| 
									
										
										
										
											2017-04-14 10:29:00 +02:00
										 |  |  | // Copyright 2016 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  | // it under the terms of the GNU Lesser General Public License as published by | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // The go-ethereum library 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 Lesser General Public License for more details. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // You should have received a copy of the GNU Lesser General Public License | 
					
						
							|  |  |  | // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package core | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"math/big" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/common" | 
					
						
							| 
									
										
										
										
											2017-04-12 16:38:31 +03:00
										 |  |  | 	"github.com/ethereum/go-ethereum/consensus" | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/core/types" | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/core/vm" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 16:38:31 +03:00
										 |  |  | // ChainContext supports retrieving headers and consensus parameters from the | 
					
						
							|  |  |  | // current blockchain to be used during transaction processing. | 
					
						
							|  |  |  | type ChainContext interface { | 
					
						
							|  |  |  | 	// Engine retrieves the chain's consensus engine. | 
					
						
							|  |  |  | 	Engine() consensus.Engine | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// GetHeader returns the hash corresponding to their hash. | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	GetHeader(common.Hash, uint64) *types.Header | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-13 13:42:19 +01:00
										 |  |  | // NewEVMBlockContext creates a new context for use in the EVM. | 
					
						
							|  |  |  | func NewEVMBlockContext(header *types.Header, chain ChainContext, author *common.Address) vm.BlockContext { | 
					
						
							| 
									
										
										
										
											2017-04-12 16:38:31 +03:00
										 |  |  | 	// If we don't have an explicit author (i.e. not mining), extract from the header | 
					
						
							|  |  |  | 	var beneficiary common.Address | 
					
						
							|  |  |  | 	if author == nil { | 
					
						
							|  |  |  | 		beneficiary, _ = chain.Engine().Author(header) // Ignore error, we're past header validation | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		beneficiary = *author | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2020-11-13 13:42:19 +01:00
										 |  |  | 	return vm.BlockContext{ | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 		CanTransfer: CanTransfer, | 
					
						
							|  |  |  | 		Transfer:    Transfer, | 
					
						
							|  |  |  | 		GetHash:     GetHashFn(header, chain), | 
					
						
							| 
									
										
										
										
											2017-04-12 16:38:31 +03:00
										 |  |  | 		Coinbase:    beneficiary, | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 		BlockNumber: new(big.Int).Set(header.Number), | 
					
						
							| 
									
										
										
										
											2019-04-02 22:28:48 +02:00
										 |  |  | 		Time:        new(big.Int).SetUint64(header.Time), | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 		Difficulty:  new(big.Int).Set(header.Difficulty), | 
					
						
							| 
									
										
										
										
											2017-11-13 13:47:27 +02:00
										 |  |  | 		GasLimit:    header.GasLimit, | 
					
						
							| 
									
										
										
										
											2020-11-13 13:42:19 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NewEVMTxContext creates a new transaction context for a single transaction. | 
					
						
							|  |  |  | func NewEVMTxContext(msg Message) vm.TxContext { | 
					
						
							|  |  |  | 	return vm.TxContext{ | 
					
						
							|  |  |  | 		Origin:   msg.From(), | 
					
						
							|  |  |  | 		GasPrice: new(big.Int).Set(msg.GasPrice()), | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // GetHashFn returns a GetHashFunc which retrieves header hashes by number | 
					
						
							| 
									
										
										
										
											2017-04-12 16:38:31 +03:00
										 |  |  | func GetHashFn(ref *types.Header, chain ChainContext) func(n uint64) common.Hash { | 
					
						
							| 
									
										
										
										
											2020-02-04 11:32:31 +01:00
										 |  |  | 	// Cache will initially contain [refHash.parent], | 
					
						
							|  |  |  | 	// Then fill up with [refHash.p, refHash.pp, refHash.ppp, ...] | 
					
						
							|  |  |  | 	var cache []common.Hash | 
					
						
							| 
									
										
										
										
											2018-03-26 12:28:46 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 	return func(n uint64) common.Hash { | 
					
						
							| 
									
										
										
										
											2018-03-26 12:28:46 +03:00
										 |  |  | 		// If there's no hash cache yet, make one | 
					
						
							| 
									
										
										
										
											2020-02-04 11:32:31 +01:00
										 |  |  | 		if len(cache) == 0 { | 
					
						
							|  |  |  | 			cache = append(cache, ref.ParentHash) | 
					
						
							| 
									
										
										
										
											2018-03-26 12:28:46 +03:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-02-04 11:32:31 +01:00
										 |  |  | 		if idx := ref.Number.Uint64() - n - 1; idx < uint64(len(cache)) { | 
					
						
							|  |  |  | 			return cache[idx] | 
					
						
							| 
									
										
										
										
											2018-03-26 12:28:46 +03:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-02-04 11:32:31 +01:00
										 |  |  | 		// No luck in the cache, but we can start iterating from the last element we already know | 
					
						
							|  |  |  | 		lastKnownHash := cache[len(cache)-1] | 
					
						
							|  |  |  | 		lastKnownNumber := ref.Number.Uint64() - uint64(len(cache)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for { | 
					
						
							|  |  |  | 			header := chain.GetHeader(lastKnownHash, lastKnownNumber) | 
					
						
							|  |  |  | 			if header == nil { | 
					
						
							|  |  |  | 				break | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			cache = append(cache, header.ParentHash) | 
					
						
							|  |  |  | 			lastKnownHash = header.ParentHash | 
					
						
							|  |  |  | 			lastKnownNumber = header.Number.Uint64() - 1 | 
					
						
							|  |  |  | 			if n == lastKnownNumber { | 
					
						
							|  |  |  | 				return lastKnownHash | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return common.Hash{} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-25 18:04:38 +07:00
										 |  |  | // CanTransfer checks whether there are enough funds in the address' account to make a transfer. | 
					
						
							| 
									
										
										
										
											2016-12-06 02:16:03 +01:00
										 |  |  | // This does not take the necessary gas in to account to make the transfer valid. | 
					
						
							|  |  |  | func CanTransfer(db vm.StateDB, addr common.Address, amount *big.Int) bool { | 
					
						
							|  |  |  | 	return db.GetBalance(addr).Cmp(amount) >= 0 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Transfer subtracts amount from sender and adds amount to recipient using the given Db | 
					
						
							|  |  |  | func Transfer(db vm.StateDB, sender, recipient common.Address, amount *big.Int) { | 
					
						
							|  |  |  | 	db.SubBalance(sender, amount) | 
					
						
							|  |  |  | 	db.AddBalance(recipient, amount) | 
					
						
							|  |  |  | } |