Skip to content

Commit

Permalink
EVM: finalize: fix handling of symbolic returndata
Browse files Browse the repository at this point in the history
We produced a `Partial` node with a message "runtime code cannot have an
abstract length" if we could not determine a concrete size for
returndata when finalizing a transaction. This changes gates that check
so it is only made if we are actually in a create transaction (which is
the only context where it is valid).
  • Loading branch information
d-xo committed Oct 7, 2024
1 parent 016d063 commit fd3d494
Show file tree
Hide file tree
Showing 3 changed files with 27 additions and 13 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Fixed bug in SMT encoding that caused empty and all-zero byte arrays to be considered equal
and hence lead to false negatives through trivially UNSAT SMT expressions
- Respect --smt-timeout in equivalence checking
- Fixed the handling of returndata with an abstract size during transaction finalization

## [0.53.0] - 2024-02-23

Expand Down
25 changes: 12 additions & 13 deletions src/EVM.hs
Original file line number Diff line number Diff line change
Expand Up @@ -1504,19 +1504,18 @@ finalize = do
creation <- use (#tx % #isCreate)
createe <- use (#state % #contract)
createeExists <- (Map.member createe) <$> use (#env % #contracts)
let onContractCode contractCode =
when (creation && createeExists) $ replaceCode createe contractCode
case output of
ConcreteBuf bs ->
onContractCode $ RuntimeCode (ConcreteRuntimeCode bs)
_ ->
case Expr.toList output of
Nothing -> do
state <- use #state
partial $
UnexpectedSymbolicArg pc' (getOpName state) "runtime code cannot have an abstract length" (wrap [output])
Just ops ->
onContractCode $ RuntimeCode (SymbolicRuntimeCode ops)
when (creation && createeExists) $
case output of
ConcreteBuf bs ->
replaceCode createe (RuntimeCode (ConcreteRuntimeCode bs))
_ ->
case Expr.toList output of
Nothing -> do
state <- use #state
partial $
UnexpectedSymbolicArg pc' (getOpName state) "runtime code cannot have an abstract length" (wrap [output])
Just ops ->
replaceCode createe (RuntimeCode (SymbolicRuntimeCode ops))
_ ->
internalError "Finalising an unfinished tx."

Expand Down
14 changes: 14 additions & 0 deletions test/test.hs
Original file line number Diff line number Diff line change
Expand Up @@ -1834,6 +1834,20 @@ tests = testGroup "hevm"
|]
(_, [Qed _]) <- withSolvers Z3 1 Nothing $ \s -> checkAssert s defaultPanicCodes c (Just (Sig "f(uint256)" [AbiUIntType 256])) [] defaultVeriOpts
putStrLnM "this should always be true, due to bitwise OR with positive value"
,
test "abstract-returndata-size" $ do
Just c <- solcRuntime "C"
[i|
contract C {
function f(uint256 x) public pure {
assembly {
return(0, x)
}
}
}
|]
expr <- withSolvers Z3 1 Nothing $ \s -> getExpr s c (Just (Sig "f(uint256)" [])) [] defaultVeriOpts
assertBoolM "The expression is partial" $ not $ Expr.containsNode isPartial expr
,
-- CopySlice check
-- uses identity precompiled contract (0x4) to copy memory
Expand Down

0 comments on commit fd3d494

Please sign in to comment.