322cbb50de
There is no good reason to keep genhd.h separate from the main blkdev.h header that includes it. So fold the contents of genhd.h into blkdev.h and remove genhd.h entirely. Signed-off-by: Christoph Hellwig <hch@lst.de> Reviewed-by: Chaitanya Kulkarni <kch@nvidia.com> Reviewed-by: Martin K. Petersen <martin.petersen@oracle.com> Link: https://lore.kernel.org/r/20220124093913.742411-4-hch@lst.de Signed-off-by: Jens Axboe <axboe@kernel.dk>
252 lines
6.1 KiB
C
252 lines
6.1 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* RDMA Network Block Driver
|
|
*
|
|
* Copyright (c) 2014 - 2018 ProfitBricks GmbH. All rights reserved.
|
|
* Copyright (c) 2018 - 2019 1&1 IONOS Cloud GmbH. All rights reserved.
|
|
* Copyright (c) 2019 - 2020 1&1 IONOS SE. All rights reserved.
|
|
*/
|
|
#undef pr_fmt
|
|
#define pr_fmt(fmt) KBUILD_MODNAME " L" __stringify(__LINE__) ": " fmt
|
|
|
|
#include <uapi/linux/limits.h>
|
|
#include <linux/kobject.h>
|
|
#include <linux/sysfs.h>
|
|
#include <linux/stat.h>
|
|
#include <linux/list.h>
|
|
#include <linux/moduleparam.h>
|
|
#include <linux/device.h>
|
|
|
|
#include "rnbd-srv.h"
|
|
|
|
static struct device *rnbd_dev;
|
|
static struct class *rnbd_dev_class;
|
|
static struct kobject *rnbd_devs_kobj;
|
|
|
|
static void rnbd_srv_dev_release(struct kobject *kobj)
|
|
{
|
|
struct rnbd_srv_dev *dev;
|
|
|
|
dev = container_of(kobj, struct rnbd_srv_dev, dev_kobj);
|
|
|
|
kfree(dev);
|
|
}
|
|
|
|
static struct kobj_type dev_ktype = {
|
|
.sysfs_ops = &kobj_sysfs_ops,
|
|
.release = rnbd_srv_dev_release
|
|
};
|
|
|
|
int rnbd_srv_create_dev_sysfs(struct rnbd_srv_dev *dev,
|
|
struct block_device *bdev,
|
|
const char *dev_name)
|
|
{
|
|
struct kobject *bdev_kobj;
|
|
int ret;
|
|
|
|
ret = kobject_init_and_add(&dev->dev_kobj, &dev_ktype,
|
|
rnbd_devs_kobj, dev_name);
|
|
if (ret) {
|
|
kobject_put(&dev->dev_kobj);
|
|
return ret;
|
|
}
|
|
|
|
dev->dev_sessions_kobj = kobject_create_and_add("sessions",
|
|
&dev->dev_kobj);
|
|
if (!dev->dev_sessions_kobj) {
|
|
ret = -ENOMEM;
|
|
goto free_dev_kobj;
|
|
}
|
|
|
|
bdev_kobj = &disk_to_dev(bdev->bd_disk)->kobj;
|
|
ret = sysfs_create_link(&dev->dev_kobj, bdev_kobj, "block_dev");
|
|
if (ret)
|
|
goto put_sess_kobj;
|
|
|
|
return 0;
|
|
|
|
put_sess_kobj:
|
|
kobject_put(dev->dev_sessions_kobj);
|
|
free_dev_kobj:
|
|
kobject_del(&dev->dev_kobj);
|
|
kobject_put(&dev->dev_kobj);
|
|
return ret;
|
|
}
|
|
|
|
void rnbd_srv_destroy_dev_sysfs(struct rnbd_srv_dev *dev)
|
|
{
|
|
sysfs_remove_link(&dev->dev_kobj, "block_dev");
|
|
kobject_del(dev->dev_sessions_kobj);
|
|
kobject_put(dev->dev_sessions_kobj);
|
|
kobject_del(&dev->dev_kobj);
|
|
kobject_put(&dev->dev_kobj);
|
|
}
|
|
|
|
static ssize_t read_only_show(struct kobject *kobj, struct kobj_attribute *attr,
|
|
char *page)
|
|
{
|
|
struct rnbd_srv_sess_dev *sess_dev;
|
|
|
|
sess_dev = container_of(kobj, struct rnbd_srv_sess_dev, kobj);
|
|
|
|
return sysfs_emit(page, "%d\n",
|
|
!(sess_dev->open_flags & FMODE_WRITE));
|
|
}
|
|
|
|
static struct kobj_attribute rnbd_srv_dev_session_ro_attr =
|
|
__ATTR_RO(read_only);
|
|
|
|
static ssize_t access_mode_show(struct kobject *kobj,
|
|
struct kobj_attribute *attr,
|
|
char *page)
|
|
{
|
|
struct rnbd_srv_sess_dev *sess_dev;
|
|
|
|
sess_dev = container_of(kobj, struct rnbd_srv_sess_dev, kobj);
|
|
|
|
return sysfs_emit(page, "%s\n",
|
|
rnbd_access_mode_str(sess_dev->access_mode));
|
|
}
|
|
|
|
static struct kobj_attribute rnbd_srv_dev_session_access_mode_attr =
|
|
__ATTR_RO(access_mode);
|
|
|
|
static ssize_t mapping_path_show(struct kobject *kobj,
|
|
struct kobj_attribute *attr, char *page)
|
|
{
|
|
struct rnbd_srv_sess_dev *sess_dev;
|
|
|
|
sess_dev = container_of(kobj, struct rnbd_srv_sess_dev, kobj);
|
|
|
|
return sysfs_emit(page, "%s\n", sess_dev->pathname);
|
|
}
|
|
|
|
static struct kobj_attribute rnbd_srv_dev_session_mapping_path_attr =
|
|
__ATTR_RO(mapping_path);
|
|
|
|
static ssize_t rnbd_srv_dev_session_force_close_show(struct kobject *kobj,
|
|
struct kobj_attribute *attr, char *page)
|
|
{
|
|
return sysfs_emit(page, "Usage: echo 1 > %s\n",
|
|
attr->attr.name);
|
|
}
|
|
|
|
static ssize_t rnbd_srv_dev_session_force_close_store(struct kobject *kobj,
|
|
struct kobj_attribute *attr,
|
|
const char *buf, size_t count)
|
|
{
|
|
struct rnbd_srv_sess_dev *sess_dev;
|
|
|
|
sess_dev = container_of(kobj, struct rnbd_srv_sess_dev, kobj);
|
|
|
|
if (!sysfs_streq(buf, "1")) {
|
|
rnbd_srv_err(sess_dev, "%s: invalid value: '%s'\n",
|
|
attr->attr.name, buf);
|
|
return -EINVAL;
|
|
}
|
|
|
|
rnbd_srv_info(sess_dev, "force close requested\n");
|
|
rnbd_srv_sess_dev_force_close(sess_dev, attr);
|
|
|
|
return count;
|
|
}
|
|
|
|
static struct kobj_attribute rnbd_srv_dev_session_force_close_attr =
|
|
__ATTR(force_close, 0644,
|
|
rnbd_srv_dev_session_force_close_show,
|
|
rnbd_srv_dev_session_force_close_store);
|
|
|
|
static struct attribute *rnbd_srv_default_dev_sessions_attrs[] = {
|
|
&rnbd_srv_dev_session_access_mode_attr.attr,
|
|
&rnbd_srv_dev_session_ro_attr.attr,
|
|
&rnbd_srv_dev_session_mapping_path_attr.attr,
|
|
&rnbd_srv_dev_session_force_close_attr.attr,
|
|
NULL,
|
|
};
|
|
|
|
static struct attribute_group rnbd_srv_default_dev_session_attr_group = {
|
|
.attrs = rnbd_srv_default_dev_sessions_attrs,
|
|
};
|
|
|
|
void rnbd_srv_destroy_dev_session_sysfs(struct rnbd_srv_sess_dev *sess_dev)
|
|
{
|
|
sysfs_remove_group(&sess_dev->kobj,
|
|
&rnbd_srv_default_dev_session_attr_group);
|
|
|
|
kobject_del(&sess_dev->kobj);
|
|
kobject_put(&sess_dev->kobj);
|
|
}
|
|
|
|
static void rnbd_srv_sess_dev_release(struct kobject *kobj)
|
|
{
|
|
struct rnbd_srv_sess_dev *sess_dev;
|
|
|
|
sess_dev = container_of(kobj, struct rnbd_srv_sess_dev, kobj);
|
|
rnbd_destroy_sess_dev(sess_dev, sess_dev->keep_id);
|
|
}
|
|
|
|
static struct kobj_type rnbd_srv_sess_dev_ktype = {
|
|
.sysfs_ops = &kobj_sysfs_ops,
|
|
.release = rnbd_srv_sess_dev_release,
|
|
};
|
|
|
|
int rnbd_srv_create_dev_session_sysfs(struct rnbd_srv_sess_dev *sess_dev)
|
|
{
|
|
int ret;
|
|
|
|
ret = kobject_init_and_add(&sess_dev->kobj, &rnbd_srv_sess_dev_ktype,
|
|
sess_dev->dev->dev_sessions_kobj, "%s",
|
|
sess_dev->sess->sessname);
|
|
if (ret) {
|
|
kobject_put(&sess_dev->kobj);
|
|
return ret;
|
|
}
|
|
|
|
ret = sysfs_create_group(&sess_dev->kobj,
|
|
&rnbd_srv_default_dev_session_attr_group);
|
|
if (ret) {
|
|
kobject_del(&sess_dev->kobj);
|
|
kobject_put(&sess_dev->kobj);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
int rnbd_srv_create_sysfs_files(void)
|
|
{
|
|
int err;
|
|
|
|
rnbd_dev_class = class_create(THIS_MODULE, "rnbd-server");
|
|
if (IS_ERR(rnbd_dev_class))
|
|
return PTR_ERR(rnbd_dev_class);
|
|
|
|
rnbd_dev = device_create(rnbd_dev_class, NULL,
|
|
MKDEV(0, 0), NULL, "ctl");
|
|
if (IS_ERR(rnbd_dev)) {
|
|
err = PTR_ERR(rnbd_dev);
|
|
goto cls_destroy;
|
|
}
|
|
rnbd_devs_kobj = kobject_create_and_add("devices", &rnbd_dev->kobj);
|
|
if (!rnbd_devs_kobj) {
|
|
err = -ENOMEM;
|
|
goto dev_destroy;
|
|
}
|
|
|
|
return 0;
|
|
|
|
dev_destroy:
|
|
device_destroy(rnbd_dev_class, MKDEV(0, 0));
|
|
cls_destroy:
|
|
class_destroy(rnbd_dev_class);
|
|
|
|
return err;
|
|
}
|
|
|
|
void rnbd_srv_destroy_sysfs_files(void)
|
|
{
|
|
kobject_del(rnbd_devs_kobj);
|
|
kobject_put(rnbd_devs_kobj);
|
|
device_destroy(rnbd_dev_class, MKDEV(0, 0));
|
|
class_destroy(rnbd_dev_class);
|
|
}
|