Go的源码是Plan 9汇编与Go语言的混合使用,即有汇编调用Go,也有Go调用汇编。在很多情况下,直接看汇编代码更直观,特别是在研究编译器生成的函数实现时更有用。下面介绍两种反汇编方法:
假设有个helloworld.go程序,源码如下:
package main
import "fmt"
func main() {
fmt.Println("Hello, World!")
}
先将代码编译成二进制,命令如下:
go build --gcflags="-l -N" -o helloworld
--gcflags="-l -N"
反汇编成AT&T汇编
AT&T是Linux操作系统最常用的汇编语言,Linux系统提供的objdump工具也是将二进制反编译成AT&T汇编,命令如下:
objdump -d helloworld > att.asm
最终反汇编的结果放在了att.asm文件。下面是汇编结果(截取了部分代码):
0000000000487200 <main.main>:
487200: 64 48 8b 0c 25 f8 ff mov %fs:0xfffffffffffffff8,%rcx
487207: ff ff
487209: 48 3b 61 10 cmp 0x10(%rcx),%rsp
48720d: 76 7a jbe 487289 <main.main+0x89>
48720f: 48 83 ec 68 sub $0x68,%rsp
487213: 48 89 6c 24 60 mov %rbp,0x60(%rsp)
487218: 48 8d 6c 24 60 lea 0x60(%rsp),%rbp
48721d: 0f 57 c0 xorps %xmm0,%xmm0
487220: 0f 11 44 24 38 movups %xmm0,0x38(%rsp)
487225: 48 8d 44 24 38 lea 0x38(%rsp),%rax
...
反汇编成Plan 9汇编
Plan 9目前看到的只有Go语言在用,而且网上的资料很匮乏,学习成本挺高的,但如果学会了Plan 9汇编,再看Go源码会非常简单,而且可以通过Go调用汇编写出高效率代码。要把二进制反汇编成Plan 9汇编,也只能用Go提供的工具,命令如下:
go tool objdump -S helloworld > plan9.asm
最终反汇编的结果放在了plan9.asm文件。下面是汇编结果(截取了部分代码):
TEXT main.main(SB) /root/program/golang/helloworld/helloworld.go
func main() {
0x487200 64488b0c25f8ffffff MOVQ FS:0xfffffff8, CX
0x487209 483b6110 CMPQ 0x10(CX), SP
0x48720d 767a JBE 0x487289
0x48720f 4883ec68 SUBQ $0x68, SP
0x487213 48896c2460 MOVQ BP, 0x60(SP)
0x487218 488d6c2460 LEAQ 0x60(SP), BP
fmt.Println("Hello, World!")
0x48721d 0f57c0 XORPS X0, X0
0x487220 0f11442438 MOVUPS X0, 0x38(SP)
0x487225 488d442438 LEAQ 0x38(SP), AX
0x48722a 4889442430 MOVQ AX, 0x30(SP)
建议
从上面展示的结果来看,两种汇编还是有点差别的。在学习Go源码时,选择用哪种汇编去看这个要根据自已的实际情况来。我的建议先必须对AT&T汇编有一定了解,因为Plan 9汇编资料实在是太少了,甚至没有一个对命令的完整介绍,一个没有汇编基础的人直接看Plan 9的一些文档,绝对会一脸绝望。在整理网上一些Plan 9汇编资料时,发现都是讲的Plan 9与AT&T的差异,所以只有了解AT&T,才能进一步学好Plan 9汇编。