vikunja-api/vendor/honnef.co/go/tools/functions/loops.go

55 lines
1.0 KiB
Go
Raw Normal View History

2018-12-31 02:18:41 +01:00
package functions
2020-05-29 22:15:21 +02:00
import "honnef.co/go/tools/ir"
2018-12-31 02:18:41 +01:00
2020-05-29 22:15:21 +02:00
type Loop struct{ *ir.BlockSet }
2018-12-31 02:18:41 +01:00
2020-05-29 22:15:21 +02:00
func FindLoops(fn *ir.Function) []Loop {
2018-12-31 02:18:41 +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-05-29 22:15:21 +02:00
set := Loop{ir.NewBlockSet(len(fn.Blocks))}
set.Add(n)
sets = append(sets, set)
2018-12-31 02:18:41 +01:00
continue
}
2020-05-29 22:15:21 +02:00
set := Loop{ir.NewBlockSet(len(fn.Blocks))}
set.Add(h)
set.Add(n)
2018-12-31 02:18:41 +01:00
for _, b := range allPredsBut(n, h, nil) {
set.Add(b)
2018-12-31 02:18:41 +01:00
}
sets = append(sets, set)
}
}
return sets
}
2020-05-29 22:15:21 +02:00
func allPredsBut(b, but *ir.BasicBlock, list []*ir.BasicBlock) []*ir.BasicBlock {
2018-12-31 02:18:41 +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
}