Skip to content

fix(fault-proof): follow genesis-rooted catch-up chains to their tip#926

Closed
seolaoh wants to merge 1 commit into
succinctlabs:mainfrom
celo-org:seolaoh/proposer-canonical-head-catchup-upstream
Closed

fix(fault-proof): follow genesis-rooted catch-up chains to their tip#926
seolaoh wants to merge 1 commit into
succinctlabs:mainfrom
celo-org:seolaoh/proposer-canonical-head-catchup-upstream

Conversation

@seolaoh

@seolaoh seolaoh commented Jun 4, 2026

Copy link
Copy Markdown
Contributor

Summary

After a prolonged L2 finalized-head stall exceeding the bond claiming window, every tracked game can resolve & close and the latest one becomes the anchor, so the proposer prunes its cache down to just the anchor game. On L2 finalized-head stall recovery, it rebuilds the backlog as a fresh genesis-rooted chain — it uses parent = u32::MAX for these games because the contract reverts when a new game is initialized with the current anchor as its parent.

compute_canonical_head treated such an alternative chain's root as the head and never followed it to the tip: the override only considered games that branch off earlier than the anchor (genesis-rooted, or a lower parent index), which selects the chain's root but not its descendants. As a result the canonical head could not advance past the root, and game creation stalled until the anchor caught up — then the cycle repeated on the next chain.

Observed in Celo internal testnet: after a ~12h L2 finalized-head stall, the proposer repeatedly stalled while creating the backlog, interleaved with execution reverted (0x346119f7) on the attempts that tried to parent on the just-promoted anchor.

