blob: 8484e2d7daba7610e312fd9c15afabdbf6e6c32b [file] [log] [blame]
Anders Carlsson6ce51fd2009-10-10 20:49:04 +00001//===--- CGCXXRtti.cpp - Emit LLVM Code for C++ RTTI descriptors ----------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9//
10// This contains code dealing with C++ code generation of RTTI descriptors.
11//
12//===----------------------------------------------------------------------===//
13
14#include "CodeGenModule.h"
15using namespace clang;
16using namespace CodeGen;
17
Mike Stump14718422009-11-14 14:25:18 +000018class RttiBuilder {
19 CodeGenModule &CGM; // Per-module state.
20 llvm::LLVMContext &VMContext;
21 const llvm::Type *Int8PtrTy;
22public:
23 RttiBuilder(CodeGenModule &cgm)
24 : CGM(cgm), VMContext(cgm.getModule().getContext()),
25 Int8PtrTy(llvm::Type::getInt8PtrTy(VMContext)) { }
26
27 llvm::Constant *BuildName(const CXXRecordDecl *RD) {
28 llvm::SmallString<256> OutName;
29 llvm::raw_svector_ostream Out(OutName);
30 mangleCXXRttiName(CGM.getMangleContext(),
31 CGM.getContext().getTagDeclType(RD), Out);
32
33 llvm::GlobalVariable::LinkageTypes linktype;
34 linktype = llvm::GlobalValue::LinkOnceODRLinkage;
35
36 llvm::Constant *C;
37 C = llvm::ConstantArray::get(VMContext, Out.str().substr(4));
38
39 llvm::Constant *s = new llvm::GlobalVariable(CGM.getModule(), C->getType(),
40 true, linktype, C,
41 Out.str());
42 s = llvm::ConstantExpr::getBitCast(s, Int8PtrTy);
43 return s;
44 };
45};
46
Anders Carlsson6ce51fd2009-10-10 20:49:04 +000047llvm::Constant *CodeGenModule::GenerateRtti(const CXXRecordDecl *RD) {
Mike Stump14718422009-11-14 14:25:18 +000048 RttiBuilder b(*this);
49
Anders Carlsson43004632009-10-30 01:26:12 +000050 const llvm::Type *Int8PtrTy = llvm::Type::getInt8PtrTy(VMContext);
Anders Carlsson6ce51fd2009-10-10 20:49:04 +000051
52 if (!getContext().getLangOptions().Rtti)
Anders Carlsson43004632009-10-30 01:26:12 +000053 return llvm::Constant::getNullValue(Int8PtrTy);
Anders Carlsson6ce51fd2009-10-10 20:49:04 +000054
55 llvm::SmallString<256> OutName;
56 llvm::raw_svector_ostream Out(OutName);
Anders Carlsson958c9f82009-10-30 01:52:02 +000057 mangleCXXRtti(getMangleContext(), Context.getTagDeclType(RD), Out);
Anders Carlsson2295f312009-10-11 21:24:51 +000058
Anders Carlsson6ce51fd2009-10-10 20:49:04 +000059 llvm::GlobalVariable::LinkageTypes linktype;
Mike Stump14718422009-11-14 14:25:18 +000060 linktype = llvm::GlobalValue::LinkOnceODRLinkage;
Anders Carlsson6ce51fd2009-10-10 20:49:04 +000061 std::vector<llvm::Constant *> info;
62 // assert(0 && "FIXME: implement rtti descriptor");
63 // FIXME: descriptor
Anders Carlsson43004632009-10-30 01:26:12 +000064 info.push_back(llvm::Constant::getNullValue(Int8PtrTy));
Mike Stump14718422009-11-14 14:25:18 +000065 info.push_back(b.BuildName(RD));
66
67 // FIXME: rest of rtti bits
Anders Carlsson6ce51fd2009-10-10 20:49:04 +000068
69 llvm::Constant *C;
Anders Carlsson43004632009-10-30 01:26:12 +000070 llvm::ArrayType *type = llvm::ArrayType::get(Int8PtrTy, info.size());
Anders Carlsson6ce51fd2009-10-10 20:49:04 +000071 C = llvm::ConstantArray::get(type, info);
Anders Carlsson43004632009-10-30 01:26:12 +000072 llvm::Constant *Rtti =
73 new llvm::GlobalVariable(getModule(), type, true, linktype, C,
74 Out.str());
75 Rtti = llvm::ConstantExpr::getBitCast(Rtti, Int8PtrTy);
Anders Carlsson6ce51fd2009-10-10 20:49:04 +000076 return Rtti;
77}