Index: llvm/lib/Transforms/IPO/LowerTypeTests.cpp =================================================================== --- llvm/lib/Transforms/IPO/LowerTypeTests.cpp +++ llvm/lib/Transforms/IPO/LowerTypeTests.cpp @@ -297,11 +297,13 @@ struct ICallBranchFunnel final : TrailingObjects { static ICallBranchFunnel *create(BumpPtrAllocator &Alloc, CallInst *CI, - ArrayRef Targets) { + ArrayRef Targets, + unsigned UniqueId) { auto *Call = static_cast( Alloc.Allocate(totalSizeToAlloc(Targets.size()), alignof(ICallBranchFunnel))); Call->CI = CI; + Call->UniqueId = UniqueId; Call->NTargets = Targets.size(); std::uninitialized_copy(Targets.begin(), Targets.end(), Call->getTrailingObjects()); @@ -313,6 +315,8 @@ return makeArrayRef(getTrailingObjects(), NTargets); } + unsigned UniqueId; + private: size_t NTargets; }; @@ -1662,11 +1666,11 @@ // identifiers. BumpPtrAllocator Alloc; struct TIInfo { - unsigned Index; + unsigned UniqueId; std::vector RefGlobals; }; DenseMap TypeIdInfo; - unsigned I = 0; + unsigned CurUniqueId = 0; SmallVector Types; struct ExportedFunctionInfo { @@ -1754,7 +1758,7 @@ for (MDNode *Type : Types) { verifyTypeMDNode(&GO, Type); auto &Info = TypeIdInfo[Type->getOperand(1)]; - Info.Index = ++I; + Info.UniqueId = ++CurUniqueId; Info.RefGlobals.push_back(GTM); } } @@ -1823,8 +1827,9 @@ } GlobalClasses.unionSets( - CurSet, GlobalClasses.findLeader(GlobalClasses.insert( - ICallBranchFunnel::create(Alloc, CI, Targets)))); + CurSet, GlobalClasses.findLeader( + GlobalClasses.insert(ICallBranchFunnel::create( + Alloc, CI, Targets, ++CurUniqueId)))); } } @@ -1861,13 +1866,15 @@ continue; ++NumTypeIdDisjointSets; - unsigned MaxIndex = 0; + unsigned MaxUniqueId = 0; for (GlobalClassesTy::member_iterator MI = GlobalClasses.member_begin(I); MI != GlobalClasses.member_end(); ++MI) { - if ((*MI).is()) - MaxIndex = std::max(MaxIndex, TypeIdInfo[MI->get()].Index); + if (auto *MD = MI->dyn_cast()) + MaxUniqueId = std::max(MaxUniqueId, TypeIdInfo[MD].UniqueId); + else if (auto *BF = MI->dyn_cast()) + MaxUniqueId = std::max(MaxUniqueId, BF->UniqueId); } - Sets.emplace_back(I, MaxIndex); + Sets.emplace_back(I, MaxUniqueId); } llvm::sort(Sets.begin(), Sets.end(), [](const std::pair &S1, @@ -1892,12 +1899,18 @@ ICallBranchFunnels.push_back(MI->get()); } - // Order type identifiers by global index for determinism. This ordering is - // stable as there is a one-to-one mapping between metadata and indices. + // Order type identifiers by unique ID for determinism. This ordering is + // stable as there is a one-to-one mapping between metadata and unique IDs. llvm::sort(TypeIds.begin(), TypeIds.end(), [&](Metadata *M1, Metadata *M2) { - return TypeIdInfo[M1].Index < TypeIdInfo[M2].Index; + return TypeIdInfo[M1].UniqueId < TypeIdInfo[M2].UniqueId; }); + // Same for the branch funnels. + llvm::sort(ICallBranchFunnels.begin(), ICallBranchFunnels.end(), + [&](ICallBranchFunnel *F1, ICallBranchFunnel *F2) { + return F1->UniqueId < F2->UniqueId; + }); + // Build bitsets for this disjoint set. buildBitSetsFromDisjointSet(TypeIds, Globals, ICallBranchFunnels); }