mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-09-24 23:28:41 +00:00
While inlining, clone llvm.dbg.func.start intrinsic and adjust
llvm.dbg.region.end instrinsic. This nested llvm.dbg.func.start/llvm.dbg.region.end pair now enables DW_TAG_inlined_subroutine support in code generator. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@69118 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
@@ -24,6 +24,7 @@
|
|||||||
#include "llvm/Support/Compiler.h"
|
#include "llvm/Support/Compiler.h"
|
||||||
#include "llvm/Transforms/Utils/ValueMapper.h"
|
#include "llvm/Transforms/Utils/ValueMapper.h"
|
||||||
#include "llvm/Analysis/ConstantFolding.h"
|
#include "llvm/Analysis/ConstantFolding.h"
|
||||||
|
#include "llvm/Analysis/DebugInfo.h"
|
||||||
#include "llvm/ADT/SmallVector.h"
|
#include "llvm/ADT/SmallVector.h"
|
||||||
#include <map>
|
#include <map>
|
||||||
using namespace llvm;
|
using namespace llvm;
|
||||||
@@ -233,10 +234,13 @@ void PruningFunctionCloner::CloneBlock(const BasicBlock *BB,
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Do not clone llvm.dbg.func.start and corresponding llvm.dbg.region.end.
|
// Do not clone llvm.dbg.region.end. It will be adjusted by the inliner.
|
||||||
if (const DbgFuncStartInst *DFSI = dyn_cast<DbgFuncStartInst>(II)) {
|
if (const DbgFuncStartInst *DFSI = dyn_cast<DbgFuncStartInst>(II)) {
|
||||||
DbgFnStart = DFSI->getSubprogram();
|
if (DbgFnStart == NULL) {
|
||||||
continue;
|
DISubprogram SP(cast<GlobalVariable>(DFSI->getSubprogram()));
|
||||||
|
if (SP.describes(BB->getParent()))
|
||||||
|
DbgFnStart = DFSI->getSubprogram();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (const DbgRegionEndInst *DREIS = dyn_cast<DbgRegionEndInst>(II)) {
|
if (const DbgRegionEndInst *DREIS = dyn_cast<DbgRegionEndInst>(II)) {
|
||||||
if (DREIS->getContext() == DbgFnStart)
|
if (DREIS->getContext() == DbgFnStart)
|
||||||
|
@@ -17,9 +17,11 @@
|
|||||||
#include "llvm/DerivedTypes.h"
|
#include "llvm/DerivedTypes.h"
|
||||||
#include "llvm/Module.h"
|
#include "llvm/Module.h"
|
||||||
#include "llvm/Instructions.h"
|
#include "llvm/Instructions.h"
|
||||||
|
#include "llvm/IntrinsicInst.h"
|
||||||
#include "llvm/Intrinsics.h"
|
#include "llvm/Intrinsics.h"
|
||||||
#include "llvm/Attributes.h"
|
#include "llvm/Attributes.h"
|
||||||
#include "llvm/Analysis/CallGraph.h"
|
#include "llvm/Analysis/CallGraph.h"
|
||||||
|
#include "llvm/Analysis/DebugInfo.h"
|
||||||
#include "llvm/Target/TargetData.h"
|
#include "llvm/Target/TargetData.h"
|
||||||
#include "llvm/ADT/SmallVector.h"
|
#include "llvm/ADT/SmallVector.h"
|
||||||
#include "llvm/ADT/StringExtras.h"
|
#include "llvm/ADT/StringExtras.h"
|
||||||
@@ -199,6 +201,31 @@ static void UpdateCallGraphAfterInlining(CallSite CS,
|
|||||||
CallerNode->removeCallEdgeFor(CS);
|
CallerNode->removeCallEdgeFor(CS);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// findFnRegionEndMarker - This is a utility routine that is used by
|
||||||
|
/// InlineFunction. Return llvm.dbg.region.end intrinsic that corresponds
|
||||||
|
/// to the llvm.dbg.func.start of the function F. Otherwise return NULL.
|
||||||
|
static const DbgRegionEndInst *findFnRegionEndMarker(const Function *F) {
|
||||||
|
|
||||||
|
GlobalVariable *FnStart = NULL;
|
||||||
|
const DbgRegionEndInst *FnEnd = NULL;
|
||||||
|
for (Function::const_iterator FI = F->begin(), FE =F->end(); FI != FE; ++FI)
|
||||||
|
for (BasicBlock::const_iterator BI = FI->begin(), BE = FI->end(); BI != BE;
|
||||||
|
++BI) {
|
||||||
|
if (FnStart == NULL) {
|
||||||
|
if (const DbgFuncStartInst *FSI = dyn_cast<DbgFuncStartInst>(BI)) {
|
||||||
|
DISubprogram SP(cast<GlobalVariable>(FSI->getSubprogram()));
|
||||||
|
assert (SP.isNull() == false && "Invalid llvm.dbg.func.start");
|
||||||
|
if (SP.describes(F))
|
||||||
|
FnStart = SP.getGV();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (const DbgRegionEndInst *REI = dyn_cast<DbgRegionEndInst>(BI))
|
||||||
|
if (REI->getContext() == FnStart)
|
||||||
|
FnEnd = REI;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return FnEnd;
|
||||||
|
}
|
||||||
|
|
||||||
// InlineFunction - This function inlines the called function into the basic
|
// InlineFunction - This function inlines the called function into the basic
|
||||||
// block of the caller. This returns false if it is not possible to inline this
|
// block of the caller. This returns false if it is not possible to inline this
|
||||||
@@ -320,6 +347,24 @@ bool llvm::InlineFunction(CallSite CS, CallGraph *CG, const TargetData *TD) {
|
|||||||
ValueMap[I] = ActualArg;
|
ValueMap[I] = ActualArg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Adjust llvm.dbg.region.end. If the CalledFunc has region end
|
||||||
|
// marker then clone that marker after next stop point at the
|
||||||
|
// call site. The function body cloner does not clone original
|
||||||
|
// region end marker from the CalledFunc. This will ensure that
|
||||||
|
// inlined function's scope ends at the right place.
|
||||||
|
const DbgRegionEndInst *DREI = findFnRegionEndMarker(CalledFunc);
|
||||||
|
if (DREI) {
|
||||||
|
for (BasicBlock::iterator BI = TheCall,
|
||||||
|
BE = TheCall->getParent()->end(); BI != BE; ++BI) {
|
||||||
|
if (DbgStopPointInst *DSPI = dyn_cast<DbgStopPointInst>(BI)) {
|
||||||
|
if (DbgRegionEndInst *NewDREI =
|
||||||
|
dyn_cast<DbgRegionEndInst>(DREI->clone()))
|
||||||
|
NewDREI->insertAfter(DSPI);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// We want the inliner to prune the code as it copies. We would LOVE to
|
// We want the inliner to prune the code as it copies. We would LOVE to
|
||||||
// have no dead or constant instructions leftover after inlining occurs
|
// have no dead or constant instructions leftover after inlining occurs
|
||||||
// (which can happen, e.g., because an argument was constant), but we'll be
|
// (which can happen, e.g., because an argument was constant), but we'll be
|
||||||
|
@@ -1,86 +0,0 @@
|
|||||||
; RUN: llvm-as < %s | opt -inline | llvm-dis | grep func.start | count 3
|
|
||||||
; RUN: llvm-as < %s | opt -inline | llvm-dis | grep region.end | count 3
|
|
||||||
%llvm.dbg.anchor.type = type { i32, i32 }
|
|
||||||
%llvm.dbg.basictype.type = type { i32, { }*, i8*, { }*, i32, i64, i64, i64, i32, i32 }
|
|
||||||
%llvm.dbg.compile_unit.type = type { i32, { }*, i32, i8*, i8*, i8*, i1, i1, i8* }
|
|
||||||
%llvm.dbg.composite.type = type { i32, { }*, i8*, { }*, i32, i64, i64, i64, i32, { }*, { }* }
|
|
||||||
%llvm.dbg.derivedtype.type = type { i32, { }*, i8*, { }*, i32, i64, i64, i64, i32, { }* }
|
|
||||||
%llvm.dbg.subprogram.type = type { i32, { }*, { }*, i8*, i8*, i8*, { }*, i32, { }*, i1, i1 }
|
|
||||||
%llvm.dbg.variable.type = type { i32, { }*, i8*, { }*, i32, { }* }
|
|
||||||
@llvm.dbg.compile_units = linkonce constant %llvm.dbg.anchor.type { i32 458752, i32 17 }, section "llvm.metadata" ; <%llvm.dbg.anchor.type*> [#uses=1]
|
|
||||||
@.str = internal constant [4 x i8] c"a.c\00", section "llvm.metadata" ; <[4 x i8]*> [#uses=1]
|
|
||||||
@.str1 = internal constant [6 x i8] c"/tmp/\00", section "llvm.metadata" ; <[6 x i8]*> [#uses=1]
|
|
||||||
@.str2 = internal constant [55 x i8] c"4.2.1 (Based on Apple Inc. build 5636) (LLVM build 00)\00", section "llvm.metadata" ; <[55 x i8]*> [#uses=1]
|
|
||||||
@llvm.dbg.compile_unit = internal constant %llvm.dbg.compile_unit.type { i32 458769, { }* bitcast (%llvm.dbg.anchor.type* @llvm.dbg.compile_units to { }*), i32 1, i8* getelementptr ([4 x i8]* @.str, i32 0, i32 0), i8* getelementptr ([6 x i8]* @.str1, i32 0, i32 0), i8* getelementptr ([55 x i8]* @.str2, i32 0, i32 0), i1 true, i1 false, i8* null }, section "llvm.metadata" ; <%llvm.dbg.compile_unit.type*> [#uses=1]
|
|
||||||
@.str3 = internal constant [4 x i8] c"int\00", section "llvm.metadata" ; <[4 x i8]*> [#uses=1]
|
|
||||||
@llvm.dbg.basictype = internal constant %llvm.dbg.basictype.type { i32 458788, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i8* getelementptr ([4 x i8]* @.str3, i32 0, i32 0), { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i32 0, i64 32, i64 32, i64 0, i32 0, i32 5 }, section "llvm.metadata" ; <%llvm.dbg.basictype.type*> [#uses=1]
|
|
||||||
@.str4 = internal constant [5 x i8] c"char\00", section "llvm.metadata" ; <[5 x i8]*> [#uses=1]
|
|
||||||
@llvm.dbg.basictype5 = internal constant %llvm.dbg.basictype.type { i32 458788, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i8* getelementptr ([5 x i8]* @.str4, i32 0, i32 0), { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i32 0, i64 8, i64 8, i64 0, i32 0, i32 6 }, section "llvm.metadata" ; <%llvm.dbg.basictype.type*> [#uses=1]
|
|
||||||
@llvm.dbg.derivedtype = internal constant %llvm.dbg.derivedtype.type { i32 458767, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i8* null, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i32 0, i64 32, i64 32, i64 0, i32 0, { }* bitcast (%llvm.dbg.basictype.type* @llvm.dbg.basictype5 to { }*) }, section "llvm.metadata" ; <%llvm.dbg.derivedtype.type*> [#uses=1]
|
|
||||||
@llvm.dbg.array = internal constant [2 x { }*] [ { }* bitcast (%llvm.dbg.basictype.type* @llvm.dbg.basictype to { }*), { }* bitcast (%llvm.dbg.derivedtype.type* @llvm.dbg.derivedtype to { }*) ], section "llvm.metadata" ; <[2 x { }*]*> [#uses=1]
|
|
||||||
@llvm.dbg.composite = internal constant %llvm.dbg.composite.type { i32 458773, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i8* null, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i32 0, i64 0, i64 0, i64 0, i32 0, { }* null, { }* bitcast ([2 x { }*]* @llvm.dbg.array to { }*) }, section "llvm.metadata" ; <%llvm.dbg.composite.type*> [#uses=1]
|
|
||||||
@llvm.dbg.subprograms = linkonce constant %llvm.dbg.anchor.type { i32 458752, i32 46 }, section "llvm.metadata" ; <%llvm.dbg.anchor.type*> [#uses=1]
|
|
||||||
@.str6 = internal constant [4 x i8] c"bar\00", section "llvm.metadata" ; <[4 x i8]*> [#uses=1]
|
|
||||||
@llvm.dbg.subprogram = internal constant %llvm.dbg.subprogram.type { i32 458798, { }* bitcast (%llvm.dbg.anchor.type* @llvm.dbg.subprograms to { }*), { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i8* getelementptr ([4 x i8]* @.str6, i32 0, i32 0), i8* getelementptr ([4 x i8]* @.str6, i32 0, i32 0), i8* null, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i32 1, { }* bitcast (%llvm.dbg.composite.type* @llvm.dbg.composite to { }*), i1 false, i1 true }, section "llvm.metadata" ; <%llvm.dbg.subprogram.type*> [#uses=1]
|
|
||||||
@.str7 = internal constant [2 x i8] c"c\00", section "llvm.metadata" ; <[2 x i8]*> [#uses=1]
|
|
||||||
@llvm.dbg.variable = internal constant %llvm.dbg.variable.type { i32 459009, { }* bitcast (%llvm.dbg.subprogram.type* @llvm.dbg.subprogram to { }*), i8* getelementptr ([2 x i8]* @.str7, i32 0, i32 0), { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i32 1, { }* bitcast (%llvm.dbg.derivedtype.type* @llvm.dbg.derivedtype to { }*) }, section "llvm.metadata" ; <%llvm.dbg.variable.type*> [#uses=1]
|
|
||||||
@llvm.dbg.array8 = internal constant [1 x { }*] [ { }* bitcast (%llvm.dbg.basictype.type* @llvm.dbg.basictype to { }*) ], section "llvm.metadata" ; <[1 x { }*]*> [#uses=1]
|
|
||||||
@llvm.dbg.composite9 = internal constant %llvm.dbg.composite.type { i32 458773, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i8* null, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i32 0, i64 0, i64 0, i64 0, i32 0, { }* null, { }* bitcast ([1 x { }*]* @llvm.dbg.array8 to { }*) }, section "llvm.metadata" ; <%llvm.dbg.composite.type*> [#uses=1]
|
|
||||||
@.str10 = internal constant [5 x i8] c"main\00", section "llvm.metadata" ; <[5 x i8]*> [#uses=1]
|
|
||||||
@llvm.dbg.subprogram11 = internal constant %llvm.dbg.subprogram.type { i32 458798, { }* bitcast (%llvm.dbg.anchor.type* @llvm.dbg.subprograms to { }*), { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i8* getelementptr ([5 x i8]* @.str10, i32 0, i32 0), i8* getelementptr ([5 x i8]* @.str10, i32 0, i32 0), i8* null, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*), i32 2, { }* bitcast (%llvm.dbg.composite.type* @llvm.dbg.composite9 to { }*), i1 false, i1 true }, section "llvm.metadata" ; <%llvm.dbg.subprogram.type*> [#uses=1]
|
|
||||||
@"\01LC" = internal constant [3 x i8] c"hi\00" ; <[3 x i8]*> [#uses=1]
|
|
||||||
|
|
||||||
define i32 @bar(i8* %c) nounwind {
|
|
||||||
entry:
|
|
||||||
%c_addr = alloca i8* ; <i8**> [#uses=3]
|
|
||||||
%retval = alloca i32 ; <i32*> [#uses=2]
|
|
||||||
%0 = alloca i32 ; <i32*> [#uses=2]
|
|
||||||
%"alloca point" = bitcast i32 0 to i32 ; <i32> [#uses=0]
|
|
||||||
call void @llvm.dbg.func.start({ }* bitcast (%llvm.dbg.subprogram.type* @llvm.dbg.subprogram to { }*))
|
|
||||||
%1 = bitcast i8** %c_addr to { }* ; <{ }*> [#uses=1]
|
|
||||||
call void @llvm.dbg.declare({ }* %1, { }* bitcast (%llvm.dbg.variable.type* @llvm.dbg.variable to { }*))
|
|
||||||
store i8* %c, i8** %c_addr
|
|
||||||
call void @llvm.dbg.stoppoint(i32 1, i32 0, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*))
|
|
||||||
%2 = load i8** %c_addr, align 4 ; <i8*> [#uses=1]
|
|
||||||
%3 = load i8* %2, align 1 ; <i8> [#uses=1]
|
|
||||||
%4 = sext i8 %3 to i32 ; <i32> [#uses=1]
|
|
||||||
%5 = add i32 %4, 42 ; <i32> [#uses=1]
|
|
||||||
store i32 %5, i32* %0, align 4
|
|
||||||
%6 = load i32* %0, align 4 ; <i32> [#uses=1]
|
|
||||||
store i32 %6, i32* %retval, align 4
|
|
||||||
br label %return
|
|
||||||
|
|
||||||
return: ; preds = %entry
|
|
||||||
%retval1 = load i32* %retval ; <i32> [#uses=1]
|
|
||||||
call void @llvm.dbg.stoppoint(i32 1, i32 0, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*))
|
|
||||||
call void @llvm.dbg.region.end({ }* bitcast (%llvm.dbg.subprogram.type* @llvm.dbg.subprogram to { }*))
|
|
||||||
ret i32 %retval1
|
|
||||||
}
|
|
||||||
|
|
||||||
declare void @llvm.dbg.func.start({ }*) nounwind
|
|
||||||
|
|
||||||
declare void @llvm.dbg.declare({ }*, { }*) nounwind
|
|
||||||
|
|
||||||
declare void @llvm.dbg.stoppoint(i32, i32, { }*) nounwind
|
|
||||||
|
|
||||||
declare void @llvm.dbg.region.end({ }*) nounwind
|
|
||||||
|
|
||||||
define i32 @main() nounwind {
|
|
||||||
entry:
|
|
||||||
%retval = alloca i32 ; <i32*> [#uses=2]
|
|
||||||
%0 = alloca i32 ; <i32*> [#uses=2]
|
|
||||||
%"alloca point" = bitcast i32 0 to i32 ; <i32> [#uses=0]
|
|
||||||
call void @llvm.dbg.func.start({ }* bitcast (%llvm.dbg.subprogram.type* @llvm.dbg.subprogram11 to { }*))
|
|
||||||
call void @llvm.dbg.stoppoint(i32 2, i32 0, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*))
|
|
||||||
%1 = call i32 @bar(i8* getelementptr ([3 x i8]* @"\01LC", i32 0, i32 0)) nounwind ; <i32> [#uses=1]
|
|
||||||
store i32 %1, i32* %0, align 4
|
|
||||||
%2 = load i32* %0, align 4 ; <i32> [#uses=1]
|
|
||||||
store i32 %2, i32* %retval, align 4
|
|
||||||
br label %return
|
|
||||||
|
|
||||||
return: ; preds = %entry
|
|
||||||
%retval1 = load i32* %retval ; <i32> [#uses=1]
|
|
||||||
call void @llvm.dbg.stoppoint(i32 2, i32 0, { }* bitcast (%llvm.dbg.compile_unit.type* @llvm.dbg.compile_unit to { }*))
|
|
||||||
call void @llvm.dbg.region.end({ }* bitcast (%llvm.dbg.subprogram.type* @llvm.dbg.subprogram11 to { }*))
|
|
||||||
ret i32 %retval1
|
|
||||||
}
|
|
Reference in New Issue
Block a user