Release 4.7 drivers/block/nbd.c
/*
* Network block device - make block devices work over TCP
*
* Note that you can not swap over this thing, yet. Seems to work but
* deadlocks sometimes - you can not swap over TCP in general.
*
* Copyright 1997-2000, 2008 Pavel Machek <pavel@ucw.cz>
* Parts copyright 2001 Steven Whitehouse <steve@chygwyn.com>
*
* This file is released under GPLv2 or later.
*
* (part of code stolen from loop.c)
*/
#include <linux/major.h>
#include <linux/blkdev.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/sched.h>
#include <linux/fs.h>
#include <linux/bio.h>
#include <linux/stat.h>
#include <linux/errno.h>
#include <linux/file.h>
#include <linux/ioctl.h>
#include <linux/mutex.h>
#include <linux/compiler.h>
#include <linux/err.h>
#include <linux/kernel.h>
#include <linux/slab.h>
#include <net/sock.h>
#include <linux/net.h>
#include <linux/kthread.h>
#include <linux/types.h>
#include <linux/debugfs.h>
#include <asm/uaccess.h>
#include <asm/types.h>
#include <linux/nbd.h>
struct nbd_device {
u32 flags;
struct socket * sock; /* If == NULL, device is not ready, yet */
int magic;
spinlock_t queue_lock;
struct list_head queue_head; /* Requests waiting result */
struct request *active_req;
wait_queue_head_t active_wq;
struct list_head waiting_queue; /* Requests to be sent */
wait_queue_head_t waiting_wq;
struct mutex tx_lock;
struct gendisk *disk;
int blksize;
loff_t bytesize;
int xmit_timeout;
bool timedout;
bool disconnect; /* a disconnect has been requested by user */
struct timer_list timeout_timer;
/* protects initialization and shutdown of the socket */
spinlock_t sock_lock;
struct task_struct *task_recv;
struct task_struct *task_send;
#if IS_ENABLED(CONFIG_DEBUG_FS)
struct dentry *dbg_dir;
#endif
};
#if IS_ENABLED(CONFIG_DEBUG_FS)
static struct dentry *nbd_dbg_dir;
#endif
#define nbd_name(nbd) ((nbd)->disk->disk_name)
#define NBD_MAGIC 0x68797548
static unsigned int nbds_max = 16;
static struct nbd_device *nbd_dev;
static int max_part;
/*
* Use just one lock (or at most 1 per NIC). Two arguments for this:
* 1. Each NIC is essentially a synchronization point for all servers
* accessed through that NIC so there's no need to have more locks
* than NICs anyway.
* 2. More locks lead to more "Dirty cache line bouncing" which will slow
* down each lock to the point where they're actually slower than just
* a single lock.
* Thanks go to Jens Axboe and Al Viro for their LKML emails explaining this!
*/
static DEFINE_SPINLOCK(nbd_lock);
static inline struct device *nbd_to_dev(struct nbd_device *nbd)
{
return disk_to_dev(nbd->disk);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 14 | 63.64% | 1 | 50.00% |
andrew morton | andrew morton | 8 | 36.36% | 1 | 50.00% |
| Total | 22 | 100.00% | 2 | 100.00% |
static bool nbd_is_connected(struct nbd_device *nbd)
{
return !!nbd->task_recv;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 18 | 100.00% | 1 | 100.00% |
| Total | 18 | 100.00% | 1 | 100.00% |
static const char *nbdcmd_to_ascii(int cmd)
{
switch (cmd) {
case NBD_CMD_READ: return "read";
case NBD_CMD_WRITE: return "write";
case NBD_CMD_DISC: return "disconnect";
case NBD_CMD_FLUSH: return "flush";
case NBD_CMD_TRIM: return "trim/discard";
}
return "invalid";
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
andrew morton | andrew morton | 38 | 76.00% | 1 | 33.33% |
paul clements | paul clements | 6 | 12.00% | 1 | 33.33% |
alex bligh | alex bligh | 6 | 12.00% | 1 | 33.33% |
| Total | 50 | 100.00% | 3 | 100.00% |
static int nbd_size_clear(struct nbd_device *nbd, struct block_device *bdev)
{
bdev->bd_inode->i_size = 0;
set_capacity(nbd->disk, 0);
kobject_uevent(&nbd_to_dev(nbd)->kobj, KOBJ_CHANGE);
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 49 | 100.00% | 1 | 100.00% |
| Total | 49 | 100.00% | 1 | 100.00% |
static void nbd_size_update(struct nbd_device *nbd, struct block_device *bdev)
{
if (!nbd_is_connected(nbd))
return;
bdev->bd_inode->i_size = nbd->bytesize;
set_capacity(nbd->disk, nbd->bytesize >> 9);
kobject_uevent(&nbd_to_dev(nbd)->kobj, KOBJ_CHANGE);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 61 | 100.00% | 1 | 100.00% |
| Total | 61 | 100.00% | 1 | 100.00% |
static int nbd_size_set(struct nbd_device *nbd, struct block_device *bdev,
int blocksize, int nr_blocks)
{
int ret;
ret = set_blocksize(bdev, blocksize);
if (ret)
return ret;
nbd->blksize = blocksize;
nbd->bytesize = (loff_t)blocksize * (loff_t)nr_blocks;
nbd_size_update(nbd, bdev);
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 71 | 100.00% | 1 | 100.00% |
| Total | 71 | 100.00% | 1 | 100.00% |
static void nbd_end_request(struct nbd_device *nbd, struct request *req)
{
int error = req->errors ? -EIO : 0;
struct request_queue *q = req->q;
unsigned long flags;
dev_dbg(nbd_to_dev(nbd), "request %p: %s\n", req,
error ? "failed" : "done");
spin_lock_irqsave(q->queue_lock, flags);
__blk_end_request_all(req, error);
spin_unlock_irqrestore(q->queue_lock, flags);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
christoph hellwig | christoph hellwig | 54 | 64.29% | 1 | 16.67% |
markus pargmann | markus pargmann | 11 | 13.10% | 1 | 16.67% |
andrew morton | andrew morton | 9 | 10.71% | 1 | 16.67% |
kiyoshi ueda | kiyoshi ueda | 7 | 8.33% | 1 | 16.67% |
jens axboe | jens axboe | 2 | 2.38% | 1 | 16.67% |
tejun heo | tejun heo | 1 | 1.19% | 1 | 16.67% |
| Total | 84 | 100.00% | 6 | 100.00% |
/*
* Forcibly shutdown the socket causing all listeners to error
*/
static void sock_shutdown(struct nbd_device *nbd)
{
spin_lock_irq(&nbd->sock_lock);
if (!nbd->sock) {
spin_unlock_irq(&nbd->sock_lock);
return;
}
dev_warn(disk_to_dev(nbd->disk), "shutting down socket\n");
kernel_sock_shutdown(nbd->sock, SHUT_RDWR);
sockfd_put(nbd->sock);
nbd->sock = NULL;
spin_unlock_irq(&nbd->sock_lock);
del_timer(&nbd->timeout_timer);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 43 | 49.43% | 3 | 42.86% |
paul clements | paul clements | 31 | 35.63% | 1 | 14.29% |
americo wang | americo wang | 6 | 6.90% | 1 | 14.29% |
wanlong gao | wanlong gao | 5 | 5.75% | 1 | 14.29% |
trond myklebust | trond myklebust | 2 | 2.30% | 1 | 14.29% |
| Total | 87 | 100.00% | 7 | 100.00% |
static void nbd_xmit_timeout(unsigned long arg)
{
struct nbd_device *nbd = (struct nbd_device *)arg;
unsigned long flags;
if (list_empty(&nbd->queue_head))
return;
spin_lock_irqsave(&nbd->sock_lock, flags);
nbd->timedout = true;
if (nbd->sock)
kernel_sock_shutdown(nbd->sock, SHUT_RDWR);
spin_unlock_irqrestore(&nbd->sock_lock, flags);
dev_err(nbd_to_dev(nbd), "Connection timed out, shutting down connection\n");
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 63 | 71.59% | 4 | 80.00% |
paul clements | paul clements | 25 | 28.41% | 1 | 20.00% |
| Total | 88 | 100.00% | 5 | 100.00% |
/*
* Send or receive packet.
*/
static int sock_xmit(struct nbd_device *nbd, int send, void *buf, int size,
int msg_flags)
{
struct socket *sock = nbd->sock;
int result;
struct msghdr msg;
struct kvec iov;
unsigned long pflags = current->flags;
if (unlikely(!sock)) {
dev_err(disk_to_dev(nbd->disk),
"Attempted %s on closed socket in sock_xmit\n",
(send ? "send" : "recv"));
return -EINVAL;
}
current->flags |= PF_MEMALLOC;
do {
sock->sk->sk_allocation = GFP_NOIO | __GFP_MEMALLOC;
iov.iov_base = buf;
iov.iov_len = size;
msg.msg_name = NULL;
msg.msg_namelen = 0;
msg.msg_control = NULL;
msg.msg_controllen = 0;
msg.msg_flags = msg_flags | MSG_NOSIGNAL;
if (send)
result = kernel_sendmsg(sock, &msg, &iov, 1, size);
else
result = kernel_recvmsg(sock, &msg, &iov, 1, size,
msg.msg_flags);
if (result <= 0) {
if (result == 0)
result = -EPIPE; /* short read */
break;
}
size -= result;
buf += result;
} while (size > 0);
tsk_restore_flags(current, pflags, PF_MEMALLOC);
if (!send && nbd->xmit_timeout)
mod_timer(&nbd->timeout_timer, jiffies + nbd->xmit_timeout);
return result;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
pre-git | pre-git | 129 | 49.05% | 3 | 18.75% |
mike snitzer | mike snitzer | 27 | 10.27% | 1 | 6.25% |
mel gorman | mel gorman | 25 | 9.51% | 1 | 6.25% |
markus pargmann | markus pargmann | 23 | 8.75% | 1 | 6.25% |
andrew morton | andrew morton | 17 | 6.46% | 2 | 12.50% |
al viro | al viro | 13 | 4.94% | 1 | 6.25% |
paul clements | paul clements | 9 | 3.42% | 1 | 6.25% |
linus torvalds | linus torvalds | 7 | 2.66% | 2 | 12.50% |
americo wang | americo wang | 6 | 2.28% | 1 | 6.25% |
namhyung kim | namhyung kim | 3 | 1.14% | 1 | 6.25% |
wanlong gao | wanlong gao | 3 | 1.14% | 1 | 6.25% |
david s. miller | david s. miller | 1 | 0.38% | 1 | 6.25% |
| Total | 263 | 100.00% | 16 | 100.00% |
static inline int sock_send_bvec(struct nbd_device *nbd, struct bio_vec *bvec,
int flags)
{
int result;
void *kaddr = kmap(bvec->bv_page);
result = sock_xmit(nbd, 1, kaddr + bvec->bv_offset,
bvec->bv_len, flags);
kunmap(bvec->bv_page);
return result;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
lou langholtz | lou langholtz | 59 | 90.77% | 1 | 25.00% |
andrew morton | andrew morton | 3 | 4.62% | 1 | 25.00% |
wanlong gao | wanlong gao | 2 | 3.08% | 1 | 25.00% |
paul clements | paul clements | 1 | 1.54% | 1 | 25.00% |
| Total | 65 | 100.00% | 4 | 100.00% |
/* always call with the tx_lock held */
static int nbd_send_req(struct nbd_device *nbd, struct request *req)
{
int result, flags;
struct nbd_request request;
unsigned long size = blk_rq_bytes(req);
u32 type;
if (req->cmd_type == REQ_TYPE_DRV_PRIV)
type = NBD_CMD_DISC;
else if (req->cmd_flags & REQ_DISCARD)
type = NBD_CMD_TRIM;
else if (req->cmd_flags & REQ_FLUSH)
type = NBD_CMD_FLUSH;
else if (rq_data_dir(req) == WRITE)
type = NBD_CMD_WRITE;
else
type = NBD_CMD_READ;
memset(&request, 0, sizeof(request));
request.magic = htonl(NBD_REQUEST_MAGIC);
request.type = htonl(type);
if (type != NBD_CMD_FLUSH && type != NBD_CMD_DISC) {
request.from = cpu_to_be64((u64)blk_rq_pos(req) << 9);
request.len = htonl(size);
}
memcpy(request.handle, &req, sizeof(req));
dev_dbg(nbd_to_dev(nbd), "request %p: sending control (%s@%llu,%uB)\n",
req, nbdcmd_to_ascii(type),
(unsigned long long)blk_rq_pos(req) << 9, blk_rq_bytes(req));
result = sock_xmit(nbd, 1, &request, sizeof(request),
(type == NBD_CMD_WRITE) ? MSG_MORE : 0);
if (result <= 0) {
dev_err(disk_to_dev(nbd->disk),
"Send control failed (result %d)\n", result);
return -EIO;
}
if (type == NBD_CMD_WRITE) {
struct req_iterator iter;
struct bio_vec bvec;
/*
* we are really probing at internals to determine
* whether to set MSG_MORE or not...
*/
rq_for_each_segment(bvec, req, iter) {
flags = 0;
if (!rq_iter_last(bvec, iter))
flags = MSG_MORE;
dev_dbg(nbd_to_dev(nbd), "request %p: sending %d bytes data\n",
req, bvec.bv_len);
result = sock_send_bvec(nbd, &bvec, flags);
if (result <= 0) {
dev_err(disk_to_dev(nbd->disk),
"Send data failed (result %d)\n",
result);
return -EIO;
}
}
}
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
pre-git | pre-git | 117 | 31.88% | 3 | 13.04% |
christoph hellwig | christoph hellwig | 66 | 17.98% | 1 | 4.35% |
linus torvalds | linus torvalds | 42 | 11.44% | 2 | 8.70% |
andrew morton | andrew morton | 42 | 11.44% | 3 | 13.04% |
markus pargmann | markus pargmann | 20 | 5.45% | 2 | 8.70% |
hani benhabiles | hani benhabiles | 17 | 4.63% | 1 | 4.35% |
tejun heo | tejun heo | 12 | 3.27% | 2 | 8.70% |
americo wang | americo wang | 12 | 3.27% | 1 | 4.35% |
neil brown | neil brown | 12 | 3.27% | 1 | 4.35% |
wanlong gao | wanlong gao | 7 | 1.91% | 1 | 4.35% |
alex bligh | alex bligh | 6 | 1.63% | 1 | 4.35% |
paul clements | paul clements | 5 | 1.36% | 1 | 4.35% |
dave jones | dave jones | 5 | 1.36% | 1 | 4.35% |
kent overstreet | kent overstreet | 3 | 0.82% | 2 | 8.70% |
lou langholtz | lou langholtz | 1 | 0.27% | 1 | 4.35% |
| Total | 367 | 100.00% | 23 | 100.00% |
static struct request *nbd_find_request(struct nbd_device *nbd,
struct request *xreq)
{
struct request *req, *tmp;
int err;
err = wait_event_interruptible(nbd->active_wq, nbd->active_req != xreq);
if (unlikely(err))
return ERR_PTR(err);
spin_lock(&nbd->queue_lock);
list_for_each_entry_safe(req, tmp, &nbd->queue_head, queuelist) {
if (req != xreq)
continue;
list_del_init(&req->queuelist);
spin_unlock(&nbd->queue_lock);
return req;
}
spin_unlock(&nbd->queue_lock);
return ERR_PTR(-ENOENT);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
dave jones | dave jones | 68 | 57.63% | 2 | 28.57% |
herbert xu | herbert xu | 26 | 22.03% | 1 | 14.29% |
cheng renquan | cheng renquan | 9 | 7.63% | 2 | 28.57% |
markus pargmann | markus pargmann | 8 | 6.78% | 1 | 14.29% |
wanlong gao | wanlong gao | 7 | 5.93% | 1 | 14.29% |
| Total | 118 | 100.00% | 7 | 100.00% |
static inline int sock_recv_bvec(struct nbd_device *nbd, struct bio_vec *bvec)
{
int result;
void *kaddr = kmap(bvec->bv_page);
result = sock_xmit(nbd, 0, kaddr + bvec->bv_offset, bvec->bv_len,
MSG_WAITALL);
kunmap(bvec->bv_page);
return result;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
lou langholtz | lou langholtz | 56 | 90.32% | 1 | 25.00% |
andrew morton | andrew morton | 3 | 4.84% | 1 | 25.00% |
wanlong gao | wanlong gao | 2 | 3.23% | 1 | 25.00% |
paul clements | paul clements | 1 | 1.61% | 1 | 25.00% |
| Total | 62 | 100.00% | 4 | 100.00% |
/* NULL returned = something went wrong, inform userspace */
static struct request *nbd_read_stat(struct nbd_device *nbd)
{
int result;
struct nbd_reply reply;
struct request *req;
reply.magic = 0;
result = sock_xmit(nbd, 0, &reply, sizeof(reply), MSG_WAITALL);
if (result <= 0) {
dev_err(disk_to_dev(nbd->disk),
"Receive control failed (result %d)\n", result);
return ERR_PTR(result);
}
if (ntohl(reply.magic) != NBD_REPLY_MAGIC) {
dev_err(disk_to_dev(nbd->disk), "Wrong magic (0x%lx)\n",
(unsigned long)ntohl(reply.magic));
return ERR_PTR(-EPROTO);
}
req = nbd_find_request(nbd, *(struct request **)reply.handle);
if (IS_ERR(req)) {
result = PTR_ERR(req);
if (result != -ENOENT)
return ERR_PTR(result);
dev_err(disk_to_dev(nbd->disk), "Unexpected reply (%p)\n",
reply.handle);
return ERR_PTR(-EBADR);
}
if (ntohl(reply.error)) {
dev_err(disk_to_dev(nbd->disk), "Other side returned error (%d)\n",
ntohl(reply.error));
req->errors++;
return req;
}
dev_dbg(nbd_to_dev(nbd), "request %p: got reply\n", req);
if (rq_data_dir(req) != WRITE) {
struct req_iterator iter;
struct bio_vec bvec;
rq_for_each_segment(bvec, req, iter) {
result = sock_recv_bvec(nbd, &bvec);
if (result <= 0) {
dev_err(disk_to_dev(nbd->disk), "Receive data failed (result %d)\n",
result);
req->errors++;
return req;
}
dev_dbg(nbd_to_dev(nbd), "request %p: got %d bytes data\n",
req, bvec.bv_len);
}
}
return req;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
pre-git | pre-git | 101 | 30.70% | 2 | 8.70% |
andrew morton | andrew morton | 56 | 17.02% | 3 | 13.04% |
michal feix | michal feix | 39 | 11.85% | 2 | 8.70% |
markus pargmann | markus pargmann | 34 | 10.33% | 2 | 8.70% |
americo wang | americo wang | 30 | 9.12% | 1 | 4.35% |
herbert xu | herbert xu | 17 | 5.17% | 1 | 4.35% |
lou langholtz | lou langholtz | 11 | 3.34% | 1 | 4.35% |
wanlong gao | wanlong gao | 11 | 3.34% | 1 | 4.35% |
linus torvalds | linus torvalds | 8 | 2.43% | 4 | 17.39% |
cheng renquan | cheng renquan | 7 | 2.13% | 1 | 4.35% |
neil brown | neil brown | 5 | 1.52% | 1 | 4.35% |
dave jones | dave jones | 4 | 1.22% | 1 | 4.35% |
christoph hellwig | christoph hellwig | 3 | 0.91% | 1 | 4.35% |
kent overstreet | kent overstreet | 2 | 0.61% | 1 | 4.35% |
adrian bunk | adrian bunk | 1 | 0.30% | 1 | 4.35% |
| Total | 329 | 100.00% | 23 | 100.00% |
static ssize_t pid_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct gendisk *disk = dev_to_disk(dev);
struct nbd_device *nbd = (struct nbd_device *)disk->private_data;
return sprintf(buf, "%d\n", task_pid_nr(nbd->task_recv));
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
paul clements | paul clements | 23 | 38.98% | 1 | 33.33% |
kay sievers | kay sievers | 18 | 30.51% | 1 | 33.33% |
markus pargmann | markus pargmann | 18 | 30.51% | 1 | 33.33% |
| Total | 59 | 100.00% | 3 | 100.00% |
static struct device_attribute pid_attr = {
.attr = { .name = "pid", .mode = S_IRUGO},
.show = pid_show,
};
static int nbd_thread_recv(struct nbd_device *nbd, struct block_device *bdev)
{
struct request *req;
int ret;
BUG_ON(nbd->magic != NBD_MAGIC);
sk_set_memalloc(nbd->sock->sk);
nbd->task_recv = current;
ret = device_create_file(disk_to_dev(nbd->disk), &pid_attr);
if (ret) {
dev_err(disk_to_dev(nbd->disk), "device_create_file failed!\n");
nbd->task_recv = NULL;
return ret;
}
nbd_size_update(nbd, bdev);
while (1) {
req = nbd_read_stat(nbd);
if (IS_ERR(req)) {
ret = PTR_ERR(req);
break;
}
nbd_end_request(nbd, req);
}
nbd_size_clear(nbd, bdev);
device_remove_file(disk_to_dev(nbd->disk), &pid_attr);
nbd->task_recv = NULL;
return ret;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 66 | 40.49% | 6 | 35.29% |
americo wang | americo wang | 29 | 17.79% | 3 | 17.65% |
pre-git | pre-git | 23 | 14.11% | 1 | 5.88% |
paul clements | paul clements | 12 | 7.36% | 1 | 5.88% |
andrew morton | andrew morton | 9 | 5.52% | 1 | 5.88% |
mel gorman | mel gorman | 9 | 5.52% | 1 | 5.88% |
wanlong gao | wanlong gao | 8 | 4.91% | 1 | 5.88% |
pavel machek | pavel machek | 3 | 1.84% | 1 | 5.88% |
tejun heo | tejun heo | 3 | 1.84% | 1 | 5.88% |
adrian bunk | adrian bunk | 1 | 0.61% | 1 | 5.88% |
| Total | 163 | 100.00% | 17 | 100.00% |
static void nbd_clear_que(struct nbd_device *nbd)
{
struct request *req;
BUG_ON(nbd->magic != NBD_MAGIC);
/*
* Because we have set nbd->sock to NULL under the tx_lock, all
* modifications to the list must have completed by now. For
* the same reason, the active_req must be NULL.
*
* As a consequence, we don't need to take the spin lock while
* purging the list here.
*/
BUG_ON(nbd->sock);
BUG_ON(nbd->active_req);
while (!list_empty(&nbd->queue_head)) {
req = list_entry(nbd->queue_head.next, struct request,
queuelist);
list_del_init(&req->queuelist);
req->errors++;
nbd_end_request(nbd, req);
}
while (!list_empty(&nbd->waiting_queue)) {
req = list_entry(nbd->waiting_queue.next, struct request,
queuelist);
list_del_init(&req->queuelist);
req->errors++;
nbd_end_request(nbd, req);
}
dev_dbg(disk_to_dev(nbd->disk), "queue cleared\n");
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
pre-git | pre-git | 50 | 33.33% | 3 | 23.08% |
paul clements | paul clements | 47 | 31.33% | 1 | 7.69% |
dave jones | dave jones | 17 | 11.33% | 2 | 15.38% |
markus pargmann | markus pargmann | 16 | 10.67% | 2 | 15.38% |
herbert xu | herbert xu | 8 | 5.33% | 1 | 7.69% |
wanlong gao | wanlong gao | 8 | 5.33% | 1 | 7.69% |
pavel machek | pavel machek | 2 | 1.33% | 1 | 7.69% |
linus torvalds | linus torvalds | 1 | 0.67% | 1 | 7.69% |
adrian bunk | adrian bunk | 1 | 0.67% | 1 | 7.69% |
| Total | 150 | 100.00% | 13 | 100.00% |
static void nbd_handle_req(struct nbd_device *nbd, struct request *req)
{
if (req->cmd_type != REQ_TYPE_FS)
goto error_out;
if (rq_data_dir(req) == WRITE &&
(nbd->flags & NBD_FLAG_READ_ONLY)) {
dev_err(disk_to_dev(nbd->disk),
"Write on read-only\n");
goto error_out;
}
req->errors = 0;
mutex_lock(&nbd->tx_lock);
if (unlikely(!nbd->sock)) {
mutex_unlock(&nbd->tx_lock);
dev_err(disk_to_dev(nbd->disk),
"Attempted send on closed socket\n");
goto error_out;
}
nbd->active_req = req;
if (nbd->xmit_timeout && list_empty_careful(&nbd->queue_head))
mod_timer(&nbd->timeout_timer, jiffies + nbd->xmit_timeout);
if (nbd_send_req(nbd, req) != 0) {
dev_err(disk_to_dev(nbd->disk), "Request send failed\n");
req->errors++;
nbd_end_request(nbd, req);
} else {
spin_lock(&nbd->queue_lock);
list_add_tail(&req->queuelist, &nbd->queue_head);
spin_unlock(&nbd->queue_lock);
}
nbd->active_req = NULL;
mutex_unlock(&nbd->tx_lock);
wake_up_all(&nbd->active_wq);
return;
error_out:
req->errors++;
nbd_end_request(nbd, req);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
herbert xu | herbert xu | 50 | 20.00% | 1 | 3.70% |
markus pargmann | markus pargmann | 32 | 12.80% | 2 | 7.41% |
pre-git | pre-git | 29 | 11.60% | 4 | 14.81% |
alan cox | alan cox | 27 | 10.80% | 1 | 3.70% |
americo wang | americo wang | 18 | 7.20% | 1 | 3.70% |
laurent vivier | laurent vivier | 18 | 7.20% | 1 | 3.70% |
wanlong gao | wanlong gao | 15 | 6.00% | 1 | 3.70% |
andrew morton | andrew morton | 15 | 6.00% | 2 | 7.41% |
paul clements | paul clements | 14 | 5.60% | 3 | 11.11% |
linus torvalds | linus torvalds | 10 | 4.00% | 1 | 3.70% |
christoph hellwig | christoph hellwig | 6 | 2.40% | 2 | 7.41% |
pavel machek | pavel machek | 4 | 1.60% | 2 | 7.41% |
al viro | al viro | 3 | 1.20% | 1 | 3.70% |
ingo molnar | ingo molnar | 3 | 1.20% | 1 | 3.70% |
dave jones | dave jones | 3 | 1.20% | 1 | 3.70% |
jens axboe | jens axboe | 2 | 0.80% | 2 | 7.41% |
chetan loke | chetan loke | 1 | 0.40% | 1 | 3.70% |
| Total | 250 | 100.00% | 27 | 100.00% |
static int nbd_thread_send(void *data)
{
struct nbd_device *nbd = data;
struct request *req;
nbd->task_send = current;
set_user_nice(current, MIN_NICE);
while (!kthread_should_stop() || !list_empty(&nbd->waiting_queue)) {
/* wait for something to do */
wait_event_interruptible(nbd->waiting_wq,
kthread_should_stop() ||
!list_empty(&nbd->waiting_queue));
/* extract request */
if (list_empty(&nbd->waiting_queue))
continue;
spin_lock_irq(&nbd->queue_lock);
req = list_entry(nbd->waiting_queue.next, struct request,
queuelist);
list_del_init(&req->queuelist);
spin_unlock_irq(&nbd->queue_lock);
/* handle request */
nbd_handle_req(nbd, req);
}
nbd->task_send = NULL;
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
laurent vivier | laurent vivier | 109 | 77.30% | 1 | 12.50% |
markus pargmann | markus pargmann | 13 | 9.22% | 2 | 25.00% |
wanlong gao | wanlong gao | 9 | 6.38% | 1 | 12.50% |
pre-git | pre-git | 7 | 4.96% | 2 | 25.00% |
linus torvalds | linus torvalds | 2 | 1.42% | 1 | 12.50% |
dongsheng yang | dongsheng yang | 1 | 0.71% | 1 | 12.50% |
| Total | 141 | 100.00% | 8 | 100.00% |
/*
* We always wait for result of write, for now. It would be nice to make it optional
* in future
* if ((rq_data_dir(req) == WRITE) && (nbd->flags & NBD_WRITE_NOCHK))
* { printk( "Warning: Ignoring result!\n"); nbd_end_request( req ); }
*/
static void nbd_request_handler(struct request_queue *q)
__releases(q->queue_lock) __acquires(q->queue_lock)
{
struct request *req;
while ((req = blk_fetch_request(q)) != NULL) {
struct nbd_device *nbd;
spin_unlock_irq(q->queue_lock);
nbd = req->rq_disk->private_data;
BUG_ON(nbd->magic != NBD_MAGIC);
dev_dbg(nbd_to_dev(nbd), "request %p: dequeued (flags=%x)\n",
req, req->cmd_type);
if (unlikely(!nbd->sock)) {
dev_err_ratelimited(disk_to_dev(nbd->disk),
"Attempted send on closed socket\n");
req->errors++;
nbd_end_request(nbd, req);
spin_lock_irq(q->queue_lock);
continue;
}
spin_lock_irq(&nbd->queue_lock);
list_add_tail(&req->queuelist, &nbd->waiting_queue);
spin_unlock_irq(&nbd->queue_lock);
wake_up(&nbd->waiting_wq);
spin_lock_irq(q->queue_lock);
}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
laurent vivier | laurent vivier | 80 | 45.45% | 1 | 8.33% |
paul clements | paul clements | 34 | 19.32% | 1 | 8.33% |
markus pargmann | markus pargmann | 19 | 10.80% | 2 | 16.67% |
alex elder | alex elder | 12 | 6.82% | 1 | 8.33% |
wanlong gao | wanlong gao | 10 | 5.68% | 1 | 8.33% |
pre-git | pre-git | 8 | 4.55% | 2 | 16.67% |
linus torvalds | linus torvalds | 6 | 3.41% | 1 | 8.33% |
americo wang | americo wang | 5 | 2.84% | 1 | 8.33% |
dan streetman | dan streetman | 1 | 0.57% | 1 | 8.33% |
tejun heo | tejun heo | 1 | 0.57% | 1 | 8.33% |
| Total | 176 | 100.00% | 12 | 100.00% |
static int nbd_set_socket(struct nbd_device *nbd, struct socket *sock)
{
int ret = 0;
spin_lock_irq(&nbd->sock_lock);
if (nbd->sock) {
ret = -EBUSY;
goto out;
}
nbd->sock = sock;
out:
spin_unlock_irq(&nbd->sock_lock);
return ret;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 64 | 100.00% | 1 | 100.00% |
| Total | 64 | 100.00% | 1 | 100.00% |
/* Reset all properties of an NBD device */
static void nbd_reset(struct nbd_device *nbd)
{
nbd->disconnect = false;
nbd->timedout = false;
nbd->blksize = 1024;
nbd->bytesize = 0;
set_capacity(nbd->disk, 0);
nbd->flags = 0;
nbd->xmit_timeout = 0;
queue_flag_clear_unlocked(QUEUE_FLAG_DISCARD, nbd->disk->queue);
del_timer_sync(&nbd->timeout_timer);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 75 | 100.00% | 1 | 100.00% |
| Total | 75 | 100.00% | 1 | 100.00% |
static void nbd_bdev_reset(struct block_device *bdev)
{
set_device_ro(bdev, false);
bdev->bd_inode->i_size = 0;
if (max_part > 0) {
blkdev_reread_part(bdev);
bdev->bd_invalidated = 1;
}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 45 | 100.00% | 1 | 100.00% |
| Total | 45 | 100.00% | 1 | 100.00% |
static void nbd_parse_flags(struct nbd_device *nbd, struct block_device *bdev)
{
if (nbd->flags & NBD_FLAG_READ_ONLY)
set_device_ro(bdev, true);
if (nbd->flags & NBD_FLAG_SEND_TRIM)
queue_flag_set_unlocked(QUEUE_FLAG_DISCARD, nbd->disk->queue);
if (nbd->flags & NBD_FLAG_SEND_FLUSH)
blk_queue_write_cache(nbd->disk->queue, true, false);
else
blk_queue_write_cache(nbd->disk->queue, false, false);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 77 | 90.59% | 1 | 50.00% |
jens axboe | jens axboe | 8 | 9.41% | 1 | 50.00% |
| Total | 85 | 100.00% | 2 | 100.00% |
static int nbd_dev_dbg_init(struct nbd_device *nbd);
static void nbd_dev_dbg_close(struct nbd_device *nbd);
/* Must be called with tx_lock held */
static int __nbd_ioctl(struct block_device *bdev, struct nbd_device *nbd,
unsigned int cmd, unsigned long arg)
{
switch (cmd) {
case NBD_DISCONNECT: {
struct request sreq;
dev_info(disk_to_dev(nbd->disk), "NBD_DISCONNECT\n");
if (!nbd->sock)
return -EINVAL;
mutex_unlock(&nbd->tx_lock);
fsync_bdev(bdev);
mutex_lock(&nbd->tx_lock);
blk_rq_init(NULL, &sreq);
sreq.cmd_type = REQ_TYPE_DRV_PRIV;
/* Check again after getting mutex back. */
if (!nbd->sock)
return -EINVAL;
nbd->disconnect = true;
nbd_send_req(nbd, &sreq);
return 0;
}
case NBD_CLEAR_SOCK:
sock_shutdown(nbd);
nbd_clear_que(nbd);
BUG_ON(!list_empty(&nbd->queue_head));
BUG_ON(!list_empty(&nbd->waiting_queue));
kill_bdev(bdev);
return 0;
case NBD_SET_SOCK: {
int err;
struct socket *sock = sockfd_lookup(arg, &err);
if (!sock)
return err;
err = nbd_set_socket(nbd, sock);
if (!err && max_part)
bdev->bd_invalidated = 1;
return err;
}
case NBD_SET_BLKSIZE: {
loff_t bsize = div_s64(nbd->bytesize, arg);
return nbd_size_set(nbd, bdev, arg, bsize);
}
case NBD_SET_SIZE:
return nbd_size_set(nbd, bdev, nbd->blksize,
arg / nbd->blksize);
case NBD_SET_SIZE_BLOCKS:
return nbd_size_set(nbd, bdev, nbd->blksize, arg);
case NBD_SET_TIMEOUT:
nbd->xmit_timeout = arg * HZ;
if (arg)
mod_timer(&nbd->timeout_timer,
jiffies + nbd->xmit_timeout);
else
del_timer_sync(&nbd->timeout_timer);
return 0;
case NBD_SET_FLAGS:
nbd->flags = arg;
return 0;
case NBD_DO_IT: {
struct task_struct *thread;
int error;
if (nbd->task_recv)
return -EBUSY;
if (!nbd->sock)
return -EINVAL;
mutex_unlock(&nbd->tx_lock);
nbd_parse_flags(nbd, bdev);
thread = kthread_run(nbd_thread_send, nbd, "%s",
nbd_name(nbd));
if (IS_ERR(thread)) {
mutex_lock(&nbd->tx_lock);
return PTR_ERR(thread);
}
nbd_dev_dbg_init(nbd);
error = nbd_thread_recv(nbd, bdev);
nbd_dev_dbg_close(nbd);
kthread_stop(thread);
mutex_lock(&nbd->tx_lock);
sock_shutdown(nbd);
nbd_clear_que(nbd);
kill_bdev(bdev);
nbd_bdev_reset(bdev);
if (nbd->disconnect) /* user requested, ignore socket errors */
error = 0;
if (nbd->timedout)
error = -ETIMEDOUT;
nbd_reset(nbd);
return error;
}
case NBD_CLEAR_QUE:
/*
* This is for compatibility only. The queue is always cleared
* by NBD_DO_IT or NBD_CLEAR_SOCK.
*/
return 0;
case NBD_PRINT_DEBUG:
dev_info(disk_to_dev(nbd->disk),
"next = %p, prev = %p, head = %p\n",
nbd->queue_head.next, nbd->queue_head.prev,
&nbd->queue_head);
return 0;
}
return -ENOTTY;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 124 | 22.10% | 12 | 24.00% |
pre-git | pre-git | 121 | 21.57% | 6 | 12.00% |
pavel machek | pavel machek | 58 | 10.34% | 2 | 4.00% |
paul clements | paul clements | 52 | 9.27% | 5 | 10.00% |
paolo bonzini | paolo bonzini | 45 | 8.02% | 2 | 4.00% |
laurent vivier | laurent vivier | 38 | 6.77% | 2 | 4.00% |
wanlong gao | wanlong gao | 29 | 5.17% | 1 | 2.00% |
andrew morton | andrew morton | 28 | 4.99% | 5 | 10.00% |
al viro | al viro | 23 | 4.10% | 5 | 10.00% |
americo wang | americo wang | 15 | 2.67% | 2 | 4.00% |
alan cox | alan cox | 9 | 1.60% | 1 | 2.00% |
fujita tomonori | fujita tomonori | 8 | 1.43% | 1 | 2.00% |
arnd bergmann | arnd bergmann | 3 | 0.53% | 1 | 2.00% |
herbert xu | herbert xu | 2 | 0.36% | 1 | 2.00% |
alex bligh | alex bligh | 2 | 0.36% | 1 | 2.00% |
kees cook | kees cook | 2 | 0.36% | 1 | 2.00% |
jens axboe | jens axboe | 1 | 0.18% | 1 | 2.00% |
christoph hellwig | christoph hellwig | 1 | 0.18% | 1 | 2.00% |
| Total | 561 | 100.00% | 50 | 100.00% |
static int nbd_ioctl(struct block_device *bdev, fmode_t mode,
unsigned int cmd, unsigned long arg)
{
struct nbd_device *nbd = bdev->bd_disk->private_data;
int error;
if (!capable(CAP_SYS_ADMIN))
return -EPERM;
BUG_ON(nbd->magic != NBD_MAGIC);
mutex_lock(&nbd->tx_lock);
error = __nbd_ioctl(bdev, nbd, cmd, arg);
mutex_unlock(&nbd->tx_lock);
return error;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
pavel machek | pavel machek | 78 | 87.64% | 1 | 25.00% |
wanlong gao | wanlong gao | 6 | 6.74% | 1 | 25.00% |
arnd bergmann | arnd bergmann | 3 | 3.37% | 1 | 25.00% |
pre-git | pre-git | 2 | 2.25% | 1 | 25.00% |
| Total | 89 | 100.00% | 4 | 100.00% |
static const struct block_device_operations nbd_fops =
{
.owner = THIS_MODULE,
.ioctl = nbd_ioctl,
.compat_ioctl = nbd_ioctl,
};
#if IS_ENABLED(CONFIG_DEBUG_FS)
static int nbd_dbg_tasks_show(struct seq_file *s, void *unused)
{
struct nbd_device *nbd = s->private;
if (nbd->task_recv)
seq_printf(s, "recv: %d\n", task_pid_nr(nbd->task_recv));
if (nbd->task_send)
seq_printf(s, "send: %d\n", task_pid_nr(nbd->task_send));
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 45 | 67.16% | 1 | 12.50% |
sven wegener | sven wegener | 9 | 13.43% | 1 | 12.50% |
pre-git | pre-git | 9 | 13.43% | 2 | 25.00% |
laurent vivier | laurent vivier | 1 | 1.49% | 1 | 12.50% |
adrian bunk | adrian bunk | 1 | 1.49% | 1 | 12.50% |
linus torvalds | linus torvalds | 1 | 1.49% | 1 | 12.50% |
al viro | al viro | 1 | 1.49% | 1 | 12.50% |
| Total | 67 | 100.00% | 8 | 100.00% |
static int nbd_dbg_tasks_open(struct inode *inode, struct file *file)
{
return single_open(file, nbd_dbg_tasks_show, inode->i_private);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 27 | 96.43% | 1 | 50.00% |
namhyung kim | namhyung kim | 1 | 3.57% | 1 | 50.00% |
| Total | 28 | 100.00% | 2 | 100.00% |
static const struct file_operations nbd_dbg_tasks_ops = {
.open = nbd_dbg_tasks_open,
.read = seq_read,
.llseek = seq_lseek,
.release = single_release,
};
static int nbd_dbg_flags_show(struct seq_file *s, void *unused)
{
struct nbd_device *nbd = s->private;
u32 flags = nbd->flags;
seq_printf(s, "Hex: 0x%08x\n\n", flags);
seq_puts(s, "Known flags:\n");
if (flags & NBD_FLAG_HAS_FLAGS)
seq_puts(s, "NBD_FLAG_HAS_FLAGS\n");
if (flags & NBD_FLAG_READ_ONLY)
seq_puts(s, "NBD_FLAG_READ_ONLY\n");
if (flags & NBD_FLAG_SEND_FLUSH)
seq_puts(s, "NBD_FLAG_SEND_FLUSH\n");
if (flags & NBD_FLAG_SEND_TRIM)
seq_puts(s, "NBD_FLAG_SEND_TRIM\n");
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 89 | 87.25% | 1 | 20.00% |
sudip mukherjee | sudip mukherjee | 6 | 5.88% | 1 | 20.00% |
namhyung kim | namhyung kim | 4 | 3.92% | 2 | 40.00% |
laurent vivier | laurent vivier | 3 | 2.94% | 1 | 20.00% |
| Total | 102 | 100.00% | 5 | 100.00% |
static int nbd_dbg_flags_open(struct inode *inode, struct file *file)
{
return single_open(file, nbd_dbg_flags_show, inode->i_private);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 28 | 100.00% | 1 | 100.00% |
| Total | 28 | 100.00% | 1 | 100.00% |
static const struct file_operations nbd_dbg_flags_ops = {
.open = nbd_dbg_flags_open,
.read = seq_read,
.llseek = seq_lseek,
.release = single_release,
};
static int nbd_dev_dbg_init(struct nbd_device *nbd)
{
struct dentry *dir;
if (!nbd_dbg_dir)
return -EIO;
dir = debugfs_create_dir(nbd_name(nbd), nbd_dbg_dir);
if (!dir) {
dev_err(nbd_to_dev(nbd), "Failed to create debugfs dir for '%s'\n",
nbd_name(nbd));
return -EIO;
}
nbd->dbg_dir = dir;
debugfs_create_file("tasks", 0444, dir, nbd, &nbd_dbg_tasks_ops);
debugfs_create_u64("size_bytes", 0444, dir, &nbd->bytesize);
debugfs_create_u32("timeout", 0444, dir, &nbd->xmit_timeout);
debugfs_create_u32("blocksize", 0444, dir, &nbd->blksize);
debugfs_create_file("flags", 0444, dir, nbd, &nbd_dbg_flags_ops);
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 142 | 100.00% | 2 | 100.00% |
| Total | 142 | 100.00% | 2 | 100.00% |
static void nbd_dev_dbg_close(struct nbd_device *nbd)
{
debugfs_remove_recursive(nbd->dbg_dir);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 18 | 100.00% | 1 | 100.00% |
| Total | 18 | 100.00% | 1 | 100.00% |
static int nbd_dbg_init(void)
{
struct dentry *dbg_dir;
dbg_dir = debugfs_create_dir("nbd", NULL);
if (!dbg_dir)
return -EIO;
nbd_dbg_dir = dbg_dir;
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 38 | 100.00% | 2 | 100.00% |
| Total | 38 | 100.00% | 2 | 100.00% |
static void nbd_dbg_close(void)
{
debugfs_remove_recursive(nbd_dbg_dir);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 13 | 100.00% | 1 | 100.00% |
| Total | 13 | 100.00% | 1 | 100.00% |
#else /* IS_ENABLED(CONFIG_DEBUG_FS) */
static int nbd_dev_dbg_init(struct nbd_device *nbd)
{
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 14 | 100.00% | 1 | 100.00% |
| Total | 14 | 100.00% | 1 | 100.00% |
static void nbd_dev_dbg_close(struct nbd_device *nbd)
{
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 10 | 100.00% | 1 | 100.00% |
| Total | 10 | 100.00% | 1 | 100.00% |
static int nbd_dbg_init(void)
{
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 11 | 100.00% | 1 | 100.00% |
| Total | 11 | 100.00% | 1 | 100.00% |
static void nbd_dbg_close(void)
{
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 7 | 100.00% | 1 | 100.00% |
| Total | 7 | 100.00% | 1 | 100.00% |
#endif
/*
* And here should be modules and kernel interface
* (Just smiley confuses emacs :-)
*/
static int __init nbd_init(void)
{
int err = -ENOMEM;
int i;
int part_shift;
BUILD_BUG_ON(sizeof(struct nbd_request) != 28);
if (max_part < 0) {
printk(KERN_ERR "nbd: max_part must be >= 0\n");
return -EINVAL;
}
part_shift = 0;
if (max_part > 0) {
part_shift = fls(max_part);
/*
* Adjust max_part according to part_shift as it is exported
* to user space so that user can know the max number of
* partition kernel should be able to manage.
*
* Note that -1 is required because partition 0 is reserved
* for the whole disk.
*/
max_part = (1UL << part_shift) - 1;
}
if ((1UL << part_shift) > DISK_MAX_PARTS)
return -EINVAL;
if (nbds_max > 1UL << (MINORBITS - part_shift))
return -EINVAL;
nbd_dev = kcalloc(nbds_max, sizeof(*nbd_dev), GFP_KERNEL);
if (!nbd_dev)
return -ENOMEM;
for (i = 0; i < nbds_max; i++) {
struct gendisk *disk = alloc_disk(1 << part_shift);
if (!disk)
goto out;
nbd_dev[i].disk = disk;
/*
* The new linux 2.5 block layer implementation requires
* every gendisk to have its very own request_queue struct.
* These structs are big so we dynamically allocate them.
*/
disk->queue = blk_init_queue(nbd_request_handler, &nbd_lock);
if (!disk->queue) {
put_disk(disk);
goto out;
}
/*
* Tell the block layer that we are not a rotational device
*/
queue_flag_set_unlocked(QUEUE_FLAG_NONROT, disk->queue);
queue_flag_clear_unlocked(QUEUE_FLAG_ADD_RANDOM, disk->queue);
disk->queue->limits.discard_granularity = 512;
blk_queue_max_discard_sectors(disk->queue, UINT_MAX);
disk->queue->limits.discard_zeroes_data = 0;
blk_queue_max_hw_sectors(disk->queue, 65536);
disk->queue->limits.max_sectors = 256;
}
if (register_blkdev(NBD_MAJOR, "nbd")) {
err = -EIO;
goto out;
}
printk(KERN_INFO "nbd: registered device at major %d\n", NBD_MAJOR);
nbd_dbg_init();
for (i = 0; i < nbds_max; i++) {
struct gendisk *disk = nbd_dev[i].disk;
nbd_dev[i].magic = NBD_MAGIC;
INIT_LIST_HEAD(&nbd_dev[i].waiting_queue);
spin_lock_init(&nbd_dev[i].queue_lock);
spin_lock_init(&nbd_dev[i].sock_lock);
INIT_LIST_HEAD(&nbd_dev[i].queue_head);
mutex_init(&nbd_dev[i].tx_lock);
init_timer(&nbd_dev[i].timeout_timer);
nbd_dev[i].timeout_timer.function = nbd_xmit_timeout;
nbd_dev[i].timeout_timer.data = (unsigned long)&nbd_dev[i];
init_waitqueue_head(&nbd_dev[i].active_wq);
init_waitqueue_head(&nbd_dev[i].waiting_wq);
disk->major = NBD_MAJOR;
disk->first_minor = i << part_shift;
disk->fops = &nbd_fops;
disk->private_data = &nbd_dev[i];
sprintf(disk->disk_name, "nbd%d", i);
nbd_reset(&nbd_dev[i]);
add_disk(disk);
}
return 0;
out:
while (i--) {
blk_cleanup_queue(nbd_dev[i].disk->queue);
put_disk(nbd_dev[i].disk);
}
kfree(nbd_dev);
return err;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 190 | 34.11% | 6 | 15.38% |
al viro | al viro | 114 | 20.47% | 6 | 15.38% |
pre-git | pre-git | 71 | 12.75% | 6 | 15.38% |
lou langholtz | lou langholtz | 39 | 7.00% | 1 | 2.56% |
paul clements | paul clements | 33 | 5.92% | 2 | 5.13% |
laurent vivier | laurent vivier | 26 | 4.67% | 2 | 5.13% |
michal belczyk | michal belczyk | 19 | 3.41% | 1 | 2.56% |
herbert xu | herbert xu | 11 | 1.97% | 1 | 2.56% |
dave jones | dave jones | 11 | 1.97% | 1 | 2.56% |
jens axboe | jens axboe | 11 | 1.97% | 3 | 7.69% |
mike snitzer | mike snitzer | 9 | 1.62% | 1 | 2.56% |
christoph hellwig | christoph hellwig | 7 | 1.26% | 2 | 5.13% |
sudip mukherjee | sudip mukherjee | 5 | 0.90% | 1 | 2.56% |
sven wegener | sven wegener | 5 | 0.90% | 1 | 2.56% |
lars marowsky-bree | lars marowsky-bree | 2 | 0.36% | 1 | 2.56% |
anton altaparmakov | anton altaparmakov | 1 | 0.18% | 1 | 2.56% |
wanlong gao | wanlong gao | 1 | 0.18% | 1 | 2.56% |
andrew morton | andrew morton | 1 | 0.18% | 1 | 2.56% |
ingo molnar | ingo molnar | 1 | 0.18% | 1 | 2.56% |
| Total | 557 | 100.00% | 39 | 100.00% |
static void __exit nbd_cleanup(void)
{
int i;
nbd_dbg_close();
for (i = 0; i < nbds_max; i++) {
struct gendisk *disk = nbd_dev[i].disk;
nbd_dev[i].magic = 0;
if (disk) {
del_gendisk(disk);
blk_cleanup_queue(disk->queue);
put_disk(disk);
}
}
unregister_blkdev(NBD_MAJOR, "nbd");
kfree(nbd_dev);
printk(KERN_INFO "nbd: unregistered device at major %d\n", NBD_MAJOR);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
lou langholtz | lou langholtz | 27 | 28.72% | 1 | 7.69% |
al viro | al viro | 24 | 25.53% | 2 | 15.38% |
pre-git | pre-git | 16 | 17.02% | 3 | 23.08% |
lars marowsky-bree | lars marowsky-bree | 10 | 10.64% | 1 | 7.69% |
sven wegener | sven wegener | 5 | 5.32% | 1 | 7.69% |
andrew morton | andrew morton | 5 | 5.32% | 2 | 15.38% |
markus pargmann | markus pargmann | 3 | 3.19% | 1 | 7.69% |
linus torvalds | linus torvalds | 3 | 3.19% | 1 | 7.69% |
christoph hellwig | christoph hellwig | 1 | 1.06% | 1 | 7.69% |
| Total | 94 | 100.00% | 13 | 100.00% |
module_init(nbd_init);
module_exit(nbd_cleanup);
MODULE_DESCRIPTION("Network Block Device");
MODULE_LICENSE("GPL");
module_param(nbds_max, int, 0444);
MODULE_PARM_DESC(nbds_max, "number of network block devices to initialize (default: 16)");
module_param(max_part, int, 0444);
MODULE_PARM_DESC(max_part, "number of partitions per device (default: 0)");
Overall Contributors
| Person | Tokens | Prop | Commits | CommitProp |
markus pargmann | markus pargmann | 1812 | 35.27% | 23 | 14.56% |
pre-git | pre-git | 736 | 14.33% | 19 | 12.03% |
paul clements | paul clements | 321 | 6.25% | 10 | 6.33% |
laurent vivier | laurent vivier | 300 | 5.84% | 2 | 1.27% |
andrew morton | andrew morton | 238 | 4.63% | 9 | 5.70% |
lou langholtz | lou langholtz | 194 | 3.78% | 1 | 0.63% |
al viro | al viro | 183 | 3.56% | 11 | 6.96% |
pavel machek | pavel machek | 147 | 2.86% | 5 | 3.16% |
christoph hellwig | christoph hellwig | 138 | 2.69% | 5 | 3.16% |
wanlong gao | wanlong gao | 125 | 2.43% | 1 | 0.63% |
herbert xu | herbert xu | 123 | 2.39% | 1 | 0.63% |
americo wang | americo wang | 121 | 2.36% | 4 | 2.53% |
dave jones | dave jones | 108 | 2.10% | 2 | 1.27% |
linus torvalds | linus torvalds | 108 | 2.10% | 9 | 5.70% |
paolo bonzini | paolo bonzini | 45 | 0.88% | 2 | 1.27% |
michal feix | michal feix | 39 | 0.76% | 2 | 1.27% |
mike snitzer | mike snitzer | 36 | 0.70% | 2 | 1.27% |
alan cox | alan cox | 36 | 0.70% | 1 | 0.63% |
mel gorman | mel gorman | 34 | 0.66% | 1 | 0.63% |
lars marowsky-bree | lars marowsky-bree | 34 | 0.66% | 1 | 0.63% |
jens axboe | jens axboe | 24 | 0.47% | 7 | 4.43% |
tejun heo | tejun heo | 20 | 0.39% | 5 | 3.16% |
michal belczyk | michal belczyk | 19 | 0.37% | 1 | 0.63% |
kay sievers | kay sievers | 19 | 0.37% | 1 | 0.63% |
sven wegener | sven wegener | 19 | 0.37% | 1 | 0.63% |
neil brown | neil brown | 17 | 0.33% | 1 | 0.63% |
hani benhabiles | hani benhabiles | 17 | 0.33% | 1 | 0.63% |
cheng renquan | cheng renquan | 16 | 0.31% | 2 | 1.27% |
alex bligh | alex bligh | 14 | 0.27% | 1 | 0.63% |
alex elder | alex elder | 12 | 0.23% | 1 | 0.63% |
sudip mukherjee | sudip mukherjee | 11 | 0.21% | 1 | 0.63% |
arnd bergmann | arnd bergmann | 10 | 0.19% | 3 | 1.90% |
namhyung kim | namhyung kim | 8 | 0.16% | 3 | 1.90% |
fujita tomonori | fujita tomonori | 8 | 0.16% | 1 | 0.63% |
kiyoshi ueda | kiyoshi ueda | 7 | 0.14% | 1 | 0.63% |
trond myklebust | trond myklebust | 5 | 0.10% | 1 | 0.63% |
kent overstreet | kent overstreet | 5 | 0.10% | 2 | 1.27% |
ingo molnar | ingo molnar | 4 | 0.08% | 1 | 0.63% |
thomas gleixner | thomas gleixner | 4 | 0.08% | 1 | 0.63% |
rusty russell | rusty russell | 4 | 0.08% | 1 | 0.63% |
adrian bunk | adrian bunk | 4 | 0.08% | 2 | 1.27% |
arnaldo carvalho de melo | arnaldo carvalho de melo | 3 | 0.06% | 1 | 0.63% |
kees cook | kees cook | 2 | 0.04% | 1 | 0.63% |
david s. miller | david s. miller | 2 | 0.04% | 2 | 1.27% |
chetan loke | chetan loke | 1 | 0.02% | 1 | 0.63% |
dongsheng yang | dongsheng yang | 1 | 0.02% | 1 | 0.63% |
dan streetman | dan streetman | 1 | 0.02% | 1 | 0.63% |
anton altaparmakov | anton altaparmakov | 1 | 0.02% | 1 | 0.63% |
alexey dobriyan | alexey dobriyan | 1 | 0.02% | 1 | 0.63% |
| Total | 5137 | 100.00% | 158 | 100.00% |
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.