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