012013
 

安装Go语言,建议通过源码安装。通过源码安装Go,一般只需要安装一个GCC编译器就可以了。(Windows下是安装MinGW)。该文分析通过源码安装Go语言的过程。

一、分析安装脚本

《探究Go中各个目录的功能》一文中提到了几个脚本,以及它们的作用。现在来分析这些脚本都做了些什么。(以linux下为例,Windows对应的脚本类似)

1、all.bash

set -e
if [ ! -f make.bash ]; then
	echo 'all.bash must be run from $GOROOT/src' 1>&2
	exit 1
fi
OLDPATH="$PATH"
. ./make.bash --no-banner
bash run.bash --no-rebuild
PATH="$OLDPATH"
$GOTOOLDIR/dist banner  # print build info

说明:

1)set -e 当脚本中某个命令返回非零退出状态时,脚本退出
2)if语句 是要求all.bash必须在make.bash所在的目录运行(也就是$GOROOT/src)
3)OLDPATH=”$PATH” 保存当前PATH
4)执行make.bash并传递–no-banner参数
5)执行run.bash并传递–no-rebuild参数
6)执行dist工具并传递 banner参数,打印build信息

2、make.bash

在make.bash中,有一些环境变量,执行make.bash过程中会用到。

①GOROOT_FINAL:最终被设置的Go root,在编译过程中,默认是Go tree的位置。

②GOHOSTARCH:为主机工具(编译器和二进制文件)指定体系结构。这种类型的二进制文件在当前系统必须是可执行的,因此设置这个环境变量的目前唯一常见原因是在AMD64机器上设置GOHOSTARCH=386。(也就是说,在ADM64上可以运行386的可执行文件)

③GO_GCFLAGS:当构建包和命令时可以通过该环境变量附带上5g/6g/8g的参数

④GO_LDFLAGS:当构建包和命令时可以通过该环境变量附带上5l/6l/8l的参数

⑤CGO_ENABLED:在构建过程中控制cgo的使用。当设置为1,在构建时,会包含所有cgo相关的文件,如带有”cgo”编译指令的.c和.go文件。当设置为0,则忽略它们(即禁用CGO)

在文件开头是一些检查:比如是否在Windows下运行了make.bash,ld的版本等。

make.bash中接下来主要的工作(也就是开始构建):

1)构建 C 引导工具 —— cmd/dist

这里首先会export GOROOT环境变量,它的值就是go源码所在路径,可见,源码安装之前并不要求一定要设置GOROOT。

这里学习一个shell脚本知识

GOROOT_FINAL=”${GOROOT_FINAL:-$GOROOT}”
这叫做参数替换,形式如下:
${parameter-default},${parameter:-default}
意思是:如果 parameter 没被 set,那么就使用 default。这两种形式大部分时候是相同的。区别是:当parameter被设置了且为空,则第一种不能输出默认值,而第二种可以。

所以,GOROOT_FINAL=”${GOROOT_FINAL:-$GOROOT}”的意思就是,如果GOROOT_FINAL没设置,则GOROOT_FINAL=$GOROOT

通过gcc编译cmd/dist

2)构建 编译器和Go引导工具

首先通过执行dist设置需要的环境变量
eval $(./cmd/dist/dist env -p)

接着构建Go引导工具:go_bootstrap,以及编译器等

3)构建 包和命令工具

这是通过go_bootstrap做的

3、run.bash

该脚本是一个测试脚本,运行标准库中的测试用例。默认情况下会重新构建go包和工具。由于make.bash会做构建的工作,all.bash中执行run.bash时,传递了–no-rebuild

二、源码安装说明

源码安装Go语言,一般只需要执行./all.bash就可以了。(Windows上执行all.bat)

根据上面的分析,这样会运行测试脚本。如果想更快的安装Go,可以直接运行make.bash(Windows上是make.bat)

整个安装过程大概如下:

# Building C bootstrap tool.
cmd/dist

