rust/hwphysics/src/common.rs
author alfadur
Tue, 30 Jul 2019 19:53:23 +0300
changeset 15281 8095853811a6
parent 15275 66c987015f2d
child 15282 478d5372eb4a
permissions -rw-r--r--
add gear lookup to the physics processor

use fpnum::{fp, FPNum};
use std::{
    collections::BinaryHeap,
    num::NonZeroU16,
    ops::{Add, Index, IndexMut},
};

pub type GearId = NonZeroU16;
pub trait GearData {}

#[derive(PartialEq, Eq, PartialOrd, Ord, Clone, Copy, Debug)]
#[repr(transparent)]
pub struct Millis(u32);

impl Millis {
    #[inline]
    pub fn new(value: u32) -> Self {
        Self(value)
    }

    #[inline]
    pub fn get(self) -> u32 {
        self.0
    }

    #[inline]
    pub fn to_fixed(self) -> FPNum {
        FPNum::new(self.0 as i32, 1000)
    }
}

impl Add for Millis {
    type Output = Self;

    fn add(self, rhs: Self) -> Self::Output {
        Self(self.0 + rhs.0)
    }
}

pub trait GearDataProcessor<T: GearData> {
    fn add(&mut self, gear_id: GearId, gear_data: T);
    fn remove(&mut self, gear_id: GearId);
}

pub trait GearDataAggregator<T: GearData> {
    fn find_processor(&mut self) -> &mut GearDataProcessor<T>;
}

pub struct GearAllocator {
    max_id: u16,
    free_ids: BinaryHeap<GearId>,
}

impl GearAllocator {
    pub fn new() -> Self {
        Self {
            max_id: 0,
            free_ids: BinaryHeap::with_capacity(1024),
        }
    }

    pub fn alloc(&mut self) -> Option<GearId> {
        self.free_ids.pop().or_else(|| {
            self.max_id.checked_add(1).and_then(|new_max_id| {
                self.max_id = new_max_id;
                NonZeroU16::new(new_max_id)
            })
        })
    }

    pub fn free(&mut self, gear_id: GearId) {
        self.free_ids.push(gear_id)
    }
}

#[derive(Clone, Copy, Default)]
pub struct LookupEntry<T> {
    pub index: u16,
    pub value: T,
}

impl<T> LookupEntry<T> {
    pub fn new(index: u16, value: T) -> Self {
        Self { index, value }
    }
}

pub struct GearDataLookup<T> {
    lookup: [LookupEntry<T>; u16::max_value() as usize],
}

impl<T: Default + Copy> GearDataLookup<T> {
    pub fn new() -> Self {
        Self {
            lookup: [LookupEntry::<T>::default(); u16::max_value() as usize],
        }
    }
}

impl<T> GearDataLookup<T> {
    pub fn get(&self, gear_id: GearId) -> &LookupEntry<T> {
        // All possible Gear IDs are valid indices
        unsafe { self.lookup.get_unchecked(gear_id.get() as usize - 1) }
    }

    pub fn get_mut(&mut self, gear_id: GearId) -> &mut LookupEntry<T> {
        // All possible Gear IDs are valid indices
        unsafe { self.lookup.get_unchecked_mut(gear_id.get() as usize - 1) }
    }
}

impl<T> Index<GearId> for GearDataLookup<T> {
    type Output = LookupEntry<T>;

    fn index(&self, index: GearId) -> &Self::Output {
        self.get(index)
    }
}

impl<T> IndexMut<GearId> for GearDataLookup<T> {
    fn index_mut(&mut self, index: GearId) -> &mut Self::Output {
        self.get_mut(index)
    }
}