|  | // Copyright 2017 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. | 
|  | // | 
|  | // Simplified dead code detector. Used for skipping certain checks | 
|  | // on unreachable code (for instance, shift checks on arch-specific code). | 
|  |  | 
|  | package main | 
|  |  | 
|  | import ( | 
|  | "go/ast" | 
|  | "go/constant" | 
|  | ) | 
|  |  | 
|  | // updateDead puts unreachable "if" and "case" nodes into f.dead. | 
|  | func (f *File) updateDead(node ast.Node) { | 
|  | if f.dead[node] { | 
|  | // The node is already marked as dead. | 
|  | return | 
|  | } | 
|  |  | 
|  | switch stmt := node.(type) { | 
|  | case *ast.IfStmt: | 
|  | // "if" branch is dead if its condition evaluates | 
|  | // to constant false. | 
|  | v := f.pkg.types[stmt.Cond].Value | 
|  | if v == nil { | 
|  | return | 
|  | } | 
|  | if !constant.BoolVal(v) { | 
|  | f.setDead(stmt.Body) | 
|  | return | 
|  | } | 
|  | f.setDead(stmt.Else) | 
|  | case *ast.SwitchStmt: | 
|  | // Case clause with empty switch tag is dead if it evaluates | 
|  | // to constant false. | 
|  | if stmt.Tag == nil { | 
|  | BodyLoopBool: | 
|  | for _, stmt := range stmt.Body.List { | 
|  | cc := stmt.(*ast.CaseClause) | 
|  | if cc.List == nil { | 
|  | // Skip default case. | 
|  | continue | 
|  | } | 
|  | for _, expr := range cc.List { | 
|  | v := f.pkg.types[expr].Value | 
|  | if v == nil || constant.BoolVal(v) { | 
|  | continue BodyLoopBool | 
|  | } | 
|  | } | 
|  | f.setDead(cc) | 
|  | } | 
|  | return | 
|  | } | 
|  |  | 
|  | // Case clause is dead if its constant value doesn't match | 
|  | // the constant value from the switch tag. | 
|  | // TODO: This handles integer comparisons only. | 
|  | v := f.pkg.types[stmt.Tag].Value | 
|  | if v == nil || v.Kind() != constant.Int { | 
|  | return | 
|  | } | 
|  | tagN, ok := constant.Uint64Val(v) | 
|  | if !ok { | 
|  | return | 
|  | } | 
|  | BodyLoopInt: | 
|  | for _, x := range stmt.Body.List { | 
|  | cc := x.(*ast.CaseClause) | 
|  | if cc.List == nil { | 
|  | // Skip default case. | 
|  | continue | 
|  | } | 
|  | for _, expr := range cc.List { | 
|  | v := f.pkg.types[expr].Value | 
|  | if v == nil { | 
|  | continue BodyLoopInt | 
|  | } | 
|  | n, ok := constant.Uint64Val(v) | 
|  | if !ok || tagN == n { | 
|  | continue BodyLoopInt | 
|  | } | 
|  | } | 
|  | f.setDead(cc) | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // setDead marks the node and all the children as dead. | 
|  | func (f *File) setDead(node ast.Node) { | 
|  | dv := deadVisitor{ | 
|  | f: f, | 
|  | } | 
|  | ast.Walk(dv, node) | 
|  | } | 
|  |  | 
|  | type deadVisitor struct { | 
|  | f *File | 
|  | } | 
|  |  | 
|  | func (dv deadVisitor) Visit(node ast.Node) ast.Visitor { | 
|  | if node == nil { | 
|  | return nil | 
|  | } | 
|  | dv.f.dead[node] = true | 
|  | return dv | 
|  | } |