/* * Copyright (C) 2014 Stefan Ringel * * Authors: * Stefan Ringel * * 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. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include #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 * * The list of section types defined and used by the ATSC specifications can be * seen in %GstMpegtsSectionATSCTableID. * * # Supported ATSC MPEG-TS sections * These are the sections for which parsing and packetizing code exists. * * ## Master Guide Table (MGT) * See: * * gst_mpegts_section_get_atsc_mgt() * * %GstMpegtsAtscMGT * * %GstMpegtsAtscMGTTable * * gst_mpegts_atsc_mgt_new() * * ## Terrestrial (TVCT) and Cable (CVCT) Virtual Channel Table * See: * * gst_mpegts_section_get_atsc_tvct() * * gst_mpegts_section_get_atsc_cvct() * * %GstMpegtsAtscVCT * * %GstMpegtsAtscVCTSource * * ## Rating Region Table (RRT) * See: * * gst_mpegts_section_get_atsc_rrt() * * %GstMpegtsAtscRRT * * gst_mpegts_atsc_rrt_new() * * ## Event Information Table (EIT) * See: * * gst_mpegts_section_get_atsc_eit() * * %GstMpegtsAtscEIT * * %GstMpegtsAtscEITEvent * * ## Extended Text Table (ETT) * See: * * gst_mpegts_section_get_atsc_ett() * * %GstMpegtsAtscETT * * ## System Time Table (STT) * See: * * gst_mpegts_section_get_atsc_stt() * * %GstMpegtsAtscSTT * * gst_mpegts_atsc_stt_new() * * # API */ /* 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; } static gboolean _packetize_mgt (GstMpegtsSection * section) { const GstMpegtsAtscMGT *mgt; guint8 *pos, *data; gsize length; guint i, j; mgt = gst_mpegts_section_get_atsc_mgt (section); if (mgt == NULL) return FALSE; if (mgt->tables_defined != mgt->tables->len) return FALSE; /* 8 byte common section fields * 1 byte protocol version * 2 byte tables_defined * 2 byte reserved / descriptors_length * 4 byte CRC */ length = 17; for (i = 0; i < mgt->tables->len; i++) { GstMpegtsAtscMGTTable *mgt_table = g_ptr_array_index (mgt->tables, 1); /* 2 byte table_type * 2 byte reserved / table_type_PID * 1 byte reserved / table_type_version_number * 4 byte number bytes * 2 byte reserved / table_type_descriptors_length */ length += 11; if (mgt_table->descriptors) { for (j = 0; j < mgt_table->descriptors->len; j++) { GstMpegtsDescriptor *descriptor = g_ptr_array_index (mgt_table->descriptors, j); length += descriptor->length + 2; } } } if (mgt->descriptors) { for (i = 0; i < mgt->descriptors->len; i++) { GstMpegtsDescriptor *descriptor = g_ptr_array_index (mgt->descriptors, i); length += descriptor->length + 2; } } _packetize_common_section (section, length); data = section->data + 8; /* protocol_version - 8 bit */ GST_WRITE_UINT8 (data, mgt->protocol_version); data += 1; /* tables_defined - 16 bit uimsbf */ GST_WRITE_UINT16_BE (data, mgt->tables_defined); data += 2; for (i = 0; i < mgt->tables_defined; i++) { GstMpegtsAtscMGTTable *mgt_table = g_ptr_array_index (mgt->tables, 1); /* table_type - 16 bit uimsbf */ GST_WRITE_UINT16_BE (data, mgt_table->table_type); data += 2; /* 3 bit reserved, 13 bit table_type_PID uimsbf */ GST_WRITE_UINT16_BE (data, mgt_table->pid | 0xe000); data += 2; /* 3 bit reserved, 5 bit table_type_version_number uimsbf */ GST_WRITE_UINT8 (data, mgt_table->version_number | 0xe0); data += 1; /* 4 bit reserved, 12 bit table_type_descriptor_length uimsbf */ pos = data; *data++ = 0xF0; *data++ = 0x00; _packetize_descriptor_array (mgt_table->descriptors, &data); /* Go back and update the descriptor length */ GST_WRITE_UINT16_BE (pos, (data - pos - 2) | 0xF000); } /* 4 bit reserved, 12 bit descriptor_length uimsbf */ pos = data; *data++ = 0xF0; *data++ = 0x00; _packetize_descriptor_array (mgt->descriptors, &data); /* Go back and update the descriptor length */ GST_WRITE_UINT16_BE (pos, (data - pos - 2) | 0xF000); return TRUE; } /** * gst_mpegts_section_from_atsc_mgt: * @mgt: (transfer full): a #GstMpegtsAtscMGT to create the #GstMpegtsSection from * * Returns: (transfer full): the #GstMpegtsSection * Since: 1.18 */ GstMpegtsSection * gst_mpegts_section_from_atsc_mgt (GstMpegtsAtscMGT * mgt) { GstMpegtsSection *section; g_return_val_if_fail (mgt != NULL, NULL); section = _gst_mpegts_section_init (0x1ffb, GST_MTS_TABLE_ID_ATSC_MASTER_GUIDE); section->subtable_extension = 0x0000; section->cached_parsed = (gpointer) mgt; section->packetizer = _packetize_mgt; section->destroy_parsed = (GDestroyNotify) _gst_mpegts_atsc_mgt_free; return section; } /** * 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; } /** * gst_mpegts_section_atsc_mgt_new: * * Returns: (transfer full): #GstMpegtsAtscMGT * Since: 1.18 */ GstMpegtsAtscMGT * gst_mpegts_atsc_mgt_new (void) { GstMpegtsAtscMGT *mgt; mgt = g_slice_new0 (GstMpegtsAtscMGT); mgt->tables = g_ptr_array_new_with_free_func ((GDestroyNotify) _gst_mpegts_atsc_mgt_table_free); mgt->descriptors = g_ptr_array_new_with_free_func ((GDestroyNotify) gst_mpegts_descriptor_free); return mgt; } /* 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; } gboolean gst_mpegts_atsc_string_segment_set_string (GstMpegtsAtscStringSegment * seg, gchar * string, guint8 compression_type, guint8 mode) { const gchar *to_encoding = NULL; gboolean ret = FALSE; gsize written; GError *err = NULL; unsigned long len; if (compression_type) { GST_FIXME ("Compressed strings not yet supported"); goto done; } switch (mode) { case 0x3f: to_encoding = "UTF-16BE"; break; default: break; } if (seg->cached_string) g_free (seg->cached_string); if (seg->compressed_data) g_free (seg->compressed_data); seg->cached_string = g_strdup (string); seg->compression_type = compression_type; seg->mode = mode; len = strlen (string); if (to_encoding && len) { gchar *converted = g_convert (string, len, to_encoding, "UTF-8", NULL, &written, &err); if (err) { GST_WARNING ("Failed to convert input string to codeset %s (%s)", to_encoding, err->message); g_error_free (err); goto done; } seg->compressed_data = (guint8 *) g_strndup (converted, written); seg->compressed_data_size = written; g_free (converted); } else { seg->compressed_data = (guint8 *) g_strndup (string, len); seg->compressed_data_size = len; } ret = TRUE; done: return ret; } 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; } static void _packetize_atsc_mult_string (GPtrArray * strings, guint8 ** data) { guint i; if (strings == NULL) return; /* 8 bit number_strings */ GST_WRITE_UINT8 (*data, strings->len); *data += 1; for (i = 0; i < strings->len; i++) { GstMpegtsAtscMultString *string; guint j; string = g_ptr_array_index (strings, i); /* 24 bit ISO_639_langcode */ GST_WRITE_UINT8 (*data, string->iso_639_langcode[0]); *data += 1; GST_WRITE_UINT8 (*data, string->iso_639_langcode[1]); *data += 1; GST_WRITE_UINT8 (*data, string->iso_639_langcode[2]); *data += 1; /* 8 bit number_segments */ GST_WRITE_UINT8 (*data, string->segments->len); *data += 1; for (j = 0; j < string->segments->len; j++) { GstMpegtsAtscStringSegment *seg; seg = g_ptr_array_index (string->segments, j); /* 8 bit compression_type */ GST_WRITE_UINT8 (*data, seg->compression_type); *data += 1; /* 8 bit mode */ GST_WRITE_UINT8 (*data, seg->mode); *data += 1; /* 8 bit number_bytes */ GST_WRITE_UINT8 (*data, seg->compressed_data_size); *data += 1; /* number_bytes compressed string */ memcpy (*data, seg->compressed_data, seg->compressed_data_size); *data += seg->compressed_data_size; } } } static gsize _get_atsc_mult_string_packetized_length (GPtrArray * strings) { gsize length = 1; guint i; for (i = 0; i < strings->len; i++) { GstMpegtsAtscMultString *string; guint j; string = g_ptr_array_index (strings, i); length += 4; for (j = 0; j < string->segments->len; j++) { GstMpegtsAtscStringSegment *seg; seg = g_ptr_array_index (string->segments, j); length += 3 + seg->compressed_data_size; } } return length; } /* 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; } static gboolean _packetize_stt (GstMpegtsSection * section) { const GstMpegtsAtscSTT *stt; guint8 *data; gsize length; guint i; stt = gst_mpegts_section_get_atsc_stt (section); if (stt == NULL) return FALSE; /* 8 byte common section fields * 1 byte protocol version * 4 byte system time * 1 byte GPS_UTC_offset * 2 byte daylight saving * 4 byte CRC */ length = 20; if (stt->descriptors) { for (i = 0; i < stt->descriptors->len; i++) { GstMpegtsDescriptor *descriptor = g_ptr_array_index (stt->descriptors, i); length += descriptor->length + 2; } } _packetize_common_section (section, length); data = section->data + 8; /* protocol_version - 8 bit */ GST_WRITE_UINT8 (data, stt->protocol_version); data += 1; /* system time - 32 bit uimsbf */ GST_WRITE_UINT32_BE (data, stt->system_time); data += 4; /* GPS_UTC_offset - 8 bit */ GST_WRITE_UINT8 (data, stt->gps_utc_offset); data += 1; /* daylight_saving - 16 bit uimsbf */ GST_WRITE_UINT8 (data, (stt->ds_status << 7) | 0x60 | (stt->ds_dayofmonth & 0x1f)); data += 1; GST_WRITE_UINT8 (data, stt->ds_hour); data += 1; _packetize_descriptor_array (stt->descriptors, &data); return TRUE; } /** * gst_mpegts_section_section_from_atsc_stt: * @stt: (transfer full): a #GstMpegtsAtscSTT to create the #GstMpegtsSection from * * Returns: (transfer full): the #GstMpegtsSection * Since: 1.18 */ GstMpegtsSection * gst_mpegts_section_from_atsc_stt (GstMpegtsAtscSTT * stt) { GstMpegtsSection *section; g_return_val_if_fail (stt != NULL, NULL); section = _gst_mpegts_section_init (0x1ffb, GST_MTS_TABLE_ID_ATSC_SYSTEM_TIME); section->subtable_extension = 0x0000; section->cached_parsed = (gpointer) stt; section->packetizer = _packetize_stt; section->destroy_parsed = (GDestroyNotify) _gst_mpegts_atsc_stt_free; return section; } /** * 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; } /** * gst_mpegts_section_atsc_stt_new: * * Returns: (transfer full): #GstMpegtsAtscSTT * Since: 1.18 */ GstMpegtsAtscSTT * gst_mpegts_atsc_stt_new (void) { GstMpegtsAtscSTT *stt; stt = g_slice_new0 (GstMpegtsAtscSTT); stt->descriptors = g_ptr_array_new_with_free_func ((GDestroyNotify) gst_mpegts_descriptor_free); return stt; } #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; } /* RRT */ static GstMpegtsAtscRRTDimensionValue * _gst_mpegts_atsc_rrt_dimension_value_copy (GstMpegtsAtscRRTDimensionValue * value) { GstMpegtsAtscRRTDimensionValue *copy; copy = g_slice_dup (GstMpegtsAtscRRTDimensionValue, value); copy->abbrev_ratings = g_ptr_array_ref (value->abbrev_ratings); copy->ratings = g_ptr_array_ref (value->ratings); return copy; } static void _gst_mpegts_atsc_rrt_dimension_value_free (GstMpegtsAtscRRTDimensionValue * value) { if (value->abbrev_ratings) g_ptr_array_unref (value->abbrev_ratings); if (value->ratings) g_ptr_array_unref (value->ratings); g_slice_free (GstMpegtsAtscRRTDimensionValue, value); } G_DEFINE_BOXED_TYPE (GstMpegtsAtscRRTDimensionValue, gst_mpegts_atsc_rrt_dimension_value, (GBoxedCopyFunc) _gst_mpegts_atsc_rrt_dimension_value_copy, (GFreeFunc) _gst_mpegts_atsc_rrt_dimension_value_free); static GstMpegtsAtscRRTDimension * _gst_mpegts_atsc_rrt_dimension_copy (GstMpegtsAtscRRTDimension * dim) { GstMpegtsAtscRRTDimension *copy; copy = g_slice_dup (GstMpegtsAtscRRTDimension, dim); copy->names = g_ptr_array_ref (dim->names); copy->values = g_ptr_array_ref (dim->values); return copy; } static void _gst_mpegts_atsc_rrt_dimension_free (GstMpegtsAtscRRTDimension * dim) { if (dim->names) g_ptr_array_unref (dim->names); if (dim->values) g_ptr_array_unref (dim->values); g_slice_free (GstMpegtsAtscRRTDimension, dim); } G_DEFINE_BOXED_TYPE (GstMpegtsAtscRRTDimension, gst_mpegts_atsc_rrt_dimension, (GBoxedCopyFunc) _gst_mpegts_atsc_rrt_dimension_copy, (GFreeFunc) _gst_mpegts_atsc_rrt_dimension_free); static GstMpegtsAtscRRT * _gst_mpegts_atsc_rrt_copy (GstMpegtsAtscRRT * rrt) { GstMpegtsAtscRRT *copy; copy = g_slice_dup (GstMpegtsAtscRRT, rrt); copy->names = g_ptr_array_ref (rrt->names); copy->dimensions = g_ptr_array_ref (rrt->dimensions); copy->descriptors = g_ptr_array_ref (rrt->descriptors); return copy; } static void _gst_mpegts_atsc_rrt_free (GstMpegtsAtscRRT * rrt) { if (rrt->names) g_ptr_array_unref (rrt->names); if (rrt->dimensions) g_ptr_array_unref (rrt->dimensions); if (rrt->descriptors) g_ptr_array_unref (rrt->descriptors); g_slice_free (GstMpegtsAtscRRT, rrt); } G_DEFINE_BOXED_TYPE (GstMpegtsAtscRRT, gst_mpegts_atsc_rrt, (GBoxedCopyFunc) _gst_mpegts_atsc_rrt_copy, (GFreeFunc) _gst_mpegts_atsc_rrt_free); static gpointer _parse_rrt (GstMpegtsSection * section) { GstMpegtsAtscRRT *rrt = NULL; guint i = 0; guint8 *data; guint16 descriptors_loop_length; guint8 text_length; rrt = g_slice_new0 (GstMpegtsAtscRRT); data = section->data; /* Skip already parsed data */ data += 8; rrt->protocol_version = GST_READ_UINT8 (data); data += 1; text_length = GST_READ_UINT8 (data); data += 1; rrt->names = _parse_atsc_mult_string (data, text_length); data += text_length; rrt->dimensions_defined = GST_READ_UINT8 (data); data += 1; rrt->dimensions = g_ptr_array_new_full (rrt->dimensions_defined, (GDestroyNotify) _gst_mpegts_atsc_rrt_dimension_free); for (i = 0; i < rrt->dimensions_defined; i++) { GstMpegtsAtscRRTDimension *dim; guint8 tmp; guint j = 0; dim = g_slice_new0 (GstMpegtsAtscRRTDimension); g_ptr_array_add (rrt->dimensions, dim); text_length = GST_READ_UINT8 (data); data += 1; dim->names = _parse_atsc_mult_string (data, text_length); data += text_length; tmp = GST_READ_UINT8 (data); data += 1; dim->graduated_scale = tmp & 0x10; dim->values_defined = tmp & 0x0f; dim->values = g_ptr_array_new_full (dim->values_defined, (GDestroyNotify) _gst_mpegts_atsc_rrt_dimension_value_free); for (j = 0; j < dim->values_defined; j++) { GstMpegtsAtscRRTDimensionValue *val; val = g_slice_new0 (GstMpegtsAtscRRTDimensionValue); g_ptr_array_add (dim->values, val); text_length = GST_READ_UINT8 (data); data += 1; val->abbrev_ratings = _parse_atsc_mult_string (data, text_length); data += text_length; text_length = GST_READ_UINT8 (data); data += 1; val->ratings = _parse_atsc_mult_string (data, text_length); data += text_length; } } descriptors_loop_length = GST_READ_UINT16_BE (data) & 0x3FF; data += 2; rrt->descriptors = gst_mpegts_parse_descriptors (data, descriptors_loop_length); return (gpointer) rrt; } static gboolean _packetize_rrt (GstMpegtsSection * section) { const GstMpegtsAtscRRT *rrt; guint8 *data, *pos; gsize length; gsize names_length; guint i, j; rrt = gst_mpegts_section_get_atsc_rrt (section); if (rrt == NULL) return FALSE; names_length = _get_atsc_mult_string_packetized_length (rrt->names); /* 8 byte common section fields * 1 byte protocol version * 1 byte rating_region_name_length * name_length bytes * 1 byte dimensions_defined * 2 byte reserved / descriptors_length * 4 byte CRC */ length = names_length + 17; for (i = 0; i < rrt->dimensions->len; i++) { GstMpegtsAtscRRTDimension *dim = g_ptr_array_index (rrt->dimensions, i); /* 1 byte dimension_name_length * 1 byte reserved / graduated_scale / values_defined */ length += 2; length += _get_atsc_mult_string_packetized_length (dim->names); for (j = 0; j < dim->values->len; j++) { GstMpegtsAtscRRTDimensionValue *val = g_ptr_array_index (dim->values, j); /* 1 byte abbrev_rating_value_length * 1 byte rating_value_length */ length += 2; length += _get_atsc_mult_string_packetized_length (val->abbrev_ratings); length += _get_atsc_mult_string_packetized_length (val->ratings); } } if (rrt->descriptors) { for (i = 0; i < rrt->descriptors->len; i++) { GstMpegtsDescriptor *descriptor = g_ptr_array_index (rrt->descriptors, i); length += descriptor->length + 2; } } if (length > 1024) { GST_WARNING ("RRT size can not exceed 1024"); return FALSE; } _packetize_common_section (section, length); data = section->data + 8; /* protocol_version - 8 bit */ GST_WRITE_UINT8 (data, rrt->protocol_version); data += 1; /* rating_region_name_length - 8 bit */ GST_WRITE_UINT8 (data, names_length); data += 1; _packetize_atsc_mult_string (rrt->names, &data); for (i = 0; i < rrt->dimensions->len; i++) { GstMpegtsAtscRRTDimension *dim = g_ptr_array_index (rrt->dimensions, i); /* dimension_name_length - 8 bit */ GST_WRITE_UINT8 (data, _get_atsc_mult_string_packetized_length (dim->names)); data += 1; _packetize_atsc_mult_string (rrt->names, &data); /* 3 bit reserved / 1 bit graduated_scale / 4 bit values_defined */ GST_WRITE_UINT8 (data, 0xe0 | ((dim->graduated_scale ? 1 : 0) << 4) | (dim-> values_defined & 0x0f)); data += 1; for (j = 0; j < dim->values->len; j++) { GstMpegtsAtscRRTDimensionValue *val = g_ptr_array_index (dim->values, j); /* abbrev_rating_value_length - 8 bit */ GST_WRITE_UINT8 (data, _get_atsc_mult_string_packetized_length (val->abbrev_ratings)); data += 1; _packetize_atsc_mult_string (val->abbrev_ratings, &data); /* rating_value_length - 8 bit */ GST_WRITE_UINT8 (data, _get_atsc_mult_string_packetized_length (val->ratings)); data += 1; _packetize_atsc_mult_string (val->ratings, &data); } } /* 6 bit reserved, 10 bit descriptor_length uimsbf */ pos = data; *data++ = 0xFC; *data++ = 0x00; _packetize_descriptor_array (rrt->descriptors, &data); /* Go back and update the descriptor length */ GST_WRITE_UINT16_BE (pos, (data - pos - 2) | 0xFC00); return TRUE; } /** * gst_mpegts_section_section_from_atsc_rrt: * @rrt: (transfer full): a #GstMpegtsAtscRRT to create the #GstMpegtsSection from * * Returns: (transfer full): the #GstMpegtsSection * Since: 1.18 */ GstMpegtsSection * gst_mpegts_section_from_atsc_rrt (GstMpegtsAtscRRT * rrt) { GstMpegtsSection *section; g_return_val_if_fail (rrt != NULL, NULL); section = _gst_mpegts_section_init (0x1ffb, GST_MTS_TABLE_ID_ATSC_RATING_REGION); /* FIXME random rating_region, what should be the default? */ section->subtable_extension = 0xff01; section->cached_parsed = (gpointer) rrt; section->packetizer = _packetize_rrt; section->destroy_parsed = (GDestroyNotify) _gst_mpegts_atsc_rrt_free; return section; } /** * gst_mpegts_section_get_atsc_rrt: * @section: a #GstMpegtsSection of type %GST_MPEGTS_SECTION_ATSC_RRT * * Returns the #GstMpegtsAtscRRT contained in the @section. * * Returns: The #GstMpegtsAtscRRT contained in the section, or %NULL if an error * happened. * Since: 1.18 */ const GstMpegtsAtscRRT * gst_mpegts_section_get_atsc_rrt (GstMpegtsSection * section) { g_return_val_if_fail (section->section_type == GST_MPEGTS_SECTION_ATSC_RRT, 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_rrt, (GDestroyNotify) _gst_mpegts_atsc_rrt_free); return (const GstMpegtsAtscRRT *) section->cached_parsed; } /** * gst_mpegts_section_atsc_rrt_dimension_value_new: * * Returns: (transfer full): #GstMpegtsAtscRRTDimensionValue * Since: 1.18 */ GstMpegtsAtscRRTDimensionValue * gst_mpegts_atsc_rrt_dimension_value_new (void) { GstMpegtsAtscRRTDimensionValue *val; val = g_slice_new0 (GstMpegtsAtscRRTDimensionValue); val->abbrev_ratings = g_ptr_array_new_with_free_func ((GDestroyNotify) _gst_mpegts_atsc_mult_string_free); val->ratings = g_ptr_array_new_with_free_func ((GDestroyNotify) _gst_mpegts_atsc_mult_string_free); return val; } /** * gst_mpegts_section_atsc_rrt_dimension_new: * * Returns: (transfer full): #GstMpegtsAtscRRTDimension * Since: 1.18 */ GstMpegtsAtscRRTDimension * gst_mpegts_atsc_rrt_dimension_new (void) { GstMpegtsAtscRRTDimension *dim; dim = g_slice_new0 (GstMpegtsAtscRRTDimension); dim->names = g_ptr_array_new_with_free_func ((GDestroyNotify) _gst_mpegts_atsc_mult_string_free); dim->values = g_ptr_array_new_with_free_func ((GDestroyNotify) _gst_mpegts_atsc_rrt_dimension_value_free); return dim; } /** * gst_mpegts_section_atsc_rrt_new: * * Returns: (transfer full): #GstMpegtsAtscRRT * Since: 1.18 */ GstMpegtsAtscRRT * gst_mpegts_atsc_rrt_new (void) { GstMpegtsAtscRRT *rrt; rrt = g_slice_new0 (GstMpegtsAtscRRT); rrt->names = g_ptr_array_new_with_free_func ((GDestroyNotify) _gst_mpegts_atsc_mult_string_free); rrt->dimensions = g_ptr_array_new_with_free_func ((GDestroyNotify) _gst_mpegts_atsc_rrt_dimension_free); rrt->descriptors = g_ptr_array_new_with_free_func ((GDestroyNotify) gst_mpegts_descriptor_free); return rrt; }