mirror of
https://github.com/odin-lang/Odin.git
synced 2025-12-29 17:34:34 +00:00
83 lines
1.5 KiB
Odin
83 lines
1.5 KiB
Odin
//+private
|
|
//+build openbsd
|
|
package sync
|
|
|
|
import "core:c"
|
|
import "core:time"
|
|
|
|
FUTEX_WAIT :: 1
|
|
FUTEX_WAKE :: 2
|
|
|
|
FUTEX_PRIVATE_FLAG :: 128
|
|
|
|
FUTEX_WAIT_PRIVATE :: (FUTEX_WAIT | FUTEX_PRIVATE_FLAG)
|
|
FUTEX_WAKE_PRIVATE :: (FUTEX_WAKE | FUTEX_PRIVATE_FLAG)
|
|
|
|
ETIMEDOUT :: 60
|
|
|
|
|
|
foreign import libc "system:c"
|
|
|
|
foreign libc {
|
|
@(link_name="futex")
|
|
_unix_futex :: proc "c" (f: ^Futex, op: c.int, val: u32, timeout: rawptr) -> c.int ---
|
|
|
|
@(link_name="__errno") __errno :: proc() -> ^int ---
|
|
}
|
|
|
|
_futex_wait :: proc "contextless" (f: ^Futex, expected: u32) -> bool {
|
|
res := _unix_futex(f, FUTEX_WAIT_PRIVATE, expected, nil)
|
|
|
|
if res != -1 {
|
|
return true
|
|
}
|
|
|
|
if __errno()^ == ETIMEDOUT {
|
|
return false
|
|
}
|
|
|
|
_panic("futex_wait failure")
|
|
}
|
|
|
|
_futex_wait_with_timeout :: proc "contextless" (f: ^Futex, expected: u32, duration: time.Duration) -> bool {
|
|
if duration <= 0 {
|
|
return false
|
|
}
|
|
|
|
timespec_t :: struct {
|
|
tv_sec: c.long,
|
|
tv_nsec: c.long,
|
|
}
|
|
|
|
res := _unix_futex(f, FUTEX_WAIT_PRIVATE, expected, ×pec_t{
|
|
tv_sec = (c.long)(duration/1e9),
|
|
tv_nsec = (c.long)(duration%1e9),
|
|
})
|
|
|
|
if res != -1 {
|
|
return true
|
|
}
|
|
|
|
if __errno()^ == ETIMEDOUT {
|
|
return false
|
|
}
|
|
|
|
_panic("futex_wait_with_timeout failure")
|
|
}
|
|
|
|
_futex_signal :: proc "contextless" (f: ^Futex) {
|
|
res := _unix_futex(f, FUTEX_WAKE_PRIVATE, 1, nil)
|
|
|
|
if res == -1 {
|
|
_panic("futex_wake_single failure")
|
|
}
|
|
}
|
|
|
|
_futex_broadcast :: proc "contextless" (f: ^Futex) {
|
|
res := _unix_futex(f, FUTEX_WAKE_PRIVATE, u32(max(i32)), nil)
|
|
|
|
if res == -1 {
|
|
_panic("_futex_wake_all failure")
|
|
}
|
|
}
|