1
0
mirror of https://github.com/chai2010/advanced-go-programming-book.git synced 2025-05-24 12:32:21 +00:00
2018-06-21 21:01:23 +08:00

12 KiB
Raw Blame History

3.1. 快速入门

在第一章的“Hello, World 的革命”一节中我们已经见过一个Go汇编程序。本节我们将通过分析简单的Go程序输出的汇编代码然后照猫画虎用汇编实现一个简单的输出程序。

实现和声明

Go汇编语言并不是一个独立的语言主要原因是因为Go汇编程序无法独立使用。Go汇编代码必须以Go包的方式被组织同时包中至少要有一个Go语言文件。如果Go汇编代码中定义的变量和函数要被其它Go语言代码引用还需要通过Go语言代码将汇编中定义的符号声明出来。用于变量的定义和函数的定义Go汇编文件类似于C语言中的.c文件。而用于导出汇编中定义符号的Go源文件类似于C语言的.h文件。

定义整数变量

为了简单我们先用Go语言定义并赋值一个整数变量然后查看生成的汇编代码。

创建pkg.go文件内容如下

package pkg

var Id = 9527

然后用以下命令查看的Go语言程序对应的伪汇编代码

$ go tool compile -S pkg.go
"".Id SNOPTRDATA size=8
        0x0000 37 25 00 00 00 00 00 00                          '.......

输出的汇编比较简单,其中"".Id对应Id变量符号变量的内存大小为8个字节。变量的初始化内容为37 25 00 00 00 00 00 00对应十六进制格式的0x2537对应十进制为9527。SNOPTRDATA是相关的标志暂时忽略。

以上的内容只是目标文件对应的汇编和Go汇编语言虽然相似当并不完全等价。Go语言官网自带了一个Go汇编语言的入门教程地址在https://golang.org/doc/asm

Go汇编语言提供了DATA命令用于初始化变量DATA命令的语法如下

DATA symbol+offset(SB)/width, value

其中symbol为变量在汇编语言中对应的符号offset是符号开始地址的偏移量width是要初始化内存的宽度大小value是要初始化的那天。其中当前包中Go语言定义的符号symbol在汇编代码中对应·symbol其中·为一个特殊的unicode符号。

采用以下命令可以给Id变量初始化为十六进制的0x2537对应十进制的9527常量需要以美元符号$开头表示:

DATA ·Id+0(SB)/1,$0x37
DATA ·Id+1(SB)/1,$0x25

变量定义好之后需要导出以共其它代码引用。Go汇编语言提供了GLOBL命令用于将符号导出

GLOBL symbol(SB), width

其中symbol对应汇编中符号的名字width为符号对应内存的大小。用以下命令将汇编中的·Id变量导出

GLOBL ·Id, $8

现在已经出版完成了用汇编定义一个整数变量的工作。

为了便于其它包使用该Id变量我们还需要在Go代码中声明该变量同时也给变量指定一个合适的类型。修改pkg.go的内容如下

package pkg

var Id int

表示声明一个int类型的Id变量。因为该变量已经在汇编中定义因此Go语言部分只是声明变量声明的变量不能含义初始化的操作。

完整的汇编代码在pkg_amd64.s中

GLOBL ·Id(SB),$8

DATA ·Id+0(SB)/1,$0x37
DATA ·Id+1(SB)/1,$0x25
DATA ·Id+2(SB)/1,$0x00
DATA ·Id+3(SB)/1,$0x00
DATA ·Id+4(SB)/1,$0x00
DATA ·Id+5(SB)/1,$0x00
DATA ·Id+6(SB)/1,$0x00
DATA ·Id+7(SB)/1,$0x00

文件名pkg_amd64.s表示为AMD64环境下的汇编代码文件。

虽然pkg包改用汇编实现但是用法和之前完全一样

package main

import pkg "pkg包的路径"

func main() {
	println(pkg.Id)
}

对于Go包的用户来说用Go汇编语言或Go语言实现并无区别。

定义字符串变量

在前一个例子中,我们通过汇编定义了一个整数变量。现在我们尝试通过汇编定义一个字符串变量。

虽然从Go语言角度看定义字符串和整数变量的写法基本相同但是字符串底层却有着比单个整数更复杂的数据结构。

