Skip to content

Commit

Permalink
EIP-7709 Implement gas costs for BLOCKHASH implementation with system…
Browse files Browse the repository at this point in the history
… contract

Signed-off-by: Luis Pinto <[email protected]>
  • Loading branch information
lu-pinto committed Dec 10, 2024
1 parent 836137a commit 702cb81
Show file tree
Hide file tree
Showing 6 changed files with 147 additions and 22 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@
import org.hyperledger.besu.ethereum.mainnet.ClearEmptyAccountStrategy.NotClearEmptyAccount;
import org.hyperledger.besu.ethereum.mainnet.ProtocolSpecBuilder.BlockValidatorBuilder;
import org.hyperledger.besu.ethereum.mainnet.blockhash.CancunBlockHashProcessor;
import org.hyperledger.besu.ethereum.mainnet.blockhash.Eip7709BlockHashProcessor;
import org.hyperledger.besu.ethereum.mainnet.blockhash.FrontierBlockHashProcessor;
import org.hyperledger.besu.ethereum.mainnet.blockhash.PragueBlockHashProcessor;
import org.hyperledger.besu.ethereum.mainnet.feemarket.BaseFeeMarket;
Expand Down Expand Up @@ -905,7 +906,7 @@ static ProtocolSpecBuilder verkleDefinition(
CoinbaseFeePriceCalculator.eip1559()))
.withdrawalsProcessor(new WithdrawalsProcessor(clearEmptyAccountStrategy))
.executionWitnessValidator(new ExecutionWitnessValidator.AllowedExecutionWitness())
.blockHashProcessor(new PragueBlockHashProcessor())
.blockHashProcessor(new Eip7709BlockHashProcessor())
.blockHeaderFunctions(new VerkleDevnetBlockHeaderFunctions())
.name("Verkle");
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@
public class ContractBasedBlockHashLookup implements BlockHashLookup {
private static final Logger LOG = LoggerFactory.getLogger(ContractBasedBlockHashLookup.class);
private static final long BLOCKHASH_SERVE_WINDOW = 256L;
private static final long WARM_STORAGE_READ_COST = 100L;

private final ProcessableBlockHeader blockHeader;
private final Address contractAddress;
Expand Down Expand Up @@ -90,6 +91,13 @@ public Hash apply(final MessageFrame frame, final Long blockNumber) {
return ZERO;
}

final UInt256 slot = UInt256.valueOf(blockNumber % historyServeWindow);
final long cost = cost(frame, slot);
if (frame.getRemainingGas() < cost) {
return Hash.EMPTY;
}
frame.decrementRemainingGas(cost);

final Hash cachedHash = hashByNumber.get(blockNumber);
if (cachedHash != null) {
return cachedHash;
Expand All @@ -102,7 +110,6 @@ public Hash apply(final MessageFrame frame, final Long blockNumber) {
return ZERO;
}

UInt256 slot = UInt256.valueOf(blockNumber % historyServeWindow);
final UInt256 value = account.getStorageValue(slot);
LOG.atTrace()
.log(
Expand All @@ -114,4 +121,18 @@ public Hash apply(final MessageFrame frame, final Long blockNumber) {
hashByNumber.put(blockNumber, blockHash);
return blockHash;
}

protected long cost(final MessageFrame frame, final UInt256 slot) {
long gas = frame.getAccessWitness().touchAndChargeStorageLoad(contractAddress, slot);

if (gas == 0) {
return getWarmStorageReadCost();
}

return gas;
}

protected long getWarmStorageReadCost() {
return WARM_STORAGE_READ_COST;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@

import static org.hyperledger.besu.evm.frame.MessageFrame.DEFAULT_MAX_STACK_SIZE;

import org.hyperledger.besu.datatypes.AccessWitness;
import org.hyperledger.besu.datatypes.Address;
import org.hyperledger.besu.datatypes.Wei;
import org.hyperledger.besu.ethereum.chain.Blockchain;
Expand Down Expand Up @@ -56,6 +57,7 @@ public class MessageFrameTestFixture {
private Optional<BlockHeader> blockHeader = Optional.empty();
private Optional<BlockHashLookup> blockHashLookup = Optional.empty();
private ExecutionContextTestFixture executionContextTestFixture;
private AccessWitness accessWitness;

public MessageFrameTestFixture parentFrame(final MessageFrame parentFrame) {
this.parentFrame = parentFrame;
Expand Down Expand Up @@ -153,6 +155,11 @@ public MessageFrameTestFixture blockHashLookup(final BlockHashLookup blockHashLo
return this;
}

public MessageFrameTestFixture accessWitness(final AccessWitness accessWitness) {
this.accessWitness = accessWitness;
return this;
}

public MessageFrame build() {
final Blockchain localBlockchain = this.blockchain.orElseGet(this::createDefaultBlockchain);
final BlockHeader localBlockHeader =
Expand Down Expand Up @@ -185,6 +192,7 @@ public MessageFrame build() {
.getBlockHashProcessor()
.createBlockHashLookup(localBlockchain, localBlockHeader)))
.maxStackSize(maxStackSize)
.accessWitness(accessWitness)
.build();
stackItems.forEach(frame::pushStackItem);
return frame;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
package org.hyperledger.besu.ethereum.vm;

import static org.assertj.core.api.Assertions.assertThat;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.eq;
import static org.mockito.Mockito.clearInvocations;
import static org.mockito.Mockito.mock;
Expand All @@ -24,16 +25,19 @@
import static org.mockito.Mockito.verifyNoMoreInteractions;
import static org.mockito.Mockito.when;

import org.hyperledger.besu.datatypes.AccessWitness;
import org.hyperledger.besu.datatypes.Address;
import org.hyperledger.besu.datatypes.Hash;
import org.hyperledger.besu.datatypes.Wei;
import org.hyperledger.besu.ethereum.core.BlockHeader;
import org.hyperledger.besu.ethereum.core.BlockHeaderTestFixture;
import org.hyperledger.besu.ethereum.core.MessageFrameTestFixture;
import org.hyperledger.besu.evm.account.Account;
import org.hyperledger.besu.evm.blockhash.BlockHashLookup;
import org.hyperledger.besu.evm.fluent.SimpleAccount;
import org.hyperledger.besu.evm.fluent.SimpleWorld;
import org.hyperledger.besu.evm.frame.MessageFrame;
import org.hyperledger.besu.evm.gascalculator.stateless.NoopAccessWitness;
import org.hyperledger.besu.evm.worldstate.WorldUpdater;

import java.util.ArrayList;
Expand All @@ -53,13 +57,18 @@ public class ContractBasedBlockHashLookupTest {
private List<BlockHeader> headers;
private BlockHashLookup lookup;
private MessageFrame frame;
private WorldUpdater worldUpdater;

@BeforeEach
void setUp() {
headers = new ArrayList<>();
frame = mock(MessageFrame.class);
final WorldUpdater worldUpdater = createWorldUpdater(0, CURRENT_BLOCK_NUMBER);
when(frame.getWorldUpdater()).thenReturn(worldUpdater);
worldUpdater = createWorldUpdater(0, CURRENT_BLOCK_NUMBER);
frame =
new MessageFrameTestFixture()
.worldUpdater(worldUpdater)
.initialGas(Long.MAX_VALUE)
.accessWitness(NoopAccessWitness.get())
.build();
lookup =
new ContractBasedBlockHashLookup(
createHeader(CURRENT_BLOCK_NUMBER, headers.getLast()),
Expand Down Expand Up @@ -94,22 +103,58 @@ void shouldGetHashOfMaxBlocksBehind() {
}

@Test
void shouldReturnEmptyHashWhenRequestedBlockHigherThanHead() {
void shouldReturnZeroHashWhenRequestedBlockHigherThanHead() {
assertThat(lookup.apply(frame, CURRENT_BLOCK_NUMBER + 20L)).isEqualTo(Hash.ZERO);
}

@Test
void shouldReturnEmptyHashWhenSystemContractNotExists() {
final WorldUpdater worldUpdater = new SimpleWorld();
when(frame.getWorldUpdater()).thenReturn(worldUpdater);
void shouldReturnZeroHashWhenSystemContractNotExists() {
worldUpdater = new SimpleWorld();
frame =
new MessageFrameTestFixture()
.worldUpdater(worldUpdater)
.accessWitness(NoopAccessWitness.get())
.build();
assertThat(lookup.apply(frame, CURRENT_BLOCK_NUMBER - 1L)).isEqualTo(Hash.ZERO);
}

@Test
void shouldReturnEmptyHashWhenParentBlockNotInContract() {
final WorldUpdater worldUpdater =
createWorldUpdater(CURRENT_BLOCK_NUMBER - 10, CURRENT_BLOCK_NUMBER);
when(frame.getWorldUpdater()).thenReturn(worldUpdater);
@SuppressWarnings("ReturnValueIgnored")
void shouldDecrementRemainingGasFromFrame() {
AccessWitness accessWitness = mock(AccessWitness.class);
when(accessWitness.touchAndChargeStorageLoad(any(), any())).thenReturn(100L);
frame =
new MessageFrameTestFixture()
.worldUpdater(worldUpdater)
.accessWitness(accessWitness)
.initialGas(200L)
.build();
lookup.apply(frame, CURRENT_BLOCK_NUMBER - 1L);
assertThat(frame.getRemainingGas()).isEqualTo(100L);
}

@Test
void insufficientGasReturnsEmptyHash() {
worldUpdater = new SimpleWorld();
AccessWitness accessWitness = mock(AccessWitness.class);
when(accessWitness.touchAndChargeStorageLoad(any(), any())).thenReturn(100L);
frame =
new MessageFrameTestFixture()
.worldUpdater(worldUpdater)
.accessWitness(accessWitness)
.initialGas(1)
.build();
assertThat(lookup.apply(frame, CURRENT_BLOCK_NUMBER - 1L)).isEqualTo(Hash.EMPTY);
}

@Test
void shouldReturnZeroHashWhenParentBlockNotInContract() {
worldUpdater = createWorldUpdater(CURRENT_BLOCK_NUMBER - 10, CURRENT_BLOCK_NUMBER);
frame =
new MessageFrameTestFixture()
.worldUpdater(worldUpdater)
.accessWitness(NoopAccessWitness.get())
.build();
lookup =
new ContractBasedBlockHashLookup(
new BlockHeaderTestFixture().number(CURRENT_BLOCK_NUMBER).buildHeader(),
Expand All @@ -121,8 +166,12 @@ void shouldReturnEmptyHashWhenParentBlockNotInContract() {

@Test
void shouldCacheBlockHashes() {
final WorldUpdater worldUpdater = createWorldUpdater(0, CURRENT_BLOCK_NUMBER);
when(frame.getWorldUpdater()).thenReturn(worldUpdater);
worldUpdater = createWorldUpdater(0, CURRENT_BLOCK_NUMBER);
frame =
new MessageFrameTestFixture()
.worldUpdater(worldUpdater)
.accessWitness(NoopAccessWitness.get())
.build();
final Account account = worldUpdater.get(STORAGE_ADDRESS);
clearInvocations(account);

Expand All @@ -147,8 +196,12 @@ void shouldCacheBlockHashes() {

@Test
void shouldGetHashWhenParentIsGenesis() {
final WorldUpdater worldUpdater = createWorldUpdater(0, 1);
when(frame.getWorldUpdater()).thenReturn(worldUpdater);
worldUpdater = createWorldUpdater(0, 1);
frame =
new MessageFrameTestFixture()
.worldUpdater(worldUpdater)
.accessWitness(NoopAccessWitness.get())
.build();
lookup =
new ContractBasedBlockHashLookup(
createHeader(1, headers.getFirst()),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -50,11 +50,18 @@ public OperationResult execute(final MessageFrame frame, final EVM evm) {
return new OperationResult(cost, null);
}

final long remainingGas = frame.getRemainingGas();
final BlockHashLookup blockHashLookup = frame.getBlockHashLookup();
final Hash blockHash = blockHashLookup.apply(frame, blockArg.toLong());
final long lookupCost = remainingGas - frame.getRemainingGas();
if (Hash.EMPTY.equals(blockHash)) {
return new OperationResult(cost + lookupCost, ExceptionalHaltReason.INSUFFICIENT_GAS);
}
// give lookupCost back as it will be taken after
frame.incrementRemainingGas(lookupCost);
frame.pushStackItem(blockHash);

return new OperationResult(cost, null);
return new OperationResult(cost + lookupCost, null);
}

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ void shouldReturnZeroWhenArgIsBiggerThanALong() {
}

@Test
void shouldReturnBlockHashUsingLookupFromFrameWhenItIsWithinTheAllowedRange() {
void returnsBlockHashWhenWithinAllowedRange() {
final Hash blockHash = Hash.hash(Bytes.fromHexString("0x1293487297"));
assertBlockHash(
100,
Expand All @@ -56,14 +56,46 @@ void shouldReturnBlockHashUsingLookupFromFrameWhenItIsWithinTheAllowedRange() {
ENOUGH_GAS);
}

@Test
void remainingGasDoesNotChange() {
final Hash blockHash = Hash.hash(Bytes.fromHexString("0x1293487297"));
assertBlockHash(
100,
blockHash,
200,
(frame, block) -> {
frame.decrementRemainingGas(100L);
if (block == 100) {
return blockHash;
}
return Hash.ZERO;
},
ENOUGH_GAS);
}

@Test
void shouldFailWithInsufficientGas() {
assertFailure(
Bytes32.fromHexString("0x64"),
ExceptionalHaltReason.INSUFFICIENT_GAS,
200,
(__, ___) -> Hash.hash(Bytes.fromHexString("0x1293487297")),
1);
1L,
20L);
}

@Test
void shouldFailWithInsufficientGasWithinLookup() {
assertFailure(
Bytes32.fromHexString("0x64"),
ExceptionalHaltReason.INSUFFICIENT_GAS,
200,
(frame, ___) -> {
frame.decrementRemainingGas(100);
return Hash.EMPTY;
},
100L,
120L);
}

private void assertBlockHash(
Expand Down Expand Up @@ -93,18 +125,21 @@ private void assertBlockHash(
.pushStackItem(UInt256.fromBytes(input))
.initialGas(initialGas)
.build();
long remainingGasBefore = frame.getRemainingGas();
blockHashOperation.execute(frame, null);
final Bytes result = frame.popStackItem();
assertThat(result).isEqualTo(expectedOutput);
assertThat(frame.stackSize()).isZero();
assertThat(frame.getRemainingGas()).isEqualTo(remainingGasBefore);
}

private void assertFailure(
final Bytes32 input,
final ExceptionalHaltReason haltReason,
final long currentBlockNumber,
final BlockHashLookup blockHashLookup,
final long initialGas) {
final long initialGas,
final long gasCost) {
final MessageFrame frame =
new TestMessageFrameBuilder()
.blockHashLookup(blockHashLookup)
Expand All @@ -114,6 +149,6 @@ private void assertFailure(
.build();
Operation.OperationResult operationResult = blockHashOperation.execute(frame, null);
assertThat(operationResult.getHaltReason()).isEqualTo(haltReason);
assertThat(frame.stackSize()).isOne();
assertThat(operationResult.gasCost).isEqualTo(gasCost);
}
}

0 comments on commit 702cb81

Please sign in to comment.