Skip to main content

dvb_si/tables/
mpe_fec.rs

1//! MPE-FEC section — ETSI EN 301 192 v1.7.1 §9.9 (table_id 0x78).
2//!
3//! Carries Reed-Solomon FEC data for a MPE-FEC frame in a DVB-H/IP datacast
4//! time-slice burst. Syntax per "Table 45 — MPE-FEC section"
5//! (`dvb-si/docs/en_301_192.md`, §9.9, PDF p. 57):
6//!
7//! ```text
8//! MPE-FEC_section() {
9//!   table_id                  8   (0x78)
10//!   section_syntax_indicator  1
11//!   private_indicator         1
12//!   reserved                  2
13//!   section_length           12
14//!   padding_columns           8
15//!   reserved_for_future_use   8
16//!   reserved                  2
17//!   reserved_for_future_use   5
18//!   current_next_indicator    1
19//!   section_number            8
20//!   last_section_number       8
21//!   real_time_parameters()   32
22//!   for (i=0; i<N; i++) { rs_data_byte 8 }
23//!   CRC_32                   32
24//! }
25//! ```
26//!
27//! Note the byte-3/4 "table_id_extension" slot is NOT a 16-bit identifier here:
28//! it is `padding_columns(8)` followed by an 8-bit `reserved_for_future_use`.
29//! There is no `version_number` field — byte 5 is reserved bits plus the
30//! `current_next_indicator`. `real_time_parameters` is the shared time-slice /
31//! MPE-FEC structure of "Table 46 — real time parameters" (§9.10): a 12-bit
32//! `delta_t`, two boundary flags, and an 18-bit `address`, packed into 4 bytes.
33//!
34//! MPE-FEC is a private section (byte-1 bit 6 = `private_indicator`), handled
35//! with the [`crate::tables::cit`] idiom. `rs_data` is a borrowed raw slice.
36//! No well-known PID — carriage is descriptor-signalled, so `PID = 0x0000`
37//! follows the [`crate::tables::dsmcc`] precedent.
38
39use crate::error::{Error, Result};
40use crate::traits::Table;
41use dvb_common::{Parse, Serialize};
42
43/// table_id for the MPE-FEC section.
44pub const TABLE_ID: u8 = 0x78;
45
46/// MPE-FEC has no well-known PID — its carriage is signalled via the
47/// time_slice_fec_identifier_descriptor. `0x0000` follows the DSM-CC precedent.
48pub const PID: u16 = 0x0000;
49
50/// Bytes 0-2: table_id (1) + flags + section_length (2).
51const HEADER_LEN: usize = 3;
52
53/// Bytes 3-7: padding_columns(1) + reserved_for_future_use(1)
54/// + reserved/rfu/cni byte(1) + section_number(1) + last_section_number(1).
55const EXTENSION_HEADER_LEN: usize = 5;
56
57/// Bytes 8-11: the 32-bit `real_time_parameters()`.
58const RTP_LEN: usize = 4;
59
60/// Bytes occupied by the trailing CRC-32 field.
61const CRC_LEN: usize = 4;
62
63/// Minimum total encoded length: header + extension + RTP + CRC.
64const MIN_LEN: usize = HEADER_LEN + EXTENSION_HEADER_LEN + RTP_LEN + CRC_LEN;
65
66/// Time-slicing / MPE-FEC real-time parameters (EN 301 192 §9.10, Table 46).
67///
68/// 32 bits: `delta_t(12)` | `table_boundary(1)` | `frame_boundary(1)`
69/// | `address(18)`.
70#[derive(Debug, Clone, Copy, PartialEq, Eq)]
71#[cfg_attr(feature = "serde", derive(serde::Serialize))]
72pub struct RealTimeParameters {
73    /// 12-bit `delta_t` — time until the start of the next burst (in units of
74    /// the MPE-FEC frame, see §9.10).
75    pub delta_t: u16,
76    /// `table_boundary` flag.
77    pub table_boundary: bool,
78    /// `frame_boundary` flag.
79    pub frame_boundary: bool,
80    /// 18-bit `address` — byte position of the first payload byte of this
81    /// section within the MPE-FEC frame table.
82    pub address: u32,
83}
84
85impl RealTimeParameters {
86    /// Decode the 4-byte real_time_parameters block.
87    fn from_bytes(b: [u8; RTP_LEN]) -> Self {
88        // delta_t(12) = b[0] | top 4 bits of b[1]
89        let delta_t = ((b[0] as u16) << 4) | ((b[1] >> 4) as u16);
90        let table_boundary = (b[1] & 0x08) != 0;
91        let frame_boundary = (b[1] & 0x04) != 0;
92        // address(18) = bottom 2 bits of b[1] | b[2] | b[3]
93        let address = (((b[1] & 0x03) as u32) << 16) | ((b[2] as u32) << 8) | (b[3] as u32);
94        RealTimeParameters {
95            delta_t,
96            table_boundary,
97            frame_boundary,
98            address,
99        }
100    }
101
102    /// Encode into the 4-byte real_time_parameters block.
103    fn to_bytes(self) -> [u8; RTP_LEN] {
104        let dt = self.delta_t & 0x0FFF;
105        let addr = self.address & 0x0003_FFFF;
106        [
107            (dt >> 4) as u8,
108            (((dt & 0x0F) as u8) << 4)
109                | (u8::from(self.table_boundary) << 3)
110                | (u8::from(self.frame_boundary) << 2)
111                | ((addr >> 16) as u8 & 0x03),
112            ((addr >> 8) & 0xFF) as u8,
113            (addr & 0xFF) as u8,
114        ]
115    }
116}
117
118/// MPE-FEC section (ETSI EN 301 192 v1.7.1 §9.9).
119#[derive(Debug, Clone, PartialEq, Eq)]
120#[cfg_attr(feature = "serde", derive(serde::Serialize))]
121pub struct MpeFec<'a> {
122    /// `private_indicator` bit from byte 1 (MPE-FEC is a private section).
123    pub private_indicator: bool,
124    /// `padding_columns` — number of padding RS columns (byte 3).
125    pub padding_columns: u8,
126    /// `current_next_indicator` bit.
127    pub current_next_indicator: bool,
128    /// section_number in the sub-table sequence.
129    pub section_number: u8,
130    /// last_section_number in the sub-table sequence.
131    pub last_section_number: u8,
132    /// The decoded real-time parameters block.
133    pub real_time_parameters: RealTimeParameters,
134    /// Raw Reed-Solomon data bytes (everything between the real_time_parameters
135    /// block and the CRC-32 trailer).
136    pub rs_data: &'a [u8],
137}
138
139impl<'a> Parse<'a> for MpeFec<'a> {
140    type Error = crate::error::Error;
141
142    fn parse(bytes: &'a [u8]) -> Result<Self> {
143        if bytes.len() < MIN_LEN {
144            return Err(Error::BufferTooShort {
145                need: MIN_LEN,
146                have: bytes.len(),
147                what: "MpeFec",
148            });
149        }
150
151        if bytes[0] != TABLE_ID {
152            return Err(Error::UnexpectedTableId {
153                table_id: bytes[0],
154                what: "MpeFec",
155                expected: &[TABLE_ID],
156            });
157        }
158
159        let section_length = (((bytes[1] & 0x0F) as usize) << 8) | bytes[2] as usize;
160        let total = HEADER_LEN + section_length;
161        if bytes.len() < total {
162            return Err(Error::SectionLengthOverflow {
163                declared: section_length,
164                available: bytes.len() - HEADER_LEN,
165            });
166        }
167
168        let private_indicator = (bytes[1] & 0x40) != 0;
169        let padding_columns = bytes[3];
170        // bytes[4] = reserved_for_future_use(8) — ignored on parse.
171        // bytes[5] = reserved(2) | reserved_for_future_use(5) | current_next_indicator(1)
172        let current_next_indicator = (bytes[5] & 0x01) != 0;
173        let section_number = bytes[6];
174        let last_section_number = bytes[7];
175
176        let rtp_start = HEADER_LEN + EXTENSION_HEADER_LEN;
177        let real_time_parameters = RealTimeParameters::from_bytes([
178            bytes[rtp_start],
179            bytes[rtp_start + 1],
180            bytes[rtp_start + 2],
181            bytes[rtp_start + 3],
182        ]);
183
184        let data_start = rtp_start + RTP_LEN;
185        let data_end = total - CRC_LEN;
186        let rs_data = &bytes[data_start..data_end];
187
188        Ok(MpeFec {
189            private_indicator,
190            padding_columns,
191            current_next_indicator,
192            section_number,
193            last_section_number,
194            real_time_parameters,
195            rs_data,
196        })
197    }
198}
199
200impl Serialize for MpeFec<'_> {
201    type Error = crate::error::Error;
202
203    fn serialized_len(&self) -> usize {
204        HEADER_LEN + EXTENSION_HEADER_LEN + RTP_LEN + self.rs_data.len() + CRC_LEN
205    }
206
207    fn serialize_into(&self, buf: &mut [u8]) -> Result<usize> {
208        let len = self.serialized_len();
209        if buf.len() < len {
210            return Err(Error::OutputBufferTooSmall {
211                need: len,
212                have: buf.len(),
213            });
214        }
215
216        let section_length = (len - HEADER_LEN) as u16;
217
218        // Byte 0: table_id.
219        buf[0] = TABLE_ID;
220        // Byte 1: section_syntax_indicator(1)=1 | private_indicator(1)
221        //         | reserved(2)=11 | section_length[11:8](4).
222        buf[1] = 0x80
223            | (u8::from(self.private_indicator) << 6)
224            | 0x30
225            | ((section_length >> 8) as u8 & 0x0F);
226        buf[2] = (section_length & 0xFF) as u8;
227
228        // Extension header.
229        buf[3] = self.padding_columns;
230        buf[4] = 0xFF; // reserved_for_future_use(8) emitted as 1s.
231                       // reserved(2)=11 | reserved_for_future_use(5)=11111 | current_next_indicator(1)
232        buf[5] = 0xFE | u8::from(self.current_next_indicator);
233        buf[6] = self.section_number;
234        buf[7] = self.last_section_number;
235
236        // real_time_parameters.
237        let rtp_start = HEADER_LEN + EXTENSION_HEADER_LEN;
238        buf[rtp_start..rtp_start + RTP_LEN].copy_from_slice(&self.real_time_parameters.to_bytes());
239
240        // rs_data.
241        let data_start = rtp_start + RTP_LEN;
242        let data_end = data_start + self.rs_data.len();
243        buf[data_start..data_end].copy_from_slice(self.rs_data);
244
245        // CRC-32 over everything up to (but not including) the CRC slot.
246        let crc = dvb_common::crc32_mpeg2::compute(&buf[..data_end]);
247        buf[data_end..len].copy_from_slice(&crc.to_be_bytes());
248
249        Ok(len)
250    }
251}
252
253impl<'a> Table<'a> for MpeFec<'a> {
254    const TABLE_ID: u8 = TABLE_ID;
255    const PID: u16 = PID;
256}
257
258impl<'a> crate::traits::TableDef<'a> for MpeFec<'a> {
259    const TABLE_ID_RANGES: &'static [(u8, u8)] = &[(TABLE_ID, TABLE_ID)];
260    const NAME: &'static str = "MPE_FEC";
261}
262
263#[cfg(test)]
264mod tests {
265    use super::*;
266
267    fn build_mpe_fec(
268        padding_columns: u8,
269        current_next: bool,
270        section_number: u8,
271        last_section_number: u8,
272        rtp: RealTimeParameters,
273        rs_data: &[u8],
274    ) -> Vec<u8> {
275        let s = MpeFec {
276            private_indicator: true,
277            padding_columns,
278            current_next_indicator: current_next,
279            section_number,
280            last_section_number,
281            real_time_parameters: rtp,
282            rs_data,
283        };
284        let mut buf = vec![0u8; s.serialized_len()];
285        s.serialize_into(&mut buf).unwrap();
286        buf
287    }
288
289    fn sample_rtp() -> RealTimeParameters {
290        RealTimeParameters {
291            delta_t: 0x0ABC,
292            table_boundary: true,
293            frame_boundary: false,
294            address: 0x0001_2345,
295        }
296    }
297
298    #[test]
299    fn parse_happy_path() {
300        let rs = [0x11u8, 0x22, 0x33, 0x44];
301        let bytes = build_mpe_fec(7, true, 1, 3, sample_rtp(), &rs);
302        let s = MpeFec::parse(&bytes).unwrap();
303        assert!(s.private_indicator);
304        assert_eq!(s.padding_columns, 7);
305        assert!(s.current_next_indicator);
306        assert_eq!(s.section_number, 1);
307        assert_eq!(s.last_section_number, 3);
308        assert_eq!(s.real_time_parameters, sample_rtp());
309        assert_eq!(s.rs_data, &rs[..]);
310    }
311
312    #[test]
313    fn parse_empty_rs_data() {
314        let bytes = build_mpe_fec(0, false, 0, 0, sample_rtp(), &[]);
315        let s = MpeFec::parse(&bytes).unwrap();
316        assert_eq!(s.padding_columns, 0);
317        assert!(!s.current_next_indicator);
318        assert!(s.rs_data.is_empty());
319        assert_eq!(s.real_time_parameters, sample_rtp());
320    }
321
322    #[test]
323    fn rtp_bit_packing_round_trips_extremes() {
324        // Exercise full-width values to pin the 12/1/1/18 split.
325        let rtp = RealTimeParameters {
326            delta_t: 0x0FFF,
327            table_boundary: false,
328            frame_boundary: true,
329            address: 0x0003_FFFF,
330        };
331        assert_eq!(RealTimeParameters::from_bytes(rtp.to_bytes()), rtp);
332    }
333
334    #[test]
335    fn parse_rejects_wrong_tag() {
336        let mut bytes = build_mpe_fec(0, true, 0, 0, sample_rtp(), &[]);
337        bytes[0] = 0x70; // not 0x78
338        assert!(matches!(
339            MpeFec::parse(&bytes).unwrap_err(),
340            Error::UnexpectedTableId { table_id: 0x70, .. }
341        ));
342    }
343
344    #[test]
345    fn parse_rejects_short_buffer() {
346        assert!(matches!(
347            MpeFec::parse(&[0x78, 0x80]).unwrap_err(),
348            Error::BufferTooShort { .. }
349        ));
350    }
351
352    #[test]
353    fn parse_rejects_section_length_overflow() {
354        let mut bytes = build_mpe_fec(0, true, 0, 0, sample_rtp(), &[]);
355        let fake_sl: u16 = (bytes.len() as u16) + 100 - HEADER_LEN as u16;
356        bytes[1] = (bytes[1] & 0xF0) | ((fake_sl >> 8) as u8 & 0x0F);
357        bytes[2] = (fake_sl & 0xFF) as u8;
358        assert!(matches!(
359            MpeFec::parse(&bytes).unwrap_err(),
360            Error::SectionLengthOverflow { .. }
361        ));
362    }
363
364    #[test]
365    fn serialize_round_trip() {
366        let rs = [0xDEu8, 0xAD, 0xBE, 0xEF, 0x00];
367        let original = MpeFec {
368            private_indicator: false,
369            padding_columns: 191,
370            current_next_indicator: false,
371            section_number: 2,
372            last_section_number: 4,
373            real_time_parameters: sample_rtp(),
374            rs_data: &rs,
375        };
376        let mut buf = vec![0u8; original.serialized_len()];
377        original.serialize_into(&mut buf).unwrap();
378        assert_eq!(MpeFec::parse(&buf).unwrap(), original);
379    }
380
381    #[test]
382    fn serialize_rejects_output_buffer_too_small() {
383        let s = MpeFec {
384            private_indicator: false,
385            padding_columns: 0,
386            current_next_indicator: true,
387            section_number: 0,
388            last_section_number: 0,
389            real_time_parameters: sample_rtp(),
390            rs_data: &[],
391        };
392        let mut buf = vec![0u8; 2];
393        assert!(matches!(
394            s.serialize_into(&mut buf).unwrap_err(),
395            Error::OutputBufferTooSmall { .. }
396        ));
397    }
398
399    #[test]
400    fn table_trait_constants() {
401        assert_eq!(<MpeFec as Table>::TABLE_ID, 0x78);
402        assert_eq!(<MpeFec as Table>::PID, 0x0000);
403    }
404
405    #[cfg(feature = "serde")]
406    #[test]
407    fn serde_json_serializes_fields() {
408        // Serialize-only: assert serialization yields valid, field-bearing JSON.
409        let rs = [0x01u8, 0x02];
410        let bytes = build_mpe_fec(12, true, 0, 0, sample_rtp(), &rs);
411        let s = MpeFec::parse(&bytes).unwrap();
412        let v: serde_json::Value = serde_json::to_value(&s).unwrap();
413        assert_eq!(v["padding_columns"], 12);
414        assert_eq!(v["current_next_indicator"], true);
415        assert_eq!(v["rs_data"], serde_json::json!([0x01, 0x02]));
416        assert_eq!(v["real_time_parameters"]["delta_t"], 0x0ABC);
417        assert_eq!(v["real_time_parameters"]["table_boundary"], true);
418    }
419}