List, This patch adds support for passthrough devices to tgtd. Please review/apply. >From 7b9518e0cd6ad4b2051c0bdbba2f5ea4aca324d0 Mon Sep 17 00:00:00 2001 From: Ronnie Sahlberg <ronniesahlberg@xxxxxxxxx> Date: Mon, 18 Aug 2008 15:59:59 +1000 Subject: [PATCH] add support for passthrough luns Signed-off-by: Ronnie Sahlberg <ronniesahlberg@xxxxxxxxx> --- usr/Makefile | 7 +- usr/bs_pt.c | 90 ++++++++++++++++++++++++++ usr/pt.c | 199 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ usr/target.c | 9 +-- usr/tgtd.h | 2 +- 5 files changed, 298 insertions(+), 9 deletions(-) create mode 100644 usr/bs_pt.c create mode 100644 usr/pt.c diff --git a/usr/Makefile b/usr/Makefile index 4245709..b7c7517 100644 --- a/usr/Makefile +++ b/usr/Makefile @@ -11,7 +11,7 @@ CFLAGS += -DISCSI TGTD_OBJS += $(addprefix iscsi/, conn.o param.o session.o \ iscsid.o target.o chap.o transport.o iscsi_tcp.o \ isns.o libcrc32c.o) -TGTD_OBJS += bs_rdwr.o bs_aio.o +TGTD_OBJS += bs_rdwr.o bs_aio.o bs_pt.o LIBS += -lcrypto ifneq ($(ISCSI_RDMA),) @@ -43,7 +43,7 @@ TGTD_OBJS += $(addprefix fcoe/,\ fcoe_if.o fcoe_dev.o \ sa_event.o sa_timer.o sa_hash_kern.o sa_state.o\ crc32_le.o crc32_le_tab.o) -TGTD_OBJS += bs_rdwr.o +TGTD_OBJS += bs_rdwr.o bs_pt.o endif INCLUDES += -I. -I../include -I$(KERNELSRC)/include @@ -57,7 +57,8 @@ LIBS += -lpthread PROGRAMS += tgtd tgtadm SCRIPTS += ../scripts/tgt-setup-lun TGTD_OBJS += tgtd.o mgmt.o target.o scsi.o log.o driver.o util.o work.o \ - parser.o spc.o sbc.o mmc.o osd.o scc.o smc.o ssc.o bs_ssc.o bs.o + parser.o spc.o sbc.o mmc.o osd.o pt.o scc.o \ + smc.o ssc.o bs_ssc.o bs.o MANPAGES = ../doc/manpages/tgtadm.8 ../doc/manpages/tgt-setup-lun.8 TGTD_DEP = $(TGTD_OBJS:.o=.d) diff --git a/usr/bs_pt.c b/usr/bs_pt.c new file mode 100644 index 0000000..299ba2c --- /dev/null +++ b/usr/bs_pt.c @@ -0,0 +1,90 @@ +/* + * Backing store for passthrough devices + * + * Copyright (C) 2008 Ronnie Sahlberg <ronniesahlberg@xxxxxxxxx> + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation, version 2 of the + * License. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA + * 02110-1301 USA + */ +#define _XOPEN_SOURCE 500 + +#include <errno.h> +#include <fcntl.h> +#include <inttypes.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <unistd.h> + +#include <linux/fs.h> +#include <sys/epoll.h> +#include <sys/ioctl.h> +#include <scsi/sg.h> + +#include "list.h" +#include "util.h" +#include "tgtd.h" +#include "scsi.h" +#include "bs_thread.h" + + + +static int open_scsi_device(const char *dev) +{ + int fd, vers; + + fd = open(dev, O_RDWR); + if (fd < 0) { + dprintf("ERROR could not open device %s\n", dev); + return -1; + } + if ((ioctl(fd, SG_GET_VERSION_NUM, &vers) < 0) || (vers < 30000)) { + dprintf("/dev is not an sg device, or old sg driver\n"); + close(fd); + return -1; + } + + return fd; +} + + +static int bs_passthrough_open(struct scsi_lu *lu, char *path, int *fd, + uint64_t *size) +{ + *fd = open_scsi_device(path); + + return 0; +} + +static void bs_passthrough_close(struct scsi_lu *lu) +{ + close(lu->fd); +} + +static int bs_passthrough_cmd_done(struct scsi_cmd *cmd) +{ + return 0; +} + +static struct backingstore_template passthrough_bst = { + .bs_name = "pt", + .bs_open = bs_passthrough_open, + .bs_close = bs_passthrough_close, +}; + +__attribute__((constructor)) static void bs_passthrough_constructor(void) +{ + register_backingstore_template(&passthrough_bst); +} diff --git a/usr/pt.c b/usr/pt.c new file mode 100644 index 0000000..480b4d7 --- /dev/null +++ b/usr/pt.c @@ -0,0 +1,199 @@ +/* + * SCSI passthrough module + * + * Copyright (C) 2008 Ronnie Sahlberg <ronniesahlberg@xxxxxxxxx> + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation, version 2 of the + * License. + * + * This program is distributed in the hope that it will be useful, but + * WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA + * 02110-1301 USA + */ +#include <errno.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <stdint.h> +#include <unistd.h> +#include <linux/fs.h> +#include <sys/stat.h> + +#include "list.h" +#include "util.h" +#include "tgtd.h" +#include <fcntl.h> +#include "target.h" +#include "tgtadm_error.h" +#include "driver.h" +#include "scsi.h" +#include "spc.h" +#include "tgtadm_error.h" +#include <sys/ioctl.h> +#include <scsi/sg.h> + + +static unsigned char scsi_command_size[8] = {6, 10, 10, 12, 16, 12, 10, 10}; + +#define COMMAND_SIZE(opcode) scsi_command_size[((opcode) >> 5) & 7] +#define CDB_SIZE(cmd) (((((cmd)->scb[0] >> 5) & 7) < 6) ? \ + COMMAND_SIZE((cmd)->scb[0]) : (cmd)->scb_len) + +#define SCSI_TIMEOUT 5000 /* ms */ + + +int scsi_io(int fd, unsigned char *cdb, unsigned char cdb_size, + int xfer_dir, unsigned char *data, int *data_size, + unsigned char *sense, unsigned int *sense_len) +{ + sg_io_hdr_t io_hdr; + + memset(&io_hdr, 0, sizeof(sg_io_hdr_t)); + io_hdr.interface_id = 'S'; + + /* CDB */ + io_hdr.cmdp = cdb; + io_hdr.cmd_len = cdb_size; + + /* Where to store the sense_data, if there was an error */ + io_hdr.sbp = sense; + io_hdr.mx_sb_len = *sense_len; + *sense_len = 0; + + /* Transfer direction, either in or out. Linux does not yet + support bidirectional SCSI transfers ? + */ + io_hdr.dxfer_direction = xfer_dir; + + /* Where to store the DATA IN/OUT from the device and how big the + buffer is + */ + io_hdr.dxferp = data; + io_hdr.dxfer_len = *data_size; + + /* SCSI timeout in ms */ + io_hdr.timeout = SCSI_TIMEOUT; + + + if (ioctl(fd, SG_IO, &io_hdr) < 0) + return -1; + + /* now for the error processing */ + if ((io_hdr.info & SG_INFO_OK_MASK) != SG_INFO_OK) { + if (io_hdr.sb_len_wr > 0) { + *sense_len = io_hdr.sb_len_wr; + return 0; + } + } + if (io_hdr.masked_status) + return -2; + + if (io_hdr.host_status) + return -3; + + if (io_hdr.driver_status) + return -4; + + return 0; +} + + + + +static int scsi_passthrough(int host_no, struct scsi_cmd *cmd) +{ + unsigned int sense_len = 32; + unsigned char sense[sense_len]; + int alloc_len; + uint8_t *data; + int direction; + + alloc_len = scsi_get_in_length(cmd); + if (alloc_len > 0) { + direction = SG_DXFER_FROM_DEV; + data = scsi_get_in_buffer(cmd); + } else { + alloc_len = scsi_get_out_length(cmd); + if (alloc_len > 0) { + direction = SG_DXFER_TO_DEV; + data = scsi_get_out_buffer(cmd); + } else { + direction = SG_DXFER_NONE; + data = NULL; + } + } + + scsi_io(cmd->dev->fd, + cmd->scb, + COMMAND_SIZE(cmd->scb[0]), + direction, + data, &alloc_len, sense, &sense_len); + if (sense_len) { + scsi_set_in_resid_by_actual(cmd, 0); + sense_data_build(cmd, sense[2], sense[12]<<8); + return SAM_STAT_CHECK_CONDITION; + } + + if (direction == SG_DXFER_FROM_DEV) + scsi_set_in_resid_by_actual(cmd, alloc_len); + + return SAM_STAT_GOOD; +} + +static int passthrough_lu_init(struct scsi_lu *lu) +{ + struct backingstore_template *bst; + + /* passthrough devices always use passthrough backingstore */ + bst = get_backingstore_template("pt"); + if (!bst) { + eprintf("failed to find bstype, pt\n"); + return TGTADM_INVALID_REQUEST; + } + lu->bst = bst; + + lu->attrs.device_type = TYPE_SPT; + + return 0; +} + +static int passthrough_lu_online(struct scsi_lu *lu) +{ + lu->attrs.online = 1; + + return 0; +} + +static int passthrough_lu_offline(struct scsi_lu *lu) +{ + lu->attrs.online = 0; + return 0; +} + +static struct device_type_template passthrough_template = { + .type = TYPE_SPT, + .lu_init = passthrough_lu_init, + .lu_config = spc_lu_config, + .lu_online = passthrough_lu_online, + .lu_offline = passthrough_lu_offline, + .lu_exit = spc_lu_exit, + .ops = { + [0x00 ... 0x9F] = {scsi_passthrough,}, + /* 0xA0 */ + {spc_report_luns,}, + [0xA1 ... 0xFF] = {scsi_passthrough,}, + } +}; + +__attribute__((constructor)) static void passthrough_init(void) +{ + device_type_register(&passthrough_template); +} diff --git a/usr/target.c b/usr/target.c index 70bf72a..e200471 100644 --- a/usr/target.c +++ b/usr/target.c @@ -1524,21 +1524,20 @@ static struct { {TYPE_ENCLOSURE, "enclosure"}, {TYPE_RBC, "rbc"}, {TYPE_OSD, "osd"}, - {TYPE_NO_LUN, "No LUN"} + {TYPE_NO_LUN, "No LUN"}, + {TYPE_SPT, "passthrough"} }; static char *print_type(int type) { int i; - char *name = NULL; for (i = 0; i < ARRAY_SIZE(disk_type_names); i++) { if (disk_type_names[i].value == type) { - name = disk_type_names[i].name; - break; + return disk_type_names[i].name; } } - return name; + return "Unknown type"; } diff --git a/usr/tgtd.h b/usr/tgtd.h index 4febcd3..a649129 100644 --- a/usr/tgtd.h +++ b/usr/tgtd.h @@ -60,7 +60,7 @@ struct lu_phy_attr { uint16_t version_desc[VERSION_DESCRIPTOR_LEN]; - char device_type; /* Peripheral device type */ + unsigned char device_type;/* Peripheral device type */ char qualifier; /* Peripheral Qualifier */ char removable; /* Removable media */ char online; /* Logical Unit online */ -- 1.5.5
Attachment:
0001-add-support-for-passthrough-luns.patch.gz
Description: GNU Zip compressed data