/***************************************************************************** * bits.h ***************************************************************************** * Copyright (C) 2001, 2002 the VideoLAN team * $Id$ * * Authors: Laurent Aimar <fenrir@via.ecp.fr> * Eric Petit <titer@videolan.org> * * Copyright (C) 2008 Lin YANG <oxcsnicho@gmail.com> * * 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 * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. *****************************************************************************/ #ifndef __BITS_H__ #define __BITS_H__ #include <glib.h> G_BEGIN_DECLS typedef struct bits_buffer_s { gint i_size; gint i_data; guint8 i_mask; guint8* p_data; } bits_buffer_t; static inline gint bits_initwrite( bits_buffer_t *p_buffer, gint i_size, void *p_data ) { p_buffer->i_size = i_size; p_buffer->i_data = 0; p_buffer->i_mask = 0x80; p_buffer->p_data = p_data; if( !p_buffer->p_data ) { if( !( p_buffer->p_data = g_slice_alloc0( i_size ) ) ) return -1; } p_buffer->p_data[0] = 0; return 0; } static inline void bits_align( bits_buffer_t *p_buffer ) { if( p_buffer->i_mask != 0x80 && p_buffer->i_data < p_buffer->i_size ) { p_buffer->i_mask = 0x80; p_buffer->i_data++; p_buffer->p_data[p_buffer->i_data] = 0x00; } } static inline void bits_write( bits_buffer_t *p_buffer, gint i_count, guint64 i_bits ) { while( i_count > 0 ) { i_count--; if( ( i_bits >> i_count )&0x01 ) { p_buffer->p_data[p_buffer->i_data] |= p_buffer->i_mask; } else { p_buffer->p_data[p_buffer->i_data] &= ~p_buffer->i_mask; } p_buffer->i_mask >>= 1; if( p_buffer->i_mask == 0 ) { p_buffer->i_data++; p_buffer->i_mask = 0x80; } } } G_END_DECLS #endif