190 lines
4.5 KiB
C
190 lines
4.5 KiB
C
|
/*
|
||
|
* ptw32_throw.c
|
||
|
*
|
||
|
* Description:
|
||
|
* This translation unit implements routines which are private to
|
||
|
* the implementation and may be used throughout it.
|
||
|
*
|
||
|
* --------------------------------------------------------------------------
|
||
|
*
|
||
|
* Pthreads-win32 - POSIX Threads Library for Win32
|
||
|
* Copyright(C) 1998 John E. Bossom
|
||
|
* Copyright(C) 1999,2005 Pthreads-win32 contributors
|
||
|
*
|
||
|
* Contact Email: rpj@callisto.canberra.edu.au
|
||
|
*
|
||
|
* The current list of contributors is contained
|
||
|
* in the file CONTRIBUTORS included with the source
|
||
|
* code distribution. The list can also be seen at the
|
||
|
* following World Wide Web location:
|
||
|
* http://sources.redhat.com/pthreads-win32/contributors.html
|
||
|
*
|
||
|
* This library is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of the GNU Lesser General Public
|
||
|
* License as published by the Free Software Foundation; either
|
||
|
* version 2 of the License, or (at your option) any later version.
|
||
|
*
|
||
|
* This library 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
|
||
|
* Lesser General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU Lesser General Public
|
||
|
* License along with this library in the file COPYING.LIB;
|
||
|
* if not, write to the Free Software Foundation, Inc.,
|
||
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||
|
*/
|
||
|
|
||
|
#include "pthread.h"
|
||
|
#include "implement.h"
|
||
|
|
||
|
#if defined(__CLEANUP_C)
|
||
|
# include <setjmp.h>
|
||
|
#endif
|
||
|
|
||
|
/*
|
||
|
* ptw32_throw
|
||
|
*
|
||
|
* All canceled and explicitly exited POSIX threads go through
|
||
|
* here. This routine knows how to exit both POSIX initiated threads and
|
||
|
* 'implicit' POSIX threads for each of the possible language modes (C,
|
||
|
* C++, and SEH).
|
||
|
*/
|
||
|
#if defined(_MSC_VER)
|
||
|
/*
|
||
|
* Ignore the warning:
|
||
|
* "C++ exception specification ignored except to indicate that
|
||
|
* the function is not __declspec(nothrow)."
|
||
|
*/
|
||
|
#pragma warning(disable:4290)
|
||
|
#endif
|
||
|
void
|
||
|
ptw32_throw (DWORD exception)
|
||
|
#if defined(__CLEANUP_CXX)
|
||
|
throw(ptw32_exception_cancel,ptw32_exception_exit)
|
||
|
#endif
|
||
|
{
|
||
|
/*
|
||
|
* Don't use pthread_self() to avoid creating an implicit POSIX thread handle
|
||
|
* unnecessarily.
|
||
|
*/
|
||
|
ptw32_thread_t * sp = (ptw32_thread_t *) pthread_getspecific (ptw32_selfThreadKey);
|
||
|
|
||
|
#if defined(__CLEANUP_SEH)
|
||
|
DWORD exceptionInformation[3];
|
||
|
#endif
|
||
|
|
||
|
sp->state = PThreadStateExiting;
|
||
|
|
||
|
if (exception != PTW32_EPS_CANCEL && exception != PTW32_EPS_EXIT)
|
||
|
{
|
||
|
/* Should never enter here */
|
||
|
exit (1);
|
||
|
}
|
||
|
|
||
|
if (NULL == sp || sp->implicit)
|
||
|
{
|
||
|
/*
|
||
|
* We're inside a non-POSIX initialised Win32 thread
|
||
|
* so there is no point to jump or throw back to. Just do an
|
||
|
* explicit thread exit here after cleaning up POSIX
|
||
|
* residue (i.e. cleanup handlers, POSIX thread handle etc).
|
||
|
*/
|
||
|
#if ! (defined(__MINGW64__) || defined(__MINGW32__)) || defined (__MSVCRT__) || defined (__DMC__)
|
||
|
unsigned exitCode = 0;
|
||
|
|
||
|
switch (exception)
|
||
|
{
|
||
|
case PTW32_EPS_CANCEL:
|
||
|
exitCode = (unsigned)(size_t) PTHREAD_CANCELED;
|
||
|
break;
|
||
|
case PTW32_EPS_EXIT:
|
||
|
if (NULL != sp)
|
||
|
{
|
||
|
exitCode = (unsigned)(size_t) sp->exitStatus;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
#if defined(PTW32_STATIC_LIB)
|
||
|
|
||
|
pthread_win32_thread_detach_np ();
|
||
|
|
||
|
#endif
|
||
|
|
||
|
#if ! (defined(__MINGW64__) || defined(__MINGW32__)) || defined (__MSVCRT__) || defined (__DMC__)
|
||
|
_endthreadex (exitCode);
|
||
|
#else
|
||
|
_endthread ();
|
||
|
#endif
|
||
|
|
||
|
}
|
||
|
|
||
|
#if defined(__CLEANUP_SEH)
|
||
|
|
||
|
|
||
|
exceptionInformation[0] = (DWORD) (exception);
|
||
|
exceptionInformation[1] = (DWORD) (0);
|
||
|
exceptionInformation[2] = (DWORD) (0);
|
||
|
|
||
|
RaiseException (EXCEPTION_PTW32_SERVICES, 0, 3, exceptionInformation);
|
||
|
|
||
|
#else /* __CLEANUP_SEH */
|
||
|
|
||
|
#if defined(__CLEANUP_C)
|
||
|
|
||
|
ptw32_pop_cleanup_all (1);
|
||
|
longjmp (sp->start_mark, exception);
|
||
|
|
||
|
#else /* __CLEANUP_C */
|
||
|
|
||
|
#if defined(__CLEANUP_CXX)
|
||
|
|
||
|
switch (exception)
|
||
|
{
|
||
|
case PTW32_EPS_CANCEL:
|
||
|
throw ptw32_exception_cancel ();
|
||
|
break;
|
||
|
case PTW32_EPS_EXIT:
|
||
|
throw ptw32_exception_exit ();
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
#else
|
||
|
|
||
|
#error ERROR [__FILE__, line __LINE__]: Cleanup type undefined.
|
||
|
|
||
|
#endif /* __CLEANUP_CXX */
|
||
|
|
||
|
#endif /* __CLEANUP_C */
|
||
|
|
||
|
#endif /* __CLEANUP_SEH */
|
||
|
|
||
|
/* Never reached */
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
ptw32_pop_cleanup_all (int execute)
|
||
|
{
|
||
|
while (NULL != ptw32_pop_cleanup (execute))
|
||
|
{
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
DWORD
|
||
|
ptw32_get_exception_services_code (void)
|
||
|
{
|
||
|
#if defined(__CLEANUP_SEH)
|
||
|
|
||
|
return EXCEPTION_PTW32_SERVICES;
|
||
|
|
||
|
#else
|
||
|
|
||
|
return (DWORD)0;
|
||
|
|
||
|
#endif
|
||
|
}
|