ref: 40091208a0cad1d442ecbc8cfdb692fdc8b0fc15
dir: /sys/src/cmd/mothra/getpix.c/
#include <u.h>
#include <libc.h>
#include <draw.h>
#include <event.h>
#include <panel.h>
#include "mothra.h"
typedef struct Pix Pix;
struct Pix{
	Pix *next;
	Image *b;
	int width;
	int height;
	char name[NNAME];
};
char *pixcmd[]={
[GIF]	"gif -9t",
[JPEG]	"jpg -9t",
[PNG]	"png -9t",
[BMP]	"bmp -9t",
[ICO]	"ico -c",
};
void getimage(Rtext *t, Www *w){
	Action *ap;
	Url *url;
	Image *b;
	int fd, typ;
	char err[512], buf[80], *s;
	Pix *p;
	ap=t->user;
	url=emalloc(sizeof(Url));
	seturl(url, ap->image, w->url->fullname);
	for(p=w->pix;p!=nil; p=p->next)
		if(strcmp(ap->image, p->name)==0 && ap->width==p->width && ap->height==p->height){
			t->b = p->b;
			w->changed=1;
			return;
		}
	fd=urlget(url, -1);
	if(fd==-1){
	Err:
		snprint(err, sizeof(err), "[img: %s: %r]", urlstr(url));
		free(t->text);
		t->text=strdup(err);
		w->changed=1;
		close(fd);
		goto Out;
	}
	typ = snooptype(fd);
	if(typ < 0 || typ >= nelem(pixcmd) || pixcmd[typ] == nil){
		werrstr("unknown image type");
		goto Err;
	}
	if((fd = pipeline(fd, "exec %s", pixcmd[typ])) < 0)
		goto Err;
	if(ap->width>0 || ap->height>0){
		s = buf;
		s += sprint(s, "exec resize");
		if(ap->width>0)
			s += sprint(s, " -x %d", ap->width);
		if(ap->height>0)
			s += sprint(s, " -y %d", ap->height);
		USED(s);
		if((fd = pipeline(fd, buf)) < 0)
			goto Err;
	}
	b=readimage(display, fd, 1);
	if(b==0){
		werrstr("can't read image");
		goto Err;
	}
	close(fd);
	p=emalloc(sizeof(Pix));
	nstrcpy(p->name, ap->image, sizeof(p->name));
	p->b=b;
	p->width=ap->width;
	p->height=ap->height;
	p->next=w->pix;
	w->pix=p;
	t->b=b;
	w->changed=1;
Out:
	freeurl(url);
}
void getpix(Rtext *t, Www *w){
	int i, pid, nworker, worker[NXPROC];
	Action *ap;
	nworker = 0;
	for(i=0; i<nelem(worker); i++)
		worker[i] = -1;
	for(;t!=0;t=t->next){
		ap=t->user;
		if(ap && ap->image){
			pid = rfork(RFFDG|RFPROC|RFMEM);
			switch(pid){
			case -1:
				fprint(2, "fork: %r\n");
				break;
			case 0:
				getimage(t, w);
				exits(0);
			default:
				for(i=0; i<nelem(worker); i++)
					if(worker[i] == -1){
						worker[i] = pid;
						nworker++;
						break;
					}
				while(nworker == nelem(worker)){
					if((pid = waitpid()) < 0)
						break;
					for(i=0; i<nelem(worker); i++)
						if(worker[i] == pid){
							worker[i] = -1;
							nworker--;
							break;
						}
				}
			}
			
		}
	}
	while(nworker > 0){
		if((pid = waitpid()) < 0)
			break;
		for(i=0; i<nelem(worker); i++)
			if(worker[i] == pid){
				worker[i] = -1;
				nworker--;
				break;
			}
	}
}
ulong countpix(void *p){
	ulong n=0;
	Pix *x;
	for(x = p; x; x = x->next)
		n += Dy(x->b->r)*bytesperline(x->b->r, x->b->depth);
	return n;
}
void freepix(void *p){
	Pix *x, *xx;
	xx = p;
	while(x = xx){
		xx = x->next;
		freeimage(x->b);
		free(x);
	}
}