Skip to content

Commit

Permalink
fix off by one stake subsidy distribution counter
Browse files Browse the repository at this point in the history
  • Loading branch information
emmazzz committed Oct 25, 2024
1 parent 41dabbc commit f16a93e
Show file tree
Hide file tree
Showing 7 changed files with 116 additions and 34 deletions.
1 change: 0 additions & 1 deletion crates/sui-framework/docs/sui-system/stake_subsidy.md
Original file line number Diff line number Diff line change
Expand Up @@ -170,7 +170,6 @@ Advance the epoch counter and draw down the subsidy for the epoch.

// Drawn down the subsidy for this epoch.
<b>let</b> <a href="stake_subsidy.md#0x3_stake_subsidy">stake_subsidy</a> = self.<a href="../sui-framework/balance.md#0x2_balance">balance</a>.split(to_withdraw);

self.distribution_counter = self.distribution_counter + 1;

// Decrease the subsidy amount only when the current period ends.
Expand Down
12 changes: 8 additions & 4 deletions crates/sui-framework/docs/sui-system/sui_system_state_inner.md
Original file line number Diff line number Diff line change
Expand Up @@ -2164,18 +2164,22 @@ gas coins.
<b>let</b> storage_charge = storage_reward.value();
<b>let</b> computation_charge = computation_reward.value();
<b>let</b> <b>mut</b> <a href="stake_subsidy.md#0x3_stake_subsidy">stake_subsidy</a> = <a href="../sui-framework/balance.md#0x2_balance_zero">balance::zero</a>();

