• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

c/c++多线程绕过mutex的保护

原作者: [db:作者] 来自: [db:来源] 收藏 邀请

多线程 绕过mutex的保护

mutex,能够解决线程安全的问题,但它不是万能的。下面的例子虽然使用了mutex,但是恶意注入了一个外部函数,导致把被mutex保护的双向链表,让一个外部的指针指向了,结果就可以通过这个外部的指针操作被保护的双向链表,也就绕过了metex的保护。

例子:

#include <list>
#include <iostream>
#include <mutex>
#include <algorithm>
#include <thread>
#include <unistd.h>

using namespace std;

//是线程安全的
class data_protect{
public:
  list<int> alist{1,2};
  mutex m;
public:  
  void add_list(int val){
    //操作双向链表时,加锁了
    lock_guard<mutex> g(m);
    alist.push_back(val);
  }
  bool contains(int val){
    //操作双向链表时,加锁了
    lock_guard<mutex> g(m);
    return find(alist.begin(), alist.end(), val) != alist.end();
  }
  //参数接受一个外来的函数,并把双向链表传递给了外来的函数
  template<typename Function>
  void do_data(Function func){
    lock_guard<mutex> g(m);
    func(alist);
  }
};

list<int>* unprotect_list;
//外来的函数接收到了双向链表的引用,并把这个双向链表暴露给外面的非保护的双向链表的指针
void bad_func(list<int>& alist){
  unprotect_list = &alist;
}

void gl(data_protect& dp){
  dp.do_data(bad_func);
  //这个插入操作没有加锁,所以不是线程安全的
  unprotect_list->push_back(99);
}
int main(){
  data_protect dp;
  gl(dp);
  //alist被改变了
  for(auto& s : dp.alist){
    cout << s << endl;
  }
}

为了避免上面的情况,最好遵守下面的准则:

不要将受保护的数据的指针或者引用传递到锁的范围之外,无论是通过函数返回它们,还是作为参数传递给用户提供的函数。

本人微信:xiaoshitou5854


鲜花

握手

雷人

路过

鸡蛋
该文章已有0人参与评论

请发表评论

全部评论

专题导读
上一篇:
C#如何异步查询数据库发布时间:2022-07-18
下一篇:
C#好书盘点发布时间:2022-07-18
热门推荐
阅读排行榜

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap