rump_allserver.c revision 1.24 1 1.24 pooka /* $NetBSD: rump_allserver.c,v 1.24 2013/09/10 17:59:52 pooka Exp $ */
2 1.1 pooka
3 1.1 pooka /*-
4 1.16 pooka * Copyright (c) 2010, 2011 Antti Kantee. All Rights Reserved.
5 1.1 pooka *
6 1.1 pooka * Redistribution and use in source and binary forms, with or without
7 1.1 pooka * modification, are permitted provided that the following conditions
8 1.1 pooka * are met:
9 1.1 pooka * 1. Redistributions of source code must retain the above copyright
10 1.1 pooka * notice, this list of conditions and the following disclaimer.
11 1.1 pooka * 2. Redistributions in binary form must reproduce the above copyright
12 1.1 pooka * notice, this list of conditions and the following disclaimer in the
13 1.1 pooka * documentation and/or other materials provided with the distribution.
14 1.1 pooka *
15 1.1 pooka * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
16 1.1 pooka * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
17 1.1 pooka * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
18 1.1 pooka * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19 1.1 pooka * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20 1.1 pooka * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
21 1.1 pooka * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22 1.1 pooka * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23 1.1 pooka * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24 1.1 pooka * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25 1.1 pooka * SUCH DAMAGE.
26 1.1 pooka */
27 1.1 pooka
28 1.24 pooka #include <rump/rumpuser_port.h>
29 1.24 pooka
30 1.1 pooka #include <sys/cdefs.h>
31 1.1 pooka #ifndef lint
32 1.24 pooka __RCSID("$NetBSD: rump_allserver.c,v 1.24 2013/09/10 17:59:52 pooka Exp $");
33 1.1 pooka #endif /* !lint */
34 1.1 pooka
35 1.1 pooka #include <sys/types.h>
36 1.4 pooka #include <sys/signal.h>
37 1.24 pooka #include <sys/stat.h>
38 1.24 pooka
39 1.24 pooka #ifdef PLATFORM_HAS_NBMODULES
40 1.7 pooka #include <sys/module.h>
41 1.24 pooka #endif
42 1.24 pooka #ifdef PLATFORM_HAS_DISKLABEL
43 1.24 pooka #include <sys/disklabel.h>
44 1.24 pooka #include <util.h>
45 1.24 pooka #endif
46 1.1 pooka
47 1.6 pooka #include <dlfcn.h>
48 1.1 pooka #include <err.h>
49 1.1 pooka #include <errno.h>
50 1.9 pooka #include <fcntl.h>
51 1.4 pooka #include <semaphore.h>
52 1.1 pooka #include <stdio.h>
53 1.1 pooka #include <stdlib.h>
54 1.24 pooka #include <stdint.h>
55 1.1 pooka #include <string.h>
56 1.1 pooka #include <unistd.h>
57 1.24 pooka
58 1.24 pooka #include <rump/rump.h>
59 1.24 pooka #include <rump/rump_syscalls.h>
60 1.1 pooka
61 1.22 joerg __dead static void
62 1.1 pooka usage(void)
63 1.1 pooka {
64 1.1 pooka
65 1.24 pooka #ifndef PLATFORM_HAS_SETGETPROGNAME
66 1.24 pooka #define getprogname() "rump_server"
67 1.24 pooka #endif
68 1.13 wiz fprintf(stderr, "usage: %s [-s] [-c ncpu] [-d drivespec] [-l libs] "
69 1.12 pooka "[-m modules] bindurl\n", getprogname());
70 1.1 pooka exit(1);
71 1.1 pooka }
72 1.1 pooka
73 1.22 joerg __dead static void
74 1.2 pooka die(int sflag, int error, const char *reason)
75 1.2 pooka {
76 1.2 pooka
77 1.2 pooka warnx("%s: %s", reason, strerror(error));
78 1.2 pooka if (!sflag)
79 1.2 pooka rump_daemonize_done(error);
80 1.2 pooka exit(1);
81 1.2 pooka }
82 1.2 pooka
83 1.4 pooka static sem_t sigsem;
84 1.4 pooka static void
85 1.4 pooka sigreboot(int sig)
86 1.4 pooka {
87 1.4 pooka
88 1.4 pooka sem_post(&sigsem);
89 1.4 pooka }
90 1.4 pooka
91 1.9 pooka static const char *const disktokens[] = {
92 1.9 pooka #define DKEY 0
93 1.9 pooka "key",
94 1.9 pooka #define DFILE 1
95 1.9 pooka "hostpath",
96 1.9 pooka #define DSIZE 2
97 1.18 pooka #define DSIZE_E -1
98 1.9 pooka "size",
99 1.14 pooka #define DOFFSET 3
100 1.14 pooka "offset",
101 1.16 pooka #define DLABEL 4
102 1.16 pooka "disklabel",
103 1.17 pooka #define DTYPE 5
104 1.17 pooka "type",
105 1.9 pooka NULL
106 1.9 pooka };
107 1.9 pooka
108 1.9 pooka struct etfsreg {
109 1.9 pooka const char *key;
110 1.9 pooka const char *hostpath;
111 1.9 pooka off_t flen;
112 1.14 pooka off_t foffset;
113 1.16 pooka char partition;
114 1.9 pooka enum rump_etfs_type type;
115 1.9 pooka };
116 1.9 pooka
117 1.17 pooka struct etfstype {
118 1.17 pooka const char *name;
119 1.17 pooka enum rump_etfs_type type;
120 1.17 pooka } etfstypes[] = {
121 1.17 pooka { "blk", RUMP_ETFS_BLK },
122 1.17 pooka { "chr", RUMP_ETFS_CHR },
123 1.17 pooka { "reg", RUMP_ETFS_REG },
124 1.17 pooka };
125 1.17 pooka
126 1.1 pooka int
127 1.1 pooka main(int argc, char *argv[])
128 1.1 pooka {
129 1.2 pooka const char *serverurl;
130 1.9 pooka struct etfsreg *etfs = NULL;
131 1.9 pooka unsigned netfs = 0, curetfs = 0;
132 1.1 pooka int error;
133 1.2 pooka int ch, sflag;
134 1.24 pooka unsigned i;
135 1.1 pooka
136 1.24 pooka #ifdef PLATFORM_HAS_NBMODULES
137 1.24 pooka char **modarray = NULL;
138 1.24 pooka unsigned nmods = 0, curmod = 0;
139 1.24 pooka #endif
140 1.24 pooka
141 1.24 pooka #ifdef PLATFORM_HAS_SETGETPROGNAME
142 1.1 pooka setprogname(argv[0]);
143 1.24 pooka #endif
144 1.1 pooka
145 1.2 pooka sflag = 0;
146 1.20 pooka while ((ch = getopt(argc, argv, "c:d:l:m:r:sv")) != -1) {
147 1.2 pooka switch (ch) {
148 1.12 pooka case 'c':
149 1.12 pooka setenv("RUMP_NCPU", optarg, 1);
150 1.12 pooka break;
151 1.9 pooka case 'd': {
152 1.9 pooka char *options, *value;
153 1.9 pooka char *key, *hostpath;
154 1.14 pooka long long flen, foffset;
155 1.16 pooka char partition;
156 1.17 pooka int ftype;
157 1.9 pooka
158 1.14 pooka flen = foffset = 0;
159 1.16 pooka partition = 0;
160 1.9 pooka key = hostpath = NULL;
161 1.17 pooka ftype = -1;
162 1.9 pooka options = optarg;
163 1.9 pooka while (*options) {
164 1.9 pooka switch (getsubopt(&options,
165 1.9 pooka __UNCONST(disktokens), &value)) {
166 1.9 pooka case DKEY:
167 1.11 pooka if (key != NULL) {
168 1.11 pooka fprintf(stderr,
169 1.11 pooka "key already given\n");
170 1.11 pooka usage();
171 1.11 pooka }
172 1.9 pooka key = value;
173 1.9 pooka break;
174 1.16 pooka
175 1.9 pooka case DFILE:
176 1.11 pooka if (hostpath != NULL) {
177 1.11 pooka fprintf(stderr,
178 1.11 pooka "hostpath already given\n");
179 1.11 pooka usage();
180 1.11 pooka }
181 1.9 pooka hostpath = value;
182 1.9 pooka break;
183 1.16 pooka
184 1.9 pooka case DSIZE:
185 1.11 pooka if (flen != 0) {
186 1.11 pooka fprintf(stderr,
187 1.11 pooka "size already given\n");
188 1.11 pooka usage();
189 1.11 pooka }
190 1.19 pooka if (strcmp(value, "host") == 0) {
191 1.18 pooka if (foffset != 0) {
192 1.18 pooka fprintf(stderr,
193 1.18 pooka "cannot specify "
194 1.18 pooka "offset with "
195 1.21 pooka "size=host\n");
196 1.18 pooka usage();
197 1.18 pooka }
198 1.18 pooka flen = DSIZE_E;
199 1.18 pooka } else {
200 1.24 pooka #ifdef PLATFORM_HAS_STRSUFTOLL
201 1.18 pooka /* XXX: off_t max? */
202 1.18 pooka flen = strsuftoll("-d size",
203 1.18 pooka value, 0, LLONG_MAX);
204 1.24 pooka #else
205 1.24 pooka flen = strtoull(value,
206 1.24 pooka NULL, 10);
207 1.24 pooka #endif
208 1.18 pooka }
209 1.9 pooka break;
210 1.14 pooka case DOFFSET:
211 1.14 pooka if (foffset != 0) {
212 1.14 pooka fprintf(stderr,
213 1.14 pooka "offset already given\n");
214 1.14 pooka usage();
215 1.14 pooka }
216 1.18 pooka if (flen == DSIZE_E) {
217 1.18 pooka fprintf(stderr, "cannot "
218 1.18 pooka "specify offset with "
219 1.21 pooka "size=host\n");
220 1.18 pooka usage();
221 1.18 pooka }
222 1.24 pooka #ifdef PLATFORM_HAS_STRSUFTOLL
223 1.14 pooka /* XXX: off_t max? */
224 1.14 pooka foffset = strsuftoll("-d offset", value,
225 1.14 pooka 0, LLONG_MAX);
226 1.24 pooka #else
227 1.24 pooka foffset = strtoull(value, NULL, 10);
228 1.24 pooka #endif
229 1.14 pooka break;
230 1.16 pooka
231 1.16 pooka case DLABEL:
232 1.16 pooka if (foffset != 0 || flen != 0) {
233 1.16 pooka fprintf(stderr,
234 1.16 pooka "disklabel needs to be "
235 1.16 pooka "used alone\n");
236 1.16 pooka usage();
237 1.16 pooka }
238 1.16 pooka if (strlen(value) != 1 ||
239 1.16 pooka *value < 'a' || *value > 'z') {
240 1.16 pooka fprintf(stderr,
241 1.16 pooka "invalid label part\n");
242 1.16 pooka usage();
243 1.16 pooka }
244 1.16 pooka partition = *value;
245 1.16 pooka break;
246 1.16 pooka
247 1.17 pooka case DTYPE:
248 1.17 pooka if (ftype != -1) {
249 1.17 pooka fprintf(stderr,
250 1.17 pooka "type already specified\n");
251 1.17 pooka usage();
252 1.17 pooka }
253 1.17 pooka
254 1.17 pooka for (i = 0;
255 1.17 pooka i < __arraycount(etfstypes);
256 1.17 pooka i++) {
257 1.17 pooka if (strcmp(etfstypes[i].name,
258 1.17 pooka value) == 0)
259 1.17 pooka break;
260 1.17 pooka }
261 1.17 pooka if (i == __arraycount(etfstypes)) {
262 1.17 pooka fprintf(stderr,
263 1.17 pooka "invalid type %s\n", value);
264 1.17 pooka usage();
265 1.17 pooka }
266 1.17 pooka ftype = etfstypes[i].type;
267 1.17 pooka break;
268 1.17 pooka
269 1.9 pooka default:
270 1.9 pooka fprintf(stderr, "invalid dtoken\n");
271 1.9 pooka usage();
272 1.9 pooka break;
273 1.9 pooka }
274 1.9 pooka }
275 1.9 pooka
276 1.16 pooka if (key == NULL || hostpath == NULL ||
277 1.16 pooka (flen == 0 && partition == 0)) {
278 1.9 pooka fprintf(stderr, "incomplete drivespec\n");
279 1.9 pooka usage();
280 1.9 pooka }
281 1.17 pooka if (ftype == -1)
282 1.17 pooka ftype = RUMP_ETFS_BLK;
283 1.9 pooka
284 1.9 pooka if (netfs - curetfs == 0) {
285 1.9 pooka etfs = realloc(etfs, (netfs+16)*sizeof(*etfs));
286 1.9 pooka if (etfs == NULL)
287 1.9 pooka err(1, "realloc etfs");
288 1.9 pooka netfs += 16;
289 1.9 pooka }
290 1.9 pooka
291 1.9 pooka etfs[curetfs].key = key;
292 1.9 pooka etfs[curetfs].hostpath = hostpath;
293 1.9 pooka etfs[curetfs].flen = flen;
294 1.14 pooka etfs[curetfs].foffset = foffset;
295 1.16 pooka etfs[curetfs].partition = partition;
296 1.17 pooka etfs[curetfs].type = ftype;
297 1.9 pooka curetfs++;
298 1.9 pooka
299 1.9 pooka break;
300 1.9 pooka }
301 1.6 pooka case 'l':
302 1.8 pooka if (dlopen(optarg, RTLD_LAZY|RTLD_GLOBAL) == NULL) {
303 1.8 pooka char pb[MAXPATHLEN];
304 1.8 pooka /* try to mimic linker -l syntax */
305 1.8 pooka
306 1.8 pooka snprintf(pb, sizeof(pb), "lib%s.so", optarg);
307 1.8 pooka if (dlopen(pb, RTLD_LAZY|RTLD_GLOBAL) == NULL) {
308 1.8 pooka errx(1, "dlopen %s failed: %s",
309 1.8 pooka pb, dlerror());
310 1.8 pooka }
311 1.8 pooka }
312 1.6 pooka break;
313 1.24 pooka #ifdef PLATFORM_HAS_NBMODULES
314 1.24 pooka case 'm': {
315 1.24 pooka
316 1.7 pooka if (nmods - curmod == 0) {
317 1.7 pooka modarray = realloc(modarray,
318 1.7 pooka (nmods+16) * sizeof(char *));
319 1.7 pooka if (modarray == NULL)
320 1.7 pooka err(1, "realloc");
321 1.7 pooka nmods += 16;
322 1.7 pooka }
323 1.7 pooka modarray[curmod++] = optarg;
324 1.24 pooka break; }
325 1.24 pooka #endif
326 1.20 pooka case 'r':
327 1.20 pooka setenv("RUMP_MEMLIMIT", optarg, 1);
328 1.20 pooka break;
329 1.2 pooka case 's':
330 1.2 pooka sflag = 1;
331 1.2 pooka break;
332 1.20 pooka case 'v':
333 1.20 pooka setenv("RUMP_VERBOSE", "1", 1);
334 1.20 pooka break;
335 1.2 pooka default:
336 1.2 pooka usage();
337 1.2 pooka /*NOTREACHED*/
338 1.2 pooka }
339 1.2 pooka }
340 1.2 pooka
341 1.2 pooka argc -= optind;
342 1.2 pooka argv += optind;
343 1.2 pooka
344 1.2 pooka if (argc != 1)
345 1.1 pooka usage();
346 1.1 pooka
347 1.2 pooka serverurl = argv[0];
348 1.2 pooka
349 1.2 pooka if (!sflag) {
350 1.2 pooka error = rump_daemonize_begin();
351 1.2 pooka if (error)
352 1.2 pooka errx(1, "rump daemonize: %s", strerror(error));
353 1.2 pooka }
354 1.2 pooka
355 1.1 pooka error = rump_init();
356 1.1 pooka if (error)
357 1.2 pooka die(sflag, error, "rump init failed");
358 1.7 pooka
359 1.24 pooka #ifdef PLATFORM_HAS_NBMODULES
360 1.9 pooka /* load modules */
361 1.7 pooka for (i = 0; i < curmod; i++) {
362 1.7 pooka struct modctl_load ml;
363 1.7 pooka
364 1.7 pooka #define ETFSKEY "/module.mod"
365 1.7 pooka if ((error = rump_pub_etfs_register(ETFSKEY,
366 1.7 pooka modarray[0], RUMP_ETFS_REG)) != 0)
367 1.7 pooka die(sflag, error, "module etfs register failed");
368 1.7 pooka memset(&ml, 0, sizeof(ml));
369 1.7 pooka ml.ml_filename = ETFSKEY;
370 1.7 pooka if (rump_sys_modctl(MODCTL_LOAD, &ml) == -1)
371 1.7 pooka die(sflag, errno, "module load failed");
372 1.7 pooka rump_pub_etfs_remove(ETFSKEY);
373 1.9 pooka #undef ETFSKEY
374 1.9 pooka }
375 1.24 pooka #endif /* PLATFORM_HAS_NBMODULES */
376 1.9 pooka
377 1.9 pooka /* register host drives */
378 1.9 pooka for (i = 0; i < curetfs; i++) {
379 1.14 pooka struct stat sb;
380 1.16 pooka off_t foffset, flen, fendoff;
381 1.18 pooka int fd, oflags;
382 1.9 pooka
383 1.18 pooka oflags = etfs[i].flen == DSIZE_E ? 0 : O_CREAT;
384 1.18 pooka fd = open(etfs[i].hostpath, O_RDWR | oflags, 0644);
385 1.9 pooka if (fd == -1)
386 1.18 pooka die(sflag, errno, "etfs hostpath open");
387 1.16 pooka
388 1.24 pooka #ifdef PLATFORM_HAS_DISKLABEL
389 1.16 pooka if (etfs[i].partition) {
390 1.24 pooka struct disklabel dl;
391 1.24 pooka char buf[1<<16];
392 1.16 pooka int partition = etfs[i].partition - 'a';
393 1.16 pooka
394 1.16 pooka pread(fd, buf, sizeof(buf), 0);
395 1.16 pooka if (disklabel_scan(&dl, buf, sizeof(buf)))
396 1.16 pooka die(sflag, ENOENT, "disklabel not found");
397 1.16 pooka
398 1.16 pooka if (partition >= dl.d_npartitions)
399 1.16 pooka die(sflag, ENOENT, "partition not available");
400 1.16 pooka
401 1.16 pooka foffset = dl.d_partitions[partition].p_offset
402 1.16 pooka << DEV_BSHIFT;
403 1.16 pooka flen = dl.d_partitions[partition].p_size
404 1.16 pooka << DEV_BSHIFT;
405 1.16 pooka } else {
406 1.24 pooka #else
407 1.24 pooka {
408 1.24 pooka #endif
409 1.16 pooka foffset = etfs[i].foffset;
410 1.16 pooka flen = etfs[i].flen;
411 1.16 pooka }
412 1.16 pooka
413 1.14 pooka if (fstat(fd, &sb) == -1)
414 1.14 pooka die(sflag, errno, "fstat etfs hostpath");
415 1.18 pooka if (flen == DSIZE_E) {
416 1.21 pooka if (sb.st_size == 0)
417 1.21 pooka die(sflag, EINVAL, "size=host, but cannot "
418 1.21 pooka "query non-zero size");
419 1.18 pooka flen = sb.st_size;
420 1.18 pooka }
421 1.18 pooka fendoff = foffset + flen;
422 1.16 pooka if (S_ISREG(sb.st_mode) && sb.st_size < fendoff) {
423 1.16 pooka if (ftruncate(fd, fendoff) == -1)
424 1.14 pooka die(sflag, errno, "truncate");
425 1.14 pooka }
426 1.9 pooka close(fd);
427 1.9 pooka
428 1.14 pooka if ((error = rump_pub_etfs_register_withsize(etfs[i].key,
429 1.16 pooka etfs[i].hostpath, etfs[i].type, foffset, flen)) != 0)
430 1.9 pooka die(sflag, error, "etfs register");
431 1.7 pooka }
432 1.7 pooka
433 1.2 pooka error = rump_init_server(serverurl);
434 1.1 pooka if (error)
435 1.2 pooka die(sflag, error, "rump server init failed");
436 1.2 pooka
437 1.2 pooka if (!sflag)
438 1.3 pooka rump_daemonize_done(RUMP_DAEMONIZE_SUCCESS);
439 1.2 pooka
440 1.4 pooka sem_init(&sigsem, 0, 0);
441 1.4 pooka signal(SIGTERM, sigreboot);
442 1.4 pooka signal(SIGINT, sigreboot);
443 1.4 pooka sem_wait(&sigsem);
444 1.4 pooka
445 1.4 pooka rump_sys_reboot(0, NULL);
446 1.5 pooka /*NOTREACHED*/
447 1.5 pooka
448 1.4 pooka return 0;
449 1.1 pooka }
450