Logs
1780556787673	2026-06-04T07:06:27.673Z	2026-06-04T07:06:27.673197Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8033038 parent_game_index=4294967295}: Creating game l2_block_number=8033038 parent_game_index=4294967295 output_root=0x08bde30acca8b731c17e6913e29f6304600e53ae608a73ca253aa5cf27d7db14
1780556837875	2026-06-04T07:07:17.875Z	2026-06-04T07:07:17.874716Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8033038 parent_game_index=4294967295}: Game created successfully game_address=0x953a18ecc5c0de88d2bc48cb8204200a3443aaf1 tx_hash=0x10e9db66c8c86687c722d7eee07dc4fcdee543031c7127aefe6ffe49284d1de5
1780556877673	2026-06-04T07:07:57.673Z	2026-06-04T07:07:57.673232Z  INFO Valid game: adding to cache game_index=10141 game_type=42 game_address=0x953a18ecc5c0de88d2bc48cb8204200a3443aaf1 parent_index=4294967295 l2_block=8033038 status=IN_PROGRESS proposal_status=Unchallenged deadline=1780560396
1780556877695	2026-06-04T07:07:57.695Z	2026-06-04T07:07:57.695771Z  INFO Canonical head updated previous_canonical_index=Some(10140) new_canonical_index=10141 l2_block=8033038 total_games=2
1780556877713	2026-06-04T07:07:57.713Z	2026-06-04T07:07:57.713697Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8033938 parent_game_index=10141}: Creating game l2_block_number=8033938 parent_game_index=10141 output_root=0x2e88ec52a5588eebfe48a0788dfae3a23fde6b1bc164def30e252b34d0d1bee1
1780556919886	2026-06-04T07:08:39.886Z	2026-06-04T07:08:39.886745Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8033938 parent_game_index=10141}: Game created successfully game_address=0x1b37ec89bffa003dfe8ee10ae38a9239b77bb445 tx_hash=0x8942442bf4b237644074e0c38a39350975d9e730ab6dc769f664714b2a08e3f1
1780556967671	2026-06-04T07:09:27.671Z	2026-06-04T07:09:27.670592Z  INFO Valid game: adding to cache game_index=10142 game_type=42 game_address=0x1b37ec89bffa003dfe8ee10ae38a9239b77bb445 parent_index=10141 l2_block=8033938 status=IN_PROGRESS proposal_status=Unchallenged deadline=1780560480
1780560520766	2026-06-04T08:08:40.766Z	2026-06-04T08:08:40.766222Z  INFO Game resolved successfully game_index=10141 game_address=0x953a18ecc5c0de88d2bc48cb8204200a3443aaf1 l2_block_end=8033038 tx_hash=0x44e72b649efae17f650ce8901c077788080623b145dfb7bcf01088947caa9b4c
1780560537683	2026-06-04T08:08:57.683Z	2026-06-04T08:08:57.683210Z  INFO Skipping resolve: game already resolved on chain game_index=10141 game_address=0x953a18ecc5c0de88d2bc48cb8204200a3443aaf1 status=DEFENDER_WINS
1780560616863	2026-06-04T08:10:16.863Z	2026-06-04T08:10:16.862721Z  INFO Game resolved successfully game_index=10142 game_address=0x1b37ec89bffa003dfe8ee10ae38a9239b77bb445 l2_block_end=8033938 tx_hash=0xb5b966a3e83dd981ce8729903d035006b856d11c05b7f3eab622b9992f3d84ba
1780560627692	2026-06-04T08:10:27.692Z	2026-06-04T08:10:27.692779Z  INFO Skipping resolve: game already resolved on chain game_index=10142 game_address=0x1b37ec89bffa003dfe8ee10ae38a9239b77bb445 status=DEFENDER_WINS
1780560657697	2026-06-04T08:10:57.697Z	2026-06-04T08:10:57.697410Z  INFO Skipping resolve: game already resolved on chain game_index=10142 game_address=0x1b37ec89bffa003dfe8ee10ae38a9239b77bb445 status=DEFENDER_WINS
1780564267979	2026-06-04T09:11:07.979Z	2026-06-04T09:11:07.979649Z  INFO [[Claiming Proposer Bonds]]: Bond claimed successfully game_index=10141 game_address=0x953a18ecc5c0de88d2bc48cb8204200a3443aaf1 l2_block_end=8033038 tx_hash=0x55599318e727780d31b52eb234eeea9b8ba753e01893e92649878a5df0eb1d8a
1780564287690	2026-06-04T09:11:27.690Z	2026-06-04T09:11:27.690007Z  INFO Skipping claim: bond already claimed on chain game_index=10141 game_address=0x953a18ecc5c0de88d2bc48cb8204200a3443aaf1
1780564347677	2026-06-04T09:12:27.677Z	2026-06-04T09:12:27.676932Z  INFO Canonical head updated previous_canonical_index=Some(10141) new_canonical_index=10142 l2_block=8033938 total_games=2
1780564347698	2026-06-04T09:12:27.698Z	2026-06-04T09:12:27.698569Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8034838 parent_game_index=10142}: Creating game l2_block_number=8034838 parent_game_index=10142 output_root=0x6ed9cd3e7daec4f3e38f2ab393a7177dabe5ab0dbb1f7468db513c6162ec7836
1780564366862	2026-06-04T09:12:46.862Z	2026-06-04T09:12:46.862232Z  INFO [[Claiming Proposer Bonds]]: Bond claimed successfully game_index=10142 game_address=0x1b37ec89bffa003dfe8ee10ae38a9239b77bb445 l2_block_end=8033938 tx_hash=0x5a80a42af09c25e125e95d49b6ff0c6c09ca5b55c69cf74b97fef7df2e68c1ee
1780564366864	2026-06-04T09:12:46.864Z	2026-06-04T09:12:46.864592Z  INFO Skipping claim: bond already claimed on chain game_index=10141 game_address=0x953a18ecc5c0de88d2bc48cb8204200a3443aaf1
1780564367006	2026-06-04T09:12:47.006Z	2026-06-04T09:12:46.992600Z  WARN Failed to handle game creation: Failed to send KMS-signed transaction
1780564367006	2026-06-04T09:12:47.006Z	
1780564367006	2026-06-04T09:12:47.006Z	Caused by:
1780564367006	2026-06-04T09:12:47.006Z	    server returned an error response: error code 3: execution reverted, data: "0x346119f7"
1780564377670	2026-06-04T09:12:57.670Z	
1780564377670	2026-06-04T09:12:57.670Z	Caused by:
1780564377670	2026-06-04T09:12:57.670Z	    server returned an error response: error code 3: execution reverted, data: "0x346119f7"
1780564377694	2026-06-04T09:12:57.694Z	2026-06-04T09:12:57.693806Z  INFO Skipping claim: bond already claimed on chain game_index=10142 game_address=0x1b37ec89bffa003dfe8ee10ae38a9239b77bb445
1780564377694	2026-06-04T09:12:57.694Z	2026-06-04T09:12:57.694728Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8034838 parent_game_index=10142}: Creating game l2_block_number=8034838 parent_game_index=10142 output_root=0x6ed9cd3e7daec4f3e38f2ab393a7177dabe5ab0dbb1f7468db513c6162ec7836
1780564377781	2026-06-04T09:12:57.781Z	2026-06-04T09:12:57.781135Z  WARN Failed to handle game creation: Failed to send KMS-signed transaction
1780564377781	2026-06-04T09:12:57.781Z	
1780564377781	2026-06-04T09:12:57.781Z	Caused by:
1780564377781	2026-06-04T09:12:57.781Z	    server returned an error response: error code 3: execution reverted, data: "0x346119f7"
1780564407671	2026-06-04T09:13:27.671Z	
1780564407671	2026-06-04T09:13:27.671Z	Caused by:
1780564407671	2026-06-04T09:13:27.671Z	    server returned an error response: error code 3: execution reverted, data: "0x346119f7"
1780564407685	2026-06-04T09:13:27.685Z	2026-06-04T09:13:27.685527Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8034838 parent_game_index=4294967295}: Creating game l2_block_number=8034838 parent_game_index=4294967295 output_root=0x6ed9cd3e7daec4f3e38f2ab393a7177dabe5ab0dbb1f7468db513c6162ec7836
1780564456867	2026-06-04T09:14:16.867Z	2026-06-04T09:14:16.867079Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8034838 parent_game_index=4294967295}: Game created successfully game_address=0x989878700fd45083755ceeb834e8cdf5a8df623e tx_hash=0xe01464ce3a1e5e5b04417393c93bfbea3d6d0ddc033532442cac3c516dd35b8c
1780564497672	2026-06-04T09:14:57.672Z	2026-06-04T09:14:57.672170Z  INFO Valid game: adding to cache game_index=10143 game_type=42 game_address=0x989878700fd45083755ceeb834e8cdf5a8df623e parent_index=4294967295 l2_block=8034838 status=IN_PROGRESS proposal_status=Unchallenged deadline=1780568016
1780564497691	2026-06-04T09:14:57.691Z	2026-06-04T09:14:57.691632Z  INFO Canonical head updated previous_canonical_index=Some(10142) new_canonical_index=10143 l2_block=8034838 total_games=2
1780564497706	2026-06-04T09:14:57.706Z	2026-06-04T09:14:57.706641Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8035738 parent_game_index=10143}: Creating game l2_block_number=8035738 parent_game_index=10143 output_root=0xa1463603f9a262208407dbd696808c5456b83115e912001451c4873d49e327e9
1780564546916	2026-06-04T09:15:46.916Z	2026-06-04T09:15:46.915343Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8035738 parent_game_index=10143}: Game created successfully game_address=0x41668875d037b5e57277168681e4a669ba39a931 tx_hash=0x735d80949f86ac3be2791b134ad421d2cb44a058eab372d4895416225236c784
1780564617668	2026-06-04T09:16:57.668Z	2026-06-04T09:16:57.667854Z  INFO Valid game: adding to cache game_index=10144 game_type=42 game_address=0x41668875d037b5e57277168681e4a669ba39a931 parent_index=10143 l2_block=8035738 status=IN_PROGRESS proposal_status=Unchallenged deadline=1780568100
1780568176989	2026-06-04T10:16:16.989Z	2026-06-04T10:16:16.988655Z  INFO Game resolved successfully game_index=10143 game_address=0x989878700fd45083755ceeb834e8cdf5a8df623e l2_block_end=8034838 tx_hash=0xac564709b5d80b1a38bccdb4b7aa3af92caf4503d3a17942d9d28b1ffbbf3540
1780568187680	2026-06-04T10:16:27.680Z	2026-06-04T10:16:27.680863Z  INFO Skipping resolve: game already resolved on chain game_index=10143 game_address=0x989878700fd45083755ceeb834e8cdf5a8df623e status=DEFENDER_WINS
1780568273848	2026-06-04T10:17:53.848Z	2026-06-04T10:17:53.847817Z  INFO Game resolved successfully game_index=10144 game_address=0x41668875d037b5e57277168681e4a669ba39a931 l2_block_end=8035738 tx_hash=0x0215d04ceaa4ffee6fcc8a5740e7571b81a070a6a1c37562f43c09e8511ed134
1780568277687	2026-06-04T10:17:57.687Z	2026-06-04T10:17:57.687503Z  INFO Skipping resolve: game already resolved on chain game_index=10144 game_address=0x41668875d037b5e57277168681e4a669ba39a931 status=DEFENDER_WINS
1780568307685	2026-06-04T10:18:27.685Z	2026-06-04T10:18:27.685253Z  INFO Skipping resolve: game already resolved on chain game_index=10144 game_address=0x41668875d037b5e57277168681e4a669ba39a931 status=DEFENDER_WINS
1780571897016	2026-06-04T11:18:17.016Z	2026-06-04T11:18:17.016221Z  INFO [[Claiming Proposer Bonds]]: Bond claimed successfully game_index=10143 game_address=0x989878700fd45083755ceeb834e8cdf5a8df623e l2_block_end=8034838 tx_hash=0xd45845d1ca6b95b29cbb574f45480ef97aa2c9fd62a987a9168fa2b01de017c9
1780571907680	2026-06-04T11:18:27.680Z	2026-06-04T11:18:27.680759Z  INFO Skipping claim: bond already claimed on chain game_index=10143 game_address=0x989878700fd45083755ceeb834e8cdf5a8df623e
1780571937669	2026-06-04T11:18:57.669Z	2026-06-04T11:18:57.669619Z  INFO Canonical head updated previous_canonical_index=Some(10143) new_canonical_index=10144 l2_block=8035738 total_games=2
1780571937703	2026-06-04T11:18:57.703Z	2026-06-04T11:18:57.703408Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8036638 parent_game_index=10144}: Creating game l2_block_number=8036638 parent_game_index=10144 output_root=0x1e6bca5a6f19343d7a8011d7219ed95c5d52e275e98fb8a8333782669acf1ff1
1780571986858	2026-06-04T11:19:46.858Z	2026-06-04T11:19:46.858405Z  INFO [[Claiming Proposer Bonds]]: Bond claimed successfully game_index=10144 game_address=0x41668875d037b5e57277168681e4a669ba39a931 l2_block_end=8035738 tx_hash=0x403c936aed1586301a9e0100929e2404a0d1b9a09562e43fd9409b7dbd80be3d
1780571986959	2026-06-04T11:19:46.959Z	2026-06-04T11:19:46.959107Z  WARN Failed to handle game creation: Failed to send KMS-signed transaction
1780571986959	2026-06-04T11:19:46.959Z	
1780571986959	2026-06-04T11:19:46.959Z	Caused by:
1780571986959	2026-06-04T11:19:46.959Z	    server returned an error response: error code -32000: execution reverted
1780571997662	2026-06-04T11:19:57.662Z	
1780571997662	2026-06-04T11:19:57.662Z	Caused by:
1780571997662	2026-06-04T11:19:57.662Z	    server returned an error response: error code -32000: execution reverted
1780571997673	2026-06-04T11:19:57.673Z	2026-06-04T11:19:57.673050Z  INFO Skipping claim: bond already claimed on chain game_index=10144 game_address=0x41668875d037b5e57277168681e4a669ba39a931
1780571997677	2026-06-04T11:19:57.677Z	2026-06-04T11:19:57.675273Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8036638 parent_game_index=10144}: Creating game l2_block_number=8036638 parent_game_index=10144 output_root=0x1e6bca5a6f19343d7a8011d7219ed95c5d52e275e98fb8a8333782669acf1ff1
1780571997750	2026-06-04T11:19:57.750Z	2026-06-04T11:19:57.750739Z  WARN Failed to handle game creation: Failed to send KMS-signed transaction
1780571997751	2026-06-04T11:19:57.751Z	
1780571997751	2026-06-04T11:19:57.751Z	Caused by:
1780571997751	2026-06-04T11:19:57.751Z	    server returned an error response: error code -32000: execution reverted
1780572027665	2026-06-04T11:20:27.665Z	
1780572027666	2026-06-04T11:20:27.666Z	Caused by:
1780572027666	2026-06-04T11:20:27.666Z	    server returned an error response: error code -32000: execution reverted
1780572027678	2026-06-04T11:20:27.678Z	2026-06-04T11:20:27.678059Z  INFO Skipping claim: bond already claimed on chain game_index=10144 game_address=0x41668875d037b5e57277168681e4a669ba39a931
1780572027678	2026-06-04T11:20:27.678Z	2026-06-04T11:20:27.678851Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8036638 parent_game_index=10144}: Creating game l2_block_number=8036638 parent_game_index=10144 output_root=0x1e6bca5a6f19343d7a8011d7219ed95c5d52e275e98fb8a8333782669acf1ff1
1780572027776	2026-06-04T11:20:27.776Z	2026-06-04T11:20:27.776122Z  WARN Failed to handle game creation: Failed to send KMS-signed transaction
1780572027776	2026-06-04T11:20:27.776Z	
1780572027776	2026-06-04T11:20:27.776Z	Caused by:
1780572027776	2026-06-04T11:20:27.776Z	    server returned an error response: error code -32000: execution reverted
1780572057661	2026-06-04T11:20:57.661Z	
1780572057661	2026-06-04T11:20:57.661Z	Caused by:
1780572057661	2026-06-04T11:20:57.661Z	    server returned an error response: error code -32000: execution reverted
1780572057675	2026-06-04T11:20:57.675Z	2026-06-04T11:20:57.674845Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8036638 parent_game_index=4294967295}: Creating game l2_block_number=8036638 parent_game_index=4294967295 output_root=0x1e6bca5a6f19343d7a8011d7219ed95c5d52e275e98fb8a8333782669acf1ff1
1780572106832	2026-06-04T11:21:46.832Z	2026-06-04T11:21:46.832616Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8036638 parent_game_index=4294967295}: Game created successfully game_address=0x592b7ad4fd3958708b874b03f12f36e081d3d8be tx_hash=0x5b46b59acabd578d50f9621a9f26398eee8793c242a145a40e6695d489fc3dc6
1780572147667	2026-06-04T11:22:27.667Z	2026-06-04T11:22:27.667072Z  INFO Valid game: adding to cache game_index=10145 game_type=42 game_address=0x592b7ad4fd3958708b874b03f12f36e081d3d8be parent_index=4294967295 l2_block=8036638 status=IN_PROGRESS proposal_status=Unchallenged deadline=1780575660
1780572147685	2026-06-04T11:22:27.685Z	2026-06-04T11:22:27.685204Z  INFO Canonical head updated previous_canonical_index=Some(10144) new_canonical_index=10145 l2_block=8036638 total_games=2
1780572147704	2026-06-04T11:22:27.704Z	2026-06-04T11:22:27.704186Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8037538 parent_game_index=10145}: Creating game l2_block_number=8037538 parent_game_index=10145 output_root=0x59acfee0c00ca8948db3dfd45668bd4b44d37a94dffc3e45a41c4b53df87cdc4
1780572196891	2026-06-04T11:23:16.891Z	2026-06-04T11:23:16.891005Z  INFO [[Proposing]]{next_l2_block_number_for_proposal=8037538 parent_game_index=10145}: Game created successfully game_address=0x89fa2828e6018a582398eff6e602d7d070653cab tx_hash=0xa4ced7ed75bbfe80649abe58416328ca7297e1032a5f5210af5ef8bd789cc4f5
1780572237671	2026-06-04T11:23:57.671Z	2026-06-04T11:23:57.671008Z  INFO Valid game: adding to cache game_index=10146 game_type=42 game_address=0x89fa2828e6018a582398eff6e602d7d070653cab parent_index=10145 l2_block=8037538 status=IN_PROGRESS proposal_status=Unchallenged deadline=1780575756

