190 lines
		
	
	
		
			4.7 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			190 lines
		
	
	
		
			4.7 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.,
 | 
						|
 *      59 Temple Place - Suite 330, Boston, MA 02111-1307, 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
 | 
						|
}
 |