メインコンテンツへスキップ
mip.watch
⌘K

MIP-4

Reserve Balance Introspection

確定 標準トラック コア MONAD_NINEにバンドル済み GitHub ↗ フォーラム ↗
アイデア ドラフト レビュー中 最終確認 確定 継続中

Add reserve balance precompile to query reserve balance violation state during transaction execution

作成者
Category Labs
作成日
2026-01-08
更新日
2026/06/15
フォーラム
8p · 11♥

ステークホルダー影響

このMIPが各対象者にとって何を変えるのかを示します。重大度とアクションフラグはモデルによって導出され、MIPのタイプ/カテゴリごとに決定論的な下限値が適用されます。

AIサマリーは利用できません — 構造的なデフォルト値で代替

要約

このMIPのAIインサイトは利用できません — 以下に構造的なデフォルト値を表示します。

何が変わるか

  • このMIPのAIインサイトは利用できません — 以下に構造的なデフォルト値を表示します。

開発者

スマートコントラクトおよびdapp開発者、RPC利用者、ツール作成者

ユーザー

ウォレットユーザー、EOA保有者、dapp訪問者

バリデーター

ノード運用者、RPC運用者、委任者

財団

Monad財団およびCategory Labsのコア開発者

仕様

引用
サブシステム executionevmstakingdb

## Abstract

Add a new precompile at address `0x1001` with a method `dippedIntoReserve` that returns whether the current execution state is in reserve balance violation.
This enables contracts to detect and recover from temporary reserve violations before transaction completion.

## Motivation

Monad's reserve balance mechanism (per the initial spec, Algorithm 3) reverts transactions that leave any touched account below its reserve threshold at execution end.
However, the check is performed post-execution: contracts have no way to know during execution whether they are in a violation state.

The reserve balance precompile allows contracts to query violation state mid-execution and adjust behavior accordingly—either by restoring balances, taking an alternative code path, or reverting early with a meaningful error.

## Specification

### Constants

| Name                           | Value        |
| ------------------------------ | ------------ |
| `GAS_DIPPED_INTO_RESERVE`      | `100`        |
| `SELECTOR_DIPPED_INTO_RESERVE` | `0x3a61584e` |

### Precompile

The new precompile has address `0x1001`, following the existing staking precompile at `0x1000`, and satisfies the following Solidity interface:

```solidity
interface IReserveBalance {
    function dippedIntoReserve() external returns (bool);
}
```

The Solidity selector for `dippedIntoReserve` is `SELECTOR_DIPPED_INTO_RESERVE (0x3a61584e)`.

### Gas Cost

Calling `dippedIntoReserve()` has a gas cost of `GAS_DIPPED_INTO_RESERVE (100)`, equivalent to the cost of one `tload`. 
Implementations should incrementally update their state to track violations of the reserve balance constraints, rather than iterating over the set of modified accounts in the current transaction on each call to the precompile.
Then `dippedIntoReserve()` should be a check of this violation state and therefore should have similar resource usage to a load from transient storage.

This was not benchmarked, but aligns with benchmarking and costing for the staking precompile's gas costs, which were calculated from the number of database loads and stores, events, and transfers.
The reserve balance check produces no events or transfers and is analagous to a load from transient storage instead of database storage and thus uses the cost for a `tload`.

### Semantics

The method `dippedIntoReserve()` evaluates the condition that `DippedIntoReserve` (Algorithm 3 of the initial spec) would return, substituting the current state for the post-execution state.
The check considers all accounts touched in the transaction, regardless of call depth.

The return value is ABI-encoded as a Solidity `bool`—i.e., a 32-byte word in returndata, consistent with standard Solidity ABI encoding.
This means callers can invoke the precompile via a normal contract call and decode the result using standard ABI decoding.

Calldata must consist of exactly the 4-byte `SELECTOR_DIPPED_INTO_RESERVE`.
Any other calldata is invalid: if the selector does not match `dippedIntoReserve()` (i.e. the selector is unrecognized) the contract must revert with the error message "method not supported".
If extra calldata is appended beyond the selector, the precompile must revert with the error message "input is invalid".

