added TTA parser and decoder

Original commit message from CVS:
added TTA parser and decoder
This commit is contained in:
Arwed v. Merkatz 2004-11-26 23:10:28 +00:00
parent 90bf594c66
commit aeb09d35ab
11 changed files with 1474 additions and 0 deletions

View file

@ -1,3 +1,16 @@
2004-11-27 Arwed v. Merkatz <v.merkatz@gmx.net>
* configure.ac:
* gst/tta/Makefile.am:
* gst/tta/crc32.h:
* gst/tta/filters.h:
* gst/tta/gsttta.c:
* gst/tta/gstttadec.c:
* gst/tta/gstttadec.h:
* gst/tta/gstttaparse.c:
* gst/tta/gstttaparse.h:
* gst/tta/ttadec.h:
added TTA parser and decoder
2004-11-26 Ronald S. Bultje <rbultje@ronald.bitfreak.net>
* gst/playback/gstplaybasebin.c: (gst_play_base_bin_class_init),

View file

@ -383,6 +383,7 @@ GST_PLUGINS_ALL="\
synaesthesia \
tags \
tcp \
tta \
typefind \
udp \
vbidec \
@ -1903,6 +1904,7 @@ gst/switch/Makefile
gst/synaesthesia/Makefile
gst/tags/Makefile
gst/tcp/Makefile
gst/tta/Makefile
gst/typefind/Makefile
gst/udp/Makefile
gst/vbidec/Makefile

12
gst/tta/Makefile.am Normal file
View file

@ -0,0 +1,12 @@
plugin_LTLIBRARIES = libgsttta.la
libgsttta_la_SOURCES = gsttta.c \
gstttaparse.c \
gstttadec.c
noinst_HEADERS = gstttaparse.h gstttadec.h \
ttadec.h crc32.h filters.h
libgsttta_la_CFLAGS = $(GST_CFLAGS)
libgsttta_la_LIBADD = $(GST_LIBS)
libgsttta_la_LDFLAGS = $(GST_PLUGIN_LDFLAGS)

114
gst/tta/crc32.h Normal file
View file

@ -0,0 +1,114 @@
/*
* crc32.h
*
* Description: CRC32 functions
* Developed by: Alexander Djourik <sasha@iszf.irk.ru>
* Pavel Zhilin <pzh@iszf.irk.ru>
*
* Copyright (c) 1999-2004 Alexander Djourik. All rights reserved.
*
*/
/*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* Please see the file COPYING in this directory for full copyright
* information.
*/
#ifndef CRC32_H
#define CRC32_H
static const unsigned long crc32_table[256] = {
0x00000000, 0x77073096, 0xee0e612c, 0x990951ba,
0x076dc419, 0x706af48f, 0xe963a535, 0x9e6495a3,
0x0edb8832, 0x79dcb8a4, 0xe0d5e91e, 0x97d2d988,
0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, 0x90bf1d91,
0x1db71064, 0x6ab020f2, 0xf3b97148, 0x84be41de,
0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7,
0x136c9856, 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec,
0x14015c4f, 0x63066cd9, 0xfa0f3d63, 0x8d080df5,
0x3b6e20c8, 0x4c69105e, 0xd56041e4, 0xa2677172,
0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b,
0x35b5a8fa, 0x42b2986c, 0xdbbbc9d6, 0xacbcf940,
0x32d86ce3, 0x45df5c75, 0xdcd60dcf, 0xabd13d59,
0x26d930ac, 0x51de003a, 0xc8d75180, 0xbfd06116,
0x21b4f4b5, 0x56b3c423, 0xcfba9599, 0xb8bda50f,
0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924,
0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d,
0x76dc4190, 0x01db7106, 0x98d220bc, 0xefd5102a,
0x71b18589, 0x06b6b51f, 0x9fbfe4a5, 0xe8b8d433,
0x7807c9a2, 0x0f00f934, 0x9609a88e, 0xe10e9818,
0x7f6a0dbb, 0x086d3d2d, 0x91646c97, 0xe6635c01,
0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e,
0x6c0695ed, 0x1b01a57b, 0x8208f4c1, 0xf50fc457,
0x65b0d9c6, 0x12b7e950, 0x8bbeb8ea, 0xfcb9887c,
0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, 0xfbd44c65,
0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2,
0x4adfa541, 0x3dd895d7, 0xa4d1c46d, 0xd3d6f4fb,
0x4369e96a, 0x346ed9fc, 0xad678846, 0xda60b8d0,
0x44042d73, 0x33031de5, 0xaa0a4c5f, 0xdd0d7cc9,
0x5005713c, 0x270241aa, 0xbe0b1010, 0xc90c2086,
0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f,
0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4,
0x59b33d17, 0x2eb40d81, 0xb7bd5c3b, 0xc0ba6cad,
0xedb88320, 0x9abfb3b6, 0x03b6e20c, 0x74b1d29a,
0xead54739, 0x9dd277af, 0x04db2615, 0x73dc1683,
0xe3630b12, 0x94643b84, 0x0d6d6a3e, 0x7a6a5aa8,
0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1,
0xf00f9344, 0x8708a3d2, 0x1e01f268, 0x6906c2fe,
0xf762575d, 0x806567cb, 0x196c3671, 0x6e6b06e7,
0xfed41b76, 0x89d32be0, 0x10da7a5a, 0x67dd4acc,
0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5,
0xd6d6a3e8, 0xa1d1937e, 0x38d8c2c4, 0x4fdff252,
0xd1bb67f1, 0xa6bc5767, 0x3fb506dd, 0x48b2364b,
0xd80d2bda, 0xaf0a1b4c, 0x36034af6, 0x41047a60,
0xdf60efc3, 0xa867df55, 0x316e8eef, 0x4669be79,
0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236,
0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f,
0xc5ba3bbe, 0xb2bd0b28, 0x2bb45a92, 0x5cb36a04,
0xc2d7ffa7, 0xb5d0cf31, 0x2cd99e8b, 0x5bdeae1d,
0x9b64c2b0, 0xec63f226, 0x756aa39c, 0x026d930a,
0x9c0906a9, 0xeb0e363f, 0x72076785, 0x05005713,
0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38,
0x92d28e9b, 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21,
0x86d3d2d4, 0xf1d4e242, 0x68ddb3f8, 0x1fda836e,
0x81be16cd, 0xf6b9265b, 0x6fb077e1, 0x18b74777,
0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c,
0x8f659eff, 0xf862ae69, 0x616bffd3, 0x166ccf45,
0xa00ae278, 0xd70dd2ee, 0x4e048354, 0x3903b3c2,
0xa7672661, 0xd06016f7, 0x4969474d, 0x3e6e77db,
0xaed16a4a, 0xd9d65adc, 0x40df0b66, 0x37d83bf0,
0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9,
0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6,
0xbad03605, 0xcdd70693, 0x54de5729, 0x23d967bf,
0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94,
0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d
};
#define UPDATE_CRC32(x, crc) crc = \
(((crc>>8) & 0x00FFFFFF) ^ crc32_table[(crc^x) & 0xFF])
static unsigned long
crc32 (unsigned char *buffer, unsigned long len) {
unsigned long i;
unsigned long crc = 0xFFFFFFFF;
for (i = 0; i < len; i++) UPDATE_CRC32(buffer[i], crc);
return (crc ^ 0xFFFFFFFF);
}
#endif /* CRC32_H */

