Skip to content

Commit

Permalink
Add hardware encoding to the compositor API.
Browse files Browse the repository at this point in the history
  • Loading branch information
jerzywilczek committed Nov 21, 2024
1 parent 6731fb5 commit 6a8dd1e
Show file tree
Hide file tree
Showing 8 changed files with 177 additions and 157 deletions.
30 changes: 28 additions & 2 deletions compositor_api/src/types/from_register_input.rs
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,10 @@ impl TryFrom<InputRtpAudioOptions> for InputAudioStream {
}
}

#[cfg(not(feature = "vk-video"))]
const NO_VULKAN_VIDEO: &str =
"Requested `vulkan_video` decoder, but this binary was compiled without the `vk-video` feature.";

impl TryFrom<RtpInput> for pipeline::RegisterInputOptions {
type Error = TypeError;

Expand All @@ -106,10 +110,20 @@ impl TryFrom<RtpInput> for pipeline::RegisterInputOptions {
.as_ref()
.map(|video| {
Ok(input::rtp::InputVideoStream {
options: match video {
InputRtpVideoOptions::FfmepgH264 => decoder::VideoDecoderOptions {
options: match video.decoder {
VideoDecoder::FfmpegH264 => decoder::VideoDecoderOptions {
decoder: pipeline::VideoDecoder::FFmpegH264,
},

#[cfg(feature = "vk-video")]
VideoDecoder::VulkanVideo => decoder::VideoDecoderOptions {
decoder: pipeline::VideoDecoder::VulkanVideoH264,
},

#[cfg(not(feature = "vk-video"))]
VideoDecoder::VulkanVideo => {
return Err(TypeError::new(NO_VULKAN_VIDEO))
}
},
})
})
Expand Down Expand Up @@ -146,6 +160,7 @@ impl TryFrom<Mp4Input> for pipeline::RegisterInputOptions {
required,
offset_ms,
should_loop,
video_decoder,
} = value;

const BAD_URL_PATH_SPEC: &str =
Expand All @@ -165,10 +180,21 @@ impl TryFrom<Mp4Input> for pipeline::RegisterInputOptions {
buffer_duration: None,
};

let video_decoder = match video_decoder.unwrap_or(VideoDecoder::FfmpegH264) {
VideoDecoder::FfmpegH264 => pipeline::VideoDecoder::FFmpegH264,

#[cfg(feature = "vk-video")]
VideoDecoder::VulkanVideo => pipeline::VideoDecoder::VulkanVideoH264,

#[cfg(not(feature = "vk-video"))]
VideoDecoder::VulkanVideo => return Err(TypeError::new(NO_VULKAN_VIDEO)),
};

Ok(pipeline::RegisterInputOptions {
input_options: input::InputOptions::Mp4(input::mp4::Mp4Options {
source,
should_loop: should_loop.unwrap_or(false),
video_decoder,
}),
queue_options,
})
Expand Down
25 changes: 21 additions & 4 deletions compositor_api/src/types/register_input.rs
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,8 @@ pub struct Mp4Input {
/// Offset in milliseconds relative to the pipeline start (start request). If offset is
/// not defined then stream is synchronized based on the first frames delivery time.
pub offset_ms: Option<f64>,
/// (**default=`ffmpeg_h264`**) The decoder to use for decoding video.
pub video_decoder: Option<VideoDecoder>,
}

/// Capture streams from devices connected to Blackmagic DeckLink card.
Expand Down Expand Up @@ -122,8 +124,23 @@ pub enum InputRtpAudioOptions {
}

#[derive(Debug, Serialize, Deserialize, Clone, JsonSchema)]
#[serde(tag = "decoder", rename_all = "snake_case", deny_unknown_fields)]
pub enum InputRtpVideoOptions {
#[serde(rename = "ffmpeg_h264")]
FfmepgH264,
#[serde(rename_all = "snake_case", deny_unknown_fields)]
pub struct InputRtpVideoOptions {
pub decoder: VideoDecoder,
}

#[derive(Debug, Serialize, Deserialize, Clone, JsonSchema)]
#[serde(rename_all = "snake_case", deny_unknown_fields)]
pub enum VideoDecoder {
/// Use the software decoder based on ffmpeg.
FfmpegH264,

/// Use hardware decoder based on Vulkan Video.
///
/// This should be faster and more scalable than teh ffmpeg decoder, if the hardware and OS
/// support it.
///
/// This requires hardware that supports Vulkan Video. Another requirement is this program has
/// to be compiled with the `vk-video` feature enabled.
VulkanVideo,
}
11 changes: 9 additions & 2 deletions compositor_pipeline/src/pipeline/input/mp4.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,10 @@ use crossbeam_channel::Receiver;
use tracing::error;

use crate::{
pipeline::decoder::{AudioDecoderOptions, VideoDecoderOptions},
pipeline::{
decoder::{AudioDecoderOptions, VideoDecoderOptions},
VideoDecoder,
},
queue::PipelineEvent,
};

Expand All @@ -20,6 +23,7 @@ pub mod mp4_file_reader;
pub struct Mp4Options {
pub source: Source,
pub should_loop: bool,
pub video_decoder: VideoDecoder,
}

pub(crate) enum Mp4ReaderOptions {
Expand Down Expand Up @@ -96,13 +100,16 @@ impl Mp4 {
should_loop: options.should_loop,
},
input_id.clone(),
options.video_decoder,
)?;

let (video_reader, video_receiver) = match video {
Some((reader, receiver)) => {
let input_receiver = VideoInputReceiver::Encoded {
chunk_receiver: receiver,
decoder_options: reader.decoder_options(),
decoder_options: VideoDecoderOptions {
decoder: options.video_decoder,
},
};
(Some(reader), Some(input_receiver))
}
Expand Down
8 changes: 5 additions & 3 deletions compositor_pipeline/src/pipeline/input/mp4/mp4_file_reader.rs
Original file line number Diff line number Diff line change
Expand Up @@ -125,6 +125,7 @@ impl Mp4FileReader<VideoDecoderOptions> {
pub(crate) fn new_video(
options: Mp4ReaderOptions,
input_id: InputId,
video_decoder: VideoDecoder,
) -> Result<Option<(Mp4FileReader<VideoDecoderOptions>, ChunkReceiver)>, Mp4Error> {
let stop_thread = Arc::new(AtomicBool::new(false));
let span = span!(Level::INFO, "MP4 video", input_id = input_id.to_string());
Expand All @@ -136,7 +137,7 @@ impl Mp4FileReader<VideoDecoderOptions> {
Self::new(
input_file,
size,
Self::find_h264_info,
|r| Self::find_h264_info(r, video_decoder),
None,
stop_thread,
span,
Expand All @@ -153,7 +154,7 @@ impl Mp4FileReader<VideoDecoderOptions> {
Self::new(
reader,
size,
Self::find_h264_info,
|r| Self::find_h264_info(r, video_decoder),
Some(fragment_receiver),
stop_thread,
span,
Expand All @@ -165,6 +166,7 @@ impl Mp4FileReader<VideoDecoderOptions> {

fn find_h264_info<Reader: Read + Seek + Send + 'static>(
reader: &mp4::Mp4Reader<Reader>,
video_decoder: VideoDecoder,
) -> Option<TrackInfo<VideoDecoderOptions, impl FnMut(mp4::Mp4Sample) -> Bytes>> {
let (&track_id, track, avc) = reader.tracks().iter().find_map(|(id, track)| {
let track_type = track.track_type().ok()?;
Expand Down Expand Up @@ -234,7 +236,7 @@ impl Mp4FileReader<VideoDecoderOptions> {
};

let decoder_options = VideoDecoderOptions {
decoder: VideoDecoder::FFmpegH264,
decoder: video_decoder,
};

Some(TrackInfo {
Expand Down
1 change: 1 addition & 0 deletions integration_tests/examples/encoded_channel_output.rs
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,7 @@ fn main() {
input_options: InputOptions::Mp4(Mp4Options {
source: Source::File(root_dir.join(BUNNY_FILE_PATH)),
should_loop: false,
video_decoder: compositor_pipeline::pipeline::VideoDecoder::FFmpegH264,
}),
queue_options: QueueInputOptions {
required: true,
Expand Down
66 changes: 66 additions & 0 deletions integration_tests/examples/mp4_hw_dec.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
use anyhow::Result;
use compositor_api::types::Resolution;
use serde_json::json;
use std::time::Duration;

use integration_tests::{
examples::{self, run_example},
ffmpeg::start_ffmpeg_receive,
};

const VIDEO_RESOLUTION: Resolution = Resolution {
width: 1280,
height: 720,
};

const IP: &str = "127.0.0.1";
const OUTPUT_PORT: u16 = 8004;

fn main() {
run_example(client_code);
}

fn client_code() -> Result<()> {
start_ffmpeg_receive(Some(OUTPUT_PORT), None)?;

examples::post(
"input/input_1/register",
&json!({
"type": "mp4",
"path": "examples/assets/BigBuckBunny.mp4",
"video_decoder": "vulkan_video",
}),
)?;

examples::post(
"output/output_1/register",
&json!({
"type": "rtp_stream",
"port": OUTPUT_PORT,
"ip": IP,
"video": {
"resolution": {
"width": VIDEO_RESOLUTION.width,
"height": VIDEO_RESOLUTION.height,
},
"encoder": {
"type": "ffmpeg_h264",
"preset": "ultrafast"
},
"initial": {
"root": {
"id": "input_1",
"type": "input_stream",
"input_id": "input_1",
}
}
}
}),
)?;

std::thread::sleep(Duration::from_millis(500));

examples::post("start", &json!({}))?;

Ok(())
}
1 change: 1 addition & 0 deletions integration_tests/examples/raw_channel_output.rs
Original file line number Diff line number Diff line change
Expand Up @@ -113,6 +113,7 @@ fn main() {
input_options: InputOptions::Mp4(Mp4Options {
source: Source::File(root_dir().join(BUNNY_FILE_PATH)),
should_loop: false,
video_decoder: compositor_pipeline::pipeline::VideoDecoder::FFmpegH264,
}),
queue_options: QueueInputOptions {
required: true,
Expand Down
Loading

0 comments on commit 6a8dd1e

Please sign in to comment.