]> de.git.xonotic.org Git - xonotic/netradiant.git/blobdiff - tools/quake3/common/threads.c
set eol-style
[xonotic/netradiant.git] / tools / quake3 / common / threads.c
index 49ded1716105b5ec244353fc6c4cea00eb53917e..463d32bdd300fbde7756afebef2a01d4ba191961 100644 (file)
-/*\r
-Copyright (C) 1999-2007 id Software, Inc. and contributors.\r
-For a list of contributors, see the accompanying CONTRIBUTORS file.\r
-\r
-This file is part of GtkRadiant.\r
-\r
-GtkRadiant is free software; you can redistribute it and/or modify\r
-it under the terms of the GNU General Public License as published by\r
-the Free Software Foundation; either version 2 of the License, or\r
-(at your option) any later version.\r
-\r
-GtkRadiant is distributed in the hope that it will be useful,\r
-but WITHOUT ANY WARRANTY; without even the implied warranty of\r
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\r
-GNU General Public License for more details.\r
-\r
-You should have received a copy of the GNU General Public License\r
-along with GtkRadiant; if not, write to the Free Software\r
-Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA\r
-*/\r
-\r
-#ifndef _WIN32\r
-// The below define is necessary to use\r
-// pthreads extensions like pthread_mutexattr_settype\r
-#define _GNU_SOURCE\r
-#include <pthread.h>\r
-#endif\r
-\r
-#include "cmdlib.h"\r
-#include "mathlib.h"\r
-#include "inout.h"\r
-#include "qthreads.h"\r
-\r
-#define        MAX_THREADS     64\r
-\r
-int            dispatch;\r
-int            workcount;\r
-int            oldf;\r
-qboolean               pacifier;\r
-\r
-qboolean       threaded;\r
-\r
-/*\r
-=============\r
-GetThreadWork\r
-\r
-=============\r
-*/\r
-int    GetThreadWork (void)\r
-{\r
-       int     r;\r
-       int     f;\r
-\r
-       ThreadLock ();\r
-\r
-       if (dispatch == workcount)\r
-       {\r
-               ThreadUnlock ();\r
-               return -1;\r
-       }\r
-\r
-       f = 10*dispatch / workcount;\r
-       if (f != oldf)\r
-       {\r
-               oldf = f;\r
-               if (pacifier)\r
-               {\r
-                       Sys_Printf ("%i...", f);\r
-                       fflush( stdout );       /* ydnar */\r
-               }\r
-       }\r
-\r
-       r = dispatch;\r
-       dispatch++;\r
-       ThreadUnlock ();\r
-\r
-       return r;\r
-}\r
-\r
-\r
-void (*workfunction) (int);\r
-\r
-void ThreadWorkerFunction (int threadnum)\r
-{\r
-       int             work;\r
-\r
-       while (1)\r
-       {\r
-               work = GetThreadWork ();\r
-               if (work == -1)\r
-                       break;\r
-//Sys_Printf ("thread %i, work %i\n", threadnum, work);\r
-               workfunction(work);\r
-       }\r
-}\r
-\r
-void RunThreadsOnIndividual (int workcnt, qboolean showpacifier, void(*func)(int))\r
-{\r
-       if (numthreads == -1)\r
-               ThreadSetDefault ();\r
-       workfunction = func;\r
-  RunThreadsOn (workcnt, showpacifier, ThreadWorkerFunction);\r
-}\r
-\r
-\r
-/*\r
-===================================================================\r
-\r
-WIN32\r
-\r
-===================================================================\r
-*/\r
-#ifdef _WIN32\r
-\r
-#define        USED\r
-\r
-#include <windows.h>\r
-\r
-int            numthreads = -1;\r
-CRITICAL_SECTION               crit;\r
-static int enter;\r
-\r
-void ThreadSetDefault (void)\r
-{\r
-       SYSTEM_INFO info;\r
-\r
-       if (numthreads == -1)   // not set manually\r
-       {\r
-               GetSystemInfo (&info);\r
-               numthreads = info.dwNumberOfProcessors;\r
-               if (numthreads < 1 || numthreads > 32)\r
-                       numthreads = 1;\r
-       }\r
-\r
-       Sys_Printf ("%i threads\n", numthreads);\r
-}\r
-\r
-\r
-void ThreadLock (void)\r
-{\r
-       if (!threaded)\r
-               return;\r
-       EnterCriticalSection (&crit);\r
-       if (enter)\r
-               Error ("Recursive ThreadLock\n");\r
-       enter = 1;\r
-}\r
-\r
-void ThreadUnlock (void)\r
-{\r
-       if (!threaded)\r
-               return;\r
-       if (!enter)\r
-               Error ("ThreadUnlock without lock\n");\r
-       enter = 0;\r
-       LeaveCriticalSection (&crit);\r
-}\r
-\r
-/*\r
-=============\r
-RunThreadsOn\r
-=============\r
-*/\r
-void RunThreadsOn (int workcnt, qboolean showpacifier, void(*func)(int))\r
-{\r
-       int             threadid[MAX_THREADS];\r
-       HANDLE  threadhandle[MAX_THREADS];\r
-       int             i;\r
-       int             start, end;\r
-\r
-       start = I_FloatTime ();\r
-       dispatch = 0;\r
-       workcount = workcnt;\r
-       oldf = -1;\r
-       pacifier = showpacifier;\r
-       threaded = qtrue;\r
-\r
-       //\r
-       // run threads in parallel\r
-       //\r
-       InitializeCriticalSection (&crit);\r
-\r
-       if (numthreads == 1)\r
-       {       // use same thread\r
-               func (0);\r
-       }\r
-       else\r
-       {\r
-               for (i=0 ; i<numthreads ; i++)\r
-               {\r
-                       threadhandle[i] = CreateThread(\r
-                          NULL,        // LPSECURITY_ATTRIBUTES lpsa,\r
-                          //0,         // DWORD cbStack,\r
-\r
-                               /* ydnar: cranking stack size to eliminate radiosity crash with 1MB stack on win32 */\r
-                               (4096 * 1024),\r
-\r
-                          (LPTHREAD_START_ROUTINE)func,        // LPTHREAD_START_ROUTINE lpStartAddr,\r
-                          (LPVOID)i,   // LPVOID lpvThreadParm,\r
-                          0,                   //   DWORD fdwCreate,\r
-                          &threadid[i]);\r
-               }\r
-\r
-               for (i=0 ; i<numthreads ; i++)\r
-                       WaitForSingleObject (threadhandle[i], INFINITE);\r
-       }\r
-       DeleteCriticalSection (&crit);\r
-\r
-       threaded = qfalse;\r
-       end = I_FloatTime ();\r
-       if (pacifier)\r
-               Sys_Printf (" (%i)\n", end-start);\r
-}\r
-\r
-\r
-#endif\r
-\r
-/*\r
-===================================================================\r
-\r
-OSF1\r
-\r
-===================================================================\r
-*/\r
-\r
-#ifdef __osf__\r
-#define        USED\r
-\r
-int            numthreads = 4;\r
-\r
-void ThreadSetDefault (void)\r
-{\r
-       if (numthreads == -1)   // not set manually\r
-       {\r
-               numthreads = 4;\r
-       }\r
-}\r
-\r
-\r
-#include <pthread.h>\r
-\r
-pthread_mutex_t        *my_mutex;\r
-\r
-void ThreadLock (void)\r
-{\r
-       if (my_mutex)\r
-               pthread_mutex_lock (my_mutex);\r
-}\r
-\r
-void ThreadUnlock (void)\r
-{\r
-       if (my_mutex)\r
-               pthread_mutex_unlock (my_mutex);\r
-}\r
-\r
-\r
-/*\r
-=============\r
-RunThreadsOn\r
-=============\r
-*/\r
-void RunThreadsOn (int workcnt, qboolean showpacifier, void(*func)(int))\r
-{\r
-       int             i;\r
-       pthread_t       work_threads[MAX_THREADS];\r
-       pthread_addr_t  status;\r
-       pthread_attr_t  attrib;\r
-       pthread_mutexattr_t     mattrib;\r
-       int             start, end;\r
-\r
-       start = I_FloatTime ();\r
-       dispatch = 0;\r
-       workcount = workcnt;\r
-       oldf = -1;\r
-       pacifier = showpacifier;\r
-       threaded = qtrue;\r
-\r
-       if (pacifier)\r
-               setbuf (stdout, NULL);\r
-\r
-       if (!my_mutex)\r
-       {\r
-               my_mutex = safe_malloc (sizeof(*my_mutex));\r
-               if (pthread_mutexattr_create (&mattrib) == -1)\r
-                       Error ("pthread_mutex_attr_create failed");\r
-               if (pthread_mutexattr_setkind_np (&mattrib, MUTEX_FAST_NP) == -1)\r
-                       Error ("pthread_mutexattr_setkind_np failed");\r
-               if (pthread_mutex_init (my_mutex, mattrib) == -1)\r
-                       Error ("pthread_mutex_init failed");\r
-       }\r
-\r
-       if (pthread_attr_create (&attrib) == -1)\r
-               Error ("pthread_attr_create failed");\r
-       if (pthread_attr_setstacksize (&attrib, 0x100000) == -1)\r
-               Error ("pthread_attr_setstacksize failed");\r
-       \r
-       for (i=0 ; i<numthreads ; i++)\r
-       {\r
-               if (pthread_create(&work_threads[i], attrib\r
-               , (pthread_startroutine_t)func, (pthread_addr_t)i) == -1)\r
-                       Error ("pthread_create failed");\r
-       }\r
-               \r
-       for (i=0 ; i<numthreads ; i++)\r
-       {\r
-               if (pthread_join (work_threads[i], &status) == -1)\r
-                       Error ("pthread_join failed");\r
-       }\r
-\r
-       threaded = qfalse;\r
-\r
-       end = I_FloatTime ();\r
-       if (pacifier)\r
-               Sys_Printf (" (%i)\n", end-start);\r
-}\r
-\r
-\r
-#endif\r
-\r
-/*\r
-===================================================================\r
-\r
-IRIX\r
-\r
-===================================================================\r
-*/\r
-\r
-#ifdef _MIPS_ISA \r
-#define        USED\r
-\r
-#include <task.h>\r
-#include <abi_mutex.h>\r
-#include <sys/types.h>\r
-#include <sys/prctl.h>\r
-\r
-\r
-int            numthreads = -1;\r
-abilock_t              lck;\r
-\r
-void ThreadSetDefault (void)\r
-{\r
-       if (numthreads == -1)\r
-               numthreads = prctl(PR_MAXPPROCS);\r
-       Sys_Printf ("%i threads\n", numthreads);\r
-       usconfig (CONF_INITUSERS, numthreads);\r
-}\r
-\r
-\r
-void ThreadLock (void)\r
-{\r
-       spin_lock (&lck);\r
-}\r
-\r
-void ThreadUnlock (void)\r
-{\r
-       release_lock (&lck);\r
-}\r
-\r
-\r
-/*\r
-=============\r
-RunThreadsOn\r
-=============\r
-*/\r
-void RunThreadsOn (int workcnt, qboolean showpacifier, void(*func)(int))\r
-{\r
-       int             i;\r
-       int             pid[MAX_THREADS];\r
-       int             start, end;\r
-\r
-       start = I_FloatTime ();\r
-       dispatch = 0;\r
-       workcount = workcnt;\r
-       oldf = -1;\r
-       pacifier = showpacifier;\r
-       threaded = qtrue;\r
-\r
-       if (pacifier)\r
-               setbuf (stdout, NULL);\r
-\r
-       init_lock (&lck);\r
-\r
-       for (i=0 ; i<numthreads-1 ; i++)\r
-       {\r
-               pid[i] = sprocsp ( (void (*)(void *, size_t))func, PR_SALL, (void *)i\r
-                       , NULL, 0x200000);              // 2 meg stacks\r
-               if (pid[i] == -1)\r
-               {\r
-                       perror ("sproc");\r
-                       Error ("sproc failed");\r
-               }\r
-       }\r
-               \r
-       func(i);\r
-                       \r
-       for (i=0 ; i<numthreads-1 ; i++)\r
-               wait (NULL);\r
-\r
-       threaded = qfalse;\r
-\r
-       end = I_FloatTime ();\r
-       if (pacifier)\r
-               Sys_Printf (" (%i)\n", end-start);\r
-}\r
-\r
-\r
-#endif\r
-\r
-\r
-/*\r
-=======================================================================\r
-\r
-  Linux pthreads\r
-\r
-=======================================================================\r
-*/\r
-\r
-#ifdef __linux__\r
-#define USED\r
-\r
-int numthreads = 4;\r
-\r
-void ThreadSetDefault (void)\r
-{\r
-       if (numthreads == -1)   // not set manually\r
-       {\r
-    /* default to one thread, only multi-thread when specifically told to */\r
-               numthreads = 1;\r
-       }\r
-  if(numthreads > 1)\r
-    Sys_Printf("threads: %d\n", numthreads);\r
-}\r
-\r
-#include <pthread.h>\r
-\r
-typedef struct pt_mutex_s\r
-{\r
-  pthread_t       *owner;\r
-  pthread_mutex_t a_mutex;\r
-  pthread_cond_t  cond;\r
-  unsigned int    lock;\r
-} pt_mutex_t;\r
-\r
-pt_mutex_t global_lock;\r
-\r
-void ThreadLock(void)\r
-{\r
-  pt_mutex_t *pt_mutex = &global_lock;\r
-\r
-  if(!threaded)\r
-    return;\r
-\r
-  pthread_mutex_lock(&pt_mutex->a_mutex);\r
-  if(pthread_equal(pthread_self(), (pthread_t)&pt_mutex->owner)) \r
-    pt_mutex->lock++;\r
-  else\r
-  {\r
-    if((!pt_mutex->owner) && (pt_mutex->lock == 0))\r
-    {\r
-      pt_mutex->owner = (pthread_t *)pthread_self();\r
-      pt_mutex->lock  = 1;\r
-    }\r
-    else\r
-    {\r
-      while(1)\r
-      {\r
-        pthread_cond_wait(&pt_mutex->cond, &pt_mutex->a_mutex);\r
-        if((!pt_mutex->owner) && (pt_mutex->lock == 0))\r
-        {\r
-          pt_mutex->owner = (pthread_t *)pthread_self();\r
-          pt_mutex->lock  = 1;\r
-          break;\r
-        }\r
-      }\r
-    }\r
-  }\r
-  pthread_mutex_unlock(&pt_mutex->a_mutex);\r
-}\r
-\r
-void ThreadUnlock(void)\r
-{\r
-  pt_mutex_t *pt_mutex = &global_lock;\r
-  \r
-  if(!threaded)\r
-    return;\r
-\r
-  pthread_mutex_lock(&pt_mutex->a_mutex);\r
-  pt_mutex->lock--;\r
-  \r
-  if(pt_mutex->lock == 0)\r
-  {\r
-    pt_mutex->owner = NULL;\r
-    pthread_cond_signal(&pt_mutex->cond);\r
-  }\r
-  \r
-  pthread_mutex_unlock(&pt_mutex->a_mutex);\r
-}\r
-\r
-void recursive_mutex_init(pthread_mutexattr_t attribs)\r
-{\r
-  pt_mutex_t *pt_mutex = &global_lock;\r
-  \r
-  pt_mutex->owner = NULL;\r
-  if(pthread_mutex_init(&pt_mutex->a_mutex, &attribs) != 0)\r
-    Error("pthread_mutex_init failed\n");\r
-  if(pthread_cond_init(&pt_mutex->cond, NULL) != 0)\r
-    Error("pthread_cond_init failed\n");\r
-  \r
-  pt_mutex->lock = 0;\r
-}\r
-\r
-/*\r
-=============\r
-RunThreadsOn\r
-=============\r
-*/\r
-void RunThreadsOn (int workcnt, qboolean showpacifier, void(*func)(int))\r
-{\r
-  pthread_mutexattr_t         mattrib;\r
-  pthread_t work_threads[MAX_THREADS];\r
-  \r
-  int    start, end;\r
-  int   i=0, status=0;\r
-  \r
-  start     = I_FloatTime ();\r
-  pacifier  = showpacifier;\r
-  \r
-  dispatch  = 0;\r
-  oldf      = -1;\r
-  workcount = workcnt;\r
-  \r
-  if(numthreads == 1)\r
-    func(0);\r
-  else\r
-  {    \r
-    threaded  = qtrue;\r
-      \r
-    if(pacifier)\r
-      setbuf(stdout, NULL);\r
-\r
-    if(pthread_mutexattr_init(&mattrib) != 0)\r
-      Error("pthread_mutexattr_init failed");\r
-#if __GLIBC_MINOR__ == 1\r
-    if (pthread_mutexattr_settype(&mattrib, PTHREAD_MUTEX_FAST_NP) != 0)\r
-#else\r
-    if (pthread_mutexattr_settype(&mattrib, PTHREAD_MUTEX_ADAPTIVE_NP) != 0)\r
-#endif\r
-      Error ("pthread_mutexattr_settype failed");\r
-    recursive_mutex_init(mattrib);\r
-\r
-    for (i=0 ; i<numthreads ; i++)\r
-    {\r
-      /* Default pthread attributes: joinable & non-realtime scheduling */\r
-      if(pthread_create(&work_threads[i], NULL, (void*)func, (void*)i) != 0)\r
-        Error("pthread_create failed");\r
-    }\r
-    for (i=0 ; i<numthreads ; i++)\r
-    {\r
-      if(pthread_join(work_threads[i], (void **)&status) != 0)\r
-        Error("pthread_join failed");\r
-    }\r
-    pthread_mutexattr_destroy(&mattrib);\r
-    threaded = qfalse;\r
-  }\r
-  \r
-  end = I_FloatTime ();\r
-  if (pacifier)\r
-    Sys_Printf (" (%i)\n", end-start);\r
-}\r
-#endif // ifdef __linux__\r
-\r
-\r
-/*\r
-=======================================================================\r
-\r
-  SINGLE THREAD\r
-\r
-=======================================================================\r
-*/\r
-\r
-#ifndef USED\r
-\r
-int            numthreads = 1;\r
-\r
-void ThreadSetDefault (void)\r
-{\r
-       numthreads = 1;\r
-}\r
-\r
-void ThreadLock (void)\r
-{\r
-}\r
-\r
-void ThreadUnlock (void)\r
-{\r
-}\r
-\r
-/*\r
-=============\r
-RunThreadsOn\r
-=============\r
-*/\r
-void RunThreadsOn (int workcnt, qboolean showpacifier, void(*func)(int))\r
-{\r
-       int             i;\r
-       int             start, end;\r
-\r
-       dispatch = 0;\r
-       workcount = workcnt;\r
-       oldf = -1;\r
-       pacifier = showpacifier;\r
-       start = I_FloatTime (); \r
-       func(0);\r
-\r
-       end = I_FloatTime ();\r
-       if (pacifier)\r
-               Sys_Printf (" (%i)\n", end-start);\r
-}\r
-\r
-#endif\r
+/*
+Copyright (C) 1999-2007 id Software, Inc. and contributors.
+For a list of contributors, see the accompanying CONTRIBUTORS file.
+
+This file is part of GtkRadiant.
+
+GtkRadiant 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 2 of the License, or
+(at your option) any later version.
+
+GtkRadiant 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.
+
+You should have received a copy of the GNU General Public License
+along with GtkRadiant; if not, write to the Free Software
+Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+*/
+
+#ifndef _WIN32
+// The below define is necessary to use
+// pthreads extensions like pthread_mutexattr_settype
+#define _GNU_SOURCE
+#include <pthread.h>
+#endif
+
+#include "cmdlib.h"
+#include "mathlib.h"
+#include "inout.h"
+#include "qthreads.h"
+
+#define        MAX_THREADS     64
+
+int            dispatch;
+int            workcount;
+int            oldf;
+qboolean               pacifier;
+
+qboolean       threaded;
+
+/*
+=============
+GetThreadWork
+
+=============
+*/
+int    GetThreadWork (void)
+{
+       int     r;
+       int     f;
+
+       ThreadLock ();
+
+       if (dispatch == workcount)
+       {
+               ThreadUnlock ();
+               return -1;
+       }
+
+       f = 10*dispatch / workcount;
+       if (f != oldf)
+       {
+               oldf = f;
+               if (pacifier)
+               {
+                       Sys_Printf ("%i...", f);
+                       fflush( stdout );       /* ydnar */
+               }
+       }
+
+       r = dispatch;
+       dispatch++;
+       ThreadUnlock ();
+
+       return r;
+}
+
+
+void (*workfunction) (int);
+
+void ThreadWorkerFunction (int threadnum)
+{
+       int             work;
+
+       while (1)
+       {
+               work = GetThreadWork ();
+               if (work == -1)
+                       break;
+//Sys_Printf ("thread %i, work %i\n", threadnum, work);
+               workfunction(work);
+       }
+}
+
+void RunThreadsOnIndividual (int workcnt, qboolean showpacifier, void(*func)(int))
+{
+       if (numthreads == -1)
+               ThreadSetDefault ();
+       workfunction = func;
+  RunThreadsOn (workcnt, showpacifier, ThreadWorkerFunction);
+}
+
+
+/*
+===================================================================
+
+WIN32
+
+===================================================================
+*/
+#ifdef _WIN32
+
+#define        USED
+
+#include <windows.h>
+
+int            numthreads = -1;
+CRITICAL_SECTION               crit;
+static int enter;
+
+void ThreadSetDefault (void)
+{
+       SYSTEM_INFO info;
+
+       if (numthreads == -1)   // not set manually
+       {
+               GetSystemInfo (&info);
+               numthreads = info.dwNumberOfProcessors;
+               if (numthreads < 1 || numthreads > 32)
+                       numthreads = 1;
+       }
+
+       Sys_Printf ("%i threads\n", numthreads);
+}
+
+
+void ThreadLock (void)
+{
+       if (!threaded)
+               return;
+       EnterCriticalSection (&crit);
+       if (enter)
+               Error ("Recursive ThreadLock\n");
+       enter = 1;
+}
+
+void ThreadUnlock (void)
+{
+       if (!threaded)
+               return;
+       if (!enter)
+               Error ("ThreadUnlock without lock\n");
+       enter = 0;
+       LeaveCriticalSection (&crit);
+}
+
+/*
+=============
+RunThreadsOn
+=============
+*/
+void RunThreadsOn (int workcnt, qboolean showpacifier, void(*func)(int))
+{
+       int             threadid[MAX_THREADS];
+       HANDLE  threadhandle[MAX_THREADS];
+       int             i;
+       int             start, end;
+
+       start = I_FloatTime ();
+       dispatch = 0;
+       workcount = workcnt;
+       oldf = -1;
+       pacifier = showpacifier;
+       threaded = qtrue;
+
+       //
+       // run threads in parallel
+       //
+       InitializeCriticalSection (&crit);
+
+       if (numthreads == 1)
+       {       // use same thread
+               func (0);
+       }
+       else
+       {
+               for (i=0 ; i<numthreads ; i++)
+               {
+                       threadhandle[i] = CreateThread(
+                          NULL,        // LPSECURITY_ATTRIBUTES lpsa,
+                          //0,         // DWORD cbStack,
+
+                               /* ydnar: cranking stack size to eliminate radiosity crash with 1MB stack on win32 */
+                               (4096 * 1024),
+
+                          (LPTHREAD_START_ROUTINE)func,        // LPTHREAD_START_ROUTINE lpStartAddr,
+                          (LPVOID)i,   // LPVOID lpvThreadParm,
+                          0,                   //   DWORD fdwCreate,
+                          &threadid[i]);
+               }
+
+               for (i=0 ; i<numthreads ; i++)
+                       WaitForSingleObject (threadhandle[i], INFINITE);
+       }
+       DeleteCriticalSection (&crit);
+
+       threaded = qfalse;
+       end = I_FloatTime ();
+       if (pacifier)
+               Sys_Printf (" (%i)\n", end-start);
+}
+
+
+#endif
+
+/*
+===================================================================
+
+OSF1
+
+===================================================================
+*/
+
+#ifdef __osf__
+#define        USED
+
+int            numthreads = 4;
+
+void ThreadSetDefault (void)
+{
+       if (numthreads == -1)   // not set manually
+       {
+               numthreads = 4;
+       }
+}
+
+
+#include <pthread.h>
+
+pthread_mutex_t        *my_mutex;
+
+void ThreadLock (void)
+{
+       if (my_mutex)
+               pthread_mutex_lock (my_mutex);
+}
+
+void ThreadUnlock (void)
+{
+       if (my_mutex)
+               pthread_mutex_unlock (my_mutex);
+}
+
+
+/*
+=============
+RunThreadsOn
+=============
+*/
+void RunThreadsOn (int workcnt, qboolean showpacifier, void(*func)(int))
+{
+       int             i;
+       pthread_t       work_threads[MAX_THREADS];
+       pthread_addr_t  status;
+       pthread_attr_t  attrib;
+       pthread_mutexattr_t     mattrib;
+       int             start, end;
+
+       start = I_FloatTime ();
+       dispatch = 0;
+       workcount = workcnt;
+       oldf = -1;
+       pacifier = showpacifier;
+       threaded = qtrue;
+
+       if (pacifier)
+               setbuf (stdout, NULL);
+
+       if (!my_mutex)
+       {
+               my_mutex = safe_malloc (sizeof(*my_mutex));
+               if (pthread_mutexattr_create (&mattrib) == -1)
+                       Error ("pthread_mutex_attr_create failed");
+               if (pthread_mutexattr_setkind_np (&mattrib, MUTEX_FAST_NP) == -1)
+                       Error ("pthread_mutexattr_setkind_np failed");
+               if (pthread_mutex_init (my_mutex, mattrib) == -1)
+                       Error ("pthread_mutex_init failed");
+       }
+
+       if (pthread_attr_create (&attrib) == -1)
+               Error ("pthread_attr_create failed");
+       if (pthread_attr_setstacksize (&attrib, 0x100000) == -1)
+               Error ("pthread_attr_setstacksize failed");
+       
+       for (i=0 ; i<numthreads ; i++)
+       {
+               if (pthread_create(&work_threads[i], attrib
+               , (pthread_startroutine_t)func, (pthread_addr_t)i) == -1)
+                       Error ("pthread_create failed");
+       }
+               
+       for (i=0 ; i<numthreads ; i++)
+       {
+               if (pthread_join (work_threads[i], &status) == -1)
+                       Error ("pthread_join failed");
+       }
+
+       threaded = qfalse;
+
+       end = I_FloatTime ();
+       if (pacifier)
+               Sys_Printf (" (%i)\n", end-start);
+}
+
+
+#endif
+
+/*
+===================================================================
+
+IRIX
+
+===================================================================
+*/
+
+#ifdef _MIPS_ISA 
+#define        USED
+
+#include <task.h>
+#include <abi_mutex.h>
+#include <sys/types.h>
+#include <sys/prctl.h>
+
+
+int            numthreads = -1;
+abilock_t              lck;
+
+void ThreadSetDefault (void)
+{
+       if (numthreads == -1)
+               numthreads = prctl(PR_MAXPPROCS);
+       Sys_Printf ("%i threads\n", numthreads);
+       usconfig (CONF_INITUSERS, numthreads);
+}
+
+
+void ThreadLock (void)
+{
+       spin_lock (&lck);
+}
+
+void ThreadUnlock (void)
+{
+       release_lock (&lck);
+}
+
+
+/*
+=============
+RunThreadsOn
+=============
+*/
+void RunThreadsOn (int workcnt, qboolean showpacifier, void(*func)(int))
+{
+       int             i;
+       int             pid[MAX_THREADS];
+       int             start, end;
+
+       start = I_FloatTime ();
+       dispatch = 0;
+       workcount = workcnt;
+       oldf = -1;
+       pacifier = showpacifier;
+       threaded = qtrue;
+
+       if (pacifier)
+               setbuf (stdout, NULL);
+
+       init_lock (&lck);
+
+       for (i=0 ; i<numthreads-1 ; i++)
+       {
+               pid[i] = sprocsp ( (void (*)(void *, size_t))func, PR_SALL, (void *)i
+                       , NULL, 0x200000);              // 2 meg stacks
+               if (pid[i] == -1)
+               {
+                       perror ("sproc");
+                       Error ("sproc failed");
+               }
+       }
+               
+       func(i);
+                       
+       for (i=0 ; i<numthreads-1 ; i++)
+               wait (NULL);
+
+       threaded = qfalse;
+
+       end = I_FloatTime ();
+       if (pacifier)
+               Sys_Printf (" (%i)\n", end-start);
+}
+
+
+#endif
+
+
+/*
+=======================================================================
+
+  Linux pthreads
+
+=======================================================================
+*/
+
+#ifdef __linux__
+#define USED
+
+int numthreads = 4;
+
+void ThreadSetDefault (void)
+{
+       if (numthreads == -1)   // not set manually
+       {
+    /* default to one thread, only multi-thread when specifically told to */
+               numthreads = 1;
+       }
+  if(numthreads > 1)
+    Sys_Printf("threads: %d\n", numthreads);
+}
+
+#include <pthread.h>
+
+typedef struct pt_mutex_s
+{
+  pthread_t       *owner;
+  pthread_mutex_t a_mutex;
+  pthread_cond_t  cond;
+  unsigned int    lock;
+} pt_mutex_t;
+
+pt_mutex_t global_lock;
+
+void ThreadLock(void)
+{
+  pt_mutex_t *pt_mutex = &global_lock;
+
+  if(!threaded)
+    return;
+
+  pthread_mutex_lock(&pt_mutex->a_mutex);
+  if(pthread_equal(pthread_self(), (pthread_t)&pt_mutex->owner)) 
+    pt_mutex->lock++;
+  else
+  {
+    if((!pt_mutex->owner) && (pt_mutex->lock == 0))
+    {
+      pt_mutex->owner = (pthread_t *)pthread_self();
+      pt_mutex->lock  = 1;
+    }
+    else
+    {
+      while(1)
+      {
+        pthread_cond_wait(&pt_mutex->cond, &pt_mutex->a_mutex);
+        if((!pt_mutex->owner) && (pt_mutex->lock == 0))
+        {
+          pt_mutex->owner = (pthread_t *)pthread_self();
+          pt_mutex->lock  = 1;
+          break;
+        }
+      }
+    }
+  }
+  pthread_mutex_unlock(&pt_mutex->a_mutex);
+}
+
+void ThreadUnlock(void)
+{
+  pt_mutex_t *pt_mutex = &global_lock;
+  
+  if(!threaded)
+    return;
+
+  pthread_mutex_lock(&pt_mutex->a_mutex);
+  pt_mutex->lock--;
+  
+  if(pt_mutex->lock == 0)
+  {
+    pt_mutex->owner = NULL;
+    pthread_cond_signal(&pt_mutex->cond);
+  }
+  
+  pthread_mutex_unlock(&pt_mutex->a_mutex);
+}
+
+void recursive_mutex_init(pthread_mutexattr_t attribs)
+{
+  pt_mutex_t *pt_mutex = &global_lock;
+  
+  pt_mutex->owner = NULL;
+  if(pthread_mutex_init(&pt_mutex->a_mutex, &attribs) != 0)
+    Error("pthread_mutex_init failed\n");
+  if(pthread_cond_init(&pt_mutex->cond, NULL) != 0)
+    Error("pthread_cond_init failed\n");
+  
+  pt_mutex->lock = 0;
+}
+
+/*
+=============
+RunThreadsOn
+=============
+*/
+void RunThreadsOn (int workcnt, qboolean showpacifier, void(*func)(int))
+{
+  pthread_mutexattr_t         mattrib;
+  pthread_t work_threads[MAX_THREADS];
+  
+  int    start, end;
+  int   i=0, status=0;
+  
+  start     = I_FloatTime ();
+  pacifier  = showpacifier;
+  
+  dispatch  = 0;
+  oldf      = -1;
+  workcount = workcnt;
+  
+  if(numthreads == 1)
+    func(0);
+  else
+  {    
+    threaded  = qtrue;
+      
+    if(pacifier)
+      setbuf(stdout, NULL);
+
+    if(pthread_mutexattr_init(&mattrib) != 0)
+      Error("pthread_mutexattr_init failed");
+#if __GLIBC_MINOR__ == 1
+    if (pthread_mutexattr_settype(&mattrib, PTHREAD_MUTEX_FAST_NP) != 0)
+#else
+    if (pthread_mutexattr_settype(&mattrib, PTHREAD_MUTEX_ADAPTIVE_NP) != 0)
+#endif
+      Error ("pthread_mutexattr_settype failed");
+    recursive_mutex_init(mattrib);
+
+    for (i=0 ; i<numthreads ; i++)
+    {
+      /* Default pthread attributes: joinable & non-realtime scheduling */
+      if(pthread_create(&work_threads[i], NULL, (void*)func, (void*)i) != 0)
+        Error("pthread_create failed");
+    }
+    for (i=0 ; i<numthreads ; i++)
+    {
+      if(pthread_join(work_threads[i], (void **)&status) != 0)
+        Error("pthread_join failed");
+    }
+    pthread_mutexattr_destroy(&mattrib);
+    threaded = qfalse;
+  }
+  
+  end = I_FloatTime ();
+  if (pacifier)
+    Sys_Printf (" (%i)\n", end-start);
+}
+#endif // ifdef __linux__
+
+
+/*
+=======================================================================
+
+  SINGLE THREAD
+
+=======================================================================
+*/
+
+#ifndef USED
+
+int            numthreads = 1;
+
+void ThreadSetDefault (void)
+{
+       numthreads = 1;
+}
+
+void ThreadLock (void)
+{
+}
+
+void ThreadUnlock (void)
+{
+}
+
+/*
+=============
+RunThreadsOn
+=============
+*/
+void RunThreadsOn (int workcnt, qboolean showpacifier, void(*func)(int))
+{
+       int             i;
+       int             start, end;
+
+       dispatch = 0;
+       workcount = workcnt;
+       oldf = -1;
+       pacifier = showpacifier;
+       start = I_FloatTime (); 
+       func(0);
+
+       end = I_FloatTime ();
+       if (pacifier)
+               Sys_Printf (" (%i)\n", end-start);
+}
+
+#endif