xref: /plan9/acme/bin/source/acd/win.c (revision 3ff48bf5ed603850fcd251ddf13025d23d693782)
19a747e4fSDavid du Colombier #include "acd.h"
29a747e4fSDavid du Colombier 
39a747e4fSDavid du Colombier Window*
newwindow(void)49a747e4fSDavid du Colombier newwindow(void)
59a747e4fSDavid du Colombier {
69a747e4fSDavid du Colombier 	char buf[12];
79a747e4fSDavid du Colombier 	Window *w;
89a747e4fSDavid du Colombier 
99a747e4fSDavid du Colombier 	w = emalloc(sizeof(Window));
109a747e4fSDavid du Colombier 	w->ctl = open("/mnt/wsys/new/ctl", ORDWR|OCEXEC);
119a747e4fSDavid du Colombier 	if(w->ctl<0 || read(w->ctl, buf, 12)!=12)
129a747e4fSDavid du Colombier 		error("can't open window ctl file: %r");
139a747e4fSDavid du Colombier 	ctlprint(w->ctl, "noscroll\n");
149a747e4fSDavid du Colombier 	w->id = atoi(buf);
159a747e4fSDavid du Colombier 	w->event = winopenfile(w, "event");
169a747e4fSDavid du Colombier 	w->addr = -1;	/* will be opened when needed */
179a747e4fSDavid du Colombier 	w->body = nil;
189a747e4fSDavid du Colombier 	w->data = -1;
199a747e4fSDavid du Colombier 	w->cevent = chancreate(sizeof(Event*), 0);
209a747e4fSDavid du Colombier 	if(w->cevent == nil)
219a747e4fSDavid du Colombier 		error("cevent is nil: %r");
229a747e4fSDavid du Colombier 	return w;
239a747e4fSDavid du Colombier }
249a747e4fSDavid du Colombier 
259a747e4fSDavid du Colombier void
winsetdump(Window * w,char * dir,char * cmd)269a747e4fSDavid du Colombier winsetdump(Window *w, char *dir, char *cmd)
279a747e4fSDavid du Colombier {
289a747e4fSDavid du Colombier 	if(dir != nil)
299a747e4fSDavid du Colombier 		ctlprint(w->ctl, "dumpdir %s\n", dir);
309a747e4fSDavid du Colombier 	if(cmd != nil)
319a747e4fSDavid du Colombier 		ctlprint(w->ctl, "dump %s\n", cmd);
329a747e4fSDavid du Colombier }
339a747e4fSDavid du Colombier 
349a747e4fSDavid du Colombier void
wineventproc(void * v)359a747e4fSDavid du Colombier wineventproc(void *v)
369a747e4fSDavid du Colombier {
379a747e4fSDavid du Colombier 	Window *w;
389a747e4fSDavid du Colombier 	int i;
399a747e4fSDavid du Colombier 
409a747e4fSDavid du Colombier 	threadsetname("wineventproc");
419a747e4fSDavid du Colombier 	w = v;
429a747e4fSDavid du Colombier 	for(i=0; ; i++){
439a747e4fSDavid du Colombier 		if(i >= NEVENT)
449a747e4fSDavid du Colombier 			i = 0;
459a747e4fSDavid du Colombier 		wingetevent(w, &w->e[i]);
469a747e4fSDavid du Colombier 		sendp(w->cevent, &w->e[i]);
479a747e4fSDavid du Colombier 	}
489a747e4fSDavid du Colombier }
499a747e4fSDavid du Colombier 
509a747e4fSDavid du Colombier int
winopenfile(Window * w,char * f)519a747e4fSDavid du Colombier winopenfile(Window *w, char *f)
529a747e4fSDavid du Colombier {
539a747e4fSDavid du Colombier 	char buf[64];
549a747e4fSDavid du Colombier 	int fd;
559a747e4fSDavid du Colombier 
569a747e4fSDavid du Colombier 	sprint(buf, "/mnt/wsys/%d/%s", w->id, f);
579a747e4fSDavid du Colombier 	fd = open(buf, ORDWR|OCEXEC);
589a747e4fSDavid du Colombier 	if(fd < 0)
599a747e4fSDavid du Colombier 		error("can't open window file %s: %r", f);
609a747e4fSDavid du Colombier 	return fd;
619a747e4fSDavid du Colombier }
629a747e4fSDavid du Colombier 
639a747e4fSDavid du Colombier void
wintagwrite(Window * w,char * s,int n)649a747e4fSDavid du Colombier wintagwrite(Window *w, char *s, int n)
659a747e4fSDavid du Colombier {
669a747e4fSDavid du Colombier 	int fd;
679a747e4fSDavid du Colombier 
689a747e4fSDavid du Colombier 	fd = winopenfile(w, "tag");
699a747e4fSDavid du Colombier 	if(write(fd, s, n) != n)
709a747e4fSDavid du Colombier 		error("tag write: %r");
719a747e4fSDavid du Colombier 	close(fd);
729a747e4fSDavid du Colombier }
739a747e4fSDavid du Colombier 
749a747e4fSDavid du Colombier void
winname(Window * w,char * s)759a747e4fSDavid du Colombier winname(Window *w, char *s)
769a747e4fSDavid du Colombier {
779a747e4fSDavid du Colombier 	ctlprint(w->ctl, "name %s\n", s);
789a747e4fSDavid du Colombier }
799a747e4fSDavid du Colombier 
809a747e4fSDavid du Colombier void
winopenbody(Window * w,int mode)819a747e4fSDavid du Colombier winopenbody(Window *w, int mode)
829a747e4fSDavid du Colombier {
839a747e4fSDavid du Colombier 	char buf[256];
849a747e4fSDavid du Colombier 
859a747e4fSDavid du Colombier 	sprint(buf, "/mnt/wsys/%d/body", w->id);
869a747e4fSDavid du Colombier 	w->body = Bopen(buf, mode|OCEXEC);
879a747e4fSDavid du Colombier 	if(w->body == nil)
889a747e4fSDavid du Colombier 		error("can't open window body file: %r");
899a747e4fSDavid du Colombier }
909a747e4fSDavid du Colombier 
919a747e4fSDavid du Colombier void
winclosebody(Window * w)929a747e4fSDavid du Colombier winclosebody(Window *w)
939a747e4fSDavid du Colombier {
949a747e4fSDavid du Colombier 	if(w->body != nil){
959a747e4fSDavid du Colombier 		Bterm(w->body);
969a747e4fSDavid du Colombier 		w->body = nil;
979a747e4fSDavid du Colombier 	}
989a747e4fSDavid du Colombier }
999a747e4fSDavid du Colombier 
1009a747e4fSDavid du Colombier void
winwritebody(Window * w,char * s,int n)1019a747e4fSDavid du Colombier winwritebody(Window *w, char *s, int n)
1029a747e4fSDavid du Colombier {
1039a747e4fSDavid du Colombier 	if(w->body == nil)
1049a747e4fSDavid du Colombier 		winopenbody(w, OWRITE);
1059a747e4fSDavid du Colombier 	if(Bwrite(w->body, s, n) != n)
1069a747e4fSDavid du Colombier 		error("write error to window: %r");
1079a747e4fSDavid du Colombier }
1089a747e4fSDavid du Colombier 
1099a747e4fSDavid du Colombier int
wingetec(Window * w)1109a747e4fSDavid du Colombier wingetec(Window *w)
1119a747e4fSDavid du Colombier {
1129a747e4fSDavid du Colombier 	if(w->nbuf == 0){
1139a747e4fSDavid du Colombier 		w->nbuf = read(w->event, w->buf, sizeof w->buf);
1149a747e4fSDavid du Colombier 		if(w->nbuf <= 0){
1159a747e4fSDavid du Colombier 			/* probably because window has exited, and only called by wineventproc, so just shut down */
1169a747e4fSDavid du Colombier 			threadexits(nil);
1179a747e4fSDavid du Colombier 		}
1189a747e4fSDavid du Colombier 		w->bufp = w->buf;
1199a747e4fSDavid du Colombier 	}
1209a747e4fSDavid du Colombier 	w->nbuf--;
1219a747e4fSDavid du Colombier 	return *w->bufp++;
1229a747e4fSDavid du Colombier }
1239a747e4fSDavid du Colombier 
1249a747e4fSDavid du Colombier int
wingeten(Window * w)1259a747e4fSDavid du Colombier wingeten(Window *w)
1269a747e4fSDavid du Colombier {
1279a747e4fSDavid du Colombier 	int n, c;
1289a747e4fSDavid du Colombier 
1299a747e4fSDavid du Colombier 	n = 0;
1309a747e4fSDavid du Colombier 	while('0'<=(c=wingetec(w)) && c<='9')
1319a747e4fSDavid du Colombier 		n = n*10+(c-'0');
1329a747e4fSDavid du Colombier 	if(c != ' ')
1339a747e4fSDavid du Colombier 		error("event number syntax");
1349a747e4fSDavid du Colombier 	return n;
1359a747e4fSDavid du Colombier }
1369a747e4fSDavid du Colombier 
1379a747e4fSDavid du Colombier int
wingeter(Window * w,char * buf,int * nb)1389a747e4fSDavid du Colombier wingeter(Window *w, char *buf, int *nb)
1399a747e4fSDavid du Colombier {
1409a747e4fSDavid du Colombier 	Rune r;
1419a747e4fSDavid du Colombier 	int n;
1429a747e4fSDavid du Colombier 
1439a747e4fSDavid du Colombier 	r = wingetec(w);
1449a747e4fSDavid du Colombier 	buf[0] = r;
1459a747e4fSDavid du Colombier 	n = 1;
1469a747e4fSDavid du Colombier 	if(r >= Runeself) {
1479a747e4fSDavid du Colombier 		while(!fullrune(buf, n))
1489a747e4fSDavid du Colombier 			buf[n++] = wingetec(w);
1499a747e4fSDavid du Colombier 		chartorune(&r, buf);
1509a747e4fSDavid du Colombier 	}
1519a747e4fSDavid du Colombier 	*nb = n;
1529a747e4fSDavid du Colombier 	return r;
1539a747e4fSDavid du Colombier }
1549a747e4fSDavid du Colombier 
1559a747e4fSDavid du Colombier void
wingetevent(Window * w,Event * e)1569a747e4fSDavid du Colombier wingetevent(Window *w, Event *e)
1579a747e4fSDavid du Colombier {
1589a747e4fSDavid du Colombier 	int i, nb;
1599a747e4fSDavid du Colombier 
1609a747e4fSDavid du Colombier 	e->c1 = wingetec(w);
1619a747e4fSDavid du Colombier 	e->c2 = wingetec(w);
1629a747e4fSDavid du Colombier 	e->q0 = wingeten(w);
1639a747e4fSDavid du Colombier 	e->q1 = wingeten(w);
1649a747e4fSDavid du Colombier 	e->flag = wingeten(w);
1659a747e4fSDavid du Colombier 	e->nr = wingeten(w);
1669a747e4fSDavid du Colombier 	if(e->nr > EVENTSIZE)
1679a747e4fSDavid du Colombier 		error("event string too long");
1689a747e4fSDavid du Colombier 	e->nb = 0;
1699a747e4fSDavid du Colombier 	for(i=0; i<e->nr; i++){
1709a747e4fSDavid du Colombier 		e->r[i] = wingeter(w, e->b+e->nb, &nb);
1719a747e4fSDavid du Colombier 		e->nb += nb;
1729a747e4fSDavid du Colombier 	}
1739a747e4fSDavid du Colombier 	e->r[e->nr] = 0;
1749a747e4fSDavid du Colombier 	e->b[e->nb] = 0;
1759a747e4fSDavid du Colombier 	if(wingetec(w) != '\n')
1769a747e4fSDavid du Colombier 		error("event syntax error");
1779a747e4fSDavid du Colombier }
1789a747e4fSDavid du Colombier 
1799a747e4fSDavid du Colombier void
winwriteevent(Window * w,Event * e)1809a747e4fSDavid du Colombier winwriteevent(Window *w, Event *e)
1819a747e4fSDavid du Colombier {
182*3ff48bf5SDavid du Colombier 	fprint(w->event, "%c%c%d %d\n", e->c1, e->c2, e->q0, e->q1);
1839a747e4fSDavid du Colombier }
1849a747e4fSDavid du Colombier 
1859a747e4fSDavid du Colombier static int
nrunes(char * s,int nb)1869a747e4fSDavid du Colombier nrunes(char *s, int nb)
1879a747e4fSDavid du Colombier {
1889a747e4fSDavid du Colombier 	int i, n;
1899a747e4fSDavid du Colombier 	Rune r;
1909a747e4fSDavid du Colombier 
1919a747e4fSDavid du Colombier 	n = 0;
1929a747e4fSDavid du Colombier 	for(i=0; i<nb; n++)
1939a747e4fSDavid du Colombier 		i += chartorune(&r, s+i);
1949a747e4fSDavid du Colombier 	return n;
1959a747e4fSDavid du Colombier }
1969a747e4fSDavid du Colombier 
1979a747e4fSDavid du Colombier void
winread(Window * w,uint q0,uint q1,char * data)1989a747e4fSDavid du Colombier winread(Window *w, uint q0, uint q1, char *data)
1999a747e4fSDavid du Colombier {
2009a747e4fSDavid du Colombier 	int m, n, nr;
2019a747e4fSDavid du Colombier 	char buf[256];
2029a747e4fSDavid du Colombier 
2039a747e4fSDavid du Colombier 	if(w->addr < 0)
2049a747e4fSDavid du Colombier 		w->addr = winopenfile(w, "addr");
2059a747e4fSDavid du Colombier 	if(w->data < 0)
2069a747e4fSDavid du Colombier 		w->data = winopenfile(w, "data");
2079a747e4fSDavid du Colombier 	m = q0;
2089a747e4fSDavid du Colombier 	while(m < q1){
2099a747e4fSDavid du Colombier 		n = sprint(buf, "#%d", m);
2109a747e4fSDavid du Colombier 		if(write(w->addr, buf, n) != n)
2119a747e4fSDavid du Colombier 			error("error writing addr: %r");
2129a747e4fSDavid du Colombier 		n = read(w->data, buf, sizeof buf);
2139a747e4fSDavid du Colombier 		if(n <= 0)
2149a747e4fSDavid du Colombier 			error("reading data: %r");
2159a747e4fSDavid du Colombier 		nr = nrunes(buf, n);
2169a747e4fSDavid du Colombier 		while(m+nr >q1){
2179a747e4fSDavid du Colombier 			do; while(n>0 && (buf[--n]&0xC0)==0x80);
2189a747e4fSDavid du Colombier 			--nr;
2199a747e4fSDavid du Colombier 		}
2209a747e4fSDavid du Colombier 		if(n == 0)
2219a747e4fSDavid du Colombier 			break;
2229a747e4fSDavid du Colombier 		memmove(data, buf, n);
2239a747e4fSDavid du Colombier 		data += n;
2249a747e4fSDavid du Colombier 		*data = 0;
2259a747e4fSDavid du Colombier 		m += nr;
2269a747e4fSDavid du Colombier 	}
2279a747e4fSDavid du Colombier }
2289a747e4fSDavid du Colombier 
2299a747e4fSDavid du Colombier void
windormant(Window * w)2309a747e4fSDavid du Colombier windormant(Window *w)
2319a747e4fSDavid du Colombier {
2329a747e4fSDavid du Colombier 	if(w->addr >= 0){
2339a747e4fSDavid du Colombier 		close(w->addr);
2349a747e4fSDavid du Colombier 		w->addr = -1;
2359a747e4fSDavid du Colombier 	}
2369a747e4fSDavid du Colombier 	if(w->body != nil){
2379a747e4fSDavid du Colombier 		Bterm(w->body);
2389a747e4fSDavid du Colombier 		w->body = nil;
2399a747e4fSDavid du Colombier 	}
2409a747e4fSDavid du Colombier 	if(w->data >= 0){
2419a747e4fSDavid du Colombier 		close(w->data);
2429a747e4fSDavid du Colombier 		w->data = -1;
2439a747e4fSDavid du Colombier 	}
2449a747e4fSDavid du Colombier }
2459a747e4fSDavid du Colombier 
2469a747e4fSDavid du Colombier 
2479a747e4fSDavid du Colombier int
windel(Window * w,int sure)2489a747e4fSDavid du Colombier windel(Window *w, int sure)
2499a747e4fSDavid du Colombier {
2509a747e4fSDavid du Colombier 	if(sure)
2519a747e4fSDavid du Colombier 		write(w->ctl, "delete\n", 7);
2529a747e4fSDavid du Colombier 	else if(write(w->ctl, "del\n", 4) != 4)
2539a747e4fSDavid du Colombier 		return 0;
2549a747e4fSDavid du Colombier 	/* event proc will die due to read error from event file */
2559a747e4fSDavid du Colombier 	windormant(w);
2569a747e4fSDavid du Colombier 	close(w->ctl);
2579a747e4fSDavid du Colombier 	w->ctl = -1;
2589a747e4fSDavid du Colombier 	close(w->event);
2599a747e4fSDavid du Colombier 	w->event = -1;
2609a747e4fSDavid du Colombier 	return 1;
2619a747e4fSDavid du Colombier }
2629a747e4fSDavid du Colombier 
2639a747e4fSDavid du Colombier void
winclean(Window * w)2649a747e4fSDavid du Colombier winclean(Window *w)
2659a747e4fSDavid du Colombier {
2669a747e4fSDavid du Colombier 	if(w->body)
2679a747e4fSDavid du Colombier 		Bflush(w->body);
2689a747e4fSDavid du Colombier 	ctlprint(w->ctl, "clean\n");
2699a747e4fSDavid du Colombier }
2709a747e4fSDavid du Colombier 
2719a747e4fSDavid du Colombier int
winsetaddr(Window * w,char * addr,int errok)2729a747e4fSDavid du Colombier winsetaddr(Window *w, char *addr, int errok)
2739a747e4fSDavid du Colombier {
2749a747e4fSDavid du Colombier 	if(w->addr < 0)
2759a747e4fSDavid du Colombier 		w->addr = winopenfile(w, "addr");
2769a747e4fSDavid du Colombier 	if(write(w->addr, addr, strlen(addr)) < 0){
2779a747e4fSDavid du Colombier 		if(!errok)
2789a747e4fSDavid du Colombier 			error("error writing addr(%s): %r", addr);
2799a747e4fSDavid du Colombier 		return 0;
2809a747e4fSDavid du Colombier 	}
2819a747e4fSDavid du Colombier 	return 1;
2829a747e4fSDavid du Colombier }
2839a747e4fSDavid du Colombier 
2849a747e4fSDavid du Colombier int
winselect(Window * w,char * addr,int errok)2859a747e4fSDavid du Colombier winselect(Window *w, char *addr, int errok)
2869a747e4fSDavid du Colombier {
2879a747e4fSDavid du Colombier 	if(winsetaddr(w, addr, errok)){
2889a747e4fSDavid du Colombier 		ctlprint(w->ctl, "dot=addr\n");
2899a747e4fSDavid du Colombier 		return 1;
2909a747e4fSDavid du Colombier 	}
2919a747e4fSDavid du Colombier 	return 0;
2929a747e4fSDavid du Colombier }
2939a747e4fSDavid du Colombier 
2949a747e4fSDavid du Colombier char*
winreadbody(Window * w,int * np)2959a747e4fSDavid du Colombier winreadbody(Window *w, int *np)	/* can't use readfile because acme doesn't report the length */
2969a747e4fSDavid du Colombier {
2979a747e4fSDavid du Colombier 	char *s;
2989a747e4fSDavid du Colombier 	int m, na, n;
2999a747e4fSDavid du Colombier 
3009a747e4fSDavid du Colombier 	if(w->body != nil)
3019a747e4fSDavid du Colombier 		winclosebody(w);
3029a747e4fSDavid du Colombier 	winopenbody(w, OREAD);
3039a747e4fSDavid du Colombier 	s = nil;
3049a747e4fSDavid du Colombier 	na = 0;
3059a747e4fSDavid du Colombier 	n = 0;
3069a747e4fSDavid du Colombier 	for(;;){
3079a747e4fSDavid du Colombier 		if(na < n+512){
3089a747e4fSDavid du Colombier 			na += 1024;
3099a747e4fSDavid du Colombier 			s = realloc(s, na+1);
3109a747e4fSDavid du Colombier 		}
3119a747e4fSDavid du Colombier 		m = Bread(w->body, s+n, na-n);
3129a747e4fSDavid du Colombier 		if(m <= 0)
3139a747e4fSDavid du Colombier 			break;
3149a747e4fSDavid du Colombier 		n += m;
3159a747e4fSDavid du Colombier 	}
3169a747e4fSDavid du Colombier 	s[n] = 0;
3179a747e4fSDavid du Colombier 	winclosebody(w);
3189a747e4fSDavid du Colombier 	*np = n;
3199a747e4fSDavid du Colombier 	return s;
3209a747e4fSDavid du Colombier }
321