2021-06-25 08:52:38 +00:00
|
|
|
//! Memory based communication channel between proxy & daemon for things such as cache
|
|
|
|
//! invalidation.
|
|
|
|
|
|
|
|
use std::os::unix::io::AsRawFd;
|
|
|
|
use std::sync::atomic::{AtomicUsize, Ordering};
|
|
|
|
use std::sync::Arc;
|
|
|
|
|
2021-07-20 11:51:54 +00:00
|
|
|
use anyhow::Error;
|
2021-06-25 08:52:38 +00:00
|
|
|
use nix::fcntl::OFlag;
|
|
|
|
use nix::sys::mman::{MapFlags, ProtFlags};
|
|
|
|
use nix::sys::stat::Mode;
|
|
|
|
use once_cell::sync::OnceCell;
|
|
|
|
|
2021-07-20 11:51:54 +00:00
|
|
|
use proxmox::tools::fs::CreateOptions;
|
2021-06-25 08:52:38 +00:00
|
|
|
use proxmox::tools::mmap::Mmap;
|
|
|
|
|
|
|
|
/// In-memory communication channel.
|
|
|
|
pub struct Memcom {
|
|
|
|
mmap: Mmap<u8>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(C)]
|
|
|
|
struct Head {
|
|
|
|
// User (user.cfg) cache generation/version.
|
|
|
|
user_cache_generation: AtomicUsize,
|
|
|
|
}
|
|
|
|
|
2021-07-03 18:42:55 +00:00
|
|
|
static INSTANCE: OnceCell<Arc<Memcom>> = OnceCell::new();
|
2021-06-25 08:52:38 +00:00
|
|
|
|
2021-07-06 09:56:35 +00:00
|
|
|
const MEMCOM_FILE_PATH: &str = pbs_buildcfg::rundir!("/proxmox-backup-memcom");
|
2021-07-20 11:51:54 +00:00
|
|
|
const EMPTY_PAGE: [u8; 4096] = [0u8; 4096];
|
2021-06-25 08:52:38 +00:00
|
|
|
|
|
|
|
impl Memcom {
|
|
|
|
/// Open the memory based communication channel singleton.
|
2021-07-03 18:42:55 +00:00
|
|
|
pub fn new() -> Result<Arc<Self>, Error> {
|
2021-06-25 08:52:38 +00:00
|
|
|
INSTANCE.get_or_try_init(Self::open).map(Arc::clone)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Actual work of `new`:
|
2021-07-03 18:42:55 +00:00
|
|
|
fn open() -> Result<Arc<Self>, Error> {
|
2021-07-20 11:51:54 +00:00
|
|
|
let user = crate::backup::backup_user()?;
|
|
|
|
let options = CreateOptions::new()
|
|
|
|
.perm(Mode::from_bits_truncate(0o660))
|
|
|
|
.owner(user.uid)
|
|
|
|
.group(user.gid);
|
|
|
|
|
|
|
|
let file = proxmox::tools::fs::atomic_open_or_create_file(
|
|
|
|
MEMCOM_FILE_PATH,
|
|
|
|
OFlag::O_RDWR | OFlag::O_CLOEXEC,
|
|
|
|
&EMPTY_PAGE, options)?;
|
2021-06-25 08:52:38 +00:00
|
|
|
|
|
|
|
let mmap = unsafe {
|
|
|
|
Mmap::<u8>::map_fd(
|
2021-07-20 11:51:54 +00:00
|
|
|
file.as_raw_fd(),
|
2021-06-25 08:52:38 +00:00
|
|
|
0,
|
|
|
|
4096,
|
|
|
|
ProtFlags::PROT_READ | ProtFlags::PROT_WRITE,
|
|
|
|
MapFlags::MAP_SHARED | MapFlags::MAP_NORESERVE | MapFlags::MAP_POPULATE,
|
|
|
|
)?
|
|
|
|
};
|
2021-07-03 18:42:55 +00:00
|
|
|
|
2021-06-25 08:52:38 +00:00
|
|
|
Ok(Arc::new(Self { mmap }))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Shortcut to get the mapped `Head` as a `Head`.
|
2021-07-03 18:42:55 +00:00
|
|
|
fn head(&self) -> &Head {
|
2021-06-25 08:52:38 +00:00
|
|
|
unsafe { &*(self.mmap.as_ptr() as *const u8 as *const Head) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the user cache generation number.
|
2021-07-03 18:42:55 +00:00
|
|
|
pub fn user_cache_generation(&self) -> usize {
|
2021-06-25 08:52:38 +00:00
|
|
|
self.head().user_cache_generation.load(Ordering::Acquire)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Increase the user cache generation number.
|
|
|
|
pub fn increase_user_cache_generation(&self) {
|
|
|
|
self.head()
|
|
|
|
.user_cache_generation
|
|
|
|
.fetch_add(1, Ordering::AcqRel);
|
|
|
|
}
|
|
|
|
}
|