@lodestar/fork-choice 1.44.0-dev.985999b30c → 1.44.0-dev.9d8b487a59

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -21,6 +21,29 @@ import {
21
21
  * Spec: gloas/fork-choice.md (PAYLOAD_TIMELY_THRESHOLD = PTC_SIZE // 2)
22
22
  */
23
23
  const PAYLOAD_TIMELY_THRESHOLD = Math.floor(PTC_SIZE / 2);
24
+ /**
25
+ * Threshold for blob data availability via PTC vote
26
+ * Spec: gloas/fork-choice.md (DATA_AVAILABILITY_TIMELY_THRESHOLD = PTC_SIZE // 2)
27
+ */
28
+ const DATA_AVAILABILITY_TIMELY_THRESHOLD = Math.floor(PTC_SIZE / 2);
29
+
30
+ /**
31
+ * popcount(attended AND NOT yes) — explicit False-vote count.
32
+ * Excludes PTC members who didn't attest (the None state).
33
+ */
34
+ export function countNoVotes(attended: BitArray, yes: BitArray): number {
35
+ const a = attended.uint8Array;
36
+ const y = yes.uint8Array;
37
+ let count = 0;
38
+ for (let i = 0; i < a.length; i++) {
39
+ let byte = a[i] & ~y[i] & 0xff;
40
+ while (byte) {
41
+ byte &= byte - 1;
42
+ count++;
43
+ }
44
+ }
45
+ return count;
46
+ }
24
47
 
25
48
  export const DEFAULT_PRUNE_THRESHOLD = 0;
26
49
  type ProposerBoost = {root: RootHex; score: number};
@@ -64,12 +87,25 @@ export class ProtoArray {
64
87
  /**
65
88
  * PTC (Payload Timeliness Committee) votes per block as bitvectors
66
89
  * Maps block root to BitArray of PTC_SIZE bits (512 mainnet, 2 minimal)
67
- * Spec: gloas/fork-choice.md#modified-store (line 148)
90
+ * Spec: gloas/fork-choice.md#modified-store (payload_timeliness_vote)
68
91
  *
69
- * Bit i is set if PTC member i voted payload_present=true
70
- * Used by is_payload_timely() to determine if payload is timely
92
+ * Bit i = PTC member i voted payloadPresent=true (timeliness YES vote)
71
93
  */
72
94
  private ptcVotes = new Map<RootHex, BitArray>();
95
+ /**
96
+ * Blob data availability votes per block.
97
+ * Spec: gloas/fork-choice.md#modified-store (payload_data_availability_vote)
98
+ *
99
+ * Bit i = PTC member i voted blobDataAvailable=true (DA YES vote)
100
+ */
101
+ private daVotes = new Map<RootHex, BitArray>();
102
+ /**
103
+ * Tracks which PTC members have attested at all (any payload_status).
104
+ * Without this, we cannot tell "didn't vote" (None) from "voted false" —
105
+ * a distinction required by payload_timeliness/payload_data_availability
106
+ * when called with the negative parameter value.
107
+ */
108
+ private ptcAttested = new Map<RootHex, BitArray>();
73
109
 
74
110
  constructor({
75
111
  pruneThreshold,
@@ -514,9 +550,11 @@ export class ProtoArray {
514
550
  // Update bestChild for PENDING → EMPTY edge
515
551
  this.maybeUpdateBestChildAndDescendant(pendingIndex, emptyIndex, currentSlot, proposerBoostRoot);
516
552
 
517
- // Initialize PTC votes for this block (all false initially)
518
- // Spec: gloas/fork-choice.md#modified-on_block (line 645)
553
+ // Initialize PTC vote bitvectors for this block.
554
+ // Spec: gloas/fork-choice.md#modified-on_block
519
555
  this.ptcVotes.set(block.blockRoot, BitArray.fromBitLen(PTC_SIZE));
556
+ this.ptcAttested.set(block.blockRoot, BitArray.fromBitLen(PTC_SIZE));
557
+ this.daVotes.set(block.blockRoot, BitArray.fromBitLen(PTC_SIZE));
520
558
  } else {
521
559
  // Pre-Gloas: Only create FULL node (payload embedded in block)
522
560
  const node: ProtoNode = {
@@ -636,15 +674,18 @@ export class ProtoArray {
636
674
 
637
675
  /**
638
676
  * Update PTC votes for multiple validators attesting to a block
639
- * Spec: gloas/fork-choice.md#new-on_payload_attestation_message
640
- *
641
- * @param blockRoot - The beacon block root being attested
642
- * @param ptcIndices - Array of PTC committee indices that voted (0..PTC_SIZE-1)
643
- * @param payloadPresent - Whether the validators attest the payload is present
677
+ * Spec: gloas/fork-choice.md#new-notify_ptc_messages
644
678
  */
645
- notifyPtcMessages(blockRoot: RootHex, ptcIndices: number[], payloadPresent: boolean): void {
679
+ notifyPtcMessages(
680
+ blockRoot: RootHex,
681
+ ptcIndices: number[],
682
+ payloadPresent: boolean,
683
+ blobDataAvailable: boolean
684
+ ): void {
646
685
  const votes = this.ptcVotes.get(blockRoot);
647
- if (votes === undefined) {
686
+ const attended = this.ptcAttested.get(blockRoot);
687
+ const daVotes = this.daVotes.get(blockRoot);
688
+ if (votes === undefined || attended === undefined || daVotes === undefined) {
648
689
  // Block not found or not a Gloas block, ignore
649
690
  return;
650
691
  }
@@ -653,8 +694,9 @@ export class ProtoArray {
653
694
  if (ptcIndex < 0 || ptcIndex >= PTC_SIZE) {
654
695
  throw new Error(`Invalid PTC index: ${ptcIndex}, must be 0..${PTC_SIZE - 1}`);
655
696
  }
656
-
657
697
  votes.set(ptcIndex, payloadPresent);
698
+ daVotes.set(ptcIndex, blobDataAvailable);
699
+ attended.set(ptcIndex, true);
658
700
  }
659
701
  }
660
702
 
@@ -669,31 +711,61 @@ export class ProtoArray {
669
711
  }
670
712
 
671
713
  /**
672
- * Check if execution payload for a block is timely
673
- * Spec: gloas/fork-choice.md#new-is_payload_timely
674
- *
675
- * Returns true if:
676
- * 1. Block has PTC votes tracked
677
- * 2. Payload is locally available (FULL variant exists in proto array)
678
- * 3. More than PAYLOAD_TIMELY_THRESHOLD (>50% of PTC) members voted payload_present=true
679
- *
680
- * @param blockRoot - The beacon block root to check
714
+ * Spec: payload_timeliness(store, root, timely=True)
681
715
  */
682
716
  isPayloadTimely(blockRoot: RootHex): boolean {
683
717
  const votes = this.ptcVotes.get(blockRoot);
684
- if (votes === undefined) {
685
- // Block not found or not a Gloas block
686
- return false;
687
- }
718
+ if (votes === undefined) return false;
719
+ if (!this.hasPayload(blockRoot)) return false;
720
+ return bitCount(votes.uint8Array) > PAYLOAD_TIMELY_THRESHOLD;
721
+ }
688
722
 
689
- // If payload is not locally available, it's not timely
690
- if (!this.hasPayload(blockRoot)) {
691
- return false;
692
- }
723
+ /**
724
+ * Spec: payload_timeliness(store, root, timely=False)
725
+ */
726
+ isPayloadNotTimely(blockRoot: RootHex): boolean {
727
+ const votes = this.ptcVotes.get(blockRoot);
728
+ const attended = this.ptcAttested.get(blockRoot);
729
+ if (votes === undefined || attended === undefined) return false;
730
+ // Spec: not verified locally → returns `not False = True`
731
+ if (!this.hasPayload(blockRoot)) return true;
732
+ return countNoVotes(attended, votes) > PAYLOAD_TIMELY_THRESHOLD;
733
+ }
734
+
735
+ /**
736
+ * Spec: payload_data_availability(store, root, available=True)
737
+ */
738
+ isPayloadDataAvailable(blockRoot: RootHex): boolean {
739
+ const daVotes = this.daVotes.get(blockRoot);
740
+ if (daVotes === undefined) return false;
741
+ if (!this.hasPayload(blockRoot)) return false;
742
+ return bitCount(daVotes.uint8Array) > DATA_AVAILABILITY_TIMELY_THRESHOLD;
743
+ }
693
744
 
694
- // Count votes for payload_present=true
695
- const yesVotes = bitCount(votes.uint8Array);
696
- return yesVotes > PAYLOAD_TIMELY_THRESHOLD;
745
+ /**
746
+ * Spec: payload_data_availability(store, root, available=False)
747
+ */
748
+ isPayloadDataNotAvailable(blockRoot: RootHex): boolean {
749
+ const daVotes = this.daVotes.get(blockRoot);
750
+ const attended = this.ptcAttested.get(blockRoot);
751
+ if (daVotes === undefined || attended === undefined) return false;
752
+ // Spec: not verified locally → returns `not False = True`
753
+ if (!this.hasPayload(blockRoot)) return true;
754
+ return countNoVotes(attended, daVotes) > DATA_AVAILABILITY_TIMELY_THRESHOLD;
755
+ }
756
+
757
+ /**
758
+ * Spec: should_build_on_full(store, head)
759
+ *
760
+ * The proposer is forced to build on the EMPTY variant (effectively reorging)
761
+ * when the PTC majority voted that the blob data is not available.
762
+ */
763
+ shouldBuildOnFull(head: ProtoBlock): boolean {
764
+ if (head.payloadStatus === PayloadStatus.PENDING) {
765
+ throw new Error("shouldBuildOnFull called with PENDING head");
766
+ }
767
+ if (head.payloadStatus === PayloadStatus.EMPTY) return false;
768
+ return !this.isPayloadDataNotAvailable(head.blockRoot);
697
769
  }
698
770
 
699
771
  /**
@@ -711,7 +783,7 @@ export class ProtoArray {
711
783
  * Spec: gloas/fork-choice.md#new-should_extend_payload
712
784
  *
713
785
  * Returns true if payload is verified (FULL variant exists) AND:
714
- * 1. Payload is timely, OR
786
+ * 1. Payload is timely AND data is available, OR
715
787
  * 2. No proposer boost root (empty/zero hash), OR
716
788
  * 3. Proposer boost root's parent is not this block, OR
717
789
  * 4. Proposer boost root extends FULL parent
@@ -724,8 +796,8 @@ export class ProtoArray {
724
796
  return false;
725
797
  }
726
798
 
727
- // Condition 1: Payload is timely
728
- if (this.isPayloadTimely(blockRoot)) {
799
+ // Condition 1: Payload is timely AND data is available
800
+ if (this.isPayloadTimely(blockRoot) && this.isPayloadDataAvailable(blockRoot)) {
729
801
  return true;
730
802
  }
731
803
 
@@ -1134,6 +1206,8 @@ export class ProtoArray {
1134
1206
  // Prune PTC votes for this block to prevent memory leak
1135
1207
  // Spec: gloas/fork-choice.md (implicit - finalized blocks don't need PTC votes)
1136
1208
  this.ptcVotes.delete(root);
1209
+ this.ptcAttested.delete(root);
1210
+ this.daVotes.delete(root);
1137
1211
  }
1138
1212
 
1139
1213
  // Store nodes prior to finalization