猿问

golang调度程序如何以及为什么在runtime/proc.go:execute中递归地运行

我试图分解 Go 调度程序的工作原理,我在runtime/proc.go中看到的是:

  1. schedule函数调用execute运行一个 goroutine

  2. 的注释execute明确指出该函数永远不会返回。它调用gogo汇编文件之一中定义的函数。

  3. gogo函数执行跳转到新 goroutine 的第一条指令的地址。

  4. 当这个 goroutine 完成后,schedule函数会被再次调用,所以我们回到了步骤 1。

如果我的理解是正确的,那么这个方案是如何避免堆栈溢出的呢?它与自动增加其大小的“无限”堆栈有关,还是我在这里遗漏了一些东西?


米琪卡哇伊
浏览 98回答 1
1回答

幕布斯6054654

所以我花了一些时间研究这个主题,现在可以尝试回答我自己的问题。整个 goroutine 生命周期变得更加复杂:新的 goroutine 在一个名为 的特殊 goroutine 中创建g0,它是线程的主 goroutine。任何对 的调用都会go func将堆栈从调用它的当前 Goroutine 更改为g0(这是在 中完成的proc.go:newproc)。当创建 goroutine 时(在 中),它的堆栈(和/或程序计数器,PC)以看起来像是由函数调用的proc.go:newproc1方式构建。这样做是为了保证当 goroutine 完成并返回时,它会返回到.goexitgoexit当schedule调用 并选择运行一个 goroutine 时,该execute函数将执行它(== 通过gogo汇编函数跳转到其地址)。Goroutine 完成后,它返回到goexit在汇编中实现的函数。该汇编函数调用proc.go:goexit1(不确定为什么需要汇编中的这个额外步骤)。该goexit1函数将当前堆栈更改为g0. 这是通过调用mcall(“机器线程调用”)来完成的,它执行参数中接收到的任何函数。在这种情况下,提供给的函数mcall是goexit0。在汇编中实现的 跳转到的堆栈帧 (SP)mcall的地址并执行to 。g0CALLgoexit0该goexit0函数在 的上下文中执行g0。它将一个已完成的 goroutine 放入空闲 goroutine 列表中,并释放其堆栈(如果之前已增加)。然后再次goexit0调用schedule,选择一个 goroutine 来运行,所以我们回到步骤 3。所以这里确实似乎没有递归。调度的 goroutine 本身从不调用schedule:这是由一个特殊的 goroutine 完成的g0。我仍然不确定我是否捕获了所有细节,因此欢迎评论和其他答案。
随时随地看视频慕课网APP

相关分类

Go
我要回答