110
gst/tta/filters.h Normal file
View file

@ -0,0 +1,110 @@
/*
* filters.h
*
* Description: TTAv1 filter functions
* Developed by: Alexander Djourik <sasha@iszf.irk.ru>
* Pavel Zhilin <pzh@iszf.irk.ru>
*
* Copyright (c) 1999-2004 Alexander Djourik. All rights reserved.
*
*/
/*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* aint with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*
* Please see the file COPYING in this directory for full copyright
* information.
*/
#ifndef FILTERS_H
#define FILTERS_H
///////// Filter Settings //////////
static long flt_set[3] = {10, 9, 10};
static void
memshl (register long *pA, register long *pB) {
*pA++ = *pB++;
*pA++ = *pB++;
*pA++ = *pB++;
*pA++ = *pB++;
*pA++ = *pB++;
*pA++ = *pB++;
*pA++ = *pB++;
*pA = *pB;
}
static void
hybrid_filter (fltst *fs, long *in) {
register long *pA = fs->dl;
register long *pB = fs->qm;
register long *pM = fs->dx;
register long sum = fs->round;
if (!fs->error) {
sum += *pA++ * *pB, pB++;
sum += *pA++ * *pB, pB++;
sum += *pA++ * *pB, pB++;
sum += *pA++ * *pB, pB++;
sum += *pA++ * *pB, pB++;
sum += *pA++ * *pB, pB++;
sum += *pA++ * *pB, pB++;
sum += *pA++ * *pB, pB++; pM += 8;
} else if (fs->error < 0) {
sum += *pA++ * (*pB -= *pM++), pB++;
sum += *pA++ * (*pB -= *pM++), pB++;
sum += *pA++ * (*pB -= *pM++), pB++;
sum += *pA++ * (*pB -= *pM++), pB++;
sum += *pA++ * (*pB -= *pM++), pB++;
sum += *pA++ * (*pB -= *pM++), pB++;
sum += *pA++ * (*pB -= *pM++), pB++;
sum += *pA++ * (*pB -= *pM++), pB++;
} else {
sum += *pA++ * (*pB += *pM++), pB++;
sum += *pA++ * (*pB += *pM++), pB++;
sum += *pA++ * (*pB += *pM++), pB++;
sum += *pA++ * (*pB += *pM++), pB++;
sum += *pA++ * (*pB += *pM++), pB++;
sum += *pA++ * (*pB += *pM++), pB++;
sum += *pA++ * (*pB += *pM++), pB++;
sum += *pA++ * (*pB += *pM++), pB++;
}
*(pM-0) = ((*(pA-1) >> 30) | 1) << 2;
*(pM-1) = ((*(pA-2) >> 30) | 1) << 1;
*(pM-2) = ((*(pA-3) >> 30) | 1) << 1;
*(pM-3) = ((*(pA-4) >> 30) | 1);
fs->error = *in;
*in += (sum >> fs->shift);
*pA = *in;
*(pA-1) = *(pA-0) - *(pA-1);
*(pA-2) = *(pA-1) - *(pA-2);
*(pA-3) = *(pA-2) - *(pA-3);
memshl (fs->dl, fs->dl + 1);
memshl (fs->dx, fs->dx + 1);
}
static void
filter_init (fltst *fs, long shift) {
memset (fs, 0, sizeof(fltst));
fs->shift = shift;
fs->round = 1 << (shift - 1);
}
#endif /* FILTERS_H */

40
gst/tta/gsttta.c Normal file
View file

