diff options
| author | Zbigniew Jędrzejewski-Szmek <zbyszek@in.waw.pl> | 2017-02-24 18:48:57 -0500 | 
|---|---|---|
| committer | GitHub <noreply@github.com> | 2017-02-24 18:48:57 -0500 | 
| commit | 78e4f19ebc4c128ec72d732068df5b6f8a0e122d (patch) | |
| tree | 9c53c3f264abedd041184d107600a41acfce036b /src/basic/cgroup-util.c | |
| parent | 13e785f7a0d2c78fbd822b1157ef9d7ee3fef805 (diff) | |
| parent | 4bb652ac2fe25d4c85072c9f2f610319ac9981ee (diff) | |
Merge pull request #5444 from poettering/cgroups-revert-no-error
Revert "core: simplify cg_[all_]unified()" and more.
Diffstat (limited to 'src/basic/cgroup-util.c')
| -rw-r--r-- | src/basic/cgroup-util.c | 167 | 
1 files changed, 120 insertions, 47 deletions
| diff --git a/src/basic/cgroup-util.c b/src/basic/cgroup-util.c index f76b7f47e5..bda5c555ad 100644 --- a/src/basic/cgroup-util.c +++ b/src/basic/cgroup-util.c @@ -208,7 +208,13 @@ int cg_rmdir(const char *controller, const char *path) {          if (r < 0 && errno != ENOENT)                  return -errno; -        if (streq(controller, SYSTEMD_CGROUP_CONTROLLER) && cg_hybrid_unified()) { +        r = cg_hybrid_unified(); +        if (r < 0) +                return r; +        if (r == 0) +                return 0; + +        if (streq(controller, SYSTEMD_CGROUP_CONTROLLER)) {                  r = cg_rmdir(SYSTEMD_CGROUP_CONTROLLER_LEGACY, path);                  if (r < 0)                          log_warning_errno(r, "Failed to remove compat systemd cgroup %s: %m", path); @@ -549,7 +555,7 @@ static const char *controller_to_dirname(const char *controller) {           * hierarchies, if it is specified. */          if (streq(controller, SYSTEMD_CGROUP_CONTROLLER)) { -                if (cg_hybrid_unified()) +                if (cg_hybrid_unified() > 0)                          controller = SYSTEMD_CGROUP_CONTROLLER_HYBRID;                  else                          controller = SYSTEMD_CGROUP_CONTROLLER_LEGACY; @@ -636,7 +642,10 @@ int cg_get_path(const char *controller, const char *path, const char *suffix, ch          if (!cg_controller_is_valid(controller))                  return -EINVAL; -        if (cg_all_unified()) +        r = cg_all_unified(); +        if (r < 0) +                return r; +        if (r > 0)                  r = join_path_unified(path, suffix, fs);          else                  r = join_path_legacy(controller, path, suffix, fs); @@ -648,6 +657,7 @@ int cg_get_path(const char *controller, const char *path, const char *suffix, ch  }  static int controller_is_accessible(const char *controller) { +        int r;          assert(controller); @@ -659,7 +669,10 @@ static int controller_is_accessible(const char *controller) {          if (!cg_controller_is_valid(controller))                  return -EINVAL; -        if (cg_all_unified()) { +        r = cg_all_unified(); +        if (r < 0) +                return r; +        if (r > 0) {                  /* We don't support named hierarchies if we are using                   * the unified hierarchy. */ @@ -736,7 +749,10 @@ int cg_trim(const char *controller, const char *path, bool delete_root) {                          return -errno;          } -        if (streq(controller, SYSTEMD_CGROUP_CONTROLLER) && cg_hybrid_unified()) { +        q = cg_hybrid_unified(); +        if (q < 0) +                return q; +        if (q > 0 && streq(controller, SYSTEMD_CGROUP_CONTROLLER)) {                  q = cg_trim(SYSTEMD_CGROUP_CONTROLLER_LEGACY, path, delete_root);                  if (q < 0)                          log_warning_errno(q, "Failed to trim compat systemd cgroup %s: %m", path); @@ -765,7 +781,11 @@ int cg_create(const char *controller, const char *path) {                  return -errno;          } -        if (streq(controller, SYSTEMD_CGROUP_CONTROLLER) && cg_hybrid_unified()) { +        r = cg_hybrid_unified(); +        if (r < 0) +                return r; + +        if (r > 0 && streq(controller, SYSTEMD_CGROUP_CONTROLLER)) {                  r = cg_create(SYSTEMD_CGROUP_CONTROLLER_LEGACY, path);                  if (r < 0)                          log_warning_errno(r, "Failed to create compat systemd cgroup %s: %m", path); @@ -812,7 +832,11 @@ int cg_attach(const char *controller, const char *path, pid_t pid) {          if (r < 0)                  return r; -        if (streq(controller, SYSTEMD_CGROUP_CONTROLLER) && cg_hybrid_unified()) { +        r = cg_hybrid_unified(); +        if (r < 0) +                return r; + +        if (r > 0 && streq(controller, SYSTEMD_CGROUP_CONTROLLER)) {                  r = cg_attach(SYSTEMD_CGROUP_CONTROLLER_LEGACY, path, pid);                  if (r < 0)                          log_warning_errno(r, "Failed to attach %d to compat systemd cgroup %s: %m", pid, path); @@ -871,7 +895,10 @@ int cg_set_group_access(          if (r < 0)                  return r; -        if (streq(controller, SYSTEMD_CGROUP_CONTROLLER) && cg_hybrid_unified()) { +        r = cg_hybrid_unified(); +        if (r < 0) +                return r; +        if (r > 0 && streq(controller, SYSTEMD_CGROUP_CONTROLLER)) {                  r = cg_set_group_access(SYSTEMD_CGROUP_CONTROLLER_LEGACY, path, mode, uid, gid);                  if (r < 0)                          log_warning_errno(r, "Failed to set group access on compat systemd cgroup %s: %m", path); @@ -906,14 +933,20 @@ int cg_set_task_access(          if (r < 0)                  return r; -        if (!cg_unified(controller)) { +        r = cg_unified_controller(controller); +        if (r < 0) +                return r; +        if (r == 0) {                  /* Compatibility, Always keep values for "tasks" in sync with                   * "cgroup.procs" */                  if (cg_get_path(controller, path, "tasks", &procs) >= 0)                          (void) chmod_and_chown(procs, mode, uid, gid);          } -        if (streq(controller, SYSTEMD_CGROUP_CONTROLLER) && cg_hybrid_unified()) { +        r = cg_hybrid_unified(); +        if (r < 0) +                return r; +        if (r > 0 && streq(controller, SYSTEMD_CGROUP_CONTROLLER)) {                  r = cg_set_task_access(SYSTEMD_CGROUP_CONTROLLER_LEGACY, path, mode, uid, gid);                  if (r < 0)                          log_warning_errno(r, "Failed to set task access on compat systemd cgroup %s: %m", path); @@ -964,7 +997,7 @@ int cg_pid_get_path(const char *controller, pid_t pid, char **path) {          char line[LINE_MAX];          const char *fs, *controller_str;          size_t cs = 0; -        bool unified; +        int unified;          assert(path);          assert(pid >= 0); @@ -975,8 +1008,10 @@ int cg_pid_get_path(const char *controller, pid_t pid, char **path) {          } else                  controller = SYSTEMD_CGROUP_CONTROLLER; -        unified = cg_unified(controller); -        if (!unified) { +        unified = cg_unified_controller(controller); +        if (unified < 0) +                return unified; +        if (unified == 0) {                  if (streq(controller, SYSTEMD_CGROUP_CONTROLLER))                          controller_str = SYSTEMD_CGROUP_CONTROLLER_LEGACY;                  else @@ -1048,7 +1083,10 @@ int cg_install_release_agent(const char *controller, const char *agent) {          assert(agent); -        if (cg_unified(controller)) /* doesn't apply to unified hierarchy */ +        r = cg_unified_controller(controller); +        if (r < 0) +                return r; +        if (r > 0) /* doesn't apply to unified hierarchy */                  return -EOPNOTSUPP;          r = cg_get_path(controller, NULL, "release_agent", &fs); @@ -1096,7 +1134,10 @@ int cg_uninstall_release_agent(const char *controller) {          _cleanup_free_ char *fs = NULL;          int r; -        if (cg_unified(controller)) /* Doesn't apply to unified hierarchy */ +        r = cg_unified_controller(controller); +        if (r < 0) +                return r; +        if (r > 0) /* Doesn't apply to unified hierarchy */                  return -EOPNOTSUPP;          r = cg_get_path(controller, NULL, "notify_on_release", &fs); @@ -1149,7 +1190,10 @@ int cg_is_empty_recursive(const char *controller, const char *path) {          if (controller && (isempty(path) || path_equal(path, "/")))                  return false; -        if (cg_unified(controller)) { +        r = cg_unified_controller(controller); +        if (r < 0) +                return r; +        if (r > 0) {                  _cleanup_free_ char *t = NULL;                  /* On the unified hierarchy we can check empty state @@ -2034,7 +2078,10 @@ int cg_create_everywhere(CGroupMask supported, CGroupMask mask, const char *path                  return r;          /* If we are in the unified hierarchy, we are done now */ -        if (cg_all_unified()) +        r = cg_all_unified(); +        if (r < 0) +                return r; +        if (r > 0)                  return 0;          /* Otherwise, do the same in the other hierarchies */ @@ -2061,7 +2108,10 @@ int cg_attach_everywhere(CGroupMask supported, const char *path, pid_t pid, cg_m          if (r < 0)                  return r; -        if (cg_all_unified()) +        r = cg_all_unified(); +        if (r < 0) +                return r; +        if (r > 0)                  return 0;          for (c = 0; c < _CGROUP_CONTROLLER_MAX; c++) { @@ -2102,7 +2152,7 @@ int cg_attach_many_everywhere(CGroupMask supported, const char *path, Set* pids,  int cg_migrate_everywhere(CGroupMask supported, const char *from, const char *to, cg_migrate_callback_t to_callback, void *userdata) {          CGroupController c; -        int r = 0; +        int r = 0, q;          if (!path_equal(from, to))  {                  r = cg_migrate_recursive(SYSTEMD_CGROUP_CONTROLLER, from, SYSTEMD_CGROUP_CONTROLLER, to, CGROUP_REMOVE); @@ -2110,7 +2160,10 @@ int cg_migrate_everywhere(CGroupMask supported, const char *from, const char *to                          return r;          } -        if (cg_all_unified()) +        q = cg_all_unified(); +        if (q < 0) +                return q; +        if (q > 0)                  return r;          for (c = 0; c < _CGROUP_CONTROLLER_MAX; c++) { @@ -2134,13 +2187,16 @@ int cg_migrate_everywhere(CGroupMask supported, const char *from, const char *to  int cg_trim_everywhere(CGroupMask supported, const char *path, bool delete_root) {          CGroupController c; -        int r; +        int r, q;          r = cg_trim(SYSTEMD_CGROUP_CONTROLLER, path, delete_root);          if (r < 0)                  return r; -        if (cg_all_unified()) +        q = cg_all_unified(); +        if (q < 0) +                return q; +        if (q > 0)                  return r;          for (c = 0; c < _CGROUP_CONTROLLER_MAX; c++) { @@ -2163,7 +2219,10 @@ int cg_mask_supported(CGroupMask *ret) {           * includes controllers we can make sense of and that are           * actually accessible. */ -        if (cg_all_unified()) { +        r = cg_all_unified(); +        if (r < 0) +                return r; +        if (r > 0) {                  _cleanup_free_ char *root = NULL, *controllers = NULL, *path = NULL;                  const char *c; @@ -2283,21 +2342,18 @@ int cg_kernel_controllers(Set *controllers) {  static thread_local CGroupUnified unified_cache = CGROUP_UNIFIED_UNKNOWN; -/* The hybrid mode was initially implemented in v232 and simply mounted - * cgroup v2 on /sys/fs/cgroup/systemd.  This unfortunately broke other - * tools (such as docker) which expected the v1 "name=systemd" hierarchy - * on /sys/fs/cgroup/systemd.  From v233 and on, the hybrid mode mountnbs - * v2 on /sys/fs/cgroup/unified and maintains "name=systemd" hierarchy - * on /sys/fs/cgroup/systemd for compatibility with other tools. +/* The hybrid mode was initially implemented in v232 and simply mounted cgroup v2 on /sys/fs/cgroup/systemd.  This + * unfortunately broke other tools (such as docker) which expected the v1 "name=systemd" hierarchy on + * /sys/fs/cgroup/systemd.  From v233 and on, the hybrid mode mountnbs v2 on /sys/fs/cgroup/unified and maintains + * "name=systemd" hierarchy on /sys/fs/cgroup/systemd for compatibility with other tools.   * - * To keep live upgrade working, we detect and support v232 layout.  When - * v232 layout is detected, to keep cgroup v2 process management but - * disable the compat dual layout, we return %true on - * cg_unified(SYSTEMD_CGROUP_CONTROLLER) and %false on cg_hybrid_unified(). + * To keep live upgrade working, we detect and support v232 layout.  When v232 layout is detected, to keep cgroup v2 + * process management but disable the compat dual layout, we return %true on + * cg_unified_controller(SYSTEMD_CGROUP_CONTROLLER) and %false on cg_hybrid_unified().   */  static thread_local bool unified_systemd_v232; -static int cg_update_unified(void) { +static int cg_unified_update(void) {          struct statfs fs; @@ -2336,24 +2392,38 @@ static int cg_update_unified(void) {          return 0;  } -bool cg_unified(const char *controller) { +int cg_unified_controller(const char *controller) { +        int r; + +        r = cg_unified_update(); +        if (r < 0) +                return r; -        assert(cg_update_unified() >= 0); +        if (unified_cache == CGROUP_UNIFIED_NONE) +                return false; -        if (streq_ptr(controller, SYSTEMD_CGROUP_CONTROLLER)) -                return unified_cache >= CGROUP_UNIFIED_SYSTEMD; -        else -                return unified_cache >= CGROUP_UNIFIED_ALL; +        if (unified_cache >= CGROUP_UNIFIED_ALL) +                return true; + +        return streq_ptr(controller, SYSTEMD_CGROUP_CONTROLLER);  } -bool cg_all_unified(void) { +int cg_all_unified(void) { +        int r; -        return cg_unified(NULL); +        r = cg_unified_update(); +        if (r < 0) +                return r; + +        return unified_cache >= CGROUP_UNIFIED_ALL;  } -bool cg_hybrid_unified(void) { +int cg_hybrid_unified(void) { +        int r; -        assert(cg_update_unified() >= 0); +        r = cg_unified_update(); +        if (r < 0) +                return r;          return unified_cache == CGROUP_UNIFIED_SYSTEMD && !unified_systemd_v232;  } @@ -2361,7 +2431,7 @@ bool cg_hybrid_unified(void) {  int cg_unified_flush(void) {          unified_cache = CGROUP_UNIFIED_UNKNOWN; -        return cg_update_unified(); +        return cg_unified_update();  }  int cg_enable_everywhere(CGroupMask supported, CGroupMask mask, const char *p) { @@ -2374,7 +2444,10 @@ int cg_enable_everywhere(CGroupMask supported, CGroupMask mask, const char *p) {          if (supported == 0)                  return 0; -        if (!cg_all_unified()) /* on the legacy hiearchy there's no joining of controllers defined */ +        r = cg_all_unified(); +        if (r < 0) +                return r; +        if (r == 0) /* on the legacy hiearchy there's no joining of controllers defined */                  return 0;          r = cg_get_path(SYSTEMD_CGROUP_CONTROLLER, p, "cgroup.subtree_control", &fs); @@ -2417,7 +2490,7 @@ bool cg_is_unified_wanted(void) {          /* If the hierarchy is already mounted, then follow whatever           * was chosen for it. */          if (cg_unified_flush() >= 0) -                return (wanted = cg_all_unified()); +                return (wanted = unified_cache >= CGROUP_UNIFIED_ALL);          /* Otherwise, let's see what the kernel command line has to say.           * Since checking is expensive, cache a non-error result. */ | 