The method `dippedIntoReserve()` is not payable and must revert with the error message "value is nonzero" when called with a nonzero value.

The precompile must be invoked via `CALL`.
Invocations via `STATICCALL`, `DELEGATECALL`, or `CALLCODE` must revert.
Invocations via [EIP-7702](https://eips.ethereum.org/EIPS/eip-7702) delegations targeting the precompile address must revert.

Reverts consume all gas provided to the call frame.
This is consistent with the behavior of Ethereum precompiles, which do not return remaining gas on failure, as opposed to Solidity functions which return unused gas on revert.

In case more than one revert condition is present, the revert error message must correspond with the conditions evaluated in the following order:

1. is not invoked via `CALL`
2. `gas < GAS_DIPPED_INTO_RESERVE`
3. `len(calldata) < 4`
4. `calldata[:4] != SELECTOR_DIPPED_INTO_RESERVE`
5. `value > 0`
6. `len(calldata) > 4`

## Rationale

A previous design proposed adding a new opcode with similar semantics.
Since this introspection feature is intended for direct use by smart contract developers (e.g., in bundler entrypoint contracts), a precompile was chosen because it can be called immediately without requiring compiler or toolchain updates.

The semantics described above (strict calldata validation, ABI-encoded return values, rejecting calls with value, conventions around `*CALL` opcodes & EIP-7702, revert messages, and all-gas-consuming reverts) are chosen for explicit consistency with the existing Monad staking precompile.

## Backwards Compatibility

This proposal adds a new precompile and does not modify existing behavior.
Contracts that do not use the precompile are unaffected.

## Security Considerations

The precompile is read-only and exposes information that is already implicitly available (the transaction will revert if violation persists).
No new attack surface is introduced.

## References

- [Monad Initial Specification](https://category-labs.github.io/category-research/monad-initial-spec-proposal.pdf)

## Copyright

Copyright and related rights waived via [CC0](../LICENSE.md).

フォーラムでの議論

8件の投稿 · 11件のいいね · 5 か月前
フォーラムで続きを読む ↗
  1. @Bruce Collie #1 2026/01/08 12:53

    github.com/monad-crypto/MIPs MIP-4: Reserve Balance Introspection main ← bruce/mip-4 opened 12:45PM - 08 Jan 26 UTC Baltoli +46 -0 This MIP adds a new opcode that allows contracts on Monad to detect whether their local execution state is in violation of reserve balance and take action to avoid reverting.

  2. @Pdobacz #4 2026/01/15 13:10

    Re gas cost Expected to be `O(N)` in the number of warm accounts (i.e., accounts with modified balances). Would it be feasible to: - Keep a cache of reserve balance check results per warm account address (with a count of failed checks for quick lookup) - Run the reserve balance check only on crossing boundaries where it actually might change. I think that is at transaction start (after auth processing) CALL with value!=0 and CREATE and SELFDESTRUCT and call frame reverting. - Run the check and update cache only for affected addresses (1 or 2 max per such op) - Have CHECKRESERVEBALANCE just return the cached result Then it would be possible to price CHECKRESERVEBALANCE at O(1) (and as cheap as an environment opcode)? The cost of checking and updating would be assumed to be included in the (expensive) ops from (2.). Cost of memory to hold the extra execution state included in co...

  3. @Bruce Collie #5 2026/01/19 11:02

    I think that should be feasible to implement. We’ll try that method once we have our tests nailed down using the eager method as a reference.

  4. @ARitz Cracker #6 2026/01/23 23:17

    Could the MIP link to the algorithm referenced? We will need to update our revm-based codebase to include this opcode.

  5. @Bruce Collie #7 2026/01/26 15:13

    Yes, we’ll update the MIP to point to the algorithm in the current version of the Monad client - roughly speaking, the opcode will represent a call to dipped_into_reserve here .