render / rpms / libvirt

Forked from rpms/libvirt 11 months ago
Clone
Pablo Greco 40546a
From e1184bde63a7ed92fb99fc4eba4b4abdf6a01815 Mon Sep 17 00:00:00 2001
Pablo Greco 40546a
Message-Id: <e1184bde63a7ed92fb99fc4eba4b4abdf6a01815@dist-git>
Pablo Greco 40546a
From: Pavel Hrdina <phrdina@redhat.com>
Pablo Greco 40546a
Date: Mon, 1 Jul 2019 17:06:27 +0200
Pablo Greco 40546a
Subject: [PATCH] vircgroup: extract v1 detect functions
Pablo Greco 40546a
MIME-Version: 1.0
Pablo Greco 40546a
Content-Type: text/plain; charset=UTF-8
Pablo Greco 40546a
Content-Transfer-Encoding: 8bit
Pablo Greco 40546a
Pablo Greco 40546a
Reviewed-by: Fabiano Fidêncio <fidencio@redhat.com>
Pablo Greco 40546a
Reviewed-by: Ján Tomko <jtomko@redhat.com>
Pablo Greco 40546a
Signed-off-by: Pavel Hrdina <phrdina@redhat.com>
Pablo Greco 40546a
(cherry picked from commit 42a3fcc02bcaec4a0c037a5c59ca63f3fc90e90f)
Pablo Greco 40546a
Pablo Greco 40546a
Resolves: https://bugzilla.redhat.com/show_bug.cgi?id=1689297
Pablo Greco 40546a
Pablo Greco 40546a
Signed-off-by: Pavel Hrdina <phrdina@redhat.com>
Pablo Greco 40546a
Message-Id: <3a918fc4eda51c5e9aff5efc93fe14886470578e.1561993100.git.phrdina@redhat.com>
Pablo Greco 40546a
Reviewed-by: Ján Tomko <jtomko@redhat.com>
Pablo Greco 40546a
---
Pablo Greco 40546a
 src/util/vircgroup.c        | 138 ++-----------------------------
Pablo Greco 40546a
 src/util/vircgroupbackend.h |  14 ++++
Pablo Greco 40546a
 src/util/vircgroupv1.c      | 158 ++++++++++++++++++++++++++++++++++++
Pablo Greco 40546a
 3 files changed, 180 insertions(+), 130 deletions(-)
Pablo Greco 40546a
Pablo Greco 40546a
diff --git a/src/util/vircgroup.c b/src/util/vircgroup.c
Pablo Greco 40546a
index 9d644d37d1..10c99a66fd 100644
Pablo Greco 40546a
--- a/src/util/vircgroup.c
Pablo Greco 40546a
+++ b/src/util/vircgroup.c
Pablo Greco 40546a
@@ -235,82 +235,6 @@ virCgroupPartitionEscape(char **path)
Pablo Greco 40546a
 }
Pablo Greco 40546a
 
Pablo Greco 40546a
 
