4516cd09c5
When dealing with a formal argument in a function argument pattern that has a default expression, there are two different things that can happen at runtime: Either we select its value from the passed attribute successfully or we need to use the default expression. Both of these may be thunks and both of these may need finalisers. However, in the former case this is taken care of elsewhere, the value will always be finalised already if necessary. In the latter case we may need to finalise the thunk resulting from the default expression. However, the thunk corresponding to the expression may never end up in the local's stack slot. Since finalisation goes by stack slot (and not constants), we need to prevent a case where we don't fall back to the default expression, but finalise anyways. Previously, we worked around this by making `OpFinalise` ignore non-thunks. Since finalisation of already evaluated thunks still crashed, the faulty compilation of function pattern arguments could still cause a crash. As a new approach, we reinstate the old behavior of `OpFinalise` to crash whenever encountering something that is either not a thunk or doesn't need finalisation. This can also help catching (similar) miscompilations in the future. To then prevent the crash, we need to track whether we have fallen back or not at runtime. This is done using an additional phantom on the stack that holds a new `FinaliseRequest` value. When it comes to finalisation we check this value and conditionally execute `OpFinalise` based on its value. Resolves b/261 and b/265 (partially). Change-Id: Ic04fb80ec671a2ba11fa645090769c335fb7f58b Reviewed-on: https://cl.tvl.fyi/c/depot/+/8705 Reviewed-by: tazjin <tazjin@tvl.su> Tested-by: BuildkiteCI Autosubmit: sterni <sternenseemann@systemli.org>
140 lines
6.3 KiB
Markdown
140 lines
6.3 KiB
Markdown
Known Optimisation Potential
|
|
============================
|
|
|
|
There are several areas of the Tvix evaluator code base where
|
|
potentially large performance gains can be achieved through
|
|
optimisations that we are already aware of.
|
|
|
|
The shape of most optimisations is that of moving more work into the
|
|
compiler to simplify the runtime execution of Nix code. This leads, in
|
|
some cases, to drastically higher complexity in both the compiler
|
|
itself and in invariants that need to be guaranteed between the
|
|
runtime and the compiler.
|
|
|
|
For this reason, and because we lack the infrastructure to adequately
|
|
track their impact (WIP), we have not yet implemented these
|
|
optimisations, but note the most important ones here.
|
|
|
|
* Use "open upvalues" [hard]
|
|
|
|
Right now, Tvix will immediately close over all upvalues that are
|
|
created and clone them into the `Closure::upvalues` array.
|
|
|
|
Instead of doing this, we can statically determine most locals that
|
|
are closed over *and escape their scope* (similar to how the
|
|
`compiler::scope::Scope` struct currently tracks whether locals are
|
|
used at all).
|
|
|
|
If we implement the machinery to track this, we can implement some
|
|
upvalues at runtime by simply sticking stack indices in the upvalue
|
|
array and only copy the values where we know that they escape.
|
|
|
|
* Avoid `with` value duplication [easy]
|
|
|
|
If a `with` makes use of a local identifier in a scope that can not
|
|
close before the with (e.g. not across `LambdaCtx` boundaries), we
|
|
can avoid the allocation of the phantom value and duplication of the
|
|
`NixAttrs` value on the stack. In this case we simply push the stack
|
|
index of the known local.
|
|
|
|
* Multiple attribute selection [medium]
|
|
|
|
An instruction could be introduced that avoids repeatedly pushing an
|
|
attribute set to/from the stack if multiple keys are being selected
|
|
from it. This occurs, for example, when inheriting from an attribute
|
|
set or when binding function formals.
|
|
|
|
* Split closure/function representation [easy]
|
|
|
|
Functions have fewer fields that need to be populated at runtime and
|
|
can directly use the `value::function::Lambda` representation where
|
|
possible.
|
|
|
|
* Apply `compiler::optimise_select` to other set operations [medium]
|
|
|
|
In addition to selects, statically known attribute resolution could
|
|
also be used for things like `?` or `with`. The latter might be a
|
|
little more complicated but is worth investigating.
|
|
|
|
* Inline fully applied builtins with equivalent operators [medium]
|
|
|
|
Some `builtins` have equivalent operators, e.g. `builtins.sub`
|
|
corresponds to the `-` operator, `builtins.hasAttr` to the `?`
|
|
operator etc. These operators additionally compile to a primitive
|
|
VM opcode, so they should be just as cheap (if not cheaper) as
|
|
a builtin application.
|
|
|
|
In case the compiler encounters a fully applied builtin (i.e.
|
|
no currying is occurring) and the `builtins` global is unshadowed,
|
|
it could compile the equivalent operator bytecode instead: For
|
|
example, `builtins.sub 20 22` would be compiled as `20 - 22`.
|
|
This would ensure that equivalent `builtins` can also benefit
|
|
from special optimisations we may implement for certain operators
|
|
(in the absence of currying). E.g. we could optimise access
|
|
to the `builtins` attribute set which a call to
|
|
`builtins.getAttr "foo" builtins` should also profit from.
|
|
|
|
* Avoid nested `VM::run` calls [hard]
|
|
|
|
Currently when encountering Nix-native callables (thunks, closures)
|
|
the VM's run loop will nest and return the value of the nested call
|
|
frame one level up. This makes the Rust call stack almost mirror the
|
|
Nix call stack, which is usually undesirable.
|
|
|
|
It is possible to detect situations where this is avoidable and
|
|
instead set up the VM in such a way that it continues and produces
|
|
the desired result in the same run loop, but this is kind of tricky
|
|
to get right - especially while other parts are still in flux.
|
|
|
|
For details consult the commit with Gerrit change ID
|
|
`I96828ab6a628136e0bac1bf03555faa4e6b74ece`, in which the initial
|
|
attempt at doing this was reverted.
|
|
|
|
* Avoid thunks if only identifier closing is required [medium]
|
|
|
|
Some constructs, like `with`, mostly do not change runtime behaviour
|
|
if thunked. However, they are wrapped in thunks to ensure that
|
|
deferred identifiers are resolved correctly.
|
|
|
|
This can be avoided, as we statically analyse the scope and should
|
|
be able to tell whether any such logic was required.
|
|
|
|
* Intern literals [easy]
|
|
|
|
Currently, the compiler emits a separate entry in the constant
|
|
table for each literal. So the program `1 + 1 + 1` will have
|
|
three entries in its `Chunk::constants` instead of only one.
|
|
|
|
* Do some list and attribute set operations in place [hard]
|
|
|
|
Algorithms that can not do a lot of work inside `builtins` like `map`,
|
|
`filter` or `foldl'` usually perform terribly if they use data structures like
|
|
lists and attribute sets.
|
|
|
|
`builtins` can do work in place on a copy of a `Value`, but naïvely expressed
|
|
recursive algorithms will usually use `//` and `++` to do a single change to a
|
|
`Value` at a time, requiring a full copy of the data structure each time.
|
|
It would be a big improvement if we could do some of these operations in place
|
|
without requiring a new copy.
|
|
|
|
There are probably two approaches: We could determine statically if a value is
|
|
reachable from elsewhere and emit a special in place instruction if not. An
|
|
easier alternative is probably to rely on reference counting at runtime: If no
|
|
other reference to a value exists, we can extend the list or update the
|
|
attribute set in place.
|
|
|
|
An **alternative** to this is using [persistent data
|
|
structures](https://en.wikipedia.org/wiki/Persistent_data_structure) or at the
|
|
very least [immutable data structures](https://docs.rs/im/latest/im/) that can
|
|
be copied more efficiently than the stock structures we are using at the
|
|
moment.
|
|
|
|
* Skip finalising unfinalised thunks or non-thunks instead of crashing [easy]
|
|
|
|
Currently `OpFinalise` crashes the VM if it is called on values that don't
|
|
need to be finalised. This helps catching miscompilations where `OpFinalise`
|
|
operates on the wrong `StackIdx`. In the case of function argument patterns,
|
|
however, this means extra VM stack and instruction overhead for dynamically
|
|
determining if finalisation is necessary or not. This wouldn't be necessary
|
|
if `OpFinalise` would just noop on any values that don't need to be finalised
|
|
(anymore).
|