use libc::{c_void, ftruncate, mmap, munmap, shm_open, shm_unlink, MAP_FAILED, MAP_SHARED, O_CREAT, O_EXCL, O_RDWR, PROT_READ, PROT_WRITE}; #[derive(Clone)] pub struct ShmPool { pub fd: i32, pub id: u32, pub addr: *mut c_void, pub size: usize, pub width: usize, } impl ShmPool { pub fn new(width: usize, height: usize, id: u32) -> std::io::Result { let size = width * height * 4; let shm_path: *const i8 = b"/chlorostart\0".as_ptr() as *const i8; let fd = unsafe { shm_open(shm_path, O_RDWR | O_EXCL | O_CREAT, 0o600) }; if fd == -1 { eprint!("shm_open in ShmPool::new() failed: "); return Err(std::io::Error::last_os_error()) } if unsafe { shm_unlink(shm_path) } == -1 { eprint!("shm_unlink in ShmPool::new() failed: "); return Err(std::io::Error::last_os_error()) } if unsafe { ftruncate(fd, size as i64) } == -1 { eprint!("ftruncate in ShmPool::new() failed: "); return Err(std::io::Error::last_os_error()) } let addr = unsafe { mmap(std::ptr::null_mut(), size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0) }; if addr == MAP_FAILED { eprint!("mmap in ShmPool::new() failed: "); return Err(std::io::Error::last_os_error()) } Ok(ShmPool { fd, id, addr, size, width, }) } pub fn resize(&mut self, size: usize) -> std::io::Result<()> { if unsafe { ftruncate(self.fd, size as i64) } == -1 { return Err(std::io::Error::last_os_error()) }; self.size = size; Ok(()) } pub fn write(&mut self, data: &Vec, offset: usize) { if offset > self.size { return; } unsafe { std::ptr::copy_nonoverlapping( data.as_ptr() as *const u32, // src: data as *const u32 self.addr.offset(4*offset as isize) as *mut u32, // dst: ShmPool address as *mut u32 if offset + data.len() * 4 <= self.size { data.len() } else { data.len() - offset } ); } } pub fn write_pixel(&mut self, data: u32, offset: isize) { // TODO: Bounds check unsafe {*(self.addr.offset(offset*4) as *mut u32) = data;} } pub fn read_pixel(&self, offset: usize) -> u32 { // TODO: Bounds check return unsafe {*(self.addr.offset(4*offset as isize) as *const u32)}; } } impl Drop for ShmPool { fn drop(&mut self) { unsafe { munmap(self.addr, self.size); } } }