use crate::ffi::*;
use crate::MemError;
use std::ffi::c_void;
use std::ptr::{self, NonNull};
pub struct Block {
raw: IOSurfaceRef,
va: NonNull<u8>,
size: usize,
id: u32,
}
unsafe impl Send for Block {}
unsafe impl Sync for Block {}
impl Block {
pub fn open(size: usize) -> Result<Self, MemError> {
if size == 0 {
return Err(MemError::ZeroSize);
}
unsafe {
let dict = CFDictionaryCreateMutable(
ptr::null(),
0,
&kCFTypeDictionaryKeyCallBacks as *const c_void,
&kCFTypeDictionaryValueCallBacks as *const c_void,
);
let sz = size as i64;
CFDictionarySetValue(dict, cf_str("IOSurfaceWidth") as _, cf_i64(sz));
CFDictionarySetValue(dict, cf_str("IOSurfaceHeight") as _, cf_i64(1));
CFDictionarySetValue(dict, cf_str("IOSurfaceBytesPerElement") as _, cf_i64(1));
CFDictionarySetValue(dict, cf_str("IOSurfaceBytesPerRow") as _, cf_i64(sz));
CFDictionarySetValue(dict, cf_str("IOSurfaceAllocSize") as _, cf_i64(sz));
CFDictionarySetValue(dict, cf_str("IOSurfacePixelFormat") as _, cf_i64(0));
let raw = IOSurfaceCreate(dict);
CFRelease(dict as CFTypeRef);
if raw.is_null() {
return Err(MemError::BlockCreateFailed);
}
let kr = IOSurfaceLock(raw, 0, ptr::null_mut());
if kr != KERN_SUCCESS {
CFRelease(raw as CFTypeRef);
return Err(MemError::BlockLockFailed(kr));
}
let base = IOSurfaceGetBaseAddress(raw);
let va = match NonNull::new(base as *mut u8) {
Some(p) => p,
None => {
IOSurfaceUnlock(raw, 0, ptr::null_mut());
CFRelease(raw as CFTypeRef);
return Err(MemError::BlockCreateFailed);
}
};
let actual_size = IOSurfaceGetAllocSize(raw);
let id = IOSurfaceGetID(raw);
Ok(Block {
raw,
va,
size: actual_size,
id,
})
}
}
#[inline(always)]
pub fn address(&self) -> *mut u8 {
self.va.as_ptr()
}
#[inline(always)]
pub fn size(&self) -> usize {
self.size
}
#[inline(always)]
pub fn id(&self) -> u32 {
self.id
}
#[inline(always)]
pub fn handle(&self) -> IOSurfaceRef {
self.raw
}
#[inline(always)]
pub fn as_bytes(&self) -> &[u8] {
unsafe { std::slice::from_raw_parts(self.va.as_ptr(), self.size) }
}
#[inline(always)]
#[allow(clippy::mut_from_ref)]
pub fn as_bytes_mut(&self) -> &mut [u8] {
unsafe { std::slice::from_raw_parts_mut(self.va.as_ptr(), self.size) }
}
#[inline(always)]
pub fn as_f32(&self) -> &[f32] {
unsafe { std::slice::from_raw_parts(self.va.as_ptr() as *const f32, self.size / 4) }
}
#[inline(always)]
#[allow(clippy::mut_from_ref)]
pub fn as_f32_mut(&self) -> &mut [f32] {
unsafe { std::slice::from_raw_parts_mut(self.va.as_ptr() as *mut f32, self.size / 4) }
}
#[inline(always)]
pub fn as_u16(&self) -> &[u16] {
unsafe { std::slice::from_raw_parts(self.va.as_ptr() as *const u16, self.size / 2) }
}
#[inline(always)]
#[allow(clippy::mut_from_ref)]
pub fn as_u16_mut(&self) -> &mut [u16] {
unsafe { std::slice::from_raw_parts_mut(self.va.as_ptr() as *mut u16, self.size / 2) }
}
}
impl Drop for Block {
#[mutants::skip] fn drop(&mut self) {
unsafe {
IOSurfaceUnlock(self.raw, 0, ptr::null_mut());
CFRelease(self.raw as CFTypeRef);
}
}
}