Blame lib/msvc-inval.c

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