gpt4 book ai didi

c - 是否可以根据/proc/self/maps中的信息进行munmap?

转载 作者:太空狗 更新时间:2023-10-29 12:12:44 24 4
gpt4 key购买 nike

我需要能够取消映射通过我正在链接的某些库打开的文件。需要这样做的原因是这些库所做的映射包含对可能需要在程序执行时重新加载的模块的引用(可能是长时间运行的执行)。问题是当我的进程持有引用时无法卸载模块。

我编写了 C 代码来解析 proc/self/maps 中的信息,以便读取映射的地址范围并计算其长度。我通过从结束地址中减去起始地址来计算长度,然后将起始地址和计算出的长度作为各自的参数传递给 munmap。问题是 munmap 因 EINVAL(无效参数)而失败。

我用 sysconf(_SC_PAGESIZE) 检查了我的机器使用的页面大小,它返回了 4096,这是我计算的长度值。 GNU 手册说 munmap 在以下情况下可能会因 EINVAL 而失败:

The memory range given was outside the user mmap range or wasn’t page aligned.

我是否遗漏了什么或者这根本不可能?我最后的办法是仔细梳理所做的系统调用并通过 strace 检查每个 mmap,但我希望这是最后的手段,谢谢。

最佳答案

您选择的方法不会有效,但不一定是您认为的原因。但是,有一种方法,所以请继续阅读......

I need to be able to unmap files that were opened through some libraries I'm linking with.

一旦您让 ELF 加载程序(例如 ld.linux.so)为您加载库,您就失去了控制。

不能只取消映射区域[不管方法]。加载程序已经为这些库完成了重定位和符号链接(symbolic link)。取消映射删除了该区域,但现在一切都中断了,因为链接器设置的各种指针现在指向空白空间。加载器将不知道你做了什么。

那么,您将如何重新映射新版本的库 [以及内存中的位置]?即使将它重新映射到相同的地址也不能保证,因为您无法调整加载程序已经完成的事情。

The reason for needing to do this is that the mappings made by these libraries hold references to modules that may need to be reloaded while the program is executing (potentially long running executions).

大多数需要更新到新版本库的程序只是重新执行它们自己。如果需要保留数据,可以制定转储/恢复机制。

但是,如果您真的想卸载/加载较新的库版本,可以使用动态 链接来完成。

不使用 ld 与(例如)libA 进行链接,而是离开 ld 命令行并让程序自己加载 libA

您使用 dlopen/dlsym/dlclose 打开/加载您控制下的库。

您必须跟踪符号表,但更改到新版本很容易。当您需要新版本时,只需执行 dlclose,然后执行 dlopen。您必须重做 dlsym 调用才能获取更新的地址,但这一切都相当简单且标准

The problem is the modules cannot be unloaded while my process is holding a reference.

原因是ELF loader做了这个。使用 dlopen 等。 al.,你没有同样的问题。

关于c - 是否可以根据/proc/self/maps中的信息进行munmap?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36732412/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com