aboutsummaryrefslogtreecommitdiff
path: root/core/sys/darwin
diff options
context:
space:
mode:
authorcardboardguru76 <stevew@tellybug.com>2025-07-30 00:28:31 +0100
committerGitHub <noreply@github.com>2025-07-30 00:28:31 +0100
commit44f039a92f79b1af5df9ada793834db43a0faea9 (patch)
tree55ce6d242b65d9973b14655d6ce5db28b723da89 /core/sys/darwin
parent9c5e577792eff2cd9c95b590526c17a4e8762d5a (diff)
parent861fa4ab68d07bc8f3827ca29946c79c7b3f744e (diff)
Merge branch 'odin-lang:master' into darwin-foundation-support-for-handmade-hero-port
Diffstat (limited to 'core/sys/darwin')
-rw-r--r--core/sys/darwin/mach_darwin.odin345
1 files changed, 333 insertions, 12 deletions
diff --git a/core/sys/darwin/mach_darwin.odin b/core/sys/darwin/mach_darwin.odin
index 4e4ed6796..19515fe87 100644
--- a/core/sys/darwin/mach_darwin.odin
+++ b/core/sys/darwin/mach_darwin.odin
@@ -5,15 +5,34 @@ foreign import mach "system:System"
import "core:c"
import "base:intrinsics"
+mach_port_t :: distinct c.uint
+task_t :: mach_port_t
+
+semaphore_t :: distinct u64
+
kern_return_t :: distinct c.int
+thread_act_t :: distinct u64
+thread_state_t :: distinct ^u32
+thread_list_t :: [^]thread_act_t
+vm_region_recurse_info_t :: distinct ^i32
+task_info_t :: distinct ^i32
+
+MACH_PORT_NULL :: 0
+MACH_PORT_DEAD :: ~mach_port_t(0)
+
+MACH_MSG_PORT_DESCRIPTOR :: 0
+
+X86_THREAD_STATE32 :: 1
+X86_THREAD_STATE64 :: 4
+ARM_THREAD_STATE64 :: 6
+
+mach_msg_option_t :: distinct i32
+name_t :: distinct cstring
-mach_port_t :: distinct c.uint
vm_map_t :: mach_port_t
mem_entry_name_port_t :: mach_port_t
ipc_space_t :: mach_port_t
thread_t :: mach_port_t
-task_t :: mach_port_t
-semaphore_t :: mach_port_t
vm_size_t :: distinct c.uintptr_t
@@ -29,11 +48,279 @@ vm_inherit_t :: distinct c.uint
mach_port_name_t :: distinct c.uint
+mach_port_right_t :: distinct c.uint
+
sync_policy_t :: distinct c.int
+mach_msg_port_descriptor_t :: struct {
+ name: mach_port_t,
+ _: u32,
+ using _: bit_field u32 {
+ _: u32 | 16,
+ disposition: u32 | 8,
+ type: u32 | 8,
+ },
+}
+
+Task_Port_Type :: enum u32 {
+ Kernel = 1,
+ Host,
+ Name,
+ Bootstrap,
+ Seatbelt = 7,
+ Access = 9,
+}
+
+Bootstrap_Error :: enum u32 {
+ Success,
+ Not_Privileged = 1100,
+ Name_In_Use = 1101,
+ Unknown_Service = 1102,
+ Service_Active = 1103,
+ Bad_Count = 1104,
+ No_Memory = 1105,
+ No_Children = 1106,
+}
+
+Msg_Type :: enum u32 {
+ Unstructured = 0,
+ Bit = 0,
+ Boolean = 0,
+ Integer_16 = 1,
+ Integer_32 = 2,
+ Char = 8,
+ Byte = 9,
+ Integer_8 = 9,
+ Real = 10,
+ Integer_64 = 11,
+ String = 12,
+ String_C = 12,
+
+ Port_Name = 15,
+
+ Move_Receive = 16,
+ Port_Receive = 16,
+ Move_Send = 17,
+ Port_Send = 17,
+ Move_Send_Once = 18,
+ Port_Send_Once = 18,
+ Copy_Send = 19,
+ Make_Send = 20,
+ Make_Send_Once = 21,
+}
+
+Msg_Header_Bits :: enum u32 {
+ Zero = 0,
+ Remote_Mask = 0xff,
+ Local_Mask = 0xff00,
+ Migrated = 0x08000000,
+ Unused = 0x07ff0000,
+ Complex_Data = 0x10000000,
+ Complex_Ports = 0x20000000,
+ Circular = 0x40000000,
+ Complex = 0x80000000,
+}
+
+mach_msg_type_t :: struct {
+ using _: bit_field u32 {
+ name: u32 | 8,
+ size: u32 | 8,
+ number: u32 | 12,
+ inline: u32 | 1,
+ longform: u32 | 1,
+ deallocate: u32 | 1,
+ unused: u32 | 1,
+ },
+}
+
+mach_msg_header_t :: struct {
+ msgh_bits: u32,
+ msgh_size: u32,
+ msgh_remote_port: mach_port_t,
+ msgh_local_port: mach_port_t,
+ msgh_voucher_port: u32,
+ msgh_id: i32,
+}
+
+mach_msg_body_t :: struct {
+ msgh_descriptor_count: u32,
+}
+
+mach_msg_trailer_t :: struct {
+ msgh_trailer_type: u32,
+ msgh_trailer_size: u32,
+}
+
+x86_thread_state32_t :: struct {
+ eax: u32,
+ ebx: u32,
+ ecx: u32,
+ edx: u32,
+ edi: u32,
+ esi: u32,
+ ebp: u32,
+ esp: u32,
+ ss: u32,
+ eflags: u32,
+ eip: u32,
+ cs: u32,
+ ds: u32,
+ es: u32,
+ fs: u32,
+ gs: u32,
+}
+X86_THREAD_STATE32_COUNT :: size_of(x86_thread_state32_t) / size_of(u32)
+
+x86_thread_state64_t :: struct #packed {
+ rax: u64,
+ rbx: u64,
+ rcx: u64,
+ rdx: u64,
+ rdi: u64,
+ rsi: u64,
+ rbp: u64,
+ rsp: u64,
+ r8: u64,
+ r9: u64,
+ r10: u64,
+ r11: u64,
+ r12: u64,
+ r13: u64,
+ r14: u64,
+ r15: u64,
+ rip: u64,
+ rflags: u64,
+ cs: u64,
+ fs: u64,
+ gs: u64,
+}
+X86_THREAD_STATE64_COUNT :: size_of(x86_thread_state64_t) / size_of(u32)
+
+arm_thread_state64_t :: struct #packed {
+ x: [29]u64,
+ fp: u64,
+ lr: u64,
+ sp: u64,
+ pc: u64,
+ cpsr: u32,
+ pad: u32,
+}
+ARM_THREAD_STATE64_COUNT :: size_of(arm_thread_state64_t) / size_of(u32)
+
+THREAD_IDENTIFIER_INFO :: 4
+thread_identifier_info :: struct {
+ thread_id: u64,
+ thread_handler: u64,
+ dispatch_qaddr: u64,
+}
+THREAD_IDENTIFIER_INFO_COUNT :: size_of(thread_identifier_info) / size_of(u32)
+
+vm_region_submap_info_64 :: struct {
+ protection: u32,
+ max_protection: u32,
+ inheritance: u32,
+ offset: u64,
+ user_tag: u32,
+ pages_residept: u32,
+ pages_shared_now_private: u32,
+ pages_swapped_out: u32,
+ pages_dirtied: u32,
+ ref_count: u32,
+ shadow_depth: u16,
+ external_pager: u8,
+ share_mode: u8,
+ is_submap: b32,
+ behavior: i32,
+ object_id: u32,
+ user_wired_count: u16,
+ pages_reusable: u32,
+}
+VM_REGION_SUBMAP_INFO_COUNT_64 :: size_of(vm_region_submap_info_64) / size_of(u32)
+
+TASK_DYLD_INFO :: 17
+task_dyld_info :: struct {
+ all_image_info_addr: u64,
+ all_image_info_size: u64,
+ all_image_info_format: i32,
+}
+TASK_DYLD_INFO_COUNT :: size_of(task_dyld_info) / size_of(u32)
+
+dyld_image_info :: struct {
+ image_load_addr: u64,
+ image_file_path: cstring,
+ image_file_mod_date: u64,
+}
+
+dyld_uuid_info :: struct {
+ image_load_addr: u64,
+ image_uuid: [16]u8,
+}
+
+dyld_all_image_infos :: struct {
+ version: u32,
+ info_array_count: u32,
+ info_array: rawptr,
+ notification: rawptr,
+ process_detached_from_shared_region: b32,
+ libSystem_initialized: b32,
+ dyld_image_load_addr: u64,
+ jit_info: rawptr,
+ dyld_version: cstring,
+ error_message: cstring,
+ termination_flags: u64,
+ core_symbolication_shm_page: rawptr,
+ system_order_flag: u64,
+ uuid_array_count: u64,
+ uuid_array: rawptr,
+ dyld_all_image_infos_addr: u64,
+ initial_image_count: u64,
+ error_kind: u64,
+ error_client_of_dylib_path: cstring,
+ error_target_dylib_path: cstring,
+ error_symbol: cstring,
+ shared_cache_slide: u64,
+ shared_cache_uuid: [16]u8,
+ shared_cache_base_addr: u64,
+ info_array_change_timestamp: u64,
+ dyld_path: cstring,
+ notify_ports: [8]mach_port_t,
+ reserved: [7]u64,
+ shared_cache_fsid: u64,
+ shared_cache_fsobjid: u64,
+ compact_dyld_image_info_addr: u64,
+ compact_dyld_image_info_size: u64,
+ platform: u32,
+ aot_info_count: u32,
+ aot_info_array: rawptr,
+ aot_info_array_change_timestamp: u64,
+ aot_shared_cache_base_address: u64,
+ aot_shared_cache_uuid: [16]u8,
+}
+
+
@(default_calling_convention="c")
foreign mach {
- mach_task_self :: proc() -> mach_port_t ---
+ mach_task_self :: proc() -> mach_port_t ---
+ mach_msg :: proc(header: rawptr, option: Msg_Option_Flags, send_size: u32, receive_limit: u32, receive_name: mach_port_t, timeout: u32, notify: mach_port_t) -> Kern_Return ---
+ mach_msg_send :: proc(header: rawptr) -> Kern_Return ---
+ mach_vm_allocate :: proc(target_task: task_t, adddress: u64, size: u64, flags: i32) -> Kern_Return ---
+ mach_vm_deallocate :: proc(target_task: task_t, adddress: ^u64, size: u64) -> Kern_Return ---
+ mach_vm_remap :: proc(target_task: task_t, page: rawptr, size: u64, mask: u64, flags: i32, src_task: task_t, src_address: u64, copy: b32, cur_protection: ^i32, max_protection: ^i32, inheritance: VM_Inherit) -> Kern_Return ---
+ mach_vm_region_recurse :: proc(target_task: task_t, address: ^u64, size: ^u64, depth: ^u32, info: vm_region_recurse_info_t, count: ^u32) -> Kern_Return ---
+ vm_page_size: u64
+ vm_page_mask: u64
+ vm_page_shift: i32
+
+ mach_port_allocate :: proc(task: task_t, right: Port_Right, name: rawptr) -> Kern_Return ---
+ mach_port_deallocate :: proc(task: task_t, name: u32) -> Kern_Return ---
+ mach_port_extract_right :: proc(task: task_t, name: u32, msgt_name: u32, poly: ^mach_port_t, poly_poly: ^mach_port_t) -> Kern_Return ---
+
+ task_get_special_port :: proc(task: task_t, port: i32, special_port: ^mach_port_t) -> Kern_Return ---
+ task_suspend :: proc(task: task_t) -> Kern_Return ---
+ task_resume :: proc(task: task_t) -> Kern_Return ---
+ task_threads :: proc(task: task_t, thread_list: ^thread_list_t, list_count: ^u32) -> Kern_Return ---
+ task_info :: proc(task: task_t, flavor: i32, info: task_info_t, count: ^u32) -> Kern_Return ---
+ task_terminate :: proc(task: task_t) -> Kern_Return ---
semaphore_create :: proc(task: task_t, semaphore: ^semaphore_t, policy: Sync_Policy, value: c.int) -> Kern_Return ---
semaphore_destroy :: proc(task: task_t, semaphore: semaphore_t) -> Kern_Return ---
@@ -44,9 +331,11 @@ foreign mach {
semaphore_wait :: proc(semaphore: semaphore_t) -> Kern_Return ---
- vm_allocate :: proc (target_task : vm_map_t, address: ^vm_address_t, size: vm_size_t, flags: VM_Flags) -> Kern_Return ---
+ thread_get_state :: proc(thread: thread_act_t, flavor: i32, thread_state: thread_state_t, old_state_count: ^u32) -> Kern_Return ---
+ thread_info :: proc(thread: thread_act_t, flavor: u32, thread_info: ^thread_identifier_info, info_count: ^u32) -> Kern_Return ---
- vm_deallocate :: proc(target_task: vm_map_t, address: vm_address_t, size: vm_size_t) -> Kern_Return ---
+ bootstrap_register2 :: proc(bp: mach_port_t, service_name: name_t, sp: mach_port_t, flags: u64) -> Kern_Return ---
+ bootstrap_look_up :: proc(bp: mach_port_t, service_name: name_t, sp: ^mach_port_t) -> Kern_Return ---
vm_map :: proc(
target_task: vm_map_t,
@@ -70,14 +359,9 @@ foreign mach {
object_handle: ^mem_entry_name_port_t,
parent_entry: mem_entry_name_port_t,
) -> Kern_Return ---
+}
- mach_port_deallocate :: proc(
- task: ipc_space_t,
- name: mach_port_name_t,
- ) -> Kern_Return ---
- vm_page_size: vm_size_t
-}
Kern_Return :: enum kern_return_t {
Success,
@@ -501,6 +785,39 @@ VM_PROT_DEFAULT :: VM_Prot_Flags{.Read, .Write}
VM_PROT_ALL :: VM_Prot_Flags{.Read, .Write, .Execute}
/*
+ * Mach msg options, defined as bits within the mach_msg_option_t type
+ */
+
+Msg_Option :: enum mach_msg_option_t {
+ Send_Msg,
+ Receive_Msg,
+
+ Send_Timeout = LOG2(0x10),
+ Send_Notify = LOG2(0x20),
+ Send_Interrupt = LOG2(0x40),
+ Send_Cancel = LOG2(0x80),
+ Receive_Timeout = LOG2(0x100),
+ Receive_Notify = LOG2(0x200),
+ Receive_Interrupt = LOG2(0x400),
+ Receive_Large = LOG2(0x800),
+ Send_Always = LOG2(0x10000),
+}
+
+Msg_Option_Flags :: distinct bit_set[Msg_Option; mach_msg_option_t]
+
+/*
+ * Enumeration of valid values for mach_port_right_t
+ */
+
+Port_Right :: enum mach_port_right_t {
+ Send,
+ Receive,
+ Send_Once,
+ Port_Set,
+ Dead_Name,
+}
+
+/*
* Enumeration of valid values for vm_inherit_t.
*/
@@ -522,3 +839,7 @@ Sync_Policy :: enum sync_policy_t {
Lifo = Fifo | Reversed,
}
+
+mach_vm_trunc_page :: proc(v: u64) -> u64 {
+ return v & ~vm_page_mask
+}