2017-11-18 17:09:20 +01:00
|
|
|
/* SPDX-License-Identifier: LGPL-2.1+ */
|
2015-01-09 22:58:29 +01:00
|
|
|
|
2015-12-03 21:13:37 +01:00
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
2015-10-27 03:01:06 +01:00
|
|
|
#include "alloc-util.h"
|
2015-10-26 18:43:29 +01:00
|
|
|
#include "device-nodes.h"
|
2015-10-26 16:18:16 +01:00
|
|
|
#include "fstab-util.h"
|
2015-12-03 21:13:37 +01:00
|
|
|
#include "macro.h"
|
2015-10-26 18:44:13 +01:00
|
|
|
#include "mount-util.h"
|
2019-03-14 13:14:33 +01:00
|
|
|
#include "nulstr-util.h"
|
2015-10-26 16:18:16 +01:00
|
|
|
#include "parse-util.h"
|
2015-09-30 22:24:52 +02:00
|
|
|
#include "path-util.h"
|
2015-10-24 22:58:24 +02:00
|
|
|
#include "string-util.h"
|
2015-01-09 22:58:29 +01:00
|
|
|
#include "strv.h"
|
|
|
|
|
2017-06-26 15:22:10 +02:00
|
|
|
int fstab_has_fstype(const char *fstype) {
|
|
|
|
_cleanup_endmntent_ FILE *f = NULL;
|
|
|
|
struct mntent *m;
|
|
|
|
|
2019-11-13 17:36:46 +01:00
|
|
|
f = setmntent(fstab_path(), "re");
|
2017-06-26 15:22:10 +02:00
|
|
|
if (!f)
|
|
|
|
return errno == ENOENT ? false : -errno;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
errno = 0;
|
|
|
|
m = getmntent(f);
|
|
|
|
if (!m)
|
|
|
|
return errno != 0 ? -errno : false;
|
|
|
|
|
|
|
|
if (streq(m->mnt_type, fstype))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-06-27 09:52:12 +02:00
|
|
|
int fstab_is_mount_point(const char *mount) {
|
2015-09-30 22:24:52 +02:00
|
|
|
_cleanup_endmntent_ FILE *f = NULL;
|
|
|
|
struct mntent *m;
|
|
|
|
|
2019-11-13 17:36:46 +01:00
|
|
|
f = setmntent(fstab_path(), "re");
|
2015-09-30 22:24:52 +02:00
|
|
|
if (!f)
|
2017-06-27 09:52:12 +02:00
|
|
|
return errno == ENOENT ? false : -errno;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
errno = 0;
|
|
|
|
m = getmntent(f);
|
|
|
|
if (!m)
|
|
|
|
return errno != 0 ? -errno : false;
|
2015-09-30 22:24:52 +02:00
|
|
|
|
|
|
|
if (path_equal(m->mnt_dir, mount))
|
|
|
|
return true;
|
2017-06-27 09:52:12 +02:00
|
|
|
}
|
2015-09-30 22:24:52 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-01-09 22:58:29 +01:00
|
|
|
int fstab_filter_options(const char *opts, const char *names,
|
|
|
|
const char **namefound, char **value, char **filtered) {
|
|
|
|
const char *name, *n = NULL, *x;
|
|
|
|
_cleanup_strv_free_ char **stor = NULL;
|
|
|
|
_cleanup_free_ char *v = NULL, **strv = NULL;
|
|
|
|
|
|
|
|
assert(names && *names);
|
|
|
|
|
|
|
|
if (!opts)
|
|
|
|
goto answer;
|
|
|
|
|
|
|
|
/* If !value and !filtered, this function is not allowed to fail. */
|
|
|
|
|
|
|
|
if (!filtered) {
|
|
|
|
const char *word, *state;
|
|
|
|
size_t l;
|
|
|
|
|
|
|
|
FOREACH_WORD_SEPARATOR(word, l, opts, ",", state)
|
|
|
|
NULSTR_FOREACH(name, names) {
|
|
|
|
if (l < strlen(name))
|
|
|
|
continue;
|
|
|
|
if (!strneq(word, name, strlen(name)))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* we know that the string is NUL
|
|
|
|
* terminated, so *x is valid */
|
|
|
|
x = word + strlen(name);
|
|
|
|
if (IN_SET(*x, '\0', '=', ',')) {
|
|
|
|
n = name;
|
|
|
|
if (value) {
|
|
|
|
free(v);
|
|
|
|
if (IN_SET(*x, '\0', ','))
|
|
|
|
v = NULL;
|
|
|
|
else {
|
|
|
|
assert(*x == '=');
|
|
|
|
x++;
|
|
|
|
v = strndup(x, l - strlen(name) - 1);
|
|
|
|
if (!v)
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
char **t, **s;
|
|
|
|
|
|
|
|
stor = strv_split(opts, ",");
|
|
|
|
if (!stor)
|
|
|
|
return -ENOMEM;
|
|
|
|
strv = memdup(stor, sizeof(char*) * (strv_length(stor) + 1));
|
|
|
|
if (!strv)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
for (s = t = strv; *s; s++) {
|
|
|
|
NULSTR_FOREACH(name, names) {
|
|
|
|
x = startswith(*s, name);
|
|
|
|
if (x && IN_SET(*x, '\0', '='))
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
|
|
|
|
*t = *s;
|
|
|
|
t++;
|
|
|
|
continue;
|
|
|
|
found:
|
2019-04-27 02:22:40 +02:00
|
|
|
/* Keep the last occurrence found */
|
2015-01-09 22:58:29 +01:00
|
|
|
n = name;
|
|
|
|
if (value) {
|
|
|
|
free(v);
|
|
|
|
if (*x == '\0')
|
|
|
|
v = NULL;
|
|
|
|
else {
|
|
|
|
assert(*x == '=');
|
|
|
|
x++;
|
|
|
|
v = strdup(x);
|
|
|
|
if (!v)
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*t = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
answer:
|
|
|
|
if (namefound)
|
|
|
|
*namefound = n;
|
|
|
|
if (filtered) {
|
|
|
|
char *f;
|
|
|
|
|
|
|
|
f = strv_join(strv, ",");
|
|
|
|
if (!f)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
*filtered = f;
|
|
|
|
}
|
2018-03-22 16:53:26 +01:00
|
|
|
if (value)
|
|
|
|
*value = TAKE_PTR(v);
|
2015-01-09 22:58:29 +01:00
|
|
|
|
|
|
|
return !!n;
|
|
|
|
}
|
|
|
|
|
2015-05-18 12:30:37 +02:00
|
|
|
int fstab_extract_values(const char *opts, const char *name, char ***values) {
|
|
|
|
_cleanup_strv_free_ char **optsv = NULL, **res = NULL;
|
|
|
|
char **s;
|
|
|
|
|
|
|
|
assert(opts);
|
|
|
|
assert(name);
|
|
|
|
assert(values);
|
|
|
|
|
|
|
|
optsv = strv_split(opts, ",");
|
|
|
|
if (!optsv)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
STRV_FOREACH(s, optsv) {
|
|
|
|
char *arg;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
arg = startswith(*s, name);
|
|
|
|
if (!arg || *arg != '=')
|
|
|
|
continue;
|
|
|
|
r = strv_extend(&res, arg + 1);
|
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2018-03-22 16:53:26 +01:00
|
|
|
*values = TAKE_PTR(res);
|
2015-05-18 12:30:37 +02:00
|
|
|
|
|
|
|
return !!*values;
|
|
|
|
}
|
|
|
|
|
2015-01-09 22:58:29 +01:00
|
|
|
int fstab_find_pri(const char *options, int *ret) {
|
|
|
|
_cleanup_free_ char *opt = NULL;
|
2019-12-03 19:36:37 +01:00
|
|
|
int r, pri;
|
2015-01-09 22:58:29 +01:00
|
|
|
|
|
|
|
assert(ret);
|
|
|
|
|
|
|
|
r = fstab_filter_options(options, "pri\0", NULL, &opt, NULL);
|
2015-01-12 05:40:46 +01:00
|
|
|
if (r < 0)
|
2015-01-09 22:58:29 +01:00
|
|
|
return r;
|
2015-01-12 05:40:46 +01:00
|
|
|
if (r == 0 || !opt)
|
|
|
|
return 0;
|
2015-01-09 22:58:29 +01:00
|
|
|
|
2019-12-03 19:36:37 +01:00
|
|
|
r = safe_atoi(opt, &pri);
|
2015-01-09 22:58:29 +01:00
|
|
|
if (r < 0)
|
|
|
|
return r;
|
|
|
|
|
2019-12-03 19:36:37 +01:00
|
|
|
*ret = pri;
|
2015-01-09 22:58:29 +01:00
|
|
|
return 1;
|
|
|
|
}
|
2015-10-26 18:43:29 +01:00
|
|
|
|
|
|
|
static char *unquote(const char *s, const char* quotes) {
|
|
|
|
size_t l;
|
|
|
|
assert(s);
|
|
|
|
|
|
|
|
/* This is rather stupid, simply removes the heading and
|
|
|
|
* trailing quotes if there is one. Doesn't care about
|
|
|
|
* escaping or anything.
|
|
|
|
*
|
2017-02-24 18:14:02 +01:00
|
|
|
* DON'T USE THIS FOR NEW CODE ANYMORE! */
|
2015-10-26 18:43:29 +01:00
|
|
|
|
|
|
|
l = strlen(s);
|
|
|
|
if (l < 2)
|
|
|
|
return strdup(s);
|
|
|
|
|
|
|
|
if (strchr(quotes, s[0]) && s[l-1] == s[0])
|
|
|
|
return strndup(s+1, l-2);
|
|
|
|
|
|
|
|
return strdup(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *tag_to_udev_node(const char *tagvalue, const char *by) {
|
|
|
|
_cleanup_free_ char *t = NULL, *u = NULL;
|
|
|
|
size_t enc_len;
|
|
|
|
|
|
|
|
u = unquote(tagvalue, QUOTES);
|
|
|
|
if (!u)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
enc_len = strlen(u) * 4 + 1;
|
|
|
|
t = new(char, enc_len);
|
|
|
|
if (!t)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (encode_devnode_name(u, t, enc_len) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
2016-10-23 17:43:27 +02:00
|
|
|
return strjoin("/dev/disk/by-", by, "/", t);
|
2015-10-26 18:43:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
char *fstab_node_to_udev_node(const char *p) {
|
|
|
|
assert(p);
|
|
|
|
|
|
|
|
if (startswith(p, "LABEL="))
|
|
|
|
return tag_to_udev_node(p+6, "label");
|
|
|
|
|
|
|
|
if (startswith(p, "UUID="))
|
|
|
|
return tag_to_udev_node(p+5, "uuid");
|
|
|
|
|
|
|
|
if (startswith(p, "PARTUUID="))
|
|
|
|
return tag_to_udev_node(p+9, "partuuid");
|
|
|
|
|
|
|
|
if (startswith(p, "PARTLABEL="))
|
|
|
|
return tag_to_udev_node(p+10, "partlabel");
|
|
|
|
|
|
|
|
return strdup(p);
|
|
|
|
}
|