2018-12-31 02:18:41 +01:00
|
|
|
package functions
|
|
|
|
|
|
|
|
import (
|
|
|
|
"honnef.co/go/tools/ssa"
|
|
|
|
)
|
|
|
|
|
2020-05-09 15:44:17 +02:00
|
|
|
func filterDebug(instr []ssa.Instruction) []ssa.Instruction {
|
|
|
|
var out []ssa.Instruction
|
|
|
|
for _, ins := range instr {
|
|
|
|
if _, ok := ins.(*ssa.DebugRef); !ok {
|
|
|
|
out = append(out, ins)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return out
|
|
|
|
}
|
|
|
|
|
2018-12-31 02:18:41 +01:00
|
|
|
// IsStub reports whether a function is a stub. A function is
|
|
|
|
// considered a stub if it has no instructions or exactly one
|
|
|
|
// instruction, which must be either returning only constant values or
|
|
|
|
// a panic.
|
2020-05-09 15:44:17 +02:00
|
|
|
func IsStub(fn *ssa.Function) bool {
|
2018-12-31 02:18:41 +01:00
|
|
|
if len(fn.Blocks) == 0 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if len(fn.Blocks) > 1 {
|
|
|
|
return false
|
|
|
|
}
|
2020-05-09 15:44:17 +02:00
|
|
|
instrs := filterDebug(fn.Blocks[0].Instrs)
|
2018-12-31 02:18:41 +01:00
|
|
|
if len(instrs) != 1 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
switch instrs[0].(type) {
|
|
|
|
case *ssa.Return:
|
|
|
|
// Since this is the only instruction, the return value must
|
|
|
|
// be a constant. We consider all constants as stubs, not just
|
|
|
|
// the zero value. This does not, unfortunately, cover zero
|
|
|
|
// initialised structs, as these cause additional
|
|
|
|
// instructions.
|
|
|
|
return true
|
|
|
|
case *ssa.Panic:
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|