2001-12-09 16:28:44 +01:00
|
|
|
/*
|
|
|
|
* bitstream.h
|
2006-06-16 00:58:06 +02:00
|
|
|
* Copyright (C) 2000-2002 Michel Lespinasse <walken@zoy.org>
|
2001-12-09 16:28:44 +01:00
|
|
|
* Copyright (C) 1999-2000 Aaron Holtzman <aholtzma@ess.engr.uvic.ca>
|
|
|
|
*
|
|
|
|
* This file is part of a52dec, a free ATSC A-52 stream decoder.
|
|
|
|
* See http://liba52.sourceforge.net/ for updates.
|
|
|
|
*
|
2005-03-23 00:27:18 +01:00
|
|
|
* Modified for use with MPlayer, changes contained in liba52_changes.diff.
|
2006-06-22 15:34:00 +02:00
|
|
|
* detailed changelog at http://svn.mplayerhq.hu/mplayer/trunk/
|
2005-03-23 00:27:18 +01:00
|
|
|
* $Id$
|
|
|
|
*
|
2001-12-09 16:28:44 +01:00
|
|
|
* a52dec 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.
|
|
|
|
*
|
|
|
|
* a52dec 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
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
2003-07-28 01:02:58 +02:00
|
|
|
/* code from ffmpeg/libavcodec */
|
|
|
|
#if defined(__GNUC__) && (__GNUC__ > 3 || __GNUC_ == 3 && __GNUC_MINOR__ > 0)
|
|
|
|
# define always_inline __attribute__((always_inline)) inline
|
|
|
|
#else
|
|
|
|
# define always_inline inline
|
|
|
|
#endif
|
|
|
|
|
2003-05-05 03:22:45 +02:00
|
|
|
#if defined(__sparc__) || defined(hpux)
|
2002-09-03 19:15:08 +02:00
|
|
|
/*
|
|
|
|
* the alt bitstream reader performs unaligned memory accesses; that doesn't work
|
2003-05-05 03:22:45 +02:00
|
|
|
* on sparc/hpux. For now, disable ALT_BITSTREAM_READER.
|
2002-09-03 19:15:08 +02:00
|
|
|
*/
|
|
|
|
#undef ALT_BITSTREAM_READER
|
|
|
|
#else
|
2001-12-17 22:53:49 +01:00
|
|
|
// alternative (faster) bitstram reader (reades upto 3 bytes over the end of the input)
|
|
|
|
#define ALT_BITSTREAM_READER
|
2003-05-08 01:25:25 +02:00
|
|
|
|
2007-07-28 16:28:38 +02:00
|
|
|
/* used to avoid misaligned exceptions on some archs (alpha, ...) */
|
2003-06-20 15:10:42 +02:00
|
|
|
#if defined (ARCH_X86) || defined(ARCH_ARMV4L)
|
2003-05-08 01:25:25 +02:00
|
|
|
# define unaligned32(a) (*(uint32_t*)(a))
|
|
|
|
#else
|
|
|
|
# ifdef __GNUC__
|
2003-07-28 01:02:58 +02:00
|
|
|
static always_inline uint32_t unaligned32(const void *v) {
|
2003-05-08 01:25:25 +02:00
|
|
|
struct Unaligned {
|
|
|
|
uint32_t i;
|
|
|
|
} __attribute__((packed));
|
|
|
|
|
|
|
|
return ((const struct Unaligned *) v)->i;
|
|
|
|
}
|
|
|
|
# elif defined(__DECC)
|
|
|
|
static inline uint32_t unaligned32(const void *v) {
|
|
|
|
return *(const __unaligned uint32_t *) v;
|
|
|
|
}
|
|
|
|
# else
|
|
|
|
static inline uint32_t unaligned32(const void *v) {
|
|
|
|
return *(const uint32_t *) v;
|
|
|
|
}
|
|
|
|
# endif
|
|
|
|
#endif //!ARCH_X86
|
|
|
|
|
2002-09-03 19:15:08 +02:00
|
|
|
#endif
|
2006-06-16 00:59:40 +02:00
|
|
|
|
2001-12-09 16:28:44 +01:00
|
|
|
/* (stolen from the kernel) */
|
|
|
|
#ifdef WORDS_BIGENDIAN
|
|
|
|
|
|
|
|
# define swab32(x) (x)
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
# if defined (__i386__)
|
|
|
|
|
|
|
|
# define swab32(x) __i386_swab32(x)
|
2006-06-16 00:58:06 +02:00
|
|
|
static inline const uint32_t __i386_swab32(uint32_t x)
|
2001-12-09 16:28:44 +01:00
|
|
|
{
|
|
|
|
__asm__("bswap %0" : "=r" (x) : "0" (x));
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
# else
|
|
|
|
|
2003-07-28 01:02:58 +02:00
|
|
|
# define swab32(x) __generic_swab32(x)
|
|
|
|
static always_inline const uint32_t __generic_swab32(uint32_t x)
|
|
|
|
{
|
|
|
|
return ((((uint8_t*)&x)[0] << 24) | (((uint8_t*)&x)[1] << 16) |
|
|
|
|
(((uint8_t*)&x)[2] << 8) | (((uint8_t*)&x)[3]));
|
|
|
|
}
|
2001-12-09 16:28:44 +01:00
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
|
2001-12-17 22:53:49 +01:00
|
|
|
#ifdef ALT_BITSTREAM_READER
|
|
|
|
extern int indx;
|
|
|
|
#endif
|
2001-12-09 16:28:44 +01:00
|
|
|
|
2006-06-16 00:58:06 +02:00
|
|
|
void a52_bitstream_set_ptr (a52_state_t * state, uint8_t * buf);
|
|
|
|
uint32_t a52_bitstream_get_bh (a52_state_t * state, uint32_t num_bits);
|
|
|
|
int32_t a52_bitstream_get_bh_2 (a52_state_t * state, uint32_t num_bits);
|
2001-12-17 22:53:49 +01:00
|
|
|
|
2006-06-16 00:58:06 +02:00
|
|
|
static inline uint32_t bitstream_get (a52_state_t * state, uint32_t num_bits)
|
2001-12-09 16:28:44 +01:00
|
|
|
{
|
2001-12-17 22:53:49 +01:00
|
|
|
#ifdef ALT_BITSTREAM_READER
|
2006-06-16 00:58:06 +02:00
|
|
|
uint32_t result= swab32( unaligned32(((uint8_t *)state->buffer_start)+(indx>>3)) );
|
2001-12-17 22:53:49 +01:00
|
|
|
|
|
|
|
result<<= (indx&0x07);
|
|
|
|
result>>= 32 - num_bits;
|
|
|
|
indx+= num_bits;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
#else
|
2001-12-09 16:28:44 +01:00
|
|
|
uint32_t result;
|
2006-06-16 00:59:40 +02:00
|
|
|
|
2006-06-16 00:58:06 +02:00
|
|
|
if (num_bits < state->bits_left) {
|
|
|
|
result = (state->current_word << (32 - state->bits_left)) >> (32 - num_bits);
|
|
|
|
state->bits_left -= num_bits;
|
2001-12-09 16:28:44 +01:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2006-06-16 00:58:06 +02:00
|
|
|
return a52_bitstream_get_bh (state, num_bits);
|
2001-12-17 22:53:49 +01:00
|
|
|
#endif
|
2001-12-09 16:28:44 +01:00
|
|
|
}
|
|
|
|
|
2006-06-16 00:58:06 +02:00
|
|
|
static inline void bitstream_skip(a52_state_t * state, int num_bits)
|
2002-01-08 15:49:17 +01:00
|
|
|
{
|
|
|
|
#ifdef ALT_BITSTREAM_READER
|
|
|
|
indx+= num_bits;
|
|
|
|
#else
|
2006-06-16 00:58:06 +02:00
|
|
|
bitstream_get(state, num_bits);
|
2002-01-08 15:49:17 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2006-06-16 00:58:06 +02:00
|
|
|
static inline int32_t bitstream_get_2 (a52_state_t * state, uint32_t num_bits)
|
2001-12-09 16:28:44 +01:00
|
|
|
{
|
2001-12-17 22:53:49 +01:00
|
|
|
#ifdef ALT_BITSTREAM_READER
|
2006-06-16 00:58:06 +02:00
|
|
|
int32_t result= swab32( unaligned32(((uint8_t *)state->buffer_start)+(indx>>3)) );
|
2001-12-17 22:53:49 +01:00
|
|
|
|
|
|
|
result<<= (indx&0x07);
|
|
|
|
result>>= 32 - num_bits;
|
|
|
|
indx+= num_bits;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
#else
|
2001-12-09 16:28:44 +01:00
|
|
|
int32_t result;
|
|
|
|
|
2006-06-16 00:58:06 +02:00
|
|
|
if (num_bits < state->bits_left) {
|
|
|
|
result = (((int32_t)state->current_word) << (32 - state->bits_left)) >> (32 - num_bits);
|
|
|
|
state->bits_left -= num_bits;
|
2001-12-09 16:28:44 +01:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2006-06-16 00:58:06 +02:00
|
|
|
return a52_bitstream_get_bh_2 (state, num_bits);
|
2001-12-17 22:53:49 +01:00
|
|
|
#endif
|
2001-12-09 16:28:44 +01:00
|
|
|
}
|