Skip to main content

dvb_si/descriptors/
content.rs

1//! Content Descriptor — ETSI EN 300 468 §6.2.9 (tag 0x54).
2//!
3//! Carried inside EIT. Classifies the event's genre via a two-nibble
4//! content type plus an 8-bit broadcaster-specific user byte.
5
6use super::descriptor_body;
7use crate::error::{Error, Result};
8use dvb_common::{Parse, Serialize};
9
10/// Descriptor tag for content_descriptor.
11pub const TAG: u8 = 0x54;
12const HEADER_LEN: usize = 2;
13const ENTRY_LEN: usize = 2;
14
15/// Content genre level-1 broad category — EN 300 468 Table 29.
16#[derive(Debug, Clone, Copy, PartialEq, Eq)]
17#[cfg_attr(feature = "serde", derive(serde::Serialize))]
18#[non_exhaustive]
19pub enum ContentGenre {
20    /// 0x0 — undefined content.
21    UndefinedContent,
22    /// 0x1 — Movie/Drama.
23    MovieDrama,
24    /// 0x2 — News/Current Affairs.
25    NewsCurrentAffairs,
26    /// 0x3 — Show/Game Show.
27    ShowGameShow,
28    /// 0x4 — Sports.
29    Sports,
30    /// 0x5 — Children/Youth programmes.
31    ChildrenYouth,
32    /// 0x6 — Music/Ballet/Dance.
33    MusicBalletDance,
34    /// 0x7 — Arts/Culture (without music).
35    ArtsCulture,
36    /// 0x8 — Social/Political issues/Economics.
37    SocialPoliticalEconomics,
38    /// 0x9 — Education/Science/Factual topics.
39    EducationScienceFactual,
40    /// 0xA — Leisure hobbies.
41    LeisureHobbies,
42    /// 0xB — Special characteristics.
43    SpecialCharacteristics,
44    /// 0xC — Adult.
45    Adult,
46    /// 0xD..=0xE — reserved for future use, preserved verbatim.
47    Reserved(u8),
48    /// 0xF — user defined, preserved verbatim.
49    UserDefined(u8),
50}
51
52impl ContentGenre {
53    /// Convert a level-1 nibble to a [`ContentGenre`].
54    ///
55    /// The input must be a 4-bit nibble value (`0..=0xF`); values outside
56    /// this range are masked with `& 0x0F`.
57    #[must_use]
58    pub fn from_nibble_1(n1: u8) -> Self {
59        let n1 = n1 & 0x0F;
60        match n1 {
61            0x0 => Self::UndefinedContent,
62            0x1 => Self::MovieDrama,
63            0x2 => Self::NewsCurrentAffairs,
64            0x3 => Self::ShowGameShow,
65            0x4 => Self::Sports,
66            0x5 => Self::ChildrenYouth,
67            0x6 => Self::MusicBalletDance,
68            0x7 => Self::ArtsCulture,
69            0x8 => Self::SocialPoliticalEconomics,
70            0x9 => Self::EducationScienceFactual,
71            0xA => Self::LeisureHobbies,
72            0xB => Self::SpecialCharacteristics,
73            0xC => Self::Adult,
74            0xD | 0xE => Self::Reserved(n1),
75            0xF => Self::UserDefined(n1),
76            _ => Self::Reserved(n1),
77        }
78    }
79
80    /// Returns the level-1 nibble for this genre (inverse of
81    /// [`ContentGenre::from_nibble_1`]).
82    #[must_use]
83    pub fn to_nibble_1(self) -> u8 {
84        self.to_u8()
85    }
86
87    /// Returns the wire byte for this value (same as the level-1 nibble).
88    #[must_use]
89    pub fn to_u8(self) -> u8 {
90        match self {
91            Self::UndefinedContent => 0x0,
92            Self::MovieDrama => 0x1,
93            Self::NewsCurrentAffairs => 0x2,
94            Self::ShowGameShow => 0x3,
95            Self::Sports => 0x4,
96            Self::ChildrenYouth => 0x5,
97            Self::MusicBalletDance => 0x6,
98            Self::ArtsCulture => 0x7,
99            Self::SocialPoliticalEconomics => 0x8,
100            Self::EducationScienceFactual => 0x9,
101            Self::LeisureHobbies => 0xA,
102            Self::SpecialCharacteristics => 0xB,
103            Self::Adult => 0xC,
104            Self::Reserved(v) => v,
105            Self::UserDefined(v) => v,
106        }
107    }
108
109    /// Returns the broad level-1 category name per EN 300 468 Table 29.
110    #[must_use]
111    pub fn name(self) -> &'static str {
112        match self {
113            Self::UndefinedContent => "undefined content",
114            Self::MovieDrama => "Movie/Drama",
115            Self::NewsCurrentAffairs => "News/Current Affairs",
116            Self::ShowGameShow => "Show/Game Show",
117            Self::Sports => "Sports",
118            Self::ChildrenYouth => "Children/Youth",
119            Self::MusicBalletDance => "Music/Ballet/Dance",
120            Self::ArtsCulture => "Arts/Culture",
121            Self::SocialPoliticalEconomics => "Social/Political/Economics",
122            Self::EducationScienceFactual => "Education/Science/Factual",
123            Self::LeisureHobbies => "Leisure hobbies",
124            Self::SpecialCharacteristics => "Special characteristics",
125            Self::Adult => "Adult",
126            Self::Reserved(_) => "reserved",
127            Self::UserDefined(_) => "user defined",
128        }
129    }
130}
131dvb_common::impl_spec_display!(ContentGenre, Reserved, UserDefined);
132
133/// Return the most specific content genre name from EN 300 468 Table 29.
134///
135/// Maps `(nibble_1, nibble_2)` to the corresponding description string.
136/// Returns `"unknown"` for unallocated combinations.
137#[must_use]
138pub fn content_genre_name(nibble_1: u8, nibble_2: u8) -> &'static str {
139    match (nibble_1, nibble_2) {
140        (0x0, 0x0..=0xF) => "undefined content",
141
142        // Movie/Drama
143        (0x1, 0x0) => "movie/drama (general)",
144        (0x1, 0x1) => "detective/thriller",
145        (0x1, 0x2) => "adventure/western/war",
146        (0x1, 0x3) => "science fiction/fantasy/horror",
147        (0x1, 0x4) => "comedy",
148        (0x1, 0x5) => "soap/melodrama/folkloric",
149        (0x1, 0x6) => "romance",
150        (0x1, 0x7) => "serious/classical/religious/historical movie/drama",
151        (0x1, 0x8) => "adult movie/drama",
152        (0x1, 0x9..=0xE) => "reserved",
153        (0x1, 0xF) => "user defined",
154
155        // News/Current Affairs
156        (0x2, 0x0) => "news/current affairs (general)",
157        (0x2, 0x1) => "news/weather report",
158        (0x2, 0x2) => "news magazine",
159        (0x2, 0x3) => "documentary",
160        (0x2, 0x4) => "discussion/interview/debate",
161        (0x2, 0x5..=0xE) => "reserved",
162        (0x2, 0xF) => "user defined",
163
164        // Show/Game Show
165        (0x3, 0x0) => "show/game show (general)",
166        (0x3, 0x1) => "game show/quiz/contest",
167        (0x3, 0x2) => "variety show",
168        (0x3, 0x3) => "talk show",
169        (0x3, 0x4..=0xE) => "reserved",
170        (0x3, 0xF) => "user defined",
171
172        // Sports
173        (0x4, 0x0) => "sports (general)",
174        (0x4, 0x1) => "special events (Olympic Games, World Cup, etc.)",
175        (0x4, 0x2) => "sports magazines",
176        (0x4, 0x3) => "football/soccer",
177        (0x4, 0x4) => "tennis/squash",
178        (0x4, 0x5) => "team sports (excluding football)",
179        (0x4, 0x6) => "athletics",
180        (0x4, 0x7) => "motor sport",
181        (0x4, 0x8) => "water sport",
182        (0x4, 0x9) => "winter sports",
183        (0x4, 0xA) => "equestrian",
184        (0x4, 0xB) => "martial sports",
185        (0x4, 0xC..=0xE) => "reserved",
186        (0x4, 0xF) => "user defined",
187
188        // Children/Youth
189        (0x5, 0x0) => "children's/youth programmes (general)",
190        (0x5, 0x1) => "pre-school children's programmes",
191        (0x5, 0x2) => "entertainment programmes for 6 to 14",
192        (0x5, 0x3) => "entertainment programmes for 10 to 16",
193        (0x5, 0x4) => "informational/educational/school programmes",
194        (0x5, 0x5) => "cartoons/puppets",
195        (0x5, 0x6..=0xE) => "reserved",
196        (0x5, 0xF) => "user defined",
197
198        // Music/Ballet/Dance
199        (0x6, 0x0) => "music/ballet/dance (general)",
200        (0x6, 0x1) => "rock/pop",
201        (0x6, 0x2) => "serious music/classical music",
202        (0x6, 0x3) => "folk/traditional music",
203        (0x6, 0x4) => "jazz",
204        (0x6, 0x5) => "musical/opera",
205        (0x6, 0x6) => "ballet",
206        (0x6, 0x7..=0xE) => "reserved",
207        (0x6, 0xF) => "user defined",
208
209        // Arts/Culture
210        (0x7, 0x0) => "arts/culture (without music, general)",
211        (0x7, 0x1) => "performing arts",
212        (0x7, 0x2) => "fine arts",
213        (0x7, 0x3) => "religion",
214        (0x7, 0x4) => "popular culture/traditional arts",
215        (0x7, 0x5) => "literature",
216        (0x7, 0x6) => "film/cinema",
217        (0x7, 0x7) => "experimental film/video",
218        (0x7, 0x8) => "broadcasting/press",
219        (0x7, 0x9) => "new media",
220        (0x7, 0xA) => "arts/culture magazines",
221        (0x7, 0xB) => "fashion",
222        (0x7, 0xC..=0xE) => "reserved",
223        (0x7, 0xF) => "user defined",
224
225        // Social/Political/Economics
226        (0x8, 0x0) => "social/political issues/economics (general)",
227        (0x8, 0x1) => "magazines/reports/documentary",
228        (0x8, 0x2) => "economics/social advisory",
229        (0x8, 0x3) => "remarkable people",
230        (0x8, 0x4..=0xE) => "reserved",
231        (0x8, 0xF) => "user defined",
232
233        // Education/Science/Factual
234        (0x9, 0x0) => "education/science/factual topics (general)",
235        (0x9, 0x1) => "nature/animals/environment",
236        (0x9, 0x2) => "technology/natural sciences",
237        (0x9, 0x3) => "medicine/physiology/psychology",
238        (0x9, 0x4) => "foreign countries/expeditions",
239        (0x9, 0x5) => "social/spiritual sciences",
240        (0x9, 0x6) => "further education",
241        (0x9, 0x7) => "languages",
242        (0x9, 0x8..=0xE) => "reserved",
243        (0x9, 0xF) => "user defined",
244
245        // Leisure hobbies
246        (0xA, 0x0) => "leisure hobbies (general)",
247        (0xA, 0x1) => "tourism/travel",
248        (0xA, 0x2) => "handicraft",
249        (0xA, 0x3) => "motoring",
250        (0xA, 0x4) => "fitness and health",
251        (0xA, 0x5) => "cooking",
252        (0xA, 0x6) => "advertisement/shopping",
253        (0xA, 0x7) => "gardening",
254        (0xA, 0x8..=0xE) => "reserved",
255        (0xA, 0xF) => "user defined",
256
257        // Special characteristics
258        (0xB, 0x0) => "original language",
259        (0xB, 0x1) => "black and white",
260        (0xB, 0x2) => "unpublished",
261        (0xB, 0x3) => "live broadcast",
262        (0xB, 0x4) => "plano-stereoscopic",
263        (0xB, 0x5) => "local or regional",
264        (0xB, 0x6..=0xE) => "reserved",
265        (0xB, 0xF) => "user defined",
266
267        // Adult
268        (0xC, 0x0) => "adult (general)",
269        (0xC, 0x1..=0xE) => "reserved",
270        (0xC, 0xF) => "user defined",
271
272        // Reserved and user-defined
273        (0xD..=0xE, 0x0..=0xF) => "reserved",
274        (0xF, 0x0..=0xF) => "user defined",
275
276        _ => "unknown",
277    }
278}
279
280/// One content classification entry.
281#[derive(Debug, Clone, Copy, PartialEq, Eq)]
282#[cfg_attr(feature = "serde", derive(serde::Serialize))]
283pub struct ContentEntry {
284    /// content_nibble_level_1 (4 bits) — broad genre (ETSI Table 29).
285    pub nibble_1: u8,
286    /// content_nibble_level_2 (4 bits) — sub-genre.
287    pub nibble_2: u8,
288    /// Broadcaster-specific user byte.
289    pub user_byte: u8,
290}
291
292impl ContentEntry {
293    /// Level-1 broad category per EN 300 468 Table 29.
294    #[must_use]
295    pub fn genre(&self) -> ContentGenre {
296        ContentGenre::from_nibble_1(self.nibble_1)
297    }
298
299    /// Most specific genre name per EN 300 468 Table 29.
300    ///
301    /// # Examples
302    /// ```
303    /// use dvb_si::descriptors::content::ContentEntry;
304    ///
305    /// let e = ContentEntry { nibble_1: 0x1, nibble_2: 0x4, user_byte: 0 };
306    /// assert_eq!(e.genre_name(), "comedy");
307    /// ```
308    #[must_use]
309    pub fn genre_name(&self) -> &'static str {
310        content_genre_name(self.nibble_1, self.nibble_2)
311    }
312}
313
314/// Content Descriptor.
315#[derive(Debug, Clone, PartialEq, Eq)]
316#[cfg_attr(feature = "serde", derive(serde::Serialize))]
317pub struct ContentDescriptor {
318    /// Entries in wire order. EIT events can carry multiple genre entries.
319    pub entries: Vec<ContentEntry>,
320}
321
322impl<'a> Parse<'a> for ContentDescriptor {
323    type Error = crate::error::Error;
324    fn parse(bytes: &'a [u8]) -> Result<Self> {
325        let body = descriptor_body(
326            bytes,
327            TAG,
328            "ContentDescriptor",
329            "unexpected tag for ContentDescriptor",
330        )?;
331
332        if body.len() % 2 != 0 {
333            return Err(Error::InvalidDescriptor {
334                tag: TAG,
335                reason: "descriptor_length must be a multiple of 2",
336            });
337        }
338
339        let mut entries = Vec::with_capacity(body.len() / ENTRY_LEN);
340
341        for chunk in body.chunks_exact(ENTRY_LEN) {
342            entries.push(ContentEntry {
343                nibble_1: chunk[0] >> 4,
344                nibble_2: chunk[0] & 0x0F,
345                user_byte: chunk[1],
346            });
347        }
348
349        Ok(Self { entries })
350    }
351}
352
353impl Serialize for ContentDescriptor {
354    type Error = crate::error::Error;
355    fn serialized_len(&self) -> usize {
356        HEADER_LEN + self.entries.len() * ENTRY_LEN
357    }
358
359    fn serialize_into(&self, buf: &mut [u8]) -> Result<usize> {
360        let len = self.serialized_len();
361        if buf.len() < len {
362            return Err(Error::OutputBufferTooSmall {
363                need: len,
364                have: buf.len(),
365            });
366        }
367        buf[0] = TAG;
368        buf[1] = (len - HEADER_LEN) as u8;
369        let mut pos = HEADER_LEN;
370        for entry in &self.entries {
371            buf[pos] = (entry.nibble_1 << 4) | entry.nibble_2;
372            buf[pos + 1] = entry.user_byte;
373            pos += ENTRY_LEN;
374        }
375        Ok(len)
376    }
377}
378impl<'a> crate::traits::DescriptorDef<'a> for ContentDescriptor {
379    const TAG: u8 = TAG;
380    const NAME: &'static str = "CONTENT";
381}
382
383#[cfg(test)]
384mod tests {
385    use super::*;
386
387    #[test]
388    fn parse_single_entry_extracts_nibbles_and_user_byte() {
389        let result = ContentDescriptor::parse(&[TAG, 2, 0x31, 0xFF]).unwrap();
390        assert_eq!(result.entries.len(), 1);
391        let e = result.entries[0];
392        assert_eq!(e.nibble_1, 3);
393        assert_eq!(e.nibble_2, 1);
394        assert_eq!(e.user_byte, 0xFF);
395    }
396
397    #[test]
398    fn parse_multiple_entries_preserves_order() {
399        let bytes = [TAG, 6, 0x31, 0xAA, 0x42, 0xBB, 0x53, 0xCC];
400        let result = ContentDescriptor::parse(&bytes).unwrap();
401        assert_eq!(result.entries.len(), 3);
402        assert_eq!(result.entries[0].nibble_1, 3);
403        assert_eq!(result.entries[0].nibble_2, 1);
404        assert_eq!(result.entries[0].user_byte, 0xAA);
405        assert_eq!(result.entries[1].nibble_1, 4);
406        assert_eq!(result.entries[1].nibble_2, 2);
407        assert_eq!(result.entries[1].user_byte, 0xBB);
408        assert_eq!(result.entries[2].nibble_1, 5);
409        assert_eq!(result.entries[2].nibble_2, 3);
410        assert_eq!(result.entries[2].user_byte, 0xCC);
411    }
412
413    #[test]
414    fn parse_rejects_wrong_tag() {
415        assert!(matches!(
416            ContentDescriptor::parse(&[0x55, 2, 0x00, 0x00]).unwrap_err(),
417            Error::InvalidDescriptor { tag: 0x55, .. }
418        ));
419    }
420
421    #[test]
422    fn parse_rejects_short_header() {
423        assert!(matches!(
424            ContentDescriptor::parse(&[TAG]).unwrap_err(),
425            Error::BufferTooShort { .. }
426        ));
427    }
428
429    #[test]
430    fn parse_rejects_body_truncation() {
431        assert!(matches!(
432            ContentDescriptor::parse(&[TAG, 4, 0x01]).unwrap_err(),
433            Error::BufferTooShort { .. }
434        ));
435    }
436
437    #[test]
438    fn parse_rejects_length_not_multiple_of_2() {
439        assert!(matches!(
440            ContentDescriptor::parse(&[TAG, 3, 0x01, 0x02, 0x03]).unwrap_err(),
441            Error::InvalidDescriptor { .. }
442        ));
443    }
444
445    #[test]
446    fn serialize_round_trip() {
447        let original = ContentDescriptor {
448            entries: vec![
449                ContentEntry {
450                    nibble_1: 3,
451                    nibble_2: 1,
452                    user_byte: 0xAA,
453                },
454                ContentEntry {
455                    nibble_1: 4,
456                    nibble_2: 2,
457                    user_byte: 0xBB,
458                },
459            ],
460        };
461        let mut buf = vec![0u8; original.serialized_len()];
462        original.serialize_into(&mut buf).unwrap();
463        let parsed = ContentDescriptor::parse(&buf).unwrap();
464        assert_eq!(parsed, original);
465    }
466
467    #[test]
468    fn empty_descriptor_valid() {
469        let bytes = [TAG, 0];
470        let result = ContentDescriptor::parse(&bytes).unwrap();
471        assert!(result.entries.is_empty());
472    }
473
474    #[test]
475    fn content_genre_nibble1_round_trip() {
476        for n1 in 0..=0xF_u8 {
477            let genre = ContentGenre::from_nibble_1(n1);
478            assert_eq!(
479                genre.to_nibble_1(),
480                n1,
481                "round-trip failed for nibble 0x{n1:02X}"
482            );
483        }
484    }
485
486    #[test]
487    fn content_genre_nibble1_masks_wide_input() {
488        assert_eq!(ContentGenre::from_nibble_1(0x21), ContentGenre::MovieDrama);
489        assert_eq!(ContentGenre::from_nibble_1(0xFF).to_nibble_1(), 0x0F);
490    }
491
492    #[test]
493    fn content_genre_name_for_all_categories() {
494        assert_eq!(ContentGenre::UndefinedContent.name(), "undefined content");
495        assert_eq!(ContentGenre::MovieDrama.name(), "Movie/Drama");
496        assert_eq!(ContentGenre::Sports.name(), "Sports");
497        assert_eq!(ContentGenre::Reserved(0xD).name(), "reserved");
498        assert_eq!(ContentGenre::UserDefined(0xF).name(), "user defined");
499    }
500}