创建pkg.go文件内容如下

package pkg

var Name = "gopher"

然后用以下命令查看的Go语言程序对应的伪汇编代码

$ go tool compile -S pkg.go
go.string."gopher" SRODATA dupok size=6
        0x0000 67 6f 70 68 65 72                                gopher
"".Name SDATA size=16
        0x0000 00 00 00 00 00 00 00 00 06 00 00 00 00 00 00 00  ................
        rel 0+8 t=1 go.string."gopher"+0

输出中出现了一个新的符号go.string."gopher",根据其长度和内容分析可以猜测是对应底层的"gopher"字符串数据。因为Go语言的字符串并不是值类型Go字符串只是一种只读的引用类型。假设多个代码中出现了相同的"gopher"字符串时程序链接后其实都是引用的同一个符号go.string."gopher"。因此该符号有一个SRODATA标志表示这个数据在只读内存段dupok表示出现多个相同符号时只保留一个就可以了。

而真正的Go字符串变量Name对应的大小却只有16个字节了。其实Name变量并没有直接对应“gopher”字符串而是对应reflect.StringHeader结构体

type reflect.StringHeader struct {
	Data uintptr
	Len  int
}

从汇编角度看Name变量其实对应的是reflect.StringHeader结构体类型。前8个字节对应底层真实字符串数据的指针也就是符号go.string."gopher"对应的地址。后8个字节对应底层真实字符串数据的有效长度这里是6个字节。

创建pkg_amd64.s文件我们尝试通过汇编代码重新定义并初始化Name字符串

GLOBL ·NameData(SB),$8
DATA  ·NameData(SB)/8,$"gopher"

GLOBL ·Name(SB),$16
DATA  ·Name+0(SB)/8,$·NameData(SB)
DATA  ·Name+8(SB)/8,$6

因为在Go汇编语言中go.string."gopher"不是一个合法的符号我们无法手工创建这是给编译器保留的部分特权因为手工创建类似符号可能打破编译器输出代码的某些规则。因此我们新创建了一个·NameData符号表示底层的字符串数据。

然后定义·Name符号为两个16字节其中前8个字节用·NameData符号对应的地址初始化后8个字节为常量6表示字符串长度。

通过以下代码测试输出Name变量

package main

import pkg "pkg包的路径"

func main() {
	println(pkg.Name)
}

在运行时将会产生类似以下错误:

pkgpath.NameData: missing Go //type information for global symbol: size 8

提示汇编中定义的NameData符号没有类型信息。其实Go汇编语言中定义的数据并没有所谓的类型每个符号只不过是对应一个内存而且。出现这种错误的原因是Go语言的垃圾回收器在扫描NameData变量的时候无法知晓该变量内部是否包含指针。因此真正错误的原因并不是NameData没有类型二是NameData变量没有标注是否会含有指针信息。

通过给NameData变量增加一个标志表示其中不会包含指针数据可以修复该错误

#include "textflag.h"

GLOBL ·NameData(SB),NOPTR,$8

通过给·NameData增加NOPTR表示其中不含指针数据。那么垃圾回收器在遇到该变量的时候就会停止内部数据的扫描。

我们也可以通过给·NameData变量在Go语言中增加一个不含指针并且大小为8个字节的类型来修改该错误

package pkg

var NameData [8]byte
var Name string

我们将NameData声明为长度为8的字节数组。因为编译器可以通过类型分析出该变量不会包含指针因此汇编代码中可以NOPTR标志信息。

在这个实现中Name字符串底层其实引用的是NameData内存对应的“gopher”字符串数据。因此如果NameData发生变化的化Name字符串的数据也会跟着变化的。

func main() {
	println(pkg.Name)

	pkg.NameData[0] = '?'
	println(pkg.Name)
}

当然这和字符串的只读定义是冲突的正常的代码需要避免出现这种情况。最好的方法是不要导出内部的NameData变量这样可以避免内部数据被无意破坏。

在用汇编定义字符串时我们完全换一种思维将底层的字符串数据和字符串头结构体定义在一起这样可以避免引入NameData符号

GLOBL ·Name(SB),$24

DATA ·Name+0(SB)/8,$·Name+16(SB)
DATA ·Name+8(SB)/8,$6
DATA ·Name+16(SB)/8,$"gopher"

