| /* |
| * Copyright (C) 2014 Stefan Ringel |
| * |
| * Authors: |
| * Stefan Ringel <linuxtv@stefanringel.de> |
| * |
| * This library is free software; 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 <string.h> |
| #include <stdlib.h> |
| |
| #include "mpegts.h" |
| #include "gstmpegts-private.h" |
| |
| /** |
| * SECTION:gst-atsc-section |
| * @title: ATSC variants of MPEG-TS sections |
| * @short_description: Sections for the various ATSC specifications |
| * @include: gst/mpegts/mpegts.h |
| * |
| */ |
| |
| /* Terrestrial/Cable Virtual Channel Table TVCT/CVCT */ |
| static GstMpegtsAtscVCTSource * |
| _gst_mpegts_atsc_vct_source_copy (GstMpegtsAtscVCTSource * source) |
| { |
| GstMpegtsAtscVCTSource *copy; |
| |
| copy = g_slice_dup (GstMpegtsAtscVCTSource, source); |
| copy->descriptors = g_ptr_array_ref (source->descriptors); |
| |
| return copy; |
| } |
| |
| static void |
| _gst_mpegts_atsc_vct_source_free (GstMpegtsAtscVCTSource * source) |
| { |
| g_free (source->short_name); |
| if (source->descriptors) |
| g_ptr_array_unref (source->descriptors); |
| g_slice_free (GstMpegtsAtscVCTSource, source); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsAtscVCTSource, gst_mpegts_atsc_vct_source, |
| (GBoxedCopyFunc) _gst_mpegts_atsc_vct_source_copy, |
| (GFreeFunc) _gst_mpegts_atsc_vct_source_free); |
| |
| static GstMpegtsAtscVCT * |
| _gst_mpegts_atsc_vct_copy (GstMpegtsAtscVCT * vct) |
| { |
| GstMpegtsAtscVCT *copy; |
| |
| copy = g_slice_dup (GstMpegtsAtscVCT, vct); |
| copy->sources = g_ptr_array_ref (vct->sources); |
| copy->descriptors = g_ptr_array_ref (vct->descriptors); |
| |
| return copy; |
| } |
| |
| static void |
| _gst_mpegts_atsc_vct_free (GstMpegtsAtscVCT * vct) |
| { |
| if (vct->sources) |
| g_ptr_array_unref (vct->sources); |
| if (vct->descriptors) |
| g_ptr_array_unref (vct->descriptors); |
| g_slice_free (GstMpegtsAtscVCT, vct); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsAtscVCT, gst_mpegts_atsc_vct, |
| (GBoxedCopyFunc) _gst_mpegts_atsc_vct_copy, |
| (GFreeFunc) _gst_mpegts_atsc_vct_free); |
| |
| static gpointer |
| _parse_atsc_vct (GstMpegtsSection * section) |
| { |
| GstMpegtsAtscVCT *vct = NULL; |
| guint8 *data, *end, source_nb; |
| guint32 tmp32; |
| guint16 descriptors_loop_length, tmp16; |
| guint i; |
| GError *err = NULL; |
| |
| vct = g_slice_new0 (GstMpegtsAtscVCT); |
| |
| data = section->data; |
| end = data + section->section_length; |
| |
| vct->transport_stream_id = section->subtable_extension; |
| |
| /* Skip already parsed data */ |
| data += 8; |
| |
| /* minimum size */ |
| if (end - data < 2 + 2 + 4) |
| goto error; |
| |
| vct->protocol_version = *data; |
| data += 1; |
| |
| source_nb = *data; |
| data += 1; |
| |
| vct->sources = g_ptr_array_new_full (source_nb, |
| (GDestroyNotify) _gst_mpegts_atsc_vct_source_free); |
| |
| for (i = 0; i < source_nb; i++) { |
| GstMpegtsAtscVCTSource *source; |
| |
| /* minimum 32 bytes for a entry, 2 bytes second descriptor |
| loop-length, 4 bytes crc */ |
| if (end - data < 32 + 2 + 4) |
| goto error; |
| |
| source = g_slice_new0 (GstMpegtsAtscVCTSource); |
| g_ptr_array_add (vct->sources, source); |
| |
| source->short_name = |
| g_convert ((gchar *) data, 14, "utf-8", "utf-16be", NULL, NULL, &err); |
| if (err) { |
| GST_WARNING ("Failed to convert VCT Source short_name to utf-8: %d %s", |
| err->code, err->message); |
| GST_MEMDUMP ("UTF-16 string", data, 14); |
| g_error_free (err); |
| } |
| data += 14; |
| |
| tmp32 = GST_READ_UINT32_BE (data); |
| source->major_channel_number = (tmp32 >> 18) & 0x03FF; |
| source->minor_channel_number = (tmp32 >> 8) & 0x03FF; |
| source->modulation_mode = tmp32 & 0xF; |
| data += 4; |
| |
| source->carrier_frequency = GST_READ_UINT32_BE (data); |
| data += 4; |
| |
| source->channel_TSID = GST_READ_UINT16_BE (data); |
| data += 2; |
| |
| source->program_number = GST_READ_UINT16_BE (data); |
| data += 2; |
| |
| tmp16 = GST_READ_UINT16_BE (data); |
| source->ETM_location = (tmp16 >> 14) & 0x3; |
| source->access_controlled = (tmp16 >> 13) & 0x1; |
| source->hidden = (tmp16 >> 12) & 0x1; |
| |
| /* only used in CVCT */ |
| source->path_select = (tmp16 >> 11) & 0x1; |
| source->out_of_band = (tmp16 >> 10) & 0x1; |
| |
| source->hide_guide = (tmp16 >> 9) & 0x1; |
| source->service_type = tmp16 & 0x3f; |
| data += 2; |
| |
| source->source_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| |
| descriptors_loop_length = GST_READ_UINT16_BE (data) & 0x03FF; |
| data += 2; |
| |
| if (end - data < descriptors_loop_length + 6) |
| goto error; |
| |
| source->descriptors = |
| gst_mpegts_parse_descriptors (data, descriptors_loop_length); |
| if (source->descriptors == NULL) |
| goto error; |
| data += descriptors_loop_length; |
| } |
| |
| descriptors_loop_length = GST_READ_UINT16_BE (data) & 0x03FF; |
| data += 2; |
| |
| if (end - data < descriptors_loop_length + 4) |
| goto error; |
| |
| vct->descriptors = |
| gst_mpegts_parse_descriptors (data, descriptors_loop_length); |
| if (vct->descriptors == NULL) |
| goto error; |
| |
| return (gpointer) vct; |
| |
| error: |
| _gst_mpegts_atsc_vct_free (vct); |
| |
| return NULL; |
| } |
| |
| /** |
| * gst_mpegts_section_get_atsc_tvct: |
| * @section: a #GstMpegtsSection of type %GST_MPEGTS_SECTION_ATSC_TVCT |
| * |
| * Returns the #GstMpegtsAtscVCT contained in the @section |
| * |
| * Returns: The #GstMpegtsAtscVCT contained in the section, or %NULL if an error |
| * happened. |
| */ |
| const GstMpegtsAtscVCT * |
| gst_mpegts_section_get_atsc_tvct (GstMpegtsSection * section) |
| { |
| g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_ATSC_TVCT, |
| NULL); |
| g_return_val_if_fail (section->cached_parsed || section->data, NULL); |
| |
| if (!section->cached_parsed) |
| section->cached_parsed = |
| __common_section_checks (section, 16, _parse_atsc_vct, |
| (GDestroyNotify) _gst_mpegts_atsc_vct_free); |
| |
| return (const GstMpegtsAtscVCT *) section->cached_parsed; |
| } |
| |
| /** |
| * gst_mpegts_section_get_atsc_cvct: |
| * @section: a #GstMpegtsSection of type %GST_MPEGTS_SECTION_ATSC_CVCT |
| * |
| * Returns the #GstMpegtsAtscVCT contained in the @section |
| * |
| * Returns: The #GstMpegtsAtscVCT contained in the section, or %NULL if an error |
| * happened. |
| */ |
| const GstMpegtsAtscVCT * |
| gst_mpegts_section_get_atsc_cvct (GstMpegtsSection * section) |
| { |
| g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_ATSC_CVCT, |
| NULL); |
| g_return_val_if_fail (section->cached_parsed || section->data, NULL); |
| |
| if (!section->cached_parsed) |
| section->cached_parsed = |
| __common_section_checks (section, 16, _parse_atsc_vct, |
| (GDestroyNotify) _gst_mpegts_atsc_vct_free); |
| |
| return (const GstMpegtsAtscVCT *) section->cached_parsed; |
| } |
| |
| /* MGT */ |
| |
| static GstMpegtsAtscMGTTable * |
| _gst_mpegts_atsc_mgt_table_copy (GstMpegtsAtscMGTTable * mgt_table) |
| { |
| GstMpegtsAtscMGTTable *copy; |
| |
| copy = g_slice_dup (GstMpegtsAtscMGTTable, mgt_table); |
| copy->descriptors = g_ptr_array_ref (mgt_table->descriptors); |
| |
| return copy; |
| } |
| |
| static void |
| _gst_mpegts_atsc_mgt_table_free (GstMpegtsAtscMGTTable * mgt_table) |
| { |
| g_ptr_array_unref (mgt_table->descriptors); |
| g_slice_free (GstMpegtsAtscMGTTable, mgt_table); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsAtscMGTTable, gst_mpegts_atsc_mgt_table, |
| (GBoxedCopyFunc) _gst_mpegts_atsc_mgt_table_copy, |
| (GFreeFunc) _gst_mpegts_atsc_mgt_table_free); |
| |
| static GstMpegtsAtscMGT * |
| _gst_mpegts_atsc_mgt_copy (GstMpegtsAtscMGT * mgt) |
| { |
| GstMpegtsAtscMGT *copy; |
| |
| copy = g_slice_dup (GstMpegtsAtscMGT, mgt); |
| copy->tables = g_ptr_array_ref (mgt->tables); |
| copy->descriptors = g_ptr_array_ref (mgt->descriptors); |
| |
| return copy; |
| } |
| |
| static void |
| _gst_mpegts_atsc_mgt_free (GstMpegtsAtscMGT * mgt) |
| { |
| g_ptr_array_unref (mgt->tables); |
| g_ptr_array_unref (mgt->descriptors); |
| g_slice_free (GstMpegtsAtscMGT, mgt); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsAtscMGT, gst_mpegts_atsc_mgt, |
| (GBoxedCopyFunc) _gst_mpegts_atsc_mgt_copy, |
| (GFreeFunc) _gst_mpegts_atsc_mgt_free); |
| |
| static gpointer |
| _parse_atsc_mgt (GstMpegtsSection * section) |
| { |
| GstMpegtsAtscMGT *mgt = NULL; |
| guint i = 0; |
| guint8 *data, *end; |
| guint16 descriptors_loop_length; |
| |
| mgt = g_slice_new0 (GstMpegtsAtscMGT); |
| |
| data = section->data; |
| end = data + section->section_length; |
| |
| /* Skip already parsed data */ |
| data += 8; |
| |
| mgt->protocol_version = GST_READ_UINT8 (data); |
| data += 1; |
| mgt->tables_defined = GST_READ_UINT16_BE (data); |
| data += 2; |
| mgt->tables = g_ptr_array_new_full (mgt->tables_defined, |
| (GDestroyNotify) _gst_mpegts_atsc_mgt_table_free); |
| for (i = 0; i < mgt->tables_defined && data + 11 < end; i++) { |
| GstMpegtsAtscMGTTable *mgt_table; |
| |
| if (data + 11 >= end) { |
| GST_WARNING ("MGT data too short to parse inner table num %d", i); |
| goto error; |
| } |
| |
| mgt_table = g_slice_new0 (GstMpegtsAtscMGTTable); |
| g_ptr_array_add (mgt->tables, mgt_table); |
| |
| mgt_table->table_type = GST_READ_UINT16_BE (data); |
| data += 2; |
| mgt_table->pid = GST_READ_UINT16_BE (data) & 0x1FFF; |
| data += 2; |
| mgt_table->version_number = GST_READ_UINT8 (data) & 0x1F; |
| data += 1; |
| mgt_table->number_bytes = GST_READ_UINT32_BE (data); |
| data += 4; |
| descriptors_loop_length = GST_READ_UINT16_BE (data) & 0x0FFF; |
| data += 2; |
| |
| if (data + descriptors_loop_length >= end) { |
| GST_WARNING ("MGT data too short to parse inner table descriptors (table " |
| "num %d", i); |
| goto error; |
| } |
| mgt_table->descriptors = |
| gst_mpegts_parse_descriptors (data, descriptors_loop_length); |
| data += descriptors_loop_length; |
| } |
| |
| descriptors_loop_length = GST_READ_UINT16_BE (data) & 0xFFF; |
| data += 2; |
| if (data + descriptors_loop_length >= end) { |
| GST_WARNING ("MGT data too short to parse descriptors"); |
| goto error; |
| } |
| mgt->descriptors = |
| gst_mpegts_parse_descriptors (data, descriptors_loop_length); |
| |
| return (gpointer) mgt; |
| |
| error: |
| _gst_mpegts_atsc_mgt_free (mgt); |
| |
| return NULL; |
| } |
| |
| |
| /** |
| * gst_mpegts_section_get_atsc_mgt: |
| * @section: a #GstMpegtsSection of type %GST_MPEGTS_SECTION_ATSC_MGT |
| * |
| * Returns the #GstMpegtsAtscMGT contained in the @section. |
| * |
| * Returns: The #GstMpegtsAtscMGT contained in the section, or %NULL if an error |
| * happened. |
| */ |
| const GstMpegtsAtscMGT * |
| gst_mpegts_section_get_atsc_mgt (GstMpegtsSection * section) |
| { |
| g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_ATSC_MGT, |
| NULL); |
| g_return_val_if_fail (section->cached_parsed || section->data, NULL); |
| |
| if (!section->cached_parsed) |
| section->cached_parsed = |
| __common_section_checks (section, 17, _parse_atsc_mgt, |
| (GDestroyNotify) _gst_mpegts_atsc_mgt_free); |
| |
| return (const GstMpegtsAtscMGT *) section->cached_parsed; |
| } |
| |
| /* Multi string structure */ |
| |
| static GstMpegtsAtscStringSegment * |
| _gst_mpegts_atsc_string_segment_copy (GstMpegtsAtscStringSegment * seg) |
| { |
| GstMpegtsAtscStringSegment *copy; |
| |
| copy = g_slice_dup (GstMpegtsAtscStringSegment, seg); |
| |
| return copy; |
| } |
| |
| static void |
| _gst_mpegts_atsc_string_segment_free (GstMpegtsAtscStringSegment * seg) |
| { |
| g_free (seg->cached_string); |
| g_slice_free (GstMpegtsAtscStringSegment, seg); |
| } |
| |
| static void |
| _gst_mpegts_atsc_string_segment_decode_string (GstMpegtsAtscStringSegment * seg) |
| { |
| const gchar *from_encoding; |
| |
| g_return_if_fail (seg->cached_string == NULL); |
| |
| if (seg->compression_type != 0) { |
| GST_FIXME ("Compressed strings not yet supported"); |
| return; |
| } |
| |
| /* FIXME add more encodings */ |
| switch (seg->mode) { |
| case 0x3F: |
| from_encoding = "UTF-16BE"; |
| break; |
| default: |
| from_encoding = NULL; |
| break; |
| } |
| |
| if (from_encoding != NULL && seg->compressed_data_size > 0) { |
| GError *err = NULL; |
| |
| seg->cached_string = |
| g_convert ((gchar *) seg->compressed_data, |
| (gssize) seg->compressed_data_size, "UTF-8", from_encoding, NULL, NULL, |
| &err); |
| |
| if (err) { |
| GST_WARNING ("Failed to convert input string from codeset %s", |
| from_encoding); |
| g_error_free (err); |
| } |
| } else { |
| seg->cached_string = |
| g_strndup ((gchar *) seg->compressed_data, seg->compressed_data_size); |
| } |
| } |
| |
| const gchar * |
| gst_mpegts_atsc_string_segment_get_string (GstMpegtsAtscStringSegment * seg) |
| { |
| if (!seg->cached_string) |
| _gst_mpegts_atsc_string_segment_decode_string (seg); |
| |
| return seg->cached_string; |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsAtscStringSegment, gst_mpegts_atsc_string_segment, |
| (GBoxedCopyFunc) _gst_mpegts_atsc_string_segment_copy, |
| (GFreeFunc) _gst_mpegts_atsc_string_segment_free); |
| |
| static GstMpegtsAtscMultString * |
| _gst_mpegts_atsc_mult_string_copy (GstMpegtsAtscMultString * mstring) |
| { |
| GstMpegtsAtscMultString *copy; |
| |
| copy = g_slice_dup (GstMpegtsAtscMultString, mstring); |
| copy->segments = g_ptr_array_ref (mstring->segments); |
| |
| return copy; |
| } |
| |
| static void |
| _gst_mpegts_atsc_mult_string_free (GstMpegtsAtscMultString * mstring) |
| { |
| g_ptr_array_unref (mstring->segments); |
| g_slice_free (GstMpegtsAtscMultString, mstring); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsAtscMultString, gst_mpegts_atsc_mult_string, |
| (GBoxedCopyFunc) _gst_mpegts_atsc_mult_string_copy, |
| (GFreeFunc) _gst_mpegts_atsc_mult_string_free); |
| |
| static GPtrArray * |
| _parse_atsc_mult_string (guint8 * data, guint datasize) |
| { |
| guint8 num_strings; |
| GPtrArray *res = NULL; |
| guint8 *end = data + datasize; |
| gint i; |
| |
| if (datasize > 0) { |
| /* 1 is the minimum entry size, so no need to check here */ |
| num_strings = GST_READ_UINT8 (data); |
| data += 1; |
| |
| res = |
| g_ptr_array_new_full (num_strings, |
| (GDestroyNotify) _gst_mpegts_atsc_mult_string_free); |
| |
| for (i = 0; i < num_strings; i++) { |
| GstMpegtsAtscMultString *mstring; |
| guint8 num_segments; |
| gint j; |
| |
| mstring = g_slice_new0 (GstMpegtsAtscMultString); |
| g_ptr_array_add (res, mstring); |
| mstring->segments = |
| g_ptr_array_new_full (num_strings, |
| (GDestroyNotify) _gst_mpegts_atsc_string_segment_free); |
| |
| /* each entry needs at least 4 bytes (lang code and segments number) */ |
| if (end - data < 4) { |
| GST_WARNING ("Data too short for multstring parsing %d", |
| (gint) (end - data)); |
| goto error; |
| } |
| |
| mstring->iso_639_langcode[0] = GST_READ_UINT8 (data); |
| data += 1; |
| mstring->iso_639_langcode[1] = GST_READ_UINT8 (data); |
| data += 1; |
| mstring->iso_639_langcode[2] = GST_READ_UINT8 (data); |
| data += 1; |
| num_segments = GST_READ_UINT8 (data); |
| data += 1; |
| |
| for (j = 0; j < num_segments; j++) { |
| GstMpegtsAtscStringSegment *seg; |
| |
| seg = g_slice_new0 (GstMpegtsAtscStringSegment); |
| g_ptr_array_add (mstring->segments, seg); |
| |
| /* each entry needs at least 3 bytes */ |
| if (end - data < 3) { |
| GST_WARNING ("Data too short for multstring parsing %d", datasize); |
| goto error; |
| } |
| |
| seg->compression_type = GST_READ_UINT8 (data); |
| data += 1; |
| seg->mode = GST_READ_UINT8 (data); |
| data += 1; |
| seg->compressed_data_size = GST_READ_UINT8 (data); |
| data += 1; |
| |
| if (end - data < seg->compressed_data_size) { |
| GST_WARNING ("Data too short for multstring parsing %d", datasize); |
| goto error; |
| } |
| |
| if (seg->compressed_data_size) |
| seg->compressed_data = data; |
| data += seg->compressed_data_size; |
| } |
| |
| } |
| } |
| return res; |
| |
| error: |
| if (res) |
| g_ptr_array_unref (res); |
| return NULL; |
| } |
| |
| /* EIT */ |
| |
| static GstMpegtsAtscEITEvent * |
| _gst_mpegts_atsc_eit_event_copy (GstMpegtsAtscEITEvent * event) |
| { |
| GstMpegtsAtscEITEvent *copy; |
| |
| copy = g_slice_dup (GstMpegtsAtscEITEvent, event); |
| copy->titles = g_ptr_array_ref (event->titles); |
| copy->descriptors = g_ptr_array_ref (event->descriptors); |
| |
| return copy; |
| } |
| |
| static void |
| _gst_mpegts_atsc_eit_event_free (GstMpegtsAtscEITEvent * event) |
| { |
| if (event->titles) |
| g_ptr_array_unref (event->titles); |
| if (event->descriptors) |
| g_ptr_array_unref (event->descriptors); |
| g_slice_free (GstMpegtsAtscEITEvent, event); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsAtscEITEvent, gst_mpegts_atsc_eit_event, |
| (GBoxedCopyFunc) _gst_mpegts_atsc_eit_event_copy, |
| (GFreeFunc) _gst_mpegts_atsc_eit_event_free); |
| |
| static GstMpegtsAtscEIT * |
| _gst_mpegts_atsc_eit_copy (GstMpegtsAtscEIT * eit) |
| { |
| GstMpegtsAtscEIT *copy; |
| |
| copy = g_slice_dup (GstMpegtsAtscEIT, eit); |
| copy->events = g_ptr_array_ref (eit->events); |
| |
| return copy; |
| } |
| |
| static void |
| _gst_mpegts_atsc_eit_free (GstMpegtsAtscEIT * eit) |
| { |
| if (eit->events) |
| g_ptr_array_unref (eit->events); |
| g_slice_free (GstMpegtsAtscEIT, eit); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsAtscEIT, gst_mpegts_atsc_eit, |
| (GBoxedCopyFunc) _gst_mpegts_atsc_eit_copy, |
| (GFreeFunc) _gst_mpegts_atsc_eit_free); |
| |
| static gpointer |
| _parse_atsc_eit (GstMpegtsSection * section) |
| { |
| GstMpegtsAtscEIT *eit = NULL; |
| guint i = 0; |
| guint8 *data, *end; |
| guint8 num_events; |
| |
| eit = g_slice_new0 (GstMpegtsAtscEIT); |
| |
| data = section->data; |
| end = data + section->section_length; |
| |
| eit->source_id = section->subtable_extension; |
| |
| /* Skip already parsed data */ |
| data += 8; |
| |
| eit->protocol_version = GST_READ_UINT8 (data); |
| data += 1; |
| num_events = GST_READ_UINT8 (data); |
| data += 1; |
| |
| eit->events = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_atsc_eit_event_free); |
| |
| for (i = 0; i < num_events; i++) { |
| GstMpegtsAtscEITEvent *event; |
| guint32 tmp; |
| guint8 text_length; |
| guint16 descriptors_loop_length; |
| |
| if (end - data < 12) { |
| GST_WARNING ("PID %d invalid EIT entry length %d with %u events", |
| section->pid, (gint) (end - 4 - data), num_events); |
| goto error; |
| } |
| |
| event = g_slice_new0 (GstMpegtsAtscEITEvent); |
| g_ptr_array_add (eit->events, event); |
| |
| event->event_id = GST_READ_UINT16_BE (data) & 0x3FFF; |
| data += 2; |
| event->start_time = GST_READ_UINT32_BE (data); |
| data += 4; |
| |
| tmp = GST_READ_UINT32_BE (data); |
| data += 4; |
| event->etm_location = (tmp >> 28) & 0x3; |
| event->length_in_seconds = (tmp >> 8) & 0x0FFFFF; |
| text_length = tmp & 0xFF; |
| |
| if (text_length > end - data - 4 - 2) { |
| GST_WARNING ("PID %d invalid EIT entry length %d with %u events", |
| section->pid, (gint) (end - 4 - data), num_events); |
| goto error; |
| } |
| event->titles = _parse_atsc_mult_string (data, text_length); |
| data += text_length; |
| |
| descriptors_loop_length = GST_READ_UINT16_BE (data) & 0x0FFF; |
| data += 2; |
| |
| if (end - data - 4 < descriptors_loop_length) { |
| GST_WARNING ("PID %d invalid EIT entry length %d with %u events", |
| section->pid, (gint) (end - 4 - data), num_events); |
| goto error; |
| } |
| |
| event->descriptors = |
| gst_mpegts_parse_descriptors (data, descriptors_loop_length); |
| data += descriptors_loop_length; |
| } |
| |
| 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: |
| _gst_mpegts_atsc_eit_free (eit); |
| |
| return NULL; |
| |
| } |
| |
| /** |
| * gst_mpegts_section_get_atsc_eit: |
| * @section: a #GstMpegtsSection of type %GST_MPEGTS_SECTION_ATSC_EIT |
| * |
| * Returns the #GstMpegtsAtscEIT contained in the @section. |
| * |
| * Returns: The #GstMpegtsAtscEIT contained in the section, or %NULL if an error |
| * happened. |
| */ |
| const GstMpegtsAtscEIT * |
| gst_mpegts_section_get_atsc_eit (GstMpegtsSection * section) |
| { |
| g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_ATSC_EIT, |
| NULL); |
| g_return_val_if_fail (section->cached_parsed || section->data, NULL); |
| |
| if (!section->cached_parsed) |
| section->cached_parsed = |
| __common_section_checks (section, 14, _parse_atsc_eit, |
| (GDestroyNotify) _gst_mpegts_atsc_eit_free); |
| |
| return (const GstMpegtsAtscEIT *) section->cached_parsed; |
| } |
| |
| |
| static GstMpegtsAtscETT * |
| _gst_mpegts_atsc_ett_copy (GstMpegtsAtscETT * ett) |
| { |
| GstMpegtsAtscETT *copy; |
| |
| copy = g_slice_dup (GstMpegtsAtscETT, ett); |
| copy->messages = g_ptr_array_ref (ett->messages); |
| |
| return copy; |
| } |
| |
| static void |
| _gst_mpegts_atsc_ett_free (GstMpegtsAtscETT * ett) |
| { |
| if (ett->messages) |
| g_ptr_array_unref (ett->messages); |
| g_slice_free (GstMpegtsAtscETT, ett); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsAtscETT, gst_mpegts_atsc_ett, |
| (GBoxedCopyFunc) _gst_mpegts_atsc_ett_copy, |
| (GFreeFunc) _gst_mpegts_atsc_ett_free); |
| |
| static gpointer |
| _parse_ett (GstMpegtsSection * section) |
| { |
| GstMpegtsAtscETT *ett = NULL; |
| guint8 *data, *end; |
| |
| ett = g_slice_new0 (GstMpegtsAtscETT); |
| |
| data = section->data; |
| end = data + section->section_length; |
| |
| ett->ett_table_id_extension = section->subtable_extension; |
| |
| /* Skip already parsed data */ |
| data += 8; |
| |
| ett->protocol_version = GST_READ_UINT8 (data); |
| data += 1; |
| ett->etm_id = GST_READ_UINT32_BE (data); |
| data += 4; |
| |
| ett->messages = _parse_atsc_mult_string (data, end - data - 4); |
| data += end - data - 4; |
| |
| if (data != end - 4) { |
| GST_WARNING ("PID %d invalid ETT parsed %d length %d", |
| section->pid, (gint) (data - section->data), section->section_length); |
| goto error; |
| } |
| |
| return (gpointer) ett; |
| |
| error: |
| _gst_mpegts_atsc_ett_free (ett); |
| |
| return NULL; |
| |
| } |
| |
| /** |
| * gst_mpegts_section_get_atsc_ett: |
| * @section: a #GstMpegtsSection of type %GST_MPEGTS_SECTION_ATSC_ETT |
| * |
| * Returns the #GstMpegtsAtscETT contained in the @section. |
| * |
| * Returns: The #GstMpegtsAtscETT contained in the section, or %NULL if an error |
| * happened. |
| */ |
| const GstMpegtsAtscETT * |
| gst_mpegts_section_get_atsc_ett (GstMpegtsSection * section) |
| { |
| g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_ATSC_ETT, |
| NULL); |
| g_return_val_if_fail (section->cached_parsed || section->data, NULL); |
| |
| if (!section->cached_parsed) |
| section->cached_parsed = __common_section_checks (section, 17, _parse_ett, |
| (GDestroyNotify) _gst_mpegts_atsc_ett_free); |
| |
| return (const GstMpegtsAtscETT *) section->cached_parsed; |
| } |
| |
| /* STT */ |
| |
| static GstMpegtsAtscSTT * |
| _gst_mpegts_atsc_stt_copy (GstMpegtsAtscSTT * stt) |
| { |
| GstMpegtsAtscSTT *copy; |
| |
| copy = g_slice_dup (GstMpegtsAtscSTT, stt); |
| copy->descriptors = g_ptr_array_ref (stt->descriptors); |
| |
| return copy; |
| } |
| |
| static void |
| _gst_mpegts_atsc_stt_free (GstMpegtsAtscSTT * stt) |
| { |
| if (stt->descriptors) |
| g_ptr_array_unref (stt->descriptors); |
| if (stt->utc_datetime) |
| gst_date_time_unref (stt->utc_datetime); |
| |
| g_slice_free (GstMpegtsAtscSTT, stt); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsAtscSTT, gst_mpegts_atsc_stt, |
| (GBoxedCopyFunc) _gst_mpegts_atsc_stt_copy, |
| (GFreeFunc) _gst_mpegts_atsc_stt_free); |
| |
| static gpointer |
| _parse_atsc_stt (GstMpegtsSection * section) |
| { |
| GstMpegtsAtscSTT *stt = NULL; |
| guint8 *data, *end; |
| guint16 daylight_saving; |
| |
| stt = g_slice_new0 (GstMpegtsAtscSTT); |
| |
| data = section->data; |
| end = data + section->section_length; |
| |
| /* Skip already parsed data */ |
| data += 8; |
| |
| stt->protocol_version = GST_READ_UINT8 (data); |
| data += 1; |
| stt->system_time = GST_READ_UINT32_BE (data); |
| data += 4; |
| stt->gps_utc_offset = GST_READ_UINT8 (data); |
| data += 1; |
| |
| daylight_saving = GST_READ_UINT16_BE (data); |
| data += 2; |
| stt->ds_status = daylight_saving >> 15; |
| stt->ds_dayofmonth = (daylight_saving >> 8) & 0x1F; |
| stt->ds_hour = daylight_saving & 0xFF; |
| |
| stt->descriptors = gst_mpegts_parse_descriptors (data, end - data - 4); |
| if (stt->descriptors == NULL) |
| goto error; |
| |
| return (gpointer) stt; |
| |
| error: |
| _gst_mpegts_atsc_stt_free (stt); |
| |
| return NULL; |
| } |
| |
| |
| /** |
| * gst_mpegts_section_get_atsc_stt: |
| * @section: a #GstMpegtsSection of type %GST_MPEGTS_SECTION_ATSC_STT |
| * |
| * Returns the #GstMpegtsAtscSTT contained in the @section. |
| * |
| * Returns: The #GstMpegtsAtscSTT contained in the section, or %NULL if an error |
| * happened. |
| */ |
| const GstMpegtsAtscSTT * |
| gst_mpegts_section_get_atsc_stt (GstMpegtsSection * section) |
| { |
| g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_ATSC_STT, |
| NULL); |
| g_return_val_if_fail (section->cached_parsed || section->data, NULL); |
| |
| if (!section->cached_parsed) |
| section->cached_parsed = |
| __common_section_checks (section, 20, _parse_atsc_stt, |
| (GDestroyNotify) _gst_mpegts_atsc_stt_free); |
| |
| return (const GstMpegtsAtscSTT *) section->cached_parsed; |
| } |
| |
| #define GPS_TO_UTC_TICKS G_GINT64_CONSTANT(315964800) |
| static GstDateTime * |
| _gst_mpegts_atsc_gps_time_to_datetime (guint32 systemtime, guint8 gps_offset) |
| { |
| return gst_date_time_new_from_unix_epoch_utc (systemtime - gps_offset + |
| GPS_TO_UTC_TICKS); |
| } |
| |
| GstDateTime * |
| gst_mpegts_atsc_stt_get_datetime_utc (GstMpegtsAtscSTT * stt) |
| { |
| if (stt->utc_datetime == NULL) |
| stt->utc_datetime = _gst_mpegts_atsc_gps_time_to_datetime (stt->system_time, |
| stt->gps_utc_offset); |
| |
| if (stt->utc_datetime) |
| return gst_date_time_ref (stt->utc_datetime); |
| return NULL; |
| } |