@ -0,0 +1,40 @@
/* GStreamer TTA plugin
* (c) 2004 Arwed v. Merkatz <v.merkatz@gmx.net>
*
* gsttta.c: plugin loader
*
* 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., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#include "gstttaparse.h"
#include "gstttadec.h"
static gboolean
plugin_init (GstPlugin * plugin)
{
return (gst_tta_parse_plugin_init (plugin) &&
gst_tta_dec_plugin_init (plugin));
}
GST_PLUGIN_DEFINE (GST_VERSION_MAJOR,
GST_VERSION_MINOR,
"tta",
"TTA lossless audio format handling",
plugin_init, VERSION, "LGPL", "gst-tta", "http://www.true-audio.com")

458
gst/tta/gstttadec.c Normal file
View file

@ -0,0 +1,458 @@
/* GStreamer TTA plugin
* (c) 2004 Arwed v. Merkatz <v.merkatz@gmx.net>
*
* based on ttalib
* (c) 1999-2004 Alexander Djourik <sasha@iszf.irk.ru>
*
* gstttadec.c: raw TTA bitstream decoder
*
* 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., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#include <gst/gst.h>
#include <math.h>
#include <string.h>
#include "gstttadec.h"
#include "ttadec.h"
#include "filters.h"
#define TTA_BUFFER_SIZE (1024 * 32 * 8)
/* this is from ttadec.h originally */
static const unsigned long bit_mask[] = {
0x00000000, 0x00000001, 0x00000003, 0x00000007,
0x0000000f, 0x0000001f, 0x0000003f, 0x0000007f,
0x000000ff, 0x000001ff, 0x000003ff, 0x000007ff,
0x00000fff, 0x00001fff, 0x00003fff, 0x00007fff,
0x0000ffff, 0x0001ffff, 0x0003ffff, 0x0007ffff,
0x000fffff, 0x001fffff, 0x003fffff, 0x007fffff,
0x00ffffff, 0x01ffffff, 0x03ffffff, 0x07ffffff,
0x0fffffff, 0x1fffffff, 0x3fffffff, 0x7fffffff,
0xffffffff
};
static const unsigned long bit_shift[] = {
0x00000001, 0x00000002, 0x00000004, 0x00000008,
0x00000010, 0x00000020, 0x00000040, 0x00000080,
0x00000100, 0x00000200, 0x00000400, 0x00000800,
0x00001000, 0x00002000, 0x00004000, 0x00008000,
0x00010000, 0x00020000, 0x00040000, 0x00080000,
0x00100000, 0x00200000, 0x00400000, 0x00800000,
0x01000000, 0x02000000, 0x04000000, 0x08000000,
0x10000000, 0x20000000, 0x40000000, 0x80000000,
0x80000000, 0x80000000, 0x80000000, 0x80000000,
0x80000000, 0x80000000, 0x80000000, 0x80000000
};
static const unsigned long *shift_16 = bit_shift + 4;
/* Filter signals and args */
enum
{
LAST_SIGNAL
};
enum
{
ARG_0
};
static GstStaticPadTemplate sink_factory = GST_STATIC_PAD_TEMPLATE ("sink",
GST_PAD_SINK,
GST_PAD_ALWAYS,
GST_STATIC_CAPS ("audio/x-tta, "
"width = (int) { 8, 16, 24 }, "
"channels = (int) { 1, 2 }, " "rate = (int) [ 8000, 96000 ]")
);
static GstStaticPadTemplate src_factory = GST_STATIC_PAD_TEMPLATE ("src",
GST_PAD_SRC,
GST_PAD_ALWAYS,
GST_STATIC_CAPS ("audio/x-raw-int, "
"width = (int) { 8, 16, 24 }, "
"depth = (int) { 8, 16, 24 }, "
"channels = (int) { 1, 2 }, "
"rate = (int) [ 8000, 96000 ], "
"endianness = (int) LITTLE_ENDIAN, " "signed = (boolean) true")
);
static void gst_tta_dec_class_init (GstTtaDecClass * klass);
static void gst_tta_dec_base_init (GstTtaDecClass * klass);
static void gst_tta_dec_init (GstTtaDec * ttadec);
static void gst_tta_dec_chain (GstPad * pad, GstData * in);
static GstElementClass *parent = NULL;
static GstPadLinkReturn
gst_tta_dec_link (GstPad * pad, const GstCaps * caps)
{
GstTtaDec *ttadec = GST_TTA_DEC (gst_pad_get_parent (pad));
GstStructure *structure = gst_caps_get_structure (caps, 0);
GstCaps *srccaps;
guint64 outsize;
guint bits;
if (!gst_caps_is_fixed (caps))
return GST_PAD_LINK_DELAYED;
gst_structure_get_int (structure, "rate", &ttadec->samplerate);
gst_structure_get_int (structure, "channels", &ttadec->channels);
gst_structure_get_int (structure, "width", &bits);
ttadec->bytes = bits / 8;
srccaps = gst_caps_new_simple ("audio/x-raw-int",
"rate", G_TYPE_INT, ttadec->samplerate,
"channels", G_TYPE_INT, ttadec->channels,
"depth", G_TYPE_INT, bits,
"width", G_TYPE_INT, bits,
"endianness", G_TYPE_INT, LITTLE_ENDIAN,
"signed", G_TYPE_BOOLEAN, TRUE, NULL);
gst_pad_set_explicit_caps (ttadec->srcpad, srccaps);
ttadec->frame_length = FRAME_TIME * ttadec->samplerate;
ttadec->tta = g_malloc (ttadec->channels * sizeof (decoder));
ttadec->cache = g_malloc (ttadec->channels * sizeof (long));
outsize = ttadec->channels * ttadec->frame_length * ttadec->bytes;
ttadec->decdata =
(guchar *) g_malloc (ttadec->channels * ttadec->frame_length *
ttadec->bytes * sizeof (guchar));
return GST_PAD_LINK_OK;
}
GType
gst_tta_dec_get_type (void)
{
static GType plugin_type = 0;
if (!plugin_type) {
static const GTypeInfo plugin_info = {
sizeof (GstTtaDecClass),
(GBaseInitFunc) gst_tta_dec_base_init,
NULL,
(GClassInitFunc) gst_tta_dec_class_init,
NULL,
NULL,
sizeof (GstTtaDec),
0,
(GInstanceInitFunc) gst_tta_dec_init,
};
plugin_type = g_type_register_static (GST_TYPE_ELEMENT,
"GstTtaDec", &plugin_info, 0);
}
return plugin_type;
}
static void
gst_tta_dec_base_init (GstTtaDecClass * klass)
{
static GstElementDetails plugin_details = {
"TTA decoder",
"Codec/Decoder/Audio",
"Decode TTA audio data",
"Arwed v. Merkatz <v.merkatz@gmx.net>"
};
GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
gst_element_class_add_pad_template (element_class,
gst_static_pad_template_get (&src_factory));
gst_element_class_add_pad_template (element_class,
gst_static_pad_template_get (&sink_factory));
gst_element_class_set_details (element_class, &plugin_details);
}
static void
gst_tta_dec_dispose (GObject * object)
{
GstTtaDec *ttadec = GST_TTA_DEC (object);
g_free (ttadec->tta);
g_free (ttadec->decdata);
g_free (ttadec->tta_buf.buffer);
G_OBJECT_CLASS (parent)->dispose (object);
}
static void
gst_tta_dec_class_init (GstTtaDecClass * klass)
{
GObjectClass *gobject_class;
GstElementClass *gstelement_class;
gobject_class = (GObjectClass *) klass;
gstelement_class = (GstElementClass *) klass;
parent = g_type_class_ref (GST_TYPE_ELEMENT);
gobject_class->dispose = gst_tta_dec_dispose;
}
static void
gst_tta_dec_init (GstTtaDec * ttadec)
{
GstElementClass *klass = GST_ELEMENT_GET_CLASS (ttadec);
ttadec->sinkpad =
gst_pad_new_from_template (gst_element_class_get_pad_template (klass,
"sink"), "sink");
gst_pad_set_link_function (ttadec->sinkpad, gst_tta_dec_link);
ttadec->srcpad =
gst_pad_new_from_template (gst_element_class_get_pad_template (klass,
"src"), "src");
gst_pad_use_explicit_caps (ttadec->srcpad);
gst_element_add_pad (GST_ELEMENT (ttadec), ttadec->sinkpad);
gst_element_add_pad (GST_ELEMENT (ttadec), ttadec->srcpad);
gst_pad_set_chain_function (ttadec->sinkpad, gst_tta_dec_chain);
ttadec->tta_buf.buffer = (guchar *) g_malloc (TTA_BUFFER_SIZE + 4);
ttadec->tta_buf.buffer_end = ttadec->tta_buf.buffer + TTA_BUFFER_SIZE;
GST_FLAG_SET (ttadec, GST_ELEMENT_EVENT_AWARE);
}
void
rice_init (adapt * rice, unsigned long k0, unsigned long k1)
{
rice->k0 = k0;
rice->k1 = k1;
rice->sum0 = shift_16[k0];
rice->sum1 = shift_16[k1];
}
void
decoder_init (decoder * tta, long nch, long byte_size)
{
long shift = flt_set[byte_size - 1];
long i;
for (i = 0; i < nch; i++) {
filter_init (&tta[i].fst, shift);
rice_init (&tta[i].rice, 10, 10);
tta[i].last = 0;
}
}
void
get_binary (tta_buffer * tta_buf, guchar * buffer, unsigned long buffersize,
unsigned long *value, unsigned long bits)
{
while (tta_buf->bit_count < bits) {
if (tta_buf->bitpos == tta_buf->buffer_end) {
int max =
TTA_BUFFER_SIZE <=
buffersize - tta_buf->offset ? TTA_BUFFER_SIZE : buffersize -
tta_buf->offset;
memcpy (tta_buf->buffer, buffer + tta_buf->offset, max);
tta_buf->offset += max;
tta_buf->bitpos = tta_buf->buffer;
}
tta_buf->bit_cache |= *tta_buf->bitpos << tta_buf->bit_count;
tta_buf->bit_count += 8;
tta_buf->bitpos++;
}
*value = tta_buf->bit_cache & bit_mask[bits];
tta_buf->bit_cache >>= bits;
tta_buf->bit_count -= bits;
tta_buf->bit_cache &= bit_mask[tta_buf->bit_count];
}
void
get_unary (tta_buffer * tta_buf, guchar * buffer, unsigned long buffersize,
unsigned long *value)
{
*value = 0;
while (!(tta_buf->bit_cache ^ bit_mask[tta_buf->bit_count])) {
if (tta_buf->bitpos == tta_buf->buffer_end) {
int max =
TTA_BUFFER_SIZE <=
buffersize - tta_buf->offset ? TTA_BUFFER_SIZE : buffersize -
tta_buf->offset;
memcpy (tta_buf->buffer, buffer + tta_buf->offset, max);
tta_buf->offset += max;
tta_buf->bitpos = tta_buf->buffer;
}
*value += tta_buf->bit_count;
tta_buf->bit_cache = *tta_buf->bitpos++;
tta_buf->bit_count = 8;
}
while (tta_buf->bit_cache & 1) {
(*value)++;
tta_buf->bit_cache >>= 1;
tta_buf->bit_count--;
}
tta_buf->bit_cache >>= 1;
tta_buf->bit_count--;
}
static void
gst_tta_dec_chain (GstPad * pad, GstData * in)
{
GstTtaDec *ttadec;
GstBuffer *outbuf, *buf = GST_BUFFER (in);
guchar *data, *p;
decoder *dec;
unsigned long outsize;
unsigned long size;
guint32 frame_samples;
long res;
long *prev;
g_return_if_fail (GST_IS_PAD (pad));
g_return_if_fail (buf != NULL);
ttadec = GST_TTA_DEC (GST_OBJECT_PARENT (pad));
g_return_if_fail (GST_IS_TTA_DEC (ttadec));
if (GST_IS_EVENT (buf)) {
gst_pad_event_default (pad, GST_EVENT (buf));
return;
}
data = GST_BUFFER_DATA (buf);
size = GST_BUFFER_SIZE (buf);
ttadec->tta_buf.bit_count = 0;
ttadec->tta_buf.bit_cache = 0;
ttadec->tta_buf.bitpos = ttadec->tta_buf.buffer_end;
ttadec->tta_buf.offset = 0;
decoder_init (ttadec->tta, ttadec->channels, ttadec->bytes);
if (GST_BUFFER_DURATION_IS_VALID (buf)) {
frame_samples =
ceil ((gdouble) (GST_BUFFER_DURATION (buf) * ttadec->samplerate) /
(gdouble) GST_SECOND);
} else {
frame_samples = ttadec->samplerate * FRAME_TIME;
}
outsize = ttadec->channels * frame_samples * ttadec->bytes;
dec = ttadec->tta;
p = ttadec->decdata;
prev = ttadec->cache;
for (res = 0;
p < ttadec->decdata + frame_samples * ttadec->channels * ttadec->bytes;) {
unsigned long unary, binary, depth, k;
long value, temp_value;
fltst *fst = &dec->fst;
adapt *rice = &dec->rice;
long *last = &dec->last;
// decode Rice unsigned
get_unary (&ttadec->tta_buf, data, size, &unary);
switch (unary) {
case 0:
depth = 0;
k = rice->k0;
break;
default:
depth = 1;
k = rice->k1;
unary--;
}
if (k) {
get_binary (&ttadec->tta_buf, data, size, &binary, k);
value = (unary << k) + binary;
} else
value = unary;
switch (depth) {
case 1:
rice->sum1 += value - (rice->sum1 >> 4);
if (rice->k1 > 0 && rice->sum1 < shift_16[rice->k1])
rice->k1--;
else if (rice->sum1 > shift_16[rice->k1 + 1])
rice->k1++;
value += bit_shift[rice->k0];
default:
rice->sum0 += value - (rice->sum0 >> 4);
if (rice->k0 > 0 && rice->sum0 < shift_16[rice->k0])
rice->k0--;
else if (rice->sum0 > shift_16[rice->k0 + 1])
rice->k0++;
}
/* this only uses a temporary variable to silence a gcc warning */
temp_value = DEC (value);
value = temp_value;
// decompress stage 1: adaptive hybrid filter
hybrid_filter (fst, &value);
// decompress stage 2: fixed order 1 prediction
switch (ttadec->bytes) {
case 1:
value += PREDICTOR1 (*last, 4);
break; // bps 8
case 2:
value += PREDICTOR1 (*last, 5);
break; // bps 16
case 3:
value += PREDICTOR1 (*last, 5);
break; // bps 24
case 4:
value += *last;
break; // bps 32
}
*last = value;
if (dec < ttadec->tta + ttadec->channels - 1) {
*prev++ = value;
dec++;
} else {
*prev = value;
if (ttadec->channels > 1) {
long *r = prev - 1;
for (*prev += *r / 2; r >= ttadec->cache; r--)
*r = *(r + 1) - *r;
for (r = ttadec->cache; r < prev; r++)
WRITE_BUFFER (r, ttadec->bytes, p);
}
WRITE_BUFFER (prev, ttadec->bytes, p);
prev = ttadec->cache;
res++;
dec = ttadec->tta;
}
}
outbuf = gst_buffer_new_and_alloc (outsize);
memcpy (GST_BUFFER_DATA (outbuf), ttadec->decdata, outsize);
GST_BUFFER_TIMESTAMP (outbuf) = GST_BUFFER_TIMESTAMP (buf);
GST_BUFFER_DURATION (outbuf) = GST_BUFFER_DURATION (buf);
gst_pad_push (ttadec->srcpad, GST_DATA (outbuf));
}
gboolean
gst_tta_dec_plugin_init (GstPlugin * plugin)
{
return gst_element_register (plugin, "ttadec",
GST_RANK_PRIMARY, GST_TYPE_TTA_DEC);
}

