xref: /plan9/sys/src/cmd/acme/acme.c (revision b8661318e8f82fa2a6c406ca28596a6aef982bd2)
17dd7cddfSDavid du Colombier #include <u.h>
27dd7cddfSDavid du Colombier #include <libc.h>
37dd7cddfSDavid du Colombier #include <draw.h>
47dd7cddfSDavid du Colombier #include <thread.h>
57dd7cddfSDavid du Colombier #include <cursor.h>
67dd7cddfSDavid du Colombier #include <mouse.h>
77dd7cddfSDavid du Colombier #include <keyboard.h>
87dd7cddfSDavid du Colombier #include <frame.h>
97dd7cddfSDavid du Colombier #include <fcall.h>
107dd7cddfSDavid du Colombier #include <plumb.h>
117dd7cddfSDavid du Colombier #include "dat.h"
127dd7cddfSDavid du Colombier #include "fns.h"
1359cc4ca5SDavid du Colombier 	/* for generating syms in mkfile only: */
1459cc4ca5SDavid du Colombier 	#include <bio.h>
1559cc4ca5SDavid du Colombier 	#include "edit.h"
167dd7cddfSDavid du Colombier 
177dd7cddfSDavid du Colombier void	mousethread(void*);
187dd7cddfSDavid du Colombier void	keyboardthread(void*);
197dd7cddfSDavid du Colombier void	waitthread(void*);
207dd7cddfSDavid du Colombier void	xfidallocthread(void*);
219a747e4fSDavid du Colombier void	newwindowthread(void*);
227dd7cddfSDavid du Colombier void plumbproc(void*);
237dd7cddfSDavid du Colombier 
247dd7cddfSDavid du Colombier Reffont	**fontcache;
257dd7cddfSDavid du Colombier int		nfontcache;
267dd7cddfSDavid du Colombier char		wdir[512] = ".";
277dd7cddfSDavid du Colombier Reffont	*reffonts[2];
287dd7cddfSDavid du Colombier int		snarffd = -1;
297dd7cddfSDavid du Colombier int		mainpid;
307dd7cddfSDavid du Colombier int		plumbsendfd;
317dd7cddfSDavid du Colombier int		plumbeditfd;
327dd7cddfSDavid du Colombier 
337dd7cddfSDavid du Colombier enum{
347dd7cddfSDavid du Colombier 	NSnarf = 1000	/* less than 1024, I/O buffer size */
357dd7cddfSDavid du Colombier };
367dd7cddfSDavid du Colombier Rune	snarfrune[NSnarf+1];
377dd7cddfSDavid du Colombier 
387dd7cddfSDavid du Colombier char		*fontnames[2] =
397dd7cddfSDavid du Colombier {
407dd7cddfSDavid du Colombier 	"/lib/font/bit/lucidasans/euro.8.font",
417dd7cddfSDavid du Colombier 	"/lib/font/bit/lucm/unicode.9.font",
427dd7cddfSDavid du Colombier };
437dd7cddfSDavid du Colombier 
447dd7cddfSDavid du Colombier Command *command;
457dd7cddfSDavid du Colombier 
467dd7cddfSDavid du Colombier void	acmeerrorinit(void);
477dd7cddfSDavid du Colombier void	readfile(Column*, char*);
487dd7cddfSDavid du Colombier int	shutdown(void*, char*);
497dd7cddfSDavid du Colombier 
507dd7cddfSDavid du Colombier void
517dd7cddfSDavid du Colombier derror(Display*, char *errorstr)
527dd7cddfSDavid du Colombier {
537dd7cddfSDavid du Colombier 	error(errorstr);
547dd7cddfSDavid du Colombier }
557dd7cddfSDavid du Colombier 
567dd7cddfSDavid du Colombier void
577dd7cddfSDavid du Colombier threadmain(int argc, char *argv[])
587dd7cddfSDavid du Colombier {
597dd7cddfSDavid du Colombier 	int i;
607dd7cddfSDavid du Colombier 	char *p, *loadfile;
617dd7cddfSDavid du Colombier 	char buf[256];
627dd7cddfSDavid du Colombier 	Column *c;
637dd7cddfSDavid du Colombier 	int ncol;
647dd7cddfSDavid du Colombier 	Display *d;
657dd7cddfSDavid du Colombier 	static void *arg[1];
667dd7cddfSDavid du Colombier 
6780ee5cbfSDavid du Colombier 	rfork(RFENVG|RFNAMEG);
6880ee5cbfSDavid du Colombier 
697dd7cddfSDavid du Colombier 	ncol = -1;
707dd7cddfSDavid du Colombier 
717dd7cddfSDavid du Colombier 	loadfile = nil;
727dd7cddfSDavid du Colombier 	ARGBEGIN{
734fec87e5SDavid du Colombier 	case 'a':
744fec87e5SDavid du Colombier 		globalautoindent = TRUE;
754fec87e5SDavid du Colombier 		break;
767dd7cddfSDavid du Colombier 	case 'b':
777dd7cddfSDavid du Colombier 		bartflag = TRUE;
787dd7cddfSDavid du Colombier 		break;
797dd7cddfSDavid du Colombier 	case 'c':
807dd7cddfSDavid du Colombier 		p = ARGF();
817dd7cddfSDavid du Colombier 		if(p == nil)
827dd7cddfSDavid du Colombier 			goto Usage;
837dd7cddfSDavid du Colombier 		ncol = atoi(p);
847dd7cddfSDavid du Colombier 		if(ncol <= 0)
857dd7cddfSDavid du Colombier 			goto Usage;
867dd7cddfSDavid du Colombier 		break;
877dd7cddfSDavid du Colombier 	case 'f':
887dd7cddfSDavid du Colombier 		fontnames[0] = ARGF();
897dd7cddfSDavid du Colombier 		if(fontnames[0] == nil)
907dd7cddfSDavid du Colombier 			goto Usage;
917dd7cddfSDavid du Colombier 		break;
927dd7cddfSDavid du Colombier 	case 'F':
937dd7cddfSDavid du Colombier 		fontnames[1] = ARGF();
947dd7cddfSDavid du Colombier 		if(fontnames[1] == nil)
957dd7cddfSDavid du Colombier 			goto Usage;
967dd7cddfSDavid du Colombier 		break;
977dd7cddfSDavid du Colombier 	case 'l':
987dd7cddfSDavid du Colombier 		loadfile = ARGF();
997dd7cddfSDavid du Colombier 		if(loadfile == nil)
1007dd7cddfSDavid du Colombier 			goto Usage;
1017dd7cddfSDavid du Colombier 		break;
1027dd7cddfSDavid du Colombier 	default:
1037dd7cddfSDavid du Colombier 	Usage:
1044fec87e5SDavid du Colombier 		fprint(2, "usage: acme -a -c ncol -f fontname -F fixedwidthfontname -l loadfile\n");
1057dd7cddfSDavid du Colombier 		exits("usage");
1067dd7cddfSDavid du Colombier 	}ARGEND
1077dd7cddfSDavid du Colombier 
108*b8661318SDavid du Colombier 	quotefmtinstall();
1097dd7cddfSDavid du Colombier 	cputype = getenv("cputype");
1107dd7cddfSDavid du Colombier 	objtype = getenv("objtype");
1117dd7cddfSDavid du Colombier 	home = getenv("home");
1127dd7cddfSDavid du Colombier 	p = getenv("tabstop");
11359cc4ca5SDavid du Colombier 	if(p != nil){
1147dd7cddfSDavid du Colombier 		maxtab = strtoul(p, nil, 0);
11559cc4ca5SDavid du Colombier 		free(p);
11659cc4ca5SDavid du Colombier 	}
1177dd7cddfSDavid du Colombier 	if(maxtab == 0)
1187dd7cddfSDavid du Colombier 		maxtab = 4;
1199a747e4fSDavid du Colombier 	if(loadfile)
1209a747e4fSDavid du Colombier 		rowloadfonts(loadfile);
1217dd7cddfSDavid du Colombier 	putenv("font", fontnames[0]);
1227dd7cddfSDavid du Colombier 	snarffd = open("/dev/snarf", OREAD|OCEXEC);
1237dd7cddfSDavid du Colombier 	if(cputype){
1247dd7cddfSDavid du Colombier 		sprint(buf, "/acme/bin/%s", cputype);
1257dd7cddfSDavid du Colombier 		bind(buf, "/bin", MBEFORE);
1267dd7cddfSDavid du Colombier 	}
1277dd7cddfSDavid du Colombier 	bind("/acme/bin", "/bin", MBEFORE);
1287dd7cddfSDavid du Colombier 	getwd(wdir, sizeof wdir);
1297dd7cddfSDavid du Colombier 
1309a747e4fSDavid du Colombier 	if(geninitdraw(nil, derror, fontnames[0], "acme", nil, Refnone) < 0){
1317dd7cddfSDavid du Colombier 		fprint(2, "acme: can't open display: %r\n");
1329a747e4fSDavid du Colombier 		exits("geninitdraw");
1337dd7cddfSDavid du Colombier 	}
1347dd7cddfSDavid du Colombier 	d = display;
1357dd7cddfSDavid du Colombier 	font = d->defaultfont;
1367dd7cddfSDavid du Colombier 
1377dd7cddfSDavid du Colombier 	reffont.f = font;
1387dd7cddfSDavid du Colombier 	reffonts[0] = &reffont;
1397dd7cddfSDavid du Colombier 	incref(&reffont);	/* one to hold up 'font' variable */
1407dd7cddfSDavid du Colombier 	incref(&reffont);	/* one to hold up reffonts[0] */
1417dd7cddfSDavid du Colombier 	fontcache = emalloc(sizeof(Reffont*));
1427dd7cddfSDavid du Colombier 	nfontcache = 1;
1437dd7cddfSDavid du Colombier 	fontcache[0] = &reffont;
1447dd7cddfSDavid du Colombier 
1457dd7cddfSDavid du Colombier 	iconinit();
1467dd7cddfSDavid du Colombier 	timerinit();
1477dd7cddfSDavid du Colombier 	rxinit();
1487dd7cddfSDavid du Colombier 
1497dd7cddfSDavid du Colombier 	cwait = threadwaitchan();
1507dd7cddfSDavid du Colombier 	ccommand = chancreate(sizeof(Command**), 0);
1517dd7cddfSDavid du Colombier 	ckill = chancreate(sizeof(Rune*), 0);
1527dd7cddfSDavid du Colombier 	cxfidalloc = chancreate(sizeof(Xfid*), 0);
1537dd7cddfSDavid du Colombier 	cxfidfree = chancreate(sizeof(Xfid*), 0);
1549a747e4fSDavid du Colombier 	cnewwindow = chancreate(sizeof(Channel*), 0);
1557dd7cddfSDavid du Colombier 	cerr = chancreate(sizeof(char*), 0);
15659cc4ca5SDavid du Colombier 	cedit = chancreate(sizeof(int), 0);
1577dd7cddfSDavid du Colombier 	cexit = chancreate(sizeof(int), 0);
1585316891fSDavid du Colombier 	cwarn = chancreate(sizeof(void*), 1);
1595316891fSDavid du Colombier 	if(cwait==nil || ccommand==nil || ckill==nil || cxfidalloc==nil || cxfidfree==nil || cerr==nil || cexit==nil || cwarn==nil){
1607dd7cddfSDavid du Colombier 		fprint(2, "acme: can't create initial channels: %r\n");
1617dd7cddfSDavid du Colombier 		exits("channels");
1627dd7cddfSDavid du Colombier 	}
1637dd7cddfSDavid du Colombier 
1647dd7cddfSDavid du Colombier 	mousectl = initmouse(nil, screen);
1657dd7cddfSDavid du Colombier 	if(mousectl == nil){
1667dd7cddfSDavid du Colombier 		fprint(2, "acme: can't initialize mouse: %r\n");
1677dd7cddfSDavid du Colombier 		exits("mouse");
1687dd7cddfSDavid du Colombier 	}
1697dd7cddfSDavid du Colombier 	mouse = mousectl;
1707dd7cddfSDavid du Colombier 	keyboardctl = initkeyboard(nil);
1717dd7cddfSDavid du Colombier 	if(keyboardctl == nil){
1727dd7cddfSDavid du Colombier 		fprint(2, "acme: can't initialize keyboard: %r\n");
1737dd7cddfSDavid du Colombier 		exits("keyboard");
1747dd7cddfSDavid du Colombier 	}
1757dd7cddfSDavid du Colombier 	mainpid = getpid();
1767dd7cddfSDavid du Colombier 	plumbeditfd = plumbopen("edit", OREAD|OCEXEC);
1777dd7cddfSDavid du Colombier 	if(plumbeditfd >= 0){
1787dd7cddfSDavid du Colombier 		cplumb = chancreate(sizeof(Plumbmsg*), 0);
1797dd7cddfSDavid du Colombier 		proccreate(plumbproc, nil, STACK);
1807dd7cddfSDavid du Colombier 	}
1817dd7cddfSDavid du Colombier 	plumbsendfd = plumbopen("send", OWRITE|OCEXEC);
1827dd7cddfSDavid du Colombier 
1837dd7cddfSDavid du Colombier 	fsysinit();
1847dd7cddfSDavid du Colombier 
1857dd7cddfSDavid du Colombier 	#define	WPERCOL	8
1867dd7cddfSDavid du Colombier 	disk = diskinit();
1877dd7cddfSDavid du Colombier 	if(loadfile)
1887dd7cddfSDavid du Colombier 		rowload(&row, loadfile, TRUE);
1897dd7cddfSDavid du Colombier 	else{
1907dd7cddfSDavid du Colombier 		rowinit(&row, screen->clipr);
1917dd7cddfSDavid du Colombier 		if(ncol < 0){
1927dd7cddfSDavid du Colombier 			if(argc == 0)
1937dd7cddfSDavid du Colombier 				ncol = 2;
1947dd7cddfSDavid du Colombier 			else{
1957dd7cddfSDavid du Colombier 				ncol = (argc+(WPERCOL-1))/WPERCOL;
1967dd7cddfSDavid du Colombier 				if(ncol < 2)
1977dd7cddfSDavid du Colombier 					ncol = 2;
1987dd7cddfSDavid du Colombier 			}
1997dd7cddfSDavid du Colombier 		}
2007dd7cddfSDavid du Colombier 		if(ncol == 0)
2017dd7cddfSDavid du Colombier 			ncol = 2;
2027dd7cddfSDavid du Colombier 		for(i=0; i<ncol; i++){
2037dd7cddfSDavid du Colombier 			c = rowadd(&row, nil, -1);
2047dd7cddfSDavid du Colombier 			if(c==nil && i==0)
2057dd7cddfSDavid du Colombier 				error("initializing columns");
2067dd7cddfSDavid du Colombier 		}
2077dd7cddfSDavid du Colombier 		c = row.col[row.ncol-1];
2087dd7cddfSDavid du Colombier 		if(argc == 0)
2097dd7cddfSDavid du Colombier 			readfile(c, wdir);
2107dd7cddfSDavid du Colombier 		else
2117dd7cddfSDavid du Colombier 			for(i=0; i<argc; i++){
2127dd7cddfSDavid du Colombier 				p = utfrrune(argv[i], '/');
2137dd7cddfSDavid du Colombier 				if((p!=nil && strcmp(p, "/guide")==0) || i/WPERCOL>=row.ncol)
2147dd7cddfSDavid du Colombier 					readfile(c, argv[i]);
2157dd7cddfSDavid du Colombier 				else
2167dd7cddfSDavid du Colombier 					readfile(row.col[i/WPERCOL], argv[i]);
2177dd7cddfSDavid du Colombier 			}
2187dd7cddfSDavid du Colombier 	}
2197dd7cddfSDavid du Colombier 	flushimage(display, 1);
2207dd7cddfSDavid du Colombier 
2217dd7cddfSDavid du Colombier 	acmeerrorinit();
2227dd7cddfSDavid du Colombier 	threadcreate(keyboardthread, nil, STACK);
2237dd7cddfSDavid du Colombier 	threadcreate(mousethread, nil, STACK);
2247dd7cddfSDavid du Colombier 	threadcreate(waitthread, nil, STACK);
2257dd7cddfSDavid du Colombier 	threadcreate(xfidallocthread, nil, STACK);
2269a747e4fSDavid du Colombier 	threadcreate(newwindowthread, nil, STACK);
2277dd7cddfSDavid du Colombier 
2289a747e4fSDavid du Colombier 	threadnotify(shutdown, 1);
2297dd7cddfSDavid du Colombier 	recvul(cexit);
2307dd7cddfSDavid du Colombier 	killprocs();
2317dd7cddfSDavid du Colombier 	threadexitsall(nil);
2327dd7cddfSDavid du Colombier }
2337dd7cddfSDavid du Colombier 
2347dd7cddfSDavid du Colombier void
2357dd7cddfSDavid du Colombier readfile(Column *c, char *s)
2367dd7cddfSDavid du Colombier {
2377dd7cddfSDavid du Colombier 	Window *w;
2387dd7cddfSDavid du Colombier 	Rune rb[256];
2397dd7cddfSDavid du Colombier 	int nb, nr;
2407dd7cddfSDavid du Colombier 	Runestr rs;
2417dd7cddfSDavid du Colombier 
2427dd7cddfSDavid du Colombier 	w = coladd(c, nil, nil, -1);
2437dd7cddfSDavid du Colombier 	cvttorunes(s, strlen(s), rb, &nb, &nr, nil);
2447dd7cddfSDavid du Colombier 	rs = cleanrname((Runestr){rb, nr});
2457dd7cddfSDavid du Colombier 	winsetname(w, rs.r, rs.nr);
2467dd7cddfSDavid du Colombier 	textload(&w->body, 0, s, 1);
2477dd7cddfSDavid du Colombier 	w->body.file->mod = FALSE;
2487dd7cddfSDavid du Colombier 	w->dirty = FALSE;
2497dd7cddfSDavid du Colombier 	winsettag(w);
2507dd7cddfSDavid du Colombier 	textscrdraw(&w->body);
2517dd7cddfSDavid du Colombier 	textsetselect(&w->tag, w->tag.file->nc, w->tag.file->nc);
2527dd7cddfSDavid du Colombier }
2537dd7cddfSDavid du Colombier 
2547dd7cddfSDavid du Colombier char *oknotes[] ={
2557dd7cddfSDavid du Colombier 	"delete",
2567dd7cddfSDavid du Colombier 	"hangup",
2577dd7cddfSDavid du Colombier 	"kill",
2587dd7cddfSDavid du Colombier 	"exit",
2597dd7cddfSDavid du Colombier 	nil
2607dd7cddfSDavid du Colombier };
2617dd7cddfSDavid du Colombier 
2627dd7cddfSDavid du Colombier int	dumping;
2637dd7cddfSDavid du Colombier 
2649a747e4fSDavid du Colombier int
2659a747e4fSDavid du Colombier shutdown(void*, char *msg)
2667dd7cddfSDavid du Colombier {
2677dd7cddfSDavid du Colombier 	int i;
2687dd7cddfSDavid du Colombier 
2697dd7cddfSDavid du Colombier 	killprocs();
2709a747e4fSDavid du Colombier 	if(!dumping && strcmp(msg, "kill")!=0 && strcmp(msg, "exit")!=0 && getpid()==mainpid){
2717dd7cddfSDavid du Colombier 		dumping = TRUE;
2727dd7cddfSDavid du Colombier 		rowdump(&row, nil);
2737dd7cddfSDavid du Colombier 	}
2747dd7cddfSDavid du Colombier 	for(i=0; oknotes[i]; i++)
2757dd7cddfSDavid du Colombier 		if(strncmp(oknotes[i], msg, strlen(oknotes[i])) == 0)
2767dd7cddfSDavid du Colombier 			threadexitsall(msg);
2777dd7cddfSDavid du Colombier 	print("acme: %s\n", msg);
2787dd7cddfSDavid du Colombier 	abort();
2799a747e4fSDavid du Colombier 	return 0;
2807dd7cddfSDavid du Colombier }
2817dd7cddfSDavid du Colombier 
2827dd7cddfSDavid du Colombier void
2837dd7cddfSDavid du Colombier killprocs(void)
2847dd7cddfSDavid du Colombier {
2857dd7cddfSDavid du Colombier 	Command *c;
2867dd7cddfSDavid du Colombier 
2877dd7cddfSDavid du Colombier 	fsysclose();
2889a747e4fSDavid du Colombier //	if(display)
2899a747e4fSDavid du Colombier //		flushimage(display, 1);
2907dd7cddfSDavid du Colombier 
2917dd7cddfSDavid du Colombier 	for(c=command; c; c=c->next)
2927dd7cddfSDavid du Colombier 		postnote(PNGROUP, c->pid, "hangup");
2937dd7cddfSDavid du Colombier 	remove(acmeerrorfile);
2947dd7cddfSDavid du Colombier }
2957dd7cddfSDavid du Colombier 
2967dd7cddfSDavid du Colombier static int errorfd;
2977dd7cddfSDavid du Colombier 
2987dd7cddfSDavid du Colombier void
2997dd7cddfSDavid du Colombier acmeerrorproc(void *)
3007dd7cddfSDavid du Colombier {
3017dd7cddfSDavid du Colombier 	char *buf;
3027dd7cddfSDavid du Colombier 	int n;
3037dd7cddfSDavid du Colombier 
3047dd7cddfSDavid du Colombier 	threadsetname("acmeerrorproc");
3057dd7cddfSDavid du Colombier 	buf = emalloc(8192+1);
3067dd7cddfSDavid du Colombier 	while((n=read(errorfd, buf, 8192)) >= 0){
3077dd7cddfSDavid du Colombier 		buf[n] = '\0';
30859cc4ca5SDavid du Colombier 		sendp(cerr, estrdup(buf));
3097dd7cddfSDavid du Colombier 	}
3107dd7cddfSDavid du Colombier }
3117dd7cddfSDavid du Colombier 
3127dd7cddfSDavid du Colombier void
3137dd7cddfSDavid du Colombier acmeerrorinit(void)
3147dd7cddfSDavid du Colombier {
3157dd7cddfSDavid du Colombier 	int fd, pfd[2];
3167dd7cddfSDavid du Colombier 	char buf[64];
3177dd7cddfSDavid du Colombier 
3187dd7cddfSDavid du Colombier 	if(pipe(pfd) < 0)
3197dd7cddfSDavid du Colombier 		error("can't create pipe");
3207dd7cddfSDavid du Colombier 	sprint(acmeerrorfile, "/srv/acme.%s.%d", getuser(), mainpid);
3217dd7cddfSDavid du Colombier 	fd = create(acmeerrorfile, OWRITE, 0666);
3227dd7cddfSDavid du Colombier 	if(fd < 0){
32359cc4ca5SDavid du Colombier 		remove(acmeerrorfile);
3247dd7cddfSDavid du Colombier   		fd = create(acmeerrorfile, OWRITE, 0666);
3257dd7cddfSDavid du Colombier 		if(fd < 0)
3267dd7cddfSDavid du Colombier 			error("can't create acmeerror file");
3277dd7cddfSDavid du Colombier 	}
3287dd7cddfSDavid du Colombier 	sprint(buf, "%d", pfd[0]);
3297dd7cddfSDavid du Colombier 	write(fd, buf, strlen(buf));
3307dd7cddfSDavid du Colombier 	close(fd);
3317dd7cddfSDavid du Colombier 	/* reopen pfd[1] close on exec */
3327dd7cddfSDavid du Colombier 	sprint(buf, "/fd/%d", pfd[1]);
3337dd7cddfSDavid du Colombier 	errorfd = open(buf, OREAD|OCEXEC);
3347dd7cddfSDavid du Colombier 	if(errorfd < 0)
3357dd7cddfSDavid du Colombier 		error("can't re-open acmeerror file");
3367dd7cddfSDavid du Colombier 	close(pfd[1]);
3377dd7cddfSDavid du Colombier 	close(pfd[0]);
3387dd7cddfSDavid du Colombier 	proccreate(acmeerrorproc, nil, STACK);
3397dd7cddfSDavid du Colombier }
3407dd7cddfSDavid du Colombier 
3417dd7cddfSDavid du Colombier void
3427dd7cddfSDavid du Colombier plumbproc(void *)
3437dd7cddfSDavid du Colombier {
3447dd7cddfSDavid du Colombier 	Plumbmsg *m;
3457dd7cddfSDavid du Colombier 
3467dd7cddfSDavid du Colombier 	threadsetname("plumbproc");
3477dd7cddfSDavid du Colombier 	for(;;){
3487dd7cddfSDavid du Colombier 		m = plumbrecv(plumbeditfd);
3497dd7cddfSDavid du Colombier 		if(m == nil)
3507dd7cddfSDavid du Colombier 			threadexits(nil);
3517dd7cddfSDavid du Colombier 		sendp(cplumb, m);
3527dd7cddfSDavid du Colombier 	}
3537dd7cddfSDavid du Colombier }
3547dd7cddfSDavid du Colombier 
3557dd7cddfSDavid du Colombier void
3567dd7cddfSDavid du Colombier keyboardthread(void *)
3577dd7cddfSDavid du Colombier {
3587dd7cddfSDavid du Colombier 	Rune r;
3597dd7cddfSDavid du Colombier 	Timer *timer;
3607dd7cddfSDavid du Colombier 	Text *t;
3617dd7cddfSDavid du Colombier 	enum { KTimer, KKey, NKALT };
3627dd7cddfSDavid du Colombier 	static Alt alts[NKALT+1];
3637dd7cddfSDavid du Colombier 
3647dd7cddfSDavid du Colombier 	alts[KTimer].c = nil;
3657dd7cddfSDavid du Colombier 	alts[KTimer].v = nil;
3667dd7cddfSDavid du Colombier 	alts[KTimer].op = CHANNOP;
3677dd7cddfSDavid du Colombier 	alts[KKey].c = keyboardctl->c;
3687dd7cddfSDavid du Colombier 	alts[KKey].v = &r;
3697dd7cddfSDavid du Colombier 	alts[KKey].op = CHANRCV;
3707dd7cddfSDavid du Colombier 	alts[NKALT].op = CHANEND;
3717dd7cddfSDavid du Colombier 
3727dd7cddfSDavid du Colombier 	timer = nil;
3737dd7cddfSDavid du Colombier 	typetext = nil;
3747dd7cddfSDavid du Colombier 	threadsetname("keyboardthread");
3757dd7cddfSDavid du Colombier 	for(;;){
3767dd7cddfSDavid du Colombier 		switch(alt(alts)){
3777dd7cddfSDavid du Colombier 		case KTimer:
3787dd7cddfSDavid du Colombier 			timerstop(timer);
3797dd7cddfSDavid du Colombier 			t = typetext;
3807dd7cddfSDavid du Colombier 			if(t!=nil && t->what==Tag){
3817dd7cddfSDavid du Colombier 				winlock(t->w, 'K');
3827dd7cddfSDavid du Colombier 				wincommit(t->w, t);
3837dd7cddfSDavid du Colombier 				winunlock(t->w);
3847dd7cddfSDavid du Colombier 				flushimage(display, 1);
3857dd7cddfSDavid du Colombier 			}
3867dd7cddfSDavid du Colombier 			alts[KTimer].c = nil;
3877dd7cddfSDavid du Colombier 			alts[KTimer].op = CHANNOP;
3887dd7cddfSDavid du Colombier 			break;
3897dd7cddfSDavid du Colombier 		case KKey:
3907dd7cddfSDavid du Colombier 		casekeyboard:
3917dd7cddfSDavid du Colombier 			typetext = rowtype(&row, r, mouse->xy);
3927dd7cddfSDavid du Colombier 			t = typetext;
3939a747e4fSDavid du Colombier 			if(t!=nil && t->col!=nil && !(r==Kdown || r==Kleft || r==Kright))	/* scrolling doesn't change activecol */
3947dd7cddfSDavid du Colombier 				activecol = t->col;
3957dd7cddfSDavid du Colombier 			if(t!=nil && t->w!=nil)
3967dd7cddfSDavid du Colombier 				t->w->body.file->curtext = &t->w->body;
3977dd7cddfSDavid du Colombier 			if(timer != nil)
3987dd7cddfSDavid du Colombier 				timercancel(timer);
3997dd7cddfSDavid du Colombier 			if(t!=nil && t->what==Tag) {
4007dd7cddfSDavid du Colombier 				timer = timerstart(500);
4017dd7cddfSDavid du Colombier 				alts[KTimer].c = timer->c;
4027dd7cddfSDavid du Colombier 				alts[KTimer].op = CHANRCV;
4037dd7cddfSDavid du Colombier 			}else{
4047dd7cddfSDavid du Colombier 				timer = nil;
4057dd7cddfSDavid du Colombier 				alts[KTimer].c = nil;
4067dd7cddfSDavid du Colombier 				alts[KTimer].op = CHANNOP;
4077dd7cddfSDavid du Colombier 			}
4087dd7cddfSDavid du Colombier 			if(nbrecv(keyboardctl->c, &r) > 0)
4097dd7cddfSDavid du Colombier 				goto casekeyboard;
4107dd7cddfSDavid du Colombier 			flushimage(display, 1);
4117dd7cddfSDavid du Colombier 			break;
4127dd7cddfSDavid du Colombier 		}
4137dd7cddfSDavid du Colombier 	}
4147dd7cddfSDavid du Colombier }
4157dd7cddfSDavid du Colombier 
4167dd7cddfSDavid du Colombier void
4177dd7cddfSDavid du Colombier mousethread(void *)
4187dd7cddfSDavid du Colombier {
4197dd7cddfSDavid du Colombier 	Text *t, *argt;
4207dd7cddfSDavid du Colombier 	int but;
4217dd7cddfSDavid du Colombier 	uint q0, q1;
4227dd7cddfSDavid du Colombier 	Window *w;
4237dd7cddfSDavid du Colombier 	Plumbmsg *pm;
4249a747e4fSDavid du Colombier 	Mouse m;
4257dd7cddfSDavid du Colombier 	char *act;
4265316891fSDavid du Colombier 	enum { MResize, MMouse, MPlumb, MWarnings, NMALT };
4277dd7cddfSDavid du Colombier 	static Alt alts[NMALT+1];
4287dd7cddfSDavid du Colombier 
4297dd7cddfSDavid du Colombier 	threadsetname("mousethread");
4307dd7cddfSDavid du Colombier 	alts[MResize].c = mousectl->resizec;
4317dd7cddfSDavid du Colombier 	alts[MResize].v = nil;
4327dd7cddfSDavid du Colombier 	alts[MResize].op = CHANRCV;
4337dd7cddfSDavid du Colombier 	alts[MMouse].c = mousectl->c;
4347dd7cddfSDavid du Colombier 	alts[MMouse].v = &mousectl->Mouse;
4357dd7cddfSDavid du Colombier 	alts[MMouse].op = CHANRCV;
4367dd7cddfSDavid du Colombier 	alts[MPlumb].c = cplumb;
4377dd7cddfSDavid du Colombier 	alts[MPlumb].v = &pm;
4387dd7cddfSDavid du Colombier 	alts[MPlumb].op = CHANRCV;
4395316891fSDavid du Colombier 	alts[MWarnings].c = cwarn;
4405316891fSDavid du Colombier 	alts[MWarnings].v = nil;
4415316891fSDavid du Colombier 	alts[MWarnings].op = CHANRCV;
4427dd7cddfSDavid du Colombier 	if(cplumb == nil)
4437dd7cddfSDavid du Colombier 		alts[MPlumb].op = CHANNOP;
4447dd7cddfSDavid du Colombier 	alts[NMALT].op = CHANEND;
4457dd7cddfSDavid du Colombier 
4467dd7cddfSDavid du Colombier 	for(;;){
4475316891fSDavid du Colombier 		qlock(&row);
4485316891fSDavid du Colombier 		flushwarnings();
4495316891fSDavid du Colombier 		qunlock(&row);
4505316891fSDavid du Colombier 		flushimage(display, 1);
4517dd7cddfSDavid du Colombier 		switch(alt(alts)){
4527dd7cddfSDavid du Colombier 		case MResize:
4537dd7cddfSDavid du Colombier 			if(getwindow(display, Refnone) < 0)
4547dd7cddfSDavid du Colombier 				error("attach to window");
4557dd7cddfSDavid du Colombier 			scrlresize();
4567dd7cddfSDavid du Colombier 			rowresize(&row, screen->clipr);
4577dd7cddfSDavid du Colombier 			break;
4587dd7cddfSDavid du Colombier 		case MPlumb:
4597dd7cddfSDavid du Colombier 			if(strcmp(pm->type, "text") == 0){
4607dd7cddfSDavid du Colombier 				act = plumblookup(pm->attr, "action");
4617dd7cddfSDavid du Colombier 				if(act==nil || strcmp(act, "showfile")==0)
4627dd7cddfSDavid du Colombier 					plumblook(pm);
4637dd7cddfSDavid du Colombier 				else if(strcmp(act, "showdata")==0)
4647dd7cddfSDavid du Colombier 					plumbshow(pm);
4657dd7cddfSDavid du Colombier 			}
4667dd7cddfSDavid du Colombier 			plumbfree(pm);
4677dd7cddfSDavid du Colombier 			break;
4685316891fSDavid du Colombier 		case MWarnings:
4695316891fSDavid du Colombier 			break;
4707dd7cddfSDavid du Colombier 		case MMouse:
4719a747e4fSDavid du Colombier 			/*
4729a747e4fSDavid du Colombier 			 * Make a copy so decisions are consistent; mousectl changes
4739a747e4fSDavid du Colombier 			 * underfoot.  Can't just receive into m because this introduces
4749a747e4fSDavid du Colombier 			 * another race; see /sys/src/libdraw/mouse.c.
4759a747e4fSDavid du Colombier 			 */
4769a747e4fSDavid du Colombier 			m = mousectl->Mouse;
4777dd7cddfSDavid du Colombier 			qlock(&row);
4789a747e4fSDavid du Colombier 			t = rowwhich(&row, m.xy);
4797dd7cddfSDavid du Colombier 			if(t!=mousetext && mousetext!=nil && mousetext->w!=nil){
4807dd7cddfSDavid du Colombier 				winlock(mousetext->w, 'M');
4817dd7cddfSDavid du Colombier 				mousetext->eq0 = ~0;
4827dd7cddfSDavid du Colombier 				wincommit(mousetext->w, mousetext);
4837dd7cddfSDavid du Colombier 				winunlock(mousetext->w);
4847dd7cddfSDavid du Colombier 			}
4857dd7cddfSDavid du Colombier 			mousetext = t;
4867dd7cddfSDavid du Colombier 			if(t == nil)
4877dd7cddfSDavid du Colombier 				goto Continue;
4887dd7cddfSDavid du Colombier 			w = t->w;
4899a747e4fSDavid du Colombier 			if(t==nil || m.buttons==0)
4907dd7cddfSDavid du Colombier 				goto Continue;
4917dd7cddfSDavid du Colombier 			but = 0;
4929a747e4fSDavid du Colombier 			if(m.buttons == 1)
4937dd7cddfSDavid du Colombier 				but = 1;
4949a747e4fSDavid du Colombier 			else if(m.buttons == 2)
4957dd7cddfSDavid du Colombier 				but = 2;
4969a747e4fSDavid du Colombier 			else if(m.buttons == 4)
4977dd7cddfSDavid du Colombier 				but = 3;
4987dd7cddfSDavid du Colombier 			barttext = t;
4999a747e4fSDavid du Colombier 			if(t->what==Body && ptinrect(m.xy, t->scrollr)){
5007dd7cddfSDavid du Colombier 				if(but){
5017dd7cddfSDavid du Colombier 					winlock(w, 'M');
5027dd7cddfSDavid du Colombier 					t->eq0 = ~0;
5037dd7cddfSDavid du Colombier 					textscroll(t, but);
5047dd7cddfSDavid du Colombier 					winunlock(w);
5057dd7cddfSDavid du Colombier 				}
5067dd7cddfSDavid du Colombier 				goto Continue;
5077dd7cddfSDavid du Colombier 			}
5089a747e4fSDavid du Colombier 			if(ptinrect(m.xy, t->scrollr)){
5097dd7cddfSDavid du Colombier 				if(but){
5107dd7cddfSDavid du Colombier 					if(t->what == Columntag)
5117dd7cddfSDavid du Colombier 						rowdragcol(&row, t->col, but);
5127dd7cddfSDavid du Colombier 					else if(t->what == Tag){
5137dd7cddfSDavid du Colombier 						coldragwin(t->col, t->w, but);
5147dd7cddfSDavid du Colombier 						if(t->w)
5157dd7cddfSDavid du Colombier 							barttext = &t->w->body;
5167dd7cddfSDavid du Colombier 					}
5177dd7cddfSDavid du Colombier 					if(t->col)
5187dd7cddfSDavid du Colombier 						activecol = t->col;
5197dd7cddfSDavid du Colombier 				}
5207dd7cddfSDavid du Colombier 				goto Continue;
5217dd7cddfSDavid du Colombier 			}
5229a747e4fSDavid du Colombier 			if(m.buttons){
5237dd7cddfSDavid du Colombier 				if(w)
5247dd7cddfSDavid du Colombier 					winlock(w, 'M');
5257dd7cddfSDavid du Colombier 				t->eq0 = ~0;
5267dd7cddfSDavid du Colombier 				if(w)
5277dd7cddfSDavid du Colombier 					wincommit(w, t);
5287dd7cddfSDavid du Colombier 				else
5297dd7cddfSDavid du Colombier 					textcommit(t, TRUE);
5309a747e4fSDavid du Colombier 				if(m.buttons & 1){
5317dd7cddfSDavid du Colombier 					textselect(t);
5327dd7cddfSDavid du Colombier 					if(w)
5337dd7cddfSDavid du Colombier 						winsettag(w);
5347dd7cddfSDavid du Colombier 					argtext = t;
5357dd7cddfSDavid du Colombier 					seltext = t;
5367dd7cddfSDavid du Colombier 					if(t->col)
5377dd7cddfSDavid du Colombier 						activecol = t->col;	/* button 1 only */
5387dd7cddfSDavid du Colombier 					if(t->w!=nil && t==&t->w->body)
5397dd7cddfSDavid du Colombier 						activewin = t->w;
5409a747e4fSDavid du Colombier 				}else if(m.buttons & 2){
5417dd7cddfSDavid du Colombier 					if(textselect2(t, &q0, &q1, &argt))
5427dd7cddfSDavid du Colombier 						execute(t, q0, q1, FALSE, argt);
5439a747e4fSDavid du Colombier 				}else if(m.buttons & 4){
5447dd7cddfSDavid du Colombier 					if(textselect3(t, &q0, &q1))
5457dd7cddfSDavid du Colombier 						look3(t, q0, q1, FALSE);
5467dd7cddfSDavid du Colombier 				}
5477dd7cddfSDavid du Colombier 				if(w)
5487dd7cddfSDavid du Colombier 					winunlock(w);
5497dd7cddfSDavid du Colombier 				goto Continue;
5507dd7cddfSDavid du Colombier 			}
5517dd7cddfSDavid du Colombier     Continue:
5527dd7cddfSDavid du Colombier 			qunlock(&row);
5537dd7cddfSDavid du Colombier 			break;
5547dd7cddfSDavid du Colombier 		}
5557dd7cddfSDavid du Colombier 	}
5567dd7cddfSDavid du Colombier }
5577dd7cddfSDavid du Colombier 
5587dd7cddfSDavid du Colombier /*
5597dd7cddfSDavid du Colombier  * There is a race between process exiting and our finding out it was ever created.
5607dd7cddfSDavid du Colombier  * This structure keeps a list of processes that have exited we haven't heard of.
5617dd7cddfSDavid du Colombier  */
5627dd7cddfSDavid du Colombier typedef struct Pid Pid;
5637dd7cddfSDavid du Colombier struct Pid
5647dd7cddfSDavid du Colombier {
5657dd7cddfSDavid du Colombier 	int	pid;
5669a747e4fSDavid du Colombier 	char	msg[ERRMAX];
5677dd7cddfSDavid du Colombier 	Pid	*next;
5687dd7cddfSDavid du Colombier };
5697dd7cddfSDavid du Colombier 
5707dd7cddfSDavid du Colombier void
5717dd7cddfSDavid du Colombier waitthread(void *)
5727dd7cddfSDavid du Colombier {
5739a747e4fSDavid du Colombier 	Waitmsg *w;
5747dd7cddfSDavid du Colombier 	Command *c, *lc;
5757dd7cddfSDavid du Colombier 	uint pid;
5767dd7cddfSDavid du Colombier 	int found, ncmd;
5777dd7cddfSDavid du Colombier 	Rune *cmd;
5787dd7cddfSDavid du Colombier 	char *err;
5797dd7cddfSDavid du Colombier 	Text *t;
5807dd7cddfSDavid du Colombier 	Pid *pids, *p, *lastp;
5817dd7cddfSDavid du Colombier 	enum { WErr, WKill, WWait, WCmd, NWALT };
5827dd7cddfSDavid du Colombier 	Alt alts[NWALT+1];
5837dd7cddfSDavid du Colombier 
5847dd7cddfSDavid du Colombier 	threadsetname("waitthread");
5857dd7cddfSDavid du Colombier 	pids = nil;
5867dd7cddfSDavid du Colombier 	alts[WErr].c = cerr;
5877dd7cddfSDavid du Colombier 	alts[WErr].v = &err;
5887dd7cddfSDavid du Colombier 	alts[WErr].op = CHANRCV;
5897dd7cddfSDavid du Colombier 	alts[WKill].c = ckill;
5907dd7cddfSDavid du Colombier 	alts[WKill].v = &cmd;
5917dd7cddfSDavid du Colombier 	alts[WKill].op = CHANRCV;
5927dd7cddfSDavid du Colombier 	alts[WWait].c = cwait;
5937dd7cddfSDavid du Colombier 	alts[WWait].v = &w;
5947dd7cddfSDavid du Colombier 	alts[WWait].op = CHANRCV;
5957dd7cddfSDavid du Colombier 	alts[WCmd].c = ccommand;
5967dd7cddfSDavid du Colombier 	alts[WCmd].v = &c;
5977dd7cddfSDavid du Colombier 	alts[WCmd].op = CHANRCV;
5987dd7cddfSDavid du Colombier 	alts[NWALT].op = CHANEND;
5997dd7cddfSDavid du Colombier 
6007dd7cddfSDavid du Colombier 	command = nil;
6017dd7cddfSDavid du Colombier 	for(;;){
6027dd7cddfSDavid du Colombier 		switch(alt(alts)){
6037dd7cddfSDavid du Colombier 		case WErr:
6047dd7cddfSDavid du Colombier 			qlock(&row);
6057dd7cddfSDavid du Colombier 			warning(nil, "%s", err);
6067dd7cddfSDavid du Colombier 			free(err);
6077dd7cddfSDavid du Colombier 			flushimage(display, 1);
6087dd7cddfSDavid du Colombier 			qunlock(&row);
6097dd7cddfSDavid du Colombier 			break;
6107dd7cddfSDavid du Colombier 		case WKill:
6117dd7cddfSDavid du Colombier 			found = FALSE;
6127dd7cddfSDavid du Colombier 			ncmd = runestrlen(cmd);
6137dd7cddfSDavid du Colombier 			for(c=command; c; c=c->next){
6147dd7cddfSDavid du Colombier 				/* -1 for blank */
6157dd7cddfSDavid du Colombier 				if(runeeq(c->name, c->nname-1, cmd, ncmd) == TRUE){
6167dd7cddfSDavid du Colombier 					if(postnote(PNGROUP, c->pid, "kill") < 0)
6177dd7cddfSDavid du Colombier 						warning(nil, "kill %S: %r\n", cmd);
6187dd7cddfSDavid du Colombier 					found = TRUE;
6197dd7cddfSDavid du Colombier 				}
6207dd7cddfSDavid du Colombier 			}
6217dd7cddfSDavid du Colombier 			if(!found)
6227dd7cddfSDavid du Colombier 				warning(nil, "Kill: no process %S\n", cmd);
6237dd7cddfSDavid du Colombier 			free(cmd);
6247dd7cddfSDavid du Colombier 			break;
6257dd7cddfSDavid du Colombier 		case WWait:
6269a747e4fSDavid du Colombier 			pid = w->pid;
6277dd7cddfSDavid du Colombier 			lc = nil;
6287dd7cddfSDavid du Colombier 			for(c=command; c; c=c->next){
6297dd7cddfSDavid du Colombier 				if(c->pid == pid){
6307dd7cddfSDavid du Colombier 					if(lc)
6317dd7cddfSDavid du Colombier 						lc->next = c->next;
6327dd7cddfSDavid du Colombier 					else
6337dd7cddfSDavid du Colombier 						command = c->next;
6347dd7cddfSDavid du Colombier 					break;
6357dd7cddfSDavid du Colombier 				}
6367dd7cddfSDavid du Colombier 				lc = c;
6377dd7cddfSDavid du Colombier 			}
6387dd7cddfSDavid du Colombier 			qlock(&row);
6397dd7cddfSDavid du Colombier 			t = &row.tag;
6407dd7cddfSDavid du Colombier 			textcommit(t, TRUE);
6417dd7cddfSDavid du Colombier 			if(c == nil){
6427dd7cddfSDavid du Colombier 				/* helper processes use this exit status */
6439a747e4fSDavid du Colombier 				if(strncmp(w->msg, "libthread", 9) != 0){
6447dd7cddfSDavid du Colombier 					p = emalloc(sizeof(Pid));
6457dd7cddfSDavid du Colombier 					p->pid = pid;
6469a747e4fSDavid du Colombier 					strncpy(p->msg, w->msg, sizeof(p->msg));
6477dd7cddfSDavid du Colombier 					p->next = pids;
6487dd7cddfSDavid du Colombier 					pids = p;
6497dd7cddfSDavid du Colombier 				}
6507dd7cddfSDavid du Colombier 			}else{
6517dd7cddfSDavid du Colombier 				if(search(t, c->name, c->nname)){
6527dd7cddfSDavid du Colombier 					textdelete(t, t->q0, t->q1, TRUE);
6537dd7cddfSDavid du Colombier 					textsetselect(t, 0, 0);
6547dd7cddfSDavid du Colombier 				}
6559a747e4fSDavid du Colombier 				if(w->msg[0])
6569a747e4fSDavid du Colombier 					warning(c->md, "%s\n", w->msg);
6577dd7cddfSDavid du Colombier 				flushimage(display, 1);
6587dd7cddfSDavid du Colombier 			}
6597dd7cddfSDavid du Colombier 			qunlock(&row);
6609a747e4fSDavid du Colombier 			free(w);
6617dd7cddfSDavid du Colombier     Freecmd:
6627dd7cddfSDavid du Colombier 			if(c){
66359cc4ca5SDavid du Colombier 				if(c->iseditcmd)
66459cc4ca5SDavid du Colombier 					sendul(cedit, 0);
6657dd7cddfSDavid du Colombier 				free(c->text);
6667dd7cddfSDavid du Colombier 				free(c->name);
6677dd7cddfSDavid du Colombier 				fsysdelid(c->md);
6687dd7cddfSDavid du Colombier 				free(c);
6697dd7cddfSDavid du Colombier 			}
6707dd7cddfSDavid du Colombier 			break;
6717dd7cddfSDavid du Colombier 		case WCmd:
6727dd7cddfSDavid du Colombier 			/* has this command already exited? */
6737dd7cddfSDavid du Colombier 			lastp = nil;
6747dd7cddfSDavid du Colombier 			for(p=pids; p!=nil; p=p->next){
6757dd7cddfSDavid du Colombier 				if(p->pid == c->pid){
6767dd7cddfSDavid du Colombier 					if(p->msg[0])
6777dd7cddfSDavid du Colombier 						warning(c->md, "%s\n", p->msg);
6787dd7cddfSDavid du Colombier 					if(lastp == nil)
6797dd7cddfSDavid du Colombier 						pids = p->next;
6807dd7cddfSDavid du Colombier 					else
6817dd7cddfSDavid du Colombier 						lastp->next = p->next;
6827dd7cddfSDavid du Colombier 					free(p);
6837dd7cddfSDavid du Colombier 					goto Freecmd;
6847dd7cddfSDavid du Colombier 				}
6857dd7cddfSDavid du Colombier 				lastp = p;
6867dd7cddfSDavid du Colombier 			}
6877dd7cddfSDavid du Colombier 			c->next = command;
6887dd7cddfSDavid du Colombier 			command = c;
6897dd7cddfSDavid du Colombier 			qlock(&row);
6907dd7cddfSDavid du Colombier 			t = &row.tag;
6917dd7cddfSDavid du Colombier 			textcommit(t, TRUE);
6927dd7cddfSDavid du Colombier 			textinsert(t, 0, c->name, c->nname, TRUE);
6937dd7cddfSDavid du Colombier 			textsetselect(t, 0, 0);
6947dd7cddfSDavid du Colombier 			flushimage(display, 1);
6957dd7cddfSDavid du Colombier 			qunlock(&row);
6967dd7cddfSDavid du Colombier 			break;
6977dd7cddfSDavid du Colombier 		}
6987dd7cddfSDavid du Colombier 	}
6997dd7cddfSDavid du Colombier }
7007dd7cddfSDavid du Colombier 
7017dd7cddfSDavid du Colombier void
7027dd7cddfSDavid du Colombier xfidallocthread(void*)
7037dd7cddfSDavid du Colombier {
7047dd7cddfSDavid du Colombier 	Xfid *xfree, *x;
7057dd7cddfSDavid du Colombier 	enum { Alloc, Free, N };
7067dd7cddfSDavid du Colombier 	static Alt alts[N+1];
7077dd7cddfSDavid du Colombier 
7087dd7cddfSDavid du Colombier 	threadsetname("xfidallocthread");
7097dd7cddfSDavid du Colombier 	alts[Alloc].c = cxfidalloc;
7107dd7cddfSDavid du Colombier 	alts[Alloc].v = nil;
7117dd7cddfSDavid du Colombier 	alts[Alloc].op = CHANRCV;
7127dd7cddfSDavid du Colombier 	alts[Free].c = cxfidfree;
7137dd7cddfSDavid du Colombier 	alts[Free].v = &x;
7147dd7cddfSDavid du Colombier 	alts[Free].op = CHANRCV;
7157dd7cddfSDavid du Colombier 	alts[N].op = CHANEND;
7167dd7cddfSDavid du Colombier 
7177dd7cddfSDavid du Colombier 	xfree = nil;
7187dd7cddfSDavid du Colombier 	for(;;){
7197dd7cddfSDavid du Colombier 		switch(alt(alts)){
7207dd7cddfSDavid du Colombier 		case Alloc:
7217dd7cddfSDavid du Colombier 			x = xfree;
7227dd7cddfSDavid du Colombier 			if(x)
7237dd7cddfSDavid du Colombier 				xfree = x->next;
7247dd7cddfSDavid du Colombier 			else{
7257dd7cddfSDavid du Colombier 				x = emalloc(sizeof(Xfid));
7267dd7cddfSDavid du Colombier 				x->c = chancreate(sizeof(void(*)(Xfid*)), 0);
7277dd7cddfSDavid du Colombier 				x->arg = x;
7287dd7cddfSDavid du Colombier 				threadcreate(xfidctl, x->arg, STACK);
7297dd7cddfSDavid du Colombier 			}
7307dd7cddfSDavid du Colombier 			sendp(cxfidalloc, x);
7317dd7cddfSDavid du Colombier 			break;
7327dd7cddfSDavid du Colombier 		case Free:
7337dd7cddfSDavid du Colombier 			x->next = xfree;
7347dd7cddfSDavid du Colombier 			xfree = x;
7357dd7cddfSDavid du Colombier 			break;
7367dd7cddfSDavid du Colombier 		}
7377dd7cddfSDavid du Colombier 	}
7387dd7cddfSDavid du Colombier }
7397dd7cddfSDavid du Colombier 
7409a747e4fSDavid du Colombier /* this thread, in the main proc, allows fsysproc to get a window made without doing graphics */
7419a747e4fSDavid du Colombier void
7429a747e4fSDavid du Colombier newwindowthread(void*)
7439a747e4fSDavid du Colombier {
7449a747e4fSDavid du Colombier 	Window *w;
7459a747e4fSDavid du Colombier 
7469a747e4fSDavid du Colombier 	threadsetname("newwindowthread");
7479a747e4fSDavid du Colombier 
7489a747e4fSDavid du Colombier 	for(;;){
7499a747e4fSDavid du Colombier 		/* only fsysproc is talking to us, so synchronization is trivial */
7509a747e4fSDavid du Colombier 		recvp(cnewwindow);
751d9306527SDavid du Colombier 		w = makenewwindow(nil);
7529a747e4fSDavid du Colombier 		winsettag(w);
7539a747e4fSDavid du Colombier 		sendp(cnewwindow, w);
7549a747e4fSDavid du Colombier 	}
7559a747e4fSDavid du Colombier }
7569a747e4fSDavid du Colombier 
7577dd7cddfSDavid du Colombier Reffont*
7587dd7cddfSDavid du Colombier rfget(int fix, int save, int setfont, char *name)
7597dd7cddfSDavid du Colombier {
7607dd7cddfSDavid du Colombier 	Reffont *r;
7617dd7cddfSDavid du Colombier 	Font *f;
7627dd7cddfSDavid du Colombier 	int i;
7637dd7cddfSDavid du Colombier 
7647dd7cddfSDavid du Colombier 	r = nil;
7657dd7cddfSDavid du Colombier 	if(name == nil){
7667dd7cddfSDavid du Colombier 		name = fontnames[fix];
7677dd7cddfSDavid du Colombier 		r = reffonts[fix];
7687dd7cddfSDavid du Colombier 	}
7697dd7cddfSDavid du Colombier 	if(r == nil){
7707dd7cddfSDavid du Colombier 		for(i=0; i<nfontcache; i++)
7717dd7cddfSDavid du Colombier 			if(strcmp(name, fontcache[i]->f->name) == 0){
7727dd7cddfSDavid du Colombier 				r = fontcache[i];
7737dd7cddfSDavid du Colombier 				goto Found;
7747dd7cddfSDavid du Colombier 			}
77559cc4ca5SDavid du Colombier 		f = openfont(display, name);
7767dd7cddfSDavid du Colombier 		if(f == nil){
7777dd7cddfSDavid du Colombier 			warning(nil, "can't open font file %s: %r\n", name);
7787dd7cddfSDavid du Colombier 			return nil;
7797dd7cddfSDavid du Colombier 		}
7807dd7cddfSDavid du Colombier 		r = emalloc(sizeof(Reffont));
7817dd7cddfSDavid du Colombier 		r->f = f;
7827dd7cddfSDavid du Colombier 		fontcache = realloc(fontcache, (nfontcache+1)*sizeof(Reffont*));
7837dd7cddfSDavid du Colombier 		fontcache[nfontcache++] = r;
7847dd7cddfSDavid du Colombier 	}
7857dd7cddfSDavid du Colombier     Found:
7867dd7cddfSDavid du Colombier 	if(save){
7877dd7cddfSDavid du Colombier 		incref(r);
7887dd7cddfSDavid du Colombier 		if(reffonts[fix])
7897dd7cddfSDavid du Colombier 			rfclose(reffonts[fix]);
7907dd7cddfSDavid du Colombier 		reffonts[fix] = r;
7919a747e4fSDavid du Colombier 		free(fontnames[fix]);
7927dd7cddfSDavid du Colombier 		fontnames[fix] = name;
7937dd7cddfSDavid du Colombier 	}
7947dd7cddfSDavid du Colombier 	if(setfont){
7957dd7cddfSDavid du Colombier 		reffont.f = r->f;
7967dd7cddfSDavid du Colombier 		incref(r);
7977dd7cddfSDavid du Colombier 		rfclose(reffonts[0]);
7987dd7cddfSDavid du Colombier 		font = r->f;
7997dd7cddfSDavid du Colombier 		reffonts[0] = r;
8007dd7cddfSDavid du Colombier 		incref(r);
8017dd7cddfSDavid du Colombier 		iconinit();
8027dd7cddfSDavid du Colombier 	}
8037dd7cddfSDavid du Colombier 	incref(r);
8047dd7cddfSDavid du Colombier 	return r;
8057dd7cddfSDavid du Colombier }
8067dd7cddfSDavid du Colombier 
8077dd7cddfSDavid du Colombier void
8087dd7cddfSDavid du Colombier rfclose(Reffont *r)
8097dd7cddfSDavid du Colombier {
8107dd7cddfSDavid du Colombier 	int i;
8117dd7cddfSDavid du Colombier 
8127dd7cddfSDavid du Colombier 	if(decref(r) == 0){
8137dd7cddfSDavid du Colombier 		for(i=0; i<nfontcache; i++)
8147dd7cddfSDavid du Colombier 			if(r == fontcache[i])
8157dd7cddfSDavid du Colombier 				break;
8167dd7cddfSDavid du Colombier 		if(i >= nfontcache)
8177dd7cddfSDavid du Colombier 			warning(nil, "internal error: can't find font in cache\n");
8187dd7cddfSDavid du Colombier 		else{
8197dd7cddfSDavid du Colombier 			nfontcache--;
8207dd7cddfSDavid du Colombier 			memmove(fontcache+i, fontcache+i+1, (nfontcache-i)*sizeof(Reffont*));
8217dd7cddfSDavid du Colombier 		}
8227dd7cddfSDavid du Colombier 		freefont(r->f);
8237dd7cddfSDavid du Colombier 		free(r);
8247dd7cddfSDavid du Colombier 	}
8257dd7cddfSDavid du Colombier }
8267dd7cddfSDavid du Colombier 
8277dd7cddfSDavid du Colombier Cursor boxcursor = {
8287dd7cddfSDavid du Colombier 	{-7, -7},
8297dd7cddfSDavid du Colombier 	{0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
8307dd7cddfSDavid du Colombier 	 0xFF, 0xFF, 0xF8, 0x1F, 0xF8, 0x1F, 0xF8, 0x1F,
8317dd7cddfSDavid du Colombier 	 0xF8, 0x1F, 0xF8, 0x1F, 0xF8, 0x1F, 0xFF, 0xFF,
8327dd7cddfSDavid du Colombier 	 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF},
8337dd7cddfSDavid du Colombier 	{0x00, 0x00, 0x7F, 0xFE, 0x7F, 0xFE, 0x7F, 0xFE,
8347dd7cddfSDavid du Colombier 	 0x70, 0x0E, 0x70, 0x0E, 0x70, 0x0E, 0x70, 0x0E,
8357dd7cddfSDavid du Colombier 	 0x70, 0x0E, 0x70, 0x0E, 0x70, 0x0E, 0x70, 0x0E,
8367dd7cddfSDavid du Colombier 	 0x7F, 0xFE, 0x7F, 0xFE, 0x7F, 0xFE, 0x00, 0x00}
8377dd7cddfSDavid du Colombier };
8387dd7cddfSDavid du Colombier 
8397dd7cddfSDavid du Colombier void
8407dd7cddfSDavid du Colombier iconinit(void)
8417dd7cddfSDavid du Colombier {
8427dd7cddfSDavid du Colombier 	Rectangle r;
8437dd7cddfSDavid du Colombier 	Image *tmp;
8447dd7cddfSDavid du Colombier 
8457dd7cddfSDavid du Colombier 	/* Blue */
8467dd7cddfSDavid du Colombier 	tagcols[BACK] = allocimagemix(display, DPalebluegreen, DWhite);
8479a747e4fSDavid du Colombier 	tagcols[HIGH] = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DPalegreygreen);
8489a747e4fSDavid du Colombier 	tagcols[BORD] = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DPurpleblue);
8497dd7cddfSDavid du Colombier 	tagcols[TEXT] = display->black;
8507dd7cddfSDavid du Colombier 	tagcols[HTEXT] = display->black;
8517dd7cddfSDavid du Colombier 
8527dd7cddfSDavid du Colombier 	/* Yellow */
8537dd7cddfSDavid du Colombier 	textcols[BACK] = allocimagemix(display, DPaleyellow, DWhite);
8549a747e4fSDavid du Colombier 	textcols[HIGH] = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DDarkyellow);
8559a747e4fSDavid du Colombier 	textcols[BORD] = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DYellowgreen);
8567dd7cddfSDavid du Colombier 	textcols[TEXT] = display->black;
8577dd7cddfSDavid du Colombier 	textcols[HTEXT] = display->black;
8587dd7cddfSDavid du Colombier 
8597dd7cddfSDavid du Colombier 	if(button){
8607dd7cddfSDavid du Colombier 		freeimage(button);
8617dd7cddfSDavid du Colombier 		freeimage(modbutton);
8627dd7cddfSDavid du Colombier 		freeimage(colbutton);
8637dd7cddfSDavid du Colombier 	}
8647dd7cddfSDavid du Colombier 
8657dd7cddfSDavid du Colombier 	r = Rect(0, 0, Scrollwid+2, font->height+1);
8669a747e4fSDavid du Colombier 	button = allocimage(display, r, screen->chan, 0, DNofill);
8677dd7cddfSDavid du Colombier 	draw(button, r, tagcols[BACK], nil, r.min);
8687dd7cddfSDavid du Colombier 	r.max.x -= 2;
8697dd7cddfSDavid du Colombier 	border(button, r, 2, tagcols[BORD], ZP);
8707dd7cddfSDavid du Colombier 
8717dd7cddfSDavid du Colombier 	r = button->r;
8729a747e4fSDavid du Colombier 	modbutton = allocimage(display, r, screen->chan, 0, DNofill);
8737dd7cddfSDavid du Colombier 	draw(modbutton, r, tagcols[BACK], nil, r.min);
8747dd7cddfSDavid du Colombier 	r.max.x -= 2;
8757dd7cddfSDavid du Colombier 	border(modbutton, r, 2, tagcols[BORD], ZP);
8767dd7cddfSDavid du Colombier 	r = insetrect(r, 2);
8779a747e4fSDavid du Colombier 	tmp = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DMedblue);
8787dd7cddfSDavid du Colombier 	draw(modbutton, r, tmp, nil, ZP);
8797dd7cddfSDavid du Colombier 	freeimage(tmp);
8807dd7cddfSDavid du Colombier 
8817dd7cddfSDavid du Colombier 	r = button->r;
8829a747e4fSDavid du Colombier 	colbutton = allocimage(display, r, screen->chan, 0, DPurpleblue);
8837dd7cddfSDavid du Colombier 
8849a747e4fSDavid du Colombier 	but2col = allocimage(display, r, screen->chan, 1, 0xAA0000FF);
8859a747e4fSDavid du Colombier 	but3col = allocimage(display, r, screen->chan, 1, 0x006600FF);
8867dd7cddfSDavid du Colombier }
8877dd7cddfSDavid du Colombier 
8887dd7cddfSDavid du Colombier /*
8897dd7cddfSDavid du Colombier  * /dev/snarf updates when the file is closed, so we must open our own
8907dd7cddfSDavid du Colombier  * fd here rather than use snarffd
8917dd7cddfSDavid du Colombier  */
8927dd7cddfSDavid du Colombier 
8937dd7cddfSDavid du Colombier /* rio truncates larges snarf buffers, so this avoids using the
8947dd7cddfSDavid du Colombier  * service if the string is huge */
8957dd7cddfSDavid du Colombier 
8967dd7cddfSDavid du Colombier #define MAXSNARF 100*1024
8977dd7cddfSDavid du Colombier 
8987dd7cddfSDavid du Colombier void
8997dd7cddfSDavid du Colombier putsnarf(void)
9007dd7cddfSDavid du Colombier {
9017dd7cddfSDavid du Colombier 	int fd, i, n;
9027dd7cddfSDavid du Colombier 
9037dd7cddfSDavid du Colombier 	if(snarffd<0 || snarfbuf.nc==0)
9047dd7cddfSDavid du Colombier 		return;
9057dd7cddfSDavid du Colombier 	if(snarfbuf.nc > MAXSNARF)
9067dd7cddfSDavid du Colombier 		return;
9077dd7cddfSDavid du Colombier 	fd = open("/dev/snarf", OWRITE);
9087dd7cddfSDavid du Colombier 	if(fd < 0)
9097dd7cddfSDavid du Colombier 		return;
9107dd7cddfSDavid du Colombier 	for(i=0; i<snarfbuf.nc; i+=n){
9117dd7cddfSDavid du Colombier 		n = snarfbuf.nc-i;
9127dd7cddfSDavid du Colombier 		if(n >= NSnarf)
9137dd7cddfSDavid du Colombier 			n = NSnarf;
9147dd7cddfSDavid du Colombier 		bufread(&snarfbuf, i, snarfrune, n);
9157dd7cddfSDavid du Colombier 		if(fprint(fd, "%.*S", n, snarfrune) < 0)
9167dd7cddfSDavid du Colombier 			break;
9177dd7cddfSDavid du Colombier 	}
9187dd7cddfSDavid du Colombier 	close(fd);
9197dd7cddfSDavid du Colombier }
9207dd7cddfSDavid du Colombier 
9217dd7cddfSDavid du Colombier void
9227dd7cddfSDavid du Colombier getsnarf()
9237dd7cddfSDavid du Colombier {
9247dd7cddfSDavid du Colombier 	int nulls;
9257dd7cddfSDavid du Colombier 
9267dd7cddfSDavid du Colombier 	if(snarfbuf.nc > MAXSNARF)
9277dd7cddfSDavid du Colombier 		return;
9287dd7cddfSDavid du Colombier 	if(snarffd < 0)
9297dd7cddfSDavid du Colombier 		return;
9307dd7cddfSDavid du Colombier 	seek(snarffd, 0, 0);
9317dd7cddfSDavid du Colombier 	bufreset(&snarfbuf);
9327dd7cddfSDavid du Colombier 	bufload(&snarfbuf, 0, snarffd, &nulls);
9337dd7cddfSDavid du Colombier }
934