xref: /Universal-ctags/gnulib/mbtowc-lock.c (revision 820c1a8d46849a90376d8eb15b319ac05439f656)
1*820c1a8dSHiroo HAYASHI /* Return the internal lock used by mbrtowc and mbrtoc32.
2*820c1a8dSHiroo HAYASHI    Copyright (C) 2019-2021 Free Software Foundation, Inc.
3*820c1a8dSHiroo HAYASHI 
4*820c1a8dSHiroo HAYASHI    This file is free software: you can redistribute it and/or modify
5*820c1a8dSHiroo HAYASHI    it under the terms of the GNU Lesser General Public License as
6*820c1a8dSHiroo HAYASHI    published by the Free Software Foundation; either version 2.1 of the
7*820c1a8dSHiroo HAYASHI    License, or (at your option) any later version.
8*820c1a8dSHiroo HAYASHI 
9*820c1a8dSHiroo HAYASHI    This file is distributed in the hope that it will be useful,
10*820c1a8dSHiroo HAYASHI    but WITHOUT ANY WARRANTY; without even the implied warranty of
11*820c1a8dSHiroo HAYASHI    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12*820c1a8dSHiroo HAYASHI    GNU Lesser General Public License for more details.
13*820c1a8dSHiroo HAYASHI 
14*820c1a8dSHiroo HAYASHI    You should have received a copy of the GNU Lesser General Public License
15*820c1a8dSHiroo HAYASHI    along with this program.  If not, see <https://www.gnu.org/licenses/>.  */
16*820c1a8dSHiroo HAYASHI 
17*820c1a8dSHiroo HAYASHI /* Written by Bruno Haible <bruno@clisp.org>, 2019-2020.  */
18*820c1a8dSHiroo HAYASHI 
19*820c1a8dSHiroo HAYASHI #include <config.h>
20*820c1a8dSHiroo HAYASHI 
21*820c1a8dSHiroo HAYASHI /* When it is known that the gl_get_mbtowc_lock function is defined
22*820c1a8dSHiroo HAYASHI    by a dependency library, it should not be defined here.  */
23*820c1a8dSHiroo HAYASHI #if OMIT_MBTOWC_LOCK
24*820c1a8dSHiroo HAYASHI 
25*820c1a8dSHiroo HAYASHI /* This declaration is solely to ensure that after preprocessing
26*820c1a8dSHiroo HAYASHI    this file is never empty.  */
27*820c1a8dSHiroo HAYASHI typedef int dummy;
28*820c1a8dSHiroo HAYASHI 
29*820c1a8dSHiroo HAYASHI #else
30*820c1a8dSHiroo HAYASHI 
31*820c1a8dSHiroo HAYASHI /* This file defines the internal lock used by mbrtowc and mbrtoc32.
32*820c1a8dSHiroo HAYASHI    It is a separate compilation unit, so that only one copy of it is
33*820c1a8dSHiroo HAYASHI    present when linking statically.  */
34*820c1a8dSHiroo HAYASHI 
35*820c1a8dSHiroo HAYASHI /* Prohibit renaming this symbol.  */
36*820c1a8dSHiroo HAYASHI # undef gl_get_mbtowc_lock
37*820c1a8dSHiroo HAYASHI 
38*820c1a8dSHiroo HAYASHI /* Macro for exporting a symbol (function, not variable) defined in this file,
39*820c1a8dSHiroo HAYASHI    when compiled into a shared library.  */
40*820c1a8dSHiroo HAYASHI # ifndef DLL_EXPORTED
41*820c1a8dSHiroo HAYASHI #  if HAVE_VISIBILITY
42*820c1a8dSHiroo HAYASHI   /* Override the effect of the compiler option '-fvisibility=hidden'.  */
43*820c1a8dSHiroo HAYASHI #   define DLL_EXPORTED __attribute__((__visibility__("default")))
44*820c1a8dSHiroo HAYASHI #  elif defined _WIN32 || defined __CYGWIN__
45*820c1a8dSHiroo HAYASHI #   define DLL_EXPORTED __declspec(dllexport)
46*820c1a8dSHiroo HAYASHI #  else
47*820c1a8dSHiroo HAYASHI #   define DLL_EXPORTED
48*820c1a8dSHiroo HAYASHI #  endif
49*820c1a8dSHiroo HAYASHI # endif
50*820c1a8dSHiroo HAYASHI 
51*820c1a8dSHiroo HAYASHI # if defined _WIN32 && !defined __CYGWIN__
52*820c1a8dSHiroo HAYASHI 
53*820c1a8dSHiroo HAYASHI #  define WIN32_LEAN_AND_MEAN  /* avoid including junk */
54*820c1a8dSHiroo HAYASHI #  include <windows.h>
55*820c1a8dSHiroo HAYASHI 
56*820c1a8dSHiroo HAYASHI #  include "windows-initguard.h"
57*820c1a8dSHiroo HAYASHI 
58*820c1a8dSHiroo HAYASHI /* The return type is a 'CRITICAL_SECTION *', not a 'glwthread_mutex_t *',
59*820c1a8dSHiroo HAYASHI    because the latter is not guaranteed to be a stable ABI in the future.  */
60*820c1a8dSHiroo HAYASHI 
61*820c1a8dSHiroo HAYASHI /* Make sure the function gets exported from DLLs.  */
62*820c1a8dSHiroo HAYASHI DLL_EXPORTED CRITICAL_SECTION *gl_get_mbtowc_lock (void);
63*820c1a8dSHiroo HAYASHI 
64*820c1a8dSHiroo HAYASHI static glwthread_initguard_t guard = GLWTHREAD_INITGUARD_INIT;
65*820c1a8dSHiroo HAYASHI static CRITICAL_SECTION lock;
66*820c1a8dSHiroo HAYASHI 
67*820c1a8dSHiroo HAYASHI /* Returns the internal lock used by mbrtowc and mbrtoc32.  */
68*820c1a8dSHiroo HAYASHI CRITICAL_SECTION *
gl_get_mbtowc_lock(void)69*820c1a8dSHiroo HAYASHI gl_get_mbtowc_lock (void)
70*820c1a8dSHiroo HAYASHI {
71*820c1a8dSHiroo HAYASHI   if (!guard.done)
72*820c1a8dSHiroo HAYASHI     {
73*820c1a8dSHiroo HAYASHI       if (InterlockedIncrement (&guard.started) == 0)
74*820c1a8dSHiroo HAYASHI         {
75*820c1a8dSHiroo HAYASHI           /* This thread is the first one to need the lock.  Initialize it.  */
76*820c1a8dSHiroo HAYASHI           InitializeCriticalSection (&lock);
77*820c1a8dSHiroo HAYASHI           guard.done = 1;
78*820c1a8dSHiroo HAYASHI         }
79*820c1a8dSHiroo HAYASHI       else
80*820c1a8dSHiroo HAYASHI         {
81*820c1a8dSHiroo HAYASHI           /* Don't let guard.started grow and wrap around.  */
82*820c1a8dSHiroo HAYASHI           InterlockedDecrement (&guard.started);
83*820c1a8dSHiroo HAYASHI           /* Yield the CPU while waiting for another thread to finish
84*820c1a8dSHiroo HAYASHI              initializing this mutex.  */
85*820c1a8dSHiroo HAYASHI           while (!guard.done)
86*820c1a8dSHiroo HAYASHI             Sleep (0);
87*820c1a8dSHiroo HAYASHI         }
88*820c1a8dSHiroo HAYASHI     }
89*820c1a8dSHiroo HAYASHI   return &lock;
90*820c1a8dSHiroo HAYASHI }
91*820c1a8dSHiroo HAYASHI 
92*820c1a8dSHiroo HAYASHI # elif HAVE_PTHREAD_API
93*820c1a8dSHiroo HAYASHI 
94*820c1a8dSHiroo HAYASHI #  include <pthread.h>
95*820c1a8dSHiroo HAYASHI 
96*820c1a8dSHiroo HAYASHI static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
97*820c1a8dSHiroo HAYASHI 
98*820c1a8dSHiroo HAYASHI /* Make sure the function gets exported from shared libraries.  */
99*820c1a8dSHiroo HAYASHI DLL_EXPORTED pthread_mutex_t *gl_get_mbtowc_lock (void);
100*820c1a8dSHiroo HAYASHI 
101*820c1a8dSHiroo HAYASHI /* Returns the internal lock used by mbrtowc and mbrtoc32.  */
102*820c1a8dSHiroo HAYASHI pthread_mutex_t *
gl_get_mbtowc_lock(void)103*820c1a8dSHiroo HAYASHI gl_get_mbtowc_lock (void)
104*820c1a8dSHiroo HAYASHI {
105*820c1a8dSHiroo HAYASHI   return &mutex;
106*820c1a8dSHiroo HAYASHI }
107*820c1a8dSHiroo HAYASHI 
108*820c1a8dSHiroo HAYASHI # elif HAVE_THREADS_H
109*820c1a8dSHiroo HAYASHI 
110*820c1a8dSHiroo HAYASHI #  include <threads.h>
111*820c1a8dSHiroo HAYASHI #  include <stdlib.h>
112*820c1a8dSHiroo HAYASHI 
113*820c1a8dSHiroo HAYASHI static int volatile init_needed = 1;
114*820c1a8dSHiroo HAYASHI static once_flag init_once = ONCE_FLAG_INIT;
115*820c1a8dSHiroo HAYASHI static mtx_t mutex;
116*820c1a8dSHiroo HAYASHI 
117*820c1a8dSHiroo HAYASHI static void
atomic_init(void)118*820c1a8dSHiroo HAYASHI atomic_init (void)
119*820c1a8dSHiroo HAYASHI {
120*820c1a8dSHiroo HAYASHI   if (mtx_init (&mutex, mtx_plain) != thrd_success)
121*820c1a8dSHiroo HAYASHI     abort ();
122*820c1a8dSHiroo HAYASHI   init_needed = 0;
123*820c1a8dSHiroo HAYASHI }
124*820c1a8dSHiroo HAYASHI 
125*820c1a8dSHiroo HAYASHI /* Make sure the function gets exported from shared libraries.  */
126*820c1a8dSHiroo HAYASHI DLL_EXPORTED mtx_t *gl_get_mbtowc_lock (void);
127*820c1a8dSHiroo HAYASHI 
128*820c1a8dSHiroo HAYASHI /* Returns the internal lock used by mbrtowc and mbrtoc32.  */
129*820c1a8dSHiroo HAYASHI mtx_t *
gl_get_mbtowc_lock(void)130*820c1a8dSHiroo HAYASHI gl_get_mbtowc_lock (void)
131*820c1a8dSHiroo HAYASHI {
132*820c1a8dSHiroo HAYASHI   if (init_needed)
133*820c1a8dSHiroo HAYASHI     call_once (&init_once, atomic_init);
134*820c1a8dSHiroo HAYASHI   return &mutex;
135*820c1a8dSHiroo HAYASHI }
136*820c1a8dSHiroo HAYASHI 
137*820c1a8dSHiroo HAYASHI # endif
138*820c1a8dSHiroo HAYASHI 
139*820c1a8dSHiroo HAYASHI # if (defined _WIN32 || defined __CYGWIN__) && !defined _MSC_VER
140*820c1a8dSHiroo HAYASHI /* Make sure the '__declspec(dllimport)' in mbrtowc.c and mbrtoc32.c does not
141*820c1a8dSHiroo HAYASHI    cause a link failure when no DLLs are involved.  */
142*820c1a8dSHiroo HAYASHI #  if defined _WIN64 || defined _LP64
143*820c1a8dSHiroo HAYASHI #   define IMP(x) __imp_##x
144*820c1a8dSHiroo HAYASHI #  else
145*820c1a8dSHiroo HAYASHI #   define IMP(x) _imp__##x
146*820c1a8dSHiroo HAYASHI #  endif
147*820c1a8dSHiroo HAYASHI void * IMP(gl_get_mbtowc_lock) = &gl_get_mbtowc_lock;
148*820c1a8dSHiroo HAYASHI # endif
149*820c1a8dSHiroo HAYASHI 
150*820c1a8dSHiroo HAYASHI #endif
151