84
gst/tta/gstttadec.h Normal file
View file

@ -0,0 +1,84 @@
/* GStreamer TTA plugin
* (c) 2004 Arwed v. Merkatz <v.merkatz@gmx.net>
*
* gstttadec.h: raw TTA bitstream decoder
*
* 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., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#ifndef __GST_TTA_DEC_H__
#define __GST_TTA_DEC_H__
#include <gst/gst.h>
#include "ttadec.h"
G_BEGIN_DECLS
/* #define's don't like whitespacey bits */
#define GST_TYPE_TTA_DEC \
(gst_tta_dec_get_type())
#define GST_TTA_DEC(obj) \
(G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_TTA_DEC,GstTtaDec))
#define GST_TTA_DEC_CLASS(klass) \
(G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_TTA_DEC,GstTtaDec))
#define GST_IS_TTA_DEC(obj) \
(G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_TTA_DEC))
#define GST_IS_TTA_DEC_CLASS(obj) \
(G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_TTA_DEC))
typedef struct _GstTtaDec GstTtaDec;
typedef struct _GstTtaDecClass GstTtaDecClass;
typedef struct _tta_buffer
{
guchar *buffer;
guchar *buffer_end;
gulong bit_count;
gulong bit_cache;
guchar *bitpos;
gulong offset;
} tta_buffer;
struct _GstTtaDec
{
GstElement element;
GstPad *sinkpad, *srcpad;
guint32 samplerate;
guint channels;
guint bytes;
long frame_length;
decoder *tta;
guchar *decdata;
tta_buffer tta_buf;
long *cache;
};
struct _GstTtaDecClass
{
GstElementClass parent;
};
GType gst_tta_dec_get_type (void);
gboolean gst_tta_dec_plugin_init (GstPlugin *plugin);
G_END_DECLS
#endif /* __GST_TTA_DEC_H__ */

