]> Joshua Wise's Git repositories - patchfork.git/blob - jorbis/src/com/jcraft/jogg/Buffer.java
a40a9def9c416d610040b4f671a47c83f2e1a74b
[patchfork.git] / jorbis / src / com / jcraft / jogg / Buffer.java
1 /* -*-mode:java; c-basic-offset:2; -*- */
2 /* JOrbis
3  * Copyright (C) 2000 ymnk, JCraft,Inc.
4  *  
5  * Written by: 2000 ymnk<ymnk@jcraft.com>
6  *   
7  * Many thanks to 
8  *   Monty <monty@xiph.org> and 
9  *   The XIPHOPHORUS Company http://www.xiph.org/ .
10  * JOrbis has been based on their awesome works, Vorbis codec.
11  *   
12  * This program is free software; you can redistribute it and/or
13  * modify it under the terms of the GNU Library General Public License
14  * as published by the Free Software Foundation; either version 2 of
15  * the License, or (at your option) any later version.
16    
17  * This program is distributed in the hope that it will be useful,
18  * but WITHOUT ANY WARRANTY; without even the implied warranty of
19  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
20  * GNU Library General Public License for more details.
21  * 
22  * You should have received a copy of the GNU Library General Public
23  * License along with this program; if not, write to the Free Software
24  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
25  */
26
27 package com.jcraft.jogg;
28
29 public class Buffer{
30   private static final int BUFFER_INCREMENT=256;
31
32   private static final int[] mask={
33     0x00000000,0x00000001,0x00000003,0x00000007,0x0000000f,
34     0x0000001f,0x0000003f,0x0000007f,0x000000ff,0x000001ff,
35     0x000003ff,0x000007ff,0x00000fff,0x00001fff,0x00003fff,
36     0x00007fff,0x0000ffff,0x0001ffff,0x0003ffff,0x0007ffff,
37     0x000fffff,0x001fffff,0x003fffff,0x007fffff,0x00ffffff,
38     0x01ffffff,0x03ffffff,0x07ffffff,0x0fffffff,0x1fffffff,
39     0x3fffffff,0x7fffffff,0xffffffff
40   };
41
42   int ptr=0;
43   byte[] buffer=null;
44   int endbit=0;
45   int endbyte=0;
46   int storage=0;
47
48   public void writeinit(){
49     buffer=new byte[BUFFER_INCREMENT]; 
50     ptr=0;
51     buffer[0]=(byte)'\0';
52     storage=BUFFER_INCREMENT;
53   }
54
55   public void write(byte[] s){
56     for(int i=0; i<s.length; i++){
57       if(s[i]==0)break;
58       write(s[i],8);
59     }
60   }
61
62   public void read(byte[] s, int bytes){
63     int i=0;
64     while(bytes--!=0){
65       s[i++]=(byte)(read(8));
66     }
67   }
68
69   void reset(){
70     ptr=0;
71     buffer[0]=(byte)'\0';
72     endbit=endbyte=0;
73   }
74
75   public void writeclear(){
76     buffer=null;
77   }
78
79   public void readinit(byte[] buf, int bytes){
80     readinit(buf, 0, bytes);
81   }
82
83   public void readinit(byte[] buf, int start, int bytes){
84 //System.err.println("readinit: start="+start+", bytes="+bytes);
85 //for(int i=0;i<bytes; i++){
86 //System.err.println(i+": "+Integer.toHexString(buf[i+start]));
87 //}
88     ptr=start;
89     buffer=buf;
90     endbit=endbyte=0;
91     storage=bytes;
92   }
93
94   public void write(int value, int bits){
95 //System.err.println("write: "+Integer.toHexString(value)+", bits="+bits+" ptr="+ptr+", storage="+storage+", endbyte="+endbyte);
96     if(endbyte+4>=storage){
97       byte[] foo=new byte[storage+BUFFER_INCREMENT];
98       System.arraycopy(buffer, 0, foo, 0, storage);
99       buffer=foo;
100       storage+=BUFFER_INCREMENT;
101     }
102
103     value&=mask[bits];
104     bits+=endbit;
105     buffer[ptr]|=(byte)(value<<endbit);
106
107     if(bits>=8){
108       buffer[ptr+1]=(byte)(value>>>(8-endbit));
109       if(bits>=16){
110         buffer[ptr+2]=(byte)(value>>>(16-endbit));  
111         if(bits>=24){
112           buffer[ptr+3]=(byte)(value>>>(24-endbit));  
113           if(bits>=32){
114             if(endbit>0)
115               buffer[ptr+4]=(byte)(value>>>(32-endbit));
116             else
117               buffer[ptr+4]=0;
118           }
119         }
120       }
121     }
122
123     endbyte+=bits/8;
124     ptr+=bits/8;
125     endbit=bits&7;
126   }
127
128   public int look(int bits){
129     int ret;
130     int m=mask[bits];
131
132     bits+=endbit;
133
134 //System.err.println("look ptr:"+ptr+", bits="+bits+", endbit="+endbit+", storage="+storage);
135
136     if(endbyte+4>=storage){
137       if(endbyte+(bits-1)/8>=storage)return(-1);
138     }
139   
140     ret=((buffer[ptr])&0xff)>>>endbit;
141 //  ret=((byte)(buffer[ptr]))>>>endbit;
142     if(bits>8){
143     ret|=((buffer[ptr+1])&0xff)<<(8-endbit);
144 //      ret|=((byte)(buffer[ptr+1]))<<(8-endbit);
145       if(bits>16){
146       ret|=((buffer[ptr+2])&0xff)<<(16-endbit);
147 //        ret|=((byte)(buffer[ptr+2]))<<(16-endbit);
148         if(bits>24){
149           ret|=((buffer[ptr+3])&0xff)<<(24-endbit);
150 //System.err.print("ret="+Integer.toHexString(ret)+", ((byte)(buffer[ptr+3]))="+Integer.toHexString(((buffer[ptr+3])&0xff)));
151 //        ret|=((byte)(buffer[ptr+3]))<<(24-endbit);
152 //System.err.println(" ->ret="+Integer.toHexString(ret));
153           if(bits>32 && endbit!=0){
154             ret|=((buffer[ptr+4])&0xff)<<(32-endbit);
155 //          ret|=((byte)(buffer[ptr+4]))<<(32-endbit);
156           }
157         }
158       }
159     }
160     return(m&ret);
161   }
162
163   public int look1(){
164     if(endbyte>=storage)return(-1);
165     return((buffer[ptr]>>endbit)&1);
166   }
167
168   public void adv(int bits){
169     bits+=endbit;
170     ptr+=bits/8;
171     endbyte+=bits/8;
172     endbit=bits&7;
173   }
174
175   public void adv1(){
176     ++endbit;
177     if(endbit>7){
178       endbit=0;
179       ptr++;
180       endbyte++;
181     }
182   }
183
184   public int read(int bits){
185 //System.err.println(this+" read: bits="+bits+", storage="+storage+", endbyte="+endbyte);
186 //System.err.println(this+" read: bits="+bits+", storage="+storage+", endbyte="+endbyte+
187 //                        ", ptr="+ptr+", endbit="+endbit+", buf[ptr]="+buffer[ptr]);
188
189     int ret;
190     int m=mask[bits];
191
192     bits+=endbit;
193
194     if(endbyte+4>=storage){
195       ret=-1;
196       if(endbyte+(bits-1)/8>=storage){
197         ptr+=bits/8;
198         endbyte+=bits/8;
199         endbit=bits&7;
200         return(ret);
201       }
202     }
203
204 /*  
205     ret=(byte)(buffer[ptr]>>>endbit);
206     if(bits>8){
207       ret|=(buffer[ptr+1]<<(8-endbit));  
208       if(bits>16){
209         ret|=(buffer[ptr+2]<<(16-endbit));  
210         if(bits>24){
211           ret|=(buffer[ptr+3]<<(24-endbit));  
212           if(bits>32 && endbit>0){
213             ret|=(buffer[ptr+4]<<(32-endbit));
214           }
215         }
216       }
217     }
218 */
219     ret=((buffer[ptr])&0xff)>>>endbit;
220     if(bits>8){
221     ret|=((buffer[ptr+1])&0xff)<<(8-endbit);
222 //      ret|=((byte)(buffer[ptr+1]))<<(8-endbit);
223       if(bits>16){
224       ret|=((buffer[ptr+2])&0xff)<<(16-endbit);
225 //        ret|=((byte)(buffer[ptr+2]))<<(16-endbit);
226         if(bits>24){
227           ret|=((buffer[ptr+3])&0xff)<<(24-endbit);
228 //        ret|=((byte)(buffer[ptr+3]))<<(24-endbit);
229           if(bits>32 && endbit!=0){
230             ret|=((buffer[ptr+4])&0xff)<<(32-endbit);
231 //          ret|=((byte)(buffer[ptr+4]))<<(32-endbit);
232           }
233         }
234       }
235     }
236
237     ret&=m;
238
239     ptr+=bits/8;
240 //    ptr=bits/8;
241     endbyte+=bits/8;
242 //    endbyte=bits/8;
243     endbit=bits&7;
244     return(ret);
245   }
246
247   public int readB(int bits){
248     //System.err.println(this+" read: bits="+bits+", storage="+storage+", endbyte="+endbyte+
249     //                        ", ptr="+ptr+", endbit="+endbit+", buf[ptr]="+buffer[ptr]);
250     int ret;
251     int m=32-bits;
252
253     bits+=endbit;
254
255     if(endbyte+4>=storage){
256       /* not the main path */
257       ret=-1;
258       if(endbyte*8+bits>storage*8) {
259         ptr+=bits/8;
260         endbyte+=bits/8;
261         endbit=bits&7;
262         return(ret);
263       }
264     }
265
266     ret=(buffer[ptr]&0xff)<<(24+endbit);
267     if(bits>8){
268       ret|=(buffer[ptr+1]&0xff)<<(16+endbit);
269       if(bits>16){
270         ret|=(buffer[ptr+2]&0xff)<<(8+endbit);
271         if(bits>24){
272           ret|=(buffer[ptr+3]&0xff)<<(endbit);
273           if(bits>32 && (endbit != 0))
274             ret|=(buffer[ptr+4]&0xff)>>(8-endbit);
275         }
276       }
277     }
278     ret=(ret>>>(m>>1))>>>((m+1)>>1);
279
280     ptr+=bits/8;
281     endbyte+=bits/8;
282     endbit=bits&7;
283     return(ret);
284   }
285
286   public int read1(){
287     int ret;
288     if(endbyte>=storage){
289       ret=-1;
290       endbit++;
291       if(endbit>7){
292         endbit=0;
293         ptr++;
294         endbyte++;
295       }
296       return(ret);
297     }
298
299     ret=(buffer[ptr]>>endbit)&1;
300
301     endbit++;
302     if(endbit>7){
303       endbit=0;
304       ptr++;
305       endbyte++;
306     }
307     return(ret);
308   }
309
310   public int bytes(){
311     return(endbyte+(endbit+7)/8);
312   }
313
314   public int bits(){
315     return(endbyte*8+endbit);
316   }
317
318   public byte[] buffer(){
319     return(buffer);
320   }
321
322   public static int ilog(int v){
323     int ret=0;
324     while(v>0){
325       ret++;
326       v>>>=1;
327     }
328     return(ret);
329   }
330
331   public static void report(String in){
332     System.err.println(in);
333     System.exit(1);
334   }
335
336   /*
337   static void cliptest(int[] b, int vals, int bits, int[] comp, int compsize){
338     int bytes;
339     byte[] buffer;
340
341     o.reset();
342     for(int i=0;i<vals;i++){
343       o.write(b[i],((bits!=0)?bits:ilog(b[i])));
344     }
345     buffer=o.buffer();
346     bytes=o.bytes();
347 System.err.println("cliptest: bytes="+bytes);
348     if(bytes!=compsize)report("wrong number of bytes!\n");
349     for(int i=0;i<bytes;i++){
350       if(buffer[i]!=(byte)comp[i]){
351         for(int j=0;j<bytes;j++){
352           System.err.println(j+": "+Integer.toHexString(buffer[j])+" "+
353                              Integer.toHexString(comp[j]));
354         }
355         report("wrote incorrect value!\n");
356       }
357     }
358 System.err.println("bits: "+bits);
359     r.readinit(buffer,bytes);
360     for(int i=0;i<vals;i++){
361       int tbit=(bits!=0)?bits:ilog(b[i]);
362 System.err.println(Integer.toHexString(b[i])+" tbit: "+tbit); 
363       if(r.look(tbit)==-1){
364         report("out of data!\n");
365       }
366       if(r.look(tbit)!=(b[i]&mask[tbit])){
367         report(i+" looked at incorrect value! "+Integer.toHexString(r.look(tbit))+", "+Integer.toHexString(b[i]&mask[tbit])+":"+b[i]+" bit="+tbit);
368       }
369       if(tbit==1){
370         if(r.look1()!=(b[i]&mask[tbit])){
371           report("looked at single bit incorrect value!\n");
372         }
373       }
374       if(tbit==1){
375         if(r.read1()!=(b[i]&mask[tbit])){
376           report("read incorrect single bit value!\n");
377         }
378       }
379       else{
380         if(r.read(tbit)!=(b[i]&mask[tbit])){
381           report("read incorrect value!\n");
382         }
383       }
384     }
385     if(r.bytes()!=bytes){
386       report("leftover bytes after read!\n");
387     }
388   }
389
390   static int[] testbuffer1=
391     {18,12,103948,4325,543,76,432,52,3,65,4,56,32,42,34,21,1,23,32,546,456,7,
392        567,56,8,8,55,3,52,342,341,4,265,7,67,86,2199,21,7,1,5,1,4};
393   static int test1size=43;
394
395   static int[] testbuffer2=
396     {216531625,1237861823,56732452,131,3212421,12325343,34547562,12313212,
397        1233432,534,5,346435231,14436467,7869299,76326614,167548585,
398        85525151,0,12321,1,349528352};
399   static int test2size=21;
400
401   static int[] large=
402     {2136531625,2137861823,56732452,131,3212421,12325343,34547562,12313212,
403        1233432,534,5,2146435231,14436467,7869299,76326614,167548585,
404        85525151,0,12321,1,2146528352};
405
406   static int[] testbuffer3=
407     {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,
408        0,1,30,1,1,1,0,0,1,0,0,0,12,0,11,0,1,0,0,1};
409   static int test3size=56;
410
411   static int onesize=33;
412   static int[] one={146,25,44,151,195,15,153,176,233,131,196,65,85,172,47,40,
413                     34,242,223,136,35,222,211,86,171,50,225,135,214,75,172,
414                     223,4};
415
416   static int twosize=6;
417   static int[] two={61,255,255,251,231,29};
418
419   static int threesize=54;
420   static int[] three={169,2,232,252,91,132,156,36,89,13,123,176,144,32,254,
421                       142,224,85,59,121,144,79,124,23,67,90,90,216,79,23,83,
422                       58,135,196,61,55,129,183,54,101,100,170,37,127,126,10,
423                       100,52,4,14,18,86,77,1};
424
425   static int foursize=38;
426   static int[] four={18,6,163,252,97,194,104,131,32,1,7,82,137,42,129,11,72,
427                      132,60,220,112,8,196,109,64,179,86,9,137,195,208,122,169,
428                      28,2,133,0,1};
429
430   static int fivesize=45;
431   static int[] five={169,2,126,139,144,172,30,4,80,72,240,59,130,218,73,62,
432                      241,24,210,44,4,20,0,248,116,49,135,100,110,130,181,169,
433                      84,75,159,2,1,0,132,192,8,0,0,18,22};
434
435   static int sixsize=7;
436   static int[] six={17,177,170,242,169,19,148};
437
438   static Buffer o=new Buffer();
439   static Buffer r=new Buffer();
440
441   public static void main(String[] arg){
442     byte[] buffer;
443     int bytes;
444 //  o=new Buffer();
445 //  r=new Buffer();
446
447     o.writeinit();
448
449     System.err.print("\nSmall preclipped packing: ");
450     cliptest(testbuffer1,test1size,0,one,onesize);
451     System.err.print("ok.");
452
453     System.err.print("\nNull bit call: ");
454     cliptest(testbuffer3,test3size,0,two,twosize);
455     System.err.print("ok.");
456
457     System.err.print("\nLarge preclipped packing: ");
458     cliptest(testbuffer2,test2size,0,three,threesize);
459     System.err.print("ok.");
460
461     System.err.print("\n32 bit preclipped packing: ");
462     o.reset();
463     for(int i=0;i<test2size;i++)
464       o.write(large[i],32);
465     buffer=o.buffer();
466     bytes=o.bytes();
467
468
469     r.readinit(buffer,bytes);
470     for(int i=0;i<test2size;i++){
471       if(r.look(32)==-1){
472         report("out of data. failed!");
473       }
474       if(r.look(32)!=large[i]){
475         System.err.print(r.look(32)+" != "+large[i]+" ("+
476                          Integer.toHexString(r.look(32))+"!="+
477                          Integer.toHexString(large[i])+")");
478         report("read incorrect value!\n");
479       }
480       r.adv(32);
481     }
482     if(r.bytes()!=bytes)report("leftover bytes after read!\n");
483     System.err.print("ok.");
484
485     System.err.print("\nSmall unclipped packing: ");
486     cliptest(testbuffer1,test1size,7,four,foursize);
487     System.err.print("ok.");
488
489     System.err.print("\nLarge unclipped packing: ");
490     cliptest(testbuffer2,test2size,17,five,fivesize);
491     System.err.print("ok.");
492
493     System.err.print("\nSingle bit unclicpped packing: ");
494     cliptest(testbuffer3,test3size,1,six,sixsize);
495     System.err.print("ok.");
496
497     System.err.print("\nTesting read past end: ");
498     r.readinit("\0\0\0\0\0\0\0\0".getBytes(),8);
499     for(int i=0;i<64;i++){
500       if(r.read(1)!=0){
501         System.err.print("failed; got -1 prematurely.\n");
502         System.exit(1);
503       }
504     }
505
506     if(r.look(1)!=-1 ||
507        r.read(1)!=-1){
508       System.err.print("failed; read past end without -1.\n");
509       System.exit(1);
510     }
511
512     r.readinit("\0\0\0\0\0\0\0\0".getBytes(),8);
513     if(r.read(30)!=0 || r.read(16)!=0){
514       System.err.print("failed 2; got -1 prematurely.\n");
515     System.exit(1);
516     }
517
518     if(r.look(18)!=0 ||
519        r.look(18)!=0){
520       System.err.print("failed 3; got -1 prematurely.\n");
521       System.exit(1);
522     }
523     if(r.look(19)!=-1 ||
524        r.look(19)!=-1){
525       System.err.print("failed; read past end without -1.\n");
526       System.exit(1);
527     }
528     if(r.look(32)!=-1 ||
529        r.look(32)!=-1){
530       System.err.print("failed; read past end without -1.\n");
531       System.exit(1);
532     }
533     System.err.print("ok.\n\n");
534   }
535   */
536 }
537
538
539
540
541
This page took 0.054199 seconds and 4 git commands to generate.