There is interest in enhancing Script’s functionality. Re-enabling opcodes is one of the main feature enhancements that have been proposed. This overflow limitation has been talked about in the Rusty Russell’s Great Script Restoration project, as well as the 64-bit arithmetic delving bitcoin post.
Here are some highlights
I would like to consolidate discussion on the overflow handling part of the Great Script restoration project here.
While not an overflow exactly, you can imagine a world where OP_DIV is re-enabled. This error handling logic could also be used for the case where the user attempts to divide by 0.
Background
CScriptNum is the data type used in bitcoin core to handle numbers in Script. Here is what happens when a stack element is interpreted as a CScriptNum that results in an overflow
operands must be in the range [-2^31 +1 to 2^31 -1]
This means inputs to opcodes such as OP_ADD, OP_SUB, OP_1ADD etc must fall within the stated range. However
results may overflow (and are valid as long as they are not used in a subsequent numeric operation). CScriptNum enforces those semantics by storing results as an int64 and allowing out-of-range values to be returned as a vector of bytes but throwing an exception if arithmetic is done or the result is interpreted as an integer.
Simply put, if my Script is
2^31-1 OP_1ADD
the result on the stack would be 2^31, which is a valid result. If I tried to do an OP_1ADD on 2^31 an this exception would occur
which then would result in this error being propagated to the user
This is because the exception thrown in CScriptNum’s constructor is caught in EvalScript()'s catch clause
What design options are out there?
Here is the design space as I see it, please comment below if I’ve missed any designs that have been deployed
Elements project
In 2022 the Elements project introduced a new set of opcodes to handle the case of overflowing numbers in Script. This upgrade also added 64 bits of precision, 64 bit specific opcodes, and a new encoding format – all of which we will ignore for the purposes of this post.
This soft fork introduces overflow handling when arithmetic computations are performed with the new 64 bit opcodes in elements.
When dealing with overflows, we explicitly return the success bit as a CScriptNum at the top of the stack and the result being the second element from the top. If the operation overflows, first the operands are pushed onto the stack followed by success bit. [a_seconda_top] overflows, the stack state after the operation is [a_seconda_top0] and if the operation does not overflow, the stack state is [res1].
This gives the user flexibility to deal if they script to have overflows using OP_IF\OP_ELSE or OP_VERIFY the success bit if they expect that operation would never fail. When defining the opcodes which can fail, we only define the success path, and assume the overflow behavior as stated above.
AJ Towns provided a critique to this design in the 64-bit arithmetic thread
Bitcoin Cash
Bitcoin cash seems to have re-added disabled opcodes such as OP_MUL and OP_DIV. They have decided to modify the exception handling behavior by setting serror exception if the result overflows. Note, this is different than how bitcoin works currently as we catch an exception thrown by CScriptNum.
Zcash / Litecoin
Both Zcash and Litecoin have retained the original behavior from bitcoin as far as I can tell.
Other bitcoin derivatives?
If you know of any interesting design choices made by other forks of bitcoin, please share them below!
Ethan Heilmans suggested design
with suggestions from AJ Towns
Future research
As mentioned above, CScriptNum currently throws an exception if an overflow occurs. If this is used in conjunction with an op code that interprets the stack top as a CScriptNum, this propagates the exception in to EvalScript() causing the main execution loop to be wrapped in a trycatch block. Questions I have are
What are other opcodes that can result in an exception that are caught by this trycatch block?
What are the performance implications of wrapping the meat and potatoes of EvalScript() in a trycatch block?
While we will likely never be able to fully migrate from the trycatch block, perhaps future soft forks could avoid having this logic?