From abcceb006891056967210357117b369a360b20f6 Mon Sep 17 00:00:00 2001 From: Marcus Gartner Date: Sat, 7 May 2022 00:11:41 -0400 Subject: [PATCH] opt: display memo cycle when potential cycle is detected The optimizer will error with "memo group optimization passes surpassed limit of 100000; there may be a cycle in the memo" when it detects a potential memo cycle. The error includes the memo in the error details to aid in debugging. However, it can be hard for a human to find the cycle in the formatted memo when it contains many groups (see #80901). This commit updates the formatted memo within the error details to include the path to the cycle, if one exists. A cycle is not searched for in other cases where the memo is formatted, to avoid the overhead of cycle detection. A simple cycle detection library has been added that the memo formatter uses to locate a path in a memo to a cycle. Release note: None --- pkg/BUILD.bazel | 1 + pkg/sql/opt/cycle/BUILD.bazel | 14 ++ pkg/sql/opt/cycle/detector.go | 118 ++++++++++++++ pkg/sql/opt/cycle/detector_test.go | 188 ++++++++++++++++++++++ pkg/sql/opt/memo/logical_props_builder.go | 5 + pkg/sql/opt/xform/BUILD.bazel | 2 + pkg/sql/opt/xform/cycle_funcs.go | 18 +++ pkg/sql/opt/xform/memo_format.go | 62 ++++++- pkg/sql/opt/xform/optimizer.go | 2 +- pkg/sql/opt/xform/rules/cycle.opt | 17 ++ pkg/sql/opt/xform/testdata/rules/cycle | 83 +++++++++- 11 files changed, 505 insertions(+), 5 deletions(-) create mode 100644 pkg/sql/opt/cycle/BUILD.bazel create mode 100644 pkg/sql/opt/cycle/detector.go create mode 100644 pkg/sql/opt/cycle/detector_test.go create mode 100644 pkg/sql/opt/xform/cycle_funcs.go diff --git a/pkg/BUILD.bazel b/pkg/BUILD.bazel index e3cd9d9be391..f5e0b29cb9ed 100644 --- a/pkg/BUILD.bazel +++ b/pkg/BUILD.bazel @@ -317,6 +317,7 @@ ALL_TESTS = [ "//pkg/sql/opt/bench:bench_test", "//pkg/sql/opt/cat:cat_test", "//pkg/sql/opt/constraint:constraint_test", + "//pkg/sql/opt/cycle:cycle_test", "//pkg/sql/opt/distribution:distribution_test", "//pkg/sql/opt/exec/execbuilder:execbuilder_test", "//pkg/sql/opt/exec/explain:explain_test", diff --git a/pkg/sql/opt/cycle/BUILD.bazel b/pkg/sql/opt/cycle/BUILD.bazel new file mode 100644 index 000000000000..b27fd3315666 --- /dev/null +++ b/pkg/sql/opt/cycle/BUILD.bazel @@ -0,0 +1,14 @@ +load("@io_bazel_rules_go//go:def.bzl", "go_library", "go_test") + +go_library( + name = "cycle", + srcs = ["detector.go"], + importpath = "github.com/cockroachdb/cockroach/pkg/sql/opt/cycle", + visibility = ["//visibility:public"], +) + +go_test( + name = "cycle_test", + srcs = ["detector_test.go"], + embed = [":cycle"], +) diff --git a/pkg/sql/opt/cycle/detector.go b/pkg/sql/opt/cycle/detector.go new file mode 100644 index 000000000000..dd2d5687c34a --- /dev/null +++ b/pkg/sql/opt/cycle/detector.go @@ -0,0 +1,118 @@ +// Copyright 2022 The Cockroach Authors. +// +// Use of this software is governed by the Business Source License +// included in the file licenses/BSL.txt. +// +// As of the Change Date specified in that file, in accordance with +// the Business Source License, use of this software will be governed +// by the Apache License, Version 2.0, included in the file +// licenses/APL.txt. + +package cycle + +// Detector finds cycles in directed graphs. Vertices are identified as unique +// integers by the caller. +// +// Example usage: +// +// var d Detector +// d.AddEdge(Vertex(0), Vertex(1)) +// d.AddEdge(Vertex(1), Vertex(2)) +// d.AddEdge(Vertex(2), Vertex(0)) +// d.FindCycleStartingAtVertex(Vertex(0)) +// => [0, 1, 2, 0], true +// +type Detector struct { + // edges are the directed edges in the graph. + edges map[Vertex][]Vertex +} + +// Vertex is a vertex in the graph. +type Vertex int + +// status represents one of three possible states of a vertex during cycle +// detection. +type status int + +const ( + // statusNotVisited indicates that a vertex has not been previously visited. + statusNotVisited status = iota + // statusVisited indicates that a vertex has been visited but a verdict on + // its involvement in a cycle has not yet been reached. + statusVisited + // statusDone indicates that a vertex has been visited and has been proven + // innocent of being involved in a cycle. + statusDone +) + +// AddEdge adds a directed edge to the graph. +func (cd *Detector) AddEdge(from, to Vertex) { + if cd.edges == nil { + cd.edges = make(map[Vertex][]Vertex) + } + cd.edges[from] = append(cd.edges[from], to) +} + +// FindCycleStartingAtVertex searches for a cycle starting from v. If one is +// found, a path containing that cycle is returned. After finding the first +// cycle, searching ceases. If no cycle is found, ok=false is returned. +func (cd *Detector) FindCycleStartingAtVertex(v Vertex) (cycle []Vertex, ok bool) { + vertices := make(map[Vertex]status) + var s stack + if ok := cd.hasCycle(v, vertices, &s); ok { + // If a cycle was found, s will contain a path that includes the cycle. + return s, true + } + return nil, false +} + +// hasCycle performs a depth-first search of the graph starting at v in search +// of a cycle. It returns true when the first cycle is found. If a cycle was +// found, s contains a path that includes the cycle. Cycles are detected by +// finding edges that backtrack to a previously visited vertex. +func (cd *Detector) hasCycle(v Vertex, vertices map[Vertex]status, s *stack) bool { + // Mark the given Vertex as visited and push it onto the stack. + vertices[v] = statusVisited + s.push(v) + + // Search all children of v for cycles. + for _, child := range cd.edges[v] { + switch vertices[child] { + case statusVisited: + // A cycle has been found. The current stack is a path to the cycle. + // Push the child onto the stack so that the cycle is more obvious + // in the path. + s.push(child) + return true + case statusNotVisited: + // Recurse if this child of v has not yet been visited. + if ok := cd.hasCycle(child, vertices, s); ok { + // If a cycle was found deeper down, propagate the result and + // halt searching other children of v. + return true + } + case statusDone: + // We have already proven that this child does not backtrack to + // anywhere further up the stack, so we do not need to recurse into + // it. + } + } + + // If we didn't return early from the loop, we did not find a cycle below + // this vertex, so we can mark the vertex as done and pop it off the stack. + vertices[v] = statusDone + s.pop() + return false +} + +// stack is a simple implementation of a stack of vertices. It allows more +// ergonomic mutation by callees than a slice. +type stack []Vertex + +func (s *stack) push(i Vertex) { + *s = append(*s, i) +} + +func (s *stack) pop() { + *s = (*s)[:len(*s)-1] +} diff --git a/pkg/sql/opt/cycle/detector_test.go b/pkg/sql/opt/cycle/detector_test.go new file mode 100644 index 000000000000..5dccee16a5e3 --- /dev/null +++ b/pkg/sql/opt/cycle/detector_test.go @@ -0,0 +1,188 @@ +// Copyright 2022 The Cockroach Authors. +// +// Use of this software is governed by the Business Source License +// included in the file licenses/BSL.txt. +// +// As of the Change Date specified in that file, in accordance with +// the Business Source License, use of this software will be governed +// by the Apache License, Version 2.0, included in the file +// licenses/APL.txt. + +package cycle + +import ( + "reflect" + "testing" +) + +type edge struct { + from Vertex + to Vertex +} + +func TestDetector(t *testing.T) { + testCases := []struct { + edges []edge + start Vertex + expected []Vertex + }{ + // No cycles. + { + edges: []edge{}, + start: 0, + expected: nil, + }, + { + edges: []edge{ + {0, 1}, + {1, 2}, + {2, 3}, + }, + start: 0, + expected: nil, + }, + { + edges: []edge{ + {0, 1}, + {0, 2}, + {0, 3}, + {1, 2}, + {2, 3}, + }, + start: 0, + expected: nil, + }, + { + edges: []edge{ + {0, 1}, + {0, 2}, + {0, 3}, + {1, 2}, + {3, 1}, + }, + start: 0, + expected: nil, + }, + { + edges: []edge{ + {0, 1}, + {0, 4}, + {1, 2}, + {1, 3}, + {4, 3}, + {4, 5}, + {5, 3}, + }, + start: 0, + expected: nil, + }, + { + edges: []edge{ + {1, 2}, + {1, 3}, + {1, 4}, + {1, 7}, + {4, 5}, + {5, 6}, + {6, 7}, + {7, 4}, + }, + start: 0, + expected: nil, + }, + + // Cycles. + { + edges: []edge{ + {0, 0}, + }, + start: 0, + expected: []Vertex{0, 0}, + }, + { + edges: []edge{ + {0, 1}, + {1, 0}, + }, + start: 0, + expected: []Vertex{0, 1, 0}, + }, + { + edges: []edge{ + {0, 1}, + {1, 0}, + }, + start: 1, + expected: []Vertex{1, 0, 1}, + }, + { + edges: []edge{ + {0, 1}, + {1, 2}, + {2, 3}, + {3, 4}, + {4, 1}, + }, + start: 0, + expected: []Vertex{0, 1, 2, 3, 4, 1}, + }, + { + edges: []edge{ + {0, 1}, + {1, 3}, + {3, 2}, + {2, 1}, + }, + start: 0, + expected: []Vertex{0, 1, 3, 2, 1}, + }, + { + edges: []edge{ + {0, 1}, + {0, 4}, + {1, 2}, + {1, 3}, + {3, 5}, + {4, 3}, + {4, 5}, + {5, 1}, + }, + start: 0, + expected: []Vertex{0, 1, 3, 5, 1}, + }, + { + edges: []edge{ + {1, 2}, + {1, 3}, + {1, 4}, + {1, 7}, + {4, 5}, + {5, 6}, + {6, 7}, + {7, 4}, + }, + start: 1, + expected: []Vertex{1, 4, 5, 6, 7, 4}, + }, + { + edges: []edge{ + {0, 1}, + {1, 2}, + {2, 3}, + {3, 3}, + }, + start: 1, + expected: []Vertex{1, 2, 3, 3}, + }, + } + for _, tc := range testCases { + var d Detector + for _, edge := range tc.edges { + d.AddEdge(edge.from, edge.to) + } + res, _ := d.FindCycleStartingAtVertex(tc.start) + if !reflect.DeepEqual(res, tc.expected) { + t.Errorf("expected %v, got %v", tc.expected, res) + } + } +} diff --git a/pkg/sql/opt/memo/logical_props_builder.go b/pkg/sql/opt/memo/logical_props_builder.go index c5186d61efab..c4635305f1f2 100644 --- a/pkg/sql/opt/memo/logical_props_builder.go +++ b/pkg/sql/opt/memo/logical_props_builder.go @@ -2578,6 +2578,11 @@ func (b *logicalPropsBuilder) buildMemoCycleTestRelProps( // from transforming mc into an empty Values expression. inputProps := mc.Input.Relational() rel.Cardinality = inputProps.Cardinality + // Make the output cols the same and the input cols to avoid assertion + // failures. + rel.OutputCols = inputProps.OutputCols + // Make the row count non-zero to avoid assertion failures. + rel.Stats.RowCount = 1 } // WithUses returns the WithUsesMap for the given expression. diff --git a/pkg/sql/opt/xform/BUILD.bazel b/pkg/sql/opt/xform/BUILD.bazel index 70cf0cbe8810..448d248ef2de 100644 --- a/pkg/sql/opt/xform/BUILD.bazel +++ b/pkg/sql/opt/xform/BUILD.bazel @@ -4,6 +4,7 @@ go_library( name = "xform", srcs = [ "coster.go", + "cycle_funcs.go", "explorer.go", "general_funcs.go", "groupby_funcs.go", @@ -30,6 +31,7 @@ go_library( "//pkg/sql/opt", "//pkg/sql/opt/cat", "//pkg/sql/opt/constraint", + "//pkg/sql/opt/cycle", "//pkg/sql/opt/distribution", "//pkg/sql/opt/idxconstraint", "//pkg/sql/opt/invertedexpr", diff --git a/pkg/sql/opt/xform/cycle_funcs.go b/pkg/sql/opt/xform/cycle_funcs.go new file mode 100644 index 000000000000..2b8e6138ece6 --- /dev/null +++ b/pkg/sql/opt/xform/cycle_funcs.go @@ -0,0 +1,18 @@ +// Copyright 2022 The Cockroach Authors. +// +// Use of this software is governed by the Business Source License +// included in the file licenses/BSL.txt. +// +// As of the Change Date specified in that file, in accordance with +// the Business Source License, use of this software will be governed +// by the Apache License, Version 2.0, included in the file +// licenses/APL.txt. + +package xform + +import "github.com/cockroachdb/cockroach/pkg/sql/opt/memo" + +// EmptySubqueryPrivate returns an empty SubqueryPrivate. +func (c *CustomFuncs) EmptySubqueryPrivate() *memo.SubqueryPrivate { + return &memo.SubqueryPrivate{} +} diff --git a/pkg/sql/opt/xform/memo_format.go b/pkg/sql/opt/xform/memo_format.go index 7a4988326738..8de29ab6e930 100644 --- a/pkg/sql/opt/xform/memo_format.go +++ b/pkg/sql/opt/xform/memo_format.go @@ -16,6 +16,7 @@ import ( "sort" "github.com/cockroachdb/cockroach/pkg/sql/opt" + "github.com/cockroachdb/cockroach/pkg/sql/opt/cycle" "github.com/cockroachdb/cockroach/pkg/sql/opt/memo" "github.com/cockroachdb/cockroach/pkg/sql/opt/props/physical" "github.com/cockroachdb/cockroach/pkg/util/treeprinter" @@ -29,6 +30,9 @@ const ( // FmtPretty performs a breadth-first topological sort on the memo groups, // and shows the root group at the top of the memo. FmtPretty FmtFlags = iota + // FmtCycle formats a memo like FmtPretty, but attempts to detect a cycle in + // the memo and include it in the formatted output. + FmtCycle ) type group struct { @@ -71,8 +75,8 @@ func (mf *memoFormatter) format() string { desc = "optimized" } tpRoot := tp.Childf( - "memo (%s, ~%dKB, required=%s)", - desc, mf.o.mem.MemoryEstimate()/1024, m.RootProps(), + "memo (%s, ~%dKB, required=%s%s)", + desc, mf.o.mem.MemoryEstimate()/1024, m.RootProps(), mf.formatCycles(), ) for i, e := range mf.groups { @@ -317,3 +321,57 @@ func firstExpr(expr opt.Expr) opt.Expr { } return expr } + +func (mf *memoFormatter) formatCycles() string { + m := mf.o.mem + var cd cycle.Detector + + if mf.flags != FmtCycle { + return "" + } + + addExpr := func(from cycle.Vertex, e opt.Expr) { + for i := 0; i < e.ChildCount(); i++ { + child := e.Child(i) + if opt.IsListItemOp(child) { + child = child.Child(0) + } + to := cycle.Vertex(mf.group(child)) + cd.AddEdge(from, to) + } + } + + addGroup := func(from cycle.Vertex, first memo.RelExpr) { + for member := first; member != nil; member = member.NextExpr() { + addExpr(from, member) + } + } + + // Add an edge to the cycle detector from a group to each of the groups it + // references. + for i, e := range mf.groups { + from := cycle.Vertex(i) + rel, ok := e.first.(memo.RelExpr) + if !ok { + addExpr(from, e.first) + continue + } + addGroup(from, rel) + } + + // Search for a cycle from the root expression group. + root := cycle.Vertex(mf.group(m.RootExpr())) + if cyclePath, ok := cd.FindCycleStartingAtVertex(root); ok { + mf.buf.Reset() + mf.buf.WriteString(", cycle=[") + for i, group := range cyclePath { + if i > 0 { + mf.buf.WriteString("->") + } + fmt.Fprintf(mf.buf, "G%d", group+1) + } + mf.buf.WriteString("]") + return mf.buf.String() + } + return "" +} diff --git a/pkg/sql/opt/xform/optimizer.go b/pkg/sql/opt/xform/optimizer.go index 916fa2283d82..2193d6829f34 100644 --- a/pkg/sql/opt/xform/optimizer.go +++ b/pkg/sql/opt/xform/optimizer.go @@ -470,7 +470,7 @@ func (o *Optimizer) optimizeGroup(grp memo.RelExpr, required *physical.Required) // times, there is likely a cycle in the memo. To avoid a stack // overflow, throw an internal error. The formatted memo is included as // an error detail to aid in debugging the cycle. - mf := makeMemoFormatter(o, FmtPretty) + mf := makeMemoFormatter(o, FmtCycle) panic(errors.WithDetail( errors.AssertionFailedf( "memo group optimization passes surpassed limit of %v; "+ diff --git a/pkg/sql/opt/xform/rules/cycle.opt b/pkg/sql/opt/xform/rules/cycle.opt index 29f2559b00ef..2196d90ae8c9 100644 --- a/pkg/sql/opt/xform/rules/cycle.opt +++ b/pkg/sql/opt/xform/rules/cycle.opt @@ -10,3 +10,20 @@ (MemoCycleTestRel $input:* $filters:*) => (MemoCycleTestRel (MemoCycleTestRel $input $filters) $filters) + +# MemoCycleTestRelRuleFilter is similar to MemoCycleTestRelRule, but creates +# a cycle in the memo through a filter expression. +[MemoCycleTestRelRuleFilter, Explore] +(MemoCycleTestRel $input:* $filters:*) +=> +(Select + $input + [ + (FiltersItem + (Exists + (MemoCycleTestRel $input $filters) + (EmptySubqueryPrivate) + ) + ) + ] +) diff --git a/pkg/sql/opt/xform/testdata/rules/cycle b/pkg/sql/opt/xform/testdata/rules/cycle index 7d9645917447..7d7aef7b0a43 100644 --- a/pkg/sql/opt/xform/testdata/rules/cycle +++ b/pkg/sql/opt/xform/testdata/rules/cycle @@ -6,10 +6,26 @@ CREATE TABLE ab ( ) ---- +exec-ddl +CREATE TABLE uv ( + u INT PRIMARY KEY, + v INT, + INDEX (v) +) +---- + +exec-ddl +CREATE TABLE xy ( + x INT PRIMARY KEY, + y INT, + INDEX (y) +) +---- + # This test ensures that a memo cycle does not cause a stack overflow. Instead, # the cycle is detected and the optimizer throws an internal error. The cycle is # created by the test-only exploration rule MemoCycleTestRelRule. -expropt +expropt disable=MemoCycleTestRelRuleFilter (MemoCycleTestRel (Scan [ (Table "ab") (Cols "a,b") ]) [ (Eq (Var "b") (Const 1 "int")) ] @@ -17,7 +33,7 @@ expropt ---- error: memo group optimization passes surpassed limit of 100000; there may be a cycle in the memo details: -memo (not optimized, ~3KB, required=[]) +memo (not optimized, ~3KB, required=[], cycle=[G1->G1]) ├── G1: (memo-cycle-test-rel G2 G3) (memo-cycle-test-rel G1 G3) ├── G2: (scan ab,cols=(1,2)) (scan ab@ab_b_idx,cols=(1,2)) │ └── [] @@ -27,3 +43,66 @@ memo (not optimized, ~3KB, required=[]) ├── G4: (eq G5 G6) ├── G5: (variable b) └── G6: (const 1) + +expropt disable=MemoCycleTestRelRuleFilter +(LeftJoin + (Scan [ (Table "ab") (Cols "a,b") ]) + (LeftJoin + (MemoCycleTestRel + (Scan [ (Table "uv") (Cols "u,v") ]) + [ (Eq (Var "v") (Const 1 "int")) ] + ) + (Scan [ (Table "xy") (Cols "x,y") ]) + [ ] + [ ] + ) + [ ] + [ ] +) +---- +error: memo group optimization passes surpassed limit of 100000; there may be a cycle in the memo +details: +memo (not optimized, ~15KB, required=[], cycle=[G1->G3->G5->G5]) + ├── G1: (left-join G2 G3 G4) + ├── G2: (scan ab,cols=()) (scan ab@ab_b_idx,cols=()) + │ └── [] + │ ├── best: (scan ab,cols=()) + │ └── cost: 1044.22 + ├── G3: (left-join G5 G6 G4) + ├── G4: (filters) + ├── G5: (memo-cycle-test-rel G7 G8) (memo-cycle-test-rel G5 G8) + ├── G6: (scan xy,cols=()) + ├── G7: (scan uv,cols=(5,6)) (scan uv@uv_v_idx,cols=(5,6)) + │ └── [] + │ ├── best: (scan uv,cols=(5,6)) + │ └── cost: 1064.42 + ├── G8: (filters G9) + ├── G9: (eq G10 G11) + ├── G10: (variable v) + └── G11: (const 1) + +# Ensure that a cycle through a filter can be detected. +expropt disable=MemoCycleTestRelRule +(MemoCycleTestRel + (Scan [ (Table "uv") (Cols "u,v") ]) + [ (Eq (Var "v") (Const 1 "int")) ] +) +---- +error: memo group optimization passes surpassed limit of 100000; there may be a cycle in the memo +details: +memo (not optimized, ~4KB, required=[], cycle=[G1->G4->G6->G9->G1]) + ├── G1: (memo-cycle-test-rel G2 G3) (select G2 G4) + ├── G2: (scan uv,cols=(1,2)) (scan uv@uv_v_idx,cols=(1,2)) + │ ├── [limit hint: 1000.00] + │ │ ├── best: (scan uv,cols=(1,2)) + │ │ └── cost: 1054.02 + │ └── [] + │ ├── best: (scan uv,cols=(1,2)) + │ └── cost: 1064.42 + ├── G3: (filters G5) + ├── G4: (filters G6) + ├── G5: (eq G7 G8) + ├── G6: (exists G9 &{ 0 unknown true}) + ├── G7: (variable v) + ├── G8: (const 1) + └── G9: (limit G1 G8)