📄 mallocr.c
字号:
list of macros. */#ifndef SEPARATE_OBJECTS#define DEFINE_MALLOC#define DEFINE_FREE#define DEFINE_REALLOC#define DEFINE_CALLOC#define DEFINE_CFREE#define DEFINE_MEMALIGN#define DEFINE_VALLOC#define DEFINE_PVALLOC#define DEFINE_MALLINFO#define DEFINE_MALLOC_STATS#define DEFINE_MALLOC_USABLE_SIZE#define DEFINE_MALLOPT#define STATIC static#else#define STATIC#endif/* Define MALLOC_LOCK and MALLOC_UNLOCK to C expressions to run to lock and unlock the malloc data structures. MALLOC_LOCK may be called recursively. */#ifndef MALLOC_LOCK#define MALLOC_LOCK#endif#ifndef MALLOC_UNLOCK#define MALLOC_UNLOCK#endif/* INTERNAL_SIZE_T is the word-size used for internal bookkeeping of chunk sizes. On a 64-bit machine, you can reduce malloc overhead by defining INTERNAL_SIZE_T to be a 32 bit `unsigned int' at the expense of not being able to handle requests greater than 2^31. This limitation is hardly ever a concern; you are encouraged to set this. However, the default version is the same as size_t.*/#ifndef INTERNAL_SIZE_T#define INTERNAL_SIZE_T size_t#endif/* Following is needed on implementations whereby long > size_t. The problem is caused because the code performs subtractions of size_t values and stores the result in long values. In the case where long > size_t and the first value is actually less than the second value, the resultant value is positive. For example, (long)(x - y) where x = 0 and y is 1 ends up being 0x00000000FFFFFFFF which is 2*31 - 1 instead of 0xFFFFFFFFFFFFFFFF. This is due to the fact that assignment from unsigned to signed won't sign extend.*/#define long_sub_size_t(x, y) \ (sizeof (long) > sizeof (INTERNAL_SIZE_T) && x < y \ ? -(long) (y - x) \ : (long) (x - y))/* REALLOC_ZERO_BYTES_FREES should be set if a call to realloc with zero bytes should be the same as a call to free. Some people think it should. Otherwise, since this malloc returns a unique pointer for malloc(0), so does realloc(p, 0). *//* #define REALLOC_ZERO_BYTES_FREES *//* WIN32 causes an emulation of sbrk to be compiled in mmap-based options are not currently supported in WIN32.*//* #define WIN32 */#ifdef WIN32#define MORECORE wsbrk#define HAVE_MMAP 0#endif/* HAVE_MEMCPY should be defined if you are not otherwise using ANSI STD C, but still have memcpy and memset in your C library and want to use them in calloc and realloc. Otherwise simple macro versions are defined here. USE_MEMCPY should be defined as 1 if you actually want to have memset and memcpy called. People report that the macro versions are often enough faster than libc versions on many systems that it is better to use them. */#define HAVE_MEMCPY /* Although the original macro is called USE_MEMCPY, newlib actually uses memmove to handle cases whereby a platform's memcpy implementation copies backwards and thus destructive overlap may occur in realloc whereby we are reclaiming free memory prior to the old allocation. */#ifndef USE_MEMCPY#ifdef HAVE_MEMCPY#define USE_MEMCPY 1#else#define USE_MEMCPY 0#endif#endif#if (__STD_C || defined(HAVE_MEMCPY)) #if __STD_Cvoid* memset(void*, int, size_t);void* memcpy(void*, const void*, size_t);void* memmove(void*, const void*, size_t);#elseVoid_t* memset();Void_t* memcpy();Void_t* memmove();#endif#endif#if USE_MEMCPY/* The following macros are only invoked with (2n+1)-multiples of INTERNAL_SIZE_T units, with a positive integer n. This is exploited for fast inline execution when n is small. */#define MALLOC_ZERO(charp, nbytes) \do { \ INTERNAL_SIZE_T mzsz = (nbytes); \ if(mzsz <= 9*sizeof(mzsz)) { \ INTERNAL_SIZE_T* mz = (INTERNAL_SIZE_T*) (charp); \ if(mzsz >= 5*sizeof(mzsz)) { *mz++ = 0; \ *mz++ = 0; \ if(mzsz >= 7*sizeof(mzsz)) { *mz++ = 0; \ *mz++ = 0; \ if(mzsz >= 9*sizeof(mzsz)) { *mz++ = 0; \ *mz++ = 0; }}} \ *mz++ = 0; \ *mz++ = 0; \ *mz = 0; \ } else memset((charp), 0, mzsz); \} while(0)#define MALLOC_COPY(dest,src,nbytes) \do { \ INTERNAL_SIZE_T mcsz = (nbytes); \ if(mcsz <= 9*sizeof(mcsz)) { \ INTERNAL_SIZE_T* mcsrc = (INTERNAL_SIZE_T*) (src); \ INTERNAL_SIZE_T* mcdst = (INTERNAL_SIZE_T*) (dest); \ if(mcsz >= 5*sizeof(mcsz)) { *mcdst++ = *mcsrc++; \ *mcdst++ = *mcsrc++; \ if(mcsz >= 7*sizeof(mcsz)) { *mcdst++ = *mcsrc++; \ *mcdst++ = *mcsrc++; \ if(mcsz >= 9*sizeof(mcsz)) { *mcdst++ = *mcsrc++; \ *mcdst++ = *mcsrc++; }}} \ *mcdst++ = *mcsrc++; \ *mcdst++ = *mcsrc++; \ *mcdst = *mcsrc ; \ } else memmove(dest, src, mcsz); \} while(0)#else /* !USE_MEMCPY *//* Use Duff's device for good zeroing/copying performance. */#define MALLOC_ZERO(charp, nbytes) \do { \ INTERNAL_SIZE_T* mzp = (INTERNAL_SIZE_T*)(charp); \ long mctmp = (nbytes)/sizeof(INTERNAL_SIZE_T), mcn; \ if (mctmp < 8) mcn = 0; else { mcn = (mctmp-1)/8; mctmp %= 8; } \ switch (mctmp) { \ case 0: for(;;) { *mzp++ = 0; \ case 7: *mzp++ = 0; \ case 6: *mzp++ = 0; \ case 5: *mzp++ = 0; \ case 4: *mzp++ = 0; \ case 3: *mzp++ = 0; \ case 2: *mzp++ = 0; \ case 1: *mzp++ = 0; if(mcn <= 0) break; mcn--; } \ } \} while(0)#define MALLOC_COPY(dest,src,nbytes) \do { \ INTERNAL_SIZE_T* mcsrc = (INTERNAL_SIZE_T*) src; \ INTERNAL_SIZE_T* mcdst = (INTERNAL_SIZE_T*) dest; \ long mctmp = (nbytes)/sizeof(INTERNAL_SIZE_T), mcn; \ if (mctmp < 8) mcn = 0; else { mcn = (mctmp-1)/8; mctmp %= 8; } \ switch (mctmp) { \ case 0: for(;;) { *mcdst++ = *mcsrc++; \ case 7: *mcdst++ = *mcsrc++; \ case 6: *mcdst++ = *mcsrc++; \ case 5: *mcdst++ = *mcsrc++; \ case 4: *mcdst++ = *mcsrc++; \ case 3: *mcdst++ = *mcsrc++; \ case 2: *mcdst++ = *mcsrc++; \ case 1: *mcdst++ = *mcsrc++; if(mcn <= 0) break; mcn--; } \ } \} while(0)#endif/* Define HAVE_MMAP to optionally make malloc() use mmap() to allocate very large blocks. These will be returned to the operating system immediately after a free().*/#ifndef HAVE_MMAP#define HAVE_MMAP 1#endif/* Define HAVE_MREMAP to make realloc() use mremap() to re-allocate large blocks. This is currently only possible on Linux with kernel versions newer than 1.3.77.*/#ifndef HAVE_MREMAP#ifdef INTERNAL_LINUX_C_LIB#define HAVE_MREMAP 1#else#define HAVE_MREMAP 0#endif#endif#if HAVE_MMAP#include <unistd.h>#include <fcntl.h>#include <sys/mman.h>#if !defined(MAP_ANONYMOUS) && defined(MAP_ANON)#define MAP_ANONYMOUS MAP_ANON#endif#endif /* HAVE_MMAP *//* Access to system page size. To the extent possible, this malloc manages memory from the system in page-size units. The following mechanics for getpagesize were adapted from bsd/gnu getpagesize.h */#ifndef LACKS_UNISTD_H# include <unistd.h>#endif#ifndef malloc_getpagesize# ifdef _SC_PAGESIZE /* some SVR4 systems omit an underscore */# ifndef _SC_PAGE_SIZE# define _SC_PAGE_SIZE _SC_PAGESIZE# endif# endif# ifdef _SC_PAGE_SIZE# define malloc_getpagesize sysconf(_SC_PAGE_SIZE)# else# if defined(BSD) || defined(DGUX) || defined(HAVE_GETPAGESIZE) extern size_t getpagesize();# define malloc_getpagesize getpagesize()# else# include <sys/param.h># ifdef EXEC_PAGESIZE# define malloc_getpagesize EXEC_PAGESIZE# else# ifdef NBPG# ifndef CLSIZE# define malloc_getpagesize NBPG# else# define malloc_getpagesize (NBPG * CLSIZE)# endif# else # ifdef NBPC# define malloc_getpagesize NBPC# else# ifdef PAGESIZE# define malloc_getpagesize PAGESIZE# else# define malloc_getpagesize (4096) /* just guess */# endif# endif# endif # endif# endif # endif#endif/* This version of malloc supports the standard SVID/XPG mallinfo routine that returns a struct containing the same kind of information you can get from malloc_stats. It should work on any SVID/XPG compliant system that has a /usr/include/malloc.h defining struct mallinfo. (If you'd like to install such a thing yourself, cut out the preliminary declarations as described above and below and save them in a malloc.h file. But there's no compelling reason to bother to do this.) The main declaration needed is the mallinfo struct that is returned (by-copy) by mallinfo(). The SVID/XPG malloinfo struct contains a bunch of fields, most of which are not even meaningful in this version of malloc. Some of these fields are are instead filled by mallinfo() with other numbers that might possibly be of interest. HAVE_USR_INCLUDE_MALLOC_H should be set if you have a /usr/include/malloc.h file that includes a declaration of struct mallinfo. If so, it is included; else an SVID2/XPG2 compliant version is declared below. These must be precisely the same for mallinfo() to work.*//* #define HAVE_USR_INCLUDE_MALLOC_H */#if HAVE_USR_INCLUDE_MALLOC_H#include "/usr/include/malloc.h"#else/* SVID2/XPG mallinfo structure */struct mallinfo { int arena; /* total space allocated from system */ int ordblks; /* number of non-inuse chunks */ int smblks; /* unused -- always zero */ int hblks; /* number of mmapped regions */ int hblkhd; /* total space in mmapped regions */ int usmblks; /* unused -- always zero */ int fsmblks; /* unused -- always zero */ int uordblks; /* total allocated space */ int fordblks; /* total non-inuse space */ int keepcost; /* top-most, releasable (via malloc_trim) space */}; /* SVID2/XPG mallopt options */#define M_MXFAST 1 /* UNUSED in this malloc */#define M_NLBLKS 2 /* UNUSED in this malloc */#define M_GRAIN 3 /* UNUSED in this malloc */#define M_KEEP 4 /* UNUSED in this malloc */#endif/* mallopt options that actually do something */#define M_TRIM_THRESHOLD -1#define M_TOP_PAD -2#define M_MMAP_THRESHOLD -3#define M_MMAP_MAX -4#ifndef DEFAULT_TRIM_THRESHOLD#define DEFAULT_TRIM_THRESHOLD (128L * 1024L)#endif/* M_TRIM_THRESHOLD is the maximum amount of unused top-most memory to keep before releasing via malloc_trim in free(). Automatic trimming is mainly useful in long-lived programs. Because trimming via sbrk can be slow on some systems, and can sometimes be wasteful (in cases where programs immediately afterward allocate more large chunks) the value should be high enough so that your overall system performance would improve by releasing. The trim threshold and the mmap control parameters (see below) can be traded off with one another. Trimming and mmapping are two different ways of releasing unused memory back to the system. Between these two, it is often possible to keep system-level demands of a long-lived program down to a bare minimum. For example, in one test suite of sessions measuring the XF86 X server on Linux, using a trim threshold of 128K and a mmap threshold of 192K led to near-minimal long term resource consumption.
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -