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;
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;
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)
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)
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) ||
372 (size > GetMaxMemoryRequest()))
377 if (memory_methods.acquire_aligned_memory_handler != (AcquireAlignedMemoryHandler) NULL)
378 return(memory_methods.acquire_aligned_memory_handler(size,CACHE_LINE_SIZE));
379 return(AcquireAlignedMemory_Actual(size));
382#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
407static inline size_t AllocationPolicy(
size_t size)
416 assert(size % (4*
sizeof(
size_t)) == 0);
417 if (size <= BlockThreshold)
418 return(size/(4*
sizeof(
size_t)));
422 if (size > (
size_t) (BlockThreshold*(1L << (MaxBlockExponent-1L))))
423 return(MaxBlocks-1L);
427 blocksize=BlockThreshold/(4*
sizeof(size_t));
428 for ( ; size > BlockThreshold; size/=2)
430 assert(blocksize > (BlockThreshold/(4*
sizeof(
size_t))));
431 assert(blocksize < (MaxBlocks-1L));
435static inline void InsertFreeBlock(
void *block,
const size_t i)
444 size=SizeOfBlock(block);
445 previous=(
void *) NULL;
446 next=memory_pool.blocks[i];
447 while ((next != (
void *) NULL) && (SizeOfBlock(next) < size))
450 next=NextBlockInList(next);
452 PreviousBlockInList(block)=previous;
453 NextBlockInList(block)=next;
454 if (previous != (
void *) NULL)
455 NextBlockInList(previous)=block;
457 memory_pool.blocks[i]=block;
458 if (next != (
void *) NULL)
459 PreviousBlockInList(next)=block;
462static inline void RemoveFreeBlock(
void *block,
const size_t i)
468 next=NextBlockInList(block);
469 previous=PreviousBlockInList(block);
470 if (previous == (
void *) NULL)
471 memory_pool.blocks[i]=next;
473 NextBlockInList(previous)=next;
474 if (next != (
void *) NULL)
475 PreviousBlockInList(next)=previous;
478static void *AcquireBlock(
size_t size)
489 size=(size_t) (size+
sizeof(
size_t)+6*
sizeof(size_t)-1) & -(4U*
sizeof(size_t));
490 i=AllocationPolicy(size);
491 block=memory_pool.blocks[i];
492 while ((block != (
void *) NULL) && (SizeOfBlock(block) < size))
493 block=NextBlockInList(block);
494 if (block == (
void *) NULL)
497 while (memory_pool.blocks[i] == (
void *) NULL)
499 block=memory_pool.blocks[i];
501 return((
void *) NULL);
503 assert((*BlockHeader(NextBlock(block)) & PreviousBlockBit) == 0);
504 assert(SizeOfBlock(block) >= size);
505 RemoveFreeBlock(block,AllocationPolicy(SizeOfBlock(block)));
506 if (SizeOfBlock(block) > size)
517 next=(
char *) block+size;
518 blocksize=SizeOfBlock(block)-size;
519 *BlockHeader(next)=blocksize;
520 *BlockFooter(next,blocksize)=blocksize;
521 InsertFreeBlock(next,AllocationPolicy(blocksize));
522 *BlockHeader(block)=size | (*BlockHeader(block) & ~SizeMask);
524 assert(size == SizeOfBlock(block));
525 *BlockHeader(NextBlock(block))|=PreviousBlockBit;
526 memory_pool.allocation+=size;
554MagickExport
void *AcquireMagickMemory(
const size_t size)
559#if !defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
560 memory=memory_methods.acquire_memory_handler(size == 0 ? 1UL : size);
563 ActivateSemaphoreInfo(&memory_semaphore);
566 LockSemaphoreInfo(memory_semaphore);
572 assert(2*
sizeof(
size_t) > (
size_t) (~SizeMask));
573 (void) memset(&memory_pool,0,
sizeof(memory_pool));
574 memory_pool.allocation=SegmentSize;
575 memory_pool.blocks[MaxBlocks]=(
void *) (-1);
576 for (i=0; i < MaxSegments; i++)
579 memory_pool.segment_pool[i].previous=
580 (&memory_pool.segment_pool[i-1]);
581 if (i != (MaxSegments-1))
582 memory_pool.segment_pool[i].next=(&memory_pool.segment_pool[i+1]);
584 free_segments=(&memory_pool.segment_pool[0]);
586 UnlockSemaphoreInfo(memory_semaphore);
588 LockSemaphoreInfo(memory_semaphore);
589 memory=AcquireBlock(size == 0 ? 1UL : size);
590 if (memory == (
void *) NULL)
592 if (ExpandHeap(size == 0 ? 1UL : size) != MagickFalse)
593 memory=AcquireBlock(size == 0 ? 1UL : size);
595 UnlockSemaphoreInfo(memory_semaphore);
628MagickExport
void *AcquireCriticalMemory(
const size_t size)
636 memory=AcquireMagickMemory(size);
637 if (memory == (
void *) NULL)
638 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
667MagickExport
void *AcquireQuantumMemory(
const size_t count,
const size_t quantum)
672 if ((HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse) ||
673 (size > GetMaxMemoryRequest()))
678 return(AcquireMagickMemory(size));
707MagickExport
MemoryInfo *AcquireVirtualMemory(
const size_t count,
708 const size_t quantum)
719 if (HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse)
724 if (virtual_anonymous_memory == 0)
726 virtual_anonymous_memory=1;
727 value=GetPolicyValue(
"system:memory-map");
728 if (LocaleCompare(value,
"anonymous") == 0)
733#if defined(MAGICKCORE_HAVE_MMAP) && defined(MAP_ANONYMOUS)
734 virtual_anonymous_memory=2;
737 value=DestroyString(value);
739 memory_info=(
MemoryInfo *) MagickAssumeAligned(AcquireAlignedMemory(1,
740 sizeof(*memory_info)));
742 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
743 (void) memset(memory_info,0,
sizeof(*memory_info));
744 memory_info->length=size;
745 memory_info->signature=MagickCoreSignature;
746 if ((virtual_anonymous_memory == 1) && (size <= GetMaxMemoryRequest()))
748 memory_info->blob=AcquireAlignedMemory(1,size);
749 if (memory_info->blob != NULL)
750 memory_info->type=AlignedVirtualMemory;
752 if (memory_info->blob == NULL)
757 memory_info->blob=NULL;
758 if (size <= GetMaxMemoryRequest())
759 memory_info->blob=MapBlob(-1,IOMode,0,size);
760 if (memory_info->blob != NULL)
761 memory_info->type=MapVirtualMemory;
770 file=AcquireUniqueFileResource(memory_info->filename);
776 offset=(MagickOffsetType) lseek(file,(off_t) (size-1),SEEK_SET);
777 if ((offset == (MagickOffsetType) (size-1)) &&
778 (write(file,
"",1) == 1))
780#if !defined(MAGICKCORE_HAVE_POSIX_FALLOCATE)
781 memory_info->blob=MapBlob(file,IOMode,0,size);
783 if (posix_fallocate(file,0,(MagickOffsetType) size) == 0)
784 memory_info->blob=MapBlob(file,IOMode,0,size);
786 if (memory_info->blob != NULL)
787 memory_info->type=MapVirtualMemory;
790 (void) RelinquishUniqueFileResource(
791 memory_info->filename);
792 *memory_info->filename=
'\0';
795 (void) close_utf8(file);
799 if (memory_info->blob == NULL)
801 memory_info->blob=AcquireQuantumMemory(1,size);
802 if (memory_info->blob != NULL)
803 memory_info->type=UnalignedVirtualMemory;
805 if (memory_info->blob == NULL)
806 memory_info=RelinquishVirtualMemory(memory_info);
839MagickExport
void *CopyMagickMemory(
void *magick_restrict destination,
840 const void *magick_restrict source,
const size_t size)
848 assert(destination != (
void *) NULL);
849 assert(source != (
const void *) NULL);
850 p=(
const unsigned char *) source;
851 q=(
unsigned char *) destination;
852 if (((q+size) < p) || (q > (p+size)))
855 default:
return(memcpy(destination,source,size));
856 case 8: *q++=(*p++); magick_fallthrough;
857 case 7: *q++=(*p++); magick_fallthrough;
858 case 6: *q++=(*p++); magick_fallthrough;
859 case 5: *q++=(*p++); magick_fallthrough;
860 case 4: *q++=(*p++); magick_fallthrough;
861 case 3: *q++=(*p++); magick_fallthrough;
862 case 2: *q++=(*p++); magick_fallthrough;
863 case 1: *q++=(*p++); magick_fallthrough;
864 case 0:
return(destination);
866 return(memmove(destination,source,size));
887MagickExport
void DestroyMagickMemory(
void)
889#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
894 ActivateSemaphoreInfo(&memory_semaphore);
895 LockSemaphoreInfo(memory_semaphore);
896 for (i=0; i < (ssize_t) memory_pool.number_segments; i++)
897 if (memory_pool.segments[i]->mapped == MagickFalse)
898 memory_methods.destroy_memory_handler(
899 memory_pool.segments[i]->allocation);
901 (
void) UnmapBlob(memory_pool.segments[i]->allocation,
902 memory_pool.segments[i]->length);
904 (void) memset(&memory_pool,0,
sizeof(memory_pool));
905 UnlockSemaphoreInfo(memory_semaphore);
906 RelinquishSemaphoreInfo(&memory_semaphore);
910#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
934static MagickBooleanType ExpandHeap(
size_t size)
954 blocksize=((size+12*
sizeof(size_t))+SegmentSize-1) & -SegmentSize;
955 assert(memory_pool.number_segments < MaxSegments);
956 segment=MapBlob(-1,IOMode,0,blocksize);
957 mapped=segment != (
void *) NULL ? MagickTrue : MagickFalse;
958 if (segment == (
void *) NULL)
959 segment=(
void *) memory_methods.acquire_memory_handler(blocksize);
960 if (segment == (
void *) NULL)
963 free_segments=segment_info->next;
964 segment_info->mapped=mapped;
965 segment_info->length=blocksize;
966 segment_info->allocation=segment;
967 segment_info->bound=(
char *) segment+blocksize;
968 i=(ssize_t) memory_pool.number_segments-1;
969 for ( ; (i >= 0) && (memory_pool.segments[i]->allocation > segment); i--)
970 memory_pool.segments[i+1]=memory_pool.segments[i];
971 memory_pool.segments[i+1]=segment_info;
972 memory_pool.number_segments++;
973 size=blocksize-12*
sizeof(size_t);
974 block=(
char *) segment_info->allocation+4*
sizeof(
size_t);
975 *BlockHeader(block)=size | PreviousBlockBit;
976 *BlockFooter(block,size)=size;
977 InsertFreeBlock(block,AllocationPolicy(size));
978 block=NextBlock(block);
979 assert(block < segment_info->bound);
980 *BlockHeader(block)=2*
sizeof(size_t);
981 *BlockHeader(NextBlock(block))=PreviousBlockBit;
1015MagickExport
void GetMagickMemoryMethods(
1016 AcquireMemoryHandler *acquire_memory_handler,
1017 ResizeMemoryHandler *resize_memory_handler,
1018 DestroyMemoryHandler *destroy_memory_handler)
1020 assert(acquire_memory_handler != (AcquireMemoryHandler *) NULL);
1021 assert(resize_memory_handler != (ResizeMemoryHandler *) NULL);
1022 assert(destroy_memory_handler != (DestroyMemoryHandler *) NULL);
1023 *acquire_memory_handler=memory_methods.acquire_memory_handler;
1024 *resize_memory_handler=memory_methods.resize_memory_handler;
1025 *destroy_memory_handler=memory_methods.destroy_memory_handler;
1046MagickExport
size_t GetMaxMemoryRequest(
void)
1048#define MinMemoryRequest "16MiB"
1050 if (max_memory_request == 0)
1055 max_memory_request=(size_t) MAGICK_SSIZE_MAX;
1056 value=GetPolicyValue(
"system:max-memory-request");
1057 if (value != (
char *) NULL)
1062 max_memory_request=MagickMax(StringToSizeType(value,100.0),
1063 StringToSizeType(MinMemoryRequest,100.0));
1064 value=DestroyString(value);
1067 return(MagickMin(max_memory_request,(
size_t) MAGICK_SSIZE_MAX));
1087MagickExport
size_t GetMaxProfileSize(
void)
1089 if (max_profile_size == 0)
1094 max_profile_size=(size_t) MAGICK_SSIZE_MAX;
1095 value=GetPolicyValue(
"system:max-profile-size");
1096 if (value != (
char *) NULL)
1101 max_profile_size=StringToSizeType(value,100.0);
1102 value=DestroyString(value);
1105 return(MagickMin(max_profile_size,(
size_t) MAGICK_SSIZE_MAX));
1130MagickExport
void *GetVirtualMemoryBlob(
const MemoryInfo *memory_info)
1132 assert(memory_info != (
const MemoryInfo *) NULL);
1133 assert(memory_info->signature == MagickCoreSignature);
1134 return(memory_info->blob);
1160MagickExport
void *RelinquishAlignedMemory(
void *memory)
1162 if (memory == (
void *) NULL)
1163 return((
void *) NULL);
1164 if (memory_methods.relinquish_aligned_memory_handler != (RelinquishAlignedMemoryHandler) NULL)
1166 memory_methods.relinquish_aligned_memory_handler(memory);
1169#if defined(MAGICKCORE_HAVE_ALIGNED_MALLOC) || defined(MAGICKCORE_HAVE_POSIX_MEMALIGN)
1171#elif defined(MAGICKCORE_HAVE__ALIGNED_MALLOC)
1172 _aligned_free(memory);
1174 RelinquishMagickMemory(actual_base_address(memory));
1202MagickExport
void *RelinquishMagickMemory(
void *memory)
1204 if (memory == (
void *) NULL)
1205 return((
void *) NULL);
1206#if !defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
1207 memory_methods.destroy_memory_handler(memory);
1209 LockSemaphoreInfo(memory_semaphore);
1210 assert((SizeOfBlock(memory) % (4*
sizeof(
size_t))) == 0);
1211 assert((*BlockHeader(NextBlock(memory)) & PreviousBlockBit) != 0);
1212 if ((*BlockHeader(memory) & PreviousBlockBit) == 0)
1220 previous=PreviousBlock(memory);
1221 RemoveFreeBlock(previous,AllocationPolicy(SizeOfBlock(previous)));
1222 *BlockHeader(previous)=(SizeOfBlock(previous)+SizeOfBlock(memory)) |
1223 (*BlockHeader(previous) & ~SizeMask);
1226 if ((*BlockHeader(NextBlock(NextBlock(memory))) & PreviousBlockBit) == 0)
1234 next=NextBlock(memory);
1235 RemoveFreeBlock(next,AllocationPolicy(SizeOfBlock(next)));
1236 *BlockHeader(memory)=(SizeOfBlock(memory)+SizeOfBlock(next)) |
1237 (*BlockHeader(memory) & ~SizeMask);
1239 *BlockFooter(memory,SizeOfBlock(memory))=SizeOfBlock(memory);
1240 *BlockHeader(NextBlock(memory))&=(~PreviousBlockBit);
1241 InsertFreeBlock(memory,AllocationPolicy(SizeOfBlock(memory)));
1242 UnlockSemaphoreInfo(memory_semaphore);
1244 return((
void *) NULL);
1272 assert(memory_info->signature == MagickCoreSignature);
1273 if (memory_info->blob != (
void *) NULL)
1274 switch (memory_info->type)
1276 case AlignedVirtualMemory:
1278 (void) ShredMagickMemory(memory_info->blob,memory_info->length);
1279 memory_info->blob=RelinquishAlignedMemory(memory_info->blob);
1282 case MapVirtualMemory:
1284 (void) UnmapBlob(memory_info->blob,memory_info->length);
1285 memory_info->blob=NULL;
1286 if (*memory_info->filename !=
'\0')
1287 (void) RelinquishUniqueFileResource(memory_info->filename);
1290 case UnalignedVirtualMemory:
1293 (void) ShredMagickMemory(memory_info->blob,memory_info->length);
1294 memory_info->blob=RelinquishMagickMemory(memory_info->blob);
1298 memory_info->signature=(~MagickCoreSignature);
1299 memory_info=(
MemoryInfo *) RelinquishAlignedMemory(memory_info);
1300 return(memory_info);
1332MagickExport
void *ResetMagickMemory(
void *memory,
int c,
const size_t size)
1334 volatile unsigned char
1335 *p = (
volatile unsigned char *) memory;
1340 assert(memory != (
void *) NULL);
1342 *p++=(
unsigned char) c;
1364MagickPrivate
void ResetMaxMemoryRequest(
void)
1366 max_memory_request=0;
1387MagickPrivate
void ResetVirtualAnonymousMemory(
void)
1389 virtual_anonymous_memory=0;
1419#if defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
1420static inline void *ResizeBlock(
void *block,
size_t size)
1425 if (block == (
void *) NULL)
1426 return(AcquireBlock(size));
1427 memory=AcquireBlock(size);
1428 if (memory == (
void *) NULL)
1429 return((
void *) NULL);
1430 if (size <= (SizeOfBlock(block)-
sizeof(
size_t)))
1431 (void) memcpy(memory,block,size);
1433 (
void) memcpy(memory,block,SizeOfBlock(block)-
sizeof(
size_t));
1434 memory_pool.allocation+=size;
1439MagickExport
void *ResizeMagickMemory(
void *memory,
const size_t size)
1444 if (memory == (
void *) NULL)
1445 return(AcquireMagickMemory(size));
1446#if !defined(MAGICKCORE_ANONYMOUS_MEMORY_SUPPORT)
1447 block=memory_methods.resize_memory_handler(memory,size == 0 ? 1UL : size);
1448 if (block == (
void *) NULL)
1449 memory=RelinquishMagickMemory(memory);
1451 LockSemaphoreInfo(memory_semaphore);
1452 block=ResizeBlock(memory,size == 0 ? 1UL : size);
1453 if (block == (
void *) NULL)
1455 if (ExpandHeap(size == 0 ? 1UL : size) == MagickFalse)
1457 UnlockSemaphoreInfo(memory_semaphore);
1458 memory=RelinquishMagickMemory(memory);
1459 ThrowFatalException(ResourceLimitFatalError,
"MemoryAllocationFailed");
1461 block=ResizeBlock(memory,size == 0 ? 1UL : size);
1462 assert(block != (
void *) NULL);
1464 UnlockSemaphoreInfo(memory_semaphore);
1465 memory=RelinquishMagickMemory(memory);
1499MagickExport
void *ResizeQuantumMemory(
void *memory,
const size_t count,
1500 const size_t quantum)
1505 if ((HeapOverflowSanityCheckGetSize(count,quantum,&size) != MagickFalse) ||
1506 (size > GetMaxMemoryRequest()))
1509 memory=RelinquishMagickMemory(memory);
1512 return(ResizeMagickMemory(memory,size));
1542MagickExport
void SetMagickAlignedMemoryMethods(
1543 AcquireAlignedMemoryHandler acquire_aligned_memory_handler,
1544 RelinquishAlignedMemoryHandler relinquish_aligned_memory_handler)
1546 memory_methods.acquire_aligned_memory_handler=acquire_aligned_memory_handler;
1547 memory_methods.relinquish_aligned_memory_handler=
1548 relinquish_aligned_memory_handler;
1581MagickExport
void SetMagickMemoryMethods(
1582 AcquireMemoryHandler acquire_memory_handler,
1583 ResizeMemoryHandler resize_memory_handler,
1584 DestroyMemoryHandler destroy_memory_handler)
1589 if (acquire_memory_handler != (AcquireMemoryHandler) NULL)
1590 memory_methods.acquire_memory_handler=acquire_memory_handler;
1591 if (resize_memory_handler != (ResizeMemoryHandler) NULL)
1592 memory_methods.resize_memory_handler=resize_memory_handler;
1593 if (destroy_memory_handler != (DestroyMemoryHandler) NULL)
1594 memory_methods.destroy_memory_handler=destroy_memory_handler;
1619MagickPrivate
void SetMaxMemoryRequest(
const MagickSizeType limit)
1621 max_memory_request=MagickMin(limit,GetMaxMemoryRequest());
1646MagickPrivate
void SetMaxProfileSize(
const MagickSizeType limit)
1648 max_profile_size=MagickMin(limit,GetMaxProfileSize());
1677MagickPrivate MagickBooleanType ShredMagickMemory(
void *memory,
1678 const size_t length)
1695 if ((memory == NULL) || (length == 0))
1696 return(MagickFalse);
1703 property=GetEnvironmentValue(
"MAGICK_SHRED_PASSES");
1704 if (property != (
char *) NULL)
1706 passes=(ssize_t) StringToInteger(property);
1707 property=DestroyString(property);
1709 property=GetPolicyValue(
"system:shred");
1710 if (property != (
char *) NULL)
1712 passes=(ssize_t) StringToInteger(property);
1713 property=DestroyString(property);
1721 quantum=(size_t) MagickMin(length,MagickMinBufferExtent);
1722 random_info=AcquireRandomInfo();
1723 key=GetRandomKey(random_info,quantum);
1724 for (i=0; i < passes; i++)
1730 *p = (
unsigned char *) memory;
1732 for (j=0; j < length; j+=quantum)
1735 SetRandomKey(random_info,quantum,GetStringInfoDatum(key));
1736 (void) memcpy(p,GetStringInfoDatum(key),(size_t)
1737 MagickMin(quantum,length-j));
1738 p+=(ptrdiff_t) quantum;
1743 key=DestroyStringInfo(key);
1744 random_info=DestroyRandomInfo(random_info);
1745 return(i < passes ? MagickFalse : MagickTrue);