When I first used the TCMalloc, I thought this solution will be good for my application because many threads allocate and release memory resources frequently.
However, there some problems were occurred.
Sometimes, memory usage of a process increased sharply within short time and there were hardly available system memory.
Finally, Linux OS operated OOM killer and kill my process.
Even though my application has some possibilities that a logic cause huge memory usage, this has never occurred before I apply the TCMalloc.
What's going wrong?
Thank to googling, I could find some similar cases that the TCMalloc's memory management is not complete.
http://stackoverflow.com/questions/15566083/tcmallocs-fragmentation
tcmalloc tries to do some smart things to anticipate your memory use, but it isn't very good about releasing memory back to the system even after you have freed it. in fact, it might be resident in memory and leading to OOM.
It's true.
You cannot trust memory management of the TCMalloc.
There are some reports about memory fragmentation issues.
https://groups.google.com/forum/#!searchin/google-perftools/ReleaseFreeMemory/google-perftools/FmeMfZ2CAJM/U15HZzZ15JIJ
For this issue, you can try some solutions.
One thing is releasing 'free memory' to system using this API.
MallocExtension::instance()->ReleaseFreeMemory();
Another thing is that you can adjust free memory rate.
see section 'Modifying Behavior In Code' in the page below
http://google-perftools.googlecode.com/svn/trunk/doc/tcmalloc.htmlHere let me show you a change of memory of one of my application as a good example.
This graph shows how much frequently memory is released when TCMALLOC_RELEASE_RATE is set to 10.
(Exact period is not guaranteed)
I will try another experiment with an open source program again in the future.
2. tc_malloc_stats
Remember that when you use the TCMalloc, you must not judge or analyze memory usage change of your process with report from OS.
To grab exact current tcmalloc stats, you have to use tc_malloc_stats() function.
(see gperftools/tcmalloc.h)
Or, you can use this too.
MallocExtension::instance()->GetStats(buffer, buffer_length);
std::cout << buffer;
refer to this:
http://gperftools.googlecode.com/svn/trunk/doc/tcmalloc.html
This is tc_malloc_stats() report.
------------------------------------------------ MALLOC: 16832 ( 0.0 MiB) Bytes in use by application MALLOC: + 38682624 ( 36.9 MiB) Bytes in page heap freelist MALLOC: + 97632 ( 0.1 MiB) Bytes in central cache freelist MALLOC: + 0 ( 0.0 MiB) Bytes in transfer cache freelist MALLOC: + 224 ( 0.0 MiB) Bytes in thread cache freelists MALLOC: + 1175704 ( 1.1 MiB) Bytes in malloc metadata MALLOC: ------------ MALLOC: = 39973016 ( 38.1 MiB) Actual memory used (physical + swap) MALLOC: + 0 ( 0.0 MiB) Bytes released to OS (aka unmapped) MALLOC: ------------ MALLOC: = 39973016 ( 38.1 MiB) Virtual address space used MALLOC: MALLOC: 10 Spans in use MALLOC: 1 Thread heaps in use MALLOC: 8192 Tcmalloc page size ------------------------------------------------ Call ReleaseFreeMemory() to release freelist memory to the OS (via madvise()). Bytes released to the OS take up virtual address space but no physical memory. |