// SPDX-License-Identifier: GPL-2.0-or-later /* Filesystem parameter parser. * * Copyright (C) 2018 Red Hat, Inc. All Rights Reserved. * Written by David Howells (dhowells@redhat.com)
*/
/** * lookup_constant - Look up a constant by name in an ordered table * @tbl: The table of constants to search. * @name: The name to look up. * @not_found: The value to return if the name is not found.
*/ int lookup_constant(conststruct constant_table *tbl, constchar *name, int not_found)
{ conststruct constant_table *p = __lookup_constant(tbl, name);
return p ? p->value : not_found;
}
EXPORT_SYMBOL(lookup_constant);
*negated = false; for (p = desc; p->name; p++) { if (strcmp(p->name, name) != 0) continue; if (likely(is_flag(p) == want_flag)) return p;
other = p;
} if (want_flag) { if (name[0] == 'n' && name[1] == 'o' && name[2]) { for (p = desc; p->name; p++) { if (strcmp(p->name, name + 2) != 0) continue; if (!(p->flags & fs_param_neg_with_no)) continue;
*negated = true; return p;
}
}
} return other;
}
/* * __fs_parse - Parse a filesystem configuration parameter * @log: The filesystem context to log errors through. * @desc: The parameter description to use. * @param: The parameter. * @result: Where to place the result of the parse * * Parse a filesystem configuration parameter and attempt a conversion for a * simple parameter for which this is requested. If successful, the determined * parameter ID is placed into @result->key, the desired type is indicated in * @result->t and any converted value is placed into an appropriate member of * the union in @result. * * The function returns the parameter number if the parameter was matched, * -ENOPARAM if it wasn't matched and @desc->ignore_unknown indicated that * unknown parameters are okay and -EINVAL if there was a conversion issue or * the parameter wasn't recognised and unknowns aren't okay.
*/ int __fs_parse(struct p_log *log, conststruct fs_parameter_spec *desc, struct fs_parameter *param, struct fs_parse_result *result)
{ conststruct fs_parameter_spec *p;
result->uint_64 = 0;
p = fs_lookup_key(desc, param, &result->negated); if (!p) return -ENOPARAM;
if (p->flags & fs_param_deprecated)
warn_plog(log, "Deprecated parameter '%s'", param->key);
/* Try to turn the type we were given into the type desired by the * parameter and give an error if we can't.
*/ if (is_flag(p)) { if (param->type != fs_value_is_flag) return inval_plog(log, "Unexpected value for '%s'",
param->key);
result->boolean = !result->negated;
} else { int ret = p->type(log, p, param, result); if (ret) return ret;
} return p->opt;
}
EXPORT_SYMBOL(__fs_parse);
/** * fs_lookup_param - Look up a path referred to by a parameter * @fc: The filesystem context to log errors through. * @param: The parameter. * @want_bdev: T if want a blockdev * @flags: Pathwalk flags passed to filename_lookup() * @_path: The result of the lookup
*/ int fs_lookup_param(struct fs_context *fc, struct fs_parameter *param, bool want_bdev, unsignedint flags, struct path *_path)
{ struct filename *f; bool put_f; int ret;
switch (param->type) { case fs_value_is_string:
f = getname_kernel(param->string); if (IS_ERR(f)) return PTR_ERR(f);
param->dirfd = AT_FDCWD;
put_f = true; break; case fs_value_is_filename:
f = param->name;
put_f = false; break; default: return invalf(fc, "%s: not usable as path", param->key);
}
ret = filename_lookup(param->dirfd, f, flags, _path, NULL); if (ret < 0) {
errorf(fc, "%s: Lookup failure for '%s'", param->key, f->name); goto out;
}
if (want_bdev &&
!S_ISBLK(d_backing_inode(_path->dentry)->i_mode)) {
path_put(_path);
_path->dentry = NULL;
_path->mnt = NULL;
errorf(fc, "%s: Non-blockdev passed as '%s'",
param->key, f->name);
ret = -ENOTBLK;
}
out: if (put_f)
putname(f); return ret;
}
EXPORT_SYMBOL(fs_lookup_param);
staticint fs_param_bad_value(struct p_log *log, struct fs_parameter *param)
{ return inval_plog(log, "Bad value for '%s'", param->key);
}
int fs_param_is_bool(struct p_log *log, conststruct fs_parameter_spec *p, struct fs_parameter *param, struct fs_parse_result *result)
{ int b; if (param->type != fs_value_is_string) return fs_param_bad_value(log, param); if (!*param->string && (p->flags & fs_param_can_be_empty)) return 0;
b = lookup_constant(bool_names, param->string, -1); if (b == -1) return fs_param_bad_value(log, param);
result->boolean = b; return 0;
}
EXPORT_SYMBOL(fs_param_is_bool);
int fs_param_is_u32(struct p_log *log, conststruct fs_parameter_spec *p, struct fs_parameter *param, struct fs_parse_result *result)
{ int base = (unsignedlong)p->data; if (param->type != fs_value_is_string) return fs_param_bad_value(log, param); if (!*param->string && (p->flags & fs_param_can_be_empty)) return 0; if (kstrtouint(param->string, base, &result->uint_32) < 0) return fs_param_bad_value(log, param); return 0;
}
EXPORT_SYMBOL(fs_param_is_u32);
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.