Blame gl/msvc-inval.c

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