buffer.java

来自「java ogg player library. for play back o」· Java 代码 · 共 542 行

JAVA
542
字号
/* -*-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");  }  */}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?