Added vm options for object execution
This commit is contained in:
		| @@ -1,33 +1,14 @@ | |||||||
| package ethpipe | package ethpipe | ||||||
|  |  | ||||||
| import ( | import "github.com/ethereum/eth-go/ethutil" | ||||||
| 	"github.com/ethereum/eth-go/ethstate" |  | ||||||
| 	"github.com/ethereum/eth-go/ethutil" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| var cnfCtr = ethutil.Hex2Bytes("661005d2720d855f1d9976f88bb10c1a3398c77f") | var cnfCtr = ethutil.Hex2Bytes("661005d2720d855f1d9976f88bb10c1a3398c77f") | ||||||
|  |  | ||||||
| type object struct { |  | ||||||
| 	*ethstate.StateObject |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (self object) StorageString(str string) *ethutil.Value { |  | ||||||
| 	if ethutil.IsHex(str) { |  | ||||||
| 		return self.Storage(ethutil.Hex2Bytes(str[2:])) |  | ||||||
| 	} else { |  | ||||||
| 		return self.Storage(ethutil.RightPadBytes([]byte(str), 32)) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func (self object) Storage(addr []byte) *ethutil.Value { |  | ||||||
| 	return self.StateObject.GetStorage(ethutil.BigD(addr)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| type config struct { | type config struct { | ||||||
| 	pipe *Pipe | 	pipe *Pipe | ||||||
| } | } | ||||||
|  |  | ||||||
| func (self *config) Get(name string) object { | func (self *config) Get(name string) *object { | ||||||
| 	configCtrl := self.pipe.World().safeGet(cnfCtr) | 	configCtrl := self.pipe.World().safeGet(cnfCtr) | ||||||
| 	var addr []byte | 	var addr []byte | ||||||
|  |  | ||||||
| @@ -39,7 +20,8 @@ func (self *config) Get(name string) object { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	objectAddr := configCtrl.GetStorage(ethutil.BigD(addr)) | 	objectAddr := configCtrl.GetStorage(ethutil.BigD(addr)) | ||||||
| 	return object{self.pipe.World().safeGet(objectAddr.Bytes())} |  | ||||||
|  | 	return &object{self.pipe.World().safeGet(objectAddr.Bytes())} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (self *config) Exist() bool { | func (self *config) Exist() bool { | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								ethpipe/object.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								ethpipe/object.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | package ethpipe | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"github.com/ethereum/eth-go/ethstate" | ||||||
|  | 	"github.com/ethereum/eth-go/ethutil" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type object struct { | ||||||
|  | 	*ethstate.StateObject | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (self *object) StorageString(str string) *ethutil.Value { | ||||||
|  | 	if ethutil.IsHex(str) { | ||||||
|  | 		return self.Storage(ethutil.Hex2Bytes(str[2:])) | ||||||
|  | 	} else { | ||||||
|  | 		return self.Storage(ethutil.RightPadBytes([]byte(str), 32)) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (self *object) StorageValue(addr *ethutil.Value) *ethutil.Value { | ||||||
|  | 	return self.Storage(addr.Bytes()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (self *object) Storage(addr []byte) *ethutil.Value { | ||||||
|  | 	return self.StateObject.GetStorage(ethutil.BigD(addr)) | ||||||
|  | } | ||||||
| @@ -13,11 +13,17 @@ import ( | |||||||
|  |  | ||||||
| var logger = ethlog.NewLogger("PIPE") | var logger = ethlog.NewLogger("PIPE") | ||||||
|  |  | ||||||
|  | type VmVars struct { | ||||||
|  | 	State *ethstate.State | ||||||
|  | } | ||||||
|  |  | ||||||
| type Pipe struct { | type Pipe struct { | ||||||
| 	obj          ethchain.EthManager | 	obj          ethchain.EthManager | ||||||
| 	stateManager *ethchain.StateManager | 	stateManager *ethchain.StateManager | ||||||
| 	blockChain   *ethchain.BlockChain | 	blockChain   *ethchain.BlockChain | ||||||
| 	world        *world | 	world        *world | ||||||
|  |  | ||||||
|  | 	Vm VmVars | ||||||
| } | } | ||||||
|  |  | ||||||
| func New(obj ethchain.EthManager) *Pipe { | func New(obj ethchain.EthManager) *Pipe { | ||||||
| @@ -40,19 +46,22 @@ func (self *Pipe) Nonce(addr []byte) uint64 { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (self *Pipe) Execute(addr []byte, data []byte, value, gas, price *ethutil.Value) ([]byte, error) { | func (self *Pipe) Execute(addr []byte, data []byte, value, gas, price *ethutil.Value) ([]byte, error) { | ||||||
| 	return self.ExecuteObject(self.World().safeGet(addr), data, value, gas, price) | 	return self.ExecuteObject(&object{self.World().safeGet(addr)}, data, value, gas, price) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (self *Pipe) ExecuteObject(object *ethstate.StateObject, data []byte, value, gas, price *ethutil.Value) ([]byte, error) { | func (self *Pipe) ExecuteObject(object *object, data []byte, value, gas, price *ethutil.Value) ([]byte, error) { | ||||||
| 	var ( | 	var ( | ||||||
| 		initiator = ethstate.NewStateObject([]byte{0}) | 		initiator   = ethstate.NewStateObject([]byte{0}) | ||||||
| 		state     = self.World().State().Copy() | 		block       = self.blockChain.CurrentBlock | ||||||
| 		block     = self.blockChain.CurrentBlock | 		stateObject = object.StateObject | ||||||
| 	) | 	) | ||||||
|  | 	if self.Vm.State == nil { | ||||||
|  | 		self.Vm.State = self.World().State().Copy() | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	vm := ethvm.New(NewEnv(state, block, value.BigInt(), initiator.Address())) | 	vm := ethvm.New(NewEnv(self.Vm.State, block, value.BigInt(), initiator.Address())) | ||||||
|  |  | ||||||
| 	closure := ethvm.NewClosure(initiator, object, object.Code, gas.BigInt(), price.BigInt()) | 	closure := ethvm.NewClosure(initiator, stateObject, object.Code, gas.BigInt(), price.BigInt()) | ||||||
| 	ret, _, err := closure.Call(vm, data) | 	ret, _, err := closure.Call(vm, data) | ||||||
|  |  | ||||||
| 	return ret, err | 	return ret, err | ||||||
| @@ -79,7 +88,7 @@ func (self *Pipe) Exists(addr []byte) bool { | |||||||
| 	return self.World().Get(addr) != nil | 	return self.World().Get(addr) != nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (self *Pipe) TransactString(key *ethcrypto.KeyPair, rec string, value, gas, price *ethutil.Value, data []byte) error { | func (self *Pipe) TransactString(key *ethcrypto.KeyPair, rec string, value, gas, price *ethutil.Value, data []byte) ([]byte, error) { | ||||||
| 	// Check if an address is stored by this address | 	// Check if an address is stored by this address | ||||||
| 	var hash []byte | 	var hash []byte | ||||||
| 	addr := self.World().Config().Get("NameReg").StorageString(rec).Bytes() | 	addr := self.World().Config().Get("NameReg").StorageString(rec).Bytes() | ||||||
| @@ -94,7 +103,7 @@ func (self *Pipe) TransactString(key *ethcrypto.KeyPair, rec string, value, gas, | |||||||
| 	return self.Transact(key, hash, value, gas, price, data) | 	return self.Transact(key, hash, value, gas, price, data) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (self *Pipe) Transact(key *ethcrypto.KeyPair, rec []byte, value, gas, price *ethutil.Value, data []byte) error { | func (self *Pipe) Transact(key *ethcrypto.KeyPair, rec []byte, value, gas, price *ethutil.Value, data []byte) ([]byte, error) { | ||||||
| 	var hash []byte | 	var hash []byte | ||||||
| 	var contractCreation bool | 	var contractCreation bool | ||||||
| 	if rec == nil { | 	if rec == nil { | ||||||
| @@ -106,7 +115,7 @@ func (self *Pipe) Transact(key *ethcrypto.KeyPair, rec []byte, value, gas, price | |||||||
| 	if contractCreation { | 	if contractCreation { | ||||||
| 		script, err := ethutil.Compile(string(data), false) | 		script, err := ethutil.Compile(string(data), false) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			return err | 			return nil, err | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		tx = ethchain.NewContractCreationTx(value.BigInt(), gas.BigInt(), price.BigInt(), script) | 		tx = ethchain.NewContractCreationTx(value.BigInt(), gas.BigInt(), price.BigInt(), script) | ||||||
| @@ -133,7 +142,9 @@ func (self *Pipe) Transact(key *ethcrypto.KeyPair, rec []byte, value, gas, price | |||||||
|  |  | ||||||
| 	if contractCreation { | 	if contractCreation { | ||||||
| 		logger.Infof("Contract addr %x", tx.CreationAddress()) | 		logger.Infof("Contract addr %x", tx.CreationAddress()) | ||||||
|  |  | ||||||
|  | 		return tx.CreationAddress(), nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return tx.Hash(), nil | ||||||
| } | } | ||||||
|   | |||||||
| @@ -26,17 +26,17 @@ func (self *world) State() *ethstate.State { | |||||||
| 	return self.pipe.stateManager.CurrentState() | 	return self.pipe.stateManager.CurrentState() | ||||||
| } | } | ||||||
|  |  | ||||||
| func (self *world) Get(addr []byte) *ethstate.StateObject { | func (self *world) Get(addr []byte) *object { | ||||||
| 	return self.State().GetStateObject(addr) | 	return &object{self.State().GetStateObject(addr)} | ||||||
| } | } | ||||||
|  |  | ||||||
| func (self *world) safeGet(addr []byte) *ethstate.StateObject { | func (self *world) safeGet(addr []byte) *ethstate.StateObject { | ||||||
| 	object := self.Get(addr) | 	object := self.State().GetStateObject(addr) | ||||||
| 	if object != nil { | 	if object == nil { | ||||||
| 		return object | 		object = ethstate.NewStateObject(addr) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return ethstate.NewStateObject(addr) | 	return object | ||||||
| } | } | ||||||
|  |  | ||||||
| func (self *world) Coinbase() *ethstate.StateObject { | func (self *world) Coinbase() *ethstate.StateObject { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user