/contrib/sdk/sources/newlib/libc/Makefile |
---|
16,7 → 16,8 |
NAME:= libc |
DEFINES:= -D_IEEE_LIBM -DHAVE_RENAME -DBUILD_LIBC -DPACKAGE_NAME=\"newlib\" |
DEFINES:= -U__WIN32__ -U_Win32 -U_WIN32 -U__MINGW32__ -U_MSC_VER |
DEFINES+= -D_IEEE_LIBM -DHAVE_RENAME -DBUILD_LIBC -DPACKAGE_NAME=\"newlib\" |
DEFINES+= -DPACKAGE_TARNAME=\"newlib\" -DPACKAGE_VERSION=\"2.4.0\" |
SEFINES+= -DPACKAGE_STRING=\"newlib\ 2.4.0\" -DPACKAGE_BUGREPORT=\"\" -DPACKAGE_URL=\"\" |
46,6 → 47,7 |
argz/buf_findstr.c \ |
argz/envz_get.c \ |
crt/console.asm \ |
crt/gthr-kos32.c \ |
crt/thread.S \ |
crt/setjmp.S \ |
crt/cpu_features.c \ |
/contrib/sdk/sources/newlib/libc/Tupfile.lua |
---|
9,7 → 9,7 |
LIBC_TOPDIR = "." |
LIBC_INCLUDES = "include" |
NAME = "libc" |
DEFINES = "-D_IEEE_LIBM -DHAVE_RENAME -DBUILD_LIBC" |
DEFINES = "-U__WIN32__ -U_Win32 -U_WIN32 -U__MINGW32__ -U_MSC_VER-D_IEEE_LIBM -DHAVE_RENAME -DBUILD_LIBC" |
INCLUDES = "-Iinclude" |
TOOLCHAIN_LIBPATH = tup.getconfig("TOOLCHAIN_LIBPATH") |
/contrib/sdk/sources/newlib/libc/crt/emutls.c |
---|
File deleted |
/contrib/sdk/sources/newlib/libc/crt/gthr-kos32.c |
---|
0,0 → 1,258 |
/* 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 |
<http://www.gnu.org/licenses/>. */ |
#include <errno.h> |
#include <kos32sys.h> |
#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; |
} |
/contrib/sdk/sources/newlib/libc/crt/gthr-kos32.h |
---|
0,0 → 1,215 |
/* Threads compatibility routines for libgcc2 and libobjc. */ |
/* Compile this one with gcc. */ |
/* Copyright (C) 1999-2015 Free Software Foundation, Inc. |
Contributed by Mumit Khan <khan@xraylith.wisc.edu>. |
This file is part of GCC. |
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 |
<http://www.gnu.org/licenses/>. */ |
#ifndef GCC_GTHR_KOS32_H |
#define GCC_GTHR_KOS32_H |
/* Make sure CONST_CAST2 (origin in system.h) is declared. */ |
#ifndef CONST_CAST2 |
#define CONST_CAST2(TOTYPE,FROMTYPE,X) ((__extension__(union {FROMTYPE _q; TOTYPE _nq;})(X))._nq) |
#endif |
#define __GTHREADS 1 |
#include <stddef.h> |
#ifndef __UNUSED_PARAM |
#define __UNUSED_PARAM(x) x |
#endif |
#ifdef __cplusplus |
extern "C" { |
#endif |
typedef unsigned long __gthread_key_t; |
typedef struct |
{ |
int done; |
long started; |
} __gthread_once_t; |
typedef struct |
{ |
volatile int lock; |
int handle; |
} __gthread_mutex_t; |
typedef struct |
{ |
volatile int lock; |
int handle; |
long depth; |
unsigned long owner; |
} __gthread_recursive_mutex_t; |
#define __GTHREAD_ONCE_INIT {0, -1} |
#define __GTHREAD_MUTEX_INIT_FUNCTION __gthread_mutex_init_function |
#define __GTHREAD_RECURSIVE_MUTEX_INIT_FUNCTION \ |
__gthread_recursive_mutex_init_function |
#if defined (_WIN32) && !defined(__CYGWIN__) |
#define MINGW32_SUPPORTS_MT_EH 1 |
/* Mingw runtime >= v0.3 provides a magic variable that is set to nonzero |
if -mthreads option was specified, or 0 otherwise. This is to get around |
the lack of weak symbols in PE-COFF. */ |
extern int _CRT_MT; |
extern int __mingwthr_key_dtor (unsigned long, void (*) (void *)); |
#endif /* _WIN32 && !__CYGWIN__ */ |
static inline int __gthread_active_p (void) |
{ |
#ifdef MINGW32_SUPPORTS_MT_EH |
return _CRT_MT; |
#else |
return 1; |
#endif |
} |
extern int __gthr_kos32_once (__gthread_once_t *, void (*) (void)); |
extern int __gthr_kos32_key_create (__gthread_key_t *, void (*) (void*)); |
extern int __gthr_kos32_key_delete (__gthread_key_t); |
extern void * __gthr_kos32_getspecific (__gthread_key_t); |
extern int __gthr_kos32_setspecific (__gthread_key_t, const void *); |
extern void __gthr_kos32_mutex_init_function (__gthread_mutex_t *); |
extern int __gthr_kos32_mutex_lock (__gthread_mutex_t *); |
extern int __gthr_kos32_mutex_trylock (__gthread_mutex_t *); |
extern int __gthr_kos32_mutex_unlock (__gthread_mutex_t *); |
extern void __gthr_kos32_recursive_mutex_init_function (__gthread_recursive_mutex_t *); |
extern int __gthr_kos32_recursive_mutex_lock (__gthread_recursive_mutex_t *); |
extern int __gthr_kos32_recursive_mutex_trylock (__gthread_recursive_mutex_t *); |
extern int __gthr_kos32_recursive_mutex_unlock (__gthread_recursive_mutex_t *); |
extern void __gthr_kos32_mutex_destroy (__gthread_mutex_t *); |
extern int __gthr_kos32_recursive_mutex_destroy (__gthread_recursive_mutex_t *); |
static inline int __gthread_once (__gthread_once_t *__once, void (*__func) (void)) |
{ |
if (__gthread_active_p ()) |
return __gthr_kos32_once (__once, __func); |
else |
return -1; |
} |
static inline int __gthread_key_create (__gthread_key_t *__key, void (*__dtor) (void *)) |
{ |
return __gthr_kos32_key_create (__key, __dtor); |
} |
static inline int __gthread_key_delete (__gthread_key_t __key) |
{ |
return __gthr_kos32_key_delete (__key); |
} |
static inline void* __gthread_getspecific (__gthread_key_t __key) |
{ |
return __gthr_kos32_getspecific (__key); |
} |
static inline int __gthread_setspecific (__gthread_key_t __key, const void *__ptr) |
{ |
return __gthr_kos32_setspecific (__key, __ptr); |
} |
static inline void __gthread_mutex_init_function (__gthread_mutex_t *__mutex) |
{ |
__gthr_kos32_mutex_init_function(__mutex); |
} |
static inline void __gthread_mutex_destroy (__gthread_mutex_t *__mutex) |
{ |
__gthr_kos32_mutex_destroy (__mutex); |
} |
static inline int __gthread_mutex_lock (__gthread_mutex_t *__mutex) |
{ |
if (__gthread_active_p ()) |
return __gthr_kos32_mutex_lock (__mutex); |
else |
return 0; |
} |
static inline int __gthread_mutex_trylock (__gthread_mutex_t *__mutex) |
{ |
if (__gthread_active_p ()) |
return __gthr_kos32_mutex_trylock (__mutex); |
else |
return 0; |
} |
static inline int __gthread_mutex_unlock (__gthread_mutex_t *__mutex) |
{ |
if (__gthread_active_p ()) |
return __gthr_kos32_mutex_unlock (__mutex); |
else |
return 0; |
} |
static inline void __gthread_recursive_mutex_init_function (__gthread_recursive_mutex_t *__mutex) |
{ |
__gthr_kos32_recursive_mutex_init_function (__mutex); |
} |
static inline int __gthread_recursive_mutex_lock (__gthread_recursive_mutex_t *__mutex) |
{ |
if (__gthread_active_p ()) |
return __gthr_kos32_recursive_mutex_lock (__mutex); |
else |
return 0; |
} |
static inline int __gthread_recursive_mutex_trylock (__gthread_recursive_mutex_t *__mutex) |
{ |
if (__gthread_active_p ()) |
return __gthr_kos32_recursive_mutex_trylock (__mutex); |
else |
return 0; |
} |
static inline int __gthread_recursive_mutex_unlock (__gthread_recursive_mutex_t *__mutex) |
{ |
if (__gthread_active_p ()) |
return __gthr_kos32_recursive_mutex_unlock (__mutex); |
else |
return 0; |
} |
static inline int __gthread_recursive_mutex_destroy (__gthread_recursive_mutex_t *__mutex) |
{ |
return __gthr_kos32_recursive_mutex_destroy (__mutex); |
} |
#ifdef __cplusplus |
} |
#endif |
#endif /* ! GCC_GTHR_WIN32_H */ |