示范代码

`

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 - 代码分析(二)的更多相关文章

  1. Linux内核启动代码分析二之开发板相关驱动程序加载分析

    Linux内核启动代码分析二之开发板相关驱动程序加载分析 1 从linux开始启动的函数start_kernel开始分析,该函数位于linux-2.6.22/init/main.c  start_ke ...

  2. Android4.0图库Gallery2代码分析(二) 数据管理和数据加载

    Android4.0图库Gallery2代码分析(二) 数据管理和数据加载 2012-09-07 11:19 8152人阅读 评论(12) 收藏 举报 代码分析android相册优化工作 Androi ...

  3. SQL注入原理及代码分析(二)

    前言 上一篇文章中,对union注入.报错注入.布尔盲注等进行了分析,接下来这篇文章,会对堆叠注入.宽字节注入.cookie注入等进行分析.第一篇文章地址:SQL注入原理及代码分析(一) 如果想要了解 ...

  4. android4.0 的图库Gallery2代码分析(二)

    最近迫于生存压力,不得不给人兼职打工.故在博文中加了个求点击的链接.麻烦有时间的博友们帮我点击一下.没时间的不用勉强啊.不过请放心,我是做技术的,肯定链接没病毒,就是我打工的淘宝店铺.嘻嘻.http: ...

  5. 信息管理代码分析<二>读取二进制文件数据

    first和end做为全局变量,分别指向链表的头和尾.建立链表的方式也比较简易,从二进制文件数据块中,依次从头到尾读取,每读取一个就建立一个结点. /*基本模型*/ EMP *emp1; while( ...

  6. FLINK流计算拓扑任务代码分析<二>

    首先 是 StreamExecutionEnvironment see = StreamExecutionEnvironment.getExecutionEnvironment(); 我们在编写 fl ...

  7. XSS原理及代码分析

    前言 XSS又叫跨站脚本攻击,是一种对网站应用程序的安全漏洞攻击技术.它允许恶意用户将代码注入网页,其他用户在浏览网页时就会受到影响.XSS分为三种:反射型,存储型,和DOM型.下面我会构造有缺陷的代 ...

  8. Effective Java 70 Document thread safety

    Principle The presence of the synchronized modifier in a method declaration is an implementation det ...

  9. 折返(Reentrancy)VS线程安全(Thread safety)

    在Wiki上,折返例如,下面的定义(接) In computing, a computer program or subroutine is called reentrant if it can be ...

  10. clang的线程安全分析模块 thread safety analysis

    介绍 Clang的线程安全分析模块是C++语言的一个扩展,能对代码中潜在的竞争条件进行警告.这种分析是完全静态的(即编译时进行),没有运行时的消耗.当前这个功能还在开发中,但它已经具备了足够的成熟度, ...

随机推荐

  1. lattice的ip不显示,如何解决

    最近ip服务器可能会遇到问题,建议客户把更新检查关掉.我们有对应的IP下载链接.   diamond在     https://www.latticesemi.com/ispupdate/ipexpr ...

  2. Webpack中Loader和Plugin的区别?编写Loader,Plugin的思路?

    一.区别 前面两节我们有提到Loader与Plugin对应的概念,先来回顾下 loader 是文件加载器,能够加载资源文件,并对这些文件进行一些处理,诸如编译.压缩等,最终一起打包到指定的文件中 pl ...

  3. IDEA操作MyBatis实现数据库增删改查

    "感谢您阅读本篇博客!如果您觉得本文对您有所帮助或启发,请不吝点赞和分享给更多的朋友.您的支持是我持续创作的动力,也欢迎留言交流,让我们一起探讨技术,共同成长!谢谢!" 前置环境 ...

  4. 技术干货 | 阿里云数据库PostgreSQL 13大版本揭秘

    简介: 阿里云RDS PostgreSQL是一款兼容开源PostgreSQL的全托管云数据库产品,自2015年首次发布以来,根据用户需求不断升级迭代,已支持9.4.10.11.12等多个版本,覆盖了高 ...

  5. IIncrementalGenerator 判断程序集之间可见关系

    本文告诉大家如何在使用 IIncrementalGenerator 进行增量的 Source Generator 生成代码时,如何判断两个程序集之间是否存在 InternalsVisibleTo 关系 ...

  6. 修复 GitLab 的 CI Runner 提示找不到 pwsh 执行文件

    本文告诉大家如何修复使用 GitLab 的 Runner 做 CI 时提示 "pwsh": executable file not found in %PATH% 错误 有两个方法 ...

  7. 从改一个老项目开始的PHP踩坑记

    php所有版本的地址: https://windows.php.net/downloads/releases/archives/ 访问控制器时省略了index.php报No input file sp ...

  8. 超好用的 Redis GUI 工具,你值得拥有

    超好用的 Redis GUI 工具,你值得拥有 提供原生的性能,并且比使用 Electron 等 Web 技术开发的同等应用程序消耗的资源少得多. 下载地址:http://www.redisant.c ...

  9. IIS 部署 docsify

    来源:https://www.cnblogs.com/yokeqi/p/14276176.html 来源:https://sspai.com/post/60534 docsify之前部署在Linux+ ...

  10. C语言结构体的内存分配

    一.结构体内存分配原则 原则一:结构体中元素按照定义顺序存放到内存中,但并不是紧密排列.从结构体存储的首地址开始 ,每一个元素存入内存中时,它都会认为内存是以自己的宽度来划分空间的,因此元素存放的位置 ...