| 
									
										
										
										
											2016-04-14 18:18:24 +02:00
										 |  |  | // Copyright 2015 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							|  |  |  | // GNU Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package runtime | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2017-01-04 20:17:24 +01:00
										 |  |  | 	"math" | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	"math/big" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2018-09-24 15:57:49 +03:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/rawdb" | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/state" | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/vm" | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/crypto" | 
					
						
							| 
									
										
										
										
											2016-10-08 00:23:45 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/params" | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-15 11:08:18 -07:00
										 |  |  | // Config is a basic type specifying certain configuration flags for running | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | // the EVM. | 
					
						
							|  |  |  | type Config struct { | 
					
						
							| 
									
										
										
										
											2016-10-20 13:36:29 +02:00
										 |  |  | 	ChainConfig *params.ChainConfig | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	Difficulty  *big.Int | 
					
						
							|  |  |  | 	Origin      common.Address | 
					
						
							|  |  |  | 	Coinbase    common.Address | 
					
						
							|  |  |  | 	BlockNumber *big.Int | 
					
						
							|  |  |  | 	Time        *big.Int | 
					
						
							| 
									
										
										
										
											2017-01-04 20:17:24 +01:00
										 |  |  | 	GasLimit    uint64 | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	GasPrice    *big.Int | 
					
						
							|  |  |  | 	Value       *big.Int | 
					
						
							|  |  |  | 	Debug       bool | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	EVMConfig   vm.Config | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 23:20:42 +01:00
										 |  |  | 	State     *state.StateDB | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	GetHashFn func(n uint64) common.Hash | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // sets defaults on the config | 
					
						
							|  |  |  | func setDefaults(cfg *Config) { | 
					
						
							| 
									
										
										
										
											2016-10-20 13:36:29 +02:00
										 |  |  | 	if cfg.ChainConfig == nil { | 
					
						
							| 
									
										
										
										
											2016-11-13 22:11:42 +01:00
										 |  |  | 		cfg.ChainConfig = ¶ms.ChainConfig{ | 
					
						
							| 
									
										
										
										
											2020-07-16 14:06:19 +02:00
										 |  |  | 			ChainID:             big.NewInt(1), | 
					
						
							|  |  |  | 			HomesteadBlock:      new(big.Int), | 
					
						
							|  |  |  | 			DAOForkBlock:        new(big.Int), | 
					
						
							|  |  |  | 			DAOForkSupport:      false, | 
					
						
							|  |  |  | 			EIP150Block:         new(big.Int), | 
					
						
							|  |  |  | 			EIP150Hash:          common.Hash{}, | 
					
						
							|  |  |  | 			EIP155Block:         new(big.Int), | 
					
						
							|  |  |  | 			EIP158Block:         new(big.Int), | 
					
						
							|  |  |  | 			ByzantiumBlock:      new(big.Int), | 
					
						
							|  |  |  | 			ConstantinopleBlock: new(big.Int), | 
					
						
							|  |  |  | 			PetersburgBlock:     new(big.Int), | 
					
						
							|  |  |  | 			IstanbulBlock:       new(big.Int), | 
					
						
							|  |  |  | 			MuirGlacierBlock:    new(big.Int), | 
					
						
							| 
									
										
										
										
											2020-10-23 08:26:57 +02:00
										 |  |  | 			YoloV2Block:         nil, | 
					
						
							| 
									
										
										
										
											2016-11-13 22:11:42 +01:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-03-01 23:32:43 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	if cfg.Difficulty == nil { | 
					
						
							|  |  |  | 		cfg.Difficulty = new(big.Int) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if cfg.Time == nil { | 
					
						
							|  |  |  | 		cfg.Time = big.NewInt(time.Now().Unix()) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-01-04 20:17:24 +01:00
										 |  |  | 	if cfg.GasLimit == 0 { | 
					
						
							|  |  |  | 		cfg.GasLimit = math.MaxUint64 | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if cfg.GasPrice == nil { | 
					
						
							|  |  |  | 		cfg.GasPrice = new(big.Int) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if cfg.Value == nil { | 
					
						
							|  |  |  | 		cfg.Value = new(big.Int) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if cfg.BlockNumber == nil { | 
					
						
							|  |  |  | 		cfg.BlockNumber = new(big.Int) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if cfg.GetHashFn == nil { | 
					
						
							|  |  |  | 		cfg.GetHashFn = func(n uint64) common.Hash { | 
					
						
							| 
									
										
										
										
											2016-02-21 18:40:27 +00:00
										 |  |  | 			return common.BytesToHash(crypto.Keccak256([]byte(new(big.Int).SetUint64(n).String()))) | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Execute executes the code using the input as call data during the execution. | 
					
						
							|  |  |  | // It returns the EVM's return value, the new state and an error if it failed. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2020-01-30 11:21:10 +01:00
										 |  |  | // Execute sets up an in-memory, temporary, environment for the execution of | 
					
						
							|  |  |  | // the given code. It makes sure that it's restored to its original state afterwards. | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | func Execute(code, input []byte, cfg *Config) ([]byte, *state.StateDB, error) { | 
					
						
							|  |  |  | 	if cfg == nil { | 
					
						
							|  |  |  | 		cfg = new(Config) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	setDefaults(cfg) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 23:20:42 +01:00
										 |  |  | 	if cfg.State == nil { | 
					
						
							| 
									
										
										
										
											2019-08-06 13:40:28 +03:00
										 |  |  | 		cfg.State, _ = state.New(common.Hash{}, state.NewDatabase(rawdb.NewMemoryDatabase()), nil) | 
					
						
							| 
									
										
										
										
											2016-02-09 23:20:42 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	var ( | 
					
						
							| 
									
										
										
										
											2018-04-10 13:12:07 +02:00
										 |  |  | 		address = common.BytesToAddress([]byte("contract")) | 
					
						
							| 
									
										
										
										
											2017-08-11 07:29:32 -04:00
										 |  |  | 		vmenv   = NewEnv(cfg) | 
					
						
							| 
									
										
										
										
											2017-02-22 23:29:59 +01:00
										 |  |  | 		sender  = vm.AccountRef(cfg.Origin) | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	) | 
					
						
							| 
									
										
										
										
											2020-10-23 08:26:57 +02:00
										 |  |  | 	if cfg.ChainConfig.IsYoloV2(vmenv.BlockNumber) { | 
					
						
							|  |  |  | 		cfg.State.AddAddressToAccessList(cfg.Origin) | 
					
						
							|  |  |  | 		cfg.State.AddAddressToAccessList(address) | 
					
						
							|  |  |  | 		for _, addr := range vmenv.ActivePrecompiles() { | 
					
						
							|  |  |  | 			cfg.State.AddAddressToAccessList(addr) | 
					
						
							|  |  |  | 			cfg.State.AddAddressToAccessList(addr) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-02-22 23:29:59 +01:00
										 |  |  | 	cfg.State.CreateAccount(address) | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	// set the receiver's (the executing contract) code for execution. | 
					
						
							| 
									
										
										
										
											2017-02-22 23:29:59 +01:00
										 |  |  | 	cfg.State.SetCode(address, code) | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 	// Call the code with the given configuration. | 
					
						
							| 
									
										
										
										
											2017-01-04 20:17:24 +01:00
										 |  |  | 	ret, _, err := vmenv.Call( | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 		sender, | 
					
						
							| 
									
										
										
										
											2018-04-10 13:12:07 +02:00
										 |  |  | 		common.BytesToAddress([]byte("contract")), | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | 		input, | 
					
						
							|  |  |  | 		cfg.GasLimit, | 
					
						
							|  |  |  | 		cfg.Value, | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 23:20:42 +01:00
										 |  |  | 	return ret, cfg.State, err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | // Create executes the code using the EVM create method | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | func Create(input []byte, cfg *Config) ([]byte, common.Address, uint64, error) { | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	if cfg == nil { | 
					
						
							|  |  |  | 		cfg = new(Config) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	setDefaults(cfg) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if cfg.State == nil { | 
					
						
							| 
									
										
										
										
											2019-08-06 13:40:28 +03:00
										 |  |  | 		cfg.State, _ = state.New(common.Hash{}, state.NewDatabase(rawdb.NewMemoryDatabase()), nil) | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	var ( | 
					
						
							| 
									
										
										
										
											2017-08-11 07:29:32 -04:00
										 |  |  | 		vmenv  = NewEnv(cfg) | 
					
						
							| 
									
										
										
										
											2017-02-22 23:29:59 +01:00
										 |  |  | 		sender = vm.AccountRef(cfg.Origin) | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	) | 
					
						
							| 
									
										
										
										
											2020-10-23 08:26:57 +02:00
										 |  |  | 	if cfg.ChainConfig.IsYoloV2(vmenv.BlockNumber) { | 
					
						
							|  |  |  | 		cfg.State.AddAddressToAccessList(cfg.Origin) | 
					
						
							|  |  |  | 		for _, addr := range vmenv.ActivePrecompiles() { | 
					
						
							|  |  |  | 			cfg.State.AddAddressToAccessList(addr) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Call the code with the given configuration. | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	code, address, leftOverGas, err := vmenv.Create( | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 		sender, | 
					
						
							|  |  |  | 		input, | 
					
						
							|  |  |  | 		cfg.GasLimit, | 
					
						
							|  |  |  | 		cfg.Value, | 
					
						
							|  |  |  | 	) | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	return code, address, leftOverGas, err | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 23:20:42 +01:00
										 |  |  | // Call executes the code given by the contract's address. It will return the | 
					
						
							|  |  |  | // EVM's return value or an error if it failed. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // Call, unlike Execute, requires a config and also requires the State field to | 
					
						
							|  |  |  | // be set. | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | func Call(address common.Address, input []byte, cfg *Config) ([]byte, uint64, error) { | 
					
						
							| 
									
										
										
										
											2016-02-09 23:20:42 +01:00
										 |  |  | 	setDefaults(cfg) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-11 07:29:32 -04:00
										 |  |  | 	vmenv := NewEnv(cfg) | 
					
						
							| 
									
										
										
										
											2016-02-09 23:20:42 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	sender := cfg.State.GetOrNewStateObject(cfg.Origin) | 
					
						
							| 
									
										
										
										
											2020-10-23 08:26:57 +02:00
										 |  |  | 	if cfg.ChainConfig.IsYoloV2(vmenv.BlockNumber) { | 
					
						
							|  |  |  | 		cfg.State.AddAddressToAccessList(cfg.Origin) | 
					
						
							|  |  |  | 		cfg.State.AddAddressToAccessList(address) | 
					
						
							|  |  |  | 		for _, addr := range vmenv.ActivePrecompiles() { | 
					
						
							|  |  |  | 			cfg.State.AddAddressToAccessList(addr) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 23:20:42 +01:00
										 |  |  | 	// Call the code with the given configuration. | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	ret, leftOverGas, err := vmenv.Call( | 
					
						
							| 
									
										
										
										
											2016-02-09 23:20:42 +01:00
										 |  |  | 		sender, | 
					
						
							|  |  |  | 		address, | 
					
						
							|  |  |  | 		input, | 
					
						
							|  |  |  | 		cfg.GasLimit, | 
					
						
							|  |  |  | 		cfg.Value, | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	return ret, leftOverGas, err | 
					
						
							| 
									
										
										
										
											2015-10-09 14:56:12 +02:00
										 |  |  | } |