美文网首页Swift
Swift编译之SIL(Swift Intermediate L

Swift编译之SIL(Swift Intermediate L

作者: YungFan | 来源:发表于2023-11-19 09:00 被阅读0次

编译过程

Swift编译过程.jpeg
  • Parse:从.swift构造 AST(抽象语法树)。
  • Sema:对 AST 进行语义分析,生成格式正确且类型检查完备的 AST。
  • SILGen:从 AST 生成 raw SIL。
  • IRGen:从 canonical SIL 生成 IR。
  • LLVM:LLVM Backend 从优化后的 IR 生成汇编代码或者目标代码。

SIL生成

// 将main.swift编译成SIL
swiftc -emit-sil main.swift > main.sil
// 将main.swift编译成SIL的同时还原毫无规则的命名
swiftc -emit-sil main.swift | xcrun swift-demangle > main.sil
// UIKit+x86_64
swiftc -emit-sil -target x86_64-apple-ios17.0.1-simulator -sdk $(xcrun --show-sdk-path --sdk iphonesimulator) ViewController.swift | xcrun swift-demangle > ViewController.sil
// UIKit+arm64
swiftc -emit-sil -target arm64-apple-ios17.0.1-simulator -sdk $(xcrun --show-sdk-path --sdk iphonesimulator) ViewController.swift | xcrun swift-demangle > ViewController.sil
// SwiftUI
swiftc -emit-sil ContentView.swift | xcrun swift-demangle > ContentView.sil

SIL常见语法

  • hidden:同一个 Swift 模块中的对象可见。
  • load A:从 A 中读取数据。
  • store A to B:将 A 中的值存储到 B 中。
  • sil_global:全局变量。
  • alloc_global:开辟全局变量的内存。
  • global_addr:获取全局变量的地址。
  • alloc_stack/dealloc_stack:开辟/销毁栈区内存空间。
  • alloc_box/dealloc_box:开辟/释放堆区内存空间。
  • bb0/bb1 ...:标记代码块。
  • alloc_ref/dealloc_ref:开辟/释放内存。
  • function_ref:获取直接派发方法引用地址。
  • class_method:通过方法表获取方法引用地址。
  • witness_method:通过 Protocol Witness Table(PWT) 获取对应的方法引用地址。
  • objc_method:获取 Objective-C 方法引用地址。
  • apply:调用方法。
  • return:返回值。
  • retain_value/release_value:引用计数 +1/引用计数 -1。
  • metatype:获取元类型。
  • begin_access/end_access:开始/结束访问目标内存。
  • switch_enum:switch 枚举,一般进行判空操作。
  • pointer_to_address:将原始指针转换为地址。
  • br lable:将控制权从当前块转移到其他块,并将进行传值。
  • $:类型标识。
  • %number:寄存器。
  • @方法名:某个方法的名字。
  • @_hasStorage:存储属性。
  • @_hasInitialValue:属性有初始值。
  • @owned:方法接收者负责销毁返回值。
  • @convention(method):Swift 方法的引用。
  • @convention(thin):Swift 方法的引用,但没有特殊的self或者context参数。
  • @convention(thick):Swift 方法的引用,拥有引用计数的上下文对象。
  • @convention(c):C 方法的引用。
  • @convention(objc_method):Objective-C 方法的引用。
  • @convention(block):Objective-C 块的引用。
  • @convention(witness_method):协议方法的引用。

案例

  • 源代码。
import Foundation

protocol SomeProtocol {
    func doSomething()
}

@objc class Person: NSObject, SomeProtocol {
    var name = "zhangsan"

    @objc dynamic func say() {
    }

    func doSomething() {
    }
}

extension Person {
    func drink() {
    }
}
  • SIL。
// 表示最终的、经过规范化的SIL代码
sil_stage canonical

// 内建模块,提供了一些底层的内建函数和类型,这些函数和类型通常用于Swift编译器生成的代码中,执行一些特定的低级操作
import Builtin
// Swift标准库的模块,提供了Swift语言的核心功能,包括标准类型、集合、字符串处理等
import Swift
// Swift语言的桥接模块,用于处理Swift与其他语言的交互,例如Objective-C
import SwiftShims

import Foundation

protocol SomeProtocol {
    func doSomething()
}

// 使用@_inheritsConvenienceInitializers注解,继承方便的初始化器
// 使用@objc标记,将类导出到Objective-C
@_inheritsConvenienceInitializers @objc class Person: NSObject, SomeProtocol {
    // 使用@_hasStorage标记,表示这个属性有存储
    // 使用@_hasInitialValue标记,表示这个属性有初始值
    @_hasStorage @_hasInitialValue var name: String { get set }
    // 使用@objc dynamic标记,将方法导出到Objective-C,并启用动态派发
    @objc dynamic func say()
    // 没有@objc标记的普通方法
    func doSomething()
    // 使用override dynamic标记,覆盖父类的初始化器并启用动态派发
    override dynamic init()
    // 使用@objc标记的析构器,将析构器导出到Objective-C
    @objc deinit
}

extension Person {
    func drink()
}

// 使用sil指令定义一个名为main的函数,程序的入口点
// main
sil @main: $@convention(c) (Int32, UnsafeMutablePointer<Optional<UnsafeMutablePointer<Int8>>>) -> Int32 {
    // 定义基本块bb0,它接受两个参数,一个是Int32,另一个是UnsafeMutablePointer<Optional<UnsafeMutablePointer<Int8>>>
    bb0(%0: $Int32, %1: $UnsafeMutablePointer<Optional<UnsafeMutablePointer<Int8>>>):
        // 使用integer_literal指令创建一个Int32类型的字面量0
        %2 = integer_literal $Builtin.Int32, 0 // user: %3,user表示这个操作数标记了在用户级别上使用了该值,表示该值在程序的实际执行中被使用了
        // 使用struct指令将上面创建的Int32字面量包装成Int32结构体
        %3 = struct $Int32(%2: $Builtin.Int32) // user: %4
        // 返回创建的Int32结构体,结束main函数
        return %3: $Int32 // id: %4,id表示用于在程序的其他部分引用该值
} // end sil function 'main'

// 使用sil hidden [transparent]指令定义一个隐藏的、透明的函数,表示初始化Person类的name属性的表达式
// sil hidden:表示这个表达式是隐藏的,即不对外部可见
// [transparent]:表示这个表达式是透明的,即不包含额外的控制流或操作
// @variable initialization expression:表示这是一个变量初始化表达式
// of main.Person.name:指明这个表达式是为Person类的name属性进行初始化
// : Swift.String:指定初始化的目标类型是Swift.String
// : $@convention(thin) () -> @owned String:指定了函数的调用约定,这个表达式的类型是一个函数类型,该函数不接受参数 (()),返回一个拥有所有权的String
// variable initialization expression of Person.name
sil hidden[transparent] @variable initialization expression of main.Person.name: Swift.String: $@convention(thin) () -> @owned String {
    bb0:
        // 使用string_literal指令创建一个UTF-8字符串字面量"zhangsan"
        %0 = string_literal utf8 "zhangsan" // user: %5
        // 使用integer_literal指令创建一个Word类型的字面量8,表示UTF-8编码单元的数量
        %1 = integer_literal $Builtin.Word, 8 // user: %5
        // 使用integer_literal指令创建一个Int1类型的字面量-1,表示字符串是否为ASCII编码
        %2 = integer_literal $Builtin.Int1, -1 // user: %5
        // 使用metatype指令获取String类型的元类型
        %3 = metatype $@thin String.Type // user: %5
        // 使用function_ref指令引用String.init(_builtinStringLiteral:utf8CodeUnitCount:isASCII:)构造函数
        // 这个构造函数接受一个指向UTF-8字符串的原始指针、UTF-8编码单元的数量、以及一个表示是否为ASCII的标志
        // function_ref String.init(_builtinStringLiteral:utf8CodeUnitCount:isASCII:)
        %4 = function_ref @Swift.String(_builtinStringLiteral: Builtin.RawPointer, utf8CodeUnitCount: Builtin.Word, isASCII: Builtin.Int1) -> Swift.String: $@convention(method) (Builtin.RawPointer, Builtin.Word, Builtin.Int1, @thin String.Type) -> @owned String // user: %5
        // 使用apply指令调用上述构造函数
        %5 = apply %4(%0, %1, %2, %3): $@convention(method) (Builtin.RawPointer, Builtin.Word, Builtin.Int1, @thin String.Type) -> @owned String // user: %6
        // 返回构造函数的结果,即初始化的字符串
        return %5: $String // id: %6
} // end sil function 'variable initialization expression of main.Person.name : Swift.String'

// String类的构造函数init(_builtinStringLiteral:utf8CodeUnitCount:isASCII:)的定义
// sil[always_inline][readonly][_semantics "string.makeUTF8"] :这是关于方法行为和优化的一些指令
// [always_inline]:提示编译器尽可能内联这个方法,以提高性能
// [readonly]:表明该方法是只读的,不对传入的参数进行修改
// [_semantics "string.makeUTF8"]:指定了该方法的语义,这对编译器进行进一步的优化和特殊处理
// @Swift.String.init(_builtinStringLiteral: Builtin.RawPointer, utf8CodeUnitCount: Builtin.Word, isASCII: Builtin.Int1) -> Swift.String:指定了这个方法的Swift类型签名
// String.init(_builtinStringLiteral:utf8CodeUnitCount:isASCII:)
sil[always_inline][readonly][_semantics "string.makeUTF8"] @Swift.String(_builtinStringLiteral: Builtin.RawPointer, utf8CodeUnitCount: Builtin.Word, isASCII: Builtin.Int1) -> Swift.String: $@convention(method) (Builtin.RawPointer, Builtin.Word, Builtin.Int1, @thin String.Type) -> @owned String

// 使用sil hidden指令定义Person类的name属性的getter方法
// Person.name.getter
sil hidden @main.Person.name.getter: Swift.String: $@convention(method) (@guaranteed Person) -> @owned String {
    // 基本块bb0,接受一个Person实例作为参数
    // %0 "self"                                      // users: %2, %1
    bb0(%0: $Person):
        // 在调试中,为参数"self"创建一个调试值
        debug_value %0: $Person, let, name "self", argno 1, implicit // id: %1
        // 使用ref_element_addr指令获取Person实例的name属性的地址
        %2 = ref_element_addr %0: $Person, #Person.name // user: %3
        // 使用begin_access指令开始对name属性的读取访问,dynamic表示动态访问
        %3 = begin_access[read][dynamic] %2: $ * String // users: %4, %6
        // 使用load指令加载name属性的值
        %4 = load %3: $ * String // users: %7, %5
        // 使用retain_value指令保留加载的字符串值
        retain_value %4: $String // id: %5
        // 使用end_access指令结束对name属性的读取访问
        end_access %3: $ * String // id: %6
        // 返回加载并保留的字符串值
        return %4: $String // id: %7
} // end sil function 'main.Person.name.getter : Swift.String'

// 使用sil hidden指令定义Person类的name属性的setter方法,接受一个String实例和一个Person实例
// Person.name.setter
sil hidden @main.Person.name.setter: Swift.String: $@convention(method) (@owned String, @guaranteed Person) -> Void {
    // 定义基本块bb0,它接受两个参数,一个是String实例,另一个是Person实例
    // %0 "value"                                     // users: %11, %8, %4, %2
    // %1 "self"                                      // users: %5, %3
    bb0(%0: $String, %1: $Person):
        // 调试信息,标记参数%0为"value",参数%1为"self"
        debug_value %0: $String, let, name "value", argno 1, implicit // id: %2
        debug_value %1: $Person, let, name "self", argno 2, implicit // id: %3
        // 保留String实例,增加其引用计数
        retain_value %0: $String // id: %4
        // 获取Person实例的name属性地址
        %5 = ref_element_addr %1: $Person, #Person.name // user: %6
        // 开始访问name属性,以便进行修改
        %6 = begin_access[modify][dynamic] %5: $ * String // users: %8, %7, %10
        // 加载原始的String实例
        %7 = load %6: $ * String // user: %9
        // 将新的String实例存储到name属性地址
        store %0 to %6: $ * String // id: %8
        // 释放原始的String实例
        release_value %7: $String // id: %9
        // 结束对name属性的访问
        end_access %6: $ * String // id: %10
        // 释放String实例,减少其引用计数
        release_value %0: $String // id: %11
        // 创建一个空元组作为返回值
        %12 = tuple() // user: %13
        // 返回空元组
        return %12: $() // id: %13
} // end sil function 'main.Person.name.setter : Swift.String'

// 使用sil hidden指令定义Person类的name属性的modify方法
// sil hidden:表示这个modify方法是隐藏的,即不对外部可见
// @main.Person.name.modify:指明这个modify方法是Person类的name属性的modify方法
// : Swift.String:指定modify方法的返回类型是Swift.String,这表示modify方法返回属性的当前值
// : $@yield_once:指定了modify方法的调用约定,其中@yield_once表示该方法会产生一次yield modify 方法通常用于实现属性的写时拷贝,在获取属性时会执行一些操作,并在修改时生成一个拷贝
// @convention(method) (@guaranteed Person) -> @yields @inout String:指定modify方法的签名,该方法接受一个Person实例作为参数,返回一个 String,并且产生一个@yield的值,表示写时拷贝的结果 方法使用@inout String表示这是一个写入操作,需要返回一个写入后的值
// Person.name.modify
sil hidden @main.Person.name.modify: Swift.String: $@yield_once @convention(method) (@guaranteed Person) -> @yields @inout String {
    // 定义基本块bb0,它接受一个Person实例作为参数
    // %0 "self"                                      // users: %2, %1
    bb0(%0: $Person):
        // 调试信息,标记参数%0为"self"
        debug_value %0: $Person, let, name "self", argno 1, implicit // id: %1
        // 获取Person实例的name属性地址
        %2 = ref_element_addr %0: $Person, #Person.name // user: %3
        // 开始访问name属性,以便进行inout修改
        %3 = begin_access[modify][dynamic] %2: $ * String // users: %5, %8, %4
        // 使用yield指令将对name属性的inout访问暂停,并传递name属性的地址
        yield %3: $ * String, resume bb1, unwind bb2 // id: %4

    bb1: // Preds: bb0
        // 结束对name属性的访问
        end_access %3: $ * String // id: %5
        // 创建一个空元组作为返回值
        %6 = tuple() // user: %7
        return %6: $() // id: %7

    bb2: // Preds: bb0
        // 在异常情况下,结束对name属性的访问并执行unwind
        end_access %3: $ * String // id: %8
        unwind // id: %9
} // end sil function 'main.Person.name.modify : Swift.String'

// 使用sil hidden指令定义Person类的say()方法
// Person.say()
sil hidden @main.Person.say() -> Void: $@convention(method) (@guaranteed Person) -> Void {
    // 定义基本块bb0,它接受一个Person实例作为参数
    // %0 "self"                                      // user: %1
    bb0(%0: $Person):
        // 调试信息,标记参数%0为"self"
        debug_value %0: $Person, let, name "self", argno 1, implicit // id: %1
        // 创建一个空元组作为返回值
        %2 = tuple() // user: %3
        // 返回空元组
        return %2: $() // id: %3
} // end sil function 'main.Person.say() -> ()'

// 使用sil private [thunk]指令定义Objective-C中的thunk实现,将@objc修饰的Person.say()映射到Swift中的main.Person.say()方法
// @objc Person.say()
sil private [thunk] @@objc main.Person.say() -> Void: $@convention(objc_method) (Person) -> Void {
    // 定义基本块bb0,它接受一个Person实例作为参数
    // %0                                             // users: %4, %3, %1
    bb0(%0: $Person):
        // 强引用计数+1,保留Person实例
        strong_retain %0: $Person // id: %1
        // 引用Swift中的Person.say()方法
        // function_ref Person.say()
        %2 = function_ref @main.Person.say() -> Void: $@convention(method) (@guaranteed Person) -> Void // user: %3
        // 调用Swift中的Person.say()方法
        %3 = apply %2(%0): $@convention(method) (@guaranteed Person) -> Void // user: %5
        // 强引用计数-1,释放Person实例
        strong_release %0: $Person // id: %4
        // 返回Swift方法的结果
        return %3: $() // id: %5
} // end sil function '@objc main.Person.say() -> ()'

// 使用sil hidden指令定义Person类的doSomething()方法
// Person.doSomething()
sil hidden @main.Person.doSomething() -> Void: $@convention(method) (@guaranteed Person) -> Void {
    // 定义基本块bb0,它接受一个Person实例作为参数
    // %0 "self"                                      // user: %1
    bb0(%0: $Person):
        // 调试信息,标记参数%0为"self"
        debug_value %0: $Person, let, name "self", argno 1, implicit // id: %1
        // 创建一个空元组作为返回值
        %2 = tuple() // user: %3
        // 返回空元组
        return %2: $() // id: %3
} // end sil function 'main.Person.doSomething() -> ()'

// 使用sil hidden指令定义Person类的__allocating_init()方法,是一个指定初始化器
// (@thick Person.Type):表示这个方法接受一个Person类型的元类型参数,即类本身的类型
// -> @owned Person:表示这个方法返回一个拥有所有权的Person实例
// Person.__allocating_init()
sil hidden @main.Person.__allocating_init() -> main.Person: $@convention(method) (@thick Person.Type) -> @owned Person {
    // 定义基本块bb0,它接受一个Person类型的元类型作为参数
    // %0 "$metatype"                                 // user: %1
    bb0(%0: $@thick Person.Type):
        // 将thick元类型转换为对应的Objective-C元类型
        %1 = thick_to_objc_metatype %0: $@thick Person.Type to $@objc_metatype Person.Type // user: %2
        // 使用alloc_ref_dynamic指令分配Person实例
        %2 = alloc_ref_dynamic[objc] %1: $@objc_metatype Person.Type, $Person // users: %3, %4
        // 使用objc_method指令调用Objective-C中的初始化器init
        %3 = objc_method %2: $Person, #Person.init!initializer.foreign: (Person.Type) -> () -> Person, $@convention(objc_method) (@owned Person) -> @owned Person // user: %4
        // 调用init初始化器
        %4 = apply %3(%2): $@convention(objc_method) (@owned Person) -> @owned Person // user: %5
        // 返回初始化后的Person实例
        return %4: $Person // id: %5
} // end sil function 'main.Person.__allocating_init() -> main.Person'

// 使用sil shared [transparent] [thunk]指令定义动态初始化器的thunk实现
// sil shared:表示这个初始化方法是共享的,即可能被其他模块访问
// [transparent]:表示这个初始化方法是透明的,即没有额外的控制流或操作
// [thunk]:表示这是一个动态分发的方法,可能涉及到动态派发
// @dynamic main.Person.init():指明这个方法是Person类的初始化方法
// -> main.Person:表示这个初始化方法的返回类型是Person类的实例
// dynamic Person.init()
sil shared[transparent][thunk] @dynamic main.Person() -> main.Person: $@convention(method) (@owned Person) -> @owned Person {
    // 定义基本块bb0,它接受一个Person实例作为参数
    // %0 "self"                                      // users: %2, %1
    bb0(%0: $Person):
        // 使用objc_method指令调用Objective-C中的初始化器init
        %1 = objc_method %0: $Person, #Person.init!initializer.foreign: (Person.Type) -> () -> Person, $@convention(objc_method) (@owned Person) -> @owned Person // user: %2
        // 调用init初始化器
        %2 = apply %1(%0): $@convention(objc_method) (@owned Person) -> @owned Person // user: %3
        // 返回初始化后的Person实例
        return %2: $Person // id: %3
} // end sil function 'dynamic main.Person.init() -> main.Person'

// 使用sil hidden指令定义Person类的init()方法
// Person.init()
sil hidden @main.Person() -> main.Person: $@convention(method) (@owned Person) -> @owned Person {
    // 定义基本块bb0,它接受一个Person实例作为参数
    // %0 "self"                                      // users: %12, %4, %3, %2
    bb0(%0: $Person):
        // 分配一个Person实例的栈空间
        %1 = alloc_stack[lexical] $Person, let, name "self", implicit // users: %20, %13, %3, %21, %22
        // 强引用计数+1,保留Person实例
        strong_retain %0: $Person // id: %2
        // 将Person实例存储到栈空间
        store %0 to %1: $ * Person // id: %3
        // 获取Person实例的name属性地址
        %4 = ref_element_addr %0: $Person, #Person.name // user: %11
        // 创建String实例,初始化为"zhangsan"
        %5 = string_literal utf8 "zhangsan" // user: %10
        %6 = integer_literal $Builtin.Word, 8 // user: %10
        %7 = integer_literal $Builtin.Int1, -1 // user: %10
        %8 = metatype $@thin String.Type // user: %10
        // 获取并调用String.init(_builtinStringLiteral:utf8CodeUnitCount:isASCII:)方法
        // function_ref String.init(_builtinStringLiteral:utf8CodeUnitCount:isASCII:)
        %9 = function_ref @Swift.String(_builtinStringLiteral: Builtin.RawPointer, utf8CodeUnitCount: Builtin.Word, isASCII: Builtin.Int1) -> Swift.String: $@convention(method) (Builtin.RawPointer, Builtin.Word, Builtin.Int1, @thin String.Type) -> @owned String // user: %10
        %10 = apply %9(%5, %6, %7, %8): $@convention(method) (Builtin.RawPointer, Builtin.Word, Builtin.Int1, @thin String.Type) -> @owned String // user: %11
        // 将创建的String实例存储到name属性地址
        store %10 to %4: $ * String // id: %11
        // 强引用计数-1,释放Person实例
        strong_release %0: $Person // id: %12
        // 加载栈空间中的Person实例
        %13 = load %1: $ * Person // user: %14
        // 对Person实例进行强转换为NSObject
        %14 = upcast %13: $Person to $NSObject // users: %15, %17
        // 调用NSObject的init方法
        %15 = unchecked_ref_cast %14: $NSObject to $Person // user: %16
        %16 = objc_super_method %15: $Person, #NSObject.init!initializer.foreign: (NSObject.Type) -> () -> NSObject, $@convention(objc_method) (@owned NSObject) -> @owned NSObject // user: %17
        %17 = apply %16(%14): $@convention(objc_method) (@owned NSObject) -> @owned NSObject // user: %18
        // 对结果进行强转换回Person类型
        %18 = unchecked_ref_cast %17: $NSObject to $Person // users: %20, %23, %19
        // 强引用计数+1,保留Person实例
        strong_retain %18: $Person // id: %19
        // 将Person实例存储到栈空间
        store %18 to %1: $ * Person // id: %20
        // 销毁栈空间中的Person实例
        destroy_addr %1: $ * Person // id: %21
        // 释放栈空间
        dealloc_stack %1: $ * Person // id: %22
        // 返回初始化后的Person实例
        return %18: $Person // id: %23
} // end sil function 'main.Person.init() -> main.Person'

// 使用sil private [thunk]指令定义Objective-C中的thunk实现,将@objc修饰的Person.init()映射到Swift中的main.Person.init()方法
// @@objc main.Person.init():指明这个协议见证是为Person类的Objective-C初始化方法
// @convention(objc_method):表示这是一个Objective-C方法
// @objc Person.init()
sil private [thunk] @@objc main.Person() -> main.Person: $@convention(objc_method) (@owned Person) -> @owned Person {
    // 定义基本块bb0,它接受一个Person实例作为参数
    // %0                                             // user: %2
    bb0(%0: $Person):
        // 引用Swift中的Person.init()方法
        // function_ref Person.init()
        %1 = function_ref @main.Person() -> main.Person: $@convention(method) (@owned Person) -> @owned Person // user: %2
        // 调用并返回对象
        %2 = apply %1(%0): $@convention(method) (@owned Person) -> @owned Person // user: %3
        return %2: $Person // id: %3
} // end sil function '@objc main.Person.init() -> main.Person'

// 使用sil hidden指令定义Person类的__deallocating_deinit方法,是一个析构器
// Person.__deallocating_deinit
sil hidden @main.Person.__deallocating_deinit: $@convention(method) (@owned Person) -> Void {
    // 定义基本块bb0,它接受一个Person实例作为参数
    // %0 "self"                                      // users: %3, %2, %1
    bb0(%0: $Person):
        // 调试信息,标记参数%0为"self"
        debug_value %0: $Person, let, name "self", argno 1, implicit // id: %1
        // 使用objc_super_method指令调用NSObject的析构器deinit
        %2 = objc_super_method %0: $Person, #NSObject.deinit!deallocator.foreign: (NSObject) -> () -> Void, $@convention(objc_method) (NSObject) -> Void // user: %4
        // 对Person实例进行强转换为NSObject类型
        %3 = upcast %0: $Person to $NSObject // user: %4
        // 调用NSObject的析构器deinit
        %4 = apply %2(%3): $@convention(objc_method) (NSObject) -> Void
        // 创建一个空元组作为返回值
        %5 = tuple() // user: %6
        // 返回空元组
        return %5: $() // id: %6
} // end sil function 'main.Person.__deallocating_deinit'

// 使用sil hidden @@objc指令定义Objective-C中的实例变量销毁方法__ivar_destroyer
// @objc Person.__ivar_destroyer
sil hidden @@objc main.Person.__ivar_destroyer: $@convention(objc_method) (Person) -> Void {
    // 定义基本块bb0,它接受一个Person实例作为参数
    // %0 "self"                                      // users: %2, %1
    bb0(%0: $Person):
        // 调试信息,标记参数%0为"self"
        debug_value %0: $Person, let, name "self", argno 1, implicit // id: %1
        // 获取Person实例的name属性地址
        %2 = ref_element_addr %0: $Person, #Person.name // user: %3
        // 开始访问name属性,指定为deinit操作
        %3 = begin_access[deinit][static] %2: $ * String // users: %5, %4
        // 销毁name属性的内存
        destroy_addr %3: $ * String // id: %4
        // 结束对name属性的访问
        end_access %3: $ * String // id: %5
        // 创建一个空元组作为返回值
        %6 = tuple() // user: %7
        // 返回空元组
        return %6: $() // id: %7
} // end sil function '@objc main.Person.__ivar_destroyer'

// 使用sil private [transparent] [thunk]指令定义协议witness实现,实现SomeProtocol协议的doSomething()方法
// sil private:表示这个协议见证是私有的,即不对外部可见
// [transparent]:表示这个协议见证是透明的,即没有额外的控制流或操作
// [thunk]:表示这是一个动态分发的协议见证,可能涉及到动态派发
// @protocol witness for main.SomeProtocol.doSomething() -> ():指定了这个协议见证是为SomeProtocol协议的doSomething()方法
// in conformance main.Person : main.SomeProtocol in main:指明这个协议见证是在Person类对SomeProtocol协议的遵循中
// @convention(witness_method: SomeProtocol):表示这是一个协议方法的见证
// (@in_guaranteed Person):表示这个方法接受一个在整个调用期间保证有效的Person实例作为参数
// -> ():表示这个方法没有返回值
// protocol witness for SomeProtocol.doSomething() in conformance Person
sil private [transparent][thunk] @protocol witness for main.SomeProtocol.doSomething() in conformance main.Person: main.SomeProtocol in main: $@convention(witness_method: SomeProtocol) (@in_guaranteed Person) -> Void {
    // 定义基本块bb0,它接受一个指向Person实例的指针作为参数
    // %0                                             // user: %1
    bb0(%0: $ * Person):
        // 加载Person实例
        %1 = load %0: $ * Person // users: %2, %3
        // 获取并调用Person实例的doSomething()方法
        %2 = class_method %1: $Person, #Person.doSomething: (Person) -> () -> Void, $@convention(method) (@guaranteed Person) -> Void // user: %3
        %3 = apply %2(%1): $@convention(method) (@guaranteed Person) -> Void
        // 创建一个空元组作为返回值
        %4 = tuple() // user: %5
        // 返回空元组
        return %4: $() // id: %5
} // end sil function 'protocol witness for main.SomeProtocol.doSomething() -> () in conformance main.Person : main.SomeProtocol in main'

// 使用sil hidden指令定义Person类的drink()方法
// sil hidden:这表示该函数是隐藏的,即它不会在模块之外可见
// @main.Person.drink():这是函数的完全限定名称,指明了该函数属于main模块中的Person类的drink()方法
// -> ():表示函数的返回类型为空元组,即函数不返回任何值
// : $@convention(method):这部分指定了函数的调用约定,表明这是一个方法,在Swift中,方法调用与函数调用有一些不同之处,例如它们使用隐式的self参数
// (@guaranteed Person) -> ():这是函数的参数列表@guaranteed是一个生命周期修饰符,表示参数Person在整个函数调用期间都是有效的Person 是参数的类型,表示该方法在调用时需要一个 Person 实例
// Person.drink()
sil hidden @main.Person.drink() -> Void: $@convention(method) (@guaranteed Person) -> Void {
    // 定义基本块bb0,它接受一个Person实例作为参数
    // %0 "self"                                      // user: %1
    bb0(%0: $Person):
        // 调试信息,标记参数%0为"self"
        // implicit表示这是一个隐式的变量,它表示这个变量并不是由程序员显式创建的,而是由编译器生成
        debug_value %0: $Person, let, name "self", argno 1, implicit // id: %1
        // 创建一个空元组作为返回值
        %2 = tuple() // user: %3
        // 返回空元组
        return %2: $() // id: %3
} // end sil function 'main.Person.drink() -> ()'

// vtable
// Person中的方法
sil_vtable Person {
    #Person.name!getter: (Person) -> () -> String: @main.Person.name.getter: Swift.String // Person.name.getter
    #Person.name!setter: (Person) -> (String) -> Void: @main.Person.name.setter: Swift.String // Person.name.setter
    #Person.name!modify: (Person) -> () -> Void: @main.Person.name.modify: Swift.String // Person.name.modify
    #Person.doSomething: (Person) -> () -> Void: @main.Person.doSomething() -> Void // Person.doSomething()
    #Person.deinit!deallocator: @main.Person.__deallocating_deinit // Person.__deallocating_deinit
}

// witness_table
// hidden:表示这个协议见证表是隐藏的,即不对外部可见。
// Person: SomeProtocol:指明这个协议见证表是为Person类对SomeProtocol协议的遵循
// module main:指定了这个协议见证表所属的模块为main 
sil_witness_table hidden Person: SomeProtocol module main {
    method #SomeProtocol.doSomething: <Self where Self: SomeProtocol>(Self) -> () -> Void: @protocol witness for main.SomeProtocol.doSomething() in conformance main.Person: main.SomeProtocol in main // protocol witness for SomeProtocol.doSomething() in conformance Person
}

// 文件映射关系
// Mappings from '#fileID' to '#filePath':
//   'main/main.swift' => 'main.swift'

参考

  1. Swift Intermediate Language (SIL)
  2. GroffLattner-SILHighLevelIR

相关文章

网友评论

    本文标题:Swift编译之SIL(Swift Intermediate L

    本文链接:https://www.haomeiwen.com/subject/lgedwdtx.html