Pablo Greco 40546a
-static int
Pablo Greco 40546a
-virCgroupResolveMountLink(const char *mntDir,
Pablo Greco 40546a
-                          const char *typeStr,
Pablo Greco 40546a
-                          virCgroupControllerPtr controller)
Pablo Greco 40546a
-{
Pablo Greco 40546a
-    VIR_AUTOFREE(char *) linkSrc = NULL;
Pablo Greco 40546a
-    VIR_AUTOFREE(char *) tmp = NULL;
Pablo Greco 40546a
-    char *dirName;
Pablo Greco 40546a
-    struct stat sb;
Pablo Greco 40546a
-
Pablo Greco 40546a
-    if (VIR_STRDUP(tmp, mntDir) < 0)
Pablo Greco 40546a
-        return -1;
Pablo Greco 40546a
-
Pablo Greco 40546a
-    dirName = strrchr(tmp, '/');
Pablo Greco 40546a
-    if (!dirName) {
Pablo Greco 40546a
-        virReportError(VIR_ERR_INTERNAL_ERROR,
Pablo Greco 40546a
-                       _("Missing '/' separator in cgroup mount '%s'"), tmp);
Pablo Greco 40546a
-        return -1;
Pablo Greco 40546a
-    }
Pablo Greco 40546a
-
Pablo Greco 40546a
-    if (!strchr(dirName + 1, ','))
Pablo Greco 40546a
-        return 0;
Pablo Greco 40546a
-
Pablo Greco 40546a
-    *dirName = '\0';
Pablo Greco 40546a
-    if (virAsprintf(&linkSrc, "%s/%s", tmp, typeStr) < 0)
Pablo Greco 40546a
-        return -1;
Pablo Greco 40546a
-    *dirName = '/';
Pablo Greco 40546a
-
Pablo Greco 40546a
-    if (lstat(linkSrc, &sb) < 0) {
Pablo Greco 40546a
-        if (errno == ENOENT) {
Pablo Greco 40546a
-            VIR_WARN("Controller %s co-mounted at %s is missing symlink at %s",
Pablo Greco 40546a
-                     typeStr, tmp, linkSrc);
Pablo Greco 40546a
-        } else {
Pablo Greco 40546a
-            virReportSystemError(errno, _("Cannot stat %s"), linkSrc);
Pablo Greco 40546a
-            return -1;
Pablo Greco 40546a
-        }
Pablo Greco 40546a
-    } else {
Pablo Greco 40546a
-        if (!S_ISLNK(sb.st_mode)) {
Pablo Greco 40546a
-            VIR_WARN("Expecting a symlink at %s for controller %s",
Pablo Greco 40546a
-                     linkSrc, typeStr);
Pablo Greco 40546a
-        } else {
Pablo Greco 40546a
-            VIR_STEAL_PTR(controller->linkPoint, linkSrc);
Pablo Greco 40546a
-        }
Pablo Greco 40546a
-    }
Pablo Greco 40546a
-
Pablo Greco 40546a
-    return 0;
Pablo Greco 40546a
-}
Pablo Greco 40546a
-
Pablo Greco 40546a
-
Pablo Greco 40546a
-static bool
Pablo Greco 40546a
-virCgroupMountOptsMatchController(const char *mntOpts,
Pablo Greco 40546a
-                                  const char *typeStr)
Pablo Greco 40546a
-{
Pablo Greco 40546a
-    const char *tmp = mntOpts;
Pablo Greco 40546a
-    int typeLen = strlen(typeStr);
Pablo Greco 40546a
-
Pablo Greco 40546a
-    while (tmp) {
Pablo Greco 40546a
-        const char *next = strchr(tmp, ',');
Pablo Greco 40546a
-        int len;
Pablo Greco 40546a
-        if (next) {
Pablo Greco 40546a
-            len = next - tmp;
Pablo Greco 40546a
-            next++;
Pablo Greco 40546a
-        } else {
Pablo Greco 40546a
-            len = strlen(tmp);
Pablo Greco 40546a
-        }
Pablo Greco 40546a
-
Pablo Greco 40546a
-        if (typeLen == len && STREQLEN(typeStr, tmp, len))
Pablo Greco 40546a
-            return true;
Pablo Greco 40546a
-
Pablo Greco 40546a
-        tmp = next;
Pablo Greco 40546a
-    }
Pablo Greco 40546a
-
Pablo Greco 40546a
-    return false;
Pablo Greco 40546a
-}
Pablo Greco 40546a
-
Pablo Greco 40546a
-
Pablo Greco 40546a
 /*
Pablo Greco 40546a
  * Process /proc/mounts figuring out what controllers are
Pablo Greco 40546a
  * mounted and where
Pablo Greco 40546a
@@ -318,7 +242,6 @@ virCgroupMountOptsMatchController(const char *mntOpts,
Pablo Greco 40546a
 static int
Pablo Greco 40546a
 virCgroupDetectMounts(virCgroupPtr group)
Pablo Greco 40546a
 {
Pablo Greco 40546a
-    size_t i;
Pablo Greco 40546a
     FILE *mounts = NULL;
Pablo Greco 40546a
     struct mntent entry;
Pablo Greco 40546a
     char buf[CGROUP_MAX_VAL];
Pablo Greco 40546a
@@ -331,34 +254,11 @@ virCgroupDetectMounts(virCgroupPtr group)
Pablo Greco 40546a
     }
Pablo Greco 40546a
 
Pablo Greco 40546a
     while (getmntent_r(mounts, &entry, buf, sizeof(buf)) != NULL) {
Pablo Greco 40546a
-        if (STRNEQ(entry.mnt_type, "cgroup"))
Pablo Greco 40546a
-            continue;
Pablo Greco 40546a
-
Pablo Greco 40546a
-        for (i = 0; i < VIR_CGROUP_CONTROLLER_LAST; i++) {
Pablo Greco 40546a
-            const char *typestr = virCgroupControllerTypeToString(i);
Pablo Greco 40546a
-
Pablo Greco 40546a
-            if (virCgroupMountOptsMatchController(entry.mnt_opts, typestr)) {
Pablo Greco 40546a
-                /* Note that the lines in /proc/mounts have the same
Pablo Greco 40546a
-                 * order than the mount operations, and that there may
Pablo Greco 40546a
-                 * be duplicates due to bind mounts. This means
Pablo Greco 40546a
-                 * that the same mount point may be processed more than
Pablo Greco 40546a
-                 * once. We need to save the results of the last one,
Pablo Greco 40546a
-                 * and we need to be careful to release the memory used
Pablo Greco 40546a
-                 * by previous processing. */
Pablo Greco 40546a
-                virCgroupControllerPtr controller = &group->controllers[i];
Pablo Greco 40546a
-
Pablo Greco 40546a
-                VIR_FREE(controller->mountPoint);
Pablo Greco 40546a
-                VIR_FREE(controller->linkPoint);
Pablo Greco 40546a
-                if (VIR_STRDUP(controller->mountPoint, entry.mnt_dir) < 0)
Pablo Greco 40546a
-                    goto cleanup;
Pablo Greco 40546a
-
Pablo Greco 40546a
-                /* If it is a co-mount it has a filename like "cpu,cpuacct"
Pablo Greco 40546a
-                 * and we must identify the symlink path */
Pablo Greco 40546a
-                if (virCgroupResolveMountLink(entry.mnt_dir, typestr,
Pablo Greco 40546a
-                                              controller) < 0) {
Pablo Greco 40546a
-                    goto cleanup;
Pablo Greco 40546a
-                }
Pablo Greco 40546a
-            }
Pablo Greco 40546a
+        if (group->backend->detectMounts(group,
Pablo Greco 40546a
+                                         entry.mnt_type,
Pablo Greco 40546a
+                                         entry.mnt_opts,
Pablo Greco 40546a
+                                         entry.mnt_dir) < 0) {
Pablo Greco 40546a
+            goto cleanup;
Pablo Greco 40546a
         }