Fix

Follow each qualifying alternative-chain root (genesis-rooted, or a lower parent index than the anchor head) to its highest-L2-block tip, instead of stopping at the root. The selection logic is extracted into ProposerState::select_canonical_head, which is a pure function of the cached games and the anchor.

Test plan

  • cargo test -p op-succinct-fp --lib — 5 new canonical_head unit tests: no-anchor, anchor subtree, and genesis / earlier-lineage catch-up chains followed to their tip
  • cargo clippy -p op-succinct-fp --lib --tests — clean
  • Soak on Celo internal testnet: confirm the proposer advances past a genesis-rooted catch-up chain after a finalized-head stall without re-stalling

Equivalent to celo-org#144

@piersy piersy deleted the seolaoh/proposer-canonical-head-catchup-upstream branch June 9, 2026 15:02
@seolaoh seolaoh restored the seolaoh/proposer-canonical-head-catchup-upstream branch June 10, 2026 08:38
After a long L2 finalized-head stall, the proposer can prune its cache
down to the anchor game and rebuild the backlog as a genesis-rooted
chain (parent = u32::MAX, since the contract reverts when a new game's
parent is the current anchor). compute_canonical_head pinned the head to
the root of that alternative chain instead of following it to the tip,
so the canonical head could not advance and game creation stalled until
the anchor caught up.

