use std::convert::TryInto;
use arrayref::{array_refs, mut_array_refs};
use solana_program::program_option::COption;
use solana_program::pubkey::{Pubkey, PUBKEY_BYTES};
use solana_program::{
clock::{DEFAULT_TICKS_PER_SECOND, DEFAULT_TICKS_PER_SLOT, SECONDS_PER_DAY},
msg,
program_error::ProgramError,
};
pub use last_update::*;
pub use lending_market::*;
pub use obligation::*;
pub use reserve::*;
use solana_maths::{Decimal, WAD};
mod last_update;
mod lending_market;
mod obligation;
mod reserve;
pub const INITIAL_COLLATERAL_RATIO: u64 = 1;
pub const INITIAL_COLLATERAL_RATE: u64 = INITIAL_COLLATERAL_RATIO * WAD;
pub const PROGRAM_VERSION: u8 = 1;
pub const UNINITIALIZED_VERSION: u8 = 0;
pub const SLOTS_PER_YEAR: u64 =
DEFAULT_TICKS_PER_SECOND / DEFAULT_TICKS_PER_SLOT * SECONDS_PER_DAY * 365;
fn pack_decimal(decimal: Decimal, dst: &mut [u8; 16]) {
*dst = decimal
.to_scaled_val()
.expect("Decimal cannot be packed")
.to_le_bytes();
}
fn unpack_decimal(src: &[u8; 16]) -> Decimal {
Decimal::from_scaled_val(u128::from_le_bytes(*src))
}
fn pack_bool(boolean: bool, dst: &mut [u8; 1]) {
*dst = (boolean as u8).to_le_bytes()
}
fn unpack_bool(src: &[u8; 1]) -> Result<bool, ProgramError> {
match u8::from_le_bytes(*src) {
0 => Ok(false),
1 => Ok(true),
_ => {
msg!("Boolean cannot be unpacked");
Err(ProgramError::InvalidAccountData)
}
}
}
fn pack_coption_key(src: &COption<Pubkey>, dst: &mut [u8; 4 + PUBKEY_BYTES]) {
#[allow(clippy::ptr_offset_with_cast)]
let (tag, body) = mut_array_refs![dst, 4, PUBKEY_BYTES];
match src {
COption::Some(key) => {
*tag = [1, 0, 0, 0];
body.copy_from_slice(key.as_ref());
}
COption::None => {
*tag = [0; 4];
}
}
}
fn unpack_coption_key(src: &[u8; 4 + PUBKEY_BYTES]) -> Result<COption<Pubkey>, ProgramError> {
#[allow(clippy::ptr_offset_with_cast)]
let (tag, body) = array_refs![src, 4, PUBKEY_BYTES];
match *tag {
[0, 0, 0, 0] => Ok(COption::None),
[1, 0, 0, 0] => Ok(COption::Some(Pubkey::new_from_array(*body))),
_ => {
msg!("COption<Pubkey> cannot be unpacked");
Err(ProgramError::InvalidAccountData)
}
}
}
pub fn pack_coption_key_compact(src: &COption<Pubkey>, dst: &mut [u8; 1 + PUBKEY_BYTES]) {
match src {
COption::Some(key) => {
dst[0] = 1;
dst[1..].copy_from_slice(key.as_ref());
}
COption::None => {
dst[0] = 0;
}
}
}
pub fn unpack_coption_key_compact(
src: &[u8; 1 + PUBKEY_BYTES],
) -> Result<COption<Pubkey>, ProgramError> {
match src[0] {
0 => Ok(COption::None),
1 => Ok(COption::Some(Pubkey::new_from_array(
src[1..].try_into().unwrap(),
))),
_ => {
msg!("COption<Pubkey> cannot be unpacked");
Err(ProgramError::InvalidAccountData)
}
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn initial_collateral_rate_sanity() {
assert_eq!(
INITIAL_COLLATERAL_RATIO.checked_mul(WAD).unwrap(),
INITIAL_COLLATERAL_RATE
);
}
}