core/vm: rework reversion to work on a higher level
This commit is contained in:
		
							parent
							
								
									b70a73cd3e
								
							
						
					
					
						commit
						f9fb70d2ee
					
				| @ -168,8 +168,10 @@ func (evm *EVM) Call(caller ContractRef, addr common.Address, input []byte, gas | |||||||
| 	// above we revert to the snapshot and consume any gas remaining. Additionally
 | 	// above we revert to the snapshot and consume any gas remaining. Additionally
 | ||||||
| 	// when we're in homestead this also counts for code storage gas errors.
 | 	// when we're in homestead this also counts for code storage gas errors.
 | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		contract.UseGas(contract.Gas) |  | ||||||
| 		evm.StateDB.RevertToSnapshot(snapshot) | 		evm.StateDB.RevertToSnapshot(snapshot) | ||||||
|  | 		if err != errExecutionReverted { | ||||||
|  | 			contract.UseGas(contract.Gas) | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	return ret, contract.Gas, err | 	return ret, contract.Gas, err | ||||||
| } | } | ||||||
| @ -207,10 +209,11 @@ func (evm *EVM) CallCode(caller ContractRef, addr common.Address, input []byte, | |||||||
| 
 | 
 | ||||||
| 	ret, err = run(evm, snapshot, contract, input) | 	ret, err = run(evm, snapshot, contract, input) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		contract.UseGas(contract.Gas) |  | ||||||
| 		evm.StateDB.RevertToSnapshot(snapshot) | 		evm.StateDB.RevertToSnapshot(snapshot) | ||||||
|  | 		if err != errExecutionReverted { | ||||||
|  | 			contract.UseGas(contract.Gas) | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	return ret, contract.Gas, err | 	return ret, contract.Gas, err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -239,10 +242,11 @@ func (evm *EVM) DelegateCall(caller ContractRef, addr common.Address, input []by | |||||||
| 
 | 
 | ||||||
| 	ret, err = run(evm, snapshot, contract, input) | 	ret, err = run(evm, snapshot, contract, input) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		contract.UseGas(contract.Gas) |  | ||||||
| 		evm.StateDB.RevertToSnapshot(snapshot) | 		evm.StateDB.RevertToSnapshot(snapshot) | ||||||
|  | 		if err != errExecutionReverted { | ||||||
|  | 			contract.UseGas(contract.Gas) | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	return ret, contract.Gas, err | 	return ret, contract.Gas, err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -281,8 +285,10 @@ func (evm *EVM) StaticCall(caller ContractRef, addr common.Address, input []byte | |||||||
| 	// when we're in Homestead this also counts for code storage gas errors.
 | 	// when we're in Homestead this also counts for code storage gas errors.
 | ||||||
| 	ret, err = run(evm, snapshot, contract, input) | 	ret, err = run(evm, snapshot, contract, input) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		contract.UseGas(contract.Gas) |  | ||||||
| 		evm.StateDB.RevertToSnapshot(snapshot) | 		evm.StateDB.RevertToSnapshot(snapshot) | ||||||
|  | 		if err != errExecutionReverted { | ||||||
|  | 			contract.UseGas(contract.Gas) | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	return ret, contract.Gas, err | 	return ret, contract.Gas, err | ||||||
| } | } | ||||||
| @ -339,18 +345,12 @@ func (evm *EVM) Create(caller ContractRef, code []byte, gas uint64, value *big.I | |||||||
| 	// When an error was returned by the EVM or when setting the creation code
 | 	// When an error was returned by the EVM or when setting the creation code
 | ||||||
| 	// above we revert to the snapshot and consume any gas remaining. Additionally
 | 	// above we revert to the snapshot and consume any gas remaining. Additionally
 | ||||||
| 	// when we're in homestead this also counts for code storage gas errors.
 | 	// when we're in homestead this also counts for code storage gas errors.
 | ||||||
| 	if maxCodeSizeExceeded || | 	if maxCodeSizeExceeded || (err != nil && (evm.ChainConfig().IsHomestead(evm.BlockNumber) || err != ErrCodeStoreOutOfGas)) { | ||||||
| 		(err != nil && (evm.ChainConfig().IsHomestead(evm.BlockNumber) || err != ErrCodeStoreOutOfGas)) { |  | ||||||
| 		contract.UseGas(contract.Gas) |  | ||||||
| 		evm.StateDB.RevertToSnapshot(snapshot) | 		evm.StateDB.RevertToSnapshot(snapshot) | ||||||
|  | 		if err != errExecutionReverted { | ||||||
|  | 			contract.UseGas(contract.Gas) | ||||||
| 		} | 		} | ||||||
| 	// If the vm returned with an error the return value should be set to nil.
 |  | ||||||
| 	// This isn't consensus critical but merely to for behaviour reasons such as
 |  | ||||||
| 	// tests, RPC calls, etc.
 |  | ||||||
| 	if err != nil { |  | ||||||
| 		ret = nil |  | ||||||
| 	} | 	} | ||||||
| 
 |  | ||||||
| 	return ret, contractAddr, contract.Gas, err | 	return ret, contractAddr, contract.Gas, err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -396,6 +396,10 @@ func gasReturn(gt params.GasTable, evm *EVM, contract *Contract, stack *Stack, m | |||||||
| 	return memoryGasCost(mem, memorySize) | 	return memoryGasCost(mem, memorySize) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func gasRevert(gt params.GasTable, evm *EVM, contract *Contract, stack *Stack, mem *Memory, memorySize uint64) (uint64, error) { | ||||||
|  | 	return memoryGasCost(mem, memorySize) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func gasSuicide(gt params.GasTable, evm *EVM, contract *Contract, stack *Stack, mem *Memory, memorySize uint64) (uint64, error) { | func gasSuicide(gt params.GasTable, evm *EVM, contract *Contract, stack *Stack, mem *Memory, memorySize uint64) (uint64, error) { | ||||||
| 	var gas uint64 | 	var gas uint64 | ||||||
| 	// EIP150 homestead gas reprice fork:
 | 	// EIP150 homestead gas reprice fork:
 | ||||||
|  | |||||||
| @ -32,6 +32,7 @@ var ( | |||||||
| 	bigZero                  = new(big.Int) | 	bigZero                  = new(big.Int) | ||||||
| 	errWriteProtection       = errors.New("evm: write protection") | 	errWriteProtection       = errors.New("evm: write protection") | ||||||
| 	errReturnDataOutOfBounds = errors.New("evm: return data out of bounds") | 	errReturnDataOutOfBounds = errors.New("evm: return data out of bounds") | ||||||
|  | 	errExecutionReverted     = errors.New("evm: execution reverted") | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func opAdd(pc *uint64, evm *EVM, contract *Contract, memory *Memory, stack *Stack) ([]byte, error) { | func opAdd(pc *uint64, evm *EVM, contract *Contract, memory *Memory, stack *Stack) ([]byte, error) { | ||||||
| @ -579,7 +580,7 @@ func opCreate(pc *uint64, evm *EVM, contract *Contract, memory *Memory, stack *S | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	contract.UseGas(gas) | 	contract.UseGas(gas) | ||||||
| 	_, addr, returnGas, suberr := evm.Create(contract, input, gas, value) | 	res, addr, returnGas, suberr := evm.Create(contract, input, gas, value) | ||||||
| 	// Push item on the stack based on the returned error. If the ruleset is
 | 	// Push item on the stack based on the returned error. If the ruleset is
 | ||||||
| 	// homestead we must check for CodeStoreOutOfGasError (homestead only
 | 	// homestead we must check for CodeStoreOutOfGasError (homestead only
 | ||||||
| 	// rule) and treat as an error, if the ruleset is frontier we must
 | 	// rule) and treat as an error, if the ruleset is frontier we must
 | ||||||
| @ -592,9 +593,11 @@ func opCreate(pc *uint64, evm *EVM, contract *Contract, memory *Memory, stack *S | |||||||
| 		stack.push(addr.Big()) | 		stack.push(addr.Big()) | ||||||
| 	} | 	} | ||||||
| 	contract.Gas += returnGas | 	contract.Gas += returnGas | ||||||
| 
 |  | ||||||
| 	evm.interpreter.intPool.put(value, offset, size) | 	evm.interpreter.intPool.put(value, offset, size) | ||||||
| 
 | 
 | ||||||
|  | 	if suberr == errExecutionReverted { | ||||||
|  | 		return res, nil | ||||||
|  | 	} | ||||||
| 	return nil, nil | 	return nil, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -622,7 +625,8 @@ func opCall(pc *uint64, evm *EVM, contract *Contract, memory *Memory, stack *Sta | |||||||
| 		stack.push(new(big.Int)) | 		stack.push(new(big.Int)) | ||||||
| 	} else { | 	} else { | ||||||
| 		stack.push(big.NewInt(1)) | 		stack.push(big.NewInt(1)) | ||||||
| 
 | 	} | ||||||
|  | 	if err == nil || err == errExecutionReverted { | ||||||
| 		memory.Set(retOffset.Uint64(), retSize.Uint64(), ret) | 		memory.Set(retOffset.Uint64(), retSize.Uint64(), ret) | ||||||
| 	} | 	} | ||||||
| 	contract.Gas += returnGas | 	contract.Gas += returnGas | ||||||
| @ -653,10 +657,10 @@ func opCallCode(pc *uint64, evm *EVM, contract *Contract, memory *Memory, stack | |||||||
| 	ret, returnGas, err := evm.CallCode(contract, address, args, gas, value) | 	ret, returnGas, err := evm.CallCode(contract, address, args, gas, value) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		stack.push(new(big.Int)) | 		stack.push(new(big.Int)) | ||||||
| 
 |  | ||||||
| 	} else { | 	} else { | ||||||
| 		stack.push(big.NewInt(1)) | 		stack.push(big.NewInt(1)) | ||||||
| 
 | 	} | ||||||
|  | 	if err == nil || err == errExecutionReverted { | ||||||
| 		memory.Set(retOffset.Uint64(), retSize.Uint64(), ret) | 		memory.Set(retOffset.Uint64(), retSize.Uint64(), ret) | ||||||
| 	} | 	} | ||||||
| 	contract.Gas += returnGas | 	contract.Gas += returnGas | ||||||
| @ -676,6 +680,8 @@ func opDelegateCall(pc *uint64, evm *EVM, contract *Contract, memory *Memory, st | |||||||
| 		stack.push(new(big.Int)) | 		stack.push(new(big.Int)) | ||||||
| 	} else { | 	} else { | ||||||
| 		stack.push(big.NewInt(1)) | 		stack.push(big.NewInt(1)) | ||||||
|  | 	} | ||||||
|  | 	if err == nil || err == errExecutionReverted { | ||||||
| 		memory.Set(outOffset.Uint64(), outSize.Uint64(), ret) | 		memory.Set(outOffset.Uint64(), outSize.Uint64(), ret) | ||||||
| 	} | 	} | ||||||
| 	contract.Gas += returnGas | 	contract.Gas += returnGas | ||||||
| @ -704,7 +710,8 @@ func opStaticCall(pc *uint64, evm *EVM, contract *Contract, memory *Memory, stac | |||||||
| 		stack.push(new(big.Int)) | 		stack.push(new(big.Int)) | ||||||
| 	} else { | 	} else { | ||||||
| 		stack.push(big.NewInt(1)) | 		stack.push(big.NewInt(1)) | ||||||
| 
 | 	} | ||||||
|  | 	if err == nil || err == errExecutionReverted { | ||||||
| 		memory.Set(retOffset.Uint64(), retSize.Uint64(), ret) | 		memory.Set(retOffset.Uint64(), retSize.Uint64(), ret) | ||||||
| 	} | 	} | ||||||
| 	contract.Gas += returnGas | 	contract.Gas += returnGas | ||||||
|  | |||||||
| @ -209,24 +209,22 @@ func (in *Interpreter) Run(snapshot int, contract *Contract, input []byte) (ret | |||||||
| 		if verifyPool { | 		if verifyPool { | ||||||
| 			verifyIntegerPool(in.intPool) | 			verifyIntegerPool(in.intPool) | ||||||
| 		} | 		} | ||||||
| 		// checks whether the operation should revert state.
 |  | ||||||
| 		if operation.reverts { |  | ||||||
| 			in.evm.StateDB.RevertToSnapshot(snapshot) |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		switch { |  | ||||||
| 		case err != nil: |  | ||||||
| 			return nil, err |  | ||||||
| 		case operation.halts: |  | ||||||
| 			return res, nil |  | ||||||
| 		case !operation.jumps: |  | ||||||
| 			pc++ |  | ||||||
| 		} |  | ||||||
| 		// if the operation clears the return data (e.g. it has returning data)
 | 		// if the operation clears the return data (e.g. it has returning data)
 | ||||||
| 		// set the last return to the result of the operation.
 | 		// set the last return to the result of the operation.
 | ||||||
| 		if operation.returns { | 		if operation.returns { | ||||||
| 			in.returnData = res | 			in.returnData = res | ||||||
| 		} | 		} | ||||||
|  | 
 | ||||||
|  | 		switch { | ||||||
|  | 		case err != nil: | ||||||
|  | 			return nil, err | ||||||
|  | 		case operation.reverts: | ||||||
|  | 			return res, errExecutionReverted | ||||||
|  | 		case operation.halts: | ||||||
|  | 			return res, nil | ||||||
|  | 		case !operation.jumps: | ||||||
|  | 			pc++ | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	return nil, nil | 	return nil, nil | ||||||
| } | } | ||||||
|  | |||||||
| @ -41,20 +41,13 @@ type operation struct { | |||||||
| 	validateStack stackValidationFunc | 	validateStack stackValidationFunc | ||||||
| 	// memorySize returns the memory size required for the operation
 | 	// memorySize returns the memory size required for the operation
 | ||||||
| 	memorySize memorySizeFunc | 	memorySize memorySizeFunc | ||||||
| 	// halts indicates whether the operation shoult halt further execution
 | 
 | ||||||
| 	// and return
 | 	halts   bool // indicates whether the operation shoult halt further execution
 | ||||||
| 	halts bool | 	jumps   bool // indicates whether the program counter should not increment
 | ||||||
| 	// jumps indicates whether operation made a jump. This prevents the program
 | 	writes  bool // determines whether this a state modifying operation
 | ||||||
| 	// counter from further incrementing.
 | 	valid   bool // indication whether the retrieved operation is valid and known
 | ||||||
| 	jumps bool | 	reverts bool // determines whether the operation reverts state (implicitly halts)
 | ||||||
| 	// writes determines whether this a state modifying operation
 | 	returns bool // determines whether the opertions sets the return data content
 | ||||||
| 	writes bool |  | ||||||
| 	// valid is used to check whether the retrieved operation is valid and known
 |  | ||||||
| 	valid bool |  | ||||||
| 	// reverts determined whether the operation reverts state
 |  | ||||||
| 	reverts bool |  | ||||||
| 	// returns determines whether the opertions sets the return data
 |  | ||||||
| 	returns bool |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| var ( | var ( | ||||||
| @ -91,10 +84,12 @@ func NewMetropolisInstructionSet() [256]operation { | |||||||
| 	} | 	} | ||||||
| 	instructionSet[REVERT] = operation{ | 	instructionSet[REVERT] = operation{ | ||||||
| 		execute:       opRevert, | 		execute:       opRevert, | ||||||
| 		gasCost:       constGasFunc(GasFastestStep), | 		gasCost:       gasRevert, | ||||||
| 		validateStack: makeStackFunc(2, 0), | 		validateStack: makeStackFunc(2, 0), | ||||||
|  | 		memorySize:    memoryRevert, | ||||||
| 		valid:         true, | 		valid:         true, | ||||||
| 		reverts:       true, | 		reverts:       true, | ||||||
|  | 		returns:       true, | ||||||
| 	} | 	} | ||||||
| 	return instructionSet | 	return instructionSet | ||||||
| } | } | ||||||
|  | |||||||
| @ -89,6 +89,10 @@ func memoryReturn(stack *Stack) *big.Int { | |||||||
| 	return calcMemSize(stack.Back(0), stack.Back(1)) | 	return calcMemSize(stack.Back(0), stack.Back(1)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func memoryRevert(stack *Stack) *big.Int { | ||||||
|  | 	return calcMemSize(stack.Back(0), stack.Back(1)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func memoryLog(stack *Stack) *big.Int { | func memoryLog(stack *Stack) *big.Int { | ||||||
| 	mSize, mStart := stack.Back(1), stack.Back(0) | 	mSize, mStart := stack.Back(1), stack.Back(0) | ||||||
| 	return calcMemSize(mStart, mSize) | 	return calcMemSize(mStart, mSize) | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user