Blame gnulib/lib/msvc-inval.c

Packit Service 51e54d
/* Invalid parameter handler for MSVC runtime libraries.
Packit Service 51e54d
   Copyright (C) 2011-2014 Free Software Foundation, Inc.
Packit Service 51e54d
Packit Service 51e54d
   This program is free software; you can redistribute it and/or modify
Packit Service 51e54d
   it under the terms of the GNU General Public License as published by
Packit Service 51e54d
   the Free Software Foundation; either version 3, or (at your option)
Packit Service 51e54d
   any later version.
Packit Service 51e54d
Packit Service 51e54d
   This program is distributed in the hope that it will be useful,
Packit Service 51e54d
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 51e54d
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit Service 51e54d
   GNU General Public License for more details.
Packit Service 51e54d
Packit Service 51e54d
   You should have received a copy of the GNU General Public License along
Packit Service 51e54d
   with this program; if not, see <http://www.gnu.org/licenses/>.  */
Packit Service 51e54d
Packit Service 51e54d
#include <config.h>
Packit Service 51e54d
Packit Service 51e54d
/* Specification.  */
Packit Service 51e54d
#include "msvc-inval.h"
Packit Service 51e54d
Packit Service 51e54d
#if HAVE_MSVC_INVALID_PARAMETER_HANDLER \
Packit Service 51e54d
    && !(MSVC_INVALID_PARAMETER_HANDLING == SANE_LIBRARY_HANDLING)
Packit Service 51e54d
Packit Service 51e54d
/* Get _invalid_parameter_handler type and _set_invalid_parameter_handler
Packit Service 51e54d
   declaration.  */
Packit Service 51e54d
# include <stdlib.h>
Packit Service 51e54d
Packit Service 51e54d
# if MSVC_INVALID_PARAMETER_HANDLING == DEFAULT_HANDLING
Packit Service 51e54d
Packit Service 51e54d
static void __cdecl
Packit Service 51e54d
gl_msvc_invalid_parameter_handler (const wchar_t *expression,
Packit Service 51e54d
                                   const wchar_t *function,
Packit Service 51e54d
                                   const wchar_t *file,
Packit Service 51e54d
                                   unsigned int line,
Packit Service 51e54d
                                   uintptr_t dummy)
Packit Service 51e54d
{
Packit Service 51e54d
}
Packit Service 51e54d
Packit Service 51e54d
# else
Packit Service 51e54d
Packit Service 51e54d
/* Get declarations of the native Windows API functions.  */
Packit Service 51e54d
#  define WIN32_LEAN_AND_MEAN
Packit Service 51e54d
#  include <windows.h>
Packit Service 51e54d
Packit Service 51e54d
#  if defined _MSC_VER
Packit Service 51e54d
Packit Service 51e54d
static void __cdecl
Packit Service 51e54d
gl_msvc_invalid_parameter_handler (const wchar_t *expression,
Packit Service 51e54d
                                   const wchar_t *function,
Packit Service 51e54d
                                   const wchar_t *file,
Packit Service 51e54d
                                   unsigned int line,
Packit Service 51e54d
                                   uintptr_t dummy)
Packit Service 51e54d
{
Packit Service 51e54d
  RaiseException (STATUS_GNULIB_INVALID_PARAMETER, 0, 0, NULL);
Packit Service 51e54d
}
Packit Service 51e54d
Packit Service 51e54d
#  else
Packit Service 51e54d
Packit Service 51e54d
/* An index to thread-local storage.  */
Packit Service 51e54d
static DWORD tls_index;
Packit Service 51e54d
static int tls_initialized /* = 0 */;
Packit Service 51e54d
Packit Service 51e54d
/* Used as a fallback only.  */
Packit Service 51e54d
static struct gl_msvc_inval_per_thread not_per_thread;
Packit Service 51e54d
Packit Service 51e54d
struct gl_msvc_inval_per_thread *
Packit Service 51e54d
gl_msvc_inval_current (void)
Packit Service 51e54d
{
Packit Service 51e54d
  if (!tls_initialized)
Packit Service 51e54d
    {
Packit Service 51e54d
      tls_index = TlsAlloc ();
Packit Service 51e54d
      tls_initialized = 1;
Packit Service 51e54d
    }
Packit Service 51e54d
  if (tls_index == TLS_OUT_OF_INDEXES)
Packit Service 51e54d
    /* TlsAlloc had failed.  */
Packit Service 51e54d
    return &not_per_thread;
Packit Service 51e54d
  else
Packit Service 51e54d
    {
Packit Service 51e54d
      struct gl_msvc_inval_per_thread *pointer =
Packit Service 51e54d
        (struct gl_msvc_inval_per_thread *) TlsGetValue (tls_index);
Packit Service 51e54d
      if (pointer == NULL)
Packit Service 51e54d
        {
Packit Service 51e54d
          /* First call.  Allocate a new 'struct gl_msvc_inval_per_thread'.  */
Packit Service 51e54d
          pointer =
Packit Service 51e54d
            (struct gl_msvc_inval_per_thread *)
Packit Service 51e54d
            malloc (sizeof (struct gl_msvc_inval_per_thread));
Packit Service 51e54d
          if (pointer == NULL)
Packit Service 51e54d
            /* Could not allocate memory.  Use the global storage.  */
Packit Service 51e54d
            pointer = &not_per_thread;
Packit Service 51e54d
          TlsSetValue (tls_index, pointer);
Packit Service 51e54d
        }
Packit Service 51e54d
      return pointer;
Packit Service 51e54d
    }
Packit Service 51e54d
}
Packit Service 51e54d
Packit Service 51e54d
static void __cdecl
Packit Service 51e54d
gl_msvc_invalid_parameter_handler (const wchar_t *expression,
Packit Service 51e54d
                                   const wchar_t *function,
Packit Service 51e54d
                                   const wchar_t *file,
Packit Service 51e54d
                                   unsigned int line,
Packit Service 51e54d
                                   uintptr_t dummy)
Packit Service 51e54d
{
Packit Service 51e54d
  struct gl_msvc_inval_per_thread *current = gl_msvc_inval_current ();
Packit Service 51e54d
  if (current->restart_valid)
Packit Service 51e54d
    longjmp (current->restart, 1);
Packit Service 51e54d
  else
Packit Service 51e54d
    /* An invalid parameter notification from outside the gnulib code.
Packit Service 51e54d
       Give the caller a chance to intervene.  */
Packit Service 51e54d
    RaiseException (STATUS_GNULIB_INVALID_PARAMETER, 0, 0, NULL);
Packit Service 51e54d
}
Packit Service 51e54d
Packit Service 51e54d
#  endif
Packit Service 51e54d
Packit Service 51e54d
# endif
Packit Service 51e54d
Packit Service 51e54d
static int gl_msvc_inval_initialized /* = 0 */;
Packit Service 51e54d
Packit Service 51e54d
void
Packit Service 51e54d
gl_msvc_inval_ensure_handler (void)
Packit Service 51e54d
{
Packit Service 51e54d
  if (gl_msvc_inval_initialized == 0)
Packit Service 51e54d
    {
Packit Service 51e54d
      _set_invalid_parameter_handler (gl_msvc_invalid_parameter_handler);
Packit Service 51e54d
      gl_msvc_inval_initialized = 1;
Packit Service 51e54d
    }
Packit Service 51e54d
}
Packit Service 51e54d
Packit Service 51e54d
#endif