-- Main file of the hevm CLI program {-# Language DataKinds #-} {-# Language DeriveAnyClass #-} module Main where import EVM (StorageModel(..)) import qualified EVM import EVM.Concrete (createAddress) import qualified EVM.FeeSchedule as FeeSchedule import qualified EVM.Fetch import qualified EVM.Stepper import EVM.SymExec import EVM.Debug import qualified EVM.Expr as Expr import EVM.SMT import qualified EVM.TTY as TTY import EVM.Solidity import EVM.Expr (litAddr) import EVM.Types hiding (word) import EVM.UnitTest (UnitTestOptions, coverageReport, coverageForUnitTestContract, getParametersFromEnvironmentVariables, testNumber, dappTest) import EVM.Dapp (findUnitTests, dappInfo, DappInfo, emptyDapp) import GHC.Natural import EVM.Format (showTraceTree, formatExpr) import Data.Word (Word64) import qualified EVM.Facts as Facts import qualified EVM.Facts.Git as Git import qualified EVM.UnitTest import GHC.Conc import Control.Lens hiding (pre, passing) import Control.Monad (void, when, forM_, unless) import Control.Monad.State.Strict (execStateT, liftIO) import Data.ByteString (ByteString) import Data.List (intercalate, isSuffixOf) import Data.Text (unpack, pack) import Data.Maybe (fromMaybe, mapMaybe) import Data.Version (showVersion) import Data.DoubleWord (Word256) import System.IO (stderr) import System.Directory (withCurrentDirectory, listDirectory) import System.Exit (exitFailure, exitWith, ExitCode(..)) import qualified Data.ByteString as ByteString import qualified Data.ByteString.Char8 as Char8 import qualified Data.ByteString.Lazy as LazyByteString import qualified Data.Map as Map import qualified Data.Text as T import qualified Data.Text.IO as T import qualified Paths_hevm as Paths import Options.Generic as Options import qualified EVM.Transaction -- This record defines the program's command-line options -- automatically via the `optparse-generic` package. data Command w = Symbolic -- Symbolically explore an abstract program, or specialized with specified env & calldata -- vm opts { code :: w ::: Maybe ByteString "Program bytecode" , calldata :: w ::: Maybe ByteString "Tx: calldata" , address :: w ::: Maybe Addr "Tx: address" , caller :: w ::: Maybe Addr "Tx: caller" , origin :: w ::: Maybe Addr "Tx: origin" , coinbase :: w ::: Maybe Addr "Block: coinbase" , value :: w ::: Maybe W256 "Tx: Eth amount" , nonce :: w ::: Maybe W256 "Nonce of origin" , gas :: w ::: Maybe Word64 "Tx: gas amount" , number :: w ::: Maybe W256 "Block: number" , timestamp :: w ::: Maybe W256 "Block: timestamp" , basefee :: w ::: Maybe W256 "Block: base fee" , priorityFee :: w ::: Maybe W256 "Tx: priority fee" , gaslimit :: w ::: Maybe Word64 "Tx: gas limit" , gasprice :: w ::: Maybe W256 "Tx: gas price" , create :: w ::: Bool "Tx: creation" , maxcodesize :: w ::: Maybe W256 "Block: max code size" , prevRandao :: w ::: Maybe W256 "Block: prevRandao" , chainid :: w ::: Maybe W256 "Env: chainId" -- remote state opts , rpc :: w ::: Maybe URL "Fetch state from a remote node" , block :: w ::: Maybe W256 "Block state is be fetched from" , state :: w ::: Maybe String "Path to state repository" , cache :: w ::: Maybe String "Path to rpc cache repository" -- symbolic execution opts , jsonFile :: w ::: Maybe String "Filename or path to dapp build output (default: out/*.solc.json)" , dappRoot :: w ::: Maybe String "Path to dapp project root directory (default: . )" , storageModel :: w ::: Maybe StorageModel "Select storage model: ConcreteS, SymbolicS (default) or InitialS" , sig :: w ::: Maybe Text "Signature of types to decode / encode" , arg :: w ::: [String] "Values to encode" , debug :: w ::: Bool "Run interactively" , getModels :: w ::: Bool "Print example testcase for each execution path" , showTree :: w ::: Bool "Print branches explored in tree view" , showReachableTree :: w ::: Bool "Print only reachable branches explored in tree view" , smttimeout :: w ::: Maybe Natural "Timeout given to SMT solver in milliseconds (default: 60000)" , maxIterations :: w ::: Maybe Integer "Number of times we may revisit a particular branching point" , solver :: w ::: Maybe Text "Used SMT solver: z3 (default) or cvc5" , smtdebug :: w ::: Bool "Print smt queries sent to the solver" , assertions :: w ::: Maybe [Word256] "Comma seperated list of solc panic codes to check for (default: everything except arithmetic overflow)" , askSmtIterations :: w ::: Maybe Integer "Number of times we may revisit a particular branching point before we consult the smt solver to check reachability (default: 5)" , numSolvers :: w ::: Maybe Natural "Number of solver instances to use (default: number of cpu cores)" } | Equivalence -- prove equivalence between two programs { codeA :: w ::: ByteString "Bytecode of the first program" , codeB :: w ::: ByteString "Bytecode of the second program" , sig :: w ::: Maybe Text "Signature of types to decode / encode" , smttimeout :: w ::: Maybe Natural "Timeout given to SMT solver in milliseconds (default: 60000)" , maxIterations :: w ::: Maybe Integer "Number of times we may revisit a particular branching point" , solver :: w ::: Maybe Text "Used SMT solver: z3 (default) or cvc5" , smtoutput :: w ::: Bool "Print verbose smt output" , smtdebug :: w ::: Bool "Print smt queries sent to the solver" , askSmtIterations :: w ::: Maybe Integer "Number of times we may revisit a particular branching point before we consult the smt solver to check reachability (default: 5)" } | Exec -- Execute a given program with specified env & calldata { code :: w ::: Maybe ByteString "Program bytecode" , calldata :: w ::: Maybe ByteString "Tx: calldata" , address :: w ::: Maybe Addr "Tx: address" , caller :: w ::: Maybe Addr "Tx: caller" , origin :: w ::: Maybe Addr "Tx: origin" , coinbase :: w ::: Maybe Addr "Block: coinbase" , value :: w ::: Maybe W256 "Tx: Eth amount" , nonce :: w ::: Maybe W256 "Nonce of origin" , gas :: w ::: Maybe Word64 "Tx: gas amount" , number :: w ::: Maybe W256 "Block: number" , timestamp :: w ::: Maybe W256 "Block: timestamp" , basefee :: w ::: Maybe W256 "Block: base fee" , priorityFee :: w ::: Maybe W256 "Tx: priority fee" , gaslimit :: w ::: Maybe Word64 "Tx: gas limit" , gasprice :: w ::: Maybe W256 "Tx: gas price" , create :: w ::: Bool "Tx: creation" , maxcodesize :: w ::: Maybe W256 "Block: max code size" , prevRandao :: w ::: Maybe W256 "Block: prevRandao" , chainid :: w ::: Maybe W256 "Env: chainId" , debug :: w ::: Bool "Run interactively" , jsontrace :: w ::: Bool "Print json trace output at every step" , trace :: w ::: Bool "Dump trace" , state :: w ::: Maybe String "Path to state repository" , cache :: w ::: Maybe String "Path to rpc cache repository" , rpc :: w ::: Maybe URL "Fetch state from a remote node" , block :: w ::: Maybe W256 "Block state is be fetched from" , jsonFile :: w ::: Maybe String "Filename or path to dapp build output (default: out/*.solc.json)" , dappRoot :: w ::: Maybe String "Path to dapp project root directory (default: . )" } | DappTest -- Run DSTest unit tests { jsonFile :: w ::: Maybe String "Filename or path to dapp build output (default: out/*.solc.json)" , dappRoot :: w ::: Maybe String "Path to dapp project root directory (default: . )" , debug :: w ::: Bool "Run interactively" , jsontrace :: w ::: Bool "Print json trace output at every step" , fuzzRuns :: w ::: Maybe Int "Number of times to run fuzz tests" , depth :: w ::: Maybe Int "Number of transactions to explore" , replay :: w ::: Maybe (Text, ByteString) "Custom fuzz case to run/debug" , rpc :: w ::: Maybe URL "Fetch state from a remote node" , verbose :: w ::: Maybe Int "Append call trace: {1} failures {2} all" , coverage :: w ::: Bool "Coverage analysis" , state :: w ::: Maybe String "Path to state repository" , cache :: w ::: Maybe String "Path to rpc cache repository" , match :: w ::: Maybe String "Test case filter - only run methods matching regex" , covMatch :: w ::: Maybe String "Coverage filter - only print coverage for files matching regex" , solver :: w ::: Maybe Text "Used SMT solver: z3 (default) or cvc5" , smtdebug :: w ::: Bool "Print smt queries sent to the solver" , ffi :: w ::: Bool "Allow the usage of the hevm.ffi() cheatcode (WARNING: this allows test authors to execute arbitrary code on your machine)" , smttimeout :: w ::: Maybe Natural "Timeout given to SMT solver in milliseconds (default: 60000)" , maxIterations :: w ::: Maybe Integer "Number of times we may revisit a particular branching point" , askSmtIterations :: w ::: Maybe Integer "Number of times we may revisit a particular branching point before we consult the smt solver to check reachability (default: 5)" } | Version deriving (Options.Generic) type URL = Text -- For some reason haskell can't derive a -- parseField instance for (Text, ByteString) instance Options.ParseField (Text, ByteString) deriving instance Options.ParseField Word256 deriving instance Options.ParseField [Word256] instance Options.ParseRecord (Command Options.Wrapped) where parseRecord = Options.parseRecordWithModifiers Options.lispCaseModifiers optsMode :: Command Options.Unwrapped -> Mode optsMode x | Main.debug x = Debug | jsontrace x = JsonTrace | otherwise = Run applyCache :: (Maybe String, Maybe String) -> IO (EVM.VM -> EVM.VM) applyCache (state, cache) = let applyState = flip Facts.apply applyCache' = flip Facts.applyCache in case (state, cache) of (Nothing, Nothing) -> do pure id (Nothing, Just cachePath) -> do facts <- Git.loadFacts (Git.RepoAt cachePath) pure $ applyCache' facts (Just statePath, Nothing) -> do facts <- Git.loadFacts (Git.RepoAt statePath) pure $ applyState facts (Just statePath, Just cachePath) -> do cacheFacts <- Git.loadFacts (Git.RepoAt cachePath) stateFacts <- Git.loadFacts (Git.RepoAt statePath) pure $ (applyState stateFacts) . (applyCache' cacheFacts) unitTestOptions :: Command Options.Unwrapped -> SolverGroup -> String -> IO UnitTestOptions unitTestOptions cmd solvers testFile = do let root = fromMaybe "." (dappRoot cmd) srcInfo <- readSolc testFile >>= \case Nothing -> error "Could not read .sol.json file" Just (contractMap, sourceCache) -> pure $ dappInfo root contractMap sourceCache vmModifier <- applyCache (state cmd, cache cmd) params <- getParametersFromEnvironmentVariables (rpc cmd) let testn = testNumber params block' = if 0 == testn then EVM.Fetch.Latest else EVM.Fetch.BlockNumber testn pure EVM.UnitTest.UnitTestOptions { EVM.UnitTest.solvers = solvers , EVM.UnitTest.rpcInfo = case rpc cmd of Just url -> Just (block', url) Nothing -> Nothing , EVM.UnitTest.maxIter = maxIterations cmd , EVM.UnitTest.askSmtIters = askSmtIterations cmd , EVM.UnitTest.smtDebug = smtdebug cmd , EVM.UnitTest.smtTimeout = smttimeout cmd , EVM.UnitTest.solver = solver cmd , EVM.UnitTest.covMatch = pack <$> covMatch cmd , EVM.UnitTest.verbose = verbose cmd , EVM.UnitTest.match = pack $ fromMaybe ".*" (match cmd) , EVM.UnitTest.maxDepth = depth cmd , EVM.UnitTest.fuzzRuns = fromMaybe 100 (fuzzRuns cmd) , EVM.UnitTest.replay = do arg' <- replay cmd return (fst arg', LazyByteString.fromStrict (hexByteString "--replay" $ strip0x $ snd arg')) , EVM.UnitTest.vmModifier = vmModifier , EVM.UnitTest.testParams = params , EVM.UnitTest.dapp = srcInfo , EVM.UnitTest.ffiAllowed = ffi cmd } main :: IO () main = do cmd <- Options.unwrapRecord "hevm -- Ethereum evaluator" let root = fromMaybe "." (dappRoot cmd) case cmd of Version {} -> putStrLn (showVersion Paths.version) Symbolic {} -> withCurrentDirectory root $ assert cmd Equivalence {} -> equivalence cmd Exec {} -> launchExec cmd DappTest {} -> withCurrentDirectory root $ do cores <- num <$> getNumProcessors withSolvers Z3 cores (smttimeout cmd) $ \solvers -> do testFile <- findJsonFile (jsonFile cmd) testOpts <- unitTestOptions cmd solvers testFile case (coverage cmd, optsMode cmd) of (False, Run) -> do res <- dappTest testOpts testFile (cache cmd) unless res exitFailure (False, Debug) -> liftIO $ TTY.main testOpts root testFile (False, JsonTrace) -> error "json traces not implemented for dappTest" (True, _) -> liftIO $ dappCoverage testOpts (optsMode cmd) testFile findJsonFile :: Maybe String -> IO String findJsonFile (Just s) = pure s findJsonFile Nothing = do outFiles <- listDirectory "out" case filter (isSuffixOf ".sol.json") outFiles of [x] -> pure ("out/" ++ x) [] -> error $ concat [ "No `*.sol.json' file found in `./out'.\n" , "Maybe you need to run `dapp build'.\n" , "You can specify a file with `--json-file'." ] xs -> error $ concat [ "Multiple `*.sol.json' files found in `./out'.\n" , "Specify one using `--json-file'.\n" , "Files found: " , intercalate ", " xs ] equivalence :: Command Options.Unwrapped -> IO () equivalence cmd = do let bytecodeA = hexByteString "--code" . strip0x $ codeA cmd bytecodeB = hexByteString "--code" . strip0x $ codeB cmd veriOpts = VeriOpts { simp = True , debug = False , maxIter = maxIterations cmd , askSmtIters = askSmtIterations cmd , rpcInfo = Nothing } withSolvers Z3 3 Nothing $ \s -> do res <- equivalenceCheck s bytecodeA bytecodeB veriOpts Nothing case containsA (Cex()) res of False -> do putStrLn "No discrepancies found" when (containsA (EVM.SymExec.Timeout()) res) $ do putStrLn "But timeout(s) occurred" exitFailure True -> do putStrLn $ "Not equivalent. Counterexample(s):" <> show res exitFailure getSrcInfo :: Command Options.Unwrapped -> IO DappInfo getSrcInfo cmd = let root = fromMaybe "." (dappRoot cmd) in case (jsonFile cmd) of Nothing -> pure emptyDapp Just json -> readSolc json >>= \case Nothing -> pure emptyDapp Just (contractMap, sourceCache) -> pure $ dappInfo root contractMap sourceCache -- Although it is tempting to fully abstract calldata and give any hints about -- the nature of the signature doing so results in significant time spent in -- consulting z3 about rather trivial matters. But with cvc5 it is quite -- pleasant! -- If function signatures are known, they should always be given for best results. assert :: Command Options.Unwrapped -> IO () assert cmd = do let block' = maybe EVM.Fetch.Latest EVM.Fetch.BlockNumber (block cmd) rpcinfo = (,) block' <$> rpc cmd decipher = hexByteString "bytes" . strip0x calldata' <- case (Main.calldata cmd, sig cmd) of -- fully abstract calldata (Nothing, Nothing) -> pure ( AbstractBuf "txdata" -- assert that the length of the calldata is never more than 2^64 -- this is way larger than would ever be allowed by the gas limit -- and avoids spurious counterexamples during abi decoding -- TODO: can we encode calldata as an array with a smaller length? , [Expr.bufLength (AbstractBuf "txtdata") .< (Lit (2 ^ (64 :: Integer)))] ) -- fully concrete calldata (Just c, Nothing) -> pure (ConcreteBuf (decipher c), []) -- calldata according to given abi with possible specializations from the `arg` list (Nothing, Just sig') -> do method' <- functionAbi sig' let typs = snd <$> view methodInputs method' pure $ symCalldata (view methodSignature method') typs (arg cmd) mempty _ -> error "incompatible options: calldata and abi" preState <- symvmFromCommand cmd calldata' let errCodes = fromMaybe defaultPanicCodes (assertions cmd) cores <- num <$> getNumProcessors let solverCount = fromMaybe cores (numSolvers cmd) withSolvers EVM.SMT.Z3 solverCount (smttimeout cmd) $ \solvers -> do if Main.debug cmd then do srcInfo <- getSrcInfo cmd void $ TTY.runFromVM solvers rpcinfo (maxIterations cmd) srcInfo preState else do let opts = VeriOpts { simp = True, debug = smtdebug cmd, maxIter = maxIterations cmd, askSmtIters = askSmtIterations cmd, rpcInfo = rpcinfo} (expr, res) <- verify solvers opts preState (Just $ checkAssertions errCodes) case res of [Qed _] -> putStrLn "\nQED: No reachable property violations discovered\n" cexs -> do let counterexamples | null (getCexs cexs) = [] | otherwise = [ "" , "Discovered the following counterexamples:" , "" ] <> fmap (formatCex (fst calldata')) (getCexs cexs) unknowns | null (getTimeouts cexs) = [] | otherwise = [ "" , "Could not determine reachability of the following end states:" , "" ] <> fmap (formatExpr) (getTimeouts cexs) T.putStrLn $ T.unlines (counterexamples <> unknowns) when (showTree cmd) $ do putStrLn "=== Expression ===\n" T.putStrLn $ formatExpr expr putStrLn "" when (showReachableTree cmd) $ do reached <- reachable solvers expr putStrLn "=== Reachable Expression ===\n" T.putStrLn (formatExpr . snd $ reached) putStrLn "" when (getModels cmd) $ do putStrLn $ "=== Models for " <> show (Expr.numBranches expr) <> " branches ===\n" ms <- produceModels solvers expr forM_ ms (showModel (fst calldata')) getCexs :: [VerifyResult] -> [SMTCex] getCexs = mapMaybe go where go (Cex cex) = Just $ snd cex go _ = Nothing getTimeouts :: [VerifyResult] -> [Expr End] getTimeouts = mapMaybe go where go (Timeout leaf) = Just leaf go _ = Nothing dappCoverage :: UnitTestOptions -> Mode -> String -> IO () dappCoverage opts _ solcFile = readSolc solcFile >>= \case Just (contractMap, sourceCache) -> do let unitTests = findUnitTests (EVM.UnitTest.match opts) $ Map.elems contractMap covs <- mconcat <$> mapM (coverageForUnitTestContract opts contractMap sourceCache) unitTests let dapp = dappInfo "." contractMap sourceCache f (k, vs) = do when (shouldPrintCoverage (EVM.UnitTest.covMatch opts) k) $ do putStr ("\x1b[0m" ++ "————— hevm coverage for ") -- Prefixed with color reset putStrLn (unpack k ++ " —————") putStrLn "" forM_ vs $ \(n, bs) -> do case ByteString.find (\x -> x /= 0x9 && x /= 0x20 && x /= 0x7d) bs of Nothing -> putStr "\x1b[38;5;240m" -- Gray (Coverage status isn't relevant) Just _ -> case n of -1 -> putStr "\x1b[38;5;240m" -- Gray (Coverage status isn't relevant) 0 -> putStr "\x1b[31m" -- Red (Uncovered) _ -> putStr "\x1b[32m" -- Green (Covered) Char8.putStrLn bs putStrLn "" mapM_ f (Map.toList (coverageReport dapp covs)) Nothing -> error ("Failed to read Solidity JSON for `" ++ solcFile ++ "'") shouldPrintCoverage :: Maybe Text -> Text -> Bool shouldPrintCoverage (Just covMatch) file = regexMatches covMatch file shouldPrintCoverage Nothing file = not (isTestOrLib file) isTestOrLib :: Text -> Bool isTestOrLib file = T.isSuffixOf ".t.sol" file || areAnyPrefixOf ["src/test/", "src/tests/", "lib/"] file areAnyPrefixOf :: [Text] -> Text -> Bool areAnyPrefixOf prefixes t = any (flip T.isPrefixOf t) prefixes launchExec :: Command Options.Unwrapped -> IO () launchExec cmd = do dapp <- getSrcInfo cmd vm <- vmFromCommand cmd -- TODO: we shouldn't need solvers to execute this code withSolvers Z3 0 Nothing $ \solvers -> do case optsMode cmd of Run -> do vm' <- execStateT (EVM.Stepper.interpret (EVM.Fetch.oracle solvers rpcinfo) . void $ EVM.Stepper.execFully) vm when (trace cmd) $ T.hPutStr stderr (showTraceTree dapp vm') case view EVM.result vm' of Nothing -> error "internal error; no EVM result" Just (EVM.VMFailure (EVM.Revert msg)) -> do let res = case msg of ConcreteBuf bs -> bs _ -> "" print $ ByteStringS res exitWith (ExitFailure 2) Just (EVM.VMFailure err) -> do print err exitWith (ExitFailure 2) Just (EVM.VMSuccess buf) -> do let msg = case buf of ConcreteBuf msg' -> msg' _ -> "" print $ ByteStringS msg case state cmd of Nothing -> pure () Just path -> Git.saveFacts (Git.RepoAt path) (Facts.vmFacts vm') case cache cmd of Nothing -> pure () Just path -> Git.saveFacts (Git.RepoAt path) (Facts.cacheFacts (view EVM.cache vm')) Debug -> void $ TTY.runFromVM solvers rpcinfo Nothing dapp vm --JsonTrace -> void $ execStateT (interpretWithTrace fetcher EVM.Stepper.runFully) vm _ -> error "TODO" where block' = maybe EVM.Fetch.Latest EVM.Fetch.BlockNumber (block cmd) rpcinfo = (,) block' <$> rpc cmd -- | Creates a (concrete) VM from command line options vmFromCommand :: Command Options.Unwrapped -> IO EVM.VM vmFromCommand cmd = do withCache <- applyCache (state cmd, cache cmd) (miner,ts,baseFee,blockNum,prevRan) <- case rpc cmd of Nothing -> return (0,Lit 0,0,0,0) Just url -> EVM.Fetch.fetchBlockFrom block' url >>= \case Nothing -> error "Could not fetch block" Just EVM.Block{..} -> return (_coinbase , _timestamp , _baseFee , _number , _prevRandao ) contract <- case (rpc cmd, address cmd, code cmd) of (Just url, Just addr', Just c) -> do EVM.Fetch.fetchContractFrom block' url addr' >>= \case Nothing -> error $ "contract not found: " <> show address' Just contract' -> -- if both code and url is given, -- fetch the contract and overwrite the code return $ EVM.initialContract (mkCode $ hexByteString "--code" $ strip0x c) & set EVM.balance (view EVM.balance contract') & set EVM.nonce (view EVM.nonce contract') & set EVM.external (view EVM.external contract') (Just url, Just addr', Nothing) -> EVM.Fetch.fetchContractFrom block' url addr' >>= \case Nothing -> error $ "contract not found: " <> show address' Just contract' -> return contract' (_, _, Just c) -> return $ EVM.initialContract (mkCode $ hexByteString "--code" $ strip0x c) (_, _, Nothing) -> error "must provide at least (rpc + address) or code" let ts' = case unlit ts of Just t -> t Nothing -> error "unexpected symbolic timestamp when executing vm test" return $ EVM.Transaction.initTx $ withCache (vm0 baseFee miner ts' blockNum prevRan contract) where block' = maybe EVM.Fetch.Latest EVM.Fetch.BlockNumber (block cmd) value' = word value 0 caller' = addr caller 0 origin' = addr origin 0 calldata' = ConcreteBuf $ bytes Main.calldata "" decipher = hexByteString "bytes" . strip0x mkCode bs = if create cmd then EVM.InitCode bs mempty else EVM.RuntimeCode (EVM.ConcreteRuntimeCode bs) address' = if create cmd then addr address (createAddress origin' (word nonce 0)) else addr address 0xacab vm0 baseFee miner ts blockNum prevRan c = EVM.makeVm $ EVM.VMOpts { EVM.vmoptContract = c , EVM.vmoptCalldata = (calldata', []) , EVM.vmoptValue = Lit value' , EVM.vmoptAddress = address' , EVM.vmoptCaller = litAddr caller' , EVM.vmoptOrigin = origin' , EVM.vmoptGas = word64 gas 0xffffffffffffffff , EVM.vmoptBaseFee = baseFee , EVM.vmoptPriorityFee = word priorityFee 0 , EVM.vmoptGaslimit = word64 gaslimit 0xffffffffffffffff , EVM.vmoptCoinbase = addr coinbase miner , EVM.vmoptNumber = word number blockNum , EVM.vmoptTimestamp = Lit $ word timestamp ts , EVM.vmoptBlockGaslimit = word64 gaslimit 0xffffffffffffffff , EVM.vmoptGasprice = word gasprice 0 , EVM.vmoptMaxCodeSize = word maxcodesize 0xffffffff , EVM.vmoptPrevRandao = word prevRandao prevRan , EVM.vmoptSchedule = FeeSchedule.berlin , EVM.vmoptChainId = word chainid 1 , EVM.vmoptCreate = create cmd , EVM.vmoptStorageBase = EVM.Concrete , EVM.vmoptTxAccessList = mempty -- TODO: support me soon , EVM.vmoptAllowFFI = False } word f def = fromMaybe def (f cmd) word64 f def = fromMaybe def (f cmd) addr f def = fromMaybe def (f cmd) bytes f def = maybe def decipher (f cmd) symvmFromCommand :: Command Options.Unwrapped -> (Expr Buf, [Prop]) -> IO (EVM.VM) symvmFromCommand cmd calldata' = do (miner,blockNum,baseFee,prevRan) <- case rpc cmd of Nothing -> return (0,0,0,0) Just url -> EVM.Fetch.fetchBlockFrom block' url >>= \case Nothing -> error "Could not fetch block" Just EVM.Block{..} -> return (_coinbase , _number , _baseFee , _prevRandao ) let caller' = Caller 0 ts = maybe Timestamp Lit (timestamp cmd) callvalue' = maybe (CallValue 0) Lit (value cmd) -- TODO: rework this, ConcreteS not needed anymore let store = case storageModel cmd of -- InitialS and SymbolicS can read and write to symbolic locations -- ConcreteS cannot (instead values can be fetched from rpc!) -- Initial defaults to 0 for uninitialized storage slots, -- whereas the values of SymbolicS are unconstrained. Just InitialS -> EmptyStore Just ConcreteS -> ConcreteStore mempty Just SymbolicS -> AbstractStore Nothing -> if create cmd then EmptyStore else AbstractStore withCache <- applyCache (state cmd, cache cmd) contract' <- case (rpc cmd, address cmd, code cmd) of (Just url, Just addr', _) -> EVM.Fetch.fetchContractFrom block' url addr' >>= \case Nothing -> error "contract not found." Just contract' -> return contract'' where contract'' = case code cmd of Nothing -> contract' -- if both code and url is given, -- fetch the contract and overwrite the code Just c -> EVM.initialContract (mkCode $ decipher c) -- TODO: fix this -- & set EVM.origStorage (view EVM.origStorage contract') & set EVM.balance (view EVM.balance contract') & set EVM.nonce (view EVM.nonce contract') & set EVM.external (view EVM.external contract') (_, _, Just c) -> return (EVM.initialContract . mkCode $ decipher c) (_, _, Nothing) -> error "must provide at least (rpc + address) or code" return $ (EVM.Transaction.initTx $ withCache $ vm0 baseFee miner ts blockNum prevRan calldata' callvalue' caller' contract') & set (EVM.env . EVM.storage) store where decipher = hexByteString "bytes" . strip0x block' = maybe EVM.Fetch.Latest EVM.Fetch.BlockNumber (block cmd) origin' = addr origin 0 mkCode bs = if create cmd then EVM.InitCode bs mempty else EVM.RuntimeCode (EVM.ConcreteRuntimeCode bs) address' = if create cmd then addr address (createAddress origin' (word nonce 0)) else addr address 0xacab vm0 baseFee miner ts blockNum prevRan cd' callvalue' caller' c = EVM.makeVm $ EVM.VMOpts { EVM.vmoptContract = c , EVM.vmoptCalldata = cd' , EVM.vmoptValue = callvalue' , EVM.vmoptAddress = address' , EVM.vmoptCaller = caller' , EVM.vmoptOrigin = origin' , EVM.vmoptGas = word64 gas 0xffffffffffffffff , EVM.vmoptGaslimit = word64 gaslimit 0xffffffffffffffff , EVM.vmoptBaseFee = baseFee , EVM.vmoptPriorityFee = word priorityFee 0 , EVM.vmoptCoinbase = addr coinbase miner , EVM.vmoptNumber = word number blockNum , EVM.vmoptTimestamp = ts , EVM.vmoptBlockGaslimit = word64 gaslimit 0xffffffffffffffff , EVM.vmoptGasprice = word gasprice 0 , EVM.vmoptMaxCodeSize = word maxcodesize 0xffffffff , EVM.vmoptPrevRandao = word prevRandao prevRan , EVM.vmoptSchedule = FeeSchedule.berlin , EVM.vmoptChainId = word chainid 1 , EVM.vmoptCreate = create cmd , EVM.vmoptStorageBase = EVM.Symbolic , EVM.vmoptTxAccessList = mempty , EVM.vmoptAllowFFI = False } word f def = fromMaybe def (f cmd) addr f def = fromMaybe def (f cmd) word64 f def = fromMaybe def (f cmd)