83#include "MagickCore/studio.h"
84#include "MagickCore/blob.h"
85#include "MagickCore/blob-private.h"
86#include "MagickCore/exception.h"
87#include "MagickCore/exception-private.h"
88#include "MagickCore/image-private.h"
89#include "MagickCore/memory_.h"
90#include "MagickCore/memory-private.h"
91#include "MagickCore/policy.h"
92#include "MagickCore/resource_.h"
93#include "MagickCore/semaphore.h"
94#include "MagickCore/string_.h"
95#include "MagickCore/string-private.h"
96#include "MagickCore/utility-private.h"
101#define BlockFooter(block,size) \
102 ((size_t *) ((char *) (block)+(size)-2*sizeof(size_t)))
103#define BlockHeader(block) ((size_t *) (block)-1)
104#define BlockThreshold 1024
105#define MaxBlockExponent 16
106#define MaxBlocks ((BlockThreshold/(4*sizeof(size_t)))+MaxBlockExponent+1)
107#define MaxSegments 1024
108#define NextBlock(block) ((char *) (block)+SizeOfBlock(block))
109#define NextBlockInList(block) (*(void **) (block))
110#define PreviousBlock(block) ((char *) (block)-(*((size_t *) (block)-2)))
111#define PreviousBlockBit 0x01
112#define PreviousBlockInList(block) (*((void **) (block)+1))
113#define SegmentSize (2*1024*1024)
114#define SizeMask (~0x01)
115#define SizeOfBlock(block) (*BlockHeader(block) & SizeMask)
122 UndefinedVirtualMemory,
123 AlignedVirtualMemory,
125 UnalignedVirtualMemory
148 acquire_memory_handler;
151 resize_memory_handler;
154 destroy_memory_handler;
156 AcquireAlignedMemoryHandler
157 acquire_aligned_memory_handler;
159 RelinquishAlignedMemoryHandler
160 relinquish_aligned_memory_handler;
161} MagickMemoryMethods;
166 filename[MagickPathExtent];
187 *blocks[MaxBlocks+1];
193 *segments[MaxSegments],
194 segment_pool[MaxSegments];
201 max_memory_request = 0,
202 max_profile_size = 0,
203 virtual_anonymous_memory = 0;
205static MagickMemoryMethods
208 (AcquireMemoryHandler) malloc,
209 (ResizeMemoryHandler) realloc,
210 (DestroyMemoryHandler) free,
211 (AcquireAlignedMemoryHandler) NULL,
212 (RelinquishAlignedMemoryHandler) NULL
214#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
221static volatile DataSegmentInfo
222 *free_segments = (DataSegmentInfo *) NULL;
227static MagickBooleanType
256#if defined(MAGICKCORE_HAVE_ALIGNED_MALLOC)
257#define AcquireAlignedMemory_Actual AcquireAlignedMemory_STDC
258static inline void *AcquireAlignedMemory_STDC(
const size_t size)
261 extent = CACHE_ALIGNED(size);
268 return(aligned_alloc(CACHE_LINE_SIZE,extent));
270#elif defined(MAGICKCORE_HAVE_POSIX_MEMALIGN)
271#define AcquireAlignedMemory_Actual AcquireAlignedMemory_POSIX
272static inline void *AcquireAlignedMemory_POSIX(
const size_t size)
277 if (posix_memalign(&memory,CACHE_LINE_SIZE,size))
281#elif defined(MAGICKCORE_HAVE__ALIGNED_MALLOC)
282#define AcquireAlignedMemory_Actual AcquireAlignedMemory_WinAPI
283static inline void *AcquireAlignedMemory_WinAPI(
const size_t size)
285 return(_aligned_malloc(size,CACHE_LINE_SIZE));
288#define ALIGNMENT_OVERHEAD \
289 (MAGICKCORE_MAX_ALIGNMENT_PADDING(CACHE_LINE_SIZE) + MAGICKCORE_SIZEOF_VOID_P)
290static inline void *reserve_space_for_actual_base_address(
void *
const p)
292 return((
void **) p+1);
295static inline void **pointer_to_space_for_actual_base_address(
void *
const p)
297 return((
void **) p-1);
300static inline void *actual_base_address(
void *
const p)
302 return(*pointer_to_space_for_actual_base_address(p));
305static inline void *align_to_cache(
void *
const p)
307 return((
void *) CACHE_ALIGNED((MagickAddressType) p));
310static inline void *adjust(
void *
const p)
312 return(align_to_cache(reserve_space_for_actual_base_address(p)));
315#define AcquireAlignedMemory_Actual AcquireAlignedMemory_Generic
316static inline void *AcquireAlignedMemory_Generic(
const size_t size)
325 #if SIZE_MAX < ALIGNMENT_OVERHEAD
326 #error "CACHE_LINE_SIZE is way too big."
328 extent=(size+ALIGNMENT_OVERHEAD);
334 p=AcquireMagickMemory(extent);
338 *pointer_to_space_for_actual_base_address(memory)=p;
343MagickExport
void *AcquireAlignedMemory(
const size_t count,
const size_t quantum)
348 if ((HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse) ||
349 (size > GetMaxMemoryRequest()))
354 if (memory_methods.acquire_aligned_memory_handler != (AcquireAlignedMemoryHandler) NULL)
355 return(memory_methods.acquire_aligned_memory_handler(size,CACHE_LINE_SIZE));
356 return(AcquireAlignedMemory_Actual(size));
359#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
384static inline size_t AllocationPolicy(
size_t size)
393 assert(size % (4*
sizeof(
size_t)) == 0);
394 if (size <= BlockThreshold)
395 return(size/(4*
sizeof(
size_t)));
399 if (size > (
size_t) (BlockThreshold*(1L << (MaxBlockExponent-1L))))
400 return(MaxBlocks-1L);
404 blocksize=BlockThreshold/(4*
sizeof(size_t));
405 for ( ; size > BlockThreshold; size/=2)
407 assert(blocksize > (BlockThreshold/(4*
sizeof(
size_t))));
408 assert(blocksize < (MaxBlocks-1L));
412static inline void InsertFreeBlock(
void *block,
const size_t i)
421 size=SizeOfBlock(block);
422 previous=(
void *) NULL;
423 next=memory_pool.blocks[i];
424 while ((next != (
void *) NULL) && (SizeOfBlock(next) < size))
427 next=NextBlockInList(next);
429 PreviousBlockInList(block)=previous;
430 NextBlockInList(block)=next;
431 if (previous != (
void *) NULL)
432 NextBlockInList(previous)=block;
434 memory_pool.blocks[i]=block;
435 if (next != (
void *) NULL)
436 PreviousBlockInList(next)=block;
439static inline void RemoveFreeBlock(
void *block,
const size_t i)
445 next=NextBlockInList(block);
446 previous=PreviousBlockInList(block);
447 if (previous == (
void *) NULL)
448 memory_pool.blocks[i]=next;
450 NextBlockInList(previous)=next;
451 if (next != (
void *) NULL)
452 PreviousBlockInList(next)=previous;
455static void *AcquireBlock(
size_t size)
466 size=(size_t) (size+
sizeof(
size_t)+6*
sizeof(
size_t)-1) & -(4U*
sizeof(size_t));
467 i=AllocationPolicy(size);
468 block=memory_pool.blocks[i];
469 while ((block != (
void *) NULL) && (SizeOfBlock(block) < size))
470 block=NextBlockInList(block);
471 if (block == (
void *) NULL)
474 while (memory_pool.blocks[i] == (
void *) NULL)
476 block=memory_pool.blocks[i];
478 return((
void *) NULL);
480 assert((*BlockHeader(NextBlock(block)) & PreviousBlockBit) == 0);
481 assert(SizeOfBlock(block) >= size);
482 RemoveFreeBlock(block,AllocationPolicy(SizeOfBlock(block)));
483 if (SizeOfBlock(block) > size)
494 next=(
char *) block+size;
495 blocksize=SizeOfBlock(block)-size;
496 *BlockHeader(next)=blocksize;
497 *BlockFooter(next,blocksize)=blocksize;
498 InsertFreeBlock(next,AllocationPolicy(blocksize));
499 *BlockHeader(block)=size | (*BlockHeader(block) & ~SizeMask);
501 assert(size == SizeOfBlock(block));
502 *BlockHeader(NextBlock(block))|=PreviousBlockBit;
503 memory_pool.allocation+=size;
531MagickExport
void *AcquireMagickMemory(
const size_t size)
536#if !defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
537 memory=memory_methods.acquire_memory_handler(size == 0 ? 1UL : size);
540 ActivateSemaphoreInfo(&memory_semaphore);
541 if (free_segments == (DataSegmentInfo *) NULL)
543 LockSemaphoreInfo(memory_semaphore);
544 if (free_segments == (DataSegmentInfo *) NULL)
549 assert(2*
sizeof(
size_t) > (
size_t) (~SizeMask));
550 (void) memset(&memory_pool,0,
sizeof(memory_pool));
551 memory_pool.allocation=SegmentSize;
552 memory_pool.blocks[MaxBlocks]=(
void *) (-1);
553 for (i=0; i < MaxSegments; i++)
556 memory_pool.segment_pool[i].previous=
557 (&memory_pool.segment_pool[i-1]);
558 if (i != (MaxSegments-1))
559 memory_pool.segment_pool[i].next=(&memory_pool.segment_pool[i+1]);
561 free_segments=(&memory_pool.segment_pool[0]);
563 UnlockSemaphoreInfo(memory_semaphore);
565 LockSemaphoreInfo(memory_semaphore);
566 memory=AcquireBlock(size == 0 ? 1UL : size);
567 if (memory == (
void *) NULL)
569 if (ExpandHeap(size == 0 ? 1UL : size) != MagickFalse)
570 memory=AcquireBlock(size == 0 ? 1UL : size);
572 UnlockSemaphoreInfo(memory_semaphore);
605MagickExport
void *AcquireCriticalMemory(
const size_t size)
607#if !defined(STDERR_FILENO)
608#define STDERR_FILENO 2
614 static const char fatal_message[] =
615 "ImageMagick: fatal error: unable to acquire critical memory\n";
623 memory=AcquireMagickMemory(size);
624 if (memory != (
void *) NULL)
626 status=write(STDERR_FILENO,fatal_message,
sizeof(fatal_message)-1);
628 MagickCoreTerminus();
657MagickExport
void *AcquireQuantumMemory(
const size_t count,
const size_t quantum)
662 if ((HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse) ||
663 (size > GetMaxMemoryRequest()))
668 return(AcquireMagickMemory(size));
697MagickExport MemoryInfo *AcquireVirtualMemory(
const size_t count,
698 const size_t quantum)
709 if (HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse)
712 return((MemoryInfo *) NULL);
714 if (virtual_anonymous_memory == 0)
716 virtual_anonymous_memory=1;
717 value=GetPolicyValue(
"system:memory-map");
718 if (LocaleCompare(value,
"anonymous") == 0)
723#if defined(MAGICKCORE_HAVE_MMAP) && defined(MAP_ANONYMOUS)
724 virtual_anonymous_memory=2;
727 value=DestroyString(value);
729 memory_info=(MemoryInfo *) MagickAssumeAligned(AcquireAlignedMemory(1,
730 sizeof(*memory_info)));
731 if (memory_info == (MemoryInfo *) NULL)
732 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
733 (void) memset(memory_info,0,
sizeof(*memory_info));
734 memory_info->length=size;
735 memory_info->signature=MagickCoreSignature;
736 if ((virtual_anonymous_memory == 1) && (size <= GetMaxMemoryRequest()))
738 memory_info->blob=AcquireAlignedMemory(1,size);
739 if (memory_info->blob != NULL)
740 memory_info->type=AlignedVirtualMemory;
742 if (memory_info->blob == NULL)
747 memory_info->blob=NULL;
748 if (size <= GetMaxMemoryRequest())
749 memory_info->blob=MapBlob(-1,IOMode,0,size);
750 if (memory_info->blob != NULL)
751 memory_info->type=MapVirtualMemory;
760 file=AcquireUniqueFileResource(memory_info->filename);
766 offset=(MagickOffsetType) lseek(file,(off_t) (size-1),SEEK_SET);
767 if ((offset == (MagickOffsetType) (size-1)) &&
768 (write(file,
"",1) == 1))
770#if !defined(MAGICKCORE_HAVE_POSIX_FALLOCATE)
771 memory_info->blob=MapBlob(file,IOMode,0,size);
773 if (posix_fallocate(file,0,(MagickOffsetType) size) == 0)
774 memory_info->blob=MapBlob(file,IOMode,0,size);
776 if (memory_info->blob != NULL)
777 memory_info->type=MapVirtualMemory;
780 (void) RelinquishUniqueFileResource(
781 memory_info->filename);
782 *memory_info->filename=
'\0';
785 (void) close_utf8(file);
789 if (memory_info->blob == NULL)
791 memory_info->blob=AcquireQuantumMemory(1,size);
792 if (memory_info->blob != NULL)
793 memory_info->type=UnalignedVirtualMemory;
795 if (memory_info->blob == NULL)
796 memory_info=RelinquishVirtualMemory(memory_info);
829MagickExport
void *CopyMagickMemory(
void *magick_restrict destination,
830 const void *magick_restrict source,
const size_t size)
838 assert(destination != (
void *) NULL);
839 assert(source != (
const void *) NULL);
840 p=(
const unsigned char *) source;
841 q=(
unsigned char *) destination;
842 if (((q+size) < p) || (q > (p+size)))
845 default:
return(memcpy(destination,source,size));
846 case 8: *q++=(*p++); magick_fallthrough;
847 case 7: *q++=(*p++); magick_fallthrough;
848 case 6: *q++=(*p++); magick_fallthrough;
849 case 5: *q++=(*p++); magick_fallthrough;
850 case 4: *q++=(*p++); magick_fallthrough;
851 case 3: *q++=(*p++); magick_fallthrough;
852 case 2: *q++=(*p++); magick_fallthrough;
853 case 1: *q++=(*p++); magick_fallthrough;
854 case 0:
return(destination);
856 return(memmove(destination,source,size));
877MagickExport
void DestroyMagickMemory(
void)
879#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
884 ActivateSemaphoreInfo(&memory_semaphore);
885 LockSemaphoreInfo(memory_semaphore);
886 for (i=0; i < (ssize_t) memory_pool.number_segments; i++)
887 if (memory_pool.segments[i]->mapped == MagickFalse)
888 memory_methods.destroy_memory_handler(
889 memory_pool.segments[i]->allocation);
891 (
void) UnmapBlob(memory_pool.segments[i]->allocation,
892 memory_pool.segments[i]->length);
893 free_segments=(DataSegmentInfo *) NULL;
894 (void) memset(&memory_pool,0,
sizeof(memory_pool));
895 UnlockSemaphoreInfo(memory_semaphore);
896 RelinquishSemaphoreInfo(&memory_semaphore);
900#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
924static MagickBooleanType ExpandHeap(
size_t size)
944 blocksize=((size+12*
sizeof(size_t))+SegmentSize-1) & -SegmentSize;
945 assert(memory_pool.number_segments < MaxSegments);
946 segment=MapBlob(-1,IOMode,0,blocksize);
947 mapped=segment != (
void *) NULL ? MagickTrue : MagickFalse;
948 if (segment == (
void *) NULL)
949 segment=(
void *) memory_methods.acquire_memory_handler(blocksize);
950 if (segment == (
void *) NULL)
952 segment_info=(DataSegmentInfo *) free_segments;
953 free_segments=segment_info->next;
954 segment_info->mapped=mapped;
955 segment_info->length=blocksize;
956 segment_info->allocation=segment;
957 segment_info->bound=(
char *) segment+blocksize;
958 i=(ssize_t) memory_pool.number_segments-1;
959 for ( ; (i >= 0) && (memory_pool.segments[i]->allocation > segment); i--)
960 memory_pool.segments[i+1]=memory_pool.segments[i];
961 memory_pool.segments[i+1]=segment_info;
962 memory_pool.number_segments++;
963 size=blocksize-12*
sizeof(size_t);
964 block=(
char *) segment_info->allocation+4*
sizeof(
size_t);
965 *BlockHeader(block)=size | PreviousBlockBit;
966 *BlockFooter(block,size)=size;
967 InsertFreeBlock(block,AllocationPolicy(size));
968 block=NextBlock(block);
969 assert(block < segment_info->bound);
970 *BlockHeader(block)=2*
sizeof(size_t);
971 *BlockHeader(NextBlock(block))=PreviousBlockBit;
1005MagickExport
void GetMagickMemoryMethods(
1006 AcquireMemoryHandler *acquire_memory_handler,
1007 ResizeMemoryHandler *resize_memory_handler,
1008 DestroyMemoryHandler *destroy_memory_handler)
1010 assert(acquire_memory_handler != (AcquireMemoryHandler *) NULL);
1011 assert(resize_memory_handler != (ResizeMemoryHandler *) NULL);
1012 assert(destroy_memory_handler != (DestroyMemoryHandler *) NULL);
1013 *acquire_memory_handler=memory_methods.acquire_memory_handler;
1014 *resize_memory_handler=memory_methods.resize_memory_handler;
1015 *destroy_memory_handler=memory_methods.destroy_memory_handler;
1036static size_t GetMaxMemoryRequestFromPolicy(
void)
1038#define MinMemoryRequest "16MiB"
1044 max_memory = (size_t) MAGICK_SSIZE_MAX;
1046 value=GetPolicyValue(
"system:max-memory-request");
1047 if (value != (
char *) NULL)
1052 max_memory=MagickMax(StringToSizeType(value,100.0),StringToSizeType(
1053 MinMemoryRequest,100.0));
1054 value=DestroyString(value);
1056 return(MagickMin(max_memory,(
size_t) MAGICK_SSIZE_MAX));
1059MagickExport
size_t GetMaxMemoryRequest(
void)
1061 if (max_memory_request == 0)
1067 max_memory_request=(size_t) MAGICK_SSIZE_MAX;
1068 max_memory_request=GetMaxMemoryRequestFromPolicy();
1070 return(max_memory_request);
1091static size_t GetMaxProfileSizeFromPolicy(
void)
1097 max=(size_t) MAGICK_SSIZE_MAX;
1099 value=GetPolicyValue(
"system:max-profile-size");
1100 if (value != (
char *) NULL)
1105 max=StringToSizeType(value,100.0);
1106 value=DestroyString(value);
1108 return(MagickMin(max,(
size_t) MAGICK_SSIZE_MAX));
1111MagickExport
size_t GetMaxProfileSize(
void)
1113 if (max_profile_size == 0)
1114 max_profile_size=GetMaxProfileSizeFromPolicy();
1115 return(max_profile_size);
1140MagickExport
void *GetVirtualMemoryBlob(
const MemoryInfo *memory_info)
1142 assert(memory_info != (
const MemoryInfo *) NULL);
1143 assert(memory_info->signature == MagickCoreSignature);
1144 return(memory_info->blob);
1170MagickExport
void *RelinquishAlignedMemory(
void *memory)
1172 if (memory == (
void *) NULL)
1173 return((
void *) NULL);
1174 if (memory_methods.relinquish_aligned_memory_handler != (RelinquishAlignedMemoryHandler) NULL)
1176 memory_methods.relinquish_aligned_memory_handler(memory);
1179#if defined(MAGICKCORE_HAVE_ALIGNED_MALLOC) || defined(MAGICKCORE_HAVE_POSIX_MEMALIGN)
1181#elif defined(MAGICKCORE_HAVE__ALIGNED_MALLOC)
1182 _aligned_free(memory);
1184 RelinquishMagickMemory(actual_base_address(memory));
1212MagickExport
void *RelinquishMagickMemory(
void *memory)
1214 if (memory == (
void *) NULL)
1215 return((
void *) NULL);
1216#if !defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
1217 memory_methods.destroy_memory_handler(memory);
1219 LockSemaphoreInfo(memory_semaphore);
1220 assert((SizeOfBlock(memory) % (4*
sizeof(
size_t))) == 0);
1221 assert((*BlockHeader(NextBlock(memory)) & PreviousBlockBit) != 0);
1222 if ((*BlockHeader(memory) & PreviousBlockBit) == 0)
1230 previous=PreviousBlock(memory);
1231 RemoveFreeBlock(previous,AllocationPolicy(SizeOfBlock(previous)));
1232 *BlockHeader(previous)=(SizeOfBlock(previous)+SizeOfBlock(memory)) |
1233 (*BlockHeader(previous) & ~SizeMask);
1236 if ((*BlockHeader(NextBlock(NextBlock(memory))) & PreviousBlockBit) == 0)
1244 next=NextBlock(memory);
1245 RemoveFreeBlock(next,AllocationPolicy(SizeOfBlock(next)));
1246 *BlockHeader(memory)=(SizeOfBlock(memory)+SizeOfBlock(next)) |
1247 (*BlockHeader(memory) & ~SizeMask);
1249 *BlockFooter(memory,SizeOfBlock(memory))=SizeOfBlock(memory);
1250 *BlockHeader(NextBlock(memory))&=(~PreviousBlockBit);
1251 InsertFreeBlock(memory,AllocationPolicy(SizeOfBlock(memory)));
1252 UnlockSemaphoreInfo(memory_semaphore);
1254 return((
void *) NULL);
1279MagickExport MemoryInfo *RelinquishVirtualMemory(MemoryInfo *memory_info)
1281 assert(memory_info != (MemoryInfo *) NULL);
1282 assert(memory_info->signature == MagickCoreSignature);
1283 if (memory_info->blob != (
void *) NULL)
1284 switch (memory_info->type)
1286 case AlignedVirtualMemory:
1288 (void) ShredMagickMemory(memory_info->blob,memory_info->length);
1289 memory_info->blob=RelinquishAlignedMemory(memory_info->blob);
1292 case MapVirtualMemory:
1294 (void) UnmapBlob(memory_info->blob,memory_info->length);
1295 memory_info->blob=NULL;
1296 if (*memory_info->filename !=
'\0')
1297 (void) RelinquishUniqueFileResource(memory_info->filename);
1300 case UnalignedVirtualMemory:
1303 (void) ShredMagickMemory(memory_info->blob,memory_info->length);
1304 memory_info->blob=RelinquishMagickMemory(memory_info->blob);
1308 memory_info->signature=(~MagickCoreSignature);
1309 memory_info=(MemoryInfo *) RelinquishAlignedMemory(memory_info);
1310 return(memory_info);
1342MagickExport
void *ResetMagickMemory(
void *memory,
int c,
const size_t size)
1344 volatile unsigned char
1345 *p = (
volatile unsigned char *) memory;
1350 assert(memory != (
void *) NULL);
1352 *p++=(
unsigned char) c;
1374MagickPrivate
void ResetVirtualAnonymousMemory(
void)
1376 virtual_anonymous_memory=0;
1406#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
1407static inline void *ResizeBlock(
void *block,
size_t size)
1412 if (block == (
void *) NULL)
1413 return(AcquireBlock(size));
1414 memory=AcquireBlock(size);
1415 if (memory == (
void *) NULL)
1416 return((
void *) NULL);
1417 if (size <= (SizeOfBlock(block)-
sizeof(
size_t)))
1418 (void) memcpy(memory,block,size);
1420 (
void) memcpy(memory,block,SizeOfBlock(block)-
sizeof(
size_t));
1421 memory_pool.allocation+=size;
1426MagickExport
void *ResizeMagickMemory(
void *memory,
const size_t size)
1431 if (memory == (
void *) NULL)
1432 return(AcquireMagickMemory(size));
1433#if !defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
1434 block=memory_methods.resize_memory_handler(memory,size == 0 ? 1UL : size);
1435 if (block == (
void *) NULL)
1436 memory=RelinquishMagickMemory(memory);
1438 LockSemaphoreInfo(memory_semaphore);
1439 block=ResizeBlock(memory,size == 0 ? 1UL : size);
1440 if (block == (
void *) NULL)
1442 if (ExpandHeap(size == 0 ? 1UL : size) == MagickFalse)
1444 UnlockSemaphoreInfo(memory_semaphore);
1445 memory=RelinquishMagickMemory(memory);
1446 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
1448 block=ResizeBlock(memory,size == 0 ? 1UL : size);
1449 assert(block != (
void *) NULL);
1451 UnlockSemaphoreInfo(memory_semaphore);
1452 memory=RelinquishMagickMemory(memory);
1486MagickExport
void *ResizeQuantumMemory(
void *memory,
const size_t count,
1487 const size_t quantum)
1492 if ((HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse) ||
1493 (size > GetMaxMemoryRequest()))
1496 memory=RelinquishMagickMemory(memory);
1499 return(ResizeMagickMemory(memory,size));
1529MagickExport
void SetMagickAlignedMemoryMethods(
1530 AcquireAlignedMemoryHandler acquire_aligned_memory_handler,
1531 RelinquishAlignedMemoryHandler relinquish_aligned_memory_handler)
1533 memory_methods.acquire_aligned_memory_handler=acquire_aligned_memory_handler;
1534 memory_methods.relinquish_aligned_memory_handler=
1535 relinquish_aligned_memory_handler;
1568MagickExport
void SetMagickMemoryMethods(
1569 AcquireMemoryHandler acquire_memory_handler,
1570 ResizeMemoryHandler resize_memory_handler,
1571 DestroyMemoryHandler destroy_memory_handler)
1576 if (acquire_memory_handler != (AcquireMemoryHandler) NULL)
1577 memory_methods.acquire_memory_handler=acquire_memory_handler;
1578 if (resize_memory_handler != (ResizeMemoryHandler) NULL)
1579 memory_methods.resize_memory_handler=resize_memory_handler;
1580 if (destroy_memory_handler != (DestroyMemoryHandler) NULL)
1581 memory_methods.destroy_memory_handler=destroy_memory_handler;
1606MagickPrivate
void SetMaxMemoryRequest(
const MagickSizeType limit)
1608 max_memory_request=(size_t) MagickMin(limit,GetMaxMemoryRequestFromPolicy());
1633MagickPrivate
void SetMaxProfileSize(
const MagickSizeType limit)
1635 max_profile_size=(size_t) MagickMin(limit,GetMaxProfileSizeFromPolicy());
1664MagickPrivate MagickBooleanType ShredMagickMemory(
void *memory,
1665 const size_t length)
1682 if ((memory == NULL) || (length == 0))
1683 return(MagickFalse);
1690 property=GetEnvironmentValue(
"MAGICK_SHRED_PASSES");
1691 if (property != (
char *) NULL)
1693 passes=(ssize_t) StringToInteger(property);
1694 property=DestroyString(property);
1696 property=GetPolicyValue(
"system:shred");
1697 if (property != (
char *) NULL)
1699 passes=(ssize_t) StringToInteger(property);
1700 property=DestroyString(property);
1708 quantum=(size_t) MagickMin(length,MagickMinBufferExtent);
1709 random_info=AcquireRandomInfo();
1710 key=GetRandomKey(random_info,quantum);
1711 for (i=0; i < passes; i++)
1717 *p = (
unsigned char *) memory;
1719 for (j=0; j < length; j+=quantum)
1722 SetRandomKey(random_info,quantum,GetStringInfoDatum(key));
1723 (void) memcpy(p,GetStringInfoDatum(key),(
size_t)
1724 MagickMin(quantum,length-j));
1725 p+=(ptrdiff_t) quantum;
1730 key=DestroyStringInfo(key);
1731 random_info=DestroyRandomInfo(random_info);
1732 return(i < passes ? MagickFalse : MagickTrue);