URL
https://opencores.org/ocsvn/openrisc/openrisc/trunk
Subversion Repositories openrisc
[/] [openrisc/] [trunk/] [rtos/] [ecos-3.0/] [packages/] [compat/] [posix/] [current/] [src/] [pthread.cxx] - Rev 786
Compare with Previous | Blame | View Log
//========================================================================== // // pthread.cxx // // POSIX pthreads implementation // //========================================================================== // ####ECOSGPLCOPYRIGHTBEGIN#### // ------------------------------------------- // This file is part of eCos, the Embedded Configurable Operating System. // Copyright (C) 1998, 1999, 2000, 2001, 2002 Free Software Foundation, Inc. // // eCos 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 or (at your option) any later // version. // // eCos 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 eCos; if not, write to the Free Software Foundation, Inc., // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. // // As a special exception, if other files instantiate templates or use // macros or inline functions from this file, or you compile this file // and link it with other works to produce a work based on this file, // this file does not by itself cause the resulting work to be covered by // the GNU General Public License. However the source code for this file // must still be made available in accordance with section (3) of the GNU // General Public License v2. // // This exception does not invalidate any other reasons why a work based // on this file might be covered by the GNU General Public License. // ------------------------------------------- // ####ECOSGPLCOPYRIGHTEND#### //========================================================================== //#####DESCRIPTIONBEGIN#### // // Author(s): nickg // Contributors: nickg, jlarmour // Date: 2000-03-27 // Purpose: POSIX pthread implementation // Description: This file contains the implementation of the POSIX pthread // functions. // // // //####DESCRIPTIONEND#### // //========================================================================== #include <pkgconf/hal.h> #include <pkgconf/kernel.h> #include <pkgconf/posix.h> #include <pkgconf/isoinfra.h> #include <pkgconf/libc_startup.h> #include <cyg/kernel/ktypes.h> // base kernel types #include <cyg/infra/cyg_trac.h> // tracing macros #include <cyg/infra/cyg_ass.h> // assertion macros #include "pprivate.h" // POSIX private header #include <stdlib.h> // malloc(), free() #include <cyg/kernel/sched.hxx> // scheduler definitions #include <cyg/kernel/thread.hxx> // thread definitions #include <cyg/kernel/clock.hxx> // clock definitions #include <cyg/kernel/sched.inl> // scheduler inlines //----------------------------------------------------------------------------- // First check that the configuration contains the elements we need #ifndef CYGPKG_KERNEL #error POSIX pthread need eCos kernel #endif #ifndef CYGSEM_KERNEL_SCHED_MLQUEUE #error POSIX pthreads need MLQ scheduler #endif #ifndef CYGSEM_KERNEL_SCHED_TIMESLICE #error POSIX pthreads need timeslicing #endif #ifndef CYGVAR_KERNEL_THREADS_DATA #error POSIX pthreads need per-thread data #endif //============================================================================= // Internal data structures // Mutex for controlling access to shared data structures Cyg_Mutex pthread_mutex CYGBLD_POSIX_INIT; // Array of pthread control structures. A pthread_t object is // "just" an index into this array. static pthread_info *thread_table[CYGNUM_POSIX_PTHREAD_THREADS_MAX]; // Count of number of threads in table. static int pthread_count = 0; // Count of number of threads that have exited and not been reaped. static int pthreads_exited; // Count of number of threads that are waiting to be joined static int pthreads_tobejoined; // Per-thread key allocation. This key map has a 1 bit set for each // key that is free, zero if it is allocated. #define KEY_MAP_TYPE cyg_uint32 #define KEY_MAP_TYPE_SIZE (sizeof(KEY_MAP_TYPE)*8) // in BITS! static KEY_MAP_TYPE thread_key[PTHREAD_KEYS_MAX/KEY_MAP_TYPE_SIZE]; static void (*key_destructor[PTHREAD_KEYS_MAX]) (void *); // Index of next pthread_info to allocate from thread_table array. static int thread_info_next = 0; // This is used to make pthread_t values unique even when reusing // a table slot. This allows CYGNUM_POSIX_PTHREAD_THREADS_MAX to range // up to 1024. #define THREAD_ID_COOKIE_INC 0x00000400 #define THREAD_ID_COOKIE_MASK (THREAD_ID_COOKIE_INC-1) static pthread_t thread_id_cookie = THREAD_ID_COOKIE_INC; //----------------------------------------------------------------------------- // Main thread. #define MAIN_DEFAULT_STACK_SIZE \ (CYGNUM_LIBC_MAIN_DEFAULT_STACK_SIZE < PTHREAD_STACK_MIN \ ? PTHREAD_STACK_MIN : CYGNUM_LIBC_MAIN_DEFAULT_STACK_SIZE) static char main_stack[MAIN_DEFAULT_STACK_SIZE]; // Thread ID of main thread. static pthread_t main_thread; //============================================================================= // Exported variables int pthread_canceled_dummy_var; // pointed to by PTHREAD_CANCELED //============================================================================= // Internal functions //----------------------------------------------------------------------------- // Private version of pthread_self() that returns a pointer to our internal // control structure. pthread_info *pthread_self_info(void) { Cyg_Thread *thread = Cyg_Thread::self(); CYG_CHECK_DATA_PTR(thread, "Illegal current thread"); pthread_info *info = (pthread_info *)thread->get_data(CYGNUM_KERNEL_THREADS_DATA_POSIX); // This assertion mustn't be enabled because sometimes we can legitimately // carefully call this as long as we realise the value can be NULL. // e.g. consider the use of this when inheriting sigmasks when in the // context of creating the main() thread. // CYG_CHECK_DATA_PTR(info, "Not a POSIX thread!!!"); return info; } externC pthread_info *pthread_info_id( pthread_t id ) { pthread_t index = id & THREAD_ID_COOKIE_MASK; pthread_info *info = thread_table[index]; // Check for a valid entry if( info == NULL ) return NULL; // Check that this is a valid entry if ( info->state == PTHREAD_STATE_FREE || info->state == PTHREAD_STATE_EXITED ) return NULL; // Check that the entry matches the id if( info->id != id ) return NULL; // Return the pointer return info; } //----------------------------------------------------------------------------- // new operator to allow us to invoke the Cyg_Thread constructor on the // pthread_info.thread_obj array. inline void *operator new(size_t size, cyg_uint8 *ptr) { return (void *)ptr; }; //----------------------------------------------------------------------------- // Optional memory allocation functions for pthread stacks. // If there is an implementation of malloc() available, define pthread_malloc() // and pthread_free() to use it. Otherwise define them to do nothing. // In the future we may want to add configuration here to permit thread stacks // to be allocated in a nominated memory pool separate from the standard malloc() // pool. Hence the (currently redundant) encapsulation of these functions. #if CYGINT_ISO_MALLOC static __inline__ CYG_ADDRWORD pthread_malloc( CYG_ADDRWORD size ) { return (CYG_ADDRWORD)malloc( size ); } static __inline__ void pthread_free( CYG_ADDRWORD m ) { free( (void *)m ); } #define PTHREAD_MALLOC #else #define pthread_malloc(_x_) (0) #define pthread_free(_x_) #endif //----------------------------------------------------------------------------- // pthread entry function. // does some housekeeping and then calls the user's start routine. static void pthread_entry(CYG_ADDRWORD data) { pthread_info *self = (pthread_info *)data; void *retval = self->start_routine(self->start_arg); pthread_exit( retval ); } //----------------------------------------------------------------------------- // Main entry function. // This is set as the start_routine of the main thread. It invokes main() // and if it returns, shuts down the system. externC void cyg_libc_invoke_main( void ); static void *call_main( void * ) { cyg_libc_invoke_main(); return NULL; // placate compiler } //----------------------------------------------------------------------------- // Check whether there is a cancel pending and if so, whether // cancellations are enabled. We do it in this order to reduce the // number of tests in the common case - when no cancellations are // pending. // We make this inline so it can be called directly below for speed static __inline__ int checkforcancel( void ) { pthread_info *self = pthread_self_info(); if( self != NULL && self->cancelpending && self->cancelstate == PTHREAD_CANCEL_ENABLE ) return 1; else return 0; } //----------------------------------------------------------------------------- // POSIX ASR // This is installed as the ASR for all POSIX threads. static void posix_asr( CYG_ADDRWORD data ) { pthread_info *self = (pthread_info *)data; #ifdef CYGPKG_POSIX_TIMERS // Call into timer subsystem to deliver any pending // timer expirations. cyg_posix_timer_asr(self); #endif #ifdef CYGPKG_POSIX_SIGNALS // Call signal subsystem to deliver any signals cyg_posix_signal_asr(self); #endif // Check for cancellation if( self->cancelpending && self->cancelstate == PTHREAD_CANCEL_ENABLE && self->canceltype == PTHREAD_CANCEL_ASYNCHRONOUS ) { // If we have a pending cancellation, cancellations are // enabled and we are in asynchronous mode, then we can do the // cancellation processing. Since pthread_exit() does // everything we need to do, we just call that here. pthread_exit(PTHREAD_CANCELED); } } //----------------------------------------------------------------------------- // The (Grim) Reaper. // This function is called to tidy up and dispose of any threads that have // exited. This work must be done from a thread other than the one exiting. // Note: this function _must_ be called with pthread_mutex locked. static void pthread_reap() { int i; // Loop over the thread table looking for exited threads. The // pthreads_exited counter springs us out of this once we have // found them all (and keeps us out if there are none to do). for( i = 0; pthreads_exited && i < CYGNUM_POSIX_PTHREAD_THREADS_MAX ; i++ ) { pthread_info *thread = thread_table[i]; if( thread != NULL && thread->state == PTHREAD_STATE_EXITED ) { // The thread has exited, so it is a candidate for being // reaped. We have to make sure that the eCos thread has // also reached EXITED state before we can tidy it up. while( thread->thread->get_state() != Cyg_Thread::EXITED ) { // The eCos thread has not yet exited. This is // probably because its priority is too low to allow // it to complete. We fix this here by raising its // priority to equal ours and then yielding. This // should eventually get it into exited state. Cyg_Thread *self = Cyg_Thread::self(); // Set thread's priority to our current dispatching priority. thread->thread->set_priority( self->get_current_priority() ); // Yield, yield self->yield(); // and keep looping until he exits. } // At this point we have a thread that we can reap. // destroy the eCos thread thread->thread->~Cyg_Thread(); // destroy the joiner condvar thread->joiner->~Cyg_Condition_Variable(); #ifdef CYGPKG_POSIX_SIGNALS // Destroy signal handling fields cyg_posix_thread_sigdestroy( thread ); #endif // Free the stack if we allocated it if( thread->freestack ) pthread_free( thread->stackmem ); // Finally, set the thread table entry to NULL so that it // may be reused. thread_table[i] = NULL; pthread_count--; pthreads_exited--; } } } //============================================================================= // Functions exported to rest of POSIX subsystem. //----------------------------------------------------------------------------- // Create the main() thread. externC void cyg_posix_pthread_start( void ) { // Initialize the per-thread data key map. for( cyg_ucount32 i = 0; i < (PTHREAD_KEYS_MAX/KEY_MAP_TYPE_SIZE); i++ ) { thread_key[i] = ~0; } // Create the main thread pthread_attr_t attr; struct sched_param schedparam; schedparam.sched_priority = CYGNUM_POSIX_MAIN_DEFAULT_PRIORITY; pthread_attr_init( &attr ); pthread_attr_setinheritsched( &attr, PTHREAD_EXPLICIT_SCHED ); pthread_attr_setstackaddr( &attr, &main_stack[sizeof(main_stack)] ); pthread_attr_setstacksize( &attr, sizeof(main_stack) ); pthread_attr_setschedpolicy( &attr, SCHED_RR ); pthread_attr_setschedparam( &attr, &schedparam ); pthread_create( &main_thread, &attr, call_main, NULL ); } #ifdef CYGPKG_POSIX_SIGNALS //----------------------------------------------------------------------------- // Look for a thread that can accept delivery of any of the signals in // the mask and release it from any wait it is in. Since this may be // called from a DSR, it cannot use any locks internally - any locking // should be done before the call. externC void cyg_posix_pthread_release_thread( sigset_t *mask ) { int i; int count = pthread_count; // Loop over the thread table looking for a thread that has a // signal mask that does not mask all the signals in mask. // FIXME: find a more efficient way of doing this. for( i = 0; count > 0 && i < CYGNUM_POSIX_PTHREAD_THREADS_MAX ; i++ ) { pthread_info *thread = thread_table[i]; if( (thread != NULL) && (thread->state <= PTHREAD_STATE_RUNNING) && ((*mask & ~thread->sigmask) != 0) ) { // This thread can service at least one of the signals in // *mask. Knock it out of its wait and make its ASR pending. thread->thread->set_asr_pending(); thread->thread->release(); break; } // Decrement count for each valid thread we find. if( thread != NULL && thread->state != PTHREAD_STATE_FREE ) count--; } } #endif //============================================================================= // General thread operations //----------------------------------------------------------------------------- // Thread creation and management. // Create a thread. externC int pthread_create ( pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine) (void *), void *arg) { PTHREAD_ENTRY(); PTHREAD_CHECK(thread); PTHREAD_CHECK(start_routine); pthread_info *self = pthread_self_info(); pthread_attr_t use_attr; // Set use_attr to the set of attributes we are going to // actually use. Either those passed in, or the default set. if( attr == NULL ) pthread_attr_init( &use_attr ); else use_attr = *attr; // Adjust the attributes to cope with the setting of inheritsched. if( use_attr.inheritsched == PTHREAD_INHERIT_SCHED ) { CYG_ASSERT( NULL != self, "Attempt to inherit sched policy from non-POSIX thread" ); #ifdef CYGDBG_USE_ASSERTS // paranoia check int i; for (i=(sizeof(thread_table)/sizeof(*thread_table))-1; i>=0; i--) { if (thread_table[i] == self) break; } CYG_ASSERT( i>=0, "Current pthread not found in table" ); #endif use_attr.schedpolicy = self->attr.schedpolicy; use_attr.schedparam = self->attr.schedparam; } CYG_ADDRWORD stackbase, stacksize; cyg_bool freestack = false; CYG_ADDRWORD stackmem = 0; // If the stack size is not valid, we can assume that it is at // least PTHREAD_STACK_MIN bytes. if( use_attr.stacksize_valid ) stacksize = use_attr.stacksize; else stacksize = PTHREAD_STACK_MIN; if( use_attr.stackaddr_valid ) { // Set up stack base and size from supplied arguments. // Calculate stack base from address and size. // FIXME: Falling stack assumed in pthread_create(). stackmem = stackbase = (CYG_ADDRWORD)use_attr.stackaddr-stacksize; } else { #ifdef PTHREAD_MALLOC stackmem = stackbase = pthread_malloc( stacksize ); if( stackmem == 0 ) PTHREAD_RETURN( EAGAIN ); freestack = true; #else PTHREAD_RETURN(EINVAL); #endif } // Get sole access to data structures pthread_mutex.lock(); // Dispose of any dead threads pthread_reap(); // Find a free slot in the thread table pthread_info *nthread; int thread_next = thread_info_next; while( thread_table[thread_next] != NULL ) { thread_next++; if( thread_next >= CYGNUM_POSIX_PTHREAD_THREADS_MAX ) thread_next = 0; // check for wrap, and return error if no slots left if( thread_next == thread_info_next ) { pthread_mutex.unlock(); if( freestack ) pthread_free( stackmem ); PTHREAD_RETURN(ENOMEM); } } nthread = (pthread_info *)stackbase; stackbase += sizeof(pthread_info); stacksize -= sizeof(pthread_info); thread_table[thread_next] = nthread; // Set new next index thread_info_next = thread_next; // step the cookie thread_id_cookie += THREAD_ID_COOKIE_INC; // Initialize the table entry nthread->state = use_attr.detachstate == PTHREAD_CREATE_JOINABLE ? PTHREAD_STATE_RUNNING : PTHREAD_STATE_DETACHED; nthread->id = thread_next+thread_id_cookie; nthread->attr = use_attr; nthread->retval = 0; nthread->start_routine = start_routine; nthread->start_arg = arg; nthread->freestack = freestack; nthread->stackmem = stackmem; nthread->cancelstate = PTHREAD_CANCEL_ENABLE; nthread->canceltype = PTHREAD_CANCEL_DEFERRED; nthread->cancelbuffer = NULL; nthread->cancelpending = false; nthread->thread_data = NULL; #ifdef CYGVAR_KERNEL_THREADS_NAME // generate a name for this thread char *name = nthread->name; static char *name_template = "pthread.00000000"; pthread_t id = nthread->id; for( int i = 0; name_template[i]; i++ ) name[i] = name_template[i]; // dump the id, in hex into the name. for( int i = 15; i >= 8; i-- ) { name[i] = "0123456789ABCDEF"[id&0xF]; id >>= 4; } #endif // Initialize the joiner condition variable nthread->joiner = new(nthread->joiner_obj) Cyg_Condition_Variable( pthread_mutex ); #ifdef CYGPKG_POSIX_SIGNALS // Initialize signal specific fields. if (NULL != self) { CYG_CHECK_DATA_PTR( self, "Attempt to inherit signal mask from bogus pthread" ); #ifdef CYGDBG_USE_ASSERTS // paranoia check int i; for (i=(sizeof(thread_table)/sizeof(*thread_table))-1; i>=0; i--) { if (thread_table[i] == self) break; } CYG_ASSERT( i>=0, "Current pthread not found in table" ); #endif } cyg_posix_thread_siginit( nthread, self ); #endif // create the underlying eCos thread nthread->thread = new(&nthread->thread_obj[0]) Cyg_Thread ( PTHREAD_ECOS_PRIORITY(use_attr.schedparam.sched_priority), pthread_entry, (CYG_ADDRWORD)nthread, #ifdef CYGVAR_KERNEL_THREADS_NAME name, #else NULL, #endif stackbase, stacksize); // Put pointer to pthread_info into eCos thread's per-thread data. nthread->thread->set_data( CYGNUM_KERNEL_THREADS_DATA_POSIX, (CYG_ADDRWORD)nthread ); // Set timeslice enable according to scheduling policy. if( use_attr.schedpolicy == SCHED_FIFO ) nthread->thread->timeslice_disable(); else nthread->thread->timeslice_enable(); // set up ASR and data nthread->thread->set_asr( posix_asr, (CYG_ADDRWORD)nthread, NULL, NULL ); // return thread ID *thread = nthread->id; pthread_count++; pthread_mutex.unlock(); // finally, set the thread going nthread->thread->resume(); PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Get current thread id. externC pthread_t pthread_self ( void ) { PTHREAD_ENTRY(); pthread_info *info = pthread_self_info(); CYG_CHECK_DATA_PTR(info, "Not a POSIX thread!!!"); return info->id; } //----------------------------------------------------------------------------- // Compare two thread identifiers. externC int pthread_equal (pthread_t thread1, pthread_t thread2) { PTHREAD_ENTRY(); return thread1 == thread2; } //----------------------------------------------------------------------------- // Terminate current thread. externC void exit(int) CYGBLD_ATTRIB_NORET; externC void pthread_exit (void *retval) { PTHREAD_ENTRY(); pthread_info *self = pthread_self_info(); // Disable cancellation requests for this thread. If cleanup // handlers exist, they will generally be issuing system calls // to clean up resources. We want these system calls to run // without cancelling, and we also want to prevent being // re-cancelled. pthread_setcancelstate(PTHREAD_CANCEL_DISABLE, NULL); // Call cancellation handlers. We eat up the buffers as we go in // case any of the routines calls pthread_exit() itself. while( self->cancelbuffer != NULL ) { struct pthread_cleanup_buffer *buffer = self->cancelbuffer; self->cancelbuffer = buffer->prev; buffer->routine(buffer->arg); } if( self->thread_data != NULL ) { // Call per-thread key destructors. // The specification of this is that we must continue to call the // destructor functions until all the per-thread data values are NULL or // we have done it PTHREAD_DESTRUCTOR_ITERATIONS times. cyg_bool destructors_called; int destructor_iterations = 0; do { destructors_called = false; for( cyg_ucount32 key = 0; key < PTHREAD_KEYS_MAX; key++ ) { // Skip unallocated keys if( thread_key[key/KEY_MAP_TYPE_SIZE] & 1<<(key%KEY_MAP_TYPE_SIZE) ) continue; // Skip NULL destructors if( key_destructor[key] == NULL ) continue; // Skip NULL data values if( self->thread_data[key] == NULL ) continue; // If it passes all that, call the destructor. // Note that NULLing the data value here is new // behaviour in the 2001 POSIX standard. { void* value = self->thread_data[key]; self->thread_data[key] = NULL; key_destructor[key](value); } // Record that we called a destructor destructors_called = true; } // Count the iteration destructor_iterations++; } while( destructors_called && (destructor_iterations <= PTHREAD_DESTRUCTOR_ITERATIONS)); } pthread_mutex.lock(); // Set the retval for any joiner self->retval = retval; // If we are already detached, go to EXITED state, otherwise // go into JOIN state. if ( PTHREAD_STATE_DETACHED == self->state ) { self->state = PTHREAD_STATE_EXITED; pthreads_exited++; } else { self->state = PTHREAD_STATE_JOIN; pthreads_tobejoined++; } // Kick any waiting joiners self->joiner->broadcast(); cyg_bool call_exit=false; // if this is the last thread (other than threads waiting to be joined) // then we need to call exit() later if ( pthreads_exited + pthreads_tobejoined == pthread_count ) call_exit=true; pthread_mutex.unlock(); // Finally, call the exit function; this will not return. if ( call_exit ) ::exit(0); else self->thread->exit(); // This loop keeps some compilers happy. pthread_exit() is marked // with the noreturn attribute, and without this they generate a // call to abort() here in case Cyg_Thread::exit() returns. for(;;) continue; } //----------------------------------------------------------------------------- // Wait for the thread to terminate. If thread_return is not NULL then // the retval from the thread's call to pthread_exit() is stored at // *thread_return. externC int pthread_join (pthread_t thread, void **thread_return) { int err = 0; PTHREAD_ENTRY(); // check for cancellation first. pthread_testcancel(); pthread_mutex.lock(); // Dispose of any dead threads pthread_reap(); pthread_info *self = pthread_self_info(); pthread_info *joinee = pthread_info_id( thread ); if( joinee == NULL ) { err = ESRCH; } if( !err && joinee == self ) { err = EDEADLK; } if ( !err ) { switch ( joinee->state ) { case PTHREAD_STATE_RUNNING: // The thread is still running, we must wait for it. while( joinee->state == PTHREAD_STATE_RUNNING ) { if ( !joinee->joiner->wait() ) // check if we were woken because we were being cancelled if ( checkforcancel() ) { err = EAGAIN; // value unimportant, just some error break; } } // check that the thread is still joinable if( joinee->state == PTHREAD_STATE_JOIN ) break; // The thread has become unjoinable while we waited, so we // fall through to complain. case PTHREAD_STATE_FREE: case PTHREAD_STATE_DETACHED: case PTHREAD_STATE_EXITED: // None of these may be joined. err = EINVAL; break; case PTHREAD_STATE_JOIN: break; } } if ( !err ) { // here, we know that joinee is a thread that has exited and is // ready to be joined. // Get the retval if( thread_return != NULL ) *thread_return = joinee->retval; // set state to exited. joinee->state = PTHREAD_STATE_EXITED; pthreads_exited++; pthreads_tobejoined--; // Dispose of any dead threads pthread_reap(); } pthread_mutex.unlock(); // check for cancellation before returning pthread_testcancel(); PTHREAD_RETURN(err); } //----------------------------------------------------------------------------- // Set the detachstate of the thread to "detached". The thread then does not // need to be joined and its resources will be freed when it exits. externC int pthread_detach (pthread_t thread) { PTHREAD_ENTRY(); int ret = 0; pthread_mutex.lock(); pthread_info *detachee = pthread_info_id( thread ); if( detachee == NULL ) ret = ESRCH; // No such thread else if( detachee->state == PTHREAD_STATE_DETACHED ) ret = EINVAL; // Already detached! else { // Set state to detached and kick any joinees to // make them return. detachee->state = PTHREAD_STATE_DETACHED; detachee->joiner->broadcast(); } // Dispose of any dead threads pthread_reap(); pthread_mutex.unlock(); PTHREAD_RETURN(ret); } //----------------------------------------------------------------------------- // Thread attribute handling. //----------------------------------------------------------------------------- // Initialize attributes object with default attributes: // detachstate == PTHREAD_CREATE_JOINABLE // scope == PTHREAD_SCOPE_SYSTEM // inheritsched == PTHREAD_INHERIT_SCHED // schedpolicy == SCHED_OTHER // schedparam == unset // stackaddr == unset // stacksize == 0 // externC int pthread_attr_init (pthread_attr_t *attr) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); attr->detachstate = PTHREAD_CREATE_JOINABLE; attr->scope = PTHREAD_SCOPE_SYSTEM; attr->inheritsched = PTHREAD_INHERIT_SCHED; attr->schedpolicy = SCHED_OTHER; attr->schedparam.sched_priority = 0; attr->stackaddr_valid = 0; attr->stackaddr = NULL; attr->stacksize_valid = 0; attr->stacksize = 0; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Destroy thread attributes object externC int pthread_attr_destroy (pthread_attr_t *attr) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); // Nothing to do here... PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Set the detachstate attribute externC int pthread_attr_setdetachstate (pthread_attr_t *attr, int detachstate) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); if( detachstate == PTHREAD_CREATE_JOINABLE || detachstate == PTHREAD_CREATE_DETACHED ) { attr->detachstate = detachstate; PTHREAD_RETURN(0); } PTHREAD_RETURN(EINVAL); } //----------------------------------------------------------------------------- // Get the detachstate attribute externC int pthread_attr_getdetachstate (const pthread_attr_t *attr, int *detachstate) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); if( detachstate != NULL ) *detachstate = attr->detachstate; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Set scheduling contention scope externC int pthread_attr_setscope (pthread_attr_t *attr, int scope) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); if( scope == PTHREAD_SCOPE_SYSTEM || scope == PTHREAD_SCOPE_PROCESS ) { if( scope == PTHREAD_SCOPE_PROCESS ) PTHREAD_RETURN(ENOTSUP); attr->scope = scope; PTHREAD_RETURN(0); } PTHREAD_RETURN(EINVAL); } //----------------------------------------------------------------------------- // Get scheduling contention scope externC int pthread_attr_getscope (const pthread_attr_t *attr, int *scope) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); if( scope != NULL ) *scope = attr->scope; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Set scheduling inheritance attribute externC int pthread_attr_setinheritsched (pthread_attr_t *attr, int inherit) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); if( inherit == PTHREAD_INHERIT_SCHED || inherit == PTHREAD_EXPLICIT_SCHED ) { attr->inheritsched = inherit; PTHREAD_RETURN(0); } PTHREAD_RETURN(EINVAL); } //----------------------------------------------------------------------------- // Get scheduling inheritance attribute externC int pthread_attr_getinheritsched (const pthread_attr_t *attr, int *inherit) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); if( inherit != NULL ) *inherit = attr->inheritsched; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Set scheduling policy externC int pthread_attr_setschedpolicy (pthread_attr_t *attr, int policy) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); if( policy == SCHED_OTHER || policy == SCHED_FIFO || policy == SCHED_RR ) { attr->schedpolicy = policy; PTHREAD_RETURN(0); } PTHREAD_RETURN(EINVAL); } //----------------------------------------------------------------------------- // Get scheduling policy externC int pthread_attr_getschedpolicy (const pthread_attr_t *attr, int *policy) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); if( policy != NULL ) *policy = attr->schedpolicy; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Set scheduling parameters externC int pthread_attr_setschedparam (pthread_attr_t *attr, const struct sched_param *param) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); PTHREAD_CHECK(param); attr->schedparam = *param; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Get scheduling parameters externC int pthread_attr_getschedparam (const pthread_attr_t *attr, struct sched_param *param) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); if( param != NULL ) *param = attr->schedparam; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Set starting address of stack. Whether this is at the start or end of // the memory block allocated for the stack depends on whether the stack // grows up or down. externC int pthread_attr_setstackaddr (pthread_attr_t *attr, void *stackaddr) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); attr->stackaddr = stackaddr; attr->stackaddr_valid = 1; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Get any previously set stack address. externC int pthread_attr_getstackaddr (const pthread_attr_t *attr, void **stackaddr) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); if( stackaddr != NULL ) { if( attr->stackaddr_valid ) { *stackaddr = attr->stackaddr; PTHREAD_RETURN(0); } // Stack address not set, return EINVAL. else PTHREAD_RETURN(EINVAL); } PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Set minimum creation stack size. externC int pthread_attr_setstacksize (pthread_attr_t *attr, size_t stacksize) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); CYG_ASSERT( stacksize >= PTHREAD_STACK_MIN, "Inadequate stack size supplied"); // Reject inadequate stack sizes if( stacksize < PTHREAD_STACK_MIN ) PTHREAD_RETURN(EINVAL); attr->stacksize_valid = 1; attr->stacksize = stacksize; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Get current minimal stack size. externC int pthread_attr_getstacksize (const pthread_attr_t *attr, size_t *stacksize) { PTHREAD_ENTRY(); PTHREAD_CHECK(attr); // Reject attempts to get a stack size when one has not been set. if( !attr->stacksize_valid ) PTHREAD_RETURN(EINVAL); if( stacksize != NULL ) *stacksize = attr->stacksize; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Thread scheduling controls //----------------------------------------------------------------------------- // Set scheduling policy and parameters for the thread externC int pthread_setschedparam (pthread_t thread_id, int policy, const struct sched_param *param) { PTHREAD_ENTRY(); if( policy != SCHED_OTHER && policy != SCHED_FIFO && policy != SCHED_RR ) PTHREAD_RETURN(EINVAL); PTHREAD_CHECK(param); // The parameters seem OK, change the thread... pthread_mutex.lock(); pthread_info *thread = pthread_info_id( thread_id ); if( thread == NULL ) { pthread_mutex.unlock(); PTHREAD_RETURN(ESRCH); } thread->attr.schedpolicy = policy; thread->attr.schedparam = *param; if ( policy == SCHED_FIFO ) thread->thread->timeslice_disable(); else thread->thread->timeslice_enable(); thread->thread->set_priority( PTHREAD_ECOS_PRIORITY( param->sched_priority )); pthread_mutex.unlock(); PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Get scheduling policy and parameters for the thread externC int pthread_getschedparam (pthread_t thread_id, int *policy, struct sched_param *param) { PTHREAD_ENTRY(); pthread_mutex.lock(); pthread_info *thread = pthread_info_id( thread_id ); if( thread == NULL ) { pthread_mutex.unlock(); PTHREAD_RETURN(ESRCH); } if( policy != NULL ) *policy = thread->attr.schedpolicy; if( param != NULL ) *param = thread->attr.schedparam; pthread_mutex.unlock(); PTHREAD_RETURN(0); } //============================================================================= // Dynamic package initialization // Call init_routine just the once per control variable. externC int pthread_once (pthread_once_t *once_control, void (*init_routine) (void)) { PTHREAD_ENTRY(); PTHREAD_CHECK( once_control ); PTHREAD_CHECK( init_routine ); pthread_once_t old; // Do a test and set on the once_control object. pthread_mutex.lock(); old = *once_control; *once_control = 1; pthread_mutex.unlock(); // If the once_control was zero, call the init_routine(). if( !old ) init_routine(); PTHREAD_RETURN(0); } //============================================================================= //Thread specific data //----------------------------------------------------------------------------- // Create a key to identify a location in the thread specific data area. // Each thread has its own distinct thread-specific data area but all are // addressed by the same keys. The destructor function is called whenever a // thread exits and the value associated with the key is non-NULL. externC int pthread_key_create (pthread_key_t *key, void (*destructor) (void *)) { PTHREAD_ENTRY(); pthread_key_t k = -1; pthread_mutex.lock(); // Find a key to allocate for( cyg_ucount32 i = 0; i < (PTHREAD_KEYS_MAX/KEY_MAP_TYPE_SIZE); i++ ) { if( thread_key[i] != 0 ) { // We have a table slot with space available // Get index of ls set bit. HAL_LSBIT_INDEX( k, thread_key[i] ); // clear it thread_key[i] &= ~(1<<k); // Add index of word k += i * KEY_MAP_TYPE_SIZE; // Install destructor key_destructor[k] = destructor; // break out with key found break; } } if( k != -1 ) { // plant a NULL in all the valid thread data slots for this // key in case we are reusing a key we used before. for( cyg_ucount32 i = 0; i < CYGNUM_POSIX_PTHREAD_THREADS_MAX ; i++ ) { pthread_info *thread = thread_table[i]; if( thread != NULL && thread->thread_data != NULL ) thread->thread_data[k] = NULL; } } pthread_mutex.unlock(); if( k == -1 ) PTHREAD_RETURN(EAGAIN); *key = k; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Delete key. externC int pthread_key_delete (pthread_key_t key) { PTHREAD_ENTRY(); pthread_mutex.lock(); // Set the key bit to 1 to indicate it is free. thread_key[key/KEY_MAP_TYPE_SIZE] |= 1<<(key%(KEY_MAP_TYPE_SIZE)); pthread_mutex.unlock(); PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Store the pointer value in the thread-specific data slot addressed // by the key. externC int pthread_setspecific (pthread_key_t key, const void *pointer) { PTHREAD_ENTRY(); if( thread_key[key/KEY_MAP_TYPE_SIZE] & 1<<(key%KEY_MAP_TYPE_SIZE) ) PTHREAD_RETURN(EINVAL); pthread_info *self = pthread_self_info(); if( self->thread_data == NULL ) { // Allocate the per-thread data table self->thread_data = (void **)self->thread->increment_stack_limit( PTHREAD_KEYS_MAX * sizeof(void *) ); // Clear out all entries for( int i = 0; i < PTHREAD_KEYS_MAX; i++ ) self->thread_data[i] = NULL; } self->thread_data[key] = (void *)pointer; PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Retrieve the pointer value in the thread-specific data slot addressed // by the key. externC void *pthread_getspecific (pthread_key_t key) { void *val; PTHREAD_ENTRY(); if( thread_key[key/KEY_MAP_TYPE_SIZE] & 1<<(key%KEY_MAP_TYPE_SIZE) ) PTHREAD_RETURN(NULL); pthread_info *self = pthread_self_info(); if( self->thread_data == NULL ) val = NULL; else val = self->thread_data[key]; PTHREAD_RETURN(val); } //============================================================================= // Thread Cancellation Functions //----------------------------------------------------------------------------- // Set cancel state of current thread to ENABLE or DISABLE. // Returns old state in *oldstate. externC int pthread_setcancelstate (int state, int *oldstate) { PTHREAD_ENTRY(); if( state != PTHREAD_CANCEL_ENABLE && state != PTHREAD_CANCEL_DISABLE ) PTHREAD_RETURN(EINVAL); pthread_mutex.lock(); pthread_info *self = pthread_self_info(); if( oldstate != NULL ) *oldstate = self->cancelstate; self->cancelstate = state; pthread_mutex.unlock(); // Note: This function may have made it possible for a pending // cancellation to now be delivered. However the standard does not // list this function as a cancellation point, so for now we do // nothing. In future we might call pthread_testcancel() here. PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Set cancel type of current thread to ASYNCHRONOUS or DEFERRED. // Returns old type in *oldtype. externC int pthread_setcanceltype (int type, int *oldtype) { PTHREAD_ENTRY(); if( type != PTHREAD_CANCEL_ASYNCHRONOUS && type != PTHREAD_CANCEL_DEFERRED ) PTHREAD_RETURN(EINVAL); pthread_mutex.lock(); pthread_info *self = pthread_self_info(); if( oldtype != NULL ) *oldtype = self->canceltype; self->canceltype = type; pthread_mutex.unlock(); // Note: This function may have made it possible for a pending // cancellation to now be delivered. However the standard does not // list this function as a cancellation point, so for now we do // nothing. In future we might call pthread_testcancel() here. PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Cancel the thread. externC int pthread_cancel (pthread_t thread) { PTHREAD_ENTRY(); pthread_mutex.lock(); pthread_info *th = pthread_info_id(thread); if( th == NULL ) { pthread_mutex.unlock(); PTHREAD_RETURN(ESRCH); } th->cancelpending = true; if ( th->cancelstate == PTHREAD_CANCEL_ENABLE ) { if ( th->canceltype == PTHREAD_CANCEL_ASYNCHRONOUS ) { // If the thread has cancellation enabled, and it is in // asynchronous mode, set the eCos thread's ASR pending to // deal with it when the thread wakes up. We also release the // thread out of any current wait to make it wake up. th->thread->set_asr_pending(); th->thread->release(); } else if ( th->canceltype == PTHREAD_CANCEL_DEFERRED ) { // If the thread has cancellation enabled, and it is in // deferred mode, wake the thread up so that cancellation // points can test for cancellation. th->thread->release(); } else CYG_FAIL("Unknown cancellation type"); } // Otherwise the thread has cancellation disabled, in which case // it is up to the thread to enable cancellation pthread_mutex.unlock(); PTHREAD_RETURN(0); } //----------------------------------------------------------------------------- // Test for a pending cancellation for the current thread and terminate // the thread if there is one. externC void pthread_testcancel (void) { PTHREAD_ENTRY_VOID(); if( checkforcancel() ) { // If we have cancellation enabled, and there is a cancellation // pending, then go ahead and do the deed. // Exit now with special retval. pthread_exit() calls the // cancellation handlers implicitly. pthread_exit(PTHREAD_CANCELED); } PTHREAD_RETURN_VOID; } //----------------------------------------------------------------------------- // These two functions actually implement the cleanup push and pop functionality. externC void pthread_cleanup_push_inner (struct pthread_cleanup_buffer *buffer, void (*routine) (void *), void *arg) { PTHREAD_ENTRY(); pthread_info *self = pthread_self_info(); buffer->routine = routine; buffer->arg = arg; buffer->prev = self->cancelbuffer; self->cancelbuffer = buffer; return; } externC void pthread_cleanup_pop_inner (struct pthread_cleanup_buffer *buffer, int execute) { PTHREAD_ENTRY(); pthread_info *self = pthread_self_info(); CYG_ASSERT( self->cancelbuffer == buffer, "Stacking error in cleanup buffers"); if( self->cancelbuffer == buffer ) { // Remove the buffer from the stack self->cancelbuffer = buffer->prev; } else { // If the top of the stack is not the buffer we expect, do not // execute it. execute = 0; } if( execute ) buffer->routine(buffer->arg); return; } // ------------------------------------------------------------------------- // eCos-specific function to measure stack usage of the supplied thread #ifdef CYGFUN_KERNEL_THREADS_STACK_MEASUREMENT externC size_t pthread_measure_stack_usage (pthread_t thread) { pthread_info *th = pthread_info_id(thread); if ( NULL == th ) return (size_t)-1; return (size_t)th->thread->measure_stack_usage(); } #endif // ------------------------------------------------------------------------- // EOF pthread.cxx