Unverified Commit 3537c316 by Wolfgang Bumiller Committed by Christian Brauner

cgroup isolation: handle devices cgroup early

Otherwise we cannot use an 'a' entry in devices.deny/allow as these are not permitted once a subdirectory was created. Without isolation we initialize the devices cgroup particularly late, so there are probably cases which cannot work with isolation. Signed-off-by: 's avatarWolfgang Bumiller <w.bumiller@proxmox.com>
parent bba910b2
...@@ -1155,11 +1155,12 @@ static int mkdir_eexist_on_last(const char *dir, mode_t mode) ...@@ -1155,11 +1155,12 @@ static int mkdir_eexist_on_last(const char *dir, mode_t mode)
return 0; return 0;
} }
static bool cgroup_tree_create(struct hierarchy *h, const char *cgroup_tree, static bool cgroup_tree_create(struct cgroup_ops *ops, struct lxc_conf *conf,
struct hierarchy *h, const char *cgroup_tree,
const char *cgroup_leaf, bool payload, const char *cgroup_leaf, bool payload,
const char *cgroup_limit_dir) const char *cgroup_limit_dir)
{ {
__do_free char *path = NULL; __do_free char *path = NULL, *limit_path = NULL;
int ret, ret_cpuset; int ret, ret_cpuset;
path = must_make_path(h->mountpoint, h->container_base_path, cgroup_leaf, NULL); path = must_make_path(h->mountpoint, h->container_base_path, cgroup_leaf, NULL);
...@@ -1170,6 +1171,37 @@ static bool cgroup_tree_create(struct hierarchy *h, const char *cgroup_tree, ...@@ -1170,6 +1171,37 @@ static bool cgroup_tree_create(struct hierarchy *h, const char *cgroup_tree,
if (ret_cpuset < 0) if (ret_cpuset < 0)
return log_error_errno(false, errno, "Failed to handle legacy cpuset controller"); return log_error_errno(false, errno, "Failed to handle legacy cpuset controller");
if (payload && cgroup_limit_dir) {
/* with isolation both parts need to not already exist */
limit_path = must_make_path(h->mountpoint,
h->container_base_path,
cgroup_limit_dir, NULL);
ret = mkdir_eexist_on_last(limit_path, 0755);
if (ret < 0)
return log_error_errno(false, errno,
"Failed to create %s limiting cgroup",
limit_path);
h->cgfd_limit = lxc_open_dirfd(limit_path);
if (h->cgfd_limit < 0)
return log_error_errno(false, errno,
"Failed to open %s", path);
h->container_limit_path = move_ptr(limit_path);
/*
* With isolation the devices legacy cgroup needs to be
* iinitialized early, as it typically contains an 'a' (all)
* line, which is not possible once a subdirectory has been
* created.
*/
if (string_in_list(h->controllers, "devices")) {
ret = ops->setup_limits_legacy(ops, conf, true);
if (ret < 0)
return ret;
}
}
ret = mkdir_eexist_on_last(path, 0755); ret = mkdir_eexist_on_last(path, 0755);
if (ret < 0) { if (ret < 0) {
/* /*
...@@ -1186,16 +1218,10 @@ static bool cgroup_tree_create(struct hierarchy *h, const char *cgroup_tree, ...@@ -1186,16 +1218,10 @@ static bool cgroup_tree_create(struct hierarchy *h, const char *cgroup_tree,
if (h->cgfd_con < 0) if (h->cgfd_con < 0)
return log_error_errno(false, errno, "Failed to open %s", path); return log_error_errno(false, errno, "Failed to open %s", path);
h->container_full_path = move_ptr(path); h->container_full_path = move_ptr(path);
if (cgroup_limit_dir) { if (h->cgfd_limit < 0)
path = must_make_path(h->mountpoint, h->container_base_path, cgroup_limit_dir, NULL);
h->cgfd_limit = lxc_open_dirfd(path);
if (h->cgfd_limit < 0)
return log_error_errno(false, errno, "Failed to open %s", path);
h->container_limit_path = move_ptr(path);
} else {
h->container_limit_path = h->container_full_path;
h->cgfd_limit = h->cgfd_con; h->cgfd_limit = h->cgfd_con;
} if (!h->container_limit_path)
h->container_limit_path = h->container_full_path;
} else { } else {
h->cgfd_mon = lxc_open_dirfd(path); h->cgfd_mon = lxc_open_dirfd(path);
if (h->cgfd_mon < 0) if (h->cgfd_mon < 0)
...@@ -1322,7 +1348,9 @@ __cgfsng_ops static inline bool cgfsng_monitor_create(struct cgroup_ops *ops, ...@@ -1322,7 +1348,9 @@ __cgfsng_ops static inline bool cgfsng_monitor_create(struct cgroup_ops *ops,
sprintf(suffix, "-%d", idx); sprintf(suffix, "-%d", idx);
for (i = 0; ops->hierarchies[i]; i++) { for (i = 0; ops->hierarchies[i]; i++) {
if (cgroup_tree_create(ops->hierarchies[i], cgroup_tree, monitor_cgroup, false, NULL)) if (cgroup_tree_create(ops, handler->conf,
ops->hierarchies[i], cgroup_tree,
monitor_cgroup, false, NULL))
continue; continue;
ERROR("Failed to create cgroup \"%s\"", ops->hierarchies[i]->monitor_full_path ?: "(null)"); ERROR("Failed to create cgroup \"%s\"", ops->hierarchies[i]->monitor_full_path ?: "(null)");
...@@ -1382,9 +1410,14 @@ __cgfsng_ops static inline bool cgfsng_payload_create(struct cgroup_ops *ops, ...@@ -1382,9 +1410,14 @@ __cgfsng_ops static inline bool cgfsng_payload_create(struct cgroup_ops *ops,
if (!limiting_cgroup) if (!limiting_cgroup)
return ret_set_errno(false, ENOMEM); return ret_set_errno(false, ENOMEM);
container_cgroup = must_make_path(limiting_cgroup, if (conf->cgroup_meta.namespace_dir) {
conf->cgroup_meta.namespace_dir, container_cgroup = must_make_path(limiting_cgroup,
NULL); conf->cgroup_meta.namespace_dir,
NULL);
} else {
/* explicit paths but without isolation */
container_cgroup = move_ptr(limiting_cgroup);
}
} else if (conf->cgroup_meta.dir) { } else if (conf->cgroup_meta.dir) {
cgroup_tree = conf->cgroup_meta.dir; cgroup_tree = conf->cgroup_meta.dir;
container_cgroup = must_concat(&len, cgroup_tree, "/", container_cgroup = must_concat(&len, cgroup_tree, "/",
...@@ -1418,7 +1451,8 @@ __cgfsng_ops static inline bool cgfsng_payload_create(struct cgroup_ops *ops, ...@@ -1418,7 +1451,8 @@ __cgfsng_ops static inline bool cgfsng_payload_create(struct cgroup_ops *ops,
sprintf(suffix, "-%d", idx); sprintf(suffix, "-%d", idx);
for (i = 0; ops->hierarchies[i]; i++) { for (i = 0; ops->hierarchies[i]; i++) {
if (cgroup_tree_create(ops->hierarchies[i], cgroup_tree, if (cgroup_tree_create(ops, handler->conf,
ops->hierarchies[i], cgroup_tree,
container_cgroup, true, container_cgroup, true,
limiting_cgroup)) limiting_cgroup))
continue; continue;
......
...@@ -1776,7 +1776,12 @@ static int lxc_spawn(struct lxc_handler *handler) ...@@ -1776,7 +1776,12 @@ static int lxc_spawn(struct lxc_handler *handler)
if (ret < 0) if (ret < 0)
goto out_delete_net; goto out_delete_net;
if (!cgroup_ops->setup_limits_legacy(cgroup_ops, handler->conf, true)) { /*
* with isolation the limiting devices cgroup was already setup, so
* only setup devices here if we have no namespace directory
*/
if (!handler->conf->cgroup_meta.namespace_dir &&
!cgroup_ops->setup_limits_legacy(cgroup_ops, handler->conf, true)) {
ERROR("Failed to setup legacy device cgroup controller limits"); ERROR("Failed to setup legacy device cgroup controller limits");
goto out_delete_net; goto out_delete_net;
} }
......
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