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

标签 c linux

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

我编写了 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,而是将 if 留在 ld 命令行中并让程序自行加载 libA

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

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

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

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

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

相关文章:

python - 把python代码翻译成c代码有那么难吗?

c - 在不同的机器上运行 UDP 客户端和服务器

c - 如何获取指定架构(x86、pic Controller )中的芯片名称和可寻址内存?

linux - shell 脚本中的返回值竞争条件

C# Raspberry pi Linux 串行端口权限即使作为 root 也被拒绝

linux - shell 找到一个文件,执行它 - 如果 'error' 则退出,如果 ' no error' 则继续

c++ - 我无法编译此 dijkstra 代码。 (算法设计手册)

c - Makefile:make运行命令

python - 在我的 apache 本地主机 Ubuntu 12.04 中安装 WSGI

linux - 获取进程的 pid - linux