Understanding Swift’s value type thread safety - 代码分析(二)
示范代码
`
func testScenarioA2() throws {
var store: Int = 100
DispatchQueue.concurrentPerform(iterations: 1_000_000) { _ in
store.negate()
_ = store
}
}
`
开启线程race诊断后,出现以下错误
Swift access race in closure #1 (Swift.Int) -> () in UnderstandStruct.testScenarioA2() throws -> ()
查看汇编
`
0x100002a30 <+0>: pushq %rbp
0x100002a31 <+1>: movq %rsp, %rbp
0x100002a34 <+4>: subq $0x20, %rsp
0x100002a38 <+8>: movq 0x8(%rbp), %rdi
0x100002a3c <+12>: movq %rsi, -0x10(%rbp)
0x100002a40 <+16>: callq 0x100003bac ; symbol stub for: __tsan_func_entry
0x100002a45 <+21>: xorl %esi, %esi
0x100002a47 <+23>: leaq -0x8(%rbp), %rax
0x100002a4b <+27>: movq %rax, %rdi
0x100002a4e <+30>: movl $0x8, %edx
0x100002a53 <+35>: callq 0x100003c06 ; symbol stub for: memset
0x100002a58 <+40>: xorl %ecx, %ecx
0x100002a5a <+42>: movl %ecx, %esi
0x100002a5c <+44>: movq -0x10(%rbp), %rdx
0x100002a60 <+48>: movq %rdx, -0x8(%rbp)
-> 0x100002a64 <+52>: movq %rdx, %rdi
0x100002a67 <+55>: movl $0x1, %edx
0x100002a6c <+60>: movq %rax, -0x18(%rbp)
0x100002a70 <+64>: callq 0x100003ba6 ; symbol stub for: __tsan_external_write
0x100002a75 <+69>: movq -0x10(%rbp), %rax
0x100002a79 <+73>: movq %rax, %rdi
0x100002a7c <+76>: callq 0x100003bbe ; symbol stub for: __tsan_read8
0x100002a81 <+81>: xorl %ecx, %ecx
0x100002a83 <+83>: movl %ecx, %eax
0x100002a85 <+85>: movq -0x10(%rbp), %rdx
0x100002a89 <+89>: subq (%rdx), %rax
0x100002a8c <+92>: seto %r8b
0x100002a90 <+96>: testb $0x1, %r8b
0x100002a94 <+100>: movq %rax, -0x20(%rbp)
0x100002a98 <+104>: jne 0x100002ac1 ; <+145> [inlined] Swift runtime failure: arithmetic overflow at main.swift:13
0x100002a9a <+106>: movq -0x10(%rbp), %rdi
0x100002a9e <+110>: callq 0x100003bc4 ; symbol stub for: __tsan_write8
0x100002aa3 <+115>: movq -0x10(%rbp), %rax
0x100002aa7 <+119>: movq -0x20(%rbp), %rcx
0x100002aab <+123>: movq %rcx, (%rax)
0x100002aae <+126>: movq %rax, %rdi
0x100002ab1 <+129>: callq 0x100003bbe ; symbol stub for: __tsan_read8
0x100002ab6 <+134>: callq 0x100003bb2 ; symbol stub for: __tsan_func_exit
0x100002abb <+139>: addq $0x20, %rsp
0x100002abf <+143>: popq %rbp
0x100002ac0 <+144>: retq
0x100002ac1 <+145>: ud2
0x100002ac3 <+147>: nopw %cs:(%rax,%rax)
0x100002acd <+157>: nopl (%rax)
`
0x100002aab <+123>: movq %rcx, (%rax)
rax是store的地址,直接将计算结果赋给指向的地址。
结论
- 对于堆中的变量 store,系统会自动生成 begin_access 和 end_access插桩的检测竞争代码,该代码不影响实际逻辑的执行
- 对于store的值的改变,不同线程中都是一条mov指令,所以这里存在的线程竞争不会导致crash
参考
`
bb0(%0 : $Int, %1 : $Int):
debug_value %0 : $Int // id: %2
debug_value_addr %1 : $Int, var, name "store", argno 2 // id: %3
%4 = begin_access [modify] [unknown] %1 : $Int // users: %7, %6
// function_ref SignedNumeric.negate()
%5 = function_ref @$ss13SignedNumericPsE6negateyyF : $@convention(method) <τ_0_0 where τ_0_0 : SignedNumeric> (@inout τ_0_0) -> () // user: %6
%6 = apply %5(%4) : $@convention(method) <τ_0_0 where τ_0_0 : SignedNumeric> (@inout τ_0_0) -> ()
end_access %4 : $Int // id: %7
%8 = begin_access [read] [unknown] %1 : $Int // users: %10, %9
%9 = load [trivial] %8 : $Int
end_access %8 : $*Int // id: %10
%11 = tuple () // user: %12
return %11 : $() // id: %12
}
`
Understanding Swift’s value type thread safety - 代码分析(二)的更多相关文章
- Linux内核启动代码分析二之开发板相关驱动程序加载分析
Linux内核启动代码分析二之开发板相关驱动程序加载分析 1 从linux开始启动的函数start_kernel开始分析,该函数位于linux-2.6.22/init/main.c start_ke ...
- Android4.0图库Gallery2代码分析(二) 数据管理和数据加载
Android4.0图库Gallery2代码分析(二) 数据管理和数据加载 2012-09-07 11:19 8152人阅读 评论(12) 收藏 举报 代码分析android相册优化工作 Androi ...
- SQL注入原理及代码分析(二)
前言 上一篇文章中,对union注入.报错注入.布尔盲注等进行了分析,接下来这篇文章,会对堆叠注入.宽字节注入.cookie注入等进行分析.第一篇文章地址:SQL注入原理及代码分析(一) 如果想要了解 ...
- android4.0 的图库Gallery2代码分析(二)
最近迫于生存压力,不得不给人兼职打工.故在博文中加了个求点击的链接.麻烦有时间的博友们帮我点击一下.没时间的不用勉强啊.不过请放心,我是做技术的,肯定链接没病毒,就是我打工的淘宝店铺.嘻嘻.http: ...
- 信息管理代码分析<二>读取二进制文件数据
first和end做为全局变量,分别指向链表的头和尾.建立链表的方式也比较简易,从二进制文件数据块中,依次从头到尾读取,每读取一个就建立一个结点. /*基本模型*/ EMP *emp1; while( ...
- FLINK流计算拓扑任务代码分析<二>
首先 是 StreamExecutionEnvironment see = StreamExecutionEnvironment.getExecutionEnvironment(); 我们在编写 fl ...
- XSS原理及代码分析
前言 XSS又叫跨站脚本攻击,是一种对网站应用程序的安全漏洞攻击技术.它允许恶意用户将代码注入网页,其他用户在浏览网页时就会受到影响.XSS分为三种:反射型,存储型,和DOM型.下面我会构造有缺陷的代 ...
- Effective Java 70 Document thread safety
Principle The presence of the synchronized modifier in a method declaration is an implementation det ...
- 折返(Reentrancy)VS线程安全(Thread safety)
在Wiki上,折返例如,下面的定义(接) In computing, a computer program or subroutine is called reentrant if it can be ...
- clang的线程安全分析模块 thread safety analysis
介绍 Clang的线程安全分析模块是C++语言的一个扩展,能对代码中潜在的竞争条件进行警告.这种分析是完全静态的(即编译时进行),没有运行时的消耗.当前这个功能还在开发中,但它已经具备了足够的成熟度, ...
随机推荐
- ArcMap分别求取矢量要素各区域的面积
本文介绍基于ArcMap软件,自动批量计算矢量图层中各个要素的面积的方法. 一次,遇到一个问题,需要分别计算ArcMap软件中一个图层的所有面要素的面积.如图,这个图层中包括多个省级行政区矢量 ...
- 重新点亮shell————变量[三]
前言 简单介绍一下shell的变量. 正文 变量的定义 变量名的命名规则 字母.数字.下划线 不以数字开头 变量的赋值 在赋值的时候不能出现空格 a =123,在等号前面有一个空格,那么会报错. 这是 ...
- Kafka 线上性能调优
Kafka 线上性能调优是一项综合工程,不仅仅是 Kafka 本身,还应该从硬件(存储.网络.CPU)以及操作系统方面来整体考量,首先我们要有一套生产部署方案,基于这套方案再进行调优,这样就有了可靠的 ...
- etcd 历史版本回溯的方法
在使用 etcd 作为配置存储或者其他的场景,如果因为误操作对其中 key 的值进行了修改,如果想要找回原来的值,可以利用 etcd 的版本机制进行回溯找回以前的值.在具体操作之前,我们首先获取一下 ...
- opensips使用drouting进行路由
操作系统 :CentOS 7.6_x64 opensips版本:2.4.9 drouting是Dynamic Routing(动态路由)的缩写,该模块可为特定呼叫选择(基于多个条件)最佳网关.今天整理 ...
- OpenYurt v1.1.0: 新增 DaemonSet 的 OTA 和 Auto 升级策略
简介: 在 OpenYurt v1.1.0 版本中,我们提供了 Auto 和 OTA 的升级策略.Auto 的升级策略重点解决由于节点 NotReady 而导致 DaemonSet升级阻塞的问题,OT ...
- EasyCV带你复现更好更快的自监督算法-FastConvMAE
简介: 近期FastConvMAE工作在EasyCV框架内首次对外开源,本文将重点介绍ConvMAE和FastConvMAE的主要工作,以及对应的代码实现,最后提供详细的教程示例如何进行FastCon ...
- 深信服智能边缘计算平台与 OpenYurt 落地方案探索与实践
简介:本文将介绍边缘计算落地的机遇与挑战,以及边缘容器开源项目 OpenYurt 在企业生产环境下的实践方案. 作者:赵震,深信服云计算开发工程师,OpenYurt 社区 Member 编者案:在 ...
- [BEX] Quasar BEX 提供的那些配置
Manifest.json https://developer.chrome.com/extensions/manifest Background Script & Content Scrip ...
- 实践探讨Python如何进行异常处理与日志记录
本文分享自华为云社区<Python异常处理与日志记录构建稳健可靠的应用>,作者:柠檬味拥抱. 异常处理和日志记录是编写可靠且易于维护的软件应用程序中至关重要的组成部分.Python提供了强 ...