target: Add a user-passthrough backstore
Add a LIO storage engine that presents commands to userspace for execution. This would allow more complex backstores to be implemented out-of-kernel, and also make experimentation a-la FUSE (but at the SCSI level -- "SUSE"?) possible. It uses a mmap()able UIO device per LUN to share a command ring and data area. The commands are raw SCSI CDBs and iovs for in/out data. The command ring is also reused for returning scsi command status and optional sense data. This implementation is based on Shaohua Li's earlier version but heavily modified. Differences include: * Shared memory allocated by kernel, not locked-down user pages * Single ring for command request and response * Offsets instead of embedded pointers * Generic SCSI CDB passthrough instead of per-cmd specialization in ring format. * Uses UIO device instead of anon_file passed in mailbox. * Optional in-kernel handling of some commands. The main reason for these differences is to permit greater resiliency if the user process dies or hangs. Things not yet implemented (on purpose): * Zero copy. The data area is flexible enough to allow page flipping or backend-allocated pages to be used by fabrics, but it's not clear these are performance wins. Can come later. * Out-of-order command completion by userspace. Possible to add by just allowing userspace to change cmd_id in rsp cmd entries, but currently not supported. * No locks between kernel cmd submission and completion routines. Sounds like it's possible, but this can come later. * Sparse allocation of mmaped area. Current code vmallocs the whole thing. If the mapped area was larger and not fully mapped then the driver would have more freedom to change cmd and data area sizes based on demand. Current code open issues: * The use of idrs may be overkill -- we maybe can replace them with a simple counter to generate cmd_ids, and a hash table to get a cmd_id's associated pointer. * Use of a free-running counter for cmd ring instead of explicit modulo math. This would require power-of-2 cmd ring size. (Add kconfig depends NET - Randy) Signed-off-by: Andy Grover <agrover@redhat.com> Signed-off-by: Nicholas Bellinger <nab@linux-iscsi.org>
This commit is contained in:
parent
ce87685128
commit
7c9e7a6fe1
|
@ -31,6 +31,13 @@ config TCM_PSCSI
|
|||
Say Y here to enable the TCM/pSCSI subsystem plugin for non-buffered
|
||||
passthrough access to Linux/SCSI device
|
||||
|
||||
config TCM_USER
|
||||
tristate "TCM/USER Subsystem Plugin for Linux"
|
||||
depends on UIO && NET
|
||||
help
|
||||
Say Y here to enable the TCM/USER subsystem plugin for a userspace
|
||||
process to handle requests
|
||||
|
||||
source "drivers/target/loopback/Kconfig"
|
||||
source "drivers/target/tcm_fc/Kconfig"
|
||||
source "drivers/target/iscsi/Kconfig"
|
||||
|
|
|
@ -22,6 +22,7 @@ obj-$(CONFIG_TARGET_CORE) += target_core_mod.o
|
|||
obj-$(CONFIG_TCM_IBLOCK) += target_core_iblock.o
|
||||
obj-$(CONFIG_TCM_FILEIO) += target_core_file.o
|
||||
obj-$(CONFIG_TCM_PSCSI) += target_core_pscsi.o
|
||||
obj-$(CONFIG_TCM_USER) += target_core_user.o
|
||||
|
||||
# Fabric modules
|
||||
obj-$(CONFIG_LOOPBACK_TARGET) += loopback/
|
||||
|
|
|
@ -232,6 +232,10 @@ void transport_subsystem_check_init(void)
|
|||
if (ret != 0)
|
||||
pr_err("Unable to load target_core_pscsi\n");
|
||||
|
||||
ret = request_module("target_core_user");
|
||||
if (ret != 0)
|
||||
pr_err("Unable to load target_core_user\n");
|
||||
|
||||
sub_api_initialized = 1;
|
||||
}
|
||||
|
||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -371,6 +371,7 @@ header-y += swab.h
|
|||
header-y += synclink.h
|
||||
header-y += sysctl.h
|
||||
header-y += sysinfo.h
|
||||
header-y += target_core_user.h
|
||||
header-y += taskstats.h
|
||||
header-y += tcp.h
|
||||
header-y += tcp_metrics.h
|
||||
|
|
|
@ -0,0 +1,142 @@
|
|||
#ifndef __TARGET_CORE_USER_H
|
||||
#define __TARGET_CORE_USER_H
|
||||
|
||||
/* This header will be used by application too */
|
||||
|
||||
#include <linux/types.h>
|
||||
#include <linux/uio.h>
|
||||
|
||||
#ifndef __packed
|
||||
#define __packed __attribute__((packed))
|
||||
#endif
|
||||
|
||||
#define TCMU_VERSION "1.0"
|
||||
|
||||
/*
|
||||
* Ring Design
|
||||
* -----------
|
||||
*
|
||||
* The mmaped area is divided into three parts:
|
||||
* 1) The mailbox (struct tcmu_mailbox, below)
|
||||
* 2) The command ring
|
||||
* 3) Everything beyond the command ring (data)
|
||||
*
|
||||
* The mailbox tells userspace the offset of the command ring from the
|
||||
* start of the shared memory region, and how big the command ring is.
|
||||
*
|
||||
* The kernel passes SCSI commands to userspace by putting a struct
|
||||
* tcmu_cmd_entry in the ring, updating mailbox->cmd_head, and poking
|
||||
* userspace via uio's interrupt mechanism.
|
||||
*
|
||||
* tcmu_cmd_entry contains a header. If the header type is PAD,
|
||||
* userspace should skip hdr->length bytes (mod cmdr_size) to find the
|
||||
* next cmd_entry.
|
||||
*
|
||||
* Otherwise, the entry will contain offsets into the mmaped area that
|
||||
* contain the cdb and data buffers -- the latter accessible via the
|
||||
* iov array. iov addresses are also offsets into the shared area.
|
||||
*
|
||||
* When userspace is completed handling the command, set
|
||||
* entry->rsp.scsi_status, fill in rsp.sense_buffer if appropriate,
|
||||
* and also set mailbox->cmd_tail equal to the old cmd_tail plus
|
||||
* hdr->length, mod cmdr_size. If cmd_tail doesn't equal cmd_head, it
|
||||
* should process the next packet the same way, and so on.
|
||||
*/
|
||||
|
||||
#define TCMU_MAILBOX_VERSION 1
|
||||
#define ALIGN_SIZE 64 /* Should be enough for most CPUs */
|
||||
|
||||
struct tcmu_mailbox {
|
||||
__u16 version;
|
||||
__u16 flags;
|
||||
__u32 cmdr_off;
|
||||
__u32 cmdr_size;
|
||||
|
||||
__u32 cmd_head;
|
||||
|
||||
/* Updated by user. On its own cacheline */
|
||||
__u32 cmd_tail __attribute__((__aligned__(ALIGN_SIZE)));
|
||||
|
||||
} __packed;
|
||||
|
||||
enum tcmu_opcode {
|
||||
TCMU_OP_PAD = 0,
|
||||
TCMU_OP_CMD,
|
||||
};
|
||||
|
||||
/*
|
||||
* Only a few opcodes, and length is 8-byte aligned, so use low bits for opcode.
|
||||
*/
|
||||
struct tcmu_cmd_entry_hdr {
|
||||
__u32 len_op;
|
||||
} __packed;
|
||||
|
||||
#define TCMU_OP_MASK 0x7
|
||||
|
||||
static inline enum tcmu_opcode tcmu_hdr_get_op(struct tcmu_cmd_entry_hdr *hdr)
|
||||
{
|
||||
return hdr->len_op & TCMU_OP_MASK;
|
||||
}
|
||||
|
||||
static inline void tcmu_hdr_set_op(struct tcmu_cmd_entry_hdr *hdr, enum tcmu_opcode op)
|
||||
{
|
||||
hdr->len_op &= ~TCMU_OP_MASK;
|
||||
hdr->len_op |= (op & TCMU_OP_MASK);
|
||||
}
|
||||
|
||||
static inline __u32 tcmu_hdr_get_len(struct tcmu_cmd_entry_hdr *hdr)
|
||||
{
|
||||
return hdr->len_op & ~TCMU_OP_MASK;
|
||||
}
|
||||
|
||||
static inline void tcmu_hdr_set_len(struct tcmu_cmd_entry_hdr *hdr, __u32 len)
|
||||
{
|
||||
hdr->len_op &= TCMU_OP_MASK;
|
||||
hdr->len_op |= len;
|
||||
}
|
||||
|
||||
/* Currently the same as SCSI_SENSE_BUFFERSIZE */
|
||||
#define TCMU_SENSE_BUFFERSIZE 96
|
||||
|
||||
struct tcmu_cmd_entry {
|
||||
struct tcmu_cmd_entry_hdr hdr;
|
||||
|
||||
uint16_t cmd_id;
|
||||
uint16_t __pad1;
|
||||
|
||||
union {
|
||||
struct {
|
||||
uint64_t cdb_off;
|
||||
uint64_t iov_cnt;
|
||||
struct iovec iov[0];
|
||||
} req;
|
||||
struct {
|
||||
uint8_t scsi_status;
|
||||
uint8_t __pad1;
|
||||
uint16_t __pad2;
|
||||
uint32_t __pad3;
|
||||
char sense_buffer[TCMU_SENSE_BUFFERSIZE];
|
||||
} rsp;
|
||||
};
|
||||
|
||||
} __packed;
|
||||
|
||||
#define TCMU_OP_ALIGN_SIZE sizeof(uint64_t)
|
||||
|
||||
enum tcmu_genl_cmd {
|
||||
TCMU_CMD_UNSPEC,
|
||||
TCMU_CMD_ADDED_DEVICE,
|
||||
TCMU_CMD_REMOVED_DEVICE,
|
||||
__TCMU_CMD_MAX,
|
||||
};
|
||||
#define TCMU_CMD_MAX (__TCMU_CMD_MAX - 1)
|
||||
|
||||
enum tcmu_genl_attr {
|
||||
TCMU_ATTR_UNSPEC,
|
||||
TCMU_ATTR_DEVICE,
|
||||
TCMU_ATTR_MINOR,
|
||||
__TCMU_ATTR_MAX,
|
||||
};
|
||||
#define TCMU_ATTR_MAX (__TCMU_ATTR_MAX - 1)
|
||||
|
||||
#endif
|
Loading…
Reference in New Issue