Blame lib/msvc-inval.c

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