2007-10-24 19:06:40 +00:00
|
|
|
//=- Deserialize.h - Generic Object Deserialization from Bitcode --*- C++ -*-=//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file was developed by Ted Kremenek and is distributed under the
|
|
|
|
// University of Illinois Open Source License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file defines the interface for generic object deserialization from
|
|
|
|
// LLVM bitcode.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLVM_BITCODE_SERIALIZE_INPUT
|
|
|
|
#define LLVM_BITCODE_SERIALIZE_INPUT
|
|
|
|
|
|
|
|
#include "llvm/Bitcode/BitstreamReader.h"
|
|
|
|
#include "llvm/Bitcode/Serialization.h"
|
2007-10-25 00:10:21 +00:00
|
|
|
#include "llvm/ADT/DenseMap.h"
|
|
|
|
#include "llvm/ADT/SmallVector.h"
|
|
|
|
#include "llvm/Support/Allocator.h"
|
2007-10-25 23:40:35 +00:00
|
|
|
#include "llvm/Support/DataTypes.h"
|
2007-10-24 19:06:40 +00:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
|
|
|
class Deserializer {
|
2007-10-28 21:17:59 +00:00
|
|
|
|
|
|
|
//===----------------------------------------------------------===//
|
|
|
|
// Internal type definitions.
|
|
|
|
//===----------------------------------------------------------===//
|
2007-10-25 00:10:21 +00:00
|
|
|
|
2007-10-28 21:17:59 +00:00
|
|
|
struct BPNode {
|
|
|
|
BPNode* Next;
|
2007-10-25 23:40:35 +00:00
|
|
|
uintptr_t& PtrRef;
|
2007-11-01 00:57:37 +00:00
|
|
|
|
2007-10-28 21:17:59 +00:00
|
|
|
BPNode(BPNode* n, uintptr_t& pref)
|
|
|
|
: Next(n), PtrRef(pref) {
|
2007-10-25 23:40:35 +00:00
|
|
|
PtrRef = 0;
|
|
|
|
}
|
2007-10-25 00:10:21 +00:00
|
|
|
};
|
|
|
|
|
2007-11-01 00:57:37 +00:00
|
|
|
struct BPEntry {
|
|
|
|
union { BPNode* Head; void* Ptr; };
|
|
|
|
|
|
|
|
BPEntry() : Head(NULL) {}
|
|
|
|
|
|
|
|
static inline bool isPod() { return true; }
|
|
|
|
|
|
|
|
void SetPtr(BPNode*& FreeList, void* P);
|
|
|
|
};
|
|
|
|
|
|
|
|
class BPKey {
|
|
|
|
unsigned Raw;
|
|
|
|
|
2007-10-28 21:17:59 +00:00
|
|
|
public:
|
2007-11-01 00:57:37 +00:00
|
|
|
BPKey(unsigned PtrId) : Raw(PtrId << 1) { assert (PtrId > 0); }
|
2007-11-05 18:13:03 +00:00
|
|
|
BPKey(unsigned code, unsigned) : Raw(code) {}
|
2007-10-28 21:17:59 +00:00
|
|
|
|
2007-11-01 00:57:37 +00:00
|
|
|
void MarkFinal() { Raw |= 0x1; }
|
|
|
|
bool hasFinalPtr() const { return Raw & 0x1 ? true : false; }
|
|
|
|
unsigned getID() const { return Raw >> 1; }
|
2007-10-28 21:17:59 +00:00
|
|
|
|
2007-11-05 18:13:03 +00:00
|
|
|
static inline BPKey getEmptyKey() { return BPKey(0,0); }
|
|
|
|
static inline BPKey getTombstoneKey() { return BPKey(1,0); }
|
2007-11-01 00:57:37 +00:00
|
|
|
static inline unsigned getHashValue(const BPKey& K) { return K.Raw & ~0x1; }
|
|
|
|
|
|
|
|
static bool isEqual(const BPKey& K1, const BPKey& K2) {
|
|
|
|
return (K1.Raw ^ K2.Raw) & ~0x1 ? false : true;
|
2007-10-28 21:17:59 +00:00
|
|
|
}
|
|
|
|
|
2007-11-01 00:57:37 +00:00
|
|
|
static bool isPod() { return true; }
|
2007-10-28 21:17:59 +00:00
|
|
|
};
|
2007-11-01 00:57:37 +00:00
|
|
|
|
|
|
|
typedef llvm::DenseMap<BPKey,BPEntry,BPKey,BPEntry> MapTy;
|
2007-10-25 00:10:21 +00:00
|
|
|
|
2007-10-28 21:17:59 +00:00
|
|
|
//===----------------------------------------------------------===//
|
|
|
|
// Internal data members.
|
|
|
|
//===----------------------------------------------------------===//
|
|
|
|
|
|
|
|
BitstreamReader& Stream;
|
|
|
|
SmallVector<uint64_t,10> Record;
|
|
|
|
unsigned RecIdx;
|
|
|
|
BumpPtrAllocator Allocator;
|
|
|
|
BPNode* FreeList;
|
2007-10-25 00:10:21 +00:00
|
|
|
MapTy BPatchMap;
|
|
|
|
|
2007-10-28 21:17:59 +00:00
|
|
|
//===----------------------------------------------------------===//
|
|
|
|
// Public Interface.
|
|
|
|
//===----------------------------------------------------------===//
|
|
|
|
|
2007-10-24 19:06:40 +00:00
|
|
|
public:
|
|
|
|
Deserializer(BitstreamReader& stream);
|
|
|
|
~Deserializer();
|
2007-10-25 00:10:21 +00:00
|
|
|
|
|
|
|
uint64_t ReadInt();
|
|
|
|
bool ReadBool() {
|
|
|
|
return ReadInt() ? true : false;
|
|
|
|
}
|
|
|
|
|
2007-10-24 19:06:40 +00:00
|
|
|
template <typename T>
|
|
|
|
inline T& Read(T& X) {
|
|
|
|
SerializeTrait<T>::Read(*this,X);
|
|
|
|
return X;
|
|
|
|
}
|
2007-10-25 00:10:21 +00:00
|
|
|
|
2007-10-24 19:06:40 +00:00
|
|
|
template <typename T>
|
|
|
|
inline T* Materialize() {
|
|
|
|
return SerializeTrait<T>::Materialize(*this);
|
|
|
|
}
|
|
|
|
|
|
|
|
char* ReadCStr(char* cstr = NULL, unsigned MaxLen=0, bool isNullTerm=true);
|
|
|
|
void ReadCStr(std::vector<char>& buff, bool isNullTerm=false);
|
2007-10-25 00:10:21 +00:00
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
inline T* ReadOwnedPtr() {
|
|
|
|
unsigned PtrId = ReadInt();
|
2007-10-25 18:42:52 +00:00
|
|
|
|
|
|
|
if (PtrId == 0)
|
|
|
|
return NULL;
|
|
|
|
|
2007-10-25 00:10:21 +00:00
|
|
|
T* x = SerializeTrait<T>::Materialize(*this);
|
|
|
|
RegisterPtr(PtrId,x);
|
|
|
|
return x;
|
|
|
|
}
|
2007-10-24 19:06:40 +00:00
|
|
|
|
2007-11-02 18:04:20 +00:00
|
|
|
template <typename T>
|
|
|
|
inline void ReadOwnedPtr(T*& Ptr) {
|
|
|
|
Ptr = ReadOwnedPtr<T>();
|
|
|
|
}
|
|
|
|
|
2007-10-28 21:17:59 +00:00
|
|
|
template <typename T>
|
2007-10-31 19:58:32 +00:00
|
|
|
void ReadPtr(T*& PtrRef) {
|
|
|
|
ReadUIntPtr(reinterpret_cast<uintptr_t&>(PtrRef));
|
|
|
|
}
|
2007-10-28 21:17:59 +00:00
|
|
|
|
2007-10-31 19:58:32 +00:00
|
|
|
template <typename T>
|
|
|
|
void ReadPtr(const T*& PtrRef) {
|
|
|
|
ReadPtr(const_cast<T*&>(PtrRef));
|
|
|
|
}
|
|
|
|
|
2007-10-28 21:17:59 +00:00
|
|
|
void ReadUIntPtr(uintptr_t& PtrRef);
|
2007-10-26 20:23:27 +00:00
|
|
|
|
2007-10-31 19:58:32 +00:00
|
|
|
template <typename T>
|
|
|
|
T& ReadRef() {
|
|
|
|
T* p = reinterpret_cast<T*>(ReadInternalRefPtr());
|
|
|
|
return *p;
|
|
|
|
}
|
|
|
|
|
2007-10-31 22:42:03 +00:00
|
|
|
void RegisterPtr(unsigned PtrId, const void* Ptr);
|
2007-10-31 19:58:32 +00:00
|
|
|
|
2007-10-31 22:42:03 +00:00
|
|
|
void RegisterPtr(const void* Ptr) {
|
|
|
|
RegisterPtr(ReadInt(),Ptr);
|
|
|
|
}
|
2007-11-05 21:36:35 +00:00
|
|
|
|
|
|
|
bool AtEnd();
|
2007-10-25 00:10:21 +00:00
|
|
|
|
2007-10-24 19:06:40 +00:00
|
|
|
private:
|
2007-10-25 00:10:21 +00:00
|
|
|
void ReadRecord();
|
|
|
|
bool inRecord();
|
2007-10-31 19:58:32 +00:00
|
|
|
uintptr_t ReadInternalRefPtr();
|
2007-11-01 00:57:37 +00:00
|
|
|
|
|
|
|
static inline bool HasFinalPtr(MapTy::value_type& V) {
|
|
|
|
return V.first.hasFinalPtr();
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uintptr_t GetFinalPtr(MapTy::value_type& V) {
|
|
|
|
return reinterpret_cast<uintptr_t>(V.second.Ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline BPNode* GetBPNode(MapTy::value_type& V) {
|
|
|
|
return V.second.Head;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void SetBPNode(MapTy::value_type& V, BPNode* N) {
|
|
|
|
V.second.Head = N;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SetPtr(MapTy::value_type& V, const void* P) {
|
|
|
|
V.first.MarkFinal();
|
|
|
|
V.second.SetPtr(FreeList,const_cast<void*>(P));
|
|
|
|
}
|
2007-10-24 19:06:40 +00:00
|
|
|
};
|
2007-10-25 00:10:21 +00:00
|
|
|
|
2007-10-24 19:06:40 +00:00
|
|
|
} // end namespace llvm
|
|
|
|
|
|
|
|
#endif
|