美文网首页
静态区析构时引发的线程安全 heap-use-after-fre

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

作者: 铸造中 | 来源:发表于2023-11-22 19:09 被阅读0次

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

背景

给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

相关文章

  • 面试题2:C++ 单例模式最简单最安全实现

    本代码采用饿汉模式,是线程安全的,而且静态对象在生命周期结束的时候也会自动析构。

  • strtok()线程不安全

    strtok函数在提取字符串时使用了静态缓冲区,因此,它是线程不安全的,多线程同时访问该静态变量时,则会出现错误

  • python 13面向对象

    构造函数 析构函数 私有 类方法 静态方法 属性方法

  • C++总结

    函数 为什么基类的析构函数用虚函数 在实现多态时,当用基类操作派用类,在析构时防止只析构基类而不析构派生类的状况发...

  • C++ 静态变量中使用了条件变量导致主线程无法退出的问题

    记录下一个因为静态变量中使用了条件变量导致主线程无法退出的问题。 涉及到的问题点: 静态变量的析构顺序,是在mai...

  • Swift中struct和class的区别

    struct 值类型,深拷贝,分配在栈上没有析构函数不能继承不会发生内存泄漏,线程安全 class 引用类型,浅拷...

  • GeekBand C++面向对象高级编程(下)(第四周):没有虚

    在C++中,如果一个类是作为父类存在时,那么析构函数必须是虚析构函数,否则在对其子类进行析构时,只会释放其父类的成...

  • swift3语法(十二)

    析构过程 析构器析构器只适用于类类型,当一个类的实例被释放之前,析构器会被立即调用。析构器用关键字deinit来标...

  • Java析构方法

    析构方法与构造方法相反,当对象脱离其作用域时(例如对象所在的方法已调用完毕),系统自动执行析构方法。析构方法往往用...

  • [C++之旅] 13 析构函数

    [C++之旅] 13 析构函数 析构函数与构造函数相反,构造函数在实例化一个对象时调用,而析构函数在销毁一个对象时...

网友评论

      本文标题:静态区析构时引发的线程安全 heap-use-after-fre

      本文链接:https://www.haomeiwen.com/subject/hllwwdtx.html