Pablo Greco 40546a
     }
Pablo Greco 40546a
 
Pablo Greco 40546a
@@ -432,7 +332,6 @@ virCgroupDetectPlacement(virCgroupPtr group,
Pablo Greco 40546a
                          pid_t pid,
Pablo Greco 40546a
                          const char *path)
Pablo Greco 40546a
 {
Pablo Greco 40546a
-    size_t i;
Pablo Greco 40546a
     FILE *mapping  = NULL;
Pablo Greco 40546a
     char line[1024];
Pablo Greco 40546a
     int ret = -1;
Pablo Greco 40546a
@@ -472,30 +371,9 @@ virCgroupDetectPlacement(virCgroupPtr group,
Pablo Greco 40546a
         controllers++;
Pablo Greco 40546a
         selfpath++;
Pablo Greco 40546a
 
Pablo Greco 40546a
-        for (i = 0; i < VIR_CGROUP_CONTROLLER_LAST; i++) {
Pablo Greco 40546a
-            const char *typestr = virCgroupControllerTypeToString(i);
Pablo Greco 40546a
-
Pablo Greco 40546a
-            if (virCgroupMountOptsMatchController(controllers, typestr) &&
Pablo Greco 40546a
-                group->controllers[i].mountPoint != NULL &&
Pablo Greco 40546a
-                group->controllers[i].placement == NULL) {
Pablo Greco 40546a
-                /*
Pablo Greco 40546a
-                 * selfpath == "/" + path="" -> "/"
Pablo Greco 40546a
-                 * selfpath == "/libvirt.service" + path == "" -> "/libvirt.service"
Pablo Greco 40546a
-                 * selfpath == "/libvirt.service" + path == "foo" -> "/libvirt.service/foo"
Pablo Greco 40546a
-                 */
Pablo Greco 40546a
-                if (i == VIR_CGROUP_CONTROLLER_SYSTEMD) {
Pablo Greco 40546a
-                    if (VIR_STRDUP(group->controllers[i].placement,
Pablo Greco 40546a
-                                   selfpath) < 0)
Pablo Greco 40546a
-                        goto cleanup;
Pablo Greco 40546a
-                } else {
Pablo Greco 40546a
-                    if (virAsprintf(&group->controllers[i].placement,
Pablo Greco 40546a
-                                    "%s%s%s", selfpath,
Pablo Greco 40546a
-                                    (STREQ(selfpath, "/") ||
Pablo Greco 40546a
-                                     STREQ(path, "") ? "" : "/"),
Pablo Greco 40546a
-                                    path) < 0)
Pablo Greco 40546a
-                        goto cleanup;
Pablo Greco 40546a
-                }
Pablo Greco 40546a
-            }
Pablo Greco 40546a
+        if (group->backend->detectPlacement(group, path, controllers,
Pablo Greco 40546a
+                                            selfpath) < 0) {
Pablo Greco 40546a
+            goto cleanup;
Pablo Greco 40546a
         }
Pablo Greco 40546a
     }
Pablo Greco 40546a
 
Pablo Greco 40546a
diff --git a/src/util/vircgroupbackend.h b/src/util/vircgroupbackend.h
Pablo Greco 40546a
index 81ee597fc8..fadc7efdcf 100644
Pablo Greco 40546a
--- a/src/util/vircgroupbackend.h
Pablo Greco 40546a
+++ b/src/util/vircgroupbackend.h
Pablo Greco 40546a
@@ -45,6 +45,18 @@ typedef int
Pablo Greco 40546a
 (*virCgroupCopyMountsCB)(virCgroupPtr group,
Pablo Greco 40546a
                          virCgroupPtr parent);
Pablo Greco 40546a
 
Pablo Greco 40546a
+typedef int
Pablo Greco 40546a
+(*virCgroupDetectMountsCB)(virCgroupPtr group,
Pablo Greco 40546a
+                           const char *mntType,
Pablo Greco 40546a
+                           const char *mntOpts,
Pablo Greco 40546a
+                           const char *mntDir);
Pablo Greco 40546a
+
Pablo Greco 40546a
+typedef int
Pablo Greco 40546a
+(*virCgroupDetectPlacementCB)(virCgroupPtr group,
Pablo Greco 40546a
+                              const char *path,
Pablo Greco 40546a
+                              const char *controllers,
Pablo Greco 40546a
+                              const char *selfpath);
Pablo Greco 40546a
+
Pablo Greco 40546a
 struct _virCgroupBackend {
Pablo Greco 40546a
     virCgroupBackendType type;
Pablo Greco 40546a
 
Pablo Greco 40546a
@@ -52,6 +64,8 @@ struct _virCgroupBackend {
Pablo Greco 40546a
     virCgroupAvailableCB available;
Pablo Greco 40546a
     virCgroupValidateMachineGroupCB validateMachineGroup;
Pablo Greco 40546a
     virCgroupCopyMountsCB copyMounts;
Pablo Greco 40546a
+    virCgroupDetectMountsCB detectMounts;
Pablo Greco 40546a
+    virCgroupDetectPlacementCB detectPlacement;
Pablo Greco 40546a
 };
Pablo Greco 40546a
 typedef struct _virCgroupBackend virCgroupBackend;
Pablo Greco 40546a
 typedef virCgroupBackend *virCgroupBackendPtr;
Pablo Greco 40546a
diff --git a/src/util/vircgroupv1.c b/src/util/vircgroupv1.c
Pablo Greco 40546a
index 50b58ab413..bd9f28f6e9 100644
Pablo Greco 40546a
--- a/src/util/vircgroupv1.c
Pablo Greco 40546a
+++ b/src/util/vircgroupv1.c
Pablo Greco 40546a
@@ -23,6 +23,7 @@
Pablo Greco 40546a
 #if defined HAVE_MNTENT_H && defined HAVE_GETMNTENT_R
Pablo Greco 40546a
 # include <mntent.h>
Pablo Greco 40546a
 #endif
Pablo Greco 40546a
+#include <sys/stat.h>
Pablo Greco 40546a
 
Pablo Greco 40546a
 #include "internal.h"
Pablo Greco 40546a
 
Pablo Greco 40546a
@@ -38,6 +39,7 @@
Pablo Greco 40546a
 #include "virlog.h"
Pablo Greco 40546a
 #include "virstring.h"
Pablo Greco 40546a
 #include "virsystemd.h"
Pablo Greco 40546a
+#include "virerror.h"
Pablo Greco 40546a
 
Pablo Greco 40546a
 VIR_LOG_INIT("util.cgroup");
Pablo Greco 40546a
 
Pablo Greco 40546a
@@ -181,12 +183,168 @@ virCgroupV1CopyMounts(virCgroupPtr group,
Pablo Greco 40546a
 }
Pablo Greco 40546a
 
Pablo Greco 40546a
 
Pablo Greco 40546a
+static int
Pablo Greco 40546a
+virCgroupV1ResolveMountLink(const char *mntDir,
Pablo Greco 40546a
+                            const char *typeStr,
Pablo Greco 40546a
+                            virCgroupControllerPtr controller)
Pablo Greco 40546a
+{
Pablo Greco 40546a
+    VIR_AUTOFREE(char *) linkSrc = NULL;
Pablo Greco 40546a
+    VIR_AUTOFREE(char *) tmp = NULL;
Pablo Greco 40546a
+    char *dirName;
Pablo Greco 40546a
+    struct stat sb;
Pablo Greco 40546a
+
Pablo Greco 40546a
+    if (VIR_STRDUP(tmp, mntDir) < 0)
Pablo Greco 40546a
+        return -1;
Pablo Greco 40546a
+
Pablo Greco 40546a
+    dirName = strrchr(tmp, '/');
Pablo Greco 40546a
+    if (!dirName) {
Pablo Greco 40546a
+        virReportError(VIR_ERR_INTERNAL_ERROR,
Pablo Greco 40546a
+                       _("Missing '/' separator in cgroup mount '%s'"), tmp);
Pablo Greco 40546a
+        return -1;
Pablo Greco 40546a
+    }
Pablo Greco 40546a
+
Pablo Greco 40546a
+    if (!strchr(dirName + 1, ','))
Pablo Greco 40546a
+        return 0;
Pablo Greco 40546a
+
Pablo Greco 40546a
+    *dirName = '\0';
Pablo Greco 40546a
+    if (virAsprintf(&linkSrc, "%s/%s", tmp, typeStr) < 0)
Pablo Greco 40546a
+        return -1;
Pablo Greco 40546a
+    *dirName = '/';
Pablo Greco 40546a
+
Pablo Greco 40546a
+    if (lstat(linkSrc, &sb) < 0) {
Pablo Greco 40546a
+        if (errno == ENOENT) {
Pablo Greco 40546a
+            VIR_WARN("Controller %s co-mounted at %s is missing symlink at %s",
Pablo Greco 40546a
+                     typeStr, tmp, linkSrc);
Pablo Greco 40546a
+        } else {
Pablo Greco 40546a
+            virReportSystemError(errno, _("Cannot stat %s"), linkSrc);
Pablo Greco 40546a
+            return -1;
Pablo Greco 40546a
+        }
Pablo Greco 40546a
+    } else {
Pablo Greco 40546a
+        if (!S_ISLNK(sb.st_mode)) {
Pablo Greco 40546a
+            VIR_WARN("Expecting a symlink at %s for controller %s",
Pablo Greco 40546a
+                     linkSrc, typeStr);
Pablo Greco 40546a
+        } else {
Pablo Greco 40546a
+            VIR_STEAL_PTR(controller->linkPoint, linkSrc);
Pablo Greco 40546a
+        }
Pablo Greco 40546a
+    }
Pablo Greco 40546a
+
Pablo Greco 40546a
+    return 0;
Pablo Greco 40546a
+}
Pablo Greco 40546a
+
Pablo Greco 40546a
+
Pablo Greco 40546a
+static bool
Pablo Greco 40546a
+virCgroupV1MountOptsMatchController(const char *mntOpts,
Pablo Greco 40546a
+                                    const char *typeStr)
Pablo Greco 40546a
+{
Pablo Greco 40546a
+    const char *tmp = mntOpts;
Pablo Greco 40546a
+    int typeLen = strlen(typeStr);
Pablo Greco 40546a
+
Pablo Greco 40546a
+    while (tmp) {
Pablo Greco 40546a
+        const char *next = strchr(tmp, ',');
Pablo Greco 40546a
+        int len;
Pablo Greco 40546a
+        if (next) {
Pablo Greco 40546a
+            len = next - tmp;
Pablo Greco 40546a
+            next++;
Pablo Greco 40546a
+        } else {
Pablo Greco 40546a
+            len = strlen(tmp);
Pablo Greco 40546a
+        }
Pablo Greco 40546a
+
Pablo Greco 40546a
+        if (typeLen == len && STREQLEN(typeStr, tmp, len))
Pablo Greco 40546a
+            return true;
Pablo Greco 40546a
+
Pablo Greco 40546a
+        tmp = next;
Pablo Greco 40546a
+    }
Pablo Greco 40546a
+
Pablo Greco 40546a
+    return false;
Pablo Greco 40546a
+}
Pablo Greco 40546a
+
Pablo Greco 40546a
+
Pablo Greco 40546a
+static int
Pablo Greco 40546a
+virCgroupV1DetectMounts(virCgroupPtr group,
Pablo Greco 40546a
+                        const char *mntType,
Pablo Greco 40546a
+                        const char *mntOpts,
Pablo Greco 40546a
+                        const char *mntDir)
Pablo Greco 40546a
+{
Pablo Greco 40546a
+    size_t i;
Pablo Greco 40546a
+
Pablo Greco 40546a
+    if (STRNEQ(mntType, "cgroup"))
Pablo Greco 40546a
+        return 0;
Pablo Greco 40546a
+
Pablo Greco 40546a
+    for (i = 0; i < VIR_CGROUP_CONTROLLER_LAST; i++) {
Pablo Greco 40546a
+        const char *typestr = virCgroupV1ControllerTypeToString(i);
Pablo Greco 40546a
+
Pablo Greco 40546a
+        if (virCgroupV1MountOptsMatchController(mntOpts, typestr)) {
Pablo Greco 40546a
+            /* Note that the lines in /proc/mounts have the same
Pablo Greco 40546a
+             * order than the mount operations, and that there may
Pablo Greco 40546a
+             * be duplicates due to bind mounts. This means
Pablo Greco 40546a
+             * that the same mount point may be processed more than
Pablo Greco 40546a
+             * once. We need to save the results of the last one,
Pablo Greco 40546a
+             * and we need to be careful to release the memory used
Pablo Greco 40546a
+             * by previous processing. */
Pablo Greco 40546a
+            virCgroupControllerPtr controller = &group->controllers[i];
Pablo Greco 40546a
+
Pablo Greco 40546a
+            VIR_FREE(controller->mountPoint);
Pablo Greco 40546a
+            VIR_FREE(controller->linkPoint);
Pablo Greco 40546a
+            if (VIR_STRDUP(controller->mountPoint, mntDir) < 0)
Pablo Greco 40546a
+                return -1;
Pablo Greco 40546a
+
Pablo Greco 40546a
+            /* If it is a co-mount it has a filename like "cpu,cpuacct"
Pablo Greco 40546a
+             * and we must identify the symlink path */
Pablo Greco 40546a
+            if (virCgroupV1ResolveMountLink(mntDir, typestr, controller) < 0)
Pablo Greco 40546a
+                return -1;
Pablo Greco 40546a
+        }
Pablo Greco 40546a
+    }
Pablo Greco 40546a
+
Pablo Greco 40546a
+    return 0;
Pablo Greco 40546a
+}
Pablo Greco 40546a
+
Pablo Greco 40546a
+
Pablo Greco 40546a
+static int
Pablo Greco 40546a
+virCgroupV1DetectPlacement(virCgroupPtr group,
Pablo Greco 40546a
+                           const char *path,
Pablo Greco 40546a
+                           const char *controllers,
Pablo Greco 40546a
+                           const char *selfpath)
Pablo Greco 40546a
+{
Pablo Greco 40546a
+    size_t i;
Pablo Greco 40546a
+
Pablo Greco 40546a
+    for (i = 0; i < VIR_CGROUP_CONTROLLER_LAST; i++) {
Pablo Greco 40546a
+        const char *typestr = virCgroupV1ControllerTypeToString(i);
Pablo Greco 40546a
+
Pablo Greco 40546a
+        if (virCgroupV1MountOptsMatchController(controllers, typestr) &&
Pablo Greco 40546a
+            group->controllers[i].mountPoint != NULL &&
Pablo Greco 40546a
+            group->controllers[i].placement == NULL) {
Pablo Greco 40546a
+            /*
Pablo Greco 40546a
+             * selfpath == "/" + path="" -> "/"
Pablo Greco 40546a
+             * selfpath == "/libvirt.service" + path == "" -> "/libvirt.service"
Pablo Greco 40546a
+             * selfpath == "/libvirt.service" + path == "foo" -> "/libvirt.service/foo"
Pablo Greco 40546a
+             */
Pablo Greco 40546a
+            if (i == VIR_CGROUP_CONTROLLER_SYSTEMD) {
Pablo Greco 40546a
+                if (VIR_STRDUP(group->controllers[i].placement,
Pablo Greco 40546a
+                               selfpath) < 0)
Pablo Greco 40546a
+                    return -1;
Pablo Greco 40546a
+            } else {
Pablo Greco 40546a
+                if (virAsprintf(&group->controllers[i].placement,
Pablo Greco 40546a
+                                "%s%s%s", selfpath,
Pablo Greco 40546a
+                                (STREQ(selfpath, "/") ||
Pablo Greco 40546a
+                                 STREQ(path, "") ? "" : "/"),
Pablo Greco 40546a
+                                path) < 0)
Pablo Greco 40546a
+                    return -1;
Pablo Greco 40546a
+            }
Pablo Greco 40546a
+        }
Pablo Greco 40546a
+    }
Pablo Greco 40546a
+
Pablo Greco 40546a
+    return 0;
Pablo Greco 40546a
+}
Pablo Greco 40546a
+
Pablo Greco 40546a
+
Pablo Greco 40546a
 virCgroupBackend virCgroupV1Backend = {
Pablo Greco 40546a
     .type = VIR_CGROUP_BACKEND_TYPE_V1,
Pablo Greco 40546a
 
Pablo Greco 40546a
     .available = virCgroupV1Available,
Pablo Greco 40546a
     .validateMachineGroup = virCgroupV1ValidateMachineGroup,
Pablo Greco 40546a
     .copyMounts = virCgroupV1CopyMounts,
Pablo Greco 40546a
+    .detectMounts = virCgroupV1DetectMounts,
Pablo Greco 40546a
+    .detectPlacement = virCgroupV1DetectPlacement,
Pablo Greco 40546a
 };
Pablo Greco 40546a
 
Pablo Greco 40546a
 
Pablo Greco 40546a
-- 
Pablo Greco 40546a
2.22.0
Pablo Greco 40546a