【问题标题】:How to avoid unexpected memory allocation如何避免意外的内存分配
【发布时间】:2019-06-27 07:36:31
【问题描述】:

我只是用c ++写了一个小的MemoryHook库并用作预加载。但是当我每次运行测试时,分配的内存量很大,大小为72704。 用法是这样的: LD_PRELOAD=./libPreload.so ./XXX

我尝试在ldd列表中查找所有相关库,但没有大小为72704。我删除了其他信息,只保留了大小,但也有这个问题。

我的初始化代码:

    static void TraceInitialize()
    {
#ifdef _DEBUG
        fprintf( stderr, "call TraceInitialize\n" );      
#endif
        pthread_mutex_lock( &s_mutexInit );
        if ( s_status == TS_INITIALIZED ) { pthread_mutex_unlock( &s_mutexInit ); return; }

        s_status = TS_INITIALIZING;

        MemoryManager::initialize();

        s_pRealMalloc       = (FUNC_MALLOC)dlsym(RTLD_NEXT, "malloc");
        s_pRealCalloc       = (FUNC_CALLOC)dlsym(RTLD_NEXT, "calloc");
        s_pRealRealloc      = (FUNC_REALLOC)dlsym(RTLD_NEXT, "realloc");
        s_pRealMemalign     = (FUNC_MEMALIGN)dlsym(RTLD_NEXT, "memalign");
        s_pRealValloc       = (FUNC_VALLOC)dlsym(RTLD_NEXT, "valloc");
        s_pRealFree         = (FUNC_FREE)dlsym(RTLD_NEXT, "free");

        assert( !( NULL == s_pRealMalloc || NULL == s_pRealCalloc || NULL == s_pRealRealloc || 
                    NULL == s_pRealMemalign || NULL == s_pRealValloc || NULL == s_pRealFree ) );

        s_status = TS_INITIALIZED;

       // printMap();

        pthread_mutex_unlock( &s_mutexInit );
    }

并跟踪 malloc 调用:

 static int s_no_hook = 0;
 void* TraceMalloc( size_t size )
 {  
      if ( s_status == TS_INITIALIZING ) return mockMemory::_mockMalloc( size );

      if ( s_status != TS_INITIALIZED )  TraceInitialize();

      void* p = _impMalloc( size, __sync_fetch_and_add( &s_no_hook, 1 ) );
      __sync_fetch_and_sub( &s_no_hook, 1 );
      return p;
  }

当我测试一个演示问题时,只有空的 main 函数。我希望不是内存未释放,但真正的输出是:

++++++++++++++ unfreed addr: 0x55cc4ae22260, size: 72704, serial: 1 ++++++++++++++
backtrace:
./libPreLoad.so(_ZN11MemoryTrace13MemoryManager14storeBacktraceEPNS0_11tagUnitNodeE+0x28)[0x7f62fd6cceba]
./libPreLoad.so(_ZN11MemoryTrace13MemoryManager10appendUnitEPvmb+0x9f)[0x7f62fd6ccbb7]
./libPreLoad.so(_ZN11MemoryTrace10_impMallocEmb+0x52)[0x7f62fd6cd4cb]
./libPreLoad.so(_ZN11MemoryTrace11TraceMallocEm+0x58)[0x7f62fd6cd286]
./libPreLoad.so(malloc+0x18)[0x7f62fd6cc812]
/usr/lib/x86_64-linux-gnu/libstdc++.so.6(+0x8f416)[0x7f62fd1cd416]
/lib64/ld-linux-x86-64.so.2(+0x10733)[0x7f62fd9e0733]
/lib64/ld-linux-x86-64.so.2(+0x10ca)[0x7f62fd9d10ca]
++++++++++++++ end ++++++++++++++

没有回溯,输出为:

++++++++++++++ unfreed addr: 0x55f799c8e260, size: 72704, serial: 0 ++++++++++++++
backtrace:
++++++++++++++ end ++++++++++++++

还有一个额外的问题是无法统计静态全局变量的内存释放。我找不到合适的时间来做这件事,即使有 attribute ((destructor(101))) .

【问题讨论】:

  • 找到一个工具来将地址(如libstdc++.so.6(+0x8f416))解码为符号名称。也许addr2line -f
  • 感谢您的回答。我现在才解决这个问题。这个内存是libstdc++在程序开始时自动申请的。

标签: c++ memory


【解决方案1】:

您看到的大分配是C++ STL。正如herererumuanswered Nikos C. 所问的那样

堆使用来自 C++ 标准库。它分配 内存供内部库在启动时使用。如果你不链接反对 它,C 和 C++ 版本之间应该有零差异。 使用 GCC 和 Clang,您可以使用以下命令编译文件:

 g++ -Wl,--根据需要 main.cpp 

这将指示链接器不要链接到未使用的库。在 您的示例代码,未使用 C++ 库,因此不应链接 反对 C++ 标准库。

【讨论】:

  • 有效!感谢您的帮助!!!我按照链接修改我的测试代码,然后它不再依赖 libstdc++ 并且测试好了。我删除了包含列表中的一行:“#include ”。我不明白为什么只包含头文件会强制我链接 libstdc++。甚至混淆了为什么 libstdc++ 在启动时必须分配内存。内存完全失控。这真是一次糟糕的体验。
  • iostream 是 c++ 标准 i/o 的标头。例如:cout 在 libstdc++ 中定义。
猜你喜欢
  • 2023-04-10
  • 2023-03-07
  • 2021-04-24
  • 1970-01-01
  • 1970-01-01
  • 2021-03-13
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多