module EVM.Exec where import EVM import EVM.Concrete (createAddress) import EVM.Types import EVM.Expr (litAddr) import qualified EVM.FeeSchedule as FeeSchedule import Optics.Core import Control.Monad.Trans.State.Strict (get, State) import Data.ByteString (ByteString) import Data.Maybe (isNothing) ethrunAddress :: Addr ethrunAddress = Addr 0x00a329c0648769a73afac7f9381e08fb43dbea72 vmForEthrunCreation :: ByteString -> VM vmForEthrunCreation creationCode = (makeVm $ VMOpts { contract = initialContract (InitCode creationCode mempty) , calldata = mempty , value = (Lit 0) , initialStorage = EmptyStore , address = createAddress ethrunAddress 1 , caller = litAddr ethrunAddress , origin = ethrunAddress , coinbase = 0 , number = 0 , timestamp = (Lit 0) , blockGaslimit = 0 , gasprice = 0 , prevRandao = 42069 , gas = 0xffffffffffffffff , gaslimit = 0xffffffffffffffff , baseFee = 0 , priorityFee = 0 , maxCodeSize = 0xffffffff , schedule = FeeSchedule.berlin , chainId = 1 , create = False , txAccessList = mempty , allowFFI = False }) & set (#env % #contracts % at ethrunAddress) (Just (initialContract (RuntimeCode (ConcreteRuntimeCode "")))) exec :: State VM VMResult exec = do vm <- get case vm.result of Nothing -> exec1 >> exec Just r -> pure r run :: State VM VM run = do vm <- get case vm.result of Nothing -> exec1 >> run Just _ -> pure vm execWhile :: (VM -> Bool) -> State VM Int execWhile p = go 0 where go i = do vm <- get if p vm && isNothing vm.result then do go $! (i + 1) else pure i