| /* |
| * gstmpegtsdescriptor.c - |
| * Copyright (C) 2013 Edward Hervey |
| * |
| * Authors: |
| * Edward Hervey <edward@collabora.com> |
| * |
| * 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 <stdlib.h> |
| #include <string.h> |
| |
| #include "mpegts.h" |
| #include "gstmpegts-private.h" |
| |
| |
| /** |
| * SECTION:gst-dvb-descriptor |
| * @title: DVB variants of MPEG-TS descriptors |
| * @short_description: Descriptors for the various DVB specifications |
| * @include: gst/mpegts/mpegts.h |
| * |
| */ |
| |
| /* |
| * TODO |
| * |
| * * Add common validation code for data presence and minimum/maximum expected |
| * size. |
| * * Add parsing methods for the following descriptors that were previously |
| * handled in mpegtsbase: |
| * * GST_MTS_DESC_DVB_DATA_BROADCAST_ID |
| * * GST_MTS_DESC_DVB_CAROUSEL_IDENTIFIER |
| */ |
| |
| #define BCD_UN(a) ((a) & 0x0f) |
| #define BCD_DEC(a) (((a) >> 4) & 0x0f) |
| #define BCD(a) (BCD_UN(a) + 10 * BCD_DEC(a)) |
| #define BCD_16(a) (BCD(a[1]) + 100 * BCD(a[0])) |
| #define BCD_28(a) (BCD_DEC(a[3]) + 10 * BCD(a[2]) + 1000 * BCD(a[1]) + 100000 * BCD(a[0])) |
| #define BCD_32(a) (BCD(a[3]) + 100 * BCD(a[2]) + 10000 * BCD(a[1]) + 1000000 * BCD(a[0])) |
| |
| /* GST_MTS_DESC_DVB_NETWORK_NAME (0x40) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_network_name: |
| * @descriptor: a %GST_MTS_DESC_DVB_NETWORK_NAME #GstMpegtsDescriptor |
| * @name: (out) (transfer full): the extracted name |
| * |
| * Parses out the dvb network name from the @descriptor: |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_network_name (const GstMpegtsDescriptor * |
| descriptor, gchar ** name) |
| { |
| g_return_val_if_fail (descriptor != NULL && name != NULL, FALSE); |
| /* We need at least one byte of data for the string */ |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_NETWORK_NAME, 1, FALSE); |
| |
| *name = get_encoding_and_convert ((gchar *) descriptor->data + 2, |
| descriptor->data[1]); |
| return TRUE; |
| } |
| |
| /** |
| * gst_mpegts_descriptor_from_dvb_network_name: |
| * @name: the network name to set |
| * |
| * Creates a #GstMpegtsDescriptor to be a %GST_MTS_DESC_DVB_NETWORK_NAME, |
| * with the network name @name. The data field of the #GstMpegtsDescriptor |
| * will be allocated, and transferred to the caller. |
| * |
| * Returns: (transfer full): the #GstMpegtsDescriptor or %NULL on fail |
| */ |
| GstMpegtsDescriptor * |
| gst_mpegts_descriptor_from_dvb_network_name (const gchar * name) |
| { |
| GstMpegtsDescriptor *descriptor; |
| guint8 *converted_name; |
| gsize size; |
| |
| g_return_val_if_fail (name != NULL, NULL); |
| |
| converted_name = dvb_text_from_utf8 (name, &size); |
| |
| if (size >= 256) { |
| g_free (converted_name); |
| return NULL; |
| } |
| |
| if (!converted_name) { |
| GST_WARNING ("Could not find proper encoding for string `%s`", name); |
| return NULL; |
| } |
| |
| descriptor = _new_descriptor (GST_MTS_DESC_DVB_NETWORK_NAME, size); |
| memcpy (descriptor->data + 2, converted_name, size); |
| g_free (converted_name); |
| |
| return descriptor; |
| } |
| |
| /* GST_MTS_DESC_DVB_SERVICE_LIST (0x41) */ |
| static void |
| _gst_mpegts_dvb_service_list_item_free (GstMpegtsDVBServiceListItem * item) |
| { |
| g_slice_free (GstMpegtsDVBServiceListItem, item); |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_service_list: |
| * @descriptor: a %GST_MTS_DESC_DVB_SERVICE_LIST #GstMpegtsDescriptor |
| * @list: (out) (transfer full) (element-type GstMpegtsDVBServiceListItem): |
| * the list of services |
| * |
| * Parses out a list of services from the @descriptor: |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_service_list (const GstMpegtsDescriptor * |
| descriptor, GPtrArray ** list) |
| { |
| guint8 *data, i; |
| |
| g_return_val_if_fail (descriptor != NULL && list != NULL, FALSE); |
| /* a entry has 3 bytes, 2 for service id, 1 for service type */ |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_SERVICE_LIST, 3, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *list = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_dvb_service_list_item_free); |
| |
| for (i = 0; i < descriptor->length - 2; i += 3) { |
| GstMpegtsDVBServiceListItem *item = |
| g_slice_new0 (GstMpegtsDVBServiceListItem); |
| |
| g_ptr_array_add (*list, item); |
| item->service_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| |
| item->type = *data; |
| data += 1; |
| } |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_STUFFING (0x42) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_stuffing: |
| * @descriptor: a %GST_MTS_DESC_DVB_STUFFING #GstMpegtsDescriptor |
| * @stuffing_bytes: (out) (transfer full): the stuffing bytes |
| * |
| * Parses out the stuffing bytes from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_stuffing (const GstMpegtsDescriptor * |
| descriptor, guint8 ** stuffing_bytes) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL && stuffing_bytes != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_STUFFING, 0, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *stuffing_bytes = g_memdup (data, descriptor->length); |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_SATELLITE_DELIVERY_SYSTEM (0x43) */ |
| /** |
| * gst_mpegts_descriptor_parse_satellite_delivery_system: |
| * @descriptor: a %GST_MTS_DESC_DVB_SATELLITE_DELIVERY_SYSTEM #GstMpegtsDescriptor |
| * @res: (out) (transfer none): the #GstMpegtsSatelliteDeliverySystemDescriptor to fill |
| * |
| * Extracts the satellite delivery system information from @descriptor. |
| * |
| * Returns: %TRUE if parsing succeeded, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_satellite_delivery_system (const GstMpegtsDescriptor |
| * descriptor, GstMpegtsSatelliteDeliverySystemDescriptor * res) |
| { |
| guint8 *data; |
| guint8 tmp; |
| |
| g_return_val_if_fail (descriptor != NULL && res != NULL, FALSE); |
| /* This descriptor is always 11 bytes long */ |
| __common_desc_checks_exact (descriptor, |
| GST_MTS_DESC_DVB_SATELLITE_DELIVERY_SYSTEM, 11, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| /* BCD coded frequency in GHz (decimal point occurs after the 3rd character) |
| * So direct BCD gives us units of (GHz / 100 000) = 10 kHz*/ |
| res->frequency = BCD_32 (data) * 10; |
| data += 4; |
| /* BCD codec position in degrees (float pointer after the 3rd character) */ |
| res->orbital_position = (BCD_16 (data)) / 10.0; |
| data += 2; |
| |
| tmp = *data; |
| res->west_east = (tmp & 0x80) == 0x80; |
| res->polarization = (tmp >> 7) & 0x03; |
| res->modulation_system = (tmp & 0x04) == 0x04; |
| if (res->modulation_system) |
| res->roll_off = (tmp >> 3 & 0x03); |
| else |
| res->roll_off = GST_MPEGTS_ROLLOFF_AUTO; |
| switch (tmp & 0x03) { |
| case 0x00: |
| res->modulation_type = GST_MPEGTS_MODULATION_QAM_AUTO; |
| break; |
| case 0x01: |
| res->modulation_type = GST_MPEGTS_MODULATION_QPSK; |
| break; |
| case 0x02: |
| res->modulation_type = GST_MPEGTS_MODULATION_PSK_8; |
| break; |
| case 0x03: |
| res->modulation_type = GST_MPEGTS_MODULATION_QAM_16; |
| break; |
| default: |
| res->modulation_type = GST_MPEGTS_MODULATION_QAM_AUTO; |
| break; |
| } |
| data += 1; |
| /* symbol_rate is in Msymbols/ (decimal point occurs after 3rd character) */ |
| /* So direct BCD gives us units of (Msymbol / 10 000) = 100 sym/s */ |
| res->symbol_rate = BCD_28 (data) * 100; |
| data += 3; |
| /* fec_inner */ |
| switch (*data >> 4) { |
| case 0x01: |
| res->fec_inner = GST_MPEGTS_FEC_1_2; |
| break; |
| case 0x02: |
| res->fec_inner = GST_MPEGTS_FEC_2_3; |
| break; |
| case 0x03: |
| res->fec_inner = GST_MPEGTS_FEC_3_4; |
| break; |
| case 0x04: |
| res->fec_inner = GST_MPEGTS_FEC_5_6; |
| break; |
| case 0x05: |
| res->fec_inner = GST_MPEGTS_FEC_7_8; |
| break; |
| case 0x06: |
| res->fec_inner = GST_MPEGTS_FEC_8_9; |
| break; |
| case 0x07: |
| res->fec_inner = GST_MPEGTS_FEC_3_5; |
| break; |
| case 0x08: |
| res->fec_inner = GST_MPEGTS_FEC_4_5; |
| break; |
| case 0x09: |
| res->fec_inner = GST_MPEGTS_FEC_9_10; |
| break; |
| case 0x0f: |
| res->fec_inner = GST_MPEGTS_FEC_NONE; |
| break; |
| default: |
| res->fec_inner = GST_MPEGTS_FEC_AUTO; |
| break; |
| } |
| |
| return TRUE; |
| } |
| |
| |
| /* GST_MTS_DESC_DVB_CABLE_DELIVERY_SYSTEM (0x44) */ |
| /** |
| * gst_mpegts_descriptor_parse_cable_delivery_system: |
| * @descriptor: a %GST_MTS_DESC_DVB_CABLE_DELIVERY_SYSTEM #GstMpegtsDescriptor |
| * @res: (out) (transfer none): the #GstMpegtsCableDeliverySystemDescriptor to fill |
| * |
| * Extracts the cable delivery system information from @descriptor. |
| * |
| * Returns: %TRUE if parsing succeeded, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_cable_delivery_system (const GstMpegtsDescriptor * |
| descriptor, GstMpegtsCableDeliverySystemDescriptor * res) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL && res != NULL, FALSE); |
| /* This descriptor is always 11 bytes long */ |
| __common_desc_checks_exact (descriptor, |
| GST_MTS_DESC_DVB_CABLE_DELIVERY_SYSTEM, 11, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| /* BCD in MHz, decimal place after the fourth character */ |
| res->frequency = BCD_32 (data) * 100; |
| data += 5; |
| /* fec_out (4bits) */ |
| res->outer_fec = *data++ & 0x0f; |
| switch (*data) { |
| case 0x00: |
| res->modulation = GST_MPEGTS_MODULATION_NONE; |
| break; |
| case 0x01: |
| res->modulation = GST_MPEGTS_MODULATION_QAM_16; |
| break; |
| case 0x02: |
| res->modulation = GST_MPEGTS_MODULATION_QAM_32; |
| break; |
| case 0x03: |
| res->modulation = GST_MPEGTS_MODULATION_QAM_64; |
| break; |
| case 0x04: |
| res->modulation = GST_MPEGTS_MODULATION_QAM_128; |
| break; |
| case 0x05: |
| res->modulation = GST_MPEGTS_MODULATION_QAM_256; |
| break; |
| default: |
| GST_WARNING ("Unsupported cable modulation type: 0x%02x", *data); |
| res->modulation = GST_MPEGTS_MODULATION_NONE; |
| break; |
| } |
| |
| data += 1; |
| /* symbol_rate is in Msymbols/ (decimal point occurs after 3rd character) */ |
| /* So direct BCD gives us units of (Msymbol / 10 000) = 100 sym/s */ |
| res->symbol_rate = BCD_28 (data) * 100; |
| data += 3; |
| /* fec_inner */ |
| switch (*data & 0xf) { |
| case 0x00: |
| res->fec_inner = GST_MPEGTS_FEC_AUTO; |
| break; |
| case 0x01: |
| res->fec_inner = GST_MPEGTS_FEC_1_2; |
| break; |
| case 0x02: |
| res->fec_inner = GST_MPEGTS_FEC_2_3; |
| break; |
| case 0x03: |
| res->fec_inner = GST_MPEGTS_FEC_3_4; |
| break; |
| case 0x04: |
| res->fec_inner = GST_MPEGTS_FEC_5_6; |
| break; |
| case 0x05: |
| res->fec_inner = GST_MPEGTS_FEC_7_8; |
| break; |
| case 0x06: |
| res->fec_inner = GST_MPEGTS_FEC_8_9; |
| break; |
| case 0x07: |
| res->fec_inner = GST_MPEGTS_FEC_3_5; |
| break; |
| case 0x08: |
| res->fec_inner = GST_MPEGTS_FEC_4_5; |
| break; |
| case 0x09: |
| res->fec_inner = GST_MPEGTS_FEC_9_10; |
| break; |
| case 0x0f: |
| res->fec_inner = GST_MPEGTS_FEC_NONE; |
| break; |
| default: |
| res->fec_inner = GST_MPEGTS_FEC_AUTO; |
| break; |
| } |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_BOUQUET_NAME (0x47) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_bouquet_name: |
| * @bouquet_name: (out) (transfer full) (allow-none): the bouquet name |
| * |
| * Extracts the bouquet name from @descriptor. |
| * |
| * Returns: %TRUE if parsing succeeded, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_bouquet_name (const GstMpegtsDescriptor * |
| descriptor, gchar ** bouquet_name) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL && bouquet_name != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_BOUQUET_NAME, 1, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *bouquet_name = |
| get_encoding_and_convert ((const gchar *) data, descriptor->length); |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_SERVICE (0x48) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_service: |
| * @descriptor: a %GST_MTS_DESC_DVB_SERVICE #GstMpegtsDescriptor |
| * @service_type: (out) (allow-none): the service type |
| * @service_name: (out) (transfer full) (allow-none): the service name |
| * @provider_name: (out) (transfer full) (allow-none): the provider name |
| * |
| * Extracts the dvb service information from @descriptor. |
| * |
| * Returns: %TRUE if parsing succeeded, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_service (const GstMpegtsDescriptor * |
| descriptor, GstMpegtsDVBServiceType * service_type, gchar ** service_name, |
| gchar ** provider_name) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL, FALSE); |
| /* Need at least 3 bytes (type and 2 bytes for the string length) */ |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_SERVICE, 3, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| if (service_type) |
| *service_type = *data; |
| data += 1; |
| if (provider_name) |
| *provider_name = get_encoding_and_convert ((const gchar *) data + 1, *data); |
| data += *data + 1; |
| if (service_name) |
| *service_name = get_encoding_and_convert ((const gchar *) data + 1, *data); |
| |
| return TRUE; |
| } |
| |
| /** |
| * gst_mpegts_descriptor_from_dvb_service: |
| * @service_type: Service type defined as a #GstMpegtsDVBServiceType |
| * @service_name: (allow-none): Name of the service |
| * @service_provider: (allow-none): Name of the service provider |
| * |
| * Fills a #GstMpegtsDescriptor to be a %GST_MTS_DESC_DVB_SERVICE. |
| * The data field of the #GstMpegtsDescriptor will be allocated, |
| * and transferred to the caller. |
| * |
| * Returns: (transfer full): the #GstMpegtsDescriptor or %NULL on fail |
| */ |
| GstMpegtsDescriptor * |
| gst_mpegts_descriptor_from_dvb_service (GstMpegtsDVBServiceType service_type, |
| const gchar * service_name, const gchar * service_provider) |
| { |
| GstMpegtsDescriptor *descriptor = NULL; |
| guint8 *conv_provider_name = NULL, *conv_service_name = NULL; |
| gsize provider_size = 0, service_size = 0; |
| guint8 *data; |
| |
| if (service_provider) { |
| conv_provider_name = dvb_text_from_utf8 (service_provider, &provider_size); |
| |
| if (!conv_provider_name) { |
| GST_WARNING ("Could not find proper encoding for string `%s`", |
| service_provider); |
| goto beach; |
| } |
| } |
| |
| if (provider_size >= 256) { |
| GST_WARNING ("Service provider string too big (%" G_GSIZE_FORMAT " > 256)", |
| provider_size); |
| goto beach; |
| } |
| |
| if (service_name) { |
| conv_service_name = dvb_text_from_utf8 (service_name, &service_size); |
| |
| if (!conv_service_name) { |
| GST_WARNING ("Could not find proper encoding for string `%s`", |
| service_name); |
| goto beach; |
| } |
| } |
| |
| if (service_size >= 256) { |
| GST_WARNING ("Service name string too big (%" G_GSIZE_FORMAT " > 256)", |
| service_size); |
| goto beach; |
| } |
| |
| descriptor = |
| _new_descriptor (GST_MTS_DESC_DVB_SERVICE, |
| 3 + provider_size + service_size); |
| |
| data = descriptor->data + 2; |
| *data++ = service_type; |
| *data++ = provider_size; |
| if (conv_provider_name) |
| memcpy (data, conv_provider_name, provider_size); |
| |
| data += provider_size; |
| *data++ = service_size; |
| if (conv_service_name) |
| memcpy (data, conv_service_name, service_size); |
| |
| beach: |
| g_free (conv_service_name); |
| g_free (conv_provider_name); |
| |
| return descriptor; |
| } |
| |
| /* GST_MTS_DESC_DVB_LINKAGE (0x4A) */ |
| static GstMpegtsDVBLinkageDescriptor * |
| _gst_mpegts_dvb_linkage_descriptor_copy (GstMpegtsDVBLinkageDescriptor * source) |
| { |
| GstMpegtsDVBLinkageDescriptor *copy; |
| |
| copy = g_slice_dup (GstMpegtsDVBLinkageDescriptor, source); |
| |
| switch (source->linkage_type) { |
| case GST_MPEGTS_DVB_LINKAGE_MOBILE_HAND_OVER: |
| copy->linkage_data = g_slice_dup (GstMpegtsDVBLinkageMobileHandOver, |
| source->linkage_data); |
| break; |
| case GST_MPEGTS_DVB_LINKAGE_EVENT: |
| copy->linkage_data = g_slice_dup (GstMpegtsDVBLinkageEvent, |
| source->linkage_data); |
| break; |
| case GST_MPEGTS_DVB_LINKAGE_EXTENDED_EVENT: |
| copy->linkage_data = g_ptr_array_ref (source->linkage_data); |
| break; |
| default: |
| break; |
| } |
| |
| copy->private_data_bytes = g_memdup (source->private_data_bytes, |
| source->private_data_length); |
| |
| return copy; |
| } |
| |
| void |
| gst_mpegts_dvb_linkage_descriptor_free (GstMpegtsDVBLinkageDescriptor * source) |
| { |
| if (source->linkage_data) |
| switch (source->linkage_type) { |
| case GST_MPEGTS_DVB_LINKAGE_MOBILE_HAND_OVER: |
| g_slice_free (GstMpegtsDVBLinkageMobileHandOver, source->linkage_data); |
| break; |
| case GST_MPEGTS_DVB_LINKAGE_EVENT: |
| g_slice_free (GstMpegtsDVBLinkageEvent, source->linkage_data); |
| break; |
| case GST_MPEGTS_DVB_LINKAGE_EXTENDED_EVENT: |
| g_ptr_array_unref (source->linkage_data); |
| break; |
| default: |
| break; |
| } |
| |
| g_free (source->private_data_bytes); |
| g_slice_free (GstMpegtsDVBLinkageDescriptor, source); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsDVBLinkageDescriptor, |
| gst_mpegts_dvb_linkage_descriptor, |
| (GBoxedCopyFunc) _gst_mpegts_dvb_linkage_descriptor_copy, |
| (GFreeFunc) gst_mpegts_dvb_linkage_descriptor_free); |
| |
| static void |
| _gst_mpegts_dvb_linkage_extened_event_free (GstMpegtsDVBLinkageExtendedEvent * |
| item) |
| { |
| g_slice_free (GstMpegtsDVBLinkageExtendedEvent, item); |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_linkage: |
| * @descriptor: a %GST_MTS_DESC_DVB_LINKAGE #GstMpegtsDescriptor |
| * @res: (out) (transfer full): the #GstMpegtsDVBLinkageDescriptor to fill |
| * |
| * Extracts the DVB linkage information from @descriptor. |
| * |
| * Returns: %TRUE if parsing succeeded, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_linkage (const GstMpegtsDescriptor * descriptor, |
| GstMpegtsDVBLinkageDescriptor ** desc) |
| { |
| guint i; |
| guint8 *data, *end; |
| GstMpegtsDVBLinkageDescriptor *res; |
| |
| g_return_val_if_fail (descriptor != NULL && desc != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_LINKAGE, 7, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| end = data + descriptor->length; |
| |
| res = g_slice_new0 (GstMpegtsDVBLinkageDescriptor); |
| |
| res->transport_stream_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| |
| res->original_network_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| |
| res->service_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| |
| res->linkage_type = *data; |
| data += 1; |
| |
| switch (res->linkage_type) { |
| case GST_MPEGTS_DVB_LINKAGE_MOBILE_HAND_OVER:{ |
| GstMpegtsDVBLinkageMobileHandOver *hand_over; |
| |
| if (end - data < 1) |
| goto fail; |
| |
| hand_over = g_slice_new0 (GstMpegtsDVBLinkageMobileHandOver); |
| res->linkage_data = (gpointer) hand_over; |
| |
| hand_over->origin_type = (*data) & 0x01; |
| hand_over->hand_over_type = (*data >> 4) & 0x0f; |
| data += 1; |
| |
| if (hand_over->hand_over_type == |
| GST_MPEGTS_DVB_LINKAGE_HAND_OVER_IDENTICAL |
| || hand_over->hand_over_type == |
| GST_MPEGTS_DVB_LINKAGE_HAND_OVER_LOCAL_VARIATION |
| || hand_over->hand_over_type == |
| GST_MPEGTS_DVB_LINKAGE_HAND_OVER_ASSOCIATED) { |
| if (end - data < 2) |
| goto fail; |
| |
| hand_over->network_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| } |
| |
| if (hand_over->origin_type == 0) { |
| if (end - data < 2) |
| goto fail; |
| |
| hand_over->initial_service_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| } |
| break; |
| } |
| case GST_MPEGTS_DVB_LINKAGE_EVENT:{ |
| GstMpegtsDVBLinkageEvent *event; |
| |
| if (end - data < 3) |
| goto fail; |
| |
| event = g_slice_new0 (GstMpegtsDVBLinkageEvent); |
| res->linkage_data = (gpointer) event; |
| |
| event->target_event_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| event->target_listed = *data & 0x01; |
| event->event_simulcast = (*data >> 1) & 0x01; |
| data += 1; |
| break; |
| } |
| case GST_MPEGTS_DVB_LINKAGE_EXTENDED_EVENT:{ |
| GPtrArray *ext_events; |
| ext_events = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_dvb_linkage_extened_event_free); |
| |
| res->linkage_data = (gpointer) ext_events; |
| |
| for (i = 0; i < *data++;) { |
| GstMpegtsDVBLinkageExtendedEvent *ext_event; |
| |
| if (end - data < 3) |
| goto fail; |
| |
| ext_event = g_slice_new0 (GstMpegtsDVBLinkageExtendedEvent); |
| g_ptr_array_add (res->linkage_data, ext_event); |
| |
| ext_event->target_event_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| i += 2; |
| |
| ext_event->target_listed = *data & 0x01; |
| ext_event->event_simulcast = (*data >> 1) & 0x01; |
| ext_event->link_type = (*data >> 3) & 0x03; |
| ext_event->target_id_type = (*data >> 5) & 0x03; |
| ext_event->original_network_id_flag = (*data >> 6) & 0x01; |
| ext_event->service_id_flag = (*data >> 7) & 0x01; |
| data += 1; |
| i += 1; |
| |
| if (ext_event->target_id_type == 3) { |
| if (end - data < 2) |
| goto fail; |
| |
| ext_event->user_defined_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| i += 2; |
| } else { |
| if (ext_event->target_id_type == 1) { |
| if (end - data < 2) |
| goto fail; |
| |
| ext_event->target_transport_stream_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| i += 2; |
| } |
| if (ext_event->original_network_id_flag) { |
| if (end - data < 2) |
| goto fail; |
| |
| ext_event->target_original_network_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| i += 2; |
| } |
| if (ext_event->service_id_flag) { |
| if (end - data < 2) |
| goto fail; |
| |
| ext_event->target_service_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| i += 2; |
| } |
| } |
| } |
| break; |
| } |
| default: |
| break; |
| } |
| |
| res->private_data_length = end - data; |
| res->private_data_bytes = g_memdup (data, res->private_data_length); |
| |
| *desc = res; |
| |
| return TRUE; |
| |
| fail: |
| gst_mpegts_dvb_linkage_descriptor_free (res); |
| return FALSE; |
| } |
| |
| /** |
| * gst_mpegts_dvb_linkage_descriptor_get_mobile_hand_over: |
| * @desc: the #GstMpegtsDVBLinkageDescriptor |
| * |
| * Returns: The #GstMpegtsDVBLinkageMobileHandOver or %NULL if an error happened |
| */ |
| const GstMpegtsDVBLinkageMobileHandOver * |
| gst_mpegts_dvb_linkage_descriptor_get_mobile_hand_over (const |
| GstMpegtsDVBLinkageDescriptor * desc) |
| { |
| g_return_val_if_fail (desc != NULL, NULL); |
| g_return_val_if_fail (desc->linkage_type == |
| GST_MPEGTS_DVB_LINKAGE_MOBILE_HAND_OVER, NULL); |
| |
| return (const GstMpegtsDVBLinkageMobileHandOver *) desc->linkage_data; |
| } |
| |
| /** |
| * gst_mpegts_dvb_linkage_descriptor_get_event: |
| * @desc: the #GstMpegtsDVBLinkageDescriptor |
| * |
| * Returns: The #GstMpegtsDVBLinkageEvent or %NULL if an error happened |
| */ |
| const GstMpegtsDVBLinkageEvent * |
| gst_mpegts_dvb_linkage_descriptor_get_event (const GstMpegtsDVBLinkageDescriptor |
| * desc) |
| { |
| g_return_val_if_fail (desc != NULL, NULL); |
| g_return_val_if_fail (desc->linkage_type == |
| GST_MPEGTS_DVB_LINKAGE_EVENT, NULL); |
| |
| return (const GstMpegtsDVBLinkageEvent *) desc->linkage_data; |
| } |
| |
| /** |
| * gst_mpegts_dvb_linkage_descriptor_get_extended_event: |
| * @desc: the #GstMpegtsDVBLinkageDescriptor |
| * |
| * Returns: (element-type GstMpegtsDVBLinkageExtendedEvent): an #GstMpegtsDVBLinkageExtendedEvent array or %NULL if an error happened |
| */ |
| const GPtrArray * |
| gst_mpegts_dvb_linkage_descriptor_get_extended_event (const |
| GstMpegtsDVBLinkageDescriptor * desc) |
| { |
| g_return_val_if_fail (desc != NULL, NULL); |
| g_return_val_if_fail (desc->linkage_type == |
| GST_MPEGTS_DVB_LINKAGE_EXTENDED_EVENT, NULL); |
| |
| return (const GPtrArray *) desc->linkage_data; |
| } |
| |
| /* GST_MTS_DESC_DVB_SHORT_EVENT (0x4D) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_short_event: |
| * @descriptor: a %GST_MTS_DESC_DVB_SHORT_EVENT #GstMpegtsDescriptor |
| * @language_code: (out) (transfer full) (allow-none): the language code |
| * @event_name: (out) (transfer full) (allow-none): the event name |
| * @text: (out) (transfer full) (allow-none): the event text |
| * |
| * Extracts the DVB short event information from @descriptor. |
| * |
| * Returns: %TRUE if parsing succeeded, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_short_event (const GstMpegtsDescriptor * |
| descriptor, gchar ** language_code, gchar ** event_name, gchar ** text) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL, FALSE); |
| /* Need at least 5 bytes (3 for language code, 2 for each string length) */ |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_SHORT_EVENT, 5, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| if (language_code) |
| *language_code = convert_lang_code (data); |
| |
| data += 3; |
| if (event_name) |
| *event_name = get_encoding_and_convert ((const gchar *) data + 1, *data); |
| data += *data + 1; |
| if (text) |
| *text = get_encoding_and_convert ((const gchar *) data + 1, *data); |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_TELETEXT (0x56) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_teletext_idx: |
| * @descriptor: a %GST_MTS_DESC_DVB_TELETEXT #GstMpegtsDescriptor |
| * @idx: The id of the teletext to get |
| * @language_code: (out) (transfer full) (allow-none): a null-terminated string |
| * @teletext_type: (out) (allow-none): #GstMpegtsDVBTeletextType |
| * @magazine_number: (out) (allow-none): |
| * @page_number: (out) (allow-none): |
| * |
| * Parses teletext number @idx in the @descriptor. The language is in ISO639 format. |
| * |
| * Returns: FALSE on out-of-bounds and errors |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_teletext_idx (const GstMpegtsDescriptor * |
| descriptor, guint idx, gchar ** language_code, |
| GstMpegtsDVBTeletextType * teletext_type, guint8 * magazine_number, |
| guint8 * page_number) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_TELETEXT, 0, FALSE); |
| |
| if (descriptor->length / 5 <= idx) |
| return FALSE; |
| |
| data = (guint8 *) descriptor->data + 2 + idx * 5; |
| |
| if (language_code) |
| *language_code = convert_lang_code (data); |
| |
| if (teletext_type) |
| *teletext_type = data[3] >> 3; |
| |
| if (magazine_number) |
| *magazine_number = data[3] & 0x07; |
| |
| if (page_number) |
| *page_number = data[4]; |
| |
| return TRUE; |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_teletext_nb: |
| * @descriptor: a %GST_MTS_DESC_DVB_TELETEXT #GstMpegtsDescriptor |
| * |
| * Find the number of teletext entries in @descriptor |
| * |
| * Returns: Number of teletext entries |
| */ |
| guint |
| gst_mpegts_descriptor_parse_dvb_teletext_nb (const GstMpegtsDescriptor * |
| descriptor) |
| { |
| g_return_val_if_fail (descriptor != NULL, 0); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_TELETEXT, 0, 0); |
| |
| return descriptor->length / 5; |
| } |
| |
| /* GST_MTS_DESC_DVB_SUBTITLING (0x59) */ |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_subtitling_idx: |
| * @descriptor: a %GST_MTS_DESC_DVB_SUBTITLING #GstMpegtsDescriptor |
| * @idx: Table id of the entry to parse |
| * @lang: (out) (transfer full): the language code |
| * @type: (out) (transfer none) (allow-none): the type of subtitling |
| * @composition_page_id: (out) (transfer none) (allow-none): the composition page id |
| * @ancillary_page_id: (out) (transfer none) (allow-none): the ancillary page id |
| * |
| * Extracts the DVB subtitling informatio from specific table id in @descriptor. |
| * |
| * Note: Use #gst_tag_get_language_code if you want to get the the |
| * ISO 639-1 language code from the returned ISO 639-2 one. |
| * |
| * Returns: %TRUE if parsing succeeded, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_subtitling_idx (const GstMpegtsDescriptor * |
| descriptor, guint idx, gchar ** lang, guint8 * type, |
| guint16 * composition_page_id, guint16 * ancillary_page_id) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL && lang != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_SUBTITLING, 0, FALSE); |
| |
| /* If we went too far, return FALSE */ |
| if (descriptor->length / 8 <= idx) |
| return FALSE; |
| |
| data = (guint8 *) descriptor->data + 2 + idx * 8; |
| |
| *lang = convert_lang_code (data); |
| |
| data += 3; |
| |
| if (type) |
| *type = *data; |
| data += 1; |
| if (composition_page_id) |
| *composition_page_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| if (ancillary_page_id) |
| *ancillary_page_id = GST_READ_UINT16_BE (data); |
| |
| return TRUE; |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_subtitling_nb: |
| * @descriptor: a %GST_MTS_DESC_DVB_SUBTITLING #GstMpegtsDescriptor |
| * |
| * Returns: The number of entries in @descriptor |
| */ |
| guint |
| gst_mpegts_descriptor_parse_dvb_subtitling_nb (const GstMpegtsDescriptor * |
| descriptor) |
| { |
| g_return_val_if_fail (descriptor != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_SUBTITLING, 0, FALSE); |
| |
| return descriptor->length / 8; |
| } |
| |
| /** |
| * gst_mpegts_descriptor_from_dvb_subtitling: |
| * @lang: (transfer none): a string containing the ISO639 language |
| * @type: subtitling type |
| * @composition: composition page id |
| * @ancillary: ancillary page id |
| */ |
| GstMpegtsDescriptor * |
| gst_mpegts_descriptor_from_dvb_subtitling (const gchar * lang, |
| guint8 type, guint16 composition, guint16 ancillary) |
| { |
| GstMpegtsDescriptor *descriptor; |
| guint8 *data; |
| |
| g_return_val_if_fail (lang != NULL, NULL); |
| |
| descriptor = _new_descriptor (GST_MTS_DESC_DVB_SUBTITLING, 8); |
| |
| data = descriptor->data + 2; |
| |
| memcpy (data, lang, 3); |
| data += 3; |
| |
| *data++ = type; |
| |
| GST_WRITE_UINT16_BE (data, composition); |
| data += 2; |
| |
| GST_WRITE_UINT16_BE (data, ancillary); |
| |
| return descriptor; |
| } |
| |
| /* GST_MTS_DESC_DVB_EXTENDED_EVENT (0x4E) */ |
| static GstMpegtsExtendedEventDescriptor * |
| _gst_mpegts_extended_event_descriptor_copy (GstMpegtsExtendedEventDescriptor * |
| source) |
| { |
| GstMpegtsExtendedEventDescriptor *copy; |
| |
| copy = g_slice_dup (GstMpegtsExtendedEventDescriptor, source); |
| copy->items = g_ptr_array_ref (source->items); |
| copy->text = g_strdup (source->text); |
| |
| return copy; |
| } |
| |
| void |
| gst_mpegts_extended_event_descriptor_free (GstMpegtsExtendedEventDescriptor * |
| source) |
| { |
| g_free (source->text); |
| g_free (source->language_code); |
| g_ptr_array_unref (source->items); |
| g_slice_free (GstMpegtsExtendedEventDescriptor, source); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsExtendedEventDescriptor, |
| gst_mpegts_extended_event_descriptor, |
| (GBoxedCopyFunc) _gst_mpegts_extended_event_descriptor_copy, |
| (GFreeFunc) gst_mpegts_extended_event_descriptor_free); |
| |
| static void |
| _gst_mpegts_extended_event_item_free (GstMpegtsExtendedEventItem * item) |
| { |
| g_free (item->item); |
| g_free (item->item_description); |
| g_slice_free (GstMpegtsExtendedEventItem, item); |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_extended_event: |
| * @descriptor: a %GST_MTS_DESC_DVB_EXTENDED_EVENT #GstMpegtsDescriptor |
| * @res: (out) (transfer full): the #GstMpegtsExtendedEventDescriptor to fill |
| * |
| * Extracts the DVB extended event information from @descriptor. |
| * |
| * Returns: %TRUE if parsing succeeded, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_extended_event (const GstMpegtsDescriptor |
| * descriptor, GstMpegtsExtendedEventDescriptor ** desc) |
| { |
| guint8 *data, *pdata; |
| guint8 tmp, len_item; |
| GstMpegtsExtendedEventItem *item; |
| GstMpegtsExtendedEventDescriptor *res; |
| |
| g_return_val_if_fail (descriptor != NULL && desc != NULL, FALSE); |
| /* Need at least 6 bytes (1 for desc number, 3 for language code, 2 for the loop length) */ |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_EXTENDED_EVENT, 6, FALSE); |
| |
| res = g_slice_new0 (GstMpegtsExtendedEventDescriptor); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| tmp = *data; |
| res->descriptor_number = tmp >> 4; |
| res->last_descriptor_number = tmp & 0x0f; |
| |
| data += 1; |
| |
| res->language_code = convert_lang_code (data); |
| data += 3; |
| |
| len_item = *data; |
| if (len_item > descriptor->length - 5) { |
| gst_mpegts_extended_event_descriptor_free (res); |
| return FALSE; |
| } |
| |
| data += 1; |
| |
| res->items = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_extended_event_item_free); |
| |
| pdata = data + len_item; |
| while (data < pdata) { |
| item = g_slice_new0 (GstMpegtsExtendedEventItem); |
| item->item_description = |
| get_encoding_and_convert ((const gchar *) data + 1, *data); |
| |
| data += *data + 1; |
| |
| item->item = get_encoding_and_convert ((const gchar *) data + 1, *data); |
| |
| data += *data + 1; |
| |
| g_ptr_array_add (res->items, item); |
| } |
| if (pdata != data) { |
| gst_mpegts_extended_event_descriptor_free (res); |
| return FALSE; |
| } |
| res->text = get_encoding_and_convert ((const gchar *) data + 1, *data); |
| |
| *desc = res; |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_COMPONENT (0x50) */ |
| static GstMpegtsComponentDescriptor * |
| _gst_mpegts_dvb_component_descriptor_copy (GstMpegtsComponentDescriptor * |
| source) |
| { |
| GstMpegtsComponentDescriptor *copy; |
| |
| copy = g_slice_dup (GstMpegtsComponentDescriptor, source); |
| copy->language_code = g_strdup (source->language_code); |
| copy->text = g_strdup (source->text); |
| |
| return copy; |
| } |
| |
| void |
| gst_mpegts_dvb_component_descriptor_free (GstMpegtsComponentDescriptor * source) |
| { |
| g_free (source->language_code); |
| g_free (source->text); |
| g_slice_free (GstMpegtsComponentDescriptor, source); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsComponentDescriptor, |
| gst_mpegts_component_descriptor, |
| (GBoxedCopyFunc) _gst_mpegts_dvb_component_descriptor_copy, |
| (GFreeFunc) gst_mpegts_dvb_component_descriptor_free); |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_component: |
| * @descriptor: a %GST_MTS_DESC_DVB_COMPONENT #GstMpegtsDescriptor |
| * @res: (out) (transfer full): the #GstMpegtsComponentDescriptor to fill |
| * |
| * Extracts the DVB component information from @descriptor. |
| * |
| * Returns: %TRUE if parsing succeeded, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_component (const GstMpegtsDescriptor |
| * descriptor, GstMpegtsComponentDescriptor ** res) |
| { |
| guint8 *data; |
| guint8 len; |
| GstMpegtsComponentDescriptor *desc; |
| |
| g_return_val_if_fail (descriptor != NULL && res != NULL, FALSE); |
| /* Need 6 bytes at least (1 for content, 1 for type, 1 for tag, 3 for language code) */ |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_COMPONENT, 6, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| desc = g_slice_new0 (GstMpegtsComponentDescriptor); |
| |
| desc->stream_content = *data & 0x0f; |
| data += 1; |
| |
| desc->component_type = *data; |
| data += 1; |
| |
| desc->component_tag = *data; |
| data += 1; |
| |
| desc->language_code = convert_lang_code (data); |
| data += 3; |
| |
| len = descriptor->length - 6; |
| if (len) |
| desc->text = get_encoding_and_convert ((const gchar *) data, len); |
| |
| *res = desc; |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_STREAM_IDENTIFIER (0x52) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_stream_identifier: |
| * @descriptor: a %GST_MTS_DESC_DVB_CONTENT #GstMpegtsDescriptor |
| * @component_tag: (out) (transfer none): the component tag |
| * |
| * Extracts the component tag from @descriptor. |
| * |
| * Returns: %TRUE if the parsing happended correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_stream_identifier (const GstMpegtsDescriptor |
| * descriptor, guint8 * component_tag) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL && component_tag != NULL, FALSE); |
| __common_desc_checks_exact (descriptor, GST_MTS_DESC_DVB_STREAM_IDENTIFIER, |
| 1, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *component_tag = *data; |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_CA_IDENTIFIER (0x53) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_ca_identifier: |
| * @descriptor: a %GST_MTS_DESC_DVB_CA_IDENTIFIER #GstMpegtsDescriptor |
| * @list: (out) (transfer full) (element-type guint16): a list of ca identifier. |
| * Edge entry identifies the CA system. Allocations of the value of this field |
| * are found in http://www.dvbservices.com |
| * |
| * Extracts ca id's from @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_ca_identifier (const GstMpegtsDescriptor * |
| descriptor, GArray ** list) |
| { |
| guint8 *data; |
| guint16 tmp; |
| guint i; |
| |
| g_return_val_if_fail (descriptor != NULL && list != NULL, FALSE); |
| /* 2 bytes = one entry */ |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_CA_IDENTIFIER, 2, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *list = g_array_new (FALSE, FALSE, sizeof (guint16)); |
| |
| for (i = 0; i < descriptor->length - 1; i += 2) { |
| tmp = GST_READ_UINT16_BE (data); |
| g_array_append_val (*list, tmp); |
| data += 2; |
| } |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_CONTENT (0x54) */ |
| static void |
| _gst_mpegts_content_free (GstMpegtsContent * content) |
| { |
| g_slice_free (GstMpegtsContent, content); |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_content: |
| * @descriptor: a %GST_MTS_DESC_DVB_CONTENT #GstMpegtsDescriptor |
| * @content: (out) (transfer full) (element-type GstMpegtsContent): #GstMpegtsContent |
| * |
| * Extracts the DVB content information from @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_content (const GstMpegtsDescriptor |
| * descriptor, GPtrArray ** content) |
| { |
| guint8 *data; |
| guint8 len, tmp; |
| guint8 i; |
| |
| g_return_val_if_fail (descriptor != NULL && content != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_CONTENT, 0, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| len = descriptor->length; |
| |
| *content = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_content_free); |
| for (i = 0; i < len;) { |
| GstMpegtsContent *cont = g_slice_new0 (GstMpegtsContent); |
| tmp = *data; |
| cont->content_nibble_1 = (tmp & 0xf0) >> 4; |
| cont->content_nibble_2 = tmp & 0x0f; |
| data += 1; |
| cont->user_byte = *data; |
| data += 1; |
| i += 2; |
| g_ptr_array_add (*content, cont); |
| } |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_PARENTAL_RATING (0x55) */ |
| static void |
| _gst_mpegts_dvb_parental_rating_item_free (GstMpegtsDVBParentalRatingItem * |
| item) |
| { |
| g_free (item->country_code); |
| g_slice_free (GstMpegtsDVBParentalRatingItem, item); |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_parental_rating: |
| * @descriptor: a %GST_MTS_DESC_DVB_PARENTAL_RATING #GstMpegtsDescriptor |
| * @rating: (out) (transfer full) (element-type GstMpegtsDVBParentalRatingItem): |
| * #GstMpegtsDVBParentalRatingItem |
| * |
| * Extracts the DVB parental rating information from @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_parental_rating (const GstMpegtsDescriptor |
| * descriptor, GPtrArray ** rating) |
| { |
| guint8 *data; |
| guint i; |
| |
| g_return_val_if_fail (descriptor != NULL && rating != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_PARENTAL_RATING, 0, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *rating = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_dvb_parental_rating_item_free); |
| |
| for (i = 0; i < descriptor->length - 3; i += 4) { |
| GstMpegtsDVBParentalRatingItem *item = |
| g_slice_new0 (GstMpegtsDVBParentalRatingItem); |
| g_ptr_array_add (*rating, item); |
| |
| item->country_code = convert_lang_code (data); |
| data += 3; |
| |
| if (g_strcmp0 (item->country_code, "BRA") == 0) { |
| /* brasil */ |
| switch (*data & 0xf) { |
| case 1: |
| item->rating = 6; |
| break; |
| case 2: |
| item->rating = 10; |
| break; |
| case 3: |
| item->rating = 12; |
| break; |
| case 4: |
| item->rating = 14; |
| break; |
| case 5: |
| item->rating = 16; |
| break; |
| case 6: |
| item->rating = 18; |
| break; |
| default: |
| item->rating = 0; |
| break; |
| } |
| } else { |
| item->rating = (*data & 0xf) + 3; |
| } |
| |
| data += 1; |
| } |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM (0x5A) */ |
| /** |
| * gst_mpegts_descriptor_parse_terrestrial_delivery_system: |
| * @descriptor: a %GST_MTS_DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM #GstMpegtsDescriptor |
| * @res: (out) (transfer none): #GstMpegtsTerrestrialDeliverySystemDescriptor |
| * |
| * Parses out the terrestrial delivery system from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_terrestrial_delivery_system (const |
| GstMpegtsDescriptor * descriptor, |
| GstMpegtsTerrestrialDeliverySystemDescriptor * res) |
| { |
| guint8 *data; |
| guint8 tmp; |
| |
| g_return_val_if_fail (descriptor != NULL && res != NULL, FALSE); |
| /* Descriptor is always 11 bytes long */ |
| __common_desc_checks_exact (descriptor, |
| GST_MTS_DESC_DVB_TERRESTRIAL_DELIVERY_SYSTEM, 11, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| res->frequency = 0; |
| res->frequency = GST_READ_UINT32_BE (data); |
| res->frequency *= 10; |
| |
| data += 4; |
| |
| tmp = *data; |
| switch ((tmp >> 5) & 0x07) { |
| case 0: |
| res->bandwidth = 8000000; |
| break; |
| case 1: |
| res->bandwidth = 7000000; |
| break; |
| case 2: |
| res->bandwidth = 6000000; |
| break; |
| case 3: |
| res->bandwidth = 5000000; |
| break; |
| default: |
| res->bandwidth = 0; |
| break; |
| } |
| |
| res->priority = (tmp >> 4) & 0x01; |
| res->time_slicing = (tmp >> 3) & 0x01; |
| res->mpe_fec = (tmp >> 2) & 0x01; |
| data += 1; |
| |
| tmp = *data; |
| switch ((tmp >> 6) & 0x03) { |
| case 0: |
| res->constellation = GST_MPEGTS_MODULATION_QPSK; |
| break; |
| case 1: |
| res->constellation = GST_MPEGTS_MODULATION_QAM_16; |
| break; |
| case 2: |
| res->constellation = GST_MPEGTS_MODULATION_QAM_64; |
| break; |
| default: |
| break; |
| } |
| |
| switch ((tmp >> 3) & 0x07) { |
| case 0: |
| res->hierarchy = GST_MPEGTS_HIERARCHY_NONE; |
| break; |
| case 1: |
| res->hierarchy = GST_MPEGTS_HIERARCHY_1; |
| break; |
| case 2: |
| res->hierarchy = GST_MPEGTS_HIERARCHY_2; |
| break; |
| case 3: |
| res->hierarchy = GST_MPEGTS_HIERARCHY_4; |
| break; |
| case 4: |
| res->hierarchy = GST_MPEGTS_HIERARCHY_NONE; |
| break; |
| case 5: |
| res->hierarchy = GST_MPEGTS_HIERARCHY_1; |
| break; |
| case 6: |
| res->hierarchy = GST_MPEGTS_HIERARCHY_2; |
| break; |
| case 7: |
| res->hierarchy = GST_MPEGTS_HIERARCHY_4; |
| break; |
| default: |
| break; |
| } |
| |
| switch (tmp & 0x07) { |
| case 0: |
| res->code_rate_hp = GST_MPEGTS_FEC_1_2; |
| break; |
| case 1: |
| res->code_rate_hp = GST_MPEGTS_FEC_2_3; |
| break; |
| case 2: |
| res->code_rate_hp = GST_MPEGTS_FEC_3_4; |
| break; |
| case 3: |
| res->code_rate_hp = GST_MPEGTS_FEC_5_6; |
| break; |
| case 4: |
| res->code_rate_hp = GST_MPEGTS_FEC_7_8; |
| break; |
| default: |
| break; |
| } |
| data += 1; |
| |
| tmp = *data; |
| switch ((tmp >> 5) & 0x07) { |
| case 0: |
| res->code_rate_lp = GST_MPEGTS_FEC_1_2; |
| break; |
| case 1: |
| res->code_rate_lp = GST_MPEGTS_FEC_2_3; |
| break; |
| case 2: |
| res->code_rate_lp = GST_MPEGTS_FEC_3_4; |
| break; |
| case 3: |
| res->code_rate_lp = GST_MPEGTS_FEC_5_6; |
| break; |
| case 4: |
| res->code_rate_lp = GST_MPEGTS_FEC_7_8; |
| break; |
| default: |
| break; |
| } |
| |
| switch ((tmp >> 3) & 0x03) { |
| case 0: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_1_32; |
| break; |
| case 1: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_1_16; |
| break; |
| case 2: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_1_8; |
| break; |
| case 3: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_1_4; |
| break; |
| default: |
| break; |
| } |
| |
| switch ((tmp >> 1) & 0x03) { |
| case 0: |
| res->transmission_mode = GST_MPEGTS_TRANSMISSION_MODE_2K; |
| break; |
| case 1: |
| res->transmission_mode = GST_MPEGTS_TRANSMISSION_MODE_8K; |
| break; |
| case 2: |
| res->transmission_mode = GST_MPEGTS_TRANSMISSION_MODE_4K; |
| break; |
| default: |
| break; |
| } |
| res->other_frequency = tmp & 0x01; |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_MULTILINGUAL_NETWORK_NAME (0x5B) */ |
| static void |
| _gst_mpegts_dvb_multilingual_network_name_item_free |
| (GstMpegtsDvbMultilingualNetworkNameItem * item) |
| { |
| g_free (item->network_name); |
| g_free (item->language_code); |
| g_slice_free (GstMpegtsDvbMultilingualNetworkNameItem, item); |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_multilingual_network_name: |
| * @descriptor: a %GST_MTS_DESC_DVB_MULTILINGUAL_NETWORK_NAME |
| * #GstMpegtsDescriptor |
| * @network_name_items: (out) (transfer full) (element-type GstMpegtsDvbMultilingualNetworkNameItem): |
| * a #GstMpegtsDvbMultilingualNetworkNameItem |
| * |
| * Parses out the multilingual network name from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_multilingual_network_name (const |
| GstMpegtsDescriptor * descriptor, GPtrArray ** network_name_items) |
| { |
| guint8 *data, i, len; |
| GstMpegtsDvbMultilingualNetworkNameItem *item; |
| |
| g_return_val_if_fail (descriptor != NULL && network_name_items != NULL, |
| FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_MULTILINGUAL_NETWORK_NAME, |
| 5, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *network_name_items = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_dvb_multilingual_network_name_item_free); |
| |
| for (i = 0; i < descriptor->length - 3;) { |
| item = g_slice_new0 (GstMpegtsDvbMultilingualNetworkNameItem); |
| g_ptr_array_add (*network_name_items, item); |
| item->language_code = convert_lang_code (data); |
| data += 3; |
| i += 3; |
| |
| len = *data; |
| item->network_name = |
| get_encoding_and_convert ((const gchar *) data + 1, len); |
| data += len + 1; |
| i += len + 1; |
| } |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_MULTILINGUAL_BOUQUET_NAME (0x5C) */ |
| static void |
| _gst_mpegts_dvb_multilingual_bouquet_name_item_free |
| (GstMpegtsDvbMultilingualBouquetNameItem * item) |
| { |
| g_free (item->language_code); |
| g_free (item->bouquet_name); |
| g_slice_free (GstMpegtsDvbMultilingualBouquetNameItem, item); |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_multilingual_bouquet_name: |
| * @descriptor: a %GST_MTS_DESC_DVB_MULTILINGUAL_BOUQUET_NAME |
| * #GstMpegtsDescriptor |
| * @bouquet_name_items: (out) (transfer full) (element-type GstMpegtsDvbMultilingualBouquetNameItem): |
| * a #GstMpegtsDvbMultilingualBouquetNameItem |
| * |
| * Parses out the multilingual bouquet name from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_multilingual_bouquet_name (const |
| GstMpegtsDescriptor * descriptor, GPtrArray ** bouquet_name_items) |
| { |
| guint8 *data, i, len; |
| GstMpegtsDvbMultilingualBouquetNameItem *item; |
| |
| g_return_val_if_fail (descriptor != NULL && bouquet_name_items != NULL, |
| FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_MULTILINGUAL_BOUQUET_NAME, |
| 5, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *bouquet_name_items = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_dvb_multilingual_bouquet_name_item_free); |
| |
| for (i = 0; i < descriptor->length - 3;) { |
| item = g_slice_new0 (GstMpegtsDvbMultilingualBouquetNameItem); |
| g_ptr_array_add (*bouquet_name_items, item); |
| item->language_code = convert_lang_code (data); |
| data += 3; |
| i += 3; |
| |
| len = *data; |
| item->bouquet_name = |
| get_encoding_and_convert ((const gchar *) data + 1, len); |
| data += len + 1; |
| i += len + 1; |
| } |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_MULTILINGUAL_SERVICE_NAME (0x5D) */ |
| static void |
| _gst_mpegts_dvb_multilingual_service_name_item_free |
| (GstMpegtsDvbMultilingualServiceNameItem * item) |
| { |
| g_free (item->provider_name); |
| g_free (item->service_name); |
| g_free (item->language_code); |
| g_slice_free (GstMpegtsDvbMultilingualServiceNameItem, item); |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_multilingual_service_name: |
| * @descriptor: a %GST_MTS_DESC_DVB_MULTILINGUAL_SERVICE_NAME |
| * #GstMpegtsDescriptor |
| * @service_name_items: (out) (transfer full) (element-type GstMpegtsDvbMultilingualServiceNameItem): |
| * a #GstMpegtsDvbMultilingualServiceNameItem |
| * |
| * Parses out the multilingual service name from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_multilingual_service_name (const |
| GstMpegtsDescriptor * descriptor, GPtrArray ** service_name_items) |
| { |
| guint8 *data, i, len; |
| GstMpegtsDvbMultilingualServiceNameItem *item; |
| |
| g_return_val_if_fail (descriptor != NULL && service_name_items != NULL, |
| FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_MULTILINGUAL_SERVICE_NAME, |
| 7, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *service_name_items = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_dvb_multilingual_service_name_item_free); |
| |
| for (i = 0; i < descriptor->length - 3;) { |
| item = g_slice_new0 (GstMpegtsDvbMultilingualServiceNameItem); |
| g_ptr_array_add (*service_name_items, item); |
| item->language_code = convert_lang_code (data); |
| data += 3; |
| i += 3; |
| |
| len = *data; |
| item->provider_name = |
| get_encoding_and_convert ((const gchar *) data + 1, len); |
| data += len + 1; |
| i += len + 1; |
| |
| len = *data; |
| item->service_name = |
| get_encoding_and_convert ((const gchar *) data + 1, len); |
| data += len + 1; |
| i += len + 1; |
| } |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_MULTILINGUAL_COMPONENT (0x5E) */ |
| static void |
| _gst_mpegts_dvb_multilingual_component_item_free |
| (GstMpegtsDvbMultilingualComponentItem * item) |
| { |
| g_free (item->language_code); |
| g_free (item->description); |
| g_slice_free (GstMpegtsDvbMultilingualComponentItem, item); |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_multilingual_component: |
| * @descriptor: a %GST_MTS_DESC_DVB_MULTILINGUAL_COMPONENT |
| * #GstMpegtsDescriptor |
| * @component_tag: the component tag |
| * @component_description_items: (out) (transfer full) (element-type GstMpegtsDvbMultilingualComponentItem): |
| * a #GstMpegtsDvbMultilingualComponentItem |
| * |
| * Parses out the multilingual component from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_multilingual_component (const |
| GstMpegtsDescriptor * descriptor, guint8 * component_tag, |
| GPtrArray ** component_description_items) |
| { |
| guint8 *data, i, len; |
| GstMpegtsDvbMultilingualComponentItem *item; |
| |
| g_return_val_if_fail (descriptor != NULL |
| && component_description_items != NULL && component_tag != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_MULTILINGUAL_COMPONENT, 6, |
| FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *component_tag = *data; |
| data += 1; |
| |
| *component_description_items = |
| g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_dvb_multilingual_component_item_free); |
| |
| for (i = 0; i < descriptor->length - 3;) { |
| item = g_slice_new0 (GstMpegtsDvbMultilingualComponentItem); |
| g_ptr_array_add (*component_description_items, item); |
| item->language_code = convert_lang_code (data); |
| data += 3; |
| i += 3; |
| |
| len = *data; |
| item->description = |
| get_encoding_and_convert ((const gchar *) data + 1, len); |
| data += len + 1; |
| i += len + 1; |
| } |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_PRIVATE_DATA_SPECIFIER (0x5F) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_private_data_specifier: |
| * @descriptor: a %GST_MTS_DESC_DVB_PRIVATE_DATA_SPECIFIER #GstMpegtsDescriptor |
| * @private_data_specifier: (out): the private data specifier id |
| * registered by http://www.dvbservices.com/ |
| * @private_data: (out) (transfer full)(allow-none): additional data or NULL |
| * @length: (out) (allow-none): length of %private_data |
| * |
| * Parses out the private data specifier from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_private_data_specifier (const |
| GstMpegtsDescriptor * descriptor, guint32 * private_data_specifier, |
| guint8 ** private_data, guint8 * length) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL |
| && private_data_specifier != NULL, FALSE); |
| __common_desc_checks (descriptor, |
| GST_MTS_DESC_DVB_PRIVATE_DATA_SPECIFIER, 4, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *private_data_specifier = GST_READ_UINT32_BE (data); |
| |
| if (length && private_data) { |
| *length = descriptor->length - 4; |
| |
| *private_data = g_memdup (data + 4, *length); |
| } |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_FREQUENCY_LIST (0x62) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_frequency_list: |
| * @descriptor: a %GST_MTS_DESC_DVB_FREQUENCY_LIST #GstMpegtsDescriptor |
| * @offset: (out): %FALSE in Hz, %TRUE in kHz |
| * @list: (out) (transfer full) (element-type guint32): a list of all frequencies in Hz/kHz |
| * depending on %offset |
| * |
| * Parses out a list of frequencies from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_frequency_list (const GstMpegtsDescriptor |
| * descriptor, gboolean * offset, GArray ** list) |
| { |
| guint8 *data, type, len; |
| guint32 freq; |
| guint i; |
| |
| g_return_val_if_fail (descriptor != NULL && offset != NULL && |
| list != NULL, FALSE); |
| /* 1 byte coding system, 4 bytes each frequency entry */ |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_FREQUENCY_LIST, 5, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| type = *data & 0x03; |
| data += 1; |
| |
| if (type == 1) { |
| /* satellite */ |
| *offset = TRUE; |
| } else { |
| /* cable, terrestrial */ |
| *offset = FALSE; |
| } |
| |
| *list = g_array_new (FALSE, FALSE, sizeof (guint32)); |
| |
| len = descriptor->length - 1; |
| |
| for (i = 0; i < len - 3; i += 4) { |
| switch (type) { |
| case 1: |
| freq = BCD_32 (data) * 10; |
| break; |
| case 2: |
| freq = BCD_32 (data) * 100; |
| break; |
| case 3: |
| freq = GST_READ_UINT32_BE (data) * 10; |
| break; |
| default: |
| break; |
| } |
| |
| g_array_append_val (*list, freq); |
| data += 4; |
| } |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_DATA_BROADCAST (0x64) */ |
| static GstMpegtsDataBroadcastDescriptor * |
| _gst_mpegts_dvb_data_broadcast_descriptor_copy (GstMpegtsDataBroadcastDescriptor |
| * source) |
| { |
| GstMpegtsDataBroadcastDescriptor *copy; |
| |
| copy = g_slice_dup (GstMpegtsDataBroadcastDescriptor, source); |
| |
| copy->selector_bytes = g_memdup (source->selector_bytes, source->length); |
| copy->language_code = g_strdup (source->language_code); |
| copy->text = g_strdup (source->text); |
| |
| return copy; |
| } |
| |
| void |
| gst_mpegts_dvb_data_broadcast_descriptor_free (GstMpegtsDataBroadcastDescriptor |
| * source) |
| { |
| g_free (source->selector_bytes); |
| g_free (source->language_code); |
| g_free (source->text); |
| g_slice_free (GstMpegtsDataBroadcastDescriptor, source); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsDataBroadcastDescriptor, |
| gst_mpegts_dvb_data_broadcast_descriptor, |
| (GBoxedCopyFunc) _gst_mpegts_dvb_data_broadcast_descriptor_copy, |
| (GFreeFunc) gst_mpegts_dvb_data_broadcast_descriptor_free); |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_data_broadcast: |
| * @descriptor: a %GST_MTS_DESC_DVB_DATA_BROADCAST #GstMpegtsDescriptor |
| * @res: (out) (transfer full): #GstMpegtsDataBroadcastDescriptor |
| * |
| * Parses out the data broadcast from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_data_broadcast (const GstMpegtsDescriptor |
| * descriptor, GstMpegtsDataBroadcastDescriptor ** desc) |
| { |
| guint8 *data; |
| GstMpegtsDataBroadcastDescriptor *res; |
| |
| g_return_val_if_fail (descriptor != NULL && desc != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_DATA_BROADCAST, 8, FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| res = g_slice_new0 (GstMpegtsDataBroadcastDescriptor); |
| |
| res->data_broadcast_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| |
| res->component_tag = *data; |
| data += 1; |
| |
| res->length = *data; |
| data += 1; |
| |
| res->selector_bytes = g_memdup (data, res->length); |
| data += res->length; |
| |
| res->language_code = convert_lang_code (data); |
| data += 3; |
| |
| res->text = get_encoding_and_convert ((const gchar *) data + 1, *data); |
| |
| *desc = res; |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_SCRAMBLING (0x65) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_scrambling: |
| * @descriptor: a %GST_MTS_DESC_DVB_SCRAMBLING #GstMpegtsDescriptor |
| * @scrambling_mode: (out): This 8-bit field identifies the selected |
| * mode of the scrambling algorithm (#GstMpegtsDVBScramblingModeType). |
| * The technical details of the scrambling algorithm are available only |
| * to bona-fide users upon signature of a Non Disclosure Agreement (NDA) |
| * administered by the DVB Common Scrambling Algorithm Custodian. |
| * |
| * Parses out the scrambling mode from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_scrambling (const GstMpegtsDescriptor * |
| descriptor, GstMpegtsDVBScramblingModeType * scrambling_mode) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL && scrambling_mode != NULL, FALSE); |
| __common_desc_checks_exact (descriptor, GST_MTS_DESC_DVB_SCRAMBLING, 1, |
| FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *scrambling_mode = *data; |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_DVB_DATA_BROADCAST_ID (0x66) */ |
| /** |
| * gst_mpegts_descriptor_parse_dvb_data_broadcast_id: |
| * @descriptor: a %GST_MTS_DESC_DVB_DATA_BROADCAST_ID #GstMpegtsDescriptor |
| * @data_broadcast_id: (out): the data broadcast id |
| * @id_selector_bytes: (out) (transfer full): the selector bytes, if present |
| * @len: (out): the length of #id_selector_bytes |
| * |
| * Parses out the data broadcast id from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_data_broadcast_id (const GstMpegtsDescriptor |
| * descriptor, guint16 * data_broadcast_id, guint8 ** id_selector_bytes, |
| guint8 * len) |
| { |
| guint8 *data; |
| |
| g_return_val_if_fail (descriptor != NULL && data_broadcast_id != NULL && |
| id_selector_bytes != NULL, FALSE); |
| __common_desc_checks (descriptor, GST_MTS_DESC_DVB_DATA_BROADCAST_ID, 2, |
| FALSE); |
| |
| data = (guint8 *) descriptor->data + 2; |
| |
| *data_broadcast_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| |
| *len = descriptor->length - 2; |
| |
| *id_selector_bytes = g_memdup (data, *len); |
| |
| return TRUE; |
| } |
| |
| /* GST_MTS_DESC_EXT_DVB_T2_DELIVERY_SYSTEM (0x7F && 0x04) */ |
| static GstMpegtsT2DeliverySystemDescriptor |
| * _gst_mpegts_t2_delivery_system_descriptor_copy |
| (GstMpegtsT2DeliverySystemDescriptor * source) |
| { |
| GstMpegtsT2DeliverySystemDescriptor *copy; |
| |
| copy = g_slice_dup (GstMpegtsT2DeliverySystemDescriptor, source); |
| copy->cells = g_ptr_array_ref (source->cells); |
| |
| return copy; |
| } |
| |
| void gst_mpegts_t2_delivery_system_descriptor_free |
| (GstMpegtsT2DeliverySystemDescriptor * source) |
| { |
| g_ptr_array_unref (source->cells); |
| g_slice_free (GstMpegtsT2DeliverySystemDescriptor, source); |
| } |
| |
| G_DEFINE_BOXED_TYPE (GstMpegtsT2DeliverySystemDescriptor, |
| gst_mpegts_t2_delivery_system_descriptor, |
| (GBoxedCopyFunc) _gst_mpegts_t2_delivery_system_descriptor_copy, |
| (GFreeFunc) gst_mpegts_t2_delivery_system_descriptor_free); |
| |
| static void |
| _gst_mpegts_t2_delivery_system_cell_extension_free |
| (GstMpegtsT2DeliverySystemCellExtension * ext) |
| { |
| g_slice_free (GstMpegtsT2DeliverySystemCellExtension, ext); |
| } |
| |
| static void |
| _gst_mpegts_t2_delivery_system_cell_free (GstMpegtsT2DeliverySystemCell * cell) |
| { |
| g_ptr_array_unref (cell->sub_cells); |
| g_array_unref (cell->centre_frequencies); |
| g_slice_free (GstMpegtsT2DeliverySystemCell, cell); |
| } |
| |
| /** |
| * gst_mpegts_descriptor_parse_dvb_t2_delivery_system: |
| * @descriptor: a %GST_MTS_DESC_EXT_DVB_T2_DELIVERY_SYSTEM #GstMpegtsDescriptor |
| * @res: (out) (transfer full): #GstMpegtsT2DeliverySystemDescriptor |
| * |
| * Parses out the DVB-T2 delivery system from the @descriptor. |
| * |
| * Returns: %TRUE if the parsing happened correctly, else %FALSE. |
| */ |
| gboolean |
| gst_mpegts_descriptor_parse_dvb_t2_delivery_system (const GstMpegtsDescriptor |
| * descriptor, GstMpegtsT2DeliverySystemDescriptor ** desc) |
| { |
| guint8 *data; |
| guint8 len, freq_len, sub_cell_len; |
| guint32 tmp_freq; |
| guint8 i; |
| GstMpegtsT2DeliverySystemDescriptor *res; |
| |
| g_return_val_if_fail (descriptor != NULL && desc != NULL, FALSE); |
| __common_desc_ext_checks (descriptor, GST_MTS_DESC_EXT_DVB_T2_DELIVERY_SYSTEM, |
| 4, FALSE); |
| |
| data = (guint8 *) descriptor->data + 3; |
| |
| res = g_slice_new0 (GstMpegtsT2DeliverySystemDescriptor); |
| |
| res->plp_id = *data; |
| data += 1; |
| |
| res->t2_system_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| |
| if (descriptor->length > 4) { |
| // FIXME: siso / miso |
| res->siso_miso = (*data >> 6) & 0x03; |
| switch ((*data >> 2) & 0x0f) { |
| case 0: |
| res->bandwidth = 8000000; |
| break; |
| case 1: |
| res->bandwidth = 7000000; |
| break; |
| case 2: |
| res->bandwidth = 6000000; |
| break; |
| case 3: |
| res->bandwidth = 5000000; |
| break; |
| case 4: |
| res->bandwidth = 10000000; |
| break; |
| case 5: |
| res->bandwidth = 1712000; |
| break; |
| default: |
| res->bandwidth = 0; |
| break; |
| } |
| data += 1; |
| |
| switch ((*data >> 5) & 0x07) { |
| case 0: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_1_32; |
| break; |
| case 1: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_1_16; |
| break; |
| case 2: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_1_8; |
| break; |
| case 3: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_1_4; |
| break; |
| case 4: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_1_128; |
| break; |
| case 5: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_19_128; |
| break; |
| case 6: |
| res->guard_interval = GST_MPEGTS_GUARD_INTERVAL_19_256; |
| break; |
| default: |
| break; |
| } |
| |
| switch ((*data >> 2) & 0x07) { |
| case 0: |
| res->transmission_mode = GST_MPEGTS_TRANSMISSION_MODE_2K; |
| break; |
| case 1: |
| res->transmission_mode = GST_MPEGTS_TRANSMISSION_MODE_8K; |
| break; |
| case 2: |
| res->transmission_mode = GST_MPEGTS_TRANSMISSION_MODE_4K; |
| break; |
| case 3: |
| res->transmission_mode = GST_MPEGTS_TRANSMISSION_MODE_1K; |
| break; |
| case 4: |
| res->transmission_mode = GST_MPEGTS_TRANSMISSION_MODE_16K; |
| break; |
| case 5: |
| res->transmission_mode = GST_MPEGTS_TRANSMISSION_MODE_32K; |
| break; |
| default: |
| break; |
| } |
| res->other_frequency = (*data >> 1) & 0x01; |
| res->tfs = (*data) & 0x01; |
| data += 1; |
| |
| len = descriptor->length - 6; |
| |
| res->cells = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_t2_delivery_system_cell_free); |
| |
| for (i = 0; i < len;) { |
| GstMpegtsT2DeliverySystemCell *cell; |
| guint8 j, k; |
| |
| cell = g_slice_new0 (GstMpegtsT2DeliverySystemCell); |
| g_ptr_array_add (res->cells, cell); |
| |
| cell->cell_id = GST_READ_UINT16_BE (data); |
| data += 2; |
| i += 2; |
| |
| cell->centre_frequencies = g_array_new (FALSE, FALSE, sizeof (guint32)); |
| |
| if (res->tfs == TRUE) { |
| freq_len = *data; |
| data += 1; |
| i += 1; |
| |
| for (j = 0; j < freq_len;) { |
| tmp_freq = GST_READ_UINT32_BE (data) * 10; |
| g_array_append_val (cell->centre_frequencies, tmp_freq); |
| data += 4; |
| j += 4; |
| i += 4; |
| } |
| } else { |
| tmp_freq = GST_READ_UINT32_BE (data) * 10; |
| g_array_append_val (cell->centre_frequencies, tmp_freq); |
| data += 4; |
| i += 4; |
| } |
| sub_cell_len = (*data); |
| data += 1; |
| i += 1; |
| |
| cell->sub_cells = g_ptr_array_new_with_free_func ((GDestroyNotify) |
| _gst_mpegts_t2_delivery_system_cell_extension_free); |
| |
| for (k = 0; k < sub_cell_len;) { |
| GstMpegtsT2DeliverySystemCellExtension *cell_ext; |
| cell_ext = g_slice_new0 (GstMpegtsT2DeliverySystemCellExtension); |
| |
| g_ptr_array_add (cell->sub_cells, cell_ext); |
| cell_ext->cell_id_extension = *data; |
| data += 1; |
| |
| cell_ext->transposer_frequency = GST_READ_UINT32_BE (data) * 10; |
| data += 4; |
| i += 5; |
| k += 5; |
| } |
| } |
| } |
| |
| *desc = res; |
| return TRUE; |
| } |