forked from OSchip/llvm-project
Make ivar rewrite C++ friendly (since C++ forces a unified namespace for typedefs/structures). The previous version of the rewriter generated both a type def and structure def for each class (which doesn't fly in C++).
llvm-svn: 48266
This commit is contained in:
parent
35f8f07c00
commit
dc5b6b2e5d
|
@ -654,11 +654,12 @@ void RewriteTest::RewriteObjCMethodDecl(ObjCMethodDecl *OMD,
|
||||||
if (OMD->isInstance()) {
|
if (OMD->isInstance()) {
|
||||||
QualType selfTy = Context->getObjCInterfaceType(OMD->getClassInterface());
|
QualType selfTy = Context->getObjCInterfaceType(OMD->getClassInterface());
|
||||||
selfTy = Context->getPointerType(selfTy);
|
selfTy = Context->getPointerType(selfTy);
|
||||||
if (ObjCSynthesizedStructs.count(OMD->getClassInterface()))
|
if (!LangOpts.Microsoft) {
|
||||||
ResultStr += "struct ";
|
if (ObjCSynthesizedStructs.count(OMD->getClassInterface()))
|
||||||
|
ResultStr += "struct ";
|
||||||
|
}
|
||||||
|
// When rewriting for Microsoft, explicitly omit the structure name.
|
||||||
ResultStr += OMD->getClassInterface()->getName();
|
ResultStr += OMD->getClassInterface()->getName();
|
||||||
if (LangOpts.Microsoft)
|
|
||||||
ResultStr += "_IMPL";
|
|
||||||
ResultStr += " *";
|
ResultStr += " *";
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -759,40 +760,43 @@ void RewriteTest::RewriteInterfaceDecl(ObjCInterfaceDecl *ClassDecl) {
|
||||||
ReplaceText(ClassDecl->getAtEndLoc(), 0, "// ", 3);
|
ReplaceText(ClassDecl->getAtEndLoc(), 0, "// ", 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// FIXME: Investigate the following comment...
|
||||||
|
/// This code is not right. It seems unnecessary. It breaks use of
|
||||||
|
/// ivar reference used as 'receiver' of an expression; as in:
|
||||||
|
/// [newInv->_container addObject:0];
|
||||||
Stmt *RewriteTest::RewriteObjCIvarRefExpr(ObjCIvarRefExpr *IV) {
|
Stmt *RewriteTest::RewriteObjCIvarRefExpr(ObjCIvarRefExpr *IV) {
|
||||||
ObjCIvarDecl *D = IV->getDecl();
|
ObjCIvarDecl *D = IV->getDecl();
|
||||||
if (IV->isFreeIvar()) {
|
if (CurMethodDecl) {
|
||||||
Expr *Replacement = new MemberExpr(IV->getBase(), true, D,
|
if (const PointerType *pType = IV->getBase()->getType()->getAsPointerType()) {
|
||||||
IV->getLocation(), D->getType());
|
ObjCInterfaceType *intT = dyn_cast<ObjCInterfaceType>(pType->getPointeeType());
|
||||||
ReplaceStmt(IV, Replacement);
|
if (CurMethodDecl->getClassInterface() == intT->getDecl()) {
|
||||||
delete IV;
|
std::string RecName = intT->getDecl()->getIdentifier()->getName();
|
||||||
return Replacement;
|
RecName += "_IMPL";
|
||||||
} else {
|
IdentifierInfo *II = &Context->Idents.get(RecName.c_str());
|
||||||
#if 0
|
RecordDecl *RD = new RecordDecl(Decl::Struct, SourceLocation(), II, 0);
|
||||||
/// This code is not right. It seems unnecessary. It breaks use of
|
assert(RD && "RewriteObjCIvarRefExpr(): Can't find RecordDecl");
|
||||||
/// ivar reference used as 'receiver' of an expression; as in:
|
QualType castT = Context->getPointerType(Context->getTagDeclType(RD));
|
||||||
/// [newInv->_container addObject:0];
|
CastExpr *castExpr = new CastExpr(castT, IV->getBase(), SourceLocation());
|
||||||
if (CurMethodDecl) {
|
// Don't forget the parens to enforce the proper binding.
|
||||||
if (const PointerType *pType = IV->getBase()->getType()->getAsPointerType()) {
|
ParenExpr *PE = new ParenExpr(SourceLocation(), SourceLocation(), castExpr);
|
||||||
ObjCInterfaceType *intT = dyn_cast<ObjCInterfaceType>(pType->getPointeeType());
|
if (IV->isFreeIvar()) {
|
||||||
if (CurMethodDecl->getClassInterface() == intT->getDecl()) {
|
MemberExpr *ME = new MemberExpr(PE, true, D, IV->getLocation(), D->getType());
|
||||||
IdentifierInfo *II = intT->getDecl()->getIdentifier();
|
ReplaceStmt(IV, ME);
|
||||||
RecordDecl *RD = new RecordDecl(Decl::Struct, SourceLocation(),
|
delete IV;
|
||||||
II, 0);
|
return ME;
|
||||||
QualType castT = Context->getPointerType(Context->getTagDeclType(RD));
|
} else {
|
||||||
|
|
||||||
CastExpr *castExpr = new CastExpr(castT, IV->getBase(), SourceLocation());
|
|
||||||
// Don't forget the parens to enforce the proper binding.
|
|
||||||
ParenExpr *PE = new ParenExpr(SourceLocation(), SourceLocation(), castExpr);
|
|
||||||
ReplaceStmt(IV->getBase(), PE);
|
ReplaceStmt(IV->getBase(), PE);
|
||||||
delete IV->getBase();
|
delete IV->getBase();
|
||||||
return PE;
|
return PE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
return IV;
|
|
||||||
}
|
}
|
||||||
|
Expr *Replacement = new MemberExpr(IV->getBase(), true, D,
|
||||||
|
IV->getLocation(), D->getType());
|
||||||
|
ReplaceStmt(IV, Replacement);
|
||||||
|
delete IV;
|
||||||
|
return Replacement;
|
||||||
}
|
}
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
@ -2169,7 +2173,7 @@ void RewriteTest::SynthesizeObjCInternalStruct(ObjCInterfaceDecl *CDecl,
|
||||||
Result += CDecl->getName();
|
Result += CDecl->getName();
|
||||||
if (LangOpts.Microsoft)
|
if (LangOpts.Microsoft)
|
||||||
Result += "_IMPL";
|
Result += "_IMPL";
|
||||||
|
|
||||||
if (NumIvars > 0) {
|
if (NumIvars > 0) {
|
||||||
const char *cursor = strchr(startBuf, '{');
|
const char *cursor = strchr(startBuf, '{');
|
||||||
assert((cursor && endBuf)
|
assert((cursor && endBuf)
|
||||||
|
@ -2182,6 +2186,7 @@ void RewriteTest::SynthesizeObjCInternalStruct(ObjCInterfaceDecl *CDecl,
|
||||||
Result += RCDecl->getName();
|
Result += RCDecl->getName();
|
||||||
if (LangOpts.Microsoft)
|
if (LangOpts.Microsoft)
|
||||||
Result += "_IMPL";
|
Result += "_IMPL";
|
||||||
|
|
||||||
// Note: We don't name the field decl. This simplifies the "codegen" for
|
// Note: We don't name the field decl. This simplifies the "codegen" for
|
||||||
// accessing a superclasses instance variables (and is similar to what gcc
|
// accessing a superclasses instance variables (and is similar to what gcc
|
||||||
// does internally). The unnamed struct field feature is enabled with
|
// does internally). The unnamed struct field feature is enabled with
|
||||||
|
@ -2232,6 +2237,7 @@ void RewriteTest::SynthesizeObjCInternalStruct(ObjCInterfaceDecl *CDecl,
|
||||||
Result += RCDecl->getName();
|
Result += RCDecl->getName();
|
||||||
if (LangOpts.Microsoft)
|
if (LangOpts.Microsoft)
|
||||||
Result += "_IMPL";
|
Result += "_IMPL";
|
||||||
|
|
||||||
// Note: We don't name the field decl. This simplifies the "codegen" for
|
// Note: We don't name the field decl. This simplifies the "codegen" for
|
||||||
// accessing a superclasses instance variables (and is similar to what gcc
|
// accessing a superclasses instance variables (and is similar to what gcc
|
||||||
// does internally). The unnamed struct field feature is enabled with
|
// does internally). The unnamed struct field feature is enabled with
|
||||||
|
|
Loading…
Reference in New Issue