1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
use solana_program::{
account_info::AccountInfo,
entrypoint::ProgramResult,
program::{invoke, invoke_signed},
pubkey::Pubkey,
rent::Rent,
system_instruction,
};
pub fn create_account<'a, 'info>(
payer: &'a AccountInfo<'info>,
new_account: &'a AccountInfo<'info>,
system_program: &'a AccountInfo<'info>,
program_owner: &Pubkey,
rent: &Rent,
space: u64,
seeds: Vec<Vec<u8>>,
) -> ProgramResult {
let current_lamports = **new_account.try_borrow_lamports()?;
if current_lamports == 0 {
invoke_signed(
&system_instruction::create_account(
payer.key,
new_account.key,
rent.minimum_balance(space as usize),
space,
program_owner,
),
&[payer.clone(), new_account.clone(), system_program.clone()],
&[seeds
.iter()
.map(|seed| seed.as_slice())
.collect::<Vec<&[u8]>>()
.as_slice()],
)
} else {
let required_lamports = rent
.minimum_balance(space as usize)
.max(1)
.saturating_sub(current_lamports);
if required_lamports > 0 {
invoke(
&system_instruction::transfer(payer.key, new_account.key, required_lamports),
&[payer.clone(), new_account.clone(), system_program.clone()],
)?;
}
invoke_signed(
&system_instruction::allocate(new_account.key, space),
&[new_account.clone(), system_program.clone()],
&[seeds
.iter()
.map(|seed| seed.as_slice())
.collect::<Vec<&[u8]>>()
.as_slice()],
)?;
invoke_signed(
&system_instruction::assign(new_account.key, program_owner),
&[new_account.clone(), system_program.clone()],
&[seeds
.iter()
.map(|seed| seed.as_slice())
.collect::<Vec<&[u8]>>()
.as_slice()],
)
}
}