/*% % l1zip.psm -- /FlateDecode image data implementation for /PSL1 % by pts@fazekas.hu at Sun Sep 22 01:01:00 CEST 2002 % formerly flatedecode.psm -- please run faltedecode_eps_helper.sh... % by pts@fazekas.hu at Fri Sep 20 23:31:36 CEST 2002 % -- Sat Sep 21 01:11:10 CEST 2002 % doesn't work -- Sat Sep 21 02:52:36 CEST 2002 % disallow preread==8, allow preread==0 WORKS -- Sat Sep 21 20:22:07 CEST 2002 % derived from statecat.psm -- a stateful, faster minimalistic zcat (gzip -cd) % implementation, implemented in pure PostScript LanguageLevel1 (Run % through cpp (C preprocessor) to get the PostScript file % statecat.psm by pts@fazekas.hu % originally rcm.c % original author: Ron McFarland , 1996 % formally muzcat.c (at Tue Dec 25 11:07:47 CET 2001) % based on statecat.ps.3 by pts@fazekas.hu % rewritten, restructured and extended at Tue Dec 25 21:00:33 CET 2001 % translated to PostScript except fvWork at Tue Dec 25 23:11:31 CET 2001 % translated to PostScript altogether at Thu Dec 27 01:10:58 CET 2001 % translation works with `cat random.file.gz random.file.gz'' at Thu Dec 27 14:19:36 CET 2001 % Thu Dec 27 18:37:03 CET 2001: % statecat.ps: 1071960 ms user time (249.293 times slower than gunzip) % function inlining at Thu Dec 27 21:51:45 CET 2001 % shorter keywords at Thu Dec 27 23:52:46 CET 2001 % at Sat Feb 23 17:05:43 CET 2002: % statecat.ps originally: 5388 bytes % after eliminate-NULLs trick: 5349 bytes % after anti-32768... trick 5229 bytes % 5041 %*/ /* % Imp: more efficient optimize CFG_FMT_ZLIB_ONLY % Imp: do `F closefile', `T closefile' in % % Not: don't inline TE_read for USE_HEXD (15*5 bytes for ACSO0g is not tolerable) % OK : verify a85_getc for really eating EOD (~>) chars % OK : find and catalogize `|' and other one-char abbrs % OK : shorter operator names (i.e `E' for `def', as in test_vm_eps.eps) % OK : shorter var names (such as `preread', `past') % Not: make global vars numbered, not named... % OK : make constU[]-3 a string, not an array % OK : make constW, constP, constL strings, not arrays % Imp: inline all except fvMktree (too long), fvFree (recursive) and fvWork (too long) % OK : verify, examine StateCatDict (45) % OK : find unused vars (such as `moo') in StateCatDict % OK : is constM or 1<< faster?? (decided 1<<) % OK : anti-`/z 42 eq', ensure `eq' compares numbers (INT_EQ) % OK : verify if/ifelse for ENDIF % OK : verify direction of `for's % OK : eliminate `stopped' context because it catches errors % OK : clear stack before `exit' % OK : verify `/name' against `name'; grep m@/@ % OK : verify that `}' is followed by correct if|ifelse|def|bind|stopped|loop|for % % Typical error: syntax: /^#/ % Typical error: syntax: m@/ +\w@ % Typical error: mispelling % Typical error: missing `pop' at `dup ... eq{exit}if' % Typical error: missing `for|if|...' after '}' */ #include "psmlib.psm" #if USE_A85D #else #if USE_HEXD #else #define USE_BINARY 1 #endif #endif #if USE_NO_BIND #define BIND_DEF def #else #define BIND_DEF bind def #endif /* --- .pin begins */ % %!PS-Adobe-3.0`E %%Inflater: pts@fazekas.hu l1zip.psm %%Pages: 1 `X%%DocumentData: `B %%LanguageLevel: 1 `I%%EndComments %%Page: 1 1 % % save`s `R % /* 88 dict dup /TokDict exch def begin */ % begin % % % best to make it first % % best to make it second % % % % % % % % % % /*!=LZW*/ % /*!=LZW*/ % % % % % % % % % % % % #if 0 /* important in LZW, but not in Flate */ % % % % #endif #if USE_A85D % % % % #endif #if USE_HEXD % % % % #endif #if USE_BINARY % #if USE_PALETTE % #endif % % #endif #if USE_SHORT_NAMES % Free for short names: *?\^_`| BFGIKLMNPQRTUVWYZ efhklmnopqrsvwz % Numerical comments denote # calls or # accesses % : B: TokSubs % /* F must be overridden with array to avoid early TokSubs */ % : G % % ? + 3 % % % 5 % % 5 % % % % 6 % % 6 % /* T must be overridden with array to avoid early TokSubs */ % % 5 % % 5 % % 7 /* formerly: C */ % % 4 % % % 7 % % % % 4 % % 3 % % % 6 % % % % % 10 % % % % 6 % /* % % 3 */ /* not a function anymore */ % % 1 % % 1 % % #endif % #if USE_CURRENTFILE #define STDIN currentfile #endif % #if USE_DEBUG2 /FP /fileposition x #endif #if USE_BINARY {mark /F currentfile/FlateDecode filter def}stopped #endif #if USE_A85D {mark /T currentfile/ASCII85Decode filter def /F T/FlateDecode filter def}stopped #endif #if USE_HEXD {mark /T currentfile/ASCIIHexDecode filter def /F T/FlateDecode filter def}stopped #endif /Z exch def cleartomark /G{`i}def % image, imagemask or `false 3 colorimage` `w `h `b[1 0 0 -1 0 `h] #if USE_TRUE true #else Z #endif % % 32768 string dup /Sbuf exch def % % fvMain % % F G F closefile #if !USE_BINARY T closefile #endif % #define ZZZ_ASET(aname,idx,b) aname idx b put #define ZZZ_AREF(aname,idx) aname idx get #define PASS #define GLOBAL_VAR0(t,name) #define SLOCAL_VAR0(t,name) #define GLOBAL_VAR(t,name) /name null def #define SLOCAL_VAR(t,name) /name null def #define GLOBAL_ARRAY(t,name,size) /name size t def #define LOCAL_VAR(a,b) #if 0 /* LINE_DEBUG */ #define put (put:__LINE__) === put #define get (get:__LINE__) === get #endif % % Imp: F closefile, T closefile /* We must define constants and global arrays first. */ #define NULL 0 #define NODESIZE 1998 /* NODESIZE%3==0 */ /** Binary (Huffman) tree of nodes. */ GLOBAL_ARRAY(array,N,NODESIZE) /** * Contains the code word lengths used for Huffman code generation in * fvMktree(). 320==288+32. 320 is large enough to hold both the code lengths * of the literal--length (288) and the distance (32) Huffman trees -- and * large enough to hold the auxilary Huffman tree (used for building the * real tree) of length 19. */ GLOBAL_ARRAY(array,Z,320) GLOBAL_ARRAY(array,Bary,17) GLOBAL_ARRAY(array,Gary,17) #if CFG_NO_VAR_S #else GLOBAL_ARRAY(string,Sbuf,32768) #endif /constZ19 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 19 packedarray def #if USE_HEXD /C(.)def #endif { /* We start declaring functions inside the brace */ #define constW { 16 17 18 0 8 7 9 6 10 5 11 4 12 3 13 2 14 1 15 } /* 19 packedarray def */ #define constU { 3 4 5 6 7 8 9 10 11 13 15 17 19 23 27 31 35 43 51 59 67 83 99 115 131 163 195 227 258 } /* 29 packedarray def */ #define constP { 0 0 0 0 0 0 0 0 1 1 1 1 2 2 2 2 3 3 3 3 4 4 4 4 5 5 5 5 0 } /* 29 packedarray def */ #define constQ { 1 2 3 4 5 7 9 13 17 25 33 49 65 97 129 193 257 385 513 769 1025 1537 2049 3073 4097 6145 8193 12289 16385 24577 } /* 30 packedarray def */ #define constL { 0 0 0 0 1 1 2 2 3 3 4 4 5 5 6 6 7 7 8 8 9 9 10 10 11 11 12 12 13 13 } /* 30 packedarray def */ #if 0 #define constM { 0 1 3 7 15 31 63 127 255 511 1023 2047 4095 8191 } /* 14 packedarray def */ #endif #if USE_A85D GLOBAL_VAR0(SINT32,xS) /* Variable `xS' was formerly called: `sx' */ GLOBAL_VAR0(SINT32,xD) /* Variable `xD' was formerly called: `dx' */ GLOBAL_VAR0(SINT32,xC) /* Variable `xC' was formerly called: `c0' */ /** - a85_getc <0..255|511> * Reads a char from `currentfile' as /ASCII85Decode */ /a85_getc{ % Simple getchar() of ASCII85Decode filter :-)), formerly /d PSM_A85_GETC }BIND_DEF #endif #if USE_NO_EOF #define my_getchar() TE_read( ) #define my_ignorechar() TE_read(pop) #else #define my_getchar() TE_read() {511}ifelse #define my_ignorechar() TE_read(pop) if #endif /** Index of the first free node in N. */ GLOBAL_VAR0(WORD,Dvar) GLOBAL_VAR0(WORD,Tvar) /* index (in S) of the first free position, [0..32767] */ /* GLOBAL_VAR(NODEN,no) -- unused */ /* GLOBAL_VAR(WORD,roo) -- unused */ /* GLOBAL_VAR(WORD,rf) -- unused */ /* GLOBAL_VAR(WORD,moo) -- unused */ /* GLOBAL_VAR(WORD,mo) -- unused */ /* GLOBAL_VAR(WORD,oooo) -- unused */ /* GLOBAL_VAR(WORD,x) -- unused */ GLOBAL_VAR0(WORD,mQ) GLOBAL_VAR0(WORD,mF) GLOBAL_VAR0(WORD,o) GLOBAL_VAR0(WORD,q) GLOBAL_VAR0(WORD,mode) /* MODE_* */ GLOBAL_VAR0(WORD,tY) /* FMT_*; formerly ty */ GLOBAL_VAR0(WORD,oO) GLOBAL_VAR0(WORD,oP) GLOBAL_VAR0(WORD,f) GLOBAL_VAR0(WORD,p) GLOBAL_VAR0(WORD,v) GLOBAL_VAR0(WORD,h) #define AREF_S(idx) AREF(Sbuf,idx) #define ASET_S(idx,val) ASET(Sbuf,idx,val) /* Reading Operations: -- fxRead8(): read a 8-bit byte at byte boundary -- fxRead16z(): read a 16-bit word at byte boundary, MSB first (ZIP) -- fxRead16(): read a 16-bit word at byte boundary, LSB first (normal) -- fxSkip(): skip to next byte boundary -- fxRead1(): read 1 bit (for Huffman) -- fvRead(): read any amount of bits (0..13), LSB first */ /* Dat: FlateDecode bit packing order is: (b7*128+...+b0)+(b15*128+...+b8) /** char, 0..255 */ SLOCAL_VAR0(WORD,preread) /** -1..-8: bits of bufread[pos] already read */ SLOCAL_VAR0(WORD,past) % static void fxInit(void) { % /fxInit { #define fxInit \ /preread 0 def \ /past -8 def % }BIND_DEF /* #define ASSERT_PREREAD_NOT_EOF() preread 255 gt ASSERT_FALSE_POP((unexpected eof)) */ #define ASSERT_NOT_EOF(val) ASSERT_TRUE(val dup 255 and eq,(non-EOF expected)) #define ASSERT_PREREAD_NOT_EOF() ASSERT_NOT_EOF(preread) /** reads 0..13 bits */ % static WORD fvRead(WORD arg) { /fvRead { DEBUG(dup (==> fvRead\() exch 42 string cvs concatstrings (\)\n) concatstrings) DEBUG(( past=) past neg 42 string cvs concatstrings ( preread=) preread 42 string cvs concatstrings (\n) concatstrings concatstrings) dup 0 INT_NE{ ASSERT_PREREAD_NOT_EOF() past % Stack: arg -opast dup preread exch bitshift exch % SHR % Stack: arg preread>>opast opast dup 3 index sub % Stack: arg ret==:preread>>-opast -opast -past==:-opast-arg dup -8 ge{ /past exch def % BUG fixed at Thu Dec 27 14:53:53 CET 2001 pop % Stack: arg ret }{ ASSERT_TRUE(dup -9 le,(-past>=9)) /preread my_getchar() def ASSERT_PREREAD_NOT_EOF() dup -16 ge{ /* _GT_ */ 8 add /past exch def 8 add % Stack: arg ret==:preread>>opast 8-opast preread exch bitshift % SHL add % Stack: arg ret'' }{ 16 add /past exch def 8 add 2 copy % Stack: arg retold 8-opast ret 8-opast preread exch bitshift add % SHL % Stack: arg retold 8-opast ret'' /preread my_getchar() def % BUG fixed at Thu Dec 27 15:40:53 CET 2001 ASSERT_TRUE(past -1 le,(-past>=1)) ASSERT_TRUE(past -4 ge,(-past<=4)) ASSERT_PREREAD_NOT_EOF() exch 8 add preread exch bitshift add % Stack: arg retold ret'' exch pop % Stack: arg ret'' }ifelse }ifelse % Stack: arg ret % exch constM exch get and % ret&=((1<0)) \ ASSERT_TRUE(past 0 INT_EQ,(past==0)) \ ASSERT_PREREAD_NOT_EOF() \ -1 2 {pop my_ignorechar()}for \ /preread my_getchar() def #define fxIgnore8z \ ASSERT_TRUE(past 0 INT_EQ,(past==0)) \ ASSERT_PREREAD_NOT_EOF() \ dup 0 INT_NE{ \ -1 2 {pop DEBUG2((fI8z)===) my_ignorechar()}for \ /preread my_getchar() def \ }{pop}ifelse % static WORD fxRead8EOF(void) { % /fxRead8EOF{ #define fxRead8EOF \ ASSERT_TRUE(past 0 INT_EQ,(past==0)) \ preread \ /* Stack: ret */ \ /preread my_getchar() def % }BIND_DEF #else #if 0 /* old-style: past==0 */ % static void fxSkip(void) { /fxSkip{ past 0 INT_NE{ ASSERT_PREREAD_NOT_EOF() /preread my_getchar() def /past 0 def }if }BIND_DEF % static WORD fxRead8(void) { /fxRead8{ ASSERT_TRUE(past 0 INT_EQ,(past==0)) ASSERT_PREREAD_NOT_EOF() preread /* Stack: ret */ /preread my_getchar() def }BIND_DEF /fxIgnore8z{ % Same as void fxIgnore8(WORD how_many_bytes), but allow 0 as arg ASSERT_TRUE(past 0 INT_EQ,(past==0)) ASSERT_PREREAD_NOT_EOF() dup 0 INT_NE{ -1 2 {pop DEBUG2((fI8z)===) my_ignorechar()}for /preread my_getchar() def }{pop}ifelse }BIND_DEF /fxRead8EOF{ ASSERT_TRUE(past 0 INT_EQ,(past==0)) preread /* Stack: ret */ /preread my_getchar() def }BIND_DEF #else #define fxSkip /past -8 def #define fxRead8 \ ASSERT_TRUE(past -8 INT_EQ,(past==-8)) \ my_getchar() ASSERT_NOT_EOF(dup) % }BIND_DEF #define fxRead8EOF \ ASSERT_TRUE(past -8 INT_EQ,(past==-8)) \ my_getchar() % }BIND_DEF #endif % static WORD fxRead16(void) { /fxRead16{ ASSERT_TRUE(past -8 INT_EQ,(past==-8)) my_getchar() ASSERT_NOT_EOF(dup) my_getchar() ASSERT_NOT_EOF(dup) 8 bitshift add }BIND_DEF % static void fxIgnore8(WORD how_many_bytes) { /fxIgnore8{ ASSERT_TRUE(dup 0 gt,(fxIgnore8arg>0)) ASSERT_TRUE(past -8 INT_EQ,(past==-8)) ASSERT_PREREAD_NOT_EOF() {my_getchar() ASSERT_NOT_EOF(dup) pop }repeat }BIND_DEF #define fxIgnore8z fxIgnore8 #endif #define fvReadq fvRead /** Reads an uint16 from the ZIP file */ #define fxRead16z fxRead16 /** * Allocates a new node and initializes it with empty content. This function * lets us get rid of the global variable F (which was buggily uninitialized * in the original scm.c). * @return pointer to the newly allocated node * @whichcode -1 */ % DEFUN_0(NODEN,fvNalloc) % /fvNalloc{ #define fvNalloc \ /* GLOBAL_REF(N) */ \ /* GLOBAL_REF(Dvar) */ \ /* LOCAL_VAR(NODEN,no) */ /* the first/next free node */ \ /* FUNCODE */ \ Dvar \ /* Stack: no */ \ ASSERT_TRUE(dup NODESIZE lt,(Dvarroot recursively. * Used I as input only. * Moved I into a argeter. * @param arg root node index in N */ % DEFUN_1_VOID(fvFree,NODEN) /fvFree{ % GLOBAL_REF(N) % GLOBAL_REF(Dvar) N exch dup NULL INT_NE{ 2 copy get fvFree 2 copy 1 add get fvFree % fvFree(AREF(N,arg+1)); 2 copy 1 add NULL put % ASET(N,arg+1,NULL); /* clear */ 2 copy 2 add 0 put % ASET(N,arg+2,0); /* clear */ 2 copy Dvar put % ASET(N,arg,Dvar); /* link to the beginning of the free list */ /Dvar exch def pop % SET(Dvar,arg); /* set it free */ }{ % BUG fixed at Thu Dec 27 11:52:59 CET 2001 pop pop }ifelse }BIND_DEF /** * Goes down (descends) enough levels in the binary tree of (struct node)s. * Reads numbers from input encoded as a Huffman tree represented by the * (struct node)s. * Called only from 2 places. * Used I as both input and output, but the output is never used * in the program. So I've put I to a argument. * @param arg root is only for input * @whichcode 3 */ % DEFUN_1(WORD,fvDescend,NODEN) % /fvDescend{ #define fvDescend \ /* GLOBAL_REF(N) */ \ /* FUNCODE */ \ DEBUG(dup (==> fvDescend\() exch 42 string cvs concatstrings (\)\n) concatstrings) \ N exch \ { /* Stack: N arg */ \ 2 copy get NULL INT_EQ{exit}if /* WHILE(NE(NULL,AREF(N,arg))) */ \ 2 copy fxRead1 add get exch pop /* SET(arg,AREF(N,arg+fxRead1())); */ \ }loop /* ENDWHILE */ \ /* Stack: N arg */ \ 2 add get \ DEBUG(dup (<-- fvDescend=) exch 42 string cvs concatstrings (\n) concatstrings) \ /* RETURN(fvDescend,AREF(N,arg+2)) */ /* arg->value; */ % }BIND_DEF /** * Allocates new (struct node)s. This is the Huffman tree * builder. It reads the global array Z: the code lengths have * been already stored there. * Used I as output only, so I moved it to the return value. * @param arg the number of entries (codes) of the Huffman tree to be built * @return the root of the Huffman tree just built * @whichcode 5 */ % DEFUN_1(NODEN,fvMktree,WORD) /fvMktree{ % BUG fixed at Thu Dec 27 11:54:02 CET 2001 % GLOBAL_REF(B) % GLOBAL_REF(G) % GLOBAL_REF(N) % GLOBAL_REF(Z) % LOCAL_VAR(WORD,moo) % LOCAL_VAR(WORD,mQ) % LOCAL_VAR(WORD,mo) % LOCAL_VAR(WORD,mF) % FUNCODE constZ19 0 17 getinterval Bary copy pop % OK to copy packedarray -> array % Stack: arg Bary Z 0 1 4 index 1 sub % Stack: arg Bary Z 0 1 arg-1 { % Stack: arg Bary Z moo 3 copy get % Stack: arg Bary Z moo Bary Z[moo] 2 copy get 1 add put % Stack: arg Bary Z moo pop }for % WHILE(LT(moo,arg)) INCR(AREF(Bary,AREF(Z,moo))); INCR(moo); ENDWHILE % Stack: arg Bary Z pop 0 0 put % ASET(Bary,0,0); Gary 0 0 put % ASET(Gary,0,0); % Stack: arg Gary Bary 0 1 15 % SET(moo,0); WHILE(LT(moo,16)) { % Stack: arg Gary Bary moo 2 copy get % Stack: arg Gary Bary moo Bary[moo] 3 index 2 index get % Stack: arg Gary Bary moo Bary[moo] Gary[moo] add dup add % BUG fixed at Thu Dec 27 15:05:51 CET 2001 % Stack: arg Gary Bary moo 2*(Bary[moo]+Gary[moo]) 3 index exch 2 index 1 add exch % Stack: arg Gary Bary moo Gary moo+1 2*(Bary[moo]+Gary[moo]) % ASET(Gary, moo+1,TWICE(AREF(Gary,moo)+AREF(Bary,moo))); put pop }for % BUG fixed at Thu Dec 27 14:59:52 CET 2001 % Stack: arg Gary Bary pop pop /* Dat: anode is the ->left pointer of the Sentinel node */ N 3 0 put % ASET(N,3,NULL); /* anode=NULL; */ % Stack: arg 1 sub 0 exch 1 exch{ % SET(moo,0); WHILE(LT(moo,arg)) % Stack: moo dup Z exch get dup 0 INT_NE{ % IF(NZ(AREF(Z,moo))) % Stack: moo Z[moo] % /* WORD o, f; */ /* struct node **f; */ dup Gary exch get /mQ exch def % SET(mQ,AREF(Gary,AREF(Z,moo))); % Stack: moo Z[moo] dup Gary exch 2 copy % Stack: moo Z[moo] Gary Z[moo] Gary Z[moo] get 1 add put % INCR(AREF(Gary,AREF(Z,moo))); % Stack: moo Z[moo] /mF 3 def % SET(mF,3); /* mF=&anode; */ % BUG fixed at Thu Dec 27 12:03:31 CET 2001 % SET(mo,AREF(Z,moo)); 1 sub -1 0 % Stack: moo Z[moo]-1 -1 0 { % WHILE(NZ(mo)); DECR(mo); % Stack: moo mo N mF get NULL INT_EQ{ N mF fvNalloc put }if % IF(EQ(AREF(N,mF),NULL)); ASET(N,mF,fvNalloc NOARGS); ENDIF 1 exch bitshift mQ and 0 INT_EQ{0}{1}ifelse N mF get add /mF exch def % SET(mF,AREF(N,mF)+TAND_P(mQ,SHL(1,mo))); % Stack: moo }for N exch % Stack: N moo N mF 2 copy fvNalloc put % ASET(N,mF,fvNalloc NOARGS); get 2 add % Stack: N moo AREF(N,mF)+2 exch put % ASET(N,AREF(N,mF)+2,moo); /* (*f)->value=moo; */ % Stack: - }{pop pop}ifelse % ENDIF }for % Stack: - N 3 get }BIND_DEF #define FMT_ZLIB 120 #define FMT_STOP 4 /* stop processing and flush STDIN */ #if CFG_FMT_ZLIB_ONLYX #else #define FMT_ZIP_STORED 0 /* file is stored, not Deflated in ZIP file */ #define FMT_GZIP 31 #define FMT_NONE 3 #define FMT_ZIP_DEFLATED 8 #endif #define MODE_BOS 0 /* at beginning of (sub)file */ #define MODE_MOB 1 /* at middle of a DEFLATE block */ #define MODE_REP 2 /* printing a repeating string in a DEFLATE block */ #define MODE_BOB 3 /* at the beginning of a block, just before BFINAL and BTYPE */ #define MODE_COPY1 4 /* copying from input to output (a stored block) */ #if CFG_FMT_ZLIB_ONLYX #else #define MODE_ZSTO 5 /* copying from input to output (FMT_ZIP_STORED) */ #endif /** * Root of the literal--length Huffman tree in N * (see RFC 1951). Its length is at most 288 with values 0..287. Values * 286 and 287 are invalid. A value in 0..255 means a literal byte, a * value of 256 indicates end-of-block, and a * value in 257..285 is a prefix (with 29 possible values) of a length * in 3..258. See subsubsection 3.2.5 of RFC 1951. */ SLOCAL_VAR0(NODEN,v) /** * Root of the distance Huffman tree in N (see RFC 1951). * Its length is at most 32 with values 0..31. Values 30 and 31 are invalid. * A value of 0..29 is a prefix (with 30 possible values) of a distance in * 1..32768. See subsubsection 3.2.5 of RFC 1951. */ SLOCAL_VAR0(NODEN,h) #define SLIDEE dup 32767 INT_EQ{pop 0}{1 add}ifelse #if CFG_FMT_ZLIB_ONLY #define fvAFB \ DEBUG2((f0:__LINE__) ===only) DEBUG2(currentfile FP ===) \ /mode MODE_BOS def \ /tY FMT_STOP def /* force EOF */ \ fxSkip \ DEBUG2((f1:__LINE__) ===only) DEBUG2(currentfile FP ===) \ 4 fxIgnore8z \ DEBUG2((f2:__LINE__) ===only) DEBUG2(currentfile FP ===) #else % static void fvAFB(void) { % /fvAFB{ /* after final block (BFINAL) */ #define fvAFB \ \ /mode MODE_BOS def /* SET(mode,MODE_BOS); */ \ fxSkip /* fxSkip(); */ \ /* ^^^ skip till we reach byte boundary. fvRead! 0..7 bits */ \ tY FMT_GZIP INT_EQ{ /* IF(EQ(tY,FMT_GZIP)) */ \ /* CRC32 and ISIZE remaining */ \ /* vvv IGNORE fxRead16(); IGNORE fxRead16(); IGNORE fxRead16(); IGNORE fxRead16(); */ \ 8 \ }{ \ tY FMT_ZLIB INT_EQ{ /* ELSE_IF(EQ(tY,FMT_ZLIB)) */ \ /* ADLER32 remaining */ \ 4 \ }{0}ifelse \ }ifelse \ fxIgnore8z % }BIND_DEF #endif /* CFG_FMT_ZLIB_ONLY */ /** Reads data from input, uncompresses and saves it to Sbuf. Sbuf.length==32768. * Saves 32768 bytes into Sbuf unless EOF prevents it. After return, the caller * may read Sbuf, but it may not modify it. * @return length to be saved to Sbuf */ % static WORD fvWork(void) { /fvWork{ % GLOBAL_REF(J) % GLOBAL_REF(Y) % GLOBAL_REF(Tvar) % GLOBAL_REF(N) % GLOBAL_REF(Dvar) { % `{'': establish fake loop context for return mode MODE_REP INT_EQ{ % IF(EQ(mode,MODE_REP)) ASSERT_TRUE(Tvar 0 INT_EQ,(Tvar==0)) ASSERT_TRUE(f 258 le,(f<=258)) % /* f==0 is OK now */ Tvar oO 1 1 f{ % WHILE(NZ(f)) -- inverse for pop % Stack: Tvar oO Sbuf 3 copy exch get % Stack: Tvar oO Sbuf Tvar S[oO] put % ASET_S(Tvar,AREF_S(oO)); SLIDEE % SLIDE(oO); exch 1 add exch % INCR(Tvar); % DECR(f); }for % ENDWHILE pop /Tvar exch def /mode MODE_MOB def }{ mode MODE_COPY1 INT_EQ{ % ELSE_IF(EQ(mode,MODE_COPY1)) % /* Now: f: copy count unsinged word16 */ ASSERT_TRUE(Tvar 0 INT_EQ,(Tvar==0)) % /* f==0 is OK now */ f 32767 gt{ /* _GT_ */ % IF((unsigned short)f>=32768) ASSERT_TRUE(f 32768 sub f 32767 and eq,((f&32767)==f-32768)) /f f 32767 and def 0 1 32767{ % WHILE(NE((unsigned short)Tvar,32768)) Sbuf exch fxRead8 put % ASET_S(Tvar,fxRead8()); % INCR(T); % DECR(f); }for % ENDWHILE /Tvar 0 def % SET(Tvar,0); % Stack: - ASSERT_TRUE(mode MODE_COPY INT_EQ,(mode==MODE_COPY1)) % bbb111 32768 exit % return (WORD)32768; outside loops }{ % ELSE Sbuf Tvar 1 1 f{ % WHILE(NZ(f)) -- inverse for pop % Stack: Sbuf Tvar 2 copy fxRead8 put % ASET_S(Tvar,fxRead8()); 1 add % INCR(Tvar); }for % ENDWHILE /Tvar exch def pop % /mode MODE_BOB def % SET(mode,MODE_BOB); % BUG fixed at Thu Dec 27 14:16:57 CET 2001 o 0 INT_NE{ % IF(NZ(o)) fvAFB % goto on_bos; SET(mode,MODE_BOS); }{ % ELSE /mode MODE_BOB def % SET(mode,MODE_BOB); % beginning of next block }ifelse % ENDIF }ifelse % ENDIF } #if CFG_FMT_ZLIB_ONLY if #else {mode MODE_ZSTO INT_EQ{ % ELSE_IF(EQ(mode,MODE_ZSTO)) % /* Now: f: copy count low unsinged word16, oP: high unsigned word16 */ % /* Now: oo: 0..1 flag for decrementing oP */ ASSERT_TRUE(Tvar 0 INT_EQ,(Tvar==0)) % /* f==0 is OK now */ f 32767 gt{ /* _GT_ */ % IF((unsigned short)f>=32768) ASSERT_TRUE(f 32768 sub f 32767 and eq,((f&32767)==f-32768)) /f f 32767 and def 0 1 32767{ % WHILE(NE((unsigned short)Tvar,32768)) Sbuf exch fxRead8 put % ASET_S(Tvar,fxRead8()); % INCR(Tvar); % DECR(f); }for % ENDWHILE /Tvar 0 def % SET(Tvar,0); % Stack: - ASSERT_TRUE(mode MODE_ZSTO INT_EQ,(mode==MODE_ZSTO)) % bbb222 32768 exit % return (WORD)32768; outside loops }{ oP 0 INT_NE{ % ELSE_IF(NZ(oP)) 0 1 32767{ % WHILE(NE((unsigned short)Tvar,32768)) Sbuf exch fxRead8 put % ASET_S(Tvar,fxRead8()); % INCR(Tvar); % DECR(f); }for % ENDWHILE /Tvar 0 def % SET(Tvar,0); % Stack: - oO 0 INT_NE{ % IF(NZ(oO)) /oP oP 1 sub def % DECR(oP); /oO 0 def % SET(oO,0); }{ % ELSE /oO 1 def % SET(oO,1); }ifelse ASSERT_TRUE(mode MODE_ZSTO INT_EQ,(mode==MODE_ZSTO)) % bbb333 32768 exit % return (WORD)32768; outside loops }{ % ELSE Sbuf Tvar 1 1 f{ % WHILE(NZ(f)) -- inverse for pop % Stack: Sbuf Tvar 2 copy fxRead8 put % ASET_S(Tvar,fxRead8()); 1 add % INCR(Tvar); }for % ENDWHILE /Tvar exch def pop /mode MODE_BOS def % SET(mode,MODE_BOS); }ifelse % oO!=0 }ifelse % f>=32768 }if % MODE==MODE_ZSTO }ifelse % MODE==MODE_COPY1 #endif /* CFG_FMT_ZLIB_ONLY */ }ifelse % MODE==MODE_REP -1 exit % don''t return yet }loop % Stack: retval||-1 { % big-big MODE_MOB||MODE_BOB||MODE_BOS loop #ifndef NDEBUG % mode ===only ( big-big\n) print #endif ASSERT_TRUE(mode MODE_MOB INT_EQ mode MODE_BOB INT_EQ mode MODE_BOS INT_EQ or or,(MODE_MOB||MODE_BOB||MODE_BOS)) % Stack: retval||-1 dup -1 INT_NE{%ccc999 exit}if mode MODE_MOB INT_EQ{ % IF(EQ(mode,MODE_MOB)); mode_mob: ASSERT_TRUE(dup -1 INT_EQ,(was -1)) pop % -1 /** * Uncompresses the data block with the Huffman codes set up. * * Reads at most 33 bits per entry. Unfortunately the block can be of * arbitrary length (terminated by an entry of 256). */ % /* vvv reads 0..15 bits, see subsubsection 3.2.7 of RFC 1951 */ { % Dat: most of the time is spent inside fvDescend, fvRead and % inside this loop v fvDescend dup 256 INT_EQ{pop -1 exit}if % WHILE(NE(oo, 256)) dup 257 lt{ % IF(LT(oO,257)) % /* ^^^ BUG corrected */ Sbuf exch Tvar exch put % ASET_S(Tvar,oO); % Stack: - % vvv SLIDE(Tvar); /* Tvar++; Tvar&=32767; */ % vvv if (Tvar==0) return (WORD)32768; /* remain in MODE_MOB */ Tvar 32767 INT_EQ{ /Tvar 0 def % bbb444 32768 exit % return % in big-big-loop and MOB-loop }if /Tvar Tvar 1 add def }{ % ELSE 257 sub % SUB(oO,257); % Stack: oO dup constU exch get exch constP exch get fvRead add /f exch def % SET(f,AREF(constU,oO) + fvRead(AREF(constP,oO))); /* fvRead! 0..5 bits */ ASSERT_TRUE(3 f le f 258 le and,(3<=f && f<=258)) h fvDescend % SET(oO,fvDescend(h)); dup constQ exch get exch constL exch get fvRead add % SET(oO,AREF(constQ,oO) + fvRead(AREF(constL,oO))); /* fvRead! 0..13 bits */ Tvar exch sub 32767 and % /* oO = oO <= Tvar ? Tvar - oO : 32768 - oO + Tvar; */ f 32768 Tvar sub sub dup 0 lt{ % IF((unsigned short)f<32768-(unsigned short)Tvar) pop % Stack: oO Tvar exch 1 1 f{ % WHILE(NZ(f)) -- inverse for pop % Stack: Tvar oO Sbuf 3 copy exch get % Stack: Tvar oO Sbuf Tvar S[oO] put % ASET_S(Tvar,AREF_S(oO)); SLIDEE % SLIDE(oO); exch 1 add exch % INCR(Tvar); % DECR(f); }for % ENDWHILE % Dat: oO doesn''t have to be remembered here pop /Tvar exch def % Stack: - ASSERT_TRUE(Tvar 32768 lt,(Tvar<32768)) }{ % ELSE /f exch def % SUB(f,32768-Tvar); % Stack: oO Tvar 1 32767{ % WHILE(NE((unsigned short)Tvar,32768)) % Stack: oO Tvar exch % Stack: Tvar oO Sbuf 3 copy exch get % Stack: Tvar oO Sbuf Tvar S[oO] put % ASET_S(Tvar,AREF_S(oO)); SLIDEE % SLIDE(oO); exch pop % INCR(Tvar); }for % ENDWHILE /oO exch def % Stack: - /Tvar 0 def % SET(Tvar,0); /mode MODE_REP def % SET(mode,MODE_REP); % bbb555 32768 exit % return (WORD)32768; % in big-big-loop and MOB-loop }ifelse % ENDIF }ifelse % ENDIF }loop % ENDWHILE % Stack: retval||-1 dup -1 INT_EQ{ % BUG fixed at Thu Dec 27 15:52:18 CET 2001 v fvFree % fvFree(v); h fvFree % fvFree(h); o 0 INT_NE{ % IF(NZ(o)) fvAFB % goto on_bos; SET(mode,MODE_BOS); }{ % ELSE /mode MODE_BOB def % SET(mode,MODE_BOB); % beginning of next block }ifelse % ENDIF }if }if % ENDIF % Stack: retval||-1 dup -1 INT_NE{%ccc888 exit}if pop DEBUG2((;)===) mode MODE_BOB INT_EQ{ % WHILE(EQ(mode,MODE_BOB)); mode_bob: DEBUG(counttomark 42 string cvs) DEBUG(( kope\n)) % assert(mode==MODE_BOB); /o fxRead1 def % SET(o,fxRead1()); /* BFINAL: 1 iff this is the last data block */ /q 2 fvRead def % SET(q,fvRead(2)); /* BTYPE: block type; 0=stored, 1=fixed Huff, 2=dyn Huff */ % #if NDEBUG % #else % fprintf(stderr, "MODE_BOB o=%d q=%d %ld\n", o, q, ftell(stdin)); % #endif DEBUG((MODE:BOB o=) o 42 string cvs ( q=) q 42 string cvs ( ) Stdin fileposition 42 string cvs (\n) concatstrings concatstrings concatstrings concatstrings concatstrings concatstrings) ASSERT_TRUE(q 0 INT_EQ q 1 INT_EQ q 2 INT_EQ or or,(q==0 || q==1 || q==2)) q 0 INT_NE{ % IF(NZ(q)) /** * Now: q==BTYPE, q==1 or q==2. (BTYPE==3 means error). Handles * compressed data blocks, see subsubsection 3.2.5 of RFC 1951. */ q 1 INT_EQ{ % IF(EQ(q,1)) /** * Initializes the fixed Huffman codes for BTYPE==1. */ % /* WORD oO; */ 287 -1 0{ % SET(oO,288); WHILE(NZ(oO)); DECR(oO); Z exch dup 144 lt{8}{dup 256 lt{9} {dup 280 lt{7}{8}ifelse}ifelse}ifelse put /* ^^^ AREF(Z,oO) = oO < 144 ? 8 : oO < 256 ? 9 : oO < 280 ? 7 : 8; */ }for % ENDWHILE /v 288 fvMktree def % SET(v,fvMktree(288)); 0 1 31{Z exch 5 put}for % SET(f,0); WHILE(LT(f,32)) ASET(Z,f,5); INCR(f); ENDWHILE /h 32 fvMktree def % SET(h,fvMktree(32)); }{ % ELSE /** * Reads dynamic Huffman codes for BTYPE==2. * * Maximum read: 5+5+4+3*19+7*289 == 2094 bits */ ASSERT_TRUE(q 2 INT_EQ,(q==2)) % /* WORD oO, oP, oPo, f, p, x, v; */ /p 5 fvRead 257 add def % SET(p, fvRead(5) + 257); /* HLIT: 257..286 (287--289 are unused) */ /oO 5 fvRead 1 add def % SET(oO, fvRead(5) + 1); /* HDIST: 1..32 */ /v 4 fvRead 4 add def % SET(v, fvRead(4) + 4); /* HCLEN: 4..19 */ /* small v */ constZ19 Z copy pop % WHILE(LT(oO,19)) ASET(Z,AREF(constW,oO), 0); INCR(oO); ENDWHILE 0 1 v 1 sub{ Z exch constW exch get 3 fvRead put % WHILE(LT(oO,v)) ASET(Z,AREF(constW,oO), fvRead(3)); INCR(oO); ENDWHILE /* small v */ }for % Stack: - (verified) /v 19 fvMktree def % SET(v,fvMktree(19)); /oP 0 def % SET(oP,0); % Stack: - Z 0 { % SET(oO,0); WHILE(LT(oO,p + oO)) % Stack: Z oO dup p oO add ge{exit}if v fvDescend % SET(oPo,fvDescend(v)); % Stack: Z oO oPo dup 16 INT_EQ{ pop oP /f 2 fvRead 3 add def % SET(oPo,oP); SET(f,3+fvRead(2)); }{ dup 17 INT_EQ{ pop 0 /f 3 fvRead 3 add def % SET(oPo,0); SET(f,3+fvRead(3)); }{ dup 18 INT_EQ{ pop 0 /f 7 fvRead 11 add def % SET(oPo,0); SET(f,11+fvRead(7)); }{ dup /oP exch def /f 1 def % SET(oP,oPo); SET(f,1); }ifelse }ifelse }ifelse % Stack: Z oO oPo 1 1 f{ % SET(q,f); WHILE(NZ(q)) pop % BUG fixed at Thu Dec 27 12:17:58 CET 2001 3 copy put % ASET(Z,oO,oPo); exch 1 add exch % INCR(oO); % DECR(q); }for % ENDWHILE pop % oPo, BUG fixed at Thu Dec 27 12:18:47 CET 2001 }loop % ENDWHILE % Stack: Z oO''==p+oO pop pop v fvFree % fvFree(v); /v p fvMktree def % SET(v,fvMktree(p)); % Stack: - % /* vvv No need for array copy, just change/pass the pointer to Z... */ oO 1 sub -1 0{ % SET(oO'',oO); WHILE(NZ(oO'')) DECR(oO''); Z exch dup % Stack: Z oO'' oO'' p add Z exch get put % ASET(Z,oO'',AREF(Z,oO'' + p)); }for % ENDWHILE /h oO fvMktree def % SET(h,fvMktree(x)); }ifelse % ENDIF /mode MODE_MOB def % SET(mode,MODE_MOB); % goto mode_mob; }{ % ELSE /* inline: fv(..., 7); */ /** * Copies a block of input to output (mostly) verbatim. This is for * BTYPE==0, non-compressed block, subsubsection 3.2.4 of RFC 1951. * (We need non-compressed because * some blocks cannot be compressed effectively, so gzip inserts them * as is.) * @whichcode 7 */ fxSkip % fxSkip(); % /* ^^^ skip till we reach byte boundary. fvRead! 0..7 bits */ /f fxRead16 def % SET(f,fxRead16()); /* length of block: 0..65535 */ % #if NDEBUG % #else % fprintf(stderr, "COPY1_BLK=%d Tvar=%d\n", f, Tvar); % #endif 2 fxIgnore8 % IGNORE fxRead16(); /* one's complement of length; ignored */ f 32768 Tvar sub sub dup 0 lt{ % IF((unsigned short)f<32768-(unsigned short)Tvar) pop Sbuf Tvar 1 1 f{ % WHILE(NZ(f)) -- inverse for pop % Stack: Sbuf Tvar 2 copy fxRead8 put % ASET_S(Tvar,fxRead8()); 1 add % INCR(Tvar); }for % ENDWHILE /Tvar exch def pop ASSERT_TRUE(mode MODE_BOB INT_EQ,(mode==MODE_BOB)) ASSERT_TRUE(Tvar 32768 lt,(Tvar<32768)) }{ % ELSE % /* Even if f>=32768 */ /f exch def % SUB(f,32768-Tvar); Tvar 1 32767{ % WHILE(NE((unsigned short)Tvar,32768)) Sbuf exch fxRead8 put % ASET_S(Tvar,fxRead8()); % INCR(Tvar); % DECR(f); }for % ENDWHILE /Tvar 0 def % SET(Tvar,0); /mode MODE_COPY1 def % SET(mode,MODE_REP); % bbb666 DEBUG(counttomark 42 string cvs) DEBUG(( whata\n)) % trtr ASSERT_TRUE(counttomark 0 eq,(seccounttomark==0)) DEBUG((whatb\n)) 32768 exit % return (WORD)32768; % exit from big-big loop }ifelse % ENDIF o 0 INT_NE{fvAFB}if % IF(NZ(o)); fvAFB(); ENDIF DEBUG( counttomark 42 string cvs) DEBUG((pikula\n)) }ifelse % ENDIF }if % ENDWHILE /* MODE_BOB */ % Stack: - mode MODE_BOS INT_EQ{ DEBUG(counttomark 42 string cvs) DEBUG(( booski\n)) DEBUG2((boost)===only) DEBUG2(currentfile FP ===) { DEBUG(counttomark 42 string cvs) DEBUG(( boosbe\n)) tY FMT_STOP INT_EQ{ % WHILE(NE(tY,FMT_STOP)) Tvar 0 INT_NE{ % /* Flush unwritten buffer. */ Tvar% BUG fixed at Thu Dec 27 14:05:32 CET 2001 /Tvar 0 def % Stack: Tvar_old }{0}ifelse % ccc777 exit % return % in big-big loop and BOS-loop }if % /* SET(oO,0); SET(oP,0); */ /* avoid GCC warnings */ % /* fxInit(); */ /* assert(Y==0); SET(J,0); SET(Y,0); */ /v NULL def /h NULL def % SET(v,NULL); SET(h,NULL); % N 0 NULL put N 1 NULL put N 2 0 put % ASET(N,0,NULL); ASET(N,1,NULL); ASET(N,2,0); /* the NULL node is initially empty */ % N 3 NULL put N 4 NULL put N 5 0 put % ASET(N,3,NULL); ASET(N,4,NULL); ASET(N,5,0); /* the Sentinel node is initially empty */ constZ19 N copy pop % eliminate-NULLs trick: overwrites 1st >=6 elements of N /Dvar 6 def % SET(Dvar,6); /* first free node is 6. `0' is NULL, `3' is Sentinel */ 6{ % SET(o,Dvar); dup NODESIZE ge{pop exit}if % WHILE (LT(o,NODESIZE)) dup N exch dup 3 add put 1 add % ASET(N,o,o+3); INCR(o); /* next free node is next node */ dup N exch NULL put 1 add % ASET(N,o,NULL); INCR(o); /* empty RIGHT */ dup N exch 0 put 1 add % ASET(N,o,0); INCR(o); /* empty NVAL */ }loop % ENDWHILE % Stack: - fxRead8EOF dup /tY exch def % SET(tY,fxRead8EOF()); /* read first byte of the header */ % tY kkkkkkkkkk dup 511 INT_EQ{ % IF(EQ(tY,511)) /* EOF */ /tY FMT_STOP def % SET(tY,FMT_STOP); }{ #if CFG_FMT_ZLIB_ONLY 1 fxIgnore8 % IGNORE fxRead8(); /* skip second header byte: 0x01 or 0x5e or 0x9c or 0xda */ #else dup FMT_ZLIB INT_EQ{ % ELSE_IF(EQ(tY,120)) /* ZLIB format */ 1 fxIgnore8 % IGNORE fxRead8(); /* skip second header byte: 0x01 or 0x5e or 0x9c or 0xda */ % /* SET(tY,FMT_ZLIB); */ }{ dup 80 INT_EQ{ % ELSE_IF(EQ(tY,80)) /* ZIP format */ 1 fxIgnore8 % IGNORE fxRead8(); /* skip second header byte: 0x48 */ /tY FMT_NONE def % SET(tY,FMT_NONE); fxRead8 % SET(o,fxRead8()); dup 3 INT_EQ{ % IF(EQ(o,3)) /* Local file header */ % fxRead8 pop % IGNORE fxRead8(); /* skip: 0x04 */ % fxRead16 pop % IGNORE fxRead16(); /* skip: version needed to extract file (0x0020) */ % fxRead16 pop % IGNORE fxRead16(); /* LOCFLG flags */ 5 fxIgnore8 /tY fxRead8 def % SET(tY,fxRead8()); /* lower half of compression method */ ASSERT_TRUE(tY FMT_ZIP_STORED INT_EQ tY FMT_ZIP_DEFLATED INT_EQ or,(ty==FMT_ZIP_STORED || ty==FMT_ZIP_DEFLATED)) % ^^^ BUG fixed at Thu Dec 27 20:55:52 CET 2001 % fxRead8 pop % IGNORE fxRead8(); /* upper half of compression method */ % fxRead16 pop fxRead16 pop % IGNORE fxRead16(); IGNORE fxRead16(); /* file modification time in MS-DOS format */ % fxRead16 pop fxRead16 pop % IGNORE fxRead16(); IGNORE fxRead16(); /* some kind of CRC-32 */ 9 fxIgnore8 /f fxRead16z def % SET(f, fxRead16z()); /* lower compressed file size */ /oP fxRead16z def % SET(oP,fxRead16z()); /* higher compressed file size */ 4 fxIgnore8 % fxRead16 pop fxRead16 pop % IGNORE fxRead16(); IGNORE fxRead16(); /* uncompressed file size */ fxRead16z % SET(oO,fxRead16z()); /* length of filename */ fxRead16z add % SET(q,fxRead16z()); /* length of extra field */ % WHILE(NZ(oO)) IGNORE fxRead8(); DECR(oO); ENDWHILE /* file name */ % WHILE(NZ(q)) IGNORE fxRead8(); DECR(q); ENDWHILE /* extra field */ % -1 1{fxRead8 pop}for fxIgnore8z }{ dup 7 INT_EQ{ % ELSE_IF(EQ(o,7)) /* Extended local header of previous file in ZIP */ % 1 1 13{fxRead8 pop}for % SET(o,0); WHILE(LT(o,13)) IGNORE fxRead8(); INCR(o); ENDWHILE /* BUGFIX: was 15 */ 13 fxIgnore8 }{ dup 5 INT_EQ{ % ELSE_IF(EQ(o,5)) /* End of Central Directory Structure in ZIP */ /* fprintf(stderr,"EOCDS\n"); */ % 1 1 17{fxRead8 pop}for % SET(o,0); WHILE(LT(o,17)) IGNORE fxRead8(); INCR(o); ENDWHILE 17 fxIgnore8 fxRead16z % SET(o,fxRead16z()); /* CML: archive comment length */ % -1 1{fxRead8 pop}for % WHILE (NZ(o)) IGNORE fxRead8(); DECR(o); ENDWHILE fxIgnore8z }{ dup 1 INT_EQ{ % ELSE_IF(EQ(o,1)) /* Central Directory Structure */ /* fprintf(stderr,"CDS\n"); */ % 1 1 25{fxRead8 pop}for % SET(oO,0); WHILE(LT(oO,25)) IGNORE fxRead8(); INCR(oO); ENDWHILE 25 fxIgnore8 fxRead16z % SET(f,fxRead16z()); /* LEN: length of file name */ fxRead16z add % SET(o,fxRead16z()); /* XLN: length of extra field */ fxRead16z add % SET(q,fxRead16z()); /* CML: length of file comment */ % fxRead16 pop fxRead16 pop fxRead16 pop % fxRead16 pop fxRead16 pop fxRead16 pop % ^^^ SET(oO,0); WHILE(LT(oO,12)) IGNORE fxRead8(); INCR(oO); ENDWHILE 12 fxIgnore8 fxIgnore8z % ^^^ WHILE(NZ(f)) IGNORE fxRead8(); DECR(f); ENDWHILE /* file name */ % WHILE(NZ(o)) IGNORE fxRead8(); DECR(o); ENDWHILE /* extra field */ % WHILE(NZ(q)) IGNORE fxRead8(); DECR(q); ENDWHILE /* file comment */ }if % o==1 }ifelse % o==5 }ifelse % o==7 }ifelse % o==3 % ENDIF /* IF ZIP structure sub-type */ % Stack: tY_old o pop }{ dup 31 INT_EQ{ % ELSE_IF(EQ(tY,31)) /* gzip/RFC 1952 format */ /* fprintf(stderr,"gzip!\n"); */ /* SET(tY,FMT_GZIP); */ /* The most simple gzip header (10 bytes): * ID1 hex 0x1f == dec 31 == FMT_GZIP * ID2 hex 0x8b * CM hex 0x08 * FLG hex 0x00 * MTIME hex 0x00, 0x00, 0x00, 0x00 (1 Jan 1970, UNIX epoch) * XFL hex 0x00 * OS hex 0xff * After that comes the compressed data stream. * After that comes the CRC32 and ISIZE (byte aligned? ?) */ 2 fxIgnore8 % fxRead16 pop % IGNORE fxRead16(); /* ignore ID2 and CM */ % kkkkkkkkkl fxRead8 % SET(o,fxRead8()); /* FLG */ % fxRead16 pop fxRead16 pop % IGNORE fxRead16(); IGNORE fxRead16(); /* ignore MTIME */ % fxRead16 pop % IGNORE fxRead16(); /* ignore XFL, OS */ 6 fxIgnore8 % Stack: o==FLG dup 2 and 0 INT_NE{ % IF(TAND_P(o,2)) /* GCONT: skip part number of continuation */ %fxRead16 pop % IGNORE fxRead16(); 2 fxIgnore8 }if % ENDIF dup 4 and 0 INT_NE{ % IF(TAND_P(o,4)) /* ignore FEXTRA */ fxRead16 fxIgnore8 % 1 1 fxRead16{fxRead8 pop}for % SET(q,fxRead16()); % WHILE(NZ(q)) IGNORE fxRead8(); DECR(q); ENDWHILE }if % ENDIF dup 8 and 0 INT_NE{ % IF(TAND_P(o,8)) {fxRead8 0 INT_EQ{exit}if}loop % WHILE(NZ(fxRead8())); PASS; ENDWHILE }if % ENDIF /* ignore FNAME */ dup 17 and 0 INT_NE{ % IF(TAND_P(o,16)) {fxRead8 0 INT_EQ{exit}if}loop % WHILE(NZ(fxRead8())); PASS; ENDWHILE }if % ENDIF /* ignore FCOMMENT */ dup 32 and 0 INT_NE{ % IF(TAND_P(o,32)) /* skip encryption header */ % fxRead16 pop fxRead16 pop fxRead16 pop % fxRead16 pop fxRead16 pop fxRead16 pop % SET(f,0); WHILE(LT(f,12)) IGNORE fxRead8(); INCR(f); ENDWHILE 12 fxIgnore8 }if % ENDIF % Stack: tY o pop }if % tY==31 (gzip/RFC 1952 format) }ifelse % tY==80 (ZIP format) }ifelse % tY==FMT_ZLIB #endif /* CFG_FMT_ZLIB_ONLY */ }ifelse % tY==511 % ENDIF /* IF file format */ % Stack: tY_old pop % /* fprintf(stderr,"tY=%d\n", tY); */ % tY bbbbbc #if CFG_FMT_ZLIB_ONLYX /* FMT_ZIP_STORED can become alive somehow... */ tY FMT_STOP INT_NE tY 3 INT_NE and{ /mode MODE_BOB def % SET(mode,MODE_BOB); -1 exit % goto mode_bob; }if #else tY FMT_ZIP_STORED INT_EQ{ % IF(EQ(tY,FMT_ZIP_STORED)) % /* fprintf(stderr,"ZIP_STORED oO=%d oP=%d\n", oO, oP); */ /oO 0 def % SET(oO,0); f 32768 Tvar sub sub dup 0 lt{ % IF((unsigned short)f<32768-(unsigned short)Tvar) pop Sbuf Tvar 1 1 f{ % WHILE(NZ(f)) -- inverse for pop % Stack: Sbuf Tvar 2 copy fxRead8 put % ASET_S(Tvar,fxRead8()); 1 add % INCR(Tvar); }for % ENDWHILE /Tvar exch def pop ASSERT_TRUE(Tvar 32768 lt,(Tvar<32768)) oP 0 INT_NE{ /mode MODE_ZSTO def % ccc555 Tvar exit % in big-big loop and BOS-loop }if % ^^^ IF(NZ(oP)); mode=MODE_ZSTO; return Tvar; ENDIF ASSERT_TRUE(mode MODE_BOS INT_EQ,(mode==MODE_BOS)) }{ % ELSE % /* Even if f>=32768 */ /f exch def % SUB(f,32768-Tvar); Tvar 1 32767{ % WHILE(NE((unsigned short)Tvar,32768)) Sbuf exch fxRead8 put % ASET_S(Tvar,fxRead8()); % INCR(Tvar); % DECR(f); }for % ENDWHILE /Tvar 0 def % SET(Tvar,0); /mode MODE_ZSTO def % SET(mode,MODE_REP); % bbb777 32768 exit % return (WORD)32768; % in big-big loop and BOS-loop }ifelse % ENDIF }{ tY FMT_STOP INT_NE tY FMT_NONE INT_NE and{ /mode MODE_BOB def % SET(mode,MODE_BOB); -1 exit % goto mode_bob; }if }ifelse % ENDIF #endif /* CFG_FMT_ZLIB_ONLY */ }loop % ENDWHILE /* outermost WHILE(NE(tY,FMT_STOP)) */ % Stack: retval }{-1}ifelse % mode==MODE_BOS % Stack: retval }loop % big-big MODE_MOB||MODE_BOB||MODE_BOS loop % Stack: retval }BIND_DEF % ENDFUN % Stdin already set #if 0 % /fvInitAll{ #define fvInitAll \ fxInit \ /tY FMT_NONE def \ /mode MODE_BOS def \ /Tvar 0 def /* make the round buffer empty */ \ TE_init % }BIND_DEF #else /fvMain{ % /XXX % STDIN read % STDIN C readhexstring % pop 0 get true_action % TE_read(===) TE_init % must be called _before_ fxInit fxInit /tY FMT_NONE def /mode MODE_BOS def /Tvar 0 def % /Tvar 0 def Tvar === quit { Sbuf 0 fvWork getinterval } /G load exec TE_read_eod % make sure that mEOD has been read }BIND_DEF #endif } /* We close the brace of declared functions */ % % %%BeginData: exec `S %%EndData end restore showpage %%Trailer %%EOF % %%EOF