mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-11-03 16:09:39 +00:00
ee55f94ac0
Instead of trying to get the size of a NULL buffer :)
2027 lines
68 KiB
C
2027 lines
68 KiB
C
/* GStreamer RIFF I/O
|
|
* Copyright (C) 2003 Ronald Bultje <rbultje@ronald.bitfreak.net>
|
|
*
|
|
* riff-media.h: RIFF-id to/from caps routines
|
|
*
|
|
* 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 "riff-ids.h"
|
|
#include "riff-media.h"
|
|
|
|
#include <gst/audio/audio.h>
|
|
|
|
#include <string.h>
|
|
#include <math.h>
|
|
|
|
GST_DEBUG_CATEGORY_EXTERN (riff_debug);
|
|
#define GST_CAT_DEFAULT riff_debug
|
|
|
|
/**
|
|
* gst_riff_create_video_caps:
|
|
* @codec_fcc: fourCC codec for this codec.
|
|
* @strh: pointer to the strh stream header structure.
|
|
* @strf: pointer to the strf stream header structure, including any
|
|
* data that is within the range of strf.size, but excluding any
|
|
* additional data withint this chunk but outside strf.size.
|
|
* @strf_data: a #GstBuffer containing the additional data in the strf
|
|
* chunk outside reach of strf.size. Ususally a palette.
|
|
* @strd_data: a #GstBuffer containing the data in the strd stream header
|
|
* chunk. Usually codec initialization data.
|
|
* @codec_name: if given, will be filled with a human-readable codec name.
|
|
*/
|
|
|
|
GstCaps *
|
|
gst_riff_create_video_caps (guint32 codec_fcc,
|
|
gst_riff_strh * strh, gst_riff_strf_vids * strf,
|
|
GstBuffer * strf_data, GstBuffer * strd_data, char **codec_name)
|
|
{
|
|
GstCaps *caps = NULL;
|
|
GstBuffer *palette = NULL;
|
|
|
|
GST_DEBUG ("video fourcc %" GST_FOURCC_FORMAT, GST_FOURCC_ARGS (codec_fcc));
|
|
|
|
switch (codec_fcc) {
|
|
case GST_RIFF_DIB: /* uncompressed RGB */
|
|
case GST_RIFF_rgb:
|
|
case GST_RIFF_RGB:
|
|
case GST_RIFF_RAW:
|
|
{
|
|
gint bpp = (strf && strf->bit_cnt != 0) ? strf->bit_cnt : 8;
|
|
|
|
if (strf) {
|
|
if (bpp == 8) {
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "RGB8P", NULL);
|
|
} else if (bpp == 24) {
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "BGR", NULL);
|
|
} else if (bpp == 32) {
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "BGRx", NULL);
|
|
} else {
|
|
GST_WARNING ("Unhandled DIB RGB depth: %d", bpp);
|
|
return NULL;
|
|
}
|
|
} else {
|
|
/* for template */
|
|
caps =
|
|
gst_caps_from_string ("video/x-raw, format = (string) "
|
|
"{ RGB8P, BGR, BGRx }");
|
|
}
|
|
|
|
palette = strf_data;
|
|
strf_data = NULL;
|
|
if (codec_name) {
|
|
if (bpp == 8)
|
|
*codec_name = g_strdup_printf ("Palettized %d-bit RGB", bpp);
|
|
else
|
|
*codec_name = g_strdup_printf ("%d-bit RGB", bpp);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case GST_MAKE_FOURCC ('G', 'R', 'E', 'Y'):
|
|
case GST_MAKE_FOURCC ('Y', '8', '0', '0'):
|
|
case GST_MAKE_FOURCC ('Y', '8', ' ', ' '):
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "GRAY8", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Uncompressed 8-bit monochrome");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('r', '2', '1', '0'):
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "r210", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Uncompressed packed RGB 10-bit 4:4:4");
|
|
break;
|
|
|
|
case GST_RIFF_I420:
|
|
case GST_RIFF_i420:
|
|
case GST_RIFF_IYUV:
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "I420", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Uncompressed planar YUV 4:2:0");
|
|
break;
|
|
|
|
case GST_RIFF_YUY2:
|
|
case GST_RIFF_yuy2:
|
|
case GST_MAKE_FOURCC ('Y', 'U', 'N', 'V'):
|
|
case GST_MAKE_FOURCC ('Y', 'U', 'Y', 'V'):
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "YUY2", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Uncompressed packed YUV 4:2:2");
|
|
break;
|
|
|
|
case GST_RIFF_YVU9:
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "YVU9", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Uncompressed packed YVU 4:1:0");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('U', 'Y', 'V', 'Y'):
|
|
case GST_MAKE_FOURCC ('2', 'v', 'u', 'y'):
|
|
case GST_MAKE_FOURCC ('H', 'D', 'Y', 'C'):
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "UYVY", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Uncompressed packed YUV 4:2:2");
|
|
break;
|
|
|
|
case GST_RIFF_YV12:
|
|
case GST_RIFF_yv12:
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "YV12", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Uncompressed packed YVU 4:2:2");
|
|
break;
|
|
case GST_MAKE_FOURCC ('v', '2', '1', '0'):
|
|
caps = gst_caps_new_simple ("video/x-raw",
|
|
"format", G_TYPE_STRING, "v210", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Uncompressed packed 10-bit YUV 4:2:2");
|
|
break;
|
|
|
|
case GST_RIFF_MJPG: /* YUY2 MJPEG */
|
|
case GST_RIFF_mJPG:
|
|
case GST_MAKE_FOURCC ('A', 'V', 'R', 'n'):
|
|
case GST_RIFF_IJPG:
|
|
case GST_MAKE_FOURCC ('i', 'j', 'p', 'g'):
|
|
case GST_RIFF_DMB1:
|
|
case GST_RIFF_dmb1:
|
|
case GST_MAKE_FOURCC ('A', 'C', 'D', 'V'):
|
|
case GST_MAKE_FOURCC ('Q', 'I', 'V', 'G'):
|
|
caps = gst_caps_new_empty_simple ("image/jpeg");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Motion JPEG");
|
|
break;
|
|
|
|
case GST_RIFF_JPEG: /* generic (mostly RGB) MJPEG */
|
|
case GST_RIFF_jpeg:
|
|
case GST_MAKE_FOURCC ('j', 'p', 'e', 'g'): /* generic (mostly RGB) MJPEG */
|
|
caps = gst_caps_new_empty_simple ("image/jpeg");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("JPEG Still Image");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('P', 'I', 'X', 'L'): /* Miro/Pinnacle fourccs */
|
|
case GST_RIFF_VIXL: /* Miro/Pinnacle fourccs */
|
|
case GST_RIFF_vixl:
|
|
caps = gst_caps_new_empty_simple ("image/jpeg");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Miro/Pinnacle Motion JPEG");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('C', 'J', 'P', 'G'):
|
|
caps = gst_caps_new_empty_simple ("image/jpeg");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Creative Webcam JPEG");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('S', 'L', 'M', 'J'):
|
|
caps = gst_caps_new_empty_simple ("image/jpeg");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("SL Motion JPEG");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('J', 'P', 'G', 'L'):
|
|
caps = gst_caps_new_empty_simple ("image/jpeg");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Pegasus Lossless JPEG");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('L', 'O', 'C', 'O'):
|
|
caps = gst_caps_new_empty_simple ("video/x-loco");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("LOCO Lossless");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('S', 'P', '5', '3'):
|
|
case GST_MAKE_FOURCC ('S', 'P', '5', '4'):
|
|
case GST_MAKE_FOURCC ('S', 'P', '5', '5'):
|
|
case GST_MAKE_FOURCC ('S', 'P', '5', '6'):
|
|
case GST_MAKE_FOURCC ('S', 'P', '5', '7'):
|
|
case GST_MAKE_FOURCC ('S', 'P', '5', '8'):
|
|
caps = gst_caps_new_empty_simple ("video/sp5x");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Sp5x-like JPEG");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('Z', 'M', 'B', 'V'):
|
|
caps = gst_caps_new_empty_simple ("video/x-zmbv");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Zip Motion Block video");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('H', 'F', 'Y', 'U'):
|
|
caps = gst_caps_new_empty_simple ("video/x-huffyuv");
|
|
if (strf) {
|
|
gst_caps_set_simple (caps, "bpp",
|
|
G_TYPE_INT, (int) strf->bit_cnt, NULL);
|
|
}
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Huffman Lossless Codec");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('M', 'P', 'E', 'G'):
|
|
case GST_MAKE_FOURCC ('M', 'P', 'G', 'I'):
|
|
case GST_MAKE_FOURCC ('m', 'p', 'g', '1'):
|
|
case GST_MAKE_FOURCC ('M', 'P', 'G', '1'):
|
|
case GST_MAKE_FOURCC ('P', 'I', 'M', '1'):
|
|
case GST_MAKE_FOURCC (0x01, 0x00, 0x00, 0x10):
|
|
caps = gst_caps_new_simple ("video/mpeg",
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE,
|
|
"mpegversion", G_TYPE_INT, 1, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("MPEG-1 video");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('M', 'P', 'G', '2'):
|
|
case GST_MAKE_FOURCC ('m', 'p', 'g', '2'):
|
|
case GST_MAKE_FOURCC ('P', 'I', 'M', '2'):
|
|
case GST_MAKE_FOURCC ('D', 'V', 'R', ' '):
|
|
case GST_MAKE_FOURCC (0x02, 0x00, 0x00, 0x10):
|
|
caps = gst_caps_new_simple ("video/mpeg",
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE,
|
|
"mpegversion", G_TYPE_INT, 2, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("MPEG-2 video");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('L', 'M', 'P', '2'):
|
|
caps = gst_caps_new_simple ("video/mpeg",
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE,
|
|
"mpegversion", G_TYPE_INT, 2, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Lead MPEG-2 video");
|
|
break;
|
|
|
|
case GST_RIFF_H263:
|
|
case GST_RIFF_h263:
|
|
case GST_RIFF_i263:
|
|
case GST_MAKE_FOURCC ('U', '2', '6', '3'):
|
|
case GST_MAKE_FOURCC ('v', 'i', 'v', '1'):
|
|
case GST_MAKE_FOURCC ('T', '2', '6', '3'):
|
|
caps = gst_caps_new_simple ("video/x-h263",
|
|
"variant", G_TYPE_STRING, "itu", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("ITU H.26n");
|
|
break;
|
|
|
|
case GST_RIFF_L263:
|
|
/* http://www.leadcodecs.com/Codecs/LEAD-H263.htm */
|
|
caps = gst_caps_new_simple ("video/x-h263",
|
|
"variant", G_TYPE_STRING, "lead", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Lead H.263");
|
|
break;
|
|
|
|
case GST_RIFF_M263:
|
|
case GST_RIFF_m263:
|
|
caps = gst_caps_new_simple ("video/x-h263",
|
|
"variant", G_TYPE_STRING, "microsoft", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft H.263");
|
|
break;
|
|
|
|
case GST_RIFF_VDOW:
|
|
caps = gst_caps_new_simple ("video/x-h263",
|
|
"variant", G_TYPE_STRING, "vdolive", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("VDOLive");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('V', 'I', 'V', 'O'):
|
|
caps = gst_caps_new_simple ("video/x-h263",
|
|
"variant", G_TYPE_STRING, "vivo", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Vivo H.263");
|
|
break;
|
|
|
|
case GST_RIFF_x263:
|
|
caps = gst_caps_new_simple ("video/x-h263",
|
|
"variant", G_TYPE_STRING, "xirlink", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Xirlink H.263");
|
|
break;
|
|
|
|
/* apparently not standard H.263...? */
|
|
case GST_MAKE_FOURCC ('I', '2', '6', '3'):
|
|
caps = gst_caps_new_simple ("video/x-intel-h263",
|
|
"variant", G_TYPE_STRING, "intel", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Intel H.263");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('V', 'X', '1', 'K'):
|
|
caps = gst_caps_new_simple ("video/x-h263",
|
|
"variant", G_TYPE_STRING, "lucent", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Lucent VX1000S H.263");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('X', '2', '6', '4'):
|
|
case GST_MAKE_FOURCC ('x', '2', '6', '4'):
|
|
case GST_MAKE_FOURCC ('H', '2', '6', '4'):
|
|
case GST_MAKE_FOURCC ('h', '2', '6', '4'):
|
|
case GST_MAKE_FOURCC ('a', 'v', 'c', '1'):
|
|
case GST_MAKE_FOURCC ('A', 'V', 'C', '1'):
|
|
caps = gst_caps_new_simple ("video/x-h264",
|
|
"variant", G_TYPE_STRING, "itu", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("ITU H.264");
|
|
break;
|
|
|
|
case GST_RIFF_VSSH:
|
|
caps = gst_caps_new_simple ("video/x-h264",
|
|
"variant", G_TYPE_STRING, "videosoft", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("VideoSoft H.264");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('L', '2', '6', '4'):
|
|
/* http://www.leadcodecs.com/Codecs/LEAD-H264.htm */
|
|
caps = gst_caps_new_simple ("video/x-h264",
|
|
"variant", G_TYPE_STRING, "lead", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Lead H.264");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('S', 'E', 'D', 'G'):
|
|
caps = gst_caps_new_simple ("video/mpeg",
|
|
"mpegversion", G_TYPE_INT, 4,
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Samsung MPEG-4");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('M', '4', 'C', 'C'):
|
|
caps = gst_caps_new_simple ("video/mpeg",
|
|
"mpegversion", G_TYPE_INT, 4,
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Divio MPEG-4");
|
|
break;
|
|
|
|
case GST_RIFF_DIV3:
|
|
case GST_MAKE_FOURCC ('d', 'i', 'v', '3'):
|
|
case GST_MAKE_FOURCC ('D', 'V', 'X', '3'):
|
|
case GST_MAKE_FOURCC ('d', 'v', 'x', '3'):
|
|
case GST_MAKE_FOURCC ('D', 'I', 'V', '4'):
|
|
case GST_MAKE_FOURCC ('d', 'i', 'v', '4'):
|
|
case GST_MAKE_FOURCC ('D', 'I', 'V', '5'):
|
|
case GST_MAKE_FOURCC ('d', 'i', 'v', '5'):
|
|
case GST_MAKE_FOURCC ('D', 'I', 'V', '6'):
|
|
case GST_MAKE_FOURCC ('d', 'i', 'v', '6'):
|
|
case GST_MAKE_FOURCC ('M', 'P', 'G', '3'):
|
|
case GST_MAKE_FOURCC ('m', 'p', 'g', '3'):
|
|
case GST_MAKE_FOURCC ('c', 'o', 'l', '0'):
|
|
case GST_MAKE_FOURCC ('C', 'O', 'L', '0'):
|
|
case GST_MAKE_FOURCC ('c', 'o', 'l', '1'):
|
|
case GST_MAKE_FOURCC ('C', 'O', 'L', '1'):
|
|
case GST_MAKE_FOURCC ('A', 'P', '4', '1'):
|
|
caps = gst_caps_new_simple ("video/x-divx",
|
|
"divxversion", G_TYPE_INT, 3, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("DivX MS-MPEG-4 Version 3");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('d', 'i', 'v', 'x'):
|
|
case GST_MAKE_FOURCC ('D', 'I', 'V', 'X'):
|
|
caps = gst_caps_new_simple ("video/x-divx",
|
|
"divxversion", G_TYPE_INT, 4, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("DivX MPEG-4 Version 4");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('B', 'L', 'Z', '0'):
|
|
caps = gst_caps_new_simple ("video/x-divx",
|
|
"divxversion", G_TYPE_INT, 4, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Blizzard DivX");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('D', 'X', '5', '0'):
|
|
caps = gst_caps_new_simple ("video/x-divx",
|
|
"divxversion", G_TYPE_INT, 5, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("DivX MPEG-4 Version 5");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('M', 'P', 'G', '4'):
|
|
case GST_MAKE_FOURCC ('M', 'P', '4', '1'):
|
|
case GST_MAKE_FOURCC ('m', 'p', '4', '1'):
|
|
caps = gst_caps_new_simple ("video/x-msmpeg",
|
|
"msmpegversion", G_TYPE_INT, 41, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft MPEG-4 4.1");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('m', 'p', '4', '2'):
|
|
case GST_MAKE_FOURCC ('M', 'P', '4', '2'):
|
|
caps = gst_caps_new_simple ("video/x-msmpeg",
|
|
"msmpegversion", G_TYPE_INT, 42, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft MPEG-4 4.2");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('m', 'p', '4', '3'):
|
|
case GST_MAKE_FOURCC ('M', 'P', '4', '3'):
|
|
caps = gst_caps_new_simple ("video/x-msmpeg",
|
|
"msmpegversion", G_TYPE_INT, 43, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft MPEG-4 4.3");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('M', 'P', '4', 'S'):
|
|
case GST_MAKE_FOURCC ('M', '4', 'S', '2'):
|
|
caps = gst_caps_new_simple ("video/mpeg",
|
|
"mpegversion", G_TYPE_INT, 4,
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft ISO MPEG-4 1.1");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('F', 'M', 'P', '4'):
|
|
case GST_MAKE_FOURCC ('U', 'M', 'P', '4'):
|
|
case GST_MAKE_FOURCC ('F', 'F', 'D', 'S'):
|
|
caps = gst_caps_new_simple ("video/mpeg",
|
|
"mpegversion", G_TYPE_INT, 4,
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("FFmpeg MPEG-4");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('3', 'I', 'V', '1'):
|
|
case GST_MAKE_FOURCC ('3', 'I', 'V', '2'):
|
|
case GST_MAKE_FOURCC ('X', 'V', 'I', 'D'):
|
|
case GST_MAKE_FOURCC ('x', 'v', 'i', 'd'):
|
|
case GST_MAKE_FOURCC ('E', 'M', '4', 'A'):
|
|
case GST_MAKE_FOURCC ('E', 'P', 'V', 'H'):
|
|
case GST_MAKE_FOURCC ('F', 'V', 'F', 'W'):
|
|
case GST_MAKE_FOURCC ('I', 'N', 'M', 'C'):
|
|
case GST_MAKE_FOURCC ('D', 'I', 'G', 'I'):
|
|
case GST_MAKE_FOURCC ('D', 'M', '2', 'K'):
|
|
case GST_MAKE_FOURCC ('D', 'C', 'O', 'D'):
|
|
case GST_MAKE_FOURCC ('M', 'V', 'X', 'M'):
|
|
case GST_MAKE_FOURCC ('P', 'M', '4', 'V'):
|
|
case GST_MAKE_FOURCC ('S', 'M', 'P', '4'):
|
|
case GST_MAKE_FOURCC ('D', 'X', 'G', 'M'):
|
|
case GST_MAKE_FOURCC ('V', 'I', 'D', 'M'):
|
|
case GST_MAKE_FOURCC ('M', '4', 'T', '3'):
|
|
case GST_MAKE_FOURCC ('G', 'E', 'O', 'X'):
|
|
case GST_MAKE_FOURCC ('M', 'P', '4', 'V'):
|
|
case GST_MAKE_FOURCC ('m', 'p', '4', 'v'):
|
|
case GST_MAKE_FOURCC ('R', 'M', 'P', '4'):
|
|
caps = gst_caps_new_simple ("video/mpeg",
|
|
"mpegversion", G_TYPE_INT, 4,
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("MPEG-4");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('3', 'i', 'v', 'd'):
|
|
case GST_MAKE_FOURCC ('3', 'I', 'V', 'D'):
|
|
caps = gst_caps_new_simple ("video/x-msmpeg",
|
|
"msmpegversion", G_TYPE_INT, 43, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft MPEG-4 4.3"); /* FIXME? */
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('D', 'V', 'S', 'D'):
|
|
case GST_MAKE_FOURCC ('d', 'v', 's', 'd'):
|
|
case GST_MAKE_FOURCC ('d', 'v', 'c', ' '):
|
|
case GST_MAKE_FOURCC ('d', 'v', '2', '5'):
|
|
caps = gst_caps_new_simple ("video/x-dv",
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE,
|
|
"dvversion", G_TYPE_INT, 25, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Generic DV");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('C', 'D', 'V', 'C'):
|
|
case GST_MAKE_FOURCC ('c', 'd', 'v', 'c'):
|
|
caps = gst_caps_new_simple ("video/x-dv",
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE,
|
|
"dvversion", G_TYPE_INT, 25, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Canopus DV");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('D', 'V', '5', '0'):
|
|
case GST_MAKE_FOURCC ('d', 'v', '5', '0'):
|
|
caps = gst_caps_new_simple ("video/x-dv",
|
|
"systemstream", G_TYPE_BOOLEAN, FALSE,
|
|
"dvversion", G_TYPE_INT, 50, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("DVCPro50 Video");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('M', 'S', 'S', '1'):
|
|
caps = gst_caps_new_simple ("video/x-wmv",
|
|
"wmvversion", G_TYPE_INT, 1, "format", G_TYPE_STRING, "MSS1", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft Windows Media 7 Screen");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('M', 'S', 'S', '2'):
|
|
caps = gst_caps_new_simple ("video/x-wmv",
|
|
"wmvversion", G_TYPE_INT, 3, "format", G_TYPE_STRING, "MSS2", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft Windows Media 9 Screen");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('W', 'M', 'V', '1'):
|
|
caps = gst_caps_new_simple ("video/x-wmv",
|
|
"wmvversion", G_TYPE_INT, 1, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft Windows Media 7");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('W', 'M', 'V', '2'):
|
|
caps = gst_caps_new_simple ("video/x-wmv",
|
|
"wmvversion", G_TYPE_INT, 2, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft Windows Media 8");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('W', 'M', 'V', '3'):
|
|
caps = gst_caps_new_simple ("video/x-wmv",
|
|
"wmvversion", G_TYPE_INT, 3, "format", G_TYPE_STRING, "WMV3", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft Windows Media 9");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('W', 'M', 'V', 'A'):
|
|
caps = gst_caps_new_simple ("video/x-wmv",
|
|
"wmvversion", G_TYPE_INT, 3, "format", G_TYPE_STRING, "WMVA", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft Windows Media Advanced Profile");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('W', 'V', 'C', '1'):
|
|
caps = gst_caps_new_simple ("video/x-wmv",
|
|
"wmvversion", G_TYPE_INT, 3, "format", G_TYPE_STRING, "WVC1", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft Windows Media VC-1");
|
|
break;
|
|
|
|
case GST_RIFF_cvid:
|
|
case GST_RIFF_CVID:
|
|
caps = gst_caps_new_empty_simple ("video/x-cinepak");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Cinepak video");
|
|
break;
|
|
|
|
case GST_RIFF_FCCH_MSVC:
|
|
case GST_RIFF_FCCH_msvc:
|
|
case GST_RIFF_CRAM:
|
|
case GST_RIFF_cram:
|
|
case GST_RIFF_WHAM:
|
|
case GST_RIFF_wham:
|
|
caps = gst_caps_new_simple ("video/x-msvideocodec",
|
|
"msvideoversion", G_TYPE_INT, 1, NULL);
|
|
if (strf) {
|
|
gst_caps_set_simple (caps, "bpp",
|
|
G_TYPE_INT, (int) strf->bit_cnt, NULL);
|
|
}
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("MS video v1");
|
|
palette = strf_data;
|
|
strf_data = NULL;
|
|
break;
|
|
|
|
case GST_RIFF_FCCH_RLE:
|
|
case GST_MAKE_FOURCC ('m', 'r', 'l', 'e'):
|
|
case GST_MAKE_FOURCC (0x1, 0x0, 0x0, 0x0): /* why, why, why? */
|
|
case GST_MAKE_FOURCC (0x2, 0x0, 0x0, 0x0): /* why, why, why? */
|
|
caps = gst_caps_new_simple ("video/x-rle",
|
|
"layout", G_TYPE_STRING, "microsoft", NULL);
|
|
palette = strf_data;
|
|
strf_data = NULL;
|
|
if (strf) {
|
|
gst_caps_set_simple (caps,
|
|
"depth", G_TYPE_INT, (gint) strf->bit_cnt, NULL);
|
|
} else {
|
|
gst_caps_set_simple (caps, "depth", GST_TYPE_INT_RANGE, 1, 64, NULL);
|
|
}
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Microsoft RLE");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('A', 'A', 'S', 'C'):
|
|
caps = gst_caps_new_empty_simple ("video/x-aasc");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Autodesk Animator");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('X', 'x', 'a', 'n'):
|
|
caps = gst_caps_new_simple ("video/x-xan",
|
|
"wcversion", G_TYPE_INT, 4, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Xan Wing Commander 4");
|
|
break;
|
|
|
|
case GST_RIFF_RT21:
|
|
case GST_RIFF_rt21:
|
|
caps = gst_caps_new_simple ("video/x-indeo",
|
|
"indeoversion", G_TYPE_INT, 2, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Intel Video 2");
|
|
break;
|
|
|
|
case GST_RIFF_IV31:
|
|
case GST_RIFF_IV32:
|
|
case GST_RIFF_iv31:
|
|
case GST_RIFF_iv32:
|
|
caps = gst_caps_new_simple ("video/x-indeo",
|
|
"indeoversion", G_TYPE_INT, 3, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Intel Video 3");
|
|
break;
|
|
|
|
case GST_RIFF_IV41:
|
|
case GST_RIFF_iv41:
|
|
caps = gst_caps_new_simple ("video/x-indeo",
|
|
"indeoversion", G_TYPE_INT, 4, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Intel Video 4");
|
|
break;
|
|
|
|
case GST_RIFF_IV50:
|
|
caps = gst_caps_new_simple ("video/x-indeo",
|
|
"indeoversion", G_TYPE_INT, 5, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Intel Video 5");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('M', 'S', 'Z', 'H'):
|
|
caps = gst_caps_new_empty_simple ("video/x-mszh");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Lossless MSZH Video");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('Z', 'L', 'I', 'B'):
|
|
caps = gst_caps_new_empty_simple ("video/x-zlib");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Lossless zlib video");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('C', 'L', 'J', 'R'):
|
|
case GST_MAKE_FOURCC ('c', 'l', 'j', 'r'):
|
|
caps = gst_caps_new_empty_simple ("video/x-cirrus-logic-accupak");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Cirrus Logipak AccuPak");
|
|
break;
|
|
|
|
case GST_RIFF_CYUV:
|
|
case GST_RIFF_cyuv:
|
|
caps = gst_caps_new_empty_simple ("video/x-compressed-yuv");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("CYUV Lossless");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('D', 'U', 'C', 'K'):
|
|
case GST_MAKE_FOURCC ('P', 'V', 'E', 'Z'):
|
|
caps = gst_caps_new_simple ("video/x-truemotion",
|
|
"trueversion", G_TYPE_INT, 1, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Duck Truemotion1");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('T', 'M', '2', '0'):
|
|
caps = gst_caps_new_simple ("video/x-truemotion",
|
|
"trueversion", G_TYPE_INT, 2, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("TrueMotion 2.0");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('V', 'P', '3', '0'):
|
|
case GST_MAKE_FOURCC ('v', 'p', '3', '0'):
|
|
case GST_MAKE_FOURCC ('V', 'P', '3', '1'):
|
|
case GST_MAKE_FOURCC ('v', 'p', '3', '1'):
|
|
case GST_MAKE_FOURCC ('V', 'P', '3', ' '):
|
|
caps = gst_caps_new_empty_simple ("video/x-vp3");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("VP3");
|
|
break;
|
|
|
|
case GST_RIFF_ULTI:
|
|
case GST_RIFF_ulti:
|
|
caps = gst_caps_new_empty_simple ("video/x-ultimotion");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("IBM UltiMotion");
|
|
break;
|
|
|
|
/* FIXME 2.0: Rename video/x-camtasia to video/x-tscc,version=1 */
|
|
case GST_MAKE_FOURCC ('T', 'S', 'C', 'C'):
|
|
case GST_MAKE_FOURCC ('t', 's', 'c', 'c'):{
|
|
if (strf) {
|
|
gint depth = (strf->bit_cnt != 0) ? (gint) strf->bit_cnt : 24;
|
|
|
|
caps = gst_caps_new_simple ("video/x-camtasia", "depth", G_TYPE_INT,
|
|
depth, NULL);
|
|
} else {
|
|
/* template caps */
|
|
caps = gst_caps_new_empty_simple ("video/x-camtasia");
|
|
}
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("TechSmith Camtasia");
|
|
break;
|
|
}
|
|
|
|
case GST_MAKE_FOURCC ('T', 'S', 'C', '2'):
|
|
case GST_MAKE_FOURCC ('t', 's', 'c', '2'):{
|
|
caps =
|
|
gst_caps_new_simple ("video/x-tscc", "tsccversion", G_TYPE_INT, 2,
|
|
NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("TechSmith Screen Capture 2");
|
|
break;
|
|
}
|
|
|
|
case GST_MAKE_FOURCC ('C', 'S', 'C', 'D'):
|
|
{
|
|
if (strf) {
|
|
gint depth = (strf->bit_cnt != 0) ? (gint) strf->bit_cnt : 24;
|
|
|
|
caps = gst_caps_new_simple ("video/x-camstudio", "depth", G_TYPE_INT,
|
|
depth, NULL);
|
|
} else {
|
|
/* template caps */
|
|
caps = gst_caps_new_empty_simple ("video/x-camstudio");
|
|
}
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Camstudio");
|
|
break;
|
|
}
|
|
|
|
case GST_MAKE_FOURCC ('V', 'C', 'R', '1'):
|
|
caps = gst_caps_new_simple ("video/x-ati-vcr",
|
|
"vcrversion", G_TYPE_INT, 1, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("ATI VCR 1");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('V', 'C', 'R', '2'):
|
|
caps = gst_caps_new_simple ("video/x-ati-vcr",
|
|
"vcrversion", G_TYPE_INT, 2, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("ATI VCR 2");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('A', 'S', 'V', '1'):
|
|
caps = gst_caps_new_simple ("video/x-asus",
|
|
"asusversion", G_TYPE_INT, 1, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Asus Video 1");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('A', 'S', 'V', '2'):
|
|
caps = gst_caps_new_simple ("video/x-asus",
|
|
"asusversion", G_TYPE_INT, 2, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Asus Video 2");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('M', 'P', 'N', 'G'):
|
|
case GST_MAKE_FOURCC ('m', 'p', 'n', 'g'):
|
|
case GST_MAKE_FOURCC ('P', 'N', 'G', ' '):
|
|
case GST_MAKE_FOURCC ('p', 'n', 'g', ' '):
|
|
caps = gst_caps_new_empty_simple ("image/png");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("PNG image");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('F', 'L', 'V', '1'):
|
|
caps = gst_caps_new_simple ("video/x-flash-video",
|
|
"flvversion", G_TYPE_INT, 1, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Flash Video 1");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('V', 'M', 'n', 'c'):
|
|
caps = gst_caps_new_simple ("video/x-vmnc",
|
|
"version", G_TYPE_INT, 1, NULL);
|
|
if (strf && strf->bit_cnt != 0)
|
|
gst_caps_set_simple (caps, "bpp", G_TYPE_INT, strf->bit_cnt, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("VMWare NC Video");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('d', 'r', 'a', 'c'):
|
|
caps = gst_caps_new_empty_simple ("video/x-dirac");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Dirac");
|
|
break;
|
|
|
|
case GST_RIFF_rpza:
|
|
case GST_RIFF_azpr:
|
|
case GST_MAKE_FOURCC ('R', 'P', 'Z', 'A'):
|
|
caps = gst_caps_new_empty_simple ("video/x-apple-video");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Apple Video (RPZA)");
|
|
break;
|
|
|
|
|
|
case GST_MAKE_FOURCC ('F', 'F', 'V', '1'):
|
|
caps = gst_caps_new_simple ("video/x-ffv",
|
|
"ffvversion", G_TYPE_INT, 1, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("FFmpeg lossless video codec");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('K', 'M', 'V', 'C'):
|
|
caps = gst_caps_new_empty_simple ("video/x-kmvc");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Karl Morton's video codec");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('v', 'p', '5', '0'):
|
|
case GST_MAKE_FOURCC ('V', 'P', '5', '0'):
|
|
caps = gst_caps_new_empty_simple ("video/x-vp5");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("On2 VP5");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('v', 'p', '6', '0'):
|
|
case GST_MAKE_FOURCC ('V', 'P', '6', '0'):
|
|
case GST_MAKE_FOURCC ('v', 'p', '6', '1'):
|
|
case GST_MAKE_FOURCC ('V', 'P', '6', '1'):
|
|
case GST_MAKE_FOURCC ('V', 'p', '6', '2'):
|
|
case GST_MAKE_FOURCC ('V', 'P', '6', '2'):
|
|
caps = gst_caps_new_empty_simple ("video/x-vp6");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("On2 VP6");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('V', 'P', '6', 'F'):
|
|
case GST_MAKE_FOURCC ('v', 'p', '6', 'f'):
|
|
case GST_MAKE_FOURCC ('F', 'L', 'V', '4'):
|
|
caps = gst_caps_new_empty_simple ("video/x-vp6-flash");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("On2 VP6");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('v', 'p', '7', '0'):
|
|
case GST_MAKE_FOURCC ('V', 'P', '7', '0'):
|
|
caps = gst_caps_new_empty_simple ("video/x-vp7");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("On2 VP7");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('V', 'P', '8', '0'):
|
|
caps = gst_caps_new_empty_simple ("video/x-vp8");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("On2 VP8");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('L', 'M', '2', '0'):
|
|
caps = gst_caps_new_empty_simple ("video/x-mimic");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Mimic webcam");
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('T', 'H', 'E', 'O'):
|
|
case GST_MAKE_FOURCC ('t', 'h', 'e', 'o'):
|
|
caps = gst_caps_new_empty_simple ("video/x-theora");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Theora video codec");
|
|
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('F', 'P', 'S', '1'):
|
|
caps = gst_caps_new_empty_simple ("video/x-fraps");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Fraps video");
|
|
|
|
break;
|
|
|
|
case GST_MAKE_FOURCC ('D', 'X', 'S', 'B'):
|
|
case GST_MAKE_FOURCC ('D', 'X', 'S', 'A'):
|
|
caps = gst_caps_new_empty_simple ("subpicture/x-xsub");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("XSUB subpicture stream");
|
|
|
|
break;
|
|
|
|
default:
|
|
GST_WARNING ("Unknown video fourcc %" GST_FOURCC_FORMAT,
|
|
GST_FOURCC_ARGS (codec_fcc));
|
|
return NULL;
|
|
}
|
|
|
|
if (strh != NULL) {
|
|
gst_caps_set_simple (caps, "framerate", GST_TYPE_FRACTION,
|
|
strh->rate, strh->scale, NULL);
|
|
} else {
|
|
gst_caps_set_simple (caps,
|
|
"framerate", GST_TYPE_FRACTION_RANGE, 0, 1, G_MAXINT, 1, NULL);
|
|
}
|
|
|
|
if (strf != NULL) {
|
|
/* raw rgb data is stored topdown, but instead of inverting the buffer, */
|
|
/* some tools just negate the height field in the header (e.g. ffmpeg) */
|
|
gst_caps_set_simple (caps,
|
|
"width", G_TYPE_INT, strf->width,
|
|
"height", G_TYPE_INT, ABS ((gint) strf->height), NULL);
|
|
} else {
|
|
gst_caps_set_simple (caps,
|
|
"width", GST_TYPE_INT_RANGE, 1, G_MAXINT,
|
|
"height", GST_TYPE_INT_RANGE, 1, G_MAXINT, NULL);
|
|
}
|
|
|
|
/* extradata */
|
|
if (strf_data || strd_data) {
|
|
GstBuffer *codec_data;
|
|
|
|
codec_data = strf_data ? strf_data : strd_data;
|
|
|
|
gst_caps_set_simple (caps, "codec_data", GST_TYPE_BUFFER, codec_data, NULL);
|
|
}
|
|
|
|
/* palette */
|
|
if (palette) {
|
|
GstBuffer *copy;
|
|
guint num_colors;
|
|
gsize size;
|
|
|
|
if (strf != NULL)
|
|
num_colors = strf->num_colors;
|
|
else
|
|
num_colors = 256;
|
|
|
|
size = gst_buffer_get_size (palette);
|
|
|
|
if (size >= (num_colors * 4)) {
|
|
guint8 *pdata;
|
|
|
|
/* palette is always at least 256*4 bytes */
|
|
pdata = g_malloc0 (MAX (size, 256 * 4));
|
|
gst_buffer_extract (palette, 0, pdata, size);
|
|
|
|
if (G_BYTE_ORDER == G_BIG_ENDIAN) {
|
|
guint8 *p = pdata;
|
|
gint n;
|
|
|
|
/* own endianness */
|
|
for (n = 0; n < num_colors; n++) {
|
|
GST_WRITE_UINT32_BE (p, GST_READ_UINT32_LE (p));
|
|
p += sizeof (guint32);
|
|
}
|
|
}
|
|
|
|
copy = gst_buffer_new_wrapped (pdata, size);
|
|
gst_caps_set_simple (caps, "palette_data", GST_TYPE_BUFFER, copy, NULL);
|
|
gst_buffer_unref (copy);
|
|
} else {
|
|
GST_WARNING ("Palette smaller than expected: broken file");
|
|
}
|
|
}
|
|
|
|
return caps;
|
|
}
|
|
|
|
static const struct
|
|
{
|
|
const guint32 ms_mask;
|
|
const GstAudioChannelPosition gst_pos;
|
|
} layout_mapping[] = {
|
|
{
|
|
0x00001, GST_AUDIO_CHANNEL_POSITION_FRONT_LEFT}, {
|
|
0x00002, GST_AUDIO_CHANNEL_POSITION_FRONT_RIGHT}, {
|
|
0x00004, GST_AUDIO_CHANNEL_POSITION_FRONT_CENTER}, {
|
|
0x00008, GST_AUDIO_CHANNEL_POSITION_LFE1}, {
|
|
0x00010, GST_AUDIO_CHANNEL_POSITION_REAR_LEFT}, {
|
|
0x00020, GST_AUDIO_CHANNEL_POSITION_REAR_RIGHT}, {
|
|
0x00040, GST_AUDIO_CHANNEL_POSITION_FRONT_LEFT_OF_CENTER}, {
|
|
0x00080, GST_AUDIO_CHANNEL_POSITION_FRONT_RIGHT_OF_CENTER}, {
|
|
0x00100, GST_AUDIO_CHANNEL_POSITION_REAR_CENTER}, {
|
|
0x00200, GST_AUDIO_CHANNEL_POSITION_SIDE_LEFT}, {
|
|
0x00400, GST_AUDIO_CHANNEL_POSITION_SIDE_RIGHT}, {
|
|
0x00800, GST_AUDIO_CHANNEL_POSITION_TOP_CENTER}, {
|
|
0x01000, GST_AUDIO_CHANNEL_POSITION_TOP_FRONT_LEFT}, {
|
|
0x02000, GST_AUDIO_CHANNEL_POSITION_TOP_FRONT_CENTER}, {
|
|
0x04000, GST_AUDIO_CHANNEL_POSITION_TOP_FRONT_RIGHT}, {
|
|
0x08000, GST_AUDIO_CHANNEL_POSITION_TOP_REAR_LEFT}, {
|
|
0x10000, GST_AUDIO_CHANNEL_POSITION_TOP_REAR_CENTER}, {
|
|
0x20000, GST_AUDIO_CHANNEL_POSITION_TOP_REAR_RIGHT}
|
|
};
|
|
|
|
#define MAX_CHANNEL_POSITIONS G_N_ELEMENTS (layout_mapping)
|
|
|
|
static gboolean
|
|
gst_riff_wavext_add_channel_mask (GstCaps * caps, gint num_channels,
|
|
guint32 layout, gint channel_reorder_map[18])
|
|
{
|
|
gint i, p;
|
|
guint64 channel_mask = 0;
|
|
GstAudioChannelPosition *from, *to;
|
|
gboolean ret = FALSE;
|
|
|
|
if (num_channels < 1) {
|
|
GST_DEBUG ("invalid number of channels: %d", num_channels);
|
|
return FALSE;
|
|
}
|
|
|
|
from = g_new (GstAudioChannelPosition, num_channels);
|
|
to = g_new (GstAudioChannelPosition, num_channels);
|
|
p = 0;
|
|
for (i = 0; i < MAX_CHANNEL_POSITIONS; ++i) {
|
|
if ((layout & layout_mapping[i].ms_mask) != 0) {
|
|
if (p >= num_channels) {
|
|
GST_WARNING ("More bits set in the channel layout map than there "
|
|
"are channels! Setting channel-mask to 0.");
|
|
channel_mask = 0;
|
|
break;
|
|
}
|
|
channel_mask |= G_GUINT64_CONSTANT (1) << layout_mapping[i].gst_pos;
|
|
from[p] = layout_mapping[i].gst_pos;
|
|
++p;
|
|
}
|
|
}
|
|
|
|
if (channel_mask > 0 && channel_reorder_map) {
|
|
if (p != num_channels) {
|
|
/* WAVEFORMATEXTENSIBLE allows to have more channels than bits in
|
|
* the channel mask. We accept this, too, and hope that downstream
|
|
* can handle this */
|
|
GST_WARNING ("Partially unknown positions in channel mask");
|
|
for (; p < num_channels; ++p)
|
|
from[p] = GST_AUDIO_CHANNEL_POSITION_INVALID;
|
|
}
|
|
memcpy (to, from, sizeof (from[0]) * num_channels);
|
|
if (!gst_audio_channel_positions_to_valid_order (to, num_channels))
|
|
goto fail;
|
|
if (!gst_audio_get_channel_reorder_map (num_channels, from, to,
|
|
channel_reorder_map))
|
|
goto fail;
|
|
}
|
|
|
|
gst_caps_set_simple (caps, "channel-mask", GST_TYPE_BITMASK, channel_mask,
|
|
NULL);
|
|
|
|
ret = TRUE;
|
|
|
|
fail:
|
|
g_free (from);
|
|
g_free (to);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static gboolean
|
|
gst_riff_wave_add_default_channel_mask (GstCaps * caps,
|
|
gint nchannels, gint channel_reorder_map[18])
|
|
{
|
|
guint64 channel_mask = 0;
|
|
static const gint reorder_maps[8][11] = {
|
|
{0,},
|
|
{0, 1},
|
|
{-1, -1, -1},
|
|
{0, 1, 2, 3},
|
|
{0, 1, 3, 4, 2},
|
|
{0, 1, 4, 5, 2, 3},
|
|
{-1, -1, -1, -1, -1, -1, -1},
|
|
{0, 1, 4, 5, 2, 3, 6, 7}
|
|
};
|
|
|
|
if (nchannels > 8) {
|
|
GST_DEBUG ("invalid number of channels: %d", nchannels);
|
|
return FALSE;
|
|
}
|
|
|
|
/* This uses the default channel mapping from ALSA which
|
|
* is used in quite a few surround test files and seems to be
|
|
* the defacto standard. The channel mapping from
|
|
* WAVE_FORMAT_EXTENSIBLE doesn't seem to be used in normal
|
|
* wav files like chan-id.wav.
|
|
* http://bugzilla.gnome.org/show_bug.cgi?id=489010
|
|
*/
|
|
switch (nchannels) {
|
|
case 1:
|
|
/* Mono => nothing */
|
|
if (channel_reorder_map)
|
|
channel_reorder_map[0] = 0;
|
|
return TRUE;
|
|
case 8:
|
|
channel_mask |=
|
|
G_GUINT64_CONSTANT (1) << GST_AUDIO_CHANNEL_POSITION_SIDE_RIGHT;
|
|
channel_mask |=
|
|
G_GUINT64_CONSTANT (1) << GST_AUDIO_CHANNEL_POSITION_SIDE_LEFT;
|
|
/* fall through */
|
|
case 6:
|
|
channel_mask |= G_GUINT64_CONSTANT (1) << GST_AUDIO_CHANNEL_POSITION_LFE1;
|
|
/* fall through */
|
|
case 5:
|
|
channel_mask |=
|
|
G_GUINT64_CONSTANT (1) << GST_AUDIO_CHANNEL_POSITION_FRONT_CENTER;
|
|
/* fall through */
|
|
case 4:
|
|
channel_mask |=
|
|
G_GUINT64_CONSTANT (1) << GST_AUDIO_CHANNEL_POSITION_REAR_RIGHT;
|
|
channel_mask |=
|
|
G_GUINT64_CONSTANT (1) << GST_AUDIO_CHANNEL_POSITION_REAR_LEFT;
|
|
/* fall through */
|
|
case 2:
|
|
channel_mask |=
|
|
G_GUINT64_CONSTANT (1) << GST_AUDIO_CHANNEL_POSITION_FRONT_RIGHT;
|
|
channel_mask |=
|
|
G_GUINT64_CONSTANT (1) << GST_AUDIO_CHANNEL_POSITION_FRONT_LEFT;
|
|
break;
|
|
default:
|
|
return FALSE;
|
|
}
|
|
|
|
if (channel_reorder_map)
|
|
memcpy (channel_reorder_map, reorder_maps[nchannels - 1],
|
|
sizeof (gint) * nchannels);
|
|
|
|
gst_caps_set_simple (caps, "channel-mask", GST_TYPE_BITMASK, channel_mask,
|
|
NULL);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static guint32
|
|
gst_riff_wavext_get_default_channel_mask (guint nchannels)
|
|
{
|
|
guint32 channel_mask = 0;
|
|
|
|
/* Set the default channel mask for the given number of channels.
|
|
* http://www.microsoft.com/whdc/device/audio/multichaud.mspx
|
|
*/
|
|
switch (nchannels) {
|
|
case 11:
|
|
channel_mask |= 0x00400;
|
|
channel_mask |= 0x00200;
|
|
case 9:
|
|
channel_mask |= 0x00100;
|
|
case 8:
|
|
channel_mask |= 0x00080;
|
|
channel_mask |= 0x00040;
|
|
case 6:
|
|
channel_mask |= 0x00020;
|
|
channel_mask |= 0x00010;
|
|
case 4:
|
|
channel_mask |= 0x00008;
|
|
case 3:
|
|
channel_mask |= 0x00004;
|
|
case 2:
|
|
channel_mask |= 0x00002;
|
|
channel_mask |= 0x00001;
|
|
break;
|
|
}
|
|
|
|
return channel_mask;
|
|
}
|
|
|
|
GstCaps *
|
|
gst_riff_create_audio_caps (guint16 codec_id,
|
|
gst_riff_strh * strh, gst_riff_strf_auds * strf,
|
|
GstBuffer * strf_data, GstBuffer * strd_data, char **codec_name,
|
|
gint channel_reorder_map[18])
|
|
{
|
|
gboolean block_align = FALSE, rate_chan = TRUE;
|
|
GstCaps *caps = NULL;
|
|
gint i;
|
|
|
|
if (channel_reorder_map)
|
|
for (i = 0; i < 18; i++)
|
|
channel_reorder_map[i] = -1;
|
|
|
|
switch (codec_id) {
|
|
case GST_RIFF_WAVE_FORMAT_PCM: /* PCM */
|
|
if (strf != NULL) {
|
|
gint ba = strf->blockalign;
|
|
gint ch = strf->channels;
|
|
gint wd, ws;
|
|
GstAudioFormat format;
|
|
|
|
if (ba > (32 / 8) * ch) {
|
|
GST_WARNING ("Invalid block align: %d > %d", ba, (32 / 8) * ch);
|
|
wd = GST_ROUND_UP_8 (strf->bits_per_sample);
|
|
} else if (ba != 0) {
|
|
/* If we have an empty blockalign, we take the width contained in
|
|
* strf->bits_per_sample */
|
|
wd = ba * 8 / ch;
|
|
} else {
|
|
wd = GST_ROUND_UP_8 (strf->bits_per_sample);
|
|
}
|
|
|
|
if (strf->bits_per_sample > 32) {
|
|
GST_WARNING ("invalid depth (%d) of pcm audio, overwriting.",
|
|
strf->bits_per_sample);
|
|
strf->bits_per_sample = wd;
|
|
}
|
|
|
|
/* in riff, the depth is stored in the size field but it just means that
|
|
* the _least_ significant bits are cleared. We can therefore just play
|
|
* the sample as if it had a depth == width */
|
|
/* For reference, the actual depth is in strf->bits_per_sample */
|
|
ws = wd;
|
|
|
|
format =
|
|
gst_audio_format_build_integer (wd != 8, G_LITTLE_ENDIAN, wd, ws);
|
|
if (format == GST_AUDIO_FORMAT_UNKNOWN) {
|
|
GST_WARNING ("Unsupported raw audio format with width %d", wd);
|
|
return NULL;
|
|
}
|
|
|
|
caps = gst_caps_new_simple ("audio/x-raw",
|
|
"format", G_TYPE_STRING, gst_audio_format_to_string (format),
|
|
"layout", G_TYPE_STRING, "interleaved",
|
|
"channels", G_TYPE_INT, ch, NULL);
|
|
|
|
/* Add default channel layout. We know no default layout for more than
|
|
* 8 channels. */
|
|
if (ch > 8)
|
|
GST_WARNING ("don't know default layout for %d channels", ch);
|
|
else if (gst_riff_wave_add_default_channel_mask (caps, ch,
|
|
channel_reorder_map))
|
|
GST_DEBUG ("using default channel layout for %d channels", ch);
|
|
else
|
|
GST_WARNING ("failed to add channel layout");
|
|
} else {
|
|
/* FIXME: this is pretty useless - we need fixed caps */
|
|
caps = gst_caps_from_string ("audio/x-raw, "
|
|
"format = (string) { S8, U8, S16LE, U16LE, S24LE, "
|
|
"U24LE, S32LE, U32LE }, " "layout = (string) interleaved");
|
|
}
|
|
if (codec_name && strf)
|
|
*codec_name = g_strdup_printf ("Uncompressed %d-bit PCM audio",
|
|
strf->bits_per_sample);
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_ADPCM:
|
|
if (strf != NULL) {
|
|
/* Many encoding tools create a wrong bitrate information in the header,
|
|
* so either we calculate the bitrate or mark it as invalid as this
|
|
* would probably confuse timing */
|
|
strf->av_bps = 0;
|
|
if (strf->channels != 0 && strf->rate != 0 && strf->blockalign != 0) {
|
|
int spb = ((strf->blockalign - strf->channels * 7) / 2) * 2;
|
|
strf->av_bps =
|
|
gst_util_uint64_scale_int (strf->rate, strf->blockalign, spb);
|
|
GST_DEBUG ("fixing av_bps to calculated value %d of MS ADPCM",
|
|
strf->av_bps);
|
|
}
|
|
}
|
|
caps = gst_caps_new_simple ("audio/x-adpcm",
|
|
"layout", G_TYPE_STRING, "microsoft", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("ADPCM audio");
|
|
block_align = TRUE;
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_IEEE_FLOAT:
|
|
if (strf != NULL) {
|
|
gint ba = strf->blockalign;
|
|
gint ch = strf->channels;
|
|
|
|
if (ba > 0 && ch > 0 && (ba == (64 / 8) * ch || ba == (32 / 8) * ch)) {
|
|
gint wd = ba * 8 / ch;
|
|
|
|
caps = gst_caps_new_simple ("audio/x-raw",
|
|
"format", G_TYPE_STRING, wd == 64 ? "F64LE" : "F32LE",
|
|
"layout", G_TYPE_STRING, "interleaved",
|
|
"channels", G_TYPE_INT, ch, NULL);
|
|
|
|
/* Add default channel layout. We know no default layout for more than
|
|
* 8 channels. */
|
|
if (ch > 8)
|
|
GST_WARNING ("don't know default layout for %d channels", ch);
|
|
else if (gst_riff_wave_add_default_channel_mask (caps, ch,
|
|
channel_reorder_map))
|
|
GST_DEBUG ("using default channel layout for %d channels", ch);
|
|
else
|
|
GST_WARNING ("failed to add channel layout");
|
|
} else {
|
|
GST_WARNING ("invalid block align %d or channel count %d", ba, ch);
|
|
return NULL;
|
|
}
|
|
} else {
|
|
/* FIXME: this is pretty useless - we need fixed caps */
|
|
caps = gst_caps_from_string ("audio/x-raw, "
|
|
"format = (string) { F32LE, F64LE }, "
|
|
"layout = (string) interleaved");
|
|
}
|
|
if (codec_name && strf)
|
|
*codec_name = g_strdup_printf ("Uncompressed %d-bit IEEE float audio",
|
|
strf->bits_per_sample);
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_IBM_CVSD:
|
|
goto unknown;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_ALAW:
|
|
if (strf != NULL) {
|
|
if (strf->bits_per_sample != 8) {
|
|
GST_WARNING ("invalid depth (%d) of alaw audio, overwriting.",
|
|
strf->bits_per_sample);
|
|
strf->bits_per_sample = 8;
|
|
strf->blockalign = (strf->bits_per_sample * strf->channels) / 8;
|
|
strf->av_bps = strf->blockalign * strf->rate;
|
|
}
|
|
if (strf->av_bps == 0 || strf->blockalign == 0) {
|
|
GST_WARNING ("fixing av_bps (%d) and blockalign (%d) of alaw audio",
|
|
strf->av_bps, strf->blockalign);
|
|
strf->blockalign = (strf->bits_per_sample * strf->channels) / 8;
|
|
strf->av_bps = strf->blockalign * strf->rate;
|
|
}
|
|
}
|
|
caps = gst_caps_new_empty_simple ("audio/x-alaw");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("A-law audio");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_WMS:
|
|
caps = gst_caps_new_empty_simple ("audio/x-wms");
|
|
if (strf != NULL) {
|
|
gst_caps_set_simple (caps,
|
|
"bitrate", G_TYPE_INT, strf->av_bps * 8,
|
|
"width", G_TYPE_INT, strf->bits_per_sample,
|
|
"depth", G_TYPE_INT, strf->bits_per_sample, NULL);
|
|
} else {
|
|
gst_caps_set_simple (caps,
|
|
"bitrate", GST_TYPE_INT_RANGE, 0, G_MAXINT, NULL);
|
|
}
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Windows Media Audio Speech");
|
|
block_align = TRUE;
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_MULAW:
|
|
if (strf != NULL) {
|
|
if (strf->bits_per_sample != 8) {
|
|
GST_WARNING ("invalid depth (%d) of mulaw audio, overwriting.",
|
|
strf->bits_per_sample);
|
|
strf->bits_per_sample = 8;
|
|
strf->blockalign = (strf->bits_per_sample * strf->channels) / 8;
|
|
strf->av_bps = strf->blockalign * strf->rate;
|
|
}
|
|
if (strf->av_bps == 0 || strf->blockalign == 0) {
|
|
GST_WARNING ("fixing av_bps (%d) and blockalign (%d) of mulaw audio",
|
|
strf->av_bps, strf->blockalign);
|
|
strf->blockalign = (strf->bits_per_sample * strf->channels) / 8;
|
|
strf->av_bps = strf->blockalign * strf->rate;
|
|
}
|
|
}
|
|
caps = gst_caps_new_empty_simple ("audio/x-mulaw");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Mu-law audio");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_OKI_ADPCM:
|
|
goto unknown;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_DVI_ADPCM:
|
|
if (strf != NULL) {
|
|
/* Many encoding tools create a wrong bitrate information in the
|
|
* header, so either we calculate the bitrate or mark it as invalid
|
|
* as this would probably confuse timing */
|
|
strf->av_bps = 0;
|
|
if (strf->channels != 0 && strf->rate != 0 && strf->blockalign != 0) {
|
|
int spb = ((strf->blockalign - strf->channels * 4) / 2) * 2;
|
|
strf->av_bps =
|
|
gst_util_uint64_scale_int (strf->rate, strf->blockalign, spb);
|
|
GST_DEBUG ("fixing av_bps to calculated value %d of IMA DVI ADPCM",
|
|
strf->av_bps);
|
|
}
|
|
}
|
|
caps = gst_caps_new_simple ("audio/x-adpcm",
|
|
"layout", G_TYPE_STRING, "dvi", NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("DVI ADPCM audio");
|
|
block_align = TRUE;
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_ADPCM_G722:
|
|
caps = gst_caps_new_empty_simple ("audio/G722");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("G722 audio");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_ITU_G726_ADPCM:
|
|
if (strf != NULL) {
|
|
gint bitrate;
|
|
bitrate = 0;
|
|
if (strf->av_bps == 2000 || strf->av_bps == 3000 || strf->av_bps == 4000
|
|
|| strf->av_bps == 5000) {
|
|
strf->blockalign = strf->av_bps / 1000;
|
|
bitrate = strf->av_bps * 8;
|
|
} else if (strf->blockalign >= 2 && strf->blockalign <= 5) {
|
|
bitrate = strf->blockalign * 8000;
|
|
}
|
|
if (bitrate > 0) {
|
|
caps = gst_caps_new_simple ("audio/x-adpcm",
|
|
"layout", G_TYPE_STRING, "g726", "bitrate", G_TYPE_INT, bitrate,
|
|
NULL);
|
|
} else {
|
|
caps = gst_caps_new_simple ("audio/x-adpcm",
|
|
"layout", G_TYPE_STRING, "g726", NULL);
|
|
}
|
|
} else {
|
|
caps = gst_caps_new_simple ("audio/x-adpcm",
|
|
"layout", G_TYPE_STRING, "g726", NULL);
|
|
}
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("G726 ADPCM audio");
|
|
block_align = TRUE;
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_DSP_TRUESPEECH:
|
|
caps = gst_caps_new_empty_simple ("audio/x-truespeech");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("DSP Group TrueSpeech");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_GSM610:
|
|
case GST_RIFF_WAVE_FORMAT_MSN:
|
|
caps = gst_caps_new_empty_simple ("audio/ms-gsm");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("MS GSM audio");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_MPEGL12: /* mp1 or mp2 */
|
|
caps = gst_caps_new_simple ("audio/mpeg",
|
|
"mpegversion", G_TYPE_INT, 1, "layer", G_TYPE_INT, 2, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("MPEG-1 layer 2");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_MPEGL3: /* mp3 */
|
|
caps = gst_caps_new_simple ("audio/mpeg",
|
|
"mpegversion", G_TYPE_INT, 1, "layer", G_TYPE_INT, 3, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("MPEG-1 layer 3");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_AMR_NB: /* amr-nb */
|
|
caps = gst_caps_new_empty_simple ("audio/AMR");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("AMR Narrow Band (NB)");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_AMR_WB: /* amr-wb */
|
|
caps = gst_caps_new_empty_simple ("audio/AMR-WB");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("AMR Wide Band (WB)");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_VORBIS1: /* ogg/vorbis mode 1 */
|
|
case GST_RIFF_WAVE_FORMAT_VORBIS2: /* ogg/vorbis mode 2 */
|
|
case GST_RIFF_WAVE_FORMAT_VORBIS3: /* ogg/vorbis mode 3 */
|
|
case GST_RIFF_WAVE_FORMAT_VORBIS1PLUS: /* ogg/vorbis mode 1+ */
|
|
case GST_RIFF_WAVE_FORMAT_VORBIS2PLUS: /* ogg/vorbis mode 2+ */
|
|
case GST_RIFF_WAVE_FORMAT_VORBIS3PLUS: /* ogg/vorbis mode 3+ */
|
|
caps = gst_caps_new_empty_simple ("audio/x-vorbis");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Vorbis");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_A52:
|
|
caps = gst_caps_new_empty_simple ("audio/x-ac3");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("AC-3 audio");
|
|
break;
|
|
case GST_RIFF_WAVE_FORMAT_DTS:
|
|
caps = gst_caps_new_empty_simple ("audio/x-dts");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("DTS audio");
|
|
/* wavparse is not always able to specify rate/channels for DTS-in-wav */
|
|
rate_chan = FALSE;
|
|
break;
|
|
case GST_RIFF_WAVE_FORMAT_AAC:
|
|
case GST_RIFF_WAVE_FORMAT_AAC_AC:
|
|
case GST_RIFF_WAVE_FORMAT_AAC_pm:
|
|
{
|
|
caps = gst_caps_new_simple ("audio/mpeg",
|
|
"mpegversion", G_TYPE_INT, 4, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("MPEG-4 AAC audio");
|
|
break;
|
|
}
|
|
case GST_RIFF_WAVE_FORMAT_WMAV1:
|
|
case GST_RIFF_WAVE_FORMAT_WMAV2:
|
|
case GST_RIFF_WAVE_FORMAT_WMAV3:
|
|
case GST_RIFF_WAVE_FORMAT_WMAV3_L:
|
|
{
|
|
gint version = (codec_id - GST_RIFF_WAVE_FORMAT_WMAV1) + 1;
|
|
|
|
block_align = TRUE;
|
|
|
|
caps = gst_caps_new_simple ("audio/x-wma",
|
|
"wmaversion", G_TYPE_INT, version, NULL);
|
|
|
|
if (codec_name) {
|
|
if (codec_id == GST_RIFF_WAVE_FORMAT_WMAV3_L)
|
|
*codec_name = g_strdup ("WMA Lossless");
|
|
else
|
|
*codec_name = g_strdup_printf ("WMA Version %d", version + 6);
|
|
}
|
|
|
|
if (strf != NULL) {
|
|
gst_caps_set_simple (caps,
|
|
"bitrate", G_TYPE_INT, strf->av_bps * 8,
|
|
"depth", G_TYPE_INT, strf->bits_per_sample, NULL);
|
|
} else {
|
|
gst_caps_set_simple (caps,
|
|
"bitrate", GST_TYPE_INT_RANGE, 0, G_MAXINT, NULL);
|
|
}
|
|
break;
|
|
}
|
|
case GST_RIFF_WAVE_FORMAT_SONY_ATRAC3:
|
|
caps = gst_caps_new_empty_simple ("audio/x-vnd.sony.atrac3");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Sony ATRAC3");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_SIREN:
|
|
caps = gst_caps_new_empty_simple ("audio/x-siren");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Siren7");
|
|
rate_chan = FALSE;
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_ADPCM_IMA_DK4:
|
|
caps =
|
|
gst_caps_new_simple ("audio/x-adpcm", "layout", G_TYPE_STRING, "dk4",
|
|
NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("IMA/DK4 ADPCM");
|
|
break;
|
|
case GST_RIFF_WAVE_FORMAT_ADPCM_IMA_DK3:
|
|
caps =
|
|
gst_caps_new_simple ("audio/x-adpcm", "layout", G_TYPE_STRING, "dk3",
|
|
NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("IMA/DK3 ADPCM");
|
|
break;
|
|
|
|
case GST_RIFF_WAVE_FORMAT_ADPCM_IMA_WAV:
|
|
caps =
|
|
gst_caps_new_simple ("audio/x-adpcm", "layout", G_TYPE_STRING, "dvi",
|
|
NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("IMA/WAV ADPCM");
|
|
break;
|
|
case GST_RIFF_WAVE_FORMAT_EXTENSIBLE:{
|
|
guint16 valid_bits_per_sample;
|
|
guint32 channel_mask;
|
|
guint32 subformat_guid[4];
|
|
GstMapInfo info;
|
|
gsize size;
|
|
|
|
if (strf_data == NULL) {
|
|
GST_WARNING ("WAVE_FORMAT_EXTENSIBLE but no strf_data buffer provided");
|
|
return NULL;
|
|
}
|
|
|
|
/* should be at least 22 bytes */
|
|
size = gst_buffer_get_size (strf_data);
|
|
|
|
if (size < 22) {
|
|
GST_WARNING ("WAVE_FORMAT_EXTENSIBLE data size is %" G_GSIZE_FORMAT
|
|
" (expected: 22)", size);
|
|
return NULL;
|
|
}
|
|
|
|
gst_buffer_map (strf_data, &info, GST_MAP_READ);
|
|
valid_bits_per_sample = GST_READ_UINT16_LE (info.data);
|
|
channel_mask = GST_READ_UINT32_LE (info.data + 2);
|
|
subformat_guid[0] = GST_READ_UINT32_LE (info.data + 6);
|
|
subformat_guid[1] = GST_READ_UINT32_LE (info.data + 10);
|
|
subformat_guid[2] = GST_READ_UINT32_LE (info.data + 14);
|
|
subformat_guid[3] = GST_READ_UINT32_LE (info.data + 18);
|
|
gst_buffer_unmap (strf_data, &info);
|
|
|
|
GST_DEBUG ("valid bps = %u", valid_bits_per_sample);
|
|
GST_DEBUG ("channel mask = 0x%08x", channel_mask);
|
|
GST_DEBUG ("GUID = %08x-%08x-%08x-%08x", subformat_guid[0],
|
|
subformat_guid[1], subformat_guid[2], subformat_guid[3]);
|
|
|
|
if (subformat_guid[1] == 0x00100000 &&
|
|
subformat_guid[2] == 0xaa000080 && subformat_guid[3] == 0x719b3800) {
|
|
if (subformat_guid[0] == 0x00000001) {
|
|
GST_DEBUG ("PCM");
|
|
if (strf != NULL && strf->blockalign != 0 && strf->channels != 0
|
|
&& strf->rate != 0) {
|
|
gint ba = strf->blockalign;
|
|
gint wd = ba * 8 / strf->channels;
|
|
gint ws;
|
|
GstAudioFormat format;
|
|
|
|
/* in riff, the depth is stored in the size field but it just
|
|
* means that the _least_ significant bits are cleared. We can
|
|
* therefore just play the sample as if it had a depth == width */
|
|
ws = wd;
|
|
|
|
/* For reference, use this to get the actual depth:
|
|
* ws = strf->bits_per_sample;
|
|
* if (valid_bits_per_sample != 0)
|
|
* ws = valid_bits_per_sample; */
|
|
|
|
format =
|
|
gst_audio_format_build_integer (wd != 8, G_LITTLE_ENDIAN, wd,
|
|
ws);
|
|
|
|
caps = gst_caps_new_simple ("audio/x-raw",
|
|
"format", G_TYPE_STRING, gst_audio_format_to_string (format),
|
|
"layout", G_TYPE_STRING, "interleaved",
|
|
"channels", G_TYPE_INT, strf->channels,
|
|
"rate", G_TYPE_INT, strf->rate, NULL);
|
|
|
|
if (codec_name) {
|
|
*codec_name = g_strdup_printf ("Uncompressed %d-bit PCM audio",
|
|
strf->bits_per_sample);
|
|
}
|
|
}
|
|
} else if (subformat_guid[0] == 0x00000003) {
|
|
GST_DEBUG ("FLOAT");
|
|
if (strf != NULL && strf->blockalign != 0 && strf->channels != 0
|
|
&& strf->rate != 0) {
|
|
gint ba = strf->blockalign;
|
|
gint wd = ba * 8 / strf->channels;
|
|
|
|
caps = gst_caps_new_simple ("audio/x-raw",
|
|
"format", G_TYPE_STRING, wd == 32 ? "F32LE" : "F64LE",
|
|
"layout", G_TYPE_STRING, "interleaved",
|
|
"channels", G_TYPE_INT, strf->channels,
|
|
"rate", G_TYPE_INT, strf->rate, NULL);
|
|
|
|
if (codec_name) {
|
|
*codec_name =
|
|
g_strdup_printf ("Uncompressed %d-bit IEEE float audio",
|
|
strf->bits_per_sample);
|
|
}
|
|
}
|
|
} else if (subformat_guid[0] == 0x0000006) {
|
|
GST_DEBUG ("ALAW");
|
|
if (strf != NULL) {
|
|
if (strf->bits_per_sample != 8) {
|
|
GST_WARNING ("invalid depth (%d) of alaw audio, overwriting.",
|
|
strf->bits_per_sample);
|
|
strf->bits_per_sample = 8;
|
|
strf->av_bps = 8;
|
|
strf->blockalign = strf->av_bps * strf->channels;
|
|
}
|
|
if (strf->av_bps == 0 || strf->blockalign == 0) {
|
|
GST_WARNING
|
|
("fixing av_bps (%d) and blockalign (%d) of alaw audio",
|
|
strf->av_bps, strf->blockalign);
|
|
strf->av_bps = strf->bits_per_sample;
|
|
strf->blockalign = strf->av_bps * strf->channels;
|
|
}
|
|
}
|
|
caps = gst_caps_new_empty_simple ("audio/x-alaw");
|
|
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("A-law audio");
|
|
} else if (subformat_guid[0] == 0x00000007) {
|
|
GST_DEBUG ("MULAW");
|
|
if (strf != NULL) {
|
|
if (strf->bits_per_sample != 8) {
|
|
GST_WARNING ("invalid depth (%d) of mulaw audio, overwriting.",
|
|
strf->bits_per_sample);
|
|
strf->bits_per_sample = 8;
|
|
strf->av_bps = 8;
|
|
strf->blockalign = strf->av_bps * strf->channels;
|
|
}
|
|
if (strf->av_bps == 0 || strf->blockalign == 0) {
|
|
GST_WARNING
|
|
("fixing av_bps (%d) and blockalign (%d) of mulaw audio",
|
|
strf->av_bps, strf->blockalign);
|
|
strf->av_bps = strf->bits_per_sample;
|
|
strf->blockalign = strf->av_bps * strf->channels;
|
|
}
|
|
}
|
|
caps = gst_caps_new_empty_simple ("audio/x-mulaw");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Mu-law audio");
|
|
} else if (subformat_guid[0] == 0x00000092) {
|
|
GST_DEBUG ("FIXME: handle DOLBY AC3 SPDIF format");
|
|
GST_DEBUG ("WAVE_FORMAT_EXTENSIBLE AC-3 SPDIF audio");
|
|
caps = gst_caps_new_empty_simple ("audio/x-ac3");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("wavext AC-3 SPDIF audio");
|
|
} else if ((subformat_guid[0] & 0xffff) ==
|
|
GST_RIFF_WAVE_FORMAT_EXTENSIBLE) {
|
|
GST_DEBUG ("WAVE_FORMAT_EXTENSIBLE nested");
|
|
} else {
|
|
/* recurse where no special consideration has yet to be identified
|
|
* for the subformat guid */
|
|
caps = gst_riff_create_audio_caps (subformat_guid[0], strh, strf,
|
|
strf_data, strd_data, codec_name, channel_reorder_map);
|
|
if (!codec_name)
|
|
GST_DEBUG ("WAVE_FORMAT_EXTENSIBLE audio");
|
|
if (caps) {
|
|
if (codec_name) {
|
|
GST_DEBUG ("WAVE_FORMAT_EXTENSIBLE %s", *codec_name);
|
|
*codec_name = g_strjoin ("wavext ", *codec_name, NULL);
|
|
}
|
|
return caps;
|
|
}
|
|
}
|
|
} else if (subformat_guid[0] == 0x6ba47966 &&
|
|
subformat_guid[1] == 0x41783f83 &&
|
|
subformat_guid[2] == 0xf0006596 && subformat_guid[3] == 0xe59262bf) {
|
|
caps = gst_caps_new_empty_simple ("application/x-ogg-avi");
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Ogg-AVI");
|
|
}
|
|
|
|
if (caps == NULL) {
|
|
GST_WARNING ("Unknown WAVE_FORMAT_EXTENSIBLE audio format");
|
|
return NULL;
|
|
}
|
|
|
|
if (strf != NULL) {
|
|
/* If channel_mask == 0 and channels > 1 let's
|
|
* assume default layout as some wav files don't have the
|
|
* channel mask set. Don't set the layout for 1 channel. */
|
|
if (channel_mask == 0 && strf->channels > 1)
|
|
channel_mask =
|
|
gst_riff_wavext_get_default_channel_mask (strf->channels);
|
|
|
|
if ((channel_mask != 0 || strf->channels > 1) &&
|
|
!gst_riff_wavext_add_channel_mask (caps, strf->channels,
|
|
channel_mask, channel_reorder_map)) {
|
|
GST_WARNING ("failed to add channel layout");
|
|
gst_caps_unref (caps);
|
|
caps = NULL;
|
|
}
|
|
rate_chan = FALSE;
|
|
}
|
|
|
|
break;
|
|
}
|
|
/* can anything decode these? pitfdll? */
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_AC8:
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_AC10:
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_AC16:
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_AC20:
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_METAVOICE:
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_METASOUND:
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_RT29HW:
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_VR12:
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_VR18:
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_TQ40:
|
|
case GST_RIFF_WAVE_FORMAT_VOXWARE_TQ60:{
|
|
caps = gst_caps_new_simple ("audio/x-voxware",
|
|
"voxwaretype", G_TYPE_INT, (gint) codec_id, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Voxware");
|
|
break;
|
|
}
|
|
default:
|
|
unknown:
|
|
GST_WARNING ("Unknown audio tag 0x%04x", codec_id);
|
|
return NULL;
|
|
}
|
|
|
|
if (strf != NULL) {
|
|
if (rate_chan) {
|
|
gst_caps_set_simple (caps,
|
|
"rate", G_TYPE_INT, strf->rate,
|
|
"channels", G_TYPE_INT, strf->channels, NULL);
|
|
}
|
|
if (block_align) {
|
|
gst_caps_set_simple (caps,
|
|
"block_align", G_TYPE_INT, strf->blockalign, NULL);
|
|
}
|
|
} else {
|
|
if (block_align) {
|
|
gst_caps_set_simple (caps,
|
|
"block_align", GST_TYPE_INT_RANGE, 1, G_MAXINT, NULL);
|
|
}
|
|
}
|
|
|
|
/* extradata */
|
|
if (strf_data || strd_data) {
|
|
gst_caps_set_simple (caps, "codec_data", GST_TYPE_BUFFER,
|
|
strf_data ? strf_data : strd_data, NULL);
|
|
}
|
|
|
|
return caps;
|
|
}
|
|
|
|
GstCaps *
|
|
gst_riff_create_iavs_caps (guint32 codec_fcc,
|
|
gst_riff_strh * strh, gst_riff_strf_iavs * strf,
|
|
GstBuffer * init_data, GstBuffer * extra_data, char **codec_name)
|
|
{
|
|
GstCaps *caps = NULL;
|
|
|
|
switch (codec_fcc) {
|
|
/* is this correct? */
|
|
case GST_MAKE_FOURCC ('D', 'V', 'S', 'D'):
|
|
case GST_MAKE_FOURCC ('d', 'v', 's', 'd'):
|
|
caps = gst_caps_new_simple ("video/x-dv",
|
|
"systemstream", G_TYPE_BOOLEAN, TRUE, NULL);
|
|
if (codec_name)
|
|
*codec_name = g_strdup ("Generic DV");
|
|
break;
|
|
|
|
default:
|
|
GST_WARNING ("Unknown IAVS fourcc %" GST_FOURCC_FORMAT,
|
|
GST_FOURCC_ARGS (codec_fcc));
|
|
return NULL;
|
|
}
|
|
|
|
return caps;
|
|
}
|
|
|
|
/*
|
|
* Functions below are for template caps. All is variable.
|
|
*/
|
|
|
|
GstCaps *
|
|
gst_riff_create_video_template_caps (void)
|
|
{
|
|
static const guint32 tags[] = {
|
|
GST_MAKE_FOURCC ('3', 'I', 'V', '1'),
|
|
GST_MAKE_FOURCC ('A', 'S', 'V', '1'),
|
|
GST_MAKE_FOURCC ('A', 'S', 'V', '2'),
|
|
GST_MAKE_FOURCC ('C', 'L', 'J', 'R'),
|
|
GST_MAKE_FOURCC ('C', 'S', 'C', 'D'),
|
|
GST_MAKE_FOURCC ('C', 'Y', 'U', 'V'),
|
|
GST_MAKE_FOURCC ('D', 'I', 'B', ' '),
|
|
GST_MAKE_FOURCC ('D', 'I', 'V', '3'),
|
|
GST_MAKE_FOURCC ('D', 'I', 'V', 'X'),
|
|
GST_MAKE_FOURCC ('D', 'U', 'C', 'K'),
|
|
GST_MAKE_FOURCC ('D', 'V', 'S', 'D'),
|
|
GST_MAKE_FOURCC ('D', 'V', '5', '0'),
|
|
GST_MAKE_FOURCC ('D', 'X', '5', '0'),
|
|
GST_MAKE_FOURCC ('M', '4', 'C', 'C'),
|
|
GST_MAKE_FOURCC ('F', 'L', 'V', '1'),
|
|
GST_MAKE_FOURCC ('F', 'L', 'V', '4'),
|
|
GST_MAKE_FOURCC ('H', '2', '6', '3'),
|
|
GST_MAKE_FOURCC ('V', 'X', '1', 'K'),
|
|
GST_MAKE_FOURCC ('H', '2', '6', '4'),
|
|
GST_MAKE_FOURCC ('H', 'F', 'Y', 'U'),
|
|
GST_MAKE_FOURCC ('I', '2', '6', '3'),
|
|
GST_MAKE_FOURCC ('I', '4', '2', '0'),
|
|
GST_MAKE_FOURCC ('I', 'V', '3', '2'),
|
|
GST_MAKE_FOURCC ('I', 'V', '4', '1'),
|
|
GST_MAKE_FOURCC ('I', 'V', '5', '0'),
|
|
GST_MAKE_FOURCC ('L', '2', '6', '3'),
|
|
GST_MAKE_FOURCC ('L', '2', '6', '4'),
|
|
GST_MAKE_FOURCC ('M', '2', '6', '3'),
|
|
GST_MAKE_FOURCC ('M', '4', 'S', '2'),
|
|
GST_MAKE_FOURCC ('M', 'J', 'P', 'G'),
|
|
GST_MAKE_FOURCC ('M', 'P', '4', '2'),
|
|
GST_MAKE_FOURCC ('M', 'P', '4', '3'),
|
|
GST_MAKE_FOURCC ('M', 'P', 'E', 'G'),
|
|
GST_MAKE_FOURCC ('M', 'P', 'G', '2'),
|
|
GST_MAKE_FOURCC ('M', 'P', 'G', '4'),
|
|
GST_MAKE_FOURCC ('M', 'S', 'Z', 'H'),
|
|
GST_MAKE_FOURCC ('P', 'N', 'G', ' '),
|
|
GST_MAKE_FOURCC ('R', 'L', 'E', ' '),
|
|
GST_MAKE_FOURCC ('R', 'T', '2', '1'),
|
|
GST_MAKE_FOURCC ('S', 'P', '5', '3'),
|
|
GST_MAKE_FOURCC ('T', 'M', '2', '0'),
|
|
GST_MAKE_FOURCC ('T', 'S', 'C', 'C'),
|
|
GST_MAKE_FOURCC ('U', 'L', 'T', 'I'),
|
|
GST_MAKE_FOURCC ('U', 'Y', 'V', 'Y'),
|
|
GST_MAKE_FOURCC ('V', 'C', 'R', '1'),
|
|
GST_MAKE_FOURCC ('V', 'C', 'R', '2'),
|
|
GST_MAKE_FOURCC ('V', 'D', 'O', 'W'),
|
|
GST_MAKE_FOURCC ('V', 'I', 'V', 'O'),
|
|
GST_MAKE_FOURCC ('V', 'M', 'n', 'c'),
|
|
GST_MAKE_FOURCC ('V', 'P', '3', ' '),
|
|
GST_MAKE_FOURCC ('V', 'S', 'S', 'H'),
|
|
GST_MAKE_FOURCC ('W', 'M', 'V', '1'),
|
|
GST_MAKE_FOURCC ('W', 'M', 'V', '2'),
|
|
GST_MAKE_FOURCC ('W', 'M', 'V', '3'),
|
|
GST_MAKE_FOURCC ('X', 'V', 'I', 'D'),
|
|
GST_MAKE_FOURCC ('X', 'x', 'a', 'n'),
|
|
GST_MAKE_FOURCC ('Y', 'U', 'Y', '2'),
|
|
GST_MAKE_FOURCC ('Y', 'V', 'U', '9'),
|
|
GST_MAKE_FOURCC ('Z', 'L', 'I', 'B'),
|
|
GST_MAKE_FOURCC ('c', 'v', 'i', 'd'),
|
|
GST_MAKE_FOURCC ('h', '2', '6', '4'),
|
|
GST_MAKE_FOURCC ('m', 's', 'v', 'c'),
|
|
GST_MAKE_FOURCC ('x', '2', '6', '3'),
|
|
GST_MAKE_FOURCC ('d', 'r', 'a', 'c'),
|
|
GST_MAKE_FOURCC ('F', 'F', 'V', '1'),
|
|
GST_MAKE_FOURCC ('K', 'M', 'V', 'C'),
|
|
GST_MAKE_FOURCC ('V', 'P', '5', '0'),
|
|
GST_MAKE_FOURCC ('V', 'P', '6', '0'),
|
|
GST_MAKE_FOURCC ('V', 'P', '6', 'F'),
|
|
GST_MAKE_FOURCC ('V', 'P', '7', '0'),
|
|
GST_MAKE_FOURCC ('V', 'P', '8', '0'),
|
|
GST_MAKE_FOURCC ('L', 'M', '2', '0'),
|
|
GST_MAKE_FOURCC ('R', 'P', 'Z', 'A'),
|
|
GST_MAKE_FOURCC ('T', 'H', 'E', 'O'),
|
|
GST_MAKE_FOURCC ('F', 'P', 'S', '1'),
|
|
GST_MAKE_FOURCC ('A', 'A', 'S', 'C'),
|
|
GST_MAKE_FOURCC ('Y', 'V', '1', '2'),
|
|
GST_MAKE_FOURCC ('L', 'O', 'C', 'O'),
|
|
GST_MAKE_FOURCC ('Z', 'M', 'B', 'V'),
|
|
GST_MAKE_FOURCC ('v', '2', '1', '0'),
|
|
GST_MAKE_FOURCC ('r', '2', '1', '0'),
|
|
/* FILL ME */
|
|
};
|
|
guint i;
|
|
GstCaps *caps, *one;
|
|
|
|
caps = gst_caps_new_empty ();
|
|
for (i = 0; i < G_N_ELEMENTS (tags); i++) {
|
|
one = gst_riff_create_video_caps (tags[i], NULL, NULL, NULL, NULL, NULL);
|
|
if (one)
|
|
gst_caps_append (caps, one);
|
|
}
|
|
|
|
return caps;
|
|
}
|
|
|
|
GstCaps *
|
|
gst_riff_create_audio_template_caps (void)
|
|
{
|
|
static const guint16 tags[] = {
|
|
GST_RIFF_WAVE_FORMAT_GSM610,
|
|
GST_RIFF_WAVE_FORMAT_MPEGL3,
|
|
GST_RIFF_WAVE_FORMAT_MPEGL12,
|
|
GST_RIFF_WAVE_FORMAT_PCM,
|
|
GST_RIFF_WAVE_FORMAT_VORBIS1,
|
|
GST_RIFF_WAVE_FORMAT_A52,
|
|
GST_RIFF_WAVE_FORMAT_DTS,
|
|
GST_RIFF_WAVE_FORMAT_AAC,
|
|
GST_RIFF_WAVE_FORMAT_ALAW,
|
|
GST_RIFF_WAVE_FORMAT_MULAW,
|
|
GST_RIFF_WAVE_FORMAT_WMS,
|
|
GST_RIFF_WAVE_FORMAT_ADPCM,
|
|
GST_RIFF_WAVE_FORMAT_DVI_ADPCM,
|
|
GST_RIFF_WAVE_FORMAT_DSP_TRUESPEECH,
|
|
GST_RIFF_WAVE_FORMAT_WMAV1,
|
|
GST_RIFF_WAVE_FORMAT_WMAV2,
|
|
GST_RIFF_WAVE_FORMAT_WMAV3,
|
|
GST_RIFF_WAVE_FORMAT_SONY_ATRAC3,
|
|
GST_RIFF_WAVE_FORMAT_IEEE_FLOAT,
|
|
GST_RIFF_WAVE_FORMAT_VOXWARE_METASOUND,
|
|
GST_RIFF_WAVE_FORMAT_ADPCM_IMA_DK4,
|
|
GST_RIFF_WAVE_FORMAT_ADPCM_IMA_DK3,
|
|
GST_RIFF_WAVE_FORMAT_ADPCM_IMA_WAV,
|
|
GST_RIFF_WAVE_FORMAT_AMR_NB,
|
|
GST_RIFF_WAVE_FORMAT_AMR_WB,
|
|
GST_RIFF_WAVE_FORMAT_SIREN,
|
|
/* FILL ME */
|
|
};
|
|
guint i;
|
|
GstCaps *caps, *one;
|
|
|
|
caps = gst_caps_new_empty ();
|
|
for (i = 0; i < G_N_ELEMENTS (tags); i++) {
|
|
one =
|
|
gst_riff_create_audio_caps (tags[i], NULL, NULL, NULL, NULL, NULL,
|
|
NULL);
|
|
if (one)
|
|
gst_caps_append (caps, one);
|
|
}
|
|
one = gst_caps_new_empty_simple ("application/x-ogg-avi");
|
|
gst_caps_append (caps, one);
|
|
|
|
return caps;
|
|
}
|
|
|
|
GstCaps *
|
|
gst_riff_create_iavs_template_caps (void)
|
|
{
|
|
static const guint32 tags[] = {
|
|
GST_MAKE_FOURCC ('D', 'V', 'S', 'D')
|
|
/* FILL ME */
|
|
};
|
|
guint i;
|
|
GstCaps *caps, *one;
|
|
|
|
caps = gst_caps_new_empty ();
|
|
for (i = 0; i < G_N_ELEMENTS (tags); i++) {
|
|
one = gst_riff_create_iavs_caps (tags[i], NULL, NULL, NULL, NULL, NULL);
|
|
if (one)
|
|
gst_caps_append (caps, one);
|
|
}
|
|
|
|
return caps;
|
|
}
|