2 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
6 % CCCC AAA CCCC H H EEEEE %
8 % C AAAAA C HHHHH EEE %
10 % CCCC A A CCCC H H EEEEE %
13 % MagickCore Pixel Cache Methods %
20 % Copyright 1999-2011 ImageMagick Studio LLC, a non-profit organization %
21 % dedicated to making software imaging solutions freely available. %
23 % You may not use this file except in compliance with the License. You may %
24 % obtain a copy of the License at %
26 % http://www.imagemagick.org/script/license.php %
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. %
34 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
43 #include "MagickCore/studio.h"
44 #include "MagickCore/blob.h"
45 #include "MagickCore/blob-private.h"
46 #include "MagickCore/cache.h"
47 #include "MagickCore/cache-private.h"
48 #include "MagickCore/color-private.h"
49 #include "MagickCore/composite-private.h"
50 #include "MagickCore/exception.h"
51 #include "MagickCore/exception-private.h"
52 #include "MagickCore/geometry.h"
53 #include "MagickCore/list.h"
54 #include "MagickCore/log.h"
55 #include "MagickCore/magick.h"
56 #include "MagickCore/memory_.h"
57 #include "MagickCore/pixel.h"
58 #include "MagickCore/pixel-accessor.h"
59 #include "MagickCore/policy.h"
60 #include "MagickCore/quantum.h"
61 #include "MagickCore/random_.h"
62 #include "MagickCore/resource_.h"
63 #include "MagickCore/semaphore.h"
64 #include "MagickCore/splay-tree.h"
65 #include "MagickCore/string_.h"
66 #include "MagickCore/string-private.h"
67 #include "MagickCore/thread-private.h"
68 #include "MagickCore/utility.h"
69 #if defined(MAGICKCORE_ZLIB_DELEGATE)
76 #define CacheTick(offset,extent) QuantumTick((MagickOffsetType) offset,extent)
81 typedef struct _MagickModulo
111 Forward declarations.
113 #if defined(__cplusplus) || defined(c_plusplus)
118 *GetVirtualPixelCache(const Image *,const VirtualPixelMethod,const ssize_t,
119 const ssize_t,const size_t,const size_t,ExceptionInfo *),
120 *GetVirtualPixelsCache(const Image *);
123 *GetVirtualMetacontentFromCache(const Image *);
125 static MagickBooleanType
126 GetOneAuthenticPixelFromCache(Image *,const ssize_t,const ssize_t,
127 PixelPacket *,ExceptionInfo *),
128 GetOneVirtualPixelFromCache(const Image *,const VirtualPixelMethod,
129 const ssize_t,const ssize_t,PixelPacket *,ExceptionInfo *),
130 OpenPixelCache(Image *,const MapMode,ExceptionInfo *),
131 ReadPixelCacheMetacontent(CacheInfo *,NexusInfo *,ExceptionInfo *),
132 ReadPixelCachePixels(CacheInfo *,NexusInfo *,ExceptionInfo *),
133 SyncAuthenticPixelsCache(Image *,ExceptionInfo *),
134 WritePixelCacheMetacontent(CacheInfo *,NexusInfo *,ExceptionInfo *),
135 WritePixelCachePixels(CacheInfo *,NexusInfo *,ExceptionInfo *);
138 *GetAuthenticPixelsCache(Image *,const ssize_t,const ssize_t,const size_t,
139 const size_t,ExceptionInfo *),
140 *QueueAuthenticPixelsCache(Image *,const ssize_t,const ssize_t,const size_t,
141 const size_t,ExceptionInfo *),
142 *SetPixelCacheNexusPixels(const Image *,const RectangleInfo *,NexusInfo *,
145 #if defined(__cplusplus) || defined(c_plusplus)
152 static volatile MagickBooleanType
153 instantiate_cache = MagickFalse;
156 *cache_semaphore = (SemaphoreInfo *) NULL;
159 *cache_resources = (SplayTreeInfo *) NULL;
162 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
166 + A c q u i r e P i x e l C a c h e %
170 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
172 % AcquirePixelCache() acquires a pixel cache.
174 % The format of the AcquirePixelCache() method is:
176 % Cache AcquirePixelCache(const size_t number_threads)
178 % A description of each parameter follows:
180 % o number_threads: the number of nexus threads.
183 MagickExport Cache AcquirePixelCache(const size_t number_threads)
188 cache_info=(CacheInfo *) AcquireAlignedMemory(1,sizeof(*cache_info));
189 if (cache_info == (CacheInfo *) NULL)
190 ThrowFatalException(ResourceLimitFatalError,"MemoryAllocationFailed");
191 (void) ResetMagickMemory(cache_info,0,sizeof(*cache_info));
192 cache_info->type=UndefinedCache;
193 cache_info->mode=IOMode;
194 cache_info->colorspace=RGBColorspace;
195 cache_info->file=(-1);
196 cache_info->id=GetMagickThreadId();
197 cache_info->number_threads=number_threads;
198 if (number_threads == 0)
199 cache_info->number_threads=GetOpenMPMaximumThreads();
200 cache_info->nexus_info=AcquirePixelCacheNexus(cache_info->number_threads);
201 if (cache_info->nexus_info == (NexusInfo **) NULL)
202 ThrowFatalException(ResourceLimitFatalError,"MemoryAllocationFailed");
203 cache_info->semaphore=AllocateSemaphoreInfo();
204 cache_info->reference_count=1;
205 cache_info->disk_semaphore=AllocateSemaphoreInfo();
206 cache_info->debug=IsEventLogging();
207 cache_info->signature=MagickSignature;
208 if ((cache_resources == (SplayTreeInfo *) NULL) &&
209 (instantiate_cache == MagickFalse))
211 if (cache_semaphore == (SemaphoreInfo *) NULL)
212 AcquireSemaphoreInfo(&cache_semaphore);
213 LockSemaphoreInfo(cache_semaphore);
214 if ((cache_resources == (SplayTreeInfo *) NULL) &&
215 (instantiate_cache == MagickFalse))
217 cache_resources=NewSplayTree((int (*)(const void *,const void *))
218 NULL,(void *(*)(void *)) NULL,(void *(*)(void *)) NULL);
219 instantiate_cache=MagickTrue;
221 UnlockSemaphoreInfo(cache_semaphore);
223 (void) AddValueToSplayTree(cache_resources,cache_info,cache_info);
224 return((Cache ) cache_info);
228 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
232 % A c q u i r e P i x e l C a c h e N e x u s %
236 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
238 % AcquirePixelCacheNexus() allocates the NexusInfo structure.
240 % The format of the AcquirePixelCacheNexus method is:
242 % NexusInfo **AcquirePixelCacheNexus(const size_t number_threads)
244 % A description of each parameter follows:
246 % o number_threads: the number of nexus threads.
249 MagickExport NexusInfo **AcquirePixelCacheNexus(const size_t number_threads)
257 nexus_info=(NexusInfo **) AcquireAlignedMemory(number_threads,
258 sizeof(*nexus_info));
259 if (nexus_info == (NexusInfo **) NULL)
260 ThrowFatalException(ResourceLimitFatalError,"MemoryAllocationFailed");
261 for (i=0; i < (ssize_t) number_threads; i++)
263 nexus_info[i]=(NexusInfo *) AcquireAlignedMemory(1,sizeof(**nexus_info));
264 if (nexus_info[i] == (NexusInfo *) NULL)
265 ThrowFatalException(ResourceLimitFatalError,"MemoryAllocationFailed");
266 (void) ResetMagickMemory(nexus_info[i],0,sizeof(*nexus_info[i]));
267 nexus_info[i]->signature=MagickSignature;
273 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
277 + A c q u i r e P i x e l C a c h e P i x e l s %
281 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
283 % AcquirePixelCachePixels() returns the pixels associated with the specified
286 % The format of the AcquirePixelCachePixels() method is:
288 % const void *AcquirePixelCachePixels(const Image *image,
289 % MagickSizeType *length,ExceptionInfo *exception)
291 % A description of each parameter follows:
293 % o image: the image.
295 % o length: the pixel cache length.
297 % o exception: return any errors or warnings in this structure.
300 MagickExport const void *AcquirePixelCachePixels(const Image *image,
301 MagickSizeType *length,ExceptionInfo *exception)
306 assert(image != (const Image *) NULL);
307 assert(image->signature == MagickSignature);
308 assert(exception != (ExceptionInfo *) NULL);
309 assert(exception->signature == MagickSignature);
310 assert(image->cache != (Cache) NULL);
311 cache_info=(CacheInfo *) image->cache;
312 assert(cache_info->signature == MagickSignature);
314 if ((cache_info->type != MemoryCache) && (cache_info->type != MapCache))
315 return((const void *) NULL);
316 *length=cache_info->length;
317 return((const void *) cache_info->pixels);
321 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
325 + C a c h e C o m p o n e n t G e n e s i s %
329 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
331 % CacheComponentGenesis() instantiates the cache component.
333 % The format of the CacheComponentGenesis method is:
335 % MagickBooleanType CacheComponentGenesis(void)
338 MagickExport MagickBooleanType CacheComponentGenesis(void)
340 AcquireSemaphoreInfo(&cache_semaphore);
345 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
349 + C a c h e C o m p o n e n t T e r m i n u s %
353 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
355 % CacheComponentTerminus() destroys the cache component.
357 % The format of the CacheComponentTerminus() method is:
359 % CacheComponentTerminus(void)
362 MagickExport void CacheComponentTerminus(void)
364 if (cache_semaphore == (SemaphoreInfo *) NULL)
365 AcquireSemaphoreInfo(&cache_semaphore);
366 LockSemaphoreInfo(cache_semaphore);
367 if (cache_resources != (SplayTreeInfo *) NULL)
368 cache_resources=DestroySplayTree(cache_resources);
369 instantiate_cache=MagickFalse;
370 UnlockSemaphoreInfo(cache_semaphore);
371 DestroySemaphoreInfo(&cache_semaphore);
375 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
379 + C l i p P i x e l C a c h e N e x u s %
383 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
385 % ClipPixelCacheNexus() clips the cache nexus as defined by the image clip
386 % mask. It returns MagickTrue if the pixel region is clipped, otherwise
389 % The format of the ClipPixelCacheNexus() method is:
391 % MagickBooleanType ClipPixelCacheNexus(Image *image,NexusInfo *nexus_info,
392 % ExceptionInfo *exception)
394 % A description of each parameter follows:
396 % o image: the image.
398 % o nexus_info: the cache nexus to clip.
400 % o exception: return any errors or warnings in this structure.
403 static MagickBooleanType ClipPixelCacheNexus(Image *image,
404 NexusInfo *nexus_info,ExceptionInfo *exception)
416 register const Quantum
429 if (image->debug != MagickFalse)
430 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
431 if (image->clip_mask == (Image *) NULL)
433 cache_info=(CacheInfo *) image->cache;
434 if (cache_info == (Cache) NULL)
436 image_nexus=AcquirePixelCacheNexus(1);
437 clip_nexus=AcquirePixelCacheNexus(1);
438 if ((image_nexus == (NexusInfo **) NULL) ||
439 (clip_nexus == (NexusInfo **) NULL))
440 ThrowBinaryException(CacheError,"UnableToGetCacheNexus",image->filename);
441 p=(const Quantum *) GetAuthenticPixelCacheNexus(image,
442 nexus_info->region.x,nexus_info->region.y,nexus_info->region.width,
443 nexus_info->region.height,image_nexus[0],exception);
444 q=nexus_info->pixels;
445 r=GetVirtualPixelsFromNexus(image->clip_mask,MaskVirtualPixelMethod,
446 nexus_info->region.x,nexus_info->region.y,nexus_info->region.width,
447 nexus_info->region.height,clip_nexus[0],exception);
448 number_pixels=(MagickSizeType) nexus_info->region.width*
449 nexus_info->region.height;
450 for (i=0; i < (ssize_t) number_pixels; i++)
452 if ((p == (const Quantum *) NULL) || (r == (const Quantum *) NULL))
454 if (GetPixelIntensity(image,r) > ((Quantum) QuantumRange/2))
456 SetPixelRed(image,GetPixelRed(image,p),q);
457 SetPixelGreen(image,GetPixelGreen(image,p),q);
458 SetPixelBlue(image,GetPixelBlue(image,p),q);
459 if (cache_info->colorspace == CMYKColorspace)
460 SetPixelBlack(image,GetPixelBlack(image,p),q);
461 SetPixelAlpha(image,GetPixelAlpha(image,p),q);
463 p+=GetPixelChannels(image);
464 q+=GetPixelChannels(image);
465 r+=GetPixelChannels(image->clip_mask);
467 clip_nexus=DestroyPixelCacheNexus(clip_nexus,1);
468 image_nexus=DestroyPixelCacheNexus(image_nexus,1);
469 if (i < (ssize_t) number_pixels)
475 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
479 + C l o n e P i x e l C a c h e %
483 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
485 % ClonePixelCache() clones a pixel cache.
487 % The format of the ClonePixelCache() method is:
489 % Cache ClonePixelCache(const Cache cache)
491 % A description of each parameter follows:
493 % o cache: the pixel cache.
496 MagickExport Cache ClonePixelCache(const Cache cache)
504 assert(cache != (const Cache) NULL);
505 cache_info=(const CacheInfo *) cache;
506 assert(cache_info->signature == MagickSignature);
507 if (cache_info->debug != MagickFalse)
508 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
509 cache_info->filename);
510 clone_info=(CacheInfo *) AcquirePixelCache(cache_info->number_threads);
511 if (clone_info == (Cache) NULL)
512 return((Cache) NULL);
513 clone_info->virtual_pixel_method=cache_info->virtual_pixel_method;
514 return((Cache ) clone_info);
518 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
522 + C l o n e P i x e l C a c h e P i x e l s %
526 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% %
527 % ClonePixelCachePixels() clones the source pixel cache to the destination
530 % The format of the ClonePixelCachePixels() method is:
532 % MagickBooleanType ClonePixelCachePixels(CacheInfo *cache_info,
533 % CacheInfo *source_info,ExceptionInfo *exception)
535 % A description of each parameter follows:
537 % o cache_info: the pixel cache.
539 % o source_info: the source pixel cache.
541 % o exception: return any errors or warnings in this structure.
545 static MagickBooleanType ClosePixelCacheOnDisk(CacheInfo *cache_info)
551 LockSemaphoreInfo(cache_info->disk_semaphore);
552 if (cache_info->file != -1)
554 status=close(cache_info->file);
555 cache_info->file=(-1);
556 RelinquishMagickResource(FileResource,1);
558 UnlockSemaphoreInfo(cache_info->disk_semaphore);
559 return(status == -1 ? MagickFalse : MagickTrue);
562 static void LimitPixelCacheDescriptors(void)
569 Limit # of open file descriptors.
571 if (GetMagickResource(FileResource) < GetMagickResourceLimit(FileResource))
573 LockSemaphoreInfo(cache_semaphore);
574 if (cache_resources == (SplayTreeInfo *) NULL)
576 UnlockSemaphoreInfo(cache_semaphore);
579 ResetSplayTreeIterator(cache_resources);
580 p=(CacheInfo *) GetNextKeyInSplayTree(cache_resources);
581 while (p != (CacheInfo *) NULL)
583 if ((p->type == DiskCache) && (p->file != -1))
585 p=(CacheInfo *) GetNextKeyInSplayTree(cache_resources);
587 for (q=p; p != (CacheInfo *) NULL; )
589 if ((p->type == DiskCache) && (p->file != -1) &&
590 (p->timestamp < q->timestamp))
592 p=(CacheInfo *) GetNextKeyInSplayTree(cache_resources);
594 if (q != (CacheInfo *) NULL)
597 Close least recently used cache.
599 (void) close(q->file);
602 UnlockSemaphoreInfo(cache_semaphore);
605 static inline MagickSizeType MagickMax(const MagickSizeType x,
606 const MagickSizeType y)
613 static inline MagickSizeType MagickMin(const MagickSizeType x,
614 const MagickSizeType y)
621 static MagickBooleanType OpenPixelCacheOnDisk(CacheInfo *cache_info,
628 Open pixel cache on disk.
630 LockSemaphoreInfo(cache_info->disk_semaphore);
631 if (cache_info->file != -1)
633 UnlockSemaphoreInfo(cache_info->disk_semaphore);
634 return(MagickTrue); /* cache already open */
636 LimitPixelCacheDescriptors();
637 if (*cache_info->cache_filename == '\0')
638 file=AcquireUniqueFileResource(cache_info->cache_filename);
644 file=open(cache_info->cache_filename,O_RDONLY | O_BINARY);
649 file=open(cache_info->cache_filename,O_WRONLY | O_CREAT | O_BINARY |
652 file=open(cache_info->cache_filename,O_WRONLY | O_BINARY,S_MODE);
658 file=open(cache_info->cache_filename,O_RDWR | O_CREAT | O_BINARY |
661 file=open(cache_info->cache_filename,O_RDWR | O_BINARY,S_MODE);
667 UnlockSemaphoreInfo(cache_info->disk_semaphore);
670 (void) AcquireMagickResource(FileResource,1);
671 cache_info->file=file;
672 cache_info->timestamp=time(0);
673 UnlockSemaphoreInfo(cache_info->disk_semaphore);
677 static inline MagickOffsetType ReadPixelCacheRegion(CacheInfo *cache_info,
678 const MagickOffsetType offset,const MagickSizeType length,
679 unsigned char *restrict buffer)
681 register MagickOffsetType
687 cache_info->timestamp=time(0);
688 #if !defined(MAGICKCORE_HAVE_PREAD)
689 LockSemaphoreInfo(cache_info->disk_semaphore);
690 if (lseek(cache_info->file,offset,SEEK_SET) < 0)
692 UnlockSemaphoreInfo(cache_info->disk_semaphore);
693 return((MagickOffsetType) -1);
697 for (i=0; i < (MagickOffsetType) length; i+=count)
699 #if !defined(MAGICKCORE_HAVE_PREAD)
700 count=read(cache_info->file,buffer+i,(size_t) MagickMin(length-i,
701 (MagickSizeType) SSIZE_MAX));
703 count=pread(cache_info->file,buffer+i,(size_t) MagickMin(length-i,
704 (MagickSizeType) SSIZE_MAX),(off_t) (offset+i));
715 #if !defined(MAGICKCORE_HAVE_PREAD)
716 UnlockSemaphoreInfo(cache_info->disk_semaphore);
721 static inline MagickOffsetType WritePixelCacheRegion(CacheInfo *cache_info,
722 const MagickOffsetType offset,const MagickSizeType length,
723 const unsigned char *restrict buffer)
725 register MagickOffsetType
731 cache_info->timestamp=time(0);
732 #if !defined(MAGICKCORE_HAVE_PWRITE)
733 LockSemaphoreInfo(cache_info->disk_semaphore);
734 if (lseek(cache_info->file,offset,SEEK_SET) < 0)
736 UnlockSemaphoreInfo(cache_info->disk_semaphore);
737 return((MagickOffsetType) -1);
741 for (i=0; i < (MagickOffsetType) length; i+=count)
743 #if !defined(MAGICKCORE_HAVE_PWRITE)
744 count=write(cache_info->file,buffer+i,(size_t) MagickMin(length-i,
745 (MagickSizeType) SSIZE_MAX));
747 count=pwrite(cache_info->file,buffer+i,(size_t) MagickMin(length-i,
748 (MagickSizeType) SSIZE_MAX),(off_t) (offset+i));
759 #if !defined(MAGICKCORE_HAVE_PWRITE)
760 UnlockSemaphoreInfo(cache_info->disk_semaphore);
765 static MagickBooleanType DiskToDiskPixelCacheClone(CacheInfo *clone_info,
766 CacheInfo *cache_info,ExceptionInfo *exception)
771 register MagickOffsetType
781 Clone pixel cache (both caches on disk).
783 if (cache_info->debug != MagickFalse)
784 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"disk => disk");
785 blob=(unsigned char *) AcquireAlignedMemory(MagickMaxBufferExtent,
787 if (blob == (unsigned char *) NULL)
789 (void) ThrowMagickException(exception,GetMagickModule(),
790 ResourceLimitError,"MemoryAllocationFailed","`%s'",
791 cache_info->filename);
794 if (OpenPixelCacheOnDisk(cache_info,ReadMode) == MagickFalse)
796 blob=(unsigned char *) RelinquishMagickMemory(blob);
797 ThrowFileException(exception,FileOpenError,"UnableToOpenFile",
798 cache_info->cache_filename);
801 if (OpenPixelCacheOnDisk(clone_info,WriteMode) == MagickFalse)
803 (void) ClosePixelCacheOnDisk(cache_info);
804 blob=(unsigned char *) RelinquishMagickMemory(blob);
805 ThrowFileException(exception,FileOpenError,"UnableToOpenFile",
806 clone_info->cache_filename);
810 for (i=0; i < (MagickOffsetType) cache_info->length; i+=count)
812 count=ReadPixelCacheRegion(cache_info,cache_info->offset+i,
813 MagickMin(cache_info->length-i,(MagickSizeType) MagickMaxBufferExtent),
817 ThrowFileException(exception,CacheError,"UnableToReadPixelCache",
818 cache_info->cache_filename);
821 length=(size_t) count;
822 count=WritePixelCacheRegion(clone_info,clone_info->offset+i,length,blob);
823 if ((MagickSizeType) count != length)
825 ThrowFileException(exception,CacheError,"UnableToWritePixelCache",
826 clone_info->cache_filename);
830 (void) ClosePixelCacheOnDisk(clone_info);
831 (void) ClosePixelCacheOnDisk(cache_info);
832 blob=(unsigned char *) RelinquishMagickMemory(blob);
833 if (i < (MagickOffsetType) cache_info->length)
838 static MagickBooleanType OptimizedPixelCacheClone(CacheInfo *clone_info,
839 CacheInfo *cache_info,ExceptionInfo *exception)
844 if ((cache_info->type != DiskCache) && (clone_info->type != DiskCache))
847 Clone pixel cache (both caches in memory).
849 if (cache_info->debug != MagickFalse)
850 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"memory => memory");
851 (void) memcpy(clone_info->pixels,cache_info->pixels,(size_t)
855 if ((clone_info->type != DiskCache) && (cache_info->type == DiskCache))
858 Clone pixel cache (one cache on disk, one in memory).
860 if (cache_info->debug != MagickFalse)
861 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"disk => memory");
862 if (OpenPixelCacheOnDisk(cache_info,ReadMode) == MagickFalse)
864 ThrowFileException(exception,FileOpenError,"UnableToOpenFile",
865 cache_info->cache_filename);
868 count=ReadPixelCacheRegion(cache_info,cache_info->offset,
869 cache_info->length,(unsigned char *) clone_info->pixels);
870 (void) ClosePixelCacheOnDisk(cache_info);
871 if ((MagickSizeType) count != cache_info->length)
873 ThrowFileException(exception,CacheError,"UnableToReadPixelCache",
874 cache_info->cache_filename);
879 if ((clone_info->type == DiskCache) && (cache_info->type != DiskCache))
882 Clone pixel cache (one cache on disk, one in memory).
884 if (clone_info->debug != MagickFalse)
885 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"memory => disk");
886 if (OpenPixelCacheOnDisk(clone_info,WriteMode) == MagickFalse)
888 ThrowFileException(exception,FileOpenError,"UnableToOpenFile",
889 clone_info->cache_filename);
892 count=WritePixelCacheRegion(clone_info,clone_info->offset,
893 clone_info->length,(unsigned char *) cache_info->pixels);
894 (void) ClosePixelCacheOnDisk(clone_info);
895 if ((MagickSizeType) count != clone_info->length)
897 ThrowFileException(exception,CacheError,"UnableToWritePixelCache",
898 clone_info->cache_filename);
904 Clone pixel cache (both caches on disk).
906 return(DiskToDiskPixelCacheClone(clone_info,cache_info,exception));
909 static MagickBooleanType UnoptimizedPixelCacheClone(CacheInfo *clone_info,
910 CacheInfo *cache_info,ExceptionInfo *exception)
933 Clone pixel cache (unoptimized).
935 if (cache_info->debug != MagickFalse)
937 if ((cache_info->type != DiskCache) && (clone_info->type != DiskCache))
938 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"memory => memory");
940 if ((clone_info->type != DiskCache) && (cache_info->type == DiskCache))
941 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"disk => memory");
943 if ((clone_info->type == DiskCache) && (cache_info->type != DiskCache))
944 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"memory => disk");
946 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"disk => disk");
948 length=(size_t) MagickMax(MagickMax(cache_info->number_channels,
949 clone_info->number_channels)*sizeof(Quantum),MagickMax(
950 cache_info->metacontent_extent,clone_info->metacontent_extent));
951 blob=(unsigned char *) AcquireAlignedMemory(length,sizeof(*blob));
952 if (blob == (unsigned char *) NULL)
954 (void) ThrowMagickException(exception,GetMagickModule(),
955 ResourceLimitError,"MemoryAllocationFailed","`%s'",
956 cache_info->filename);
959 (void) ResetMagickMemory(blob,0,length*sizeof(*blob));
962 if (cache_info->type == DiskCache)
964 if (OpenPixelCacheOnDisk(cache_info,ReadMode) == MagickFalse)
966 blob=(unsigned char *) RelinquishMagickMemory(blob);
967 ThrowFileException(exception,FileOpenError,"UnableToOpenFile",
968 cache_info->cache_filename);
971 cache_offset=cache_info->offset;
973 if (clone_info->type == DiskCache)
975 if ((cache_info->type == DiskCache) &&
976 (strcmp(cache_info->cache_filename,clone_info->cache_filename) == 0))
978 (void) ClosePixelCacheOnDisk(clone_info);
979 *clone_info->cache_filename='\0';
981 if (OpenPixelCacheOnDisk(clone_info,WriteMode) == MagickFalse)
983 if (cache_info->type == DiskCache)
984 (void) ClosePixelCacheOnDisk(cache_info);
985 blob=(unsigned char *) RelinquishMagickMemory(blob);
986 ThrowFileException(exception,FileOpenError,"UnableToOpenFile",
987 clone_info->cache_filename);
990 clone_offset=clone_info->offset;
993 Clone pixel channels.
996 for (y=0; y < (ssize_t) cache_info->rows; y++)
998 for (x=0; x < (ssize_t) cache_info->columns; x++)
1001 Read a set of pixel channels.
1003 length=cache_info->number_channels*sizeof(Quantum);
1004 if (cache_info->type != DiskCache)
1005 (void) memcpy(blob,(unsigned char *) cache_info->pixels+cache_offset,
1009 count=ReadPixelCacheRegion(cache_info,cache_offset,length,blob);
1010 if ((MagickSizeType) count != length)
1016 cache_offset+=length;
1017 if ((y < (ssize_t) clone_info->rows) &&
1018 (x < (ssize_t) clone_info->columns))
1021 Write a set of pixel channels.
1023 length=clone_info->number_channels*sizeof(Quantum);
1024 if (clone_info->type != DiskCache)
1025 (void) memcpy((unsigned char *) clone_info->pixels+clone_offset,
1029 count=WritePixelCacheRegion(clone_info,clone_offset,length,
1031 if ((MagickSizeType) count != length)
1037 clone_offset+=length;
1040 length=clone_info->number_channels*sizeof(Quantum);
1041 (void) ResetMagickMemory(blob,0,length*sizeof(*blob));
1042 for ( ; x < (ssize_t) clone_info->columns; x++)
1045 Set remaining columns with transparent pixel channels.
1047 if (clone_info->type != DiskCache)
1048 (void) memcpy((unsigned char *) clone_info->pixels+clone_offset,blob,
1052 count=WritePixelCacheRegion(clone_info,clone_offset,length,blob);
1053 if ((MagickSizeType) count != length)
1059 clone_offset+=length;
1062 length=clone_info->number_channels*sizeof(Quantum);
1063 (void) ResetMagickMemory(blob,0,length*sizeof(*blob));
1064 for ( ; y < (ssize_t) clone_info->rows; y++)
1067 Set remaining rows with transparent pixels.
1069 for (x=0; x < (ssize_t) clone_info->columns; x++)
1071 if (clone_info->type != DiskCache)
1072 (void) memcpy((unsigned char *) clone_info->pixels+clone_offset,blob,
1076 count=WritePixelCacheRegion(clone_info,clone_offset,length,blob);
1077 if ((MagickSizeType) count != length)
1083 clone_offset+=length;
1086 if ((cache_info->metacontent_extent != 0) &&
1087 (clone_info->metacontent_extent != 0))
1092 for (y=0; y < (ssize_t) cache_info->rows; y++)
1094 for (x=0; x < (ssize_t) cache_info->columns; x++)
1097 Read a set of metacontent.
1099 length=cache_info->metacontent_extent;
1100 if (cache_info->type != DiskCache)
1101 (void) memcpy(blob,(unsigned char *) cache_info->pixels+
1102 cache_offset,length);
1105 count=ReadPixelCacheRegion(cache_info,cache_offset,length,blob);
1106 if ((MagickSizeType) count != length)
1112 cache_offset+=length;
1113 if ((y < (ssize_t) clone_info->rows) &&
1114 (x < (ssize_t) clone_info->columns))
1117 Write a set of metacontent.
1119 length=clone_info->metacontent_extent;
1120 if (clone_info->type != DiskCache)
1121 (void) memcpy((unsigned char *) clone_info->pixels+clone_offset,
1125 count=WritePixelCacheRegion(clone_info,clone_offset,length,
1127 if ((MagickSizeType) count != length)
1133 clone_offset+=length;
1136 length=clone_info->metacontent_extent;
1137 (void) ResetMagickMemory(blob,0,length*sizeof(*blob));
1138 for ( ; x < (ssize_t) clone_info->columns; x++)
1141 Set remaining columns with metacontent.
1143 if (clone_info->type != DiskCache)
1144 (void) memcpy((unsigned char *) clone_info->pixels+clone_offset,
1148 count=WritePixelCacheRegion(clone_info,clone_offset,length,
1150 if ((MagickSizeType) count != length)
1156 clone_offset+=length;
1159 length=clone_info->metacontent_extent;
1160 (void) ResetMagickMemory(blob,0,length*sizeof(*blob));
1161 for ( ; y < (ssize_t) clone_info->rows; y++)
1164 Set remaining rows with metacontent.
1166 for (x=0; x < (ssize_t) clone_info->columns; x++)
1168 if (clone_info->type != DiskCache)
1169 (void) memcpy((unsigned char *) clone_info->pixels+clone_offset,
1173 count=WritePixelCacheRegion(clone_info,clone_offset,length,blob);
1174 if ((MagickSizeType) count != length)
1180 clone_offset+=length;
1184 if (clone_info->type == DiskCache)
1185 (void) ClosePixelCacheOnDisk(clone_info);
1186 if (cache_info->type == DiskCache)
1187 (void) ClosePixelCacheOnDisk(cache_info);
1188 blob=(unsigned char *) RelinquishMagickMemory(blob);
1192 static MagickBooleanType ClonePixelCachePixels(CacheInfo *clone_info,
1193 CacheInfo *cache_info,ExceptionInfo *exception)
1195 if (cache_info->type == PingCache)
1197 if ((cache_info->columns == clone_info->columns) &&
1198 (cache_info->rows == clone_info->rows) &&
1199 (cache_info->number_channels == clone_info->number_channels) &&
1200 (cache_info->metacontent_extent == clone_info->metacontent_extent))
1201 return(OptimizedPixelCacheClone(clone_info,cache_info,exception));
1202 return(UnoptimizedPixelCacheClone(clone_info,cache_info,exception));
1206 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1210 + C l o n e P i x e l C a c h e M e t h o d s %
1214 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1216 % ClonePixelCacheMethods() clones the pixel cache methods from one cache to
1219 % The format of the ClonePixelCacheMethods() method is:
1221 % void ClonePixelCacheMethods(Cache clone,const Cache cache)
1223 % A description of each parameter follows:
1225 % o clone: Specifies a pointer to a Cache structure.
1227 % o cache: the pixel cache.
1230 MagickExport void ClonePixelCacheMethods(Cache clone,const Cache cache)
1236 assert(clone != (Cache) NULL);
1237 source_info=(CacheInfo *) clone;
1238 assert(source_info->signature == MagickSignature);
1239 if (source_info->debug != MagickFalse)
1240 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
1241 source_info->filename);
1242 assert(cache != (Cache) NULL);
1243 cache_info=(CacheInfo *) cache;
1244 assert(cache_info->signature == MagickSignature);
1245 source_info->methods=cache_info->methods;
1249 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1253 + D e s t r o y I m a g e P i x e l C a c h e %
1257 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1259 % DestroyImagePixelCache() deallocates memory associated with the pixel cache.
1261 % The format of the DestroyImagePixelCache() method is:
1263 % void DestroyImagePixelCache(Image *image)
1265 % A description of each parameter follows:
1267 % o image: the image.
1270 static void DestroyImagePixelCache(Image *image)
1272 assert(image != (Image *) NULL);
1273 assert(image->signature == MagickSignature);
1274 if (image->debug != MagickFalse)
1275 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
1276 if (image->cache == (void *) NULL)
1278 image->cache=DestroyPixelCache(image->cache);
1282 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1286 + D e s t r o y I m a g e P i x e l s %
1290 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1292 % DestroyImagePixels() deallocates memory associated with the pixel cache.
1294 % The format of the DestroyImagePixels() method is:
1296 % void DestroyImagePixels(Image *image)
1298 % A description of each parameter follows:
1300 % o image: the image.
1303 MagickExport void DestroyImagePixels(Image *image)
1308 assert(image != (const Image *) NULL);
1309 assert(image->signature == MagickSignature);
1310 if (image->debug != MagickFalse)
1311 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
1312 assert(image->cache != (Cache) NULL);
1313 cache_info=(CacheInfo *) image->cache;
1314 assert(cache_info->signature == MagickSignature);
1315 if (cache_info->methods.destroy_pixel_handler != (DestroyPixelHandler) NULL)
1317 cache_info->methods.destroy_pixel_handler(image);
1320 image->cache=DestroyPixelCache(image->cache);
1324 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1328 + D e s t r o y P i x e l C a c h e %
1332 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1334 % DestroyPixelCache() deallocates memory associated with the pixel cache.
1336 % The format of the DestroyPixelCache() method is:
1338 % Cache DestroyPixelCache(Cache cache)
1340 % A description of each parameter follows:
1342 % o cache: the pixel cache.
1346 static inline void RelinquishPixelCachePixels(CacheInfo *cache_info)
1348 switch (cache_info->type)
1352 if (cache_info->mapped == MagickFalse)
1353 cache_info->pixels=(Quantum *) RelinquishMagickMemory(
1354 cache_info->pixels);
1356 cache_info->pixels=(Quantum *) UnmapBlob(cache_info->pixels,
1357 (size_t) cache_info->length);
1358 RelinquishMagickResource(MemoryResource,cache_info->length);
1363 cache_info->pixels=(Quantum *) UnmapBlob(cache_info->pixels,(size_t)
1364 cache_info->length);
1365 RelinquishMagickResource(MapResource,cache_info->length);
1369 if (cache_info->file != -1)
1370 (void) ClosePixelCacheOnDisk(cache_info);
1371 RelinquishMagickResource(DiskResource,cache_info->length);
1377 cache_info->type=UndefinedCache;
1378 cache_info->mapped=MagickFalse;
1379 cache_info->metacontent=(void *) NULL;
1382 MagickExport Cache DestroyPixelCache(Cache cache)
1387 assert(cache != (Cache) NULL);
1388 cache_info=(CacheInfo *) cache;
1389 assert(cache_info->signature == MagickSignature);
1390 if (cache_info->debug != MagickFalse)
1391 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
1392 cache_info->filename);
1393 LockSemaphoreInfo(cache_info->semaphore);
1394 cache_info->reference_count--;
1395 if (cache_info->reference_count != 0)
1397 UnlockSemaphoreInfo(cache_info->semaphore);
1398 return((Cache) NULL);
1400 UnlockSemaphoreInfo(cache_info->semaphore);
1401 if (cache_resources != (SplayTreeInfo *) NULL)
1402 (void) DeleteNodeByValueFromSplayTree(cache_resources,cache_info);
1403 if (cache_info->debug != MagickFalse)
1406 message[MaxTextExtent];
1408 (void) FormatLocaleString(message,MaxTextExtent,"destroy %s",
1409 cache_info->filename);
1410 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"%s",message);
1412 if ((cache_info->mode == ReadMode) || ((cache_info->type != MapCache) &&
1413 (cache_info->type != DiskCache)))
1414 RelinquishPixelCachePixels(cache_info);
1417 RelinquishPixelCachePixels(cache_info);
1418 (void) RelinquishUniqueFileResource(cache_info->cache_filename);
1420 *cache_info->cache_filename='\0';
1421 if (cache_info->nexus_info != (NexusInfo **) NULL)
1422 cache_info->nexus_info=DestroyPixelCacheNexus(cache_info->nexus_info,
1423 cache_info->number_threads);
1424 if (cache_info->random_info != (RandomInfo *) NULL)
1425 cache_info->random_info=DestroyRandomInfo(cache_info->random_info);
1426 if (cache_info->disk_semaphore != (SemaphoreInfo *) NULL)
1427 DestroySemaphoreInfo(&cache_info->disk_semaphore);
1428 if (cache_info->semaphore != (SemaphoreInfo *) NULL)
1429 DestroySemaphoreInfo(&cache_info->semaphore);
1430 cache_info->signature=(~MagickSignature);
1431 cache_info=(CacheInfo *) RelinquishMagickMemory(cache_info);
1437 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1441 + D e s t r o y P i x e l C a c h e N e x u s %
1445 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1447 % DestroyPixelCacheNexus() destroys a pixel cache nexus.
1449 % The format of the DestroyPixelCacheNexus() method is:
1451 % NexusInfo **DestroyPixelCacheNexus(NexusInfo *nexus_info,
1452 % const size_t number_threads)
1454 % A description of each parameter follows:
1456 % o nexus_info: the nexus to destroy.
1458 % o number_threads: the number of nexus threads.
1462 static inline void RelinquishCacheNexusPixels(NexusInfo *nexus_info)
1464 if (nexus_info->mapped == MagickFalse)
1465 (void) RelinquishMagickMemory(nexus_info->cache);
1467 (void) UnmapBlob(nexus_info->cache,(size_t) nexus_info->length);
1468 nexus_info->cache=(Quantum *) NULL;
1469 nexus_info->pixels=(Quantum *) NULL;
1470 nexus_info->metacontent=(void *) NULL;
1471 nexus_info->length=0;
1472 nexus_info->mapped=MagickFalse;
1475 MagickExport NexusInfo **DestroyPixelCacheNexus(NexusInfo **nexus_info,
1476 const size_t number_threads)
1481 assert(nexus_info != (NexusInfo **) NULL);
1482 for (i=0; i < (ssize_t) number_threads; i++)
1484 if (nexus_info[i]->cache != (Quantum *) NULL)
1485 RelinquishCacheNexusPixels(nexus_info[i]);
1486 nexus_info[i]->signature=(~MagickSignature);
1487 nexus_info[i]=(NexusInfo *) RelinquishMagickMemory(nexus_info[i]);
1489 nexus_info=(NexusInfo **) RelinquishMagickMemory(nexus_info);
1494 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1498 % G e t A u t h e n t i c M e t a c o n t e n t %
1502 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1504 % GetAuthenticMetacontent() returns the authentic metacontent corresponding
1505 % with the last call to QueueAuthenticPixels() or GetVirtualPixels(). NULL is
1506 % returned if the associated pixels are not available.
1508 % The format of the GetAuthenticMetacontent() method is:
1510 % void *GetAuthenticMetacontent(const Image *image)
1512 % A description of each parameter follows:
1514 % o image: the image.
1517 MagickExport void *GetAuthenticMetacontent(const Image *image)
1523 id = GetOpenMPThreadId();
1528 assert(image != (const Image *) NULL);
1529 assert(image->signature == MagickSignature);
1530 assert(image->cache != (Cache) NULL);
1531 cache_info=(CacheInfo *) image->cache;
1532 assert(cache_info->signature == MagickSignature);
1533 if (cache_info->methods.get_authentic_metacontent_from_handler !=
1534 (GetAuthenticMetacontentFromHandler) NULL)
1536 metacontent=cache_info->methods.
1537 get_authentic_metacontent_from_handler(image);
1538 return(metacontent);
1540 assert(id < (int) cache_info->number_threads);
1541 metacontent=GetPixelCacheNexusMetacontent(cache_info,
1542 cache_info->nexus_info[id]);
1543 return(metacontent);
1547 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1551 + G e t A u t h e n t i c M e t a c o n t e n t F r o m C a c h e %
1555 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1557 % GetAuthenticMetacontentFromCache() returns the meta-content corresponding
1558 % with the last call to QueueAuthenticPixelsCache() or
1559 % GetAuthenticPixelsCache().
1561 % The format of the GetAuthenticMetacontentFromCache() method is:
1563 % void *GetAuthenticMetacontentFromCache(const Image *image)
1565 % A description of each parameter follows:
1567 % o image: the image.
1570 static void *GetAuthenticMetacontentFromCache(const Image *image)
1576 id = GetOpenMPThreadId();
1581 assert(image != (const Image *) NULL);
1582 assert(image->signature == MagickSignature);
1583 assert(image->cache != (Cache) NULL);
1584 cache_info=(CacheInfo *) image->cache;
1585 assert(cache_info->signature == MagickSignature);
1586 assert(id < (int) cache_info->number_threads);
1587 metacontent=GetPixelCacheNexusMetacontent(image->cache,
1588 cache_info->nexus_info[id]);
1589 return(metacontent);
1593 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1597 + G e t A u t h e n t i c P i x e l C a c h e N e x u s %
1601 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1603 % GetAuthenticPixelCacheNexus() gets authentic pixels from the in-memory or
1604 % disk pixel cache as defined by the geometry parameters. A pointer to the
1605 % pixels is returned if the pixels are transferred, otherwise a NULL is
1608 % The format of the GetAuthenticPixelCacheNexus() method is:
1610 % Quantum *GetAuthenticPixelCacheNexus(Image *image,const ssize_t x,
1611 % const ssize_t y,const size_t columns,const size_t rows,
1612 % NexusInfo *nexus_info,ExceptionInfo *exception)
1614 % A description of each parameter follows:
1616 % o image: the image.
1618 % o x,y,columns,rows: These values define the perimeter of a region of
1621 % o nexus_info: the cache nexus to return.
1623 % o exception: return any errors or warnings in this structure.
1627 static inline MagickBooleanType IsPixelAuthentic(const CacheInfo *cache_info,
1628 NexusInfo *nexus_info)
1636 if (cache_info->type == PingCache)
1638 offset=(MagickOffsetType) nexus_info->region.y*cache_info->columns+
1639 nexus_info->region.x;
1640 status=nexus_info->pixels == (cache_info->pixels+offset*
1641 cache_info->number_channels) ? MagickTrue : MagickFalse;
1645 MagickExport Quantum *GetAuthenticPixelCacheNexus(Image *image,
1646 const ssize_t x,const ssize_t y,const size_t columns,const size_t rows,
1647 NexusInfo *nexus_info,ExceptionInfo *exception)
1656 Transfer pixels from the cache.
1658 assert(image != (Image *) NULL);
1659 assert(image->signature == MagickSignature);
1660 pixels=QueueAuthenticNexus(image,x,y,columns,rows,nexus_info,exception);
1661 if (pixels == (Quantum *) NULL)
1662 return((Quantum *) NULL);
1663 cache_info=(CacheInfo *) image->cache;
1664 assert(cache_info->signature == MagickSignature);
1665 if (IsPixelAuthentic(cache_info,nexus_info) != MagickFalse)
1667 if (ReadPixelCachePixels(cache_info,nexus_info,exception) == MagickFalse)
1668 return((Quantum *) NULL);
1669 if (cache_info->metacontent_extent != 0)
1670 if (ReadPixelCacheMetacontent(cache_info,nexus_info,exception) == MagickFalse)
1671 return((Quantum *) NULL);
1676 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1680 + G e t A u t h e n t i c P i x e l s F r o m C a c h e %
1684 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1686 % GetAuthenticPixelsFromCache() returns the pixels associated with the last
1687 % call to the QueueAuthenticPixelsCache() or GetAuthenticPixelsCache() methods.
1689 % The format of the GetAuthenticPixelsFromCache() method is:
1691 % Quantum *GetAuthenticPixelsFromCache(const Image image)
1693 % A description of each parameter follows:
1695 % o image: the image.
1698 static Quantum *GetAuthenticPixelsFromCache(const Image *image)
1704 id = GetOpenMPThreadId();
1706 assert(image != (const Image *) NULL);
1707 assert(image->signature == MagickSignature);
1708 assert(image->cache != (Cache) NULL);
1709 cache_info=(CacheInfo *) image->cache;
1710 assert(cache_info->signature == MagickSignature);
1711 assert(id < (int) cache_info->number_threads);
1712 return(GetPixelCacheNexusPixels(image->cache,cache_info->nexus_info[id]));
1716 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1720 % G e t A u t h e n t i c P i x e l Q u e u e %
1724 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1726 % GetAuthenticPixelQueue() returns the authentic pixels associated
1727 % corresponding with the last call to QueueAuthenticPixels() or
1728 % GetAuthenticPixels().
1730 % The format of the GetAuthenticPixelQueue() method is:
1732 % Quantum *GetAuthenticPixelQueue(const Image image)
1734 % A description of each parameter follows:
1736 % o image: the image.
1739 MagickExport Quantum *GetAuthenticPixelQueue(const Image *image)
1745 id = GetOpenMPThreadId();
1747 assert(image != (const Image *) NULL);
1748 assert(image->signature == MagickSignature);
1749 assert(image->cache != (Cache) NULL);
1750 cache_info=(CacheInfo *) image->cache;
1751 assert(cache_info->signature == MagickSignature);
1752 if (cache_info->methods.get_authentic_pixels_from_handler !=
1753 (GetAuthenticPixelsFromHandler) NULL)
1754 return(cache_info->methods.get_authentic_pixels_from_handler(image));
1755 assert(id < (int) cache_info->number_threads);
1756 return(GetPixelCacheNexusPixels(cache_info,cache_info->nexus_info[id]));
1760 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1764 % G e t A u t h e n t i c P i x e l s %
1767 % % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1769 % GetAuthenticPixels() obtains a pixel region for read/write access. If the
1770 % region is successfully accessed, a pointer to a Quantum array
1771 % representing the region is returned, otherwise NULL is returned.
1773 % The returned pointer may point to a temporary working copy of the pixels
1774 % or it may point to the original pixels in memory. Performance is maximized
1775 % if the selected region is part of one row, or one or more full rows, since
1776 % then there is opportunity to access the pixels in-place (without a copy)
1777 % if the image is in memory, or in a memory-mapped file. The returned pointer
1778 % must *never* be deallocated by the user.
1780 % Pixels accessed via the returned pointer represent a simple array of type
1781 % Quantum. If the image has corresponding metacontent,call
1782 % GetAuthenticMetacontent() after invoking GetAuthenticPixels() to obtain the
1783 % meta-content corresponding to the region. Once the Quantum array has
1784 % been updated, the changes must be saved back to the underlying image using
1785 % SyncAuthenticPixels() or they may be lost.
1787 % The format of the GetAuthenticPixels() method is:
1789 % Quantum *GetAuthenticPixels(Image *image,const ssize_t x,
1790 % const ssize_t y,const size_t columns,const size_t rows,
1791 % ExceptionInfo *exception)
1793 % A description of each parameter follows:
1795 % o image: the image.
1797 % o x,y,columns,rows: These values define the perimeter of a region of
1800 % o exception: return any errors or warnings in this structure.
1803 MagickExport Quantum *GetAuthenticPixels(Image *image,const ssize_t x,
1804 const ssize_t y,const size_t columns,const size_t rows,
1805 ExceptionInfo *exception)
1811 id = GetOpenMPThreadId();
1816 assert(image != (Image *) NULL);
1817 assert(image->signature == MagickSignature);
1818 assert(image->cache != (Cache) NULL);
1819 cache_info=(CacheInfo *) image->cache;
1820 assert(cache_info->signature == MagickSignature);
1821 if (cache_info->methods.get_authentic_pixels_handler !=
1822 (GetAuthenticPixelsHandler) NULL)
1824 pixels=cache_info->methods.get_authentic_pixels_handler(image,x,y,columns,
1828 assert(id < (int) cache_info->number_threads);
1829 pixels=GetAuthenticPixelCacheNexus(image,x,y,columns,rows,
1830 cache_info->nexus_info[id],exception);
1835 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1839 + G e t A u t h e n t i c P i x e l s C a c h e %
1843 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1845 % GetAuthenticPixelsCache() gets pixels from the in-memory or disk pixel cache
1846 % as defined by the geometry parameters. A pointer to the pixels is returned
1847 % if the pixels are transferred, otherwise a NULL is returned.
1849 % The format of the GetAuthenticPixelsCache() method is:
1851 % Quantum *GetAuthenticPixelsCache(Image *image,const ssize_t x,
1852 % const ssize_t y,const size_t columns,const size_t rows,
1853 % ExceptionInfo *exception)
1855 % A description of each parameter follows:
1857 % o image: the image.
1859 % o x,y,columns,rows: These values define the perimeter of a region of
1862 % o exception: return any errors or warnings in this structure.
1865 static Quantum *GetAuthenticPixelsCache(Image *image,const ssize_t x,
1866 const ssize_t y,const size_t columns,const size_t rows,
1867 ExceptionInfo *exception)
1873 id = GetOpenMPThreadId();
1878 assert(image != (const Image *) NULL);
1879 assert(image->signature == MagickSignature);
1880 assert(image->cache != (Cache) NULL);
1881 cache_info=(CacheInfo *) image->cache;
1882 if (cache_info == (Cache) NULL)
1883 return((Quantum *) NULL);
1884 assert(cache_info->signature == MagickSignature);
1885 assert(id < (int) cache_info->number_threads);
1886 pixels=GetAuthenticPixelCacheNexus(image,x,y,columns,rows,
1887 cache_info->nexus_info[id],exception);
1892 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1896 + G e t I m a g e E x t e n t %
1900 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1902 % GetImageExtent() returns the extent of the pixels associated corresponding
1903 % with the last call to QueueAuthenticPixels() or GetAuthenticPixels().
1905 % The format of the GetImageExtent() method is:
1907 % MagickSizeType GetImageExtent(const Image *image)
1909 % A description of each parameter follows:
1911 % o image: the image.
1914 MagickExport MagickSizeType GetImageExtent(const Image *image)
1920 id = GetOpenMPThreadId();
1922 assert(image != (Image *) NULL);
1923 assert(image->signature == MagickSignature);
1924 if (image->debug != MagickFalse)
1925 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
1926 assert(image->cache != (Cache) NULL);
1927 cache_info=(CacheInfo *) image->cache;
1928 assert(cache_info->signature == MagickSignature);
1929 assert(id < (int) cache_info->number_threads);
1930 return(GetPixelCacheNexusExtent(cache_info,cache_info->nexus_info[id]));
1934 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1938 + G e t I m a g e P i x e l C a c h e %
1942 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1944 % GetImagePixelCache() ensures that there is only a single reference to the
1945 % pixel cache to be modified, updating the provided cache pointer to point to
1946 % a clone of the original pixel cache if necessary.
1948 % The format of the GetImagePixelCache method is:
1950 % Cache GetImagePixelCache(Image *image,const MagickBooleanType clone,
1951 % ExceptionInfo *exception)
1953 % A description of each parameter follows:
1955 % o image: the image.
1957 % o clone: any value other than MagickFalse clones the cache pixels.
1959 % o exception: return any errors or warnings in this structure.
1962 static inline MagickBooleanType ValidatePixelCacheMorphology(const Image *image)
1968 Does the image match the pixel cache morphology?
1970 cache_info=(CacheInfo *) image->cache;
1971 if ((image->storage_class != cache_info->storage_class) ||
1972 (image->colorspace != cache_info->colorspace) ||
1973 (image->matte != cache_info->matte) ||
1974 (image->columns != cache_info->columns) ||
1975 (image->rows != cache_info->rows) ||
1976 (image->number_channels != cache_info->number_channels) ||
1977 (image->metacontent_extent != cache_info->metacontent_extent) ||
1978 (cache_info->nexus_info == (NexusInfo **) NULL) ||
1979 (cache_info->number_threads < GetOpenMPMaximumThreads()))
1980 return(MagickFalse);
1984 static Cache GetImagePixelCache(Image *image,const MagickBooleanType clone,
1985 ExceptionInfo *exception)
1994 static MagickSizeType
2003 LockSemaphoreInfo(image->semaphore);
2004 if (cpu_throttle == 0)
2010 Set CPU throttle in milleseconds.
2012 cpu_throttle=MagickResourceInfinity;
2013 limit=GetEnvironmentValue("MAGICK_THROTTLE");
2014 if (limit == (char *) NULL)
2015 limit=GetPolicyValue("throttle");
2016 if (limit != (char *) NULL)
2018 cpu_throttle=(MagickSizeType) StringToInteger(limit);
2019 limit=DestroyString(limit);
2022 if ((cpu_throttle != MagickResourceInfinity) && ((cycles++ % 32) == 0))
2023 MagickDelay(cpu_throttle);
2024 if (time_limit == 0)
2027 Set the exire time in seconds.
2029 time_limit=GetMagickResourceLimit(TimeResource);
2030 cache_genesis=time((time_t *) NULL);
2032 if ((time_limit != MagickResourceInfinity) &&
2033 ((MagickSizeType) (time((time_t *) NULL)-cache_genesis) >= time_limit))
2034 ThrowFatalException(ResourceLimitFatalError,"TimeLimitExceeded");
2035 assert(image->cache != (Cache) NULL);
2036 cache_info=(CacheInfo *) image->cache;
2037 destroy=MagickFalse;
2038 if ((cache_info->reference_count > 1) || (cache_info->mode == ReadMode))
2040 LockSemaphoreInfo(cache_info->semaphore);
2041 if ((cache_info->reference_count > 1) || (cache_info->mode == ReadMode))
2052 clone_image=(*image);
2053 clone_image.semaphore=AllocateSemaphoreInfo();
2054 clone_image.reference_count=1;
2055 clone_image.cache=ClonePixelCache(cache_info);
2056 clone_info=(CacheInfo *) clone_image.cache;
2057 status=OpenPixelCache(&clone_image,IOMode,exception);
2058 if (status != MagickFalse)
2060 if (clone != MagickFalse)
2061 status=ClonePixelCachePixels(clone_info,cache_info,exception);
2062 if (status != MagickFalse)
2065 image->cache=clone_image.cache;
2068 DestroySemaphoreInfo(&clone_image.semaphore);
2070 UnlockSemaphoreInfo(cache_info->semaphore);
2072 if (destroy != MagickFalse)
2073 cache_info=(CacheInfo *) DestroyPixelCache(cache_info);
2074 if (status != MagickFalse)
2077 Ensure the image matches the pixel cache morphology.
2079 image->taint=MagickTrue;
2080 image->type=UndefinedType;
2081 if (1 && image->colorspace == GRAYColorspace)
2082 image->colorspace=RGBColorspace;
2083 if (ValidatePixelCacheMorphology(image) == MagickFalse)
2084 status=OpenPixelCache(image,IOMode,exception);
2086 UnlockSemaphoreInfo(image->semaphore);
2087 if (status == MagickFalse)
2088 return((Cache) NULL);
2089 return(image->cache);
2093 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2097 % G e t O n e A u t h e n t i c P i x e l %
2101 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2103 % GetOneAuthenticPixel() returns a single pixel at the specified (x,y)
2104 % location. The image background color is returned if an error occurs.
2106 % The format of the GetOneAuthenticPixel() method is:
2108 % MagickBooleanType GetOneAuthenticPixel(const Image image,const ssize_t x,
2109 % const ssize_t y,PixelPacket *pixel,ExceptionInfo *exception)
2111 % A description of each parameter follows:
2113 % o image: the image.
2115 % o x,y: These values define the location of the pixel to return.
2117 % o pixel: return a pixel at the specified (x,y) location.
2119 % o exception: return any errors or warnings in this structure.
2122 MagickExport MagickBooleanType GetOneAuthenticPixel(Image *image,
2123 const ssize_t x,const ssize_t y,PixelPacket *pixel,ExceptionInfo *exception)
2131 assert(image != (Image *) NULL);
2132 assert(image->signature == MagickSignature);
2133 assert(image->cache != (Cache) NULL);
2134 cache_info=(CacheInfo *) image->cache;
2135 assert(cache_info->signature == MagickSignature);
2136 *pixel=image->background_color;
2137 if (cache_info->methods.get_one_authentic_pixel_from_handler !=
2138 (GetOneAuthenticPixelFromHandler) NULL)
2139 return(cache_info->methods.get_one_authentic_pixel_from_handler(image,x,y,
2141 q=GetAuthenticPixelsCache(image,x,y,1UL,1UL,exception);
2142 if (q == (Quantum *) NULL)
2143 return(MagickFalse);
2144 GetPixelPacket(image,q,pixel);
2149 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2153 + G e t O n e A u t h e n t i c P i x e l F r o m C a c h e %
2157 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2159 % GetOneAuthenticPixelFromCache() returns a single pixel at the specified (x,y)
2160 % location. The image background color is returned if an error occurs.
2162 % The format of the GetOneAuthenticPixelFromCache() method is:
2164 % MagickBooleanType GetOneAuthenticPixelFromCache(const Image image,
2165 % const ssize_t x,const ssize_t y,PixelPacket *pixel,
2166 % ExceptionInfo *exception)
2168 % A description of each parameter follows:
2170 % o image: the image.
2172 % o x,y: These values define the location of the pixel to return.
2174 % o pixel: return a pixel at the specified (x,y) location.
2176 % o exception: return any errors or warnings in this structure.
2179 static MagickBooleanType GetOneAuthenticPixelFromCache(Image *image,
2180 const ssize_t x,const ssize_t y,PixelPacket *pixel,ExceptionInfo *exception)
2186 id = GetOpenMPThreadId();
2191 assert(image != (const Image *) NULL);
2192 assert(image->signature == MagickSignature);
2193 assert(image->cache != (Cache) NULL);
2194 cache_info=(CacheInfo *) image->cache;
2195 assert(cache_info->signature == MagickSignature);
2196 assert(id < (int) cache_info->number_threads);
2197 *pixel=image->background_color;
2198 q=GetAuthenticPixelCacheNexus(image,x,y,1UL,1UL,cache_info->nexus_info[id],
2200 if (q == (Quantum *) NULL)
2201 return(MagickFalse);
2202 GetPixelPacket(image,q,pixel);
2207 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2211 % G e t O n e V i r t u a l M a g i c k P i x e l %
2215 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2217 % GetOneVirtualMagickPixel() returns a single pixel at the specified (x,y)
2218 % location. The image background color is returned if an error occurs. If
2219 % you plan to modify the pixel, use GetOneAuthenticPixel() instead.
2221 % The format of the GetOneVirtualMagickPixel() method is:
2223 % MagickBooleanType GetOneVirtualMagickPixel(const Image image,
2224 % const ssize_t x,const ssize_t y,PixelInfo *pixel,
2225 % ExceptionInfo exception)
2227 % A description of each parameter follows:
2229 % o image: the image.
2231 % o x,y: these values define the location of the pixel to return.
2233 % o pixel: return a pixel at the specified (x,y) location.
2235 % o exception: return any errors or warnings in this structure.
2238 MagickExport MagickBooleanType GetOneVirtualMagickPixel(const Image *image,
2239 const ssize_t x,const ssize_t y,PixelInfo *pixel,
2240 ExceptionInfo *exception)
2246 id = GetOpenMPThreadId();
2248 register const Quantum
2251 assert(image != (const Image *) NULL);
2252 assert(image->signature == MagickSignature);
2253 assert(image->cache != (Cache) NULL);
2254 cache_info=(CacheInfo *) image->cache;
2255 assert(cache_info->signature == MagickSignature);
2256 assert(id < (int) cache_info->number_threads);
2257 pixels=GetVirtualPixelsFromNexus(image,GetPixelCacheVirtualMethod(image),x,y,
2258 1UL,1UL,cache_info->nexus_info[id],exception);
2259 GetPixelInfo(image,pixel);
2260 if (pixels == (const Quantum *) NULL)
2261 return(MagickFalse);
2262 SetPixelInfo(image,pixels,pixel);
2267 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2271 % G e t O n e V i r t u a l M e t h o d P i x e l %
2275 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2277 % GetOneVirtualMethodPixel() returns a single pixel at the specified (x,y)
2278 % location as defined by specified pixel method. The image background color
2279 % is returned if an error occurs. If you plan to modify the pixel, use
2280 % GetOneAuthenticPixel() instead.
2282 % The format of the GetOneVirtualMethodPixel() method is:
2284 % MagickBooleanType GetOneVirtualMethodPixel(const Image image,
2285 % const VirtualPixelMethod virtual_pixel_method,const ssize_t x,
2286 % const ssize_t y,PixelPacket *pixel,ExceptionInfo exception)
2288 % A description of each parameter follows:
2290 % o image: the image.
2292 % o virtual_pixel_method: the virtual pixel method.
2294 % o x,y: These values define the location of the pixel to return.
2296 % o pixel: return a pixel at the specified (x,y) location.
2298 % o exception: return any errors or warnings in this structure.
2301 MagickExport MagickBooleanType GetOneVirtualMethodPixel(const Image *image,
2302 const VirtualPixelMethod virtual_pixel_method,const ssize_t x,const ssize_t y,
2303 PixelPacket *pixel,ExceptionInfo *exception)
2309 id = GetOpenMPThreadId();
2314 assert(image != (const Image *) NULL);
2315 assert(image->signature == MagickSignature);
2316 assert(image->cache != (Cache) NULL);
2317 cache_info=(CacheInfo *) image->cache;
2318 assert(cache_info->signature == MagickSignature);
2319 *pixel=image->background_color;
2320 if (cache_info->methods.get_one_virtual_pixel_from_handler !=
2321 (GetOneVirtualPixelFromHandler) NULL)
2322 return(cache_info->methods.get_one_virtual_pixel_from_handler(image,
2323 virtual_pixel_method,x,y,pixel,exception));
2324 assert(id < (int) cache_info->number_threads);
2325 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,x,y,1UL,1UL,
2326 cache_info->nexus_info[id],exception);
2327 if (p == (const Quantum *) NULL)
2328 return(MagickFalse);
2329 GetPixelPacket(image,p,pixel);
2330 if (image->colorspace == CMYKColorspace)
2331 pixel->black=GetPixelBlack(image,p);
2336 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2340 % G e t O n e V i r t u a l P i x e l %
2344 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2346 % GetOneVirtualPixel() returns a single virtual pixel at the specified
2347 % (x,y) location. The image background color is returned if an error occurs.
2348 % If you plan to modify the pixel, use GetOneAuthenticPixel() instead.
2350 % The format of the GetOneVirtualPixel() method is:
2352 % MagickBooleanType GetOneVirtualPixel(const Image image,const ssize_t x,
2353 % const ssize_t y,PixelPacket *pixel,ExceptionInfo exception)
2355 % A description of each parameter follows:
2357 % o image: the image.
2359 % o x,y: These values define the location of the pixel to return.
2361 % o pixel: return a pixel at the specified (x,y) location.
2363 % o exception: return any errors or warnings in this structure.
2366 MagickExport MagickBooleanType GetOneVirtualPixel(const Image *image,
2367 const ssize_t x,const ssize_t y,PixelPacket *pixel,ExceptionInfo *exception)
2373 id = GetOpenMPThreadId();
2378 assert(image != (const Image *) NULL);
2379 assert(image->signature == MagickSignature);
2380 assert(image->cache != (Cache) NULL);
2381 cache_info=(CacheInfo *) image->cache;
2382 assert(cache_info->signature == MagickSignature);
2383 *pixel=image->background_color;
2384 if (cache_info->methods.get_one_virtual_pixel_from_handler !=
2385 (GetOneVirtualPixelFromHandler) NULL)
2386 return(cache_info->methods.get_one_virtual_pixel_from_handler(image,
2387 GetPixelCacheVirtualMethod(image),x,y,pixel,exception));
2388 assert(id < (int) cache_info->number_threads);
2389 p=GetVirtualPixelsFromNexus(image,GetPixelCacheVirtualMethod(image),x,y,
2390 1UL,1UL,cache_info->nexus_info[id],exception);
2391 if (p == (const Quantum *) NULL)
2392 return(MagickFalse);
2393 GetPixelPacket(image,p,pixel);
2394 if (image->colorspace == CMYKColorspace)
2395 pixel->black=GetPixelBlack(image,p);
2400 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2404 + G e t O n e V i r t u a l P i x e l F r o m C a c h e %
2408 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2410 % GetOneVirtualPixelFromCache() returns a single virtual pixel at the
2411 % specified (x,y) location. The image background color is returned if an
2414 % The format of the GetOneVirtualPixelFromCache() method is:
2416 % MagickBooleanType GetOneVirtualPixelFromCache(const Image image,
2417 % const VirtualPixelMethod method,const ssize_t x,const ssize_t y,
2418 % PixelPacket *pixel,ExceptionInfo *exception)
2420 % A description of each parameter follows:
2422 % o image: the image.
2424 % o virtual_pixel_method: the virtual pixel method.
2426 % o x,y: These values define the location of the pixel to return.
2428 % o pixel: return a pixel at the specified (x,y) location.
2430 % o exception: return any errors or warnings in this structure.
2433 static MagickBooleanType GetOneVirtualPixelFromCache(const Image *image,
2434 const VirtualPixelMethod virtual_pixel_method,const ssize_t x,const ssize_t y,
2435 PixelPacket *pixel,ExceptionInfo *exception)
2441 id = GetOpenMPThreadId();
2446 assert(image != (const Image *) NULL);
2447 assert(image->signature == MagickSignature);
2448 assert(image->cache != (Cache) NULL);
2449 cache_info=(CacheInfo *) image->cache;
2450 assert(cache_info->signature == MagickSignature);
2451 assert(id < (int) cache_info->number_threads);
2452 *pixel=image->background_color;
2453 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,x,y,1UL,1UL,
2454 cache_info->nexus_info[id],exception);
2455 if (p == (const Quantum *) NULL)
2456 return(MagickFalse);
2457 GetPixelPacket(image,p,pixel);
2458 if (image->colorspace == CMYKColorspace)
2459 pixel->black=GetPixelBlack(image,p);
2464 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2468 + G e t P i x e l C a c h e C o l o r s p a c e %
2472 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2474 % GetPixelCacheColorspace() returns the class type of the pixel cache.
2476 % The format of the GetPixelCacheColorspace() method is:
2478 % Colorspace GetPixelCacheColorspace(Cache cache)
2480 % A description of each parameter follows:
2482 % o cache: the pixel cache.
2485 MagickExport ColorspaceType GetPixelCacheColorspace(const Cache cache)
2490 assert(cache != (Cache) NULL);
2491 cache_info=(CacheInfo *) cache;
2492 assert(cache_info->signature == MagickSignature);
2493 if (cache_info->debug != MagickFalse)
2494 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
2495 cache_info->filename);
2496 return(cache_info->colorspace);
2500 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2504 + G e t P i x e l C a c h e M e t h o d s %
2508 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2510 % GetPixelCacheMethods() initializes the CacheMethods structure.
2512 % The format of the GetPixelCacheMethods() method is:
2514 % void GetPixelCacheMethods(CacheMethods *cache_methods)
2516 % A description of each parameter follows:
2518 % o cache_methods: Specifies a pointer to a CacheMethods structure.
2521 MagickExport void GetPixelCacheMethods(CacheMethods *cache_methods)
2523 assert(cache_methods != (CacheMethods *) NULL);
2524 (void) ResetMagickMemory(cache_methods,0,sizeof(*cache_methods));
2525 cache_methods->get_virtual_pixel_handler=GetVirtualPixelCache;
2526 cache_methods->get_virtual_pixels_handler=GetVirtualPixelsCache;
2527 cache_methods->get_virtual_metacontent_from_handler=
2528 GetVirtualMetacontentFromCache;
2529 cache_methods->get_one_virtual_pixel_from_handler=GetOneVirtualPixelFromCache;
2530 cache_methods->get_authentic_pixels_handler=GetAuthenticPixelsCache;
2531 cache_methods->get_authentic_metacontent_from_handler=
2532 GetAuthenticMetacontentFromCache;
2533 cache_methods->get_authentic_pixels_from_handler=GetAuthenticPixelsFromCache;
2534 cache_methods->get_one_authentic_pixel_from_handler=
2535 GetOneAuthenticPixelFromCache;
2536 cache_methods->queue_authentic_pixels_handler=QueueAuthenticPixelsCache;
2537 cache_methods->sync_authentic_pixels_handler=SyncAuthenticPixelsCache;
2538 cache_methods->destroy_pixel_handler=DestroyImagePixelCache;
2542 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2546 + G e t P i x e l C a c h e N e x u s E x t e n t %
2550 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2552 % GetPixelCacheNexusExtent() returns the extent of the pixels associated
2553 % corresponding with the last call to SetPixelCacheNexusPixels() or
2554 % GetPixelCacheNexusPixels().
2556 % The format of the GetPixelCacheNexusExtent() method is:
2558 % MagickSizeType GetPixelCacheNexusExtent(const Cache cache,
2559 % NexusInfo *nexus_info)
2561 % A description of each parameter follows:
2563 % o nexus_info: the nexus info.
2566 MagickExport MagickSizeType GetPixelCacheNexusExtent(const Cache cache,
2567 NexusInfo *nexus_info)
2575 assert(cache != (const Cache) NULL);
2576 cache_info=(CacheInfo *) cache;
2577 assert(cache_info->signature == MagickSignature);
2578 extent=(MagickSizeType) nexus_info->region.width*nexus_info->region.height;
2580 return((MagickSizeType) cache_info->columns*cache_info->rows);
2585 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2589 + G e t P i x e l C a c h e N e x u s M e t a c o n t e n t %
2593 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2595 % GetPixelCacheNexusMetacontent() returns the meta-content for the specified
2598 % The format of the GetPixelCacheNexusMetacontent() method is:
2600 % void *GetPixelCacheNexusMetacontent(const Cache cache,
2601 % NexusInfo *nexus_info)
2603 % A description of each parameter follows:
2605 % o cache: the pixel cache.
2607 % o nexus_info: the cache nexus to return the meta-content.
2610 MagickExport void *GetPixelCacheNexusMetacontent(const Cache cache,
2611 NexusInfo *nexus_info)
2616 assert(cache != (const Cache) NULL);
2617 cache_info=(CacheInfo *) cache;
2618 assert(cache_info->signature == MagickSignature);
2619 if (cache_info->storage_class == UndefinedClass)
2620 return((void *) NULL);
2621 return(nexus_info->metacontent);
2625 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2629 + G e t P i x e l C a c h e N e x u s P i x e l s %
2633 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2635 % GetPixelCacheNexusPixels() returns the pixels associated with the specified
2638 % The format of the GetPixelCacheNexusPixels() method is:
2640 % Quantum *GetPixelCacheNexusPixels(const Cache cache,
2641 % NexusInfo *nexus_info)
2643 % A description of each parameter follows:
2645 % o cache: the pixel cache.
2647 % o nexus_info: the cache nexus to return the pixels.
2650 MagickExport Quantum *GetPixelCacheNexusPixels(const Cache cache,
2651 NexusInfo *nexus_info)
2656 assert(cache != (const Cache) NULL);
2657 cache_info=(CacheInfo *) cache;
2658 assert(cache_info->signature == MagickSignature);
2659 if (cache_info->storage_class == UndefinedClass)
2660 return((Quantum *) NULL);
2661 return(nexus_info->pixels);
2665 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2669 + G e t P i x e l C a c h e P i x e l s %
2673 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2675 % GetPixelCachePixels() returns the pixels associated with the specified image.
2677 % The format of the GetPixelCachePixels() method is:
2679 % void *GetPixelCachePixels(Image *image,MagickSizeType *length,
2680 % ExceptionInfo *exception)
2682 % A description of each parameter follows:
2684 % o image: the image.
2686 % o length: the pixel cache length.
2688 % o exception: return any errors or warnings in this structure.
2691 MagickExport void *GetPixelCachePixels(Image *image,MagickSizeType *length,
2692 ExceptionInfo *exception)
2697 assert(image != (const Image *) NULL);
2698 assert(image->signature == MagickSignature);
2699 assert(image->cache != (Cache) NULL);
2700 assert(length != (MagickSizeType *) NULL);
2701 assert(exception != (ExceptionInfo *) NULL);
2702 assert(exception->signature == MagickSignature);
2703 cache_info=(CacheInfo *) image->cache;
2704 assert(cache_info->signature == MagickSignature);
2706 if ((cache_info->type != MemoryCache) && (cache_info->type != MapCache))
2707 return((void *) NULL);
2708 *length=cache_info->length;
2709 return((void *) cache_info->pixels);
2713 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2717 + G e t P i x e l C a c h e S t o r a g e C l a s s %
2721 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2723 % GetPixelCacheStorageClass() returns the class type of the pixel cache.
2725 % The format of the GetPixelCacheStorageClass() method is:
2727 % ClassType GetPixelCacheStorageClass(Cache cache)
2729 % A description of each parameter follows:
2731 % o type: GetPixelCacheStorageClass returns DirectClass or PseudoClass.
2733 % o cache: the pixel cache.
2736 MagickExport ClassType GetPixelCacheStorageClass(const Cache cache)
2741 assert(cache != (Cache) NULL);
2742 cache_info=(CacheInfo *) cache;
2743 assert(cache_info->signature == MagickSignature);
2744 if (cache_info->debug != MagickFalse)
2745 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
2746 cache_info->filename);
2747 return(cache_info->storage_class);
2751 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2755 + G e t P i x e l C a c h e T i l e S i z e %
2759 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2761 % GetPixelCacheTileSize() returns the pixel cache tile size.
2763 % The format of the GetPixelCacheTileSize() method is:
2765 % void GetPixelCacheTileSize(const Image *image,size_t *width,
2768 % A description of each parameter follows:
2770 % o image: the image.
2772 % o width: the optimize cache tile width in pixels.
2774 % o height: the optimize cache tile height in pixels.
2777 MagickExport void GetPixelCacheTileSize(const Image *image,size_t *width,
2783 assert(image != (Image *) NULL);
2784 assert(image->signature == MagickSignature);
2785 if (image->debug != MagickFalse)
2786 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
2787 cache_info=(CacheInfo *) image->cache;
2788 assert(cache_info->signature == MagickSignature);
2789 *width=2048UL/(cache_info->number_channels*sizeof(Quantum));
2790 if (GetPixelCacheType(image) == DiskCache)
2791 *width=8192UL/(cache_info->number_channels*sizeof(Quantum));
2796 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2800 + G e t P i x e l C a c h e T y p e %
2804 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2806 % GetPixelCacheType() returns the pixel cache type (e.g. memory, disk, etc.).
2808 % The format of the GetPixelCacheType() method is:
2810 % CacheType GetPixelCacheType(const Image *image)
2812 % A description of each parameter follows:
2814 % o image: the image.
2817 MagickExport CacheType GetPixelCacheType(const Image *image)
2822 assert(image != (Image *) NULL);
2823 assert(image->signature == MagickSignature);
2824 assert(image->cache != (Cache) NULL);
2825 cache_info=(CacheInfo *) image->cache;
2826 assert(cache_info->signature == MagickSignature);
2827 return(cache_info->type);
2831 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2835 + G e t P i x e l C a c h e V i r t u a l M e t h o d %
2839 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2841 % GetPixelCacheVirtualMethod() gets the "virtual pixels" method for the
2842 % pixel cache. A virtual pixel is any pixel access that is outside the
2843 % boundaries of the image cache.
2845 % The format of the GetPixelCacheVirtualMethod() method is:
2847 % VirtualPixelMethod GetPixelCacheVirtualMethod(const Image *image)
2849 % A description of each parameter follows:
2851 % o image: the image.
2854 MagickExport VirtualPixelMethod GetPixelCacheVirtualMethod(const Image *image)
2859 assert(image != (Image *) NULL);
2860 assert(image->signature == MagickSignature);
2861 assert(image->cache != (Cache) NULL);
2862 cache_info=(CacheInfo *) image->cache;
2863 assert(cache_info->signature == MagickSignature);
2864 return(cache_info->virtual_pixel_method);
2868 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2872 + G e t V i r t u a l M e t a c o n t e n t F r o m C a c h e %
2876 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2878 % GetVirtualMetacontentFromCache() returns the meta-content corresponding with
2879 % the last call to QueueAuthenticPixelsCache() or GetVirtualPixelCache().
2881 % The format of the GetVirtualMetacontentFromCache() method is:
2883 % void *GetVirtualMetacontentFromCache(const Image *image)
2885 % A description of each parameter follows:
2887 % o image: the image.
2890 static const void *GetVirtualMetacontentFromCache(const Image *image)
2896 id = GetOpenMPThreadId();
2901 assert(image != (const Image *) NULL);
2902 assert(image->signature == MagickSignature);
2903 assert(image->cache != (Cache) NULL);
2904 cache_info=(CacheInfo *) image->cache;
2905 assert(cache_info->signature == MagickSignature);
2906 assert(id < (int) cache_info->number_threads);
2907 metacontent=GetVirtualMetacontentFromNexus(cache_info,
2908 cache_info->nexus_info[id]);
2909 return(metacontent);
2913 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2917 + G e t V i r t u a l M e t a c o n t e n t F r o m N e x u s %
2921 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2923 % GetVirtualMetacontentFromNexus() returns the meta-content for the specified
2926 % The format of the GetVirtualMetacontentFromNexus() method is:
2928 % const void *GetVirtualMetacontentFromNexus(const Cache cache,
2929 % NexusInfo *nexus_info)
2931 % A description of each parameter follows:
2933 % o cache: the pixel cache.
2935 % o nexus_info: the cache nexus to return the meta-content.
2938 MagickExport const void *GetVirtualMetacontentFromNexus(const Cache cache,
2939 NexusInfo *nexus_info)
2944 assert(cache != (Cache) NULL);
2945 cache_info=(CacheInfo *) cache;
2946 assert(cache_info->signature == MagickSignature);
2947 if (cache_info->storage_class == UndefinedClass)
2948 return((void *) NULL);
2949 return(nexus_info->metacontent);
2953 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2957 % G e t V i r t u a l M e t a c o n t e n t %
2961 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2963 % GetVirtualMetacontent() returns the virtual metacontent corresponding with
2964 % the last call to QueueAuthenticPixels() or GetVirtualPixels(). NULL is
2965 % returned if the meta-content are not available.
2967 % The format of the GetVirtualMetacontent() method is:
2969 % const void *GetVirtualMetacontent(const Image *image)
2971 % A description of each parameter follows:
2973 % o image: the image.
2976 MagickExport const void *GetVirtualMetacontent(const Image *image)
2982 id = GetOpenMPThreadId();
2987 assert(image != (const Image *) NULL);
2988 assert(image->signature == MagickSignature);
2989 assert(image->cache != (Cache) NULL);
2990 cache_info=(CacheInfo *) image->cache;
2991 assert(cache_info->signature == MagickSignature);
2992 if (cache_info->methods.get_virtual_metacontent_from_handler !=
2993 (GetVirtualMetacontentFromHandler) NULL)
2995 metacontent=cache_info->methods.
2996 get_virtual_metacontent_from_handler(image);
2997 return(metacontent);
2999 assert(id < (int) cache_info->number_threads);
3000 metacontent=GetVirtualMetacontentFromNexus(cache_info,
3001 cache_info->nexus_info[id]);
3002 return(metacontent);
3006 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3010 + G e t V i r t u a l P i x e l s F r o m N e x u s %
3014 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3016 % GetVirtualPixelsFromNexus() gets virtual pixels from the in-memory or disk
3017 % pixel cache as defined by the geometry parameters. A pointer to the pixels
3018 % is returned if the pixels are transferred, otherwise a NULL is returned.
3020 % The format of the GetVirtualPixelsFromNexus() method is:
3022 % Quantum *GetVirtualPixelsFromNexus(const Image *image,
3023 % const VirtualPixelMethod method,const ssize_t x,const ssize_t y,
3024 % const size_t columns,const size_t rows,NexusInfo *nexus_info,
3025 % ExceptionInfo *exception)
3027 % A description of each parameter follows:
3029 % o image: the image.
3031 % o virtual_pixel_method: the virtual pixel method.
3033 % o x,y,columns,rows: These values define the perimeter of a region of
3036 % o nexus_info: the cache nexus to acquire.
3038 % o exception: return any errors or warnings in this structure.
3045 0, 48, 12, 60, 3, 51, 15, 63,
3046 32, 16, 44, 28, 35, 19, 47, 31,
3047 8, 56, 4, 52, 11, 59, 7, 55,
3048 40, 24, 36, 20, 43, 27, 39, 23,
3049 2, 50, 14, 62, 1, 49, 13, 61,
3050 34, 18, 46, 30, 33, 17, 45, 29,
3051 10, 58, 6, 54, 9, 57, 5, 53,
3052 42, 26, 38, 22, 41, 25, 37, 21
3055 static inline ssize_t DitherX(const ssize_t x,const size_t columns)
3060 index=x+DitherMatrix[x & 0x07]-32L;
3063 if (index >= (ssize_t) columns)
3064 return((ssize_t) columns-1L);
3068 static inline ssize_t DitherY(const ssize_t y,const size_t rows)
3073 index=y+DitherMatrix[y & 0x07]-32L;
3076 if (index >= (ssize_t) rows)
3077 return((ssize_t) rows-1L);
3081 static inline ssize_t EdgeX(const ssize_t x,const size_t columns)
3085 if (x >= (ssize_t) columns)
3086 return((ssize_t) (columns-1));
3090 static inline ssize_t EdgeY(const ssize_t y,const size_t rows)
3094 if (y >= (ssize_t) rows)
3095 return((ssize_t) (rows-1));
3099 static inline ssize_t RandomX(RandomInfo *random_info,const size_t columns)
3101 return((ssize_t) (columns*GetPseudoRandomValue(random_info)));
3104 static inline ssize_t RandomY(RandomInfo *random_info,const size_t rows)
3106 return((ssize_t) (rows*GetPseudoRandomValue(random_info)));
3109 static inline MagickModulo VirtualPixelModulo(const ssize_t offset,
3110 const size_t extent)
3116 Compute the remainder of dividing offset by extent. It returns not only
3117 the quotient (tile the offset falls in) but also the positive remainer
3118 within that tile such that 0 <= remainder < extent. This method is
3119 essentially a ldiv() using a floored modulo division rather than the
3120 normal default truncated modulo division.
3122 modulo.quotient=offset/(ssize_t) extent;
3125 modulo.remainder=offset-modulo.quotient*(ssize_t) extent;
3129 MagickExport const Quantum *GetVirtualPixelsFromNexus(const Image *image,
3130 const VirtualPixelMethod virtual_pixel_method,const ssize_t x,const ssize_t y,
3131 const size_t columns,const size_t rows,NexusInfo *nexus_info,
3132 ExceptionInfo *exception)
3149 virtual_pixel[MaxPixelChannels];
3154 register const Quantum
3167 register unsigned char
3174 *virtual_metacontent;
3179 assert(image != (const Image *) NULL);
3180 assert(image->signature == MagickSignature);
3181 assert(image->cache != (Cache) NULL);
3182 cache_info=(CacheInfo *) image->cache;
3183 assert(cache_info->signature == MagickSignature);
3184 if (cache_info->type == UndefinedCache)
3185 return((const Quantum *) NULL);
3188 region.width=columns;
3190 pixels=SetPixelCacheNexusPixels(image,®ion,nexus_info,exception);
3191 if (pixels == (Quantum *) NULL)
3192 return((const Quantum *) NULL);
3193 offset=(MagickOffsetType) nexus_info->region.y*cache_info->columns+
3194 nexus_info->region.x;
3195 length=(MagickSizeType) (nexus_info->region.height-1L)*cache_info->columns+
3196 nexus_info->region.width-1L;
3197 number_pixels=(MagickSizeType) cache_info->columns*cache_info->rows;
3198 if ((offset >= 0) && (((MagickSizeType) offset+length) < number_pixels))
3199 if ((x >= 0) && ((ssize_t) (x+columns) <= (ssize_t) cache_info->columns) &&
3200 (y >= 0) && ((ssize_t) (y+rows) <= (ssize_t) cache_info->rows))
3206 Pixel request is inside cache extents.
3208 if (IsPixelAuthentic(cache_info,nexus_info) != MagickFalse)
3210 status=ReadPixelCachePixels(cache_info,nexus_info,exception);
3211 if (status == MagickFalse)
3212 return((const Quantum *) NULL);
3213 if (cache_info->metacontent_extent != 0)
3215 status=ReadPixelCacheMetacontent(cache_info,nexus_info,exception);
3216 if (status == MagickFalse)
3217 return((const Quantum *) NULL);
3222 Pixel request is outside cache extents.
3225 s=(unsigned char *) GetPixelCacheNexusMetacontent(cache_info,nexus_info);
3226 virtual_nexus=AcquirePixelCacheNexus(1);
3227 if (virtual_nexus == (NexusInfo **) NULL)
3229 if (virtual_nexus != (NexusInfo **) NULL)
3230 virtual_nexus=DestroyPixelCacheNexus(virtual_nexus,1);
3231 (void) ThrowMagickException(exception,GetMagickModule(),CacheError,
3232 "UnableToGetCacheNexus","`%s'",image->filename);
3233 return((const Quantum *) NULL);
3235 (void) ResetMagickMemory(virtual_pixel,0,cache_info->number_channels*
3236 sizeof(*virtual_pixel));
3237 virtual_metacontent=(void *) NULL;
3238 switch (virtual_pixel_method)
3240 case BackgroundVirtualPixelMethod:
3241 case BlackVirtualPixelMethod:
3242 case GrayVirtualPixelMethod:
3243 case TransparentVirtualPixelMethod:
3244 case MaskVirtualPixelMethod:
3245 case WhiteVirtualPixelMethod:
3246 case EdgeVirtualPixelMethod:
3247 case CheckerTileVirtualPixelMethod:
3248 case HorizontalTileVirtualPixelMethod:
3249 case VerticalTileVirtualPixelMethod:
3251 if (cache_info->metacontent_extent != 0)
3254 Acquire a metacontent buffer.
3256 virtual_metacontent=(void *) AcquireAlignedMemory(1,
3257 cache_info->metacontent_extent);
3258 if (virtual_metacontent == (void *) NULL)
3260 virtual_nexus=DestroyPixelCacheNexus(virtual_nexus,1);
3261 (void) ThrowMagickException(exception,GetMagickModule(),
3262 CacheError,"UnableToGetCacheNexus","`%s'",image->filename);
3263 return((const Quantum *) NULL);
3265 (void) ResetMagickMemory(virtual_metacontent,0,
3266 cache_info->metacontent_extent);
3268 switch (virtual_pixel_method)
3270 case BlackVirtualPixelMethod:
3272 for (i=0; i < (ssize_t) cache_info->number_channels; i++)
3273 SetPixelChannel(image,(PixelChannel) i,0,virtual_pixel);
3274 SetPixelAlpha(image,OpaqueAlpha,virtual_pixel);
3277 case GrayVirtualPixelMethod:
3279 for (i=0; i < (ssize_t) cache_info->number_channels; i++)
3280 SetPixelChannel(image,(PixelChannel) i,QuantumRange/2,virtual_pixel);
3281 SetPixelAlpha(image,OpaqueAlpha,virtual_pixel);
3284 case TransparentVirtualPixelMethod:
3286 for (i=0; i < (ssize_t) cache_info->number_channels; i++)
3287 SetPixelChannel(image,(PixelChannel) i,0,virtual_pixel);
3288 SetPixelAlpha(image,TransparentAlpha,virtual_pixel);
3291 case MaskVirtualPixelMethod:
3292 case WhiteVirtualPixelMethod:
3294 for (i=0; i < (ssize_t) cache_info->number_channels; i++)
3295 SetPixelChannel(image,(PixelChannel) i,QuantumRange,virtual_pixel);
3296 SetPixelAlpha(image,OpaqueAlpha,virtual_pixel);
3301 SetPixelRed(image,image->background_color.red,virtual_pixel);
3302 SetPixelGreen(image,image->background_color.green,virtual_pixel);
3303 SetPixelBlue(image,image->background_color.blue,virtual_pixel);
3304 if (image->colorspace == CMYKColorspace)
3305 SetPixelBlack(image,image->background_color.black,virtual_pixel);
3306 SetPixelAlpha(image,image->background_color.alpha,virtual_pixel);
3315 for (v=0; v < (ssize_t) rows; v++)
3317 for (u=0; u < (ssize_t) columns; u+=length)
3319 length=(MagickSizeType) MagickMin(cache_info->columns-(x+u),columns-u);
3320 if ((((x+u) < 0) || ((x+u) >= (ssize_t) cache_info->columns)) ||
3321 (((y+v) < 0) || ((y+v) >= (ssize_t) cache_info->rows)) ||
3329 Transfer a single pixel.
3331 length=(MagickSizeType) 1;
3332 switch (virtual_pixel_method)
3336 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,
3337 EdgeX(x+u,cache_info->columns),EdgeY(y+v,cache_info->rows),
3338 1UL,1UL,*virtual_nexus,exception);
3339 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3342 case RandomVirtualPixelMethod:
3344 if (cache_info->random_info == (RandomInfo *) NULL)
3345 cache_info->random_info=AcquireRandomInfo();
3346 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,
3347 RandomX(cache_info->random_info,cache_info->columns),
3348 RandomY(cache_info->random_info,cache_info->rows),1UL,1UL,
3349 *virtual_nexus,exception);
3350 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3353 case DitherVirtualPixelMethod:
3355 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,
3356 DitherX(x+u,cache_info->columns),DitherY(y+v,cache_info->rows),
3357 1UL,1UL,*virtual_nexus,exception);
3358 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3361 case TileVirtualPixelMethod:
3363 x_modulo=VirtualPixelModulo(x+u,cache_info->columns);
3364 y_modulo=VirtualPixelModulo(y+v,cache_info->rows);
3365 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,
3366 x_modulo.remainder,y_modulo.remainder,1UL,1UL,*virtual_nexus,
3368 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3371 case MirrorVirtualPixelMethod:
3373 x_modulo=VirtualPixelModulo(x+u,cache_info->columns);
3374 if ((x_modulo.quotient & 0x01) == 1L)
3375 x_modulo.remainder=(ssize_t) cache_info->columns-
3376 x_modulo.remainder-1L;
3377 y_modulo=VirtualPixelModulo(y+v,cache_info->rows);
3378 if ((y_modulo.quotient & 0x01) == 1L)
3379 y_modulo.remainder=(ssize_t) cache_info->rows-
3380 y_modulo.remainder-1L;
3381 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,
3382 x_modulo.remainder,y_modulo.remainder,1UL,1UL,*virtual_nexus,
3384 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3387 case HorizontalTileEdgeVirtualPixelMethod:
3389 x_modulo=VirtualPixelModulo(x+u,cache_info->columns);
3390 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,
3391 x_modulo.remainder,EdgeY(y+v,cache_info->rows),1UL,1UL,
3392 *virtual_nexus,exception);
3393 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3396 case VerticalTileEdgeVirtualPixelMethod:
3398 y_modulo=VirtualPixelModulo(y+v,cache_info->rows);
3399 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,
3400 EdgeX(x+u,cache_info->columns),y_modulo.remainder,1UL,1UL,
3401 *virtual_nexus,exception);
3402 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3405 case BackgroundVirtualPixelMethod:
3406 case BlackVirtualPixelMethod:
3407 case GrayVirtualPixelMethod:
3408 case TransparentVirtualPixelMethod:
3409 case MaskVirtualPixelMethod:
3410 case WhiteVirtualPixelMethod:
3413 r=virtual_metacontent;
3416 case EdgeVirtualPixelMethod:
3417 case CheckerTileVirtualPixelMethod:
3419 x_modulo=VirtualPixelModulo(x+u,cache_info->columns);
3420 y_modulo=VirtualPixelModulo(y+v,cache_info->rows);
3421 if (((x_modulo.quotient ^ y_modulo.quotient) & 0x01) != 0L)
3424 r=virtual_metacontent;
3427 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,
3428 x_modulo.remainder,y_modulo.remainder,1UL,1UL,*virtual_nexus,
3430 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3433 case HorizontalTileVirtualPixelMethod:
3435 if (((y+v) < 0) || ((y+v) >= (ssize_t) cache_info->rows))
3438 r=virtual_metacontent;
3441 x_modulo=VirtualPixelModulo(x+u,cache_info->columns);
3442 y_modulo=VirtualPixelModulo(y+v,cache_info->rows);
3443 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,
3444 x_modulo.remainder,y_modulo.remainder,1UL,1UL,*virtual_nexus,
3446 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3449 case VerticalTileVirtualPixelMethod:
3451 if (((x+u) < 0) || ((x+u) >= (ssize_t) cache_info->columns))
3454 r=virtual_metacontent;
3457 x_modulo=VirtualPixelModulo(x+u,cache_info->columns);
3458 y_modulo=VirtualPixelModulo(y+v,cache_info->rows);
3459 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,
3460 x_modulo.remainder,y_modulo.remainder,1UL,1UL,*virtual_nexus,
3462 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3466 if (p == (const Quantum *) NULL)
3468 (void) memcpy(q,p,(size_t) length*cache_info->number_channels*
3470 q+=cache_info->number_channels;
3471 if ((s != (void *) NULL) && (r != (const void *) NULL))
3473 (void) memcpy(s,r,(size_t) cache_info->metacontent_extent);
3474 s+=cache_info->metacontent_extent;
3479 Transfer a run of pixels.
3481 p=GetVirtualPixelsFromNexus(image,virtual_pixel_method,x+u,y+v,(size_t)
3482 length,1UL,*virtual_nexus,exception);
3483 if (p == (const Quantum *) NULL)
3485 r=GetVirtualMetacontentFromNexus(cache_info,*virtual_nexus);
3486 (void) memcpy(q,p,(size_t) length*cache_info->number_channels*sizeof(*p));
3487 q+=length*cache_info->number_channels;
3488 if ((r != (void *) NULL) && (s != (const void *) NULL))
3490 (void) memcpy(s,r,(size_t) length);
3491 s+=length*cache_info->metacontent_extent;
3498 if (virtual_metacontent != (void *) NULL)
3499 virtual_metacontent=(void *) RelinquishMagickMemory(virtual_metacontent);
3500 virtual_nexus=DestroyPixelCacheNexus(virtual_nexus,1);
3505 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3509 + G e t V i r t u a l P i x e l C a c h e %
3513 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3515 % GetVirtualPixelCache() get virtual pixels from the in-memory or disk pixel
3516 % cache as defined by the geometry parameters. A pointer to the pixels
3517 % is returned if the pixels are transferred, otherwise a NULL is returned.
3519 % The format of the GetVirtualPixelCache() method is:
3521 % const Quantum *GetVirtualPixelCache(const Image *image,
3522 % const VirtualPixelMethod virtual_pixel_method,const ssize_t x,
3523 % const ssize_t y,const size_t columns,const size_t rows,
3524 % ExceptionInfo *exception)
3526 % A description of each parameter follows:
3528 % o image: the image.
3530 % o virtual_pixel_method: the virtual pixel method.
3532 % o x,y,columns,rows: These values define the perimeter of a region of
3535 % o exception: return any errors or warnings in this structure.
3538 static const Quantum *GetVirtualPixelCache(const Image *image,
3539 const VirtualPixelMethod virtual_pixel_method,const ssize_t x,const ssize_t y,
3540 const size_t columns,const size_t rows,ExceptionInfo *exception)
3546 id = GetOpenMPThreadId();
3551 assert(image != (const Image *) NULL);
3552 assert(image->signature == MagickSignature);
3553 assert(image->cache != (Cache) NULL);
3554 cache_info=(CacheInfo *) image->cache;
3555 assert(cache_info->signature == MagickSignature);
3556 assert(id < (int) cache_info->number_threads);
3557 pixels=GetVirtualPixelsFromNexus(image,virtual_pixel_method,x,y,columns,rows,
3558 cache_info->nexus_info[id],exception);
3563 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3567 % G e t V i r t u a l P i x e l Q u e u e %
3571 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3573 % GetVirtualPixelQueue() returns the virtual pixels associated corresponding
3574 % with the last call to QueueAuthenticPixels() or GetVirtualPixels().
3576 % The format of the GetVirtualPixelQueue() method is:
3578 % const Quantum *GetVirtualPixelQueue(const Image image)
3580 % A description of each parameter follows:
3582 % o image: the image.
3585 MagickExport const Quantum *GetVirtualPixelQueue(const Image *image)
3591 id = GetOpenMPThreadId();
3593 assert(image != (const Image *) NULL);
3594 assert(image->signature == MagickSignature);
3595 assert(image->cache != (Cache) NULL);
3596 cache_info=(CacheInfo *) image->cache;
3597 assert(cache_info->signature == MagickSignature);
3598 if (cache_info->methods.get_virtual_pixels_handler !=
3599 (GetVirtualPixelsHandler) NULL)
3600 return(cache_info->methods.get_virtual_pixels_handler(image));
3601 assert(id < (int) cache_info->number_threads);
3602 return(GetVirtualPixelsNexus(cache_info,cache_info->nexus_info[id]));
3606 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3610 % G e t V i r t u a l P i x e l s %
3614 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3616 % GetVirtualPixels() returns an immutable pixel region. If the
3617 % region is successfully accessed, a pointer to it is returned, otherwise
3618 % NULL is returned. The returned pointer may point to a temporary working
3619 % copy of the pixels or it may point to the original pixels in memory.
3620 % Performance is maximized if the selected region is part of one row, or one
3621 % or more full rows, since there is opportunity to access the pixels in-place
3622 % (without a copy) if the image is in memory, or in a memory-mapped file. The
3623 % returned pointer must *never* be deallocated by the user.
3625 % Pixels accessed via the returned pointer represent a simple array of type
3626 % Quantum. If the image type is CMYK or the storage class is PseudoClass,
3627 % call GetAuthenticMetacontent() after invoking GetAuthenticPixels() to
3628 % access the meta-content (of type void) corresponding to the the
3631 % If you plan to modify the pixels, use GetAuthenticPixels() instead.
3633 % Note, the GetVirtualPixels() and GetAuthenticPixels() methods are not thread-
3634 % safe. In a threaded environment, use GetCacheViewVirtualPixels() or
3635 % GetCacheViewAuthenticPixels() instead.
3637 % The format of the GetVirtualPixels() method is:
3639 % const Quantum *GetVirtualPixels(const Image *image,const ssize_t x,
3640 % const ssize_t y,const size_t columns,const size_t rows,
3641 % ExceptionInfo *exception)
3643 % A description of each parameter follows:
3645 % o image: the image.
3647 % o x,y,columns,rows: These values define the perimeter of a region of
3650 % o exception: return any errors or warnings in this structure.
3653 MagickExport const Quantum *GetVirtualPixels(const Image *image,
3654 const ssize_t x,const ssize_t y,const size_t columns,const size_t rows,
3655 ExceptionInfo *exception)
3661 id = GetOpenMPThreadId();
3666 assert(image != (const Image *) NULL);
3667 assert(image->signature == MagickSignature);
3668 assert(image->cache != (Cache) NULL);
3669 cache_info=(CacheInfo *) image->cache;
3670 assert(cache_info->signature == MagickSignature);
3671 if (cache_info->methods.get_virtual_pixel_handler !=
3672 (GetVirtualPixelHandler) NULL)
3673 return(cache_info->methods.get_virtual_pixel_handler(image,
3674 GetPixelCacheVirtualMethod(image),x,y,columns,rows,exception));
3675 assert(id < (int) cache_info->number_threads);
3676 pixels=GetVirtualPixelsFromNexus(image,GetPixelCacheVirtualMethod(image),x,y,
3677 columns,rows,cache_info->nexus_info[id],exception);
3682 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3686 + G e t V i r t u a l P i x e l s F r o m C a c h e %
3690 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3692 % GetVirtualPixelsCache() returns the pixels associated corresponding with the
3693 % last call to QueueAuthenticPixelsCache() or GetVirtualPixelCache().
3695 % The format of the GetVirtualPixelsCache() method is:
3697 % Quantum *GetVirtualPixelsCache(const Image *image)
3699 % A description of each parameter follows:
3701 % o image: the image.
3704 static const Quantum *GetVirtualPixelsCache(const Image *image)
3710 id = GetOpenMPThreadId();
3712 assert(image != (const Image *) NULL);
3713 assert(image->signature == MagickSignature);
3714 assert(image->cache != (Cache) NULL);
3715 cache_info=(CacheInfo *) image->cache;
3716 assert(cache_info->signature == MagickSignature);
3717 assert(id < (int) cache_info->number_threads);
3718 return(GetVirtualPixelsNexus(image->cache,cache_info->nexus_info[id]));
3722 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3726 + G e t V i r t u a l P i x e l s N e x u s %
3730 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3732 % GetVirtualPixelsNexus() returns the pixels associated with the specified
3735 % The format of the GetVirtualPixelsNexus() method is:
3737 % const Quantum *GetVirtualPixelsNexus(const Cache cache,
3738 % NexusInfo *nexus_info)
3740 % A description of each parameter follows:
3742 % o cache: the pixel cache.
3744 % o nexus_info: the cache nexus to return the colormap pixels.
3747 MagickExport const Quantum *GetVirtualPixelsNexus(const Cache cache,
3748 NexusInfo *nexus_info)
3753 assert(cache != (Cache) NULL);
3754 cache_info=(CacheInfo *) cache;
3755 assert(cache_info->signature == MagickSignature);
3756 if (cache_info->storage_class == UndefinedClass)
3757 return((Quantum *) NULL);
3758 return((const Quantum *) nexus_info->pixels);
3762 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3766 + M a s k P i x e l C a c h e N e x u s %
3770 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3772 % MaskPixelCacheNexus() masks the cache nexus as defined by the image mask.
3773 % The method returns MagickTrue if the pixel region is masked, otherwise
3776 % The format of the MaskPixelCacheNexus() method is:
3778 % MagickBooleanType MaskPixelCacheNexus(Image *image,
3779 % NexusInfo *nexus_info,ExceptionInfo *exception)
3781 % A description of each parameter follows:
3783 % o image: the image.
3785 % o nexus_info: the cache nexus to clip.
3787 % o exception: return any errors or warnings in this structure.
3791 static inline void MagickPixelCompositeMask(const PixelInfo *p,
3792 const MagickRealType alpha,const PixelInfo *q,
3793 const MagickRealType beta,PixelInfo *composite)
3798 if (alpha == TransparentAlpha)
3803 gamma=1.0-QuantumScale*QuantumScale*alpha*beta;
3804 gamma=1.0/(gamma <= MagickEpsilon ? 1.0 : gamma);
3805 composite->red=gamma*MagickOver_(p->red,alpha,q->red,beta);
3806 composite->green=gamma*MagickOver_(p->green,alpha,q->green,beta);
3807 composite->blue=gamma*MagickOver_(p->blue,alpha,q->blue,beta);
3808 if ((p->colorspace == CMYKColorspace) && (q->colorspace == CMYKColorspace))
3809 composite->black=gamma*MagickOver_(p->black,alpha,q->black,beta);
3812 static MagickBooleanType MaskPixelCacheNexus(Image *image,NexusInfo *nexus_info,
3813 ExceptionInfo *exception)
3829 register const Quantum
3842 if (image->debug != MagickFalse)
3843 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
3844 if (image->mask == (Image *) NULL)
3845 return(MagickFalse);
3846 cache_info=(CacheInfo *) image->cache;
3847 if (cache_info == (Cache) NULL)
3848 return(MagickFalse);
3849 image_nexus=AcquirePixelCacheNexus(1);
3850 clip_nexus=AcquirePixelCacheNexus(1);
3851 if ((image_nexus == (NexusInfo **) NULL) ||
3852 (clip_nexus == (NexusInfo **) NULL))
3853 ThrowBinaryException(CacheError,"UnableToGetCacheNexus",image->filename);
3854 p=(const Quantum *) GetAuthenticPixelCacheNexus(image,
3855 nexus_info->region.x,nexus_info->region.y,nexus_info->region.width,
3856 nexus_info->region.height,image_nexus[0],exception);
3857 q=nexus_info->pixels;
3858 r=GetVirtualPixelsFromNexus(image->mask,MaskVirtualPixelMethod,
3859 nexus_info->region.x,nexus_info->region.y,nexus_info->region.width,
3860 nexus_info->region.height,clip_nexus[0],&image->exception);
3861 GetPixelInfo(image,&alpha);
3862 GetPixelInfo(image,&beta);
3863 number_pixels=(MagickSizeType) nexus_info->region.width*
3864 nexus_info->region.height;
3865 for (i=0; i < (ssize_t) number_pixels; i++)
3867 if ((p == (const Quantum *) NULL) || (r == (const Quantum *) NULL))
3869 SetPixelInfo(image,p,&alpha);
3870 SetPixelInfo(image,q,&beta);
3871 MagickPixelCompositeMask(&beta,(MagickRealType) GetPixelIntensity(image,r),
3872 &alpha,alpha.alpha,&beta);
3873 SetPixelRed(image,ClampToQuantum(beta.red),q);
3874 SetPixelGreen(image,ClampToQuantum(beta.green),q);
3875 SetPixelBlue(image,ClampToQuantum(beta.blue),q);
3876 if (cache_info->colorspace == CMYKColorspace)
3877 SetPixelBlack(image,ClampToQuantum(beta.black),q);
3878 SetPixelAlpha(image,ClampToQuantum(beta.alpha),q);
3883 clip_nexus=DestroyPixelCacheNexus(clip_nexus,1);
3884 image_nexus=DestroyPixelCacheNexus(image_nexus,1);
3885 if (i < (ssize_t) number_pixels)
3886 return(MagickFalse);
3891 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3895 + O p e n P i x e l C a c h e %
3899 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
3901 % OpenPixelCache() allocates the pixel cache. This includes defining the cache
3902 % dimensions, allocating space for the image pixels and optionally the
3903 % metacontent, and memory mapping the cache if it is disk based. The cache
3904 % nexus array is initialized as well.
3906 % The format of the OpenPixelCache() method is:
3908 % MagickBooleanType OpenPixelCache(Image *image,const MapMode mode,
3909 % ExceptionInfo *exception)
3911 % A description of each parameter follows:
3913 % o image: the image.
3915 % o mode: ReadMode, WriteMode, or IOMode.
3917 % o exception: return any errors or warnings in this structure.
3921 static inline void AllocatePixelCachePixels(CacheInfo *cache_info)
3923 cache_info->mapped=MagickFalse;
3924 cache_info->pixels=(Quantum *) AcquireAlignedMemory(1,(size_t)
3925 cache_info->length);
3926 if (cache_info->pixels == (Quantum *) NULL)
3928 cache_info->mapped=MagickTrue;
3929 cache_info->pixels=(Quantum *) MapBlob(-1,IOMode,0,(size_t)
3930 cache_info->length);
3934 static MagickBooleanType ExtendCache(Image *image,MagickSizeType length)
3944 cache_info=(CacheInfo *) image->cache;
3945 if (image->debug != MagickFalse)
3948 format[MaxTextExtent],
3949 message[MaxTextExtent];
3951 (void) FormatMagickSize(length,MagickFalse,format);
3952 (void) FormatLocaleString(message,MaxTextExtent,
3953 "extend %s (%s[%d], disk, %s)",cache_info->filename,
3954 cache_info->cache_filename,cache_info->file,format);
3955 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"%s",message);
3957 if (length != (MagickSizeType) ((MagickOffsetType) length))
3958 return(MagickFalse);
3959 extent=(MagickOffsetType) lseek(cache_info->file,0,SEEK_END);
3961 return(MagickFalse);
3962 if ((MagickSizeType) extent >= length)
3964 offset=(MagickOffsetType) length-1;
3965 count=WritePixelCacheRegion(cache_info,offset,1,(const unsigned char *) "");
3966 return(count == (MagickOffsetType) 1 ? MagickTrue : MagickFalse);
3969 static MagickBooleanType OpenPixelCache(Image *image,const MapMode mode,
3970 ExceptionInfo *exception)
3977 format[MaxTextExtent],
3978 message[MaxTextExtent];
3991 assert(image != (const Image *) NULL);
3992 assert(image->signature == MagickSignature);
3993 assert(image->cache != (Cache) NULL);
3994 if (image->debug != MagickFalse)
3995 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
3996 if ((image->columns == 0) || (image->rows == 0))
3997 ThrowBinaryException(CacheError,"NoPixelsDefinedInCache",image->filename);
3998 StandardPixelChannelMap(image);
3999 cache_info=(CacheInfo *) image->cache;
4000 assert(cache_info->signature == MagickSignature);
4001 source_info=(*cache_info);
4002 source_info.file=(-1);
4003 (void) FormatLocaleString(cache_info->filename,MaxTextExtent,"%s[%.20g]",
4004 image->filename,(double) GetImageIndexInList(image));
4005 cache_info->storage_class=image->storage_class;
4006 cache_info->colorspace=image->colorspace;
4007 cache_info->matte=image->matte;
4008 cache_info->rows=image->rows;
4009 cache_info->columns=image->columns;
4010 cache_info->number_channels=GetPixelChannels(image);
4011 cache_info->metacontent_extent=image->metacontent_extent;
4012 cache_info->mode=mode;
4013 if (image->ping != MagickFalse)
4015 cache_info->type=PingCache;
4016 cache_info->pixels=(Quantum *) NULL;
4017 cache_info->metacontent=(void *) NULL;
4018 cache_info->length=0;
4021 number_pixels=(MagickSizeType) cache_info->columns*cache_info->rows;
4022 packet_size=cache_info->number_channels*sizeof(Quantum);
4023 if (image->metacontent_extent != 0)
4024 packet_size+=cache_info->metacontent_extent;
4025 length=number_pixels*packet_size;
4026 columns=(size_t) (length/cache_info->rows/packet_size);
4027 if (cache_info->columns != columns)
4028 ThrowBinaryException(ResourceLimitError,"PixelCacheAllocationFailed",
4030 cache_info->length=length;
4031 if ((cache_info->type != UndefinedCache) &&
4032 (cache_info->columns <= source_info.columns) &&
4033 (cache_info->rows <= source_info.rows) &&
4034 (cache_info->number_channels <= source_info.number_channels) &&
4035 (cache_info->metacontent_extent <= source_info.metacontent_extent))
4038 Inline pixel cache clone optimization.
4040 if ((cache_info->columns == source_info.columns) &&
4041 (cache_info->rows == source_info.rows) &&
4042 (cache_info->number_channels == source_info.number_channels) &&
4043 (cache_info->metacontent_extent == source_info.metacontent_extent))
4045 return(ClonePixelCachePixels(cache_info,&source_info,exception));
4047 status=AcquireMagickResource(AreaResource,cache_info->length);
4048 length=number_pixels*(cache_info->number_channels*sizeof(Quantum)+
4049 cache_info->metacontent_extent);
4050 if ((status != MagickFalse) && (length == (MagickSizeType) ((size_t) length)))
4052 status=AcquireMagickResource(MemoryResource,cache_info->length);
4053 if (((cache_info->type == UndefinedCache) && (status != MagickFalse)) ||
4054 (cache_info->type == MemoryCache))
4056 AllocatePixelCachePixels(cache_info);
4057 if (cache_info->pixels == (Quantum *) NULL)
4058 cache_info->pixels=source_info.pixels;
4062 Create memory pixel cache.
4065 if (image->debug != MagickFalse)
4067 (void) FormatMagickSize(cache_info->length,MagickTrue,
4069 (void) FormatLocaleString(message,MaxTextExtent,
4070 "open %s (%s memory, %.20gx%.20gx%.20g %s)",
4071 cache_info->filename,cache_info->mapped != MagickFalse ?
4072 "anonymous" : "heap",(double) cache_info->columns,(double)
4073 cache_info->rows,(double) cache_info->number_channels,
4075 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"%s",
4078 cache_info->type=MemoryCache;
4079 cache_info->metacontent=(void *) NULL;
4080 if (cache_info->metacontent_extent != 0)
4081 cache_info->metacontent=(void *) (cache_info->pixels+
4082 number_pixels*cache_info->number_channels);
4083 if (source_info.storage_class != UndefinedClass)
4085 status=ClonePixelCachePixels(cache_info,&source_info,
4087 RelinquishPixelCachePixels(&source_info);
4092 RelinquishMagickResource(MemoryResource,cache_info->length);
4095 Create pixel cache on disk.
4097 status=AcquireMagickResource(DiskResource,cache_info->length);
4098 if (status == MagickFalse)
4100 (void) ThrowMagickException(exception,GetMagickModule(),CacheError,
4101 "CacheResourcesExhausted","`%s'",image->filename);
4102 return(MagickFalse);
4104 if (OpenPixelCacheOnDisk(cache_info,mode) == MagickFalse)
4106 RelinquishMagickResource(DiskResource,cache_info->length);
4107 ThrowFileException(exception,CacheError,"UnableToOpenPixelCache",
4109 return(MagickFalse);
4111 status=ExtendCache(image,(MagickSizeType) cache_info->offset+
4112 cache_info->length);
4113 if (status == MagickFalse)
4115 ThrowFileException(exception,CacheError,"UnableToExtendCache",
4117 return(MagickFalse);
4119 length=number_pixels*(cache_info->number_channels*sizeof(Quantum)+
4120 cache_info->metacontent_extent);
4121 status=AcquireMagickResource(AreaResource,cache_info->length);
4122 if ((status == MagickFalse) || (length != (MagickSizeType) ((size_t) length)))
4123 cache_info->type=DiskCache;
4126 status=AcquireMagickResource(MapResource,cache_info->length);
4127 if ((status == MagickFalse) && (cache_info->type != MapCache) &&
4128 (cache_info->type != MemoryCache))
4129 cache_info->type=DiskCache;
4132 cache_info->pixels=(Quantum *) MapBlob(cache_info->file,mode,
4133 cache_info->offset,(size_t) cache_info->length);
4134 if (cache_info->pixels == (Quantum *) NULL)
4136 cache_info->type=DiskCache;
4137 cache_info->pixels=source_info.pixels;
4142 Create file-backed memory-mapped pixel cache.
4145 (void) ClosePixelCacheOnDisk(cache_info);
4146 cache_info->type=MapCache;
4147 cache_info->mapped=MagickTrue;
4148 cache_info->metacontent=(void *) NULL;
4149 if (cache_info->metacontent_extent != 0)
4150 cache_info->metacontent=(void *) (cache_info->pixels+
4151 number_pixels*cache_info->number_channels);
4152 if (source_info.storage_class != UndefinedClass)
4154 status=ClonePixelCachePixels(cache_info,&source_info,
4156 RelinquishPixelCachePixels(&source_info);
4158 if (image->debug != MagickFalse)
4160 (void) FormatMagickSize(cache_info->length,MagickTrue,
4162 (void) FormatLocaleString(message,MaxTextExtent,
4163 "open %s (%s[%d], memory-mapped, %.20gx%.20gx%.20g %s)",
4164 cache_info->filename,cache_info->cache_filename,
4165 cache_info->file,(double) cache_info->columns,(double)
4166 cache_info->rows,(double) cache_info->number_channels,
4168 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"%s",
4174 RelinquishMagickResource(MapResource,cache_info->length);
4177 if ((source_info.type != UndefinedCache) && (mode != ReadMode))
4179 status=ClonePixelCachePixels(cache_info,&source_info,exception);
4180 RelinquishPixelCachePixels(&source_info);
4182 if (image->debug != MagickFalse)
4184 (void) FormatMagickSize(cache_info->length,MagickFalse,format);
4185 (void) FormatLocaleString(message,MaxTextExtent,
4186 "open %s (%s[%d], disk, %.20gx%.20gx%.20g %s)",cache_info->filename,
4187 cache_info->cache_filename,cache_info->file,(double)
4188 cache_info->columns,(double) cache_info->rows,(double)
4189 cache_info->number_channels,format);
4190 (void) LogMagickEvent(CacheEvent,GetMagickModule(),"%s",message);
4196 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4200 + P e r s i s t P i x e l C a c h e %
4204 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4206 % PersistPixelCache() attaches to or initializes a persistent pixel cache. A
4207 % persistent pixel cache is one that resides on disk and is not destroyed
4208 % when the program exits.
4210 % The format of the PersistPixelCache() method is:
4212 % MagickBooleanType PersistPixelCache(Image *image,const char *filename,
4213 % const MagickBooleanType attach,MagickOffsetType *offset,
4214 % ExceptionInfo *exception)
4216 % A description of each parameter follows:
4218 % o image: the image.
4220 % o filename: the persistent pixel cache filename.
4222 % o attach: A value other than zero initializes the persistent pixel cache.
4224 % o initialize: A value other than zero initializes the persistent pixel
4227 % o offset: the offset in the persistent cache to store pixels.
4229 % o exception: return any errors or warnings in this structure.
4232 MagickExport MagickBooleanType PersistPixelCache(Image *image,
4233 const char *filename,const MagickBooleanType attach,MagickOffsetType *offset,
4234 ExceptionInfo *exception)
4249 assert(image != (Image *) NULL);
4250 assert(image->signature == MagickSignature);
4251 if (image->debug != MagickFalse)
4252 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
4253 assert(image->cache != (void *) NULL);
4254 assert(filename != (const char *) NULL);
4255 assert(offset != (MagickOffsetType *) NULL);
4256 page_size=GetMagickPageSize();
4257 cache_info=(CacheInfo *) image->cache;
4258 assert(cache_info->signature == MagickSignature);
4259 if (attach != MagickFalse)
4262 Attach existing persistent pixel cache.
4264 if (image->debug != MagickFalse)
4265 (void) LogMagickEvent(CacheEvent,GetMagickModule(),
4266 "attach persistent cache");
4267 (void) CopyMagickString(cache_info->cache_filename,filename,
4269 cache_info->type=DiskCache;
4270 cache_info->offset=(*offset);
4271 if (OpenPixelCache(image,ReadMode,exception) == MagickFalse)
4272 return(MagickFalse);
4273 *offset+=cache_info->length+page_size-(cache_info->length % page_size);
4276 if ((cache_info->mode != ReadMode) && (cache_info->type != MemoryCache) &&
4277 (cache_info->reference_count == 1))
4279 LockSemaphoreInfo(cache_info->semaphore);
4280 if ((cache_info->mode != ReadMode) &&
4281 (cache_info->type != MemoryCache) &&
4282 (cache_info->reference_count == 1))
4288 Usurp existing persistent pixel cache.
4290 status=rename(cache_info->cache_filename,filename);
4293 (void) CopyMagickString(cache_info->cache_filename,filename,
4295 *offset+=cache_info->length+page_size-(cache_info->length %
4297 UnlockSemaphoreInfo(cache_info->semaphore);
4298 cache_info=(CacheInfo *) ReferencePixelCache(cache_info);
4299 if (image->debug != MagickFalse)
4300 (void) LogMagickEvent(CacheEvent,GetMagickModule(),
4301 "Usurp resident persistent cache");
4305 UnlockSemaphoreInfo(cache_info->semaphore);
4308 Clone persistent pixel cache.
4310 clone_image=(*image);
4311 clone_info=(CacheInfo *) clone_image.cache;
4312 image->cache=ClonePixelCache(cache_info);
4313 cache_info=(CacheInfo *) ReferencePixelCache(image->cache);
4314 (void) CopyMagickString(cache_info->cache_filename,filename,MaxTextExtent);
4315 cache_info->type=DiskCache;
4316 cache_info->offset=(*offset);
4317 cache_info=(CacheInfo *) image->cache;
4318 status=OpenPixelCache(image,IOMode,exception);
4319 if (status != MagickFalse)
4320 status=ClonePixelCachePixels(cache_info,clone_info,&image->exception);
4321 *offset+=cache_info->length+page_size-(cache_info->length % page_size);
4322 clone_info=(CacheInfo *) DestroyPixelCache(clone_info);
4327 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4331 + Q u e u e A u t h e n t i c N e x u s %
4335 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4337 % QueueAuthenticNexus() allocates an region to store image pixels as defined
4338 % by the region rectangle and returns a pointer to the region. This region is
4339 % subsequently transferred from the pixel cache with
4340 % SyncAuthenticPixelsCache(). A pointer to the pixels is returned if the
4341 % pixels are transferred, otherwise a NULL is returned.
4343 % The format of the QueueAuthenticNexus() method is:
4345 % Quantum *QueueAuthenticNexus(Image *image,const ssize_t x,
4346 % const ssize_t y,const size_t columns,const size_t rows,
4347 % NexusInfo *nexus_info,ExceptionInfo *exception)
4349 % A description of each parameter follows:
4351 % o image: the image.
4353 % o x,y,columns,rows: These values define the perimeter of a region of
4356 % o nexus_info: the cache nexus to set.
4358 % o exception: return any errors or warnings in this structure.
4361 MagickExport Quantum *QueueAuthenticNexus(Image *image,const ssize_t x,
4362 const ssize_t y,const size_t columns,const size_t rows,NexusInfo *nexus_info,
4363 ExceptionInfo *exception)
4378 Validate pixel cache geometry.
4380 assert(image != (const Image *) NULL);
4381 assert(image->signature == MagickSignature);
4382 assert(image->cache != (Cache) NULL);
4383 cache_info=(CacheInfo *) GetImagePixelCache(image,MagickTrue,exception);
4384 assert(cache_info->signature == MagickSignature);
4385 if (cache_info == (Cache) NULL)
4386 return((Quantum *) NULL);
4387 if ((cache_info->columns == 0) && (cache_info->rows == 0))
4389 (void) ThrowMagickException(exception,GetMagickModule(),CacheError,
4390 "NoPixelsDefinedInCache","`%s'",image->filename);
4391 return((Quantum *) NULL);
4393 if ((x < 0) || (y < 0) || (x >= (ssize_t) cache_info->columns) ||
4394 (y >= (ssize_t) cache_info->rows))
4396 (void) ThrowMagickException(exception,GetMagickModule(),CacheError,
4397 "PixelsAreNotAuthentic","`%s'",image->filename);
4398 return((Quantum *) NULL);
4400 offset=(MagickOffsetType) y*cache_info->columns+x;
4402 return((Quantum *) NULL);
4403 number_pixels=(MagickSizeType) cache_info->columns*cache_info->rows;
4404 offset+=(MagickOffsetType) (rows-1)*cache_info->columns+columns-1;
4405 if ((MagickSizeType) offset >= number_pixels)
4406 return((Quantum *) NULL);
4412 region.width=columns;
4414 return(SetPixelCacheNexusPixels(image,®ion,nexus_info,exception));
4418 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4422 + Q u e u e A u t h e n t i c P i x e l s C a c h e %
4426 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4428 % QueueAuthenticPixelsCache() allocates an region to store image pixels as
4429 % defined by the region rectangle and returns a pointer to the region. This
4430 % region is subsequently transferred from the pixel cache with
4431 % SyncAuthenticPixelsCache(). A pointer to the pixels is returned if the
4432 % pixels are transferred, otherwise a NULL is returned.
4434 % The format of the QueueAuthenticPixelsCache() method is:
4436 % Quantum *QueueAuthenticPixelsCache(Image *image,const ssize_t x,
4437 % const ssize_t y,const size_t columns,const size_t rows,
4438 % ExceptionInfo *exception)
4440 % A description of each parameter follows:
4442 % o image: the image.
4444 % o x,y,columns,rows: These values define the perimeter of a region of
4447 % o exception: return any errors or warnings in this structure.
4450 static Quantum *QueueAuthenticPixelsCache(Image *image,const ssize_t x,
4451 const ssize_t y,const size_t columns,const size_t rows,
4452 ExceptionInfo *exception)
4458 id = GetOpenMPThreadId();
4463 assert(image != (const Image *) NULL);
4464 assert(image->signature == MagickSignature);
4465 assert(image->cache != (Cache) NULL);
4466 cache_info=(CacheInfo *) image->cache;
4467 assert(cache_info->signature == MagickSignature);
4468 assert(id < (int) cache_info->number_threads);
4469 pixels=QueueAuthenticNexus(image,x,y,columns,rows,cache_info->nexus_info[id],
4475 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4479 % Q u e u e A u t h e n t i c P i x e l s %
4483 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4485 % QueueAuthenticPixels() queues a mutable pixel region. If the region is
4486 % successfully initialized a pointer to a Quantum array representing the
4487 % region is returned, otherwise NULL is returned. The returned pointer may
4488 % point to a temporary working buffer for the pixels or it may point to the
4489 % final location of the pixels in memory.
4491 % Write-only access means that any existing pixel values corresponding to
4492 % the region are ignored. This is useful if the initial image is being
4493 % created from scratch, or if the existing pixel values are to be
4494 % completely replaced without need to refer to their pre-existing values.
4495 % The application is free to read and write the pixel buffer returned by
4496 % QueueAuthenticPixels() any way it pleases. QueueAuthenticPixels() does not
4497 % initialize the pixel array values. Initializing pixel array values is the
4498 % application's responsibility.
4500 % Performance is maximized if the selected region is part of one row, or
4501 % one or more full rows, since then there is opportunity to access the
4502 % pixels in-place (without a copy) if the image is in memory, or in a
4503 % memory-mapped file. The returned pointer must *never* be deallocated
4506 % Pixels accessed via the returned pointer represent a simple array of type
4507 % Quantum. If the image type is CMYK or the storage class is PseudoClass,
4508 % call GetAuthenticMetacontent() after invoking GetAuthenticPixels() to
4509 % obtain the meta-content (of type void) corresponding to the region.
4510 % Once the Quantum (and/or Quantum) array has been updated, the
4511 % changes must be saved back to the underlying image using
4512 % SyncAuthenticPixels() or they may be lost.
4514 % The format of the QueueAuthenticPixels() method is:
4516 % Quantum *QueueAuthenticPixels(Image *image,const ssize_t x,
4517 % const ssize_t y,const size_t columns,const size_t rows,
4518 % ExceptionInfo *exception)
4520 % A description of each parameter follows:
4522 % o image: the image.
4524 % o x,y,columns,rows: These values define the perimeter of a region of
4527 % o exception: return any errors or warnings in this structure.
4530 MagickExport Quantum *QueueAuthenticPixels(Image *image,const ssize_t x,
4531 const ssize_t y,const size_t columns,const size_t rows,
4532 ExceptionInfo *exception)
4538 id = GetOpenMPThreadId();
4543 assert(image != (Image *) NULL);
4544 assert(image->signature == MagickSignature);
4545 assert(image->cache != (Cache) NULL);
4546 cache_info=(CacheInfo *) image->cache;
4547 assert(cache_info->signature == MagickSignature);
4548 if (cache_info->methods.queue_authentic_pixels_handler !=
4549 (QueueAuthenticPixelsHandler) NULL)
4551 pixels=cache_info->methods.queue_authentic_pixels_handler(image,x,y,
4552 columns,rows,exception);
4555 assert(id < (int) cache_info->number_threads);
4556 pixels=QueueAuthenticNexus(image,x,y,columns,rows,cache_info->nexus_info[id],
4562 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4566 + R e a d P i x e l C a c h e M e t a c o n t e n t %
4570 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4572 % ReadPixelCacheMetacontent() reads metacontent from the specified region of
4575 % The format of the ReadPixelCacheMetacontent() method is:
4577 % MagickBooleanType ReadPixelCacheMetacontent(CacheInfo *cache_info,
4578 % NexusInfo *nexus_info,ExceptionInfo *exception)
4580 % A description of each parameter follows:
4582 % o cache_info: the pixel cache.
4584 % o nexus_info: the cache nexus to read the metacontent.
4586 % o exception: return any errors or warnings in this structure.
4589 static MagickBooleanType ReadPixelCacheMetacontent(CacheInfo *cache_info,
4590 NexusInfo *nexus_info,ExceptionInfo *exception)
4603 register unsigned char
4609 if (cache_info->metacontent_extent == 0)
4610 return(MagickFalse);
4611 if (IsPixelAuthentic(cache_info,nexus_info) != MagickFalse)
4613 offset=(MagickOffsetType) nexus_info->region.y*cache_info->columns+
4614 nexus_info->region.x;
4615 length=(MagickSizeType) nexus_info->region.width*
4616 cache_info->metacontent_extent;
4617 rows=nexus_info->region.height;
4619 q=(unsigned char *) nexus_info->metacontent;
4620 switch (cache_info->type)
4625 register unsigned char
4629 Read meta-content from memory.
4631 if ((cache_info->columns == nexus_info->region.width) &&
4632 (extent == (MagickSizeType) ((size_t) extent)))
4637 p=(unsigned char *) cache_info->metacontent+offset*
4638 cache_info->metacontent_extent;
4639 for (y=0; y < (ssize_t) rows; y++)
4641 (void) memcpy(q,p,(size_t) length);
4642 p+=cache_info->metacontent_extent*cache_info->columns;
4643 q+=cache_info->metacontent_extent*nexus_info->region.width;
4650 Read meta content from disk.
4652 if (OpenPixelCacheOnDisk(cache_info,IOMode) == MagickFalse)
4654 ThrowFileException(exception,FileOpenError,"UnableToOpenFile",
4655 cache_info->cache_filename);
4656 return(MagickFalse);
4658 if ((cache_info->columns == nexus_info->region.width) &&
4659 (extent <= MagickMaxBufferExtent))
4664 extent=(MagickSizeType) cache_info->columns*cache_info->rows;
4665 for (y=0; y < (ssize_t) rows; y++)
4667 count=ReadPixelCacheRegion(cache_info,cache_info->offset+extent*
4668 cache_info->number_channels*sizeof(Quantum)+offset*
4669 cache_info->metacontent_extent,length,(unsigned char *) q);
4670 if ((MagickSizeType) count != length)
4672 offset+=cache_info->columns;
4673 q+=cache_info->metacontent_extent*nexus_info->region.width;
4675 if (y < (ssize_t) rows)
4677 ThrowFileException(exception,CacheError,"UnableToReadPixelCache",
4678 cache_info->cache_filename);
4679 return(MagickFalse);
4686 if ((cache_info->debug != MagickFalse) &&
4687 (CacheTick(nexus_info->region.y,cache_info->rows) != MagickFalse))
4688 (void) LogMagickEvent(CacheEvent,GetMagickModule(),
4689 "%s[%.20gx%.20g%+.20g%+.20g]",cache_info->filename,(double)
4690 nexus_info->region.width,(double) nexus_info->region.height,(double)
4691 nexus_info->region.x,(double) nexus_info->region.y);
4696 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4700 + R e a d P i x e l C a c h e P i x e l s %
4704 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4706 % ReadPixelCachePixels() reads pixels from the specified region of the pixel
4709 % The format of the ReadPixelCachePixels() method is:
4711 % MagickBooleanType ReadPixelCachePixels(CacheInfo *cache_info,
4712 % NexusInfo *nexus_info,ExceptionInfo *exception)
4714 % A description of each parameter follows:
4716 % o cache_info: the pixel cache.
4718 % o nexus_info: the cache nexus to read the pixels.
4720 % o exception: return any errors or warnings in this structure.
4723 static MagickBooleanType ReadPixelCachePixels(CacheInfo *cache_info,
4724 NexusInfo *nexus_info,ExceptionInfo *exception)
4743 if (IsPixelAuthentic(cache_info,nexus_info) != MagickFalse)
4745 offset=(MagickOffsetType) nexus_info->region.y*cache_info->columns+
4746 nexus_info->region.x;
4747 length=(MagickSizeType) nexus_info->region.width*cache_info->number_channels*
4749 rows=nexus_info->region.height;
4751 q=nexus_info->pixels;
4752 switch (cache_info->type)
4761 Read pixels from memory.
4763 if ((cache_info->columns == nexus_info->region.width) &&
4764 (extent == (MagickSizeType) ((size_t) extent)))
4769 p=cache_info->pixels+offset*cache_info->number_channels;
4770 for (y=0; y < (ssize_t) rows; y++)
4772 (void) memcpy(q,p,(size_t) length);
4773 p+=cache_info->number_channels*cache_info->columns;
4774 q+=cache_info->number_channels*nexus_info->region.width;
4781 Read pixels from disk.
4783 if (OpenPixelCacheOnDisk(cache_info,IOMode) == MagickFalse)
4785 ThrowFileException(exception,FileOpenError,"UnableToOpenFile",
4786 cache_info->cache_filename);
4787 return(MagickFalse);
4789 if ((cache_info->columns == nexus_info->region.width) &&
4790 (extent <= MagickMaxBufferExtent))
4795 for (y=0; y < (ssize_t) rows; y++)
4797 count=ReadPixelCacheRegion(cache_info,cache_info->offset+offset*
4798 cache_info->number_channels*sizeof(*q),length,(unsigned char *) q);
4799 if ((MagickSizeType) count != length)
4801 offset+=cache_info->columns;
4802 q+=cache_info->number_channels*nexus_info->region.width;
4804 if (y < (ssize_t) rows)
4806 ThrowFileException(exception,CacheError,"UnableToReadPixelCache",
4807 cache_info->cache_filename);
4808 return(MagickFalse);
4815 if ((cache_info->debug != MagickFalse) &&
4816 (CacheTick(nexus_info->region.y,cache_info->rows) != MagickFalse))
4817 (void) LogMagickEvent(CacheEvent,GetMagickModule(),
4818 "%s[%.20gx%.20g%+.20g%+.20g]",cache_info->filename,(double)
4819 nexus_info->region.width,(double) nexus_info->region.height,(double)
4820 nexus_info->region.x,(double) nexus_info->region.y);
4825 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4829 + R e f e r e n c e P i x e l C a c h e %
4833 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4835 % ReferencePixelCache() increments the reference count associated with the
4836 % pixel cache returning a pointer to the cache.
4838 % The format of the ReferencePixelCache method is:
4840 % Cache ReferencePixelCache(Cache cache_info)
4842 % A description of each parameter follows:
4844 % o cache_info: the pixel cache.
4847 MagickExport Cache ReferencePixelCache(Cache cache)
4852 assert(cache != (Cache *) NULL);
4853 cache_info=(CacheInfo *) cache;
4854 assert(cache_info->signature == MagickSignature);
4855 LockSemaphoreInfo(cache_info->semaphore);
4856 cache_info->reference_count++;
4857 UnlockSemaphoreInfo(cache_info->semaphore);
4862 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4866 + S e t P i x e l C a c h e M e t h o d s %
4870 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4872 % SetPixelCacheMethods() sets the image pixel methods to the specified ones.
4874 % The format of the SetPixelCacheMethods() method is:
4876 % SetPixelCacheMethods(Cache *,CacheMethods *cache_methods)
4878 % A description of each parameter follows:
4880 % o cache: the pixel cache.
4882 % o cache_methods: Specifies a pointer to a CacheMethods structure.
4885 MagickExport void SetPixelCacheMethods(Cache cache,CacheMethods *cache_methods)
4890 GetOneAuthenticPixelFromHandler
4891 get_one_authentic_pixel_from_handler;
4893 GetOneVirtualPixelFromHandler
4894 get_one_virtual_pixel_from_handler;
4897 Set cache pixel methods.
4899 assert(cache != (Cache) NULL);
4900 assert(cache_methods != (CacheMethods *) NULL);
4901 cache_info=(CacheInfo *) cache;
4902 assert(cache_info->signature == MagickSignature);
4903 if (cache_info->debug != MagickFalse)
4904 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",
4905 cache_info->filename);
4906 if (cache_methods->get_virtual_pixel_handler != (GetVirtualPixelHandler) NULL)
4907 cache_info->methods.get_virtual_pixel_handler=
4908 cache_methods->get_virtual_pixel_handler;
4909 if (cache_methods->destroy_pixel_handler != (DestroyPixelHandler) NULL)
4910 cache_info->methods.destroy_pixel_handler=
4911 cache_methods->destroy_pixel_handler;
4912 if (cache_methods->get_virtual_metacontent_from_handler !=
4913 (GetVirtualMetacontentFromHandler) NULL)
4914 cache_info->methods.get_virtual_metacontent_from_handler=
4915 cache_methods->get_virtual_metacontent_from_handler;
4916 if (cache_methods->get_authentic_pixels_handler !=
4917 (GetAuthenticPixelsHandler) NULL)
4918 cache_info->methods.get_authentic_pixels_handler=
4919 cache_methods->get_authentic_pixels_handler;
4920 if (cache_methods->queue_authentic_pixels_handler !=
4921 (QueueAuthenticPixelsHandler) NULL)
4922 cache_info->methods.queue_authentic_pixels_handler=
4923 cache_methods->queue_authentic_pixels_handler;
4924 if (cache_methods->sync_authentic_pixels_handler !=
4925 (SyncAuthenticPixelsHandler) NULL)
4926 cache_info->methods.sync_authentic_pixels_handler=
4927 cache_methods->sync_authentic_pixels_handler;
4928 if (cache_methods->get_authentic_pixels_from_handler !=
4929 (GetAuthenticPixelsFromHandler) NULL)
4930 cache_info->methods.get_authentic_pixels_from_handler=
4931 cache_methods->get_authentic_pixels_from_handler;
4932 if (cache_methods->get_authentic_metacontent_from_handler !=
4933 (GetAuthenticMetacontentFromHandler) NULL)
4934 cache_info->methods.get_authentic_metacontent_from_handler=
4935 cache_methods->get_authentic_metacontent_from_handler;
4936 get_one_virtual_pixel_from_handler=
4937 cache_info->methods.get_one_virtual_pixel_from_handler;
4938 if (get_one_virtual_pixel_from_handler !=
4939 (GetOneVirtualPixelFromHandler) NULL)
4940 cache_info->methods.get_one_virtual_pixel_from_handler=
4941 cache_methods->get_one_virtual_pixel_from_handler;
4942 get_one_authentic_pixel_from_handler=
4943 cache_methods->get_one_authentic_pixel_from_handler;
4944 if (get_one_authentic_pixel_from_handler !=
4945 (GetOneAuthenticPixelFromHandler) NULL)
4946 cache_info->methods.get_one_authentic_pixel_from_handler=
4947 cache_methods->get_one_authentic_pixel_from_handler;
4951 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4955 + S e t P i x e l C a c h e N e x u s P i x e l s %
4959 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
4961 % SetPixelCacheNexusPixels() defines the region of the cache for the
4962 % specified cache nexus.
4964 % The format of the SetPixelCacheNexusPixels() method is:
4966 % Quantum SetPixelCacheNexusPixels(const Image *image,
4967 % const RectangleInfo *region,NexusInfo *nexus_info,
4968 % ExceptionInfo *exception)
4970 % A description of each parameter follows:
4972 % o image: the image.
4974 % o region: A pointer to the RectangleInfo structure that defines the
4975 % region of this particular cache nexus.
4977 % o nexus_info: the cache nexus to set.
4979 % o exception: return any errors or warnings in this structure.
4983 static inline MagickBooleanType AcquireCacheNexusPixels(CacheInfo *cache_info,
4984 NexusInfo *nexus_info,ExceptionInfo *exception)
4986 if (nexus_info->length != (MagickSizeType) ((size_t) nexus_info->length))
4987 return(MagickFalse);
4988 nexus_info->mapped=MagickFalse;
4989 nexus_info->cache=(Quantum *) AcquireAlignedMemory(1,(size_t)
4990 nexus_info->length);
4991 if (nexus_info->cache == (Quantum *) NULL)
4993 nexus_info->mapped=MagickTrue;
4994 nexus_info->cache=(Quantum *) MapBlob(-1,IOMode,0,(size_t)
4995 nexus_info->length);
4997 if (nexus_info->cache == (Quantum *) NULL)
4999 (void) ThrowMagickException(exception,GetMagickModule(),
5000 ResourceLimitError,"MemoryAllocationFailed","`%s'",
5001 cache_info->filename);
5002 return(MagickFalse);
5007 static Quantum *SetPixelCacheNexusPixels(const Image *image,
5008 const RectangleInfo *region,NexusInfo *nexus_info,ExceptionInfo *exception)
5020 cache_info=(CacheInfo *) image->cache;
5021 assert(cache_info->signature == MagickSignature);
5022 if (cache_info->type == UndefinedCache)
5023 return((Quantum *) NULL);
5024 nexus_info->region=(*region);
5025 if ((cache_info->type != DiskCache) && (cache_info->type != PingCache) &&
5026 (image->clip_mask == (Image *) NULL) && (image->mask == (Image *) NULL))
5032 x=nexus_info->region.x+(ssize_t) nexus_info->region.width-1;
5033 y=nexus_info->region.y+(ssize_t) nexus_info->region.height-1;
5034 if (((nexus_info->region.x >= 0) && (x < (ssize_t) cache_info->columns) &&
5035 (nexus_info->region.y >= 0) && (y < (ssize_t) cache_info->rows)) &&
5036 ((nexus_info->region.height == 1UL) || ((nexus_info->region.x == 0) &&
5037 ((nexus_info->region.width == cache_info->columns) ||
5038 ((nexus_info->region.width % cache_info->columns) == 0)))))
5044 Pixels are accessed directly from memory.
5046 offset=(MagickOffsetType) nexus_info->region.y*cache_info->columns+
5047 nexus_info->region.x;
5048 nexus_info->pixels=cache_info->pixels+cache_info->number_channels*
5050 nexus_info->metacontent=(void *) NULL;
5051 if (cache_info->metacontent_extent != 0)
5052 nexus_info->metacontent=(unsigned char *) cache_info->metacontent+
5053 offset*cache_info->metacontent_extent;
5054 return(nexus_info->pixels);
5058 Pixels are stored in a cache region until they are synced to the cache.
5060 number_pixels=(MagickSizeType) nexus_info->region.width*
5061 nexus_info->region.height;
5062 length=number_pixels*cache_info->number_channels*sizeof(Quantum);
5063 if (cache_info->metacontent_extent != 0)
5064 length+=number_pixels*cache_info->metacontent_extent;
5065 if (nexus_info->cache == (Quantum *) NULL)
5067 nexus_info->length=length;
5068 status=AcquireCacheNexusPixels(cache_info,nexus_info,exception);
5069 if (status == MagickFalse)
5071 nexus_info->length=0;
5072 return((Quantum *) NULL);
5076 if (nexus_info->length != length)
5078 RelinquishCacheNexusPixels(nexus_info);
5079 nexus_info->length=length;
5080 status=AcquireCacheNexusPixels(cache_info,nexus_info,exception);
5081 if (status == MagickFalse)
5083 nexus_info->length=0;
5084 return((Quantum *) NULL);
5087 nexus_info->pixels=nexus_info->cache;
5088 nexus_info->metacontent=(void *) NULL;
5089 if (cache_info->metacontent_extent != 0)
5090 nexus_info->metacontent=(void *) (nexus_info->pixels+number_pixels*
5091 cache_info->number_channels);
5092 return(nexus_info->pixels);
5096 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5100 % S e t P i x e l C a c h e V i r t u a l M e t h o d %
5104 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5106 % SetPixelCacheVirtualMethod() sets the "virtual pixels" method for the
5107 % pixel cache and returns the previous setting. A virtual pixel is any pixel
5108 % access that is outside the boundaries of the image cache.
5110 % The format of the SetPixelCacheVirtualMethod() method is:
5112 % VirtualPixelMethod SetPixelCacheVirtualMethod(const Image *image,
5113 % const VirtualPixelMethod virtual_pixel_method)
5115 % A description of each parameter follows:
5117 % o image: the image.
5119 % o virtual_pixel_method: choose the type of virtual pixel.
5122 MagickExport VirtualPixelMethod SetPixelCacheVirtualMethod(const Image *image,
5123 const VirtualPixelMethod virtual_pixel_method)
5131 assert(image != (Image *) NULL);
5132 assert(image->signature == MagickSignature);
5133 if (image->debug != MagickFalse)
5134 (void) LogMagickEvent(TraceEvent,GetMagickModule(),"%s",image->filename);
5135 assert(image->cache != (Cache) NULL);
5136 cache_info=(CacheInfo *) image->cache;
5137 assert(cache_info->signature == MagickSignature);
5138 method=cache_info->virtual_pixel_method;
5139 cache_info->virtual_pixel_method=virtual_pixel_method;
5144 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5148 + S y n c A u t h e n t i c P i x e l C a c h e N e x u s %
5152 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5154 % SyncAuthenticPixelCacheNexus() saves the authentic image pixels to the
5155 % in-memory or disk cache. The method returns MagickTrue if the pixel region
5156 % is synced, otherwise MagickFalse.
5158 % The format of the SyncAuthenticPixelCacheNexus() method is:
5160 % MagickBooleanType SyncAuthenticPixelCacheNexus(Image *image,
5161 % NexusInfo *nexus_info,ExceptionInfo *exception)
5163 % A description of each parameter follows:
5165 % o image: the image.
5167 % o nexus_info: the cache nexus to sync.
5169 % o exception: return any errors or warnings in this structure.
5172 MagickExport MagickBooleanType SyncAuthenticPixelCacheNexus(Image *image,
5173 NexusInfo *nexus_info,ExceptionInfo *exception)
5182 Transfer pixels to the cache.
5184 assert(image != (Image *) NULL);
5185 assert(image->signature == MagickSignature);
5186 if (image->cache == (Cache) NULL)
5187 ThrowBinaryException(CacheError,"PixelCacheIsNotOpen",image->filename);
5188 cache_info=(CacheInfo *) image->cache;
5189 assert(cache_info->signature == MagickSignature);
5190 if (cache_info->type == UndefinedCache)
5191 return(MagickFalse);
5192 if ((image->clip_mask != (Image *) NULL) &&
5193 (ClipPixelCacheNexus(image,nexus_info,exception) == MagickFalse))
5194 return(MagickFalse);
5195 if ((image->mask != (Image *) NULL) &&
5196 (MaskPixelCacheNexus(image,nexus_info,exception) == MagickFalse))
5197 return(MagickFalse);
5198 if (IsPixelAuthentic(cache_info,nexus_info) != MagickFalse)
5200 assert(cache_info->signature == MagickSignature);
5201 status=WritePixelCachePixels(cache_info,nexus_info,exception);
5202 if ((cache_info->metacontent_extent != 0) &&
5203 (WritePixelCacheMetacontent(cache_info,nexus_info,exception) == MagickFalse))
5204 return(MagickFalse);
5209 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5213 + S y n c A u t h e n t i c P i x e l C a c h e %
5217 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5219 % SyncAuthenticPixelsCache() saves the authentic image pixels to the in-memory
5220 % or disk cache. The method returns MagickTrue if the pixel region is synced,
5221 % otherwise MagickFalse.
5223 % The format of the SyncAuthenticPixelsCache() method is:
5225 % MagickBooleanType SyncAuthenticPixelsCache(Image *image,
5226 % ExceptionInfo *exception)
5228 % A description of each parameter follows:
5230 % o image: the image.
5232 % o exception: return any errors or warnings in this structure.
5235 static MagickBooleanType SyncAuthenticPixelsCache(Image *image,
5236 ExceptionInfo *exception)
5242 id = GetOpenMPThreadId();
5247 assert(image != (Image *) NULL);
5248 assert(image->signature == MagickSignature);
5249 assert(image->cache != (Cache) NULL);
5250 cache_info=(CacheInfo *) image->cache;
5251 assert(cache_info->signature == MagickSignature);
5252 assert(id < (int) cache_info->number_threads);
5253 status=SyncAuthenticPixelCacheNexus(image,cache_info->nexus_info[id],
5259 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5263 % S y n c A u t h e n t i c P i x e l s %
5267 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5269 % SyncAuthenticPixels() saves the image pixels to the in-memory or disk cache.
5270 % The method returns MagickTrue if the pixel region is flushed, otherwise
5273 % The format of the SyncAuthenticPixels() method is:
5275 % MagickBooleanType SyncAuthenticPixels(Image *image,
5276 % ExceptionInfo *exception)
5278 % A description of each parameter follows:
5280 % o image: the image.
5282 % o exception: return any errors or warnings in this structure.
5285 MagickExport MagickBooleanType SyncAuthenticPixels(Image *image,
5286 ExceptionInfo *exception)
5292 id = GetOpenMPThreadId();
5297 assert(image != (Image *) NULL);
5298 assert(image->signature == MagickSignature);
5299 assert(image->cache != (Cache) NULL);
5300 cache_info=(CacheInfo *) image->cache;
5301 assert(cache_info->signature == MagickSignature);
5302 if (cache_info->methods.sync_authentic_pixels_handler !=
5303 (SyncAuthenticPixelsHandler) NULL)
5305 status=cache_info->methods.sync_authentic_pixels_handler(image,
5309 assert(id < (int) cache_info->number_threads);
5310 status=SyncAuthenticPixelCacheNexus(image,cache_info->nexus_info[id],
5316 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5320 + W r i t e P i x e l C a c h e M e t a c o n t e n t %
5324 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5326 % WritePixelCacheMetacontent() writes the meta-content to the specified region
5327 % of the pixel cache.
5329 % The format of the WritePixelCacheMetacontent() method is:
5331 % MagickBooleanType WritePixelCacheMetacontent(CacheInfo *cache_info,
5332 % NexusInfo *nexus_info,ExceptionInfo *exception)
5334 % A description of each parameter follows:
5336 % o cache_info: the pixel cache.
5338 % o nexus_info: the cache nexus to write the meta-content.
5340 % o exception: return any errors or warnings in this structure.
5343 static MagickBooleanType WritePixelCacheMetacontent(CacheInfo *cache_info,
5344 NexusInfo *nexus_info,ExceptionInfo *exception)
5354 register const unsigned char
5363 if (cache_info->metacontent_extent == 0)
5364 return(MagickFalse);
5365 if (IsPixelAuthentic(cache_info,nexus_info) != MagickFalse)
5367 offset=(MagickOffsetType) nexus_info->region.y*cache_info->columns+
5368 nexus_info->region.x;
5369 length=(MagickSizeType) nexus_info->region.width*
5370 cache_info->metacontent_extent;
5371 rows=nexus_info->region.height;
5372 extent=(MagickSizeType) length*rows;
5373 p=(unsigned char *) nexus_info->metacontent;
5374 switch (cache_info->type)
5379 register unsigned char
5383 Write associated pixels to memory.
5385 if ((cache_info->columns == nexus_info->region.width) &&
5386 (extent == (MagickSizeType) ((size_t) extent)))
5391 q=(unsigned char *) cache_info->metacontent+offset*
5392 cache_info->metacontent_extent;
5393 for (y=0; y < (ssize_t) rows; y++)
5395 (void) memcpy(q,p,(size_t) length);
5396 p+=nexus_info->region.width*cache_info->metacontent_extent;
5397 q+=cache_info->columns*cache_info->metacontent_extent;
5404 Write associated pixels to disk.
5406 if (OpenPixelCacheOnDisk(cache_info,IOMode) == MagickFalse)
5408 ThrowFileException(exception,FileOpenError,"UnableToOpenFile",
5409 cache_info->cache_filename);
5410 return(MagickFalse);
5412 if ((cache_info->columns == nexus_info->region.width) &&
5413 (extent <= MagickMaxBufferExtent))
5418 extent=(MagickSizeType) cache_info->columns*cache_info->rows;
5419 for (y=0; y < (ssize_t) rows; y++)
5421 count=WritePixelCacheRegion(cache_info,cache_info->offset+extent*
5422 cache_info->number_channels*sizeof(Quantum)+offset*
5423 cache_info->metacontent_extent,length,(const unsigned char *) p);
5424 if ((MagickSizeType) count != length)
5426 p+=nexus_info->region.width*cache_info->metacontent_extent;
5427 offset+=cache_info->columns;
5429 if (y < (ssize_t) rows)
5431 ThrowFileException(exception,CacheError,"UnableToWritePixelCache",
5432 cache_info->cache_filename);
5433 return(MagickFalse);
5440 if ((cache_info->debug != MagickFalse) &&
5441 (CacheTick(nexus_info->region.y,cache_info->rows) != MagickFalse))
5442 (void) LogMagickEvent(CacheEvent,GetMagickModule(),
5443 "%s[%.20gx%.20g%+.20g%+.20g]",cache_info->filename,(double)
5444 nexus_info->region.width,(double) nexus_info->region.height,(double)
5445 nexus_info->region.x,(double) nexus_info->region.y);
5450 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5454 + W r i t e C a c h e P i x e l s %
5458 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
5460 % WritePixelCachePixels() writes image pixels to the specified region of the
5463 % The format of the WritePixelCachePixels() method is:
5465 % MagickBooleanType WritePixelCachePixels(CacheInfo *cache_info,
5466 % NexusInfo *nexus_info,ExceptionInfo *exception)
5468 % A description of each parameter follows:
5470 % o cache_info: the pixel cache.
5472 % o nexus_info: the cache nexus to write the pixels.
5474 % o exception: return any errors or warnings in this structure.
5477 static MagickBooleanType WritePixelCachePixels(CacheInfo *cache_info,
5478 NexusInfo *nexus_info,ExceptionInfo *exception)
5488 register const Quantum
5497 if (IsPixelAuthentic(cache_info,nexus_info) != MagickFalse)
5499 offset=(MagickOffsetType) nexus_info->region.y*cache_info->columns+
5500 nexus_info->region.x;
5501 length=(MagickSizeType) nexus_info->region.width*cache_info->number_channels*
5503 rows=nexus_info->region.height;
5505 p=nexus_info->pixels;
5506 switch (cache_info->type)
5515 Write pixels to memory.
5517 if ((cache_info->columns == nexus_info->region.width) &&
5518 (extent == (MagickSizeType) ((size_t) extent)))
5523 q=cache_info->pixels+offset*cache_info->number_channels;
5524 for (y=0; y < (ssize_t) rows; y++)
5526 (void) memcpy(q,p,(size_t) length);
5527 p+=nexus_info->region.width*cache_info->number_channels;
5528 q+=cache_info->columns*cache_info->number_channels;
5535 Write pixels to disk.
5537 if (OpenPixelCacheOnDisk(cache_info,IOMode) == MagickFalse)
5539 ThrowFileException(exception,FileOpenError,"UnableToOpenFile",
5540 cache_info->cache_filename);
5541 return(MagickFalse);
5543 if ((cache_info->columns == nexus_info->region.width) &&
5544 (extent <= MagickMaxBufferExtent))
5549 for (y=0; y < (ssize_t) rows; y++)
5551 count=WritePixelCacheRegion(cache_info,cache_info->offset+offset*
5552 cache_info->number_channels*sizeof(*p),length,(const unsigned char *)
5554 if ((MagickSizeType) count != length)
5556 p+=nexus_info->region.width*cache_info->number_channels;
5557 offset+=cache_info->columns;
5559 if (y < (ssize_t) rows)
5561 ThrowFileException(exception,CacheError,"UnableToWritePixelCache",
5562 cache_info->cache_filename);
5563 return(MagickFalse);
5570 if ((cache_info->debug != MagickFalse) &&
5571 (CacheTick(nexus_info->region.y,cache_info->rows) != MagickFalse))
5572 (void) LogMagickEvent(CacheEvent,GetMagickModule(),
5573 "%s[%.20gx%.20g%+.20g%+.20g]",cache_info->filename,(double)
5574 nexus_info->region.width,(double) nexus_info->region.height,(double)
5575 nexus_info->region.x,(double) nexus_info->region.y);