mirror of
https://github.com/odin-lang/Odin.git
synced 2025-12-28 17:04:34 +00:00
Fix @objc_implement methods not respecting @objc_is_class_method
Fix incorrect type encoding for objc_class
This commit is contained in:
@@ -31,5 +31,6 @@ foreign ObjC {
|
||||
class_getInstanceVariable :: proc "c" (cls : objc_Class, name: cstring) -> objc_Ivar ---
|
||||
class_getInstanceSize :: proc "c" (cls : objc_Class) -> uint ---
|
||||
ivar_getOffset :: proc "c" (v: objc_Ivar) -> uintptr ---
|
||||
object_getClass :: proc "c" (obj: objc_id) -> objc_Class ---
|
||||
}
|
||||
|
||||
|
||||
@@ -1300,18 +1300,8 @@ String lb_get_objc_type_encoding(Type *t, isize pointer_depth = 0) {
|
||||
|
||||
const bool is_union = base->kind == Type_Union;
|
||||
if (!is_union) {
|
||||
// Check for objc_SEL
|
||||
if (internal_check_is_assignable_to(base, t_objc_SEL)) {
|
||||
return str_lit(":");
|
||||
}
|
||||
|
||||
// Check for objc_Class
|
||||
if (internal_check_is_assignable_to(base, t_objc_SEL)) {
|
||||
return str_lit("#");
|
||||
}
|
||||
|
||||
// Treat struct as an Objective-C Class?
|
||||
if (has_type_got_objc_class_attribute(base) && pointer_depth == 0) {
|
||||
if (has_type_got_objc_class_attribute(t) && pointer_depth == 0) {
|
||||
return str_lit("#");
|
||||
}
|
||||
}
|
||||
@@ -1354,6 +1344,17 @@ String lb_get_objc_type_encoding(Type *t, isize pointer_depth = 0) {
|
||||
return str_lit("?");
|
||||
|
||||
case Type_Pointer: {
|
||||
// NOTE: These types are pointers, so we must check here for special cases
|
||||
// Check for objc_SEL
|
||||
if (internal_check_is_assignable_to(t, t_objc_SEL)) {
|
||||
return str_lit(":");
|
||||
}
|
||||
|
||||
// Check for objc_Class
|
||||
if (internal_check_is_assignable_to(t, t_objc_Class)) {
|
||||
return str_lit("#");
|
||||
}
|
||||
|
||||
String pointee = lb_get_objc_type_encoding(t->Pointer.elem, pointer_depth +1);
|
||||
// Special case for Objective-C Objects
|
||||
if (pointer_depth == 0 && pointee == "@") {
|
||||
@@ -1645,6 +1646,21 @@ gb_internal void lb_finalize_objc_names(lbGenerator *gen, lbProcedure *p) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check if it has any class methods ahead of time so that we know to grab the meta_class
|
||||
lbValue meta_class_value = {};
|
||||
for (const ObjcMethodData &md : *methods) {
|
||||
if (!md.ac.objc_is_class_method) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// Get the meta_class
|
||||
args.count = 1;
|
||||
args[0] = class_value;
|
||||
meta_class_value = lb_emit_runtime_call(p, "object_getClass", args);
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
for (const ObjcMethodData &md : *methods) {
|
||||
GB_ASSERT( md.proc_entity->kind == Entity_Procedure);
|
||||
Type *method_type = md.proc_entity->type;
|
||||
@@ -1770,8 +1786,10 @@ gb_internal void lb_finalize_objc_names(lbGenerator *gen, lbProcedure *p) {
|
||||
GB_ASSERT(sel_address);
|
||||
lbValue selector_value = lb_addr_load(p, *sel_address);
|
||||
|
||||
lbValue target_class = !md.ac.objc_is_class_method ? class_value : meta_class_value;
|
||||
|
||||
args.count = 4;
|
||||
args[0] = class_value; // Class
|
||||
args[0] = target_class; // Class
|
||||
args[1] = selector_value; // SEL
|
||||
args[2] = lbValue { wrapper_proc->value, wrapper_proc->type };
|
||||
args[3] = lb_const_value(m, t_cstring, exact_value_string(method_encoding));
|
||||
|
||||
Reference in New Issue
Block a user