Blame gl/msvc-inval.c

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