| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | // Copyright 2017 The go-ethereum Authors | 
					
						
							|  |  |  | // This file is part of go-ethereum. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // go-ethereum is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // it under the terms of the GNU General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // go-ethereum 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 General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  | // along with go-ethereum. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2019-10-28 06:55:20 -07:00
										 |  |  | 	"bytes" | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	"encoding/json" | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2018-04-26 10:30:23 +02:00
										 |  |  | 	"math/big" | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2018-04-26 10:30:23 +02:00
										 |  |  | 	goruntime "runtime" | 
					
						
							| 
									
										
										
										
											2017-05-23 09:34:04 +02:00
										 |  |  | 	"runtime/pprof" | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 	"testing" | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-01 11:52:57 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/cmd/evm/internal/compiler" | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/cmd/utils" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core" | 
					
						
							| 
									
										
										
										
											2018-09-24 15:57:49 +03:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/rawdb" | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/state" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/core/vm" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/core/vm/runtime" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/log" | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/params" | 
					
						
							| 
									
										
										
										
											2020-11-25 21:00:23 +01:00
										 |  |  | 	"gopkg.in/urfave/cli.v1" | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var runCommand = cli.Command{ | 
					
						
							|  |  |  | 	Action:      runCmd, | 
					
						
							|  |  |  | 	Name:        "run", | 
					
						
							|  |  |  | 	Usage:       "run arbitrary evm binary", | 
					
						
							|  |  |  | 	ArgsUsage:   "<code>", | 
					
						
							|  |  |  | 	Description: `The run command runs arbitrary EVM code.`, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | // readGenesis will read the given JSON format genesis file and return | 
					
						
							|  |  |  | // the initialized Genesis structure | 
					
						
							|  |  |  | func readGenesis(genesisPath string) *core.Genesis { | 
					
						
							|  |  |  | 	// Make sure we have a valid genesis JSON | 
					
						
							|  |  |  | 	//genesisPath := ctx.Args().First() | 
					
						
							|  |  |  | 	if len(genesisPath) == 0 { | 
					
						
							|  |  |  | 		utils.Fatalf("Must supply path to genesis JSON file") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	file, err := os.Open(genesisPath) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		utils.Fatalf("Failed to read genesis file: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer file.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	genesis := new(core.Genesis) | 
					
						
							|  |  |  | 	if err := json.NewDecoder(file).Decode(genesis); err != nil { | 
					
						
							|  |  |  | 		utils.Fatalf("invalid genesis file: %v", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return genesis | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 12:40:07 +02:00
										 |  |  | type execStats struct { | 
					
						
							|  |  |  | 	time           time.Duration // The execution time. | 
					
						
							|  |  |  | 	allocs         int64         // The number of heap allocations during execution. | 
					
						
							|  |  |  | 	bytesAllocated int64         // The cumulative number of bytes allocated during execution. | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 12:40:07 +02:00
										 |  |  | func timedExec(bench bool, execFunc func() ([]byte, uint64, error)) (output []byte, gasLeft uint64, stats execStats, err error) { | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 	if bench { | 
					
						
							|  |  |  | 		result := testing.Benchmark(func(b *testing.B) { | 
					
						
							|  |  |  | 			for i := 0; i < b.N; i++ { | 
					
						
							|  |  |  | 				output, gasLeft, err = execFunc() | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Get the average execution time from the benchmarking result. | 
					
						
							|  |  |  | 		// There are other useful stats here that could be reported. | 
					
						
							| 
									
										
										
										
											2020-04-01 12:40:07 +02:00
										 |  |  | 		stats.time = time.Duration(result.NsPerOp()) | 
					
						
							|  |  |  | 		stats.allocs = result.AllocsPerOp() | 
					
						
							|  |  |  | 		stats.bytesAllocated = result.AllocedBytesPerOp() | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2020-04-01 12:40:07 +02:00
										 |  |  | 		var memStatsBefore, memStatsAfter goruntime.MemStats | 
					
						
							|  |  |  | 		goruntime.ReadMemStats(&memStatsBefore) | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 		startTime := time.Now() | 
					
						
							|  |  |  | 		output, gasLeft, err = execFunc() | 
					
						
							| 
									
										
										
										
											2020-04-01 12:40:07 +02:00
										 |  |  | 		stats.time = time.Since(startTime) | 
					
						
							|  |  |  | 		goruntime.ReadMemStats(&memStatsAfter) | 
					
						
							|  |  |  | 		stats.allocs = int64(memStatsAfter.Mallocs - memStatsBefore.Mallocs) | 
					
						
							|  |  |  | 		stats.bytesAllocated = int64(memStatsAfter.TotalAlloc - memStatsBefore.TotalAlloc) | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 12:40:07 +02:00
										 |  |  | 	return output, gasLeft, stats, err | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | func runCmd(ctx *cli.Context) error { | 
					
						
							|  |  |  | 	glogger := log.NewGlogHandler(log.StreamHandler(os.Stderr, log.TerminalFormat(false))) | 
					
						
							|  |  |  | 	glogger.Verbosity(log.Lvl(ctx.GlobalInt(VerbosityFlag.Name))) | 
					
						
							|  |  |  | 	log.Root().SetHandler(glogger) | 
					
						
							| 
									
										
										
										
											2017-06-21 14:52:31 +02:00
										 |  |  | 	logconfig := &vm.LogConfig{ | 
					
						
							| 
									
										
										
										
											2021-09-13 18:59:52 +02:00
										 |  |  | 		EnableMemory:     !ctx.GlobalBool(DisableMemoryFlag.Name), | 
					
						
							|  |  |  | 		DisableStack:     ctx.GlobalBool(DisableStackFlag.Name), | 
					
						
							|  |  |  | 		DisableStorage:   ctx.GlobalBool(DisableStorageFlag.Name), | 
					
						
							|  |  |  | 		EnableReturnData: !ctx.GlobalBool(DisableReturnDataFlag.Name), | 
					
						
							|  |  |  | 		Debug:            ctx.GlobalBool(DebugFlag.Name), | 
					
						
							| 
									
										
										
										
											2017-06-21 14:52:31 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var ( | 
					
						
							| 
									
										
										
										
											2018-09-20 08:24:53 +02:00
										 |  |  | 		tracer        vm.Tracer | 
					
						
							|  |  |  | 		debugLogger   *vm.StructLogger | 
					
						
							|  |  |  | 		statedb       *state.StateDB | 
					
						
							|  |  |  | 		chainConfig   *params.ChainConfig | 
					
						
							|  |  |  | 		sender        = common.BytesToAddress([]byte("sender")) | 
					
						
							|  |  |  | 		receiver      = common.BytesToAddress([]byte("receiver")) | 
					
						
							|  |  |  | 		genesisConfig *core.Genesis | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	) | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	if ctx.GlobalBool(MachineFlag.Name) { | 
					
						
							| 
									
										
										
										
											2018-10-15 12:28:44 +02:00
										 |  |  | 		tracer = vm.NewJSONLogger(logconfig, os.Stdout) | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	} else if ctx.GlobalBool(DebugFlag.Name) { | 
					
						
							| 
									
										
										
										
											2017-06-21 14:52:31 +02:00
										 |  |  | 		debugLogger = vm.NewStructLogger(logconfig) | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 		tracer = debugLogger | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2017-06-21 14:52:31 +02:00
										 |  |  | 		debugLogger = vm.NewStructLogger(logconfig) | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if ctx.GlobalString(GenesisFlag.Name) != "" { | 
					
						
							|  |  |  | 		gen := readGenesis(ctx.GlobalString(GenesisFlag.Name)) | 
					
						
							| 
									
										
										
										
											2018-09-20 08:24:53 +02:00
										 |  |  | 		genesisConfig = gen | 
					
						
							| 
									
										
										
										
											2018-09-24 15:57:49 +03:00
										 |  |  | 		db := rawdb.NewMemoryDatabase() | 
					
						
							| 
									
										
										
										
											2018-02-05 18:40:32 +02:00
										 |  |  | 		genesis := gen.ToBlock(db) | 
					
						
							| 
									
										
										
										
											2019-08-06 13:40:28 +03:00
										 |  |  | 		statedb, _ = state.New(genesis.Root(), state.NewDatabase(db), nil) | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 		chainConfig = gen.Config | 
					
						
							|  |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2019-08-06 13:40:28 +03:00
										 |  |  | 		statedb, _ = state.New(common.Hash{}, state.NewDatabase(rawdb.NewMemoryDatabase()), nil) | 
					
						
							| 
									
										
										
										
											2018-09-20 08:24:53 +02:00
										 |  |  | 		genesisConfig = new(core.Genesis) | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	if ctx.GlobalString(SenderFlag.Name) != "" { | 
					
						
							|  |  |  | 		sender = common.HexToAddress(ctx.GlobalString(SenderFlag.Name)) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	statedb.CreateAccount(sender) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-15 11:31:36 +02:00
										 |  |  | 	if ctx.GlobalString(ReceiverFlag.Name) != "" { | 
					
						
							|  |  |  | 		receiver = common.HexToAddress(ctx.GlobalString(ReceiverFlag.Name)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 	var code []byte | 
					
						
							| 
									
										
										
										
											2019-06-25 10:03:04 +02:00
										 |  |  | 	codeFileFlag := ctx.GlobalString(CodeFileFlag.Name) | 
					
						
							|  |  |  | 	codeFlag := ctx.GlobalString(CodeFlag.Name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-15 11:31:36 +02:00
										 |  |  | 	// The '--code' or '--codefile' flag overrides code in state | 
					
						
							| 
									
										
										
										
											2019-06-25 10:03:04 +02:00
										 |  |  | 	if codeFileFlag != "" || codeFlag != "" { | 
					
						
							| 
									
										
										
										
											2017-08-15 11:31:36 +02:00
										 |  |  | 		var hexcode []byte | 
					
						
							| 
									
										
										
										
											2019-06-25 10:03:04 +02:00
										 |  |  | 		if codeFileFlag != "" { | 
					
						
							|  |  |  | 			var err error | 
					
						
							|  |  |  | 			// If - is specified, it means that code comes from stdin | 
					
						
							|  |  |  | 			if codeFileFlag == "-" { | 
					
						
							|  |  |  | 				//Try reading from stdin | 
					
						
							|  |  |  | 				if hexcode, err = ioutil.ReadAll(os.Stdin); err != nil { | 
					
						
							|  |  |  | 					fmt.Printf("Could not load code from stdin: %v\n", err) | 
					
						
							|  |  |  | 					os.Exit(1) | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				// Codefile with hex assembly | 
					
						
							|  |  |  | 				if hexcode, err = ioutil.ReadFile(codeFileFlag); err != nil { | 
					
						
							|  |  |  | 					fmt.Printf("Could not load code from file: %v\n", err) | 
					
						
							|  |  |  | 					os.Exit(1) | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2017-08-15 11:31:36 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2019-06-25 10:03:04 +02:00
										 |  |  | 			hexcode = []byte(codeFlag) | 
					
						
							| 
									
										
										
										
											2017-08-15 11:31:36 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-10-28 06:55:20 -07:00
										 |  |  | 		hexcode = bytes.TrimSpace(hexcode) | 
					
						
							| 
									
										
										
										
											2019-06-25 10:03:04 +02:00
										 |  |  | 		if len(hexcode)%2 != 0 { | 
					
						
							|  |  |  | 			fmt.Printf("Invalid input length for hex data (%d)\n", len(hexcode)) | 
					
						
							|  |  |  | 			os.Exit(1) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		code = common.FromHex(string(hexcode)) | 
					
						
							| 
									
										
										
										
											2017-08-15 11:31:36 +02:00
										 |  |  | 	} else if fn := ctx.Args().First(); len(fn) > 0 { | 
					
						
							|  |  |  | 		// EASM-file to compile | 
					
						
							| 
									
										
										
										
											2017-03-01 11:52:57 +01:00
										 |  |  | 		src, err := ioutil.ReadFile(fn) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		bin, err := compiler.Compile(fn, src, false) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		code = common.Hex2Bytes(bin) | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	initialGas := ctx.GlobalUint64(GasFlag.Name) | 
					
						
							| 
									
										
										
										
											2018-09-20 08:24:53 +02:00
										 |  |  | 	if genesisConfig.GasLimit != 0 { | 
					
						
							|  |  |  | 		initialGas = genesisConfig.GasLimit | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-03-01 10:19:15 +01:00
										 |  |  | 	runtimeConfig := runtime.Config{ | 
					
						
							| 
									
										
										
										
											2018-04-26 10:30:23 +02:00
										 |  |  | 		Origin:      sender, | 
					
						
							|  |  |  | 		State:       statedb, | 
					
						
							|  |  |  | 		GasLimit:    initialGas, | 
					
						
							|  |  |  | 		GasPrice:    utils.GlobalBig(ctx, PriceFlag.Name), | 
					
						
							|  |  |  | 		Value:       utils.GlobalBig(ctx, ValueFlag.Name), | 
					
						
							| 
									
										
										
										
											2018-09-20 08:24:53 +02:00
										 |  |  | 		Difficulty:  genesisConfig.Difficulty, | 
					
						
							|  |  |  | 		Time:        new(big.Int).SetUint64(genesisConfig.Timestamp), | 
					
						
							|  |  |  | 		Coinbase:    genesisConfig.Coinbase, | 
					
						
							|  |  |  | 		BlockNumber: new(big.Int).SetUint64(genesisConfig.Number), | 
					
						
							| 
									
										
										
										
											2017-03-01 10:19:15 +01:00
										 |  |  | 		EVMConfig: vm.Config{ | 
					
						
							| 
									
										
										
										
											2021-07-06 22:03:09 +02:00
										 |  |  | 			Tracer: tracer, | 
					
						
							|  |  |  | 			Debug:  ctx.GlobalBool(DebugFlag.Name) || ctx.GlobalBool(MachineFlag.Name), | 
					
						
							| 
									
										
										
										
											2017-03-01 10:19:15 +01:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-23 09:34:04 +02:00
										 |  |  | 	if cpuProfilePath := ctx.GlobalString(CPUProfileFlag.Name); cpuProfilePath != "" { | 
					
						
							|  |  |  | 		f, err := os.Create(cpuProfilePath) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2018-08-01 19:09:08 +03:00
										 |  |  | 			fmt.Println("could not create CPU profile: ", err) | 
					
						
							| 
									
										
										
										
											2017-05-23 09:34:04 +02:00
										 |  |  | 			os.Exit(1) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if err := pprof.StartCPUProfile(f); err != nil { | 
					
						
							| 
									
										
										
										
											2018-08-01 19:09:08 +03:00
										 |  |  | 			fmt.Println("could not start CPU profile: ", err) | 
					
						
							| 
									
										
										
										
											2017-05-23 09:34:04 +02:00
										 |  |  | 			os.Exit(1) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		defer pprof.StopCPUProfile() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	if chainConfig != nil { | 
					
						
							|  |  |  | 		runtimeConfig.ChainConfig = chainConfig | 
					
						
							| 
									
										
										
										
											2019-09-13 22:32:20 +02:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		runtimeConfig.ChainConfig = params.AllEthashProtocolChanges | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-17 06:45:54 -08:00
										 |  |  | 	var hexInput []byte | 
					
						
							|  |  |  | 	if inputFileFlag := ctx.GlobalString(InputFileFlag.Name); inputFileFlag != "" { | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 		var err error | 
					
						
							| 
									
										
										
										
											2019-11-17 06:45:54 -08:00
										 |  |  | 		if hexInput, err = ioutil.ReadFile(inputFileFlag); err != nil { | 
					
						
							|  |  |  | 			fmt.Printf("could not load input from file: %v\n", err) | 
					
						
							|  |  |  | 			os.Exit(1) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		hexInput = []byte(ctx.GlobalString(InputFlag.Name)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	input := common.FromHex(string(bytes.TrimSpace(hexInput))) | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var execFunc func() ([]byte, uint64, error) | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	if ctx.GlobalBool(CreateFlag.Name) { | 
					
						
							| 
									
										
										
										
											2019-11-17 06:45:54 -08:00
										 |  |  | 		input = append(code, input...) | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 		execFunc = func() ([]byte, uint64, error) { | 
					
						
							|  |  |  | 			output, _, gasLeft, err := runtime.Create(input, &runtimeConfig) | 
					
						
							|  |  |  | 			return output, gasLeft, err | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2017-08-15 11:31:36 +02:00
										 |  |  | 		if len(code) > 0 { | 
					
						
							|  |  |  | 			statedb.SetCode(receiver, code) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 		execFunc = func() ([]byte, uint64, error) { | 
					
						
							|  |  |  | 			return runtime.Call(receiver, input, &runtimeConfig) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 12:40:07 +02:00
										 |  |  | 	bench := ctx.GlobalBool(BenchFlag.Name) | 
					
						
							|  |  |  | 	output, leftOverGas, stats, err := timedExec(bench, execFunc) | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if ctx.GlobalBool(DumpFlag.Name) { | 
					
						
							| 
									
										
										
										
											2018-11-29 09:29:12 +01:00
										 |  |  | 		statedb.Commit(true) | 
					
						
							| 
									
										
										
										
											2017-06-27 15:57:06 +02:00
										 |  |  | 		statedb.IntermediateRoot(true) | 
					
						
							| 
									
										
										
											
												cmd/geth, eth, core: snapshot dump + unify with trie dump (#22795)
* cmd/geth, eth, core: snapshot dump + unify with trie dump
* cmd/evm: dump API fixes
* cmd/geth, core, eth: fix some remaining errors
* cmd/evm: dump - add limit, support address startkey, address review concerns
* cmd, core/state, eth: minor polishes, fix snap dump crash, unify format
Co-authored-by: Péter Szilágyi <peterke@gmail.com>
											
										 
											2021-05-12 10:05:39 +02:00
										 |  |  | 		fmt.Println(string(statedb.Dump(nil))) | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-03-01 10:19:15 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-23 09:34:04 +02:00
										 |  |  | 	if memProfilePath := ctx.GlobalString(MemProfileFlag.Name); memProfilePath != "" { | 
					
						
							|  |  |  | 		f, err := os.Create(memProfilePath) | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2018-08-01 19:09:08 +03:00
										 |  |  | 			fmt.Println("could not create memory profile: ", err) | 
					
						
							| 
									
										
										
										
											2017-05-23 09:34:04 +02:00
										 |  |  | 			os.Exit(1) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if err := pprof.WriteHeapProfile(f); err != nil { | 
					
						
							| 
									
										
										
										
											2018-08-01 19:09:08 +03:00
										 |  |  | 			fmt.Println("could not write memory profile: ", err) | 
					
						
							| 
									
										
										
										
											2017-05-23 09:34:04 +02:00
										 |  |  | 			os.Exit(1) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		f.Close() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-01 10:19:15 +01:00
										 |  |  | 	if ctx.GlobalBool(DebugFlag.Name) { | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 		if debugLogger != nil { | 
					
						
							| 
									
										
										
										
											2018-08-01 19:09:08 +03:00
										 |  |  | 			fmt.Fprintln(os.Stderr, "#### TRACE ####") | 
					
						
							|  |  |  | 			vm.WriteTrace(os.Stderr, debugLogger.StructLogs()) | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-08-01 19:09:08 +03:00
										 |  |  | 		fmt.Fprintln(os.Stderr, "#### LOGS ####") | 
					
						
							|  |  |  | 		vm.WriteLogs(os.Stderr, statedb.Logs()) | 
					
						
							| 
									
										
										
										
											2017-05-23 09:34:04 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 12:40:07 +02:00
										 |  |  | 	if bench || ctx.GlobalBool(StatDumpFlag.Name) { | 
					
						
							|  |  |  | 		fmt.Fprintf(os.Stderr, `EVM gas used:    %d | 
					
						
							|  |  |  | execution time:  %v | 
					
						
							|  |  |  | allocations:     %d | 
					
						
							|  |  |  | allocated bytes: %d | 
					
						
							|  |  |  | `, initialGas-leftOverGas, stats.time, stats.allocs, stats.bytesAllocated) | 
					
						
							| 
									
										
										
										
											2017-06-07 17:09:08 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-04-06 18:43:36 +08:00
										 |  |  | 	if tracer == nil { | 
					
						
							| 
									
										
										
										
											2019-12-18 09:43:18 +01:00
										 |  |  | 		fmt.Printf("0x%x\n", output) | 
					
						
							| 
									
										
										
										
											2017-08-23 13:37:18 +02:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2018-08-01 19:09:08 +03:00
										 |  |  | 			fmt.Printf(" error: %v\n", err) | 
					
						
							| 
									
										
										
										
											2017-08-23 13:37:18 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-03-01 01:11:24 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } |