Zbigniew Jędrzejewski-Szmek 62fe94
From 4f52d3fe2da7c3449b7fbfaa7c64a83354d3b56c Mon Sep 17 00:00:00 2001
Zbigniew Jędrzejewski-Szmek 62fe94
From: =?UTF-8?q?Zbigniew=20J=C4=99drzejewski-Szmek?= <zbyszek@in.waw.pl>
Zbigniew Jędrzejewski-Szmek 62fe94
Date: Sat, 27 Sep 2014 22:02:04 -0400
Zbigniew Jędrzejewski-Szmek 62fe94
Subject: [PATCH] fstab-generator: properly deal with discard as non-last
Zbigniew Jędrzejewski-Szmek 62fe94
 option
Zbigniew Jędrzejewski-Szmek 62fe94
Zbigniew Jędrzejewski-Szmek 62fe94
Previous code would only return correct results when discard
Zbigniew Jędrzejewski-Szmek 62fe94
was the last option.
Zbigniew Jędrzejewski-Szmek 62fe94
Zbigniew Jędrzejewski-Szmek 62fe94
While at it, avoid incorrect behaviour for (invalid) 'pri' option
Zbigniew Jędrzejewski-Szmek 62fe94
not followed by '=...', and also do not return -1 as the error code.
Zbigniew Jędrzejewski-Szmek 62fe94
---
Zbigniew Jędrzejewski-Szmek 62fe94
 src/core/swap.c                       |  2 +-
Zbigniew Jędrzejewski-Szmek 62fe94
 src/fstab-generator/fstab-generator.c | 73 ++++++++++++++++++++++++++---------
Zbigniew Jędrzejewski-Szmek 62fe94
 2 files changed, 55 insertions(+), 20 deletions(-)
Zbigniew Jędrzejewski-Szmek 62fe94
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/src/core/swap.c b/src/core/swap.c
Zbigniew Jędrzejewski-Szmek 62fe94
index 2e12824049..36c9e029e9 100644
Zbigniew Jędrzejewski-Szmek 62fe94
--- a/src/core/swap.c
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/src/core/swap.c
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -610,7 +610,7 @@ static void swap_dump(Unit *u, FILE *f, const char *prefix) {
Zbigniew Jędrzejewski-Szmek 62fe94
                         prefix, p->priority,
Zbigniew Jędrzejewski-Szmek 62fe94
                         prefix, yes_no(p->noauto),
Zbigniew Jędrzejewski-Szmek 62fe94
                         prefix, yes_no(p->nofail),
Zbigniew Jędrzejewski-Szmek 62fe94
-                        prefix, p->discard);
Zbigniew Jędrzejewski-Szmek 62fe94
+                        prefix, p->discard ?: "none");
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
         if (s->control_pid > 0)
