blob: 9d759a3e6448308462bfb675ae34a2ed255ad0fc [file] [log] [blame]
Duncan Sands264d2e72012-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 Carruthed0881b2012-12-03 16:50:05 +000014#include "llvm-c/TargetMachine.h"
Duncan Sands264d2e72012-04-11 10:25:24 +000015#include "llvm-c/Core.h"
16#include "llvm-c/Target.h"
Chandler Carruth93dcdc42015-01-31 11:17:59 +000017#include "llvm/Analysis/TargetTransformInfo.h"
Chandler Carruth9fb823b2013-01-02 11:36:10 +000018#include "llvm/IR/DataLayout.h"
19#include "llvm/IR/Module.h"
Duncan Sands264d2e72012-04-11 10:25:24 +000020#include "llvm/PassManager.h"
Chandler Carruthed0881b2012-12-03 16:50:05 +000021#include "llvm/Support/CodeGen.h"
Benjamin Kramerd59664f2014-04-29 23:26:49 +000022#include "llvm/Support/FileSystem.h"
Chandler Carruthed0881b2012-12-03 16:50:05 +000023#include "llvm/Support/FormattedStream.h"
Chandler Carruth8a8cd2b2014-01-07 11:48:04 +000024#include "llvm/Support/Host.h"
Chandler Carruthed0881b2012-12-03 16:50:05 +000025#include "llvm/Support/TargetRegistry.h"
26#include "llvm/Support/raw_ostream.h"
27#include "llvm/Target/TargetMachine.h"
Eric Christopherd9134482014-08-04 21:25:23 +000028#include "llvm/Target/TargetSubtargetInfo.h"
Duncan Sands264d2e72012-04-11 10:25:24 +000029#include <cassert>
30#include <cstdlib>
31#include <cstring>
32
33using namespace llvm;
34
Eric Christopher04d4e932013-04-22 22:47:22 +000035inline TargetMachine *unwrap(LLVMTargetMachineRef P) {
36 return reinterpret_cast<TargetMachine*>(P);
37}
38inline Target *unwrap(LLVMTargetRef P) {
39 return reinterpret_cast<Target*>(P);
40}
41inline LLVMTargetMachineRef wrap(const TargetMachine *P) {
42 return
43 reinterpret_cast<LLVMTargetMachineRef>(const_cast<TargetMachine*>(P));
44}
45inline LLVMTargetRef wrap(const Target * P) {
46 return reinterpret_cast<LLVMTargetRef>(const_cast<Target*>(P));
47}
Duncan Sands264d2e72012-04-11 10:25:24 +000048
49LLVMTargetRef LLVMGetFirstTarget() {
Anders Waldenborga89c1e32013-10-17 10:25:24 +000050 if(TargetRegistry::begin() == TargetRegistry::end()) {
Craig Topper062a2ba2014-04-25 05:30:21 +000051 return nullptr;
Anders Waldenborga89c1e32013-10-17 10:25:24 +000052 }
53
54 const Target* target = &*TargetRegistry::begin();
55 return wrap(target);
Duncan Sands264d2e72012-04-11 10:25:24 +000056}
57LLVMTargetRef LLVMGetNextTarget(LLVMTargetRef T) {
58 return wrap(unwrap(T)->getNext());
59}
60
Peter Zotovb2c8b8a2013-11-15 02:51:01 +000061LLVMTargetRef LLVMGetTargetFromName(const char *Name) {
Peter Zotov5a6cfda2013-11-15 02:51:18 +000062 StringRef NameRef = Name;
Peter Zotov7b61b752013-11-06 10:25:18 +000063 for (TargetRegistry::iterator IT = TargetRegistry::begin(),
64 IE = TargetRegistry::end(); IT != IE; ++IT) {
Peter Zotov5a6cfda2013-11-15 02:51:18 +000065 if (IT->getName() == NameRef)
Peter Zotovb2c8b8a2013-11-15 02:51:01 +000066 return wrap(&*IT);
Peter Zotov7b61b752013-11-06 10:25:18 +000067 }
68
Craig Topper062a2ba2014-04-25 05:30:21 +000069 return nullptr;
Peter Zotov7b61b752013-11-06 10:25:18 +000070}
71
72LLVMBool LLVMGetTargetFromTriple(const char* TripleStr, LLVMTargetRef *T,
73 char **ErrorMessage) {
74 std::string Error;
75
76 *T = wrap(TargetRegistry::lookupTarget(TripleStr, Error));
77
78 if (!*T) {
79 if (ErrorMessage)
80 *ErrorMessage = strdup(Error.c_str());
81
82 return 1;
83 }
84
85 return 0;
86}
87
Duncan Sands264d2e72012-04-11 10:25:24 +000088const char * LLVMGetTargetName(LLVMTargetRef T) {
89 return unwrap(T)->getName();
90}
91
92const char * LLVMGetTargetDescription(LLVMTargetRef T) {
93 return unwrap(T)->getShortDescription();
94}
95
96LLVMBool LLVMTargetHasJIT(LLVMTargetRef T) {
97 return unwrap(T)->hasJIT();
98}
99
100LLVMBool LLVMTargetHasTargetMachine(LLVMTargetRef T) {
101 return unwrap(T)->hasTargetMachine();
102}
103
104LLVMBool LLVMTargetHasAsmBackend(LLVMTargetRef T) {
105 return unwrap(T)->hasMCAsmBackend();
106}
107
Peter Zotov0e38fc82013-11-15 02:51:12 +0000108LLVMTargetMachineRef LLVMCreateTargetMachine(LLVMTargetRef T,
109 const char* Triple, const char* CPU, const char* Features,
110 LLVMCodeGenOptLevel Level, LLVMRelocMode Reloc,
111 LLVMCodeModel CodeModel) {
Duncan Sands264d2e72012-04-11 10:25:24 +0000112 Reloc::Model RM;
113 switch (Reloc){
114 case LLVMRelocStatic:
115 RM = Reloc::Static;
116 break;
117 case LLVMRelocPIC:
118 RM = Reloc::PIC_;
119 break;
120 case LLVMRelocDynamicNoPic:
121 RM = Reloc::DynamicNoPIC;
122 break;
123 default:
124 RM = Reloc::Default;
125 break;
126 }
127
Filip Pizlo85e0d272013-05-01 22:58:00 +0000128 CodeModel::Model CM = unwrap(CodeModel);
Duncan Sands264d2e72012-04-11 10:25:24 +0000129
Filip Pizlo85e0d272013-05-01 22:58:00 +0000130 CodeGenOpt::Level OL;
Duncan Sands264d2e72012-04-11 10:25:24 +0000131 switch (Level) {
132 case LLVMCodeGenLevelNone:
133 OL = CodeGenOpt::None;
134 break;
135 case LLVMCodeGenLevelLess:
136 OL = CodeGenOpt::Less;
137 break;
138 case LLVMCodeGenLevelAggressive:
139 OL = CodeGenOpt::Aggressive;
140 break;
141 default:
142 OL = CodeGenOpt::Default;
143 break;
144 }
145
146 TargetOptions opt;
147 return wrap(unwrap(T)->createTargetMachine(Triple, CPU, Features, opt, RM,
148 CM, OL));
149}
150
151
152void LLVMDisposeTargetMachine(LLVMTargetMachineRef T) {
153 delete unwrap(T);
154}
155
156LLVMTargetRef LLVMGetTargetMachineTarget(LLVMTargetMachineRef T) {
157 const Target* target = &(unwrap(T)->getTarget());
158 return wrap(target);
159}
160
161char* LLVMGetTargetMachineTriple(LLVMTargetMachineRef T) {
162 std::string StringRep = unwrap(T)->getTargetTriple();
163 return strdup(StringRep.c_str());
164}
165
166char* LLVMGetTargetMachineCPU(LLVMTargetMachineRef T) {
167 std::string StringRep = unwrap(T)->getTargetCPU();
168 return strdup(StringRep.c_str());
169}
170
171char* LLVMGetTargetMachineFeatureString(LLVMTargetMachineRef T) {
172 std::string StringRep = unwrap(T)->getTargetFeatureString();
173 return strdup(StringRep.c_str());
174}
175
176LLVMTargetDataRef LLVMGetTargetMachineData(LLVMTargetMachineRef T) {
Eric Christopher8b770652015-01-26 19:03:15 +0000177 return wrap(unwrap(T)->getDataLayout());
Duncan Sands264d2e72012-04-11 10:25:24 +0000178}
179
Peter Zotov7b61b752013-11-06 10:25:18 +0000180void LLVMSetTargetMachineAsmVerbosity(LLVMTargetMachineRef T,
181 LLVMBool VerboseAsm) {
182 unwrap(T)->setAsmVerbosityDefault(VerboseAsm);
183}
184
Tom Stellardec924c52013-04-16 23:12:56 +0000185static LLVMBool LLVMTargetMachineEmit(LLVMTargetMachineRef T, LLVMModuleRef M,
186 formatted_raw_ostream &OS, LLVMCodeGenFileType codegen, char **ErrorMessage) {
Duncan Sands264d2e72012-04-11 10:25:24 +0000187 TargetMachine* TM = unwrap(T);
188 Module* Mod = unwrap(M);
189
190 PassManager pass;
191
192 std::string error;
193
Eric Christopher8b770652015-01-26 19:03:15 +0000194 const DataLayout *td = TM->getDataLayout();
Duncan Sands264d2e72012-04-11 10:25:24 +0000195
196 if (!td) {
Micah Villmowcdfe20b2012-10-08 16:38:25 +0000197 error = "No DataLayout in TargetMachine";
Duncan Sands264d2e72012-04-11 10:25:24 +0000198 *ErrorMessage = strdup(error.c_str());
199 return true;
200 }
Rafael Espindola339430f2014-02-25 23:25:17 +0000201 Mod->setDataLayout(td);
Rafael Espindolac435adc2014-09-10 21:27:43 +0000202 pass.add(new DataLayoutPass());
Duncan Sands264d2e72012-04-11 10:25:24 +0000203
204 TargetMachine::CodeGenFileType ft;
205 switch (codegen) {
206 case LLVMAssemblyFile:
207 ft = TargetMachine::CGFT_AssemblyFile;
208 break;
209 default:
210 ft = TargetMachine::CGFT_ObjectFile;
211 break;
212 }
Tom Stellardec924c52013-04-16 23:12:56 +0000213 if (TM->addPassesToEmitFile(pass, OS, ft)) {
Nick Lewycky7b287ee2013-03-10 22:01:44 +0000214 error = "TargetMachine can't emit a file of this type";
Duncan Sands264d2e72012-04-11 10:25:24 +0000215 *ErrorMessage = strdup(error.c_str());
216 return true;
217 }
218
219 pass.run(*Mod);
220
Tom Stellardec924c52013-04-16 23:12:56 +0000221 OS.flush();
Duncan Sands264d2e72012-04-11 10:25:24 +0000222 return false;
223}
Tom Stellardec924c52013-04-16 23:12:56 +0000224
225LLVMBool LLVMTargetMachineEmitToFile(LLVMTargetMachineRef T, LLVMModuleRef M,
226 char* Filename, LLVMCodeGenFileType codegen, char** ErrorMessage) {
Rafael Espindola3fd1e992014-08-25 18:16:47 +0000227 std::error_code EC;
228 raw_fd_ostream dest(Filename, EC, sys::fs::F_None);
229 if (EC) {
230 *ErrorMessage = strdup(EC.message().c_str());
Tom Stellardec924c52013-04-16 23:12:56 +0000231 return true;
232 }
Anders Waldenborg39f5d7d2013-10-17 10:39:35 +0000233 formatted_raw_ostream destf(dest);
Tom Stellardec924c52013-04-16 23:12:56 +0000234 bool Result = LLVMTargetMachineEmit(T, M, destf, codegen, ErrorMessage);
235 dest.flush();
236 return Result;
237}
238
239LLVMBool LLVMTargetMachineEmitToMemoryBuffer(LLVMTargetMachineRef T,
240 LLVMModuleRef M, LLVMCodeGenFileType codegen, char** ErrorMessage,
241 LLVMMemoryBufferRef *OutMemBuf) {
Alp Tokere69170a2014-06-26 22:52:05 +0000242 std::string CodeString;
243 raw_string_ostream OStream(CodeString);
244 formatted_raw_ostream Out(OStream);
Tom Stellardec924c52013-04-16 23:12:56 +0000245 bool Result = LLVMTargetMachineEmit(T, M, Out, codegen, ErrorMessage);
Alp Tokere69170a2014-06-26 22:52:05 +0000246 OStream.flush();
Tom Stellardec924c52013-04-16 23:12:56 +0000247
Alp Tokere69170a2014-06-26 22:52:05 +0000248 std::string &Data = OStream.str();
249 *OutMemBuf = LLVMCreateMemoryBufferWithMemoryRangeCopy(Data.c_str(),
250 Data.length(), "");
Tom Stellardec924c52013-04-16 23:12:56 +0000251 return Result;
252}
Peter Zotov7b61b752013-11-06 10:25:18 +0000253
254char *LLVMGetDefaultTargetTriple(void) {
255 return strdup(sys::getDefaultTargetTriple().c_str());
256}
Juergen Ributzka5fe955c2014-01-23 19:23:28 +0000257
258void LLVMAddAnalysisPasses(LLVMTargetMachineRef T, LLVMPassManagerRef PM) {
Chandler Carruth93dcdc42015-01-31 11:17:59 +0000259 unwrap(PM)->add(createTargetTransformInfoWrapperPass(unwrap(T)->getTTI()));
Juergen Ributzka5fe955c2014-01-23 19:23:28 +0000260}