#![warn(missing_docs)]
use guestmem::GuestMemory;
use inspect::InspectMut;
use mesh::RecvError;
use std::task::Context;
use std::task::Poll;
use vmcore::line_interrupt::LineInterrupt;
#[expect(missing_docs)] pub struct GenerationIdRuntimeDeps {
pub gm: GuestMemory,
pub generation_id_recv: mesh::Receiver<[u8; 16]>,
pub notify_interrupt: LineInterrupt,
}
#[derive(InspectMut)]
#[inspect(extra = "GenerationId::inspect_extra")]
pub struct GenerationId {
#[inspect(skip)]
rt: GenerationIdRuntimeDeps,
id: [u8; 16],
#[inspect(with = "ptr_to_opt_u64")]
ptr: [Option<u32>; 2],
}
impl GenerationId {
fn inspect_extra(&mut self, resp: &mut inspect::Response<'_>) {
resp.field_mut_with("force_update", |update| {
let v = if let Some(update) = update {
let update = update.parse()?;
if update {
let mut id = [0; 16];
getrandom::getrandom(&mut id).unwrap();
self.set_and_update_generation_id(id);
tracing::info!("Force updated genid...");
}
update
} else {
false
};
Result::<_, std::str::ParseBoolError>::Ok(v.to_string())
});
}
}
impl GenerationId {
pub fn new(initial_generation_id: [u8; 16], platform: GenerationIdRuntimeDeps) -> Self {
Self {
id: initial_generation_id,
ptr: [None, None],
rt: platform,
}
}
pub fn reset(&mut self) {
self.ptr = [None; 2];
}
pub fn write_generation_id_low(&mut self, data: u32) {
self.set_ptr(0, data)
}
pub fn write_generation_id_high(&mut self, data: u32) {
self.set_ptr(1, data)
}
pub fn poll(&mut self, cx: &mut Context<'_>) {
while let Poll::Ready(val) = self.rt.generation_id_recv.poll_recv(cx) {
match val {
Ok(val) => self.set_and_update_generation_id(val),
Err(RecvError::Closed) => break,
Err(err) => {
tracing::error!(
error = &err as &dyn std::error::Error,
"Error receiving generation ID"
);
break;
}
}
}
}
fn update_generation_id(&self) {
if let Some(ptr) = ptr_to_opt_u64(&self.ptr) {
if let Err(e) = self.rt.gm.write_at(ptr, &self.id) {
tracelimit::error_ratelimited!(
error = &e as &dyn std::error::Error,
"failed to write generation ID"
)
}
}
}
fn set_and_update_generation_id(&mut self, val: [u8; 16]) {
self.id = val;
self.update_generation_id();
self.rt.notify_interrupt.set_level(true);
self.rt.notify_interrupt.set_level(false);
}
fn set_ptr(&mut self, index: usize, data: u32) {
self.ptr[index] = Some(data);
self.update_generation_id();
}
}
fn ptr_to_opt_u64(ptr: &[Option<u32>; 2]) -> Option<u64> {
Some(((ptr[1]? as u64) << 32) | ptr[0]? as u64)
}
mod save_restore {
use super::*;
use vmcore::save_restore::RestoreError;
use vmcore::save_restore::SaveError;
use vmcore::save_restore::SaveRestore;
mod state {
use mesh::payload::Protobuf;
#[derive(Protobuf)]
#[mesh(package = "firmware.generation_id")]
pub struct SavedState {
#[mesh(1)]
pub id: [u8; 16],
#[mesh(2)]
pub ptr: [Option<u32>; 2],
}
}
impl SaveRestore for GenerationId {
type SavedState = state::SavedState;
fn save(&mut self) -> Result<Self::SavedState, SaveError> {
let Self { rt: _, id, ptr } = *self;
let saved_state = state::SavedState { id, ptr };
Ok(saved_state)
}
fn restore(&mut self, state: Self::SavedState) -> Result<(), RestoreError> {
let state::SavedState { id, ptr } = state;
self.id = id;
self.ptr = ptr;
Ok(())
}
}
}