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