Zbigniew Jędrzejewski-Szmek 62fe94
                 fprintf(f,
Zbigniew Jędrzejewski-Szmek 62fe94
diff --git a/src/fstab-generator/fstab-generator.c b/src/fstab-generator/fstab-generator.c
Zbigniew Jędrzejewski-Szmek 62fe94
index 5569325a16..5dafcba3c0 100644
Zbigniew Jędrzejewski-Szmek 62fe94
--- a/src/fstab-generator/fstab-generator.c
Zbigniew Jędrzejewski-Szmek 62fe94
+++ b/src/fstab-generator/fstab-generator.c
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -46,34 +46,70 @@ static int arg_root_rw = -1;
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
 static int mount_find_pri(struct mntent *me, int *ret) {
Zbigniew Jędrzejewski-Szmek 62fe94
-        char *end, *pri;
Zbigniew Jędrzejewski-Szmek 62fe94
+        char *end, *opt;
Zbigniew Jędrzejewski-Szmek 62fe94
         unsigned long r;
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
         assert(me);
Zbigniew Jędrzejewski-Szmek 62fe94
         assert(ret);
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
-        pri = hasmntopt(me, "pri");
Zbigniew Jędrzejewski-Szmek 62fe94
-        if (!pri)
Zbigniew Jędrzejewski-Szmek 62fe94
+        opt = hasmntopt(me, "pri");
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!opt)
Zbigniew Jędrzejewski-Szmek 62fe94
                 return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
-        pri += 4;
Zbigniew Jędrzejewski-Szmek 62fe94
+        opt += strlen("pri");
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (*opt != '=')
Zbigniew Jędrzejewski-Szmek 62fe94
+                return -EINVAL;
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
         errno = 0;
Zbigniew Jędrzejewski-Szmek 62fe94
-        r = strtoul(pri, &end, 10);
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = strtoul(opt + 1, &end, 10);
Zbigniew Jędrzejewski-Szmek 62fe94
         if (errno > 0)
Zbigniew Jędrzejewski-Szmek 62fe94
                 return -errno;
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
-        if (end == pri || (*end != ',' && *end != 0))
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (end == opt + 1 || (*end != ',' && *end != 0))
Zbigniew Jędrzejewski-Szmek 62fe94
                 return -EINVAL;
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
         *ret = (int) r;
Zbigniew Jędrzejewski-Szmek 62fe94
         return 1;
Zbigniew Jędrzejewski-Szmek 62fe94
 }
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
+static int mount_find_discard(struct mntent *me, char **ret) {
Zbigniew Jędrzejewski-Szmek 62fe94
+        char *opt, *ans;
Zbigniew Jędrzejewski-Szmek 62fe94
+        size_t len;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(me);
Zbigniew Jędrzejewski-Szmek 62fe94
+        assert(ret);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        opt = hasmntopt(me, "discard");
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!opt)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return 0;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        opt += strlen("discard");
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (*opt == ',' || *opt == '\0')
Zbigniew Jędrzejewski-Szmek 62fe94
+                ans = strdup("all");
Zbigniew Jędrzejewski-Szmek 62fe94
+        else {
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (*opt != '=')
Zbigniew Jędrzejewski-Szmek 62fe94
+                        return -EINVAL;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                len = strcspn(opt + 1, ",");
Zbigniew Jędrzejewski-Szmek 62fe94
+                if (len == 0)
Zbigniew Jędrzejewski-Szmek 62fe94
+                        return -EINVAL;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+                ans = strndup(opt + 1, len);
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (!ans)
Zbigniew Jędrzejewski-Szmek 62fe94
+                return -ENOMEM;
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        *ret = ans;
Zbigniew Jędrzejewski-Szmek 62fe94
+        return 1;
Zbigniew Jędrzejewski-Szmek 62fe94
+}
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
 static int add_swap(const char *what, struct mntent *me) {
Zbigniew Jędrzejewski-Szmek 62fe94
         _cleanup_free_ char *name = NULL, *unit = NULL, *lnk = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
         _cleanup_fclose_ FILE *f = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
-        char *discard = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
+        _cleanup_free_ char *discard = NULL;
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
         bool noauto;
Zbigniew Jędrzejewski-Szmek 62fe94
         int r, pri = -1;
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -89,7 +125,13 @@ static int add_swap(const char *what, struct mntent *me) {
Zbigniew Jędrzejewski-Szmek 62fe94
         r = mount_find_pri(me, &pri);
Zbigniew Jędrzejewski-Szmek 62fe94
         if (r < 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
                 log_error("Failed to parse priority");
Zbigniew Jędrzejewski-Szmek 62fe94
-                return pri;
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
+        }
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        r = mount_find_discard(me, &discard);
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (r < 0) {
Zbigniew Jędrzejewski-Szmek 62fe94
+                log_error("Failed to parse discard");
Zbigniew Jędrzejewski-Szmek 62fe94
+                return r;
Zbigniew Jędrzejewski-Szmek 62fe94
         }
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
         noauto = !!hasmntopt(me, "noauto");
Zbigniew Jędrzejewski-Szmek 62fe94
@@ -120,18 +162,11 @@ static int add_swap(const char *what, struct mntent *me) {
Zbigniew Jędrzejewski-Szmek 62fe94
                 "What=%s\n",
Zbigniew Jędrzejewski-Szmek 62fe94
                 what);
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
-        discard = mount_test_option(me->mnt_opts, "discard");
Zbigniew Jędrzejewski-Szmek 62fe94
-        if (discard) {
Zbigniew Jędrzejewski-Szmek 62fe94
-                discard = strpbrk(discard, "=");
Zbigniew Jędrzejewski-Szmek 62fe94
-                fprintf(f,
Zbigniew Jędrzejewski-Szmek 62fe94
-                        "Discard=%s\n",
Zbigniew Jędrzejewski-Szmek 62fe94
-                        discard ? discard+1 : "all");
Zbigniew Jędrzejewski-Szmek 62fe94
-        }
Zbigniew Jędrzejewski-Szmek 62fe94
-
Zbigniew Jędrzejewski-Szmek 62fe94
         if (pri >= 0)
Zbigniew Jędrzejewski-Szmek 62fe94
-                fprintf(f,
Zbigniew Jędrzejewski-Szmek 62fe94
-                        "Priority=%i\n",
Zbigniew Jędrzejewski-Szmek 62fe94
-                        pri);
Zbigniew Jędrzejewski-Szmek 62fe94
+                fprintf(f, "Priority=%i\n", pri);
Zbigniew Jędrzejewski-Szmek 62fe94
+
Zbigniew Jędrzejewski-Szmek 62fe94
+        if (discard)
Zbigniew Jędrzejewski-Szmek 62fe94
+                fprintf(f, "Discard=%s\n", discard);
Zbigniew Jędrzejewski-Szmek 62fe94
 
Zbigniew Jędrzejewski-Szmek 62fe94
         fflush(f);
Zbigniew Jędrzejewski-Szmek 62fe94
         if (ferror(f)) {