542 lines
14 KiB
Java
542 lines
14 KiB
Java
|
/* -*-mode:java; c-basic-offset:2; -*- */
|
||
|
/* JOrbis
|
||
|
* Copyright (C) 2000 ymnk, JCraft,Inc.
|
||
|
*
|
||
|
* Written by: 2000 ymnk<ymnk@jcraft.com>
|
||
|
*
|
||
|
* Many thanks to
|
||
|
* Monty <monty@xiph.org> and
|
||
|
* The XIPHOPHORUS Company http://www.xiph.org/ .
|
||
|
* JOrbis has been based on their awesome works, Vorbis codec.
|
||
|
*
|
||
|
* This program 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 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 Library General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU Library General Public
|
||
|
* License along with this program; if not, write to the Free Software
|
||
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
||
|
*/
|
||
|
|
||
|
package com.jcraft.jogg;
|
||
|
|
||
|
public class Buffer{
|
||
|
private static final int BUFFER_INCREMENT=256;
|
||
|
|
||
|
private static final int[] 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
|
||
|
};
|
||
|
|
||
|
int ptr=0;
|
||
|
byte[] buffer=null;
|
||
|
int endbit=0;
|
||
|
int endbyte=0;
|
||
|
int storage=0;
|
||
|
|
||
|
public void writeinit(){
|
||
|
buffer=new byte[BUFFER_INCREMENT];
|
||
|
ptr=0;
|
||
|
buffer[0]=(byte)'\0';
|
||
|
storage=BUFFER_INCREMENT;
|
||
|
}
|
||
|
|
||
|
public void write(byte[] s){
|
||
|
for(int i=0; i<s.length; i++){
|
||
|
if(s[i]==0)break;
|
||
|
write(s[i],8);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void read(byte[] s, int bytes){
|
||
|
int i=0;
|
||
|
while(bytes--!=0){
|
||
|
s[i++]=(byte)(read(8));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void reset(){
|
||
|
ptr=0;
|
||
|
buffer[0]=(byte)'\0';
|
||
|
endbit=endbyte=0;
|
||
|
}
|
||
|
|
||
|
public void writeclear(){
|
||
|
buffer=null;
|
||
|
}
|
||
|
|
||
|
public void readinit(byte[] buf, int bytes){
|
||
|
readinit(buf, 0, bytes);
|
||
|
}
|
||
|
|
||
|
public void readinit(byte[] buf, int start, int bytes){
|
||
|
//System.err.println("readinit: start="+start+", bytes="+bytes);
|
||
|
//for(int i=0;i<bytes; i++){
|
||
|
//System.err.println(i+": "+Integer.toHexString(buf[i+start]));
|
||
|
//}
|
||
|
ptr=start;
|
||
|
buffer=buf;
|
||
|
endbit=endbyte=0;
|
||
|
storage=bytes;
|
||
|
}
|
||
|
|
||
|
public void write(int value, int bits){
|
||
|
//System.err.println("write: "+Integer.toHexString(value)+", bits="+bits+" ptr="+ptr+", storage="+storage+", endbyte="+endbyte);
|
||
|
if(endbyte+4>=storage){
|
||
|
byte[] foo=new byte[storage+BUFFER_INCREMENT];
|
||
|
System.arraycopy(buffer, 0, foo, 0, storage);
|
||
|
buffer=foo;
|
||
|
storage+=BUFFER_INCREMENT;
|
||
|
}
|
||
|
|
||
|
value&=mask[bits];
|
||
|
bits+=endbit;
|
||
|
buffer[ptr]|=(byte)(value<<endbit);
|
||
|
|
||
|
if(bits>=8){
|
||
|
buffer[ptr+1]=(byte)(value>>>(8-endbit));
|
||
|
if(bits>=16){
|
||
|
buffer[ptr+2]=(byte)(value>>>(16-endbit));
|
||
|
if(bits>=24){
|
||
|
buffer[ptr+3]=(byte)(value>>>(24-endbit));
|
||
|
if(bits>=32){
|
||
|
if(endbit>0)
|
||
|
buffer[ptr+4]=(byte)(value>>>(32-endbit));
|
||
|
else
|
||
|
buffer[ptr+4]=0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
endbyte+=bits/8;
|
||
|
ptr+=bits/8;
|
||
|
endbit=bits&7;
|
||
|
}
|
||
|
|
||
|
public int look(int bits){
|
||
|
int ret;
|
||
|
int m=mask[bits];
|
||
|
|
||
|
bits+=endbit;
|
||
|
|
||
|
//System.err.println("look ptr:"+ptr+", bits="+bits+", endbit="+endbit+", storage="+storage);
|
||
|
|
||
|
if(endbyte+4>=storage){
|
||
|
if(endbyte+(bits-1)/8>=storage)return(-1);
|
||
|
}
|
||
|
|
||
|
ret=((buffer[ptr])&0xff)>>>endbit;
|
||
|
// ret=((byte)(buffer[ptr]))>>>endbit;
|
||
|
if(bits>8){
|
||
|
ret|=((buffer[ptr+1])&0xff)<<(8-endbit);
|
||
|
// ret|=((byte)(buffer[ptr+1]))<<(8-endbit);
|
||
|
if(bits>16){
|
||
|
ret|=((buffer[ptr+2])&0xff)<<(16-endbit);
|
||
|
// ret|=((byte)(buffer[ptr+2]))<<(16-endbit);
|
||
|
if(bits>24){
|
||
|
ret|=((buffer[ptr+3])&0xff)<<(24-endbit);
|
||
|
//System.err.print("ret="+Integer.toHexString(ret)+", ((byte)(buffer[ptr+3]))="+Integer.toHexString(((buffer[ptr+3])&0xff)));
|
||
|
// ret|=((byte)(buffer[ptr+3]))<<(24-endbit);
|
||
|
//System.err.println(" ->ret="+Integer.toHexString(ret));
|
||
|
if(bits>32 && endbit!=0){
|
||
|
ret|=((buffer[ptr+4])&0xff)<<(32-endbit);
|
||
|
// ret|=((byte)(buffer[ptr+4]))<<(32-endbit);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return(m&ret);
|
||
|
}
|
||
|
|
||
|
public int look1(){
|
||
|
if(endbyte>=storage)return(-1);
|
||
|
return((buffer[ptr]>>endbit)&1);
|
||
|
}
|
||
|
|
||
|
public void adv(int bits){
|
||
|
bits+=endbit;
|
||
|
ptr+=bits/8;
|
||
|
endbyte+=bits/8;
|
||
|
endbit=bits&7;
|
||
|
}
|
||
|
|
||
|
public void adv1(){
|
||
|
++endbit;
|
||
|
if(endbit>7){
|
||
|
endbit=0;
|
||
|
ptr++;
|
||
|
endbyte++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public int read(int bits){
|
||
|
//System.err.println(this+" read: bits="+bits+", storage="+storage+", endbyte="+endbyte);
|
||
|
//System.err.println(this+" read: bits="+bits+", storage="+storage+", endbyte="+endbyte+
|
||
|
// ", ptr="+ptr+", endbit="+endbit+", buf[ptr]="+buffer[ptr]);
|
||
|
|
||
|
int ret;
|
||
|
int m=mask[bits];
|
||
|
|
||
|
bits+=endbit;
|
||
|
|
||
|
if(endbyte+4>=storage){
|
||
|
ret=-1;
|
||
|
if(endbyte+(bits-1)/8>=storage){
|
||
|
ptr+=bits/8;
|
||
|
endbyte+=bits/8;
|
||
|
endbit=bits&7;
|
||
|
return(ret);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
ret=(byte)(buffer[ptr]>>>endbit);
|
||
|
if(bits>8){
|
||
|
ret|=(buffer[ptr+1]<<(8-endbit));
|
||
|
if(bits>16){
|
||
|
ret|=(buffer[ptr+2]<<(16-endbit));
|
||
|
if(bits>24){
|
||
|
ret|=(buffer[ptr+3]<<(24-endbit));
|
||
|
if(bits>32 && endbit>0){
|
||
|
ret|=(buffer[ptr+4]<<(32-endbit));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
*/
|
||
|
ret=((buffer[ptr])&0xff)>>>endbit;
|
||
|
if(bits>8){
|
||
|
ret|=((buffer[ptr+1])&0xff)<<(8-endbit);
|
||
|
// ret|=((byte)(buffer[ptr+1]))<<(8-endbit);
|
||
|
if(bits>16){
|
||
|
ret|=((buffer[ptr+2])&0xff)<<(16-endbit);
|
||
|
// ret|=((byte)(buffer[ptr+2]))<<(16-endbit);
|
||
|
if(bits>24){
|
||
|
ret|=((buffer[ptr+3])&0xff)<<(24-endbit);
|
||
|
// ret|=((byte)(buffer[ptr+3]))<<(24-endbit);
|
||
|
if(bits>32 && endbit!=0){
|
||
|
ret|=((buffer[ptr+4])&0xff)<<(32-endbit);
|
||
|
// ret|=((byte)(buffer[ptr+4]))<<(32-endbit);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ret&=m;
|
||
|
|
||
|
ptr+=bits/8;
|
||
|
// ptr=bits/8;
|
||
|
endbyte+=bits/8;
|
||
|
// endbyte=bits/8;
|
||
|
endbit=bits&7;
|
||
|
return(ret);
|
||
|
}
|
||
|
|
||
|
public int readB(int bits){
|
||
|
//System.err.println(this+" read: bits="+bits+", storage="+storage+", endbyte="+endbyte+
|
||
|
// ", ptr="+ptr+", endbit="+endbit+", buf[ptr]="+buffer[ptr]);
|
||
|
int ret;
|
||
|
int m=32-bits;
|
||
|
|
||
|
bits+=endbit;
|
||
|
|
||
|
if(endbyte+4>=storage){
|
||
|
/* not the main path */
|
||
|
ret=-1;
|
||
|
if(endbyte*8+bits>storage*8) {
|
||
|
ptr+=bits/8;
|
||
|
endbyte+=bits/8;
|
||
|
endbit=bits&7;
|
||
|
return(ret);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ret=(buffer[ptr]&0xff)<<(24+endbit);
|
||
|
if(bits>8){
|
||
|
ret|=(buffer[ptr+1]&0xff)<<(16+endbit);
|
||
|
if(bits>16){
|
||
|
ret|=(buffer[ptr+2]&0xff)<<(8+endbit);
|
||
|
if(bits>24){
|
||
|
ret|=(buffer[ptr+3]&0xff)<<(endbit);
|
||
|
if(bits>32 && (endbit != 0))
|
||
|
ret|=(buffer[ptr+4]&0xff)>>(8-endbit);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
ret=(ret>>>(m>>1))>>>((m+1)>>1);
|
||
|
|
||
|
ptr+=bits/8;
|
||
|
endbyte+=bits/8;
|
||
|
endbit=bits&7;
|
||
|
return(ret);
|
||
|
}
|
||
|
|
||
|
public int read1(){
|
||
|
int ret;
|
||
|
if(endbyte>=storage){
|
||
|
ret=-1;
|
||
|
endbit++;
|
||
|
if(endbit>7){
|
||
|
endbit=0;
|
||
|
ptr++;
|
||
|
endbyte++;
|
||
|
}
|
||
|
return(ret);
|
||
|
}
|
||
|
|
||
|
ret=(buffer[ptr]>>endbit)&1;
|
||
|
|
||
|
endbit++;
|
||
|
if(endbit>7){
|
||
|
endbit=0;
|
||
|
ptr++;
|
||
|
endbyte++;
|
||
|
}
|
||
|
return(ret);
|
||
|
}
|
||
|
|
||
|
public int bytes(){
|
||
|
return(endbyte+(endbit+7)/8);
|
||
|
}
|
||
|
|
||
|
public int bits(){
|
||
|
return(endbyte*8+endbit);
|
||
|
}
|
||
|
|
||
|
public byte[] buffer(){
|
||
|
return(buffer);
|
||
|
}
|
||
|
|
||
|
public static int ilog(int v){
|
||
|
int ret=0;
|
||
|
while(v>0){
|
||
|
ret++;
|
||
|
v>>>=1;
|
||
|
}
|
||
|
return(ret);
|
||
|
}
|
||
|
|
||
|
public static void report(String in){
|
||
|
System.err.println(in);
|
||
|
System.exit(1);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
static void cliptest(int[] b, int vals, int bits, int[] comp, int compsize){
|
||
|
int bytes;
|
||
|
byte[] buffer;
|
||
|
|
||
|
o.reset();
|
||
|
for(int i=0;i<vals;i++){
|
||
|
o.write(b[i],((bits!=0)?bits:ilog(b[i])));
|
||
|
}
|
||
|
buffer=o.buffer();
|
||
|
bytes=o.bytes();
|
||
|
System.err.println("cliptest: bytes="+bytes);
|
||
|
if(bytes!=compsize)report("wrong number of bytes!\n");
|
||
|
for(int i=0;i<bytes;i++){
|
||
|
if(buffer[i]!=(byte)comp[i]){
|
||
|
for(int j=0;j<bytes;j++){
|
||
|
System.err.println(j+": "+Integer.toHexString(buffer[j])+" "+
|
||
|
Integer.toHexString(comp[j]));
|
||
|
}
|
||
|
report("wrote incorrect value!\n");
|
||
|
}
|
||
|
}
|
||
|
System.err.println("bits: "+bits);
|
||
|
r.readinit(buffer,bytes);
|
||
|
for(int i=0;i<vals;i++){
|
||
|
int tbit=(bits!=0)?bits:ilog(b[i]);
|
||
|
System.err.println(Integer.toHexString(b[i])+" tbit: "+tbit);
|
||
|
if(r.look(tbit)==-1){
|
||
|
report("out of data!\n");
|
||
|
}
|
||
|
if(r.look(tbit)!=(b[i]&mask[tbit])){
|
||
|
report(i+" looked at incorrect value! "+Integer.toHexString(r.look(tbit))+", "+Integer.toHexString(b[i]&mask[tbit])+":"+b[i]+" bit="+tbit);
|
||
|
}
|
||
|
if(tbit==1){
|
||
|
if(r.look1()!=(b[i]&mask[tbit])){
|
||
|
report("looked at single bit incorrect value!\n");
|
||
|
}
|
||
|
}
|
||
|
if(tbit==1){
|
||
|
if(r.read1()!=(b[i]&mask[tbit])){
|
||
|
report("read incorrect single bit value!\n");
|
||
|
}
|
||
|
}
|
||
|
else{
|
||
|
if(r.read(tbit)!=(b[i]&mask[tbit])){
|
||
|
report("read incorrect value!\n");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if(r.bytes()!=bytes){
|
||
|
report("leftover bytes after read!\n");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static int[] testbuffer1=
|
||
|
{18,12,103948,4325,543,76,432,52,3,65,4,56,32,42,34,21,1,23,32,546,456,7,
|
||
|
567,56,8,8,55,3,52,342,341,4,265,7,67,86,2199,21,7,1,5,1,4};
|
||
|
static int test1size=43;
|
||
|
|
||
|
static int[] testbuffer2=
|
||
|
{216531625,1237861823,56732452,131,3212421,12325343,34547562,12313212,
|
||
|
1233432,534,5,346435231,14436467,7869299,76326614,167548585,
|
||
|
85525151,0,12321,1,349528352};
|
||
|
static int test2size=21;
|
||
|
|
||
|
static int[] large=
|
||
|
{2136531625,2137861823,56732452,131,3212421,12325343,34547562,12313212,
|
||
|
1233432,534,5,2146435231,14436467,7869299,76326614,167548585,
|
||
|
85525151,0,12321,1,2146528352};
|
||
|
|
||
|
static int[] testbuffer3=
|
||
|
{1,0,14,0,1,0,12,0,1,0,0,0,1,1,0,1,0,1,0,1,0,1,0,1,0,1,0,0,1,1,1,1,1,0,0,1,
|
||
|
0,1,30,1,1,1,0,0,1,0,0,0,12,0,11,0,1,0,0,1};
|
||
|
static int test3size=56;
|
||
|
|
||
|
static int onesize=33;
|
||
|
static int[] one={146,25,44,151,195,15,153,176,233,131,196,65,85,172,47,40,
|
||
|
34,242,223,136,35,222,211,86,171,50,225,135,214,75,172,
|
||
|
223,4};
|
||
|
|
||
|
static int twosize=6;
|
||
|
static int[] two={61,255,255,251,231,29};
|
||
|
|
||
|
static int threesize=54;
|
||
|
static int[] three={169,2,232,252,91,132,156,36,89,13,123,176,144,32,254,
|
||
|
142,224,85,59,121,144,79,124,23,67,90,90,216,79,23,83,
|
||
|
58,135,196,61,55,129,183,54,101,100,170,37,127,126,10,
|
||
|
100,52,4,14,18,86,77,1};
|
||
|
|
||
|
static int foursize=38;
|
||
|
static int[] four={18,6,163,252,97,194,104,131,32,1,7,82,137,42,129,11,72,
|
||
|
132,60,220,112,8,196,109,64,179,86,9,137,195,208,122,169,
|
||
|
28,2,133,0,1};
|
||
|
|
||
|
static int fivesize=45;
|
||
|
static int[] five={169,2,126,139,144,172,30,4,80,72,240,59,130,218,73,62,
|
||
|
241,24,210,44,4,20,0,248,116,49,135,100,110,130,181,169,
|
||
|
84,75,159,2,1,0,132,192,8,0,0,18,22};
|
||
|
|
||
|
static int sixsize=7;
|
||
|
static int[] six={17,177,170,242,169,19,148};
|
||
|
|
||
|
static Buffer o=new Buffer();
|
||
|
static Buffer r=new Buffer();
|
||
|
|
||
|
public static void main(String[] arg){
|
||
|
byte[] buffer;
|
||
|
int bytes;
|
||
|
// o=new Buffer();
|
||
|
// r=new Buffer();
|
||
|
|
||
|
o.writeinit();
|
||
|
|
||
|
System.err.print("\nSmall preclipped packing: ");
|
||
|
cliptest(testbuffer1,test1size,0,one,onesize);
|
||
|
System.err.print("ok.");
|
||
|
|
||
|
System.err.print("\nNull bit call: ");
|
||
|
cliptest(testbuffer3,test3size,0,two,twosize);
|
||
|
System.err.print("ok.");
|
||
|
|
||
|
System.err.print("\nLarge preclipped packing: ");
|
||
|
cliptest(testbuffer2,test2size,0,three,threesize);
|
||
|
System.err.print("ok.");
|
||
|
|
||
|
System.err.print("\n32 bit preclipped packing: ");
|
||
|
o.reset();
|
||
|
for(int i=0;i<test2size;i++)
|
||
|
o.write(large[i],32);
|
||
|
buffer=o.buffer();
|
||
|
bytes=o.bytes();
|
||
|
|
||
|
|
||
|
r.readinit(buffer,bytes);
|
||
|
for(int i=0;i<test2size;i++){
|
||
|
if(r.look(32)==-1){
|
||
|
report("out of data. failed!");
|
||
|
}
|
||
|
if(r.look(32)!=large[i]){
|
||
|
System.err.print(r.look(32)+" != "+large[i]+" ("+
|
||
|
Integer.toHexString(r.look(32))+"!="+
|
||
|
Integer.toHexString(large[i])+")");
|
||
|
report("read incorrect value!\n");
|
||
|
}
|
||
|
r.adv(32);
|
||
|
}
|
||
|
if(r.bytes()!=bytes)report("leftover bytes after read!\n");
|
||
|
System.err.print("ok.");
|
||
|
|
||
|
System.err.print("\nSmall unclipped packing: ");
|
||
|
cliptest(testbuffer1,test1size,7,four,foursize);
|
||
|
System.err.print("ok.");
|
||
|
|
||
|
System.err.print("\nLarge unclipped packing: ");
|
||
|
cliptest(testbuffer2,test2size,17,five,fivesize);
|
||
|
System.err.print("ok.");
|
||
|
|
||
|
System.err.print("\nSingle bit unclicpped packing: ");
|
||
|
cliptest(testbuffer3,test3size,1,six,sixsize);
|
||
|
System.err.print("ok.");
|
||
|
|
||
|
System.err.print("\nTesting read past end: ");
|
||
|
r.readinit("\0\0\0\0\0\0\0\0".getBytes(),8);
|
||
|
for(int i=0;i<64;i++){
|
||
|
if(r.read(1)!=0){
|
||
|
System.err.print("failed; got -1 prematurely.\n");
|
||
|
System.exit(1);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if(r.look(1)!=-1 ||
|
||
|
r.read(1)!=-1){
|
||
|
System.err.print("failed; read past end without -1.\n");
|
||
|
System.exit(1);
|
||
|
}
|
||
|
|
||
|
r.readinit("\0\0\0\0\0\0\0\0".getBytes(),8);
|
||
|
if(r.read(30)!=0 || r.read(16)!=0){
|
||
|
System.err.print("failed 2; got -1 prematurely.\n");
|
||
|
System.exit(1);
|
||
|
}
|
||
|
|
||
|
if(r.look(18)!=0 ||
|
||
|
r.look(18)!=0){
|
||
|
System.err.print("failed 3; got -1 prematurely.\n");
|
||
|
System.exit(1);
|
||
|
}
|
||
|
if(r.look(19)!=-1 ||
|
||
|
r.look(19)!=-1){
|
||
|
System.err.print("failed; read past end without -1.\n");
|
||
|
System.exit(1);
|
||
|
}
|
||
|
if(r.look(32)!=-1 ||
|
||
|
r.look(32)!=-1){
|
||
|
System.err.print("failed; read past end without -1.\n");
|
||
|
System.exit(1);
|
||
|
}
|
||
|
System.err.print("ok.\n\n");
|
||
|
}
|
||
|
*/
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|