静态区析构时引发的线程安全 heap-use-after-free

静态区析构时引发的线程安全

背景

给openssl 1.0.2 是非线程安全的,需要CRYPTO_set_locking_callback设置函数来控制加锁和解锁.
example.cpp

std::vector<std::mutex> g_openssl_locks{static_cast<size_t>(CRYPTO_num_locks())};

//可能是多个线程在调用这个函数
void openssl_locking_function(int mode, int n, const char * /* file */, int /* line */) {
    if(mode & CRYPTO_LOCK) {
        g_openssl_locks[n].lock();
    }
    else {
        g_openssl_locks[n].unlock();
    }
}

CRYPTO_set_locking_callback(openssl_locking_function);

如果此时程序强行退出可能出现线程安全错误.
比如用instrument ThreadSanitizer运行测试的话就会报错heap-use-after-free

原因

当程序退出的时候,会销毁全局/静态对象. 此时别的线程可能还没有终止,最后访问了一个已经被析构的对象从而引发未知的问题.
调用 exit() 函数时,程序的终止流程通常遵循以下步骤:

  1. 调用 exit() 函数:这可以发生在程序的任何地方,不限于主线程。
  2. 执行 exit() 的初始操作:开始终止程序,但不立即关闭所有线程。
  3. 销毁静态存储期对象:全局对象和静态局部对象会被销毁,调用它们的析构函数。
  4. 调用 atexit() 注册的函数:如果有通过 atexit() 注册的函数,这些函数会按照注册的逆序被调用。
  5. 其他线程的强制终止:程序中的其他线程将被强制性地终止。这些线程不会正常完成它们的执行路径。
  6. 清理和关闭:进行最终的清理操作,包括关闭所有打开的文件和释放其他系统资源。
  7. 程序终止:最后,控制权返回操作系统,程序完全结束。

在这个过程中,并没有为线程提供一个完整的、有序的终止机制。这是因为 exit() 的设计是为了迅速终止程序,而不是等待或协调线程的安全退出。因此,当使用多线程时,如果需要优雅地关闭线程,通常建议使用其他同步机制来确保线程能够安全地完成它们的工作,而不是依赖 exit() 来结束程序。

The primary problem with destruction of static-duration objects is access to static-duration objects after their destructors have executed, thus resulting in undefined behavior. To prevent this problem, we require that all user threads finish before destruction begins. For threads that do not naturally finish, mechanisms to terminate threads are proposed in N2447 Multi-threading Library for Standard C++ and its initial incorporation in N2521 Working Draft, Standard for Programming Language C++.

这也是为什么C++标准中也提及了,需要在释放全局/静态变量之前,要保证所有线程都结束了.

编译器自带线程安全优化

各编译器也正对这一点有了优化.
对于GCC >=4.3的版本已经默认属性fthreadsafe-static保证了静态变量线程安全. 此属性默认是开启的,大概就是能保证在线程都停止之后再析构. 有兴趣可以继续研究
参考链接:
https://gcc.gnu.org/onlinedocs/gcc/gcc-command-options/options-controlling-c%2B%2B-dialect.html#cmdoption-fthreadsafe-statics
各编译器所支持的功能如下:

image.png

解决方案

如果你不能保证代码会被什么编译器和版本运行. 要兼容的话可以考虑这2个方案.

  • 指定顺序

遵循C++标准. 如果你能控制子线程的话. 你只需要按照C++标准规定的顺序即可. 先结束子线程再析构.

  • 利用堆的特性. 通过将对象new出来放在堆区. 堆区的生命周期是等待delete操作来释放. 它不受exit()影响.
std::vector<std::mutex>* g_openssl_locks = new std::vector<std::mutex>(static_cast<size_t>(CRYPTO_num_locks()));

void openssl_locking_function(int mode, int n, const char * /* file */, int /* line */) { 
    if(mode & CRYPTO_LOCK) {
        (*g_openssl_locks)[n].lock();
    }
    else {
        (*g_openssl_locks)[n].unlock();
    }
}

int g_sslinit = SetSslLocking();

此方式,没人调用释放g_openssl_locks. 它会等待程序完全结束后被操作系统回收. 保证了顺序.

参考:

https://developer.aliyun.com/article/793257
https://zhuanlan.zhihu.com/p/656683028

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

推荐阅读更多精彩内容