# Building compilers and Go bootstrap tool for host, linux/amd64.
lib9
libbio
libmach
misc/pprof
……
pkg/text/template/parse
pkg/text/template
pkg/go/doc
pkg/go/build
cmd/go

# Building packages and commands for linux/amd64.
runtime
errors
sync/atomic
sync
io
……
testing
testing/iotest
testing/quick

# Testing packages.
ok cmd/api 0.031s
? cmd/cgo [no test files]
ok cmd/fix 3.558s
ok cmd/go 0.022s
……
? unsafe [no test files]

real 3m6.056s
user 2m29.517s
sys 0m25.134s
# GOMAXPROCS=2 runtime -cpu=1,2,4
ok runtime 6.605s

# sync -cpu=10
ok sync 0.428s

# Testing race detector.
ok flag 1.044s

# ../misc/cgo/stdio

# ../misc/cgo/life

# ../misc/cgo/test
scatter = 0×430490
hello from C
PASS
ok _/home/polaris/go/misc/cgo/test 1.137s

# ../misc/cgo/testso

# ../doc/progs

real 0m19.110s
user 0m15.341s
sys 0m2.904s

# ../doc/articles/wiki
run.bash: 行 92: make: 未找到命令
PASS

# ../doc/codewalk

# ../misc/dashboard/builder ../misc/goplay

# ../test/bench/shootout
fasta
reverse-complement
nbody
binary-tree
binary-tree-freelist
fannkuch
fannkuch-parallel
regex-dna
regex-dna-parallel
spectral-norm
k-nucleotide
k-nucleotide-parallel
mandelbrot
meteor-contest
pidigits
threadring
chameneosredux

# ../test/bench/go1
ok _/home/polaris/go/test/bench/go1 4.942s

# ../test

real 1m38.036s
user 1m14.701s
sys 0m16.645s

# Checking API compatibility.
+pkg crypto/x509, const PEMCipher3DES PEMCipher
+pkg crypto/x509, const PEMCipherAES128 PEMCipher
+pkg crypto/x509, const PEMCipherAES192 PEMCipher
+pkg crypto/x509, const PEMCipherAES256 PEMCipher
+pkg crypto/x509, const PEMCipherDES PEMCipher
+pkg crypto/x509, func EncryptPEMBlock(io.Reader, string, []byte, []byte, PEMCipher)
……
+pkg reflect, func SliceOf(Type) Type
+pkg regexp, method (*Regexp) Split(string, int) []string
~pkg text/template/parse, type DotNode bool
~pkg text/template/parse, type Node interface { Copy, String, Type }

ALL TESTS PASSED


Installed Go for linux/amd64 in /home/polaris/go
Installed commands in /home/polaris/go/bin
*** You need to add /home/polaris/go/bin to your PATH.

三、带中文注释的脚本

以下是我加上了注释的make.bash脚本(关键部分)

echo '# Building C bootstrap tool.'
echo cmd/dist
# export当前Go源码所在跟目录为GOROOT
export GOROOT="$(cd .. && pwd)"
# 如果GOROOT_FINAL没有设置,则使用$GOROOT的值当做GOROOT_FINAL
GOROOT_FINAL="${GOROOT_FINAL:-$GOROOT}"
DEFGOROOT='-DGOROOT_FINAL="'"$GOROOT_FINAL"'"'

# 如果在amd64机子上编译Go本身为32位,可以设置 $GOHOSTARCH=386。不建议这么做
mflag=""
case "$GOHOSTARCH" in
386) mflag=-m32;;
amd64) mflag=-m64;;
esac

