#[cfg(test)]
mod h264_test;
use crate::{
error::Error,
packetizer::{Depacketizer, Payloader},
};
use anyhow::Result;
use bytes::{BufMut, Bytes, BytesMut};
#[derive(Debug, Copy, Clone)]
pub struct H264Payloader;
pub const STAPA_NALU_TYPE: u8 = 24;
pub const FUA_NALU_TYPE: u8 = 28;
pub const FUA_HEADER_SIZE: isize = 2;
pub const STAPA_HEADER_SIZE: usize = 1;
pub const STAPA_NALU_LENGTH_SIZE: usize = 2;
pub const NALU_TYPE_BITMASK: u8 = 0x1F;
pub const NALU_REF_IDC_BITMASK: u8 = 0x60;
pub const FUA_START_BITMASK: u8 = 0x80;
pub static ANNEXB_NALUSTART_CODE: Bytes = Bytes::from_static(&[0x00, 0x00, 0x00, 0x01]);
fn next_ind(nalu: &Bytes, start: usize) -> (isize, isize) {
let mut zero_count = 0;
for (i, &b) in nalu[start..].iter().enumerate() {
if b == 0 {
zero_count += 1;
continue;
} else if b == 1 && zero_count >= 2 {
return ((start + i - zero_count) as isize, zero_count as isize + 1);
}
zero_count = 0
}
(-1, -1)
}
fn emit(nalu: &Bytes, mtu: usize, payloads: &mut Vec<Bytes>) {
let nalu_type = nalu[0] & NALU_TYPE_BITMASK;
let nalu_ref_idc = nalu[0] & NALU_REF_IDC_BITMASK;
if nalu_type == 9 || nalu_type == 12 {
return;
}
if nalu.len() <= mtu {
payloads.push(nalu.clone());
return;
}
let max_fragment_size = mtu as isize - FUA_HEADER_SIZE;
let nalu_data = nalu;
let mut nalu_data_index = 1;
let nalu_data_length = nalu.len() as isize - nalu_data_index;
let mut nalu_data_remaining = nalu_data_length;
if std::cmp::min(max_fragment_size, nalu_data_remaining) <= 0 {
return;
}
while nalu_data_remaining > 0 {
let current_fragment_size = std::cmp::min(max_fragment_size, nalu_data_remaining);
let mut out = BytesMut::with_capacity((FUA_HEADER_SIZE + current_fragment_size) as usize);
let b0 = FUA_NALU_TYPE | nalu_ref_idc;
out.put_u8(b0);
let mut b1 = nalu_type;
if nalu_data_remaining == nalu_data_length {
b1 |= 1 << 7;
} else if nalu_data_remaining - current_fragment_size == 0 {
b1 |= 1 << 6;
}
out.put_u8(b1);
out.put(
&nalu_data
[nalu_data_index as usize..(nalu_data_index + current_fragment_size) as usize],
);
payloads.push(out.freeze());
nalu_data_remaining -= current_fragment_size;
nalu_data_index += current_fragment_size;
}
}
impl Payloader for H264Payloader {
fn payload(&self, mtu: usize, payload: &Bytes) -> Result<Vec<Bytes>> {
if payload.is_empty() || mtu == 0 {
return Ok(vec![]);
}
let mut payloads = vec![];
let (mut next_ind_start, mut next_ind_len) = next_ind(payload, 0);
if next_ind_start == -1 {
emit(payload, mtu, &mut payloads);
} else {
while next_ind_start != -1 {
let prev_start = (next_ind_start + next_ind_len) as usize;
let (next_ind_start2, next_ind_len2) = next_ind(payload, prev_start);
next_ind_start = next_ind_start2;
next_ind_len = next_ind_len2;
if next_ind_start != -1 {
emit(
&payload.slice(prev_start..next_ind_start as usize),
mtu,
&mut payloads,
);
} else {
emit(&payload.slice(prev_start..), mtu, &mut payloads);
}
}
}
Ok(payloads)
}
fn clone_to(&self) -> Box<dyn Payloader + Send + Sync> {
Box::new(*self)
}
}
#[derive(PartialEq, Debug, Default, Clone)]
pub struct H264Packet {
pub payload: Bytes,
}
impl Depacketizer for H264Packet {
fn depacketize(&mut self, packet: &Bytes) -> Result<()> {
if packet.len() <= 2 {
return Err(Error::ErrShortPacket.into());
}
let mut payload = BytesMut::new();
let b0 = packet[0];
let nalu_type = b0 & NALU_TYPE_BITMASK;
match nalu_type {
1..=23 => {
payload.put(&*ANNEXB_NALUSTART_CODE);
payload.put(&*packet.clone());
self.payload = payload.freeze();
}
STAPA_NALU_TYPE => {
let mut curr_offset = STAPA_HEADER_SIZE;
while curr_offset < packet.len() {
let nalu_size =
((packet[curr_offset] as usize) << 8) | packet[curr_offset + 1] as usize;
curr_offset += STAPA_NALU_LENGTH_SIZE;
if packet.len() < curr_offset + nalu_size {
return Err(Error::StapASizeLargerThanBuffer(
nalu_size,
packet.len() - curr_offset,
)
.into());
}
payload.put(&*ANNEXB_NALUSTART_CODE);
payload.put(&*packet.slice(curr_offset..curr_offset + nalu_size));
curr_offset += nalu_size;
}
self.payload = payload.freeze();
}
FUA_NALU_TYPE => {
if packet.len() < FUA_HEADER_SIZE as usize {
return Err(Error::ErrShortPacket.into());
}
let b1 = packet[1];
if b1 & FUA_START_BITMASK != 0 {
let nalu_ref_idc = b0 & NALU_REF_IDC_BITMASK;
let fragmented_nalu_type = b1 & NALU_TYPE_BITMASK;
payload.put(&*ANNEXB_NALUSTART_CODE);
payload.put_u8(nalu_ref_idc | fragmented_nalu_type);
payload.put_slice(&*packet.slice(FUA_HEADER_SIZE as usize..));
self.payload = payload.freeze();
} else {
self.payload = packet.slice(FUA_HEADER_SIZE as usize..);
}
}
_ => return Err(Error::NaluTypeIsNotHandled(nalu_type).into()),
}
Ok(())
}
}