github.com/amarpal/go-tools@v0.0.0-20240422043104-40142f59f616/staticcheck/sa9006/sa9006.go (about) 1 package sa9006 2 3 import ( 4 "fmt" 5 "go/ast" 6 "go/types" 7 8 "github.com/amarpal/go-tools/analysis/code" 9 "github.com/amarpal/go-tools/analysis/lint" 10 "github.com/amarpal/go-tools/analysis/report" 11 "github.com/amarpal/go-tools/pattern" 12 13 "golang.org/x/tools/go/analysis" 14 "golang.org/x/tools/go/analysis/passes/inspect" 15 ) 16 17 var SCAnalyzer = lint.InitializeAnalyzer(&lint.Analyzer{ 18 Analyzer: &analysis.Analyzer{ 19 Name: "SA9006", 20 Run: run, 21 Requires: []*analysis.Analyzer{inspect.Analyzer}, 22 }, 23 Doc: &lint.Documentation{ 24 Title: `Dubious bit shifting of a fixed size integer value`, 25 Text: `Bit shifting a value past its size will always clear the value. 26 27 For instance: 28 29 v := int8(42) 30 v >>= 8 31 32 will always result in 0. 33 34 This check flags bit shifting operations on fixed size integer values only. 35 That is, int, uint and uintptr are never flagged to avoid potential false 36 positives in somewhat exotic but valid bit twiddling tricks: 37 38 // Clear any value above 32 bits if integers are more than 32 bits. 39 func f(i int) int { 40 v := i >> 32 41 v = v << 32 42 return i-v 43 }`, 44 Since: "2020.2", 45 Severity: lint.SeverityWarning, 46 // Technically this should be MergeIfAll, because the type of 47 // v might be different for different build tags. Practically, 48 // don't write code that depends on that. 49 MergeIf: lint.MergeIfAny, 50 }, 51 }) 52 53 var Analyzer = SCAnalyzer.Analyzer 54 var ( 55 checkFixedLengthTypeShiftQ = pattern.MustParse(` 56 (Or 57 (AssignStmt _ (Or ">>=" "<<=") _) 58 (BinaryExpr _ (Or ">>" "<<") _)) 59 `) 60 ) 61 62 func run(pass *analysis.Pass) (interface{}, error) { 63 isDubiousShift := func(x, y ast.Expr) (int64, int64, bool) { 64 typ, ok := pass.TypesInfo.TypeOf(x).Underlying().(*types.Basic) 65 if !ok { 66 return 0, 0, false 67 } 68 switch typ.Kind() { 69 case types.Int8, types.Int16, types.Int32, types.Int64, 70 types.Uint8, types.Uint16, types.Uint32, types.Uint64: 71 // We're only interested in fixed–size types. 72 default: 73 return 0, 0, false 74 } 75 76 const bitsInByte = 8 77 typeBits := pass.TypesSizes.Sizeof(typ) * bitsInByte 78 79 shiftLength, ok := code.ExprToInt(pass, y) 80 if !ok { 81 return 0, 0, false 82 } 83 84 return typeBits, shiftLength, shiftLength >= typeBits 85 } 86 87 fn := func(node ast.Node) { 88 if _, ok := code.Match(pass, checkFixedLengthTypeShiftQ, node); !ok { 89 return 90 } 91 92 switch e := node.(type) { 93 case *ast.AssignStmt: 94 if size, shift, yes := isDubiousShift(e.Lhs[0], e.Rhs[0]); yes { 95 report.Report(pass, e, fmt.Sprintf("shifting %d-bit value by %d bits will always clear it", size, shift)) 96 } 97 case *ast.BinaryExpr: 98 if size, shift, yes := isDubiousShift(e.X, e.Y); yes { 99 report.Report(pass, e, fmt.Sprintf("shifting %d-bit value by %d bits will always clear it", size, shift)) 100 } 101 } 102 } 103 code.Preorder(pass, fn, (*ast.AssignStmt)(nil), (*ast.BinaryExpr)(nil)) 104 105 return nil, nil 106 }