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