blob: 7c644fcfcaa810e54d6358f3b5f56cac6b1a2a38 [file] [log] [blame]
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +00001//===-- TargetMachine.cpp -------------------------------------------------===//
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 file implements the LLVM-C part of TargetMachine.h
11//
12//===----------------------------------------------------------------------===//
13
Chandler Carruthd04a8d42012-12-03 16:50:05 +000014#include "llvm-c/TargetMachine.h"
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +000015#include "llvm-c/Core.h"
16#include "llvm-c/Target.h"
Chandler Carruth0b8c9a82013-01-02 11:36:10 +000017#include "llvm/IR/DataLayout.h"
18#include "llvm/IR/Module.h"
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +000019#include "llvm/PassManager.h"
Eric Christopher3e397312013-04-22 22:47:22 +000020#include "llvm/Wrap.h"
Chandler Carruthd04a8d42012-12-03 16:50:05 +000021#include "llvm/Support/CodeGen.h"
22#include "llvm/Support/FormattedStream.h"
23#include "llvm/Support/TargetRegistry.h"
24#include "llvm/Support/raw_ostream.h"
25#include "llvm/Target/TargetMachine.h"
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +000026#include <cassert>
27#include <cstdlib>
28#include <cstring>
29
30using namespace llvm;
31
Eric Christopher3e397312013-04-22 22:47:22 +000032inline DataLayout *unwrap(LLVMTargetDataRef P) {
33 return reinterpret_cast<DataLayout*>(P);
34}
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +000035
Eric Christopher3e397312013-04-22 22:47:22 +000036inline LLVMTargetDataRef wrap(const DataLayout *P) {
37 return reinterpret_cast<LLVMTargetDataRef>(const_cast<DataLayout*>(P));
38}
39
40inline TargetLibraryInfo *unwrap(LLVMTargetLibraryInfoRef P) {
41 return reinterpret_cast<TargetLibraryInfo*>(P);
42}
43
44inline LLVMTargetLibraryInfoRef wrap(const TargetLibraryInfo *P) {
45 TargetLibraryInfo *X = const_cast<TargetLibraryInfo*>(P);
46 return reinterpret_cast<LLVMTargetLibraryInfoRef>(X);
47}
48
49inline TargetMachine *unwrap(LLVMTargetMachineRef P) {
50 return reinterpret_cast<TargetMachine*>(P);
51}
52inline Target *unwrap(LLVMTargetRef P) {
53 return reinterpret_cast<Target*>(P);
54}
55inline LLVMTargetMachineRef wrap(const TargetMachine *P) {
56 return
57 reinterpret_cast<LLVMTargetMachineRef>(const_cast<TargetMachine*>(P));
58}
59inline LLVMTargetRef wrap(const Target * P) {
60 return reinterpret_cast<LLVMTargetRef>(const_cast<Target*>(P));
61}
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +000062
63LLVMTargetRef LLVMGetFirstTarget() {
64 const Target* target = &*TargetRegistry::begin();
65 return wrap(target);
66}
67LLVMTargetRef LLVMGetNextTarget(LLVMTargetRef T) {
68 return wrap(unwrap(T)->getNext());
69}
70
71const char * LLVMGetTargetName(LLVMTargetRef T) {
72 return unwrap(T)->getName();
73}
74
75const char * LLVMGetTargetDescription(LLVMTargetRef T) {
76 return unwrap(T)->getShortDescription();
77}
78
79LLVMBool LLVMTargetHasJIT(LLVMTargetRef T) {
80 return unwrap(T)->hasJIT();
81}
82
83LLVMBool LLVMTargetHasTargetMachine(LLVMTargetRef T) {
84 return unwrap(T)->hasTargetMachine();
85}
86
87LLVMBool LLVMTargetHasAsmBackend(LLVMTargetRef T) {
88 return unwrap(T)->hasMCAsmBackend();
89}
90
91LLVMTargetMachineRef LLVMCreateTargetMachine(LLVMTargetRef T, char* Triple,
92 char* CPU, char* Features, LLVMCodeGenOptLevel Level, LLVMRelocMode Reloc,
93 LLVMCodeModel CodeModel) {
94 Reloc::Model RM;
95 switch (Reloc){
96 case LLVMRelocStatic:
97 RM = Reloc::Static;
98 break;
99 case LLVMRelocPIC:
100 RM = Reloc::PIC_;
101 break;
102 case LLVMRelocDynamicNoPic:
103 RM = Reloc::DynamicNoPIC;
104 break;
105 default:
106 RM = Reloc::Default;
107 break;
108 }
109
110 CodeModel::Model CM;
111 switch (CodeModel) {
112 case LLVMCodeModelJITDefault:
113 CM = CodeModel::JITDefault;
114 break;
115 case LLVMCodeModelSmall:
116 CM = CodeModel::Small;
117 break;
118 case LLVMCodeModelKernel:
119 CM = CodeModel::Kernel;
120 break;
121 case LLVMCodeModelMedium:
122 CM = CodeModel::Medium;
123 break;
124 case LLVMCodeModelLarge:
125 CM = CodeModel::Large;
126 break;
127 default:
128 CM = CodeModel::Default;
129 break;
130 }
131 CodeGenOpt::Level OL;
132
133 switch (Level) {
134 case LLVMCodeGenLevelNone:
135 OL = CodeGenOpt::None;
136 break;
137 case LLVMCodeGenLevelLess:
138 OL = CodeGenOpt::Less;
139 break;
140 case LLVMCodeGenLevelAggressive:
141 OL = CodeGenOpt::Aggressive;
142 break;
143 default:
144 OL = CodeGenOpt::Default;
145 break;
146 }
147
148 TargetOptions opt;
149 return wrap(unwrap(T)->createTargetMachine(Triple, CPU, Features, opt, RM,
150 CM, OL));
151}
152
153
154void LLVMDisposeTargetMachine(LLVMTargetMachineRef T) {
155 delete unwrap(T);
156}
157
158LLVMTargetRef LLVMGetTargetMachineTarget(LLVMTargetMachineRef T) {
159 const Target* target = &(unwrap(T)->getTarget());
160 return wrap(target);
161}
162
163char* LLVMGetTargetMachineTriple(LLVMTargetMachineRef T) {
164 std::string StringRep = unwrap(T)->getTargetTriple();
165 return strdup(StringRep.c_str());
166}
167
168char* LLVMGetTargetMachineCPU(LLVMTargetMachineRef T) {
169 std::string StringRep = unwrap(T)->getTargetCPU();
170 return strdup(StringRep.c_str());
171}
172
173char* LLVMGetTargetMachineFeatureString(LLVMTargetMachineRef T) {
174 std::string StringRep = unwrap(T)->getTargetFeatureString();
175 return strdup(StringRep.c_str());
176}
177
178LLVMTargetDataRef LLVMGetTargetMachineData(LLVMTargetMachineRef T) {
Micah Villmow3574eca2012-10-08 16:38:25 +0000179 return wrap(unwrap(T)->getDataLayout());
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +0000180}
181
Tom Stellardad74f332013-04-16 23:12:56 +0000182static LLVMBool LLVMTargetMachineEmit(LLVMTargetMachineRef T, LLVMModuleRef M,
183 formatted_raw_ostream &OS, LLVMCodeGenFileType codegen, char **ErrorMessage) {
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +0000184 TargetMachine* TM = unwrap(T);
185 Module* Mod = unwrap(M);
186
187 PassManager pass;
188
189 std::string error;
190
Micah Villmow3574eca2012-10-08 16:38:25 +0000191 const DataLayout* td = TM->getDataLayout();
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +0000192
193 if (!td) {
Micah Villmow3574eca2012-10-08 16:38:25 +0000194 error = "No DataLayout in TargetMachine";
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +0000195 *ErrorMessage = strdup(error.c_str());
196 return true;
197 }
Micah Villmow3574eca2012-10-08 16:38:25 +0000198 pass.add(new DataLayout(*td));
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +0000199
200 TargetMachine::CodeGenFileType ft;
201 switch (codegen) {
202 case LLVMAssemblyFile:
203 ft = TargetMachine::CGFT_AssemblyFile;
204 break;
205 default:
206 ft = TargetMachine::CGFT_ObjectFile;
207 break;
208 }
Tom Stellardad74f332013-04-16 23:12:56 +0000209 if (TM->addPassesToEmitFile(pass, OS, ft)) {
Nick Lewyckyf3a64c02013-03-10 22:01:44 +0000210 error = "TargetMachine can't emit a file of this type";
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +0000211 *ErrorMessage = strdup(error.c_str());
212 return true;
213 }
214
215 pass.run(*Mod);
216
Tom Stellardad74f332013-04-16 23:12:56 +0000217 OS.flush();
Duncan Sandsd6b7b8f2012-04-11 10:25:24 +0000218 return false;
219}
Tom Stellardad74f332013-04-16 23:12:56 +0000220
221LLVMBool LLVMTargetMachineEmitToFile(LLVMTargetMachineRef T, LLVMModuleRef M,
222 char* Filename, LLVMCodeGenFileType codegen, char** ErrorMessage) {
223 std::string error;
224 raw_fd_ostream dest(Filename, error, raw_fd_ostream::F_Binary);
225 formatted_raw_ostream destf(dest);
226 if (!error.empty()) {
227 *ErrorMessage = strdup(error.c_str());
228 return true;
229 }
230 bool Result = LLVMTargetMachineEmit(T, M, destf, codegen, ErrorMessage);
231 dest.flush();
232 return Result;
233}
234
235LLVMBool LLVMTargetMachineEmitToMemoryBuffer(LLVMTargetMachineRef T,
236 LLVMModuleRef M, LLVMCodeGenFileType codegen, char** ErrorMessage,
237 LLVMMemoryBufferRef *OutMemBuf) {
238 std::string CodeString;
239 raw_string_ostream OStream(CodeString);
240 formatted_raw_ostream Out(OStream);
241 bool Result = LLVMTargetMachineEmit(T, M, Out, codegen, ErrorMessage);
242 OStream.flush();
243
244 std::string &Data = OStream.str();
245 *OutMemBuf = LLVMCreateMemoryBufferWithMemoryRangeCopy(Data.c_str(),
246 Data.length(), "");
247 return Result;
248}