Skip to content

Commit

Permalink
execinfra: correctly reset the eval.Context in InternalClose
Browse files Browse the repository at this point in the history
This commit fixes the way we restore the `eval.Context.Context` in
`ProcessorBase.InternalClose`. Whenever a processor is started with
`StartInternal`, we derive a tracing span, and in order for all things
related to that processor (including the evaluation of the builtin
functions) to be correctly attributed to the tracing span of the processor,
we update the eval context with the new context. Most processors get
a fresh copy of the eval context via `FlowCtx.NewEvalCtx`, but a couple
(that are deemed to be "safe") do not do that (since they don't actually
evaluate any expressions) and, instead, reuse the "main" eval context.
Then, when the processor is closed, we want to reset the eval context to
its original form, and we had a bug here - we would use the context that
was passed into `StartInternal` rather than the "original" one, so we
could leak the context with an already finished tracing span outside.
This is now fixed.

This bug could occur for example when we have a `FlowCoordinator` (which
doesn't create a copy of the eval context) at the root of the main query
and then there are some postqueries - the eval context used by the
postqueries would refer to the context from the main query with
a finished tracing span. However, the bug was mitigated by explicitly
setting the context before running each postquery (due to an unrelated
old issue).

One could argue that since the processors modify the eval context, they
should always use a copy of the eval context. That would be unfortunate
since it would lead to increased allocations with the only modification
being this context-mangering business. The fix in this commit seems more
appropriate (both philosophically speaking and performance-wise).

Release note: None
  • Loading branch information
yuzefovich committed Oct 2, 2022
1 parent 3a9f5b8 commit 2aef9c7
Show file tree
Hide file tree
Showing 2 changed files with 5 additions and 12 deletions.
6 changes: 5 additions & 1 deletion pkg/sql/execinfra/processorsbase.go
Original file line number Diff line number Diff line change
Expand Up @@ -358,6 +358,9 @@ type ProcessorBaseNoHelper struct {
// origCtx is the context from which ctx was derived. InternalClose() resets
// ctx to this.
origCtx context.Context
// evalOrigCtx is the original context that was stored in the eval.Context.
// InternalClose() uses it to correctly reset the eval.Context.
evalOrigCtx context.Context

State procState

Expand Down Expand Up @@ -862,6 +865,7 @@ func (pb *ProcessorBaseNoHelper) StartInternal(ctx context.Context, name string)
pb.span.SetTag(execinfrapb.ProcessorIDTagKey, attribute.IntValue(int(pb.ProcessorID)))
}
}
pb.evalOrigCtx = pb.EvalCtx.Context
pb.EvalCtx.Context = pb.Ctx
return pb.Ctx
}
Expand Down Expand Up @@ -893,7 +897,7 @@ func (pb *ProcessorBaseNoHelper) InternalClose() bool {
// Reset the context so that any incidental uses after this point do not
// access the finished span.
pb.Ctx = pb.origCtx
pb.EvalCtx.Context = pb.origCtx
pb.EvalCtx.Context = pb.evalOrigCtx
return true
}

Expand Down
11 changes: 0 additions & 11 deletions pkg/sql/instrumentation.go
Original file line number Diff line number Diff line change
Expand Up @@ -710,18 +710,7 @@ func (ih *instrumentationHelper) SetIndexRecommendations(
isInternal,
) {
f := opc.optimizer.Factory()
// EvalContext() has the context with the already closed span, so we
// need to update with the current context.
// The replacement of the context here isn't ideal, but the current
// implementation of contexts would need to change
// significantly to accommodate this case.
evalCtx := opc.p.EvalContext()
oldCtx := evalCtx.Context
evalCtx.Context = ctx
defer func() {
evalCtx.Context = oldCtx
}()

f.Init(evalCtx, &opc.catalog)
f.FoldingControl().AllowStableFolds()
bld := optbuilder.New(ctx, &opc.p.semaCtx, evalCtx, &opc.catalog, f, opc.p.stmt.AST)
Expand Down

0 comments on commit 2aef9c7

Please sign in to comment.