mirror of
https://git.proxmox.com/git/mirror_zfs.git
synced 2025-01-03 23:09:35 +03:00
97143b9d31
`snprintf()` is meant to protect against buffer overflows, but operating on the buffer using its return value, possibly by calling it again, can cause a buffer overflow, because it will return how many characters it would have written if it had enough space even when it did not. In a number of places, we repeatedly call snprintf() by successively incrementing a buffer offset and decrementing a buffer length, by its return value. This is a potentially unsafe usage of `snprintf()` whenever the buffer length is reached. CodeQL complained about this. To fix this, we introduce `kmem_scnprintf()`, which will return 0 when the buffer is zero or the number of written characters, minus 1 to exclude the NULL character, when the buffer was too small. In all other cases, it behaves like snprintf(). The name is inspired by the Linux and XNU kernels' `scnprintf()`. The implementation was written before I thought to look at `scnprintf()` and had a good name for it, but it turned out to have identical semantics to the Linux kernel version. That lead to the name, `kmem_scnprintf()`. CodeQL only catches this issue in loops, so repeated use of snprintf() outside of a loop was not caught. As a result, a thorough audit of the codebase was done to examine all instances of `snprintf()` usage for potential problems and a few were caught. Fixes for them are included in this patch. Unfortunately, ZED is one of the places where `snprintf()` is potentially used incorrectly. Since using `kmem_scnprintf()` in it would require changing how it is linked, we modify its usage to make it safe, no matter what buffer length is used. In addition, there was a bug in the use of the return value where the NULL format character was not being written by pwrite(). That has been fixed. Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Signed-off-by: Richard Yao <richard.yao@alumni.stonybrook.edu> Closes #14098
702 lines
18 KiB
C
702 lines
18 KiB
C
/*
|
|
* CDDL HEADER START
|
|
*
|
|
* The contents of this file are subject to the terms of the
|
|
* Common Development and Distribution License (the "License").
|
|
* You may not use this file except in compliance with the License.
|
|
*
|
|
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
|
|
* or https://opensource.org/licenses/CDDL-1.0.
|
|
* See the License for the specific language governing permissions
|
|
* and limitations under the License.
|
|
*
|
|
* When distributing Covered Code, include this CDDL HEADER in each
|
|
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
|
|
* If applicable, add the following below this CDDL HEADER, with the
|
|
* fields enclosed by brackets "[]" replaced with your own identifying
|
|
* information: Portions Copyright [yyyy] [name of copyright owner]
|
|
*
|
|
* CDDL HEADER END
|
|
*/
|
|
/*
|
|
* Copyright (c) 2018, 2019 by Delphix. All rights reserved.
|
|
*/
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/param.h>
|
|
#include <sys/zfeature.h>
|
|
#include <sys/zfs_ioctl.h>
|
|
#include <sys/zfs_sysfs.h>
|
|
#include <sys/kmem.h>
|
|
#include <sys/fs/zfs.h>
|
|
#include <linux/kobject.h>
|
|
|
|
#include "zfs_prop.h"
|
|
|
|
#if !defined(_KERNEL)
|
|
#error kernel builds only
|
|
#endif
|
|
|
|
/*
|
|
* ZFS Module sysfs support
|
|
*
|
|
* This extends our sysfs '/sys/module/zfs' entry to include feature
|
|
* and property attributes. The primary consumer of this information
|
|
* is user processes, like the zfs CLI, that need to know what the
|
|
* current loaded ZFS module supports. The libzfs binary will consult
|
|
* this information when instantiating the zfs|zpool property tables
|
|
* and the pool features table.
|
|
*
|
|
* The added top-level directories are:
|
|
* /sys/module/zfs
|
|
* ├── features.kernel
|
|
* ├── features.pool
|
|
* ├── properties.dataset
|
|
* └── properties.pool
|
|
*
|
|
* The local interface for the zfs kobjects includes:
|
|
* zfs_kobj_init()
|
|
* zfs_kobj_add()
|
|
* zfs_kobj_release()
|
|
* zfs_kobj_add_attr()
|
|
* zfs_kobj_fini()
|
|
*/
|
|
|
|
/*
|
|
* A zfs_mod_kobj_t represents a zfs kobject under '/sys/module/zfs'
|
|
*/
|
|
typedef struct zfs_mod_kobj zfs_mod_kobj_t;
|
|
struct zfs_mod_kobj {
|
|
struct kobject zko_kobj;
|
|
struct kobj_type zko_kobj_type;
|
|
struct sysfs_ops zko_sysfs_ops;
|
|
size_t zko_attr_count;
|
|
struct attribute *zko_attr_list; /* allocated */
|
|
struct attribute_group zko_default_group; /* .attrs allocated */
|
|
const struct attribute_group *zko_default_groups[2];
|
|
size_t zko_child_count;
|
|
zfs_mod_kobj_t *zko_children; /* allocated */
|
|
};
|
|
|
|
#define ATTR_TABLE_SIZE(cnt) (sizeof (struct attribute) * (cnt))
|
|
/* Note +1 for NULL terminator slot */
|
|
#define DEFAULT_ATTR_SIZE(cnt) (sizeof (struct attribute *) * (cnt + 1))
|
|
#define CHILD_TABLE_SIZE(cnt) (sizeof (zfs_mod_kobj_t) * (cnt))
|
|
|
|
/*
|
|
* These are the top-level kobjects under '/sys/module/zfs/'
|
|
*/
|
|
static zfs_mod_kobj_t kernel_features_kobj;
|
|
static zfs_mod_kobj_t pool_features_kobj;
|
|
static zfs_mod_kobj_t dataset_props_kobj;
|
|
static zfs_mod_kobj_t vdev_props_kobj;
|
|
static zfs_mod_kobj_t pool_props_kobj;
|
|
|
|
/*
|
|
* The show function is used to provide the content
|
|
* of an attribute into a PAGE_SIZE buffer.
|
|
*/
|
|
typedef ssize_t (*sysfs_show_func)(struct kobject *, struct attribute *,
|
|
char *);
|
|
|
|
static void
|
|
zfs_kobj_fini(zfs_mod_kobj_t *zkobj)
|
|
{
|
|
/* finalize any child kobjects */
|
|
if (zkobj->zko_child_count != 0) {
|
|
ASSERT(zkobj->zko_children);
|
|
for (int i = 0; i < zkobj->zko_child_count; i++)
|
|
zfs_kobj_fini(&zkobj->zko_children[i]);
|
|
}
|
|
|
|
/* kobject_put() will call zfs_kobj_release() to release memory */
|
|
kobject_del(&zkobj->zko_kobj);
|
|
kobject_put(&zkobj->zko_kobj);
|
|
}
|
|
|
|
static void
|
|
zfs_kobj_release(struct kobject *kobj)
|
|
{
|
|
zfs_mod_kobj_t *zkobj = container_of(kobj, zfs_mod_kobj_t, zko_kobj);
|
|
|
|
if (zkobj->zko_attr_list != NULL) {
|
|
ASSERT3S(zkobj->zko_attr_count, !=, 0);
|
|
kmem_free(zkobj->zko_attr_list,
|
|
ATTR_TABLE_SIZE(zkobj->zko_attr_count));
|
|
zkobj->zko_attr_list = NULL;
|
|
}
|
|
|
|
if (zkobj->zko_default_group.attrs != NULL) {
|
|
kmem_free(zkobj->zko_default_group.attrs,
|
|
DEFAULT_ATTR_SIZE(zkobj->zko_attr_count));
|
|
zkobj->zko_default_group.attrs = NULL;
|
|
}
|
|
|
|
if (zkobj->zko_child_count != 0) {
|
|
ASSERT(zkobj->zko_children);
|
|
|
|
kmem_free(zkobj->zko_children,
|
|
CHILD_TABLE_SIZE(zkobj->zko_child_count));
|
|
zkobj->zko_child_count = 0;
|
|
zkobj->zko_children = NULL;
|
|
}
|
|
|
|
zkobj->zko_attr_count = 0;
|
|
}
|
|
|
|
#ifndef sysfs_attr_init
|
|
#define sysfs_attr_init(attr) do {} while (0)
|
|
#endif
|
|
|
|
static void
|
|
zfs_kobj_add_attr(zfs_mod_kobj_t *zkobj, int attr_num, const char *attr_name)
|
|
{
|
|
VERIFY3U(attr_num, <, zkobj->zko_attr_count);
|
|
ASSERT(zkobj->zko_attr_list);
|
|
ASSERT(zkobj->zko_default_group.attrs);
|
|
|
|
zkobj->zko_attr_list[attr_num].name = attr_name;
|
|
zkobj->zko_attr_list[attr_num].mode = 0444;
|
|
zkobj->zko_default_group.attrs[attr_num] =
|
|
&zkobj->zko_attr_list[attr_num];
|
|
sysfs_attr_init(&zkobj->zko_attr_list[attr_num]);
|
|
}
|
|
|
|
static int
|
|
zfs_kobj_init(zfs_mod_kobj_t *zkobj, int attr_cnt, int child_cnt,
|
|
sysfs_show_func show_func)
|
|
{
|
|
/*
|
|
* Initialize object's attributes. Count can be zero.
|
|
*/
|
|
if (attr_cnt > 0) {
|
|
zkobj->zko_attr_list = kmem_zalloc(ATTR_TABLE_SIZE(attr_cnt),
|
|
KM_SLEEP);
|
|
if (zkobj->zko_attr_list == NULL)
|
|
return (ENOMEM);
|
|
}
|
|
/* this will always have at least one slot for NULL termination */
|
|
zkobj->zko_default_group.attrs =
|
|
kmem_zalloc(DEFAULT_ATTR_SIZE(attr_cnt), KM_SLEEP);
|
|
if (zkobj->zko_default_group.attrs == NULL) {
|
|
if (zkobj->zko_attr_list != NULL) {
|
|
kmem_free(zkobj->zko_attr_list,
|
|
ATTR_TABLE_SIZE(attr_cnt));
|
|
}
|
|
return (ENOMEM);
|
|
}
|
|
zkobj->zko_attr_count = attr_cnt;
|
|
zkobj->zko_default_groups[0] = &zkobj->zko_default_group;
|
|
#ifdef HAVE_SYSFS_DEFAULT_GROUPS
|
|
zkobj->zko_kobj_type.default_groups = zkobj->zko_default_groups;
|
|
#else
|
|
zkobj->zko_kobj_type.default_attrs = zkobj->zko_default_group.attrs;
|
|
#endif
|
|
|
|
if (child_cnt > 0) {
|
|
zkobj->zko_children = kmem_zalloc(CHILD_TABLE_SIZE(child_cnt),
|
|
KM_SLEEP);
|
|
if (zkobj->zko_children == NULL) {
|
|
if (zkobj->zko_default_group.attrs != NULL) {
|
|
kmem_free(zkobj->zko_default_group.attrs,
|
|
DEFAULT_ATTR_SIZE(attr_cnt));
|
|
}
|
|
if (zkobj->zko_attr_list != NULL) {
|
|
kmem_free(zkobj->zko_attr_list,
|
|
ATTR_TABLE_SIZE(attr_cnt));
|
|
}
|
|
return (ENOMEM);
|
|
}
|
|
zkobj->zko_child_count = child_cnt;
|
|
}
|
|
|
|
zkobj->zko_sysfs_ops.show = show_func;
|
|
zkobj->zko_kobj_type.sysfs_ops = &zkobj->zko_sysfs_ops;
|
|
zkobj->zko_kobj_type.release = zfs_kobj_release;
|
|
|
|
return (0);
|
|
}
|
|
|
|
static int
|
|
zfs_kobj_add(zfs_mod_kobj_t *zkobj, struct kobject *parent, const char *name)
|
|
{
|
|
/* zko_default_group.attrs must be NULL terminated */
|
|
ASSERT(zkobj->zko_default_group.attrs != NULL);
|
|
ASSERT(zkobj->zko_default_group.attrs[zkobj->zko_attr_count] == NULL);
|
|
|
|
kobject_init(&zkobj->zko_kobj, &zkobj->zko_kobj_type);
|
|
return (kobject_add(&zkobj->zko_kobj, parent, name));
|
|
}
|
|
|
|
/*
|
|
* Each zfs property has these common attributes
|
|
*/
|
|
static const char *const zprop_attrs[] = {
|
|
"type",
|
|
"readonly",
|
|
"setonce",
|
|
"visible",
|
|
"values",
|
|
"default",
|
|
"datasets" /* zfs properties only */
|
|
};
|
|
|
|
#define ZFS_PROP_ATTR_COUNT ARRAY_SIZE(zprop_attrs)
|
|
#define ZPOOL_PROP_ATTR_COUNT (ZFS_PROP_ATTR_COUNT - 1)
|
|
|
|
static const char *const zprop_types[] = {
|
|
"number",
|
|
"string",
|
|
"index",
|
|
};
|
|
|
|
typedef struct zfs_type_map {
|
|
zfs_type_t ztm_type;
|
|
const char *ztm_name;
|
|
} zfs_type_map_t;
|
|
|
|
static const zfs_type_map_t type_map[] = {
|
|
{ZFS_TYPE_FILESYSTEM, "filesystem"},
|
|
{ZFS_TYPE_SNAPSHOT, "snapshot"},
|
|
{ZFS_TYPE_VOLUME, "volume"},
|
|
{ZFS_TYPE_BOOKMARK, "bookmark"}
|
|
};
|
|
|
|
/*
|
|
* Show the content for a zfs property attribute
|
|
*/
|
|
static ssize_t
|
|
zprop_sysfs_show(const char *attr_name, const zprop_desc_t *property,
|
|
char *buf, size_t buflen)
|
|
{
|
|
const char *show_str;
|
|
char number[32];
|
|
|
|
/* For dataset properties list the dataset types that apply */
|
|
if (strcmp(attr_name, "datasets") == 0 &&
|
|
property->pd_types != ZFS_TYPE_POOL) {
|
|
int len = 0;
|
|
|
|
for (int i = 0; i < ARRAY_SIZE(type_map); i++) {
|
|
if (type_map[i].ztm_type & property->pd_types) {
|
|
len += kmem_scnprintf(buf + len, buflen - len,
|
|
"%s ", type_map[i].ztm_name);
|
|
}
|
|
}
|
|
len += kmem_scnprintf(buf + len, buflen - len, "\n");
|
|
return (len);
|
|
}
|
|
|
|
if (strcmp(attr_name, "type") == 0) {
|
|
show_str = zprop_types[property->pd_proptype];
|
|
} else if (strcmp(attr_name, "readonly") == 0) {
|
|
show_str = property->pd_attr == PROP_READONLY ? "1" : "0";
|
|
} else if (strcmp(attr_name, "setonce") == 0) {
|
|
show_str = property->pd_attr == PROP_ONETIME ? "1" : "0";
|
|
} else if (strcmp(attr_name, "visible") == 0) {
|
|
show_str = property->pd_visible ? "1" : "0";
|
|
} else if (strcmp(attr_name, "values") == 0) {
|
|
show_str = property->pd_values ? property->pd_values : "";
|
|
} else if (strcmp(attr_name, "default") == 0) {
|
|
switch (property->pd_proptype) {
|
|
case PROP_TYPE_NUMBER:
|
|
(void) snprintf(number, sizeof (number), "%llu",
|
|
(u_longlong_t)property->pd_numdefault);
|
|
show_str = number;
|
|
break;
|
|
case PROP_TYPE_STRING:
|
|
show_str = property->pd_strdefault ?
|
|
property->pd_strdefault : "";
|
|
break;
|
|
case PROP_TYPE_INDEX:
|
|
if (zprop_index_to_string(property->pd_propnum,
|
|
property->pd_numdefault, &show_str,
|
|
property->pd_types) != 0) {
|
|
show_str = "";
|
|
}
|
|
break;
|
|
default:
|
|
return (0);
|
|
}
|
|
} else {
|
|
return (0);
|
|
}
|
|
|
|
return (snprintf(buf, buflen, "%s\n", show_str));
|
|
}
|
|
|
|
static ssize_t
|
|
dataset_property_show(struct kobject *kobj, struct attribute *attr, char *buf)
|
|
{
|
|
zfs_prop_t prop = zfs_name_to_prop(kobject_name(kobj));
|
|
zprop_desc_t *prop_tbl = zfs_prop_get_table();
|
|
ssize_t len;
|
|
|
|
ASSERT3U(prop, <, ZFS_NUM_PROPS);
|
|
|
|
len = zprop_sysfs_show(attr->name, &prop_tbl[prop], buf, PAGE_SIZE);
|
|
|
|
return (len);
|
|
}
|
|
|
|
static ssize_t
|
|
vdev_property_show(struct kobject *kobj, struct attribute *attr, char *buf)
|
|
{
|
|
vdev_prop_t prop = vdev_name_to_prop(kobject_name(kobj));
|
|
zprop_desc_t *prop_tbl = vdev_prop_get_table();
|
|
ssize_t len;
|
|
|
|
ASSERT3U(prop, <, VDEV_NUM_PROPS);
|
|
|
|
len = zprop_sysfs_show(attr->name, &prop_tbl[prop], buf, PAGE_SIZE);
|
|
|
|
return (len);
|
|
}
|
|
|
|
static ssize_t
|
|
pool_property_show(struct kobject *kobj, struct attribute *attr, char *buf)
|
|
{
|
|
zpool_prop_t prop = zpool_name_to_prop(kobject_name(kobj));
|
|
zprop_desc_t *prop_tbl = zpool_prop_get_table();
|
|
ssize_t len;
|
|
|
|
ASSERT3U(prop, <, ZPOOL_NUM_PROPS);
|
|
|
|
len = zprop_sysfs_show(attr->name, &prop_tbl[prop], buf, PAGE_SIZE);
|
|
|
|
return (len);
|
|
}
|
|
|
|
/*
|
|
* ZFS kernel feature attributes for '/sys/module/zfs/features.kernel'
|
|
*
|
|
* This list is intended for kernel features that don't have a pool feature
|
|
* association or that extend existing user kernel interfaces.
|
|
*
|
|
* A user process can easily check if the running zfs kernel module
|
|
* supports the new feature.
|
|
*/
|
|
static const char *const zfs_kernel_features[] = {
|
|
/* --> Add new kernel features here */
|
|
"com.delphix:vdev_initialize",
|
|
"org.zfsonlinux:vdev_trim",
|
|
"org.openzfs:l2arc_persistent",
|
|
};
|
|
|
|
#define KERNEL_FEATURE_COUNT ARRAY_SIZE(zfs_kernel_features)
|
|
|
|
static ssize_t
|
|
kernel_feature_show(struct kobject *kobj, struct attribute *attr, char *buf)
|
|
{
|
|
if (strcmp(attr->name, "supported") == 0)
|
|
return (snprintf(buf, PAGE_SIZE, "yes\n"));
|
|
return (0);
|
|
}
|
|
|
|
static void
|
|
kernel_feature_to_kobj(zfs_mod_kobj_t *parent, int slot, const char *name)
|
|
{
|
|
zfs_mod_kobj_t *zfs_kobj = &parent->zko_children[slot];
|
|
|
|
ASSERT3U(slot, <, KERNEL_FEATURE_COUNT);
|
|
ASSERT(name);
|
|
|
|
int err = zfs_kobj_init(zfs_kobj, 1, 0, kernel_feature_show);
|
|
if (err)
|
|
return;
|
|
|
|
zfs_kobj_add_attr(zfs_kobj, 0, "supported");
|
|
|
|
err = zfs_kobj_add(zfs_kobj, &parent->zko_kobj, name);
|
|
if (err)
|
|
zfs_kobj_release(&zfs_kobj->zko_kobj);
|
|
}
|
|
|
|
static int
|
|
zfs_kernel_features_init(zfs_mod_kobj_t *zfs_kobj, struct kobject *parent)
|
|
{
|
|
/*
|
|
* Create a parent kobject to host kernel features.
|
|
*
|
|
* '/sys/module/zfs/features.kernel'
|
|
*/
|
|
int err = zfs_kobj_init(zfs_kobj, 0, KERNEL_FEATURE_COUNT,
|
|
kernel_feature_show);
|
|
if (err)
|
|
return (err);
|
|
err = zfs_kobj_add(zfs_kobj, parent, ZFS_SYSFS_KERNEL_FEATURES);
|
|
if (err) {
|
|
zfs_kobj_release(&zfs_kobj->zko_kobj);
|
|
return (err);
|
|
}
|
|
|
|
/*
|
|
* Now create a kobject for each feature.
|
|
*
|
|
* '/sys/module/zfs/features.kernel/<feature>'
|
|
*/
|
|
for (int f = 0; f < KERNEL_FEATURE_COUNT; f++)
|
|
kernel_feature_to_kobj(zfs_kobj, f, zfs_kernel_features[f]);
|
|
|
|
return (0);
|
|
}
|
|
|
|
/*
|
|
* Each pool feature has these common attributes
|
|
*/
|
|
static const char *const pool_feature_attrs[] = {
|
|
"description",
|
|
"guid",
|
|
"uname",
|
|
"readonly_compatible",
|
|
"required_for_mos",
|
|
"activate_on_enable",
|
|
"per_dataset"
|
|
};
|
|
|
|
#define ZPOOL_FEATURE_ATTR_COUNT ARRAY_SIZE(pool_feature_attrs)
|
|
|
|
/*
|
|
* Show the content for the given zfs pool feature attribute
|
|
*/
|
|
static ssize_t
|
|
pool_feature_show(struct kobject *kobj, struct attribute *attr, char *buf)
|
|
{
|
|
spa_feature_t fid;
|
|
|
|
if (zfeature_lookup_guid(kobject_name(kobj), &fid) != 0)
|
|
return (0);
|
|
|
|
ASSERT3U(fid, <, SPA_FEATURES);
|
|
|
|
zfeature_flags_t flags = spa_feature_table[fid].fi_flags;
|
|
const char *show_str = NULL;
|
|
|
|
if (strcmp(attr->name, "description") == 0) {
|
|
show_str = spa_feature_table[fid].fi_desc;
|
|
} else if (strcmp(attr->name, "guid") == 0) {
|
|
show_str = spa_feature_table[fid].fi_guid;
|
|
} else if (strcmp(attr->name, "uname") == 0) {
|
|
show_str = spa_feature_table[fid].fi_uname;
|
|
} else if (strcmp(attr->name, "readonly_compatible") == 0) {
|
|
show_str = flags & ZFEATURE_FLAG_READONLY_COMPAT ? "1" : "0";
|
|
} else if (strcmp(attr->name, "required_for_mos") == 0) {
|
|
show_str = flags & ZFEATURE_FLAG_MOS ? "1" : "0";
|
|
} else if (strcmp(attr->name, "activate_on_enable") == 0) {
|
|
show_str = flags & ZFEATURE_FLAG_ACTIVATE_ON_ENABLE ? "1" : "0";
|
|
} else if (strcmp(attr->name, "per_dataset") == 0) {
|
|
show_str = flags & ZFEATURE_FLAG_PER_DATASET ? "1" : "0";
|
|
}
|
|
if (show_str == NULL)
|
|
return (0);
|
|
|
|
return (snprintf(buf, PAGE_SIZE, "%s\n", show_str));
|
|
}
|
|
|
|
static void
|
|
pool_feature_to_kobj(zfs_mod_kobj_t *parent, spa_feature_t fid,
|
|
const char *name)
|
|
{
|
|
zfs_mod_kobj_t *zfs_kobj = &parent->zko_children[fid];
|
|
|
|
ASSERT3U(fid, <, SPA_FEATURES);
|
|
ASSERT(name);
|
|
|
|
int err = zfs_kobj_init(zfs_kobj, ZPOOL_FEATURE_ATTR_COUNT, 0,
|
|
pool_feature_show);
|
|
if (err)
|
|
return;
|
|
|
|
for (int i = 0; i < ZPOOL_FEATURE_ATTR_COUNT; i++)
|
|
zfs_kobj_add_attr(zfs_kobj, i, pool_feature_attrs[i]);
|
|
|
|
err = zfs_kobj_add(zfs_kobj, &parent->zko_kobj, name);
|
|
if (err)
|
|
zfs_kobj_release(&zfs_kobj->zko_kobj);
|
|
}
|
|
|
|
static int
|
|
zfs_pool_features_init(zfs_mod_kobj_t *zfs_kobj, struct kobject *parent)
|
|
{
|
|
/*
|
|
* Create a parent kobject to host pool features.
|
|
*
|
|
* '/sys/module/zfs/features.pool'
|
|
*/
|
|
int err = zfs_kobj_init(zfs_kobj, 0, SPA_FEATURES, pool_feature_show);
|
|
if (err)
|
|
return (err);
|
|
err = zfs_kobj_add(zfs_kobj, parent, ZFS_SYSFS_POOL_FEATURES);
|
|
if (err) {
|
|
zfs_kobj_release(&zfs_kobj->zko_kobj);
|
|
return (err);
|
|
}
|
|
|
|
/*
|
|
* Now create a kobject for each feature.
|
|
*
|
|
* '/sys/module/zfs/features.pool/<feature>'
|
|
*/
|
|
for (spa_feature_t i = 0; i < SPA_FEATURES; i++)
|
|
pool_feature_to_kobj(zfs_kobj, i, spa_feature_table[i].fi_guid);
|
|
|
|
return (0);
|
|
}
|
|
|
|
typedef struct prop_to_kobj_arg {
|
|
zprop_desc_t *p2k_table;
|
|
zfs_mod_kobj_t *p2k_parent;
|
|
sysfs_show_func p2k_show_func;
|
|
int p2k_attr_count;
|
|
} prop_to_kobj_arg_t;
|
|
|
|
static int
|
|
zprop_to_kobj(int prop, void *args)
|
|
{
|
|
prop_to_kobj_arg_t *data = args;
|
|
zfs_mod_kobj_t *parent = data->p2k_parent;
|
|
zfs_mod_kobj_t *zfs_kobj = &parent->zko_children[prop];
|
|
const char *name = data->p2k_table[prop].pd_name;
|
|
int err;
|
|
|
|
ASSERT(name);
|
|
|
|
err = zfs_kobj_init(zfs_kobj, data->p2k_attr_count, 0,
|
|
data->p2k_show_func);
|
|
if (err)
|
|
return (ZPROP_CONT);
|
|
|
|
for (int i = 0; i < data->p2k_attr_count; i++)
|
|
zfs_kobj_add_attr(zfs_kobj, i, zprop_attrs[i]);
|
|
|
|
err = zfs_kobj_add(zfs_kobj, &parent->zko_kobj, name);
|
|
if (err)
|
|
zfs_kobj_release(&zfs_kobj->zko_kobj);
|
|
|
|
return (ZPROP_CONT);
|
|
}
|
|
|
|
static int
|
|
zfs_sysfs_properties_init(zfs_mod_kobj_t *zfs_kobj, struct kobject *parent,
|
|
zfs_type_t type)
|
|
{
|
|
prop_to_kobj_arg_t context;
|
|
const char *name;
|
|
int err;
|
|
|
|
/*
|
|
* Create a parent kobject to host properties.
|
|
*
|
|
* '/sys/module/zfs/properties.<type>'
|
|
*/
|
|
if (type == ZFS_TYPE_POOL) {
|
|
name = ZFS_SYSFS_POOL_PROPERTIES;
|
|
context.p2k_table = zpool_prop_get_table();
|
|
context.p2k_attr_count = ZPOOL_PROP_ATTR_COUNT;
|
|
context.p2k_parent = zfs_kobj;
|
|
context.p2k_show_func = pool_property_show;
|
|
err = zfs_kobj_init(zfs_kobj, 0, ZPOOL_NUM_PROPS,
|
|
pool_property_show);
|
|
} else if (type == ZFS_TYPE_VDEV) {
|
|
name = ZFS_SYSFS_VDEV_PROPERTIES;
|
|
context.p2k_table = vdev_prop_get_table();
|
|
context.p2k_attr_count = ZPOOL_PROP_ATTR_COUNT;
|
|
context.p2k_parent = zfs_kobj;
|
|
context.p2k_show_func = vdev_property_show;
|
|
err = zfs_kobj_init(zfs_kobj, 0, VDEV_NUM_PROPS,
|
|
vdev_property_show);
|
|
} else {
|
|
name = ZFS_SYSFS_DATASET_PROPERTIES;
|
|
context.p2k_table = zfs_prop_get_table();
|
|
context.p2k_attr_count = ZFS_PROP_ATTR_COUNT;
|
|
context.p2k_parent = zfs_kobj;
|
|
context.p2k_show_func = dataset_property_show;
|
|
err = zfs_kobj_init(zfs_kobj, 0, ZFS_NUM_PROPS,
|
|
dataset_property_show);
|
|
}
|
|
|
|
if (err)
|
|
return (err);
|
|
|
|
err = zfs_kobj_add(zfs_kobj, parent, name);
|
|
if (err) {
|
|
zfs_kobj_release(&zfs_kobj->zko_kobj);
|
|
return (err);
|
|
}
|
|
|
|
/*
|
|
* Create a kobject for each property.
|
|
*
|
|
* '/sys/module/zfs/properties.<type>/<property>'
|
|
*/
|
|
(void) zprop_iter_common(zprop_to_kobj, &context, B_TRUE,
|
|
B_FALSE, type);
|
|
|
|
return (err);
|
|
}
|
|
|
|
void
|
|
zfs_sysfs_init(void)
|
|
{
|
|
struct kobject *parent;
|
|
#if defined(CONFIG_ZFS) && !defined(CONFIG_ZFS_MODULE)
|
|
parent = kobject_create_and_add("zfs", fs_kobj);
|
|
#else
|
|
parent = &(((struct module *)(THIS_MODULE))->mkobj).kobj;
|
|
#endif
|
|
int err;
|
|
|
|
if (parent == NULL)
|
|
return;
|
|
|
|
err = zfs_kernel_features_init(&kernel_features_kobj, parent);
|
|
if (err)
|
|
return;
|
|
|
|
err = zfs_pool_features_init(&pool_features_kobj, parent);
|
|
if (err) {
|
|
zfs_kobj_fini(&kernel_features_kobj);
|
|
return;
|
|
}
|
|
|
|
err = zfs_sysfs_properties_init(&pool_props_kobj, parent,
|
|
ZFS_TYPE_POOL);
|
|
if (err) {
|
|
zfs_kobj_fini(&kernel_features_kobj);
|
|
zfs_kobj_fini(&pool_features_kobj);
|
|
return;
|
|
}
|
|
|
|
err = zfs_sysfs_properties_init(&vdev_props_kobj, parent,
|
|
ZFS_TYPE_VDEV);
|
|
if (err) {
|
|
zfs_kobj_fini(&kernel_features_kobj);
|
|
zfs_kobj_fini(&pool_features_kobj);
|
|
zfs_kobj_fini(&pool_props_kobj);
|
|
return;
|
|
}
|
|
|
|
err = zfs_sysfs_properties_init(&dataset_props_kobj, parent,
|
|
ZFS_TYPE_FILESYSTEM);
|
|
if (err) {
|
|
zfs_kobj_fini(&kernel_features_kobj);
|
|
zfs_kobj_fini(&pool_features_kobj);
|
|
zfs_kobj_fini(&pool_props_kobj);
|
|
zfs_kobj_fini(&vdev_props_kobj);
|
|
return;
|
|
}
|
|
}
|
|
|
|
void
|
|
zfs_sysfs_fini(void)
|
|
{
|
|
/*
|
|
* Remove top-level kobjects; each will remove any children kobjects
|
|
*/
|
|
zfs_kobj_fini(&kernel_features_kobj);
|
|
zfs_kobj_fini(&pool_features_kobj);
|
|
zfs_kobj_fini(&pool_props_kobj);
|
|
zfs_kobj_fini(&vdev_props_kobj);
|
|
zfs_kobj_fini(&dataset_props_kobj);
|
|
}
|