83#include "magick/studio.h"
84#include "magick/blob.h"
85#include "magick/blob-private.h"
86#include "magick/exception.h"
87#include "magick/exception-private.h"
88#include "magick/image-private.h"
89#include "magick/memory_.h"
90#include "magick/memory-private.h"
91#include "magick/policy.h"
92#include "magick/random_.h"
93#include "magick/resource_.h"
94#include "magick/semaphore.h"
95#include "magick/string_.h"
96#include "magick/string-private.h"
97#include "magick/utility-private.h"
102#define BlockFooter(block,size) \
103 ((size_t *) ((char *) (block)+(size)-2*sizeof(size_t)))
104#define BlockHeader(block) ((size_t *) (block)-1)
105#define BlockThreshold 1024
106#define MaxBlockExponent 16
107#define MaxBlocks ((BlockThreshold/(4*sizeof(size_t)))+MaxBlockExponent+1)
108#define MaxSegments 1024
109#define NextBlock(block) ((char *) (block)+SizeOfBlock(block))
110#define NextBlockInList(block) (*(void **) (block))
111#define PreviousBlock(block) ((char *) (block)-(*((size_t *) (block)-2)))
112#define PreviousBlockBit 0x01
113#define PreviousBlockInList(block) (*((void **) (block)+1))
114#define SegmentSize (2*1024*1024)
115#define SizeMask (~0x01)
116#define SizeOfBlock(block) (*BlockHeader(block) & SizeMask)
123 UndefinedVirtualMemory,
124 AlignedVirtualMemory,
126 UnalignedVirtualMemory
149 acquire_memory_handler;
152 resize_memory_handler;
155 destroy_memory_handler;
157 AcquireAlignedMemoryHandler
158 acquire_aligned_memory_handler;
160 RelinquishAlignedMemoryHandler
161 relinquish_aligned_memory_handler;
162} MagickMemoryMethods;
167 filename[MagickPathExtent];
188 *blocks[MaxBlocks+1];
194 *segments[MaxSegments],
195 segment_pool[MaxSegments];
202 max_memory_request = 0,
203 virtual_anonymous_memory = 0;
206static void *MSCMalloc(
size_t size)
208 return(malloc(size));
211static void *MSCRealloc(
void* ptr,
size_t size)
213 return(realloc(ptr,size));
216static void MSCFree(
void* ptr)
222static MagickMemoryMethods
226 (AcquireMemoryHandler) MSCMalloc,
227 (ResizeMemoryHandler) MSCRealloc,
228 (DestroyMemoryHandler) MSCFree,
230 (AcquireMemoryHandler) malloc,
231 (ResizeMemoryHandler) realloc,
232 (DestroyMemoryHandler) free,
234 (AcquireAlignedMemoryHandler) NULL,
235 (RelinquishAlignedMemoryHandler) NULL
237#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
244static volatile DataSegmentInfo
245 *free_segments = (DataSegmentInfo *) NULL;
250static MagickBooleanType
279#if defined(MAGICKCORE_HAVE_ALIGNED_MALLOC)
280#define AcquireAlignedMemory_Actual AcquireAlignedMemory_STDC
281static inline void *AcquireAlignedMemory_STDC(
const size_t size)
284 extent = CACHE_ALIGNED(size);
291 return(aligned_alloc(CACHE_LINE_SIZE,extent));
293#elif defined(MAGICKCORE_HAVE_POSIX_MEMALIGN)
294#define AcquireAlignedMemory_Actual AcquireAlignedMemory_POSIX
295static inline void *AcquireAlignedMemory_POSIX(
const size_t size)
300 if (posix_memalign(&memory,CACHE_LINE_SIZE,size))
304#elif defined(MAGICKCORE_HAVE__ALIGNED_MALLOC)
305#define AcquireAlignedMemory_Actual AcquireAlignedMemory_WinAPI
306static inline void *AcquireAlignedMemory_WinAPI(
const size_t size)
308 return(_aligned_malloc(size,CACHE_LINE_SIZE));
311#define ALIGNMENT_OVERHEAD \
312 (MAGICKCORE_MAX_ALIGNMENT_PADDING(CACHE_LINE_SIZE) + MAGICKCORE_SIZEOF_VOID_P)
313static inline void *reserve_space_for_actual_base_address(
void *
const p)
315 return((
void **) p+1);
318static inline void **pointer_to_space_for_actual_base_address(
void *
const p)
320 return((
void **) p-1);
323static inline void *actual_base_address(
void *
const p)
325 return(*pointer_to_space_for_actual_base_address(p));
328static inline void *align_to_cache(
void *
const p)
330 return((
void *) CACHE_ALIGNED((MagickAddressType) p));
333static inline void *adjust(
void *
const p)
335 return(align_to_cache(reserve_space_for_actual_base_address(p)));
338#define AcquireAlignedMemory_Actual AcquireAlignedMemory_Generic
339static inline void *AcquireAlignedMemory_Generic(
const size_t size)
348 #if SIZE_MAX < ALIGNMENT_OVERHEAD
349 #error "CACHE_LINE_SIZE is way too big."
351 extent=(size+ALIGNMENT_OVERHEAD);
357 p=AcquireMagickMemory(extent);
361 *pointer_to_space_for_actual_base_address(memory)=p;
366MagickExport
void *AcquireAlignedMemory(
const size_t count,
const size_t quantum)
371 if (HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse)
376 if (memory_methods.acquire_aligned_memory_handler != (AcquireAlignedMemoryHandler) NULL)
377 return(memory_methods.acquire_aligned_memory_handler(size,CACHE_LINE_SIZE));
378 return(AcquireAlignedMemory_Actual(size));
381#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
406static inline size_t AllocationPolicy(
size_t size)
415 assert(size % (4*
sizeof(
size_t)) == 0);
416 if (size <= BlockThreshold)
417 return(size/(4*
sizeof(
size_t)));
421 if (size > (
size_t) (BlockThreshold*(1L << (MaxBlockExponent-1L))))
422 return(MaxBlocks-1L);
426 blocksize=BlockThreshold/(4*
sizeof(size_t));
427 for ( ; size > BlockThreshold; size/=2)
429 assert(blocksize > (BlockThreshold/(4*
sizeof(
size_t))));
430 assert(blocksize < (MaxBlocks-1L));
434static inline void InsertFreeBlock(
void *block,
const size_t i)
443 size=SizeOfBlock(block);
444 previous=(
void *) NULL;
445 next=memory_pool.blocks[i];
446 while ((next != (
void *) NULL) && (SizeOfBlock(next) < size))
449 next=NextBlockInList(next);
451 PreviousBlockInList(block)=previous;
452 NextBlockInList(block)=next;
453 if (previous != (
void *) NULL)
454 NextBlockInList(previous)=block;
456 memory_pool.blocks[i]=block;
457 if (next != (
void *) NULL)
458 PreviousBlockInList(next)=block;
461static inline void RemoveFreeBlock(
void *block,
const size_t i)
467 next=NextBlockInList(block);
468 previous=PreviousBlockInList(block);
469 if (previous == (
void *) NULL)
470 memory_pool.blocks[i]=next;
472 NextBlockInList(previous)=next;
473 if (next != (
void *) NULL)
474 PreviousBlockInList(next)=previous;
477static void *AcquireBlock(
size_t size)
488 size=(size_t) (size+
sizeof(
size_t)+6*
sizeof(
size_t)-1) & -(4U*
sizeof(size_t));
489 i=AllocationPolicy(size);
490 block=memory_pool.blocks[i];
491 while ((block != (
void *) NULL) && (SizeOfBlock(block) < size))
492 block=NextBlockInList(block);
493 if (block == (
void *) NULL)
496 while (memory_pool.blocks[i] == (
void *) NULL)
498 block=memory_pool.blocks[i];
500 return((
void *) NULL);
502 assert((*BlockHeader(NextBlock(block)) & PreviousBlockBit) == 0);
503 assert(SizeOfBlock(block) >= size);
504 RemoveFreeBlock(block,AllocationPolicy(SizeOfBlock(block)));
505 if (SizeOfBlock(block) > size)
516 next=(
char *) block+size;
517 blocksize=SizeOfBlock(block)-size;
518 *BlockHeader(next)=blocksize;
519 *BlockFooter(next,blocksize)=blocksize;
520 InsertFreeBlock(next,AllocationPolicy(blocksize));
521 *BlockHeader(block)=size | (*BlockHeader(block) & ~SizeMask);
523 assert(size == SizeOfBlock(block));
524 *BlockHeader(NextBlock(block))|=PreviousBlockBit;
525 memory_pool.allocation+=size;
553MagickExport
void *AcquireMagickMemory(
const size_t size)
558#if !defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
559 memory=memory_methods.acquire_memory_handler(size == 0 ? 1UL : size);
562 ActivateSemaphoreInfo(&memory_semaphore);
563 if (free_segments == (DataSegmentInfo *) NULL)
565 LockSemaphoreInfo(memory_semaphore);
566 if (free_segments == (DataSegmentInfo *) NULL)
571 assert(2*
sizeof(
size_t) > (
size_t) (~SizeMask));
572 (void) memset(&memory_pool,0,
sizeof(memory_pool));
573 memory_pool.allocation=SegmentSize;
574 memory_pool.blocks[MaxBlocks]=(
void *) (-1);
575 for (i=0; i < MaxSegments; i++)
578 memory_pool.segment_pool[i].previous=
579 (&memory_pool.segment_pool[i-1]);
580 if (i != (MaxSegments-1))
581 memory_pool.segment_pool[i].next=(&memory_pool.segment_pool[i+1]);
583 free_segments=(&memory_pool.segment_pool[0]);
585 UnlockSemaphoreInfo(memory_semaphore);
587 LockSemaphoreInfo(memory_semaphore);
588 memory=AcquireBlock(size == 0 ? 1UL : size);
589 if (memory == (
void *) NULL)
591 if (ExpandHeap(size == 0 ? 1UL : size) != MagickFalse)
592 memory=AcquireBlock(size == 0 ? 1UL : size);
594 UnlockSemaphoreInfo(memory_semaphore);
627MagickExport
void *AcquireCriticalMemory(
const size_t size)
629#if !defined(STDERR_FILENO)
630#define STDERR_FILENO 2
633 static const char fatal_message[] =
634 "ImageMagick: fatal error: unable to acquire critical memory\n";
642 memory=AcquireMagickMemory(size);
643 if (memory != (
void *) NULL)
645 (void) write(STDERR_FILENO,fatal_message,
sizeof(fatal_message)-1);
646 MagickCoreTerminus();
675MagickExport
void *AcquireQuantumMemory(
const size_t count,
const size_t quantum)
680 if ((HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse) ||
681 (size > GetMaxMemoryRequest()))
686 return(AcquireMagickMemory(size));
715MagickExport MemoryInfo *AcquireVirtualMemory(
const size_t count,
716 const size_t quantum)
727 if (HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse)
730 return((MemoryInfo *) NULL);
732 if (virtual_anonymous_memory == 0)
734 virtual_anonymous_memory=1;
735 value=GetPolicyValue(
"system:memory-map");
736 if (LocaleCompare(value,
"anonymous") == 0)
741#if defined(MAGICKCORE_HAVE_MMAP) && defined(MAP_ANONYMOUS)
742 virtual_anonymous_memory=2;
745 value=DestroyString(value);
747 memory_info=(MemoryInfo *) MagickAssumeAligned(AcquireAlignedMemory(1,
748 sizeof(*memory_info)));
749 if (memory_info == (MemoryInfo *) NULL)
750 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
751 (void) memset(memory_info,0,
sizeof(*memory_info));
752 memory_info->length=size;
753 memory_info->signature=MagickCoreSignature;
754 if ((virtual_anonymous_memory == 1) && (size <= GetMaxMemoryRequest()))
756 memory_info->blob=AcquireAlignedMemory(1,size);
757 if (memory_info->blob != NULL)
758 memory_info->type=AlignedVirtualMemory;
760 if (memory_info->blob == NULL)
765 memory_info->blob=NULL;
766 if (size <= GetMaxMemoryRequest())
767 memory_info->blob=MapBlob(-1,IOMode,0,size);
768 if (memory_info->blob != NULL)
769 memory_info->type=MapVirtualMemory;
778 file=AcquireUniqueFileResource(memory_info->filename);
784 offset=(MagickOffsetType) lseek(file,size-1,SEEK_SET);
785 if ((offset == (MagickOffsetType) (size-1)) &&
786 (write(file,
"",1) == 1))
788#if !defined(MAGICKCORE_HAVE_POSIX_FALLOCATE)
789 memory_info->blob=MapBlob(file,IOMode,0,size);
791 if (posix_fallocate(file,0,(MagickOffsetType) size) == 0)
792 memory_info->blob=MapBlob(file,IOMode,0,size);
794 if (memory_info->blob != NULL)
795 memory_info->type=MapVirtualMemory;
798 (void) RelinquishUniqueFileResource(
799 memory_info->filename);
800 *memory_info->filename=
'\0';
807 if (memory_info->blob == NULL)
809 memory_info->blob=AcquireQuantumMemory(1,size);
810 if (memory_info->blob != NULL)
811 memory_info->type=UnalignedVirtualMemory;
813 if (memory_info->blob == NULL)
814 memory_info=RelinquishVirtualMemory(memory_info);
847MagickExport
void *CopyMagickMemory(
void *magick_restrict destination,
848 const void *magick_restrict source,
const size_t size)
856 assert(destination != (
void *) NULL);
857 assert(source != (
const void *) NULL);
858 p=(
const unsigned char *) source;
859 q=(
unsigned char *) destination;
860 if (((q+size) < p) || (q > (p+size)))
863 default:
return(memcpy(destination,source,size));
864 case 8: *q++=(*p++); magick_fallthrough;
865 case 7: *q++=(*p++); magick_fallthrough;
866 case 6: *q++=(*p++); magick_fallthrough;
867 case 5: *q++=(*p++); magick_fallthrough;
868 case 4: *q++=(*p++); magick_fallthrough;
869 case 3: *q++=(*p++); magick_fallthrough;
870 case 2: *q++=(*p++); magick_fallthrough;
871 case 1: *q++=(*p++); magick_fallthrough;
872 case 0:
return(destination);
874 return(memmove(destination,source,size));
895MagickExport
void DestroyMagickMemory(
void)
897#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
902 ActivateSemaphoreInfo(&memory_semaphore);
903 LockSemaphoreInfo(memory_semaphore);
904 for (i=0; i < (ssize_t) memory_pool.number_segments; i++)
905 if (memory_pool.segments[i]->mapped == MagickFalse)
906 memory_methods.destroy_memory_handler(
907 memory_pool.segments[i]->allocation);
909 (
void) UnmapBlob(memory_pool.segments[i]->allocation,
910 memory_pool.segments[i]->length);
911 free_segments=(DataSegmentInfo *) NULL;
912 (void) memset(&memory_pool,0,
sizeof(memory_pool));
913 UnlockSemaphoreInfo(memory_semaphore);
914 DestroySemaphoreInfo(&memory_semaphore);
918#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
942static MagickBooleanType ExpandHeap(
size_t size)
962 blocksize=((size+12*
sizeof(size_t))+SegmentSize-1) & -SegmentSize;
963 assert(memory_pool.number_segments < MaxSegments);
964 segment=MapBlob(-1,IOMode,0,blocksize);
965 mapped=segment != (
void *) NULL ? MagickTrue : MagickFalse;
966 if (segment == (
void *) NULL)
967 segment=(
void *) memory_methods.acquire_memory_handler(blocksize);
968 if (segment == (
void *) NULL)
970 segment_info=(DataSegmentInfo *) free_segments;
971 free_segments=segment_info->next;
972 segment_info->mapped=mapped;
973 segment_info->length=blocksize;
974 segment_info->allocation=segment;
975 segment_info->bound=(
char *) segment+blocksize;
976 i=(ssize_t) memory_pool.number_segments-1;
977 for ( ; (i >= 0) && (memory_pool.segments[i]->allocation > segment); i--)
978 memory_pool.segments[i+1]=memory_pool.segments[i];
979 memory_pool.segments[i+1]=segment_info;
980 memory_pool.number_segments++;
981 size=blocksize-12*
sizeof(size_t);
982 block=(
char *) segment_info->allocation+4*
sizeof(
size_t);
983 *BlockHeader(block)=size | PreviousBlockBit;
984 *BlockFooter(block,size)=size;
985 InsertFreeBlock(block,AllocationPolicy(size));
986 block=NextBlock(block);
987 assert(block < segment_info->bound);
988 *BlockHeader(block)=2*
sizeof(size_t);
989 *BlockHeader(NextBlock(block))=PreviousBlockBit;
1023MagickExport
void GetMagickMemoryMethods(
1024 AcquireMemoryHandler *acquire_memory_handler,
1025 ResizeMemoryHandler *resize_memory_handler,
1026 DestroyMemoryHandler *destroy_memory_handler)
1028 assert(acquire_memory_handler != (AcquireMemoryHandler *) NULL);
1029 assert(resize_memory_handler != (ResizeMemoryHandler *) NULL);
1030 assert(destroy_memory_handler != (DestroyMemoryHandler *) NULL);
1031 *acquire_memory_handler=memory_methods.acquire_memory_handler;
1032 *resize_memory_handler=memory_methods.resize_memory_handler;
1033 *destroy_memory_handler=memory_methods.destroy_memory_handler;
1054MagickExport
size_t GetMaxMemoryRequest(
void)
1056#define MinMemoryRequest "16MiB"
1058 if (max_memory_request == 0)
1063 max_memory_request=(size_t) MAGICK_SSIZE_MAX;
1064 value=GetPolicyValue(
"system:max-memory-request");
1065 if (value != (
char *) NULL)
1070 max_memory_request=MagickMax(StringToSizeType(value,100.0),
1071 StringToSizeType(MinMemoryRequest,100.0));
1072 value=DestroyString(value);
1075 return(MagickMin(max_memory_request,(
size_t) MAGICK_SSIZE_MAX));
1100MagickExport
void *GetVirtualMemoryBlob(
const MemoryInfo *memory_info)
1102 assert(memory_info != (
const MemoryInfo *) NULL);
1103 assert(memory_info->signature == MagickCoreSignature);
1104 return(memory_info->blob);
1130MagickExport
void *RelinquishAlignedMemory(
void *memory)
1132 if (memory == (
void *) NULL)
1133 return((
void *) NULL);
1134 if (memory_methods.relinquish_aligned_memory_handler != (RelinquishAlignedMemoryHandler) NULL)
1136 memory_methods.relinquish_aligned_memory_handler(memory);
1139#if defined(MAGICKCORE_HAVE_ALIGNED_MALLOC) || defined(MAGICKCORE_HAVE_POSIX_MEMALIGN)
1141#elif defined(MAGICKCORE_HAVE__ALIGNED_MALLOC)
1142 _aligned_free(memory);
1144 RelinquishMagickMemory(actual_base_address(memory));
1172MagickExport
void *RelinquishMagickMemory(
void *memory)
1174 if (memory == (
void *) NULL)
1175 return((
void *) NULL);
1176#if !defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
1177 memory_methods.destroy_memory_handler(memory);
1179 LockSemaphoreInfo(memory_semaphore);
1180 assert((SizeOfBlock(memory) % (4*
sizeof(
size_t))) == 0);
1181 assert((*BlockHeader(NextBlock(memory)) & PreviousBlockBit) != 0);
1182 if ((*BlockHeader(memory) & PreviousBlockBit) == 0)
1190 previous=PreviousBlock(memory);
1191 RemoveFreeBlock(previous,AllocationPolicy(SizeOfBlock(previous)));
1192 *BlockHeader(previous)=(SizeOfBlock(previous)+SizeOfBlock(memory)) |
1193 (*BlockHeader(previous) & ~SizeMask);
1196 if ((*BlockHeader(NextBlock(NextBlock(memory))) & PreviousBlockBit) == 0)
1204 next=NextBlock(memory);
1205 RemoveFreeBlock(next,AllocationPolicy(SizeOfBlock(next)));
1206 *BlockHeader(memory)=(SizeOfBlock(memory)+SizeOfBlock(next)) |
1207 (*BlockHeader(memory) & ~SizeMask);
1209 *BlockFooter(memory,SizeOfBlock(memory))=SizeOfBlock(memory);
1210 *BlockHeader(NextBlock(memory))&=(~PreviousBlockBit);
1211 InsertFreeBlock(memory,AllocationPolicy(SizeOfBlock(memory)));
1212 UnlockSemaphoreInfo(memory_semaphore);
1214 return((
void *) NULL);
1239MagickExport MemoryInfo *RelinquishVirtualMemory(MemoryInfo *memory_info)
1241 assert(memory_info != (MemoryInfo *) NULL);
1242 assert(memory_info->signature == MagickCoreSignature);
1243 if (memory_info->blob != (
void *) NULL)
1244 switch (memory_info->type)
1246 case AlignedVirtualMemory:
1248 (void) ShredMagickMemory(memory_info->blob,memory_info->length);
1249 memory_info->blob=RelinquishAlignedMemory(memory_info->blob);
1252 case MapVirtualMemory:
1254 (void) UnmapBlob(memory_info->blob,memory_info->length);
1255 memory_info->blob=NULL;
1256 if (*memory_info->filename !=
'\0')
1257 (void) RelinquishUniqueFileResource(memory_info->filename);
1260 case UnalignedVirtualMemory:
1263 (void) ShredMagickMemory(memory_info->blob,memory_info->length);
1264 memory_info->blob=RelinquishMagickMemory(memory_info->blob);
1268 memory_info->signature=(~MagickCoreSignature);
1269 memory_info=(MemoryInfo *) RelinquishAlignedMemory(memory_info);
1270 return(memory_info);
1303MagickExport
void *ResetMagickMemory(
void *memory,
int c,
const size_t size)
1305 volatile unsigned char
1306 *p = (
volatile unsigned char *) memory;
1311 assert(memory != (
void *) NULL);
1313 *p++=(
unsigned char) c;
1335MagickPrivate
void ResetMaxMemoryRequest(
void)
1337 max_memory_request=0;
1358MagickPrivate
void ResetVirtualAnonymousMemory(
void)
1360 virtual_anonymous_memory=0;
1390#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
1391static inline void *ResizeBlock(
void *block,
size_t size)
1396 if (block == (
void *) NULL)
1397 return(AcquireBlock(size));
1398 memory=AcquireBlock(size);
1399 if (memory == (
void *) NULL)
1400 return((
void *) NULL);
1401 if (size <= (SizeOfBlock(block)-
sizeof(
size_t)))
1402 (void) memcpy(memory,block,size);
1404 (
void) memcpy(memory,block,SizeOfBlock(block)-
sizeof(
size_t));
1405 memory_pool.allocation+=size;
1410MagickExport
void *ResizeMagickMemory(
void *memory,
const size_t size)
1415 if (memory == (
void *) NULL)
1416 return(AcquireMagickMemory(size));
1417#if !defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
1418 block=memory_methods.resize_memory_handler(memory,size == 0 ? 1UL : size);
1419 if (block == (
void *) NULL)
1420 memory=RelinquishMagickMemory(memory);
1422 LockSemaphoreInfo(memory_semaphore);
1423 block=ResizeBlock(memory,size == 0 ? 1UL : size);
1424 if (block == (
void *) NULL)
1426 if (ExpandHeap(size == 0 ? 1UL : size) == MagickFalse)
1428 UnlockSemaphoreInfo(memory_semaphore);
1429 memory=RelinquishMagickMemory(memory);
1430 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
1432 block=ResizeBlock(memory,size == 0 ? 1UL : size);
1433 assert(block != (
void *) NULL);
1435 UnlockSemaphoreInfo(memory_semaphore);
1436 memory=RelinquishMagickMemory(memory);
1470MagickExport
void *ResizeQuantumMemory(
void *memory,
const size_t count,
1471 const size_t quantum)
1476 if ((HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse) ||
1477 (size > GetMaxMemoryRequest()))
1480 memory=RelinquishMagickMemory(memory);
1483 if (size > GetMaxMemoryRequest())
1485 return(ResizeMagickMemory(memory,size));
1515MagickExport
void SetMagickAlignedMemoryMethods(
1516 AcquireAlignedMemoryHandler acquire_aligned_memory_handler,
1517 RelinquishAlignedMemoryHandler relinquish_aligned_memory_handler)
1519 memory_methods.acquire_aligned_memory_handler=acquire_aligned_memory_handler;
1520 memory_methods.relinquish_aligned_memory_handler=
1521 relinquish_aligned_memory_handler;
1554MagickExport
void SetMagickMemoryMethods(
1555 AcquireMemoryHandler acquire_memory_handler,
1556 ResizeMemoryHandler resize_memory_handler,
1557 DestroyMemoryHandler destroy_memory_handler)
1562 if (acquire_memory_handler != (AcquireMemoryHandler) NULL)
1563 memory_methods.acquire_memory_handler=acquire_memory_handler;
1564 if (resize_memory_handler != (ResizeMemoryHandler) NULL)
1565 memory_methods.resize_memory_handler=resize_memory_handler;
1566 if (destroy_memory_handler != (DestroyMemoryHandler) NULL)
1567 memory_methods.destroy_memory_handler=destroy_memory_handler;
1592MagickPrivate
void SetMaxMemoryRequest(
const MagickSizeType limit)
1594 max_memory_request=MagickMin(limit,GetMaxMemoryRequest());
1623MagickPrivate MagickBooleanType ShredMagickMemory(
void *memory,
1624 const size_t length)
1641 if ((memory == NULL) || (length == 0))
1642 return(MagickFalse);
1649 property=GetEnvironmentValue(
"MAGICK_SHRED_PASSES");
1650 if (property != (
char *) NULL)
1652 passes=(ssize_t) StringToInteger(property);
1653 property=DestroyString(property);
1655 property=GetPolicyValue(
"system:shred");
1656 if (property != (
char *) NULL)
1658 passes=(ssize_t) StringToInteger(property);
1659 property=DestroyString(property);
1667 quantum=(size_t) MagickMin(length,MagickMinBufferExtent);
1668 random_info=AcquireRandomInfo();
1669 key=GetRandomKey(random_info,quantum);
1670 for (i=0; i < passes; i++)
1676 *p = (
unsigned char *) memory;
1678 for (j=0; j < length; j+=quantum)
1681 SetRandomKey(random_info,quantum,GetStringInfoDatum(key));
1682 (void) memcpy(p,GetStringInfoDatum(key),(
size_t)
1683 MagickMin(quantum,length-j));
1684 p+=(ptrdiff_t) quantum;
1689 key=DestroyStringInfo(key);
1690 random_info=DestroyRandomInfo(random_info);
1691 return(i < passes ? MagickFalse : MagickTrue);