A Block represents a basic block: a list of statements and expressions that are always evaluated sequentially.
A block may have 0-2 successors: zero for a return block or a block that calls a function such as panic that never returns; one for a normal (jump) block; and two for a conditional (if) block.
type Block struct { Nodes []ast.Node // statements, expressions, and ValueSpecs Succs []*Block // successor nodes in the graph Index int32 // index within CFG.Blocks Live bool // block is reachable from entry // contains filtered or unexported fields }
func (b *Block) Return() (ret *ast.ReturnStmt)
Return returns the return statement at the end of this block if present, nil otherwise.
func (b *Block) String() string
A CFG represents the control-flow graph of a single function.
The entry point is Blocks[0]; there may be multiple return blocks.
type CFG struct { Blocks []*Block // block[0] is entry; order otherwise undefined }
func New(body *ast.BlockStmt, mayReturn func(*ast.CallExpr) bool) *CFG
New returns a new control-flow graph for the specified function body, which must be non-nil.
The CFG builder calls mayReturn to determine whether a given function call may return. For example, calls to panic, os.Exit, and log.Fatal do not return, so the builder can remove infeasible graph edges following such calls. The builder calls mayReturn only for a CallExpr beneath an ExprStmt.
func (g *CFG) Format(fset *token.FileSet) string
Format formats the control-flow graph for ease of debugging.