[RFC PATCH 1/8] virCgroupV1GetBlkioIo(Device)Serviced: Refactor extraction of cgroup data

Peter Krempa pkrempa at redhat.com
Thu Apr 13 08:15:49 UTC 2023


Rewrite the code to improve maintainability and also re-do construction
of error messages which are assembled from non-translatable parts.

Closes: https://gitlab.com/libvirt/libvirt/-/issues/455
Signed-off-by: Peter Krempa <pkrempa at redhat.com>
---
 src/util/vircgroupv1.c | 264 ++++++++++++++++++-----------------------
 1 file changed, 117 insertions(+), 147 deletions(-)

diff --git a/src/util/vircgroupv1.c b/src/util/vircgroupv1.c
index c0cac2a6b6..77c7e209ce 100644
--- a/src/util/vircgroupv1.c
+++ b/src/util/vircgroupv1.c
@@ -1051,189 +1051,159 @@ virCgroupV1GetBlkioWeight(virCgroup *group,
 }


+
 static int
-virCgroupV1GetBlkioIoServiced(virCgroup *group,
-                              long long *bytes_read,
-                              long long *bytes_write,
-                              long long *requests_read,
-                              long long *requests_write)
+virCgroupV1GetBlkioIoServicedOne(virCgroup *group,
+                                 const char *field,
+                                 const char *devpath,
+                                 long long *dataRead,
+                                 long long *dataWrite)
 {
-    long long stats_val;
-    g_autofree char *str1 = NULL;
-    g_autofree char *str2 = NULL;
-    char *p1 = NULL;
-    char *p2 = NULL;
-    size_t i;
+    g_autofree char *serviced = NULL;
+    g_autofree char *filterWrite = NULL;
+    g_autofree char *filterRead = NULL;
+    unsigned long long tmpval;
+    char *tmp;
+    size_t len;

-    const char *value_names[] = {
-        "Read ",
-        "Write "
-    };
-    long long *bytes_ptrs[] = {
-        bytes_read,
-        bytes_write
-    };
-    long long *requests_ptrs[] = {
-        requests_read,
-        requests_write
-    };
-
-    *bytes_read = 0;
-    *bytes_write = 0;
-    *requests_read = 0;
-    *requests_write = 0;
-
-    if (virCgroupGetValueStr(group,
-                             VIR_CGROUP_CONTROLLER_BLKIO,
-                             "blkio.throttle.io_service_bytes", &str1) < 0)
-        return -1;
+    *dataRead = 0;
+    *dataWrite = 0;

-    if (virCgroupGetValueStr(group,
-                             VIR_CGROUP_CONTROLLER_BLKIO,
-                             "blkio.throttle.io_serviced", &str2) < 0)
+    if (virCgroupGetValueStr(group, VIR_CGROUP_CONTROLLER_BLKIO, field, &serviced) < 0)
         return -1;

-    /* sum up all entries of the same kind, from all devices */
-    for (i = 0; i < G_N_ELEMENTS(value_names); i++) {
-        p1 = str1;
-        p2 = str2;
-
-        while ((p1 = strstr(p1, value_names[i]))) {
-            p1 += strlen(value_names[i]);
-            if (virStrToLong_ll(p1, &p1, 10, &stats_val) < 0) {
-                virReportError(VIR_ERR_INTERNAL_ERROR,
-                               _("Cannot parse byte %1$sstat '%2$s'"),
-                               value_names[i],
-                               p1);
-                return -1;
-            }
+    if (devpath) {
+        g_autofree char *devstr = NULL;

-            if (stats_val < 0 ||
-                (stats_val > 0 && *bytes_ptrs[i] > (LLONG_MAX - stats_val)))
-            {
-                virReportError(VIR_ERR_OVERFLOW,
-                               _("Sum of byte %1$sstat overflows"),
-                               value_names[i]);
-                return -1;
-            }
-            *bytes_ptrs[i] += stats_val;
-        }
+        if (!(devstr = virCgroupGetBlockDevString(devpath)))
+            return -1;

-        while ((p2 = strstr(p2, value_names[i]))) {
-            p2 += strlen(value_names[i]);
-            if (virStrToLong_ll(p2, &p2, 10, &stats_val) < 0) {
-                virReportError(VIR_ERR_INTERNAL_ERROR,
-                               _("Cannot parse %1$srequest stat '%2$s'"),
-                               value_names[i],
-                               p2);
-                return -1;
-            }
+        filterWrite = g_strdup_printf("%sWrite ", devstr);
+        filterRead = g_strdup_printf("%sRead ", devstr);

-            if (stats_val < 0 ||
-                (stats_val > 0 && *requests_ptrs[i] > (LLONG_MAX - stats_val)))
-            {
-                virReportError(VIR_ERR_OVERFLOW,
-                               _("Sum of %1$srequest stat overflows"),
-                               value_names[i]);
-                return -1;
-            }
-            *requests_ptrs[i] += stats_val;
+        if (!strstr(serviced, filterWrite) ||
+            !strstr(serviced, filterRead)) {
+            virReportError(VIR_ERR_INTERNAL_ERROR,
+                           _("Cannot find blkio cgroup stats (%1$s) for block device '%2$s' (%3$s)"),
+                           field, devstr, devpath);
+            return -1;
         }
+    } else {
+        filterWrite = g_strdup("Write ");
+        filterRead = g_strdup("Read ");
     }

-    return 0;
-}
-
-
-static int
-virCgroupV1GetBlkioIoDeviceServiced(virCgroup *group,
-                                    const char *path,
-                                    long long *bytes_read,
-                                    long long *bytes_write,
-                                    long long *requests_read,
-                                    long long *requests_write)
-{
-    g_autofree char *str1 = NULL;
-    g_autofree char *str2 = NULL;
-    g_autofree char *str3 = NULL;
-    char *p1 = NULL;
-    char *p2 = NULL;
-    size_t i;
-
-    const char *value_names[] = {
-        "Read ",
-        "Write "
-    };
-    long long *bytes_ptrs[] = {
-        bytes_read,
-        bytes_write
-    };
-    long long *requests_ptrs[] = {
-        requests_read,
-        requests_write
-    };
-
-    if (virCgroupGetValueStr(group,
-                             VIR_CGROUP_CONTROLLER_BLKIO,
-                             "blkio.throttle.io_service_bytes", &str1) < 0)
-        return -1;
+    len = strlen(filterRead);

-    if (virCgroupGetValueStr(group,
-                             VIR_CGROUP_CONTROLLER_BLKIO,
-                             "blkio.throttle.io_serviced", &str2) < 0)
-        return -1;
+    for (tmp = strstr(serviced, filterRead); tmp; tmp = strstr(tmp, filterRead)) {
+        char *cur = tmp;
+        tmp += len;

-    if (!(str3 = virCgroupGetBlockDevString(path)))
-        return -1;
+        VIR_DEBUG("filter='%s' line='%.*s'", filterRead, (int) (strchr(tmp, '\n') - tmp), tmp);

-    if (!(p1 = strstr(str1, str3))) {
-        virReportError(VIR_ERR_INTERNAL_ERROR,
-                       _("Cannot find byte stats for block device '%1$s'"),
-                       str3);
-        return -1;
-    }
+        if (virStrToLong_ullp(tmp, &tmp, 10, &tmpval) < 0) {
+            char *eol;

-    if (!(p2 = strstr(str2, str3))) {
-        virReportError(VIR_ERR_INTERNAL_ERROR,
-                       _("Cannot find request stats for block device '%1$s'"),
-                       str3);
-        return -1;
-    }
+            if ((eol = strchr(cur, '\n')))
+                *eol = '\0';

-    for (i = 0; i < G_N_ELEMENTS(value_names); i++) {
-        if (!(p1 = strstr(p1, value_names[i]))) {
             virReportError(VIR_ERR_INTERNAL_ERROR,
-                           _("Cannot find byte %1$sstats for block device '%2$s'"),
-                           value_names[i], str3);
+                           _("Cannot parse blkio cgroup (%1$s) entry '%2$s'"),
+                           field, cur);
             return -1;
         }

-        if (virStrToLong_ll(p1 + strlen(value_names[i]), &p1, 10, bytes_ptrs[i]) < 0) {
-            virReportError(VIR_ERR_INTERNAL_ERROR,
-                           _("Cannot parse %1$sstat '%2$s'"),
-                           value_names[i], p1 + strlen(value_names[i]));
+        if (tmpval + *dataRead > LLONG_MAX) {
+            virReportError(VIR_ERR_OVERFLOW,
+                           _("overflow in sum of statistic for blkio cgroup (%1$s) field '%2$s'"),
+                           field, filterRead);
             return -1;
         }

-        if (!(p2 = strstr(p2, value_names[i]))) {
+        *dataRead += tmpval;
+    }
+
+    len = strlen(filterWrite);
+
+    for (tmp = strstr(serviced, filterWrite); tmp; tmp = strstr(tmp, filterWrite)) {
+        char *cur = tmp;
+        tmp += len;
+
+        VIR_DEBUG("filter='%s' line='%.*s'", filterWrite, (int) (strchr(cur, '\n') - cur), cur);
+
+        if (virStrToLong_ullp(tmp, &tmp, 10, &tmpval) < 0) {
+            char *eol;
+
+            if ((eol = strchr(cur, '\n')))
+                *eol = '\0';
+
             virReportError(VIR_ERR_INTERNAL_ERROR,
-                           _("Cannot find request %1$sstats for block device '%2$s'"),
-                           value_names[i], str3);
+                           _("Cannot parse blkio cgroup ('%1$s') entry '%2$s'"),
+                           field, cur);
             return -1;
         }

-        if (virStrToLong_ll(p2 + strlen(value_names[i]), &p2, 10, requests_ptrs[i]) < 0) {
-            virReportError(VIR_ERR_INTERNAL_ERROR,
-                           _("Cannot parse %1$sstat '%2$s'"),
-                           value_names[i], p2 + strlen(value_names[i]));
+        if (tmpval + *dataWrite > LLONG_MAX) {
+            virReportError(VIR_ERR_OVERFLOW,
+                           _("overflow in sum of statistic for blkio cgroup (%1$s) field '%2$s'"),
+                           field, filterWrite);
             return -1;
         }
+
+        *dataWrite += tmpval;
     }

     return 0;
 }


+static int
+virCgroupV1GetBlkioIoServicedInternal(virCgroup *group,
+                                      const char *devpath,
+                                      long long *bytes_read,
+                                      long long *bytes_write,
+                                      long long *requests_read,
+                                      long long *requests_write)
+{
+    if (virCgroupV1GetBlkioIoServicedOne(group, "blkio.throttle.io_service_bytes",
+                                         devpath, bytes_read, bytes_write) < 0)
+        return -1;
+
+    if (virCgroupV1GetBlkioIoServicedOne(group, "blkio.throttle.io_serviced",
+                                         devpath, requests_read, requests_write) < 0)
+        return -1;
+
+    return 0;
+}
+
+
+static int
+virCgroupV1GetBlkioIoServiced(virCgroup *group,
+                              long long *bytes_read,
+                              long long *bytes_write,
+                              long long *requests_read,
+                              long long *requests_write)
+{
+    return virCgroupV1GetBlkioIoServicedInternal(group, NULL,
+                                                 bytes_read, bytes_write,
+                                                 requests_read, requests_write);
+}
+
+
+static int
+virCgroupV1GetBlkioIoDeviceServiced(virCgroup *group,
+                                    const char *path,
+                                    long long *bytes_read,
+                                    long long *bytes_write,
+                                    long long *requests_read,
+                                    long long *requests_write)
+{
+    return virCgroupV1GetBlkioIoServicedInternal(group, path,
+                                                 bytes_read, bytes_write,
+                                                 requests_read, requests_write);
+}
+
+
 static int
 virCgroupV1SetBlkioDeviceWeight(virCgroup *group,
                                 const char *devPath,
-- 
2.39.2



More information about the libvir-list mailing list