418
gst/tta/gstttaparse.c Normal file
View file

@ -0,0 +1,418 @@
/* GStreamer TTA plugin
* (c) 2004 Arwed v. Merkatz <v.merkatz@gmx.net>
*
* gstttaparse.c: TTA file parser
*
* 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., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#include <gst/gst.h>
#include <math.h>
#include "gstttaparse.h"
#include "ttadec.h"
#include "crc32.h"
GST_DEBUG_CATEGORY_STATIC (gst_tta_parse_debug);
#define GST_CAT_DEFAULT gst_tta_parse_debug
/* Filter signals and args */
enum
{
LAST_SIGNAL
};
enum
{
ARG_0
};
static GstStaticPadTemplate sink_factory = GST_STATIC_PAD_TEMPLATE ("sink",
GST_PAD_SINK,
GST_PAD_ALWAYS,
GST_STATIC_CAPS ("audio/x-ttafile")
);
static GstStaticPadTemplate src_factory = GST_STATIC_PAD_TEMPLATE ("src",
GST_PAD_SRC,
GST_PAD_ALWAYS,
GST_STATIC_CAPS ("audio/x-tta, "
"width = (int) { 8, 16, 24 }, "
"channels = (int) { 1, 2 }, " "rate = (int) [ 8000, 96000 ]")
);
static void gst_tta_parse_class_init (GstTtaParseClass * klass);
static void gst_tta_parse_base_init (GstTtaParseClass * klass);
static void gst_tta_parse_init (GstTtaParse * ttaparse);
static void gst_tta_parse_chain (GstPad * pad, GstData * in);
static GstElementClass *parent = NULL;
GType
gst_tta_parse_get_type (void)
{
static GType plugin_type = 0;
if (!plugin_type) {
static const GTypeInfo plugin_info = {
sizeof (GstTtaParseClass),
(GBaseInitFunc) gst_tta_parse_base_init,
NULL,
(GClassInitFunc) gst_tta_parse_class_init,
NULL,
NULL,
sizeof (GstTtaParse),
0,
(GInstanceInitFunc) gst_tta_parse_init,
};
plugin_type = g_type_register_static (GST_TYPE_ELEMENT,
"GstTtaParse", &plugin_info, 0);
}
return plugin_type;
}
static void
gst_tta_parse_base_init (GstTtaParseClass * klass)
{
static GstElementDetails plugin_details = {
"TTA file parser",
"Codec/Demuxer/Audio",
"Parses TTA files",
"Arwed v. Merkatz <v.merkatz@gmx.net>"
};
GstElementClass *element_class = GST_ELEMENT_CLASS (klass);
gst_element_class_add_pad_template (element_class,
gst_static_pad_template_get (&src_factory));
gst_element_class_add_pad_template (element_class,
gst_static_pad_template_get (&sink_factory));
gst_element_class_set_details (element_class, &plugin_details);
}
static void
gst_tta_parse_dispose (GObject * object)
{
GstTtaParse *ttaparse = GST_TTA_PARSE (object);
g_free (ttaparse->index);
G_OBJECT_CLASS (parent)->dispose (object);
}
static void
gst_tta_parse_class_init (GstTtaParseClass * klass)
{
GObjectClass *gobject_class;
GstElementClass *gstelement_class;
gobject_class = (GObjectClass *) klass;
gstelement_class = (GstElementClass *) klass;
parent = g_type_class_ref (GST_TYPE_ELEMENT);
gobject_class->dispose = gst_tta_parse_dispose;
}
static gboolean
gst_tta_src_query (GstPad * pad, GstQueryType type,
GstFormat * format, gint64 * value)
{
GstTtaParse *ttaparse = GST_TTA_PARSE (gst_pad_get_parent (pad));
if (type == GST_QUERY_TOTAL) {
if (*format == GST_FORMAT_TIME) {
if ((ttaparse->data_length == 0) || (ttaparse->samplerate == 0)) {
*value = 0;
return FALSE;
}
*value =
((gdouble) ttaparse->data_length / (gdouble) ttaparse->samplerate) *
GST_SECOND;
GST_DEBUG_OBJECT (ttaparse, "got queried for time, returned %lli",
*value);
return TRUE;
}
} else {
return gst_pad_query_default (pad, type, format, value);
}
return FALSE;
}
static gboolean
gst_tta_src_event (GstPad * pad, GstEvent * event)
{
GstTtaParse *ttaparse = GST_TTA_PARSE (gst_pad_get_parent (pad));
switch (GST_EVENT_TYPE (event)) {
case GST_EVENT_SEEK:
{
if (GST_EVENT_SEEK_FORMAT (event) == GST_FORMAT_TIME) {
GST_DEBUG_OBJECT (ttaparse, "got seek event");
GstEvent *seek_event;
guint64 time = GST_EVENT_SEEK_OFFSET (event);
guint64 seek_frame = time / (FRAME_TIME * 1000000000);
guint64 seekpos = ttaparse->index[seek_frame].pos;
GST_DEBUG_OBJECT (ttaparse, "seeking to %u", (guint) seekpos);
seek_event =
gst_event_new_seek (GST_FORMAT_BYTES | GST_SEEK_METHOD_SET |
GST_SEEK_FLAG_ACCURATE, seekpos);
gst_event_unref (event);
if (gst_pad_send_event (GST_PAD_PEER (ttaparse->sinkpad), seek_event)) {
gst_pad_event_default (ttaparse->srcpad,
gst_event_new (GST_EVENT_FLUSH));
return TRUE;
} else {
GST_LOG_OBJECT (ttaparse, "seek failed");
return FALSE;
}
} else {
return gst_pad_send_event (pad, event);
}
break;
}
default:
return gst_pad_send_event (pad, event);
break;
}
}
static void
gst_tta_parse_init (GstTtaParse * ttaparse)
{
GstElementClass *klass = GST_ELEMENT_GET_CLASS (ttaparse);
ttaparse->sinkpad =
gst_pad_new_from_template (gst_element_class_get_pad_template (klass,
"sink"), "sink");
ttaparse->srcpad =
gst_pad_new_from_template (gst_element_class_get_pad_template (klass,
"src"), "src");
gst_pad_use_explicit_caps (ttaparse->srcpad);
gst_pad_set_query_function (ttaparse->srcpad, gst_tta_src_query);
gst_pad_set_event_function (ttaparse->srcpad, gst_tta_src_event);
gst_element_add_pad (GST_ELEMENT (ttaparse), ttaparse->sinkpad);
gst_element_add_pad (GST_ELEMENT (ttaparse), ttaparse->srcpad);
gst_pad_set_chain_function (ttaparse->sinkpad, gst_tta_parse_chain);
ttaparse->silent = FALSE;
ttaparse->header_parsed = FALSE;
ttaparse->partialbuf = NULL;
ttaparse->seek_ok = FALSE;
ttaparse->current_frame = 0;
ttaparse->data_length = 0;
ttaparse->samplerate = 0;
GST_FLAG_SET (ttaparse, GST_ELEMENT_EVENT_AWARE);
}
static void
gst_tta_handle_event (GstPad * pad, GstBuffer * buffer)
{
GstEvent *event = GST_EVENT (buffer);
GstTtaParse *ttaparse = GST_TTA_PARSE (gst_pad_get_parent (pad));
GST_DEBUG_OBJECT (ttaparse, "got some event");
switch (GST_EVENT_TYPE (event)) {
case GST_EVENT_DISCONTINUOUS:
{
GstEvent *discont;
guint64 offset = GST_EVENT_DISCONT_OFFSET (event, 0).value;
int i;
GST_DEBUG_OBJECT (ttaparse, "discont with offset: %u", offset);
for (i = 0; i < ttaparse->num_frames; i++) {
if (offset == ttaparse->index[i].pos) {
GST_DEBUG_OBJECT (ttaparse, "setting current frame to %i", i);
discont = gst_event_new_discontinuous (FALSE,
GST_FORMAT_TIME, ttaparse->index[i].time, NULL);
gst_event_unref (event);
gst_buffer_unref (ttaparse->partialbuf);
ttaparse->partialbuf = NULL;
ttaparse->current_frame = i;
gst_pad_event_default (pad, gst_event_new (GST_EVENT_FLUSH));
gst_pad_event_default (pad, discont);
GST_DEBUG_OBJECT (ttaparse, "sent discont event");
return;
}
}
break;
}
default:
gst_pad_event_default (pad, event);
break;
}
}
static void
gst_tta_parse_chain (GstPad * pad, GstData * in)
{
GstTtaParse *ttaparse;
GstBuffer *outbuf, *buf = GST_BUFFER (in);
guchar *data;
gint i;
guint64 size, offset = 0;
GstCaps *caps;
g_return_if_fail (GST_IS_PAD (pad));
g_return_if_fail (buf != NULL);
ttaparse = GST_TTA_PARSE (GST_OBJECT_PARENT (pad));
g_return_if_fail (GST_IS_TTA_PARSE (ttaparse));
if (GST_IS_EVENT (buf)) {
gst_tta_handle_event (pad, buf);
return;
}
if (ttaparse->partialbuf) {
GstBuffer *newbuf;
newbuf = gst_buffer_merge (ttaparse->partialbuf, buf);
gst_buffer_unref (buf);
gst_buffer_unref (ttaparse->partialbuf);
ttaparse->partialbuf = newbuf;
} else {
ttaparse->partialbuf = buf;
}
size = GST_BUFFER_SIZE (ttaparse->partialbuf);
data = GST_BUFFER_DATA (ttaparse->partialbuf);
if (!ttaparse->header_parsed) {
if ((*data == 'T') && (*(data + 1) == 'T') && (*(data + 2) == 'A')) {
double frame_length;
int num_frames;
guint32 datasize = 0;
guint32 crc;
offset = offset + 4;
offset = offset + 2;
ttaparse->channels = GST_READ_UINT16_LE (data + offset);
offset = offset + 2;
ttaparse->bits = GST_READ_UINT16_LE (data + offset);
offset += 2;
ttaparse->samplerate = GST_READ_UINT32_LE (data + offset);
frame_length = FRAME_TIME * ttaparse->samplerate;
offset += 4;
ttaparse->data_length = GST_READ_UINT32_LE (data + offset);
offset += 4;
num_frames = (ttaparse->data_length / frame_length) + 1;
crc = crc32 (data, 18);
if (crc != GST_READ_UINT32_LE (data + offset)) {
GST_WARNING_OBJECT (ttaparse, "Header CRC wrong!");
}
offset += 4;
GST_INFO_OBJECT (ttaparse,
"channels: %u, bits: %u, samplerate: %u, data_length: %u, num_frames: %u",
ttaparse->channels, ttaparse->bits, ttaparse->samplerate,
ttaparse->data_length, num_frames);
ttaparse->index =
(GstTtaIndex *) g_malloc (num_frames * sizeof (GstTtaIndex));
ttaparse->num_frames = num_frames;
for (i = 0; i < num_frames; i++) {
ttaparse->index[i].size = GST_READ_UINT32_LE (data + offset);
ttaparse->index[i].pos = GST_BUFFER_OFFSET (ttaparse->partialbuf) + (num_frames) * 4 + 4 + datasize + 22; // 22 == header size, +4 for the TTA1
ttaparse->index[i].time = i * FRAME_TIME * 1000000000;
offset += 4;
datasize += ttaparse->index[i].size;
}
GST_DEBUG_OBJECT (ttaparse, "Datasize: %u", datasize);
crc = crc32 (data + 22, num_frames * 4);
if (crc != GST_READ_UINT32_LE (data + offset)) {
GST_WARNING_OBJECT (ttaparse, "Seek table CRC wrong!");
} else {
ttaparse->seek_ok = TRUE;
/*
g_print("allowing seeking!\n");
g_print("dumping index:\n");
for (i = 0; i < ttaparse->num_frames; i++) {
g_print("frame %u: offset = %llu, time=%llu, size=%u\n",
i,
ttaparse->index[i].pos,
ttaparse->index[i].time,
ttaparse->index[i].size);
}
*/
}
offset += 4;
ttaparse->header_parsed = TRUE;
caps = gst_caps_new_simple ("audio/x-tta",
"width", G_TYPE_INT, ttaparse->bits,
"channels", G_TYPE_INT, ttaparse->channels,
"rate", G_TYPE_INT, ttaparse->samplerate, NULL);
gst_pad_set_explicit_caps (ttaparse->srcpad, caps);
}
}
i = ttaparse->current_frame;
while (size - offset >= ttaparse->index[i].size) {
guint32 crc;
crc = crc32 (data + offset, ttaparse->index[i].size - 4);
if (crc != GST_READ_UINT32_LE (data + offset + ttaparse->index[i].size - 4)) {
GST_WARNING_OBJECT (ttaparse, "Frame %u corrupted :(", i);
GST_WARNING_OBJECT (ttaparse, "calculated crc: %u, got crc: %u", crc,
GST_READ_UINT32_LE (data + offset + ttaparse->index[i].size - 4));
}
outbuf =
gst_buffer_create_sub (ttaparse->partialbuf, offset,
ttaparse->index[i].size - 4);
GST_BUFFER_TIMESTAMP (outbuf) = ttaparse->index[i].time;
if (ttaparse->current_frame + 1 == ttaparse->num_frames) {
guint32 samples =
ttaparse->data_length % (gint64) (ttaparse->samplerate * FRAME_TIME);
gdouble frametime = (gdouble) samples / (gdouble) ttaparse->samplerate;
GST_BUFFER_DURATION (outbuf) = (guint64) (frametime * GST_SECOND);
} else {
GST_BUFFER_DURATION (outbuf) = FRAME_TIME * 1000000000;
}
gst_pad_push (ttaparse->srcpad, GST_DATA (outbuf));
offset += ttaparse->index[i].size;
ttaparse->current_frame++;
i = ttaparse->current_frame;
}
if (size - offset > 0) {
glong remainder = size - offset;
outbuf = gst_buffer_create_sub (ttaparse->partialbuf, offset, remainder);
gst_buffer_unref (ttaparse->partialbuf);
ttaparse->partialbuf = outbuf;
} else {
gst_buffer_unref (ttaparse->partialbuf);
ttaparse->partialbuf = NULL;
}
}
gboolean
gst_tta_parse_plugin_init (GstPlugin * plugin)
{
if (!gst_element_register (plugin, "ttaparse",
GST_RANK_PRIMARY, GST_TYPE_TTA_PARSE)) {
return FALSE;
}
GST_DEBUG_CATEGORY_INIT (gst_tta_parse_debug, "ttaparse", 0,
"tta file parser");
return TRUE;
}

