| 
									
										
										
										
											2014-12-04 10:28:02 +01:00
										 |  |  | package core | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"math/big" | 
					
						
							| 
									
										
										
										
											2014-12-20 02:21:13 +01:00
										 |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-13 20:31:31 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/crypto" | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/state" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/vm" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type Execution struct { | 
					
						
							| 
									
										
										
										
											2014-12-19 00:23:00 +01:00
										 |  |  | 	env               vm.Environment | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | 	address, input    []byte | 
					
						
							|  |  |  | 	Gas, price, value *big.Int | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-19 00:18:52 +01:00
										 |  |  | func NewExecution(env vm.Environment, address, input []byte, gas, gasPrice, value *big.Int) *Execution { | 
					
						
							| 
									
										
										
										
											2014-12-19 00:23:00 +01:00
										 |  |  | 	return &Execution{env: env, address: address, input: input, Gas: gas, price: gasPrice, value: value} | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (self *Execution) Addr() []byte { | 
					
						
							|  |  |  | 	return self.address | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-02 16:14:12 +01:00
										 |  |  | func (self *Execution) Call(codeAddr []byte, caller vm.ContextRef) ([]byte, error) { | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | 	// Retrieve the executing code | 
					
						
							| 
									
										
										
										
											2014-12-19 00:23:00 +01:00
										 |  |  | 	code := self.env.State().GetCode(codeAddr) | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return self.exec(code, codeAddr, caller) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-02 16:14:12 +01:00
										 |  |  | func (self *Execution) exec(code, contextAddr []byte, caller vm.ContextRef) (ret []byte, err error) { | 
					
						
							| 
									
										
										
										
											2014-12-19 00:23:00 +01:00
										 |  |  | 	env := self.env | 
					
						
							| 
									
										
										
										
											2015-01-20 15:49:12 +01:00
										 |  |  | 	evm := vm.New(env) | 
					
						
							| 
									
										
										
										
											2014-12-19 00:23:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if env.Depth() == vm.MaxCallDepth { | 
					
						
							| 
									
										
										
										
											2015-01-05 17:37:30 +01:00
										 |  |  | 		caller.ReturnGas(self.Gas, self.price) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-17 23:58:52 +01:00
										 |  |  | 		return nil, vm.DepthError{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-13 20:31:31 +01:00
										 |  |  | 	vsnapshot := env.State().Copy() | 
					
						
							|  |  |  | 	if len(self.address) == 0 { | 
					
						
							|  |  |  | 		// Generate a new address | 
					
						
							|  |  |  | 		nonce := env.State().GetNonce(caller.Address()) | 
					
						
							|  |  |  | 		self.address = crypto.CreateAddress(caller.Address(), nonce) | 
					
						
							|  |  |  | 		env.State().SetNonce(caller.Address(), nonce+1) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-09 20:27:57 +01:00
										 |  |  | 	from, to := env.State().GetStateObject(caller.Address()), env.State().GetOrNewStateObject(self.address) | 
					
						
							| 
									
										
										
										
											2015-01-12 14:40:40 +01:00
										 |  |  | 	err = env.Transfer(from, to, self.value) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2015-01-13 20:31:31 +01:00
										 |  |  | 		env.State().Set(vsnapshot) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-12 14:40:40 +01:00
										 |  |  | 		caller.ReturnGas(self.Gas, self.price) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-13 20:31:31 +01:00
										 |  |  | 		return nil, fmt.Errorf("insufficient funds to transfer value. Req %v, has %v", self.value, from.Balance()) | 
					
						
							| 
									
										
										
										
											2014-12-09 20:27:57 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | 	snapshot := env.State().Copy() | 
					
						
							| 
									
										
										
										
											2014-12-20 02:21:13 +01:00
										 |  |  | 	start := time.Now() | 
					
						
							| 
									
										
										
										
											2014-12-19 00:23:00 +01:00
										 |  |  | 	ret, err = evm.Run(to, caller, code, self.value, self.Gas, self.price, self.input) | 
					
						
							| 
									
										
										
										
											2014-12-20 02:21:13 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		env.State().Set(snapshot) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	chainlogger.Debugf("vm took %v\n", time.Since(start)) | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-02 16:14:12 +01:00
										 |  |  | func (self *Execution) Create(caller vm.ContextRef) (ret []byte, err error, account *state.StateObject) { | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | 	ret, err = self.exec(self.input, nil, caller) | 
					
						
							| 
									
										
										
										
											2014-12-19 00:23:00 +01:00
										 |  |  | 	account = self.env.State().GetStateObject(self.address) | 
					
						
							| 
									
										
										
										
											2014-12-03 17:06:54 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return | 
					
						
							|  |  |  | } |