146 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			146 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|  | /* Alloc.c -- Memory allocation functions
 | ||
|  | 2017-06-15 : Igor Pavlov : Public domain */ | ||
|  | 
 | ||
|  | #include "Precomp.h"
 | ||
|  | 
 | ||
|  | #ifdef _WIN32
 | ||
|  | #include <windows.h>
 | ||
|  | #endif
 | ||
|  | #include <stdlib.h>
 | ||
|  | 
 | ||
|  | #include "Alloc.h"
 | ||
|  | 
 | ||
|  | /* #define _SZ_ALLOC_DEBUG */ | ||
|  | 
 | ||
|  | /* use _SZ_ALLOC_DEBUG to debug alloc/free operations */ | ||
|  | #ifdef _SZ_ALLOC_DEBUG
 | ||
|  | #include <stdio.h>
 | ||
|  | int g_allocCount = 0; | ||
|  | int g_allocCountMid = 0; | ||
|  | int g_allocCountBig = 0; | ||
|  | #endif
 | ||
|  | 
 | ||
|  | void *MyAlloc(size_t size) | ||
|  | { | ||
|  |   if (size == 0) | ||
|  |     return NULL; | ||
|  |   #ifdef _SZ_ALLOC_DEBUG
 | ||
|  |   { | ||
|  |     void *p = malloc(size); | ||
|  |     fprintf(stderr, "\nAlloc %10u bytes, count = %10d,  addr = %8X", size, g_allocCount++, (unsigned)p); | ||
|  |     return p; | ||
|  |   } | ||
|  |   #else
 | ||
|  |   return malloc(size); | ||
|  |   #endif
 | ||
|  | } | ||
|  | 
 | ||
|  | void MyFree(void *address) | ||
|  | { | ||
|  |   #ifdef _SZ_ALLOC_DEBUG
 | ||
|  |   if (address) | ||
|  |     fprintf(stderr, "\nFree; count = %10d,  addr = %8X", --g_allocCount, (unsigned)address); | ||
|  |   #endif
 | ||
|  |   free(address); | ||
|  | } | ||
|  | 
 | ||
|  | #ifdef _WIN32
 | ||
|  | 
 | ||
|  | void *MidAlloc(size_t size) | ||
|  | { | ||
|  |   if (size == 0) | ||
|  |     return NULL; | ||
|  |   #ifdef _SZ_ALLOC_DEBUG
 | ||
|  |   fprintf(stderr, "\nAlloc_Mid %10d bytes;  count = %10d", size, g_allocCountMid++); | ||
|  |   #endif
 | ||
|  |   return VirtualAlloc(NULL, size, MEM_COMMIT, PAGE_READWRITE); | ||
|  | } | ||
|  | 
 | ||
|  | void MidFree(void *address) | ||
|  | { | ||
|  |   #ifdef _SZ_ALLOC_DEBUG
 | ||
|  |   if (address) | ||
|  |     fprintf(stderr, "\nFree_Mid; count = %10d", --g_allocCountMid); | ||
|  |   #endif
 | ||
|  |   if (!address) | ||
|  |     return; | ||
|  |   VirtualFree(address, 0, MEM_RELEASE); | ||
|  | } | ||
|  | 
 | ||
|  | #ifndef MEM_LARGE_PAGES
 | ||
|  | #undef _7ZIP_LARGE_PAGES
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | #ifdef _7ZIP_LARGE_PAGES
 | ||
|  | SIZE_T g_LargePageSize = 0; | ||
|  | typedef SIZE_T (WINAPI *GetLargePageMinimumP)(); | ||
|  | #endif
 | ||
|  | 
 | ||
|  | void SetLargePageSize() | ||
|  | { | ||
|  |   #ifdef _7ZIP_LARGE_PAGES
 | ||
|  |   SIZE_T size; | ||
|  |   GetLargePageMinimumP largePageMinimum = (GetLargePageMinimumP) | ||
|  |         GetProcAddress(GetModuleHandle(TEXT("kernel32.dll")), "GetLargePageMinimum"); | ||
|  |   if (!largePageMinimum) | ||
|  |     return; | ||
|  |   size = largePageMinimum(); | ||
|  |   if (size == 0 || (size & (size - 1)) != 0) | ||
|  |     return; | ||
|  |   g_LargePageSize = size; | ||
|  |   #endif
 | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | void *BigAlloc(size_t size) | ||
|  | { | ||
|  |   if (size == 0) | ||
|  |     return NULL; | ||
|  |   #ifdef _SZ_ALLOC_DEBUG
 | ||
|  |   fprintf(stderr, "\nAlloc_Big %10u bytes;  count = %10d", size, g_allocCountBig++); | ||
|  |   #endif
 | ||
|  |    | ||
|  |   #ifdef _7ZIP_LARGE_PAGES
 | ||
|  |   { | ||
|  |     SIZE_T ps = g_LargePageSize; | ||
|  |     if (ps != 0 && ps <= (1 << 30) && size > (ps / 2)) | ||
|  |     { | ||
|  |       size_t size2; | ||
|  |       ps--; | ||
|  |       size2 = (size + ps) & ~ps; | ||
|  |       if (size2 >= size) | ||
|  |       { | ||
|  |         void *res = VirtualAlloc(NULL, size2, MEM_COMMIT | MEM_LARGE_PAGES, PAGE_READWRITE); | ||
|  |         if (res) | ||
|  |           return res; | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  |   #endif
 | ||
|  | 
 | ||
|  |   return VirtualAlloc(NULL, size, MEM_COMMIT, PAGE_READWRITE); | ||
|  | } | ||
|  | 
 | ||
|  | void BigFree(void *address) | ||
|  | { | ||
|  |   #ifdef _SZ_ALLOC_DEBUG
 | ||
|  |   if (address) | ||
|  |     fprintf(stderr, "\nFree_Big; count = %10d", --g_allocCountBig); | ||
|  |   #endif
 | ||
|  |    | ||
|  |   if (!address) | ||
|  |     return; | ||
|  |   VirtualFree(address, 0, MEM_RELEASE); | ||
|  | } | ||
|  | 
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | 
 | ||
|  | static void *SzAlloc(ISzAllocPtr p, size_t size) { UNUSED_VAR(p); return MyAlloc(size); } | ||
|  | static void SzFree(ISzAllocPtr p, void *address) { UNUSED_VAR(p); MyFree(address); } | ||
|  | ISzAlloc const g_Alloc = { SzAlloc, SzFree }; | ||
|  | 
 | ||
|  | static void *SzBigAlloc(ISzAllocPtr p, size_t size) { UNUSED_VAR(p); return BigAlloc(size); } | ||
|  | static void SzBigFree(ISzAllocPtr p, void *address) { UNUSED_VAR(p); BigFree(address); } | ||
|  | ISzAlloc const g_BigAlloc = { SzBigAlloc, SzBigFree }; |