mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-06 23:45:35 +00:00
b6411ae74c
For each lib we build export its own API in headers when we're building it, otherwise import the API from the headers. This fixes linker warnings on Windows when building with MSVC. The problem was that we had defined all GST_*_API decorators unconditionally to GST_EXPORT. This was intentional and only supposed to be temporary, but caused linker warnings because we tell the linker that we want to export all symbols even those from externall DLLs, and when the linker notices that they were in external DLLS and not present locally it warns. What we need to do when building each library is: export the library's own symbols and import all other symbols. To this end we define e.g. BUILDING_GST_FOO and then we define the GST_FOO_API decorator either to export or to import symbols depending on whether BUILDING_GST_FOO is set or not. That way external users of each library API automatically get the import. While we're at it, add new GST_API_EXPORT in config.h and use that for GST_*_API decorators instead of GST_EXPORT. The right export define depends on the toolchain and whether we're using -fvisibility=hidden or not, so it's better to set it to the right thing directly than hard-coding a compiler whitelist in the public header. We put the export define into config.h instead of passing it via the command line to the compiler because it might contain spaces and brackets and in the autotools scenario we'd have to pass that through multiple layers of plumbing and Makefile/shell escaping and we're just not going to be *that* lucky. The export define is only used if we're compiling our lib, not by external users of the lib headers, so it's not a problem to put it into config.h Also, this means all .c files of libs need to include config.h to get the export marker defined, so fix up a few that didn't include config.h. This commit depends on a common submodule commit that makes gst-glib-gen.mak add an #include "config.h" to generated enum/marshal .c files for the autotools build. https://bugzilla.gnome.org/show_bug.cgi?id=797185
1157 lines
30 KiB
C
1157 lines
30 KiB
C
/*
|
|
* 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.
|
|
*/
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "mpegts.h"
|
|
#include "gstmpegts-private.h"
|
|
|
|
/**
|
|
* SECTION:gstmpegtsdescriptor
|
|
* @title: Base MPEG-TS descriptors
|
|
* @short_description: Descriptors for ITU H.222.0 | ISO/IEC 13818-1
|
|
* @include: gst/mpegts/mpegts.h
|
|
*
|
|
* These are the base descriptor types and methods.
|
|
*
|
|
* For more details, refer to the ITU H.222.0 or ISO/IEC 13818-1 specifications
|
|
* and other specifications mentionned in the documentation.
|
|
*/
|
|
|
|
/* FIXME : Move this to proper file once we have a C file for ATSC/ISDB descriptors */
|
|
/**
|
|
* SECTION:gst-atsc-descriptor
|
|
* @title: ATSC variants of MPEG-TS descriptors
|
|
* @short_description: Descriptors for the various ATSC specifications
|
|
* @include: gst/mpegts/mpegts.h
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* SECTION:gst-isdb-descriptor
|
|
* @title: ISDB variants of MPEG-TS descriptors
|
|
* @short_description: Descriptors for the various ISDB 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
|
|
* * GST_MTS_DESC_DVB_FREQUENCY_LIST
|
|
*/
|
|
|
|
#define MAX_KNOWN_ICONV 25
|
|
|
|
/* First column is the original encoding,
|
|
* second column is the target encoding */
|
|
|
|
static GIConv __iconvs[MAX_KNOWN_ICONV][MAX_KNOWN_ICONV];
|
|
|
|
/* All these conversions will be to UTF8 */
|
|
typedef enum
|
|
{
|
|
_ICONV_UNKNOWN = -1,
|
|
_ICONV_ISO8859_1,
|
|
_ICONV_ISO8859_2,
|
|
_ICONV_ISO8859_3,
|
|
_ICONV_ISO8859_4,
|
|
_ICONV_ISO8859_5,
|
|
_ICONV_ISO8859_6,
|
|
_ICONV_ISO8859_7,
|
|
_ICONV_ISO8859_8,
|
|
_ICONV_ISO8859_9,
|
|
_ICONV_ISO8859_10,
|
|
_ICONV_ISO8859_11,
|
|
_ICONV_ISO8859_12,
|
|
_ICONV_ISO8859_13,
|
|
_ICONV_ISO8859_14,
|
|
_ICONV_ISO8859_15,
|
|
_ICONV_UCS_2BE,
|
|
_ICONV_EUC_KR,
|
|
_ICONV_GB2312,
|
|
_ICONV_UTF_16BE,
|
|
_ICONV_ISO10646_UTF8,
|
|
_ICONV_ISO6937,
|
|
_ICONV_UTF8,
|
|
/* Insert more here if needed */
|
|
_ICONV_MAX
|
|
} LocalIconvCode;
|
|
|
|
static const gchar *iconvtablename[] = {
|
|
"iso-8859-1",
|
|
"iso-8859-2",
|
|
"iso-8859-3",
|
|
"iso-8859-4",
|
|
"iso-8859-5",
|
|
"iso-8859-6",
|
|
"iso-8859-7",
|
|
"iso-8859-8",
|
|
"iso-8859-9",
|
|
"iso-8859-10",
|
|
"iso-8859-11",
|
|
"iso-8859-12",
|
|
"iso-8859-13",
|
|
"iso-8859-14",
|
|
"iso-8859-15",
|
|
"UCS-2BE",
|
|
"EUC-KR",
|
|
"GB2312",
|
|
"UTF-16BE",
|
|
"ISO-10646/UTF8",
|
|
"iso6937",
|
|
"utf-8"
|
|
/* Insert more here if needed */
|
|
};
|
|
|
|
void
|
|
__initialize_descriptors (void)
|
|
{
|
|
guint i, j;
|
|
|
|
/* Initialize converters */
|
|
/* FIXME : How/when should we close them ??? */
|
|
for (i = 0; i < MAX_KNOWN_ICONV; i++) {
|
|
for (j = 0; j < MAX_KNOWN_ICONV; j++)
|
|
__iconvs[i][j] = ((GIConv) - 1);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* @text: The text you want to get the encoding from
|
|
* @start_text: Location where the beginning of the actual text is stored
|
|
* @is_multibyte: Location where information whether it's a multibyte encoding
|
|
* or not is stored
|
|
* @returns: GIconv for conversion or NULL
|
|
*/
|
|
static LocalIconvCode
|
|
get_encoding (const gchar * text, guint * start_text, gboolean * is_multibyte)
|
|
{
|
|
LocalIconvCode encoding;
|
|
guint8 firstbyte;
|
|
|
|
*is_multibyte = FALSE;
|
|
*start_text = 0;
|
|
|
|
firstbyte = (guint8) text[0];
|
|
|
|
/* A wrong value */
|
|
g_return_val_if_fail (firstbyte != 0x00, _ICONV_UNKNOWN);
|
|
|
|
if (firstbyte <= 0x0B) {
|
|
/* 0x01 => iso 8859-5 */
|
|
encoding = firstbyte + _ICONV_ISO8859_4;
|
|
*start_text = 1;
|
|
goto beach;
|
|
}
|
|
|
|
/* ETSI EN 300 468, "Selection of character table" */
|
|
switch (firstbyte) {
|
|
case 0x0C:
|
|
case 0x0D:
|
|
case 0x0E:
|
|
case 0x0F:
|
|
/* RESERVED */
|
|
encoding = _ICONV_UNKNOWN;
|
|
break;
|
|
case 0x10:
|
|
{
|
|
guint16 table;
|
|
|
|
table = GST_READ_UINT16_BE (text + 1);
|
|
|
|
if (table < 17)
|
|
encoding = _ICONV_UNKNOWN + table;
|
|
else
|
|
encoding = _ICONV_UNKNOWN;
|
|
*start_text = 3;
|
|
break;
|
|
}
|
|
case 0x11:
|
|
encoding = _ICONV_UCS_2BE;
|
|
*start_text = 1;
|
|
*is_multibyte = TRUE;
|
|
break;
|
|
case 0x12:
|
|
/* EUC-KR implements KSX1001 */
|
|
encoding = _ICONV_EUC_KR;
|
|
*start_text = 1;
|
|
*is_multibyte = TRUE;
|
|
break;
|
|
case 0x13:
|
|
encoding = _ICONV_GB2312;
|
|
*start_text = 1;
|
|
break;
|
|
case 0x14:
|
|
encoding = _ICONV_UTF_16BE;
|
|
*start_text = 1;
|
|
*is_multibyte = TRUE;
|
|
break;
|
|
case 0x15:
|
|
/* TODO : Where does this come from ?? */
|
|
encoding = _ICONV_ISO10646_UTF8;
|
|
*start_text = 1;
|
|
break;
|
|
case 0x16:
|
|
case 0x17:
|
|
case 0x18:
|
|
case 0x19:
|
|
case 0x1A:
|
|
case 0x1B:
|
|
case 0x1C:
|
|
case 0x1D:
|
|
case 0x1E:
|
|
case 0x1F:
|
|
/* RESERVED */
|
|
encoding = _ICONV_UNKNOWN;
|
|
break;
|
|
default:
|
|
encoding = _ICONV_ISO6937;
|
|
break;
|
|
}
|
|
|
|
beach:
|
|
GST_DEBUG
|
|
("Found encoding %d, first byte is 0x%02x, start_text: %u, is_multibyte: %d",
|
|
encoding, firstbyte, *start_text, *is_multibyte);
|
|
|
|
return encoding;
|
|
}
|
|
|
|
static GIConv
|
|
_get_iconv (LocalIconvCode from, LocalIconvCode to)
|
|
{
|
|
if (__iconvs[from][to] == (GIConv) - 1)
|
|
__iconvs[from][to] = g_iconv_open (iconvtablename[to],
|
|
iconvtablename[from]);
|
|
return __iconvs[from][to];
|
|
}
|
|
|
|
static void
|
|
_encode_control_codes (gchar * text, gsize length, gboolean is_multibyte)
|
|
{
|
|
gsize pos = 0;
|
|
|
|
while (pos < length) {
|
|
if (is_multibyte) {
|
|
guint16 code = GST_READ_UINT16_BE (text + pos);
|
|
if (code == 0x000A) {
|
|
text[pos] = 0xE0;
|
|
text[pos + 1] = 0x8A;
|
|
}
|
|
pos += 2;
|
|
} else {
|
|
guint8 code = text[pos];
|
|
if (code == 0x0A)
|
|
text[pos] = 0x8A;
|
|
pos++;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* dvb_text_from_utf8:
|
|
* @text: The text to convert. This should be in UTF-8 format
|
|
* @out_size: (out): the byte length of the new text
|
|
*
|
|
* Converts UTF-8 strings to text characters compliant with EN 300 468.
|
|
* The converted text can be used directly in DVB #GstMpegtsDescriptor
|
|
*
|
|
* The function will try different character maps until the string is
|
|
* completely converted.
|
|
*
|
|
* The function tries the default ISO 6937 character map first.
|
|
*
|
|
* If no character map that contains all characters could be found, the
|
|
* string is converted to ISO 6937 with unknown characters set to `?`.
|
|
*
|
|
* Returns: (transfer full): byte array of size @out_size
|
|
*/
|
|
guint8 *
|
|
dvb_text_from_utf8 (const gchar * text, gsize * out_size)
|
|
{
|
|
GError *error = NULL;
|
|
gchar *out_text;
|
|
guint8 *out_buffer;
|
|
guint encoding;
|
|
GIConv giconv = (GIConv) - 1;
|
|
|
|
/* We test character maps one-by-one. Start with the default */
|
|
encoding = _ICONV_ISO6937;
|
|
giconv = _get_iconv (_ICONV_UTF8, encoding);
|
|
out_text = g_convert_with_iconv (text, -1, giconv, NULL, out_size, &error);
|
|
|
|
if (out_text) {
|
|
GST_DEBUG ("Using default ISO6937 encoding");
|
|
goto out;
|
|
}
|
|
|
|
g_clear_error (&error);
|
|
|
|
for (encoding = _ICONV_ISO8859_1; encoding <= _ICONV_ISO10646_UTF8;
|
|
encoding++) {
|
|
giconv = _get_iconv (_ICONV_UTF8, encoding);
|
|
if (giconv == (GIConv) - 1)
|
|
continue;
|
|
out_text = g_convert_with_iconv (text, -1, giconv, NULL, out_size, &error);
|
|
|
|
if (out_text) {
|
|
GST_DEBUG ("Found suitable character map - %s", iconvtablename[encoding]);
|
|
goto out;
|
|
}
|
|
|
|
g_clear_error (&error);
|
|
}
|
|
|
|
out_text = g_convert_with_fallback (text, -1, iconvtablename[_ICONV_ISO6937],
|
|
iconvtablename[_ICONV_UTF8], "?", NULL, out_size, &error);
|
|
|
|
out:
|
|
|
|
if (error) {
|
|
GST_WARNING ("Could not convert from utf-8: %s", error->message);
|
|
g_error_free (error);
|
|
g_free (out_text);
|
|
return NULL;
|
|
}
|
|
|
|
switch (encoding) {
|
|
case _ICONV_ISO6937:
|
|
/* Default encoding contains no selection bytes. */
|
|
_encode_control_codes (out_text, *out_size, FALSE);
|
|
return (guint8 *) out_text;
|
|
case _ICONV_ISO8859_1:
|
|
case _ICONV_ISO8859_2:
|
|
case _ICONV_ISO8859_3:
|
|
case _ICONV_ISO8859_4:
|
|
/* These character sets requires 3 selection bytes */
|
|
_encode_control_codes (out_text, *out_size, FALSE);
|
|
out_buffer = g_malloc (*out_size + 3);
|
|
out_buffer[0] = 0x10;
|
|
out_buffer[1] = 0x00;
|
|
out_buffer[2] = encoding - _ICONV_ISO8859_1 + 1;
|
|
memcpy (out_buffer + 3, out_text, *out_size);
|
|
*out_size += 3;
|
|
g_free (out_text);
|
|
return out_buffer;
|
|
case _ICONV_ISO8859_5:
|
|
case _ICONV_ISO8859_6:
|
|
case _ICONV_ISO8859_7:
|
|
case _ICONV_ISO8859_8:
|
|
case _ICONV_ISO8859_9:
|
|
case _ICONV_ISO8859_10:
|
|
case _ICONV_ISO8859_11:
|
|
case _ICONV_ISO8859_12:
|
|
case _ICONV_ISO8859_13:
|
|
case _ICONV_ISO8859_14:
|
|
case _ICONV_ISO8859_15:
|
|
/* These character sets requires 1 selection byte */
|
|
_encode_control_codes (out_text, *out_size, FALSE);
|
|
out_buffer = g_malloc (*out_size + 1);
|
|
out_buffer[0] = encoding - _ICONV_ISO8859_5 + 1;
|
|
memcpy (out_buffer + 1, out_text, *out_size);
|
|
*out_size += 1;
|
|
g_free (out_text);
|
|
return out_buffer;
|
|
case _ICONV_UCS_2BE:
|
|
case _ICONV_EUC_KR:
|
|
case _ICONV_UTF_16BE:
|
|
/* These character sets requires 1 selection byte */
|
|
_encode_control_codes (out_text, *out_size, TRUE);
|
|
out_buffer = g_malloc (*out_size + 1);
|
|
out_buffer[0] = encoding - _ICONV_UCS_2BE + 0x11;
|
|
memcpy (out_buffer + 1, out_text, *out_size);
|
|
*out_size += 1;
|
|
g_free (out_text);
|
|
return out_buffer;
|
|
case _ICONV_GB2312:
|
|
case _ICONV_ISO10646_UTF8:
|
|
/* These character sets requires 1 selection byte */
|
|
_encode_control_codes (out_text, *out_size, FALSE);
|
|
out_buffer = g_malloc (*out_size + 1);
|
|
out_buffer[0] = encoding - _ICONV_UCS_2BE + 0x11;
|
|
memcpy (out_buffer + 1, out_text, *out_size);
|
|
*out_size += 1;
|
|
g_free (out_text);
|
|
return out_buffer;
|
|
default:
|
|
g_free (out_text);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* @text: The text to convert. It may include pango markup (<b> and </b>)
|
|
* @length: The length of the string -1 if it's nul-terminated
|
|
* @start: Where to start converting in the text
|
|
* @encoding: The encoding of text
|
|
* @is_multibyte: Whether the encoding is a multibyte encoding
|
|
* @error: The location to store the error, or NULL to ignore errors
|
|
* @returns: UTF-8 encoded string
|
|
*
|
|
* Convert text to UTF-8.
|
|
*/
|
|
static gchar *
|
|
convert_to_utf8 (const gchar * text, gint length, guint start,
|
|
GIConv giconv, gboolean is_multibyte, GError ** error)
|
|
{
|
|
gchar *new_text;
|
|
gchar *tmp, *pos;
|
|
gint i;
|
|
|
|
text += start;
|
|
|
|
pos = tmp = g_malloc (length * 2);
|
|
|
|
if (is_multibyte) {
|
|
if (length == -1) {
|
|
while (*text != '\0') {
|
|
guint16 code = GST_READ_UINT16_BE (text);
|
|
|
|
switch (code) {
|
|
case 0xE086: /* emphasis on */
|
|
case 0xE087: /* emphasis off */
|
|
/* skip it */
|
|
break;
|
|
case 0xE08A:{
|
|
pos[0] = 0x00; /* 0x00 0x0A is a new line */
|
|
pos[1] = 0x0A;
|
|
pos += 2;
|
|
break;
|
|
}
|
|
default:
|
|
pos[0] = text[0];
|
|
pos[1] = text[1];
|
|
pos += 2;
|
|
break;
|
|
}
|
|
|
|
text += 2;
|
|
}
|
|
} else {
|
|
for (i = 0; i < length; i += 2) {
|
|
guint16 code = GST_READ_UINT16_BE (text);
|
|
|
|
switch (code) {
|
|
case 0xE086: /* emphasis on */
|
|
case 0xE087: /* emphasis off */
|
|
/* skip it */
|
|
break;
|
|
case 0xE08A:{
|
|
pos[0] = 0x00; /* 0x00 0x0A is a new line */
|
|
pos[1] = 0x0A;
|
|
pos += 2;
|
|
break;
|
|
}
|
|
default:
|
|
pos[0] = text[0];
|
|
pos[1] = text[1];
|
|
pos += 2;
|
|
break;
|
|
}
|
|
|
|
text += 2;
|
|
}
|
|
}
|
|
} else {
|
|
if (length == -1) {
|
|
while (*text != '\0') {
|
|
guint8 code = (guint8) (*text);
|
|
|
|
switch (code) {
|
|
case 0x86: /* emphasis on */
|
|
case 0x87: /* emphasis off */
|
|
/* skip it */
|
|
break;
|
|
case 0x8A:
|
|
*pos = '\n';
|
|
pos += 1;
|
|
break;
|
|
default:
|
|
*pos = *text;
|
|
pos += 1;
|
|
break;
|
|
}
|
|
|
|
text++;
|
|
}
|
|
} else {
|
|
for (i = 0; i < length; i++) {
|
|
guint8 code = (guint8) (*text);
|
|
|
|
switch (code) {
|
|
case 0x86: /* emphasis on */
|
|
case 0x87: /* emphasis off */
|
|
/* skip it */
|
|
break;
|
|
case 0x8A:
|
|
*pos = '\n';
|
|
pos += 1;
|
|
break;
|
|
default:
|
|
*pos = *text;
|
|
pos += 1;
|
|
break;
|
|
}
|
|
|
|
text++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pos > tmp) {
|
|
gsize bread = 0;
|
|
new_text =
|
|
g_convert_with_iconv (tmp, pos - tmp, giconv, &bread, NULL, error);
|
|
GST_DEBUG ("Converted to : %s", new_text);
|
|
} else {
|
|
new_text = g_strdup ("");
|
|
}
|
|
|
|
g_free (tmp);
|
|
|
|
return new_text;
|
|
}
|
|
|
|
gchar *
|
|
get_encoding_and_convert (const gchar * text, guint length)
|
|
{
|
|
GError *error = NULL;
|
|
gchar *converted_str;
|
|
guint start_text = 0;
|
|
gboolean is_multibyte;
|
|
LocalIconvCode encoding;
|
|
GIConv giconv = (GIConv) - 1;
|
|
|
|
g_return_val_if_fail (text != NULL, NULL);
|
|
|
|
if (text == NULL || length == 0)
|
|
return g_strdup ("");
|
|
|
|
encoding = get_encoding (text, &start_text, &is_multibyte);
|
|
|
|
if (encoding > _ICONV_UNKNOWN && encoding < _ICONV_MAX) {
|
|
GST_DEBUG ("Encoding %s", iconvtablename[encoding]);
|
|
giconv = _get_iconv (encoding, _ICONV_UTF8);
|
|
} else {
|
|
GST_FIXME ("Could not detect encoding. Returning NULL string");
|
|
converted_str = NULL;
|
|
goto beach;
|
|
}
|
|
|
|
converted_str = convert_to_utf8 (text, length - start_text, start_text,
|
|
giconv, is_multibyte, &error);
|
|
if (error != NULL) {
|
|
GST_WARNING ("Could not convert string: %s", error->message);
|
|
g_free (converted_str);
|
|
g_error_free (error);
|
|
error = NULL;
|
|
|
|
if (encoding >= _ICONV_ISO8859_2 && encoding <= _ICONV_ISO8859_15) {
|
|
/* Sometimes using the standard 8859-1 set fixes issues */
|
|
GST_DEBUG ("Encoding %s", iconvtablename[_ICONV_ISO8859_1]);
|
|
giconv = _get_iconv (_ICONV_ISO8859_1, _ICONV_UTF8);
|
|
|
|
GST_INFO ("Trying encoding ISO 8859-1");
|
|
converted_str = convert_to_utf8 (text, length, 1, giconv, FALSE, &error);
|
|
if (error != NULL) {
|
|
GST_WARNING
|
|
("Could not convert string while assuming encoding ISO 8859-1: %s",
|
|
error->message);
|
|
g_error_free (error);
|
|
goto failed;
|
|
}
|
|
} else if (encoding == _ICONV_ISO6937) {
|
|
|
|
/* The first part of ISO 6937 is identical to ISO 8859-9, but
|
|
* they differ in the second part. Some channels don't
|
|
* provide the first byte that indicates ISO 8859-9 encoding.
|
|
* If decoding from ISO 6937 failed, we try ISO 8859-9 here.
|
|
*/
|
|
giconv = _get_iconv (_ICONV_ISO8859_9, _ICONV_UTF8);
|
|
|
|
GST_INFO ("Trying encoding ISO 8859-9");
|
|
converted_str = convert_to_utf8 (text, length, 0, giconv, FALSE, &error);
|
|
if (error != NULL) {
|
|
GST_WARNING
|
|
("Could not convert string while assuming encoding ISO 8859-9: %s",
|
|
error->message);
|
|
g_error_free (error);
|
|
goto failed;
|
|
}
|
|
} else
|
|
goto failed;
|
|
}
|
|
|
|
beach:
|
|
return converted_str;
|
|
|
|
failed:
|
|
{
|
|
text += start_text;
|
|
return g_strndup (text, length - start_text);
|
|
}
|
|
}
|
|
|
|
gchar *
|
|
convert_lang_code (guint8 * data)
|
|
{
|
|
gchar *code;
|
|
/* the iso language code and country code is always 3 byte long */
|
|
code = g_malloc0 (4);
|
|
memcpy (code, data, 3);
|
|
|
|
return code;
|
|
}
|
|
|
|
void
|
|
_packetize_descriptor_array (GPtrArray * array, guint8 ** out_data)
|
|
{
|
|
guint i;
|
|
GstMpegtsDescriptor *descriptor;
|
|
|
|
g_return_if_fail (out_data != NULL);
|
|
g_return_if_fail (*out_data != NULL);
|
|
|
|
if (array == NULL)
|
|
return;
|
|
|
|
for (i = 0; i < array->len; i++) {
|
|
descriptor = g_ptr_array_index (array, i);
|
|
|
|
memcpy (*out_data, descriptor->data, descriptor->length + 2);
|
|
*out_data += descriptor->length + 2;
|
|
}
|
|
}
|
|
|
|
GstMpegtsDescriptor *
|
|
_new_descriptor (guint8 tag, guint8 length)
|
|
{
|
|
GstMpegtsDescriptor *descriptor;
|
|
guint8 *data;
|
|
|
|
descriptor = g_slice_new (GstMpegtsDescriptor);
|
|
|
|
descriptor->tag = tag;
|
|
descriptor->tag_extension = 0;
|
|
descriptor->length = length;
|
|
|
|
descriptor->data = g_malloc (length + 2);
|
|
|
|
data = descriptor->data;
|
|
|
|
*data++ = descriptor->tag;
|
|
*data = descriptor->length;
|
|
|
|
return descriptor;
|
|
}
|
|
|
|
GstMpegtsDescriptor *
|
|
_new_descriptor_with_extension (guint8 tag, guint8 tag_extension, guint8 length)
|
|
{
|
|
GstMpegtsDescriptor *descriptor;
|
|
guint8 *data;
|
|
|
|
descriptor = g_slice_new (GstMpegtsDescriptor);
|
|
|
|
descriptor->tag = tag;
|
|
descriptor->tag_extension = tag_extension;
|
|
descriptor->length = length + 1;
|
|
|
|
descriptor->data = g_malloc (length + 3);
|
|
|
|
data = descriptor->data;
|
|
|
|
*data++ = descriptor->tag;
|
|
*data++ = descriptor->length;
|
|
*data = descriptor->tag_extension;
|
|
|
|
return descriptor;
|
|
}
|
|
|
|
static GstMpegtsDescriptor *
|
|
_copy_descriptor (GstMpegtsDescriptor * desc)
|
|
{
|
|
GstMpegtsDescriptor *copy;
|
|
|
|
copy = g_slice_dup (GstMpegtsDescriptor, desc);
|
|
copy->data = g_memdup (desc->data, desc->length + 2);
|
|
|
|
return copy;
|
|
}
|
|
|
|
/**
|
|
* gst_mpegts_descriptor_free:
|
|
* @desc: The descriptor to free
|
|
*
|
|
* Frees @desc
|
|
*/
|
|
void
|
|
gst_mpegts_descriptor_free (GstMpegtsDescriptor * desc)
|
|
{
|
|
g_free ((gpointer) desc->data);
|
|
g_slice_free (GstMpegtsDescriptor, desc);
|
|
}
|
|
|
|
G_DEFINE_BOXED_TYPE (GstMpegtsDescriptor, gst_mpegts_descriptor,
|
|
(GBoxedCopyFunc) _copy_descriptor,
|
|
(GBoxedFreeFunc) gst_mpegts_descriptor_free);
|
|
|
|
/**
|
|
* gst_mpegts_parse_descriptors:
|
|
* @buffer: (transfer none): descriptors to parse
|
|
* @buf_len: Size of @buffer
|
|
*
|
|
* Parses the descriptors present in @buffer and returns them as an
|
|
* array.
|
|
*
|
|
* Note: The data provided in @buffer will not be copied.
|
|
*
|
|
* Returns: (transfer full) (element-type GstMpegtsDescriptor): an
|
|
* array of the parsed descriptors or %NULL if there was an error.
|
|
* Release with #g_array_unref when done with it.
|
|
*/
|
|
GPtrArray *
|
|
gst_mpegts_parse_descriptors (guint8 * buffer, gsize buf_len)
|
|
{
|
|
GPtrArray *res;
|
|
guint8 length;
|
|
guint8 *data;
|
|
guint i, nb_desc = 0;
|
|
|
|
/* fast-path */
|
|
if (buf_len == 0)
|
|
return g_ptr_array_new ();
|
|
|
|
data = buffer;
|
|
|
|
GST_MEMDUMP ("Full descriptor array", buffer, buf_len);
|
|
|
|
while (data - buffer < buf_len) {
|
|
data++; /* skip tag */
|
|
length = *data++;
|
|
|
|
if (data - buffer > buf_len) {
|
|
GST_WARNING ("invalid descriptor length %d now at %d max %"
|
|
G_GSIZE_FORMAT, length, (gint) (data - buffer), buf_len);
|
|
return NULL;
|
|
}
|
|
|
|
data += length;
|
|
nb_desc++;
|
|
}
|
|
|
|
GST_DEBUG ("Saw %d descriptors, read %" G_GSIZE_FORMAT " bytes",
|
|
nb_desc, (gsize) (data - buffer));
|
|
|
|
if (data - buffer != buf_len) {
|
|
GST_WARNING ("descriptors size %d expected %" G_GSIZE_FORMAT,
|
|
(gint) (data - buffer), buf_len);
|
|
return NULL;
|
|
}
|
|
|
|
res =
|
|
g_ptr_array_new_full (nb_desc + 1,
|
|
(GDestroyNotify) gst_mpegts_descriptor_free);
|
|
|
|
data = buffer;
|
|
|
|
for (i = 0; i < nb_desc; i++) {
|
|
GstMpegtsDescriptor *desc = g_slice_new0 (GstMpegtsDescriptor);
|
|
|
|
desc->data = data;
|
|
desc->tag = *data++;
|
|
desc->length = *data++;
|
|
/* Copy the data now that we known the size */
|
|
desc->data = g_memdup (desc->data, desc->length + 2);
|
|
GST_LOG ("descriptor 0x%02x length:%d", desc->tag, desc->length);
|
|
GST_MEMDUMP ("descriptor", desc->data + 2, desc->length);
|
|
/* extended descriptors */
|
|
if (G_UNLIKELY (desc->tag == 0x7f))
|
|
desc->tag_extension = *data;
|
|
|
|
data += desc->length;
|
|
|
|
/* Set the descriptor in the array */
|
|
g_ptr_array_index (res, i) = desc;
|
|
}
|
|
|
|
res->len = nb_desc;
|
|
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* gst_mpegts_find_descriptor:
|
|
* @descriptors: (element-type GstMpegtsDescriptor) (transfer none): an array
|
|
* of #GstMpegtsDescriptor
|
|
* @tag: the tag to look for
|
|
*
|
|
* Finds the first descriptor of type @tag in the array.
|
|
*
|
|
* Note: To look for descriptors that can be present more than once in an
|
|
* array of descriptors, iterate the #GArray manually.
|
|
*
|
|
* Returns: (transfer none): the first descriptor matchin @tag, else %NULL.
|
|
*/
|
|
const GstMpegtsDescriptor *
|
|
gst_mpegts_find_descriptor (GPtrArray * descriptors, guint8 tag)
|
|
{
|
|
guint i, nb_desc;
|
|
|
|
g_return_val_if_fail (descriptors != NULL, NULL);
|
|
|
|
nb_desc = descriptors->len;
|
|
for (i = 0; i < nb_desc; i++) {
|
|
GstMpegtsDescriptor *desc = g_ptr_array_index (descriptors, i);
|
|
if (desc->tag == tag)
|
|
return (const GstMpegtsDescriptor *) desc;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/* GST_MTS_DESC_REGISTRATION (0x05) */
|
|
/**
|
|
* gst_mpegts_descriptor_from_registration:
|
|
* @format_identifier: (transfer none): a 4 character format identifier string
|
|
* @additional_info: (transfer none) (allow-none) (array length=additional_info_length): pointer to optional additional info
|
|
* @additional_info_length: length of the optional @additional_info
|
|
*
|
|
* Creates a %GST_MTS_DESC_REGISTRATION #GstMpegtsDescriptor
|
|
*
|
|
* Return: #GstMpegtsDescriptor, %NULL on failure
|
|
*/
|
|
GstMpegtsDescriptor *
|
|
gst_mpegts_descriptor_from_registration (const gchar * format_identifier,
|
|
guint8 * additional_info, gsize additional_info_length)
|
|
{
|
|
GstMpegtsDescriptor *descriptor;
|
|
|
|
g_return_val_if_fail (format_identifier != NULL, NULL);
|
|
g_return_val_if_fail (additional_info_length > 0 || !additional_info, NULL);
|
|
|
|
descriptor = _new_descriptor (GST_MTS_DESC_REGISTRATION,
|
|
4 + additional_info_length);
|
|
|
|
memcpy (descriptor->data + 2, format_identifier, 4);
|
|
if (additional_info && (additional_info_length > 0))
|
|
memcpy (descriptor->data + 6, additional_info, additional_info_length);
|
|
|
|
return descriptor;
|
|
}
|
|
|
|
/* GST_MTS_DESC_CA (0x09) */
|
|
|
|
/**
|
|
* gst_mpegts_descriptor_parse_ca:
|
|
* @descriptor: a %GST_MTS_DESC_CA #GstMpegtsDescriptor
|
|
* @ca_system_id: (out): the type of CA system used
|
|
* @ca_pid: (out): The PID containing ECM or EMM data
|
|
* @private_data: (out) (allow-none) (array length=private_data_size): The private data
|
|
* @private_data_size: (out) (allow-none): The size of @private_data in bytes
|
|
*
|
|
* Extracts the Conditional Access information from @descriptor.
|
|
*
|
|
* Returns: %TRUE if parsing succeeded, else %FALSE.
|
|
*/
|
|
|
|
gboolean
|
|
gst_mpegts_descriptor_parse_ca (GstMpegtsDescriptor * descriptor,
|
|
guint16 * ca_system_id, guint16 * ca_pid,
|
|
const guint8 ** private_data, gsize * private_data_size)
|
|
{
|
|
guint8 *data;
|
|
|
|
g_return_val_if_fail (descriptor != NULL && ca_system_id != NULL
|
|
&& ca_pid != NULL, FALSE);
|
|
/* The smallest CA is 4 bytes (though not having any private data
|
|
* sounds a bit ... weird) */
|
|
__common_desc_checks (descriptor, GST_MTS_DESC_CA, 4, FALSE);
|
|
|
|
data = (guint8 *) descriptor->data + 2;
|
|
*ca_system_id = GST_READ_UINT16_BE (data);
|
|
data += 2;
|
|
*ca_pid = GST_READ_UINT16_BE (data) & 0x1fff;
|
|
data += 2;
|
|
if (private_data && private_data_size) {
|
|
*private_data = data;
|
|
*private_data_size = descriptor->length - 4;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* GST_MTS_DESC_ISO_639_LANGUAGE (0x0A) */
|
|
static GstMpegtsISO639LanguageDescriptor *
|
|
_gst_mpegts_iso_639_language_descriptor_copy (GstMpegtsISO639LanguageDescriptor
|
|
* source)
|
|
{
|
|
GstMpegtsISO639LanguageDescriptor *copy;
|
|
guint i;
|
|
|
|
copy = g_slice_dup (GstMpegtsISO639LanguageDescriptor, source);
|
|
|
|
for (i = 0; i < source->nb_language; i++) {
|
|
copy->language[i] = g_strdup (source->language[i]);
|
|
}
|
|
|
|
return copy;
|
|
}
|
|
|
|
void
|
|
gst_mpegts_iso_639_language_descriptor_free (GstMpegtsISO639LanguageDescriptor
|
|
* desc)
|
|
{
|
|
guint i;
|
|
|
|
for (i = 0; i < desc->nb_language; i++) {
|
|
g_free (desc->language[i]);
|
|
}
|
|
g_slice_free (GstMpegtsISO639LanguageDescriptor, desc);
|
|
}
|
|
|
|
G_DEFINE_BOXED_TYPE (GstMpegtsISO639LanguageDescriptor,
|
|
gst_mpegts_iso_639_language,
|
|
(GBoxedCopyFunc) _gst_mpegts_iso_639_language_descriptor_copy,
|
|
(GFreeFunc) gst_mpegts_iso_639_language_descriptor_free);
|
|
|
|
/**
|
|
* gst_mpegts_descriptor_parse_iso_639_language:
|
|
* @descriptor: a %GST_MTS_DESC_ISO_639_LANGUAGE #GstMpegtsDescriptor
|
|
* @res: (out) (transfer full): the #GstMpegtsISO639LanguageDescriptor to fill
|
|
*
|
|
* Extracts the iso 639-2 language information from @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_iso_639_language (const GstMpegtsDescriptor *
|
|
descriptor, GstMpegtsISO639LanguageDescriptor ** desc)
|
|
{
|
|
guint i;
|
|
guint8 *data;
|
|
GstMpegtsISO639LanguageDescriptor *res;
|
|
|
|
g_return_val_if_fail (descriptor != NULL && desc != NULL, FALSE);
|
|
/* This descriptor can be empty, no size check needed */
|
|
__common_desc_check_base (descriptor, GST_MTS_DESC_ISO_639_LANGUAGE, FALSE);
|
|
|
|
data = (guint8 *) descriptor->data + 2;
|
|
|
|
res = g_slice_new0 (GstMpegtsISO639LanguageDescriptor);
|
|
|
|
/* Each language is 3 + 1 bytes */
|
|
res->nb_language = descriptor->length / 4;
|
|
for (i = 0; i < res->nb_language; i++) {
|
|
res->language[i] = convert_lang_code (data);
|
|
res->audio_type[i] = data[3];
|
|
data += 4;
|
|
}
|
|
|
|
*desc = res;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
/**
|
|
* gst_mpegts_descriptor_parse_iso_639_language_idx:
|
|
* @descriptor: a %GST_MTS_DESC_ISO_639_LANGUAGE #GstMpegtsDescriptor
|
|
* @idx: Table id of the language to parse
|
|
* @lang: (out) (transfer full): 4-byte gchar array to hold the language code
|
|
* @audio_type: (out) (transfer none) (allow-none): the #GstMpegtsIso639AudioType to set
|
|
*
|
|
* Extracts the iso 639-2 language information 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_iso_639_language_idx (const GstMpegtsDescriptor *
|
|
descriptor, guint idx, gchar ** lang, GstMpegtsIso639AudioType * audio_type)
|
|
{
|
|
guint8 *data;
|
|
|
|
g_return_val_if_fail (descriptor != NULL && lang != NULL, FALSE);
|
|
/* This descriptor can be empty, no size check needed */
|
|
__common_desc_check_base (descriptor, GST_MTS_DESC_ISO_639_LANGUAGE, FALSE);
|
|
|
|
if (descriptor->length / 4 <= idx)
|
|
return FALSE;
|
|
|
|
data = (guint8 *) descriptor->data + 2 + idx * 4;
|
|
|
|
*lang = convert_lang_code (data);
|
|
|
|
data += 3;
|
|
|
|
if (audio_type)
|
|
*audio_type = *data;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* gst_mpegts_descriptor_parse_iso_639_language_nb:
|
|
* @descriptor: a %GST_MTS_DESC_ISO_639_LANGUAGE #GstMpegtsDescriptor
|
|
*
|
|
* Returns: The number of languages in @descriptor
|
|
*/
|
|
guint
|
|
gst_mpegts_descriptor_parse_iso_639_language_nb (const GstMpegtsDescriptor *
|
|
descriptor)
|
|
{
|
|
g_return_val_if_fail (descriptor != NULL, 0);
|
|
/* This descriptor can be empty, no size check needed */
|
|
__common_desc_check_base (descriptor, GST_MTS_DESC_ISO_639_LANGUAGE, FALSE);
|
|
|
|
return descriptor->length / 4;
|
|
}
|
|
|
|
/**
|
|
* gst_mpegts_descriptor_from_iso_639_language:
|
|
* @language: (transfer none): ISO-639-2 language 3-char code
|
|
*
|
|
* Creates a %GST_MTS_DESC_ISO_639_LANGUAGE #GstMpegtsDescriptor with
|
|
* a single language
|
|
*
|
|
* Return: #GstMpegtsDescriptor, %NULL on failure
|
|
*/
|
|
GstMpegtsDescriptor *
|
|
gst_mpegts_descriptor_from_iso_639_language (const gchar * language)
|
|
{
|
|
GstMpegtsDescriptor *descriptor;
|
|
|
|
g_return_val_if_fail (language != NULL, NULL);
|
|
|
|
descriptor = _new_descriptor (GST_MTS_DESC_ISO_639_LANGUAGE, 4); /* a language takes 4 bytes */
|
|
|
|
memcpy (descriptor->data + 2, language, 3);
|
|
descriptor->data[2 + 3] = 0; /* set audio type to undefined */
|
|
|
|
return descriptor;
|
|
}
|
|
|
|
/**
|
|
* gst_mpegts_descriptor_parse_logical_channel:
|
|
* @descriptor: a %GST_MTS_DESC_DTG_LOGICAL_CHANNEL #GstMpegtsDescriptor
|
|
* @res: (out) (transfer none): the #GstMpegtsLogicalChannelDescriptor to fill
|
|
*
|
|
* Extracts the logical channels from @descriptor.
|
|
*
|
|
* Returns: %TRUE if parsing succeeded, else %FALSE.
|
|
*/
|
|
gboolean
|
|
gst_mpegts_descriptor_parse_logical_channel (const GstMpegtsDescriptor *
|
|
descriptor, GstMpegtsLogicalChannelDescriptor * res)
|
|
{
|
|
guint i;
|
|
guint8 *data;
|
|
|
|
g_return_val_if_fail (descriptor != NULL && res != NULL, FALSE);
|
|
/* This descriptor loop can be empty, no size check required */
|
|
__common_desc_check_base (descriptor, GST_MTS_DESC_DTG_LOGICAL_CHANNEL,
|
|
FALSE);
|
|
|
|
data = (guint8 *) descriptor->data + 2;
|
|
|
|
res->nb_channels = descriptor->length / 4;
|
|
|
|
for (i = 0; i < res->nb_channels; i++) {
|
|
res->channels[i].service_id = GST_READ_UINT16_BE (data);
|
|
data += 2;
|
|
res->channels[i].visible_service = *data >> 7;
|
|
res->channels[i].logical_channel_number =
|
|
GST_READ_UINT16_BE (data) & 0x03ff;
|
|
data += 2;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/**
|
|
* gst_mpegts_descriptor_from_custom:
|
|
* @tag: descriptor tag
|
|
* @data: (transfer none) (array length=length): descriptor data (after tag and length field)
|
|
* @length: length of @data
|
|
*
|
|
* Creates a #GstMpegtsDescriptor with custom @tag and @data
|
|
*
|
|
* Returns: #GstMpegtsDescriptor
|
|
*/
|
|
GstMpegtsDescriptor *
|
|
gst_mpegts_descriptor_from_custom (guint8 tag, const guint8 * data,
|
|
gsize length)
|
|
{
|
|
GstMpegtsDescriptor *descriptor;
|
|
|
|
g_return_val_if_fail (length > 0 || !data, NULL);
|
|
|
|
descriptor = _new_descriptor (tag, length);
|
|
|
|
if (data && (length > 0))
|
|
memcpy (descriptor->data + 2, data, length);
|
|
|
|
return descriptor;
|
|
}
|
|
|
|
/**
|
|
* gst_mpegts_descriptor_from_custom_with_extension:
|
|
* @tag: descriptor tag
|
|
* @tag_extension: descriptor tag extension
|
|
* @data: (transfer none) (array length=length): descriptor data (after tag and length field)
|
|
* @length: length of @data
|
|
*
|
|
* Creates a #GstMpegtsDescriptor with custom @tag, @tag_extension and @data
|
|
*
|
|
* Returns: #GstMpegtsDescriptor
|
|
*/
|
|
GstMpegtsDescriptor *
|
|
gst_mpegts_descriptor_from_custom_with_extension (guint8 tag,
|
|
guint8 tag_extension, const guint8 * data, gsize length)
|
|
{
|
|
GstMpegtsDescriptor *descriptor;
|
|
|
|
descriptor = _new_descriptor_with_extension (tag, tag_extension, length);
|
|
|
|
if (data && (length > 0))
|
|
memcpy (descriptor->data + 3, data, length);
|
|
|
|
return descriptor;
|
|
}
|