MagickCore 7.1.1-43
Convert, Edit, Or Compose Bitmap Images
Loading...
Searching...
No Matches
thread-private.h
1/*
2 Copyright @ 1999 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)
27extern "C" {
28#endif
29
30#define magick_number_threads(source,destination,chunk,factor) \
31 num_threads(GetMagickNumberThreads((source),(destination),(chunk),(factor)))
32#if defined(__clang__) || (__GNUC__ > 3) || ((__GNUC__ == 3) && (__GNUC_MINOR__ > 10))
33#define MagickCachePrefetch(address,mode,locality) \
34 __builtin_prefetch(address,mode,locality)
35#else
36#define MagickCachePrefetch(address,mode,locality) \
37 magick_unreferenced(address); \
38 magick_unreferenced(mode); \
39 magick_unreferenced(locality);
40#endif
41
42#if defined(MAGICKCORE_THREAD_SUPPORT)
43 typedef pthread_mutex_t MagickMutexType;
44#elif defined(MAGICKCORE_WINDOWS_SUPPORT)
45 typedef CRITICAL_SECTION MagickMutexType;
46#else
47 typedef size_t MagickMutexType;
48#endif
49
50static inline int GetMagickNumberThreads(const Image *source,
51 const Image *destination,const size_t chunk,const int factor)
52{
53#define WorkLoadFactor (64UL << factor)
54
55 const CacheType
56 destination_type = (CacheType) GetImagePixelCacheType(destination),
57 source_type = (CacheType) GetImagePixelCacheType(source);
58
59 int
60 number_threads;
61
62 /*
63 Return number of threads dependent on cache type and work load.
64 */
65 number_threads=(int) MagickMax(MagickMin(chunk/WorkLoadFactor,
66 GetMagickResourceLimit(ThreadResource)),1);
67 if (((source_type != MemoryCache) && (source_type != MapCache)) ||
68 ((destination_type != MemoryCache) && (destination_type != MapCache)))
69 number_threads=MagickMin(number_threads,2);
70 return(number_threads);
71}
72
73static inline MagickThreadType GetMagickThreadId(void)
74{
75#if defined(MAGICKCORE_THREAD_SUPPORT)
76 return(pthread_self());
77#elif defined(MAGICKCORE_WINDOWS_SUPPORT)
78 return(GetCurrentThreadId());
79#else
80 return(getpid());
81#endif
82}
83
84static inline void GetMagickThreadFilename(const char *filename,
85 char *thread_filename)
86{
87 MagickThreadType
88 id;
89
90 char
91 thread_id[2*sizeof(id)+1];
92
93 ssize_t
94 i;
95
96 unsigned char
97 bytes[sizeof(id)];
98
99 id=GetMagickThreadId();
100 (void) memcpy(bytes,&id,sizeof(id));
101 for (i=0; i < (ssize_t) sizeof(bytes); i++)
102 (void) FormatLocaleString(thread_id+2*i,MagickPathExtent,"%02x",bytes[i]);
103 thread_id[sizeof(thread_id)-1]='\0';
104 (void) FormatLocaleString(thread_filename,MagickPathExtent,"%s|%s",thread_id,
105 filename);
106}
107
108static inline size_t GetMagickThreadSignature(void)
109{
110#if defined(MAGICKCORE_THREAD_SUPPORT)
111 {
112 union
113 {
114 pthread_t
115 id;
116
117 size_t
118 signature;
119 } magick_thread;
120
121 magick_thread.signature=0UL;
122 magick_thread.id=pthread_self();
123 return(magick_thread.signature);
124 }
125#elif defined(MAGICKCORE_WINDOWS_SUPPORT)
126 return((size_t) GetCurrentThreadId());
127#else
128 return((size_t) getpid());
129#endif
130}
131
132static inline MagickBooleanType IsMagickThreadEqual(const MagickThreadType id)
133{
134#if defined(MAGICKCORE_THREAD_SUPPORT)
135 if (pthread_equal(id,pthread_self()) != 0)
136 return(MagickTrue);
137#elif defined(MAGICKCORE_WINDOWS_SUPPORT)
138 if (id == GetCurrentThreadId())
139 return(MagickTrue);
140#else
141 if (id == getpid())
142 return(MagickTrue);
143#endif
144 return(MagickFalse);
145}
146
147/*
148 Lightweight OpenMP methods.
149*/
150static inline size_t GetOpenMPMaximumThreads(void)
151{
152#if defined(MAGICKCORE_OPENMP_SUPPORT)
153 return((size_t) omp_get_max_threads());
154#else
155 return(1);
156#endif
157}
158
159static inline int GetOpenMPThreadId(void)
160{
161#if defined(MAGICKCORE_OPENMP_SUPPORT)
162 return(omp_get_thread_num());
163#else
164 return(0);
165#endif
166}
167
168#if defined(MAGICKCORE_OPENMP_SUPPORT)
169static inline void SetOpenMPMaximumThreads(const int threads)
170{
171 omp_set_num_threads(threads);
172#else
173static inline void SetOpenMPMaximumThreads(const int magick_unused(threads))
174{
175 magick_unreferenced(threads);
176#endif
177}
178
179#if defined(MAGICKCORE_OPENMP_SUPPORT)
180static inline void SetOpenMPNested(const int value)
181{
182 omp_set_nested(value);
183#else
184static inline void SetOpenMPNested(const int magick_unused(value))
185{
186 magick_unreferenced(value);
187#endif
188}
189
190#if defined(__cplusplus) || defined(c_plusplus)
191}
192#endif
193
194#endif