core,state: finish implementing Eip 1283

This commit is contained in:
Martin Holst Swende 2018-08-12 14:47:03 +02:00 committed by Péter Szilágyi
parent 58374e28d9
commit caa2c23a38
No known key found for this signature in database
GPG Key ID: E9AE538CEDF8293D
6 changed files with 41 additions and 16 deletions

View File

@ -183,6 +183,16 @@ func (self *stateObject) GetState(db Database, key common.Hash) common.Hash {
return value return value
} }
// GetOriginalStateValue returns the state value that is currently in the Trie, that is, ignoring any
// changes that have been made but not yet written to trie.
func (self *stateObject) GetOriginalStateValue(db Database, key common.Hash) common.Hash{
if original, exist:= self.originalValue[key]; exist {
// original value has been set, return it
return original
}
return self.GetState(db, key)
}
// SetState updates a value in account storage. // SetState updates a value in account storage.
func (self *stateObject) SetState(db Database, key, value common.Hash) { func (self *stateObject) SetState(db Database, key, value common.Hash) {
prev := self.GetState(db, key) prev := self.GetState(db, key)

View File

@ -255,6 +255,14 @@ func (self *StateDB) GetState(addr common.Address, bhash common.Hash) common.Has
return common.Hash{} return common.Hash{}
} }
func (self *StateDB) GetStateOriginal(addr common.Address, bhash common.Hash) common.Hash {
stateObject := self.getStateObject(addr)
if stateObject != nil {
return stateObject.GetOriginalStateValue(self.db, bhash)
}
return common.Hash{}
}
// Database retrieves the low level database supporting the lower level trie ops. // Database retrieves the low level database supporting the lower level trie ops.
func (self *StateDB) Database() Database { func (self *StateDB) Database() Database {
return self.db return self.db

View File

@ -17,11 +17,9 @@
package vm package vm
import ( import (
"bytes"
"github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/common/math" "github.com/ethereum/go-ethereum/common/math"
"github.com/ethereum/go-ethereum/params" "github.com/ethereum/go-ethereum/params"
"math/big"
) )
// memoryGasCosts calculates the quadratic gas for memory expansion. It does so // memoryGasCosts calculates the quadratic gas for memory expansion. It does so
@ -117,7 +115,7 @@ func gasReturnDataCopy(gt params.GasTable, evm *EVM, contract *Contract, stack *
return gas, nil return gas, nil
} }
func gasSStoreOld(gt params.GasTable, evm *EVM, contract *Contract, stack *Stack, mem *Memory, memorySize uint64) (uint64, error) { func gasSStore(gt params.GasTable, evm *EVM, contract *Contract, stack *Stack, mem *Memory, memorySize uint64) (uint64, error) {
var ( var (
y, x = stack.Back(1), stack.Back(0) y, x = stack.Back(1), stack.Back(0)
val = evm.StateDB.GetState(contract.Address(), common.BigToHash(x)) val = evm.StateDB.GetState(contract.Address(), common.BigToHash(x))
@ -139,10 +137,11 @@ func gasSStoreOld(gt params.GasTable, evm *EVM, contract *Contract, stack *Stack
} }
} }
func gasSStore(gt params.GasTable, evm *EVM, contract *Contract, stack *Stack, mem *Memory, memorySize uint64) (uint64, error) { // gasSStoreEip1283 calculates SSTORE gas cost according to EIP-1283
func gasSStoreEip1283(gt params.GasTable, evm *EVM, contract *Contract, stack *Stack, mem *Memory, memorySize uint64) (uint64, error) {
var ( var (
y, x = stack.Back(1), stack.Back(0) y, x = stack.Back(1), stack.Back(0)
current = evm.StateDB.GetState(contract.Address(), common.BigToHash(x)) current = evm.StateDB.GetState(contract.Address(), common.BigToHash(x))
) )
//1. If current value equals new value (this is a no-op), 200 gas is deducted. //1. If current value equals new value (this is a no-op), 200 gas is deducted.
//2. If current value does not equal new value //2. If current value does not equal new value
@ -161,33 +160,31 @@ func gasSStore(gt params.GasTable, evm *EVM, contract *Contract, stack *Stack, m
// 1. current == new // 1. current == new
return 200, nil return 200, nil
} }
// Todo, get this value original := evm.StateDB.GetStateOriginal(contract.Address(), common.BigToHash(x))
original := common.Hash{}
// 2 // 2
if original == current { // 2.1 if original == current { // 2.1
if original == (common.Hash{}){ // 2.1.1 if original == (common.Hash{}) { // 2.1.1
return 20000, nil return 20000, nil
} }
// 2.1.2 // 2.1.2
if new == (common.Hash{}){ if new == (common.Hash{}) {
evm.StateDB.AddRefund(15000) evm.StateDB.AddRefund(15000)
} }
return 5000, nil return 5000, nil
} }
// 2.2 // 2.2
if original != (common.Hash{}){ // 2.2.1 if original != (common.Hash{}) { // 2.2.1
if current == (common.Hash{}){ // 2.2.1.1 if current == (common.Hash{}) { // 2.2.1.1
evm.StateDB.SubRefund(15000) evm.StateDB.SubRefund(15000)
}else{ } else {
// 2.2.1.2 // 2.2.1.2
evm.StateDB.AddRefund(15000) evm.StateDB.AddRefund(15000)
} }
} }
if original == new { // 2.2.2 if original == new { // 2.2.2
if original == (common.Hash{}){ if original == (common.Hash{}) {
evm.StateDB.AddRefund(19800) evm.StateDB.AddRefund(19800)
}else{ } else {
evm.StateDB.AddRefund(4800) evm.StateDB.AddRefund(4800)
} }
} }

View File

@ -44,6 +44,7 @@ type StateDB interface {
GetRefund() uint64 GetRefund() uint64
GetState(common.Address, common.Hash) common.Hash GetState(common.Address, common.Hash) common.Hash
GetStateOriginal(common.Address, common.Hash) common.Hash
SetState(common.Address, common.Hash, common.Hash) SetState(common.Address, common.Hash, common.Hash)
Suicide(common.Address) bool Suicide(common.Address) bool

View File

@ -95,6 +95,14 @@ func newConstantinopleInstructionSet() [256]operation {
writes: true, writes: true,
returns: true, returns: true,
} }
instructionSet[SSTORE] = operation{
execute: opSstore,
gasCost: gasSStoreEip1283,
validateStack: makeStackFunc(2, 0),
valid: true,
writes: true,
}
return instructionSet return instructionSet
} }

View File

@ -59,6 +59,7 @@ func (NoopStateDB) AddRefund(uint64)
func (NoopStateDB) SubRefund(uint64) {} func (NoopStateDB) SubRefund(uint64) {}
func (NoopStateDB) GetRefund() uint64 { return 0 } func (NoopStateDB) GetRefund() uint64 { return 0 }
func (NoopStateDB) GetState(common.Address, common.Hash) common.Hash { return common.Hash{} } func (NoopStateDB) GetState(common.Address, common.Hash) common.Hash { return common.Hash{} }
func (NoopStateDB) GetStateOriginal(common.Address, common.Hash) common.Hash { return common.Hash{} }
func (NoopStateDB) SetState(common.Address, common.Hash, common.Hash) {} func (NoopStateDB) SetState(common.Address, common.Hash, common.Hash) {}
func (NoopStateDB) Suicide(common.Address) bool { return false } func (NoopStateDB) Suicide(common.Address) bool { return false }
func (NoopStateDB) HasSuicided(common.Address) bool { return false } func (NoopStateDB) HasSuicided(common.Address) bool { return false }