use inspect::Inspect;
use parking_lot::Mutex;
use std::collections::HashMap;
use std::ops::Deref;
use std::sync::Arc;
use thiserror::Error;
pub use vmbus_core::protocol::GpadlId;
use vmbus_ring::gparange::MultiPagedRangeBuf;
pub type TeardownFn = Box<dyn FnOnce() + Send>;
struct Teardown(TeardownFn);
impl std::fmt::Debug for Teardown {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.pad("Teardown")
}
}
#[derive(Debug)]
struct GpadlState {
tearing_down: bool,
map_count: usize,
teardown: Option<Teardown>,
}
#[derive(Debug)]
struct Gpadl {
id: GpadlId,
buf: MultiPagedRangeBuf<Vec<u64>>,
state: Mutex<GpadlState>,
}
impl Inspect for Gpadl {
fn inspect(&self, req: inspect::Request<'_>) {
let state = self.state.lock();
req.respond()
.field("tearing_down", state.tearing_down)
.field("map_count", state.map_count)
.child("ranges", |req| {
let mut resp = req.respond();
for (i, range) in self.buf.iter().enumerate() {
resp.child(&i.to_string(), |req| {
req.respond()
.hex("len", range.len())
.hex("offset", range.offset())
.field(
"pages",
range
.gpns()
.iter()
.map(|gpa| format!("{:x}", gpa))
.collect::<Vec<_>>()
.join(" "),
);
});
}
});
}
}
#[derive(Debug)]
pub struct GpadlView(Arc<Gpadl>);
impl Clone for GpadlView {
fn clone(&self) -> Self {
let clone = GpadlView(self.0.clone());
let mut state = self.0.state.lock();
state.map_count += 1;
clone
}
}
impl GpadlView {
pub fn id(&self) -> GpadlId {
self.0.id
}
}
impl Deref for GpadlView {
type Target = MultiPagedRangeBuf<Vec<u64>>;
fn deref(&self) -> &Self::Target {
&self.0.buf
}
}
impl Drop for GpadlView {
fn drop(&mut self) {
let teardown = {
let mut state = self.0.state.lock();
state.map_count -= 1;
if state.map_count == 0 && state.tearing_down {
state.teardown.take()
} else {
None
}
};
if let Some(Teardown(teardown)) = teardown {
teardown();
}
}
}
#[derive(Debug)]
pub struct GpadlMap {
map: Mutex<HashMap<GpadlId, Arc<Gpadl>>>,
}
impl Inspect for GpadlMap {
fn inspect(&self, req: inspect::Request<'_>) {
req.respond()
.fields("", self.map.lock().iter().map(|(id, gpadl)| (&id.0, gpadl)));
}
}
impl GpadlMap {
pub fn new() -> Arc<Self> {
Arc::new(GpadlMap {
map: Default::default(),
})
}
pub fn add(&self, id: GpadlId, buf: MultiPagedRangeBuf<Vec<u64>>) {
let gpadl = Arc::new(Gpadl {
id,
buf,
state: Mutex::new(GpadlState {
tearing_down: false,
map_count: 0,
teardown: None,
}),
});
let mut map = self.map.lock();
map.insert(id, gpadl);
}
pub fn remove(&self, id: GpadlId, f: TeardownFn) -> Option<TeardownFn> {
let gpadl = {
let mut map = self.map.lock();
map.remove(&id).unwrap()
};
let mut state = gpadl.state.lock();
assert!(!state.tearing_down && state.teardown.is_none());
state.tearing_down = true;
if state.map_count > 0 {
state.teardown = Some(Teardown(f));
None
} else {
Some(f)
}
}
pub fn view(self: Arc<Self>) -> GpadlMapView {
GpadlMapView(Some(self))
}
}
#[derive(Debug, Default, Clone)]
pub struct GpadlMapView(Option<Arc<GpadlMap>>);
#[derive(Debug, Error)]
#[error("unknown gpadl ID {:#x}", (.0).0)]
pub struct UnknownGpadlId(GpadlId);
impl GpadlMapView {
pub fn map(&self, id: GpadlId) -> Result<GpadlView, UnknownGpadlId> {
self.try_map(id).ok_or(UnknownGpadlId(id))
}
fn try_map(&self, id: GpadlId) -> Option<GpadlView> {
let gpadl = {
let map = self.0.as_ref()?.map.lock();
map.get(&id)?.clone()
};
{
let mut state = gpadl.state.lock();
if state.tearing_down {
return None;
}
state.map_count += 1;
}
Some(GpadlView(gpadl))
}
}