xref: /plan9-contrib/sys/src/cmd/jpg/writerawimage.c (revision 9a747e4fd48b9f4522c70c07e8f882a15030f964)
17dd7cddfSDavid du Colombier #include <u.h>
27dd7cddfSDavid du Colombier #include <libc.h>
37dd7cddfSDavid du Colombier #include <bio.h>
47dd7cddfSDavid du Colombier #include <draw.h>
57dd7cddfSDavid du Colombier #include "imagefile.h"
67dd7cddfSDavid du Colombier 
77dd7cddfSDavid du Colombier /*
87dd7cddfSDavid du Colombier  * Hacked version for writing from Rawimage to file.
959cc4ca5SDavid du Colombier  * Assumes 8 bits per component.
107dd7cddfSDavid du Colombier  */
117dd7cddfSDavid du Colombier 
127dd7cddfSDavid du Colombier #define	HSHIFT	3	/* HSHIFT==5 runs slightly faster, but hash table is 64x bigger */
137dd7cddfSDavid du Colombier #define	NHASH	(1<<(HSHIFT*NMATCH))
147dd7cddfSDavid du Colombier #define	HMASK	(NHASH-1)
157dd7cddfSDavid du Colombier #define	hupdate(h, c)	((((h)<<HSHIFT)^(c))&HMASK)
167dd7cddfSDavid du Colombier typedef struct Hlist Hlist;
177dd7cddfSDavid du Colombier struct Hlist{
187dd7cddfSDavid du Colombier 	uchar *s;
197dd7cddfSDavid du Colombier 	Hlist *next, *prev;
207dd7cddfSDavid du Colombier };
217dd7cddfSDavid du Colombier 
227dd7cddfSDavid du Colombier int
writerawimage(int fd,Rawimage * i)237dd7cddfSDavid du Colombier writerawimage(int fd, Rawimage *i)
247dd7cddfSDavid du Colombier {
257dd7cddfSDavid du Colombier 	uchar *outbuf, *outp, *eout;		/* encoded data, pointer, end */
267dd7cddfSDavid du Colombier 	uchar *loutp;				/* start of encoded line */
277dd7cddfSDavid du Colombier 	Hlist *hash;				/* heads of hash chains of past strings */
287dd7cddfSDavid du Colombier 	Hlist *chain, *hp;			/* hash chain members, pointer */
297dd7cddfSDavid du Colombier 	Hlist *cp;				/* next Hlist to fall out of window */
307dd7cddfSDavid du Colombier 	int h;					/* hash value */
317dd7cddfSDavid du Colombier 	uchar *line, *eline;			/* input line, end pointer */
327dd7cddfSDavid du Colombier 	uchar *data, *edata;			/* input buffer, end pointer */
337dd7cddfSDavid du Colombier 	ulong n;				/* length of input buffer */
347dd7cddfSDavid du Colombier 	int bpl;				/* input line length */
357dd7cddfSDavid du Colombier 	int offs, runlen;			/* offset, length of consumed data */
367dd7cddfSDavid du Colombier 	uchar dumpbuf[NDUMP];			/* dump accumulator */
377dd7cddfSDavid du Colombier 	int ndump;				/* length of dump accumulator */
38*9a747e4fSDavid du Colombier 	int ncblock;				/* size of buffer */
397dd7cddfSDavid du Colombier 	Rectangle r;
407dd7cddfSDavid du Colombier 	uchar *p, *q, *s, *es, *t;
417dd7cddfSDavid du Colombier 	char hdr[11+5*12+1], buf[16];
427dd7cddfSDavid du Colombier 	ulong desc;
437dd7cddfSDavid du Colombier 
447dd7cddfSDavid du Colombier 	r = i->r;
457dd7cddfSDavid du Colombier 	switch(i->chandesc){
467dd7cddfSDavid du Colombier 	default:
477dd7cddfSDavid du Colombier 		werrstr("can't handle chandesc %d", i->chandesc);
48*9a747e4fSDavid du Colombier 		return -1;
497dd7cddfSDavid du Colombier 	case CY:
507dd7cddfSDavid du Colombier 		bpl = Dx(r);
517dd7cddfSDavid du Colombier 		desc = GREY8;
527dd7cddfSDavid du Colombier 		break;
5359cc4ca5SDavid du Colombier 	case CYA16:
5459cc4ca5SDavid du Colombier 		bpl = 2*Dx(r);
5559cc4ca5SDavid du Colombier 		desc = CHAN2(CGrey, 8, CAlpha, 8);
5659cc4ca5SDavid du Colombier 		break;
577dd7cddfSDavid du Colombier 	case CRGBV:
587dd7cddfSDavid du Colombier 		bpl = Dx(r);
597dd7cddfSDavid du Colombier 		desc = CMAP8;
607dd7cddfSDavid du Colombier 		break;
6159cc4ca5SDavid du Colombier 	case CRGBVA16:
6259cc4ca5SDavid du Colombier 		bpl = 2*Dx(r);
6359cc4ca5SDavid du Colombier 		desc = CHAN2(CMap, 8, CAlpha, 8);
6459cc4ca5SDavid du Colombier 		break;
657dd7cddfSDavid du Colombier 	case CRGB24:
667dd7cddfSDavid du Colombier 		bpl = 3*Dx(r);
677dd7cddfSDavid du Colombier 		desc = RGB24;
687dd7cddfSDavid du Colombier 		break;
6959cc4ca5SDavid du Colombier 	case CRGBA32:
7059cc4ca5SDavid du Colombier 		bpl = 4*Dx(r);
7159cc4ca5SDavid du Colombier 		desc = RGBA32;
7259cc4ca5SDavid du Colombier 		break;
737dd7cddfSDavid du Colombier 	}
74*9a747e4fSDavid du Colombier 	ncblock = _compblocksize(r, bpl/Dx(r));
75*9a747e4fSDavid du Colombier 	outbuf = malloc(ncblock);
76*9a747e4fSDavid du Colombier 	hash = malloc(NHASH*sizeof(Hlist));
77*9a747e4fSDavid du Colombier 	chain = malloc(NMEM*sizeof(Hlist));
78*9a747e4fSDavid du Colombier 	if(outbuf == 0 || hash == 0 || chain == 0){
79*9a747e4fSDavid du Colombier 	ErrOut:
80*9a747e4fSDavid du Colombier 		free(outbuf);
81*9a747e4fSDavid du Colombier 		free(hash);
82*9a747e4fSDavid du Colombier 		free(chain);
83*9a747e4fSDavid du Colombier 		return -1;
84*9a747e4fSDavid du Colombier 	}
857dd7cddfSDavid du Colombier 	n = Dy(r)*bpl;
867dd7cddfSDavid du Colombier 	data = i->chans[0];
877dd7cddfSDavid du Colombier 	sprint(hdr, "compressed\n%11s %11d %11d %11d %11d ",
887dd7cddfSDavid du Colombier 		chantostr(buf, desc), r.min.x, r.min.y, r.max.x, r.max.y);
89*9a747e4fSDavid du Colombier 	if(write(fd, hdr, 11+5*12) != 11+5*12){
90*9a747e4fSDavid du Colombier 		werrstr("i/o error writing header");
917dd7cddfSDavid du Colombier 		goto ErrOut;
92*9a747e4fSDavid du Colombier 	}
937dd7cddfSDavid du Colombier 	edata = data+n;
94*9a747e4fSDavid du Colombier 	eout = outbuf+ncblock;
957dd7cddfSDavid du Colombier 	line = data;
967dd7cddfSDavid du Colombier 	r.max.y = r.min.y;
977dd7cddfSDavid du Colombier 	while(line != edata){
987dd7cddfSDavid du Colombier 		memset(hash, 0, NHASH*sizeof(Hlist));
997dd7cddfSDavid du Colombier 		memset(chain, 0, NMEM*sizeof(Hlist));
1007dd7cddfSDavid du Colombier 		cp = chain;
1017dd7cddfSDavid du Colombier 		h = 0;
1027dd7cddfSDavid du Colombier 		outp = outbuf;
1037dd7cddfSDavid du Colombier 		for(n = 0; n != NMATCH; n++)
1047dd7cddfSDavid du Colombier 			h = hupdate(h, line[n]);
1057dd7cddfSDavid du Colombier 		loutp = outbuf;
1067dd7cddfSDavid du Colombier 		while(line != edata){
1077dd7cddfSDavid du Colombier 			ndump = 0;
1087dd7cddfSDavid du Colombier 			eline = line+bpl;
1097dd7cddfSDavid du Colombier 			for(p = line; p != eline; ){
1107dd7cddfSDavid du Colombier 				if(eline-p < NRUN)
1117dd7cddfSDavid du Colombier 					es = eline;
1127dd7cddfSDavid du Colombier 				else
1137dd7cddfSDavid du Colombier 					es = p+NRUN;
1147dd7cddfSDavid du Colombier 				q = 0;
1157dd7cddfSDavid du Colombier 				runlen = 0;
1167dd7cddfSDavid du Colombier 				for(hp = hash[h].next; hp; hp = hp->next){
1177dd7cddfSDavid du Colombier 					s = p + runlen;
1187dd7cddfSDavid du Colombier 					if(s >= es)
1197dd7cddfSDavid du Colombier 						continue;
1207dd7cddfSDavid du Colombier 					t = hp->s + runlen;
1217dd7cddfSDavid du Colombier 					for(; s >= p; s--)
1227dd7cddfSDavid du Colombier 						if(*s != *t--)
1237dd7cddfSDavid du Colombier 							goto matchloop;
1247dd7cddfSDavid du Colombier 					t += runlen+2;
1257dd7cddfSDavid du Colombier 					s += runlen+2;
1267dd7cddfSDavid du Colombier 					for(; s < es; s++)
1277dd7cddfSDavid du Colombier 						if(*s != *t++)
1287dd7cddfSDavid du Colombier 							break;
1297dd7cddfSDavid du Colombier 					n = s-p;
1307dd7cddfSDavid du Colombier 					if(n > runlen){
1317dd7cddfSDavid du Colombier 						runlen = n;
1327dd7cddfSDavid du Colombier 						q = hp->s;
1337dd7cddfSDavid du Colombier 						if(n == NRUN)
1347dd7cddfSDavid du Colombier 							break;
1357dd7cddfSDavid du Colombier 					}
1367dd7cddfSDavid du Colombier 			matchloop: ;
1377dd7cddfSDavid du Colombier 				}
1387dd7cddfSDavid du Colombier 				if(runlen < NMATCH){
1397dd7cddfSDavid du Colombier 					if(ndump == NDUMP){
1407dd7cddfSDavid du Colombier 						if(eout-outp < ndump+1)
1417dd7cddfSDavid du Colombier 							goto Bfull;
1427dd7cddfSDavid du Colombier 						*outp++ = ndump-1+128;
1437dd7cddfSDavid du Colombier 						memmove(outp, dumpbuf, ndump);
1447dd7cddfSDavid du Colombier 						outp += ndump;
1457dd7cddfSDavid du Colombier 						ndump = 0;
1467dd7cddfSDavid du Colombier 					}
1477dd7cddfSDavid du Colombier 					dumpbuf[ndump++] = *p;
1487dd7cddfSDavid du Colombier 					runlen = 1;
1497dd7cddfSDavid du Colombier 				}
1507dd7cddfSDavid du Colombier 				else{
1517dd7cddfSDavid du Colombier 					if(ndump != 0){
1527dd7cddfSDavid du Colombier 						if(eout-outp < ndump+1)
1537dd7cddfSDavid du Colombier 							goto Bfull;
1547dd7cddfSDavid du Colombier 						*outp++ = ndump-1+128;
1557dd7cddfSDavid du Colombier 						memmove(outp, dumpbuf, ndump);
1567dd7cddfSDavid du Colombier 						outp += ndump;
1577dd7cddfSDavid du Colombier 						ndump = 0;
1587dd7cddfSDavid du Colombier 					}
1597dd7cddfSDavid du Colombier 					offs = p-q-1;
1607dd7cddfSDavid du Colombier 					if(eout-outp < 2)
1617dd7cddfSDavid du Colombier 						goto Bfull;
1627dd7cddfSDavid du Colombier 					*outp++ = ((runlen-NMATCH)<<2) + (offs>>8);
1637dd7cddfSDavid du Colombier 					*outp++ = offs&255;
1647dd7cddfSDavid du Colombier 				}
1657dd7cddfSDavid du Colombier 				for(q = p+runlen; p != q; p++){
1667dd7cddfSDavid du Colombier 					if(cp->prev)
1677dd7cddfSDavid du Colombier 						cp->prev->next = 0;
1687dd7cddfSDavid du Colombier 					cp->next = hash[h].next;
1697dd7cddfSDavid du Colombier 					cp->prev = &hash[h];
1707dd7cddfSDavid du Colombier 					if(cp->next)
1717dd7cddfSDavid du Colombier 						cp->next->prev = cp;
1727dd7cddfSDavid du Colombier 					cp->prev->next = cp;
1737dd7cddfSDavid du Colombier 					cp->s = p;
1747dd7cddfSDavid du Colombier 					if(++cp == &chain[NMEM])
1757dd7cddfSDavid du Colombier 						cp = chain;
1767dd7cddfSDavid du Colombier 					if(edata-p > NMATCH)
1777dd7cddfSDavid du Colombier 						h = hupdate(h, p[NMATCH]);
1787dd7cddfSDavid du Colombier 				}
1797dd7cddfSDavid du Colombier 			}
1807dd7cddfSDavid du Colombier 			if(ndump != 0){
1817dd7cddfSDavid du Colombier 				if(eout-outp < ndump+1)
1827dd7cddfSDavid du Colombier 					goto Bfull;
1837dd7cddfSDavid du Colombier 				*outp++ = ndump-1+128;
1847dd7cddfSDavid du Colombier 				memmove(outp, dumpbuf, ndump);
1857dd7cddfSDavid du Colombier 				outp += ndump;
1867dd7cddfSDavid du Colombier 			}
1877dd7cddfSDavid du Colombier 			line = eline;
1887dd7cddfSDavid du Colombier 			loutp = outp;
1897dd7cddfSDavid du Colombier 			r.max.y++;
1907dd7cddfSDavid du Colombier 		}
1917dd7cddfSDavid du Colombier 	Bfull:
192*9a747e4fSDavid du Colombier 		if(loutp == outbuf){
193*9a747e4fSDavid du Colombier 			werrstr("compressor out of sync");
1947dd7cddfSDavid du Colombier 			goto ErrOut;
195*9a747e4fSDavid du Colombier 		}
1967dd7cddfSDavid du Colombier 		n = loutp-outbuf;
1977dd7cddfSDavid du Colombier 		sprint(hdr, "%11d %11ld ", r.max.y, n);
1987dd7cddfSDavid du Colombier 		write(fd, hdr, 2*12);
1997dd7cddfSDavid du Colombier 		write(fd, outbuf, n);
2007dd7cddfSDavid du Colombier 		r.min.y = r.max.y;
2017dd7cddfSDavid du Colombier 	}
2027dd7cddfSDavid du Colombier 	free(outbuf);
2037dd7cddfSDavid du Colombier 	free(hash);
2047dd7cddfSDavid du Colombier 	free(chain);
2057dd7cddfSDavid du Colombier 	return 0;
2067dd7cddfSDavid du Colombier }
207