// during the transition from epoch N <b>to</b> epoch N + 1, ctx.<a href="sui_system_state_inner.md#0x3_sui_system_state_inner_epoch">epoch</a>() will <b>return</b> N
<b>let</b> old_epoch = ctx.<a href="sui_system_state_inner.md#0x3_sui_system_state_inner_epoch">epoch</a>();
// Include stake subsidy in the rewards given out <b>to</b> validators and stakers.
// Delay distributing any stake subsidies until after `stake_subsidy_start_epoch`.
// And <b>if</b> this epoch is shorter than the regular epoch duration, don't distribute any stake subsidy.
<b>if</b> (ctx.<a href="sui_system_state_inner.md#0x3_sui_system_state_inner_epoch">epoch</a>() &gt;= self.parameters.stake_subsidy_start_epoch &&
<b>if</b> (old_epoch &gt;= self.parameters.stake_subsidy_start_epoch &&
epoch_start_timestamp_ms &gt;= prev_epoch_start_timestamp + self.parameters.epoch_duration_ms)
{
// special case for epoch 560 -&gt; 561 change bug. add extra subsidies for "safe mode"
// <b>where</b> reward distribution was skipped. <b>use</b> distribution counter and epoch check <b>to</b>
// avoiding affecting devnet and testnet
<b>if</b> (self.<a href="stake_subsidy.md#0x3_stake_subsidy">stake_subsidy</a>.get_distribution_counter() == 540 && ctx.<a href="sui_system_state_inner.md#0x3_sui_system_state_inner_epoch">epoch</a>() &gt; 560) {
<b>let</b> first_safe_mode_epoch = 561;
<b>let</b> safe_mode_epoch_count = ctx.<a href="sui_system_state_inner.md#0x3_sui_system_state_inner_epoch">epoch</a>() - first_safe_mode_epoch;
<b>if</b> (self.<a href="stake_subsidy.md#0x3_stake_subsidy">stake_subsidy</a>.get_distribution_counter() == 540 && old_epoch &gt; 560) {
// safe mode was entered on the change from 560 <b>to</b> 561. so 560 was the first epoch without proper subsidy distribution
<b>let</b> first_safe_mode_epoch = 560;
<b>let</b> safe_mode_epoch_count = old_epoch - first_safe_mode_epoch;
safe_mode_epoch_count.do!(|_| {
<a href="stake_subsidy.md#0x3_stake_subsidy">stake_subsidy</a>.join(self.<a href="stake_subsidy.md#0x3_stake_subsidy">stake_subsidy</a>.<a href="sui_system_state_inner.md#0x3_sui_system_state_inner_advance_epoch">advance_epoch</a>());
});
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -65,7 +65,6 @@ module sui_system::stake_subsidy {

// Drawn down the subsidy for this epoch.
let stake_subsidy = self.balance.split(to_withdraw);

self.distribution_counter = self.distribution_counter + 1;

// Decrease the subsidy amount only when the current period ends.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -866,18 +866,22 @@ module sui_system::sui_system_state_inner {
let storage_charge = storage_reward.value();
let computation_charge = computation_reward.value();
let mut stake_subsidy = balance::zero();

// during the transition from epoch N to epoch N + 1, ctx.epoch() will return N
let old_epoch = ctx.epoch();
// Include stake subsidy in the rewards given out to validators and stakers.
// Delay distributing any stake subsidies until after `stake_subsidy_start_epoch`.
// And if this epoch is shorter than the regular epoch duration, don't distribute any stake subsidy.
if (ctx.epoch() >= self.parameters.stake_subsidy_start_epoch &&
if (old_epoch >= self.parameters.stake_subsidy_start_epoch &&
epoch_start_timestamp_ms >= prev_epoch_start_timestamp + self.parameters.epoch_duration_ms)
{
// special case for epoch 560 -> 561 change bug. add extra subsidies for "safe mode"
// where reward distribution was skipped. use distribution counter and epoch check to
// avoiding affecting devnet and testnet
if (self.stake_subsidy.get_distribution_counter() == 540 && ctx.epoch() > 560) {
let first_safe_mode_epoch = 561;
let safe_mode_epoch_count = ctx.epoch() - first_safe_mode_epoch;
if (self.stake_subsidy.get_distribution_counter() == 540 && old_epoch > 560) {
// safe mode was entered on the change from 560 to 561. so 560 was the first epoch without proper subsidy distribution
let first_safe_mode_epoch = 560;
let safe_mode_epoch_count = old_epoch - first_safe_mode_epoch;
safe_mode_epoch_count.do!(|_| {
stake_subsidy.join(self.stake_subsidy.advance_epoch());
});
Expand Down Expand Up @@ -1141,6 +1145,11 @@ module sui_system::sui_system_state_inner {
self.stake_subsidy.set_distribution_counter(counter)
}

#[test_only]
public(package) fun epoch_duration_ms(self: &SuiSystemStateInnerV2): u64 {
self.parameters.epoch_duration_ms
}

// CAUTION: THIS CODE IS ONLY FOR TESTING AND THIS MACRO MUST NEVER EVER BE REMOVED. Creates a
// candidate validator - bypassing the proof of possession check and other metadata validation
// in the process.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -493,32 +493,103 @@ module sui_system::rewards_distribution_tests {
test_scenario::return_shared(system_state);
}

#[test]
fun test_stake_subsidy_with_safe_mode() {
use std::unit_test::assert_eq;
fun check_distribution_counter_invariant(system: &mut SuiSystemState, ctx: &TxContext) {
assert!(ctx.epoch() == system.epoch());
// first subsidy distribution was at epoch 20, so counter should always be ahead by 20
assert_eq(system.get_stake_subsidy_distribution_counter() + 20, ctx.epoch());
}

#[test]
fun test_stake_subsidy_with_safe_mode_epoch_562_to_563() {
set_up_sui_system_state_with_big_amounts();

let mut test = test_scenario::begin(VALIDATOR_ADDR_1);
let mut sui_system = test.take_shared<SuiSystemState>();
let ctx = test.ctx();
// mimic state during epoch 562, if we're in safe mode since the 560 -> 561 epoch change
let start_epoch: u64 = 562;
let start_distribution_counter = 540;
let epoch_start_time = 100000000000;
let epoch_duration = sui_system.inner_mut_for_testing().epoch_duration_ms();

// increment epoch number (safe mode emulation)
start_epoch.do!(|_| ctx.increment_epoch_number());
sui_system.set_epoch_for_testing(start_epoch);
sui_system.set_stake_subsidy_distribution_counter(start_distribution_counter);

assert!(ctx.epoch() == start_epoch);
assert!(ctx.epoch() == sui_system.epoch());
assert!(sui_system.get_stake_subsidy_distribution_counter() == start_distribution_counter);

// perform advance epoch
sui_system
.inner_mut_for_testing()
.advance_epoch(start_epoch + 1, 65, balance::zero(), balance::zero(), 0, 0, 0, 0, epoch_start_time, ctx)
.destroy_for_testing(); // balance returned from `advance_epoch`
ctx.increment_epoch_number();

// should distribute 3 epochs worth of subsidies: 560, 561, 562
assert_eq(sui_system.get_stake_subsidy_distribution_counter(), start_distribution_counter + 3);
check_distribution_counter_invariant(&mut sui_system, ctx);

// ensure that next epoch change only distributes one epoch's worth
sui_system
.inner_mut_for_testing()
.advance_epoch(start_epoch + 2, 65, balance::zero(), balance::zero(), 0, 0, 0, 0, epoch_start_time + epoch_duration, ctx)
.destroy_for_testing(); // balance returned from `advance_epoch`
ctx.increment_epoch_number();

// should distribute 1 epoch's worth of subsidies: 563 only
assert_eq(sui_system.get_stake_subsidy_distribution_counter(), start_distribution_counter + 4);
check_distribution_counter_invariant(&mut sui_system, ctx);

test_scenario::return_shared(sui_system);
test.end();
}

#[test]
fun test_stake_subsidy_with_safe_mode_epoch_563_to_564() {
set_up_sui_system_state_with_big_amounts();

let mut test = test_scenario::begin(VALIDATOR_ADDR_1);
let mut sui_system = test.take_shared<SuiSystemState>();
let ctx = test.ctx();
// mimic state during epoch 563, if we're in safe mode since the 560 -> 561 epoch change
let start_epoch: u64 = 563;
let start_distribution_counter = 540;
let epoch_start_time = 100000000000;
let epoch_duration = sui_system.inner_mut_for_testing().epoch_duration_ms();

// increment epoch number (safe mode emulation)
562u64.do!(|_| ctx.increment_epoch_number());
sui_system.set_epoch_for_testing(562);
sui_system.set_stake_subsidy_distribution_counter(540);
start_epoch.do!(|_| ctx.increment_epoch_number());
sui_system.set_epoch_for_testing(start_epoch);
sui_system.set_stake_subsidy_distribution_counter(start_distribution_counter);

assert!(ctx.epoch() == 562);
assert!(sui_system.get_stake_subsidy_distribution_counter() == 540);
assert!(ctx.epoch() == start_epoch);
assert!(ctx.epoch() == sui_system.epoch());
assert!(sui_system.get_stake_subsidy_distribution_counter() == start_distribution_counter);

// perform advance epoch
sui_system
.inner_mut_for_testing()
.advance_epoch(563, 65, balance::zero(), balance::zero(), 0, 0, 0, 0, 100000000000, ctx)
.advance_epoch(start_epoch + 1, 65, balance::zero(), balance::zero(), 0, 0, 0, 0, epoch_start_time, ctx)
.destroy_for_testing(); // balance returned from `advance_epoch`
ctx.increment_epoch_number();

// should distribute 4 epochs worth of subsidies: 560, 561, 562, 563
assert_eq(sui_system.get_stake_subsidy_distribution_counter(), start_distribution_counter + 4);
check_distribution_counter_invariant(&mut sui_system, ctx);

// ensure that next epoch change only distributes one epoch's worth
sui_system
.inner_mut_for_testing()
.advance_epoch(start_epoch + 2, 65, balance::zero(), balance::zero(), 0, 0, 0, 0, epoch_start_time + epoch_duration, ctx)
.destroy_for_testing(); // balance returned from `advance_epoch`
ctx.increment_epoch_number();

assert_eq!(sui_system.get_stake_subsidy_distribution_counter(), 542);
// should distribute 1 epoch's worth of subsidies
assert_eq(sui_system.get_stake_subsidy_distribution_counter(), start_distribution_counter + 5);
check_distribution_counter_invariant(&mut sui_system, ctx);

test_scenario::return_shared(sui_system);
test.end();
Expand Down
Binary file modified crates/sui-framework/packages_compiled/sui-system
Binary file not shown.
Original file line number Diff line number Diff line change
Expand Up @@ -240,56 +240,56 @@ validators:
next_epoch_worker_address: ~
extra_fields:
id:
id: "0x25021b06e348c636a0971154e2fa451e8adb77944032ca139d6a8a78905c8667"
id: "0x03a63a15f9e8e024f98f47d797d2fa22baf2aeafee5aec4d66f09d9689c1a2c3"
size: 0
voting_power: 10000
operation_cap_id: "0x57863a99dd53b91959baa7f78224dcaf75a233408e33decbc7eff4525c94994a"
operation_cap_id: "0x009c04b7e355d845db8c5c4e2c79f94e1579ae28531a59a9411d1c71923ad619"
gas_price: 1000
staking_pool:
id: "0xf23a550978da24f82b454e8652eed7ab54ccd9432c6b3e4fed2346a8933f6ab6"
id: "0x758d1e73ea89a62fda28afdd7a0b327f8c37c849609d1badd6189557cc45a323"
activation_epoch: 0
deactivation_epoch: ~
sui_balance: 20000000000000000
rewards_pool:
value: 0
pool_token_balance: 20000000000000000
exchange_rates:
id: "0x241c206ff1b9dc817023ea4ab6d47ead2e52f1ff5486a4384f2aeda3047c3ba4"
id: "0x36ff3b4f616e5a027a671fc510c1db7baee6ec9159857d20b0ec458b38471d7a"
size: 1
pending_stake: 0
pending_total_sui_withdraw: 0
pending_pool_token_withdraw: 0
extra_fields:
id:
id: "0x285598570152e0252d02bc206348058ff05238374dd2899849e783d31041d0a5"
id: "0x6a92a767673917836d5ca96ec5035750f734a123e2dc00d9fa6823872c48af84"
size: 0
commission_rate: 200
next_epoch_stake: 20000000000000000
next_epoch_gas_price: 1000
next_epoch_commission_rate: 200
extra_fields:
id:
id: "0x200f6ce3e9e5cb38c095a5a2fffc93592b1715a2783e7d2b03f6990b222955d3"
id: "0xb2cb5fe76aa57d4abbb22ba2e49015b3e70e4a13f9f1ec84233358c5a6b580c9"
size: 0
pending_active_validators:
contents:
id: "0x9658c7b137b76541dcf33e8e0ebf70f4f53740b3d791e736468a27327f81afbd"
id: "0x2ef68115e3f22e2280feb00c57956531e48f518da60dbaa3fa4edefae45b19c8"
size: 0
pending_removals: []
staking_pool_mappings:
id: "0xd9db436196009d729b3be781aabdbd3f730d2a1fa32a6cb6760b3be1293f3564"
id: "0xaf4b76ff1f408e0bd825ef4942ffd31a62e928d8592edec331ff3a3221af2a14"
size: 1
inactive_validators:
id: "0x07a63d81a71e1b91eec484cd926e73bddca885d748581fcc9a5b3bc3efa0a060"
id: "0xd59da3a9b3aeb6b47f700dade9cc6c2a9c5054db52bf7efdff2d6b8864c02e75"
size: 0
validator_candidates:
id: "0x90a30f4334af3cbc8b0057a111ca36a3f43b14e472d1d6ca39c3b42b0aa24ce6"
id: "0xc14f302bf04a1debb67ed60b8a69fc743fc2d0112ee08a798b0329c97f2187f0"
size: 0
at_risk_validators:
contents: []
extra_fields:
id:
id: "0x38642ab5aefbdeaa8c1f9ed4c306a78660c2ed8da3d95fba22d7f04e2cf8dc28"
id: "0x4933a6d414b1d9e9b4c7234899ebab473fa50ceea51b698779a6e043035647ef"
size: 0
storage_fund:
total_object_storage_rebates:
Expand All @@ -306,7 +306,7 @@ parameters:
validator_low_stake_grace_period: 7
extra_fields:
id:
id: "0x5c893fa7618f2b3fe0754fa13180768e5bf74d502243760543ac78aa10384e4e"
id: "0x2e2f5f04aa912fa92c78411e5255460698199aa068b61148084b44f84616b5ef"
size: 0
reference_gas_price: 1000
validator_report_records:
Expand All @@ -320,7 +320,7 @@ stake_subsidy:
stake_subsidy_decrease_rate: 1000
extra_fields:
id:
id: "0x6f5ad342308a3bf8446888b8599edab6f7a49490c20d4d5d46b6948c790e076c"
id: "0x23b2a35c924ac30a0ea2e8b7fa4643e0078ae345778a7083cda02b79428043fc"
size: 0
safe_mode: false
safe_mode_storage_rewards:
Expand All @@ -332,6 +332,6 @@ safe_mode_non_refundable_storage_fee: 0
epoch_start_timestamp_ms: 10
extra_fields:
id:
id: "0x406c443f9696c9314bfab3e942d5dcb1663a671c3be52a2502c3c8aa626820bb"
id: "0x5b1db70db5006a12cc40708a295a77502167e893c18326d6628cf947ed364c56"
size: 0

0 comments on commit f16a93e

Please sign in to comment.