82
gst/tta/gstttaparse.h Normal file
View file

@ -0,0 +1,82 @@
/* GStreamer TTA plugin
* (c) 2004 Arwed v. Merkatz <v.merkatz@gmx.net>
*
* gstttaparse.h: TTA file parser
*
* 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., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#ifndef __GST_TTA_PARSE_H__
#define __GST_TTA_PARSE_H__
#include <gst/gst.h>
G_BEGIN_DECLS
/* #define's don't like whitespacey bits */
#define GST_TYPE_TTA_PARSE \
(gst_tta_parse_get_type())
#define GST_TTA_PARSE(obj) \
(G_TYPE_CHECK_INSTANCE_CAST((obj),GST_TYPE_TTA_PARSE,GstTtaParse))
#define GST_TTA_PARSE_CLASS(klass) \
(G_TYPE_CHECK_CLASS_CAST((klass),GST_TYPE_TTA_PARSE,GstTtaParse))
#define GST_IS_TTA_PARSE(obj) \
(G_TYPE_CHECK_INSTANCE_TYPE((obj),GST_TYPE_TTA_PARSE))
#define GST_IS_TTA_PARSE_CLASS(obj) \
(G_TYPE_CHECK_CLASS_TYPE((klass),GST_TYPE_TTA_PARSE))
typedef struct _GstTtaParse GstTtaParse;
typedef struct _GstTtaParseClass GstTtaParseClass;
typedef struct _GstTtaIndex {
guint32 frameno;
guint32 size; /* size of frame frameno */
guint64 pos; /* start of the frame */
guint64 time; /* in nanoseconds */
} GstTtaIndex;
struct _GstTtaParse
{
GstElement element;
GstPad *sinkpad, *srcpad;
gboolean silent;
gboolean header_parsed;
GstBuffer *partialbuf;
guint32 samplerate;
guint16 channels;
guint16 bits;
guint32 data_length;
GstTtaIndex *index;
guint num_frames;
gboolean seek_ok;
guint32 current_frame;
};
struct _GstTtaParseClass
{
GstElementClass parent;
};
GType gst_tta_parse_get_type (void);
gboolean gst_tta_parse_plugin_init (GstPlugin *plugin);
G_END_DECLS
#endif /* __GST_TTA_PARSE_H__ */

