-
Notifications
You must be signed in to change notification settings - Fork 3.8k
/
memo.go
497 lines (453 loc) · 21.1 KB
/
memo.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
// Copyright 2018 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 memo
import (
"context"
"github.com/cockroachdb/cockroach/pkg/sql/opt"
"github.com/cockroachdb/cockroach/pkg/sql/opt/cat"
"github.com/cockroachdb/cockroach/pkg/sql/opt/props"
"github.com/cockroachdb/cockroach/pkg/sql/opt/props/physical"
"github.com/cockroachdb/cockroach/pkg/sql/sem/tree"
"github.com/cockroachdb/cockroach/pkg/util/duration"
"github.com/cockroachdb/cockroach/pkg/util/timeutil/pgdate"
"github.com/cockroachdb/errors"
"github.com/cockroachdb/redact"
)
// Memo is a data structure for efficiently storing a forest of query plans.
// Conceptually, the memo is composed of a numbered set of equivalency classes
// called groups where each group contains a set of logically equivalent
// expressions. Two expressions are considered logically equivalent if:
//
// 1. They return the same number and data type of columns. However, order and
// naming of columns doesn't matter.
// 2. They return the same number of rows, with the same values in each row.
// However, order of rows doesn't matter.
//
// The different expressions in a single group are called memo expressions
// (memo-ized expressions). The children of a memo expression can themselves be
// part of memo groups. Therefore, the memo forest is composed of every possible
// combination of parent expression with its possible child expressions,
// recursively applied.
//
// Memo expressions can be relational (e.g. join) or scalar (e.g. <). Operators
// are always both logical (specify results) and physical (specify results and
// a particular implementation). This means that even a "raw" unoptimized
// expression tree can be executed (naively). Both relational and scalar
// operators are uniformly represented as nodes in memo expression trees, which
// facilitates tree pattern matching and replacement. However, because scalar
// expression memo groups never have more than one expression, scalar
// expressions can use a simpler representation.
//
// Because memo groups contain logically equivalent expressions, all the memo
// expressions in a group share the same logical properties. However, it's
// possible for two logically equivalent expression to be placed in different
// memo groups. This occurs because determining logical equivalency of two
// relational expressions is too complex to perform 100% correctly. A
// correctness failure (i.e. considering two expressions logically equivalent
// when they are not) results in invalid transformations and invalid plans.
// But placing two logically equivalent expressions in different groups has a
// much gentler failure mode: the memo and transformations are less efficient.
// Expressions within the memo may have different physical properties. For
// example, a memo group might contain both hash join and merge join
// expressions which produce the same set of output rows, but produce them in
// different orders.
//
// Expressions are inserted into the memo by the factory, which ensure that
// expressions have been fully normalized before insertion (see the comment in
// factory.go for more details). A new group is created only when unique
// normalized expressions are created by the factory during construction or
// rewrite of the tree. Uniqueness is determined by "interning" each expression,
// which means that multiple equivalent expressions are mapped to a single
// in-memory instance. This allows interned expressions to be checked for
// equivalence by simple pointer comparison. For example:
//
// SELECT * FROM a, b WHERE a.x = b.x
//
// After insertion into the memo, the memo would contain these six groups, with
// numbers substituted for pointers to the normalized expression in each group:
//
// G6: [inner-join [G1 G2 G5]]
// G5: [eq [G3 G4]]
// G4: [variable b.x]
// G3: [variable a.x]
// G2: [scan b]
// G1: [scan a]
//
// Each leaf expressions is interned by hashing its operator type and any
// private field values. Expressions higher in the tree can then rely on the
// fact that all children have been interned, and include their pointer values
// in its hash value. Therefore, the memo need only hash the expression's fields
// in order to determine whether the expression already exists in the memo.
// Walking the subtree is not necessary.
//
// The normalizing factory will never add more than one expression to a memo
// group. But the explorer does add denormalized expressions to existing memo
// groups, since oftentimes one of these equivalent, but denormalized
// expressions will have a lower cost than the initial normalized expression
// added by the factory. For example, the join commutativity transformation
// expands the memo like this:
//
// G6: [inner-join [G1 G2 G5]] [inner-join [G2 G1 G5]]
// G5: [eq [G3 G4]]
// G4: [variable b.x]
// G3: [variable a.x]
// G2: [scan b]
// G1: [scan a]
//
// See the comments in explorer.go for more details.
type Memo struct {
// metadata provides information about the columns and tables used in this
// particular query.
metadata opt.Metadata
// interner interns all expressions in the memo, ensuring that there is at
// most one instance of each expression in the memo.
interner interner
// logPropsBuilder is inlined in the memo so that it can be reused each time
// scalar or relational properties need to be built.
logPropsBuilder logicalPropsBuilder
// rootExpr is the root expression of the memo expression forest. It is set
// via a call to SetRoot. After optimization, it is set to be the root of the
// lowest cost tree in the forest.
rootExpr opt.Expr
// rootProps are the physical properties required of the root memo expression.
// It is set via a call to SetRoot.
rootProps *physical.Required
// memEstimate is the approximate memory usage of the memo, in bytes.
memEstimate int64
// The following are selected fields from SessionData which can affect
// planning. We need to cross-check these before reusing a cached memo.
// NOTE: If you add new fields here, be sure to add them to the relevant
// fields in explain_bundle.go.
reorderJoinsLimit int
zigzagJoinEnabled bool
useHistograms bool
useMultiColStats bool
localityOptimizedSearch bool
safeUpdates bool
preferLookupJoinsForFKs bool
saveTablesPrefix string
dateStyleEnabled bool
intervalStyleEnabled bool
dateStyle pgdate.DateStyle
intervalStyle duration.IntervalStyle
propagateInputOrdering bool
disallowFullTableScans bool
largeFullScanRows float64
nullOrderedLast bool
costScansWithDefaultColSize bool
testingOptimizerRandomSeed int64
testingOptimizerCostPerturbation float64
testingOptimizerDisableRuleProbability float64
useImprovedDisjunctionStats bool
alwaysUseHistograms bool
// curRank is the highest currently in-use scalar expression rank.
curRank opt.ScalarRank
// curWithID is the highest currently in-use WITH ID.
curWithID opt.WithID
newGroupFn func(opt.Expr)
// disableCheckExpr disables expression validation performed by CheckExpr,
// if the crdb_test build tag is set. If the crdb_test build tag is not set,
// CheckExpr is always a no-op, so disableCheckExpr has no effect. This is
// set to true for the optsteps test command to prevent CheckExpr from
// erring with partially normalized expressions.
disableCheckExpr bool
// WARNING: if you add more members, add initialization code in Init (if
// reusing allocated data structures is desired).
}
// Init initializes a new empty memo instance, or resets existing state so it
// can be reused. It must be called before use (or reuse). The memo collects
// information about the context in which it is compiled from the evalContext
// argument. If any of that changes, then the memo must be invalidated (see the
// IsStale method for more details).
func (m *Memo) Init(evalCtx *tree.EvalContext) {
// This initialization pattern ensures that fields are not unwittingly
// reused. Field reuse must be explicit.
*m = Memo{
metadata: m.metadata,
reorderJoinsLimit: int(evalCtx.SessionData().ReorderJoinsLimit),
zigzagJoinEnabled: evalCtx.SessionData().ZigzagJoinEnabled,
useHistograms: evalCtx.SessionData().OptimizerUseHistograms,
useMultiColStats: evalCtx.SessionData().OptimizerUseMultiColStats,
localityOptimizedSearch: evalCtx.SessionData().LocalityOptimizedSearch,
safeUpdates: evalCtx.SessionData().SafeUpdates,
preferLookupJoinsForFKs: evalCtx.SessionData().PreferLookupJoinsForFKs,
saveTablesPrefix: evalCtx.SessionData().SaveTablesPrefix,
intervalStyleEnabled: evalCtx.SessionData().IntervalStyleEnabled,
dateStyleEnabled: evalCtx.SessionData().DateStyleEnabled,
dateStyle: evalCtx.SessionData().GetDateStyle(),
intervalStyle: evalCtx.SessionData().GetIntervalStyle(),
propagateInputOrdering: evalCtx.SessionData().PropagateInputOrdering,
disallowFullTableScans: evalCtx.SessionData().DisallowFullTableScans,
largeFullScanRows: evalCtx.SessionData().LargeFullScanRows,
nullOrderedLast: evalCtx.SessionData().NullOrderedLast,
costScansWithDefaultColSize: evalCtx.SessionData().CostScansWithDefaultColSize,
testingOptimizerRandomSeed: evalCtx.SessionData().TestingOptimizerRandomSeed,
testingOptimizerCostPerturbation: evalCtx.SessionData().TestingOptimizerCostPerturbation,
testingOptimizerDisableRuleProbability: evalCtx.SessionData().TestingOptimizerDisableRuleProbability,
useImprovedDisjunctionStats: evalCtx.SessionData().OptimizerUseImprovedDisjunctionStats,
alwaysUseHistograms: evalCtx.SessionData().OptimizerAlwaysUseHistograms,
}
m.metadata.Init()
m.logPropsBuilder.init(evalCtx, m)
}
// ResetLogProps resets the logPropsBuilder. It should be used in combination
// with the perturb-cost OptTester flag in order to update the query plan tree
// after optimization is complete with the real computed cost, not the perturbed
// cost.
func (m *Memo) ResetLogProps(evalCtx *tree.EvalContext) {
m.logPropsBuilder.init(evalCtx, m)
}
// NotifyOnNewGroup sets a callback function which is invoked each time we
// create a new memo group.
func (m *Memo) NotifyOnNewGroup(fn func(opt.Expr)) {
m.newGroupFn = fn
}
// IsEmpty returns true if there are no expressions in the memo.
func (m *Memo) IsEmpty() bool {
// Root expression can be nil before optimization and interner is empty after
// exploration, so check both.
return m.interner.Count() == 0 && m.rootExpr == nil
}
// MemoryEstimate returns a rough estimate of the memo's memory usage, in bytes.
// It only includes memory usage that is proportional to the size and complexity
// of the query, rather than constant overhead bytes.
func (m *Memo) MemoryEstimate() int64 {
// Multiply by 2 to take rough account of allocation fragmentation, private
// data, list overhead, properties, etc.
return m.memEstimate * 2
}
// Metadata returns the metadata instance associated with the memo.
func (m *Memo) Metadata() *opt.Metadata {
return &m.metadata
}
// RootExpr returns the root memo expression previously set via a call to
// SetRoot.
func (m *Memo) RootExpr() opt.Expr {
return m.rootExpr
}
// RootProps returns the physical properties required of the root memo group,
// previously set via a call to SetRoot.
func (m *Memo) RootProps() *physical.Required {
return m.rootProps
}
// SetRoot stores the root memo expression when it is a relational expression,
// and also stores the physical properties required of the root group.
func (m *Memo) SetRoot(e RelExpr, phys *physical.Required) {
m.rootExpr = e
if m.rootProps != phys {
m.rootProps = m.InternPhysicalProps(phys)
}
// Once memo is optimized, release reference to the eval context and free up
// the memory used by the interner.
if m.IsOptimized() {
m.logPropsBuilder.clear()
m.interner = interner{}
}
}
// SetScalarRoot stores the root memo expression when it is a scalar expression.
// Used only for testing.
func (m *Memo) SetScalarRoot(scalar opt.ScalarExpr) {
m.rootExpr = scalar
}
// HasPlaceholders returns true if the memo contains at least one placeholder
// operator.
func (m *Memo) HasPlaceholders() bool {
rel, ok := m.rootExpr.(RelExpr)
if !ok {
panic(errors.AssertionFailedf("placeholders only supported when memo root is relational"))
}
return rel.Relational().HasPlaceholder
}
// IsStale returns true if the memo has been invalidated by changes to any of
// its dependencies. Once a memo is known to be stale, it must be ejected from
// any query cache or prepared statement and replaced with a recompiled memo
// that takes into account the changes. IsStale checks the following
// dependencies:
//
// 1. Current database: this can change name resolution.
// 2. Current search path: this can change name resolution.
// 3. Current location: this determines time zone, and can change how time-
// related types are constructed and compared.
// 4. Data source schema: this determines most aspects of how the query is
// compiled.
// 5. Data source privileges: current user may no longer have access to one or
// more data sources.
//
// This function cannot swallow errors and return only a boolean, as it may
// perform KV operations on behalf of the transaction associated with the
// provided catalog, and those errors are required to be propagated.
func (m *Memo) IsStale(
ctx context.Context, evalCtx *tree.EvalContext, catalog cat.Catalog,
) (bool, error) {
// Memo is stale if fields from SessionData that can affect planning have
// changed.
if m.reorderJoinsLimit != int(evalCtx.SessionData().ReorderJoinsLimit) ||
m.zigzagJoinEnabled != evalCtx.SessionData().ZigzagJoinEnabled ||
m.useHistograms != evalCtx.SessionData().OptimizerUseHistograms ||
m.useMultiColStats != evalCtx.SessionData().OptimizerUseMultiColStats ||
m.localityOptimizedSearch != evalCtx.SessionData().LocalityOptimizedSearch ||
m.safeUpdates != evalCtx.SessionData().SafeUpdates ||
m.preferLookupJoinsForFKs != evalCtx.SessionData().PreferLookupJoinsForFKs ||
m.saveTablesPrefix != evalCtx.SessionData().SaveTablesPrefix ||
m.intervalStyleEnabled != evalCtx.SessionData().IntervalStyleEnabled ||
m.dateStyleEnabled != evalCtx.SessionData().DateStyleEnabled ||
m.dateStyle != evalCtx.SessionData().GetDateStyle() ||
m.intervalStyle != evalCtx.SessionData().GetIntervalStyle() ||
m.propagateInputOrdering != evalCtx.SessionData().PropagateInputOrdering ||
m.disallowFullTableScans != evalCtx.SessionData().DisallowFullTableScans ||
m.largeFullScanRows != evalCtx.SessionData().LargeFullScanRows ||
m.nullOrderedLast != evalCtx.SessionData().NullOrderedLast ||
m.costScansWithDefaultColSize != evalCtx.SessionData().CostScansWithDefaultColSize ||
m.testingOptimizerRandomSeed != evalCtx.SessionData().TestingOptimizerRandomSeed ||
m.testingOptimizerCostPerturbation != evalCtx.SessionData().TestingOptimizerCostPerturbation ||
m.testingOptimizerDisableRuleProbability != evalCtx.SessionData().TestingOptimizerDisableRuleProbability ||
m.useImprovedDisjunctionStats != evalCtx.SessionData().OptimizerUseImprovedDisjunctionStats ||
m.alwaysUseHistograms != evalCtx.SessionData().OptimizerAlwaysUseHistograms {
return true, nil
}
// Memo is stale if the fingerprint of any object in the memo's metadata has
// changed, or if the current user no longer has sufficient privilege to
// access the object.
if depsUpToDate, err := m.Metadata().CheckDependencies(ctx, catalog); err != nil {
return true, err
} else if !depsUpToDate {
return true, nil
}
return false, nil
}
// InternPhysicalProps adds the given physical props to the memo if they haven't
// yet been added. If the same props was added previously, then return a pointer
// to the previously added props. This allows interned physical props to be
// compared for equality using simple pointer comparison.
func (m *Memo) InternPhysicalProps(phys *physical.Required) *physical.Required {
// Special case physical properties that require nothing of operator.
if !phys.Defined() {
return physical.MinRequired
}
return m.interner.InternPhysicalProps(phys)
}
// SetBestProps updates the physical properties, provided ordering, and cost of
// a relational expression's memo group (see the relevant methods of RelExpr).
// It is called by the optimizer once it determines the expression in the group
// that is part of the lowest cost tree (for the overall query).
func (m *Memo) SetBestProps(
e RelExpr, required *physical.Required, provided *physical.Provided, cost Cost,
) {
if e.RequiredPhysical() != nil {
if e.RequiredPhysical() != required ||
!e.ProvidedPhysical().Equals(provided) ||
e.Cost() != cost {
panic(errors.AssertionFailedf(
"cannot overwrite %s / %s (%.9g) with %s / %s (%.9g)",
e.RequiredPhysical(),
e.ProvidedPhysical(),
redact.Safe(e.Cost()),
required.String(),
provided.String(), // Call String() so provided doesn't escape.
cost,
))
}
return
}
bp := e.bestProps()
bp.required = required
bp.provided = provided
bp.cost = cost
}
// ResetCost updates the cost of a relational expression's memo group. It
// should *only* be called by Optimizer.RecomputeCost() for testing purposes.
func (m *Memo) ResetCost(e RelExpr, cost Cost) {
e.bestProps().cost = cost
}
// IsOptimized returns true if the memo has been fully optimized.
func (m *Memo) IsOptimized() bool {
// The memo is optimized once the root expression has its physical properties
// assigned.
rel, ok := m.rootExpr.(RelExpr)
return ok && rel.RequiredPhysical() != nil
}
// NextRank returns a new rank that can be assigned to a scalar expression.
func (m *Memo) NextRank() opt.ScalarRank {
m.curRank++
return m.curRank
}
// CopyNextRankFrom copies the next ScalarRank from the other memo.
func (m *Memo) CopyNextRankFrom(other *Memo) {
m.curRank = other.curRank
}
// RequestColStat calculates and returns the column statistic calculated on the
// relational expression.
func (m *Memo) RequestColStat(
expr RelExpr, cols opt.ColSet,
) (colStat *props.ColumnStatistic, ok bool) {
// When SetRoot is called, the statistics builder may have been cleared.
// If this happens, we can't serve the request anymore.
if m.logPropsBuilder.sb.md != nil {
return m.logPropsBuilder.sb.colStat(cols, expr), true
}
return nil, false
}
// RequestColStatTable calculates and returns the column statistic in table
// tabId.
func (m *Memo) RequestColStatTable(
tabID opt.TableID, colSet opt.ColSet,
) (colStat *props.ColumnStatistic, ok bool) {
// When SetRoot is called, the statistics builder may have been cleared.
// If this happens, we can't serve the request anymore.
if m.logPropsBuilder.sb.md != nil {
return m.logPropsBuilder.sb.colStatTable(tabID, colSet), true
}
return nil, false
}
// RowsProcessed calculates and returns the number of rows processed by the
// relational expression. It is currently only supported for joins.
func (m *Memo) RowsProcessed(expr RelExpr) (_ float64, ok bool) {
// When SetRoot is called, the statistics builder may have been cleared.
// If this happens, we can't serve the request anymore.
if m.logPropsBuilder.sb.md != nil {
return m.logPropsBuilder.sb.rowsProcessed(expr), true
}
return 0, false
}
// NextWithID returns a not-yet-assigned identifier for a WITH expression.
func (m *Memo) NextWithID() opt.WithID {
m.curWithID++
return m.curWithID
}
// Detach is used when we detach a memo that is to be reused later (either for
// execbuilding or with AssignPlaceholders). New expressions should no longer be
// constructed in this memo.
func (m *Memo) Detach() {
m.interner = interner{}
// It is important to not hold on to the EvalCtx in the logicalPropsBuilder
// (#57059).
m.logPropsBuilder = logicalPropsBuilder{}
// Clear all column statistics from every relational expression in the memo.
// This is used to free up the potentially large amount of memory used by
// histograms.
var clearColStats func(parent opt.Expr)
clearColStats = func(parent opt.Expr) {
for i, n := 0, parent.ChildCount(); i < n; i++ {
child := parent.Child(i)
clearColStats(child)
}
switch t := parent.(type) {
case RelExpr:
t.Relational().Stats.ColStats = props.ColStatsMap{}
}
}
clearColStats(m.RootExpr())
}
// DisableCheckExpr disables expression validation performed by CheckExpr,
// if the crdb_test build tag is set. If the crdb_test build tag is not set,
// CheckExpr is always a no-op, so DisableCheckExpr has no effect.
func (m *Memo) DisableCheckExpr() {
m.disableCheckExpr = true
}