X-Git-Url: http://git.megacz.com/?a=blobdiff_plain;f=rts%2Fposix%2FOSThreads.c;h=1b85fd41f871ddc0edb37eded24b08674bda3824;hb=ebb1d38825c6eafeb1e1ff9885b1557e7d065e49;hp=b30d08545db742cef179867a97430519ef9b7f61;hpb=88b35c172f9434fd98b700f706074d142914a8bb;p=ghc-hetmet.git diff --git a/rts/posix/OSThreads.c b/rts/posix/OSThreads.c index b30d085..1b85fd4 100644 --- a/rts/posix/OSThreads.c +++ b/rts/posix/OSThreads.c @@ -7,8 +7,9 @@ * * --------------------------------------------------------------------------*/ -#if defined(DEBUG) && defined(__linux__) +#if defined(__linux__) /* We want GNU extensions in DEBUG mode for mutex error checking */ +/* We also want the affinity API, which requires _GNU_SOURCE */ #define _GNU_SOURCE #endif @@ -26,6 +27,14 @@ #error pthreads.h is required for the threaded RTS on Posix platforms #endif +#if defined(HAVE_SCHED_H) +#include +#endif + +#ifdef HAVE_UNISTD_H +#include +#endif + /* * This (allegedly) OS threads independent layer was initially * abstracted away from code that used Pthreads, so the functions @@ -93,13 +102,21 @@ osThreadId() return pthread_self(); } +rtsBool +osThreadIsAlive(OSThreadId id STG_UNUSED) +{ + // no good way to implement this on POSIX, AFAICT. Returning true + // is safe. + return rtsTrue; +} + void initMutex(Mutex* pMut) { -#if defined(DEBUG) && defined(linux_HOST_OS) +#if defined(DEBUG) pthread_mutexattr_t attr; pthread_mutexattr_init(&attr); - pthread_mutexattr_settype(&attr,PTHREAD_MUTEX_ERRORCHECK_NP); + pthread_mutexattr_settype(&attr,PTHREAD_MUTEX_ERRORCHECK); pthread_mutex_init(pMut,&attr); #else pthread_mutex_init(pMut,NULL); @@ -140,6 +157,15 @@ setThreadLocalVar (ThreadLocalKey *key, void *value) } } +void +freeThreadLocalKey (ThreadLocalKey *key) +{ + int r; + if ((r = pthread_key_delete(*key)) != 0) { + barf("freeThreadLocalKey: %s", strerror(r)); + } +} + static void * forkOS_createThreadWrapper ( void * entry ) { @@ -162,6 +188,44 @@ forkOS_createThread ( HsStablePtr entry ) return result; } +nat +getNumberOfProcessors (void) +{ + static nat nproc = 0; + + if (nproc == 0) { +#if defined(HAVE_SYSCONF) && defined(_SC_NPROCESSORS_ONLN) + nproc = sysconf(_SC_NPROCESSORS_ONLN); +#elif defined(HAVE_SYSCONF) && defined(_SC_NPROCESSORS_CONF) + nproc = sysconf(_SC_NPROCESSORS_CONF); +#else + nproc = 1; +#endif + } + + return nproc; +} + +// Schedules the thread to run on CPU n of m. m may be less than the +// number of physical CPUs, in which case, the thread will be allowed +// to run on CPU n, n+m, n+2m etc. +void +setThreadAffinity (nat n, nat m) +{ +#if defined(HAVE_SCHED_H) && defined(HAVE_SCHED_SETAFFINITY) + nat nproc; + cpu_set_t cs; + nat i; + + nproc = getNumberOfProcessors(); + CPU_ZERO(&cs); + for (i = n; i < nproc; i+=m) { + CPU_SET(n, &cs); + } + sched_setaffinity(0, sizeof(cpu_set_t), &cs); +#endif +} + #else /* !defined(THREADED_RTS) */ int