由于名称篡改,输出文件中可见的内容是什么

What is visible in the output file as a result of name mangling?

本文关键字:是什么 文件 于名 输出      更新时间:2023-10-16

名称篡改后是否只有公共函数可见?私有和受保护的功能会发生什么?类成员变量、常量、枚举等呢?嵌套类?

我知道每个编译器都会有不同的错误,但我想它们会有相同的错误吗?是这样吗?

编译完一个类后,如何检查输出文件以查看哪些是可见的?

例如,如果我有下面的类,并且它是编译的,那么最终用户什么时候可以通过检查输出文件来确定(他们将无法访问源代码)。

class SomeSwearWord{
private:
int _anotherSwearWord;
int getNextSwearWord();
protected:
std::string _aCurseWord;
std::string refineMyCurseWord();
public:
int _aVeryBadWord;
int getPreviousBadWord();
}

编辑:为了举例的目的,让我们考虑LLVM或gcc。。。

编辑:无法访问源代码的客户是否能够仅从可执行文件或对象文件中确定上述诅咒词?

编辑:有人能给出一个全面的答案吗?实际的平台并不重要,我只是想知道什么是可见的。我知道不同的编译器会有不同的混乱,但我只想明白,在最坏的情况下,一个局外人能看到多少?

所有可能需要重新定位的东西都在输出文件中可见。所以,这是任何在结果二进制中有固定位置的东西。

  • 成员不可见
  • 非虚拟成员功能可见
  • 虚拟成员功能可见
  • 类名可能是可见的,因为它们的构造函数需要在某个地方调用,或者它们的名称是完整函数名的一部分
  • 静态成员可见
  • 静态成员函数可见
  • 枚举、结构、并集和类本身根本不存在
  • C字符串(即"MyString")不是作为符号出现的,而是作为只读数据出现的

因此,在您的示例中,没有成员可见,但如果调用了所有函数,则所有函数都可见。

虚拟函数名称是存在的,因为您的vtable是一个可以重新定位到所有虚拟函数的数组。

通过在链接时具有一个定义文件来抑制它们在输出文件中的存在,该文件包含您想要呈现的输出的确切列表。在Windows上,这是共享库的默认设置(即,所有内容都隐藏),但在Linux/Mac上,默认设置是使它们可见并可覆盖。

当谈到"名称篡改"时,我们实际上是在谈论如何将函数实例化(包括模板、参数等)转换为一个唯一的字符串。被提及的名字本身就一字不差地存在着。

例如,您的示例将产生以下符号(GCC):

_ZN13SomeSwearWord16getNextSwearWordEv
_ZN13SomeSwearWord17refineMySwearWordEv
_ZN13SomeSwearWord18getPreviousBadWordEv

请注意,返回类型不存在。您可以通过objdump -t <myBinary>从对象文件或可执行文件中获取