Commit 679aaa56 authored by Ruslan Kuprieiev's avatar Ruslan Kuprieiev Committed by Pavel Emelyanov

libcriu: add ability to use local options structure

Having ability to have your own options structure is quite nice
and allows much more flexible use of libcriu in cases when you
want to have a bunch of instances of options structures.

This patch also allows users to use raw CriuOpts structure
modified in any suitable way, whether by libcriu's criu_local_set
methods or by using protobuf-c directly.

It is also worth noting, that backward-compatibility in API and ABI
is preserved.
Signed-off-by: 's avatarRuslan Kuprieiev <rkuprieiev@cloudlinux.com>
Acked-by: 's avatarTycho Andersen <tycho.andersen@canonical.com>
Signed-off-by: 's avatarPavel Emelyanov <xemul@parallels.com>
parent 337ba4f3
...@@ -19,7 +19,7 @@ ...@@ -19,7 +19,7 @@
const char *criu_lib_version = CRIU_VERSION; const char *criu_lib_version = CRIU_VERSION;
static char *service_address = CR_DEFAULT_SERVICE_ADDRESS; static char *service_address = CR_DEFAULT_SERVICE_ADDRESS;
static CriuOpts *opts; static CriuOpts *global_opts;
static int (*notify)(char *action, criu_notify_arg_t na); static int (*notify)(char *action, criu_notify_arg_t na);
static int saved_errno; static int saved_errno;
...@@ -31,166 +31,381 @@ void criu_set_service_address(char *path) ...@@ -31,166 +31,381 @@ void criu_set_service_address(char *path)
service_address = CR_DEFAULT_SERVICE_ADDRESS; service_address = CR_DEFAULT_SERVICE_ADDRESS;
} }
int criu_init_opts(void) int criu_local_init_opts(void **o)
{ {
if (opts) { if (*o) {
notify = NULL; notify = NULL;
criu_opts__free_unpacked(opts, NULL); criu_opts__free_unpacked((CriuOpts *)*o, NULL);
} }
opts = malloc(sizeof(CriuOpts)); *o = malloc(sizeof(CriuOpts));
if (opts == NULL) { if (*o == NULL) {
perror("Can't allocate memory for criu opts"); perror("Can't allocate memory for criu opts");
return -1; return -1;
} }
criu_opts__init(opts); criu_opts__init((CriuOpts *)*o);
return 0; return 0;
} }
void criu_set_notify_cb(int (*cb)(char *action, criu_notify_arg_t na)) int criu_init_opts(void)
{
return criu_local_init_opts((void **)&global_opts);
}
void criu_local_set_notify_cb(void *o, int (*cb)(char *action, criu_notify_arg_t na))
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
notify = cb; notify = cb;
opts->has_notify_scripts = true; opts->has_notify_scripts = true;
opts->notify_scripts = true; opts->notify_scripts = true;
} }
void criu_set_notify_cb(int (*cb)(char *action, criu_notify_arg_t na))
{
criu_local_set_notify_cb((void *)global_opts, cb);
}
int criu_notify_pid(criu_notify_arg_t na) int criu_notify_pid(criu_notify_arg_t na)
{ {
return na->has_pid ? na->pid : 0; return na->has_pid ? na->pid : 0;
} }
void criu_set_pid(int pid) void criu_local_set_pid(void *o, int pid)
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_pid = true; opts->has_pid = true;
opts->pid = pid; opts->pid = pid;
} }
void criu_set_images_dir_fd(int fd) void criu_set_pid(int pid)
{ {
criu_local_set_pid((void *)global_opts, pid);
}
void criu_local_set_images_dir_fd(void *o, int fd)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->images_dir_fd = fd; opts->images_dir_fd = fd;
} }
void criu_set_parent_images(char *path) void criu_set_images_dir_fd(int fd)
{ {
criu_local_set_images_dir_fd((void *)global_opts, fd);
}
void criu_local_set_parent_images(void *o, char *path)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->parent_img = strdup(path); opts->parent_img = strdup(path);
} }
void criu_set_track_mem(bool track_mem) void criu_set_parent_images(char *path)
{ {
criu_local_set_parent_images((void *)global_opts, path);
}
void criu_local_set_track_mem(void *o, bool track_mem)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_track_mem = true; opts->has_track_mem = true;
opts->track_mem = track_mem; opts->track_mem = track_mem;
} }
void criu_set_auto_dedup(bool auto_dedup) void criu_set_track_mem(bool track_mem)
{ {
criu_local_set_track_mem((void *)global_opts, track_mem);
}
void criu_local_set_auto_dedup(void *o, bool auto_dedup)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_auto_dedup = true; opts->has_auto_dedup = true;
opts->auto_dedup = auto_dedup; opts->auto_dedup = auto_dedup;
} }
void criu_set_force_irmap(bool force_irmap) void criu_set_auto_dedup(bool auto_dedup)
{ {
criu_local_set_auto_dedup((void *)global_opts, auto_dedup);
}
void criu_local_set_force_irmap(void *o, bool force_irmap)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_force_irmap = true; opts->has_force_irmap = true;
opts->force_irmap = force_irmap; opts->force_irmap = force_irmap;
} }
void criu_set_link_remap(bool link_remap) void criu_set_force_irmap(bool force_irmap)
{ {
criu_local_set_force_irmap((void *)global_opts, force_irmap);
}
void criu_local_set_link_remap(void *o, bool link_remap)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_link_remap = true; opts->has_link_remap = true;
opts->link_remap = link_remap; opts->link_remap = link_remap;
} }
void criu_set_work_dir_fd(int fd) void criu_set_link_remap(bool link_remap)
{
criu_local_set_link_remap((void *)global_opts, link_remap);
}
void criu_local_set_work_dir_fd(void *o, int fd)
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_work_dir_fd = true; opts->has_work_dir_fd = true;
opts->work_dir_fd = fd; opts->work_dir_fd = fd;
} }
void criu_set_leave_running(bool leave_running) void criu_set_work_dir_fd(int fd)
{
criu_local_set_work_dir_fd((void *)global_opts, fd);
}
void criu_local_set_leave_running(void *o, bool leave_running)
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_leave_running = true; opts->has_leave_running = true;
opts->leave_running = leave_running; opts->leave_running = leave_running;
} }
void criu_set_ext_unix_sk(bool ext_unix_sk) void criu_set_leave_running(bool leave_running)
{
criu_local_set_leave_running((void *)global_opts, leave_running);
}
void criu_local_set_ext_unix_sk(void *o, bool ext_unix_sk)
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_ext_unix_sk = true; opts->has_ext_unix_sk = true;
opts->ext_unix_sk = ext_unix_sk; opts->ext_unix_sk = ext_unix_sk;
} }
void criu_set_tcp_established(bool tcp_established) void criu_set_ext_unix_sk(bool ext_unix_sk)
{
criu_local_set_ext_unix_sk((void *)global_opts, ext_unix_sk);
}
void criu_local_set_tcp_established(void *o, bool tcp_established)
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_tcp_established = true; opts->has_tcp_established = true;
opts->tcp_established = tcp_established; opts->tcp_established = tcp_established;
} }
void criu_set_evasive_devices(bool evasive_devices) void criu_set_tcp_established(bool tcp_established)
{
criu_local_set_tcp_established((void *)global_opts, tcp_established);
}
void criu_local_set_evasive_devices(void *o, bool evasive_devices)
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_evasive_devices = true; opts->has_evasive_devices = true;
opts->evasive_devices = evasive_devices; opts->evasive_devices = evasive_devices;
} }
void criu_set_shell_job(bool shell_job) void criu_set_evasive_devices(bool evasive_devices)
{
criu_local_set_evasive_devices((void *)global_opts, evasive_devices);
}
void criu_local_set_shell_job(void *o, bool shell_job)
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_shell_job = true; opts->has_shell_job = true;
opts->shell_job = shell_job; opts->shell_job = shell_job;
} }
void criu_set_file_locks(bool file_locks) void criu_set_shell_job(bool shell_job)
{
criu_local_set_shell_job((void *)global_opts, shell_job);
}
void criu_local_set_file_locks(void *o, bool file_locks)
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_file_locks = true; opts->has_file_locks = true;
opts->file_locks = file_locks; opts->file_locks = file_locks;
} }
void criu_set_log_level(int log_level) void criu_set_file_locks(bool file_locks)
{
criu_local_set_file_locks((void *)global_opts, file_locks);
}
void criu_local_set_log_level(void *o, int log_level)
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_log_level = true; opts->has_log_level = true;
opts->log_level = log_level; opts->log_level = log_level;
} }
void criu_set_root(char *root) void criu_set_log_level(int log_level)
{
criu_local_set_log_level((void *)global_opts, log_level);
}
void criu_local_set_root(void *o, char *root)
{ {
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->root = strdup(root); opts->root = strdup(root);
} }
void criu_set_manage_cgroups(bool manage) void criu_set_root(char *root)
{ {
criu_local_set_root((void *)global_opts, root);
}
void criu_local_set_manage_cgroups(void *o, bool manage)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_manage_cgroups = true; opts->has_manage_cgroups = true;
opts->manage_cgroups = manage; opts->manage_cgroups = manage;
} }
void criu_set_auto_ext_mnt(bool val) void criu_set_manage_cgroups(bool manage)
{ {
criu_local_set_manage_cgroups((void *)global_opts, manage);
}
void criu_local_set_auto_ext_mnt(void *o, bool val)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_auto_ext_mnt = true; opts->has_auto_ext_mnt = true;
opts->auto_ext_mnt = val; opts->auto_ext_mnt = val;
} }
void criu_set_ext_sharing(bool val) void criu_set_auto_ext_mnt(bool val)
{ {
criu_local_set_auto_ext_mnt((void *)global_opts, val);
}
void criu_local_set_ext_sharing(void *o, bool val)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_ext_sharing = true; opts->has_ext_sharing = true;
opts->ext_sharing = val; opts->ext_sharing = val;
} }
void criu_set_ext_masters(bool val) void criu_set_ext_sharing(bool val)
{ {
criu_local_set_ext_sharing((void *)global_opts, val);
}
void criu_local_set_ext_masters(void *o, bool val)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_ext_masters = true; opts->has_ext_masters = true;
opts->ext_masters = val; opts->ext_masters = val;
} }
void criu_set_log_file(char *log_file) void criu_set_ext_masters(bool val)
{ {
criu_local_set_ext_masters((void *)global_opts, val);
}
void criu_local_set_log_file(void *o, char *log_file)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->log_file = strdup(log_file); opts->log_file = strdup(log_file);
} }
void criu_set_cpu_cap(unsigned int cap) void criu_set_log_file(char *log_file)
{ {
criu_local_set_log_file((void *)global_opts, log_file);
}
void criu_local_set_cpu_cap(void *o, unsigned int cap)
{
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->has_cpu_cap = true; opts->has_cpu_cap = true;
opts->cpu_cap = cap; opts->cpu_cap = cap;
} }
int criu_set_exec_cmd(int argc, char *argv[]) void criu_set_cpu_cap(unsigned int cap)
{
criu_local_set_cpu_cap((void *)global_opts, cap);
}
int criu_local_set_exec_cmd(void *o, int argc, char *argv[])
{ {
int i; int i;
CriuOpts *opts;
opts = (CriuOpts *)o;
opts->n_exec_cmd = argc; opts->n_exec_cmd = argc;
opts->exec_cmd = malloc((argc) * sizeof(char *)); opts->exec_cmd = malloc((argc) * sizeof(char *));
...@@ -214,10 +429,18 @@ out: ...@@ -214,10 +429,18 @@ out:
return -ENOMEM; return -ENOMEM;
} }
int criu_add_ext_mount(char *key, char *val) int criu_set_exec_cmd(int argc, char *argv[])
{
return criu_local_set_exec_cmd((void *)global_opts, argc, argv);
}
int criu_local_add_ext_mount(void *o, char *key, char *val)
{ {
int nr; int nr;
ExtMountMap **a, *m; ExtMountMap **a, *m;
CriuOpts *opts;
opts = (CriuOpts *)o;
m = malloc(sizeof(*m)); m = malloc(sizeof(*m));
if (!m) if (!m)
...@@ -251,10 +474,18 @@ er: ...@@ -251,10 +474,18 @@ er:
return -ENOMEM; return -ENOMEM;
} }
int criu_add_cg_root(char *ctrl, char *path) int criu_add_ext_mount(char *key, char *val)
{
return criu_local_add_ext_mount((void *)global_opts, key, val);
}
int criu_local_add_cg_root(void *o, char *ctrl, char *path)
{ {
int nr; int nr;
CgroupRoot **a, *root; CgroupRoot **a, *root;
CriuOpts *opts;
opts = (CriuOpts *)o;
root = malloc(sizeof(*root)); root = malloc(sizeof(*root));
if (!root) if (!root)
...@@ -291,10 +522,19 @@ er_r: ...@@ -291,10 +522,19 @@ er_r:
er: er:
return -ENOMEM; return -ENOMEM;
} }
int criu_add_veth_pair(char *in, char *out)
int criu_add_cg_root(char *ctrl, char *path)
{
return criu_local_add_cg_root((void *)global_opts, ctrl, path);
}
int criu_local_add_veth_pair(void *o, char *in, char *out)
{ {
int nr; int nr;
CriuVethPair **a, *p; CriuVethPair **a, *p;
CriuOpts *opts;
opts = (CriuOpts *)o;
p = malloc(sizeof(*p)); p = malloc(sizeof(*p));
if (!p) if (!p)
...@@ -328,11 +568,19 @@ er: ...@@ -328,11 +568,19 @@ er:
return -ENOMEM; return -ENOMEM;
} }
int criu_add_enable_fs(char *fs) int criu_add_veth_pair(char *in, char *out)
{
return criu_local_add_veth_pair((void *)global_opts, in, out);
}
int criu_local_add_enable_fs(void *o, char *fs)
{ {
int nr; int nr;
char *str = NULL; char *str = NULL;
char **ptr = NULL; char **ptr = NULL;
CriuOpts *opts;
opts = (CriuOpts *)o;
str = strdup(fs); str = strdup(fs);
if (!str) if (!str)
...@@ -359,11 +607,20 @@ err: ...@@ -359,11 +607,20 @@ err:
return -ENOMEM; return -ENOMEM;
} }
int criu_add_skip_mnt(char *mnt) int criu_add_enable_fs(char *fs)
{
return criu_local_add_enable_fs((void *)global_opts, fs);
}
int criu_local_add_skip_mnt(void *o, char *mnt)
{ {
int nr; int nr;
char *str = NULL; char *str = NULL;
char **ptr = NULL; char **ptr = NULL;
CriuOpts *opts;
opts = (CriuOpts *)o;
str = strdup(mnt); str = strdup(mnt);
if (!str) if (!str)
...@@ -390,6 +647,11 @@ err: ...@@ -390,6 +647,11 @@ err:
return -ENOMEM; return -ENOMEM;
} }
int criu_add_skip_mnt(char *mnt)
{
return criu_local_add_skip_mnt((void *)global_opts, mnt);
}
static CriuResp *recv_resp(int socket_fd) static CriuResp *recv_resp(int socket_fd)
{ {
unsigned char *buf; unsigned char *buf;
...@@ -588,11 +850,14 @@ exit: ...@@ -588,11 +850,14 @@ exit:
return ret; return ret;
} }
int criu_dump(void) int criu_local_dump(void *o)
{ {
int ret = -1; int ret = -1;
CriuReq req = CRIU_REQ__INIT; CriuReq req = CRIU_REQ__INIT;
CriuResp *resp = NULL; CriuResp *resp = NULL;
CriuOpts *opts;
opts = (CriuOpts *)o;
saved_errno = 0; saved_errno = 0;
...@@ -620,11 +885,19 @@ exit: ...@@ -620,11 +885,19 @@ exit:
return ret; return ret;
} }
int criu_dump_iters(int (*more)(criu_predump_info pi)) int criu_dump(void)
{
return criu_local_dump((void *)global_opts);
}
int criu_local_dump_iters(void *o, int (*more)(criu_predump_info pi))
{ {
int ret = -1, fd = -1, uret; int ret = -1, fd = -1, uret;
CriuReq req = CRIU_REQ__INIT; CriuReq req = CRIU_REQ__INIT;
CriuResp *resp = NULL; CriuResp *resp = NULL;
CriuOpts *opts;
opts = (CriuOpts *)o;
saved_errno = 0; saved_errno = 0;
...@@ -684,11 +957,19 @@ exit: ...@@ -684,11 +957,19 @@ exit:
return ret; return ret;
} }
int criu_restore(void) int criu_dump_iters(int (*more)(criu_predump_info pi))
{
return criu_local_dump_iters((void *)global_opts, more);
}
int criu_local_restore(void *o)
{ {
int ret = -1; int ret = -1;
CriuReq req = CRIU_REQ__INIT; CriuReq req = CRIU_REQ__INIT;
CriuResp *resp = NULL; CriuResp *resp = NULL;
CriuOpts *opts;
opts = (CriuOpts *)o;
saved_errno = 0; saved_errno = 0;
...@@ -713,11 +994,19 @@ exit: ...@@ -713,11 +994,19 @@ exit:
return ret; return ret;
} }
int criu_restore_child(void) int criu_restore(void)
{
return criu_local_restore((void *)global_opts);
}
int criu_local_restore_child(void *o)
{ {
int sks[2], pid, ret = -1; int sks[2], pid, ret = -1;
CriuReq req = CRIU_REQ__INIT; CriuReq req = CRIU_REQ__INIT;
CriuResp *resp = NULL; CriuResp *resp = NULL;
CriuOpts *opts;
opts = (CriuOpts *)o;
if (socketpair(PF_LOCAL, SOCK_SEQPACKET, 0, sks)) if (socketpair(PF_LOCAL, SOCK_SEQPACKET, 0, sks))
goto out; goto out;
...@@ -782,3 +1071,8 @@ err: ...@@ -782,3 +1071,8 @@ err:
close(sks[0]); close(sks[0]);
goto out; goto out;
} }
int criu_restore_child(void)
{
return criu_local_restore_child((void *)global_opts);
}
...@@ -58,6 +58,8 @@ int criu_add_cg_root(char *ctrl, char *path); ...@@ -58,6 +58,8 @@ int criu_add_cg_root(char *ctrl, char *path);
int criu_add_enable_fs(char *fs); int criu_add_enable_fs(char *fs);
int criu_add_skip_mnt(char *mnt); int criu_add_skip_mnt(char *mnt);
/* /*
* The criu_notify_arg_t na argument is an opaque * The criu_notify_arg_t na argument is an opaque
* value that callbacks (cb-s) should pass into * value that callbacks (cb-s) should pass into
...@@ -113,4 +115,47 @@ int criu_restore_child(void); ...@@ -113,4 +115,47 @@ int criu_restore_child(void);
typedef void *criu_predump_info; typedef void *criu_predump_info;
int criu_dump_iters(int (*more)(criu_predump_info pi)); int criu_dump_iters(int (*more)(criu_predump_info pi));
/*
* Same as the list above, but lets you have your very own options
* structure and lets you set individual options in it.
*/
int criu_local_init_opts(void **opts);
void criu_local_set_pid(void *opts, int pid);
void criu_local_set_images_dir_fd(void *opts, int fd); /* must be set for dump/restore */
void criu_local_set_parent_images(void *opts, char *path);
void criu_local_set_work_dir_fd(void *opts, int fd);
void criu_local_set_leave_running(void *opts, bool leave_running);
void criu_local_set_ext_unix_sk(void *opts, bool ext_unix_sk);
void criu_local_set_tcp_established(void *opts, bool tcp_established);
void criu_local_set_evasive_devices(void *opts, bool evasive_devices);
void criu_local_set_shell_job(void *opts, bool shell_job);
void criu_local_set_file_locks(void *opts, bool file_locks);
void criu_local_set_track_mem(void *opts, bool track_mem);
void criu_local_set_auto_dedup(void *opts, bool auto_dedup);
void criu_local_set_force_irmap(void *opts, bool force_irmap);
void criu_local_set_link_remap(void *opts, bool link_remap);
void criu_local_set_log_level(void *opts, int log_level);
void criu_local_set_log_file(void *opts, char *log_file);
void criu_local_set_cpu_cap(void *opts, unsigned int cap);
void criu_local_set_root(void *opts, char *root);
void criu_local_set_manage_cgroups(void *opts, bool manage);
void criu_local_set_auto_ext_mnt(void *opts, bool val);
void criu_local_set_ext_sharing(void *opts, bool val);
void criu_local_set_ext_masters(void *opts, bool val);
int criu_local_set_exec_cmd(void *opts, int argc, char *argv[]);
int criu_local_add_ext_mount(void *opts, char *key, char *val);
int criu_local_add_veth_pair(void *opts, char *in, char *out);
int criu_local_add_cg_root(void *opts, char *ctrl, char *path);
int criu_local_add_enable_fs(void *opts, char *fs);
int criu_local_add_skip_mnt(void *opts, char *mnt);
void criu_local_set_notify_cb(void *opts, int (*cb)(char *action, criu_notify_arg_t na));
int criu_local_dump(void *opts);
int criu_local_restore(void *opts);
int criu_local_restore_child(void *opts);
int criu_local_dump_iters(void *opts, int (*more)(criu_predump_info pi));
#endif /* __CRIU_LIB_H__ */ #endif /* __CRIU_LIB_H__ */
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment