[SCSI] ibmvfc: Add FC Passthru support
Adds support for FC passthru via BSG. Signed-off-by: Brian King <brking@linux.vnet.ibm.com> Signed-off-by: James Bottomley <James.Bottomley@suse.de>
This commit is contained in:
parent
4a5c4a5ed2
commit
d31429e151
|
@ -39,6 +39,7 @@
|
||||||
#include <scsi/scsi_device.h>
|
#include <scsi/scsi_device.h>
|
||||||
#include <scsi/scsi_tcq.h>
|
#include <scsi/scsi_tcq.h>
|
||||||
#include <scsi/scsi_transport_fc.h>
|
#include <scsi/scsi_transport_fc.h>
|
||||||
|
#include <scsi/scsi_bsg_fc.h>
|
||||||
#include "ibmvfc.h"
|
#include "ibmvfc.h"
|
||||||
|
|
||||||
static unsigned int init_timeout = IBMVFC_INIT_TIMEOUT;
|
static unsigned int init_timeout = IBMVFC_INIT_TIMEOUT;
|
||||||
|
@ -1674,6 +1675,276 @@ static void ibmvfc_sync_completion(struct ibmvfc_event *evt)
|
||||||
complete(&evt->comp);
|
complete(&evt->comp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ibmvfc_bsg_timeout_done - Completion handler for cancelling BSG commands
|
||||||
|
* @evt: struct ibmvfc_event
|
||||||
|
*
|
||||||
|
**/
|
||||||
|
static void ibmvfc_bsg_timeout_done(struct ibmvfc_event *evt)
|
||||||
|
{
|
||||||
|
struct ibmvfc_host *vhost = evt->vhost;
|
||||||
|
|
||||||
|
ibmvfc_free_event(evt);
|
||||||
|
vhost->aborting_passthru = 0;
|
||||||
|
dev_info(vhost->dev, "Passthru command cancelled\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ibmvfc_bsg_timeout - Handle a BSG timeout
|
||||||
|
* @job: struct fc_bsg_job that timed out
|
||||||
|
*
|
||||||
|
* Returns:
|
||||||
|
* 0 on success / other on failure
|
||||||
|
**/
|
||||||
|
static int ibmvfc_bsg_timeout(struct fc_bsg_job *job)
|
||||||
|
{
|
||||||
|
struct ibmvfc_host *vhost = shost_priv(job->shost);
|
||||||
|
unsigned long port_id = (unsigned long)job->dd_data;
|
||||||
|
struct ibmvfc_event *evt;
|
||||||
|
struct ibmvfc_tmf *tmf;
|
||||||
|
unsigned long flags;
|
||||||
|
int rc;
|
||||||
|
|
||||||
|
ENTER;
|
||||||
|
spin_lock_irqsave(vhost->host->host_lock, flags);
|
||||||
|
if (vhost->aborting_passthru || vhost->state != IBMVFC_ACTIVE) {
|
||||||
|
__ibmvfc_reset_host(vhost);
|
||||||
|
spin_unlock_irqrestore(vhost->host->host_lock, flags);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
vhost->aborting_passthru = 1;
|
||||||
|
evt = ibmvfc_get_event(vhost);
|
||||||
|
ibmvfc_init_event(evt, ibmvfc_bsg_timeout_done, IBMVFC_MAD_FORMAT);
|
||||||
|
|
||||||
|
tmf = &evt->iu.tmf;
|
||||||
|
memset(tmf, 0, sizeof(*tmf));
|
||||||
|
tmf->common.version = 1;
|
||||||
|
tmf->common.opcode = IBMVFC_TMF_MAD;
|
||||||
|
tmf->common.length = sizeof(*tmf);
|
||||||
|
tmf->scsi_id = port_id;
|
||||||
|
tmf->cancel_key = IBMVFC_PASSTHRU_CANCEL_KEY;
|
||||||
|
tmf->my_cancel_key = IBMVFC_INTERNAL_CANCEL_KEY;
|
||||||
|
rc = ibmvfc_send_event(evt, vhost, default_timeout);
|
||||||
|
|
||||||
|
if (rc != 0) {
|
||||||
|
vhost->aborting_passthru = 0;
|
||||||
|
dev_err(vhost->dev, "Failed to send cancel event. rc=%d\n", rc);
|
||||||
|
rc = -EIO;
|
||||||
|
} else
|
||||||
|
dev_info(vhost->dev, "Cancelling passthru command to port id 0x%lx\n",
|
||||||
|
port_id);
|
||||||
|
|
||||||
|
spin_unlock_irqrestore(vhost->host->host_lock, flags);
|
||||||
|
|
||||||
|
LEAVE;
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ibmvfc_bsg_plogi - PLOGI into a target to handle a BSG command
|
||||||
|
* @vhost: struct ibmvfc_host to send command
|
||||||
|
* @port_id: port ID to send command
|
||||||
|
*
|
||||||
|
* Returns:
|
||||||
|
* 0 on success / other on failure
|
||||||
|
**/
|
||||||
|
static int ibmvfc_bsg_plogi(struct ibmvfc_host *vhost, unsigned int port_id)
|
||||||
|
{
|
||||||
|
struct ibmvfc_port_login *plogi;
|
||||||
|
struct ibmvfc_target *tgt;
|
||||||
|
struct ibmvfc_event *evt;
|
||||||
|
union ibmvfc_iu rsp_iu;
|
||||||
|
unsigned long flags;
|
||||||
|
int rc = 0, issue_login = 1;
|
||||||
|
|
||||||
|
ENTER;
|
||||||
|
spin_lock_irqsave(vhost->host->host_lock, flags);
|
||||||
|
list_for_each_entry(tgt, &vhost->targets, queue) {
|
||||||
|
if (tgt->scsi_id == port_id) {
|
||||||
|
issue_login = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!issue_login)
|
||||||
|
goto unlock_out;
|
||||||
|
if (unlikely((rc = ibmvfc_host_chkready(vhost))))
|
||||||
|
goto unlock_out;
|
||||||
|
|
||||||
|
evt = ibmvfc_get_event(vhost);
|
||||||
|
ibmvfc_init_event(evt, ibmvfc_sync_completion, IBMVFC_MAD_FORMAT);
|
||||||
|
plogi = &evt->iu.plogi;
|
||||||
|
memset(plogi, 0, sizeof(*plogi));
|
||||||
|
plogi->common.version = 1;
|
||||||
|
plogi->common.opcode = IBMVFC_PORT_LOGIN;
|
||||||
|
plogi->common.length = sizeof(*plogi);
|
||||||
|
plogi->scsi_id = port_id;
|
||||||
|
evt->sync_iu = &rsp_iu;
|
||||||
|
init_completion(&evt->comp);
|
||||||
|
|
||||||
|
rc = ibmvfc_send_event(evt, vhost, default_timeout);
|
||||||
|
spin_unlock_irqrestore(vhost->host->host_lock, flags);
|
||||||
|
|
||||||
|
if (rc)
|
||||||
|
return -EIO;
|
||||||
|
|
||||||
|
wait_for_completion(&evt->comp);
|
||||||
|
|
||||||
|
if (rsp_iu.plogi.common.status)
|
||||||
|
rc = -EIO;
|
||||||
|
|
||||||
|
spin_lock_irqsave(vhost->host->host_lock, flags);
|
||||||
|
ibmvfc_free_event(evt);
|
||||||
|
unlock_out:
|
||||||
|
spin_unlock_irqrestore(vhost->host->host_lock, flags);
|
||||||
|
LEAVE;
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ibmvfc_bsg_request - Handle a BSG request
|
||||||
|
* @job: struct fc_bsg_job to be executed
|
||||||
|
*
|
||||||
|
* Returns:
|
||||||
|
* 0 on success / other on failure
|
||||||
|
**/
|
||||||
|
static int ibmvfc_bsg_request(struct fc_bsg_job *job)
|
||||||
|
{
|
||||||
|
struct ibmvfc_host *vhost = shost_priv(job->shost);
|
||||||
|
struct fc_rport *rport = job->rport;
|
||||||
|
struct ibmvfc_passthru_mad *mad;
|
||||||
|
struct ibmvfc_event *evt;
|
||||||
|
union ibmvfc_iu rsp_iu;
|
||||||
|
unsigned long flags, port_id = -1;
|
||||||
|
unsigned int code = job->request->msgcode;
|
||||||
|
int rc = 0, req_seg, rsp_seg, issue_login = 0;
|
||||||
|
u32 fc_flags, rsp_len;
|
||||||
|
|
||||||
|
ENTER;
|
||||||
|
job->reply->reply_payload_rcv_len = 0;
|
||||||
|
if (rport)
|
||||||
|
port_id = rport->port_id;
|
||||||
|
|
||||||
|
switch (code) {
|
||||||
|
case FC_BSG_HST_ELS_NOLOGIN:
|
||||||
|
port_id = (job->request->rqst_data.h_els.port_id[0] << 16) |
|
||||||
|
(job->request->rqst_data.h_els.port_id[1] << 8) |
|
||||||
|
job->request->rqst_data.h_els.port_id[2];
|
||||||
|
case FC_BSG_RPT_ELS:
|
||||||
|
fc_flags = IBMVFC_FC_ELS;
|
||||||
|
break;
|
||||||
|
case FC_BSG_HST_CT:
|
||||||
|
issue_login = 1;
|
||||||
|
port_id = (job->request->rqst_data.h_ct.port_id[0] << 16) |
|
||||||
|
(job->request->rqst_data.h_ct.port_id[1] << 8) |
|
||||||
|
job->request->rqst_data.h_ct.port_id[2];
|
||||||
|
case FC_BSG_RPT_CT:
|
||||||
|
fc_flags = IBMVFC_FC_CT_IU;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return -ENOTSUPP;
|
||||||
|
};
|
||||||
|
|
||||||
|
if (port_id == -1)
|
||||||
|
return -EINVAL;
|
||||||
|
if (!mutex_trylock(&vhost->passthru_mutex))
|
||||||
|
return -EBUSY;
|
||||||
|
|
||||||
|
job->dd_data = (void *)port_id;
|
||||||
|
req_seg = dma_map_sg(vhost->dev, job->request_payload.sg_list,
|
||||||
|
job->request_payload.sg_cnt, DMA_TO_DEVICE);
|
||||||
|
|
||||||
|
if (!req_seg) {
|
||||||
|
mutex_unlock(&vhost->passthru_mutex);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
rsp_seg = dma_map_sg(vhost->dev, job->reply_payload.sg_list,
|
||||||
|
job->reply_payload.sg_cnt, DMA_FROM_DEVICE);
|
||||||
|
|
||||||
|
if (!rsp_seg) {
|
||||||
|
dma_unmap_sg(vhost->dev, job->request_payload.sg_list,
|
||||||
|
job->request_payload.sg_cnt, DMA_TO_DEVICE);
|
||||||
|
mutex_unlock(&vhost->passthru_mutex);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (req_seg > 1 || rsp_seg > 1) {
|
||||||
|
rc = -EINVAL;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (issue_login)
|
||||||
|
rc = ibmvfc_bsg_plogi(vhost, port_id);
|
||||||
|
|
||||||
|
spin_lock_irqsave(vhost->host->host_lock, flags);
|
||||||
|
|
||||||
|
if (unlikely(rc || (rport && (rc = fc_remote_port_chkready(rport)))) ||
|
||||||
|
unlikely((rc = ibmvfc_host_chkready(vhost)))) {
|
||||||
|
spin_unlock_irqrestore(vhost->host->host_lock, flags);
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
evt = ibmvfc_get_event(vhost);
|
||||||
|
ibmvfc_init_event(evt, ibmvfc_sync_completion, IBMVFC_MAD_FORMAT);
|
||||||
|
mad = &evt->iu.passthru;
|
||||||
|
|
||||||
|
memset(mad, 0, sizeof(*mad));
|
||||||
|
mad->common.version = 1;
|
||||||
|
mad->common.opcode = IBMVFC_PASSTHRU;
|
||||||
|
mad->common.length = sizeof(*mad) - sizeof(mad->fc_iu) - sizeof(mad->iu);
|
||||||
|
|
||||||
|
mad->cmd_ioba.va = (u64)evt->crq.ioba +
|
||||||
|
offsetof(struct ibmvfc_passthru_mad, iu);
|
||||||
|
mad->cmd_ioba.len = sizeof(mad->iu);
|
||||||
|
|
||||||
|
mad->iu.cmd_len = job->request_payload.payload_len;
|
||||||
|
mad->iu.rsp_len = job->reply_payload.payload_len;
|
||||||
|
mad->iu.flags = fc_flags;
|
||||||
|
mad->iu.cancel_key = IBMVFC_PASSTHRU_CANCEL_KEY;
|
||||||
|
|
||||||
|
mad->iu.cmd.va = sg_dma_address(job->request_payload.sg_list);
|
||||||
|
mad->iu.cmd.len = sg_dma_len(job->request_payload.sg_list);
|
||||||
|
mad->iu.rsp.va = sg_dma_address(job->reply_payload.sg_list);
|
||||||
|
mad->iu.rsp.len = sg_dma_len(job->reply_payload.sg_list);
|
||||||
|
mad->iu.scsi_id = port_id;
|
||||||
|
mad->iu.tag = (u64)evt;
|
||||||
|
rsp_len = mad->iu.rsp.len;
|
||||||
|
|
||||||
|
evt->sync_iu = &rsp_iu;
|
||||||
|
init_completion(&evt->comp);
|
||||||
|
rc = ibmvfc_send_event(evt, vhost, 0);
|
||||||
|
spin_unlock_irqrestore(vhost->host->host_lock, flags);
|
||||||
|
|
||||||
|
if (rc) {
|
||||||
|
rc = -EIO;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
wait_for_completion(&evt->comp);
|
||||||
|
|
||||||
|
if (rsp_iu.passthru.common.status)
|
||||||
|
rc = -EIO;
|
||||||
|
else
|
||||||
|
job->reply->reply_payload_rcv_len = rsp_len;
|
||||||
|
|
||||||
|
spin_lock_irqsave(vhost->host->host_lock, flags);
|
||||||
|
ibmvfc_free_event(evt);
|
||||||
|
spin_unlock_irqrestore(vhost->host->host_lock, flags);
|
||||||
|
job->reply->result = rc;
|
||||||
|
job->job_done(job);
|
||||||
|
rc = 0;
|
||||||
|
out:
|
||||||
|
dma_unmap_sg(vhost->dev, job->request_payload.sg_list,
|
||||||
|
job->request_payload.sg_cnt, DMA_TO_DEVICE);
|
||||||
|
dma_unmap_sg(vhost->dev, job->reply_payload.sg_list,
|
||||||
|
job->reply_payload.sg_cnt, DMA_FROM_DEVICE);
|
||||||
|
mutex_unlock(&vhost->passthru_mutex);
|
||||||
|
LEAVE;
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ibmvfc_reset_device - Reset the device with the specified reset type
|
* ibmvfc_reset_device - Reset the device with the specified reset type
|
||||||
* @sdev: scsi device to reset
|
* @sdev: scsi device to reset
|
||||||
|
@ -3918,6 +4189,8 @@ static void ibmvfc_tgt_add_rport(struct ibmvfc_target *tgt)
|
||||||
rport->supported_classes |= FC_COS_CLASS2;
|
rport->supported_classes |= FC_COS_CLASS2;
|
||||||
if (tgt->service_parms.class3_parms[0] & 0x80000000)
|
if (tgt->service_parms.class3_parms[0] & 0x80000000)
|
||||||
rport->supported_classes |= FC_COS_CLASS3;
|
rport->supported_classes |= FC_COS_CLASS3;
|
||||||
|
if (rport->rqst_q)
|
||||||
|
blk_queue_max_hw_segments(rport->rqst_q, 1);
|
||||||
} else
|
} else
|
||||||
tgt_dbg(tgt, "rport add failed\n");
|
tgt_dbg(tgt, "rport add failed\n");
|
||||||
spin_unlock_irqrestore(vhost->host->host_lock, flags);
|
spin_unlock_irqrestore(vhost->host->host_lock, flags);
|
||||||
|
@ -4357,6 +4630,7 @@ static int ibmvfc_probe(struct vio_dev *vdev, const struct vio_device_id *id)
|
||||||
init_waitqueue_head(&vhost->work_wait_q);
|
init_waitqueue_head(&vhost->work_wait_q);
|
||||||
init_waitqueue_head(&vhost->init_wait_q);
|
init_waitqueue_head(&vhost->init_wait_q);
|
||||||
INIT_WORK(&vhost->rport_add_work_q, ibmvfc_rport_add_thread);
|
INIT_WORK(&vhost->rport_add_work_q, ibmvfc_rport_add_thread);
|
||||||
|
mutex_init(&vhost->passthru_mutex);
|
||||||
|
|
||||||
if ((rc = ibmvfc_alloc_mem(vhost)))
|
if ((rc = ibmvfc_alloc_mem(vhost)))
|
||||||
goto free_scsi_host;
|
goto free_scsi_host;
|
||||||
|
@ -4389,6 +4663,8 @@ static int ibmvfc_probe(struct vio_dev *vdev, const struct vio_device_id *id)
|
||||||
goto remove_shost;
|
goto remove_shost;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (shost_to_fc_host(shost)->rqst_q)
|
||||||
|
blk_queue_max_hw_segments(shost_to_fc_host(shost)->rqst_q, 1);
|
||||||
dev_set_drvdata(dev, vhost);
|
dev_set_drvdata(dev, vhost);
|
||||||
spin_lock(&ibmvfc_driver_lock);
|
spin_lock(&ibmvfc_driver_lock);
|
||||||
list_add_tail(&vhost->queue, &ibmvfc_head);
|
list_add_tail(&vhost->queue, &ibmvfc_head);
|
||||||
|
@ -4517,6 +4793,9 @@ static struct fc_function_template ibmvfc_transport_functions = {
|
||||||
|
|
||||||
.get_starget_port_id = ibmvfc_get_starget_port_id,
|
.get_starget_port_id = ibmvfc_get_starget_port_id,
|
||||||
.show_starget_port_id = 1,
|
.show_starget_port_id = 1,
|
||||||
|
|
||||||
|
.bsg_request = ibmvfc_bsg_request,
|
||||||
|
.bsg_timeout = ibmvfc_bsg_timeout,
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -58,9 +58,10 @@
|
||||||
* 1 for ERP
|
* 1 for ERP
|
||||||
* 1 for initialization
|
* 1 for initialization
|
||||||
* 1 for NPIV Logout
|
* 1 for NPIV Logout
|
||||||
|
* 2 for BSG passthru
|
||||||
* 2 for each discovery thread
|
* 2 for each discovery thread
|
||||||
*/
|
*/
|
||||||
#define IBMVFC_NUM_INTERNAL_REQ (1 + 1 + 1 + (disc_threads * 2))
|
#define IBMVFC_NUM_INTERNAL_REQ (1 + 1 + 1 + 2 + (disc_threads * 2))
|
||||||
|
|
||||||
#define IBMVFC_MAD_SUCCESS 0x00
|
#define IBMVFC_MAD_SUCCESS 0x00
|
||||||
#define IBMVFC_MAD_NOT_SUPPORTED 0xF1
|
#define IBMVFC_MAD_NOT_SUPPORTED 0xF1
|
||||||
|
@ -466,7 +467,10 @@ struct ibmvfc_passthru_iu {
|
||||||
u16 error;
|
u16 error;
|
||||||
u32 flags;
|
u32 flags;
|
||||||
#define IBMVFC_FC_ELS 0x01
|
#define IBMVFC_FC_ELS 0x01
|
||||||
|
#define IBMVFC_FC_CT_IU 0x02
|
||||||
u32 cancel_key;
|
u32 cancel_key;
|
||||||
|
#define IBMVFC_PASSTHRU_CANCEL_KEY 0x80000000
|
||||||
|
#define IBMVFC_INTERNAL_CANCEL_KEY 0x80000001
|
||||||
u32 reserved;
|
u32 reserved;
|
||||||
struct srp_direct_buf cmd;
|
struct srp_direct_buf cmd;
|
||||||
struct srp_direct_buf rsp;
|
struct srp_direct_buf rsp;
|
||||||
|
@ -693,6 +697,7 @@ struct ibmvfc_host {
|
||||||
int disc_buf_sz;
|
int disc_buf_sz;
|
||||||
int log_level;
|
int log_level;
|
||||||
struct ibmvfc_discover_targets_buf *disc_buf;
|
struct ibmvfc_discover_targets_buf *disc_buf;
|
||||||
|
struct mutex passthru_mutex;
|
||||||
int task_set;
|
int task_set;
|
||||||
int init_retries;
|
int init_retries;
|
||||||
int discovery_threads;
|
int discovery_threads;
|
||||||
|
@ -702,6 +707,7 @@ struct ibmvfc_host {
|
||||||
int delay_init;
|
int delay_init;
|
||||||
int scan_complete;
|
int scan_complete;
|
||||||
int logged_in;
|
int logged_in;
|
||||||
|
int aborting_passthru;
|
||||||
int events_to_log;
|
int events_to_log;
|
||||||
#define IBMVFC_AE_LINKUP 0x0001
|
#define IBMVFC_AE_LINKUP 0x0001
|
||||||
#define IBMVFC_AE_LINKDOWN 0x0002
|
#define IBMVFC_AE_LINKDOWN 0x0002
|
||||||
|
|
Loading…
Reference in New Issue