提交 dcfbbe8a 编写于 作者: weixin_48148422's avatar weixin_48148422

implement memory leak detection

上级 f7e40d0e
...@@ -125,7 +125,11 @@ IF (NOT DEFINED TD_CLUSTER) ...@@ -125,7 +125,11 @@ IF (NOT DEFINED TD_CLUSTER)
# debug flag # debug flag
# #
# ADD_DEFINITIONS(-D_CHECK_HEADER_FILE_) # ADD_DEFINITIONS(-D_CHECK_HEADER_FILE_)
# ADD_DEFINITIONS(-D_TAOS_MEM_TEST_)
# TAOS_MEM_CHECK
# 1 to test memory allocation failure
# 2 to check memory leak
ADD_DEFINITIONS(-DTAOS_MEM_CHECK=2)
IF (TD_CLUSTER) IF (TD_CLUSTER)
ADD_DEFINITIONS(-DCLUSTER) ADD_DEFINITIONS(-DCLUSTER)
......
...@@ -187,18 +187,43 @@ static FORCE_INLINE void taosEncryptPass(uint8_t *inBuf, unsigned int inLen, cha ...@@ -187,18 +187,43 @@ static FORCE_INLINE void taosEncryptPass(uint8_t *inBuf, unsigned int inLen, cha
char *taosIpStr(uint32_t ipInt); char *taosIpStr(uint32_t ipInt);
#ifdef _TAOS_MEM_TEST_ extern void taos_dump_memory_leak_at_exit(const char* path);
#if TAOS_MEM_CHECK == 1
// Use during test to simulate the success and failure scenarios of memory allocation // Use during test to simulate the success and failure scenarios of memory allocation
extern void* taos_malloc(unsigned int size, char* _func); extern void* taos_malloc(size_t size, const char* func);
extern void* taos_calloc(unsigned int num, unsigned int size, char* _func); extern void* taos_calloc(size_t num, size_t size, const char* func);
extern void* taos_realloc(void* ptr, unsigned int size, char* _func); extern void* taos_realloc(void* ptr, size_t size, const char* func);
extern void taos_free(void* ptr); extern void taos_free(void* ptr);
extern char* taos_strdup(const char* str, const char* func);
extern char* taos_strndup(const char* str, size_t size, const char* func);
#define malloc(size) taos_malloc(size, __FUNCTION__) #define malloc(size) taos_malloc(size, __FUNCTION__)
#define calloc(num, size) taos_calloc(num, size, __FUNCTION__) #define calloc(num, size) taos_calloc(num, size, __FUNCTION__)
#define realloc(ptr, size) taos_realloc(ptr, size, __FUNCTION__) #define realloc(ptr, size) taos_realloc(ptr, size, __FUNCTION__)
#define free(ptr) taos_free(ptr) #define free(ptr) taos_free(ptr)
#endif #define strdup(str) taos_strdup(str, __FUNCTION__)
#define strndup(str, size) taos_strndup(str, size, __FUNCTION__)
#elif TAOS_MEM_CHECK == 2
extern void* taos_malloc(size_t size, const char* file, uint32_t line);
extern void* taos_calloc(size_t num, size_t size, const char* file, uint32_t line);
extern void* taos_realloc(void* ptr, size_t size, const char* file, uint32_t line);
extern void taos_free(void* ptr, const char* file, uint32_t line);
extern char* taos_strdup(const char* str, const char* file, uint32_t line);
extern char* taos_strndup(const char* str, size_t size, const char* file, uint32_t line);
extern ssize_t taos_getline(char **lineptr, size_t *n, FILE *stream, const char* file, uint32_t line);
#define malloc(size) taos_malloc(size, __FILE__, __LINE__)
#define calloc(num, size) taos_calloc(num, size, __FILE__, __LINE__)
#define realloc(ptr, size) taos_realloc(ptr, size, __FILE__, __LINE__)
#define free(ptr) taos_free(ptr, __FILE__, __LINE__)
#define strdup(str) taos_strdup(str, __FILE__, __LINE__)
#define strndup(str, size) taos_strndup(str, size, __FILE__, __LINE__)
#define getline(lineptr, n, stream) taos_getline(lineptr, n, stream, __FILE__, __LINE__)
#endif
#ifdef __cplusplus #ifdef __cplusplus
} }
......
...@@ -81,6 +81,7 @@ bool taosGetProcMemory(float *memoryUsedMB) { ...@@ -81,6 +81,7 @@ bool taosGetProcMemory(float *memoryUsedMB) {
char * line = NULL; char * line = NULL;
while (!feof(fp)) { while (!feof(fp)) {
tfree(line); tfree(line);
len = 0;
getline(&line, &len, fp); getline(&line, &len, fp);
if (line == NULL) { if (line == NULL) {
break; break;
...@@ -137,7 +138,7 @@ bool taosGetProcCpuInfo(ProcCpuInfo *cpuInfo) { ...@@ -137,7 +138,7 @@ bool taosGetProcCpuInfo(ProcCpuInfo *cpuInfo) {
return false; return false;
} }
size_t len; size_t len = 0;
char * line = NULL; char * line = NULL;
getline(&line, &len, fp); getline(&line, &len, fp);
if (line == NULL) { if (line == NULL) {
...@@ -409,6 +410,7 @@ bool taosGetCardInfo(int64_t *bytes) { ...@@ -409,6 +410,7 @@ bool taosGetCardInfo(int64_t *bytes) {
while (!feof(fp)) { while (!feof(fp)) {
tfree(line); tfree(line);
len = 0;
getline(&line, &len, fp); getline(&line, &len, fp);
if (line == NULL) { if (line == NULL) {
break; break;
...@@ -480,6 +482,7 @@ bool taosReadProcIO(int64_t *readbyte, int64_t *writebyte) { ...@@ -480,6 +482,7 @@ bool taosReadProcIO(int64_t *readbyte, int64_t *writebyte) {
while (!feof(fp)) { while (!feof(fp)) {
tfree(line); tfree(line);
len = 0;
getline(&line, &len, fp); getline(&line, &len, fp);
if (line == NULL) { if (line == NULL) {
break; break;
......
...@@ -61,6 +61,14 @@ int main(int argc, char *argv[]) { ...@@ -61,6 +61,14 @@ int main(int argc, char *argv[]) {
return 0; return 0;
} else if (strcmp(argv[i], "-k") == 0) { } else if (strcmp(argv[i], "-k") == 0) {
dnodeParseParameterK(); dnodeParseParameterK();
#if TAOS_MEM_CHECK == 2
} else if (strcmp(argv[i], "--check-mem-leak") == 0) {
if ((i < argc - 1) && (argv[i+1][0] != '-')) {
taos_dump_memory_leak_at_exit(argv[++i]);
} else {
taos_dump_memory_leak_at_exit(NULL);
}
#endif
} }
} }
......
...@@ -15,12 +15,15 @@ ...@@ -15,12 +15,15 @@
#include "os.h" #include "os.h"
#include "tlog.h" #include "tlog.h"
#include "os.h"
#if TAOS_MEM_CHECK == 1
extern int32_t taosGetTimestampSec(); extern int32_t taosGetTimestampSec();
static int32_t startTime = 0; static int32_t startTime = 0;
static int64_t m_curLimit = 100*1024; static int64_t m_curLimit = 100 * 1024;
bool isMallocMem(unsigned int size, char* _func) { static bool isMallocMem(size_t size, const char* func) {
if (0 == startTime) { if (0 == startTime) {
startTime = taosGetTimestampSec(); startTime = taosGetTimestampSec();
return true; return true;
...@@ -29,9 +32,9 @@ bool isMallocMem(unsigned int size, char* _func) { ...@@ -29,9 +32,9 @@ bool isMallocMem(unsigned int size, char* _func) {
if (currentTime - startTime < 10) return true; if (currentTime - startTime < 10) return true;
} }
if (size > m_curLimit) { if (size > m_curLimit) {
if (3 == rand() % 20) { if (3 == rand() % 20) {
pTrace("====no alloc mem in func: %s, size:%d", _func, size); pTrace("====no alloc mem in func: %s, size:%d", func, size);
return false; return false;
} }
} }
...@@ -39,40 +42,294 @@ bool isMallocMem(unsigned int size, char* _func) { ...@@ -39,40 +42,294 @@ bool isMallocMem(unsigned int size, char* _func) {
return true; return true;
} }
void* taos_malloc(unsigned int size, char* _func) { void* taos_malloc(size_t size, const char* func) {
if (false == isMallocMem(size, func)) {
return NULL;
}
return malloc(size);
}
if (false == isMallocMem(size, _func)) { void* taos_calloc(size_t num, size_t size, const char* func) {
if (false == isMallocMem(size, func)) {
return NULL; return NULL;
} }
return calloc(num, size);
void *p = NULL; }
p = malloc(size);
return p; void* taos_realloc(void* ptr, size_t size, const char* func) {
if (false == isMallocMem(size, func)) {
return NULL;
}
return realloc(ptr, size);
}
void taos_free(void* ptr) { free(ptr); }
char* taos_strdup(const char* str, const char* func) {
size_t len = strlen(str);
return isMallocMem(len + 1, func) ? strdup(str) : NULL;
}
char* taos_strndup(const char* str, size_t size, const char* func) {
size_t len = strlen(str);
if (len > size) {
len = size;
}
return isMallocMem(len + 1, func) ? strndup(str, len) : NULL;
}
#elif TAOS_MEM_CHECK == 2
#define MAGIC 0x55AA
typedef struct SMemBlock {
const char* file;
uint16_t line;
uint16_t magic;
uint32_t size;
struct SMemBlock* prev;
struct SMemBlock* next;
// TODO: need pading in 32bit platform
char data[0];
} SMemBlock;
static SMemBlock *blocks = NULL;
static uintptr_t lock = 0;
static FILE* fpMemLeak = NULL;
static void add_mem_block(SMemBlock* blk) {
blk->prev = NULL;
while (atomic_val_compare_exchange_ptr(&lock, 0, 1) != 0);
blk->next = blocks;
if (blocks != NULL) {
blocks->prev = blk;
}
blocks = blk;
atomic_store_ptr(&lock, 0);
}
static void remove_mem_block(SMemBlock* blk) {
while (atomic_val_compare_exchange_ptr(&lock, 0, 1) != 0);
if (blocks == blk) {
blocks = blk->next;
}
if (blk->prev != NULL) {
blk->prev->next = blk->next;
}
if (blk->next != NULL) {
blk->next->prev = blk->prev;
}
atomic_store_ptr(&lock, 0);
blk->prev = NULL;
blk->next = NULL;
} }
void* taos_calloc(unsigned int num, unsigned int size, char* _func) { void* taos_malloc(size_t size, const char* file, uint32_t line) {
if (size == 0) {
if (false == isMallocMem(size, _func)) { return NULL;
}
SMemBlock *blk = (SMemBlock*)malloc(size + sizeof(SMemBlock));
if (blk == NULL) {
return NULL; return NULL;
} }
void *p = NULL; if (line > UINT16_MAX && fpMemLeak != NULL) {
p = calloc(num, size); fprintf(fpMemLeak, "%s:%d: line number too large.\n", file, line);
}
if (size > UINT32_MAX && fpMemLeak != NULL) {
fprintf(fpMemLeak, "%s:%d: size too large: %lld.\n", file, line, size);
}
blk->file = file;
blk->line = (uint16_t)line;
blk->magic = MAGIC;
blk->size = size;
add_mem_block(blk);
return blk->data;
}
void* taos_calloc(size_t num, size_t size, const char* file, uint32_t line) {
size *= num;
void* p = taos_malloc(size, file, line);
if (p != NULL) {
memset(p, 0, size);
}
return p; return p;
} }
void* taos_realloc(void* ptr, unsigned int size, char* _func) { void* taos_realloc(void* ptr, size_t size, const char* file, uint32_t line) {
if (size == 0) {
if (false == isMallocMem(size, _func)) { taos_free(ptr, file, line);
return NULL;
}
if (ptr == NULL) {
return taos_malloc(size, file, line);
}
SMemBlock* blk = ((char*)ptr) - sizeof(SMemBlock);
if (blk->magic != MAGIC) {
if (fpMemLeak != NULL) {
fprintf(fpMemLeak, "%s:%d: memory not allocated by 'taos_malloc'.\n", file, line);
}
return realloc(ptr, size);
}
remove_mem_block(blk);
void* p = realloc(blk, size + sizeof(SMemBlock));
if (p == NULL) {
add_mem_block(blk);
return NULL; return NULL;
} }
void *p = NULL; if (size > UINT32_MAX && fpMemLeak != NULL) {
p = realloc(ptr, size); fprintf(fpMemLeak, "%s:%d: size too large: %lld.\n", file, line, size);
}
blk = (SMemBlock*)p;
blk->size = size;
add_mem_block(blk);
return blk->data;
}
void taos_free(void* ptr, const char* file, uint32_t line) {
if (ptr == NULL) {
return;
}
SMemBlock* blk = ((char*)ptr) - sizeof(SMemBlock);
if (blk->magic != MAGIC) {
if (fpMemLeak != NULL) {
fprintf(fpMemLeak, "%s:%d: memory not allocated by 'taos_malloc'.\n", file, line);
}
free(ptr);
return;
}
remove_mem_block(blk);
free(blk);
}
char* taos_strdup(const char* str, const char* file, uint32_t line) {
size_t len = strlen(str);
char *p = taos_malloc(len + 1, file, line);
if (p != NULL) {
memcpy(p, str, len);
p[len] = 0;
}
return p;
}
char* taos_strndup(const char* str, size_t size, const char* file, uint32_t line) {
size_t len = strlen(str);
if (len > size) {
len = size;
}
char *p = taos_malloc(len + 1, file, line);
if (p != NULL) {
memcpy(p, str, len);
p[len] = 0;
}
return p; return p;
} }
void taos_free(void* ptr) { ssize_t taos_getline(char **lineptr, size_t *n, FILE *stream, const char* file, uint32_t line) {
free(ptr); char* buf = NULL;
size_t bufSize = 0;
ssize_t size = getline(&buf, &bufSize, stream);
if (size != -1) {
if (*n < size + 1) {
void* p = taos_realloc(*lineptr, size + 1, file, line);
if (p == NULL) {
free(buf);
return -1;
}
*lineptr = (char*)p;
*n = size + 1;
}
memcpy(*lineptr, buf, size + 1);
}
free(buf);
return size;
}
static void dump_memory_leak() {
const char* hex = "0123456789ABCDEF";
const char* fmt = ":%d: addr=0x%p, size=%d, content(first 16 bytes)='";
size_t numOfBlk = 0, totalSize = 0;
fputs("memory blocks allocated but not freed before exit:\n\n", fpMemLeak);
while (atomic_val_compare_exchange_ptr(&lock, 0, 1) != 0);
for (SMemBlock* blk = blocks; blk != NULL; blk = blk->next) {
++numOfBlk;
totalSize += blk->size;
fputs(blk->file, fpMemLeak);
fprintf(fpMemLeak, fmt, blk->line, blk->data, blk->size);
uint8_t c = (uint8_t)(blk->data[0]);
fputc(hex[c >> 4], fpMemLeak);
fputc(hex[c & 0x0f], fpMemLeak);
size_t size = blk->size > 16 ? 16 : blk->size;
for (size_t i = 1; i < size; ++i) {
c = (uint8_t)(blk->data[i]);
fputc(' ', fpMemLeak);
fputc(hex[c >> 4], fpMemLeak);
fputc(hex[c & 0x0f], fpMemLeak);
}
fputs("'\n", fpMemLeak);
}
atomic_store_ptr(&lock, 0);
fprintf("\nnumber of blocks: %lld, total bytes: %lld\n", numOfBlk, totalSize);
if (fpMemLeak != stdout) {
fclose(fpMemLeak);
fpMemLeak = NULL;
}
}
static void dump_memory_leak_at_sig(int sig) {
fprintf(fpMemLeak, "signal %d received, exiting...\n", sig);
dump_memory_leak();
struct sigaction act = {0};
act.sa_handler = SIG_DFL;
sigaction(sig, &act, NULL);
}
void taos_dump_memory_leak_at_exit(const char* path) {
if (path == NULL || path[0] == 0) {
fpMemLeak = stdout;
} else if ((fpMemLeak = fopen(path, "w")) == NULL) {
printf("failed to open memory leak dump file '%s', errno=%d\n", path, errno);
return;
}
atexit(dump_memory_leak);
struct sigaction act = {0};
act.sa_handler = dump_memory_leak_at_sig;
sigaction(SIGFPE, &act, NULL);
sigaction(SIGSEGV, &act, NULL);
sigaction(SIGILL, &act, NULL);
} }
#endif
#if TAOS_MEM_CHECK != 2
void taos_dump_memory_leak_at_exit(const char* path) {
printf("memory leak detection not enabled!")
}
#endif
\ No newline at end of file
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册