在新的结构中Name符号对应的内存从16字节变为24字节多出的8个字节用户存放底层的“gopher”字符串。·Name符号前16个字节依然对应reflect.StringHeader结构体Data部分对应$·Name+16(SB)表示数据的地址为Name符号往后偏移16个字节的位置Len部分依然对应6个字节的长度。

定义main函数

前面的例子已经展示了如何通过汇编定义整型和字符串类型变量。我们现在将尝试用汇编实现函数,然后输出一个字符串。

先创建main.go文件创建并初始化字符串变量同时声明main函数

package main

var helloworld = "你好, 世界"

func main()

然后创建main_amd64.s文件里面对应main函数的实现

TEXT ·main(SB), $16-0
	MOVQ ·helloworld+0(SB), AX; MOVQ AX, 0(SP)
	MOVQ ·helloworld+8(SB), BX; MOVQ BX, 8(SP)
	CALL runtime·printstring(SB)
	CALL runtime·printnl(SB)
	RET

TEXT ·main(SB), $16-0用于定义main函数,其中$16-0表示main函数的帧大小是16个字节对应string头的大小用于给runtime·printstring函数传递参数),0表示main函数没有参数和返回值。main函数内部通过调用运行时内部的runtime·printstring(SB)函数来打印字符串。然后调用runtime·printnl打印换行符号。

Go语言函数在函数调用时完全通过栈传递调用参数和返回值。先通过MOVQ指令将helloworld对应的字符串头部结构体的16个字节复制到栈指针SP对应的16字节的空间然后通过CALL指令调用对应函数。最后使用RET指令表示当前函数返回。

特殊字符

Go语言函数或方法符号在编译为目标文件后目标文件中的每个符号均包含对应包的绝对导入路径。因此目标文件的符号可能非常复杂比如“path/to/pkg.(*SomeType).SomeMethod”或“go.string."abc"”。目标文件的符号名中不仅仅包含普通的字母还可能包含诸多特殊字符。而Go语言的汇编器是从plan9移植过来的二把刀并不能处理这些特殊的字符导致了用Go汇编语言手工实现Go诸多特性时遇到种种限制。

Go汇编语言同样遵循Go语言少即是多的哲学它只保留了最基本的特性定义变量和全局函数。同时为了简化Go汇编器的词法扫描程序的实现特别引入了Unicode中的中点·和大写的除法/对应的Unicode码点为U+00B7U+2215。汇编器编译后,中点·会被替换为ASCII中的点“.”大写点除法会被替换为ASCII码中的除法“/”,比如math/rand·Int会被替换为math/rand.Int。这样可以将点和浮点数中的小数点、大写的除法和表达式中的除法符号分开,可以简化汇编程序此法分析部分的实现。

即使暂时抛开Go汇编语言设计取舍的问题中点·和除法/两个字符的如何输入就是一个挑战。这两个字符在 https://golang.org/doc/asm 文档中均有描述,因此直接从该页面复制是最简单可靠的方式。

如果是macOS系统则有以下几种方法输入中点·:在不开输入法时,可直接用 option+shift+9 输入;如果是自带的简体拼音输入法,输入左上角~键对应·如果是自带的Unicode输入法则可以输入对应的Unicode码点。

没有分号

Go汇编语言中分号可以用于分隔同一行内的多个语句。下面是用分号混乱排版的汇编代码

TEXT ·main(SB), $16-0; MOVQ ·helloworld+0(SB), AX; MOVQ ·helloworld+8(SB), BX;
MOVQ AX, 0(SP);MOVQ BX, 8(SP);CALL runtime·printstring(SB);
CALL runtime·printnl(SB);
RET;

和Go语言一样也可以省略行尾的分号。当遇到末尾时汇编器会自动插入分号。下面是省略分号后的代码

TEXT ·main(SB), $16-0
	MOVQ ·helloworld+0(SB), AX; MOVQ AX, 0(SP)
	MOVQ ·helloworld+8(SB), BX; MOVQ BX, 8(SP)
	CALL runtime·printstring(SB)
	CALL runtime·printnl(SB)
	RET

和Go语言一样语句之间多个连续的空白字符和一个空格是等价的。