17dd7cddfSDavid du Colombier #include <u.h>
27dd7cddfSDavid du Colombier #include <libc.h>
37dd7cddfSDavid du Colombier #include <bio.h>
47dd7cddfSDavid du Colombier #include <thread.h>
55d459b5aSDavid du Colombier #include <plumb.h>
67dd7cddfSDavid du Colombier #include "dat.h"
77dd7cddfSDavid du Colombier
87dd7cddfSDavid du Colombier Window*
newwindow(void)97dd7cddfSDavid du Colombier newwindow(void)
107dd7cddfSDavid du Colombier {
117dd7cddfSDavid du Colombier char buf[12];
127dd7cddfSDavid du Colombier Window *w;
137dd7cddfSDavid du Colombier
147dd7cddfSDavid du Colombier w = emalloc(sizeof(Window));
157dd7cddfSDavid du Colombier w->ctl = open("/mnt/wsys/new/ctl", ORDWR|OCEXEC);
167dd7cddfSDavid du Colombier if(w->ctl<0 || read(w->ctl, buf, 12)!=12)
177dd7cddfSDavid du Colombier error("can't open window ctl file: %r");
187dd7cddfSDavid du Colombier ctlprint(w->ctl, "noscroll\n");
197dd7cddfSDavid du Colombier w->id = atoi(buf);
207dd7cddfSDavid du Colombier w->event = winopenfile(w, "event");
217dd7cddfSDavid du Colombier w->addr = -1; /* will be opened when needed */
227dd7cddfSDavid du Colombier w->body = nil;
237dd7cddfSDavid du Colombier w->data = -1;
247dd7cddfSDavid du Colombier w->cevent = chancreate(sizeof(Event*), 0);
257dd7cddfSDavid du Colombier return w;
267dd7cddfSDavid du Colombier }
277dd7cddfSDavid du Colombier
287dd7cddfSDavid du Colombier void
winsetdump(Window * w,char * dir,char * cmd)297dd7cddfSDavid du Colombier winsetdump(Window *w, char *dir, char *cmd)
307dd7cddfSDavid du Colombier {
317dd7cddfSDavid du Colombier if(dir != nil)
327dd7cddfSDavid du Colombier ctlprint(w->ctl, "dumpdir %s\n", dir);
337dd7cddfSDavid du Colombier if(cmd != nil)
347dd7cddfSDavid du Colombier ctlprint(w->ctl, "dump %s\n", cmd);
357dd7cddfSDavid du Colombier }
367dd7cddfSDavid du Colombier
377dd7cddfSDavid du Colombier void
wineventproc(void * v)387dd7cddfSDavid du Colombier wineventproc(void *v)
397dd7cddfSDavid du Colombier {
407dd7cddfSDavid du Colombier Window *w;
417dd7cddfSDavid du Colombier int i;
427dd7cddfSDavid du Colombier
437dd7cddfSDavid du Colombier w = v;
447dd7cddfSDavid du Colombier for(i=0; ; i++){
457dd7cddfSDavid du Colombier if(i >= NEVENT)
467dd7cddfSDavid du Colombier i = 0;
477dd7cddfSDavid du Colombier wingetevent(w, &w->e[i]);
487dd7cddfSDavid du Colombier sendp(w->cevent, &w->e[i]);
497dd7cddfSDavid du Colombier }
507dd7cddfSDavid du Colombier }
517dd7cddfSDavid du Colombier
529a747e4fSDavid du Colombier static int
winopenfile1(Window * w,char * f,int m)539a747e4fSDavid du Colombier winopenfile1(Window *w, char *f, int m)
547dd7cddfSDavid du Colombier {
557dd7cddfSDavid du Colombier char buf[64];
567dd7cddfSDavid du Colombier int fd;
577dd7cddfSDavid du Colombier
587dd7cddfSDavid du Colombier sprint(buf, "/mnt/wsys/%d/%s", w->id, f);
599a747e4fSDavid du Colombier fd = open(buf, m|OCEXEC);
607dd7cddfSDavid du Colombier if(fd < 0)
617dd7cddfSDavid du Colombier error("can't open window file %s: %r", f);
627dd7cddfSDavid du Colombier return fd;
637dd7cddfSDavid du Colombier }
647dd7cddfSDavid du Colombier
659a747e4fSDavid du Colombier int
winopenfile(Window * w,char * f)669a747e4fSDavid du Colombier winopenfile(Window *w, char *f)
679a747e4fSDavid du Colombier {
689a747e4fSDavid du Colombier return winopenfile1(w, f, ORDWR);
699a747e4fSDavid du Colombier }
709a747e4fSDavid du Colombier
717dd7cddfSDavid du Colombier void
wintagwrite(Window * w,char * s,int n)727dd7cddfSDavid du Colombier wintagwrite(Window *w, char *s, int n)
737dd7cddfSDavid du Colombier {
747dd7cddfSDavid du Colombier int fd;
757dd7cddfSDavid du Colombier
767dd7cddfSDavid du Colombier fd = winopenfile(w, "tag");
777dd7cddfSDavid du Colombier if(write(fd, s, n) != n)
787dd7cddfSDavid du Colombier error("tag write: %r");
797dd7cddfSDavid du Colombier close(fd);
807dd7cddfSDavid du Colombier }
817dd7cddfSDavid du Colombier
827dd7cddfSDavid du Colombier void
winname(Window * w,char * s)837dd7cddfSDavid du Colombier winname(Window *w, char *s)
847dd7cddfSDavid du Colombier {
857dd7cddfSDavid du Colombier ctlprint(w->ctl, "name %s\n", s);
867dd7cddfSDavid du Colombier }
877dd7cddfSDavid du Colombier
887dd7cddfSDavid du Colombier void
winopenbody(Window * w,int mode)897dd7cddfSDavid du Colombier winopenbody(Window *w, int mode)
907dd7cddfSDavid du Colombier {
917dd7cddfSDavid du Colombier char buf[256];
927dd7cddfSDavid du Colombier
937dd7cddfSDavid du Colombier sprint(buf, "/mnt/wsys/%d/body", w->id);
947dd7cddfSDavid du Colombier w->body = Bopen(buf, mode|OCEXEC);
957dd7cddfSDavid du Colombier if(w->body == nil)
967dd7cddfSDavid du Colombier error("can't open window body file: %r");
977dd7cddfSDavid du Colombier }
987dd7cddfSDavid du Colombier
997dd7cddfSDavid du Colombier void
winclosebody(Window * w)1007dd7cddfSDavid du Colombier winclosebody(Window *w)
1017dd7cddfSDavid du Colombier {
1027dd7cddfSDavid du Colombier if(w->body != nil){
1037dd7cddfSDavid du Colombier Bterm(w->body);
1047dd7cddfSDavid du Colombier w->body = nil;
1057dd7cddfSDavid du Colombier }
1067dd7cddfSDavid du Colombier }
1077dd7cddfSDavid du Colombier
1087dd7cddfSDavid du Colombier void
winwritebody(Window * w,char * s,int n)1097dd7cddfSDavid du Colombier winwritebody(Window *w, char *s, int n)
1107dd7cddfSDavid du Colombier {
1117dd7cddfSDavid du Colombier if(w->body == nil)
1127dd7cddfSDavid du Colombier winopenbody(w, OWRITE);
1137dd7cddfSDavid du Colombier if(Bwrite(w->body, s, n) != n)
1147dd7cddfSDavid du Colombier error("write error to window: %r");
1157dd7cddfSDavid du Colombier }
1167dd7cddfSDavid du Colombier
1177dd7cddfSDavid du Colombier int
wingetec(Window * w)1187dd7cddfSDavid du Colombier wingetec(Window *w)
1197dd7cddfSDavid du Colombier {
1207dd7cddfSDavid du Colombier if(w->nbuf == 0){
1217dd7cddfSDavid du Colombier w->nbuf = read(w->event, w->buf, sizeof w->buf);
1227dd7cddfSDavid du Colombier if(w->nbuf <= 0){
1237dd7cddfSDavid du Colombier /* probably because window has exited, and only called by wineventproc, so just shut down */
1247dd7cddfSDavid du Colombier threadexits(nil);
1257dd7cddfSDavid du Colombier }
1267dd7cddfSDavid du Colombier w->bufp = w->buf;
1277dd7cddfSDavid du Colombier }
1287dd7cddfSDavid du Colombier w->nbuf--;
1297dd7cddfSDavid du Colombier return *w->bufp++;
1307dd7cddfSDavid du Colombier }
1317dd7cddfSDavid du Colombier
1327dd7cddfSDavid du Colombier int
wingeten(Window * w)1337dd7cddfSDavid du Colombier wingeten(Window *w)
1347dd7cddfSDavid du Colombier {
1357dd7cddfSDavid du Colombier int n, c;
1367dd7cddfSDavid du Colombier
1377dd7cddfSDavid du Colombier n = 0;
1387dd7cddfSDavid du Colombier while('0'<=(c=wingetec(w)) && c<='9')
1397dd7cddfSDavid du Colombier n = n*10+(c-'0');
1407dd7cddfSDavid du Colombier if(c != ' ')
1417dd7cddfSDavid du Colombier error("event number syntax");
1427dd7cddfSDavid du Colombier return n;
1437dd7cddfSDavid du Colombier }
1447dd7cddfSDavid du Colombier
1457dd7cddfSDavid du Colombier int
wingeter(Window * w,char * buf,int * nb)1467dd7cddfSDavid du Colombier wingeter(Window *w, char *buf, int *nb)
1477dd7cddfSDavid du Colombier {
1487dd7cddfSDavid du Colombier Rune r;
1497dd7cddfSDavid du Colombier int n;
1507dd7cddfSDavid du Colombier
1517dd7cddfSDavid du Colombier r = wingetec(w);
1527dd7cddfSDavid du Colombier buf[0] = r;
1537dd7cddfSDavid du Colombier n = 1;
1547dd7cddfSDavid du Colombier if(r >= Runeself) {
1557dd7cddfSDavid du Colombier while(!fullrune(buf, n))
1567dd7cddfSDavid du Colombier buf[n++] = wingetec(w);
1577dd7cddfSDavid du Colombier chartorune(&r, buf);
1587dd7cddfSDavid du Colombier }
1597dd7cddfSDavid du Colombier *nb = n;
1607dd7cddfSDavid du Colombier return r;
1617dd7cddfSDavid du Colombier }
1627dd7cddfSDavid du Colombier
1637dd7cddfSDavid du Colombier void
wingetevent(Window * w,Event * e)1647dd7cddfSDavid du Colombier wingetevent(Window *w, Event *e)
1657dd7cddfSDavid du Colombier {
1667dd7cddfSDavid du Colombier int i, nb;
1677dd7cddfSDavid du Colombier
1687dd7cddfSDavid du Colombier e->c1 = wingetec(w);
1697dd7cddfSDavid du Colombier e->c2 = wingetec(w);
1707dd7cddfSDavid du Colombier e->q0 = wingeten(w);
1717dd7cddfSDavid du Colombier e->q1 = wingeten(w);
1727dd7cddfSDavid du Colombier e->flag = wingeten(w);
1737dd7cddfSDavid du Colombier e->nr = wingeten(w);
1747dd7cddfSDavid du Colombier if(e->nr > EVENTSIZE)
1757dd7cddfSDavid du Colombier error("event string too long");
1767dd7cddfSDavid du Colombier e->nb = 0;
1777dd7cddfSDavid du Colombier for(i=0; i<e->nr; i++){
1787dd7cddfSDavid du Colombier e->r[i] = wingeter(w, e->b+e->nb, &nb);
1797dd7cddfSDavid du Colombier e->nb += nb;
1807dd7cddfSDavid du Colombier }
1817dd7cddfSDavid du Colombier e->r[e->nr] = 0;
1827dd7cddfSDavid du Colombier e->b[e->nb] = 0;
1837dd7cddfSDavid du Colombier if(wingetec(w) != '\n')
1847dd7cddfSDavid du Colombier error("event syntax error");
1857dd7cddfSDavid du Colombier }
1867dd7cddfSDavid du Colombier
1877dd7cddfSDavid du Colombier void
winwriteevent(Window * w,Event * e)1887dd7cddfSDavid du Colombier winwriteevent(Window *w, Event *e)
1897dd7cddfSDavid du Colombier {
1909a747e4fSDavid du Colombier fprint(w->event, "%c%c%d %d\n", e->c1, e->c2, e->q0, e->q1);
1917dd7cddfSDavid du Colombier }
1927dd7cddfSDavid du Colombier
1937dd7cddfSDavid du Colombier void
winread(Window * w,uint q0,uint q1,char * data)1947dd7cddfSDavid du Colombier winread(Window *w, uint q0, uint q1, char *data)
1957dd7cddfSDavid du Colombier {
1967dd7cddfSDavid du Colombier int m, n, nr;
1977dd7cddfSDavid du Colombier char buf[256];
1987dd7cddfSDavid du Colombier
1997dd7cddfSDavid du Colombier if(w->addr < 0)
2007dd7cddfSDavid du Colombier w->addr = winopenfile(w, "addr");
2017dd7cddfSDavid du Colombier if(w->data < 0)
2027dd7cddfSDavid du Colombier w->data = winopenfile(w, "data");
2037dd7cddfSDavid du Colombier m = q0;
2047dd7cddfSDavid du Colombier while(m < q1){
2057dd7cddfSDavid du Colombier n = sprint(buf, "#%d", m);
2067dd7cddfSDavid du Colombier if(write(w->addr, buf, n) != n)
2077dd7cddfSDavid du Colombier error("error writing addr: %r");
2087dd7cddfSDavid du Colombier n = read(w->data, buf, sizeof buf);
2097dd7cddfSDavid du Colombier if(n <= 0)
2107dd7cddfSDavid du Colombier error("reading data: %r");
2119a747e4fSDavid du Colombier nr = utfnlen(buf, n);
2127dd7cddfSDavid du Colombier while(m+nr >q1){
2137dd7cddfSDavid du Colombier do; while(n>0 && (buf[--n]&0xC0)==0x80);
2147dd7cddfSDavid du Colombier --nr;
2157dd7cddfSDavid du Colombier }
2167dd7cddfSDavid du Colombier if(n == 0)
2177dd7cddfSDavid du Colombier break;
2187dd7cddfSDavid du Colombier memmove(data, buf, n);
2197dd7cddfSDavid du Colombier data += n;
2207dd7cddfSDavid du Colombier *data = 0;
2217dd7cddfSDavid du Colombier m += nr;
2227dd7cddfSDavid du Colombier }
2237dd7cddfSDavid du Colombier }
2247dd7cddfSDavid du Colombier
2257dd7cddfSDavid du Colombier void
windormant(Window * w)2267dd7cddfSDavid du Colombier windormant(Window *w)
2277dd7cddfSDavid du Colombier {
2287dd7cddfSDavid du Colombier if(w->addr >= 0){
2297dd7cddfSDavid du Colombier close(w->addr);
2307dd7cddfSDavid du Colombier w->addr = -1;
2317dd7cddfSDavid du Colombier }
2327dd7cddfSDavid du Colombier if(w->body != nil){
2337dd7cddfSDavid du Colombier Bterm(w->body);
2347dd7cddfSDavid du Colombier w->body = nil;
2357dd7cddfSDavid du Colombier }
2367dd7cddfSDavid du Colombier if(w->data >= 0){
2377dd7cddfSDavid du Colombier close(w->data);
2387dd7cddfSDavid du Colombier w->data = -1;
2397dd7cddfSDavid du Colombier }
2407dd7cddfSDavid du Colombier }
2417dd7cddfSDavid du Colombier
2427dd7cddfSDavid du Colombier
2437dd7cddfSDavid du Colombier int
windel(Window * w,int sure)2447dd7cddfSDavid du Colombier windel(Window *w, int sure)
2457dd7cddfSDavid du Colombier {
2467dd7cddfSDavid du Colombier if(sure)
2477dd7cddfSDavid du Colombier write(w->ctl, "delete\n", 7);
2487dd7cddfSDavid du Colombier else if(write(w->ctl, "del\n", 4) != 4)
2497dd7cddfSDavid du Colombier return 0;
2507dd7cddfSDavid du Colombier /* event proc will die due to read error from event file */
2517dd7cddfSDavid du Colombier windormant(w);
2527dd7cddfSDavid du Colombier close(w->ctl);
2537dd7cddfSDavid du Colombier w->ctl = -1;
2547dd7cddfSDavid du Colombier close(w->event);
2557dd7cddfSDavid du Colombier w->event = -1;
2567dd7cddfSDavid du Colombier return 1;
2577dd7cddfSDavid du Colombier }
2587dd7cddfSDavid du Colombier
2597dd7cddfSDavid du Colombier void
winclean(Window * w)2607dd7cddfSDavid du Colombier winclean(Window *w)
2617dd7cddfSDavid du Colombier {
2627dd7cddfSDavid du Colombier if(w->body)
2637dd7cddfSDavid du Colombier Bflush(w->body);
2647dd7cddfSDavid du Colombier ctlprint(w->ctl, "clean\n");
2657dd7cddfSDavid du Colombier }
2667dd7cddfSDavid du Colombier
2677dd7cddfSDavid du Colombier int
winsetaddr(Window * w,char * addr,int errok)2687dd7cddfSDavid du Colombier winsetaddr(Window *w, char *addr, int errok)
2697dd7cddfSDavid du Colombier {
2707dd7cddfSDavid du Colombier if(w->addr < 0)
2717dd7cddfSDavid du Colombier w->addr = winopenfile(w, "addr");
2727dd7cddfSDavid du Colombier if(write(w->addr, addr, strlen(addr)) < 0){
2737dd7cddfSDavid du Colombier if(!errok)
2747dd7cddfSDavid du Colombier error("error writing addr(%s): %r", addr);
2757dd7cddfSDavid du Colombier return 0;
2767dd7cddfSDavid du Colombier }
2777dd7cddfSDavid du Colombier return 1;
2787dd7cddfSDavid du Colombier }
2797dd7cddfSDavid du Colombier
2807dd7cddfSDavid du Colombier int
winselect(Window * w,char * addr,int errok)2817dd7cddfSDavid du Colombier winselect(Window *w, char *addr, int errok)
2827dd7cddfSDavid du Colombier {
2837dd7cddfSDavid du Colombier if(winsetaddr(w, addr, errok)){
2847dd7cddfSDavid du Colombier ctlprint(w->ctl, "dot=addr\n");
2857dd7cddfSDavid du Colombier return 1;
2867dd7cddfSDavid du Colombier }
2877dd7cddfSDavid du Colombier return 0;
2887dd7cddfSDavid du Colombier }
2897dd7cddfSDavid du Colombier
2907dd7cddfSDavid du Colombier char*
winreadbody(Window * w,int * np)2917dd7cddfSDavid du Colombier winreadbody(Window *w, int *np) /* can't use readfile because acme doesn't report the length */
2927dd7cddfSDavid du Colombier {
2937dd7cddfSDavid du Colombier char *s;
2947dd7cddfSDavid du Colombier int m, na, n;
2957dd7cddfSDavid du Colombier
2967dd7cddfSDavid du Colombier if(w->body != nil)
2977dd7cddfSDavid du Colombier winclosebody(w);
2987dd7cddfSDavid du Colombier winopenbody(w, OREAD);
2997dd7cddfSDavid du Colombier s = nil;
3007dd7cddfSDavid du Colombier na = 0;
3017dd7cddfSDavid du Colombier n = 0;
3027dd7cddfSDavid du Colombier for(;;){
3037dd7cddfSDavid du Colombier if(na < n+512){
3047dd7cddfSDavid du Colombier na += 1024;
3057dd7cddfSDavid du Colombier s = realloc(s, na+1);
3067dd7cddfSDavid du Colombier }
3077dd7cddfSDavid du Colombier m = Bread(w->body, s+n, na-n);
3087dd7cddfSDavid du Colombier if(m <= 0)
3097dd7cddfSDavid du Colombier break;
3107dd7cddfSDavid du Colombier n += m;
3117dd7cddfSDavid du Colombier }
3127dd7cddfSDavid du Colombier s[n] = 0;
3137dd7cddfSDavid du Colombier winclosebody(w);
3147dd7cddfSDavid du Colombier *np = n;
3157dd7cddfSDavid du Colombier return s;
3167dd7cddfSDavid du Colombier }
3179a747e4fSDavid du Colombier
3189a747e4fSDavid du Colombier char*
winselection(Window * w)3199a747e4fSDavid du Colombier winselection(Window *w)
3209a747e4fSDavid du Colombier {
3219a747e4fSDavid du Colombier int fd, m, n;
3229a747e4fSDavid du Colombier char *buf;
323*d9306527SDavid du Colombier char tmp[256];
3249a747e4fSDavid du Colombier
3259a747e4fSDavid du Colombier fd = winopenfile1(w, "rdsel", OREAD);
3269a747e4fSDavid du Colombier if(fd < 0)
3279a747e4fSDavid du Colombier error("can't open rdsel: %r");
3289a747e4fSDavid du Colombier n = 0;
3299a747e4fSDavid du Colombier buf = nil;
3309a747e4fSDavid du Colombier for(;;){
331*d9306527SDavid du Colombier m = read(fd, tmp, sizeof tmp);
3329a747e4fSDavid du Colombier if(m <= 0)
3339a747e4fSDavid du Colombier break;
334*d9306527SDavid du Colombier buf = erealloc(buf, n+m+1);
335*d9306527SDavid du Colombier memmove(buf+n, tmp, m);
336*d9306527SDavid du Colombier n += m;
3379a747e4fSDavid du Colombier buf[n] = '\0';
3389a747e4fSDavid du Colombier }
3399a747e4fSDavid du Colombier close(fd);
3409a747e4fSDavid du Colombier return buf;
3419a747e4fSDavid du Colombier }
342