You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
238 lines
4.7 KiB
238 lines
4.7 KiB
/*-------------------------------------------------------------------------
|
|
* drawElements Thread Library
|
|
* ---------------------------
|
|
*
|
|
* Copyright 2014 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*
|
|
*//*!
|
|
* \file
|
|
* \brief Unix implementation of thread management.
|
|
*//*--------------------------------------------------------------------*/
|
|
|
|
#include "deThread.h"
|
|
|
|
#if (DE_OS == DE_OS_UNIX || DE_OS == DE_OS_OSX || DE_OS == DE_OS_ANDROID || DE_OS == DE_OS_SYMBIAN || DE_OS == DE_OS_IOS || DE_OS == DE_OS_QNX)
|
|
|
|
#include "deMemory.h"
|
|
#include "deInt32.h"
|
|
|
|
#if !defined(_XOPEN_SOURCE) || (_XOPEN_SOURCE < 500)
|
|
# error "You are using too old posix API!"
|
|
#endif
|
|
|
|
#include <unistd.h>
|
|
#include <pthread.h>
|
|
#include <sched.h>
|
|
#if (DE_OS == DE_OS_UNIX) || (DE_OS == DE_OS_ANDROID)
|
|
# include <sys/syscall.h>
|
|
#endif
|
|
|
|
#if (DE_OS == DE_OS_OSX) || (DE_OS == DE_OS_IOS)
|
|
# if !defined(_SC_NPROCESSORS_CONF)
|
|
# define _SC_NPROCESSORS_CONF 57
|
|
# endif
|
|
# if !defined(_SC_NPROCESSORS_ONLN)
|
|
# define _SC_NPROCESSORS_ONLN 58
|
|
# endif
|
|
#endif
|
|
|
|
typedef struct Thread_s
|
|
{
|
|
pthread_t thread;
|
|
deThreadFunc func;
|
|
void* arg;
|
|
} Thread;
|
|
|
|
DE_STATIC_ASSERT(sizeof(deThread) >= sizeof(Thread*));
|
|
|
|
static void* startThread (void* entryPtr)
|
|
{
|
|
Thread* thread = (Thread*)entryPtr;
|
|
deThreadFunc func = thread->func;
|
|
void* arg = thread->arg;
|
|
|
|
/* Start actual thread. */
|
|
func(arg);
|
|
|
|
return DE_NULL;
|
|
}
|
|
|
|
deThread deThread_create (deThreadFunc func, void* arg, const deThreadAttributes* attributes)
|
|
{
|
|
pthread_attr_t attr;
|
|
Thread* thread = (Thread*)deCalloc(sizeof(Thread));
|
|
|
|
if (!thread)
|
|
return 0;
|
|
|
|
thread->func = func;
|
|
thread->arg = arg;
|
|
|
|
if (pthread_attr_init(&attr) != 0)
|
|
{
|
|
deFree(thread);
|
|
return 0;
|
|
}
|
|
|
|
/* \todo [2009-11-12 pyry] Map attributes. */
|
|
DE_UNREF(attributes);
|
|
|
|
if (pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE) != 0)
|
|
{
|
|
pthread_attr_destroy(&attr);
|
|
deFree(thread);
|
|
return 0;
|
|
}
|
|
|
|
if (pthread_create(&thread->thread, &attr, startThread, thread) != 0)
|
|
{
|
|
pthread_attr_destroy(&attr);
|
|
deFree(thread);
|
|
return 0;
|
|
}
|
|
DE_ASSERT(thread->thread);
|
|
|
|
pthread_attr_destroy(&attr);
|
|
|
|
return (deThread)thread;
|
|
}
|
|
|
|
deBool deThread_join (deThread threadptr)
|
|
{
|
|
Thread* thread = (Thread*)threadptr;
|
|
int ret;
|
|
|
|
DE_ASSERT(thread->thread);
|
|
ret = pthread_join(thread->thread, DE_NULL);
|
|
|
|
/* If join fails for some reason, at least mark as detached. */
|
|
if (ret != 0)
|
|
pthread_detach(thread->thread);
|
|
|
|
/* Thread is no longer valid as far as we are concerned. */
|
|
thread->thread = 0;
|
|
|
|
return (ret == 0);
|
|
}
|
|
|
|
void deThread_destroy (deThread threadptr)
|
|
{
|
|
Thread* thread = (Thread*)threadptr;
|
|
|
|
if (thread->thread)
|
|
{
|
|
/* Not joined, detach. */
|
|
int ret = pthread_detach(thread->thread);
|
|
DE_ASSERT(ret == 0);
|
|
DE_UNREF(ret);
|
|
}
|
|
|
|
deFree(thread);
|
|
}
|
|
|
|
void deSleep (deUint32 milliseconds)
|
|
{
|
|
/* Maximum value for usleep is 10^6. */
|
|
deUint32 seconds = milliseconds / 1000;
|
|
|
|
milliseconds = milliseconds - seconds * 1000;
|
|
|
|
if (seconds > 0)
|
|
sleep(seconds);
|
|
|
|
usleep((useconds_t)milliseconds * (useconds_t)1000);
|
|
}
|
|
|
|
void deYield (void)
|
|
{
|
|
sched_yield();
|
|
}
|
|
|
|
#if (DE_OS == DE_OS_UNIX) || (DE_OS == DE_OS_ANDROID)
|
|
|
|
deUint32 deGetNumAvailableLogicalCores (void)
|
|
{
|
|
#if !defined(__FreeBSD__)
|
|
unsigned long mask = 0;
|
|
const unsigned int maskSize = sizeof(mask);
|
|
long ret;
|
|
|
|
deMemset(&mask, 0, sizeof(mask));
|
|
|
|
ret = syscall(__NR_sched_getaffinity, 0, maskSize, &mask);
|
|
|
|
if (ret > 0)
|
|
{
|
|
return (deUint32)dePop64(mask);
|
|
}
|
|
else
|
|
{
|
|
#endif
|
|
#if defined(_SC_NPROCESSORS_ONLN)
|
|
const long count = sysconf(_SC_NPROCESSORS_ONLN);
|
|
|
|
if (count <= 0)
|
|
return 1;
|
|
else
|
|
return (deUint32)count;
|
|
#else
|
|
return 1;
|
|
#endif
|
|
|
|
#if !defined(__FreeBSD__)
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#else
|
|
|
|
deUint32 deGetNumAvailableLogicalCores (void)
|
|
{
|
|
#if defined(_SC_NPROCESSORS_ONLN)
|
|
const long count = sysconf(_SC_NPROCESSORS_ONLN);
|
|
|
|
if (count <= 0)
|
|
return 1;
|
|
else
|
|
return (deUint32)count;
|
|
#else
|
|
return 1;
|
|
#endif
|
|
}
|
|
|
|
#endif
|
|
|
|
deUint32 deGetNumTotalLogicalCores (void)
|
|
{
|
|
#if defined(_SC_NPROCESSORS_CONF)
|
|
const long count = sysconf(_SC_NPROCESSORS_CONF);
|
|
|
|
if (count <= 0)
|
|
return 1;
|
|
else
|
|
return (deUint32)count;
|
|
#else
|
|
return 1;
|
|
#endif
|
|
}
|
|
|
|
deUint32 deGetNumTotalPhysicalCores (void)
|
|
{
|
|
/* \todo [2015-04-09 pyry] Parse /proc/cpuinfo perhaps? */
|
|
return deGetNumTotalLogicalCores();
|
|
}
|
|
|
|
#endif /* DE_OS */
|