You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			356 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			356 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===--- TransGCAttrs.cpp - Transformations to ARC mode --------------------===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include "Transforms.h"
 | ||
|  | #include "Internals.h"
 | ||
|  | #include "clang/AST/ASTContext.h"
 | ||
|  | #include "clang/Basic/SourceManager.h"
 | ||
|  | #include "clang/Lex/Lexer.h"
 | ||
|  | #include "clang/Sema/SemaDiagnostic.h"
 | ||
|  | #include "llvm/ADT/SmallString.h"
 | ||
|  | #include "llvm/ADT/TinyPtrVector.h"
 | ||
|  | #include "llvm/Support/SaveAndRestore.h"
 | ||
|  | 
 | ||
|  | using namespace clang; | ||
|  | using namespace arcmt; | ||
|  | using namespace trans; | ||
|  | 
 | ||
|  | namespace { | ||
|  | 
 | ||
|  | /// \brief Collects all the places where GC attributes __strong/__weak occur.
 | ||
|  | class GCAttrsCollector : public RecursiveASTVisitor<GCAttrsCollector> { | ||
|  |   MigrationContext &MigrateCtx; | ||
|  |   bool FullyMigratable; | ||
|  |   std::vector<ObjCPropertyDecl *> &AllProps; | ||
|  | 
 | ||
|  |   typedef RecursiveASTVisitor<GCAttrsCollector> base; | ||
|  | public: | ||
|  |   GCAttrsCollector(MigrationContext &ctx, | ||
|  |                    std::vector<ObjCPropertyDecl *> &AllProps) | ||
|  |     : MigrateCtx(ctx), FullyMigratable(false), | ||
|  |       AllProps(AllProps) { } | ||
|  | 
 | ||
|  |   bool shouldWalkTypesOfTypeLocs() const { return false; } | ||
|  | 
 | ||
|  |   bool VisitAttributedTypeLoc(AttributedTypeLoc TL) { | ||
|  |     handleAttr(TL); | ||
|  |     return true; | ||
|  |   } | ||
|  | 
 | ||
|  |   bool TraverseDecl(Decl *D) { | ||
|  |     if (!D || D->isImplicit()) | ||
|  |       return true; | ||
|  | 
 | ||
|  |     SaveAndRestore<bool> Save(FullyMigratable, isMigratable(D)); | ||
|  |      | ||
|  |     if (ObjCPropertyDecl *PropD = dyn_cast<ObjCPropertyDecl>(D)) { | ||
|  |       lookForAttribute(PropD, PropD->getTypeSourceInfo()); | ||
|  |       AllProps.push_back(PropD); | ||
|  |     } else if (DeclaratorDecl *DD = dyn_cast<DeclaratorDecl>(D)) { | ||
|  |       lookForAttribute(DD, DD->getTypeSourceInfo()); | ||
|  |     } | ||
|  |     return base::TraverseDecl(D); | ||
|  |   } | ||
|  | 
 | ||
|  |   void lookForAttribute(Decl *D, TypeSourceInfo *TInfo) { | ||
|  |     if (!TInfo) | ||
|  |       return; | ||
|  |     TypeLoc TL = TInfo->getTypeLoc(); | ||
|  |     while (TL) { | ||
|  |       if (QualifiedTypeLoc QL = TL.getAs<QualifiedTypeLoc>()) { | ||
|  |         TL = QL.getUnqualifiedLoc(); | ||
|  |       } else if (AttributedTypeLoc Attr = TL.getAs<AttributedTypeLoc>()) { | ||
|  |         if (handleAttr(Attr, D)) | ||
|  |           break; | ||
|  |         TL = Attr.getModifiedLoc(); | ||
|  |       } else if (ArrayTypeLoc Arr = TL.getAs<ArrayTypeLoc>()) { | ||
|  |         TL = Arr.getElementLoc(); | ||
|  |       } else if (PointerTypeLoc PT = TL.getAs<PointerTypeLoc>()) { | ||
|  |         TL = PT.getPointeeLoc(); | ||
|  |       } else if (ReferenceTypeLoc RT = TL.getAs<ReferenceTypeLoc>()) | ||
|  |         TL = RT.getPointeeLoc(); | ||
|  |       else | ||
|  |         break; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   bool handleAttr(AttributedTypeLoc TL, Decl *D = nullptr) { | ||
|  |     if (TL.getAttrKind() != AttributedType::attr_objc_ownership) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     SourceLocation Loc = TL.getAttrNameLoc(); | ||
|  |     unsigned RawLoc = Loc.getRawEncoding(); | ||
|  |     if (MigrateCtx.AttrSet.count(RawLoc)) | ||
|  |       return true; | ||
|  | 
 | ||
|  |     ASTContext &Ctx = MigrateCtx.Pass.Ctx; | ||
|  |     SourceManager &SM = Ctx.getSourceManager(); | ||
|  |     if (Loc.isMacroID()) | ||
|  |       Loc = SM.getImmediateExpansionRange(Loc).first; | ||
|  |     SmallString<32> Buf; | ||
|  |     bool Invalid = false; | ||
|  |     StringRef Spell = Lexer::getSpelling( | ||
|  |                                   SM.getSpellingLoc(TL.getAttrEnumOperandLoc()), | ||
|  |                                   Buf, SM, Ctx.getLangOpts(), &Invalid); | ||
|  |     if (Invalid) | ||
|  |       return false; | ||
|  |     MigrationContext::GCAttrOccurrence::AttrKind Kind; | ||
|  |     if (Spell == "strong") | ||
|  |       Kind = MigrationContext::GCAttrOccurrence::Strong; | ||
|  |     else if (Spell == "weak") | ||
|  |       Kind = MigrationContext::GCAttrOccurrence::Weak; | ||
|  |     else | ||
|  |       return false; | ||
|  |   | ||
|  |     MigrateCtx.AttrSet.insert(RawLoc); | ||
|  |     MigrateCtx.GCAttrs.push_back(MigrationContext::GCAttrOccurrence()); | ||
|  |     MigrationContext::GCAttrOccurrence &Attr = MigrateCtx.GCAttrs.back(); | ||
|  | 
 | ||
|  |     Attr.Kind = Kind; | ||
|  |     Attr.Loc = Loc; | ||
|  |     Attr.ModifiedType = TL.getModifiedLoc().getType(); | ||
|  |     Attr.Dcl = D; | ||
|  |     Attr.FullyMigratable = FullyMigratable; | ||
|  |     return true; | ||
|  |   } | ||
|  | 
 | ||
|  |   bool isMigratable(Decl *D) { | ||
|  |     if (isa<TranslationUnitDecl>(D)) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     if (isInMainFile(D)) | ||
|  |       return true; | ||
|  | 
 | ||
|  |     if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) | ||
|  |       return FD->hasBody(); | ||
|  | 
 | ||
|  |     if (ObjCContainerDecl *ContD = dyn_cast<ObjCContainerDecl>(D)) | ||
|  |       return hasObjCImpl(ContD); | ||
|  | 
 | ||
|  |     if (CXXRecordDecl *RD = dyn_cast<CXXRecordDecl>(D)) { | ||
|  |       for (const auto *MI : RD->methods()) { | ||
|  |         if (MI->isOutOfLine()) | ||
|  |           return true; | ||
|  |       } | ||
|  |       return false; | ||
|  |     } | ||
|  | 
 | ||
|  |     return isMigratable(cast<Decl>(D->getDeclContext())); | ||
|  |   } | ||
|  | 
 | ||
|  |   static bool hasObjCImpl(Decl *D) { | ||
|  |     if (!D) | ||
|  |       return false; | ||
|  |     if (ObjCContainerDecl *ContD = dyn_cast<ObjCContainerDecl>(D)) { | ||
|  |       if (ObjCInterfaceDecl *ID = dyn_cast<ObjCInterfaceDecl>(ContD)) | ||
|  |         return ID->getImplementation() != nullptr; | ||
|  |       if (ObjCCategoryDecl *CD = dyn_cast<ObjCCategoryDecl>(ContD)) | ||
|  |         return CD->getImplementation() != nullptr; | ||
|  |       return isa<ObjCImplDecl>(ContD); | ||
|  |     } | ||
|  |     return false; | ||
|  |   } | ||
|  | 
 | ||
|  |   bool isInMainFile(Decl *D) { | ||
|  |     if (!D) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     for (auto I : D->redecls()) | ||
|  |       if (!isInMainFile(I->getLocation())) | ||
|  |         return false; | ||
|  |      | ||
|  |     return true; | ||
|  |   } | ||
|  | 
 | ||
|  |   bool isInMainFile(SourceLocation Loc) { | ||
|  |     if (Loc.isInvalid()) | ||
|  |       return false; | ||
|  | 
 | ||
|  |     SourceManager &SM = MigrateCtx.Pass.Ctx.getSourceManager(); | ||
|  |     return SM.isInFileID(SM.getExpansionLoc(Loc), SM.getMainFileID()); | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | } // anonymous namespace
 | ||
|  | 
 | ||
|  | static void errorForGCAttrsOnNonObjC(MigrationContext &MigrateCtx) { | ||
|  |   TransformActions &TA = MigrateCtx.Pass.TA; | ||
|  | 
 | ||
|  |   for (unsigned i = 0, e = MigrateCtx.GCAttrs.size(); i != e; ++i) { | ||
|  |     MigrationContext::GCAttrOccurrence &Attr = MigrateCtx.GCAttrs[i]; | ||
|  |     if (Attr.FullyMigratable && Attr.Dcl) { | ||
|  |       if (Attr.ModifiedType.isNull()) | ||
|  |         continue; | ||
|  |       if (!Attr.ModifiedType->isObjCRetainableType()) { | ||
|  |         TA.reportError("GC managed memory will become unmanaged in ARC", | ||
|  |                        Attr.Loc); | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | static void checkWeakGCAttrs(MigrationContext &MigrateCtx) { | ||
|  |   TransformActions &TA = MigrateCtx.Pass.TA; | ||
|  | 
 | ||
|  |   for (unsigned i = 0, e = MigrateCtx.GCAttrs.size(); i != e; ++i) { | ||
|  |     MigrationContext::GCAttrOccurrence &Attr = MigrateCtx.GCAttrs[i]; | ||
|  |     if (Attr.Kind == MigrationContext::GCAttrOccurrence::Weak) { | ||
|  |       if (Attr.ModifiedType.isNull() || | ||
|  |           !Attr.ModifiedType->isObjCRetainableType()) | ||
|  |         continue; | ||
|  |       if (!canApplyWeak(MigrateCtx.Pass.Ctx, Attr.ModifiedType, | ||
|  |                         /*AllowOnUnknownClass=*/true)) { | ||
|  |         Transaction Trans(TA); | ||
|  |         if (!MigrateCtx.RemovedAttrSet.count(Attr.Loc.getRawEncoding())) | ||
|  |           TA.replaceText(Attr.Loc, "__weak", "__unsafe_unretained"); | ||
|  |         TA.clearDiagnostic(diag::err_arc_weak_no_runtime, | ||
|  |                            diag::err_arc_unsupported_weak_class, | ||
|  |                            Attr.Loc); | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | typedef llvm::TinyPtrVector<ObjCPropertyDecl *> IndivPropsTy; | ||
|  | 
 | ||
|  | static void checkAllAtProps(MigrationContext &MigrateCtx, | ||
|  |                             SourceLocation AtLoc, | ||
|  |                             IndivPropsTy &IndProps) { | ||
|  |   if (IndProps.empty()) | ||
|  |     return; | ||
|  | 
 | ||
|  |   for (IndivPropsTy::iterator | ||
|  |          PI = IndProps.begin(), PE = IndProps.end(); PI != PE; ++PI) { | ||
|  |     QualType T = (*PI)->getType(); | ||
|  |     if (T.isNull() || !T->isObjCRetainableType()) | ||
|  |       return; | ||
|  |   } | ||
|  | 
 | ||
|  |   SmallVector<std::pair<AttributedTypeLoc, ObjCPropertyDecl *>, 4> ATLs; | ||
|  |   bool hasWeak = false, hasStrong = false; | ||
|  |   ObjCPropertyDecl::PropertyAttributeKind | ||
|  |     Attrs = ObjCPropertyDecl::OBJC_PR_noattr; | ||
|  |   for (IndivPropsTy::iterator | ||
|  |          PI = IndProps.begin(), PE = IndProps.end(); PI != PE; ++PI) { | ||
|  |     ObjCPropertyDecl *PD = *PI; | ||
|  |     Attrs = PD->getPropertyAttributesAsWritten(); | ||
|  |     TypeSourceInfo *TInfo = PD->getTypeSourceInfo(); | ||
|  |     if (!TInfo) | ||
|  |       return; | ||
|  |     TypeLoc TL = TInfo->getTypeLoc(); | ||
|  |     if (AttributedTypeLoc ATL = | ||
|  |             TL.getAs<AttributedTypeLoc>()) { | ||
|  |       ATLs.push_back(std::make_pair(ATL, PD)); | ||
|  |       if (TInfo->getType().getObjCLifetime() == Qualifiers::OCL_Weak) { | ||
|  |         hasWeak = true; | ||
|  |       } else if (TInfo->getType().getObjCLifetime() == Qualifiers::OCL_Strong) | ||
|  |         hasStrong = true; | ||
|  |       else | ||
|  |         return; | ||
|  |     } | ||
|  |   } | ||
|  |   if (ATLs.empty()) | ||
|  |     return; | ||
|  |   if (hasWeak && hasStrong) | ||
|  |     return; | ||
|  | 
 | ||
|  |   TransformActions &TA = MigrateCtx.Pass.TA; | ||
|  |   Transaction Trans(TA); | ||
|  | 
 | ||
|  |   if (GCAttrsCollector::hasObjCImpl( | ||
|  |                               cast<Decl>(IndProps.front()->getDeclContext()))) { | ||
|  |     if (hasWeak) | ||
|  |       MigrateCtx.AtPropsWeak.insert(AtLoc.getRawEncoding()); | ||
|  | 
 | ||
|  |   } else { | ||
|  |     StringRef toAttr = "strong"; | ||
|  |     if (hasWeak) { | ||
|  |       if (canApplyWeak(MigrateCtx.Pass.Ctx, IndProps.front()->getType(), | ||
|  |                        /*AllowOnUnkwownClass=*/true)) | ||
|  |         toAttr = "weak"; | ||
|  |       else | ||
|  |         toAttr = "unsafe_unretained"; | ||
|  |     } | ||
|  |     if (Attrs & ObjCPropertyDecl::OBJC_PR_assign) | ||
|  |       MigrateCtx.rewritePropertyAttribute("assign", toAttr, AtLoc); | ||
|  |     else | ||
|  |       MigrateCtx.addPropertyAttribute(toAttr, AtLoc); | ||
|  |   } | ||
|  | 
 | ||
|  |   for (unsigned i = 0, e = ATLs.size(); i != e; ++i) { | ||
|  |     SourceLocation Loc = ATLs[i].first.getAttrNameLoc(); | ||
|  |     if (Loc.isMacroID()) | ||
|  |       Loc = MigrateCtx.Pass.Ctx.getSourceManager() | ||
|  |                                          .getImmediateExpansionRange(Loc).first; | ||
|  |     TA.remove(Loc); | ||
|  |     TA.clearDiagnostic(diag::err_objc_property_attr_mutually_exclusive, AtLoc); | ||
|  |     TA.clearDiagnostic(diag::err_arc_inconsistent_property_ownership, | ||
|  |                        ATLs[i].second->getLocation()); | ||
|  |     MigrateCtx.RemovedAttrSet.insert(Loc.getRawEncoding()); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | static void checkAllProps(MigrationContext &MigrateCtx, | ||
|  |                           std::vector<ObjCPropertyDecl *> &AllProps) { | ||
|  |   typedef llvm::TinyPtrVector<ObjCPropertyDecl *> IndivPropsTy; | ||
|  |   llvm::DenseMap<unsigned, IndivPropsTy> AtProps; | ||
|  | 
 | ||
|  |   for (unsigned i = 0, e = AllProps.size(); i != e; ++i) { | ||
|  |     ObjCPropertyDecl *PD = AllProps[i]; | ||
|  |     if (PD->getPropertyAttributesAsWritten() & | ||
|  |           (ObjCPropertyDecl::OBJC_PR_assign | | ||
|  |            ObjCPropertyDecl::OBJC_PR_readonly)) { | ||
|  |       SourceLocation AtLoc = PD->getAtLoc(); | ||
|  |       if (AtLoc.isInvalid()) | ||
|  |         continue; | ||
|  |       unsigned RawAt = AtLoc.getRawEncoding(); | ||
|  |       AtProps[RawAt].push_back(PD); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   for (llvm::DenseMap<unsigned, IndivPropsTy>::iterator | ||
|  |          I = AtProps.begin(), E = AtProps.end(); I != E; ++I) { | ||
|  |     SourceLocation AtLoc = SourceLocation::getFromRawEncoding(I->first); | ||
|  |     IndivPropsTy &IndProps = I->second; | ||
|  |     checkAllAtProps(MigrateCtx, AtLoc, IndProps); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | void GCAttrsTraverser::traverseTU(MigrationContext &MigrateCtx) { | ||
|  |   std::vector<ObjCPropertyDecl *> AllProps; | ||
|  |   GCAttrsCollector(MigrateCtx, AllProps).TraverseDecl( | ||
|  |                                   MigrateCtx.Pass.Ctx.getTranslationUnitDecl()); | ||
|  | 
 | ||
|  |   errorForGCAttrsOnNonObjC(MigrateCtx); | ||
|  |   checkAllProps(MigrateCtx, AllProps); | ||
|  |   checkWeakGCAttrs(MigrateCtx); | ||
|  | } | ||
|  | 
 | ||
|  | void MigrationContext::dumpGCAttrs() { | ||
|  |   llvm::errs() << "\n################\n"; | ||
|  |   for (unsigned i = 0, e = GCAttrs.size(); i != e; ++i) { | ||
|  |     GCAttrOccurrence &Attr = GCAttrs[i]; | ||
|  |     llvm::errs() << "KIND: " | ||
|  |         << (Attr.Kind == GCAttrOccurrence::Strong ? "strong" : "weak"); | ||
|  |     llvm::errs() << "\nLOC: "; | ||
|  |     Attr.Loc.dump(Pass.Ctx.getSourceManager()); | ||
|  |     llvm::errs() << "\nTYPE: "; | ||
|  |     Attr.ModifiedType.dump(); | ||
|  |     if (Attr.Dcl) { | ||
|  |       llvm::errs() << "DECL:\n"; | ||
|  |       Attr.Dcl->dump(); | ||
|  |     } else { | ||
|  |       llvm::errs() << "DECL: NONE"; | ||
|  |     } | ||
|  |     llvm::errs() << "\nMIGRATABLE: " << Attr.FullyMigratable; | ||
|  |     llvm::errs() << "\n----------------\n"; | ||
|  |   } | ||
|  |   llvm::errs() << "\n################\n"; | ||
|  | } |