通过汇编看golang函数的多返回值

golang这门语言,有个比较好的特性,就是支持函数的多返回值。想C,C++,Java等这些语言,是不支持函数多返回的。但是C,C++可以使用传递指针,实现函数多返回。但是,你有没有想过,golang是怎样实现函数多返回值的呢?
我们知道,C,C++是通过寄存器实现函数返回值的,也就是先把返回值写入到一个寄存器中,然后再从寄存器中,读到函数的返回值。golang也是这样实现的吗?

伟大的思想家孔子曾说过,在源码面前一切都如同裸奔。后来,鲁迅先生,总结了孔子的思想,说出了,在汇编面前,一切语法都是纸老虎。

下面我们通过golang的汇编指令,来看一下golang是怎样实现函数的多返回值的

在看汇编之前,我们先用go的debug函数看下函数的栈信息
代码很简单,不用解释了

package main

import (
    "fmt"
    "runtime/debug"
)

func main() {
    one(3)
}

func one(a int) (int, int) {
    fmt.Println(string(debug.Stack()))
    return a, a + 5
}


我标红的这一行,就是one 函数的栈信息,第一个参数 0x3 很好理解,就是我们传入的参数3, 但是后面这两个是啥?还有,我明明只传了一个参数,为啥会传入三个参数?

到这里,我也就不卖关子了,直接说了,后面这两个参数,就是one函数返回值的地址,也就是说,one函数返回值地址不在one函数中,而是在调用one函数的mian函数中。golang的函数返回值,和C,C++的不同,golang的返回值是通过栈内地址实现的(返回值的地址是由函数调用者提供)。

package main

func main() {
    var b, c *int
    one(3, b, c)
}

func one(a int, b, c *int) {
}

也就是说,刚开始的那段代码,和这段在功能实现上,没有什么差别,只是golang编译器提供的一个语法糖。

下面通过汇编来看一下
这次我们不是对深入分析golang的汇编,只是从汇编层面,验证我们之前结论(golang函数多返回问题)
所以,不会死磕plan9汇编语法,说实话,plan9的很多知识我也不懂,大学没开过汇编的课程,这些东西都是因为兴趣自学的。

golang用的是plan9汇编,看plan9之前,先了解一下plan9的几个概念

go汇编中有4个伪寄存器

  • FP: Frame pointer,指向栈底位置,一般用来引用函数的输入参数,用来访问函数的参数
  • PC: Program counter: 程序计数器,用于分支和跳转
  • SB: Static base pointer: 一般用于声明函数或者全局变量
  • SP: Stack pointer:指向当前栈帧的局部变量的开始位置(栈顶位置),一般用来引用函数的局部变量

我们用这段代码进行汇编

package main

func main() {
    one(3)
}

func one(a int) (int, int) {
    return a, a + 5
}

使用 go tool compile -N -l -S main.go 得到汇编代码

"".main STEXT nosplit size=2 args=0x0 locals=0x0
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:3)      TEXT    "".main(SB), NOSPLIT|ABIInternal, $0-0
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:3)      FUNCDATA        $0, gclocals·33cdeccccebe80329f1fdbee7f5874cb(SB)
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:3)      FUNCDATA        $1, gclocals·33cdeccccebe80329f1fdbee7f5874cb(SB)
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:3)      FUNCDATA        $3, gclocals·33cdeccccebe80329f1fdbee7f5874cb(SB)
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:4)      PCDATA  $2, $0
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:4)      PCDATA  $0, $0
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:4)      XCHGL   AX, AX
        0x0001 00001 (<unknown line number>)    RET
        0x0000 90 c3                                            ..
"".one STEXT nosplit size=20 args=0x18 locals=0x0
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:7)      TEXT    "".one(SB), NOSPLIT|ABIInternal, $0-24
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:7)      FUNCDATA        $0, gclocals·33cdeccccebe80329f1fdbee7f5874cb(SB)
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:7)      FUNCDATA        $1, gclocals·33cdeccccebe80329f1fdbee7f5874cb(SB)
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:7)      FUNCDATA        $3, gclocals·33cdeccccebe80329f1fdbee7f5874cb(SB)
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:8)      PCDATA  $2, $0
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:8)      PCDATA  $0, $0
        0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:8)      MOVQ    "".a+8(SP), AX
        0x0005 00005 (C:\Users\bruce\Desktop\go\main.go:8)      MOVQ    AX, "".~r1+16(SP)
        0x000a 00010 (C:\Users\bruce\Desktop\go\main.go:8)      ADDQ    $5, AX
        0x000e 00014 (C:\Users\bruce\Desktop\go\main.go:8)      MOVQ    AX, "".~r2+24(SP)
        0x0013 00019 (C:\Users\bruce\Desktop\go\main.go:8)      RET
        0x0000 48 8b 44 24 08 48 89 44 24 10 48 83 c0 05 48 89  H.D$.H.D$.H...H.
        0x0010 44 24 18 c3                                      D$..

我只截取了和main,one函数相关的部分
TEXT "".one(SB), NOSPLIT|ABIInternal, $0-24这行最后, $0-24 的含义,0代表one函数的栈帧大小(局部变量+可能需要的额外调用函数的参数空间的总大小),因为one函数中没有额外开销,所有大小是0,24是传入参数和返回值的大小,单位是字节。传入的参数和返回值都是int,在64位机器上,大小是8个字节,64位。


简单画一下栈的示意图

看一下这句0x0000 00000 (C:\Users\bruce\Desktop\go\main.go:8) MOVQ "".a+8(SP), AX
SP寄存器指向的是栈顶的位置,AX 是一个通用寄存器
MOVQ指令 把 参数a 也就是(SP+8)的值搬到AX中


0x0005 00005 (C:\Users\bruce\Desktop\go\main.go:8) MOVQ AX, "".~r1+16(SP)
同样,把AX中的值搬到r1(返回值b)


0x000a 00010 (C:\Users\bruce\Desktop\go\main.go:8) ADDQ $5, AX
ADDQ指令把AX值+5

0x000e 00014 (C:\Users\bruce\Desktop\go\main.go:8) MOVQ AX, "".~r2+24(SP)
最后把AX的值搬到r2(返回值c)

0x0013 00019 (C:\Users\bruce\Desktop\go\main.go:8) RET
最后RET指令,one函数结束

总结

通过对golang进行汇编,真实了之前的结论

golang函数的多返回值不是通过寄存器传递,使用过使用调用值提供的地址,赋值实现的

先写这些吧,我也是刚接触golang的汇编,文中如有不正确的地方,还请在评论区指出

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 156,265评论 4 359
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,274评论 1 288
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 106,087评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,479评论 0 203
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 51,782评论 3 285
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,218评论 1 207
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,594评论 2 309
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,316评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 33,955评论 1 237
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,274评论 2 240
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,803评论 1 255
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,177评论 2 250
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,732评论 3 229
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 25,953评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,687评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,263评论 2 267
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,189评论 2 258