Taylor — A Performant Language for the Ethereum Virtual Machine (EVM)
Read it on GitHub: https://github.com/loredanacirstea/articles/blob/master/articles/Taylor-EVM_compiled.md
Taylor has started as an interpreted functional language for the EVM, being born out of our Pipeline project. Pipeline is a textual and visual language for flow-based programming, where function inputs and outputs can be connected to create complex atomic operations. These atomic operations are interpreted by our Pipeline graph interpreter smart contract or the graphs can be deployed as standalone, smart contracts, optimized for smaller transaction gas costs.
With Taylor, we created a Turing complete language, that can be interpreted by the EVM, with operations that can run both on the stack or on memory frames. This allows Taylor to do recursive operations that are not possible now on Solidity or Yul (the language that Solidity is based on) (https://github.com/ethereum/solidity/issues/9622).
For example, Taylor can interpret this recursive implementation of the Fibonacci series with a higher number of iterations (bounded by gas limit instead of stack limit), because, unlike Solidity or Yul, it can leave the stack empty between function calls:
(def! fibonacci (fn* (n) (if (or (eq n 1) (eq n 2)) 1 (add(fibonacci (sub n 1)) (fibonacci (sub n 2)) ) )))
While building the Taylor interpreter and other EVM interpreters, neither Solidity, nor Yul was flexible enough to build them efficiently. We created a macro language exactly for this purpose — mASM. mASM produces EVM assembly.
Today we are announcing the next step for Taylor: as a compiled language.
taylor-evm
is an intermediate language that gives developers access to all EVM instructions and to several out-of-the-box compositions. It compiles to EVM bytecode, which can be deployed as a smart contract. It supersedes mASM.
Its canonical form looks like (add 34 (add 3 4))
. But, it also has an indentation-based form with optional parenthesis:
add
34
add
3
4
or
add
34
(add 3 4)
You can directly manipulate the stack:
(list
0x02
0x04
(dup 2)
(swap 1)
(add 5))
You can have access to change control flow:
(if (eq 3 4) (add 0x03 0x04) (add 0x05 0x06))switch 0x4444
(case 0x1000 (add 0x1 0x1))
(case 0x4444 (add 0x2 0x2))
(case 0x2000 (add 0x3 0x3))
(default 0x55)
You can use loops. The following is the efficient, non-recursive way to implement the Fibonacci function (fib):
(list
0x00
0x01
(loop 1 8
(list
(add (dup 4) (dup 4))
(swap 3)
(swap 4)
(pop)
)
)
(mem-store 0x00)
(return 0x00 0x20)
)
Hence, the Fibonacci function can be written as a smart contract like this:
define fib ["uint256"] ["uint256"]
if (eq 0x0 (dup 1))
(pass 0x0)
if (eq 0x1 (dup 1))
(pass 0x01)
list
add
(fib (sub (dup 4) 1 ))
(fib (sub (dup 3) 2 ))
(swap 1)
(pop)to-deploy
list
(include "fib")
(fib (calldata-load 0x0))
(mem-store 0x00)
(return 0x00 0x20)
Or, with indentation-only, like this:
if
eq
0x1
dup
1
pass
0x01
list
add
fib
sub
dup
4
1
fib
sub
dup
3
2
swap
1
pop
The generated assembly code (asm) is:
0x4b
0x0d
0x00
codecopy
0x4b
0x00
return
stop
after_do_fib_14
0x00
calldataload
fib
jumpafter_do_fib_14:
0x00
mstore
0x20
0x00
returnfib:
dup1
0x00
eq
ifsource_9
jumpielsesource_9:
dup1
0x01
eq
ifsource_8
jumpielsesource_8:
after_do_fib_13
0x02
dup3
sub
fib
jumpafter_do_fib_13:
after_do_fib_12
0x01
dup4
sub
fib
jumpafter_do_fib_12:
add
swap1
pop
endif_8
jumpifsource_8:
fib_end
jumpendif_8:
endif_9
jumpifsource_9:
fib_end
jumpendif_9:fib_end:
swap1
jump
Comparison
The above Fibonacci function contract is equivalent to the following contract in Yul:
object "TestFib" {
code {
datacopy(0, dataoffset("Runtime"), datasize("Runtime"))
return(0, datasize("Runtime"))
}
object "Runtime" {
code {
let input := calldataload(0x0)
let res := fib(input)
mstore(0x0, res)
return(0x0, 0x20)
function fib(nth) -> result {
switch nth
case 0 { result := 0 }
case 1 { result := 1 }
default {
result := add(fib(sub(nth, 2)), fib(sub(nth, 1)))
}}}}}
At this stage, these are the stats from our own Cometh EVM debugger when calling fib(16)
, fib(18)
, fib(19)
, fib(30)
:
The performance of the compiled Taylor relative to Yul is a consistent 94% on gas and 89.7% on execution steps with a more important margin on stack usage.
Compiled vs. Interpreted
The compiled version of Taylor fulfills the usecases that Solidity and Yul fulfill — optimizing for transaction execution in the EVM.
The interpreted version of Taylor optimizes for flexibility, upgradability, and off-chain processing, verified by the EVM.
As far as we know Solidity’s internals, the compiled code is generated by Yul: it is based on Yul. If we managed to outperform Yul, it is the only comparison we need: Solidity will be performing equal to or less than Yul. In order to achieve maximum performance, we had to use EVM assembly language (ASM) directly and for this purpose, we had to develop our own macro language: mASM. And the interpreted Taylor is based on mAsm while the compiled Taylor is built (using itself) straight from ASM. Taylor is presently used as a language interpreted by the EVM, compiled for the EVM, and interpreted by JavaScript. Therefore a developer could use the same language to develop web3 dApps. However, amongst the 3 flavors of Taylor, the compiled one (that we demoed today) is the most under-developed: work on stack and memory management is underway with the intent to outperform our present implementation.
Video Demo
This technology was created by and for volunteers at The Laurel Project. You may join this tech effort: https://forms.gle/WmSaSbxhHiiA2qZV7. https://www.reddit.com/r/provable_laurel/.