6

我们的程序中有一个自定义的内存管理器,我们所有的 malloc/free 调用都由内存管理器管理,但是在程序的初始,getpwuid() 将被调用,并且在某些客户的机器上激活了 nss_ldap 它会调用来自 libc 的 malloc 不是来自我们的内存管理器,这会导致我们的内存管理器出现错误,来自 gdb 的堆栈报告是:

Breakpoint 2, 0x0000003df8cc6eb0 in brk () from /lib64/libc.so.6
0  0x0000003df8cc6eb0 in brk () from /lib64/libc.so.6
1  0x0000003df8cc6f72 in sbrk () from /lib64/libc.so.6
2  0x0000003df8c73d29 in __default_morecore () from /lib64/libc.so.6
3  0x0000003df8c70090 in _int_malloc () from /lib64/libc.so.6
4  0x0000003df8c70c9d in malloc () from /lib64/libc.so.6
5  0x0000003df880fc65 in __tls_get_addr () from /lib64/ld-linux-x86-64.so.2
6  0x00002aaaae302a7c in _nss_ldap_inc_depth () from /lib64/libnss_ldap.so.2
7  0x00002aaaae2f91a4 in _nss_ldap_enter () from /lib64/libnss_ldap.so.2
8  0x00002aaaae2f942c in _nss_ldap_getbyname () from /lib64/libnss_ldap.so.2
9  0x00002aaaae2f9aa9 in _nss_ldap_getpwuid_r () from /lib64/libnss_ldap.so.2
10 0x0000003df8c947c5 in getpwuid_r@@GLIBC_2.2.5 () from /lib64/libc.so.6
11 0x0000003df8c9412f in getpwuid () from /lib64/libc.so.6
12 0x0000000001414be3 in lc_username ()

我跟踪了_nss_ldap_inc_depth()的代码,似乎__tls_get_addr()被调用了,因为使用了线程本地存储,我尝试将内存管理器更改为共享库,但__tls_get_addr()仍然从libc调用malloc ,我怎样才能让它调用我们的内存管理器而不是 libc 的?

4

2 回答 2

2

您可以使用LD_PRELOAD在任何其他库(包括 glibc)之前加载您的库,它将被链接,例如:

$ LD_PRELOAD=/path/to/library/libmymalloc.so /bin/myprog

这里有一个教程展示了它是如何工作的,它甚至有一个示例插入malloc

于 2012-11-21T07:32:48.793 回答
0

您可以更改内存管理器以使用mmapinstread of brk.

一个进程中只能有一个用户brk。因此,如果您还没有替换所有调用malloc和相关函数(calloc以及strdup更多),则不能使用brk.

mmap但是,没有这样的问题。您的内存管理器可以使用mmap,并且malloc仍然可以并行工作。

于 2012-11-21T08:48:36.950 回答