1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134
|
//===--- DerivedConformanceError.cpp - Derived Error ----------------------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
//
// This file implements implicit derivation of the Error
// protocol.
//
//===----------------------------------------------------------------------===//
#include "CodeSynthesis.h"
#include "TypeChecker.h"
#include "DerivedConformances.h"
#include "swift/AST/Decl.h"
#include "swift/AST/Stmt.h"
#include "swift/AST/Expr.h"
#include "swift/AST/Module.h"
#include "swift/AST/Types.h"
#include "swift/AST/SwiftNameTranslation.h"
using namespace swift;
using namespace swift::objc_translation;
static std::pair<BraceStmt *, bool>
deriveBodyBridgedNSError_enum_nsErrorDomain(AbstractFunctionDecl *domainDecl,
void *) {
// enum SomeEnum {
// @derived
// static var _nsErrorDomain: String {
// return String(reflecting: self)
// }
// }
auto M = domainDecl->getParentModule();
auto &C = M->getASTContext();
auto self = domainDecl->getImplicitSelfDecl();
auto selfRef = new (C) DeclRefExpr(self, DeclNameLoc(), /*implicit*/ true);
auto stringType = TypeExpr::createImplicitForDecl(
DeclNameLoc(), C.getStringDecl(), domainDecl,
C.getStringDecl()->getInterfaceType());
auto *argList = ArgumentList::forImplicitSingle(
C, C.getIdentifier("reflecting"), selfRef);
auto *initReflectingCall = CallExpr::createImplicit(C, stringType, argList);
auto *ret = ReturnStmt::createImplicit(C, initReflectingCall);
auto body = BraceStmt::create(C, SourceLoc(), ASTNode(ret), SourceLoc());
return { body, /*isTypeChecked=*/false };
}
static std::pair<BraceStmt *, bool>
deriveBodyBridgedNSError_printAsObjCEnum_nsErrorDomain(
AbstractFunctionDecl *domainDecl, void *) {
// enum SomeEnum {
// @derived
// static var _nsErrorDomain: String {
// return "ModuleName.SomeEnum"
// }
// }
auto M = domainDecl->getParentModule();
auto &C = M->getASTContext();
auto TC = domainDecl->getInnermostTypeContext();
auto ED = TC->getSelfEnumDecl();
StringRef value(C.AllocateCopy(getErrorDomainStringForObjC(ED)));
auto string = new (C) StringLiteralExpr(value, SourceRange(), /*implicit*/ true);
auto *ret = ReturnStmt::createImplicit(C, SourceLoc(), string);
auto body = BraceStmt::create(C, SourceLoc(),
ASTNode(ret),
SourceLoc());
return { body, /*isTypeChecked=*/false };
}
static ValueDecl *
deriveBridgedNSError_enum_nsErrorDomain(
DerivedConformance &derived,
std::pair<BraceStmt *, bool> (*synthesizer)(AbstractFunctionDecl *, void*)) {
// enum SomeEnum {
// @derived
// static var _nsErrorDomain: String {
// ...
// }
// }
auto stringTy = derived.Context.getStringType();
// Define the property.
VarDecl *propDecl;
PatternBindingDecl *pbDecl;
std::tie(propDecl, pbDecl) = derived.declareDerivedProperty(
DerivedConformance::SynthesizedIntroducer::Var,
derived.Context.Id_nsErrorDomain, stringTy, stringTy, /*isStatic=*/true,
/*isFinal=*/true);
addNonIsolatedToSynthesized(derived.Nominal, propDecl);
// Define the getter.
auto getterDecl = derived.addGetterToReadOnlyDerivedProperty(
propDecl, stringTy);
getterDecl->setBodySynthesizer(synthesizer);
derived.addMembersToConformanceContext({propDecl, pbDecl});
return propDecl;
}
ValueDecl *DerivedConformance::deriveBridgedNSError(ValueDecl *requirement) {
if (!isa<EnumDecl>(Nominal))
return nullptr;
if (requirement->getBaseName() == Context.Id_nsErrorDomain) {
auto synthesizer = deriveBodyBridgedNSError_enum_nsErrorDomain;
auto scope = Nominal->getFormalAccessScope(Nominal->getModuleScopeContext());
if (scope.isPublic() || scope.isInternal())
// PrintAsClang may print this domain, so we should make sure we use the
// same string it will.
synthesizer = deriveBodyBridgedNSError_printAsObjCEnum_nsErrorDomain;
return deriveBridgedNSError_enum_nsErrorDomain(*this, synthesizer);
}
Context.Diags.diagnose(requirement->getLoc(),
diag::broken_errortype_requirement);
return nullptr;
}
|