iOS静/动态库开发支持bitcode

作者: Worthy | 来源:发表于2018-12-27 17:57 被阅读39次

bitcode是什么

Bitcode is an intermediate representation of a compiled program. Apps you upload to iTunes Connect that contain bitcode will be compiled and linked on the App Store. Including bitcode will allow Apple to re-optimize your app binary in the future without the need to submit a new version of your app to the store.

根据官方文档的解释,bitcode是被编译程序的一种中间形式的代码。包含bitcode配置的程序将会在App store上被编译和链接。bitcode允许苹果在后期重新优化程序的二进制文件,而不需要重新提交一个新的版本到App store上。
我们知道,iOS编译使用的是Clang/LLVM编译工具链。LLVM是编译器和工具链技术的集合,Clang 是LLVM的前端。C,C++ 和 Objective-C代码通过Clang编译生成中间代码LLVM IR,再经由后端生成目标平台上的本地字节码。
LLVM比较特别的地方在于LLVM IR,它把编译过程分为了三层。第一层支持多种语言作为输入,无论是OC,C++,Swift代码,最终转化为中间代码LLVM IR。第二层是共享优化器,专门对中间代码优化。第三层是不同目标平台字节码生成。这样的好处在于:想要增加增加一门新语言,只需要增加前端;想要支持新的CPU架构,只需要增加后端。

LLVM架构

bitcode本质上就是LLVM IR的字节流格式,我们可以尝试生成IR代码,看一看庐山真面目。首先,创建一个test.c:

 #include <stdio.h> 
 int main(int argc, char **argv) 
 { 
    printf("hello world\n"); 
    return 0; 
 }

调用clang指令-emit-llvm:

 clang test.c -S -emit-llvm -o test.bc

生成的test.bc代码如下:

; ModuleID = 'test.c'
source_filename = "test.c"
target datalayout = "e-m:o-i64:64-f80:128-n8:16:32:64-S128"
target triple = "x86_64-apple-macosx10.14.0"

@.str = private unnamed_addr constant [13 x i8] c"hello world\0A\00", align 1

; Function Attrs: noinline nounwind optnone ssp uwtable
define i32 @main(i32, i8**) #0 {
  %3 = alloca i32, align 4
  %4 = alloca i32, align 4
  %5 = alloca i8**, align 8
  store i32 0, i32* %3, align 4
  store i32 %0, i32* %4, align 4
  store i8** %1, i8*** %5, align 8
  %6 = call i32 (i8*, ...) @printf(i8* getelementptr inbounds ([13 x i8], [13 x i8]* @.str, i32 0, i32 0))
  ret i32 0
}

declare i32 @printf(i8*, ...) #1

attributes #0 = { noinline nounwind optnone ssp uwtable "correctly-rounded-divide-sqrt-fp-math"="false" "disable-tail-calls"="false" "less-precise-fpmad"="false" "no-frame-pointer-elim"="true" "no-frame-pointer-elim-non-leaf" "no-infs-fp-math"="false" "no-jump-tables"="false" "no-nans-fp-math"="false" "no-signed-zeros-fp-math"="false" "no-trapping-math"="false" "stack-protector-buffer-size"="8" "target-cpu"="penryn" "target-features"="+cx16,+fxsr,+mmx,+sahf,+sse,+sse2,+sse3,+sse4.1,+ssse3,+x87" "unsafe-fp-math"="false" "use-soft-float"="false" }
attributes #1 = { "correctly-rounded-divide-sqrt-fp-math"="false" "disable-tail-calls"="false" "less-precise-fpmad"="false" "no-frame-pointer-elim"="true" "no-frame-pointer-elim-non-leaf" "no-infs-fp-math"="false" "no-nans-fp-math"="false" "no-signed-zeros-fp-math"="false" "no-trapping-math"="false" "stack-protector-buffer-size"="8" "target-cpu"="penryn" "target-features"="+cx16,+fxsr,+mmx,+sahf,+sse,+sse2,+sse3,+sse4.1,+ssse3,+x87" "unsafe-fp-math"="false" "use-soft-float"="false" }

!llvm.module.flags = !{!0, !1}
!llvm.ident = !{!2}

!0 = !{i32 1, !"wchar_size", i32 4}
!1 = !{i32 7, !"PIC Level", i32 2}
!2 = !{!"Apple LLVM version 10.0.0 (clang-1000.11.45.5)"}

可以看到,生成的中间代码IR相比汇编代码更为直观易懂。下面简单总结一下bitcode的优缺点:
优点:Apple可以根据目标机型编译生成性能最优,包大小最小的可执行文件。当编译器更新时,Apple可以对提交包含bitcode的应用进行优化而不需要我们重新编译提交。
缺点:bitcode本质上是LLVM IR,相比汇编代码更容易被分析,安全性会下降。

如何支持bitcode

Framework支持bitcode就是在编译好的目标文件中增加一个__LLVM段用来存放编译生成的中间代码IR。应用支持bitcode的前提是项目中所有第三方库都支持bitcode,否则就只能关闭bitcode选项。
bitcode选项在xcode工程配置中默认是打开的,如果Framework是纯OC代码,只需要在工程选项Other C flags中增加-fembed-bitcode即可。如果Framework依赖了其他第三方库,需要所有第三方库也需要支持bitcode。假如第三方库使用clang编译,需要增加编译选项-fembed-bitcode
编译完成后,通过otool命令查看目标文件是否包含存放bitcode的__LLVM段:

otool -l xxx | grep __LLVM

如果已经包含bitcode,会有如下结果:

segname __LLVM
segname __LLVM

如何剥离bitcode

如果想要剥离掉bitcode,通过xcode提供的bitcode_strip命令也可以很方便的做到:

xcrun bitcode_strip input [ -r | -m | -l ] -o output

相关文章

网友评论

    本文标题:iOS静/动态库开发支持bitcode

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