DbgInfoIntrinsics no longer appear in an instruction's use list; so clean up looking for them in use iterations and remove OnlyUsedByDbgInfoIntrinsics()

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@94111 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Victor Hernandez 2010-01-21 23:05:53 +00:00
parent eca35b76ce
commit 6c146eefbf
3 changed files with 7 additions and 90 deletions

View File

@ -27,7 +27,6 @@ class PHINode;
class AllocaInst; class AllocaInst;
class ConstantExpr; class ConstantExpr;
class TargetData; class TargetData;
class DbgInfoIntrinsic;
template<typename T> class SmallVectorImpl; template<typename T> class SmallVectorImpl;
@ -154,12 +153,6 @@ AllocaInst *DemoteRegToStack(Instruction &X,
/// The phi node is deleted and it returns the pointer to the alloca inserted. /// The phi node is deleted and it returns the pointer to the alloca inserted.
AllocaInst *DemotePHIToStack(PHINode *P, Instruction *AllocaPoint = 0); AllocaInst *DemotePHIToStack(PHINode *P, Instruction *AllocaPoint = 0);
/// OnlyUsedByDbgIntrinsics - Return true if the instruction I is only used
/// by DbgIntrinsics. If DbgInUses is specified then the vector is filled
/// with DbgInfoIntrinsic that use the instruction I.
bool OnlyUsedByDbgInfoIntrinsics(Instruction *I,
SmallVectorImpl<DbgInfoIntrinsic *> *DbgInUses = 0);
} // End llvm namespace } // End llvm namespace
#endif #endif

View File

