|  | // Copyright 2018 The Go Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | package escape | 
|  |  | 
|  | import ( | 
|  | "cmd/compile/internal/base" | 
|  | "cmd/compile/internal/ir" | 
|  | "fmt" | 
|  | ) | 
|  |  | 
|  | // stmt evaluates a single Go statement. | 
|  | func (e *escape) stmt(n ir.Node) { | 
|  | if n == nil { | 
|  | return | 
|  | } | 
|  |  | 
|  | lno := ir.SetPos(n) | 
|  | defer func() { | 
|  | base.Pos = lno | 
|  | }() | 
|  |  | 
|  | if base.Flag.LowerM > 2 { | 
|  | fmt.Printf("%v:[%d] %v stmt: %v\n", base.FmtPos(base.Pos), e.loopDepth, e.curfn, n) | 
|  | } | 
|  |  | 
|  | e.stmts(n.Init()) | 
|  |  | 
|  | switch n.Op() { | 
|  | default: | 
|  | base.Fatalf("unexpected stmt: %v", n) | 
|  |  | 
|  | case ir.ODCLCONST, ir.ODCLTYPE, ir.OFALL, ir.OINLMARK: | 
|  | // nop | 
|  |  | 
|  | case ir.OBREAK, ir.OCONTINUE, ir.OGOTO: | 
|  | // TODO(mdempsky): Handle dead code? | 
|  |  | 
|  | case ir.OBLOCK: | 
|  | n := n.(*ir.BlockStmt) | 
|  | e.stmts(n.List) | 
|  |  | 
|  | case ir.ODCL: | 
|  | // Record loop depth at declaration. | 
|  | n := n.(*ir.Decl) | 
|  | if !ir.IsBlank(n.X) { | 
|  | e.dcl(n.X) | 
|  | } | 
|  |  | 
|  | case ir.OLABEL: | 
|  | n := n.(*ir.LabelStmt) | 
|  | switch e.labels[n.Label] { | 
|  | case nonlooping: | 
|  | if base.Flag.LowerM > 2 { | 
|  | fmt.Printf("%v:%v non-looping label\n", base.FmtPos(base.Pos), n) | 
|  | } | 
|  | case looping: | 
|  | if base.Flag.LowerM > 2 { | 
|  | fmt.Printf("%v: %v looping label\n", base.FmtPos(base.Pos), n) | 
|  | } | 
|  | e.loopDepth++ | 
|  | default: | 
|  | base.Fatalf("label missing tag") | 
|  | } | 
|  | delete(e.labels, n.Label) | 
|  |  | 
|  | case ir.OIF: | 
|  | n := n.(*ir.IfStmt) | 
|  | e.discard(n.Cond) | 
|  | e.block(n.Body) | 
|  | e.block(n.Else) | 
|  |  | 
|  | case ir.OFOR, ir.OFORUNTIL: | 
|  | n := n.(*ir.ForStmt) | 
|  | e.loopDepth++ | 
|  | e.discard(n.Cond) | 
|  | e.stmt(n.Post) | 
|  | e.block(n.Body) | 
|  | e.loopDepth-- | 
|  |  | 
|  | case ir.ORANGE: | 
|  | // for Key, Value = range X { Body } | 
|  | n := n.(*ir.RangeStmt) | 
|  |  | 
|  | // X is evaluated outside the loop. | 
|  | tmp := e.newLoc(nil, false) | 
|  | e.expr(tmp.asHole(), n.X) | 
|  |  | 
|  | e.loopDepth++ | 
|  | ks := e.addrs([]ir.Node{n.Key, n.Value}) | 
|  | if n.X.Type().IsArray() { | 
|  | e.flow(ks[1].note(n, "range"), tmp) | 
|  | } else { | 
|  | e.flow(ks[1].deref(n, "range-deref"), tmp) | 
|  | } | 
|  | e.reassigned(ks, n) | 
|  |  | 
|  | e.block(n.Body) | 
|  | e.loopDepth-- | 
|  |  | 
|  | case ir.OSWITCH: | 
|  | n := n.(*ir.SwitchStmt) | 
|  |  | 
|  | if guard, ok := n.Tag.(*ir.TypeSwitchGuard); ok { | 
|  | var ks []hole | 
|  | if guard.Tag != nil { | 
|  | for _, cas := range n.Cases { | 
|  | cv := cas.Var | 
|  | k := e.dcl(cv) // type switch variables have no ODCL. | 
|  | if cv.Type().HasPointers() { | 
|  | ks = append(ks, k.dotType(cv.Type(), cas, "switch case")) | 
|  | } | 
|  | } | 
|  | } | 
|  | e.expr(e.teeHole(ks...), n.Tag.(*ir.TypeSwitchGuard).X) | 
|  | } else { | 
|  | e.discard(n.Tag) | 
|  | } | 
|  |  | 
|  | for _, cas := range n.Cases { | 
|  | e.discards(cas.List) | 
|  | e.block(cas.Body) | 
|  | } | 
|  |  | 
|  | case ir.OSELECT: | 
|  | n := n.(*ir.SelectStmt) | 
|  | for _, cas := range n.Cases { | 
|  | e.stmt(cas.Comm) | 
|  | e.block(cas.Body) | 
|  | } | 
|  | case ir.ORECV: | 
|  | // TODO(mdempsky): Consider e.discard(n.Left). | 
|  | n := n.(*ir.UnaryExpr) | 
|  | e.exprSkipInit(e.discardHole(), n) // already visited n.Ninit | 
|  | case ir.OSEND: | 
|  | n := n.(*ir.SendStmt) | 
|  | e.discard(n.Chan) | 
|  | e.assignHeap(n.Value, "send", n) | 
|  |  | 
|  | case ir.OAS: | 
|  | n := n.(*ir.AssignStmt) | 
|  | e.assignList([]ir.Node{n.X}, []ir.Node{n.Y}, "assign", n) | 
|  | case ir.OASOP: | 
|  | n := n.(*ir.AssignOpStmt) | 
|  | // TODO(mdempsky): Worry about OLSH/ORSH? | 
|  | e.assignList([]ir.Node{n.X}, []ir.Node{n.Y}, "assign", n) | 
|  | case ir.OAS2: | 
|  | n := n.(*ir.AssignListStmt) | 
|  | e.assignList(n.Lhs, n.Rhs, "assign-pair", n) | 
|  |  | 
|  | case ir.OAS2DOTTYPE: // v, ok = x.(type) | 
|  | n := n.(*ir.AssignListStmt) | 
|  | e.assignList(n.Lhs, n.Rhs, "assign-pair-dot-type", n) | 
|  | case ir.OAS2MAPR: // v, ok = m[k] | 
|  | n := n.(*ir.AssignListStmt) | 
|  | e.assignList(n.Lhs, n.Rhs, "assign-pair-mapr", n) | 
|  | case ir.OAS2RECV, ir.OSELRECV2: // v, ok = <-ch | 
|  | n := n.(*ir.AssignListStmt) | 
|  | e.assignList(n.Lhs, n.Rhs, "assign-pair-receive", n) | 
|  |  | 
|  | case ir.OAS2FUNC: | 
|  | n := n.(*ir.AssignListStmt) | 
|  | e.stmts(n.Rhs[0].Init()) | 
|  | ks := e.addrs(n.Lhs) | 
|  | e.call(ks, n.Rhs[0]) | 
|  | e.reassigned(ks, n) | 
|  | case ir.ORETURN: | 
|  | n := n.(*ir.ReturnStmt) | 
|  | results := e.curfn.Type().Results().FieldSlice() | 
|  | dsts := make([]ir.Node, len(results)) | 
|  | for i, res := range results { | 
|  | dsts[i] = res.Nname.(*ir.Name) | 
|  | } | 
|  | e.assignList(dsts, n.Results, "return", n) | 
|  | case ir.OCALLFUNC, ir.OCALLMETH, ir.OCALLINTER, ir.OINLCALL, ir.OCLOSE, ir.OCOPY, ir.ODELETE, ir.OPANIC, ir.OPRINT, ir.OPRINTN, ir.ORECOVER: | 
|  | e.call(nil, n) | 
|  | case ir.OGO, ir.ODEFER: | 
|  | n := n.(*ir.GoDeferStmt) | 
|  | e.goDeferStmt(n) | 
|  |  | 
|  | case ir.OTAILCALL: | 
|  | // TODO(mdempsky): Treat like a normal call? esc.go used to just ignore it. | 
|  | } | 
|  | } | 
|  |  | 
|  | func (e *escape) stmts(l ir.Nodes) { | 
|  | for _, n := range l { | 
|  | e.stmt(n) | 
|  | } | 
|  | } | 
|  |  | 
|  | // block is like stmts, but preserves loopDepth. | 
|  | func (e *escape) block(l ir.Nodes) { | 
|  | old := e.loopDepth | 
|  | e.stmts(l) | 
|  | e.loopDepth = old | 
|  | } | 
|  |  | 
|  | func (e *escape) dcl(n *ir.Name) hole { | 
|  | if n.Curfn != e.curfn || n.IsClosureVar() { | 
|  | base.Fatalf("bad declaration of %v", n) | 
|  | } | 
|  | loc := e.oldLoc(n) | 
|  | loc.loopDepth = e.loopDepth | 
|  | return loc.asHole() | 
|  | } |