futex_freebsd.odin 1.3 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071
  1. //+private
  2. //+build freebsd
  3. package sync
  4. import "core:c"
  5. import "core:time"
  6. UMTX_OP_WAIT :: 2
  7. UMTX_OP_WAKE :: 3
  8. ETIMEDOUT :: 60
  9. foreign import libc "system:c"
  10. foreign libc {
  11. _umtx_op :: proc "c" (obj: rawptr, op: c.int, val: c.ulong, uaddr: rawptr, uaddr2: rawptr) -> c.int ---
  12. __error :: proc "c" () -> ^c.int ---
  13. }
  14. _futex_wait :: proc(f: ^Futex, expected: u32) -> bool {
  15. timeout := [2]i64{14400, 0} // 4 hours
  16. for {
  17. res := _umtx_op(f, UMTX_OP_WAIT, c.ulong(expected), nil, &timeout)
  18. if res != -1 {
  19. return true
  20. }
  21. if __error()^ == ETIMEDOUT {
  22. continue
  23. }
  24. panic("_futex_wait failure")
  25. }
  26. unreachable()
  27. }
  28. _futex_wait_with_timeout :: proc(f: ^Futex, expected: u32, duration: time.Duration) -> bool {
  29. if duration <= 0 {
  30. return false
  31. }
  32. timeout := [2]i64{i64(duration/1e9), i64(duration%1e9)}
  33. res := _umtx_op(f, UMTX_OP_WAIT, c.ulong(expected), nil, &timeout)
  34. if res != -1 {
  35. return true
  36. }
  37. if __error()^ == ETIMEDOUT {
  38. return false
  39. }
  40. panic("_futex_wait_with_timeout failure")
  41. }
  42. _futex_signal :: proc(f: ^Futex) {
  43. res := _umtx_op(f, UMTX_OP_WAKE, 1, nil, nil)
  44. if res == -1 {
  45. panic("_futex_signal failure")
  46. }
  47. }
  48. _futex_broadcast :: proc(f: ^Futex) {
  49. res := _umtx_op(f, UMTX_OP_WAKE, c.ulong(max(i32)), nil, nil)
  50. if res == -1 {
  51. panic("_futex_broadcast failure")
  52. }
  53. }