-
-
Notifications
You must be signed in to change notification settings - Fork 838
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
video/external: Add WebCodecs H.264 decoder
- Loading branch information
1 parent
9cc1406
commit 540e896
Showing
5 changed files
with
277 additions
and
1 deletion.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,251 @@ | ||
use std::cell::RefCell; | ||
use std::rc::Rc; | ||
|
||
use crate::decoder::VideoDecoder; | ||
|
||
use ruffle_render::bitmap::BitmapFormat; | ||
use ruffle_video::error::Error; | ||
use ruffle_video::frame::{DecodedFrame, EncodedFrame, FrameDependency}; | ||
|
||
use js_sys::Uint8Array; | ||
use wasm_bindgen::prelude::*; | ||
use web_sys::{ | ||
DomException, EncodedVideoChunk, EncodedVideoChunkInit, EncodedVideoChunkType, | ||
VideoDecoder as WebVideoDecoder, VideoDecoderConfig, VideoDecoderInit, VideoFrame, | ||
VideoPixelFormat, | ||
}; | ||
|
||
/// H264 video decoder. | ||
pub struct H264Decoder { | ||
/// How many bytes are used to store the length of the NALU (1, 2, 3, or 4). | ||
length_size: u8, | ||
|
||
decoder: WebVideoDecoder, | ||
|
||
// Simply keeping these objects alive, as they are used by the JS side. | ||
// See: https://rustwasm.github.io/wasm-bindgen/examples/closures.html | ||
#[allow(dead_code)] | ||
output_callback: Closure<dyn Fn(VideoFrame)>, | ||
#[allow(dead_code)] | ||
error_callback: Closure<dyn Fn(DomException)>, | ||
|
||
last_frame: Rc<RefCell<Option<DecodedFrame>>>, | ||
} | ||
|
||
impl H264Decoder { | ||
/// `extradata` should hold "AVCC (MP4) format" decoder configuration, including PPS and SPS. | ||
/// Make sure it has any start code emulation prevention "three bytes" removed. | ||
pub fn new() -> Self { | ||
let last_frame = Rc::new(RefCell::new(None)); | ||
let lf = last_frame.clone(); | ||
// TODO: set up tracing log subscriber into these closures ... somehow | ||
let output = move |output: &VideoFrame| { | ||
let visible_rect = output.visible_rect().unwrap(); | ||
|
||
match output.format().unwrap() { | ||
VideoPixelFormat::I420 => { | ||
let mut data: Vec<u8> = | ||
vec![ | ||
0; | ||
visible_rect.width() as usize * visible_rect.height() as usize * 3 / 2 | ||
]; | ||
let _ = output.copy_to_with_u8_array(&mut data); | ||
last_frame.replace(Some(DecodedFrame::new( | ||
visible_rect.width() as u32, | ||
visible_rect.height() as u32, | ||
BitmapFormat::Yuv420p, | ||
data, | ||
))); | ||
} | ||
VideoPixelFormat::Bgrx => { | ||
let mut data: Vec<u8> = | ||
vec![0; visible_rect.width() as usize * visible_rect.height() as usize * 4]; | ||
let _ = output.copy_to_with_u8_array(&mut data); | ||
for pixel in data.chunks_mut(4) { | ||
pixel.swap(0, 2); | ||
pixel[3] = 0xff; | ||
} | ||
last_frame.replace(Some(DecodedFrame::new( | ||
visible_rect.width() as u32, | ||
visible_rect.height() as u32, | ||
BitmapFormat::Rgba, | ||
data, | ||
))); | ||
} | ||
_ => { | ||
panic!("unsupported pixel format: {:?}", output.format().unwrap()); | ||
} | ||
}; | ||
}; | ||
|
||
fn error(error: &DomException) { | ||
tracing::error!("webcodecs error {:}", error.message()); | ||
} | ||
|
||
let output_callback = Closure::new(move |frame| output(&frame)); | ||
let error_callback = Closure::new(move |exception| error(&exception)); | ||
|
||
let decoder = WebVideoDecoder::new(&VideoDecoderInit::new( | ||
error_callback.as_ref().unchecked_ref(), | ||
output_callback.as_ref().unchecked_ref(), | ||
)) | ||
.unwrap(); | ||
|
||
Self { | ||
length_size: 0, | ||
decoder, | ||
output_callback, | ||
error_callback, | ||
last_frame: lf, | ||
} | ||
} | ||
} | ||
|
||
impl Default for H264Decoder { | ||
fn default() -> Self { | ||
Self::new() | ||
} | ||
} | ||
|
||
impl VideoDecoder for H264Decoder { | ||
fn configure_decoder(&mut self, configuration_data: &[u8]) -> Result<(), Error> { | ||
// extradata[0]: configuration version, always 1 | ||
// extradata[1]: profile | ||
// extradata[2]: compatibility | ||
// extradata[3]: level | ||
// extradata[4]: 6 reserved bits | NALU length size - 1 | ||
|
||
// The codec string is the profile, compatibility, and level bytes as hex. | ||
|
||
self.length_size = (configuration_data[4] & 0b0000_0011) + 1; | ||
|
||
tracing::warn!("length_size: {}", self.length_size); | ||
|
||
let codec_string = format!( | ||
"avc1.{:02x}{:02x}{:02x}", | ||
configuration_data[1], configuration_data[2], configuration_data[3] | ||
); | ||
let config = VideoDecoderConfig::new(&codec_string); | ||
tracing::warn!("configuring decoder: {:?}", &configuration_data[1..4]); | ||
tracing::info!("{:?}", self.decoder.state()); | ||
|
||
let data = Uint8Array::from(configuration_data); | ||
config.set_description(&data); | ||
config.set_optimize_for_latency(true); | ||
let _ = self.decoder.configure(&config); | ||
tracing::info!("{:?}", self.decoder.state()); | ||
Ok(()) | ||
} | ||
|
||
fn preload_frame(&mut self, encoded_frame: EncodedFrame<'_>) -> Result<FrameDependency, Error> { | ||
tracing::warn!("preloading frame"); | ||
|
||
let mut is_key = false; | ||
let mut offset = 0; | ||
|
||
while offset < encoded_frame.data.len() { | ||
let mut encoded_len = 0; | ||
|
||
for i in 0..self.length_size { | ||
encoded_len = (encoded_len << 8) | encoded_frame.data[offset + i as usize] as u32; | ||
} | ||
|
||
tracing::warn!( | ||
"encoded_len: {}, chunk length: {}", | ||
encoded_len, | ||
encoded_frame.data.len() | ||
); | ||
|
||
let nal_unit_type = | ||
encoded_frame.data[offset + self.length_size as usize] & 0b0001_1111; | ||
|
||
tracing::warn!("nal_unit_type: {}", nal_unit_type); | ||
|
||
if nal_unit_type == 5u8 { | ||
is_key = true; | ||
} | ||
|
||
offset += encoded_len as usize + self.length_size as usize; | ||
} | ||
|
||
// 3.62 instantaneous decoding refresh (IDR) picture: | ||
// After the decoding of an IDR picture all following coded pictures in decoding order can | ||
// be decoded without inter prediction from any picture decoded prior to the IDR picture. | ||
if is_key { | ||
// openh264_sys::NAL_SLICE_IDR as u8 | ||
tracing::info!("is key"); | ||
Ok(FrameDependency::None) | ||
} else { | ||
tracing::info!("is not key"); | ||
Ok(FrameDependency::Past) | ||
} | ||
} | ||
|
||
fn decode_frame(&mut self, encoded_frame: EncodedFrame<'_>) -> Result<DecodedFrame, Error> { | ||
tracing::warn!("decoding frame {}", encoded_frame.frame_id); | ||
tracing::info!("{:?}", self.decoder.state()); | ||
|
||
tracing::warn!("queue size: {}", self.decoder.decode_queue_size()); | ||
|
||
let mut offset = 0; | ||
|
||
while offset < encoded_frame.data.len() { | ||
let mut encoded_len = 0; | ||
|
||
for i in 0..self.length_size { | ||
encoded_len = (encoded_len << 8) | encoded_frame.data[offset + i as usize] as u32; | ||
} | ||
|
||
tracing::warn!( | ||
"encoded_len: {}, chunk length: {}", | ||
encoded_len, | ||
encoded_frame.data.len() | ||
); | ||
|
||
let nal_unit_type = | ||
encoded_frame.data[offset + self.length_size as usize] & 0b0001_1111; | ||
|
||
tracing::warn!("nal_unit_type: {}", nal_unit_type); | ||
|
||
if nal_unit_type != 6u8 && nal_unit_type != 7u8 && nal_unit_type != 8u8 { | ||
// skipping SEI NALus, SPS NALus, and PPS NALus | ||
// 3.62 instantaneous decoding refresh (IDR) picture: | ||
// After the decoding of an IDR picture all following coded pictures in decoding order can | ||
// be decoded without inter prediction from any picture decoded prior to the IDR picture. | ||
let frame_type = if nal_unit_type == 5u8 { | ||
// openh264_sys::NAL_SLICE_IDR as u8 | ||
tracing::info!("is key"); | ||
EncodedVideoChunkType::Key | ||
} else { | ||
tracing::info!("is not key"); | ||
EncodedVideoChunkType::Delta | ||
}; | ||
let timestamp = (encoded_frame.frame_id as f64 - 1.0) * 1000000.0 * 0.5; | ||
tracing::warn!("timestamp: {}", timestamp); | ||
let data = Uint8Array::from( | ||
&encoded_frame.data | ||
[offset..offset + encoded_len as usize + self.length_size as usize], | ||
); | ||
let init = EncodedVideoChunkInit::new(&data, timestamp, frame_type); | ||
let chunk = EncodedVideoChunk::new(&init).unwrap(); | ||
|
||
let _ = self.decoder.decode(&chunk); | ||
tracing::info!("{:?}", self.decoder.state()); | ||
} | ||
|
||
offset += encoded_len as usize + self.length_size as usize; | ||
} | ||
|
||
assert!( | ||
offset == encoded_frame.data.len(), | ||
"Incomplete NALu at the end" | ||
); | ||
|
||
match self.last_frame.borrow_mut().take() { | ||
Some(frame) => Ok(frame), | ||
None => Err(Error::DecoderError( | ||
"No output frame produced by the decoder".into(), | ||
)), | ||
} | ||
} | ||
} |