📄 ttmemory.c
字号:
/******************************************************************* * * ttmemory.c 1.2 * * Memory management component (body). * * Copyright 1996-1999 by * David Turner, Robert Wilhelm, and Werner Lemberg. * * This file is part of the FreeType project, and may only be used * modified and distributed under the terms of the FreeType project * license, LICENSE.TXT. By continuing to use, modify, or distribute * this file you indicate that you have read the license and * understand and accept it fully. * * * Changes between 1.1 and 1.2: * * - the font pool is gone. * * - introduced the FREE macro and the Free function for * future use in destructors. * * - Init_FontPool() is now a macro to allow the compilation of * 'legacy' applications (all four test programs have been updated). * ******************************************************************/#include "ttdebug.h"#include "ttmemory.h"#include "ttengine.h"/* required by the tracing mode */#undef TT_COMPONENT#define TT_COMPONENT trace_memory#ifdef DEBUG_MEMORY#include <stdio.h>#define MAX_TRACKED_BLOCKS 1024 struct TMemRec_ { void* base; Long size; }; typedef struct TMemRec_ TMemRec; static TMemRec pointers[MAX_TRACKED_BLOCKS + 1]; static Int num_alloc; static Int num_free; static Int num_realloc; /* counts only `real' reallocations (i.e., an existing buffer will be resized to a value larger than zero */ static Int fail_alloc; static Int fail_realloc; static Int fail_free;#endif /* DEBUG_MEMORY */#ifndef TT_CONFIG_OPTION_THREAD_SAFE Long TTMemory_Allocated; Long TTMemory_MaxAllocated;#endif/******************************************************************* * * Function : TT_Alloc * * Description : Allocates memory from the heap buffer. * * Input : Size size of the memory to be allocated * P pointer to a buffer pointer * * Output : Error code. * * NOTE : The newly allocated block should _always_ be zeroed * on return. Many parts of the engine rely on this to * work properly. * ******************************************************************/ EXPORT_FUNC TT_Error TT_Alloc( ULong Size, void** P ) {#ifdef DEBUG_MEMORY Int i;#endif if ( !P ) return TT_Err_Invalid_Argument; if ( Size > (size_t)-1 ) return TT_Err_Out_Of_Memory; if ( Size > 0 ) { *P = (void*)malloc( Size ); if ( !*P ) return TT_Err_Out_Of_Memory;#ifndef TT_CONFIG_OPTION_THREAD_SAFE TTMemory_Allocated += Size; TTMemory_MaxAllocated += Size;#endif#ifdef DEBUG_MEMORY num_alloc++; i = 0; while ( i < MAX_TRACKED_BLOCKS && pointers[i].base != NULL ) i++; if ( i >= MAX_TRACKED_BLOCKS ) fail_alloc++; else { pointers[i].base = *P; pointers[i].size = Size; }#endif /* DEBUG_MEMORY */ MEM_Set( *P, 0, Size ); } else *P = NULL; return TT_Err_Ok; }#ifdef TT_CONFIG_OPTION_EXTEND_ENGINE/******************************************************************* * * Function : TT_Realloc * * Description : Reallocates memory from the heap buffer. * * Input : Size new size of the memory to be allocated; * if zero, TT_Free() will be called * P pointer to a buffer pointer; if *P == NULL, * TT_Alloc() will be called * * Output : Error code. * * NOTES : It's not necessary to zero the memory in case the * reallocated buffer is larger than before -- the * application has to take care of this. * * If the memory request fails, TT_Free() will be * called on *P, and TT_Err_Out_Of_Memory returned. * ******************************************************************/ EXPORT_FUNC TT_Error TT_Realloc( ULong Size, void** P ) { void* Q;#ifdef DEBUG_MEMORY Int i;#endif if ( !P ) return TT_Err_Invalid_Argument; if ( !*P ) return TT_Alloc( Size, P ); if ( Size == 0 ) return TT_Free( P ); if ( Size > (size_t)-1 ) { TT_Free( *P ); return TT_Err_Out_Of_Memory; } Q = (void*)realloc( *P, Size ); if ( !Q ) { TT_Free( *P ); return TT_Err_Out_Of_Memory; }#ifdef DEBUG_MEMORY num_realloc++; i = 0; while ( i < MAX_TRACKED_BLOCKS && pointers[i].base != *P ) i++; if ( i >= MAX_TRACKED_BLOCKS ) fail_realloc++; else {#ifndef TT_CONFIG_OPTION_THREAD_SAFE TTMemory_Allocated += Size - pointers[i].size; if ( Size > pointers[i].size ) TTMemory_MaxAllocated += Size - pointers[i].size;#endif pointers[i].base = Q; pointers[i].size = size; }#endif /* DEBUG_MEMORY */ *P = Q; return TT_Err_Ok; }#endif /* TT_CONFIG_OPTION_EXTEND_ENGINE *//******************************************************************* * * Function : TT_Free * * Description : Releases a previously allocated block of memory. * * Input : P pointer to memory block * * Output : Always SUCCESS. * * Note : The pointer must _always_ be set to NULL by this function. * ******************************************************************/ EXPORT_FUNC TT_Error TT_Free( void** P ) {#ifdef DEBUG_MEMORY Int i;#endif /* DEBUG_MEMORY */ if ( !P || !*P ) return TT_Err_Ok;#ifdef DEBUG_MEMORY num_free++; i = 0; while ( i < MAX_TRACKED_BLOCKS && pointers[i].base != *P ) i++; if ( i >= MAX_TRACKED_BLOCKS ) fail_free++; else {#ifndef TT_CONFIG_OPTION_THREAD_SAFE TTMemory_Allocated -= pointers[i].size;#endif pointers[i].base = NULL; pointers[i].size = 0; }#endif /* DEBUG_MEMORY */ free( *P ); *P = NULL; return TT_Err_Ok; }/******************************************************************* * * Function : TTMemory_Init * * Description : Initializes the memory. * * Output : Always SUCCESS. * ******************************************************************/ LOCAL_FUNC TT_Error TTMemory_Init( void ) {#ifdef DEBUG_MEMORY Int i; for ( i = 0; i < MAX_TRACKED_BLOCKS; i++ ) { pointers[i].base = NULL; pointers[i].size = 0; } num_alloc = 0; num_realloc = 0; num_free = 0; fail_alloc = 0; fail_realloc = 0; fail_free = 0;#endif#ifndef TT_CONFIG_OPTION_THREAD_SAFE TTMemory_Allocated = 0; TTMemory_MaxAllocated = 0;#endif return TT_Err_Ok; }/******************************************************************* * * Function : TTMemory_Done * * Description : Finalizes memory usage. * * Output : Always SUCCESS. * ******************************************************************/ LOCAL_FUNC TT_Error TTMemory_Done( void ) {#ifdef DEBUG_MEMORY Int i, num_leaked, tot_leaked; num_leaked = 0; tot_leaked = 0; for ( i = 0; i < MAX_TRACKED_BLOCKS; i++ ) { if ( pointers[i].base ) { num_leaked ++; tot_leaked += pointers[i].size; } } fprintf( stderr, "%d memory allocations, of which %d failed\n", num_alloc, fail_alloc ); fprintf( stderr, "%d memory reallocations, of which %d failed\n", num_realloc, fail_realloc ); fprintf( stderr, "%d memory frees, of which %d failed\n", num_free, fail_free ); if ( num_leaked > 0 ) { fprintf( stderr, "There are %d leaked memory blocks, totalizing %d bytes\n", num_leaked, tot_leaked ); for ( i = 0; i < MAX_TRACKED_BLOCKS; i++ ) { if ( pointers[i].base ) { fprintf( stderr, "index: %4d (base: $%08lx, size: %08ld)\n", i, (long)pointers[i].base, pointers[i].size ); } } } else fprintf( stderr, "No memory leaks !\n" );#endif /* DEBUG_MEMORY */ return TT_Err_Ok; }/* END */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -