futex_openbsd.odin 1.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778
  1. //+private
  2. //+build openbsd
  3. package sync
  4. import "core:c"
  5. import "core:os"
  6. import "core:time"
  7. FUTEX_WAIT :: 1
  8. FUTEX_WAKE :: 2
  9. FUTEX_PRIVATE_FLAG :: 128
  10. FUTEX_WAIT_PRIVATE :: (FUTEX_WAIT | FUTEX_PRIVATE_FLAG)
  11. FUTEX_WAKE_PRIVATE :: (FUTEX_WAKE | FUTEX_PRIVATE_FLAG)
  12. foreign import libc "system:c"
  13. foreign libc {
  14. @(link_name="futex")
  15. _unix_futex :: proc "c" (f: ^Futex, op: c.int, val: u32, timeout: rawptr) -> c.int ---
  16. }
  17. _futex_wait :: proc "contextless" (f: ^Futex, expected: u32) -> bool {
  18. res := _unix_futex(f, FUTEX_WAIT_PRIVATE, expected, nil)
  19. if res != -1 {
  20. return true
  21. }
  22. if os.Errno(os.get_last_error()) == os.ETIMEDOUT {
  23. return false
  24. }
  25. _panic("futex_wait failure")
  26. }
  27. _futex_wait_with_timeout :: proc "contextless" (f: ^Futex, expected: u32, duration: time.Duration) -> bool {
  28. if duration <= 0 {
  29. return false
  30. }
  31. timespec_t :: struct {
  32. tv_sec: c.long,
  33. tv_nsec: c.long,
  34. }
  35. res := _unix_futex(f, FUTEX_WAIT_PRIVATE, expected, &timespec_t{
  36. tv_sec = (c.long)(duration/1e9),
  37. tv_nsec = (c.long)(duration%1e9),
  38. })
  39. if res != -1 {
  40. return true
  41. }
  42. if os.Errno(os.get_last_error()) == os.ETIMEDOUT {
  43. return false
  44. }
  45. _panic("futex_wait_with_timeout failure")
  46. }
  47. _futex_signal :: proc "contextless" (f: ^Futex) {
  48. res := _unix_futex(f, FUTEX_WAKE_PRIVATE, 1, nil)
  49. if res == -1 {
  50. _panic("futex_wake_single failure")
  51. }
  52. }
  53. _futex_broadcast :: proc "contextless" (f: ^Futex) {
  54. res := _unix_futex(f, FUTEX_WAKE_PRIVATE, u32(max(i32)), nil)
  55. if res == -1 {
  56. _panic("_futex_wake_all failure")
  57. }
  58. }