@ -85,10 +85,6 @@ namespace {
/// isUnsafe - This is set to true if the alloca cannot be SROA'd. /// isUnsafe - This is set to true if the alloca cannot be SROA'd.
bool isUnsafe : 1; bool isUnsafe : 1;
/// needsCleanup - This is set to true if there is some use of the alloca
/// that requires cleanup.
bool needsCleanup : 1;
/// isMemCpySrc - This is true if this aggregate is memcpy'd from. /// isMemCpySrc - This is true if this aggregate is memcpy'd from.
bool isMemCpySrc : 1; bool isMemCpySrc : 1;
@ -96,15 +92,14 @@ namespace {
bool isMemCpyDst : 1; bool isMemCpyDst : 1;
AllocaInfo() AllocaInfo()
: isUnsafe(false), needsCleanup(false), : isUnsafe(false), isMemCpySrc(false), isMemCpyDst(false) {}
isMemCpySrc(false), isMemCpyDst(false) {}
}; };
unsigned SRThreshold; unsigned SRThreshold;
void MarkUnsafe(AllocaInfo &I) { I.isUnsafe = true; } void MarkUnsafe(AllocaInfo &I) { I.isUnsafe = true; }
int isSafeAllocaToScalarRepl(AllocaInst *AI); bool isSafeAllocaToScalarRepl(AllocaInst *AI);
void isSafeForScalarRepl(Instruction *I, AllocaInst *AI, uint64_t Offset, void isSafeForScalarRepl(Instruction *I, AllocaInst *AI, uint64_t Offset,
AllocaInfo &Info); AllocaInfo &Info);
@ -119,7 +114,6 @@ namespace {
void DoScalarReplacement(AllocaInst *AI, void DoScalarReplacement(AllocaInst *AI,
std::vector<AllocaInst*> &WorkList); std::vector<AllocaInst*> &WorkList);
void DeleteDeadInstructions(); void DeleteDeadInstructions();
void CleanupAllocaUsers(Value *V);
AllocaInst *AddNewAlloca(Function &F, const Type *Ty, AllocaInst *Base); AllocaInst *AddNewAlloca(Function &F, const Type *Ty, AllocaInst *Base);
void RewriteForScalarRepl(Instruction *I, AllocaInst *AI, uint64_t Offset, void RewriteForScalarRepl(Instruction *I, AllocaInst *AI, uint64_t Offset,
@ -281,14 +275,7 @@ bool SROA::performScalarRepl(Function &F) {
getNumSAElements(AI->getAllocatedType()) <= SRThreshold/4) { getNumSAElements(AI->getAllocatedType()) <= SRThreshold/4) {
// Check that all of the users of the allocation are capable of being // Check that all of the users of the allocation are capable of being
// transformed. // transformed.
switch (isSafeAllocaToScalarRepl(AI)) { if (isSafeAllocaToScalarRepl(AI)) {
default: llvm_unreachable("Unexpected value!");
case 0: // Not safe to scalar replace.
break;
case 1: // Safe, but requires cleanup/canonicalizations first
CleanupAllocaUsers(AI);
// FALL THROUGH.
case 3: // Safe to scalar replace.
DoScalarReplacement(AI, WorkList); DoScalarReplacement(AI, WorkList);
Changed = true; Changed = true;
continue; continue;
@ -437,14 +424,6 @@ void SROA::isSafeForScalarRepl(Instruction *I, AllocaInst *AI, uint64_t Offset,
SIType, true, Info); SIType, true, Info);
} else } else
MarkUnsafe(Info); MarkUnsafe(Info);
} else if (isa<DbgInfoIntrinsic>(UI)) {
// If one user is DbgInfoIntrinsic then check if all users are
// DbgInfoIntrinsics.
if (OnlyUsedByDbgInfoIntrinsics(I)) {
Info.needsCleanup = true;
return;
}
MarkUnsafe(Info);
} else { } else {
DEBUG(errs() << " Transformation preventing inst: " << *User << '\n'); DEBUG(errs() << " Transformation preventing inst: " << *User << '\n');
MarkUnsafe(Info); MarkUnsafe(Info);
@ -1150,7 +1129,7 @@ static bool HasPadding(const Type *Ty, const TargetData &TD) {
/// isSafeStructAllocaToScalarRepl - Check to see if the specified allocation of /// isSafeStructAllocaToScalarRepl - Check to see if the specified allocation of
/// an aggregate can be broken down into elements. Return 0 if not, 3 if safe, /// an aggregate can be broken down into elements. Return 0 if not, 3 if safe,
/// or 1 if safe after canonicalization has been performed. /// or 1 if safe after canonicalization has been performed.
int SROA::isSafeAllocaToScalarRepl(AllocaInst *AI) { bool SROA::isSafeAllocaToScalarRepl(AllocaInst *AI) {
// Loop over the use list of the alloca. We can only transform it if all of // Loop over the use list of the alloca. We can only transform it if all of
// the users are safe to transform. // the users are safe to transform.
AllocaInfo Info; AllocaInfo Info;
@ -1158,7 +1137,7 @@ int SROA::isSafeAllocaToScalarRepl(AllocaInst *AI) {
isSafeForScalarRepl(AI, AI, 0, Info); isSafeForScalarRepl(AI, AI, 0, Info);
if (Info.isUnsafe) { if (Info.isUnsafe) {
DEBUG(dbgs() << "Cannot transform: " << *AI << '\n'); DEBUG(dbgs() << "Cannot transform: " << *AI << '\n');
return 0; return false;
} }
// Okay, we know all the users are promotable. If the aggregate is a memcpy // Okay, we know all the users are promotable. If the aggregate is a memcpy
@ -1168,29 +1147,9 @@ int SROA::isSafeAllocaToScalarRepl(AllocaInst *AI) {
// struct. // struct.
if (Info.isMemCpySrc && Info.isMemCpyDst && if (Info.isMemCpySrc && Info.isMemCpyDst &&
HasPadding(AI->getAllocatedType(), *TD)) HasPadding(AI->getAllocatedType(), *TD))
return 0; return false;
// If we require cleanup, return 1, otherwise return 3. return true;
return Info.needsCleanup ? 1 : 3;
}
/// CleanupAllocaUsers - If SROA reported that it can promote the specified
/// allocation, but only if cleaned up, perform the cleanups required.
void SROA::CleanupAllocaUsers(Value *V) {
for (Value::use_iterator UI = V->use_begin(), E = V->use_end();
UI != E; ) {
User *U = *UI++;
Instruction *I = cast<Instruction>(U);
SmallVector<DbgInfoIntrinsic *, 2> DbgInUses;
if (!isa<StoreInst>(I) && OnlyUsedByDbgInfoIntrinsics(I, &DbgInUses)) {
// Safe to remove debug info uses.
while (!DbgInUses.empty()) {
DbgInfoIntrinsic *DI = DbgInUses.pop_back_val();
DI->eraseFromParent();
}
I->eraseFromParent();
}
}
} }
/// MergeInType - Add the 'In' type to the accumulated type (Accum) so far at /// MergeInType - Add the 'In' type to the accumulated type (Accum) so far at
@ -1325,10 +1284,6 @@ bool SROA::CanConvertToScalar(Value *V, bool &IsNotTrivial, const Type *&VecTy,
} }
} }
// Ignore dbg intrinsic.
if (isa<DbgInfoIntrinsic>(User))
continue;
// Otherwise, we cannot handle this! // Otherwise, we cannot handle this!
return false; return false;
} }
@ -1453,18 +1408,11 @@ void SROA::ConvertUsesToScalar(Value *Ptr, AllocaInst *NewAI, uint64_t Offset) {
} else { } else {
// Noop transfer. Src == Dst // Noop transfer. Src == Dst
} }
MTI->eraseFromParent(); MTI->eraseFromParent();
continue; continue;
} }
// If user is a dbg info intrinsic then it is safe to remove it.
if (isa<DbgInfoIntrinsic>(User)) {
User->eraseFromParent();
continue;
}
llvm_unreachable("Unsupported operation!"); llvm_unreachable("Unsupported operation!");
} }
} }

View File

@ -609,30 +609,6 @@ bool llvm::TryToSimplifyUncondBranchFromEmptyBlock(BasicBlock *BB) {
return true; return true;
} }
/// OnlyUsedByDbgIntrinsics - Return true if the instruction I is only used
/// by DbgIntrinsics. If DbgInUses is specified then the vector is filled
/// with the DbgInfoIntrinsic that use the instruction I.
bool llvm::OnlyUsedByDbgInfoIntrinsics(Instruction *I,
SmallVectorImpl<DbgInfoIntrinsic *> *DbgInUses) {
if (DbgInUses)
DbgInUses->clear();
for (Value::use_iterator UI = I->use_begin(), UE = I->use_end(); UI != UE;
++UI) {
if (DbgInfoIntrinsic *DI = dyn_cast<DbgInfoIntrinsic>(*UI)) {
if (DbgInUses)
DbgInUses->push_back(DI);
} else {
if (DbgInUses)
DbgInUses->clear();
return false;
}
}
return true;
}
/// EliminateDuplicatePHINodes - Check for and eliminate duplicate PHI /// EliminateDuplicatePHINodes - Check for and eliminate duplicate PHI
/// nodes in this block. This doesn't try to be clever about PHI nodes /// nodes in this block. This doesn't try to be clever about PHI nodes
/// which differ only in the order of the incoming values, but instcombine /// which differ only in the order of the incoming values, but instcombine