17dd7cddfSDavid du Colombier #include <u.h>
27dd7cddfSDavid du Colombier #include <libc.h>
37dd7cddfSDavid du Colombier #include <draw.h>
47dd7cddfSDavid du Colombier
57dd7cddfSDavid du Colombier #define HSHIFT 3 /* HSHIFT==5 runs slightly faster, but hash table is 64x bigger */
67dd7cddfSDavid du Colombier #define NHASH (1<<(HSHIFT*NMATCH))
77dd7cddfSDavid du Colombier #define HMASK (NHASH-1)
87dd7cddfSDavid du Colombier #define hupdate(h, c) ((((h)<<HSHIFT)^(c))&HMASK)
97dd7cddfSDavid du Colombier typedef struct Hlist Hlist;
107dd7cddfSDavid du Colombier struct Hlist{
117dd7cddfSDavid du Colombier uchar *s;
127dd7cddfSDavid du Colombier Hlist *next, *prev;
137dd7cddfSDavid du Colombier };
147dd7cddfSDavid du Colombier
157dd7cddfSDavid du Colombier int
writeimage(int fd,Image * i,int dolock)167dd7cddfSDavid du Colombier writeimage(int fd, Image *i, int dolock)
177dd7cddfSDavid du Colombier {
187dd7cddfSDavid du Colombier uchar *outbuf, *outp, *eout; /* encoded data, pointer, end */
197dd7cddfSDavid du Colombier uchar *loutp; /* start of encoded line */
207dd7cddfSDavid du Colombier Hlist *hash; /* heads of hash chains of past strings */
217dd7cddfSDavid du Colombier Hlist *chain, *hp; /* hash chain members, pointer */
227dd7cddfSDavid du Colombier Hlist *cp; /* next Hlist to fall out of window */
237dd7cddfSDavid du Colombier int h; /* hash value */
247dd7cddfSDavid du Colombier uchar *line, *eline; /* input line, end pointer */
257dd7cddfSDavid du Colombier uchar *data, *edata; /* input buffer, end pointer */
267dd7cddfSDavid du Colombier ulong n; /* length of input buffer */
277dd7cddfSDavid du Colombier ulong nb; /* # of bytes returned by unloadimage */
287dd7cddfSDavid du Colombier int bpl; /* input line length */
297dd7cddfSDavid du Colombier int offs, runlen; /* offset, length of consumed data */
307dd7cddfSDavid du Colombier uchar dumpbuf[NDUMP]; /* dump accumulator */
317dd7cddfSDavid du Colombier int ndump; /* length of dump accumulator */
327dd7cddfSDavid du Colombier int miny, dy; /* y values while unloading input */
33*9a747e4fSDavid du Colombier int chunk, ncblock;
347dd7cddfSDavid du Colombier Rectangle r;
357dd7cddfSDavid du Colombier uchar *p, *q, *s, *es, *t;
367dd7cddfSDavid du Colombier char hdr[11+5*12+1];
377dd7cddfSDavid du Colombier char cbuf[20];
387dd7cddfSDavid du Colombier
39*9a747e4fSDavid du Colombier chunk = i->display->bufsize - 32; /* a little room for header */
407dd7cddfSDavid du Colombier r = i->r;
417dd7cddfSDavid du Colombier bpl = bytesperline(r, i->depth);
427dd7cddfSDavid du Colombier n = Dy(r)*bpl;
437dd7cddfSDavid du Colombier data = malloc(n);
44*9a747e4fSDavid du Colombier ncblock = _compblocksize(r, i->depth);
45*9a747e4fSDavid du Colombier outbuf = malloc(ncblock);
467dd7cddfSDavid du Colombier hash = malloc(NHASH*sizeof(Hlist));
477dd7cddfSDavid du Colombier chain = malloc(NMEM*sizeof(Hlist));
487dd7cddfSDavid du Colombier if(data == 0 || outbuf == 0 || hash == 0 || chain == 0){
497dd7cddfSDavid du Colombier ErrOut:
507dd7cddfSDavid du Colombier free(data);
517dd7cddfSDavid du Colombier free(outbuf);
527dd7cddfSDavid du Colombier free(hash);
537dd7cddfSDavid du Colombier free(chain);
547dd7cddfSDavid du Colombier return -1;
557dd7cddfSDavid du Colombier }
567dd7cddfSDavid du Colombier for(miny = r.min.y; miny != r.max.y; miny += dy){
577dd7cddfSDavid du Colombier dy = r.max.y-miny;
58*9a747e4fSDavid du Colombier if(dy*bpl > chunk)
59*9a747e4fSDavid du Colombier dy = chunk/bpl;
607dd7cddfSDavid du Colombier if(dolock)
617dd7cddfSDavid du Colombier lockdisplay(i->display);
627dd7cddfSDavid du Colombier nb = unloadimage(i, Rect(r.min.x, miny, r.max.x, miny+dy),
637dd7cddfSDavid du Colombier data+(miny-r.min.y)*bpl, dy*bpl);
647dd7cddfSDavid du Colombier if(dolock)
657dd7cddfSDavid du Colombier unlockdisplay(i->display);
667dd7cddfSDavid du Colombier if(nb != dy*bpl)
677dd7cddfSDavid du Colombier goto ErrOut;
687dd7cddfSDavid du Colombier }
697dd7cddfSDavid du Colombier sprint(hdr, "compressed\n%11s %11d %11d %11d %11d ",
707dd7cddfSDavid du Colombier chantostr(cbuf, i->chan), r.min.x, r.min.y, r.max.x, r.max.y);
717dd7cddfSDavid du Colombier if(write(fd, hdr, 11+5*12) != 11+5*12)
727dd7cddfSDavid du Colombier goto ErrOut;
737dd7cddfSDavid du Colombier edata = data+n;
74*9a747e4fSDavid du Colombier eout = outbuf+ncblock;
757dd7cddfSDavid du Colombier line = data;
767dd7cddfSDavid du Colombier r.max.y = r.min.y;
777dd7cddfSDavid du Colombier while(line != edata){
787dd7cddfSDavid du Colombier memset(hash, 0, NHASH*sizeof(Hlist));
797dd7cddfSDavid du Colombier memset(chain, 0, NMEM*sizeof(Hlist));
807dd7cddfSDavid du Colombier cp = chain;
817dd7cddfSDavid du Colombier h = 0;
827dd7cddfSDavid du Colombier outp = outbuf;
837dd7cddfSDavid du Colombier for(n = 0; n != NMATCH; n++)
847dd7cddfSDavid du Colombier h = hupdate(h, line[n]);
857dd7cddfSDavid du Colombier loutp = outbuf;
867dd7cddfSDavid du Colombier while(line != edata){
877dd7cddfSDavid du Colombier ndump = 0;
887dd7cddfSDavid du Colombier eline = line+bpl;
897dd7cddfSDavid du Colombier for(p = line; p != eline; ){
907dd7cddfSDavid du Colombier if(eline-p < NRUN)
917dd7cddfSDavid du Colombier es = eline;
927dd7cddfSDavid du Colombier else
937dd7cddfSDavid du Colombier es = p+NRUN;
947dd7cddfSDavid du Colombier q = 0;
957dd7cddfSDavid du Colombier runlen = 0;
967dd7cddfSDavid du Colombier for(hp = hash[h].next; hp; hp = hp->next){
977dd7cddfSDavid du Colombier s = p + runlen;
987dd7cddfSDavid du Colombier if(s >= es)
997dd7cddfSDavid du Colombier continue;
1007dd7cddfSDavid du Colombier t = hp->s + runlen;
1017dd7cddfSDavid du Colombier for(; s >= p; s--)
1027dd7cddfSDavid du Colombier if(*s != *t--)
1037dd7cddfSDavid du Colombier goto matchloop;
1047dd7cddfSDavid du Colombier t += runlen+2;
1057dd7cddfSDavid du Colombier s += runlen+2;
1067dd7cddfSDavid du Colombier for(; s < es; s++)
1077dd7cddfSDavid du Colombier if(*s != *t++)
1087dd7cddfSDavid du Colombier break;
1097dd7cddfSDavid du Colombier n = s-p;
1107dd7cddfSDavid du Colombier if(n > runlen){
1117dd7cddfSDavid du Colombier runlen = n;
1127dd7cddfSDavid du Colombier q = hp->s;
1137dd7cddfSDavid du Colombier if(n == NRUN)
1147dd7cddfSDavid du Colombier break;
1157dd7cddfSDavid du Colombier }
1167dd7cddfSDavid du Colombier matchloop: ;
1177dd7cddfSDavid du Colombier }
1187dd7cddfSDavid du Colombier if(runlen < NMATCH){
1197dd7cddfSDavid du Colombier if(ndump == NDUMP){
1207dd7cddfSDavid du Colombier if(eout-outp < ndump+1)
1217dd7cddfSDavid du Colombier goto Bfull;
1227dd7cddfSDavid du Colombier *outp++ = ndump-1+128;
1237dd7cddfSDavid du Colombier memmove(outp, dumpbuf, ndump);
1247dd7cddfSDavid du Colombier outp += ndump;
1257dd7cddfSDavid du Colombier ndump = 0;
1267dd7cddfSDavid du Colombier }
1277dd7cddfSDavid du Colombier dumpbuf[ndump++] = *p;
1287dd7cddfSDavid du Colombier runlen = 1;
1297dd7cddfSDavid du Colombier }
1307dd7cddfSDavid du Colombier else{
1317dd7cddfSDavid du Colombier if(ndump != 0){
1327dd7cddfSDavid du Colombier if(eout-outp < ndump+1)
1337dd7cddfSDavid du Colombier goto Bfull;
1347dd7cddfSDavid du Colombier *outp++ = ndump-1+128;
1357dd7cddfSDavid du Colombier memmove(outp, dumpbuf, ndump);
1367dd7cddfSDavid du Colombier outp += ndump;
1377dd7cddfSDavid du Colombier ndump = 0;
1387dd7cddfSDavid du Colombier }
1397dd7cddfSDavid du Colombier offs = p-q-1;
1407dd7cddfSDavid du Colombier if(eout-outp < 2)
1417dd7cddfSDavid du Colombier goto Bfull;
1427dd7cddfSDavid du Colombier *outp++ = ((runlen-NMATCH)<<2) + (offs>>8);
1437dd7cddfSDavid du Colombier *outp++ = offs&255;
1447dd7cddfSDavid du Colombier }
1457dd7cddfSDavid du Colombier for(q = p+runlen; p != q; p++){
1467dd7cddfSDavid du Colombier if(cp->prev)
1477dd7cddfSDavid du Colombier cp->prev->next = 0;
1487dd7cddfSDavid du Colombier cp->next = hash[h].next;
1497dd7cddfSDavid du Colombier cp->prev = &hash[h];
1507dd7cddfSDavid du Colombier if(cp->next)
1517dd7cddfSDavid du Colombier cp->next->prev = cp;
1527dd7cddfSDavid du Colombier cp->prev->next = cp;
1537dd7cddfSDavid du Colombier cp->s = p;
1547dd7cddfSDavid du Colombier if(++cp == &chain[NMEM])
1557dd7cddfSDavid du Colombier cp = chain;
1567dd7cddfSDavid du Colombier if(edata-p > NMATCH)
1577dd7cddfSDavid du Colombier h = hupdate(h, p[NMATCH]);
1587dd7cddfSDavid du Colombier }
1597dd7cddfSDavid du Colombier }
1607dd7cddfSDavid du Colombier if(ndump != 0){
1617dd7cddfSDavid du Colombier if(eout-outp < ndump+1)
1627dd7cddfSDavid du Colombier goto Bfull;
1637dd7cddfSDavid du Colombier *outp++ = ndump-1+128;
1647dd7cddfSDavid du Colombier memmove(outp, dumpbuf, ndump);
1657dd7cddfSDavid du Colombier outp += ndump;
1667dd7cddfSDavid du Colombier }
1677dd7cddfSDavid du Colombier line = eline;
1687dd7cddfSDavid du Colombier loutp = outp;
1697dd7cddfSDavid du Colombier r.max.y++;
1707dd7cddfSDavid du Colombier }
1717dd7cddfSDavid du Colombier Bfull:
1727dd7cddfSDavid du Colombier if(loutp == outbuf)
1737dd7cddfSDavid du Colombier goto ErrOut;
1747dd7cddfSDavid du Colombier n = loutp-outbuf;
1757dd7cddfSDavid du Colombier sprint(hdr, "%11d %11ld ", r.max.y, n);
1767dd7cddfSDavid du Colombier write(fd, hdr, 2*12);
1777dd7cddfSDavid du Colombier write(fd, outbuf, n);
1787dd7cddfSDavid du Colombier r.min.y = r.max.y;
1797dd7cddfSDavid du Colombier }
1807dd7cddfSDavid du Colombier free(data);
1817dd7cddfSDavid du Colombier free(outbuf);
1827dd7cddfSDavid du Colombier free(hash);
1837dd7cddfSDavid du Colombier free(chain);
1847dd7cddfSDavid du Colombier return 0;
1857dd7cddfSDavid du Colombier }
186