From 34062713c3a190967bfcd8c00a7ab41f825d05a7 Mon Sep 17 00:00:00 2001 From: Ian Douglas Scott Date: Tue, 1 Oct 2024 14:55:58 -0700 Subject: [PATCH] WIP image-capture-source / image-copy-capture --- Cargo.toml | 2 +- examples/image_copy_capture.rs | 83 +++++++ src/dmabuf.rs | 5 +- src/image_copy_capture/mod.rs | 417 +++++++++++++++++++++++++++++++++ src/lib.rs | 1 + 5 files changed, 505 insertions(+), 3 deletions(-) create mode 100644 examples/image_copy_capture.rs create mode 100644 src/image_copy_capture/mod.rs diff --git a/Cargo.toml b/Cargo.toml index b46aa56dd..dabbce653 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -27,7 +27,7 @@ thiserror = "1.0.30" wayland-backend = "0.3.0" wayland-client = "0.31.1" wayland-cursor = "0.31.0" -wayland-protocols = { version = "0.32.1", features = ["client", "staging", "unstable"] } +wayland-protocols = { version = "0.32.4", features = ["client", "staging", "unstable"] } wayland-protocols-wlr = { version = "0.3.1", features = ["client"] } wayland-scanner = "0.31.0" wayland-csd-frame = "0.3.0" diff --git a/examples/image_copy_capture.rs b/examples/image_copy_capture.rs new file mode 100644 index 000000000..353691966 --- /dev/null +++ b/examples/image_copy_capture.rs @@ -0,0 +1,83 @@ +use smithay_client_toolkit::{ + delegate_image_copy_capture, + image_copy_capture::{ + BufferConstraints, Frame, ImageCopyCaptureHandler, ImageCopyCaptureState, + ImageCopyFrameData, ImageCopyFrameDataExt, ImageCopySessionData, ImageCopySessionDataExt, + }, +}; +use wayland_client::{Connection, QueueHandle, WEnum}; +use wayland_protocols::ext::image_copy_capture::v1::client::{ + ext_image_copy_capture_cursor_session_v1, ext_image_copy_capture_frame_v1, + ext_image_copy_capture_manager_v1, ext_image_copy_capture_session_v1, +}; + +struct State { + image_copy_capture_state: ImageCopyCaptureState, +} + +struct SessionData { + session_data: ImageCopySessionData, +} + +impl ImageCopySessionDataExt for SessionData { + fn image_copy_session_data(&self) -> &ImageCopySessionData { + &self.session_data + } +} + +struct FrameData { + frame_data: ImageCopyFrameData, +} + +impl ImageCopyFrameDataExt for FrameData { + fn image_copy_frame_data(&self) -> &ImageCopyFrameData { + &self.frame_data + } +} + +struct CursorSessionData {} + +fn main() {} + +impl ImageCopyCaptureHandler for State { + fn image_copy_capture_state(&mut self) -> &mut ImageCopyCaptureState { + &mut self.image_copy_capture_state + } + + fn buffer_constraints( + &mut self, + conn: &Connection, + qh: &QueueHandle, + session: &ext_image_copy_capture_session_v1::ExtImageCopyCaptureSessionV1, + constraints: BufferConstraints, + ) { + } + + fn stopped( + &mut self, + conn: &Connection, + qh: &QueueHandle, + session: &ext_image_copy_capture_session_v1::ExtImageCopyCaptureSessionV1, + ) { + } + + fn ready( + &mut self, + conn: &Connection, + qh: &QueueHandle, + image_copy_frame: &ext_image_copy_capture_frame_v1::ExtImageCopyCaptureFrameV1, + frame: Frame, + ) { + } + + fn failed( + &mut self, + conn: &Connection, + qh: &QueueHandle, + image_copy_frame: &ext_image_copy_capture_frame_v1::ExtImageCopyCaptureFrameV1, + reason: WEnum, + ) { + } +} + +delegate_image_copy_capture!(State, session: [SessionData], frame: [FrameData], cursor_session: [CursorSessionData]); diff --git a/src/dmabuf.rs b/src/dmabuf.rs index a947b58ca..a1a54ee86 100644 --- a/src/dmabuf.rs +++ b/src/dmabuf.rs @@ -14,9 +14,9 @@ use wayland_protocols::wp::linux_dmabuf::zv1::client::{ // Workaround until `libc` updates to FreeBSD 12 ABI #[cfg(target_os = "freebsd")] -type dev_t = u64; +pub(crate) type dev_t = u64; #[cfg(not(target_os = "freebsd"))] -use libc::dev_t; +pub(crate) use libc::dev_t; /// A preference tranche of dmabuf formats #[derive(Debug)] @@ -42,6 +42,7 @@ impl Default for DmabufFeedbackTranche { /// A single dmabuf format/modifier pair // Must have correct representation to be able to mmap format table +#[derive(Clone)] #[repr(C)] pub struct DmabufFormat { /// Fourcc format diff --git a/src/image_copy_capture/mod.rs b/src/image_copy_capture/mod.rs new file mode 100644 index 000000000..b396aed7e --- /dev/null +++ b/src/image_copy_capture/mod.rs @@ -0,0 +1,417 @@ +use crate::{ + dmabuf::dev_t, + error::GlobalError, + globals::GlobalData, + registry::GlobalProxy, +}; +use std::{sync::Mutex, time::Duration}; +use wayland_client::{ + globals::GlobalList, + protocol::{wl_buffer, wl_output, wl_shm, wl_surface}, + Connection, Dispatch, Proxy, QueueHandle, WEnum, +}; +use wayland_protocols::ext::image_capture_source::v1::client::{ + ext_foreign_toplevel_image_capture_source_manager_v1, ext_image_capture_source_v1, + ext_output_image_capture_source_manager_v1, +}; +use wayland_protocols::ext::image_copy_capture::v1::client::{ + ext_image_copy_capture_cursor_session_v1, ext_image_copy_capture_frame_v1, + ext_image_copy_capture_manager_v1, ext_image_copy_capture_session_v1, +}; + +#[derive(Debug, Default, Clone)] +pub struct BufferConstraints { + pub size: (u32, u32), + pub shm_formats: Vec>, + pub dmabuf_device: Option, + pub dmabuf_formats: Vec<(u32, Vec)>, +} + +pub trait ImageCopySessionDataExt { + fn image_copy_session_data(&self) -> &ImageCopySessionData; +} + +#[derive(Default, Debug)] +pub struct ImageCopySessionData { + constraints: Mutex, +} + +impl ImageCopySessionDataExt for ImageCopySessionData { + fn image_copy_session_data(&self) -> &ImageCopySessionData { + self + } +} + +#[derive(Clone, Debug)] +pub struct Rect { + pub x: i32, + pub y: i32, + pub width: i32, + pub height: i32, +} + +#[derive(Clone, Debug)] +pub struct Frame { + pub transform: WEnum, + pub damage: Vec, + // TODO: Better type for this? + pub presentation_time: Option, +} + +impl Default for Frame { + fn default() -> Self { + Self { + transform: WEnum::Value(wl_output::Transform::Normal), + damage: Vec::new(), + presentation_time: None, + } + } +} + +#[derive(Default, Debug)] +pub struct ImageCopyFrameData { + frame: Mutex, +} + +pub trait ImageCopyFrameDataExt { + fn image_copy_frame_data(&self) -> &ImageCopyFrameData; +} + +impl ImageCopyFrameDataExt for ImageCopyFrameData { + fn image_copy_frame_data(&self) -> &ImageCopyFrameData { + self + } +} + +pub trait ImageCopyCaptureHandler: Sized { + fn image_copy_capture_state(&mut self) -> &mut ImageCopyCaptureState; + + fn buffer_constraints( + &mut self, + conn: &Connection, + qh: &QueueHandle, + session: &ext_image_copy_capture_session_v1::ExtImageCopyCaptureSessionV1, + constraints: BufferConstraints, + ); + + fn stopped( + &mut self, + conn: &Connection, + qh: &QueueHandle, + session: &ext_image_copy_capture_session_v1::ExtImageCopyCaptureSessionV1, + ); + + fn ready( + &mut self, + conn: &Connection, + qh: &QueueHandle, + image_copy_frame: &ext_image_copy_capture_frame_v1::ExtImageCopyCaptureFrameV1, + frame: Frame, + ); + + fn failed( + &mut self, + conn: &Connection, + qh: &QueueHandle, + image_copy_frame: &ext_image_copy_capture_frame_v1::ExtImageCopyCaptureFrameV1, + reason: WEnum, + ); + + fn cursor_enter( + &mut self, + _conn: &Connection, + _qh: &QueueHandle, + _cursor_session: &ext_image_copy_capture_cursor_session_v1::ExtImageCopyCaptureCursorSessionV1, + ) { + } + + fn cursor_leave( + &mut self, + _conn: &Connection, + _qh: &QueueHandle, + _cursor_session: &ext_image_copy_capture_cursor_session_v1::ExtImageCopyCaptureCursorSessionV1, + ) { + } + + fn cursor_position( + &mut self, + _conn: &Connection, + _qh: &QueueHandle, + _cursor_session: &ext_image_copy_capture_cursor_session_v1::ExtImageCopyCaptureCursorSessionV1, + _x: i32, + _y: i32, + ) { + } + + fn cursor_hotspot( + &mut self, + _conn: &Connection, + _qh: &QueueHandle, + _cursor_session: &ext_image_copy_capture_cursor_session_v1::ExtImageCopyCaptureCursorSessionV1, + _x: i32, + _y: i32, + ) { + } +} + +#[derive(Debug)] +pub struct ImageCopyCaptureState { + foreign_toplevel_source_manager: GlobalProxy, + output_source_manager: GlobalProxy, + capture_manager: GlobalProxy, +} + +impl ImageCopyCaptureState { + pub fn new(globals: &GlobalList, qh: &QueueHandle) -> Self + where + D: Dispatch, + D: Dispatch< + ext_output_image_capture_source_manager_v1::ExtOutputImageCaptureSourceManagerV1, + GlobalData>, + D: Dispatch, + D: 'static, + { + let foreign_toplevel_source_manager = + GlobalProxy::from(globals.bind(qh, 1..=1, GlobalData)); + let output_source_manager = GlobalProxy::from(globals.bind(qh, 1..=1, GlobalData)); + let capture_manager = GlobalProxy::from(globals.bind(qh, 1..=1, GlobalData)); + Self { foreign_toplevel_source_manager, output_source_manager, capture_manager } + } +} + +impl Dispatch + for ImageCopyCaptureState +where + D: Dispatch, +{ + fn event( + _: &mut D, + _: &ext_image_capture_source_v1::ExtImageCaptureSourceV1, + _: ext_image_capture_source_v1::Event, + _: &GlobalData, + _: &Connection, + _: &QueueHandle, + ) { + unreachable!() + } +} + +impl + Dispatch< + ext_output_image_capture_source_manager_v1::ExtOutputImageCaptureSourceManagerV1, + GlobalData, + D, + > for ImageCopyCaptureState +where + D: Dispatch< + ext_output_image_capture_source_manager_v1::ExtOutputImageCaptureSourceManagerV1, + GlobalData, + >, +{ + fn event( + _: &mut D, + _: &ext_output_image_capture_source_manager_v1::ExtOutputImageCaptureSourceManagerV1, + _: ext_output_image_capture_source_manager_v1::Event, + _: &GlobalData, + _: &Connection, + _: &QueueHandle, + ) { + unreachable!() + } +} + +impl Dispatch for ImageCopyCaptureState +where + D: Dispatch +{ + fn event( + _: &mut D, + _: &ext_foreign_toplevel_image_capture_source_manager_v1::ExtForeignToplevelImageCaptureSourceManagerV1, + _: ext_foreign_toplevel_image_capture_source_manager_v1::Event, + _: &GlobalData, + _: &Connection, + _: &QueueHandle, + ) { + unreachable!() + } +} + +impl Dispatch + for ImageCopyCaptureState +where + D: Dispatch, +{ + fn event( + _: &mut D, + _: &ext_image_copy_capture_manager_v1::ExtImageCopyCaptureManagerV1, + _: ext_image_copy_capture_manager_v1::Event, + _: &GlobalData, + _: &Connection, + _: &QueueHandle, + ) { + unreachable!() + } +} + +impl Dispatch + for ImageCopyCaptureState +where + D: Dispatch + + ImageCopyCaptureHandler, + U: ImageCopySessionDataExt, +{ + fn event( + state: &mut D, + proxy: &ext_image_copy_capture_session_v1::ExtImageCopyCaptureSessionV1, + event: ext_image_copy_capture_session_v1::Event, + udata: &U, + conn: &Connection, + qh: &QueueHandle, + ) { + let constraints = &udata.image_copy_session_data().constraints; + match event { + ext_image_copy_capture_session_v1::Event::BufferSize { width, height } => { + constraints.lock().unwrap().size = (width, height); + } + ext_image_copy_capture_session_v1::Event::ShmFormat { format } => { + constraints.lock().unwrap().shm_formats.push(format); + } + ext_image_copy_capture_session_v1::Event::DmabufDevice { device } => { + let device = dev_t::from_ne_bytes(device.try_into().unwrap()); + constraints.lock().unwrap().dmabuf_device = Some(device); + } + ext_image_copy_capture_session_v1::Event::DmabufFormat { format, modifiers } => { + let modifiers = modifiers + .chunks_exact(8) + .map(|x| u64::from_ne_bytes(x.try_into().unwrap())) + .collect(); + constraints.lock().unwrap().dmabuf_formats.push((format, modifiers)); + } + ext_image_copy_capture_session_v1::Event::Done => { + let constraints = constraints.lock().unwrap().clone(); + state.buffer_constraints(conn, qh, proxy, constraints); + } + ext_image_copy_capture_session_v1::Event::Stopped => { + state.stopped(conn, qh, proxy); + proxy.destroy(); + } + _ => unreachable!(), + } + } +} + +impl Dispatch + for ImageCopyCaptureState +where + D: Dispatch + + ImageCopyCaptureHandler, + U: ImageCopyFrameDataExt, +{ + fn event( + state: &mut D, + proxy: &ext_image_copy_capture_frame_v1::ExtImageCopyCaptureFrameV1, + event: ext_image_copy_capture_frame_v1::Event, + udata: &U, + conn: &Connection, + qh: &QueueHandle, + ) { + let frame = &udata.image_copy_frame_data().frame; + match event { + ext_image_copy_capture_frame_v1::Event::Transform { transform } => { + frame.lock().unwrap().transform = transform; + } + ext_image_copy_capture_frame_v1::Event::Damage { x, y, width, height } => { + frame.lock().unwrap().damage.push(Rect { x, y, width, height }); + } + ext_image_copy_capture_frame_v1::Event::PresentationTime { + tv_sec_hi, + tv_sec_lo, + tv_nsec, + } => { + let secs = (u64::from(tv_sec_hi) << 32) + u64::from(tv_sec_lo); + let duration = Duration::new(secs, tv_nsec); + frame.lock().unwrap().presentation_time = Some(duration); + } + ext_image_copy_capture_frame_v1::Event::Ready => { + let frame = frame.lock().unwrap().clone(); + state.ready(conn, qh, proxy, frame); + } + ext_image_copy_capture_frame_v1::Event::Failed { reason } => { + state.failed(conn, qh, proxy, reason); + proxy.destroy(); + } + _ => unreachable!(), + } + } +} + +// TODO +impl + Dispatch + for ImageCopyCaptureState +where + D: Dispatch + + ImageCopyCaptureHandler, +{ + fn event( + state: &mut D, + proxy: &ext_image_copy_capture_cursor_session_v1::ExtImageCopyCaptureCursorSessionV1, + event: ext_image_copy_capture_cursor_session_v1::Event, + _: &U, + conn: &Connection, + qh: &QueueHandle, + ) { + match event { + ext_image_copy_capture_cursor_session_v1::Event::Enter => { + state.cursor_enter(conn, qh, proxy); + } + ext_image_copy_capture_cursor_session_v1::Event::Leave => { + state.cursor_leave(conn, qh, proxy); + } + ext_image_copy_capture_cursor_session_v1::Event::Position { x, y } => { + state.cursor_position(conn, qh, proxy, x, y); + } + ext_image_copy_capture_cursor_session_v1::Event::Hotspot { x, y } => { + state.cursor_hotspot(conn, qh, proxy, x, y); + } + _ => unreachable!(), + } + } +} + +#[macro_export] +macro_rules! delegate_image_copy_capture { + ($(@<$( $lt:tt $( : $clt:tt $(+ $dlt:tt )* )? ),+>)? $ty: ty) => { + $crate::delegate_image_copy_capture($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty, + session: $crate::image_copy_capture::ImageCopySessionData, frame: $crate::image_copy_capture::ImageCopyFrameData, cursor_session: $crate::globals::GlobalData); + }; + ($(@<$( $lt:tt $( : $clt:tt $(+ $dlt:tt )* )? ),+>)? $ty: ty, session: [$($session_data:ty),* $(,)?], frame: [$($frame_data:ty),* $(,)?], cursor_session: [$($cursor_session_data:ty),* $(,)?]) => { + $crate::reexports::client::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [ + $crate::reexports::protocols::ext::image_capture_source::v1::client::ext_image_capture_source_v1::ExtImageCaptureSourceV1: $crate::globals::GlobalData + ] => $crate::image_copy_capture::ImageCopyCaptureState); + $crate::reexports::client::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [ + $crate::reexports::protocols::ext::image_capture_source::v1::client::ext_output_image_capture_source_manager_v1::ExtOutputImageCaptureSourceManagerV1: $crate::globals::GlobalData + ] => $crate::image_copy_capture::ImageCopyCaptureState); + $crate::reexports::client::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [ + $crate::reexports::protocols::ext::image_capture_source::v1::client::ext_foreign_toplevel_image_capture_source_manager_v1::ExtForeignToplevelImageCaptureSourceManagerV1: $crate::globals::GlobalData + ] => $crate::image_copy_capture::ImageCopyCaptureState); + $crate::reexports::client::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [ + $crate::reexports::protocols::ext::image_copy_capture::v1::client::ext_image_copy_capture_manager_v1::ExtImageCopyCaptureManagerV1: $crate::globals::GlobalData + ] => $crate::image_copy_capture::ImageCopyCaptureState); + $crate::reexports::client::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [ + $( + $crate::reexports::protocols::ext::image_copy_capture::v1::client::ext_image_copy_capture_session_v1::ExtImageCopyCaptureSessionV1: $session_data + ),* + ] => $crate::image_copy_capture::ImageCopyCaptureState); + $crate::reexports::client::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [ + $( + $crate::reexports::protocols::ext::image_copy_capture::v1::client::ext_image_copy_capture_frame_v1::ExtImageCopyCaptureFrameV1: $frame_data + ),* + ] => $crate::image_copy_capture::ImageCopyCaptureState); + $crate::reexports::client::delegate_dispatch!($(@< $( $lt $( : $clt $(+ $dlt )* )? ),+ >)? $ty: [ + $( + $crate::reexports::protocols::ext::image_copy_capture::v1::client::ext_image_copy_capture_cursor_session_v1::ExtImageCopyCaptureCursorSessionV1: $cursor_session_data + ),* + ] => $crate::image_copy_capture::ImageCopyCaptureState); + } +} diff --git a/src/lib.rs b/src/lib.rs index b0a1435ee..f41a22dbf 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -25,6 +25,7 @@ pub mod dmabuf; pub mod error; pub mod foreign_toplevel_list; pub mod globals; +pub mod image_copy_capture; pub mod output; pub mod primary_selection; pub mod registry;