2017-08-15 05:04:53 +07:00
|
|
|
/*
|
|
|
|
* Add configfs and memory store: Kyungchan Koh <kkc6196@fb.com> and
|
|
|
|
* Shaohua Li <shli@fb.com>
|
|
|
|
*/
|
2013-10-25 17:52:25 +07:00
|
|
|
#include <linux/module.h>
|
2013-12-21 06:11:01 +07:00
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
#include <linux/moduleparam.h>
|
|
|
|
#include <linux/sched.h>
|
|
|
|
#include <linux/fs.h>
|
|
|
|
#include <linux/blkdev.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/blk-mq.h>
|
|
|
|
#include <linux/hrtimer.h>
|
2015-11-13 02:25:10 +07:00
|
|
|
#include <linux/lightnvm.h>
|
2017-08-15 05:04:53 +07:00
|
|
|
#include <linux/configfs.h>
|
2013-10-25 17:52:25 +07:00
|
|
|
|
|
|
|
struct nullb_cmd {
|
|
|
|
struct list_head list;
|
|
|
|
struct llist_node ll_list;
|
|
|
|
struct call_single_data csd;
|
|
|
|
struct request *rq;
|
|
|
|
struct bio *bio;
|
|
|
|
unsigned int tag;
|
|
|
|
struct nullb_queue *nq;
|
null_blk: set a separate timer for each command
For the Timer IRQ mode (i.e., when command completions are delayed),
there is one timer for each CPU. Each of these timers
. has a completion queue associated with it, containing all the
command completions to be executed when the timer fires;
. is set, and a new completion-to-execute is inserted into its
completion queue, every time the dispatch code for a new command
happens to be executed on the CPU related to the timer.
This implies that, if the dispatch of a new command happens to be
executed on a CPU whose timer has already been set, but has not yet
fired, then the timer is set again, to the completion time of the
newly arrived command. When the timer eventually fires, all its queued
completions are executed.
This way of handling delayed command completions entails the following
problem: if more than one command completion is inserted into the
queue of a timer before the timer fires, then the expiration time for
the timer is moved forward every time each of these completions is
enqueued. As a consequence, only the last completion enqueued enjoys a
correct execution time, while all previous completions are unjustly
delayed until the last completion is executed (and at that time they
are executed all together).
Specifically, if all the above completions are enqueued almost at the
same time, then the problem is negligible. On the opposite end, if
every completion is enqueued a while after the previous completion was
enqueued (in the extreme case, it is enqueued only right before the
timer would have expired), then every enqueued completion, except for
the last one, experiences an inflated delay, proportional to the number
of completions enqueued after it. In the end, commands, and thus I/O
requests, may be completed at an arbitrarily lower rate than the
desired one.
This commit addresses this issue by replacing per-CPU timers with
per-command timers, i.e., by associating an individual timer with each
command.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna Avanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:17 +07:00
|
|
|
struct hrtimer timer;
|
2013-10-25 17:52:25 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
struct nullb_queue {
|
|
|
|
unsigned long *tag_map;
|
|
|
|
wait_queue_head_t wait;
|
|
|
|
unsigned int queue_depth;
|
2017-08-15 05:04:52 +07:00
|
|
|
struct nullb_device *dev;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
|
|
|
struct nullb_cmd *cmds;
|
|
|
|
};
|
|
|
|
|
2017-08-15 05:04:53 +07:00
|
|
|
/*
|
|
|
|
* Status flags for nullb_device.
|
|
|
|
*
|
|
|
|
* CONFIGURED: Device has been configured and turned on. Cannot reconfigure.
|
|
|
|
* UP: Device is currently on and visible in userspace.
|
|
|
|
*/
|
|
|
|
enum nullb_device_flags {
|
|
|
|
NULLB_DEV_FL_CONFIGURED = 0,
|
|
|
|
NULLB_DEV_FL_UP = 1,
|
|
|
|
};
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
struct nullb_device {
|
|
|
|
struct nullb *nullb;
|
2017-08-15 05:04:53 +07:00
|
|
|
struct config_item item;
|
|
|
|
unsigned long flags; /* device flags */
|
2017-08-15 05:04:52 +07:00
|
|
|
|
|
|
|
unsigned long size; /* device size in MB */
|
|
|
|
unsigned long completion_nsec; /* time in ns to complete a request */
|
|
|
|
unsigned int submit_queues; /* number of submission queues */
|
|
|
|
unsigned int home_node; /* home node for the device */
|
|
|
|
unsigned int queue_mode; /* block interface */
|
|
|
|
unsigned int blocksize; /* block size */
|
|
|
|
unsigned int irqmode; /* IRQ completion handler */
|
|
|
|
unsigned int hw_queue_depth; /* queue depth */
|
|
|
|
bool use_lightnvm; /* register as a LightNVM device */
|
|
|
|
bool blocking; /* blocking blk-mq device */
|
|
|
|
bool use_per_node_hctx; /* use per-node allocation for hardware context */
|
|
|
|
};
|
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
struct nullb {
|
2017-08-15 05:04:52 +07:00
|
|
|
struct nullb_device *dev;
|
2013-10-25 17:52:25 +07:00
|
|
|
struct list_head list;
|
|
|
|
unsigned int index;
|
|
|
|
struct request_queue *q;
|
|
|
|
struct gendisk *disk;
|
2016-09-16 19:25:07 +07:00
|
|
|
struct nvm_dev *ndev;
|
2017-06-21 03:22:01 +07:00
|
|
|
struct blk_mq_tag_set *tag_set;
|
|
|
|
struct blk_mq_tag_set __tag_set;
|
2013-10-25 17:52:25 +07:00
|
|
|
struct hrtimer timer;
|
|
|
|
unsigned int queue_depth;
|
|
|
|
spinlock_t lock;
|
|
|
|
|
|
|
|
struct nullb_queue *queues;
|
|
|
|
unsigned int nr_queues;
|
2015-11-13 02:25:10 +07:00
|
|
|
char disk_name[DISK_NAME_LEN];
|
2013-10-25 17:52:25 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
static LIST_HEAD(nullb_list);
|
|
|
|
static struct mutex lock;
|
|
|
|
static int null_major;
|
|
|
|
static int nullb_indexes;
|
2015-11-19 18:50:08 +07:00
|
|
|
static struct kmem_cache *ppa_cache;
|
2017-06-21 03:22:01 +07:00
|
|
|
static struct blk_mq_tag_set tag_set;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
|
|
|
enum {
|
|
|
|
NULL_IRQ_NONE = 0,
|
|
|
|
NULL_IRQ_SOFTIRQ = 1,
|
|
|
|
NULL_IRQ_TIMER = 2,
|
2014-02-10 18:24:40 +07:00
|
|
|
};
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2014-02-10 18:24:40 +07:00
|
|
|
enum {
|
2013-10-25 17:52:25 +07:00
|
|
|
NULL_Q_BIO = 0,
|
|
|
|
NULL_Q_RQ = 1,
|
|
|
|
NULL_Q_MQ = 2,
|
|
|
|
};
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static int g_submit_queues = 1;
|
|
|
|
module_param_named(submit_queues, g_submit_queues, int, S_IRUGO);
|
2013-10-25 17:52:25 +07:00
|
|
|
MODULE_PARM_DESC(submit_queues, "Number of submission queues");
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static int g_home_node = NUMA_NO_NODE;
|
|
|
|
module_param_named(home_node, g_home_node, int, S_IRUGO);
|
2013-10-25 17:52:25 +07:00
|
|
|
MODULE_PARM_DESC(home_node, "Home node for the device");
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static int g_queue_mode = NULL_Q_MQ;
|
2014-11-27 04:45:48 +07:00
|
|
|
|
|
|
|
static int null_param_store_val(const char *str, int *val, int min, int max)
|
|
|
|
{
|
|
|
|
int ret, new_val;
|
|
|
|
|
|
|
|
ret = kstrtoint(str, 10, &new_val);
|
|
|
|
if (ret)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
if (new_val < min || new_val > max)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
*val = new_val;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int null_set_queue_mode(const char *str, const struct kernel_param *kp)
|
|
|
|
{
|
2017-08-15 05:04:52 +07:00
|
|
|
return null_param_store_val(str, &g_queue_mode, NULL_Q_BIO, NULL_Q_MQ);
|
2014-11-27 04:45:48 +07:00
|
|
|
}
|
|
|
|
|
2015-05-27 08:39:38 +07:00
|
|
|
static const struct kernel_param_ops null_queue_mode_param_ops = {
|
2014-11-27 04:45:48 +07:00
|
|
|
.set = null_set_queue_mode,
|
|
|
|
.get = param_get_int,
|
|
|
|
};
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
device_param_cb(queue_mode, &null_queue_mode_param_ops, &g_queue_mode, S_IRUGO);
|
2014-06-12 04:13:50 +07:00
|
|
|
MODULE_PARM_DESC(queue_mode, "Block interface to use (0=bio,1=rq,2=multiqueue)");
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static int g_gb = 250;
|
|
|
|
module_param_named(gb, g_gb, int, S_IRUGO);
|
2013-10-25 17:52:25 +07:00
|
|
|
MODULE_PARM_DESC(gb, "Size in GB");
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static int g_bs = 512;
|
|
|
|
module_param_named(bs, g_bs, int, S_IRUGO);
|
2013-10-25 17:52:25 +07:00
|
|
|
MODULE_PARM_DESC(bs, "Block size (in bytes)");
|
|
|
|
|
2017-06-21 03:22:01 +07:00
|
|
|
static int nr_devices = 1;
|
2013-10-25 17:52:25 +07:00
|
|
|
module_param(nr_devices, int, S_IRUGO);
|
|
|
|
MODULE_PARM_DESC(nr_devices, "Number of devices to register");
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static bool g_use_lightnvm;
|
|
|
|
module_param_named(use_lightnvm, g_use_lightnvm, bool, S_IRUGO);
|
2015-11-13 02:25:10 +07:00
|
|
|
MODULE_PARM_DESC(use_lightnvm, "Register as a LightNVM device");
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static bool g_blocking;
|
|
|
|
module_param_named(blocking, g_blocking, bool, S_IRUGO);
|
2017-03-31 02:44:26 +07:00
|
|
|
MODULE_PARM_DESC(blocking, "Register as a blocking blk-mq driver device");
|
|
|
|
|
2017-06-21 03:22:01 +07:00
|
|
|
static bool shared_tags;
|
|
|
|
module_param(shared_tags, bool, S_IRUGO);
|
|
|
|
MODULE_PARM_DESC(shared_tags, "Share tag set between devices for blk-mq");
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static int g_irqmode = NULL_IRQ_SOFTIRQ;
|
2014-11-27 04:45:48 +07:00
|
|
|
|
|
|
|
static int null_set_irqmode(const char *str, const struct kernel_param *kp)
|
|
|
|
{
|
2017-08-15 05:04:52 +07:00
|
|
|
return null_param_store_val(str, &g_irqmode, NULL_IRQ_NONE,
|
2014-11-27 04:45:48 +07:00
|
|
|
NULL_IRQ_TIMER);
|
|
|
|
}
|
|
|
|
|
2015-05-27 08:39:38 +07:00
|
|
|
static const struct kernel_param_ops null_irqmode_param_ops = {
|
2014-11-27 04:45:48 +07:00
|
|
|
.set = null_set_irqmode,
|
|
|
|
.get = param_get_int,
|
|
|
|
};
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
device_param_cb(irqmode, &null_irqmode_param_ops, &g_irqmode, S_IRUGO);
|
2013-10-25 17:52:25 +07:00
|
|
|
MODULE_PARM_DESC(irqmode, "IRQ completion handler. 0-none, 1-softirq, 2-timer");
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static unsigned long g_completion_nsec = 10000;
|
|
|
|
module_param_named(completion_nsec, g_completion_nsec, ulong, S_IRUGO);
|
2013-10-25 17:52:25 +07:00
|
|
|
MODULE_PARM_DESC(completion_nsec, "Time in ns to complete a request in hardware. Default: 10,000ns");
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static int g_hw_queue_depth = 64;
|
|
|
|
module_param_named(hw_queue_depth, g_hw_queue_depth, int, S_IRUGO);
|
2013-10-25 17:52:25 +07:00
|
|
|
MODULE_PARM_DESC(hw_queue_depth, "Queue depth for each hardware queue. Default: 64");
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static bool g_use_per_node_hctx;
|
|
|
|
module_param_named(use_per_node_hctx, g_use_per_node_hctx, bool, S_IRUGO);
|
2013-12-21 06:11:00 +07:00
|
|
|
MODULE_PARM_DESC(use_per_node_hctx, "Use per-node allocation for hardware context queues. Default: false");
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2017-08-15 05:04:53 +07:00
|
|
|
static struct nullb_device *null_alloc_dev(void);
|
|
|
|
static void null_free_dev(struct nullb_device *dev);
|
|
|
|
|
|
|
|
static inline struct nullb_device *to_nullb_device(struct config_item *item)
|
|
|
|
{
|
|
|
|
return item ? container_of(item, struct nullb_device, item) : NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline ssize_t nullb_device_uint_attr_show(unsigned int val, char *page)
|
|
|
|
{
|
|
|
|
return snprintf(page, PAGE_SIZE, "%u\n", val);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline ssize_t nullb_device_ulong_attr_show(unsigned long val,
|
|
|
|
char *page)
|
|
|
|
{
|
|
|
|
return snprintf(page, PAGE_SIZE, "%lu\n", val);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline ssize_t nullb_device_bool_attr_show(bool val, char *page)
|
|
|
|
{
|
|
|
|
return snprintf(page, PAGE_SIZE, "%u\n", val);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t nullb_device_uint_attr_store(unsigned int *val,
|
|
|
|
const char *page, size_t count)
|
|
|
|
{
|
|
|
|
unsigned int tmp;
|
|
|
|
int result;
|
|
|
|
|
|
|
|
result = kstrtouint(page, 0, &tmp);
|
|
|
|
if (result)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
*val = tmp;
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t nullb_device_ulong_attr_store(unsigned long *val,
|
|
|
|
const char *page, size_t count)
|
|
|
|
{
|
|
|
|
int result;
|
|
|
|
unsigned long tmp;
|
|
|
|
|
|
|
|
result = kstrtoul(page, 0, &tmp);
|
|
|
|
if (result)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
*val = tmp;
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t nullb_device_bool_attr_store(bool *val, const char *page,
|
|
|
|
size_t count)
|
|
|
|
{
|
|
|
|
bool tmp;
|
|
|
|
int result;
|
|
|
|
|
|
|
|
result = kstrtobool(page, &tmp);
|
|
|
|
if (result)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
*val = tmp;
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The following macro should only be used with TYPE = {uint, ulong, bool}. */
|
|
|
|
#define NULLB_DEVICE_ATTR(NAME, TYPE) \
|
|
|
|
static ssize_t \
|
|
|
|
nullb_device_##NAME##_show(struct config_item *item, char *page) \
|
|
|
|
{ \
|
|
|
|
return nullb_device_##TYPE##_attr_show( \
|
|
|
|
to_nullb_device(item)->NAME, page); \
|
|
|
|
} \
|
|
|
|
static ssize_t \
|
|
|
|
nullb_device_##NAME##_store(struct config_item *item, const char *page, \
|
|
|
|
size_t count) \
|
|
|
|
{ \
|
|
|
|
if (test_bit(NULLB_DEV_FL_CONFIGURED, &to_nullb_device(item)->flags)) \
|
|
|
|
return -EBUSY; \
|
|
|
|
return nullb_device_##TYPE##_attr_store( \
|
|
|
|
&to_nullb_device(item)->NAME, page, count); \
|
|
|
|
} \
|
|
|
|
CONFIGFS_ATTR(nullb_device_, NAME);
|
|
|
|
|
|
|
|
NULLB_DEVICE_ATTR(size, ulong);
|
|
|
|
NULLB_DEVICE_ATTR(completion_nsec, ulong);
|
|
|
|
NULLB_DEVICE_ATTR(submit_queues, uint);
|
|
|
|
NULLB_DEVICE_ATTR(home_node, uint);
|
|
|
|
NULLB_DEVICE_ATTR(queue_mode, uint);
|
|
|
|
NULLB_DEVICE_ATTR(blocksize, uint);
|
|
|
|
NULLB_DEVICE_ATTR(irqmode, uint);
|
|
|
|
NULLB_DEVICE_ATTR(hw_queue_depth, uint);
|
|
|
|
NULLB_DEVICE_ATTR(use_lightnvm, bool);
|
|
|
|
NULLB_DEVICE_ATTR(blocking, bool);
|
|
|
|
NULLB_DEVICE_ATTR(use_per_node_hctx, bool);
|
|
|
|
|
|
|
|
static struct configfs_attribute *nullb_device_attrs[] = {
|
|
|
|
&nullb_device_attr_size,
|
|
|
|
&nullb_device_attr_completion_nsec,
|
|
|
|
&nullb_device_attr_submit_queues,
|
|
|
|
&nullb_device_attr_home_node,
|
|
|
|
&nullb_device_attr_queue_mode,
|
|
|
|
&nullb_device_attr_blocksize,
|
|
|
|
&nullb_device_attr_irqmode,
|
|
|
|
&nullb_device_attr_hw_queue_depth,
|
|
|
|
&nullb_device_attr_use_lightnvm,
|
|
|
|
&nullb_device_attr_blocking,
|
|
|
|
&nullb_device_attr_use_per_node_hctx,
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
|
|
|
|
static void nullb_device_release(struct config_item *item)
|
|
|
|
{
|
|
|
|
null_free_dev(to_nullb_device(item));
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct configfs_item_operations nullb_device_ops = {
|
|
|
|
.release = nullb_device_release,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct config_item_type nullb_device_type = {
|
|
|
|
.ct_item_ops = &nullb_device_ops,
|
|
|
|
.ct_attrs = nullb_device_attrs,
|
|
|
|
.ct_owner = THIS_MODULE,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct
|
|
|
|
config_item *nullb_group_make_item(struct config_group *group, const char *name)
|
|
|
|
{
|
|
|
|
struct nullb_device *dev;
|
|
|
|
|
|
|
|
dev = null_alloc_dev();
|
|
|
|
if (!dev)
|
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
|
|
|
|
config_item_init_type_name(&dev->item, name, &nullb_device_type);
|
|
|
|
|
|
|
|
return &dev->item;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
nullb_group_drop_item(struct config_group *group, struct config_item *item)
|
|
|
|
{
|
|
|
|
config_item_put(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t memb_group_features_show(struct config_item *item, char *page)
|
|
|
|
{
|
|
|
|
return snprintf(page, PAGE_SIZE, "\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
CONFIGFS_ATTR_RO(memb_group_, features);
|
|
|
|
|
|
|
|
static struct configfs_attribute *nullb_group_attrs[] = {
|
|
|
|
&memb_group_attr_features,
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct configfs_group_operations nullb_group_ops = {
|
|
|
|
.make_item = nullb_group_make_item,
|
|
|
|
.drop_item = nullb_group_drop_item,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct config_item_type nullb_group_type = {
|
|
|
|
.ct_group_ops = &nullb_group_ops,
|
|
|
|
.ct_attrs = nullb_group_attrs,
|
|
|
|
.ct_owner = THIS_MODULE,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct configfs_subsystem nullb_subsys = {
|
|
|
|
.su_group = {
|
|
|
|
.cg_item = {
|
|
|
|
.ci_namebuf = "nullb",
|
|
|
|
.ci_type = &nullb_group_type,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static struct nullb_device *null_alloc_dev(void)
|
|
|
|
{
|
|
|
|
struct nullb_device *dev;
|
|
|
|
|
|
|
|
dev = kzalloc(sizeof(*dev), GFP_KERNEL);
|
|
|
|
if (!dev)
|
|
|
|
return NULL;
|
|
|
|
dev->size = g_gb * 1024;
|
|
|
|
dev->completion_nsec = g_completion_nsec;
|
|
|
|
dev->submit_queues = g_submit_queues;
|
|
|
|
dev->home_node = g_home_node;
|
|
|
|
dev->queue_mode = g_queue_mode;
|
|
|
|
dev->blocksize = g_bs;
|
|
|
|
dev->irqmode = g_irqmode;
|
|
|
|
dev->hw_queue_depth = g_hw_queue_depth;
|
|
|
|
dev->use_lightnvm = g_use_lightnvm;
|
|
|
|
dev->blocking = g_blocking;
|
|
|
|
dev->use_per_node_hctx = g_use_per_node_hctx;
|
|
|
|
return dev;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void null_free_dev(struct nullb_device *dev)
|
|
|
|
{
|
|
|
|
kfree(dev);
|
|
|
|
}
|
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
static void put_tag(struct nullb_queue *nq, unsigned int tag)
|
|
|
|
{
|
|
|
|
clear_bit_unlock(tag, nq->tag_map);
|
|
|
|
|
|
|
|
if (waitqueue_active(&nq->wait))
|
|
|
|
wake_up(&nq->wait);
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned int get_tag(struct nullb_queue *nq)
|
|
|
|
{
|
|
|
|
unsigned int tag;
|
|
|
|
|
|
|
|
do {
|
|
|
|
tag = find_first_zero_bit(nq->tag_map, nq->queue_depth);
|
|
|
|
if (tag >= nq->queue_depth)
|
|
|
|
return -1U;
|
|
|
|
} while (test_and_set_bit_lock(tag, nq->tag_map));
|
|
|
|
|
|
|
|
return tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void free_cmd(struct nullb_cmd *cmd)
|
|
|
|
{
|
|
|
|
put_tag(cmd->nq, cmd->tag);
|
|
|
|
}
|
|
|
|
|
null_blk: set a separate timer for each command
For the Timer IRQ mode (i.e., when command completions are delayed),
there is one timer for each CPU. Each of these timers
. has a completion queue associated with it, containing all the
command completions to be executed when the timer fires;
. is set, and a new completion-to-execute is inserted into its
completion queue, every time the dispatch code for a new command
happens to be executed on the CPU related to the timer.
This implies that, if the dispatch of a new command happens to be
executed on a CPU whose timer has already been set, but has not yet
fired, then the timer is set again, to the completion time of the
newly arrived command. When the timer eventually fires, all its queued
completions are executed.
This way of handling delayed command completions entails the following
problem: if more than one command completion is inserted into the
queue of a timer before the timer fires, then the expiration time for
the timer is moved forward every time each of these completions is
enqueued. As a consequence, only the last completion enqueued enjoys a
correct execution time, while all previous completions are unjustly
delayed until the last completion is executed (and at that time they
are executed all together).
Specifically, if all the above completions are enqueued almost at the
same time, then the problem is negligible. On the opposite end, if
every completion is enqueued a while after the previous completion was
enqueued (in the extreme case, it is enqueued only right before the
timer would have expired), then every enqueued completion, except for
the last one, experiences an inflated delay, proportional to the number
of completions enqueued after it. In the end, commands, and thus I/O
requests, may be completed at an arbitrarily lower rate than the
desired one.
This commit addresses this issue by replacing per-CPU timers with
per-command timers, i.e., by associating an individual timer with each
command.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna Avanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:17 +07:00
|
|
|
static enum hrtimer_restart null_cmd_timer_expired(struct hrtimer *timer);
|
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
static struct nullb_cmd *__alloc_cmd(struct nullb_queue *nq)
|
|
|
|
{
|
|
|
|
struct nullb_cmd *cmd;
|
|
|
|
unsigned int tag;
|
|
|
|
|
|
|
|
tag = get_tag(nq);
|
|
|
|
if (tag != -1U) {
|
|
|
|
cmd = &nq->cmds[tag];
|
|
|
|
cmd->tag = tag;
|
|
|
|
cmd->nq = nq;
|
2017-08-15 05:04:52 +07:00
|
|
|
if (nq->dev->irqmode == NULL_IRQ_TIMER) {
|
null_blk: set a separate timer for each command
For the Timer IRQ mode (i.e., when command completions are delayed),
there is one timer for each CPU. Each of these timers
. has a completion queue associated with it, containing all the
command completions to be executed when the timer fires;
. is set, and a new completion-to-execute is inserted into its
completion queue, every time the dispatch code for a new command
happens to be executed on the CPU related to the timer.
This implies that, if the dispatch of a new command happens to be
executed on a CPU whose timer has already been set, but has not yet
fired, then the timer is set again, to the completion time of the
newly arrived command. When the timer eventually fires, all its queued
completions are executed.
This way of handling delayed command completions entails the following
problem: if more than one command completion is inserted into the
queue of a timer before the timer fires, then the expiration time for
the timer is moved forward every time each of these completions is
enqueued. As a consequence, only the last completion enqueued enjoys a
correct execution time, while all previous completions are unjustly
delayed until the last completion is executed (and at that time they
are executed all together).
Specifically, if all the above completions are enqueued almost at the
same time, then the problem is negligible. On the opposite end, if
every completion is enqueued a while after the previous completion was
enqueued (in the extreme case, it is enqueued only right before the
timer would have expired), then every enqueued completion, except for
the last one, experiences an inflated delay, proportional to the number
of completions enqueued after it. In the end, commands, and thus I/O
requests, may be completed at an arbitrarily lower rate than the
desired one.
This commit addresses this issue by replacing per-CPU timers with
per-command timers, i.e., by associating an individual timer with each
command.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna Avanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:17 +07:00
|
|
|
hrtimer_init(&cmd->timer, CLOCK_MONOTONIC,
|
|
|
|
HRTIMER_MODE_REL);
|
|
|
|
cmd->timer.function = null_cmd_timer_expired;
|
|
|
|
}
|
2013-10-25 17:52:25 +07:00
|
|
|
return cmd;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct nullb_cmd *alloc_cmd(struct nullb_queue *nq, int can_wait)
|
|
|
|
{
|
|
|
|
struct nullb_cmd *cmd;
|
|
|
|
DEFINE_WAIT(wait);
|
|
|
|
|
|
|
|
cmd = __alloc_cmd(nq);
|
|
|
|
if (cmd || !can_wait)
|
|
|
|
return cmd;
|
|
|
|
|
|
|
|
do {
|
|
|
|
prepare_to_wait(&nq->wait, &wait, TASK_UNINTERRUPTIBLE);
|
|
|
|
cmd = __alloc_cmd(nq);
|
|
|
|
if (cmd)
|
|
|
|
break;
|
|
|
|
|
|
|
|
io_schedule();
|
|
|
|
} while (1);
|
|
|
|
|
|
|
|
finish_wait(&nq->wait, &wait);
|
|
|
|
return cmd;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void end_cmd(struct nullb_cmd *cmd)
|
|
|
|
{
|
null_blk: guarantee device restart in all irq modes
In single-queue (block layer) mode,the function null_rq_prep_fn stops
the device if alloc_cmd fails. Then, once stopped, the device must be
restarted on the next command completion, so that the request(s) for
which alloc_cmd failed can be requeued. Otherwise the device hangs.
Unfortunately, device restart is currently performed only for delayed
completions, i.e., in irqmode==2. This fact causes hangs, for the
above reasons, with the other irqmodes in combination with single-queue
block layer.
This commits addresses this issue by making sure that, if stopped, the
device is properly restarted for all irqmodes on completions.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna AVanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:18 +07:00
|
|
|
struct request_queue *q = NULL;
|
2017-08-15 05:04:52 +07:00
|
|
|
int queue_mode = cmd->nq->dev->queue_mode;
|
null_blk: guarantee device restart in all irq modes
In single-queue (block layer) mode,the function null_rq_prep_fn stops
the device if alloc_cmd fails. Then, once stopped, the device must be
restarted on the next command completion, so that the request(s) for
which alloc_cmd failed can be requeued. Otherwise the device hangs.
Unfortunately, device restart is currently performed only for delayed
completions, i.e., in irqmode==2. This fact causes hangs, for the
above reasons, with the other irqmodes in combination with single-queue
block layer.
This commits addresses this issue by making sure that, if stopped, the
device is properly restarted for all irqmodes on completions.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna AVanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:18 +07:00
|
|
|
|
2015-12-15 16:56:40 +07:00
|
|
|
if (cmd->rq)
|
|
|
|
q = cmd->rq->q;
|
|
|
|
|
2014-02-10 18:24:40 +07:00
|
|
|
switch (queue_mode) {
|
|
|
|
case NULL_Q_MQ:
|
2017-06-03 14:38:04 +07:00
|
|
|
blk_mq_end_request(cmd->rq, BLK_STS_OK);
|
2014-02-10 18:24:40 +07:00
|
|
|
return;
|
|
|
|
case NULL_Q_RQ:
|
|
|
|
INIT_LIST_HEAD(&cmd->rq->queuelist);
|
2017-06-03 14:38:04 +07:00
|
|
|
blk_end_request_all(cmd->rq, BLK_STS_OK);
|
2014-02-10 18:24:40 +07:00
|
|
|
break;
|
|
|
|
case NULL_Q_BIO:
|
2015-07-20 20:29:37 +07:00
|
|
|
bio_endio(cmd->bio);
|
2015-12-29 03:02:47 +07:00
|
|
|
break;
|
2014-02-10 18:24:40 +07:00
|
|
|
}
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2015-12-29 03:02:47 +07:00
|
|
|
free_cmd(cmd);
|
|
|
|
|
null_blk: guarantee device restart in all irq modes
In single-queue (block layer) mode,the function null_rq_prep_fn stops
the device if alloc_cmd fails. Then, once stopped, the device must be
restarted on the next command completion, so that the request(s) for
which alloc_cmd failed can be requeued. Otherwise the device hangs.
Unfortunately, device restart is currently performed only for delayed
completions, i.e., in irqmode==2. This fact causes hangs, for the
above reasons, with the other irqmodes in combination with single-queue
block layer.
This commits addresses this issue by making sure that, if stopped, the
device is properly restarted for all irqmodes on completions.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna AVanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:18 +07:00
|
|
|
/* Restart queue if needed, as we are freeing a tag */
|
2015-12-29 03:02:47 +07:00
|
|
|
if (queue_mode == NULL_Q_RQ && blk_queue_stopped(q)) {
|
null_blk: guarantee device restart in all irq modes
In single-queue (block layer) mode,the function null_rq_prep_fn stops
the device if alloc_cmd fails. Then, once stopped, the device must be
restarted on the next command completion, so that the request(s) for
which alloc_cmd failed can be requeued. Otherwise the device hangs.
Unfortunately, device restart is currently performed only for delayed
completions, i.e., in irqmode==2. This fact causes hangs, for the
above reasons, with the other irqmodes in combination with single-queue
block layer.
This commits addresses this issue by making sure that, if stopped, the
device is properly restarted for all irqmodes on completions.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna AVanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:18 +07:00
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
spin_lock_irqsave(q->queue_lock, flags);
|
2015-12-29 03:02:47 +07:00
|
|
|
blk_start_queue_async(q);
|
null_blk: guarantee device restart in all irq modes
In single-queue (block layer) mode,the function null_rq_prep_fn stops
the device if alloc_cmd fails. Then, once stopped, the device must be
restarted on the next command completion, so that the request(s) for
which alloc_cmd failed can be requeued. Otherwise the device hangs.
Unfortunately, device restart is currently performed only for delayed
completions, i.e., in irqmode==2. This fact causes hangs, for the
above reasons, with the other irqmodes in combination with single-queue
block layer.
This commits addresses this issue by making sure that, if stopped, the
device is properly restarted for all irqmodes on completions.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna AVanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:18 +07:00
|
|
|
spin_unlock_irqrestore(q->queue_lock, flags);
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
null_blk: guarantee device restart in all irq modes
In single-queue (block layer) mode,the function null_rq_prep_fn stops
the device if alloc_cmd fails. Then, once stopped, the device must be
restarted on the next command completion, so that the request(s) for
which alloc_cmd failed can be requeued. Otherwise the device hangs.
Unfortunately, device restart is currently performed only for delayed
completions, i.e., in irqmode==2. This fact causes hangs, for the
above reasons, with the other irqmodes in combination with single-queue
block layer.
This commits addresses this issue by making sure that, if stopped, the
device is properly restarted for all irqmodes on completions.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna AVanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:18 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static enum hrtimer_restart null_cmd_timer_expired(struct hrtimer *timer)
|
|
|
|
{
|
|
|
|
end_cmd(container_of(timer, struct nullb_cmd, timer));
|
2013-10-25 17:52:25 +07:00
|
|
|
|
|
|
|
return HRTIMER_NORESTART;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void null_cmd_end_timer(struct nullb_cmd *cmd)
|
|
|
|
{
|
2017-08-15 05:04:52 +07:00
|
|
|
ktime_t kt = cmd->nq->dev->completion_nsec;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
null_blk: set a separate timer for each command
For the Timer IRQ mode (i.e., when command completions are delayed),
there is one timer for each CPU. Each of these timers
. has a completion queue associated with it, containing all the
command completions to be executed when the timer fires;
. is set, and a new completion-to-execute is inserted into its
completion queue, every time the dispatch code for a new command
happens to be executed on the CPU related to the timer.
This implies that, if the dispatch of a new command happens to be
executed on a CPU whose timer has already been set, but has not yet
fired, then the timer is set again, to the completion time of the
newly arrived command. When the timer eventually fires, all its queued
completions are executed.
This way of handling delayed command completions entails the following
problem: if more than one command completion is inserted into the
queue of a timer before the timer fires, then the expiration time for
the timer is moved forward every time each of these completions is
enqueued. As a consequence, only the last completion enqueued enjoys a
correct execution time, while all previous completions are unjustly
delayed until the last completion is executed (and at that time they
are executed all together).
Specifically, if all the above completions are enqueued almost at the
same time, then the problem is negligible. On the opposite end, if
every completion is enqueued a while after the previous completion was
enqueued (in the extreme case, it is enqueued only right before the
timer would have expired), then every enqueued completion, except for
the last one, experiences an inflated delay, proportional to the number
of completions enqueued after it. In the end, commands, and thus I/O
requests, may be completed at an arbitrarily lower rate than the
desired one.
This commit addresses this issue by replacing per-CPU timers with
per-command timers, i.e., by associating an individual timer with each
command.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna Avanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:17 +07:00
|
|
|
hrtimer_start(&cmd->timer, kt, HRTIMER_MODE_REL);
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void null_softirq_done_fn(struct request *rq)
|
|
|
|
{
|
2017-08-15 05:04:52 +07:00
|
|
|
struct nullb *nullb = rq->q->queuedata;
|
|
|
|
|
|
|
|
if (nullb->dev->queue_mode == NULL_Q_MQ)
|
2014-06-17 00:40:25 +07:00
|
|
|
end_cmd(blk_mq_rq_to_pdu(rq));
|
|
|
|
else
|
|
|
|
end_cmd(rq->special);
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void null_handle_cmd(struct nullb_cmd *cmd)
|
|
|
|
{
|
|
|
|
/* Complete IO by inline, softirq or timer */
|
2017-08-15 05:04:52 +07:00
|
|
|
switch (cmd->nq->dev->irqmode) {
|
2013-10-25 17:52:25 +07:00
|
|
|
case NULL_IRQ_SOFTIRQ:
|
2017-08-15 05:04:52 +07:00
|
|
|
switch (cmd->nq->dev->queue_mode) {
|
2014-02-10 18:24:40 +07:00
|
|
|
case NULL_Q_MQ:
|
2017-04-20 21:03:09 +07:00
|
|
|
blk_mq_complete_request(cmd->rq);
|
2014-02-10 18:24:40 +07:00
|
|
|
break;
|
|
|
|
case NULL_Q_RQ:
|
|
|
|
blk_complete_request(cmd->rq);
|
|
|
|
break;
|
|
|
|
case NULL_Q_BIO:
|
|
|
|
/*
|
|
|
|
* XXX: no proper submitting cpu information available.
|
|
|
|
*/
|
|
|
|
end_cmd(cmd);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case NULL_IRQ_NONE:
|
2013-10-25 17:52:25 +07:00
|
|
|
end_cmd(cmd);
|
|
|
|
break;
|
|
|
|
case NULL_IRQ_TIMER:
|
|
|
|
null_cmd_end_timer(cmd);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct nullb_queue *nullb_to_queue(struct nullb *nullb)
|
|
|
|
{
|
|
|
|
int index = 0;
|
|
|
|
|
|
|
|
if (nullb->nr_queues != 1)
|
|
|
|
index = raw_smp_processor_id() / ((nr_cpu_ids + nullb->nr_queues - 1) / nullb->nr_queues);
|
|
|
|
|
|
|
|
return &nullb->queues[index];
|
|
|
|
}
|
|
|
|
|
2015-11-06 00:41:16 +07:00
|
|
|
static blk_qc_t null_queue_bio(struct request_queue *q, struct bio *bio)
|
2013-10-25 17:52:25 +07:00
|
|
|
{
|
|
|
|
struct nullb *nullb = q->queuedata;
|
|
|
|
struct nullb_queue *nq = nullb_to_queue(nullb);
|
|
|
|
struct nullb_cmd *cmd;
|
|
|
|
|
|
|
|
cmd = alloc_cmd(nq, 1);
|
|
|
|
cmd->bio = bio;
|
|
|
|
|
|
|
|
null_handle_cmd(cmd);
|
2015-11-06 00:41:16 +07:00
|
|
|
return BLK_QC_T_NONE;
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int null_rq_prep_fn(struct request_queue *q, struct request *req)
|
|
|
|
{
|
|
|
|
struct nullb *nullb = q->queuedata;
|
|
|
|
struct nullb_queue *nq = nullb_to_queue(nullb);
|
|
|
|
struct nullb_cmd *cmd;
|
|
|
|
|
|
|
|
cmd = alloc_cmd(nq, 0);
|
|
|
|
if (cmd) {
|
|
|
|
cmd->rq = req;
|
|
|
|
req->special = cmd;
|
|
|
|
return BLKPREP_OK;
|
|
|
|
}
|
2015-06-02 06:35:10 +07:00
|
|
|
blk_stop_queue(q);
|
2013-10-25 17:52:25 +07:00
|
|
|
|
|
|
|
return BLKPREP_DEFER;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void null_request_fn(struct request_queue *q)
|
|
|
|
{
|
|
|
|
struct request *rq;
|
|
|
|
|
|
|
|
while ((rq = blk_fetch_request(q)) != NULL) {
|
|
|
|
struct nullb_cmd *cmd = rq->special;
|
|
|
|
|
|
|
|
spin_unlock_irq(q->queue_lock);
|
|
|
|
null_handle_cmd(cmd);
|
|
|
|
spin_lock_irq(q->queue_lock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-03 14:38:05 +07:00
|
|
|
static blk_status_t null_queue_rq(struct blk_mq_hw_ctx *hctx,
|
2014-10-30 00:14:52 +07:00
|
|
|
const struct blk_mq_queue_data *bd)
|
2013-10-25 17:52:25 +07:00
|
|
|
{
|
2014-10-30 00:14:52 +07:00
|
|
|
struct nullb_cmd *cmd = blk_mq_rq_to_pdu(bd->rq);
|
2017-08-15 05:04:52 +07:00
|
|
|
struct nullb_queue *nq = hctx->driver_data;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2017-03-31 02:44:26 +07:00
|
|
|
might_sleep_if(hctx->flags & BLK_MQ_F_BLOCKING);
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (nq->dev->irqmode == NULL_IRQ_TIMER) {
|
null_blk: set a separate timer for each command
For the Timer IRQ mode (i.e., when command completions are delayed),
there is one timer for each CPU. Each of these timers
. has a completion queue associated with it, containing all the
command completions to be executed when the timer fires;
. is set, and a new completion-to-execute is inserted into its
completion queue, every time the dispatch code for a new command
happens to be executed on the CPU related to the timer.
This implies that, if the dispatch of a new command happens to be
executed on a CPU whose timer has already been set, but has not yet
fired, then the timer is set again, to the completion time of the
newly arrived command. When the timer eventually fires, all its queued
completions are executed.
This way of handling delayed command completions entails the following
problem: if more than one command completion is inserted into the
queue of a timer before the timer fires, then the expiration time for
the timer is moved forward every time each of these completions is
enqueued. As a consequence, only the last completion enqueued enjoys a
correct execution time, while all previous completions are unjustly
delayed until the last completion is executed (and at that time they
are executed all together).
Specifically, if all the above completions are enqueued almost at the
same time, then the problem is negligible. On the opposite end, if
every completion is enqueued a while after the previous completion was
enqueued (in the extreme case, it is enqueued only right before the
timer would have expired), then every enqueued completion, except for
the last one, experiences an inflated delay, proportional to the number
of completions enqueued after it. In the end, commands, and thus I/O
requests, may be completed at an arbitrarily lower rate than the
desired one.
This commit addresses this issue by replacing per-CPU timers with
per-command timers, i.e., by associating an individual timer with each
command.
Signed-off-by: Paolo Valente <paolo.valente@unimore.it>
Signed-off-by: Arianna Avanzini <avanzini@google.com>
Signed-off-by: Jens Axboe <axboe@fb.com>
2015-12-01 17:48:17 +07:00
|
|
|
hrtimer_init(&cmd->timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
|
|
|
|
cmd->timer.function = null_cmd_timer_expired;
|
|
|
|
}
|
2014-10-30 00:14:52 +07:00
|
|
|
cmd->rq = bd->rq;
|
2017-08-15 05:04:52 +07:00
|
|
|
cmd->nq = nq;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2014-10-30 00:14:52 +07:00
|
|
|
blk_mq_start_request(bd->rq);
|
2014-09-14 06:40:09 +07:00
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
null_handle_cmd(cmd);
|
2017-06-03 14:38:05 +07:00
|
|
|
return BLK_STS_OK;
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
2017-03-31 03:39:16 +07:00
|
|
|
static const struct blk_mq_ops null_mq_ops = {
|
2013-10-25 17:52:25 +07:00
|
|
|
.queue_rq = null_queue_rq,
|
2014-02-10 18:24:40 +07:00
|
|
|
.complete = null_softirq_done_fn,
|
2013-10-25 17:52:25 +07:00
|
|
|
};
|
|
|
|
|
2015-08-31 19:17:18 +07:00
|
|
|
static void cleanup_queue(struct nullb_queue *nq)
|
|
|
|
{
|
|
|
|
kfree(nq->tag_map);
|
|
|
|
kfree(nq->cmds);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cleanup_queues(struct nullb *nullb)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < nullb->nr_queues; i++)
|
|
|
|
cleanup_queue(&nullb->queues[i]);
|
|
|
|
|
|
|
|
kfree(nullb->queues);
|
|
|
|
}
|
|
|
|
|
2015-11-13 02:25:10 +07:00
|
|
|
#ifdef CONFIG_NVM
|
|
|
|
|
2017-06-03 14:38:04 +07:00
|
|
|
static void null_lnvm_end_io(struct request *rq, blk_status_t status)
|
2015-11-13 02:25:10 +07:00
|
|
|
{
|
|
|
|
struct nvm_rq *rqd = rq->end_io_data;
|
|
|
|
|
2017-06-03 14:38:04 +07:00
|
|
|
/* XXX: lighnvm core seems to expect NVM_RSP_* values here.. */
|
|
|
|
rqd->error = status ? -EIO : 0;
|
2017-01-31 19:17:17 +07:00
|
|
|
nvm_end_io(rqd);
|
2015-11-13 02:25:10 +07:00
|
|
|
|
|
|
|
blk_put_request(rq);
|
|
|
|
}
|
|
|
|
|
2015-12-06 17:25:48 +07:00
|
|
|
static int null_lnvm_submit_io(struct nvm_dev *dev, struct nvm_rq *rqd)
|
2015-11-13 02:25:10 +07:00
|
|
|
{
|
2015-12-06 17:25:48 +07:00
|
|
|
struct request_queue *q = dev->q;
|
2015-11-13 02:25:10 +07:00
|
|
|
struct request *rq;
|
|
|
|
struct bio *bio = rqd->bio;
|
|
|
|
|
2017-01-31 22:57:31 +07:00
|
|
|
rq = blk_mq_alloc_request(q,
|
|
|
|
op_is_write(bio_op(bio)) ? REQ_OP_DRV_OUT : REQ_OP_DRV_IN, 0);
|
2015-11-13 02:25:10 +07:00
|
|
|
if (IS_ERR(rq))
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2017-04-20 04:01:25 +07:00
|
|
|
blk_init_request_from_bio(rq, bio);
|
2015-11-13 02:25:10 +07:00
|
|
|
|
|
|
|
rq->end_io_data = rqd;
|
|
|
|
|
|
|
|
blk_execute_rq_nowait(q, NULL, rq, 0, null_lnvm_end_io);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-12-06 17:25:48 +07:00
|
|
|
static int null_lnvm_id(struct nvm_dev *dev, struct nvm_id *id)
|
2015-11-13 02:25:10 +07:00
|
|
|
{
|
2017-08-15 05:04:52 +07:00
|
|
|
struct nullb *nullb = dev->q->queuedata;
|
|
|
|
sector_t size = (sector_t)nullb->dev->size * 1024 * 1024ULL;
|
2015-11-19 18:50:09 +07:00
|
|
|
sector_t blksize;
|
2015-11-13 02:25:10 +07:00
|
|
|
struct nvm_id_group *grp;
|
|
|
|
|
|
|
|
id->ver_id = 0x1;
|
|
|
|
id->vmnt = 0;
|
2016-02-04 21:13:27 +07:00
|
|
|
id->cap = 0x2;
|
2015-11-13 02:25:10 +07:00
|
|
|
id->dom = 0x1;
|
2015-11-19 18:50:09 +07:00
|
|
|
|
|
|
|
id->ppaf.blk_offset = 0;
|
|
|
|
id->ppaf.blk_len = 16;
|
|
|
|
id->ppaf.pg_offset = 16;
|
|
|
|
id->ppaf.pg_len = 16;
|
|
|
|
id->ppaf.sect_offset = 32;
|
|
|
|
id->ppaf.sect_len = 8;
|
|
|
|
id->ppaf.pln_offset = 40;
|
|
|
|
id->ppaf.pln_len = 8;
|
|
|
|
id->ppaf.lun_offset = 48;
|
|
|
|
id->ppaf.lun_len = 8;
|
|
|
|
id->ppaf.ch_offset = 56;
|
|
|
|
id->ppaf.ch_len = 8;
|
2015-11-13 02:25:10 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
sector_div(size, nullb->dev->blocksize); /* convert size to pages */
|
2016-01-14 05:04:08 +07:00
|
|
|
size >>= 8; /* concert size to pgs pr blk */
|
2017-01-31 19:17:15 +07:00
|
|
|
grp = &id->grp;
|
2015-11-13 02:25:10 +07:00
|
|
|
grp->mtype = 0;
|
2015-11-19 18:50:09 +07:00
|
|
|
grp->fmtype = 0;
|
2015-11-13 02:25:10 +07:00
|
|
|
grp->num_ch = 1;
|
|
|
|
grp->num_pg = 256;
|
2015-11-19 18:50:09 +07:00
|
|
|
blksize = size;
|
2016-01-14 05:04:08 +07:00
|
|
|
size >>= 16;
|
2015-11-19 18:50:09 +07:00
|
|
|
grp->num_lun = size + 1;
|
2016-01-14 05:04:08 +07:00
|
|
|
sector_div(blksize, grp->num_lun);
|
2015-11-19 18:50:09 +07:00
|
|
|
grp->num_blk = blksize;
|
|
|
|
grp->num_pln = 1;
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
grp->fpg_sz = nullb->dev->blocksize;
|
|
|
|
grp->csecs = nullb->dev->blocksize;
|
2015-11-13 02:25:10 +07:00
|
|
|
grp->trdt = 25000;
|
|
|
|
grp->trdm = 25000;
|
|
|
|
grp->tprt = 500000;
|
|
|
|
grp->tprm = 500000;
|
|
|
|
grp->tbet = 1500000;
|
|
|
|
grp->tbem = 1500000;
|
|
|
|
grp->mpos = 0x010101; /* single plane rwe */
|
2017-08-15 05:04:52 +07:00
|
|
|
grp->cpar = nullb->dev->hw_queue_depth;
|
2015-11-13 02:25:10 +07:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-12-06 17:25:48 +07:00
|
|
|
static void *null_lnvm_create_dma_pool(struct nvm_dev *dev, char *name)
|
2015-11-13 02:25:10 +07:00
|
|
|
{
|
|
|
|
mempool_t *virtmem_pool;
|
|
|
|
|
2015-11-19 18:50:08 +07:00
|
|
|
virtmem_pool = mempool_create_slab_pool(64, ppa_cache);
|
2015-11-13 02:25:10 +07:00
|
|
|
if (!virtmem_pool) {
|
|
|
|
pr_err("null_blk: Unable to create virtual memory pool\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return virtmem_pool;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void null_lnvm_destroy_dma_pool(void *pool)
|
|
|
|
{
|
|
|
|
mempool_destroy(pool);
|
|
|
|
}
|
|
|
|
|
2015-12-06 17:25:48 +07:00
|
|
|
static void *null_lnvm_dev_dma_alloc(struct nvm_dev *dev, void *pool,
|
2015-11-13 02:25:10 +07:00
|
|
|
gfp_t mem_flags, dma_addr_t *dma_handler)
|
|
|
|
{
|
|
|
|
return mempool_alloc(pool, mem_flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void null_lnvm_dev_dma_free(void *pool, void *entry,
|
|
|
|
dma_addr_t dma_handler)
|
|
|
|
{
|
|
|
|
mempool_free(entry, pool);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct nvm_dev_ops null_lnvm_dev_ops = {
|
|
|
|
.identity = null_lnvm_id,
|
|
|
|
.submit_io = null_lnvm_submit_io,
|
|
|
|
|
|
|
|
.create_dma_pool = null_lnvm_create_dma_pool,
|
|
|
|
.destroy_dma_pool = null_lnvm_destroy_dma_pool,
|
|
|
|
.dev_dma_alloc = null_lnvm_dev_dma_alloc,
|
|
|
|
.dev_dma_free = null_lnvm_dev_dma_free,
|
|
|
|
|
|
|
|
/* Simulate nvme protocol restriction */
|
|
|
|
.max_phys_sect = 64,
|
|
|
|
};
|
2016-09-16 19:25:05 +07:00
|
|
|
|
|
|
|
static int null_nvm_register(struct nullb *nullb)
|
|
|
|
{
|
2016-09-16 19:25:07 +07:00
|
|
|
struct nvm_dev *dev;
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
dev = nvm_alloc_dev(0);
|
|
|
|
if (!dev)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
dev->q = nullb->q;
|
|
|
|
memcpy(dev->name, nullb->disk_name, DISK_NAME_LEN);
|
|
|
|
dev->ops = &null_lnvm_dev_ops;
|
|
|
|
|
|
|
|
rv = nvm_register(dev);
|
|
|
|
if (rv) {
|
|
|
|
kfree(dev);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
nullb->ndev = dev;
|
|
|
|
return 0;
|
2016-09-16 19:25:05 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void null_nvm_unregister(struct nullb *nullb)
|
|
|
|
{
|
2016-09-16 19:25:07 +07:00
|
|
|
nvm_unregister(nullb->ndev);
|
2016-09-16 19:25:05 +07:00
|
|
|
}
|
2015-11-13 02:25:10 +07:00
|
|
|
#else
|
2016-09-16 19:25:05 +07:00
|
|
|
static int null_nvm_register(struct nullb *nullb)
|
|
|
|
{
|
2016-11-16 22:26:11 +07:00
|
|
|
pr_err("null_blk: CONFIG_NVM needs to be enabled for LightNVM\n");
|
2016-09-16 19:25:05 +07:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
static void null_nvm_unregister(struct nullb *nullb) {}
|
2015-11-13 02:25:10 +07:00
|
|
|
#endif /* CONFIG_NVM */
|
|
|
|
|
2016-09-16 19:25:05 +07:00
|
|
|
static void null_del_dev(struct nullb *nullb)
|
|
|
|
{
|
2017-08-15 05:04:52 +07:00
|
|
|
struct nullb_device *dev = nullb->dev;
|
|
|
|
|
2016-09-16 19:25:05 +07:00
|
|
|
list_del_init(&nullb->list);
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (dev->use_lightnvm)
|
2016-09-16 19:25:05 +07:00
|
|
|
null_nvm_unregister(nullb);
|
|
|
|
else
|
|
|
|
del_gendisk(nullb->disk);
|
|
|
|
blk_cleanup_queue(nullb->q);
|
2017-08-15 05:04:52 +07:00
|
|
|
if (dev->queue_mode == NULL_Q_MQ &&
|
|
|
|
nullb->tag_set == &nullb->__tag_set)
|
2017-06-21 03:22:01 +07:00
|
|
|
blk_mq_free_tag_set(nullb->tag_set);
|
2017-08-15 05:04:52 +07:00
|
|
|
if (!dev->use_lightnvm)
|
2016-09-16 19:25:05 +07:00
|
|
|
put_disk(nullb->disk);
|
|
|
|
cleanup_queues(nullb);
|
|
|
|
kfree(nullb);
|
2017-08-15 05:04:52 +07:00
|
|
|
dev->nullb = NULL;
|
2016-09-16 19:25:05 +07:00
|
|
|
}
|
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
static int null_open(struct block_device *bdev, fmode_t mode)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void null_release(struct gendisk *disk, fmode_t mode)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct block_device_operations null_fops = {
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.open = null_open,
|
|
|
|
.release = null_release,
|
|
|
|
};
|
|
|
|
|
2017-06-21 03:22:01 +07:00
|
|
|
static void null_init_queue(struct nullb *nullb, struct nullb_queue *nq)
|
|
|
|
{
|
|
|
|
BUG_ON(!nullb);
|
|
|
|
BUG_ON(!nq);
|
|
|
|
|
|
|
|
init_waitqueue_head(&nq->wait);
|
|
|
|
nq->queue_depth = nullb->queue_depth;
|
2017-08-15 05:04:52 +07:00
|
|
|
nq->dev = nullb->dev;
|
2017-06-21 03:22:01 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void null_init_queues(struct nullb *nullb)
|
|
|
|
{
|
|
|
|
struct request_queue *q = nullb->q;
|
|
|
|
struct blk_mq_hw_ctx *hctx;
|
|
|
|
struct nullb_queue *nq;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
queue_for_each_hw_ctx(q, hctx, i) {
|
|
|
|
if (!hctx->nr_ctx || !hctx->tags)
|
|
|
|
continue;
|
|
|
|
nq = &nullb->queues[i];
|
|
|
|
hctx->driver_data = nq;
|
|
|
|
null_init_queue(nullb, nq);
|
|
|
|
nullb->nr_queues++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
static int setup_commands(struct nullb_queue *nq)
|
|
|
|
{
|
|
|
|
struct nullb_cmd *cmd;
|
|
|
|
int i, tag_size;
|
|
|
|
|
|
|
|
nq->cmds = kzalloc(nq->queue_depth * sizeof(*cmd), GFP_KERNEL);
|
|
|
|
if (!nq->cmds)
|
2013-12-18 19:41:43 +07:00
|
|
|
return -ENOMEM;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
|
|
|
tag_size = ALIGN(nq->queue_depth, BITS_PER_LONG) / BITS_PER_LONG;
|
|
|
|
nq->tag_map = kzalloc(tag_size * sizeof(unsigned long), GFP_KERNEL);
|
|
|
|
if (!nq->tag_map) {
|
|
|
|
kfree(nq->cmds);
|
2013-12-18 19:41:43 +07:00
|
|
|
return -ENOMEM;
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < nq->queue_depth; i++) {
|
|
|
|
cmd = &nq->cmds[i];
|
|
|
|
INIT_LIST_HEAD(&cmd->list);
|
|
|
|
cmd->ll_list.next = NULL;
|
|
|
|
cmd->tag = -1U;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int setup_queues(struct nullb *nullb)
|
|
|
|
{
|
2017-08-15 05:04:52 +07:00
|
|
|
nullb->queues = kzalloc(nullb->dev->submit_queues *
|
|
|
|
sizeof(struct nullb_queue), GFP_KERNEL);
|
2013-10-25 17:52:25 +07:00
|
|
|
if (!nullb->queues)
|
2013-12-18 19:41:43 +07:00
|
|
|
return -ENOMEM;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
|
|
|
nullb->nr_queues = 0;
|
2017-08-15 05:04:52 +07:00
|
|
|
nullb->queue_depth = nullb->dev->hw_queue_depth;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2013-12-18 19:41:43 +07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int init_driver_queues(struct nullb *nullb)
|
|
|
|
{
|
|
|
|
struct nullb_queue *nq;
|
|
|
|
int i, ret = 0;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
for (i = 0; i < nullb->dev->submit_queues; i++) {
|
2013-10-25 17:52:25 +07:00
|
|
|
nq = &nullb->queues[i];
|
2013-12-18 19:41:43 +07:00
|
|
|
|
|
|
|
null_init_queue(nullb, nq);
|
|
|
|
|
|
|
|
ret = setup_commands(nq);
|
|
|
|
if (ret)
|
2014-10-22 20:34:21 +07:00
|
|
|
return ret;
|
2013-10-25 17:52:25 +07:00
|
|
|
nullb->nr_queues++;
|
|
|
|
}
|
2013-12-18 19:41:43 +07:00
|
|
|
return 0;
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
2016-09-16 19:25:05 +07:00
|
|
|
static int null_gendisk_register(struct nullb *nullb)
|
2013-10-25 17:52:25 +07:00
|
|
|
{
|
|
|
|
struct gendisk *disk;
|
|
|
|
sector_t size;
|
2016-09-16 19:25:05 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
disk = nullb->disk = alloc_disk_node(1, nullb->dev->home_node);
|
2016-09-16 19:25:05 +07:00
|
|
|
if (!disk)
|
|
|
|
return -ENOMEM;
|
2017-08-15 05:04:52 +07:00
|
|
|
size = (sector_t)nullb->dev->size * 1024 * 1024ULL;
|
2016-09-16 19:25:05 +07:00
|
|
|
set_capacity(disk, size >> 9);
|
|
|
|
|
|
|
|
disk->flags |= GENHD_FL_EXT_DEVT | GENHD_FL_SUPPRESS_PARTITION_INFO;
|
|
|
|
disk->major = null_major;
|
|
|
|
disk->first_minor = nullb->index;
|
|
|
|
disk->fops = &null_fops;
|
|
|
|
disk->private_data = nullb;
|
|
|
|
disk->queue = nullb->q;
|
|
|
|
strncpy(disk->disk_name, nullb->disk_name, DISK_NAME_LEN);
|
|
|
|
|
|
|
|
add_disk(disk);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static int null_init_tag_set(struct nullb *nullb, struct blk_mq_tag_set *set)
|
2017-06-21 03:22:01 +07:00
|
|
|
{
|
|
|
|
set->ops = &null_mq_ops;
|
2017-08-15 05:04:52 +07:00
|
|
|
set->nr_hw_queues = nullb ? nullb->dev->submit_queues :
|
|
|
|
g_submit_queues;
|
|
|
|
set->queue_depth = nullb ? nullb->dev->hw_queue_depth :
|
|
|
|
g_hw_queue_depth;
|
|
|
|
set->numa_node = nullb ? nullb->dev->home_node : g_home_node;
|
2017-06-21 03:22:01 +07:00
|
|
|
set->cmd_size = sizeof(struct nullb_cmd);
|
|
|
|
set->flags = BLK_MQ_F_SHOULD_MERGE;
|
|
|
|
set->driver_data = NULL;
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (nullb->dev->blocking)
|
2017-06-21 03:22:01 +07:00
|
|
|
set->flags |= BLK_MQ_F_BLOCKING;
|
|
|
|
|
|
|
|
return blk_mq_alloc_tag_set(set);
|
|
|
|
}
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
static int null_add_dev(struct nullb_device *dev)
|
2016-09-16 19:25:05 +07:00
|
|
|
{
|
|
|
|
struct nullb *nullb;
|
2014-09-02 23:38:49 +07:00
|
|
|
int rv;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
nullb = kzalloc_node(sizeof(*nullb), GFP_KERNEL, dev->home_node);
|
2014-09-02 23:38:49 +07:00
|
|
|
if (!nullb) {
|
|
|
|
rv = -ENOMEM;
|
2014-04-16 03:14:00 +07:00
|
|
|
goto out;
|
2014-09-02 23:38:49 +07:00
|
|
|
}
|
2017-08-15 05:04:52 +07:00
|
|
|
nullb->dev = dev;
|
|
|
|
dev->nullb = nullb;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
|
|
|
spin_lock_init(&nullb->lock);
|
|
|
|
|
2014-09-02 23:38:49 +07:00
|
|
|
rv = setup_queues(nullb);
|
|
|
|
if (rv)
|
2014-04-16 03:14:00 +07:00
|
|
|
goto out_free_nullb;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (dev->queue_mode == NULL_Q_MQ) {
|
2017-06-21 03:22:01 +07:00
|
|
|
if (shared_tags) {
|
|
|
|
nullb->tag_set = &tag_set;
|
|
|
|
rv = 0;
|
|
|
|
} else {
|
|
|
|
nullb->tag_set = &nullb->__tag_set;
|
2017-08-15 05:04:52 +07:00
|
|
|
rv = null_init_tag_set(nullb, nullb->tag_set);
|
2017-06-21 03:22:01 +07:00
|
|
|
}
|
|
|
|
|
2014-09-02 23:38:49 +07:00
|
|
|
if (rv)
|
2014-04-16 03:14:00 +07:00
|
|
|
goto out_cleanup_queues;
|
|
|
|
|
2017-06-21 03:22:01 +07:00
|
|
|
nullb->q = blk_mq_init_queue(nullb->tag_set);
|
2015-01-02 21:25:27 +07:00
|
|
|
if (IS_ERR(nullb->q)) {
|
2014-09-02 23:38:49 +07:00
|
|
|
rv = -ENOMEM;
|
2014-04-16 03:14:00 +07:00
|
|
|
goto out_cleanup_tags;
|
2014-09-02 23:38:49 +07:00
|
|
|
}
|
2017-06-21 03:22:01 +07:00
|
|
|
null_init_queues(nullb);
|
2017-08-15 05:04:52 +07:00
|
|
|
} else if (dev->queue_mode == NULL_Q_BIO) {
|
|
|
|
nullb->q = blk_alloc_queue_node(GFP_KERNEL, dev->home_node);
|
2014-09-02 23:38:49 +07:00
|
|
|
if (!nullb->q) {
|
|
|
|
rv = -ENOMEM;
|
2014-04-16 03:14:00 +07:00
|
|
|
goto out_cleanup_queues;
|
2014-09-02 23:38:49 +07:00
|
|
|
}
|
2013-10-25 17:52:25 +07:00
|
|
|
blk_queue_make_request(nullb->q, null_queue_bio);
|
2014-10-22 20:34:21 +07:00
|
|
|
rv = init_driver_queues(nullb);
|
|
|
|
if (rv)
|
|
|
|
goto out_cleanup_blk_queue;
|
2013-10-25 17:52:25 +07:00
|
|
|
} else {
|
2017-08-15 05:04:52 +07:00
|
|
|
nullb->q = blk_init_queue_node(null_request_fn, &nullb->lock,
|
|
|
|
dev->home_node);
|
2014-09-02 23:38:49 +07:00
|
|
|
if (!nullb->q) {
|
|
|
|
rv = -ENOMEM;
|
2014-04-16 03:14:00 +07:00
|
|
|
goto out_cleanup_queues;
|
2014-09-02 23:38:49 +07:00
|
|
|
}
|
2013-10-25 17:52:25 +07:00
|
|
|
blk_queue_prep_rq(nullb->q, null_rq_prep_fn);
|
2014-04-16 03:14:00 +07:00
|
|
|
blk_queue_softirq_done(nullb->q, null_softirq_done_fn);
|
2014-10-22 20:34:21 +07:00
|
|
|
rv = init_driver_queues(nullb);
|
|
|
|
if (rv)
|
|
|
|
goto out_cleanup_blk_queue;
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
nullb->q->queuedata = nullb;
|
|
|
|
queue_flag_set_unlocked(QUEUE_FLAG_NONROT, nullb->q);
|
2014-10-04 23:55:32 +07:00
|
|
|
queue_flag_clear_unlocked(QUEUE_FLAG_ADD_RANDOM, nullb->q);
|
2013-10-25 17:52:25 +07:00
|
|
|
|
|
|
|
mutex_lock(&lock);
|
|
|
|
nullb->index = nullb_indexes++;
|
|
|
|
mutex_unlock(&lock);
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
blk_queue_logical_block_size(nullb->q, dev->blocksize);
|
|
|
|
blk_queue_physical_block_size(nullb->q, dev->blocksize);
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2015-11-13 02:25:10 +07:00
|
|
|
sprintf(nullb->disk_name, "nullb%d", nullb->index);
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (dev->use_lightnvm)
|
2016-09-16 19:25:05 +07:00
|
|
|
rv = null_nvm_register(nullb);
|
|
|
|
else
|
|
|
|
rv = null_gendisk_register(nullb);
|
2015-11-13 02:25:10 +07:00
|
|
|
|
2016-09-16 19:25:05 +07:00
|
|
|
if (rv)
|
|
|
|
goto out_cleanup_blk_queue;
|
2016-02-11 20:49:13 +07:00
|
|
|
|
|
|
|
mutex_lock(&lock);
|
|
|
|
list_add_tail(&nullb->list, &nullb_list);
|
|
|
|
mutex_unlock(&lock);
|
2016-03-04 23:27:04 +07:00
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
return 0;
|
2014-04-16 03:14:00 +07:00
|
|
|
out_cleanup_blk_queue:
|
|
|
|
blk_cleanup_queue(nullb->q);
|
|
|
|
out_cleanup_tags:
|
2017-08-15 05:04:52 +07:00
|
|
|
if (dev->queue_mode == NULL_Q_MQ && nullb->tag_set == &nullb->__tag_set)
|
2017-06-21 03:22:01 +07:00
|
|
|
blk_mq_free_tag_set(nullb->tag_set);
|
2014-04-16 03:14:00 +07:00
|
|
|
out_cleanup_queues:
|
|
|
|
cleanup_queues(nullb);
|
|
|
|
out_free_nullb:
|
|
|
|
kfree(nullb);
|
|
|
|
out:
|
2017-08-15 05:04:52 +07:00
|
|
|
null_free_dev(dev);
|
2014-09-02 23:38:49 +07:00
|
|
|
return rv;
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static int __init null_init(void)
|
|
|
|
{
|
2015-12-09 03:47:34 +07:00
|
|
|
int ret = 0;
|
2013-10-25 17:52:25 +07:00
|
|
|
unsigned int i;
|
2015-12-09 03:47:34 +07:00
|
|
|
struct nullb *nullb;
|
2017-08-15 05:04:52 +07:00
|
|
|
struct nullb_device *dev;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (g_bs > PAGE_SIZE) {
|
2014-01-21 18:29:59 +07:00
|
|
|
pr_warn("null_blk: invalid block size\n");
|
|
|
|
pr_warn("null_blk: defaults block size to %lu\n", PAGE_SIZE);
|
2017-08-15 05:04:52 +07:00
|
|
|
g_bs = PAGE_SIZE;
|
2014-01-21 18:29:59 +07:00
|
|
|
}
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (g_use_lightnvm && g_bs != 4096) {
|
2015-11-19 18:50:08 +07:00
|
|
|
pr_warn("null_blk: LightNVM only supports 4k block size\n");
|
|
|
|
pr_warn("null_blk: defaults block size to 4k\n");
|
2017-08-15 05:04:52 +07:00
|
|
|
g_bs = 4096;
|
2015-11-19 18:50:08 +07:00
|
|
|
}
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (g_use_lightnvm && g_queue_mode != NULL_Q_MQ) {
|
2015-11-13 02:25:10 +07:00
|
|
|
pr_warn("null_blk: LightNVM only supported for blk-mq\n");
|
|
|
|
pr_warn("null_blk: defaults queue mode to blk-mq\n");
|
2017-08-15 05:04:52 +07:00
|
|
|
g_queue_mode = NULL_Q_MQ;
|
2015-11-13 02:25:10 +07:00
|
|
|
}
|
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (g_queue_mode == NULL_Q_MQ && g_use_per_node_hctx) {
|
|
|
|
if (g_submit_queues != nr_online_nodes) {
|
2017-08-02 23:26:39 +07:00
|
|
|
pr_warn("null_blk: submit_queues param is set to %u.\n",
|
2013-12-18 19:41:44 +07:00
|
|
|
nr_online_nodes);
|
2017-08-15 05:04:52 +07:00
|
|
|
g_submit_queues = nr_online_nodes;
|
2013-12-21 06:11:01 +07:00
|
|
|
}
|
2017-08-15 05:04:52 +07:00
|
|
|
} else if (g_submit_queues > nr_cpu_ids)
|
|
|
|
g_submit_queues = nr_cpu_ids;
|
|
|
|
else if (g_submit_queues <= 0)
|
|
|
|
g_submit_queues = 1;
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (g_queue_mode == NULL_Q_MQ && shared_tags) {
|
|
|
|
ret = null_init_tag_set(NULL, &tag_set);
|
2017-07-06 22:00:07 +07:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-08-15 05:04:53 +07:00
|
|
|
config_group_init(&nullb_subsys.su_group);
|
|
|
|
mutex_init(&nullb_subsys.su_mutex);
|
|
|
|
|
|
|
|
ret = configfs_register_subsystem(&nullb_subsys);
|
|
|
|
if (ret)
|
|
|
|
goto err_tagset;
|
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
mutex_init(&lock);
|
|
|
|
|
|
|
|
null_major = register_blkdev(0, "nullb");
|
2017-07-06 22:00:07 +07:00
|
|
|
if (null_major < 0) {
|
|
|
|
ret = null_major;
|
2017-08-15 05:04:53 +07:00
|
|
|
goto err_conf;
|
2017-07-06 22:00:07 +07:00
|
|
|
}
|
2013-10-25 17:52:25 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (g_use_lightnvm) {
|
2015-11-19 18:50:08 +07:00
|
|
|
ppa_cache = kmem_cache_create("ppa_cache", 64 * sizeof(u64),
|
|
|
|
0, 0, NULL);
|
|
|
|
if (!ppa_cache) {
|
|
|
|
pr_err("null_blk: unable to create ppa cache\n");
|
2015-12-09 03:47:34 +07:00
|
|
|
ret = -ENOMEM;
|
|
|
|
goto err_ppa;
|
2015-11-19 18:50:08 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
for (i = 0; i < nr_devices; i++) {
|
2017-08-15 05:04:52 +07:00
|
|
|
dev = null_alloc_dev();
|
|
|
|
if (!dev)
|
|
|
|
goto err_dev;
|
|
|
|
ret = null_add_dev(dev);
|
|
|
|
if (ret) {
|
|
|
|
null_free_dev(dev);
|
2015-12-09 03:47:34 +07:00
|
|
|
goto err_dev;
|
2017-08-15 05:04:52 +07:00
|
|
|
}
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
pr_info("null: module loaded\n");
|
|
|
|
return 0;
|
2015-12-09 03:47:34 +07:00
|
|
|
|
|
|
|
err_dev:
|
|
|
|
while (!list_empty(&nullb_list)) {
|
|
|
|
nullb = list_entry(nullb_list.next, struct nullb, list);
|
2017-08-15 05:04:52 +07:00
|
|
|
dev = nullb->dev;
|
2015-12-09 03:47:34 +07:00
|
|
|
null_del_dev(nullb);
|
2017-08-15 05:04:52 +07:00
|
|
|
null_free_dev(dev);
|
2015-12-09 03:47:34 +07:00
|
|
|
}
|
2015-11-19 18:50:08 +07:00
|
|
|
kmem_cache_destroy(ppa_cache);
|
2015-12-09 03:47:34 +07:00
|
|
|
err_ppa:
|
|
|
|
unregister_blkdev(null_major, "nullb");
|
2017-08-15 05:04:53 +07:00
|
|
|
err_conf:
|
|
|
|
configfs_unregister_subsystem(&nullb_subsys);
|
2017-07-06 22:00:07 +07:00
|
|
|
err_tagset:
|
2017-08-15 05:04:52 +07:00
|
|
|
if (g_queue_mode == NULL_Q_MQ && shared_tags)
|
2017-07-06 22:00:07 +07:00
|
|
|
blk_mq_free_tag_set(&tag_set);
|
2015-12-09 03:47:34 +07:00
|
|
|
return ret;
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit null_exit(void)
|
|
|
|
{
|
|
|
|
struct nullb *nullb;
|
|
|
|
|
2017-08-15 05:04:53 +07:00
|
|
|
configfs_unregister_subsystem(&nullb_subsys);
|
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
unregister_blkdev(null_major, "nullb");
|
|
|
|
|
|
|
|
mutex_lock(&lock);
|
|
|
|
while (!list_empty(&nullb_list)) {
|
2017-08-15 05:04:52 +07:00
|
|
|
struct nullb_device *dev;
|
|
|
|
|
2013-10-25 17:52:25 +07:00
|
|
|
nullb = list_entry(nullb_list.next, struct nullb, list);
|
2017-08-15 05:04:52 +07:00
|
|
|
dev = nullb->dev;
|
2013-10-25 17:52:25 +07:00
|
|
|
null_del_dev(nullb);
|
2017-08-15 05:04:52 +07:00
|
|
|
null_free_dev(dev);
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
mutex_unlock(&lock);
|
2015-11-19 18:50:08 +07:00
|
|
|
|
2017-08-15 05:04:52 +07:00
|
|
|
if (g_queue_mode == NULL_Q_MQ && shared_tags)
|
2017-06-21 03:22:01 +07:00
|
|
|
blk_mq_free_tag_set(&tag_set);
|
|
|
|
|
2015-11-19 18:50:08 +07:00
|
|
|
kmem_cache_destroy(ppa_cache);
|
2013-10-25 17:52:25 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
module_init(null_init);
|
|
|
|
module_exit(null_exit);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Jens Axboe <jaxboe@fusionio.com>");
|
|
|
|
MODULE_LICENSE("GPL");
|