1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
use core::ffi::c_void;
#[cfg(not(objfw))]
use std::os::raw::c_char;
use std::os::raw::c_int;
#[cfg(apple)]
use std::os::raw::c_uint;

#[cfg(not(objfw))]
use crate::{objc_AssociationPolicy, BOOL};
use crate::{objc_object, OpaqueData};

/// An opaque type that represents an instance variable.
#[repr(C)]
pub struct objc_ivar {
    _priv: [u8; 0],
    _p: OpaqueData,
}

/// A nullable pointer to the start of a method implementation.
///
/// Not all APIs are guaranteed to take NULL values; read the docs!
pub type IMP = Option<unsafe extern "C" fn()>;

// /// Not available on macOS x86.
// ///
// /// Remember that this is non-null!
// #[cfg(apple_new)]
// pub type objc_hook_getClass =
//     unsafe extern "C" fn(name: *const c_char, out_cls: *mut *const crate::objc_class) -> BOOL;
//
// /// Not available on macOS x86.
// ///
// /// Remember that this is non-null!
// #[cfg(apple_new)]
// pub type objc_hook_lazyClassNamer =
//     unsafe extern "C" fn(cls: *const crate::objc_class) -> *const c_char;

extern_c! {
    #[cfg(not(objfw))]
    pub fn imp_getBlock(imp: IMP) -> *mut objc_object;
    #[cfg(not(objfw))]
    pub fn imp_implementationWithBlock(block: *mut objc_object) -> IMP;
    #[cfg(not(objfw))]
    pub fn imp_removeBlock(imp: IMP) -> BOOL;

    #[cfg(not(objfw))]
    pub fn ivar_getName(ivar: *const objc_ivar) -> *const c_char;
    #[cfg(not(objfw))]
    pub fn ivar_getOffset(ivar: *const objc_ivar) -> isize;
    #[cfg(not(objfw))]
    pub fn ivar_getTypeEncoding(ivar: *const objc_ivar) -> *const c_char;

    #[cfg(apple)]
    pub fn objc_copyClassNamesForImage(
        image: *const c_char,
        out_len: *mut c_uint,
    ) -> *mut *const c_char;
    #[cfg(apple)]
    pub fn objc_copyImageNames(out_len: *mut c_uint) -> *mut *const c_char;

    // Instead of being able to change this, it's a weak symbol on GNUStep.
    #[cfg(any(apple, objfw))]
    pub fn objc_enumerationMutation(obj: *mut objc_object);
    #[cfg(any(apple, objfw))]
    pub fn objc_setEnumerationMutationHandler(
        handler: Option<unsafe extern "C" fn(obj: *mut objc_object)>,
    );

    #[cfg(not(objfw))]
    pub fn objc_getAssociatedObject(
        object: *const objc_object,
        key: *const c_void,
    ) -> *const objc_object;
    #[cfg(not(objfw))]
    pub fn objc_setAssociatedObject(
        object: *mut objc_object,
        key: *const c_void,
        value: *mut objc_object,
        policy: objc_AssociationPolicy,
    );
    #[cfg(not(objfw))]
    pub fn objc_removeAssociatedObjects(object: *mut objc_object);

    #[cfg(any(apple, objfw))]
    pub fn objc_setForwardHandler(fwd: *mut c_void, fwd_stret: *mut c_void);
    // These two are defined in:
    // - Apple: objc-sync.h
    // - GNUStep: dtable.h / associate.m
    // - ObjFW: ObjFW-RT.h
    pub fn objc_sync_enter(obj: *mut objc_object) -> c_int;
    pub fn objc_sync_exit(obj: *mut objc_object) -> c_int;

    // Available in macOS 10.14.4
    // /// Not available on macOS x86.
    // ///
    // /// Remember that this is non-null!
    // #[cfg(apple_new)]
    // pub fn objc_setHook_getClass(
    //     new_value: objc_hook_getClass,
    //     out_old_value: *mut objc_hook_getClass,
    // );
    // Available in macOS 11
    // /// Not available on macOS x86.
    // ///
    // /// Remember that this is non-null!
    // #[cfg(apple_new)]
    // pub fn objc_setHook_lazyClassNamer(
    //     new_value: objc_hook_lazyClassNamer,
    //     out_old_value: *mut objc_hook_lazyClassNamer,
    // );

    // #[deprecated = "not recommended"]
    // #[cfg(apple)]
    // pub fn _objc_flush_caches

    // #[cfg(gnustep)]
    // objc_test_capability
}