| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | /* | 
					
						
							|  |  |  | 	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/>. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @authors | 
					
						
							|  |  |  |  * 	Jeffrey Wilcke <i@jev.io> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"flag" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"log" | 
					
						
							|  |  |  | 	"math/big" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"runtime" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core" | 
					
						
							| 
									
										
										
										
											2015-03-24 15:27:05 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/state" | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							| 
									
										
										
										
											2015-03-24 15:27:05 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/vm" | 
					
						
							| 
									
										
										
										
											2014-11-10 12:42:16 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/ethdb" | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/logger" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							|  |  |  | 	code     = flag.String("code", "", "evm code") | 
					
						
							|  |  |  | 	loglevel = flag.Int("log", 4, "log level") | 
					
						
							| 
									
										
										
										
											2014-12-04 12:09:22 +01:00
										 |  |  | 	gas      = flag.String("gas", "1000000000", "gas amount") | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 	price    = flag.String("price", "0", "gas price") | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	value    = flag.String("value", "0", "tx value") | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 	dump     = flag.Bool("dump", false, "dump state after run") | 
					
						
							| 
									
										
										
										
											2014-11-11 22:51:26 +01:00
										 |  |  | 	data     = flag.String("data", "", "data") | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func perr(v ...interface{}) { | 
					
						
							|  |  |  | 	fmt.Println(v...) | 
					
						
							|  |  |  | 	//os.Exit(1) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func main() { | 
					
						
							|  |  |  | 	flag.Parse() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	logger.AddLogSystem(logger.NewStdLogSystem(os.Stdout, log.LstdFlags, logger.LogLevel(*loglevel))) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-10 21:09:12 +02:00
										 |  |  | 	vm.Debug = true | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	db, _ := ethdb.NewMemDatabase() | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | 	statedb := state.New(common.Hash{}, db) | 
					
						
							| 
									
										
										
										
											2015-04-01 11:42:02 +02:00
										 |  |  | 	sender := statedb.CreateAccount(common.StringToAddress("sender")) | 
					
						
							|  |  |  | 	receiver := statedb.CreateAccount(common.StringToAddress("receiver")) | 
					
						
							| 
									
										
										
										
											2015-03-16 11:27:38 +01:00
										 |  |  | 	receiver.SetCode(common.Hex2Bytes(*code)) | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | 	vmenv := NewEnv(statedb, common.StringToAddress("evmuser"), common.Big(*value)) | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	tstart := time.Now() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-16 11:27:38 +01:00
										 |  |  | 	ret, e := vmenv.Call(sender, receiver.Address(), common.Hex2Bytes(*data), common.Big(*gas), common.Big(*price), common.Big(*value)) | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	logger.Flush() | 
					
						
							|  |  |  | 	if e != nil { | 
					
						
							|  |  |  | 		perr(e) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-10 12:42:16 +01:00
										 |  |  | 	if *dump { | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 		fmt.Println(string(statedb.Dump())) | 
					
						
							| 
									
										
										
										
											2014-11-10 12:42:16 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-10 21:09:12 +02:00
										 |  |  | 	vm.StdErrFormat(vmenv.StructLogs()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 	var mem runtime.MemStats | 
					
						
							|  |  |  | 	runtime.ReadMemStats(&mem) | 
					
						
							|  |  |  | 	fmt.Printf("vm took %v\n", time.Since(tstart)) | 
					
						
							|  |  |  | 	fmt.Printf(`alloc:      %d | 
					
						
							|  |  |  | 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) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fmt.Printf("%x\n", ret) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | type VMEnv struct { | 
					
						
							| 
									
										
										
										
											2014-12-04 12:09:22 +01:00
										 |  |  | 	state *state.StateDB | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	block *types.Block | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | 	transactor *common.Address | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	value      *big.Int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	depth int | 
					
						
							|  |  |  | 	Gas   *big.Int | 
					
						
							|  |  |  | 	time  int64 | 
					
						
							| 
									
										
										
										
											2015-06-10 16:46:43 +02:00
										 |  |  | 	logs  []vm.StructLog | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | func NewEnv(state *state.StateDB, transactor common.Address, value *big.Int) *VMEnv { | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	return &VMEnv{ | 
					
						
							|  |  |  | 		state:      state, | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | 		transactor: &transactor, | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 		value:      value, | 
					
						
							|  |  |  | 		time:       time.Now().Unix(), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | func (self *VMEnv) State() *state.StateDB    { return self.state } | 
					
						
							|  |  |  | func (self *VMEnv) Origin() common.Address   { return *self.transactor } | 
					
						
							|  |  |  | func (self *VMEnv) BlockNumber() *big.Int    { return common.Big0 } | 
					
						
							|  |  |  | func (self *VMEnv) Coinbase() common.Address { return *self.transactor } | 
					
						
							|  |  |  | func (self *VMEnv) Time() int64              { return self.time } | 
					
						
							|  |  |  | func (self *VMEnv) Difficulty() *big.Int     { return common.Big1 } | 
					
						
							|  |  |  | func (self *VMEnv) BlockHash() []byte        { return make([]byte, 32) } | 
					
						
							|  |  |  | func (self *VMEnv) Value() *big.Int          { return self.value } | 
					
						
							|  |  |  | func (self *VMEnv) GasLimit() *big.Int       { return big.NewInt(1000000000) } | 
					
						
							|  |  |  | func (self *VMEnv) VmType() vm.Type          { return vm.StdVmTy } | 
					
						
							|  |  |  | func (self *VMEnv) Depth() int               { return 0 } | 
					
						
							|  |  |  | func (self *VMEnv) SetDepth(i int)           { self.depth = i } | 
					
						
							|  |  |  | func (self *VMEnv) GetHash(n uint64) common.Hash { | 
					
						
							| 
									
										
										
										
											2015-01-06 11:07:05 +01:00
										 |  |  | 	if self.block.Number().Cmp(big.NewInt(int64(n))) == 0 { | 
					
						
							|  |  |  | 		return self.block.Hash() | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | 	return common.Hash{} | 
					
						
							| 
									
										
										
										
											2015-01-06 11:07:05 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-06-10 16:46:43 +02:00
										 |  |  | func (self *VMEnv) AddStructLog(log vm.StructLog) { | 
					
						
							|  |  |  | 	self.logs = append(self.logs, log) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func (self *VMEnv) StructLogs() []vm.StructLog { | 
					
						
							|  |  |  | 	return self.logs | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-04-08 20:45:39 +02:00
										 |  |  | func (self *VMEnv) AddLog(log *state.Log) { | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	self.state.AddLog(log) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func (self *VMEnv) Transfer(from, to vm.Account, amount *big.Int) error { | 
					
						
							|  |  |  | 	return vm.Transfer(from, to, amount) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | func (self *VMEnv) vm(addr *common.Address, data []byte, gas, price, value *big.Int) *core.Execution { | 
					
						
							| 
									
										
										
										
											2014-12-21 15:13:06 +01:00
										 |  |  | 	return core.NewExecution(self, addr, data, gas, price, value) | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | func (self *VMEnv) Call(caller vm.ContextRef, addr common.Address, data []byte, gas, price, value *big.Int) ([]byte, error) { | 
					
						
							|  |  |  | 	exe := self.vm(&addr, data, gas, price, value) | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	ret, err := exe.Call(addr, caller) | 
					
						
							|  |  |  | 	self.Gas = exe.Gas | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret, err | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-03-22 12:12:29 +01:00
										 |  |  | func (self *VMEnv) CallCode(caller vm.ContextRef, addr common.Address, data []byte, gas, price, value *big.Int) ([]byte, error) { | 
					
						
							|  |  |  | 	a := caller.Address() | 
					
						
							|  |  |  | 	exe := self.vm(&a, data, gas, price, value) | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	return exe.Call(addr, caller) | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 15:27:05 +01:00
										 |  |  | func (self *VMEnv) Create(caller vm.ContextRef, data []byte, gas, price, value *big.Int) ([]byte, error, vm.ContextRef) { | 
					
						
							|  |  |  | 	exe := self.vm(nil, data, gas, price, value) | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	return exe.Create(caller) | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | } |