newlib/winsup/cygwin/pthread.cc
Christopher Faylor 893ac8e03c Replace valid memory checks with new myfault class "exception handling", almost
everywhere.  Leave some thread.cc stuff alone for now.
* cygtls.h: Kludge some definitions to avoid including a problematic windows
header.
(_cygtls::_myfault): New entry.
(_cygtls::_myfault_errno): Ditto.
(_cygtls::fault_guarded): New function.
(_cygtls::setup_fault): Ditto.
(_cygtls::return_from_fault): Ditto.
(_cygtls::clear_fault): Ditto.
(myfault): New class.
* exceptions.cc (handle_exceptions): Handle case of guarded fault in system
routine.
* gendef: Add another entry point for setjmp that the compiler doesn't know
about and won't complain about.
* gentls_offsets: Just include windows.h rather than kludging a HANDLE def.
* miscfuncs.cc (check_null_str): Delete.
(check_null_empty_str): Ditto.
(check_null_empty_str_errno): Ditto.
(check_null_str_errno): Ditto.
(__check_null_invalid_struct): Ditto.
(__check_null_invalid_struct_errno): Ditto.
(__check_invalid_read_ptr): Ditto.
(__check_invalid_read_ptr_errno): Ditto.
(dummytest): New function.
(check_iovec_for_read): Delete.
(chec_iovec): Rename from check_iovec_for_write.  Take a read/write parameter.
* tlsoffsets.h: Regenerate.
* winsup.h: Remove check_* declarations.
(check_iovec_for_read): Delete declaration.  Turn into a define instead.
(check_iovec_for_write): Ditto.
(check_iovec): New declaration.
* thread.h: Use ifdef guard name consistent with other header files.
2005-07-03 02:40:30 +00:00

249 lines
4.6 KiB
C++

/* pthread.cc: posix pthread interface for Cygwin
Copyright 1998, 1999, 2000, 2001, 2002, 2003 Red Hat, Inc.
Originally written by Marco Fuykschot <marco@ddi.nl>
This file is part of Cygwin.
This software is a copyrighted work licensed under the terms of the
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
details. */
#include "winsup.h"
#include "thread.h"
#include "cygerrno.h"
#include <stdarg.h>
#include <sys/fcntl.h>
#include "cygtls.h"
extern "C"
{
/* ThreadCreation */
int
pthread_create (pthread_t * thread, const pthread_attr_t * attr,
void *(*start_routine) (void *), void *arg)
{
return pthread::create (thread, attr, start_routine, arg);
}
int
pthread_once (pthread_once_t * once_control, void (*init_routine) (void))
{
return pthread::once (once_control, init_routine);
}
int
pthread_atfork (void (*prepare)(void), void (*parent)(void), void (*child)(void))
{
return pthread::atfork (prepare, parent, child);
}
/* Thread Exit */
void
pthread_exit (void *value_ptr)
{
return pthread::self ()->exit (value_ptr);
}
int
pthread_join (pthread_t thread, void **return_val)
{
return pthread::join (&thread, (void **) return_val);
}
int
pthread_detach (pthread_t thread)
{
return pthread::detach (&thread);
}
/* This isn't a posix call... should we keep it? */
int
pthread_suspend (pthread_t thread)
{
return pthread::suspend (&thread);
}
/* same */
int
pthread_continue (pthread_t thread)
{
return pthread::resume (&thread);
}
unsigned long
pthread_getsequence_np (pthread_t * thread)
{
if (!pthread::is_good_object (thread))
return EINVAL;
return (*thread)->getsequence_np ();
}
/* ID */
pthread_t pthread_self ()
{
return pthread::self ();
}
/* Mutexes */
int
pthread_mutex_init (pthread_mutex_t * mutex, const pthread_mutexattr_t * attr)
{
return pthread_mutex::init (mutex, attr);
}
/* Synchronisation */
int
pthread_cond_init (pthread_cond_t * cond, const pthread_condattr_t * attr)
{
return pthread_cond::init (cond, attr);
}
/* RW Locks */
int
pthread_rwlock_init (pthread_rwlock_t *rwlock, const pthread_rwlockattr_t *attr)
{
return pthread_rwlock::init (rwlock, attr);
}
/* Cancelability */
int
pthread_cancel (pthread_t thread)
{
return pthread::cancel (thread);
}
int
pthread_setcancelstate (int state, int *oldstate)
{
return pthread::self ()->setcancelstate (state, oldstate);
}
int
pthread_setcanceltype (int type, int *oldtype)
{
return pthread::self ()->setcanceltype (type, oldtype);
}
void
pthread_testcancel (void)
{
pthread::self ()->testcancel ();
}
void
_pthread_cleanup_push (__pthread_cleanup_handler *handler)
{
pthread::self ()->push_cleanup_handler (handler);
}
void
_pthread_cleanup_pop (int execute)
{
pthread::self ()->pop_cleanup_handler (execute);
}
/* Semaphores */
int
sem_init (sem_t * sem, int pshared, unsigned int value)
{
return semaphore::init (sem, pshared, value);
}
int
sem_destroy (sem_t * sem)
{
return semaphore::destroy (sem);
}
/* Mangle semaphore name to follow windows naming rules. Prepend "Global\"
if running on terminal service aware machine. Substitute invalid backslash
by forward slash characters, hoping not to collide. */
static bool
mangle_sem_name (char *mangled, const char *name)
{
myfault efault;
if (efault.faulted (EFAULT))
return false;
if (!*name)
{
set_errno (ENOENT);
return false;
}
int len = strlen (name);
if (len >= CYG_MAX_PATH
|| (wincap.has_terminal_services () && len >= CYG_MAX_PATH - 7))
{
set_errno (EINVAL);
return false;
}
strcpy (mangled, wincap.has_terminal_services () ? "Global\\" : "");
char *d = mangled + strlen (mangled);
const char *s = name;
while (*s)
*d++ = (*s == '\\') ? '/' : *s++;
*d = '\0';
return true;
}
sem_t *
sem_open (const char *name, int oflag, ...)
{
mode_t mode = 0;
unsigned int value = 0;
if (oflag & O_CREAT)
{
va_list ap;
va_start (ap, oflag);
mode = va_arg (ap, mode_t);
value = va_arg (ap, unsigned int);
va_end (ap);
}
char mangled_name[CYG_MAX_PATH];
if (!mangle_sem_name (mangled_name, name))
return NULL;
return semaphore::open (mangled_name, oflag, mode, value);
}
int
sem_close (sem_t * sem)
{
return semaphore::destroy (sem);
}
int
sem_wait (sem_t * sem)
{
return semaphore::wait (sem);
}
int
sem_trywait (sem_t * sem)
{
return semaphore::trywait (sem);
}
int
sem_timedwait (sem_t * sem, const struct timespec *abstime)
{
return semaphore::timedwait (sem, abstime);
}
int
sem_post (sem_t * sem)
{
return semaphore::post (sem);
}
int
sem_getvalue (sem_t * sem, int *sval)
{
return semaphore::getvalue (sem, sval);
}
}