/* * gstmpegtssection.c - * Copyright (C) 2013 Edward Hervey * Copyright (C) 2011, Hewlett-Packard Development Company, L.P. * Copyright (C) 2007 Alessandro Decina * 2010 Edward Hervey * Author: Youness Alaoui , Collabora Ltd. * Author: Sebastian Dröge , Collabora Ltd. * Author: Edward Hervey , Collabora Ltd. * * Authors: * Alessandro Decina * Zaheer Abbas Merali * Edward Hervey * * This library is free softwagre; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. */ #include #include #include "mpegts.h" #include "gstmpegts-private.h" /** * SECTION:gst-dvb-section * @title: DVB variants of MPEG-TS sections * @short_description: Sections for the various DVB specifications * @include: gst/mpegts/mpegts.h * */ /* * TODO * * * Check minimum size for section parsing in the various * gst_mpegts_section_get_() methods * * * Implement parsing code for * * BAT * * CAT * * TSDT */ static inline GstDateTime * _parse_utc_time (guint8 * data) { guint year, month, day, hour, minute, second; guint16 mjd; guint8 *utc_ptr; mjd = GST_READ_UINT16_BE (data); if (mjd == G_MAXUINT16) return NULL; /* See EN 300 468 Annex C */ year = (guint32) (((mjd - 15078.2) / 365.25)); month = (guint8) ((mjd - 14956.1 - (guint) (year * 365.25)) / 30.6001); day = mjd - 14956 - (guint) (year * 365.25) - (guint) (month * 30.6001); if (month == 14 || month == 15) { year++; month = month - 1 - 12; } else { month--; } year += 1900; utc_ptr = data + 2; /* First digit of hours cannot exceeed 2 (max: 23 hours) */ hour = ((utc_ptr[0] & 0x30) >> 4) * 10 + (utc_ptr[0] & 0x0F); /* First digit of minutes cannot exced 5 (max: 59 mins) */ minute = ((utc_ptr[1] & 0x70) >> 4) * 10 + (utc_ptr[1] & 0x0F); /* first digit of seconds cannot exceed 5 (max: 59 seconds) */ second = ((utc_ptr[2] & 0x70) >> 4) * 10 + (utc_ptr[2] & 0x0F); /* Time is UTC */ return gst_date_time_new (0.0, year, month, day, hour, minute, (gdouble) second); } /* Event Information Table */ static GstMpegTsEITEvent * _gst_mpegts_eit_event_copy (GstMpegTsEITEvent * eit) { GstMpegTsEITEvent *copy; copy = g_slice_dup (GstMpegTsEITEvent, eit); copy->start_time = gst_date_time_ref (eit->start_time); copy->descriptors = g_ptr_array_ref (eit->descriptors); return copy; } static void _gst_mpegts_eit_event_free (GstMpegTsEITEvent * eit) { if (eit->start_time) gst_date_time_unref (eit->start_time); g_ptr_array_unref (eit->descriptors); g_slice_free (GstMpegTsEITEvent, eit); } G_DEFINE_BOXED_TYPE (GstMpegTsEITEvent, gst_mpegts_eit_event, (GBoxedCopyFunc) _gst_mpegts_eit_event_copy, (GFreeFunc) _gst_mpegts_eit_event_free); static GstMpegTsEIT * _gst_mpegts_eit_copy (GstMpegTsEIT * eit) { GstMpegTsEIT *copy; copy = g_slice_dup (GstMpegTsEIT, eit); copy->events = g_ptr_array_ref (eit->events); return copy; } static void _gst_mpegts_eit_free (GstMpegTsEIT * eit) { g_ptr_array_unref (eit->events); g_slice_free (GstMpegTsEIT, eit); } G_DEFINE_BOXED_TYPE (GstMpegTsEIT, gst_mpegts_eit, (GBoxedCopyFunc) _gst_mpegts_eit_copy, (GFreeFunc) _gst_mpegts_eit_free); static gpointer _parse_eit (GstMpegTsSection * section) { GstMpegTsEIT *eit = NULL; guint i = 0, allocated_events = 12; guint8 *data, *end, *duration_ptr; guint16 descriptors_loop_length; eit = g_slice_new0 (GstMpegTsEIT); data = section->data; end = data + section->section_length; /* Skip already parsed data */ data += 8; eit->transport_stream_id = GST_READ_UINT16_BE (data); data += 2; eit->original_network_id = GST_READ_UINT16_BE (data); data += 2; eit->segment_last_section_number = *data++; eit->last_table_id = *data++; eit->actual_stream = (section->table_id == 0x4E || (section->table_id >= 0x50 && section->table_id <= 0x5F)); eit->present_following = (section->table_id == 0x4E || section->table_id == 0x4F); eit->events = g_ptr_array_new_full (allocated_events, (GDestroyNotify) _gst_mpegts_eit_event_free); while (data < end - 4) { GstMpegTsEITEvent *event; /* 12 is the minimum entry size + CRC */ if (end - data < 12 + 4) { GST_WARNING ("PID %d invalid EIT entry length %d", section->pid, (gint) (end - 4 - data)); goto error; } event = g_slice_new0 (GstMpegTsEITEvent); g_ptr_array_add (eit->events, event); event->event_id = GST_READ_UINT16_BE (data); data += 2; event->start_time = _parse_utc_time (data); duration_ptr = data + 5; event->duration = (((duration_ptr[0] & 0xF0) >> 4) * 10 + (duration_ptr[0] & 0x0F)) * 60 * 60 + (((duration_ptr[1] & 0xF0) >> 4) * 10 + (duration_ptr[1] & 0x0F)) * 60 + ((duration_ptr[2] & 0xF0) >> 4) * 10 + (duration_ptr[2] & 0x0F); data += 8; event->running_status = *data >> 5; event->free_CA_mode = (*data >> 4) & 0x01; descriptors_loop_length = GST_READ_UINT16_BE (data) & 0x0FFF; data += 2; event->descriptors = gst_mpegts_parse_descriptors (data, descriptors_loop_length); if (event->descriptors == NULL) goto error; data += descriptors_loop_length; i += 1; } if (data != end - 4) { GST_WARNING ("PID %d invalid EIT parsed %d length %d", section->pid, (gint) (data - section->data), section->section_length); goto error; } return (gpointer) eit; error: if (eit) _gst_mpegts_eit_free (eit); return NULL; } /** * gst_mpegts_section_get_eit: * @section: a #GstMpegTsSection of type %GST_MPEGTS_SECTION_EIT * * Returns the #GstMpegTsEIT contained in the @section. * * Returns: The #GstMpegTsEIT contained in the section, or %NULL if an error * happened. */ const GstMpegTsEIT * gst_mpegts_section_get_eit (GstMpegTsSection * section) { g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_EIT, NULL); g_return_val_if_fail (section->cached_parsed || section->data, NULL); if (!section->cached_parsed) section->cached_parsed = __common_desc_checks (section, 18, _parse_eit, (GDestroyNotify) _gst_mpegts_eit_free); return (const GstMpegTsEIT *) section->cached_parsed; } /* Bouquet Association Table */ static GstMpegTsBATStream * _gst_mpegts_bat_stream_copy (GstMpegTsBATStream * bat) { GstMpegTsBATStream *copy; copy = g_slice_dup (GstMpegTsBATStream, bat); copy->descriptors = g_ptr_array_ref (bat->descriptors); return copy; } static void _gst_mpegts_bat_stream_free (GstMpegTsBATStream * bat) { g_ptr_array_unref (bat->descriptors); g_slice_free (GstMpegTsBATStream, bat); } G_DEFINE_BOXED_TYPE (GstMpegTsBATStream, gst_mpegts_bat_stream, (GBoxedCopyFunc) _gst_mpegts_bat_stream_copy, (GFreeFunc) _gst_mpegts_bat_stream_free); static GstMpegTsBAT * _gst_mpegts_bat_copy (GstMpegTsBAT * bat) { GstMpegTsBAT *copy; copy = g_slice_dup (GstMpegTsBAT, bat); copy->descriptors = g_ptr_array_ref (bat->descriptors); copy->streams = g_ptr_array_ref (bat->streams); return copy; } static void _gst_mpegts_bat_free (GstMpegTsBAT * bat) { g_ptr_array_unref (bat->descriptors); g_ptr_array_unref (bat->streams); g_slice_free (GstMpegTsBAT, bat); } G_DEFINE_BOXED_TYPE (GstMpegTsBAT, gst_mpegts_bat, (GBoxedCopyFunc) _gst_mpegts_bat_copy, (GFreeFunc) _gst_mpegts_bat_free); static gpointer _parse_bat (GstMpegTsSection * section) { GstMpegTsBAT *bat = NULL; guint i = 0, allocated_streams = 12; guint8 *data, *end, *entry_begin; guint16 descriptors_loop_length, transport_stream_loop_length; GST_DEBUG ("BAT"); bat = g_slice_new0 (GstMpegTsBAT); data = section->data; end = data + section->section_length; /* Skip already parsed data */ data += 8; descriptors_loop_length = GST_READ_UINT16_BE (data) & 0x0FFF; data += 2; /* see if the buffer is large enough */ if (descriptors_loop_length && (data + descriptors_loop_length > end - 4)) { GST_WARNING ("PID %d invalid BAT descriptors loop length %d", section->pid, descriptors_loop_length); goto error; } bat->descriptors = gst_mpegts_parse_descriptors (data, descriptors_loop_length); if (bat->descriptors == NULL) goto error; data += descriptors_loop_length; transport_stream_loop_length = GST_READ_UINT16_BE (data) & 0x0FFF; data += 2; if (G_UNLIKELY (transport_stream_loop_length > (end - 4 - data))) { GST_WARNING ("PID 0x%04x invalid BAT (transport_stream_loop_length too big)", section->pid); goto error; } bat->streams = g_ptr_array_new_full (allocated_streams, (GDestroyNotify) _gst_mpegts_bat_stream_free); /* read up to the CRC */ while (transport_stream_loop_length - 4 > 0) { GstMpegTsBATStream *stream = g_slice_new0 (GstMpegTsBATStream); g_ptr_array_add (bat->streams, stream); if (transport_stream_loop_length < 6) { /* each entry must be at least 6 bytes (+ 4bytes CRC) */ GST_WARNING ("PID %d invalid BAT entry size %d", section->pid, transport_stream_loop_length); goto error; } entry_begin = data; stream->transport_stream_id = GST_READ_UINT16_BE (data); data += 2; stream->original_network_id = GST_READ_UINT16_BE (data); data += 2; descriptors_loop_length = GST_READ_UINT16_BE (data) & 0x0FFF; data += 2; GST_DEBUG ("descriptors_loop_length %d", descriptors_loop_length); if (descriptors_loop_length && (data + descriptors_loop_length > end - 4)) { GST_WARNING ("PID %d invalid BAT entry %d descriptors loop length %d (only have %" G_GSIZE_FORMAT ")", section->pid, section->subtable_extension, descriptors_loop_length, end - 4 - data); goto error; } stream->descriptors = gst_mpegts_parse_descriptors (data, descriptors_loop_length); if (stream->descriptors == NULL) goto error; data += descriptors_loop_length; i += 1; transport_stream_loop_length -= data - entry_begin; } if (data != end - 4) { GST_WARNING ("PID %d invalid BAT parsed %d length %d", section->pid, (gint) (data - section->data), section->section_length); goto error; } return (gpointer) bat; error: if (bat) _gst_mpegts_bat_free (bat); return NULL; } /** * gst_mpegts_section_get_bat: * @section: a #GstMpegTsSection of type %GST_MPEGTS_SECTION_BAT * * Returns the #GstMpegTsBAT contained in the @section. * * Returns: The #GstMpegTsBAT contained in the section, or %NULL if an error * happened. */ const GstMpegTsBAT * gst_mpegts_section_get_bat (GstMpegTsSection * section) { g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_BAT, NULL); g_return_val_if_fail (section->cached_parsed || section->data, NULL); if (!section->cached_parsed) section->cached_parsed = __common_desc_checks (section, 16, _parse_bat, (GDestroyNotify) _gst_mpegts_bat_free); return (const GstMpegTsBAT *) section->cached_parsed; } /* Network Information Table */ static GstMpegTsNITStream * _gst_mpegts_nit_stream_copy (GstMpegTsNITStream * nit) { GstMpegTsNITStream *copy; copy = g_slice_dup (GstMpegTsNITStream, nit); copy->descriptors = g_ptr_array_ref (nit->descriptors); return copy; } static void _gst_mpegts_nit_stream_free (GstMpegTsNITStream * nit) { g_ptr_array_unref (nit->descriptors); g_slice_free (GstMpegTsNITStream, nit); } G_DEFINE_BOXED_TYPE (GstMpegTsNITStream, gst_mpegts_nit_stream, (GBoxedCopyFunc) _gst_mpegts_nit_stream_copy, (GFreeFunc) _gst_mpegts_nit_stream_free); static GstMpegTsNIT * _gst_mpegts_nit_copy (GstMpegTsNIT * nit) { GstMpegTsNIT *copy = g_slice_dup (GstMpegTsNIT, nit); copy->descriptors = g_ptr_array_ref (nit->descriptors); copy->streams = g_ptr_array_ref (nit->streams); return copy; } static void _gst_mpegts_nit_free (GstMpegTsNIT * nit) { g_ptr_array_unref (nit->descriptors); g_ptr_array_unref (nit->streams); g_slice_free (GstMpegTsNIT, nit); } G_DEFINE_BOXED_TYPE (GstMpegTsNIT, gst_mpegts_nit, (GBoxedCopyFunc) _gst_mpegts_nit_copy, (GFreeFunc) _gst_mpegts_nit_free); static gpointer _parse_nit (GstMpegTsSection * section) { GstMpegTsNIT *nit = NULL; guint i = 0, allocated_streams = 12; guint8 *data, *end, *entry_begin; guint16 descriptors_loop_length, transport_stream_loop_length; GST_DEBUG ("NIT"); nit = g_slice_new0 (GstMpegTsNIT); data = section->data; end = data + section->section_length; /* Set network id, and skip the rest of what is already parsed */ nit->network_id = section->subtable_extension; data += 8; nit->actual_network = section->table_id == 0x40; descriptors_loop_length = GST_READ_UINT16_BE (data) & 0x0FFF; data += 2; /* see if the buffer is large enough */ if (descriptors_loop_length && (data + descriptors_loop_length > end - 4)) { GST_WARNING ("PID %d invalid NIT descriptors loop length %d", section->pid, descriptors_loop_length); goto error; } nit->descriptors = gst_mpegts_parse_descriptors (data, descriptors_loop_length); if (nit->descriptors == NULL) goto error; data += descriptors_loop_length; transport_stream_loop_length = GST_READ_UINT16_BE (data) & 0x0FFF; data += 2; if (G_UNLIKELY (transport_stream_loop_length > (end - 4 - data))) { GST_WARNING ("PID 0x%04x invalid NIT (transport_stream_loop_length too big)", section->pid); goto error; } nit->streams = g_ptr_array_new_full (allocated_streams, (GDestroyNotify) _gst_mpegts_nit_stream_free); /* read up to the CRC */ while (transport_stream_loop_length - 4 > 0) { GstMpegTsNITStream *stream = g_slice_new0 (GstMpegTsNITStream); g_ptr_array_add (nit->streams, stream); if (transport_stream_loop_length < 6) { /* each entry must be at least 6 bytes (+ 4bytes CRC) */ GST_WARNING ("PID %d invalid NIT entry size %d", section->pid, transport_stream_loop_length); goto error; } entry_begin = data; stream->transport_stream_id = GST_READ_UINT16_BE (data); data += 2; stream->original_network_id = GST_READ_UINT16_BE (data); data += 2; descriptors_loop_length = GST_READ_UINT16_BE (data) & 0x0FFF; data += 2; GST_DEBUG ("descriptors_loop_length %d", descriptors_loop_length); if (descriptors_loop_length && (data + descriptors_loop_length > end - 4)) { GST_WARNING ("PID %d invalid NIT entry %d descriptors loop length %d (only have %" G_GSIZE_FORMAT ")", section->pid, section->subtable_extension, descriptors_loop_length, end - 4 - data); goto error; } stream->descriptors = gst_mpegts_parse_descriptors (data, descriptors_loop_length); if (stream->descriptors == NULL) goto error; data += descriptors_loop_length; i += 1; transport_stream_loop_length -= data - entry_begin; } if (data != end - 4) { GST_WARNING ("PID %d invalid NIT parsed %d length %d", section->pid, (gint) (data - section->data), section->section_length); goto error; } return (gpointer) nit; error: if (nit) _gst_mpegts_nit_free (nit); return NULL; } /** * gst_mpegts_section_get_nit: * @section: a #GstMpegTsSection of type %GST_MPEGTS_SECTION_NIT * * Returns the #GstMpegTsNIT contained in the @section. * * Returns: The #GstMpegTsNIT contained in the section, or %NULL if an error * happened. */ const GstMpegTsNIT * gst_mpegts_section_get_nit (GstMpegTsSection * section) { g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_NIT, NULL); g_return_val_if_fail (section->cached_parsed || section->data, NULL); if (!section->cached_parsed) section->cached_parsed = __common_desc_checks (section, 16, _parse_nit, (GDestroyNotify) _gst_mpegts_nit_free); return (const GstMpegTsNIT *) section->cached_parsed; } /** * gst_mpegts_nit_new: * * Allocates and initializes a #GstMpegTsNIT. * * Returns: (transfer full): A newly allocated #GstMpegTsNIT */ GstMpegTsNIT * gst_mpegts_nit_new (void) { GstMpegTsNIT *nit; nit = g_slice_new0 (GstMpegTsNIT); nit->descriptors = g_ptr_array_new_with_free_func ((GDestroyNotify) _free_descriptor); nit->streams = g_ptr_array_new_with_free_func ((GDestroyNotify) _gst_mpegts_nit_stream_free); return nit; } /** * gst_mpegts_nit_stream_new: * * Allocates and initializes a #GstMpegTsNITStream * * Returns: (transfer full): A newly allocated #GstMpegTsNITStream */ GstMpegTsNITStream * gst_mpegts_nit_stream_new (void) { GstMpegTsNITStream *stream; stream = g_slice_new0 (GstMpegTsNITStream); stream->descriptors = g_ptr_array_new_with_free_func ( (GDestroyNotify) _free_descriptor); return stream; } static gboolean _packetize_nit (GstMpegTsSection * section) { gsize length, network_length, loop_length; const GstMpegTsNIT *nit; GstMpegTsNITStream *stream; GstMpegTsDescriptor *descriptor; guint i, j; guint8 *data, *pos; nit = gst_mpegts_section_get_nit (section); if (nit == NULL) return FALSE; /* 8 byte common section fields 2 byte network_descriptors_length 2 byte transport_stream_loop_length 4 byte CRC */ length = 16; /* Find length of network descriptors */ network_length = 0; if (nit->descriptors) { for (i = 0; i < nit->descriptors->len; i++) { descriptor = g_ptr_array_index (nit->descriptors, i); network_length += descriptor->length + 2; } } /* Find length of loop */ loop_length = 0; if (nit->streams) { for (i = 0; i < nit->streams->len; i++) { stream = g_ptr_array_index (nit->streams, i); loop_length += 6; if (stream->descriptors) { for (j = 0; j < stream->descriptors->len; j++) { descriptor = g_ptr_array_index (stream->descriptors, j); loop_length += descriptor->length + 2; } } } } length += network_length + loop_length; /* Max length of NIT section is 1024 bytes */ g_return_val_if_fail (length <= 1024, FALSE); _packetize_common_section (section, length); data = section->data + 8; /* reserved - 4 bit network_descriptors_length - 12 bit uimsbf */ GST_WRITE_UINT16_BE (data, network_length | 0xF000); data += 2; _packetize_descriptor_array (nit->descriptors, &data); /* reserved - 4 bit transport_stream_loop_length - 12 bit uimsbf */ GST_WRITE_UINT16_BE (data, loop_length | 0xF000); data += 2; if (nit->streams) { for (i = 0; i < nit->streams->len; i++) { stream = g_ptr_array_index (nit->streams, i); /* transport_stream_id - 16 bit uimsbf */ GST_WRITE_UINT16_BE (data, stream->transport_stream_id); data += 2; /* original_network_id - 16 bit uimsbf */ GST_WRITE_UINT16_BE (data, stream->original_network_id); data += 2; /* reserved - 4 bit transport_descriptors_length - 12 bit uimsbf Set length to zero, and update in loop */ pos = data; *data++ = 0xF0; *data++ = 0x00; _packetize_descriptor_array (stream->descriptors, &data); /* Go back and update the descriptor length */ GST_WRITE_UINT16_BE (pos, (data - pos - 2) | 0xF000); } } return TRUE; } /** * gst_mpegts_section_from_nit: * @nit: (transfer full): a #GstMpegTsNIT to create the #GstMpegTsSection from * * Ownership of @nit is taken. The data in @nit is managed by the #GstMpegTsSection * * Returns: (transfer full): the #GstMpegTsSection */ GstMpegTsSection * gst_mpegts_section_from_nit (GstMpegTsNIT * nit) { GstMpegTsSection *section; g_return_val_if_fail (nit != NULL, NULL); if (nit->actual_network) section = _gst_mpegts_section_init (0x10, GST_MTS_TABLE_ID_NETWORK_INFORMATION_ACTUAL_NETWORK); else section = _gst_mpegts_section_init (0x10, GST_MTS_TABLE_ID_NETWORK_INFORMATION_OTHER_NETWORK); section->subtable_extension = nit->network_id; section->cached_parsed = (gpointer) nit; section->packetizer = _packetize_nit; section->destroy_parsed = (GDestroyNotify) _gst_mpegts_nit_free; return section; } /* Service Description Table (SDT) */ static GstMpegTsSDTService * _gst_mpegts_sdt_service_copy (GstMpegTsSDTService * sdt) { GstMpegTsSDTService *copy = g_slice_dup (GstMpegTsSDTService, sdt); copy->descriptors = g_ptr_array_ref (sdt->descriptors); return copy; } static void _gst_mpegts_sdt_service_free (GstMpegTsSDTService * sdt) { g_ptr_array_unref (sdt->descriptors); g_slice_free (GstMpegTsSDTService, sdt); } G_DEFINE_BOXED_TYPE (GstMpegTsSDTService, gst_mpegts_sdt_service, (GBoxedCopyFunc) _gst_mpegts_sdt_service_copy, (GFreeFunc) _gst_mpegts_sdt_service_free); static GstMpegTsSDT * _gst_mpegts_sdt_copy (GstMpegTsSDT * sdt) { GstMpegTsSDT *copy = g_slice_dup (GstMpegTsSDT, sdt); copy->services = g_ptr_array_ref (sdt->services); return copy; } static void _gst_mpegts_sdt_free (GstMpegTsSDT * sdt) { g_ptr_array_unref (sdt->services); g_slice_free (GstMpegTsSDT, sdt); } G_DEFINE_BOXED_TYPE (GstMpegTsSDT, gst_mpegts_sdt, (GBoxedCopyFunc) _gst_mpegts_sdt_copy, (GFreeFunc) _gst_mpegts_sdt_free); static gpointer _parse_sdt (GstMpegTsSection * section) { GstMpegTsSDT *sdt = NULL; guint i = 0, allocated_services = 8; guint8 *data, *end, *entry_begin; guint tmp; guint sdt_info_length; guint descriptors_loop_length; GST_DEBUG ("SDT"); sdt = g_slice_new0 (GstMpegTsSDT); data = section->data; end = data + section->section_length; /* Skip common fields */ data += 8; sdt->original_network_id = GST_READ_UINT16_BE (data); data += 2; /* skip reserved byte */ data += 1; sdt->actual_ts = section->table_id == 0x42; sdt_info_length = section->section_length - 11; sdt->services = g_ptr_array_new_full (allocated_services, (GDestroyNotify) _gst_mpegts_sdt_service_free); /* read up to the CRC */ while (sdt_info_length - 4 > 0) { GstMpegTsSDTService *service = g_slice_new0 (GstMpegTsSDTService); g_ptr_array_add (sdt->services, service); entry_begin = data; if (sdt_info_length + 5 < 4) { /* each entry must be at least 5 bytes (+4 bytes for the CRC) */ GST_WARNING ("PID %d invalid SDT entry size %d", section->pid, sdt_info_length); goto error; } service->service_id = GST_READ_UINT16_BE (data); data += 2; service->EIT_schedule_flag = ((*data & 0x02) == 2); service->EIT_present_following_flag = (*data & 0x01) == 1; data += 1; tmp = GST_READ_UINT16_BE (data); service->running_status = (*data >> 5) & 0x07; service->free_CA_mode = (*data >> 4) & 0x01; descriptors_loop_length = tmp & 0x0FFF; data += 2; if (descriptors_loop_length && (data + descriptors_loop_length > end - 4)) { GST_WARNING ("PID %d invalid SDT entry %d descriptors loop length %d", section->pid, service->service_id, descriptors_loop_length); goto error; } service->descriptors = gst_mpegts_parse_descriptors (data, descriptors_loop_length); if (!service->descriptors) goto error; data += descriptors_loop_length; sdt_info_length -= data - entry_begin; i += 1; } if (data != end - 4) { GST_WARNING ("PID %d invalid SDT parsed %d length %d", section->pid, (gint) (data - section->data), section->section_length); goto error; } return sdt; error: if (sdt) _gst_mpegts_sdt_free (sdt); return NULL; } /** * gst_mpegts_section_get_sdt: * @section: a #GstMpegTsSection of type %GST_MPEGTS_SECTION_SDT * * Returns the #GstMpegTsSDT contained in the @section. * * Returns: The #GstMpegTsSDT contained in the section, or %NULL if an error * happened. */ const GstMpegTsSDT * gst_mpegts_section_get_sdt (GstMpegTsSection * section) { g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_SDT, NULL); g_return_val_if_fail (section->cached_parsed || section->data, NULL); if (!section->cached_parsed) section->cached_parsed = __common_desc_checks (section, 15, _parse_sdt, (GDestroyNotify) _gst_mpegts_sdt_free); return (const GstMpegTsSDT *) section->cached_parsed; } /* Time and Date Table (TDT) */ static gpointer _parse_tdt (GstMpegTsSection * section) { return (gpointer) _parse_utc_time (section->data + 3); } /** * gst_mpegts_section_get_tdt: * @section: a #GstMpegTsSection of type %GST_MPEGTS_SECTION_TDT * * Returns the #GstDateTime of the TDT * * Returns: The #GstDateTime contained in the section, or %NULL * if an error happened. Release with #gst_date_time_unref when done. */ GstDateTime * gst_mpegts_section_get_tdt (GstMpegTsSection * section) { g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_TDT, NULL); g_return_val_if_fail (section->cached_parsed || section->data, NULL); if (!section->cached_parsed) section->cached_parsed = __common_desc_checks (section, 8, _parse_tdt, (GDestroyNotify) gst_date_time_unref); if (section->cached_parsed) return gst_date_time_ref ((GstDateTime *) section->cached_parsed); return NULL; } /* Time Offset Table (TOT) */ static GstMpegTsTOT * _gst_mpegts_tot_copy (GstMpegTsTOT * tot) { GstMpegTsTOT *copy = g_slice_dup (GstMpegTsTOT, tot); if (tot->utc_time) copy->utc_time = gst_date_time_ref (tot->utc_time); copy->descriptors = g_ptr_array_ref (tot->descriptors); return copy; } static void _gst_mpegts_tot_free (GstMpegTsTOT * tot) { if (tot->utc_time) gst_date_time_unref (tot->utc_time); g_ptr_array_unref (tot->descriptors); g_slice_free (GstMpegTsTOT, tot); } G_DEFINE_BOXED_TYPE (GstMpegTsTOT, gst_mpegts_tot, (GBoxedCopyFunc) _gst_mpegts_tot_copy, (GFreeFunc) _gst_mpegts_tot_free); static gpointer _parse_tot (GstMpegTsSection * section) { guint8 *data; GstMpegTsTOT *tot; guint16 desc_len; GST_DEBUG ("TOT"); tot = g_slice_new0 (GstMpegTsTOT); tot->utc_time = _parse_utc_time (section->data + 3); /* Skip 5 bytes from utc_time (+3 of initial offset) */ data = section->data + 8; desc_len = GST_READ_UINT16_BE (data) & 0xFFF; data += 2; tot->descriptors = gst_mpegts_parse_descriptors (data, desc_len); return (gpointer) tot; } /** * gst_mpegts_section_get_tot: * @section: a #GstMpegTsSection of type %GST_MPEGTS_SECTION_TOT * * Returns the #GstMpegTsTOT contained in the @section. * * Returns: The #GstMpegTsTOT contained in the section, or %NULL if an error * happened. */ const GstMpegTsTOT * gst_mpegts_section_get_tot (GstMpegTsSection * section) { g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_TOT, NULL); g_return_val_if_fail (section->cached_parsed || section->data, NULL); if (!section->cached_parsed) section->cached_parsed = __common_desc_checks (section, 14, _parse_tot, (GDestroyNotify) _gst_mpegts_tot_free); return (const GstMpegTsTOT *) section->cached_parsed; }