use solana_program::{
account_info::AccountInfo, entrypoint::ProgramResult, program::invoke,
program_error::ProgramError, pubkey::Pubkey, rent::Rent,
system_instruction, sysvar::Sysvar,
};
#[inline(always)]
pub(crate) fn create_pda<'a, 'info>(
target_account: &'a AccountInfo<'info>,
owner: &Pubkey,
space: usize,
pda_seeds: &[&[u8]],
pda_bump: u8,
system_program: &'a AccountInfo<'info>,
payer: &'a AccountInfo<'info>,
) -> ProgramResult {
let pda_bump_slice = &[pda_bump];
let pda_signer_seeds = [pda_seeds, &[pda_bump_slice]].concat();
let rent = Rent::get()?;
if target_account.lamports().eq(&0) {
solana_program::program::invoke_signed(
&solana_program::system_instruction::create_account(
payer.key,
target_account.key,
rent.minimum_balance(space),
space as u64,
owner,
),
&[
payer.clone(),
target_account.clone(),
system_program.clone(),
],
&[&pda_signer_seeds],
)?;
} else {
let rent_exempt_balance = rent
.minimum_balance(space)
.saturating_sub(target_account.lamports());
if rent_exempt_balance.gt(&0) {
solana_program::program::invoke(
&solana_program::system_instruction::transfer(
payer.key,
target_account.key,
rent_exempt_balance,
),
&[
payer.as_ref().clone(),
target_account.as_ref().clone(),
system_program.as_ref().clone(),
],
)?;
}
solana_program::program::invoke_signed(
&solana_program::system_instruction::allocate(
target_account.key,
space as u64,
),
&[
target_account.as_ref().clone(),
system_program.as_ref().clone(),
],
&[&pda_signer_seeds],
)?;
solana_program::program::invoke_signed(
&solana_program::system_instruction::assign(
target_account.key,
owner,
),
&[
target_account.as_ref().clone(),
system_program.as_ref().clone(),
],
&[&pda_signer_seeds],
)?;
}
Ok(())
}
pub(crate) fn resize_pda<'a, 'info>(
payer: &'a AccountInfo<'info>,
pda: &'a AccountInfo<'info>,
system_program: &'a AccountInfo<'info>,
new_size: usize,
) -> Result<(), ProgramError> {
let new_minimum_balance = Rent::default().minimum_balance(new_size);
let lamports_diff = new_minimum_balance.saturating_sub(pda.lamports());
invoke(
&system_instruction::transfer(payer.key, pda.key, lamports_diff),
&[payer.clone(), pda.clone(), system_program.clone()],
)?;
pda.realloc(new_size, false)?;
Ok(())
}
#[inline(always)]
pub(crate) fn close_pda<'a, 'info>(
target_account: &'a AccountInfo<'info>,
destination: &'a AccountInfo<'info>,
) -> ProgramResult {
let dest_starting_lamports = destination.lamports();
**destination.lamports.borrow_mut() = dest_starting_lamports
.checked_add(target_account.lamports())
.unwrap();
**target_account.lamports.borrow_mut() = 0;
target_account.assign(&solana_program::system_program::ID);
target_account.realloc(0, false).map_err(Into::into)
}