关于Bsymbolic

位于同一个动态库中的外部函数调用

一个工程如下:
头文件common.h:

#ifndef COMMON_H_H
#define COMMON_H_H

void funa(void);
void funb(void);

#endif

源文件a.c:

#include "common.h"

void funa(void)
{

}

源文件b.c:

#include "common.h"

void funb(void)
{
    funa();
}

CMakeLists.txt中的内容如下:

cmake_minimum_required(VERSION 3.18)

project(sub C)

add_library(sub SHARED a.c b.c)

编译后,查看libsub.so中funb的反汇编:


image.png

image.png

funa明明就在funb附近, funb却要调用funa@plt,在运行时再去找funa。
实际上,就算funb和funa在同一个源文件中,funb也不会直接调用funa,而是去调用funa@plt。

Bsymbolic在动态库中的作用

修改CMakeLists.txt如下:

cmake_minimum_required(VERSION 3.18)

project(sub C)

add_link_options("-Wl,-Bsymbolic")
add_library(sub SHARED a.c b.c)

编译后,查看libsub.so中funb的反汇编:


image.png

可以看到,这个时候funb直接调用了funa,不会再等到运行时去找funa了。

Bsymbolic对静态库有没有用?

没有用。静态库只是一堆.o文件的打包。.c编译成.o之后是什么样子,打包到.a中还是什么样子,没有重定向过程。不管带不带Bsymbolic,编译为静态库后,funb的反汇编都是:


image.png

这里的00 00 00 00实际上是相对于callq下一条指令的偏移。因为funb不知道funa在什么位置,所以先全填0,等到链接进application或者so时再做重定向。

Bsymbolic的副作用

Bsymbolic可以使动态库优先使用内部的符号,防止当动态库内部的符号和外部符号同名时,被外部符号覆盖掉。
但同时,Bsymbolic也有一个副作用:动态库内部的符号,对外部来说,是独立的、不受影响的。虽然外部也能“看到”动态库内部的符号,但实际上那是一个副本。
以一个小程序为例:
头文件common.h中的内容如下:

#pragma once
#include <map>
#include <string>

namespace SUB
{

class A
{
public:
    static int i;
    static std::map<int, int> m;
    static void Print(void);
};

extern std::string s;
extern int arr[10];

}

源文件sub.cc中的内容如下:

#include "common.h"

#include <iostream>

namespace SUB
{

int A::i = 123456;
std::map<int, int> A::m{{1, 1}, {2, 2}, {3, 3}};

void A::Print(void)
{
    std::cout<< "A::i: " << &A::i << ", " << A::i << std::endl;
    std::cout<< "A::m: " << &A::m << ", size: " << A::m.size() << std::endl;
    std::cout<< "s: " << &s << ", " << s << std::endl;
    std::cout<< "arr: " << arr << ", " << arr[0] << " " << arr[9] << std::endl;
}

std::string s("hello");
int arr[10] = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};

}

源文件main.cc中的内容如下:

#include <iostream>

#include "common.h"

int main(void)
{
    std::cout<< "SUB::A::i: " << &SUB::A::i << ", " << SUB::A::i << std::endl;
    std::cout<< "SUB::A::m: " << &SUB::A::m << ", size: " << SUB::A::m.size() << std::endl;
    std::cout<< "SUB::s: " << &SUB::s << ", " << SUB::s << std::endl;
    std::cout<< "SUB::arr: " << &SUB::arr << ", " << SUB::arr[0] << " " << SUB::arr[9] << std::endl;

    SUB::A::Print();
}

编译命令:

g++ ../sub.cc -fPIC -shared -Wl,-Bsymbolic -o libsub.so
g++ ../main.cc ./libsub.so   -o main

运行结果:


image.png

虽然main中也能看到libsub.so中定义的全局变量、类静态成员变量,但它们的地址并不一样,不是同一个东西。
另外,在libsub.so之外实例化的复合数据类型(如STL中的map、string),并没有被正确初始化。
若动态库中确实有全局变量、类静态成员变量需要对外导出,可使用Bsymbolic-functions选项,缩小影响范围。例如:

g++ ../sub.cc -fPIC -shared -Wl,-Bsymbolic-functions -o libsub.so
g++ ../main.cc ./libsub.so   -o main
image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容