在运行时读取 libstdc++ 版本
Reading libstdc++ version at runtime
我的应用程序受到旧版本 libstdc++ 中的 a bug 的影响,数据丢失相当严重。如何使用 -rpath
或 LD_LIBRARY_PATH
select 正确的库版本的补救措施是已知的,但对于部署和构建中的更改并不可靠。在我自己被咬了不止一次之后,我想停止痛苦并引入运行时检查以获取足够新的 libstdc++ 版本。如果部署未能使用正确的版本,我如何访问该版本以打印一条大警告消息。请注意,我需要 minor 版本 libstdc++.so.6.0.25
a.k.a。 GLIBCXX_3.4.25
随 gcc 8 一起提供。
这是一个linux程序,它简单地列出了它拥有的DSOs的绝对真实路径
已加载(由 dl_iterate_phdr 枚举)的可访问文件。
(所有 linux 程序加载 linux-vdso.so
,这实际上不是文件)。
main.cpp
#include <link.h>
#include <climits>
#include <cstdlib>
#include <string>
#include <vector>
#include <iostream>
int
get_next_SO_path(dl_phdr_info *info, size_t, void *p_SO_list)
{
auto & SO_list =
*static_cast<std::vector<std::string> *>(p_SO_list);
auto p_SO_path = realpath(info->dlpi_name,NULL);
if (p_SO_path) {
SO_list.emplace_back(p_SO_path);
free(p_SO_path);
}
return 0;
}
std::vector<std::string>
get_SO_realpaths()
{
std::vector<std::string> SO_paths;
dl_iterate_phdr(get_next_SO_path, &SO_paths);
return SO_paths;
}
int main()
{
auto SO_paths = get_SO_realpaths();
for (auto const & SO_path : SO_paths) {
std::cout << SO_path << std::endl;
}
return 0;
}
对我来说运行起来像:
$ g++ -Wall -Wextra main.cpp && ./a.out
/usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.25
/lib/x86_64-linux-gnu/libgcc_s.so.1
/lib/x86_64-linux-gnu/libc-2.27.so
/lib/x86_64-linux-gnu/libm-2.27.so
/lib/x86_64-linux-gnu/ld-2.27.so
如你所见,完整版出现了。通过一些文件名解析,你
可以从那里拿走。根据 get_SO_realpaths
获取整个 DSO 列表,
在寻找任何 libstdc++
之前,如果你愿意,你可以检测到异常的可能性
正在加载多个 libstdc++
。
我的应用程序受到旧版本 libstdc++ 中的 a bug 的影响,数据丢失相当严重。如何使用 -rpath
或 LD_LIBRARY_PATH
select 正确的库版本的补救措施是已知的,但对于部署和构建中的更改并不可靠。在我自己被咬了不止一次之后,我想停止痛苦并引入运行时检查以获取足够新的 libstdc++ 版本。如果部署未能使用正确的版本,我如何访问该版本以打印一条大警告消息。请注意,我需要 minor 版本 libstdc++.so.6.0.25
a.k.a。 GLIBCXX_3.4.25
随 gcc 8 一起提供。
这是一个linux程序,它简单地列出了它拥有的DSOs的绝对真实路径
已加载(由 dl_iterate_phdr 枚举)的可访问文件。
(所有 linux 程序加载 linux-vdso.so
,这实际上不是文件)。
main.cpp
#include <link.h>
#include <climits>
#include <cstdlib>
#include <string>
#include <vector>
#include <iostream>
int
get_next_SO_path(dl_phdr_info *info, size_t, void *p_SO_list)
{
auto & SO_list =
*static_cast<std::vector<std::string> *>(p_SO_list);
auto p_SO_path = realpath(info->dlpi_name,NULL);
if (p_SO_path) {
SO_list.emplace_back(p_SO_path);
free(p_SO_path);
}
return 0;
}
std::vector<std::string>
get_SO_realpaths()
{
std::vector<std::string> SO_paths;
dl_iterate_phdr(get_next_SO_path, &SO_paths);
return SO_paths;
}
int main()
{
auto SO_paths = get_SO_realpaths();
for (auto const & SO_path : SO_paths) {
std::cout << SO_path << std::endl;
}
return 0;
}
对我来说运行起来像:
$ g++ -Wall -Wextra main.cpp && ./a.out
/usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.25
/lib/x86_64-linux-gnu/libgcc_s.so.1
/lib/x86_64-linux-gnu/libc-2.27.so
/lib/x86_64-linux-gnu/libm-2.27.so
/lib/x86_64-linux-gnu/ld-2.27.so
如你所见,完整版出现了。通过一些文件名解析,你
可以从那里拿走。根据 get_SO_realpaths
获取整个 DSO 列表,
在寻找任何 libstdc++
之前,如果你愿意,你可以检测到异常的可能性
正在加载多个 libstdc++
。