mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-27 13:30:05 +00:00
Make instcombine promote inline asm calls to 'nounwind'
calls. Remove special casing of inline asm from the inliner. There is a potential problem: the verifier rejects invokes of inline asm (not sure why). If an asm call is not marked "nounwind" in some .ll, and instcombine is not run, but the inliner is run, then an illegal module will be created. This is bad but I'm not sure what the best approach is. I'm tempted to remove the check in the verifier... git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@45073 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
2c6fd8c7ce
commit
ece2c04d53
@ -73,6 +73,9 @@ public:
|
||||
/// @brief Determine if the call does not access or only reads memory.
|
||||
bool onlyReadsMemory() const;
|
||||
|
||||
/// @brief Determine if the call cannot unwind.
|
||||
bool isNoUnwind() const;
|
||||
|
||||
/// getType - Return the type of the instruction that generated this call site
|
||||
///
|
||||
const Type *getType() const { return I->getType(); }
|
||||
|
@ -7972,6 +7972,19 @@ Instruction *InstCombiner::visitCallSite(CallSite CS) {
|
||||
}
|
||||
}
|
||||
|
||||
if (isa<InlineAsm>(Callee) && !CS.isNoUnwind()) {
|
||||
// Inline asm calls cannot throw - mark them 'nounwind'.
|
||||
const ParamAttrsList *PAL = CS.getParamAttrs();
|
||||
uint16_t RAttributes = PAL ? PAL->getParamAttrs(0) : 0;
|
||||
RAttributes |= ParamAttr::NoUnwind;
|
||||
|
||||
ParamAttrsVector modVec;
|
||||
modVec.push_back(ParamAttrsWithIndex::get(0, RAttributes));
|
||||
PAL = ParamAttrsList::getModified(PAL, modVec);
|
||||
CS.setParamAttrs(PAL);
|
||||
Changed = true;
|
||||
}
|
||||
|
||||
return Changed ? CS.getInstruction() : 0;
|
||||
}
|
||||
|
||||
|
@ -69,9 +69,8 @@ static void HandleInlinedInvoke(InvokeInst *II, BasicBlock *FirstNewBlock,
|
||||
if (!isa<CallInst>(I)) continue;
|
||||
CallInst *CI = cast<CallInst>(I);
|
||||
|
||||
// If this call cannot unwind or is an inline asm, don't
|
||||
// convert it to an invoke.
|
||||
if (CI->isNoUnwind() || isa<InlineAsm>(CI->getCalledValue()))
|
||||
// If this call cannot unwind, don't convert it to an invoke.
|
||||
if (CI->isNoUnwind())
|
||||
continue;
|
||||
|
||||
// Convert this function call into an invoke instruction.
|
||||
|
@ -65,6 +65,12 @@ bool CallSite::onlyReadsMemory() const {
|
||||
else
|
||||
return cast<InvokeInst>(I)->onlyReadsMemory();
|
||||
}
|
||||
bool CallSite::isNoUnwind() const {
|
||||
if (CallInst *CI = dyn_cast<CallInst>(I))
|
||||
return CI->isNoUnwind();
|
||||
else
|
||||
return cast<InvokeInst>(I)->isNoUnwind();
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
3
test/CFrontend/2007-12-16-AsmNoUnwind.c
Normal file
3
test/CFrontend/2007-12-16-AsmNoUnwind.c
Normal file
@ -0,0 +1,3 @@
|
||||
// RUN: %llvmgcc %s -S -o - | grep nounwind
|
||||
|
||||
void bar() { asm (""); }
|
@ -8,7 +8,7 @@ target triple = "i686-pc-linux-gnu"
|
||||
|
||||
define void @bc__support__high_resolution_time__clock() {
|
||||
entry:
|
||||
call void asm "rdtsc\0A\09movl %eax, $0\0A\09movl %edx, $1", "=*imr,=*imr,~{dirflag},~{fpsr},~{flags},~{dx},~{ax}"( i32* null, i32* null )
|
||||
call void asm "rdtsc\0A\09movl %eax, $0\0A\09movl %edx, $1", "=*imr,=*imr,~{dirflag},~{fpsr},~{flags},~{dx},~{ax}"( i32* null, i32* null ) nounwind
|
||||
unreachable
|
||||
}
|
||||
|
||||
|
7
test/Transforms/InstCombine/2007-12-16-AsmNoUnwind.ll
Normal file
7
test/Transforms/InstCombine/2007-12-16-AsmNoUnwind.ll
Normal file
@ -0,0 +1,7 @@
|
||||
; RUN: llvm-as < %s | opt -instcombine | llvm-dis | grep nounwind
|
||||
|
||||
define void @bar() {
|
||||
entry:
|
||||
call void asm sideeffect "", "~{dirflag},~{fpsr},~{flags}"( )
|
||||
ret void
|
||||
}
|
Loading…
Reference in New Issue
Block a user