xref: /inferno-os/libmemdraw/cload.c (revision 37da2899f40661e3e9631e497da8dc59b971cbd0)
1*37da2899SCharles.Forsyth #include "lib9.h"
2*37da2899SCharles.Forsyth #include "draw.h"
3*37da2899SCharles.Forsyth #include "memdraw.h"
4*37da2899SCharles.Forsyth 
5*37da2899SCharles.Forsyth int
cloadmemimage(Memimage * i,Rectangle r,uchar * data,int ndata)6*37da2899SCharles.Forsyth cloadmemimage(Memimage *i, Rectangle r, uchar *data, int ndata)
7*37da2899SCharles.Forsyth {
8*37da2899SCharles.Forsyth 	int y, bpl, c, cnt, offs;
9*37da2899SCharles.Forsyth 	uchar mem[NMEM], *memp, *omemp, *emem, *linep, *elinep, *u, *eu;
10*37da2899SCharles.Forsyth 
11*37da2899SCharles.Forsyth 	if(!rectinrect(r, i->r))
12*37da2899SCharles.Forsyth 		return -1;
13*37da2899SCharles.Forsyth 	bpl = bytesperline(r, i->depth);
14*37da2899SCharles.Forsyth 	u = data;
15*37da2899SCharles.Forsyth 	eu = data+ndata;
16*37da2899SCharles.Forsyth 	memp = mem;
17*37da2899SCharles.Forsyth 	emem = mem+NMEM;
18*37da2899SCharles.Forsyth 	y = r.min.y;
19*37da2899SCharles.Forsyth 	linep = byteaddr(i, Pt(r.min.x, y));
20*37da2899SCharles.Forsyth 	elinep = linep+bpl;
21*37da2899SCharles.Forsyth 	for(;;){
22*37da2899SCharles.Forsyth 		if(linep == elinep){
23*37da2899SCharles.Forsyth 			if(++y == r.max.y)
24*37da2899SCharles.Forsyth 				break;
25*37da2899SCharles.Forsyth 			linep = byteaddr(i, Pt(r.min.x, y));
26*37da2899SCharles.Forsyth 			elinep = linep+bpl;
27*37da2899SCharles.Forsyth 		}
28*37da2899SCharles.Forsyth 		if(u == eu){	/* buffer too small */
29*37da2899SCharles.Forsyth 			return -1;
30*37da2899SCharles.Forsyth 		}
31*37da2899SCharles.Forsyth 		c = *u++;
32*37da2899SCharles.Forsyth 		if(c >= 128){
33*37da2899SCharles.Forsyth 			for(cnt=c-128+1; cnt!=0 ;--cnt){
34*37da2899SCharles.Forsyth 				if(u == eu){		/* buffer too small */
35*37da2899SCharles.Forsyth 					return -1;
36*37da2899SCharles.Forsyth 				}
37*37da2899SCharles.Forsyth 				if(linep == elinep){	/* phase error */
38*37da2899SCharles.Forsyth 					return -1;
39*37da2899SCharles.Forsyth 				}
40*37da2899SCharles.Forsyth 				*linep++ = *u;
41*37da2899SCharles.Forsyth 				*memp++ = *u++;
42*37da2899SCharles.Forsyth 				if(memp == emem)
43*37da2899SCharles.Forsyth 					memp = mem;
44*37da2899SCharles.Forsyth 			}
45*37da2899SCharles.Forsyth 		}
46*37da2899SCharles.Forsyth 		else{
47*37da2899SCharles.Forsyth 			if(u == eu)	/* short buffer */
48*37da2899SCharles.Forsyth 				return -1;
49*37da2899SCharles.Forsyth 			offs = *u++ + ((c&3)<<8)+1;
50*37da2899SCharles.Forsyth 			if(memp-mem < offs)
51*37da2899SCharles.Forsyth 				omemp = memp+(NMEM-offs);
52*37da2899SCharles.Forsyth 			else
53*37da2899SCharles.Forsyth 				omemp = memp-offs;
54*37da2899SCharles.Forsyth 			for(cnt=(c>>2)+NMATCH; cnt!=0; --cnt){
55*37da2899SCharles.Forsyth 				if(linep == elinep)	/* phase error */
56*37da2899SCharles.Forsyth 					return -1;
57*37da2899SCharles.Forsyth 				*linep++ = *omemp;
58*37da2899SCharles.Forsyth 				*memp++ = *omemp++;
59*37da2899SCharles.Forsyth 				if(omemp == emem)
60*37da2899SCharles.Forsyth 					omemp = mem;
61*37da2899SCharles.Forsyth 				if(memp == emem)
62*37da2899SCharles.Forsyth 					memp = mem;
63*37da2899SCharles.Forsyth 			}
64*37da2899SCharles.Forsyth 		}
65*37da2899SCharles.Forsyth 	}
66*37da2899SCharles.Forsyth 	return u-data;
67*37da2899SCharles.Forsyth }
68