ref: b86a12149ade500326a238753c31b6e0178d3b5b
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){ int pfd[2]; Action *ap; Url url; Image *b; int fd, typ; char err[512], buf[80], *s; Pix *p; ap=t->user; 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]", url.reltext); free(t->text); t->text=strdup(err); w->changed=1; close(fd); return; } 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); 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; } 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); } }