mirror of
https://github.com/rustdesk/rustdesk.git
synced 2025-06-07 09:52:49 +08:00
398 lines
11 KiB
Rust
398 lines
11 KiB
Rust
pub use self::vpxcodec::*;
|
|
use hbb_common::{
|
|
log,
|
|
message_proto::{video_frame, Chroma, VideoFrame},
|
|
};
|
|
use std::slice;
|
|
|
|
cfg_if! {
|
|
if #[cfg(quartz)] {
|
|
mod quartz;
|
|
pub use self::quartz::*;
|
|
} else if #[cfg(x11)] {
|
|
cfg_if! {
|
|
if #[cfg(feature="wayland")] {
|
|
mod linux;
|
|
mod wayland;
|
|
mod x11;
|
|
pub use self::linux::*;
|
|
pub use self::x11::Frame;
|
|
pub use self::wayland::set_map_err;
|
|
} else {
|
|
mod x11;
|
|
pub use self::x11::*;
|
|
}
|
|
}
|
|
} else if #[cfg(dxgi)] {
|
|
mod dxgi;
|
|
pub use self::dxgi::*;
|
|
} else if #[cfg(target_os = "android")] {
|
|
mod android;
|
|
pub use self::android::*;
|
|
}else {
|
|
//TODO: Fallback implementation.
|
|
}
|
|
}
|
|
|
|
pub mod codec;
|
|
pub mod convert;
|
|
#[cfg(feature = "hwcodec")]
|
|
pub mod hwcodec;
|
|
#[cfg(feature = "mediacodec")]
|
|
pub mod mediacodec;
|
|
pub mod vpxcodec;
|
|
pub use self::convert::*;
|
|
pub const STRIDE_ALIGN: usize = 64; // commonly used in libvpx vpx_img_alloc caller
|
|
pub const HW_STRIDE_ALIGN: usize = 0; // recommended by av_frame_get_buffer
|
|
|
|
pub mod aom;
|
|
pub mod record;
|
|
mod vpx;
|
|
|
|
#[repr(usize)]
|
|
#[derive(Copy, Clone)]
|
|
pub enum ImageFormat {
|
|
Raw,
|
|
ABGR,
|
|
ARGB,
|
|
}
|
|
#[repr(C)]
|
|
pub struct ImageRgb {
|
|
pub raw: Vec<u8>,
|
|
pub w: usize,
|
|
pub h: usize,
|
|
pub fmt: ImageFormat,
|
|
pub stride: usize,
|
|
}
|
|
|
|
impl ImageRgb {
|
|
pub fn new(fmt: ImageFormat, stride: usize) -> Self {
|
|
Self {
|
|
raw: Vec::new(),
|
|
w: 0,
|
|
h: 0,
|
|
fmt,
|
|
stride,
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
pub fn fmt(&self) -> ImageFormat {
|
|
self.fmt
|
|
}
|
|
|
|
#[inline]
|
|
pub fn stride(&self) -> usize {
|
|
self.stride
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
pub fn would_block_if_equal(old: &mut Vec<u8>, b: &[u8]) -> std::io::Result<()> {
|
|
// does this really help?
|
|
if b == &old[..] {
|
|
return Err(std::io::ErrorKind::WouldBlock.into());
|
|
}
|
|
old.resize(b.len(), 0);
|
|
old.copy_from_slice(b);
|
|
Ok(())
|
|
}
|
|
|
|
pub trait TraitCapturer {
|
|
// We doesn't support
|
|
#[cfg(not(any(target_os = "ios")))]
|
|
fn frame<'a>(&'a mut self, timeout: std::time::Duration) -> std::io::Result<Frame<'a>>;
|
|
|
|
#[cfg(windows)]
|
|
fn is_gdi(&self) -> bool;
|
|
#[cfg(windows)]
|
|
fn set_gdi(&mut self) -> bool;
|
|
}
|
|
|
|
pub trait TraitFrame {
|
|
fn data(&self) -> &[u8];
|
|
|
|
fn stride(&self) -> Vec<usize>;
|
|
|
|
fn pixfmt(&self) -> Pixfmt;
|
|
}
|
|
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
|
|
pub enum Pixfmt {
|
|
BGRA,
|
|
RGBA,
|
|
I420,
|
|
NV12,
|
|
I444,
|
|
}
|
|
|
|
pub struct EncodeYuvFormat {
|
|
pub pixfmt: Pixfmt,
|
|
pub w: usize,
|
|
pub h: usize,
|
|
pub stride: Vec<usize>,
|
|
pub u: usize,
|
|
pub v: usize,
|
|
}
|
|
|
|
#[cfg(x11)]
|
|
#[inline]
|
|
pub fn is_x11() -> bool {
|
|
hbb_common::platform::linux::is_x11_or_headless()
|
|
}
|
|
|
|
#[cfg(x11)]
|
|
#[inline]
|
|
pub fn is_cursor_embedded() -> bool {
|
|
if is_x11() {
|
|
x11::IS_CURSOR_EMBEDDED
|
|
} else {
|
|
wayland::is_cursor_embedded()
|
|
}
|
|
}
|
|
|
|
#[cfg(not(x11))]
|
|
#[inline]
|
|
pub fn is_cursor_embedded() -> bool {
|
|
false
|
|
}
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
pub enum CodecName {
|
|
VP8,
|
|
VP9,
|
|
AV1,
|
|
H264(String),
|
|
H265(String),
|
|
}
|
|
|
|
#[derive(PartialEq, Debug, Clone)]
|
|
pub enum CodecFormat {
|
|
VP8,
|
|
VP9,
|
|
AV1,
|
|
H264,
|
|
H265,
|
|
Unknown,
|
|
}
|
|
|
|
impl From<&VideoFrame> for CodecFormat {
|
|
fn from(it: &VideoFrame) -> Self {
|
|
match it.union {
|
|
Some(video_frame::Union::Vp8s(_)) => CodecFormat::VP8,
|
|
Some(video_frame::Union::Vp9s(_)) => CodecFormat::VP9,
|
|
Some(video_frame::Union::Av1s(_)) => CodecFormat::AV1,
|
|
Some(video_frame::Union::H264s(_)) => CodecFormat::H264,
|
|
Some(video_frame::Union::H265s(_)) => CodecFormat::H265,
|
|
_ => CodecFormat::Unknown,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<&CodecName> for CodecFormat {
|
|
fn from(value: &CodecName) -> Self {
|
|
match value {
|
|
CodecName::VP8 => Self::VP8,
|
|
CodecName::VP9 => Self::VP9,
|
|
CodecName::AV1 => Self::AV1,
|
|
CodecName::H264(_) => Self::H264,
|
|
CodecName::H265(_) => Self::H265,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl ToString for CodecFormat {
|
|
fn to_string(&self) -> String {
|
|
match self {
|
|
CodecFormat::VP8 => "VP8".into(),
|
|
CodecFormat::VP9 => "VP9".into(),
|
|
CodecFormat::AV1 => "AV1".into(),
|
|
CodecFormat::H264 => "H264".into(),
|
|
CodecFormat::H265 => "H265".into(),
|
|
CodecFormat::Unknown => "Unknow".into(),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
pub enum Error {
|
|
FailedCall(String),
|
|
BadPtr(String),
|
|
}
|
|
|
|
impl std::fmt::Display for Error {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::result::Result<(), std::fmt::Error> {
|
|
write!(f, "{:?}", self)
|
|
}
|
|
}
|
|
|
|
impl std::error::Error for Error {}
|
|
|
|
pub type Result<T> = std::result::Result<T, Error>;
|
|
|
|
#[macro_export]
|
|
macro_rules! generate_call_macro {
|
|
($func_name:ident, $allow_err:expr) => {
|
|
macro_rules! $func_name {
|
|
($x:expr) => {{
|
|
let result = unsafe { $x };
|
|
let result_int = unsafe { std::mem::transmute::<_, i32>(result) };
|
|
if result_int != 0 {
|
|
let message = format!(
|
|
"errcode={} {}:{}:{}:{}",
|
|
result_int,
|
|
module_path!(),
|
|
file!(),
|
|
line!(),
|
|
column!()
|
|
);
|
|
if $allow_err {
|
|
log::warn!("Failed to call {}, {}", stringify!($func_name), message);
|
|
} else {
|
|
return Err(crate::Error::FailedCall(message).into());
|
|
}
|
|
}
|
|
result
|
|
}};
|
|
}
|
|
};
|
|
}
|
|
|
|
#[macro_export]
|
|
macro_rules! generate_call_ptr_macro {
|
|
($func_name:ident) => {
|
|
macro_rules! $func_name {
|
|
($x:expr) => {{
|
|
let result = unsafe { $x };
|
|
let result_int = unsafe { std::mem::transmute::<_, isize>(result) };
|
|
if result_int == 0 {
|
|
return Err(crate::Error::BadPtr(format!(
|
|
"errcode={} {}:{}:{}:{}",
|
|
result_int,
|
|
module_path!(),
|
|
file!(),
|
|
line!(),
|
|
column!()
|
|
))
|
|
.into());
|
|
}
|
|
result
|
|
}};
|
|
}
|
|
};
|
|
}
|
|
|
|
pub trait GoogleImage {
|
|
fn width(&self) -> usize;
|
|
fn height(&self) -> usize;
|
|
fn stride(&self) -> Vec<i32>;
|
|
fn planes(&self) -> Vec<*mut u8>;
|
|
fn chroma(&self) -> Chroma;
|
|
fn get_bytes_per_row(w: usize, fmt: ImageFormat, stride: usize) -> usize {
|
|
let bytes_per_pixel = match fmt {
|
|
ImageFormat::Raw => 3,
|
|
ImageFormat::ARGB | ImageFormat::ABGR => 4,
|
|
};
|
|
// https://github.com/lemenkov/libyuv/blob/6900494d90ae095d44405cd4cc3f346971fa69c9/source/convert_argb.cc#L128
|
|
// https://github.com/lemenkov/libyuv/blob/6900494d90ae095d44405cd4cc3f346971fa69c9/source/convert_argb.cc#L129
|
|
(w * bytes_per_pixel + stride - 1) & !(stride - 1)
|
|
}
|
|
// rgb [in/out] fmt and stride must be set in ImageRgb
|
|
fn to(&self, rgb: &mut ImageRgb) {
|
|
rgb.w = self.width();
|
|
rgb.h = self.height();
|
|
let bytes_per_row = Self::get_bytes_per_row(rgb.w, rgb.fmt, rgb.stride());
|
|
rgb.raw.resize(rgb.h * bytes_per_row, 0);
|
|
let stride = self.stride();
|
|
let planes = self.planes();
|
|
unsafe {
|
|
match (self.chroma(), rgb.fmt()) {
|
|
(Chroma::I420, ImageFormat::Raw) => {
|
|
super::I420ToRAW(
|
|
planes[0],
|
|
stride[0],
|
|
planes[1],
|
|
stride[1],
|
|
planes[2],
|
|
stride[2],
|
|
rgb.raw.as_mut_ptr(),
|
|
bytes_per_row as _,
|
|
self.width() as _,
|
|
self.height() as _,
|
|
);
|
|
}
|
|
(Chroma::I420, ImageFormat::ARGB) => {
|
|
super::I420ToARGB(
|
|
planes[0],
|
|
stride[0],
|
|
planes[1],
|
|
stride[1],
|
|
planes[2],
|
|
stride[2],
|
|
rgb.raw.as_mut_ptr(),
|
|
bytes_per_row as _,
|
|
self.width() as _,
|
|
self.height() as _,
|
|
);
|
|
}
|
|
(Chroma::I420, ImageFormat::ABGR) => {
|
|
super::I420ToABGR(
|
|
planes[0],
|
|
stride[0],
|
|
planes[1],
|
|
stride[1],
|
|
planes[2],
|
|
stride[2],
|
|
rgb.raw.as_mut_ptr(),
|
|
bytes_per_row as _,
|
|
self.width() as _,
|
|
self.height() as _,
|
|
);
|
|
}
|
|
(Chroma::I444, ImageFormat::ARGB) => {
|
|
super::I444ToARGB(
|
|
planes[0],
|
|
stride[0],
|
|
planes[1],
|
|
stride[1],
|
|
planes[2],
|
|
stride[2],
|
|
rgb.raw.as_mut_ptr(),
|
|
bytes_per_row as _,
|
|
self.width() as _,
|
|
self.height() as _,
|
|
);
|
|
}
|
|
(Chroma::I444, ImageFormat::ABGR) => {
|
|
super::I444ToABGR(
|
|
planes[0],
|
|
stride[0],
|
|
planes[1],
|
|
stride[1],
|
|
planes[2],
|
|
stride[2],
|
|
rgb.raw.as_mut_ptr(),
|
|
bytes_per_row as _,
|
|
self.width() as _,
|
|
self.height() as _,
|
|
);
|
|
}
|
|
// (Chroma::I444, ImageFormat::Raw), new version libyuv have I444ToRAW
|
|
_ => log::error!("unsupported pixfmt:{:?}", self.chroma()),
|
|
}
|
|
}
|
|
}
|
|
fn data(&self) -> (&[u8], &[u8], &[u8]) {
|
|
unsafe {
|
|
let stride = self.stride();
|
|
let planes = self.planes();
|
|
let h = (self.height() as usize + 1) & !1;
|
|
let n = stride[0] as usize * h;
|
|
let y = slice::from_raw_parts(planes[0], n);
|
|
let n = stride[1] as usize * (h >> 1);
|
|
let u = slice::from_raw_parts(planes[1], n);
|
|
let v = slice::from_raw_parts(planes[2], n);
|
|
(y, u, v)
|
|
}
|
|
}
|
|
}
|