/* Implementation of Kos32-specific threads compatibility routines for libgcc2. GCC is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version. GCC is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. Under Section 7 of GPL version 3, you are granted additional permissions described in the GCC Runtime Library Exception, version 3.1, as published by the Free Software Foundation. You should have received a copy of the GNU General Public License and a copy of the GCC Runtime Library Exception along with this program; see the files COPYING3 and COPYING.RUNTIME respectively. If not, see . */ #include #include "gthr-kos32.h" #define FUTEX_INIT 0 #define FUTEX_DESTROY 1 #define FUTEX_WAIT 2 #define FUTEX_WAKE 3 #define exchange_acquire(ptr, new) \ __atomic_exchange_4((ptr), (new), __ATOMIC_ACQUIRE) #define exchange_release(ptr, new) \ __atomic_exchange_4((ptr), (new), __ATOMIC_RELEASE) int __gthr_kos32_once (__gthread_once_t *once, void (*func) (void)) { if (once == NULL || func == NULL) return EINVAL; if (! once->done) { if (__sync_add_and_fetch(&(once->started), 1) == 0) { (*func) (); once->done = 1; } else { while (! once->done) yield(); } } return 0; }; int __gthr_kos32_key_create (__gthread_key_t *key, void (*dtor) (void *) __attribute__((unused))) { int status = 0; unsigned int tls_index = tls_alloc(); if (tls_index != 0xFFFFFFFF) { *key = tls_index; #ifdef MINGW32_SUPPORTS_MT_EH /* Mingw runtime will run the dtors in reverse order for each thread when the thread exits. */ status = __mingwthr_key_dtor (*key, dtor); #endif } else status = -1; return status; } int __gthr_kos32_key_delete (__gthread_key_t key) { return tls_free(key); } void* __gthr_kos32_getspecific (__gthread_key_t key) { return tls_get(key); } int __gthr_kos32_setspecific (__gthread_key_t key, const void *ptr) { return tls_set(key, CONST_CAST2(void *, const void *, ptr)); } void __gthr_kos32_mutex_init_function (__gthread_mutex_t *mutex) { int handle; mutex->lock = 0; __asm__ volatile( "int $0x40\t" :"=a"(handle) :"a"(77),"b"(FUTEX_INIT),"c"(mutex)); mutex->handle = handle; } void __gthr_kos32_mutex_destroy (__gthread_mutex_t *mutex) { int retval; __asm__ volatile( "int $0x40\t" :"=a"(retval) :"a"(77),"b"(FUTEX_DESTROY),"c"(mutex->handle)); } int __gthr_kos32_mutex_lock (__gthread_mutex_t *mutex) { int tmp; if( __sync_fetch_and_add(&mutex->lock, 1) == 0) return 0; while (exchange_acquire (&mutex->lock, 2) != 0) { __asm__ volatile( "int $0x40\t\n" :"=a"(tmp) :"a"(77),"b"(FUTEX_WAIT), "c"(mutex->handle),"d"(2),"S"(0)); } return 0; } int __gthr_kos32_mutex_trylock (__gthread_mutex_t *mutex) { int zero = 0; return !__atomic_compare_exchange_4(&mutex->lock, &zero, 1,0,__ATOMIC_ACQUIRE,__ATOMIC_RELAXED); } int __gthr_kos32_mutex_unlock (__gthread_mutex_t *mutex) { int prev; prev = exchange_release (&mutex->lock, 0); if (prev != 1) { __asm__ volatile( "int $0x40\t" :"=a"(prev) :"a"(77),"b"(FUTEX_WAKE), "c"(mutex->handle),"d"(1)); }; return 0; } void __gthr_kos32_recursive_mutex_init_function (__gthread_recursive_mutex_t *mutex) { int handle; mutex->lock = 0; __asm__ volatile( "int $0x40\t" :"=a"(handle) :"a"(77),"b"(FUTEX_INIT),"c"(mutex)); mutex->handle = handle; mutex->depth = 0; mutex->owner = 0; } int __gthr_kos32_recursive_mutex_lock (__gthread_recursive_mutex_t *mutex) { int tmp; unsigned long me = (unsigned long)tls_get(TLS_KEY_LOW_STACK); if( __sync_fetch_and_add(&mutex->lock, 1) == 0) { mutex->depth = 1; mutex->owner = me; return 0; } else if (mutex->owner == me) { __sync_fetch_and_sub(&mutex->lock, 1); ++(mutex->depth); } else while (exchange_acquire (&mutex->lock, 2) != 0) { __asm__ volatile( "int $0x40\t\n" :"=a"(tmp) :"a"(77),"b"(FUTEX_WAIT), "c"(mutex->handle),"d"(2),"S"(0)); mutex->depth = 1; mutex->owner = me; }; return 0; } int __gthr_kos32_recursive_mutex_trylock (__gthread_recursive_mutex_t *mutex) { unsigned long me = (unsigned long)tls_get(TLS_KEY_LOW_STACK); int zero = 0; if(__atomic_compare_exchange_4(&mutex->lock, &zero, 1,0,__ATOMIC_ACQUIRE,__ATOMIC_RELAXED)) { mutex->depth = 1; mutex->owner = me; } else if (mutex->owner == me) ++(mutex->depth); else return 1; return 0; } int __gthr_kos32_recursive_mutex_unlock (__gthread_recursive_mutex_t *mutex) { --(mutex->depth); if (mutex->depth == 0) { int prev; prev = exchange_release (&mutex->lock, 0); if (prev != 1) { __asm__ volatile( "int $0x40\t" :"=a"(prev) :"a"(77),"b"(FUTEX_WAKE), "c"(mutex->handle),"d"(1)); }; mutex->owner = 0; }; return 0; } int __gthr_kos32_recursive_mutex_destroy (__gthread_recursive_mutex_t *mutex) { int retval; __asm__ volatile( "int $0x40\t" :"=a"(retval) :"a"(77),"b"(FUTEX_DESTROY),"c"(mutex->handle)); return 0; }