| /* |
| * one_time_construction.cpp |
| * |
| * Copyright 2006 The Android Open Source Project |
| * |
| * This file contains C++ ABI support functions for one time |
| * constructors as defined in the "Run-time ABI for the ARM Architecture" |
| * section 4.4.2 |
| */ |
| |
| #include <stddef.h> |
| #include <sys/atomics.h> |
| |
| extern "C" int __futex_wait(volatile void *ftx, int val, const struct timespec *timeout); |
| extern "C" int __futex_wake(volatile void *ftx, int count); |
| |
| |
| extern "C" int __cxa_guard_acquire(int volatile * gv) |
| { |
| // 0 -> 2, return 1 |
| // 2 -> 6, wait and return 0 |
| // 6 untouched, wait and return 0 |
| // 1 untouched, return 0 |
| retry: |
| if (__atomic_cmpxchg(0, 0x2, gv) == 0) |
| return 1; |
| |
| __atomic_cmpxchg(0x2, 0x6, gv); // Indicate there is a waiter |
| __futex_wait(gv, 0x6, NULL); |
| if(*gv != 1) // __cxa_guard_abort was called, let every thread try since there is no return code for this condition |
| goto retry; |
| return 0; |
| } |
| |
| extern "C" void __cxa_guard_release(int volatile * gv) |
| { |
| // 2 -> 1 |
| // 6 -> 1, and wake |
| if (__atomic_cmpxchg(0x2, 0x1, gv) == 0) |
| return; |
| |
| *gv = 0x1; |
| __futex_wake(gv, 0x7fffffff); |
| } |
| |
| extern "C" void __cxa_guard_abort(int volatile * gv) |
| { |
| *gv = 0; |
| __futex_wake(gv, 0x7fffffff); |
| } |