ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/gvpe/src/util.C
Revision: 1.15
Committed: Fri Jun 11 15:56:34 2004 UTC (19 years, 11 months ago) by pcg
Content type: text/plain
Branch: MAIN
CVS Tags: rel-1_7
Changes since 1.14: +5 -5 lines
Log Message:
*** empty log message ***

File Contents

# Content
1 /*
2 util.C -- process management and other utility functions
3 Copyright (C) 2003-2004 Marc Lehmann <pcg@goof.com>
4
5 Some of these are taken from tinc, see the AUTHORS file.
6
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2 of the License, or
10 (at your option) any later version.
11
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
16
17 You should have received a copy of the GNU General Public License
18 along with this program; if not, write to the Free Software
19 Foundation, Inc. 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 */
21
22 #include "config.h"
23
24 #include <cstdio>
25 #include <cstdlib>
26 #include <cstring>
27
28 #include <errno.h>
29 #include <signal.h>
30 #include <sys/types.h>
31 #include <sys/wait.h>
32 #include <unistd.h>
33 #include <time.h>
34
35 #include "netcompat.h"
36
37 #include "gettext.h"
38 #include "pidfile.h"
39 #include "dropin.h"
40
41 #include "global.h"
42 #include "conf.h"
43 #include "util.h"
44 #include "slog.h"
45
46 int
47 write_pidfile (void)
48 {
49 int pid;
50
51 pid = check_pid (conf.pidfilename);
52
53 if (pid)
54 {
55 fprintf (stderr, _("A gvpe daemon is already running with pid %d.\n"), pid);
56 return 1;
57 }
58
59 /* if it's locked, write-protected, or whatever */
60 if (!write_pid (conf.pidfilename))
61 return 1;
62
63 return 0;
64 }
65
66 int
67 kill_other (int signal)
68 {
69 int pid;
70
71 pid = read_pid (conf.pidfilename);
72
73 if (!pid)
74 {
75 fprintf (stderr, _("No other gvpe daemon is running.\n"));
76 return 1;
77 }
78
79 errno = 0; /* No error, sometimes errno is only changed on error */
80
81 /* ESRCH is returned when no process with that pid is found */
82 if (kill (pid, signal) && errno == ESRCH)
83 {
84 fprintf (stderr, _("The gvpe daemon is no longer running. "));
85
86 fprintf (stderr, _("Removing stale lock file.\n"));
87 remove_pid (conf.pidfilename);
88 }
89
90 return 0;
91 }
92
93 int
94 detach (int do_detach)
95 {
96 /* First check if we can open a fresh new pidfile */
97
98 if (write_pidfile ())
99 return -1;
100
101 /* If we succeeded in doing that, detach */
102
103 log_to (0);
104
105 if (do_detach)
106 {
107 if (daemon (0, 0) < 0)
108 {
109 log_to (LOGTO_SYSLOG | LOGTO_STDERR);
110
111 slog (L_ERR, _("couldn't detach from terminal: %s"), strerror (errno));
112 return -1;
113 }
114
115 /* Now UPDATE the pid in the pidfile, because we changed it... */
116
117 if (!write_pid (conf.pidfilename))
118 return -1;
119
120 log_to (LOGTO_SYSLOG);
121 }
122 else
123 log_to (LOGTO_SYSLOG | LOGTO_STDERR);
124
125 slog (L_INFO, _("gvpe daemon %s (%s %s) starting"), VERSION, __DATE__, __TIME__);
126
127 return 0;
128 }
129
130 void run_script (const run_script_cb &cb, bool wait)
131 {
132 int pid;
133
134 if ((pid = fork ()) == 0)
135 {
136 char *filename;
137 asprintf (&filename, "%s/%s", confbase, cb());
138 execl (filename, filename, (char *) 0);
139 exit (126);
140 }
141 else if (pid > 0)
142 {
143 if (wait)
144 {
145 waitpid (pid, 0, 0);
146 /* TODO: check status */
147 }
148 }
149 }
150
151 #if ENABLE_HTTP_PROXY
152 // works like strdup
153 u8 *
154 base64_encode (const u8 *data, unsigned int len)
155 {
156 const static char base64[65] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
157
158 unsigned int t, i;
159 const u8 *end = data + len;
160 u8 *res = new u8 [4 * ((len + 2) / 3) + 1];
161 u8 *out = res;
162
163 while (data <= end - 3)
164 {
165 t = (((data[0] << 8) | data[1]) << 8) | data[2];
166 data += 3;
167
168 *out++ = base64[(t >> 18) & 0x3f];
169 *out++ = base64[(t >> 12) & 0x3f];
170 *out++ = base64[(t >> 6) & 0x3f];
171 *out++ = base64[(t ) & 0x3f];
172 }
173
174 for (t = 0, i = 0; data < end; i++)
175 t = (t << 8) | *data++;
176
177 switch (i)
178 {
179 case 2:
180 *out++ = base64[(t >> 10) & 0x3f];
181 *out++ = base64[(t >> 4) & 0x3f];
182 *out++ = base64[(t << 2) & 0x3f];
183 *out++ = '=';
184 break;
185 case 1:
186 *out++ = base64[(t >> 2) & 0x3f];
187 *out++ = base64[(t << 4) & 0x3f];
188 *out++ = '=';
189 *out++ = '=';
190 break;
191 }
192
193 *out++ = 0;
194
195 return res;
196 }
197 #endif
198
199 void
200 id2mac (unsigned int id, void *m)
201 {
202 mac &p = *(mac *)m;
203
204 if (id)
205 {
206 p[0] = 0xfe;
207 p[1] = 0xfd;
208 p[2] = 0x80;
209 p[3] = 0x00;
210 p[4] = id >> 8;
211 p[5] = id;
212 }
213 else
214 {
215 p[0] = 0xff;
216 p[1] = 0xff;
217 p[2] = 0xff;
218 p[3] = 0xff;
219 p[4] = 0xff;
220 p[5] = 0xff;
221 }
222 }
223