rump_allserver.c revision 1.13 1 1.13 wiz /* $NetBSD: rump_allserver.c,v 1.13 2011/01/03 12:18:25 wiz Exp $ */
2 1.1 pooka
3 1.1 pooka /*-
4 1.1 pooka * Copyright (c) 2010 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.1 pooka #include <sys/cdefs.h>
29 1.1 pooka #ifndef lint
30 1.13 wiz __RCSID("$NetBSD: rump_allserver.c,v 1.13 2011/01/03 12:18:25 wiz Exp $");
31 1.1 pooka #endif /* !lint */
32 1.1 pooka
33 1.1 pooka #include <sys/types.h>
34 1.4 pooka #include <sys/signal.h>
35 1.7 pooka #include <sys/module.h>
36 1.1 pooka
37 1.1 pooka #include <rump/rump.h>
38 1.4 pooka #include <rump/rump_syscalls.h>
39 1.1 pooka
40 1.6 pooka #include <dlfcn.h>
41 1.1 pooka #include <err.h>
42 1.1 pooka #include <errno.h>
43 1.9 pooka #include <fcntl.h>
44 1.4 pooka #include <semaphore.h>
45 1.1 pooka #include <stdio.h>
46 1.1 pooka #include <stdlib.h>
47 1.1 pooka #include <string.h>
48 1.1 pooka #include <unistd.h>
49 1.1 pooka
50 1.1 pooka static void
51 1.1 pooka usage(void)
52 1.1 pooka {
53 1.1 pooka
54 1.13 wiz fprintf(stderr, "usage: %s [-s] [-c ncpu] [-d drivespec] [-l libs] "
55 1.12 pooka "[-m modules] bindurl\n", getprogname());
56 1.1 pooka exit(1);
57 1.1 pooka }
58 1.1 pooka
59 1.2 pooka static void
60 1.2 pooka die(int sflag, int error, const char *reason)
61 1.2 pooka {
62 1.2 pooka
63 1.2 pooka warnx("%s: %s", reason, strerror(error));
64 1.2 pooka if (!sflag)
65 1.2 pooka rump_daemonize_done(error);
66 1.2 pooka exit(1);
67 1.2 pooka }
68 1.2 pooka
69 1.4 pooka static sem_t sigsem;
70 1.4 pooka static void
71 1.4 pooka sigreboot(int sig)
72 1.4 pooka {
73 1.4 pooka
74 1.4 pooka sem_post(&sigsem);
75 1.4 pooka }
76 1.4 pooka
77 1.9 pooka static const char *const disktokens[] = {
78 1.9 pooka #define DKEY 0
79 1.9 pooka "key",
80 1.9 pooka #define DFILE 1
81 1.9 pooka "hostpath",
82 1.9 pooka #define DSIZE 2
83 1.9 pooka "size",
84 1.9 pooka NULL
85 1.9 pooka };
86 1.9 pooka
87 1.9 pooka struct etfsreg {
88 1.9 pooka const char *key;
89 1.9 pooka const char *hostpath;
90 1.9 pooka off_t flen;
91 1.9 pooka enum rump_etfs_type type;
92 1.9 pooka };
93 1.9 pooka
94 1.1 pooka int
95 1.1 pooka main(int argc, char *argv[])
96 1.1 pooka {
97 1.2 pooka const char *serverurl;
98 1.7 pooka char **modarray = NULL;
99 1.7 pooka unsigned nmods = 0, curmod = 0, i;
100 1.9 pooka struct etfsreg *etfs = NULL;
101 1.9 pooka unsigned netfs = 0, curetfs = 0;
102 1.1 pooka int error;
103 1.2 pooka int ch, sflag;
104 1.12 pooka int ncpu;
105 1.1 pooka
106 1.1 pooka setprogname(argv[0]);
107 1.1 pooka
108 1.2 pooka sflag = 0;
109 1.12 pooka while ((ch = getopt(argc, argv, "c:d:l:m:s")) != -1) {
110 1.2 pooka switch (ch) {
111 1.12 pooka case 'c':
112 1.12 pooka ncpu = atoi(optarg);
113 1.12 pooka /* XXX: MAXCPUS is from host, not from kernel */
114 1.12 pooka if (ncpu < 1 || ncpu > MAXCPUS)
115 1.12 pooka err(1, "CPU count needs to be between "
116 1.12 pooka "1 and %d\n", MAXCPUS);
117 1.12 pooka setenv("RUMP_NCPU", optarg, 1);
118 1.12 pooka break;
119 1.9 pooka case 'd': {
120 1.9 pooka char *options, *value;
121 1.9 pooka char *key, *hostpath;
122 1.9 pooka long long flen;
123 1.9 pooka
124 1.9 pooka flen = 0;
125 1.9 pooka key = hostpath = NULL;
126 1.9 pooka options = optarg;
127 1.9 pooka while (*options) {
128 1.9 pooka switch (getsubopt(&options,
129 1.9 pooka __UNCONST(disktokens), &value)) {
130 1.9 pooka case DKEY:
131 1.11 pooka if (key != NULL) {
132 1.11 pooka fprintf(stderr,
133 1.11 pooka "key already given\n");
134 1.11 pooka usage();
135 1.11 pooka }
136 1.9 pooka key = value;
137 1.9 pooka break;
138 1.9 pooka case DFILE:
139 1.11 pooka if (hostpath != NULL) {
140 1.11 pooka fprintf(stderr,
141 1.11 pooka "hostpath already given\n");
142 1.11 pooka usage();
143 1.11 pooka }
144 1.9 pooka hostpath = value;
145 1.9 pooka break;
146 1.9 pooka case DSIZE:
147 1.11 pooka if (flen != 0) {
148 1.11 pooka fprintf(stderr,
149 1.11 pooka "size already given\n");
150 1.11 pooka usage();
151 1.11 pooka }
152 1.9 pooka /* XXX: off_t max? */
153 1.9 pooka flen = strsuftoll("-d size", value,
154 1.9 pooka 0, LLONG_MAX);
155 1.9 pooka break;
156 1.9 pooka default:
157 1.9 pooka fprintf(stderr, "invalid dtoken\n");
158 1.9 pooka usage();
159 1.9 pooka break;
160 1.9 pooka }
161 1.9 pooka }
162 1.9 pooka
163 1.9 pooka if (key == NULL || hostpath == NULL || flen == 0) {
164 1.9 pooka fprintf(stderr, "incomplete drivespec\n");
165 1.9 pooka usage();
166 1.9 pooka }
167 1.9 pooka
168 1.9 pooka if (netfs - curetfs == 0) {
169 1.9 pooka etfs = realloc(etfs, (netfs+16)*sizeof(*etfs));
170 1.9 pooka if (etfs == NULL)
171 1.9 pooka err(1, "realloc etfs");
172 1.9 pooka netfs += 16;
173 1.9 pooka }
174 1.9 pooka
175 1.9 pooka etfs[curetfs].key = key;
176 1.9 pooka etfs[curetfs].hostpath = hostpath;
177 1.9 pooka etfs[curetfs].flen = flen;
178 1.9 pooka etfs[curetfs].type = RUMP_ETFS_BLK;
179 1.9 pooka curetfs++;
180 1.9 pooka
181 1.9 pooka break;
182 1.9 pooka }
183 1.6 pooka case 'l':
184 1.8 pooka if (dlopen(optarg, RTLD_LAZY|RTLD_GLOBAL) == NULL) {
185 1.8 pooka char pb[MAXPATHLEN];
186 1.8 pooka /* try to mimic linker -l syntax */
187 1.8 pooka
188 1.8 pooka snprintf(pb, sizeof(pb), "lib%s.so", optarg);
189 1.8 pooka if (dlopen(pb, RTLD_LAZY|RTLD_GLOBAL) == NULL) {
190 1.8 pooka errx(1, "dlopen %s failed: %s",
191 1.8 pooka pb, dlerror());
192 1.8 pooka }
193 1.8 pooka }
194 1.6 pooka break;
195 1.7 pooka case 'm':
196 1.7 pooka if (nmods - curmod == 0) {
197 1.7 pooka modarray = realloc(modarray,
198 1.7 pooka (nmods+16) * sizeof(char *));
199 1.7 pooka if (modarray == NULL)
200 1.7 pooka err(1, "realloc");
201 1.7 pooka nmods += 16;
202 1.7 pooka }
203 1.7 pooka modarray[curmod++] = optarg;
204 1.7 pooka break;
205 1.2 pooka case 's':
206 1.2 pooka sflag = 1;
207 1.2 pooka break;
208 1.2 pooka default:
209 1.2 pooka usage();
210 1.2 pooka /*NOTREACHED*/
211 1.2 pooka }
212 1.2 pooka }
213 1.2 pooka
214 1.2 pooka argc -= optind;
215 1.2 pooka argv += optind;
216 1.2 pooka
217 1.2 pooka if (argc != 1)
218 1.1 pooka usage();
219 1.1 pooka
220 1.2 pooka serverurl = argv[0];
221 1.2 pooka
222 1.2 pooka if (!sflag) {
223 1.2 pooka error = rump_daemonize_begin();
224 1.2 pooka if (error)
225 1.2 pooka errx(1, "rump daemonize: %s", strerror(error));
226 1.2 pooka }
227 1.2 pooka
228 1.1 pooka error = rump_init();
229 1.1 pooka if (error)
230 1.2 pooka die(sflag, error, "rump init failed");
231 1.7 pooka
232 1.9 pooka /* load modules */
233 1.7 pooka for (i = 0; i < curmod; i++) {
234 1.7 pooka struct modctl_load ml;
235 1.7 pooka
236 1.7 pooka #define ETFSKEY "/module.mod"
237 1.7 pooka if ((error = rump_pub_etfs_register(ETFSKEY,
238 1.7 pooka modarray[0], RUMP_ETFS_REG)) != 0)
239 1.7 pooka die(sflag, error, "module etfs register failed");
240 1.7 pooka memset(&ml, 0, sizeof(ml));
241 1.7 pooka ml.ml_filename = ETFSKEY;
242 1.7 pooka if (rump_sys_modctl(MODCTL_LOAD, &ml) == -1)
243 1.7 pooka die(sflag, errno, "module load failed");
244 1.7 pooka rump_pub_etfs_remove(ETFSKEY);
245 1.9 pooka #undef ETFSKEY
246 1.9 pooka }
247 1.9 pooka
248 1.9 pooka /* register host drives */
249 1.9 pooka for (i = 0; i < curetfs; i++) {
250 1.9 pooka int fd;
251 1.9 pooka
252 1.9 pooka fd = open(etfs[i].hostpath, O_RDWR | O_CREAT, 0755);
253 1.9 pooka if (fd == -1)
254 1.9 pooka die(sflag, error, "etfs hostpath create");
255 1.9 pooka if (ftruncate(fd, etfs[i].flen) == -1)
256 1.9 pooka die(sflag, error, "truncate");
257 1.9 pooka close(fd);
258 1.9 pooka
259 1.9 pooka if ((error = rump_pub_etfs_register(etfs[i].key,
260 1.9 pooka etfs[i].hostpath, etfs[i].type)) != 0)
261 1.9 pooka die(sflag, error, "etfs register");
262 1.7 pooka }
263 1.7 pooka
264 1.2 pooka error = rump_init_server(serverurl);
265 1.1 pooka if (error)
266 1.2 pooka die(sflag, error, "rump server init failed");
267 1.2 pooka
268 1.2 pooka if (!sflag)
269 1.3 pooka rump_daemonize_done(RUMP_DAEMONIZE_SUCCESS);
270 1.2 pooka
271 1.4 pooka sem_init(&sigsem, 0, 0);
272 1.4 pooka signal(SIGTERM, sigreboot);
273 1.4 pooka signal(SIGINT, sigreboot);
274 1.4 pooka sem_wait(&sigsem);
275 1.4 pooka
276 1.4 pooka rump_sys_reboot(0, NULL);
277 1.5 pooka /*NOTREACHED*/
278 1.5 pooka
279 1.4 pooka return 0;
280 1.1 pooka }
281