Line data Source code
1 : /**********************************************************************
2 : *
3 : * Project: CPL - Common Portability Library
4 : * Purpose: CPL Multi-Threading, and process handling portability functions.
5 : * Author: Frank Warmerdam, warmerdam@pobox.com
6 : *
7 : **********************************************************************
8 : * Copyright (c) 2002, Frank Warmerdam
9 : * Copyright (c) 2009-2013, Even Rouault <even dot rouault at spatialys.com>
10 : *
11 : * SPDX-License-Identifier: MIT
12 : ****************************************************************************/
13 :
14 : #ifndef _GNU_SOURCE
15 : #define _GNU_SOURCE
16 : #endif
17 :
18 : // Include cpl_config.h BEFORE cpl_multiproc.h, as the later may undefine
19 : // CPL_MULTIPROC_PTHREAD for mingw case.
20 :
21 : #include "cpl_config.h"
22 : #include "cpl_multiproc.h"
23 :
24 : #ifdef CHECK_THREAD_CAN_ALLOCATE_TLS
25 : #include <cassert>
26 : #endif
27 : #include <cerrno>
28 : #include <cmath>
29 : #include <cstddef>
30 : #include <cstdio>
31 : #include <cstdlib>
32 : #include <cstring>
33 : #include <ctime>
34 : #include <algorithm>
35 :
36 : #include "cpl_atomic_ops.h"
37 : #include "cpl_conv.h"
38 : #include "cpl_error.h"
39 : #include "cpl_string.h"
40 : #include "cpl_vsi.h"
41 :
42 : #if defined(CPL_MULTIPROC_STUB) && !defined(DEBUG)
43 : #define MUTEX_NONE
44 : #endif
45 :
46 : // #define DEBUG_MUTEX
47 :
48 : #if defined(DEBUG) && \
49 : (defined(__GNUC__) && (defined(__i386__) || defined(__x86_64__)))
50 : #ifndef DEBUG_CONTENTION
51 : #define DEBUG_CONTENTION
52 : #endif
53 : #endif
54 :
55 : typedef struct _CPLSpinLock CPLSpinLock;
56 :
57 : struct _CPLLock
58 : {
59 : CPLLockType eType;
60 :
61 : union
62 : {
63 : CPLMutex *hMutex;
64 : CPLSpinLock *hSpinLock;
65 : } u;
66 :
67 : #ifdef DEBUG_CONTENTION
68 : bool bDebugPerfAsked;
69 : bool bDebugPerf;
70 : volatile int nCurrentHolders;
71 : GUIntBig nStartTime;
72 : GIntBig nMaxDiff;
73 : double dfAvgDiff;
74 : GUIntBig nIters;
75 : #endif
76 : };
77 :
78 : #ifdef DEBUG_CONTENTION
79 :
80 : #if defined(__x86_64)
81 : #define GCC_CPUID(level, a, b, c, d) \
82 : __asm__ volatile("xchgq %%rbx, %q1\n" \
83 : "cpuid\n" \
84 : "xchgq %%rbx, %q1" \
85 : : "=a"(a), "=r"(b), "=c"(c), "=d"(d) \
86 : : "0"(level))
87 : #else
88 : #define GCC_CPUID(level, a, b, c, d) \
89 : __asm__ volatile("xchgl %%ebx, %1\n" \
90 : "cpuid\n" \
91 : "xchgl %%ebx, %1" \
92 : : "=a"(a), "=r"(b), "=c"(c), "=d"(d) \
93 : : "0"(level))
94 : #endif
95 :
96 1952510 : static GUIntBig CPLrdtsc()
97 : {
98 : unsigned int a;
99 : unsigned int d;
100 : unsigned int unused1;
101 : unsigned int unused2;
102 : unsigned int unused3;
103 : unsigned int unused4;
104 1952510 : GCC_CPUID(0, unused1, unused2, unused3, unused4);
105 1952510 : __asm__ volatile("rdtsc" : "=a"(a), "=d"(d));
106 1952520 : return static_cast<GUIntBig>(a) | (static_cast<GUIntBig>(d) << 32);
107 : }
108 :
109 1952520 : static GUIntBig CPLrdtscp()
110 : {
111 : unsigned int a;
112 : unsigned int d;
113 : unsigned int unused1;
114 : unsigned int unused2;
115 : unsigned int unused3;
116 : unsigned int unused4;
117 1952520 : __asm__ volatile("rdtscp" : "=a"(a), "=d"(d));
118 1952520 : GCC_CPUID(0, unused1, unused2, unused3, unused4);
119 1952520 : return static_cast<GUIntBig>(a) | (static_cast<GUIntBig>(d) << 32);
120 : }
121 : #endif
122 :
123 : static CPLSpinLock *CPLCreateSpinLock(); // Returned NON acquired.
124 : static int CPLCreateOrAcquireSpinLockInternal(CPLLock **);
125 : static int CPLAcquireSpinLock(CPLSpinLock *);
126 : static void CPLReleaseSpinLock(CPLSpinLock *);
127 : static void CPLDestroySpinLock(CPLSpinLock *);
128 :
129 : #ifndef CPL_MULTIPROC_PTHREAD
130 : #ifndef MUTEX_NONE
131 : static CPLMutex *CPLCreateOrAcquireMasterMutex(double);
132 : static CPLMutex *&CPLCreateOrAcquireMasterMutexInternal(double);
133 : static CPLMutex *CPLCreateUnacquiredMutex();
134 : #endif
135 : #endif
136 :
137 : // We don't want it to be publicly used since it solves rather tricky issues
138 : // that are better to remain hidden.
139 : void CPLFinalizeTLS();
140 :
141 : /************************************************************************/
142 : /* CPLMutexHolder() */
143 : /************************************************************************/
144 :
145 : #ifdef MUTEX_NONE
146 : CPLMutexHolder::CPLMutexHolder(CPLMutex ** /* phMutex */,
147 : double /* dfWaitInSeconds */,
148 : const char * /* pszFileIn */, int /* nLineIn */,
149 : int /* nOptions */)
150 : {
151 : }
152 :
153 : #else
154 67420800 : CPLMutexHolder::CPLMutexHolder(CPLMutex **phMutex, double dfWaitInSeconds,
155 67420800 : const char *pszFileIn, int nLineIn, int nOptions)
156 67420800 : : hMutex(nullptr), pszFile(pszFileIn), nLine(nLineIn)
157 : {
158 67420800 : if (phMutex == nullptr)
159 : {
160 0 : fprintf(stderr, "CPLMutexHolder: phMutex )) NULL !\n");
161 0 : hMutex = nullptr;
162 0 : return;
163 : }
164 :
165 : #ifdef DEBUG_MUTEX
166 : // There is no way to use CPLDebug() here because it works with
167 : // mutexes itself so we will fall in infinite recursion.
168 : // fprintf() will do the job right.
169 : fprintf(stderr, "CPLMutexHolder: Request %p for pid %ld at %d/%s.\n",
170 : *phMutex, static_cast<long>(CPLGetPID()), nLine, pszFile);
171 : #else
172 : // TODO(schwehr): Find a better way to do handle this.
173 : (void)pszFile;
174 : (void)nLine;
175 : #endif
176 :
177 67420800 : if (!CPLCreateOrAcquireMutexEx(phMutex, dfWaitInSeconds, nOptions))
178 : {
179 53 : fprintf(stderr, "CPLMutexHolder: Failed to acquire mutex!\n");
180 0 : hMutex = nullptr;
181 : }
182 : else
183 : {
184 : #ifdef DEBUG_MUTEX
185 : fprintf(stderr, "CPLMutexHolder: Acquired %p for pid %ld at %d/%s.\n",
186 : *phMutex, static_cast<long>(CPLGetPID()), nLine, pszFile);
187 : #endif
188 :
189 67425700 : hMutex = *phMutex;
190 : }
191 : }
192 : #endif // ndef MUTEX_NONE
193 :
194 : /************************************************************************/
195 : /* CPLMutexHolder() */
196 : /************************************************************************/
197 :
198 : #ifdef MUTEX_NONE
199 : CPLMutexHolder::CPLMutexHolder(CPLMutex * /* hMutexIn */,
200 : double /* dfWaitInSeconds */,
201 : const char * /* pszFileIn */, int /* nLineIn */)
202 : {
203 : }
204 : #else
205 :
206 15091 : static CPLMutex *GetMutexHolderMutexMember(CPLMutex *hMutexIn,
207 : double dfWaitInSeconds)
208 : {
209 15091 : if (hMutexIn && !CPLAcquireMutex(hMutexIn, dfWaitInSeconds))
210 : {
211 0 : fprintf(stderr, "CPLMutexHolder: Failed to acquire mutex!\n");
212 0 : return nullptr;
213 : }
214 15091 : return hMutexIn;
215 : }
216 :
217 15091 : CPLMutexHolder::CPLMutexHolder(CPLMutex *hMutexIn, double dfWaitInSeconds,
218 15091 : const char *pszFileIn, int nLineIn)
219 15091 : : hMutex(GetMutexHolderMutexMember(hMutexIn, dfWaitInSeconds)),
220 15091 : pszFile(pszFileIn), nLine(nLineIn)
221 : {
222 15091 : }
223 : #endif // ndef MUTEX_NONE
224 :
225 : /************************************************************************/
226 : /* ~CPLMutexHolder() */
227 : /************************************************************************/
228 :
229 : #ifdef MUTEX_NONE
230 : CPLMutexHolder::~CPLMutexHolder()
231 : {
232 : }
233 : #else
234 134882000 : CPLMutexHolder::~CPLMutexHolder()
235 : {
236 67441000 : if (hMutex != nullptr)
237 : {
238 : #ifdef DEBUG_MUTEX
239 : fprintf(stderr, "~CPLMutexHolder: Release %p for pid %ld at %d/%s.\n",
240 : hMutex, static_cast<long>(CPLGetPID()), nLine, pszFile);
241 : #endif
242 67440900 : CPLReleaseMutex(hMutex);
243 : }
244 67440800 : }
245 : #endif // ndef MUTEX_NONE
246 :
247 1376130 : int CPLCreateOrAcquireMutex(CPLMutex **phMutex, double dfWaitInSeconds)
248 : {
249 1376130 : return CPLCreateOrAcquireMutexEx(phMutex, dfWaitInSeconds,
250 1376150 : CPL_MUTEX_RECURSIVE);
251 : }
252 :
253 : /************************************************************************/
254 : /* CPLCreateOrAcquireMutex() */
255 : /************************************************************************/
256 :
257 : #ifndef CPL_MULTIPROC_PTHREAD
258 :
259 : #ifndef MUTEX_NONE
260 : CPLMutex *CPLCreateUnacquiredMutex()
261 : {
262 : CPLMutex *hMutex = CPLCreateMutex();
263 : if (hMutex)
264 : {
265 : CPLReleaseMutex(hMutex);
266 : }
267 : return hMutex;
268 : }
269 :
270 : CPLMutex *&
271 : CPLCreateOrAcquireMasterMutexInternal(double dfWaitInSeconds = 1000.0)
272 : {
273 : // The dynamic initialization of the block scope hCOAMutex
274 : // with static storage duration is thread-safe in C++11
275 : static CPLMutex *hCOAMutex = CPLCreateUnacquiredMutex();
276 :
277 : // WARNING: although adding an CPLAssert(hCOAMutex); might seem logical
278 : // here, do not enable it (see comment below). It calls CPLError that
279 : // uses the hCOAMutex itself leading to recursive mutex acquisition
280 : // and likely a stack overflow.
281 :
282 : if (!hCOAMutex)
283 : {
284 : // Fall back to this, ironically, NOT thread-safe re-initialisation of
285 : // hCOAMutex in case of a memory error or call to CPLCleanupMasterMutex
286 : // sequenced in an unusual, unexpected or erroneous way.
287 : // For example, an unusual sequence could be:
288 : // GDALDriverManager has been instantiated,
289 : // then OGRCleanupAll is called which calls CPLCleanupMasterMutex,
290 : // then CPLFreeConfig is called which acquires the hCOAMutex
291 : // that has already been released and destroyed.
292 :
293 : hCOAMutex = CPLCreateUnacquiredMutex();
294 : }
295 :
296 : if (hCOAMutex)
297 : {
298 : CPLAcquireMutex(hCOAMutex, dfWaitInSeconds);
299 : }
300 :
301 : return hCOAMutex;
302 : }
303 :
304 : CPLMutex *CPLCreateOrAcquireMasterMutex(double dfWaitInSeconds = 1000.0)
305 : {
306 : CPLMutex *hCOAMutex =
307 : CPLCreateOrAcquireMasterMutexInternal(dfWaitInSeconds);
308 : return hCOAMutex;
309 : }
310 : #endif
311 :
312 : #ifdef MUTEX_NONE
313 :
314 : int CPLCreateOrAcquireMutexEx(CPLMutex **phMutex, double dfWaitInSeconds,
315 : int nOptions)
316 : {
317 : return false;
318 : }
319 : #else
320 : int CPLCreateOrAcquireMutexEx(CPLMutex **phMutex, double dfWaitInSeconds,
321 : int nOptions)
322 : {
323 : bool bSuccess = false;
324 :
325 : CPLMutex *hCOAMutex = CPLCreateOrAcquireMasterMutex(dfWaitInSeconds);
326 : if (hCOAMutex == nullptr)
327 : {
328 : *phMutex = nullptr;
329 : return FALSE;
330 : }
331 :
332 : if (*phMutex == nullptr)
333 : {
334 : *phMutex = CPLCreateMutexEx(nOptions);
335 : bSuccess = *phMutex != nullptr;
336 : CPLReleaseMutex(hCOAMutex);
337 : }
338 : else
339 : {
340 : CPLReleaseMutex(hCOAMutex);
341 :
342 : bSuccess = CPL_TO_BOOL(CPLAcquireMutex(*phMutex, dfWaitInSeconds));
343 : }
344 :
345 : return bSuccess;
346 : }
347 : #endif // ndef MUTEX_NONE
348 :
349 : /************************************************************************/
350 : /* CPLCreateOrAcquireMutexInternal() */
351 : /************************************************************************/
352 :
353 : #ifdef MUTEX_NONE
354 : static bool CPLCreateOrAcquireMutexInternal(CPLLock **phLock,
355 : double dfWaitInSeconds,
356 : CPLLockType eType)
357 : {
358 : return false;
359 : }
360 : #else
361 : static bool CPLCreateOrAcquireMutexInternal(CPLLock **phLock,
362 : double dfWaitInSeconds,
363 : CPLLockType eType)
364 :
365 : {
366 : bool bSuccess = false;
367 :
368 : CPLMutex *hCOAMutex = CPLCreateOrAcquireMasterMutex(dfWaitInSeconds);
369 : if (hCOAMutex == nullptr)
370 : {
371 : *phLock = nullptr;
372 : return FALSE;
373 : }
374 :
375 : if (*phLock == nullptr)
376 : {
377 : *phLock = static_cast<CPLLock *>(calloc(1, sizeof(CPLLock)));
378 : if (*phLock)
379 : {
380 : (*phLock)->eType = eType;
381 : (*phLock)->u.hMutex = CPLCreateMutexEx(
382 : (eType == LOCK_RECURSIVE_MUTEX) ? CPL_MUTEX_RECURSIVE
383 : : CPL_MUTEX_ADAPTIVE);
384 : if ((*phLock)->u.hMutex == nullptr)
385 : {
386 : free(*phLock);
387 : *phLock = nullptr;
388 : }
389 : }
390 : bSuccess = *phLock != nullptr;
391 : CPLReleaseMutex(hCOAMutex);
392 : }
393 : else
394 : {
395 : CPLReleaseMutex(hCOAMutex);
396 :
397 : bSuccess =
398 : CPL_TO_BOOL(CPLAcquireMutex((*phLock)->u.hMutex, dfWaitInSeconds));
399 : }
400 :
401 : return bSuccess;
402 : }
403 : #endif // ndef MUTEX_NONE
404 :
405 : #endif // CPL_MULTIPROC_PTHREAD
406 :
407 : /************************************************************************/
408 : /* CPLCleanupMasterMutex() */
409 : /************************************************************************/
410 :
411 1381 : void CPLCleanupMasterMutex()
412 : {
413 : #ifndef CPL_MULTIPROC_PTHREAD
414 : #ifndef MUTEX_NONE
415 : CPLMutex *&hCOAMutex = CPLCreateOrAcquireMasterMutexInternal();
416 : if (hCOAMutex != nullptr)
417 : {
418 : CPLReleaseMutex(hCOAMutex);
419 : CPLDestroyMutex(hCOAMutex);
420 : hCOAMutex = nullptr;
421 : }
422 : #endif
423 : #endif
424 1381 : }
425 :
426 : /************************************************************************/
427 : /* CPLCleanupTLSList() */
428 : /* */
429 : /* Free resources associated with a TLS vector (implementation */
430 : /* independent). */
431 : /************************************************************************/
432 :
433 3063 : static void CPLCleanupTLSList(void **papTLSList)
434 :
435 : {
436 : #ifdef DEBUG_VERBOSE
437 : printf("CPLCleanupTLSList(%p)\n", papTLSList); /*ok*/
438 : #endif
439 :
440 3063 : if (papTLSList == nullptr)
441 0 : return;
442 :
443 101009 : for (int i = 0; i < CTLS_MAX; i++)
444 : {
445 97946 : if (papTLSList[i] != nullptr && papTLSList[i + CTLS_MAX] != nullptr)
446 : {
447 5992 : CPLTLSFreeFunc pfnFree =
448 5992 : reinterpret_cast<CPLTLSFreeFunc>(papTLSList[i + CTLS_MAX]);
449 5992 : pfnFree(papTLSList[i]);
450 5992 : papTLSList[i] = nullptr;
451 : }
452 : }
453 :
454 3063 : CPLFree(papTLSList);
455 : }
456 :
457 : #if defined(CPL_MULTIPROC_STUB)
458 : /************************************************************************/
459 : /* ==================================================================== */
460 : /* CPL_MULTIPROC_STUB */
461 : /* */
462 : /* Stub implementation. Mutexes don't provide exclusion, file */
463 : /* locking is achieved with extra "lock files", and thread */
464 : /* creation doesn't work. The PID is always just one. */
465 : /* ==================================================================== */
466 : /************************************************************************/
467 :
468 : /************************************************************************/
469 : /* CPLGetNumCPUs() */
470 : /************************************************************************/
471 :
472 : int CPLGetNumCPUs()
473 : {
474 : return 1;
475 : }
476 :
477 : /************************************************************************/
478 : /* CPLGetThreadingModel() */
479 : /************************************************************************/
480 :
481 : const char *CPLGetThreadingModel()
482 :
483 : {
484 : return "stub";
485 : }
486 :
487 : /************************************************************************/
488 : /* CPLCreateMutex() */
489 : /************************************************************************/
490 :
491 : #ifdef MUTEX_NONE
492 : CPLMutex *CPLCreateMutex()
493 : {
494 : return (CPLMutex *)0xdeadbeef;
495 : }
496 : #else
497 : CPLMutex *CPLCreateMutex()
498 : {
499 : unsigned char *pabyMutex = static_cast<unsigned char *>(malloc(4));
500 : if (pabyMutex == nullptr)
501 : return nullptr;
502 :
503 : pabyMutex[0] = 1;
504 : pabyMutex[1] = 'r';
505 : pabyMutex[2] = 'e';
506 : pabyMutex[3] = 'd';
507 :
508 : return (CPLMutex *)pabyMutex;
509 : }
510 : #endif
511 :
512 : CPLMutex *CPLCreateMutexEx(int /*nOptions*/)
513 :
514 : {
515 : return CPLCreateMutex();
516 : }
517 :
518 : /************************************************************************/
519 : /* CPLAcquireMutex() */
520 : /************************************************************************/
521 :
522 : #ifdef MUTEX_NONE
523 : int CPLAcquireMutex(CPLMutex *hMutex, double /* dfWaitInSeconds */)
524 : {
525 : return TRUE;
526 : }
527 : #else
528 : int CPLAcquireMutex(CPLMutex *hMutex, double /*dfWaitInSeconds*/)
529 : {
530 : unsigned char *pabyMutex = reinterpret_cast<unsigned char *>(hMutex);
531 :
532 : CPLAssert(pabyMutex[1] == 'r' && pabyMutex[2] == 'e' &&
533 : pabyMutex[3] == 'd');
534 :
535 : pabyMutex[0] += 1;
536 :
537 : return TRUE;
538 : }
539 : #endif // ! MUTEX_NONE
540 :
541 : /************************************************************************/
542 : /* CPLReleaseMutex() */
543 : /************************************************************************/
544 :
545 : #ifdef MUTEX_NONE
546 : void CPLReleaseMutex(CPLMutex * /* hMutex */)
547 : {
548 : }
549 : #else
550 : void CPLReleaseMutex(CPLMutex *hMutex)
551 : {
552 : unsigned char *pabyMutex = reinterpret_cast<unsigned char *>(hMutex);
553 :
554 : CPLAssert(pabyMutex[1] == 'r' && pabyMutex[2] == 'e' &&
555 : pabyMutex[3] == 'd');
556 :
557 : if (pabyMutex[0] < 1)
558 : CPLDebug("CPLMultiProc",
559 : "CPLReleaseMutex() called on mutex with %d as ref count!",
560 : pabyMutex[0]);
561 :
562 : pabyMutex[0] -= 1;
563 : }
564 : #endif
565 :
566 : /************************************************************************/
567 : /* CPLDestroyMutex() */
568 : /************************************************************************/
569 :
570 : #ifdef MUTEX_NONE
571 : void CPLDestroyMutex(CPLMutex * /* hMutex */)
572 : {
573 : }
574 : #else
575 : void CPLDestroyMutex(CPLMutex *hMutex)
576 : {
577 : unsigned char *pabyMutex = reinterpret_cast<unsigned char *>(hMutex);
578 :
579 : CPLAssert(pabyMutex[1] == 'r' && pabyMutex[2] == 'e' &&
580 : pabyMutex[3] == 'd');
581 :
582 : free(pabyMutex);
583 : }
584 : #endif
585 :
586 : /************************************************************************/
587 : /* CPLCreateCond() */
588 : /************************************************************************/
589 :
590 : CPLCond *CPLCreateCond()
591 : {
592 : return nullptr;
593 : }
594 :
595 : /************************************************************************/
596 : /* CPLCondWait() */
597 : /************************************************************************/
598 :
599 : void CPLCondWait(CPLCond * /* hCond */, CPLMutex * /* hMutex */)
600 : {
601 : }
602 :
603 : /************************************************************************/
604 : /* CPLCondTimedWait() */
605 : /************************************************************************/
606 :
607 : CPLCondTimedWaitReason CPLCondTimedWait(CPLCond * /* hCond */,
608 : CPLMutex * /* hMutex */, double)
609 : {
610 : return COND_TIMED_WAIT_OTHER;
611 : }
612 :
613 : /************************************************************************/
614 : /* CPLCondSignal() */
615 : /************************************************************************/
616 :
617 : void CPLCondSignal(CPLCond * /* hCond */)
618 : {
619 : }
620 :
621 : /************************************************************************/
622 : /* CPLCondBroadcast() */
623 : /************************************************************************/
624 :
625 : void CPLCondBroadcast(CPLCond * /* hCond */)
626 : {
627 : }
628 :
629 : /************************************************************************/
630 : /* CPLDestroyCond() */
631 : /************************************************************************/
632 :
633 : void CPLDestroyCond(CPLCond * /* hCond */)
634 : {
635 : }
636 :
637 : /************************************************************************/
638 : /* CPLLockFile() */
639 : /* */
640 : /* Lock a file. This implementation has a terrible race */
641 : /* condition. If we don't succeed in opening the lock file, we */
642 : /* assume we can create one and own the target file, but other */
643 : /* processes might easily try creating the target file at the */
644 : /* same time, overlapping us. Death! Mayhem! The traditional */
645 : /* solution is to use open() with _O_CREAT|_O_EXCL but this */
646 : /* function and these arguments aren't trivially portable. */
647 : /* Also, this still leaves a race condition on NFS drivers */
648 : /* (apparently). */
649 : /************************************************************************/
650 :
651 : void *CPLLockFile(const char *pszPath, double dfWaitInSeconds)
652 :
653 : {
654 : /* -------------------------------------------------------------------- */
655 : /* We use a lock file with a name derived from the file we want */
656 : /* to lock to represent the file being locked. Note that for */
657 : /* the stub implementation the target file does not even need */
658 : /* to exist to be locked. */
659 : /* -------------------------------------------------------------------- */
660 : char *pszLockFilename =
661 : static_cast<char *>(CPLMalloc(strlen(pszPath) + 30));
662 : snprintf(pszLockFilename, strlen(pszPath) + 30, "%s.lock", pszPath);
663 :
664 : FILE *fpLock = fopen(pszLockFilename, "r");
665 : while (fpLock != nullptr && dfWaitInSeconds > 0.0)
666 : {
667 : fclose(fpLock);
668 : CPLSleep(std::min(dfWaitInSeconds, 0.5));
669 : dfWaitInSeconds -= 0.5;
670 :
671 : fpLock = fopen(pszLockFilename, "r");
672 : }
673 :
674 : if (fpLock != nullptr)
675 : {
676 : fclose(fpLock);
677 : CPLFree(pszLockFilename);
678 : return nullptr;
679 : }
680 :
681 : fpLock = fopen(pszLockFilename, "w");
682 :
683 : if (fpLock == nullptr)
684 : {
685 : CPLFree(pszLockFilename);
686 : return nullptr;
687 : }
688 :
689 : fwrite("held\n", 1, 5, fpLock);
690 : fclose(fpLock);
691 :
692 : return pszLockFilename;
693 : }
694 :
695 : /************************************************************************/
696 : /* CPLUnlockFile() */
697 : /************************************************************************/
698 :
699 : void CPLUnlockFile(void *hLock)
700 :
701 : {
702 : char *pszLockFilename = static_cast<char *>(hLock);
703 :
704 : if (hLock == nullptr)
705 : return;
706 :
707 : VSIUnlink(pszLockFilename);
708 :
709 : CPLFree(pszLockFilename);
710 : }
711 :
712 : /************************************************************************/
713 : /* CPLGetPID() */
714 : /************************************************************************/
715 :
716 : GIntBig CPLGetPID()
717 :
718 : {
719 : return 1;
720 : }
721 :
722 : /************************************************************************/
723 : /* CPLCreateThread(); */
724 : /************************************************************************/
725 :
726 : int CPLCreateThread(CPLThreadFunc /* pfnMain */, void * /* pArg */)
727 : {
728 : CPLDebug("CPLCreateThread", "Fails to dummy implementation");
729 :
730 : return -1;
731 : }
732 :
733 : /************************************************************************/
734 : /* CPLCreateJoinableThread() */
735 : /************************************************************************/
736 :
737 : CPLJoinableThread *CPLCreateJoinableThread(CPLThreadFunc /* pfnMain */,
738 : void * /* pThreadArg */)
739 : {
740 : CPLDebug("CPLCreateJoinableThread", "Fails to dummy implementation");
741 :
742 : return nullptr;
743 : }
744 :
745 : /************************************************************************/
746 : /* CPLJoinThread() */
747 : /************************************************************************/
748 :
749 : void CPLJoinThread(CPLJoinableThread * /* hJoinableThread */)
750 : {
751 : }
752 :
753 : /************************************************************************/
754 : /* CPLSleep() */
755 : /************************************************************************/
756 :
757 : void CPLSleep(double dfWaitInSeconds)
758 : {
759 : time_t ltime;
760 :
761 : time(<ime);
762 : const time_t ttime = ltime + static_cast<int>(dfWaitInSeconds + 0.5);
763 :
764 : for (; ltime < ttime; time(<ime))
765 : {
766 : // Currently we just busy wait. Perhaps we could at least block on io?
767 : }
768 : }
769 :
770 : /************************************************************************/
771 : /* CPLGetTLSList() */
772 : /************************************************************************/
773 :
774 : static void **papTLSList = nullptr;
775 :
776 : static void **CPLGetTLSList(int *pbMemoryErrorOccurred)
777 :
778 : {
779 : if (pbMemoryErrorOccurred)
780 : *pbMemoryErrorOccurred = FALSE;
781 : if (papTLSList == nullptr)
782 : {
783 : papTLSList =
784 : static_cast<void **>(VSICalloc(sizeof(void *), CTLS_MAX * 2));
785 : if (papTLSList == nullptr)
786 : {
787 : if (pbMemoryErrorOccurred)
788 : {
789 : *pbMemoryErrorOccurred = TRUE;
790 : fprintf(stderr,
791 : "CPLGetTLSList() failed to allocate TLS list!\n");
792 : return nullptr;
793 : }
794 : CPLEmergencyError("CPLGetTLSList() failed to allocate TLS list!");
795 : }
796 : }
797 :
798 : return papTLSList;
799 : }
800 :
801 : /************************************************************************/
802 : /* CPLFinalizeTLS() */
803 : /************************************************************************/
804 :
805 : void CPLFinalizeTLS()
806 : {
807 : CPLCleanupTLS();
808 : }
809 :
810 : /************************************************************************/
811 : /* CPLCleanupTLS() */
812 : /************************************************************************/
813 :
814 : void CPLCleanupTLS()
815 :
816 : {
817 : CPLCleanupTLSList(papTLSList);
818 : papTLSList = nullptr;
819 : }
820 :
821 : // endif CPL_MULTIPROC_STUB
822 :
823 : #elif defined(CPL_MULTIPROC_WIN32)
824 :
825 : /************************************************************************/
826 : /* ==================================================================== */
827 : /* CPL_MULTIPROC_WIN32 */
828 : /* */
829 : /* WIN32 Implementation of multiprocessing functions. */
830 : /* ==================================================================== */
831 : /************************************************************************/
832 :
833 : /* InitializeCriticalSectionAndSpinCount requires _WIN32_WINNT >= 0x403 */
834 : #undef _WIN32_WINNT
835 : #define _WIN32_WINNT 0x0500
836 :
837 : #include <windows.h>
838 :
839 : /************************************************************************/
840 : /* CPLGetNumCPUs() */
841 : /************************************************************************/
842 :
843 : int CPLGetNumCPUs()
844 : {
845 : SYSTEM_INFO info;
846 : GetSystemInfo(&info);
847 : const DWORD dwNum = info.dwNumberOfProcessors;
848 : if (dwNum < 1)
849 : return 1;
850 : return static_cast<int>(dwNum);
851 : }
852 :
853 : /************************************************************************/
854 : /* CPLGetThreadingModel() */
855 : /************************************************************************/
856 :
857 : const char *CPLGetThreadingModel()
858 :
859 : {
860 : return "win32";
861 : }
862 :
863 : /************************************************************************/
864 : /* CPLCreateMutex() */
865 : /************************************************************************/
866 :
867 : CPLMutex *CPLCreateMutex()
868 :
869 : {
870 : #ifdef USE_WIN32_MUTEX
871 : HANDLE hMutex = CreateMutex(nullptr, TRUE, nullptr);
872 :
873 : return (CPLMutex *)hMutex;
874 : #else
875 :
876 : // Do not use CPLMalloc() since its debugging infrastructure
877 : // can call the CPL*Mutex functions.
878 : CRITICAL_SECTION *pcs =
879 : static_cast<CRITICAL_SECTION *>(malloc(sizeof(*pcs)));
880 : if (pcs)
881 : {
882 : InitializeCriticalSectionAndSpinCount(pcs, 4000);
883 : EnterCriticalSection(pcs);
884 : }
885 :
886 : return reinterpret_cast<CPLMutex *>(pcs);
887 : #endif
888 : }
889 :
890 : CPLMutex *CPLCreateMutexEx(int /* nOptions */)
891 :
892 : {
893 : return CPLCreateMutex();
894 : }
895 :
896 : /************************************************************************/
897 : /* CPLAcquireMutex() */
898 : /************************************************************************/
899 :
900 : int CPLAcquireMutex(CPLMutex *hMutexIn, double dfWaitInSeconds)
901 :
902 : {
903 : #ifdef USE_WIN32_MUTEX
904 : HANDLE hMutex = (HANDLE)hMutexIn;
905 : const DWORD hr =
906 : WaitForSingleObject(hMutex, static_cast<int>(dfWaitInSeconds * 1000));
907 :
908 : return hr != WAIT_TIMEOUT;
909 : #else
910 : CRITICAL_SECTION *pcs = reinterpret_cast<CRITICAL_SECTION *>(hMutexIn);
911 : BOOL ret;
912 :
913 : if (dfWaitInSeconds >= 1000.0)
914 : {
915 : // We assume this is the synonymous for infinite, so it is more
916 : // efficient to use EnterCriticalSection() directly
917 : EnterCriticalSection(pcs);
918 : ret = TRUE;
919 : }
920 : else
921 : {
922 : while ((ret = TryEnterCriticalSection(pcs)) == 0 &&
923 : dfWaitInSeconds > 0.0)
924 : {
925 : CPLSleep(std::min(dfWaitInSeconds, 0.01));
926 : dfWaitInSeconds -= 0.01;
927 : }
928 : }
929 :
930 : return ret;
931 : #endif
932 : }
933 :
934 : /************************************************************************/
935 : /* CPLReleaseMutex() */
936 : /************************************************************************/
937 :
938 : void CPLReleaseMutex(CPLMutex *hMutexIn)
939 :
940 : {
941 : #ifdef USE_WIN32_MUTEX
942 : HANDLE hMutex = (HANDLE)hMutexIn;
943 :
944 : ReleaseMutex(hMutex);
945 : #else
946 : CRITICAL_SECTION *pcs = reinterpret_cast<CRITICAL_SECTION *>(hMutexIn);
947 :
948 : LeaveCriticalSection(pcs);
949 : #endif
950 : }
951 :
952 : /************************************************************************/
953 : /* CPLDestroyMutex() */
954 : /************************************************************************/
955 :
956 : void CPLDestroyMutex(CPLMutex *hMutexIn)
957 :
958 : {
959 : #ifdef USE_WIN32_MUTEX
960 : HANDLE hMutex = (HANDLE)hMutexIn;
961 :
962 : CloseHandle(hMutex);
963 : #else
964 : CRITICAL_SECTION *pcs = reinterpret_cast<CRITICAL_SECTION *>(hMutexIn);
965 :
966 : DeleteCriticalSection(pcs);
967 : free(pcs);
968 : #endif
969 : }
970 :
971 : /************************************************************************/
972 : /* CPLCreateCond() */
973 : /************************************************************************/
974 :
975 : struct _WaiterItem
976 : {
977 : HANDLE hEvent;
978 : struct _WaiterItem *psNext;
979 : };
980 : typedef struct _WaiterItem WaiterItem;
981 :
982 : typedef struct
983 : {
984 : CPLMutex *hInternalMutex;
985 : WaiterItem *psWaiterList;
986 : } Win32Cond;
987 :
988 : CPLCond *CPLCreateCond()
989 : {
990 : Win32Cond *psCond = static_cast<Win32Cond *>(malloc(sizeof(Win32Cond)));
991 : if (psCond == nullptr)
992 : return nullptr;
993 : psCond->hInternalMutex = CPLCreateMutex();
994 : if (psCond->hInternalMutex == nullptr)
995 : {
996 : free(psCond);
997 : return nullptr;
998 : }
999 : CPLReleaseMutex(psCond->hInternalMutex);
1000 : psCond->psWaiterList = nullptr;
1001 : return reinterpret_cast<CPLCond *>(psCond);
1002 : }
1003 :
1004 : /************************************************************************/
1005 : /* CPLCondWait() */
1006 : /************************************************************************/
1007 :
1008 : static void CPLTLSFreeEvent(void *pData)
1009 : {
1010 : CloseHandle(static_cast<HANDLE>(pData));
1011 : }
1012 :
1013 : void CPLCondWait(CPLCond *hCond, CPLMutex *hClientMutex)
1014 : {
1015 : CPLCondTimedWait(hCond, hClientMutex, -1);
1016 : }
1017 :
1018 : /************************************************************************/
1019 : /* CPLCondTimedWait() */
1020 : /************************************************************************/
1021 :
1022 : CPLCondTimedWaitReason CPLCondTimedWait(CPLCond *hCond, CPLMutex *hClientMutex,
1023 : double dfWaitInSeconds)
1024 : {
1025 : Win32Cond *psCond = reinterpret_cast<Win32Cond *>(hCond);
1026 :
1027 : HANDLE hEvent = static_cast<HANDLE>(CPLGetTLS(CTLS_WIN32_COND));
1028 : if (hEvent == nullptr)
1029 : {
1030 : hEvent = CreateEvent(nullptr, /* security attributes */
1031 : 0, /* manual reset = no */
1032 : 0, /* initial state = unsignaled */
1033 : nullptr /* no name */);
1034 : CPLAssert(hEvent != nullptr);
1035 :
1036 : CPLSetTLSWithFreeFunc(CTLS_WIN32_COND, hEvent, CPLTLSFreeEvent);
1037 : }
1038 :
1039 : /* Insert the waiter into the waiter list of the condition */
1040 : CPLAcquireMutex(psCond->hInternalMutex, 1000.0);
1041 :
1042 : WaiterItem *psItem = static_cast<WaiterItem *>(malloc(sizeof(WaiterItem)));
1043 : CPLAssert(psItem != nullptr);
1044 :
1045 : psItem->hEvent = hEvent;
1046 : psItem->psNext = psCond->psWaiterList;
1047 :
1048 : psCond->psWaiterList = psItem;
1049 :
1050 : CPLReleaseMutex(psCond->hInternalMutex);
1051 :
1052 : // Release the client mutex before waiting for the event being signaled.
1053 : CPLReleaseMutex(hClientMutex);
1054 :
1055 : // Ideally we would check that we do not get WAIT_FAILED but it is hard
1056 : // to report a failure.
1057 : auto ret = WaitForSingleObject(
1058 : hEvent, dfWaitInSeconds < 0 ? INFINITE
1059 : : static_cast<int>(dfWaitInSeconds * 1000));
1060 :
1061 : // Reacquire the client mutex.
1062 : CPLAcquireMutex(hClientMutex, 1000.0);
1063 :
1064 : if (ret == WAIT_OBJECT_0)
1065 : return COND_TIMED_WAIT_COND;
1066 : if (ret == WAIT_TIMEOUT)
1067 : return COND_TIMED_WAIT_TIME_OUT;
1068 : return COND_TIMED_WAIT_OTHER;
1069 : }
1070 :
1071 : /************************************************************************/
1072 : /* CPLCondSignal() */
1073 : /************************************************************************/
1074 :
1075 : void CPLCondSignal(CPLCond *hCond)
1076 : {
1077 : Win32Cond *psCond = reinterpret_cast<Win32Cond *>(hCond);
1078 :
1079 : // Signal the first registered event, and remove it from the list.
1080 : CPLAcquireMutex(psCond->hInternalMutex, 1000.0);
1081 :
1082 : WaiterItem *psIter = psCond->psWaiterList;
1083 : if (psIter != nullptr)
1084 : {
1085 : SetEvent(psIter->hEvent);
1086 : psCond->psWaiterList = psIter->psNext;
1087 : free(psIter);
1088 : }
1089 :
1090 : CPLReleaseMutex(psCond->hInternalMutex);
1091 : }
1092 :
1093 : /************************************************************************/
1094 : /* CPLCondBroadcast() */
1095 : /************************************************************************/
1096 :
1097 : void CPLCondBroadcast(CPLCond *hCond)
1098 : {
1099 : Win32Cond *psCond = reinterpret_cast<Win32Cond *>(hCond);
1100 :
1101 : // Signal all the registered events, and remove them from the list.
1102 : CPLAcquireMutex(psCond->hInternalMutex, 1000.0);
1103 :
1104 : WaiterItem *psIter = psCond->psWaiterList;
1105 : while (psIter != nullptr)
1106 : {
1107 : WaiterItem *psNext = psIter->psNext;
1108 : SetEvent(psIter->hEvent);
1109 : free(psIter);
1110 : psIter = psNext;
1111 : }
1112 : psCond->psWaiterList = nullptr;
1113 :
1114 : CPLReleaseMutex(psCond->hInternalMutex);
1115 : }
1116 :
1117 : /************************************************************************/
1118 : /* CPLDestroyCond() */
1119 : /************************************************************************/
1120 :
1121 : void CPLDestroyCond(CPLCond *hCond)
1122 : {
1123 : Win32Cond *psCond = reinterpret_cast<Win32Cond *>(hCond);
1124 : CPLDestroyMutex(psCond->hInternalMutex);
1125 : psCond->hInternalMutex = nullptr;
1126 : CPLAssert(psCond->psWaiterList == nullptr);
1127 : free(psCond);
1128 : }
1129 :
1130 : /************************************************************************/
1131 : /* CPLLockFile() */
1132 : /************************************************************************/
1133 :
1134 : void *CPLLockFile(const char *pszPath, double dfWaitInSeconds)
1135 :
1136 : {
1137 : char *pszLockFilename =
1138 : static_cast<char *>(CPLMalloc(strlen(pszPath) + 30));
1139 : snprintf(pszLockFilename, strlen(pszPath) + 30, "%s.lock", pszPath);
1140 :
1141 : // FIXME: use CreateFileW()
1142 : HANDLE hLockFile =
1143 : CreateFileA(pszLockFilename, GENERIC_WRITE, 0, nullptr, CREATE_NEW,
1144 : FILE_ATTRIBUTE_NORMAL | FILE_FLAG_DELETE_ON_CLOSE, nullptr);
1145 :
1146 : while (GetLastError() == ERROR_ALREADY_EXISTS && dfWaitInSeconds > 0.0)
1147 : {
1148 : CloseHandle(hLockFile);
1149 : CPLSleep(std::min(dfWaitInSeconds, 0.125));
1150 : dfWaitInSeconds -= 0.125;
1151 :
1152 : hLockFile = CreateFileA(
1153 : pszLockFilename, GENERIC_WRITE, 0, nullptr, CREATE_NEW,
1154 : FILE_ATTRIBUTE_NORMAL | FILE_FLAG_DELETE_ON_CLOSE, nullptr);
1155 : }
1156 :
1157 : CPLFree(pszLockFilename);
1158 :
1159 : if (hLockFile == INVALID_HANDLE_VALUE)
1160 : return nullptr;
1161 :
1162 : if (GetLastError() == ERROR_ALREADY_EXISTS)
1163 : {
1164 : CloseHandle(hLockFile);
1165 : return nullptr;
1166 : }
1167 :
1168 : return static_cast<void *>(hLockFile);
1169 : }
1170 :
1171 : /************************************************************************/
1172 : /* CPLUnlockFile() */
1173 : /************************************************************************/
1174 :
1175 : void CPLUnlockFile(void *hLock)
1176 :
1177 : {
1178 : HANDLE hLockFile = static_cast<HANDLE>(hLock);
1179 :
1180 : CloseHandle(hLockFile);
1181 : }
1182 :
1183 : /************************************************************************/
1184 : /* CPLGetPID() */
1185 : /************************************************************************/
1186 :
1187 : GIntBig CPLGetPID()
1188 :
1189 : {
1190 : return static_cast<GIntBig>(GetCurrentThreadId());
1191 : }
1192 :
1193 : /************************************************************************/
1194 : /* CPLStdCallThreadJacket() */
1195 : /************************************************************************/
1196 :
1197 : typedef struct
1198 : {
1199 : void *pAppData;
1200 : CPLThreadFunc pfnMain;
1201 : HANDLE hThread;
1202 : } CPLStdCallThreadInfo;
1203 :
1204 : static DWORD WINAPI CPLStdCallThreadJacket(void *pData)
1205 :
1206 : {
1207 : CPLStdCallThreadInfo *psInfo = static_cast<CPLStdCallThreadInfo *>(pData);
1208 :
1209 : psInfo->pfnMain(psInfo->pAppData);
1210 :
1211 : if (psInfo->hThread == nullptr)
1212 : CPLFree(psInfo); // Only for detached threads.
1213 :
1214 : CPLCleanupTLS();
1215 :
1216 : return 0;
1217 : }
1218 :
1219 : /************************************************************************/
1220 : /* CPLCreateThread() */
1221 : /* */
1222 : /* The WIN32 CreateThread() call requires an entry point that */
1223 : /* has __stdcall conventions, so we provide a jacket function */
1224 : /* to supply that. */
1225 : /************************************************************************/
1226 :
1227 : int CPLCreateThread(CPLThreadFunc pfnMain, void *pThreadArg)
1228 :
1229 : {
1230 : CPLStdCallThreadInfo *psInfo = static_cast<CPLStdCallThreadInfo *>(
1231 : CPLCalloc(sizeof(CPLStdCallThreadInfo), 1));
1232 : psInfo->pAppData = pThreadArg;
1233 : psInfo->pfnMain = pfnMain;
1234 : psInfo->hThread = nullptr;
1235 :
1236 : DWORD nThreadId = 0;
1237 : HANDLE hThread =
1238 : CreateThread(nullptr, 0, CPLStdCallThreadJacket, psInfo, 0, &nThreadId);
1239 :
1240 : if (hThread == nullptr)
1241 : return -1;
1242 :
1243 : CloseHandle(hThread);
1244 :
1245 : return nThreadId;
1246 : }
1247 :
1248 : /************************************************************************/
1249 : /* CPLCreateJoinableThread() */
1250 : /************************************************************************/
1251 :
1252 : CPLJoinableThread *CPLCreateJoinableThread(CPLThreadFunc pfnMain,
1253 : void *pThreadArg)
1254 :
1255 : {
1256 : CPLStdCallThreadInfo *psInfo = static_cast<CPLStdCallThreadInfo *>(
1257 : CPLCalloc(sizeof(CPLStdCallThreadInfo), 1));
1258 : psInfo->pAppData = pThreadArg;
1259 : psInfo->pfnMain = pfnMain;
1260 :
1261 : DWORD nThreadId = 0;
1262 : HANDLE hThread =
1263 : CreateThread(nullptr, 0, CPLStdCallThreadJacket, psInfo, 0, &nThreadId);
1264 :
1265 : if (hThread == nullptr)
1266 : return nullptr;
1267 :
1268 : psInfo->hThread = hThread;
1269 : return reinterpret_cast<CPLJoinableThread *>(psInfo);
1270 : }
1271 :
1272 : /************************************************************************/
1273 : /* CPLJoinThread() */
1274 : /************************************************************************/
1275 :
1276 : void CPLJoinThread(CPLJoinableThread *hJoinableThread)
1277 : {
1278 : CPLStdCallThreadInfo *psInfo =
1279 : reinterpret_cast<CPLStdCallThreadInfo *>(hJoinableThread);
1280 :
1281 : WaitForSingleObject(psInfo->hThread, INFINITE);
1282 : CloseHandle(psInfo->hThread);
1283 : CPLFree(psInfo);
1284 : }
1285 :
1286 : /************************************************************************/
1287 : /* CPLSleep() */
1288 : /************************************************************************/
1289 :
1290 : void CPLSleep(double dfWaitInSeconds)
1291 :
1292 : {
1293 : Sleep(static_cast<DWORD>(dfWaitInSeconds * 1000.0));
1294 : }
1295 :
1296 : static bool bTLSKeySetup = false;
1297 : static DWORD nTLSKey = 0;
1298 :
1299 : /************************************************************************/
1300 : /* CPLGetTLSList() */
1301 : /************************************************************************/
1302 :
1303 : static void **CPLGetTLSList(int *pbMemoryErrorOccurred)
1304 :
1305 : {
1306 : void **papTLSList = nullptr;
1307 :
1308 : if (pbMemoryErrorOccurred)
1309 : *pbMemoryErrorOccurred = FALSE;
1310 : if (!bTLSKeySetup)
1311 : {
1312 : nTLSKey = TlsAlloc();
1313 : if (nTLSKey == TLS_OUT_OF_INDEXES)
1314 : {
1315 : if (pbMemoryErrorOccurred)
1316 : {
1317 : *pbMemoryErrorOccurred = TRUE;
1318 : fprintf(stderr, "CPLGetTLSList(): TlsAlloc() failed!\n");
1319 : return nullptr;
1320 : }
1321 : CPLEmergencyError("CPLGetTLSList(): TlsAlloc() failed!");
1322 : }
1323 : bTLSKeySetup = true;
1324 : }
1325 :
1326 : papTLSList = static_cast<void **>(TlsGetValue(nTLSKey));
1327 : if (papTLSList == nullptr)
1328 : {
1329 : papTLSList =
1330 : static_cast<void **>(VSICalloc(sizeof(void *), CTLS_MAX * 2));
1331 : if (papTLSList == nullptr)
1332 : {
1333 : if (pbMemoryErrorOccurred)
1334 : {
1335 : *pbMemoryErrorOccurred = TRUE;
1336 : fprintf(stderr,
1337 : "CPLGetTLSList() failed to allocate TLS list!\n");
1338 : return nullptr;
1339 : }
1340 : CPLEmergencyError("CPLGetTLSList() failed to allocate TLS list!");
1341 : }
1342 : if (TlsSetValue(nTLSKey, papTLSList) == 0)
1343 : {
1344 : if (pbMemoryErrorOccurred)
1345 : {
1346 : *pbMemoryErrorOccurred = TRUE;
1347 : fprintf(stderr, "CPLGetTLSList(): TlsSetValue() failed!\n");
1348 : return nullptr;
1349 : }
1350 : CPLEmergencyError("CPLGetTLSList(): TlsSetValue() failed!");
1351 : }
1352 : }
1353 :
1354 : return papTLSList;
1355 : }
1356 :
1357 : /************************************************************************/
1358 : /* CPLFinalizeTLS() */
1359 : /************************************************************************/
1360 :
1361 : void CPLFinalizeTLS()
1362 : {
1363 : CPLCleanupTLS();
1364 : }
1365 :
1366 : /************************************************************************/
1367 : /* CPLCleanupTLS() */
1368 : /************************************************************************/
1369 :
1370 : void CPLCleanupTLS()
1371 :
1372 : {
1373 : if (!bTLSKeySetup)
1374 : return;
1375 :
1376 : void **papTLSList = static_cast<void **>(TlsGetValue(nTLSKey));
1377 : if (papTLSList == nullptr)
1378 : return;
1379 :
1380 : TlsSetValue(nTLSKey, nullptr);
1381 :
1382 : CPLCleanupTLSList(papTLSList);
1383 : }
1384 :
1385 : // endif CPL_MULTIPROC_WIN32
1386 :
1387 : #elif defined(CPL_MULTIPROC_PTHREAD)
1388 :
1389 : #include <pthread.h>
1390 : #include <time.h>
1391 : #include <unistd.h>
1392 : #include <sys/time.h>
1393 :
1394 : #ifdef HAVE_SCHED_GETAFFINITY
1395 : #include <sched.h>
1396 : #endif
1397 :
1398 : /************************************************************************/
1399 : /* ==================================================================== */
1400 : /* CPL_MULTIPROC_PTHREAD */
1401 : /* */
1402 : /* PTHREAD Implementation of multiprocessing functions. */
1403 : /* ==================================================================== */
1404 : /************************************************************************/
1405 :
1406 : /************************************************************************/
1407 : /* CPLGetNumCPUs() */
1408 : /************************************************************************/
1409 :
1410 4408 : int CPLGetNumCPUs()
1411 : {
1412 : int nCPUs;
1413 : #ifdef _SC_NPROCESSORS_ONLN
1414 4408 : nCPUs = static_cast<int>(sysconf(_SC_NPROCESSORS_ONLN));
1415 : #else
1416 : nCPUs = 1;
1417 : #endif
1418 :
1419 : #ifdef HAVE_SCHED_GETAFFINITY
1420 4408 : if (nCPUs > 1)
1421 : {
1422 4408 : cpu_set_t *set = CPU_ALLOC(nCPUs);
1423 4408 : if (set)
1424 : {
1425 4408 : size_t sizeof_set = CPU_ALLOC_SIZE(nCPUs);
1426 4408 : CPU_ZERO_S(sizeof_set, set);
1427 4408 : if (sched_getaffinity(getpid(), sizeof_set, set) == 0)
1428 4408 : nCPUs = CPU_COUNT_S(sizeof_set, set);
1429 : else
1430 0 : CPLDebug("CPL", "sched_getaffinity() failed");
1431 4408 : CPU_FREE(set);
1432 : }
1433 : }
1434 : #endif
1435 :
1436 4408 : return nCPUs;
1437 : }
1438 :
1439 : /************************************************************************/
1440 : /* CPLCreateOrAcquireMutex() */
1441 : /************************************************************************/
1442 : #ifdef HAVE_GCC_WARNING_ZERO_AS_NULL_POINTER_CONSTANT
1443 : #pragma GCC diagnostic push
1444 : #pragma GCC diagnostic ignored "-Wzero-as-null-pointer-constant"
1445 : #endif
1446 : static pthread_mutex_t global_mutex = PTHREAD_MUTEX_INITIALIZER;
1447 : #ifdef HAVE_GCC_WARNING_ZERO_AS_NULL_POINTER_CONSTANT
1448 : #pragma GCC diagnostic pop
1449 : #endif
1450 :
1451 : static CPLMutex *CPLCreateMutexInternal(bool bAlreadyInGlobalLock,
1452 : int nOptions);
1453 :
1454 68786100 : int CPLCreateOrAcquireMutexEx(CPLMutex **phMutex, double dfWaitInSeconds,
1455 : int nOptions)
1456 :
1457 : {
1458 68786100 : pthread_mutex_lock(&global_mutex);
1459 68802200 : if (*phMutex == nullptr)
1460 : {
1461 32594 : *phMutex = CPLCreateMutexInternal(true, nOptions);
1462 32594 : const bool bSuccess = *phMutex != nullptr;
1463 32594 : pthread_mutex_unlock(&global_mutex);
1464 32594 : if (!bSuccess)
1465 0 : return false;
1466 : }
1467 : else
1468 : {
1469 68769700 : pthread_mutex_unlock(&global_mutex);
1470 : }
1471 :
1472 68801900 : return CPL_TO_BOOL(CPLAcquireMutex(*phMutex, dfWaitInSeconds));
1473 : }
1474 :
1475 : /************************************************************************/
1476 : /* CPLCreateOrAcquireMutexInternal() */
1477 : /************************************************************************/
1478 :
1479 5046 : static bool CPLCreateOrAcquireMutexInternal(CPLLock **phLock,
1480 : double dfWaitInSeconds,
1481 : CPLLockType eType)
1482 : {
1483 5046 : pthread_mutex_lock(&global_mutex);
1484 5046 : if (*phLock == nullptr)
1485 : {
1486 546 : *phLock = static_cast<CPLLock *>(calloc(1, sizeof(CPLLock)));
1487 546 : if (*phLock)
1488 : {
1489 546 : (*phLock)->eType = eType;
1490 1092 : (*phLock)->u.hMutex = CPLCreateMutexInternal(
1491 546 : true, eType == LOCK_RECURSIVE_MUTEX ? CPL_MUTEX_RECURSIVE
1492 : : CPL_MUTEX_ADAPTIVE);
1493 546 : if ((*phLock)->u.hMutex == nullptr)
1494 : {
1495 0 : free(*phLock);
1496 0 : *phLock = nullptr;
1497 : }
1498 : }
1499 546 : const bool bSuccess = *phLock != nullptr;
1500 546 : pthread_mutex_unlock(&global_mutex);
1501 546 : if (!bSuccess)
1502 0 : return false;
1503 : }
1504 : else
1505 : {
1506 4500 : pthread_mutex_unlock(&global_mutex);
1507 : }
1508 :
1509 5046 : return CPL_TO_BOOL(CPLAcquireMutex((*phLock)->u.hMutex, dfWaitInSeconds));
1510 : }
1511 :
1512 : /************************************************************************/
1513 : /* CPLGetThreadingModel() */
1514 : /************************************************************************/
1515 :
1516 29 : const char *CPLGetThreadingModel()
1517 :
1518 : {
1519 29 : return "pthread";
1520 : }
1521 :
1522 : /************************************************************************/
1523 : /* CPLCreateMutex() */
1524 : /************************************************************************/
1525 :
1526 : typedef struct _MutexLinkedElt MutexLinkedElt;
1527 :
1528 : struct _MutexLinkedElt
1529 : {
1530 : pthread_mutex_t sMutex;
1531 : int nOptions;
1532 : _MutexLinkedElt *psPrev;
1533 : _MutexLinkedElt *psNext;
1534 : };
1535 :
1536 : static MutexLinkedElt *psMutexList = nullptr;
1537 :
1538 110797 : static void CPLInitMutex(MutexLinkedElt *psItem)
1539 : {
1540 110797 : if (psItem->nOptions == CPL_MUTEX_REGULAR)
1541 : {
1542 : #ifdef HAVE_GCC_WARNING_ZERO_AS_NULL_POINTER_CONSTANT
1543 : #pragma GCC diagnostic push
1544 : #pragma GCC diagnostic ignored "-Wzero-as-null-pointer-constant"
1545 : #endif
1546 0 : pthread_mutex_t tmp_mutex = PTHREAD_MUTEX_INITIALIZER;
1547 : #ifdef HAVE_GCC_WARNING_ZERO_AS_NULL_POINTER_CONSTANT
1548 : #pragma GCC diagnostic pop
1549 : #endif
1550 0 : psItem->sMutex = tmp_mutex;
1551 0 : return;
1552 : }
1553 :
1554 : // When an adaptive mutex is required, we can safely fallback to regular
1555 : // mutex if we don't have HAVE_PTHREAD_MUTEX_ADAPTIVE_NP.
1556 110797 : if (psItem->nOptions == CPL_MUTEX_ADAPTIVE)
1557 : {
1558 : #if defined(HAVE_PTHREAD_MUTEX_ADAPTIVE_NP)
1559 : pthread_mutexattr_t attr;
1560 616 : pthread_mutexattr_init(&attr);
1561 616 : pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_ADAPTIVE_NP);
1562 616 : pthread_mutex_init(&(psItem->sMutex), &attr);
1563 : #else
1564 : pthread_mutex_t tmp_mutex = PTHREAD_MUTEX_INITIALIZER;
1565 : psItem->sMutex = tmp_mutex;
1566 : #endif
1567 616 : return;
1568 : }
1569 :
1570 : #if defined(PTHREAD_MUTEX_RECURSIVE) || defined(HAVE_PTHREAD_MUTEX_RECURSIVE)
1571 : {
1572 : pthread_mutexattr_t attr;
1573 110181 : pthread_mutexattr_init(&attr);
1574 110179 : pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);
1575 110179 : pthread_mutex_init(&(psItem->sMutex), &attr);
1576 : }
1577 : // BSDs have PTHREAD_MUTEX_RECURSIVE as an enum, not a define.
1578 : // But they have #define MUTEX_TYPE_COUNTING_FAST PTHREAD_MUTEX_RECURSIVE
1579 : #elif defined(MUTEX_TYPE_COUNTING_FAST)
1580 : {
1581 : pthread_mutexattr_t attr;
1582 : pthread_mutexattr_init(&attr);
1583 : pthread_mutexattr_settype(&attr, MUTEX_TYPE_COUNTING_FAST);
1584 : pthread_mutex_init(&(psItem->sMutex), &attr);
1585 : }
1586 : #elif defined(PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP)
1587 : pthread_mutex_t tmp_mutex = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;
1588 : psItem->sMutex = tmp_mutex;
1589 : #else
1590 : #error "Recursive mutexes apparently unsupported, configure --without-threads"
1591 : #endif
1592 : }
1593 :
1594 110779 : static CPLMutex *CPLCreateMutexInternal(bool bAlreadyInGlobalLock, int nOptions)
1595 : {
1596 : MutexLinkedElt *psItem =
1597 110779 : static_cast<MutexLinkedElt *>(malloc(sizeof(MutexLinkedElt)));
1598 110779 : if (psItem == nullptr)
1599 : {
1600 0 : fprintf(stderr, "CPLCreateMutexInternal() failed.\n");
1601 0 : return nullptr;
1602 : }
1603 :
1604 110779 : if (!bAlreadyInGlobalLock)
1605 77646 : pthread_mutex_lock(&global_mutex);
1606 110790 : psItem->psPrev = nullptr;
1607 110790 : psItem->psNext = psMutexList;
1608 110790 : if (psMutexList)
1609 109383 : psMutexList->psPrev = psItem;
1610 110790 : psMutexList = psItem;
1611 110790 : if (!bAlreadyInGlobalLock)
1612 77657 : pthread_mutex_unlock(&global_mutex);
1613 :
1614 110790 : psItem->nOptions = nOptions;
1615 110790 : CPLInitMutex(psItem);
1616 :
1617 110790 : return reinterpret_cast<CPLMutex *>(psItem);
1618 : }
1619 :
1620 77566 : CPLMutex *CPLCreateMutex()
1621 : {
1622 77566 : CPLMutex *mutex = CPLCreateMutexInternal(false, CPL_MUTEX_RECURSIVE);
1623 77580 : if (mutex)
1624 77578 : CPLAcquireMutex(mutex, 0);
1625 77580 : return mutex;
1626 : }
1627 :
1628 70 : CPLMutex *CPLCreateMutexEx(int nOptions)
1629 : {
1630 70 : CPLMutex *mutex = CPLCreateMutexInternal(false, nOptions);
1631 70 : if (mutex)
1632 70 : CPLAcquireMutex(mutex, 0);
1633 70 : return mutex;
1634 : }
1635 :
1636 : /************************************************************************/
1637 : /* CPLAcquireMutex() */
1638 : /************************************************************************/
1639 :
1640 94486600 : int CPLAcquireMutex(CPLMutex *hMutexIn, double /* dfWaitInSeconds */)
1641 : {
1642 : // TODO: Need to add timeout support.
1643 94486600 : MutexLinkedElt *psItem = reinterpret_cast<MutexLinkedElt *>(hMutexIn);
1644 94486600 : const int err = pthread_mutex_lock(&(psItem->sMutex));
1645 :
1646 94488000 : if (err != 0)
1647 : {
1648 0 : if (err == EDEADLK)
1649 0 : fprintf(stderr, "CPLAcquireMutex: Error = %d/EDEADLK\n", err);
1650 : else
1651 0 : fprintf(stderr, "CPLAcquireMutex: Error = %d (%s)\n", err,
1652 : strerror(err));
1653 :
1654 0 : return FALSE;
1655 : }
1656 :
1657 94488000 : return TRUE;
1658 : }
1659 :
1660 : /************************************************************************/
1661 : /* CPLReleaseMutex() */
1662 : /************************************************************************/
1663 :
1664 94487700 : void CPLReleaseMutex(CPLMutex *hMutexIn)
1665 :
1666 : {
1667 94487700 : MutexLinkedElt *psItem = reinterpret_cast<MutexLinkedElt *>(hMutexIn);
1668 94487700 : const int err = pthread_mutex_unlock(&(psItem->sMutex));
1669 94486200 : if (err != 0)
1670 : {
1671 0 : fprintf(stderr, "CPLReleaseMutex: Error = %d (%s)\n", err,
1672 : strerror(err));
1673 : }
1674 94486200 : }
1675 :
1676 : /************************************************************************/
1677 : /* CPLDestroyMutex() */
1678 : /************************************************************************/
1679 :
1680 105107 : void CPLDestroyMutex(CPLMutex *hMutexIn)
1681 :
1682 : {
1683 105107 : MutexLinkedElt *psItem = reinterpret_cast<MutexLinkedElt *>(hMutexIn);
1684 105107 : const int err = pthread_mutex_destroy(&(psItem->sMutex));
1685 105105 : if (err != 0)
1686 : {
1687 0 : fprintf(stderr, "CPLDestroyMutex: Error = %d (%s)\n", err,
1688 : strerror(err));
1689 : }
1690 105105 : pthread_mutex_lock(&global_mutex);
1691 105107 : if (psItem->psPrev)
1692 69023 : psItem->psPrev->psNext = psItem->psNext;
1693 105107 : if (psItem->psNext)
1694 101324 : psItem->psNext->psPrev = psItem->psPrev;
1695 105107 : if (psItem == psMutexList)
1696 36084 : psMutexList = psItem->psNext;
1697 105107 : pthread_mutex_unlock(&global_mutex);
1698 105107 : free(hMutexIn);
1699 105107 : }
1700 :
1701 : /************************************************************************/
1702 : /* CPLReinitAllMutex() */
1703 : /************************************************************************/
1704 :
1705 : // Used by gdalclientserver.cpp just after forking, to avoid
1706 : // deadlocks while mixing threads with fork.
1707 : void CPLReinitAllMutex(); // TODO(schwehr): Put this in a header.
1708 :
1709 0 : void CPLReinitAllMutex()
1710 : {
1711 0 : MutexLinkedElt *psItem = psMutexList;
1712 0 : while (psItem != nullptr)
1713 : {
1714 0 : CPLInitMutex(psItem);
1715 0 : psItem = psItem->psNext;
1716 : }
1717 : #ifdef HAVE_GCC_WARNING_ZERO_AS_NULL_POINTER_CONSTANT
1718 : #pragma GCC diagnostic push
1719 : #pragma GCC diagnostic ignored "-Wzero-as-null-pointer-constant"
1720 : #endif
1721 0 : pthread_mutex_t tmp_global_mutex = PTHREAD_MUTEX_INITIALIZER;
1722 : #ifdef HAVE_GCC_WARNING_ZERO_AS_NULL_POINTER_CONSTANT
1723 : #pragma GCC diagnostic pop
1724 : #endif
1725 0 : global_mutex = tmp_global_mutex;
1726 0 : }
1727 :
1728 : /************************************************************************/
1729 : /* CPLCreateCond() */
1730 : /************************************************************************/
1731 :
1732 34000 : CPLCond *CPLCreateCond()
1733 : {
1734 : pthread_cond_t *pCond =
1735 34000 : static_cast<pthread_cond_t *>(malloc(sizeof(pthread_cond_t)));
1736 34000 : if (pCond && pthread_cond_init(pCond, nullptr) == 0)
1737 33984 : return reinterpret_cast<CPLCond *>(pCond);
1738 10 : fprintf(stderr, "CPLCreateCond() failed.\n");
1739 0 : free(pCond);
1740 0 : return nullptr;
1741 : }
1742 :
1743 : /************************************************************************/
1744 : /* CPLCondWait() */
1745 : /************************************************************************/
1746 :
1747 999 : void CPLCondWait(CPLCond *hCond, CPLMutex *hMutex)
1748 : {
1749 999 : pthread_cond_t *pCond = reinterpret_cast<pthread_cond_t *>(hCond);
1750 999 : MutexLinkedElt *psItem = reinterpret_cast<MutexLinkedElt *>(hMutex);
1751 999 : pthread_mutex_t *pMutex = &(psItem->sMutex);
1752 999 : pthread_cond_wait(pCond, pMutex);
1753 999 : }
1754 :
1755 : /************************************************************************/
1756 : /* CPLCondTimedWait() */
1757 : /************************************************************************/
1758 :
1759 0 : CPLCondTimedWaitReason CPLCondTimedWait(CPLCond *hCond, CPLMutex *hMutex,
1760 : double dfWaitInSeconds)
1761 : {
1762 0 : pthread_cond_t *pCond = reinterpret_cast<pthread_cond_t *>(hCond);
1763 0 : MutexLinkedElt *psItem = reinterpret_cast<MutexLinkedElt *>(hMutex);
1764 0 : pthread_mutex_t *pMutex = &(psItem->sMutex);
1765 : struct timeval tv;
1766 : struct timespec ts;
1767 :
1768 0 : gettimeofday(&tv, nullptr);
1769 0 : ts.tv_sec = time(nullptr) + static_cast<int>(dfWaitInSeconds);
1770 0 : ts.tv_nsec =
1771 0 : static_cast<int>(tv.tv_usec) * 1000 +
1772 0 : static_cast<int>(1000 * 1000 * 1000 * fmod(dfWaitInSeconds, 1));
1773 0 : ts.tv_sec += ts.tv_nsec / (1000 * 1000 * 1000);
1774 0 : ts.tv_nsec %= (1000 * 1000 * 1000);
1775 0 : int ret = pthread_cond_timedwait(pCond, pMutex, &ts);
1776 0 : if (ret == 0)
1777 0 : return COND_TIMED_WAIT_COND;
1778 0 : else if (ret == ETIMEDOUT)
1779 0 : return COND_TIMED_WAIT_TIME_OUT;
1780 : else
1781 0 : return COND_TIMED_WAIT_OTHER;
1782 : }
1783 :
1784 : /************************************************************************/
1785 : /* CPLCondSignal() */
1786 : /************************************************************************/
1787 :
1788 32617 : void CPLCondSignal(CPLCond *hCond)
1789 : {
1790 32617 : pthread_cond_t *pCond = reinterpret_cast<pthread_cond_t *>(hCond);
1791 32617 : pthread_cond_signal(pCond);
1792 32617 : }
1793 :
1794 : /************************************************************************/
1795 : /* CPLCondBroadcast() */
1796 : /************************************************************************/
1797 :
1798 120 : void CPLCondBroadcast(CPLCond *hCond)
1799 : {
1800 120 : pthread_cond_t *pCond = reinterpret_cast<pthread_cond_t *>(hCond);
1801 120 : pthread_cond_broadcast(pCond);
1802 120 : }
1803 :
1804 : /************************************************************************/
1805 : /* CPLDestroyCond() */
1806 : /************************************************************************/
1807 :
1808 34005 : void CPLDestroyCond(CPLCond *hCond)
1809 : {
1810 34005 : pthread_cond_t *pCond = reinterpret_cast<pthread_cond_t *>(hCond);
1811 34005 : pthread_cond_destroy(pCond);
1812 34005 : free(hCond);
1813 34005 : }
1814 :
1815 : /************************************************************************/
1816 : /* CPLLockFile() */
1817 : /* */
1818 : /* This is really a stub implementation, see first */
1819 : /* CPLLockFile() for caveats. */
1820 : /************************************************************************/
1821 :
1822 3 : void *CPLLockFile(const char *pszPath, double dfWaitInSeconds)
1823 :
1824 : {
1825 : /* -------------------------------------------------------------------- */
1826 : /* We use a lock file with a name derived from the file we want */
1827 : /* to lock to represent the file being locked. Note that for */
1828 : /* the stub implementation the target file does not even need */
1829 : /* to exist to be locked. */
1830 : /* -------------------------------------------------------------------- */
1831 3 : const size_t nLen = strlen(pszPath) + 30;
1832 3 : char *pszLockFilename = static_cast<char *>(CPLMalloc(nLen));
1833 3 : snprintf(pszLockFilename, nLen, "%s.lock", pszPath);
1834 :
1835 3 : FILE *fpLock = fopen(pszLockFilename, "r");
1836 3 : while (fpLock != nullptr && dfWaitInSeconds > 0.0)
1837 : {
1838 0 : fclose(fpLock);
1839 0 : CPLSleep(std::min(dfWaitInSeconds, 0.5));
1840 0 : dfWaitInSeconds -= 0.5;
1841 :
1842 0 : fpLock = fopen(pszLockFilename, "r");
1843 : }
1844 :
1845 3 : if (fpLock != nullptr)
1846 : {
1847 0 : fclose(fpLock);
1848 0 : CPLFree(pszLockFilename);
1849 0 : return nullptr;
1850 : }
1851 :
1852 3 : fpLock = fopen(pszLockFilename, "w");
1853 :
1854 3 : if (fpLock == nullptr)
1855 : {
1856 0 : CPLFree(pszLockFilename);
1857 0 : return nullptr;
1858 : }
1859 :
1860 3 : fwrite("held\n", 1, 5, fpLock);
1861 3 : fclose(fpLock);
1862 :
1863 3 : return pszLockFilename;
1864 : }
1865 :
1866 : /************************************************************************/
1867 : /* CPLUnlockFile() */
1868 : /************************************************************************/
1869 :
1870 3 : void CPLUnlockFile(void *hLock)
1871 :
1872 : {
1873 3 : char *pszLockFilename = static_cast<char *>(hLock);
1874 :
1875 3 : if (hLock == nullptr)
1876 0 : return;
1877 :
1878 3 : VSIUnlink(pszLockFilename);
1879 :
1880 3 : CPLFree(pszLockFilename);
1881 : }
1882 :
1883 : /************************************************************************/
1884 : /* CPLGetPID() */
1885 : /************************************************************************/
1886 :
1887 20217900 : GIntBig CPLGetPID()
1888 :
1889 : {
1890 20217900 : return reinterpret_cast<GIntBig>(reinterpret_cast<void *>(pthread_self()));
1891 : }
1892 :
1893 : static pthread_key_t oTLSKey;
1894 : static pthread_once_t oTLSKeySetup = PTHREAD_ONCE_INIT;
1895 :
1896 : /************************************************************************/
1897 : /* CPLMake_key() */
1898 : /************************************************************************/
1899 :
1900 1414 : static void CPLMake_key()
1901 :
1902 : {
1903 1414 : if (pthread_key_create(&oTLSKey, reinterpret_cast<void (*)(void *)>(
1904 1414 : CPLCleanupTLSList)) != 0)
1905 : {
1906 0 : CPLError(CE_Fatal, CPLE_AppDefined, "pthread_key_create() failed!");
1907 : }
1908 1414 : }
1909 :
1910 : /************************************************************************/
1911 : /* CPLGetTLSList() */
1912 : /************************************************************************/
1913 :
1914 60126100 : static void **CPLGetTLSList(int *pbMemoryErrorOccurred)
1915 :
1916 : {
1917 60126100 : if (pbMemoryErrorOccurred)
1918 57177900 : *pbMemoryErrorOccurred = FALSE;
1919 :
1920 60126100 : if (pthread_once(&oTLSKeySetup, CPLMake_key) != 0)
1921 : {
1922 0 : if (pbMemoryErrorOccurred)
1923 : {
1924 0 : fprintf(stderr, "CPLGetTLSList(): pthread_once() failed!\n");
1925 0 : *pbMemoryErrorOccurred = TRUE;
1926 0 : return nullptr;
1927 : }
1928 0 : CPLEmergencyError("CPLGetTLSList(): pthread_once() failed!");
1929 : }
1930 :
1931 60105800 : void **papTLSList = static_cast<void **>(pthread_getspecific(oTLSKey));
1932 60111800 : if (papTLSList == nullptr)
1933 : {
1934 : papTLSList =
1935 16781 : static_cast<void **>(VSICalloc(sizeof(void *), CTLS_MAX * 2));
1936 3588 : if (papTLSList == nullptr)
1937 : {
1938 0 : if (pbMemoryErrorOccurred)
1939 : {
1940 0 : fprintf(stderr,
1941 : "CPLGetTLSList() failed to allocate TLS list!\n");
1942 0 : *pbMemoryErrorOccurred = TRUE;
1943 0 : return nullptr;
1944 : }
1945 0 : CPLEmergencyError("CPLGetTLSList() failed to allocate TLS list!");
1946 : }
1947 3588 : if (pthread_setspecific(oTLSKey, papTLSList) != 0)
1948 : {
1949 0 : if (pbMemoryErrorOccurred)
1950 : {
1951 0 : fprintf(stderr,
1952 : "CPLGetTLSList(): pthread_setspecific() failed!\n");
1953 0 : *pbMemoryErrorOccurred = TRUE;
1954 0 : return nullptr;
1955 : }
1956 0 : CPLEmergencyError("CPLGetTLSList(): pthread_setspecific() failed!");
1957 : }
1958 : }
1959 :
1960 60098700 : return papTLSList;
1961 : }
1962 :
1963 : /************************************************************************/
1964 : /* CPLStdCallThreadJacket() */
1965 : /************************************************************************/
1966 :
1967 : typedef struct
1968 : {
1969 : void *pAppData;
1970 : CPLThreadFunc pfnMain;
1971 : pthread_t hThread;
1972 : bool bJoinable;
1973 : #ifdef CHECK_THREAD_CAN_ALLOCATE_TLS
1974 : bool bInitSucceeded;
1975 : bool bInitDone;
1976 : pthread_mutex_t sMutex;
1977 : pthread_cond_t sCond;
1978 : #endif
1979 : } CPLStdCallThreadInfo;
1980 :
1981 4829 : static void *CPLStdCallThreadJacket(void *pData)
1982 :
1983 : {
1984 4829 : CPLStdCallThreadInfo *psInfo = static_cast<CPLStdCallThreadInfo *>(pData);
1985 :
1986 : #ifdef CHECK_THREAD_CAN_ALLOCATE_TLS
1987 : int bMemoryError = FALSE;
1988 : CPLGetTLSList(&bMemoryError);
1989 : if (bMemoryError)
1990 : goto error;
1991 :
1992 : assert(pthread_mutex_lock(&(psInfo->sMutex)) == 0);
1993 : psInfo->bInitDone = true;
1994 : assert(pthread_cond_signal(&(psInfo->sCond)) == 0);
1995 : assert(pthread_mutex_unlock(&(psInfo->sMutex)) == 0);
1996 : #endif
1997 :
1998 4829 : psInfo->pfnMain(psInfo->pAppData);
1999 :
2000 3790 : if (!psInfo->bJoinable)
2001 1 : CPLFree(psInfo);
2002 :
2003 3789 : return nullptr;
2004 :
2005 : #ifdef CHECK_THREAD_CAN_ALLOCATE_TLS
2006 : error:
2007 : assert(pthread_mutex_lock(&(psInfo->sMutex)) == 0);
2008 : psInfo->bInitSucceeded = false;
2009 : psInfo->bInitDone = true;
2010 : assert(pthread_cond_signal(&(psInfo->sCond)) == 0);
2011 : assert(pthread_mutex_unlock(&(psInfo->sMutex)) == 0);
2012 : return nullptr;
2013 : #endif
2014 : }
2015 :
2016 : /************************************************************************/
2017 : /* CPLCreateThread() */
2018 : /* */
2019 : /* The WIN32 CreateThread() call requires an entry point that */
2020 : /* has __stdcall conventions, so we provide a jacket function */
2021 : /* to supply that. */
2022 : /************************************************************************/
2023 :
2024 1 : int CPLCreateThread(CPLThreadFunc pfnMain, void *pThreadArg)
2025 :
2026 : {
2027 : CPLStdCallThreadInfo *psInfo = static_cast<CPLStdCallThreadInfo *>(
2028 1 : VSI_CALLOC_VERBOSE(sizeof(CPLStdCallThreadInfo), 1));
2029 1 : if (psInfo == nullptr)
2030 0 : return -1;
2031 1 : psInfo->pAppData = pThreadArg;
2032 1 : psInfo->pfnMain = pfnMain;
2033 1 : psInfo->bJoinable = false;
2034 : #ifdef CHECK_THREAD_CAN_ALLOCATE_TLS
2035 : psInfo->bInitSucceeded = true;
2036 : psInfo->bInitDone = false;
2037 : pthread_mutex_t sMutex = PTHREAD_MUTEX_INITIALIZER;
2038 : psInfo->sMutex = sMutex;
2039 : if (pthread_cond_init(&(psInfo->sCond), nullptr) != 0)
2040 : {
2041 : CPLFree(psInfo);
2042 : fprintf(stderr, "CPLCreateThread() failed.\n");
2043 : return -1;
2044 : }
2045 : #endif
2046 :
2047 : pthread_attr_t hThreadAttr;
2048 1 : pthread_attr_init(&hThreadAttr);
2049 1 : pthread_attr_setdetachstate(&hThreadAttr, PTHREAD_CREATE_DETACHED);
2050 1 : if (pthread_create(&(psInfo->hThread), &hThreadAttr, CPLStdCallThreadJacket,
2051 1 : static_cast<void *>(psInfo)) != 0)
2052 : {
2053 : #ifdef CHECK_THREAD_CAN_ALLOCATE_TLS
2054 : pthread_cond_destroy(&(psInfo->sCond));
2055 : #endif
2056 0 : CPLFree(psInfo);
2057 0 : fprintf(stderr, "CPLCreateThread() failed.\n");
2058 0 : return -1;
2059 : }
2060 :
2061 : #ifdef CHECK_THREAD_CAN_ALLOCATE_TLS
2062 : bool bInitSucceeded;
2063 : while (true)
2064 : {
2065 : assert(pthread_mutex_lock(&(psInfo->sMutex)) == 0);
2066 : bool bInitDone = psInfo->bInitDone;
2067 : if (!bInitDone)
2068 : assert(pthread_cond_wait(&(psInfo->sCond), &(psInfo->sMutex)) == 0);
2069 : bInitSucceeded = psInfo->bInitSucceeded;
2070 : assert(pthread_mutex_unlock(&(psInfo->sMutex)) == 0);
2071 : if (bInitDone)
2072 : break;
2073 : }
2074 :
2075 : pthread_cond_destroy(&(psInfo->sCond));
2076 :
2077 : if (!bInitSucceeded)
2078 : {
2079 : CPLFree(psInfo);
2080 : fprintf(stderr, "CPLCreateThread() failed.\n");
2081 : return -1;
2082 : }
2083 : #endif
2084 :
2085 1 : return 1; // Can we return the actual thread pid?
2086 : }
2087 :
2088 : /************************************************************************/
2089 : /* CPLCreateJoinableThread() */
2090 : /************************************************************************/
2091 :
2092 4828 : CPLJoinableThread *CPLCreateJoinableThread(CPLThreadFunc pfnMain,
2093 : void *pThreadArg)
2094 :
2095 : {
2096 : CPLStdCallThreadInfo *psInfo = static_cast<CPLStdCallThreadInfo *>(
2097 4828 : VSI_CALLOC_VERBOSE(sizeof(CPLStdCallThreadInfo), 1));
2098 4828 : if (psInfo == nullptr)
2099 0 : return nullptr;
2100 4828 : psInfo->pAppData = pThreadArg;
2101 4828 : psInfo->pfnMain = pfnMain;
2102 4828 : psInfo->bJoinable = true;
2103 : #ifdef CHECK_THREAD_CAN_ALLOCATE_TLS
2104 : psInfo->bInitSucceeded = true;
2105 : psInfo->bInitDone = false;
2106 : pthread_mutex_t sMutex = PTHREAD_MUTEX_INITIALIZER;
2107 : psInfo->sMutex = sMutex;
2108 : {
2109 : int err = pthread_cond_init(&(psInfo->sCond), nullptr);
2110 : if (err != 0)
2111 : {
2112 : CPLFree(psInfo);
2113 : fprintf(stderr, "CPLCreateJoinableThread() failed: %s.\n",
2114 : strerror(err));
2115 : return nullptr;
2116 : }
2117 : }
2118 : #endif
2119 :
2120 : pthread_attr_t hThreadAttr;
2121 4828 : pthread_attr_init(&hThreadAttr);
2122 4828 : pthread_attr_setdetachstate(&hThreadAttr, PTHREAD_CREATE_JOINABLE);
2123 : int err =
2124 4828 : pthread_create(&(psInfo->hThread), &hThreadAttr, CPLStdCallThreadJacket,
2125 : static_cast<void *>(psInfo));
2126 4828 : if (err != 0)
2127 : {
2128 : #ifdef CHECK_THREAD_CAN_ALLOCATE_TLS
2129 : pthread_cond_destroy(&(psInfo->sCond));
2130 : #endif
2131 0 : CPLFree(psInfo);
2132 0 : fprintf(stderr, "CPLCreateJoinableThread() failed: %s.\n",
2133 : strerror(err));
2134 0 : return nullptr;
2135 : }
2136 :
2137 : #ifdef CHECK_THREAD_CAN_ALLOCATE_TLS
2138 : bool bInitSucceeded;
2139 : while (true)
2140 : {
2141 : assert(pthread_mutex_lock(&(psInfo->sMutex)) == 0);
2142 : bool bInitDone = psInfo->bInitDone;
2143 : if (!bInitDone)
2144 : assert(pthread_cond_wait(&(psInfo->sCond), &(psInfo->sMutex)) == 0);
2145 : bInitSucceeded = psInfo->bInitSucceeded;
2146 : assert(pthread_mutex_unlock(&(psInfo->sMutex)) == 0);
2147 : if (bInitDone)
2148 : break;
2149 : }
2150 :
2151 : pthread_cond_destroy(&(psInfo->sCond));
2152 :
2153 : if (!bInitSucceeded)
2154 : {
2155 : void *status;
2156 : pthread_join(psInfo->hThread, &status);
2157 : CPLFree(psInfo);
2158 : fprintf(stderr, "CPLCreateJoinableThread() failed.\n");
2159 : return nullptr;
2160 : }
2161 : #endif
2162 :
2163 4828 : return reinterpret_cast<CPLJoinableThread *>(psInfo);
2164 : }
2165 :
2166 : /************************************************************************/
2167 : /* CPLJoinThread() */
2168 : /************************************************************************/
2169 :
2170 3789 : void CPLJoinThread(CPLJoinableThread *hJoinableThread)
2171 : {
2172 3789 : CPLStdCallThreadInfo *psInfo =
2173 : reinterpret_cast<CPLStdCallThreadInfo *>(hJoinableThread);
2174 3789 : if (psInfo == nullptr)
2175 0 : return;
2176 :
2177 : void *status;
2178 3789 : pthread_join(psInfo->hThread, &status);
2179 :
2180 3789 : CPLFree(psInfo);
2181 : }
2182 :
2183 : /************************************************************************/
2184 : /* CPLSleep() */
2185 : /************************************************************************/
2186 :
2187 68 : void CPLSleep(double dfWaitInSeconds)
2188 :
2189 : {
2190 : struct timespec sRequest;
2191 : struct timespec sRemain;
2192 :
2193 68 : sRequest.tv_sec = static_cast<int>(floor(dfWaitInSeconds));
2194 68 : sRequest.tv_nsec =
2195 68 : static_cast<int>((dfWaitInSeconds - sRequest.tv_sec) * 1000000000);
2196 68 : nanosleep(&sRequest, &sRemain);
2197 68 : }
2198 :
2199 : /************************************************************************/
2200 : /* CPLFinalizeTLS() */
2201 : /************************************************************************/
2202 :
2203 440 : void CPLFinalizeTLS()
2204 : {
2205 440 : CPLCleanupTLS();
2206 : // See #5509 for the explanation why this may be needed.
2207 440 : pthread_key_delete(oTLSKey);
2208 440 : }
2209 :
2210 : /************************************************************************/
2211 : /* CPLCleanupTLS() */
2212 : /************************************************************************/
2213 :
2214 1382 : void CPLCleanupTLS()
2215 :
2216 : {
2217 1382 : void **papTLSList = static_cast<void **>(pthread_getspecific(oTLSKey));
2218 1382 : if (papTLSList == nullptr)
2219 1 : return;
2220 :
2221 1381 : pthread_setspecific(oTLSKey, nullptr);
2222 :
2223 1381 : CPLCleanupTLSList(papTLSList);
2224 : }
2225 :
2226 : /************************************************************************/
2227 : /* CPLCreateSpinLock() */
2228 : /************************************************************************/
2229 :
2230 : #if defined(HAVE_PTHREAD_SPIN_LOCK)
2231 : #define HAVE_SPINLOCK_IMPL
2232 :
2233 : struct _CPLSpinLock
2234 : {
2235 : pthread_spinlock_t spin;
2236 : };
2237 :
2238 : CPLSpinLock *CPLCreateSpinLock()
2239 : {
2240 : CPLSpinLock *psSpin =
2241 : static_cast<CPLSpinLock *>(malloc(sizeof(CPLSpinLock)));
2242 : if (psSpin != nullptr &&
2243 : pthread_spin_init(&(psSpin->spin), PTHREAD_PROCESS_PRIVATE) == 0)
2244 : {
2245 : return psSpin;
2246 : }
2247 : else
2248 : {
2249 : fprintf(stderr, "CPLCreateSpinLock() failed.\n");
2250 : free(psSpin);
2251 : return nullptr;
2252 : }
2253 : }
2254 :
2255 : /************************************************************************/
2256 : /* CPLAcquireSpinLock() */
2257 : /************************************************************************/
2258 :
2259 : int CPLAcquireSpinLock(CPLSpinLock *psSpin)
2260 : {
2261 : return pthread_spin_lock(&(psSpin->spin)) == 0;
2262 : }
2263 :
2264 : /************************************************************************/
2265 : /* CPLCreateOrAcquireSpinLockInternal() */
2266 : /************************************************************************/
2267 :
2268 : int CPLCreateOrAcquireSpinLockInternal(CPLLock **ppsLock)
2269 : {
2270 : pthread_mutex_lock(&global_mutex);
2271 : if (*ppsLock == nullptr)
2272 : {
2273 : *ppsLock = static_cast<CPLLock *>(calloc(1, sizeof(CPLLock)));
2274 : if (*ppsLock != nullptr)
2275 : {
2276 : (*ppsLock)->eType = LOCK_SPIN;
2277 : (*ppsLock)->u.hSpinLock = CPLCreateSpinLock();
2278 : if ((*ppsLock)->u.hSpinLock == nullptr)
2279 : {
2280 : free(*ppsLock);
2281 : *ppsLock = nullptr;
2282 : }
2283 : }
2284 : }
2285 : pthread_mutex_unlock(&global_mutex);
2286 : // coverity[missing_unlock]
2287 : return (*ppsLock != nullptr && CPLAcquireSpinLock((*ppsLock)->u.hSpinLock));
2288 : }
2289 :
2290 : /************************************************************************/
2291 : /* CPLReleaseSpinLock() */
2292 : /************************************************************************/
2293 :
2294 : void CPLReleaseSpinLock(CPLSpinLock *psSpin)
2295 : {
2296 : pthread_spin_unlock(&(psSpin->spin));
2297 : }
2298 :
2299 : /************************************************************************/
2300 : /* CPLDestroySpinLock() */
2301 : /************************************************************************/
2302 :
2303 : void CPLDestroySpinLock(CPLSpinLock *psSpin)
2304 : {
2305 : pthread_spin_destroy(&(psSpin->spin));
2306 : free(psSpin);
2307 : }
2308 : #endif // HAVE_PTHREAD_SPIN_LOCK
2309 :
2310 : #endif // def CPL_MULTIPROC_PTHREAD
2311 :
2312 : /************************************************************************/
2313 : /* CPLGetTLS() */
2314 : /************************************************************************/
2315 :
2316 1389510 : void *CPLGetTLS(int nIndex)
2317 :
2318 : {
2319 1389510 : void **l_papTLSList = CPLGetTLSList(nullptr);
2320 :
2321 1389450 : CPLAssert(nIndex >= 0 && nIndex < CTLS_MAX);
2322 :
2323 1389440 : return l_papTLSList[nIndex];
2324 : }
2325 :
2326 : /************************************************************************/
2327 : /* CPLGetTLSEx() */
2328 : /************************************************************************/
2329 :
2330 57211300 : void *CPLGetTLSEx(int nIndex, int *pbMemoryErrorOccurred)
2331 :
2332 : {
2333 57211300 : void **l_papTLSList = CPLGetTLSList(pbMemoryErrorOccurred);
2334 57173800 : if (l_papTLSList == nullptr)
2335 0 : return nullptr;
2336 :
2337 57173800 : CPLAssert(nIndex >= 0 && nIndex < CTLS_MAX);
2338 :
2339 57172800 : return l_papTLSList[nIndex];
2340 : }
2341 :
2342 : /************************************************************************/
2343 : /* CPLSetTLS() */
2344 : /************************************************************************/
2345 :
2346 17458 : void CPLSetTLS(int nIndex, void *pData, int bFreeOnExit)
2347 :
2348 : {
2349 17458 : CPLSetTLSWithFreeFunc(nIndex, pData, (bFreeOnExit) ? CPLFree : nullptr);
2350 17447 : }
2351 :
2352 : /************************************************************************/
2353 : /* CPLSetTLSWithFreeFunc() */
2354 : /************************************************************************/
2355 :
2356 : // Warning: The CPLTLSFreeFunc must not in any case directly or indirectly
2357 : // use or fetch any TLS data, or a terminating thread will hang!
2358 1547650 : void CPLSetTLSWithFreeFunc(int nIndex, void *pData, CPLTLSFreeFunc pfnFree)
2359 :
2360 : {
2361 1547650 : void **l_papTLSList = CPLGetTLSList(nullptr);
2362 :
2363 1538910 : CPLAssert(nIndex >= 0 && nIndex < CTLS_MAX);
2364 :
2365 1536880 : l_papTLSList[nIndex] = pData;
2366 1536880 : l_papTLSList[CTLS_MAX + nIndex] = reinterpret_cast<void *>(pfnFree);
2367 1536880 : }
2368 :
2369 : /************************************************************************/
2370 : /* CPLSetTLSWithFreeFuncEx() */
2371 : /************************************************************************/
2372 :
2373 : // Warning: the CPLTLSFreeFunc must not in any case directly or indirectly
2374 : // use or fetch any TLS data, or a terminating thread will hang!
2375 1372 : void CPLSetTLSWithFreeFuncEx(int nIndex, void *pData, CPLTLSFreeFunc pfnFree,
2376 : int *pbMemoryErrorOccurred)
2377 :
2378 : {
2379 1372 : void **l_papTLSList = CPLGetTLSList(pbMemoryErrorOccurred);
2380 :
2381 1372 : CPLAssert(nIndex >= 0 && nIndex < CTLS_MAX);
2382 :
2383 1372 : l_papTLSList[nIndex] = pData;
2384 1372 : l_papTLSList[CTLS_MAX + nIndex] = reinterpret_cast<void *>(pfnFree);
2385 1372 : }
2386 : #ifndef HAVE_SPINLOCK_IMPL
2387 :
2388 : // No spinlock specific API? Fallback to mutex.
2389 :
2390 : /************************************************************************/
2391 : /* CPLCreateSpinLock() */
2392 : /************************************************************************/
2393 :
2394 33762 : CPLSpinLock *CPLCreateSpinLock(void)
2395 : {
2396 33762 : CPLSpinLock *psSpin = reinterpret_cast<CPLSpinLock *>(CPLCreateMutex());
2397 33758 : if (psSpin)
2398 33752 : CPLReleaseSpinLock(psSpin);
2399 33759 : return psSpin;
2400 : }
2401 :
2402 : /************************************************************************/
2403 : /* CPLCreateOrAcquireSpinLock() */
2404 : /************************************************************************/
2405 :
2406 4 : int CPLCreateOrAcquireSpinLockInternal(CPLLock **ppsLock)
2407 : {
2408 4 : return CPLCreateOrAcquireMutexInternal(ppsLock, 1000, LOCK_ADAPTIVE_MUTEX);
2409 : }
2410 :
2411 : /************************************************************************/
2412 : /* CPLAcquireSpinLock() */
2413 : /************************************************************************/
2414 :
2415 6547240 : int CPLAcquireSpinLock(CPLSpinLock *psSpin)
2416 : {
2417 6547240 : return CPLAcquireMutex(reinterpret_cast<CPLMutex *>(psSpin), 1000);
2418 : }
2419 :
2420 : /************************************************************************/
2421 : /* CPLReleaseSpinLock() */
2422 : /************************************************************************/
2423 :
2424 6581120 : void CPLReleaseSpinLock(CPLSpinLock *psSpin)
2425 : {
2426 6581120 : CPLReleaseMutex(reinterpret_cast<CPLMutex *>(psSpin));
2427 6581210 : }
2428 :
2429 : /************************************************************************/
2430 : /* CPLDestroySpinLock() */
2431 : /************************************************************************/
2432 :
2433 33763 : void CPLDestroySpinLock(CPLSpinLock *psSpin)
2434 : {
2435 33763 : CPLDestroyMutex(reinterpret_cast<CPLMutex *>(psSpin));
2436 33765 : }
2437 :
2438 : #endif // HAVE_SPINLOCK_IMPL
2439 :
2440 : /************************************************************************/
2441 : /* CPLCreateLock() */
2442 : /************************************************************************/
2443 :
2444 33833 : CPLLock *CPLCreateLock(CPLLockType eType)
2445 : {
2446 33833 : switch (eType)
2447 : {
2448 70 : case LOCK_RECURSIVE_MUTEX:
2449 : case LOCK_ADAPTIVE_MUTEX:
2450 : {
2451 140 : CPLMutex *hMutex = CPLCreateMutexEx(eType == LOCK_RECURSIVE_MUTEX
2452 70 : ? CPL_MUTEX_RECURSIVE
2453 : : CPL_MUTEX_ADAPTIVE);
2454 70 : if (!hMutex)
2455 0 : return nullptr;
2456 70 : CPLReleaseMutex(hMutex);
2457 70 : CPLLock *psLock = static_cast<CPLLock *>(malloc(sizeof(CPLLock)));
2458 70 : if (psLock == nullptr)
2459 : {
2460 0 : fprintf(stderr, "CPLCreateLock() failed.\n");
2461 0 : CPLDestroyMutex(hMutex);
2462 0 : return nullptr;
2463 : }
2464 70 : psLock->eType = eType;
2465 70 : psLock->u.hMutex = hMutex;
2466 : #ifdef DEBUG_CONTENTION
2467 70 : psLock->bDebugPerf = false;
2468 70 : psLock->bDebugPerfAsked = false;
2469 70 : psLock->nCurrentHolders = 0;
2470 70 : psLock->nStartTime = 0;
2471 : #endif
2472 70 : return psLock;
2473 : }
2474 33761 : case LOCK_SPIN:
2475 : {
2476 33761 : CPLSpinLock *hSpinLock = CPLCreateSpinLock();
2477 33760 : if (!hSpinLock)
2478 0 : return nullptr;
2479 33760 : CPLLock *psLock = static_cast<CPLLock *>(malloc(sizeof(CPLLock)));
2480 33760 : if (psLock == nullptr)
2481 : {
2482 0 : fprintf(stderr, "CPLCreateLock() failed.\n");
2483 0 : CPLDestroySpinLock(hSpinLock);
2484 0 : return nullptr;
2485 : }
2486 33760 : psLock->eType = eType;
2487 33760 : psLock->u.hSpinLock = hSpinLock;
2488 : #ifdef DEBUG_CONTENTION
2489 33760 : psLock->bDebugPerf = false;
2490 33760 : psLock->bDebugPerfAsked = false;
2491 33760 : psLock->nCurrentHolders = 0;
2492 33760 : psLock->nStartTime = 0;
2493 : #endif
2494 33760 : return psLock;
2495 : }
2496 2 : default:
2497 2 : CPLAssert(false);
2498 : return nullptr;
2499 : }
2500 : }
2501 :
2502 : /************************************************************************/
2503 : /* CPLCreateOrAcquireLock() */
2504 : /************************************************************************/
2505 :
2506 5046 : int CPLCreateOrAcquireLock(CPLLock **ppsLock, CPLLockType eType)
2507 : {
2508 : #ifdef DEBUG_CONTENTION
2509 5046 : GUIntBig nStartTime = 0;
2510 5046 : if ((*ppsLock) && (*ppsLock)->bDebugPerfAsked)
2511 4 : nStartTime = CPLrdtsc();
2512 : #endif
2513 5046 : int ret = 0;
2514 :
2515 5046 : switch (eType)
2516 : {
2517 5042 : case LOCK_RECURSIVE_MUTEX:
2518 : case LOCK_ADAPTIVE_MUTEX:
2519 : {
2520 5042 : ret = CPLCreateOrAcquireMutexInternal(ppsLock, 1000, eType);
2521 5042 : break;
2522 : }
2523 4 : case LOCK_SPIN:
2524 : {
2525 4 : ret = CPLCreateOrAcquireSpinLockInternal(ppsLock);
2526 4 : break;
2527 : }
2528 0 : default:
2529 0 : CPLAssert(false);
2530 : return FALSE;
2531 : }
2532 : #ifdef DEBUG_CONTENTION
2533 5050 : if (ret && (*ppsLock)->bDebugPerfAsked &&
2534 4 : CPLAtomicInc(&((*ppsLock)->nCurrentHolders)) == 1)
2535 : {
2536 4 : (*ppsLock)->bDebugPerf = true;
2537 4 : (*ppsLock)->nStartTime = nStartTime;
2538 : }
2539 : #endif
2540 5046 : return ret;
2541 : }
2542 :
2543 : /************************************************************************/
2544 : /* CPLAcquireLock() */
2545 : /************************************************************************/
2546 :
2547 24130700 : int CPLAcquireLock(CPLLock *psLock)
2548 : {
2549 : #ifdef DEBUG_CONTENTION
2550 24130700 : GUIntBig nStartTime = 0;
2551 24130700 : if (psLock->bDebugPerfAsked)
2552 1952510 : nStartTime = CPLrdtsc();
2553 : #endif
2554 : int ret;
2555 24130700 : if (psLock->eType == LOCK_SPIN)
2556 6547260 : ret = CPLAcquireSpinLock(psLock->u.hSpinLock);
2557 : else
2558 17583400 : ret = CPLAcquireMutex(psLock->u.hMutex, 1000);
2559 : #ifdef DEBUG_CONTENTION
2560 26083700 : if (ret && psLock->bDebugPerfAsked &&
2561 1952510 : CPLAtomicInc(&(psLock->nCurrentHolders)) == 1)
2562 : {
2563 1952510 : psLock->bDebugPerf = true;
2564 1952510 : psLock->nStartTime = nStartTime;
2565 : }
2566 : #endif
2567 24131200 : return ret;
2568 : }
2569 :
2570 : /************************************************************************/
2571 : /* CPLReleaseLock() */
2572 : /************************************************************************/
2573 :
2574 24136300 : void CPLReleaseLock(CPLLock *psLock)
2575 : {
2576 : #ifdef DEBUG_CONTENTION
2577 24136300 : bool bHitMaxDiff = false;
2578 24136300 : GIntBig nMaxDiff = 0;
2579 24136300 : double dfAvgDiff = 0;
2580 24136300 : if (psLock->bDebugPerf && CPLAtomicDec(&(psLock->nCurrentHolders)) == 0)
2581 : {
2582 1952520 : const GUIntBig nStopTime = CPLrdtscp();
2583 : // coverity[missing_lock:FALSE]
2584 1952520 : const GIntBig nDiffTime =
2585 1952520 : static_cast<GIntBig>(nStopTime - psLock->nStartTime);
2586 1952520 : if (nDiffTime > psLock->nMaxDiff)
2587 : {
2588 29 : bHitMaxDiff = true;
2589 29 : psLock->nMaxDiff = nDiffTime;
2590 : }
2591 1952520 : nMaxDiff = psLock->nMaxDiff;
2592 1952520 : psLock->nIters++;
2593 1952520 : psLock->dfAvgDiff += (nDiffTime - psLock->dfAvgDiff) / psLock->nIters;
2594 1952520 : dfAvgDiff = psLock->dfAvgDiff;
2595 : }
2596 : #endif
2597 24136300 : if (psLock->eType == LOCK_SPIN)
2598 6547320 : CPLReleaseSpinLock(psLock->u.hSpinLock);
2599 : else
2600 17589000 : CPLReleaseMutex(psLock->u.hMutex);
2601 : #ifdef DEBUG_CONTENTION
2602 24136300 : if (psLock->bDebugPerf &&
2603 1952490 : (bHitMaxDiff || (psLock->nIters % 1000000) == (1000000 - 1)))
2604 : {
2605 29 : CPLDebug("LOCK", "Lock contention : max = " CPL_FRMT_GIB ", avg = %.0f",
2606 : nMaxDiff, dfAvgDiff);
2607 : }
2608 : #endif
2609 24136300 : }
2610 :
2611 : /************************************************************************/
2612 : /* CPLDestroyLock() */
2613 : /************************************************************************/
2614 :
2615 34243 : void CPLDestroyLock(CPLLock *psLock)
2616 : {
2617 34243 : if (psLock->eType == LOCK_SPIN)
2618 33763 : CPLDestroySpinLock(psLock->u.hSpinLock);
2619 : else
2620 480 : CPLDestroyMutex(psLock->u.hMutex);
2621 34245 : free(psLock);
2622 34245 : }
2623 :
2624 : /************************************************************************/
2625 : /* CPLLockSetDebugPerf() */
2626 : /************************************************************************/
2627 :
2628 : #ifdef DEBUG_CONTENTION
2629 5046 : void CPLLockSetDebugPerf(CPLLock *psLock, int bEnableIn)
2630 : {
2631 5046 : psLock->bDebugPerfAsked = CPL_TO_BOOL(bEnableIn);
2632 5046 : }
2633 : #else
2634 : void CPLLockSetDebugPerf(CPLLock * /* psLock */, int bEnableIn)
2635 : {
2636 : if (!bEnableIn)
2637 : return;
2638 :
2639 : static bool bOnce = false;
2640 : if (!bOnce)
2641 : {
2642 : bOnce = true;
2643 : CPLDebug("LOCK", "DEBUG_CONTENTION not available");
2644 : }
2645 : }
2646 : #endif
2647 :
2648 : /************************************************************************/
2649 : /* CPLLockHolder() */
2650 : /************************************************************************/
2651 :
2652 5046 : CPLLockHolder::CPLLockHolder(CPLLock **phLock, CPLLockType eType,
2653 5046 : const char *pszFileIn, int nLineIn)
2654 :
2655 : {
2656 : #ifndef MUTEX_NONE
2657 5046 : pszFile = pszFileIn;
2658 5046 : nLine = nLineIn;
2659 :
2660 : #ifdef DEBUG_MUTEX
2661 : // XXX: There is no way to use CPLDebug() here because it works with
2662 : // mutexes itself so we will fall in infinite recursion. Good old
2663 : // fprintf() will do the job right.
2664 : fprintf(stderr, "CPLLockHolder: Request %p for pid %ld at %d/%s.\n",
2665 : *phLock, static_cast<long>(CPLGetPID()), nLine, pszFile);
2666 : #endif
2667 :
2668 5046 : if (!CPLCreateOrAcquireLock(phLock, eType))
2669 : {
2670 0 : fprintf(stderr, "CPLLockHolder: Failed to acquire lock!\n");
2671 0 : hLock = nullptr;
2672 : }
2673 : else
2674 : {
2675 : #ifdef DEBUG_MUTEX
2676 : fprintf(stderr, "CPLLockHolder: Acquired %p for pid %ld at %d/%s.\n",
2677 : *phLock, static_cast<long>(CPLGetPID()), nLine, pszFile);
2678 : #endif
2679 :
2680 5046 : hLock = *phLock;
2681 : }
2682 : #endif // ndef MUTEX_NONE
2683 5046 : }
2684 :
2685 : /************************************************************************/
2686 : /* CPLLockHolder() */
2687 : /************************************************************************/
2688 :
2689 24126100 : CPLLockHolder::CPLLockHolder(CPLLock *hLockIn, const char *pszFileIn,
2690 24126100 : int nLineIn)
2691 :
2692 : {
2693 : #ifndef MUTEX_NONE
2694 24126100 : pszFile = pszFileIn;
2695 24126100 : nLine = nLineIn;
2696 24126100 : hLock = hLockIn;
2697 :
2698 24126100 : if (hLock != nullptr)
2699 : {
2700 24126000 : if (!CPLAcquireLock(hLock))
2701 : {
2702 0 : fprintf(stderr, "CPLLockHolder: Failed to acquire lock!\n");
2703 0 : hLock = nullptr;
2704 : }
2705 : }
2706 : #endif // ndef MUTEX_NONE
2707 24126500 : }
2708 :
2709 : /************************************************************************/
2710 : /* ~CPLLockHolder() */
2711 : /************************************************************************/
2712 :
2713 48263000 : CPLLockHolder::~CPLLockHolder()
2714 :
2715 : {
2716 : #ifndef MUTEX_NONE
2717 24131400 : if (hLock != nullptr)
2718 : {
2719 : #ifdef DEBUG_MUTEX
2720 : fprintf(stderr, "~CPLLockHolder: Release %p for pid %ld at %d/%s.\n",
2721 : hLock, static_cast<long>(CPLGetPID()), nLine, pszFile);
2722 : #endif
2723 24131500 : CPLReleaseLock(hLock);
2724 : }
2725 : #endif // ndef MUTEX_NONE
2726 24131500 : }
2727 :
2728 : /************************************************************************/
2729 : /* CPLGetCurrentProcessID() */
2730 : /************************************************************************/
2731 :
2732 : #ifdef CPL_MULTIPROC_WIN32
2733 :
2734 : int CPLGetCurrentProcessID()
2735 : {
2736 : return GetCurrentProcessId();
2737 : }
2738 :
2739 : #else
2740 :
2741 : #include <sys/types.h>
2742 : #include <unistd.h>
2743 :
2744 2336 : int CPLGetCurrentProcessID()
2745 : {
2746 2336 : return getpid();
2747 : }
2748 :
2749 : #endif
|