C ABI rework

This commit is contained in:
David Gonzalez Martin 2025-03-03 20:12:08 -06:00
parent 39444e4a61
commit 8e7d81d8ff
19 changed files with 4165 additions and 1323 deletions

View File

@ -78,9 +78,11 @@ const LLVM = struct {
else => "HOME",
};
const home_path = env.get(home_env) orelse unreachable;
const is_ci = std.mem.eql(u8, (env.get("BB_CI") orelse "0"), "1");
const download_dir = try std.mem.concat(b.allocator, u8, &.{ home_path, "/Downloads" });
std.fs.makeDirAbsolute(download_dir) catch {};
const llvm_base = try std.mem.concat(b.allocator, u8, &.{ "llvm-", @tagName(target.result.cpu.arch), "-", @tagName(target.result.os.tag), "-", @tagName(CmakeBuildType.from_zig_build_type(optimize)) });
const cmake_build_type = if (is_ci) CmakeBuildType.from_zig_build_type(optimize) else CmakeBuildType.Release;
const llvm_base = try std.mem.concat(b.allocator, u8, &.{ "llvm-", @tagName(target.result.cpu.arch), "-", @tagName(target.result.os.tag), "-", @tagName(cmake_build_type) });
const base = try std.mem.concat(b.allocator, u8, &.{ download_dir, "/", llvm_base });
const full_path = try std.mem.concat(b.allocator, u8, &.{ base, "/bin/llvm-config" });
@ -306,7 +308,7 @@ pub fn build(b: *std.Build) !void {
env = try std.process.getEnvMap(b.allocator);
target = b.standardTargetOptions(.{});
optimize = b.standardOptimizeOption(.{});
system_llvm = b.option(bool, "system_llvm", "Link against system LLVM libraries") orelse true;
system_llvm = b.option(bool, "system_llvm", "Link against system LLVM libraries") orelse false;
const path = env.get("PATH") orelse unreachable;
const c_abi_module = b.createModule(.{
@ -322,7 +324,7 @@ pub fn build(b: *std.Build) !void {
.optimize = optimize,
});
c_abi.addCSourceFiles(.{
.files = &.{"src/c_abi.c"},
.files = &.{"tests/c_abi.c"},
.flags = &.{"-g"},
});
@ -363,8 +365,8 @@ pub fn build(b: *std.Build) !void {
const exe_unit_tests = b.addTest(.{
.root_module = exe_mod,
.link_libc = true,
});
exe_unit_tests.linkLibC();
llvm.link(exe);

View File

@ -33,6 +33,28 @@ pub const Intrinsic = enum {
};
pub const Attribute = opaque {
pub const List = opaque {
pub const Options = extern struct {
function: Attribute.Function,
@"return": Attribute.Argument,
argument_pointer: [*]const Attribute.Argument,
argument_length: u64,
comptime {
assert(@sizeOf(Options) == @sizeOf(Attribute.Function) + @sizeOf(Attribute.Argument) + @sizeOf([*]const Attribute.Argument) + @sizeOf(u64));
}
};
pub fn build(context: *Context, function_attributes: Attribute.Function, return_attributes: Attribute.Argument, argument_attributes: []const Attribute.Argument, call_site: bool) *Attribute.List {
return api.llvm_attribute_list_build(context, &Options{
.function = function_attributes,
.@"return" = return_attributes,
.argument_pointer = argument_attributes.ptr,
.argument_length = argument_attributes.len,
}, call_site);
}
};
pub const Index = enum(c_uint) {
@"return" = 0,
function = 0xffff_ffff,
@ -42,6 +64,172 @@ pub const Attribute = opaque {
pub const Kind = enum(c_uint) {
_,
};
pub const FramePointerKind = enum(u2) {
none = 0,
reserved = 1,
non_leaf = 2,
all = 3,
};
pub const ZeroCallUsedRegsKind = enum(u4) {
all = 0,
skip = 1 << 0,
used_gpr_arg = only_used | only_gpr | only_arg,
used_gpr = only_used | only_gpr,
used_arg = only_used | only_arg,
used = only_used,
all_gpr_arg = only_gpr | only_arg,
all_gpr = only_gpr,
all_arg = only_arg,
const only_used = 1 << 1;
const only_gpr = 1 << 2;
const only_arg = 1 << 3;
};
pub const FPClassTest = packed struct(u10) {
s_nan: bool = false,
q_nan: bool = false,
neg_inf: bool = false,
neg_normal: bool = false,
neg_subnormal: bool = false,
neg_zero: bool = false,
pos_zero: bool = false,
pos_subnormal: bool = false,
pos_normal: bool = false,
pos_inf: bool = false,
};
pub const UnwindTableKind = enum(u2) {
none = 0,
sync = 1,
@"async" = 2,
pub const default = UnwindTableKind.@"async";
};
pub const Argument = extern struct {
semantic_type: *Type,
abi_type: *Type,
dereferenceable_bytes: u64,
alignment: u32,
flags: packed struct(u32) {
no_alias: bool,
non_null: bool,
no_undef: bool,
sign_extend: bool,
zero_extend: bool,
in_reg: bool,
no_fp_class: FPClassTest,
struct_return: bool,
writable: bool,
dead_on_unwind: bool,
in_alloca: bool,
dereferenceable: bool,
dereferenceable_or_null: bool,
nest: bool,
by_value: bool,
by_reference: bool,
no_capture: bool,
_: u6 = 0,
},
comptime {
assert(@sizeOf(Attribute.Argument) == 2 * @sizeOf(*Type) + 2 * @sizeOf(u64));
}
};
pub const Function = extern struct {
prefer_vector_width: String,
stack_protector_buffer_size: String,
definition_probe_stack: String,
definition_stack_probe_size: String,
flags0: packed struct(u64) {
noreturn: bool,
cmse_ns_call: bool,
nounwind: bool,
returns_twice: bool,
cold: bool,
hot: bool,
no_duplicate: bool,
convergent: bool,
no_merge: bool,
will_return: bool,
no_caller_saved_registers: bool,
no_cf_check: bool,
no_callback: bool,
alloc_size: bool,
uniform_work_group_size: bool,
aarch64_pstate_sm_body: bool,
aarch64_pstate_sm_enabled: bool,
aarch64_pstate_sm_compatible: bool,
aarch64_preserves_za: bool,
aarch64_in_za: bool,
aarch64_out_za: bool,
aarch64_inout_za: bool,
aarch64_preserves_zt0: bool,
aarch64_in_zt0: bool,
aarch64_out_zt0: bool,
aarch64_inout_zt0: bool,
optimize_for_size: bool,
min_size: bool,
no_red_zone: bool,
indirect_tls_seg_refs: bool,
no_implicit_floats: bool,
sample_profile_suffix_elision_policy: bool,
memory_none: bool,
memory_readonly: bool,
memory_inaccessible_or_arg_memory_only: bool,
memory_arg_memory_only: bool,
strict_fp: bool,
no_inline: bool,
always_inline: bool,
guard_no_cf: bool,
// TODO: branch protection function attributes
// TODO: cpu features
// CALL-SITE ATTRIBUTES
call_no_builtins: bool,
// DEFINITION-SITE ATTRIBUTES
definition_frame_pointer_kind: FramePointerKind,
definition_less_precise_fpmad: bool,
definition_null_pointer_is_valid: bool,
definition_no_trapping_fp_math: bool,
definition_no_infs_fp_math: bool,
definition_no_nans_fp_math: bool,
definition_approx_func_fp_math: bool,
definition_unsafe_fp_math: bool,
definition_use_soft_float: bool,
definition_no_signed_zeroes_fp_math: bool,
definition_stack_realignment: bool,
definition_backchain: bool,
definition_split_stack: bool,
definition_speculative_load_hardening: bool,
definition_zero_call_used_registers: ZeroCallUsedRegsKind,
// TODO: denormal builtins
definition_non_lazy_bind: bool,
definition_cmse_nonsecure_entry: bool,
definition_unwind_table_kind: UnwindTableKind,
},
flags1: packed struct(u64) {
definition_disable_tail_calls: bool,
definition_stack_protect_strong: bool,
definition_stack_protect: bool,
definition_stack_protect_req: bool,
definition_aarch64_new_za: bool,
definition_aarch64_new_zt0: bool,
definition_optimize_none: bool,
definition_naked: bool,
definition_inline_hint: bool,
reserved: u55 = 0,
},
comptime {
assert(@sizeOf(Attribute.Function) == 10 * @sizeOf(u64));
}
};
};
pub const CodeModel = enum(u8) {
@ -74,19 +262,19 @@ pub const Target = opaque {
/// This is ABI-compatible with C++
pub const Options = extern struct {
flags0: packed struct(u64) {
unsafe_fp_math: u1,
no_infs_fp_math: u1,
no_nans_fp_math: u1,
no_trapping_fp_math: u1,
no_signed_zeroes_fp_math: u1,
approx_func_fp_match: u1,
enable_aix_extended_altivec_abi: u1,
honor_sign_dependent_rounding_fp_math: u1,
no_zeroes_in_bss: u1,
guaranteed_tail_call_optimization: u1,
stack_symbol_ordering: u1,
enable_fast_isel: u1,
enable_global_isel: u1,
unsafe_fp_math: bool,
no_infs_fp_math: bool,
no_nans_fp_math: bool,
no_trapping_fp_math: bool,
no_signed_zeroes_fp_math: bool,
approx_func_fp_math: bool,
enable_aix_extended_altivec_abi: bool,
honor_sign_dependent_rounding_fp_math: bool,
no_zeroes_in_bss: bool,
guaranteed_tail_call_optimization: bool,
stack_symbol_ordering: bool,
enable_fast_isel: bool,
enable_global_isel: bool,
global_isel_abort_mode: enum(u2) {
disable = 0,
enable = 1,
@ -97,27 +285,27 @@ pub const Target = opaque {
always = 1,
never = 2,
},
use_init_array: u1,
disable_integrated_assembler: u1,
function_sections: u1,
data_sections: u1,
ignore_xcoff_visibility: u1,
xcoff_traceback_table: u1,
unique_section_names: u1,
unique_basic_block_section_names: u1,
separate_named_sections: u1,
trap_unreachable: u1,
no_trap_after_noreturn: u1,
use_init_array: bool,
disable_integrated_assembler: bool,
function_sections: bool,
data_sections: bool,
ignore_xcoff_visibility: bool,
xcoff_traceback_table: bool,
unique_section_names: bool,
unique_basic_block_section_names: bool,
separate_named_sections: bool,
trap_unreachable: bool,
no_trap_after_noreturn: bool,
tls_size: u8,
emulated_tls: u1,
enable_tls_descriptors: u1,
enable_ipra: u1,
emit_stack_size_section: u1,
enable_machine_outliner: u1,
enable_machine_function_splitter: u1,
supports_default_outlining: u1,
emit_address_significance_table: u1,
bb_address_map: u1,
emulated_tls: bool,
enable_tls_descriptors: bool,
enable_ipra: bool,
emit_stack_size_section: bool,
enable_machine_outliner: bool,
enable_machine_function_splitter: bool,
supports_default_outlining: bool,
emit_address_significance_table: bool,
bb_address_map: bool,
bb_sections: enum(u3) {
all = 0,
list = 1,
@ -125,19 +313,19 @@ pub const Target = opaque {
preset = 3,
none = 4,
},
emit_call_site_information: u1,
supports_debug_entry_values: u1,
enable_debug_entry_values: u1,
value_tracking_variable_locations: u1,
force_dwarf_frame_section: u1,
xray_function_index: u1,
debug_strict_dwarf: u1,
hotpatch: u1,
ppc_gen_scalar_mass_entries: u1,
jmc_instrument: u1,
enable_cfi_fixup: u1,
mis_expect: u1,
xcoff_read_only_pointers: u1,
emit_call_site_information: bool,
supports_debug_entry_values: bool,
enable_debug_entry_values: bool,
value_tracking_variable_locations: bool,
force_dwarf_frame_section: bool,
xray_function_index: bool,
debug_strict_dwarf: bool,
hotpatch: bool,
ppc_gen_scalar_mass_entries: bool,
jmc_instrument: bool,
enable_cfi_fixup: bool,
mis_expect: bool,
xcoff_read_only_pointers: bool,
float_abi: enum(u2) {
default = 0,
soft = 1,
@ -200,56 +388,56 @@ pub const Target = opaque {
return .{
.binutils_version = .{ 0, 0 },
.flags0 = .{
.unsafe_fp_math = 0,
.no_infs_fp_math = 0,
.no_nans_fp_math = 0,
.no_trapping_fp_math = 1,
.no_signed_zeroes_fp_math = 0,
.approx_func_fp_match = 0,
.enable_aix_extended_altivec_abi = 0,
.honor_sign_dependent_rounding_fp_math = 0,
.no_zeroes_in_bss = 0,
.guaranteed_tail_call_optimization = 0,
.stack_symbol_ordering = 1,
.enable_fast_isel = 0,
.enable_global_isel = 0,
.unsafe_fp_math = false,
.no_infs_fp_math = false,
.no_nans_fp_math = false,
.no_trapping_fp_math = true,
.no_signed_zeroes_fp_math = false,
.approx_func_fp_math = false,
.enable_aix_extended_altivec_abi = false,
.honor_sign_dependent_rounding_fp_math = false,
.no_zeroes_in_bss = false,
.guaranteed_tail_call_optimization = false,
.stack_symbol_ordering = true,
.enable_fast_isel = false,
.enable_global_isel = false,
.global_isel_abort_mode = .enable,
.swift_async_frame_pointer = .always,
.use_init_array = 0,
.disable_integrated_assembler = 0,
.function_sections = 0,
.data_sections = 0,
.ignore_xcoff_visibility = 0,
.xcoff_traceback_table = 1,
.unique_section_names = 1,
.unique_basic_block_section_names = 0,
.separate_named_sections = 0,
.trap_unreachable = 0,
.no_trap_after_noreturn = 0,
.use_init_array = false,
.disable_integrated_assembler = false,
.function_sections = false,
.data_sections = false,
.ignore_xcoff_visibility = false,
.xcoff_traceback_table = true,
.unique_section_names = true,
.unique_basic_block_section_names = false,
.separate_named_sections = false,
.trap_unreachable = false,
.no_trap_after_noreturn = false,
.tls_size = 0,
.emulated_tls = 0,
.enable_tls_descriptors = 0,
.enable_ipra = 0,
.emit_stack_size_section = 0,
.enable_machine_outliner = 0,
.enable_machine_function_splitter = 0,
.supports_default_outlining = 0,
.emit_address_significance_table = 0,
.bb_address_map = 0,
.emulated_tls = false,
.enable_tls_descriptors = false,
.enable_ipra = false,
.emit_stack_size_section = false,
.enable_machine_outliner = false,
.enable_machine_function_splitter = false,
.supports_default_outlining = false,
.emit_address_significance_table = false,
.bb_address_map = false,
.bb_sections = .none,
.emit_call_site_information = 0,
.supports_debug_entry_values = 0,
.enable_debug_entry_values = 0,
.value_tracking_variable_locations = 0,
.force_dwarf_frame_section = 0,
.xray_function_index = 1,
.debug_strict_dwarf = 0,
.hotpatch = 0,
.ppc_gen_scalar_mass_entries = 0,
.jmc_instrument = 0,
.enable_cfi_fixup = 0,
.mis_expect = 0,
.xcoff_read_only_pointers = 0,
.emit_call_site_information = false,
.supports_debug_entry_values = false,
.enable_debug_entry_values = false,
.value_tracking_variable_locations = false,
.force_dwarf_frame_section = false,
.xray_function_index = true,
.debug_strict_dwarf = false,
.hotpatch = false,
.ppc_gen_scalar_mass_entries = false,
.jmc_instrument = false,
.enable_cfi_fixup = false,
.mis_expect = false,
.xcoff_read_only_pointers = false,
.float_abi = .default,
.thread_model = .posix,
},
@ -271,28 +459,28 @@ pub const Target = opaque {
.integrated_assembler_search_path_pointer = null,
.integrated_assembler_search_path_count = 0,
.flags = .{
.relax_all = 0,
.no_exec_stack = 0,
.fatal_warnings = 0,
.no_warn = 0,
.no_deprecated_warn = 0,
.no_type_check = 0,
.save_temp_labels = 0,
.incremental_linker_compatible = 0,
.fdpic = 0,
.show_mc_encoding = 0,
.show_mc_inst = 0,
.asm_verbose = 0,
.preserve_asm_comments = 1,
.dwarf64 = 0,
.crel = 0,
.x86_relax_relocations = 1,
.x86_sse2_avx = 0,
.relax_all = false,
.no_exec_stack = false,
.fatal_warnings = false,
.no_warn = false,
.no_deprecated_warn = false,
.no_type_check = false,
.save_temp_labels = false,
.incremental_linker_compatible = false,
.fdpic = false,
.show_mc_encoding = false,
.show_mc_inst = false,
.asm_verbose = false,
.preserve_asm_comments = true,
.dwarf64 = false,
.crel = false,
.x86_relax_relocations = true,
.x86_sse2_avx = false,
.emit_dwarf_unwind = .default,
.use_dwarf_directory = .default,
.debug_compression_type = .none,
.emit_compact_unwind_non_canonical = 0,
.ppc_use_full_register_names = 0,
.emit_compact_unwind_non_canonical = false,
.ppc_use_full_register_names = false,
},
},
};
@ -337,23 +525,23 @@ pub const MCTargetOptions = extern struct {
integrated_assembler_search_path_pointer: ?[*]const String,
integrated_assembler_search_path_count: u64,
flags: packed struct(u32) {
relax_all: u1,
no_exec_stack: u1,
fatal_warnings: u1,
no_warn: u1,
no_deprecated_warn: u1,
no_type_check: u1,
save_temp_labels: u1,
incremental_linker_compatible: u1,
fdpic: u1,
show_mc_encoding: u1,
show_mc_inst: u1,
asm_verbose: u1,
preserve_asm_comments: u1,
dwarf64: u1,
crel: u1,
x86_relax_relocations: u1,
x86_sse2_avx: u1,
relax_all: bool,
no_exec_stack: bool,
fatal_warnings: bool,
no_warn: bool,
no_deprecated_warn: bool,
no_type_check: bool,
save_temp_labels: bool,
incremental_linker_compatible: bool,
fdpic: bool,
show_mc_encoding: bool,
show_mc_inst: bool,
asm_verbose: bool,
preserve_asm_comments: bool,
dwarf64: bool,
crel: bool,
x86_relax_relocations: bool,
x86_sse2_avx: bool,
emit_dwarf_unwind: enum(u2) {
always = 0,
no_compact_unwind = 1,
@ -369,8 +557,8 @@ pub const MCTargetOptions = extern struct {
zlib = 1,
zstd = 2,
},
emit_compact_unwind_non_canonical: u1,
ppc_use_full_register_names: u1,
emit_compact_unwind_non_canonical: bool,
ppc_use_full_register_names: bool,
reserved: PaddingType = 0,
},
@ -410,16 +598,16 @@ pub const OptimizationLevel = enum(u3) {
/// This is ABI-compatible with C++
pub const OptimizationPipelineOptions = packed struct(u64) {
optimization_level: OptimizationLevel,
debug_info: u1,
loop_unrolling: u1,
loop_interleaving: u1,
loop_vectorization: u1,
slp_vectorization: u1,
merge_functions: u1,
call_graph_profile: u1,
unified_lto: u1,
assignment_tracking: u1,
verify_module: u1,
debug_info: bool,
loop_unrolling: bool,
loop_interleaving: bool,
loop_vectorization: bool,
slp_vectorization: bool,
merge_functions: bool,
call_graph_profile: bool,
unified_lto: bool,
assignment_tracking: bool,
verify_module: bool,
reserved: PaddingType = 0,
const padding_bit_count = 51;
@ -437,10 +625,10 @@ pub const OptimizationPipelineOptions = packed struct(u64) {
const Create = packed struct {
optimization_level: OptimizationLevel,
debug_info: u1,
debug_info: bool,
};
pub fn default(create: Create) OptimizationPipelineOptions {
const pref_speed = @intFromBool(create.optimization_level.prefers_speed());
const pref_speed = create.optimization_level.prefers_speed();
return .{
.optimization_level = create.optimization_level,
.debug_info = create.debug_info,
@ -449,10 +637,10 @@ pub const OptimizationPipelineOptions = packed struct(u64) {
.loop_vectorization = pref_speed,
.slp_vectorization = pref_speed,
.merge_functions = pref_speed,
.call_graph_profile = 0,
.unified_lto = 0,
.call_graph_profile = false,
.unified_lto = false,
.assignment_tracking = create.debug_info,
.verify_module = @intFromBool(lib.optimization_mode == .ReleaseSafe or lib.optimization_mode == .Debug),
.verify_module = lib.optimization_mode == .ReleaseSafe or lib.optimization_mode == .Debug,
};
}
};
@ -467,8 +655,8 @@ pub const CodeGenerationPipelineOptions = extern struct {
object_file = 1,
null = 2,
},
optimize_when_possible: u1,
verify_module: u1,
optimize_when_possible: bool,
verify_module: bool,
reserved: PaddingType = 0,
},
@ -515,7 +703,7 @@ pub const Context = opaque {
pub const create_builder = api.LLVMCreateBuilderInContext;
pub fn create_basic_block(context: *Context, name: []const u8, parent: *Function) *BasicBlock {
pub fn create_basic_block(context: *Context, name: []const u8, parent: ?*Function) *BasicBlock {
return api.llvm_context_create_basic_block(context, String.from_slice(name), parent);
}
@ -541,16 +729,30 @@ pub const Context = opaque {
return api.LLVMIntrinsicGetType(context, intrinsic_id, parameter_types.ptr, parameter_types.len);
}
pub fn create_string_attribute(context: *Context, attribute_name: []const u8, attribute_value: []const u8) *Attribute {
return api.LLVMCreateStringAttribute(context, attribute_name.ptr, @intCast(attribute_name.len), attribute_value.ptr, @intCast(attribute_value.len));
pub fn get_anonymous_constant_struct(context: *Context, constant_values: []const *Constant, is_packed: bool) *Constant {
return api.LLVMConstStructInContext(context, constant_values.ptr, @intCast(constant_values.len), @intFromBool(is_packed));
}
pub const create_enum_attribute = api.LLVMCreateEnumAttribute;
pub const create_type_attribute = api.LLVMCreateTypeAttribute;
// pub fn create_string_attribute(context: *Context, attribute_name: []const u8, attribute_value: []const u8) *Attribute {
// return api.LLVMCreateStringAttribute(context, attribute_name.ptr, @intCast(attribute_name.len), attribute_value.ptr, @intCast(attribute_value.len));
// }
//
// pub const create_enum_attribute = api.LLVMCreateEnumAttribute;
// pub const create_type_attribute = api.LLVMCreateTypeAttribute;
};
pub const BasicBlock = opaque {
pub const delete = api.llvm_basic_block_delete;
pub const erase_from_parent = api.LLVMDeleteBasicBlock;
pub const get_terminator = api.LLVMGetBasicBlockTerminator;
pub const is_empty = api.llvm_basic_block_is_empty;
pub const user_begin = api.llvm_basic_block_user_begin;
pub const get_next = api.LLVMGetNextBasicBlock;
pub const get_parent = api.LLVMGetBasicBlockParent;
pub fn to_value(basic_block: *BasicBlock) *Value {
return @ptrCast(basic_block);
}
};
pub const Module = opaque {
@ -610,6 +812,7 @@ pub const VerifyResult = struct {
pub const Builder = opaque {
pub const position_at_end = api.LLVMPositionBuilderAtEnd;
pub const clear_insertion_position = api.LLVMClearInsertionPosition;
pub const get_insert_block = api.LLVMGetInsertBlock;
pub const create_ret = api.LLVMBuildRet;
@ -690,6 +893,7 @@ pub const Builder = opaque {
return api.LLVMBuildICmp(builder, predicate, left, right, "");
}
pub const create_branch = api.LLVMBuildBr;
pub const create_conditional_branch = api.LLVMBuildCondBr;
pub fn create_call(builder: *Builder, function_type: *Type.Function, function_value: *Value, arguments: []const *Value) *Value {
@ -708,6 +912,10 @@ pub const Builder = opaque {
return api.LLVMBuildInsertValue(builder, aggregate, element, index, "");
}
pub fn create_extract_value(builder: *Builder, aggregate: *Value, index: c_uint) *Value {
return api.LLVMBuildExtractValue(builder, aggregate, index, "");
}
pub fn create_zero_extend(builder: *Builder, value: *Value, destination_type: *Type) *Value {
return api.LLVMBuildZExt(builder, value, destination_type, "");
}
@ -731,6 +939,12 @@ pub const Builder = opaque {
pub const create_unreachable = api.LLVMBuildUnreachable;
pub const create_memcpy = api.LLVMBuildMemCpy;
pub fn create_vaarg(builder: *Builder, va_list: *Value, arg_type: *Type) *Value {
return api.LLVMBuildVAArg(builder, va_list, arg_type, "");
}
pub const find_return_value_dominating_store = api.llvm_find_return_value_dominating_store;
};
pub const GlobalValue = opaque {
@ -739,6 +953,10 @@ pub const GlobalValue = opaque {
pub const GlobalVariable = opaque {
pub const add_debug_info = api.llvm_global_variable_add_debug_info;
pub const get_initializer = api.LLVMGetInitializer;
pub const set_initializer = api.LLVMSetInitializer;
pub const erase_from_parent = api.LLVMDeleteGlobal;
pub const delete = api.llvm_global_variable_delete;
pub fn to_value(global_variable: *GlobalVariable) *Value {
return @ptrCast(global_variable);
}
@ -776,7 +994,9 @@ pub const Function = opaque {
pub const get_arguments = api.LLVMGetParams;
pub const add_attribute = api.LLVMAddAttributeAtIndex;
pub const set_attributes = api.llvm_function_set_attributes;
pub const get_last_basic_block = api.LLVMGetLastBasicBlock;
pub const append_basic_block = api.LLVMAppendExistingBasicBlock;
};
pub const Constant = opaque {
@ -784,17 +1004,19 @@ pub const Constant = opaque {
return @ptrCast(constant);
}
pub fn to_global_variable(constant: *Constant) *GlobalVariable {
assert(constant.to_value().get_kind() == .global_variable);
return @ptrCast(constant);
}
pub const Integer = opaque {
pub fn to_value(constant: *Constant.Integer) *Value {
return @ptrCast(constant);
}
};
pub const Array = opaque {
pub fn to_value(constant: *Constant.Array) *Value {
return @ptrCast(constant);
}
};
pub const get_sign_extended_value = api.LLVMConstIntGetSExtValue;
pub const get_zero_extended_value = api.LLVMConstIntGetZExtValue;
};
pub const Argument = opaque {
@ -808,7 +1030,20 @@ pub const Value = opaque {
pub const get_kind = api.LLVMGetValueKind;
pub const set_alignment = api.LLVMSetAlignment;
pub fn set_name(value: *Value, name: []const u8) void {
api.LLVMSetValueName2(value, name.ptr, name.len);
}
// The operand API is from the User class, but it would work nonetheless
pub const get_operand = api.LLVMGetOperand;
pub const is_call_instruction = api.LLVMIsACallInst;
pub const use_empty = api.llvm_value_use_empty;
pub const has_one_use = api.llvm_value_has_one_use;
pub const replace_all_uses_with = api.LLVMReplaceAllUsesWith;
pub const to_branch = api.llvm_value_to_branch;
pub fn is_constant(value: *Value) bool {
return api.LLVMIsConstant(value) != 0;
@ -829,19 +1064,9 @@ pub const Value = opaque {
return @ptrCast(value);
}
pub fn get_calling_convention(value: *Value) CallingConvention {
const kind = value.get_kind();
switch (kind) {
.Instruction => {
const call = value.to_instruction().to_call();
return call.get_calling_convention();
},
.Function => {
const function = value.to_function();
return function.get_calling_convention();
},
else => unreachable,
}
pub fn to_global_variable(value: *Value) *GlobalVariable {
assert(value.get_kind() == .GlobalVariable);
return @ptrCast(value);
}
pub const Kind = enum(c_uint) {
@ -884,14 +1109,35 @@ pub const Instruction = opaque {
pub fn to_value(instruction: *Instruction) *Value {
return @ptrCast(instruction);
}
pub fn to_call(instruction: *Instruction) *Instruction.Call {
assert(instruction.to_value().is_call_instruction() != null);
pub fn to_call_base(instruction: *Instruction) *Instruction.CallBase {
assert(instruction.is_call_base());
return @ptrCast(instruction);
}
pub const Call = opaque {
pub const is_call_base = api.llvm_instruction_is_call_base;
pub const erase_from_parent = api.LLVMInstructionEraseFromParent;
pub const get_parent = api.LLVMGetInstructionParent;
pub const Branch = opaque {
pub const is_conditional = api.LLVMIsConditional;
pub const get_successor = api.LLVMGetSuccessor;
pub fn to_instruction(branch: *Branch) *Instruction {
return @ptrCast(branch);
}
};
pub const CallBase = opaque {
pub const set_calling_convention = api.LLVMSetInstructionCallConv;
pub const get_calling_convention = api.LLVMGetInstructionCallConv;
pub const add_attribute = api.LLVMAddCallSiteAttribute;
pub const set_attributes = api.llvm_call_base_set_attributes;
};
pub const Store = opaque {
pub fn to_instruction(store: *Store) *Instruction {
return @ptrCast(store);
}
};
};
@ -999,6 +1245,10 @@ pub const DI = struct {
pub const Record = opaque {};
pub const Type = opaque {
// TODO: typecheck
pub fn to_subroutine(ty: *DI.Type) *Subroutine {
return @ptrCast(ty);
}
pub const Subroutine = opaque {
pub fn to_type(subroutine: *Subroutine) *DI.Type {
return @ptrCast(subroutine);
@ -1126,6 +1376,10 @@ pub const Type = opaque {
};
pub const Struct = opaque {
pub fn get_constant(struct_type: *Type.Struct, constant_values: []const *Constant) *Constant {
return api.LLVMConstNamedStruct(struct_type, constant_values.ptr, @intCast(constant_values.len));
}
pub fn to_type(struct_type: *Type.Struct) *Type {
return @ptrCast(struct_type);
}
@ -1152,9 +1406,12 @@ pub const Type = opaque {
return api.LLVMArrayType2(element_type, element_count);
}
pub fn get_constant_array(element_type: *Type, values: []const *Constant) *Constant.Array {
pub fn get_constant_array(element_type: *Type, values: []const *Constant) *Constant {
return api.LLVMConstArray2(element_type, values.ptr, values.len);
}
pub const get_size = api.LLVMSizeOf;
pub const get_alignment = api.LLVMAlignOf;
};
pub const Dwarf = struct {
@ -1285,9 +1542,9 @@ pub fn lookup_intrinsic_id(name: []const u8) Intrinsic.Id {
return api.LLVMLookupIntrinsicID(name.ptr, name.len);
}
pub fn lookup_attribute_kind(name: []const u8) Attribute.Kind {
return api.LLVMGetEnumAttributeKindForName(name.ptr, name.len);
}
// pub fn lookup_attribute_kind(name: []const u8) Attribute.Kind {
// return api.LLVMGetEnumAttributeKindForName(name.ptr, name.len);
// }
pub const IntPredicate = enum(c_int) {
eq = 32,
@ -1429,14 +1686,14 @@ pub const ObjectGenerate = struct {
path: []const u8,
optimization_level: ?OptimizationLevel,
debug_info: bool,
optimize_when_possible: u1,
optimize_when_possible: bool,
};
pub fn object_generate(module: *Module, target_machine: *Target.Machine, generate: ObjectGenerate) CodeGenerationPipelineResult {
module.set_target(target_machine);
if (generate.optimization_level) |optimization_level| {
module.run_optimization_pipeline(target_machine, OptimizationPipelineOptions.default(.{ .optimization_level = optimization_level, .debug_info = @intFromBool(generate.debug_info) }));
module.run_optimization_pipeline(target_machine, OptimizationPipelineOptions.default(.{ .optimization_level = optimization_level, .debug_info = generate.debug_info }));
}
// const mod_string = module.to_string();
@ -1448,7 +1705,7 @@ pub fn object_generate(module: *Module, target_machine: *Target.Machine, generat
.flags = .{
.code_generation_file_type = .object_file,
.optimize_when_possible = generate.optimize_when_possible,
.verify_module = 1,
.verify_module = true,
},
});

File diff suppressed because it is too large Load Diff

View File

@ -22,7 +22,7 @@ fn invoke(name: []const u8) !void {
inline for (@typeInfo(BuildMode).@"enum".fields) |f| {
const build_mode = @field(BuildMode, f.name);
inline for ([2]bool{ false, true }) |has_debug_info| {
inline for ([2]bool{ true, false }) |has_debug_info| {
// Bootstrap
{
var tmp_dir = std.testing.tmpDir(.{});
@ -329,6 +329,30 @@ test "c_struct_with_array" {
try invsrc(@src());
}
test "indirect" {
try invsrc(@src());
}
test "indirect_struct" {
try invsrc(@src());
}
test "u1_return" {
try invsrc(@src());
}
test "small_struct_ints" {
try invsrc(@src());
}
test "c_med_struct_ints" {
try invsrc(@src());
}
test "c_abi" {
try invsrc(@src());
}
// test "varargs" {
// try invsrc(@src());
// }

View File

@ -122,7 +122,6 @@ pub const file = struct {
}, .{
.read = 1,
});
defer fd.close();
var result: []u8 = undefined;
const ptr = @as(*[2]u64, @ptrCast(&result));
ptr[0] = 0;
@ -133,6 +132,7 @@ pub const file = struct {
const file_buffer = arena.allocate_bytes(file_size, 1);
result = file_buffer[0..file_size];
fd.read(result, file_size);
fd.close();
}
return result;
@ -2504,7 +2504,18 @@ pub fn format_va(buffer: []u8, format_string: [*:0]const u8, variable_arguments:
switch (next_ch) {
left_brace => os.abort(),
'c' => {
os.abort();
read_byte_count += 1;
assert(format_string[read_byte_count] == 's');
read_byte_count += 1;
assert(format_string[read_byte_count] == 't');
read_byte_count += 1;
assert(format_string[read_byte_count] == 'r');
read_byte_count += 1;
const c_string = @cVaArg(variable_arguments, [*:0]const u8);
const str = cstring.to_slice(c_string);
@memcpy(buffer[written_byte_count..][0..str.len], str);
written_byte_count += str.len;
},
'f' => {
os.abort();
@ -2653,12 +2664,12 @@ test "parse integer hexadecimal" {
fn vprint(format_string: [*:0]const u8, args: *VariableArguments) void {
var buffer: [16 * 1024]u8 = undefined;
const slice = format_va(&buffer, format_string, args);
print_string(slice);
const byte_count = format_va(&buffer, format_string, args);
print_string(buffer[0..byte_count]);
}
pub fn print(format_string: [*:0]const u8, ...) callconv(.C) void {
const args = @cVaStart();
var args = @cVaStart();
vprint(format_string, &args);
@cVaEnd(&args);
}

View File

@ -7,6 +7,7 @@ typedef uint64_t u64;
#define EXPORT extern "C"
#define fn static
#define array_length(arr) (sizeof(arr) / sizeof((arr)[0]))
#include "llvm/Config/llvm-config.h"
@ -70,6 +71,11 @@ EXPORT void llvm_global_variable_add_debug_info(GlobalVariable& global_variable,
global_variable.addDebugInfo(debug_global_variable);
}
EXPORT void llvm_global_variable_delete(GlobalVariable* global_variable)
{
delete global_variable;
}
EXPORT Function* llvm_module_create_function(Module* module, FunctionType* function_type, GlobalValue::LinkageTypes linkage_type, unsigned address_space, BBLLVMString name)
{
auto* function = Function::Create(function_type, linkage_type, address_space, name.string_ref(), module);
@ -102,6 +108,102 @@ EXPORT BasicBlock* llvm_context_create_basic_block(LLVMContext& context, BBLLVMS
return basic_block;
}
EXPORT bool llvm_value_has_one_use(Value& value)
{
auto result = value.hasOneUse();
return result;
}
EXPORT Value* llvm_basic_block_user_begin(BasicBlock* basic_block)
{
Value* value = *basic_block->user_begin();
return value;
}
EXPORT void llvm_basic_block_delete(BasicBlock* basic_block)
{
delete basic_block;
}
EXPORT BranchInst* llvm_value_to_branch(Value* value)
{
auto* result = dyn_cast<BranchInst>(value);
return result;
}
// If there are multiple uses of the return-value slot, just check
// for something immediately preceding the IP. Sometimes this can
// happen with how we generate implicit-returns; it can also happen
// with noreturn cleanups.
fn StoreInst* get_store_if_valid(User* user, Value* return_alloca, Type* element_type)
{
auto *SI = dyn_cast<llvm::StoreInst>(user);
if (!SI || SI->getPointerOperand() != return_alloca ||
SI->getValueOperand()->getType() != element_type)
return nullptr;
// These aren't actually possible for non-coerced returns, and we
// only care about non-coerced returns on this code path.
// All memory instructions inside __try block are volatile.
assert(!SI->isAtomic() &&
(!SI->isVolatile()
//|| CGF.currentFunctionUsesSEHTry())
));
return SI;
}
// copy of static llvm::StoreInst *findDominatingStoreToReturnValue(CodeGenFunction &CGF) {
// in clang/lib/CodeGen/CGCall.cpp:3526 in LLVM 19
EXPORT StoreInst* llvm_find_return_value_dominating_store(IRBuilder<>& builder, Value* return_alloca, Type* element_type)
{
// Check if a User is a store which pointerOperand is the ReturnValue.
// We are looking for stores to the ReturnValue, not for stores of the
// ReturnValue to some other location.
if (!return_alloca->hasOneUse()) {
llvm::BasicBlock *IP = builder.GetInsertBlock();
if (IP->empty()) return nullptr;
// Look at directly preceding instruction, skipping bitcasts and lifetime
// markers.
for (llvm::Instruction &I : make_range(IP->rbegin(), IP->rend())) {
if (isa<llvm::BitCastInst>(&I))
continue;
if (auto *II = dyn_cast<llvm::IntrinsicInst>(&I))
if (II->getIntrinsicID() == llvm::Intrinsic::lifetime_end)
continue;
return get_store_if_valid(&I, return_alloca, element_type);
}
return nullptr;
}
llvm::StoreInst *store = get_store_if_valid(return_alloca->user_back(), return_alloca, element_type);
if (!store) return nullptr;
// Now do a first-and-dirty dominance check: just walk up the
// single-predecessors chain from the current insertion point.
llvm::BasicBlock *StoreBB = store->getParent();
llvm::BasicBlock *IP = builder.GetInsertBlock();
llvm::SmallPtrSet<llvm::BasicBlock *, 4> SeenBBs;
while (IP != StoreBB) {
if (!SeenBBs.insert(IP).second || !(IP = IP->getSinglePredecessor()))
return nullptr;
}
// Okay, the store's basic block dominates the insertion point; we
// can do our thing.
return store;
}
EXPORT bool llvm_value_use_empty(Value& value)
{
return value.use_empty();
}
EXPORT bool llvm_basic_block_is_empty(BasicBlock& basic_block)
{
return basic_block.empty();
}
EXPORT AllocaInst* llvm_builder_create_alloca(IRBuilder<>& builder, Type* type, unsigned address_space, BBLLVMString name)
{
const DataLayout &data_layout = builder.GetInsertBlock()->getDataLayout();
@ -109,6 +211,721 @@ EXPORT AllocaInst* llvm_builder_create_alloca(IRBuilder<>& builder, Type* type,
return builder.Insert(new AllocaInst(type, address_space, 0, alignment), name.string_ref());
}
enum class BBLLVMAttributeFramePointerKind : u8
{
None = 0,
Reserved = 1,
NonLeaf = 2,
All = 3,
};
const unsigned BB_LLVM_ONLY_USED = 1U << 1;
const unsigned BB_LLVM_ONLY_GPR = 1U << 2;
const unsigned BB_LLVM_ONLY_ARG = 1U << 3;
enum class BBLLVMZeroCallUsedRegsKind : unsigned int
{
// Don't zero any call-used regs.
Skip = 1U << 0,
// Only zeros call-used GPRs used in the fn and pass args.
UsedGPRArg = BB_LLVM_ONLY_USED | BB_LLVM_ONLY_GPR | BB_LLVM_ONLY_ARG,
// Only zeros call-used GPRs used in the fn.
UsedGPR = BB_LLVM_ONLY_USED | BB_LLVM_ONLY_GPR,
// Only zeros call-used regs used in the fn and pass args.
UsedArg = BB_LLVM_ONLY_USED | BB_LLVM_ONLY_ARG,
// Only zeros call-used regs used in the fn.
Used = BB_LLVM_ONLY_USED,
// Zeros all call-used GPRs that pass args.
AllGPRArg = BB_LLVM_ONLY_GPR | BB_LLVM_ONLY_ARG,
// Zeros all call-used GPRs.
AllGPR = BB_LLVM_ONLY_GPR,
// Zeros all call-used regs that pass args.
AllArg = BB_LLVM_ONLY_ARG,
// Zeros all call-used regs.
All = 0,
};
enum class BBLLVMFPClassTest : unsigned
{
None = 0,
SNan = 0x0001,
QNan = 0x0002,
NegInf = 0x0004,
NegNormal = 0x0008,
NegSubnormal = 0x0010,
NegZero = 0x0020,
PosZero = 0x0040,
PosSubnormal = 0x0080,
PosNormal = 0x0100,
PosInf = 0x0200,
Nan = SNan | QNan,
Inf = PosInf | NegInf,
Normal = PosNormal | NegNormal,
Subnormal = PosSubnormal | NegSubnormal,
Zero = PosZero | NegZero,
PosFinite = PosNormal | PosSubnormal | PosZero,
NegFinite = NegNormal | NegSubnormal | NegZero,
Finite = PosFinite | NegFinite,
Positive = PosFinite | PosInf,
Negative = NegFinite | NegInf,
AllFlags = Nan | Inf | Finite,
};
enum class BBLLVMUWTableKind
{
None = 0, ///< No unwind table requested
Sync = 1, ///< "Synchronous" unwind tables
Async = 2, ///< "Asynchronous" unwind tables (instr precise)
Default = 2,
};
struct BBLLVMArgumentAttributes
{
Type* semantic_type;
Type* abi_type;
u64 dereferenceable_bytes;
u32 alignment;
u32 no_alias:1;
u32 non_null:1;
u32 no_undef:1;
u32 sign_extend:1;
u32 zero_extend:1;
u32 in_reg:1;
u32 no_fp_class:10;
u32 struct_return:1;
u32 writable:1;
u32 dead_on_unwind:1;
u32 in_alloca:1;
u32 dereferenceable:1;
u32 dereferenceable_or_null:1;
u32 nest:1;
u32 by_value:1;
u32 by_reference:1;
u32 no_capture:1;
u32 _:6;
};
static_assert(sizeof(BBLLVMArgumentAttributes) == 2 * sizeof(Type*) + 2 * sizeof(u64));
fn AttributeSet build_argument_attributes(LLVMContext& context, const BBLLVMArgumentAttributes& attributes)
{
AttrBuilder builder(context);
if (attributes.alignment)
{
builder.addAlignmentAttr(attributes.alignment);
}
if (attributes.no_alias)
{
builder.addAttribute(Attribute::NoAlias);
}
if (attributes.non_null)
{
builder.addAttribute(Attribute::NonNull);
}
if (attributes.no_undef)
{
builder.addAttribute(Attribute::NoUndef);
}
if (attributes.sign_extend)
{
builder.addAttribute(Attribute::SExt);
}
if (attributes.zero_extend)
{
builder.addAttribute(Attribute::ZExt);
}
if (attributes.in_reg)
{
builder.addAttribute(Attribute::InReg);
}
if (attributes.no_fp_class)
{
__builtin_trap(); // TODO
}
if (attributes.struct_return)
{
builder.addStructRetAttr(attributes.semantic_type);
}
if (attributes.writable)
{
builder.addAttribute(Attribute::Writable);
}
if (attributes.dead_on_unwind)
{
builder.addAttribute(Attribute::DeadOnUnwind);
}
if (attributes.in_alloca)
{
__builtin_trap(); // TODO
}
if (attributes.dereferenceable)
{
builder.addDereferenceableAttr(attributes.dereferenceable_bytes);
}
if (attributes.dereferenceable_or_null)
{
builder.addDereferenceableOrNullAttr(attributes.dereferenceable_bytes);
}
if (attributes.nest)
{
builder.addAttribute(Attribute::Nest);
}
if (attributes.by_value)
{
builder.addByValAttr(attributes.semantic_type);
}
if (attributes.by_reference)
{
builder.addByRefAttr(attributes.semantic_type);
}
if (attributes.no_capture)
{
builder.addAttribute(Attribute::NoCapture);
}
auto attribute_set = AttributeSet::get(context, builder);
return attribute_set;
}
struct BBLLVMFunctionAttributesFlags0
{
u64 noreturn:1;
u64 cmse_ns_call:1;
u64 nounwind:1;
u64 returns_twice:1;
u64 cold:1;
u64 hot:1;
u64 no_duplicate:1;
u64 convergent:1;
u64 no_merge:1;
u64 will_return:1;
u64 no_caller_saved_registers:1;
u64 no_cf_check:1;
u64 no_callback:1;
u64 alloc_size:1;
u64 uniform_work_group_size:1;
u64 aarch64_pstate_sm_body:1;
u64 aarch64_pstate_sm_enabled:1;
u64 aarch64_pstate_sm_compatible:1;
u64 aarch64_preserves_za:1;
u64 aarch64_in_za:1;
u64 aarch64_out_za:1;
u64 aarch64_inout_za:1;
u64 aarch64_preserves_zt0:1;
u64 aarch64_in_zt0:1;
u64 aarch64_out_zt0:1;
u64 aarch64_inout_zt0:1;
u64 optimize_for_size:1;
u64 min_size:1;
u64 no_red_zone:1;
u64 indirect_tls_seg_refs:1;
u64 no_implicit_floats:1;
u64 sample_profile_suffix_elision_policy:1;
u64 memory_none:1;
u64 memory_readonly:1;
u64 memory_inaccessible_or_arg_memory_only:1;
u64 memory_arg_memory_only:1;
u64 strict_fp:1;
u64 no_inline:1;
u64 always_inline:1;
u64 guard_no_cf:1;
// TODO: branch protection function attributes
// TODO: cpu features
// Call-site begin
u64 call_no_builtins:1;
u64 definition_frame_pointer_kind:2;
u64 definition_less_precise_fpmad:1;
u64 definition_null_pointer_is_valid:1;
u64 definition_no_trapping_fp_math:1;
u64 definition_no_infs_fp_math:1;
u64 definition_no_nans_fp_math:1;
u64 definition_approx_func_fp_math:1;
u64 definition_unsafe_fp_math:1;
u64 definition_use_soft_float:1;
u64 definition_no_signed_zeroes_fp_math:1;
u64 definition_stack_realignment:1;
u64 definition_backchain:1;
u64 definition_split_stack:1;
u64 definition_speculative_load_hardening:1;
u64 definition_zero_call_used_registers:4;
// TODO: denormal builtins
u64 definition_non_lazy_bind:1;
u64 definition_cmse_nonsecure_entry:1;
u64 definition_unwind_table_kind:2;
};
static_assert(sizeof(BBLLVMFunctionAttributesFlags0) == sizeof(u64));
struct BBLLVMFunctionAttributesFlags1
{
u64 definition_disable_tail_calls:1;
u64 definition_stack_protect_strong:1;
u64 definition_stack_protect:1;
u64 definition_stack_protect_req:1;
u64 definition_aarch64_new_za:1;
u64 definition_aarch64_new_zt0:1;
u64 definition_optimize_none:1;
u64 definition_naked:1;
u64 definition_inline_hint:1;
u64 _:55;
};
static_assert(sizeof(BBLLVMFunctionAttributesFlags1) == sizeof(u64));
struct BBLLVMFunctionAttributes
{
BBLLVMString prefer_vector_width;
BBLLVMString stack_protector_buffer_size;
BBLLVMString definition_probe_stack;
BBLLVMString definition_stack_probe_size;
BBLLVMFunctionAttributesFlags0 flags0;
BBLLVMFunctionAttributesFlags1 flags1;
};
static_assert(sizeof(BBLLVMFunctionAttributes) == 10 * sizeof(u64));
struct BBLLVMAttributeList
{
BBLLVMFunctionAttributes function;
BBLLVMArgumentAttributes return_;
const BBLLVMArgumentAttributes* argument_pointer;
u64 argument_count;
};
static_assert(sizeof(BBLLVMAttributeList) == sizeof(BBLLVMFunctionAttributes) + sizeof(BBLLVMArgumentAttributes) + sizeof(void*) + sizeof(u64));
typedef void* BBLLVMAttributeListHandle;
EXPORT BBLLVMAttributeListHandle llvm_attribute_list_build(LLVMContext& context, const BBLLVMAttributeList& attributes, bool call_site)
{
AttrBuilder function_attribute_builder(context);
if (attributes.function.prefer_vector_width.length)
{
function_attribute_builder.addAttribute("prefer-vector-width", attributes.function.prefer_vector_width.string_ref());
}
if (attributes.function.stack_protector_buffer_size.length)
{
function_attribute_builder.addAttribute("stack-protector-buffer-size", attributes.function.stack_protector_buffer_size.string_ref());
}
if (attributes.function.flags0.noreturn)
{
function_attribute_builder.addAttribute(Attribute::NoReturn);
}
if (attributes.function.flags0.cmse_ns_call)
{
function_attribute_builder.addAttribute("cmse_nonsecure_call");
}
if (attributes.function.flags0.nounwind)
{
function_attribute_builder.addAttribute(Attribute::NoUnwind);
}
if (attributes.function.flags0.returns_twice)
{
function_attribute_builder.addAttribute(Attribute::ReturnsTwice);
}
if (attributes.function.flags0.cold)
{
function_attribute_builder.addAttribute(Attribute::Cold);
}
if (attributes.function.flags0.hot)
{
function_attribute_builder.addAttribute(Attribute::Hot);
}
if (attributes.function.flags0.no_duplicate)
{
function_attribute_builder.addAttribute(Attribute::NoDuplicate);
}
if (attributes.function.flags0.convergent)
{
function_attribute_builder.addAttribute(Attribute::Convergent);
}
if (attributes.function.flags0.no_merge)
{
function_attribute_builder.addAttribute(Attribute::NoMerge);
}
if (attributes.function.flags0.will_return)
{
function_attribute_builder.addAttribute(Attribute::WillReturn);
}
if (attributes.function.flags0.no_caller_saved_registers)
{
function_attribute_builder.addAttribute("no-caller-saved-registers");
}
if (attributes.function.flags0.no_cf_check)
{
function_attribute_builder.addAttribute(Attribute::NoCfCheck);
}
if (attributes.function.flags0.no_callback)
{
function_attribute_builder.addAttribute(Attribute::NoCallback);
}
if (attributes.function.flags0.alloc_size)
{
__builtin_trap(); // TODO
}
if (attributes.function.flags0.uniform_work_group_size)
{
__builtin_trap(); // TODO
}
if (attributes.function.flags0.aarch64_pstate_sm_body)
{
function_attribute_builder.addAttribute("aarch64_pstate_sm_body");
}
if (attributes.function.flags0.aarch64_pstate_sm_enabled)
{
function_attribute_builder.addAttribute("aarch64_pstate_sm_enabled");
}
if (attributes.function.flags0.aarch64_pstate_sm_compatible)
{
function_attribute_builder.addAttribute("aarch64_pstate_sm_compatible");
}
if (attributes.function.flags0.aarch64_preserves_za)
{
function_attribute_builder.addAttribute("aarch64_preserves_za");
}
if (attributes.function.flags0.aarch64_in_za)
{
function_attribute_builder.addAttribute("aarch64_in_za");
}
if (attributes.function.flags0.aarch64_out_za)
{
function_attribute_builder.addAttribute("aarch64_out_za");
}
if (attributes.function.flags0.aarch64_inout_za)
{
function_attribute_builder.addAttribute("aarch64_inout_za");
}
if (attributes.function.flags0.aarch64_preserves_zt0)
{
function_attribute_builder.addAttribute("aarch64_preserves_zt0");
}
if (attributes.function.flags0.aarch64_in_zt0)
{
function_attribute_builder.addAttribute("aarch64_in_zt0");
}
if (attributes.function.flags0.aarch64_out_zt0)
{
function_attribute_builder.addAttribute("aarch64_out_zt0");
}
if (attributes.function.flags0.aarch64_inout_zt0)
{
function_attribute_builder.addAttribute("aarch64_inout_zt0");
}
if (attributes.function.flags0.optimize_for_size)
{
function_attribute_builder.addAttribute(Attribute::OptimizeForSize);
}
if (attributes.function.flags0.min_size)
{
function_attribute_builder.addAttribute(Attribute::MinSize);
}
if (attributes.function.flags0.no_red_zone)
{
function_attribute_builder.addAttribute(Attribute::NoRedZone);
}
if (attributes.function.flags0.indirect_tls_seg_refs)
{
function_attribute_builder.addAttribute("indirect-tls-seg-refs");
}
if (attributes.function.flags0.no_implicit_floats)
{
function_attribute_builder.addAttribute(Attribute::NoImplicitFloat);
}
if (attributes.function.flags0.sample_profile_suffix_elision_policy)
{
function_attribute_builder.addAttribute("sample-profile-suffix-elision-policy", "selected");
}
if (attributes.function.flags0.memory_none)
{
function_attribute_builder.addMemoryAttr(llvm::MemoryEffects::none());
}
if (attributes.function.flags0.memory_readonly)
{
function_attribute_builder.addMemoryAttr(llvm::MemoryEffects::readOnly());
}
if (attributes.function.flags0.memory_inaccessible_or_arg_memory_only)
{
function_attribute_builder.addMemoryAttr(llvm::MemoryEffects::inaccessibleOrArgMemOnly());
}
if (attributes.function.flags0.memory_arg_memory_only)
{
Attribute attribute = function_attribute_builder.getAttribute(Attribute::Memory);
function_attribute_builder.addMemoryAttr(attribute.getMemoryEffects() | llvm::MemoryEffects::argMemOnly());
}
// TODO: branch protection function attributes
// TODO: cpu features
if (call_site)
{
if (attributes.function.flags0.call_no_builtins)
{
function_attribute_builder.addAttribute(Attribute::NoBuiltin);
}
}
else
{
if (attributes.function.definition_probe_stack.length)
{
function_attribute_builder.addAttribute("probe-stack", attributes.function.definition_probe_stack.string_ref());
}
if (attributes.function.definition_stack_probe_size.length)
{
function_attribute_builder.addAttribute("stack-probe-size", attributes.function.definition_stack_probe_size.string_ref());
}
StringRef frame_pointer_kind_name;
switch ((BBLLVMAttributeFramePointerKind) attributes.function.flags0.definition_frame_pointer_kind)
{
case BBLLVMAttributeFramePointerKind::None: frame_pointer_kind_name = "none"; break;
case BBLLVMAttributeFramePointerKind::Reserved: frame_pointer_kind_name = "reserved"; break;
case BBLLVMAttributeFramePointerKind::NonLeaf: frame_pointer_kind_name = "non-leaf"; break;
case BBLLVMAttributeFramePointerKind::All: frame_pointer_kind_name = "all"; break;
}
function_attribute_builder.addAttribute("frame-pointer", frame_pointer_kind_name);
if (attributes.function.flags0.definition_less_precise_fpmad)
{
function_attribute_builder.addAttribute("less-precise-fp-mad", "true");
}
if (attributes.function.flags0.definition_null_pointer_is_valid)
{
function_attribute_builder.addAttribute(Attribute::NullPointerIsValid);
}
if (attributes.function.flags0.definition_no_trapping_fp_math)
{
function_attribute_builder.addAttribute("no-trapping-math", "true");
}
if (attributes.function.flags0.definition_no_infs_fp_math)
{
function_attribute_builder.addAttribute("no-infs-fp-math", "true");
}
if (attributes.function.flags0.definition_no_nans_fp_math)
{
function_attribute_builder.addAttribute("no-nans-fp-math", "true");
}
if (attributes.function.flags0.definition_approx_func_fp_math)
{
function_attribute_builder.addAttribute("approx-func-fp-math", "true");
}
if (attributes.function.flags0.definition_unsafe_fp_math)
{
function_attribute_builder.addAttribute("unsafe-fp-math", "true");
}
if (attributes.function.flags0.definition_use_soft_float)
{
function_attribute_builder.addAttribute("use-soft-float", "true");
}
if (attributes.function.flags0.definition_no_signed_zeroes_fp_math)
{
function_attribute_builder.addAttribute("no-signed-zeros-fp-math", "true");
}
if (attributes.function.flags0.definition_stack_realignment)
{
function_attribute_builder.addAttribute("stackrealign");
}
if (attributes.function.flags0.definition_backchain)
{
function_attribute_builder.addAttribute("backchain");
}
if (attributes.function.flags0.definition_split_stack)
{
function_attribute_builder.addAttribute("split-stack");
}
if (attributes.function.flags0.definition_speculative_load_hardening)
{
function_attribute_builder.addAttribute("split-stack");
}
if (attributes.function.flags0.definition_zero_call_used_registers)
{
__builtin_trap(); // TODO
}
// TODO: denormal builtins
if (attributes.function.flags0.definition_non_lazy_bind)
{
function_attribute_builder.addAttribute(Attribute::NonLazyBind);
}
if (attributes.function.flags0.definition_cmse_nonsecure_entry)
{
function_attribute_builder.addAttribute("cmse_nonsecure_entry");
}
UWTableKind unwind_table_kind;
switch ((BBLLVMUWTableKind)attributes.function.flags0.definition_unwind_table_kind)
{
case BBLLVMUWTableKind::None: unwind_table_kind = UWTableKind::None; break;
case BBLLVMUWTableKind::Sync: unwind_table_kind = UWTableKind::Sync; break;
case BBLLVMUWTableKind::Async: unwind_table_kind = UWTableKind::Async; break;
}
function_attribute_builder.addUWTableAttr(unwind_table_kind);
if (attributes.function.flags1.definition_disable_tail_calls)
{
function_attribute_builder.addAttribute("disable-tail-calls", "true");
}
if (attributes.function.flags1.definition_stack_protect_strong)
{
function_attribute_builder.addAttribute(Attribute::StackProtectStrong);
}
if (attributes.function.flags1.definition_stack_protect)
{
function_attribute_builder.addAttribute(Attribute::StackProtect);
}
if (attributes.function.flags1.definition_stack_protect_req)
{
function_attribute_builder.addAttribute(Attribute::StackProtectReq);
}
if (attributes.function.flags1.definition_aarch64_new_za)
{
function_attribute_builder.addAttribute("aarch64_new_za");
}
if (attributes.function.flags1.definition_aarch64_new_zt0)
{
function_attribute_builder.addAttribute("aarch64_new_zt0");
}
if (attributes.function.flags1.definition_optimize_none)
{
function_attribute_builder.addAttribute(Attribute::OptimizeNone);
}
if (attributes.function.flags1.definition_naked)
{
function_attribute_builder.addAttribute(Attribute::Naked);
}
if (attributes.function.flags1.definition_inline_hint)
{
function_attribute_builder.addAttribute(Attribute::InlineHint);
}
}
auto function_attributes = AttributeSet::get(context, function_attribute_builder);
auto return_attributes = build_argument_attributes(context, attributes.return_);
AttributeSet argument_attribute_buffer[128];
assert(attributes.argument_count < array_length(argument_attribute_buffer));
for (u64 i = 0; i < attributes.argument_count; i += 1)
{
auto attribute_set = build_argument_attributes(context, attributes.argument_pointer[i]);
argument_attribute_buffer[i] = attribute_set;
}
ArrayRef<AttributeSet> argument_attributes = ArrayRef(argument_attribute_buffer, attributes.argument_count);
auto attribute_list = AttributeList::get(context, function_attributes, return_attributes, argument_attributes);
static_assert(sizeof(AttributeList) == sizeof(uintptr_t));
return *(BBLLVMAttributeListHandle*)&attribute_list;
}
EXPORT bool llvm_instruction_is_call_base(Instruction* instruction)
{
return isa<CallBase>(instruction);
}
EXPORT void llvm_function_set_attributes(Function& function, BBLLVMAttributeListHandle attribute_list_handle)
{
auto attribute_list = *(AttributeList*)&attribute_list_handle;
function.setAttributes(attribute_list);
}
EXPORT void llvm_call_base_set_attributes(CallBase& call, BBLLVMAttributeListHandle attribute_list_handle)
{
auto attribute_list = *(AttributeList*)&attribute_list_handle;
call.setAttributes(attribute_list);
}
fn BBLLVMString stream_to_string(raw_string_ostream& stream)
{
// No need to call stream.flush(); because it's string-based
@ -150,7 +967,6 @@ EXPORT bool llvm_function_verify(Function& function, BBLLVMString* error_message
return !result;
}
EXPORT bool llvm_module_verify(const Module& module, BBLLVMString* error_message)
{
std::string message_buffer;
@ -396,7 +1212,7 @@ struct BBLLVMTargetOptions
u64 no_nans_fp_math:1;
u64 no_trapping_fp_math:1;
u64 no_signed_zeroes_fp_math:1;
u64 approx_func_fp_match:1;
u64 approx_func_fp_math:1;
u64 enable_aix_extended_altivec_abi:1;
u64 honor_sign_dependent_rounding_fp_math:1;
u64 no_zeroes_in_bss:1;
@ -525,7 +1341,7 @@ EXPORT TargetMachine* llvm_create_target_machine(const BBLLVMTargetMachineCreate
target_options.NoNaNsFPMath = create.target_options.no_nans_fp_math;
target_options.NoTrappingFPMath = create.target_options.no_trapping_fp_math;
target_options.NoSignedZerosFPMath = create.target_options.no_signed_zeroes_fp_math;
target_options.ApproxFuncFPMath = create.target_options.approx_func_fp_match;
target_options.ApproxFuncFPMath = create.target_options.approx_func_fp_math;
target_options.EnableAIXExtendedAltivecABI = create.target_options.enable_aix_extended_altivec_abi;
target_options.HonorSignDependentRoundingFPMathOption = create.target_options.honor_sign_dependent_rounding_fp_math;
target_options.NoZerosInBSS = create.target_options.no_zeroes_in_bss;

View File

@ -7,21 +7,39 @@ pub extern fn llvm_context_create_module(context: *llvm.Context, name: llvm.Stri
pub extern fn LLVMContextCreate() *llvm.Context;
pub extern fn LLVMCreateBuilderInContext(context: *llvm.Context) *llvm.Builder;
pub extern fn LLVMIsACallInst(value: *llvm.Value) ?*llvm.Instruction.Call;
pub extern fn LLVMGetOperand(value: *llvm.Value, index: c_uint) *llvm.Value;
pub extern fn LLVMSetAlignment(value: *llvm.Value, alignment: c_uint) void;
pub extern fn llvm_instruction_is_call_base(instruction: *llvm.Instruction) bool;
// Module
pub extern fn llvm_module_create_global_variable(module: *llvm.Module, global_type: *llvm.Type, is_constant: bool, linkage: llvm.LinkageType, initial_value: *llvm.Constant, name: llvm.String, before: ?*llvm.GlobalVariable, thread_local_mode: llvm.ThreadLocalMode, address_space: c_uint, externally_initialized: bool) *llvm.GlobalVariable;
pub extern fn llvm_module_create_function(module: *llvm.Module, function_type: *llvm.Type.Function, linkage_type: llvm.LinkageType, address_space: c_uint, name: llvm.String) *llvm.Function;
pub extern fn llvm_context_create_basic_block(context: *llvm.Context, name: llvm.String, parent: *llvm.Function) *llvm.BasicBlock;
pub extern fn llvm_context_create_basic_block(context: *llvm.Context, name: llvm.String, parent: ?*llvm.Function) *llvm.BasicBlock;
pub extern fn LLVMGetNextBasicBlock(basic_block: *llvm.BasicBlock) ?*llvm.BasicBlock;
pub extern fn LLVMDeleteBasicBlock(basic_block: *llvm.BasicBlock) void;
pub extern fn LLVMGetLastBasicBlock(function: *llvm.Function) *llvm.BasicBlock;
pub extern fn LLVMGetBasicBlockParent(basic_block: *llvm.BasicBlock) ?*llvm.BasicBlock;
pub extern fn LLVMAppendExistingBasicBlock(function: *llvm.Function, basic_block: *llvm.BasicBlock) void;
pub extern fn LLVMSetValueName2(value: *llvm.Value, name_pointer: [*]const u8, name_length: usize) void;
pub extern fn llvm_value_use_empty(value: *llvm.Value) bool;
pub extern fn llvm_value_has_one_use(value: *llvm.Value) bool;
pub extern fn llvm_value_to_branch(value: ?*llvm.Value) ?*llvm.Instruction.Branch;
pub extern fn LLVMReplaceAllUsesWith(old: *llvm.Value, new: *llvm.Value) void;
pub extern fn LLVMGetSuccessor(branch: *llvm.Instruction.Branch, index: c_uint) *llvm.BasicBlock;
pub extern fn LLVMIsConditional(branch: *llvm.Instruction.Branch) bool;
pub extern fn LLVMGetInstructionParent(instruction: *llvm.Instruction) *llvm.BasicBlock;
pub extern fn llvm_basic_block_is_empty(basic_block: *llvm.BasicBlock) bool;
pub extern fn llvm_basic_block_user_begin(basic_block: *llvm.BasicBlock) ?*llvm.Value;
pub extern fn llvm_basic_block_delete(basic_block: *llvm.BasicBlock) void;
pub extern fn LLVMGetBasicBlockTerminator(basic_block: *llvm.BasicBlock) ?*llvm.Value;
pub extern fn LLVMSetFunctionCallConv(function: *llvm.Function, calling_convention: llvm.CallingConvention) void;
pub extern fn LLVMGetFunctionCallConv(function: *llvm.Function) llvm.CallingConvention;
pub extern fn LLVMSetInstructionCallConv(instruction: *llvm.Instruction.Call, calling_convention: llvm.CallingConvention) void;
pub extern fn LLVMGetInstructionCallConv(instruction: *llvm.Instruction.Call) llvm.CallingConvention;
pub extern fn LLVMSetInstructionCallConv(instruction: *llvm.Instruction.CallBase, calling_convention: llvm.CallingConvention) void;
pub extern fn LLVMGetParams(function: *llvm.Function, argument_buffer: [*]*llvm.Argument) void;
@ -33,7 +51,13 @@ pub extern fn llvm_module_to_string(module: *llvm.Module) llvm.String;
// Builder API
pub extern fn LLVMPositionBuilderAtEnd(builder: *llvm.Builder, basic_block: *llvm.BasicBlock) void;
pub extern fn LLVMGetInsertBlock(builder: *llvm.Builder) *llvm.BasicBlock;
pub extern fn LLVMClearInsertionPosition(builder: *llvm.Builder) void;
pub extern fn LLVMGetInsertBlock(builder: *llvm.Builder) ?*llvm.BasicBlock;
pub extern fn llvm_find_return_value_dominating_store(builder: *llvm.Builder, return_alloca: *llvm.Value, element_type: *llvm.Type) ?*llvm.Instruction.Store;
pub extern fn LLVMDeleteInstruction(instruction: *llvm.Instruction) void;
pub extern fn LLVMInstructionEraseFromParent(instruction: *llvm.Instruction) void;
pub extern fn LLVMBuildRet(builder: *llvm.Builder, value: ?*llvm.Value) void;
pub extern fn LLVMBuildAdd(builder: *llvm.Builder, left: *llvm.Value, right: *llvm.Value, name: [*:0]const u8) *llvm.Value;
@ -50,6 +74,7 @@ pub extern fn LLVMBuildAnd(builder: *llvm.Builder, left: *llvm.Value, right: *ll
pub extern fn LLVMBuildOr(builder: *llvm.Builder, left: *llvm.Value, right: *llvm.Value, name: [*:0]const u8) *llvm.Value;
pub extern fn LLVMBuildXor(builder: *llvm.Builder, left: *llvm.Value, right: *llvm.Value, name: [*:0]const u8) *llvm.Value;
pub extern fn LLVMBuildICmp(builder: *llvm.Builder, predicate: llvm.IntPredicate, left: *llvm.Value, right: *llvm.Value, name: [*:0]const u8) *llvm.Value;
pub extern fn LLVMBuildBr(builder: *llvm.Builder, block: *llvm.BasicBlock) *llvm.Value;
pub extern fn LLVMBuildCondBr(builder: *llvm.Builder, condition: *llvm.Value, taken: *llvm.BasicBlock, not_taken: *llvm.BasicBlock) *llvm.Value;
pub extern fn llvm_builder_create_alloca(builder: *llvm.Builder, ty: *llvm.Type, address_space: c_uint, name: llvm.String) *llvm.Value;
@ -60,9 +85,12 @@ pub extern fn LLVMBuildStructGEP2(builder: *llvm.Builder, struct_type: *llvm.Typ
pub extern fn LLVMBuildInBoundsGEP2(builder: *llvm.Builder, ty: *llvm.Type, aggregate: *llvm.Value, index_pointer: [*]const *llvm.Value, index_count: c_uint, name: [*:0]const u8) *llvm.Value;
pub extern fn LLVMBuildInsertValue(builder: *llvm.Builder, aggregate: *llvm.Value, element: *llvm.Value, index: c_uint, name: [*:0]const u8) *llvm.Value;
pub extern fn LLVMBuildExtractValue(builder: *llvm.Builder, aggregate: *llvm.Value, index: c_uint, name: [*:0]const u8) *llvm.Value;
pub extern fn LLVMBuildUnreachable(builder: *llvm.Builder) *llvm.Value;
pub extern fn LLVMBuildMemCpy(builder: *llvm.Builder, destination: *llvm.Value, destination_alignment: c_uint, source: *llvm.Value, source_alignment: c_uint, size: *llvm.Value) *llvm.Value;
pub extern fn LLVMBuildVAArg(builder: *llvm.Builder, va_list: *llvm.Value, arg_type: *llvm.Type, name: [*:0]const u8) *llvm.Value;
// Casts
pub extern fn LLVMBuildZExt(builder: *llvm.Builder, value: *llvm.Value, destination_type: *llvm.Type, name: [*:0]const u8) *llvm.Value;
pub extern fn LLVMBuildSExt(builder: *llvm.Builder, value: *llvm.Value, destination_type: *llvm.Type, name: [*:0]const u8) *llvm.Value;
@ -73,7 +101,13 @@ pub extern fn LLVMBuildTrunc(builder: *llvm.Builder, value: *llvm.Value, destina
pub extern fn LLVMSetCurrentDebugLocation2(builder: *llvm.Builder, location: ?*llvm.DI.Location) void;
pub extern fn LLVMTypeOf(value: *llvm.Value) *llvm.Type;
pub extern fn LLVMSizeOf(ty: *llvm.Type) *llvm.Constant;
pub extern fn LLVMAlignOf(ty: *llvm.Type) *llvm.Constant;
pub extern fn LLVMGlobalGetValueType(value: *llvm.GlobalValue) *llvm.Type;
pub extern fn LLVMGetInitializer(global_variable: *llvm.GlobalVariable) *llvm.Constant;
pub extern fn LLVMSetInitializer(global_variable: *llvm.GlobalVariable, initializer: *llvm.Constant) void;
pub extern fn LLVMDeleteGlobal(global_variable: *llvm.GlobalVariable) void;
pub extern fn llvm_global_variable_delete(global_variable: *llvm.GlobalVariable) void;
pub extern fn llvm_value_is_instruction(value: *llvm.Value) bool;
// Intrinsics
@ -82,15 +116,19 @@ pub extern fn LLVMGetIntrinsicDeclaration(module: *llvm.Module, intrinsic_id: ll
pub extern fn LLVMIntrinsicGetType(context: *llvm.Context, intrinsic_id: llvm.Intrinsic.Id, parameter_type_pointer: [*]const *llvm.Type, parameter_type_count: usize) *llvm.Type.Function;
// Attributes
pub extern fn LLVMGetEnumAttributeKindForName(name_pointer: [*]const u8, name_length: usize) llvm.Attribute.Kind;
pub extern fn llvm_attribute_list_build(context: *llvm.Context, options: *const llvm.Attribute.List.Options, call_site: bool) *llvm.Attribute.List;
pub extern fn llvm_function_set_attributes(function: *llvm.Function, attribute_list: *llvm.Attribute.List) void;
pub extern fn llvm_call_base_set_attributes(function: *llvm.Instruction.CallBase, attribute_list: *llvm.Attribute.List) void;
pub extern fn LLVMCreateEnumAttribute(context: *llvm.Context, kind: llvm.Attribute.Kind, value: u64) *llvm.Attribute;
pub extern fn LLVMCreateTypeAttribute(context: *llvm.Context, kind: llvm.Attribute.Kind, ty: *llvm.Type) *llvm.Attribute;
pub extern fn LLVMCreateConstantRangeAttribute(context: *llvm.Context, kind: llvm.Attribute.Kind, bit_count: c_uint, lower_words: [*]const u64, upper_words: [*]const u64) *llvm.Attribute;
pub extern fn LLVMCreateStringAttribute(context: *llvm.Context, key_pointer: [*]const u8, key_length: c_uint, value_pointer: [*]const u8, value_length: usize) *llvm.Attribute;
pub extern fn LLVMAddAttributeAtIndex(function: *llvm.Function, attribute_index: llvm.Attribute.Index, attribute: *llvm.Attribute) void;
pub extern fn LLVMAddCallSiteAttribute(call: *llvm.Instruction.Call, attribute_index: llvm.Attribute.Index, attribute: *llvm.Attribute) void;
// pub extern fn LLVMGetEnumAttributeKindForName(name_pointer: [*]const u8, name_length: usize) llvm.Attribute.Kind;
//
// pub extern fn LLVMCreateEnumAttribute(context: *llvm.Context, kind: llvm.Attribute.Kind, value: u64) *llvm.Attribute;
// pub extern fn LLVMCreateTypeAttribute(context: *llvm.Context, kind: llvm.Attribute.Kind, ty: *llvm.Type) *llvm.Attribute;
// pub extern fn LLVMCreateConstantRangeAttribute(context: *llvm.Context, kind: llvm.Attribute.Kind, bit_count: c_uint, lower_words: [*]const u64, upper_words: [*]const u64) *llvm.Attribute;
// pub extern fn LLVMCreateStringAttribute(context: *llvm.Context, key_pointer: [*]const u8, key_length: c_uint, value_pointer: [*]const u8, value_length: usize) *llvm.Attribute;
//
// pub extern fn LLVMAddAttributeAtIndex(function: *llvm.Function, attribute_index: llvm.Attribute.Index, attribute: *llvm.Attribute) void;
// pub extern fn LLVMAddCallSiteAttribute(call: *llvm.Instruction.Call, attribute_index: llvm.Attribute.Index, attribute: *llvm.Attribute) void;
// TYPES
// Types: integers
@ -115,7 +153,7 @@ pub extern fn LLVMFunctionType(return_type: *llvm.Type, parameter_type_pointer:
pub extern fn LLVMIsFunctionVarArg(function_type: *llvm.Type.Function) Bool;
pub extern fn LLVMGetReturnType(function_type: *llvm.Type.Function) *llvm.Type;
pub extern fn LLVMSetSubprogram(function: *llvm.Function, subprogram: *llvm.DI.Subprogram) void;
pub extern fn LLVMGetSubprogram(function: *llvm.Function) *llvm.DI.Subprogram;
pub extern fn LLVMGetSubprogram(function: *llvm.Function) ?*llvm.DI.Subprogram;
pub extern fn LLVMCountParamTypes(function_type: *llvm.Type.Function) c_uint;
pub extern fn LLVMGetParamTypes(function_type: *llvm.Type.Function, types: [*]*llvm.Type) void;
@ -142,7 +180,11 @@ pub extern fn llvm_integer_type_get_bit_count(integer_type: *llvm.Type.Integer)
// VALUES
pub extern fn LLVMGetPoison(type: *llvm.Type) *llvm.Value;
pub extern fn LLVMConstInt(type: *llvm.Type.Integer, value: c_ulonglong, sign_extend: Bool) *llvm.Constant.Integer;
pub extern fn LLVMConstArray2(element_type: *llvm.Type, value_pointer: [*]const *llvm.Constant, value_length: u64) *llvm.Constant.Array;
pub extern fn LLVMConstIntGetZExtValue(constant: *llvm.Constant) u64;
pub extern fn LLVMConstIntGetSExtValue(constant: *llvm.Constant) i64;
pub extern fn LLVMConstArray2(element_type: *llvm.Type, value_pointer: [*]const *llvm.Constant, value_length: u64) *llvm.Constant;
pub extern fn LLVMConstStructInContext(context: *llvm.Context, constant_value_pointer: [*]const *llvm.Constant, constant_value_count: c_uint, is_packed: c_uint) *llvm.Constant;
pub extern fn LLVMConstNamedStruct(struct_type: *llvm.Type.Struct, constant_value_pointer: [*]const *llvm.Constant, constant_value_count: c_uint) *llvm.Constant;
pub extern fn LLVMGetValueKind(value: *llvm.Value) llvm.Value.Kind;
pub extern fn LLVMIsConstant(value: *llvm.Value) Bool;

View File

@ -45,7 +45,7 @@ pub fn main(argc: c_int, argv: [*:null]const ?[*:0]const u8) callconv(.C) c_int
.build_mode = .debug_none,
.content = file_content,
.path = file_path,
.has_debug_info = true,
.has_debug_info = false,
.target = converter.Target.get_native(),
});
return 0;

View File

@ -280,7 +280,7 @@ require = fn (ok: u1) void
require(x.a == 4);
require(x.b == 155);
>res = c_modify_by_ref_param({ .val = 1, .arr = undefined, });
>res = c_modify_by_ref_param({ .val = 1, .arr = [ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 ] });
require(res.val == 42);
>function_pointer = &c_func_ptr_byval;

View File

@ -0,0 +1,51 @@
require = fn (ok: u1) void
{
if (!ok)
{
#trap();
}
}
MedStructInts = struct
{
x: s32,
y: s32,
z: s32,
}
bb_ret_med_struct_ints = fn [cc(c)] () MedStructInts
{
return {
.x = 1,
.y = 2,
.z = 3,
};
}
c_med_struct_ints = fn [cc(c)] (s: MedStructInts) void
{
require(s.x == 1);
require(s.y == 2);
require(s.z == 3);
>s2 = bb_ret_med_struct_ints();
require(s2.x == 1);
require(s2.y == 2);
require(s2.z == 3);
}
[export] main = fn [cc(c)] () s32
{
>med: MedStructInts = {
.x = 1,
.y = 2,
.z = 3,
};
c_med_struct_ints(med);
>med2 = bb_ret_med_struct_ints();
require(med2.x == 1);
require(med2.y == 2);
require(med2.z == 3);
return 0;
}

View File

@ -5,8 +5,19 @@ SplitStructInt = struct
c: u32,
}
require = fn (ok: u1) void
{
if (!ok)
{
#trap();
}
}
bb_split_struct_ints = fn [cc(c)] (x: SplitStructInt) void
{
require(x.a == 1234);
require(x.b == 100);
require(x.c == 1337);
}
[export] main = fn [cc(c)] () s32
@ -20,6 +31,6 @@ bb_split_struct_ints = fn [cc(c)] (x: SplitStructInt) void
bb_split_struct_ints(split);
>a: s32 = #truncate(split.a);
>b: s32 = #extend(split.b);
>c: s32 = split.c;
>c: s32 = #extend(split.c);
return a + b + 3 - c;
}

View File

@ -1,4 +1,4 @@
require = fn (ok: u1) void
require = fn [cc(c)] (ok: u1) void
{
if (!ok)
{

45
tests/indirect.bbb Normal file
View File

@ -0,0 +1,45 @@
require = fn (ok: u1) void
{
if (!ok)
{
#trap();
}
}
S = struct
{
a: u32,
b: u32,
c: u32,
d: u32,
e: u32,
f: u32,
}
ret = fn [cc(c)] () S
{
return { .a = 56, .b = 57, .c = 58, .d = 59, .e = 60, .f = 61 };
}
arg = fn [cc(c)] (s: S) void
{
require(s.a == 56);
require(s.b == 57);
require(s.c == 58);
require(s.d == 59);
require(s.e == 60);
require(s.f == 61);
}
[export] main = fn [cc(c)] () s32
{
>s = ret();
require(s.a == 56);
require(s.b == 57);
require(s.c == 58);
require(s.d == 59);
require(s.e == 60);
require(s.f == 61);
arg(s);
return 0;
}

46
tests/indirect_struct.bbb Normal file
View File

@ -0,0 +1,46 @@
Struct_u64_u64 = struct
{
a: u64,
b: u64,
}
require = fn (ok: u1) void
{
if (!ok)
{
#trap();
}
}
c_struct_u64_u64_5 = fn [cc(c)] (a: u64, b: u64, c: u64, d: u64, e: u64, s: Struct_u64_u64) void
{
require(s.a == 33);
require(s.b == 34);
}
c_struct_u64_u64_6 = fn [cc(c)] (a: u64, b: u64, c: u64, d: u64, e: u64, f: u64, s: Struct_u64_u64) void
{
require(s.a == 35);
require(s.b == 36);
}
c_struct_u64_u64_7 = fn [cc(c)] (a: u64, b: u64, c: u64, d: u64, e: u64, f: u64, g: u64, s: Struct_u64_u64) void
{
require(s.a == 37);
require(s.b == 38);
}
c_struct_u64_u64_8 = fn [cc(c)] (a: u64, b: u64, c: u64, d: u64, e: u64, f: u64, g: u64, h: u64, s: Struct_u64_u64) void
{
require(s.a == 39);
require(s.b == 40);
}
[export] main = fn [cc(c)] () s32
{
c_struct_u64_u64_5(0, 0, 0, 0, 0, { .a = 33, .b = 34, });
c_struct_u64_u64_6(0, 0, 0, 0, 0, 0, { .a = 35, .b = 36, });
c_struct_u64_u64_7(0, 0, 0, 0, 0, 0, 0, { .a = 37, .b = 38, });
c_struct_u64_u64_8(0, 0, 0, 0, 0, 0, 0, 0, { .a = 39, .b = 40, });
return 0;
}

View File

@ -7,5 +7,5 @@ modify = fn (v: &s32) void
{
>value: s32 = 0;
modify(&value);
return #extend(value == 1);
return #extend(value == 0);
}

View File

@ -0,0 +1,47 @@
SmallStructInts = struct
{
a: u8,
b: u8,
c: u8,
d: u8,
}
bb_ret_small_struct_ints = fn [cc(c)] () SmallStructInts
{
return {
.a = 1,
.b = 2,
.c = 3,
.d = 4,
};
}
require = fn(ok: u1) void
{
if (!ok)
{
#trap();
}
}
c_small_struct_ints = fn [cc(c)] (x: SmallStructInts) void
{
require(x.a == 1);
require(x.b == 2);
require(x.c == 3);
require(x.d == 4);
>y = bb_ret_small_struct_ints();
require(y.a == 1);
require(y.b == 2);
require(y.c == 3);
require(y.d == 4);
}
[export] main = fn [cc(c)] () s32
{
>s: SmallStructInts = { .a = 1, .b = 2, .c = 3, .d = 4 };
c_small_struct_ints(s);
return 0;
}

11
tests/u1_return.bbb Normal file
View File

@ -0,0 +1,11 @@
foo = fn [cc(c)] () u1
{
>result: u1 = 0;
return result;
}
[export] main = fn [cc(c)] () s32
{
>result = foo();
return #extend(result);
}

66
tests/varargs.bbb Normal file
View File

@ -0,0 +1,66 @@
S = struct
{
a: u32,
b: u32,
c: u64,
d: u64,
e: u64
}
require = fn (ok: u1) void
{
if (!ok)
{
#trap();
}
}
va_arg_function = fn [cc(c)] (first_arg: u32, ...) void
{
>va = #va_start();
>a = #va_arg(&va, u32);
>b = #va_arg(&va, S);
>c = #va_arg(&va, s64);
>d = #va_arg(&va, s32);
require(first_arg == 123456789);
require(a == 123);
require(b.a == 1);
require(b.b == 2);
require(b.c == 3);
require(b.d == 4);
require(b.e == 5);
require(c == -1);
require(d == -2);
#va_end(&va);
}
S2 = struct
{
a: u64,
b: u64,
}
va_arg_function2 = fn [cc(c)] (...) void
{
>va = #va_start();
>s2 = #va_arg(&va, S2);
require(s2.a == 8);
require(s2.b == 9);
#va_end(&va);
}
[export] main = fn [cc(c)] () s32
{
>first_arg: u32 = 123456789;
>a: u32 = 123;
>b: S = { .a = 1, .b = 2, .c = 3, .d = 4, .e = 5 };
>c: s64 = -1;
>d: s32 = -2;
va_arg_function(first_arg, a, b, c, d);
>s2: S2 = { .a = 8, .b = 9 };
va_arg_function2(s2);
return 0;
}