Swift高级进阶-Swift编译过程,”SIL代码“,“IR语法”

swift编译过程

如果不懂LLVM,Clang的同学可以去了解下它的知识点 一些文章中有详细介绍 OC 的编译过程 ,本文来探索一下 Swift 的编译过程。Swift 的编译过程中使用 Swiftc ,与 Clang 一样,SwiftcLLVM编译架构的一个前端。

swiftc常用命令:

-dump-ast              解析和类型检查源文件 & 转换成 AST
-dump-parse            解析源文件 & 转换成 AST  
  -emit-assembly         生成汇编文件
  -emit-bc               生成 LLVM Bitcode 文件
  -emit-executable       生成已链接的可执行文件
  -emit-imported-modules 生成已导入的库
  -emit-ir               生成 LLVM IR 文件
  -emit-library          生成已连接的库
  -emit-object           生成目标文件
  -emit-silgen           生成 raw SIL 文件(第一个阶段)
  -emit-sil              生成 canonical SIL 文件(第2个阶段)
  -index-file            为源文件生成索引数据
  -print-ast             解析和类型检查源文件 & 转换成更简约的格式更好的 AST
  -typecheck             解析和类型检查源文件

swift 的编译流程:

image.png

与 Clang 相比, LLVM前端的流程中,在AST 和 IR之间,多了一层中间语言SIL ( Swift Intermediate Language ) 这么做的目的是希望弥补一些 Clang 编译器的缺陷,如无法执行一些高级分析,可靠的诊断和优化,而 AST 和LLVM IR 都不是合适的选择。因此,SIL应运而生,用来解决现有的缺陷。

SIL代码

SIL 官方文档传送门

由于 SILSwift 在编译过程中的中间产物, 通过 SIL 可以了解swift 底层的实现细节,帮助我们理解一些问题)

源码生成 SIL 的命令如下:

// 将m ain.swift 编译成 SIL 代码
swiftc -emit-sil main.swift 

// 将 main.swift 编译成 SIL,并保存到 main.sil 文件中
swiftc -emit-sil main.swift >> main.sil

// 将 main.swift 编译成 SIL的同时, 将命名重整后的符号恢复原样,并保存到 main.sil 文件中
swiftc -emit-sil main.swift | xcrun swift-demangle >> main.sil


SIL 语法和 IR 语法有点相似,常见语法含义:

  • load: 读取数据
  • sil_global:标记变量为全局变量
  • hidden: 标记只针对同一个Swift模块中的对象可见
  • alloc_global: 开辟全局变量的内存
  • global_addr: 获取全局变量的地址
  • ref_element_addr: 获取元素地址
  • init_existential_addr: 指令会生成 Existential Container 结构, 包裹着实例变量和协议对应的 PWT
  • destroy_addr
  • bb0 / bb1 ... : basic block 数字,表示一个代码块,SIL中没有分支语句,只有入口和出口
  • alloc_ref / dealloc_ref: 开辟/释放内存
  • function_ref: 获取直接派发函数地址.
  • class_method: 通过函数表获取方法.
  • witness_method: 通过 PWT 获取对应的函数地址
  • objc_method : 获取OC 方法地址
  • apply:调用函数
  • store A to B : 把A 的值存储到B中。
  • begin_access / end_access: 开始、结束访问
    • [modify] / [read] / [deinit] :修改型访问、读取型访问、删除型访问
    • [dynamic]:动态访问
    • [static]:静态访问
  • retain_value: 引用计数 + 1
  • release_value: 引用计数 - 1
  • metatype 获得元类型
    • @thick 描述元类型代表的形式,是引用 对象类型或是其子类,
    • @thin 代表一个确切的 类型不需要存储,
  • $ : 类型标识
  • %: 表示寄存器,类似局部常量,赋值后不可修改。如果再需要新的寄存器,就增加寄存器编号,这样操作有利于编译器的优化;后续进行降级操作 ,才会把这些带编号的虚拟寄存器 转换成对应体系结构的真实寄存器
  • @ : SIL中所有标识符均以@符号开头
    • @main 方法名字是 main
    • @_hasStorage 标识属性是存储属性
    • @_hasInitialValue 标识属性有初始值
    • @owned 代表函数接收者负责销毁返回值
    • @convention 这个标识用于明确指定当函数调用时参数返回值应该如何被处理
      • @convention(c) 表示使用C函数的方式进行调用
      • @convention(swift) 纯Swift函数的默认调用方式
      • @convention(method) 柯里化的函数调用方式
      • @convention(witness_method) 协议方法调用,它等同于convention(method),除了在处理范型类型参数时
      • @convention(objc_method) Objective-C方式调用

