Implement encoding SimpleBlocks

This commit is contained in:
Tangent 128 2017-08-12 16:48:24 -04:00
parent 01e81c6a9f
commit dd376e146e
2 changed files with 37 additions and 2 deletions

View file

@ -12,6 +12,13 @@ pub fn main() {
encode_webm_element(WebmElement::Cluster, &mut cursor).unwrap(); encode_webm_element(WebmElement::Cluster, &mut cursor).unwrap();
encode_webm_element(WebmElement::Timecode(0), &mut cursor).unwrap(); encode_webm_element(WebmElement::Timecode(0), &mut cursor).unwrap();
encode_webm_element(WebmElement::SimpleBlock {
track: 3,
flags: 0x0,
timecode: 123,
data: "Hello, World".as_bytes()
}, &mut cursor).unwrap();
encode_webm_element(WebmElement::Cluster, &mut cursor).unwrap(); encode_webm_element(WebmElement::Cluster, &mut cursor).unwrap();
encode_webm_element(WebmElement::Timecode(1000), &mut cursor).unwrap(); encode_webm_element(WebmElement::Timecode(1000), &mut cursor).unwrap();

View file

@ -1,5 +1,5 @@
use std::io::{Error as IoError, ErrorKind, Result as IoResult, Write, Seek}; use std::io::{Cursor, Error as IoError, ErrorKind, Result as IoResult, Write, Seek};
use bytes::{BigEndian, ByteOrder}; use bytes::{BigEndian, BufMut, ByteOrder};
use ebml::*; use ebml::*;
const SEGMENT_ID: u64 = 0x08538067; const SEGMENT_ID: u64 = 0x08538067;
@ -80,6 +80,33 @@ fn decode_simple_block(bytes: &[u8]) -> Result<WebmElement, Error> {
} }
} }
pub fn encode_simple_block<T: Write>(element: WebmElement, output: &mut T) -> IoResult<()> {
if let WebmElement::SimpleBlock{
track,
timecode,
flags,
data
} = element {
// limiting number of tracks for now
if track > 31 {
return Err(IoError::new(ErrorKind::InvalidInput, WriteError::OutOfRange));
}
let header_len = 1 + 2 + 1;
encode_tag_header(SIMPLE_BLOCK_ID, Varint::Value((header_len + data.len()) as u64), output)?;
encode_varint(Varint::Value(track), output)?;
let mut buffer = Cursor::new([0; 3]);
buffer.put_i16::<BigEndian>(timecode);
buffer.put_u8(flags);
output.write_all(&buffer.get_ref()[..])?;
output.write_all(data)
} else {
Err(IoError::new(ErrorKind::InvalidInput, WriteError::OutOfRange))
}
}
pub fn encode_webm_element<T: Write + Seek>(element: WebmElement, output: &mut T) -> IoResult<()> { pub fn encode_webm_element<T: Write + Seek>(element: WebmElement, output: &mut T) -> IoResult<()> {
match element { match element {
WebmElement::EbmlHead => encode_element(EBML_HEAD_ID, output, |output| { WebmElement::EbmlHead => encode_element(EBML_HEAD_ID, output, |output| {
@ -90,6 +117,7 @@ pub fn encode_webm_element<T: Write + Seek>(element: WebmElement, output: &mut T
WebmElement::Cues => Ok(()), WebmElement::Cues => Ok(()),
WebmElement::Cluster => encode_tag_header(CLUSTER_ID, Varint::Unknown, output), WebmElement::Cluster => encode_tag_header(CLUSTER_ID, Varint::Unknown, output),
WebmElement::Timecode(time) => encode_integer(TIMECODE_ID, time, output), WebmElement::Timecode(time) => encode_integer(TIMECODE_ID, time, output),
WebmElement::SimpleBlock {..} => encode_simple_block(element, output),
_ => Err(IoError::new(ErrorKind::InvalidInput, WriteError::OutOfRange)) _ => Err(IoError::new(ErrorKind::InvalidInput, WriteError::OutOfRange))
} }
} }