| 
									
										
										
										
											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> | 
					
						
							|  |  |  |  * @date 2014 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package main | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"flag" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"log" | 
					
						
							|  |  |  | 	"math/big" | 
					
						
							|  |  |  | 	"os" | 
					
						
							|  |  |  | 	"runtime" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							| 
									
										
										
										
											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/ethutil" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/logger" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/state" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/trie" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/vm" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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))) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 12:35:23 +01:00
										 |  |  | 	ethutil.ReadConfig("/tmp/evmtest", "/tmp/evm", "") | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	db, _ := ethdb.NewMemDatabase() | 
					
						
							|  |  |  | 	statedb := state.New(trie.New(db, "")) | 
					
						
							|  |  |  | 	sender := statedb.NewStateObject([]byte("sender")) | 
					
						
							|  |  |  | 	receiver := statedb.NewStateObject([]byte("receiver")) | 
					
						
							| 
									
										
										
										
											2014-12-04 12:09:22 +01:00
										 |  |  | 	//receiver.SetCode([]byte(*code)) | 
					
						
							|  |  |  | 	receiver.SetCode(ethutil.Hex2Bytes(*code)) | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	vmenv := NewEnv(statedb, []byte("evmuser"), ethutil.Big(*value)) | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	tstart := time.Now() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	ret, e := vmenv.Call(sender, receiver.Address(), ethutil.Hex2Bytes(*data), ethutil.Big(*gas), ethutil.Big(*price), ethutil.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
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	transactor []byte | 
					
						
							|  |  |  | 	value      *big.Int | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	depth int | 
					
						
							|  |  |  | 	Gas   *big.Int | 
					
						
							|  |  |  | 	time  int64 | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 12:09:22 +01:00
										 |  |  | func NewEnv(state *state.StateDB, transactor []byte, value *big.Int) *VMEnv { | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | 	return &VMEnv{ | 
					
						
							|  |  |  | 		state:      state, | 
					
						
							|  |  |  | 		transactor: transactor, | 
					
						
							|  |  |  | 		value:      value, | 
					
						
							|  |  |  | 		time:       time.Now().Unix(), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 12:09:22 +01:00
										 |  |  | func (self *VMEnv) State() *state.StateDB { return self.state } | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | func (self *VMEnv) Origin() []byte        { return self.transactor } | 
					
						
							|  |  |  | func (self *VMEnv) BlockNumber() *big.Int { return ethutil.Big0 } | 
					
						
							|  |  |  | func (self *VMEnv) PrevHash() []byte      { return make([]byte, 32) } | 
					
						
							|  |  |  | func (self *VMEnv) Coinbase() []byte      { return self.transactor } | 
					
						
							|  |  |  | func (self *VMEnv) Time() int64           { return self.time } | 
					
						
							|  |  |  | func (self *VMEnv) Difficulty() *big.Int  { return ethutil.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) } | 
					
						
							| 
									
										
										
										
											2014-12-04 12:09:22 +01:00
										 |  |  | func (self *VMEnv) Depth() int            { return 0 } | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | func (self *VMEnv) SetDepth(i int)        { self.depth = i } | 
					
						
							| 
									
										
										
										
											2014-12-10 10:57:19 +01: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) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *VMEnv) vm(addr, data []byte, gas, price, value *big.Int) *core.Execution { | 
					
						
							|  |  |  | 	evm := vm.New(self, vm.DebugVmTy) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return core.NewExecution(evm, addr, data, gas, price, value) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *VMEnv) Call(caller vm.ClosureRef, addr, data []byte, gas, price, value *big.Int) ([]byte, error) { | 
					
						
							|  |  |  | 	exe := self.vm(addr, data, gas, price, value) | 
					
						
							|  |  |  | 	ret, err := exe.Call(addr, caller) | 
					
						
							|  |  |  | 	self.Gas = exe.Gas | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret, err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func (self *VMEnv) CallCode(caller vm.ClosureRef, addr, data []byte, gas, price, value *big.Int) ([]byte, error) { | 
					
						
							|  |  |  | 	exe := self.vm(caller.Address(), data, gas, price, value) | 
					
						
							|  |  |  | 	return exe.Call(addr, caller) | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 11:30:41 +01:00
										 |  |  | func (self *VMEnv) Create(caller vm.ClosureRef, addr, data []byte, gas, price, value *big.Int) ([]byte, error, vm.ClosureRef) { | 
					
						
							|  |  |  | 	exe := self.vm(addr, data, gas, price, value) | 
					
						
							|  |  |  | 	return exe.Create(caller) | 
					
						
							| 
									
										
										
										
											2014-11-10 11:47:37 +01:00
										 |  |  | } |