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
71
72
73
74
75
76
77
78
79
80
81
82
use crate::error::TokenError;
use solana_sdk::program_error::ProgramError;
pub trait IsInitialized {
fn is_initialized(&self) -> bool;
}
pub trait Sealed: Sized {}
pub trait Pack: Sealed {
const LEN: usize;
#[doc(hidden)]
fn pack_into_slice(&self, dst: &mut [u8]);
#[doc(hidden)]
fn unpack_from_slice(src: &[u8]) -> Result<Self, ProgramError>;
fn unpack(input: &[u8]) -> Result<Self, ProgramError>
where
Self: IsInitialized,
{
let value = Self::unpack_unchecked(input)?;
if value.is_initialized() {
Ok(value)
} else {
Err(TokenError::UninitializedState.into())
}
}
fn unpack_unchecked(input: &[u8]) -> Result<Self, ProgramError> {
if input.len() < Self::LEN {
println!("ilen {:?} tlen {:?}", input.len(), Self::LEN);
return Err(ProgramError::InvalidAccountData);
}
Ok(Self::unpack_from_slice(input)?)
}
#[inline(never)]
fn unpack_mut<F, U>(input: &mut [u8], f: &mut F) -> Result<U, ProgramError>
where
F: FnMut(&mut Self) -> Result<U, ProgramError>,
Self: IsInitialized,
{
let mut t = Self::unpack(input)?;
let u = f(&mut t)?;
Self::pack(t, input)?;
Ok(u)
}
#[inline(never)]
fn unpack_unchecked_mut<F, U>(input: &mut [u8], f: &mut F) -> Result<U, ProgramError>
where
F: FnMut(&mut Self) -> Result<U, ProgramError>,
{
let mut t = Self::unpack_unchecked(input)?;
let u = f(&mut t)?;
Self::pack(t, input)?;
Ok(u)
}
fn pack(src: Self, dst: &mut [u8]) -> Result<(), ProgramError> {
if dst.len() < Self::LEN {
println!("dlen {:?} tlen {:?}", dst.len(), Self::LEN);
return Err(ProgramError::InvalidAccountData);
}
src.pack_into_slice(dst);
Ok(())
}
}