Golang源码之初始化流程

在用Go编写应用程序的时候,可以认为main.main是整个应用程序的入口,但站在整个Go程序的角度来看,却并非如此。在main.main函数之前,Go底层已经做了大量的初始化工作,下面开始从程序真正的入口开始了解下初始化前的工作。

入口

通过GDB可以找到程序入口:

go build -gcflags "-N -l" -o test test.go

info files会列出程序的入口地址:

(gdb) info files
Symbols from "/home/sandydu/program/golang/test".
Local exec file:
        `/home/sandydu/program/golang/test', file type elf64-x86-64.
        Entry point: 0x452890

Entry point: 0x452890这个就是我们要打的入口地址,直接对这个地址打断点,GDB就会自动列出断点的具体位置:

 (gdb) b *0x452890
Breakpoint 1 at 0x452890: file /usr/local/go/src/runtime/rt0_linux_amd64.s, line 8.

rt0_linux_amd64.s文件的第8行就是真正的入口地址,看下详细代码:

@(src/runtime/rt0_linux_amd64.s:8)

TEXT _rt0_arm64_linux(SB),NOSPLIT|NOFRAME,$0
	MOVD	0(RSP), R0	// argc
	ADD	    $8, RSP, R1	// argv
	BL	main(SB)

_rt0_arm64_linux的工作就是把命令行参数argc、argv放到寄存器,然后跳转到本文件的main继续执行,代码如下:

@(src/runtime/rt0_linux_amd64.s:98)

TEXT main(SB),NOSPLIT|NOFRAME,$0
	MOVD	$runtime·rt0_go(SB), R2
	BL	(R2)
exit:
	MOVD $0, R0
	MOVD	$94, R8	// sys_exit
	SVC
	B	exit

再继续跳转到runtime·rt0_go执行(没搞懂为啥要这样跳来跳去,直接点不好嘛。。。)

runtime·rt0_go开始进入初始化的流程,这个方法主要工作如下:

  • 处理命令行参数,将argc,argv放入栈。
  • 获取CPU相关信息
  • 将g0(0号goroutine)存入TLS,将m0(0号线程)存入TLS
  • 调用runtime.args,去 stack 里读取参数和环境变量
  • 调用runtime.osinit,获取CPU核数
  • 调用runtime.schedinit,这个函数功能很多,主要初始化调度相关的信息,后面详细介绍
  • 调用runtime.newproc,创建一个新的协程,并执行runtime.main,这个函数会调用我们熟悉的main.init和main.main
  • 创建 一个m,并调用schedule开始进入调度状态,整个程序开始run起来
TEXT runtime·rt0_go(SB),NOSPLIT,$0
	// 将参数argc、argv放到堆栈上
	MOVQ	DI, AX		// argc
	MOVQ	SI, BX		// argv
	SUBQ	$(4*8+7), SP		// 2args 2auto
	ANDQ	$~15, SP
	MOVQ	AX, 16(SP)
	MOVQ	BX, 24(SP)

	// 初始化g0的stackguard和stack
	// g0.stackguard0 = (-64*1024+104)(SP)
	// g0.stackguard1 = (-64*1024+104)(SP)
	// g0.stack.l0 = (-64*1024+104)(SP)
	// g0.stack.hi = SP
	MOVQ	$runtime·g0(SB), DI
	LEAQ	(-64*1024+104)(SP), BX
	MOVQ	BX, g_stackguard0(DI)
	MOVQ	BX, g_stackguard1(DI)
	MOVQ	BX, (g_stack+stack_lo)(DI)
	MOVQ	SP, (g_stack+stack_hi)(DI)

	// 通过CPUID指令获取CPU信息
	// 详见https://c9x.me/x86/html/file_module_x86_id_45.html
	MOVL	$0, AX
	CPUID
	MOVL	AX, SI
	CMPL	AX, $0
	JE	nocpuinfo

	// Figure out how to serialize RDTSC.
	// On Intel processors LFENCE is enough. AMD requires MFENCE.
	// Don't know about the rest, so let's do MFENCE.
	CMPL	BX, $0x756E6547  // "Genu"
	JNE	notintel
	CMPL	DX, $0x49656E69  // "ineI"
	JNE	notintel
	CMPL	CX, $0x6C65746E  // "ntel"
	JNE	notintel
	MOVB	$1, runtime·isIntel(SB)
	MOVB	$1, runtime·lfenceBeforeRdtsc(SB)
notintel:

	// Load EAX=1 cpuid flags
	MOVL	$1, AX
	CPUID
	MOVL	AX, runtime·processorVersionInfo(SB)

nocpuinfo:
	// 如果启用了cgo,就执行_cgo_init函数
	MOVQ	_cgo_init(SB), AX
	TESTQ	AX, AX
	JZ	needtls
	// g0 already in DI
	MOVQ	DI, CX	// Win64 uses CX for first parameter
	MOVQ	$setg_gcc<>(SB), SI
	CALL	AX

	// 重新更新g0的g0的stackguard和stack
	MOVQ	$runtime·g0(SB), CX
	MOVQ	(g_stack+stack_lo)(CX), AX
	ADDQ	$const__StackGuard, AX
	MOVQ	AX, g_stackguard0(CX)
	MOVQ	AX, g_stackguard1(CX)
	
needtls:
	// 初始化TLS
	LEAQ	runtime·m0+m_tls(SB), DI
	CALL	runtime·settls(SB)

	// 验证TLS是否初始化成功
	get_tls(BX)
	MOVQ	$0x123, g(BX)
	MOVQ	runtime·m0+m_tls(SB), AX
	CMPQ	AX, $0x123
	JEQ 2(PC)
	CALL	runtime·abort(SB)
ok:
	// 把g0放入TLS,后面可以通过getg函数找到了
	get_tls(BX)
	LEAQ	runtime·g0(SB), CX
	MOVQ	CX, g(BX)
	LEAQ	runtime·m0(SB), AX

	// 将g0存到m0中,m->g0 = g0
	MOVQ	CX, m_g0(AX)
	// 将m0存到g0中,g0->m = m0
	MOVQ	AX, g_m(CX)

	CLD				// convention is D is always left cleared
	CALL	runtime·check(SB)

	// 处理参数
	MOVL	16(SP), AX		// copy argc
	MOVL	AX, 0(SP)
	MOVQ	24(SP), AX		// copy argv
	MOVQ	AX, 8(SP)
	CALL	runtime·args(SB)

	// 初始化OS:获取CPU数量
	CALL	runtime·osinit(SB)
	
	// 初始化调度,非常重要的函数,后续详解
	CALL	runtime·schedinit(SB)

	// 创建一个goruntine并执行,执行函数为runtime.main
	// 即: go runtime.main()
	MOVQ	$runtime·mainPC(SB), AX		// entry
	PUSHQ	AX
	PUSHQ	$0			// arg size
	CALL	runtime·newproc(SB)
	POPQ	AX
	POPQ	AX

	// 启动这个m,即m0,里面会调用schedule函数进入调度状态
	CALL	runtime·mstart(SB)

	CALL	runtime·abort(SB)	// mstart should never return
	RET

	MOVQ	$runtime·debugCallV1(SB), AX
	RET

DATA	runtime·mainPC+0(SB)/8,$runtime·main(SB)
GLOBL	runtime·mainPC(SB),RODATA,$8
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值