Blame


1 b3994ec5 2003-12-11 devnull #include <u.h>
2 b3994ec5 2003-12-11 devnull #include <libc.h>
3 b3994ec5 2003-12-11 devnull #include <draw.h>
4 b3994ec5 2003-12-11 devnull #include <thread.h>
5 b3994ec5 2003-12-11 devnull #include <cursor.h>
6 b3994ec5 2003-12-11 devnull #include <mouse.h>
7 b3994ec5 2003-12-11 devnull #include <keyboard.h>
8 b3994ec5 2003-12-11 devnull #include <frame.h>
9 b3994ec5 2003-12-11 devnull #include <fcall.h>
10 b3994ec5 2003-12-11 devnull #include <plumb.h>
11 b3994ec5 2003-12-11 devnull #include "dat.h"
12 b3994ec5 2003-12-11 devnull #include "fns.h"
13 b3994ec5 2003-12-11 devnull /* for generating syms in mkfile only: */
14 b3994ec5 2003-12-11 devnull #include <bio.h>
15 b3994ec5 2003-12-11 devnull #include "edit.h"
16 b3994ec5 2003-12-11 devnull
17 b3994ec5 2003-12-11 devnull void mousethread(void*);
18 b3994ec5 2003-12-11 devnull void keyboardthread(void*);
19 b3994ec5 2003-12-11 devnull void waitthread(void*);
20 b3994ec5 2003-12-11 devnull void xfidallocthread(void*);
21 b3994ec5 2003-12-11 devnull void newwindowthread(void*);
22 b3994ec5 2003-12-11 devnull void plumbproc(void*);
23 b3994ec5 2003-12-11 devnull
24 b3994ec5 2003-12-11 devnull Reffont **fontcache;
25 b3994ec5 2003-12-11 devnull int nfontcache;
26 b3994ec5 2003-12-11 devnull char wdir[512] = ".";
27 b3994ec5 2003-12-11 devnull Reffont *reffonts[2];
28 b3994ec5 2003-12-11 devnull int snarffd = -1;
29 b3994ec5 2003-12-11 devnull int mainpid;
30 d8c78a8d 2005-01-10 devnull int swapscrollbuttons = FALSE;
31 cd5a7378 2008-04-17 rsc char *mtpt;
32 b3994ec5 2003-12-11 devnull
33 b3994ec5 2003-12-11 devnull enum{
34 b3994ec5 2003-12-11 devnull NSnarf = 1000 /* less than 1024, I/O buffer size */
35 b3994ec5 2003-12-11 devnull };
36 b3994ec5 2003-12-11 devnull Rune snarfrune[NSnarf+1];
37 b3994ec5 2003-12-11 devnull
38 b3994ec5 2003-12-11 devnull char *fontnames[2] =
39 b3994ec5 2003-12-11 devnull {
40 4590ad02 2006-03-19 devnull "/lib/font/bit/lucsans/euro.8.font",
41 cbeb0b26 2006-04-01 devnull "/lib/font/bit/lucm/unicode.9.font"
42 b3994ec5 2003-12-11 devnull };
43 b3994ec5 2003-12-11 devnull
44 b3994ec5 2003-12-11 devnull Command *command;
45 b3994ec5 2003-12-11 devnull
46 334cb1e9 2004-12-27 devnull void shutdownthread(void*);
47 b3994ec5 2003-12-11 devnull void acmeerrorinit(void);
48 b3994ec5 2003-12-11 devnull void readfile(Column*, char*);
49 4da83e7c 2004-03-05 devnull static int shutdown(void*, char*);
50 b3994ec5 2003-12-11 devnull
51 b3994ec5 2003-12-11 devnull void
52 b3994ec5 2003-12-11 devnull derror(Display *d, char *errorstr)
53 b3994ec5 2003-12-11 devnull {
54 b3994ec5 2003-12-11 devnull USED(d);
55 b3994ec5 2003-12-11 devnull error(errorstr);
56 b3994ec5 2003-12-11 devnull }
57 b3994ec5 2003-12-11 devnull
58 b3994ec5 2003-12-11 devnull void
59 b3994ec5 2003-12-11 devnull threadmain(int argc, char *argv[])
60 b3994ec5 2003-12-11 devnull {
61 b3994ec5 2003-12-11 devnull int i;
62 b3994ec5 2003-12-11 devnull char *p, *loadfile;
63 b3994ec5 2003-12-11 devnull Column *c;
64 b3994ec5 2003-12-11 devnull int ncol;
65 b3994ec5 2003-12-11 devnull Display *d;
66 b3994ec5 2003-12-11 devnull
67 b3994ec5 2003-12-11 devnull rfork(RFENVG|RFNAMEG);
68 b3994ec5 2003-12-11 devnull
69 b3994ec5 2003-12-11 devnull ncol = -1;
70 b3994ec5 2003-12-11 devnull
71 b3994ec5 2003-12-11 devnull loadfile = nil;
72 b3994ec5 2003-12-11 devnull ARGBEGIN{
73 012a8a02 2004-10-22 devnull case 'D':
74 012a8a02 2004-10-22 devnull {extern int _threaddebuglevel;
75 012a8a02 2004-10-22 devnull _threaddebuglevel = ~0;
76 012a8a02 2004-10-22 devnull }
77 012a8a02 2004-10-22 devnull break;
78 5a8e63b2 2004-02-29 devnull case 'a':
79 5a8e63b2 2004-02-29 devnull globalautoindent = TRUE;
80 5a8e63b2 2004-02-29 devnull break;
81 b3994ec5 2003-12-11 devnull case 'b':
82 b3994ec5 2003-12-11 devnull bartflag = TRUE;
83 b3994ec5 2003-12-11 devnull break;
84 b3994ec5 2003-12-11 devnull case 'c':
85 b3994ec5 2003-12-11 devnull p = ARGF();
86 b3994ec5 2003-12-11 devnull if(p == nil)
87 b3994ec5 2003-12-11 devnull goto Usage;
88 b3994ec5 2003-12-11 devnull ncol = atoi(p);
89 b3994ec5 2003-12-11 devnull if(ncol <= 0)
90 b3994ec5 2003-12-11 devnull goto Usage;
91 b3994ec5 2003-12-11 devnull break;
92 b3994ec5 2003-12-11 devnull case 'f':
93 b3994ec5 2003-12-11 devnull fontnames[0] = ARGF();
94 b3994ec5 2003-12-11 devnull if(fontnames[0] == nil)
95 b3994ec5 2003-12-11 devnull goto Usage;
96 b3994ec5 2003-12-11 devnull break;
97 b3994ec5 2003-12-11 devnull case 'F':
98 b3994ec5 2003-12-11 devnull fontnames[1] = ARGF();
99 b3994ec5 2003-12-11 devnull if(fontnames[1] == nil)
100 b3994ec5 2003-12-11 devnull goto Usage;
101 b3994ec5 2003-12-11 devnull break;
102 b3994ec5 2003-12-11 devnull case 'l':
103 b3994ec5 2003-12-11 devnull loadfile = ARGF();
104 b3994ec5 2003-12-11 devnull if(loadfile == nil)
105 cd5a7378 2008-04-17 rsc goto Usage;
106 cd5a7378 2008-04-17 rsc break;
107 cd5a7378 2008-04-17 rsc case 'm':
108 cd5a7378 2008-04-17 rsc mtpt = ARGF();
109 cd5a7378 2008-04-17 rsc if(mtpt == nil)
110 b3994ec5 2003-12-11 devnull goto Usage;
111 b3994ec5 2003-12-11 devnull break;
112 d8c78a8d 2005-01-10 devnull case 'r':
113 d8c78a8d 2005-01-10 devnull swapscrollbuttons = TRUE;
114 d8c78a8d 2005-01-10 devnull break;
115 be36ff68 2004-04-29 devnull case 'W':
116 be36ff68 2004-04-29 devnull winsize = ARGF();
117 be36ff68 2004-04-29 devnull if(winsize == nil)
118 be36ff68 2004-04-29 devnull goto Usage;
119 be36ff68 2004-04-29 devnull break;
120 b3994ec5 2003-12-11 devnull default:
121 b3994ec5 2003-12-11 devnull Usage:
122 be36ff68 2004-04-29 devnull fprint(2, "usage: acme -a -c ncol -f fontname -F fixedwidthfontname -l loadfile -W winsize\n");
123 38c10d1a 2005-01-17 devnull threadexitsall("usage");
124 b3994ec5 2003-12-11 devnull }ARGEND
125 b3994ec5 2003-12-11 devnull
126 6d7fdb24 2004-12-27 devnull fontnames[0] = estrdup(fontnames[0]);
127 6d7fdb24 2004-12-27 devnull fontnames[1] = estrdup(fontnames[1]);
128 6d7fdb24 2004-12-27 devnull
129 c591e1e6 2004-04-25 devnull quotefmtinstall();
130 b3994ec5 2003-12-11 devnull cputype = getenv("cputype");
131 b3994ec5 2003-12-11 devnull objtype = getenv("objtype");
132 c8b6342d 2005-01-13 devnull home = getenv("HOME");
133 b3994ec5 2003-12-11 devnull p = getenv("tabstop");
134 b3994ec5 2003-12-11 devnull if(p != nil){
135 b3994ec5 2003-12-11 devnull maxtab = strtoul(p, nil, 0);
136 b3994ec5 2003-12-11 devnull free(p);
137 b3994ec5 2003-12-11 devnull }
138 b3994ec5 2003-12-11 devnull if(maxtab == 0)
139 b3994ec5 2003-12-11 devnull maxtab = 4;
140 b3994ec5 2003-12-11 devnull if(loadfile)
141 b3994ec5 2003-12-11 devnull rowloadfonts(loadfile);
142 b3994ec5 2003-12-11 devnull putenv("font", fontnames[0]);
143 b3994ec5 2003-12-11 devnull snarffd = open("/dev/snarf", OREAD|OCEXEC);
144 b3994ec5 2003-12-11 devnull /*
145 b3994ec5 2003-12-11 devnull if(cputype){
146 b3994ec5 2003-12-11 devnull sprint(buf, "/acme/bin/%s", cputype);
147 b3994ec5 2003-12-11 devnull bind(buf, "/bin", MBEFORE);
148 b3994ec5 2003-12-11 devnull }
149 b3994ec5 2003-12-11 devnull bind("/acme/bin", "/bin", MBEFORE);
150 b3994ec5 2003-12-11 devnull */
151 b3994ec5 2003-12-11 devnull getwd(wdir, sizeof wdir);
152 b3994ec5 2003-12-11 devnull
153 b3994ec5 2003-12-11 devnull /*
154 b3994ec5 2003-12-11 devnull if(geninitdraw(nil, derror, fontnames[0], "acme", nil, Refnone) < 0){
155 b3994ec5 2003-12-11 devnull fprint(2, "acme: can't open display: %r\n");
156 38c10d1a 2005-01-17 devnull threadexitsall("geninitdraw");
157 b3994ec5 2003-12-11 devnull }
158 b3994ec5 2003-12-11 devnull */
159 b3994ec5 2003-12-11 devnull if(initdraw(derror, fontnames[0], "acme") < 0){
160 b3994ec5 2003-12-11 devnull fprint(2, "acme: can't open display: %r\n");
161 38c10d1a 2005-01-17 devnull threadexitsall("initdraw");
162 b3994ec5 2003-12-11 devnull }
163 b3994ec5 2003-12-11 devnull
164 b3994ec5 2003-12-11 devnull d = display;
165 b3994ec5 2003-12-11 devnull font = d->defaultfont;
166 cbeb0b26 2006-04-01 devnull /*assert(font); */
167 b3994ec5 2003-12-11 devnull
168 b3994ec5 2003-12-11 devnull reffont.f = font;
169 b3994ec5 2003-12-11 devnull reffonts[0] = &reffont;
170 b3994ec5 2003-12-11 devnull incref(&reffont.ref); /* one to hold up 'font' variable */
171 b3994ec5 2003-12-11 devnull incref(&reffont.ref); /* one to hold up reffonts[0] */
172 b3994ec5 2003-12-11 devnull fontcache = emalloc(sizeof(Reffont*));
173 b3994ec5 2003-12-11 devnull nfontcache = 1;
174 b3994ec5 2003-12-11 devnull fontcache[0] = &reffont;
175 b3994ec5 2003-12-11 devnull
176 b3994ec5 2003-12-11 devnull iconinit();
177 b3994ec5 2003-12-11 devnull timerinit();
178 b3994ec5 2003-12-11 devnull rxinit();
179 b3994ec5 2003-12-11 devnull
180 b3994ec5 2003-12-11 devnull cwait = threadwaitchan();
181 b3994ec5 2003-12-11 devnull ccommand = chancreate(sizeof(Command**), 0);
182 b3994ec5 2003-12-11 devnull ckill = chancreate(sizeof(Rune*), 0);
183 b3994ec5 2003-12-11 devnull cxfidalloc = chancreate(sizeof(Xfid*), 0);
184 b3994ec5 2003-12-11 devnull cxfidfree = chancreate(sizeof(Xfid*), 0);
185 b3994ec5 2003-12-11 devnull cnewwindow = chancreate(sizeof(Channel*), 0);
186 b3994ec5 2003-12-11 devnull cerr = chancreate(sizeof(char*), 0);
187 b3994ec5 2003-12-11 devnull cedit = chancreate(sizeof(int), 0);
188 b3994ec5 2003-12-11 devnull cexit = chancreate(sizeof(int), 0);
189 8ad51794 2004-03-25 devnull cwarn = chancreate(sizeof(void*), 1);
190 8ad51794 2004-03-25 devnull if(cwait==nil || ccommand==nil || ckill==nil || cxfidalloc==nil || cxfidfree==nil || cerr==nil || cexit==nil || cwarn==nil){
191 b3994ec5 2003-12-11 devnull fprint(2, "acme: can't create initial channels: %r\n");
192 38c10d1a 2005-01-17 devnull threadexitsall("channels");
193 b3994ec5 2003-12-11 devnull }
194 cb1ac4c4 2008-03-07 rsc chansetname(ccommand, "ccommand");
195 cb1ac4c4 2008-03-07 rsc chansetname(ckill, "ckill");
196 cb1ac4c4 2008-03-07 rsc chansetname(cxfidalloc, "cxfidalloc");
197 cb1ac4c4 2008-03-07 rsc chansetname(cxfidfree, "cxfidfree");
198 cb1ac4c4 2008-03-07 rsc chansetname(cnewwindow, "cnewwindow");
199 cb1ac4c4 2008-03-07 rsc chansetname(cerr, "cerr");
200 cb1ac4c4 2008-03-07 rsc chansetname(cedit, "cedit");
201 cb1ac4c4 2008-03-07 rsc chansetname(cexit, "cexit");
202 cb1ac4c4 2008-03-07 rsc chansetname(cwarn, "cwarn");
203 b3994ec5 2003-12-11 devnull
204 b3994ec5 2003-12-11 devnull mousectl = initmouse(nil, screen);
205 b3994ec5 2003-12-11 devnull if(mousectl == nil){
206 b3994ec5 2003-12-11 devnull fprint(2, "acme: can't initialize mouse: %r\n");
207 38c10d1a 2005-01-17 devnull threadexitsall("mouse");
208 b3994ec5 2003-12-11 devnull }
209 b3994ec5 2003-12-11 devnull mouse = &mousectl->m;
210 b3994ec5 2003-12-11 devnull keyboardctl = initkeyboard(nil);
211 b3994ec5 2003-12-11 devnull if(keyboardctl == nil){
212 b3994ec5 2003-12-11 devnull fprint(2, "acme: can't initialize keyboard: %r\n");
213 38c10d1a 2005-01-17 devnull threadexitsall("keyboard");
214 b3994ec5 2003-12-11 devnull }
215 b3994ec5 2003-12-11 devnull mainpid = getpid();
216 2277c5d7 2004-03-21 devnull startplumbing();
217 2277c5d7 2004-03-21 devnull /*
218 b3994ec5 2003-12-11 devnull plumbeditfd = plumbopen("edit", OREAD|OCEXEC);
219 b3994ec5 2003-12-11 devnull if(plumbeditfd < 0)
220 b3994ec5 2003-12-11 devnull fprint(2, "acme: can't initialize plumber: %r\n");
221 b3994ec5 2003-12-11 devnull else{
222 b3994ec5 2003-12-11 devnull cplumb = chancreate(sizeof(Plumbmsg*), 0);
223 5a8e63b2 2004-02-29 devnull threadcreate(plumbproc, nil, STACK);
224 b3994ec5 2003-12-11 devnull }
225 b3994ec5 2003-12-11 devnull plumbsendfd = plumbopen("send", OWRITE|OCEXEC);
226 2277c5d7 2004-03-21 devnull */
227 b3994ec5 2003-12-11 devnull
228 b3994ec5 2003-12-11 devnull fsysinit();
229 b3994ec5 2003-12-11 devnull
230 b3994ec5 2003-12-11 devnull #define WPERCOL 8
231 b3994ec5 2003-12-11 devnull disk = diskinit();
232 6d7fdb24 2004-12-27 devnull if(!loadfile || !rowload(&row, loadfile, TRUE)){
233 b3994ec5 2003-12-11 devnull rowinit(&row, screen->clipr);
234 b3994ec5 2003-12-11 devnull if(ncol < 0){
235 b3994ec5 2003-12-11 devnull if(argc == 0)
236 b3994ec5 2003-12-11 devnull ncol = 2;
237 b3994ec5 2003-12-11 devnull else{
238 b3994ec5 2003-12-11 devnull ncol = (argc+(WPERCOL-1))/WPERCOL;
239 b3994ec5 2003-12-11 devnull if(ncol < 2)
240 b3994ec5 2003-12-11 devnull ncol = 2;
241 b3994ec5 2003-12-11 devnull }
242 b3994ec5 2003-12-11 devnull }
243 b3994ec5 2003-12-11 devnull if(ncol == 0)
244 b3994ec5 2003-12-11 devnull ncol = 2;
245 b3994ec5 2003-12-11 devnull for(i=0; i<ncol; i++){
246 b3994ec5 2003-12-11 devnull c = rowadd(&row, nil, -1);
247 b3994ec5 2003-12-11 devnull if(c==nil && i==0)
248 b3994ec5 2003-12-11 devnull error("initializing columns");
249 b3994ec5 2003-12-11 devnull }
250 b3994ec5 2003-12-11 devnull c = row.col[row.ncol-1];
251 b3994ec5 2003-12-11 devnull if(argc == 0)
252 b3994ec5 2003-12-11 devnull readfile(c, wdir);
253 b3994ec5 2003-12-11 devnull else
254 b3994ec5 2003-12-11 devnull for(i=0; i<argc; i++){
255 b3994ec5 2003-12-11 devnull p = utfrrune(argv[i], '/');
256 b3994ec5 2003-12-11 devnull if((p!=nil && strcmp(p, "/guide")==0) || i/WPERCOL>=row.ncol)
257 b3994ec5 2003-12-11 devnull readfile(c, argv[i]);
258 b3994ec5 2003-12-11 devnull else
259 b3994ec5 2003-12-11 devnull readfile(row.col[i/WPERCOL], argv[i]);
260 b3994ec5 2003-12-11 devnull }
261 b3994ec5 2003-12-11 devnull }
262 b3994ec5 2003-12-11 devnull flushimage(display, 1);
263 b3994ec5 2003-12-11 devnull
264 b3994ec5 2003-12-11 devnull acmeerrorinit();
265 b3994ec5 2003-12-11 devnull threadcreate(keyboardthread, nil, STACK);
266 b3994ec5 2003-12-11 devnull threadcreate(mousethread, nil, STACK);
267 b3994ec5 2003-12-11 devnull threadcreate(waitthread, nil, STACK);
268 b3994ec5 2003-12-11 devnull threadcreate(xfidallocthread, nil, STACK);
269 b3994ec5 2003-12-11 devnull threadcreate(newwindowthread, nil, STACK);
270 334cb1e9 2004-12-27 devnull /* threadcreate(shutdownthread, nil, STACK); */
271 93a6f8d7 2005-01-04 devnull threadnotify(shutdown, 1);
272 b3994ec5 2003-12-11 devnull recvul(cexit);
273 b3994ec5 2003-12-11 devnull killprocs();
274 b3994ec5 2003-12-11 devnull threadexitsall(nil);
275 b3994ec5 2003-12-11 devnull }
276 b3994ec5 2003-12-11 devnull
277 b3994ec5 2003-12-11 devnull void
278 b3994ec5 2003-12-11 devnull readfile(Column *c, char *s)
279 b3994ec5 2003-12-11 devnull {
280 b3994ec5 2003-12-11 devnull Window *w;
281 b3994ec5 2003-12-11 devnull Rune rb[256];
282 93611313 2006-06-16 devnull int nr;
283 b3994ec5 2003-12-11 devnull Runestr rs;
284 b3994ec5 2003-12-11 devnull
285 b3994ec5 2003-12-11 devnull w = coladd(c, nil, nil, -1);
286 93611313 2006-06-16 devnull if(s[0] != '/')
287 93611313 2006-06-16 devnull runesnprint(rb, sizeof rb, "%s/%s", wdir, s);
288 93611313 2006-06-16 devnull else
289 93611313 2006-06-16 devnull runesnprint(rb, sizeof rb, "%s", s);
290 93611313 2006-06-16 devnull nr = runestrlen(rb);
291 8ad51794 2004-03-25 devnull rs = cleanrname(runestr(rb, nr));
292 b3994ec5 2003-12-11 devnull winsetname(w, rs.r, rs.nr);
293 b3994ec5 2003-12-11 devnull textload(&w->body, 0, s, 1);
294 b3994ec5 2003-12-11 devnull w->body.file->mod = FALSE;
295 b3994ec5 2003-12-11 devnull w->dirty = FALSE;
296 b3994ec5 2003-12-11 devnull winsettag(w);
297 cd1d0ab0 2006-01-12 devnull winresize(w, w->r, FALSE, TRUE);
298 b3994ec5 2003-12-11 devnull textscrdraw(&w->body);
299 b3994ec5 2003-12-11 devnull textsetselect(&w->tag, w->tag.file->b.nc, w->tag.file->b.nc);
300 b3994ec5 2003-12-11 devnull }
301 b3994ec5 2003-12-11 devnull
302 429f8aa4 2005-02-15 devnull char *ignotes[] = {
303 429f8aa4 2005-02-15 devnull "sys: write on closed pipe",
304 429f8aa4 2005-02-15 devnull "sys: ttin",
305 429f8aa4 2005-02-15 devnull "sys: ttou",
306 429f8aa4 2005-02-15 devnull "sys: tstp",
307 429f8aa4 2005-02-15 devnull nil
308 429f8aa4 2005-02-15 devnull };
309 429f8aa4 2005-02-15 devnull
310 b3994ec5 2003-12-11 devnull char *oknotes[] ={
311 b3994ec5 2003-12-11 devnull "delete",
312 b3994ec5 2003-12-11 devnull "hangup",
313 b3994ec5 2003-12-11 devnull "kill",
314 b3994ec5 2003-12-11 devnull "exit",
315 b3994ec5 2003-12-11 devnull nil
316 b3994ec5 2003-12-11 devnull };
317 b3994ec5 2003-12-11 devnull
318 b3994ec5 2003-12-11 devnull int dumping;
319 b3994ec5 2003-12-11 devnull
320 4da83e7c 2004-03-05 devnull static int
321 b3994ec5 2003-12-11 devnull shutdown(void *v, char *msg)
322 b3994ec5 2003-12-11 devnull {
323 b3994ec5 2003-12-11 devnull int i;
324 b3994ec5 2003-12-11 devnull
325 b3994ec5 2003-12-11 devnull USED(v);
326 429f8aa4 2005-02-15 devnull
327 429f8aa4 2005-02-15 devnull for(i=0; ignotes[i]; i++)
328 429f8aa4 2005-02-15 devnull if(strncmp(ignotes[i], msg, strlen(ignotes[i])) == 0)
329 429f8aa4 2005-02-15 devnull return 1;
330 429f8aa4 2005-02-15 devnull
331 b3994ec5 2003-12-11 devnull killprocs();
332 b3994ec5 2003-12-11 devnull if(!dumping && strcmp(msg, "kill")!=0 && strcmp(msg, "exit")!=0 && getpid()==mainpid){
333 b3994ec5 2003-12-11 devnull dumping = TRUE;
334 b3994ec5 2003-12-11 devnull rowdump(&row, nil);
335 b3994ec5 2003-12-11 devnull }
336 b3994ec5 2003-12-11 devnull for(i=0; oknotes[i]; i++)
337 b3994ec5 2003-12-11 devnull if(strncmp(oknotes[i], msg, strlen(oknotes[i])) == 0)
338 b3994ec5 2003-12-11 devnull threadexitsall(msg);
339 b3994ec5 2003-12-11 devnull print("acme: %s\n", msg);
340 b3994ec5 2003-12-11 devnull return 0;
341 b3994ec5 2003-12-11 devnull }
342 b3994ec5 2003-12-11 devnull
343 334cb1e9 2004-12-27 devnull /*
344 b3994ec5 2003-12-11 devnull void
345 334cb1e9 2004-12-27 devnull shutdownthread(void *v)
346 334cb1e9 2004-12-27 devnull {
347 334cb1e9 2004-12-27 devnull char *msg;
348 334cb1e9 2004-12-27 devnull Channel *c;
349 334cb1e9 2004-12-27 devnull
350 334cb1e9 2004-12-27 devnull USED(v);
351 334cb1e9 2004-12-27 devnull
352 391363f5 2005-09-26 devnull threadsetname("shutdown");
353 334cb1e9 2004-12-27 devnull c = threadnotechan();
354 334cb1e9 2004-12-27 devnull while((msg = recvp(c)) != nil)
355 334cb1e9 2004-12-27 devnull shutdown(nil, msg);
356 334cb1e9 2004-12-27 devnull }
357 334cb1e9 2004-12-27 devnull */
358 334cb1e9 2004-12-27 devnull
359 334cb1e9 2004-12-27 devnull void
360 b3994ec5 2003-12-11 devnull killprocs(void)
361 b3994ec5 2003-12-11 devnull {
362 b3994ec5 2003-12-11 devnull Command *c;
363 b3994ec5 2003-12-11 devnull
364 b3994ec5 2003-12-11 devnull fsysclose();
365 cbeb0b26 2006-04-01 devnull /* if(display) */
366 cbeb0b26 2006-04-01 devnull /* flushimage(display, 1); */
367 b3994ec5 2003-12-11 devnull
368 b3994ec5 2003-12-11 devnull for(c=command; c; c=c->next)
369 b3994ec5 2003-12-11 devnull postnote(PNGROUP, c->pid, "hangup");
370 b3994ec5 2003-12-11 devnull }
371 b3994ec5 2003-12-11 devnull
372 b3994ec5 2003-12-11 devnull static int errorfd;
373 b3994ec5 2003-12-11 devnull int erroutfd;
374 b3994ec5 2003-12-11 devnull
375 b3994ec5 2003-12-11 devnull void
376 b3994ec5 2003-12-11 devnull acmeerrorproc(void *v)
377 b3994ec5 2003-12-11 devnull {
378 b3994ec5 2003-12-11 devnull char *buf;
379 b3994ec5 2003-12-11 devnull int n;
380 b3994ec5 2003-12-11 devnull
381 b3994ec5 2003-12-11 devnull USED(v);
382 b3994ec5 2003-12-11 devnull threadsetname("acmeerrorproc");
383 b3994ec5 2003-12-11 devnull buf = emalloc(8192+1);
384 334cb1e9 2004-12-27 devnull while((n=read(errorfd, buf, 8192)) >= 0){
385 b3994ec5 2003-12-11 devnull buf[n] = '\0';
386 b3994ec5 2003-12-11 devnull sendp(cerr, estrdup(buf));
387 b3994ec5 2003-12-11 devnull }
388 b3994ec5 2003-12-11 devnull }
389 b3994ec5 2003-12-11 devnull
390 b3994ec5 2003-12-11 devnull void
391 b3994ec5 2003-12-11 devnull acmeerrorinit(void)
392 b3994ec5 2003-12-11 devnull {
393 5a8e63b2 2004-02-29 devnull int pfd[2];
394 b3994ec5 2003-12-11 devnull
395 b3994ec5 2003-12-11 devnull if(pipe(pfd) < 0)
396 b3994ec5 2003-12-11 devnull error("can't create pipe");
397 b3994ec5 2003-12-11 devnull #if 0
398 b3994ec5 2003-12-11 devnull sprint(acmeerrorfile, "/srv/acme.%s.%d", getuser(), mainpid);
399 b3994ec5 2003-12-11 devnull fd = create(acmeerrorfile, OWRITE, 0666);
400 b3994ec5 2003-12-11 devnull if(fd < 0){
401 b3994ec5 2003-12-11 devnull remove(acmeerrorfile);
402 b3994ec5 2003-12-11 devnull fd = create(acmeerrorfile, OWRITE, 0666);
403 b3994ec5 2003-12-11 devnull if(fd < 0)
404 b3994ec5 2003-12-11 devnull error("can't create acmeerror file");
405 b3994ec5 2003-12-11 devnull }
406 b3994ec5 2003-12-11 devnull sprint(buf, "%d", pfd[0]);
407 b3994ec5 2003-12-11 devnull write(fd, buf, strlen(buf));
408 b3994ec5 2003-12-11 devnull close(fd);
409 b3994ec5 2003-12-11 devnull /* reopen pfd[1] close on exec */
410 b3994ec5 2003-12-11 devnull sprint(buf, "/fd/%d", pfd[1]);
411 b3994ec5 2003-12-11 devnull errorfd = open(buf, OREAD|OCEXEC);
412 b3994ec5 2003-12-11 devnull #endif
413 b3994ec5 2003-12-11 devnull fcntl(pfd[0], F_SETFD, FD_CLOEXEC);
414 b3994ec5 2003-12-11 devnull fcntl(pfd[1], F_SETFD, FD_CLOEXEC);
415 b3994ec5 2003-12-11 devnull erroutfd = pfd[0];
416 b3994ec5 2003-12-11 devnull errorfd = pfd[1];
417 b3994ec5 2003-12-11 devnull if(errorfd < 0)
418 b3994ec5 2003-12-11 devnull error("can't re-open acmeerror file");
419 334cb1e9 2004-12-27 devnull proccreate(acmeerrorproc, nil, STACK);
420 b3994ec5 2003-12-11 devnull }
421 b3994ec5 2003-12-11 devnull
422 2277c5d7 2004-03-21 devnull /*
423 b3994ec5 2003-12-11 devnull void
424 b3994ec5 2003-12-11 devnull plumbproc(void *v)
425 b3994ec5 2003-12-11 devnull {
426 b3994ec5 2003-12-11 devnull Plumbmsg *m;
427 b3994ec5 2003-12-11 devnull
428 b3994ec5 2003-12-11 devnull USED(v);
429 b3994ec5 2003-12-11 devnull threadsetname("plumbproc");
430 b3994ec5 2003-12-11 devnull for(;;){
431 5a8e63b2 2004-02-29 devnull m = threadplumbrecv(plumbeditfd);
432 b3994ec5 2003-12-11 devnull if(m == nil)
433 b3994ec5 2003-12-11 devnull threadexits(nil);
434 b3994ec5 2003-12-11 devnull sendp(cplumb, m);
435 b3994ec5 2003-12-11 devnull }
436 b3994ec5 2003-12-11 devnull }
437 2277c5d7 2004-03-21 devnull */
438 b3994ec5 2003-12-11 devnull
439 b3994ec5 2003-12-11 devnull void
440 b3994ec5 2003-12-11 devnull keyboardthread(void *v)
441 b3994ec5 2003-12-11 devnull {
442 b3994ec5 2003-12-11 devnull Rune r;
443 b3994ec5 2003-12-11 devnull Timer *timer;
444 b3994ec5 2003-12-11 devnull Text *t;
445 b3994ec5 2003-12-11 devnull enum { KTimer, KKey, NKALT };
446 b3994ec5 2003-12-11 devnull static Alt alts[NKALT+1];
447 b3994ec5 2003-12-11 devnull
448 b3994ec5 2003-12-11 devnull USED(v);
449 b3994ec5 2003-12-11 devnull alts[KTimer].c = nil;
450 b3994ec5 2003-12-11 devnull alts[KTimer].v = nil;
451 b3994ec5 2003-12-11 devnull alts[KTimer].op = CHANNOP;
452 b3994ec5 2003-12-11 devnull alts[KKey].c = keyboardctl->c;
453 b3994ec5 2003-12-11 devnull alts[KKey].v = &r;
454 b3994ec5 2003-12-11 devnull alts[KKey].op = CHANRCV;
455 b3994ec5 2003-12-11 devnull alts[NKALT].op = CHANEND;
456 b3994ec5 2003-12-11 devnull
457 b3994ec5 2003-12-11 devnull timer = nil;
458 b3994ec5 2003-12-11 devnull typetext = nil;
459 b3994ec5 2003-12-11 devnull threadsetname("keyboardthread");
460 b3994ec5 2003-12-11 devnull for(;;){
461 b3994ec5 2003-12-11 devnull switch(alt(alts)){
462 b3994ec5 2003-12-11 devnull case KTimer:
463 b3994ec5 2003-12-11 devnull timerstop(timer);
464 b3994ec5 2003-12-11 devnull t = typetext;
465 b3994ec5 2003-12-11 devnull if(t!=nil && t->what==Tag){
466 b3994ec5 2003-12-11 devnull winlock(t->w, 'K');
467 b3994ec5 2003-12-11 devnull wincommit(t->w, t);
468 b3994ec5 2003-12-11 devnull winunlock(t->w);
469 b3994ec5 2003-12-11 devnull flushimage(display, 1);
470 b3994ec5 2003-12-11 devnull }
471 b3994ec5 2003-12-11 devnull alts[KTimer].c = nil;
472 b3994ec5 2003-12-11 devnull alts[KTimer].op = CHANNOP;
473 b3994ec5 2003-12-11 devnull break;
474 b3994ec5 2003-12-11 devnull case KKey:
475 b3994ec5 2003-12-11 devnull casekeyboard:
476 b3994ec5 2003-12-11 devnull typetext = rowtype(&row, r, mouse->xy);
477 b3994ec5 2003-12-11 devnull t = typetext;
478 b3994ec5 2003-12-11 devnull if(t!=nil && t->col!=nil && !(r==Kdown || r==Kleft || r==Kright)) /* scrolling doesn't change activecol */
479 b3994ec5 2003-12-11 devnull activecol = t->col;
480 b3994ec5 2003-12-11 devnull if(t!=nil && t->w!=nil)
481 b3994ec5 2003-12-11 devnull t->w->body.file->curtext = &t->w->body;
482 b3994ec5 2003-12-11 devnull if(timer != nil)
483 b3994ec5 2003-12-11 devnull timercancel(timer);
484 b3994ec5 2003-12-11 devnull if(t!=nil && t->what==Tag) {
485 b3994ec5 2003-12-11 devnull timer = timerstart(500);
486 b3994ec5 2003-12-11 devnull alts[KTimer].c = timer->c;
487 b3994ec5 2003-12-11 devnull alts[KTimer].op = CHANRCV;
488 b3994ec5 2003-12-11 devnull }else{
489 b3994ec5 2003-12-11 devnull timer = nil;
490 b3994ec5 2003-12-11 devnull alts[KTimer].c = nil;
491 b3994ec5 2003-12-11 devnull alts[KTimer].op = CHANNOP;
492 b3994ec5 2003-12-11 devnull }
493 b3994ec5 2003-12-11 devnull if(nbrecv(keyboardctl->c, &r) > 0)
494 b3994ec5 2003-12-11 devnull goto casekeyboard;
495 b3994ec5 2003-12-11 devnull flushimage(display, 1);
496 b3994ec5 2003-12-11 devnull break;
497 b3994ec5 2003-12-11 devnull }
498 b3994ec5 2003-12-11 devnull }
499 b3994ec5 2003-12-11 devnull }
500 b3994ec5 2003-12-11 devnull
501 b3994ec5 2003-12-11 devnull void
502 b3994ec5 2003-12-11 devnull mousethread(void *v)
503 b3994ec5 2003-12-11 devnull {
504 b3994ec5 2003-12-11 devnull Text *t, *argt;
505 b3994ec5 2003-12-11 devnull int but;
506 b3994ec5 2003-12-11 devnull uint q0, q1;
507 b3994ec5 2003-12-11 devnull Window *w;
508 b3994ec5 2003-12-11 devnull Plumbmsg *pm;
509 b3994ec5 2003-12-11 devnull Mouse m;
510 b3994ec5 2003-12-11 devnull char *act;
511 8ad51794 2004-03-25 devnull enum { MResize, MMouse, MPlumb, MWarnings, NMALT };
512 b3994ec5 2003-12-11 devnull static Alt alts[NMALT+1];
513 b3994ec5 2003-12-11 devnull
514 b3994ec5 2003-12-11 devnull USED(v);
515 b3994ec5 2003-12-11 devnull threadsetname("mousethread");
516 b3994ec5 2003-12-11 devnull alts[MResize].c = mousectl->resizec;
517 b3994ec5 2003-12-11 devnull alts[MResize].v = nil;
518 b3994ec5 2003-12-11 devnull alts[MResize].op = CHANRCV;
519 b3994ec5 2003-12-11 devnull alts[MMouse].c = mousectl->c;
520 b3994ec5 2003-12-11 devnull alts[MMouse].v = &mousectl->m;
521 b3994ec5 2003-12-11 devnull alts[MMouse].op = CHANRCV;
522 b3994ec5 2003-12-11 devnull alts[MPlumb].c = cplumb;
523 b3994ec5 2003-12-11 devnull alts[MPlumb].v = &pm;
524 b3994ec5 2003-12-11 devnull alts[MPlumb].op = CHANRCV;
525 8ad51794 2004-03-25 devnull alts[MWarnings].c = cwarn;
526 8ad51794 2004-03-25 devnull alts[MWarnings].v = nil;
527 8ad51794 2004-03-25 devnull alts[MWarnings].op = CHANRCV;
528 b3994ec5 2003-12-11 devnull if(cplumb == nil)
529 b3994ec5 2003-12-11 devnull alts[MPlumb].op = CHANNOP;
530 b3994ec5 2003-12-11 devnull alts[NMALT].op = CHANEND;
531 b3994ec5 2003-12-11 devnull
532 b3994ec5 2003-12-11 devnull for(;;){
533 8ad51794 2004-03-25 devnull qlock(&row.lk);
534 8ad51794 2004-03-25 devnull flushwarnings();
535 8ad51794 2004-03-25 devnull qunlock(&row.lk);
536 8ad51794 2004-03-25 devnull flushimage(display, 1);
537 b3994ec5 2003-12-11 devnull switch(alt(alts)){
538 b3994ec5 2003-12-11 devnull case MResize:
539 b3994ec5 2003-12-11 devnull if(getwindow(display, Refnone) < 0)
540 b3994ec5 2003-12-11 devnull error("attach to window");
541 b3994ec5 2003-12-11 devnull draw(screen, screen->r, display->white, nil, ZP);
542 b3994ec5 2003-12-11 devnull scrlresize();
543 b3994ec5 2003-12-11 devnull rowresize(&row, screen->clipr);
544 b3994ec5 2003-12-11 devnull break;
545 b3994ec5 2003-12-11 devnull case MPlumb:
546 b3994ec5 2003-12-11 devnull if(strcmp(pm->type, "text") == 0){
547 b3994ec5 2003-12-11 devnull act = plumblookup(pm->attr, "action");
548 b3994ec5 2003-12-11 devnull if(act==nil || strcmp(act, "showfile")==0)
549 b3994ec5 2003-12-11 devnull plumblook(pm);
550 b3994ec5 2003-12-11 devnull else if(strcmp(act, "showdata")==0)
551 b3994ec5 2003-12-11 devnull plumbshow(pm);
552 b3994ec5 2003-12-11 devnull }
553 8ad51794 2004-03-25 devnull plumbfree(pm);
554 b3994ec5 2003-12-11 devnull break;
555 8ad51794 2004-03-25 devnull case MWarnings:
556 8ad51794 2004-03-25 devnull break;
557 b3994ec5 2003-12-11 devnull case MMouse:
558 b3994ec5 2003-12-11 devnull /*
559 b3994ec5 2003-12-11 devnull * Make a copy so decisions are consistent; mousectl changes
560 b3994ec5 2003-12-11 devnull * underfoot. Can't just receive into m because this introduces
561 b3994ec5 2003-12-11 devnull * another race; see /sys/src/libdraw/mouse.c.
562 b3994ec5 2003-12-11 devnull */
563 b3994ec5 2003-12-11 devnull m = mousectl->m;
564 b3994ec5 2003-12-11 devnull qlock(&row.lk);
565 b3994ec5 2003-12-11 devnull t = rowwhich(&row, m.xy);
566 b3994ec5 2003-12-11 devnull if(t!=mousetext && mousetext!=nil && mousetext->w!=nil){
567 b3994ec5 2003-12-11 devnull winlock(mousetext->w, 'M');
568 b3994ec5 2003-12-11 devnull mousetext->eq0 = ~0;
569 b3994ec5 2003-12-11 devnull wincommit(mousetext->w, mousetext);
570 b3994ec5 2003-12-11 devnull winunlock(mousetext->w);
571 b3994ec5 2003-12-11 devnull }
572 b3994ec5 2003-12-11 devnull mousetext = t;
573 b3994ec5 2003-12-11 devnull if(t == nil)
574 b3994ec5 2003-12-11 devnull goto Continue;
575 b3994ec5 2003-12-11 devnull w = t->w;
576 b3994ec5 2003-12-11 devnull if(t==nil || m.buttons==0)
577 b3994ec5 2003-12-11 devnull goto Continue;
578 b3994ec5 2003-12-11 devnull but = 0;
579 b3994ec5 2003-12-11 devnull if(m.buttons == 1)
580 b3994ec5 2003-12-11 devnull but = 1;
581 b3994ec5 2003-12-11 devnull else if(m.buttons == 2)
582 b3994ec5 2003-12-11 devnull but = 2;
583 b3994ec5 2003-12-11 devnull else if(m.buttons == 4)
584 b3994ec5 2003-12-11 devnull but = 3;
585 b3994ec5 2003-12-11 devnull barttext = t;
586 b3994ec5 2003-12-11 devnull if(t->what==Body && ptinrect(m.xy, t->scrollr)){
587 b3994ec5 2003-12-11 devnull if(but){
588 d8c78a8d 2005-01-10 devnull if(swapscrollbuttons){
589 d8c78a8d 2005-01-10 devnull if(but == 1)
590 d8c78a8d 2005-01-10 devnull but = 3;
591 d8c78a8d 2005-01-10 devnull else if(but == 3)
592 d8c78a8d 2005-01-10 devnull but = 1;
593 d8c78a8d 2005-01-10 devnull }
594 b3994ec5 2003-12-11 devnull winlock(w, 'M');
595 b3994ec5 2003-12-11 devnull t->eq0 = ~0;
596 b3994ec5 2003-12-11 devnull textscroll(t, but);
597 b3994ec5 2003-12-11 devnull winunlock(w);
598 b3994ec5 2003-12-11 devnull }
599 c16f1f73 2004-06-09 devnull goto Continue;
600 c16f1f73 2004-06-09 devnull }
601 c16f1f73 2004-06-09 devnull /* scroll buttons, wheels, etc. */
602 f8dea3c1 2005-12-16 devnull if(w != nil && (m.buttons & (8|16))){
603 c16f1f73 2004-06-09 devnull if(m.buttons & 8)
604 c16f1f73 2004-06-09 devnull but = Kscrolloneup;
605 c16f1f73 2004-06-09 devnull else
606 c16f1f73 2004-06-09 devnull but = Kscrollonedown;
607 c16f1f73 2004-06-09 devnull winlock(w, 'M');
608 c16f1f73 2004-06-09 devnull t->eq0 = ~0;
609 c16f1f73 2004-06-09 devnull texttype(t, but);
610 c16f1f73 2004-06-09 devnull winunlock(w);
611 b3994ec5 2003-12-11 devnull goto Continue;
612 b3994ec5 2003-12-11 devnull }
613 b3994ec5 2003-12-11 devnull if(ptinrect(m.xy, t->scrollr)){
614 b3994ec5 2003-12-11 devnull if(but){
615 b3994ec5 2003-12-11 devnull if(t->what == Columntag)
616 b3994ec5 2003-12-11 devnull rowdragcol(&row, t->col, but);
617 b3994ec5 2003-12-11 devnull else if(t->what == Tag){
618 b3994ec5 2003-12-11 devnull coldragwin(t->col, t->w, but);
619 b3994ec5 2003-12-11 devnull if(t->w)
620 b3994ec5 2003-12-11 devnull barttext = &t->w->body;
621 b3994ec5 2003-12-11 devnull }
622 b3994ec5 2003-12-11 devnull if(t->col)
623 b3994ec5 2003-12-11 devnull activecol = t->col;
624 b3994ec5 2003-12-11 devnull }
625 b3994ec5 2003-12-11 devnull goto Continue;
626 b3994ec5 2003-12-11 devnull }
627 b3994ec5 2003-12-11 devnull if(m.buttons){
628 b3994ec5 2003-12-11 devnull if(w)
629 b3994ec5 2003-12-11 devnull winlock(w, 'M');
630 b3994ec5 2003-12-11 devnull t->eq0 = ~0;
631 b3994ec5 2003-12-11 devnull if(w)
632 b3994ec5 2003-12-11 devnull wincommit(w, t);
633 b3994ec5 2003-12-11 devnull else
634 b3994ec5 2003-12-11 devnull textcommit(t, TRUE);
635 b3994ec5 2003-12-11 devnull if(m.buttons & 1){
636 b3994ec5 2003-12-11 devnull textselect(t);
637 b3994ec5 2003-12-11 devnull if(w)
638 b3994ec5 2003-12-11 devnull winsettag(w);
639 b3994ec5 2003-12-11 devnull argtext = t;
640 b3994ec5 2003-12-11 devnull seltext = t;
641 b3994ec5 2003-12-11 devnull if(t->col)
642 b3994ec5 2003-12-11 devnull activecol = t->col; /* button 1 only */
643 b3994ec5 2003-12-11 devnull if(t->w!=nil && t==&t->w->body)
644 b3994ec5 2003-12-11 devnull activewin = t->w;
645 b3994ec5 2003-12-11 devnull }else if(m.buttons & 2){
646 b3994ec5 2003-12-11 devnull if(textselect2(t, &q0, &q1, &argt))
647 b3994ec5 2003-12-11 devnull execute(t, q0, q1, FALSE, argt);
648 b3994ec5 2003-12-11 devnull }else if(m.buttons & 4){
649 b3994ec5 2003-12-11 devnull if(textselect3(t, &q0, &q1))
650 b3994ec5 2003-12-11 devnull look3(t, q0, q1, FALSE);
651 b3994ec5 2003-12-11 devnull }
652 b3994ec5 2003-12-11 devnull if(w)
653 b3994ec5 2003-12-11 devnull winunlock(w);
654 b3994ec5 2003-12-11 devnull goto Continue;
655 b3994ec5 2003-12-11 devnull }
656 b3994ec5 2003-12-11 devnull Continue:
657 b3994ec5 2003-12-11 devnull qunlock(&row.lk);
658 b3994ec5 2003-12-11 devnull break;
659 b3994ec5 2003-12-11 devnull }
660 b3994ec5 2003-12-11 devnull }
661 b3994ec5 2003-12-11 devnull }
662 b3994ec5 2003-12-11 devnull
663 b3994ec5 2003-12-11 devnull /*
664 b3994ec5 2003-12-11 devnull * There is a race between process exiting and our finding out it was ever created.
665 b3994ec5 2003-12-11 devnull * This structure keeps a list of processes that have exited we haven't heard of.
666 b3994ec5 2003-12-11 devnull */
667 b3994ec5 2003-12-11 devnull typedef struct Pid Pid;
668 b3994ec5 2003-12-11 devnull struct Pid
669 b3994ec5 2003-12-11 devnull {
670 b3994ec5 2003-12-11 devnull int pid;
671 b3994ec5 2003-12-11 devnull char msg[ERRMAX];
672 b3994ec5 2003-12-11 devnull Pid *next;
673 b3994ec5 2003-12-11 devnull };
674 b3994ec5 2003-12-11 devnull
675 b3994ec5 2003-12-11 devnull void
676 b3994ec5 2003-12-11 devnull waitthread(void *v)
677 b3994ec5 2003-12-11 devnull {
678 b3994ec5 2003-12-11 devnull Waitmsg *w;
679 b3994ec5 2003-12-11 devnull Command *c, *lc;
680 b3994ec5 2003-12-11 devnull uint pid;
681 b3994ec5 2003-12-11 devnull int found, ncmd;
682 b3994ec5 2003-12-11 devnull Rune *cmd;
683 b3994ec5 2003-12-11 devnull char *err;
684 b3994ec5 2003-12-11 devnull Text *t;
685 b3994ec5 2003-12-11 devnull Pid *pids, *p, *lastp;
686 b3994ec5 2003-12-11 devnull enum { WErr, WKill, WWait, WCmd, NWALT };
687 b3994ec5 2003-12-11 devnull Alt alts[NWALT+1];
688 b3994ec5 2003-12-11 devnull
689 b3994ec5 2003-12-11 devnull USED(v);
690 b3994ec5 2003-12-11 devnull threadsetname("waitthread");
691 b3994ec5 2003-12-11 devnull pids = nil;
692 b3994ec5 2003-12-11 devnull alts[WErr].c = cerr;
693 b3994ec5 2003-12-11 devnull alts[WErr].v = &err;
694 b3994ec5 2003-12-11 devnull alts[WErr].op = CHANRCV;
695 b3994ec5 2003-12-11 devnull alts[WKill].c = ckill;
696 b3994ec5 2003-12-11 devnull alts[WKill].v = &cmd;
697 b3994ec5 2003-12-11 devnull alts[WKill].op = CHANRCV;
698 b3994ec5 2003-12-11 devnull alts[WWait].c = cwait;
699 b3994ec5 2003-12-11 devnull alts[WWait].v = &w;
700 b3994ec5 2003-12-11 devnull alts[WWait].op = CHANRCV;
701 b3994ec5 2003-12-11 devnull alts[WCmd].c = ccommand;
702 b3994ec5 2003-12-11 devnull alts[WCmd].v = &c;
703 b3994ec5 2003-12-11 devnull alts[WCmd].op = CHANRCV;
704 b3994ec5 2003-12-11 devnull alts[NWALT].op = CHANEND;
705 b3994ec5 2003-12-11 devnull
706 b3994ec5 2003-12-11 devnull command = nil;
707 b3994ec5 2003-12-11 devnull for(;;){
708 b3994ec5 2003-12-11 devnull switch(alt(alts)){
709 b3994ec5 2003-12-11 devnull case WErr:
710 b3994ec5 2003-12-11 devnull qlock(&row.lk);
711 b3994ec5 2003-12-11 devnull warning(nil, "%s", err);
712 b3994ec5 2003-12-11 devnull free(err);
713 b3994ec5 2003-12-11 devnull flushimage(display, 1);
714 b3994ec5 2003-12-11 devnull qunlock(&row.lk);
715 b3994ec5 2003-12-11 devnull break;
716 b3994ec5 2003-12-11 devnull case WKill:
717 b3994ec5 2003-12-11 devnull found = FALSE;
718 b3994ec5 2003-12-11 devnull ncmd = runestrlen(cmd);
719 b3994ec5 2003-12-11 devnull for(c=command; c; c=c->next){
720 b3994ec5 2003-12-11 devnull /* -1 for blank */
721 b3994ec5 2003-12-11 devnull if(runeeq(c->name, c->nname-1, cmd, ncmd) == TRUE){
722 b3994ec5 2003-12-11 devnull if(postnote(PNGROUP, c->pid, "kill") < 0)
723 b3994ec5 2003-12-11 devnull warning(nil, "kill %S: %r\n", cmd);
724 b3994ec5 2003-12-11 devnull found = TRUE;
725 b3994ec5 2003-12-11 devnull }
726 b3994ec5 2003-12-11 devnull }
727 b3994ec5 2003-12-11 devnull if(!found)
728 b3994ec5 2003-12-11 devnull warning(nil, "Kill: no process %S\n", cmd);
729 b3994ec5 2003-12-11 devnull free(cmd);
730 b3994ec5 2003-12-11 devnull break;
731 b3994ec5 2003-12-11 devnull case WWait:
732 b3994ec5 2003-12-11 devnull pid = w->pid;
733 b3994ec5 2003-12-11 devnull lc = nil;
734 b3994ec5 2003-12-11 devnull for(c=command; c; c=c->next){
735 b3994ec5 2003-12-11 devnull if(c->pid == pid){
736 b3994ec5 2003-12-11 devnull if(lc)
737 b3994ec5 2003-12-11 devnull lc->next = c->next;
738 b3994ec5 2003-12-11 devnull else
739 b3994ec5 2003-12-11 devnull command = c->next;
740 b3994ec5 2003-12-11 devnull break;
741 b3994ec5 2003-12-11 devnull }
742 b3994ec5 2003-12-11 devnull lc = c;
743 b3994ec5 2003-12-11 devnull }
744 b3994ec5 2003-12-11 devnull qlock(&row.lk);
745 b3994ec5 2003-12-11 devnull t = &row.tag;
746 b3994ec5 2003-12-11 devnull textcommit(t, TRUE);
747 b3994ec5 2003-12-11 devnull if(c == nil){
748 b3994ec5 2003-12-11 devnull /* helper processes use this exit status */
749 b3994ec5 2003-12-11 devnull if(strncmp(w->msg, "libthread", 9) != 0){
750 b3994ec5 2003-12-11 devnull p = emalloc(sizeof(Pid));
751 b3994ec5 2003-12-11 devnull p->pid = pid;
752 b3994ec5 2003-12-11 devnull strncpy(p->msg, w->msg, sizeof(p->msg));
753 b3994ec5 2003-12-11 devnull p->next = pids;
754 b3994ec5 2003-12-11 devnull pids = p;
755 b3994ec5 2003-12-11 devnull }
756 b3994ec5 2003-12-11 devnull }else{
757 b3994ec5 2003-12-11 devnull if(search(t, c->name, c->nname)){
758 b3994ec5 2003-12-11 devnull textdelete(t, t->q0, t->q1, TRUE);
759 b3994ec5 2003-12-11 devnull textsetselect(t, 0, 0);
760 b3994ec5 2003-12-11 devnull }
761 b3994ec5 2003-12-11 devnull if(w->msg[0])
762 334cb1e9 2004-12-27 devnull warning(c->md, "%.*S: exit %s\n", c->nname-1, c->name, w->msg);
763 b3994ec5 2003-12-11 devnull flushimage(display, 1);
764 b3994ec5 2003-12-11 devnull }
765 b3994ec5 2003-12-11 devnull qunlock(&row.lk);
766 b3994ec5 2003-12-11 devnull free(w);
767 b3994ec5 2003-12-11 devnull Freecmd:
768 b3994ec5 2003-12-11 devnull if(c){
769 b3994ec5 2003-12-11 devnull if(c->iseditcmd)
770 b3994ec5 2003-12-11 devnull sendul(cedit, 0);
771 b3994ec5 2003-12-11 devnull free(c->text);
772 b3994ec5 2003-12-11 devnull free(c->name);
773 b3994ec5 2003-12-11 devnull fsysdelid(c->md);
774 b3994ec5 2003-12-11 devnull free(c);
775 b3994ec5 2003-12-11 devnull }
776 b3994ec5 2003-12-11 devnull break;
777 b3994ec5 2003-12-11 devnull case WCmd:
778 b3994ec5 2003-12-11 devnull /* has this command already exited? */
779 b3994ec5 2003-12-11 devnull lastp = nil;
780 b3994ec5 2003-12-11 devnull for(p=pids; p!=nil; p=p->next){
781 b3994ec5 2003-12-11 devnull if(p->pid == c->pid){
782 b3994ec5 2003-12-11 devnull if(p->msg[0])
783 b3994ec5 2003-12-11 devnull warning(c->md, "%s\n", p->msg);
784 b3994ec5 2003-12-11 devnull if(lastp == nil)
785 b3994ec5 2003-12-11 devnull pids = p->next;
786 b3994ec5 2003-12-11 devnull else
787 b3994ec5 2003-12-11 devnull lastp->next = p->next;
788 b3994ec5 2003-12-11 devnull free(p);
789 b3994ec5 2003-12-11 devnull goto Freecmd;
790 b3994ec5 2003-12-11 devnull }
791 b3994ec5 2003-12-11 devnull lastp = p;
792 b3994ec5 2003-12-11 devnull }
793 b3994ec5 2003-12-11 devnull c->next = command;
794 b3994ec5 2003-12-11 devnull command = c;
795 b3994ec5 2003-12-11 devnull qlock(&row.lk);
796 b3994ec5 2003-12-11 devnull t = &row.tag;
797 b3994ec5 2003-12-11 devnull textcommit(t, TRUE);
798 b3994ec5 2003-12-11 devnull textinsert(t, 0, c->name, c->nname, TRUE);
799 b3994ec5 2003-12-11 devnull textsetselect(t, 0, 0);
800 b3994ec5 2003-12-11 devnull flushimage(display, 1);
801 b3994ec5 2003-12-11 devnull qunlock(&row.lk);
802 b3994ec5 2003-12-11 devnull break;
803 b3994ec5 2003-12-11 devnull }
804 b3994ec5 2003-12-11 devnull }
805 b3994ec5 2003-12-11 devnull }
806 b3994ec5 2003-12-11 devnull
807 b3994ec5 2003-12-11 devnull void
808 b3994ec5 2003-12-11 devnull xfidallocthread(void *v)
809 b3994ec5 2003-12-11 devnull {
810 b3994ec5 2003-12-11 devnull Xfid *xfree, *x;
811 b3994ec5 2003-12-11 devnull enum { Alloc, Free, N };
812 b3994ec5 2003-12-11 devnull static Alt alts[N+1];
813 b3994ec5 2003-12-11 devnull
814 b3994ec5 2003-12-11 devnull USED(v);
815 b3994ec5 2003-12-11 devnull threadsetname("xfidallocthread");
816 b3994ec5 2003-12-11 devnull alts[Alloc].c = cxfidalloc;
817 b3994ec5 2003-12-11 devnull alts[Alloc].v = nil;
818 b3994ec5 2003-12-11 devnull alts[Alloc].op = CHANRCV;
819 b3994ec5 2003-12-11 devnull alts[Free].c = cxfidfree;
820 b3994ec5 2003-12-11 devnull alts[Free].v = &x;
821 b3994ec5 2003-12-11 devnull alts[Free].op = CHANRCV;
822 b3994ec5 2003-12-11 devnull alts[N].op = CHANEND;
823 b3994ec5 2003-12-11 devnull
824 b3994ec5 2003-12-11 devnull xfree = nil;
825 b3994ec5 2003-12-11 devnull for(;;){
826 b3994ec5 2003-12-11 devnull switch(alt(alts)){
827 b3994ec5 2003-12-11 devnull case Alloc:
828 b3994ec5 2003-12-11 devnull x = xfree;
829 b3994ec5 2003-12-11 devnull if(x)
830 b3994ec5 2003-12-11 devnull xfree = x->next;
831 b3994ec5 2003-12-11 devnull else{
832 b3994ec5 2003-12-11 devnull x = emalloc(sizeof(Xfid));
833 b3994ec5 2003-12-11 devnull x->c = chancreate(sizeof(void(*)(Xfid*)), 0);
834 334cb1e9 2004-12-27 devnull chansetname(x->c, "xc%p", x->c);
835 b3994ec5 2003-12-11 devnull x->arg = x;
836 b3994ec5 2003-12-11 devnull threadcreate(xfidctl, x->arg, STACK);
837 b3994ec5 2003-12-11 devnull }
838 b3994ec5 2003-12-11 devnull sendp(cxfidalloc, x);
839 b3994ec5 2003-12-11 devnull break;
840 b3994ec5 2003-12-11 devnull case Free:
841 b3994ec5 2003-12-11 devnull x->next = xfree;
842 b3994ec5 2003-12-11 devnull xfree = x;
843 b3994ec5 2003-12-11 devnull break;
844 b3994ec5 2003-12-11 devnull }
845 b3994ec5 2003-12-11 devnull }
846 b3994ec5 2003-12-11 devnull }
847 b3994ec5 2003-12-11 devnull
848 b3994ec5 2003-12-11 devnull /* this thread, in the main proc, allows fsysproc to get a window made without doing graphics */
849 b3994ec5 2003-12-11 devnull void
850 b3994ec5 2003-12-11 devnull newwindowthread(void *v)
851 b3994ec5 2003-12-11 devnull {
852 b3994ec5 2003-12-11 devnull Window *w;
853 b3994ec5 2003-12-11 devnull
854 b3994ec5 2003-12-11 devnull USED(v);
855 b3994ec5 2003-12-11 devnull threadsetname("newwindowthread");
856 b3994ec5 2003-12-11 devnull
857 b3994ec5 2003-12-11 devnull for(;;){
858 b3994ec5 2003-12-11 devnull /* only fsysproc is talking to us, so synchronization is trivial */
859 b3994ec5 2003-12-11 devnull recvp(cnewwindow);
860 b3994ec5 2003-12-11 devnull w = makenewwindow(nil);
861 b3994ec5 2003-12-11 devnull winsettag(w);
862 b3994ec5 2003-12-11 devnull sendp(cnewwindow, w);
863 b3994ec5 2003-12-11 devnull }
864 b3994ec5 2003-12-11 devnull }
865 b3994ec5 2003-12-11 devnull
866 b3994ec5 2003-12-11 devnull Reffont*
867 b3994ec5 2003-12-11 devnull rfget(int fix, int save, int setfont, char *name)
868 b3994ec5 2003-12-11 devnull {
869 b3994ec5 2003-12-11 devnull Reffont *r;
870 b3994ec5 2003-12-11 devnull Font *f;
871 b3994ec5 2003-12-11 devnull int i;
872 b3994ec5 2003-12-11 devnull
873 b3994ec5 2003-12-11 devnull r = nil;
874 b3994ec5 2003-12-11 devnull if(name == nil){
875 b3994ec5 2003-12-11 devnull name = fontnames[fix];
876 b3994ec5 2003-12-11 devnull r = reffonts[fix];
877 b3994ec5 2003-12-11 devnull }
878 b3994ec5 2003-12-11 devnull if(r == nil){
879 b3994ec5 2003-12-11 devnull for(i=0; i<nfontcache; i++)
880 b3994ec5 2003-12-11 devnull if(strcmp(name, fontcache[i]->f->name) == 0){
881 b3994ec5 2003-12-11 devnull r = fontcache[i];
882 b3994ec5 2003-12-11 devnull goto Found;
883 b3994ec5 2003-12-11 devnull }
884 b3994ec5 2003-12-11 devnull f = openfont(display, name);
885 b3994ec5 2003-12-11 devnull if(f == nil){
886 b3994ec5 2003-12-11 devnull warning(nil, "can't open font file %s: %r\n", name);
887 b3994ec5 2003-12-11 devnull return nil;
888 b3994ec5 2003-12-11 devnull }
889 b3994ec5 2003-12-11 devnull r = emalloc(sizeof(Reffont));
890 b3994ec5 2003-12-11 devnull r->f = f;
891 6d7fdb24 2004-12-27 devnull fontcache = erealloc(fontcache, (nfontcache+1)*sizeof(Reffont*));
892 b3994ec5 2003-12-11 devnull fontcache[nfontcache++] = r;
893 b3994ec5 2003-12-11 devnull }
894 b3994ec5 2003-12-11 devnull Found:
895 b3994ec5 2003-12-11 devnull if(save){
896 b3994ec5 2003-12-11 devnull incref(&r->ref);
897 b3994ec5 2003-12-11 devnull if(reffonts[fix])
898 b3994ec5 2003-12-11 devnull rfclose(reffonts[fix]);
899 b3994ec5 2003-12-11 devnull reffonts[fix] = r;
900 cb1ac4c4 2008-03-07 rsc if(name != fontnames[fix]){
901 6d7fdb24 2004-12-27 devnull free(fontnames[fix]);
902 6d7fdb24 2004-12-27 devnull fontnames[fix] = estrdup(name);
903 6d7fdb24 2004-12-27 devnull }
904 b3994ec5 2003-12-11 devnull }
905 b3994ec5 2003-12-11 devnull if(setfont){
906 b3994ec5 2003-12-11 devnull reffont.f = r->f;
907 b3994ec5 2003-12-11 devnull incref(&r->ref);
908 b3994ec5 2003-12-11 devnull rfclose(reffonts[0]);
909 b3994ec5 2003-12-11 devnull font = r->f;
910 b3994ec5 2003-12-11 devnull reffonts[0] = r;
911 b3994ec5 2003-12-11 devnull incref(&r->ref);
912 b3994ec5 2003-12-11 devnull iconinit();
913 b3994ec5 2003-12-11 devnull }
914 b3994ec5 2003-12-11 devnull incref(&r->ref);
915 b3994ec5 2003-12-11 devnull return r;
916 b3994ec5 2003-12-11 devnull }
917 b3994ec5 2003-12-11 devnull
918 b3994ec5 2003-12-11 devnull void
919 b3994ec5 2003-12-11 devnull rfclose(Reffont *r)
920 b3994ec5 2003-12-11 devnull {
921 b3994ec5 2003-12-11 devnull int i;
922 b3994ec5 2003-12-11 devnull
923 b3994ec5 2003-12-11 devnull if(decref(&r->ref) == 0){
924 b3994ec5 2003-12-11 devnull for(i=0; i<nfontcache; i++)
925 b3994ec5 2003-12-11 devnull if(r == fontcache[i])
926 b3994ec5 2003-12-11 devnull break;
927 b3994ec5 2003-12-11 devnull if(i >= nfontcache)
928 b3994ec5 2003-12-11 devnull warning(nil, "internal error: can't find font in cache\n");
929 b3994ec5 2003-12-11 devnull else{
930 b3994ec5 2003-12-11 devnull nfontcache--;
931 b3994ec5 2003-12-11 devnull memmove(fontcache+i, fontcache+i+1, (nfontcache-i)*sizeof(Reffont*));
932 b3994ec5 2003-12-11 devnull }
933 b3994ec5 2003-12-11 devnull freefont(r->f);
934 b3994ec5 2003-12-11 devnull free(r);
935 b3994ec5 2003-12-11 devnull }
936 b3994ec5 2003-12-11 devnull }
937 b3994ec5 2003-12-11 devnull
938 b3994ec5 2003-12-11 devnull Cursor boxcursor = {
939 b3994ec5 2003-12-11 devnull {-7, -7},
940 b3994ec5 2003-12-11 devnull {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
941 b3994ec5 2003-12-11 devnull 0xFF, 0xFF, 0xF8, 0x1F, 0xF8, 0x1F, 0xF8, 0x1F,
942 b3994ec5 2003-12-11 devnull 0xF8, 0x1F, 0xF8, 0x1F, 0xF8, 0x1F, 0xFF, 0xFF,
943 b3994ec5 2003-12-11 devnull 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF},
944 b3994ec5 2003-12-11 devnull {0x00, 0x00, 0x7F, 0xFE, 0x7F, 0xFE, 0x7F, 0xFE,
945 b3994ec5 2003-12-11 devnull 0x70, 0x0E, 0x70, 0x0E, 0x70, 0x0E, 0x70, 0x0E,
946 b3994ec5 2003-12-11 devnull 0x70, 0x0E, 0x70, 0x0E, 0x70, 0x0E, 0x70, 0x0E,
947 b3994ec5 2003-12-11 devnull 0x7F, 0xFE, 0x7F, 0xFE, 0x7F, 0xFE, 0x00, 0x00}
948 b3994ec5 2003-12-11 devnull };
949 b3994ec5 2003-12-11 devnull
950 b3994ec5 2003-12-11 devnull void
951 b3994ec5 2003-12-11 devnull iconinit(void)
952 b3994ec5 2003-12-11 devnull {
953 b3994ec5 2003-12-11 devnull Rectangle r;
954 b3994ec5 2003-12-11 devnull Image *tmp;
955 b3994ec5 2003-12-11 devnull
956 b3994ec5 2003-12-11 devnull /* Blue */
957 b3994ec5 2003-12-11 devnull tagcols[BACK] = allocimagemix(display, DPalebluegreen, DWhite);
958 b3994ec5 2003-12-11 devnull tagcols[HIGH] = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DPalegreygreen);
959 b3994ec5 2003-12-11 devnull tagcols[BORD] = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DPurpleblue);
960 b3994ec5 2003-12-11 devnull tagcols[TEXT] = display->black;
961 b3994ec5 2003-12-11 devnull tagcols[HTEXT] = display->black;
962 b3994ec5 2003-12-11 devnull
963 b3994ec5 2003-12-11 devnull /* Yellow */
964 b3994ec5 2003-12-11 devnull textcols[BACK] = allocimagemix(display, DPaleyellow, DWhite);
965 b3994ec5 2003-12-11 devnull textcols[HIGH] = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DDarkyellow);
966 b3994ec5 2003-12-11 devnull textcols[BORD] = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DYellowgreen);
967 b3994ec5 2003-12-11 devnull textcols[TEXT] = display->black;
968 b3994ec5 2003-12-11 devnull textcols[HTEXT] = display->black;
969 b3994ec5 2003-12-11 devnull
970 b3994ec5 2003-12-11 devnull if(button){
971 b3994ec5 2003-12-11 devnull freeimage(button);
972 b3994ec5 2003-12-11 devnull freeimage(modbutton);
973 b3994ec5 2003-12-11 devnull freeimage(colbutton);
974 b3994ec5 2003-12-11 devnull }
975 b3994ec5 2003-12-11 devnull
976 b3994ec5 2003-12-11 devnull r = Rect(0, 0, Scrollwid+2, font->height+1);
977 b3994ec5 2003-12-11 devnull button = allocimage(display, r, screen->chan, 0, DNofill);
978 b3994ec5 2003-12-11 devnull draw(button, r, tagcols[BACK], nil, r.min);
979 b3994ec5 2003-12-11 devnull r.max.x -= 2;
980 b3994ec5 2003-12-11 devnull border(button, r, 2, tagcols[BORD], ZP);
981 b3994ec5 2003-12-11 devnull
982 b3994ec5 2003-12-11 devnull r = button->r;
983 b3994ec5 2003-12-11 devnull modbutton = allocimage(display, r, screen->chan, 0, DNofill);
984 b3994ec5 2003-12-11 devnull draw(modbutton, r, tagcols[BACK], nil, r.min);
985 b3994ec5 2003-12-11 devnull r.max.x -= 2;
986 b3994ec5 2003-12-11 devnull border(modbutton, r, 2, tagcols[BORD], ZP);
987 b3994ec5 2003-12-11 devnull r = insetrect(r, 2);
988 b3994ec5 2003-12-11 devnull tmp = allocimage(display, Rect(0,0,1,1), screen->chan, 1, DMedblue);
989 b3994ec5 2003-12-11 devnull draw(modbutton, r, tmp, nil, ZP);
990 b3994ec5 2003-12-11 devnull freeimage(tmp);
991 b3994ec5 2003-12-11 devnull
992 b3994ec5 2003-12-11 devnull r = button->r;
993 b3994ec5 2003-12-11 devnull colbutton = allocimage(display, r, screen->chan, 0, DPurpleblue);
994 b3994ec5 2003-12-11 devnull
995 b3994ec5 2003-12-11 devnull but2col = allocimage(display, r, screen->chan, 1, 0xAA0000FF);
996 b3994ec5 2003-12-11 devnull but3col = allocimage(display, r, screen->chan, 1, 0x006600FF);
997 b3994ec5 2003-12-11 devnull }
998 b3994ec5 2003-12-11 devnull
999 b3994ec5 2003-12-11 devnull /*
1000 b3994ec5 2003-12-11 devnull * /dev/snarf updates when the file is closed, so we must open our own
1001 b3994ec5 2003-12-11 devnull * fd here rather than use snarffd
1002 b3994ec5 2003-12-11 devnull */
1003 b3994ec5 2003-12-11 devnull
1004 b3994ec5 2003-12-11 devnull /* rio truncates larges snarf buffers, so this avoids using the
1005 b3994ec5 2003-12-11 devnull * service if the string is huge */
1006 b3994ec5 2003-12-11 devnull
1007 b3994ec5 2003-12-11 devnull #define MAXSNARF 100*1024
1008 b3994ec5 2003-12-11 devnull
1009 b3994ec5 2003-12-11 devnull void
1010 b3994ec5 2003-12-11 devnull acmeputsnarf(void)
1011 b3994ec5 2003-12-11 devnull {
1012 5a8e63b2 2004-02-29 devnull int i, n;
1013 5a8e63b2 2004-02-29 devnull Fmt f;
1014 5a8e63b2 2004-02-29 devnull char *s;
1015 b3994ec5 2003-12-11 devnull
1016 5a8e63b2 2004-02-29 devnull if(snarfbuf.nc==0)
1017 b3994ec5 2003-12-11 devnull return;
1018 b3994ec5 2003-12-11 devnull if(snarfbuf.nc > MAXSNARF)
1019 b3994ec5 2003-12-11 devnull return;
1020 5a8e63b2 2004-02-29 devnull
1021 5a8e63b2 2004-02-29 devnull fmtstrinit(&f);
1022 b3994ec5 2003-12-11 devnull for(i=0; i<snarfbuf.nc; i+=n){
1023 b3994ec5 2003-12-11 devnull n = snarfbuf.nc-i;
1024 b3994ec5 2003-12-11 devnull if(n >= NSnarf)
1025 b3994ec5 2003-12-11 devnull n = NSnarf;
1026 b3994ec5 2003-12-11 devnull bufread(&snarfbuf, i, snarfrune, n);
1027 5a8e63b2 2004-02-29 devnull if(fmtprint(&f, "%.*S", n, snarfrune) < 0)
1028 b3994ec5 2003-12-11 devnull break;
1029 b3994ec5 2003-12-11 devnull }
1030 5a8e63b2 2004-02-29 devnull s = fmtstrflush(&f);
1031 5a8e63b2 2004-02-29 devnull if(s && s[0])
1032 5a8e63b2 2004-02-29 devnull putsnarf(s);
1033 5a8e63b2 2004-02-29 devnull free(s);
1034 b3994ec5 2003-12-11 devnull }
1035 b3994ec5 2003-12-11 devnull
1036 b3994ec5 2003-12-11 devnull void
1037 5a8e63b2 2004-02-29 devnull acmegetsnarf(void)
1038 b3994ec5 2003-12-11 devnull {
1039 5a8e63b2 2004-02-29 devnull char *s;
1040 5a8e63b2 2004-02-29 devnull int nb, nr, nulls, len;
1041 5a8e63b2 2004-02-29 devnull Rune *r;
1042 b3994ec5 2003-12-11 devnull
1043 5a8e63b2 2004-02-29 devnull s = getsnarf();
1044 5a8e63b2 2004-02-29 devnull if(s == nil || s[0]==0){
1045 5a8e63b2 2004-02-29 devnull free(s);
1046 b3994ec5 2003-12-11 devnull return;
1047 5a8e63b2 2004-02-29 devnull }
1048 5a8e63b2 2004-02-29 devnull
1049 5a8e63b2 2004-02-29 devnull len = strlen(s);
1050 5a8e63b2 2004-02-29 devnull r = runemalloc(len+1);
1051 5a8e63b2 2004-02-29 devnull cvttorunes(s, len, r, &nb, &nr, &nulls);
1052 b3994ec5 2003-12-11 devnull bufreset(&snarfbuf);
1053 5a8e63b2 2004-02-29 devnull bufinsert(&snarfbuf, 0, r, nr);
1054 5a8e63b2 2004-02-29 devnull free(r);
1055 5a8e63b2 2004-02-29 devnull free(s);
1056 cd5a7378 2008-04-17 rsc }
1057 cd5a7378 2008-04-17 rsc
1058 cd5a7378 2008-04-17 rsc int
1059 cd5a7378 2008-04-17 rsc ismtpt(char *file)
1060 cd5a7378 2008-04-17 rsc {
1061 cd5a7378 2008-04-17 rsc int n;
1062 cd5a7378 2008-04-17 rsc
1063 5d32c407 2008-05-08 rsc if(mtpt == nil)
1064 5d32c407 2008-05-08 rsc return 0;
1065 5d32c407 2008-05-08 rsc
1066 cd5a7378 2008-04-17 rsc /* This is not foolproof, but it will stop a lot of them. */
1067 cd5a7378 2008-04-17 rsc n = strlen(mtpt);
1068 cd5a7378 2008-04-17 rsc return strncmp(file, mtpt, n) == 0 && ((n > 0 && mtpt[n-1] == '/') || file[n] == '/' || file[n] == 0);
1069 b3994ec5 2003-12-11 devnull }