aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorgingerBill <bill@gingerbill.org>2025-01-30 10:37:21 +0000
committergingerBill <bill@gingerbill.org>2025-01-30 10:37:21 +0000
commit5efa3510cfeebf498f78b35a213bab0c1014a85e (patch)
tree8d0040882adf0bf2a6c5a5a8a91d99906b5664da /src
parent998d6525b7e1cc046f9cf58577fa71ee88b7c9a0 (diff)
parent2656ecd4e17d448f1d972270bde87f75bc096d0d (diff)
Merge branch 'master' into bill/sdl3
Diffstat (limited to 'src')
-rw-r--r--src/big_int.cpp5
-rw-r--r--src/checker.cpp9
-rw-r--r--src/checker.hpp1
-rw-r--r--src/entity.cpp1
-rw-r--r--src/linker.cpp24
-rw-r--r--src/llvm_backend.cpp4
-rw-r--r--src/llvm_backend.hpp9
-rw-r--r--src/llvm_backend_expr.cpp21
-rw-r--r--src/llvm_backend_utility.cpp88
9 files changed, 105 insertions, 57 deletions
diff --git a/src/big_int.cpp b/src/big_int.cpp
index 8e476f090..0b0a9a400 100644
--- a/src/big_int.cpp
+++ b/src/big_int.cpp
@@ -251,7 +251,10 @@ gb_internal void big_int_from_string(BigInt *dst, String const &s, bool *success
exp *= 10;
exp += v;
}
- big_int_exp_u64(dst, &b, exp, success);
+ BigInt tmp = {};
+ mp_init(&tmp);
+ big_int_exp_u64(&tmp, &b, exp, success);
+ big_int_mul_eq(dst, &tmp);
}
if (is_negative) {
diff --git a/src/checker.cpp b/src/checker.cpp
index 85077a5c5..baa1e0d2b 100644
--- a/src/checker.cpp
+++ b/src/checker.cpp
@@ -5040,6 +5040,12 @@ gb_internal DECL_ATTRIBUTE_PROC(foreign_import_decl_attribute) {
ac->extra_linker_flags = ev.value_string;
}
return true;
+ } else if (name == "ignore_duplicates") {
+ if (value != nullptr) {
+ error(elem, "Expected no parameter for '%.*s'", LIT(name));
+ }
+ ac->ignore_duplicates = true;
+ return true;
}
return false;
}
@@ -5190,6 +5196,9 @@ gb_internal void check_add_foreign_import_decl(CheckerContext *ctx, Ast *decl) {
if (ac.foreign_import_priority_index != 0) {
e->LibraryName.priority_index = ac.foreign_import_priority_index;
}
+ if (ac.ignore_duplicates) {
+ e->LibraryName.ignore_duplicates = true;
+ }
String extra_linker_flags = string_trim_whitespace(ac.extra_linker_flags);
if (extra_linker_flags.len != 0) {
e->LibraryName.extra_linker_flags = extra_linker_flags;
diff --git a/src/checker.hpp b/src/checker.hpp
index 3951fcefe..4634047c0 100644
--- a/src/checker.hpp
+++ b/src/checker.hpp
@@ -140,6 +140,7 @@ struct AttributeContext {
bool instrumentation_enter : 1;
bool instrumentation_exit : 1;
bool rodata : 1;
+ bool ignore_duplicates : 1;
u32 optimization_mode; // ProcedureOptimizationMode
i64 foreign_import_priority_index;
String extra_linker_flags;
diff --git a/src/entity.cpp b/src/entity.cpp
index 802b381f9..d137a8674 100644
--- a/src/entity.cpp
+++ b/src/entity.cpp
@@ -274,6 +274,7 @@ struct Entity {
Slice<String> paths;
String name;
i64 priority_index;
+ bool ignore_duplicates;
String extra_linker_flags;
} LibraryName;
i32 Nil;
diff --git a/src/linker.cpp b/src/linker.cpp
index 59e6d8dc1..cf2ef638d 100644
--- a/src/linker.cpp
+++ b/src/linker.cpp
@@ -449,6 +449,26 @@ gb_internal i32 linker_stage(LinkerData *gen) {
if (extra_linker_flags.len != 0) {
lib_str = gb_string_append_fmt(lib_str, " %.*s", LIT(extra_linker_flags));
}
+
+ if (build_context.metrics.os == TargetOs_darwin) {
+ // Print frameworks first
+ for (String lib : e->LibraryName.paths) {
+ lib = string_trim_whitespace(lib);
+ if (lib.len == 0) {
+ continue;
+ }
+ if (string_ends_with(lib, str_lit(".framework"))) {
+ if (string_set_update(&min_libs_set, lib)) {
+ continue;
+ }
+
+ String lib_name = lib;
+ lib_name = remove_extension_from_path(lib_name);
+ lib_str = gb_string_append_fmt(lib_str, " -framework %.*s ", LIT(lib_name));
+ }
+ }
+ }
+
for (String lib : e->LibraryName.paths) {
lib = string_trim_whitespace(lib);
if (lib.len == 0) {
@@ -541,7 +561,9 @@ gb_internal i32 linker_stage(LinkerData *gen) {
short_circuit = true;
} else if (string_ends_with(lib, str_lit(".dylib"))) {
short_circuit = true;
- } else if (string_ends_with(lib, str_lit(".so"))) {
+ } else if (string_ends_with(lib, str_lit(".so"))) {
+ short_circuit = true;
+ } else if (e->LibraryName.ignore_duplicates) {
short_circuit = true;
}
diff --git a/src/llvm_backend.cpp b/src/llvm_backend.cpp
index 696ced0df..29fa67f3f 100644
--- a/src/llvm_backend.cpp
+++ b/src/llvm_backend.cpp
@@ -1147,14 +1147,14 @@ gb_internal void lb_finalize_objc_names(lbProcedure *p) {
String name = entry.key;
args[0] = lb_const_value(m, t_cstring, exact_value_string(name));
lbValue ptr = lb_emit_runtime_call(p, "objc_lookUpClass", args);
- lb_addr_store(p, entry.value, ptr);
+ lb_addr_store(p, entry.value.local_module_addr, ptr);
}
for (auto const &entry : m->objc_selectors) {
String name = entry.key;
args[0] = lb_const_value(m, t_cstring, exact_value_string(name));
lbValue ptr = lb_emit_runtime_call(p, "sel_registerName", args);
- lb_addr_store(p, entry.value, ptr);
+ lb_addr_store(p, entry.value.local_module_addr, ptr);
}
lb_end_procedure_body(p);
diff --git a/src/llvm_backend.hpp b/src/llvm_backend.hpp
index 42d283a1e..a0775ac3b 100644
--- a/src/llvm_backend.hpp
+++ b/src/llvm_backend.hpp
@@ -143,6 +143,11 @@ struct lbPadType {
LLVMTypeRef type;
};
+struct lbObjcRef {
+ Entity * entity;
+ lbAddr local_module_addr;
+};
+
struct lbModule {
LLVMModuleRef mod;
LLVMContextRef ctx;
@@ -196,8 +201,8 @@ struct lbModule {
RecursiveMutex debug_values_mutex;
PtrMap<void *, LLVMMetadataRef> debug_values;
- StringMap<lbAddr> objc_classes;
- StringMap<lbAddr> objc_selectors;
+ StringMap<lbObjcRef> objc_classes;
+ StringMap<lbObjcRef> objc_selectors;
PtrMap<Type *, lbAddr> map_cell_info_map; // address of runtime.Map_Info
PtrMap<Type *, lbAddr> map_info_map; // address of runtime.Map_Cell_Info
diff --git a/src/llvm_backend_expr.cpp b/src/llvm_backend_expr.cpp
index df9dca801..871536927 100644
--- a/src/llvm_backend_expr.cpp
+++ b/src/llvm_backend_expr.cpp
@@ -4294,6 +4294,17 @@ gb_internal lbAddr lb_build_addr_index_expr(lbProcedure *p, Ast *expr) {
gb_internal lbAddr lb_build_addr_slice_expr(lbProcedure *p, Ast *expr) {
ast_node(se, SliceExpr, expr);
+ lbAddr addr = lb_build_addr(p, se->expr);
+ lbValue base = lb_addr_load(p, addr);
+ Type *type = base_type(base.type);
+
+ if (is_type_pointer(type)) {
+ type = base_type(type_deref(type));
+ addr = lb_addr(base);
+ base = lb_addr_load(p, addr);
+ }
+
+
lbValue low = lb_const_int(p->module, t_int, 0);
lbValue high = {};
@@ -4306,16 +4317,6 @@ gb_internal lbAddr lb_build_addr_slice_expr(lbProcedure *p, Ast *expr) {
bool no_indices = se->low == nullptr && se->high == nullptr;
- lbAddr addr = lb_build_addr(p, se->expr);
- lbValue base = lb_addr_load(p, addr);
- Type *type = base_type(base.type);
-
- if (is_type_pointer(type)) {
- type = base_type(type_deref(type));
- addr = lb_addr(base);
- base = lb_addr_load(p, addr);
- }
-
switch (type->kind) {
case Type_Slice: {
Type *slice_type = type;
diff --git a/src/llvm_backend_utility.cpp b/src/llvm_backend_utility.cpp
index 7b7c9d6e9..8910bd67a 100644
--- a/src/llvm_backend_utility.cpp
+++ b/src/llvm_backend_utility.cpp
@@ -2093,23 +2093,34 @@ gb_internal void lb_set_wasm_export_attributes(LLVMValueRef value, String export
gb_internal lbAddr lb_handle_objc_find_or_register_selector(lbProcedure *p, String const &name) {
- lbAddr *found = string_map_get(&p->module->objc_selectors, name);
+ lbObjcRef *found = string_map_get(&p->module->objc_selectors, name);
if (found) {
- return *found;
- } else {
- lbModule *default_module = &p->module->gen->default_module;
- Entity *e = nullptr;
- lbAddr default_addr = lb_add_global_generated(default_module, t_objc_SEL, {}, &e);
+ return found->local_module_addr;
+ }
- lbValue ptr = lb_find_value_from_entity(p->module, e);
- lbAddr local_addr = lb_addr(ptr);
+ lbModule *default_module = &p->module->gen->default_module;
+ Entity *entity = {};
- string_map_set(&default_module->objc_selectors, name, default_addr);
- if (default_module != p->module) {
- string_map_set(&p->module->objc_selectors, name, local_addr);
+ if (default_module != p->module) {
+ found = string_map_get(&default_module->objc_selectors, name);
+ if (found) {
+ entity = found->entity;
}
- return local_addr;
}
+
+ if (!entity) {
+ lbAddr default_addr = lb_add_global_generated(default_module, t_objc_SEL, {}, &entity);
+ string_map_set(&default_module->objc_selectors, name, lbObjcRef{entity, default_addr});
+ }
+
+ lbValue ptr = lb_find_value_from_entity(p->module, entity);
+ lbAddr local_addr = lb_addr(ptr);
+
+ if (default_module != p->module) {
+ string_map_set(&p->module->objc_selectors, name, lbObjcRef{entity, local_addr});
+ }
+
+ return local_addr;
}
gb_internal lbValue lb_handle_objc_find_selector(lbProcedure *p, Ast *expr) {
@@ -2139,23 +2150,34 @@ gb_internal lbValue lb_handle_objc_register_selector(lbProcedure *p, Ast *expr)
}
gb_internal lbAddr lb_handle_objc_find_or_register_class(lbProcedure *p, String const &name) {
- lbAddr *found = string_map_get(&p->module->objc_classes, name);
+ lbObjcRef *found = string_map_get(&p->module->objc_classes, name);
if (found) {
- return *found;
- } else {
- lbModule *default_module = &p->module->gen->default_module;
- Entity *e = nullptr;
- lbAddr default_addr = lb_add_global_generated(default_module, t_objc_SEL, {}, &e);
+ return found->local_module_addr;
+ }
- lbValue ptr = lb_find_value_from_entity(p->module, e);
- lbAddr local_addr = lb_addr(ptr);
+ lbModule *default_module = &p->module->gen->default_module;
+ Entity *entity = {};
- string_map_set(&default_module->objc_classes, name, default_addr);
- if (default_module != p->module) {
- string_map_set(&p->module->objc_classes, name, local_addr);
+ if (default_module != p->module) {
+ found = string_map_get(&default_module->objc_classes, name);
+ if (found) {
+ entity = found->entity;
}
- return local_addr;
}
+
+ if (!entity) {
+ lbAddr default_addr = lb_add_global_generated(default_module, t_objc_Class, {}, &entity);
+ string_map_set(&default_module->objc_classes, name, lbObjcRef{entity, default_addr});
+ }
+
+ lbValue ptr = lb_find_value_from_entity(p->module, entity);
+ lbAddr local_addr = lb_addr(ptr);
+
+ if (default_module != p->module) {
+ string_map_set(&p->module->objc_classes, name, lbObjcRef{entity, local_addr});
+ }
+
+ return local_addr;
}
gb_internal lbValue lb_handle_objc_find_class(lbProcedure *p, Ast *expr) {
@@ -2196,23 +2218,7 @@ gb_internal lbValue lb_handle_objc_id(lbProcedure *p, Ast *expr) {
GB_ASSERT(e->kind == Entity_TypeName);
String name = e->TypeName.objc_class_name;
- lbAddr *found = string_map_get(&p->module->objc_classes, name);
- if (found) {
- return lb_addr_load(p, *found);
- } else {
- lbModule *default_module = &p->module->gen->default_module;
- Entity *e = nullptr;
- lbAddr default_addr = lb_add_global_generated(default_module, t_objc_Class, {}, &e);
-
- lbValue ptr = lb_find_value_from_entity(p->module, e);
- lbAddr local_addr = lb_addr(ptr);
-
- string_map_set(&default_module->objc_classes, name, default_addr);
- if (default_module != p->module) {
- string_map_set(&p->module->objc_classes, name, local_addr);
- }
- return lb_addr_load(p, local_addr);
- }
+ return lb_addr_load(p, lb_handle_objc_find_or_register_class(p, name));
}
return lb_build_expr(p, expr);