1 /*
2   Copyright 1999-2019 ImageMagick Studio LLC, a non-profit organization
3   dedicated to making software imaging solutions freely available.
4 
5   You may not use this file except in compliance with the License.  You may
6   obtain a copy of the License at
7 
8     https://imagemagick.org/script/license.php
9 
10   Unless required by applicable law or agreed to in writing, software
11   distributed under the License is distributed on an "AS IS" BASIS,
12   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   See the License for the specific language governing permissions and
14   limitations under the License.
15 
16   MagickCore private methods for internal threading.
17 */
18 #ifndef MAGICKCORE_THREAD_PRIVATE_H
19 #define MAGICKCORE_THREAD_PRIVATE_H
20 
21 #include "MagickCore/cache.h"
22 #include "MagickCore/image-private.h"
23 #include "MagickCore/resource_.h"
24 #include "MagickCore/thread_.h"
25 
26 #if defined(__cplusplus) || defined(c_plusplus)
27 extern "C" {
28 #endif
29 
30 /*
31   Number of threads bounded by the amount of work and any thread resource limit.
32   The limit is 2 if the pixel cache type is not memory or memory-mapped.
33 */
34 #define magick_number_threads(source,destination,chunk,multithreaded) \
35   num_threads((multithreaded) == 0 ? 1 : \
36     ((GetImagePixelCacheType(source) != MemoryCache) && \
37      (GetImagePixelCacheType(source) != MapCache)) || \
38     ((GetImagePixelCacheType(destination) != MemoryCache) && \
39      (GetImagePixelCacheType(destination) != MapCache)) ? \
40     MagickMax(MagickMin(GetMagickResourceLimit(ThreadResource),2),1) : \
41     MagickMax(MagickMin((ssize_t) GetMagickResourceLimit(ThreadResource),(ssize_t) (chunk)/64),1))
42 
43 #if defined(__clang__) || (__GNUC__ > 3) || ((__GNUC__ == 3) && (__GNUC_MINOR__ > 10))
44 #define MagickCachePrefetch(address,mode,locality) \
45   __builtin_prefetch(address,mode,locality)
46 #else
47 #define MagickCachePrefetch(address,mode,locality) \
48   magick_unreferenced(address); \
49   magick_unreferenced(mode); \
50   magick_unreferenced(locality);
51 #endif
52 
53 #if defined(MAGICKCORE_THREAD_SUPPORT)
54   typedef pthread_mutex_t MagickMutexType;
55 #elif defined(MAGICKCORE_WINDOWS_SUPPORT)
56   typedef CRITICAL_SECTION MagickMutexType;
57 #else
58   typedef size_t MagickMutexType;
59 #endif
60 
GetMagickThreadId(void)61 static inline MagickThreadType GetMagickThreadId(void)
62 {
63 #if defined(MAGICKCORE_THREAD_SUPPORT)
64   return(pthread_self());
65 #elif defined(MAGICKCORE_WINDOWS_SUPPORT)
66   return(GetCurrentThreadId());
67 #else
68   return(getpid());
69 #endif
70 }
71 
GetMagickThreadSignature(void)72 static inline size_t GetMagickThreadSignature(void)
73 {
74 #if defined(MAGICKCORE_THREAD_SUPPORT)
75   {
76     union
77     {
78       pthread_t
79         id;
80 
81       size_t
82         signature;
83     } magick_thread;
84 
85     magick_thread.signature=0UL;
86     magick_thread.id=pthread_self();
87     return(magick_thread.signature);
88   }
89 #elif defined(MAGICKCORE_WINDOWS_SUPPORT)
90   return((size_t) GetCurrentThreadId());
91 #else
92   return((size_t) getpid());
93 #endif
94 }
95 
IsMagickThreadEqual(const MagickThreadType id)96 static inline MagickBooleanType IsMagickThreadEqual(const MagickThreadType id)
97 {
98 #if defined(MAGICKCORE_THREAD_SUPPORT)
99   if (pthread_equal(id,pthread_self()) != 0)
100     return(MagickTrue);
101 #elif defined(MAGICKCORE_WINDOWS_SUPPORT)
102   if (id == GetCurrentThreadId())
103     return(MagickTrue);
104 #else
105   if (id == getpid())
106     return(MagickTrue);
107 #endif
108   return(MagickFalse);
109 }
110 
111 /*
112   Lightweight OpenMP methods.
113 */
GetOpenMPMaximumThreads(void)114 static inline size_t GetOpenMPMaximumThreads(void)
115 {
116 #if defined(MAGICKCORE_OPENMP_SUPPORT)
117   return(omp_get_max_threads());
118 #else
119   return(1);
120 #endif
121 }
122 
GetOpenMPThreadId(void)123 static inline int GetOpenMPThreadId(void)
124 {
125 #if defined(MAGICKCORE_OPENMP_SUPPORT)
126   return(omp_get_thread_num());
127 #else
128   return(0);
129 #endif
130 }
131 
SetOpenMPMaximumThreads(const int threads)132 static inline void SetOpenMPMaximumThreads(const int threads)
133 {
134 #if defined(MAGICKCORE_OPENMP_SUPPORT)
135   omp_set_num_threads(threads);
136 #else
137   (void) threads;
138 #endif
139 }
140 
SetOpenMPNested(const int value)141 static inline void SetOpenMPNested(const int value)
142 {
143 #if defined(MAGICKCORE_OPENMP_SUPPORT)
144   omp_set_nested(value);
145 #else
146   (void) value;
147 #endif
148 }
149 
150 #if defined(__cplusplus) || defined(c_plusplus)
151 }
152 #endif
153 
154 #endif
155