2007-04-22 06:23:29 +00:00
|
|
|
//===- BitcodeReader.cpp - Internal BitcodeReader implementation ----------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-12-29 20:36:04 +00:00
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
2007-04-22 06:23:29 +00:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This header defines the BitcodeReader class.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2007-04-29 07:54:31 +00:00
|
|
|
#include "llvm/Bitcode/ReaderWriter.h"
|
2007-04-22 06:23:29 +00:00
|
|
|
#include "BitcodeReader.h"
|
2007-04-24 03:30:34 +00:00
|
|
|
#include "llvm/Constants.h"
|
2007-04-22 06:23:29 +00:00
|
|
|
#include "llvm/DerivedTypes.h"
|
2007-05-06 01:58:20 +00:00
|
|
|
#include "llvm/InlineAsm.h"
|
2007-05-01 07:01:57 +00:00
|
|
|
#include "llvm/Instructions.h"
|
2009-07-07 20:18:58 +00:00
|
|
|
#include "llvm/LLVMContext.h"
|
2009-05-10 20:57:05 +00:00
|
|
|
#include "llvm/MDNode.h"
|
2007-04-22 06:23:29 +00:00
|
|
|
#include "llvm/Module.h"
|
2009-07-20 21:19:07 +00:00
|
|
|
#include "llvm/Operator.h"
|
2007-08-04 01:51:18 +00:00
|
|
|
#include "llvm/AutoUpgrade.h"
|
2007-04-23 21:26:05 +00:00
|
|
|
#include "llvm/ADT/SmallString.h"
|
2008-02-26 19:38:17 +00:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2007-04-24 04:04:35 +00:00
|
|
|
#include "llvm/Support/MathExtras.h"
|
2007-04-29 07:54:31 +00:00
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
2008-05-10 08:32:32 +00:00
|
|
|
#include "llvm/OperandTraits.h"
|
2007-04-22 06:23:29 +00:00
|
|
|
using namespace llvm;
|
|
|
|
|
2007-05-18 04:02:46 +00:00
|
|
|
void BitcodeReader::FreeState() {
|
2007-04-29 07:54:31 +00:00
|
|
|
delete Buffer;
|
2007-05-18 04:02:46 +00:00
|
|
|
Buffer = 0;
|
|
|
|
std::vector<PATypeHolder>().swap(TypeList);
|
|
|
|
ValueList.clear();
|
2008-03-12 02:25:52 +00:00
|
|
|
|
2008-09-26 22:53:05 +00:00
|
|
|
std::vector<AttrListPtr>().swap(MAttributes);
|
2007-05-18 04:02:46 +00:00
|
|
|
std::vector<BasicBlock*>().swap(FunctionBBs);
|
|
|
|
std::vector<Function*>().swap(FunctionsWithBodies);
|
|
|
|
DeferredFunctionInfo.clear();
|
2007-04-29 07:54:31 +00:00
|
|
|
}
|
|
|
|
|
2007-05-04 03:30:17 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Helper functions to implement forward reference resolution, etc.
|
|
|
|
//===----------------------------------------------------------------------===//
|
2007-04-29 07:54:31 +00:00
|
|
|
|
2007-04-22 06:23:29 +00:00
|
|
|
/// ConvertToString - Convert a string from a record into an std::string, return
|
|
|
|
/// true on failure.
|
2007-04-23 21:26:05 +00:00
|
|
|
template<typename StrTy>
|
2007-04-22 06:23:29 +00:00
|
|
|
static bool ConvertToString(SmallVector<uint64_t, 64> &Record, unsigned Idx,
|
2007-04-23 21:26:05 +00:00
|
|
|
StrTy &Result) {
|
2007-05-04 19:11:41 +00:00
|
|
|
if (Idx > Record.size())
|
2007-04-22 06:23:29 +00:00
|
|
|
return true;
|
|
|
|
|
2007-05-04 19:11:41 +00:00
|
|
|
for (unsigned i = Idx, e = Record.size(); i != e; ++i)
|
|
|
|
Result += (char)Record[i];
|
2007-04-22 06:23:29 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GlobalValue::LinkageTypes GetDecodedLinkage(unsigned Val) {
|
|
|
|
switch (Val) {
|
|
|
|
default: // Map unknown/new linkages to external
|
2009-07-20 01:03:30 +00:00
|
|
|
case 0: return GlobalValue::ExternalLinkage;
|
|
|
|
case 1: return GlobalValue::WeakAnyLinkage;
|
|
|
|
case 2: return GlobalValue::AppendingLinkage;
|
|
|
|
case 3: return GlobalValue::InternalLinkage;
|
|
|
|
case 4: return GlobalValue::LinkOnceAnyLinkage;
|
|
|
|
case 5: return GlobalValue::DLLImportLinkage;
|
|
|
|
case 6: return GlobalValue::DLLExportLinkage;
|
|
|
|
case 7: return GlobalValue::ExternalWeakLinkage;
|
|
|
|
case 8: return GlobalValue::CommonLinkage;
|
|
|
|
case 9: return GlobalValue::PrivateLinkage;
|
Introduce new linkage types linkonce_odr, weak_odr, common_odr
and extern_weak_odr. These are the same as the non-odr versions,
except that they indicate that the global will only be overridden
by an *equivalent* global. In C, a function with weak linkage can
be overridden by a function which behaves completely differently.
This means that IP passes have to skip weak functions, since any
deductions made from the function definition might be wrong, since
the definition could be replaced by something completely different
at link time. This is not allowed in C++, thanks to the ODR
(One-Definition-Rule): if a function is replaced by another at
link-time, then the new function must be the same as the original
function. If a language knows that a function or other global can
only be overridden by an equivalent global, it can give it the
weak_odr linkage type, and the optimizers will understand that it
is alright to make deductions based on the function body. The
code generators on the other hand map weak and weak_odr linkage
to the same thing.
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@66339 91177308-0d34-0410-b5e6-96231b3b80d8
2009-03-07 15:45:40 +00:00
|
|
|
case 10: return GlobalValue::WeakODRLinkage;
|
|
|
|
case 11: return GlobalValue::LinkOnceODRLinkage;
|
2009-04-13 05:44:34 +00:00
|
|
|
case 12: return GlobalValue::AvailableExternallyLinkage;
|
2009-07-20 01:03:30 +00:00
|
|
|
case 13: return GlobalValue::LinkerPrivateLinkage;
|
2007-04-22 06:23:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static GlobalValue::VisibilityTypes GetDecodedVisibility(unsigned Val) {
|
|
|
|
switch (Val) {
|
|
|
|
default: // Map unknown visibilities to default.
|
|
|
|
case 0: return GlobalValue::DefaultVisibility;
|
|
|
|
case 1: return GlobalValue::HiddenVisibility;
|
2007-04-29 20:56:48 +00:00
|
|
|
case 2: return GlobalValue::ProtectedVisibility;
|
2007-04-22 06:23:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-04-24 07:07:11 +00:00
|
|
|
static int GetDecodedCastOpcode(unsigned Val) {
|
|
|
|
switch (Val) {
|
|
|
|
default: return -1;
|
|
|
|
case bitc::CAST_TRUNC : return Instruction::Trunc;
|
|
|
|
case bitc::CAST_ZEXT : return Instruction::ZExt;
|
|
|
|
case bitc::CAST_SEXT : return Instruction::SExt;
|
|
|
|
case bitc::CAST_FPTOUI : return Instruction::FPToUI;
|
|
|
|
case bitc::CAST_FPTOSI : return Instruction::FPToSI;
|
|
|
|
case bitc::CAST_UITOFP : return Instruction::UIToFP;
|
|
|
|
case bitc::CAST_SITOFP : return Instruction::SIToFP;
|
|
|
|
case bitc::CAST_FPTRUNC : return Instruction::FPTrunc;
|
|
|
|
case bitc::CAST_FPEXT : return Instruction::FPExt;
|
|
|
|
case bitc::CAST_PTRTOINT: return Instruction::PtrToInt;
|
|
|
|
case bitc::CAST_INTTOPTR: return Instruction::IntToPtr;
|
|
|
|
case bitc::CAST_BITCAST : return Instruction::BitCast;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static int GetDecodedBinaryOpcode(unsigned Val, const Type *Ty) {
|
|
|
|
switch (Val) {
|
|
|
|
default: return -1;
|
2009-06-04 22:49:04 +00:00
|
|
|
case bitc::BINOP_ADD:
|
|
|
|
return Ty->isFPOrFPVector() ? Instruction::FAdd : Instruction::Add;
|
|
|
|
case bitc::BINOP_SUB:
|
|
|
|
return Ty->isFPOrFPVector() ? Instruction::FSub : Instruction::Sub;
|
|
|
|
case bitc::BINOP_MUL:
|
|
|
|
return Ty->isFPOrFPVector() ? Instruction::FMul : Instruction::Mul;
|
2007-04-24 07:07:11 +00:00
|
|
|
case bitc::BINOP_UDIV: return Instruction::UDiv;
|
|
|
|
case bitc::BINOP_SDIV:
|
|
|
|
return Ty->isFPOrFPVector() ? Instruction::FDiv : Instruction::SDiv;
|
|
|
|
case bitc::BINOP_UREM: return Instruction::URem;
|
|
|
|
case bitc::BINOP_SREM:
|
|
|
|
return Ty->isFPOrFPVector() ? Instruction::FRem : Instruction::SRem;
|
|
|
|
case bitc::BINOP_SHL: return Instruction::Shl;
|
|
|
|
case bitc::BINOP_LSHR: return Instruction::LShr;
|
|
|
|
case bitc::BINOP_ASHR: return Instruction::AShr;
|
|
|
|
case bitc::BINOP_AND: return Instruction::And;
|
|
|
|
case bitc::BINOP_OR: return Instruction::Or;
|
|
|
|
case bitc::BINOP_XOR: return Instruction::Xor;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-10 08:32:32 +00:00
|
|
|
namespace llvm {
|
2007-04-24 05:48:56 +00:00
|
|
|
namespace {
|
|
|
|
/// @brief A class for maintaining the slot number definition
|
|
|
|
/// as a placeholder for the actual definition for forward constants defs.
|
|
|
|
class ConstantPlaceHolder : public ConstantExpr {
|
|
|
|
ConstantPlaceHolder(); // DO NOT IMPLEMENT
|
|
|
|
void operator=(const ConstantPlaceHolder &); // DO NOT IMPLEMENT
|
2008-04-06 20:25:17 +00:00
|
|
|
public:
|
|
|
|
// allocate space for exactly one operand
|
|
|
|
void *operator new(size_t s) {
|
|
|
|
return User::operator new(s, 1);
|
|
|
|
}
|
2009-07-07 20:18:58 +00:00
|
|
|
explicit ConstantPlaceHolder(const Type *Ty, LLVMContext& Context)
|
2008-05-10 08:32:32 +00:00
|
|
|
: ConstantExpr(Ty, Instruction::UserOp1, &Op<0>(), 1) {
|
2009-07-07 20:18:58 +00:00
|
|
|
Op<0>() = Context.getUndef(Type::Int32Ty);
|
2007-04-24 05:48:56 +00:00
|
|
|
}
|
2008-08-21 02:34:16 +00:00
|
|
|
|
|
|
|
/// @brief Methods to support type inquiry through isa, cast, and dyn_cast.
|
|
|
|
static inline bool classof(const ConstantPlaceHolder *) { return true; }
|
|
|
|
static bool classof(const Value *V) {
|
|
|
|
return isa<ConstantExpr>(V) &&
|
|
|
|
cast<ConstantExpr>(V)->getOpcode() == Instruction::UserOp1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-05-10 08:32:32 +00:00
|
|
|
/// Provide fast operand accessors
|
2009-03-31 22:55:09 +00:00
|
|
|
//DECLARE_TRANSPARENT_OPERAND_ACCESSORS(Value);
|
2007-04-24 05:48:56 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2009-03-31 22:55:09 +00:00
|
|
|
// FIXME: can we inherit this from ConstantExpr?
|
2008-05-10 08:32:32 +00:00
|
|
|
template <>
|
|
|
|
struct OperandTraits<ConstantPlaceHolder> : FixedNumOperandTraits<1> {
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2009-03-31 22:55:09 +00:00
|
|
|
|
|
|
|
void BitcodeReaderValueList::AssignValue(Value *V, unsigned Idx) {
|
|
|
|
if (Idx == size()) {
|
|
|
|
push_back(V);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Idx >= size())
|
|
|
|
resize(Idx+1);
|
|
|
|
|
|
|
|
WeakVH &OldV = ValuePtrs[Idx];
|
|
|
|
if (OldV == 0) {
|
|
|
|
OldV = V;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle constants and non-constants (e.g. instrs) differently for
|
|
|
|
// efficiency.
|
|
|
|
if (Constant *PHC = dyn_cast<Constant>(&*OldV)) {
|
|
|
|
ResolveConstants.push_back(std::make_pair(PHC, Idx));
|
|
|
|
OldV = V;
|
|
|
|
} else {
|
|
|
|
// If there was a forward reference to this value, replace it.
|
|
|
|
Value *PrevVal = OldV;
|
|
|
|
OldV->replaceAllUsesWith(V);
|
|
|
|
delete PrevVal;
|
2008-05-10 08:32:32 +00:00
|
|
|
}
|
|
|
|
}
|
2009-03-31 22:55:09 +00:00
|
|
|
|
2008-05-10 08:32:32 +00:00
|
|
|
|
2007-04-24 05:48:56 +00:00
|
|
|
Constant *BitcodeReaderValueList::getConstantFwdRef(unsigned Idx,
|
|
|
|
const Type *Ty) {
|
2009-03-31 22:55:09 +00:00
|
|
|
if (Idx >= size())
|
2008-05-10 08:32:32 +00:00
|
|
|
resize(Idx + 1);
|
2007-04-24 05:48:56 +00:00
|
|
|
|
2009-03-31 22:55:09 +00:00
|
|
|
if (Value *V = ValuePtrs[Idx]) {
|
2007-05-01 07:01:57 +00:00
|
|
|
assert(Ty == V->getType() && "Type mismatch in constant table!");
|
|
|
|
return cast<Constant>(V);
|
2007-04-24 07:07:11 +00:00
|
|
|
}
|
2007-04-24 05:48:56 +00:00
|
|
|
|
|
|
|
// Create and return a placeholder, which will later be RAUW'd.
|
2009-07-07 20:18:58 +00:00
|
|
|
Constant *C = new ConstantPlaceHolder(Ty, Context);
|
2009-03-31 22:55:09 +00:00
|
|
|
ValuePtrs[Idx] = C;
|
2007-04-24 05:48:56 +00:00
|
|
|
return C;
|
|
|
|
}
|
|
|
|
|
2007-05-01 07:01:57 +00:00
|
|
|
Value *BitcodeReaderValueList::getValueFwdRef(unsigned Idx, const Type *Ty) {
|
2009-03-31 22:55:09 +00:00
|
|
|
if (Idx >= size())
|
2008-05-10 08:32:32 +00:00
|
|
|
resize(Idx + 1);
|
2007-05-01 07:01:57 +00:00
|
|
|
|
2009-03-31 22:55:09 +00:00
|
|
|
if (Value *V = ValuePtrs[Idx]) {
|
2007-05-01 07:01:57 +00:00
|
|
|
assert((Ty == 0 || Ty == V->getType()) && "Type mismatch in value table!");
|
|
|
|
return V;
|
|
|
|
}
|
|
|
|
|
2007-05-02 05:16:49 +00:00
|
|
|
// No type specified, must be invalid reference.
|
|
|
|
if (Ty == 0) return 0;
|
|
|
|
|
2007-05-01 07:01:57 +00:00
|
|
|
// Create and return a placeholder, which will later be RAUW'd.
|
|
|
|
Value *V = new Argument(Ty);
|
2009-03-31 22:55:09 +00:00
|
|
|
ValuePtrs[Idx] = V;
|
2007-05-01 07:01:57 +00:00
|
|
|
return V;
|
|
|
|
}
|
|
|
|
|
2008-08-21 02:34:16 +00:00
|
|
|
/// ResolveConstantForwardRefs - Once all constants are read, this method bulk
|
|
|
|
/// resolves any forward references. The idea behind this is that we sometimes
|
|
|
|
/// get constants (such as large arrays) which reference *many* forward ref
|
|
|
|
/// constants. Replacing each of these causes a lot of thrashing when
|
|
|
|
/// building/reuniquing the constant. Instead of doing this, we look at all the
|
|
|
|
/// uses and rewrite all the place holders at once for any constant that uses
|
|
|
|
/// a placeholder.
|
|
|
|
void BitcodeReaderValueList::ResolveConstantForwardRefs() {
|
|
|
|
// Sort the values by-pointer so that they are efficient to look up with a
|
|
|
|
// binary search.
|
|
|
|
std::sort(ResolveConstants.begin(), ResolveConstants.end());
|
|
|
|
|
|
|
|
SmallVector<Constant*, 64> NewOps;
|
|
|
|
|
|
|
|
while (!ResolveConstants.empty()) {
|
2009-03-31 22:55:09 +00:00
|
|
|
Value *RealVal = operator[](ResolveConstants.back().second);
|
2008-08-21 02:34:16 +00:00
|
|
|
Constant *Placeholder = ResolveConstants.back().first;
|
|
|
|
ResolveConstants.pop_back();
|
|
|
|
|
|
|
|
// Loop over all users of the placeholder, updating them to reference the
|
|
|
|
// new value. If they reference more than one placeholder, update them all
|
|
|
|
// at once.
|
|
|
|
while (!Placeholder->use_empty()) {
|
2008-08-21 17:31:45 +00:00
|
|
|
Value::use_iterator UI = Placeholder->use_begin();
|
|
|
|
|
2008-08-21 02:34:16 +00:00
|
|
|
// If the using object isn't uniqued, just update the operands. This
|
|
|
|
// handles instructions and initializers for global variables.
|
2008-08-21 17:31:45 +00:00
|
|
|
if (!isa<Constant>(*UI) || isa<GlobalValue>(*UI)) {
|
|
|
|
UI.getUse().set(RealVal);
|
2008-08-21 02:34:16 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, we have a constant that uses the placeholder. Replace that
|
|
|
|
// constant with a new constant that has *all* placeholder uses updated.
|
2008-08-21 17:31:45 +00:00
|
|
|
Constant *UserC = cast<Constant>(*UI);
|
2008-08-21 02:34:16 +00:00
|
|
|
for (User::op_iterator I = UserC->op_begin(), E = UserC->op_end();
|
|
|
|
I != E; ++I) {
|
|
|
|
Value *NewOp;
|
|
|
|
if (!isa<ConstantPlaceHolder>(*I)) {
|
|
|
|
// Not a placeholder reference.
|
|
|
|
NewOp = *I;
|
|
|
|
} else if (*I == Placeholder) {
|
|
|
|
// Common case is that it just references this one placeholder.
|
|
|
|
NewOp = RealVal;
|
|
|
|
} else {
|
|
|
|
// Otherwise, look up the placeholder in ResolveConstants.
|
|
|
|
ResolveConstantsTy::iterator It =
|
|
|
|
std::lower_bound(ResolveConstants.begin(), ResolveConstants.end(),
|
|
|
|
std::pair<Constant*, unsigned>(cast<Constant>(*I),
|
|
|
|
0));
|
|
|
|
assert(It != ResolveConstants.end() && It->first == *I);
|
2009-03-31 22:55:09 +00:00
|
|
|
NewOp = operator[](It->second);
|
2008-08-21 02:34:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
NewOps.push_back(cast<Constant>(NewOp));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make the new constant.
|
|
|
|
Constant *NewC;
|
|
|
|
if (ConstantArray *UserCA = dyn_cast<ConstantArray>(UserC)) {
|
2009-07-07 20:18:58 +00:00
|
|
|
NewC = Context.getConstantArray(UserCA->getType(), &NewOps[0],
|
|
|
|
NewOps.size());
|
2008-08-21 02:34:16 +00:00
|
|
|
} else if (ConstantStruct *UserCS = dyn_cast<ConstantStruct>(UserC)) {
|
2009-07-07 20:18:58 +00:00
|
|
|
NewC = Context.getConstantStruct(&NewOps[0], NewOps.size(),
|
|
|
|
UserCS->getType()->isPacked());
|
2008-08-21 02:34:16 +00:00
|
|
|
} else if (isa<ConstantVector>(UserC)) {
|
2009-07-07 20:18:58 +00:00
|
|
|
NewC = Context.getConstantVector(&NewOps[0], NewOps.size());
|
2009-05-10 20:57:05 +00:00
|
|
|
} else {
|
|
|
|
assert(isa<ConstantExpr>(UserC) && "Must be a ConstantExpr.");
|
2008-08-21 02:34:16 +00:00
|
|
|
NewC = cast<ConstantExpr>(UserC)->getWithOperands(&NewOps[0],
|
|
|
|
NewOps.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
UserC->replaceAllUsesWith(NewC);
|
|
|
|
UserC->destroyConstant();
|
|
|
|
NewOps.clear();
|
|
|
|
}
|
|
|
|
|
2009-05-10 20:57:05 +00:00
|
|
|
// Update all ValueHandles, they should be the only users at this point.
|
|
|
|
Placeholder->replaceAllUsesWith(RealVal);
|
2008-08-21 02:34:16 +00:00
|
|
|
delete Placeholder;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-04-22 06:23:29 +00:00
|
|
|
|
|
|
|
const Type *BitcodeReader::getTypeByID(unsigned ID, bool isTypeTable) {
|
|
|
|
// If the TypeID is in range, return it.
|
|
|
|
if (ID < TypeList.size())
|
|
|
|
return TypeList[ID].get();
|
|
|
|
if (!isTypeTable) return 0;
|
|
|
|
|
|
|
|
// The type table allows forward references. Push as many Opaque types as
|
|
|
|
// needed to get up to ID.
|
|
|
|
while (TypeList.size() <= ID)
|
2009-07-07 20:18:58 +00:00
|
|
|
TypeList.push_back(Context.getOpaqueType());
|
2007-04-22 06:23:29 +00:00
|
|
|
return TypeList.back().get();
|
|
|
|
}
|
|
|
|
|
2007-05-04 03:30:17 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Functions for parsing blocks from the bitcode file
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2008-09-25 21:00:45 +00:00
|
|
|
bool BitcodeReader::ParseAttributeBlock() {
|
2007-05-05 00:17:00 +00:00
|
|
|
if (Stream.EnterSubBlock(bitc::PARAMATTR_BLOCK_ID))
|
2007-05-04 03:30:17 +00:00
|
|
|
return Error("Malformed block record");
|
|
|
|
|
2008-09-26 22:53:05 +00:00
|
|
|
if (!MAttributes.empty())
|
2007-05-04 03:30:17 +00:00
|
|
|
return Error("Multiple PARAMATTR blocks found!");
|
|
|
|
|
|
|
|
SmallVector<uint64_t, 64> Record;
|
|
|
|
|
2008-09-25 21:00:45 +00:00
|
|
|
SmallVector<AttributeWithIndex, 8> Attrs;
|
2007-05-04 03:30:17 +00:00
|
|
|
|
|
|
|
// Read all the records.
|
|
|
|
while (1) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
|
|
|
if (Code == bitc::END_BLOCK) {
|
|
|
|
if (Stream.ReadBlockEnd())
|
|
|
|
return Error("Error at end of PARAMATTR block");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == bitc::ENTER_SUBBLOCK) {
|
|
|
|
// No known subblocks, always skip them.
|
|
|
|
Stream.ReadSubBlockID();
|
|
|
|
if (Stream.SkipBlock())
|
|
|
|
return Error("Malformed block record");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == bitc::DEFINE_ABBREV) {
|
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read a record.
|
|
|
|
Record.clear();
|
|
|
|
switch (Stream.ReadRecord(Code, Record)) {
|
|
|
|
default: // Default behavior: ignore.
|
|
|
|
break;
|
|
|
|
case bitc::PARAMATTR_CODE_ENTRY: { // ENTRY: [paramidx0, attr0, ...]
|
|
|
|
if (Record.size() & 1)
|
|
|
|
return Error("Invalid ENTRY record");
|
|
|
|
|
2008-10-05 18:22:09 +00:00
|
|
|
// FIXME : Remove this autoupgrade code in LLVM 3.0.
|
2008-09-26 22:53:05 +00:00
|
|
|
// If Function attributes are using index 0 then transfer them
|
2008-10-05 18:22:09 +00:00
|
|
|
// to index ~0. Index 0 is used for return value attributes but used to be
|
|
|
|
// used for function attributes.
|
2008-09-26 22:53:05 +00:00
|
|
|
Attributes RetAttribute = Attribute::None;
|
|
|
|
Attributes FnAttribute = Attribute::None;
|
2007-05-04 03:30:17 +00:00
|
|
|
for (unsigned i = 0, e = Record.size(); i != e; i += 2) {
|
2008-12-19 09:38:31 +00:00
|
|
|
// FIXME: remove in LLVM 3.0
|
|
|
|
// The alignment is stored as a 16-bit raw value from bits 31--16.
|
|
|
|
// We shift the bits above 31 down by 11 bits.
|
|
|
|
|
|
|
|
unsigned Alignment = (Record[i+1] & (0xffffull << 16)) >> 16;
|
|
|
|
if (Alignment && !isPowerOf2_32(Alignment))
|
|
|
|
return Error("Alignment is not a power of two.");
|
|
|
|
|
|
|
|
Attributes ReconstitutedAttr = Record[i+1] & 0xffff;
|
|
|
|
if (Alignment)
|
|
|
|
ReconstitutedAttr |= Attribute::constructAlignmentFromInt(Alignment);
|
|
|
|
ReconstitutedAttr |= (Record[i+1] & (0xffffull << 32)) >> 11;
|
|
|
|
Record[i+1] = ReconstitutedAttr;
|
|
|
|
|
2008-09-26 22:53:05 +00:00
|
|
|
if (Record[i] == 0)
|
|
|
|
RetAttribute = Record[i+1];
|
|
|
|
else if (Record[i] == ~0U)
|
|
|
|
FnAttribute = Record[i+1];
|
|
|
|
}
|
2008-10-05 18:22:09 +00:00
|
|
|
|
|
|
|
unsigned OldRetAttrs = (Attribute::NoUnwind|Attribute::NoReturn|
|
|
|
|
Attribute::ReadOnly|Attribute::ReadNone);
|
|
|
|
|
|
|
|
if (FnAttribute == Attribute::None && RetAttribute != Attribute::None &&
|
|
|
|
(RetAttribute & OldRetAttrs) != 0) {
|
|
|
|
if (FnAttribute == Attribute::None) { // add a slot so they get added.
|
|
|
|
Record.push_back(~0U);
|
|
|
|
Record.push_back(0);
|
2008-09-26 22:53:05 +00:00
|
|
|
}
|
2008-10-05 18:22:09 +00:00
|
|
|
|
|
|
|
FnAttribute |= RetAttribute & OldRetAttrs;
|
|
|
|
RetAttribute &= ~OldRetAttrs;
|
2008-09-26 22:53:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned i = 0, e = Record.size(); i != e; i += 2) {
|
2008-10-05 18:22:09 +00:00
|
|
|
if (Record[i] == 0) {
|
|
|
|
if (RetAttribute != Attribute::None)
|
|
|
|
Attrs.push_back(AttributeWithIndex::get(0, RetAttribute));
|
|
|
|
} else if (Record[i] == ~0U) {
|
|
|
|
if (FnAttribute != Attribute::None)
|
|
|
|
Attrs.push_back(AttributeWithIndex::get(~0U, FnAttribute));
|
|
|
|
} else if (Record[i+1] != Attribute::None)
|
2008-09-25 21:00:45 +00:00
|
|
|
Attrs.push_back(AttributeWithIndex::get(Record[i], Record[i+1]));
|
2007-05-04 03:30:17 +00:00
|
|
|
}
|
2008-03-12 02:25:52 +00:00
|
|
|
|
2008-09-26 22:53:05 +00:00
|
|
|
MAttributes.push_back(AttrListPtr::get(Attrs.begin(), Attrs.end()));
|
2007-05-04 03:30:17 +00:00
|
|
|
Attrs.clear();
|
|
|
|
break;
|
|
|
|
}
|
2007-11-20 14:09:29 +00:00
|
|
|
}
|
2007-05-04 03:30:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-05-01 05:01:34 +00:00
|
|
|
bool BitcodeReader::ParseTypeTable() {
|
2007-05-05 00:17:00 +00:00
|
|
|
if (Stream.EnterSubBlock(bitc::TYPE_BLOCK_ID))
|
2007-04-22 06:23:29 +00:00
|
|
|
return Error("Malformed block record");
|
|
|
|
|
|
|
|
if (!TypeList.empty())
|
|
|
|
return Error("Multiple TYPE_BLOCKs found!");
|
|
|
|
|
|
|
|
SmallVector<uint64_t, 64> Record;
|
|
|
|
unsigned NumRecords = 0;
|
|
|
|
|
|
|
|
// Read all the records for this type table.
|
|
|
|
while (1) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
|
|
|
if (Code == bitc::END_BLOCK) {
|
|
|
|
if (NumRecords != TypeList.size())
|
|
|
|
return Error("Invalid type forward reference in TYPE_BLOCK");
|
2007-04-24 18:15:21 +00:00
|
|
|
if (Stream.ReadBlockEnd())
|
|
|
|
return Error("Error at end of type table block");
|
|
|
|
return false;
|
2007-04-22 06:23:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == bitc::ENTER_SUBBLOCK) {
|
|
|
|
// No known subblocks, always skip them.
|
|
|
|
Stream.ReadSubBlockID();
|
|
|
|
if (Stream.SkipBlock())
|
|
|
|
return Error("Malformed block record");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2007-04-23 16:04:05 +00:00
|
|
|
if (Code == bitc::DEFINE_ABBREV) {
|
2007-04-23 18:58:34 +00:00
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
2007-04-22 06:23:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Read a record.
|
|
|
|
Record.clear();
|
|
|
|
const Type *ResultTy = 0;
|
|
|
|
switch (Stream.ReadRecord(Code, Record)) {
|
|
|
|
default: // Default behavior: unknown type.
|
|
|
|
ResultTy = 0;
|
|
|
|
break;
|
|
|
|
case bitc::TYPE_CODE_NUMENTRY: // TYPE_CODE_NUMENTRY: [numentries]
|
|
|
|
// TYPE_CODE_NUMENTRY contains a count of the number of types in the
|
|
|
|
// type list. This allows us to reserve space.
|
|
|
|
if (Record.size() < 1)
|
|
|
|
return Error("Invalid TYPE_CODE_NUMENTRY record");
|
|
|
|
TypeList.reserve(Record[0]);
|
|
|
|
continue;
|
|
|
|
case bitc::TYPE_CODE_VOID: // VOID
|
|
|
|
ResultTy = Type::VoidTy;
|
|
|
|
break;
|
|
|
|
case bitc::TYPE_CODE_FLOAT: // FLOAT
|
|
|
|
ResultTy = Type::FloatTy;
|
|
|
|
break;
|
|
|
|
case bitc::TYPE_CODE_DOUBLE: // DOUBLE
|
|
|
|
ResultTy = Type::DoubleTy;
|
|
|
|
break;
|
2007-08-03 01:03:46 +00:00
|
|
|
case bitc::TYPE_CODE_X86_FP80: // X86_FP80
|
|
|
|
ResultTy = Type::X86_FP80Ty;
|
|
|
|
break;
|
|
|
|
case bitc::TYPE_CODE_FP128: // FP128
|
|
|
|
ResultTy = Type::FP128Ty;
|
|
|
|
break;
|
|
|
|
case bitc::TYPE_CODE_PPC_FP128: // PPC_FP128
|
|
|
|
ResultTy = Type::PPC_FP128Ty;
|
|
|
|
break;
|
2007-04-22 06:23:29 +00:00
|
|
|
case bitc::TYPE_CODE_LABEL: // LABEL
|
|
|
|
ResultTy = Type::LabelTy;
|
|
|
|
break;
|
|
|
|
case bitc::TYPE_CODE_OPAQUE: // OPAQUE
|
|
|
|
ResultTy = 0;
|
|
|
|
break;
|
2009-05-30 05:06:04 +00:00
|
|
|
case bitc::TYPE_CODE_METADATA: // METADATA
|
|
|
|
ResultTy = Type::MetadataTy;
|
|
|
|
break;
|
2007-04-22 06:23:29 +00:00
|
|
|
case bitc::TYPE_CODE_INTEGER: // INTEGER: [width]
|
|
|
|
if (Record.size() < 1)
|
|
|
|
return Error("Invalid Integer type record");
|
|
|
|
|
2009-07-07 20:18:58 +00:00
|
|
|
ResultTy = Context.getIntegerType(Record[0]);
|
2007-04-22 06:23:29 +00:00
|
|
|
break;
|
2007-12-11 08:59:05 +00:00
|
|
|
case bitc::TYPE_CODE_POINTER: { // POINTER: [pointee type] or
|
|
|
|
// [pointee type, address space]
|
2007-04-22 06:23:29 +00:00
|
|
|
if (Record.size() < 1)
|
|
|
|
return Error("Invalid POINTER type record");
|
2007-12-11 08:59:05 +00:00
|
|
|
unsigned AddressSpace = 0;
|
|
|
|
if (Record.size() == 2)
|
|
|
|
AddressSpace = Record[1];
|
2009-07-07 20:18:58 +00:00
|
|
|
ResultTy = Context.getPointerType(getTypeByID(Record[0], true),
|
|
|
|
AddressSpace);
|
2007-04-22 06:23:29 +00:00
|
|
|
break;
|
2007-12-11 08:59:05 +00:00
|
|
|
}
|
2007-04-22 06:23:29 +00:00
|
|
|
case bitc::TYPE_CODE_FUNCTION: {
|
2007-11-27 17:48:06 +00:00
|
|
|
// FIXME: attrid is dead, remove it in LLVM 3.0
|
|
|
|
// FUNCTION: [vararg, attrid, retty, paramty x N]
|
|
|
|
if (Record.size() < 3)
|
2007-04-22 06:23:29 +00:00
|
|
|
return Error("Invalid FUNCTION type record");
|
|
|
|
std::vector<const Type*> ArgTys;
|
2007-11-27 17:48:06 +00:00
|
|
|
for (unsigned i = 3, e = Record.size(); i != e; ++i)
|
2007-05-04 19:11:41 +00:00
|
|
|
ArgTys.push_back(getTypeByID(Record[i], true));
|
2007-04-22 06:23:29 +00:00
|
|
|
|
2009-07-07 20:18:58 +00:00
|
|
|
ResultTy = Context.getFunctionType(getTypeByID(Record[2], true), ArgTys,
|
2007-11-27 13:23:08 +00:00
|
|
|
Record[0]);
|
2007-04-22 06:23:29 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::TYPE_CODE_STRUCT: { // STRUCT: [ispacked, eltty x N]
|
2007-05-06 08:21:50 +00:00
|
|
|
if (Record.size() < 1)
|
2007-04-22 06:23:29 +00:00
|
|
|
return Error("Invalid STRUCT type record");
|
|
|
|
std::vector<const Type*> EltTys;
|
2007-05-04 19:11:41 +00:00
|
|
|
for (unsigned i = 1, e = Record.size(); i != e; ++i)
|
|
|
|
EltTys.push_back(getTypeByID(Record[i], true));
|
2009-07-07 20:18:58 +00:00
|
|
|
ResultTy = Context.getStructType(EltTys, Record[0]);
|
2007-04-22 06:23:29 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case bitc::TYPE_CODE_ARRAY: // ARRAY: [numelts, eltty]
|
|
|
|
if (Record.size() < 2)
|
|
|
|
return Error("Invalid ARRAY type record");
|
2009-07-07 20:18:58 +00:00
|
|
|
ResultTy = Context.getArrayType(getTypeByID(Record[1], true), Record[0]);
|
2007-04-22 06:23:29 +00:00
|
|
|
break;
|
|
|
|
case bitc::TYPE_CODE_VECTOR: // VECTOR: [numelts, eltty]
|
|
|
|
if (Record.size() < 2)
|
|
|
|
return Error("Invalid VECTOR type record");
|
2009-07-07 20:18:58 +00:00
|
|
|
ResultTy = Context.getVectorType(getTypeByID(Record[1], true), Record[0]);
|
2007-04-22 06:23:29 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (NumRecords == TypeList.size()) {
|
|
|
|
// If this is a new type slot, just append it.
|
2009-07-07 20:18:58 +00:00
|
|
|
TypeList.push_back(ResultTy ? ResultTy : Context.getOpaqueType());
|
2007-04-22 06:23:29 +00:00
|
|
|
++NumRecords;
|
|
|
|
} else if (ResultTy == 0) {
|
|
|
|
// Otherwise, this was forward referenced, so an opaque type was created,
|
|
|
|
// but the result type is actually just an opaque. Leave the one we
|
|
|
|
// created previously.
|
|
|
|
++NumRecords;
|
|
|
|
} else {
|
|
|
|
// Otherwise, this was forward referenced, so an opaque type was created.
|
|
|
|
// Resolve the opaque type to the real type now.
|
|
|
|
assert(NumRecords < TypeList.size() && "Typelist imbalance");
|
|
|
|
const OpaqueType *OldTy = cast<OpaqueType>(TypeList[NumRecords++].get());
|
|
|
|
|
|
|
|
// Don't directly push the new type on the Tab. Instead we want to replace
|
|
|
|
// the opaque type we previously inserted with the new concrete value. The
|
|
|
|
// refinement from the abstract (opaque) type to the new type causes all
|
|
|
|
// uses of the abstract type to use the concrete type (NewTy). This will
|
|
|
|
// also cause the opaque type to be deleted.
|
|
|
|
const_cast<OpaqueType*>(OldTy)->refineAbstractTypeTo(ResultTy);
|
|
|
|
|
|
|
|
// This should have replaced the old opaque type with the new type in the
|
2007-04-24 04:04:35 +00:00
|
|
|
// value table... or with a preexisting type that was already in the
|
|
|
|
// system. Let's just make sure it did.
|
2007-04-22 06:23:29 +00:00
|
|
|
assert(TypeList[NumRecords-1].get() != OldTy &&
|
|
|
|
"refineAbstractType didn't work!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-05-01 05:01:34 +00:00
|
|
|
bool BitcodeReader::ParseTypeSymbolTable() {
|
2007-05-05 00:17:00 +00:00
|
|
|
if (Stream.EnterSubBlock(bitc::TYPE_SYMTAB_BLOCK_ID))
|
2007-04-22 06:23:29 +00:00
|
|
|
return Error("Malformed block record");
|
|
|
|
|
|
|
|
SmallVector<uint64_t, 64> Record;
|
|
|
|
|
|
|
|
// Read all the records for this type table.
|
|
|
|
std::string TypeName;
|
|
|
|
while (1) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
2007-04-24 18:15:21 +00:00
|
|
|
if (Code == bitc::END_BLOCK) {
|
|
|
|
if (Stream.ReadBlockEnd())
|
|
|
|
return Error("Error at end of type symbol table block");
|
|
|
|
return false;
|
|
|
|
}
|
2007-04-22 06:23:29 +00:00
|
|
|
|
|
|
|
if (Code == bitc::ENTER_SUBBLOCK) {
|
|
|
|
// No known subblocks, always skip them.
|
|
|
|
Stream.ReadSubBlockID();
|
|
|
|
if (Stream.SkipBlock())
|
|
|
|
return Error("Malformed block record");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2007-04-23 16:04:05 +00:00
|
|
|
if (Code == bitc::DEFINE_ABBREV) {
|
2007-04-23 18:58:34 +00:00
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
2007-04-22 06:23:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Read a record.
|
|
|
|
Record.clear();
|
|
|
|
switch (Stream.ReadRecord(Code, Record)) {
|
|
|
|
default: // Default behavior: unknown type.
|
|
|
|
break;
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::TST_CODE_ENTRY: // TST_ENTRY: [typeid, namechar x N]
|
2007-04-22 06:23:29 +00:00
|
|
|
if (ConvertToString(Record, 1, TypeName))
|
|
|
|
return Error("Invalid TST_ENTRY record");
|
|
|
|
unsigned TypeID = Record[0];
|
|
|
|
if (TypeID >= TypeList.size())
|
|
|
|
return Error("Invalid Type ID in TST_ENTRY record");
|
|
|
|
|
|
|
|
TheModule->addTypeName(TypeName, TypeList[TypeID].get());
|
|
|
|
TypeName.clear();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-05-01 05:01:34 +00:00
|
|
|
bool BitcodeReader::ParseValueSymbolTable() {
|
2007-05-05 00:17:00 +00:00
|
|
|
if (Stream.EnterSubBlock(bitc::VALUE_SYMTAB_BLOCK_ID))
|
2007-04-23 21:26:05 +00:00
|
|
|
return Error("Malformed block record");
|
|
|
|
|
|
|
|
SmallVector<uint64_t, 64> Record;
|
|
|
|
|
|
|
|
// Read all the records for this value table.
|
|
|
|
SmallString<128> ValueName;
|
|
|
|
while (1) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
2007-04-24 18:15:21 +00:00
|
|
|
if (Code == bitc::END_BLOCK) {
|
|
|
|
if (Stream.ReadBlockEnd())
|
|
|
|
return Error("Error at end of value symbol table block");
|
|
|
|
return false;
|
|
|
|
}
|
2007-04-23 21:26:05 +00:00
|
|
|
if (Code == bitc::ENTER_SUBBLOCK) {
|
|
|
|
// No known subblocks, always skip them.
|
|
|
|
Stream.ReadSubBlockID();
|
|
|
|
if (Stream.SkipBlock())
|
|
|
|
return Error("Malformed block record");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == bitc::DEFINE_ABBREV) {
|
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read a record.
|
|
|
|
Record.clear();
|
|
|
|
switch (Stream.ReadRecord(Code, Record)) {
|
|
|
|
default: // Default behavior: unknown type.
|
|
|
|
break;
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::VST_CODE_ENTRY: { // VST_ENTRY: [valueid, namechar x N]
|
2007-04-23 21:26:05 +00:00
|
|
|
if (ConvertToString(Record, 1, ValueName))
|
2009-05-31 06:07:28 +00:00
|
|
|
return Error("Invalid VST_ENTRY record");
|
2007-04-23 21:26:05 +00:00
|
|
|
unsigned ValueID = Record[0];
|
|
|
|
if (ValueID >= ValueList.size())
|
|
|
|
return Error("Invalid Value ID in VST_ENTRY record");
|
|
|
|
Value *V = ValueList[ValueID];
|
|
|
|
|
2009-07-26 00:34:27 +00:00
|
|
|
V->setName(StringRef(ValueName.data(), ValueName.size()));
|
2007-04-23 21:26:05 +00:00
|
|
|
ValueName.clear();
|
|
|
|
break;
|
2007-05-04 01:43:33 +00:00
|
|
|
}
|
|
|
|
case bitc::VST_CODE_BBENTRY: {
|
2007-05-03 22:18:21 +00:00
|
|
|
if (ConvertToString(Record, 1, ValueName))
|
|
|
|
return Error("Invalid VST_BBENTRY record");
|
|
|
|
BasicBlock *BB = getBasicBlock(Record[0]);
|
|
|
|
if (BB == 0)
|
|
|
|
return Error("Invalid BB ID in VST_BBENTRY record");
|
|
|
|
|
2009-07-26 00:34:27 +00:00
|
|
|
BB->setName(StringRef(ValueName.data(), ValueName.size()));
|
2007-05-03 22:18:21 +00:00
|
|
|
ValueName.clear();
|
|
|
|
break;
|
2007-04-23 21:26:05 +00:00
|
|
|
}
|
2007-05-04 01:43:33 +00:00
|
|
|
}
|
2007-04-23 21:26:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-22 17:43:22 +00:00
|
|
|
bool BitcodeReader::ParseMetadata() {
|
|
|
|
unsigned NextValueNo = ValueList.size();
|
|
|
|
|
|
|
|
if (Stream.EnterSubBlock(bitc::METADATA_BLOCK_ID))
|
|
|
|
return Error("Malformed block record");
|
|
|
|
|
|
|
|
SmallVector<uint64_t, 64> Record;
|
|
|
|
|
|
|
|
// Read all the records.
|
|
|
|
while (1) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
|
|
|
if (Code == bitc::END_BLOCK) {
|
|
|
|
if (Stream.ReadBlockEnd())
|
|
|
|
return Error("Error at end of PARAMATTR block");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == bitc::ENTER_SUBBLOCK) {
|
|
|
|
// No known subblocks, always skip them.
|
|
|
|
Stream.ReadSubBlockID();
|
|
|
|
if (Stream.SkipBlock())
|
|
|
|
return Error("Malformed block record");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == bitc::DEFINE_ABBREV) {
|
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read a record.
|
|
|
|
Record.clear();
|
|
|
|
switch (Stream.ReadRecord(Code, Record)) {
|
|
|
|
default: // Default behavior: ignore.
|
|
|
|
break;
|
2009-07-23 01:07:34 +00:00
|
|
|
case bitc::METADATA_NODE: {
|
|
|
|
if (Record.empty() || Record.size() % 2 == 1)
|
|
|
|
return Error("Invalid METADATA_NODE record");
|
|
|
|
|
|
|
|
unsigned Size = Record.size();
|
|
|
|
SmallVector<Value*, 8> Elts;
|
|
|
|
for (unsigned i = 0; i != Size; i += 2) {
|
|
|
|
const Type *Ty = getTypeByID(Record[i], false);
|
|
|
|
if (Ty != Type::VoidTy)
|
|
|
|
Elts.push_back(ValueList.getValueFwdRef(Record[i+1], Ty));
|
|
|
|
else
|
|
|
|
Elts.push_back(NULL);
|
|
|
|
}
|
|
|
|
Value *V = Context.getMDNode(&Elts[0], Elts.size());
|
|
|
|
ValueList.AssignValue(V, NextValueNo++);
|
|
|
|
break;
|
|
|
|
}
|
2009-07-22 17:43:22 +00:00
|
|
|
case bitc::METADATA_STRING: {
|
|
|
|
unsigned MDStringLength = Record.size();
|
|
|
|
SmallString<8> String;
|
|
|
|
String.resize(MDStringLength);
|
|
|
|
for (unsigned i = 0; i != MDStringLength; ++i)
|
|
|
|
String[i] = Record[i];
|
2009-07-25 06:02:13 +00:00
|
|
|
Value *V = Context.getMDString(StringRef(String.data(), String.size()));
|
2009-07-22 17:43:22 +00:00
|
|
|
ValueList.AssignValue(V, NextValueNo++);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-04-24 04:04:35 +00:00
|
|
|
/// DecodeSignRotatedValue - Decode a signed value stored with the sign bit in
|
|
|
|
/// the LSB for dense VBR encoding.
|
|
|
|
static uint64_t DecodeSignRotatedValue(uint64_t V) {
|
|
|
|
if ((V & 1) == 0)
|
|
|
|
return V >> 1;
|
|
|
|
if (V != 1)
|
|
|
|
return -(V >> 1);
|
|
|
|
// There is no such thing as -0 with integers. "-0" really means MININT.
|
|
|
|
return 1ULL << 63;
|
|
|
|
}
|
|
|
|
|
2007-04-26 02:46:40 +00:00
|
|
|
/// ResolveGlobalAndAliasInits - Resolve all of the initializers for global
|
|
|
|
/// values and aliases that we can.
|
|
|
|
bool BitcodeReader::ResolveGlobalAndAliasInits() {
|
|
|
|
std::vector<std::pair<GlobalVariable*, unsigned> > GlobalInitWorklist;
|
|
|
|
std::vector<std::pair<GlobalAlias*, unsigned> > AliasInitWorklist;
|
|
|
|
|
|
|
|
GlobalInitWorklist.swap(GlobalInits);
|
|
|
|
AliasInitWorklist.swap(AliasInits);
|
|
|
|
|
|
|
|
while (!GlobalInitWorklist.empty()) {
|
2007-04-26 03:27:58 +00:00
|
|
|
unsigned ValID = GlobalInitWorklist.back().second;
|
2007-04-26 02:46:40 +00:00
|
|
|
if (ValID >= ValueList.size()) {
|
|
|
|
// Not ready to resolve this yet, it requires something later in the file.
|
2007-04-26 03:27:58 +00:00
|
|
|
GlobalInits.push_back(GlobalInitWorklist.back());
|
2007-04-26 02:46:40 +00:00
|
|
|
} else {
|
|
|
|
if (Constant *C = dyn_cast<Constant>(ValueList[ValID]))
|
|
|
|
GlobalInitWorklist.back().first->setInitializer(C);
|
|
|
|
else
|
|
|
|
return Error("Global variable initializer is not a constant!");
|
|
|
|
}
|
|
|
|
GlobalInitWorklist.pop_back();
|
|
|
|
}
|
|
|
|
|
|
|
|
while (!AliasInitWorklist.empty()) {
|
|
|
|
unsigned ValID = AliasInitWorklist.back().second;
|
|
|
|
if (ValID >= ValueList.size()) {
|
|
|
|
AliasInits.push_back(AliasInitWorklist.back());
|
|
|
|
} else {
|
|
|
|
if (Constant *C = dyn_cast<Constant>(ValueList[ValID]))
|
2007-04-28 14:57:59 +00:00
|
|
|
AliasInitWorklist.back().first->setAliasee(C);
|
2007-04-26 02:46:40 +00:00
|
|
|
else
|
|
|
|
return Error("Alias initializer is not a constant!");
|
|
|
|
}
|
|
|
|
AliasInitWorklist.pop_back();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2009-07-20 21:19:07 +00:00
|
|
|
static void SetOptimizationFlags(Value *V, uint64_t Flags) {
|
|
|
|
if (OverflowingBinaryOperator *OBO =
|
|
|
|
dyn_cast<OverflowingBinaryOperator>(V)) {
|
|
|
|
if (Flags & (1 << bitc::OBO_NO_SIGNED_OVERFLOW))
|
|
|
|
OBO->setHasNoSignedOverflow(true);
|
|
|
|
if (Flags & (1 << bitc::OBO_NO_UNSIGNED_OVERFLOW))
|
|
|
|
OBO->setHasNoUnsignedOverflow(true);
|
|
|
|
} else if (SDivOperator *Div = dyn_cast<SDivOperator>(V)) {
|
|
|
|
if (Flags & (1 << bitc::SDIV_EXACT))
|
|
|
|
Div->setIsExact(true);
|
|
|
|
}
|
|
|
|
}
|
2007-04-26 02:46:40 +00:00
|
|
|
|
2007-05-01 05:01:34 +00:00
|
|
|
bool BitcodeReader::ParseConstants() {
|
2007-05-05 00:17:00 +00:00
|
|
|
if (Stream.EnterSubBlock(bitc::CONSTANTS_BLOCK_ID))
|
2007-04-24 03:30:34 +00:00
|
|
|
return Error("Malformed block record");
|
|
|
|
|
|
|
|
SmallVector<uint64_t, 64> Record;
|
|
|
|
|
|
|
|
// Read all the records for this value table.
|
|
|
|
const Type *CurTy = Type::Int32Ty;
|
2007-04-24 05:48:56 +00:00
|
|
|
unsigned NextCstNo = ValueList.size();
|
2007-04-24 03:30:34 +00:00
|
|
|
while (1) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
2008-08-21 02:34:16 +00:00
|
|
|
if (Code == bitc::END_BLOCK)
|
|
|
|
break;
|
2007-04-24 03:30:34 +00:00
|
|
|
|
|
|
|
if (Code == bitc::ENTER_SUBBLOCK) {
|
|
|
|
// No known subblocks, always skip them.
|
|
|
|
Stream.ReadSubBlockID();
|
|
|
|
if (Stream.SkipBlock())
|
|
|
|
return Error("Malformed block record");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == bitc::DEFINE_ABBREV) {
|
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read a record.
|
|
|
|
Record.clear();
|
|
|
|
Value *V = 0;
|
2009-07-20 21:19:07 +00:00
|
|
|
unsigned BitCode = Stream.ReadRecord(Code, Record);
|
|
|
|
switch (BitCode) {
|
2007-04-24 03:30:34 +00:00
|
|
|
default: // Default behavior: unknown constant
|
|
|
|
case bitc::CST_CODE_UNDEF: // UNDEF
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getUndef(CurTy);
|
2007-04-24 03:30:34 +00:00
|
|
|
break;
|
|
|
|
case bitc::CST_CODE_SETTYPE: // SETTYPE: [typeid]
|
|
|
|
if (Record.empty())
|
|
|
|
return Error("Malformed CST_SETTYPE record");
|
|
|
|
if (Record[0] >= TypeList.size())
|
|
|
|
return Error("Invalid Type ID in CST_SETTYPE record");
|
|
|
|
CurTy = TypeList[Record[0]];
|
2007-04-24 04:04:35 +00:00
|
|
|
continue; // Skip the ValueList manipulation.
|
2007-04-24 03:30:34 +00:00
|
|
|
case bitc::CST_CODE_NULL: // NULL
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getNullValue(CurTy);
|
2007-04-24 03:30:34 +00:00
|
|
|
break;
|
|
|
|
case bitc::CST_CODE_INTEGER: // INTEGER: [intval]
|
2007-04-24 04:04:35 +00:00
|
|
|
if (!isa<IntegerType>(CurTy) || Record.empty())
|
|
|
|
return Error("Invalid CST_INTEGER record");
|
2009-07-24 23:12:02 +00:00
|
|
|
V = ConstantInt::get(CurTy, DecodeSignRotatedValue(Record[0]));
|
2007-04-24 04:04:35 +00:00
|
|
|
break;
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::CST_CODE_WIDE_INTEGER: {// WIDE_INTEGER: [n x intval]
|
|
|
|
if (!isa<IntegerType>(CurTy) || Record.empty())
|
2007-04-24 04:04:35 +00:00
|
|
|
return Error("Invalid WIDE_INTEGER record");
|
|
|
|
|
2007-05-04 19:11:41 +00:00
|
|
|
unsigned NumWords = Record.size();
|
2007-04-24 17:22:05 +00:00
|
|
|
SmallVector<uint64_t, 8> Words;
|
|
|
|
Words.resize(NumWords);
|
2007-04-24 04:04:35 +00:00
|
|
|
for (unsigned i = 0; i != NumWords; ++i)
|
2007-05-04 19:11:41 +00:00
|
|
|
Words[i] = DecodeSignRotatedValue(Record[i]);
|
2009-07-24 23:12:02 +00:00
|
|
|
V = ConstantInt::get(Context,
|
|
|
|
APInt(cast<IntegerType>(CurTy)->getBitWidth(),
|
|
|
|
NumWords, &Words[0]));
|
2007-04-24 04:04:35 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-09-11 18:32:33 +00:00
|
|
|
case bitc::CST_CODE_FLOAT: { // FLOAT: [fpval]
|
2007-04-24 04:04:35 +00:00
|
|
|
if (Record.empty())
|
|
|
|
return Error("Invalid FLOAT record");
|
|
|
|
if (CurTy == Type::FloatTy)
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantFP(APFloat(APInt(32, (uint32_t)Record[0])));
|
2007-04-24 04:04:35 +00:00
|
|
|
else if (CurTy == Type::DoubleTy)
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantFP(APFloat(APInt(64, Record[0])));
|
2009-03-23 21:16:53 +00:00
|
|
|
else if (CurTy == Type::X86_FP80Ty) {
|
|
|
|
// Bits are not stored the same way as a normal i80 APInt, compensate.
|
|
|
|
uint64_t Rearrange[2];
|
|
|
|
Rearrange[0] = (Record[1] & 0xffffLL) | (Record[0] << 16);
|
|
|
|
Rearrange[1] = Record[0] >> 48;
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantFP(APFloat(APInt(80, 2, Rearrange)));
|
2009-03-23 21:16:53 +00:00
|
|
|
} else if (CurTy == Type::FP128Ty)
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantFP(APFloat(APInt(128, 2, &Record[0]), true));
|
2007-09-06 18:13:44 +00:00
|
|
|
else if (CurTy == Type::PPC_FP128Ty)
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantFP(APFloat(APInt(128, 2, &Record[0])));
|
2007-04-24 03:30:34 +00:00
|
|
|
else
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getUndef(CurTy);
|
2007-04-24 03:30:34 +00:00
|
|
|
break;
|
2007-09-11 18:32:33 +00:00
|
|
|
}
|
2007-04-24 05:48:56 +00:00
|
|
|
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::CST_CODE_AGGREGATE: {// AGGREGATE: [n x value number]
|
|
|
|
if (Record.empty())
|
2007-04-24 05:48:56 +00:00
|
|
|
return Error("Invalid CST_AGGREGATE record");
|
|
|
|
|
2007-05-04 19:11:41 +00:00
|
|
|
unsigned Size = Record.size();
|
2007-04-24 05:48:56 +00:00
|
|
|
std::vector<Constant*> Elts;
|
|
|
|
|
|
|
|
if (const StructType *STy = dyn_cast<StructType>(CurTy)) {
|
|
|
|
for (unsigned i = 0; i != Size; ++i)
|
2007-05-04 19:11:41 +00:00
|
|
|
Elts.push_back(ValueList.getConstantFwdRef(Record[i],
|
2007-04-24 05:48:56 +00:00
|
|
|
STy->getElementType(i)));
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantStruct(STy, Elts);
|
2007-04-24 05:48:56 +00:00
|
|
|
} else if (const ArrayType *ATy = dyn_cast<ArrayType>(CurTy)) {
|
|
|
|
const Type *EltTy = ATy->getElementType();
|
|
|
|
for (unsigned i = 0; i != Size; ++i)
|
2007-05-04 19:11:41 +00:00
|
|
|
Elts.push_back(ValueList.getConstantFwdRef(Record[i], EltTy));
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantArray(ATy, Elts);
|
2007-04-24 05:48:56 +00:00
|
|
|
} else if (const VectorType *VTy = dyn_cast<VectorType>(CurTy)) {
|
|
|
|
const Type *EltTy = VTy->getElementType();
|
|
|
|
for (unsigned i = 0; i != Size; ++i)
|
2007-05-04 19:11:41 +00:00
|
|
|
Elts.push_back(ValueList.getConstantFwdRef(Record[i], EltTy));
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantVector(Elts);
|
2007-04-24 05:48:56 +00:00
|
|
|
} else {
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getUndef(CurTy);
|
2007-04-24 05:48:56 +00:00
|
|
|
}
|
2007-04-24 07:07:11 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-05-06 00:35:24 +00:00
|
|
|
case bitc::CST_CODE_STRING: { // STRING: [values]
|
|
|
|
if (Record.empty())
|
|
|
|
return Error("Invalid CST_AGGREGATE record");
|
2007-04-24 07:07:11 +00:00
|
|
|
|
2007-05-06 00:35:24 +00:00
|
|
|
const ArrayType *ATy = cast<ArrayType>(CurTy);
|
|
|
|
const Type *EltTy = ATy->getElementType();
|
|
|
|
|
|
|
|
unsigned Size = Record.size();
|
|
|
|
std::vector<Constant*> Elts;
|
2007-05-06 00:53:07 +00:00
|
|
|
for (unsigned i = 0; i != Size; ++i)
|
2009-07-24 23:12:02 +00:00
|
|
|
Elts.push_back(ConstantInt::get(EltTy, Record[i]));
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantArray(ATy, Elts);
|
2007-05-06 00:53:07 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case bitc::CST_CODE_CSTRING: { // CSTRING: [values]
|
|
|
|
if (Record.empty())
|
|
|
|
return Error("Invalid CST_AGGREGATE record");
|
|
|
|
|
|
|
|
const ArrayType *ATy = cast<ArrayType>(CurTy);
|
|
|
|
const Type *EltTy = ATy->getElementType();
|
2007-05-06 00:35:24 +00:00
|
|
|
|
2007-05-06 00:53:07 +00:00
|
|
|
unsigned Size = Record.size();
|
|
|
|
std::vector<Constant*> Elts;
|
2007-05-06 00:35:24 +00:00
|
|
|
for (unsigned i = 0; i != Size; ++i)
|
2009-07-24 23:12:02 +00:00
|
|
|
Elts.push_back(ConstantInt::get(EltTy, Record[i]));
|
2009-07-07 20:18:58 +00:00
|
|
|
Elts.push_back(Context.getNullValue(EltTy));
|
|
|
|
V = Context.getConstantArray(ATy, Elts);
|
2007-05-06 00:35:24 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-04-24 07:07:11 +00:00
|
|
|
case bitc::CST_CODE_CE_BINOP: { // CE_BINOP: [opcode, opval, opval]
|
|
|
|
if (Record.size() < 3) return Error("Invalid CE_BINOP record");
|
|
|
|
int Opc = GetDecodedBinaryOpcode(Record[0], CurTy);
|
2007-04-24 18:15:21 +00:00
|
|
|
if (Opc < 0) {
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getUndef(CurTy); // Unknown binop.
|
2007-04-24 18:15:21 +00:00
|
|
|
} else {
|
|
|
|
Constant *LHS = ValueList.getConstantFwdRef(Record[1], CurTy);
|
|
|
|
Constant *RHS = ValueList.getConstantFwdRef(Record[2], CurTy);
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantExpr(Opc, LHS, RHS);
|
2007-04-24 18:15:21 +00:00
|
|
|
}
|
2009-07-20 21:19:07 +00:00
|
|
|
if (Record.size() >= 4)
|
|
|
|
SetOptimizationFlags(V, Record[3]);
|
2007-04-24 07:07:11 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case bitc::CST_CODE_CE_CAST: { // CE_CAST: [opcode, opty, opval]
|
|
|
|
if (Record.size() < 3) return Error("Invalid CE_CAST record");
|
|
|
|
int Opc = GetDecodedCastOpcode(Record[0]);
|
2007-04-24 18:15:21 +00:00
|
|
|
if (Opc < 0) {
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getUndef(CurTy); // Unknown cast.
|
2007-04-24 18:15:21 +00:00
|
|
|
} else {
|
|
|
|
const Type *OpTy = getTypeByID(Record[1]);
|
2007-05-06 07:33:01 +00:00
|
|
|
if (!OpTy) return Error("Invalid CE_CAST record");
|
2007-04-24 18:15:21 +00:00
|
|
|
Constant *Op = ValueList.getConstantFwdRef(Record[2], OpTy);
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantExprCast(Opc, Op, CurTy);
|
2007-04-24 18:15:21 +00:00
|
|
|
}
|
2007-04-24 07:07:11 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case bitc::CST_CODE_CE_GEP: { // CE_GEP: [n x operands]
|
2007-05-04 19:11:41 +00:00
|
|
|
if (Record.size() & 1) return Error("Invalid CE_GEP record");
|
2007-04-24 07:07:11 +00:00
|
|
|
SmallVector<Constant*, 16> Elts;
|
2007-05-04 19:11:41 +00:00
|
|
|
for (unsigned i = 0, e = Record.size(); i != e; i += 2) {
|
2007-04-24 07:07:11 +00:00
|
|
|
const Type *ElTy = getTypeByID(Record[i]);
|
|
|
|
if (!ElTy) return Error("Invalid CE_GEP record");
|
|
|
|
Elts.push_back(ValueList.getConstantFwdRef(Record[i+1], ElTy));
|
|
|
|
}
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantExprGetElementPtr(Elts[0], &Elts[1],
|
|
|
|
Elts.size()-1);
|
2007-04-24 18:15:21 +00:00
|
|
|
break;
|
2007-04-24 07:07:11 +00:00
|
|
|
}
|
|
|
|
case bitc::CST_CODE_CE_SELECT: // CE_SELECT: [opval#, opval#, opval#]
|
|
|
|
if (Record.size() < 3) return Error("Invalid CE_SELECT record");
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantExprSelect(ValueList.getConstantFwdRef(Record[0],
|
2007-04-24 07:07:11 +00:00
|
|
|
Type::Int1Ty),
|
|
|
|
ValueList.getConstantFwdRef(Record[1],CurTy),
|
|
|
|
ValueList.getConstantFwdRef(Record[2],CurTy));
|
|
|
|
break;
|
|
|
|
case bitc::CST_CODE_CE_EXTRACTELT: { // CE_EXTRACTELT: [opty, opval, opval]
|
|
|
|
if (Record.size() < 3) return Error("Invalid CE_EXTRACTELT record");
|
|
|
|
const VectorType *OpTy =
|
|
|
|
dyn_cast_or_null<VectorType>(getTypeByID(Record[0]));
|
|
|
|
if (OpTy == 0) return Error("Invalid CE_EXTRACTELT record");
|
|
|
|
Constant *Op0 = ValueList.getConstantFwdRef(Record[1], OpTy);
|
2009-02-03 02:11:28 +00:00
|
|
|
Constant *Op1 = ValueList.getConstantFwdRef(Record[2], Type::Int32Ty);
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantExprExtractElement(Op0, Op1);
|
2007-04-24 07:07:11 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case bitc::CST_CODE_CE_INSERTELT: { // CE_INSERTELT: [opval, opval, opval]
|
|
|
|
const VectorType *OpTy = dyn_cast<VectorType>(CurTy);
|
|
|
|
if (Record.size() < 3 || OpTy == 0)
|
|
|
|
return Error("Invalid CE_INSERTELT record");
|
|
|
|
Constant *Op0 = ValueList.getConstantFwdRef(Record[0], OpTy);
|
|
|
|
Constant *Op1 = ValueList.getConstantFwdRef(Record[1],
|
|
|
|
OpTy->getElementType());
|
|
|
|
Constant *Op2 = ValueList.getConstantFwdRef(Record[2], Type::Int32Ty);
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantExprInsertElement(Op0, Op1, Op2);
|
2007-04-24 07:07:11 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case bitc::CST_CODE_CE_SHUFFLEVEC: { // CE_SHUFFLEVEC: [opval, opval, opval]
|
|
|
|
const VectorType *OpTy = dyn_cast<VectorType>(CurTy);
|
|
|
|
if (Record.size() < 3 || OpTy == 0)
|
2009-02-12 21:28:33 +00:00
|
|
|
return Error("Invalid CE_SHUFFLEVEC record");
|
2007-04-24 07:07:11 +00:00
|
|
|
Constant *Op0 = ValueList.getConstantFwdRef(Record[0], OpTy);
|
|
|
|
Constant *Op1 = ValueList.getConstantFwdRef(Record[1], OpTy);
|
2009-07-07 20:18:58 +00:00
|
|
|
const Type *ShufTy = Context.getVectorType(Type::Int32Ty,
|
|
|
|
OpTy->getNumElements());
|
2007-04-24 07:07:11 +00:00
|
|
|
Constant *Op2 = ValueList.getConstantFwdRef(Record[2], ShufTy);
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantExprShuffleVector(Op0, Op1, Op2);
|
2007-04-24 07:07:11 +00:00
|
|
|
break;
|
|
|
|
}
|
2009-02-12 21:28:33 +00:00
|
|
|
case bitc::CST_CODE_CE_SHUFVEC_EX: { // [opty, opval, opval, opval]
|
|
|
|
const VectorType *RTy = dyn_cast<VectorType>(CurTy);
|
|
|
|
const VectorType *OpTy = dyn_cast<VectorType>(getTypeByID(Record[0]));
|
|
|
|
if (Record.size() < 4 || RTy == 0 || OpTy == 0)
|
|
|
|
return Error("Invalid CE_SHUFVEC_EX record");
|
|
|
|
Constant *Op0 = ValueList.getConstantFwdRef(Record[1], OpTy);
|
|
|
|
Constant *Op1 = ValueList.getConstantFwdRef(Record[2], OpTy);
|
2009-07-07 20:18:58 +00:00
|
|
|
const Type *ShufTy = Context.getVectorType(Type::Int32Ty,
|
|
|
|
RTy->getNumElements());
|
2009-02-12 21:28:33 +00:00
|
|
|
Constant *Op2 = ValueList.getConstantFwdRef(Record[3], ShufTy);
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantExprShuffleVector(Op0, Op1, Op2);
|
2009-02-12 21:28:33 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-04-24 07:07:11 +00:00
|
|
|
case bitc::CST_CODE_CE_CMP: { // CE_CMP: [opty, opval, opval, pred]
|
|
|
|
if (Record.size() < 4) return Error("Invalid CE_CMP record");
|
|
|
|
const Type *OpTy = getTypeByID(Record[0]);
|
|
|
|
if (OpTy == 0) return Error("Invalid CE_CMP record");
|
|
|
|
Constant *Op0 = ValueList.getConstantFwdRef(Record[1], OpTy);
|
|
|
|
Constant *Op1 = ValueList.getConstantFwdRef(Record[2], OpTy);
|
|
|
|
|
|
|
|
if (OpTy->isFloatingPoint())
|
2009-07-07 20:18:58 +00:00
|
|
|
V = Context.getConstantExprFCmp(Record[3], Op0, Op1);
|
2008-05-12 19:01:56 +00:00
|
|
|
else
|
2009-07-08 03:04:38 +00:00
|
|
|
V = Context.getConstantExprICmp(Record[3], Op0, Op1);
|
2007-04-24 07:07:11 +00:00
|
|
|
break;
|
2007-04-24 05:48:56 +00:00
|
|
|
}
|
2007-05-06 01:58:20 +00:00
|
|
|
case bitc::CST_CODE_INLINEASM: {
|
|
|
|
if (Record.size() < 2) return Error("Invalid INLINEASM record");
|
|
|
|
std::string AsmStr, ConstrStr;
|
|
|
|
bool HasSideEffects = Record[0];
|
|
|
|
unsigned AsmStrSize = Record[1];
|
|
|
|
if (2+AsmStrSize >= Record.size())
|
|
|
|
return Error("Invalid INLINEASM record");
|
|
|
|
unsigned ConstStrSize = Record[2+AsmStrSize];
|
|
|
|
if (3+AsmStrSize+ConstStrSize > Record.size())
|
|
|
|
return Error("Invalid INLINEASM record");
|
|
|
|
|
|
|
|
for (unsigned i = 0; i != AsmStrSize; ++i)
|
|
|
|
AsmStr += (char)Record[2+i];
|
|
|
|
for (unsigned i = 0; i != ConstStrSize; ++i)
|
|
|
|
ConstrStr += (char)Record[3+AsmStrSize+i];
|
|
|
|
const PointerType *PTy = cast<PointerType>(CurTy);
|
|
|
|
V = InlineAsm::get(cast<FunctionType>(PTy->getElementType()),
|
|
|
|
AsmStr, ConstrStr, HasSideEffects);
|
|
|
|
break;
|
|
|
|
}
|
2007-04-24 05:48:56 +00:00
|
|
|
}
|
|
|
|
|
2007-05-01 07:01:57 +00:00
|
|
|
ValueList.AssignValue(V, NextCstNo);
|
2007-04-24 05:48:56 +00:00
|
|
|
++NextCstNo;
|
2007-04-24 03:30:34 +00:00
|
|
|
}
|
2008-08-21 02:34:16 +00:00
|
|
|
|
|
|
|
if (NextCstNo != ValueList.size())
|
|
|
|
return Error("Invalid constant reference!");
|
|
|
|
|
|
|
|
if (Stream.ReadBlockEnd())
|
|
|
|
return Error("Error at end of constants block");
|
|
|
|
|
|
|
|
// Once all the constants have been read, go through and resolve forward
|
|
|
|
// references.
|
|
|
|
ValueList.ResolveConstantForwardRefs();
|
|
|
|
return false;
|
2007-04-24 03:30:34 +00:00
|
|
|
}
|
2007-04-22 06:23:29 +00:00
|
|
|
|
2007-05-01 05:52:21 +00:00
|
|
|
/// RememberAndSkipFunctionBody - When we see the block for a function body,
|
|
|
|
/// remember where it is and then skip it. This lets us lazily deserialize the
|
|
|
|
/// functions.
|
|
|
|
bool BitcodeReader::RememberAndSkipFunctionBody() {
|
2007-05-01 04:59:48 +00:00
|
|
|
// Get the function we are talking about.
|
|
|
|
if (FunctionsWithBodies.empty())
|
|
|
|
return Error("Insufficient function protos");
|
|
|
|
|
|
|
|
Function *Fn = FunctionsWithBodies.back();
|
|
|
|
FunctionsWithBodies.pop_back();
|
|
|
|
|
|
|
|
// Save the current stream state.
|
|
|
|
uint64_t CurBit = Stream.GetCurrentBitNo();
|
|
|
|
DeferredFunctionInfo[Fn] = std::make_pair(CurBit, Fn->getLinkage());
|
|
|
|
|
|
|
|
// Set the functions linkage to GhostLinkage so we know it is lazily
|
|
|
|
// deserialized.
|
|
|
|
Fn->setLinkage(GlobalValue::GhostLinkage);
|
|
|
|
|
|
|
|
// Skip over the function block for now.
|
|
|
|
if (Stream.SkipBlock())
|
|
|
|
return Error("Malformed block record");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-05-01 05:01:34 +00:00
|
|
|
bool BitcodeReader::ParseModule(const std::string &ModuleID) {
|
2007-04-22 06:23:29 +00:00
|
|
|
// Reject multiple MODULE_BLOCK's in a single bitstream.
|
|
|
|
if (TheModule)
|
|
|
|
return Error("Multiple MODULE_BLOCKs in same stream");
|
|
|
|
|
2007-05-05 00:17:00 +00:00
|
|
|
if (Stream.EnterSubBlock(bitc::MODULE_BLOCK_ID))
|
2007-04-22 06:23:29 +00:00
|
|
|
return Error("Malformed block record");
|
|
|
|
|
|
|
|
// Otherwise, create the module.
|
2009-07-01 16:58:40 +00:00
|
|
|
TheModule = new Module(ModuleID, Context);
|
2007-04-22 06:23:29 +00:00
|
|
|
|
|
|
|
SmallVector<uint64_t, 64> Record;
|
|
|
|
std::vector<std::string> SectionTable;
|
2008-08-17 18:44:35 +00:00
|
|
|
std::vector<std::string> GCTable;
|
2007-04-22 06:23:29 +00:00
|
|
|
|
|
|
|
// Read all the records for this module.
|
|
|
|
while (!Stream.AtEndOfStream()) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
2007-04-24 00:21:45 +00:00
|
|
|
if (Code == bitc::END_BLOCK) {
|
2007-05-01 05:52:21 +00:00
|
|
|
if (Stream.ReadBlockEnd())
|
|
|
|
return Error("Error at end of module block");
|
|
|
|
|
|
|
|
// Patch the initializers for globals and aliases up.
|
2007-04-26 02:46:40 +00:00
|
|
|
ResolveGlobalAndAliasInits();
|
|
|
|
if (!GlobalInits.empty() || !AliasInits.empty())
|
2007-04-24 00:21:45 +00:00
|
|
|
return Error("Malformed global initializer set");
|
2007-05-01 04:59:48 +00:00
|
|
|
if (!FunctionsWithBodies.empty())
|
|
|
|
return Error("Too few function bodies found");
|
2007-05-01 05:52:21 +00:00
|
|
|
|
2007-08-04 01:51:18 +00:00
|
|
|
// Look for intrinsic functions which need to be upgraded at some point
|
|
|
|
for (Module::iterator FI = TheModule->begin(), FE = TheModule->end();
|
|
|
|
FI != FE; ++FI) {
|
2007-12-17 22:33:23 +00:00
|
|
|
Function* NewFn;
|
|
|
|
if (UpgradeIntrinsicFunction(FI, NewFn))
|
2007-08-04 01:51:18 +00:00
|
|
|
UpgradedIntrinsics.push_back(std::make_pair(FI, NewFn));
|
|
|
|
}
|
|
|
|
|
2007-05-01 05:52:21 +00:00
|
|
|
// Force deallocation of memory for these vectors to favor the client that
|
|
|
|
// want lazy deserialization.
|
|
|
|
std::vector<std::pair<GlobalVariable*, unsigned> >().swap(GlobalInits);
|
|
|
|
std::vector<std::pair<GlobalAlias*, unsigned> >().swap(AliasInits);
|
|
|
|
std::vector<Function*>().swap(FunctionsWithBodies);
|
2007-04-24 18:15:21 +00:00
|
|
|
return false;
|
2007-04-24 00:21:45 +00:00
|
|
|
}
|
2007-04-22 06:23:29 +00:00
|
|
|
|
|
|
|
if (Code == bitc::ENTER_SUBBLOCK) {
|
|
|
|
switch (Stream.ReadSubBlockID()) {
|
|
|
|
default: // Skip unknown content.
|
|
|
|
if (Stream.SkipBlock())
|
|
|
|
return Error("Malformed block record");
|
|
|
|
break;
|
2007-05-05 18:57:30 +00:00
|
|
|
case bitc::BLOCKINFO_BLOCK_ID:
|
|
|
|
if (Stream.ReadBlockInfoBlock())
|
|
|
|
return Error("Malformed BlockInfoBlock");
|
|
|
|
break;
|
2007-05-04 03:30:17 +00:00
|
|
|
case bitc::PARAMATTR_BLOCK_ID:
|
2008-09-25 21:00:45 +00:00
|
|
|
if (ParseAttributeBlock())
|
2007-05-04 03:30:17 +00:00
|
|
|
return true;
|
|
|
|
break;
|
2007-04-22 06:23:29 +00:00
|
|
|
case bitc::TYPE_BLOCK_ID:
|
2007-05-01 05:01:34 +00:00
|
|
|
if (ParseTypeTable())
|
2007-04-22 06:23:29 +00:00
|
|
|
return true;
|
|
|
|
break;
|
|
|
|
case bitc::TYPE_SYMTAB_BLOCK_ID:
|
2007-05-01 05:01:34 +00:00
|
|
|
if (ParseTypeSymbolTable())
|
2007-04-22 06:23:29 +00:00
|
|
|
return true;
|
|
|
|
break;
|
2007-04-23 21:26:05 +00:00
|
|
|
case bitc::VALUE_SYMTAB_BLOCK_ID:
|
2007-05-01 05:01:34 +00:00
|
|
|
if (ParseValueSymbolTable())
|
2007-04-23 21:26:05 +00:00
|
|
|
return true;
|
|
|
|
break;
|
2007-04-24 03:30:34 +00:00
|
|
|
case bitc::CONSTANTS_BLOCK_ID:
|
2007-05-01 05:01:34 +00:00
|
|
|
if (ParseConstants() || ResolveGlobalAndAliasInits())
|
2007-04-24 03:30:34 +00:00
|
|
|
return true;
|
|
|
|
break;
|
2009-07-22 17:43:22 +00:00
|
|
|
case bitc::METADATA_BLOCK_ID:
|
|
|
|
if (ParseMetadata())
|
|
|
|
return true;
|
|
|
|
break;
|
2007-05-01 04:59:48 +00:00
|
|
|
case bitc::FUNCTION_BLOCK_ID:
|
|
|
|
// If this is the first function body we've seen, reverse the
|
|
|
|
// FunctionsWithBodies list.
|
|
|
|
if (!HasReversedFunctionsWithBodies) {
|
|
|
|
std::reverse(FunctionsWithBodies.begin(), FunctionsWithBodies.end());
|
|
|
|
HasReversedFunctionsWithBodies = true;
|
|
|
|
}
|
|
|
|
|
2007-05-01 05:52:21 +00:00
|
|
|
if (RememberAndSkipFunctionBody())
|
2007-05-01 04:59:48 +00:00
|
|
|
return true;
|
|
|
|
break;
|
2007-04-22 06:23:29 +00:00
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2007-04-23 16:04:05 +00:00
|
|
|
if (Code == bitc::DEFINE_ABBREV) {
|
2007-04-23 18:58:34 +00:00
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
2007-04-22 06:23:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Read a record.
|
|
|
|
switch (Stream.ReadRecord(Code, Record)) {
|
|
|
|
default: break; // Default behavior, ignore unknown content.
|
|
|
|
case bitc::MODULE_CODE_VERSION: // VERSION: [version#]
|
|
|
|
if (Record.size() < 1)
|
|
|
|
return Error("Malformed MODULE_CODE_VERSION");
|
|
|
|
// Only version #0 is supported so far.
|
|
|
|
if (Record[0] != 0)
|
|
|
|
return Error("Unknown bitstream version!");
|
|
|
|
break;
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::MODULE_CODE_TRIPLE: { // TRIPLE: [strchr x N]
|
2007-04-22 06:23:29 +00:00
|
|
|
std::string S;
|
|
|
|
if (ConvertToString(Record, 0, S))
|
|
|
|
return Error("Invalid MODULE_CODE_TRIPLE record");
|
|
|
|
TheModule->setTargetTriple(S);
|
|
|
|
break;
|
|
|
|
}
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::MODULE_CODE_DATALAYOUT: { // DATALAYOUT: [strchr x N]
|
2007-04-22 06:23:29 +00:00
|
|
|
std::string S;
|
|
|
|
if (ConvertToString(Record, 0, S))
|
|
|
|
return Error("Invalid MODULE_CODE_DATALAYOUT record");
|
|
|
|
TheModule->setDataLayout(S);
|
|
|
|
break;
|
|
|
|
}
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::MODULE_CODE_ASM: { // ASM: [strchr x N]
|
2007-04-22 06:23:29 +00:00
|
|
|
std::string S;
|
|
|
|
if (ConvertToString(Record, 0, S))
|
|
|
|
return Error("Invalid MODULE_CODE_ASM record");
|
|
|
|
TheModule->setModuleInlineAsm(S);
|
|
|
|
break;
|
|
|
|
}
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::MODULE_CODE_DEPLIB: { // DEPLIB: [strchr x N]
|
2007-04-22 06:23:29 +00:00
|
|
|
std::string S;
|
|
|
|
if (ConvertToString(Record, 0, S))
|
|
|
|
return Error("Invalid MODULE_CODE_DEPLIB record");
|
|
|
|
TheModule->addLibrary(S);
|
|
|
|
break;
|
|
|
|
}
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::MODULE_CODE_SECTIONNAME: { // SECTIONNAME: [strchr x N]
|
2007-04-22 06:23:29 +00:00
|
|
|
std::string S;
|
|
|
|
if (ConvertToString(Record, 0, S))
|
|
|
|
return Error("Invalid MODULE_CODE_SECTIONNAME record");
|
|
|
|
SectionTable.push_back(S);
|
|
|
|
break;
|
|
|
|
}
|
2008-08-17 18:44:35 +00:00
|
|
|
case bitc::MODULE_CODE_GCNAME: { // SECTIONNAME: [strchr x N]
|
2007-12-10 03:18:06 +00:00
|
|
|
std::string S;
|
|
|
|
if (ConvertToString(Record, 0, S))
|
2008-08-17 18:44:35 +00:00
|
|
|
return Error("Invalid MODULE_CODE_GCNAME record");
|
|
|
|
GCTable.push_back(S);
|
2007-12-10 03:18:06 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-12-11 08:59:05 +00:00
|
|
|
// GLOBALVAR: [pointer type, isconst, initid,
|
2007-04-22 06:23:29 +00:00
|
|
|
// linkage, alignment, section, visibility, threadlocal]
|
|
|
|
case bitc::MODULE_CODE_GLOBALVAR: {
|
2007-04-23 16:04:05 +00:00
|
|
|
if (Record.size() < 6)
|
2007-04-22 06:23:29 +00:00
|
|
|
return Error("Invalid MODULE_CODE_GLOBALVAR record");
|
|
|
|
const Type *Ty = getTypeByID(Record[0]);
|
|
|
|
if (!isa<PointerType>(Ty))
|
|
|
|
return Error("Global not a pointer type!");
|
2007-12-11 08:59:05 +00:00
|
|
|
unsigned AddressSpace = cast<PointerType>(Ty)->getAddressSpace();
|
2007-04-22 06:23:29 +00:00
|
|
|
Ty = cast<PointerType>(Ty)->getElementType();
|
|
|
|
|
|
|
|
bool isConstant = Record[1];
|
|
|
|
GlobalValue::LinkageTypes Linkage = GetDecodedLinkage(Record[3]);
|
|
|
|
unsigned Alignment = (1 << Record[4]) >> 1;
|
|
|
|
std::string Section;
|
|
|
|
if (Record[5]) {
|
|
|
|
if (Record[5]-1 >= SectionTable.size())
|
|
|
|
return Error("Invalid section ID");
|
|
|
|
Section = SectionTable[Record[5]-1];
|
|
|
|
}
|
2007-04-23 16:04:05 +00:00
|
|
|
GlobalValue::VisibilityTypes Visibility = GlobalValue::DefaultVisibility;
|
2007-05-06 19:27:46 +00:00
|
|
|
if (Record.size() > 6)
|
|
|
|
Visibility = GetDecodedVisibility(Record[6]);
|
2007-04-23 16:04:05 +00:00
|
|
|
bool isThreadLocal = false;
|
2007-05-06 19:27:46 +00:00
|
|
|
if (Record.size() > 7)
|
|
|
|
isThreadLocal = Record[7];
|
2007-04-22 06:23:29 +00:00
|
|
|
|
|
|
|
GlobalVariable *NewGV =
|
2009-07-08 19:03:57 +00:00
|
|
|
new GlobalVariable(*TheModule, Ty, isConstant, Linkage, 0, "", 0,
|
2007-12-11 08:59:05 +00:00
|
|
|
isThreadLocal, AddressSpace);
|
2007-04-22 06:23:29 +00:00
|
|
|
NewGV->setAlignment(Alignment);
|
|
|
|
if (!Section.empty())
|
|
|
|
NewGV->setSection(Section);
|
|
|
|
NewGV->setVisibility(Visibility);
|
|
|
|
NewGV->setThreadLocal(isThreadLocal);
|
|
|
|
|
2007-04-23 21:26:05 +00:00
|
|
|
ValueList.push_back(NewGV);
|
|
|
|
|
2007-04-24 00:18:21 +00:00
|
|
|
// Remember which value to use for the global initializer.
|
|
|
|
if (unsigned InitID = Record[2])
|
|
|
|
GlobalInits.push_back(std::make_pair(NewGV, InitID-1));
|
2007-04-22 06:23:29 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-05-08 05:38:01 +00:00
|
|
|
// FUNCTION: [type, callingconv, isproto, linkage, paramattr,
|
2008-08-17 18:44:35 +00:00
|
|
|
// alignment, section, visibility, gc]
|
2007-04-22 06:23:29 +00:00
|
|
|
case bitc::MODULE_CODE_FUNCTION: {
|
2007-05-08 05:38:01 +00:00
|
|
|
if (Record.size() < 8)
|
2007-04-22 06:23:29 +00:00
|
|
|
return Error("Invalid MODULE_CODE_FUNCTION record");
|
|
|
|
const Type *Ty = getTypeByID(Record[0]);
|
|
|
|
if (!isa<PointerType>(Ty))
|
|
|
|
return Error("Function not a pointer type!");
|
|
|
|
const FunctionType *FTy =
|
|
|
|
dyn_cast<FunctionType>(cast<PointerType>(Ty)->getElementType());
|
|
|
|
if (!FTy)
|
|
|
|
return Error("Function not a pointer to function type!");
|
|
|
|
|
2008-04-06 20:25:17 +00:00
|
|
|
Function *Func = Function::Create(FTy, GlobalValue::ExternalLinkage,
|
|
|
|
"", TheModule);
|
2007-04-22 06:23:29 +00:00
|
|
|
|
|
|
|
Func->setCallingConv(Record[1]);
|
2007-05-01 04:59:48 +00:00
|
|
|
bool isProto = Record[2];
|
2007-04-22 06:23:29 +00:00
|
|
|
Func->setLinkage(GetDecodedLinkage(Record[3]));
|
2008-09-25 21:00:45 +00:00
|
|
|
Func->setAttributes(getAttributes(Record[4]));
|
2007-05-08 05:38:01 +00:00
|
|
|
|
|
|
|
Func->setAlignment((1 << Record[5]) >> 1);
|
|
|
|
if (Record[6]) {
|
|
|
|
if (Record[6]-1 >= SectionTable.size())
|
2007-04-22 06:23:29 +00:00
|
|
|
return Error("Invalid section ID");
|
2007-05-08 05:38:01 +00:00
|
|
|
Func->setSection(SectionTable[Record[6]-1]);
|
2007-04-22 06:23:29 +00:00
|
|
|
}
|
2007-05-08 05:38:01 +00:00
|
|
|
Func->setVisibility(GetDecodedVisibility(Record[7]));
|
2007-12-10 03:18:06 +00:00
|
|
|
if (Record.size() > 8 && Record[8]) {
|
2008-08-17 18:44:35 +00:00
|
|
|
if (Record[8]-1 > GCTable.size())
|
|
|
|
return Error("Invalid GC ID");
|
|
|
|
Func->setGC(GCTable[Record[8]-1].c_str());
|
2007-12-10 03:18:06 +00:00
|
|
|
}
|
2007-04-23 21:26:05 +00:00
|
|
|
ValueList.push_back(Func);
|
2007-05-01 04:59:48 +00:00
|
|
|
|
|
|
|
// If this is a function with a body, remember the prototype we are
|
|
|
|
// creating now, so that we can match up the body with them later.
|
|
|
|
if (!isProto)
|
|
|
|
FunctionsWithBodies.push_back(Func);
|
2007-04-22 06:23:29 +00:00
|
|
|
break;
|
|
|
|
}
|
2008-03-12 00:49:19 +00:00
|
|
|
// ALIAS: [alias type, aliasee val#, linkage]
|
2008-03-11 21:40:17 +00:00
|
|
|
// ALIAS: [alias type, aliasee val#, linkage, visibility]
|
2007-04-26 03:27:58 +00:00
|
|
|
case bitc::MODULE_CODE_ALIAS: {
|
2007-04-26 02:46:40 +00:00
|
|
|
if (Record.size() < 3)
|
|
|
|
return Error("Invalid MODULE_ALIAS record");
|
|
|
|
const Type *Ty = getTypeByID(Record[0]);
|
|
|
|
if (!isa<PointerType>(Ty))
|
|
|
|
return Error("Function not a pointer type!");
|
|
|
|
|
|
|
|
GlobalAlias *NewGA = new GlobalAlias(Ty, GetDecodedLinkage(Record[2]),
|
|
|
|
"", 0, TheModule);
|
2008-03-12 00:49:19 +00:00
|
|
|
// Old bitcode files didn't have visibility field.
|
|
|
|
if (Record.size() > 3)
|
|
|
|
NewGA->setVisibility(GetDecodedVisibility(Record[3]));
|
2007-04-26 02:46:40 +00:00
|
|
|
ValueList.push_back(NewGA);
|
|
|
|
AliasInits.push_back(std::make_pair(NewGA, Record[1]));
|
|
|
|
break;
|
2007-04-22 06:23:29 +00:00
|
|
|
}
|
2007-04-26 03:27:58 +00:00
|
|
|
/// MODULE_CODE_PURGEVALS: [numvals]
|
|
|
|
case bitc::MODULE_CODE_PURGEVALS:
|
|
|
|
// Trim down the value list to the specified size.
|
|
|
|
if (Record.size() < 1 || Record[0] > ValueList.size())
|
|
|
|
return Error("Invalid MODULE_PURGEVALS record");
|
|
|
|
ValueList.shrinkTo(Record[0]);
|
|
|
|
break;
|
|
|
|
}
|
2007-04-22 06:23:29 +00:00
|
|
|
Record.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
return Error("Premature end of bitstream");
|
|
|
|
}
|
|
|
|
|
2007-04-29 07:54:31 +00:00
|
|
|
bool BitcodeReader::ParseBitcode() {
|
2007-04-22 06:23:29 +00:00
|
|
|
TheModule = 0;
|
|
|
|
|
2007-04-29 07:54:31 +00:00
|
|
|
if (Buffer->getBufferSize() & 3)
|
2007-04-22 06:23:29 +00:00
|
|
|
return Error("Bitcode stream should be a multiple of 4 bytes in length");
|
|
|
|
|
2007-04-29 07:54:31 +00:00
|
|
|
unsigned char *BufPtr = (unsigned char *)Buffer->getBufferStart();
|
2008-07-09 05:14:23 +00:00
|
|
|
unsigned char *BufEnd = BufPtr+Buffer->getBufferSize();
|
|
|
|
|
|
|
|
// If we have a wrapper header, parse it and ignore the non-bc file contents.
|
|
|
|
// The magic number is 0x0B17C0DE stored in little endian.
|
2009-04-06 20:54:32 +00:00
|
|
|
if (isBitcodeWrapper(BufPtr, BufEnd))
|
|
|
|
if (SkipBitcodeWrapperHeader(BufPtr, BufEnd))
|
2008-07-09 05:14:23 +00:00
|
|
|
return Error("Invalid bitcode wrapper header");
|
|
|
|
|
2009-04-26 20:59:02 +00:00
|
|
|
StreamFile.init(BufPtr, BufEnd);
|
|
|
|
Stream.init(StreamFile);
|
2007-04-22 06:23:29 +00:00
|
|
|
|
|
|
|
// Sniff for the signature.
|
|
|
|
if (Stream.Read(8) != 'B' ||
|
|
|
|
Stream.Read(8) != 'C' ||
|
|
|
|
Stream.Read(4) != 0x0 ||
|
|
|
|
Stream.Read(4) != 0xC ||
|
|
|
|
Stream.Read(4) != 0xE ||
|
|
|
|
Stream.Read(4) != 0xD)
|
|
|
|
return Error("Invalid bitcode signature");
|
|
|
|
|
|
|
|
// We expect a number of well-defined blocks, though we don't necessarily
|
|
|
|
// need to understand them all.
|
|
|
|
while (!Stream.AtEndOfStream()) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
|
|
|
|
|
|
|
if (Code != bitc::ENTER_SUBBLOCK)
|
|
|
|
return Error("Invalid record at top-level");
|
|
|
|
|
|
|
|
unsigned BlockID = Stream.ReadSubBlockID();
|
|
|
|
|
|
|
|
// We only know the MODULE subblock ID.
|
2007-05-05 00:17:00 +00:00
|
|
|
switch (BlockID) {
|
|
|
|
case bitc::BLOCKINFO_BLOCK_ID:
|
|
|
|
if (Stream.ReadBlockInfoBlock())
|
|
|
|
return Error("Malformed BlockInfoBlock");
|
|
|
|
break;
|
|
|
|
case bitc::MODULE_BLOCK_ID:
|
2007-05-01 05:01:34 +00:00
|
|
|
if (ParseModule(Buffer->getBufferIdentifier()))
|
2007-04-22 06:23:29 +00:00
|
|
|
return true;
|
2007-05-05 00:17:00 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
if (Stream.SkipBlock())
|
|
|
|
return Error("Malformed block record");
|
|
|
|
break;
|
2007-04-22 06:23:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2007-04-29 07:54:31 +00:00
|
|
|
|
2007-05-01 04:59:48 +00:00
|
|
|
|
2007-05-01 05:52:21 +00:00
|
|
|
/// ParseFunctionBody - Lazily parse the specified function body block.
|
|
|
|
bool BitcodeReader::ParseFunctionBody(Function *F) {
|
2007-05-05 00:17:00 +00:00
|
|
|
if (Stream.EnterSubBlock(bitc::FUNCTION_BLOCK_ID))
|
2007-05-01 05:52:21 +00:00
|
|
|
return Error("Malformed block record");
|
|
|
|
|
|
|
|
unsigned ModuleValueListSize = ValueList.size();
|
|
|
|
|
|
|
|
// Add all the function arguments to the value table.
|
|
|
|
for(Function::arg_iterator I = F->arg_begin(), E = F->arg_end(); I != E; ++I)
|
|
|
|
ValueList.push_back(I);
|
|
|
|
|
2007-05-01 07:01:57 +00:00
|
|
|
unsigned NextValueNo = ValueList.size();
|
2007-05-02 04:27:25 +00:00
|
|
|
BasicBlock *CurBB = 0;
|
|
|
|
unsigned CurBBNo = 0;
|
|
|
|
|
2007-05-01 05:52:21 +00:00
|
|
|
// Read all the records.
|
|
|
|
SmallVector<uint64_t, 64> Record;
|
|
|
|
while (1) {
|
|
|
|
unsigned Code = Stream.ReadCode();
|
|
|
|
if (Code == bitc::END_BLOCK) {
|
|
|
|
if (Stream.ReadBlockEnd())
|
|
|
|
return Error("Error at end of function block");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == bitc::ENTER_SUBBLOCK) {
|
|
|
|
switch (Stream.ReadSubBlockID()) {
|
|
|
|
default: // Skip unknown content.
|
|
|
|
if (Stream.SkipBlock())
|
|
|
|
return Error("Malformed block record");
|
|
|
|
break;
|
|
|
|
case bitc::CONSTANTS_BLOCK_ID:
|
|
|
|
if (ParseConstants()) return true;
|
2007-05-01 07:01:57 +00:00
|
|
|
NextValueNo = ValueList.size();
|
2007-05-01 05:52:21 +00:00
|
|
|
break;
|
|
|
|
case bitc::VALUE_SYMTAB_BLOCK_ID:
|
|
|
|
if (ParseValueSymbolTable()) return true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Code == bitc::DEFINE_ABBREV) {
|
|
|
|
Stream.ReadAbbrevRecord();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read a record.
|
|
|
|
Record.clear();
|
2007-05-01 07:01:57 +00:00
|
|
|
Instruction *I = 0;
|
2009-07-20 21:19:07 +00:00
|
|
|
unsigned BitCode = Stream.ReadRecord(Code, Record);
|
|
|
|
switch (BitCode) {
|
2007-05-01 07:01:57 +00:00
|
|
|
default: // Default behavior: reject
|
|
|
|
return Error("Unknown instruction");
|
2007-05-01 05:52:21 +00:00
|
|
|
case bitc::FUNC_CODE_DECLAREBLOCKS: // DECLAREBLOCKS: [nblocks]
|
2007-05-01 07:01:57 +00:00
|
|
|
if (Record.size() < 1 || Record[0] == 0)
|
|
|
|
return Error("Invalid DECLAREBLOCKS record");
|
2007-05-01 05:52:21 +00:00
|
|
|
// Create all the basic blocks for the function.
|
2007-05-03 22:09:51 +00:00
|
|
|
FunctionBBs.resize(Record[0]);
|
2007-05-01 05:52:21 +00:00
|
|
|
for (unsigned i = 0, e = FunctionBBs.size(); i != e; ++i)
|
2008-04-06 20:25:17 +00:00
|
|
|
FunctionBBs[i] = BasicBlock::Create("", F);
|
2007-05-01 07:01:57 +00:00
|
|
|
CurBB = FunctionBBs[0];
|
|
|
|
continue;
|
|
|
|
|
2007-05-06 00:21:25 +00:00
|
|
|
case bitc::FUNC_CODE_INST_BINOP: { // BINOP: [opval, ty, opval, opcode]
|
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *LHS, *RHS;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, LHS) ||
|
|
|
|
getValue(Record, OpNum, LHS->getType(), RHS) ||
|
2009-07-20 21:19:07 +00:00
|
|
|
OpNum+1 > Record.size())
|
2007-05-06 00:21:25 +00:00
|
|
|
return Error("Invalid BINOP record");
|
|
|
|
|
2009-07-20 21:19:07 +00:00
|
|
|
int Opc = GetDecodedBinaryOpcode(Record[OpNum++], LHS->getType());
|
2007-05-06 00:21:25 +00:00
|
|
|
if (Opc == -1) return Error("Invalid BINOP record");
|
2008-05-16 19:29:10 +00:00
|
|
|
I = BinaryOperator::Create((Instruction::BinaryOps)Opc, LHS, RHS);
|
2009-07-20 21:19:07 +00:00
|
|
|
if (OpNum < Record.size())
|
|
|
|
SetOptimizationFlags(I, Record[3]);
|
2007-05-01 07:01:57 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-05-06 00:21:25 +00:00
|
|
|
case bitc::FUNC_CODE_INST_CAST: { // CAST: [opval, opty, destty, castopc]
|
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Op;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Op) ||
|
|
|
|
OpNum+2 != Record.size())
|
|
|
|
return Error("Invalid CAST record");
|
|
|
|
|
|
|
|
const Type *ResTy = getTypeByID(Record[OpNum]);
|
|
|
|
int Opc = GetDecodedCastOpcode(Record[OpNum+1]);
|
|
|
|
if (Opc == -1 || ResTy == 0)
|
2007-05-02 04:27:25 +00:00
|
|
|
return Error("Invalid CAST record");
|
2008-05-16 19:29:10 +00:00
|
|
|
I = CastInst::Create((Instruction::CastOps)Opc, Op, ResTy);
|
2007-05-02 04:27:25 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-05-04 19:11:41 +00:00
|
|
|
case bitc::FUNC_CODE_INST_GEP: { // GEP: [n x operands]
|
2007-05-06 00:00:00 +00:00
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *BasePtr;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, BasePtr))
|
2007-05-02 05:16:49 +00:00
|
|
|
return Error("Invalid GEP record");
|
|
|
|
|
2007-05-02 05:46:45 +00:00
|
|
|
SmallVector<Value*, 16> GEPIdx;
|
2007-05-06 00:00:00 +00:00
|
|
|
while (OpNum != Record.size()) {
|
|
|
|
Value *Op;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Op))
|
2007-05-02 05:16:49 +00:00
|
|
|
return Error("Invalid GEP record");
|
2007-05-06 00:00:00 +00:00
|
|
|
GEPIdx.push_back(Op);
|
2007-05-02 05:16:49 +00:00
|
|
|
}
|
|
|
|
|
2008-04-06 20:25:17 +00:00
|
|
|
I = GetElementPtrInst::Create(BasePtr, GEPIdx.begin(), GEPIdx.end());
|
2007-05-02 05:16:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-05-31 00:58:22 +00:00
|
|
|
case bitc::FUNC_CODE_INST_EXTRACTVAL: {
|
|
|
|
// EXTRACTVAL: [opty, opval, n x indices]
|
2008-05-23 01:55:30 +00:00
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Agg;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Agg))
|
|
|
|
return Error("Invalid EXTRACTVAL record");
|
|
|
|
|
2008-05-31 00:58:22 +00:00
|
|
|
SmallVector<unsigned, 4> EXTRACTVALIdx;
|
|
|
|
for (unsigned RecSize = Record.size();
|
|
|
|
OpNum != RecSize; ++OpNum) {
|
|
|
|
uint64_t Index = Record[OpNum];
|
|
|
|
if ((unsigned)Index != Index)
|
|
|
|
return Error("Invalid EXTRACTVAL index");
|
|
|
|
EXTRACTVALIdx.push_back((unsigned)Index);
|
2008-05-23 01:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
I = ExtractValueInst::Create(Agg,
|
|
|
|
EXTRACTVALIdx.begin(), EXTRACTVALIdx.end());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-05-31 00:58:22 +00:00
|
|
|
case bitc::FUNC_CODE_INST_INSERTVAL: {
|
|
|
|
// INSERTVAL: [opty, opval, opty, opval, n x indices]
|
2008-05-23 01:55:30 +00:00
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Agg;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Agg))
|
|
|
|
return Error("Invalid INSERTVAL record");
|
|
|
|
Value *Val;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Val))
|
|
|
|
return Error("Invalid INSERTVAL record");
|
|
|
|
|
2008-05-31 00:58:22 +00:00
|
|
|
SmallVector<unsigned, 4> INSERTVALIdx;
|
|
|
|
for (unsigned RecSize = Record.size();
|
|
|
|
OpNum != RecSize; ++OpNum) {
|
|
|
|
uint64_t Index = Record[OpNum];
|
|
|
|
if ((unsigned)Index != Index)
|
|
|
|
return Error("Invalid INSERTVAL index");
|
|
|
|
INSERTVALIdx.push_back((unsigned)Index);
|
2008-05-23 01:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
I = InsertValueInst::Create(Agg, Val,
|
|
|
|
INSERTVALIdx.begin(), INSERTVALIdx.end());
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-05-06 00:21:25 +00:00
|
|
|
case bitc::FUNC_CODE_INST_SELECT: { // SELECT: [opval, ty, opval, opval]
|
2008-09-16 01:01:33 +00:00
|
|
|
// obsolete form of select
|
|
|
|
// handles select i1 ... in old bitcode
|
2007-05-06 00:21:25 +00:00
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *TrueVal, *FalseVal, *Cond;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, TrueVal) ||
|
|
|
|
getValue(Record, OpNum, TrueVal->getType(), FalseVal) ||
|
2008-09-09 02:08:49 +00:00
|
|
|
getValue(Record, OpNum, Type::Int1Ty, Cond))
|
2007-05-02 05:16:49 +00:00
|
|
|
return Error("Invalid SELECT record");
|
2008-09-16 01:01:33 +00:00
|
|
|
|
|
|
|
I = SelectInst::Create(Cond, TrueVal, FalseVal);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case bitc::FUNC_CODE_INST_VSELECT: {// VSELECT: [ty,opval,opval,predty,pred]
|
|
|
|
// new form of select
|
|
|
|
// handles select i1 or select [N x i1]
|
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *TrueVal, *FalseVal, *Cond;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, TrueVal) ||
|
|
|
|
getValue(Record, OpNum, TrueVal->getType(), FalseVal) ||
|
|
|
|
getValueTypePair(Record, OpNum, NextValueNo, Cond))
|
|
|
|
return Error("Invalid SELECT record");
|
2008-09-09 01:02:47 +00:00
|
|
|
|
|
|
|
// select condition can be either i1 or [N x i1]
|
2008-09-16 01:01:33 +00:00
|
|
|
if (const VectorType* vector_type =
|
|
|
|
dyn_cast<const VectorType>(Cond->getType())) {
|
2008-09-09 01:02:47 +00:00
|
|
|
// expect <n x i1>
|
|
|
|
if (vector_type->getElementType() != Type::Int1Ty)
|
|
|
|
return Error("Invalid SELECT condition type");
|
|
|
|
} else {
|
|
|
|
// expect i1
|
|
|
|
if (Cond->getType() != Type::Int1Ty)
|
|
|
|
return Error("Invalid SELECT condition type");
|
|
|
|
}
|
2007-05-06 00:21:25 +00:00
|
|
|
|
2008-04-06 20:25:17 +00:00
|
|
|
I = SelectInst::Create(Cond, TrueVal, FalseVal);
|
2007-05-02 05:16:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case bitc::FUNC_CODE_INST_EXTRACTELT: { // EXTRACTELT: [opty, opval, opval]
|
2007-05-06 00:21:25 +00:00
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Vec, *Idx;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Vec) ||
|
|
|
|
getValue(Record, OpNum, Type::Int32Ty, Idx))
|
2007-05-02 05:16:49 +00:00
|
|
|
return Error("Invalid EXTRACTELT record");
|
2009-07-25 02:28:41 +00:00
|
|
|
I = ExtractElementInst::Create(Vec, Idx);
|
2007-05-02 05:16:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case bitc::FUNC_CODE_INST_INSERTELT: { // INSERTELT: [ty, opval,opval,opval]
|
2007-05-06 00:21:25 +00:00
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Vec, *Elt, *Idx;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Vec) ||
|
|
|
|
getValue(Record, OpNum,
|
|
|
|
cast<VectorType>(Vec->getType())->getElementType(), Elt) ||
|
|
|
|
getValue(Record, OpNum, Type::Int32Ty, Idx))
|
2007-05-02 05:16:49 +00:00
|
|
|
return Error("Invalid INSERTELT record");
|
2008-04-06 20:25:17 +00:00
|
|
|
I = InsertElementInst::Create(Vec, Elt, Idx);
|
2007-05-02 05:16:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-05-06 00:21:25 +00:00
|
|
|
case bitc::FUNC_CODE_INST_SHUFFLEVEC: {// SHUFFLEVEC: [opval,ty,opval,opval]
|
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Vec1, *Vec2, *Mask;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Vec1) ||
|
|
|
|
getValue(Record, OpNum, Vec1->getType(), Vec2))
|
|
|
|
return Error("Invalid SHUFFLEVEC record");
|
|
|
|
|
2008-11-10 04:46:22 +00:00
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Mask))
|
2007-05-02 05:16:49 +00:00
|
|
|
return Error("Invalid SHUFFLEVEC record");
|
|
|
|
I = new ShuffleVectorInst(Vec1, Vec2, Mask);
|
|
|
|
break;
|
|
|
|
}
|
2008-11-10 04:46:22 +00:00
|
|
|
|
2009-07-08 03:04:38 +00:00
|
|
|
case bitc::FUNC_CODE_INST_CMP: // CMP: [opty, opval, opval, pred]
|
|
|
|
// Old form of ICmp/FCmp returning bool
|
|
|
|
// Existed to differentiate between icmp/fcmp and vicmp/vfcmp which were
|
|
|
|
// both legal on vectors but had different behaviour.
|
2008-09-16 01:01:33 +00:00
|
|
|
case bitc::FUNC_CODE_INST_CMP2: { // CMP2: [opty, opval, opval, pred]
|
2009-07-08 03:04:38 +00:00
|
|
|
// FCmp/ICmp returning bool or vector of bool
|
|
|
|
|
2008-09-09 01:02:47 +00:00
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *LHS, *RHS;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, LHS) ||
|
|
|
|
getValue(Record, OpNum, LHS->getType(), RHS) ||
|
|
|
|
OpNum+1 != Record.size())
|
2009-07-08 03:04:38 +00:00
|
|
|
return Error("Invalid CMP record");
|
2008-09-09 01:02:47 +00:00
|
|
|
|
|
|
|
if (LHS->getType()->isFPOrFPVector())
|
2009-07-09 23:48:35 +00:00
|
|
|
I = new FCmpInst(Context, (FCmpInst::Predicate)Record[OpNum], LHS, RHS);
|
2009-07-08 03:04:38 +00:00
|
|
|
else
|
2009-07-09 23:48:35 +00:00
|
|
|
I = new ICmpInst(Context, (ICmpInst::Predicate)Record[OpNum], LHS, RHS);
|
2008-09-09 01:02:47 +00:00
|
|
|
break;
|
|
|
|
}
|
2009-07-08 03:04:38 +00:00
|
|
|
|
2008-02-22 02:49:49 +00:00
|
|
|
case bitc::FUNC_CODE_INST_GETRESULT: { // GETRESULT: [ty, val, n]
|
|
|
|
if (Record.size() != 2)
|
|
|
|
return Error("Invalid GETRESULT record");
|
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Op;
|
|
|
|
getValueTypePair(Record, OpNum, NextValueNo, Op);
|
|
|
|
unsigned Index = Record[1];
|
2008-07-23 00:34:11 +00:00
|
|
|
I = ExtractValueInst::Create(Op, Index);
|
2008-02-22 02:49:49 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-05-02 05:16:49 +00:00
|
|
|
|
2007-05-02 04:27:25 +00:00
|
|
|
case bitc::FUNC_CODE_INST_RET: // RET: [opty,opval<optional>]
|
2008-02-26 01:29:32 +00:00
|
|
|
{
|
|
|
|
unsigned Size = Record.size();
|
|
|
|
if (Size == 0) {
|
2008-04-06 20:25:17 +00:00
|
|
|
I = ReturnInst::Create();
|
2008-02-26 01:29:32 +00:00
|
|
|
break;
|
2008-07-23 00:34:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned OpNum = 0;
|
|
|
|
SmallVector<Value *,4> Vs;
|
|
|
|
do {
|
|
|
|
Value *Op = NULL;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Op))
|
|
|
|
return Error("Invalid RET record");
|
|
|
|
Vs.push_back(Op);
|
|
|
|
} while(OpNum != Record.size());
|
|
|
|
|
|
|
|
const Type *ReturnType = F->getReturnType();
|
|
|
|
if (Vs.size() > 1 ||
|
|
|
|
(isa<StructType>(ReturnType) &&
|
|
|
|
(Vs.empty() || Vs[0]->getType() != ReturnType))) {
|
2009-07-07 20:18:58 +00:00
|
|
|
Value *RV = Context.getUndef(ReturnType);
|
2008-07-23 00:34:11 +00:00
|
|
|
for (unsigned i = 0, e = Vs.size(); i != e; ++i) {
|
|
|
|
I = InsertValueInst::Create(RV, Vs[i], i, "mrv");
|
|
|
|
CurBB->getInstList().push_back(I);
|
|
|
|
ValueList.AssignValue(I, NextValueNo++);
|
|
|
|
RV = I;
|
|
|
|
}
|
|
|
|
I = ReturnInst::Create(RV);
|
2008-02-26 01:29:32 +00:00
|
|
|
break;
|
|
|
|
}
|
2008-07-23 00:34:11 +00:00
|
|
|
|
|
|
|
I = ReturnInst::Create(Vs[0]);
|
|
|
|
break;
|
2007-05-02 04:27:25 +00:00
|
|
|
}
|
2007-05-02 05:46:45 +00:00
|
|
|
case bitc::FUNC_CODE_INST_BR: { // BR: [bb#, bb#, opval] or [bb#]
|
2007-05-03 22:09:51 +00:00
|
|
|
if (Record.size() != 1 && Record.size() != 3)
|
2007-05-02 05:46:45 +00:00
|
|
|
return Error("Invalid BR record");
|
|
|
|
BasicBlock *TrueDest = getBasicBlock(Record[0]);
|
|
|
|
if (TrueDest == 0)
|
|
|
|
return Error("Invalid BR record");
|
|
|
|
|
|
|
|
if (Record.size() == 1)
|
2008-04-06 20:25:17 +00:00
|
|
|
I = BranchInst::Create(TrueDest);
|
2007-05-02 05:46:45 +00:00
|
|
|
else {
|
|
|
|
BasicBlock *FalseDest = getBasicBlock(Record[1]);
|
|
|
|
Value *Cond = getFnValueByID(Record[2], Type::Int1Ty);
|
|
|
|
if (FalseDest == 0 || Cond == 0)
|
|
|
|
return Error("Invalid BR record");
|
2008-04-06 20:25:17 +00:00
|
|
|
I = BranchInst::Create(TrueDest, FalseDest, Cond);
|
2007-05-02 05:46:45 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case bitc::FUNC_CODE_INST_SWITCH: { // SWITCH: [opty, opval, n, n x ops]
|
|
|
|
if (Record.size() < 3 || (Record.size() & 1) == 0)
|
|
|
|
return Error("Invalid SWITCH record");
|
|
|
|
const Type *OpTy = getTypeByID(Record[0]);
|
|
|
|
Value *Cond = getFnValueByID(Record[1], OpTy);
|
|
|
|
BasicBlock *Default = getBasicBlock(Record[2]);
|
|
|
|
if (OpTy == 0 || Cond == 0 || Default == 0)
|
|
|
|
return Error("Invalid SWITCH record");
|
|
|
|
unsigned NumCases = (Record.size()-3)/2;
|
2008-04-06 20:25:17 +00:00
|
|
|
SwitchInst *SI = SwitchInst::Create(Cond, Default, NumCases);
|
2007-05-02 05:46:45 +00:00
|
|
|
for (unsigned i = 0, e = NumCases; i != e; ++i) {
|
|
|
|
ConstantInt *CaseVal =
|
|
|
|
dyn_cast_or_null<ConstantInt>(getFnValueByID(Record[3+i*2], OpTy));
|
|
|
|
BasicBlock *DestBB = getBasicBlock(Record[1+3+i*2]);
|
|
|
|
if (CaseVal == 0 || DestBB == 0) {
|
|
|
|
delete SI;
|
|
|
|
return Error("Invalid SWITCH record!");
|
|
|
|
}
|
|
|
|
SI->addCase(CaseVal, DestBB);
|
|
|
|
}
|
|
|
|
I = SI;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2007-11-27 13:23:08 +00:00
|
|
|
case bitc::FUNC_CODE_INST_INVOKE: {
|
|
|
|
// INVOKE: [attrs, cc, normBB, unwindBB, fnty, op0,op1,op2, ...]
|
2007-05-08 05:38:01 +00:00
|
|
|
if (Record.size() < 4) return Error("Invalid INVOKE record");
|
2008-09-25 21:00:45 +00:00
|
|
|
AttrListPtr PAL = getAttributes(Record[0]);
|
2007-05-08 05:38:01 +00:00
|
|
|
unsigned CCInfo = Record[1];
|
|
|
|
BasicBlock *NormalBB = getBasicBlock(Record[2]);
|
|
|
|
BasicBlock *UnwindBB = getBasicBlock(Record[3]);
|
2007-05-06 00:00:00 +00:00
|
|
|
|
2007-05-08 05:38:01 +00:00
|
|
|
unsigned OpNum = 4;
|
2007-05-06 00:00:00 +00:00
|
|
|
Value *Callee;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Callee))
|
2007-05-02 05:46:45 +00:00
|
|
|
return Error("Invalid INVOKE record");
|
|
|
|
|
2007-05-06 00:00:00 +00:00
|
|
|
const PointerType *CalleeTy = dyn_cast<PointerType>(Callee->getType());
|
|
|
|
const FunctionType *FTy = !CalleeTy ? 0 :
|
2007-05-02 05:46:45 +00:00
|
|
|
dyn_cast<FunctionType>(CalleeTy->getElementType());
|
|
|
|
|
|
|
|
// Check that the right number of fixed parameters are here.
|
2007-05-06 00:00:00 +00:00
|
|
|
if (FTy == 0 || NormalBB == 0 || UnwindBB == 0 ||
|
|
|
|
Record.size() < OpNum+FTy->getNumParams())
|
2007-05-02 05:46:45 +00:00
|
|
|
return Error("Invalid INVOKE record");
|
2007-05-06 00:00:00 +00:00
|
|
|
|
2007-05-02 05:46:45 +00:00
|
|
|
SmallVector<Value*, 16> Ops;
|
2007-05-06 00:00:00 +00:00
|
|
|
for (unsigned i = 0, e = FTy->getNumParams(); i != e; ++i, ++OpNum) {
|
|
|
|
Ops.push_back(getFnValueByID(Record[OpNum], FTy->getParamType(i)));
|
|
|
|
if (Ops.back() == 0) return Error("Invalid INVOKE record");
|
2007-05-02 05:46:45 +00:00
|
|
|
}
|
|
|
|
|
2007-05-06 00:00:00 +00:00
|
|
|
if (!FTy->isVarArg()) {
|
|
|
|
if (Record.size() != OpNum)
|
2007-05-02 05:46:45 +00:00
|
|
|
return Error("Invalid INVOKE record");
|
2007-05-06 00:00:00 +00:00
|
|
|
} else {
|
|
|
|
// Read type/value pairs for varargs params.
|
|
|
|
while (OpNum != Record.size()) {
|
|
|
|
Value *Op;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Op))
|
2007-05-02 05:46:45 +00:00
|
|
|
return Error("Invalid INVOKE record");
|
2007-05-06 00:00:00 +00:00
|
|
|
Ops.push_back(Op);
|
2007-05-02 05:46:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-15 10:04:30 +00:00
|
|
|
I = InvokeInst::Create(Callee, NormalBB, UnwindBB,
|
|
|
|
Ops.begin(), Ops.end());
|
2007-05-03 22:34:03 +00:00
|
|
|
cast<InvokeInst>(I)->setCallingConv(CCInfo);
|
2008-09-25 21:00:45 +00:00
|
|
|
cast<InvokeInst>(I)->setAttributes(PAL);
|
2007-05-02 05:46:45 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-05-02 04:27:25 +00:00
|
|
|
case bitc::FUNC_CODE_INST_UNWIND: // UNWIND
|
|
|
|
I = new UnwindInst();
|
|
|
|
break;
|
|
|
|
case bitc::FUNC_CODE_INST_UNREACHABLE: // UNREACHABLE
|
|
|
|
I = new UnreachableInst();
|
|
|
|
break;
|
2007-05-06 00:21:25 +00:00
|
|
|
case bitc::FUNC_CODE_INST_PHI: { // PHI: [ty, val0,bb0, ...]
|
2007-05-04 19:11:41 +00:00
|
|
|
if (Record.size() < 1 || ((Record.size()-1)&1))
|
2007-05-03 18:58:09 +00:00
|
|
|
return Error("Invalid PHI record");
|
|
|
|
const Type *Ty = getTypeByID(Record[0]);
|
|
|
|
if (!Ty) return Error("Invalid PHI record");
|
|
|
|
|
2008-04-06 20:25:17 +00:00
|
|
|
PHINode *PN = PHINode::Create(Ty);
|
2008-04-13 00:14:42 +00:00
|
|
|
PN->reserveOperandSpace((Record.size()-1)/2);
|
2007-05-03 18:58:09 +00:00
|
|
|
|
2007-05-04 19:11:41 +00:00
|
|
|
for (unsigned i = 0, e = Record.size()-1; i != e; i += 2) {
|
|
|
|
Value *V = getFnValueByID(Record[1+i], Ty);
|
|
|
|
BasicBlock *BB = getBasicBlock(Record[2+i]);
|
2007-05-03 18:58:09 +00:00
|
|
|
if (!V || !BB) return Error("Invalid PHI record");
|
|
|
|
PN->addIncoming(V, BB);
|
|
|
|
}
|
|
|
|
I = PN;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case bitc::FUNC_CODE_INST_MALLOC: { // MALLOC: [instty, op, align]
|
|
|
|
if (Record.size() < 3)
|
|
|
|
return Error("Invalid MALLOC record");
|
|
|
|
const PointerType *Ty =
|
|
|
|
dyn_cast_or_null<PointerType>(getTypeByID(Record[0]));
|
|
|
|
Value *Size = getFnValueByID(Record[1], Type::Int32Ty);
|
|
|
|
unsigned Align = Record[2];
|
|
|
|
if (!Ty || !Size) return Error("Invalid MALLOC record");
|
2009-07-15 23:53:25 +00:00
|
|
|
I = new MallocInst(Ty->getElementType(), Size, (1 << Align) >> 1);
|
2007-05-03 18:58:09 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-05-06 00:21:25 +00:00
|
|
|
case bitc::FUNC_CODE_INST_FREE: { // FREE: [op, opty]
|
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Op;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Op) ||
|
|
|
|
OpNum != Record.size())
|
2007-05-03 18:58:09 +00:00
|
|
|
return Error("Invalid FREE record");
|
|
|
|
I = new FreeInst(Op);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case bitc::FUNC_CODE_INST_ALLOCA: { // ALLOCA: [instty, op, align]
|
|
|
|
if (Record.size() < 3)
|
|
|
|
return Error("Invalid ALLOCA record");
|
|
|
|
const PointerType *Ty =
|
|
|
|
dyn_cast_or_null<PointerType>(getTypeByID(Record[0]));
|
|
|
|
Value *Size = getFnValueByID(Record[1], Type::Int32Ty);
|
|
|
|
unsigned Align = Record[2];
|
|
|
|
if (!Ty || !Size) return Error("Invalid ALLOCA record");
|
2009-07-15 23:53:25 +00:00
|
|
|
I = new AllocaInst(Ty->getElementType(), Size, (1 << Align) >> 1);
|
2007-05-03 18:58:09 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-05-03 22:04:19 +00:00
|
|
|
case bitc::FUNC_CODE_INST_LOAD: { // LOAD: [opty, op, align, vol]
|
2007-05-06 00:00:00 +00:00
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Op;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Op) ||
|
|
|
|
OpNum+2 != Record.size())
|
2007-05-06 00:21:25 +00:00
|
|
|
return Error("Invalid LOAD record");
|
2007-05-06 00:00:00 +00:00
|
|
|
|
|
|
|
I = new LoadInst(Op, "", Record[OpNum+1], (1 << Record[OpNum]) >> 1);
|
2007-05-03 22:04:19 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-12-11 08:59:05 +00:00
|
|
|
case bitc::FUNC_CODE_INST_STORE2: { // STORE2:[ptrty, ptr, val, align, vol]
|
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Val, *Ptr;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Ptr) ||
|
|
|
|
getValue(Record, OpNum,
|
|
|
|
cast<PointerType>(Ptr->getType())->getElementType(), Val) ||
|
|
|
|
OpNum+2 != Record.size())
|
|
|
|
return Error("Invalid STORE record");
|
|
|
|
|
|
|
|
I = new StoreInst(Val, Ptr, Record[OpNum+1], (1 << Record[OpNum]) >> 1);
|
|
|
|
break;
|
|
|
|
}
|
2007-05-06 00:21:25 +00:00
|
|
|
case bitc::FUNC_CODE_INST_STORE: { // STORE:[val, valty, ptr, align, vol]
|
2007-12-11 08:59:05 +00:00
|
|
|
// FIXME: Legacy form of store instruction. Should be removed in LLVM 3.0.
|
2007-05-06 00:21:25 +00:00
|
|
|
unsigned OpNum = 0;
|
|
|
|
Value *Val, *Ptr;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Val) ||
|
2009-07-07 20:18:58 +00:00
|
|
|
getValue(Record, OpNum,
|
|
|
|
Context.getPointerTypeUnqual(Val->getType()), Ptr)||
|
2007-05-06 00:21:25 +00:00
|
|
|
OpNum+2 != Record.size())
|
2007-05-03 22:04:19 +00:00
|
|
|
return Error("Invalid STORE record");
|
2007-05-06 00:21:25 +00:00
|
|
|
|
|
|
|
I = new StoreInst(Val, Ptr, Record[OpNum+1], (1 << Record[OpNum]) >> 1);
|
2007-05-01 05:52:21 +00:00
|
|
|
break;
|
2007-05-03 22:04:19 +00:00
|
|
|
}
|
2007-11-27 13:23:08 +00:00
|
|
|
case bitc::FUNC_CODE_INST_CALL: {
|
|
|
|
// CALL: [paramattrs, cc, fnty, fnid, arg0, arg1...]
|
|
|
|
if (Record.size() < 3)
|
2007-05-03 22:04:19 +00:00
|
|
|
return Error("Invalid CALL record");
|
2007-05-06 00:00:00 +00:00
|
|
|
|
2008-09-25 21:00:45 +00:00
|
|
|
AttrListPtr PAL = getAttributes(Record[0]);
|
2007-05-08 05:38:01 +00:00
|
|
|
unsigned CCInfo = Record[1];
|
|
|
|
|
|
|
|
unsigned OpNum = 2;
|
2007-05-06 00:00:00 +00:00
|
|
|
Value *Callee;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Callee))
|
|
|
|
return Error("Invalid CALL record");
|
|
|
|
|
|
|
|
const PointerType *OpTy = dyn_cast<PointerType>(Callee->getType());
|
2007-05-03 22:04:19 +00:00
|
|
|
const FunctionType *FTy = 0;
|
|
|
|
if (OpTy) FTy = dyn_cast<FunctionType>(OpTy->getElementType());
|
2007-05-06 00:00:00 +00:00
|
|
|
if (!FTy || Record.size() < FTy->getNumParams()+OpNum)
|
2007-05-03 22:04:19 +00:00
|
|
|
return Error("Invalid CALL record");
|
|
|
|
|
|
|
|
SmallVector<Value*, 16> Args;
|
|
|
|
// Read the fixed params.
|
2007-05-06 00:00:00 +00:00
|
|
|
for (unsigned i = 0, e = FTy->getNumParams(); i != e; ++i, ++OpNum) {
|
2007-11-05 21:20:28 +00:00
|
|
|
if (FTy->getParamType(i)->getTypeID()==Type::LabelTyID)
|
|
|
|
Args.push_back(getBasicBlock(Record[OpNum]));
|
|
|
|
else
|
|
|
|
Args.push_back(getFnValueByID(Record[OpNum], FTy->getParamType(i)));
|
2007-05-03 22:04:19 +00:00
|
|
|
if (Args.back() == 0) return Error("Invalid CALL record");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read type/value pairs for varargs params.
|
|
|
|
if (!FTy->isVarArg()) {
|
2007-05-06 00:00:00 +00:00
|
|
|
if (OpNum != Record.size())
|
2007-05-03 22:04:19 +00:00
|
|
|
return Error("Invalid CALL record");
|
|
|
|
} else {
|
2007-05-06 00:00:00 +00:00
|
|
|
while (OpNum != Record.size()) {
|
|
|
|
Value *Op;
|
|
|
|
if (getValueTypePair(Record, OpNum, NextValueNo, Op))
|
|
|
|
return Error("Invalid CALL record");
|
|
|
|
Args.push_back(Op);
|
2007-05-03 22:04:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-04-06 20:25:17 +00:00
|
|
|
I = CallInst::Create(Callee, Args.begin(), Args.end());
|
2007-05-03 22:34:03 +00:00
|
|
|
cast<CallInst>(I)->setCallingConv(CCInfo>>1);
|
|
|
|
cast<CallInst>(I)->setTailCall(CCInfo & 1);
|
2008-09-25 21:00:45 +00:00
|
|
|
cast<CallInst>(I)->setAttributes(PAL);
|
2007-05-03 22:04:19 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case bitc::FUNC_CODE_INST_VAARG: { // VAARG: [valistty, valist, instty]
|
|
|
|
if (Record.size() < 3)
|
|
|
|
return Error("Invalid VAARG record");
|
|
|
|
const Type *OpTy = getTypeByID(Record[0]);
|
|
|
|
Value *Op = getFnValueByID(Record[1], OpTy);
|
|
|
|
const Type *ResTy = getTypeByID(Record[2]);
|
|
|
|
if (!OpTy || !Op || !ResTy)
|
|
|
|
return Error("Invalid VAARG record");
|
|
|
|
I = new VAArgInst(Op, ResTy);
|
|
|
|
break;
|
|
|
|
}
|
2007-05-01 07:01:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add instruction to end of current BB. If there is no current BB, reject
|
|
|
|
// this file.
|
|
|
|
if (CurBB == 0) {
|
|
|
|
delete I;
|
|
|
|
return Error("Invalid instruction with no BB");
|
|
|
|
}
|
|
|
|
CurBB->getInstList().push_back(I);
|
|
|
|
|
|
|
|
// If this was a terminator instruction, move to the next block.
|
|
|
|
if (isa<TerminatorInst>(I)) {
|
|
|
|
++CurBBNo;
|
|
|
|
CurBB = CurBBNo < FunctionBBs.size() ? FunctionBBs[CurBBNo] : 0;
|
2007-05-01 05:52:21 +00:00
|
|
|
}
|
2007-05-01 07:01:57 +00:00
|
|
|
|
|
|
|
// Non-void values get registered in the value table for future use.
|
|
|
|
if (I && I->getType() != Type::VoidTy)
|
|
|
|
ValueList.AssignValue(I, NextValueNo++);
|
2007-05-01 05:52:21 +00:00
|
|
|
}
|
|
|
|
|
2007-05-01 07:01:57 +00:00
|
|
|
// Check the function list for unresolved values.
|
|
|
|
if (Argument *A = dyn_cast<Argument>(ValueList.back())) {
|
|
|
|
if (A->getParent() == 0) {
|
|
|
|
// We found at least one unresolved value. Nuke them all to avoid leaks.
|
|
|
|
for (unsigned i = ModuleValueListSize, e = ValueList.size(); i != e; ++i){
|
|
|
|
if ((A = dyn_cast<Argument>(ValueList.back())) && A->getParent() == 0) {
|
2009-07-07 20:18:58 +00:00
|
|
|
A->replaceAllUsesWith(Context.getUndef(A->getType()));
|
2007-05-01 07:01:57 +00:00
|
|
|
delete A;
|
|
|
|
}
|
|
|
|
}
|
2007-05-04 03:50:29 +00:00
|
|
|
return Error("Never resolved value found in function!");
|
2007-05-01 07:01:57 +00:00
|
|
|
}
|
|
|
|
}
|
2007-05-01 05:52:21 +00:00
|
|
|
|
|
|
|
// Trim the value list down to the size it was before we parsed this function.
|
|
|
|
ValueList.shrinkTo(ModuleValueListSize);
|
|
|
|
std::vector<BasicBlock*>().swap(FunctionBBs);
|
|
|
|
|
2007-05-01 04:59:48 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-05-18 04:02:46 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// ModuleProvider implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
|
|
|
|
bool BitcodeReader::materializeFunction(Function *F, std::string *ErrInfo) {
|
|
|
|
// If it already is material, ignore the request.
|
2007-07-05 17:07:56 +00:00
|
|
|
if (!F->hasNotBeenReadFromBitcode()) return false;
|
2007-05-18 04:02:46 +00:00
|
|
|
|
|
|
|
DenseMap<Function*, std::pair<uint64_t, unsigned> >::iterator DFII =
|
|
|
|
DeferredFunctionInfo.find(F);
|
|
|
|
assert(DFII != DeferredFunctionInfo.end() && "Deferred function not found!");
|
|
|
|
|
|
|
|
// Move the bit stream to the saved position of the deferred function body and
|
|
|
|
// restore the real linkage type for the function.
|
|
|
|
Stream.JumpToBit(DFII->second.first);
|
|
|
|
F->setLinkage((GlobalValue::LinkageTypes)DFII->second.second);
|
|
|
|
|
|
|
|
if (ParseFunctionBody(F)) {
|
|
|
|
if (ErrInfo) *ErrInfo = ErrorString;
|
|
|
|
return true;
|
|
|
|
}
|
2007-08-04 01:51:18 +00:00
|
|
|
|
|
|
|
// Upgrade any old intrinsic calls in the function.
|
|
|
|
for (UpgradedIntrinsicMap::iterator I = UpgradedIntrinsics.begin(),
|
|
|
|
E = UpgradedIntrinsics.end(); I != E; ++I) {
|
|
|
|
if (I->first != I->second) {
|
|
|
|
for (Value::use_iterator UI = I->first->use_begin(),
|
|
|
|
UE = I->first->use_end(); UI != UE; ) {
|
|
|
|
if (CallInst* CI = dyn_cast<CallInst>(*UI++))
|
|
|
|
UpgradeIntrinsicCall(CI, I->second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-05-18 04:02:46 +00:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BitcodeReader::dematerializeFunction(Function *F) {
|
|
|
|
// If this function isn't materialized, or if it is a proto, this is a noop.
|
2007-07-05 17:07:56 +00:00
|
|
|
if (F->hasNotBeenReadFromBitcode() || F->isDeclaration())
|
2007-05-18 04:02:46 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
assert(DeferredFunctionInfo.count(F) && "No info to read function later?");
|
|
|
|
|
|
|
|
// Just forget the function body, we can remat it later.
|
|
|
|
F->deleteBody();
|
|
|
|
F->setLinkage(GlobalValue::GhostLinkage);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Module *BitcodeReader::materializeModule(std::string *ErrInfo) {
|
2009-06-16 05:15:21 +00:00
|
|
|
// Iterate over the module, deserializing any functions that are still on
|
|
|
|
// disk.
|
|
|
|
for (Module::iterator F = TheModule->begin(), E = TheModule->end();
|
|
|
|
F != E; ++F)
|
2007-07-05 17:07:56 +00:00
|
|
|
if (F->hasNotBeenReadFromBitcode() &&
|
2007-05-18 04:02:46 +00:00
|
|
|
materializeFunction(F, ErrInfo))
|
|
|
|
return 0;
|
2007-08-04 01:51:18 +00:00
|
|
|
|
|
|
|
// Upgrade any intrinsic calls that slipped through (should not happen!) and
|
|
|
|
// delete the old functions to clean up. We can't do this unless the entire
|
|
|
|
// module is materialized because there could always be another function body
|
|
|
|
// with calls to the old function.
|
|
|
|
for (std::vector<std::pair<Function*, Function*> >::iterator I =
|
|
|
|
UpgradedIntrinsics.begin(), E = UpgradedIntrinsics.end(); I != E; ++I) {
|
|
|
|
if (I->first != I->second) {
|
|
|
|
for (Value::use_iterator UI = I->first->use_begin(),
|
|
|
|
UE = I->first->use_end(); UI != UE; ) {
|
|
|
|
if (CallInst* CI = dyn_cast<CallInst>(*UI++))
|
|
|
|
UpgradeIntrinsicCall(CI, I->second);
|
|
|
|
}
|
2009-04-01 01:43:03 +00:00
|
|
|
if (!I->first->use_empty())
|
|
|
|
I->first->replaceAllUsesWith(I->second);
|
2007-08-04 01:51:18 +00:00
|
|
|
I->first->eraseFromParent();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
std::vector<std::pair<Function*, Function*> >().swap(UpgradedIntrinsics);
|
|
|
|
|
2007-05-18 04:02:46 +00:00
|
|
|
return TheModule;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// This method is provided by the parent ModuleProvde class and overriden
|
|
|
|
/// here. It simply releases the module from its provided and frees up our
|
|
|
|
/// state.
|
|
|
|
/// @brief Release our hold on the generated module
|
|
|
|
Module *BitcodeReader::releaseModule(std::string *ErrInfo) {
|
|
|
|
// Since we're losing control of this Module, we must hand it back complete
|
|
|
|
Module *M = ModuleProvider::releaseModule(ErrInfo);
|
|
|
|
FreeState();
|
|
|
|
return M;
|
|
|
|
}
|
|
|
|
|
2007-05-01 04:59:48 +00:00
|
|
|
|
2007-04-29 07:54:31 +00:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// External interface
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
/// getBitcodeModuleProvider - lazy function-at-a-time loading from a file.
|
|
|
|
///
|
|
|
|
ModuleProvider *llvm::getBitcodeModuleProvider(MemoryBuffer *Buffer,
|
2009-07-01 23:13:44 +00:00
|
|
|
LLVMContext& Context,
|
2007-04-29 07:54:31 +00:00
|
|
|
std::string *ErrMsg) {
|
2009-07-01 16:58:40 +00:00
|
|
|
BitcodeReader *R = new BitcodeReader(Buffer, Context);
|
2007-04-29 07:54:31 +00:00
|
|
|
if (R->ParseBitcode()) {
|
|
|
|
if (ErrMsg)
|
|
|
|
*ErrMsg = R->getErrorString();
|
|
|
|
|
|
|
|
// Don't let the BitcodeReader dtor delete 'Buffer'.
|
|
|
|
R->releaseMemoryBuffer();
|
|
|
|
delete R;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return R;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// ParseBitcodeFile - Read the specified bitcode file, returning the module.
|
|
|
|
/// If an error occurs, return null and fill in *ErrMsg if non-null.
|
2009-07-01 23:13:44 +00:00
|
|
|
Module *llvm::ParseBitcodeFile(MemoryBuffer *Buffer, LLVMContext& Context,
|
2009-07-01 16:58:40 +00:00
|
|
|
std::string *ErrMsg){
|
2007-04-29 07:54:31 +00:00
|
|
|
BitcodeReader *R;
|
2009-07-01 16:58:40 +00:00
|
|
|
R = static_cast<BitcodeReader*>(getBitcodeModuleProvider(Buffer, Context,
|
|
|
|
ErrMsg));
|
2007-04-29 07:54:31 +00:00
|
|
|
if (!R) return 0;
|
|
|
|
|
2007-05-18 04:02:46 +00:00
|
|
|
// Read in the entire module.
|
|
|
|
Module *M = R->materializeModule(ErrMsg);
|
|
|
|
|
|
|
|
// Don't let the BitcodeReader dtor delete 'Buffer', regardless of whether
|
|
|
|
// there was an error.
|
2007-04-29 07:54:31 +00:00
|
|
|
R->releaseMemoryBuffer();
|
2007-05-18 04:02:46 +00:00
|
|
|
|
|
|
|
// If there was no error, tell ModuleProvider not to delete it when its dtor
|
|
|
|
// is run.
|
|
|
|
if (M)
|
|
|
|
M = R->releaseModule(ErrMsg);
|
2009-06-16 05:15:21 +00:00
|
|
|
|
2007-04-29 07:54:31 +00:00
|
|
|
delete R;
|
|
|
|
return M;
|
|
|
|
}
|