| 1 |
#include "stlport_prefix.h" |
| 2 |
|
| 3 |
#if defined(__unix) && defined(__GNUC__) |
| 4 |
|
| 5 |
#ifdef __FreeBSD__ |
| 6 |
# include <osreldate.h> |
| 7 |
#endif |
| 8 |
|
| 9 |
#if (defined(__FreeBSD__) && (__FreeBSD_version < 503001)) || defined(__sun) || defined (__hpux) |
| 10 |
/* Note: __cxa_finalize and __cxa_atexit present in libc in FreeBSD 5.3 */ |
| 11 |
|
| 12 |
#include <stdlib.h> |
| 13 |
#include <stdio.h> |
| 14 |
#include <pthread.h> |
| 15 |
|
| 16 |
/* __asm__ (".symver " "__cxa_finalize" "," "__cxa_finalize" "@" "STLPORT_5_0_0"); */ |
| 17 |
/* __asm__ (".symver " "__cxa_finalize" "," "__cxa_finalize" "@@" "STLPORT_5_0_0"); */ |
| 18 |
|
| 19 |
/* Not atomic! */ |
| 20 |
/* But we can use static mutexes here: I hope that performance issue isn't very |
| 21 |
significant on unloading (for only few calls, ~10) - ptr */ |
| 22 |
|
| 23 |
/* |
| 24 |
#define atomic_compare_and_exchange_bool_acq(mem, newval, oldval) \ |
| 25 |
({ __typeof (mem) __gmemp = (mem); \ |
| 26 |
__typeof (*mem) __gnewval = (newval); \ |
| 27 |
\ |
| 28 |
*__gmemp == (oldval) ? (*__gmemp = __gnewval, 0) : 1; }) |
| 29 |
*/ |
| 30 |
|
| 31 |
enum { |
| 32 |
ef_free, /* `ef_free' MUST be zero! */ |
| 33 |
ef_us, |
| 34 |
ef_on, |
| 35 |
ef_at, |
| 36 |
ef_cxa |
| 37 |
}; |
| 38 |
|
| 39 |
struct exit_function |
| 40 |
{ |
| 41 |
/* `flavour' should be of type of the `enum' above but since we need |
| 42 |
this element in an atomic operation we have to use `long int'. */ |
| 43 |
long int flavor; |
| 44 |
union { |
| 45 |
void (*at)(void); |
| 46 |
struct { |
| 47 |
void (*fn)(int status, void *arg); |
| 48 |
void *arg; |
| 49 |
} on; |
| 50 |
struct { |
| 51 |
void (*fn)(void *arg, int status); |
| 52 |
void *arg; |
| 53 |
void *dso_handle; |
| 54 |
} cxa; |
| 55 |
} func; |
| 56 |
}; |
| 57 |
|
| 58 |
struct exit_function_list |
| 59 |
{ |
| 60 |
struct exit_function_list *next; |
| 61 |
size_t idx; |
| 62 |
struct exit_function fns[32]; |
| 63 |
}; |
| 64 |
|
| 65 |
struct exit_function *__new_exitfn (void); |
| 66 |
|
| 67 |
/* Register a function to be called by exit or when a shared library |
| 68 |
is unloaded. This function is only called from code generated by |
| 69 |
the C++ compiler. */ |
| 70 |
int __cxa_atexit(void (*func)(void *), void *arg, void *d) |
| 71 |
{ |
| 72 |
struct exit_function *new = __new_exitfn (); |
| 73 |
|
| 74 |
if ( new == NULL ) |
| 75 |
return -1; |
| 76 |
|
| 77 |
new->flavor = ef_cxa; |
| 78 |
new->func.cxa.fn = (void (*) (void *, int)) func; |
| 79 |
new->func.cxa.arg = arg; |
| 80 |
new->func.cxa.dso_handle = d; |
| 81 |
return 0; |
| 82 |
} |
| 83 |
|
| 84 |
|
| 85 |
/* We change global data, so we need locking. */ |
| 86 |
#ifdef __linux__ |
| 87 |
static pthread_mutex_t lock = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP; |
| 88 |
#endif |
| 89 |
/* #ifdef __FreeBSD__ */ |
| 90 |
#if 0 |
| 91 |
static pthread_mutex_t lock = |
| 92 |
{ PTHREAD_MUTEX_RECURSIVE /* PTHREAD_MUTEX_DEFAULT */, PTHREAD_PRIO_NONE, {NULL,NULL}, |
| 93 |
NULL, { NULL }, /* MUTEX_FLAGS_PRIVATE */ 0x1, 0, 0, 0, {NULL, NULL}, |
| 94 |
{ 0, 0, 0, 0 } }; |
| 95 |
#endif |
| 96 |
#ifdef __sun |
| 97 |
static pthread_mutex_t lock = |
| 98 |
{{0, 0, 0, PTHREAD_MUTEX_RECURSIVE, _MUTEX_MAGIC}, {{{0}}}, 0}; |
| 99 |
#endif |
| 100 |
#ifdef __hpux |
| 101 |
static pthread_mutex_t lock = PTHREAD_MUTEX_RECURSIVE_INITIALIZER_NP; |
| 102 |
# ifdef __ia64 |
| 103 |
void *__dso_handle = (void *) &__dso_handle; |
| 104 |
# endif |
| 105 |
#endif |
| 106 |
|
| 107 |
|
| 108 |
static struct exit_function_list initial; |
| 109 |
struct exit_function_list *__exit_funcs = &initial; |
| 110 |
|
| 111 |
struct exit_function *__new_exitfn(void) |
| 112 |
{ |
| 113 |
struct exit_function_list *l; |
| 114 |
size_t i = 0; |
| 115 |
|
| 116 |
#ifndef __FreeBSD__ |
| 117 |
pthread_mutex_lock( &lock ); |
| 118 |
#endif |
| 119 |
|
| 120 |
for (l = __exit_funcs; l != NULL; l = l->next) { |
| 121 |
for (i = 0; i < l->idx; ++i) |
| 122 |
if (l->fns[i].flavor == ef_free) |
| 123 |
break; |
| 124 |
if ( i < l->idx ) |
| 125 |
break; |
| 126 |
|
| 127 |
if (l->idx < sizeof (l->fns) / sizeof (l->fns[0])) { |
| 128 |
i = l->idx++; |
| 129 |
break; |
| 130 |
} |
| 131 |
} |
| 132 |
|
| 133 |
if (l == NULL) { |
| 134 |
l = (struct exit_function_list *)malloc( sizeof(struct exit_function_list) ); |
| 135 |
if (l != NULL) { |
| 136 |
l->next = __exit_funcs; |
| 137 |
__exit_funcs = l; |
| 138 |
|
| 139 |
l->idx = 1; |
| 140 |
i = 0; |
| 141 |
} |
| 142 |
} |
| 143 |
|
| 144 |
/* Mark entry as used, but we don't know the flavor now. */ |
| 145 |
if ( l != NULL ) |
| 146 |
l->fns[i].flavor = ef_us; |
| 147 |
|
| 148 |
#ifndef __FreeBSD__ |
| 149 |
pthread_mutex_unlock( &lock ); |
| 150 |
#endif |
| 151 |
|
| 152 |
return l == NULL ? NULL : &l->fns[i]; |
| 153 |
} |
| 154 |
|
| 155 |
/* If D is non-NULL, call all functions registered with `__cxa_atexit' |
| 156 |
with the same dso handle. Otherwise, if D is NULL, call all of the |
| 157 |
registered handlers. */ |
| 158 |
|
| 159 |
/* |
| 160 |
* Note, that original __cxa_finalize don't use lock, but use __exit_funcs |
| 161 |
* i.e. global data. |
| 162 |
*/ |
| 163 |
void __cxa_finalize(void *d) |
| 164 |
{ |
| 165 |
struct exit_function_list *funcs; |
| 166 |
|
| 167 |
#ifndef __FreeBSD__ |
| 168 |
pthread_mutex_lock( &lock ); |
| 169 |
#endif |
| 170 |
|
| 171 |
for (funcs = __exit_funcs; funcs; funcs = funcs->next) { |
| 172 |
struct exit_function *f; |
| 173 |
|
| 174 |
for (f = &funcs->fns[funcs->idx - 1]; f >= &funcs->fns[0]; --f) { |
| 175 |
if ( (d == NULL || d == f->func.cxa.dso_handle) && (f->flavor == ef_cxa) ) { |
| 176 |
f->flavor = ef_free; |
| 177 |
(*f->func.cxa.fn) (f->func.cxa.arg, 0); |
| 178 |
} |
| 179 |
} |
| 180 |
} |
| 181 |
|
| 182 |
/* Remove the registered fork handlers. We do not have to |
| 183 |
unregister anything if the program is going to terminate anyway. */ |
| 184 |
#ifdef UNREGISTER_ATFORK |
| 185 |
if (d != NULL) |
| 186 |
UNREGISTER_ATFORK (d); |
| 187 |
#endif |
| 188 |
#ifndef __FreeBSD__ |
| 189 |
pthread_mutex_unlock( &lock ); |
| 190 |
#endif |
| 191 |
} |
| 192 |
|
| 193 |
/* __asm__ (".symver " "__cxa_finalize" "," "__cxa_finalize" "@@" "STLPORT_5_0_0"); */ |
| 194 |
/* void __cxa_finalize(void *d) __attribute__ ((weak)); */ |
| 195 |
|
| 196 |
#endif /* OS name */ |
| 197 |
#endif /* __unix */ |