chiark / gitweb /
1a7ecd77a0eb5ef7e7c9119b2f8f71763d91d784
[xtoys] / xwait.c
1 /* -*-c-*-
2  *
3  * $Id: xwait.c,v 1.4 1998/11/21 22:41:19 mdw Exp $
4  *
5  * Wait until prodded by another X client
6  *
7  * (c) 1998 Straylight/Edgeware
8  */
9
10 /*----- Licensing notice --------------------------------------------------* 
11  *
12  * This file is part of the Edgeware X tools collection.
13  *
14  * X tools is free software; you can redistribute it and/or modify
15  * it under the terms of the GNU General Public License as published by
16  * the Free Software Foundation; either version 2 of the License, or
17  * (at your option) any later version.
18  * 
19  * X tools is distributed in the hope that it will be useful,
20  * but WITHOUT ANY WARRANTY; without even the implied warranty of
21  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
22  * GNU General Public License for more details.
23  * 
24  * You should have received a copy of the GNU General Public License
25  * along with X tools; if not, write to the Free Software Foundation,
26  * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
27  */
28
29 /*----- Revision history --------------------------------------------------* 
30  *
31  * $Log: xwait.c,v $
32  * Revision 1.4  1998/11/21 22:41:19  mdw
33  * Reap children which die before I get my signal handler installed.
34  *
35  * Revision 1.3  1998/11/21 22:30:27  mdw
36  * Support GNU-style long options throughout, and introduce proper help
37  * text to all programs.  Update manual pages to match.
38  *
39  * Revision 1.2  1998/11/18 21:25:06  mdw
40  * Reap dead children as they arrive.  The previous shell may have
41  * carelessly left them behind.
42  *
43  * Revision 1.1  1998/11/16 23:00:49  mdw
44  * Initial versions.
45  *
46  */
47
48 /*----- Header files ------------------------------------------------------*/
49
50 #include <signal.h>
51 #include <stdio.h>
52 #include <stdlib.h>
53 #include <string.h>
54
55 #include <sys/types.h>
56 #include <sys/wait.h>
57 #include <unistd.h>
58
59 #include <X11/Xlib.h>
60 #include <X11/Xutil.h>
61
62 #include "mdwopt.h"
63 #include "quis.h"
64 #include "xwait.h"
65
66 /*----- Main code ---------------------------------------------------------*/
67
68 /* --- @sigchld@ --- */
69
70 static void sigchld(int sig)
71 {
72   while (waitpid(-1, 0, WNOHANG) > 0)
73     ;
74 }
75
76 /* --- @main@ --- */
77
78 static void version(FILE *fp)
79 {
80   fprintf(fp, "%s (xtoys version " VERSION ")\n", QUIS);
81 }
82
83 static void usage(FILE *fp)
84 {
85   fprintf(fp, "Usage: %s [-f] [-d DISPLAY] [-a ATOM] [-m MSG]\n", QUIS);
86 }
87
88 int main(int argc, char *argv[])
89 {
90   char *display = 0;
91   Display *dpy;
92   Atom xwait_die;
93   XEvent ev;
94   char *atom = XWAIT_DIE;
95   char *msg = XWAIT_DIE_MSG;
96   unsigned f = 0;
97
98   enum {
99     f_force = 1
100   };
101
102   /* --- Parse options --- */
103
104   ego(argv[0]);
105
106   for (;;) {
107     static struct option opt[] = {
108       { "help",         0,                      0,      'h' },
109       { "usage",        0,                      0,      'u' },
110       { "version",      0,                      0,      'v' },
111       { "display",      required_argument,      0,      'd' },
112       { "atom",         required_argument,      0,      'a' },
113       { "msg",          required_argument,      0,      'm' },
114       { "force",        0,                      0,      'f' },
115       { 0,              0,                      0,      0 }
116     };
117
118     int i = getopt_long(argc, argv, "d:a:m:f", opt, 0);
119     if (i < 0)
120       break;
121     switch (i) {
122       case 'h':
123         version(stdout);
124         fputs("\n", stdout);
125         usage(stdout);
126         fputs(
127 "\n"
128 "Waits until signalled by `xtell' or `xshutdown'.  Specifically, waits\n"
129 "until a property with name ATOM is written to the root window with\n"
130 "contents MSG.\n"
131 "\n"
132 "Options:\n"
133 "\n"
134 "-h, --help             Display this help text\n"
135 "-u, --usage            Display a short usage summary\n"
136 "-v, --version          Display the program's version number\n"
137 "\n"
138 "-d, --display=DISPLAY  Choose X display to connect to\n"
139 "-f, --force            Run even if this property is waited for by another\n"
140 "                       process\n"
141 "-a, --atom=ATOM                Choose property name to listen for\n"
142 "-m, --msg=MSG          Choose value of property to wait for\n",
143           stdout);
144         exit(0);
145         break;
146       case 'u':
147         usage(stdout);
148         exit(0);
149         break;
150       case 'v':
151         version(stdout);
152         exit(0);
153         break;
154         
155       case 'd':
156         display = optarg;
157         break;
158       case 'a':
159         atom = optarg;
160         break;
161       case 'm':
162         msg = optarg;
163         break;
164       case 'f':
165         f |= f_force;
166         break;
167       default:
168         usage(stderr);
169         exit(EXIT_FAILURE);
170         break;
171     }
172   }
173
174   /* --- Connect to the X display --- */
175
176   dpy = XOpenDisplay(display);
177   if (!dpy) {
178     fprintf(stderr, "xwait: couldn't open display\n");
179     exit(EXIT_FAILURE);
180   }
181
182   /* --- Fetch the property name atom --- */
183
184   xwait_die = XInternAtom(dpy, atom, False);
185
186   /* --- Mark ourselves as listening to all the screens --- */
187
188   {
189     int i;
190     int nsc = ScreenCount(dpy);
191
192     /* --- First pass: make sure there's not a process already here --- */
193
194     if ((f & f_force) == 0) {
195       for (i = 0; i < nsc; i++) {
196         Window win = RootWindow(dpy, i);
197         XTextProperty prop;
198
199         if (XGetTextProperty(dpy, win, &prop, xwait_die)) {
200           fprintf(stderr, "xwait: already waiting for `%s'\n", atom);
201           exit(EXIT_FAILURE);
202         }
203       }
204     }
205
206     /* --- Second pass: set up listening to the property --- */
207
208     for (i = 0; i < nsc; i++) {
209       Window win = RootWindow(dpy, i);
210       XTextProperty prop;
211       char *imsg = "XWAIT_READY";
212
213       XStringListToTextProperty(&imsg, 1, &prop);
214       XSetTextProperty(dpy, win, &prop, xwait_die);
215       XSelectInput(dpy, win, PropertyChangeMask);
216     }
217   }
218
219   /* --- Set up a handler when children die --- *
220    *
221    * I don't fork any children?  Why is this useful?  Because I've been
222    * execed from a shell which started lots of background processes, and
223    * they'll zombie themselves otherwise.
224    */
225
226   {
227     struct sigaction sa;
228     sigset_t ss, oss;
229
230     /* --- Set the handler up --- */
231
232     sa.sa_handler = sigchld;
233     sigemptyset(&sa.sa_mask);
234     sigaddset(&sa.sa_mask, SIGCHLD);
235     sa.sa_flags = 0;
236     sigaction(SIGCHLD, &sa, 0);
237
238     /* --- Now reap any which have been waiting around so far --- */
239
240     sigemptyset(&ss);
241     sigaddset(&ss, SIGCHLD);
242     sigprocmask(SIG_BLOCK, &ss, &oss);
243     sigchld(SIGCHLD);
244     sigprocmask(SIG_SETMASK, &oss, 0);
245   }
246
247   /* --- Now wait for an event --- */
248
249   for (;;) {
250     XNextEvent(dpy, &ev);
251     switch (ev.type) {
252       case PropertyNotify:
253         if (ev.xproperty.atom == xwait_die) {
254           XTextProperty prop;
255           char **sl;
256           int c;
257
258           if (XGetTextProperty(dpy, ev.xproperty.window, &prop, xwait_die)) {
259             XTextPropertyToStringList(&prop, &sl, &c);
260             if (strcmp(sl[0], msg) == 0)
261               goto exit;
262             XFreeStringList(sl);
263           }
264         }
265     }
266   }
267
268   /* --- Finished: remove the property from all the screens --- */
269
270 exit:
271   {
272     int i;
273     int nsc = ScreenCount(dpy);
274
275     for (i = 0; i < nsc; i++)
276       XDeleteProperty(dpy, RootWindow(dpy, i), xwait_die);
277   }
278
279   /* --- Go away --- */
280
281   XCloseDisplay(dpy);
282   return (0);
283 }
284
285 /*----- That's all, folks -------------------------------------------------*/