ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/rxvt-unicode/src/rxvtd.C
Revision: 1.30
Committed: Wed Aug 9 22:01:07 2006 UTC (17 years, 9 months ago) by root
Content type: text/plain
Branch: MAIN
Changes since 1.29: +5 -2 lines
Log Message:
*** empty log message ***

File Contents

# User Rev Content
1 root 1.29 /*----------------------------------------------------------------------*
2 pcg 1.14 * File: rxvtd.C
3     *----------------------------------------------------------------------*
4     *
5     * All portions of code are copyright by their respective author/s.
6 root 1.28 * Copyright (c) 2003-2006 Marc Lehmann <pcg@goof.com>
7 pcg 1.14 *
8     * This program is free software; you can redistribute it and/or modify
9     * it under the terms of the GNU General Public License as published by
10     * the Free Software Foundation; either version 2 of the License, or
11     * (at your option) any later version.
12     *
13     * This program is distributed in the hope that it will be useful,
14     * but WITHOUT ANY WARRANTY; without even the implied warranty of
15     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16     * GNU General Public License for more details.
17     *
18     * You should have received a copy of the GNU General Public License
19     * along with this program; if not, write to the Free Software
20     * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21     *----------------------------------------------------------------------*/
22    
23 pcg 1.9 #include "../config.h"
24 pcg 1.1 #include <cstdio>
25     #include <cstdlib>
26 pcg 1.3 #include <cstdarg>
27     #include <cstring>
28 pcg 1.1
29     #include <unistd.h>
30 pcg 1.12 #include <fcntl.h>
31 pcg 1.5 #include <sys/types.h>
32     #include <sys/stat.h>
33 pcg 1.1 #include <sys/socket.h>
34     #include <sys/un.h>
35    
36 pcg 1.3 #include <cerrno>
37    
38 root 1.27 #include "rxvt.h"
39     #include "rxvtdaemon.h"
40     #include "libptytty.h"
41     #include "iom.h"
42 pcg 1.3
43 pcg 1.1 struct server : rxvt_connection {
44 pcg 1.10 log_callback log_cb;
45 root 1.22 getfd_callback getfd_cb;
46 pcg 1.10
47 pcg 1.1 void read_cb (io_watcher &w, short revents); io_watcher read_ev;
48 pcg 1.10 void log_msg (const char *msg);
49 root 1.22 int getfd (int remote_fd);
50 pcg 1.1
51     server (int fd)
52 pcg 1.10 : read_ev (this, &server::read_cb),
53 root 1.22 log_cb (this, &server::log_msg),
54     getfd_cb (this, &server::getfd)
55 pcg 1.1 {
56     this->fd = fd;
57     read_ev.start (fd, EVENT_READ);
58     }
59    
60 pcg 1.3 void err (const char *format = 0, ...);
61 pcg 1.1 };
62    
63 pcg 1.5 struct unix_listener {
64 pcg 1.1 int fd;
65    
66     void accept_cb (io_watcher &w, short revents); io_watcher accept_ev;
67    
68 pcg 1.5 unix_listener (const char *sockname);
69 pcg 1.1 };
70    
71 pcg 1.5 unix_listener::unix_listener (const char *sockname)
72     : accept_ev (this, &unix_listener::accept_cb)
73 pcg 1.1 {
74 root 1.30 sockaddr_un sa;
75    
76     if (strlen(sockname) >= sizeof(sa.sun_path))
77     rxvt_fatal ("socket name too long, aborting.\n");
78    
79 root 1.24 if ((fd = socket (AF_UNIX, SOCK_STREAM, 0)) < 0)
80 pcg 1.1 {
81     perror ("unable to create listening socket");
82     exit (EXIT_FAILURE);
83     }
84    
85 pcg 1.12 fcntl (fd, F_SETFD, FD_CLOEXEC);
86    
87 pcg 1.1 sa.sun_family = AF_UNIX;
88 pcg 1.5 strcpy (sa.sun_path, sockname);
89 pcg 1.1
90     unlink (rxvt_connection::unix_sockname ());
91    
92 pcg 1.5 mode_t omask = umask (0077);
93    
94 pcg 1.1 if (bind (fd, (sockaddr *)&sa, sizeof (sa)))
95     {
96     perror ("unable to bind listening socket");
97     exit (EXIT_FAILURE);
98     }
99    
100 pcg 1.5 umask (omask);
101    
102 pcg 1.1 if (listen (fd, 5))
103     {
104     perror ("unable to listen on socket");
105     exit (EXIT_FAILURE);
106     }
107    
108     accept_ev.start (fd, EVENT_READ);
109     }
110    
111 pcg 1.5 void unix_listener::accept_cb (io_watcher &w, short revents)
112 pcg 1.1 {
113     int fd2 = accept (fd, 0, 0);
114    
115     if (fd2 >= 0)
116 pcg 1.12 {
117     fcntl (fd2, F_SETFD, FD_CLOEXEC);
118     new server (fd2);
119     }
120 pcg 1.1 }
121    
122 root 1.22 int server::getfd (int remote_fd)
123     {
124     send ("GETFD");
125     send (remote_fd);
126 root 1.27 return ptytty::recv_fd (fd);
127 root 1.22 }
128    
129 pcg 1.10 void server::log_msg (const char *msg)
130     {
131     send ("MSG"), send (msg);
132     }
133    
134 pcg 1.3 void server::err (const char *format, ...)
135 pcg 1.1 {
136 pcg 1.3 if (format)
137     {
138     char err[1024];
139    
140     va_list ap;
141     va_start (ap, format);
142     vsnprintf (err, 1024, format, ap);
143     va_end (ap);
144    
145 pcg 1.10 send ("MSG"), send (err);
146 pcg 1.3 }
147    
148 pcg 1.10 send ("END", 0);
149 pcg 1.1 close (fd);
150     delete this;
151     }
152    
153     void server::read_cb (io_watcher &w, short revents)
154     {
155 pcg 1.2 auto_str tok;
156 pcg 1.1
157 pcg 1.2 if (recv (tok))
158 pcg 1.1 {
159 pcg 1.2 if (!strcmp (tok, "NEW"))
160 pcg 1.1 {
161 pcg 1.9 stringvec *argv = new stringvec;
162     stringvec *envv = new stringvec;
163 pcg 1.2
164     for (;;)
165     {
166     if (!recv (tok))
167     return err ();
168    
169     if (!strcmp (tok, "END"))
170     break;
171 pcg 1.3 else if (!strcmp (tok, "ENV") && recv (tok))
172 root 1.26 envv->push_back (strdup (tok));
173 pcg 1.3 else if (!strcmp (tok, "CWD") && recv (tok))
174     {
175     if (chdir (tok))
176     err ("unable to change to working directory to '%s': %s",
177     (char *)tok, strerror (errno));
178     }
179 pcg 1.2 else if (!strcmp (tok, "ARG") && recv (tok))
180 root 1.26 argv->push_back (strdup (tok));
181 pcg 1.2 else
182 pcg 1.3 return err ("protocol error: unexpected NEW token");
183 pcg 1.2 }
184    
185 pcg 1.9 envv->push_back (0);
186 pcg 1.3
187     {
188 pcg 1.10 rxvt_term *term = new rxvt_term;
189    
190     term->log_hook = &log_cb;
191 root 1.22 term->getfd_hook = &getfd_cb;
192 pcg 1.2
193 pcg 1.10 bool success;
194    
195     try
196     {
197 root 1.26 success = term->init (argv, envv);
198 pcg 1.10 }
199     catch (const class rxvt_failure_exception &e)
200     {
201     success = false;
202     }
203    
204 root 1.15 term->log_hook = 0;
205    
206 root 1.20 chdir ("/");
207 pcg 1.10
208     if (!success)
209     term->destroy ();
210    
211     send ("END"); send (success ? 1 : 0);
212 pcg 1.3 }
213 pcg 1.1 }
214     else
215 pcg 1.3 return err ("protocol error: request '%s' unsupported.", (char *)tok);
216 pcg 1.1 }
217     else
218 pcg 1.2 return err ();
219 pcg 1.1 }
220    
221 root 1.21 int opt_fork, opt_opendisplay, opt_quiet;
222    
223 pcg 1.1 int
224 pcg 1.8 main (int argc, const char *const *argv)
225 pcg 1.1 {
226 root 1.25 rxvt_init ();
227    
228 root 1.21 for (int i = 1; i < argc; i++)
229     {
230     if (!strcmp (argv [i], "-f") || !strcmp (argv [i], "--fork"))
231     opt_fork = 1;
232     else if (!strcmp (argv [i], "-o") || !strcmp (argv [i], "--opendisplay"))
233     opt_opendisplay = 1;
234     else if (!strcmp (argv [i], "-q") || !strcmp (argv [i], "--quiet"))
235     opt_quiet = 1;
236     else
237     {
238     rxvt_log ("%s: unknown option '%s', aborting.\n", argv [0], argv [i]);
239     return EXIT_FAILURE;
240     }
241     }
242    
243 root 1.20 chdir ("/");
244    
245 root 1.21 if (opt_opendisplay)
246     displays.get (getenv ("DISPLAY")); // open display and never release it
247    
248 pcg 1.5 char *sockname = rxvt_connection::unix_sockname ();
249     unix_listener l (sockname);
250 root 1.21
251     if (!opt_quiet)
252     {
253     printf ("rxvt-unicode daemon listening on %s.\n", sockname);
254     fflush (stdout);
255     }
256    
257 pcg 1.5 free (sockname);
258    
259 root 1.21 if (opt_fork)
260     {
261     pid_t pid = fork ();
262    
263     if (pid < 0)
264     {
265     rxvt_log ("unable to fork daemon, aborting.\n");
266     return EXIT_FAILURE;
267     }
268     else if (pid > 0)
269     _exit (EXIT_SUCCESS);
270     }
271    
272 root 1.18 io_manager::loop ();
273 pcg 1.1
274     return EXIT_SUCCESS;
275     }
276 pcg 1.7