Blame lib/msvc-inval.c

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