2006-02-16 10:11:48 +00:00
|
|
|
/*
|
2011-04-08 13:03:26 -07:00
|
|
|
Simple DirectMedia Layer
|
|
|
|
Copyright (C) 1997-2011 Sam Lantinga <slouken@libsdl.org>
|
2006-02-16 10:11:48 +00:00
|
|
|
|
2011-04-08 13:03:26 -07:00
|
|
|
This software is provided 'as-is', without any express or implied
|
|
|
|
warranty. In no event will the authors be held liable for any damages
|
|
|
|
arising from the use of this software.
|
2006-02-16 10:11:48 +00:00
|
|
|
|
2011-04-08 13:03:26 -07:00
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
|
|
including commercial applications, and to alter it and redistribute it
|
|
|
|
freely, subject to the following restrictions:
|
2006-02-16 10:11:48 +00:00
|
|
|
|
2011-04-08 13:03:26 -07:00
|
|
|
1. The origin of this software must not be misrepresented; you must not
|
|
|
|
claim that you wrote the original software. If you use this software
|
|
|
|
in a product, an acknowledgment in the product documentation would be
|
|
|
|
appreciated but is not required.
|
|
|
|
2. Altered source versions must be plainly marked as such, and must not be
|
|
|
|
misrepresented as being the original software.
|
|
|
|
3. This notice may not be removed or altered from any source distribution.
|
2006-02-16 10:11:48 +00:00
|
|
|
*/
|
2011-10-13 01:08:30 -04:00
|
|
|
|
2006-02-21 08:46:50 +00:00
|
|
|
#include "SDL_config.h"
|
2006-02-16 10:11:48 +00:00
|
|
|
|
|
|
|
#include <pthread.h>
|
2011-10-02 00:29:16 -04:00
|
|
|
|
|
|
|
#if HAVE_PTHREAD_NP_H
|
|
|
|
#include <pthread_np.h>
|
|
|
|
#endif
|
|
|
|
|
2006-02-16 10:11:48 +00:00
|
|
|
#include <signal.h>
|
2011-03-25 13:48:48 -07:00
|
|
|
#ifdef __LINUX__
|
2011-03-25 12:44:06 -07:00
|
|
|
#include <sys/time.h>
|
|
|
|
#include <sys/resource.h>
|
|
|
|
#include <sys/syscall.h>
|
2011-10-13 01:08:30 -04:00
|
|
|
#include <unistd.h>
|
|
|
|
extern int pthread_setname_np (pthread_t __target_thread, __const char *__name) __THROW __nonnull ((2));
|
2011-03-25 12:44:06 -07:00
|
|
|
#endif
|
2006-02-16 10:11:48 +00:00
|
|
|
|
2011-10-02 00:29:16 -04:00
|
|
|
#include "SDL_platform.h"
|
2006-02-16 10:11:48 +00:00
|
|
|
#include "SDL_thread.h"
|
|
|
|
#include "../SDL_thread_c.h"
|
|
|
|
#include "../SDL_systhread.h"
|
|
|
|
|
|
|
|
/* List of signals to mask in the subthreads */
|
2009-06-03 04:37:27 +00:00
|
|
|
static const int sig_list[] = {
|
2006-07-10 21:04:37 +00:00
|
|
|
SIGHUP, SIGINT, SIGQUIT, SIGPIPE, SIGALRM, SIGTERM, SIGCHLD, SIGWINCH,
|
|
|
|
SIGVTALRM, SIGPROF, 0
|
2006-02-16 10:11:48 +00:00
|
|
|
};
|
|
|
|
|
2011-03-25 12:44:06 -07:00
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
static void *
|
|
|
|
RunThread(void *data)
|
2006-02-16 10:11:48 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
SDL_RunThread(data);
|
2011-09-20 17:48:29 -04:00
|
|
|
return NULL;
|
2006-02-16 10:11:48 +00:00
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
int
|
|
|
|
SDL_SYS_CreateThread(SDL_Thread * thread, void *args)
|
2006-02-16 10:11:48 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
pthread_attr_t type;
|
|
|
|
|
|
|
|
/* Set the thread attributes */
|
|
|
|
if (pthread_attr_init(&type) != 0) {
|
|
|
|
SDL_SetError("Couldn't initialize pthread attributes");
|
|
|
|
return (-1);
|
|
|
|
}
|
|
|
|
pthread_attr_setdetachstate(&type, PTHREAD_CREATE_JOINABLE);
|
|
|
|
|
|
|
|
/* Create the thread and go! */
|
|
|
|
if (pthread_create(&thread->handle, &type, RunThread, args) != 0) {
|
|
|
|
SDL_SetError("Not enough resources to create thread");
|
|
|
|
return (-1);
|
|
|
|
}
|
2006-03-02 03:24:20 +00:00
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
return (0);
|
2006-02-16 10:11:48 +00:00
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
void
|
2011-10-02 00:29:16 -04:00
|
|
|
SDL_SYS_SetupThread(const char *name)
|
2006-02-16 10:11:48 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
int i;
|
|
|
|
sigset_t mask;
|
2006-02-16 10:11:48 +00:00
|
|
|
|
2011-10-02 00:49:52 -04:00
|
|
|
#if ( (__MACOSX__ && (MAC_OS_X_VERSION_MAX_ALLOWED >= 1060)) || \
|
|
|
|
(__IPHONEOS__ && (__IPHONE_OS_VERSION_MAX_ALLOWED >= 30200)) )
|
2011-10-02 00:29:16 -04:00
|
|
|
if (pthread_setname_np != NULL) { pthread_setname_np(name); }
|
|
|
|
#elif HAVE_PTHREAD_SETNAME_NP
|
|
|
|
pthread_setname_np(pthread_self(), name);
|
|
|
|
#elif HAVE_PTHREAD_SET_NAME_NP
|
|
|
|
pthread_set_name_np(pthread_self(), name);
|
|
|
|
#endif
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
/* Mask asynchronous signals for this thread */
|
|
|
|
sigemptyset(&mask);
|
|
|
|
for (i = 0; sig_list[i]; ++i) {
|
|
|
|
sigaddset(&mask, sig_list[i]);
|
|
|
|
}
|
|
|
|
pthread_sigmask(SIG_BLOCK, &mask, 0);
|
2006-02-16 10:11:48 +00:00
|
|
|
|
|
|
|
#ifdef PTHREAD_CANCEL_ASYNCHRONOUS
|
2006-07-10 21:04:37 +00:00
|
|
|
/* Allow ourselves to be asynchronously cancelled */
|
|
|
|
{
|
|
|
|
int oldstate;
|
|
|
|
pthread_setcanceltype(PTHREAD_CANCEL_ASYNCHRONOUS, &oldstate);
|
|
|
|
}
|
2006-02-16 10:11:48 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2009-12-16 04:48:11 +00:00
|
|
|
SDL_threadID
|
2006-07-10 21:04:37 +00:00
|
|
|
SDL_ThreadID(void)
|
2006-02-16 10:11:48 +00:00
|
|
|
{
|
2009-12-16 04:48:11 +00:00
|
|
|
return ((SDL_threadID) pthread_self());
|
2006-02-16 10:11:48 +00:00
|
|
|
}
|
|
|
|
|
2011-03-25 10:47:49 -07:00
|
|
|
int
|
2011-03-25 12:44:06 -07:00
|
|
|
SDL_SYS_SetThreadPriority(SDL_ThreadPriority priority)
|
2011-03-25 10:47:49 -07:00
|
|
|
{
|
2011-03-25 13:48:48 -07:00
|
|
|
#ifdef __LINUX__
|
2011-03-25 12:44:06 -07:00
|
|
|
int value;
|
|
|
|
|
|
|
|
if (priority == SDL_THREAD_PRIORITY_LOW) {
|
|
|
|
value = 19;
|
|
|
|
} else if (priority == SDL_THREAD_PRIORITY_HIGH) {
|
|
|
|
value = -20;
|
|
|
|
} else {
|
|
|
|
value = 0;
|
|
|
|
}
|
|
|
|
if (setpriority(PRIO_PROCESS, syscall(SYS_gettid), value) < 0) {
|
2011-03-25 12:54:21 -07:00
|
|
|
/* Note that this fails if you're trying to set high priority
|
|
|
|
and you don't have root permission. BUT DON'T RUN AS ROOT!
|
|
|
|
*/
|
2011-03-25 12:44:06 -07:00
|
|
|
SDL_SetError("setpriority() failed");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
#else
|
2011-03-25 10:47:49 -07:00
|
|
|
struct sched_param sched;
|
|
|
|
int policy;
|
2011-03-25 12:44:06 -07:00
|
|
|
pthread_t thread = pthread_self();
|
2011-03-25 10:47:49 -07:00
|
|
|
|
2011-03-25 12:44:06 -07:00
|
|
|
if (pthread_getschedparam(thread, &policy, &sched) < 0) {
|
2011-03-25 10:47:49 -07:00
|
|
|
SDL_SetError("pthread_getschedparam() failed");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (priority == SDL_THREAD_PRIORITY_LOW) {
|
|
|
|
sched.sched_priority = sched_get_priority_min(policy);
|
|
|
|
} else if (priority == SDL_THREAD_PRIORITY_HIGH) {
|
|
|
|
sched.sched_priority = sched_get_priority_max(policy);
|
|
|
|
} else {
|
|
|
|
int min_priority = sched_get_priority_min(policy);
|
|
|
|
int max_priority = sched_get_priority_max(policy);
|
2011-03-25 12:44:06 -07:00
|
|
|
sched.sched_priority = (min_priority + (max_priority - min_priority) / 2);
|
2011-03-25 10:47:49 -07:00
|
|
|
}
|
2011-03-25 12:44:06 -07:00
|
|
|
if (pthread_setschedparam(thread, policy, &sched) < 0) {
|
2011-03-25 10:47:49 -07:00
|
|
|
SDL_SetError("pthread_setschedparam() failed");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
2011-03-25 12:44:06 -07:00
|
|
|
#endif /* linux */
|
2011-03-25 10:47:49 -07:00
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
void
|
|
|
|
SDL_SYS_WaitThread(SDL_Thread * thread)
|
2006-02-16 10:11:48 +00:00
|
|
|
{
|
2006-07-10 21:04:37 +00:00
|
|
|
pthread_join(thread->handle, 0);
|
2006-02-16 10:11:48 +00:00
|
|
|
}
|
|
|
|
|
2006-07-10 21:04:37 +00:00
|
|
|
/* vi: set ts=4 sw=4 expandtab: */
|