LibOpCallNP
Contains the call operation. This allows sources to be treated in a
function-like manner. Primarily intended as a way for expression authors to
create reusable logic inline with their expression, in a way that mimics how
words and stack consumption works at the Solidity level.
Similarities between call
and a traditional function:
- The source is called with a set of 0+ inputs.
- The source returns a set of 0+ outputs.
- The source has a fixed number of inputs and outputs.
- When the source executes it has its own stack/scope.
- Sources use lexical scoping rules for named LHS items.
- The source can be called from multiple places.
- The source can
call
other sources. - The source is stateless across calls (although it can use words like get/set to read/write external state).
- The caller and callee have to agree on the number of inputs (but not outputs, see below).
- Generally speaking, the behaviour of a source can be reasoned about
without needing to know the context in which it is called. Which is the
basic requirement for reusability.
Differences between
call
and a traditional function: - The caller defines the number of outputs to be returned, NOT the callee. This is because the caller is responsible for allocating space on the stack for the outputs, and the callee is responsible for providing the outputs. The only limitation is that the caller cannot request more outputs than the callee has available. This means that two calls to the same source can return different numbers of outputs in different contexts.
- The inputs to a source are considered to be the top of the callee's stack from the perspective of the caller. This means that the inputs are eligible to be read as outputs, if the caller chooses to do so.
- The sources are not named, they are identified by their index in the bytecode. Tooling can provide sugar over this but the underlying representation is just an index.
- Sources are not "first class" like functions often are, i.e. they cannot be passed as arguments to other sources or otherwise be treated as values.
- Recursion is not supported. This is because currently there is no laziness
in the interpreter, so a recursive call would result in an infinite loop
unconditionally (even when wrapped in an
if
). This may change in the future. - The memory allocation for a source must be known at compile time.
- There's no way to return early from a source.
The order of inputs and outputs is designed so that the visual representation
of a source call matches the visual representation of a function call. This
requires some reversals of order "under the hood" while copying data around
but it makes the behaviour of
call
more intuitive. Illustrative example:
* Final result */
* a = 2 */
* b = 9 */
a b: call<1 2>(10 5); ten five:, a b: int-div(ten five) 9;
Functions
integrity
function integrity(IntegrityCheckStateNP memory state, Operand operand) internal pure returns (uint256, uint256);
run
The call
word is conceptually very simple. It takes a source index, a
number of outputs, and a number of inputs. It then runs the standard
eval loop for the source, with a starting stack pointer above the inputs,
and then copies the outputs to the calling stack.
function run(InterpreterStateNP memory state, Operand operand, Pointer stackTop) internal view returns (Pointer);