xref: /plan9/acme/bin/source/win/win.c (revision 13ec27127e6fa2d1a530af3cca3d8494602f7465)
17dd7cddfSDavid du Colombier #include <u.h>
27dd7cddfSDavid du Colombier #include <libc.h>
37dd7cddfSDavid du Colombier #include <bio.h>
47dd7cddfSDavid du Colombier #include <thread.h>
57dd7cddfSDavid du Colombier #include "dat.h"
67dd7cddfSDavid du Colombier 
77dd7cddfSDavid du Colombier Window*
newwindow(void)87dd7cddfSDavid du Colombier newwindow(void)
97dd7cddfSDavid du Colombier {
107dd7cddfSDavid du Colombier 	char buf[12];
117dd7cddfSDavid du Colombier 	Window *w;
127dd7cddfSDavid du Colombier 
137dd7cddfSDavid du Colombier 	w = emalloc(sizeof(Window));
147dd7cddfSDavid du Colombier 	w->ctl = open("/mnt/wsys/new/ctl", ORDWR|OCEXEC);
157dd7cddfSDavid du Colombier 	if(w->ctl<0 || read(w->ctl, buf, 12)!=12)
167dd7cddfSDavid du Colombier 		error("can't open window ctl file: %r");
177dd7cddfSDavid du Colombier 	ctlprint(w->ctl, "noscroll\n");
187dd7cddfSDavid du Colombier 	w->id = atoi(buf);
197dd7cddfSDavid du Colombier 	w->event = winopenfile(w, "event");
207dd7cddfSDavid du Colombier 	w->addr = winopenfile(w, "addr");
217dd7cddfSDavid du Colombier 	w->body = winopenfile(w, "body");
227dd7cddfSDavid du Colombier 	w->data = winopenfile(w, "data");
237dd7cddfSDavid du Colombier 	w->cevent = chancreate(sizeof(Event*), 0);
247dd7cddfSDavid du Colombier 	return w;
257dd7cddfSDavid du Colombier }
267dd7cddfSDavid du Colombier 
277dd7cddfSDavid du Colombier void
winsetdump(Window * w,char * dir,char * cmd)287dd7cddfSDavid du Colombier winsetdump(Window *w, char *dir, char *cmd)
297dd7cddfSDavid du Colombier {
307dd7cddfSDavid du Colombier 	if(dir != nil)
317dd7cddfSDavid du Colombier 		ctlprint(w->ctl, "dumpdir %s\n", dir);
327dd7cddfSDavid du Colombier 	if(cmd != nil)
337dd7cddfSDavid du Colombier 		ctlprint(w->ctl, "dump %s\n", cmd);
347dd7cddfSDavid du Colombier }
357dd7cddfSDavid du Colombier 
367dd7cddfSDavid du Colombier void
wineventproc(void * v)377dd7cddfSDavid du Colombier wineventproc(void *v)
387dd7cddfSDavid du Colombier {
397dd7cddfSDavid du Colombier 	Window *w;
407dd7cddfSDavid du Colombier 	int i;
417dd7cddfSDavid du Colombier 
427dd7cddfSDavid du Colombier 	w = v;
437dd7cddfSDavid du Colombier 	for(i=0; ; i++){
447dd7cddfSDavid du Colombier 		if(i >= NEVENT)
457dd7cddfSDavid du Colombier 			i = 0;
467dd7cddfSDavid du Colombier 		wingetevent(w, &w->e[i]);
477dd7cddfSDavid du Colombier 		sendp(w->cevent, &w->e[i]);
487dd7cddfSDavid du Colombier 	}
497dd7cddfSDavid du Colombier }
507dd7cddfSDavid du Colombier 
517dd7cddfSDavid du Colombier int
winopenfile(Window * w,char * f)527dd7cddfSDavid du Colombier winopenfile(Window *w, char *f)
537dd7cddfSDavid du Colombier {
547dd7cddfSDavid du Colombier 	char buf[64];
557dd7cddfSDavid du Colombier 	int fd;
567dd7cddfSDavid du Colombier 
577dd7cddfSDavid du Colombier 	sprint(buf, "/mnt/wsys/%d/%s", w->id, f);
587dd7cddfSDavid du Colombier 	fd = open(buf, ORDWR|OCEXEC);
597dd7cddfSDavid du Colombier 	if(fd < 0)
607dd7cddfSDavid du Colombier 		error("can't open window file %s: %r", f);
617dd7cddfSDavid du Colombier 	return fd;
627dd7cddfSDavid du Colombier }
637dd7cddfSDavid du Colombier 
647dd7cddfSDavid du Colombier void
wintagwrite(Window * w,char * s,int n)657dd7cddfSDavid du Colombier wintagwrite(Window *w, char *s, int n)
667dd7cddfSDavid du Colombier {
677dd7cddfSDavid du Colombier 	int fd;
687dd7cddfSDavid du Colombier 
697dd7cddfSDavid du Colombier 	fd = winopenfile(w, "tag");
707dd7cddfSDavid du Colombier 	if(write(fd, s, n) != n)
717dd7cddfSDavid du Colombier 		error("tag write: %r");
727dd7cddfSDavid du Colombier 	close(fd);
737dd7cddfSDavid du Colombier }
747dd7cddfSDavid du Colombier 
757dd7cddfSDavid du Colombier void
winname(Window * w,char * s)767dd7cddfSDavid du Colombier winname(Window *w, char *s)
777dd7cddfSDavid du Colombier {
787dd7cddfSDavid du Colombier 	ctlprint(w->ctl, "name %s\n", s);
797dd7cddfSDavid du Colombier }
807dd7cddfSDavid du Colombier 
817dd7cddfSDavid du Colombier int
wingetec(Window * w)827dd7cddfSDavid du Colombier wingetec(Window *w)
837dd7cddfSDavid du Colombier {
847dd7cddfSDavid du Colombier 	if(w->nbuf == 0){
857dd7cddfSDavid du Colombier 		w->nbuf = read(w->event, w->buf, sizeof w->buf);
867dd7cddfSDavid du Colombier 		if(w->nbuf <= 0){
877dd7cddfSDavid du Colombier 			/* probably because window has exited, and only called by wineventproc, so just shut down */
887dd7cddfSDavid du Colombier 			threadexits(nil);
897dd7cddfSDavid du Colombier 		}
907dd7cddfSDavid du Colombier 		w->bufp = w->buf;
917dd7cddfSDavid du Colombier 	}
927dd7cddfSDavid du Colombier 	w->nbuf--;
937dd7cddfSDavid du Colombier 	return *w->bufp++;
947dd7cddfSDavid du Colombier }
957dd7cddfSDavid du Colombier 
967dd7cddfSDavid du Colombier int
wingeten(Window * w)977dd7cddfSDavid du Colombier wingeten(Window *w)
987dd7cddfSDavid du Colombier {
997dd7cddfSDavid du Colombier 	int n, c;
1007dd7cddfSDavid du Colombier 
1017dd7cddfSDavid du Colombier 	n = 0;
1027dd7cddfSDavid du Colombier 	while('0'<=(c=wingetec(w)) && c<='9')
1037dd7cddfSDavid du Colombier 		n = n*10+(c-'0');
1047dd7cddfSDavid du Colombier 	if(c != ' ')
1057dd7cddfSDavid du Colombier 		error("event number syntax");
1067dd7cddfSDavid du Colombier 	return n;
1077dd7cddfSDavid du Colombier }
1087dd7cddfSDavid du Colombier 
1097dd7cddfSDavid du Colombier int
wingeter(Window * w,char * buf,int * nb)1107dd7cddfSDavid du Colombier wingeter(Window *w, char *buf, int *nb)
1117dd7cddfSDavid du Colombier {
1127dd7cddfSDavid du Colombier 	Rune r;
1137dd7cddfSDavid du Colombier 	int n;
1147dd7cddfSDavid du Colombier 
1157dd7cddfSDavid du Colombier 	r = wingetec(w);
1167dd7cddfSDavid du Colombier 	buf[0] = r;
1177dd7cddfSDavid du Colombier 	n = 1;
1187dd7cddfSDavid du Colombier 	if(r >= Runeself) {
1197dd7cddfSDavid du Colombier 		while(!fullrune(buf, n))
1207dd7cddfSDavid du Colombier 			buf[n++] = wingetec(w);
1217dd7cddfSDavid du Colombier 		chartorune(&r, buf);
1227dd7cddfSDavid du Colombier 	}
1237dd7cddfSDavid du Colombier 	*nb = n;
1247dd7cddfSDavid du Colombier 	return r;
1257dd7cddfSDavid du Colombier }
1267dd7cddfSDavid du Colombier 
1277dd7cddfSDavid du Colombier void
wingetevent(Window * w,Event * e)1287dd7cddfSDavid du Colombier wingetevent(Window *w, Event *e)
1297dd7cddfSDavid du Colombier {
1307dd7cddfSDavid du Colombier 	int i, nb;
1317dd7cddfSDavid du Colombier 
1327dd7cddfSDavid du Colombier 	e->c1 = wingetec(w);
1337dd7cddfSDavid du Colombier 	e->c2 = wingetec(w);
1347dd7cddfSDavid du Colombier 	e->q0 = wingeten(w);
1357dd7cddfSDavid du Colombier 	e->q1 = wingeten(w);
1367dd7cddfSDavid du Colombier 	e->flag = wingeten(w);
1377dd7cddfSDavid du Colombier 	e->nr = wingeten(w);
1387dd7cddfSDavid du Colombier 	if(e->nr > EVENTSIZE)
1397dd7cddfSDavid du Colombier 		error("event string too long");
1407dd7cddfSDavid du Colombier 	e->nb = 0;
1417dd7cddfSDavid du Colombier 	for(i=0; i<e->nr; i++){
1427dd7cddfSDavid du Colombier 		e->r[i] = wingeter(w, e->b+e->nb, &nb);
1437dd7cddfSDavid du Colombier 		e->nb += nb;
1447dd7cddfSDavid du Colombier 	}
1457dd7cddfSDavid du Colombier 	e->r[e->nr] = 0;
1467dd7cddfSDavid du Colombier 	e->b[e->nb] = 0;
1477dd7cddfSDavid du Colombier 	if(wingetec(w) != '\n')
1487dd7cddfSDavid du Colombier 		error("event syntax error");
1497dd7cddfSDavid du Colombier }
1507dd7cddfSDavid du Colombier 
1517dd7cddfSDavid du Colombier void
winwriteevent(Window * w,Event * e)1527dd7cddfSDavid du Colombier winwriteevent(Window *w, Event *e)
1537dd7cddfSDavid du Colombier {
1549a747e4fSDavid du Colombier 	fprint(w->event, "%c%c%d %d\n", e->c1, e->c2, e->q0, e->q1);
1557dd7cddfSDavid du Colombier }
1567dd7cddfSDavid du Colombier 
1577dd7cddfSDavid du Colombier static int
nrunes(char * s,int nb)1587dd7cddfSDavid du Colombier nrunes(char *s, int nb)
1597dd7cddfSDavid du Colombier {
1607dd7cddfSDavid du Colombier 	int i, n;
1617dd7cddfSDavid du Colombier 	Rune r;
1627dd7cddfSDavid du Colombier 
1637dd7cddfSDavid du Colombier 	n = 0;
1647dd7cddfSDavid du Colombier 	for(i=0; i<nb; n++)
1657dd7cddfSDavid du Colombier 		i += chartorune(&r, s+i);
1667dd7cddfSDavid du Colombier 	return n;
1677dd7cddfSDavid du Colombier }
1687dd7cddfSDavid du Colombier 
1697dd7cddfSDavid du Colombier int
winread(Window * w,uint q0,uint q1,char * data)1707dd7cddfSDavid du Colombier winread(Window *w, uint q0, uint q1, char *data)
1717dd7cddfSDavid du Colombier {
1727dd7cddfSDavid du Colombier 	int m, n, nr, nb;
1737dd7cddfSDavid du Colombier 	char buf[256];
1747dd7cddfSDavid du Colombier 
1757dd7cddfSDavid du Colombier 	if(w->addr < 0)
1767dd7cddfSDavid du Colombier 		w->addr = winopenfile(w, "addr");
1777dd7cddfSDavid du Colombier 	if(w->data < 0)
1787dd7cddfSDavid du Colombier 		w->data = winopenfile(w, "data");
1797dd7cddfSDavid du Colombier 	m = q0;
1807dd7cddfSDavid du Colombier 	nb = 0;
1817dd7cddfSDavid du Colombier 	while(m < q1){
1827dd7cddfSDavid du Colombier 		n = sprint(buf, "#%d", m);
1837dd7cddfSDavid du Colombier 		if(write(w->addr, buf, n) != n)
1847dd7cddfSDavid du Colombier 			error("error writing addr: %r");
1857dd7cddfSDavid du Colombier 		n = read(w->data, buf, sizeof buf);
186*13ec2712SDavid du Colombier 		if(n < 0)
1877dd7cddfSDavid du Colombier 			error("reading data: %r");
1887dd7cddfSDavid du Colombier 		nr = nrunes(buf, n);
1897dd7cddfSDavid du Colombier 		while(m+nr >q1){
1907dd7cddfSDavid du Colombier 			do; while(n>0 && (buf[--n]&0xC0)==0x80);
1917dd7cddfSDavid du Colombier 			--nr;
1927dd7cddfSDavid du Colombier 		}
1937dd7cddfSDavid du Colombier 		if(n == 0)
1947dd7cddfSDavid du Colombier 			break;
1957dd7cddfSDavid du Colombier 		memmove(data, buf, n);
1967dd7cddfSDavid du Colombier 		nb += n;
1977dd7cddfSDavid du Colombier 		data += n;
1987dd7cddfSDavid du Colombier 		*data = 0;
1997dd7cddfSDavid du Colombier 		m += nr;
2007dd7cddfSDavid du Colombier 	}
2017dd7cddfSDavid du Colombier 	return nb;
2027dd7cddfSDavid du Colombier }
2037dd7cddfSDavid du Colombier 
2047dd7cddfSDavid du Colombier void
windormant(Window * w)2057dd7cddfSDavid du Colombier windormant(Window *w)
2067dd7cddfSDavid du Colombier {
2077dd7cddfSDavid du Colombier 	if(w->addr >= 0){
2087dd7cddfSDavid du Colombier 		close(w->addr);
2097dd7cddfSDavid du Colombier 		w->addr = -1;
2107dd7cddfSDavid du Colombier 	}
2117dd7cddfSDavid du Colombier 	if(w->body >= 0){
2127dd7cddfSDavid du Colombier 		close(w->body);
2137dd7cddfSDavid du Colombier 		w->body = -1;
2147dd7cddfSDavid du Colombier 	}
2157dd7cddfSDavid du Colombier 	if(w->data >= 0){
2167dd7cddfSDavid du Colombier 		close(w->data);
2177dd7cddfSDavid du Colombier 		w->data = -1;
2187dd7cddfSDavid du Colombier 	}
2197dd7cddfSDavid du Colombier }
2207dd7cddfSDavid du Colombier 
2217dd7cddfSDavid du Colombier int
windel(Window * w,int sure)2227dd7cddfSDavid du Colombier windel(Window *w, int sure)
2237dd7cddfSDavid du Colombier {
2247dd7cddfSDavid du Colombier 	if(sure)
2257dd7cddfSDavid du Colombier 		write(w->ctl, "delete\n", 7);
2267dd7cddfSDavid du Colombier 	else if(write(w->ctl, "del\n", 4) != 4)
2277dd7cddfSDavid du Colombier 		return 0;
2287dd7cddfSDavid du Colombier 	/* event proc will die due to read error from event file */
2297dd7cddfSDavid du Colombier 	windormant(w);
2307dd7cddfSDavid du Colombier 	close(w->ctl);
2317dd7cddfSDavid du Colombier 	w->ctl = -1;
2327dd7cddfSDavid du Colombier 	close(w->event);
2337dd7cddfSDavid du Colombier 	w->event = -1;
2347dd7cddfSDavid du Colombier 	return 1;
2357dd7cddfSDavid du Colombier }
2367dd7cddfSDavid du Colombier 
2377dd7cddfSDavid du Colombier void
winclean(Window * w)2387dd7cddfSDavid du Colombier winclean(Window *w)
2397dd7cddfSDavid du Colombier {
2407dd7cddfSDavid du Colombier 	ctlprint(w->ctl, "clean\n");
2417dd7cddfSDavid du Colombier }
2427dd7cddfSDavid du Colombier 
2437dd7cddfSDavid du Colombier int
winsetaddr(Window * w,char * addr,int errok)2447dd7cddfSDavid du Colombier winsetaddr(Window *w, char *addr, int errok)
2457dd7cddfSDavid du Colombier {
2467dd7cddfSDavid du Colombier 	if(w->addr < 0)
2477dd7cddfSDavid du Colombier 		w->addr = winopenfile(w, "addr");
2487dd7cddfSDavid du Colombier 	if(write(w->addr, addr, strlen(addr)) < 0){
2497dd7cddfSDavid du Colombier 		if(!errok)
2507dd7cddfSDavid du Colombier 			error("error writing addr(%s): %r", addr);
2517dd7cddfSDavid du Colombier 		return 0;
2527dd7cddfSDavid du Colombier 	}
2537dd7cddfSDavid du Colombier 	return 1;
2547dd7cddfSDavid du Colombier }
2557dd7cddfSDavid du Colombier 
2567dd7cddfSDavid du Colombier int
winselect(Window * w,char * addr,int errok)2577dd7cddfSDavid du Colombier winselect(Window *w, char *addr, int errok)
2587dd7cddfSDavid du Colombier {
2597dd7cddfSDavid du Colombier 	if(winsetaddr(w, addr, errok)){
2607dd7cddfSDavid du Colombier 		ctlprint(w->ctl, "dot=addr\n");
2617dd7cddfSDavid du Colombier 		return 1;
2627dd7cddfSDavid du Colombier 	}
2637dd7cddfSDavid du Colombier 	return 0;
2647dd7cddfSDavid du Colombier }
265