forked from OSchip/llvm-project
Speculatively revert r110610 " Make ObjCInterfaceDecl redeclarable,
and create separate decl nodes for forward declarations and the definition," which appears to be causing significant Objective-C breakage. llvm-svn: 110803
This commit is contained in:
parent
6a98131468
commit
1c28331b57
|
@ -49,11 +49,6 @@ public:
|
||||||
/// elements). Use Decl::getNextDeclarator() to walk the chain.
|
/// elements). Use Decl::getNextDeclarator() to walk the chain.
|
||||||
virtual void HandleTopLevelDecl(DeclGroupRef D);
|
virtual void HandleTopLevelDecl(DeclGroupRef D);
|
||||||
|
|
||||||
/// HandleInterestingDecl - Handle the specified interesting declaration. This
|
|
||||||
/// is called by the PCH reader when deserializing things that might interest
|
|
||||||
/// the consumer. The default implementation forwards to HandleTopLevelDecl.
|
|
||||||
virtual void HandleInterestingDecl(DeclGroupRef D);
|
|
||||||
|
|
||||||
/// HandleTranslationUnit - This method is called when the ASTs for entire
|
/// HandleTranslationUnit - This method is called when the ASTs for entire
|
||||||
/// translation unit have been parsed.
|
/// translation unit have been parsed.
|
||||||
virtual void HandleTranslationUnit(ASTContext &Ctx) {}
|
virtual void HandleTranslationUnit(ASTContext &Ctx) {}
|
||||||
|
|
|
@ -668,10 +668,7 @@ public:
|
||||||
unsigned NumArgs,
|
unsigned NumArgs,
|
||||||
const TemplateArgument *Args);
|
const TemplateArgument *Args);
|
||||||
|
|
||||||
/// getObjCInterfaceType - Return the unique reference to the type for the
|
QualType getObjCInterfaceType(const ObjCInterfaceDecl *Decl);
|
||||||
/// specified ObjC interface decl.
|
|
||||||
QualType getObjCInterfaceType(const ObjCInterfaceDecl *Decl,
|
|
||||||
const ObjCInterfaceDecl *PrevDecl = 0);
|
|
||||||
|
|
||||||
QualType getObjCObjectType(QualType Base,
|
QualType getObjCObjectType(QualType Base,
|
||||||
ObjCProtocolDecl * const *Protocols,
|
ObjCProtocolDecl * const *Protocols,
|
||||||
|
|
|
@ -450,10 +450,7 @@ public:
|
||||||
/// Unlike C++, ObjC is a single-rooted class model. In Cocoa, classes
|
/// Unlike C++, ObjC is a single-rooted class model. In Cocoa, classes
|
||||||
/// typically inherit from NSObject (an exception is NSProxy).
|
/// typically inherit from NSObject (an exception is NSProxy).
|
||||||
///
|
///
|
||||||
class ObjCInterfaceDecl : public ObjCContainerDecl,
|
class ObjCInterfaceDecl : public ObjCContainerDecl {
|
||||||
public Redeclarable<ObjCInterfaceDecl> {
|
|
||||||
typedef Redeclarable<ObjCInterfaceDecl> redeclarable_base;
|
|
||||||
|
|
||||||
/// TypeForDecl - This indicates the Type object that represents this
|
/// TypeForDecl - This indicates the Type object that represents this
|
||||||
/// TypeDecl. It is a cache maintained by ASTContext::getObjCInterfaceType
|
/// TypeDecl. It is a cache maintained by ASTContext::getObjCInterfaceType
|
||||||
mutable Type *TypeForDecl;
|
mutable Type *TypeForDecl;
|
||||||
|
@ -477,52 +474,15 @@ class ObjCInterfaceDecl : public ObjCContainerDecl,
|
||||||
SourceLocation EndLoc; // marks the '>', '}', or identifier.
|
SourceLocation EndLoc; // marks the '>', '}', or identifier.
|
||||||
|
|
||||||
ObjCInterfaceDecl(DeclContext *DC, SourceLocation atLoc, IdentifierInfo *Id,
|
ObjCInterfaceDecl(DeclContext *DC, SourceLocation atLoc, IdentifierInfo *Id,
|
||||||
SourceLocation CLoc, ObjCInterfaceDecl *PrevDecl,
|
SourceLocation CLoc, bool FD, bool isInternal);
|
||||||
bool FD, bool isInternal);
|
|
||||||
|
|
||||||
protected:
|
|
||||||
virtual ObjCInterfaceDecl *getNextRedeclaration() {
|
|
||||||
return RedeclLink.getNext();
|
|
||||||
}
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
typedef redeclarable_base::redecl_iterator redecl_iterator;
|
|
||||||
redecl_iterator redecls_begin() const {
|
|
||||||
return redeclarable_base::redecls_begin();
|
|
||||||
}
|
|
||||||
redecl_iterator redecls_end() const {
|
|
||||||
return redeclarable_base::redecls_end();
|
|
||||||
}
|
|
||||||
|
|
||||||
static ObjCInterfaceDecl *Create(ASTContext &C, DeclContext *DC,
|
static ObjCInterfaceDecl *Create(ASTContext &C, DeclContext *DC,
|
||||||
SourceLocation atLoc,
|
SourceLocation atLoc,
|
||||||
IdentifierInfo *Id,
|
IdentifierInfo *Id,
|
||||||
SourceLocation ClassLoc,
|
SourceLocation ClassLoc = SourceLocation(),
|
||||||
ObjCInterfaceDecl *PrevDecl,
|
|
||||||
bool ForwardDecl = false,
|
bool ForwardDecl = false,
|
||||||
bool isInternal = false);
|
bool isInternal = false);
|
||||||
|
|
||||||
static ObjCInterfaceDecl *Create(ASTContext &C, EmptyShell Empty);
|
|
||||||
|
|
||||||
void setPreviousDeclaration(ObjCInterfaceDecl *Prev) {
|
|
||||||
// Adopt a type pointer if it exists.
|
|
||||||
TypeForDecl = Prev->TypeForDecl;
|
|
||||||
redeclarable_base::setPreviousDeclaration(Prev);
|
|
||||||
}
|
|
||||||
|
|
||||||
virtual ObjCInterfaceDecl *getCanonicalDecl() {
|
|
||||||
return getFirstDeclaration();
|
|
||||||
}
|
|
||||||
const ObjCInterfaceDecl *getCanonicalDecl() const {
|
|
||||||
return const_cast<ObjCInterfaceDecl*>(this)->getCanonicalDecl();
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Get the interface declaration that is a definition, if there is one.
|
|
||||||
ObjCInterfaceDecl *getDefinition();
|
|
||||||
const ObjCInterfaceDecl *getDefinition() const {
|
|
||||||
return const_cast<ObjCInterfaceDecl*>(this)->getDefinition();
|
|
||||||
}
|
|
||||||
|
|
||||||
const ObjCProtocolList &getReferencedProtocols() const {
|
const ObjCProtocolList &getReferencedProtocols() const {
|
||||||
return ReferencedProtocols;
|
return ReferencedProtocols;
|
||||||
}
|
}
|
||||||
|
@ -575,7 +535,6 @@ public:
|
||||||
ASTContext &C);
|
ASTContext &C);
|
||||||
|
|
||||||
bool isForwardDecl() const { return ForwardDecl; }
|
bool isForwardDecl() const { return ForwardDecl; }
|
||||||
bool isDefinition() const { return !ForwardDecl; }
|
|
||||||
void setForwardDecl(bool val) { ForwardDecl = val; }
|
void setForwardDecl(bool val) { ForwardDecl = val; }
|
||||||
|
|
||||||
ObjCInterfaceDecl *getSuperClass() const { return SuperClass; }
|
ObjCInterfaceDecl *getSuperClass() const { return SuperClass; }
|
||||||
|
|
|
@ -3038,7 +3038,7 @@ class ObjCInterfaceType : public ObjCObjectType {
|
||||||
friend class ASTContext; // ASTContext creates these.
|
friend class ASTContext; // ASTContext creates these.
|
||||||
public:
|
public:
|
||||||
/// getDecl - Get the declaration of this interface.
|
/// getDecl - Get the declaration of this interface.
|
||||||
ObjCInterfaceDecl *getDecl() const;
|
ObjCInterfaceDecl *getDecl() const { return Decl; }
|
||||||
|
|
||||||
bool isSugared() const { return false; }
|
bool isSugared() const { return false; }
|
||||||
QualType desugar() const { return QualType(this, 0); }
|
QualType desugar() const { return QualType(this, 0); }
|
||||||
|
|
|
@ -17,6 +17,3 @@ using namespace clang;
|
||||||
|
|
||||||
void ASTConsumer::HandleTopLevelDecl(DeclGroupRef D) {}
|
void ASTConsumer::HandleTopLevelDecl(DeclGroupRef D) {}
|
||||||
|
|
||||||
void ASTConsumer::HandleInterestingDecl(DeclGroupRef D) {
|
|
||||||
HandleTopLevelDecl(D);
|
|
||||||
}
|
|
||||||
|
|
|
@ -872,7 +872,7 @@ void ASTContext::ShallowCollectObjCIvars(const ObjCInterfaceDecl *OI,
|
||||||
/// CollectNonClassIvars -
|
/// CollectNonClassIvars -
|
||||||
/// This routine collects all other ivars which are not declared in the class.
|
/// This routine collects all other ivars which are not declared in the class.
|
||||||
/// This includes synthesized ivars (via @synthesize) and those in
|
/// This includes synthesized ivars (via @synthesize) and those in
|
||||||
/// class's @implementation.
|
// class's @implementation.
|
||||||
///
|
///
|
||||||
void ASTContext::CollectNonClassIvars(const ObjCInterfaceDecl *OI,
|
void ASTContext::CollectNonClassIvars(const ObjCInterfaceDecl *OI,
|
||||||
llvm::SmallVectorImpl<ObjCIvarDecl*> &Ivars) {
|
llvm::SmallVectorImpl<ObjCIvarDecl*> &Ivars) {
|
||||||
|
@ -2212,24 +2212,18 @@ QualType ASTContext::getObjCObjectPointerType(QualType ObjectT) {
|
||||||
return QualType(QType, 0);
|
return QualType(QType, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
QualType ASTContext::getObjCInterfaceType(const ObjCInterfaceDecl *Decl,
|
/// getObjCInterfaceType - Return the unique reference to the type for the
|
||||||
const ObjCInterfaceDecl *PrevDecl) {
|
/// specified ObjC interface decl. The list of protocols is optional.
|
||||||
assert(Decl && "Passed null for Decl param");
|
QualType ASTContext::getObjCInterfaceType(const ObjCInterfaceDecl *Decl) {
|
||||||
|
if (Decl->TypeForDecl)
|
||||||
if (Decl->TypeForDecl) return QualType(Decl->TypeForDecl, 0);
|
|
||||||
|
|
||||||
if (PrevDecl) {
|
|
||||||
assert(PrevDecl->TypeForDecl && "previous decl has no TypeForDecl");
|
|
||||||
Decl->TypeForDecl = PrevDecl->TypeForDecl;
|
|
||||||
return QualType(PrevDecl->TypeForDecl, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(!Decl->getPreviousDeclaration() &&
|
|
||||||
"interface has previous declaration");
|
|
||||||
|
|
||||||
Decl->TypeForDecl = new (*this, TypeAlignment) ObjCInterfaceType(Decl);
|
|
||||||
Types.push_back(Decl->TypeForDecl);
|
|
||||||
return QualType(Decl->TypeForDecl, 0);
|
return QualType(Decl->TypeForDecl, 0);
|
||||||
|
|
||||||
|
// FIXME: redeclarations?
|
||||||
|
void *Mem = Allocate(sizeof(ObjCInterfaceType), TypeAlignment);
|
||||||
|
ObjCInterfaceType *T = new (Mem) ObjCInterfaceType(Decl);
|
||||||
|
Decl->TypeForDecl = T;
|
||||||
|
Types.push_back(T);
|
||||||
|
return QualType(T, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// getTypeOfExprType - Unlike many "get<Type>" functions, we can't unique
|
/// getTypeOfExprType - Unlike many "get<Type>" functions, we can't unique
|
||||||
|
|
|
@ -2467,18 +2467,6 @@ Decl *ASTNodeImporter::VisitObjCProtocolDecl(ObjCProtocolDecl *D) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Decl *ASTNodeImporter::VisitObjCInterfaceDecl(ObjCInterfaceDecl *D) {
|
Decl *ASTNodeImporter::VisitObjCInterfaceDecl(ObjCInterfaceDecl *D) {
|
||||||
// If this interface has a definition in the translation unit we're coming
|
|
||||||
// from, but this particular declaration is not that definition, import the
|
|
||||||
// definition and map to that.
|
|
||||||
ObjCInterfaceDecl *Definition = D->getDefinition();
|
|
||||||
if (Definition && Definition != D) {
|
|
||||||
Decl *ImportedDef = Importer.Import(Definition);
|
|
||||||
if (!ImportedDef)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
return Importer.Imported(D, ImportedDef);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Import the major distinguishing characteristics of an @interface.
|
// Import the major distinguishing characteristics of an @interface.
|
||||||
DeclContext *DC, *LexicalDC;
|
DeclContext *DC, *LexicalDC;
|
||||||
DeclarationName Name;
|
DeclarationName Name;
|
||||||
|
@ -2503,7 +2491,7 @@ Decl *ASTNodeImporter::VisitObjCInterfaceDecl(ObjCInterfaceDecl *D) {
|
||||||
ToIface = ObjCInterfaceDecl::Create(Importer.getToContext(),
|
ToIface = ObjCInterfaceDecl::Create(Importer.getToContext(),
|
||||||
DC, Loc,
|
DC, Loc,
|
||||||
Name.getAsIdentifierInfo(),
|
Name.getAsIdentifierInfo(),
|
||||||
Importer.Import(D->getClassLoc()), 0,
|
Importer.Import(D->getClassLoc()),
|
||||||
D->isForwardDecl(),
|
D->isForwardDecl(),
|
||||||
D->isImplicitInterfaceDecl());
|
D->isImplicitInterfaceDecl());
|
||||||
ToIface->setForwardDecl(D->isForwardDecl());
|
ToIface->setForwardDecl(D->isForwardDecl());
|
||||||
|
|
|
@ -45,14 +45,6 @@ void ObjCProtocolList::set(ObjCProtocolDecl* const* InList, unsigned Elts,
|
||||||
// ObjCInterfaceDecl
|
// ObjCInterfaceDecl
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
ObjCInterfaceDecl *ObjCInterfaceDecl::getDefinition() {
|
|
||||||
for (redecl_iterator I = redecls_begin(), E = redecls_end(); I != E; ++I) {
|
|
||||||
if (I->isDefinition())
|
|
||||||
return *I;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// getIvarDecl - This method looks up an ivar in this ContextDecl.
|
/// getIvarDecl - This method looks up an ivar in this ContextDecl.
|
||||||
///
|
///
|
||||||
ObjCIvarDecl *
|
ObjCIvarDecl *
|
||||||
|
@ -440,30 +432,18 @@ ObjCInterfaceDecl *ObjCInterfaceDecl::Create(ASTContext &C,
|
||||||
SourceLocation atLoc,
|
SourceLocation atLoc,
|
||||||
IdentifierInfo *Id,
|
IdentifierInfo *Id,
|
||||||
SourceLocation ClassLoc,
|
SourceLocation ClassLoc,
|
||||||
ObjCInterfaceDecl *PrevDecl,
|
|
||||||
bool ForwardDecl, bool isInternal){
|
bool ForwardDecl, bool isInternal){
|
||||||
ObjCInterfaceDecl *D = new (C) ObjCInterfaceDecl(DC, atLoc, Id, ClassLoc,
|
return new (C) ObjCInterfaceDecl(DC, atLoc, Id, ClassLoc, ForwardDecl,
|
||||||
PrevDecl, ForwardDecl,
|
|
||||||
isInternal);
|
isInternal);
|
||||||
C.getObjCInterfaceType(D, PrevDecl);
|
|
||||||
return D;
|
|
||||||
}
|
|
||||||
|
|
||||||
ObjCInterfaceDecl *ObjCInterfaceDecl::Create(ASTContext &C, EmptyShell) {
|
|
||||||
return new (C) ObjCInterfaceDecl(0, SourceLocation(), 0, SourceLocation(),
|
|
||||||
0, false, false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ObjCInterfaceDecl::
|
ObjCInterfaceDecl::
|
||||||
ObjCInterfaceDecl(DeclContext *DC, SourceLocation atLoc, IdentifierInfo *Id,
|
ObjCInterfaceDecl(DeclContext *DC, SourceLocation atLoc, IdentifierInfo *Id,
|
||||||
SourceLocation CLoc, ObjCInterfaceDecl *PrevDecl,
|
SourceLocation CLoc, bool FD, bool isInternal)
|
||||||
bool FD, bool isInternal)
|
|
||||||
: ObjCContainerDecl(ObjCInterface, DC, atLoc, Id),
|
: ObjCContainerDecl(ObjCInterface, DC, atLoc, Id),
|
||||||
TypeForDecl(0), SuperClass(0),
|
TypeForDecl(0), SuperClass(0),
|
||||||
CategoryList(0), ForwardDecl(FD), InternalInterface(isInternal),
|
CategoryList(0), ForwardDecl(FD), InternalInterface(isInternal),
|
||||||
ClassLoc(CLoc) {
|
ClassLoc(CLoc) {
|
||||||
if (PrevDecl)
|
|
||||||
setPreviousDeclaration(PrevDecl);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ObjCImplementationDecl *ObjCInterfaceDecl::getImplementation() const {
|
ObjCImplementationDecl *ObjCInterfaceDecl::getImplementation() const {
|
||||||
|
|
|
@ -727,19 +727,12 @@ void DeclPrinter::VisitObjCImplementationDecl(ObjCImplementationDecl *OID) {
|
||||||
|
|
||||||
void DeclPrinter::VisitObjCInterfaceDecl(ObjCInterfaceDecl *OID) {
|
void DeclPrinter::VisitObjCInterfaceDecl(ObjCInterfaceDecl *OID) {
|
||||||
std::string I = OID->getNameAsString();
|
std::string I = OID->getNameAsString();
|
||||||
|
|
||||||
if (OID->isForwardDecl()) {
|
|
||||||
// These shouldn't be directly visited, but in case they are, write them
|
|
||||||
// as an @class declaration.
|
|
||||||
Out << "@class " << I;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
ObjCInterfaceDecl *SID = OID->getSuperClass();
|
ObjCInterfaceDecl *SID = OID->getSuperClass();
|
||||||
|
|
||||||
Out << "@interface " << I;
|
|
||||||
if (SID)
|
if (SID)
|
||||||
Out << " : " << SID;
|
Out << "@interface " << I << " : " << SID;
|
||||||
|
else
|
||||||
|
Out << "@interface " << I;
|
||||||
|
|
||||||
// Protocols?
|
// Protocols?
|
||||||
const ObjCList<ObjCProtocolDecl> &Protocols = OID->getReferencedProtocols();
|
const ObjCList<ObjCProtocolDecl> &Protocols = OID->getReferencedProtocols();
|
||||||
|
|
|
@ -360,17 +360,6 @@ const ObjCObjectPointerType *Type::getAsObjCInterfacePointerType() const {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
ObjCInterfaceDecl *ObjCInterfaceType::getDecl() const {
|
|
||||||
for (ObjCInterfaceDecl::redecl_iterator I = Decl->redecls_begin(),
|
|
||||||
E = Decl->redecls_end();
|
|
||||||
I != E; ++I) {
|
|
||||||
if (I->isDefinition())
|
|
||||||
return *I;
|
|
||||||
}
|
|
||||||
// If we can't find a definition, return whatever we have.
|
|
||||||
return Decl;
|
|
||||||
}
|
|
||||||
|
|
||||||
const CXXRecordDecl *Type::getCXXRecordDeclForPointerType() const {
|
const CXXRecordDecl *Type::getCXXRecordDeclForPointerType() const {
|
||||||
if (const PointerType *PT = getAs<PointerType>())
|
if (const PointerType *PT = getAs<PointerType>())
|
||||||
if (const RecordType *RT = PT->getPointeeType()->getAs<RecordType>())
|
if (const RecordType *RT = PT->getPointeeType()->getAs<RecordType>())
|
||||||
|
|
|
@ -320,9 +320,6 @@ public:
|
||||||
Unit.addTopLevelDecl(D);
|
Unit.addTopLevelDecl(D);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// We're not interested in "interesting" decls.
|
|
||||||
void HandleInterestingDecl(DeclGroupRef) {}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class TopLevelDeclTrackerAction : public ASTFrontendAction {
|
class TopLevelDeclTrackerAction : public ASTFrontendAction {
|
||||||
|
|
|
@ -3027,7 +3027,7 @@ void PCHReader::PassInterestingDeclsToConsumer() {
|
||||||
while (!InterestingDecls.empty()) {
|
while (!InterestingDecls.empty()) {
|
||||||
DeclGroupRef DG(InterestingDecls.front());
|
DeclGroupRef DG(InterestingDecls.front());
|
||||||
InterestingDecls.pop_front();
|
InterestingDecls.pop_front();
|
||||||
Consumer->HandleInterestingDecl(DG);
|
Consumer->HandleTopLevelDecl(DG);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -34,7 +34,7 @@ namespace clang {
|
||||||
const pch::DeclID ThisDeclID;
|
const pch::DeclID ThisDeclID;
|
||||||
const PCHReader::RecordData &Record;
|
const PCHReader::RecordData &Record;
|
||||||
unsigned &Idx;
|
unsigned &Idx;
|
||||||
pch::TypeID TypeIDForDecl;
|
pch::TypeID TypeIDForTypeDecl;
|
||||||
|
|
||||||
uint64_t GetCurrentCursorOffset();
|
uint64_t GetCurrentCursorOffset();
|
||||||
|
|
||||||
|
@ -43,7 +43,7 @@ namespace clang {
|
||||||
pch::DeclID thisDeclID, const PCHReader::RecordData &Record,
|
pch::DeclID thisDeclID, const PCHReader::RecordData &Record,
|
||||||
unsigned &Idx)
|
unsigned &Idx)
|
||||||
: Reader(Reader), Cursor(Cursor), ThisDeclID(thisDeclID), Record(Record),
|
: Reader(Reader), Cursor(Cursor), ThisDeclID(thisDeclID), Record(Record),
|
||||||
Idx(Idx), TypeIDForDecl(0) { }
|
Idx(Idx), TypeIDForTypeDecl(0) { }
|
||||||
|
|
||||||
void Visit(Decl *D);
|
void Visit(Decl *D);
|
||||||
|
|
||||||
|
@ -132,11 +132,9 @@ uint64_t PCHDeclReader::GetCurrentCursorOffset() {
|
||||||
void PCHDeclReader::Visit(Decl *D) {
|
void PCHDeclReader::Visit(Decl *D) {
|
||||||
DeclVisitor<PCHDeclReader, void>::Visit(D);
|
DeclVisitor<PCHDeclReader, void>::Visit(D);
|
||||||
|
|
||||||
// if we have a fully initialized Decl, we can safely read its type now.
|
|
||||||
if (TypeDecl *TD = dyn_cast<TypeDecl>(D)) {
|
if (TypeDecl *TD = dyn_cast<TypeDecl>(D)) {
|
||||||
TD->setTypeForDecl(Reader.GetType(TypeIDForDecl).getTypePtr());
|
// if we have a fully initialized TypeDecl, we can safely read its type now.
|
||||||
} else if (ObjCInterfaceDecl *ID = dyn_cast<ObjCInterfaceDecl>(D)) {
|
TD->setTypeForDecl(Reader.GetType(TypeIDForTypeDecl).getTypePtr());
|
||||||
ID->setTypeForDecl(Reader.GetType(TypeIDForDecl).getTypePtr());
|
|
||||||
} else if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
|
} else if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
|
||||||
// FunctionDecl's body was written last after all other Stmts/Exprs.
|
// FunctionDecl's body was written last after all other Stmts/Exprs.
|
||||||
if (Record[Idx++])
|
if (Record[Idx++])
|
||||||
|
@ -172,7 +170,7 @@ void PCHDeclReader::VisitNamedDecl(NamedDecl *ND) {
|
||||||
void PCHDeclReader::VisitTypeDecl(TypeDecl *TD) {
|
void PCHDeclReader::VisitTypeDecl(TypeDecl *TD) {
|
||||||
VisitNamedDecl(TD);
|
VisitNamedDecl(TD);
|
||||||
// Delay type reading until after we have fully initialized the decl.
|
// Delay type reading until after we have fully initialized the decl.
|
||||||
TypeIDForDecl = Record[Idx++];
|
TypeIDForTypeDecl = Record[Idx++];
|
||||||
}
|
}
|
||||||
|
|
||||||
void PCHDeclReader::VisitTypedefDecl(TypedefDecl *TD) {
|
void PCHDeclReader::VisitTypedefDecl(TypedefDecl *TD) {
|
||||||
|
@ -369,11 +367,7 @@ void PCHDeclReader::VisitObjCContainerDecl(ObjCContainerDecl *CD) {
|
||||||
|
|
||||||
void PCHDeclReader::VisitObjCInterfaceDecl(ObjCInterfaceDecl *ID) {
|
void PCHDeclReader::VisitObjCInterfaceDecl(ObjCInterfaceDecl *ID) {
|
||||||
VisitObjCContainerDecl(ID);
|
VisitObjCContainerDecl(ID);
|
||||||
ID->setForwardDecl(Record[Idx++]);
|
ID->setTypeForDecl(Reader.GetType(Record[Idx++]).getTypePtr());
|
||||||
ID->setImplicitInterfaceDecl(Record[Idx++]);
|
|
||||||
VisitRedeclarable(ID);
|
|
||||||
// Must delay type reading until the redecl chain is complete.
|
|
||||||
TypeIDForDecl = Record[Idx++];
|
|
||||||
ID->setSuperClass(cast_or_null<ObjCInterfaceDecl>
|
ID->setSuperClass(cast_or_null<ObjCInterfaceDecl>
|
||||||
(Reader.GetDecl(Record[Idx++])));
|
(Reader.GetDecl(Record[Idx++])));
|
||||||
unsigned NumProtocols = Record[Idx++];
|
unsigned NumProtocols = Record[Idx++];
|
||||||
|
@ -394,6 +388,8 @@ void PCHDeclReader::VisitObjCInterfaceDecl(ObjCInterfaceDecl *ID) {
|
||||||
IVars.push_back(cast<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++])));
|
IVars.push_back(cast<ObjCIvarDecl>(Reader.GetDecl(Record[Idx++])));
|
||||||
ID->setCategoryList(
|
ID->setCategoryList(
|
||||||
cast_or_null<ObjCCategoryDecl>(Reader.GetDecl(Record[Idx++])));
|
cast_or_null<ObjCCategoryDecl>(Reader.GetDecl(Record[Idx++])));
|
||||||
|
ID->setForwardDecl(Record[Idx++]);
|
||||||
|
ID->setImplicitInterfaceDecl(Record[Idx++]);
|
||||||
ID->setClassLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
ID->setClassLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
||||||
ID->setSuperClassLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
ID->setSuperClassLoc(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
||||||
ID->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
ID->setLocEnd(SourceLocation::getFromRawEncoding(Record[Idx++]));
|
||||||
|
@ -1463,7 +1459,7 @@ Decl *PCHReader::ReadDeclRecord(unsigned Index, pch::DeclID ID) {
|
||||||
Selector(), QualType(), 0, 0);
|
Selector(), QualType(), 0, 0);
|
||||||
break;
|
break;
|
||||||
case pch::DECL_OBJC_INTERFACE:
|
case pch::DECL_OBJC_INTERFACE:
|
||||||
D = ObjCInterfaceDecl::Create(*Context, Decl::EmptyShell());
|
D = ObjCInterfaceDecl::Create(*Context, 0, SourceLocation(), 0);
|
||||||
break;
|
break;
|
||||||
case pch::DECL_OBJC_IVAR:
|
case pch::DECL_OBJC_IVAR:
|
||||||
D = ObjCIvarDecl::Create(*Context, 0, SourceLocation(), 0, QualType(), 0,
|
D = ObjCIvarDecl::Create(*Context, 0, SourceLocation(), 0, QualType(), 0,
|
||||||
|
|
|
@ -305,9 +305,7 @@ void PCHTypeWriter::VisitInjectedClassNameType(const InjectedClassNameType *T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void PCHTypeWriter::VisitObjCInterfaceType(const ObjCInterfaceType *T) {
|
void PCHTypeWriter::VisitObjCInterfaceType(const ObjCInterfaceType *T) {
|
||||||
// The stored declaration must be the first, but getDecl() returns the
|
Writer.AddDeclRef(T->getDecl(), Record);
|
||||||
// definition.
|
|
||||||
Writer.AddDeclRef(T->getDecl()->getFirstDeclaration(), Record);
|
|
||||||
Code = pch::TYPE_OBJC_INTERFACE;
|
Code = pch::TYPE_OBJC_INTERFACE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -340,9 +340,6 @@ void PCHDeclWriter::VisitObjCContainerDecl(ObjCContainerDecl *D) {
|
||||||
|
|
||||||
void PCHDeclWriter::VisitObjCInterfaceDecl(ObjCInterfaceDecl *D) {
|
void PCHDeclWriter::VisitObjCInterfaceDecl(ObjCInterfaceDecl *D) {
|
||||||
VisitObjCContainerDecl(D);
|
VisitObjCContainerDecl(D);
|
||||||
Record.push_back(D->isForwardDecl());
|
|
||||||
Record.push_back(D->isImplicitInterfaceDecl());
|
|
||||||
VisitRedeclarable(D);
|
|
||||||
Writer.AddTypeRef(QualType(D->getTypeForDecl(), 0), Record);
|
Writer.AddTypeRef(QualType(D->getTypeForDecl(), 0), Record);
|
||||||
Writer.AddDeclRef(D->getSuperClass(), Record);
|
Writer.AddDeclRef(D->getSuperClass(), Record);
|
||||||
Record.push_back(D->protocol_size());
|
Record.push_back(D->protocol_size());
|
||||||
|
@ -359,6 +356,8 @@ void PCHDeclWriter::VisitObjCInterfaceDecl(ObjCInterfaceDecl *D) {
|
||||||
IEnd = D->ivar_end(); I != IEnd; ++I)
|
IEnd = D->ivar_end(); I != IEnd; ++I)
|
||||||
Writer.AddDeclRef(*I, Record);
|
Writer.AddDeclRef(*I, Record);
|
||||||
Writer.AddDeclRef(D->getCategoryList(), Record);
|
Writer.AddDeclRef(D->getCategoryList(), Record);
|
||||||
|
Record.push_back(D->isForwardDecl());
|
||||||
|
Record.push_back(D->isImplicitInterfaceDecl());
|
||||||
Writer.AddSourceLocation(D->getClassLoc(), Record);
|
Writer.AddSourceLocation(D->getClassLoc(), Record);
|
||||||
Writer.AddSourceLocation(D->getSuperClassLoc(), Record);
|
Writer.AddSourceLocation(D->getSuperClassLoc(), Record);
|
||||||
Writer.AddSourceLocation(D->getLocEnd(), Record);
|
Writer.AddSourceLocation(D->getLocEnd(), Record);
|
||||||
|
|
|
@ -89,7 +89,7 @@ void Sema::ActOnTranslationUnitScope(SourceLocation Loc, Scope *S) {
|
||||||
ObjCInterfaceDecl *ProtocolDecl =
|
ObjCInterfaceDecl *ProtocolDecl =
|
||||||
ObjCInterfaceDecl::Create(Context, CurContext, SourceLocation(),
|
ObjCInterfaceDecl::Create(Context, CurContext, SourceLocation(),
|
||||||
&Context.Idents.get("Protocol"),
|
&Context.Idents.get("Protocol"),
|
||||||
SourceLocation(), 0, true);
|
SourceLocation(), true);
|
||||||
Context.setObjCProtoType(Context.getObjCInterfaceType(ProtocolDecl));
|
Context.setObjCProtoType(Context.getObjCInterfaceType(ProtocolDecl));
|
||||||
PushOnScopeChains(ProtocolDecl, TUScope, false);
|
PushOnScopeChains(ProtocolDecl, TUScope, false);
|
||||||
}
|
}
|
||||||
|
|
|
@ -3748,31 +3748,23 @@ static void AddInterfaceResults(DeclContext *Ctx, DeclContext *CurContext,
|
||||||
for (DeclContext::decl_iterator D = Ctx->decls_begin(),
|
for (DeclContext::decl_iterator D = Ctx->decls_begin(),
|
||||||
DEnd = Ctx->decls_end();
|
DEnd = Ctx->decls_end();
|
||||||
D != DEnd; ++D) {
|
D != DEnd; ++D) {
|
||||||
// Record any interfaces we find. Forward declarations are never registered
|
// Record any interfaces we find.
|
||||||
// in the lexical contest, so if we're only looking for those, don't bother.
|
|
||||||
if (!OnlyForwardDeclarations)
|
|
||||||
if (ObjCInterfaceDecl *Class = dyn_cast<ObjCInterfaceDecl>(*D))
|
if (ObjCInterfaceDecl *Class = dyn_cast<ObjCInterfaceDecl>(*D))
|
||||||
if (!OnlyUnimplemented || !Class->getImplementation())
|
if ((!OnlyForwardDeclarations || Class->isForwardDecl()) &&
|
||||||
|
(!OnlyUnimplemented || !Class->getImplementation()))
|
||||||
Results.AddResult(Result(Class, 0), CurContext, 0, false);
|
Results.AddResult(Result(Class, 0), CurContext, 0, false);
|
||||||
|
|
||||||
// Record any forward-declared interfaces we find.
|
// Record any forward-declared interfaces we find.
|
||||||
if (ObjCClassDecl *Forward = dyn_cast<ObjCClassDecl>(*D)) {
|
if (ObjCClassDecl *Forward = dyn_cast<ObjCClassDecl>(*D)) {
|
||||||
for (ObjCClassDecl::iterator C = Forward->begin(), CEnd = Forward->end();
|
for (ObjCClassDecl::iterator C = Forward->begin(), CEnd = Forward->end();
|
||||||
C != CEnd; ++C) {
|
C != CEnd; ++C)
|
||||||
ObjCInterfaceDecl *IDecl = C->getInterface();
|
if ((!OnlyForwardDeclarations || C->getInterface()->isForwardDecl()) &&
|
||||||
ObjCInterfaceDecl *IDef = IDecl->getDefinition();
|
(!OnlyUnimplemented || !C->getInterface()->getImplementation()))
|
||||||
// If there's a definition, and we're looking for everything, then we
|
Results.AddResult(Result(C->getInterface(), 0), CurContext,
|
||||||
// already added the decl in question above.
|
|
||||||
if (!OnlyForwardDeclarations && !OnlyUnimplemented && IDef)
|
|
||||||
continue;
|
|
||||||
if ((!OnlyForwardDeclarations || !IDef) &&
|
|
||||||
(!OnlyUnimplemented || !IDef || !IDef->getImplementation()))
|
|
||||||
Results.AddResult(Result(IDecl, 0), CurContext,
|
|
||||||
0, false);
|
0, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
void Sema::CodeCompleteObjCInterfaceDecl(Scope *S) {
|
void Sema::CodeCompleteObjCInterfaceDecl(Scope *S) {
|
||||||
ResultBuilder Results(*this);
|
ResultBuilder Results(*this);
|
||||||
|
|
|
@ -606,6 +606,45 @@ void Sema::ActOnPopScope(SourceLocation Loc, Scope *S) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// \brief Look for an Objective-C class in the translation unit.
|
||||||
|
///
|
||||||
|
/// \param Id The name of the Objective-C class we're looking for. If
|
||||||
|
/// typo-correction fixes this name, the Id will be updated
|
||||||
|
/// to the fixed name.
|
||||||
|
///
|
||||||
|
/// \param IdLoc The location of the name in the translation unit.
|
||||||
|
///
|
||||||
|
/// \param TypoCorrection If true, this routine will attempt typo correction
|
||||||
|
/// if there is no class with the given name.
|
||||||
|
///
|
||||||
|
/// \returns The declaration of the named Objective-C class, or NULL if the
|
||||||
|
/// class could not be found.
|
||||||
|
ObjCInterfaceDecl *Sema::getObjCInterfaceDecl(IdentifierInfo *&Id,
|
||||||
|
SourceLocation IdLoc,
|
||||||
|
bool TypoCorrection) {
|
||||||
|
// The third "scope" argument is 0 since we aren't enabling lazy built-in
|
||||||
|
// creation from this context.
|
||||||
|
NamedDecl *IDecl = LookupSingleName(TUScope, Id, IdLoc, LookupOrdinaryName);
|
||||||
|
|
||||||
|
if (!IDecl && TypoCorrection) {
|
||||||
|
// Perform typo correction at the given location, but only if we
|
||||||
|
// find an Objective-C class name.
|
||||||
|
LookupResult R(*this, Id, IdLoc, LookupOrdinaryName);
|
||||||
|
if (CorrectTypo(R, TUScope, 0, 0, false, CTC_NoKeywords) &&
|
||||||
|
(IDecl = R.getAsSingle<ObjCInterfaceDecl>())) {
|
||||||
|
Diag(IdLoc, diag::err_undef_interface_suggest)
|
||||||
|
<< Id << IDecl->getDeclName()
|
||||||
|
<< FixItHint::CreateReplacement(IdLoc, IDecl->getNameAsString());
|
||||||
|
Diag(IDecl->getLocation(), diag::note_previous_decl)
|
||||||
|
<< IDecl->getDeclName();
|
||||||
|
|
||||||
|
Id = IDecl->getIdentifier();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return dyn_cast_or_null<ObjCInterfaceDecl>(IDecl);
|
||||||
|
}
|
||||||
|
|
||||||
/// getNonFieldDeclScope - Retrieves the innermost scope, starting
|
/// getNonFieldDeclScope - Retrieves the innermost scope, starting
|
||||||
/// from S, where a non-field would be declared. This routine copes
|
/// from S, where a non-field would be declared. This routine copes
|
||||||
/// with the difference between C and C++ scoping rules in structs and
|
/// with the difference between C and C++ scoping rules in structs and
|
||||||
|
|
|
@ -20,50 +20,6 @@
|
||||||
#include "clang/Parse/DeclSpec.h"
|
#include "clang/Parse/DeclSpec.h"
|
||||||
using namespace clang;
|
using namespace clang;
|
||||||
|
|
||||||
/// \brief Look for an Objective-C class in the translation unit.
|
|
||||||
///
|
|
||||||
/// \param Id The name of the Objective-C class we're looking for. If
|
|
||||||
/// typo-correction fixes this name, the Id will be updated
|
|
||||||
/// to the fixed name.
|
|
||||||
///
|
|
||||||
/// \param IdLoc The location of the name in the translation unit.
|
|
||||||
///
|
|
||||||
/// \param TypoCorrection If true, this routine will attempt typo correction
|
|
||||||
/// if there is no class with the given name.
|
|
||||||
///
|
|
||||||
/// \returns The declaration of the named Objective-C class, which is also the
|
|
||||||
/// definition if one is available, or NULL if the class could not be found.
|
|
||||||
ObjCInterfaceDecl *Sema::getObjCInterfaceDecl(IdentifierInfo *&Id,
|
|
||||||
SourceLocation IdLoc,
|
|
||||||
bool TypoCorrection) {
|
|
||||||
// The third "scope" argument is 0 since we aren't enabling lazy built-in
|
|
||||||
// creation from this context.
|
|
||||||
NamedDecl *Decl = LookupSingleName(TUScope, Id, IdLoc, LookupOrdinaryName);
|
|
||||||
|
|
||||||
if (!Decl && TypoCorrection) {
|
|
||||||
// Perform typo correction at the given location, but only if we
|
|
||||||
// find an Objective-C class name.
|
|
||||||
LookupResult R(*this, Id, IdLoc, LookupOrdinaryName);
|
|
||||||
if (CorrectTypo(R, TUScope, 0, 0, false, CTC_NoKeywords) &&
|
|
||||||
(Decl = R.getAsSingle<ObjCInterfaceDecl>())) {
|
|
||||||
Diag(IdLoc, diag::err_undef_interface_suggest)
|
|
||||||
<< Id << Decl->getDeclName()
|
|
||||||
<< FixItHint::CreateReplacement(IdLoc, Decl->getNameAsString());
|
|
||||||
Diag(Decl->getLocation(), diag::note_previous_decl)
|
|
||||||
<< Decl->getDeclName();
|
|
||||||
|
|
||||||
Id = Decl->getIdentifier();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ObjCInterfaceDecl *IDecl = dyn_cast_or_null<ObjCInterfaceDecl>(Decl);
|
|
||||||
if (IDecl) {
|
|
||||||
if (ObjCInterfaceDecl *Def = IDecl->getDefinition())
|
|
||||||
IDecl = Def;
|
|
||||||
}
|
|
||||||
return IDecl;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// ActOnStartOfObjCMethodDef - This routine sets up parameters; invisible
|
/// ActOnStartOfObjCMethodDef - This routine sets up parameters; invisible
|
||||||
/// and user declared, in the method definition's AST.
|
/// and user declared, in the method definition's AST.
|
||||||
void Sema::ActOnStartOfObjCMethodDef(Scope *FnBodyScope, DeclPtrTy D) {
|
void Sema::ActOnStartOfObjCMethodDef(Scope *FnBodyScope, DeclPtrTy D) {
|
||||||
|
@ -109,8 +65,6 @@ ActOnStartClassInterface(SourceLocation AtInterfaceLoc,
|
||||||
SourceLocation EndProtoLoc, AttributeList *AttrList) {
|
SourceLocation EndProtoLoc, AttributeList *AttrList) {
|
||||||
assert(ClassName && "Missing class identifier");
|
assert(ClassName && "Missing class identifier");
|
||||||
|
|
||||||
bool Invalid = false;
|
|
||||||
|
|
||||||
// Check for another declaration kind with the same name.
|
// Check for another declaration kind with the same name.
|
||||||
NamedDecl *PrevDecl = LookupSingleName(TUScope, ClassName, ClassLoc,
|
NamedDecl *PrevDecl = LookupSingleName(TUScope, ClassName, ClassLoc,
|
||||||
LookupOrdinaryName, ForRedeclaration);
|
LookupOrdinaryName, ForRedeclaration);
|
||||||
|
@ -118,34 +72,41 @@ ActOnStartClassInterface(SourceLocation AtInterfaceLoc,
|
||||||
if (PrevDecl && !isa<ObjCInterfaceDecl>(PrevDecl)) {
|
if (PrevDecl && !isa<ObjCInterfaceDecl>(PrevDecl)) {
|
||||||
Diag(ClassLoc, diag::err_redefinition_different_kind) << ClassName;
|
Diag(ClassLoc, diag::err_redefinition_different_kind) << ClassName;
|
||||||
Diag(PrevDecl->getLocation(), diag::note_previous_definition);
|
Diag(PrevDecl->getLocation(), diag::note_previous_definition);
|
||||||
// Set the new decl invalid and ignore the old.
|
|
||||||
Invalid = true;
|
|
||||||
PrevDecl = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ObjCInterfaceDecl *ODecl = cast_or_null<ObjCInterfaceDecl>(PrevDecl);
|
ObjCInterfaceDecl* IDecl = dyn_cast_or_null<ObjCInterfaceDecl>(PrevDecl);
|
||||||
if (ODecl) {
|
if (IDecl) {
|
||||||
// Class already seen. Is it a forward declaration?
|
// Class already seen. Is it a forward declaration?
|
||||||
if (ObjCInterfaceDecl *Def = ODecl->getDefinition()) {
|
if (!IDecl->isForwardDecl()) {
|
||||||
Invalid = true;
|
|
||||||
Diag(AtInterfaceLoc, diag::err_duplicate_class_def) << Def->getDeclName();
|
|
||||||
Diag(Def->getLocation(), diag::note_previous_definition);
|
|
||||||
|
|
||||||
// Return the previous class interface and ignore the new one.
|
|
||||||
return DeclPtrTy::make(ODecl);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ObjCInterfaceDecl *IDecl =
|
|
||||||
ObjCInterfaceDecl::Create(Context, CurContext, AtInterfaceLoc,
|
|
||||||
ClassName, ClassLoc, ODecl);
|
|
||||||
if (Invalid)
|
|
||||||
IDecl->setInvalidDecl();
|
IDecl->setInvalidDecl();
|
||||||
|
Diag(AtInterfaceLoc, diag::err_duplicate_class_def)<<IDecl->getDeclName();
|
||||||
|
Diag(IDecl->getLocation(), diag::note_previous_definition);
|
||||||
|
|
||||||
|
// Return the previous class interface.
|
||||||
|
// FIXME: don't leak the objects passed in!
|
||||||
|
return DeclPtrTy::make(IDecl);
|
||||||
|
} else {
|
||||||
|
IDecl->setLocation(AtInterfaceLoc);
|
||||||
|
IDecl->setForwardDecl(false);
|
||||||
|
IDecl->setClassLoc(ClassLoc);
|
||||||
|
|
||||||
|
// Since this ObjCInterfaceDecl was created by a forward declaration,
|
||||||
|
// we now add it to the DeclContext since it wasn't added before
|
||||||
|
// (see ActOnForwardClassDeclaration).
|
||||||
|
IDecl->setLexicalDeclContext(CurContext);
|
||||||
|
CurContext->addDecl(IDecl);
|
||||||
|
|
||||||
|
if (AttrList)
|
||||||
|
ProcessDeclAttributeList(TUScope, IDecl, AttrList);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
IDecl = ObjCInterfaceDecl::Create(Context, CurContext, AtInterfaceLoc,
|
||||||
|
ClassName, ClassLoc);
|
||||||
if (AttrList)
|
if (AttrList)
|
||||||
ProcessDeclAttributeList(TUScope, IDecl, AttrList);
|
ProcessDeclAttributeList(TUScope, IDecl, AttrList);
|
||||||
|
|
||||||
PushOnScopeChains(IDecl, TUScope);
|
PushOnScopeChains(IDecl, TUScope);
|
||||||
|
}
|
||||||
|
|
||||||
if (SuperName) {
|
if (SuperName) {
|
||||||
// Check if a different kind of symbol declared in this scope.
|
// Check if a different kind of symbol declared in this scope.
|
||||||
|
@ -164,8 +125,6 @@ ActOnStartClassInterface(SourceLocation AtInterfaceLoc,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Since we just pushed IDecl on the scope chain, if PrevDecl is the same
|
|
||||||
// class, it will be the same declaration.
|
|
||||||
if (PrevDecl == IDecl) {
|
if (PrevDecl == IDecl) {
|
||||||
Diag(SuperLoc, diag::err_recursive_superclass)
|
Diag(SuperLoc, diag::err_recursive_superclass)
|
||||||
<< SuperName << ClassName << SourceRange(AtInterfaceLoc, ClassLoc);
|
<< SuperName << ClassName << SourceRange(AtInterfaceLoc, ClassLoc);
|
||||||
|
@ -184,8 +143,8 @@ ActOnStartClassInterface(SourceLocation AtInterfaceLoc,
|
||||||
if (const TypedefDecl *TDecl = dyn_cast_or_null<TypedefDecl>(PrevDecl)) {
|
if (const TypedefDecl *TDecl = dyn_cast_or_null<TypedefDecl>(PrevDecl)) {
|
||||||
QualType T = TDecl->getUnderlyingType();
|
QualType T = TDecl->getUnderlyingType();
|
||||||
if (T->isObjCObjectType()) {
|
if (T->isObjCObjectType()) {
|
||||||
if (NamedDecl *NDecl = T->getAs<ObjCObjectType>()->getInterface())
|
if (NamedDecl *IDecl = T->getAs<ObjCObjectType>()->getInterface())
|
||||||
SuperClassDecl = dyn_cast<ObjCInterfaceDecl>(NDecl);
|
SuperClassDecl = dyn_cast<ObjCInterfaceDecl>(IDecl);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -200,11 +159,6 @@ ActOnStartClassInterface(SourceLocation AtInterfaceLoc,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (SuperClassDecl) {
|
|
||||||
if (ObjCInterfaceDecl *Def = SuperClassDecl->getDefinition())
|
|
||||||
SuperClassDecl = Def;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!dyn_cast_or_null<TypedefDecl>(PrevDecl)) {
|
if (!dyn_cast_or_null<TypedefDecl>(PrevDecl)) {
|
||||||
if (!SuperClassDecl)
|
if (!SuperClassDecl)
|
||||||
Diag(SuperLoc, diag::err_undef_superclass)
|
Diag(SuperLoc, diag::err_undef_superclass)
|
||||||
|
@ -576,7 +530,7 @@ Sema::DeclPtrTy Sema::ActOnStartClassImplementation(
|
||||||
IdentifierInfo *ClassName, SourceLocation ClassLoc,
|
IdentifierInfo *ClassName, SourceLocation ClassLoc,
|
||||||
IdentifierInfo *SuperClassname,
|
IdentifierInfo *SuperClassname,
|
||||||
SourceLocation SuperClassLoc) {
|
SourceLocation SuperClassLoc) {
|
||||||
ObjCInterfaceDecl *IDecl = 0, *ODecl = 0;
|
ObjCInterfaceDecl* IDecl = 0;
|
||||||
// Check for another declaration kind with the same name.
|
// Check for another declaration kind with the same name.
|
||||||
NamedDecl *PrevDecl
|
NamedDecl *PrevDecl
|
||||||
= LookupSingleName(TUScope, ClassName, ClassLoc, LookupOrdinaryName,
|
= LookupSingleName(TUScope, ClassName, ClassLoc, LookupOrdinaryName,
|
||||||
|
@ -584,10 +538,11 @@ Sema::DeclPtrTy Sema::ActOnStartClassImplementation(
|
||||||
if (PrevDecl && !isa<ObjCInterfaceDecl>(PrevDecl)) {
|
if (PrevDecl && !isa<ObjCInterfaceDecl>(PrevDecl)) {
|
||||||
Diag(ClassLoc, diag::err_redefinition_different_kind) << ClassName;
|
Diag(ClassLoc, diag::err_redefinition_different_kind) << ClassName;
|
||||||
Diag(PrevDecl->getLocation(), diag::note_previous_definition);
|
Diag(PrevDecl->getLocation(), diag::note_previous_definition);
|
||||||
} else if ((ODecl = cast_or_null<ObjCInterfaceDecl>(PrevDecl))) {
|
} else if ((IDecl = dyn_cast_or_null<ObjCInterfaceDecl>(PrevDecl))) {
|
||||||
// If we can't find a definition of the interface, warn.
|
// If this is a forward declaration of an interface, warn.
|
||||||
if (!(IDecl = ODecl->getDefinition())) {
|
if (IDecl->isForwardDecl()) {
|
||||||
Diag(ClassLoc, diag::warn_undef_interface) << ClassName;
|
Diag(ClassLoc, diag::warn_undef_interface) << ClassName;
|
||||||
|
IDecl = 0;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// We did not find anything with the name ClassName; try to correct for
|
// We did not find anything with the name ClassName; try to correct for
|
||||||
|
@ -597,7 +552,7 @@ Sema::DeclPtrTy Sema::ActOnStartClassImplementation(
|
||||||
(IDecl = R.getAsSingle<ObjCInterfaceDecl>())) {
|
(IDecl = R.getAsSingle<ObjCInterfaceDecl>())) {
|
||||||
// Suggest the (potentially) correct interface name. However, put the
|
// Suggest the (potentially) correct interface name. However, put the
|
||||||
// fix-it hint itself in a separate note, since changing the name in
|
// fix-it hint itself in a separate note, since changing the name in
|
||||||
// the warning would make the fix-it change semantics. Also, don't
|
// the warning would make the fix-it change semantics.However, don't
|
||||||
// provide a code-modification hint or use the typo name for recovery,
|
// provide a code-modification hint or use the typo name for recovery,
|
||||||
// because this is just a warning. The program may actually be correct.
|
// because this is just a warning. The program may actually be correct.
|
||||||
Diag(ClassLoc, diag::warn_undef_interface_suggest)
|
Diag(ClassLoc, diag::warn_undef_interface_suggest)
|
||||||
|
@ -644,11 +599,16 @@ Sema::DeclPtrTy Sema::ActOnStartClassImplementation(
|
||||||
// FIXME: Do we support attributes on the @implementation? If so we should
|
// FIXME: Do we support attributes on the @implementation? If so we should
|
||||||
// copy them over.
|
// copy them over.
|
||||||
IDecl = ObjCInterfaceDecl::Create(Context, CurContext, AtClassImplLoc,
|
IDecl = ObjCInterfaceDecl::Create(Context, CurContext, AtClassImplLoc,
|
||||||
ClassName, ClassLoc, ODecl, false, true);
|
ClassName, ClassLoc, false, true);
|
||||||
IDecl->setSuperClass(SDecl);
|
IDecl->setSuperClass(SDecl);
|
||||||
IDecl->setLocEnd(ClassLoc);
|
IDecl->setLocEnd(ClassLoc);
|
||||||
|
|
||||||
PushOnScopeChains(IDecl, TUScope);
|
PushOnScopeChains(IDecl, TUScope);
|
||||||
|
} else {
|
||||||
|
// Mark the interface as being completed, even if it was just as
|
||||||
|
// @class ....;
|
||||||
|
// declaration; the user cannot reopen it.
|
||||||
|
IDecl->setForwardDecl(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
ObjCImplementationDecl* IMPDecl =
|
ObjCImplementationDecl* IMPDecl =
|
||||||
|
@ -659,15 +619,15 @@ Sema::DeclPtrTy Sema::ActOnStartClassImplementation(
|
||||||
return DeclPtrTy::make(IMPDecl);
|
return DeclPtrTy::make(IMPDecl);
|
||||||
|
|
||||||
// Check that there is no duplicate implementation of this class.
|
// Check that there is no duplicate implementation of this class.
|
||||||
if (IDecl && IDecl->getImplementation()) {
|
if (IDecl->getImplementation()) {
|
||||||
|
// FIXME: Don't leak everything!
|
||||||
Diag(ClassLoc, diag::err_dup_implementation_class) << ClassName;
|
Diag(ClassLoc, diag::err_dup_implementation_class) << ClassName;
|
||||||
Diag(IDecl->getImplementation()->getLocation(),
|
Diag(IDecl->getImplementation()->getLocation(),
|
||||||
diag::note_previous_definition);
|
diag::note_previous_definition);
|
||||||
} else {
|
} else { // add it to the list.
|
||||||
IDecl->setImplementation(IMPDecl);
|
IDecl->setImplementation(IMPDecl);
|
||||||
PushOnScopeChains(IMPDecl, TUScope);
|
PushOnScopeChains(IMPDecl, TUScope);
|
||||||
}
|
}
|
||||||
|
|
||||||
return DeclPtrTy::make(IMPDecl);
|
return DeclPtrTy::make(IMPDecl);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1069,10 +1029,10 @@ Sema::ActOnForwardClassDeclaration(SourceLocation AtClassLoc,
|
||||||
PrevDecl = OI->getInterface();
|
PrevDecl = OI->getInterface();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ObjCInterfaceDecl *ODecl = dyn_cast_or_null<ObjCInterfaceDecl>(PrevDecl);
|
ObjCInterfaceDecl *IDecl = dyn_cast_or_null<ObjCInterfaceDecl>(PrevDecl);
|
||||||
ObjCInterfaceDecl *IDecl =
|
if (!IDecl) { // Not already seen? Make a forward decl.
|
||||||
ObjCInterfaceDecl::Create(Context, CurContext, AtClassLoc,
|
IDecl = ObjCInterfaceDecl::Create(Context, CurContext, AtClassLoc,
|
||||||
IdentList[i], IdentLocs[i], ODecl, true);
|
IdentList[i], IdentLocs[i], true);
|
||||||
|
|
||||||
// Push the ObjCInterfaceDecl on the scope chain but do *not* add it to
|
// Push the ObjCInterfaceDecl on the scope chain but do *not* add it to
|
||||||
// the current DeclContext. This prevents clients that walk DeclContext
|
// the current DeclContext. This prevents clients that walk DeclContext
|
||||||
|
@ -1081,6 +1041,7 @@ Sema::ActOnForwardClassDeclaration(SourceLocation AtClassLoc,
|
||||||
// sure this declaration is visible for name lookup.
|
// sure this declaration is visible for name lookup.
|
||||||
PushOnScopeChains(IDecl, TUScope, false);
|
PushOnScopeChains(IDecl, TUScope, false);
|
||||||
CurContext->makeDeclVisibleInContext(IDecl, true);
|
CurContext->makeDeclVisibleInContext(IDecl, true);
|
||||||
|
}
|
||||||
|
|
||||||
Interfaces.push_back(IDecl);
|
Interfaces.push_back(IDecl);
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,7 +17,7 @@ void function(Foo * arg)
|
||||||
|
|
||||||
// CHECK-scan: [1:1 - 8:1] Invalid Cursor => NoDeclFound
|
// CHECK-scan: [1:1 - 8:1] Invalid Cursor => NoDeclFound
|
||||||
// CHECK-scan: [8:1 - 8:8] UnexposedDecl=:8:1
|
// CHECK-scan: [8:1 - 8:8] UnexposedDecl=:8:1
|
||||||
// CHECK-scan: [8:8 - 8:11] ObjCClassRef=Foo:8:8
|
// CHECK-scan: [8:8 - 8:11] ObjCClassRef=Foo:10:12
|
||||||
// CHECK-scan: [8:11 - 10:1] Invalid Cursor => NoDeclFound
|
// CHECK-scan: [8:11 - 10:1] Invalid Cursor => NoDeclFound
|
||||||
// CHECK-scan: [10:1 - 11:5] ObjCInterfaceDecl=Foo:10:12
|
// CHECK-scan: [10:1 - 11:5] ObjCInterfaceDecl=Foo:10:12
|
||||||
// CHECK-scan: [11:5 - 13:6] Invalid Cursor => NoDeclFound
|
// CHECK-scan: [11:5 - 13:6] Invalid Cursor => NoDeclFound
|
||||||
|
|
Loading…
Reference in New Issue