1 /*===-- llvm_ocaml.h - LLVM Ocaml Glue --------------------------*- C++ -*-===*\
3 |* The LLVM Compiler Infrastructure *|
5 |* This file was developed by Gordon Henriksen and is distributed under the *|
6 |* University of Illinois Open Source License. See LICENSE.TXT for details. *|
8 |*===----------------------------------------------------------------------===*|
10 |* This file glues LLVM's ocaml interface to its C interface. These functions *|
11 |* are by and large transparent wrappers to the corresponding C functions. *|
13 |* Note that these functions intentionally take liberties with the CAMLparamX *|
14 |* macros, since most of the parameters are not GC heap objects. *|
16 \*===----------------------------------------------------------------------===*/
18 #include "llvm-c/Core.h"
19 #include "caml/alloc.h"
20 #include "caml/mlvalues.h"
21 #include "caml/memory.h"
25 /*===-- Modules -----------------------------------------------------------===*/
27 /* string -> llmodule */
28 CAMLprim value llvm_create_module(value ModuleID) {
29 return (value) LLVMModuleCreateWithName(String_val(ModuleID));
32 /* llmodule -> unit */
33 CAMLprim value llvm_dispose_module(value M) {
34 LLVMDisposeModule((LLVMModuleRef) M);
38 /* string -> lltype -> llmodule -> bool */
39 CAMLprim value llvm_add_type_name(value Name, value Ty, value M) {
40 int res = LLVMAddTypeName((LLVMModuleRef) M,
41 String_val(Name), (LLVMTypeRef) Ty);
42 return Val_bool(res == 0);
46 /*===-- Types -------------------------------------------------------------===*/
48 /* lltype -> type_kind */
49 CAMLprim value llvm_classify_type(value Ty) {
50 return Val_int(LLVMGetTypeKind((LLVMTypeRef) Ty));
53 /* lltype -> lltype -> unit */
54 CAMLprim value llvm_refine_abstract_type(value ConcreteTy, value AbstractTy) {
55 LLVMRefineAbstractType((LLVMTypeRef) AbstractTy, (LLVMTypeRef) ConcreteTy);
56 return (value) Val_unit;
59 /*--... Operations on integer types ........................................--*/
62 CAMLprim value llvm_i1_type (value Unit) { return (value) LLVMInt1Type(); }
63 CAMLprim value llvm_i8_type (value Unit) { return (value) LLVMInt8Type(); }
64 CAMLprim value llvm_i16_type(value Unit) { return (value) LLVMInt16Type(); }
65 CAMLprim value llvm_i32_type(value Unit) { return (value) LLVMInt32Type(); }
66 CAMLprim value llvm_i64_type(value Unit) { return (value) LLVMInt64Type(); }
69 CAMLprim value llvm_make_integer_type(value Width) {
70 return (value) LLVMCreateIntegerType(Int_val(Width));
74 CAMLprim value llvm_integer_bitwidth(value IntegerTy) {
75 return Val_int(LLVMGetIntegerTypeWidth((LLVMTypeRef) IntegerTy));
78 /*--... Operations on real types ...........................................--*/
81 CAMLprim value llvm_float_type(value Unit) {
82 return (value) LLVMFloatType();
86 CAMLprim value llvm_double_type(value Unit) {
87 return (value) LLVMDoubleType();
91 CAMLprim value llvm_x86fp80_type(value Unit) {
92 return (value) LLVMX86FP80Type();
96 CAMLprim value llvm_fp128_type(value Unit) {
97 return (value) LLVMFP128Type();
101 CAMLprim value llvm_ppc_fp128_type(value Unit) {
102 return (value) LLVMPPCFP128Type();
105 /*--... Operations on function types .......................................--*/
107 /* lltype -> lltype array -> bool -> lltype */
108 CAMLprim value llvm_make_function_type(value RetTy, value ParamTys,
110 return (value) LLVMCreateFunctionType((LLVMTypeRef) RetTy,
111 (LLVMTypeRef *) ParamTys,
112 Wosize_val(ParamTys),
117 CAMLprim value llvm_is_var_arg(value FunTy) {
118 return Val_bool(LLVMIsFunctionVarArg((LLVMTypeRef) FunTy));
121 /* lltype -> lltype */
122 CAMLprim value llvm_return_type(value FunTy) {
123 return (value) LLVMGetFunctionReturnType((LLVMTypeRef) FunTy);
126 /* lltype -> lltype array */
127 CAMLprim value llvm_param_types(value FunTy) {
128 unsigned Count = LLVMGetFunctionParamCount((LLVMTypeRef) FunTy);
129 LLVMTypeRef *FunTys = alloca(Count * sizeof(LLVMTypeRef));
131 /* copy into an ocaml array */
133 value ParamTys = caml_alloc(Count, 0);
135 LLVMGetFunctionParamTypes((LLVMTypeRef) FunTy, FunTys);
136 for (i = 0; i != Count; ++i)
137 Store_field(ParamTys, i, (value) FunTys[i]);
142 /*--... Operations on struct types .........................................--*/
144 /* lltype array -> bool -> lltype */
145 CAMLprim value llvm_make_struct_type(value ElementTypes, value Packed) {
146 return (value) LLVMCreateStructType((LLVMTypeRef *) ElementTypes,
147 Wosize_val(ElementTypes),
151 /* lltype -> lltype array */
152 CAMLprim value llvm_element_types(value StructTy) {
153 unsigned Count = LLVMGetStructElementCount((LLVMTypeRef) StructTy);
154 LLVMTypeRef *Tys = alloca(Count * sizeof(LLVMTypeRef));
156 /* copy into an ocaml array */
158 value ElementTys = caml_alloc(Count, 0);
160 LLVMGetStructElementTypes((LLVMTypeRef) StructTy, Tys);
161 for (i = 0; i != Count; ++i)
162 Store_field(ElementTys, i, (value) Tys[i]);
167 CAMLprim value llvm_is_packed(value StructTy) {
168 return Val_bool(LLVMIsPackedStruct((LLVMTypeRef) StructTy));
171 /*--... Operations on array, pointer, and vector types .....................--*/
173 /* lltype -> int -> lltype */
174 CAMLprim value llvm_make_array_type(value ElementTy, value Count) {
175 return (value) LLVMCreateArrayType((LLVMTypeRef) ElementTy, Int_val(Count));
178 /* lltype -> lltype */
179 CAMLprim value llvm_make_pointer_type(value ElementTy) {
180 return (value) LLVMCreatePointerType((LLVMTypeRef) ElementTy);
183 /* lltype -> int -> lltype */
184 CAMLprim value llvm_make_vector_type(value ElementTy, value Count) {
185 return (value) LLVMCreateVectorType((LLVMTypeRef) ElementTy, Int_val(Count));
188 /* lltype -> lltype */
189 CAMLprim value llvm_element_type(value Ty) {
190 return (value) LLVMGetElementType((LLVMTypeRef) Ty);
194 CAMLprim value llvm_array_length(value ArrayTy) {
195 return Val_int(LLVMGetArrayLength((LLVMTypeRef) ArrayTy));
199 CAMLprim value llvm_vector_size(value VectorTy) {
200 return Val_int(LLVMGetVectorSize((LLVMTypeRef) VectorTy));
203 /*--... Operations on other types ..........................................--*/
206 CAMLprim value llvm_void_type (value Unit) { return (value) LLVMVoidType(); }
207 CAMLprim value llvm_label_type(value Unit) { return (value) LLVMLabelType(); }
210 CAMLprim value llvm_make_opaque_type(value Unit) {
211 return (value) LLVMCreateOpaqueType();
215 /*===-- VALUES ------------------------------------------------------------===*/
217 /* llvalue -> lltype */
218 CAMLprim value llvm_type_of(value Val) {
219 return (value) LLVMGetTypeOfValue((LLVMValueRef) Val);
222 /* llvalue -> string */
223 CAMLprim value llvm_value_name(value Val) {
224 return caml_copy_string(LLVMGetValueName((LLVMValueRef) Val));
227 /* string -> llvalue -> unit */
228 CAMLprim value llvm_set_value_name(value Name, value Val) {
229 LLVMSetValueName((LLVMValueRef) Val, String_val(Name));
233 /*--... Operations on constants of (mostly) any type .......................--*/
235 /* lltype -> llvalue */
236 CAMLprim value llvm_make_null(value Ty) {
237 return (value) LLVMGetNull((LLVMTypeRef) Ty);
240 /* lltype -> llvalue */
241 CAMLprim value llvm_make_all_ones(value Ty) {
242 return (value) LLVMGetAllOnes((LLVMTypeRef) Ty);
245 /* lltype -> llvalue */
246 CAMLprim value llvm_make_undef(value Ty) {
247 return (value) LLVMGetUndef((LLVMTypeRef) Ty);
250 /* llvalue -> bool */
251 CAMLprim value llvm_is_null(value Val) {
252 return Val_bool(LLVMIsNull((LLVMValueRef) Val));
255 /*--... Operations on scalar constants .....................................--*/
257 /* lltype -> int -> bool -> llvalue */
258 CAMLprim value llvm_make_int_constant(value IntTy, value N, value SExt) {
259 /* GCC warns if we use the ternary operator. */
260 unsigned long long N2;
262 N2 = (intnat) Int_val(N);
264 N2 = (uintnat) Int_val(N);
266 return (value) LLVMGetIntConstant((LLVMTypeRef) IntTy, N2, Bool_val(SExt));
269 /* lltype -> float -> llvalue */
270 CAMLprim value llvm_make_real_constant(value RealTy, value N) {
271 return (value) LLVMGetRealConstant((LLVMTypeRef) RealTy, Double_val(N));
274 /*--... Operations on composite constants ..................................--*/
276 /* string -> bool -> llvalue */
277 CAMLprim value llvm_make_string_constant(value Str, value NullTerminate) {
278 return (value) LLVMGetStringConstant(String_val(Str),
280 Bool_val(NullTerminate) == 0);
283 /* lltype -> llvalue array -> llvalue */
284 CAMLprim value llvm_make_array_constant(value ElementTy, value ElementVals) {
285 return (value) LLVMGetArrayConstant((LLVMTypeRef) ElementTy,
286 (LLVMValueRef*) Op_val(ElementVals),
287 Wosize_val(ElementVals));
290 /* llvalue array -> bool -> llvalue */
291 CAMLprim value llvm_make_struct_constant(value ElementVals, value Packed) {
292 return (value) LLVMGetStructConstant((LLVMValueRef*) Op_val(ElementVals),
293 Wosize_val(ElementVals),
297 /* llvalue array -> llvalue */
298 CAMLprim value llvm_make_vector_constant(value ElementVals) {
299 return (value) LLVMGetVectorConstant((LLVMValueRef*) Op_val(ElementVals),
300 Wosize_val(ElementVals));
303 /*--... Operations on global variables, functions, and aliases (globals) ...--*/
305 /* llvalue -> bool */
306 CAMLprim value llvm_is_declaration(value Global) {
307 return Val_bool(LLVMIsDeclaration((LLVMValueRef) Global));
310 /* llvalue -> linkage */
311 CAMLprim value llvm_linkage(value Global) {
312 return Val_int(LLVMGetLinkage((LLVMValueRef) Global));
315 /* linkage -> llvalue -> unit */
316 CAMLprim value llvm_set_linkage(value Linkage, value Global) {
317 LLVMSetLinkage((LLVMValueRef) Global, Int_val(Linkage));
321 /* llvalue -> string */
322 CAMLprim value llvm_section(value Global) {
323 return caml_copy_string(LLVMGetSection((LLVMValueRef) Global));
326 /* string -> llvalue -> unit */
327 CAMLprim value llvm_set_section(value Section, value Global) {
328 LLVMSetSection((LLVMValueRef) Global, String_val(Section));
332 /* llvalue -> visibility */
333 CAMLprim value llvm_visibility(value Global) {
334 return Val_int(LLVMGetVisibility((LLVMValueRef) Global));
337 /* visibility -> llvalue -> unit */
338 CAMLprim value llvm_set_visibility(value Viz, value Global) {
339 LLVMSetVisibility((LLVMValueRef) Global, Int_val(Viz));
344 CAMLprim value llvm_alignment(value Global) {
345 return Val_int(LLVMGetAlignment((LLVMValueRef) Global));
348 /* int -> llvalue -> unit */
349 CAMLprim value llvm_set_alignment(value Bytes, value Global) {
350 LLVMSetAlignment((LLVMValueRef) Global, Int_val(Bytes));
354 /*--... Operations on global variables .....................................--*/
356 /* lltype -> string -> llmodule -> llvalue */
357 CAMLprim value llvm_add_global(value Ty, value Name, value M) {
358 return (value) LLVMAddGlobal((LLVMModuleRef) M,
359 (LLVMTypeRef) Ty, String_val(Name));
362 /* lltype -> string -> llmodule -> llvalue */
363 CAMLprim value llvm_declare_global(value Ty, value Name, value M) {
364 return (value) LLVMAddGlobal((LLVMModuleRef) M,
365 (LLVMTypeRef) Ty, String_val(Name));
368 /* string -> llvalue -> llmodule -> llvalue */
369 CAMLprim value llvm_define_global(value Name, value ConstantVal, value M) {
370 LLVMValueRef Initializer = (LLVMValueRef) ConstantVal;
371 LLVMValueRef GlobalVar = LLVMAddGlobal((LLVMModuleRef) M,
372 LLVMGetTypeOfValue(Initializer),
374 LLVMSetInitializer(GlobalVar, Initializer);
375 return (value) GlobalVar;
378 /* llvalue -> unit */
379 CAMLprim value llvm_delete_global(value GlobalVar) {
380 LLVMDeleteGlobal((LLVMValueRef) GlobalVar);
384 /* llvalue -> llvalue */
385 CAMLprim value llvm_global_initializer(value GlobalVar) {
386 return (value) LLVMGetInitializer((LLVMValueRef) GlobalVar);
389 /* llvalue -> llvalue -> unit */
390 CAMLprim value llvm_set_initializer(value ConstantVal, value GlobalVar) {
391 LLVMSetInitializer((LLVMValueRef) GlobalVar, (LLVMValueRef) ConstantVal);
395 /* llvalue -> unit */
396 CAMLprim value llvm_remove_initializer(value GlobalVar) {
397 LLVMSetInitializer((LLVMValueRef) GlobalVar, NULL);
401 /* llvalue -> bool */
402 CAMLprim value llvm_is_thread_local(value GlobalVar) {
403 return Val_bool(LLVMIsThreadLocal((LLVMValueRef) GlobalVar));
406 /* bool -> llvalue -> unit */
407 CAMLprim value llvm_set_thread_local(value IsThreadLocal, value GlobalVar) {
408 LLVMSetThreadLocal((LLVMValueRef) GlobalVar, Bool_val(IsThreadLocal));