1 /*
2 * Copyright (C) 2012 Citrix Ltd.
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU Lesser General Public License as published
6 * by the Free Software Foundation; version 2.1 only. with the special
7 * exception on linking described in file LICENSE.
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU Lesser General Public License for more details.
13 */
14
15 #include "libxl_osdeps.h"
16
17 #include "libxl_internal.h"
18
19 /* stream_fd is as from the caller (eventually, the application).
20 * It may be 0, 1 or 2, in which case we need to dup it elsewhere.
21 * The actual fd value is not included in the supplied argnums; rather
22 * it will be automatically supplied by run_helper as the 2nd argument.
23 *
24 * preserve_fds are fds that the caller is intending to pass to the
25 * helper so which need cloexec clearing. They may not be 0, 1 or 2.
26 * An entry may be -1 in which case it will be ignored.
27 */
28 static void run_helper(libxl__egc *egc, libxl__save_helper_state *shs,
29 const char *mode_arg,
30 int stream_fd, int back_channel_fd,
31 const int *preserve_fds, int num_preserve_fds,
32 const unsigned long *argnums, int num_argnums);
33
34 static void helper_failed(libxl__egc*, libxl__save_helper_state *shs, int rc);
35 static void helper_stop(libxl__egc *egc, libxl__ao_abortable*, int rc);
36 static void helper_stdout_readable(libxl__egc *egc, libxl__ev_fd *ev,
37 int fd, short events, short revents);
38 static void helper_exited(libxl__egc *egc, libxl__ev_child *ch,
39 pid_t pid, int status);
40 static void helper_done(libxl__egc *egc, libxl__save_helper_state *shs);
41
42 /*----- entrypoints -----*/
43
libxl__xc_domain_restore(libxl__egc * egc,libxl__domain_create_state * dcs,libxl__save_helper_state * shs,int hvm,int pae)44 void libxl__xc_domain_restore(libxl__egc *egc, libxl__domain_create_state *dcs,
45 libxl__save_helper_state *shs,
46 int hvm, int pae)
47 {
48 STATE_AO_GC(dcs->ao);
49
50 /* Convenience aliases */
51 const uint32_t domid = dcs->guest_domid;
52 const int restore_fd = dcs->libxc_fd;
53 const int send_back_fd = dcs->send_back_fd;
54 libxl__domain_build_state *const state = &dcs->build_state;
55
56 unsigned cbflags =
57 libxl__srm_callout_enumcallbacks_restore(&shs->callbacks.restore.a);
58
59 const unsigned long argnums[] = {
60 domid,
61 state->store_port,
62 state->store_domid, state->console_port,
63 state->console_domid,
64 hvm, pae,
65 cbflags, dcs->restore_params.checkpointed_stream,
66 };
67
68 shs->ao = ao;
69 shs->domid = domid;
70 shs->recv_callback = libxl__srm_callout_received_restore;
71 if (dcs->restore_params.checkpointed_stream ==
72 LIBXL_CHECKPOINTED_STREAM_COLO)
73 shs->completion_callback = libxl__colo_restore_teardown;
74 else
75 shs->completion_callback = libxl__xc_domain_restore_done;
76 shs->caller_state = dcs;
77 shs->need_results = 1;
78
79 run_helper(egc, shs, "--restore-domain", restore_fd, send_back_fd, 0, 0,
80 argnums, ARRAY_SIZE(argnums));
81 }
82
libxl__xc_domain_save(libxl__egc * egc,libxl__domain_save_state * dss,libxl__save_helper_state * shs)83 void libxl__xc_domain_save(libxl__egc *egc, libxl__domain_save_state *dss,
84 libxl__save_helper_state *shs)
85 {
86 STATE_AO_GC(dss->ao);
87
88 unsigned cbflags =
89 libxl__srm_callout_enumcallbacks_save(&shs->callbacks.save.a);
90
91 const unsigned long argnums[] = {
92 dss->domid, dss->xcflags, dss->hvm, cbflags,
93 dss->checkpointed_stream,
94 };
95
96 shs->ao = ao;
97 shs->domid = dss->domid;
98 shs->recv_callback = libxl__srm_callout_received_save;
99 shs->completion_callback = libxl__xc_domain_save_done;
100 shs->caller_state = dss;
101 shs->need_results = 0;
102
103 run_helper(egc, shs, "--save-domain", dss->fd, dss->recv_fd,
104 NULL, 0,
105 argnums, ARRAY_SIZE(argnums));
106 return;
107 }
108
109
libxl__xc_domain_saverestore_async_callback_done(libxl__egc * egc,libxl__save_helper_state * shs,int return_value)110 void libxl__xc_domain_saverestore_async_callback_done(libxl__egc *egc,
111 libxl__save_helper_state *shs, int return_value)
112 {
113 shs->egc = egc;
114 libxl__srm_callout_sendreply(return_value, shs);
115 shs->egc = 0;
116 }
117
libxl__save_helper_init(libxl__save_helper_state * shs)118 void libxl__save_helper_init(libxl__save_helper_state *shs)
119 {
120 libxl__ao_abortable_init(&shs->abrt);
121 libxl__ev_fd_init(&shs->readable);
122 libxl__ev_child_init(&shs->child);
123 }
124
125 /*----- helper execution -----*/
126
127 /* This function can not fail. */
dup_cloexec(libxl__gc * gc,int fd,const char * what)128 static int dup_cloexec(libxl__gc *gc, int fd, const char *what)
129 {
130 int dup_fd = fd;
131
132 if (fd <= 2) {
133 dup_fd = dup(fd);
134 if (dup_fd < 0) {
135 LOGE(ERROR,"dup %s", what);
136 exit(-1);
137 }
138 }
139 libxl_fd_set_cloexec(CTX, dup_fd, 0);
140
141 return dup_fd;
142 }
143
144 /*
145 * Both save and restore share four parameters:
146 * 1) Path to libxl-save-helper.
147 * 2) --[restore|save]-domain.
148 * 3) stream file descriptor.
149 * 4) back channel file descriptor.
150 * n) save/restore specific parameters.
151 * 5) A \0 at the end.
152 */
153 #define HELPER_NR_ARGS 5
run_helper(libxl__egc * egc,libxl__save_helper_state * shs,const char * mode_arg,int stream_fd,int back_channel_fd,const int * preserve_fds,int num_preserve_fds,const unsigned long * argnums,int num_argnums)154 static void run_helper(libxl__egc *egc, libxl__save_helper_state *shs,
155 const char *mode_arg,
156 int stream_fd, int back_channel_fd,
157 const int *preserve_fds, int num_preserve_fds,
158 const unsigned long *argnums, int num_argnums)
159 {
160 STATE_AO_GC(shs->ao);
161 const char *args[HELPER_NR_ARGS + num_argnums];
162 const char **arg = args;
163 int i, rc;
164
165 /* Resources we must free */
166 libxl__carefd *childs_pipes[2] = { 0,0 };
167
168 /* Convenience aliases */
169 const uint32_t domid = shs->domid;
170
171 shs->rc = 0;
172 shs->completed = 0;
173 shs->pipes[0] = shs->pipes[1] = 0;
174 libxl__save_helper_init(shs);
175
176 shs->abrt.ao = shs->ao;
177 shs->abrt.callback = helper_stop;
178 rc = libxl__ao_abortable_register(&shs->abrt);
179 if (rc) goto out;
180
181 shs->stdin_what = GCSPRINTF("domain %"PRIu32" save/restore helper"
182 " stdin pipe", domid);
183 shs->stdout_what = GCSPRINTF("domain %"PRIu32" save/restore helper"
184 " stdout pipe", domid);
185
186 *arg++ = getenv("LIBXL_SAVE_HELPER") ?: LIBEXEC_BIN "/" "libxl-save-helper";
187 *arg++ = mode_arg;
188 const char **stream_fd_arg = arg++;
189 const char **back_channel_fd_arg = arg++;
190 for (i=0; i<num_argnums; i++)
191 *arg++ = GCSPRINTF("%lu", argnums[i]);
192 *arg++ = 0;
193 assert(arg == args + ARRAY_SIZE(args));
194
195 libxl__carefd_begin();
196 int childfd;
197 for (childfd=0; childfd<2; childfd++) {
198 /* Setting up the pipe for the child's fd childfd */
199 int fds[2];
200 if (libxl_pipe(CTX,fds)) {
201 rc = ERROR_FAIL;
202 libxl__carefd_unlock();
203 goto out;
204 }
205 int childs_end = childfd==0 ? 0 /*read*/ : 1 /*write*/;
206 int our_end = childfd==0 ? 1 /*write*/ : 0 /*read*/;
207 childs_pipes[childfd] = libxl__carefd_record(CTX, fds[childs_end]);
208 shs->pipes[childfd] = libxl__carefd_record(CTX, fds[our_end]);
209 }
210 libxl__carefd_unlock();
211
212 pid_t pid = libxl__ev_child_fork(gc, &shs->child, helper_exited);
213 if (!pid) {
214 stream_fd = dup_cloexec(gc, stream_fd, "migration stream fd");
215 *stream_fd_arg = GCSPRINTF("%d", stream_fd);
216
217 if (back_channel_fd >= 0)
218 back_channel_fd = dup_cloexec(gc, back_channel_fd,
219 "migration back channel fd");
220 *back_channel_fd_arg = GCSPRINTF("%d", back_channel_fd);
221
222 for (i=0; i<num_preserve_fds; i++)
223 if (preserve_fds[i] >= 0) {
224 assert(preserve_fds[i] > 2);
225 libxl_fd_set_cloexec(CTX, preserve_fds[i], 0);
226 }
227
228 libxl__exec(gc,
229 libxl__carefd_fd(childs_pipes[0]),
230 libxl__carefd_fd(childs_pipes[1]),
231 -1,
232 args[0], (char**)args, 0);
233 }
234
235 libxl__carefd_close(childs_pipes[0]);
236 libxl__carefd_close(childs_pipes[1]);
237
238 rc = libxl__ev_fd_register(gc, &shs->readable, helper_stdout_readable,
239 libxl__carefd_fd(shs->pipes[1]), POLLIN|POLLPRI);
240 if (rc) goto out;
241 return;
242
243 out:
244 libxl__carefd_close(childs_pipes[0]);
245 libxl__carefd_close(childs_pipes[1]);
246 helper_failed(egc, shs, rc);;
247 }
248
helper_failed(libxl__egc * egc,libxl__save_helper_state * shs,int rc)249 static void helper_failed(libxl__egc *egc, libxl__save_helper_state *shs,
250 int rc)
251 {
252 STATE_AO_GC(shs->ao);
253
254 if (!shs->rc)
255 shs->rc = rc;
256
257 libxl__ev_fd_deregister(gc, &shs->readable);
258
259 if (!libxl__save_helper_inuse(shs)) {
260 helper_done(egc, shs);
261 return;
262 }
263
264 libxl__kill(gc, shs->child.pid, SIGKILL, "save/restore helper");
265 }
266
helper_stop(libxl__egc * egc,libxl__ao_abortable * abrt,int rc)267 static void helper_stop(libxl__egc *egc, libxl__ao_abortable *abrt, int rc)
268 {
269 libxl__save_helper_state *shs = CONTAINER_OF(abrt, *shs, abrt);
270 STATE_AO_GC(shs->ao);
271
272 if (!libxl__save_helper_inuse(shs)) {
273 helper_failed(egc, shs, rc);
274 return;
275 }
276
277 if (!shs->rc)
278 shs->rc = rc;
279
280 libxl__kill(gc, shs->child.pid, SIGTERM, "save/restore helper");
281 }
282
libxl__save_helper_abort(libxl__egc * egc,libxl__save_helper_state * shs)283 void libxl__save_helper_abort(libxl__egc *egc,
284 libxl__save_helper_state *shs)
285 {
286 helper_stop(egc, &shs->abrt, ERROR_FAIL);
287 }
288
helper_stdout_readable(libxl__egc * egc,libxl__ev_fd * ev,int fd,short events,short revents)289 static void helper_stdout_readable(libxl__egc *egc, libxl__ev_fd *ev,
290 int fd, short events, short revents)
291 {
292 libxl__save_helper_state *shs = CONTAINER_OF(ev, *shs, readable);
293 STATE_AO_GC(shs->ao);
294 int rc, errnoval;
295
296 if (revents & (POLLERR|POLLPRI)) {
297 LOGD(ERROR, shs->domid, "%s signaled POLLERR|POLLPRI (%#x)",
298 shs->stdout_what, revents);
299 rc = ERROR_FAIL;
300 out:
301 /* this is here because otherwise we bypass the decl of msg[] */
302 helper_failed(egc, shs, rc);
303 return;
304 }
305
306 uint16_t msglen;
307 errnoval = libxl_read_exactly(CTX, fd, &msglen, sizeof(msglen),
308 shs->stdout_what, "ipc msg header");
309 if (errnoval) { rc = ERROR_FAIL; goto out; }
310
311 unsigned char msg[msglen];
312 errnoval = libxl_read_exactly(CTX, fd, msg, msglen,
313 shs->stdout_what, "ipc msg body");
314 if (errnoval) { rc = ERROR_FAIL; goto out; }
315
316 shs->egc = egc;
317 shs->recv_callback(msg, msglen, shs);
318 shs->egc = 0;
319 return;
320 }
321
helper_exited(libxl__egc * egc,libxl__ev_child * ch,pid_t pid,int status)322 static void helper_exited(libxl__egc *egc, libxl__ev_child *ch,
323 pid_t pid, int status)
324 {
325 libxl__save_helper_state *shs = CONTAINER_OF(ch, *shs, child);
326 STATE_AO_GC(shs->ao);
327
328 /* Convenience aliases */
329 const uint32_t domid = shs->domid;
330
331 const char *what =
332 GCSPRINTF("domain %"PRIu32" save/restore helper", domid);
333
334 if (status) {
335 libxl_report_child_exitstatus(CTX, XTL_ERROR, what, pid, status);
336 if (!shs->rc)
337 shs->rc = ERROR_FAIL;
338 }
339
340 if (shs->need_results) {
341 if (!shs->rc) {
342 LOGD(ERROR,shs->domid,"%s exited without providing results",what);
343 shs->rc = ERROR_FAIL;
344 }
345 }
346
347 if (!shs->completed) {
348 if (!shs->rc) {
349 LOGD(ERROR,shs->domid,"%s exited without signaling completion",what);
350 shs->rc = ERROR_FAIL;
351 }
352 }
353
354 helper_done(egc, shs);
355 return;
356 }
357
helper_done(libxl__egc * egc,libxl__save_helper_state * shs)358 static void helper_done(libxl__egc *egc, libxl__save_helper_state *shs)
359 {
360 STATE_AO_GC(shs->ao);
361
362 libxl__ao_abortable_deregister(&shs->abrt);
363 libxl__ev_fd_deregister(gc, &shs->readable);
364 libxl__carefd_close(shs->pipes[0]); shs->pipes[0] = 0;
365 libxl__carefd_close(shs->pipes[1]); shs->pipes[1] = 0;
366 assert(!libxl__save_helper_inuse(shs));
367
368 shs->egc = egc;
369 shs->completion_callback(egc, shs->caller_state,
370 shs->rc, shs->retval, shs->errnoval);
371 shs->egc = 0;
372 }
373
374 /*----- generic helpers for the autogenerated code -----*/
375
376 const libxl__srm_save_autogen_callbacks*
libxl__srm_callout_get_callbacks_save(void * user)377 libxl__srm_callout_get_callbacks_save(void *user)
378 {
379 libxl__save_helper_state *shs = user;
380 return &shs->callbacks.save.a;
381 }
382
383 const libxl__srm_restore_autogen_callbacks*
libxl__srm_callout_get_callbacks_restore(void * user)384 libxl__srm_callout_get_callbacks_restore(void *user)
385 {
386 libxl__save_helper_state *shs = user;
387 return &shs->callbacks.restore.a;
388 }
389
libxl__srm_callout_sendreply(int r,void * user)390 void libxl__srm_callout_sendreply(int r, void *user)
391 {
392 libxl__save_helper_state *shs = user;
393 libxl__egc *egc = shs->egc;
394 STATE_AO_GC(shs->ao);
395 int errnoval;
396
397 errnoval = libxl_write_exactly(CTX, libxl__carefd_fd(shs->pipes[0]),
398 &r, sizeof(r), shs->stdin_what,
399 "callback return value");
400 if (errnoval)
401 helper_failed(egc, shs, ERROR_FAIL);
402 }
403
libxl__srm_callout_callback_log(uint32_t level,uint32_t errnoval,const char * context,const char * formatted,void * user)404 void libxl__srm_callout_callback_log(uint32_t level, uint32_t errnoval,
405 const char *context, const char *formatted, void *user)
406 {
407 libxl__save_helper_state *shs = user;
408 STATE_AO_GC(shs->ao);
409 xtl_log(CTX->lg, level, errnoval, context, "%s", formatted);
410 }
411
libxl__srm_callout_callback_progress(const char * context,const char * doing_what,unsigned long done,unsigned long total,void * user)412 void libxl__srm_callout_callback_progress(const char *context,
413 const char *doing_what, unsigned long done,
414 unsigned long total, void *user)
415 {
416 libxl__save_helper_state *shs = user;
417 STATE_AO_GC(shs->ao);
418 xtl_progress(CTX->lg, context, doing_what, done, total);
419 }
420
libxl__srm_callout_callback_complete(int retval,int errnoval,void * user)421 int libxl__srm_callout_callback_complete(int retval, int errnoval,
422 void *user)
423 {
424 libxl__save_helper_state *shs = user;
425 STATE_AO_GC(shs->ao);
426
427 shs->completed = 1;
428 shs->retval = retval;
429 shs->errnoval = errnoval;
430 libxl__ev_fd_deregister(gc, &shs->readable);
431 return 0;
432 }
433