| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // Copyright 2014 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 | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							| 
									
										
										
										
											2015-07-07 02:54:22 +02:00
										 |  |  | // GNU General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU General Public License | 
					
						
							| 
									
										
										
										
											2015-07-22 18:48:40 +02:00
										 |  |  | // along with go-ethereum. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-07 05:08:16 +02:00
										 |  |  | // evm executes EVM code snippets. | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | package main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2016-10-29 10:07:38 +01:00
										 |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	goruntime "runtime" | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/cmd/utils" | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2015-03-24 15:27:05 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/state" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/core/vm" | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/vm/runtime" | 
					
						
							| 
									
										
										
										
											2016-10-01 15:44:53 +03:00
										 |  |  | 	"github.com/ethereum/go-ethereum/crypto" | 
					
						
							| 
									
										
										
										
											2014-11-10 12:42:16 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/ethdb" | 
					
						
							| 
									
										
										
										
											2015-07-17 23:09:36 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/logger/glog" | 
					
						
							| 
									
										
										
										
											2016-06-09 11:44:42 +02:00
										 |  |  | 	"gopkg.in/urfave/cli.v1" | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-05 13:08:41 +02:00
										 |  |  | var gitCommit = "" // Git SHA1 commit hash of the release (set via linker flags) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | var ( | 
					
						
							| 
									
										
										
										
											2016-09-05 13:08:41 +02:00
										 |  |  | 	app = utils.NewApp(gitCommit, "the evm command line interface") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 	DebugFlag = cli.BoolFlag{ | 
					
						
							|  |  |  | 		Name:  "debug", | 
					
						
							|  |  |  | 		Usage: "output full trace logs", | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	CodeFlag = cli.StringFlag{ | 
					
						
							|  |  |  | 		Name:  "code", | 
					
						
							|  |  |  | 		Usage: "EVM code", | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-10-29 10:07:38 +01:00
										 |  |  | 	CodeFileFlag = cli.StringFlag{ | 
					
						
							|  |  |  | 		Name:  "codefile", | 
					
						
							|  |  |  | 		Usage: "file containing EVM code", | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 	GasFlag = cli.StringFlag{ | 
					
						
							|  |  |  | 		Name:  "gas", | 
					
						
							|  |  |  | 		Usage: "gas limit for the evm", | 
					
						
							|  |  |  | 		Value: "10000000000", | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	PriceFlag = cli.StringFlag{ | 
					
						
							|  |  |  | 		Name:  "price", | 
					
						
							|  |  |  | 		Usage: "price set for the evm", | 
					
						
							|  |  |  | 		Value: "0", | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	ValueFlag = cli.StringFlag{ | 
					
						
							|  |  |  | 		Name:  "value", | 
					
						
							|  |  |  | 		Usage: "value set for the evm", | 
					
						
							|  |  |  | 		Value: "0", | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	DumpFlag = cli.BoolFlag{ | 
					
						
							|  |  |  | 		Name:  "dump", | 
					
						
							|  |  |  | 		Usage: "dumps the state after the run", | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	InputFlag = cli.StringFlag{ | 
					
						
							|  |  |  | 		Name:  "input", | 
					
						
							|  |  |  | 		Usage: "input for the EVM", | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	SysStatFlag = cli.BoolFlag{ | 
					
						
							|  |  |  | 		Name:  "sysstat", | 
					
						
							|  |  |  | 		Usage: "display system stats", | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-08-30 10:19:10 +02:00
										 |  |  | 	VerbosityFlag = cli.IntFlag{ | 
					
						
							|  |  |  | 		Name:  "verbosity", | 
					
						
							|  |  |  | 		Usage: "sets the verbosity level", | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-06-14 16:09:27 +02:00
										 |  |  | 	CreateFlag = cli.BoolFlag{ | 
					
						
							|  |  |  | 		Name:  "create", | 
					
						
							|  |  |  | 		Usage: "indicates the action should be create rather than call", | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-01-05 11:52:10 +01:00
										 |  |  | 	DisableGasMeteringFlag = cli.BoolFlag{ | 
					
						
							|  |  |  | 		Name:  "nogasmetering", | 
					
						
							|  |  |  | 		Usage: "disable gas metering", | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | func init() { | 
					
						
							|  |  |  | 	app.Flags = []cli.Flag{ | 
					
						
							| 
									
										
										
										
											2016-06-14 16:09:27 +02:00
										 |  |  | 		CreateFlag, | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 		DebugFlag, | 
					
						
							| 
									
										
										
										
											2015-08-30 10:19:10 +02:00
										 |  |  | 		VerbosityFlag, | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 		SysStatFlag, | 
					
						
							|  |  |  | 		CodeFlag, | 
					
						
							| 
									
										
										
										
											2016-10-29 10:07:38 +01:00
										 |  |  | 		CodeFileFlag, | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 		GasFlag, | 
					
						
							|  |  |  | 		PriceFlag, | 
					
						
							|  |  |  | 		ValueFlag, | 
					
						
							|  |  |  | 		DumpFlag, | 
					
						
							|  |  |  | 		InputFlag, | 
					
						
							| 
									
										
										
										
											2017-01-05 11:52:10 +01:00
										 |  |  | 		DisableGasMeteringFlag, | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	app.Action = run | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-10 11:23:00 +03:00
										 |  |  | func run(ctx *cli.Context) error { | 
					
						
							| 
									
										
										
										
											2015-07-17 23:09:36 +02:00
										 |  |  | 	glog.SetToStderr(true) | 
					
						
							| 
									
										
										
										
											2015-08-30 10:19:10 +02:00
										 |  |  | 	glog.SetV(ctx.GlobalInt(VerbosityFlag.Name)) | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	db, _ := ethdb.NewMemDatabase() | 
					
						
							| 
									
										
										
										
											2015-10-06 16:35:55 +02:00
										 |  |  | 	statedb, _ := state.New(common.Hash{}, db) | 
					
						
							| 
									
										
										
										
											2015-04-01 11:42:02 +02:00
										 |  |  | 	sender := statedb.CreateAccount(common.StringToAddress("sender")) | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-19 15:19:51 +01:00
										 |  |  | 	logger := vm.NewStructLogger(nil) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 	tstart := time.Now() | 
					
						
							| 
									
										
										
										
											2016-06-14 16:09:27 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	var ( | 
					
						
							| 
									
										
										
										
											2016-10-29 10:07:38 +01:00
										 |  |  | 		code []byte | 
					
						
							|  |  |  | 		ret  []byte | 
					
						
							|  |  |  | 		err  error | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 	) | 
					
						
							| 
									
										
										
										
											2016-06-14 16:09:27 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-29 10:07:38 +01:00
										 |  |  | 	if ctx.GlobalString(CodeFlag.Name) != "" { | 
					
						
							|  |  |  | 		code = common.Hex2Bytes(ctx.GlobalString(CodeFlag.Name)) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		var hexcode []byte | 
					
						
							|  |  |  | 		if ctx.GlobalString(CodeFileFlag.Name) != "" { | 
					
						
							|  |  |  | 			var err error | 
					
						
							|  |  |  | 			hexcode, err = ioutil.ReadFile(ctx.GlobalString(CodeFileFlag.Name)) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				fmt.Printf("Could not load code from file: %v\n", err) | 
					
						
							|  |  |  | 				os.Exit(1) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			var err error | 
					
						
							|  |  |  | 			hexcode, err = ioutil.ReadAll(os.Stdin) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							|  |  |  | 				fmt.Printf("Could not load code from stdin: %v\n", err) | 
					
						
							|  |  |  | 				os.Exit(1) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		code = common.Hex2Bytes(string(hexcode[:])) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-14 16:09:27 +02:00
										 |  |  | 	if ctx.GlobalBool(CreateFlag.Name) { | 
					
						
							| 
									
										
										
										
											2016-10-29 10:07:38 +01:00
										 |  |  | 		input := append(code, common.Hex2Bytes(ctx.GlobalString(InputFlag.Name))...) | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 		ret, _, err = runtime.Create(input, &runtime.Config{ | 
					
						
							|  |  |  | 			Origin:   sender.Address(), | 
					
						
							|  |  |  | 			State:    statedb, | 
					
						
							| 
									
										
										
										
											2017-01-04 20:17:24 +01:00
										 |  |  | 			GasLimit: common.Big(ctx.GlobalString(GasFlag.Name)).Uint64(), | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 			GasPrice: common.Big(ctx.GlobalString(PriceFlag.Name)), | 
					
						
							|  |  |  | 			Value:    common.Big(ctx.GlobalString(ValueFlag.Name)), | 
					
						
							|  |  |  | 			EVMConfig: vm.Config{ | 
					
						
							| 
									
										
										
										
											2017-01-05 11:52:10 +01:00
										 |  |  | 				Tracer:             logger, | 
					
						
							| 
									
										
										
										
											2017-01-22 22:14:09 +01:00
										 |  |  | 				Debug:              ctx.GlobalBool(DebugFlag.Name), | 
					
						
							| 
									
										
										
										
											2017-01-05 11:52:10 +01:00
										 |  |  | 				DisableGasMetering: ctx.GlobalBool(DisableGasMeteringFlag.Name), | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 			}, | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2016-06-14 16:09:27 +02:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		receiver := statedb.CreateAccount(common.StringToAddress("receiver")) | 
					
						
							| 
									
										
										
										
											2016-10-01 15:44:53 +03:00
										 |  |  | 		receiver.SetCode(crypto.Keccak256Hash(code), code) | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		ret, err = runtime.Call(receiver.Address(), common.Hex2Bytes(ctx.GlobalString(InputFlag.Name)), &runtime.Config{ | 
					
						
							|  |  |  | 			Origin:   sender.Address(), | 
					
						
							|  |  |  | 			State:    statedb, | 
					
						
							| 
									
										
										
										
											2017-01-04 20:17:24 +01:00
										 |  |  | 			GasLimit: common.Big(ctx.GlobalString(GasFlag.Name)).Uint64(), | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 			GasPrice: common.Big(ctx.GlobalString(PriceFlag.Name)), | 
					
						
							|  |  |  | 			Value:    common.Big(ctx.GlobalString(ValueFlag.Name)), | 
					
						
							|  |  |  | 			EVMConfig: vm.Config{ | 
					
						
							| 
									
										
										
										
											2017-01-05 11:52:10 +01:00
										 |  |  | 				Tracer:             logger, | 
					
						
							| 
									
										
										
										
											2017-01-22 22:14:09 +01:00
										 |  |  | 				Debug:              ctx.GlobalBool(DebugFlag.Name), | 
					
						
							| 
									
										
										
										
											2017-01-05 11:52:10 +01:00
										 |  |  | 				DisableGasMetering: ctx.GlobalBool(DisableGasMeteringFlag.Name), | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 			}, | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2016-06-14 16:09:27 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 	vmdone := time.Since(tstart) | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 	if ctx.GlobalBool(DumpFlag.Name) { | 
					
						
							| 
									
										
										
										
											2016-10-20 13:36:29 +02:00
										 |  |  | 		statedb.Commit(true) | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 		fmt.Println(string(statedb.Dump())) | 
					
						
							| 
									
										
										
										
											2014-11-10 12:42:16 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-08-19 15:19:51 +01:00
										 |  |  | 	vm.StdErrFormat(logger.StructLogs()) | 
					
						
							| 
									
										
										
										
											2015-06-10 21:09:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 	if ctx.GlobalBool(SysStatFlag.Name) { | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 		var mem goruntime.MemStats | 
					
						
							|  |  |  | 		goruntime.ReadMemStats(&mem) | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 		fmt.Printf("vm took %v\n", vmdone) | 
					
						
							|  |  |  | 		fmt.Printf(`alloc:      %d | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | tot alloc:  %d | 
					
						
							|  |  |  | no. malloc: %d | 
					
						
							|  |  |  | heap alloc: %d | 
					
						
							|  |  |  | heap objs:  %d | 
					
						
							|  |  |  | num gc:     %d | 
					
						
							|  |  |  | `, mem.Alloc, mem.TotalAlloc, mem.Mallocs, mem.HeapAlloc, mem.HeapObjects, mem.NumGC) | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-17 23:09:36 +02:00
										 |  |  | 	fmt.Printf("OUT: 0x%x", ret) | 
					
						
							| 
									
										
										
										
											2016-06-14 16:09:27 +02:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		fmt.Printf(" error: %v", err) | 
					
						
							| 
									
										
										
										
											2015-07-17 23:09:36 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	fmt.Println() | 
					
						
							| 
									
										
										
										
											2016-06-10 11:23:00 +03:00
										 |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2015-07-17 15:43:16 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func main() { | 
					
						
							|  |  |  | 	if err := app.Run(os.Args); err != nil { | 
					
						
							|  |  |  | 		fmt.Fprintln(os.Stderr, err) | 
					
						
							|  |  |  | 		os.Exit(1) | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | } |