2004-01-14 07:45:37 -08:00
|
|
|
/* cygtls.cc
|
2003-12-23 08:43:45 -08:00
|
|
|
|
|
|
|
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"
|
2005-07-02 19:40:30 -07:00
|
|
|
#define USE_SYS_TYPES_FD_SET
|
2003-12-23 08:43:45 -08:00
|
|
|
#include "cygtls.h"
|
|
|
|
#include <syslog.h>
|
2010-12-28 22:33:30 -08:00
|
|
|
#include <stdlib.h>
|
2004-01-14 07:45:37 -08:00
|
|
|
#include "path.h"
|
|
|
|
#include "fhandler.h"
|
|
|
|
#include "dtable.h"
|
|
|
|
#include "cygheap.h"
|
2004-01-18 21:46:54 -08:00
|
|
|
#include "sigproc.h"
|
2010-02-28 07:54:25 -08:00
|
|
|
#include "exception.h"
|
2004-01-14 07:45:37 -08:00
|
|
|
|
2003-12-23 08:43:45 -08:00
|
|
|
/* Two calls to get the stack right... */
|
|
|
|
void
|
2004-02-11 19:01:58 -08:00
|
|
|
_cygtls::call (DWORD (*func) (void *, void *), void *arg)
|
2003-12-23 08:43:45 -08:00
|
|
|
{
|
|
|
|
char buf[CYGTLS_PADSIZE];
|
2010-02-28 07:54:25 -08:00
|
|
|
/* Initialize this thread's ability to respond to things like
|
|
|
|
SIGSEGV or SIGFPE. */
|
|
|
|
exception protect;
|
2006-05-24 19:33:13 -07:00
|
|
|
_my_tls.call2 (func, arg, buf);
|
2003-12-23 08:43:45 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2004-02-11 19:01:58 -08:00
|
|
|
_cygtls::call2 (DWORD (*func) (void *, void *), void *arg, void *buf)
|
2003-12-23 08:43:45 -08:00
|
|
|
{
|
2014-11-28 04:10:12 -08:00
|
|
|
/* If func is pthread_wrapper, the final stack hasn't been set up yet.
|
|
|
|
This only happens in pthread_wrapper itself. Thus it doesn't make
|
|
|
|
sense to call init_thread or perform BLODA detection. pthread_wrapper
|
|
|
|
eventually calls init_thread by itself. */
|
|
|
|
if ((void *) func != (void *) pthread_wrapper)
|
2015-12-15 09:28:03 -08:00
|
|
|
init_thread (buf, func);
|
2012-02-27 03:55:27 -08:00
|
|
|
|
2004-01-14 07:45:37 -08:00
|
|
|
DWORD res = func (arg, buf);
|
2006-05-24 19:33:13 -07:00
|
|
|
remove (INFINITE);
|
2006-05-30 19:14:17 -07:00
|
|
|
/* Don't call ExitThread on the main thread since we may have been
|
|
|
|
dynamically loaded. */
|
2006-06-01 17:09:50 -07:00
|
|
|
if ((void *) func != (void *) dll_crt0_1
|
|
|
|
&& (void *) func != (void *) dll_dllcrt0_1)
|
2012-12-21 11:32:43 -08:00
|
|
|
ExitThread (res);
|
2003-12-23 08:43:45 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2004-02-11 19:01:58 -08:00
|
|
|
_cygtls::init_thread (void *x, DWORD (*func) (void *, void *))
|
2003-12-23 08:43:45 -08:00
|
|
|
{
|
|
|
|
if (x)
|
|
|
|
{
|
2006-05-15 20:14:24 -07:00
|
|
|
memset (this, 0, sizeof (*this));
|
2011-01-11 00:05:51 -08:00
|
|
|
_REENT_INIT_PTR (&local_clib);
|
2003-12-23 08:43:45 -08:00
|
|
|
stackptr = stack;
|
2015-06-19 06:58:23 -07:00
|
|
|
altstack.ss_flags = SS_DISABLE;
|
2003-12-23 08:43:45 -08:00
|
|
|
if (_GLOBAL_REENT)
|
|
|
|
{
|
|
|
|
local_clib._stdin = _GLOBAL_REENT->_stdin;
|
|
|
|
local_clib._stdout = _GLOBAL_REENT->_stdout;
|
|
|
|
local_clib._stderr = _GLOBAL_REENT->_stderr;
|
2004-09-06 21:05:14 -07:00
|
|
|
local_clib.__sdidinit = _GLOBAL_REENT->__sdidinit ? -1 : 0;
|
2003-12-23 08:43:45 -08:00
|
|
|
local_clib.__cleanup = _GLOBAL_REENT->__cleanup;
|
2004-01-26 10:52:02 -08:00
|
|
|
local_clib.__sglue._niobs = 3;
|
|
|
|
local_clib.__sglue._iobs = &_GLOBAL_REENT->__sf[0];
|
2003-12-23 08:43:45 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-02-06 10:24:11 -08:00
|
|
|
thread_id = GetCurrentThreadId ();
|
2005-12-05 12:20:18 -08:00
|
|
|
initialized = CYGTLS_INITIALIZED;
|
2003-12-23 08:43:45 -08:00
|
|
|
errno_addr = &(local_clib._errno);
|
2011-08-03 09:40:48 -07:00
|
|
|
locals.cw_timer = NULL;
|
2004-01-14 07:45:37 -08:00
|
|
|
|
|
|
|
if ((void *) func == (void *) cygthread::stub
|
|
|
|
|| (void *) func == (void *) cygthread::simplestub)
|
|
|
|
return;
|
|
|
|
|
2012-08-09 12:58:53 -07:00
|
|
|
cygheap->add_tls (this);
|
2003-12-23 08:43:45 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2004-02-11 19:01:58 -08:00
|
|
|
_cygtls::fixup_after_fork ()
|
|
|
|
{
|
2004-03-15 20:39:38 -08:00
|
|
|
if (sig)
|
|
|
|
{
|
|
|
|
pop ();
|
|
|
|
sig = 0;
|
|
|
|
}
|
2005-10-23 16:47:45 -07:00
|
|
|
stacklock = spinning = 0;
|
2012-07-21 15:58:20 -07:00
|
|
|
signal_arrived = NULL;
|
2009-07-06 08:42:01 -07:00
|
|
|
locals.select.sockevt = NULL;
|
2011-08-03 09:40:48 -07:00
|
|
|
locals.cw_timer = NULL;
|
2004-03-11 19:09:28 -08:00
|
|
|
wq.thread_ev = NULL;
|
2004-02-11 19:01:58 -08:00
|
|
|
}
|
|
|
|
|
2005-03-16 09:07:32 -08:00
|
|
|
#define free_local(x) \
|
|
|
|
if (locals.x) \
|
|
|
|
{ \
|
|
|
|
free (locals.x); \
|
|
|
|
locals.x = NULL; \
|
|
|
|
}
|
|
|
|
|
2004-02-11 19:01:58 -08:00
|
|
|
void
|
|
|
|
_cygtls::remove (DWORD wait)
|
2003-12-23 08:43:45 -08:00
|
|
|
{
|
2006-03-12 20:26:57 -08:00
|
|
|
initialized = 0;
|
2009-07-06 08:42:01 -07:00
|
|
|
if (exit_state >= ES_FINAL)
|
2005-03-02 16:36:49 -08:00
|
|
|
return;
|
2006-01-05 08:26:22 -08:00
|
|
|
|
2013-04-23 02:44:36 -07:00
|
|
|
debug_printf ("wait %u", wait);
|
2011-04-21 01:10:28 -07:00
|
|
|
|
2014-11-28 12:46:13 -08:00
|
|
|
HANDLE mutex = cygheap->remove_tls (this);
|
2014-11-28 04:10:12 -08:00
|
|
|
remove_wq (wait);
|
|
|
|
|
2011-04-21 01:10:28 -07:00
|
|
|
/* FIXME: Need some sort of atthreadexit function to allow things like
|
|
|
|
select to control this themselves. */
|
|
|
|
|
2015-10-23 05:30:40 -07:00
|
|
|
remove_pending_sigs ();
|
2012-07-21 15:58:20 -07:00
|
|
|
if (signal_arrived)
|
|
|
|
{
|
|
|
|
HANDLE h = signal_arrived;
|
|
|
|
signal_arrived = NULL;
|
|
|
|
CloseHandle (h);
|
|
|
|
}
|
|
|
|
|
2011-04-21 01:10:28 -07:00
|
|
|
/* Close handle and free memory used by select. */
|
2011-04-18 08:51:54 -07:00
|
|
|
if (locals.select.sockevt)
|
2005-04-05 10:13:35 -07:00
|
|
|
{
|
2011-04-21 01:10:28 -07:00
|
|
|
CloseHandle (locals.select.sockevt);
|
|
|
|
locals.select.sockevt = NULL;
|
|
|
|
free_local (select.ser_num);
|
|
|
|
free_local (select.w4);
|
2005-04-05 10:13:35 -07:00
|
|
|
}
|
2011-04-21 01:10:28 -07:00
|
|
|
/* Free memory used by network functions. */
|
|
|
|
free_local (ntoa_buf);
|
|
|
|
free_local (protoent_buf);
|
|
|
|
free_local (servent_buf);
|
|
|
|
free_local (hostent_buf);
|
2008-03-07 03:24:51 -08:00
|
|
|
/* Free temporary TLS path buffers. */
|
2014-08-25 07:53:49 -07:00
|
|
|
locals.pathbufs.destroy ();
|
2013-07-19 04:54:51 -07:00
|
|
|
/* Close timer handle. */
|
|
|
|
if (locals.cw_timer)
|
|
|
|
NtClose (locals.cw_timer);
|
2014-11-28 12:46:13 -08:00
|
|
|
if (mutex)
|
|
|
|
{
|
|
|
|
ReleaseMutex (mutex);
|
|
|
|
CloseHandle (mutex);
|
|
|
|
}
|
2003-12-23 08:43:45 -08:00
|
|
|
}
|
2014-08-25 12:47:44 -07:00
|
|
|
|
|
|
|
#ifdef __x86_64__
|
|
|
|
void san::leave ()
|
|
|
|
{
|
|
|
|
/* Restore tls_pathbuf counters in case of error. */
|
|
|
|
_my_tls.locals.pathbufs._counters = _cnt;
|
2015-07-07 11:45:06 -07:00
|
|
|
_my_tls.andreas = _clemente;
|
2014-08-25 12:47:44 -07:00
|
|
|
}
|
|
|
|
#endif
|