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