Follow each qualifying alternative-chain root (genesis-rooted, or a lower
parent index than the anchor head) to its highest-block tip. Extract the
selection into ProposerState::select_canonical_head and cover it with
unit tests.
@seolaoh seolaoh force-pushed the seolaoh/proposer-canonical-head-catchup-upstream branch from 01577eb to 37a9e03 Compare June 10, 2026 08:38
KamiD pushed a commit to okx/op-succinct that referenced this pull request Jun 11, 2026
Backport of upstream PR succinctlabs#926. After an L2 finalized-head stall plus bond
claim window expiry, the proposer prunes its cache to the anchor and
rebuilds the backlog as a fresh genesis-rooted chain (parent=u32::MAX).
compute_canonical_head used to stop at such a chain's root instead of
following it to its tip, pinning the head and stalling proposals.

Extract the selection into a pure ProposerState::select_canonical_head
method and add unit tests covering anchor-subtree, genesis-rooted
catch-up, deep genesis chain, and earlier-lineage override cases. The
tz-proposer shares ProposerState with the main proposer, so the fix
benefits tz flows automatically.
KamiD pushed a commit to okx/op-succinct that referenced this pull request Jun 11, 2026
…inctlabs#927

Upstream replaced succinctlabs#926 with succinctlabs#927, adding two increments on top of the
same core selection logic:
- Emit a tracing::debug! when the canonical head is chosen from a
  catch-up chain outside the anchor subtree, so operators can tell the
  proposer is recovering on a non-descendant lineage.
- Cover the multi-chain case in a new unit test
  (multiple_catchup_chains_select_highest_tip): after repeated
  stall/recovery cycles, several genesis-rooted catch-up chains can sit
  in the cache, and the head must be the highest tip pooled across all
  of them, not the tip of the chain whose root sits highest.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants