sdk: Add new version of `StakeState` to avoid breaking downstream users (#32736)

* sdk: Rename `StakeState` -> `StakeStateWithFlags`

* Add back `StakeFlags` with a deprecation warning
This commit is contained in:
Jon Cinque 2023-08-09 00:05:40 +02:00 committed by GitHub
parent ef318c23ae
commit 8e4a9a94ed
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
24 changed files with 782 additions and 466 deletions

View File

@ -6,24 +6,24 @@ use {
bincode::deserialize,
solana_sdk::{
clock::{Epoch, UnixTimestamp},
stake::state::{Authorized, Delegation, Lockup, Meta, Stake, StakeState},
stake::state::{Authorized, Delegation, Lockup, Meta, Stake, StakeStateWithFlags},
},
};
pub fn parse_stake(data: &[u8]) -> Result<StakeAccountType, ParseAccountError> {
let stake_state: StakeState = deserialize(data)
let stake_state: StakeStateWithFlags = deserialize(data)
.map_err(|_| ParseAccountError::AccountNotParsable(ParsableAccount::Stake))?;
let parsed_account = match stake_state {
StakeState::Uninitialized => StakeAccountType::Uninitialized,
StakeState::Initialized(meta) => StakeAccountType::Initialized(UiStakeAccount {
StakeStateWithFlags::Uninitialized => StakeAccountType::Uninitialized,
StakeStateWithFlags::Initialized(meta) => StakeAccountType::Initialized(UiStakeAccount {
meta: meta.into(),
stake: None,
}),
StakeState::Stake(meta, stake, _) => StakeAccountType::Delegated(UiStakeAccount {
StakeStateWithFlags::Stake(meta, stake, _) => StakeAccountType::Delegated(UiStakeAccount {
meta: meta.into(),
stake: Some(stake.into()),
}),
StakeState::RewardsPool => StakeAccountType::RewardsPool,
StakeStateWithFlags::RewardsPool => StakeAccountType::RewardsPool,
};
Ok(parsed_account)
}
@ -146,7 +146,7 @@ mod test {
#[test]
#[allow(deprecated)]
fn test_parse_stake() {
let stake_state = StakeState::Uninitialized;
let stake_state = StakeStateWithFlags::Uninitialized;
let stake_data = serialize(&stake_state).unwrap();
assert_eq!(
parse_stake(&stake_data).unwrap(),
@ -167,7 +167,7 @@ mod test {
lockup,
};
let stake_state = StakeState::Initialized(meta);
let stake_state = StakeStateWithFlags::Initialized(meta);
let stake_data = serialize(&stake_state).unwrap();
assert_eq!(
parse_stake(&stake_data).unwrap(),
@ -200,7 +200,7 @@ mod test {
credits_observed: 10,
};
let stake_state = StakeState::Stake(meta, stake, StakeFlags::empty());
let stake_state = StakeStateWithFlags::Stake(meta, stake, StakeFlags::empty());
let stake_data = serialize(&stake_state).unwrap();
assert_eq!(
parse_stake(&stake_data).unwrap(),
@ -230,7 +230,7 @@ mod test {
})
);
let stake_state = StakeState::RewardsPool;
let stake_state = StakeStateWithFlags::RewardsPool;
let stake_data = serialize(&stake_state).unwrap();
assert_eq!(
parse_stake(&stake_data).unwrap(),

View File

@ -54,7 +54,7 @@ use {
rpc_port::DEFAULT_RPC_PORT_STR,
signature::Signature,
slot_history,
stake::{self, state::StakeState},
stake::{self, state::StakeStateWithFlags},
system_instruction,
sysvar::{
self,
@ -1768,7 +1768,7 @@ pub fn process_show_stakes(
// Use server-side filtering if only one vote account is provided
if vote_account_pubkeys.len() == 1 {
program_accounts_config.filters = Some(vec![
// Filter by `StakeState::Stake(_, _)`
// Filter by `StakeStateWithFlags::Stake(_, _)`
RpcFilterType::Memcmp(Memcmp::new_base58_encoded(0, &[2, 0, 0, 0])),
// Filter by `Delegation::voter_pubkey`, which begins at byte offset 124
RpcFilterType::Memcmp(Memcmp::new_base58_encoded(
@ -1809,7 +1809,7 @@ pub fn process_show_stakes(
for (stake_pubkey, stake_account) in all_stake_accounts {
if let Ok(stake_state) = stake_account.state() {
match stake_state {
StakeState::Initialized(_) => {
StakeStateWithFlags::Initialized(_) => {
if vote_account_pubkeys.is_none() {
stake_accounts.push(CliKeyedStakeState {
stake_pubkey: stake_pubkey.to_string(),
@ -1824,7 +1824,7 @@ pub fn process_show_stakes(
});
}
}
StakeState::Stake(_, stake, _) => {
StakeStateWithFlags::Stake(_, stake, _) => {
if vote_account_pubkeys.is_none()
|| vote_account_pubkeys
.unwrap()
@ -2157,7 +2157,7 @@ impl RentLengthValue {
pub fn length(&self) -> usize {
match self {
Self::Nonce => NonceState::size(),
Self::Stake => StakeState::size_of(),
Self::Stake => StakeStateWithFlags::size_of(),
Self::System => 0,
Self::Vote => VoteState::size_of(),
Self::Bytes(l) => *l,

View File

@ -47,7 +47,10 @@ use {
stake::{
self,
instruction::{self as stake_instruction, LockupArgs, StakeError},
state::{Authorized, Lockup, Meta, StakeActivationStatus, StakeAuthorize, StakeState},
state::{
Authorized, Lockup, Meta, StakeActivationStatus, StakeAuthorize,
StakeStateWithFlags,
},
tools::{acceptable_reference_epoch_credits, eligible_for_deactivate_delinquent},
},
stake_history::{Epoch, StakeHistory},
@ -1422,7 +1425,7 @@ pub fn process_create_stake_account(
}
let minimum_balance =
rpc_client.get_minimum_balance_for_rent_exemption(StakeState::size_of())?;
rpc_client.get_minimum_balance_for_rent_exemption(StakeStateWithFlags::size_of())?;
if lamports < minimum_balance {
return Err(CliError::BadParameter(format!(
@ -1500,8 +1503,8 @@ pub fn process_stake_authorize(
let authority = config.signers[*authority];
if let Some(current_stake_account) = current_stake_account {
let authorized = match current_stake_account {
StakeState::Stake(Meta { authorized, .. }, ..) => Some(authorized),
StakeState::Initialized(Meta { authorized, .. }) => Some(authorized),
StakeStateWithFlags::Stake(Meta { authorized, .. }, ..) => Some(authorized),
StakeStateWithFlags::Initialized(Meta { authorized, .. }) => Some(authorized),
_ => None,
};
if let Some(authorized) = authorized {
@ -1630,7 +1633,7 @@ pub fn process_deactivate_stake_account(
let vote_account_address = match stake_account.state() {
Ok(stake_state) => match stake_state {
StakeState::Stake(_, stake, _) => stake.delegation.voter_pubkey,
StakeStateWithFlags::Stake(_, stake, _) => stake.delegation.voter_pubkey,
_ => {
return Err(CliError::BadParameter(format!(
"{stake_account_address} is not a delegated stake account",
@ -1895,7 +1898,7 @@ pub fn process_split_stake(
}
let minimum_balance =
rpc_client.get_minimum_balance_for_rent_exemption(StakeState::size_of())?;
rpc_client.get_minimum_balance_for_rent_exemption(StakeStateWithFlags::size_of())?;
if lamports < minimum_balance {
return Err(CliError::BadParameter(format!(
@ -2116,8 +2119,8 @@ pub fn process_stake_set_lockup(
if !sign_only {
let state = get_stake_account_state(rpc_client, stake_account_pubkey, config.commitment)?;
let lockup = match state {
StakeState::Stake(Meta { lockup, .. }, ..) => Some(lockup),
StakeState::Initialized(Meta { lockup, .. }) => Some(lockup),
StakeStateWithFlags::Stake(Meta { lockup, .. }, ..) => Some(lockup),
StakeStateWithFlags::Initialized(Meta { lockup, .. }) => Some(lockup),
_ => None,
};
if let Some(lockup) = lockup {
@ -2184,14 +2187,14 @@ fn u64_some_if_not_zero(n: u64) -> Option<u64> {
pub fn build_stake_state(
account_balance: u64,
stake_state: &StakeState,
stake_state: &StakeStateWithFlags,
use_lamports_unit: bool,
stake_history: &StakeHistory,
clock: &Clock,
new_rate_activation_epoch: Option<Epoch>,
) -> CliStakeState {
match stake_state {
StakeState::Stake(
StakeStateWithFlags::Stake(
Meta {
rent_exempt_reserve,
authorized,
@ -2248,16 +2251,16 @@ pub fn build_stake_state(
..CliStakeState::default()
}
}
StakeState::RewardsPool => CliStakeState {
StakeStateWithFlags::RewardsPool => CliStakeState {
stake_type: CliStakeType::RewardsPool,
account_balance,
..CliStakeState::default()
},
StakeState::Uninitialized => CliStakeState {
StakeStateWithFlags::Uninitialized => CliStakeState {
account_balance,
..CliStakeState::default()
},
StakeState::Initialized(Meta {
StakeStateWithFlags::Initialized(Meta {
rent_exempt_reserve,
authorized,
lockup,
@ -2285,7 +2288,7 @@ fn get_stake_account_state(
rpc_client: &RpcClient,
stake_account_pubkey: &Pubkey,
commitment_config: CommitmentConfig,
) -> Result<StakeState, Box<dyn std::error::Error>> {
) -> Result<StakeStateWithFlags, Box<dyn std::error::Error>> {
let stake_account = rpc_client
.get_account_with_commitment(stake_account_pubkey, commitment_config)?
.value

View File

@ -26,7 +26,7 @@ use {
stake::{
self,
instruction::LockupArgs,
state::{Lockup, StakeAuthorize, StakeState},
state::{Lockup, StakeAuthorize, StakeStateWithFlags},
},
},
solana_streamer::socket::SocketAddrSpace,
@ -162,10 +162,10 @@ fn test_stake_redelegation() {
// `stake_keypair` should now be delegated to `vote_keypair` and fully activated
let stake_account = rpc_client.get_account(&stake_keypair.pubkey()).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let rent_exempt_reserve = match stake_state {
StakeState::Stake(meta, stake, _) => {
StakeStateWithFlags::Stake(meta, stake, _) => {
assert_eq!(stake.delegation.voter_pubkey, vote_keypair.pubkey());
meta.rent_exempt_reserve
}
@ -268,10 +268,10 @@ fn test_stake_redelegation() {
// `stake2_keypair` should now be delegated to `vote2_keypair` and fully activated
let stake2_account = rpc_client.get_account(&stake2_keypair.pubkey()).unwrap();
let stake2_state: StakeState = stake2_account.state().unwrap();
let stake2_state: StakeStateWithFlags = stake2_account.state().unwrap();
match stake2_state {
StakeState::Stake(_meta, stake, _) => {
StakeStateWithFlags::Stake(_meta, stake, _) => {
assert_eq!(stake.delegation.voter_pubkey, vote2_keypair.pubkey());
}
_ => panic!("Unexpected stake2 state!"),
@ -966,9 +966,9 @@ fn test_stake_authorize() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let current_authority = match stake_state {
StakeState::Initialized(meta) => meta.authorized.staker,
StakeStateWithFlags::Initialized(meta) => meta.authorized.staker,
_ => panic!("Unexpected stake state!"),
};
assert_eq!(current_authority, online_authority_pubkey);
@ -1008,9 +1008,11 @@ fn test_stake_authorize() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let (current_staker, current_withdrawer) = match stake_state {
StakeState::Initialized(meta) => (meta.authorized.staker, meta.authorized.withdrawer),
StakeStateWithFlags::Initialized(meta) => {
(meta.authorized.staker, meta.authorized.withdrawer)
}
_ => panic!("Unexpected stake state!"),
};
assert_eq!(current_staker, online_authority2_pubkey);
@ -1040,9 +1042,9 @@ fn test_stake_authorize() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let current_authority = match stake_state {
StakeState::Initialized(meta) => meta.authorized.staker,
StakeStateWithFlags::Initialized(meta) => meta.authorized.staker,
_ => panic!("Unexpected stake state!"),
};
assert_eq!(current_authority, offline_authority_pubkey);
@ -1097,9 +1099,9 @@ fn test_stake_authorize() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let current_authority = match stake_state {
StakeState::Initialized(meta) => meta.authorized.staker,
StakeStateWithFlags::Initialized(meta) => meta.authorized.staker,
_ => panic!("Unexpected stake state!"),
};
assert_eq!(current_authority, nonced_authority_pubkey);
@ -1184,9 +1186,9 @@ fn test_stake_authorize() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let current_authority = match stake_state {
StakeState::Initialized(meta) => meta.authorized.staker,
StakeStateWithFlags::Initialized(meta) => meta.authorized.staker,
_ => panic!("Unexpected stake state!"),
};
assert_eq!(current_authority, online_authority_pubkey);
@ -1434,7 +1436,7 @@ fn test_stake_split() {
// Create stake account, identity is authority
let stake_balance = rpc_client
.get_minimum_balance_for_rent_exemption(StakeState::size_of())
.get_minimum_balance_for_rent_exemption(StakeStateWithFlags::size_of())
.unwrap()
+ 10_000_000_000;
let stake_keypair = keypair_from_seed(&[0u8; 32]).unwrap();
@ -1587,7 +1589,7 @@ fn test_stake_set_lockup() {
// Create stake account, identity is authority
let stake_balance = rpc_client
.get_minimum_balance_for_rent_exemption(StakeState::size_of())
.get_minimum_balance_for_rent_exemption(StakeStateWithFlags::size_of())
.unwrap()
+ 10_000_000_000;
@ -1645,9 +1647,9 @@ fn test_stake_set_lockup() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let current_lockup = match stake_state {
StakeState::Initialized(meta) => meta.lockup,
StakeStateWithFlags::Initialized(meta) => meta.lockup,
_ => panic!("Unexpected stake state!"),
};
assert_eq!(
@ -1704,9 +1706,9 @@ fn test_stake_set_lockup() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let current_lockup = match stake_state {
StakeState::Initialized(meta) => meta.lockup,
StakeStateWithFlags::Initialized(meta) => meta.lockup,
_ => panic!("Unexpected stake state!"),
};
assert_eq!(
@ -1811,9 +1813,9 @@ fn test_stake_set_lockup() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let current_lockup = match stake_state {
StakeState::Initialized(meta) => meta.lockup,
StakeStateWithFlags::Initialized(meta) => meta.lockup,
_ => panic!("Unexpected stake state!"),
};
assert_eq!(
@ -2187,9 +2189,9 @@ fn test_stake_checked_instructions() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let current_authority = match stake_state {
StakeState::Initialized(meta) => meta.authorized.staker,
StakeStateWithFlags::Initialized(meta) => meta.authorized.staker,
_ => panic!("Unexpected stake state!"),
};
assert_eq!(current_authority, staker_pubkey);
@ -2244,9 +2246,9 @@ fn test_stake_checked_instructions() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let current_authority = match stake_state {
StakeState::Initialized(meta) => meta.authorized.withdrawer,
StakeStateWithFlags::Initialized(meta) => meta.authorized.withdrawer,
_ => panic!("Unexpected stake state!"),
};
assert_eq!(current_authority, new_withdrawer_pubkey);
@ -2293,9 +2295,9 @@ fn test_stake_checked_instructions() {
};
process_command(&config).unwrap();
let stake_account = rpc_client.get_account(&stake_account_pubkey).unwrap();
let stake_state: StakeState = stake_account.state().unwrap();
let stake_state: StakeStateWithFlags = stake_account.state().unwrap();
let current_lockup = match stake_state {
StakeState::Initialized(meta) => meta.lockup,
StakeStateWithFlags::Initialized(meta) => meta.lockup,
_ => panic!("Unexpected stake state!"),
};
assert_eq!(

View File

@ -11,8 +11,8 @@ msc {
VoteSigner <:> Validator [label="register\n\n(optional)"];
Validator => Cluster [label="VoteState::Initialize(VoteSigner)"];
StakerX => Cluster [label="StakeState::Delegate(Validator)"];
StakerY => Cluster [label="StakeState::Delegate(Validator)"];
StakerX => Cluster [label="StakeStateWithFlags::Delegate(Validator)"];
StakerY => Cluster [label="StakeStateWithFlags::Delegate(Validator)"];
|||;
Validator box Cluster [label="\nvalidate\n"];

View File

@ -14,7 +14,7 @@ A separate Stake account \(created by a staker\) names a Vote account to which t
Any number of Stake accounts can delegate to a single Vote account without an interactive action from the identity controlling the Vote account or submitting votes to the account.
The total stake allocated to a Vote account can be calculated by the sum of all the Stake accounts that have the Vote account pubkey as the `StakeState::Stake::voter_pubkey`.
The total stake allocated to a Vote account can be calculated by the sum of all the Stake accounts that have the Vote account pubkey as the `StakeStateWithFlags::Stake::voter_pubkey`.
## Vote and Stake accounts
@ -62,13 +62,13 @@ Updates the account with a new authorized voter or withdrawer, according to the
- `account[1]` - RO - `sysvar::slot_hashes` A list of some N most recent slots and their hashes for the vote to be verified against.
- `account[2]` - RO - `sysvar::clock` The current network time, expressed in slots, epochs.
### StakeState
### StakeStateWithFlags
A StakeState takes one of four forms, StakeState::Uninitialized, StakeState::Initialized, StakeState::Stake, and StakeState::RewardsPool. Only the first three forms are used in staking, but only StakeState::Stake is interesting. All RewardsPools are created at genesis.
A StakeStateWithFlags takes one of four forms, StakeStateWithFlags::Uninitialized, StakeStateWithFlags::Initialized, StakeStateWithFlags::Stake, and StakeStateWithFlags::RewardsPool. Only the first three forms are used in staking, but only StakeStateWithFlags::Stake is interesting. All RewardsPools are created at genesis.
### StakeState::Stake
### StakeStateWithFlags::Stake
StakeState::Stake is the current delegation preference of the **staker** and contains the following state information:
StakeStateWithFlags::Stake is the current delegation preference of the **staker** and contains the following state information:
- Account::lamports - The lamports available for staking.
- `stake` - the staked amount \(subject to warmup and cooldown\) for generating rewards, always less than or equal to Account::lamports.
@ -79,7 +79,7 @@ StakeState::Stake is the current delegation preference of the **staker** and con
- `authorized_staker` - the pubkey of the entity that must sign delegation, activation, and deactivation transactions.
- `authorized_withdrawer` - the identity of the entity in charge of the lamports of this account, separate from the account's address, and the authorized staker.
### StakeState::RewardsPool
### StakeStateWithFlags::RewardsPool
To avoid a single network-wide lock or contention in redemption, 256 RewardsPools are part of genesis under pre-determined keys, each with std::u64::MAX credits to be able to satisfy redemptions according to point value.
@ -87,9 +87,9 @@ The Stakes and the RewardsPool are accounts that are owned by the same `Stake` p
### StakeInstruction::DelegateStake
The Stake account is moved from Initialized to StakeState::Stake form, or from a deactivated (i.e. fully cooled-down) StakeState::Stake to activated StakeState::Stake. This is how stakers choose the vote account and validator node to which their stake account lamports are delegated. The transaction must be signed by the stake's `authorized_staker`.
The Stake account is moved from Initialized to StakeStateWithFlags::Stake form, or from a deactivated (i.e. fully cooled-down) StakeStateWithFlags::Stake to activated StakeStateWithFlags::Stake. This is how stakers choose the vote account and validator node to which their stake account lamports are delegated. The transaction must be signed by the stake's `authorized_staker`.
- `account[0]` - RW - The StakeState::Stake instance. `StakeState::Stake::credits_observed` is initialized to `VoteState::credits`, `StakeState::Stake::voter_pubkey` is initialized to `account[1]`. If this is the initial delegation of stake, `StakeState::Stake::stake` is initialized to the account's balance in lamports, `StakeState::Stake::activated` is initialized to the current Bank epoch, and `StakeState::Stake::deactivated` is initialized to std::u64::MAX
- `account[0]` - RW - The StakeStateWithFlags::Stake instance. `StakeStateWithFlags::Stake::credits_observed` is initialized to `VoteState::credits`, `StakeStateWithFlags::Stake::voter_pubkey` is initialized to `account[1]`. If this is the initial delegation of stake, `StakeStateWithFlags::Stake::stake` is initialized to the account's balance in lamports, `StakeStateWithFlags::Stake::activated` is initialized to the current Bank epoch, and `StakeStateWithFlags::Stake::deactivated` is initialized to std::u64::MAX
- `account[1]` - R - The VoteState instance.
- `account[2]` - R - sysvar::clock account, carries information about current Bank epoch.
- `account[3]` - R - sysvar::stakehistory account, carries information about stake history.
@ -99,25 +99,25 @@ The Stake account is moved from Initialized to StakeState::Stake form, or from a
Updates the account with a new authorized staker or withdrawer, according to the StakeAuthorize parameter \(`Staker` or `Withdrawer`\). The transaction must be by signed by the Stakee account's current `authorized_staker` or `authorized_withdrawer`. Any stake lock-up must have expired, or the lock-up custodian must also sign the transaction.
- `account[0]` - RW - The StakeState.
- `account[0]` - RW - The StakeStateWithFlags.
`StakeState::authorized_staker` or `authorized_withdrawer` is set to to `Pubkey`.
`StakeStateWithFlags::authorized_staker` or `authorized_withdrawer` is set to to `Pubkey`.
### StakeInstruction::Deactivate
A staker may wish to withdraw from the network. To do so he must first deactivate his stake, and wait for cooldown.
The transaction must be signed by the stake's `authorized_staker`.
- `account[0]` - RW - The StakeState::Stake instance that is deactivating.
- `account[0]` - RW - The StakeStateWithFlags::Stake instance that is deactivating.
- `account[1]` - R - sysvar::clock account from the Bank that carries current epoch.
StakeState::Stake::deactivated is set to the current epoch + cooldown. The account's stake will ramp down to zero by that epoch, and Account::lamports will be available for withdrawal.
StakeStateWithFlags::Stake::deactivated is set to the current epoch + cooldown. The account's stake will ramp down to zero by that epoch, and Account::lamports will be available for withdrawal.
### StakeInstruction::Withdraw\(u64\)
Lamports build up over time in a Stake account and any excess over activated stake can be withdrawn. The transaction must be signed by the stake's `authorized_withdrawer`.
- `account[0]` - RW - The StakeState::Stake from which to withdraw.
- `account[0]` - RW - The StakeStateWithFlags::Stake from which to withdraw.
- `account[1]` - RW - Account that should be credited with the withdrawn lamports.
- `account[2]` - R - sysvar::clock account from the Bank that carries current epoch, to calculate stake.
- `account[3]` - R - sysvar::stake_history account from the Bank that carries stake warmup/cooldown history.

View File

@ -31,7 +31,7 @@ use {
rent::Rent,
signature::{Keypair, Signer},
signer::keypair::read_keypair_file,
stake::state::StakeState,
stake::state::StakeStateWithFlags,
system_program, timing,
},
solana_stake_program::stake_state,
@ -141,7 +141,7 @@ fn main() -> Result<(), Box<dyn error::Error>> {
.to_string();
// stake account
let default_bootstrap_validator_stake_lamports = &sol_to_lamports(0.5)
.max(rent.minimum_balance(StakeState::size_of()))
.max(rent.minimum_balance(StakeStateWithFlags::size_of()))
.to_string();
let default_target_tick_duration =
@ -444,7 +444,7 @@ fn main() -> Result<(), Box<dyn error::Error>> {
let bootstrap_validator_stake_lamports = rent_exempt_check(
&matches,
"bootstrap_validator_stake_lamports",
rent.minimum_balance(StakeState::size_of()),
rent.minimum_balance(StakeStateWithFlags::size_of()),
)?;
let bootstrap_stake_authorized_pubkey =

View File

@ -11,7 +11,7 @@ use {
pubkey::Pubkey,
stake::{
self,
state::{Authorized, Lockup, StakeState},
state::{Authorized, Lockup, StakeStateWithFlags},
},
system_program,
timing::years_as_slots,
@ -107,7 +107,9 @@ pub fn create_and_add_stakes(
let mut address_generator = AddressGenerator::new(&authorized.staker, &stake::program::id());
let stake_rent_reserve = genesis_config.rent.minimum_balance(StakeState::size_of());
let stake_rent_reserve = genesis_config
.rent
.minimum_balance(StakeStateWithFlags::size_of());
for unlock in unlocks {
let lamports = unlock.amount(stakes_lamports);
@ -193,7 +195,9 @@ mod tests {
.iter()
.all(|(_pubkey, account)| account.lamports <= granularity
|| account.lamports - granularity
<= genesis_config.rent.minimum_balance(StakeState::size_of())));
<= genesis_config
.rent
.minimum_balance(StakeStateWithFlags::size_of())));
}
// #[ignore]
@ -238,7 +242,7 @@ mod tests {
..Rent::default()
};
let reserve = rent.minimum_balance(StakeState::size_of());
let reserve = rent.minimum_balance(StakeStateWithFlags::size_of());
let staker_reserve = rent.minimum_balance(0);
// verify that a small remainder ends up in the last stake

View File

@ -74,7 +74,7 @@ use {
pubkey::Pubkey,
rent::Rent,
shred_version::compute_shred_version,
stake::{self, state::StakeState},
stake::{self, state::StakeStateWithFlags},
system_program,
transaction::{
MessageHash, SanitizedTransaction, SimpleAddressLoader, VersionedTransaction,
@ -1293,7 +1293,7 @@ fn main() {
.max(VoteState::get_rent_exempt_reserve(&rent))
.to_string();
let default_bootstrap_validator_stake_lamports = &sol_to_lamports(0.5)
.max(rent.minimum_balance(StakeState::size_of()))
.max(rent.minimum_balance(StakeStateWithFlags::size_of()))
.to_string();
let default_graph_vote_account_mode = GraphVoteAccountMode::default();
@ -2768,7 +2768,7 @@ fn main() {
value_t_or_exit!(arg_matches, "bootstrap_validator_lamports", u64);
let bootstrap_validator_stake_lamports =
value_t_or_exit!(arg_matches, "bootstrap_validator_stake_lamports", u64);
let minimum_stake_lamports = rent.minimum_balance(StakeState::size_of());
let minimum_stake_lamports = rent.minimum_balance(StakeStateWithFlags::size_of());
if bootstrap_validator_stake_lamports < minimum_stake_lamports {
eprintln!(
"Error: insufficient --bootstrap-validator-stake-lamports. \
@ -2996,7 +2996,9 @@ fn main() {
.unwrap()
.into_iter()
{
if let Ok(StakeState::Stake(meta, stake, _)) = account.state() {
if let Ok(StakeStateWithFlags::Stake(meta, stake, _)) =
account.state()
{
if vote_accounts_to_destake
.contains(&stake.delegation.voter_pubkey)
{
@ -3006,7 +3008,9 @@ fn main() {
address, stake.delegation.voter_pubkey,
);
}
account.set_state(&StakeState::Initialized(meta)).unwrap();
account
.set_state(&StakeStateWithFlags::Initialized(meta))
.unwrap();
bank.store_account(&address, &account);
}
}

View File

@ -18,7 +18,7 @@ use {
signature::{Keypair, Signer},
stake::{
instruction as stake_instruction,
state::{Authorized, Lockup, StakeActivationStatus, StakeState},
state::{Authorized, Lockup, StakeActivationStatus, StakeStateWithFlags},
},
system_instruction, system_program,
sysvar::{
@ -271,7 +271,7 @@ async fn stake_rewards_from_warp() {
.expect("account exists")
.unwrap();
let stake_state: StakeState = deserialize(&account.data).unwrap();
let stake_state: StakeStateWithFlags = deserialize(&account.data).unwrap();
let stake_history: StakeHistory = deserialize(&stake_history_account.data).unwrap();
let clock: Clock = deserialize(&clock_account.data).unwrap();
let stake = stake_state.stake().unwrap();
@ -387,7 +387,7 @@ async fn stake_rewards_filter_bench_core(num_stake_accounts: u64) {
.expect("account exists")
.unwrap();
let stake_state: StakeState = deserialize(&account.data).unwrap();
let stake_state: StakeStateWithFlags = deserialize(&account.data).unwrap();
let stake_history: StakeHistory = deserialize(&stake_history_account.data).unwrap();
let clock: Clock = deserialize(&clock_account.data).unwrap();
let stake = stake_state.stake().unwrap();
@ -409,7 +409,7 @@ async fn check_credits_observed(
.await
.unwrap()
.unwrap();
let stake_state: StakeState = deserialize(&stake_account.data).unwrap();
let stake_state: StakeStateWithFlags = deserialize(&stake_account.data).unwrap();
assert_eq!(
stake_state.stake().unwrap().credits_observed,
expected_credits
@ -465,7 +465,7 @@ async fn stake_merge_immediately_after_activation() {
.await
.unwrap()
.unwrap();
let stake_state: StakeState = deserialize(&stake_account.data).unwrap();
let stake_state: StakeStateWithFlags = deserialize(&stake_account.data).unwrap();
assert_eq!(stake_state.stake().unwrap().credits_observed, 300);
assert!(stake_account.lamports > stake_lamports);
@ -476,7 +476,7 @@ async fn stake_merge_immediately_after_activation() {
.await
.unwrap()
.unwrap();
let stake_state: StakeState = deserialize(&stake_account.data).unwrap();
let stake_state: StakeStateWithFlags = deserialize(&stake_account.data).unwrap();
assert_eq!(stake_state.stake().unwrap().credits_observed, 300);
assert_eq!(stake_account.lamports, stake_lamports);

File diff suppressed because it is too large Load Diff

View File

@ -73,28 +73,34 @@ pub(crate) fn null_tracer() -> Option<impl Fn(&InflationPointCalculationEvent)>
}
// utility function, used by Stakes, tests
pub fn from<T: ReadableAccount + StateMut<StakeState>>(account: &T) -> Option<StakeState> {
pub fn from<T: ReadableAccount + StateMut<StakeStateWithFlags>>(
account: &T,
) -> Option<StakeStateWithFlags> {
account.state().ok()
}
pub fn stake_from<T: ReadableAccount + StateMut<StakeState>>(account: &T) -> Option<Stake> {
from(account).and_then(|state: StakeState| state.stake())
pub fn stake_from<T: ReadableAccount + StateMut<StakeStateWithFlags>>(
account: &T,
) -> Option<Stake> {
from(account).and_then(|state: StakeStateWithFlags| state.stake())
}
pub fn delegation_from(account: &AccountSharedData) -> Option<Delegation> {
from(account).and_then(|state: StakeState| state.delegation())
from(account).and_then(|state: StakeStateWithFlags| state.delegation())
}
pub fn authorized_from(account: &AccountSharedData) -> Option<Authorized> {
from(account).and_then(|state: StakeState| state.authorized())
from(account).and_then(|state: StakeStateWithFlags| state.authorized())
}
pub fn lockup_from<T: ReadableAccount + StateMut<StakeState>>(account: &T) -> Option<Lockup> {
from(account).and_then(|state: StakeState| state.lockup())
pub fn lockup_from<T: ReadableAccount + StateMut<StakeStateWithFlags>>(
account: &T,
) -> Option<Lockup> {
from(account).and_then(|state: StakeStateWithFlags| state.lockup())
}
pub fn meta_from(account: &AccountSharedData) -> Option<Meta> {
from(account).and_then(|state: StakeState| state.meta())
from(account).and_then(|state: StakeStateWithFlags| state.meta())
}
pub(crate) fn new_warmup_cooldown_rate_epoch(invoke_context: &InvokeContext) -> Option<Epoch> {
@ -470,13 +476,13 @@ pub fn initialize(
lockup: &Lockup,
rent: &Rent,
) -> Result<(), InstructionError> {
if stake_account.get_data().len() != StakeState::size_of() {
if stake_account.get_data().len() != StakeStateWithFlags::size_of() {
return Err(InstructionError::InvalidAccountData);
}
if let StakeState::Uninitialized = stake_account.get_state()? {
if let StakeStateWithFlags::Uninitialized = stake_account.get_state()? {
let rent_exempt_reserve = rent.minimum_balance(stake_account.get_data().len());
if stake_account.get_lamports() >= rent_exempt_reserve {
stake_account.set_state(&StakeState::Initialized(Meta {
stake_account.set_state(&StakeStateWithFlags::Initialized(Meta {
rent_exempt_reserve,
authorized: *authorized,
lockup: *lockup,
@ -502,7 +508,7 @@ pub fn authorize(
custodian: Option<&Pubkey>,
) -> Result<(), InstructionError> {
match stake_account.get_state()? {
StakeState::Stake(mut meta, stake, stake_flags) => {
StakeStateWithFlags::Stake(mut meta, stake, stake_flags) => {
meta.authorized.authorize(
signers,
new_authority,
@ -513,9 +519,9 @@ pub fn authorize(
None
},
)?;
stake_account.set_state(&StakeState::Stake(meta, stake, stake_flags))
stake_account.set_state(&StakeStateWithFlags::Stake(meta, stake, stake_flags))
}
StakeState::Initialized(mut meta) => {
StakeStateWithFlags::Initialized(mut meta) => {
meta.authorized.authorize(
signers,
new_authority,
@ -526,7 +532,7 @@ pub fn authorize(
None
},
)?;
stake_account.set_state(&StakeState::Initialized(meta))
stake_account.set_state(&StakeStateWithFlags::Initialized(meta))
}
_ => Err(InstructionError::InvalidAccountData),
}
@ -593,7 +599,7 @@ pub fn delegate(
let mut stake_account = instruction_context
.try_borrow_instruction_account(transaction_context, stake_account_index)?;
match stake_account.get_state()? {
StakeState::Initialized(meta) => {
StakeStateWithFlags::Initialized(meta) => {
meta.authorized.check(signers, StakeAuthorize::Staker)?;
let ValidatedDelegatedInfo { stake_amount } =
validate_delegated_amount(&stake_account, &meta, feature_set)?;
@ -603,9 +609,13 @@ pub fn delegate(
&vote_state?.convert_to_current(),
clock.epoch,
);
stake_account.set_state(&StakeState::Stake(meta, stake, StakeFlags::empty()))
stake_account.set_state(&StakeStateWithFlags::Stake(
meta,
stake,
StakeFlags::empty(),
))
}
StakeState::Stake(meta, mut stake, stake_flags) => {
StakeStateWithFlags::Stake(meta, mut stake, stake_flags) => {
meta.authorized.check(signers, StakeAuthorize::Staker)?;
let ValidatedDelegatedInfo { stake_amount } =
validate_delegated_amount(&stake_account, &meta, feature_set)?;
@ -618,7 +628,7 @@ pub fn delegate(
clock,
stake_history,
)?;
stake_account.set_state(&StakeState::Stake(meta, stake, stake_flags))
stake_account.set_state(&StakeStateWithFlags::Stake(meta, stake, stake_flags))
}
_ => Err(InstructionError::InvalidAccountData),
}
@ -629,11 +639,11 @@ pub fn deactivate(
clock: &Clock,
signers: &HashSet<Pubkey>,
) -> Result<(), InstructionError> {
if let StakeState::Stake(meta, mut stake, stake_flags) = stake_account.get_state()? {
if let StakeStateWithFlags::Stake(meta, mut stake, stake_flags) = stake_account.get_state()? {
meta.authorized.check(signers, StakeAuthorize::Staker)?;
stake.deactivate(clock.epoch)?;
stake_account.set_state(&StakeState::Stake(meta, stake, stake_flags))
stake_account.set_state(&StakeStateWithFlags::Stake(meta, stake, stake_flags))
} else {
Err(InstructionError::InvalidAccountData)
}
@ -646,13 +656,13 @@ pub fn set_lockup(
clock: &Clock,
) -> Result<(), InstructionError> {
match stake_account.get_state()? {
StakeState::Initialized(mut meta) => {
StakeStateWithFlags::Initialized(mut meta) => {
meta.set_lockup(lockup, signers, clock)?;
stake_account.set_state(&StakeState::Initialized(meta))
stake_account.set_state(&StakeStateWithFlags::Initialized(meta))
}
StakeState::Stake(mut meta, stake, stake_flags) => {
StakeStateWithFlags::Stake(mut meta, stake, stake_flags) => {
meta.set_lockup(lockup, signers, clock)?;
stake_account.set_state(&StakeState::Stake(meta, stake, stake_flags))
stake_account.set_state(&StakeStateWithFlags::Stake(meta, stake, stake_flags))
}
_ => Err(InstructionError::InvalidAccountData),
}
@ -672,10 +682,10 @@ pub fn split(
if *split.get_owner() != id() {
return Err(InstructionError::IncorrectProgramId);
}
if split.get_data().len() != StakeState::size_of() {
if split.get_data().len() != StakeStateWithFlags::size_of() {
return Err(InstructionError::InvalidAccountData);
}
if !matches!(split.get_state()?, StakeState::Uninitialized) {
if !matches!(split.get_state()?, StakeStateWithFlags::Uninitialized) {
return Err(InstructionError::InvalidAccountData);
}
let split_lamport_balance = split.get_lamports();
@ -689,7 +699,7 @@ pub fn split(
drop(stake_account);
match stake_state {
StakeState::Stake(meta, mut stake, stake_flags) => {
StakeStateWithFlags::Stake(meta, mut stake, stake_flags) => {
meta.authorized.check(signers, StakeAuthorize::Staker)?;
let minimum_delegation = crate::get_minimum_delegation(&invoke_context.feature_set);
let validated_split_info = validate_split_amount(
@ -759,13 +769,17 @@ pub fn split(
let mut stake_account = instruction_context
.try_borrow_instruction_account(transaction_context, stake_account_index)?;
stake_account.set_state(&StakeState::Stake(meta, stake, stake_flags))?;
stake_account.set_state(&StakeStateWithFlags::Stake(meta, stake, stake_flags))?;
drop(stake_account);
let mut split = instruction_context
.try_borrow_instruction_account(transaction_context, split_index)?;
split.set_state(&StakeState::Stake(split_meta, split_stake, stake_flags))?;
split.set_state(&StakeStateWithFlags::Stake(
split_meta,
split_stake,
stake_flags,
))?;
}
StakeState::Initialized(meta) => {
StakeStateWithFlags::Initialized(meta) => {
meta.authorized.check(signers, StakeAuthorize::Staker)?;
let validated_split_info = validate_split_amount(
invoke_context,
@ -782,9 +796,9 @@ pub fn split(
split_meta.rent_exempt_reserve = validated_split_info.destination_rent_exempt_reserve;
let mut split = instruction_context
.try_borrow_instruction_account(transaction_context, split_index)?;
split.set_state(&StakeState::Initialized(split_meta))?;
split.set_state(&StakeStateWithFlags::Initialized(split_meta))?;
}
StakeState::Uninitialized => {
StakeStateWithFlags::Uninitialized => {
let stake_pubkey = transaction_context.get_key_of_account_at_index(
instruction_context
.get_index_of_instruction_account_in_transaction(stake_account_index)?,
@ -800,7 +814,7 @@ pub fn split(
let mut stake_account = instruction_context
.try_borrow_instruction_account(transaction_context, stake_account_index)?;
if lamports == stake_account.get_lamports() {
stake_account.set_state(&StakeState::Uninitialized)?;
stake_account.set_state(&StakeStateWithFlags::Uninitialized)?;
}
drop(stake_account);
@ -870,7 +884,7 @@ pub fn merge(
}
// Source is about to be drained, deinitialize its state
source_account.set_state(&StakeState::Uninitialized)?;
source_account.set_state(&StakeStateWithFlags::Uninitialized)?;
// Drain the source stake account
let lamports = source_account.get_lamports();
@ -902,18 +916,18 @@ pub fn redelegate(
);
return Err(InstructionError::IncorrectProgramId);
}
if uninitialized_stake_account.get_data().len() != StakeState::size_of() {
if uninitialized_stake_account.get_data().len() != StakeStateWithFlags::size_of() {
ic_msg!(
invoke_context,
"expected uninitialized stake account data len to be {}, not {}",
StakeState::size_of(),
StakeStateWithFlags::size_of(),
uninitialized_stake_account.get_data().len()
);
return Err(InstructionError::InvalidAccountData);
}
if !matches!(
uninitialized_stake_account.get_state()?,
StakeState::Uninitialized
StakeStateWithFlags::Uninitialized
) {
ic_msg!(
invoke_context,
@ -938,7 +952,7 @@ pub fn redelegate(
let vote_state = vote_account.get_state::<VoteStateVersions>()?;
let (stake_meta, effective_stake) =
if let StakeState::Stake(meta, stake, _stake_flags) = stake_account.get_state()? {
if let StakeStateWithFlags::Stake(meta, stake, _stake_flags) = stake_account.get_state()? {
let stake_history = invoke_context.get_sysvar_cache().get_stake_history()?;
let status = stake.delegation.stake_activating_and_deactivating(
clock.epoch,
@ -987,7 +1001,7 @@ pub fn redelegate(
&uninitialized_stake_meta,
&invoke_context.feature_set,
)?;
uninitialized_stake_account.set_state(&StakeState::Stake(
uninitialized_stake_account.set_state(&StakeStateWithFlags::Stake(
uninitialized_stake_meta,
new_stake(
stake_amount,
@ -1027,7 +1041,7 @@ pub fn withdraw(
let mut stake_account = instruction_context
.try_borrow_instruction_account(transaction_context, stake_account_index)?;
let (lockup, reserve, is_staked) = match stake_account.get_state()? {
StakeState::Stake(meta, stake, _stake_flag) => {
StakeStateWithFlags::Stake(meta, stake, _stake_flag) => {
meta.authorized
.check(&signers, StakeAuthorize::Withdrawer)?;
// if we have a deactivation epoch and we're in cooldown
@ -1045,13 +1059,13 @@ pub fn withdraw(
let staked_and_reserve = checked_add(staked, meta.rent_exempt_reserve)?;
(meta.lockup, staked_and_reserve, staked != 0)
}
StakeState::Initialized(meta) => {
StakeStateWithFlags::Initialized(meta) => {
meta.authorized
.check(&signers, StakeAuthorize::Withdrawer)?;
// stake accounts must have a balance >= rent_exempt_reserve
(meta.lockup, meta.rent_exempt_reserve, false)
}
StakeState::Uninitialized => {
StakeStateWithFlags::Uninitialized => {
if !signers.contains(stake_account.get_key()) {
return Err(InstructionError::MissingRequiredSignature);
}
@ -1097,7 +1111,7 @@ pub fn withdraw(
// Deinitialize state upon zero balance
if lamports == stake_account.get_lamports() {
stake_account.set_state(&StakeState::Uninitialized)?;
stake_account.set_state(&StakeStateWithFlags::Uninitialized)?;
}
stake_account.checked_sub_lamports(lamports)?;
@ -1142,7 +1156,7 @@ pub(crate) fn deactivate_delinquent(
return Err(StakeError::InsufficientReferenceVotes.into());
}
if let StakeState::Stake(meta, mut stake, stake_flags) = stake_account.get_state()? {
if let StakeStateWithFlags::Stake(meta, mut stake, stake_flags) = stake_account.get_state()? {
if stake.delegation.voter_pubkey != *delinquent_vote_account_pubkey {
return Err(StakeError::VoteAddressMismatch.into());
}
@ -1151,7 +1165,7 @@ pub(crate) fn deactivate_delinquent(
// voted in the last `MINIMUM_DELINQUENT_EPOCHS_FOR_DEACTIVATION`
if eligible_for_deactivate_delinquent(&delinquent_vote_state.epoch_credits, current_epoch) {
stake.deactivate(current_epoch)?;
stake_account.set_state(&StakeState::Stake(meta, stake, stake_flags))
stake_account.set_state(&StakeStateWithFlags::Stake(meta, stake, stake_flags))
} else {
Err(StakeError::MinimumDelinquentEpochsForDeactivationNotMet.into())
}
@ -1312,13 +1326,13 @@ impl MergeKind {
fn get_if_mergeable(
invoke_context: &InvokeContext,
stake_state: &StakeState,
stake_state: &StakeStateWithFlags,
stake_lamports: u64,
clock: &Clock,
stake_history: &StakeHistory,
) -> Result<Self, InstructionError> {
match stake_state {
StakeState::Stake(meta, stake, stake_flags) => {
StakeStateWithFlags::Stake(meta, stake, stake_flags) => {
// stake must not be in a transient state. Transient here meaning
// activating or deactivating with non-zero effective stake.
let status = stake.delegation.stake_activating_and_deactivating(
@ -1338,7 +1352,7 @@ impl MergeKind {
}
}
}
StakeState::Initialized(meta) => {
StakeStateWithFlags::Initialized(meta) => {
Ok(Self::Inactive(*meta, stake_lamports, StakeFlags::empty()))
}
_ => Err(InstructionError::InvalidAccountData),
@ -1412,7 +1426,7 @@ impl MergeKind {
invoke_context: &InvokeContext,
source: Self,
clock: &Clock,
) -> Result<Option<StakeState>, InstructionError> {
) -> Result<Option<StakeStateWithFlags>, InstructionError> {
Self::metas_can_merge(invoke_context, self.meta(), source.meta(), clock)?;
self.active_stake()
.zip(source.active_stake())
@ -1439,7 +1453,7 @@ impl MergeKind {
Self::Inactive(_, source_lamports, source_stake_flags),
) => {
stake.delegation.stake = checked_add(stake.delegation.stake, source_lamports)?;
Some(StakeState::Stake(
Some(StakeStateWithFlags::Stake(
meta,
stake,
stake_flags.union(source_stake_flags),
@ -1459,7 +1473,7 @@ impl MergeKind {
source_lamports,
source_stake.credits_observed,
)?;
Some(StakeState::Stake(
Some(StakeStateWithFlags::Stake(
meta,
stake,
stake_flags.union(source_stake_flags),
@ -1476,7 +1490,7 @@ impl MergeKind {
source_stake.delegation.stake,
source_stake.credits_observed,
)?;
Some(StakeState::Stake(meta, stake, StakeFlags::empty()))
Some(StakeStateWithFlags::Stake(meta, stake, StakeFlags::empty()))
}
_ => return Err(StakeError::MergeMismatch.into()),
};
@ -1555,7 +1569,7 @@ fn stake_weighted_credits_observed(
#[doc(hidden)]
pub fn redeem_rewards(
rewarded_epoch: Epoch,
stake_state: StakeState,
stake_state: StakeStateWithFlags,
stake_account: &mut AccountSharedData,
vote_state: &VoteState,
point_value: &PointValue,
@ -1563,7 +1577,7 @@ pub fn redeem_rewards(
inflation_point_calc_tracer: Option<impl Fn(&InflationPointCalculationEvent)>,
new_rate_activation_epoch: Option<Epoch>,
) -> Result<(u64, u64), InstructionError> {
if let StakeState::Stake(meta, mut stake, stake_flags) = stake_state {
if let StakeStateWithFlags::Stake(meta, mut stake, stake_flags) = stake_state {
if let Some(inflation_point_calc_tracer) = inflation_point_calc_tracer.as_ref() {
inflation_point_calc_tracer(
&InflationPointCalculationEvent::EffectiveStakeAtRewardedEpoch(stake.stake(
@ -1590,7 +1604,7 @@ pub fn redeem_rewards(
new_rate_activation_epoch,
) {
stake_account.checked_add_lamports(stakers_reward)?;
stake_account.set_state(&StakeState::Stake(meta, stake, stake_flags))?;
stake_account.set_state(&StakeStateWithFlags::Stake(meta, stake, stake_flags))?;
Ok((stakers_reward, voters_reward))
} else {
@ -1604,12 +1618,12 @@ pub fn redeem_rewards(
// utility function, used by runtime
#[doc(hidden)]
pub fn calculate_points(
stake_state: &StakeState,
stake_state: &StakeStateWithFlags,
vote_state: &VoteState,
stake_history: Option<&StakeHistory>,
new_rate_activation_epoch: Option<Epoch>,
) -> Result<u128, InstructionError> {
if let StakeState::Stake(_meta, stake, _stake_flags) = stake_state {
if let StakeStateWithFlags::Stake(_meta, stake, _stake_flags) = stake_state {
Ok(calculate_stake_points(
stake,
vote_state,
@ -1678,7 +1692,7 @@ pub fn create_lockup_stake_account(
rent: &Rent,
lamports: u64,
) -> AccountSharedData {
let mut stake_account = AccountSharedData::new(lamports, StakeState::size_of(), &id());
let mut stake_account = AccountSharedData::new(lamports, StakeStateWithFlags::size_of(), &id());
let rent_exempt_reserve = rent.minimum_balance(stake_account.data().len());
assert!(
@ -1687,7 +1701,7 @@ pub fn create_lockup_stake_account(
);
stake_account
.set_state(&StakeState::Initialized(Meta {
.set_state(&StakeStateWithFlags::Initialized(Meta {
authorized: *authorized,
lockup: *lockup,
rent_exempt_reserve,
@ -1742,14 +1756,14 @@ fn do_create_account(
lamports: u64,
activation_epoch: Epoch,
) -> AccountSharedData {
let mut stake_account = AccountSharedData::new(lamports, StakeState::size_of(), &id());
let mut stake_account = AccountSharedData::new(lamports, StakeStateWithFlags::size_of(), &id());
let vote_state = vote_state::from(vote_account).expect("vote_state");
let rent_exempt_reserve = rent.minimum_balance(stake_account.data().len());
stake_account
.set_state(&StakeState::Stake(
.set_state(&StakeStateWithFlags::Stake(
Meta {
authorized: Authorized::auto(authorized),
rent_exempt_reserve,
@ -1925,10 +1939,10 @@ mod tests {
#[test]
fn test_stake_state_stake_from_fail() {
let mut stake_account = AccountSharedData::new(0, StakeState::size_of(), &id());
let mut stake_account = AccountSharedData::new(0, StakeStateWithFlags::size_of(), &id());
stake_account
.set_state(&StakeState::default())
.set_state(&StakeStateWithFlags::default())
.expect("set_state");
assert_eq!(stake_from(&stake_account), None);
@ -2985,7 +2999,7 @@ mod tests {
#[ignore]
#[should_panic]
fn test_dbg_stake_minimum_balance() {
let minimum_balance = Rent::default().minimum_balance(StakeState::size_of());
let minimum_balance = Rent::default().minimum_balance(StakeStateWithFlags::size_of());
panic!(
"stake minimum_balance: {} lamports, {} SOL",
minimum_balance,
@ -3230,7 +3244,7 @@ mod tests {
let authority_pubkey = Pubkey::new_unique();
let initial_lamports = 4242424242;
let rent = Rent::default();
let rent_exempt_reserve = rent.minimum_balance(StakeState::size_of());
let rent_exempt_reserve = rent.minimum_balance(StakeStateWithFlags::size_of());
let stake_lamports = rent_exempt_reserve + initial_lamports;
let new_rate_activation_epoch = Some(0);
@ -3240,8 +3254,8 @@ mod tests {
};
let mut stake_account = AccountSharedData::new_data_with_space(
stake_lamports,
&StakeState::Uninitialized,
StakeState::size_of(),
&StakeStateWithFlags::Uninitialized,
StakeStateWithFlags::size_of(),
&id(),
)
.expect("stake_account");
@ -3262,7 +3276,9 @@ mod tests {
);
// RewardsPool state fails
stake_account.set_state(&StakeState::RewardsPool).unwrap();
stake_account
.set_state(&StakeStateWithFlags::RewardsPool)
.unwrap();
assert_eq!(
MergeKind::get_if_mergeable(
&invoke_context,
@ -3277,7 +3293,7 @@ mod tests {
// Initialized state succeeds
stake_account
.set_state(&StakeState::Initialized(meta))
.set_state(&StakeStateWithFlags::Initialized(meta))
.unwrap();
assert_eq!(
MergeKind::get_if_mergeable(
@ -3325,7 +3341,11 @@ mod tests {
..Stake::default()
};
stake_account
.set_state(&StakeState::Stake(meta, stake, StakeFlags::empty()))
.set_state(&StakeStateWithFlags::Stake(
meta,
stake,
StakeFlags::empty(),
))
.unwrap();
// activation_epoch succeeds
assert_eq!(

View File

@ -68,7 +68,7 @@ use {
message::SanitizedMessage,
pubkey::{Pubkey, PUBKEY_BYTES},
signature::{Keypair, Signature, Signer},
stake::state::{StakeActivationStatus, StakeState},
stake::state::{StakeActivationStatus, StakeStateWithFlags},
stake_history::StakeHistory,
system_instruction,
sysvar::stake_history,
@ -1731,7 +1731,7 @@ impl JsonRpcRequestProcessor {
let stake_account = bank
.get_account(pubkey)
.ok_or_else(|| Error::invalid_params("Invalid param: account not found".to_string()))?;
let stake_state: StakeState = stake_account
let stake_state: StakeStateWithFlags = stake_account
.state()
.map_err(|_| Error::invalid_params("Invalid param: not a stake account".to_string()))?;
let delegation = stake_state.delegation();

View File

@ -635,7 +635,7 @@ mod tests {
signature::{Keypair, Signer},
stake::{
self, instruction as stake_instruction,
state::{Authorized, Lockup, StakeAuthorize, StakeState},
state::{Authorized, Lockup, StakeAuthorize, StakeStateWithFlags},
},
system_instruction, system_program, system_transaction,
transaction::{self, Transaction},
@ -907,7 +907,7 @@ mod tests {
let balance = {
let bank = bank_forks.read().unwrap().working_bank();
let rent = &bank.rent_collector().rent;
rent.minimum_balance(StakeState::size_of())
rent.minimum_balance(StakeStateWithFlags::size_of())
};
let tx = system_transaction::transfer(&alice, &from.pubkey(), balance, blockhash);

View File

@ -174,7 +174,7 @@ use {
},
},
solana_stake_program::stake_state::{
self, InflationPointCalculationEvent, PointValue, StakeState,
self, InflationPointCalculationEvent, PointValue, StakeStateWithFlags,
},
solana_system_program::{get_system_account_kind, SystemAccountKind},
solana_vote_program::vote_state::VoteState,
@ -3122,7 +3122,7 @@ impl Bank {
let delegation = stake_account.delegation();
let (mut stake_account, stake_state) =
<(AccountSharedData, StakeState)>::from(stake_account);
<(AccountSharedData, StakeStateWithFlags)>::from(stake_account);
let vote_pubkey = delegation.voter_pubkey;
let Some(vote_account) = get_vote_account(&vote_pubkey) else {
return None;
@ -3253,7 +3253,7 @@ impl Bank {
}
});
let (mut stake_account, stake_state) =
<(AccountSharedData, StakeState)>::from(stake_account);
<(AccountSharedData, StakeStateWithFlags)>::from(stake_account);
let redeemed = stake_state::redeem_rewards(
rewarded_epoch,
stake_state,

View File

@ -96,7 +96,7 @@ use {
},
transaction_context::{TransactionAccount, TransactionContext},
},
solana_stake_program::stake_state::{self, StakeState},
solana_stake_program::stake_state::{self, StakeStateWithFlags},
solana_vote_program::{
vote_instruction,
vote_state::{
@ -4417,7 +4417,7 @@ fn test_bank_cloned_stake_delegations() {
let (vote_balance, stake_balance) = {
let rent = &bank.rent_collector().rent;
let vote_rent_exempt_reserve = rent.minimum_balance(VoteState::size_of());
let stake_rent_exempt_reserve = rent.minimum_balance(StakeState::size_of());
let stake_rent_exempt_reserve = rent.minimum_balance(StakeStateWithFlags::size_of());
let minimum_delegation = solana_stake_program::get_minimum_delegation(&bank.feature_set);
(
vote_rent_exempt_reserve,

View File

@ -10,7 +10,7 @@ use {
pubkey::Pubkey,
rent::Rent,
signature::{Keypair, Signer},
stake::state::StakeState,
stake::state::StakeStateWithFlags,
system_program,
},
solana_stake_program::stake_state,
@ -23,7 +23,7 @@ const VALIDATOR_LAMPORTS: u64 = 42;
// fun fact: rustc is very close to make this const fn.
pub fn bootstrap_validator_stake_lamports() -> u64 {
Rent::default().minimum_balance(StakeState::size_of())
Rent::default().minimum_balance(StakeStateWithFlags::size_of())
}
// Number of lamports automatically used for genesis accounts

View File

@ -7,7 +7,7 @@ use {
solana_sdk::{
account::ReadableAccount,
pubkey::Pubkey,
stake::{self, state::StakeState},
stake::{self, state::StakeStateWithFlags},
},
solana_stake_program::stake_state,
std::{collections::HashSet, sync::Arc},
@ -53,14 +53,14 @@ pub fn calculate_non_circulating_supply(bank: &Arc<Bank>) -> ScanResult<NonCircu
for (pubkey, account) in stake_accounts.iter() {
let stake_account = stake_state::from(account).unwrap_or_default();
match stake_account {
StakeState::Initialized(meta) => {
StakeStateWithFlags::Initialized(meta) => {
if meta.lockup.is_in_force(&clock, None)
|| withdraw_authority_list.contains(&meta.authorized.withdrawer)
{
non_circulating_accounts_set.insert(*pubkey);
}
}
StakeState::Stake(meta, _stake, _stake_flags) => {
StakeStateWithFlags::Stake(meta, _stake, _stake_flags) => {
if meta.lockup.is_in_force(&clock, None)
|| withdraw_authority_list.contains(&meta.authorized.withdrawer)
{
@ -264,8 +264,8 @@ mod tests {
};
let stake_account = Account::new_data_with_space(
balance,
&StakeState::Initialized(meta),
StakeState::size_of(),
&StakeStateWithFlags::Initialized(meta),
StakeStateWithFlags::size_of(),
&stake::program::id(),
)
.unwrap();

View File

@ -6,7 +6,7 @@ use {
account_utils::StateMut,
instruction::InstructionError,
pubkey::Pubkey,
stake::state::{Delegation, StakeState},
stake::state::{Delegation, StakeStateWithFlags},
},
std::marker::PhantomData,
thiserror::Error,
@ -19,7 +19,7 @@ use {
#[derive(Clone, Debug, Default)]
pub struct StakeAccount<T> {
account: AccountSharedData,
stake_state: StakeState,
stake_state: StakeStateWithFlags,
_phantom: PhantomData<T>,
}
@ -29,7 +29,7 @@ pub enum Error {
#[error(transparent)]
InstructionError(#[from] InstructionError),
#[error("Invalid delegation: {0:?}")]
InvalidDelegation(Box<StakeState>),
InvalidDelegation(Box<StakeStateWithFlags>),
#[error("Invalid stake account owner: {0}")]
InvalidOwner(/*owner:*/ Pubkey),
}
@ -41,7 +41,7 @@ impl<T> StakeAccount<T> {
}
#[inline]
pub(crate) fn stake_state(&self) -> &StakeState {
pub(crate) fn stake_state(&self) -> &StakeStateWithFlags {
&self.stake_state
}
}
@ -61,7 +61,7 @@ impl TryFrom<AccountSharedData> for StakeAccount<Delegation> {
if account.owner() != &solana_stake_program::id() {
return Err(Error::InvalidOwner(*account.owner()));
}
let stake_state: StakeState = account.state()?;
let stake_state: StakeStateWithFlags = account.state()?;
if stake_state.delegation().is_none() {
return Err(Error::InvalidDelegation(Box::new(stake_state)));
}
@ -73,7 +73,7 @@ impl TryFrom<AccountSharedData> for StakeAccount<Delegation> {
}
}
impl<T> From<StakeAccount<T>> for (AccountSharedData, StakeState) {
impl<T> From<StakeAccount<T>> for (AccountSharedData, StakeStateWithFlags) {
#[inline]
fn from(stake_account: StakeAccount<T>) -> Self {
(stake_account.account, stake_account.stake_state)
@ -102,7 +102,7 @@ impl AbiExample for StakeAccount<Delegation> {
},
};
let stake_state =
StakeState::Stake(Meta::example(), Stake::example(), StakeFlags::example());
StakeStateWithFlags::Stake(Meta::example(), Stake::example(), StakeFlags::example());
let mut account = Account::example();
account.data.resize(200, 0u8);
account.owner = solana_stake_program::id();

View File

@ -179,7 +179,7 @@ impl StakesCache {
/// [`Stakes<Delegation>`] is equivalent to the old code and is used for backward
/// compatibility in [`crate::bank::BankFieldsToDeserialize`].
/// But banks cache [`Stakes<StakeAccount>`] which includes the entire stake
/// account and StakeState deserialized from the account. Doing so, will remove
/// account and StakeStateWithFlags deserialized from the account. Doing so, will remove
/// the need to load the stake account from accounts-db when working with
/// stake-delegations.
#[derive(Default, Clone, PartialEq, Debug, Deserialize, Serialize, AbiExample)]

View File

@ -19,7 +19,7 @@ use {
signature::{Keypair, Signer},
stake::{
self, instruction as stake_instruction,
state::{Authorized, Lockup, StakeState},
state::{Authorized, Lockup, StakeStateWithFlags},
},
sysvar::{self, stake_history::StakeHistory},
},
@ -145,7 +145,7 @@ fn test_stake_create_and_split_single_signature() {
let lamports = {
let rent = &bank.rent_collector().rent;
let rent_exempt_reserve = rent.minimum_balance(StakeState::size_of());
let rent_exempt_reserve = rent.minimum_balance(StakeStateWithFlags::size_of());
let minimum_delegation = solana_stake_program::get_minimum_delegation(&bank.feature_set);
2 * (rent_exempt_reserve + minimum_delegation)
};
@ -221,7 +221,7 @@ fn test_stake_create_and_split_to_existing_system_account() {
let lamports = {
let rent = &bank.rent_collector().rent;
let rent_exempt_reserve = rent.minimum_balance(StakeState::size_of());
let rent_exempt_reserve = rent.minimum_balance(StakeStateWithFlags::size_of());
let minimum_delegation = solana_stake_program::get_minimum_delegation(&bank.feature_set);
2 * (rent_exempt_reserve + minimum_delegation)
};
@ -314,7 +314,7 @@ fn test_stake_account_lifetime() {
let rent = &bank.rent_collector().rent;
(
rent.minimum_balance(VoteState::size_of()),
rent.minimum_balance(StakeState::size_of()),
rent.minimum_balance(StakeStateWithFlags::size_of()),
solana_stake_program::get_minimum_delegation(&bank.feature_set),
)
};
@ -367,7 +367,7 @@ fn test_stake_account_lifetime() {
// Test that correct lamports are staked
let account = bank.get_account(&stake_pubkey).expect("account not found");
let stake_state = account.state().expect("couldn't unpack account data");
if let StakeState::Stake(_meta, stake, _stake_flags) = stake_state {
if let StakeStateWithFlags::Stake(_meta, stake, _stake_flags) = stake_state {
assert_eq!(stake.delegation.stake, stake_starting_delegation,);
} else {
panic!("wrong account type found")
@ -391,7 +391,7 @@ fn test_stake_account_lifetime() {
// Test that lamports are still staked
let account = bank.get_account(&stake_pubkey).expect("account not found");
let stake_state = account.state().expect("couldn't unpack account data");
if let StakeState::Stake(_meta, stake, _stake_flags) = stake_state {
if let StakeStateWithFlags::Stake(_meta, stake, _stake_flags) = stake_state {
assert_eq!(stake.delegation.stake, stake_starting_delegation,);
} else {
panic!("wrong account type found")
@ -615,7 +615,7 @@ fn test_create_stake_account_from_seed() {
let authorized = Authorized::auto(&mint_pubkey);
let (balance, delegation) = {
let rent = &bank.rent_collector().rent;
let rent_exempt_reserve = rent.minimum_balance(StakeState::size_of());
let rent_exempt_reserve = rent.minimum_balance(StakeStateWithFlags::size_of());
let minimum_delegation = solana_stake_program::get_minimum_delegation(&bank.feature_set);
(rent_exempt_reserve + minimum_delegation, minimum_delegation)
};
@ -641,7 +641,7 @@ fn test_create_stake_account_from_seed() {
// Test that correct lamports are staked
let account = bank.get_account(&stake_pubkey).expect("account not found");
let stake_state = account.state().expect("couldn't unpack account data");
if let StakeState::Stake(_meta, stake, _) = stake_state {
if let StakeStateWithFlags::Stake(_meta, stake, _) = stake_state {
assert_eq!(stake.delegation.stake, delegation);
} else {
panic!("wrong account type found")

View File

@ -8,7 +8,7 @@ use {
pubkey::Pubkey,
stake::{
program::id,
state::{Authorized, Lockup, StakeAuthorize, StakeState},
state::{Authorized, Lockup, StakeAuthorize, StakeStateWithFlags},
},
system_instruction, sysvar,
},
@ -364,7 +364,7 @@ pub fn create_account_with_seed(
base,
seed,
lamports,
StakeState::size_of() as u64,
StakeStateWithFlags::size_of() as u64,
&id(),
),
initialize(stake_pubkey, authorized, lockup),
@ -383,7 +383,7 @@ pub fn create_account(
from_pubkey,
stake_pubkey,
lamports,
StakeState::size_of() as u64,
StakeStateWithFlags::size_of() as u64,
&id(),
),
initialize(stake_pubkey, authorized, lockup),
@ -405,7 +405,7 @@ pub fn create_account_with_seed_checked(
base,
seed,
lamports,
StakeState::size_of() as u64,
StakeStateWithFlags::size_of() as u64,
&id(),
),
initialize_checked(stake_pubkey, authorized),
@ -423,7 +423,7 @@ pub fn create_account_checked(
from_pubkey,
stake_pubkey,
lamports,
StakeState::size_of() as u64,
StakeStateWithFlags::size_of() as u64,
&id(),
),
initialize_checked(stake_pubkey, authorized),
@ -452,7 +452,7 @@ pub fn split(
split_stake_pubkey: &Pubkey,
) -> Vec<Instruction> {
vec![
system_instruction::allocate(split_stake_pubkey, StakeState::size_of() as u64),
system_instruction::allocate(split_stake_pubkey, StakeStateWithFlags::size_of() as u64),
system_instruction::assign(split_stake_pubkey, &id()),
_split(
stake_pubkey,
@ -476,7 +476,7 @@ pub fn split_with_seed(
split_stake_pubkey,
base,
seed,
StakeState::size_of() as u64,
StakeStateWithFlags::size_of() as u64,
&id(),
),
_split(
@ -796,7 +796,10 @@ pub fn redelegate(
uninitialized_stake_pubkey: &Pubkey,
) -> Vec<Instruction> {
vec![
system_instruction::allocate(uninitialized_stake_pubkey, StakeState::size_of() as u64),
system_instruction::allocate(
uninitialized_stake_pubkey,
StakeStateWithFlags::size_of() as u64,
),
system_instruction::assign(uninitialized_stake_pubkey, &id()),
_redelegate(
stake_pubkey,
@ -820,7 +823,7 @@ pub fn redelegate_with_seed(
uninitialized_stake_pubkey,
base,
seed,
StakeState::size_of() as u64,
StakeStateWithFlags::size_of() as u64,
&id(),
),
_redelegate(

View File

@ -1,4 +1,7 @@
#![allow(clippy::integer_arithmetic)]
// Remove the following `allow` when `StakeState` is removed, required to avoid
// warnings from uses of deprecated types during trait derivations.
#![allow(deprecated)]
use {
crate::{
@ -33,14 +36,17 @@ pub fn warmup_cooldown_rate(current_epoch: Epoch, new_rate_activation_epoch: Opt
#[derive(Debug, Default, Serialize, Deserialize, PartialEq, Clone, Copy, AbiExample)]
#[allow(clippy::large_enum_variant)]
#[deprecated(
since = "1.17.0",
note = "Please use `StakeStateWithFlags` instead, and match the third `StakeFlags` field when matching `StakeStateWithFlags::Stake` to resolve any breakage. For example, `if let StakeState::Stake(meta, stake)` becomes `if let StakeStateWithFlags::Stake(meta, stake, _stake_flags)`."
)]
pub enum StakeState {
#[default]
Uninitialized,
Initialized(Meta),
Stake(Meta, Stake, StakeFlags),
Stake(Meta, Stake),
RewardsPool,
}
impl BorshDeserialize for StakeState {
fn deserialize_reader<R: io::Read>(reader: &mut R) -> io::Result<Self> {
let enum_value = u32::deserialize_reader(reader)?;
@ -53,8 +59,7 @@ impl BorshDeserialize for StakeState {
2 => {
let meta: Meta = BorshDeserialize::deserialize_reader(reader)?;
let stake: Stake = BorshDeserialize::deserialize_reader(reader)?;
let stake_flags: StakeFlags = BorshDeserialize::deserialize_reader(reader)?;
Ok(StakeState::Stake(meta, stake, stake_flags))
Ok(StakeState::Stake(meta, stake))
}
3 => Ok(StakeState::RewardsPool),
_ => Err(io::Error::new(
@ -64,7 +69,6 @@ impl BorshDeserialize for StakeState {
}
}
}
impl BorshSerialize for StakeState {
fn serialize<W: io::Write>(&self, writer: &mut W) -> io::Result<()> {
match self {
@ -73,17 +77,15 @@ impl BorshSerialize for StakeState {
writer.write_all(&1u32.to_le_bytes())?;
meta.serialize(writer)
}
StakeState::Stake(meta, stake, stake_flags) => {
StakeState::Stake(meta, stake) => {
writer.write_all(&2u32.to_le_bytes())?;
meta.serialize(writer)?;
stake.serialize(writer)?;
stake_flags.serialize(writer)
stake.serialize(writer)
}
StakeState::RewardsPool => writer.write_all(&3u32.to_le_bytes()),
}
}
}
impl StakeState {
/// The fixed number of bytes used to serialize each stake account
pub const fn size_of() -> usize {
@ -92,21 +94,21 @@ impl StakeState {
pub fn stake(&self) -> Option<Stake> {
match self {
StakeState::Stake(_meta, stake, _stake_flags) => Some(*stake),
StakeState::Stake(_meta, stake) => Some(*stake),
_ => None,
}
}
pub fn delegation(&self) -> Option<Delegation> {
match self {
StakeState::Stake(_meta, stake, _stake_flags) => Some(stake.delegation),
StakeState::Stake(_meta, stake) => Some(stake.delegation),
_ => None,
}
}
pub fn authorized(&self) -> Option<Authorized> {
match self {
StakeState::Stake(meta, _stake, _stake_flags) => Some(meta.authorized),
StakeState::Stake(meta, _stake) => Some(meta.authorized),
StakeState::Initialized(meta) => Some(meta.authorized),
_ => None,
}
@ -118,13 +120,107 @@ impl StakeState {
pub fn meta(&self) -> Option<Meta> {
match self {
StakeState::Stake(meta, _stake, _stake_flags) => Some(*meta),
StakeState::Stake(meta, _stake) => Some(*meta),
StakeState::Initialized(meta) => Some(*meta),
_ => None,
}
}
}
#[derive(Debug, Default, Serialize, Deserialize, PartialEq, Clone, Copy, AbiExample)]
#[allow(clippy::large_enum_variant)]
pub enum StakeStateWithFlags {
#[default]
Uninitialized,
Initialized(Meta),
Stake(Meta, Stake, StakeFlags),
RewardsPool,
}
impl BorshDeserialize for StakeStateWithFlags {
fn deserialize_reader<R: io::Read>(reader: &mut R) -> io::Result<Self> {
let enum_value = u32::deserialize_reader(reader)?;
match enum_value {
0 => Ok(StakeStateWithFlags::Uninitialized),
1 => {
let meta = Meta::deserialize_reader(reader)?;
Ok(StakeStateWithFlags::Initialized(meta))
}
2 => {
let meta: Meta = BorshDeserialize::deserialize_reader(reader)?;
let stake: Stake = BorshDeserialize::deserialize_reader(reader)?;
let stake_flags: StakeFlags = BorshDeserialize::deserialize_reader(reader)?;
Ok(StakeStateWithFlags::Stake(meta, stake, stake_flags))
}
3 => Ok(StakeStateWithFlags::RewardsPool),
_ => Err(io::Error::new(
io::ErrorKind::InvalidData,
"Invalid enum value",
)),
}
}
}
impl BorshSerialize for StakeStateWithFlags {
fn serialize<W: io::Write>(&self, writer: &mut W) -> io::Result<()> {
match self {
StakeStateWithFlags::Uninitialized => writer.write_all(&0u32.to_le_bytes()),
StakeStateWithFlags::Initialized(meta) => {
writer.write_all(&1u32.to_le_bytes())?;
meta.serialize(writer)
}
StakeStateWithFlags::Stake(meta, stake, stake_flags) => {
writer.write_all(&2u32.to_le_bytes())?;
meta.serialize(writer)?;
stake.serialize(writer)?;
stake_flags.serialize(writer)
}
StakeStateWithFlags::RewardsPool => writer.write_all(&3u32.to_le_bytes()),
}
}
}
impl StakeStateWithFlags {
/// The fixed number of bytes used to serialize each stake account
pub const fn size_of() -> usize {
200 // see test_size_of
}
pub fn stake(&self) -> Option<Stake> {
match self {
StakeStateWithFlags::Stake(_meta, stake, _stake_flags) => Some(*stake),
_ => None,
}
}
pub fn delegation(&self) -> Option<Delegation> {
match self {
StakeStateWithFlags::Stake(_meta, stake, _stake_flags) => Some(stake.delegation),
_ => None,
}
}
pub fn authorized(&self) -> Option<Authorized> {
match self {
StakeStateWithFlags::Stake(meta, _stake, _stake_flags) => Some(meta.authorized),
StakeStateWithFlags::Initialized(meta) => Some(meta.authorized),
_ => None,
}
}
pub fn lockup(&self) -> Option<Lockup> {
self.meta().map(|meta| meta.lockup)
}
pub fn meta(&self) -> Option<Meta> {
match self {
StakeStateWithFlags::Stake(meta, _stake, _stake_flags) => Some(*meta),
StakeStateWithFlags::Initialized(meta) => Some(*meta),
_ => None,
}
}
}
#[derive(Debug, Serialize, Deserialize, PartialEq, Eq, Clone, Copy, AbiExample)]
pub enum StakeAuthorize {
Staker,
@ -615,6 +711,123 @@ mod test {
bincode::serialize,
};
fn check_borsh_deserialization(stake: StakeStateWithFlags) {
let serialized = serialize(&stake).unwrap();
let deserialized = StakeStateWithFlags::try_from_slice(&serialized).unwrap();
assert_eq!(stake, deserialized);
}
fn check_borsh_serialization(stake: StakeStateWithFlags) {
let bincode_serialized = serialize(&stake).unwrap();
let borsh_serialized = StakeStateWithFlags::try_to_vec(&stake).unwrap();
assert_eq!(bincode_serialized, borsh_serialized);
}
#[test]
fn test_size_of() {
assert_eq!(
StakeStateWithFlags::size_of(),
std::mem::size_of::<StakeStateWithFlags>()
);
}
#[test]
fn bincode_vs_borsh_deserialization() {
check_borsh_deserialization(StakeStateWithFlags::Uninitialized);
check_borsh_deserialization(StakeStateWithFlags::RewardsPool);
check_borsh_deserialization(StakeStateWithFlags::Initialized(Meta {
rent_exempt_reserve: u64::MAX,
authorized: Authorized {
staker: Pubkey::new_unique(),
withdrawer: Pubkey::new_unique(),
},
lockup: Lockup::default(),
}));
check_borsh_deserialization(StakeStateWithFlags::Stake(
Meta {
rent_exempt_reserve: 1,
authorized: Authorized {
staker: Pubkey::new_unique(),
withdrawer: Pubkey::new_unique(),
},
lockup: Lockup::default(),
},
Stake {
delegation: Delegation {
voter_pubkey: Pubkey::new_unique(),
stake: u64::MAX,
activation_epoch: Epoch::MAX,
deactivation_epoch: Epoch::MAX,
..Delegation::default()
},
credits_observed: 1,
},
StakeFlags::empty(),
));
}
#[test]
fn bincode_vs_borsh_serialization() {
check_borsh_serialization(StakeStateWithFlags::Uninitialized);
check_borsh_serialization(StakeStateWithFlags::RewardsPool);
check_borsh_serialization(StakeStateWithFlags::Initialized(Meta {
rent_exempt_reserve: u64::MAX,
authorized: Authorized {
staker: Pubkey::new_unique(),
withdrawer: Pubkey::new_unique(),
},
lockup: Lockup::default(),
}));
check_borsh_serialization(StakeStateWithFlags::Stake(
Meta {
rent_exempt_reserve: 1,
authorized: Authorized {
staker: Pubkey::new_unique(),
withdrawer: Pubkey::new_unique(),
},
lockup: Lockup::default(),
},
Stake {
delegation: Delegation {
voter_pubkey: Pubkey::new_unique(),
stake: u64::MAX,
activation_epoch: Epoch::MAX,
deactivation_epoch: Epoch::MAX,
..Default::default()
},
credits_observed: 1,
},
StakeFlags::MUST_FULLY_ACTIVATE_BEFORE_DEACTIVATION_IS_PERMITTED,
));
}
#[test]
fn borsh_deserialization_live_data() {
let data = [
1, 0, 0, 0, 128, 213, 34, 0, 0, 0, 0, 0, 133, 0, 79, 231, 141, 29, 73, 61, 232, 35,
119, 124, 168, 12, 120, 216, 195, 29, 12, 166, 139, 28, 36, 182, 186, 154, 246, 149,
224, 109, 52, 100, 133, 0, 79, 231, 141, 29, 73, 61, 232, 35, 119, 124, 168, 12, 120,
216, 195, 29, 12, 166, 139, 28, 36, 182, 186, 154, 246, 149, 224, 109, 52, 100, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0,
];
// As long as we get the 4-byte enum and the first field right, then
// we're sure the rest works out
let deserialized = try_from_slice_unchecked::<StakeStateWithFlags>(&data).unwrap();
assert_matches!(
deserialized,
StakeStateWithFlags::Initialized(Meta {
rent_exempt_reserve: 2282880,
..
})
);
}
mod deprecated {
use super::*;
fn check_borsh_deserialization(stake: StakeState) {
let serialized = serialize(&stake).unwrap();
let deserialized = StakeState::try_from_slice(&serialized).unwrap();
@ -659,11 +872,10 @@ mod test {
stake: u64::MAX,
activation_epoch: Epoch::MAX,
deactivation_epoch: Epoch::MAX,
..Delegation::default()
warmup_cooldown_rate: f64::MAX,
},
credits_observed: 1,
},
StakeFlags::empty(),
));
}
@ -694,11 +906,10 @@ mod test {
stake: u64::MAX,
activation_epoch: Epoch::MAX,
deactivation_epoch: Epoch::MAX,
..Default::default()
warmup_cooldown_rate: f64::MAX,
},
credits_observed: 1,
},
StakeFlags::MUST_FULLY_ACTIVATE_BEFORE_DEACTIVATION_IS_PERMITTED,
));
}
@ -706,14 +917,14 @@ mod test {
fn borsh_deserialization_live_data() {
let data = [
1, 0, 0, 0, 128, 213, 34, 0, 0, 0, 0, 0, 133, 0, 79, 231, 141, 29, 73, 61, 232, 35,
119, 124, 168, 12, 120, 216, 195, 29, 12, 166, 139, 28, 36, 182, 186, 154, 246, 149,
224, 109, 52, 100, 133, 0, 79, 231, 141, 29, 73, 61, 232, 35, 119, 124, 168, 12, 120,
216, 195, 29, 12, 166, 139, 28, 36, 182, 186, 154, 246, 149, 224, 109, 52, 100, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0,
119, 124, 168, 12, 120, 216, 195, 29, 12, 166, 139, 28, 36, 182, 186, 154, 246,
149, 224, 109, 52, 100, 133, 0, 79, 231, 141, 29, 73, 61, 232, 35, 119, 124, 168,
12, 120, 216, 195, 29, 12, 166, 139, 28, 36, 182, 186, 154, 246, 149, 224, 109, 52,
100, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
];
// As long as we get the 4-byte enum and the first field right, then
// we're sure the rest works out
@ -727,3 +938,4 @@ mod test {
);
}
}
}

View File

@ -289,7 +289,7 @@ mod tests {
client::SyncClient,
genesis_config::create_genesis_config,
signature::{Keypair, Signer},
stake::state::StakeState,
stake::state::StakeStateWithFlags,
},
solana_stake_program::stake_state,
};
@ -297,7 +297,8 @@ mod tests {
fn create_bank(lamports: u64) -> (Bank, Keypair, u64, u64) {
let (genesis_config, mint_keypair) = create_genesis_config(lamports);
let bank = Bank::new_for_tests(&genesis_config);
let stake_rent = bank.get_minimum_balance_for_rent_exemption(StakeState::size_of());
let stake_rent =
bank.get_minimum_balance_for_rent_exemption(StakeStateWithFlags::size_of());
let system_rent = bank.get_minimum_balance_for_rent_exemption(0);
(bank, mint_keypair, stake_rent, system_rent)
}