2019-11-07 20:05:39 +01:00
|
|
|
package functions
|
|
|
|
|
2020-02-26 19:45:19 +01:00
|
|
|
import "honnef.co/go/tools/ir"
|
2019-11-07 20:05:39 +01:00
|
|
|
|
2020-02-26 19:45:19 +01:00
|
|
|
type Loop struct{ *ir.BlockSet }
|
2019-11-07 20:05:39 +01:00
|
|
|
|
2020-02-26 19:45:19 +01:00
|
|
|
func FindLoops(fn *ir.Function) []Loop {
|
2019-11-07 20:05:39 +01:00
|
|
|
if fn.Blocks == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
tree := fn.DomPreorder()
|
|
|
|
var sets []Loop
|
|
|
|
for _, h := range tree {
|
|
|
|
for _, n := range h.Preds {
|
|
|
|
if !h.Dominates(n) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// n is a back-edge to h
|
|
|
|
// h is the loop header
|
|
|
|
if n == h {
|
2020-02-26 19:45:19 +01:00
|
|
|
set := Loop{ir.NewBlockSet(len(fn.Blocks))}
|
2019-11-07 20:05:39 +01:00
|
|
|
set.Add(n)
|
|
|
|
sets = append(sets, set)
|
|
|
|
continue
|
|
|
|
}
|
2020-02-26 19:45:19 +01:00
|
|
|
set := Loop{ir.NewBlockSet(len(fn.Blocks))}
|
2019-11-07 20:05:39 +01:00
|
|
|
set.Add(h)
|
|
|
|
set.Add(n)
|
|
|
|
for _, b := range allPredsBut(n, h, nil) {
|
|
|
|
set.Add(b)
|
|
|
|
}
|
|
|
|
sets = append(sets, set)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return sets
|
|
|
|
}
|
|
|
|
|
2020-02-26 19:45:19 +01:00
|
|
|
func allPredsBut(b, but *ir.BasicBlock, list []*ir.BasicBlock) []*ir.BasicBlock {
|
2019-11-07 20:05:39 +01:00
|
|
|
outer:
|
|
|
|
for _, pred := range b.Preds {
|
|
|
|
if pred == but {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, p := range list {
|
|
|
|
// TODO improve big-o complexity of this function
|
|
|
|
if pred == p {
|
|
|
|
continue outer
|
|
|
|
}
|
|
|
|
}
|
|
|
|
list = append(list, pred)
|
|
|
|
list = allPredsBut(pred, but, list)
|
|
|
|
}
|
|
|
|
return list
|
|
|
|
}
|