141
gst/tta/ttadec.h Normal file
View file

@ -0,0 +1,141 @@
/*
* ttadec.h
*
* Description: TTAv1 decoder definitions and prototypes
* Developed by: Alexander Djourik <sasha@iszf.irk.ru>
* Pavel Zhilin <pzh@iszf.irk.ru>
*
* Copyright (c) 1999-2004 Alexander Djourik. All rights reserved.
*
*/
/*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* Please see the file COPYING in this directory for full copyright
* information.
*/
#ifndef TTADEC_H_
#define TTADEC_H_
#ifdef _WIN32
#pragma pack(1)
#define __ATTRIBUTE_PACKED__
#else
#define __ATTRIBUTE_PACKED__ __attribute__((packed))
#endif
#define TTA1_SIGN 0x31415454
#define FRAME_TIME 1.04489795918367346939
#define MAX_ORDER 8
#ifndef WAVE_FORMAT_PCM
#define WAVE_FORMAT_PCM 1
#endif
#ifdef _WIN32
typedef unsigned __int64 uint64;
#else
typedef unsigned long long uint64;
#endif
/**
* moved to gstttadec.c to silence gcc warnings
*/
/*
static const unsigned long bit_mask[] = {
0x00000000, 0x00000001, 0x00000003, 0x00000007,
0x0000000f, 0x0000001f, 0x0000003f, 0x0000007f,
0x000000ff, 0x000001ff, 0x000003ff, 0x000007ff,
0x00000fff, 0x00001fff, 0x00003fff, 0x00007fff,
0x0000ffff, 0x0001ffff, 0x0003ffff, 0x0007ffff,
0x000fffff, 0x001fffff, 0x003fffff, 0x007fffff,
0x00ffffff, 0x01ffffff, 0x03ffffff, 0x07ffffff,
0x0fffffff, 0x1fffffff, 0x3fffffff, 0x7fffffff,
0xffffffff
};
static const unsigned long bit_shift[] = {
0x00000001, 0x00000002, 0x00000004, 0x00000008,
0x00000010, 0x00000020, 0x00000040, 0x00000080,
0x00000100, 0x00000200, 0x00000400, 0x00000800,
0x00001000, 0x00002000, 0x00004000, 0x00008000,
0x00010000, 0x00020000, 0x00040000, 0x00080000,
0x00100000, 0x00200000, 0x00400000, 0x00800000,
0x01000000, 0x02000000, 0x04000000, 0x08000000,
0x10000000, 0x20000000, 0x40000000, 0x80000000,
0x80000000, 0x80000000, 0x80000000, 0x80000000,
0x80000000, 0x80000000, 0x80000000, 0x80000000
};
static const unsigned long *shift_16 = bit_shift + 4;
*/
typedef unsigned char byte;
#ifdef _BIG_ENDIAN
#define ENDSWAP_INT16(x) (((((x)>>8)&0xFF)|(((x)&0xFF)<<8)))
#define ENDSWAP_INT32(x) (((((x)>>24)&0xFF)|(((x)>>8)&0xFF00)|(((x)&0xFF00)<<8)|(((x)&0xFF)<<24)))
#define WRITE_BUFFER(x, bsize, out) { \
if (bsize > 2) *out++ = (byte)(*x >> 16); \
if (bsize > 1) *out++ = (byte)(*x >> 8); \
*out++ = (byte) *x; }
#else
#define ENDSWAP_INT16(x) (x)
#define ENDSWAP_INT32(x) (x)
#define WRITE_BUFFER(x, bsize, out) { \
*out++ = (byte) *x; \
if (bsize > 1) *out++ = (byte)(*x >> 8); \
if (bsize > 2) *out++ = (byte)(*x >> 16); }
#endif
#define PREDICTOR1(x, k) ((long)((((uint64)x << k) - x) >> k))
#define DEC(x) (((x)&1)?(++(x)>>1):(-(x)>>1))
typedef struct {
unsigned long TTAid;
unsigned short AudioFormat;
unsigned short NumChannels;
unsigned short BitsPerSample;
unsigned long SampleRate;
unsigned long DataLength;
unsigned long CRC32;
} __ATTRIBUTE_PACKED__ tta_hdr;
typedef struct {
unsigned long k0;
unsigned long k1;
unsigned long sum0;
unsigned long sum1;
} adapt;
typedef struct {
long shift;
long round;
long error;
long mutex;
long qm[MAX_ORDER+1];
long dx[MAX_ORDER+1];
long dl[MAX_ORDER+1];
} fltst;
typedef struct {
fltst fst;
adapt rice;
long last;
} decoder;
#endif /* TTADEC_H_ */