2009-10-11 22:13:54 +00:00
|
|
|
//===--- CGVtable.h - Emit LLVM Code for C++ vtables ----------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This contains code dealing with C++ code generation of virtual tables.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef CLANG_CODEGEN_CGVTABLE_H
|
|
|
|
#define CLANG_CODEGEN_CGVTABLE_H
|
|
|
|
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2009-12-06 22:01:30 +00:00
|
|
|
#include "llvm/ADT/DenseSet.h"
|
2009-12-06 00:23:49 +00:00
|
|
|
#include "llvm/GlobalVariable.h"
|
2009-11-13 17:08:56 +00:00
|
|
|
#include "GlobalDecl.h"
|
2009-10-11 22:13:54 +00:00
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
class CXXRecordDecl;
|
2009-11-26 13:09:03 +00:00
|
|
|
|
2009-10-11 22:13:54 +00:00
|
|
|
namespace CodeGen {
|
|
|
|
class CodeGenModule;
|
2009-11-26 02:32:05 +00:00
|
|
|
|
2009-11-26 13:09:03 +00:00
|
|
|
/// ThunkAdjustment - Virtual and non-virtual adjustment for thunks.
|
|
|
|
class ThunkAdjustment {
|
|
|
|
public:
|
2009-11-26 02:32:05 +00:00
|
|
|
ThunkAdjustment(int64_t NonVirtual, int64_t Virtual)
|
2009-11-26 13:09:03 +00:00
|
|
|
: NonVirtual(NonVirtual),
|
2009-11-26 02:32:05 +00:00
|
|
|
Virtual(Virtual) { }
|
2009-11-26 13:09:03 +00:00
|
|
|
|
2009-11-26 02:32:05 +00:00
|
|
|
ThunkAdjustment()
|
|
|
|
: NonVirtual(0), Virtual(0) { }
|
2009-11-26 13:09:03 +00:00
|
|
|
|
2009-11-26 02:32:05 +00:00
|
|
|
// isEmpty - Return whether this thunk adjustment is empty.
|
2009-11-26 13:09:03 +00:00
|
|
|
bool isEmpty() const {
|
2009-11-26 02:32:05 +00:00
|
|
|
return NonVirtual == 0 && Virtual == 0;
|
|
|
|
}
|
2009-11-26 13:09:03 +00:00
|
|
|
|
2009-11-26 02:32:05 +00:00
|
|
|
/// NonVirtual - The non-virtual adjustment.
|
|
|
|
int64_t NonVirtual;
|
2009-11-26 13:09:03 +00:00
|
|
|
|
2009-11-26 02:32:05 +00:00
|
|
|
/// Virtual - The virtual adjustment.
|
|
|
|
int64_t Virtual;
|
|
|
|
};
|
|
|
|
|
2009-11-26 03:09:37 +00:00
|
|
|
/// CovariantThunkAdjustment - Adjustment of the 'this' pointer and the
|
|
|
|
/// return pointer for covariant thunks.
|
2009-11-26 13:09:03 +00:00
|
|
|
class CovariantThunkAdjustment {
|
|
|
|
public:
|
2009-11-26 03:09:37 +00:00
|
|
|
CovariantThunkAdjustment(const ThunkAdjustment &ThisAdjustment,
|
|
|
|
const ThunkAdjustment &ReturnAdjustment)
|
|
|
|
: ThisAdjustment(ThisAdjustment), ReturnAdjustment(ReturnAdjustment) { }
|
|
|
|
|
|
|
|
CovariantThunkAdjustment() { }
|
|
|
|
|
|
|
|
ThunkAdjustment ThisAdjustment;
|
|
|
|
ThunkAdjustment ReturnAdjustment;
|
|
|
|
};
|
|
|
|
|
2010-01-13 20:11:15 +00:00
|
|
|
// BaseSubobject - Uniquely identifies a direct or indirect base class.
|
|
|
|
// Stores both the base class decl and the offset from the most derived class to
|
|
|
|
// the base class.
|
|
|
|
class BaseSubobject {
|
|
|
|
/// Base - The base class declaration.
|
|
|
|
const CXXRecordDecl *Base;
|
|
|
|
|
|
|
|
/// BaseOffset - The offset from the most derived class to the base class.
|
|
|
|
uint64_t BaseOffset;
|
|
|
|
|
|
|
|
public:
|
|
|
|
BaseSubobject(const CXXRecordDecl *Base, uint64_t BaseOffset)
|
|
|
|
: Base(Base), BaseOffset(BaseOffset) { }
|
|
|
|
|
|
|
|
/// getBase - Returns the base class declaration.
|
|
|
|
const CXXRecordDecl *getBase() const { return Base; }
|
|
|
|
|
|
|
|
/// getBaseOffset - Returns the base class offset.
|
|
|
|
uint64_t getBaseOffset() const { return BaseOffset; }
|
2010-01-14 01:39:42 +00:00
|
|
|
|
2010-01-13 20:11:15 +00:00
|
|
|
friend bool operator==(const BaseSubobject &LHS, const BaseSubobject &RHS) {
|
|
|
|
return LHS.Base == RHS.Base && LHS.BaseOffset == RHS.BaseOffset;
|
|
|
|
}
|
|
|
|
};
|
2010-01-14 01:39:42 +00:00
|
|
|
|
|
|
|
} // end namespace CodeGen
|
|
|
|
} // end namespace clang
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
|
|
|
template<> struct DenseMapInfo<clang::CodeGen::BaseSubobject> {
|
|
|
|
static clang::CodeGen::BaseSubobject getEmptyKey() {
|
|
|
|
return clang::CodeGen::BaseSubobject(
|
|
|
|
DenseMapInfo<const clang::CXXRecordDecl *>::getEmptyKey(),
|
|
|
|
DenseMapInfo<uint64_t>::getEmptyKey());
|
|
|
|
}
|
|
|
|
|
|
|
|
static clang::CodeGen::BaseSubobject getTombstoneKey() {
|
|
|
|
return clang::CodeGen::BaseSubobject(
|
|
|
|
DenseMapInfo<const clang::CXXRecordDecl *>::getTombstoneKey(),
|
|
|
|
DenseMapInfo<uint64_t>::getTombstoneKey());
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned getHashValue(const clang::CodeGen::BaseSubobject &Base) {
|
|
|
|
return
|
|
|
|
DenseMapInfo<const clang::CXXRecordDecl *>::getHashValue(Base.getBase()) ^
|
|
|
|
DenseMapInfo<uint64_t>::getHashValue(Base.getBaseOffset());
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool isEqual(const clang::CodeGen::BaseSubobject &LHS,
|
|
|
|
const clang::CodeGen::BaseSubobject &RHS) {
|
|
|
|
return LHS == RHS;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-01-14 02:29:07 +00:00
|
|
|
// It's OK to treat BaseSubobject as a POD type.
|
|
|
|
template <> struct isPodLike<clang::CodeGen::BaseSubobject> {
|
|
|
|
static const bool value = true;
|
|
|
|
};
|
|
|
|
|
2010-01-14 01:39:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace CodeGen {
|
|
|
|
|
2010-03-23 04:11:45 +00:00
|
|
|
class CodeGenVTables {
|
2009-12-07 23:56:34 +00:00
|
|
|
public:
|
|
|
|
typedef std::vector<std::pair<GlobalDecl, ThunkAdjustment> >
|
|
|
|
AdjustmentVectorTy;
|
|
|
|
|
2010-01-02 18:02:32 +00:00
|
|
|
typedef std::pair<const CXXRecordDecl *, uint64_t> CtorVtable_t;
|
|
|
|
typedef llvm::DenseMap<CtorVtable_t, int64_t> AddrSubMap_t;
|
|
|
|
typedef llvm::DenseMap<const CXXRecordDecl *, AddrSubMap_t *> AddrMap_t;
|
|
|
|
llvm::DenseMap<const CXXRecordDecl *, AddrMap_t*> AddressPoints;
|
|
|
|
|
2010-01-14 02:29:07 +00:00
|
|
|
typedef llvm::DenseMap<BaseSubobject, uint64_t> AddressPointsMapTy;
|
|
|
|
|
2009-12-07 23:56:34 +00:00
|
|
|
private:
|
2009-10-11 22:13:54 +00:00
|
|
|
CodeGenModule &CGM;
|
2009-11-26 13:09:03 +00:00
|
|
|
|
2009-10-11 22:13:54 +00:00
|
|
|
/// MethodVtableIndices - Contains the index (relative to the vtable address
|
|
|
|
/// point) where the function pointer for a virtual function is stored.
|
2009-11-13 17:08:56 +00:00
|
|
|
typedef llvm::DenseMap<GlobalDecl, int64_t> MethodVtableIndicesTy;
|
2009-10-11 22:13:54 +00:00
|
|
|
MethodVtableIndicesTy MethodVtableIndices;
|
2009-11-26 13:09:03 +00:00
|
|
|
|
2009-10-11 22:13:54 +00:00
|
|
|
typedef std::pair<const CXXRecordDecl *,
|
|
|
|
const CXXRecordDecl *> ClassPairTy;
|
2009-11-26 13:09:03 +00:00
|
|
|
|
2010-03-11 07:15:17 +00:00
|
|
|
/// VirtualBaseClassOffsetOffsets - Contains the vtable offset (relative to
|
|
|
|
/// the address point) in bytes where the offsets for virtual bases of a class
|
|
|
|
/// are stored.
|
|
|
|
typedef llvm::DenseMap<ClassPairTy, int64_t>
|
|
|
|
VirtualBaseClassOffsetOffsetsMapTy;
|
|
|
|
VirtualBaseClassOffsetOffsetsMapTy VirtualBaseClassOffsetOffsets;
|
2009-11-10 07:44:33 +00:00
|
|
|
|
2009-12-06 00:01:05 +00:00
|
|
|
/// Vtables - All the vtables which have been defined.
|
|
|
|
llvm::DenseMap<const CXXRecordDecl *, llvm::GlobalVariable *> Vtables;
|
2009-11-27 20:47:55 +00:00
|
|
|
|
|
|
|
/// NumVirtualFunctionPointers - Contains the number of virtual function
|
|
|
|
/// pointers in the vtable for a given record decl.
|
|
|
|
llvm::DenseMap<const CXXRecordDecl *, uint64_t> NumVirtualFunctionPointers;
|
|
|
|
|
2009-12-07 23:56:34 +00:00
|
|
|
typedef llvm::DenseMap<GlobalDecl, AdjustmentVectorTy> SavedAdjustmentsTy;
|
|
|
|
SavedAdjustmentsTy SavedAdjustments;
|
|
|
|
llvm::DenseSet<const CXXRecordDecl*> SavedAdjustmentRecords;
|
2009-12-06 22:01:30 +00:00
|
|
|
|
2010-01-02 01:01:18 +00:00
|
|
|
typedef llvm::DenseMap<ClassPairTy, uint64_t> SubVTTIndiciesTy;
|
|
|
|
SubVTTIndiciesTy SubVTTIndicies;
|
|
|
|
|
2009-11-27 20:47:55 +00:00
|
|
|
/// getNumVirtualFunctionPointers - Return the number of virtual function
|
|
|
|
/// pointers in the vtable for a given record decl.
|
|
|
|
uint64_t getNumVirtualFunctionPointers(const CXXRecordDecl *RD);
|
|
|
|
|
|
|
|
void ComputeMethodVtableIndices(const CXXRecordDecl *RD);
|
2010-03-10 02:19:29 +00:00
|
|
|
|
2009-12-06 00:23:49 +00:00
|
|
|
llvm::GlobalVariable *
|
|
|
|
GenerateVtable(llvm::GlobalVariable::LinkageTypes Linkage,
|
2009-12-06 00:53:22 +00:00
|
|
|
bool GenerateDefinition, const CXXRecordDecl *LayoutClass,
|
2010-02-28 00:36:23 +00:00
|
|
|
const CXXRecordDecl *RD, uint64_t Offset, bool IsVirtual,
|
2010-01-14 02:29:07 +00:00
|
|
|
AddressPointsMapTy& AddressPoints);
|
2009-12-06 01:09:21 +00:00
|
|
|
|
|
|
|
llvm::GlobalVariable *GenerateVTT(llvm::GlobalVariable::LinkageTypes Linkage,
|
2010-01-02 01:01:18 +00:00
|
|
|
bool GenerateDefinition,
|
2009-12-06 01:09:21 +00:00
|
|
|
const CXXRecordDecl *RD);
|
|
|
|
|
2010-03-23 04:59:02 +00:00
|
|
|
/// EmitThunks - Emit the associated thunks for the given global decl.
|
|
|
|
void EmitThunks(GlobalDecl GD);
|
|
|
|
|
2009-10-11 22:13:54 +00:00
|
|
|
public:
|
2010-03-23 04:11:45 +00:00
|
|
|
CodeGenVTables(CodeGenModule &CGM)
|
2009-10-11 22:13:54 +00:00
|
|
|
: CGM(CGM) { }
|
|
|
|
|
2010-01-02 01:01:18 +00:00
|
|
|
/// needsVTTParameter - Return whether the given global decl needs a VTT
|
|
|
|
/// parameter, which it does if it's a base constructor or destructor with
|
|
|
|
/// virtual bases.
|
|
|
|
static bool needsVTTParameter(GlobalDecl GD);
|
|
|
|
|
|
|
|
/// getSubVTTIndex - Return the index of the sub-VTT for the base class of the
|
|
|
|
/// given record decl.
|
|
|
|
uint64_t getSubVTTIndex(const CXXRecordDecl *RD, const CXXRecordDecl *Base);
|
|
|
|
|
2009-10-11 22:13:54 +00:00
|
|
|
/// getMethodVtableIndex - Return the index (relative to the vtable address
|
2009-11-26 13:09:03 +00:00
|
|
|
/// point) where the function pointer for the given virtual function is
|
2009-10-11 22:13:54 +00:00
|
|
|
/// stored.
|
2009-11-27 20:47:55 +00:00
|
|
|
uint64_t getMethodVtableIndex(GlobalDecl GD);
|
2009-11-26 13:09:03 +00:00
|
|
|
|
2010-03-11 07:15:17 +00:00
|
|
|
/// getVirtualBaseOffsetOffset - Return the offset in bytes (relative to the
|
|
|
|
/// vtable address point) where the offset of the virtual base that contains
|
|
|
|
/// the given base is stored, otherwise, if no virtual base contains the given
|
2009-10-13 22:54:56 +00:00
|
|
|
/// class, return 0. Base must be a virtual base class or an unambigious
|
|
|
|
/// base.
|
2010-03-11 07:15:17 +00:00
|
|
|
int64_t getVirtualBaseOffsetOffset(const CXXRecordDecl *RD,
|
|
|
|
const CXXRecordDecl *VBase);
|
2009-11-10 07:44:33 +00:00
|
|
|
|
2009-12-07 23:56:34 +00:00
|
|
|
AdjustmentVectorTy *getAdjustments(GlobalDecl GD);
|
2009-12-06 22:01:30 +00:00
|
|
|
|
2009-12-06 00:01:05 +00:00
|
|
|
llvm::GlobalVariable *getVtable(const CXXRecordDecl *RD);
|
2010-01-14 02:29:07 +00:00
|
|
|
|
|
|
|
/// CtorVtableInfo - Information about a constructor vtable.
|
|
|
|
struct CtorVtableInfo {
|
|
|
|
/// Vtable - The vtable itself.
|
|
|
|
llvm::GlobalVariable *Vtable;
|
|
|
|
|
|
|
|
/// AddressPoints - The address points in this constructor vtable.
|
|
|
|
AddressPointsMapTy AddressPoints;
|
|
|
|
|
|
|
|
CtorVtableInfo() : Vtable(0) { }
|
|
|
|
};
|
|
|
|
|
|
|
|
CtorVtableInfo getCtorVtable(const CXXRecordDecl *RD,
|
2010-02-28 00:36:23 +00:00
|
|
|
const BaseSubobject &Base,
|
|
|
|
bool BaseIsVirtual);
|
2009-11-30 23:41:22 +00:00
|
|
|
|
2010-01-02 01:01:18 +00:00
|
|
|
llvm::GlobalVariable *getVTT(const CXXRecordDecl *RD);
|
2009-11-30 23:41:22 +00:00
|
|
|
|
2010-03-23 04:15:00 +00:00
|
|
|
// EmitVTableRelatedData - Will emit any thunks that the global decl might
|
|
|
|
// have, as well as the vtable itself if the global decl is the key function.
|
|
|
|
void EmitVTableRelatedData(GlobalDecl GD);
|
2010-03-10 02:19:29 +00:00
|
|
|
|
|
|
|
/// GenerateClassData - Generate all the class data requires to be generated
|
|
|
|
/// upon definition of a KeyFunction. This includes the vtable, the
|
|
|
|
/// rtti data structure and the VTT.
|
|
|
|
///
|
|
|
|
/// \param Linkage - The desired linkage of the vtable, the RTTI and the VTT.
|
|
|
|
void GenerateClassData(llvm::GlobalVariable::LinkageTypes Linkage,
|
|
|
|
const CXXRecordDecl *RD);
|
2009-10-11 22:13:54 +00:00
|
|
|
};
|
2009-11-26 13:09:03 +00:00
|
|
|
|
2010-01-14 02:29:07 +00:00
|
|
|
} // end namespace CodeGen
|
|
|
|
} // end namespace clang
|
2009-10-11 22:13:54 +00:00
|
|
|
#endif
|