Skip to content

Commit

Permalink
Use Arc/Mutex instead of Rc/RefCell in PixmanImage
Browse files Browse the repository at this point in the history
Allows `PixmanImage` to be sent to another thread.
  • Loading branch information
ids1024 committed Dec 4, 2024
1 parent f988a7e commit 0057af0
Showing 1 changed file with 25 additions and 28 deletions.
53 changes: 25 additions & 28 deletions src/backend/renderer/pixman/mod.rs
Original file line number Diff line number Diff line change
@@ -1,8 +1,7 @@
//! Implementation of the rendering traits using pixman
use std::{
cell::RefCell,
rc::Rc,
sync::atomic::{AtomicBool, Ordering},
use std::sync::{
atomic::{AtomicBool, Ordering},
Arc, Mutex,
};

use drm_fourcc::{DrmFormat, DrmFourcc, DrmModifier};
Expand Down Expand Up @@ -91,12 +90,12 @@ struct PixmanImageInner {
#[cfg(feature = "wayland_frontend")]
buffer: Option<Weak<wl_buffer::WlBuffer>>,
dmabuf: Option<PixmanDmabufMapping>,
image: RefCell<Image<'static, 'static>>,
image: Mutex<Image<'static, 'static>>,
_flipped: bool, /* TODO: What about flipped textures? */
}

#[derive(Debug, Clone)]
struct PixmanImage(Rc<PixmanImageInner>);
struct PixmanImage(Arc<PixmanImageInner>);

impl PixmanImage {
#[profiling::function]
Expand Down Expand Up @@ -131,12 +130,12 @@ pub struct PixmanTexture(PixmanImage);
impl From<pixman::Image<'static, 'static>> for PixmanTexture {
#[inline]
fn from(image: pixman::Image<'static, 'static>) -> Self {
Self(PixmanImage(Rc::new(PixmanImageInner {
Self(PixmanImage(Arc::new(PixmanImageInner {
#[cfg(feature = "wayland_frontend")]
buffer: None,
dmabuf: None,
_flipped: false,
image: RefCell::new(image),
image: Mutex::new(image),
})))
}
}
Expand Down Expand Up @@ -168,7 +167,7 @@ impl Drop for DmabufReadGuard {
struct TextureAccessor<'l> {
#[cfg(feature = "wayland_frontend")]
buffer: Option<wl_buffer::WlBuffer>,
image: &'l RefCell<Image<'static, 'static>>,
image: &'l Mutex<Image<'static, 'static>>,
_guard: Option<DmabufReadGuard>,
}

Expand All @@ -177,7 +176,7 @@ impl TextureAccessor<'_> {
where
F: for<'a> FnOnce(&'a mut Image<'static, 'static>) -> R,
{
let mut image = self.image.borrow_mut();
let mut image = self.image.lock().unwrap();

#[cfg(feature = "wayland_frontend")]
if let Some(buffer) = self.buffer.as_ref() {
Expand All @@ -196,7 +195,7 @@ impl TextureAccessor<'_> {
});
}

let mut remapped_image = unsafe {
let remapped_image = unsafe {
// SAFETY: We guarantee that this image is only used for reading,
// so it is safe to cast the ptr to *mut
Image::from_raw_mut(
Expand All @@ -210,11 +209,9 @@ impl TextureAccessor<'_> {
}
.map_err(|_| PixmanError::ImportFailed)?;

// We no longer need the original image, so release it to prevent double borrowing
std::mem::drop(image);
*image = remapped_image;

let res = f(&mut remapped_image);
self.image.replace(remapped_image);
let res = f(&mut image);
Ok(res)
} else {
Ok(f(&mut image))
Expand All @@ -235,15 +232,15 @@ impl PixmanTexture {

impl Texture for PixmanTexture {
fn width(&self) -> u32 {
self.0 .0.image.borrow().width() as u32
self.0 .0.image.lock().unwrap().width() as u32
}

fn height(&self) -> u32 {
self.0 .0.image.borrow().height() as u32
self.0 .0.image.lock().unwrap().height() as u32
}

fn format(&self) -> Option<DrmFourcc> {
DrmFourcc::try_from(self.0 .0.image.borrow().format()).ok()
DrmFourcc::try_from(self.0 .0.image.lock().unwrap().format()).ok()
}
}

Expand Down Expand Up @@ -271,7 +268,7 @@ impl PixmanFrame<'_> {
let mut binding;
let target_image = match self.renderer.target.as_mut().ok_or(PixmanError::NoTargetBound)? {
PixmanTarget::Image { image, .. } => {
binding = image.0.image.borrow_mut();
binding = image.0.image.lock().unwrap();
&mut *binding
}
PixmanTarget::RenderBuffer(b) => &mut b.0,
Expand Down Expand Up @@ -382,7 +379,7 @@ impl Frame for PixmanFrame<'_> {
let mut binding;
let target_image = match self.renderer.target.as_mut().ok_or(PixmanError::NoTargetBound)? {
PixmanTarget::Image { image, .. } => {
binding = image.0.image.borrow_mut();
binding = image.0.image.lock().unwrap();
&mut *binding
}
PixmanTarget::RenderBuffer(b) => &mut b.0,
Expand Down Expand Up @@ -758,14 +755,14 @@ impl PixmanRenderer {
}
.map_err(|_| PixmanError::ImportFailed)?;

Ok(PixmanImage(Rc::new(PixmanImageInner {
Ok(PixmanImage(Arc::new(PixmanImageInner {
#[cfg(feature = "wayland_frontend")]
buffer: None,
dmabuf: Some(PixmanDmabufMapping {
dmabuf: dmabuf.weak(),
_mapping: dmabuf_mapping,
}),
image: RefCell::new(image),
image: Mutex::new(image),
_flipped: false,
})))
}
Expand Down Expand Up @@ -878,11 +875,11 @@ impl ImportMem for PixmanRenderer {
unsafe {
std::ptr::copy_nonoverlapping(data.as_ptr(), image.data() as *mut u8, expected_len);
}
Ok(PixmanTexture(PixmanImage(Rc::new(PixmanImageInner {
Ok(PixmanTexture(PixmanImage(Arc::new(PixmanImageInner {
#[cfg(feature = "wayland_frontend")]
buffer: None,
dmabuf: None,
image: RefCell::new(image),
image: Mutex::new(image),
_flipped: flipped,
}))))
}
Expand All @@ -903,7 +900,7 @@ impl ImportMem for PixmanRenderer {
return Err(PixmanError::ImportFailed);
}

let mut image = texture.0 .0.image.borrow_mut();
let mut image = texture.0 .0.image.lock().unwrap();
let stride = image.stride();
let expected_len = stride * image.height();

Expand Down Expand Up @@ -990,7 +987,7 @@ impl ExportMem for PixmanRenderer {
let target_image = match target {
PixmanTarget::Image { dmabuf, image } => {
dmabuf.sync_plane(0, DmabufSyncFlags::START | DmabufSyncFlags::READ)?;
binding = image.0.image.borrow();
binding = image.0.image.lock().unwrap();
&*binding
}
PixmanTarget::RenderBuffer(b) => &b.0,
Expand Down Expand Up @@ -1126,10 +1123,10 @@ impl ImportMemWl for PixmanRenderer {
.map_err(|_| PixmanError::ImportFailed)?;
std::result::Result::<_, PixmanError>::Ok(image)
})??;
Ok(PixmanTexture(PixmanImage(Rc::new(PixmanImageInner {
Ok(PixmanTexture(PixmanImage(Arc::new(PixmanImageInner {
buffer: Some(buffer.downgrade()),
dmabuf: None,
image: RefCell::new(image),
image: Mutex::new(image),
_flipped: false,
}))))
}
Expand Down

0 comments on commit 0057af0

Please sign in to comment.