常见ARM64汇编指令

bl : 地址跳转

blr : 带返回的地址跳转, 跳转回指令后面跟随寄存器中保存的地址

mov: 把一个寄存器里的值,复制到另一个寄存器

🌰 mov x0, x8 把 x0 的值,复制到 x8 中

ldr: 把内存中的值,读取到寄存器中

🌰 ldr x0, [x0, x8] 把 x0 + x8 的地址里面的值,写到 x0 寄存器 中

str: 把寄存器里面的值写入到内存中

🌰 str x0, [x0, x8] 把寄存器 x0 的值,写入到 x0 + x8 的地址中

4. 常见的 IR 语法:

官方文档传送门☞官方文档

  • @ 全局标识
  • % 局部标识
  • alloca 开辟空间
  • align 内存对齐
  • i32 32位, 4字节
  • store 写入内存
  • load 读取数据
  • call 调用函数
  • ret 返回

bitcast

以新的数据类型去读取原类的值(也就是类型转换)


//以ty2 的数据类型 去读取 ty 类型的 value,读取步长是ty2
<result> = bitcast <ty> <value> to <ty2>
// eg: bitcast (%struct.str* @global to i8*)
// 以i8* 的数据类型 去读取 struct.str* 类型的 global,读取步长是i8*

getelementptr

  • 指令(GER):进行地址计算,来获取复合数据结构子元素的地址,并不访问内存
    • 至少要有2个参数
    • 第一个参数是类型 & 地址,也就是开始读取内存的变量首地址
    • 第二个及以后的参数,表示要进行计算的参数,比如结构体或者数据的第几个元素
// 获取复合数据类型的指定元素的地址
getelementptr <ty> <value>, <index>, <index>, ...

如果 p 是一个数组, 第二个参数表示获取数组里索引是1的值,也就是p[1]

getelementptr %struct.munger_struct* %P, i32 1

如果 p 是一个数组,里面放了几个结构体, 第二个参数依旧表示p[1],第3个参数表示获取数组索引为1的结构体的索引为0的元素,即p[1][0]

getelementptr %struct.munger_struct* %P, i32 1, i32 0

如果结构体内还有数据类型的嵌套,再获取其内部嵌套的值,可以继续追加参数, 所以说参数至少有2个,第一个是数据的首地址,第二个参数开始都表示索引值。

官网:


struct munger_struct {
  int f1;
  int f2;
};
void munge(struct munger_struct *P) {
  P[0].f1 = P[1].f1 + P[2].f2;
}
...
munger_struct Array[3];
...
munge(Array);

转成IR代码:

void %munge(%struct.munger_struct* %P) {
entry:
  %tmp = getelementptr %struct.munger_struct* %P, i32 1, i32 0
  %tmp = load i32* %tmp
  %tmp6 = getelementptr %struct.munger_struct* %P, i32 2, i32 1
  %tmp7 = load i32* %tmp6
  %tmp8 = add i32 %tmp7, %tmp
  %tmp9 = getelementptr %struct.munger_struct* %P, i32 0, i32 0
  store i32 %tmp8, i32* %tmp9
  ret void
}


getelementptr inbounds:

  • 与 getelementptr 类似, 不过它将类型与值分开了,分别作为2个参数。所以这个指令,至少有3个参数,前两个是类型和值,第三个开始是索引参数

<result> = getelementptr inbounds <ty>, <ty>* <ptrval>{, [inrange] <ty> <idx>}*

[图片上传失败...(image-3be4d6-1639387053332)]

extractvalue

// 获取复合数据类型的指定元素的值
<result> = extractvalue <aggregate type> <val>, <idx>{, <idx>}*

insertvalue

// 像复合类型中的指定索引位置插入值
<result> = insertvalue <aggregate type> <val>, <ty> <elt>, <idx>{, <idx>}* ; 
// insertvalue %struct.tm %ret_val.fca.0.insert, i32 %b1, 1

// 向类型为%struct.tm的%ret_val.fca.0.insert变量的索引为 1 的位置,插入%b1

结构体

%T = type {<type list>}
// eg: %swift.refcounted = type { %swift.type*, i64 }

数组


[<elementnumber> x <elementtype>]
// eg: alloca[24 x i8], align8 表示数组里面放了24个 i8 的整数

指针

<type> *
//eg: i64* 表示64 位的整型

青山不改,绿水长流,后会有期,感谢每一位佳人的支持!

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

推荐阅读更多精彩内容