P0 - UDP send buffer: set SO_SNDBUF=2MB to prevent EAGAIN on large IDR frames (218KB/256KB keyframes caused 18+ EAGAIN bursts). Actual Linux buffer 4096KB confirmed. P1 - VBV rate limiting: cap rc_max_rate=bitrate and rc_buffer_size= bitrate/4 for WebRTC encode path, preventing oversized IDR frames. Stats: integrate PipelineStats into cap_portal (dropped_count), state.rs (wlroots path), webrtc.rs (browser getStats enhancement + stats panel).
1629 lines
60 KiB
Rust
1629 lines
60 KiB
Rust
use std::collections::HashMap;
|
|
use std::mem;
|
|
use std::os::fd::{AsFd, OwnedFd};
|
|
use std::os::unix::io::FromRawFd;
|
|
use std::path::{Path, PathBuf};
|
|
use std::sync::atomic::{AtomicBool, Ordering};
|
|
use std::sync::Arc;
|
|
use std::time::Instant;
|
|
|
|
use anyhow::Result;
|
|
use wayland_client::backend::ObjectId;
|
|
use wayland_client::globals::{GlobalList, GlobalListContents};
|
|
use wayland_client::protocol::wl_buffer::WlBuffer;
|
|
use wayland_client::protocol::wl_output::WlOutput;
|
|
use wayland_client::protocol::wl_registry::WlRegistry;
|
|
use wayland_client::{event_created_child, Dispatch, Proxy, QueueHandle};
|
|
use wayland_protocols::wp::linux_dmabuf::zv1::client::zwp_linux_buffer_params_v1::{
|
|
Event as BufferParamsEvent, Flags as BufferParamsFlags, ZwpLinuxBufferParamsV1,
|
|
};
|
|
use wayland_protocols::wp::linux_dmabuf::zv1::client::zwp_linux_dmabuf_feedback_v1::{
|
|
Event as DmabufFeedbackEvent, ZwpLinuxDmabufFeedbackV1,
|
|
};
|
|
use wayland_protocols::wp::linux_dmabuf::zv1::client::zwp_linux_dmabuf_v1::{
|
|
Event as DmabufEvent, ZwpLinuxDmabufV1,
|
|
};
|
|
use wayland_protocols::xdg::xdg_output::zv1::client::zxdg_output_manager_v1::ZxdgOutputManagerV1;
|
|
use wayland_protocols::xdg::xdg_output::zv1::client::zxdg_output_v1::{
|
|
Event as XdgOutputEvent, ZxdgOutputV1,
|
|
};
|
|
use wayland_protocols_wlr::output_management::v1::client::zwlr_output_head_v1::{
|
|
self, Event as WlrHeadEvent, ZwlrOutputHeadV1,
|
|
};
|
|
use wayland_protocols_wlr::output_management::v1::client::zwlr_output_manager_v1::{
|
|
self, Event as WlrOutputManagerEvent, ZwlrOutputManagerV1,
|
|
};
|
|
use wayland_protocols_wlr::output_management::v1::client::zwlr_output_mode_v1::ZwlrOutputModeV1;
|
|
use wayland_protocols_wlr::screencopy::v1::client::zwlr_screencopy_frame_v1::{
|
|
Event as ScreencopyFrameEvent, ZwlrScreencopyFrameV1,
|
|
};
|
|
use wayland_protocols_wlr::screencopy::v1::client::zwlr_screencopy_manager_v1::ZwlrScreencopyManagerV1;
|
|
|
|
use ffmpeg_next as ff;
|
|
use ffmpeg_next::ffi;
|
|
|
|
use crate::args::Args;
|
|
use crate::avhw::{AvHwDevCtx, EncState, SwEncState};
|
|
use crate::cap_wlr_screencopy::CapWlrScreencopy;
|
|
use crate::fps_limit::FpsLimit;
|
|
use crate::stats::{FrameTimings, PipelineStats};
|
|
use crate::transform::{transpose_if_transform_transposed, Transform};
|
|
use crate::webrtc::WebRtcState;
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// CaptureSource trait
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/// Screen capture backend trait.
|
|
pub trait CaptureSource: Sized + 'static {
|
|
type Frame: Send;
|
|
|
|
fn new(
|
|
gm: &GlobalList,
|
|
output: &WlOutput,
|
|
output_info: &OutputInfo,
|
|
qh: &QueueHandle<State<Self>>,
|
|
) -> Result<Self>;
|
|
|
|
fn alloc_frame(&mut self) -> Option<Self::Frame>;
|
|
|
|
fn queue_copy(&mut self, buffer: &WlBuffer, qh: &QueueHandle<State<Self>>);
|
|
|
|
fn on_done_with_frame(&mut self, frame: Self::Frame);
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Output info types
|
|
// ---------------------------------------------------------------------------
|
|
|
|
pub struct OutputInfo {
|
|
pub name: String,
|
|
pub transform: Transform,
|
|
pub physical_size: (i32, i32),
|
|
pub logical_position: (i32, i32),
|
|
}
|
|
|
|
pub struct PartialOutputInfo {
|
|
pub name: Option<String>,
|
|
/// Name from wl_output::Name (v4) — used to match wlr-output-management heads
|
|
pub wl_name: Option<String>,
|
|
pub transform: Option<Transform>,
|
|
pub physical_size: Option<(i32, i32)>,
|
|
pub logical_position: Option<(i32, i32)>,
|
|
// Pixel dimensions from Mode event — preparatory for Phase 2 resolution logic
|
|
pub mode_size: Option<(i32, i32)>,
|
|
pub done_count: u32,
|
|
}
|
|
|
|
impl Default for PartialOutputInfo {
|
|
fn default() -> Self {
|
|
Self {
|
|
name: None,
|
|
wl_name: None,
|
|
transform: None,
|
|
physical_size: None,
|
|
logical_position: None,
|
|
mode_size: None,
|
|
done_count: 0,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Stores head info from wlr-output-management for name-based matching with wl_output.
|
|
struct WlrHeadInfo {
|
|
position: Option<(i32, i32)>,
|
|
}
|
|
|
|
/// User data for XdgOutput dispatch to identify which WlOutput it belongs to.
|
|
pub struct OutputId(pub u32);
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// StreamingEncoder
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/// Wraps the two possible encoder backends for the streaming stage.
|
|
///
|
|
/// - `Mp4(EncState)` — hardware VAAPI encoder writing to an MP4 file
|
|
/// - `WebRtc(SwEncState)` — software encoder feeding H.264 NALUs into a WebRTC channel
|
|
pub enum StreamingEncoder {
|
|
Mp4(EncState),
|
|
WebRtc(SwEncState),
|
|
}
|
|
|
|
impl StreamingEncoder {
|
|
fn frames_rgb(&self) -> &crate::avhw::AvHwFrameCtx {
|
|
match self {
|
|
StreamingEncoder::Mp4(enc) => enc.frames_rgb(),
|
|
StreamingEncoder::WebRtc(enc) => enc.frames_rgb(),
|
|
}
|
|
}
|
|
|
|
fn encode_frame(&mut self, hw_frame: &ffmpeg_next::frame::Video) -> anyhow::Result<()> {
|
|
match self {
|
|
StreamingEncoder::Mp4(enc) => enc.encode_frame(hw_frame),
|
|
StreamingEncoder::WebRtc(enc) => enc.encode_frame(hw_frame),
|
|
}
|
|
}
|
|
|
|
pub fn flush(&mut self) -> anyhow::Result<()> {
|
|
match self {
|
|
StreamingEncoder::Mp4(enc) => enc.flush(),
|
|
StreamingEncoder::WebRtc(enc) => enc.flush(),
|
|
}
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// EncConstructionStage
|
|
// ---------------------------------------------------------------------------
|
|
|
|
pub enum EncConstructionStage<S: CaptureSource> {
|
|
ProbingOutputs {
|
|
outputs: Vec<PartialOutputInfo>,
|
|
bound_outputs: Vec<WlOutput>,
|
|
output_names: Vec<u32>,
|
|
screencopy_manager: Option<ZwlrScreencopyManagerV1>,
|
|
dmabuf: Option<ZwpLinuxDmabufV1>,
|
|
dmabuf_feedback: Option<ZwpLinuxDmabufFeedbackV1>,
|
|
xdg_output_manager: Option<ZxdgOutputManagerV1>,
|
|
wlr_output_manager: Option<ZwlrOutputManagerV1>,
|
|
wlr_manager_done: bool,
|
|
wlr_heads: HashMap<String, WlrHeadInfo>,
|
|
wlr_head_proxy_to_name: HashMap<ObjectId, String>,
|
|
},
|
|
EverythingButFmt {
|
|
output_info: OutputInfo,
|
|
output: WlOutput,
|
|
hw_device_ctx: AvHwDevCtx,
|
|
cap: S,
|
|
screencopy_manager: ZwlrScreencopyManagerV1,
|
|
dmabuf: ZwpLinuxDmabufV1,
|
|
},
|
|
Streaming {
|
|
output_info: OutputInfo,
|
|
output: WlOutput,
|
|
enc: StreamingEncoder,
|
|
cap: S,
|
|
screencopy_manager: ZwlrScreencopyManagerV1,
|
|
dmabuf: ZwpLinuxDmabufV1,
|
|
},
|
|
Intermediate,
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// InFlightSurface
|
|
// ---------------------------------------------------------------------------
|
|
|
|
pub enum InFlightSurface<S: CaptureSource> {
|
|
None,
|
|
AllocQueued,
|
|
Allocd(S::Frame),
|
|
CopyQueued {
|
|
surface: ff::frame::Video,
|
|
drm_map: ff::ffi::AVDRMFrameDescriptor,
|
|
frame: S::Frame,
|
|
buffer: WlBuffer,
|
|
},
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// State
|
|
// ---------------------------------------------------------------------------
|
|
|
|
pub struct State<S: CaptureSource> {
|
|
pub stage: EncConstructionStage<S>,
|
|
pub in_flight_surface: InFlightSurface<S>,
|
|
pub starting_timestamp: Option<i64>,
|
|
pub stats_start_time: Option<Instant>,
|
|
pub stats_last_time: Option<Instant>,
|
|
pub stats_frames: u64,
|
|
pub first_frame: bool,
|
|
pub args: Args,
|
|
pub errored: bool,
|
|
pub gm: GlobalList,
|
|
pub fps_limit: FpsLimit<S::Frame>,
|
|
pub qhandle: QueueHandle<State<S>>,
|
|
pub drm_device: Option<PathBuf>,
|
|
pub drm_device_from_compositor: Option<PathBuf>,
|
|
pub webrtc: Option<WebRtcState>,
|
|
pub webrtc_tx: Option<crossbeam_channel::Sender<Vec<u8>>>,
|
|
webrtc_rx: Option<crossbeam_channel::Receiver<Vec<u8>>>,
|
|
webrtc_frames_sent: u64,
|
|
webrtc_paused: Option<Arc<AtomicBool>>,
|
|
stats: PipelineStats,
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Helpers
|
|
// ---------------------------------------------------------------------------
|
|
|
|
/// Scan /dev/dri for all available DRM render nodes (renderD*), sorted by node number.
|
|
pub(crate) fn find_drm_render_nodes() -> Vec<PathBuf> {
|
|
let Ok(entries) = std::fs::read_dir("/dev/dri") else {
|
|
return Vec::new();
|
|
};
|
|
|
|
let mut nodes: Vec<(u32, PathBuf)> = entries
|
|
.filter_map(Result::ok)
|
|
.filter_map(|entry| {
|
|
let path = entry.path();
|
|
let name = path.file_name()?.to_str()?;
|
|
let number = name.strip_prefix("renderD")?.parse::<u32>().ok()?;
|
|
std::fs::metadata(&path).ok()?;
|
|
Some((number, path))
|
|
})
|
|
.collect();
|
|
nodes.sort_by_key(|(number, _)| *number);
|
|
nodes.into_iter().map(|(_, path)| path).collect()
|
|
}
|
|
|
|
/// Scan /dev/dri for the first available DRM render node (renderD*).
|
|
fn find_drm_render_node() -> Option<PathBuf> {
|
|
find_drm_render_nodes().into_iter().next()
|
|
}
|
|
|
|
impl<S: CaptureSource> State<S> {
|
|
fn resolve_drm_path(&self) -> PathBuf {
|
|
self.drm_device
|
|
.clone()
|
|
.or_else(|| self.drm_device_from_compositor.clone())
|
|
.or_else(find_drm_render_node)
|
|
.unwrap_or_else(|| PathBuf::from("/dev/dri/renderD128"))
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// State<S> methods
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> State<S> {
|
|
pub fn new(gm: GlobalList, args: Args, qhandle: QueueHandle<State<S>>) -> Result<Self> {
|
|
let fps = args.fps;
|
|
let drm_device = args.drm_device.as_ref().map(PathBuf::from);
|
|
|
|
let (webrtc, webrtc_tx, webrtc_rx, webrtc_paused) = if args.port > 0 {
|
|
let (tx, rx) = crossbeam_channel::bounded(32);
|
|
let wrtc = WebRtcState::new(args.port, args.fps)?;
|
|
// paused=true until first WebRTC client connects
|
|
let paused = Arc::new(AtomicBool::new(true));
|
|
(Some(wrtc), Some(tx), Some(rx), Some(paused))
|
|
} else {
|
|
(None, None, None, None)
|
|
};
|
|
|
|
let mut state = Self {
|
|
stage: EncConstructionStage::ProbingOutputs {
|
|
outputs: Vec::new(),
|
|
bound_outputs: Vec::new(),
|
|
output_names: Vec::new(),
|
|
screencopy_manager: None,
|
|
dmabuf: None,
|
|
dmabuf_feedback: None,
|
|
xdg_output_manager: None,
|
|
wlr_output_manager: None,
|
|
wlr_manager_done: false,
|
|
wlr_heads: HashMap::new(),
|
|
wlr_head_proxy_to_name: HashMap::new(),
|
|
},
|
|
in_flight_surface: InFlightSurface::None,
|
|
starting_timestamp: None,
|
|
stats_start_time: None,
|
|
stats_last_time: None,
|
|
stats_frames: 0,
|
|
first_frame: true,
|
|
fps_limit: FpsLimit::new(fps),
|
|
args,
|
|
errored: false,
|
|
gm,
|
|
qhandle,
|
|
drm_device,
|
|
drm_device_from_compositor: None,
|
|
webrtc,
|
|
webrtc_tx,
|
|
webrtc_rx,
|
|
webrtc_frames_sent: 0,
|
|
webrtc_paused,
|
|
stats: PipelineStats::new(),
|
|
};
|
|
|
|
// registry_queue_init consumes registry events internally during its
|
|
// initial roundtrip and does NOT forward them to our Dispatch impl.
|
|
// We must manually bind the initial globals here.
|
|
state.bind_initial_globals();
|
|
|
|
Ok(state)
|
|
}
|
|
|
|
/// Iterate over the GlobalList from registry_queue_init and bind all
|
|
/// globals we care about. This is necessary because registry_queue_init
|
|
/// consumes registry events during its internal roundtrip without forwarding
|
|
/// them to our Dispatch<WlRegistry> handler.
|
|
fn bind_initial_globals(&mut self) {
|
|
use wayland_client::globals::Global;
|
|
|
|
let globals: Vec<Global> = self.gm.contents().clone_list();
|
|
let registry = self.gm.registry();
|
|
let qhandle = &self.qhandle;
|
|
|
|
// Sort globals so that managers are bound BEFORE wl_output.
|
|
// This ensures xdg_output_manager and zwlr_output_manager are available
|
|
// when we bind wl_output, so we can immediately get xdg_output / wlr head.
|
|
let globals = {
|
|
fn priority(interface: &str) -> u8 {
|
|
match interface {
|
|
"zwlr_screencopy_manager_v1" => 0,
|
|
"zwp_linux_dmabuf_v1" => 0,
|
|
"zxdg_output_manager_v1" => 1,
|
|
"zwlr_output_manager_v1" => 1,
|
|
"wl_output" => 2,
|
|
_ => 3,
|
|
}
|
|
}
|
|
let mut g = globals;
|
|
g.sort_by_key(|g| priority(&g.interface));
|
|
g
|
|
};
|
|
|
|
for Global {
|
|
name,
|
|
interface,
|
|
version,
|
|
} in globals
|
|
{
|
|
match interface.as_str() {
|
|
"zwlr_screencopy_manager_v1" => {
|
|
let v = version.min(3);
|
|
tracing::debug!("Init: binding zwlr_screencopy_manager_v1 v{v} (name={name})");
|
|
let mgr: ZwlrScreencopyManagerV1 = registry.bind(name, v, qhandle, ());
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
screencopy_manager, ..
|
|
} = &mut self.stage
|
|
{
|
|
*screencopy_manager = Some(mgr);
|
|
}
|
|
}
|
|
"zwp_linux_dmabuf_v1" => {
|
|
let v = version.min(4);
|
|
tracing::debug!("Init: binding zwp_linux_dmabuf_v1 v{v} (name={name})");
|
|
let proxy: ZwpLinuxDmabufV1 = registry.bind(name, v, qhandle, ());
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
dmabuf,
|
|
dmabuf_feedback,
|
|
..
|
|
} = &mut self.stage
|
|
{
|
|
*dmabuf = Some(proxy.clone());
|
|
if v >= 4 {
|
|
let feedback = proxy.get_default_feedback(qhandle, ());
|
|
*dmabuf_feedback = Some(feedback);
|
|
}
|
|
}
|
|
}
|
|
"zxdg_output_manager_v1" => {
|
|
let v = version.min(3);
|
|
tracing::debug!("Init: binding zxdg_output_manager_v1 v{v} (name={name})");
|
|
let xdg_mgr: ZxdgOutputManagerV1 = registry.bind(name, v, qhandle, ());
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
bound_outputs,
|
|
xdg_output_manager,
|
|
output_names,
|
|
..
|
|
} = &mut self.stage
|
|
{
|
|
for (i, output) in bound_outputs.iter().enumerate() {
|
|
let oname = output_names.get(i).copied().unwrap_or(0);
|
|
let output_id = OutputId(oname);
|
|
xdg_mgr.get_xdg_output(output, qhandle, output_id);
|
|
}
|
|
*xdg_output_manager = Some(xdg_mgr);
|
|
}
|
|
}
|
|
"zwlr_output_manager_v1" => {
|
|
let v = version.min(4);
|
|
tracing::debug!("Init: binding zwlr_output_manager_v1 v{v} (name={name})");
|
|
let mgr: ZwlrOutputManagerV1 = registry.bind(name, v, qhandle, ());
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
wlr_output_manager, ..
|
|
} = &mut self.stage
|
|
{
|
|
*wlr_output_manager = Some(mgr);
|
|
}
|
|
}
|
|
"wl_output" => {
|
|
let v = version.min(4);
|
|
tracing::debug!("Init: binding wl_output v{v} (name={name})");
|
|
let output: WlOutput = registry.bind(name, v, qhandle, OutputId(name));
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
outputs,
|
|
bound_outputs,
|
|
output_names,
|
|
xdg_output_manager,
|
|
..
|
|
} = &mut self.stage
|
|
{
|
|
outputs.push(PartialOutputInfo::default());
|
|
bound_outputs.push(output.clone());
|
|
output_names.push(name);
|
|
if let Some(xdg_mgr) = xdg_output_manager {
|
|
let output_id = OutputId(name);
|
|
xdg_mgr.get_xdg_output(&output, qhandle, output_id);
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn queue_alloc_frame(&mut self)
|
|
where
|
|
State<S>: Dispatch<ZwlrScreencopyFrameV1, ()>,
|
|
{
|
|
let (manager, output) = match &self.stage {
|
|
EncConstructionStage::Streaming {
|
|
screencopy_manager,
|
|
output,
|
|
..
|
|
} => (screencopy_manager.clone(), output.clone()),
|
|
EncConstructionStage::EverythingButFmt {
|
|
screencopy_manager,
|
|
output,
|
|
..
|
|
} => (screencopy_manager.clone(), output.clone()),
|
|
_ => return,
|
|
};
|
|
match &self.in_flight_surface {
|
|
InFlightSurface::None => {}
|
|
_ => return,
|
|
}
|
|
let _frame_proxy = manager.capture_output(1, &output, &self.qhandle, ());
|
|
self.in_flight_surface = InFlightSurface::AllocQueued;
|
|
}
|
|
|
|
pub fn on_frame_allocd(&mut self, frame: S::Frame, format: u32, width: u32, height: u32) {
|
|
let (frames_rgb_ctx, dmabuf, cap) = match &mut self.stage {
|
|
EncConstructionStage::Streaming {
|
|
output_info: _,
|
|
output: _,
|
|
enc,
|
|
dmabuf,
|
|
cap,
|
|
screencopy_manager: _,
|
|
} => (enc.frames_rgb().as_ptr(), dmabuf, cap),
|
|
_ => {
|
|
tracing::warn!("on_frame_allocd: not in Streaming stage");
|
|
return;
|
|
}
|
|
};
|
|
|
|
let mut surface = ff::frame::Video::empty();
|
|
// SAFETY: frames_rgb_ctx is a valid AVHWFramesContext pointer; surface
|
|
// is a freshly allocated empty Video frame.
|
|
let ret = unsafe { ffi::av_hwframe_get_buffer(frames_rgb_ctx, surface.as_mut_ptr(), 0) };
|
|
if ret < 0 {
|
|
tracing::error!("av_hwframe_get_buffer failed: {}", crate::avhw::ff_err(ret));
|
|
self.errored = true;
|
|
return;
|
|
}
|
|
|
|
let mut map_frame = ff::frame::Video::empty();
|
|
// SAFETY: Setting format to DRM_PRIME and calling av_hwframe_map creates
|
|
// a mapped view of the GPU surface with DMA-BUF file descriptors.
|
|
unsafe {
|
|
(*map_frame.as_mut_ptr()).format = ffi::AVPixelFormat::AV_PIX_FMT_DRM_PRIME as i32;
|
|
}
|
|
let ret = unsafe { ffi::av_hwframe_map(map_frame.as_mut_ptr(), surface.as_ptr(), 0) };
|
|
if ret < 0 {
|
|
tracing::error!("av_hwframe_map failed: {}", crate::avhw::ff_err(ret));
|
|
self.errored = true;
|
|
return;
|
|
}
|
|
|
|
// SAFETY: After av_hwframe_map with DRM_PRIME format, data[0] points to
|
|
// a valid AVDRMFrameDescriptor.
|
|
let desc: ff::ffi::AVDRMFrameDescriptor = unsafe {
|
|
let desc_ptr = (*map_frame.as_ptr()).data[0] as *const ff::ffi::AVDRMFrameDescriptor;
|
|
std::ptr::read(desc_ptr)
|
|
};
|
|
|
|
let params = dmabuf.create_params(&self.qhandle, ());
|
|
|
|
for layer_idx in 0..desc.nb_layers as usize {
|
|
let layer = &desc.layers[layer_idx];
|
|
for p in 0..layer.nb_planes as usize {
|
|
let plane = &layer.planes[p];
|
|
let obj = &desc.objects[plane.object_index as usize];
|
|
let mod_hi = (obj.format_modifier >> 32) as u32;
|
|
let mod_lo = (obj.format_modifier & 0xFFFF_FFFF) as u32;
|
|
// SAFETY: obj.fd is a valid DMA-BUF fd. We dup because params.add()
|
|
// takes ownership of the fd, and the original fd is owned by map_frame.
|
|
let fd_dup = unsafe { libc::dup(obj.fd) };
|
|
if fd_dup < 0 {
|
|
tracing::error!("failed to dup dma-buf fd: {}", std::io::Error::last_os_error());
|
|
// wayland-client does not auto-destroy params on Drop.
|
|
params.destroy();
|
|
self.errored = true;
|
|
return;
|
|
}
|
|
// SAFETY: fd_dup is valid freshly-duped fd.
|
|
let fd_owned = unsafe { OwnedFd::from_raw_fd(fd_dup) };
|
|
params.add(
|
|
fd_owned.as_fd(),
|
|
p as u32,
|
|
plane.offset as u32,
|
|
plane.pitch as u32,
|
|
mod_hi,
|
|
mod_lo,
|
|
);
|
|
}
|
|
}
|
|
|
|
let wl_buffer = params.create_immed(
|
|
width as i32,
|
|
height as i32,
|
|
format,
|
|
BufferParamsFlags::empty(),
|
|
&self.qhandle,
|
|
(),
|
|
);
|
|
self.in_flight_surface = InFlightSurface::CopyQueued {
|
|
surface,
|
|
drm_map: desc,
|
|
frame,
|
|
buffer: wl_buffer,
|
|
};
|
|
let buffer_ref = match &self.in_flight_surface {
|
|
InFlightSurface::CopyQueued { buffer, .. } => buffer,
|
|
_ => unreachable!("just set to CopyQueued"),
|
|
};
|
|
cap.queue_copy(buffer_ref, &self.qhandle);
|
|
}
|
|
|
|
pub fn on_copy_complete(&mut self, tv_sec: u64, tv_usec: u32)
|
|
where
|
|
S::Frame: Default,
|
|
{
|
|
self.stats.record_capture();
|
|
|
|
let (mut surface, _drm_map, frame, buffer) =
|
|
match mem::replace(&mut self.in_flight_surface, InFlightSurface::None) {
|
|
InFlightSurface::CopyQueued {
|
|
surface,
|
|
drm_map,
|
|
frame,
|
|
buffer,
|
|
} => (surface, drm_map, frame, buffer),
|
|
other => {
|
|
tracing::warn!("on_copy_complete: unexpected state");
|
|
self.in_flight_surface = other;
|
|
return;
|
|
}
|
|
};
|
|
let fps = self.args.fps as i64;
|
|
// PTS in frame-number units (encoder time_base = 1/fps)
|
|
let pts = (tv_sec as i64) * fps + (tv_usec as i64) * fps / 1_000_000;
|
|
surface.set_pts(Some(pts));
|
|
drop(buffer);
|
|
let cap = match &mut self.stage {
|
|
EncConstructionStage::Streaming { cap, .. } => cap,
|
|
_ => {
|
|
tracing::warn!("on_copy_complete: not in Streaming stage");
|
|
return;
|
|
}
|
|
};
|
|
cap.on_done_with_frame(frame);
|
|
let enc = match &mut self.stage {
|
|
EncConstructionStage::Streaming { enc, .. } => enc,
|
|
_ => unreachable!("already checked Streaming above"),
|
|
};
|
|
let should_encode = if self.first_frame {
|
|
self.first_frame = false;
|
|
true
|
|
} else {
|
|
self.fps_limit
|
|
.on_new_frame(S::Frame::default(), Instant::now())
|
|
.is_some()
|
|
};
|
|
if should_encode {
|
|
let encode_start = Instant::now();
|
|
if let Err(e) = enc.encode_frame(&surface) {
|
|
tracing::error!("encode_frame failed: {}", e);
|
|
self.errored = true;
|
|
}
|
|
let encode_elapsed = encode_start.elapsed().as_micros() as u64;
|
|
self.stats.record_encode(&FrameTimings {
|
|
total_us: encode_elapsed,
|
|
..Default::default()
|
|
});
|
|
}
|
|
self.stats_frames += 1;
|
|
if let Some(last) = self.stats_last_time {
|
|
if last.elapsed() >= std::time::Duration::from_secs(10) {
|
|
let delta = self.stats_frames;
|
|
let fps = delta as f64 / last.elapsed().as_secs_f64();
|
|
tracing::info!(frames = self.stats_frames, fps = format!("{fps:.1}"), "encoding stats");
|
|
self.stats_last_time = Some(std::time::Instant::now());
|
|
self.stats_frames = 0;
|
|
}
|
|
} else {
|
|
self.stats_start_time = Some(std::time::Instant::now());
|
|
self.stats_last_time = Some(std::time::Instant::now());
|
|
}
|
|
}
|
|
|
|
pub fn on_copy_fail(&mut self)
|
|
where
|
|
S::Frame: Default,
|
|
{
|
|
tracing::error!("compositor copy failed");
|
|
let taken = mem::replace(&mut self.in_flight_surface, InFlightSurface::None);
|
|
match taken {
|
|
InFlightSurface::CopyQueued { buffer, frame, .. } => {
|
|
drop(buffer);
|
|
if let EncConstructionStage::Streaming { cap, .. } = &mut self.stage {
|
|
cap.on_done_with_frame(frame);
|
|
}
|
|
}
|
|
other => {
|
|
self.in_flight_surface = other;
|
|
}
|
|
}
|
|
self.errored = true;
|
|
}
|
|
|
|
pub fn poll_webrtc(&mut self) -> Result<()> {
|
|
let Some(ref mut wrtc) = self.webrtc else { return Ok(()) };
|
|
|
|
wrtc.handle_signaling()?;
|
|
wrtc.poll_and_feed()?;
|
|
|
|
let connected = wrtc.is_connected();
|
|
|
|
if let Some(ref paused) = self.webrtc_paused {
|
|
let was_paused = paused.load(Ordering::Relaxed);
|
|
let now_paused = !connected;
|
|
if was_paused && !now_paused {
|
|
tracing::info!("WebRTC client connected, resuming encoding");
|
|
} else if !was_paused && now_paused {
|
|
tracing::warn!("WebRTC client disconnected, pausing encoding");
|
|
}
|
|
paused.store(now_paused, Ordering::Relaxed);
|
|
}
|
|
|
|
if let Some(ref rx) = self.webrtc_rx {
|
|
let mut count = 0u32;
|
|
while let Ok(data) = rx.try_recv() {
|
|
if !connected {
|
|
continue;
|
|
}
|
|
count += 1;
|
|
if let Err(e) = wrtc.write_h264_frame(&data, self.webrtc_frames_sent, self.args.fps) {
|
|
tracing::debug!("WebRTC write frame error: {e}");
|
|
}
|
|
self.stats.record_send(0.0, None);
|
|
self.webrtc_frames_sent = self.webrtc_frames_sent.saturating_add(1);
|
|
}
|
|
if count > 0 {
|
|
tracing::debug!("WebRTC forwarded {count} frames from channel");
|
|
}
|
|
}
|
|
|
|
if self.args.stats && self.stats.should_snapshot() {
|
|
self.stats.set_queue_depths(
|
|
0,
|
|
self.webrtc_rx.as_ref().map(|r| r.len()).unwrap_or(0),
|
|
);
|
|
let snap = self.stats.snapshot_and_reset();
|
|
tracing::info!("stats: {snap}");
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
pub fn negotiate_format(&mut self, format: u32, width: u32, height: u32) {
|
|
let stage_data = match mem::replace(&mut self.stage, EncConstructionStage::Intermediate) {
|
|
EncConstructionStage::EverythingButFmt {
|
|
output_info,
|
|
output,
|
|
hw_device_ctx,
|
|
cap,
|
|
screencopy_manager,
|
|
dmabuf,
|
|
} => (
|
|
output_info,
|
|
output,
|
|
hw_device_ctx,
|
|
cap,
|
|
screencopy_manager,
|
|
dmabuf,
|
|
),
|
|
other => {
|
|
tracing::warn!("negotiate_format: not in EverythingButFmt stage");
|
|
self.stage = other;
|
|
return;
|
|
}
|
|
};
|
|
let (output_info, output, hw_device_ctx, cap, screencopy_manager, dmabuf) = stage_data;
|
|
let drm_path = self.resolve_drm_path();
|
|
let fps = self.args.fps;
|
|
let bitrate = self
|
|
.args
|
|
.bitrate
|
|
.unwrap_or_else(|| 2 * (width as u64) * (height as u64) * (fps as u64) / 100);
|
|
|
|
let enc = if let Some(ref tx) = self.webrtc_tx {
|
|
let (enc_w, enc_h) =
|
|
transpose_if_transform_transposed(output_info.transform, width as i32, height as i32);
|
|
let actual_gop_size = self.args.gop_size.unwrap_or((fps / 2).max(10));
|
|
match SwEncState::new_webrtc(
|
|
&drm_path,
|
|
width,
|
|
height,
|
|
enc_w as u32,
|
|
enc_h as u32,
|
|
fps,
|
|
bitrate,
|
|
actual_gop_size,
|
|
tx.clone(),
|
|
self.webrtc_paused.as_ref().expect("webrtc_paused must exist when webrtc_tx exists").clone(),
|
|
) {
|
|
Ok(enc) => StreamingEncoder::WebRtc(enc),
|
|
Err(e) => {
|
|
tracing::error!("SwEncState::new_webrtc failed: {}", e);
|
|
self.errored = true;
|
|
return;
|
|
}
|
|
}
|
|
} else {
|
|
let output_path = self.args.output.as_deref().expect("output required for MP4 mode");
|
|
match crate::avhw::create_encoder(
|
|
&drm_path,
|
|
Path::new(output_path),
|
|
width,
|
|
height,
|
|
fps,
|
|
output_info.transform,
|
|
self.args.bitrate,
|
|
self.args.gop_size,
|
|
Some(hw_device_ctx),
|
|
) {
|
|
Ok(enc) => StreamingEncoder::Mp4(enc),
|
|
Err(e) => {
|
|
tracing::error!("EncState::new failed: {}", e);
|
|
self.errored = true;
|
|
return;
|
|
}
|
|
}
|
|
};
|
|
tracing::info!(
|
|
"Encoder initialized: {}x{} format={} bitrate={}",
|
|
width,
|
|
height,
|
|
format,
|
|
bitrate
|
|
);
|
|
self.stage = EncConstructionStage::Streaming {
|
|
output_info,
|
|
output,
|
|
enc,
|
|
cap,
|
|
screencopy_manager,
|
|
dmabuf,
|
|
};
|
|
}
|
|
|
|
fn try_finalize_output(&mut self, _idx: usize) -> bool {
|
|
// Merge wlr head position info into outputs (needed for niri path)
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
outputs, wlr_heads, ..
|
|
} = &mut self.stage
|
|
{
|
|
for info in outputs.iter_mut() {
|
|
if info.logical_position.is_none() {
|
|
if let Some(ref wl_name) = info.wl_name {
|
|
if let Some(head_info) = wlr_heads.get(wl_name) {
|
|
info.logical_position = head_info.position;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
let (target_idx, output_count) = match &self.stage {
|
|
EncConstructionStage::ProbingOutputs {
|
|
outputs,
|
|
xdg_output_manager,
|
|
wlr_manager_done,
|
|
..
|
|
} => {
|
|
let has_xdg = xdg_output_manager.is_some();
|
|
let output_count = outputs.len();
|
|
let idx = if let Some(ref name) = self.args.output_name {
|
|
let pos = outputs
|
|
.iter()
|
|
.position(|o| o.name.as_deref() == Some(name.as_str()));
|
|
match pos {
|
|
Some(i) => Some(i),
|
|
None => {
|
|
let all_probed = outputs.iter().all(|o| o.done_count >= 1);
|
|
if all_probed {
|
|
let available: Vec<&str> =
|
|
outputs.iter().filter_map(|o| o.name.as_deref()).collect();
|
|
tracing::error!(
|
|
"Output '{}' not found. Available outputs: {:?}",
|
|
name,
|
|
available
|
|
);
|
|
self.errored = true;
|
|
}
|
|
None
|
|
}
|
|
}
|
|
} else if outputs.iter().all(|o| o.done_count >= 1) {
|
|
if outputs.is_empty() {
|
|
return false;
|
|
}
|
|
Some(0)
|
|
} else {
|
|
None
|
|
};
|
|
match idx {
|
|
Some(i) => {
|
|
let info = &outputs[i];
|
|
if has_xdg {
|
|
// xdg-output path (Sway/Hyprland) — strict checks
|
|
if info.done_count < 2
|
|
|| info.name.is_none()
|
|
|| info.transform.is_none()
|
|
|| info.physical_size.is_none()
|
|
|| info.logical_position.is_none()
|
|
{
|
|
return false;
|
|
}
|
|
} else {
|
|
// wlr-output-management path (niri) — relaxed checks
|
|
if info.done_count < 1 || !wlr_manager_done {
|
|
return false;
|
|
}
|
|
if info.transform.is_none() || info.physical_size.is_none() {
|
|
return false;
|
|
}
|
|
// name and logical_position can use defaults
|
|
}
|
|
(i, output_count)
|
|
}
|
|
None => return false,
|
|
}
|
|
}
|
|
_ => return false,
|
|
};
|
|
|
|
let probing = match mem::replace(&mut self.stage, EncConstructionStage::Intermediate) {
|
|
s @ EncConstructionStage::ProbingOutputs { .. } => s,
|
|
other => {
|
|
self.stage = other;
|
|
return false;
|
|
}
|
|
};
|
|
|
|
let (
|
|
outputs,
|
|
bound_outputs,
|
|
output_names,
|
|
screencopy_manager,
|
|
dmabuf,
|
|
dmabuf_feedback,
|
|
_xdg_output_manager,
|
|
_wlr_output_manager,
|
|
_wlr_manager_done,
|
|
_wlr_heads,
|
|
_wlr_head_proxy_to_name,
|
|
) = match probing {
|
|
EncConstructionStage::ProbingOutputs {
|
|
outputs,
|
|
bound_outputs,
|
|
output_names,
|
|
screencopy_manager,
|
|
dmabuf,
|
|
dmabuf_feedback,
|
|
xdg_output_manager,
|
|
wlr_output_manager,
|
|
wlr_manager_done,
|
|
wlr_heads,
|
|
wlr_head_proxy_to_name,
|
|
} => (
|
|
outputs,
|
|
bound_outputs,
|
|
output_names,
|
|
screencopy_manager,
|
|
dmabuf,
|
|
dmabuf_feedback,
|
|
xdg_output_manager,
|
|
wlr_output_manager,
|
|
wlr_manager_done,
|
|
wlr_heads,
|
|
wlr_head_proxy_to_name,
|
|
),
|
|
_ => unreachable!(),
|
|
};
|
|
// Destroy feedback object — prevents server-side resource leak
|
|
if let Some(feedback) = dmabuf_feedback {
|
|
feedback.destroy();
|
|
}
|
|
|
|
let info = &outputs[target_idx];
|
|
let output_info = OutputInfo {
|
|
name: info
|
|
.name
|
|
.clone()
|
|
.or(info.wl_name.clone())
|
|
.unwrap_or_else(|| format!("output-{}", output_names[target_idx])),
|
|
transform: info.transform.unwrap(),
|
|
physical_size: info.physical_size.unwrap(),
|
|
logical_position: info.logical_position.unwrap_or((0, 0)),
|
|
};
|
|
let output = bound_outputs[target_idx].clone();
|
|
|
|
let screencopy_manager = match screencopy_manager {
|
|
Some(m) => m,
|
|
None => {
|
|
tracing::error!("No screencopy manager bound");
|
|
self.errored = true;
|
|
return false;
|
|
}
|
|
};
|
|
let dmabuf = match dmabuf {
|
|
Some(d) => d,
|
|
None => {
|
|
tracing::error!("No dmabuf manager bound");
|
|
self.errored = true;
|
|
return false;
|
|
}
|
|
};
|
|
|
|
let drm_path = self.resolve_drm_path();
|
|
|
|
let hw_device_ctx = match AvHwDevCtx::new_vaapi(&drm_path) {
|
|
Ok(ctx) => ctx,
|
|
Err(e) => {
|
|
tracing::error!("Failed to create VAAPI device: {}", e);
|
|
self.errored = true;
|
|
return false;
|
|
}
|
|
};
|
|
|
|
let cap = match S::new(&self.gm, &output, &output_info, &self.qhandle) {
|
|
Ok(c) => c,
|
|
Err(e) => {
|
|
tracing::error!("Failed to create capture source: {}", e);
|
|
self.errored = true;
|
|
return false;
|
|
}
|
|
};
|
|
|
|
tracing::info!("Selected output: {}", output_info.name);
|
|
if self.args.output_name.is_none() && output_count > 1 {
|
|
tracing::warn!(
|
|
"Multiple outputs found, using '{}'. Use --output-name to select.",
|
|
output_info.name
|
|
);
|
|
}
|
|
self.stage = EncConstructionStage::EverythingButFmt {
|
|
output_info,
|
|
output,
|
|
hw_device_ctx,
|
|
cap,
|
|
screencopy_manager,
|
|
dmabuf,
|
|
};
|
|
|
|
true
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<WlRegistry, GlobalListContents>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<WlRegistry, GlobalListContents> for State<S> {
|
|
fn event(
|
|
state: &mut Self,
|
|
registry: &WlRegistry,
|
|
event: wayland_client::protocol::wl_registry::Event,
|
|
_data: &GlobalListContents,
|
|
_conn: &wayland_client::Connection,
|
|
qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
use wayland_client::protocol::wl_registry::Event as RegistryEvent;
|
|
|
|
match event {
|
|
RegistryEvent::Global {
|
|
name,
|
|
interface,
|
|
version,
|
|
} => match interface.as_str() {
|
|
"zwlr_screencopy_manager_v1" => {
|
|
let v = version.min(3);
|
|
tracing::debug!("Binding zwlr_screencopy_manager_v1 v{v} (name={name})");
|
|
let mgr: ZwlrScreencopyManagerV1 = registry.bind(name, v, qhandle, ());
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
screencopy_manager, ..
|
|
} = &mut state.stage
|
|
{
|
|
*screencopy_manager = Some(mgr);
|
|
}
|
|
}
|
|
"zwp_linux_dmabuf_v1" => {
|
|
let v = version.min(4);
|
|
tracing::debug!("Binding zwp_linux_dmabuf_v1 v{v} (name={name})");
|
|
let proxy: ZwpLinuxDmabufV1 = registry.bind(name, v, qhandle, ());
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
dmabuf,
|
|
dmabuf_feedback,
|
|
..
|
|
} = &mut state.stage
|
|
{
|
|
*dmabuf = Some(proxy.clone());
|
|
if v >= 4 {
|
|
let feedback = proxy.get_default_feedback(qhandle, ());
|
|
*dmabuf_feedback = Some(feedback);
|
|
}
|
|
}
|
|
}
|
|
"wl_output" => {
|
|
let v = version.min(4);
|
|
tracing::debug!("Binding wl_output v{v} (name={name})");
|
|
let output: WlOutput = registry.bind(name, v, qhandle, OutputId(name));
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
outputs,
|
|
bound_outputs,
|
|
output_names,
|
|
xdg_output_manager,
|
|
..
|
|
} = &mut state.stage
|
|
{
|
|
outputs.push(PartialOutputInfo::default());
|
|
bound_outputs.push(output.clone());
|
|
output_names.push(name);
|
|
if let Some(xdg_mgr) = xdg_output_manager {
|
|
let output_id = OutputId(name);
|
|
xdg_mgr.get_xdg_output(&output, qhandle, output_id);
|
|
}
|
|
}
|
|
}
|
|
"zxdg_output_manager_v1" => {
|
|
let v = version.min(3);
|
|
tracing::debug!("Binding zxdg_output_manager_v1 v{v} (name={name})");
|
|
let xdg_mgr: ZxdgOutputManagerV1 = registry.bind(name, v, qhandle, ());
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
bound_outputs,
|
|
xdg_output_manager,
|
|
output_names,
|
|
..
|
|
} = &mut state.stage
|
|
{
|
|
for (i, output) in bound_outputs.iter().enumerate() {
|
|
let oname = output_names.get(i).copied().unwrap_or(0);
|
|
let output_id = OutputId(oname);
|
|
xdg_mgr.get_xdg_output(output, qhandle, output_id);
|
|
}
|
|
*xdg_output_manager = Some(xdg_mgr);
|
|
}
|
|
}
|
|
"zwlr_output_manager_v1" => {
|
|
let v = version.min(4);
|
|
tracing::debug!("Binding zwlr_output_manager_v1 v{v} (name={name})");
|
|
let mgr: ZwlrOutputManagerV1 = registry.bind(name, v, qhandle, ());
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
wlr_output_manager, ..
|
|
} = &mut state.stage
|
|
{
|
|
*wlr_output_manager = Some(mgr);
|
|
}
|
|
}
|
|
_ => {}
|
|
},
|
|
RegistryEvent::GlobalRemove { name } => {
|
|
tracing::debug!("Global removed: name={name}");
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<WlOutput, ()>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<WlOutput, OutputId> for State<S> {
|
|
fn event(
|
|
state: &mut Self,
|
|
_proxy: &WlOutput,
|
|
event: wayland_client::protocol::wl_output::Event,
|
|
data: &OutputId,
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
use wayland_client::protocol::wl_output::Event as OutputEvent;
|
|
use wayland_client::protocol::wl_output::Mode as WlMode;
|
|
use wayland_client::protocol::wl_output::Transform as WlTransform;
|
|
|
|
let OutputId(target_name) = data;
|
|
let idx = match &state.stage {
|
|
EncConstructionStage::ProbingOutputs { output_names, .. } => {
|
|
output_names.iter().position(|&n| n == *target_name)
|
|
}
|
|
_ => None,
|
|
};
|
|
let idx = match idx {
|
|
Some(i) => i,
|
|
None => return,
|
|
};
|
|
|
|
match event {
|
|
OutputEvent::Geometry {
|
|
transform,
|
|
physical_width,
|
|
physical_height,
|
|
..
|
|
} => {
|
|
let t = match transform {
|
|
wayland_client::WEnum::Value(WlTransform::Normal) => Transform::Normal,
|
|
wayland_client::WEnum::Value(WlTransform::_90) => Transform::Normal90,
|
|
wayland_client::WEnum::Value(WlTransform::_180) => Transform::Normal180,
|
|
wayland_client::WEnum::Value(WlTransform::_270) => Transform::Normal270,
|
|
wayland_client::WEnum::Value(WlTransform::Flipped) => Transform::Flipped,
|
|
wayland_client::WEnum::Value(WlTransform::Flipped90) => Transform::Flipped90,
|
|
wayland_client::WEnum::Value(WlTransform::Flipped180) => Transform::Flipped180,
|
|
wayland_client::WEnum::Value(WlTransform::Flipped270) => Transform::Flipped270,
|
|
_ => Transform::Normal,
|
|
};
|
|
if let EncConstructionStage::ProbingOutputs { outputs, .. } = &mut state.stage {
|
|
if let Some(info) = outputs.get_mut(idx) {
|
|
info.transform = Some(t);
|
|
info.physical_size = Some((physical_width, physical_height));
|
|
}
|
|
}
|
|
}
|
|
OutputEvent::Mode {
|
|
width,
|
|
height,
|
|
flags,
|
|
..
|
|
} => {
|
|
let is_current = matches!(flags, wayland_client::WEnum::Value(WlMode::Current));
|
|
if is_current {
|
|
if let EncConstructionStage::ProbingOutputs { outputs, .. } = &mut state.stage {
|
|
if let Some(info) = outputs.get_mut(idx) {
|
|
info.mode_size = Some((width, height));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
OutputEvent::Done => {
|
|
if let EncConstructionStage::ProbingOutputs { outputs, .. } = &mut state.stage {
|
|
if let Some(info) = outputs.get_mut(idx) {
|
|
info.done_count += 1;
|
|
if info.done_count >= 1 {
|
|
state.try_finalize_output(idx);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
OutputEvent::Name { name } => {
|
|
if let EncConstructionStage::ProbingOutputs { outputs, .. } = &mut state.stage {
|
|
if let Some(info) = outputs.get_mut(idx) {
|
|
info.wl_name = Some(name);
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<ZxdgOutputV1, OutputId>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<ZxdgOutputV1, OutputId> for State<S> {
|
|
fn event(
|
|
state: &mut Self,
|
|
_proxy: &ZxdgOutputV1,
|
|
event: <ZxdgOutputV1 as Proxy>::Event,
|
|
data: &OutputId,
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
let target_name = data.0;
|
|
let idx = match &state.stage {
|
|
EncConstructionStage::ProbingOutputs { output_names, .. } => {
|
|
output_names.iter().position(|&n| n == target_name)
|
|
}
|
|
_ => None,
|
|
};
|
|
let idx = match idx {
|
|
Some(i) => i,
|
|
None => return,
|
|
};
|
|
|
|
match event {
|
|
XdgOutputEvent::Name { name } => {
|
|
if let EncConstructionStage::ProbingOutputs { outputs, .. } = &mut state.stage {
|
|
if let Some(info) = outputs.get_mut(idx) {
|
|
info.name = Some(name);
|
|
}
|
|
}
|
|
}
|
|
XdgOutputEvent::LogicalPosition { x, y } => {
|
|
if let EncConstructionStage::ProbingOutputs { outputs, .. } = &mut state.stage {
|
|
if let Some(info) = outputs.get_mut(idx) {
|
|
info.logical_position = Some((x, y));
|
|
}
|
|
}
|
|
}
|
|
XdgOutputEvent::LogicalSize { .. } => {}
|
|
XdgOutputEvent::Done => {
|
|
if let EncConstructionStage::ProbingOutputs { outputs, .. } = &mut state.stage {
|
|
if let Some(info) = outputs.get_mut(idx) {
|
|
info.done_count += 1;
|
|
if info.done_count >= 1 {
|
|
state.try_finalize_output(idx);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<ZwpLinuxDmabufV1, ()>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<ZwpLinuxDmabufV1, ()> for State<S> {
|
|
fn event(
|
|
_state: &mut Self,
|
|
_proxy: &ZwpLinuxDmabufV1,
|
|
event: <ZwpLinuxDmabufV1 as Proxy>::Event,
|
|
_data: &(),
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
match event {
|
|
DmabufEvent::Format { .. } => {}
|
|
DmabufEvent::Modifier { .. } => {}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<S: CaptureSource> Dispatch<ZwpLinuxDmabufFeedbackV1, ()> for State<S> {
|
|
fn event(
|
|
state: &mut Self,
|
|
_proxy: &ZwpLinuxDmabufFeedbackV1,
|
|
event: <ZwpLinuxDmabufFeedbackV1 as Proxy>::Event,
|
|
_data: &(),
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
match event {
|
|
DmabufFeedbackEvent::MainDevice { device } => {
|
|
if device.len() >= 8 {
|
|
let dev_bytes: [u8; 8] = device[..8].try_into().unwrap_or([0u8; 8]);
|
|
let dev = u64::from_ne_bytes(dev_bytes);
|
|
let minor = ((dev & 0xFF) | ((dev >> 12) & 0xFFFFFF00)) as u32;
|
|
let path = PathBuf::from(format!("/dev/dri/renderD{}", minor));
|
|
if path.exists() {
|
|
tracing::info!(
|
|
"Compositor DRM device: {} (dev_t: {})",
|
|
path.display(),
|
|
dev
|
|
);
|
|
state.drm_device_from_compositor = Some(path);
|
|
} else {
|
|
tracing::warn!(
|
|
"Compositor reported DRM device {} (dev_t: {}) but path does not exist",
|
|
path.display(),
|
|
dev
|
|
);
|
|
}
|
|
} else {
|
|
tracing::warn!(
|
|
"main_device event with unexpected data length: {}",
|
|
device.len()
|
|
);
|
|
}
|
|
}
|
|
DmabufFeedbackEvent::FormatTable { .. } => {}
|
|
DmabufFeedbackEvent::Done => {}
|
|
DmabufFeedbackEvent::TrancheDone => {}
|
|
DmabufFeedbackEvent::TrancheTargetDevice { .. } => {}
|
|
DmabufFeedbackEvent::TrancheFormats { .. } => {}
|
|
DmabufFeedbackEvent::TrancheFlags { .. } => {}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<ZwpLinuxBufferParamsV1, ()>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<ZwpLinuxBufferParamsV1, ()> for State<S> {
|
|
fn event(
|
|
state: &mut Self,
|
|
proxy: &ZwpLinuxBufferParamsV1,
|
|
event: <ZwpLinuxBufferParamsV1 as Proxy>::Event,
|
|
_data: &(),
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
match event {
|
|
BufferParamsEvent::Created { .. } => {
|
|
tracing::debug!("DMA-BUF buffer created");
|
|
}
|
|
BufferParamsEvent::Failed => {
|
|
tracing::error!("DMA-BUF buffer creation failed");
|
|
let taken = mem::replace(&mut state.in_flight_surface, InFlightSurface::None);
|
|
match taken {
|
|
InFlightSurface::CopyQueued { buffer, frame, .. } => {
|
|
drop(buffer);
|
|
if let EncConstructionStage::Streaming { cap, .. } = &mut state.stage {
|
|
cap.on_done_with_frame(frame);
|
|
}
|
|
}
|
|
other => {
|
|
state.in_flight_surface = other;
|
|
}
|
|
}
|
|
proxy.destroy();
|
|
state.errored = true;
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<ZwlrScreencopyFrameV1, ()> for CapWlrScreencopy
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl Dispatch<ZwlrScreencopyFrameV1, ()> for State<CapWlrScreencopy> {
|
|
fn event(
|
|
state: &mut Self,
|
|
proxy: &ZwlrScreencopyFrameV1,
|
|
event: <ZwlrScreencopyFrameV1 as Proxy>::Event,
|
|
_data: &(),
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<CapWlrScreencopy>>,
|
|
) {
|
|
match event {
|
|
// SHM buffer offer — in v3 the compositor enumerates supported buffer
|
|
// types (buffer and/or linux_dmabuf) before buffer_done. We only
|
|
// support DMA-BUF, so just log and wait for linux_dmabuf / buffer_done.
|
|
ScreencopyFrameEvent::Buffer { .. } => {
|
|
tracing::debug!("Received SHM Buffer offer — only DMA-BUF capture is supported");
|
|
}
|
|
ScreencopyFrameEvent::LinuxDmabuf {
|
|
format,
|
|
width,
|
|
height,
|
|
} => {
|
|
tracing::debug!("Screencopy LinuxDmabuf: format={format}, {width}x{height}");
|
|
|
|
if !matches!(state.in_flight_surface, InFlightSurface::AllocQueued) {
|
|
tracing::warn!("Received LinuxDmabuf while no frame allocation was queued");
|
|
return;
|
|
}
|
|
|
|
if matches!(state.stage, EncConstructionStage::EverythingButFmt { .. }) {
|
|
state.negotiate_format(format, width, height);
|
|
if state.errored {
|
|
return;
|
|
}
|
|
}
|
|
if let EncConstructionStage::Streaming { cap, .. } = &mut state.stage {
|
|
cap.current_frame = Some(proxy.clone());
|
|
}
|
|
state.on_frame_allocd((), format, width, height);
|
|
}
|
|
// v3 terminal event: all buffer offers have been enumerated.
|
|
// If still AllocQueued, the compositor never sent linux_dmabuf —
|
|
// DMA-BUF screencopy is unsupported, so we must error out.
|
|
ScreencopyFrameEvent::BufferDone => {
|
|
if matches!(state.in_flight_surface, InFlightSurface::AllocQueued) {
|
|
tracing::error!(
|
|
"Compositor did not offer DMA-BUF screencopy (only SHM); \
|
|
DMA-BUF capture is required"
|
|
);
|
|
state.in_flight_surface = InFlightSurface::None;
|
|
proxy.destroy();
|
|
state.errored = true;
|
|
}
|
|
}
|
|
ScreencopyFrameEvent::Ready {
|
|
tv_sec_hi,
|
|
tv_sec_lo,
|
|
tv_nsec,
|
|
} => {
|
|
let tv_sec = (tv_sec_hi as u64) << 32 | tv_sec_lo as u64;
|
|
let tv_usec = tv_nsec / 1000;
|
|
tracing::trace!("Screencopy ready: tv_sec={tv_sec}, tv_usec={tv_usec}");
|
|
state.on_copy_complete(tv_sec, tv_usec);
|
|
}
|
|
ScreencopyFrameEvent::Failed => {
|
|
tracing::error!("Screencopy frame failed");
|
|
state.on_copy_fail();
|
|
}
|
|
ScreencopyFrameEvent::Damage { .. } => {}
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<ZxdgOutputManagerV1, ()>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<ZxdgOutputManagerV1, ()> for State<S> {
|
|
fn event(
|
|
_state: &mut Self,
|
|
_proxy: &ZxdgOutputManagerV1,
|
|
_event: <ZxdgOutputManagerV1 as Proxy>::Event,
|
|
_data: &(),
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<ZwlrOutputManagerV1, ()>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<ZwlrOutputManagerV1, ()> for State<S> {
|
|
fn event(
|
|
state: &mut Self,
|
|
_proxy: &ZwlrOutputManagerV1,
|
|
event: <ZwlrOutputManagerV1 as Proxy>::Event,
|
|
_data: &(),
|
|
_conn: &wayland_client::Connection,
|
|
qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
match event {
|
|
WlrOutputManagerEvent::Head { head } => {
|
|
let _head: ZwlrOutputHeadV1 = head;
|
|
tracing::debug!("wlr output head advertised");
|
|
}
|
|
WlrOutputManagerEvent::Done { .. } => {
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
wlr_manager_done,
|
|
outputs,
|
|
..
|
|
} = &mut state.stage
|
|
{
|
|
*wlr_manager_done = true;
|
|
let count = outputs.len();
|
|
for idx in 0..count {
|
|
state.try_finalize_output(idx);
|
|
}
|
|
}
|
|
}
|
|
WlrOutputManagerEvent::Finished { .. } => {
|
|
tracing::warn!("zwlr_output_manager_v1::Finished received during probing");
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
event_created_child!(State<S>, ZwlrOutputManagerV1, [
|
|
zwlr_output_manager_v1::EVT_HEAD_OPCODE => (ZwlrOutputHeadV1, ()),
|
|
]);
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<ZwlrOutputHeadV1, ()>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<ZwlrOutputHeadV1, ()> for State<S> {
|
|
fn event(
|
|
state: &mut Self,
|
|
proxy: &ZwlrOutputHeadV1,
|
|
event: <ZwlrOutputHeadV1 as Proxy>::Event,
|
|
_data: &(),
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
match event {
|
|
WlrHeadEvent::Name { name } => {
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
wlr_heads,
|
|
wlr_head_proxy_to_name,
|
|
..
|
|
} = &mut state.stage
|
|
{
|
|
wlr_heads
|
|
.entry(name.clone())
|
|
.or_insert(WlrHeadInfo { position: None });
|
|
wlr_head_proxy_to_name.insert(proxy.id(), name);
|
|
}
|
|
}
|
|
WlrHeadEvent::Position { x, y } => {
|
|
if let EncConstructionStage::ProbingOutputs {
|
|
wlr_heads,
|
|
wlr_head_proxy_to_name,
|
|
..
|
|
} = &mut state.stage
|
|
{
|
|
if let Some(name) = wlr_head_proxy_to_name.get(&proxy.id()) {
|
|
if let Some(head) = wlr_heads.get_mut(name) {
|
|
head.position = Some((x, y));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
WlrHeadEvent::Finished { .. } => {
|
|
tracing::debug!("zwlr_output_head_v1::Finished received");
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
event_created_child!(State<S>, ZwlrOutputHeadV1, [
|
|
zwlr_output_head_v1::EVT_MODE_OPCODE => (ZwlrOutputModeV1, ()),
|
|
]);
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<ZwlrOutputModeV1, ()>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<ZwlrOutputModeV1, ()> for State<S> {
|
|
fn event(
|
|
_state: &mut Self,
|
|
_proxy: &ZwlrOutputModeV1,
|
|
_event: <ZwlrOutputModeV1 as Proxy>::Event,
|
|
_data: &(),
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<ZwlrScreencopyManagerV1, ()>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<ZwlrScreencopyManagerV1, ()> for State<S> {
|
|
fn event(
|
|
_state: &mut Self,
|
|
_proxy: &ZwlrScreencopyManagerV1,
|
|
_event: <ZwlrScreencopyManagerV1 as Proxy>::Event,
|
|
_data: &(),
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
// Dispatch<WlBuffer, ()>
|
|
// ---------------------------------------------------------------------------
|
|
|
|
impl<S: CaptureSource> Dispatch<WlBuffer, ()> for State<S> {
|
|
fn event(
|
|
_state: &mut Self,
|
|
_proxy: &WlBuffer,
|
|
event: <WlBuffer as Proxy>::Event,
|
|
_data: &(),
|
|
_conn: &wayland_client::Connection,
|
|
_qhandle: &QueueHandle<State<S>>,
|
|
) {
|
|
if let wayland_client::protocol::wl_buffer::Event::Release = event {
|
|
tracing::trace!("WlBuffer released");
|
|
}
|
|
}
|
|
}
|