Blame lib/msvc-inval.c

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