前面大概说过bison与yacc的语法,那么其实go也是用bison来实现的,下面来说说怎么阅读go语言源代码。要看代码首先得找到代码的入口,怎么找呢?你可能首先想到grep main是的,一般来说这是一个没有争议的办法,但是对于go语言来说,这个办法对,也不对。怎么说呢?我们用gdb来进行debug就会看到所有的go语言程序的入口居然都在src/lib9/main.c里的main函数。


#include 
#define NOPLAN9DEFINES
#include 

extern void p9main(int, char**);

int
main(int argc, char **argv)
{
    p9main(argc, argv);
    exits("main");
    return 99; 
}

从go/include/libc.h中里我们可以看到这么一句


#ifndef NOPLAN9DEFINES
#define main    p9main
#endif

也就是说,在go语言源代码里main函数都重命名了,所有的函数入口都变成了src/lib9/mina.c,也实际的入口各自的代码里,因此说入口是各自的main这句话,对也不对。

那么先大概说一下go语言的代码结构


.
|-- bin//我自己定义的安装目录
|-- doc//相关文档,godoc就是利用这个目录
|   |-- codelab
|   |   `-- wiki
|   |-- codewalk
|   |-- devel
|   |-- gopher
|   |-- progs
|   `-- talks
|       `-- io2010
|-- include//一些全局的定义文件
|-- lib//一些lib9相关的全局库
|   |-- codereview
|   `-- godoc
|-- misc//一些工具
|   |-- arm
|   |-- bash
|   |-- bbedit
|   |-- cgo
|   |   |-- gmp
|   |   |-- life
|   |   `-- stdio
|   |-- dashboard
|   |   `-- godashboard
|   |-- emacs
|   |-- kate
|   |-- nacl
|   |-- vim
|   |   |-- ftdetect
|   |   `-- syntax
|   `-- xcode
|-- pkg//编译好的go库
|   `-- linux_amd64
|       |-- archive
|       |-- compress
|       |-- container
|       |-- crypto
|       |-- debug
|       |-- encoding
|       |-- exp
|       |-- go
|       |-- hash
|       |-- http
|       |-- image
|       |-- io
|       |-- os
|       |-- rpc
|       |-- runtime
|       `-- testing
|-- src
|   |-- cmd//go语言相关的程序
|   |   |-- 5a
|   |   |-- 5c
|   |   |-- 5g
|   |   |-- 5l
|   |   |-- 6a//amd64的汇编器
|   |   |-- 6c//amd64的c编译器
|   |   |-- 6g//amd64的编译器
|   |   |-- 6l//amd64的链接器
|   |   |-- 8a
|   |   |-- 8c
|   |   |-- 8g
|   |   |-- 8l
|   |   |-- cc
|   |   |-- cgo
|   |   |-- cov
|   |   |-- ebnflint
|   |   |-- gc
|   |   |-- godefs
|   |   |-- godoc
|   |   |-- gofmt
|   |   |-- goinstall
|   |   |-- gopack
|   |   |-- gotest
|   |   |-- goyacc
|   |   |-- hgpatch
|   |   |-- ld
|   |   |-- nm
|   |   `-- prof
|   |-- lib9//lib9相关的库
|   |   |-- fmt
|   |   `-- utf
|   |-- libbio//全局库
|   |-- libcgo//cgo的库
|   |-- libmach//库
|   `-- pkg//go库
|       |-- archive
|       |-- asn1
|       |-- big
|       |-- bufio
|       |-- bytes
|       |-- cmath
|       |-- compress
|       |-- container
|       |-- crypto
|       |-- debug
|       |-- ebnf
|       |-- encoding
|       |-- exec
|       |-- exp
|       |-- expvar
|       |-- flag
|       |-- fmt
|       |-- go
|       |-- gob
|       |-- hash
|       |-- html
|       |-- http
|       |-- image
|       |-- io
|       |-- json
|       |-- log
|       |-- math
|       |-- mime
|       |-- net
|       |-- netchan
|       |-- nntp
|       |-- once
|       |-- os
|       |-- patch
|       |-- path
|       |-- rand
|       |-- reflect
|       |-- regexp
|       |-- rpc
|       |-- runtime
|       |-- scanner
|       |-- sort
|       |-- strconv
|       |-- strings
|       |-- sync
|       |-- syscall
|       |-- syslog
|       |-- tabwriter
|       |-- template
|       |-- testing
|       |-- time
|       |-- unicode
|       |-- unsafe
|       |-- utf16
|       |-- utf8
|       |-- websocket
|       `-- xml
`-- test//测试
    |-- bench
    |-- bugs
    |-- chan
    |-- fixedbugs
    |   |-- bug083.dir
    |   |-- bug088.dir
    |   |-- bug106.dir
    |   |-- bug133.dir
    |   |-- bug160.dir
    |   |-- bug191.dir
    |   |-- bug222.dir
    |   |-- bug226.dir
    |   |-- bug248.dir
    |   `-- bug282.dir
    |-- garbage
    |-- interface
    |-- ken
    |-- nilptr
    `-- syntax

我们先来看一个go语言的hello world文件hello.go

package main

func main(){
    println("Hello World");
}

那么这个文件要被编译链接成一个可执行文件,那么要执行如下命令:

6g hello.o
6l hello.6 -o hello

很显然最关键的就是6g和6l两个程序,那么我们就从6g开始

首先gdb -tui打开tui界面的gdb程序

然后输入file 6g载入6g程序,就可以看到

/home/hoping/go/src/lib9/main.c,然后输入start后,step进入了/home/hoping/go/src/cmd/gc/lex.c的main函数

因此我们只需要看这个文件就大概可以看到程序的执行流程了。接下来暂时不需要gdb了,我们换vi和ctags。

在go目录执行ctags -R *来生成tag,然后用vi打开lex.c进入main函数,其实即使你看了这个函数也不会有太多收获。

正如我们前面说过的那样,6g是用bison的,那么显然我们需要找到bison文件和yylex函数即可。既然入口在gc文件夹,那么猜想y文件也在gc文件夹,于是进入ls *.y之后可以看到go.y。而go.y的输入全是由yylex函数来提供的。所以首先我们来看看yylex函数,在go文件夹下执行vim -t yylex,然后选择gc文件夹那个。