1#![deny(missing_docs)]
2
3pub mod big_vec;
6pub mod error;
7pub mod instruction;
8pub mod processor;
9pub mod state;
10
11#[cfg(not(feature = "no-entrypoint"))]
12pub mod entrypoint;
13
14pub use solana_program;
16use {
17 crate::state::Fee,
18 solana_program::{pubkey::Pubkey, stake::state::Meta},
19 std::num::NonZeroU32,
20};
21
22const AUTHORITY_DEPOSIT: &[u8] = b"deposit";
24
25const AUTHORITY_WITHDRAW: &[u8] = b"withdraw";
27
28const TRANSIENT_STAKE_SEED_PREFIX: &[u8] = b"transient";
30
31const EPHEMERAL_STAKE_SEED_PREFIX: &[u8] = b"ephemeral";
33
34pub const MINIMUM_ACTIVE_STAKE: u64 = 1_000_000;
37
38pub const MINIMUM_RESERVE_LAMPORTS: u64 = 1;
42
43pub const MAX_VALIDATORS_TO_UPDATE: usize = 5;
46
47pub const MAX_WITHDRAWAL_FEE_INCREASE: Fee = Fee {
51 numerator: 3,
52 denominator: 2,
53};
54pub const WITHDRAWAL_BASELINE_FEE: Fee = Fee {
56 numerator: 1,
57 denominator: 1000,
58};
59
60pub const MAX_TRANSIENT_STAKE_ACCOUNTS: usize = 10;
63
64#[inline]
67pub fn minimum_stake_lamports(meta: &Meta, stake_program_minimum_delegation: u64) -> u64 {
68 meta.rent_exempt_reserve
69 .saturating_add(minimum_delegation(stake_program_minimum_delegation))
70}
71
72#[inline]
74pub fn minimum_delegation(stake_program_minimum_delegation: u64) -> u64 {
75 std::cmp::max(stake_program_minimum_delegation, MINIMUM_ACTIVE_STAKE)
76}
77
78#[inline]
81pub fn minimum_reserve_lamports(meta: &Meta) -> u64 {
82 meta.rent_exempt_reserve
83 .saturating_add(MINIMUM_RESERVE_LAMPORTS)
84}
85
86pub fn find_deposit_authority_program_address(
88 program_id: &Pubkey,
89 stake_pool_address: &Pubkey,
90) -> (Pubkey, u8) {
91 Pubkey::find_program_address(
92 &[stake_pool_address.as_ref(), AUTHORITY_DEPOSIT],
93 program_id,
94 )
95}
96
97pub fn find_withdraw_authority_program_address(
99 program_id: &Pubkey,
100 stake_pool_address: &Pubkey,
101) -> (Pubkey, u8) {
102 Pubkey::find_program_address(
103 &[stake_pool_address.as_ref(), AUTHORITY_WITHDRAW],
104 program_id,
105 )
106}
107
108pub fn find_stake_program_address(
110 program_id: &Pubkey,
111 vote_account_address: &Pubkey,
112 stake_pool_address: &Pubkey,
113 seed: Option<NonZeroU32>,
114) -> (Pubkey, u8) {
115 let seed = seed.map(|s| s.get().to_le_bytes());
116 Pubkey::find_program_address(
117 &[
118 vote_account_address.as_ref(),
119 stake_pool_address.as_ref(),
120 seed.as_ref().map(|s| s.as_slice()).unwrap_or(&[]),
121 ],
122 program_id,
123 )
124}
125
126pub fn find_transient_stake_program_address(
128 program_id: &Pubkey,
129 vote_account_address: &Pubkey,
130 stake_pool_address: &Pubkey,
131 seed: u64,
132) -> (Pubkey, u8) {
133 Pubkey::find_program_address(
134 &[
135 TRANSIENT_STAKE_SEED_PREFIX,
136 vote_account_address.as_ref(),
137 stake_pool_address.as_ref(),
138 &seed.to_le_bytes(),
139 ],
140 program_id,
141 )
142}
143
144pub fn find_ephemeral_stake_program_address(
146 program_id: &Pubkey,
147 stake_pool_address: &Pubkey,
148 seed: u64,
149) -> (Pubkey, u8) {
150 Pubkey::find_program_address(
151 &[
152 EPHEMERAL_STAKE_SEED_PREFIX,
153 stake_pool_address.as_ref(),
154 &seed.to_le_bytes(),
155 ],
156 program_id,
157 )
158}
159
160solana_program::declare_id!("spooqgqqDxZgVc3pR6EvuVFZJ1kj7ABM4Hccz1gwAN1");
161
162#[cfg(test)]
163mod test {
164 use super::*;
165
166 #[test]
167 fn validator_stake_account_derivation() {
168 let vote = Pubkey::new_unique();
169 let stake_pool = Pubkey::new_unique();
170 let function_derived = find_stake_program_address(&id(), &vote, &stake_pool, None);
171 let hand_derived =
172 Pubkey::find_program_address(&[vote.as_ref(), stake_pool.as_ref()], &id());
173 assert_eq!(function_derived, hand_derived);
174 }
175}