1*37da2899SCharles.Forsyth #include "lib9.h"
2*37da2899SCharles.Forsyth #include "draw.h"
3*37da2899SCharles.Forsyth #include "memdraw.h"
4*37da2899SCharles.Forsyth
5*37da2899SCharles.Forsyth #define CHUNK 8000
6*37da2899SCharles.Forsyth
7*37da2899SCharles.Forsyth #define HSHIFT 3 /* HSHIFT==5 runs slightly faster, but hash table is 64x bigger */
8*37da2899SCharles.Forsyth #define NHASH (1<<(HSHIFT*NMATCH))
9*37da2899SCharles.Forsyth #define HMASK (NHASH-1)
10*37da2899SCharles.Forsyth #define hupdate(h, c) ((((h)<<HSHIFT)^(c))&HMASK)
11*37da2899SCharles.Forsyth typedef struct Hlist Hlist;
12*37da2899SCharles.Forsyth struct Hlist{
13*37da2899SCharles.Forsyth uchar *s;
14*37da2899SCharles.Forsyth Hlist *next, *prev;
15*37da2899SCharles.Forsyth };
16*37da2899SCharles.Forsyth
17*37da2899SCharles.Forsyth int
writememimage(int fd,Memimage * i)18*37da2899SCharles.Forsyth writememimage(int fd, Memimage *i)
19*37da2899SCharles.Forsyth {
20*37da2899SCharles.Forsyth uchar *outbuf, *outp, *eout; /* encoded data, pointer, end */
21*37da2899SCharles.Forsyth uchar *loutp; /* start of encoded line */
22*37da2899SCharles.Forsyth Hlist *hash; /* heads of hash chains of past strings */
23*37da2899SCharles.Forsyth Hlist *chain, *hp; /* hash chain members, pointer */
24*37da2899SCharles.Forsyth Hlist *cp; /* next Hlist to fall out of window */
25*37da2899SCharles.Forsyth int h; /* hash value */
26*37da2899SCharles.Forsyth uchar *line, *eline; /* input line, end pointer */
27*37da2899SCharles.Forsyth uchar *data, *edata; /* input buffer, end pointer */
28*37da2899SCharles.Forsyth ulong n; /* length of input buffer */
29*37da2899SCharles.Forsyth ulong nb; /* # of bytes returned by unloadimage */
30*37da2899SCharles.Forsyth int bpl; /* input line length */
31*37da2899SCharles.Forsyth int offs, runlen; /* offset, length of consumed data */
32*37da2899SCharles.Forsyth uchar dumpbuf[NDUMP]; /* dump accumulator */
33*37da2899SCharles.Forsyth int ndump; /* length of dump accumulator */
34*37da2899SCharles.Forsyth int miny, dy; /* y values while unloading input */
35*37da2899SCharles.Forsyth int ncblock; /* size of compressed blocks */
36*37da2899SCharles.Forsyth Rectangle r;
37*37da2899SCharles.Forsyth uchar *p, *q, *s, *es, *t;
38*37da2899SCharles.Forsyth char hdr[11+5*12+1];
39*37da2899SCharles.Forsyth char cbuf[20];
40*37da2899SCharles.Forsyth
41*37da2899SCharles.Forsyth r = i->r;
42*37da2899SCharles.Forsyth bpl = bytesperline(r, i->depth);
43*37da2899SCharles.Forsyth n = Dy(r)*bpl;
44*37da2899SCharles.Forsyth data = malloc(n);
45*37da2899SCharles.Forsyth ncblock = _compblocksize(r, i->depth);
46*37da2899SCharles.Forsyth outbuf = malloc(ncblock);
47*37da2899SCharles.Forsyth hash = malloc(NHASH*sizeof(Hlist));
48*37da2899SCharles.Forsyth chain = malloc(NMEM*sizeof(Hlist));
49*37da2899SCharles.Forsyth if(data == 0 || outbuf == 0 || hash == 0 || chain == 0){
50*37da2899SCharles.Forsyth ErrOut:
51*37da2899SCharles.Forsyth free(data);
52*37da2899SCharles.Forsyth free(outbuf);
53*37da2899SCharles.Forsyth free(hash);
54*37da2899SCharles.Forsyth free(chain);
55*37da2899SCharles.Forsyth return -1;
56*37da2899SCharles.Forsyth }
57*37da2899SCharles.Forsyth for(miny = r.min.y; miny != r.max.y; miny += dy){
58*37da2899SCharles.Forsyth dy = r.max.y-miny;
59*37da2899SCharles.Forsyth if(dy*bpl > CHUNK)
60*37da2899SCharles.Forsyth dy = CHUNK/bpl;
61*37da2899SCharles.Forsyth nb = unloadmemimage(i, Rect(r.min.x, miny, r.max.x, miny+dy),
62*37da2899SCharles.Forsyth data+(miny-r.min.y)*bpl, dy*bpl);
63*37da2899SCharles.Forsyth if(nb != dy*bpl)
64*37da2899SCharles.Forsyth goto ErrOut;
65*37da2899SCharles.Forsyth }
66*37da2899SCharles.Forsyth sprint(hdr, "compressed\n%11s %11d %11d %11d %11d ",
67*37da2899SCharles.Forsyth chantostr(cbuf, i->chan), r.min.x, r.min.y, r.max.x, r.max.y);
68*37da2899SCharles.Forsyth if(write(fd, hdr, 11+5*12) != 11+5*12)
69*37da2899SCharles.Forsyth goto ErrOut;
70*37da2899SCharles.Forsyth edata = data+n;
71*37da2899SCharles.Forsyth eout = outbuf+ncblock;
72*37da2899SCharles.Forsyth line = data;
73*37da2899SCharles.Forsyth r.max.y = r.min.y;
74*37da2899SCharles.Forsyth while(line != edata){
75*37da2899SCharles.Forsyth memset(hash, 0, NHASH*sizeof(Hlist));
76*37da2899SCharles.Forsyth memset(chain, 0, NMEM*sizeof(Hlist));
77*37da2899SCharles.Forsyth cp = chain;
78*37da2899SCharles.Forsyth h = 0;
79*37da2899SCharles.Forsyth outp = outbuf;
80*37da2899SCharles.Forsyth for(n = 0; n != NMATCH; n++)
81*37da2899SCharles.Forsyth h = hupdate(h, line[n]);
82*37da2899SCharles.Forsyth loutp = outbuf;
83*37da2899SCharles.Forsyth while(line != edata){
84*37da2899SCharles.Forsyth ndump = 0;
85*37da2899SCharles.Forsyth eline = line+bpl;
86*37da2899SCharles.Forsyth for(p = line; p != eline; ){
87*37da2899SCharles.Forsyth if(eline-p < NRUN)
88*37da2899SCharles.Forsyth es = eline;
89*37da2899SCharles.Forsyth else
90*37da2899SCharles.Forsyth es = p+NRUN;
91*37da2899SCharles.Forsyth q = 0;
92*37da2899SCharles.Forsyth runlen = 0;
93*37da2899SCharles.Forsyth for(hp = hash[h].next; hp; hp = hp->next){
94*37da2899SCharles.Forsyth s = p + runlen;
95*37da2899SCharles.Forsyth if(s >= es)
96*37da2899SCharles.Forsyth continue;
97*37da2899SCharles.Forsyth t = hp->s + runlen;
98*37da2899SCharles.Forsyth for(; s >= p; s--)
99*37da2899SCharles.Forsyth if(*s != *t--)
100*37da2899SCharles.Forsyth goto matchloop;
101*37da2899SCharles.Forsyth t += runlen+2;
102*37da2899SCharles.Forsyth s += runlen+2;
103*37da2899SCharles.Forsyth for(; s < es; s++)
104*37da2899SCharles.Forsyth if(*s != *t++)
105*37da2899SCharles.Forsyth break;
106*37da2899SCharles.Forsyth n = s-p;
107*37da2899SCharles.Forsyth if(n > runlen){
108*37da2899SCharles.Forsyth runlen = n;
109*37da2899SCharles.Forsyth q = hp->s;
110*37da2899SCharles.Forsyth if(n == NRUN)
111*37da2899SCharles.Forsyth break;
112*37da2899SCharles.Forsyth }
113*37da2899SCharles.Forsyth matchloop: ;
114*37da2899SCharles.Forsyth }
115*37da2899SCharles.Forsyth if(runlen < NMATCH){
116*37da2899SCharles.Forsyth if(ndump == NDUMP){
117*37da2899SCharles.Forsyth if(eout-outp < ndump+1)
118*37da2899SCharles.Forsyth goto Bfull;
119*37da2899SCharles.Forsyth *outp++ = ndump-1+128;
120*37da2899SCharles.Forsyth memmove(outp, dumpbuf, ndump);
121*37da2899SCharles.Forsyth outp += ndump;
122*37da2899SCharles.Forsyth ndump = 0;
123*37da2899SCharles.Forsyth }
124*37da2899SCharles.Forsyth dumpbuf[ndump++] = *p;
125*37da2899SCharles.Forsyth runlen = 1;
126*37da2899SCharles.Forsyth }
127*37da2899SCharles.Forsyth else{
128*37da2899SCharles.Forsyth if(ndump != 0){
129*37da2899SCharles.Forsyth if(eout-outp < ndump+1)
130*37da2899SCharles.Forsyth goto Bfull;
131*37da2899SCharles.Forsyth *outp++ = ndump-1+128;
132*37da2899SCharles.Forsyth memmove(outp, dumpbuf, ndump);
133*37da2899SCharles.Forsyth outp += ndump;
134*37da2899SCharles.Forsyth ndump = 0;
135*37da2899SCharles.Forsyth }
136*37da2899SCharles.Forsyth offs = p-q-1;
137*37da2899SCharles.Forsyth if(eout-outp < 2)
138*37da2899SCharles.Forsyth goto Bfull;
139*37da2899SCharles.Forsyth *outp++ = ((runlen-NMATCH)<<2) + (offs>>8);
140*37da2899SCharles.Forsyth *outp++ = offs&255;
141*37da2899SCharles.Forsyth }
142*37da2899SCharles.Forsyth for(q = p+runlen; p != q; p++){
143*37da2899SCharles.Forsyth if(cp->prev)
144*37da2899SCharles.Forsyth cp->prev->next = 0;
145*37da2899SCharles.Forsyth cp->next = hash[h].next;
146*37da2899SCharles.Forsyth cp->prev = &hash[h];
147*37da2899SCharles.Forsyth if(cp->next)
148*37da2899SCharles.Forsyth cp->next->prev = cp;
149*37da2899SCharles.Forsyth cp->prev->next = cp;
150*37da2899SCharles.Forsyth cp->s = p;
151*37da2899SCharles.Forsyth if(++cp == &chain[NMEM])
152*37da2899SCharles.Forsyth cp = chain;
153*37da2899SCharles.Forsyth if(edata-p > NMATCH)
154*37da2899SCharles.Forsyth h = hupdate(h, p[NMATCH]);
155*37da2899SCharles.Forsyth }
156*37da2899SCharles.Forsyth }
157*37da2899SCharles.Forsyth if(ndump != 0){
158*37da2899SCharles.Forsyth if(eout-outp < ndump+1)
159*37da2899SCharles.Forsyth goto Bfull;
160*37da2899SCharles.Forsyth *outp++ = ndump-1+128;
161*37da2899SCharles.Forsyth memmove(outp, dumpbuf, ndump);
162*37da2899SCharles.Forsyth outp += ndump;
163*37da2899SCharles.Forsyth }
164*37da2899SCharles.Forsyth line = eline;
165*37da2899SCharles.Forsyth loutp = outp;
166*37da2899SCharles.Forsyth r.max.y++;
167*37da2899SCharles.Forsyth }
168*37da2899SCharles.Forsyth Bfull:
169*37da2899SCharles.Forsyth if(loutp == outbuf)
170*37da2899SCharles.Forsyth goto ErrOut;
171*37da2899SCharles.Forsyth n = loutp-outbuf;
172*37da2899SCharles.Forsyth sprint(hdr, "%11d %11ld ", r.max.y, n);
173*37da2899SCharles.Forsyth write(fd, hdr, 2*12);
174*37da2899SCharles.Forsyth write(fd, outbuf, n);
175*37da2899SCharles.Forsyth r.min.y = r.max.y;
176*37da2899SCharles.Forsyth }
177*37da2899SCharles.Forsyth free(data);
178*37da2899SCharles.Forsyth free(outbuf);
179*37da2899SCharles.Forsyth free(hash);
180*37da2899SCharles.Forsyth free(chain);
181*37da2899SCharles.Forsyth return 0;
182*37da2899SCharles.Forsyth }
183