2008-01-05 06:32:30 +08:00
|
|
|
//===--- SemaExprObjC.cpp - Semantic Analysis for ObjC Expressions --------===//
|
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2008-01-05 06:32:30 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements semantic analysis for Objective-C expressions.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2010-08-26 06:03:47 +08:00
|
|
|
#include "clang/Sema/SemaInternal.h"
|
2008-01-05 06:32:30 +08:00
|
|
|
#include "clang/AST/ASTContext.h"
|
|
|
|
#include "clang/AST/DeclObjC.h"
|
2008-05-30 05:12:08 +08:00
|
|
|
#include "clang/AST/ExprObjC.h"
|
2011-06-16 07:02:42 +08:00
|
|
|
#include "clang/AST/StmtVisitor.h"
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
#include "clang/AST/TypeLoc.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "clang/Analysis/DomainSpecific/CocoaConventions.h"
|
|
|
|
#include "clang/Edit/Commit.h"
|
|
|
|
#include "clang/Edit/Rewriters.h"
|
2009-03-10 05:12:44 +08:00
|
|
|
#include "clang/Lex/Preprocessor.h"
|
2012-12-04 17:13:33 +08:00
|
|
|
#include "clang/Sema/Initialization.h"
|
|
|
|
#include "clang/Sema/Lookup.h"
|
|
|
|
#include "clang/Sema/Scope.h"
|
|
|
|
#include "clang/Sema/ScopeInfo.h"
|
|
|
|
#include "llvm/ADT/SmallString.h"
|
2019-03-08 12:45:37 +08:00
|
|
|
#include "llvm/Support/ConvertUTF.h"
|
2009-03-10 05:12:44 +08:00
|
|
|
|
2008-01-05 06:32:30 +08:00
|
|
|
using namespace clang;
|
2011-02-03 17:00:02 +08:00
|
|
|
using namespace sema;
|
2011-10-03 14:36:45 +08:00
|
|
|
using llvm::makeArrayRef;
|
2008-01-05 06:32:30 +08:00
|
|
|
|
2010-08-27 07:41:50 +08:00
|
|
|
ExprResult Sema::ParseObjCStringLiteral(SourceLocation *AtLocs,
|
2015-12-25 07:58:11 +08:00
|
|
|
ArrayRef<Expr *> Strings) {
|
2009-02-18 14:13:04 +08:00
|
|
|
// Most ObjC strings are formed out of a single piece. However, we *can*
|
|
|
|
// have strings formed out of multiple @ strings with multiple pptokens in
|
|
|
|
// each one, e.g. @"foo" "bar" @"baz" "qux" which need to be turned into one
|
|
|
|
// StringLiteral for ObjCStringLiteral to hold onto.
|
2015-12-25 07:58:11 +08:00
|
|
|
StringLiteral *S = cast<StringLiteral>(Strings[0]);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-18 14:13:04 +08:00
|
|
|
// If we have a multi-part string, merge it all together.
|
2015-12-25 07:58:11 +08:00
|
|
|
if (Strings.size() != 1) {
|
2008-01-05 06:32:30 +08:00
|
|
|
// Concatenate objc strings.
|
2012-02-05 10:13:05 +08:00
|
|
|
SmallString<128> StrBuf;
|
2011-07-23 18:55:15 +08:00
|
|
|
SmallVector<SourceLocation, 8> StrLocs;
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2015-12-25 07:58:11 +08:00
|
|
|
for (Expr *E : Strings) {
|
|
|
|
S = cast<StringLiteral>(E);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2011-07-27 13:40:30 +08:00
|
|
|
// ObjC strings can't be wide or UTF.
|
|
|
|
if (!S->isAscii()) {
|
2018-08-10 05:08:08 +08:00
|
|
|
Diag(S->getBeginLoc(), diag::err_cfstring_literal_not_string_constant)
|
|
|
|
<< S->getSourceRange();
|
2009-02-18 14:13:04 +08:00
|
|
|
return true;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-08-17 20:54:38 +08:00
|
|
|
// Append the string.
|
|
|
|
StrBuf += S->getString();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-18 14:48:40 +08:00
|
|
|
// Get the locations of the string tokens.
|
|
|
|
StrLocs.append(S->tokloc_begin(), S->tokloc_end());
|
2008-01-05 06:32:30 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-18 14:48:40 +08:00
|
|
|
// Create the aggregate string with the appropriate content and location
|
|
|
|
// information.
|
2014-02-25 20:26:20 +08:00
|
|
|
const ConstantArrayType *CAT = Context.getAsConstantArrayType(S->getType());
|
|
|
|
assert(CAT && "String literal not of constant array type!");
|
|
|
|
QualType StrTy = Context.getConstantArrayType(
|
|
|
|
CAT->getElementType(), llvm::APInt(32, StrBuf.size() + 1),
|
|
|
|
CAT->getSizeModifier(), CAT->getIndexTypeCVRQualifiers());
|
|
|
|
S = StringLiteral::Create(Context, StrBuf, StringLiteral::Ascii,
|
|
|
|
/*Pascal=*/false, StrTy, &StrLocs[0],
|
|
|
|
StrLocs.size());
|
2008-01-05 06:32:30 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
return BuildObjCStringLiteral(AtLocs[0], S);
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
ExprResult Sema::BuildObjCStringLiteral(SourceLocation AtLoc, StringLiteral *S){
|
2009-02-18 14:01:06 +08:00
|
|
|
// Verify that this composite string is acceptable for ObjC strings.
|
|
|
|
if (CheckObjCString(S))
|
2008-01-05 06:32:30 +08:00
|
|
|
return true;
|
2009-02-18 14:06:56 +08:00
|
|
|
|
|
|
|
// Initialize the constant string interface lazily. This assumes
|
2009-04-07 22:18:33 +08:00
|
|
|
// the NSString interface is seen in this translation unit. Note: We
|
|
|
|
// don't use NSConstantString, since the runtime team considers this
|
|
|
|
// interface private (even though it appears in the header files).
|
2009-02-18 14:06:56 +08:00
|
|
|
QualType Ty = Context.getObjCConstantStringInterface();
|
|
|
|
if (!Ty.isNull()) {
|
2009-07-11 07:34:53 +08:00
|
|
|
Ty = Context.getObjCObjectPointerType(Ty);
|
2012-03-11 15:00:24 +08:00
|
|
|
} else if (getLangOpts().NoConstantCFStrings) {
|
2014-05-26 14:22:03 +08:00
|
|
|
IdentifierInfo *NSIdent=nullptr;
|
2012-03-11 15:00:24 +08:00
|
|
|
std::string StringClass(getLangOpts().ObjCConstantStringClass);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2010-10-20 01:19:29 +08:00
|
|
|
if (StringClass.empty())
|
|
|
|
NSIdent = &Context.Idents.get("NSConstantString");
|
|
|
|
else
|
|
|
|
NSIdent = &Context.Idents.get(StringClass);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
NamedDecl *IF = LookupSingleName(TUScope, NSIdent, AtLoc,
|
2010-04-24 07:19:04 +08:00
|
|
|
LookupOrdinaryName);
|
|
|
|
if (ObjCInterfaceDecl *StrIF = dyn_cast_or_null<ObjCInterfaceDecl>(IF)) {
|
|
|
|
Context.setObjCConstantStringInterface(StrIF);
|
|
|
|
Ty = Context.getObjCConstantStringInterface();
|
|
|
|
Ty = Context.getObjCObjectPointerType(Ty);
|
|
|
|
} else {
|
|
|
|
// If there is no NSConstantString interface defined then treat this
|
|
|
|
// as error and recover from it.
|
2018-08-10 05:08:08 +08:00
|
|
|
Diag(S->getBeginLoc(), diag::err_no_nsconstant_string_class)
|
|
|
|
<< NSIdent << S->getSourceRange();
|
2010-04-24 07:19:04 +08:00
|
|
|
Ty = Context.getObjCIdType();
|
|
|
|
}
|
2008-06-22 05:44:18 +08:00
|
|
|
} else {
|
2012-04-19 08:25:12 +08:00
|
|
|
IdentifierInfo *NSIdent = NSAPIObj->getNSClassId(NSAPI::ClassId_NSString);
|
2012-03-07 04:05:56 +08:00
|
|
|
NamedDecl *IF = LookupSingleName(TUScope, NSIdent, AtLoc,
|
2010-04-16 06:33:43 +08:00
|
|
|
LookupOrdinaryName);
|
2009-02-18 14:06:56 +08:00
|
|
|
if (ObjCInterfaceDecl *StrIF = dyn_cast_or_null<ObjCInterfaceDecl>(IF)) {
|
|
|
|
Context.setObjCConstantStringInterface(StrIF);
|
|
|
|
Ty = Context.getObjCConstantStringInterface();
|
2009-07-11 07:34:53 +08:00
|
|
|
Ty = Context.getObjCObjectPointerType(Ty);
|
2009-02-18 14:06:56 +08:00
|
|
|
} else {
|
2012-02-24 06:51:36 +08:00
|
|
|
// If there is no NSString interface defined, implicitly declare
|
|
|
|
// a @class NSString; and use that instead. This is to make sure
|
|
|
|
// type of an NSString literal is represented correctly, instead of
|
|
|
|
// being an 'id' type.
|
|
|
|
Ty = Context.getObjCNSStringType();
|
|
|
|
if (Ty.isNull()) {
|
2018-07-31 03:24:48 +08:00
|
|
|
ObjCInterfaceDecl *NSStringIDecl =
|
|
|
|
ObjCInterfaceDecl::Create (Context,
|
|
|
|
Context.getTranslationUnitDecl(),
|
|
|
|
SourceLocation(), NSIdent,
|
Parsing, semantic analysis, and AST for Objective-C type parameters.
Produce type parameter declarations for Objective-C type parameters,
and attach lists of type parameters to Objective-C classes,
categories, forward declarations, and extensions as
appropriate. Perform semantic analysis of type bounds for type
parameters, both in isolation and across classes/categories/extensions
to ensure consistency.
Also handle (de-)serialization of Objective-C type parameter lists,
along with sundry other things one must do to add a new declaration to
Clang.
Note that Objective-C type parameters are typedef name declarations,
like typedefs and C++11 type aliases, in support of type erasure.
Part of rdar://problem/6294649.
llvm-svn: 241541
2015-07-07 11:57:15 +08:00
|
|
|
nullptr, nullptr, SourceLocation());
|
2012-02-24 06:51:36 +08:00
|
|
|
Ty = Context.getObjCInterfaceType(NSStringIDecl);
|
|
|
|
Context.setObjCNSStringType(Ty);
|
|
|
|
}
|
|
|
|
Ty = Context.getObjCObjectPointerType(Ty);
|
2009-02-18 14:06:56 +08:00
|
|
|
}
|
2008-06-22 05:44:18 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
return new (Context) ObjCStringLiteral(S, Ty, AtLoc);
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Emits an error if the given method does not exist, or if the return
|
2012-05-13 01:32:44 +08:00
|
|
|
/// type is not an Objective-C object.
|
|
|
|
static bool validateBoxingMethod(Sema &S, SourceLocation Loc,
|
|
|
|
const ObjCInterfaceDecl *Class,
|
|
|
|
Selector Sel, const ObjCMethodDecl *Method) {
|
|
|
|
if (!Method) {
|
|
|
|
// FIXME: Is there a better way to avoid quotes than using getName()?
|
|
|
|
S.Diag(Loc, diag::err_undeclared_boxing_method) << Sel << Class->getName();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure the return type is reasonable.
|
2014-01-26 00:55:45 +08:00
|
|
|
QualType ReturnType = Method->getReturnType();
|
2012-05-13 01:32:44 +08:00
|
|
|
if (!ReturnType->isObjCObjectPointerType()) {
|
|
|
|
S.Diag(Loc, diag::err_objc_literal_method_sig)
|
|
|
|
<< Sel;
|
|
|
|
S.Diag(Method->getLocation(), diag::note_objc_literal_method_return)
|
|
|
|
<< ReturnType;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Maps ObjCLiteralKind to NSClassIdKindKind
|
2015-07-24 13:09:40 +08:00
|
|
|
static NSAPI::NSClassIdKindKind ClassKindFromLiteralKind(
|
|
|
|
Sema::ObjCLiteralKind LiteralKind) {
|
|
|
|
switch (LiteralKind) {
|
|
|
|
case Sema::LK_Array:
|
|
|
|
return NSAPI::ClassId_NSArray;
|
|
|
|
case Sema::LK_Dictionary:
|
|
|
|
return NSAPI::ClassId_NSDictionary;
|
|
|
|
case Sema::LK_Numeric:
|
|
|
|
return NSAPI::ClassId_NSNumber;
|
|
|
|
case Sema::LK_String:
|
|
|
|
return NSAPI::ClassId_NSString;
|
|
|
|
case Sema::LK_Boxed:
|
|
|
|
return NSAPI::ClassId_NSValue;
|
|
|
|
|
|
|
|
// there is no corresponding matching
|
|
|
|
// between LK_None/LK_Block and NSClassIdKindKind
|
|
|
|
case Sema::LK_Block:
|
|
|
|
case Sema::LK_None:
|
2015-07-24 20:47:27 +08:00
|
|
|
break;
|
2015-07-24 13:09:40 +08:00
|
|
|
}
|
2015-07-24 20:47:27 +08:00
|
|
|
llvm_unreachable("LiteralKind can't be converted into a ClassKind");
|
2015-07-24 13:09:40 +08:00
|
|
|
}
|
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Validates ObjCInterfaceDecl availability.
|
2015-07-24 13:09:40 +08:00
|
|
|
/// ObjCInterfaceDecl, used to create ObjC literals, should be defined
|
|
|
|
/// if clang not in a debugger mode.
|
|
|
|
static bool ValidateObjCLiteralInterfaceDecl(Sema &S, ObjCInterfaceDecl *Decl,
|
|
|
|
SourceLocation Loc,
|
|
|
|
Sema::ObjCLiteralKind LiteralKind) {
|
|
|
|
if (!Decl) {
|
|
|
|
NSAPI::NSClassIdKindKind Kind = ClassKindFromLiteralKind(LiteralKind);
|
|
|
|
IdentifierInfo *II = S.NSAPIObj->getNSClassId(Kind);
|
|
|
|
S.Diag(Loc, diag::err_undeclared_objc_literal_class)
|
|
|
|
<< II->getName() << LiteralKind;
|
|
|
|
return false;
|
|
|
|
} else if (!Decl->hasDefinition() && !S.getLangOpts().DebuggerObjCLiteral) {
|
|
|
|
S.Diag(Loc, diag::err_undeclared_objc_literal_class)
|
|
|
|
<< Decl->getName() << LiteralKind;
|
|
|
|
S.Diag(Decl->getLocation(), diag::note_forward_class);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Looks up ObjCInterfaceDecl of a given NSClassIdKindKind.
|
2015-07-24 13:09:40 +08:00
|
|
|
/// Used to create ObjC literals, such as NSDictionary (@{}),
|
|
|
|
/// NSArray (@[]) and Boxed Expressions (@())
|
|
|
|
static ObjCInterfaceDecl *LookupObjCInterfaceDeclForLiteral(Sema &S,
|
|
|
|
SourceLocation Loc,
|
|
|
|
Sema::ObjCLiteralKind LiteralKind) {
|
|
|
|
NSAPI::NSClassIdKindKind ClassKind = ClassKindFromLiteralKind(LiteralKind);
|
|
|
|
IdentifierInfo *II = S.NSAPIObj->getNSClassId(ClassKind);
|
|
|
|
NamedDecl *IF = S.LookupSingleName(S.TUScope, II, Loc,
|
|
|
|
Sema::LookupOrdinaryName);
|
|
|
|
ObjCInterfaceDecl *ID = dyn_cast_or_null<ObjCInterfaceDecl>(IF);
|
|
|
|
if (!ID && S.getLangOpts().DebuggerObjCLiteral) {
|
|
|
|
ASTContext &Context = S.Context;
|
|
|
|
TranslationUnitDecl *TU = Context.getTranslationUnitDecl();
|
|
|
|
ID = ObjCInterfaceDecl::Create (Context, TU, SourceLocation(), II,
|
|
|
|
nullptr, nullptr, SourceLocation());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ValidateObjCLiteralInterfaceDecl(S, ID, Loc, LiteralKind)) {
|
|
|
|
ID = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ID;
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Retrieve the NSNumber factory method that should be used to create
|
2012-03-07 04:05:56 +08:00
|
|
|
/// an Objective-C literal for the given type.
|
|
|
|
static ObjCMethodDecl *getNSNumberFactoryMethod(Sema &S, SourceLocation Loc,
|
2012-04-19 08:25:12 +08:00
|
|
|
QualType NumberType,
|
|
|
|
bool isLiteral = false,
|
|
|
|
SourceRange R = SourceRange()) {
|
2013-02-21 06:23:23 +08:00
|
|
|
Optional<NSAPI::NSNumberLiteralMethodKind> Kind =
|
|
|
|
S.NSAPIObj->getNSNumberFactoryMethodKind(NumberType);
|
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
if (!Kind) {
|
2012-04-19 08:25:12 +08:00
|
|
|
if (isLiteral) {
|
|
|
|
S.Diag(Loc, diag::err_invalid_nsnumber_type)
|
|
|
|
<< NumberType << R;
|
|
|
|
}
|
2014-05-26 14:22:03 +08:00
|
|
|
return nullptr;
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
// If we already looked up this method, we're done.
|
|
|
|
if (S.NSNumberLiteralMethods[*Kind])
|
|
|
|
return S.NSNumberLiteralMethods[*Kind];
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
Selector Sel = S.NSAPIObj->getNSNumberLiteralSelector(*Kind,
|
|
|
|
/*Instance=*/false);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
ASTContext &CX = S.Context;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
// Look up the NSNumber class, if we haven't done so already. It's cached
|
|
|
|
// in the Sema instance.
|
|
|
|
if (!S.NSNumberDecl) {
|
2015-07-24 13:09:40 +08:00
|
|
|
S.NSNumberDecl = LookupObjCInterfaceDeclForLiteral(S, Loc,
|
|
|
|
Sema::LK_Numeric);
|
2012-04-19 08:25:12 +08:00
|
|
|
if (!S.NSNumberDecl) {
|
2014-05-26 14:22:03 +08:00
|
|
|
return nullptr;
|
2012-04-19 08:25:12 +08:00
|
|
|
}
|
2015-02-17 00:17:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (S.NSNumberPointer.isNull()) {
|
2012-04-19 08:25:12 +08:00
|
|
|
// generate the pointer to NSNumber type.
|
2012-05-13 01:32:52 +08:00
|
|
|
QualType NSNumberObject = CX.getObjCInterfaceType(S.NSNumberDecl);
|
|
|
|
S.NSNumberPointer = CX.getObjCObjectPointerType(NSNumberObject);
|
2012-04-19 08:25:12 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
// Look for the appropriate method within NSNumber.
|
2012-05-13 01:32:52 +08:00
|
|
|
ObjCMethodDecl *Method = S.NSNumberDecl->lookupClassMethod(Sel);
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!Method && S.getLangOpts().DebuggerObjCLiteral) {
|
2012-04-19 08:25:12 +08:00
|
|
|
// create a stub definition this NSNumber factory method.
|
2014-05-26 14:22:03 +08:00
|
|
|
TypeSourceInfo *ReturnTInfo = nullptr;
|
2014-01-26 00:55:45 +08:00
|
|
|
Method =
|
|
|
|
ObjCMethodDecl::Create(CX, SourceLocation(), SourceLocation(), Sel,
|
|
|
|
S.NSNumberPointer, ReturnTInfo, S.NSNumberDecl,
|
|
|
|
/*isInstance=*/false, /*isVariadic=*/false,
|
|
|
|
/*isPropertyAccessor=*/false,
|
|
|
|
/*isImplicitlyDeclared=*/true,
|
|
|
|
/*isDefined=*/false, ObjCMethodDecl::Required,
|
|
|
|
/*HasRelatedResultType=*/false);
|
2012-03-07 04:05:56 +08:00
|
|
|
ParmVarDecl *value = ParmVarDecl::Create(S.Context, Method,
|
|
|
|
SourceLocation(), SourceLocation(),
|
2012-04-19 08:25:12 +08:00
|
|
|
&CX.Idents.get("value"),
|
2014-05-26 14:22:03 +08:00
|
|
|
NumberType, /*TInfo=*/nullptr,
|
|
|
|
SC_None, nullptr);
|
2013-05-05 08:41:58 +08:00
|
|
|
Method->setMethodParams(S.Context, value, None);
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
2012-05-13 01:32:44 +08:00
|
|
|
if (!validateBoxingMethod(S, Loc, S.NSNumberDecl, Sel, Method))
|
2014-05-26 14:22:03 +08:00
|
|
|
return nullptr;
|
2012-03-07 04:05:56 +08:00
|
|
|
|
|
|
|
// Note: if the parameter type is out-of-line, we'll catch it later in the
|
|
|
|
// implicit conversion.
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
S.NSNumberLiteralMethods[*Kind] = Method;
|
|
|
|
return Method;
|
|
|
|
}
|
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
/// BuildObjCNumericLiteral - builds an ObjCBoxedExpr AST node for the
|
|
|
|
/// numeric literal expression. Type of the expression will be "NSNumber *".
|
2012-03-07 04:05:56 +08:00
|
|
|
ExprResult Sema::BuildObjCNumericLiteral(SourceLocation AtLoc, Expr *Number) {
|
|
|
|
// Determine the type of the literal.
|
|
|
|
QualType NumberType = Number->getType();
|
|
|
|
if (CharacterLiteral *Char = dyn_cast<CharacterLiteral>(Number)) {
|
|
|
|
// In C, character literals have type 'int'. That's not the type we want
|
|
|
|
// to use to determine the Objective-c literal kind.
|
|
|
|
switch (Char->getKind()) {
|
|
|
|
case CharacterLiteral::Ascii:
|
2016-01-08 04:59:26 +08:00
|
|
|
case CharacterLiteral::UTF8:
|
2012-03-07 04:05:56 +08:00
|
|
|
NumberType = Context.CharTy;
|
|
|
|
break;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
case CharacterLiteral::Wide:
|
2013-05-10 18:08:40 +08:00
|
|
|
NumberType = Context.getWideCharType();
|
2012-03-07 04:05:56 +08:00
|
|
|
break;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
case CharacterLiteral::UTF16:
|
|
|
|
NumberType = Context.Char16Ty;
|
|
|
|
break;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
case CharacterLiteral::UTF32:
|
|
|
|
NumberType = Context.Char32Ty;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
// Look for the appropriate method within NSNumber.
|
|
|
|
// Construct the literal.
|
2012-05-02 05:47:19 +08:00
|
|
|
SourceRange NR(Number->getSourceRange());
|
2012-04-19 08:25:12 +08:00
|
|
|
ObjCMethodDecl *Method = getNSNumberFactoryMethod(*this, AtLoc, NumberType,
|
2012-05-02 05:47:19 +08:00
|
|
|
true, NR);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (!Method)
|
|
|
|
return ExprError();
|
|
|
|
|
|
|
|
// Convert the number to the type that the parameter expects.
|
2014-07-07 17:02:20 +08:00
|
|
|
ParmVarDecl *ParamDecl = Method->parameters()[0];
|
2012-05-02 05:47:19 +08:00
|
|
|
InitializedEntity Entity = InitializedEntity::InitializeParameter(Context,
|
|
|
|
ParamDecl);
|
|
|
|
ExprResult ConvertedNumber = PerformCopyInitialization(Entity,
|
|
|
|
SourceLocation(),
|
2014-05-29 22:05:12 +08:00
|
|
|
Number);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (ConvertedNumber.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
Number = ConvertedNumber.get();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-05-02 05:47:19 +08:00
|
|
|
// Use the effective source range of the literal, including the leading '@'.
|
2012-03-07 04:05:56 +08:00
|
|
|
return MaybeBindToTemporary(
|
2012-05-02 05:47:19 +08:00
|
|
|
new (Context) ObjCBoxedExpr(Number, NSNumberPointer, Method,
|
|
|
|
SourceRange(AtLoc, NR.getEnd())));
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
2018-07-31 03:24:48 +08:00
|
|
|
ExprResult Sema::ActOnObjCBoolLiteral(SourceLocation AtLoc,
|
2012-03-07 04:05:56 +08:00
|
|
|
SourceLocation ValueLoc,
|
|
|
|
bool Value) {
|
|
|
|
ExprResult Inner;
|
2012-03-11 15:00:24 +08:00
|
|
|
if (getLangOpts().CPlusPlus) {
|
2012-03-07 04:05:56 +08:00
|
|
|
Inner = ActOnCXXBoolLiteral(ValueLoc, Value? tok::kw_true : tok::kw_false);
|
|
|
|
} else {
|
2018-07-31 03:24:48 +08:00
|
|
|
// C doesn't actually have a way to represent literal values of type
|
2012-03-07 04:05:56 +08:00
|
|
|
// _Bool. So, we'll use 0/1 and implicit cast to _Bool.
|
|
|
|
Inner = ActOnIntegerConstant(ValueLoc, Value? 1 : 0);
|
2018-07-31 03:24:48 +08:00
|
|
|
Inner = ImpCastExprToType(Inner.get(), Context.BoolTy,
|
2012-03-07 04:05:56 +08:00
|
|
|
CK_IntegralToBoolean);
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
return BuildObjCNumericLiteral(AtLoc, Inner.get());
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Check that the given expression is a valid element of an Objective-C
|
2012-03-07 04:05:56 +08:00
|
|
|
/// collection literal.
|
2018-07-31 03:24:48 +08:00
|
|
|
static ExprResult CheckObjCCollectionLiteralElement(Sema &S, Expr *Element,
|
2013-08-14 07:44:55 +08:00
|
|
|
QualType T,
|
|
|
|
bool ArrayLiteral = false) {
|
2012-03-07 04:05:56 +08:00
|
|
|
// If the expression is type-dependent, there's nothing for us to do.
|
|
|
|
if (Element->isTypeDependent())
|
|
|
|
return Element;
|
|
|
|
|
|
|
|
ExprResult Result = S.CheckPlaceholderExpr(Element);
|
|
|
|
if (Result.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
Element = Result.get();
|
|
|
|
|
2018-07-31 03:24:48 +08:00
|
|
|
// In C++, check for an implicit conversion to an Objective-C object pointer
|
2012-03-07 04:05:56 +08:00
|
|
|
// type.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (S.getLangOpts().CPlusPlus && Element->getType()->isRecordType()) {
|
2012-03-07 04:05:56 +08:00
|
|
|
InitializedEntity Entity
|
2012-05-13 01:32:52 +08:00
|
|
|
= InitializedEntity::InitializeParameter(S.Context, T,
|
|
|
|
/*Consumed=*/false);
|
2018-08-10 05:08:08 +08:00
|
|
|
InitializationKind Kind = InitializationKind::CreateCopy(
|
|
|
|
Element->getBeginLoc(), SourceLocation());
|
2013-05-03 23:05:50 +08:00
|
|
|
InitializationSequence Seq(S, Entity, Kind, Element);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (!Seq.Failed())
|
2012-08-24 07:38:35 +08:00
|
|
|
return Seq.Perform(S, Entity, Kind, Element);
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Expr *OrigElement = Element;
|
|
|
|
|
|
|
|
// Perform lvalue-to-rvalue conversion.
|
|
|
|
Result = S.DefaultLvalueConversion(Element);
|
|
|
|
if (Result.isInvalid())
|
|
|
|
return ExprError();
|
2018-07-31 03:24:48 +08:00
|
|
|
Element = Result.get();
|
2012-03-07 04:05:56 +08:00
|
|
|
|
|
|
|
// Make sure that we have an Objective-C pointer type or block.
|
|
|
|
if (!Element->getType()->isObjCObjectPointerType() &&
|
|
|
|
!Element->getType()->isBlockPointerType()) {
|
|
|
|
bool Recovered = false;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
// If this is potentially an Objective-C numeric literal, add the '@'.
|
2018-07-31 03:24:48 +08:00
|
|
|
if (isa<IntegerLiteral>(OrigElement) ||
|
2012-03-07 04:05:56 +08:00
|
|
|
isa<CharacterLiteral>(OrigElement) ||
|
|
|
|
isa<FloatingLiteral>(OrigElement) ||
|
|
|
|
isa<ObjCBoolLiteralExpr>(OrigElement) ||
|
|
|
|
isa<CXXBoolLiteralExpr>(OrigElement)) {
|
|
|
|
if (S.NSAPIObj->getNSNumberFactoryMethodKind(OrigElement->getType())) {
|
|
|
|
int Which = isa<CharacterLiteral>(OrigElement) ? 1
|
|
|
|
: (isa<CXXBoolLiteralExpr>(OrigElement) ||
|
|
|
|
isa<ObjCBoolLiteralExpr>(OrigElement)) ? 2
|
|
|
|
: 3;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(OrigElement->getBeginLoc(), diag::err_box_literal_collection)
|
|
|
|
<< Which << OrigElement->getSourceRange()
|
|
|
|
<< FixItHint::CreateInsertion(OrigElement->getBeginLoc(), "@");
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2018-08-10 05:08:08 +08:00
|
|
|
Result =
|
|
|
|
S.BuildObjCNumericLiteral(OrigElement->getBeginLoc(), OrigElement);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (Result.isInvalid())
|
|
|
|
return ExprError();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
Element = Result.get();
|
|
|
|
Recovered = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If this is potentially an Objective-C string literal, add the '@'.
|
|
|
|
else if (StringLiteral *String = dyn_cast<StringLiteral>(OrigElement)) {
|
|
|
|
if (String->isAscii()) {
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(OrigElement->getBeginLoc(), diag::err_box_literal_collection)
|
|
|
|
<< 0 << OrigElement->getSourceRange()
|
|
|
|
<< FixItHint::CreateInsertion(OrigElement->getBeginLoc(), "@");
|
2012-03-07 04:05:56 +08:00
|
|
|
|
2018-08-10 05:08:08 +08:00
|
|
|
Result = S.BuildObjCStringLiteral(OrigElement->getBeginLoc(), String);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (Result.isInvalid())
|
|
|
|
return ExprError();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
Element = Result.get();
|
|
|
|
Recovered = true;
|
|
|
|
}
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
if (!Recovered) {
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(Element->getBeginLoc(), diag::err_invalid_collection_element)
|
|
|
|
<< Element->getType();
|
2012-03-07 04:05:56 +08:00
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
}
|
2013-08-14 07:44:55 +08:00
|
|
|
if (ArrayLiteral)
|
2013-10-10 06:34:33 +08:00
|
|
|
if (ObjCStringLiteral *getString =
|
|
|
|
dyn_cast<ObjCStringLiteral>(OrigElement)) {
|
|
|
|
if (StringLiteral *SL = getString->getString()) {
|
|
|
|
unsigned numConcat = SL->getNumConcatenated();
|
|
|
|
if (numConcat > 1) {
|
|
|
|
// Only warn if the concatenated string doesn't come from a macro.
|
|
|
|
bool hasMacro = false;
|
|
|
|
for (unsigned i = 0; i < numConcat ; ++i)
|
|
|
|
if (SL->getStrTokenLoc(i).isMacroID()) {
|
|
|
|
hasMacro = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!hasMacro)
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(Element->getBeginLoc(),
|
2013-10-10 06:34:33 +08:00
|
|
|
diag::warn_concatenated_nsarray_literal)
|
2018-08-10 05:08:08 +08:00
|
|
|
<< Element->getType();
|
2013-10-10 06:34:33 +08:00
|
|
|
}
|
|
|
|
}
|
2013-08-14 07:44:55 +08:00
|
|
|
}
|
|
|
|
|
2018-07-31 03:24:48 +08:00
|
|
|
// Make sure that the element has the type that the container factory
|
|
|
|
// function expects.
|
2012-03-07 04:05:56 +08:00
|
|
|
return S.PerformCopyInitialization(
|
2018-08-10 05:08:08 +08:00
|
|
|
InitializedEntity::InitializeParameter(S.Context, T,
|
|
|
|
/*Consumed=*/false),
|
|
|
|
Element->getBeginLoc(), Element);
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
ExprResult Sema::BuildObjCBoxedExpr(SourceRange SR, Expr *ValueExpr) {
|
|
|
|
if (ValueExpr->isTypeDependent()) {
|
2018-07-31 03:24:48 +08:00
|
|
|
ObjCBoxedExpr *BoxedExpr =
|
2014-05-26 14:22:03 +08:00
|
|
|
new (Context) ObjCBoxedExpr(ValueExpr, Context.DependentTy, nullptr, SR);
|
2014-05-29 22:05:12 +08:00
|
|
|
return BoxedExpr;
|
2012-04-19 08:25:12 +08:00
|
|
|
}
|
2014-05-26 14:22:03 +08:00
|
|
|
ObjCMethodDecl *BoxingMethod = nullptr;
|
2012-04-19 08:25:12 +08:00
|
|
|
QualType BoxedType;
|
|
|
|
// Convert the expression to an RValue, so we can check for pointer types...
|
|
|
|
ExprResult RValue = DefaultFunctionArrayLvalueConversion(ValueExpr);
|
|
|
|
if (RValue.isInvalid()) {
|
|
|
|
return ExprError();
|
|
|
|
}
|
2015-07-24 13:09:40 +08:00
|
|
|
SourceLocation Loc = SR.getBegin();
|
2012-04-19 08:25:12 +08:00
|
|
|
ValueExpr = RValue.get();
|
2012-05-02 05:47:19 +08:00
|
|
|
QualType ValueType(ValueExpr->getType());
|
2012-04-19 08:25:12 +08:00
|
|
|
if (const PointerType *PT = ValueType->getAs<PointerType>()) {
|
|
|
|
QualType PointeeType = PT->getPointeeType();
|
|
|
|
if (Context.hasSameUnqualifiedType(PointeeType, Context.CharTy)) {
|
|
|
|
|
|
|
|
if (!NSStringDecl) {
|
2015-07-24 13:09:40 +08:00
|
|
|
NSStringDecl = LookupObjCInterfaceDeclForLiteral(*this, Loc,
|
|
|
|
Sema::LK_String);
|
2012-04-19 08:25:12 +08:00
|
|
|
if (!NSStringDecl) {
|
|
|
|
return ExprError();
|
|
|
|
}
|
2012-05-13 01:32:52 +08:00
|
|
|
QualType NSStringObject = Context.getObjCInterfaceType(NSStringDecl);
|
|
|
|
NSStringPointer = Context.getObjCObjectPointerType(NSStringObject);
|
2012-04-19 08:25:12 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2019-03-08 12:45:37 +08:00
|
|
|
// The boxed expression can be emitted as a compile time constant if it is
|
|
|
|
// a string literal whose character encoding is compatible with UTF-8.
|
|
|
|
if (auto *CE = dyn_cast<ImplicitCastExpr>(ValueExpr))
|
|
|
|
if (CE->getCastKind() == CK_ArrayToPointerDecay)
|
|
|
|
if (auto *SL =
|
|
|
|
dyn_cast<StringLiteral>(CE->getSubExpr()->IgnoreParens())) {
|
|
|
|
assert((SL->isAscii() || SL->isUTF8()) &&
|
|
|
|
"unexpected character encoding");
|
|
|
|
StringRef Str = SL->getString();
|
|
|
|
const llvm::UTF8 *StrBegin = Str.bytes_begin();
|
|
|
|
const llvm::UTF8 *StrEnd = Str.bytes_end();
|
|
|
|
// Check that this is a valid UTF-8 string.
|
|
|
|
if (llvm::isLegalUTF8String(&StrBegin, StrEnd)) {
|
|
|
|
BoxedType = Context.getAttributedType(
|
|
|
|
AttributedType::getNullabilityAttrKind(
|
|
|
|
NullabilityKind::NonNull),
|
|
|
|
NSStringPointer, NSStringPointer);
|
|
|
|
return new (Context) ObjCBoxedExpr(CE, BoxedType, nullptr, SR);
|
|
|
|
}
|
|
|
|
|
|
|
|
Diag(SL->getBeginLoc(), diag::warn_objc_boxing_invalid_utf8_string)
|
|
|
|
<< NSStringPointer << SL->getSourceRange();
|
|
|
|
}
|
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
if (!StringWithUTF8StringMethod) {
|
|
|
|
IdentifierInfo *II = &Context.Idents.get("stringWithUTF8String");
|
|
|
|
Selector stringWithUTF8String = Context.Selectors.getUnarySelector(II);
|
|
|
|
|
|
|
|
// Look for the appropriate method within NSString.
|
2012-05-13 01:32:44 +08:00
|
|
|
BoxingMethod = NSStringDecl->lookupClassMethod(stringWithUTF8String);
|
|
|
|
if (!BoxingMethod && getLangOpts().DebuggerObjCLiteral) {
|
2012-04-19 08:25:12 +08:00
|
|
|
// Debugger needs to work even if NSString hasn't been defined.
|
2014-05-26 14:22:03 +08:00
|
|
|
TypeSourceInfo *ReturnTInfo = nullptr;
|
2014-01-26 00:55:45 +08:00
|
|
|
ObjCMethodDecl *M = ObjCMethodDecl::Create(
|
|
|
|
Context, SourceLocation(), SourceLocation(), stringWithUTF8String,
|
|
|
|
NSStringPointer, ReturnTInfo, NSStringDecl,
|
|
|
|
/*isInstance=*/false, /*isVariadic=*/false,
|
|
|
|
/*isPropertyAccessor=*/false,
|
|
|
|
/*isImplicitlyDeclared=*/true,
|
|
|
|
/*isDefined=*/false, ObjCMethodDecl::Required,
|
|
|
|
/*HasRelatedResultType=*/false);
|
2012-05-13 01:32:52 +08:00
|
|
|
QualType ConstCharType = Context.CharTy.withConst();
|
2012-04-19 08:25:12 +08:00
|
|
|
ParmVarDecl *value =
|
|
|
|
ParmVarDecl::Create(Context, M,
|
|
|
|
SourceLocation(), SourceLocation(),
|
|
|
|
&Context.Idents.get("value"),
|
2012-05-13 01:32:52 +08:00
|
|
|
Context.getPointerType(ConstCharType),
|
2014-05-26 14:22:03 +08:00
|
|
|
/*TInfo=*/nullptr,
|
|
|
|
SC_None, nullptr);
|
2013-05-05 08:41:58 +08:00
|
|
|
M->setMethodParams(Context, value, None);
|
2012-05-13 01:32:44 +08:00
|
|
|
BoxingMethod = M;
|
2012-04-19 08:25:12 +08:00
|
|
|
}
|
2012-05-12 23:53:41 +08:00
|
|
|
|
2015-07-24 13:09:40 +08:00
|
|
|
if (!validateBoxingMethod(*this, Loc, NSStringDecl,
|
2012-05-13 01:32:44 +08:00
|
|
|
stringWithUTF8String, BoxingMethod))
|
|
|
|
return ExprError();
|
|
|
|
|
|
|
|
StringWithUTF8StringMethod = BoxingMethod;
|
2012-04-19 08:25:12 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
BoxingMethod = StringWithUTF8StringMethod;
|
|
|
|
BoxedType = NSStringPointer;
|
2017-11-09 05:33:15 +08:00
|
|
|
// Transfer the nullability from method's return type.
|
|
|
|
Optional<NullabilityKind> Nullability =
|
|
|
|
BoxingMethod->getReturnType()->getNullability(Context);
|
|
|
|
if (Nullability)
|
|
|
|
BoxedType = Context.getAttributedType(
|
|
|
|
AttributedType::getNullabilityAttrKind(*Nullability), BoxedType,
|
|
|
|
BoxedType);
|
2012-04-19 08:25:12 +08:00
|
|
|
}
|
2012-05-02 05:47:19 +08:00
|
|
|
} else if (ValueType->isBuiltinType()) {
|
2012-04-19 08:25:12 +08:00
|
|
|
// The other types we support are numeric, char and BOOL/bool. We could also
|
|
|
|
// provide limited support for structure types, such as NSRange, NSRect, and
|
|
|
|
// NSSize. See NSValue (NSValueGeometryExtensions) in <Foundation/NSGeometry.h>
|
|
|
|
// for more details.
|
|
|
|
|
|
|
|
// Check for a top-level character literal.
|
|
|
|
if (const CharacterLiteral *Char =
|
|
|
|
dyn_cast<CharacterLiteral>(ValueExpr->IgnoreParens())) {
|
|
|
|
// In C, character literals have type 'int'. That's not the type we want
|
|
|
|
// to use to determine the Objective-c literal kind.
|
|
|
|
switch (Char->getKind()) {
|
|
|
|
case CharacterLiteral::Ascii:
|
2016-01-08 04:59:26 +08:00
|
|
|
case CharacterLiteral::UTF8:
|
2012-04-19 08:25:12 +08:00
|
|
|
ValueType = Context.CharTy;
|
|
|
|
break;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
case CharacterLiteral::Wide:
|
2013-05-10 18:08:40 +08:00
|
|
|
ValueType = Context.getWideCharType();
|
2012-04-19 08:25:12 +08:00
|
|
|
break;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
case CharacterLiteral::UTF16:
|
|
|
|
ValueType = Context.Char16Ty;
|
|
|
|
break;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
case CharacterLiteral::UTF32:
|
|
|
|
ValueType = Context.Char32Ty;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// FIXME: Do I need to do anything special with BoolTy expressions?
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
// Look for the appropriate method within NSNumber.
|
2015-07-24 13:09:40 +08:00
|
|
|
BoxingMethod = getNSNumberFactoryMethod(*this, Loc, ValueType);
|
2012-04-19 08:25:12 +08:00
|
|
|
BoxedType = NSNumberPointer;
|
2012-05-16 03:17:44 +08:00
|
|
|
} else if (const EnumType *ET = ValueType->getAs<EnumType>()) {
|
|
|
|
if (!ET->getDecl()->isComplete()) {
|
2015-07-24 13:09:40 +08:00
|
|
|
Diag(Loc, diag::err_objc_incomplete_boxed_expression_type)
|
2012-05-16 03:17:44 +08:00
|
|
|
<< ValueType << ValueExpr->getSourceRange();
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2015-07-24 13:09:40 +08:00
|
|
|
BoxingMethod = getNSNumberFactoryMethod(*this, Loc,
|
2012-05-16 03:17:44 +08:00
|
|
|
ET->getDecl()->getIntegerType());
|
|
|
|
BoxedType = NSNumberPointer;
|
2015-06-26 13:28:36 +08:00
|
|
|
} else if (ValueType->isObjCBoxableRecordType()) {
|
|
|
|
// Support for structure types, that marked as objc_boxable
|
|
|
|
// struct __attribute__((objc_boxable)) s { ... };
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-06-26 13:28:36 +08:00
|
|
|
// Look up the NSValue class, if we haven't done so already. It's cached
|
|
|
|
// in the Sema instance.
|
|
|
|
if (!NSValueDecl) {
|
2015-07-24 13:09:40 +08:00
|
|
|
NSValueDecl = LookupObjCInterfaceDeclForLiteral(*this, Loc,
|
|
|
|
Sema::LK_Boxed);
|
2015-06-26 13:28:36 +08:00
|
|
|
if (!NSValueDecl) {
|
|
|
|
return ExprError();
|
|
|
|
}
|
2015-07-24 13:09:40 +08:00
|
|
|
|
2015-06-26 13:28:36 +08:00
|
|
|
// generate the pointer to NSValue type.
|
|
|
|
QualType NSValueObject = Context.getObjCInterfaceType(NSValueDecl);
|
|
|
|
NSValuePointer = Context.getObjCObjectPointerType(NSValueObject);
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-06-26 13:28:36 +08:00
|
|
|
if (!ValueWithBytesObjCTypeMethod) {
|
|
|
|
IdentifierInfo *II[] = {
|
|
|
|
&Context.Idents.get("valueWithBytes"),
|
|
|
|
&Context.Idents.get("objCType")
|
|
|
|
};
|
|
|
|
Selector ValueWithBytesObjCType = Context.Selectors.getSelector(2, II);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-06-26 13:28:36 +08:00
|
|
|
// Look for the appropriate method within NSValue.
|
|
|
|
BoxingMethod = NSValueDecl->lookupClassMethod(ValueWithBytesObjCType);
|
|
|
|
if (!BoxingMethod && getLangOpts().DebuggerObjCLiteral) {
|
|
|
|
// Debugger needs to work even if NSValue hasn't been defined.
|
|
|
|
TypeSourceInfo *ReturnTInfo = nullptr;
|
|
|
|
ObjCMethodDecl *M = ObjCMethodDecl::Create(
|
|
|
|
Context,
|
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
ValueWithBytesObjCType,
|
|
|
|
NSValuePointer,
|
|
|
|
ReturnTInfo,
|
|
|
|
NSValueDecl,
|
|
|
|
/*isInstance=*/false,
|
|
|
|
/*isVariadic=*/false,
|
|
|
|
/*isPropertyAccessor=*/false,
|
|
|
|
/*isImplicitlyDeclared=*/true,
|
|
|
|
/*isDefined=*/false,
|
|
|
|
ObjCMethodDecl::Required,
|
|
|
|
/*HasRelatedResultType=*/false);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-06-26 13:28:36 +08:00
|
|
|
SmallVector<ParmVarDecl *, 2> Params;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-06-26 13:28:36 +08:00
|
|
|
ParmVarDecl *bytes =
|
|
|
|
ParmVarDecl::Create(Context, M,
|
|
|
|
SourceLocation(), SourceLocation(),
|
|
|
|
&Context.Idents.get("bytes"),
|
|
|
|
Context.VoidPtrTy.withConst(),
|
|
|
|
/*TInfo=*/nullptr,
|
|
|
|
SC_None, nullptr);
|
|
|
|
Params.push_back(bytes);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-06-26 13:28:36 +08:00
|
|
|
QualType ConstCharType = Context.CharTy.withConst();
|
|
|
|
ParmVarDecl *type =
|
|
|
|
ParmVarDecl::Create(Context, M,
|
|
|
|
SourceLocation(), SourceLocation(),
|
|
|
|
&Context.Idents.get("type"),
|
|
|
|
Context.getPointerType(ConstCharType),
|
|
|
|
/*TInfo=*/nullptr,
|
|
|
|
SC_None, nullptr);
|
|
|
|
Params.push_back(type);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-06-26 13:28:36 +08:00
|
|
|
M->setMethodParams(Context, Params, None);
|
|
|
|
BoxingMethod = M;
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-07-24 13:09:40 +08:00
|
|
|
if (!validateBoxingMethod(*this, Loc, NSValueDecl,
|
2015-06-26 13:28:36 +08:00
|
|
|
ValueWithBytesObjCType, BoxingMethod))
|
|
|
|
return ExprError();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-06-26 13:28:36 +08:00
|
|
|
ValueWithBytesObjCTypeMethod = BoxingMethod;
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-06-26 13:28:36 +08:00
|
|
|
if (!ValueType.isTriviallyCopyableType(Context)) {
|
2015-07-24 13:09:40 +08:00
|
|
|
Diag(Loc, diag::err_objc_non_trivially_copyable_boxed_expression_type)
|
2015-06-26 13:28:36 +08:00
|
|
|
<< ValueType << ValueExpr->getSourceRange();
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
|
|
|
BoxingMethod = ValueWithBytesObjCTypeMethod;
|
|
|
|
BoxedType = NSValuePointer;
|
2012-04-19 08:25:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!BoxingMethod) {
|
2015-07-24 13:09:40 +08:00
|
|
|
Diag(Loc, diag::err_objc_illegal_boxed_expression_type)
|
2012-04-19 08:25:12 +08:00
|
|
|
<< ValueType << ValueExpr->getSourceRange();
|
|
|
|
return ExprError();
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-07-24 13:09:40 +08:00
|
|
|
DiagnoseUseOfDecl(BoxingMethod, Loc);
|
2015-06-26 13:28:36 +08:00
|
|
|
|
|
|
|
ExprResult ConvertedValueExpr;
|
|
|
|
if (ValueType->isObjCBoxableRecordType()) {
|
|
|
|
InitializedEntity IE = InitializedEntity::InitializeTemporary(ValueType);
|
2018-07-31 03:24:48 +08:00
|
|
|
ConvertedValueExpr = PerformCopyInitialization(IE, ValueExpr->getExprLoc(),
|
2015-06-26 13:28:36 +08:00
|
|
|
ValueExpr);
|
|
|
|
} else {
|
|
|
|
// Convert the expression to the type that the parameter requires.
|
|
|
|
ParmVarDecl *ParamDecl = BoxingMethod->parameters()[0];
|
|
|
|
InitializedEntity IE = InitializedEntity::InitializeParameter(Context,
|
|
|
|
ParamDecl);
|
|
|
|
ConvertedValueExpr = PerformCopyInitialization(IE, SourceLocation(),
|
|
|
|
ValueExpr);
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-04-19 08:25:12 +08:00
|
|
|
if (ConvertedValueExpr.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
ValueExpr = ConvertedValueExpr.get();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
|
|
|
ObjCBoxedExpr *BoxedExpr =
|
2012-04-19 08:25:12 +08:00
|
|
|
new (Context) ObjCBoxedExpr(ValueExpr, BoxedType,
|
|
|
|
BoxingMethod, SR);
|
|
|
|
return MaybeBindToTemporary(BoxedExpr);
|
|
|
|
}
|
|
|
|
|
2012-07-31 13:14:30 +08:00
|
|
|
/// Build an ObjC subscript pseudo-object expression, given that
|
|
|
|
/// that's supported by the runtime.
|
2012-03-07 04:05:56 +08:00
|
|
|
ExprResult Sema::BuildObjCSubscriptExpression(SourceLocation RB, Expr *BaseExpr,
|
|
|
|
Expr *IndexExpr,
|
|
|
|
ObjCMethodDecl *getterMethod,
|
|
|
|
ObjCMethodDecl *setterMethod) {
|
2013-11-02 05:58:17 +08:00
|
|
|
assert(!LangOpts.isSubscriptPointerArithmetic());
|
2012-06-20 14:18:46 +08:00
|
|
|
|
2012-07-31 13:14:30 +08:00
|
|
|
// We can't get dependent types here; our callers should have
|
|
|
|
// filtered them out.
|
|
|
|
assert((!BaseExpr->isTypeDependent() && !IndexExpr->isTypeDependent()) &&
|
|
|
|
"base or index cannot have dependent type here");
|
|
|
|
|
|
|
|
// Filter out placeholders in the index. In theory, overloads could
|
|
|
|
// be preserved here, although that might not actually work correctly.
|
2012-03-07 04:05:56 +08:00
|
|
|
ExprResult Result = CheckPlaceholderExpr(IndexExpr);
|
|
|
|
if (Result.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
IndexExpr = Result.get();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-07-31 13:14:30 +08:00
|
|
|
// Perform lvalue-to-rvalue conversion on the base.
|
2012-03-07 04:05:56 +08:00
|
|
|
Result = DefaultLvalueConversion(BaseExpr);
|
|
|
|
if (Result.isInvalid())
|
|
|
|
return ExprError();
|
|
|
|
BaseExpr = Result.get();
|
2012-07-31 13:14:30 +08:00
|
|
|
|
|
|
|
// Build the pseudo-object expression.
|
2015-12-31 12:43:19 +08:00
|
|
|
return new (Context) ObjCSubscriptRefExpr(
|
|
|
|
BaseExpr, IndexExpr, Context.PseudoObjectTy, VK_LValue, OK_ObjCSubscript,
|
|
|
|
getterMethod, setterMethod, RB);
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ExprResult Sema::BuildObjCArrayLiteral(SourceRange SR, MultiExprArg Elements) {
|
2015-07-24 13:09:40 +08:00
|
|
|
SourceLocation Loc = SR.getBegin();
|
2012-03-07 04:05:56 +08:00
|
|
|
|
2015-07-24 13:09:40 +08:00
|
|
|
if (!NSArrayDecl) {
|
|
|
|
NSArrayDecl = LookupObjCInterfaceDeclForLiteral(*this, Loc,
|
|
|
|
Sema::LK_Array);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (!NSArrayDecl) {
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
}
|
2015-07-24 13:09:40 +08:00
|
|
|
|
2014-08-09 01:31:14 +08:00
|
|
|
// Find the arrayWithObjects:count: method, if we haven't done so already.
|
2014-08-09 02:29:52 +08:00
|
|
|
QualType IdT = Context.getObjCIdType();
|
2012-03-07 04:05:56 +08:00
|
|
|
if (!ArrayWithObjectsMethod) {
|
|
|
|
Selector
|
2014-10-29 02:28:16 +08:00
|
|
|
Sel = NSAPIObj->getNSArraySelector(NSAPI::NSArr_arrayWithObjectsCount);
|
|
|
|
ObjCMethodDecl *Method = NSArrayDecl->lookupClassMethod(Sel);
|
2012-05-13 01:32:44 +08:00
|
|
|
if (!Method && getLangOpts().DebuggerObjCLiteral) {
|
2014-05-26 14:22:03 +08:00
|
|
|
TypeSourceInfo *ReturnTInfo = nullptr;
|
2014-01-26 00:55:45 +08:00
|
|
|
Method = ObjCMethodDecl::Create(
|
|
|
|
Context, SourceLocation(), SourceLocation(), Sel, IdT, ReturnTInfo,
|
2014-10-29 02:28:16 +08:00
|
|
|
Context.getTranslationUnitDecl(), false /*Instance*/,
|
2014-01-26 00:55:45 +08:00
|
|
|
false /*isVariadic*/,
|
|
|
|
/*isPropertyAccessor=*/false,
|
|
|
|
/*isImplicitlyDeclared=*/true, /*isDefined=*/false,
|
|
|
|
ObjCMethodDecl::Required, false);
|
2012-03-07 04:05:56 +08:00
|
|
|
SmallVector<ParmVarDecl *, 2> Params;
|
2012-05-13 01:32:44 +08:00
|
|
|
ParmVarDecl *objects = ParmVarDecl::Create(Context, Method,
|
2012-05-13 01:32:52 +08:00
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
&Context.Idents.get("objects"),
|
|
|
|
Context.getPointerType(IdT),
|
2014-05-26 14:22:03 +08:00
|
|
|
/*TInfo=*/nullptr,
|
|
|
|
SC_None, nullptr);
|
2012-03-07 04:05:56 +08:00
|
|
|
Params.push_back(objects);
|
2012-05-13 01:32:44 +08:00
|
|
|
ParmVarDecl *cnt = ParmVarDecl::Create(Context, Method,
|
2012-05-13 01:32:52 +08:00
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
&Context.Idents.get("cnt"),
|
|
|
|
Context.UnsignedLongTy,
|
2014-05-26 14:22:03 +08:00
|
|
|
/*TInfo=*/nullptr, SC_None,
|
|
|
|
nullptr);
|
2012-03-07 04:05:56 +08:00
|
|
|
Params.push_back(cnt);
|
2013-05-05 08:41:58 +08:00
|
|
|
Method->setMethodParams(Context, Params, None);
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
2015-07-24 13:09:40 +08:00
|
|
|
if (!validateBoxingMethod(*this, Loc, NSArrayDecl, Sel, Method))
|
2012-03-07 04:05:56 +08:00
|
|
|
return ExprError();
|
2012-05-13 01:32:44 +08:00
|
|
|
|
2012-05-13 01:32:56 +08:00
|
|
|
// Dig out the type that all elements should be converted to.
|
2014-07-07 17:02:20 +08:00
|
|
|
QualType T = Method->parameters()[0]->getType();
|
2012-05-13 01:32:56 +08:00
|
|
|
const PointerType *PtrT = T->getAs<PointerType>();
|
2018-07-31 03:24:48 +08:00
|
|
|
if (!PtrT ||
|
2012-05-13 01:32:56 +08:00
|
|
|
!Context.hasSameUnqualifiedType(PtrT->getPointeeType(), IdT)) {
|
|
|
|
Diag(SR.getBegin(), diag::err_objc_literal_method_sig)
|
|
|
|
<< Sel;
|
2014-07-07 17:02:20 +08:00
|
|
|
Diag(Method->parameters()[0]->getLocation(),
|
2012-05-13 01:32:56 +08:00
|
|
|
diag::note_objc_literal_method_param)
|
2018-07-31 03:24:48 +08:00
|
|
|
<< 0 << T
|
2012-05-13 01:32:56 +08:00
|
|
|
<< Context.getPointerType(IdT.withConst());
|
|
|
|
return ExprError();
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-05-13 01:32:56 +08:00
|
|
|
// Check that the 'count' parameter is integral.
|
2014-07-07 17:02:20 +08:00
|
|
|
if (!Method->parameters()[1]->getType()->isIntegerType()) {
|
2012-05-13 01:32:56 +08:00
|
|
|
Diag(SR.getBegin(), diag::err_objc_literal_method_sig)
|
|
|
|
<< Sel;
|
2014-07-07 17:02:20 +08:00
|
|
|
Diag(Method->parameters()[1]->getLocation(),
|
2012-05-13 01:32:56 +08:00
|
|
|
diag::note_objc_literal_method_param)
|
2018-07-31 03:24:48 +08:00
|
|
|
<< 1
|
2014-07-07 17:02:20 +08:00
|
|
|
<< Method->parameters()[1]->getType()
|
2012-05-13 01:32:56 +08:00
|
|
|
<< "integral";
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
|
|
|
// We've found a good +arrayWithObjects:count: method. Save it!
|
2012-05-13 01:32:44 +08:00
|
|
|
ArrayWithObjectsMethod = Method;
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
2014-07-07 17:02:20 +08:00
|
|
|
QualType ObjectsType = ArrayWithObjectsMethod->parameters()[0]->getType();
|
2012-05-13 01:32:56 +08:00
|
|
|
QualType RequiredType = ObjectsType->castAs<PointerType>()->getPointeeType();
|
2012-03-07 04:05:56 +08:00
|
|
|
|
|
|
|
// Check that each of the elements provided is valid in a collection literal,
|
|
|
|
// performing conversions as necessary.
|
2012-08-24 07:38:35 +08:00
|
|
|
Expr **ElementsBuffer = Elements.data();
|
2012-03-07 04:05:56 +08:00
|
|
|
for (unsigned I = 0, N = Elements.size(); I != N; ++I) {
|
|
|
|
ExprResult Converted = CheckObjCCollectionLiteralElement(*this,
|
|
|
|
ElementsBuffer[I],
|
2013-08-14 07:44:55 +08:00
|
|
|
RequiredType, true);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (Converted.isInvalid())
|
|
|
|
return ExprError();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
ElementsBuffer[I] = Converted.get();
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
|
|
|
QualType Ty
|
2012-03-07 04:05:56 +08:00
|
|
|
= Context.getObjCObjectPointerType(
|
|
|
|
Context.getObjCInterfaceType(NSArrayDecl));
|
|
|
|
|
|
|
|
return MaybeBindToTemporary(
|
2012-08-24 07:38:35 +08:00
|
|
|
ObjCArrayLiteral::Create(Context, Elements, Ty,
|
2014-10-29 02:28:16 +08:00
|
|
|
ArrayWithObjectsMethod, SR));
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
2015-12-25 07:58:15 +08:00
|
|
|
ExprResult Sema::BuildObjCDictionaryLiteral(SourceRange SR,
|
|
|
|
MutableArrayRef<ObjCDictionaryElement> Elements) {
|
2015-07-24 13:09:40 +08:00
|
|
|
SourceLocation Loc = SR.getBegin();
|
2012-03-07 04:05:56 +08:00
|
|
|
|
2015-07-24 13:09:40 +08:00
|
|
|
if (!NSDictionaryDecl) {
|
|
|
|
NSDictionaryDecl = LookupObjCInterfaceDeclForLiteral(*this, Loc,
|
|
|
|
Sema::LK_Dictionary);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (!NSDictionaryDecl) {
|
2015-07-24 13:09:40 +08:00
|
|
|
return ExprError();
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
}
|
2015-07-24 13:09:40 +08:00
|
|
|
|
2014-10-29 02:28:16 +08:00
|
|
|
// Find the dictionaryWithObjects:forKeys:count: method, if we haven't done
|
|
|
|
// so already.
|
2014-08-09 02:29:52 +08:00
|
|
|
QualType IdT = Context.getObjCIdType();
|
2012-03-07 04:05:56 +08:00
|
|
|
if (!DictionaryWithObjectsMethod) {
|
2014-10-29 02:28:16 +08:00
|
|
|
Selector Sel = NSAPIObj->getNSDictionarySelector(
|
|
|
|
NSAPI::NSDict_dictionaryWithObjectsForKeysCount);
|
|
|
|
ObjCMethodDecl *Method = NSDictionaryDecl->lookupClassMethod(Sel);
|
2012-05-13 01:32:44 +08:00
|
|
|
if (!Method && getLangOpts().DebuggerObjCLiteral) {
|
2018-07-31 03:24:48 +08:00
|
|
|
Method = ObjCMethodDecl::Create(Context,
|
2012-03-07 04:05:56 +08:00
|
|
|
SourceLocation(), SourceLocation(), Sel,
|
|
|
|
IdT,
|
2014-05-26 14:22:03 +08:00
|
|
|
nullptr /*TypeSourceInfo */,
|
2012-03-07 04:05:56 +08:00
|
|
|
Context.getTranslationUnitDecl(),
|
2014-10-29 02:28:16 +08:00
|
|
|
false /*Instance*/, false/*isVariadic*/,
|
2012-10-11 00:42:25 +08:00
|
|
|
/*isPropertyAccessor=*/false,
|
2012-03-07 04:05:56 +08:00
|
|
|
/*isImplicitlyDeclared=*/true, /*isDefined=*/false,
|
|
|
|
ObjCMethodDecl::Required,
|
|
|
|
false);
|
|
|
|
SmallVector<ParmVarDecl *, 3> Params;
|
2012-05-13 01:32:44 +08:00
|
|
|
ParmVarDecl *objects = ParmVarDecl::Create(Context, Method,
|
2012-05-13 01:32:52 +08:00
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
&Context.Idents.get("objects"),
|
|
|
|
Context.getPointerType(IdT),
|
2014-05-26 14:22:03 +08:00
|
|
|
/*TInfo=*/nullptr, SC_None,
|
|
|
|
nullptr);
|
2012-03-07 04:05:56 +08:00
|
|
|
Params.push_back(objects);
|
2012-05-13 01:32:44 +08:00
|
|
|
ParmVarDecl *keys = ParmVarDecl::Create(Context, Method,
|
2012-05-13 01:32:52 +08:00
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
&Context.Idents.get("keys"),
|
|
|
|
Context.getPointerType(IdT),
|
2014-05-26 14:22:03 +08:00
|
|
|
/*TInfo=*/nullptr, SC_None,
|
|
|
|
nullptr);
|
2012-03-07 04:05:56 +08:00
|
|
|
Params.push_back(keys);
|
2012-05-13 01:32:44 +08:00
|
|
|
ParmVarDecl *cnt = ParmVarDecl::Create(Context, Method,
|
2012-05-13 01:32:52 +08:00
|
|
|
SourceLocation(),
|
|
|
|
SourceLocation(),
|
|
|
|
&Context.Idents.get("cnt"),
|
|
|
|
Context.UnsignedLongTy,
|
2014-05-26 14:22:03 +08:00
|
|
|
/*TInfo=*/nullptr, SC_None,
|
|
|
|
nullptr);
|
2012-03-07 04:05:56 +08:00
|
|
|
Params.push_back(cnt);
|
2013-05-05 08:41:58 +08:00
|
|
|
Method->setMethodParams(Context, Params, None);
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
|
|
|
|
2012-05-13 01:32:44 +08:00
|
|
|
if (!validateBoxingMethod(*this, SR.getBegin(), NSDictionaryDecl, Sel,
|
|
|
|
Method))
|
|
|
|
return ExprError();
|
|
|
|
|
2012-05-13 01:32:56 +08:00
|
|
|
// Dig out the type that all values should be converted to.
|
2014-07-07 17:02:20 +08:00
|
|
|
QualType ValueT = Method->parameters()[0]->getType();
|
2012-05-13 01:32:56 +08:00
|
|
|
const PointerType *PtrValue = ValueT->getAs<PointerType>();
|
2018-07-31 03:24:48 +08:00
|
|
|
if (!PtrValue ||
|
2012-05-13 01:32:56 +08:00
|
|
|
!Context.hasSameUnqualifiedType(PtrValue->getPointeeType(), IdT)) {
|
|
|
|
Diag(SR.getBegin(), diag::err_objc_literal_method_sig)
|
|
|
|
<< Sel;
|
2014-07-07 17:02:20 +08:00
|
|
|
Diag(Method->parameters()[0]->getLocation(),
|
2012-05-13 01:32:56 +08:00
|
|
|
diag::note_objc_literal_method_param)
|
|
|
|
<< 0 << ValueT
|
|
|
|
<< Context.getPointerType(IdT.withConst());
|
|
|
|
return ExprError();
|
|
|
|
}
|
2012-03-07 04:05:56 +08:00
|
|
|
|
2012-05-13 01:32:56 +08:00
|
|
|
// Dig out the type that all keys should be converted to.
|
2014-07-07 17:02:20 +08:00
|
|
|
QualType KeyT = Method->parameters()[1]->getType();
|
2012-05-13 01:32:56 +08:00
|
|
|
const PointerType *PtrKey = KeyT->getAs<PointerType>();
|
2018-07-31 03:24:48 +08:00
|
|
|
if (!PtrKey ||
|
2012-05-13 01:32:56 +08:00
|
|
|
!Context.hasSameUnqualifiedType(PtrKey->getPointeeType(),
|
|
|
|
IdT)) {
|
|
|
|
bool err = true;
|
|
|
|
if (PtrKey) {
|
|
|
|
if (QIDNSCopying.isNull()) {
|
|
|
|
// key argument of selector is id<NSCopying>?
|
|
|
|
if (ObjCProtocolDecl *NSCopyingPDecl =
|
|
|
|
LookupProtocol(&Context.Idents.get("NSCopying"), SR.getBegin())) {
|
|
|
|
ObjCProtocolDecl *PQ[] = {NSCopyingPDecl};
|
2018-07-31 03:24:48 +08:00
|
|
|
QIDNSCopying =
|
2015-07-07 11:57:35 +08:00
|
|
|
Context.getObjCObjectType(Context.ObjCBuiltinIdTy, { },
|
|
|
|
llvm::makeArrayRef(
|
|
|
|
(ObjCProtocolDecl**) PQ,
|
2015-07-07 11:58:42 +08:00
|
|
|
1),
|
|
|
|
false);
|
2012-05-13 01:32:56 +08:00
|
|
|
QIDNSCopying = Context.getObjCObjectPointerType(QIDNSCopying);
|
|
|
|
}
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
2012-05-13 01:32:56 +08:00
|
|
|
if (!QIDNSCopying.isNull())
|
|
|
|
err = !Context.hasSameUnqualifiedType(PtrKey->getPointeeType(),
|
|
|
|
QIDNSCopying);
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-05-13 01:32:56 +08:00
|
|
|
if (err) {
|
|
|
|
Diag(SR.getBegin(), diag::err_objc_literal_method_sig)
|
|
|
|
<< Sel;
|
2014-07-07 17:02:20 +08:00
|
|
|
Diag(Method->parameters()[1]->getLocation(),
|
2012-05-13 01:32:56 +08:00
|
|
|
diag::note_objc_literal_method_param)
|
|
|
|
<< 1 << KeyT
|
|
|
|
<< Context.getPointerType(IdT.withConst());
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that the 'count' parameter is integral.
|
2014-07-07 17:02:20 +08:00
|
|
|
QualType CountType = Method->parameters()[2]->getType();
|
2012-05-13 01:32:56 +08:00
|
|
|
if (!CountType->isIntegerType()) {
|
2012-03-07 04:05:56 +08:00
|
|
|
Diag(SR.getBegin(), diag::err_objc_literal_method_sig)
|
2012-05-13 01:32:56 +08:00
|
|
|
<< Sel;
|
2014-07-07 17:02:20 +08:00
|
|
|
Diag(Method->parameters()[2]->getLocation(),
|
2012-03-07 04:05:56 +08:00
|
|
|
diag::note_objc_literal_method_param)
|
2012-05-13 01:32:56 +08:00
|
|
|
<< 2 << CountType
|
|
|
|
<< "integral";
|
2012-03-07 04:05:56 +08:00
|
|
|
return ExprError();
|
|
|
|
}
|
2012-05-13 01:32:56 +08:00
|
|
|
|
|
|
|
// We've found a good +dictionaryWithObjects:keys:count: method; save it!
|
|
|
|
DictionaryWithObjectsMethod = Method;
|
2012-03-07 04:05:56 +08:00
|
|
|
}
|
2012-05-13 01:32:56 +08:00
|
|
|
|
2014-07-07 17:02:20 +08:00
|
|
|
QualType ValuesT = DictionaryWithObjectsMethod->parameters()[0]->getType();
|
2012-05-13 01:32:56 +08:00
|
|
|
QualType ValueT = ValuesT->castAs<PointerType>()->getPointeeType();
|
2014-07-07 17:02:20 +08:00
|
|
|
QualType KeysT = DictionaryWithObjectsMethod->parameters()[1]->getType();
|
2012-05-13 01:32:56 +08:00
|
|
|
QualType KeyT = KeysT->castAs<PointerType>()->getPointeeType();
|
2012-03-07 04:05:56 +08:00
|
|
|
|
2018-07-31 03:24:48 +08:00
|
|
|
// Check that each of the keys and values provided is valid in a collection
|
2012-03-07 04:05:56 +08:00
|
|
|
// literal, performing conversions as necessary.
|
|
|
|
bool HasPackExpansions = false;
|
2015-12-25 07:58:15 +08:00
|
|
|
for (ObjCDictionaryElement &Element : Elements) {
|
2012-03-07 04:05:56 +08:00
|
|
|
// Check the key.
|
2015-12-25 07:58:15 +08:00
|
|
|
ExprResult Key = CheckObjCCollectionLiteralElement(*this, Element.Key,
|
2012-03-07 04:05:56 +08:00
|
|
|
KeyT);
|
|
|
|
if (Key.isInvalid())
|
|
|
|
return ExprError();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
// Check the value.
|
|
|
|
ExprResult Value
|
2015-12-25 07:58:15 +08:00
|
|
|
= CheckObjCCollectionLiteralElement(*this, Element.Value, ValueT);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (Value.isInvalid())
|
|
|
|
return ExprError();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-12-25 07:58:15 +08:00
|
|
|
Element.Key = Key.get();
|
|
|
|
Element.Value = Value.get();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-12-25 07:58:15 +08:00
|
|
|
if (Element.EllipsisLoc.isInvalid())
|
2012-03-07 04:05:56 +08:00
|
|
|
continue;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-12-25 07:58:15 +08:00
|
|
|
if (!Element.Key->containsUnexpandedParameterPack() &&
|
|
|
|
!Element.Value->containsUnexpandedParameterPack()) {
|
|
|
|
Diag(Element.EllipsisLoc,
|
2012-03-07 04:05:56 +08:00
|
|
|
diag::err_pack_expansion_without_parameter_packs)
|
2018-08-10 05:08:08 +08:00
|
|
|
<< SourceRange(Element.Key->getBeginLoc(),
|
2018-08-10 05:09:38 +08:00
|
|
|
Element.Value->getEndLoc());
|
2012-03-07 04:05:56 +08:00
|
|
|
return ExprError();
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
HasPackExpansions = true;
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
QualType Ty
|
|
|
|
= Context.getObjCObjectPointerType(
|
2013-08-19 15:57:02 +08:00
|
|
|
Context.getObjCInterfaceType(NSDictionaryDecl));
|
|
|
|
return MaybeBindToTemporary(ObjCDictionaryLiteral::Create(
|
2015-12-25 07:58:15 +08:00
|
|
|
Context, Elements, HasPackExpansions, Ty,
|
2014-10-29 02:28:16 +08:00
|
|
|
DictionaryWithObjectsMethod, SR));
|
2008-01-05 06:32:30 +08:00
|
|
|
}
|
|
|
|
|
2011-05-15 04:32:39 +08:00
|
|
|
ExprResult Sema::BuildObjCEncodeExpression(SourceLocation AtLoc,
|
2010-04-20 23:39:42 +08:00
|
|
|
TypeSourceInfo *EncodedTypeInfo,
|
2009-06-08 02:45:35 +08:00
|
|
|
SourceLocation RParenLoc) {
|
2010-04-20 23:39:42 +08:00
|
|
|
QualType EncodedType = EncodedTypeInfo->getType();
|
2009-06-08 02:45:35 +08:00
|
|
|
QualType StrTy;
|
2009-09-09 23:08:12 +08:00
|
|
|
if (EncodedType->isDependentType())
|
2009-06-08 02:45:35 +08:00
|
|
|
StrTy = Context.DependentTy;
|
|
|
|
else {
|
2011-06-17 06:34:44 +08:00
|
|
|
if (!EncodedType->getAsArrayTypeUnsafe() && //// Incomplete array is handled.
|
|
|
|
!EncodedType->isVoidType()) // void is handled too.
|
2011-05-15 04:32:39 +08:00
|
|
|
if (RequireCompleteType(AtLoc, EncodedType,
|
2012-05-05 00:32:21 +08:00
|
|
|
diag::err_incomplete_type_objc_at_encode,
|
|
|
|
EncodedTypeInfo->getTypeLoc()))
|
2011-05-15 04:32:39 +08:00
|
|
|
return ExprError();
|
|
|
|
|
2009-06-08 02:45:35 +08:00
|
|
|
std::string Str;
|
2014-08-23 07:17:52 +08:00
|
|
|
QualType NotEncodedT;
|
|
|
|
Context.getObjCEncodingForType(EncodedType, Str, nullptr, &NotEncodedT);
|
|
|
|
if (!NotEncodedT.isNull())
|
|
|
|
Diag(AtLoc, diag::warn_incomplete_encoded_type)
|
|
|
|
<< EncodedType << NotEncodedT;
|
2009-06-08 02:45:35 +08:00
|
|
|
|
|
|
|
// The type of @encode is the same as the type of the corresponding string,
|
|
|
|
// which is an array type.
|
Implement __builtin_LINE() et. al. to support source location capture.
Summary:
This patch implements the source location builtins `__builtin_LINE(), `__builtin_FUNCTION()`, `__builtin_FILE()` and `__builtin_COLUMN()`. These builtins are needed to implement [`std::experimental::source_location`](https://rawgit.com/cplusplus/fundamentals-ts/v2/main.html#reflection.src_loc.creation).
With the exception of `__builtin_COLUMN`, GCC also implements these builtins, and Clangs behavior is intended to match as closely as possible.
Reviewers: rsmith, joerg, aaron.ballman, bogner, majnemer, shafik, martong
Reviewed By: rsmith
Subscribers: rnkovacs, loskutov, riccibruno, mgorny, kunitoki, alexr, majnemer, hfinkel, cfe-commits
Differential Revision: https://reviews.llvm.org/D37035
llvm-svn: 360937
2019-05-17 05:04:15 +08:00
|
|
|
StrTy = Context.getStringLiteralArrayType(Context.CharTy, Str.size());
|
2009-06-08 02:45:35 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-04-20 23:39:42 +08:00
|
|
|
return new (Context) ObjCEncodeExpr(StrTy, EncodedTypeInfo, AtLoc, RParenLoc);
|
2009-06-08 02:45:35 +08:00
|
|
|
}
|
|
|
|
|
2010-08-27 07:41:50 +08:00
|
|
|
ExprResult Sema::ParseObjCEncodeExpression(SourceLocation AtLoc,
|
|
|
|
SourceLocation EncodeLoc,
|
|
|
|
SourceLocation LParenLoc,
|
|
|
|
ParsedType ty,
|
|
|
|
SourceLocation RParenLoc) {
|
2009-08-19 09:28:28 +08:00
|
|
|
// FIXME: Preserve type source info ?
|
2010-04-20 23:39:42 +08:00
|
|
|
TypeSourceInfo *TInfo;
|
|
|
|
QualType EncodedType = GetTypeFromParser(ty, &TInfo);
|
|
|
|
if (!TInfo)
|
|
|
|
TInfo = Context.getTrivialTypeSourceInfo(EncodedType,
|
2015-11-15 10:31:46 +08:00
|
|
|
getLocForEndOfToken(LParenLoc));
|
2008-01-05 06:32:30 +08:00
|
|
|
|
2010-04-20 23:39:42 +08:00
|
|
|
return BuildObjCEncodeExpression(AtLoc, TInfo, RParenLoc);
|
2008-01-05 06:32:30 +08:00
|
|
|
}
|
|
|
|
|
2014-03-13 02:34:01 +08:00
|
|
|
static bool HelperToDiagnoseMismatchedMethodsInGlobalPool(Sema &S,
|
|
|
|
SourceLocation AtLoc,
|
2014-06-25 01:02:19 +08:00
|
|
|
SourceLocation LParenLoc,
|
|
|
|
SourceLocation RParenLoc,
|
2014-03-13 02:34:01 +08:00
|
|
|
ObjCMethodDecl *Method,
|
|
|
|
ObjCMethodList &MethList) {
|
|
|
|
ObjCMethodList *M = &MethList;
|
|
|
|
bool Warned = false;
|
|
|
|
for (M = M->getNext(); M; M=M->getNext()) {
|
2014-12-27 11:58:08 +08:00
|
|
|
ObjCMethodDecl *MatchingMethodDecl = M->getMethod();
|
2014-03-13 02:34:01 +08:00
|
|
|
if (MatchingMethodDecl == Method ||
|
|
|
|
isa<ObjCImplDecl>(MatchingMethodDecl->getDeclContext()) ||
|
|
|
|
MatchingMethodDecl->getSelector() != Method->getSelector())
|
|
|
|
continue;
|
|
|
|
if (!S.MatchTwoMethodDeclarations(Method,
|
|
|
|
MatchingMethodDecl, Sema::MMS_loose)) {
|
|
|
|
if (!Warned) {
|
|
|
|
Warned = true;
|
2016-12-03 07:00:28 +08:00
|
|
|
S.Diag(AtLoc, diag::warn_multiple_selectors)
|
2014-06-25 01:02:19 +08:00
|
|
|
<< Method->getSelector() << FixItHint::CreateInsertion(LParenLoc, "(")
|
|
|
|
<< FixItHint::CreateInsertion(RParenLoc, ")");
|
2014-03-13 02:34:01 +08:00
|
|
|
S.Diag(Method->getLocation(), diag::note_method_declared_at)
|
|
|
|
<< Method->getDeclName();
|
|
|
|
}
|
|
|
|
S.Diag(MatchingMethodDecl->getLocation(), diag::note_method_declared_at)
|
|
|
|
<< MatchingMethodDecl->getDeclName();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Warned;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void DiagnoseMismatchedSelectors(Sema &S, SourceLocation AtLoc,
|
2014-06-25 01:02:19 +08:00
|
|
|
ObjCMethodDecl *Method,
|
|
|
|
SourceLocation LParenLoc,
|
|
|
|
SourceLocation RParenLoc,
|
|
|
|
bool WarnMultipleSelectors) {
|
|
|
|
if (!WarnMultipleSelectors ||
|
2016-12-03 07:00:28 +08:00
|
|
|
S.Diags.isIgnored(diag::warn_multiple_selectors, SourceLocation()))
|
2014-03-13 02:34:01 +08:00
|
|
|
return;
|
|
|
|
bool Warned = false;
|
|
|
|
for (Sema::GlobalMethodPool::iterator b = S.MethodPool.begin(),
|
|
|
|
e = S.MethodPool.end(); b != e; b++) {
|
|
|
|
// first, instance methods
|
|
|
|
ObjCMethodList &InstMethList = b->second.first;
|
2014-06-25 01:02:19 +08:00
|
|
|
if (HelperToDiagnoseMismatchedMethodsInGlobalPool(S, AtLoc, LParenLoc, RParenLoc,
|
2014-03-13 02:34:01 +08:00
|
|
|
Method, InstMethList))
|
|
|
|
Warned = true;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2014-03-13 02:34:01 +08:00
|
|
|
// second, class methods
|
|
|
|
ObjCMethodList &ClsMethList = b->second.second;
|
2014-06-25 01:02:19 +08:00
|
|
|
if (HelperToDiagnoseMismatchedMethodsInGlobalPool(S, AtLoc, LParenLoc, RParenLoc,
|
|
|
|
Method, ClsMethList) || Warned)
|
2014-03-13 02:34:01 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-27 07:41:50 +08:00
|
|
|
ExprResult Sema::ParseObjCSelectorExpression(Selector Sel,
|
|
|
|
SourceLocation AtLoc,
|
|
|
|
SourceLocation SelLoc,
|
|
|
|
SourceLocation LParenLoc,
|
2014-06-25 01:02:19 +08:00
|
|
|
SourceLocation RParenLoc,
|
|
|
|
bool WarnMultipleSelectors) {
|
2013-01-23 02:35:43 +08:00
|
|
|
ObjCMethodDecl *Method = LookupInstanceMethodInGlobalPool(Sel,
|
2015-04-16 01:26:21 +08:00
|
|
|
SourceRange(LParenLoc, RParenLoc));
|
2013-01-23 02:35:43 +08:00
|
|
|
if (!Method)
|
|
|
|
Method = LookupFactoryMethodInGlobalPool(Sel,
|
2009-06-17 00:25:00 +08:00
|
|
|
SourceRange(LParenLoc, RParenLoc));
|
2013-06-06 02:46:14 +08:00
|
|
|
if (!Method) {
|
|
|
|
if (const ObjCMethodDecl *OM = SelectorsForTypoCorrection(Sel)) {
|
|
|
|
Selector MatchedSel = OM->getSelector();
|
|
|
|
SourceRange SelectorRange(LParenLoc.getLocWithOffset(1),
|
|
|
|
RParenLoc.getLocWithOffset(-1));
|
|
|
|
Diag(SelLoc, diag::warn_undeclared_selector_with_typo)
|
|
|
|
<< Sel << MatchedSel
|
|
|
|
<< FixItHint::CreateReplacement(SelectorRange, MatchedSel.getAsString());
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2013-06-06 02:46:14 +08:00
|
|
|
} else
|
|
|
|
Diag(SelLoc, diag::warn_undeclared_selector) << Sel;
|
2014-03-13 02:34:01 +08:00
|
|
|
} else
|
2014-06-25 01:02:19 +08:00
|
|
|
DiagnoseMismatchedSelectors(*this, AtLoc, Method, LParenLoc, RParenLoc,
|
|
|
|
WarnMultipleSelectors);
|
2015-03-27 08:55:05 +08:00
|
|
|
|
2014-05-10 03:51:39 +08:00
|
|
|
if (Method &&
|
|
|
|
Method->getImplementationControl() != ObjCMethodDecl::Optional &&
|
2015-03-27 08:55:05 +08:00
|
|
|
!getSourceManager().isInSystemHeader(Method->getLocation()))
|
|
|
|
ReferencedSelectors.insert(std::make_pair(Sel, AtLoc));
|
2010-07-23 02:24:20 +08:00
|
|
|
|
2018-07-31 03:24:48 +08:00
|
|
|
// In ARC, forbid the user from using @selector for
|
2011-06-16 07:02:42 +08:00
|
|
|
// retain/release/autorelease/dealloc/retainCount.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (getLangOpts().ObjCAutoRefCount) {
|
2011-06-16 07:02:42 +08:00
|
|
|
switch (Sel.getMethodFamily()) {
|
|
|
|
case OMF_retain:
|
|
|
|
case OMF_release:
|
|
|
|
case OMF_autorelease:
|
|
|
|
case OMF_retainCount:
|
|
|
|
case OMF_dealloc:
|
2018-07-31 03:24:48 +08:00
|
|
|
Diag(AtLoc, diag::err_arc_illegal_selector) <<
|
2011-06-16 07:02:42 +08:00
|
|
|
Sel << SourceRange(LParenLoc, RParenLoc);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case OMF_None:
|
|
|
|
case OMF_alloc:
|
|
|
|
case OMF_copy:
|
2011-08-29 06:35:17 +08:00
|
|
|
case OMF_finalize:
|
2011-06-16 07:02:42 +08:00
|
|
|
case OMF_init:
|
|
|
|
case OMF_mutableCopy:
|
|
|
|
case OMF_new:
|
|
|
|
case OMF_self:
|
2014-08-23 00:57:26 +08:00
|
|
|
case OMF_initialize:
|
2011-07-06 06:38:59 +08:00
|
|
|
case OMF_performSelector:
|
2011-06-16 07:02:42 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-02-18 14:06:56 +08:00
|
|
|
QualType Ty = Context.getObjCSelType();
|
2010-02-04 04:11:42 +08:00
|
|
|
return new (Context) ObjCSelectorExpr(Ty, Sel, AtLoc, RParenLoc);
|
2008-01-05 06:32:30 +08:00
|
|
|
}
|
|
|
|
|
2010-08-27 07:41:50 +08:00
|
|
|
ExprResult Sema::ParseObjCProtocolExpression(IdentifierInfo *ProtocolId,
|
|
|
|
SourceLocation AtLoc,
|
|
|
|
SourceLocation ProtoLoc,
|
|
|
|
SourceLocation LParenLoc,
|
2012-05-16 08:50:02 +08:00
|
|
|
SourceLocation ProtoIdLoc,
|
2010-08-27 07:41:50 +08:00
|
|
|
SourceLocation RParenLoc) {
|
2012-05-16 08:50:02 +08:00
|
|
|
ObjCProtocolDecl* PDecl = LookupProtocol(ProtocolId, ProtoIdLoc);
|
2008-01-05 06:32:30 +08:00
|
|
|
if (!PDecl) {
|
2008-11-19 16:23:25 +08:00
|
|
|
Diag(ProtoLoc, diag::err_undeclared_protocol) << ProtocolId;
|
2008-01-05 06:32:30 +08:00
|
|
|
return true;
|
|
|
|
}
|
2018-08-18 06:18:08 +08:00
|
|
|
if (!PDecl->hasDefinition()) {
|
|
|
|
Diag(ProtoLoc, diag::err_atprotocol_protocol) << PDecl;
|
|
|
|
Diag(PDecl->getLocation(), diag::note_entity_declared_at) << PDecl;
|
|
|
|
} else {
|
2014-07-26 03:45:01 +08:00
|
|
|
PDecl = PDecl->getDefinition();
|
2018-08-18 06:18:08 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-02-18 14:06:56 +08:00
|
|
|
QualType Ty = Context.getObjCProtoType();
|
|
|
|
if (Ty.isNull())
|
2008-01-05 06:32:30 +08:00
|
|
|
return true;
|
2009-07-11 07:34:53 +08:00
|
|
|
Ty = Context.getObjCObjectPointerType(Ty);
|
2012-05-16 08:50:02 +08:00
|
|
|
return new (Context) ObjCProtocolExpr(Ty, PDecl, AtLoc, ProtoIdLoc, RParenLoc);
|
2008-01-05 06:32:30 +08:00
|
|
|
}
|
|
|
|
|
2011-02-03 17:00:02 +08:00
|
|
|
/// Try to capture an implicit reference to 'self'.
|
2012-02-04 06:47:37 +08:00
|
|
|
ObjCMethodDecl *Sema::tryCaptureObjCSelf(SourceLocation Loc) {
|
|
|
|
DeclContext *DC = getFunctionLevelDeclContext();
|
2011-02-03 17:00:02 +08:00
|
|
|
|
|
|
|
// If we're not in an ObjC method, error out. Note that, unlike the
|
|
|
|
// C++ case, we don't require an instance method --- class methods
|
|
|
|
// still have a 'self', and we really do still need to capture it!
|
|
|
|
ObjCMethodDecl *method = dyn_cast<ObjCMethodDecl>(DC);
|
|
|
|
if (!method)
|
2014-05-26 14:22:03 +08:00
|
|
|
return nullptr;
|
2011-02-03 17:00:02 +08:00
|
|
|
|
2012-02-18 17:37:24 +08:00
|
|
|
tryCaptureVariable(method->getSelfDecl(), Loc);
|
2011-02-03 17:00:02 +08:00
|
|
|
|
|
|
|
return method;
|
|
|
|
}
|
|
|
|
|
2011-09-10 04:05:21 +08:00
|
|
|
static QualType stripObjCInstanceType(ASTContext &Context, QualType T) {
|
2015-06-20 02:14:38 +08:00
|
|
|
QualType origType = T;
|
|
|
|
if (auto nullability = AttributedType::stripOuterNullability(T)) {
|
|
|
|
if (T == Context.getObjCInstanceType()) {
|
|
|
|
return Context.getAttributedType(
|
|
|
|
AttributedType::getNullabilityAttrKind(*nullability),
|
|
|
|
Context.getObjCIdType(),
|
|
|
|
Context.getObjCIdType());
|
|
|
|
}
|
|
|
|
|
|
|
|
return origType;
|
|
|
|
}
|
|
|
|
|
2011-09-10 04:05:21 +08:00
|
|
|
if (T == Context.getObjCInstanceType())
|
|
|
|
return Context.getObjCIdType();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2015-06-20 02:14:38 +08:00
|
|
|
return origType;
|
2011-09-10 04:05:21 +08:00
|
|
|
}
|
|
|
|
|
2015-06-20 02:14:38 +08:00
|
|
|
/// Determine the result type of a message send based on the receiver type,
|
|
|
|
/// method, and the kind of message send.
|
|
|
|
///
|
|
|
|
/// This is the "base" result type, which will still need to be adjusted
|
|
|
|
/// to account for nullability.
|
|
|
|
static QualType getBaseMessageSendResultType(Sema &S,
|
|
|
|
QualType ReceiverType,
|
|
|
|
ObjCMethodDecl *Method,
|
|
|
|
bool isClassMessage,
|
|
|
|
bool isSuperMessage) {
|
2011-06-11 09:09:30 +08:00
|
|
|
assert(Method && "Must have a method");
|
|
|
|
if (!Method->hasRelatedResultType())
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
return Method->getSendResultType(ReceiverType);
|
2015-06-20 02:14:38 +08:00
|
|
|
|
|
|
|
ASTContext &Context = S.Context;
|
|
|
|
|
|
|
|
// Local function that transfers the nullability of the method's
|
|
|
|
// result type to the returned result.
|
|
|
|
auto transferNullability = [&](QualType type) -> QualType {
|
|
|
|
// If the method's result type has nullability, extract it.
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
if (auto nullability = Method->getSendResultType(ReceiverType)
|
|
|
|
->getNullability(Context)){
|
2015-06-20 02:14:38 +08:00
|
|
|
// Strip off any outer nullability sugar from the provided type.
|
|
|
|
(void)AttributedType::stripOuterNullability(type);
|
|
|
|
|
|
|
|
// Form a new attributed type using the method result type's nullability.
|
|
|
|
return Context.getAttributedType(
|
|
|
|
AttributedType::getNullabilityAttrKind(*nullability),
|
|
|
|
type,
|
|
|
|
type);
|
|
|
|
}
|
|
|
|
|
|
|
|
return type;
|
|
|
|
};
|
|
|
|
|
2011-06-11 09:09:30 +08:00
|
|
|
// If a method has a related return type:
|
|
|
|
// - if the method found is an instance method, but the message send
|
|
|
|
// was a class message send, T is the declared return type of the method
|
|
|
|
// found
|
|
|
|
if (Method->isInstanceMethod() && isClassMessage)
|
2018-07-31 03:24:48 +08:00
|
|
|
return stripObjCInstanceType(Context,
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
Method->getSendResultType(ReceiverType));
|
2015-06-20 02:14:38 +08:00
|
|
|
|
|
|
|
// - if the receiver is super, T is a pointer to the class of the
|
2011-06-11 09:09:30 +08:00
|
|
|
// enclosing method definition
|
|
|
|
if (isSuperMessage) {
|
2015-06-20 02:14:38 +08:00
|
|
|
if (ObjCMethodDecl *CurMethod = S.getCurMethodDecl())
|
|
|
|
if (ObjCInterfaceDecl *Class = CurMethod->getClassInterface()) {
|
|
|
|
return transferNullability(
|
|
|
|
Context.getObjCObjectPointerType(
|
|
|
|
Context.getObjCInterfaceType(Class)));
|
|
|
|
}
|
2011-06-11 09:09:30 +08:00
|
|
|
}
|
2015-06-20 02:14:38 +08:00
|
|
|
|
2011-06-11 09:09:30 +08:00
|
|
|
// - if the receiver is the name of a class U, T is a pointer to U
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
if (ReceiverType->getAsObjCInterfaceType())
|
2015-06-20 02:14:38 +08:00
|
|
|
return transferNullability(Context.getObjCObjectPointerType(ReceiverType));
|
|
|
|
// - if the receiver is of type Class or qualified Class type,
|
2011-06-11 09:09:30 +08:00
|
|
|
// T is the declared return type of the method.
|
|
|
|
if (ReceiverType->isObjCClassType() ||
|
|
|
|
ReceiverType->isObjCQualifiedClassType())
|
2018-07-31 03:24:48 +08:00
|
|
|
return stripObjCInstanceType(Context,
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
Method->getSendResultType(ReceiverType));
|
2015-06-20 02:14:38 +08:00
|
|
|
|
2011-06-11 09:09:30 +08:00
|
|
|
// - if the receiver is id, qualified id, Class, or qualified Class, T
|
|
|
|
// is the receiver type, otherwise
|
|
|
|
// - T is the type of the receiver expression.
|
2015-06-20 02:14:38 +08:00
|
|
|
return transferNullability(ReceiverType);
|
|
|
|
}
|
|
|
|
|
2018-12-21 06:11:11 +08:00
|
|
|
QualType Sema::getMessageSendResultType(const Expr *Receiver,
|
|
|
|
QualType ReceiverType,
|
2015-06-20 02:14:38 +08:00
|
|
|
ObjCMethodDecl *Method,
|
|
|
|
bool isClassMessage,
|
|
|
|
bool isSuperMessage) {
|
|
|
|
// Produce the result type.
|
|
|
|
QualType resultType = getBaseMessageSendResultType(*this, ReceiverType,
|
|
|
|
Method,
|
|
|
|
isClassMessage,
|
|
|
|
isSuperMessage);
|
|
|
|
|
2015-06-20 02:25:57 +08:00
|
|
|
// If this is a class message, ignore the nullability of the receiver.
|
2018-12-21 06:11:11 +08:00
|
|
|
if (isClassMessage) {
|
|
|
|
// In a class method, class messages to 'self' that return instancetype can
|
|
|
|
// be typed as the current class. We can safely do this in ARC because self
|
|
|
|
// can't be reassigned, and we do it unsafely outside of ARC because in
|
|
|
|
// practice people never reassign self in class methods and there's some
|
|
|
|
// virtue in not being aggressively pedantic.
|
|
|
|
if (Receiver && Receiver->isObjCSelfExpr()) {
|
|
|
|
assert(ReceiverType->isObjCClassType() && "expected a Class self");
|
|
|
|
QualType T = Method->getSendResultType(ReceiverType);
|
|
|
|
AttributedType::stripOuterNullability(T);
|
|
|
|
if (T == Context.getObjCInstanceType()) {
|
|
|
|
const ObjCMethodDecl *MD = cast<ObjCMethodDecl>(
|
|
|
|
cast<ImplicitParamDecl>(
|
|
|
|
cast<DeclRefExpr>(Receiver->IgnoreParenImpCasts())->getDecl())
|
|
|
|
->getDeclContext());
|
|
|
|
assert(MD->isClassMethod() && "expected a class method");
|
|
|
|
QualType NewResultType = Context.getObjCObjectPointerType(
|
|
|
|
Context.getObjCInterfaceType(MD->getClassInterface()));
|
|
|
|
if (auto Nullability = resultType->getNullability(Context))
|
|
|
|
NewResultType = Context.getAttributedType(
|
|
|
|
AttributedType::getNullabilityAttrKind(*Nullability),
|
|
|
|
NewResultType, NewResultType);
|
|
|
|
return NewResultType;
|
|
|
|
}
|
|
|
|
}
|
2015-06-20 02:25:57 +08:00
|
|
|
return resultType;
|
2018-12-21 06:11:11 +08:00
|
|
|
}
|
2015-06-20 02:25:57 +08:00
|
|
|
|
2018-07-27 01:51:13 +08:00
|
|
|
// There is nothing left to do if the result type cannot have a nullability
|
|
|
|
// specifier.
|
|
|
|
if (!resultType->canHaveNullability())
|
|
|
|
return resultType;
|
|
|
|
|
2015-06-20 02:14:38 +08:00
|
|
|
// Map the nullability of the result into a table index.
|
|
|
|
unsigned receiverNullabilityIdx = 0;
|
|
|
|
if (auto nullability = ReceiverType->getNullability(Context))
|
|
|
|
receiverNullabilityIdx = 1 + static_cast<unsigned>(*nullability);
|
|
|
|
|
|
|
|
unsigned resultNullabilityIdx = 0;
|
|
|
|
if (auto nullability = resultType->getNullability(Context))
|
|
|
|
resultNullabilityIdx = 1 + static_cast<unsigned>(*nullability);
|
|
|
|
|
|
|
|
// The table of nullability mappings, indexed by the receiver's nullability
|
|
|
|
// and then the result type's nullability.
|
|
|
|
static const uint8_t None = 0;
|
|
|
|
static const uint8_t NonNull = 1;
|
|
|
|
static const uint8_t Nullable = 2;
|
|
|
|
static const uint8_t Unspecified = 3;
|
|
|
|
static const uint8_t nullabilityMap[4][4] = {
|
|
|
|
// None NonNull Nullable Unspecified
|
|
|
|
/* None */ { None, None, Nullable, None },
|
|
|
|
/* NonNull */ { None, NonNull, Nullable, Unspecified },
|
|
|
|
/* Nullable */ { Nullable, Nullable, Nullable, Nullable },
|
|
|
|
/* Unspecified */ { None, Unspecified, Nullable, Unspecified }
|
|
|
|
};
|
|
|
|
|
|
|
|
unsigned newResultNullabilityIdx
|
|
|
|
= nullabilityMap[receiverNullabilityIdx][resultNullabilityIdx];
|
|
|
|
if (newResultNullabilityIdx == resultNullabilityIdx)
|
|
|
|
return resultType;
|
|
|
|
|
|
|
|
// Strip off the existing nullability. This removes as little type sugar as
|
|
|
|
// possible.
|
|
|
|
do {
|
|
|
|
if (auto attributed = dyn_cast<AttributedType>(resultType.getTypePtr())) {
|
|
|
|
resultType = attributed->getModifiedType();
|
|
|
|
} else {
|
|
|
|
resultType = resultType.getDesugaredType(Context);
|
|
|
|
}
|
|
|
|
} while (resultType->getNullability(Context));
|
|
|
|
|
|
|
|
// Add nullability back if needed.
|
|
|
|
if (newResultNullabilityIdx > 0) {
|
|
|
|
auto newNullability
|
|
|
|
= static_cast<NullabilityKind>(newResultNullabilityIdx-1);
|
|
|
|
return Context.getAttributedType(
|
|
|
|
AttributedType::getNullabilityAttrKind(newNullability),
|
|
|
|
resultType, resultType);
|
|
|
|
}
|
|
|
|
|
|
|
|
return resultType;
|
2011-06-11 09:09:30 +08:00
|
|
|
}
|
|
|
|
|
2013-03-19 15:04:25 +08:00
|
|
|
/// Look for an ObjC method whose result type exactly matches the given type.
|
|
|
|
static const ObjCMethodDecl *
|
|
|
|
findExplicitInstancetypeDeclarer(const ObjCMethodDecl *MD,
|
|
|
|
QualType instancetype) {
|
2014-01-26 00:55:45 +08:00
|
|
|
if (MD->getReturnType() == instancetype)
|
|
|
|
return MD;
|
2013-03-19 15:04:25 +08:00
|
|
|
|
|
|
|
// For these purposes, a method in an @implementation overrides a
|
|
|
|
// declaration in the @interface.
|
|
|
|
if (const ObjCImplDecl *impl =
|
|
|
|
dyn_cast<ObjCImplDecl>(MD->getDeclContext())) {
|
|
|
|
const ObjCContainerDecl *iface;
|
2018-07-31 03:24:48 +08:00
|
|
|
if (const ObjCCategoryImplDecl *catImpl =
|
2013-03-19 15:04:25 +08:00
|
|
|
dyn_cast<ObjCCategoryImplDecl>(impl)) {
|
|
|
|
iface = catImpl->getCategoryDecl();
|
|
|
|
} else {
|
|
|
|
iface = impl->getClassInterface();
|
|
|
|
}
|
|
|
|
|
2018-07-31 03:24:48 +08:00
|
|
|
const ObjCMethodDecl *ifaceMD =
|
2013-03-19 15:04:25 +08:00
|
|
|
iface->getMethod(MD->getSelector(), MD->isInstanceMethod());
|
|
|
|
if (ifaceMD) return findExplicitInstancetypeDeclarer(ifaceMD, instancetype);
|
|
|
|
}
|
|
|
|
|
|
|
|
SmallVector<const ObjCMethodDecl *, 4> overrides;
|
|
|
|
MD->getOverriddenMethods(overrides);
|
|
|
|
for (unsigned i = 0, e = overrides.size(); i != e; ++i) {
|
|
|
|
if (const ObjCMethodDecl *result =
|
|
|
|
findExplicitInstancetypeDeclarer(overrides[i], instancetype))
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-05-26 14:22:03 +08:00
|
|
|
return nullptr;
|
2013-03-19 15:04:25 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void Sema::EmitRelatedResultTypeNoteForReturn(QualType destType) {
|
|
|
|
// Only complain if we're in an ObjC method and the required return
|
|
|
|
// type doesn't match the method's declared return type.
|
|
|
|
ObjCMethodDecl *MD = dyn_cast<ObjCMethodDecl>(CurContext);
|
|
|
|
if (!MD || !MD->hasRelatedResultType() ||
|
2014-01-26 00:55:45 +08:00
|
|
|
Context.hasSameUnqualifiedType(destType, MD->getReturnType()))
|
2013-03-19 15:04:25 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
// Look for a method overridden by this method which explicitly uses
|
|
|
|
// 'instancetype'.
|
|
|
|
if (const ObjCMethodDecl *overridden =
|
|
|
|
findExplicitInstancetypeDeclarer(MD, Context.getObjCInstanceType())) {
|
2014-08-01 21:20:09 +08:00
|
|
|
SourceRange range = overridden->getReturnTypeSourceRange();
|
|
|
|
SourceLocation loc = range.getBegin();
|
2013-03-19 15:04:25 +08:00
|
|
|
if (loc.isInvalid())
|
|
|
|
loc = overridden->getLocation();
|
|
|
|
Diag(loc, diag::note_related_result_type_explicit)
|
|
|
|
<< /*current method*/ 1 << range;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, if we have an interesting method family, note that.
|
|
|
|
// This should always trigger if the above didn't.
|
|
|
|
if (ObjCMethodFamily family = MD->getMethodFamily())
|
|
|
|
Diag(MD->getLocation(), diag::note_related_result_type_family)
|
|
|
|
<< /*current method*/ 1
|
|
|
|
<< family;
|
|
|
|
}
|
|
|
|
|
2011-06-11 09:09:30 +08:00
|
|
|
void Sema::EmitRelatedResultTypeNote(const Expr *E) {
|
|
|
|
E = E->IgnoreParenImpCasts();
|
|
|
|
const ObjCMessageExpr *MsgSend = dyn_cast<ObjCMessageExpr>(E);
|
|
|
|
if (!MsgSend)
|
|
|
|
return;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-11 09:09:30 +08:00
|
|
|
const ObjCMethodDecl *Method = MsgSend->getMethodDecl();
|
|
|
|
if (!Method)
|
|
|
|
return;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-11 09:09:30 +08:00
|
|
|
if (!Method->hasRelatedResultType())
|
|
|
|
return;
|
2014-01-26 00:55:45 +08:00
|
|
|
|
|
|
|
if (Context.hasSameUnqualifiedType(
|
|
|
|
Method->getReturnType().getNonReferenceType(), MsgSend->getType()))
|
2011-06-11 09:09:30 +08:00
|
|
|
return;
|
2014-01-26 00:55:45 +08:00
|
|
|
|
|
|
|
if (!Context.hasSameUnqualifiedType(Method->getReturnType(),
|
2011-09-08 09:46:34 +08:00
|
|
|
Context.getObjCInstanceType()))
|
|
|
|
return;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-11 09:09:30 +08:00
|
|
|
Diag(Method->getLocation(), diag::note_related_result_type_inferred)
|
|
|
|
<< Method->isInstanceMethod() << Method->getSelector()
|
|
|
|
<< MsgSend->getType();
|
|
|
|
}
|
2011-02-03 17:00:02 +08:00
|
|
|
|
2018-12-21 06:11:11 +08:00
|
|
|
bool Sema::CheckMessageArgumentTypes(
|
|
|
|
const Expr *Receiver, QualType ReceiverType, MultiExprArg Args,
|
|
|
|
Selector Sel, ArrayRef<SourceLocation> SelectorLocs, ObjCMethodDecl *Method,
|
|
|
|
bool isClassMessage, bool isSuperMessage, SourceLocation lbrac,
|
|
|
|
SourceLocation rbrac, SourceRange RecRange, QualType &ReturnType,
|
|
|
|
ExprValueKind &VK) {
|
2013-05-01 08:24:09 +08:00
|
|
|
SourceLocation SelLoc;
|
|
|
|
if (!SelectorLocs.empty() && SelectorLocs.front().isValid())
|
|
|
|
SelLoc = SelectorLocs.front();
|
|
|
|
else
|
|
|
|
SelLoc = lbrac;
|
|
|
|
|
2008-09-11 08:01:56 +08:00
|
|
|
if (!Method) {
|
2008-09-11 08:04:36 +08:00
|
|
|
// Apply default argument promotion as for (C99 6.5.2.2p6).
|
2013-05-10 08:27:15 +08:00
|
|
|
for (unsigned i = 0, e = Args.size(); i != e; i++) {
|
2010-04-23 00:44:27 +08:00
|
|
|
if (Args[i]->isTypeDependent())
|
|
|
|
continue;
|
|
|
|
|
2013-03-04 15:34:02 +08:00
|
|
|
ExprResult result;
|
|
|
|
if (getLangOpts().DebuggerSupport) {
|
|
|
|
QualType paramTy; // ignored
|
2013-05-01 08:24:09 +08:00
|
|
|
result = checkUnknownAnyArg(SelLoc, Args[i], paramTy);
|
2013-03-04 15:34:02 +08:00
|
|
|
} else {
|
|
|
|
result = DefaultArgumentPromotion(Args[i]);
|
|
|
|
}
|
|
|
|
if (result.isInvalid())
|
2011-04-09 02:41:53 +08:00
|
|
|
return true;
|
2014-05-29 18:55:11 +08:00
|
|
|
Args[i] = result.get();
|
2010-04-23 00:44:27 +08:00
|
|
|
}
|
2008-09-11 08:04:36 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
unsigned DiagID;
|
2012-03-11 15:00:24 +08:00
|
|
|
if (getLangOpts().ObjCAutoRefCount)
|
2011-06-16 07:02:42 +08:00
|
|
|
DiagID = diag::err_arc_method_not_found;
|
|
|
|
else
|
|
|
|
DiagID = isClassMessage ? diag::warn_class_method_not_found
|
|
|
|
: diag::warn_inst_method_not_found;
|
2013-05-15 07:24:17 +08:00
|
|
|
if (!getLangOpts().DebuggerSupport) {
|
2013-06-18 23:31:36 +08:00
|
|
|
const ObjCMethodDecl *OMD = SelectorsForTypoCorrection(Sel, ReceiverType);
|
2013-06-19 01:10:58 +08:00
|
|
|
if (OMD && !OMD->isInvalidDecl()) {
|
2013-06-18 23:31:36 +08:00
|
|
|
if (getLangOpts().ObjCAutoRefCount)
|
2016-12-03 06:38:31 +08:00
|
|
|
DiagID = diag::err_method_not_found_with_typo;
|
2013-06-18 23:31:36 +08:00
|
|
|
else
|
|
|
|
DiagID = isClassMessage ? diag::warn_class_method_not_found_with_typo
|
|
|
|
: diag::warn_instance_method_not_found_with_typo;
|
2013-06-18 01:10:54 +08:00
|
|
|
Selector MatchedSel = OMD->getSelector();
|
|
|
|
SourceRange SelectorRange(SelectorLocs.front(), SelectorLocs.back());
|
2014-08-13 06:16:41 +08:00
|
|
|
if (MatchedSel.isUnarySelector())
|
|
|
|
Diag(SelLoc, DiagID)
|
|
|
|
<< Sel<< isClassMessage << MatchedSel
|
|
|
|
<< FixItHint::CreateReplacement(SelectorRange, MatchedSel.getAsString());
|
|
|
|
else
|
|
|
|
Diag(SelLoc, DiagID) << Sel<< isClassMessage << MatchedSel;
|
2013-06-18 01:10:54 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
Diag(SelLoc, DiagID)
|
2018-07-31 03:24:48 +08:00
|
|
|
<< Sel << isClassMessage << SourceRange(SelectorLocs.front(),
|
2012-09-01 01:03:18 +08:00
|
|
|
SelectorLocs.back());
|
2013-05-15 07:24:17 +08:00
|
|
|
// Find the class to which we are sending this message.
|
|
|
|
if (ReceiverType->isObjCObjectPointerType()) {
|
2014-08-20 07:39:17 +08:00
|
|
|
if (ObjCInterfaceDecl *ThisClass =
|
|
|
|
ReceiverType->getAs<ObjCObjectPointerType>()->getInterfaceDecl()) {
|
|
|
|
Diag(ThisClass->getLocation(), diag::note_receiver_class_declared);
|
|
|
|
if (!RecRange.isInvalid())
|
|
|
|
if (ThisClass->lookupClassMethod(Sel))
|
|
|
|
Diag(RecRange.getBegin(),diag::note_receiver_expr_here)
|
|
|
|
<< FixItHint::CreateReplacement(RecRange,
|
|
|
|
ThisClass->getNameAsString());
|
|
|
|
}
|
2013-05-15 07:24:17 +08:00
|
|
|
}
|
|
|
|
}
|
2011-07-14 01:56:40 +08:00
|
|
|
|
|
|
|
// In debuggers, we want to use __unknown_anytype for these
|
|
|
|
// results so that clients can cast them.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (getLangOpts().DebuggerSupport) {
|
2011-07-14 01:56:40 +08:00
|
|
|
ReturnType = Context.UnknownAnyTy;
|
|
|
|
} else {
|
|
|
|
ReturnType = Context.getObjCIdType();
|
|
|
|
}
|
2010-11-18 14:31:45 +08:00
|
|
|
VK = VK_RValue;
|
2008-09-11 08:01:56 +08:00
|
|
|
return false;
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2018-12-21 06:11:11 +08:00
|
|
|
ReturnType = getMessageSendResultType(Receiver, ReceiverType, Method,
|
|
|
|
isClassMessage, isSuperMessage);
|
2014-01-26 00:55:45 +08:00
|
|
|
VK = Expr::getValueKindForType(Method->getReturnType());
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2008-09-11 08:50:25 +08:00
|
|
|
unsigned NumNamedArgs = Sel.getNumArgs();
|
2010-04-08 08:30:06 +08:00
|
|
|
// Method might have more arguments than selector indicates. This is due
|
|
|
|
// to addition of c-style arguments in method.
|
|
|
|
if (Method->param_size() > Sel.getNumArgs())
|
|
|
|
NumNamedArgs = Method->param_size();
|
|
|
|
// FIXME. This need be cleaned up.
|
2013-05-10 08:27:15 +08:00
|
|
|
if (Args.size() < NumNamedArgs) {
|
2013-05-01 08:24:09 +08:00
|
|
|
Diag(SelLoc, diag::err_typecheck_call_too_few_args)
|
2013-05-10 08:27:15 +08:00
|
|
|
<< 2 << NumNamedArgs << static_cast<unsigned>(Args.size());
|
2010-04-08 08:30:06 +08:00
|
|
|
return false;
|
|
|
|
}
|
2008-09-11 08:50:25 +08:00
|
|
|
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
// Compute the set of type arguments to be substituted into each parameter
|
|
|
|
// type.
|
|
|
|
Optional<ArrayRef<QualType>> typeArgs
|
|
|
|
= ReceiverType->getObjCSubstitutions(Method->getDeclContext());
|
2009-04-12 16:11:20 +08:00
|
|
|
bool IsError = false;
|
2008-09-11 08:50:25 +08:00
|
|
|
for (unsigned i = 0; i < NumNamedArgs; i++) {
|
2010-04-23 00:44:27 +08:00
|
|
|
// We can't do any type-checking on a type-dependent argument.
|
|
|
|
if (Args[i]->isTypeDependent())
|
|
|
|
continue;
|
|
|
|
|
2008-01-05 06:32:30 +08:00
|
|
|
Expr *argExpr = Args[i];
|
2010-04-23 00:44:27 +08:00
|
|
|
|
2014-07-07 17:02:20 +08:00
|
|
|
ParmVarDecl *param = Method->parameters()[i];
|
2008-01-05 06:32:30 +08:00
|
|
|
assert(argExpr && "CheckMessageArgumentTypes(): missing expression");
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2018-03-02 09:53:15 +08:00
|
|
|
if (param->hasAttr<NoEscapeAttr>())
|
|
|
|
if (auto *BE = dyn_cast<BlockExpr>(
|
|
|
|
argExpr->IgnoreParenNoopCasts(Context)))
|
|
|
|
BE->getBlockDecl()->setDoesNotEscape();
|
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
// Strip the unbridged-cast placeholder expression off unless it's
|
|
|
|
// a consumed argument.
|
|
|
|
if (argExpr->hasPlaceholderType(BuiltinType::ARCUnbridgedCast) &&
|
|
|
|
!param->hasAttr<CFConsumedAttr>())
|
|
|
|
argExpr = stripARCUnbridgedCast(argExpr);
|
|
|
|
|
2012-11-14 08:49:39 +08:00
|
|
|
// If the parameter is __unknown_anytype, infer its type
|
|
|
|
// from the argument.
|
|
|
|
if (param->getType() == Context.UnknownAnyTy) {
|
2013-03-04 15:34:02 +08:00
|
|
|
QualType paramType;
|
2013-05-01 08:24:09 +08:00
|
|
|
ExprResult argE = checkUnknownAnyArg(SelLoc, argExpr, paramType);
|
2013-03-04 15:34:02 +08:00
|
|
|
if (argE.isInvalid()) {
|
2012-11-14 08:49:39 +08:00
|
|
|
IsError = true;
|
2013-03-04 15:34:02 +08:00
|
|
|
} else {
|
2014-05-29 18:55:11 +08:00
|
|
|
Args[i] = argE.get();
|
2012-11-14 08:49:39 +08:00
|
|
|
|
2013-03-04 15:34:02 +08:00
|
|
|
// Update the parameter type in-place.
|
|
|
|
param->setType(paramType);
|
|
|
|
}
|
|
|
|
continue;
|
2012-11-14 08:49:39 +08:00
|
|
|
}
|
|
|
|
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
QualType origParamType = param->getType();
|
|
|
|
QualType paramType = param->getType();
|
|
|
|
if (typeArgs)
|
|
|
|
paramType = paramType.substObjCTypeArgs(
|
|
|
|
Context,
|
|
|
|
*typeArgs,
|
|
|
|
ObjCSubstitutionContext::Parameter);
|
|
|
|
|
2010-04-22 07:24:10 +08:00
|
|
|
if (RequireCompleteType(argExpr->getSourceRange().getBegin(),
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
paramType,
|
2012-05-05 00:32:21 +08:00
|
|
|
diag::err_call_incomplete_argument, argExpr))
|
2010-04-22 07:24:10 +08:00
|
|
|
return true;
|
|
|
|
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
InitializedEntity Entity
|
|
|
|
= InitializedEntity::InitializeParameter(Context, param, paramType);
|
2014-08-01 01:39:50 +08:00
|
|
|
ExprResult ArgE = PerformCopyInitialization(Entity, SourceLocation(), argExpr);
|
2010-04-22 07:24:10 +08:00
|
|
|
if (ArgE.isInvalid())
|
|
|
|
IsError = true;
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
else {
|
2014-05-29 18:55:11 +08:00
|
|
|
Args[i] = ArgE.getAs<Expr>();
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
|
|
|
|
// If we are type-erasing a block to a block-compatible
|
|
|
|
// Objective-C pointer type, we may need to extend the lifetime
|
|
|
|
// of the block object.
|
|
|
|
if (typeArgs && Args[i]->isRValue() && paramType->isBlockPointerType() &&
|
2015-10-02 09:05:29 +08:00
|
|
|
Args[i]->getType()->isBlockPointerType() &&
|
|
|
|
origParamType->isObjCObjectPointerType()) {
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
ExprResult arg = Args[i];
|
|
|
|
maybeExtendBlockObject(arg);
|
|
|
|
Args[i] = arg.get();
|
|
|
|
}
|
|
|
|
}
|
2008-01-05 06:32:30 +08:00
|
|
|
}
|
2008-09-11 08:50:25 +08:00
|
|
|
|
|
|
|
// Promote additional arguments to variadic methods.
|
|
|
|
if (Method->isVariadic()) {
|
2013-05-10 08:27:15 +08:00
|
|
|
for (unsigned i = NumNamedArgs, e = Args.size(); i < e; ++i) {
|
2010-04-23 00:44:27 +08:00
|
|
|
if (Args[i]->isTypeDependent())
|
|
|
|
continue;
|
|
|
|
|
2012-05-13 01:32:52 +08:00
|
|
|
ExprResult Arg = DefaultVariadicArgumentPromotion(Args[i], VariadicMethod,
|
2014-05-26 14:22:03 +08:00
|
|
|
nullptr);
|
2011-04-09 02:41:53 +08:00
|
|
|
IsError |= Arg.isInvalid();
|
2014-05-29 18:55:11 +08:00
|
|
|
Args[i] = Arg.get();
|
2010-04-23 00:44:27 +08:00
|
|
|
}
|
2008-09-11 08:50:25 +08:00
|
|
|
} else {
|
|
|
|
// Check for extra arguments to non-variadic methods.
|
2013-05-10 08:27:15 +08:00
|
|
|
if (Args.size() != NumNamedArgs) {
|
2018-08-10 05:08:08 +08:00
|
|
|
Diag(Args[NumNamedArgs]->getBeginLoc(),
|
2008-11-19 13:08:23 +08:00
|
|
|
diag::err_typecheck_call_too_many_args)
|
2018-08-10 05:08:08 +08:00
|
|
|
<< 2 /*method*/ << NumNamedArgs << static_cast<unsigned>(Args.size())
|
|
|
|
<< Method->getSourceRange()
|
|
|
|
<< SourceRange(Args[NumNamedArgs]->getBeginLoc(),
|
2018-08-10 05:09:38 +08:00
|
|
|
Args.back()->getEndLoc());
|
2008-09-11 08:50:25 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-10 08:27:15 +08:00
|
|
|
DiagnoseSentinelCalls(Method, SelLoc, Args);
|
2012-01-18 04:03:31 +08:00
|
|
|
|
|
|
|
// Do additional checkings on method.
|
2013-05-10 08:27:15 +08:00
|
|
|
IsError |= CheckObjCMethodCall(
|
2014-08-31 00:55:39 +08:00
|
|
|
Method, SelLoc, makeArrayRef(Args.data(), Args.size()));
|
2012-01-18 04:03:31 +08:00
|
|
|
|
2009-04-12 16:11:20 +08:00
|
|
|
return IsError;
|
2008-01-05 06:32:30 +08:00
|
|
|
}
|
|
|
|
|
2014-01-04 02:32:18 +08:00
|
|
|
bool Sema::isSelfExpr(Expr *RExpr) {
|
2011-03-28 03:53:47 +08:00
|
|
|
// 'self' is objc 'self' in an objc method only.
|
2014-01-04 02:32:18 +08:00
|
|
|
ObjCMethodDecl *Method =
|
|
|
|
dyn_cast_or_null<ObjCMethodDecl>(CurContext->getNonClosureAncestor());
|
|
|
|
return isSelfExpr(RExpr, Method);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Sema::isSelfExpr(Expr *receiver, const ObjCMethodDecl *method) {
|
2011-11-06 17:01:30 +08:00
|
|
|
if (!method) return false;
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
receiver = receiver->IgnoreParenLValueCasts();
|
|
|
|
if (DeclRefExpr *DRE = dyn_cast<DeclRefExpr>(receiver))
|
2011-11-06 17:01:30 +08:00
|
|
|
if (DRE->getDecl() == method->getSelfDecl())
|
2011-09-28 00:10:05 +08:00
|
|
|
return true;
|
|
|
|
return false;
|
2009-03-04 23:11:40 +08:00
|
|
|
}
|
|
|
|
|
2011-10-26 01:37:35 +08:00
|
|
|
/// LookupMethodInType - Look up a method in an ObjCObjectType.
|
|
|
|
ObjCMethodDecl *Sema::LookupMethodInObjectType(Selector sel, QualType type,
|
|
|
|
bool isInstance) {
|
|
|
|
const ObjCObjectType *objType = type->castAs<ObjCObjectType>();
|
|
|
|
if (ObjCInterfaceDecl *iface = objType->getInterface()) {
|
|
|
|
// Look it up in the main interface (and categories, etc.)
|
|
|
|
if (ObjCMethodDecl *method = iface->lookupMethod(sel, isInstance))
|
|
|
|
return method;
|
|
|
|
|
|
|
|
// Okay, look for "private" methods declared in any
|
|
|
|
// @implementations we've seen.
|
2012-07-28 03:07:44 +08:00
|
|
|
if (ObjCMethodDecl *method = iface->lookupPrivateMethod(sel, isInstance))
|
|
|
|
return method;
|
2011-10-26 01:37:35 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check qualifiers.
|
2014-03-17 23:55:30 +08:00
|
|
|
for (const auto *I : objType->quals())
|
|
|
|
if (ObjCMethodDecl *method = I->lookupMethod(sel, isInstance))
|
2011-10-26 01:37:35 +08:00
|
|
|
return method;
|
|
|
|
|
2014-05-26 14:22:03 +08:00
|
|
|
return nullptr;
|
2011-10-26 01:37:35 +08:00
|
|
|
}
|
|
|
|
|
2018-07-31 03:24:48 +08:00
|
|
|
/// LookupMethodInQualifiedType - Lookups up a method in protocol qualifier
|
2011-03-10 04:18:06 +08:00
|
|
|
/// list of a qualified objective pointer type.
|
|
|
|
ObjCMethodDecl *Sema::LookupMethodInQualifiedType(Selector Sel,
|
|
|
|
const ObjCObjectPointerType *OPT,
|
|
|
|
bool Instance)
|
|
|
|
{
|
2014-05-26 14:22:03 +08:00
|
|
|
ObjCMethodDecl *MD = nullptr;
|
2014-03-18 00:14:00 +08:00
|
|
|
for (const auto *PROTO : OPT->quals()) {
|
2011-03-10 04:18:06 +08:00
|
|
|
if ((MD = PROTO->lookupMethod(Sel, Instance))) {
|
|
|
|
return MD;
|
|
|
|
}
|
|
|
|
}
|
2014-05-26 14:22:03 +08:00
|
|
|
return nullptr;
|
2011-03-10 04:18:06 +08:00
|
|
|
}
|
|
|
|
|
2010-04-11 15:45:24 +08:00
|
|
|
/// HandleExprPropertyRefExpr - Handle foo.bar where foo is a pointer to an
|
|
|
|
/// objective C interface. This is a property reference expression.
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Sema::
|
2010-04-11 15:45:24 +08:00
|
|
|
HandleExprPropertyRefExpr(const ObjCObjectPointerType *OPT,
|
2011-06-28 08:00:52 +08:00
|
|
|
Expr *BaseExpr, SourceLocation OpLoc,
|
|
|
|
DeclarationName MemberName,
|
2010-10-15 00:04:05 +08:00
|
|
|
SourceLocation MemberLoc,
|
|
|
|
SourceLocation SuperLoc, QualType SuperType,
|
|
|
|
bool Super) {
|
2010-04-11 15:45:24 +08:00
|
|
|
const ObjCInterfaceType *IFaceT = OPT->getInterfaceType();
|
|
|
|
ObjCInterfaceDecl *IFace = IFaceT->getDecl();
|
2012-05-20 00:03:58 +08:00
|
|
|
|
2012-05-20 00:34:46 +08:00
|
|
|
if (!MemberName.isIdentifier()) {
|
2011-04-21 02:19:55 +08:00
|
|
|
Diag(MemberLoc, diag::err_invalid_property_name)
|
|
|
|
<< MemberName << QualType(OPT, 0);
|
|
|
|
return ExprError();
|
|
|
|
}
|
2012-05-20 00:34:46 +08:00
|
|
|
|
|
|
|
IdentifierInfo *Member = MemberName.getAsIdentifierInfo();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-11-15 06:10:01 +08:00
|
|
|
SourceRange BaseRange = Super? SourceRange(SuperLoc)
|
|
|
|
: BaseExpr->getSourceRange();
|
2018-07-31 03:24:48 +08:00
|
|
|
if (RequireCompleteType(MemberLoc, OPT->getPointeeType(),
|
2012-05-05 00:32:21 +08:00
|
|
|
diag::err_property_not_found_forward_class,
|
|
|
|
MemberName, BaseRange))
|
2010-12-16 08:56:28 +08:00
|
|
|
return ExprError();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2016-01-29 02:49:28 +08:00
|
|
|
if (ObjCPropertyDecl *PD = IFace->FindPropertyDeclaration(
|
|
|
|
Member, ObjCPropertyQueryKind::OBJC_PR_query_instance)) {
|
2010-04-11 15:45:24 +08:00
|
|
|
// Check whether we can reference this property.
|
|
|
|
if (DiagnoseUseOfDecl(PD, MemberLoc))
|
|
|
|
return ExprError();
|
2010-10-15 00:04:05 +08:00
|
|
|
if (Super)
|
2014-05-29 22:05:12 +08:00
|
|
|
return new (Context)
|
|
|
|
ObjCPropertyRefExpr(PD, Context.PseudoObjectTy, VK_LValue,
|
|
|
|
OK_ObjCProperty, MemberLoc, SuperLoc, SuperType);
|
2010-10-15 00:04:05 +08:00
|
|
|
else
|
2014-05-29 22:05:12 +08:00
|
|
|
return new (Context)
|
|
|
|
ObjCPropertyRefExpr(PD, Context.PseudoObjectTy, VK_LValue,
|
|
|
|
OK_ObjCProperty, MemberLoc, BaseExpr);
|
2010-04-11 15:45:24 +08:00
|
|
|
}
|
|
|
|
// Check protocols on qualified interfaces.
|
2014-03-18 00:14:00 +08:00
|
|
|
for (const auto *I : OPT->quals())
|
2016-01-29 02:49:28 +08:00
|
|
|
if (ObjCPropertyDecl *PD = I->FindPropertyDeclaration(
|
|
|
|
Member, ObjCPropertyQueryKind::OBJC_PR_query_instance)) {
|
2010-04-11 15:45:24 +08:00
|
|
|
// Check whether we can reference this property.
|
|
|
|
if (DiagnoseUseOfDecl(PD, MemberLoc))
|
|
|
|
return ExprError();
|
2012-04-20 05:44:57 +08:00
|
|
|
|
2010-10-15 00:04:05 +08:00
|
|
|
if (Super)
|
2014-05-29 22:05:12 +08:00
|
|
|
return new (Context) ObjCPropertyRefExpr(
|
|
|
|
PD, Context.PseudoObjectTy, VK_LValue, OK_ObjCProperty, MemberLoc,
|
|
|
|
SuperLoc, SuperType);
|
2010-10-15 00:04:05 +08:00
|
|
|
else
|
2014-05-29 22:05:12 +08:00
|
|
|
return new (Context)
|
|
|
|
ObjCPropertyRefExpr(PD, Context.PseudoObjectTy, VK_LValue,
|
|
|
|
OK_ObjCProperty, MemberLoc, BaseExpr);
|
2010-04-11 15:45:24 +08:00
|
|
|
}
|
|
|
|
// If that failed, look for an "implicit" property by seeing if the nullary
|
|
|
|
// selector is implemented.
|
|
|
|
|
|
|
|
// FIXME: The logic for looking up nullary and unary selectors should be
|
|
|
|
// shared with the code in ActOnInstanceMessage.
|
|
|
|
|
|
|
|
Selector Sel = PP.getSelectorTable().getNullarySelector(Member);
|
|
|
|
ObjCMethodDecl *Getter = IFace->lookupInstanceMethod(Sel);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2016-06-29 07:01:49 +08:00
|
|
|
// May be found in property's qualified list.
|
2011-03-10 06:17:12 +08:00
|
|
|
if (!Getter)
|
|
|
|
Getter = LookupMethodInQualifiedType(Sel, OPT, true);
|
2010-04-11 15:45:24 +08:00
|
|
|
|
|
|
|
// If this reference is in an @implementation, check for 'private' methods.
|
|
|
|
if (!Getter)
|
2010-12-04 07:37:08 +08:00
|
|
|
Getter = IFace->lookupPrivateMethod(Sel);
|
2010-04-11 15:45:24 +08:00
|
|
|
|
|
|
|
if (Getter) {
|
|
|
|
// Check if we can reference this property.
|
|
|
|
if (DiagnoseUseOfDecl(Getter, MemberLoc))
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
// If we found a getter then this may be a valid dot-reference, we
|
|
|
|
// will look for the matching setter, in case it is needed.
|
|
|
|
Selector SetterSel =
|
2013-06-08 06:29:12 +08:00
|
|
|
SelectorTable::constructSetterSelector(PP.getIdentifierTable(),
|
|
|
|
PP.getSelectorTable(), Member);
|
2010-04-11 15:45:24 +08:00
|
|
|
ObjCMethodDecl *Setter = IFace->lookupInstanceMethod(SetterSel);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2016-06-29 07:01:49 +08:00
|
|
|
// May be found in property's qualified list.
|
2011-03-10 06:17:12 +08:00
|
|
|
if (!Setter)
|
|
|
|
Setter = LookupMethodInQualifiedType(SetterSel, OPT, true);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2010-04-11 15:45:24 +08:00
|
|
|
if (!Setter) {
|
|
|
|
// If this reference is in an @implementation, also check for 'private'
|
|
|
|
// methods.
|
2010-12-04 07:37:08 +08:00
|
|
|
Setter = IFace->lookupPrivateMethod(SetterSel);
|
2010-04-11 15:45:24 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2010-04-11 15:45:24 +08:00
|
|
|
if (Setter && DiagnoseUseOfDecl(Setter, MemberLoc))
|
|
|
|
return ExprError();
|
|
|
|
|
2014-08-09 06:33:24 +08:00
|
|
|
// Special warning if member name used in a property-dot for a setter accessor
|
|
|
|
// does not use a property with same name; e.g. obj.X = ... for a property with
|
|
|
|
// name 'x'.
|
2016-01-29 02:49:28 +08:00
|
|
|
if (Setter && Setter->isImplicit() && Setter->isPropertyAccessor() &&
|
|
|
|
!IFace->FindPropertyDeclaration(
|
|
|
|
Member, ObjCPropertyQueryKind::OBJC_PR_query_instance)) {
|
2014-08-16 01:39:00 +08:00
|
|
|
if (const ObjCPropertyDecl *PDecl = Setter->findPropertyDecl()) {
|
|
|
|
// Do not warn if user is using property-dot syntax to make call to
|
|
|
|
// user named setter.
|
|
|
|
if (!(PDecl->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_setter))
|
2014-08-09 06:33:24 +08:00
|
|
|
Diag(MemberLoc,
|
|
|
|
diag::warn_property_access_suggest)
|
|
|
|
<< MemberName << QualType(OPT, 0) << PDecl->getName()
|
|
|
|
<< FixItHint::CreateReplacement(MemberLoc, PDecl->getName());
|
2014-08-16 01:39:00 +08:00
|
|
|
}
|
2014-08-09 06:33:24 +08:00
|
|
|
}
|
|
|
|
|
2010-12-23 03:46:35 +08:00
|
|
|
if (Getter || Setter) {
|
2010-10-15 00:04:05 +08:00
|
|
|
if (Super)
|
2014-05-29 22:05:12 +08:00
|
|
|
return new (Context)
|
|
|
|
ObjCPropertyRefExpr(Getter, Setter, Context.PseudoObjectTy, VK_LValue,
|
|
|
|
OK_ObjCProperty, MemberLoc, SuperLoc, SuperType);
|
2010-10-15 00:04:05 +08:00
|
|
|
else
|
2014-05-29 22:05:12 +08:00
|
|
|
return new (Context)
|
|
|
|
ObjCPropertyRefExpr(Getter, Setter, Context.PseudoObjectTy, VK_LValue,
|
|
|
|
OK_ObjCProperty, MemberLoc, BaseExpr);
|
2010-10-15 00:04:05 +08:00
|
|
|
|
2010-04-11 15:45:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Attempt to correct for typos in property names.
|
2019-03-26 01:08:51 +08:00
|
|
|
DeclFilterCCC<ObjCPropertyDecl> CCC{};
|
|
|
|
if (TypoCorrection Corrected = CorrectTypo(
|
|
|
|
DeclarationNameInfo(MemberName, MemberLoc), LookupOrdinaryName,
|
|
|
|
nullptr, nullptr, CCC, CTK_ErrorRecovery, IFace, false, OPT)) {
|
2011-06-29 00:20:02 +08:00
|
|
|
DeclarationName TypoResult = Corrected.getCorrection();
|
2016-06-29 07:01:49 +08:00
|
|
|
if (TypoResult.isIdentifier() &&
|
|
|
|
TypoResult.getAsIdentifierInfo() == Member) {
|
|
|
|
// There is no need to try the correction if it is the same.
|
|
|
|
NamedDecl *ChosenDecl =
|
|
|
|
Corrected.isKeyword() ? nullptr : Corrected.getFoundDecl();
|
|
|
|
if (ChosenDecl && isa<ObjCPropertyDecl>(ChosenDecl))
|
|
|
|
if (cast<ObjCPropertyDecl>(ChosenDecl)->isClassProperty()) {
|
|
|
|
// This is a class property, we should not use the instance to
|
|
|
|
// access it.
|
|
|
|
Diag(MemberLoc, diag::err_class_property_found) << MemberName
|
|
|
|
<< OPT->getInterfaceDecl()->getName()
|
|
|
|
<< FixItHint::CreateReplacement(BaseExpr->getSourceRange(),
|
|
|
|
OPT->getInterfaceDecl()->getName());
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
diagnoseTypo(Corrected, PDiag(diag::err_property_not_found_suggest)
|
|
|
|
<< MemberName << QualType(OPT, 0));
|
|
|
|
return HandleExprPropertyRefExpr(OPT, BaseExpr, OpLoc,
|
|
|
|
TypoResult, MemberLoc,
|
|
|
|
SuperLoc, SuperType, Super);
|
|
|
|
}
|
2010-04-11 15:45:24 +08:00
|
|
|
}
|
2011-02-17 09:26:14 +08:00
|
|
|
ObjCInterfaceDecl *ClassDeclared;
|
2018-07-31 03:24:48 +08:00
|
|
|
if (ObjCIvarDecl *Ivar =
|
2011-02-17 09:26:14 +08:00
|
|
|
IFace->lookupInstanceVariable(Member, ClassDeclared)) {
|
|
|
|
QualType T = Ivar->getType();
|
2018-07-31 03:24:48 +08:00
|
|
|
if (const ObjCObjectPointerType * OBJPT =
|
2011-02-17 09:26:14 +08:00
|
|
|
T->getAsObjCInterfacePointerType()) {
|
2018-07-31 03:24:48 +08:00
|
|
|
if (RequireCompleteType(MemberLoc, OBJPT->getPointeeType(),
|
2012-05-05 00:32:21 +08:00
|
|
|
diag::err_property_not_as_forward_class,
|
|
|
|
MemberName, BaseExpr))
|
2011-11-15 06:10:01 +08:00
|
|
|
return ExprError();
|
2011-02-17 09:26:14 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
Diag(MemberLoc,
|
2011-06-28 08:00:52 +08:00
|
|
|
diag::err_ivar_access_using_property_syntax_suggest)
|
|
|
|
<< MemberName << QualType(OPT, 0) << Ivar->getDeclName()
|
|
|
|
<< FixItHint::CreateReplacement(OpLoc, "->");
|
|
|
|
return ExprError();
|
2011-02-17 09:26:14 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2010-04-11 15:45:24 +08:00
|
|
|
Diag(MemberLoc, diag::err_property_not_found)
|
|
|
|
<< MemberName << QualType(OPT, 0);
|
2010-12-23 03:46:35 +08:00
|
|
|
if (Setter)
|
2010-04-11 15:45:24 +08:00
|
|
|
Diag(Setter->getLocation(), diag::note_getter_unavailable)
|
2010-12-23 03:46:35 +08:00
|
|
|
<< MemberName << BaseExpr->getSourceRange();
|
2010-04-11 15:45:24 +08:00
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Sema::
|
2010-04-11 16:28:14 +08:00
|
|
|
ActOnClassPropertyRefExpr(IdentifierInfo &receiverName,
|
|
|
|
IdentifierInfo &propertyName,
|
|
|
|
SourceLocation receiverNameLoc,
|
|
|
|
SourceLocation propertyNameLoc) {
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-01-04 02:01:57 +08:00
|
|
|
IdentifierInfo *receiverNamePtr = &receiverName;
|
2010-04-16 06:33:43 +08:00
|
|
|
ObjCInterfaceDecl *IFace = getObjCInterfaceDecl(receiverNamePtr,
|
|
|
|
receiverNameLoc);
|
2011-06-11 09:09:30 +08:00
|
|
|
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
QualType SuperType;
|
2014-05-26 14:22:03 +08:00
|
|
|
if (!IFace) {
|
2010-04-11 16:28:14 +08:00
|
|
|
// If the "receiver" is 'super' in a method, handle it as an expression-like
|
|
|
|
// property reference.
|
2011-02-03 17:00:02 +08:00
|
|
|
if (receiverNamePtr->isStr("super")) {
|
2012-02-04 06:47:37 +08:00
|
|
|
if (ObjCMethodDecl *CurMethod = tryCaptureObjCSelf(receiverNameLoc)) {
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
if (auto classDecl = CurMethod->getClassInterface()) {
|
|
|
|
SuperType = QualType(classDecl->getSuperClassType(), 0);
|
2015-01-21 00:53:34 +08:00
|
|
|
if (CurMethod->isInstanceMethod()) {
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
if (SuperType.isNull()) {
|
2015-01-21 00:53:34 +08:00
|
|
|
// The current class does not have a superclass.
|
2016-12-03 06:38:31 +08:00
|
|
|
Diag(receiverNameLoc, diag::err_root_class_cannot_use_super)
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
<< CurMethod->getClassInterface()->getIdentifier();
|
2015-01-21 00:53:34 +08:00
|
|
|
return ExprError();
|
|
|
|
}
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
QualType T = Context.getObjCObjectPointerType(SuperType);
|
2015-01-21 00:53:34 +08:00
|
|
|
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
return HandleExprPropertyRefExpr(T->castAs<ObjCObjectPointerType>(),
|
2015-01-21 00:53:34 +08:00
|
|
|
/*BaseExpr*/nullptr,
|
|
|
|
SourceLocation()/*OpLoc*/,
|
|
|
|
&propertyName,
|
|
|
|
propertyNameLoc,
|
|
|
|
receiverNameLoc, T, true);
|
2013-03-12 06:26:33 +08:00
|
|
|
}
|
2010-04-11 16:28:14 +08:00
|
|
|
|
2015-01-21 00:53:34 +08:00
|
|
|
// Otherwise, if this is a class method, try dispatching to our
|
|
|
|
// superclass.
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
IFace = CurMethod->getClassInterface()->getSuperClass();
|
2015-01-21 00:53:34 +08:00
|
|
|
}
|
2010-04-11 16:28:14 +08:00
|
|
|
}
|
2011-02-03 17:00:02 +08:00
|
|
|
}
|
2014-05-26 14:22:03 +08:00
|
|
|
|
|
|
|
if (!IFace) {
|
2013-12-24 17:48:30 +08:00
|
|
|
Diag(receiverNameLoc, diag::err_expected_either) << tok::identifier
|
|
|
|
<< tok::l_paren;
|
2010-04-11 16:28:14 +08:00
|
|
|
return ExprError();
|
|
|
|
}
|
2010-03-23 05:02:34 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2017-02-21 07:45:49 +08:00
|
|
|
Selector GetterSel;
|
|
|
|
Selector SetterSel;
|
|
|
|
if (auto PD = IFace->FindPropertyDeclaration(
|
|
|
|
&propertyName, ObjCPropertyQueryKind::OBJC_PR_query_class)) {
|
|
|
|
GetterSel = PD->getGetterName();
|
|
|
|
SetterSel = PD->getSetterName();
|
|
|
|
} else {
|
|
|
|
GetterSel = PP.getSelectorTable().getNullarySelector(&propertyName);
|
|
|
|
SetterSel = SelectorTable::constructSetterSelector(
|
|
|
|
PP.getIdentifierTable(), PP.getSelectorTable(), &propertyName);
|
|
|
|
}
|
|
|
|
|
2010-04-11 16:28:14 +08:00
|
|
|
// Search for a declared property first.
|
2017-02-21 07:45:49 +08:00
|
|
|
ObjCMethodDecl *Getter = IFace->lookupClassMethod(GetterSel);
|
2009-03-10 05:12:44 +08:00
|
|
|
|
|
|
|
// If this reference is in an @implementation, check for 'private' methods.
|
|
|
|
if (!Getter)
|
2017-02-21 07:45:49 +08:00
|
|
|
Getter = IFace->lookupPrivateClassMethod(GetterSel);
|
2009-03-10 05:12:44 +08:00
|
|
|
|
|
|
|
if (Getter) {
|
|
|
|
// FIXME: refactor/share with ActOnMemberReference().
|
|
|
|
// Check if we can reference this property.
|
|
|
|
if (DiagnoseUseOfDecl(Getter, propertyNameLoc))
|
|
|
|
return ExprError();
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2009-03-10 05:12:44 +08:00
|
|
|
// Look for the matching setter, in case it is needed.
|
2009-06-30 10:36:12 +08:00
|
|
|
ObjCMethodDecl *Setter = IFace->lookupClassMethod(SetterSel);
|
2009-03-10 05:12:44 +08:00
|
|
|
if (!Setter) {
|
|
|
|
// If this reference is in an @implementation, also check for 'private'
|
|
|
|
// methods.
|
2014-04-22 04:22:17 +08:00
|
|
|
Setter = IFace->lookupPrivateClassMethod(SetterSel);
|
2009-03-10 05:12:44 +08:00
|
|
|
}
|
|
|
|
// Look through local category implementations associated with the class.
|
2009-07-21 08:06:20 +08:00
|
|
|
if (!Setter)
|
|
|
|
Setter = IFace->getCategoryClassMethod(SetterSel);
|
2009-03-10 05:12:44 +08:00
|
|
|
|
|
|
|
if (Setter && DiagnoseUseOfDecl(Setter, propertyNameLoc))
|
|
|
|
return ExprError();
|
|
|
|
|
|
|
|
if (Getter || Setter) {
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
if (!SuperType.isNull())
|
2014-05-29 22:05:12 +08:00
|
|
|
return new (Context)
|
|
|
|
ObjCPropertyRefExpr(Getter, Setter, Context.PseudoObjectTy, VK_LValue,
|
|
|
|
OK_ObjCProperty, propertyNameLoc, receiverNameLoc,
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
SuperType);
|
2014-05-29 22:05:12 +08:00
|
|
|
|
|
|
|
return new (Context) ObjCPropertyRefExpr(
|
|
|
|
Getter, Setter, Context.PseudoObjectTy, VK_LValue, OK_ObjCProperty,
|
|
|
|
propertyNameLoc, receiverNameLoc, IFace);
|
2009-03-10 05:12:44 +08:00
|
|
|
}
|
|
|
|
return ExprError(Diag(propertyNameLoc, diag::err_property_not_found)
|
|
|
|
<< &propertyName << Context.getObjCInterfaceType(IFace));
|
|
|
|
}
|
|
|
|
|
2012-01-13 09:32:50 +08:00
|
|
|
namespace {
|
|
|
|
|
2019-03-26 01:08:51 +08:00
|
|
|
class ObjCInterfaceOrSuperCCC final : public CorrectionCandidateCallback {
|
2012-01-13 09:32:50 +08:00
|
|
|
public:
|
|
|
|
ObjCInterfaceOrSuperCCC(ObjCMethodDecl *Method) {
|
|
|
|
// Determine whether "super" is acceptable in the current context.
|
|
|
|
if (Method && Method->getClassInterface())
|
|
|
|
WantObjCSuper = Method->getClassInterface()->getSuperClass();
|
|
|
|
}
|
|
|
|
|
2014-03-12 12:55:44 +08:00
|
|
|
bool ValidateCandidate(const TypoCorrection &candidate) override {
|
2012-01-13 09:32:50 +08:00
|
|
|
return candidate.getCorrectionDeclAs<ObjCInterfaceDecl>() ||
|
|
|
|
candidate.isKeyword("super");
|
|
|
|
}
|
2019-03-26 01:08:51 +08:00
|
|
|
|
|
|
|
std::unique_ptr<CorrectionCandidateCallback> clone() override {
|
|
|
|
return llvm::make_unique<ObjCInterfaceOrSuperCCC>(*this);
|
|
|
|
}
|
2012-01-13 09:32:50 +08:00
|
|
|
};
|
|
|
|
|
2016-02-13 06:53:10 +08:00
|
|
|
} // end anonymous namespace
|
2012-01-13 09:32:50 +08:00
|
|
|
|
2010-04-14 10:46:37 +08:00
|
|
|
Sema::ObjCMessageKind Sema::getObjCMessageKind(Scope *S,
|
2010-04-22 04:38:13 +08:00
|
|
|
IdentifierInfo *Name,
|
2010-04-14 10:46:37 +08:00
|
|
|
SourceLocation NameLoc,
|
|
|
|
bool IsSuper,
|
2010-04-22 04:38:13 +08:00
|
|
|
bool HasTrailingDot,
|
2010-08-24 13:47:05 +08:00
|
|
|
ParsedType &ReceiverType) {
|
2016-01-16 07:43:34 +08:00
|
|
|
ReceiverType = nullptr;
|
2010-04-22 04:38:13 +08:00
|
|
|
|
2010-04-14 10:46:37 +08:00
|
|
|
// If the identifier is "super" and there is no trailing dot, we're
|
2010-10-15 06:11:03 +08:00
|
|
|
// messaging super. If the identifier is "super" and there is a
|
|
|
|
// trailing dot, it's an instance message.
|
|
|
|
if (IsSuper && S->isInObjcMethodScope())
|
|
|
|
return HasTrailingDot? ObjCInstanceMessage : ObjCSuperMessage;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2010-04-14 10:46:37 +08:00
|
|
|
LookupResult Result(*this, Name, NameLoc, LookupOrdinaryName);
|
|
|
|
LookupName(Result, S);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2010-04-14 10:46:37 +08:00
|
|
|
switch (Result.getResultKind()) {
|
|
|
|
case LookupResult::NotFound:
|
2010-04-20 04:09:36 +08:00
|
|
|
// Normal name lookup didn't find anything. If we're in an
|
|
|
|
// Objective-C method, look for ivars. If we find one, we're done!
|
2010-10-15 06:11:03 +08:00
|
|
|
// FIXME: This is a hack. Ivar lookup should be part of normal
|
|
|
|
// lookup.
|
2010-04-20 04:09:36 +08:00
|
|
|
if (ObjCMethodDecl *Method = getCurMethodDecl()) {
|
2011-11-09 08:22:48 +08:00
|
|
|
if (!Method->getClassInterface()) {
|
|
|
|
// Fall back: let the parser try to parse it as an instance message.
|
|
|
|
return ObjCInstanceMessage;
|
|
|
|
}
|
|
|
|
|
2010-04-20 04:09:36 +08:00
|
|
|
ObjCInterfaceDecl *ClassDeclared;
|
2018-07-31 03:24:48 +08:00
|
|
|
if (Method->getClassInterface()->lookupInstanceVariable(Name,
|
2010-04-20 04:09:36 +08:00
|
|
|
ClassDeclared))
|
|
|
|
return ObjCInstanceMessage;
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2010-04-14 10:46:37 +08:00
|
|
|
// Break out; we'll perform typo correction below.
|
|
|
|
break;
|
|
|
|
|
|
|
|
case LookupResult::NotFoundInCurrentInstantiation:
|
|
|
|
case LookupResult::FoundOverloaded:
|
|
|
|
case LookupResult::FoundUnresolvedValue:
|
|
|
|
case LookupResult::Ambiguous:
|
|
|
|
Result.suppressDiagnostics();
|
|
|
|
return ObjCInstanceMessage;
|
|
|
|
|
|
|
|
case LookupResult::Found: {
|
2011-02-08 08:23:07 +08:00
|
|
|
// If the identifier is a class or not, and there is a trailing dot,
|
|
|
|
// it's an instance message.
|
|
|
|
if (HasTrailingDot)
|
|
|
|
return ObjCInstanceMessage;
|
2010-04-14 10:46:37 +08:00
|
|
|
// We found something. If it's a type, then we have a class
|
|
|
|
// message. Otherwise, it's an instance message.
|
|
|
|
NamedDecl *ND = Result.getFoundDecl();
|
2010-04-22 04:38:13 +08:00
|
|
|
QualType T;
|
|
|
|
if (ObjCInterfaceDecl *Class = dyn_cast<ObjCInterfaceDecl>(ND))
|
|
|
|
T = Context.getObjCInterfaceType(Class);
|
2013-04-05 02:45:52 +08:00
|
|
|
else if (TypeDecl *Type = dyn_cast<TypeDecl>(ND)) {
|
2010-04-22 04:38:13 +08:00
|
|
|
T = Context.getTypeDeclType(Type);
|
2013-04-05 02:45:52 +08:00
|
|
|
DiagnoseUseOfDecl(Type, NameLoc);
|
|
|
|
}
|
|
|
|
else
|
2010-04-22 04:38:13 +08:00
|
|
|
return ObjCInstanceMessage;
|
|
|
|
|
|
|
|
// We have a class message, and T is the type we're
|
|
|
|
// messaging. Build source-location information for it.
|
|
|
|
TypeSourceInfo *TSInfo = Context.getTrivialTypeSourceInfo(T, NameLoc);
|
2010-08-24 13:47:05 +08:00
|
|
|
ReceiverType = CreateParsedType(T, TSInfo);
|
2010-04-22 04:38:13 +08:00
|
|
|
return ObjCClassMessage;
|
2010-04-14 10:46:37 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-26 01:08:51 +08:00
|
|
|
ObjCInterfaceOrSuperCCC CCC(getCurMethodDecl());
|
2014-10-28 02:07:29 +08:00
|
|
|
if (TypoCorrection Corrected = CorrectTypo(
|
2019-03-26 01:08:51 +08:00
|
|
|
Result.getLookupNameInfo(), Result.getLookupKind(), S, nullptr, CCC,
|
2014-10-28 02:07:29 +08:00
|
|
|
CTK_ErrorRecovery, nullptr, false, nullptr, false)) {
|
2012-01-13 09:32:50 +08:00
|
|
|
if (Corrected.isKeyword()) {
|
|
|
|
// If we've found the keyword "super" (the only keyword that would be
|
|
|
|
// returned by CorrectTypo), this is a send to super.
|
2013-08-17 08:46:16 +08:00
|
|
|
diagnoseTypo(Corrected,
|
|
|
|
PDiag(diag::err_unknown_receiver_suggest) << Name);
|
2010-04-15 04:04:41 +08:00
|
|
|
return ObjCSuperMessage;
|
2012-01-13 09:32:50 +08:00
|
|
|
} else if (ObjCInterfaceDecl *Class =
|
2013-08-17 08:46:16 +08:00
|
|
|
Corrected.getCorrectionDeclAs<ObjCInterfaceDecl>()) {
|
2012-01-13 09:32:50 +08:00
|
|
|
// If we found a declaration, correct when it refers to an Objective-C
|
|
|
|
// class.
|
2013-08-17 08:46:16 +08:00
|
|
|
diagnoseTypo(Corrected,
|
|
|
|
PDiag(diag::err_unknown_receiver_suggest) << Name);
|
2012-01-13 09:32:50 +08:00
|
|
|
QualType T = Context.getObjCInterfaceType(Class);
|
|
|
|
TypeSourceInfo *TSInfo = Context.getTrivialTypeSourceInfo(T, NameLoc);
|
|
|
|
ReceiverType = CreateParsedType(T, TSInfo);
|
|
|
|
return ObjCClassMessage;
|
2010-04-14 10:46:37 +08:00
|
|
|
}
|
|
|
|
}
|
2013-08-17 08:46:16 +08:00
|
|
|
|
2010-04-14 10:46:37 +08:00
|
|
|
// Fall back: let the parser try to parse it as an instance message.
|
|
|
|
return ObjCInstanceMessage;
|
|
|
|
}
|
2009-03-10 05:12:44 +08:00
|
|
|
|
2018-07-31 03:24:48 +08:00
|
|
|
ExprResult Sema::ActOnSuperMessage(Scope *S,
|
2010-09-15 22:51:05 +08:00
|
|
|
SourceLocation SuperLoc,
|
|
|
|
Selector Sel,
|
|
|
|
SourceLocation LBracLoc,
|
2011-10-03 14:36:17 +08:00
|
|
|
ArrayRef<SourceLocation> SelectorLocs,
|
2010-09-15 22:51:05 +08:00
|
|
|
SourceLocation RBracLoc,
|
|
|
|
MultiExprArg Args) {
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// Determine whether we are inside a method or not.
|
2012-02-04 06:47:37 +08:00
|
|
|
ObjCMethodDecl *Method = tryCaptureObjCSelf(SuperLoc);
|
2010-04-22 04:01:04 +08:00
|
|
|
if (!Method) {
|
|
|
|
Diag(SuperLoc, diag::err_invalid_receiver_to_message_super);
|
|
|
|
return ExprError();
|
|
|
|
}
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
|
2010-04-22 04:01:04 +08:00
|
|
|
ObjCInterfaceDecl *Class = Method->getClassInterface();
|
|
|
|
if (!Class) {
|
2016-12-03 06:38:31 +08:00
|
|
|
Diag(SuperLoc, diag::err_no_super_class_message)
|
2010-04-22 04:01:04 +08:00
|
|
|
<< Method->getDeclName();
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
return ExprError();
|
2010-04-12 13:38:43 +08:00
|
|
|
}
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
QualType SuperTy(Class->getSuperClassType(), 0);
|
|
|
|
if (SuperTy.isNull()) {
|
2010-04-22 04:01:04 +08:00
|
|
|
// The current class does not have a superclass.
|
2016-12-03 06:38:31 +08:00
|
|
|
Diag(SuperLoc, diag::err_root_class_cannot_use_super)
|
2011-01-24 01:21:34 +08:00
|
|
|
<< Class->getIdentifier();
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2010-04-22 04:01:04 +08:00
|
|
|
// We are in a method whose class has a superclass, so 'super'
|
|
|
|
// is acting as a keyword.
|
2012-10-20 00:05:26 +08:00
|
|
|
if (Method->getSelector() == Sel)
|
|
|
|
getCurFunction()->ObjCShouldCallSuper = false;
|
2011-08-23 01:25:57 +08:00
|
|
|
|
2012-10-20 00:05:26 +08:00
|
|
|
if (Method->isInstanceMethod()) {
|
2010-04-22 04:01:04 +08:00
|
|
|
// Since we are in an instance method, this is an instance
|
|
|
|
// message to the superclass instance.
|
|
|
|
SuperTy = Context.getObjCObjectPointerType(SuperTy);
|
2014-05-26 14:22:03 +08:00
|
|
|
return BuildInstanceMessage(nullptr, SuperTy, SuperLoc,
|
|
|
|
Sel, /*Method=*/nullptr,
|
2012-08-24 05:35:17 +08:00
|
|
|
LBracLoc, SelectorLocs, RBracLoc, Args);
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2010-04-22 04:01:04 +08:00
|
|
|
// Since we are in a class method, this is a class message to
|
|
|
|
// the superclass.
|
2014-05-26 14:22:03 +08:00
|
|
|
return BuildClassMessage(/*ReceiverTypeInfo=*/nullptr,
|
Substitute type arguments into uses of Objective-C interface members.
When messaging a method that was defined in an Objective-C class (or
category or extension thereof) that has type parameters, substitute
the type arguments for those type parameters. Similarly, substitute
into property accesses, instance variables, and other references.
This includes general infrastructure for substituting the type
arguments associated with an ObjCObject(Pointer)Type into a type
referenced within a particular context, handling all of the
substitutions required to deal with (e.g.) inheritance involving
parameterized classes. In cases where no type arguments are available
(e.g., because we're messaging via some unspecialized type, id, etc.),
we substitute in the type bounds for the type parameters instead.
Example:
@interface NSSet<T : id<NSCopying>> : NSObject <NSCopying>
- (T)firstObject;
@end
void f(NSSet<NSString *> *stringSet, NSSet *anySet) {
[stringSet firstObject]; // produces NSString*
[anySet firstObject]; // produces id<NSCopying> (the bound)
}
When substituting for the type parameters given an unspecialized
context (i.e., no specific type arguments were given), substituting
the type bounds unconditionally produces type signatures that are too
strong compared to the pre-generics signatures. Instead, use the
following rule:
- In covariant positions, such as method return types, replace type
parameters with “id” or “Class” (the latter only when the type
parameter bound is “Class” or qualified class, e.g,
“Class<NSCopying>”)
- In other positions (e.g., parameter types), replace type
parameters with their type bounds.
- When a specialized Objective-C object or object pointer type
contains a type parameter in its type arguments (e.g.,
NSArray<T>*, but not NSArray<NSString *> *), replace the entire
object/object pointer type with its unspecialized version (e.g.,
NSArray *).
llvm-svn: 241543
2015-07-07 11:57:53 +08:00
|
|
|
SuperTy,
|
2014-05-26 14:22:03 +08:00
|
|
|
SuperLoc, Sel, /*Method=*/nullptr,
|
2012-08-24 05:35:17 +08:00
|
|
|
LBracLoc, SelectorLocs, RBracLoc, Args);
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
}
|
|
|
|
|
2012-01-12 10:34:39 +08:00
|
|
|
ExprResult Sema::BuildClassMessageImplicit(QualType ReceiverType,
|
|
|
|
bool isSuperReceiver,
|
|
|
|
SourceLocation Loc,
|
|
|
|
Selector Sel,
|
|
|
|
ObjCMethodDecl *Method,
|
|
|
|
MultiExprArg Args) {
|
2014-05-26 14:22:03 +08:00
|
|
|
TypeSourceInfo *receiverTypeInfo = nullptr;
|
2012-01-12 10:34:39 +08:00
|
|
|
if (!ReceiverType.isNull())
|
|
|
|
receiverTypeInfo = Context.getTrivialTypeSourceInfo(ReceiverType);
|
|
|
|
|
|
|
|
return BuildClassMessage(receiverTypeInfo, ReceiverType,
|
|
|
|
/*SuperLoc=*/isSuperReceiver ? Loc : SourceLocation(),
|
|
|
|
Sel, Method, Loc, Loc, Loc, Args,
|
|
|
|
/*isImplicit=*/true);
|
|
|
|
}
|
|
|
|
|
2012-03-07 04:05:56 +08:00
|
|
|
static void applyCocoaAPICheck(Sema &S, const ObjCMessageExpr *Msg,
|
|
|
|
unsigned DiagID,
|
|
|
|
bool (*refactor)(const ObjCMessageExpr *,
|
|
|
|
const NSAPI &, edit::Commit &)) {
|
|
|
|
SourceLocation MsgLoc = Msg->getExprLoc();
|
2014-06-16 07:30:39 +08:00
|
|
|
if (S.Diags.isIgnored(DiagID, MsgLoc))
|
2012-03-07 04:05:56 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
SourceManager &SM = S.SourceMgr;
|
|
|
|
edit::Commit ECommit(SM, S.LangOpts);
|
|
|
|
if (refactor(Msg,*S.NSAPIObj, ECommit)) {
|
|
|
|
DiagnosticBuilder Builder = S.Diag(MsgLoc, DiagID)
|
|
|
|
<< Msg->getSelector() << Msg->getSourceRange();
|
|
|
|
// FIXME: Don't emit diagnostic at all if fixits are non-commitable.
|
|
|
|
if (!ECommit.isCommitable())
|
|
|
|
return;
|
|
|
|
for (edit::Commit::edit_iterator
|
|
|
|
I = ECommit.edit_begin(), E = ECommit.edit_end(); I != E; ++I) {
|
|
|
|
const edit::Commit::Edit &Edit = *I;
|
|
|
|
switch (Edit.Kind) {
|
|
|
|
case edit::Commit::Act_Insert:
|
|
|
|
Builder.AddFixItHint(FixItHint::CreateInsertion(Edit.OrigLoc,
|
|
|
|
Edit.Text,
|
|
|
|
Edit.BeforePrev));
|
|
|
|
break;
|
|
|
|
case edit::Commit::Act_InsertFromRange:
|
|
|
|
Builder.AddFixItHint(
|
|
|
|
FixItHint::CreateInsertionFromRange(Edit.OrigLoc,
|
|
|
|
Edit.getInsertFromRange(SM),
|
|
|
|
Edit.BeforePrev));
|
|
|
|
break;
|
|
|
|
case edit::Commit::Act_Remove:
|
|
|
|
Builder.AddFixItHint(FixItHint::CreateRemoval(Edit.getFileRange(SM)));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void checkCocoaAPI(Sema &S, const ObjCMessageExpr *Msg) {
|
|
|
|
applyCocoaAPICheck(S, Msg, diag::warn_objc_redundant_literal_use,
|
|
|
|
edit::rewriteObjCRedundantCallWithLiteral);
|
|
|
|
}
|
|
|
|
|
2017-03-06 23:58:34 +08:00
|
|
|
static void checkFoundationAPI(Sema &S, SourceLocation Loc,
|
|
|
|
const ObjCMethodDecl *Method,
|
|
|
|
ArrayRef<Expr *> Args, QualType ReceiverType,
|
|
|
|
bool IsClassObjectCall) {
|
|
|
|
// Check if this is a performSelector method that uses a selector that returns
|
|
|
|
// a record or a vector type.
|
2017-03-23 18:46:05 +08:00
|
|
|
if (Method->getSelector().getMethodFamily() != OMF_performSelector ||
|
|
|
|
Args.empty())
|
2017-03-06 23:58:34 +08:00
|
|
|
return;
|
|
|
|
const auto *SE = dyn_cast<ObjCSelectorExpr>(Args[0]->IgnoreParens());
|
|
|
|
if (!SE)
|
|
|
|
return;
|
|
|
|
ObjCMethodDecl *ImpliedMethod;
|
|
|
|
if (!IsClassObjectCall) {
|
|
|
|
const auto *OPT = ReceiverType->getAs<ObjCObjectPointerType>();
|
|
|
|
if (!OPT || !OPT->getInterfaceDecl())
|
|
|
|
return;
|
|
|
|
ImpliedMethod =
|
|
|
|
OPT->getInterfaceDecl()->lookupInstanceMethod(SE->getSelector());
|
|
|
|
if (!ImpliedMethod)
|
|
|
|
ImpliedMethod =
|
|
|
|
OPT->getInterfaceDecl()->lookupPrivateMethod(SE->getSelector());
|
|
|
|
} else {
|
|
|
|
const auto *IT = ReceiverType->getAs<ObjCInterfaceType>();
|
|
|
|
if (!IT)
|
|
|
|
return;
|
|
|
|
ImpliedMethod = IT->getDecl()->lookupClassMethod(SE->getSelector());
|
|
|
|
if (!ImpliedMethod)
|
|
|
|
ImpliedMethod =
|
|
|
|
IT->getDecl()->lookupPrivateClassMethod(SE->getSelector());
|
|
|
|
}
|
|
|
|
if (!ImpliedMethod)
|
|
|
|
return;
|
|
|
|
QualType Ret = ImpliedMethod->getReturnType();
|
|
|
|
if (Ret->isRecordType() || Ret->isVectorType() || Ret->isExtVectorType()) {
|
|
|
|
QualType Ret = ImpliedMethod->getReturnType();
|
|
|
|
S.Diag(Loc, diag::warn_objc_unsafe_perform_selector)
|
|
|
|
<< Method->getSelector()
|
|
|
|
<< (!Ret->isRecordType()
|
|
|
|
? /*Vector*/ 2
|
|
|
|
: Ret->isUnionType() ? /*Union*/ 1 : /*Struct*/ 0);
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(ImpliedMethod->getBeginLoc(),
|
2017-03-06 23:58:34 +08:00
|
|
|
diag::note_objc_unsafe_perform_selector_method_declared_here)
|
|
|
|
<< ImpliedMethod->getSelector() << Ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Diagnose use of %s directive in an NSString which is being passed
|
2014-09-10 07:10:54 +08:00
|
|
|
/// as formatting string to formatting method.
|
|
|
|
static void
|
|
|
|
DiagnoseCStringFormatDirectiveInObjCAPI(Sema &S,
|
|
|
|
ObjCMethodDecl *Method,
|
|
|
|
Selector Sel,
|
|
|
|
Expr **Args, unsigned NumArgs) {
|
2014-09-12 03:13:23 +08:00
|
|
|
unsigned Idx = 0;
|
|
|
|
bool Format = false;
|
2014-09-10 07:10:54 +08:00
|
|
|
ObjCStringFormatFamily SFFamily = Sel.getStringFormatFamily();
|
|
|
|
if (SFFamily == ObjCStringFormatFamily::SFF_NSString) {
|
2014-09-12 03:13:23 +08:00
|
|
|
Idx = 0;
|
|
|
|
Format = true;
|
|
|
|
}
|
|
|
|
else if (Method) {
|
|
|
|
for (const auto *I : Method->specific_attrs<FormatAttr>()) {
|
|
|
|
if (S.GetFormatNSStringIdx(I, Idx)) {
|
|
|
|
Format = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!Format || NumArgs <= Idx)
|
|
|
|
return;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2014-09-12 03:13:23 +08:00
|
|
|
Expr *FormatExpr = Args[Idx];
|
|
|
|
if (ObjCStringLiteral *OSL =
|
|
|
|
dyn_cast<ObjCStringLiteral>(FormatExpr->IgnoreParenImpCasts())) {
|
|
|
|
StringLiteral *FormatString = OSL->getString();
|
|
|
|
if (S.FormatStringHasSArg(FormatString)) {
|
|
|
|
S.Diag(FormatExpr->getExprLoc(), diag::warn_objc_cdirective_format_string)
|
2014-09-10 07:10:54 +08:00
|
|
|
<< "%s" << 0 << 0;
|
2014-09-12 03:13:23 +08:00
|
|
|
if (Method)
|
|
|
|
S.Diag(Method->getLocation(), diag::note_method_declared_at)
|
2014-09-10 07:10:54 +08:00
|
|
|
<< Method->getDeclName();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Build an Objective-C class message expression.
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
///
|
|
|
|
/// This routine takes care of both normal class messages and
|
|
|
|
/// class messages to the superclass.
|
|
|
|
///
|
|
|
|
/// \param ReceiverTypeInfo Type source information that describes the
|
|
|
|
/// receiver of this message. This may be NULL, in which case we are
|
|
|
|
/// sending to the superclass and \p SuperLoc must be a valid source
|
|
|
|
/// location.
|
|
|
|
|
|
|
|
/// \param ReceiverType The type of the object receiving the
|
|
|
|
/// message. When \p ReceiverTypeInfo is non-NULL, this is the same
|
|
|
|
/// type as that refers to. For a superclass send, this is the type of
|
|
|
|
/// the superclass.
|
|
|
|
///
|
|
|
|
/// \param SuperLoc The location of the "super" keyword in a
|
|
|
|
/// superclass message.
|
|
|
|
///
|
|
|
|
/// \param Sel The selector to which the message is being sent.
|
|
|
|
///
|
2010-04-23 01:01:48 +08:00
|
|
|
/// \param Method The method that this class message is invoking, if
|
|
|
|
/// already known.
|
|
|
|
///
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
/// \param LBracLoc The location of the opening square bracket ']'.
|
|
|
|
///
|
2012-06-22 16:10:18 +08:00
|
|
|
/// \param RBracLoc The location of the closing square bracket ']'.
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
///
|
2012-06-22 16:10:18 +08:00
|
|
|
/// \param ArgsIn The message arguments.
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Sema::BuildClassMessage(TypeSourceInfo *ReceiverTypeInfo,
|
2010-09-15 22:51:05 +08:00
|
|
|
QualType ReceiverType,
|
|
|
|
SourceLocation SuperLoc,
|
|
|
|
Selector Sel,
|
|
|
|
ObjCMethodDecl *Method,
|
2018-07-31 03:24:48 +08:00
|
|
|
SourceLocation LBracLoc,
|
2011-10-03 14:36:17 +08:00
|
|
|
ArrayRef<SourceLocation> SelectorLocs,
|
2010-09-15 22:51:05 +08:00
|
|
|
SourceLocation RBracLoc,
|
2012-01-12 10:34:39 +08:00
|
|
|
MultiExprArg ArgsIn,
|
|
|
|
bool isImplicit) {
|
2010-09-15 22:51:05 +08:00
|
|
|
SourceLocation Loc = SuperLoc.isValid()? SuperLoc
|
2010-09-16 09:51:54 +08:00
|
|
|
: ReceiverTypeInfo->getTypeLoc().getSourceRange().getBegin();
|
2010-09-15 22:51:05 +08:00
|
|
|
if (LBracLoc.isInvalid()) {
|
|
|
|
Diag(Loc, diag::err_missing_open_square_message_send)
|
|
|
|
<< FixItHint::CreateInsertion(Loc, "[");
|
|
|
|
LBracLoc = Loc;
|
|
|
|
}
|
2018-03-30 01:34:09 +08:00
|
|
|
ArrayRef<SourceLocation> SelectorSlotLocs;
|
2013-05-01 08:24:09 +08:00
|
|
|
if (!SelectorLocs.empty() && SelectorLocs.front().isValid())
|
2018-03-30 01:34:09 +08:00
|
|
|
SelectorSlotLocs = SelectorLocs;
|
2013-05-01 08:24:09 +08:00
|
|
|
else
|
2018-03-30 01:34:09 +08:00
|
|
|
SelectorSlotLocs = Loc;
|
|
|
|
SourceLocation SelLoc = SelectorSlotLocs.front();
|
2013-05-01 08:24:09 +08:00
|
|
|
|
2010-04-23 00:44:27 +08:00
|
|
|
if (ReceiverType->isDependentType()) {
|
|
|
|
// If the receiver type is dependent, we can't type-check anything
|
|
|
|
// at this point. Build a dependent expression.
|
|
|
|
unsigned NumArgs = ArgsIn.size();
|
2012-08-24 07:38:35 +08:00
|
|
|
Expr **Args = ArgsIn.data();
|
2010-04-23 00:44:27 +08:00
|
|
|
assert(SuperLoc.isInvalid() && "Message to super with dependent type");
|
2014-05-29 22:05:12 +08:00
|
|
|
return ObjCMessageExpr::Create(
|
|
|
|
Context, ReceiverType, VK_RValue, LBracLoc, ReceiverTypeInfo, Sel,
|
|
|
|
SelectorLocs, /*Method=*/nullptr, makeArrayRef(Args, NumArgs), RBracLoc,
|
|
|
|
isImplicit);
|
2010-04-23 00:44:27 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// Find the class to which we are sending this message.
|
2014-05-26 14:22:03 +08:00
|
|
|
ObjCInterfaceDecl *Class = nullptr;
|
2010-05-15 19:32:37 +08:00
|
|
|
const ObjCObjectType *ClassType = ReceiverType->getAs<ObjCObjectType>();
|
|
|
|
if (!ClassType || !(Class = ClassType->getInterface())) {
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
Diag(Loc, diag::err_invalid_receiver_class_message)
|
|
|
|
<< ReceiverType;
|
|
|
|
return ExprError();
|
2008-07-26 03:39:00 +08:00
|
|
|
}
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
assert(Class && "We don't know which class we're messaging?");
|
2011-10-16 03:18:36 +08:00
|
|
|
// objc++ diagnoses during typename annotation.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!getLangOpts().CPlusPlus)
|
2018-03-30 01:34:09 +08:00
|
|
|
(void)DiagnoseUseOfDecl(Class, SelectorSlotLocs);
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// Find the method we are messaging.
|
2010-04-23 01:01:48 +08:00
|
|
|
if (!Method) {
|
2018-07-31 03:24:48 +08:00
|
|
|
SourceRange TypeRange
|
2011-11-15 06:10:01 +08:00
|
|
|
= SuperLoc.isValid()? SourceRange(SuperLoc)
|
|
|
|
: ReceiverTypeInfo->getTypeLoc().getSourceRange();
|
2012-05-05 00:32:21 +08:00
|
|
|
if (RequireCompleteType(Loc, Context.getObjCInterfaceType(Class),
|
2012-03-11 15:00:24 +08:00
|
|
|
(getLangOpts().ObjCAutoRefCount
|
2012-05-05 00:32:21 +08:00
|
|
|
? diag::err_arc_receiver_forward_class
|
|
|
|
: diag::warn_receiver_forward_class),
|
|
|
|
TypeRange)) {
|
2010-04-23 01:01:48 +08:00
|
|
|
// A forward class used in messaging is treated as a 'Class'
|
2018-07-31 03:24:48 +08:00
|
|
|
Method = LookupFactoryMethodInGlobalPool(Sel,
|
2010-04-23 01:01:48 +08:00
|
|
|
SourceRange(LBracLoc, RBracLoc));
|
2012-03-11 15:00:24 +08:00
|
|
|
if (Method && !getLangOpts().ObjCAutoRefCount)
|
2010-04-23 01:01:48 +08:00
|
|
|
Diag(Method->getLocation(), diag::note_method_sent_forward_class)
|
|
|
|
<< Method->getDeclName();
|
|
|
|
}
|
|
|
|
if (!Method)
|
|
|
|
Method = Class->lookupClassMethod(Sel);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2010-04-23 01:01:48 +08:00
|
|
|
// If we have an implementation in scope, check "private" methods.
|
|
|
|
if (!Method)
|
2012-07-28 03:07:44 +08:00
|
|
|
Method = Class->lookupPrivateClassMethod(Sel);
|
2008-07-26 03:39:00 +08:00
|
|
|
|
2018-09-11 06:20:09 +08:00
|
|
|
if (Method && DiagnoseUseOfDecl(Method, SelectorSlotLocs,
|
|
|
|
nullptr, false, false, Class))
|
2010-04-23 01:01:48 +08:00
|
|
|
return ExprError();
|
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// Check the argument types and determine the result type.
|
|
|
|
QualType ReturnType;
|
2010-11-18 14:31:45 +08:00
|
|
|
ExprValueKind VK = VK_RValue;
|
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
unsigned NumArgs = ArgsIn.size();
|
2012-08-24 07:38:35 +08:00
|
|
|
Expr **Args = ArgsIn.data();
|
2018-12-21 06:11:11 +08:00
|
|
|
if (CheckMessageArgumentTypes(/*Receiver=*/nullptr, ReceiverType,
|
|
|
|
MultiExprArg(Args, NumArgs), Sel, SelectorLocs,
|
|
|
|
Method, true, SuperLoc.isValid(), LBracLoc,
|
|
|
|
RBracLoc, SourceRange(), ReturnType, VK))
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
return ExprError();
|
2008-06-24 23:50:53 +08:00
|
|
|
|
2014-01-26 00:55:45 +08:00
|
|
|
if (Method && !Method->getReturnType()->isVoidType() &&
|
|
|
|
RequireCompleteType(LBracLoc, Method->getReturnType(),
|
2011-01-11 11:23:19 +08:00
|
|
|
diag::err_illegal_message_expr_incomplete_type))
|
|
|
|
return ExprError();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2014-08-23 03:52:49 +08:00
|
|
|
// Warn about explicit call of +initialize on its own class. But not on 'super'.
|
2014-08-26 05:27:38 +08:00
|
|
|
if (Method && Method->getMethodFamily() == OMF_initialize) {
|
|
|
|
if (!SuperLoc.isValid()) {
|
|
|
|
const ObjCInterfaceDecl *ID =
|
|
|
|
dyn_cast<ObjCInterfaceDecl>(Method->getDeclContext());
|
|
|
|
if (ID == Class) {
|
|
|
|
Diag(Loc, diag::warn_direct_initialize_call);
|
|
|
|
Diag(Method->getLocation(), diag::note_method_declared_at)
|
|
|
|
<< Method->getDeclName();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (ObjCMethodDecl *CurMeth = getCurMethodDecl()) {
|
|
|
|
// [super initialize] is allowed only within an +initialize implementation
|
|
|
|
if (CurMeth->getMethodFamily() != OMF_initialize) {
|
|
|
|
Diag(Loc, diag::warn_direct_super_initialize_call);
|
|
|
|
Diag(Method->getLocation(), diag::note_method_declared_at)
|
|
|
|
<< Method->getDeclName();
|
|
|
|
Diag(CurMeth->getLocation(), diag::note_method_declared_at)
|
|
|
|
<< CurMeth->getDeclName();
|
|
|
|
}
|
2014-08-23 00:57:26 +08:00
|
|
|
}
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2014-09-10 07:10:54 +08:00
|
|
|
DiagnoseCStringFormatDirectiveInObjCAPI(*this, Method, Sel, Args, NumArgs);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// Construct the appropriate ObjCMessageExpr.
|
2012-03-07 04:05:56 +08:00
|
|
|
ObjCMessageExpr *Result;
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
if (SuperLoc.isValid())
|
2018-07-31 03:24:48 +08:00
|
|
|
Result = ObjCMessageExpr::Create(Context, ReturnType, VK, LBracLoc,
|
|
|
|
SuperLoc, /*IsInstanceSuper=*/false,
|
2011-10-03 14:36:17 +08:00
|
|
|
ReceiverType, Sel, SelectorLocs,
|
2011-10-03 14:36:45 +08:00
|
|
|
Method, makeArrayRef(Args, NumArgs),
|
2012-01-12 10:34:39 +08:00
|
|
|
RBracLoc, isImplicit);
|
2012-03-07 04:05:56 +08:00
|
|
|
else {
|
2018-07-31 03:24:48 +08:00
|
|
|
Result = ObjCMessageExpr::Create(Context, ReturnType, VK, LBracLoc,
|
2011-10-03 14:36:17 +08:00
|
|
|
ReceiverTypeInfo, Sel, SelectorLocs,
|
2011-10-03 14:36:45 +08:00
|
|
|
Method, makeArrayRef(Args, NumArgs),
|
2012-01-12 10:34:39 +08:00
|
|
|
RBracLoc, isImplicit);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (!isImplicit)
|
|
|
|
checkCocoaAPI(*this, Result);
|
|
|
|
}
|
2017-03-06 23:58:34 +08:00
|
|
|
if (Method)
|
|
|
|
checkFoundationAPI(*this, SelLoc, Method, makeArrayRef(Args, NumArgs),
|
|
|
|
ReceiverType, /*IsClassObjectCall=*/true);
|
2010-05-22 13:17:18 +08:00
|
|
|
return MaybeBindToTemporary(Result);
|
2008-01-05 06:32:30 +08:00
|
|
|
}
|
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// ActOnClassMessage - used for both unary and keyword messages.
|
2008-01-05 06:32:30 +08:00
|
|
|
// ArgExprs is optional - if it is present, the number of expressions
|
|
|
|
// is obtained from Sel.getNumArgs().
|
2018-07-31 03:24:48 +08:00
|
|
|
ExprResult Sema::ActOnClassMessage(Scope *S,
|
2010-09-16 07:19:31 +08:00
|
|
|
ParsedType Receiver,
|
|
|
|
Selector Sel,
|
|
|
|
SourceLocation LBracLoc,
|
2011-10-03 14:36:17 +08:00
|
|
|
ArrayRef<SourceLocation> SelectorLocs,
|
2010-09-16 07:19:31 +08:00
|
|
|
SourceLocation RBracLoc,
|
|
|
|
MultiExprArg Args) {
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
TypeSourceInfo *ReceiverTypeInfo;
|
|
|
|
QualType ReceiverType = GetTypeFromParser(Receiver, &ReceiverTypeInfo);
|
|
|
|
if (ReceiverType.isNull())
|
|
|
|
return ExprError();
|
2009-09-09 23:08:12 +08:00
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
if (!ReceiverTypeInfo)
|
|
|
|
ReceiverTypeInfo = Context.getTrivialTypeSourceInfo(ReceiverType, LBracLoc);
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2018-07-31 03:24:48 +08:00
|
|
|
return BuildClassMessage(ReceiverTypeInfo, ReceiverType,
|
2014-05-26 14:22:03 +08:00
|
|
|
/*SuperLoc=*/SourceLocation(), Sel,
|
|
|
|
/*Method=*/nullptr, LBracLoc, SelectorLocs, RBracLoc,
|
|
|
|
Args);
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
}
|
2008-11-18 06:29:32 +08:00
|
|
|
|
2012-01-12 10:34:39 +08:00
|
|
|
ExprResult Sema::BuildInstanceMessageImplicit(Expr *Receiver,
|
|
|
|
QualType ReceiverType,
|
|
|
|
SourceLocation Loc,
|
|
|
|
Selector Sel,
|
|
|
|
ObjCMethodDecl *Method,
|
|
|
|
MultiExprArg Args) {
|
|
|
|
return BuildInstanceMessage(Receiver, ReceiverType,
|
|
|
|
/*SuperLoc=*/!Receiver ? Loc : SourceLocation(),
|
|
|
|
Sel, Method, Loc, Loc, Loc, Args,
|
|
|
|
/*isImplicit=*/true);
|
|
|
|
}
|
|
|
|
|
2017-03-16 01:16:41 +08:00
|
|
|
static bool isMethodDeclaredInRootProtocol(Sema &S, const ObjCMethodDecl *M) {
|
|
|
|
if (!S.NSAPIObj)
|
|
|
|
return false;
|
|
|
|
const auto *Protocol = dyn_cast<ObjCProtocolDecl>(M->getDeclContext());
|
|
|
|
if (!Protocol)
|
|
|
|
return false;
|
|
|
|
const IdentifierInfo *II = S.NSAPIObj->getNSClassId(NSAPI::ClassId_NSObject);
|
|
|
|
if (const auto *RootClass = dyn_cast_or_null<ObjCInterfaceDecl>(
|
2018-08-10 05:08:08 +08:00
|
|
|
S.LookupSingleName(S.TUScope, II, Protocol->getBeginLoc(),
|
2017-03-16 01:16:41 +08:00
|
|
|
Sema::LookupOrdinaryName))) {
|
|
|
|
for (const ObjCProtocolDecl *P : RootClass->all_referenced_protocols()) {
|
|
|
|
if (P->getCanonicalDecl() == Protocol->getCanonicalDecl())
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:00:01 +08:00
|
|
|
/// Build an Objective-C instance message expression.
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
///
|
|
|
|
/// This routine takes care of both normal instance messages and
|
|
|
|
/// instance messages to the superclass instance.
|
|
|
|
///
|
|
|
|
/// \param Receiver The expression that computes the object that will
|
|
|
|
/// receive this message. This may be empty, in which case we are
|
|
|
|
/// sending to the superclass instance and \p SuperLoc must be a valid
|
|
|
|
/// source location.
|
|
|
|
///
|
|
|
|
/// \param ReceiverType The (static) type of the object receiving the
|
|
|
|
/// message. When a \p Receiver expression is provided, this is the
|
|
|
|
/// same type as that expression. For a superclass instance send, this
|
|
|
|
/// is a pointer to the type of the superclass.
|
|
|
|
///
|
|
|
|
/// \param SuperLoc The location of the "super" keyword in a
|
|
|
|
/// superclass instance message.
|
|
|
|
///
|
|
|
|
/// \param Sel The selector to which the message is being sent.
|
|
|
|
///
|
2010-04-23 01:01:48 +08:00
|
|
|
/// \param Method The method that this instance message is invoking, if
|
|
|
|
/// already known.
|
|
|
|
///
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
/// \param LBracLoc The location of the opening square bracket ']'.
|
|
|
|
///
|
2012-06-22 16:10:18 +08:00
|
|
|
/// \param RBracLoc The location of the closing square bracket ']'.
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
///
|
2012-06-22 16:10:18 +08:00
|
|
|
/// \param ArgsIn The message arguments.
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Sema::BuildInstanceMessage(Expr *Receiver,
|
2010-12-11 04:08:27 +08:00
|
|
|
QualType ReceiverType,
|
|
|
|
SourceLocation SuperLoc,
|
|
|
|
Selector Sel,
|
|
|
|
ObjCMethodDecl *Method,
|
2018-07-31 03:24:48 +08:00
|
|
|
SourceLocation LBracLoc,
|
2011-10-03 14:36:17 +08:00
|
|
|
ArrayRef<SourceLocation> SelectorLocs,
|
2010-12-11 04:08:27 +08:00
|
|
|
SourceLocation RBracLoc,
|
2012-01-12 10:34:39 +08:00
|
|
|
MultiExprArg ArgsIn,
|
|
|
|
bool isImplicit) {
|
2016-11-04 14:11:54 +08:00
|
|
|
assert((Receiver || SuperLoc.isValid()) && "If the Receiver is null, the "
|
|
|
|
"SuperLoc must be valid so we can "
|
|
|
|
"use it instead.");
|
|
|
|
|
2010-09-15 22:51:05 +08:00
|
|
|
// The location of the receiver.
|
2018-08-10 05:08:08 +08:00
|
|
|
SourceLocation Loc = SuperLoc.isValid() ? SuperLoc : Receiver->getBeginLoc();
|
2013-05-01 08:24:09 +08:00
|
|
|
SourceRange RecRange =
|
|
|
|
SuperLoc.isValid()? SuperLoc : Receiver->getSourceRange();
|
2018-03-30 01:34:09 +08:00
|
|
|
ArrayRef<SourceLocation> SelectorSlotLocs;
|
2013-05-01 08:24:09 +08:00
|
|
|
if (!SelectorLocs.empty() && SelectorLocs.front().isValid())
|
2018-03-30 01:34:09 +08:00
|
|
|
SelectorSlotLocs = SelectorLocs;
|
2013-05-01 08:24:09 +08:00
|
|
|
else
|
2018-03-30 01:34:09 +08:00
|
|
|
SelectorSlotLocs = Loc;
|
|
|
|
SourceLocation SelLoc = SelectorSlotLocs.front();
|
2013-05-01 08:24:09 +08:00
|
|
|
|
2010-09-15 22:51:05 +08:00
|
|
|
if (LBracLoc.isInvalid()) {
|
|
|
|
Diag(Loc, diag::err_missing_open_square_message_send)
|
|
|
|
<< FixItHint::CreateInsertion(Loc, "[");
|
|
|
|
LBracLoc = Loc;
|
|
|
|
}
|
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// If we have a receiver expression, perform appropriate promotions
|
|
|
|
// and determine receiver type.
|
|
|
|
if (Receiver) {
|
2011-10-18 02:40:02 +08:00
|
|
|
if (Receiver->hasPlaceholderType()) {
|
2011-12-01 09:37:36 +08:00
|
|
|
ExprResult Result;
|
|
|
|
if (Receiver->getType() == Context.UnknownAnyTy)
|
|
|
|
Result = forceUnknownAnyToType(Receiver, Context.getObjCIdType());
|
|
|
|
else
|
|
|
|
Result = CheckPlaceholderExpr(Receiver);
|
|
|
|
if (Result.isInvalid()) return ExprError();
|
2014-05-29 18:55:11 +08:00
|
|
|
Receiver = Result.get();
|
2011-10-18 02:40:02 +08:00
|
|
|
}
|
|
|
|
|
2010-04-23 00:44:27 +08:00
|
|
|
if (Receiver->isTypeDependent()) {
|
|
|
|
// If the receiver is type-dependent, we can't type-check anything
|
|
|
|
// at this point. Build a dependent expression.
|
|
|
|
unsigned NumArgs = ArgsIn.size();
|
2012-08-24 07:38:35 +08:00
|
|
|
Expr **Args = ArgsIn.data();
|
2010-04-23 00:44:27 +08:00
|
|
|
assert(SuperLoc.isInvalid() && "Message to super with dependent type");
|
2014-05-29 22:05:12 +08:00
|
|
|
return ObjCMessageExpr::Create(
|
|
|
|
Context, Context.DependentTy, VK_RValue, LBracLoc, Receiver, Sel,
|
|
|
|
SelectorLocs, /*Method=*/nullptr, makeArrayRef(Args, NumArgs),
|
|
|
|
RBracLoc, isImplicit);
|
2010-04-23 00:44:27 +08:00
|
|
|
}
|
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// If necessary, apply function/array conversion to the receiver.
|
|
|
|
// C99 6.7.5.3p[7,8].
|
2011-04-09 02:41:53 +08:00
|
|
|
ExprResult Result = DefaultFunctionArrayLvalueConversion(Receiver);
|
|
|
|
if (Result.isInvalid())
|
|
|
|
return ExprError();
|
2014-05-29 18:55:11 +08:00
|
|
|
Receiver = Result.get();
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
ReceiverType = Receiver->getType();
|
2013-03-01 17:20:14 +08:00
|
|
|
|
|
|
|
// If the receiver is an ObjC pointer, a block pointer, or an
|
|
|
|
// __attribute__((NSObject)) pointer, we don't need to do any
|
|
|
|
// special conversion in order to look up a receiver.
|
|
|
|
if (ReceiverType->isObjCRetainableType()) {
|
|
|
|
// do nothing
|
|
|
|
} else if (!getLangOpts().ObjCAutoRefCount &&
|
|
|
|
!Context.getObjCIdType().isNull() &&
|
2018-07-31 03:24:48 +08:00
|
|
|
(ReceiverType->isPointerType() ||
|
2013-03-01 17:20:14 +08:00
|
|
|
ReceiverType->isIntegerType())) {
|
|
|
|
// Implicitly convert integers and pointers to 'id' but emit a warning.
|
|
|
|
// But not in ARC.
|
|
|
|
Diag(Loc, diag::warn_bad_receiver_type)
|
2018-07-31 03:24:48 +08:00
|
|
|
<< ReceiverType
|
2013-03-01 17:20:14 +08:00
|
|
|
<< Receiver->getSourceRange();
|
|
|
|
if (ReceiverType->isPointerType()) {
|
2018-07-31 03:24:48 +08:00
|
|
|
Receiver = ImpCastExprToType(Receiver, Context.getObjCIdType(),
|
2014-05-29 18:55:11 +08:00
|
|
|
CK_CPointerToObjCPointerCast).get();
|
2013-03-01 17:20:14 +08:00
|
|
|
} else {
|
|
|
|
// TODO: specialized warning on null receivers?
|
|
|
|
bool IsNull = Receiver->isNullPointerConstant(Context,
|
|
|
|
Expr::NPC_ValueDependentIsNull);
|
|
|
|
CastKind Kind = IsNull ? CK_NullToPointer : CK_IntegralToPointer;
|
|
|
|
Receiver = ImpCastExprToType(Receiver, Context.getObjCIdType(),
|
2014-05-29 18:55:11 +08:00
|
|
|
Kind).get();
|
2013-03-01 17:20:14 +08:00
|
|
|
}
|
|
|
|
ReceiverType = Receiver->getType();
|
|
|
|
} else if (getLangOpts().CPlusPlus) {
|
2013-11-08 06:34:54 +08:00
|
|
|
// The receiver must be a complete type.
|
|
|
|
if (RequireCompleteType(Loc, Receiver->getType(),
|
|
|
|
diag::err_incomplete_receiver_type))
|
|
|
|
return ExprError();
|
|
|
|
|
2013-03-01 17:20:14 +08:00
|
|
|
ExprResult result = PerformContextuallyConvertToObjCPointer(Receiver);
|
|
|
|
if (result.isUsable()) {
|
2014-05-29 18:55:11 +08:00
|
|
|
Receiver = result.get();
|
2013-03-01 17:20:14 +08:00
|
|
|
ReceiverType = Receiver->getType();
|
|
|
|
}
|
|
|
|
}
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
}
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
|
2017-08-26 00:12:17 +08:00
|
|
|
if (ReceiverType->isObjCIdType() && !isImplicit)
|
|
|
|
Diag(Receiver->getExprLoc(), diag::warn_messaging_unqualified_id);
|
|
|
|
|
2013-03-01 17:20:14 +08:00
|
|
|
// There's a somewhat weird interaction here where we assume that we
|
|
|
|
// won't actually have a method unless we also don't need to do some
|
|
|
|
// of the more detailed type-checking on the receiver.
|
|
|
|
|
2010-04-23 01:01:48 +08:00
|
|
|
if (!Method) {
|
2015-07-07 11:58:42 +08:00
|
|
|
// Handle messages to id and __kindof types (where we use the
|
|
|
|
// global method pool).
|
|
|
|
const ObjCObjectType *typeBound = nullptr;
|
|
|
|
bool receiverIsIdLike = ReceiverType->isObjCIdOrObjectKindOfType(Context,
|
|
|
|
typeBound);
|
|
|
|
if (receiverIsIdLike || ReceiverType->isBlockPointerType() ||
|
2010-04-23 01:01:48 +08:00
|
|
|
(Receiver && Context.isObjCNSObjectType(Receiver->getType()))) {
|
2016-04-08 03:30:20 +08:00
|
|
|
SmallVector<ObjCMethodDecl*, 4> Methods;
|
2016-04-08 03:32:24 +08:00
|
|
|
// If we have a type bound, further filter the methods.
|
2016-04-08 03:30:20 +08:00
|
|
|
CollectMultipleMethodsInGlobalPool(Sel, Methods, true/*InstanceFirst*/,
|
2016-04-08 03:32:24 +08:00
|
|
|
true/*CheckTheOther*/, typeBound);
|
2016-04-08 03:30:20 +08:00
|
|
|
if (!Methods.empty()) {
|
2016-09-01 09:26:58 +08:00
|
|
|
// We choose the first method as the initial candidate, then try to
|
2016-04-08 03:30:20 +08:00
|
|
|
// select a better one.
|
|
|
|
Method = Methods[0];
|
|
|
|
|
2014-08-14 05:24:14 +08:00
|
|
|
if (ObjCMethodDecl *BestMethod =
|
2016-04-08 03:30:20 +08:00
|
|
|
SelectBestMethod(Sel, ArgsIn, Method->isInstanceMethod(), Methods))
|
2014-08-14 05:24:14 +08:00
|
|
|
Method = BestMethod;
|
2016-04-08 03:30:20 +08:00
|
|
|
|
2015-04-16 01:26:21 +08:00
|
|
|
if (!AreMultipleMethodsInGlobalPool(Sel, Method,
|
|
|
|
SourceRange(LBracLoc, RBracLoc),
|
2016-04-08 03:30:20 +08:00
|
|
|
receiverIsIdLike, Methods))
|
2018-03-30 01:34:09 +08:00
|
|
|
DiagnoseUseOfDecl(Method, SelectorSlotLocs);
|
2014-11-08 07:51:15 +08:00
|
|
|
}
|
2015-07-07 11:58:42 +08:00
|
|
|
} else if (ReceiverType->isObjCClassOrClassKindOfType() ||
|
2010-04-23 01:01:48 +08:00
|
|
|
ReceiverType->isObjCQualifiedClassType()) {
|
|
|
|
// Handle messages to Class.
|
2014-12-27 11:58:08 +08:00
|
|
|
// We allow sending a message to a qualified Class ("Class<foo>"), which
|
|
|
|
// is ok as long as one of the protocols implements the selector (if not,
|
|
|
|
// warn).
|
2015-07-07 11:58:42 +08:00
|
|
|
if (!ReceiverType->isObjCClassOrClassKindOfType()) {
|
|
|
|
const ObjCObjectPointerType *QClassTy
|
|
|
|
= ReceiverType->getAsObjCQualifiedClassType();
|
2011-04-07 02:40:08 +08:00
|
|
|
// Search protocols for class methods.
|
|
|
|
Method = LookupMethodInQualifiedType(Sel, QClassTy, false);
|
|
|
|
if (!Method) {
|
|
|
|
Method = LookupMethodInQualifiedType(Sel, QClassTy, true);
|
|
|
|
// warn if instance method found for a Class message.
|
2017-03-16 01:16:41 +08:00
|
|
|
if (Method && !isMethodDeclaredInRootProtocol(*this, Method)) {
|
2013-05-01 08:24:09 +08:00
|
|
|
Diag(SelLoc, diag::warn_instance_method_on_class_found)
|
2011-04-07 02:40:08 +08:00
|
|
|
<< Method->getSelector() << Sel;
|
2012-02-28 06:55:11 +08:00
|
|
|
Diag(Method->getLocation(), diag::note_method_declared_at)
|
|
|
|
<< Method->getDeclName();
|
2011-04-07 02:40:08 +08:00
|
|
|
}
|
2009-03-04 23:11:40 +08:00
|
|
|
}
|
2011-04-07 02:40:08 +08:00
|
|
|
} else {
|
|
|
|
if (ObjCMethodDecl *CurMeth = getCurMethodDecl()) {
|
|
|
|
if (ObjCInterfaceDecl *ClassDecl = CurMeth->getClassInterface()) {
|
2019-02-05 07:30:57 +08:00
|
|
|
// As a guess, try looking for the method in the current interface.
|
|
|
|
// This very well may not produce the "right" method.
|
2018-09-11 06:20:09 +08:00
|
|
|
|
2011-04-07 02:40:08 +08:00
|
|
|
// First check the public methods in the class interface.
|
|
|
|
Method = ClassDecl->lookupClassMethod(Sel);
|
|
|
|
|
|
|
|
if (!Method)
|
2012-07-28 03:07:44 +08:00
|
|
|
Method = ClassDecl->lookupPrivateClassMethod(Sel);
|
2018-09-11 06:20:09 +08:00
|
|
|
|
2019-02-05 07:30:57 +08:00
|
|
|
if (Method && DiagnoseUseOfDecl(Method, SelectorSlotLocs))
|
2018-09-11 06:20:09 +08:00
|
|
|
return ExprError();
|
2011-04-07 02:40:08 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!Method) {
|
|
|
|
// If not messaging 'self', look for any factory method named 'Sel'.
|
2011-09-28 00:10:05 +08:00
|
|
|
if (!Receiver || !isSelfExpr(Receiver)) {
|
2016-04-08 03:30:20 +08:00
|
|
|
// If no class (factory) method was found, check if an _instance_
|
|
|
|
// method of the same name exists in the root class only.
|
|
|
|
SmallVector<ObjCMethodDecl*, 4> Methods;
|
|
|
|
CollectMultipleMethodsInGlobalPool(Sel, Methods,
|
|
|
|
false/*InstanceFirst*/,
|
|
|
|
true/*CheckTheOther*/);
|
|
|
|
if (!Methods.empty()) {
|
2016-09-01 09:26:58 +08:00
|
|
|
// We choose the first method as the initial candidate, then try
|
2016-04-08 03:30:20 +08:00
|
|
|
// to select a better one.
|
|
|
|
Method = Methods[0];
|
|
|
|
|
2018-04-06 23:14:32 +08:00
|
|
|
// If we find an instance method, emit warning.
|
2016-04-08 03:30:20 +08:00
|
|
|
if (Method->isInstanceMethod()) {
|
|
|
|
if (const ObjCInterfaceDecl *ID =
|
|
|
|
dyn_cast<ObjCInterfaceDecl>(Method->getDeclContext())) {
|
|
|
|
if (ID->getSuperClass())
|
|
|
|
Diag(SelLoc, diag::warn_root_inst_method_not_found)
|
|
|
|
<< Sel << SourceRange(LBracLoc, RBracLoc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ObjCMethodDecl *BestMethod =
|
|
|
|
SelectBestMethod(Sel, ArgsIn, Method->isInstanceMethod(),
|
|
|
|
Methods))
|
|
|
|
Method = BestMethod;
|
2011-04-07 02:40:08 +08:00
|
|
|
}
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2014-05-26 14:22:03 +08:00
|
|
|
ObjCInterfaceDecl *ClassDecl = nullptr;
|
2010-04-23 01:01:48 +08:00
|
|
|
|
|
|
|
// We allow sending a message to a qualified ID ("id<foo>"), which is ok as
|
|
|
|
// long as one of the protocols implements the selector (if not, warn).
|
2012-06-24 02:39:57 +08:00
|
|
|
// And as long as message is not deprecated/unavailable (warn if it is).
|
2018-07-31 03:24:48 +08:00
|
|
|
if (const ObjCObjectPointerType *QIdTy
|
2010-04-23 01:01:48 +08:00
|
|
|
= ReceiverType->getAsObjCQualifiedIdType()) {
|
|
|
|
// Search protocols for instance methods.
|
2011-03-10 06:17:12 +08:00
|
|
|
Method = LookupMethodInQualifiedType(Sel, QIdTy, true);
|
|
|
|
if (!Method)
|
|
|
|
Method = LookupMethodInQualifiedType(Sel, QIdTy, false);
|
2018-03-30 01:34:09 +08:00
|
|
|
if (Method && DiagnoseUseOfDecl(Method, SelectorSlotLocs))
|
2012-06-24 02:39:57 +08:00
|
|
|
return ExprError();
|
2010-04-23 01:01:48 +08:00
|
|
|
} else if (const ObjCObjectPointerType *OCIType
|
|
|
|
= ReceiverType->getAsObjCInterfacePointerType()) {
|
|
|
|
// We allow sending a message to a pointer to an interface (an object).
|
|
|
|
ClassDecl = OCIType->getInterfaceDecl();
|
2011-06-16 07:02:42 +08:00
|
|
|
|
2011-11-15 06:10:01 +08:00
|
|
|
// Try to complete the type. Under ARC, this is a hard error from which
|
|
|
|
// we don't try to recover.
|
2015-12-19 06:40:25 +08:00
|
|
|
// FIXME: In the non-ARC case, this will still be a hard error if the
|
|
|
|
// definition is found in a module that's not visible.
|
2014-05-26 14:22:03 +08:00
|
|
|
const ObjCInterfaceDecl *forwardClass = nullptr;
|
2011-11-15 06:10:01 +08:00
|
|
|
if (RequireCompleteType(Loc, OCIType->getPointeeType(),
|
2012-03-11 15:00:24 +08:00
|
|
|
getLangOpts().ObjCAutoRefCount
|
2012-05-05 00:32:21 +08:00
|
|
|
? diag::err_arc_receiver_forward_instance
|
|
|
|
: diag::warn_receiver_forward_instance,
|
|
|
|
Receiver? Receiver->getSourceRange()
|
|
|
|
: SourceRange(SuperLoc))) {
|
2012-03-11 15:00:24 +08:00
|
|
|
if (getLangOpts().ObjCAutoRefCount)
|
2011-11-15 06:10:01 +08:00
|
|
|
return ExprError();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-11-15 06:10:01 +08:00
|
|
|
forwardClass = OCIType->getInterfaceDecl();
|
2018-08-10 05:08:08 +08:00
|
|
|
Diag(Receiver ? Receiver->getBeginLoc() : SuperLoc,
|
|
|
|
diag::note_receiver_is_id);
|
2014-05-26 14:22:03 +08:00
|
|
|
Method = nullptr;
|
2011-12-15 13:27:12 +08:00
|
|
|
} else {
|
|
|
|
Method = ClassDecl->lookupInstanceMethod(Sel);
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2010-04-23 01:01:48 +08:00
|
|
|
|
2011-03-10 06:17:12 +08:00
|
|
|
if (!Method)
|
2010-04-23 01:01:48 +08:00
|
|
|
// Search protocol qualifiers.
|
2011-03-10 06:17:12 +08:00
|
|
|
Method = LookupMethodInQualifiedType(Sel, OCIType, true);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2010-04-23 01:01:48 +08:00
|
|
|
if (!Method) {
|
|
|
|
// If we have implementations in scope, check "private" methods.
|
2012-07-28 03:07:44 +08:00
|
|
|
Method = ClassDecl->lookupPrivateMethod(Sel);
|
2010-04-23 01:01:48 +08:00
|
|
|
|
2012-03-11 15:00:24 +08:00
|
|
|
if (!Method && getLangOpts().ObjCAutoRefCount) {
|
2013-05-01 08:24:09 +08:00
|
|
|
Diag(SelLoc, diag::err_arc_may_not_respond)
|
|
|
|
<< OCIType->getPointeeType() << Sel << RecRange
|
2012-11-28 09:27:44 +08:00
|
|
|
<< SourceRange(SelectorLocs.front(), SelectorLocs.back());
|
2011-06-16 07:02:42 +08:00
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2011-09-28 00:10:05 +08:00
|
|
|
if (!Method && (!Receiver || !isSelfExpr(Receiver))) {
|
2010-04-23 01:01:48 +08:00
|
|
|
// If we still haven't found a method, look in the global pool. This
|
|
|
|
// behavior isn't very desirable, however we need it for GCC
|
|
|
|
// compatibility. FIXME: should we deviate??
|
|
|
|
if (OCIType->qual_empty()) {
|
2016-04-08 03:30:20 +08:00
|
|
|
SmallVector<ObjCMethodDecl*, 4> Methods;
|
|
|
|
CollectMultipleMethodsInGlobalPool(Sel, Methods,
|
|
|
|
true/*InstanceFirst*/,
|
|
|
|
false/*CheckTheOther*/);
|
|
|
|
if (!Methods.empty()) {
|
2016-09-01 09:26:58 +08:00
|
|
|
// We choose the first method as the initial candidate, then try
|
2016-04-08 03:30:20 +08:00
|
|
|
// to select a better one.
|
|
|
|
Method = Methods[0];
|
|
|
|
|
|
|
|
if (ObjCMethodDecl *BestMethod =
|
|
|
|
SelectBestMethod(Sel, ArgsIn, Method->isInstanceMethod(),
|
|
|
|
Methods))
|
2015-04-16 01:26:21 +08:00
|
|
|
Method = BestMethod;
|
2016-04-08 03:30:20 +08:00
|
|
|
|
2015-04-16 01:26:21 +08:00
|
|
|
AreMultipleMethodsInGlobalPool(Sel, Method,
|
|
|
|
SourceRange(LBracLoc, RBracLoc),
|
2016-04-08 03:30:20 +08:00
|
|
|
true/*receiverIdOrClass*/,
|
|
|
|
Methods);
|
2015-04-16 01:26:21 +08:00
|
|
|
}
|
2010-12-21 08:44:01 +08:00
|
|
|
if (Method && !forwardClass)
|
2013-05-01 08:24:09 +08:00
|
|
|
Diag(SelLoc, diag::warn_maynot_respond)
|
|
|
|
<< OCIType->getInterfaceDecl()->getIdentifier()
|
|
|
|
<< Sel << RecRange;
|
2010-04-23 01:01:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-03-30 01:34:09 +08:00
|
|
|
if (Method && DiagnoseUseOfDecl(Method, SelectorSlotLocs, forwardClass))
|
2010-04-23 01:01:48 +08:00
|
|
|
return ExprError();
|
2011-09-09 14:11:02 +08:00
|
|
|
} else {
|
2013-03-01 17:20:14 +08:00
|
|
|
// Reject other random receiver types (e.g. structs).
|
|
|
|
Diag(Loc, diag::err_bad_receiver_type)
|
|
|
|
<< ReceiverType << Receiver->getSourceRange();
|
|
|
|
return ExprError();
|
2010-04-23 01:01:48 +08:00
|
|
|
}
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
}
|
2008-07-21 13:57:44 +08:00
|
|
|
}
|
2009-09-09 23:08:12 +08:00
|
|
|
|
2014-03-18 05:41:40 +08:00
|
|
|
FunctionScopeInfo *DIFunctionScopeInfo =
|
|
|
|
(Method && Method->getMethodFamily() == OMF_init)
|
2014-05-26 14:22:03 +08:00
|
|
|
? getEnclosingFunction() : nullptr;
|
|
|
|
|
2014-03-18 05:41:40 +08:00
|
|
|
if (DIFunctionScopeInfo &&
|
|
|
|
DIFunctionScopeInfo->ObjCIsDesignatedInit &&
|
2013-12-04 05:11:43 +08:00
|
|
|
(SuperLoc.isValid() || isSelfExpr(Receiver))) {
|
|
|
|
bool isDesignatedInitChain = false;
|
|
|
|
if (SuperLoc.isValid()) {
|
|
|
|
if (const ObjCObjectPointerType *
|
|
|
|
OCIType = ReceiverType->getAsObjCInterfacePointerType()) {
|
|
|
|
if (const ObjCInterfaceDecl *ID = OCIType->getInterfaceDecl()) {
|
2013-12-13 11:48:17 +08:00
|
|
|
// Either we know this is a designated initializer or we
|
|
|
|
// conservatively assume it because we don't know for sure.
|
|
|
|
if (!ID->declaresOrInheritsDesignatedInitializers() ||
|
|
|
|
ID->isDesignatedInitializer(Sel)) {
|
2013-12-04 05:11:43 +08:00
|
|
|
isDesignatedInitChain = true;
|
2014-03-18 05:41:40 +08:00
|
|
|
DIFunctionScopeInfo->ObjCWarnForNoDesignatedInitChain = false;
|
2013-12-04 05:11:43 +08:00
|
|
|
}
|
|
|
|
}
|
2013-12-04 05:11:36 +08:00
|
|
|
}
|
|
|
|
}
|
2013-12-04 05:11:43 +08:00
|
|
|
if (!isDesignatedInitChain) {
|
2014-05-26 14:22:03 +08:00
|
|
|
const ObjCMethodDecl *InitMethod = nullptr;
|
2013-12-04 05:11:43 +08:00
|
|
|
bool isDesignated =
|
|
|
|
getCurMethodDecl()->isDesignatedInitializerForTheInterface(&InitMethod);
|
|
|
|
assert(isDesignated && InitMethod);
|
|
|
|
(void)isDesignated;
|
|
|
|
Diag(SelLoc, SuperLoc.isValid() ?
|
|
|
|
diag::warn_objc_designated_init_non_designated_init_call :
|
|
|
|
diag::warn_objc_designated_init_non_super_designated_init_call);
|
|
|
|
Diag(InitMethod->getLocation(),
|
|
|
|
diag::note_objc_designated_init_marked_here);
|
|
|
|
}
|
2013-12-04 05:11:36 +08:00
|
|
|
}
|
|
|
|
|
2014-03-18 05:41:40 +08:00
|
|
|
if (DIFunctionScopeInfo &&
|
|
|
|
DIFunctionScopeInfo->ObjCIsSecondaryInit &&
|
2013-12-04 05:11:49 +08:00
|
|
|
(SuperLoc.isValid() || isSelfExpr(Receiver))) {
|
|
|
|
if (SuperLoc.isValid()) {
|
|
|
|
Diag(SelLoc, diag::warn_objc_secondary_init_super_init_call);
|
|
|
|
} else {
|
2014-03-18 05:41:40 +08:00
|
|
|
DIFunctionScopeInfo->ObjCWarnForNoInitDelegation = false;
|
2013-12-04 05:11:49 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// Check the message arguments.
|
|
|
|
unsigned NumArgs = ArgsIn.size();
|
2012-08-24 07:38:35 +08:00
|
|
|
Expr **Args = ArgsIn.data();
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
QualType ReturnType;
|
2010-11-18 14:31:45 +08:00
|
|
|
ExprValueKind VK = VK_RValue;
|
2010-12-01 09:07:24 +08:00
|
|
|
bool ClassMessage = (ReceiverType->isObjCClassType() ||
|
|
|
|
ReceiverType->isObjCQualifiedClassType());
|
2018-12-21 06:11:11 +08:00
|
|
|
if (CheckMessageArgumentTypes(Receiver, ReceiverType,
|
|
|
|
MultiExprArg(Args, NumArgs), Sel, SelectorLocs,
|
|
|
|
Method, ClassMessage, SuperLoc.isValid(),
|
2014-08-20 07:39:17 +08:00
|
|
|
LBracLoc, RBracLoc, RecRange, ReturnType, VK))
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
return ExprError();
|
2014-01-26 00:55:45 +08:00
|
|
|
|
|
|
|
if (Method && !Method->getReturnType()->isVoidType() &&
|
|
|
|
RequireCompleteType(LBracLoc, Method->getReturnType(),
|
2011-01-11 11:23:19 +08:00
|
|
|
diag::err_illegal_message_expr_incomplete_type))
|
|
|
|
return ExprError();
|
Overhaul the AST representation of Objective-C message send
expressions, to improve source-location information, clarify the
actual receiver of the message, and pave the way for proper C++
support. The ObjCMessageExpr node represents four different kinds of
message sends in a single AST node:
1) Send to a object instance described by an expression (e.g., [x method:5])
2) Send to a class described by the class name (e.g., [NSString method:5])
3) Send to a superclass class (e.g, [super method:5] in class method)
4) Send to a superclass instance (e.g., [super method:5] in instance method)
Previously these four cases where tangled together. Now, they have
more distinct representations. Specific changes:
1) Unchanged; the object instance is represented by an Expr*.
2) Previously stored the ObjCInterfaceDecl* referring to the class
receiving the message. Now stores a TypeSourceInfo* so that we know
how the class was spelled. This both maintains typedef information
and opens the door for more complicated C++ types (e.g., dependent
types). There was an alternative, unused representation of these
sends by naming the class via an IdentifierInfo *. In practice, we
either had an ObjCInterfaceDecl *, from which we would get the
IdentifierInfo *, or we fell into the case below...
3) Previously represented by a class message whose IdentifierInfo *
referred to "super". Sema and CodeGen would use isStr("super") to
determine if they had a send to super. Now represented as a
"class super" send, where we have both the location of the "super"
keyword and the ObjCInterfaceDecl* of the superclass we're
targetting (statically).
4) Previously represented by an instance message whose receiver is a
an ObjCSuperExpr, which Sema and CodeGen would check for via
isa<ObjCSuperExpr>(). Now represented as an "instance super" send,
where we have both the location of the "super" keyword and the
ObjCInterfaceDecl* of the superclass we're targetting
(statically). Note that ObjCSuperExpr only has one remaining use in
the AST, which is for "super.prop" references.
The new representation of ObjCMessageExpr is 2 pointers smaller than
the old one, since it combines more storage. It also eliminates a leak
when we loaded message-send expressions from a precompiled header. The
representation also feels much cleaner to me; comments welcome!
This patch attempts to maintain the same semantics we previously had
with Objective-C message sends. In several places, there are massive
changes that boil down to simply replacing a nested-if structure such
as:
if (message has a receiver expression) {
// instance message
if (isa<ObjCSuperExpr>(...)) {
// send to super
} else {
// send to an object
}
} else {
// class message
if (name->isStr("super")) {
// class send to super
} else {
// send to class
}
}
with a switch
switch (E->getReceiverKind()) {
case ObjCMessageExpr::SuperInstance: ...
case ObjCMessageExpr::Instance: ...
case ObjCMessageExpr::SuperClass: ...
case ObjCMessageExpr::Class:...
}
There are quite a few places (particularly in the checkers) where
send-to-super is effectively ignored. I've placed FIXMEs in most of
them, and attempted to address send-to-super in a reasonable way. This
could use some review.
llvm-svn: 101972
2010-04-21 08:45:42 +08:00
|
|
|
|
2018-07-31 03:24:48 +08:00
|
|
|
// In ARC, forbid the user from sending messages to
|
2011-06-16 07:02:42 +08:00
|
|
|
// retain/release/autorelease/dealloc/retainCount explicitly.
|
2012-03-11 15:00:24 +08:00
|
|
|
if (getLangOpts().ObjCAutoRefCount) {
|
2011-06-16 07:02:42 +08:00
|
|
|
ObjCMethodFamily family =
|
|
|
|
(Method ? Method->getMethodFamily() : Sel.getMethodFamily());
|
|
|
|
switch (family) {
|
|
|
|
case OMF_init:
|
|
|
|
if (Method)
|
|
|
|
checkInitMethod(Method, ReceiverType);
|
2017-12-20 06:06:11 +08:00
|
|
|
break;
|
2011-06-16 07:02:42 +08:00
|
|
|
|
|
|
|
case OMF_None:
|
|
|
|
case OMF_alloc:
|
|
|
|
case OMF_copy:
|
2011-08-29 06:35:17 +08:00
|
|
|
case OMF_finalize:
|
2011-06-16 07:02:42 +08:00
|
|
|
case OMF_mutableCopy:
|
|
|
|
case OMF_new:
|
|
|
|
case OMF_self:
|
2014-08-23 00:57:26 +08:00
|
|
|
case OMF_initialize:
|
2011-06-16 07:02:42 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case OMF_dealloc:
|
|
|
|
case OMF_retain:
|
|
|
|
case OMF_release:
|
|
|
|
case OMF_autorelease:
|
|
|
|
case OMF_retainCount:
|
2013-05-01 08:24:09 +08:00
|
|
|
Diag(SelLoc, diag::err_arc_illegal_explicit_message)
|
|
|
|
<< Sel << RecRange;
|
2011-06-16 07:02:42 +08:00
|
|
|
break;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-07-06 06:38:59 +08:00
|
|
|
case OMF_performSelector:
|
|
|
|
if (Method && NumArgs >= 1) {
|
2017-02-21 01:55:15 +08:00
|
|
|
if (const auto *SelExp =
|
|
|
|
dyn_cast<ObjCSelectorExpr>(Args[0]->IgnoreParens())) {
|
2011-07-06 06:38:59 +08:00
|
|
|
Selector ArgSel = SelExp->getSelector();
|
2018-07-31 03:24:48 +08:00
|
|
|
ObjCMethodDecl *SelMethod =
|
2011-07-06 06:38:59 +08:00
|
|
|
LookupInstanceMethodInGlobalPool(ArgSel,
|
|
|
|
SelExp->getSourceRange());
|
|
|
|
if (!SelMethod)
|
|
|
|
SelMethod =
|
|
|
|
LookupFactoryMethodInGlobalPool(ArgSel,
|
|
|
|
SelExp->getSourceRange());
|
|
|
|
if (SelMethod) {
|
|
|
|
ObjCMethodFamily SelFamily = SelMethod->getMethodFamily();
|
|
|
|
switch (SelFamily) {
|
|
|
|
case OMF_alloc:
|
|
|
|
case OMF_copy:
|
|
|
|
case OMF_mutableCopy:
|
|
|
|
case OMF_new:
|
|
|
|
case OMF_init:
|
|
|
|
// Issue error, unless ns_returns_not_retained.
|
|
|
|
if (!SelMethod->hasAttr<NSReturnsNotRetainedAttr>()) {
|
2018-07-31 03:24:48 +08:00
|
|
|
// selector names a +1 method
|
|
|
|
Diag(SelLoc,
|
2011-07-06 06:38:59 +08:00
|
|
|
diag::err_arc_perform_selector_retains);
|
2012-02-28 06:55:11 +08:00
|
|
|
Diag(SelMethod->getLocation(), diag::note_method_declared_at)
|
|
|
|
<< SelMethod->getDeclName();
|
2011-07-06 06:38:59 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// +0 call. OK. unless ns_returns_retained.
|
|
|
|
if (SelMethod->hasAttr<NSReturnsRetainedAttr>()) {
|
|
|
|
// selector names a +1 method
|
2018-07-31 03:24:48 +08:00
|
|
|
Diag(SelLoc,
|
2011-07-06 06:38:59 +08:00
|
|
|
diag::err_arc_perform_selector_retains);
|
2012-02-28 06:55:11 +08:00
|
|
|
Diag(SelMethod->getLocation(), diag::note_method_declared_at)
|
|
|
|
<< SelMethod->getDeclName();
|
2011-07-06 06:38:59 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// error (may leak).
|
2011-10-03 14:36:17 +08:00
|
|
|
Diag(SelLoc, diag::warn_arc_perform_selector_leaks);
|
2011-07-06 06:38:59 +08:00
|
|
|
Diag(Args[0]->getExprLoc(), diag::note_used_here);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-10 07:10:54 +08:00
|
|
|
DiagnoseCStringFormatDirectiveInObjCAPI(*this, Method, Sel, Args, NumArgs);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// Construct the appropriate ObjCMessageExpr instance.
|
2011-06-16 07:02:42 +08:00
|
|
|
ObjCMessageExpr *Result;
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
if (SuperLoc.isValid())
|
2010-11-18 14:31:45 +08:00
|
|
|
Result = ObjCMessageExpr::Create(Context, ReturnType, VK, LBracLoc,
|
2010-05-22 13:17:18 +08:00
|
|
|
SuperLoc, /*IsInstanceSuper=*/true,
|
2018-07-31 03:24:48 +08:00
|
|
|
ReceiverType, Sel, SelectorLocs, Method,
|
2012-01-12 10:34:39 +08:00
|
|
|
makeArrayRef(Args, NumArgs), RBracLoc,
|
|
|
|
isImplicit);
|
2012-03-07 04:05:56 +08:00
|
|
|
else {
|
2010-11-18 14:31:45 +08:00
|
|
|
Result = ObjCMessageExpr::Create(Context, ReturnType, VK, LBracLoc,
|
2011-10-03 14:36:17 +08:00
|
|
|
Receiver, Sel, SelectorLocs, Method,
|
2012-01-12 10:34:39 +08:00
|
|
|
makeArrayRef(Args, NumArgs), RBracLoc,
|
|
|
|
isImplicit);
|
2012-03-07 04:05:56 +08:00
|
|
|
if (!isImplicit)
|
|
|
|
checkCocoaAPI(*this, Result);
|
|
|
|
}
|
2017-03-06 23:58:34 +08:00
|
|
|
if (Method) {
|
|
|
|
bool IsClassObjectCall = ClassMessage;
|
|
|
|
// 'self' message receivers in class methods should be treated as message
|
|
|
|
// sends to the class object in order for the semantic checks to be
|
|
|
|
// performed correctly. Messages to 'super' already count as class messages,
|
|
|
|
// so they don't need to be handled here.
|
|
|
|
if (Receiver && isSelfExpr(Receiver)) {
|
|
|
|
if (const auto *OPT = ReceiverType->getAs<ObjCObjectPointerType>()) {
|
|
|
|
if (OPT->getObjectType()->isObjCClass()) {
|
|
|
|
if (const auto *CurMeth = getCurMethodDecl()) {
|
|
|
|
IsClassObjectCall = true;
|
|
|
|
ReceiverType =
|
|
|
|
Context.getObjCInterfaceType(CurMeth->getClassInterface());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
checkFoundationAPI(*this, SelLoc, Method, makeArrayRef(Args, NumArgs),
|
|
|
|
ReceiverType, IsClassObjectCall);
|
|
|
|
}
|
2011-06-16 07:02:42 +08:00
|
|
|
|
2012-03-11 15:00:24 +08:00
|
|
|
if (getLangOpts().ObjCAutoRefCount) {
|
2011-06-16 07:02:42 +08:00
|
|
|
// In ARC, annotate delegate init calls.
|
|
|
|
if (Result->getMethodFamily() == OMF_init &&
|
2011-09-28 00:10:05 +08:00
|
|
|
(SuperLoc.isValid() || isSelfExpr(Receiver))) {
|
2011-06-16 07:02:42 +08:00
|
|
|
// Only consider init calls *directly* in init implementations,
|
|
|
|
// not within blocks.
|
|
|
|
ObjCMethodDecl *method = dyn_cast<ObjCMethodDecl>(CurContext);
|
|
|
|
if (method && method->getMethodFamily() == OMF_init) {
|
|
|
|
// The implicit assignment to self means we also don't want to
|
|
|
|
// consume the result.
|
|
|
|
Result->setDelegateInitCall(true);
|
2014-05-29 22:05:12 +08:00
|
|
|
return Result;
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// In ARC, check for message sends which are likely to introduce
|
|
|
|
// retain cycles.
|
|
|
|
checkRetainCycles(Result);
|
2017-03-30 01:55:11 +08:00
|
|
|
}
|
-Warc-repeated-use-of-weak: Check messages to property accessors as well.
Previously, [foo weakProp] was not being treated the same as foo.weakProp.
Now, for every explicit message send, we check if it's a property access,
and if so, if the property is weak. Then for every assignment of a
message, we have to do the same thing again.
This is a potentially expensive increase because determining whether a
method is a property accessor requires searching through the methods it
overrides. However, without it -Warc-repeated-use-of-weak will miss cases
from people who prefer not to use dot syntax. If this turns out to be
too expensive, we can try caching the result somewhere, or even lose
precision by not checking superclass methods. The warning is off-by-default,
though.
<rdar://problem/12407765>
llvm-svn: 165718
2012-10-12 00:06:21 +08:00
|
|
|
|
2017-03-30 01:55:11 +08:00
|
|
|
if (getLangOpts().ObjCWeak) {
|
-Warc-repeated-use-of-weak: Check messages to property accessors as well.
Previously, [foo weakProp] was not being treated the same as foo.weakProp.
Now, for every explicit message send, we check if it's a property access,
and if so, if the property is weak. Then for every assignment of a
message, we have to do the same thing again.
This is a potentially expensive increase because determining whether a
method is a property accessor requires searching through the methods it
overrides. However, without it -Warc-repeated-use-of-weak will miss cases
from people who prefer not to use dot syntax. If this turns out to be
too expensive, we can try caching the result somewhere, or even lose
precision by not checking superclass methods. The warning is off-by-default,
though.
<rdar://problem/12407765>
llvm-svn: 165718
2012-10-12 00:06:21 +08:00
|
|
|
if (!isImplicit && Method) {
|
|
|
|
if (const ObjCPropertyDecl *Prop = Method->findPropertyDecl()) {
|
|
|
|
bool IsWeak =
|
|
|
|
Prop->getPropertyAttributes() & ObjCPropertyDecl::OBJC_PR_weak;
|
|
|
|
if (!IsWeak && Sel.isUnarySelector())
|
|
|
|
IsWeak = ReturnType.getObjCLifetime() & Qualifiers::OCL_Weak;
|
2019-01-11 04:12:16 +08:00
|
|
|
if (IsWeak && !isUnevaluatedContext() &&
|
2014-06-16 07:30:39 +08:00
|
|
|
!Diags.isIgnored(diag::warn_arc_repeated_use_of_weak, LBracLoc))
|
|
|
|
getCurFunction()->recordUseOfWeak(Result, Prop);
|
-Warc-repeated-use-of-weak: Check messages to property accessors as well.
Previously, [foo weakProp] was not being treated the same as foo.weakProp.
Now, for every explicit message send, we check if it's a property access,
and if so, if the property is weak. Then for every assignment of a
message, we have to do the same thing again.
This is a potentially expensive increase because determining whether a
method is a property accessor requires searching through the methods it
overrides. However, without it -Warc-repeated-use-of-weak will miss cases
from people who prefer not to use dot syntax. If this turns out to be
too expensive, we can try caching the result somewhere, or even lose
precision by not checking superclass methods. The warning is off-by-default,
though.
<rdar://problem/12407765>
llvm-svn: 165718
2012-10-12 00:06:21 +08:00
|
|
|
}
|
|
|
|
}
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2015-03-05 01:55:52 +08:00
|
|
|
|
|
|
|
CheckObjCCircularContainer(Result);
|
|
|
|
|
2010-05-22 13:17:18 +08:00
|
|
|
return MaybeBindToTemporary(Result);
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
}
|
|
|
|
|
2013-01-23 02:35:43 +08:00
|
|
|
static void RemoveSelectorFromWarningCache(Sema &S, Expr* Arg) {
|
|
|
|
if (ObjCSelectorExpr *OSE =
|
|
|
|
dyn_cast<ObjCSelectorExpr>(Arg->IgnoreParenCasts())) {
|
|
|
|
Selector Sel = OSE->getSelector();
|
|
|
|
SourceLocation Loc = OSE->getAtLoc();
|
2015-03-27 08:55:05 +08:00
|
|
|
auto Pos = S.ReferencedSelectors.find(Sel);
|
2013-01-23 02:35:43 +08:00
|
|
|
if (Pos != S.ReferencedSelectors.end() && Pos->second == Loc)
|
|
|
|
S.ReferencedSelectors.erase(Pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
// ActOnInstanceMessage - used for both unary and keyword messages.
|
|
|
|
// ArgExprs is optional - if it is present, the number of expressions
|
|
|
|
// is obtained from Sel.getNumArgs().
|
2010-08-24 14:29:42 +08:00
|
|
|
ExprResult Sema::ActOnInstanceMessage(Scope *S,
|
2018-07-31 03:24:48 +08:00
|
|
|
Expr *Receiver,
|
2010-08-24 14:29:42 +08:00
|
|
|
Selector Sel,
|
|
|
|
SourceLocation LBracLoc,
|
2011-10-03 14:36:17 +08:00
|
|
|
ArrayRef<SourceLocation> SelectorLocs,
|
2010-08-24 14:29:42 +08:00
|
|
|
SourceLocation RBracLoc,
|
|
|
|
MultiExprArg Args) {
|
Rework the Parser-Sema interaction for Objective-C message
sends. Major changes include:
- Expanded the interface from two actions (ActOnInstanceMessage,
ActOnClassMessage), where ActOnClassMessage also handled sends to
"super" by checking whether the identifier was "super", to three
actions (ActOnInstanceMessage, ActOnClassMessage,
ActOnSuperMessage). Code completion has the same changes.
- The parser now resolves the type to which we are sending a class
message, so ActOnClassMessage now accepts a TypeTy* (rather than
an IdentifierInfo *). This opens the door to more interesting
types (for Objective-C++ support).
- Split ActOnInstanceMessage and ActOnClassMessage into parser
action functions (with their original names) and semantic
functions (BuildInstanceMessage and BuildClassMessage,
respectively). At present, this split is onyl used by
ActOnSuperMessage, which decides which kind of super message it
has and forwards to the appropriate Build*Message. In the future,
Build*Message will be used by template instantiation.
- Use getObjCMessageKind() within the disambiguation of Objective-C
message sends vs. array designators.
Two notes about substandard bits in this patch:
- There is some redundancy in the code in ParseObjCMessageExpr and
ParseInitializerWithPotentialDesignator; this will be addressed
shortly by centralizing the mapping from identifiers to type names
for the message receiver.
- There is some #if 0'd code that won't likely ever be used---it
handles the use of 'super' in methods whose class does not have a
superclass---but could be used to model GCC's behavior more
closely. This code will die in my next check-in, but I want it in
Subversion.
llvm-svn: 102021
2010-04-22 03:57:20 +08:00
|
|
|
if (!Receiver)
|
|
|
|
return ExprError();
|
2013-02-16 02:34:15 +08:00
|
|
|
|
|
|
|
// A ParenListExpr can show up while doing error recovery with invalid code.
|
|
|
|
if (isa<ParenListExpr>(Receiver)) {
|
|
|
|
ExprResult Result = MaybeConvertParenListExprToParenExpr(S, Receiver);
|
|
|
|
if (Result.isInvalid()) return ExprError();
|
2014-05-29 18:55:11 +08:00
|
|
|
Receiver = Result.get();
|
2013-02-16 02:34:15 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2013-01-23 03:05:17 +08:00
|
|
|
if (RespondsToSelectorSel.isNull()) {
|
|
|
|
IdentifierInfo *SelectorId = &Context.Idents.get("respondsToSelector");
|
|
|
|
RespondsToSelectorSel = Context.Selectors.getUnarySelector(SelectorId);
|
|
|
|
}
|
|
|
|
if (Sel == RespondsToSelectorSel)
|
2013-01-23 02:35:43 +08:00
|
|
|
RemoveSelectorFromWarningCache(*this, Args[0]);
|
2014-05-26 14:22:03 +08:00
|
|
|
|
2010-08-24 07:25:46 +08:00
|
|
|
return BuildInstanceMessage(Receiver, Receiver->getType(),
|
2014-05-26 14:22:03 +08:00
|
|
|
/*SuperLoc=*/SourceLocation(), Sel,
|
|
|
|
/*Method=*/nullptr, LBracLoc, SelectorLocs,
|
|
|
|
RBracLoc, Args);
|
2008-01-05 06:32:30 +08:00
|
|
|
}
|
2008-04-07 13:30:13 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
enum ARCConversionTypeClass {
|
2011-10-01 09:01:08 +08:00
|
|
|
/// int, void, struct A
|
2011-06-16 07:02:42 +08:00
|
|
|
ACTC_none,
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
/// id, void (^)()
|
2011-06-16 07:02:42 +08:00
|
|
|
ACTC_retainable,
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
/// id*, id***, void (^*)(),
|
|
|
|
ACTC_indirectRetainable,
|
|
|
|
|
|
|
|
/// void* might be a normal C type, or it might a CF type.
|
|
|
|
ACTC_voidPtr,
|
|
|
|
|
|
|
|
/// struct A*
|
|
|
|
ACTC_coreFoundation
|
2011-06-16 07:02:42 +08:00
|
|
|
};
|
2016-02-13 06:53:10 +08:00
|
|
|
|
2011-10-01 09:01:08 +08:00
|
|
|
static bool isAnyRetainable(ARCConversionTypeClass ACTC) {
|
|
|
|
return (ACTC == ACTC_retainable ||
|
|
|
|
ACTC == ACTC_coreFoundation ||
|
|
|
|
ACTC == ACTC_voidPtr);
|
|
|
|
}
|
2016-02-13 06:53:10 +08:00
|
|
|
|
2011-10-01 09:01:08 +08:00
|
|
|
static bool isAnyCLike(ARCConversionTypeClass ACTC) {
|
|
|
|
return ACTC == ACTC_none ||
|
|
|
|
ACTC == ACTC_voidPtr ||
|
|
|
|
ACTC == ACTC_coreFoundation;
|
|
|
|
}
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
static ARCConversionTypeClass classifyTypeForARCConversion(QualType type) {
|
2011-10-01 09:01:08 +08:00
|
|
|
bool isIndirect = false;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
// Ignore an outermost reference type.
|
2011-10-01 09:01:08 +08:00
|
|
|
if (const ReferenceType *ref = type->getAs<ReferenceType>()) {
|
2011-06-16 07:02:42 +08:00
|
|
|
type = ref->getPointeeType();
|
2011-10-01 09:01:08 +08:00
|
|
|
isIndirect = true;
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
// Drill through pointers and arrays recursively.
|
|
|
|
while (true) {
|
|
|
|
if (const PointerType *ptr = type->getAs<PointerType>()) {
|
|
|
|
type = ptr->getPointeeType();
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
// The first level of pointer may be the innermost pointer on a CF type.
|
|
|
|
if (!isIndirect) {
|
|
|
|
if (type->isVoidType()) return ACTC_voidPtr;
|
|
|
|
if (type->isRecordType()) return ACTC_coreFoundation;
|
|
|
|
}
|
2011-06-16 07:02:42 +08:00
|
|
|
} else if (const ArrayType *array = type->getAsArrayTypeUnsafe()) {
|
|
|
|
type = QualType(array->getElementType()->getBaseElementTypeUnsafe(), 0);
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
2011-10-01 09:01:08 +08:00
|
|
|
isIndirect = true;
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-10-01 09:01:08 +08:00
|
|
|
if (isIndirect) {
|
|
|
|
if (type->isObjCARCBridgableType())
|
|
|
|
return ACTC_indirectRetainable;
|
|
|
|
return ACTC_none;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type->isObjCARCBridgableType())
|
|
|
|
return ACTC_retainable;
|
|
|
|
|
|
|
|
return ACTC_none;
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
2011-10-01 09:01:08 +08:00
|
|
|
/// A result from the cast checker.
|
|
|
|
enum ACCResult {
|
|
|
|
/// Cannot be casted.
|
|
|
|
ACC_invalid,
|
|
|
|
|
|
|
|
/// Can be safely retained or not retained.
|
|
|
|
ACC_bottom,
|
|
|
|
|
|
|
|
/// Can be casted at +0.
|
|
|
|
ACC_plusZero,
|
|
|
|
|
|
|
|
/// Can be casted at +1.
|
|
|
|
ACC_plusOne
|
|
|
|
};
|
|
|
|
ACCResult merge(ACCResult left, ACCResult right) {
|
|
|
|
if (left == right) return left;
|
|
|
|
if (left == ACC_bottom) return right;
|
|
|
|
if (right == ACC_bottom) return left;
|
|
|
|
return ACC_invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A checker which white-lists certain expressions whose conversion
|
|
|
|
/// to or from retainable type would otherwise be forbidden in ARC.
|
|
|
|
class ARCCastChecker : public StmtVisitor<ARCCastChecker, ACCResult> {
|
|
|
|
typedef StmtVisitor<ARCCastChecker, ACCResult> super;
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
ASTContext &Context;
|
2011-10-01 09:01:08 +08:00
|
|
|
ARCConversionTypeClass SourceClass;
|
|
|
|
ARCConversionTypeClass TargetClass;
|
2012-07-28 06:37:07 +08:00
|
|
|
bool Diagnose;
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
static bool isCFType(QualType type) {
|
|
|
|
// Someday this can use ns_bridged. For now, it has to do this.
|
|
|
|
return type->isCARCBridgableType();
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
public:
|
|
|
|
ARCCastChecker(ASTContext &Context, ARCConversionTypeClass source,
|
2012-07-28 06:37:07 +08:00
|
|
|
ARCConversionTypeClass target, bool diagnose)
|
|
|
|
: Context(Context), SourceClass(source), TargetClass(target),
|
|
|
|
Diagnose(diagnose) {}
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
using super::Visit;
|
|
|
|
ACCResult Visit(Expr *e) {
|
|
|
|
return super::Visit(e->IgnoreParens());
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
ACCResult VisitStmt(Stmt *s) {
|
|
|
|
return ACC_invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Null pointer constants can be casted however you please.
|
|
|
|
ACCResult VisitExpr(Expr *e) {
|
|
|
|
if (e->isNullPointerConstant(Context, Expr::NPC_ValueDependentIsNotNull))
|
|
|
|
return ACC_bottom;
|
|
|
|
return ACC_invalid;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Objective-C string literals can be safely casted.
|
|
|
|
ACCResult VisitObjCStringLiteral(ObjCStringLiteral *e) {
|
|
|
|
// If we're casting to any retainable type, go ahead. Global
|
|
|
|
// strings are immune to retains, so this is bottom.
|
|
|
|
if (isAnyRetainable(TargetClass)) return ACC_bottom;
|
|
|
|
|
|
|
|
return ACC_invalid;
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-10-01 09:01:08 +08:00
|
|
|
/// Look through certain implicit and explicit casts.
|
|
|
|
ACCResult VisitCastExpr(CastExpr *e) {
|
2011-06-16 07:02:42 +08:00
|
|
|
switch (e->getCastKind()) {
|
|
|
|
case CK_NullToPointer:
|
2011-10-01 09:01:08 +08:00
|
|
|
return ACC_bottom;
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
case CK_NoOp:
|
|
|
|
case CK_LValueToRValue:
|
|
|
|
case CK_BitCast:
|
2011-09-09 13:25:32 +08:00
|
|
|
case CK_CPointerToObjCPointerCast:
|
|
|
|
case CK_BlockPointerToObjCPointerCast:
|
2011-06-16 07:02:42 +08:00
|
|
|
case CK_AnyPointerToBlockPointerCast:
|
|
|
|
return Visit(e->getSubExpr());
|
2011-10-01 09:01:08 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
default:
|
2011-10-01 09:01:08 +08:00
|
|
|
return ACC_invalid;
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
|
|
|
}
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
/// Look through unary extension.
|
|
|
|
ACCResult VisitUnaryExtension(UnaryOperator *e) {
|
2011-06-16 07:02:42 +08:00
|
|
|
return Visit(e->getSubExpr());
|
|
|
|
}
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
/// Ignore the LHS of a comma operator.
|
|
|
|
ACCResult VisitBinComma(BinaryOperator *e) {
|
2011-06-16 07:02:42 +08:00
|
|
|
return Visit(e->getRHS());
|
|
|
|
}
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
/// Conditional operators are okay if both sides are okay.
|
|
|
|
ACCResult VisitConditionalOperator(ConditionalOperator *e) {
|
|
|
|
ACCResult left = Visit(e->getTrueExpr());
|
|
|
|
if (left == ACC_invalid) return ACC_invalid;
|
|
|
|
return merge(left, Visit(e->getFalseExpr()));
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2011-10-01 09:01:08 +08:00
|
|
|
|
2011-11-06 17:01:30 +08:00
|
|
|
/// Look through pseudo-objects.
|
|
|
|
ACCResult VisitPseudoObjectExpr(PseudoObjectExpr *e) {
|
|
|
|
// If we're getting here, we should always have a result.
|
|
|
|
return Visit(e->getResultExpr());
|
|
|
|
}
|
|
|
|
|
2011-10-01 09:01:08 +08:00
|
|
|
/// Statement expressions are okay if their result expression is okay.
|
|
|
|
ACCResult VisitStmtExpr(StmtExpr *e) {
|
2011-06-16 07:02:42 +08:00
|
|
|
return Visit(e->getSubStmt()->body_back());
|
|
|
|
}
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
/// Some declaration references are okay.
|
|
|
|
ACCResult VisitDeclRefExpr(DeclRefExpr *e) {
|
|
|
|
VarDecl *var = dyn_cast<VarDecl>(e->getDecl());
|
2015-02-26 04:09:06 +08:00
|
|
|
// References to global constants are okay.
|
2011-10-01 09:01:08 +08:00
|
|
|
if (isAnyRetainable(TargetClass) &&
|
|
|
|
isAnyRetainable(SourceClass) &&
|
|
|
|
var &&
|
2017-04-12 06:01:33 +08:00
|
|
|
!var->hasDefinition(Context) &&
|
2015-02-26 04:09:06 +08:00
|
|
|
var->getType().isConstQualified()) {
|
|
|
|
|
|
|
|
// In system headers, they can also be assumed to be immune to retains.
|
|
|
|
// These are things like 'kCFStringTransformToLatin'.
|
|
|
|
if (Context.getSourceManager().isInSystemHeader(var->getLocation()))
|
|
|
|
return ACC_bottom;
|
|
|
|
|
|
|
|
return ACC_plusZero;
|
2011-10-01 09:01:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Nothing else.
|
|
|
|
return ACC_invalid;
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
|
|
|
|
2011-10-01 09:01:08 +08:00
|
|
|
/// Some calls are okay.
|
|
|
|
ACCResult VisitCallExpr(CallExpr *e) {
|
|
|
|
if (FunctionDecl *fn = e->getDirectCallee())
|
|
|
|
if (ACCResult result = checkCallToFunction(fn))
|
|
|
|
return result;
|
|
|
|
|
|
|
|
return super::VisitCallExpr(e);
|
2011-06-22 03:42:38 +08:00
|
|
|
}
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
ACCResult checkCallToFunction(FunctionDecl *fn) {
|
|
|
|
// Require a CF*Ref return type.
|
2014-01-26 00:55:45 +08:00
|
|
|
if (!isCFType(fn->getReturnType()))
|
2011-10-01 09:01:08 +08:00
|
|
|
return ACC_invalid;
|
|
|
|
|
|
|
|
if (!isAnyRetainable(TargetClass))
|
|
|
|
return ACC_invalid;
|
|
|
|
|
|
|
|
// Honor an explicit 'not retained' attribute.
|
|
|
|
if (fn->hasAttr<CFReturnsNotRetainedAttr>())
|
|
|
|
return ACC_plusZero;
|
|
|
|
|
|
|
|
// Honor an explicit 'retained' attribute, except that for
|
|
|
|
// now we're not going to permit implicit handling of +1 results,
|
|
|
|
// because it's a bit frightening.
|
|
|
|
if (fn->hasAttr<CFReturnsRetainedAttr>())
|
2012-07-28 07:55:46 +08:00
|
|
|
return Diagnose ? ACC_plusOne
|
|
|
|
: ACC_invalid; // ACC_plusOne if we start accepting this
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
// Recognize this specific builtin function, which is used by CFSTR.
|
|
|
|
unsigned builtinID = fn->getBuiltinID();
|
|
|
|
if (builtinID == Builtin::BI__builtin___CFStringMakeConstantString)
|
|
|
|
return ACC_bottom;
|
|
|
|
|
2012-07-28 07:55:46 +08:00
|
|
|
// Otherwise, don't do anything implicit with an unaudited function.
|
|
|
|
if (!fn->hasAttr<CFAuditedTransferAttr>())
|
|
|
|
return ACC_invalid;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-07-28 06:37:07 +08:00
|
|
|
// Otherwise, it's +0 unless it follows the create convention.
|
|
|
|
if (ento::coreFoundation::followsCreateRule(fn))
|
2018-07-31 03:24:48 +08:00
|
|
|
return Diagnose ? ACC_plusOne
|
2012-07-28 06:37:07 +08:00
|
|
|
: ACC_invalid; // ACC_plusOne if we start accepting this
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
return ACC_plusZero;
|
|
|
|
}
|
|
|
|
|
|
|
|
ACCResult VisitObjCMessageExpr(ObjCMessageExpr *e) {
|
|
|
|
return checkCallToMethod(e->getMethodDecl());
|
|
|
|
}
|
|
|
|
|
|
|
|
ACCResult VisitObjCPropertyRefExpr(ObjCPropertyRefExpr *e) {
|
|
|
|
ObjCMethodDecl *method;
|
|
|
|
if (e->isExplicitProperty())
|
|
|
|
method = e->getExplicitProperty()->getGetterMethodDecl();
|
|
|
|
else
|
|
|
|
method = e->getImplicitPropertyGetter();
|
|
|
|
return checkCallToMethod(method);
|
|
|
|
}
|
|
|
|
|
|
|
|
ACCResult checkCallToMethod(ObjCMethodDecl *method) {
|
|
|
|
if (!method) return ACC_invalid;
|
|
|
|
|
|
|
|
// Check for message sends to functions returning CF types. We
|
|
|
|
// just obey the Cocoa conventions with these, even though the
|
|
|
|
// return type is CF.
|
2014-01-26 00:55:45 +08:00
|
|
|
if (!isAnyRetainable(TargetClass) || !isCFType(method->getReturnType()))
|
2011-10-01 09:01:08 +08:00
|
|
|
return ACC_invalid;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-10-01 09:01:08 +08:00
|
|
|
// If the method is explicitly marked not-retained, it's +0.
|
|
|
|
if (method->hasAttr<CFReturnsNotRetainedAttr>())
|
|
|
|
return ACC_plusZero;
|
|
|
|
|
|
|
|
// If the method is explicitly marked as returning retained, or its
|
|
|
|
// selector follows a +1 Cocoa convention, treat it as +1.
|
|
|
|
if (method->hasAttr<CFReturnsRetainedAttr>())
|
|
|
|
return ACC_plusOne;
|
|
|
|
|
|
|
|
switch (method->getSelector().getMethodFamily()) {
|
|
|
|
case OMF_alloc:
|
|
|
|
case OMF_copy:
|
|
|
|
case OMF_mutableCopy:
|
|
|
|
case OMF_new:
|
|
|
|
return ACC_plusOne;
|
|
|
|
|
|
|
|
default:
|
|
|
|
// Otherwise, treat it as +0.
|
|
|
|
return ACC_plusZero;
|
2011-06-22 03:42:38 +08:00
|
|
|
}
|
2011-06-22 01:38:29 +08:00
|
|
|
}
|
2011-10-01 09:01:08 +08:00
|
|
|
};
|
2016-02-13 06:53:10 +08:00
|
|
|
} // end anonymous namespace
|
2011-06-21 04:54:42 +08:00
|
|
|
|
2012-06-01 08:10:47 +08:00
|
|
|
bool Sema::isKnownName(StringRef name) {
|
|
|
|
if (name.empty())
|
|
|
|
return false;
|
|
|
|
LookupResult R(*this, &Context.Idents.get(name), SourceLocation(),
|
2012-02-02 06:56:20 +08:00
|
|
|
Sema::LookupOrdinaryName);
|
2012-06-01 08:10:47 +08:00
|
|
|
return LookupName(R, TUScope, false);
|
2012-02-02 06:56:20 +08:00
|
|
|
}
|
|
|
|
|
2012-02-17 01:31:07 +08:00
|
|
|
static void addFixitForObjCARCConversion(Sema &S,
|
|
|
|
DiagnosticBuilder &DiagB,
|
|
|
|
Sema::CheckedConversionKind CCK,
|
|
|
|
SourceLocation afterLParen,
|
|
|
|
QualType castType,
|
|
|
|
Expr *castExpr,
|
2013-02-23 06:02:53 +08:00
|
|
|
Expr *realCast,
|
2012-02-17 01:31:07 +08:00
|
|
|
const char *bridgeKeyword,
|
|
|
|
const char *CFBridgeName) {
|
|
|
|
// We handle C-style and implicit casts here.
|
|
|
|
switch (CCK) {
|
|
|
|
case Sema::CCK_ImplicitConversion:
|
2018-06-28 04:30:34 +08:00
|
|
|
case Sema::CCK_ForBuiltinOverloadedOp:
|
2012-02-17 01:31:07 +08:00
|
|
|
case Sema::CCK_CStyleCast:
|
2013-02-23 06:02:53 +08:00
|
|
|
case Sema::CCK_OtherCast:
|
2012-02-17 01:31:07 +08:00
|
|
|
break;
|
|
|
|
case Sema::CCK_FunctionalCast:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CFBridgeName) {
|
2013-02-23 06:02:53 +08:00
|
|
|
if (CCK == Sema::CCK_OtherCast) {
|
|
|
|
if (const CXXNamedCastExpr *NCE = dyn_cast<CXXNamedCastExpr>(realCast)) {
|
|
|
|
SourceRange range(NCE->getOperatorLoc(),
|
|
|
|
NCE->getAngleBrackets().getEnd());
|
|
|
|
SmallString<32> BridgeCall;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2013-02-23 06:02:53 +08:00
|
|
|
SourceManager &SM = S.getSourceManager();
|
|
|
|
char PrevChar = *SM.getCharacterData(range.getBegin().getLocWithOffset(-1));
|
|
|
|
if (Lexer::isIdentifierBodyChar(PrevChar, S.getLangOpts()))
|
|
|
|
BridgeCall += ' ';
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2013-02-23 06:02:53 +08:00
|
|
|
BridgeCall += CFBridgeName;
|
|
|
|
DiagB.AddFixItHint(FixItHint::CreateReplacement(range, BridgeCall));
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2012-02-17 01:31:07 +08:00
|
|
|
Expr *castedE = castExpr;
|
|
|
|
if (CStyleCastExpr *CCE = dyn_cast<CStyleCastExpr>(castedE))
|
|
|
|
castedE = CCE->getSubExpr();
|
|
|
|
castedE = castedE->IgnoreImpCasts();
|
|
|
|
SourceRange range = castedE->getSourceRange();
|
2012-06-07 09:10:31 +08:00
|
|
|
|
|
|
|
SmallString<32> BridgeCall;
|
|
|
|
|
|
|
|
SourceManager &SM = S.getSourceManager();
|
|
|
|
char PrevChar = *SM.getCharacterData(range.getBegin().getLocWithOffset(-1));
|
|
|
|
if (Lexer::isIdentifierBodyChar(PrevChar, S.getLangOpts()))
|
|
|
|
BridgeCall += ' ';
|
|
|
|
|
|
|
|
BridgeCall += CFBridgeName;
|
|
|
|
|
2012-02-17 01:31:07 +08:00
|
|
|
if (isa<ParenExpr>(castedE)) {
|
|
|
|
DiagB.AddFixItHint(FixItHint::CreateInsertion(range.getBegin(),
|
2012-06-07 09:10:31 +08:00
|
|
|
BridgeCall));
|
2012-02-17 01:31:07 +08:00
|
|
|
} else {
|
2012-06-07 09:10:31 +08:00
|
|
|
BridgeCall += '(';
|
2012-02-17 01:31:07 +08:00
|
|
|
DiagB.AddFixItHint(FixItHint::CreateInsertion(range.getBegin(),
|
2012-06-07 09:10:31 +08:00
|
|
|
BridgeCall));
|
2012-02-17 01:31:07 +08:00
|
|
|
DiagB.AddFixItHint(FixItHint::CreateInsertion(
|
2015-11-15 10:31:46 +08:00
|
|
|
S.getLocForEndOfToken(range.getEnd()),
|
2012-02-17 01:31:07 +08:00
|
|
|
")"));
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CCK == Sema::CCK_CStyleCast) {
|
|
|
|
DiagB.AddFixItHint(FixItHint::CreateInsertion(afterLParen, bridgeKeyword));
|
2013-02-23 06:02:53 +08:00
|
|
|
} else if (CCK == Sema::CCK_OtherCast) {
|
|
|
|
if (const CXXNamedCastExpr *NCE = dyn_cast<CXXNamedCastExpr>(realCast)) {
|
|
|
|
std::string castCode = "(";
|
|
|
|
castCode += bridgeKeyword;
|
|
|
|
castCode += castType.getAsString();
|
|
|
|
castCode += ")";
|
|
|
|
SourceRange Range(NCE->getOperatorLoc(),
|
|
|
|
NCE->getAngleBrackets().getEnd());
|
|
|
|
DiagB.AddFixItHint(FixItHint::CreateReplacement(Range, castCode));
|
|
|
|
}
|
2012-02-17 01:31:07 +08:00
|
|
|
} else {
|
|
|
|
std::string castCode = "(";
|
|
|
|
castCode += bridgeKeyword;
|
|
|
|
castCode += castType.getAsString();
|
|
|
|
castCode += ")";
|
|
|
|
Expr *castedE = castExpr->IgnoreImpCasts();
|
|
|
|
SourceRange range = castedE->getSourceRange();
|
|
|
|
if (isa<ParenExpr>(castedE)) {
|
|
|
|
DiagB.AddFixItHint(FixItHint::CreateInsertion(range.getBegin(),
|
|
|
|
castCode));
|
|
|
|
} else {
|
|
|
|
castCode += "(";
|
|
|
|
DiagB.AddFixItHint(FixItHint::CreateInsertion(range.getBegin(),
|
|
|
|
castCode));
|
|
|
|
DiagB.AddFixItHint(FixItHint::CreateInsertion(
|
2015-11-15 10:31:46 +08:00
|
|
|
S.getLocForEndOfToken(range.getEnd()),
|
2012-02-17 01:31:07 +08:00
|
|
|
")"));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-10 06:04:26 +08:00
|
|
|
template <typename T>
|
|
|
|
static inline T *getObjCBridgeAttr(const TypedefType *TD) {
|
|
|
|
TypedefNameDecl *TDNDecl = TD->getDecl();
|
|
|
|
QualType QT = TDNDecl->getUnderlyingType();
|
|
|
|
if (QT->isPointerType()) {
|
|
|
|
QT = QT->getPointeeType();
|
|
|
|
if (const RecordType *RT = QT->getAs<RecordType>())
|
2014-06-12 03:10:46 +08:00
|
|
|
if (RecordDecl *RD = RT->getDecl()->getMostRecentDecl())
|
2013-12-19 21:20:36 +08:00
|
|
|
return RD->getAttr<T>();
|
2013-12-10 06:04:26 +08:00
|
|
|
}
|
2014-05-26 14:22:03 +08:00
|
|
|
return nullptr;
|
2013-12-10 06:04:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static ObjCBridgeRelatedAttr *ObjCBridgeRelatedAttrFromType(QualType T,
|
|
|
|
TypedefNameDecl *&TDNDecl) {
|
|
|
|
while (const TypedefType *TD = dyn_cast<TypedefType>(T.getTypePtr())) {
|
|
|
|
TDNDecl = TD->getDecl();
|
|
|
|
if (ObjCBridgeRelatedAttr *ObjCBAttr =
|
|
|
|
getObjCBridgeAttr<ObjCBridgeRelatedAttr>(TD))
|
|
|
|
return ObjCBAttr;
|
|
|
|
T = TDNDecl->getUnderlyingType();
|
|
|
|
}
|
2014-05-26 14:22:03 +08:00
|
|
|
return nullptr;
|
2013-12-10 06:04:26 +08:00
|
|
|
}
|
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
static void
|
|
|
|
diagnoseObjCARCConversion(Sema &S, SourceRange castRange,
|
|
|
|
QualType castType, ARCConversionTypeClass castACTC,
|
2013-02-23 06:02:53 +08:00
|
|
|
Expr *castExpr, Expr *realCast,
|
|
|
|
ARCConversionTypeClass exprACTC,
|
2011-10-18 02:40:02 +08:00
|
|
|
Sema::CheckedConversionKind CCK) {
|
|
|
|
SourceLocation loc =
|
|
|
|
(castRange.isValid() ? castRange.getBegin() : castExpr->getExprLoc());
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
if (S.makeUnavailableInSystemHeader(loc,
|
2015-10-28 13:03:19 +08:00
|
|
|
UnavailableAttr::IR_ARCForbiddenConversion))
|
2011-10-18 02:40:02 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
QualType castExprType = castExpr->getType();
|
2016-02-13 09:41:41 +08:00
|
|
|
// Defer emitting a diagnostic for bridge-related casts; that will be
|
|
|
|
// handled by CheckObjCBridgeRelatedConversions.
|
2014-05-26 14:22:03 +08:00
|
|
|
TypedefNameDecl *TDNDecl = nullptr;
|
2013-12-10 06:04:26 +08:00
|
|
|
if ((castACTC == ACTC_coreFoundation && exprACTC == ACTC_retainable &&
|
|
|
|
ObjCBridgeRelatedAttrFromType(castType, TDNDecl)) ||
|
|
|
|
(exprACTC == ACTC_coreFoundation && castACTC == ACTC_retainable &&
|
2014-06-19 07:22:38 +08:00
|
|
|
ObjCBridgeRelatedAttrFromType(castExprType, TDNDecl)))
|
2013-12-10 06:04:26 +08:00
|
|
|
return;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
unsigned srcKind = 0;
|
|
|
|
switch (exprACTC) {
|
|
|
|
case ACTC_none:
|
|
|
|
case ACTC_coreFoundation:
|
|
|
|
case ACTC_voidPtr:
|
|
|
|
srcKind = (castExprType->isPointerType() ? 1 : 0);
|
|
|
|
break;
|
|
|
|
case ACTC_retainable:
|
|
|
|
srcKind = (castExprType->isBlockPointerType() ? 2 : 3);
|
|
|
|
break;
|
|
|
|
case ACTC_indirectRetainable:
|
|
|
|
srcKind = 4;
|
|
|
|
break;
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
// Check whether this could be fixed with a bridge cast.
|
2015-11-15 10:31:46 +08:00
|
|
|
SourceLocation afterLParen = S.getLocForEndOfToken(castRange.getBegin());
|
2011-10-18 02:40:02 +08:00
|
|
|
SourceLocation noteLoc = afterLParen.isValid() ? afterLParen : loc;
|
|
|
|
|
2018-06-28 04:30:34 +08:00
|
|
|
unsigned convKindForDiag = Sema::isCast(CCK) ? 0 : 1;
|
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
// Bridge from an ARC type to a CF type.
|
|
|
|
if (castACTC == ACTC_retainable && isAnyRetainable(exprACTC)) {
|
2012-02-02 06:56:20 +08:00
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
S.Diag(loc, diag::err_arc_cast_requires_bridge)
|
2018-06-28 04:30:34 +08:00
|
|
|
<< convKindForDiag
|
2011-10-18 02:40:02 +08:00
|
|
|
<< 2 // of C pointer type
|
|
|
|
<< castExprType
|
|
|
|
<< unsigned(castType->isBlockPointerType()) // to ObjC|block type
|
|
|
|
<< castType
|
|
|
|
<< castRange
|
|
|
|
<< castExpr->getSourceRange();
|
2012-06-01 08:10:47 +08:00
|
|
|
bool br = S.isKnownName("CFBridgingRelease");
|
2018-07-31 03:24:48 +08:00
|
|
|
ACCResult CreateRule =
|
2012-07-28 07:55:46 +08:00
|
|
|
ARCCastChecker(S.Context, exprACTC, castACTC, true).Visit(castExpr);
|
2012-07-31 09:07:43 +08:00
|
|
|
assert(CreateRule != ACC_bottom && "This cast should already be accepted.");
|
2012-07-28 07:55:46 +08:00
|
|
|
if (CreateRule != ACC_plusOne)
|
2012-07-28 05:34:23 +08:00
|
|
|
{
|
2018-07-31 03:24:48 +08:00
|
|
|
DiagnosticBuilder DiagB =
|
2013-02-22 09:22:48 +08:00
|
|
|
(CCK != Sema::CCK_OtherCast) ? S.Diag(noteLoc, diag::note_arc_bridge)
|
|
|
|
: S.Diag(noteLoc, diag::note_arc_cstyle_bridge);
|
2014-05-26 14:22:03 +08:00
|
|
|
|
2012-07-28 05:34:23 +08:00
|
|
|
addFixitForObjCARCConversion(S, DiagB, CCK, afterLParen,
|
2014-05-26 14:22:03 +08:00
|
|
|
castType, castExpr, realCast, "__bridge ",
|
|
|
|
nullptr);
|
2012-07-28 05:34:23 +08:00
|
|
|
}
|
2012-07-29 02:59:49 +08:00
|
|
|
if (CreateRule != ACC_plusZero)
|
2012-07-28 05:34:23 +08:00
|
|
|
{
|
2013-02-22 09:22:48 +08:00
|
|
|
DiagnosticBuilder DiagB =
|
|
|
|
(CCK == Sema::CCK_OtherCast && !br) ?
|
|
|
|
S.Diag(noteLoc, diag::note_arc_cstyle_bridge_transfer) << castExprType :
|
|
|
|
S.Diag(br ? castExpr->getExprLoc() : noteLoc,
|
|
|
|
diag::note_arc_bridge_transfer)
|
|
|
|
<< castExprType << br;
|
2014-05-26 14:22:03 +08:00
|
|
|
|
2012-07-28 05:34:23 +08:00
|
|
|
addFixitForObjCARCConversion(S, DiagB, CCK, afterLParen,
|
2013-02-23 06:02:53 +08:00
|
|
|
castType, castExpr, realCast, "__bridge_transfer ",
|
2014-05-26 14:22:03 +08:00
|
|
|
br ? "CFBridgingRelease" : nullptr);
|
2012-07-28 05:34:23 +08:00
|
|
|
}
|
2011-10-18 02:40:02 +08:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
// Bridge from a CF type to an ARC type.
|
|
|
|
if (exprACTC == ACTC_retainable && isAnyRetainable(castACTC)) {
|
2012-06-01 08:10:47 +08:00
|
|
|
bool br = S.isKnownName("CFBridgingRetain");
|
2011-10-18 02:40:02 +08:00
|
|
|
S.Diag(loc, diag::err_arc_cast_requires_bridge)
|
2018-06-28 04:30:34 +08:00
|
|
|
<< convKindForDiag
|
2011-10-18 02:40:02 +08:00
|
|
|
<< unsigned(castExprType->isBlockPointerType()) // of ObjC|block type
|
|
|
|
<< castExprType
|
|
|
|
<< 2 // to C pointer type
|
|
|
|
<< castType
|
|
|
|
<< castRange
|
|
|
|
<< castExpr->getSourceRange();
|
2018-07-31 03:24:48 +08:00
|
|
|
ACCResult CreateRule =
|
2012-07-28 07:55:46 +08:00
|
|
|
ARCCastChecker(S.Context, exprACTC, castACTC, true).Visit(castExpr);
|
2012-07-31 09:07:43 +08:00
|
|
|
assert(CreateRule != ACC_bottom && "This cast should already be accepted.");
|
2012-07-28 07:55:46 +08:00
|
|
|
if (CreateRule != ACC_plusOne)
|
2012-07-28 05:34:23 +08:00
|
|
|
{
|
2013-02-22 09:22:48 +08:00
|
|
|
DiagnosticBuilder DiagB =
|
|
|
|
(CCK != Sema::CCK_OtherCast) ? S.Diag(noteLoc, diag::note_arc_bridge)
|
|
|
|
: S.Diag(noteLoc, diag::note_arc_cstyle_bridge);
|
2012-07-28 05:34:23 +08:00
|
|
|
addFixitForObjCARCConversion(S, DiagB, CCK, afterLParen,
|
2014-05-26 14:22:03 +08:00
|
|
|
castType, castExpr, realCast, "__bridge ",
|
|
|
|
nullptr);
|
2012-07-28 05:34:23 +08:00
|
|
|
}
|
2012-07-29 02:59:49 +08:00
|
|
|
if (CreateRule != ACC_plusZero)
|
2012-07-28 05:34:23 +08:00
|
|
|
{
|
2013-02-22 09:22:48 +08:00
|
|
|
DiagnosticBuilder DiagB =
|
|
|
|
(CCK == Sema::CCK_OtherCast && !br) ?
|
|
|
|
S.Diag(noteLoc, diag::note_arc_cstyle_bridge_retained) << castType :
|
|
|
|
S.Diag(br ? castExpr->getExprLoc() : noteLoc,
|
|
|
|
diag::note_arc_bridge_retained)
|
|
|
|
<< castType << br;
|
2014-05-26 14:22:03 +08:00
|
|
|
|
2012-07-28 05:34:23 +08:00
|
|
|
addFixitForObjCARCConversion(S, DiagB, CCK, afterLParen,
|
2013-02-23 06:02:53 +08:00
|
|
|
castType, castExpr, realCast, "__bridge_retained ",
|
2014-05-26 14:22:03 +08:00
|
|
|
br ? "CFBridgingRetain" : nullptr);
|
2012-07-28 05:34:23 +08:00
|
|
|
}
|
2011-10-18 02:40:02 +08:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
S.Diag(loc, diag::err_arc_mismatched_cast)
|
2018-06-28 04:30:34 +08:00
|
|
|
<< !convKindForDiag
|
2011-10-18 02:40:02 +08:00
|
|
|
<< srcKind << castExprType << castType
|
|
|
|
<< castRange << castExpr->getSourceRange();
|
|
|
|
}
|
|
|
|
|
2013-11-22 04:50:32 +08:00
|
|
|
template <typename TB>
|
2014-06-12 00:52:44 +08:00
|
|
|
static bool CheckObjCBridgeNSCast(Sema &S, QualType castType, Expr *castExpr,
|
|
|
|
bool &HadTheAttribute, bool warn) {
|
2013-11-16 06:18:17 +08:00
|
|
|
QualType T = castExpr->getType();
|
2014-06-12 00:52:44 +08:00
|
|
|
HadTheAttribute = false;
|
2013-11-16 06:18:17 +08:00
|
|
|
while (const TypedefType *TD = dyn_cast<TypedefType>(T.getTypePtr())) {
|
|
|
|
TypedefNameDecl *TDNDecl = TD->getDecl();
|
2013-11-22 04:50:32 +08:00
|
|
|
if (TB *ObjCBAttr = getObjCBridgeAttr<TB>(TD)) {
|
2013-11-19 08:09:48 +08:00
|
|
|
if (IdentifierInfo *Parm = ObjCBAttr->getBridgedType()) {
|
2014-06-12 00:52:44 +08:00
|
|
|
HadTheAttribute = true;
|
2014-12-12 06:56:26 +08:00
|
|
|
if (Parm->isStr("id"))
|
|
|
|
return true;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2014-05-26 14:22:03 +08:00
|
|
|
NamedDecl *Target = nullptr;
|
2013-11-16 06:18:17 +08:00
|
|
|
// Check for an existing type with this name.
|
|
|
|
LookupResult R(S, DeclarationName(Parm), SourceLocation(),
|
|
|
|
Sema::LookupOrdinaryName);
|
|
|
|
if (S.LookupName(R, S.TUScope)) {
|
2013-11-16 09:45:25 +08:00
|
|
|
Target = R.getFoundDecl();
|
|
|
|
if (Target && isa<ObjCInterfaceDecl>(Target)) {
|
|
|
|
ObjCInterfaceDecl *ExprClass = cast<ObjCInterfaceDecl>(Target);
|
|
|
|
if (const ObjCObjectPointerType *InterfacePointerType =
|
|
|
|
castType->getAsObjCInterfacePointerType()) {
|
|
|
|
ObjCInterfaceDecl *CastClass
|
|
|
|
= InterfacePointerType->getObjectType()->getInterface();
|
2013-11-20 08:32:12 +08:00
|
|
|
if ((CastClass == ExprClass) ||
|
2015-04-11 06:07:47 +08:00
|
|
|
(CastClass && CastClass->isSuperClassOf(ExprClass)))
|
2014-06-12 00:52:44 +08:00
|
|
|
return true;
|
|
|
|
if (warn)
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(castExpr->getBeginLoc(), diag::warn_objc_invalid_bridge)
|
|
|
|
<< T << Target->getName() << castType->getPointeeType();
|
2014-06-12 00:52:44 +08:00
|
|
|
return false;
|
2013-11-20 08:32:12 +08:00
|
|
|
} else if (castType->isObjCIdType() ||
|
|
|
|
(S.Context.ObjCObjectAdoptsQTypeProtocols(
|
|
|
|
castType, ExprClass)))
|
|
|
|
// ok to cast to 'id'.
|
|
|
|
// casting to id<p-list> is ok if bridge type adopts all of
|
|
|
|
// p-list protocols.
|
2014-06-12 00:52:44 +08:00
|
|
|
return true;
|
2013-11-20 08:32:12 +08:00
|
|
|
else {
|
2014-06-12 00:52:44 +08:00
|
|
|
if (warn) {
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(castExpr->getBeginLoc(), diag::warn_objc_invalid_bridge)
|
|
|
|
<< T << Target->getName() << castType;
|
|
|
|
S.Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
|
|
|
S.Diag(Target->getBeginLoc(), diag::note_declared_at);
|
2014-06-12 00:52:44 +08:00
|
|
|
}
|
|
|
|
return false;
|
2013-11-17 07:22:37 +08:00
|
|
|
}
|
2013-11-16 06:18:17 +08:00
|
|
|
}
|
2015-04-10 07:39:53 +08:00
|
|
|
} else if (!castType->isObjCIdType()) {
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(castExpr->getBeginLoc(),
|
|
|
|
diag::err_objc_cf_bridged_not_interface)
|
|
|
|
<< castExpr->getType() << Parm;
|
|
|
|
S.Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
2015-04-10 07:39:53 +08:00
|
|
|
if (Target)
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(Target->getBeginLoc(), diag::note_declared_at);
|
2013-11-16 06:18:17 +08:00
|
|
|
}
|
2014-06-12 00:52:44 +08:00
|
|
|
return true;
|
2013-11-16 06:18:17 +08:00
|
|
|
}
|
2014-06-12 00:52:44 +08:00
|
|
|
return false;
|
2013-11-16 06:18:17 +08:00
|
|
|
}
|
|
|
|
T = TDNDecl->getUnderlyingType();
|
|
|
|
}
|
2014-06-12 00:52:44 +08:00
|
|
|
return true;
|
2013-11-16 06:18:17 +08:00
|
|
|
}
|
|
|
|
|
2013-11-22 04:50:32 +08:00
|
|
|
template <typename TB>
|
2014-06-12 00:52:44 +08:00
|
|
|
static bool CheckObjCBridgeCFCast(Sema &S, QualType castType, Expr *castExpr,
|
|
|
|
bool &HadTheAttribute, bool warn) {
|
2013-11-17 03:16:32 +08:00
|
|
|
QualType T = castType;
|
2014-06-12 00:52:44 +08:00
|
|
|
HadTheAttribute = false;
|
2013-11-17 03:16:32 +08:00
|
|
|
while (const TypedefType *TD = dyn_cast<TypedefType>(T.getTypePtr())) {
|
|
|
|
TypedefNameDecl *TDNDecl = TD->getDecl();
|
2013-11-22 04:50:32 +08:00
|
|
|
if (TB *ObjCBAttr = getObjCBridgeAttr<TB>(TD)) {
|
2013-11-19 08:09:48 +08:00
|
|
|
if (IdentifierInfo *Parm = ObjCBAttr->getBridgedType()) {
|
2014-06-12 00:52:44 +08:00
|
|
|
HadTheAttribute = true;
|
2015-03-11 02:41:23 +08:00
|
|
|
if (Parm->isStr("id"))
|
|
|
|
return true;
|
|
|
|
|
2014-05-26 14:22:03 +08:00
|
|
|
NamedDecl *Target = nullptr;
|
2013-11-17 03:16:32 +08:00
|
|
|
// Check for an existing type with this name.
|
|
|
|
LookupResult R(S, DeclarationName(Parm), SourceLocation(),
|
|
|
|
Sema::LookupOrdinaryName);
|
|
|
|
if (S.LookupName(R, S.TUScope)) {
|
|
|
|
Target = R.getFoundDecl();
|
|
|
|
if (Target && isa<ObjCInterfaceDecl>(Target)) {
|
|
|
|
ObjCInterfaceDecl *CastClass = cast<ObjCInterfaceDecl>(Target);
|
|
|
|
if (const ObjCObjectPointerType *InterfacePointerType =
|
|
|
|
castExpr->getType()->getAsObjCInterfacePointerType()) {
|
|
|
|
ObjCInterfaceDecl *ExprClass
|
|
|
|
= InterfacePointerType->getObjectType()->getInterface();
|
2013-11-20 08:32:12 +08:00
|
|
|
if ((CastClass == ExprClass) ||
|
|
|
|
(ExprClass && CastClass->isSuperClassOf(ExprClass)))
|
2014-06-12 00:52:44 +08:00
|
|
|
return true;
|
|
|
|
if (warn) {
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(castExpr->getBeginLoc(),
|
|
|
|
diag::warn_objc_invalid_bridge_to_cf)
|
|
|
|
<< castExpr->getType()->getPointeeType() << T;
|
|
|
|
S.Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
2014-06-12 00:52:44 +08:00
|
|
|
}
|
|
|
|
return false;
|
2013-11-20 08:32:12 +08:00
|
|
|
} else if (castExpr->getType()->isObjCIdType() ||
|
|
|
|
(S.Context.QIdProtocolsAdoptObjCObjectProtocols(
|
|
|
|
castExpr->getType(), CastClass)))
|
|
|
|
// ok to cast an 'id' expression to a CFtype.
|
|
|
|
// ok to cast an 'id<plist>' expression to CFtype provided plist
|
|
|
|
// adopts all of CFtype's ObjetiveC's class plist.
|
2014-06-12 00:52:44 +08:00
|
|
|
return true;
|
2013-11-20 08:32:12 +08:00
|
|
|
else {
|
2014-06-12 00:52:44 +08:00
|
|
|
if (warn) {
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(castExpr->getBeginLoc(),
|
|
|
|
diag::warn_objc_invalid_bridge_to_cf)
|
|
|
|
<< castExpr->getType() << castType;
|
|
|
|
S.Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
|
|
|
S.Diag(Target->getBeginLoc(), diag::note_declared_at);
|
2014-06-12 00:52:44 +08:00
|
|
|
}
|
|
|
|
return false;
|
2013-11-17 03:16:32 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(castExpr->getBeginLoc(),
|
|
|
|
diag::err_objc_ns_bridged_invalid_cfobject)
|
|
|
|
<< castExpr->getType() << castType;
|
|
|
|
S.Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
2013-11-17 03:16:32 +08:00
|
|
|
if (Target)
|
2018-08-10 05:08:08 +08:00
|
|
|
S.Diag(Target->getBeginLoc(), diag::note_declared_at);
|
2014-06-12 00:52:44 +08:00
|
|
|
return true;
|
2013-11-17 03:16:32 +08:00
|
|
|
}
|
2014-06-12 00:52:44 +08:00
|
|
|
return false;
|
2013-11-17 03:16:32 +08:00
|
|
|
}
|
|
|
|
T = TDNDecl->getUnderlyingType();
|
|
|
|
}
|
2014-06-12 00:52:44 +08:00
|
|
|
return true;
|
2013-11-17 03:16:32 +08:00
|
|
|
}
|
|
|
|
|
2013-11-21 08:39:36 +08:00
|
|
|
void Sema::CheckTollFreeBridgeCast(QualType castType, Expr *castExpr) {
|
2018-10-31 04:31:30 +08:00
|
|
|
if (!getLangOpts().ObjC)
|
2014-04-30 00:12:56 +08:00
|
|
|
return;
|
2013-12-06 00:25:25 +08:00
|
|
|
// warn in presence of __bridge casting to or from a toll free bridge cast.
|
2013-11-21 08:39:36 +08:00
|
|
|
ARCConversionTypeClass exprACTC = classifyTypeForARCConversion(castExpr->getType());
|
|
|
|
ARCConversionTypeClass castACTC = classifyTypeForARCConversion(castType);
|
2013-11-22 04:50:32 +08:00
|
|
|
if (castACTC == ACTC_retainable && exprACTC == ACTC_coreFoundation) {
|
2014-06-12 00:52:44 +08:00
|
|
|
bool HasObjCBridgeAttr;
|
|
|
|
bool ObjCBridgeAttrWillNotWarn =
|
|
|
|
CheckObjCBridgeNSCast<ObjCBridgeAttr>(*this, castType, castExpr, HasObjCBridgeAttr,
|
|
|
|
false);
|
|
|
|
if (ObjCBridgeAttrWillNotWarn && HasObjCBridgeAttr)
|
|
|
|
return;
|
|
|
|
bool HasObjCBridgeMutableAttr;
|
|
|
|
bool ObjCBridgeMutableAttrWillNotWarn =
|
|
|
|
CheckObjCBridgeNSCast<ObjCBridgeMutableAttr>(*this, castType, castExpr,
|
|
|
|
HasObjCBridgeMutableAttr, false);
|
|
|
|
if (ObjCBridgeMutableAttrWillNotWarn && HasObjCBridgeMutableAttr)
|
|
|
|
return;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2014-06-12 00:52:44 +08:00
|
|
|
if (HasObjCBridgeAttr)
|
|
|
|
CheckObjCBridgeNSCast<ObjCBridgeAttr>(*this, castType, castExpr, HasObjCBridgeAttr,
|
|
|
|
true);
|
|
|
|
else if (HasObjCBridgeMutableAttr)
|
|
|
|
CheckObjCBridgeNSCast<ObjCBridgeMutableAttr>(*this, castType, castExpr,
|
|
|
|
HasObjCBridgeMutableAttr, true);
|
2013-11-22 04:50:32 +08:00
|
|
|
}
|
|
|
|
else if (castACTC == ACTC_coreFoundation && exprACTC == ACTC_retainable) {
|
2014-06-12 00:52:44 +08:00
|
|
|
bool HasObjCBridgeAttr;
|
|
|
|
bool ObjCBridgeAttrWillNotWarn =
|
|
|
|
CheckObjCBridgeCFCast<ObjCBridgeAttr>(*this, castType, castExpr, HasObjCBridgeAttr,
|
|
|
|
false);
|
|
|
|
if (ObjCBridgeAttrWillNotWarn && HasObjCBridgeAttr)
|
|
|
|
return;
|
|
|
|
bool HasObjCBridgeMutableAttr;
|
|
|
|
bool ObjCBridgeMutableAttrWillNotWarn =
|
|
|
|
CheckObjCBridgeCFCast<ObjCBridgeMutableAttr>(*this, castType, castExpr,
|
|
|
|
HasObjCBridgeMutableAttr, false);
|
|
|
|
if (ObjCBridgeMutableAttrWillNotWarn && HasObjCBridgeMutableAttr)
|
|
|
|
return;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2014-06-12 00:52:44 +08:00
|
|
|
if (HasObjCBridgeAttr)
|
|
|
|
CheckObjCBridgeCFCast<ObjCBridgeAttr>(*this, castType, castExpr, HasObjCBridgeAttr,
|
|
|
|
true);
|
|
|
|
else if (HasObjCBridgeMutableAttr)
|
|
|
|
CheckObjCBridgeCFCast<ObjCBridgeMutableAttr>(*this, castType, castExpr,
|
|
|
|
HasObjCBridgeMutableAttr, true);
|
2013-11-22 04:50:32 +08:00
|
|
|
}
|
2013-11-21 08:39:36 +08:00
|
|
|
}
|
|
|
|
|
2014-06-27 05:22:16 +08:00
|
|
|
void Sema::CheckObjCBridgeRelatedCast(QualType castType, Expr *castExpr) {
|
|
|
|
QualType SrcType = castExpr->getType();
|
|
|
|
if (ObjCPropertyRefExpr *PRE = dyn_cast<ObjCPropertyRefExpr>(castExpr)) {
|
|
|
|
if (PRE->isExplicitProperty()) {
|
|
|
|
if (ObjCPropertyDecl *PDecl = PRE->getExplicitProperty())
|
|
|
|
SrcType = PDecl->getType();
|
|
|
|
}
|
|
|
|
else if (PRE->isImplicitProperty()) {
|
|
|
|
if (ObjCMethodDecl *Getter = PRE->getImplicitPropertyGetter())
|
|
|
|
SrcType = Getter->getReturnType();
|
|
|
|
}
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2014-06-27 05:22:16 +08:00
|
|
|
ARCConversionTypeClass srcExprACTC = classifyTypeForARCConversion(SrcType);
|
|
|
|
ARCConversionTypeClass castExprACTC = classifyTypeForARCConversion(castType);
|
|
|
|
if (srcExprACTC != ACTC_retainable || castExprACTC != ACTC_coreFoundation)
|
|
|
|
return;
|
2018-08-10 05:08:08 +08:00
|
|
|
CheckObjCBridgeRelatedConversions(castExpr->getBeginLoc(), castType, SrcType,
|
|
|
|
castExpr);
|
2014-06-27 05:22:16 +08:00
|
|
|
}
|
|
|
|
|
2014-05-11 01:40:11 +08:00
|
|
|
bool Sema::CheckTollFreeBridgeStaticCast(QualType castType, Expr *castExpr,
|
|
|
|
CastKind &Kind) {
|
2018-10-31 04:31:30 +08:00
|
|
|
if (!getLangOpts().ObjC)
|
2014-05-11 01:40:11 +08:00
|
|
|
return false;
|
|
|
|
ARCConversionTypeClass exprACTC =
|
|
|
|
classifyTypeForARCConversion(castExpr->getType());
|
|
|
|
ARCConversionTypeClass castACTC = classifyTypeForARCConversion(castType);
|
|
|
|
if ((castACTC == ACTC_retainable && exprACTC == ACTC_coreFoundation) ||
|
|
|
|
(castACTC == ACTC_coreFoundation && exprACTC == ACTC_retainable)) {
|
|
|
|
CheckTollFreeBridgeCast(castType, castExpr);
|
|
|
|
Kind = (castACTC == ACTC_coreFoundation) ? CK_BitCast
|
|
|
|
: CK_CPointerToObjCPointerCast;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2013-12-07 08:34:23 +08:00
|
|
|
|
|
|
|
bool Sema::checkObjCBridgeRelatedComponents(SourceLocation Loc,
|
|
|
|
QualType DestType, QualType SrcType,
|
|
|
|
ObjCInterfaceDecl *&RelatedClass,
|
|
|
|
ObjCMethodDecl *&ClassMethod,
|
|
|
|
ObjCMethodDecl *&InstanceMethod,
|
|
|
|
TypedefNameDecl *&TDNDecl,
|
2016-01-14 07:36:34 +08:00
|
|
|
bool CfToNs, bool Diagnose) {
|
2013-12-07 08:34:23 +08:00
|
|
|
QualType T = CfToNs ? SrcType : DestType;
|
2013-12-10 06:04:26 +08:00
|
|
|
ObjCBridgeRelatedAttr *ObjCBAttr = ObjCBridgeRelatedAttrFromType(T, TDNDecl);
|
|
|
|
if (!ObjCBAttr)
|
|
|
|
return false;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2013-12-10 06:04:26 +08:00
|
|
|
IdentifierInfo *RCId = ObjCBAttr->getRelatedClass();
|
|
|
|
IdentifierInfo *CMId = ObjCBAttr->getClassMethod();
|
|
|
|
IdentifierInfo *IMId = ObjCBAttr->getInstanceMethod();
|
|
|
|
if (!RCId)
|
|
|
|
return false;
|
2014-05-26 14:22:03 +08:00
|
|
|
NamedDecl *Target = nullptr;
|
2013-12-10 06:04:26 +08:00
|
|
|
// Check for an existing type with this name.
|
|
|
|
LookupResult R(*this, DeclarationName(RCId), SourceLocation(),
|
|
|
|
Sema::LookupOrdinaryName);
|
|
|
|
if (!LookupName(R, TUScope)) {
|
2016-01-14 07:36:34 +08:00
|
|
|
if (Diagnose) {
|
|
|
|
Diag(Loc, diag::err_objc_bridged_related_invalid_class) << RCId
|
|
|
|
<< SrcType << DestType;
|
2018-08-10 05:08:08 +08:00
|
|
|
Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
2016-01-14 07:36:34 +08:00
|
|
|
}
|
2013-12-10 06:04:26 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Target = R.getFoundDecl();
|
|
|
|
if (Target && isa<ObjCInterfaceDecl>(Target))
|
|
|
|
RelatedClass = cast<ObjCInterfaceDecl>(Target);
|
|
|
|
else {
|
2016-01-14 07:36:34 +08:00
|
|
|
if (Diagnose) {
|
|
|
|
Diag(Loc, diag::err_objc_bridged_related_invalid_class_name) << RCId
|
|
|
|
<< SrcType << DestType;
|
2018-08-10 05:08:08 +08:00
|
|
|
Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
2016-01-14 07:36:34 +08:00
|
|
|
if (Target)
|
2018-08-10 05:08:08 +08:00
|
|
|
Diag(Target->getBeginLoc(), diag::note_declared_at);
|
2016-01-14 07:36:34 +08:00
|
|
|
}
|
2013-12-10 06:04:26 +08:00
|
|
|
return false;
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2013-12-10 06:04:26 +08:00
|
|
|
// Check for an existing class method with the given selector name.
|
|
|
|
if (CfToNs && CMId) {
|
|
|
|
Selector Sel = Context.Selectors.getUnarySelector(CMId);
|
|
|
|
ClassMethod = RelatedClass->lookupMethod(Sel, false);
|
|
|
|
if (!ClassMethod) {
|
2016-01-14 07:36:34 +08:00
|
|
|
if (Diagnose) {
|
|
|
|
Diag(Loc, diag::err_objc_bridged_related_known_method)
|
|
|
|
<< SrcType << DestType << Sel << false;
|
2018-08-10 05:08:08 +08:00
|
|
|
Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
2016-01-14 07:36:34 +08:00
|
|
|
}
|
2013-12-10 06:04:26 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2013-12-10 06:04:26 +08:00
|
|
|
// Check for an existing instance method with the given selector name.
|
|
|
|
if (!CfToNs && IMId) {
|
|
|
|
Selector Sel = Context.Selectors.getNullarySelector(IMId);
|
|
|
|
InstanceMethod = RelatedClass->lookupMethod(Sel, true);
|
|
|
|
if (!InstanceMethod) {
|
2016-01-14 07:36:34 +08:00
|
|
|
if (Diagnose) {
|
|
|
|
Diag(Loc, diag::err_objc_bridged_related_known_method)
|
|
|
|
<< SrcType << DestType << Sel << true;
|
2018-08-10 05:08:08 +08:00
|
|
|
Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
2016-01-14 07:36:34 +08:00
|
|
|
}
|
2013-12-10 06:04:26 +08:00
|
|
|
return false;
|
2013-12-07 08:34:23 +08:00
|
|
|
}
|
|
|
|
}
|
2013-12-10 06:04:26 +08:00
|
|
|
return true;
|
2013-12-07 08:34:23 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
Sema::CheckObjCBridgeRelatedConversions(SourceLocation Loc,
|
2013-12-11 01:08:13 +08:00
|
|
|
QualType DestType, QualType SrcType,
|
2016-01-14 07:36:34 +08:00
|
|
|
Expr *&SrcExpr, bool Diagnose) {
|
2013-12-07 08:34:23 +08:00
|
|
|
ARCConversionTypeClass rhsExprACTC = classifyTypeForARCConversion(SrcType);
|
|
|
|
ARCConversionTypeClass lhsExprACTC = classifyTypeForARCConversion(DestType);
|
|
|
|
bool CfToNs = (rhsExprACTC == ACTC_coreFoundation && lhsExprACTC == ACTC_retainable);
|
|
|
|
bool NsToCf = (rhsExprACTC == ACTC_retainable && lhsExprACTC == ACTC_coreFoundation);
|
|
|
|
if (!CfToNs && !NsToCf)
|
|
|
|
return false;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2013-12-07 08:34:23 +08:00
|
|
|
ObjCInterfaceDecl *RelatedClass;
|
2014-05-26 14:22:03 +08:00
|
|
|
ObjCMethodDecl *ClassMethod = nullptr;
|
|
|
|
ObjCMethodDecl *InstanceMethod = nullptr;
|
|
|
|
TypedefNameDecl *TDNDecl = nullptr;
|
2013-12-07 08:34:23 +08:00
|
|
|
if (!checkObjCBridgeRelatedComponents(Loc, DestType, SrcType, RelatedClass,
|
2016-01-14 07:36:34 +08:00
|
|
|
ClassMethod, InstanceMethod, TDNDecl,
|
|
|
|
CfToNs, Diagnose))
|
2013-12-07 08:34:23 +08:00
|
|
|
return false;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2013-12-07 08:34:23 +08:00
|
|
|
if (CfToNs) {
|
|
|
|
// Implicit conversion from CF to ObjC object is needed.
|
2013-12-11 01:08:13 +08:00
|
|
|
if (ClassMethod) {
|
2016-01-14 07:36:34 +08:00
|
|
|
if (Diagnose) {
|
|
|
|
std::string ExpressionString = "[";
|
|
|
|
ExpressionString += RelatedClass->getNameAsString();
|
|
|
|
ExpressionString += " ";
|
|
|
|
ExpressionString += ClassMethod->getSelector().getAsString();
|
2018-08-10 05:09:38 +08:00
|
|
|
SourceLocation SrcExprEndLoc =
|
|
|
|
getLocForEndOfToken(SrcExpr->getEndLoc());
|
2016-01-14 07:36:34 +08:00
|
|
|
// Provide a fixit: [RelatedClass ClassMethod SrcExpr]
|
|
|
|
Diag(Loc, diag::err_objc_bridged_related_known_method)
|
2018-08-10 05:08:08 +08:00
|
|
|
<< SrcType << DestType << ClassMethod->getSelector() << false
|
|
|
|
<< FixItHint::CreateInsertion(SrcExpr->getBeginLoc(),
|
|
|
|
ExpressionString)
|
|
|
|
<< FixItHint::CreateInsertion(SrcExprEndLoc, "]");
|
|
|
|
Diag(RelatedClass->getBeginLoc(), diag::note_declared_at);
|
|
|
|
Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2016-02-13 09:41:41 +08:00
|
|
|
QualType receiverType = Context.getObjCInterfaceType(RelatedClass);
|
|
|
|
// Argument.
|
|
|
|
Expr *args[] = { SrcExpr };
|
|
|
|
ExprResult msg = BuildClassMessageImplicit(receiverType, false,
|
2013-12-17 06:54:37 +08:00
|
|
|
ClassMethod->getLocation(),
|
|
|
|
ClassMethod->getSelector(), ClassMethod,
|
|
|
|
MultiExprArg(args, 1));
|
2016-02-13 09:41:41 +08:00
|
|
|
SrcExpr = msg.get();
|
|
|
|
}
|
2013-12-17 06:54:37 +08:00
|
|
|
return true;
|
2013-12-11 01:08:13 +08:00
|
|
|
}
|
2013-12-07 08:34:23 +08:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Implicit conversion from ObjC type to CF object is needed.
|
2013-12-11 01:08:13 +08:00
|
|
|
if (InstanceMethod) {
|
2016-01-14 07:36:34 +08:00
|
|
|
if (Diagnose) {
|
|
|
|
std::string ExpressionString;
|
|
|
|
SourceLocation SrcExprEndLoc =
|
2018-08-10 05:09:38 +08:00
|
|
|
getLocForEndOfToken(SrcExpr->getEndLoc());
|
2016-01-14 07:36:34 +08:00
|
|
|
if (InstanceMethod->isPropertyAccessor())
|
|
|
|
if (const ObjCPropertyDecl *PDecl =
|
|
|
|
InstanceMethod->findPropertyDecl()) {
|
|
|
|
// fixit: ObjectExpr.propertyname when it is aproperty accessor.
|
|
|
|
ExpressionString = ".";
|
|
|
|
ExpressionString += PDecl->getNameAsString();
|
|
|
|
Diag(Loc, diag::err_objc_bridged_related_known_method)
|
|
|
|
<< SrcType << DestType << InstanceMethod->getSelector() << true
|
|
|
|
<< FixItHint::CreateInsertion(SrcExprEndLoc, ExpressionString);
|
|
|
|
}
|
|
|
|
if (ExpressionString.empty()) {
|
|
|
|
// Provide a fixit: [ObjectExpr InstanceMethod]
|
|
|
|
ExpressionString = " ";
|
|
|
|
ExpressionString += InstanceMethod->getSelector().getAsString();
|
|
|
|
ExpressionString += "]";
|
|
|
|
|
2013-12-11 07:18:06 +08:00
|
|
|
Diag(Loc, diag::err_objc_bridged_related_known_method)
|
2016-01-14 07:36:34 +08:00
|
|
|
<< SrcType << DestType << InstanceMethod->getSelector() << true
|
2018-08-10 05:08:08 +08:00
|
|
|
<< FixItHint::CreateInsertion(SrcExpr->getBeginLoc(), "[")
|
2016-01-14 07:36:34 +08:00
|
|
|
<< FixItHint::CreateInsertion(SrcExprEndLoc, ExpressionString);
|
2013-12-11 07:18:06 +08:00
|
|
|
}
|
2018-08-10 05:08:08 +08:00
|
|
|
Diag(RelatedClass->getBeginLoc(), diag::note_declared_at);
|
|
|
|
Diag(TDNDecl->getBeginLoc(), diag::note_declared_at);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2016-02-13 09:41:41 +08:00
|
|
|
ExprResult msg =
|
|
|
|
BuildInstanceMessageImplicit(SrcExpr, SrcType,
|
|
|
|
InstanceMethod->getLocation(),
|
|
|
|
InstanceMethod->getSelector(),
|
|
|
|
InstanceMethod, None);
|
|
|
|
SrcExpr = msg.get();
|
|
|
|
}
|
2013-12-17 06:54:37 +08:00
|
|
|
return true;
|
2013-12-11 01:08:13 +08:00
|
|
|
}
|
2013-12-07 08:34:23 +08:00
|
|
|
}
|
2013-12-17 06:54:37 +08:00
|
|
|
return false;
|
2013-12-07 08:34:23 +08:00
|
|
|
}
|
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
Sema::ARCConversionResult
|
2017-03-30 02:09:02 +08:00
|
|
|
Sema::CheckObjCConversion(SourceRange castRange, QualType castType,
|
|
|
|
Expr *&castExpr, CheckedConversionKind CCK,
|
|
|
|
bool Diagnose, bool DiagnoseCFAudited,
|
|
|
|
BinaryOperatorKind Opc) {
|
2011-06-16 07:02:42 +08:00
|
|
|
QualType castExprType = castExpr->getType();
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
// For the purposes of the classification, we assume reference types
|
|
|
|
// will bind to temporaries.
|
|
|
|
QualType effCastType = castType;
|
|
|
|
if (const ReferenceType *ref = castType->getAs<ReferenceType>())
|
|
|
|
effCastType = ref->getPointeeType();
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
ARCConversionTypeClass exprACTC = classifyTypeForARCConversion(castExprType);
|
2011-10-01 09:01:08 +08:00
|
|
|
ARCConversionTypeClass castACTC = classifyTypeForARCConversion(effCastType);
|
2011-10-29 04:06:07 +08:00
|
|
|
if (exprACTC == castACTC) {
|
2016-02-13 09:41:41 +08:00
|
|
|
// Check for viability and report error if casting an rvalue to a
|
2011-10-29 04:06:07 +08:00
|
|
|
// life-time qualifier.
|
2016-02-13 09:41:41 +08:00
|
|
|
if (castACTC == ACTC_retainable &&
|
2011-10-29 04:06:07 +08:00
|
|
|
(CCK == CCK_CStyleCast || CCK == CCK_OtherCast) &&
|
2016-01-14 07:36:34 +08:00
|
|
|
castType != castExprType) {
|
2011-10-29 08:06:10 +08:00
|
|
|
const Type *DT = castType.getTypePtr();
|
|
|
|
QualType QDT = castType;
|
|
|
|
// We desugar some types but not others. We ignore those
|
|
|
|
// that cannot happen in a cast; i.e. auto, and those which
|
|
|
|
// should not be de-sugared; i.e typedef.
|
|
|
|
if (const ParenType *PT = dyn_cast<ParenType>(DT))
|
|
|
|
QDT = PT->desugar();
|
|
|
|
else if (const TypeOfType *TP = dyn_cast<TypeOfType>(DT))
|
|
|
|
QDT = TP->desugar();
|
|
|
|
else if (const AttributedType *AT = dyn_cast<AttributedType>(DT))
|
|
|
|
QDT = AT->desugar();
|
|
|
|
if (QDT != castType &&
|
|
|
|
QDT.getObjCLifetime() != Qualifiers::OCL_None) {
|
2016-02-13 09:41:41 +08:00
|
|
|
if (Diagnose) {
|
2018-07-31 03:24:48 +08:00
|
|
|
SourceLocation loc = (castRange.isValid() ? castRange.getBegin()
|
2016-02-13 09:41:41 +08:00
|
|
|
: castExpr->getExprLoc());
|
|
|
|
Diag(loc, diag::err_arc_nolifetime_behavior);
|
|
|
|
}
|
|
|
|
return ACR_error;
|
2011-10-29 08:06:10 +08:00
|
|
|
}
|
2011-10-29 04:06:07 +08:00
|
|
|
}
|
|
|
|
return ACR_okay;
|
|
|
|
}
|
2017-03-30 02:09:02 +08:00
|
|
|
|
|
|
|
// The life-time qualifier cast check above is all we need for ObjCWeak.
|
|
|
|
// ObjCAutoRefCount has more restrictions on what is legal.
|
|
|
|
if (!getLangOpts().ObjCAutoRefCount)
|
|
|
|
return ACR_okay;
|
|
|
|
|
2011-10-18 02:40:02 +08:00
|
|
|
if (isAnyCLike(exprACTC) && isAnyCLike(castACTC)) return ACR_okay;
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
// Allow all of these types to be cast to integer types (but not
|
|
|
|
// vice-versa).
|
|
|
|
if (castACTC == ACTC_none && castType->isIntegralType(Context))
|
2011-10-18 02:40:02 +08:00
|
|
|
return ACR_okay;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
// Allow casts between pointers to lifetime types (e.g., __strong id*)
|
|
|
|
// and pointers to void (e.g., cv void *). Casting from void* to lifetime*
|
|
|
|
// must be explicit.
|
2011-10-01 09:01:08 +08:00
|
|
|
if (exprACTC == ACTC_indirectRetainable && castACTC == ACTC_voidPtr)
|
2011-10-18 02:40:02 +08:00
|
|
|
return ACR_okay;
|
2011-10-01 09:01:08 +08:00
|
|
|
if (castACTC == ACTC_indirectRetainable && exprACTC == ACTC_voidPtr &&
|
2018-06-28 04:30:34 +08:00
|
|
|
isCast(CCK))
|
2011-10-18 02:40:02 +08:00
|
|
|
return ACR_okay;
|
2011-10-01 09:01:08 +08:00
|
|
|
|
2012-07-28 06:37:07 +08:00
|
|
|
switch (ARCCastChecker(Context, exprACTC, castACTC, false).Visit(castExpr)) {
|
2011-10-01 09:01:08 +08:00
|
|
|
// For invalid casts, fall through.
|
|
|
|
case ACC_invalid:
|
|
|
|
break;
|
|
|
|
|
|
|
|
// Do nothing for both bottom and +0.
|
|
|
|
case ACC_bottom:
|
|
|
|
case ACC_plusZero:
|
2011-10-18 02:40:02 +08:00
|
|
|
return ACR_okay;
|
2011-10-01 09:01:08 +08:00
|
|
|
|
|
|
|
// If the result is +1, consume it here.
|
|
|
|
case ACC_plusOne:
|
|
|
|
castExpr = ImplicitCastExpr::Create(Context, castExpr->getType(),
|
|
|
|
CK_ARCConsumeObject, castExpr,
|
2014-05-26 14:22:03 +08:00
|
|
|
nullptr, VK_RValue);
|
2016-06-22 04:29:17 +08:00
|
|
|
Cleanup.setExprNeedsCleanups(true);
|
2011-10-18 02:40:02 +08:00
|
|
|
return ACR_okay;
|
2011-10-01 09:01:08 +08:00
|
|
|
}
|
2011-10-18 02:40:02 +08:00
|
|
|
|
|
|
|
// If this is a non-implicit cast from id or block type to a
|
|
|
|
// CoreFoundation type, delay complaining in case the cast is used
|
|
|
|
// in an acceptable context.
|
2018-06-28 04:30:34 +08:00
|
|
|
if (exprACTC == ACTC_retainable && isAnyRetainable(castACTC) && isCast(CCK))
|
2011-10-18 02:40:02 +08:00
|
|
|
return ACR_unbridged;
|
|
|
|
|
2016-02-13 09:41:41 +08:00
|
|
|
// Issue a diagnostic about a missing @-sign when implicit casting a cstring
|
|
|
|
// to 'NSString *', instead of falling through to report a "bridge cast"
|
|
|
|
// diagnostic.
|
2013-12-19 05:04:43 +08:00
|
|
|
if (castACTC == ACTC_retainable && exprACTC == ACTC_none &&
|
2016-01-14 07:36:34 +08:00
|
|
|
ConversionToObjCStringLiteralCheck(castType, castExpr, Diagnose))
|
2016-02-13 09:41:41 +08:00
|
|
|
return ACR_error;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2013-08-01 05:40:51 +08:00
|
|
|
// Do not issue "bridge cast" diagnostic when implicit casting
|
|
|
|
// a retainable object to a CF type parameter belonging to an audited
|
|
|
|
// CF API function. Let caller issue a normal type mismatched diagnostic
|
|
|
|
// instead.
|
2016-02-13 09:41:41 +08:00
|
|
|
if ((!DiagnoseCFAudited || exprACTC != ACTC_retainable ||
|
|
|
|
castACTC != ACTC_coreFoundation) &&
|
|
|
|
!(exprACTC == ACTC_voidPtr && castACTC == ACTC_retainable &&
|
|
|
|
(Opc == BO_NE || Opc == BO_EQ))) {
|
|
|
|
if (Diagnose)
|
2016-01-14 07:36:34 +08:00
|
|
|
diagnoseObjCARCConversion(*this, castRange, castType, castACTC, castExpr,
|
|
|
|
castExpr, exprACTC, CCK);
|
2016-02-13 09:41:41 +08:00
|
|
|
return ACR_error;
|
|
|
|
}
|
2011-10-18 02:40:02 +08:00
|
|
|
return ACR_okay;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Given that we saw an expression with the ARCUnbridgedCastTy
|
|
|
|
/// placeholder type, complain bitterly.
|
|
|
|
void Sema::diagnoseARCUnbridgedCast(Expr *e) {
|
|
|
|
// We expect the spurious ImplicitCastExpr to already have been stripped.
|
|
|
|
assert(!e->hasPlaceholderType(BuiltinType::ARCUnbridgedCast));
|
|
|
|
CastExpr *realCast = cast<CastExpr>(e->IgnoreParens());
|
|
|
|
|
|
|
|
SourceRange castRange;
|
|
|
|
QualType castType;
|
|
|
|
CheckedConversionKind CCK;
|
|
|
|
|
|
|
|
if (CStyleCastExpr *cast = dyn_cast<CStyleCastExpr>(realCast)) {
|
|
|
|
castRange = SourceRange(cast->getLParenLoc(), cast->getRParenLoc());
|
|
|
|
castType = cast->getTypeAsWritten();
|
|
|
|
CCK = CCK_CStyleCast;
|
|
|
|
} else if (ExplicitCastExpr *cast = dyn_cast<ExplicitCastExpr>(realCast)) {
|
|
|
|
castRange = cast->getTypeInfoAsWritten()->getTypeLoc().getSourceRange();
|
|
|
|
castType = cast->getTypeAsWritten();
|
|
|
|
CCK = CCK_OtherCast;
|
|
|
|
} else {
|
2017-05-09 09:54:51 +08:00
|
|
|
llvm_unreachable("Unexpected ImplicitCastExpr");
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2011-10-18 02:40:02 +08:00
|
|
|
|
|
|
|
ARCConversionTypeClass castACTC =
|
|
|
|
classifyTypeForARCConversion(castType.getNonReferenceType());
|
|
|
|
|
|
|
|
Expr *castExpr = realCast->getSubExpr();
|
|
|
|
assert(classifyTypeForARCConversion(castExpr->getType()) == ACTC_retainable);
|
|
|
|
|
|
|
|
diagnoseObjCARCConversion(*this, castRange, castType, castACTC,
|
2013-02-23 06:02:53 +08:00
|
|
|
castExpr, realCast, ACTC_retainable, CCK);
|
2011-10-18 02:40:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// stripARCUnbridgedCast - Given an expression of ARCUnbridgedCast
|
|
|
|
/// type, remove the placeholder cast.
|
|
|
|
Expr *Sema::stripARCUnbridgedCast(Expr *e) {
|
|
|
|
assert(e->hasPlaceholderType(BuiltinType::ARCUnbridgedCast));
|
|
|
|
|
|
|
|
if (ParenExpr *pe = dyn_cast<ParenExpr>(e)) {
|
|
|
|
Expr *sub = stripARCUnbridgedCast(pe->getSubExpr());
|
|
|
|
return new (Context) ParenExpr(pe->getLParen(), pe->getRParen(), sub);
|
|
|
|
} else if (UnaryOperator *uo = dyn_cast<UnaryOperator>(e)) {
|
|
|
|
assert(uo->getOpcode() == UO_Extension);
|
|
|
|
Expr *sub = stripARCUnbridgedCast(uo->getSubExpr());
|
2018-01-09 21:07:03 +08:00
|
|
|
return new (Context)
|
|
|
|
UnaryOperator(sub, UO_Extension, sub->getType(), sub->getValueKind(),
|
|
|
|
sub->getObjectKind(), uo->getOperatorLoc(), false);
|
2011-10-18 02:40:02 +08:00
|
|
|
} else if (GenericSelectionExpr *gse = dyn_cast<GenericSelectionExpr>(e)) {
|
|
|
|
assert(!gse->isResultDependent());
|
|
|
|
|
|
|
|
unsigned n = gse->getNumAssocs();
|
2019-01-29 20:57:11 +08:00
|
|
|
SmallVector<Expr *, 4> subExprs;
|
|
|
|
SmallVector<TypeSourceInfo *, 4> subTypes;
|
|
|
|
subExprs.reserve(n);
|
|
|
|
subTypes.reserve(n);
|
|
|
|
for (const GenericSelectionExpr::Association &assoc : gse->associations()) {
|
|
|
|
subTypes.push_back(assoc.getTypeSourceInfo());
|
|
|
|
Expr *sub = assoc.getAssociationExpr();
|
|
|
|
if (assoc.isSelected())
|
2011-10-18 02:40:02 +08:00
|
|
|
sub = stripARCUnbridgedCast(sub);
|
2019-01-29 20:57:11 +08:00
|
|
|
subExprs.push_back(sub);
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2011-10-18 02:40:02 +08:00
|
|
|
|
2019-01-26 22:15:10 +08:00
|
|
|
return GenericSelectionExpr::Create(
|
|
|
|
Context, gse->getGenericLoc(), gse->getControllingExpr(), subTypes,
|
|
|
|
subExprs, gse->getDefaultLoc(), gse->getRParenLoc(),
|
|
|
|
gse->containsUnexpandedParameterPack(), gse->getResultIndex());
|
2011-10-18 02:40:02 +08:00
|
|
|
} else {
|
|
|
|
assert(isa<ImplicitCastExpr>(e) && "bad form of unbridged cast!");
|
|
|
|
return cast<ImplicitCastExpr>(e)->getSubExpr();
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-08 07:04:17 +08:00
|
|
|
bool Sema::CheckObjCARCUnavailableWeakConversion(QualType castType,
|
|
|
|
QualType exprType) {
|
2018-07-31 03:24:48 +08:00
|
|
|
QualType canCastType =
|
2011-07-08 07:04:17 +08:00
|
|
|
Context.getCanonicalType(castType).getUnqualifiedType();
|
2018-07-31 03:24:48 +08:00
|
|
|
QualType canExprType =
|
2011-07-08 07:04:17 +08:00
|
|
|
Context.getCanonicalType(exprType).getUnqualifiedType();
|
|
|
|
if (isa<ObjCObjectPointerType>(canCastType) &&
|
|
|
|
castType.getObjCLifetime() == Qualifiers::OCL_Weak &&
|
|
|
|
canExprType->isObjCObjectPointerType()) {
|
|
|
|
if (const ObjCObjectPointerType *ObjT =
|
|
|
|
canExprType->getAs<ObjCObjectPointerType>())
|
2012-08-23 14:16:52 +08:00
|
|
|
if (const ObjCInterfaceDecl *ObjI = ObjT->getInterfaceDecl())
|
|
|
|
return !ObjI->isArcWeakrefUnavailable();
|
2011-07-08 07:04:17 +08:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-07-07 14:58:02 +08:00
|
|
|
/// Look for an ObjCReclaimReturnedObject cast and destroy it.
|
|
|
|
static Expr *maybeUndoReclaimObject(Expr *e) {
|
2017-10-10 09:24:33 +08:00
|
|
|
Expr *curExpr = e, *prevExpr = nullptr;
|
|
|
|
|
|
|
|
// Walk down the expression until we hit an implicit cast of kind
|
|
|
|
// ARCReclaimReturnedObject or an Expr that is neither a Paren nor a Cast.
|
|
|
|
while (true) {
|
|
|
|
if (auto *pe = dyn_cast<ParenExpr>(curExpr)) {
|
|
|
|
prevExpr = curExpr;
|
|
|
|
curExpr = pe->getSubExpr();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (auto *ce = dyn_cast<CastExpr>(curExpr)) {
|
|
|
|
if (auto *ice = dyn_cast<ImplicitCastExpr>(ce))
|
|
|
|
if (ice->getCastKind() == CK_ARCReclaimReturnedObject) {
|
|
|
|
if (!prevExpr)
|
|
|
|
return ice->getSubExpr();
|
|
|
|
if (auto *pe = dyn_cast<ParenExpr>(prevExpr))
|
|
|
|
pe->setSubExpr(ice->getSubExpr());
|
|
|
|
else
|
|
|
|
cast<CastExpr>(prevExpr)->setSubExpr(ice->getSubExpr());
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
|
|
|
prevExpr = curExpr;
|
|
|
|
curExpr = ce->getSubExpr();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Break out of the loop if curExpr is neither a Paren nor a Cast.
|
|
|
|
break;
|
|
|
|
}
|
2011-07-07 14:58:02 +08:00
|
|
|
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
ExprResult Sema::BuildObjCBridgedCast(SourceLocation LParenLoc,
|
|
|
|
ObjCBridgeCastKind Kind,
|
|
|
|
SourceLocation BridgeKeywordLoc,
|
|
|
|
TypeSourceInfo *TSInfo,
|
|
|
|
Expr *SubExpr) {
|
2011-08-26 08:48:42 +08:00
|
|
|
ExprResult SubResult = UsualUnaryConversions(SubExpr);
|
|
|
|
if (SubResult.isInvalid()) return ExprError();
|
2014-05-29 18:55:11 +08:00
|
|
|
SubExpr = SubResult.get();
|
2011-08-26 08:48:42 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
QualType T = TSInfo->getType();
|
|
|
|
QualType FromType = SubExpr->getType();
|
|
|
|
|
2011-09-09 13:25:32 +08:00
|
|
|
CastKind CK;
|
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
bool MustConsume = false;
|
|
|
|
if (T->isDependentType() || SubExpr->isTypeDependent()) {
|
|
|
|
// Okay: we'll build a dependent expression type.
|
2011-09-09 13:25:32 +08:00
|
|
|
CK = CK_Dependent;
|
2011-06-16 07:02:42 +08:00
|
|
|
} else if (T->isObjCARCBridgableType() && FromType->isCARCBridgableType()) {
|
|
|
|
// Casting CF -> id
|
2011-09-09 13:25:32 +08:00
|
|
|
CK = (T->isBlockPointerType() ? CK_AnyPointerToBlockPointerCast
|
|
|
|
: CK_CPointerToObjCPointerCast);
|
2011-06-16 07:02:42 +08:00
|
|
|
switch (Kind) {
|
|
|
|
case OBC_Bridge:
|
|
|
|
break;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-02-02 06:56:20 +08:00
|
|
|
case OBC_BridgeRetained: {
|
2012-06-01 08:10:47 +08:00
|
|
|
bool br = isKnownName("CFBridgingRelease");
|
2011-06-16 07:02:42 +08:00
|
|
|
Diag(BridgeKeywordLoc, diag::err_arc_bridge_cast_wrong_kind)
|
|
|
|
<< 2
|
|
|
|
<< FromType
|
|
|
|
<< (T->isBlockPointerType()? 1 : 0)
|
|
|
|
<< T
|
|
|
|
<< SubExpr->getSourceRange()
|
|
|
|
<< Kind;
|
|
|
|
Diag(BridgeKeywordLoc, diag::note_arc_bridge)
|
|
|
|
<< FixItHint::CreateReplacement(BridgeKeywordLoc, "__bridge");
|
|
|
|
Diag(BridgeKeywordLoc, diag::note_arc_bridge_transfer)
|
2012-02-02 06:56:20 +08:00
|
|
|
<< FromType << br
|
2018-07-31 03:24:48 +08:00
|
|
|
<< FixItHint::CreateReplacement(BridgeKeywordLoc,
|
|
|
|
br ? "CFBridgingRelease "
|
2012-02-02 06:56:20 +08:00
|
|
|
: "__bridge_transfer ");
|
2011-06-16 07:02:42 +08:00
|
|
|
|
|
|
|
Kind = OBC_Bridge;
|
|
|
|
break;
|
2012-02-02 06:56:20 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
case OBC_BridgeTransfer:
|
|
|
|
// We must consume the Objective-C object produced by the cast.
|
|
|
|
MustConsume = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (T->isCARCBridgableType() && FromType->isObjCARCBridgableType()) {
|
|
|
|
// Okay: id -> CF
|
2011-09-09 13:25:32 +08:00
|
|
|
CK = CK_BitCast;
|
2011-06-16 07:02:42 +08:00
|
|
|
switch (Kind) {
|
2014-10-29 01:26:21 +08:00
|
|
|
case OBC_Bridge:
|
|
|
|
// Reclaiming a value that's going to be __bridge-casted to CF
|
|
|
|
// is very dangerous, so we don't do it.
|
|
|
|
SubExpr = maybeUndoReclaimObject(SubExpr);
|
|
|
|
break;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
|
|
|
case OBC_BridgeRetained:
|
2011-06-16 07:02:42 +08:00
|
|
|
// Produce the object before casting it.
|
|
|
|
SubExpr = ImplicitCastExpr::Create(Context, FromType,
|
2011-09-10 14:18:15 +08:00
|
|
|
CK_ARCProduceObject,
|
2014-05-26 14:22:03 +08:00
|
|
|
SubExpr, nullptr, VK_RValue);
|
2011-06-16 07:02:42 +08:00
|
|
|
break;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2012-02-02 06:56:20 +08:00
|
|
|
case OBC_BridgeTransfer: {
|
2012-06-01 08:10:47 +08:00
|
|
|
bool br = isKnownName("CFBridgingRetain");
|
2011-06-16 07:02:42 +08:00
|
|
|
Diag(BridgeKeywordLoc, diag::err_arc_bridge_cast_wrong_kind)
|
|
|
|
<< (FromType->isBlockPointerType()? 1 : 0)
|
|
|
|
<< FromType
|
|
|
|
<< 2
|
|
|
|
<< T
|
|
|
|
<< SubExpr->getSourceRange()
|
|
|
|
<< Kind;
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
Diag(BridgeKeywordLoc, diag::note_arc_bridge)
|
|
|
|
<< FixItHint::CreateReplacement(BridgeKeywordLoc, "__bridge ");
|
|
|
|
Diag(BridgeKeywordLoc, diag::note_arc_bridge_retained)
|
2012-02-02 06:56:20 +08:00
|
|
|
<< T << br
|
2018-07-31 03:24:48 +08:00
|
|
|
<< FixItHint::CreateReplacement(BridgeKeywordLoc,
|
2012-02-02 06:56:20 +08:00
|
|
|
br ? "CFBridgingRetain " : "__bridge_retained");
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
Kind = OBC_Bridge;
|
|
|
|
break;
|
|
|
|
}
|
2012-02-02 06:56:20 +08:00
|
|
|
}
|
2011-06-16 07:02:42 +08:00
|
|
|
} else {
|
|
|
|
Diag(LParenLoc, diag::err_arc_bridge_cast_incompatible)
|
|
|
|
<< FromType << T << Kind
|
|
|
|
<< SubExpr->getSourceRange()
|
|
|
|
<< TSInfo->getTypeLoc().getSourceRange();
|
|
|
|
return ExprError();
|
|
|
|
}
|
|
|
|
|
2011-09-09 13:25:32 +08:00
|
|
|
Expr *Result = new (Context) ObjCBridgedCastExpr(LParenLoc, Kind, CK,
|
2011-06-16 07:02:42 +08:00
|
|
|
BridgeKeywordLoc,
|
|
|
|
TSInfo, SubExpr);
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
if (MustConsume) {
|
2016-06-22 04:29:17 +08:00
|
|
|
Cleanup.setExprNeedsCleanups(true);
|
2018-07-31 03:24:48 +08:00
|
|
|
Result = ImplicitCastExpr::Create(Context, T, CK_ARCConsumeObject, Result,
|
2014-05-26 14:22:03 +08:00
|
|
|
nullptr, VK_RValue);
|
2011-06-16 07:02:42 +08:00
|
|
|
}
|
2018-07-31 03:24:48 +08:00
|
|
|
|
2011-06-16 07:02:42 +08:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
ExprResult Sema::ActOnObjCBridgedCast(Scope *S,
|
|
|
|
SourceLocation LParenLoc,
|
|
|
|
ObjCBridgeCastKind Kind,
|
|
|
|
SourceLocation BridgeKeywordLoc,
|
|
|
|
ParsedType Type,
|
|
|
|
SourceLocation RParenLoc,
|
|
|
|
Expr *SubExpr) {
|
2014-05-26 14:22:03 +08:00
|
|
|
TypeSourceInfo *TSInfo = nullptr;
|
2011-06-16 07:02:42 +08:00
|
|
|
QualType T = GetTypeFromParser(Type, &TSInfo);
|
2013-11-21 08:39:36 +08:00
|
|
|
if (Kind == OBC_Bridge)
|
|
|
|
CheckTollFreeBridgeCast(T, SubExpr);
|
2011-06-16 07:02:42 +08:00
|
|
|
if (!TSInfo)
|
|
|
|
TSInfo = Context.getTrivialTypeSourceInfo(T, LParenLoc);
|
2018-07-31 03:24:48 +08:00
|
|
|
return BuildObjCBridgedCast(LParenLoc, Kind, BridgeKeywordLoc, TSInfo,
|
2011-06-16 07:02:42 +08:00
|
|
|
SubExpr);
|
|
|
|
}
|