cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 1 | /* |
| 2 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 3 | % % |
| 4 | % % |
| 5 | % % |
| 6 | % CCCC OOO DDDD EEEEE RRRR % |
| 7 | % C O O D D E R R % |
| 8 | % C O O D D EEE RRRR % |
| 9 | % C O O D D E R R % |
| 10 | % CCCC OOO DDDD EEEEE R R % |
| 11 | % % |
| 12 | % % |
| 13 | % MagickCore Image Coder Methods % |
| 14 | % % |
| 15 | % Software Design % |
cristy | de984cd | 2013-12-01 14:49:27 +0000 | [diff] [blame] | 16 | % Cristy % |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 17 | % May 2001 % |
| 18 | % % |
| 19 | % % |
Cristy | f775a5c | 2019-11-26 14:27:47 -0500 | [diff] [blame] | 20 | % Copyright 1999-2020 ImageMagick Studio LLC, a non-profit organization % |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 21 | % dedicated to making software imaging solutions freely available. % |
| 22 | % % |
| 23 | % You may not use this file except in compliance with the License. You may % |
| 24 | % obtain a copy of the License at % |
| 25 | % % |
Cristy | 9ddfcca | 2018-09-09 19:46:34 -0400 | [diff] [blame] | 26 | % https://imagemagick.org/script/license.php % |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 27 | % % |
| 28 | % Unless required by applicable law or agreed to in writing, software % |
| 29 | % distributed under the License is distributed on an "AS IS" BASIS, % |
| 30 | % WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. % |
| 31 | % See the License for the specific language governing permissions and % |
| 32 | % limitations under the License. % |
| 33 | % % |
| 34 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 35 | % |
| 36 | % |
| 37 | */ |
| 38 | |
| 39 | /* |
| 40 | Include declarations. |
| 41 | */ |
cristy | 4c08aed | 2011-07-01 19:47:50 +0000 | [diff] [blame] | 42 | #include "MagickCore/studio.h" |
| 43 | #include "MagickCore/blob.h" |
| 44 | #include "MagickCore/client.h" |
| 45 | #include "MagickCore/coder.h" |
cristy | 5ff4eaf | 2011-09-03 01:38:02 +0000 | [diff] [blame] | 46 | #include "MagickCore/coder-private.h" |
cristy | 4c08aed | 2011-07-01 19:47:50 +0000 | [diff] [blame] | 47 | #include "MagickCore/configure.h" |
| 48 | #include "MagickCore/draw.h" |
| 49 | #include "MagickCore/exception.h" |
| 50 | #include "MagickCore/exception-private.h" |
cristy | 4c08aed | 2011-07-01 19:47:50 +0000 | [diff] [blame] | 51 | #include "MagickCore/log.h" |
| 52 | #include "MagickCore/memory_.h" |
Dirk Lemstra | 06344a0 | 2017-10-15 10:10:01 +0200 | [diff] [blame] | 53 | #include "MagickCore/memory-private.h" |
cristy | 4c08aed | 2011-07-01 19:47:50 +0000 | [diff] [blame] | 54 | #include "MagickCore/option.h" |
| 55 | #include "MagickCore/semaphore.h" |
| 56 | #include "MagickCore/string_.h" |
| 57 | #include "MagickCore/splay-tree.h" |
| 58 | #include "MagickCore/token.h" |
| 59 | #include "MagickCore/utility.h" |
cristy | d1dd6e4 | 2011-09-04 01:46:08 +0000 | [diff] [blame] | 60 | #include "MagickCore/utility-private.h" |
Dirk Lemstra | 826cbeb | 2018-10-05 00:12:46 +0200 | [diff] [blame] | 61 | #include "coders/coders.h" |
| 62 | |
| 63 | /* |
| 64 | Define declarations. |
| 65 | */ |
| 66 | #define AddMagickCoder(coder) Magick ## coder ## Aliases |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 67 | |
| 68 | /* |
cristy | 54a531d | 2009-10-21 17:58:01 +0000 | [diff] [blame] | 69 | Typedef declarations. |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 70 | */ |
cristy | e3e77a1 | 2009-10-16 00:47:21 +0000 | [diff] [blame] | 71 | typedef struct _CoderMapInfo |
| 72 | { |
| 73 | const char |
| 74 | *magick, |
| 75 | *name; |
| 76 | } CoderMapInfo; |
cristy | 54a531d | 2009-10-21 17:58:01 +0000 | [diff] [blame] | 77 | |
| 78 | /* |
| 79 | Static declarations. |
| 80 | */ |
cristy | e3e77a1 | 2009-10-16 00:47:21 +0000 | [diff] [blame] | 81 | static const CoderMapInfo |
| 82 | CoderMap[] = |
| 83 | { |
Dirk Lemstra | 826cbeb | 2018-10-05 00:12:46 +0200 | [diff] [blame] | 84 | #include "coders/coders-list.h" |
Dirk Lemstra | 7ad38d1 | 2018-10-05 07:52:22 +0200 | [diff] [blame] | 85 | }; |
cristy | e3e77a1 | 2009-10-16 00:47:21 +0000 | [diff] [blame] | 86 | |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 87 | static SemaphoreInfo |
| 88 | *coder_semaphore = (SemaphoreInfo *) NULL; |
| 89 | |
| 90 | static SplayTreeInfo |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 91 | *coder_cache = (SplayTreeInfo *) NULL; |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 92 | |
| 93 | /* |
| 94 | Forward declarations. |
| 95 | */ |
| 96 | static MagickBooleanType |
Dirk Lemstra | e2939a4 | 2018-10-04 23:20:08 +0200 | [diff] [blame] | 97 | IsCoderTreeInstantiated(ExceptionInfo *); |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 98 | |
| 99 | /* |
| 100 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 101 | % % |
| 102 | % % |
| 103 | % % |
cristy | 6c6322e | 2014-09-09 22:39:21 +0000 | [diff] [blame] | 104 | + A c q u i r e C o d e r C a c h e % |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 105 | % % |
| 106 | % % |
| 107 | % % |
| 108 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 109 | % |
cristy | 6c6322e | 2014-09-09 22:39:21 +0000 | [diff] [blame] | 110 | % AcquireCoderCache() caches one or more coder configurations which provides a |
| 111 | % mapping between coder attributes and a coder name. |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 112 | % |
| 113 | % The format of the AcquireCoderCache coder is: |
| 114 | % |
| 115 | % SplayTreeInfo *AcquireCoderCache(const char *filename, |
| 116 | % ExceptionInfo *exception) |
| 117 | % |
| 118 | % A description of each parameter follows: |
| 119 | % |
| 120 | % o filename: the font file name. |
| 121 | % |
| 122 | % o exception: return any errors or warnings in this structure. |
| 123 | % |
| 124 | */ |
| 125 | |
| 126 | static void *DestroyCoderNode(void *coder_info) |
| 127 | { |
| 128 | register CoderInfo |
| 129 | *p; |
| 130 | |
| 131 | p=(CoderInfo *) coder_info; |
| 132 | if (p->exempt == MagickFalse) |
| 133 | { |
| 134 | if (p->path != (char *) NULL) |
| 135 | p->path=DestroyString(p->path); |
| 136 | if (p->name != (char *) NULL) |
| 137 | p->name=DestroyString(p->name); |
| 138 | if (p->magick != (char *) NULL) |
| 139 | p->magick=DestroyString(p->magick); |
| 140 | } |
| 141 | return(RelinquishMagickMemory(p)); |
| 142 | } |
| 143 | |
Dirk Lemstra | e2939a4 | 2018-10-04 23:20:08 +0200 | [diff] [blame] | 144 | static SplayTreeInfo *AcquireCoderCache(ExceptionInfo *exception) |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 145 | { |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 146 | MagickStatusType |
| 147 | status; |
| 148 | |
| 149 | register ssize_t |
| 150 | i; |
| 151 | |
| 152 | SplayTreeInfo |
dirk | b18d864 | 2016-07-17 19:32:43 +0200 | [diff] [blame] | 153 | *cache; |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 154 | |
| 155 | /* |
Dirk Lemstra | e2939a4 | 2018-10-04 23:20:08 +0200 | [diff] [blame] | 156 | Load built-in coder map. |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 157 | */ |
dirk | b18d864 | 2016-07-17 19:32:43 +0200 | [diff] [blame] | 158 | cache=NewSplayTree(CompareSplayTreeString,RelinquishMagickMemory, |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 159 | DestroyCoderNode); |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 160 | status=MagickTrue; |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 161 | for (i=0; i < (ssize_t) (sizeof(CoderMap)/sizeof(*CoderMap)); i++) |
| 162 | { |
| 163 | CoderInfo |
| 164 | *coder_info; |
| 165 | |
| 166 | register const CoderMapInfo |
| 167 | *p; |
| 168 | |
| 169 | p=CoderMap+i; |
| 170 | coder_info=(CoderInfo *) AcquireMagickMemory(sizeof(*coder_info)); |
| 171 | if (coder_info == (CoderInfo *) NULL) |
| 172 | { |
| 173 | (void) ThrowMagickException(exception,GetMagickModule(), |
| 174 | ResourceLimitError,"MemoryAllocationFailed","`%s'",p->name); |
| 175 | continue; |
| 176 | } |
Cristy | 81bfff2 | 2018-03-10 07:58:31 -0500 | [diff] [blame] | 177 | (void) memset(coder_info,0,sizeof(*coder_info)); |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 178 | coder_info->path=(char *) "[built-in]"; |
| 179 | coder_info->magick=(char *) p->magick; |
| 180 | coder_info->name=(char *) p->name; |
| 181 | coder_info->exempt=MagickTrue; |
cristy | e1c94d9 | 2015-06-28 12:16:33 +0000 | [diff] [blame] | 182 | coder_info->signature=MagickCoreSignature; |
dirk | b18d864 | 2016-07-17 19:32:43 +0200 | [diff] [blame] | 183 | status&=AddValueToSplayTree(cache,ConstantString(coder_info->magick), |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 184 | coder_info); |
| 185 | if (status == MagickFalse) |
| 186 | (void) ThrowMagickException(exception,GetMagickModule(), |
| 187 | ResourceLimitError,"MemoryAllocationFailed","`%s'",coder_info->name); |
| 188 | } |
dirk | b18d864 | 2016-07-17 19:32:43 +0200 | [diff] [blame] | 189 | return(cache); |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 190 | } |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 191 | |
| 192 | /* |
| 193 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 194 | % % |
| 195 | % % |
| 196 | % % |
cristy | f34a145 | 2009-10-24 22:29:27 +0000 | [diff] [blame] | 197 | + C o d e r C o m p o n e n t G e n e s i s % |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 198 | % % |
| 199 | % % |
| 200 | % % |
| 201 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 202 | % |
cristy | f34a145 | 2009-10-24 22:29:27 +0000 | [diff] [blame] | 203 | % CoderComponentGenesis() instantiates the coder component. |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 204 | % |
cristy | f34a145 | 2009-10-24 22:29:27 +0000 | [diff] [blame] | 205 | % The format of the CoderComponentGenesis method is: |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 206 | % |
cristy | f34a145 | 2009-10-24 22:29:27 +0000 | [diff] [blame] | 207 | % MagickBooleanType CoderComponentGenesis(void) |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 208 | % |
| 209 | */ |
cristy | 5ff4eaf | 2011-09-03 01:38:02 +0000 | [diff] [blame] | 210 | MagickPrivate MagickBooleanType CoderComponentGenesis(void) |
cristy | f34a145 | 2009-10-24 22:29:27 +0000 | [diff] [blame] | 211 | { |
cristy | 7c97706 | 2014-04-04 14:05:53 +0000 | [diff] [blame] | 212 | if (coder_semaphore == (SemaphoreInfo *) NULL) |
| 213 | coder_semaphore=AcquireSemaphoreInfo(); |
cristy | f34a145 | 2009-10-24 22:29:27 +0000 | [diff] [blame] | 214 | return(MagickTrue); |
| 215 | } |
| 216 | |
| 217 | /* |
| 218 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 219 | % % |
| 220 | % % |
| 221 | % % |
| 222 | + C o d e r C o m p o n e n t T e r m i n u s % |
| 223 | % % |
| 224 | % % |
| 225 | % % |
| 226 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 227 | % |
| 228 | % CoderComponentTerminus() destroys the coder component. |
| 229 | % |
| 230 | % The format of the CoderComponentTerminus method is: |
| 231 | % |
| 232 | % CoderComponentTerminus(void) |
| 233 | % |
| 234 | */ |
cristy | 5ff4eaf | 2011-09-03 01:38:02 +0000 | [diff] [blame] | 235 | MagickPrivate void CoderComponentTerminus(void) |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 236 | { |
cristy | 18b1744 | 2009-10-25 18:36:48 +0000 | [diff] [blame] | 237 | if (coder_semaphore == (SemaphoreInfo *) NULL) |
cristy | 04b11db | 2014-02-16 15:10:39 +0000 | [diff] [blame] | 238 | ActivateSemaphoreInfo(&coder_semaphore); |
cristy | f84a193 | 2010-01-03 18:00:18 +0000 | [diff] [blame] | 239 | LockSemaphoreInfo(coder_semaphore); |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 240 | if (coder_cache != (SplayTreeInfo *) NULL) |
| 241 | coder_cache=DestroySplayTree(coder_cache); |
cristy | f84a193 | 2010-01-03 18:00:18 +0000 | [diff] [blame] | 242 | UnlockSemaphoreInfo(coder_semaphore); |
cristy | 3d162a9 | 2014-02-16 14:05:06 +0000 | [diff] [blame] | 243 | RelinquishSemaphoreInfo(&coder_semaphore); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 244 | } |
| 245 | |
| 246 | /* |
| 247 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 248 | % % |
| 249 | % % |
| 250 | % % |
| 251 | + G e t C o d e r I n f o % |
| 252 | % % |
| 253 | % % |
| 254 | % % |
| 255 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 256 | % |
| 257 | % GetCoderInfo searches the coder list for the specified name and if found |
| 258 | % returns attributes for that coder. |
| 259 | % |
| 260 | % The format of the GetCoderInfo method is: |
| 261 | % |
| 262 | % const CoderInfo *GetCoderInfo(const char *name,ExceptionInfo *exception) |
| 263 | % |
| 264 | % A description of each parameter follows: |
| 265 | % |
| 266 | % o name: the coder name. |
| 267 | % |
| 268 | % o exception: return any errors or warnings in this structure. |
| 269 | % |
| 270 | */ |
| 271 | MagickExport const CoderInfo *GetCoderInfo(const char *name, |
| 272 | ExceptionInfo *exception) |
| 273 | { |
| 274 | assert(exception != (ExceptionInfo *) NULL); |
cristy | 904e591 | 2014-03-15 19:53:14 +0000 | [diff] [blame] | 275 | if (IsCoderTreeInstantiated(exception) == MagickFalse) |
| 276 | return((const CoderInfo *) NULL); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 277 | if ((name == (const char *) NULL) || (LocaleCompare(name,"*") == 0)) |
Cristy | 1d8b1f1 | 2017-09-16 10:13:23 -0400 | [diff] [blame] | 278 | return((const CoderInfo *) GetRootValueFromSplayTree(coder_cache)); |
| 279 | return((const CoderInfo *) GetValueFromSplayTree(coder_cache,name)); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 280 | } |
| 281 | |
| 282 | /* |
| 283 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 284 | % % |
| 285 | % % |
| 286 | % % |
| 287 | % G e t C o d e r I n f o L i s t % |
| 288 | % % |
| 289 | % % |
| 290 | % % |
| 291 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 292 | % |
| 293 | % GetCoderInfoList() returns any coder_map that match the specified pattern. |
| 294 | % The format of the GetCoderInfoList function is: |
| 295 | % |
| 296 | % const CoderInfo **GetCoderInfoList(const char *pattern, |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 297 | % size_t *number_coders,ExceptionInfo *exception) |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 298 | % |
| 299 | % A description of each parameter follows: |
| 300 | % |
| 301 | % o pattern: Specifies a pointer to a text string containing a pattern. |
| 302 | % |
| 303 | % o number_coders: This integer returns the number of coders in the list. |
| 304 | % |
| 305 | % o exception: return any errors or warnings in this structure. |
| 306 | % |
| 307 | */ |
| 308 | |
| 309 | static int CoderInfoCompare(const void *x,const void *y) |
| 310 | { |
| 311 | const CoderInfo |
| 312 | **p, |
| 313 | **q; |
| 314 | |
| 315 | p=(const CoderInfo **) x, |
| 316 | q=(const CoderInfo **) y; |
| 317 | if (LocaleCompare((*p)->path,(*q)->path) == 0) |
| 318 | return(LocaleCompare((*p)->name,(*q)->name)); |
| 319 | return(LocaleCompare((*p)->path,(*q)->path)); |
| 320 | } |
| 321 | |
| 322 | MagickExport const CoderInfo **GetCoderInfoList(const char *pattern, |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 323 | size_t *number_coders,ExceptionInfo *exception) |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 324 | { |
| 325 | const CoderInfo |
| 326 | **coder_map; |
| 327 | |
| 328 | register const CoderInfo |
| 329 | *p; |
| 330 | |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 331 | register ssize_t |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 332 | i; |
| 333 | |
| 334 | /* |
| 335 | Allocate coder list. |
| 336 | */ |
| 337 | assert(pattern != (char *) NULL); |
| 338 | (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",pattern); |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 339 | assert(number_coders != (size_t *) NULL); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 340 | *number_coders=0; |
| 341 | p=GetCoderInfo("*",exception); |
| 342 | if (p == (const CoderInfo *) NULL) |
| 343 | return((const CoderInfo **) NULL); |
| 344 | coder_map=(const CoderInfo **) AcquireQuantumMemory((size_t) |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 345 | GetNumberOfNodesInSplayTree(coder_cache)+1UL,sizeof(*coder_map)); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 346 | if (coder_map == (const CoderInfo **) NULL) |
| 347 | return((const CoderInfo **) NULL); |
| 348 | /* |
| 349 | Generate coder list. |
| 350 | */ |
cristy | f84a193 | 2010-01-03 18:00:18 +0000 | [diff] [blame] | 351 | LockSemaphoreInfo(coder_semaphore); |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 352 | ResetSplayTreeIterator(coder_cache); |
| 353 | p=(const CoderInfo *) GetNextValueInSplayTree(coder_cache); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 354 | for (i=0; p != (const CoderInfo *) NULL; ) |
| 355 | { |
| 356 | if ((p->stealth == MagickFalse) && |
| 357 | (GlobExpression(p->name,pattern,MagickFalse) != MagickFalse)) |
| 358 | coder_map[i++]=p; |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 359 | p=(const CoderInfo *) GetNextValueInSplayTree(coder_cache); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 360 | } |
cristy | f84a193 | 2010-01-03 18:00:18 +0000 | [diff] [blame] | 361 | UnlockSemaphoreInfo(coder_semaphore); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 362 | qsort((void *) coder_map,(size_t) i,sizeof(*coder_map),CoderInfoCompare); |
| 363 | coder_map[i]=(CoderInfo *) NULL; |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 364 | *number_coders=(size_t) i; |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 365 | return(coder_map); |
| 366 | } |
| 367 | |
| 368 | /* |
| 369 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 370 | % % |
| 371 | % % |
| 372 | % % |
| 373 | % G e t C o d e r L i s t % |
| 374 | % % |
| 375 | % % |
| 376 | % % |
| 377 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 378 | % |
| 379 | % GetCoderList() returns any coder_map that match the specified pattern. |
| 380 | % |
| 381 | % The format of the GetCoderList function is: |
| 382 | % |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 383 | % char **GetCoderList(const char *pattern,size_t *number_coders, |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 384 | % ExceptionInfo *exception) |
| 385 | % |
| 386 | % A description of each parameter follows: |
| 387 | % |
| 388 | % o pattern: Specifies a pointer to a text string containing a pattern. |
| 389 | % |
| 390 | % o number_coders: This integer returns the number of coders in the list. |
| 391 | % |
| 392 | % o exception: return any errors or warnings in this structure. |
| 393 | % |
| 394 | */ |
| 395 | |
| 396 | static int CoderCompare(const void *x,const void *y) |
| 397 | { |
| 398 | register const char |
| 399 | **p, |
| 400 | **q; |
| 401 | |
| 402 | p=(const char **) x; |
| 403 | q=(const char **) y; |
| 404 | return(LocaleCompare(*p,*q)); |
| 405 | } |
| 406 | |
| 407 | MagickExport char **GetCoderList(const char *pattern, |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 408 | size_t *number_coders,ExceptionInfo *exception) |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 409 | { |
| 410 | char |
| 411 | **coder_map; |
| 412 | |
| 413 | register const CoderInfo |
| 414 | *p; |
| 415 | |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 416 | register ssize_t |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 417 | i; |
| 418 | |
| 419 | /* |
| 420 | Allocate coder list. |
| 421 | */ |
| 422 | assert(pattern != (char *) NULL); |
| 423 | (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",pattern); |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 424 | assert(number_coders != (size_t *) NULL); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 425 | *number_coders=0; |
| 426 | p=GetCoderInfo("*",exception); |
| 427 | if (p == (const CoderInfo *) NULL) |
| 428 | return((char **) NULL); |
| 429 | coder_map=(char **) AcquireQuantumMemory((size_t) |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 430 | GetNumberOfNodesInSplayTree(coder_cache)+1UL,sizeof(*coder_map)); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 431 | if (coder_map == (char **) NULL) |
| 432 | return((char **) NULL); |
| 433 | /* |
| 434 | Generate coder list. |
| 435 | */ |
cristy | f84a193 | 2010-01-03 18:00:18 +0000 | [diff] [blame] | 436 | LockSemaphoreInfo(coder_semaphore); |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 437 | ResetSplayTreeIterator(coder_cache); |
| 438 | p=(const CoderInfo *) GetNextValueInSplayTree(coder_cache); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 439 | for (i=0; p != (const CoderInfo *) NULL; ) |
| 440 | { |
| 441 | if ((p->stealth == MagickFalse) && |
| 442 | (GlobExpression(p->name,pattern,MagickFalse) != MagickFalse)) |
| 443 | coder_map[i++]=ConstantString(p->name); |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 444 | p=(const CoderInfo *) GetNextValueInSplayTree(coder_cache); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 445 | } |
cristy | f84a193 | 2010-01-03 18:00:18 +0000 | [diff] [blame] | 446 | UnlockSemaphoreInfo(coder_semaphore); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 447 | qsort((void *) coder_map,(size_t) i,sizeof(*coder_map),CoderCompare); |
| 448 | coder_map[i]=(char *) NULL; |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 449 | *number_coders=(size_t) i; |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 450 | return(coder_map); |
| 451 | } |
| 452 | |
| 453 | /* |
| 454 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 455 | % % |
| 456 | % % |
| 457 | % % |
cristy | 904e591 | 2014-03-15 19:53:14 +0000 | [diff] [blame] | 458 | + I s C o d e r T r e e I n s t a n t i a t e d % |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 459 | % % |
| 460 | % % |
| 461 | % % |
| 462 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 463 | % |
cristy | cd2cd18 | 2014-03-18 12:10:55 +0000 | [diff] [blame] | 464 | % IsCoderTreeInstantiated() determines if the coder tree is instantiated. If |
| 465 | % not, it instantiates the tree and returns it. |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 466 | % |
cristy | 904e591 | 2014-03-15 19:53:14 +0000 | [diff] [blame] | 467 | % The format of the IsCoderInstantiated method is: |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 468 | % |
cristy | 904e591 | 2014-03-15 19:53:14 +0000 | [diff] [blame] | 469 | % MagickBooleanType IsCoderTreeInstantiated(ExceptionInfo *exception) |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 470 | % |
| 471 | % A description of each parameter follows. |
| 472 | % |
| 473 | % o exception: return any errors or warnings in this structure. |
| 474 | % |
| 475 | */ |
cristy | 904e591 | 2014-03-15 19:53:14 +0000 | [diff] [blame] | 476 | static MagickBooleanType IsCoderTreeInstantiated(ExceptionInfo *exception) |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 477 | { |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 478 | if (coder_cache == (SplayTreeInfo *) NULL) |
| 479 | { |
| 480 | if (coder_semaphore == (SemaphoreInfo *) NULL) |
| 481 | ActivateSemaphoreInfo(&coder_semaphore); |
| 482 | LockSemaphoreInfo(coder_semaphore); |
| 483 | if (coder_cache == (SplayTreeInfo *) NULL) |
Dirk Lemstra | e2939a4 | 2018-10-04 23:20:08 +0200 | [diff] [blame] | 484 | coder_cache=AcquireCoderCache(exception); |
cristy | 86e5ac9 | 2014-03-16 19:27:39 +0000 | [diff] [blame] | 485 | UnlockSemaphoreInfo(coder_semaphore); |
| 486 | } |
| 487 | return(coder_cache != (SplayTreeInfo *) NULL ? MagickTrue : MagickFalse); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 488 | } |
| 489 | |
| 490 | /* |
| 491 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 492 | % % |
| 493 | % % |
| 494 | % % |
| 495 | % L i s t C o d e r I n f o % |
| 496 | % % |
| 497 | % % |
| 498 | % % |
| 499 | %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% |
| 500 | % |
| 501 | % ListCoderInfo() lists the coder info to a file. |
| 502 | % |
| 503 | % The format of the ListCoderInfo coder is: |
| 504 | % |
| 505 | % MagickBooleanType ListCoderInfo(FILE *file,ExceptionInfo *exception) |
| 506 | % |
| 507 | % A description of each parameter follows. |
| 508 | % |
| 509 | % o file: An pointer to a FILE. |
| 510 | % |
| 511 | % o exception: return any errors or warnings in this structure. |
| 512 | % |
| 513 | */ |
| 514 | MagickExport MagickBooleanType ListCoderInfo(FILE *file, |
| 515 | ExceptionInfo *exception) |
| 516 | { |
| 517 | const char |
| 518 | *path; |
| 519 | |
| 520 | const CoderInfo |
| 521 | **coder_info; |
| 522 | |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 523 | register ssize_t |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 524 | i; |
| 525 | |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 526 | size_t |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 527 | number_coders; |
| 528 | |
cristy | 9d314ff | 2011-03-09 01:30:28 +0000 | [diff] [blame] | 529 | ssize_t |
| 530 | j; |
| 531 | |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 532 | if (file == (const FILE *) NULL) |
| 533 | file=stdout; |
| 534 | coder_info=GetCoderInfoList("*",&number_coders,exception); |
| 535 | if (coder_info == (const CoderInfo **) NULL) |
| 536 | return(MagickFalse); |
| 537 | path=(const char *) NULL; |
cristy | bb50337 | 2010-05-27 20:51:26 +0000 | [diff] [blame] | 538 | for (i=0; i < (ssize_t) number_coders; i++) |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 539 | { |
| 540 | if (coder_info[i]->stealth != MagickFalse) |
| 541 | continue; |
| 542 | if ((path == (const char *) NULL) || |
| 543 | (LocaleCompare(path,coder_info[i]->path) != 0)) |
| 544 | { |
| 545 | if (coder_info[i]->path != (char *) NULL) |
cristy | b51dff5 | 2011-05-19 16:55:47 +0000 | [diff] [blame] | 546 | (void) FormatLocaleFile(file,"\nPath: %s\n\n",coder_info[i]->path); |
Dirk Lemstra | c1e154d | 2018-10-30 18:30:25 +0100 | [diff] [blame] | 547 | (void) FormatLocaleFile(file,"Magick Coder\n"); |
cristy | 1e60481 | 2011-05-19 18:07:50 +0000 | [diff] [blame] | 548 | (void) FormatLocaleFile(file, |
| 549 | "-------------------------------------------------" |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 550 | "------------------------------\n"); |
| 551 | } |
| 552 | path=coder_info[i]->path; |
cristy | b51dff5 | 2011-05-19 16:55:47 +0000 | [diff] [blame] | 553 | (void) FormatLocaleFile(file,"%s",coder_info[i]->magick); |
Dirk Lemstra | c1e154d | 2018-10-30 18:30:25 +0100 | [diff] [blame] | 554 | for (j=(ssize_t) strlen(coder_info[i]->magick); j <= 15; j++) |
cristy | b51dff5 | 2011-05-19 16:55:47 +0000 | [diff] [blame] | 555 | (void) FormatLocaleFile(file," "); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 556 | if (coder_info[i]->name != (char *) NULL) |
cristy | b51dff5 | 2011-05-19 16:55:47 +0000 | [diff] [blame] | 557 | (void) FormatLocaleFile(file,"%s",coder_info[i]->name); |
| 558 | (void) FormatLocaleFile(file,"\n"); |
cristy | 3ed852e | 2009-09-05 21:47:34 +0000 | [diff] [blame] | 559 | } |
| 560 | coder_info=(const CoderInfo **) RelinquishMagickMemory((void *) coder_info); |
| 561 | (void) fflush(file); |
| 562 | return(MagickTrue); |
| 563 | } |