Swift的高级中间语言:SIL
简介
在LLVM的官方文档中对Swift的编译器设计描述如下: Swift编程语言是在LLVM上构建,并且使用LLVM IR和LLVM的后端去生成代码。但是Swift编译器还包含新的高级别的中间语言,称为SIL
。SIL
会对Swift进行较高级别的语义分析和优化。 我们下面分析一下SIL
设计的动机和SIL
的应用,包括高级别的语义分析,诊断转换,去虚拟化,特化,引用计数优化,TBAA(Type Based Alias Analysis)等。并且会在某些流程中加入对SIL
和LLVM IR对比。
SIL
介绍
SIL
是为了实现swift编程语言而设计的,包含高级语义信息的SSA格式的中间语言.SIL
包含以下功能:
-
一系列的高级别优化保障,用于对运行时和诊断行为提供可预测的底线
-
对swift语言数据流分析强制要求,对不满足强制要求的问题产生诊断。例如变量和结构体必须明确初始化,代码可达性即方法return的检测,switch的覆盖率
-
确保高级别优化。包含retain/release优化,动态方法的去虚拟化(devirtualization,不了解虚函数可以查看之前文章static vs dynamic dispatch),闭包内联,内存初始化提升和泛型方法实例 化.
-
可用于分配"脆弱"内联的稳定分配格式,将Swift库组件的泛型优化为二进制。
和LLVM IR不同,SIL
一般是target无关的独立格式的表示,可用于代码分发.但是也可以和LLVM一样表达具体target概念. 如果想查看更多SIL
的实现和SIL
通道的开发信息,可以查看SIL开发手册(原英文文档为SILProgrammersManual.md)。
我们下面对Clang的Swift编译器的传递流程进行对比:
编译流程对比
Clang编译器流程
imageClang编译流程存在以下问题:
-
在源码和LLVM IR直接存在非常大的抽象鸿沟
-
IR不适用对源码进行分析和检查 使用了Analysis通过CFG进行分析,分析和代码生成是两部分
-
CFG(控制流图)不够精确
-
CFG不是主道(hot path)
-
在CFG和IR降级中会出现重复分析,做无用功
Swift编译器流程
Swift作为一个高级别和安全的语言具有以下特点:
高级别语言
-
通过代码充分的展示语言的特性
-
支持基于协议的泛型
安全语言
-
充分的数据流检查:未初始化变量,函数返回处理检测,这些项在检测不合格时会产生对应的编译错误
-
边界和溢出的检测
Swift编译流程图如下:
imageSwift编译器提供的SIL
具有以下优势:
-
对程序语义信息重复表示
-
可以用于代码生成和分析 Clang不可以
-
处于编译器的主道
-
可以连接源码和LLVM的抽象鸿沟
SIL的设计
SIL流程分析
Swift编译器作为高级编译器,具有以下严格的传递流程结构。 Swift编译器的流程如下
-
Parse: 语法分析组件从Swift源码构成AST
-
语义分析组件对AST进行类型检查,并对其进行类型信息注释。
-
SILGen组件从AST形成"生的(raw)"SIL
-
一系列在 生 SIL上运行的,用于确定优化和诊断合格,对不合格的代码嵌入特定的语言诊断。这些操作一定会执行,即使在
-Onone
选项下也不例外。之后产生 正式(canonical) SIL. -
一般情况下,是否在正式
SIL
上运行SIL
优化是可选的,这个检测可以提升结果可执行文件的性能.可以通过优化级别来控制,在-Onone
模式下不会执行. -
IRGen会将
正式SIL
降级为LLVM IR. -
LLVM后端提供LLVM优化,执行LLVM代码生成器并产生二进制码.
SIL操作流程分析
SILGen
SILGen
遍历Swift进行了类型检查的AST,产生 raw SIL.SILGen
产生的SIL
格式具有如下属性:
-
属性会被加载和存储在可变内存地址,而不是使用严格的SSA(静态单赋值形式:每个变量仅被赋值一次)。这和Clang前端产生的繁重的LLVM IR(例如初始化
alloca
)类似。但是Swift的变量在大多数情况下使用了引用计数器,使得变量可以被retained,release和被闭包引用。 -
数据流检测。例如明确的内存分配,方法return检查,switch覆盖等.此环节目前不是强制执行的
-
transparent
函数优化目前未实现.
这些特性会被接下来的确保优化和诊断检查使用,这两项在 raw SIL
上一定会运行。
确保优化和诊断检查
在SILGen
之后,会在raw SIL
上运行确定顺序的优化。我们并不希望编译器产生的诊断改变编译器的进展,所以这些优化的设计是简单和可预测.
-
Mandatory inlining: 强制内联对于transparent函数进行内联。
透明函数即,如果一个函数只会受到入参的变化,那么这个函数每次的调用都会是相同的,同样的入参一定会返回一样的返回值,在确定入参的时候,返回值是可预测的。这样的函数,就可以进行内联优化。
-
内存提升实现分为两个优化阶段:
将
alloc_box
结构优化为alloc_stack
提升无暴露地址(
non_address-exposed
)的alloc_stack
说明到SSA注册. -
常数传播: Constant propagation折叠常量表达,繁殖常量值.如果在计算常量表达式时出现算术溢出,就会产生警告.
-
返回分析查证每个方法在每个代码路径只返回一个值,并且不会在定义的末端出现无返回值的错误.如果不需要返回值的函数return了也会报错.
-
临界拆分: critical edge splitting不支持任意的基础block参数通过终端进行临界拆分. 在 Advanced Compiler Design & Implementation的第13.3章节,第407,408页这样描述临界分裂
这个算法的核心作用体现为:流程图中的临界如果在流分析前被拆分的话,会使得运算更近高效. 原文: A key point in the algorithm is that it can be much more effective if the critical edges in the flowgraph have been split before the flow analysis is performed.
如果诊断通道完成后,会产生规范SIL.
-
泛型特化: Generic specialization
-
在
-Onone
模式下的ARC性能优化.
说完了处理raw SIL
的特定流程,我们对上面提到的优化通道: optimization passes
进行下说明.
泛型优化
SIL
获取语言特定的类型信息,使得无法在LLVM IR实现的高级优化在swift编译器中得以实现.
- 泛型特化分析泛型函数的特定调用,并生成新的特定版本的函数.然后将泛型的特定用法全部重写为对应的特定函数的指甲调用. 例如
func min<T: Comparable>(x: T, y: T) -> T {
return y < x ? y : x
}
从普通的泛型展开
func min<T: Comparable>(x: T, y: T, FTable: FunctionTable) -> T {
let xCopy = FTable.copy(x)
let yCopy = FTable.copy(y)
let m = FTable.lessThan(yCopy, xCopy) ? y : x
FTable.release(x)
FTable.release(y)
return m
}
在确定入参类型时,比如Int,可以优化为
func min<Int>(x: Int, y: Int) -> Int {
return y < x ? y : x
}
从而减少泛型调用的开销
-
witness和虚函数表的去虚拟化优化通过给定类型去查找关联的类的虚函数表或者类型的witness表,并将虚函数调用替换为调用函数映射
-
性能内联
-
引用计数优化
-
内存提升/优化
-
高级领域特定优化swift编译器对基础的swift类型容器(类似Array或String)实现了高级优化.领域特定优化需要在标准库和优化器之间定义交互.详情可以参考 :ref:
HighLevelSILOptimizations
SIL语法
SIL
依赖于swift的类型系统和声明,所以SIL
语法是swift的延伸.一个.sil
文件是一个增加了SIL
定义的swift源文件.swift源文件只会针对声明进行语法分析.swift的func
方法体(除了嵌套声明)和最高阶的代码会被SIL
语法分析器忽略.在.sil
文件中没有隐式import.如果使用swift
或者Buildin
标准组件的话必须明确的引入. 以下是一个.sil
文件的示例
sil_stage canonical
import Swift
// 定义用于SIL函数的类型
struct Point {
var x : Double
var y : Double
}
class Button {
func onClick()
func onMouseDown()
func onMouseUp()
}
// 定义一个swift函数,函数体会被SIL忽略
func taxicabNorm(_ a:Point) -> Double {
return a.x + a.y
}
// 定义一个SIL函数
// @_T5norms11taxicabNormfT1aV5norms5Point_Sd 是swift函数名taxicabNorm重整之后的命名
sil @_T5norms11taxicabNormfT1aV5norms5Point_Sd : $(Point) -> Double {
bb0(%0 : $Point):
// func Swift.+(Double, Double) -> Double
%1 = function_ref @_Tsoi1pfTSdSd_Sd
%2 = struct_extract %0 : $Point, #Point.x //萃取Point结构体内的x
%3 = struct_extract %0 : $Point, #Point.y ////萃取Point结构体内的y
%4 = apply %1(%2, %3) : $(Double, Double) -> Double //冒号前为计算体实现通过引用的展开,冒号后为类型说明
return %4 : Double //返回值
}
// 定义一个SIL虚函数表,匹配的是动态分派中函数实现的id,这个动态分派是在已知的静态类的类型虚函数表中
sil_vtable Button {
#Button.onClick!1: @_TC5norms6Button7onClickfS0_FT_T_
#Button.onMouseDown!1: @_TC5norms6Button11onMouseDownfS0_FT_T_
#Button.onMouseUp!1: @_TC5norms6Button9onMouseUpfS0_FT_T_
}
SIL阶段
decl ::= sil-stage-decl
sil-stage-decl ::= 'sil_stage' sil-stage
sil-stage ::= 'raw'
sil-stage ::= 'canonical'
基于操作的不同阶段,SIL
拥有不同的声明.
-
Raw SIL, 生的SIL是通过
SILGen
产生的,并未经过保证优化或者诊断通道.Raw SIL
可能没有完善结构的SSA图表.可能会包含数据流错误.一些说明可能会以非规范的方式展示,例如无地址的assign
和destory_addr
的数值.Raw SIL
不应该用于本地代码的生成或分发. -
Canonical SIL,规范SIL是在保证优化和诊断之后的
SIL
.数据流错误必须被消除掉,肯定说明也必须被规范化为更简单的形式.性能优化和本地代码是生成都是从这种格式衍生的.包含这种格式SIL
的组件可以被分发.SIL
文件通过在顶部声明sil_stage raw
或sil_stage canonical
来说明当前的操作阶段.一个文件之后出现一种阶段的声明.
SIL类型
sil-type ::= '/pre> '*'? generic-parameter-list? type
SIL
的类型是通过$
符号进行标记的。SIL
的类型系统和swift的密切相关.所以$
之后的类型会根据swift的类型语法进行语法分析。
类型降级: type lowering
swift的正式类型系统,倾向于对大量的类型信息进行抽象概括.但是SIL
目标是展示更多的实现细节,这个区别也体现在SIL
的类型系统中.所以把正式类型降级为较低类型的操作称为类型降级。
提取区别:Abstraction Difference
包含未约束类型的通用函数一定会被非直接调用.比如分配充足内存和创建地址指针指向这块地址。如下的泛型函数
func generateArray<T>(n : Int, generator : () -> T) -> [T]
函数generator
会通过一个隐式指针,指向存储在一个非直接调用的地址中,(可以参考之前static vs dynamic dispatch中虚函数表的设计和实现).在处理任意类型值时操作都是一样的.
-
我们不希望对
generateArray
的每个T
的类型去产生一个新的拷贝 -
我们不希望对每个类型进行普遍声明
-
我们不希望通过
T
的类型动态的去构造对于genetator
的调用但是我们也不希望现有的通用系统对我们的非通用代码进行低效处理。例如,我们希望
()->Int
可以直接返回结果。但是()->Int
是()->T
的代替(subsitution),对于generateArray<Int>
的调用应该向generator传递()->Int
。 所以一个正式类型在通用上下文中的表现可能会因为正式类型的的代替而不同.我们将这种不同成为提取区别.
SIL
对于类型的提取区别的设计是,在每个级别的代替中,提取数值都可以被使用。
为了可以实现如上设计,泛型实例的正式类型应该一直使用非替换正式类型的提取方式进行降级.例如
struct Generator<T> {
var fn : () -> T
}
var intGen : Generator<Int>
其中intGen.fn
拥有代替类型()->Int
,可以被降级为@callee_owned () -> Int
,可以直接返回结果.但是如果更恰当的使用非代替方式,()->T
就会变成@callee_owned () -> @out Int
当使用非代替的提取方式进行类型降级时,可以看做将拥有相同构造的类型中的具体类型替换为现有类型,以此来实现类型降级. 对于g
的Generator<(Int, Int) -> Float>
,g.fn
是使用()->T
进行降级的,简单理解就是,类型是否是具体类型,如果是,才能进行提取方式进行降级,不然只能产生
@callee_owned () -> @owned @callee_owned (@in (Int, Int)) -> @out Float.
所以提取区别来代替通用函数中类型的标准是:是否是具体类型.is materializable or not
这个系统具有通过重复代替的方式实现提取方式的属性.所以可以把降级的类型看做提取方式的编码. SILGen
已经拥有了使用提取方式转换类型的工序. 目前只有函数和元祖类型会通过提取区别进行改变.
合法的SIL类型
SIL
类型的值应该是这样的:
-
可被加载的SIL类型,
$T
-
合法SIL类型的地址
$*T
或者如果T
是一个合法的SIL类型需要满足以下条件 不展开,需要查看SIL语法中的Legal SIL Types
类型T
满足一下条件才是一个合法的SIL类型
- 函数类型符合
SIL
的约束条件 - metatype可以描述功能
- 原组的内部元素,类型也是合法的
SIL
类型 - 可选
Optional<U>
,U
也是合法类型 - 非函数,原组,可选类型,metatype,或者l-value类型的合法的Swift类型
- 包含合法类型的
@box
注意,在递归条件内的类型,还需要是正式类型。例如泛型内的参数,仍然是Swift类型,而不是SIL
降级类型。
地址类型
地址类型$*T
指针指向的是任意引用的值或者$T
。
地址不是引用计数指针,不能被retained或released。
Box类型
本地变量和非直接的数值类型都是存储在堆上的,@box T
是一个引用计数类型,指向的是包含了多种T
的盒子。盒子使用的是Swift的原生引用计数。
Metatype类型
SIL
内的metatype类型必须描述自身表示:
-
@thin
意思是不需要内存 -
@thick
指存储的是类型的引用或类型子类的引用 -
@objc
指存储的是一个OC类对象的表示而不是Swift类型对象。
函数类型
SIL
中的函数类型和Swift中的函数类型有以下区别:
-
SIL
函数可能是泛型。例如,通过function_ref
返回一个泛型函数类型。 -
SIL
函数可以声明为@noescape
。@noescape
函数类型必须是convention(thin)
或者@callee_guatanteed
。 -
SIL
函数类型声明了以下几种处理上下文的情景:-
@convention(thin)
不要求上下文。这种类型也可以通过@noescape
声明。 -
@callee_guatanteed
会被认为直接参数。也意味着convention(thick)
。 -
@callee_owned
上下文值被认为是不拥有的直接参数。也意味着convention(thick)
。 -
@convention(block)
上下文值被认为是不拥有的直接参数。 - 其他函数类型会被描述为
Properties of Types
和Calling Convention
-
-
SIL
函数必须声明参数的协议。非直接的参数类型是*T
,直接参数类型是T
-
@in
是非直接参数。地址必须是已经初始化的对象,函数负责销毁内部持有的值。 -
@inout
是非直接参数。内存必须是已经初始化的对象。在函数返回之前,必须保证内存是被初始化的。
-
-
SIL
函数需要声明返回值的协议。-
@out
是非直接的结果。地址必须是未初始化的对象。
-
VTables
<pre spellcheck="false" class="md-fences md-end-block ty-contain-cm decl ::= sil-vtable
sil-vtable ::= 'sil_vtable' identifier '{' sil-vtable-entry* '}'
sil-vtable-entry ::= sil-decl-ref ':' sil-linkage? sil-function-name
SIL
使用class_method, super_method, objc_method,和 objc_super_method来表示类方法的动态分派dynamic dispatch
class_method 和 super_method的实现是通过sil_vtable
进行追踪的.sil_vtable
的声明包含一个类的所有方法.
class A {
func foo()
func bar()
func bas()
}
sil @A_foo : $@convention(thin) (@owned A) -> ()
sil @A_bar : $@convention(thin) (@owned A) -> ()
sil @A_bas : $@convention(thin) (@owned A) -> ()
sil_vtable A {
#A.foo!1: @A_foo
#A.bar!1: @A_bar
#A.bas!1: @A_bas
}
class B : A {
func bar()
}
sil @B_bar : $@convention(thin) (@owned B) -> ()
sil_vtable B {
#A.foo!1: @A_foo
#A.bar!1: @B_bar
#A.bas!1: @A_bas
}
class C : B {
func bas()
}
sil @C_bas : $@convention(thin) (@owned C) -> ()
sil_vtable C {
#A.foo!1: @A_foo
#A.bar!1: @B_bar
#A.bas!1: @C_bas
}
swift的AST包含重载关系,可以用于在SIL
的虚函数表中查找衍生类重载方法. 为了避免SIL
方法是thunks,方法名是连接在原始方法实现之前.
Witness Tables
decl ::= sil-witness-table
sil-witness-table ::= 'sil_witness_table' sil-linkage?
normal-protocol-conformance '{' sil-witness-entry* '}'
SIL
将通用类型动态分派所需的信息编码为witness表.这些信息用于在生成二进制码时产生运行时分配表(runtime dispatch table).也可以用于对特定通用函数的SIL
优化.每个明确的一致性声明都会产生witness表.通用类型的所有实例共享一个通用witness表.衍生类会继承基类的witness表.
protocol-conformance ::= normal-protocol-conformance
protocol-conformance ::= 'inherit' '(' protocol-conformance ')'
protocol-conformance ::= 'specialize' '<' substitution* '>'
'(' protocol-conformance ')'
protocol-conformance ::= 'dependent'
normal-protocol-conformance ::= identifier ':' identifier 'module' identifier
witness的关键在于协议一致性.它是对于具体类型协议一致性的唯一标识.
-
标准的协议一致性命名了一种协议方法需要遵守的类型.属于该类型或扩展的组件,需要提供遵守协议方法的声明
-
如果派生类实现了基类遵守的协议,会体现为继承协议一致性,简单引用基类的协议一致性即可.
-
如果通用类型的实例遵守一个协议,是通过特定遵守的方式去实现的.这种方式向标准一致性提供了用于通用类型的通用参数构建.
witness table
只会直接关联标准一致性.继承和特定一致性是在标准一致性下的间接引用.
sil-witness-entry ::= 'base_protocol' identifier ':' protocol-conformance
sil-witness-entry ::= 'method' sil-decl-ref ':' sil-function-name
sil-witness-entry ::= 'associated_type' identifier
sil-witness-entry ::= 'associated_type_protocol'
'(' identifier ':' identifier ')' ':' protocol-conformance
witness table
由以下内容构成
-
基协议项提供的对于协议一致性的引用,可以用于witness协议的继承协议
-
方法项将协议中要求方法映射为
SIL
中实现了witness类型的方法.每个方法项必须对应witness协议中的要求方法 -
associate type关联类型项将必须实现的协议方法中的关联类型映射为符合witness的类型.注意witness类似是一个资源级别的swift类型,不是
SIL
类型(上面分析过SIL
类型和swift类型的区别).关联类型项必须覆盖witness协议中的所有强制关联项. -
关联类型协议项将关联类型中的协议映射为关联类型的协议一致性.
witness table作用
swift中的协议是通过结构体实现的,可以支持交互.例如参数,属性都可以是结构体.当将结构体传递给协议参数时,结构体特定的部分可能会丢失(在编译期).协议的witness table就可以发挥作用(在运行时).
Default Witness Tables
decl ::= sil-default-witness-table
sil-default-witness-table ::= 'sil_default_witness_table'
identifier minimum-witness-table-size
'{' sil-default-witness-entry* '}'
minimum-witness-table-size ::= integer
SIL
编码要求默认witness table有开放(resilient)的默认实现.包含以下条件
-
强制方法有默认实现
-
不是协议中最后一个默认方法或继承的强制方法,都有开放的默认实现.
强制方法的开放的默认实现,存储在协议的元数据中. 默认witness表关键在在自身协议.只有公共可见协议才需要默认witness表.私有协议和内部协议是对外部组件不可见的,所以他们没有增加新的强制方法的开放性问题.
sil-default-witness-entry ::= 'method' sil-decl-ref ':' sil-function-name
默认witness表目前只包含一项内容
- 方法像,将协议中的要求方法映射到
SIL
中实现了管理所有witness类型的方法.
全局变量
数据流错误
数据流错误可能存在于Raw SIL
中,swift从语义上将那些条件定义为错误,所以他们必须使用诊断通道进行诊断,并且不能存在于规范SIL中. 定义初始化 swift要求所有的本地变量在使用前必须被初始化.在构造函数中,结构体,枚举或类类型的实例变量必须在对象被使用前初始化. 未全面覆盖(unreachable)的控制流 unreachable
在raw SIL
中生成,标记错误的控制流.例如对于非Void
的函数没有返回值,或者switch
没有完全覆盖所有的条件.这种dead code
消解的保证,可以避免unreachable
的基础block,也可以避免方法返回不合法的空类型.
运行时错误
一些操作,比如无条件的检查转换次数失败或者编译器Buildin.trap
.都会引起运行时错误,这种错误会无条件的终止当前操作.如果可以检验运行时错误会发生或者已经发生.只要将它们排列到程序操作之后就可以将这些运行时错误重新安排.例如对于没有确定开始和结尾的for循环代码
// Given unknown start and end values, this loop may overflow
for var i = unknownStartValue; i != unknownEndValue; ++i {
...
}
会将内存溢出挂起,产生loop的关联运行时错误,之后检测循环的起始和结束点.只要循环体对于当前的操作没有可见影响即可.
未定义的行为
某些操作的错误使用成为未定义行为.例如对于Buildin.RawPointer
的不可用未检测的类型转换.或者使用低于LLVM说明的编译器内建函数,调用当前LLVM不支持的行为.SIL
程序中的未定义行为是无意义的,就像C中的未定义行为一样,没有语义对其进行预测.未定义行为不应该被合法的SIL文件触发,但是在SIL
级别不一定会被检测和证实.
调用协议
以下内容讨论swift函数是如何生成SIL
的.
swift调用协议 @convention(swift) swift本地方法默认使用siwft调用协议是. 入参为原组的函数被递归解构为单独的参数,即包含被调用的基础块的入口,也包含调用者的apply
说明
func foo(_ x:Int, y:Int)
sil @foo : $(x:Int, y:Int) -> () {
entry(%x : $Int, %y : $Int):
...
}
func bar(_ x:Int, y:(Int, Int))
sil @bar : $(x:Int, y:(Int, Int)) -> () {
entry(%x : $Int, %y0 : $Int, %y1 : $Int):
...
}
func call_foo_and_bar() {
foo(1, 2)
bar(4, (5, 6))
}
sil @call_foo_and_bar : $() -> () {
entry:
...
%foo = function_ref @foo : $(x:Int, y:Int) -> ()
%foo_result = apply %foo(%1, %2) : $(x:Int, y:Int) -> ()
...
%bar = function_ref @bar : $(x:Int, y:(Int, Int)) -> ()
%bar_result = apply %bar(%4, %5, %6) : $(x:Int, y:(Int, Int)) -> ()
}
调用以繁琐数据类型作为入参和输出值的函数时
func foo(_ x:Int, y:Float) -> UnicodeScalar
foo(x, y)
在SIL
内如下体现
%foo = constant_ref $(Int, Float) -> UnicodeScalar, @foo
%z = apply %foo(%x, %y) : $(Int, Float) -> UnicodeScalar
swift方法调用协议@convention(method) 方法调用协议用于独立方法的调用协议.柯里化method,使用self
作为内部和外部参数.如果是非柯里化函数,self会在最后被传入
struct Foo {
func method(_ x:Int) -> Int {}
}
sil @Foo_method_1 : $((x : Int), @inout Foo) -> Int { ... }
witness方法调用协议@convention(witness_method) witness方法调用协议是用于witness tables中的协议witness方法.它几乎等同于方法调用协议,只有对通用类型参数处理方面不同.对于非witness方法来说,机器协议可能会通过方法类型将方法签名进行静态转换.但是因为witness必须在Self
类型下进行多态分配,所以Self
相关的元数据必须通过最大化的提取规则传输.
C调用协议@convention(c) 在swift的C组件编译器中,C类型会被SIL
对照到swift类型.C的函数入参和返回值,都会被SIL
平台调用协议忽略. SIL
和swift目前都不能调用包含可变参数的C的方法.
OC调用协议@convention(objc_method) SIL
中的OC方法使用规范和ARC内一致.也可以从OC定义中引入属性. 使用@convention(block)
并不会影响block的引用计数.
在SIL
中OC方法的self参数是非柯里化的最后一个参数.就像原生swift方法
@objc class NSString {
func stringByPaddingToLength(Int) withString(NSString) startingAtIndex(Int)
}
sil @NSString_stringByPaddingToLength_withString_startingAtIndex \
: $((Int, NSString, Int), NSString)
IR级别的将self
作为第一个参数的行为在SIL
中提取了的.比如现存的_cmd
方法参数.
基于类型的别名分析: type based alias analysis
SIL
提供了两种类型别名分析(TBAA
: Type Based Alias Analysis):类TBAA和类型访问TBAA
指令集
感兴趣可以自行查看SIL指令集
初始化和销毁
alloc_stack
sil-instruction ::= 'alloc_stack' sil-type (',' debug-var-attr)*
%1 = alloc_stack $T
// %1 has type $*T
在栈区开辟充分符合T
类型的内存空间。指令的返回结果是初始化的内存地址。
如果类型的尺寸在运行时才能确定,编译器必须动态初始化内存。所以并不能确保内存一定是初始化在栈区,例如如果是特别大的数值,可能会在堆区初始化,栈区持有指针。
alloc_stack
标记了值声明周期的开始。在结束时必须使用dealloc_stack
销毁。
内存不能被retain,如果想初始化可retain的类型,使用alloc_box
。
总结:alloc_stack
在栈区为值类型开辟内存。不使用引用计数。
alloc_box
sil-instruction ::= 'alloc_box' sil-type (',' debug-var-attr)*
%1 = alloc_box $T
// %1 has type $@box T
在堆上开辟足够大的内存来支持各种类型的T
,以@box
持有引用计数。这个指令的结果是@box
的引用计数持有的box,project_box
是要来过去box内部的值的地址的。
box初始化时引用计数为1,但是内存并不会被初始化。box持有内部的值,在引用计数为0时使用destory_addr
对内部值进行释放,无法释放box的值没有被初始化的情况。这时候需要用到dealloc_box
。
总结:alloc_box
在堆上初始化指针类型的值,并且需要手动管理内存。
alloc_box和alloc_stack对比
alloc_box
和alloc_stack
最大的区别在于值的生命周期。举例,如果在闭包之外有一个变量声明,在闭包内使用了该变量。变量的值是可以被修改的,所以需要使用alloc_box
来引用变量。
对于var
声明的变量,因为可以多次修改它的值,甚至在作用域外也可以修改。所以使用alloc_box
管理引用计数。
优化:Alloc box to stack
在SILGen阶段,会对闭包内使用变量的情况,通过alloc_box
进行管理。
在SIL guaranteed transformations阶段,即生成正式SIL的阶段,会对于在闭包内没有进行值修改的变量内存分配进行优化,将alloc_box
替换为alloc_stack
。这个功能是在AllocBoxToStack组件内实现的。内部实现是将堆区不必要的初始化移动到栈区。