mirror of
https://github.com/rustdesk/rustdesk.git
synced 2026-05-07 22:58:10 +03:00
1529 lines
52 KiB
Rust
1529 lines
52 KiB
Rust
use std::collections::HashMap;
|
|
use std::error::Error;
|
|
use std::os::unix::io::AsRawFd;
|
|
use std::process::Command;
|
|
use std::sync::{
|
|
atomic::{AtomicBool, AtomicU8, Ordering},
|
|
Arc, Mutex,
|
|
};
|
|
use std::time::Duration;
|
|
use tracing::{debug, error, trace, warn};
|
|
|
|
use dbus::{
|
|
arg::{OwnedFd, PropMap, RefArg, Variant},
|
|
blocking::{Proxy, SyncConnection},
|
|
message::{MatchRule, MessageType},
|
|
Message,
|
|
};
|
|
|
|
use gstreamer as gst;
|
|
use gstreamer::prelude::*;
|
|
use gstreamer_app::AppSink;
|
|
|
|
use lazy_static::lazy_static;
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use hbb_common::{bail, config, platform::linux::CMD_SH, serde_json, tokio, ResultType};
|
|
|
|
use super::capturable::PixelProvider;
|
|
use super::capturable::{Capturable, Recorder};
|
|
use super::display::{clear_wayland_displays_cache, get_displays, Displays};
|
|
use super::remote_desktop_portal::OrgFreedesktopPortalRemoteDesktop as remote_desktop_portal;
|
|
use super::request_portal::OrgFreedesktopPortalRequestResponse;
|
|
use super::screencast_portal::OrgFreedesktopPortalScreenCast as screencast_portal;
|
|
|
|
lazy_static! {
|
|
pub static ref RDP_SESSION_INFO: Mutex<Option<RdpSessionInfo>> = Mutex::new(None);
|
|
}
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
// For KDE Plasma only, because GNOME provides position info.
|
|
struct PipewireDisplayOffsetCache {
|
|
// We need to compare the displays, because:
|
|
// 1. On Archlinux KDE Plasma
|
|
// 2. One display, and connect, remember share choice.
|
|
// 3. Plug in another monitor.
|
|
// 4. The portal will reuse the restore token, no new share choice dialog, but the share screen is different.
|
|
// The controlling side will see the new monitor.
|
|
// All displays as one string for easy comparison
|
|
// name1-x1-y1-width1-height1;name2-x2-y2-width2-height2;...
|
|
display_key: String,
|
|
restore_token: String,
|
|
offsets: Vec<(i32, i32)>,
|
|
}
|
|
|
|
// KDE Plasma may not provide position info
|
|
static HAS_POSITION_ATTR: AtomicBool = AtomicBool::new(false);
|
|
static IS_SERVER_RUNNING: AtomicU8 = AtomicU8::new(0); // 0: uninitialized, 1:true, 2: false
|
|
|
|
impl PipewireDisplayOffsetCache {
|
|
fn displays_to_key(displays: &Arc<Displays>) -> String {
|
|
displays
|
|
.displays
|
|
.iter()
|
|
.map(|d| format!("{}-{}-{}-{}-{}", d.name, d.x, d.y, d.width, d.height))
|
|
.collect::<Vec<String>>()
|
|
.join(";")
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
pub fn close_session() {
|
|
let _ = RDP_SESSION_INFO.lock().unwrap().take();
|
|
clear_wayland_displays_cache();
|
|
HAS_POSITION_ATTR.store(false, Ordering::SeqCst);
|
|
}
|
|
|
|
#[inline]
|
|
pub fn is_rdp_session_hold() -> bool {
|
|
RDP_SESSION_INFO.lock().unwrap().is_some()
|
|
}
|
|
|
|
pub fn try_close_session() {
|
|
let mut rdp_info = RDP_SESSION_INFO.lock().unwrap();
|
|
let mut close = false;
|
|
if let Some(rdp_info) = &*rdp_info {
|
|
// If is server running and restore token is supported, there's no need to keep the session.
|
|
if is_server_running() && rdp_info.is_support_restore_token {
|
|
close = true;
|
|
}
|
|
}
|
|
if close {
|
|
*rdp_info = None;
|
|
clear_wayland_displays_cache();
|
|
HAS_POSITION_ATTR.store(false, Ordering::SeqCst);
|
|
}
|
|
}
|
|
|
|
pub struct RdpSessionInfo {
|
|
pub conn: Arc<SyncConnection>,
|
|
pub streams: Vec<PwStreamInfo>,
|
|
pub fd: OwnedFd,
|
|
pub session: dbus::Path<'static>,
|
|
pub is_support_restore_token: bool,
|
|
pub resolution: Arc<Mutex<Option<(usize, usize)>>>,
|
|
}
|
|
#[derive(Debug, Clone, Copy)]
|
|
pub struct PwStreamInfo {
|
|
pub path: u64,
|
|
source_type: u64,
|
|
position: (i32, i32),
|
|
size: (usize, usize),
|
|
}
|
|
|
|
impl PwStreamInfo {
|
|
pub fn get_size(&self) -> (usize, usize) {
|
|
self.size
|
|
}
|
|
|
|
pub fn get_position(&self) -> (i32, i32) {
|
|
self.position
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
pub struct DBusError(String);
|
|
|
|
impl std::fmt::Display for DBusError {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
let Self(s) = self;
|
|
write!(f, "{}", s)
|
|
}
|
|
}
|
|
|
|
impl Error for DBusError {}
|
|
|
|
#[derive(Debug)]
|
|
pub struct GStreamerError(String);
|
|
|
|
impl std::fmt::Display for GStreamerError {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
let Self(s) = self;
|
|
write!(f, "{}", s)
|
|
}
|
|
}
|
|
|
|
impl Error for GStreamerError {}
|
|
|
|
#[derive(Clone)]
|
|
pub struct PipeWireCapturable {
|
|
// connection needs to be kept alive for recording
|
|
dbus_conn: Arc<SyncConnection>,
|
|
fd: OwnedFd,
|
|
path: u64,
|
|
source_type: u64,
|
|
pub primary: bool,
|
|
pub position: (i32, i32),
|
|
pub logical_size: (usize, usize),
|
|
pub physical_size: (usize, usize),
|
|
}
|
|
|
|
impl PipeWireCapturable {
|
|
fn new(
|
|
conn: Arc<SyncConnection>,
|
|
fd: OwnedFd,
|
|
resolution: Arc<Mutex<Option<(usize, usize)>>>,
|
|
stream: &PwStreamInfo,
|
|
) -> Self {
|
|
// alternative to get screen resolution as stream.size is not always correct ex: on fractional scaling
|
|
// https://github.com/rustdesk/rustdesk/issues/6116#issuecomment-1817724244
|
|
let physical_size = get_res(Self {
|
|
dbus_conn: conn.clone(),
|
|
fd: fd.clone(),
|
|
path: stream.path,
|
|
source_type: stream.source_type,
|
|
primary: false,
|
|
position: stream.position,
|
|
logical_size: stream.size,
|
|
physical_size: (0, 0),
|
|
})
|
|
.unwrap_or(stream.size);
|
|
*resolution.lock().unwrap() = Some(physical_size);
|
|
Self {
|
|
dbus_conn: conn,
|
|
fd,
|
|
path: stream.path,
|
|
source_type: stream.source_type,
|
|
primary: false,
|
|
position: stream.position,
|
|
logical_size: stream.size,
|
|
physical_size,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl std::fmt::Debug for PipeWireCapturable {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
write!(
|
|
f,
|
|
"PipeWireCapturable {{dbus: {}, fd: {}, path: {}, source_type: {}}}",
|
|
self.dbus_conn.unique_name(),
|
|
self.fd.as_raw_fd(),
|
|
self.path,
|
|
self.source_type
|
|
)
|
|
}
|
|
}
|
|
|
|
impl Capturable for PipeWireCapturable {
|
|
fn name(&self) -> String {
|
|
let type_str = match self.source_type {
|
|
1 => "Desktop",
|
|
2 => "Window",
|
|
_ => "Unknow",
|
|
};
|
|
format!("Pipewire {}, path: {}", type_str, self.path)
|
|
}
|
|
|
|
fn geometry_relative(&self) -> Result<(f64, f64, f64, f64), Box<dyn Error>> {
|
|
Ok((0.0, 0.0, 1.0, 1.0))
|
|
}
|
|
|
|
fn before_input(&mut self) -> Result<(), Box<dyn Error>> {
|
|
Ok(())
|
|
}
|
|
|
|
fn recorder(&self, _capture_cursor: bool) -> Result<Box<dyn Recorder>, Box<dyn Error>> {
|
|
Ok(Box::new(PipeWireRecorder::new(self.clone())?))
|
|
}
|
|
}
|
|
|
|
fn get_res(capturable: PipeWireCapturable) -> Result<(usize, usize), Box<dyn Error>> {
|
|
let rec = PipeWireRecorder::new(capturable)?;
|
|
if let Some(sample) = rec
|
|
.appsink
|
|
.try_pull_sample(gst::ClockTime::from_mseconds(300))
|
|
{
|
|
let cap = sample
|
|
.get_caps()
|
|
.ok_or("Failed get caps")?
|
|
.get_structure(0)
|
|
.ok_or("Failed to get structure")?;
|
|
let w: i32 = cap.get_value("width")?.get_some()?;
|
|
let h: i32 = cap.get_value("height")?.get_some()?;
|
|
let w = w as usize;
|
|
let h = h as usize;
|
|
Ok((w, h))
|
|
} else {
|
|
Err(Box::new(GStreamerError(
|
|
"Error getting screen resolution".into(),
|
|
)))
|
|
}
|
|
}
|
|
|
|
pub struct PipeWireRecorder {
|
|
buffer: Option<gst::MappedBuffer<gst::buffer::Readable>>,
|
|
buffer_cropped: Vec<u8>,
|
|
pix_fmt: String,
|
|
is_cropped: bool,
|
|
pipeline: gst::Pipeline,
|
|
appsink: AppSink,
|
|
width: usize,
|
|
height: usize,
|
|
saved_raw_data: Vec<u8>, // for faster compare and copy
|
|
}
|
|
|
|
impl PipeWireRecorder {
|
|
pub fn new(capturable: PipeWireCapturable) -> ResultType<Self> {
|
|
let pipeline = gst::Pipeline::new(None);
|
|
|
|
let src = gst::ElementFactory::make("pipewiresrc", None)?;
|
|
src.set_property("fd", &capturable.fd.as_raw_fd())?;
|
|
src.set_property("path", &format!("{}", capturable.path))?;
|
|
src.set_property("keepalive_time", &1_000.as_raw_fd())?;
|
|
|
|
// For some reason pipewire blocks on destruction of AppSink if this is not set to true,
|
|
// see: https://gitlab.freedesktop.org/pipewire/pipewire/-/issues/982
|
|
src.set_property("always-copy", &true)?;
|
|
|
|
let sink = gst::ElementFactory::make("appsink", None)?;
|
|
sink.set_property("drop", &true)?;
|
|
sink.set_property("max-buffers", &1u32)?;
|
|
|
|
pipeline.add_many(&[&src, &sink])?;
|
|
src.link(&sink)?;
|
|
|
|
let appsink = sink
|
|
.dynamic_cast::<AppSink>()
|
|
.map_err(|_| GStreamerError("Sink element is expected to be an appsink!".into()))?;
|
|
let mut caps = gst::Caps::new_empty();
|
|
caps.merge_structure(gst::structure::Structure::new(
|
|
"video/x-raw",
|
|
&[("format", &"BGRx")],
|
|
));
|
|
caps.merge_structure(gst::structure::Structure::new(
|
|
"video/x-raw",
|
|
&[("format", &"RGBx")],
|
|
));
|
|
appsink.set_caps(Some(&caps));
|
|
|
|
// [Workaround]
|
|
// Crash may occur if there are multiple pipelines started at the same time.
|
|
// `pipeline.get_state()` can significantly reduce the probability of crashes,
|
|
// but cannot completely resolve this issue.
|
|
// Adding a short sleep period can also reduce the probability of crashes.
|
|
debug!(
|
|
"[gstreamer] Setting pipeline {} to PLAYING state...",
|
|
capturable.fd.as_raw_fd()
|
|
);
|
|
pipeline.set_state(gst::State::Playing)?;
|
|
|
|
// If `is_server_running()` is false, it means using remote_desktop_portal,
|
|
// which does not use multiple streams, so no need to wait for state change.
|
|
if is_server_running() {
|
|
// Wait for the state change to actually complete before proceeding.
|
|
// The 2000ms timeout for pipeline state change was chosen based on empirical testing.
|
|
let state_change = pipeline.get_state(gst::ClockTime::from_mseconds(2000));
|
|
match state_change {
|
|
(Ok(_), gst::State::Playing, _) => {
|
|
debug!(
|
|
"[gstreamer] Pipeline {} state confirmed as PLAYING.",
|
|
capturable.fd.as_raw_fd()
|
|
);
|
|
}
|
|
(result, state, pending) => {
|
|
warn!(
|
|
"[gstreamer] Pipeline {} state change incomplete: result={:?}, state={:?}, pending={:?}",
|
|
capturable.fd.as_raw_fd(), result, state, pending
|
|
);
|
|
}
|
|
}
|
|
std::thread::sleep(std::time::Duration::from_millis(150));
|
|
}
|
|
|
|
Ok(Self {
|
|
pipeline,
|
|
appsink,
|
|
buffer: None,
|
|
pix_fmt: "".into(),
|
|
width: 0,
|
|
height: 0,
|
|
buffer_cropped: vec![],
|
|
is_cropped: false,
|
|
saved_raw_data: Vec::new(),
|
|
})
|
|
}
|
|
}
|
|
|
|
impl Recorder for PipeWireRecorder {
|
|
fn capture(&mut self, timeout_ms: u64) -> Result<PixelProvider<'_>, Box<dyn Error>> {
|
|
if let Some(sample) = self
|
|
.appsink
|
|
.try_pull_sample(gst::ClockTime::from_mseconds(timeout_ms))
|
|
{
|
|
let cap = sample
|
|
.get_caps()
|
|
.ok_or("Failed get caps")?
|
|
.get_structure(0)
|
|
.ok_or("Failed to get structure")?;
|
|
let w: i32 = cap.get_value("width")?.get_some()?;
|
|
let h: i32 = cap.get_value("height")?.get_some()?;
|
|
let w = w as usize;
|
|
let h = h as usize;
|
|
self.pix_fmt = cap
|
|
.get::<&str>("format")?
|
|
.ok_or("Failed to get pixel format")?
|
|
.to_string();
|
|
|
|
let buf = sample
|
|
.get_buffer_owned()
|
|
.ok_or_else(|| GStreamerError("Failed to get owned buffer.".into()))?;
|
|
let mut crop = buf
|
|
.get_meta::<gstreamer_video::VideoCropMeta>()
|
|
.map(|m| m.get_rect());
|
|
// only crop if necessary
|
|
if Some((0, 0, w as u32, h as u32)) == crop {
|
|
crop = None;
|
|
}
|
|
let buf = buf
|
|
.into_mapped_buffer_readable()
|
|
.map_err(|_| GStreamerError("Failed to map buffer.".into()))?;
|
|
if let Err(..) = crate::would_block_if_equal(&mut self.saved_raw_data, buf.as_slice()) {
|
|
return Ok(PixelProvider::NONE);
|
|
}
|
|
let buf_size = buf.get_size();
|
|
// BGRx is 4 bytes per pixel
|
|
if buf_size != (w * h * 4) {
|
|
// for some reason the width and height of the caps do not guarantee correct buffer
|
|
// size, so ignore those buffers, see:
|
|
// https://gitlab.freedesktop.org/pipewire/pipewire/-/issues/985
|
|
trace!(
|
|
"Size of mapped buffer: {} does NOT match size of capturable {}x{}@BGRx, \
|
|
dropping it!",
|
|
buf_size,
|
|
w,
|
|
h
|
|
);
|
|
} else {
|
|
// Copy region specified by crop into self.buffer_cropped
|
|
// TODO: Figure out if ffmpeg provides a zero copy alternative
|
|
if let Some((x_off, y_off, w_crop, h_crop)) = crop {
|
|
let x_off = x_off as usize;
|
|
let y_off = y_off as usize;
|
|
let w_crop = w_crop as usize;
|
|
let h_crop = h_crop as usize;
|
|
self.buffer_cropped.clear();
|
|
let data = buf.as_slice();
|
|
// BGRx is 4 bytes per pixel
|
|
self.buffer_cropped.reserve(w_crop * h_crop * 4);
|
|
for y in y_off..(y_off + h_crop) {
|
|
let i = 4 * (w * y + x_off);
|
|
self.buffer_cropped.extend(&data[i..i + 4 * w_crop]);
|
|
}
|
|
self.width = w_crop;
|
|
self.height = h_crop;
|
|
} else {
|
|
self.width = w;
|
|
self.height = h;
|
|
}
|
|
self.is_cropped = crop.is_some();
|
|
self.buffer = Some(buf);
|
|
}
|
|
} else {
|
|
return Ok(PixelProvider::NONE);
|
|
}
|
|
if self.buffer.is_none() {
|
|
return Err(Box::new(GStreamerError("No buffer available!".into())));
|
|
}
|
|
let buf = if self.is_cropped {
|
|
self.buffer_cropped.as_slice()
|
|
} else {
|
|
self.buffer
|
|
.as_ref()
|
|
.ok_or("Failed to get buffer as ref")?
|
|
.as_slice()
|
|
};
|
|
match self.pix_fmt.as_str() {
|
|
"BGRx" => Ok(PixelProvider::BGR0(self.width, self.height, buf)),
|
|
"RGBx" => Ok(PixelProvider::RGB0(self.width, self.height, buf)),
|
|
_ => Err(Box::new(GStreamerError(format!(
|
|
"Unreachable! Unknown pix_fmt, {}",
|
|
&self.pix_fmt
|
|
)))),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Drop for PipeWireRecorder {
|
|
fn drop(&mut self) {
|
|
if let Err(err) = self.pipeline.set_state(gst::State::Null) {
|
|
warn!("Failed to stop GStreamer pipeline: {}.", err);
|
|
}
|
|
// Wait for state change to complete to avoid races during PipeWire teardown.
|
|
let _ = self.pipeline.get_state(gst::ClockTime::from_mseconds(2000));
|
|
}
|
|
}
|
|
|
|
fn handle_response<F>(
|
|
conn: &SyncConnection,
|
|
path: dbus::Path<'static>,
|
|
mut f: F,
|
|
failure_out: Arc<AtomicBool>,
|
|
) -> Result<dbus::channel::Token, dbus::Error>
|
|
where
|
|
F: FnMut(
|
|
OrgFreedesktopPortalRequestResponse,
|
|
&SyncConnection,
|
|
&Message,
|
|
) -> Result<(), Box<dyn Error>>
|
|
+ Send
|
|
+ Sync
|
|
+ 'static,
|
|
{
|
|
let mut m = MatchRule::new();
|
|
m.path = Some(path);
|
|
m.msg_type = Some(MessageType::Signal);
|
|
m.sender = Some("org.freedesktop.portal.Desktop".into());
|
|
m.interface = Some("org.freedesktop.portal.Request".into());
|
|
conn.add_match(m, move |r: OrgFreedesktopPortalRequestResponse, c, m| {
|
|
debug!("Response from DBus: response: {:?}, message: {:?}", r, m);
|
|
match r.response {
|
|
0 => {}
|
|
1 => {
|
|
warn!("DBus response: User cancelled interaction.");
|
|
failure_out.store(true, Ordering::SeqCst);
|
|
return true;
|
|
}
|
|
c => {
|
|
warn!("DBus response: Unknown error, code: {}.", c);
|
|
failure_out.store(true, Ordering::SeqCst);
|
|
return true;
|
|
}
|
|
}
|
|
if let Err(err) = f(r, c, m) {
|
|
warn!("Error requesting screen capture via dbus: {}", err);
|
|
failure_out.store(true, Ordering::SeqCst);
|
|
}
|
|
true
|
|
})
|
|
}
|
|
|
|
pub fn get_portal(conn: &SyncConnection) -> Proxy<&SyncConnection> {
|
|
conn.with_proxy(
|
|
"org.freedesktop.portal.Desktop",
|
|
"/org/freedesktop/portal/desktop",
|
|
Duration::from_millis(1000),
|
|
)
|
|
}
|
|
|
|
fn streams_from_response(response: OrgFreedesktopPortalRequestResponse) -> Vec<PwStreamInfo> {
|
|
(move || {
|
|
Some(
|
|
response
|
|
.results
|
|
.get("streams")?
|
|
.as_iter()?
|
|
.next()?
|
|
.as_iter()?
|
|
.filter_map(|stream| {
|
|
let mut itr = stream.as_iter()?;
|
|
let path = itr.next()?.as_u64()?;
|
|
let (keys, values): (Vec<(usize, &dyn RefArg)>, Vec<(usize, &dyn RefArg)>) =
|
|
itr.next()?
|
|
.as_iter()?
|
|
.enumerate()
|
|
.partition(|(i, _)| i % 2 == 0);
|
|
let attributes = keys
|
|
.iter()
|
|
.filter_map(|(_, key)| Some(key.as_str()?.to_owned()))
|
|
.zip(
|
|
values
|
|
.iter()
|
|
.map(|(_, arg)| *arg)
|
|
.collect::<Vec<&dyn RefArg>>(),
|
|
)
|
|
.collect::<HashMap<String, &dyn RefArg>>();
|
|
let mut info = PwStreamInfo {
|
|
path,
|
|
source_type: attributes
|
|
.get("source_type")
|
|
.map_or(Some(0), |v| v.as_u64())?,
|
|
position: (0, 0),
|
|
size: (0, 0),
|
|
};
|
|
let v = attributes
|
|
.get("size")?
|
|
.as_iter()?
|
|
.filter_map(|v| {
|
|
Some(
|
|
v.as_iter()?
|
|
.map(|x| x.as_i64().unwrap_or(0))
|
|
.collect::<Vec<i64>>(),
|
|
)
|
|
})
|
|
.next();
|
|
if let Some(v) = v {
|
|
if v.len() == 2 {
|
|
info.size.0 = v[0] as _;
|
|
info.size.1 = v[1] as _;
|
|
}
|
|
}
|
|
if let Some(pos) = attributes.get("position") {
|
|
let v = pos
|
|
.as_iter()?
|
|
.filter_map(|v| {
|
|
Some(
|
|
v.as_iter()?
|
|
.map(|x| x.as_i64().unwrap_or(0))
|
|
.collect::<Vec<i64>>(),
|
|
)
|
|
})
|
|
.next();
|
|
if let Some(v) = v {
|
|
if v.len() == 2 {
|
|
info.position.0 = v[0] as _;
|
|
info.position.1 = v[1] as _;
|
|
HAS_POSITION_ATTR.store(true, Ordering::SeqCst);
|
|
}
|
|
}
|
|
}
|
|
Some(info)
|
|
})
|
|
.collect::<Vec<PwStreamInfo>>(),
|
|
)
|
|
})()
|
|
.unwrap_or_default()
|
|
}
|
|
|
|
static mut INIT: bool = false;
|
|
const RESTORE_TOKEN: &str = "restore_token";
|
|
const RESTORE_TOKEN_CONF_KEY: &str = "wayland-restore-token";
|
|
const PIPEWIRE_DISPLAY_OFFSET_CONF_KEY: &str = "wayland-pipewire-display-offset";
|
|
|
|
pub fn get_available_cursor_modes() -> Result<u32, dbus::Error> {
|
|
let conn = SyncConnection::new_session()?;
|
|
let portal = get_portal(&conn);
|
|
portal.available_cursor_modes()
|
|
}
|
|
|
|
// mostly inspired by https://gitlab.gnome.org/-/snippets/39
|
|
pub fn request_remote_desktop(
|
|
capture_cursor: bool,
|
|
) -> ResultType<(
|
|
SyncConnection,
|
|
OwnedFd,
|
|
Vec<PwStreamInfo>,
|
|
dbus::Path<'static>,
|
|
bool,
|
|
)> {
|
|
unsafe {
|
|
if !INIT {
|
|
gstreamer::init()?;
|
|
INIT = true;
|
|
}
|
|
}
|
|
let conn = SyncConnection::new_session()?;
|
|
let portal = get_portal(&conn);
|
|
let mut args: PropMap = HashMap::new();
|
|
let fd: Arc<Mutex<Option<OwnedFd>>> = Arc::new(Mutex::new(None));
|
|
let fd_res = fd.clone();
|
|
let streams: Arc<Mutex<Vec<PwStreamInfo>>> = Arc::new(Mutex::new(Vec::new()));
|
|
let streams_res = streams.clone();
|
|
let failure = Arc::new(AtomicBool::new(false));
|
|
let failure_res = failure.clone();
|
|
let session: Arc<Mutex<Option<dbus::Path>>> = Arc::new(Mutex::new(None));
|
|
let session_res = session.clone();
|
|
args.insert(
|
|
"session_handle_token".to_string(),
|
|
Variant(Box::new("u1".to_string())),
|
|
);
|
|
args.insert(
|
|
"handle_token".to_string(),
|
|
Variant(Box::new("u1".to_string())),
|
|
);
|
|
|
|
let mut is_support_restore_token = false;
|
|
if let Ok(version) = screencast_portal::version(&portal) {
|
|
if version >= 4 {
|
|
is_support_restore_token = true;
|
|
}
|
|
}
|
|
|
|
// The following code may be improved.
|
|
// https://flatpak.github.io/xdg-desktop-portal/#:~:text=To%20avoid%20a%20race%20condition
|
|
// To avoid a race condition
|
|
// between the caller subscribing to the signal after receiving the reply for the method call and the signal getting emitted,
|
|
// a convention for Request object paths has been established that allows
|
|
// the caller to subscribe to the signal before making the method call.
|
|
let path;
|
|
if is_server_running() {
|
|
path = screencast_portal::create_session(&portal, args)?;
|
|
} else {
|
|
path = remote_desktop_portal::create_session(&portal, args)?;
|
|
}
|
|
handle_response(
|
|
&conn,
|
|
path,
|
|
on_create_session_response(
|
|
fd.clone(),
|
|
streams.clone(),
|
|
session.clone(),
|
|
failure.clone(),
|
|
is_support_restore_token,
|
|
capture_cursor,
|
|
),
|
|
failure_res.clone(),
|
|
)?;
|
|
|
|
// wait 3 minutes for user interaction
|
|
for _ in 0..1800 {
|
|
conn.process(Duration::from_millis(100))?;
|
|
// Once we got a file descriptor we are done!
|
|
if fd_res.lock().unwrap().is_some() {
|
|
break;
|
|
}
|
|
|
|
if failure_res.load(Ordering::SeqCst) {
|
|
break;
|
|
}
|
|
}
|
|
let fd_res = fd_res.lock().unwrap();
|
|
let streams_res = streams_res.lock().unwrap();
|
|
let session_res = session_res.lock().unwrap();
|
|
|
|
if let Some(fd_res) = fd_res.clone() {
|
|
if let Some(session) = session_res.clone() {
|
|
if !streams_res.is_empty() {
|
|
return Ok((
|
|
conn,
|
|
fd_res,
|
|
streams_res.clone(),
|
|
session,
|
|
is_support_restore_token,
|
|
));
|
|
}
|
|
}
|
|
}
|
|
bail!("Failed to obtain screen capture. You may need to upgrade the PipeWire library for better compatibility. Please check https://github.com/rustdesk/rustdesk/issues/8600#issuecomment-2254720954 for more details.")
|
|
}
|
|
|
|
fn on_create_session_response(
|
|
fd: Arc<Mutex<Option<OwnedFd>>>,
|
|
streams: Arc<Mutex<Vec<PwStreamInfo>>>,
|
|
session: Arc<Mutex<Option<dbus::Path<'static>>>>,
|
|
failure: Arc<AtomicBool>,
|
|
is_support_restore_token: bool,
|
|
capture_cursor: bool,
|
|
) -> impl Fn(
|
|
OrgFreedesktopPortalRequestResponse,
|
|
&SyncConnection,
|
|
&dbus::Message,
|
|
) -> Result<(), Box<dyn Error>> {
|
|
move |r: OrgFreedesktopPortalRequestResponse, c, _| {
|
|
let ses: dbus::Path = r
|
|
.results
|
|
.get("session_handle")
|
|
.ok_or_else(|| {
|
|
DBusError(format!(
|
|
"Failed to obtain session_handle from response: {:?}",
|
|
r
|
|
))
|
|
})?
|
|
.as_str()
|
|
.ok_or_else(|| DBusError("Failed to convert session_handle to string.".into()))?
|
|
.to_string()
|
|
.into();
|
|
|
|
let mut session = match session.lock() {
|
|
Ok(session) => session,
|
|
Err(_) => return Err(Box::new(DBusError("Failed to lock session.".into()))),
|
|
};
|
|
session.replace(ses.clone());
|
|
|
|
let portal = get_portal(c);
|
|
let mut args: PropMap = HashMap::new();
|
|
// See `is_server_running()` to understand the following code.
|
|
if is_server_running() {
|
|
if is_support_restore_token {
|
|
let restore_token = config::LocalConfig::get_option(RESTORE_TOKEN_CONF_KEY);
|
|
if !restore_token.is_empty() {
|
|
args.insert(RESTORE_TOKEN.to_string(), Variant(Box::new(restore_token)));
|
|
}
|
|
// persist_mode may be configured by the user.
|
|
args.insert("persist_mode".to_string(), Variant(Box::new(2u32)));
|
|
}
|
|
args.insert(
|
|
"handle_token".to_string(),
|
|
Variant(Box::new("u3".to_string())),
|
|
);
|
|
// https://flatpak.github.io/xdg-desktop-portal/docs/doc-org.freedesktop.portal.ScreenCast.html
|
|
if is_server_running() {
|
|
args.insert("multiple".into(), Variant(Box::new(true)));
|
|
}
|
|
args.insert("types".into(), Variant(Box::new(1u32))); //| 2u32)));
|
|
|
|
if capture_cursor {
|
|
get_available_cursor_modes().ok().map(|modes| {
|
|
if modes & 0x2 != 0 {
|
|
args.insert("cursor_mode".to_string(), Variant(Box::new(2u32)));
|
|
}
|
|
});
|
|
}
|
|
|
|
let path = portal.select_sources(ses.clone(), args)?;
|
|
handle_response(
|
|
c,
|
|
path,
|
|
on_select_sources_response(
|
|
fd.clone(),
|
|
streams.clone(),
|
|
failure.clone(),
|
|
ses,
|
|
is_support_restore_token,
|
|
),
|
|
failure.clone(),
|
|
)?;
|
|
} else {
|
|
// TODO: support persist_mode for remote_desktop_portal
|
|
// https://flatpak.github.io/xdg-desktop-portal/docs/doc-org.freedesktop.portal.RemoteDesktop.html
|
|
|
|
args.insert(
|
|
"handle_token".to_string(),
|
|
Variant(Box::new("u2".to_string())),
|
|
);
|
|
args.insert("types".to_string(), Variant(Box::new(7u32)));
|
|
|
|
let path = portal.select_devices(ses.clone(), args)?;
|
|
handle_response(
|
|
c,
|
|
path,
|
|
on_select_devices_response(
|
|
fd.clone(),
|
|
streams.clone(),
|
|
failure.clone(),
|
|
ses,
|
|
is_support_restore_token,
|
|
),
|
|
failure.clone(),
|
|
)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
fn on_select_devices_response(
|
|
fd: Arc<Mutex<Option<OwnedFd>>>,
|
|
streams: Arc<Mutex<Vec<PwStreamInfo>>>,
|
|
failure: Arc<AtomicBool>,
|
|
session: dbus::Path<'static>,
|
|
is_support_restore_token: bool,
|
|
) -> impl Fn(
|
|
OrgFreedesktopPortalRequestResponse,
|
|
&SyncConnection,
|
|
&dbus::Message,
|
|
) -> Result<(), Box<dyn Error>> {
|
|
move |_: OrgFreedesktopPortalRequestResponse, c, _| {
|
|
let portal = get_portal(c);
|
|
let mut args: PropMap = HashMap::new();
|
|
args.insert(
|
|
"handle_token".to_string(),
|
|
Variant(Box::new("u3".to_string())),
|
|
);
|
|
// https://flatpak.github.io/xdg-desktop-portal/docs/doc-org.freedesktop.portal.ScreenCast.html
|
|
if is_server_running() {
|
|
args.insert("multiple".into(), Variant(Box::new(true)));
|
|
}
|
|
args.insert("types".into(), Variant(Box::new(1u32))); //| 2u32)));
|
|
|
|
let session = session.clone();
|
|
let path = portal.select_sources(session.clone(), args)?;
|
|
handle_response(
|
|
c,
|
|
path,
|
|
on_select_sources_response(
|
|
fd.clone(),
|
|
streams.clone(),
|
|
failure.clone(),
|
|
session,
|
|
is_support_restore_token,
|
|
),
|
|
failure.clone(),
|
|
)?;
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
fn on_select_sources_response(
|
|
fd: Arc<Mutex<Option<OwnedFd>>>,
|
|
streams: Arc<Mutex<Vec<PwStreamInfo>>>,
|
|
failure: Arc<AtomicBool>,
|
|
session: dbus::Path<'static>,
|
|
is_support_restore_token: bool,
|
|
) -> impl Fn(
|
|
OrgFreedesktopPortalRequestResponse,
|
|
&SyncConnection,
|
|
&dbus::Message,
|
|
) -> Result<(), Box<dyn Error>> {
|
|
move |_: OrgFreedesktopPortalRequestResponse, c, _| {
|
|
let portal = get_portal(c);
|
|
let mut args: PropMap = HashMap::new();
|
|
args.insert(
|
|
"handle_token".to_string(),
|
|
Variant(Box::new("u4".to_string())),
|
|
);
|
|
let path;
|
|
if is_server_running() {
|
|
path = screencast_portal::start(&portal, session.clone(), "", args)?;
|
|
} else {
|
|
path = remote_desktop_portal::start(&portal, session.clone(), "", args)?;
|
|
}
|
|
handle_response(
|
|
c,
|
|
path,
|
|
on_start_response(
|
|
fd.clone(),
|
|
streams.clone(),
|
|
session.clone(),
|
|
is_support_restore_token,
|
|
),
|
|
failure.clone(),
|
|
)?;
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
fn on_start_response(
|
|
fd: Arc<Mutex<Option<OwnedFd>>>,
|
|
streams: Arc<Mutex<Vec<PwStreamInfo>>>,
|
|
session: dbus::Path<'static>,
|
|
is_support_restore_token: bool,
|
|
) -> impl Fn(
|
|
OrgFreedesktopPortalRequestResponse,
|
|
&SyncConnection,
|
|
&dbus::Message,
|
|
) -> Result<(), Box<dyn Error>> {
|
|
move |r: OrgFreedesktopPortalRequestResponse, c, _| {
|
|
let portal = get_portal(c);
|
|
// See `is_server_running()` to understand the following code.
|
|
if is_server_running() {
|
|
if is_support_restore_token {
|
|
if let Some(restore_token) = r.results.get(RESTORE_TOKEN) {
|
|
if let Some(restore_token) = restore_token.as_str() {
|
|
config::LocalConfig::set_option(
|
|
RESTORE_TOKEN_CONF_KEY.to_owned(),
|
|
restore_token.to_owned(),
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
streams
|
|
.clone()
|
|
.lock()
|
|
.unwrap()
|
|
.append(&mut streams_from_response(r));
|
|
fd.clone()
|
|
.lock()
|
|
.unwrap()
|
|
.replace(portal.open_pipe_wire_remote(session.clone(), HashMap::new())?);
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
pub fn get_capturables() -> Result<Vec<PipeWireCapturable>, Box<dyn Error>> {
|
|
let mut rdp_connection = match RDP_SESSION_INFO.lock() {
|
|
Ok(conn) => conn,
|
|
Err(err) => return Err(Box::new(err)),
|
|
};
|
|
|
|
if rdp_connection.is_none() {
|
|
let (conn, fd, streams, session, is_support_restore_token) = request_remote_desktop(false)?;
|
|
let conn = Arc::new(conn);
|
|
|
|
let rdp_info = RdpSessionInfo {
|
|
conn,
|
|
streams,
|
|
fd,
|
|
session,
|
|
is_support_restore_token,
|
|
resolution: Arc::new(Mutex::new(None)),
|
|
};
|
|
*rdp_connection = Some(rdp_info);
|
|
}
|
|
|
|
let rdp_info = match rdp_connection.as_mut() {
|
|
Some(res) => res,
|
|
None => {
|
|
return Err(Box::new(DBusError("RDP response is None.".into())));
|
|
}
|
|
};
|
|
|
|
Ok(rdp_info
|
|
.streams
|
|
.iter()
|
|
.map(|s| {
|
|
PipeWireCapturable::new(
|
|
rdp_info.conn.clone(),
|
|
rdp_info.fd.clone(),
|
|
rdp_info.resolution.clone(),
|
|
s,
|
|
)
|
|
})
|
|
.collect())
|
|
}
|
|
|
|
// If `is_server_running()` is true, then `screencast_portal::start` is called.
|
|
// Otherwise, `remote_desktop_portal::start` is called.
|
|
//
|
|
// If `is_server_running()` is true, `--service` process is running,
|
|
// then we can use uinput as the input method.
|
|
// Otherwise, we have to use remote_desktop_portal's input method.
|
|
//
|
|
// `screencast_portal` supports restore_token and persist_mode if the version is greater than or equal to 4.
|
|
// `remote_desktop_portal` does not support restore_token and persist_mode.
|
|
pub(crate) fn is_server_running() -> bool {
|
|
let v = IS_SERVER_RUNNING.load(Ordering::SeqCst);
|
|
if v > 0 {
|
|
return v == 1;
|
|
}
|
|
|
|
let app_name = config::APP_NAME.read().unwrap().clone().to_lowercase();
|
|
let output = match Command::new(CMD_SH.as_str())
|
|
.arg("-c")
|
|
.arg(&format!("ps aux | grep {}", app_name))
|
|
.output()
|
|
{
|
|
Ok(output) => output,
|
|
Err(_) => {
|
|
return false;
|
|
}
|
|
};
|
|
|
|
let output_str = String::from_utf8_lossy(&output.stdout);
|
|
let is_running = output_str.contains(&format!("{} --server", app_name));
|
|
IS_SERVER_RUNNING.store(if is_running { 1 } else { 2 }, Ordering::SeqCst);
|
|
is_running
|
|
}
|
|
|
|
// The logical size reported by portal may be different from the size reported by `get_displays()`.
|
|
// So we need to use the workaround here.
|
|
// 1. openSUSE, KDE Plasma
|
|
// 2. Kubuntu 24.04 TLS, after running `sudo apt install plasma-workspace-wayland`
|
|
// Maybe it's a bug, and we can remove this workaround in the future.
|
|
pub fn try_fix_logical_size(shared_displays: &mut Vec<crate::Display>) {
|
|
if !is_server_running() {
|
|
return;
|
|
}
|
|
|
|
let wayland_displays = get_displays();
|
|
if wayland_displays.displays.is_empty() {
|
|
return;
|
|
}
|
|
|
|
for sd in shared_displays.iter_mut() {
|
|
if let crate::Display::WAYLAND(d) = sd {
|
|
let capturable = &mut d.0;
|
|
for wd in wayland_displays.displays.iter() {
|
|
if capturable.position.0 == wd.x && capturable.position.1 == wd.y {
|
|
if let Some(logical_size) = wd.logical_size {
|
|
if capturable.physical_size.0 != wd.width as usize
|
|
|| capturable.physical_size.1 != wd.height as usize
|
|
{
|
|
// If "Full Workspace" is selected in the portal dialog,
|
|
// the physical size reported by portal may not match the display info.
|
|
debug!(
|
|
"Physical size of capturable ({:?}) does not match display info: ({:?}) - ({:?}). Skipping logical size fix.",
|
|
capturable.position,
|
|
capturable.physical_size,
|
|
(wd.width as usize, wd.height as usize)
|
|
);
|
|
break;
|
|
}
|
|
|
|
if capturable.logical_size.0 != logical_size.0 as usize
|
|
|| capturable.logical_size.1 != logical_size.1 as usize
|
|
{
|
|
warn!(
|
|
"Fixing logical size of capturable from {:?} to {:?} based on display info {:?}.",
|
|
capturable.logical_size,
|
|
logical_size,
|
|
wd
|
|
);
|
|
capturable.logical_size =
|
|
(logical_size.0 as usize, logical_size.1 as usize);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn fill_displays(
|
|
mouse_move_to: impl Fn(i32, i32),
|
|
get_cursor_pos: fn() -> Option<(i32, i32)>,
|
|
shared_displays: &mut Vec<crate::Display>,
|
|
) -> ResultType<()> {
|
|
if !is_server_running() {
|
|
return Ok(());
|
|
}
|
|
|
|
let mut rdp_connection = RDP_SESSION_INFO.lock().unwrap();
|
|
let rdp_info = match rdp_connection.as_mut() {
|
|
Some(res) => res,
|
|
None => {
|
|
// Unreachable
|
|
bail!("RDP session info is None when filling display positions.");
|
|
}
|
|
};
|
|
|
|
let all_displays = get_displays();
|
|
if !HAS_POSITION_ATTR.load(Ordering::SeqCst) {
|
|
if all_displays.displays.len() > 1 {
|
|
debug!("Multiple Wayland displays detected, adjusting stream positions accordingly.");
|
|
try_fill_positions(
|
|
mouse_move_to,
|
|
get_cursor_pos,
|
|
&all_displays,
|
|
shared_displays,
|
|
&mut rdp_info.streams,
|
|
)?;
|
|
}
|
|
HAS_POSITION_ATTR.store(true, Ordering::SeqCst);
|
|
}
|
|
|
|
if all_displays.displays.len() > 1 {
|
|
sort_streams(&all_displays, shared_displays, &mut rdp_info.streams);
|
|
}
|
|
|
|
shared_displays.iter_mut().next().map(|d| {
|
|
if let crate::Display::WAYLAND(d) = d {
|
|
d.0.primary = true;
|
|
}
|
|
});
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn try_fill_positions(
|
|
mouse_move_to: impl Fn(i32, i32),
|
|
get_cursor_pos: fn() -> Option<(i32, i32)>,
|
|
displays: &Arc<Displays>,
|
|
shared_displays: &mut Vec<crate::Display>,
|
|
streams: &mut Vec<PwStreamInfo>,
|
|
) -> ResultType<()> {
|
|
let pipewire_display_offset = config::LocalConfig::get_option(PIPEWIRE_DISPLAY_OFFSET_CONF_KEY);
|
|
if !pipewire_display_offset.is_empty() {
|
|
if try_fill_positions_from_cache(
|
|
pipewire_display_offset,
|
|
displays,
|
|
shared_displays,
|
|
streams,
|
|
) {
|
|
return Ok(());
|
|
}
|
|
config::LocalConfig::set_option(PIPEWIRE_DISPLAY_OFFSET_CONF_KEY.to_owned(), "".to_owned());
|
|
}
|
|
|
|
let mut multi_matched_indices = Vec::new();
|
|
for (i, sd) in shared_displays.iter_mut().enumerate() {
|
|
if let crate::Display::WAYLAND(d) = sd {
|
|
let capturable = &mut d.0;
|
|
let mut match_count = 0;
|
|
for wd in displays.displays.iter() {
|
|
if capturable.physical_size.0 == wd.width as usize
|
|
&& capturable.physical_size.1 == wd.height as usize
|
|
{
|
|
capturable.position = (wd.x, wd.y);
|
|
if let Some(pw_stream) = streams.get_mut(i) {
|
|
pw_stream.position = (wd.x, wd.y);
|
|
}
|
|
match_count += 1;
|
|
}
|
|
}
|
|
if match_count == 0 {
|
|
warn!(
|
|
"No matching display found for capturable with size {:?}.",
|
|
capturable.physical_size
|
|
);
|
|
} else if match_count > 1 {
|
|
multi_matched_indices.push(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
if !multi_matched_indices.is_empty() {
|
|
fill_multi_matched_positions(
|
|
mouse_move_to,
|
|
get_cursor_pos,
|
|
displays,
|
|
shared_displays,
|
|
streams,
|
|
multi_matched_indices,
|
|
)?;
|
|
}
|
|
|
|
save_positions_to_cache(displays, shared_displays);
|
|
Ok(())
|
|
}
|
|
|
|
fn try_fill_positions_from_cache(
|
|
cache_str: String,
|
|
displays: &Arc<Displays>,
|
|
shared_displays: &mut Vec<crate::Display>,
|
|
streams: &mut Vec<PwStreamInfo>,
|
|
) -> bool {
|
|
let Ok(cache) = serde_json::from_str::<PipewireDisplayOffsetCache>(&cache_str) else {
|
|
return false;
|
|
};
|
|
|
|
if cache.offsets.len() != shared_displays.len() {
|
|
return false;
|
|
}
|
|
|
|
let display_key = PipewireDisplayOffsetCache::displays_to_key(displays);
|
|
if cache.display_key != display_key {
|
|
return false;
|
|
}
|
|
|
|
let restore_token = config::LocalConfig::get_option(RESTORE_TOKEN_CONF_KEY);
|
|
if cache.restore_token != restore_token {
|
|
return false;
|
|
}
|
|
|
|
for (i, sd) in shared_displays.iter_mut().enumerate() {
|
|
if let crate::Display::WAYLAND(d) = sd {
|
|
let capturable = &mut d.0;
|
|
if let Some((x_off, y_off)) = cache.offsets.get(i) {
|
|
capturable.position = (*x_off, *y_off);
|
|
if let Some(pw_stream) = streams.get_mut(i) {
|
|
pw_stream.position = (*x_off, *y_off);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
true
|
|
}
|
|
|
|
fn save_positions_to_cache(displays: &Arc<Displays>, shared_displays: &Vec<crate::Display>) {
|
|
let restore_token = config::LocalConfig::get_option(RESTORE_TOKEN_CONF_KEY);
|
|
if restore_token.is_empty() {
|
|
return;
|
|
}
|
|
|
|
let mut offsets = Vec::new();
|
|
for sd in shared_displays.iter() {
|
|
if let crate::Display::WAYLAND(d) = sd {
|
|
let capturable = &d.0;
|
|
offsets.push((capturable.position.0, capturable.position.1));
|
|
}
|
|
}
|
|
|
|
let display_key = PipewireDisplayOffsetCache::displays_to_key(displays);
|
|
let cache = PipewireDisplayOffsetCache {
|
|
display_key,
|
|
restore_token,
|
|
offsets,
|
|
};
|
|
|
|
if let Ok(s) = serde_json::to_string(&cache) {
|
|
config::LocalConfig::set_option(PIPEWIRE_DISPLAY_OFFSET_CONF_KEY.to_owned(), s);
|
|
}
|
|
}
|
|
|
|
fn compare_left_up_corner(w: usize, d1: &[u8], d2: &[u8]) -> bool {
|
|
if w == 0 {
|
|
return false;
|
|
}
|
|
if d1.len() != d2.len() {
|
|
return false;
|
|
}
|
|
let bpp = 4; // BGR0/RGB0
|
|
let stride = w.saturating_mul(bpp);
|
|
if stride == 0 || d1.len() < stride || d2.len() < stride {
|
|
return false;
|
|
}
|
|
let h = d1.len() / stride;
|
|
if h == 0 {
|
|
return false;
|
|
}
|
|
|
|
let roi_w = std::cmp::min(36, w);
|
|
let roi_h = std::cmp::min(36, h);
|
|
let mut diff_px = 0usize;
|
|
let total_px = roi_w * roi_h;
|
|
// Minimum number of differing pixels required to consider images different.
|
|
const MIN_DIFF_PIXELS: usize = 8;
|
|
// Divisor for threshold calculation: allows up to 1/8 of ROI pixels to differ before returning true.
|
|
const DIFF_THRESHOLD_DIVISOR: usize = 8;
|
|
let threshold = std::cmp::max(MIN_DIFF_PIXELS, total_px / DIFF_THRESHOLD_DIVISOR);
|
|
|
|
for y in 0..roi_h {
|
|
let row_off = y * stride;
|
|
for x in 0..roi_w {
|
|
let i = row_off + x * bpp;
|
|
let a = &d1[i..i + bpp];
|
|
let b = &d2[i..i + bpp];
|
|
if a != b {
|
|
diff_px += 1;
|
|
if diff_px >= threshold {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
false
|
|
}
|
|
|
|
fn fill_multi_matched_positions(
|
|
mouse_move_to: impl Fn(i32, i32),
|
|
get_cursor_pos: fn() -> Option<(i32, i32)>,
|
|
displays: &Arc<Displays>,
|
|
shared_displays: &mut Vec<crate::Display>,
|
|
streams: &mut Vec<PwStreamInfo>,
|
|
multi_matched_indices: Vec<usize>,
|
|
) -> ResultType<()> {
|
|
debug!(
|
|
"Multiple capturables ({:?}) match the same display size, attempting to disambiguate positions.",
|
|
&multi_matched_indices);
|
|
if multi_matched_indices.is_empty() {
|
|
return Ok(());
|
|
}
|
|
|
|
let is_support_embeded_cursor = get_available_cursor_modes()
|
|
.ok()
|
|
.map(|modes| modes & 0x2 != 0)
|
|
.unwrap_or(false);
|
|
if is_support_embeded_cursor {
|
|
fill_multi_matched_positions_cursor(
|
|
mouse_move_to,
|
|
get_cursor_pos,
|
|
displays,
|
|
shared_displays,
|
|
streams,
|
|
multi_matched_indices,
|
|
)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn mouse_move_to_(
|
|
mouse_move_to: &impl Fn(i32, i32),
|
|
get_cursor_pos: fn() -> Option<(i32, i32)>,
|
|
x: i32,
|
|
y: i32,
|
|
) {
|
|
const MOVE_MOUSE_TIMEOUT: Duration = Duration::from_millis(150);
|
|
let start = std::time::Instant::now();
|
|
while start.elapsed() < MOVE_MOUSE_TIMEOUT {
|
|
mouse_move_to(x, y);
|
|
std::thread::sleep(Duration::from_millis(20));
|
|
if let Some((x1, y1)) = get_cursor_pos() {
|
|
if x1 == x && y1 == y {
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
warn!(
|
|
"Failed to move mouse to ({}, {}) within timeout: {:?}.",
|
|
x, y, &MOVE_MOUSE_TIMEOUT
|
|
);
|
|
}
|
|
|
|
fn fill_multi_matched_positions_cursor(
|
|
mouse_move_to: impl Fn(i32, i32),
|
|
get_cursor_pos: fn() -> Option<(i32, i32)>,
|
|
displays: &Arc<Displays>,
|
|
shared_displays: &mut Vec<crate::Display>,
|
|
streams: &mut Vec<PwStreamInfo>,
|
|
multi_matched_indices: Vec<usize>,
|
|
) -> ResultType<()> {
|
|
// This creates a new remote desktop session for cursor-based position detection.
|
|
// The session is temporary, used only for disambiguation, and is dropped after detection completes.
|
|
let (conn, fd, streams_with_cursor, _session, _is_support_restore_token) =
|
|
request_remote_desktop(true)?;
|
|
let conn = Arc::new(conn);
|
|
|
|
let mut matched_indices = Vec::new();
|
|
const CAPTURE_TIMEOUT_MS: u64 = 1_000;
|
|
for idx in multi_matched_indices {
|
|
match (
|
|
shared_displays.get_mut(idx),
|
|
streams.get_mut(idx),
|
|
streams_with_cursor.get(idx),
|
|
) {
|
|
(Some(crate::Display::WAYLAND(d)), Some(pw_stream), Some(pw_stream_with_cursor)) => {
|
|
// Check if only one display matches the size
|
|
let mut match_count = 0;
|
|
for (i, wd) in displays.displays.iter().enumerate() {
|
|
if matched_indices.contains(&i) {
|
|
continue;
|
|
}
|
|
if d.0.physical_size.0 == wd.width as usize
|
|
&& d.0.physical_size.1 == wd.height as usize
|
|
{
|
|
match_count += 1;
|
|
}
|
|
}
|
|
if match_count == 0 {
|
|
error!(
|
|
"No matching display found for capturable with size {:?}.",
|
|
d.0.physical_size
|
|
);
|
|
continue;
|
|
}
|
|
if match_count == 1 {
|
|
for (i, wd) in displays.displays.iter().enumerate() {
|
|
if matched_indices.contains(&i) {
|
|
continue;
|
|
}
|
|
if d.0.physical_size.0 == wd.width as usize
|
|
&& d.0.physical_size.1 == wd.height as usize
|
|
{
|
|
d.0.position = (wd.x, wd.y);
|
|
pw_stream.position = (wd.x, wd.y);
|
|
matched_indices.push(i);
|
|
debug!(
|
|
"Disambiguated position for capturable with size {:?} to ({}, {}).",
|
|
d.0.physical_size, wd.x, wd.y
|
|
);
|
|
break;
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
|
|
// Move the mouse to a neutral position first,
|
|
// to avoid interference from previous position.
|
|
mouse_move_to_(&mouse_move_to, get_cursor_pos, 300, 300);
|
|
|
|
let mut rec = PipeWireRecorder::new(PipeWireCapturable {
|
|
dbus_conn: conn.clone(),
|
|
fd: fd.clone(),
|
|
path: pw_stream_with_cursor.path,
|
|
source_type: pw_stream_with_cursor.source_type,
|
|
primary: false,
|
|
position: pw_stream_with_cursor.position,
|
|
logical_size: pw_stream_with_cursor.size,
|
|
physical_size: (0, 0),
|
|
})?;
|
|
// Take first frame and copy owned buffer to avoid borrow across second capture
|
|
let (is_bgr, w, first_buf): (bool, usize, Vec<u8>) =
|
|
match rec.capture(CAPTURE_TIMEOUT_MS) {
|
|
Ok(PixelProvider::BGR0(w, _, data1)) => (true, w, data1.to_vec()),
|
|
Ok(PixelProvider::RGB0(w, _, data1)) => (false, w, data1.to_vec()),
|
|
Ok(_) => {
|
|
error!("Unexpected pixel format on first capture.");
|
|
continue;
|
|
}
|
|
Err(e) => {
|
|
error!(
|
|
"Failed to capture screen for position disambiguation: {}",
|
|
e
|
|
);
|
|
continue;
|
|
}
|
|
};
|
|
|
|
let matched_len = matched_indices.len();
|
|
for (i, wd) in displays.displays.iter().enumerate() {
|
|
if matched_indices.contains(&i) {
|
|
continue;
|
|
}
|
|
|
|
if wd.width as usize == d.0.physical_size.0
|
|
&& wd.height as usize == d.0.physical_size.1
|
|
{
|
|
mouse_move_to_(&mouse_move_to, get_cursor_pos, wd.x + 8, wd.y + 8);
|
|
rec.saved_raw_data.clear();
|
|
match rec.capture(CAPTURE_TIMEOUT_MS) {
|
|
Ok(PixelProvider::BGR0(_, _, data2)) if is_bgr => {
|
|
if compare_left_up_corner(w, &first_buf, data2) {
|
|
d.0.position = (wd.x, wd.y);
|
|
pw_stream.position = (wd.x, wd.y);
|
|
matched_indices.push(i);
|
|
debug!(
|
|
"Disambiguated position for capturable with size {:?} to ({}, {}).",
|
|
d.0.physical_size, wd.x, wd.y
|
|
);
|
|
break;
|
|
}
|
|
}
|
|
Ok(PixelProvider::RGB0(_, _, data2)) if !is_bgr => {
|
|
if compare_left_up_corner(w, &first_buf, data2) {
|
|
d.0.position = (wd.x, wd.y);
|
|
pw_stream.position = (wd.x, wd.y);
|
|
matched_indices.push(i);
|
|
debug!(
|
|
"Disambiguated position for capturable with size {:?} to ({}, {}).",
|
|
d.0.physical_size, wd.x, wd.y
|
|
);
|
|
break;
|
|
}
|
|
}
|
|
Ok(_) => {
|
|
// unreachable
|
|
error!("Pixel format changed between captures, cannot disambiguate position.");
|
|
}
|
|
Err(e) => {
|
|
error!(
|
|
"Failed to capture screen for position disambiguation: {}",
|
|
e
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if matched_len == matched_indices.len() {
|
|
error!(
|
|
"Failed to disambiguate position for capturable with size {:?}.",
|
|
d.0.physical_size
|
|
);
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn sort_streams(
|
|
displays: &Arc<Displays>,
|
|
shared_displays: &mut Vec<crate::Display>,
|
|
streams: &mut Vec<PwStreamInfo>,
|
|
) {
|
|
if streams.is_empty() {
|
|
// unreachable
|
|
error!("No streams available to sort.");
|
|
return;
|
|
}
|
|
|
|
// put the main display first, then the rest by the order of displays
|
|
let mut display_order: Vec<(i32, i32)> = Vec::new();
|
|
if let Some(d) = displays.displays.get(displays.primary) {
|
|
display_order.push((d.x, d.y));
|
|
}
|
|
for (i, d) in displays.displays.iter().enumerate() {
|
|
if i != displays.primary {
|
|
display_order.push((d.x, d.y));
|
|
}
|
|
}
|
|
|
|
let mut sorted_streams = Vec::new();
|
|
let mut sorted_shared_displays = Vec::new();
|
|
// Move matching items in order without cloning
|
|
for (x, y) in display_order.into_iter() {
|
|
for i in 0..streams.len() {
|
|
if streams[i].position.0 == x && streams[i].position.1 == y {
|
|
sorted_streams.push(streams.remove(i));
|
|
// shared_displays.len() must be equal to streams.len()
|
|
// But we still check the length to avoid panic
|
|
if shared_displays.len() > i {
|
|
sorted_shared_displays.push(shared_displays.remove(i));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
*streams = sorted_streams;
|
|
*shared_displays = sorted_shared_displays;
|
|
}
|