# gcc编译:编译cmd/dist下所有的c文件
# -m:指定处理器架构,以便进行优化(-m32、-m64)或为空(一般为空)
# -O:优化选项,一般为:-O2。优化得到的程序比没优化的要小,执行速度可能也有所提高
# -Wall:生成所有警告信息
# -Werror:所有警告信息都变成错误
# -ggdb:为gdb生成调试信息(-g是生成调试信息)
# -o:生成指定的输出文件
# -I:指定额外的文件搜索路径
# -D:相当于C语言中的#define GOROOT_FINAL="$GOROOT_FINAL"
gcc $mflag -O2 -Wall -Werror -ggdb -o cmd/dist/dist -Icmd/dist "$DEFGOROOT" cmd/dist/*.c

# 编译完 dist 工具后,运行dist。目的是设置相关环境变量
# 比如:$GOTOOLDIR 环境变量就是这里设置的
eval $(./cmd/dist/dist env -p)
echo

# 运行make.bash时传递--dist-tool参数可以仅编译dist工具
if [ "$1" = "--dist-tool" ]; then
	# Stop after building dist tool.
	mkdir -p "$GOTOOLDIR"
	if [ "$2" != "" ]; then
		cp cmd/dist/dist "$2"
	fi
	mv cmd/dist/dist "$GOTOOLDIR"/dist
	exit 0
fi

# 构建 编译器和Go引导工具
# $GOHOSTOS/$GOHOSTARCH 是运行dist设置的
echo "# Building compilers and Go bootstrap tool for host, $GOHOSTOS/$GOHOSTARCH."
# 表示重新构建
buildall="-a"
# 传递--no-clean 表示不重新构建
if [ "$1" = "--no-clean" ]; then
	buildall=""
fi
# 构建Go引导工具
./cmd/dist/dist bootstrap $buildall -v # builds go_bootstrap
# Delay move of dist tool to now, because bootstrap may clear tool directory.
mv cmd/dist/dist "$GOTOOLDIR"/dist
"$GOTOOLDIR"/go_bootstrap clean -i std
echo

# $GOHOSTARCH 与 $GOARCH的区别:($GOHOSTOS 与 $GOOS的区别一样)
#	GOARCH 表示Go写出来的程序会在什么架构运行(目标操作系统);
#	GOHOSTARCH 表示运行make.bash这个脚本的系统架构
# 一般它们是相等的,只有在需要交叉编译时才会不一样。
if [ "$GOHOSTARCH" != "$GOARCH" -o "$GOHOSTOS" != "$GOOS" ]; then
	# 即使交叉编译,本机的Go环境还是必须得有
	echo "# Building packages and commands for host, $GOHOSTOS/$GOHOSTARCH."
	GOOS=$GOHOSTOS GOARCH=$GOHOSTARCH \
		"$GOTOOLDIR"/go_bootstrap install -gcflags "$GO_GCFLAGS" -ldflags "$GO_LDFLAGS" -v std
	echo
fi

echo "# Building packages and commands for $GOOS/$GOARCH."
"$GOTOOLDIR"/go_bootstrap install -gcflags "$GO_GCFLAGS" -ldflags "$GO_LDFLAGS" -v std
echo

rm -f "$GOTOOLDIR"/go_bootstrap

# 是否打印安装成功的提示信息。一般为:
#	Installed Go for $GOOS/$GOARCH in $GOROOT
#	Installed commands in $GOROOT/bin
if [ "$1" != "--no-banner" ]; then
	"$GOTOOLDIR"/dist banner
fi

为了方便,将带注释的脚本放在了github上:go_install_bash_comment

注:有些是根据脚本中的英文翻译的,如不明白,请看脚本的注释;如有不对,请指正!谢谢!

说明:这是初稿,很多地方还没有仔细分析。

15,824 浏览数

  6 条评论 到 “分析源码安装Go的过程(初稿)”

评论 (4) Pingbacks (2)
  1. 好文~~~

  2. 这篇帅啊~

  3. 你确定你用源码安装过

 评论

您可以使用这些 HTML 标签和属性: <a href="" title=""> <abbr title=""> <acronym title=""> <b> <blockquote cite=""> <cite> <code> <del datetime=""> <em> <i> <q cite=""> <strike> <strong>