2009-03-03 19:28:35 -08:00
|
|
|
/*
|
|
|
|
* 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>
|
2010-06-11 13:18:41 -07:00
|
|
|
#include <bionic_futex.h>
|
|
|
|
#include <bionic_atomic_inline.h>
|
2009-03-03 19:28:35 -08:00
|
|
|
|
|
|
|
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:
|
2011-11-15 15:47:02 +01:00
|
|
|
if (__bionic_cmpxchg(0, 0x2, gv) == 0) {
|
2010-06-11 13:18:41 -07:00
|
|
|
ANDROID_MEMBAR_FULL();
|
2009-03-03 19:28:35 -08:00
|
|
|
return 1;
|
2010-06-11 13:18:41 -07:00
|
|
|
}
|
2011-11-15 15:47:02 +01:00
|
|
|
__bionic_cmpxchg(0x2, 0x6, gv); // Indicate there is a waiter
|
2009-03-03 19:28:35 -08:00
|
|
|
__futex_wait(gv, 0x6, NULL);
|
2010-06-11 13:18:41 -07:00
|
|
|
|
2009-03-03 19:28:35 -08:00
|
|
|
if(*gv != 1) // __cxa_guard_abort was called, let every thread try since there is no return code for this condition
|
|
|
|
goto retry;
|
2010-06-11 13:18:41 -07:00
|
|
|
|
|
|
|
ANDROID_MEMBAR_FULL();
|
2009-03-03 19:28:35 -08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" void __cxa_guard_release(int volatile * gv)
|
|
|
|
{
|
|
|
|
// 2 -> 1
|
|
|
|
// 6 -> 1, and wake
|
2010-06-11 13:18:41 -07:00
|
|
|
ANDROID_MEMBAR_FULL();
|
2011-11-15 15:47:02 +01:00
|
|
|
if (__bionic_cmpxchg(0x2, 0x1, gv) == 0) {
|
2009-03-03 19:28:35 -08:00
|
|
|
return;
|
2010-06-11 13:18:41 -07:00
|
|
|
}
|
2009-03-03 19:28:35 -08:00
|
|
|
|
|
|
|
*gv = 0x1;
|
|
|
|
__futex_wake(gv, 0x7fffffff);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" void __cxa_guard_abort(int volatile * gv)
|
|
|
|
{
|
2010-06-11 13:18:41 -07:00
|
|
|
ANDROID_MEMBAR_FULL();
|
2009-03-03 19:28:35 -08:00
|
|
|
*gv = 0;
|
|
|
|
__futex_wake(gv, 0x7fffffff);
|
|
|
|
}
|