mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-26 04:32:44 +01:00
add bitcode reader support for blockaddress. We can now fully
round trip blockaddress through .ll and .bc files, so add a testcase. There are still a bunch of places in the optimizer and other places that need to be updated to work with these constructs, but at least the basics are in now. llvm-svn: 85377
This commit is contained in:
parent
075e225442
commit
1efa0c450a
@ -1192,6 +1192,22 @@ bool BitcodeReader::ParseConstants() {
|
|||||||
AsmStr, ConstrStr, HasSideEffects, IsAlignStack);
|
AsmStr, ConstrStr, HasSideEffects, IsAlignStack);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case bitc::CST_CODE_BLOCKADDRESS:{
|
||||||
|
if (Record.size() < 3) return Error("Invalid CE_BLOCKADDRESS record");
|
||||||
|
const Type *FnTy = getTypeByID(Record[0]);
|
||||||
|
if (FnTy == 0) return Error("Invalid CE_BLOCKADDRESS record");
|
||||||
|
Function *Fn =
|
||||||
|
dyn_cast_or_null<Function>(ValueList.getConstantFwdRef(Record[1],FnTy));
|
||||||
|
if (Fn == 0) return Error("Invalid CE_BLOCKADDRESS record");
|
||||||
|
|
||||||
|
GlobalVariable *FwdRef = new GlobalVariable(*Fn->getParent(),
|
||||||
|
Type::getInt8Ty(Context),
|
||||||
|
false, GlobalValue::InternalLinkage,
|
||||||
|
0, "");
|
||||||
|
BlockAddrFwdRefs[Fn].push_back(std::make_pair(Record[2], FwdRef));
|
||||||
|
V = FwdRef;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ValueList.AssignValue(V, NextCstNo);
|
ValueList.AssignValue(V, NextCstNo);
|
||||||
@ -2248,6 +2264,27 @@ bool BitcodeReader::ParseFunctionBody(Function *F) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// See if anything took the address of blocks in this function. If so,
|
||||||
|
// resolve them now.
|
||||||
|
/// BlockAddrFwdRefs - These are blockaddr references to basic blocks. These
|
||||||
|
/// are resolved lazily when functions are loaded.
|
||||||
|
DenseMap<Function*, std::vector<BlockAddrRefTy> >::iterator BAFRI =
|
||||||
|
BlockAddrFwdRefs.find(F);
|
||||||
|
if (BAFRI != BlockAddrFwdRefs.end()) {
|
||||||
|
std::vector<BlockAddrRefTy> &RefList = BAFRI->second;
|
||||||
|
for (unsigned i = 0, e = RefList.size(); i != e; ++i) {
|
||||||
|
unsigned BlockIdx = RefList[i].first;
|
||||||
|
if (BlockIdx >= FunctionBBs.size())
|
||||||
|
return Error("Invalid blockaddress block #");
|
||||||
|
|
||||||
|
GlobalVariable *FwdRef = RefList[i].second;
|
||||||
|
FwdRef->replaceAllUsesWith(BlockAddress::get(F, FunctionBBs[BlockIdx]));
|
||||||
|
FwdRef->eraseFromParent();
|
||||||
|
}
|
||||||
|
|
||||||
|
BlockAddrFwdRefs.erase(BAFRI);
|
||||||
|
}
|
||||||
|
|
||||||
// Trim the value list down to the size it was before we parsed this function.
|
// Trim the value list down to the size it was before we parsed this function.
|
||||||
ValueList.shrinkTo(ModuleValueListSize);
|
ValueList.shrinkTo(ModuleValueListSize);
|
||||||
std::vector<BasicBlock*>().swap(FunctionBBs);
|
std::vector<BasicBlock*>().swap(FunctionBBs);
|
||||||
|
@ -94,7 +94,7 @@ public:
|
|||||||
class BitcodeReaderMDValueList {
|
class BitcodeReaderMDValueList {
|
||||||
std::vector<WeakVH> MDValuePtrs;
|
std::vector<WeakVH> MDValuePtrs;
|
||||||
|
|
||||||
LLVMContext& Context;
|
LLVMContext &Context;
|
||||||
public:
|
public:
|
||||||
BitcodeReaderMDValueList(LLVMContext& C) : Context(C) {}
|
BitcodeReaderMDValueList(LLVMContext& C) : Context(C) {}
|
||||||
|
|
||||||
@ -122,7 +122,7 @@ public:
|
|||||||
};
|
};
|
||||||
|
|
||||||
class BitcodeReader : public ModuleProvider {
|
class BitcodeReader : public ModuleProvider {
|
||||||
LLVMContext& Context;
|
LLVMContext &Context;
|
||||||
MemoryBuffer *Buffer;
|
MemoryBuffer *Buffer;
|
||||||
BitstreamReader StreamFile;
|
BitstreamReader StreamFile;
|
||||||
BitstreamCursor Stream;
|
BitstreamCursor Stream;
|
||||||
@ -163,6 +163,12 @@ class BitcodeReader : public ModuleProvider {
|
|||||||
/// map contains info about where to find deferred function body (in the
|
/// map contains info about where to find deferred function body (in the
|
||||||
/// stream) and what linkage the original function had.
|
/// stream) and what linkage the original function had.
|
||||||
DenseMap<Function*, std::pair<uint64_t, unsigned> > DeferredFunctionInfo;
|
DenseMap<Function*, std::pair<uint64_t, unsigned> > DeferredFunctionInfo;
|
||||||
|
|
||||||
|
/// BlockAddrFwdRefs - These are blockaddr references to basic blocks. These
|
||||||
|
/// are resolved lazily when functions are loaded.
|
||||||
|
typedef std::pair<unsigned, GlobalVariable*> BlockAddrRefTy;
|
||||||
|
DenseMap<Function*, std::vector<BlockAddrRefTy> > BlockAddrFwdRefs;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit BitcodeReader(MemoryBuffer *buffer, LLVMContext& C)
|
explicit BitcodeReader(MemoryBuffer *buffer, LLVMContext& C)
|
||||||
: Context(C), Buffer(buffer), ErrorString(0), ValueList(C), MDValueList(C) {
|
: Context(C), Buffer(buffer), ErrorString(0), ValueList(C), MDValueList(C) {
|
||||||
|
@ -25,13 +25,19 @@ Case4: ; preds = %0
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Addr = global i8* blockaddress(@indbrtest, %BB1)
|
||||||
|
@Addr3 = global i8* blockaddress(@squared, %Case1)
|
||||||
|
|
||||||
|
|
||||||
define i32 @indbrtest(i8* %P, i32* %Q) {
|
define i32 @indbrtest(i8* %P, i32* %Q) {
|
||||||
indirectbr i8* %P, [label %BB1, label %BB2, label %BB3]
|
indirectbr i8* %P, [label %BB1, label %BB2, label %BB3]
|
||||||
BB1:
|
BB1:
|
||||||
indirectbr i32* %Q, []
|
indirectbr i32* %Q, []
|
||||||
BB2:
|
BB2:
|
||||||
indirectbr i32* %Q, [label %BB1, label %BB2]
|
%R = bitcast i8* blockaddress(@indbrtest, %BB3) to i8*
|
||||||
|
indirectbr i8* %R, [label %BB1, label %BB2, label %BB3]
|
||||||
BB3:
|
BB3:
|
||||||
ret i32 2
|
ret i32 2
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user