Merge git://git.kernel.org/pub/scm/linux/kernel/git/jejb/scsi-misc-2.6
* git://git.kernel.org/pub/scm/linux/kernel/git/jejb/scsi-misc-2.6: (222 commits) [SCSI] zfcp: Remove flag ZFCP_STATUS_FSFREQ_TMFUNCNOTSUPP [SCSI] zfcp: Activate fc4s attributes for zfcp in FC transport class [SCSI] zfcp: Block scsi_eh thread for rport state BLOCKED [SCSI] zfcp: Update FSF error reporting [SCSI] zfcp: Improve ELS ADISC handling [SCSI] zfcp: Simplify handling of ct and els requests [SCSI] zfcp: Remove ZFCP_DID_MASK [SCSI] zfcp: Move WKA port to zfcp FC code [SCSI] zfcp: Use common code definitions for FC CT structs [SCSI] zfcp: Use common code definitions for FC ELS structs [SCSI] zfcp: Update FCP protocol related code [SCSI] zfcp: Dont fail SCSI commands when transitioning to blocked fc_rport [SCSI] zfcp: Assign scheduled work to driver queue [SCSI] zfcp: Remove STATUS_COMMON_REMOVE flag as it is not required anymore [SCSI] zfcp: Implement module unloading [SCSI] zfcp: Merge trace code for fsf requests in one function [SCSI] zfcp: Access ports and units with container_of in sysfs code [SCSI] zfcp: Remove suspend callback [SCSI] zfcp: Remove global config_mutex [SCSI] zfcp: Replace local reference counting with common kref ...
This commit is contained in:
commit
382f51fe2f
|
@ -0,0 +1,8 @@
|
||||||
|
What: /sys/bus/pci/drivers/qla2xxx/.../devices/*
|
||||||
|
Date: September 2009
|
||||||
|
Contact: QLogic Linux Driver <linux-driver@qlogic.com>
|
||||||
|
Description: qla2xxx-udev.sh currently looks for uevent CHANGE events to
|
||||||
|
signal a firmware-dump has been generated by the driver and is
|
||||||
|
ready for retrieval.
|
||||||
|
Users: qla2xxx-udev.sh. Proposed changes should be mailed to
|
||||||
|
linux-driver@qlogic.com
|
|
@ -1,3 +1,65 @@
|
||||||
|
1 Release Date : Tues. July 28, 2009 10:12:45 PST 2009 -
|
||||||
|
(emaild-id:megaraidlinux@lsi.com)
|
||||||
|
Bo Yang
|
||||||
|
|
||||||
|
2 Current Version : 00.00.04.12
|
||||||
|
3 Older Version : 00.00.04.10
|
||||||
|
|
||||||
|
1. Change the AEN sys PD update from scsi_scan to
|
||||||
|
scsi_add_device and scsi_remove_device.
|
||||||
|
2. Takeoff the debug print-out in aen_polling routine.
|
||||||
|
|
||||||
|
1 Release Date : Thur. July 02, 2009 10:12:45 PST 2009 -
|
||||||
|
(emaild-id:megaraidlinux@lsi.com)
|
||||||
|
Bo Yang
|
||||||
|
|
||||||
|
2 Current Version : 00.00.04.10
|
||||||
|
3 Older Version : 00.00.04.08
|
||||||
|
|
||||||
|
1. Add the 3 mins timeout during the controller initialize.
|
||||||
|
2. Add the fix for 64bit sense date errors.
|
||||||
|
|
||||||
|
1 Release Date : Tues. May 05, 2009 10:12:45 PST 2009 -
|
||||||
|
(emaild-id:megaraidlinux@lsi.com)
|
||||||
|
Bo Yang
|
||||||
|
|
||||||
|
2 Current Version : 00.00.04.08
|
||||||
|
3 Older Version : 00.00.04.06
|
||||||
|
|
||||||
|
1. Add the fix of pending in FW after deleted the logic drives.
|
||||||
|
2. Add the fix of deallocating memory after get pdlist.
|
||||||
|
|
||||||
|
1 Release Date : Tues. March 26, 2009 10:12:45 PST 2009 -
|
||||||
|
(emaild-id:megaraidlinux@lsi.com)
|
||||||
|
Bo Yang
|
||||||
|
|
||||||
|
2 Current Version : 00.00.04.06
|
||||||
|
3 Older Version : 00.00.04.04
|
||||||
|
|
||||||
|
1. Add the fix of the driver cmd empty fix of the driver cmd empty.
|
||||||
|
2. Add the fix of the driver MSM AEN CMD cause the system slow.
|
||||||
|
|
||||||
|
1 Release Date : Tues. March 03, 2009 10:12:45 PST 2009 -
|
||||||
|
(emaild-id:megaraidlinux@lsi.com)
|
||||||
|
Bo Yang
|
||||||
|
|
||||||
|
2 Current Version : 00.00.04.04
|
||||||
|
3 Older Version : 00.00.04.01
|
||||||
|
|
||||||
|
1. Add the Tape drive fix to the driver: If the command is for
|
||||||
|
the tape device, set the pthru timeout to the os layer timeout value.
|
||||||
|
|
||||||
|
2. Add Poll_wait mechanism to Gen-2 Linux driv.
|
||||||
|
In the aen handler, driver needs to wakeup poll handler similar to
|
||||||
|
the way it raises SIGIO.
|
||||||
|
|
||||||
|
3. Add new controller new SAS2 support to the driver.
|
||||||
|
|
||||||
|
4. Report the unconfigured PD (system PD) to OS.
|
||||||
|
|
||||||
|
5. Add the IEEE SGL support to the driver
|
||||||
|
|
||||||
|
6. Reasign the Application cmds to SAS2 controller
|
||||||
|
|
||||||
1 Release Date : Thur.July. 24 11:41:51 PST 2008 -
|
1 Release Date : Thur.July. 24 11:41:51 PST 2008 -
|
||||||
(emaild-id:megaraidlinux@lsi.com)
|
(emaild-id:megaraidlinux@lsi.com)
|
||||||
|
|
15
MAINTAINERS
15
MAINTAINERS
|
@ -4196,6 +4196,13 @@ W: http://www.pmc-sierra.com/
|
||||||
S: Supported
|
S: Supported
|
||||||
F: drivers/scsi/pmcraid.*
|
F: drivers/scsi/pmcraid.*
|
||||||
|
|
||||||
|
PMC SIERRA PM8001 DRIVER
|
||||||
|
M: jack_wang@usish.com
|
||||||
|
M: lindar_liu@usish.com
|
||||||
|
L: linux-scsi@vger.kernel.org
|
||||||
|
S: Supported
|
||||||
|
F: drivers/scsi/pm8001/
|
||||||
|
|
||||||
POSIX CLOCKS and TIMERS
|
POSIX CLOCKS and TIMERS
|
||||||
M: Thomas Gleixner <tglx@linutronix.de>
|
M: Thomas Gleixner <tglx@linutronix.de>
|
||||||
S: Supported
|
S: Supported
|
||||||
|
@ -5772,6 +5779,14 @@ L: netdev@vger.kernel.org
|
||||||
S: Maintained
|
S: Maintained
|
||||||
F: drivers/net/vmxnet3/
|
F: drivers/net/vmxnet3/
|
||||||
|
|
||||||
|
VMware PVSCSI driver
|
||||||
|
M: Alok Kataria <akataria@vmware.com>
|
||||||
|
M: VMware PV-Drivers <pv-drivers@vmware.com>
|
||||||
|
L: linux-scsi@vger.kernel.org
|
||||||
|
S: Maintained
|
||||||
|
F: drivers/scsi/vmw_pvscsi.c
|
||||||
|
F: drivers/scsi/vmw_pvscsi.h
|
||||||
|
|
||||||
VOLTAGE AND CURRENT REGULATOR FRAMEWORK
|
VOLTAGE AND CURRENT REGULATOR FRAMEWORK
|
||||||
M: Liam Girdwood <lrg@slimlogic.co.uk>
|
M: Liam Girdwood <lrg@slimlogic.co.uk>
|
||||||
M: Mark Brown <broonie@opensource.wolfsonmicro.com>
|
M: Mark Brown <broonie@opensource.wolfsonmicro.com>
|
||||||
|
|
|
@ -1208,6 +1208,7 @@ void ata_scsi_slave_destroy(struct scsi_device *sdev)
|
||||||
* ata_scsi_change_queue_depth - SCSI callback for queue depth config
|
* ata_scsi_change_queue_depth - SCSI callback for queue depth config
|
||||||
* @sdev: SCSI device to configure queue depth for
|
* @sdev: SCSI device to configure queue depth for
|
||||||
* @queue_depth: new queue depth
|
* @queue_depth: new queue depth
|
||||||
|
* @reason: calling context
|
||||||
*
|
*
|
||||||
* This is libata standard hostt->change_queue_depth callback.
|
* This is libata standard hostt->change_queue_depth callback.
|
||||||
* SCSI will call into this callback when user tries to set queue
|
* SCSI will call into this callback when user tries to set queue
|
||||||
|
@ -1219,12 +1220,16 @@ void ata_scsi_slave_destroy(struct scsi_device *sdev)
|
||||||
* RETURNS:
|
* RETURNS:
|
||||||
* Newly configured queue depth.
|
* Newly configured queue depth.
|
||||||
*/
|
*/
|
||||||
int ata_scsi_change_queue_depth(struct scsi_device *sdev, int queue_depth)
|
int ata_scsi_change_queue_depth(struct scsi_device *sdev, int queue_depth,
|
||||||
|
int reason)
|
||||||
{
|
{
|
||||||
struct ata_port *ap = ata_shost_to_port(sdev->host);
|
struct ata_port *ap = ata_shost_to_port(sdev->host);
|
||||||
struct ata_device *dev;
|
struct ata_device *dev;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
|
if (reason != SCSI_QDEPTH_DEFAULT)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (queue_depth < 1 || queue_depth == sdev->queue_depth)
|
if (queue_depth < 1 || queue_depth == sdev->queue_depth)
|
||||||
return sdev->queue_depth;
|
return sdev->queue_depth;
|
||||||
|
|
||||||
|
|
|
@ -1975,7 +1975,7 @@ static int nv_swncq_slave_config(struct scsi_device *sdev)
|
||||||
ata_id_c_string(dev->id, model_num, ATA_ID_PROD, sizeof(model_num));
|
ata_id_c_string(dev->id, model_num, ATA_ID_PROD, sizeof(model_num));
|
||||||
|
|
||||||
if (strncmp(model_num, "Maxtor", 6) == 0) {
|
if (strncmp(model_num, "Maxtor", 6) == 0) {
|
||||||
ata_scsi_change_queue_depth(sdev, 1);
|
ata_scsi_change_queue_depth(sdev, 1, SCSI_QDEPTH_DEFAULT);
|
||||||
ata_dev_printk(dev, KERN_NOTICE,
|
ata_dev_printk(dev, KERN_NOTICE,
|
||||||
"Disabling SWNCQ mode (depth %x)\n", sdev->queue_depth);
|
"Disabling SWNCQ mode (depth %x)\n", sdev->queue_depth);
|
||||||
}
|
}
|
||||||
|
|
|
@ -625,6 +625,7 @@ static struct iscsi_transport iscsi_iser_transport = {
|
||||||
ISCSI_USERNAME | ISCSI_PASSWORD |
|
ISCSI_USERNAME | ISCSI_PASSWORD |
|
||||||
ISCSI_USERNAME_IN | ISCSI_PASSWORD_IN |
|
ISCSI_USERNAME_IN | ISCSI_PASSWORD_IN |
|
||||||
ISCSI_FAST_ABORT | ISCSI_ABORT_TMO |
|
ISCSI_FAST_ABORT | ISCSI_ABORT_TMO |
|
||||||
|
ISCSI_LU_RESET_TMO | ISCSI_TGT_RESET_TMO |
|
||||||
ISCSI_PING_TMO | ISCSI_RECV_TMO |
|
ISCSI_PING_TMO | ISCSI_RECV_TMO |
|
||||||
ISCSI_IFACE_NAME | ISCSI_INITIATOR_NAME,
|
ISCSI_IFACE_NAME | ISCSI_INITIATOR_NAME,
|
||||||
.host_param_mask = ISCSI_HOST_HWADDRESS |
|
.host_param_mask = ISCSI_HOST_HWADDRESS |
|
||||||
|
|
|
@ -1116,8 +1116,9 @@ static int pg_init_limit_reached(struct multipath *m, struct pgpath *pgpath)
|
||||||
return limit_reached;
|
return limit_reached;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void pg_init_done(struct dm_path *path, int errors)
|
static void pg_init_done(void *data, int errors)
|
||||||
{
|
{
|
||||||
|
struct dm_path *path = data;
|
||||||
struct pgpath *pgpath = path_to_pgpath(path);
|
struct pgpath *pgpath = path_to_pgpath(path);
|
||||||
struct priority_group *pg = pgpath->pg;
|
struct priority_group *pg = pgpath->pg;
|
||||||
struct multipath *m = pg->m;
|
struct multipath *m = pg->m;
|
||||||
|
@ -1183,12 +1184,11 @@ static void pg_init_done(struct dm_path *path, int errors)
|
||||||
|
|
||||||
static void activate_path(struct work_struct *work)
|
static void activate_path(struct work_struct *work)
|
||||||
{
|
{
|
||||||
int ret;
|
|
||||||
struct pgpath *pgpath =
|
struct pgpath *pgpath =
|
||||||
container_of(work, struct pgpath, activate_path);
|
container_of(work, struct pgpath, activate_path);
|
||||||
|
|
||||||
ret = scsi_dh_activate(bdev_get_queue(pgpath->path.dev->bdev));
|
scsi_dh_activate(bdev_get_queue(pgpath->path.dev->bdev),
|
||||||
pg_init_done(&pgpath->path, ret);
|
pg_init_done, &pgpath->path);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -76,8 +76,8 @@
|
||||||
#define COPYRIGHT "Copyright (c) 1999-2008 " MODULEAUTHOR
|
#define COPYRIGHT "Copyright (c) 1999-2008 " MODULEAUTHOR
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define MPT_LINUX_VERSION_COMMON "3.04.12"
|
#define MPT_LINUX_VERSION_COMMON "3.04.13"
|
||||||
#define MPT_LINUX_PACKAGE_NAME "@(#)mptlinux-3.04.12"
|
#define MPT_LINUX_PACKAGE_NAME "@(#)mptlinux-3.04.13"
|
||||||
#define WHAT_MAGIC_STRING "@" "(" "#" ")"
|
#define WHAT_MAGIC_STRING "@" "(" "#" ")"
|
||||||
|
|
||||||
#define show_mptmod_ver(s,ver) \
|
#define show_mptmod_ver(s,ver) \
|
||||||
|
|
|
@ -621,11 +621,8 @@ __mptctl_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
|
||||||
*/
|
*/
|
||||||
iocnumX = khdr.iocnum & 0xFF;
|
iocnumX = khdr.iocnum & 0xFF;
|
||||||
if (((iocnum = mpt_verify_adapter(iocnumX, &iocp)) < 0) ||
|
if (((iocnum = mpt_verify_adapter(iocnumX, &iocp)) < 0) ||
|
||||||
(iocp == NULL)) {
|
(iocp == NULL))
|
||||||
printk(KERN_DEBUG MYNAM "%s::mptctl_ioctl() @%d - ioc%d not found!\n",
|
|
||||||
__FILE__, __LINE__, iocnumX);
|
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
}
|
|
||||||
|
|
||||||
if (!iocp->active) {
|
if (!iocp->active) {
|
||||||
printk(KERN_DEBUG MYNAM "%s::mptctl_ioctl() @%d - Controller disabled.\n",
|
printk(KERN_DEBUG MYNAM "%s::mptctl_ioctl() @%d - Controller disabled.\n",
|
||||||
|
|
|
@ -792,11 +792,36 @@ mptscsih_io_done(MPT_ADAPTER *ioc, MPT_FRAME_HDR *mf, MPT_FRAME_HDR *mr)
|
||||||
* precedence!
|
* precedence!
|
||||||
*/
|
*/
|
||||||
sc->result = (DID_OK << 16) | scsi_status;
|
sc->result = (DID_OK << 16) | scsi_status;
|
||||||
if (scsi_state & MPI_SCSI_STATE_AUTOSENSE_VALID) {
|
if (!(scsi_state & MPI_SCSI_STATE_AUTOSENSE_VALID)) {
|
||||||
/* Have already saved the status and sense data
|
|
||||||
|
/*
|
||||||
|
* For an Errata on LSI53C1030
|
||||||
|
* When the length of request data
|
||||||
|
* and transfer data are different
|
||||||
|
* with result of command (READ or VERIFY),
|
||||||
|
* DID_SOFT_ERROR is set.
|
||||||
*/
|
*/
|
||||||
;
|
if (ioc->bus_type == SPI) {
|
||||||
} else {
|
if (pScsiReq->CDB[0] == READ_6 ||
|
||||||
|
pScsiReq->CDB[0] == READ_10 ||
|
||||||
|
pScsiReq->CDB[0] == READ_12 ||
|
||||||
|
pScsiReq->CDB[0] == READ_16 ||
|
||||||
|
pScsiReq->CDB[0] == VERIFY ||
|
||||||
|
pScsiReq->CDB[0] == VERIFY_16) {
|
||||||
|
if (scsi_bufflen(sc) !=
|
||||||
|
xfer_cnt) {
|
||||||
|
sc->result =
|
||||||
|
DID_SOFT_ERROR << 16;
|
||||||
|
printk(KERN_WARNING "Errata"
|
||||||
|
"on LSI53C1030 occurred."
|
||||||
|
"sc->req_bufflen=0x%02x,"
|
||||||
|
"xfer_cnt=0x%02x\n",
|
||||||
|
scsi_bufflen(sc),
|
||||||
|
xfer_cnt);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (xfer_cnt < sc->underflow) {
|
if (xfer_cnt < sc->underflow) {
|
||||||
if (scsi_status == SAM_STAT_BUSY)
|
if (scsi_status == SAM_STAT_BUSY)
|
||||||
sc->result = SAM_STAT_BUSY;
|
sc->result = SAM_STAT_BUSY;
|
||||||
|
@ -835,7 +860,58 @@ mptscsih_io_done(MPT_ADAPTER *ioc, MPT_FRAME_HDR *mf, MPT_FRAME_HDR *mr)
|
||||||
sc->result = (DID_OK << 16) | scsi_status;
|
sc->result = (DID_OK << 16) | scsi_status;
|
||||||
if (scsi_state == 0) {
|
if (scsi_state == 0) {
|
||||||
;
|
;
|
||||||
} else if (scsi_state & MPI_SCSI_STATE_AUTOSENSE_VALID) {
|
} else if (scsi_state &
|
||||||
|
MPI_SCSI_STATE_AUTOSENSE_VALID) {
|
||||||
|
|
||||||
|
/*
|
||||||
|
* For potential trouble on LSI53C1030.
|
||||||
|
* (date:2007.xx.)
|
||||||
|
* It is checked whether the length of
|
||||||
|
* request data is equal to
|
||||||
|
* the length of transfer and residual.
|
||||||
|
* MEDIUM_ERROR is set by incorrect data.
|
||||||
|
*/
|
||||||
|
if ((ioc->bus_type == SPI) &&
|
||||||
|
(sc->sense_buffer[2] & 0x20)) {
|
||||||
|
u32 difftransfer;
|
||||||
|
difftransfer =
|
||||||
|
sc->sense_buffer[3] << 24 |
|
||||||
|
sc->sense_buffer[4] << 16 |
|
||||||
|
sc->sense_buffer[5] << 8 |
|
||||||
|
sc->sense_buffer[6];
|
||||||
|
if (((sc->sense_buffer[3] & 0x80) ==
|
||||||
|
0x80) && (scsi_bufflen(sc)
|
||||||
|
!= xfer_cnt)) {
|
||||||
|
sc->sense_buffer[2] =
|
||||||
|
MEDIUM_ERROR;
|
||||||
|
sc->sense_buffer[12] = 0xff;
|
||||||
|
sc->sense_buffer[13] = 0xff;
|
||||||
|
printk(KERN_WARNING"Errata"
|
||||||
|
"on LSI53C1030 occurred."
|
||||||
|
"sc->req_bufflen=0x%02x,"
|
||||||
|
"xfer_cnt=0x%02x\n" ,
|
||||||
|
scsi_bufflen(sc),
|
||||||
|
xfer_cnt);
|
||||||
|
}
|
||||||
|
if (((sc->sense_buffer[3] & 0x80)
|
||||||
|
!= 0x80) &&
|
||||||
|
(scsi_bufflen(sc) !=
|
||||||
|
xfer_cnt + difftransfer)) {
|
||||||
|
sc->sense_buffer[2] =
|
||||||
|
MEDIUM_ERROR;
|
||||||
|
sc->sense_buffer[12] = 0xff;
|
||||||
|
sc->sense_buffer[13] = 0xff;
|
||||||
|
printk(KERN_WARNING
|
||||||
|
"Errata on LSI53C1030 occurred"
|
||||||
|
"sc->req_bufflen=0x%02x,"
|
||||||
|
" xfer_cnt=0x%02x,"
|
||||||
|
"difftransfer=0x%02x\n",
|
||||||
|
scsi_bufflen(sc),
|
||||||
|
xfer_cnt,
|
||||||
|
difftransfer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If running against circa 200003dd 909 MPT f/w,
|
* If running against circa 200003dd 909 MPT f/w,
|
||||||
* may get this (AUTOSENSE_VALID) for actual TASK_SET_FULL
|
* may get this (AUTOSENSE_VALID) for actual TASK_SET_FULL
|
||||||
|
@ -2275,11 +2351,12 @@ mptscsih_slave_destroy(struct scsi_device *sdev)
|
||||||
* mptscsih_change_queue_depth - This function will set a devices queue depth
|
* mptscsih_change_queue_depth - This function will set a devices queue depth
|
||||||
* @sdev: per scsi_device pointer
|
* @sdev: per scsi_device pointer
|
||||||
* @qdepth: requested queue depth
|
* @qdepth: requested queue depth
|
||||||
|
* @reason: calling context
|
||||||
*
|
*
|
||||||
* Adding support for new 'change_queue_depth' api.
|
* Adding support for new 'change_queue_depth' api.
|
||||||
*/
|
*/
|
||||||
int
|
int
|
||||||
mptscsih_change_queue_depth(struct scsi_device *sdev, int qdepth)
|
mptscsih_change_queue_depth(struct scsi_device *sdev, int qdepth, int reason)
|
||||||
{
|
{
|
||||||
MPT_SCSI_HOST *hd = shost_priv(sdev->host);
|
MPT_SCSI_HOST *hd = shost_priv(sdev->host);
|
||||||
VirtTarget *vtarget;
|
VirtTarget *vtarget;
|
||||||
|
@ -2291,6 +2368,9 @@ mptscsih_change_queue_depth(struct scsi_device *sdev, int qdepth)
|
||||||
starget = scsi_target(sdev);
|
starget = scsi_target(sdev);
|
||||||
vtarget = starget->hostdata;
|
vtarget = starget->hostdata;
|
||||||
|
|
||||||
|
if (reason != SCSI_QDEPTH_DEFAULT)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (ioc->bus_type == SPI) {
|
if (ioc->bus_type == SPI) {
|
||||||
if (!(vtarget->tflags & MPT_TARGET_FLAGS_Q_YES))
|
if (!(vtarget->tflags & MPT_TARGET_FLAGS_Q_YES))
|
||||||
max_depth = 1;
|
max_depth = 1;
|
||||||
|
@ -2357,7 +2437,8 @@ mptscsih_slave_configure(struct scsi_device *sdev)
|
||||||
ioc->name, vtarget->negoFlags, vtarget->maxOffset,
|
ioc->name, vtarget->negoFlags, vtarget->maxOffset,
|
||||||
vtarget->minSyncFactor));
|
vtarget->minSyncFactor));
|
||||||
|
|
||||||
mptscsih_change_queue_depth(sdev, MPT_SCSI_CMD_PER_DEV_HIGH);
|
mptscsih_change_queue_depth(sdev, MPT_SCSI_CMD_PER_DEV_HIGH,
|
||||||
|
SCSI_QDEPTH_DEFAULT);
|
||||||
dsprintk(ioc, printk(MYIOC_s_DEBUG_FMT
|
dsprintk(ioc, printk(MYIOC_s_DEBUG_FMT
|
||||||
"tagged %d, simple %d, ordered %d\n",
|
"tagged %d, simple %d, ordered %d\n",
|
||||||
ioc->name,sdev->tagged_supported, sdev->simple_tags,
|
ioc->name,sdev->tagged_supported, sdev->simple_tags,
|
||||||
|
|
|
@ -128,7 +128,8 @@ extern int mptscsih_taskmgmt_complete(MPT_ADAPTER *ioc, MPT_FRAME_HDR *mf, MPT_F
|
||||||
extern int mptscsih_scandv_complete(MPT_ADAPTER *ioc, MPT_FRAME_HDR *mf, MPT_FRAME_HDR *r);
|
extern int mptscsih_scandv_complete(MPT_ADAPTER *ioc, MPT_FRAME_HDR *mf, MPT_FRAME_HDR *r);
|
||||||
extern int mptscsih_event_process(MPT_ADAPTER *ioc, EventNotificationReply_t *pEvReply);
|
extern int mptscsih_event_process(MPT_ADAPTER *ioc, EventNotificationReply_t *pEvReply);
|
||||||
extern int mptscsih_ioc_reset(MPT_ADAPTER *ioc, int post_reset);
|
extern int mptscsih_ioc_reset(MPT_ADAPTER *ioc, int post_reset);
|
||||||
extern int mptscsih_change_queue_depth(struct scsi_device *sdev, int qdepth);
|
extern int mptscsih_change_queue_depth(struct scsi_device *sdev, int qdepth,
|
||||||
|
int reason);
|
||||||
extern u8 mptscsih_raid_id_to_num(MPT_ADAPTER *ioc, u8 channel, u8 id);
|
extern u8 mptscsih_raid_id_to_num(MPT_ADAPTER *ioc, u8 channel, u8 id);
|
||||||
extern int mptscsih_is_phys_disk(MPT_ADAPTER *ioc, u8 channel, u8 id);
|
extern int mptscsih_is_phys_disk(MPT_ADAPTER *ioc, u8 channel, u8 id);
|
||||||
extern struct device_attribute *mptscsih_host_attrs[];
|
extern struct device_attribute *mptscsih_host_attrs[];
|
||||||
|
|
|
@ -31,6 +31,7 @@
|
||||||
#include <linux/miscdevice.h>
|
#include <linux/miscdevice.h>
|
||||||
#include <linux/seq_file.h>
|
#include <linux/seq_file.h>
|
||||||
#include "zfcp_ext.h"
|
#include "zfcp_ext.h"
|
||||||
|
#include "zfcp_fc.h"
|
||||||
|
|
||||||
#define ZFCP_BUS_ID_SIZE 20
|
#define ZFCP_BUS_ID_SIZE 20
|
||||||
|
|
||||||
|
@ -80,48 +81,40 @@ int zfcp_reqlist_isempty(struct zfcp_adapter *adapter)
|
||||||
|
|
||||||
static void __init zfcp_init_device_configure(char *busid, u64 wwpn, u64 lun)
|
static void __init zfcp_init_device_configure(char *busid, u64 wwpn, u64 lun)
|
||||||
{
|
{
|
||||||
struct ccw_device *ccwdev;
|
struct ccw_device *cdev;
|
||||||
struct zfcp_adapter *adapter;
|
struct zfcp_adapter *adapter;
|
||||||
struct zfcp_port *port;
|
struct zfcp_port *port;
|
||||||
struct zfcp_unit *unit;
|
struct zfcp_unit *unit;
|
||||||
|
|
||||||
ccwdev = get_ccwdev_by_busid(&zfcp_ccw_driver, busid);
|
cdev = get_ccwdev_by_busid(&zfcp_ccw_driver, busid);
|
||||||
if (!ccwdev)
|
if (!cdev)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (ccw_device_set_online(ccwdev))
|
if (ccw_device_set_online(cdev))
|
||||||
goto out_ccwdev;
|
goto out_ccw_device;
|
||||||
|
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
adapter = dev_get_drvdata(&ccwdev->dev);
|
|
||||||
if (!adapter)
|
if (!adapter)
|
||||||
goto out_unlock;
|
goto out_ccw_device;
|
||||||
zfcp_adapter_get(adapter);
|
|
||||||
|
|
||||||
port = zfcp_get_port_by_wwpn(adapter, wwpn);
|
port = zfcp_get_port_by_wwpn(adapter, wwpn);
|
||||||
if (!port)
|
if (!port)
|
||||||
goto out_port;
|
goto out_port;
|
||||||
|
|
||||||
zfcp_port_get(port);
|
|
||||||
unit = zfcp_unit_enqueue(port, lun);
|
unit = zfcp_unit_enqueue(port, lun);
|
||||||
if (IS_ERR(unit))
|
if (IS_ERR(unit))
|
||||||
goto out_unit;
|
goto out_unit;
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
|
||||||
|
|
||||||
zfcp_erp_unit_reopen(unit, 0, "auidc_1", NULL);
|
zfcp_erp_unit_reopen(unit, 0, "auidc_1", NULL);
|
||||||
zfcp_erp_wait(adapter);
|
zfcp_erp_wait(adapter);
|
||||||
flush_work(&unit->scsi_work);
|
flush_work(&unit->scsi_work);
|
||||||
|
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
|
||||||
zfcp_unit_put(unit);
|
|
||||||
out_unit:
|
out_unit:
|
||||||
zfcp_port_put(port);
|
put_device(&port->sysfs_device);
|
||||||
out_port:
|
out_port:
|
||||||
zfcp_adapter_put(adapter);
|
zfcp_ccw_adapter_put(adapter);
|
||||||
out_unlock:
|
out_ccw_device:
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
put_device(&cdev->dev);
|
||||||
out_ccwdev:
|
|
||||||
put_device(&ccwdev->dev);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -167,7 +160,7 @@ static int __init zfcp_module_init(void)
|
||||||
int retval = -ENOMEM;
|
int retval = -ENOMEM;
|
||||||
|
|
||||||
zfcp_data.gpn_ft_cache = zfcp_cache_hw_align("zfcp_gpn",
|
zfcp_data.gpn_ft_cache = zfcp_cache_hw_align("zfcp_gpn",
|
||||||
sizeof(struct ct_iu_gpn_ft_req));
|
sizeof(struct zfcp_fc_gpn_ft_req));
|
||||||
if (!zfcp_data.gpn_ft_cache)
|
if (!zfcp_data.gpn_ft_cache)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
@ -182,12 +175,14 @@ static int __init zfcp_module_init(void)
|
||||||
goto out_sr_cache;
|
goto out_sr_cache;
|
||||||
|
|
||||||
zfcp_data.gid_pn_cache = zfcp_cache_hw_align("zfcp_gid",
|
zfcp_data.gid_pn_cache = zfcp_cache_hw_align("zfcp_gid",
|
||||||
sizeof(struct zfcp_gid_pn_data));
|
sizeof(struct zfcp_fc_gid_pn));
|
||||||
if (!zfcp_data.gid_pn_cache)
|
if (!zfcp_data.gid_pn_cache)
|
||||||
goto out_gid_cache;
|
goto out_gid_cache;
|
||||||
|
|
||||||
mutex_init(&zfcp_data.config_mutex);
|
zfcp_data.adisc_cache = zfcp_cache_hw_align("zfcp_adisc",
|
||||||
rwlock_init(&zfcp_data.config_lock);
|
sizeof(struct zfcp_fc_els_adisc));
|
||||||
|
if (!zfcp_data.adisc_cache)
|
||||||
|
goto out_adisc_cache;
|
||||||
|
|
||||||
zfcp_data.scsi_transport_template =
|
zfcp_data.scsi_transport_template =
|
||||||
fc_attach_transport(&zfcp_transport_functions);
|
fc_attach_transport(&zfcp_transport_functions);
|
||||||
|
@ -200,7 +195,7 @@ static int __init zfcp_module_init(void)
|
||||||
goto out_misc;
|
goto out_misc;
|
||||||
}
|
}
|
||||||
|
|
||||||
retval = zfcp_ccw_register();
|
retval = ccw_driver_register(&zfcp_ccw_driver);
|
||||||
if (retval) {
|
if (retval) {
|
||||||
pr_err("The zfcp device driver could not register with "
|
pr_err("The zfcp device driver could not register with "
|
||||||
"the common I/O layer\n");
|
"the common I/O layer\n");
|
||||||
|
@ -216,6 +211,8 @@ out_ccw_register:
|
||||||
out_misc:
|
out_misc:
|
||||||
fc_release_transport(zfcp_data.scsi_transport_template);
|
fc_release_transport(zfcp_data.scsi_transport_template);
|
||||||
out_transport:
|
out_transport:
|
||||||
|
kmem_cache_destroy(zfcp_data.adisc_cache);
|
||||||
|
out_adisc_cache:
|
||||||
kmem_cache_destroy(zfcp_data.gid_pn_cache);
|
kmem_cache_destroy(zfcp_data.gid_pn_cache);
|
||||||
out_gid_cache:
|
out_gid_cache:
|
||||||
kmem_cache_destroy(zfcp_data.sr_buffer_cache);
|
kmem_cache_destroy(zfcp_data.sr_buffer_cache);
|
||||||
|
@ -229,6 +226,20 @@ out:
|
||||||
|
|
||||||
module_init(zfcp_module_init);
|
module_init(zfcp_module_init);
|
||||||
|
|
||||||
|
static void __exit zfcp_module_exit(void)
|
||||||
|
{
|
||||||
|
ccw_driver_unregister(&zfcp_ccw_driver);
|
||||||
|
misc_deregister(&zfcp_cfdc_misc);
|
||||||
|
fc_release_transport(zfcp_data.scsi_transport_template);
|
||||||
|
kmem_cache_destroy(zfcp_data.adisc_cache);
|
||||||
|
kmem_cache_destroy(zfcp_data.gid_pn_cache);
|
||||||
|
kmem_cache_destroy(zfcp_data.sr_buffer_cache);
|
||||||
|
kmem_cache_destroy(zfcp_data.qtcb_cache);
|
||||||
|
kmem_cache_destroy(zfcp_data.gpn_ft_cache);
|
||||||
|
}
|
||||||
|
|
||||||
|
module_exit(zfcp_module_exit);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* zfcp_get_unit_by_lun - find unit in unit list of port by FCP LUN
|
* zfcp_get_unit_by_lun - find unit in unit list of port by FCP LUN
|
||||||
* @port: pointer to port to search for unit
|
* @port: pointer to port to search for unit
|
||||||
|
@ -238,12 +249,18 @@ module_init(zfcp_module_init);
|
||||||
*/
|
*/
|
||||||
struct zfcp_unit *zfcp_get_unit_by_lun(struct zfcp_port *port, u64 fcp_lun)
|
struct zfcp_unit *zfcp_get_unit_by_lun(struct zfcp_port *port, u64 fcp_lun)
|
||||||
{
|
{
|
||||||
|
unsigned long flags;
|
||||||
struct zfcp_unit *unit;
|
struct zfcp_unit *unit;
|
||||||
|
|
||||||
list_for_each_entry(unit, &port->unit_list_head, list)
|
read_lock_irqsave(&port->unit_list_lock, flags);
|
||||||
if ((unit->fcp_lun == fcp_lun) &&
|
list_for_each_entry(unit, &port->unit_list, list)
|
||||||
!(atomic_read(&unit->status) & ZFCP_STATUS_COMMON_REMOVE))
|
if (unit->fcp_lun == fcp_lun) {
|
||||||
|
if (!get_device(&unit->sysfs_device))
|
||||||
|
unit = NULL;
|
||||||
|
read_unlock_irqrestore(&port->unit_list_lock, flags);
|
||||||
return unit;
|
return unit;
|
||||||
|
}
|
||||||
|
read_unlock_irqrestore(&port->unit_list_lock, flags);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -257,18 +274,35 @@ struct zfcp_unit *zfcp_get_unit_by_lun(struct zfcp_port *port, u64 fcp_lun)
|
||||||
struct zfcp_port *zfcp_get_port_by_wwpn(struct zfcp_adapter *adapter,
|
struct zfcp_port *zfcp_get_port_by_wwpn(struct zfcp_adapter *adapter,
|
||||||
u64 wwpn)
|
u64 wwpn)
|
||||||
{
|
{
|
||||||
|
unsigned long flags;
|
||||||
struct zfcp_port *port;
|
struct zfcp_port *port;
|
||||||
|
|
||||||
list_for_each_entry(port, &adapter->port_list_head, list)
|
read_lock_irqsave(&adapter->port_list_lock, flags);
|
||||||
if ((port->wwpn == wwpn) &&
|
list_for_each_entry(port, &adapter->port_list, list)
|
||||||
!(atomic_read(&port->status) & ZFCP_STATUS_COMMON_REMOVE))
|
if (port->wwpn == wwpn) {
|
||||||
|
if (!get_device(&port->sysfs_device))
|
||||||
|
port = NULL;
|
||||||
|
read_unlock_irqrestore(&adapter->port_list_lock, flags);
|
||||||
return port;
|
return port;
|
||||||
|
}
|
||||||
|
read_unlock_irqrestore(&adapter->port_list_lock, flags);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void zfcp_sysfs_unit_release(struct device *dev)
|
/**
|
||||||
|
* zfcp_unit_release - dequeue unit
|
||||||
|
* @dev: pointer to device
|
||||||
|
*
|
||||||
|
* waits until all work is done on unit and removes it then from the unit->list
|
||||||
|
* of the associated port.
|
||||||
|
*/
|
||||||
|
static void zfcp_unit_release(struct device *dev)
|
||||||
{
|
{
|
||||||
kfree(container_of(dev, struct zfcp_unit, sysfs_device));
|
struct zfcp_unit *unit = container_of(dev, struct zfcp_unit,
|
||||||
|
sysfs_device);
|
||||||
|
|
||||||
|
put_device(&unit->port->sysfs_device);
|
||||||
|
kfree(unit);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -276,43 +310,40 @@ static void zfcp_sysfs_unit_release(struct device *dev)
|
||||||
* @port: pointer to port where unit is added
|
* @port: pointer to port where unit is added
|
||||||
* @fcp_lun: FCP LUN of unit to be enqueued
|
* @fcp_lun: FCP LUN of unit to be enqueued
|
||||||
* Returns: pointer to enqueued unit on success, ERR_PTR on error
|
* Returns: pointer to enqueued unit on success, ERR_PTR on error
|
||||||
* Locks: config_mutex must be held to serialize changes to the unit list
|
|
||||||
*
|
*
|
||||||
* Sets up some unit internal structures and creates sysfs entry.
|
* Sets up some unit internal structures and creates sysfs entry.
|
||||||
*/
|
*/
|
||||||
struct zfcp_unit *zfcp_unit_enqueue(struct zfcp_port *port, u64 fcp_lun)
|
struct zfcp_unit *zfcp_unit_enqueue(struct zfcp_port *port, u64 fcp_lun)
|
||||||
{
|
{
|
||||||
struct zfcp_unit *unit;
|
struct zfcp_unit *unit;
|
||||||
|
int retval = -ENOMEM;
|
||||||
|
|
||||||
read_lock_irq(&zfcp_data.config_lock);
|
get_device(&port->sysfs_device);
|
||||||
if (zfcp_get_unit_by_lun(port, fcp_lun)) {
|
|
||||||
read_unlock_irq(&zfcp_data.config_lock);
|
unit = zfcp_get_unit_by_lun(port, fcp_lun);
|
||||||
return ERR_PTR(-EINVAL);
|
if (unit) {
|
||||||
|
put_device(&unit->sysfs_device);
|
||||||
|
retval = -EEXIST;
|
||||||
|
goto err_out;
|
||||||
}
|
}
|
||||||
read_unlock_irq(&zfcp_data.config_lock);
|
|
||||||
|
|
||||||
unit = kzalloc(sizeof(struct zfcp_unit), GFP_KERNEL);
|
unit = kzalloc(sizeof(struct zfcp_unit), GFP_KERNEL);
|
||||||
if (!unit)
|
if (!unit)
|
||||||
return ERR_PTR(-ENOMEM);
|
goto err_out;
|
||||||
|
|
||||||
atomic_set(&unit->refcount, 0);
|
|
||||||
init_waitqueue_head(&unit->remove_wq);
|
|
||||||
INIT_WORK(&unit->scsi_work, zfcp_scsi_scan);
|
|
||||||
|
|
||||||
unit->port = port;
|
unit->port = port;
|
||||||
unit->fcp_lun = fcp_lun;
|
unit->fcp_lun = fcp_lun;
|
||||||
|
unit->sysfs_device.parent = &port->sysfs_device;
|
||||||
|
unit->sysfs_device.release = zfcp_unit_release;
|
||||||
|
|
||||||
if (dev_set_name(&unit->sysfs_device, "0x%016llx",
|
if (dev_set_name(&unit->sysfs_device, "0x%016llx",
|
||||||
(unsigned long long) fcp_lun)) {
|
(unsigned long long) fcp_lun)) {
|
||||||
kfree(unit);
|
kfree(unit);
|
||||||
return ERR_PTR(-ENOMEM);
|
goto err_out;
|
||||||
}
|
}
|
||||||
unit->sysfs_device.parent = &port->sysfs_device;
|
retval = -EINVAL;
|
||||||
unit->sysfs_device.release = zfcp_sysfs_unit_release;
|
|
||||||
dev_set_drvdata(&unit->sysfs_device, unit);
|
|
||||||
|
|
||||||
/* mark unit unusable as long as sysfs registration is not complete */
|
INIT_WORK(&unit->scsi_work, zfcp_scsi_scan);
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_REMOVE, &unit->status);
|
|
||||||
|
|
||||||
spin_lock_init(&unit->latencies.lock);
|
spin_lock_init(&unit->latencies.lock);
|
||||||
unit->latencies.write.channel.min = 0xFFFFFFFF;
|
unit->latencies.write.channel.min = 0xFFFFFFFF;
|
||||||
|
@ -324,50 +355,30 @@ struct zfcp_unit *zfcp_unit_enqueue(struct zfcp_port *port, u64 fcp_lun)
|
||||||
|
|
||||||
if (device_register(&unit->sysfs_device)) {
|
if (device_register(&unit->sysfs_device)) {
|
||||||
put_device(&unit->sysfs_device);
|
put_device(&unit->sysfs_device);
|
||||||
return ERR_PTR(-EINVAL);
|
goto err_out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sysfs_create_group(&unit->sysfs_device.kobj,
|
if (sysfs_create_group(&unit->sysfs_device.kobj,
|
||||||
&zfcp_sysfs_unit_attrs)) {
|
&zfcp_sysfs_unit_attrs))
|
||||||
device_unregister(&unit->sysfs_device);
|
goto err_out_put;
|
||||||
return ERR_PTR(-EINVAL);
|
|
||||||
}
|
|
||||||
|
|
||||||
zfcp_unit_get(unit);
|
write_lock_irq(&port->unit_list_lock);
|
||||||
|
list_add_tail(&unit->list, &port->unit_list);
|
||||||
|
write_unlock_irq(&port->unit_list_lock);
|
||||||
|
|
||||||
write_lock_irq(&zfcp_data.config_lock);
|
|
||||||
list_add_tail(&unit->list, &port->unit_list_head);
|
|
||||||
atomic_clear_mask(ZFCP_STATUS_COMMON_REMOVE, &unit->status);
|
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_RUNNING, &unit->status);
|
atomic_set_mask(ZFCP_STATUS_COMMON_RUNNING, &unit->status);
|
||||||
|
|
||||||
write_unlock_irq(&zfcp_data.config_lock);
|
|
||||||
|
|
||||||
zfcp_port_get(port);
|
|
||||||
|
|
||||||
return unit;
|
return unit;
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
err_out_put:
|
||||||
* zfcp_unit_dequeue - dequeue unit
|
|
||||||
* @unit: pointer to zfcp_unit
|
|
||||||
*
|
|
||||||
* waits until all work is done on unit and removes it then from the unit->list
|
|
||||||
* of the associated port.
|
|
||||||
*/
|
|
||||||
void zfcp_unit_dequeue(struct zfcp_unit *unit)
|
|
||||||
{
|
|
||||||
wait_event(unit->remove_wq, atomic_read(&unit->refcount) == 0);
|
|
||||||
write_lock_irq(&zfcp_data.config_lock);
|
|
||||||
list_del(&unit->list);
|
|
||||||
write_unlock_irq(&zfcp_data.config_lock);
|
|
||||||
zfcp_port_put(unit->port);
|
|
||||||
sysfs_remove_group(&unit->sysfs_device.kobj, &zfcp_sysfs_unit_attrs);
|
|
||||||
device_unregister(&unit->sysfs_device);
|
device_unregister(&unit->sysfs_device);
|
||||||
|
err_out:
|
||||||
|
put_device(&port->sysfs_device);
|
||||||
|
return ERR_PTR(retval);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int zfcp_allocate_low_mem_buffers(struct zfcp_adapter *adapter)
|
static int zfcp_allocate_low_mem_buffers(struct zfcp_adapter *adapter)
|
||||||
{
|
{
|
||||||
/* must only be called with zfcp_data.config_mutex taken */
|
|
||||||
adapter->pool.erp_req =
|
adapter->pool.erp_req =
|
||||||
mempool_create_kmalloc_pool(1, sizeof(struct zfcp_fsf_req));
|
mempool_create_kmalloc_pool(1, sizeof(struct zfcp_fsf_req));
|
||||||
if (!adapter->pool.erp_req)
|
if (!adapter->pool.erp_req)
|
||||||
|
@ -405,9 +416,9 @@ static int zfcp_allocate_low_mem_buffers(struct zfcp_adapter *adapter)
|
||||||
if (!adapter->pool.status_read_data)
|
if (!adapter->pool.status_read_data)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
adapter->pool.gid_pn_data =
|
adapter->pool.gid_pn =
|
||||||
mempool_create_slab_pool(1, zfcp_data.gid_pn_cache);
|
mempool_create_slab_pool(1, zfcp_data.gid_pn_cache);
|
||||||
if (!adapter->pool.gid_pn_data)
|
if (!adapter->pool.gid_pn)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -415,7 +426,6 @@ static int zfcp_allocate_low_mem_buffers(struct zfcp_adapter *adapter)
|
||||||
|
|
||||||
static void zfcp_free_low_mem_buffers(struct zfcp_adapter *adapter)
|
static void zfcp_free_low_mem_buffers(struct zfcp_adapter *adapter)
|
||||||
{
|
{
|
||||||
/* zfcp_data.config_mutex must be held */
|
|
||||||
if (adapter->pool.erp_req)
|
if (adapter->pool.erp_req)
|
||||||
mempool_destroy(adapter->pool.erp_req);
|
mempool_destroy(adapter->pool.erp_req);
|
||||||
if (adapter->pool.scsi_req)
|
if (adapter->pool.scsi_req)
|
||||||
|
@ -428,8 +438,8 @@ static void zfcp_free_low_mem_buffers(struct zfcp_adapter *adapter)
|
||||||
mempool_destroy(adapter->pool.status_read_req);
|
mempool_destroy(adapter->pool.status_read_req);
|
||||||
if (adapter->pool.status_read_data)
|
if (adapter->pool.status_read_data)
|
||||||
mempool_destroy(adapter->pool.status_read_data);
|
mempool_destroy(adapter->pool.status_read_data);
|
||||||
if (adapter->pool.gid_pn_data)
|
if (adapter->pool.gid_pn)
|
||||||
mempool_destroy(adapter->pool.gid_pn_data);
|
mempool_destroy(adapter->pool.gid_pn);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -497,53 +507,56 @@ static void zfcp_destroy_adapter_work_queue(struct zfcp_adapter *adapter)
|
||||||
* zfcp_adapter_enqueue - enqueue a new adapter to the list
|
* zfcp_adapter_enqueue - enqueue a new adapter to the list
|
||||||
* @ccw_device: pointer to the struct cc_device
|
* @ccw_device: pointer to the struct cc_device
|
||||||
*
|
*
|
||||||
* Returns: 0 if a new adapter was successfully enqueued
|
* Returns: struct zfcp_adapter*
|
||||||
* -ENOMEM if alloc failed
|
|
||||||
* Enqueues an adapter at the end of the adapter list in the driver data.
|
* Enqueues an adapter at the end of the adapter list in the driver data.
|
||||||
* All adapter internal structures are set up.
|
* All adapter internal structures are set up.
|
||||||
* Proc-fs entries are also created.
|
* Proc-fs entries are also created.
|
||||||
* locks: config_mutex must be held to serialize changes to the adapter list
|
|
||||||
*/
|
*/
|
||||||
int zfcp_adapter_enqueue(struct ccw_device *ccw_device)
|
struct zfcp_adapter *zfcp_adapter_enqueue(struct ccw_device *ccw_device)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter;
|
struct zfcp_adapter *adapter;
|
||||||
|
|
||||||
/*
|
if (!get_device(&ccw_device->dev))
|
||||||
* Note: It is safe to release the list_lock, as any list changes
|
return ERR_PTR(-ENODEV);
|
||||||
* are protected by the config_mutex, which must be held to get here
|
|
||||||
*/
|
|
||||||
|
|
||||||
adapter = kzalloc(sizeof(struct zfcp_adapter), GFP_KERNEL);
|
adapter = kzalloc(sizeof(struct zfcp_adapter), GFP_KERNEL);
|
||||||
if (!adapter)
|
if (!adapter) {
|
||||||
return -ENOMEM;
|
put_device(&ccw_device->dev);
|
||||||
|
return ERR_PTR(-ENOMEM);
|
||||||
|
}
|
||||||
|
|
||||||
|
kref_init(&adapter->ref);
|
||||||
|
|
||||||
ccw_device->handler = NULL;
|
ccw_device->handler = NULL;
|
||||||
adapter->ccw_device = ccw_device;
|
adapter->ccw_device = ccw_device;
|
||||||
atomic_set(&adapter->refcount, 0);
|
|
||||||
|
INIT_WORK(&adapter->stat_work, _zfcp_status_read_scheduler);
|
||||||
|
INIT_WORK(&adapter->scan_work, zfcp_fc_scan_ports);
|
||||||
|
|
||||||
if (zfcp_qdio_setup(adapter))
|
if (zfcp_qdio_setup(adapter))
|
||||||
goto qdio_failed;
|
goto failed;
|
||||||
|
|
||||||
if (zfcp_allocate_low_mem_buffers(adapter))
|
if (zfcp_allocate_low_mem_buffers(adapter))
|
||||||
goto low_mem_buffers_failed;
|
goto failed;
|
||||||
|
|
||||||
if (zfcp_reqlist_alloc(adapter))
|
if (zfcp_reqlist_alloc(adapter))
|
||||||
goto low_mem_buffers_failed;
|
goto failed;
|
||||||
|
|
||||||
if (zfcp_dbf_adapter_register(adapter))
|
if (zfcp_dbf_adapter_register(adapter))
|
||||||
goto debug_register_failed;
|
goto failed;
|
||||||
|
|
||||||
if (zfcp_setup_adapter_work_queue(adapter))
|
if (zfcp_setup_adapter_work_queue(adapter))
|
||||||
goto work_queue_failed;
|
goto failed;
|
||||||
|
|
||||||
if (zfcp_fc_gs_setup(adapter))
|
if (zfcp_fc_gs_setup(adapter))
|
||||||
goto generic_services_failed;
|
goto failed;
|
||||||
|
|
||||||
|
rwlock_init(&adapter->port_list_lock);
|
||||||
|
INIT_LIST_HEAD(&adapter->port_list);
|
||||||
|
|
||||||
init_waitqueue_head(&adapter->remove_wq);
|
|
||||||
init_waitqueue_head(&adapter->erp_ready_wq);
|
init_waitqueue_head(&adapter->erp_ready_wq);
|
||||||
init_waitqueue_head(&adapter->erp_done_wqh);
|
init_waitqueue_head(&adapter->erp_done_wqh);
|
||||||
|
|
||||||
INIT_LIST_HEAD(&adapter->port_list_head);
|
|
||||||
INIT_LIST_HEAD(&adapter->erp_ready_head);
|
INIT_LIST_HEAD(&adapter->erp_ready_head);
|
||||||
INIT_LIST_HEAD(&adapter->erp_running_head);
|
INIT_LIST_HEAD(&adapter->erp_running_head);
|
||||||
|
|
||||||
|
@ -553,83 +566,85 @@ int zfcp_adapter_enqueue(struct ccw_device *ccw_device)
|
||||||
rwlock_init(&adapter->abort_lock);
|
rwlock_init(&adapter->abort_lock);
|
||||||
|
|
||||||
if (zfcp_erp_thread_setup(adapter))
|
if (zfcp_erp_thread_setup(adapter))
|
||||||
goto erp_thread_failed;
|
goto failed;
|
||||||
|
|
||||||
INIT_WORK(&adapter->stat_work, _zfcp_status_read_scheduler);
|
|
||||||
INIT_WORK(&adapter->scan_work, _zfcp_fc_scan_ports_later);
|
|
||||||
|
|
||||||
adapter->service_level.seq_print = zfcp_print_sl;
|
adapter->service_level.seq_print = zfcp_print_sl;
|
||||||
|
|
||||||
/* mark adapter unusable as long as sysfs registration is not complete */
|
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_REMOVE, &adapter->status);
|
|
||||||
|
|
||||||
dev_set_drvdata(&ccw_device->dev, adapter);
|
dev_set_drvdata(&ccw_device->dev, adapter);
|
||||||
|
|
||||||
if (sysfs_create_group(&ccw_device->dev.kobj,
|
if (sysfs_create_group(&ccw_device->dev.kobj,
|
||||||
&zfcp_sysfs_adapter_attrs))
|
&zfcp_sysfs_adapter_attrs))
|
||||||
goto sysfs_failed;
|
goto failed;
|
||||||
|
|
||||||
atomic_clear_mask(ZFCP_STATUS_COMMON_REMOVE, &adapter->status);
|
|
||||||
|
|
||||||
if (!zfcp_adapter_scsi_register(adapter))
|
if (!zfcp_adapter_scsi_register(adapter))
|
||||||
return 0;
|
return adapter;
|
||||||
|
|
||||||
sysfs_failed:
|
failed:
|
||||||
zfcp_erp_thread_kill(adapter);
|
zfcp_adapter_unregister(adapter);
|
||||||
erp_thread_failed:
|
return ERR_PTR(-ENOMEM);
|
||||||
zfcp_fc_gs_destroy(adapter);
|
}
|
||||||
generic_services_failed:
|
|
||||||
|
void zfcp_adapter_unregister(struct zfcp_adapter *adapter)
|
||||||
|
{
|
||||||
|
struct ccw_device *cdev = adapter->ccw_device;
|
||||||
|
|
||||||
|
cancel_work_sync(&adapter->scan_work);
|
||||||
|
cancel_work_sync(&adapter->stat_work);
|
||||||
zfcp_destroy_adapter_work_queue(adapter);
|
zfcp_destroy_adapter_work_queue(adapter);
|
||||||
work_queue_failed:
|
|
||||||
|
zfcp_fc_wka_ports_force_offline(adapter->gs);
|
||||||
|
zfcp_adapter_scsi_unregister(adapter);
|
||||||
|
sysfs_remove_group(&cdev->dev.kobj, &zfcp_sysfs_adapter_attrs);
|
||||||
|
|
||||||
|
zfcp_erp_thread_kill(adapter);
|
||||||
zfcp_dbf_adapter_unregister(adapter->dbf);
|
zfcp_dbf_adapter_unregister(adapter->dbf);
|
||||||
debug_register_failed:
|
|
||||||
dev_set_drvdata(&ccw_device->dev, NULL);
|
|
||||||
kfree(adapter->req_list);
|
|
||||||
low_mem_buffers_failed:
|
|
||||||
zfcp_free_low_mem_buffers(adapter);
|
|
||||||
qdio_failed:
|
|
||||||
zfcp_qdio_destroy(adapter->qdio);
|
zfcp_qdio_destroy(adapter->qdio);
|
||||||
kfree(adapter);
|
|
||||||
return -ENOMEM;
|
zfcp_ccw_adapter_put(adapter); /* final put to release */
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* zfcp_adapter_dequeue - remove the adapter from the resource list
|
* zfcp_adapter_release - remove the adapter from the resource list
|
||||||
* @adapter: pointer to struct zfcp_adapter which should be removed
|
* @ref: pointer to struct kref
|
||||||
* locks: adapter list write lock is assumed to be held by caller
|
* locks: adapter list write lock is assumed to be held by caller
|
||||||
*/
|
*/
|
||||||
void zfcp_adapter_dequeue(struct zfcp_adapter *adapter)
|
void zfcp_adapter_release(struct kref *ref)
|
||||||
{
|
{
|
||||||
int retval = 0;
|
struct zfcp_adapter *adapter = container_of(ref, struct zfcp_adapter,
|
||||||
unsigned long flags;
|
ref);
|
||||||
|
struct ccw_device *cdev = adapter->ccw_device;
|
||||||
|
|
||||||
cancel_work_sync(&adapter->stat_work);
|
|
||||||
zfcp_fc_wka_ports_force_offline(adapter->gs);
|
|
||||||
sysfs_remove_group(&adapter->ccw_device->dev.kobj,
|
|
||||||
&zfcp_sysfs_adapter_attrs);
|
|
||||||
dev_set_drvdata(&adapter->ccw_device->dev, NULL);
|
dev_set_drvdata(&adapter->ccw_device->dev, NULL);
|
||||||
/* sanity check: no pending FSF requests */
|
|
||||||
spin_lock_irqsave(&adapter->req_list_lock, flags);
|
|
||||||
retval = zfcp_reqlist_isempty(adapter);
|
|
||||||
spin_unlock_irqrestore(&adapter->req_list_lock, flags);
|
|
||||||
if (!retval)
|
|
||||||
return;
|
|
||||||
|
|
||||||
zfcp_fc_gs_destroy(adapter);
|
zfcp_fc_gs_destroy(adapter);
|
||||||
zfcp_erp_thread_kill(adapter);
|
|
||||||
zfcp_destroy_adapter_work_queue(adapter);
|
|
||||||
zfcp_dbf_adapter_unregister(adapter->dbf);
|
|
||||||
zfcp_free_low_mem_buffers(adapter);
|
zfcp_free_low_mem_buffers(adapter);
|
||||||
zfcp_qdio_destroy(adapter->qdio);
|
|
||||||
kfree(adapter->req_list);
|
kfree(adapter->req_list);
|
||||||
kfree(adapter->fc_stats);
|
kfree(adapter->fc_stats);
|
||||||
kfree(adapter->stats_reset_data);
|
kfree(adapter->stats_reset_data);
|
||||||
kfree(adapter);
|
kfree(adapter);
|
||||||
|
put_device(&cdev->dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void zfcp_sysfs_port_release(struct device *dev)
|
/**
|
||||||
|
* zfcp_device_unregister - remove port, unit from system
|
||||||
|
* @dev: reference to device which is to be removed
|
||||||
|
* @grp: related reference to attribute group
|
||||||
|
*
|
||||||
|
* Helper function to unregister port, unit from system
|
||||||
|
*/
|
||||||
|
void zfcp_device_unregister(struct device *dev,
|
||||||
|
const struct attribute_group *grp)
|
||||||
{
|
{
|
||||||
kfree(container_of(dev, struct zfcp_port, sysfs_device));
|
sysfs_remove_group(&dev->kobj, grp);
|
||||||
|
device_unregister(dev);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void zfcp_port_release(struct device *dev)
|
||||||
|
{
|
||||||
|
struct zfcp_port *port = container_of(dev, struct zfcp_port,
|
||||||
|
sysfs_device);
|
||||||
|
|
||||||
|
zfcp_ccw_adapter_put(port->adapter);
|
||||||
|
kfree(port);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -639,7 +654,6 @@ static void zfcp_sysfs_port_release(struct device *dev)
|
||||||
* @status: initial status for the port
|
* @status: initial status for the port
|
||||||
* @d_id: destination id of the remote port to be enqueued
|
* @d_id: destination id of the remote port to be enqueued
|
||||||
* Returns: pointer to enqueued port on success, ERR_PTR on error
|
* Returns: pointer to enqueued port on success, ERR_PTR on error
|
||||||
* Locks: config_mutex must be held to serialize changes to the port list
|
|
||||||
*
|
*
|
||||||
* All port internal structures are set up and the sysfs entry is generated.
|
* All port internal structures are set up and the sysfs entry is generated.
|
||||||
* d_id is used to enqueue ports with a well known address like the Directory
|
* d_id is used to enqueue ports with a well known address like the Directory
|
||||||
|
@ -649,20 +663,24 @@ struct zfcp_port *zfcp_port_enqueue(struct zfcp_adapter *adapter, u64 wwpn,
|
||||||
u32 status, u32 d_id)
|
u32 status, u32 d_id)
|
||||||
{
|
{
|
||||||
struct zfcp_port *port;
|
struct zfcp_port *port;
|
||||||
|
int retval = -ENOMEM;
|
||||||
|
|
||||||
read_lock_irq(&zfcp_data.config_lock);
|
kref_get(&adapter->ref);
|
||||||
if (zfcp_get_port_by_wwpn(adapter, wwpn)) {
|
|
||||||
read_unlock_irq(&zfcp_data.config_lock);
|
port = zfcp_get_port_by_wwpn(adapter, wwpn);
|
||||||
return ERR_PTR(-EINVAL);
|
if (port) {
|
||||||
|
put_device(&port->sysfs_device);
|
||||||
|
retval = -EEXIST;
|
||||||
|
goto err_out;
|
||||||
}
|
}
|
||||||
read_unlock_irq(&zfcp_data.config_lock);
|
|
||||||
|
|
||||||
port = kzalloc(sizeof(struct zfcp_port), GFP_KERNEL);
|
port = kzalloc(sizeof(struct zfcp_port), GFP_KERNEL);
|
||||||
if (!port)
|
if (!port)
|
||||||
return ERR_PTR(-ENOMEM);
|
goto err_out;
|
||||||
|
|
||||||
|
rwlock_init(&port->unit_list_lock);
|
||||||
|
INIT_LIST_HEAD(&port->unit_list);
|
||||||
|
|
||||||
init_waitqueue_head(&port->remove_wq);
|
|
||||||
INIT_LIST_HEAD(&port->unit_list_head);
|
|
||||||
INIT_WORK(&port->gid_pn_work, zfcp_fc_port_did_lookup);
|
INIT_WORK(&port->gid_pn_work, zfcp_fc_port_did_lookup);
|
||||||
INIT_WORK(&port->test_link_work, zfcp_fc_link_test_work);
|
INIT_WORK(&port->test_link_work, zfcp_fc_link_test_work);
|
||||||
INIT_WORK(&port->rport_work, zfcp_scsi_rport_work);
|
INIT_WORK(&port->rport_work, zfcp_scsi_rport_work);
|
||||||
|
@ -671,58 +689,38 @@ struct zfcp_port *zfcp_port_enqueue(struct zfcp_adapter *adapter, u64 wwpn,
|
||||||
port->d_id = d_id;
|
port->d_id = d_id;
|
||||||
port->wwpn = wwpn;
|
port->wwpn = wwpn;
|
||||||
port->rport_task = RPORT_NONE;
|
port->rport_task = RPORT_NONE;
|
||||||
|
port->sysfs_device.parent = &adapter->ccw_device->dev;
|
||||||
/* mark port unusable as long as sysfs registration is not complete */
|
port->sysfs_device.release = zfcp_port_release;
|
||||||
atomic_set_mask(status | ZFCP_STATUS_COMMON_REMOVE, &port->status);
|
|
||||||
atomic_set(&port->refcount, 0);
|
|
||||||
|
|
||||||
if (dev_set_name(&port->sysfs_device, "0x%016llx",
|
if (dev_set_name(&port->sysfs_device, "0x%016llx",
|
||||||
(unsigned long long)wwpn)) {
|
(unsigned long long)wwpn)) {
|
||||||
kfree(port);
|
kfree(port);
|
||||||
return ERR_PTR(-ENOMEM);
|
goto err_out;
|
||||||
}
|
}
|
||||||
port->sysfs_device.parent = &adapter->ccw_device->dev;
|
retval = -EINVAL;
|
||||||
port->sysfs_device.release = zfcp_sysfs_port_release;
|
|
||||||
dev_set_drvdata(&port->sysfs_device, port);
|
|
||||||
|
|
||||||
if (device_register(&port->sysfs_device)) {
|
if (device_register(&port->sysfs_device)) {
|
||||||
put_device(&port->sysfs_device);
|
put_device(&port->sysfs_device);
|
||||||
return ERR_PTR(-EINVAL);
|
goto err_out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sysfs_create_group(&port->sysfs_device.kobj,
|
if (sysfs_create_group(&port->sysfs_device.kobj,
|
||||||
&zfcp_sysfs_port_attrs)) {
|
&zfcp_sysfs_port_attrs))
|
||||||
device_unregister(&port->sysfs_device);
|
goto err_out_put;
|
||||||
return ERR_PTR(-EINVAL);
|
|
||||||
}
|
|
||||||
|
|
||||||
zfcp_port_get(port);
|
write_lock_irq(&adapter->port_list_lock);
|
||||||
|
list_add_tail(&port->list, &adapter->port_list);
|
||||||
|
write_unlock_irq(&adapter->port_list_lock);
|
||||||
|
|
||||||
write_lock_irq(&zfcp_data.config_lock);
|
atomic_set_mask(status | ZFCP_STATUS_COMMON_RUNNING, &port->status);
|
||||||
list_add_tail(&port->list, &adapter->port_list_head);
|
|
||||||
atomic_clear_mask(ZFCP_STATUS_COMMON_REMOVE, &port->status);
|
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_RUNNING, &port->status);
|
|
||||||
|
|
||||||
write_unlock_irq(&zfcp_data.config_lock);
|
|
||||||
|
|
||||||
zfcp_adapter_get(adapter);
|
|
||||||
return port;
|
return port;
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
err_out_put:
|
||||||
* zfcp_port_dequeue - dequeues a port from the port list of the adapter
|
|
||||||
* @port: pointer to struct zfcp_port which should be removed
|
|
||||||
*/
|
|
||||||
void zfcp_port_dequeue(struct zfcp_port *port)
|
|
||||||
{
|
|
||||||
write_lock_irq(&zfcp_data.config_lock);
|
|
||||||
list_del(&port->list);
|
|
||||||
write_unlock_irq(&zfcp_data.config_lock);
|
|
||||||
wait_event(port->remove_wq, atomic_read(&port->refcount) == 0);
|
|
||||||
cancel_work_sync(&port->rport_work); /* usually not necessary */
|
|
||||||
zfcp_adapter_put(port->adapter);
|
|
||||||
sysfs_remove_group(&port->sysfs_device.kobj, &zfcp_sysfs_port_attrs);
|
|
||||||
device_unregister(&port->sysfs_device);
|
device_unregister(&port->sysfs_device);
|
||||||
|
err_out:
|
||||||
|
zfcp_ccw_adapter_put(adapter);
|
||||||
|
return ERR_PTR(retval);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -13,28 +13,34 @@
|
||||||
|
|
||||||
#define ZFCP_MODEL_PRIV 0x4
|
#define ZFCP_MODEL_PRIV 0x4
|
||||||
|
|
||||||
static int zfcp_ccw_suspend(struct ccw_device *cdev)
|
static DEFINE_SPINLOCK(zfcp_ccw_adapter_ref_lock);
|
||||||
|
|
||||||
|
struct zfcp_adapter *zfcp_ccw_adapter_by_cdev(struct ccw_device *cdev)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter = dev_get_drvdata(&cdev->dev);
|
struct zfcp_adapter *adapter;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
if (!adapter)
|
spin_lock_irqsave(&zfcp_ccw_adapter_ref_lock, flags);
|
||||||
return 0;
|
adapter = dev_get_drvdata(&cdev->dev);
|
||||||
|
if (adapter)
|
||||||
|
kref_get(&adapter->ref);
|
||||||
|
spin_unlock_irqrestore(&zfcp_ccw_adapter_ref_lock, flags);
|
||||||
|
return adapter;
|
||||||
|
}
|
||||||
|
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
void zfcp_ccw_adapter_put(struct zfcp_adapter *adapter)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
zfcp_erp_adapter_shutdown(adapter, 0, "ccsusp1", NULL);
|
spin_lock_irqsave(&zfcp_ccw_adapter_ref_lock, flags);
|
||||||
zfcp_erp_wait(adapter);
|
kref_put(&adapter->ref, zfcp_adapter_release);
|
||||||
|
spin_unlock_irqrestore(&zfcp_ccw_adapter_ref_lock, flags);
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int zfcp_ccw_activate(struct ccw_device *cdev)
|
static int zfcp_ccw_activate(struct ccw_device *cdev)
|
||||||
|
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter = dev_get_drvdata(&cdev->dev);
|
struct zfcp_adapter *adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
|
|
||||||
if (!adapter)
|
if (!adapter)
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -46,6 +52,8 @@ static int zfcp_ccw_activate(struct ccw_device *cdev)
|
||||||
zfcp_erp_wait(adapter);
|
zfcp_erp_wait(adapter);
|
||||||
flush_work(&adapter->scan_work);
|
flush_work(&adapter->scan_work);
|
||||||
|
|
||||||
|
zfcp_ccw_adapter_put(adapter);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -67,28 +75,28 @@ int zfcp_ccw_priv_sch(struct zfcp_adapter *adapter)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* zfcp_ccw_probe - probe function of zfcp driver
|
* zfcp_ccw_probe - probe function of zfcp driver
|
||||||
* @ccw_device: pointer to belonging ccw device
|
* @cdev: pointer to belonging ccw device
|
||||||
*
|
*
|
||||||
* This function gets called by the common i/o layer for each FCP
|
* This function gets called by the common i/o layer for each FCP
|
||||||
* device found on the current system. This is only a stub to make cio
|
* device found on the current system. This is only a stub to make cio
|
||||||
* work: To only allocate adapter resources for devices actually used,
|
* work: To only allocate adapter resources for devices actually used,
|
||||||
* the allocation is deferred to the first call to ccw_set_online.
|
* the allocation is deferred to the first call to ccw_set_online.
|
||||||
*/
|
*/
|
||||||
static int zfcp_ccw_probe(struct ccw_device *ccw_device)
|
static int zfcp_ccw_probe(struct ccw_device *cdev)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* zfcp_ccw_remove - remove function of zfcp driver
|
* zfcp_ccw_remove - remove function of zfcp driver
|
||||||
* @ccw_device: pointer to belonging ccw device
|
* @cdev: pointer to belonging ccw device
|
||||||
*
|
*
|
||||||
* This function gets called by the common i/o layer and removes an adapter
|
* This function gets called by the common i/o layer and removes an adapter
|
||||||
* from the system. Task of this function is to get rid of all units and
|
* from the system. Task of this function is to get rid of all units and
|
||||||
* ports that belong to this adapter. And in addition all resources of this
|
* ports that belong to this adapter. And in addition all resources of this
|
||||||
* adapter will be freed too.
|
* adapter will be freed too.
|
||||||
*/
|
*/
|
||||||
static void zfcp_ccw_remove(struct ccw_device *ccw_device)
|
static void zfcp_ccw_remove(struct ccw_device *cdev)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter;
|
struct zfcp_adapter *adapter;
|
||||||
struct zfcp_port *port, *p;
|
struct zfcp_port *port, *p;
|
||||||
|
@ -96,49 +104,37 @@ static void zfcp_ccw_remove(struct ccw_device *ccw_device)
|
||||||
LIST_HEAD(unit_remove_lh);
|
LIST_HEAD(unit_remove_lh);
|
||||||
LIST_HEAD(port_remove_lh);
|
LIST_HEAD(port_remove_lh);
|
||||||
|
|
||||||
ccw_device_set_offline(ccw_device);
|
ccw_device_set_offline(cdev);
|
||||||
|
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
adapter = dev_get_drvdata(&ccw_device->dev);
|
|
||||||
if (!adapter)
|
if (!adapter)
|
||||||
goto out;
|
return;
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
|
||||||
|
|
||||||
cancel_work_sync(&adapter->scan_work);
|
write_lock_irq(&adapter->port_list_lock);
|
||||||
|
list_for_each_entry_safe(port, p, &adapter->port_list, list) {
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
write_lock(&port->unit_list_lock);
|
||||||
|
list_for_each_entry_safe(unit, u, &port->unit_list, list)
|
||||||
/* this also removes the scsi devices, so call it first */
|
|
||||||
zfcp_adapter_scsi_unregister(adapter);
|
|
||||||
|
|
||||||
write_lock_irq(&zfcp_data.config_lock);
|
|
||||||
list_for_each_entry_safe(port, p, &adapter->port_list_head, list) {
|
|
||||||
list_for_each_entry_safe(unit, u, &port->unit_list_head, list) {
|
|
||||||
list_move(&unit->list, &unit_remove_lh);
|
list_move(&unit->list, &unit_remove_lh);
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_REMOVE,
|
write_unlock(&port->unit_list_lock);
|
||||||
&unit->status);
|
|
||||||
}
|
|
||||||
list_move(&port->list, &port_remove_lh);
|
list_move(&port->list, &port_remove_lh);
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_REMOVE, &port->status);
|
|
||||||
}
|
}
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_REMOVE, &adapter->status);
|
write_unlock_irq(&adapter->port_list_lock);
|
||||||
write_unlock_irq(&zfcp_data.config_lock);
|
zfcp_ccw_adapter_put(adapter); /* put from zfcp_ccw_adapter_by_cdev */
|
||||||
|
|
||||||
list_for_each_entry_safe(port, p, &port_remove_lh, list) {
|
|
||||||
list_for_each_entry_safe(unit, u, &unit_remove_lh, list)
|
list_for_each_entry_safe(unit, u, &unit_remove_lh, list)
|
||||||
zfcp_unit_dequeue(unit);
|
zfcp_device_unregister(&unit->sysfs_device,
|
||||||
zfcp_port_dequeue(port);
|
&zfcp_sysfs_unit_attrs);
|
||||||
}
|
|
||||||
wait_event(adapter->remove_wq, atomic_read(&adapter->refcount) == 0);
|
|
||||||
zfcp_adapter_dequeue(adapter);
|
|
||||||
|
|
||||||
out:
|
list_for_each_entry_safe(port, p, &port_remove_lh, list)
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
zfcp_device_unregister(&port->sysfs_device,
|
||||||
|
&zfcp_sysfs_port_attrs);
|
||||||
|
|
||||||
|
zfcp_adapter_unregister(adapter);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* zfcp_ccw_set_online - set_online function of zfcp driver
|
* zfcp_ccw_set_online - set_online function of zfcp driver
|
||||||
* @ccw_device: pointer to belonging ccw device
|
* @cdev: pointer to belonging ccw device
|
||||||
*
|
*
|
||||||
* This function gets called by the common i/o layer and sets an
|
* This function gets called by the common i/o layer and sets an
|
||||||
* adapter into state online. The first call will allocate all
|
* adapter into state online. The first call will allocate all
|
||||||
|
@ -149,23 +145,20 @@ out:
|
||||||
* the SCSI stack, that the QDIO queues will be set up and that the
|
* the SCSI stack, that the QDIO queues will be set up and that the
|
||||||
* adapter will be opened.
|
* adapter will be opened.
|
||||||
*/
|
*/
|
||||||
static int zfcp_ccw_set_online(struct ccw_device *ccw_device)
|
static int zfcp_ccw_set_online(struct ccw_device *cdev)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter;
|
struct zfcp_adapter *adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
int ret = 0;
|
|
||||||
|
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
|
||||||
adapter = dev_get_drvdata(&ccw_device->dev);
|
|
||||||
|
|
||||||
if (!adapter) {
|
if (!adapter) {
|
||||||
ret = zfcp_adapter_enqueue(ccw_device);
|
adapter = zfcp_adapter_enqueue(cdev);
|
||||||
if (ret) {
|
|
||||||
dev_err(&ccw_device->dev,
|
if (IS_ERR(adapter)) {
|
||||||
|
dev_err(&cdev->dev,
|
||||||
"Setting up data structures for the "
|
"Setting up data structures for the "
|
||||||
"FCP adapter failed\n");
|
"FCP adapter failed\n");
|
||||||
goto out;
|
return PTR_ERR(adapter);
|
||||||
}
|
}
|
||||||
adapter = dev_get_drvdata(&ccw_device->dev);
|
kref_get(&adapter->ref);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* initialize request counter */
|
/* initialize request counter */
|
||||||
|
@ -177,58 +170,61 @@ static int zfcp_ccw_set_online(struct ccw_device *ccw_device)
|
||||||
zfcp_erp_adapter_reopen(adapter, ZFCP_STATUS_COMMON_ERP_FAILED,
|
zfcp_erp_adapter_reopen(adapter, ZFCP_STATUS_COMMON_ERP_FAILED,
|
||||||
"ccsonl2", NULL);
|
"ccsonl2", NULL);
|
||||||
zfcp_erp_wait(adapter);
|
zfcp_erp_wait(adapter);
|
||||||
out:
|
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
|
||||||
if (!ret)
|
|
||||||
flush_work(&adapter->scan_work);
|
flush_work(&adapter->scan_work);
|
||||||
return ret;
|
|
||||||
|
zfcp_ccw_adapter_put(adapter);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* zfcp_ccw_set_offline - set_offline function of zfcp driver
|
* zfcp_ccw_set_offline - set_offline function of zfcp driver
|
||||||
* @ccw_device: pointer to belonging ccw device
|
* @cdev: pointer to belonging ccw device
|
||||||
*
|
*
|
||||||
* This function gets called by the common i/o layer and sets an adapter
|
* This function gets called by the common i/o layer and sets an adapter
|
||||||
* into state offline.
|
* into state offline.
|
||||||
*/
|
*/
|
||||||
static int zfcp_ccw_set_offline(struct ccw_device *ccw_device)
|
static int zfcp_ccw_set_offline(struct ccw_device *cdev)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter;
|
struct zfcp_adapter *adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
|
|
||||||
|
if (!adapter)
|
||||||
|
return 0;
|
||||||
|
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
|
||||||
adapter = dev_get_drvdata(&ccw_device->dev);
|
|
||||||
zfcp_erp_adapter_shutdown(adapter, 0, "ccsoff1", NULL);
|
zfcp_erp_adapter_shutdown(adapter, 0, "ccsoff1", NULL);
|
||||||
zfcp_erp_wait(adapter);
|
zfcp_erp_wait(adapter);
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
|
||||||
|
zfcp_ccw_adapter_put(adapter);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* zfcp_ccw_notify - ccw notify function
|
* zfcp_ccw_notify - ccw notify function
|
||||||
* @ccw_device: pointer to belonging ccw device
|
* @cdev: pointer to belonging ccw device
|
||||||
* @event: indicates if adapter was detached or attached
|
* @event: indicates if adapter was detached or attached
|
||||||
*
|
*
|
||||||
* This function gets called by the common i/o layer if an adapter has gone
|
* This function gets called by the common i/o layer if an adapter has gone
|
||||||
* or reappeared.
|
* or reappeared.
|
||||||
*/
|
*/
|
||||||
static int zfcp_ccw_notify(struct ccw_device *ccw_device, int event)
|
static int zfcp_ccw_notify(struct ccw_device *cdev, int event)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter = dev_get_drvdata(&ccw_device->dev);
|
struct zfcp_adapter *adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
|
|
||||||
|
if (!adapter)
|
||||||
|
return 1;
|
||||||
|
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case CIO_GONE:
|
case CIO_GONE:
|
||||||
dev_warn(&adapter->ccw_device->dev,
|
dev_warn(&cdev->dev, "The FCP device has been detached\n");
|
||||||
"The FCP device has been detached\n");
|
|
||||||
zfcp_erp_adapter_shutdown(adapter, 0, "ccnoti1", NULL);
|
zfcp_erp_adapter_shutdown(adapter, 0, "ccnoti1", NULL);
|
||||||
break;
|
break;
|
||||||
case CIO_NO_PATH:
|
case CIO_NO_PATH:
|
||||||
dev_warn(&adapter->ccw_device->dev,
|
dev_warn(&cdev->dev,
|
||||||
"The CHPID for the FCP device is offline\n");
|
"The CHPID for the FCP device is offline\n");
|
||||||
zfcp_erp_adapter_shutdown(adapter, 0, "ccnoti2", NULL);
|
zfcp_erp_adapter_shutdown(adapter, 0, "ccnoti2", NULL);
|
||||||
break;
|
break;
|
||||||
case CIO_OPER:
|
case CIO_OPER:
|
||||||
dev_info(&adapter->ccw_device->dev,
|
dev_info(&cdev->dev, "The FCP device is operational again\n");
|
||||||
"The FCP device is operational again\n");
|
|
||||||
zfcp_erp_modify_adapter_status(adapter, "ccnoti3", NULL,
|
zfcp_erp_modify_adapter_status(adapter, "ccnoti3", NULL,
|
||||||
ZFCP_STATUS_COMMON_RUNNING,
|
ZFCP_STATUS_COMMON_RUNNING,
|
||||||
ZFCP_SET);
|
ZFCP_SET);
|
||||||
|
@ -236,11 +232,13 @@ static int zfcp_ccw_notify(struct ccw_device *ccw_device, int event)
|
||||||
"ccnoti4", NULL);
|
"ccnoti4", NULL);
|
||||||
break;
|
break;
|
||||||
case CIO_BOXED:
|
case CIO_BOXED:
|
||||||
dev_warn(&adapter->ccw_device->dev, "The FCP device "
|
dev_warn(&cdev->dev, "The FCP device did not respond within "
|
||||||
"did not respond within the specified time\n");
|
"the specified time\n");
|
||||||
zfcp_erp_adapter_shutdown(adapter, 0, "ccnoti5", NULL);
|
zfcp_erp_adapter_shutdown(adapter, 0, "ccnoti5", NULL);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
zfcp_ccw_adapter_put(adapter);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -250,18 +248,16 @@ static int zfcp_ccw_notify(struct ccw_device *ccw_device, int event)
|
||||||
*/
|
*/
|
||||||
static void zfcp_ccw_shutdown(struct ccw_device *cdev)
|
static void zfcp_ccw_shutdown(struct ccw_device *cdev)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter;
|
struct zfcp_adapter *adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
|
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
|
||||||
adapter = dev_get_drvdata(&cdev->dev);
|
|
||||||
if (!adapter)
|
if (!adapter)
|
||||||
goto out;
|
return;
|
||||||
|
|
||||||
zfcp_erp_adapter_shutdown(adapter, 0, "ccshut1", NULL);
|
zfcp_erp_adapter_shutdown(adapter, 0, "ccshut1", NULL);
|
||||||
zfcp_erp_wait(adapter);
|
zfcp_erp_wait(adapter);
|
||||||
zfcp_erp_thread_kill(adapter);
|
zfcp_erp_thread_kill(adapter);
|
||||||
out:
|
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
zfcp_ccw_adapter_put(adapter);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ccw_driver zfcp_ccw_driver = {
|
struct ccw_driver zfcp_ccw_driver = {
|
||||||
|
@ -274,18 +270,7 @@ struct ccw_driver zfcp_ccw_driver = {
|
||||||
.set_offline = zfcp_ccw_set_offline,
|
.set_offline = zfcp_ccw_set_offline,
|
||||||
.notify = zfcp_ccw_notify,
|
.notify = zfcp_ccw_notify,
|
||||||
.shutdown = zfcp_ccw_shutdown,
|
.shutdown = zfcp_ccw_shutdown,
|
||||||
.freeze = zfcp_ccw_suspend,
|
.freeze = zfcp_ccw_set_offline,
|
||||||
.thaw = zfcp_ccw_activate,
|
.thaw = zfcp_ccw_activate,
|
||||||
.restore = zfcp_ccw_activate,
|
.restore = zfcp_ccw_activate,
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* zfcp_ccw_register - ccw register function
|
|
||||||
*
|
|
||||||
* Registers the driver at the common i/o layer. This function will be called
|
|
||||||
* at module load time/system start.
|
|
||||||
*/
|
|
||||||
int __init zfcp_ccw_register(void)
|
|
||||||
{
|
|
||||||
return ccw_driver_register(&zfcp_ccw_driver);
|
|
||||||
}
|
|
||||||
|
|
|
@ -86,22 +86,17 @@ static int zfcp_cfdc_copy_to_user(void __user *user_buffer,
|
||||||
static struct zfcp_adapter *zfcp_cfdc_get_adapter(u32 devno)
|
static struct zfcp_adapter *zfcp_cfdc_get_adapter(u32 devno)
|
||||||
{
|
{
|
||||||
char busid[9];
|
char busid[9];
|
||||||
struct ccw_device *ccwdev;
|
struct ccw_device *cdev;
|
||||||
struct zfcp_adapter *adapter = NULL;
|
struct zfcp_adapter *adapter;
|
||||||
|
|
||||||
snprintf(busid, sizeof(busid), "0.0.%04x", devno);
|
snprintf(busid, sizeof(busid), "0.0.%04x", devno);
|
||||||
ccwdev = get_ccwdev_by_busid(&zfcp_ccw_driver, busid);
|
cdev = get_ccwdev_by_busid(&zfcp_ccw_driver, busid);
|
||||||
if (!ccwdev)
|
if (!cdev)
|
||||||
goto out;
|
return NULL;
|
||||||
|
|
||||||
adapter = dev_get_drvdata(&ccwdev->dev);
|
adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
if (!adapter)
|
|
||||||
goto out_put;
|
|
||||||
|
|
||||||
zfcp_adapter_get(adapter);
|
put_device(&cdev->dev);
|
||||||
out_put:
|
|
||||||
put_device(&ccwdev->dev);
|
|
||||||
out:
|
|
||||||
return adapter;
|
return adapter;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -212,7 +207,6 @@ static long zfcp_cfdc_dev_ioctl(struct file *file, unsigned int command,
|
||||||
retval = -ENXIO;
|
retval = -ENXIO;
|
||||||
goto free_buffer;
|
goto free_buffer;
|
||||||
}
|
}
|
||||||
zfcp_adapter_get(adapter);
|
|
||||||
|
|
||||||
retval = zfcp_cfdc_sg_setup(data->command, fsf_cfdc->sg,
|
retval = zfcp_cfdc_sg_setup(data->command, fsf_cfdc->sg,
|
||||||
data_user->control_file);
|
data_user->control_file);
|
||||||
|
@ -245,7 +239,7 @@ static long zfcp_cfdc_dev_ioctl(struct file *file, unsigned int command,
|
||||||
free_sg:
|
free_sg:
|
||||||
zfcp_sg_free_table(fsf_cfdc->sg, ZFCP_CFDC_PAGES);
|
zfcp_sg_free_table(fsf_cfdc->sg, ZFCP_CFDC_PAGES);
|
||||||
adapter_put:
|
adapter_put:
|
||||||
zfcp_adapter_put(adapter);
|
zfcp_ccw_adapter_put(adapter);
|
||||||
free_buffer:
|
free_buffer:
|
||||||
kfree(data);
|
kfree(data);
|
||||||
no_mem_sense:
|
no_mem_sense:
|
||||||
|
|
|
@ -13,6 +13,7 @@
|
||||||
#include <asm/debug.h>
|
#include <asm/debug.h>
|
||||||
#include "zfcp_dbf.h"
|
#include "zfcp_dbf.h"
|
||||||
#include "zfcp_ext.h"
|
#include "zfcp_ext.h"
|
||||||
|
#include "zfcp_fc.h"
|
||||||
|
|
||||||
static u32 dbfsize = 4;
|
static u32 dbfsize = 4;
|
||||||
|
|
||||||
|
@ -177,8 +178,7 @@ void _zfcp_dbf_hba_fsf_response(const char *tag2, int level,
|
||||||
|
|
||||||
case FSF_QTCB_SEND_ELS:
|
case FSF_QTCB_SEND_ELS:
|
||||||
send_els = (struct zfcp_send_els *)fsf_req->data;
|
send_els = (struct zfcp_send_els *)fsf_req->data;
|
||||||
response->u.els.d_id = qtcb->bottom.support.d_id;
|
response->u.els.d_id = ntoh24(qtcb->bottom.support.d_id);
|
||||||
response->u.els.ls_code = send_els->ls_code >> 24;
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case FSF_QTCB_ABORT_FCP_CMND:
|
case FSF_QTCB_ABORT_FCP_CMND:
|
||||||
|
@ -348,7 +348,6 @@ static void zfcp_dbf_hba_view_response(char **p,
|
||||||
|
|
||||||
case FSF_QTCB_SEND_ELS:
|
case FSF_QTCB_SEND_ELS:
|
||||||
zfcp_dbf_out(p, "d_id", "0x%06x", r->u.els.d_id);
|
zfcp_dbf_out(p, "d_id", "0x%06x", r->u.els.d_id);
|
||||||
zfcp_dbf_out(p, "ls_code", "0x%02x", r->u.els.ls_code);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case FSF_QTCB_ABORT_FCP_CMND:
|
case FSF_QTCB_ABORT_FCP_CMND:
|
||||||
|
@ -677,14 +676,14 @@ void zfcp_dbf_rec_action(char *id2, struct zfcp_erp_action *erp_action)
|
||||||
/**
|
/**
|
||||||
* zfcp_dbf_san_ct_request - trace event for issued CT request
|
* zfcp_dbf_san_ct_request - trace event for issued CT request
|
||||||
* @fsf_req: request containing issued CT data
|
* @fsf_req: request containing issued CT data
|
||||||
|
* @d_id: destination id where ct request is sent to
|
||||||
*/
|
*/
|
||||||
void zfcp_dbf_san_ct_request(struct zfcp_fsf_req *fsf_req)
|
void zfcp_dbf_san_ct_request(struct zfcp_fsf_req *fsf_req, u32 d_id)
|
||||||
{
|
{
|
||||||
struct zfcp_send_ct *ct = (struct zfcp_send_ct *)fsf_req->data;
|
struct zfcp_fsf_ct_els *ct = (struct zfcp_fsf_ct_els *)fsf_req->data;
|
||||||
struct zfcp_wka_port *wka_port = ct->wka_port;
|
struct zfcp_adapter *adapter = fsf_req->adapter;
|
||||||
struct zfcp_adapter *adapter = wka_port->adapter;
|
|
||||||
struct zfcp_dbf *dbf = adapter->dbf;
|
struct zfcp_dbf *dbf = adapter->dbf;
|
||||||
struct ct_hdr *hdr = sg_virt(ct->req);
|
struct fc_ct_hdr *hdr = sg_virt(ct->req);
|
||||||
struct zfcp_dbf_san_record *r = &dbf->san_buf;
|
struct zfcp_dbf_san_record *r = &dbf->san_buf;
|
||||||
struct zfcp_dbf_san_record_ct_request *oct = &r->u.ct_req;
|
struct zfcp_dbf_san_record_ct_request *oct = &r->u.ct_req;
|
||||||
int level = 3;
|
int level = 3;
|
||||||
|
@ -695,19 +694,18 @@ void zfcp_dbf_san_ct_request(struct zfcp_fsf_req *fsf_req)
|
||||||
strncpy(r->tag, "octc", ZFCP_DBF_TAG_SIZE);
|
strncpy(r->tag, "octc", ZFCP_DBF_TAG_SIZE);
|
||||||
r->fsf_reqid = fsf_req->req_id;
|
r->fsf_reqid = fsf_req->req_id;
|
||||||
r->fsf_seqno = fsf_req->seq_no;
|
r->fsf_seqno = fsf_req->seq_no;
|
||||||
r->s_id = fc_host_port_id(adapter->scsi_host);
|
oct->d_id = d_id;
|
||||||
r->d_id = wka_port->d_id;
|
oct->cmd_req_code = hdr->ct_cmd;
|
||||||
oct->cmd_req_code = hdr->cmd_rsp_code;
|
oct->revision = hdr->ct_rev;
|
||||||
oct->revision = hdr->revision;
|
oct->gs_type = hdr->ct_fs_type;
|
||||||
oct->gs_type = hdr->gs_type;
|
oct->gs_subtype = hdr->ct_fs_subtype;
|
||||||
oct->gs_subtype = hdr->gs_subtype;
|
oct->options = hdr->ct_options;
|
||||||
oct->options = hdr->options;
|
oct->max_res_size = hdr->ct_mr_size;
|
||||||
oct->max_res_size = hdr->max_res_size;
|
oct->len = min((int)ct->req->length - (int)sizeof(struct fc_ct_hdr),
|
||||||
oct->len = min((int)ct->req->length - (int)sizeof(struct ct_hdr),
|
|
||||||
ZFCP_DBF_SAN_MAX_PAYLOAD);
|
ZFCP_DBF_SAN_MAX_PAYLOAD);
|
||||||
debug_event(dbf->san, level, r, sizeof(*r));
|
debug_event(dbf->san, level, r, sizeof(*r));
|
||||||
zfcp_dbf_hexdump(dbf->san, r, sizeof(*r), level,
|
zfcp_dbf_hexdump(dbf->san, r, sizeof(*r), level,
|
||||||
(void *)hdr + sizeof(struct ct_hdr), oct->len);
|
(void *)hdr + sizeof(struct fc_ct_hdr), oct->len);
|
||||||
spin_unlock_irqrestore(&dbf->san_lock, flags);
|
spin_unlock_irqrestore(&dbf->san_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -717,10 +715,9 @@ void zfcp_dbf_san_ct_request(struct zfcp_fsf_req *fsf_req)
|
||||||
*/
|
*/
|
||||||
void zfcp_dbf_san_ct_response(struct zfcp_fsf_req *fsf_req)
|
void zfcp_dbf_san_ct_response(struct zfcp_fsf_req *fsf_req)
|
||||||
{
|
{
|
||||||
struct zfcp_send_ct *ct = (struct zfcp_send_ct *)fsf_req->data;
|
struct zfcp_fsf_ct_els *ct = (struct zfcp_fsf_ct_els *)fsf_req->data;
|
||||||
struct zfcp_wka_port *wka_port = ct->wka_port;
|
struct zfcp_adapter *adapter = fsf_req->adapter;
|
||||||
struct zfcp_adapter *adapter = wka_port->adapter;
|
struct fc_ct_hdr *hdr = sg_virt(ct->resp);
|
||||||
struct ct_hdr *hdr = sg_virt(ct->resp);
|
|
||||||
struct zfcp_dbf *dbf = adapter->dbf;
|
struct zfcp_dbf *dbf = adapter->dbf;
|
||||||
struct zfcp_dbf_san_record *r = &dbf->san_buf;
|
struct zfcp_dbf_san_record *r = &dbf->san_buf;
|
||||||
struct zfcp_dbf_san_record_ct_response *rct = &r->u.ct_resp;
|
struct zfcp_dbf_san_record_ct_response *rct = &r->u.ct_resp;
|
||||||
|
@ -732,25 +729,23 @@ void zfcp_dbf_san_ct_response(struct zfcp_fsf_req *fsf_req)
|
||||||
strncpy(r->tag, "rctc", ZFCP_DBF_TAG_SIZE);
|
strncpy(r->tag, "rctc", ZFCP_DBF_TAG_SIZE);
|
||||||
r->fsf_reqid = fsf_req->req_id;
|
r->fsf_reqid = fsf_req->req_id;
|
||||||
r->fsf_seqno = fsf_req->seq_no;
|
r->fsf_seqno = fsf_req->seq_no;
|
||||||
r->s_id = wka_port->d_id;
|
rct->cmd_rsp_code = hdr->ct_cmd;
|
||||||
r->d_id = fc_host_port_id(adapter->scsi_host);
|
rct->revision = hdr->ct_rev;
|
||||||
rct->cmd_rsp_code = hdr->cmd_rsp_code;
|
rct->reason_code = hdr->ct_reason;
|
||||||
rct->revision = hdr->revision;
|
rct->expl = hdr->ct_explan;
|
||||||
rct->reason_code = hdr->reason_code;
|
rct->vendor_unique = hdr->ct_vendor;
|
||||||
rct->expl = hdr->reason_code_expl;
|
rct->max_res_size = hdr->ct_mr_size;
|
||||||
rct->vendor_unique = hdr->vendor_unique;
|
rct->len = min((int)ct->resp->length - (int)sizeof(struct fc_ct_hdr),
|
||||||
rct->max_res_size = hdr->max_res_size;
|
|
||||||
rct->len = min((int)ct->resp->length - (int)sizeof(struct ct_hdr),
|
|
||||||
ZFCP_DBF_SAN_MAX_PAYLOAD);
|
ZFCP_DBF_SAN_MAX_PAYLOAD);
|
||||||
debug_event(dbf->san, level, r, sizeof(*r));
|
debug_event(dbf->san, level, r, sizeof(*r));
|
||||||
zfcp_dbf_hexdump(dbf->san, r, sizeof(*r), level,
|
zfcp_dbf_hexdump(dbf->san, r, sizeof(*r), level,
|
||||||
(void *)hdr + sizeof(struct ct_hdr), rct->len);
|
(void *)hdr + sizeof(struct fc_ct_hdr), rct->len);
|
||||||
spin_unlock_irqrestore(&dbf->san_lock, flags);
|
spin_unlock_irqrestore(&dbf->san_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void zfcp_dbf_san_els(const char *tag, int level,
|
static void zfcp_dbf_san_els(const char *tag, int level,
|
||||||
struct zfcp_fsf_req *fsf_req, u32 s_id, u32 d_id,
|
struct zfcp_fsf_req *fsf_req, u32 d_id,
|
||||||
u8 ls_code, void *buffer, int buflen)
|
void *buffer, int buflen)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter = fsf_req->adapter;
|
struct zfcp_adapter *adapter = fsf_req->adapter;
|
||||||
struct zfcp_dbf *dbf = adapter->dbf;
|
struct zfcp_dbf *dbf = adapter->dbf;
|
||||||
|
@ -762,9 +757,7 @@ static void zfcp_dbf_san_els(const char *tag, int level,
|
||||||
strncpy(rec->tag, tag, ZFCP_DBF_TAG_SIZE);
|
strncpy(rec->tag, tag, ZFCP_DBF_TAG_SIZE);
|
||||||
rec->fsf_reqid = fsf_req->req_id;
|
rec->fsf_reqid = fsf_req->req_id;
|
||||||
rec->fsf_seqno = fsf_req->seq_no;
|
rec->fsf_seqno = fsf_req->seq_no;
|
||||||
rec->s_id = s_id;
|
rec->u.els.d_id = d_id;
|
||||||
rec->d_id = d_id;
|
|
||||||
rec->u.els.ls_code = ls_code;
|
|
||||||
debug_event(dbf->san, level, rec, sizeof(*rec));
|
debug_event(dbf->san, level, rec, sizeof(*rec));
|
||||||
zfcp_dbf_hexdump(dbf->san, rec, sizeof(*rec), level,
|
zfcp_dbf_hexdump(dbf->san, rec, sizeof(*rec), level,
|
||||||
buffer, min(buflen, ZFCP_DBF_SAN_MAX_PAYLOAD));
|
buffer, min(buflen, ZFCP_DBF_SAN_MAX_PAYLOAD));
|
||||||
|
@ -777,11 +770,10 @@ static void zfcp_dbf_san_els(const char *tag, int level,
|
||||||
*/
|
*/
|
||||||
void zfcp_dbf_san_els_request(struct zfcp_fsf_req *fsf_req)
|
void zfcp_dbf_san_els_request(struct zfcp_fsf_req *fsf_req)
|
||||||
{
|
{
|
||||||
struct zfcp_send_els *els = (struct zfcp_send_els *)fsf_req->data;
|
struct zfcp_fsf_ct_els *els = (struct zfcp_fsf_ct_els *)fsf_req->data;
|
||||||
|
u32 d_id = ntoh24(fsf_req->qtcb->bottom.support.d_id);
|
||||||
|
|
||||||
zfcp_dbf_san_els("oels", 2, fsf_req,
|
zfcp_dbf_san_els("oels", 2, fsf_req, d_id,
|
||||||
fc_host_port_id(els->adapter->scsi_host),
|
|
||||||
els->d_id, *(u8 *) sg_virt(els->req),
|
|
||||||
sg_virt(els->req), els->req->length);
|
sg_virt(els->req), els->req->length);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -791,12 +783,11 @@ void zfcp_dbf_san_els_request(struct zfcp_fsf_req *fsf_req)
|
||||||
*/
|
*/
|
||||||
void zfcp_dbf_san_els_response(struct zfcp_fsf_req *fsf_req)
|
void zfcp_dbf_san_els_response(struct zfcp_fsf_req *fsf_req)
|
||||||
{
|
{
|
||||||
struct zfcp_send_els *els = (struct zfcp_send_els *)fsf_req->data;
|
struct zfcp_fsf_ct_els *els = (struct zfcp_fsf_ct_els *)fsf_req->data;
|
||||||
|
u32 d_id = ntoh24(fsf_req->qtcb->bottom.support.d_id);
|
||||||
|
|
||||||
zfcp_dbf_san_els("rels", 2, fsf_req, els->d_id,
|
zfcp_dbf_san_els("rels", 2, fsf_req, d_id,
|
||||||
fc_host_port_id(els->adapter->scsi_host),
|
sg_virt(els->resp), els->resp->length);
|
||||||
*(u8 *)sg_virt(els->req), sg_virt(els->resp),
|
|
||||||
els->resp->length);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -805,16 +796,13 @@ void zfcp_dbf_san_els_response(struct zfcp_fsf_req *fsf_req)
|
||||||
*/
|
*/
|
||||||
void zfcp_dbf_san_incoming_els(struct zfcp_fsf_req *fsf_req)
|
void zfcp_dbf_san_incoming_els(struct zfcp_fsf_req *fsf_req)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter = fsf_req->adapter;
|
|
||||||
struct fsf_status_read_buffer *buf =
|
struct fsf_status_read_buffer *buf =
|
||||||
(struct fsf_status_read_buffer *)fsf_req->data;
|
(struct fsf_status_read_buffer *)fsf_req->data;
|
||||||
int length = (int)buf->length -
|
int length = (int)buf->length -
|
||||||
(int)((void *)&buf->payload - (void *)buf);
|
(int)((void *)&buf->payload - (void *)buf);
|
||||||
|
|
||||||
zfcp_dbf_san_els("iels", 1, fsf_req, buf->d_id,
|
zfcp_dbf_san_els("iels", 1, fsf_req, ntoh24(buf->d_id),
|
||||||
fc_host_port_id(adapter->scsi_host),
|
(void *)buf->payload.data, length);
|
||||||
buf->payload.data[0], (void *)buf->payload.data,
|
|
||||||
length);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int zfcp_dbf_san_view_format(debug_info_t *id, struct debug_view *view,
|
static int zfcp_dbf_san_view_format(debug_info_t *id, struct debug_view *view,
|
||||||
|
@ -829,11 +817,10 @@ static int zfcp_dbf_san_view_format(debug_info_t *id, struct debug_view *view,
|
||||||
zfcp_dbf_tag(&p, "tag", r->tag);
|
zfcp_dbf_tag(&p, "tag", r->tag);
|
||||||
zfcp_dbf_out(&p, "fsf_reqid", "0x%0Lx", r->fsf_reqid);
|
zfcp_dbf_out(&p, "fsf_reqid", "0x%0Lx", r->fsf_reqid);
|
||||||
zfcp_dbf_out(&p, "fsf_seqno", "0x%08x", r->fsf_seqno);
|
zfcp_dbf_out(&p, "fsf_seqno", "0x%08x", r->fsf_seqno);
|
||||||
zfcp_dbf_out(&p, "s_id", "0x%06x", r->s_id);
|
|
||||||
zfcp_dbf_out(&p, "d_id", "0x%06x", r->d_id);
|
|
||||||
|
|
||||||
if (strncmp(r->tag, "octc", ZFCP_DBF_TAG_SIZE) == 0) {
|
if (strncmp(r->tag, "octc", ZFCP_DBF_TAG_SIZE) == 0) {
|
||||||
struct zfcp_dbf_san_record_ct_request *ct = &r->u.ct_req;
|
struct zfcp_dbf_san_record_ct_request *ct = &r->u.ct_req;
|
||||||
|
zfcp_dbf_out(&p, "d_id", "0x%06x", ct->d_id);
|
||||||
zfcp_dbf_out(&p, "cmd_req_code", "0x%04x", ct->cmd_req_code);
|
zfcp_dbf_out(&p, "cmd_req_code", "0x%04x", ct->cmd_req_code);
|
||||||
zfcp_dbf_out(&p, "revision", "0x%02x", ct->revision);
|
zfcp_dbf_out(&p, "revision", "0x%02x", ct->revision);
|
||||||
zfcp_dbf_out(&p, "gs_type", "0x%02x", ct->gs_type);
|
zfcp_dbf_out(&p, "gs_type", "0x%02x", ct->gs_type);
|
||||||
|
@ -852,7 +839,7 @@ static int zfcp_dbf_san_view_format(debug_info_t *id, struct debug_view *view,
|
||||||
strncmp(r->tag, "rels", ZFCP_DBF_TAG_SIZE) == 0 ||
|
strncmp(r->tag, "rels", ZFCP_DBF_TAG_SIZE) == 0 ||
|
||||||
strncmp(r->tag, "iels", ZFCP_DBF_TAG_SIZE) == 0) {
|
strncmp(r->tag, "iels", ZFCP_DBF_TAG_SIZE) == 0) {
|
||||||
struct zfcp_dbf_san_record_els *els = &r->u.els;
|
struct zfcp_dbf_san_record_els *els = &r->u.els;
|
||||||
zfcp_dbf_out(&p, "ls_code", "0x%02x", els->ls_code);
|
zfcp_dbf_out(&p, "d_id", "0x%06x", els->d_id);
|
||||||
}
|
}
|
||||||
return p - out_buf;
|
return p - out_buf;
|
||||||
}
|
}
|
||||||
|
@ -870,8 +857,9 @@ void _zfcp_dbf_scsi(const char *tag, const char *tag2, int level,
|
||||||
struct zfcp_dbf_scsi_record *rec = &dbf->scsi_buf;
|
struct zfcp_dbf_scsi_record *rec = &dbf->scsi_buf;
|
||||||
struct zfcp_dbf_dump *dump = (struct zfcp_dbf_dump *)rec;
|
struct zfcp_dbf_dump *dump = (struct zfcp_dbf_dump *)rec;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
struct fcp_rsp_iu *fcp_rsp;
|
struct fcp_resp_with_ext *fcp_rsp;
|
||||||
char *fcp_rsp_info = NULL, *fcp_sns_info = NULL;
|
struct fcp_resp_rsp_info *fcp_rsp_info = NULL;
|
||||||
|
char *fcp_sns_info = NULL;
|
||||||
int offset = 0, buflen = 0;
|
int offset = 0, buflen = 0;
|
||||||
|
|
||||||
spin_lock_irqsave(&dbf->scsi_lock, flags);
|
spin_lock_irqsave(&dbf->scsi_lock, flags);
|
||||||
|
@ -895,20 +883,22 @@ void _zfcp_dbf_scsi(const char *tag, const char *tag2, int level,
|
||||||
rec->scsi_allowed = scsi_cmnd->allowed;
|
rec->scsi_allowed = scsi_cmnd->allowed;
|
||||||
}
|
}
|
||||||
if (fsf_req != NULL) {
|
if (fsf_req != NULL) {
|
||||||
fcp_rsp = (struct fcp_rsp_iu *)
|
fcp_rsp = (struct fcp_resp_with_ext *)
|
||||||
&(fsf_req->qtcb->bottom.io.fcp_rsp);
|
&(fsf_req->qtcb->bottom.io.fcp_rsp);
|
||||||
fcp_rsp_info = (unsigned char *) &fcp_rsp[1];
|
fcp_rsp_info = (struct fcp_resp_rsp_info *)
|
||||||
fcp_sns_info =
|
&fcp_rsp[1];
|
||||||
zfcp_get_fcp_sns_info_ptr(fcp_rsp);
|
fcp_sns_info = (char *) &fcp_rsp[1];
|
||||||
|
if (fcp_rsp->resp.fr_flags & FCP_RSP_LEN_VAL)
|
||||||
|
fcp_sns_info += fcp_rsp->ext.fr_sns_len;
|
||||||
|
|
||||||
rec->rsp_validity = fcp_rsp->validity.value;
|
rec->rsp_validity = fcp_rsp->resp.fr_flags;
|
||||||
rec->rsp_scsi_status = fcp_rsp->scsi_status;
|
rec->rsp_scsi_status = fcp_rsp->resp.fr_status;
|
||||||
rec->rsp_resid = fcp_rsp->fcp_resid;
|
rec->rsp_resid = fcp_rsp->ext.fr_resid;
|
||||||
if (fcp_rsp->validity.bits.fcp_rsp_len_valid)
|
if (fcp_rsp->resp.fr_flags & FCP_RSP_LEN_VAL)
|
||||||
rec->rsp_code = *(fcp_rsp_info + 3);
|
rec->rsp_code = fcp_rsp_info->rsp_code;
|
||||||
if (fcp_rsp->validity.bits.fcp_sns_len_valid) {
|
if (fcp_rsp->resp.fr_flags & FCP_SNS_LEN_VAL) {
|
||||||
buflen = min((int)fcp_rsp->fcp_sns_len,
|
buflen = min(fcp_rsp->ext.fr_sns_len,
|
||||||
ZFCP_DBF_SCSI_MAX_FCP_SNS_INFO);
|
(u32)ZFCP_DBF_SCSI_MAX_FCP_SNS_INFO);
|
||||||
rec->sns_info_len = buflen;
|
rec->sns_info_len = buflen;
|
||||||
memcpy(rec->sns_info, fcp_sns_info,
|
memcpy(rec->sns_info, fcp_sns_info,
|
||||||
min(buflen,
|
min(buflen,
|
||||||
|
@ -1067,6 +1057,8 @@ err_out:
|
||||||
*/
|
*/
|
||||||
void zfcp_dbf_adapter_unregister(struct zfcp_dbf *dbf)
|
void zfcp_dbf_adapter_unregister(struct zfcp_dbf *dbf)
|
||||||
{
|
{
|
||||||
|
if (!dbf)
|
||||||
|
return;
|
||||||
debug_unregister(dbf->scsi);
|
debug_unregister(dbf->scsi);
|
||||||
debug_unregister(dbf->san);
|
debug_unregister(dbf->san);
|
||||||
debug_unregister(dbf->hba);
|
debug_unregister(dbf->hba);
|
||||||
|
|
|
@ -22,6 +22,7 @@
|
||||||
#ifndef ZFCP_DBF_H
|
#ifndef ZFCP_DBF_H
|
||||||
#define ZFCP_DBF_H
|
#define ZFCP_DBF_H
|
||||||
|
|
||||||
|
#include <scsi/fc/fc_fcp.h>
|
||||||
#include "zfcp_ext.h"
|
#include "zfcp_ext.h"
|
||||||
#include "zfcp_fsf.h"
|
#include "zfcp_fsf.h"
|
||||||
#include "zfcp_def.h"
|
#include "zfcp_def.h"
|
||||||
|
@ -122,7 +123,6 @@ struct zfcp_dbf_hba_record_response {
|
||||||
} unit;
|
} unit;
|
||||||
struct {
|
struct {
|
||||||
u32 d_id;
|
u32 d_id;
|
||||||
u8 ls_code;
|
|
||||||
} els;
|
} els;
|
||||||
} u;
|
} u;
|
||||||
} __attribute__ ((packed));
|
} __attribute__ ((packed));
|
||||||
|
@ -166,6 +166,7 @@ struct zfcp_dbf_san_record_ct_request {
|
||||||
u8 options;
|
u8 options;
|
||||||
u16 max_res_size;
|
u16 max_res_size;
|
||||||
u32 len;
|
u32 len;
|
||||||
|
u32 d_id;
|
||||||
} __attribute__ ((packed));
|
} __attribute__ ((packed));
|
||||||
|
|
||||||
struct zfcp_dbf_san_record_ct_response {
|
struct zfcp_dbf_san_record_ct_response {
|
||||||
|
@ -179,16 +180,13 @@ struct zfcp_dbf_san_record_ct_response {
|
||||||
} __attribute__ ((packed));
|
} __attribute__ ((packed));
|
||||||
|
|
||||||
struct zfcp_dbf_san_record_els {
|
struct zfcp_dbf_san_record_els {
|
||||||
u8 ls_code;
|
u32 d_id;
|
||||||
u32 len;
|
|
||||||
} __attribute__ ((packed));
|
} __attribute__ ((packed));
|
||||||
|
|
||||||
struct zfcp_dbf_san_record {
|
struct zfcp_dbf_san_record {
|
||||||
u8 tag[ZFCP_DBF_TAG_SIZE];
|
u8 tag[ZFCP_DBF_TAG_SIZE];
|
||||||
u64 fsf_reqid;
|
u64 fsf_reqid;
|
||||||
u32 fsf_seqno;
|
u32 fsf_seqno;
|
||||||
u32 s_id;
|
|
||||||
u32 d_id;
|
|
||||||
union {
|
union {
|
||||||
struct zfcp_dbf_san_record_ct_request ct_req;
|
struct zfcp_dbf_san_record_ct_request ct_req;
|
||||||
struct zfcp_dbf_san_record_ct_response ct_resp;
|
struct zfcp_dbf_san_record_ct_response ct_resp;
|
||||||
|
@ -343,7 +341,7 @@ static inline
|
||||||
void zfcp_dbf_scsi_devreset(const char *tag, u8 flag, struct zfcp_unit *unit,
|
void zfcp_dbf_scsi_devreset(const char *tag, u8 flag, struct zfcp_unit *unit,
|
||||||
struct scsi_cmnd *scsi_cmnd)
|
struct scsi_cmnd *scsi_cmnd)
|
||||||
{
|
{
|
||||||
zfcp_dbf_scsi(flag == FCP_TARGET_RESET ? "trst" : "lrst", tag, 1,
|
zfcp_dbf_scsi(flag == FCP_TMF_TGT_RESET ? "trst" : "lrst", tag, 1,
|
||||||
unit->port->adapter->dbf, scsi_cmnd, NULL, 0);
|
unit->port->adapter->dbf, scsi_cmnd, NULL, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -71,131 +71,6 @@
|
||||||
/* timeout value for "default timer" for fsf requests */
|
/* timeout value for "default timer" for fsf requests */
|
||||||
#define ZFCP_FSF_REQUEST_TIMEOUT (60*HZ)
|
#define ZFCP_FSF_REQUEST_TIMEOUT (60*HZ)
|
||||||
|
|
||||||
/*************** FIBRE CHANNEL PROTOCOL SPECIFIC DEFINES ********************/
|
|
||||||
|
|
||||||
/* task attribute values in FCP-2 FCP_CMND IU */
|
|
||||||
#define SIMPLE_Q 0
|
|
||||||
#define HEAD_OF_Q 1
|
|
||||||
#define ORDERED_Q 2
|
|
||||||
#define ACA_Q 4
|
|
||||||
#define UNTAGGED 5
|
|
||||||
|
|
||||||
/* task management flags in FCP-2 FCP_CMND IU */
|
|
||||||
#define FCP_CLEAR_ACA 0x40
|
|
||||||
#define FCP_TARGET_RESET 0x20
|
|
||||||
#define FCP_LOGICAL_UNIT_RESET 0x10
|
|
||||||
#define FCP_CLEAR_TASK_SET 0x04
|
|
||||||
#define FCP_ABORT_TASK_SET 0x02
|
|
||||||
|
|
||||||
#define FCP_CDB_LENGTH 16
|
|
||||||
|
|
||||||
#define ZFCP_DID_MASK 0x00FFFFFF
|
|
||||||
|
|
||||||
/* FCP(-2) FCP_CMND IU */
|
|
||||||
struct fcp_cmnd_iu {
|
|
||||||
u64 fcp_lun; /* FCP logical unit number */
|
|
||||||
u8 crn; /* command reference number */
|
|
||||||
u8 reserved0:5; /* reserved */
|
|
||||||
u8 task_attribute:3; /* task attribute */
|
|
||||||
u8 task_management_flags; /* task management flags */
|
|
||||||
u8 add_fcp_cdb_length:6; /* additional FCP_CDB length */
|
|
||||||
u8 rddata:1; /* read data */
|
|
||||||
u8 wddata:1; /* write data */
|
|
||||||
u8 fcp_cdb[FCP_CDB_LENGTH];
|
|
||||||
} __attribute__((packed));
|
|
||||||
|
|
||||||
/* FCP(-2) FCP_RSP IU */
|
|
||||||
struct fcp_rsp_iu {
|
|
||||||
u8 reserved0[10];
|
|
||||||
union {
|
|
||||||
struct {
|
|
||||||
u8 reserved1:3;
|
|
||||||
u8 fcp_conf_req:1;
|
|
||||||
u8 fcp_resid_under:1;
|
|
||||||
u8 fcp_resid_over:1;
|
|
||||||
u8 fcp_sns_len_valid:1;
|
|
||||||
u8 fcp_rsp_len_valid:1;
|
|
||||||
} bits;
|
|
||||||
u8 value;
|
|
||||||
} validity;
|
|
||||||
u8 scsi_status;
|
|
||||||
u32 fcp_resid;
|
|
||||||
u32 fcp_sns_len;
|
|
||||||
u32 fcp_rsp_len;
|
|
||||||
} __attribute__((packed));
|
|
||||||
|
|
||||||
|
|
||||||
#define RSP_CODE_GOOD 0
|
|
||||||
#define RSP_CODE_LENGTH_MISMATCH 1
|
|
||||||
#define RSP_CODE_FIELD_INVALID 2
|
|
||||||
#define RSP_CODE_RO_MISMATCH 3
|
|
||||||
#define RSP_CODE_TASKMAN_UNSUPP 4
|
|
||||||
#define RSP_CODE_TASKMAN_FAILED 5
|
|
||||||
|
|
||||||
/* see fc-fs */
|
|
||||||
#define LS_RSCN 0x61
|
|
||||||
#define LS_LOGO 0x05
|
|
||||||
#define LS_PLOGI 0x03
|
|
||||||
|
|
||||||
struct fcp_rscn_head {
|
|
||||||
u8 command;
|
|
||||||
u8 page_length; /* always 0x04 */
|
|
||||||
u16 payload_len;
|
|
||||||
} __attribute__((packed));
|
|
||||||
|
|
||||||
struct fcp_rscn_element {
|
|
||||||
u8 reserved:2;
|
|
||||||
u8 event_qual:4;
|
|
||||||
u8 addr_format:2;
|
|
||||||
u32 nport_did:24;
|
|
||||||
} __attribute__((packed));
|
|
||||||
|
|
||||||
/* see fc-ph */
|
|
||||||
struct fcp_logo {
|
|
||||||
u32 command;
|
|
||||||
u32 nport_did;
|
|
||||||
u64 nport_wwpn;
|
|
||||||
} __attribute__((packed));
|
|
||||||
|
|
||||||
/*
|
|
||||||
* FC-FS stuff
|
|
||||||
*/
|
|
||||||
#define R_A_TOV 10 /* seconds */
|
|
||||||
|
|
||||||
#define ZFCP_LS_RLS 0x0f
|
|
||||||
#define ZFCP_LS_ADISC 0x52
|
|
||||||
#define ZFCP_LS_RPS 0x56
|
|
||||||
#define ZFCP_LS_RSCN 0x61
|
|
||||||
#define ZFCP_LS_RNID 0x78
|
|
||||||
|
|
||||||
struct zfcp_ls_adisc {
|
|
||||||
u8 code;
|
|
||||||
u8 field[3];
|
|
||||||
u32 hard_nport_id;
|
|
||||||
u64 wwpn;
|
|
||||||
u64 wwnn;
|
|
||||||
u32 nport_id;
|
|
||||||
} __attribute__ ((packed));
|
|
||||||
|
|
||||||
/*
|
|
||||||
* FC-GS-2 stuff
|
|
||||||
*/
|
|
||||||
#define ZFCP_CT_REVISION 0x01
|
|
||||||
#define ZFCP_CT_DIRECTORY_SERVICE 0xFC
|
|
||||||
#define ZFCP_CT_NAME_SERVER 0x02
|
|
||||||
#define ZFCP_CT_SYNCHRONOUS 0x00
|
|
||||||
#define ZFCP_CT_SCSI_FCP 0x08
|
|
||||||
#define ZFCP_CT_UNABLE_TO_PERFORM_CMD 0x09
|
|
||||||
#define ZFCP_CT_GID_PN 0x0121
|
|
||||||
#define ZFCP_CT_GPN_FT 0x0172
|
|
||||||
#define ZFCP_CT_ACCEPT 0x8002
|
|
||||||
#define ZFCP_CT_REJECT 0x8001
|
|
||||||
|
|
||||||
/*
|
|
||||||
* FC-GS-4 stuff
|
|
||||||
*/
|
|
||||||
#define ZFCP_CT_TIMEOUT (3 * R_A_TOV)
|
|
||||||
|
|
||||||
/*************** ADAPTER/PORT/UNIT AND FSF_REQ STATUS FLAGS ******************/
|
/*************** ADAPTER/PORT/UNIT AND FSF_REQ STATUS FLAGS ******************/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -205,7 +80,6 @@ struct zfcp_ls_adisc {
|
||||||
#define ZFCP_COMMON_FLAGS 0xfff00000
|
#define ZFCP_COMMON_FLAGS 0xfff00000
|
||||||
|
|
||||||
/* common status bits */
|
/* common status bits */
|
||||||
#define ZFCP_STATUS_COMMON_REMOVE 0x80000000
|
|
||||||
#define ZFCP_STATUS_COMMON_RUNNING 0x40000000
|
#define ZFCP_STATUS_COMMON_RUNNING 0x40000000
|
||||||
#define ZFCP_STATUS_COMMON_ERP_FAILED 0x20000000
|
#define ZFCP_STATUS_COMMON_ERP_FAILED 0x20000000
|
||||||
#define ZFCP_STATUS_COMMON_UNBLOCKED 0x10000000
|
#define ZFCP_STATUS_COMMON_UNBLOCKED 0x10000000
|
||||||
|
@ -222,21 +96,10 @@ struct zfcp_ls_adisc {
|
||||||
#define ZFCP_STATUS_ADAPTER_ERP_PENDING 0x00000100
|
#define ZFCP_STATUS_ADAPTER_ERP_PENDING 0x00000100
|
||||||
#define ZFCP_STATUS_ADAPTER_LINK_UNPLUGGED 0x00000200
|
#define ZFCP_STATUS_ADAPTER_LINK_UNPLUGGED 0x00000200
|
||||||
|
|
||||||
/* FC-PH/FC-GS well-known address identifiers for generic services */
|
|
||||||
#define ZFCP_DID_WKA 0xFFFFF0
|
|
||||||
|
|
||||||
/* remote port status */
|
/* remote port status */
|
||||||
#define ZFCP_STATUS_PORT_PHYS_OPEN 0x00000001
|
#define ZFCP_STATUS_PORT_PHYS_OPEN 0x00000001
|
||||||
#define ZFCP_STATUS_PORT_LINK_TEST 0x00000002
|
#define ZFCP_STATUS_PORT_LINK_TEST 0x00000002
|
||||||
|
|
||||||
/* well known address (WKA) port status*/
|
|
||||||
enum zfcp_wka_status {
|
|
||||||
ZFCP_WKA_PORT_OFFLINE,
|
|
||||||
ZFCP_WKA_PORT_CLOSING,
|
|
||||||
ZFCP_WKA_PORT_OPENING,
|
|
||||||
ZFCP_WKA_PORT_ONLINE,
|
|
||||||
};
|
|
||||||
|
|
||||||
/* logical unit status */
|
/* logical unit status */
|
||||||
#define ZFCP_STATUS_UNIT_SHARED 0x00000004
|
#define ZFCP_STATUS_UNIT_SHARED 0x00000004
|
||||||
#define ZFCP_STATUS_UNIT_READONLY 0x00000008
|
#define ZFCP_STATUS_UNIT_READONLY 0x00000008
|
||||||
|
@ -247,10 +110,7 @@ enum zfcp_wka_status {
|
||||||
#define ZFCP_STATUS_FSFREQ_CLEANUP 0x00000010
|
#define ZFCP_STATUS_FSFREQ_CLEANUP 0x00000010
|
||||||
#define ZFCP_STATUS_FSFREQ_ABORTSUCCEEDED 0x00000040
|
#define ZFCP_STATUS_FSFREQ_ABORTSUCCEEDED 0x00000040
|
||||||
#define ZFCP_STATUS_FSFREQ_ABORTNOTNEEDED 0x00000080
|
#define ZFCP_STATUS_FSFREQ_ABORTNOTNEEDED 0x00000080
|
||||||
#define ZFCP_STATUS_FSFREQ_ABORTED 0x00000100
|
|
||||||
#define ZFCP_STATUS_FSFREQ_TMFUNCFAILED 0x00000200
|
#define ZFCP_STATUS_FSFREQ_TMFUNCFAILED 0x00000200
|
||||||
#define ZFCP_STATUS_FSFREQ_TMFUNCNOTSUPP 0x00000400
|
|
||||||
#define ZFCP_STATUS_FSFREQ_RETRY 0x00000800
|
|
||||||
#define ZFCP_STATUS_FSFREQ_DISMISSED 0x00001000
|
#define ZFCP_STATUS_FSFREQ_DISMISSED 0x00001000
|
||||||
|
|
||||||
/************************* STRUCTURE DEFINITIONS *****************************/
|
/************************* STRUCTURE DEFINITIONS *****************************/
|
||||||
|
@ -265,125 +125,10 @@ struct zfcp_adapter_mempool {
|
||||||
mempool_t *scsi_abort;
|
mempool_t *scsi_abort;
|
||||||
mempool_t *status_read_req;
|
mempool_t *status_read_req;
|
||||||
mempool_t *status_read_data;
|
mempool_t *status_read_data;
|
||||||
mempool_t *gid_pn_data;
|
mempool_t *gid_pn;
|
||||||
mempool_t *qtcb_pool;
|
mempool_t *qtcb_pool;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
|
||||||
* header for CT_IU
|
|
||||||
*/
|
|
||||||
struct ct_hdr {
|
|
||||||
u8 revision; // 0x01
|
|
||||||
u8 in_id[3]; // 0x00
|
|
||||||
u8 gs_type; // 0xFC Directory Service
|
|
||||||
u8 gs_subtype; // 0x02 Name Server
|
|
||||||
u8 options; // 0x00 single bidirectional exchange
|
|
||||||
u8 reserved0;
|
|
||||||
u16 cmd_rsp_code; // 0x0121 GID_PN, or 0x0100 GA_NXT
|
|
||||||
u16 max_res_size; // <= (4096 - 16) / 4
|
|
||||||
u8 reserved1;
|
|
||||||
u8 reason_code;
|
|
||||||
u8 reason_code_expl;
|
|
||||||
u8 vendor_unique;
|
|
||||||
} __attribute__ ((packed));
|
|
||||||
|
|
||||||
/* nameserver request CT_IU -- for requests where
|
|
||||||
* a port name is required */
|
|
||||||
struct ct_iu_gid_pn_req {
|
|
||||||
struct ct_hdr header;
|
|
||||||
u64 wwpn;
|
|
||||||
} __attribute__ ((packed));
|
|
||||||
|
|
||||||
/* FS_ACC IU and data unit for GID_PN nameserver request */
|
|
||||||
struct ct_iu_gid_pn_resp {
|
|
||||||
struct ct_hdr header;
|
|
||||||
u32 d_id;
|
|
||||||
} __attribute__ ((packed));
|
|
||||||
|
|
||||||
struct ct_iu_gpn_ft_req {
|
|
||||||
struct ct_hdr header;
|
|
||||||
u8 flags;
|
|
||||||
u8 domain_id_scope;
|
|
||||||
u8 area_id_scope;
|
|
||||||
u8 fc4_type;
|
|
||||||
} __attribute__ ((packed));
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* struct zfcp_send_ct - used to pass parameters to function zfcp_fsf_send_ct
|
|
||||||
* @wka_port: port where the request is sent to
|
|
||||||
* @req: scatter-gather list for request
|
|
||||||
* @resp: scatter-gather list for response
|
|
||||||
* @handler: handler function (called for response to the request)
|
|
||||||
* @handler_data: data passed to handler function
|
|
||||||
* @completion: completion for synchronization purposes
|
|
||||||
* @status: used to pass error status to calling function
|
|
||||||
*/
|
|
||||||
struct zfcp_send_ct {
|
|
||||||
struct zfcp_wka_port *wka_port;
|
|
||||||
struct scatterlist *req;
|
|
||||||
struct scatterlist *resp;
|
|
||||||
void (*handler)(unsigned long);
|
|
||||||
unsigned long handler_data;
|
|
||||||
struct completion *completion;
|
|
||||||
int status;
|
|
||||||
};
|
|
||||||
|
|
||||||
/* used for name server requests in error recovery */
|
|
||||||
struct zfcp_gid_pn_data {
|
|
||||||
struct zfcp_send_ct ct;
|
|
||||||
struct scatterlist req;
|
|
||||||
struct scatterlist resp;
|
|
||||||
struct ct_iu_gid_pn_req ct_iu_req;
|
|
||||||
struct ct_iu_gid_pn_resp ct_iu_resp;
|
|
||||||
struct zfcp_port *port;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* struct zfcp_send_els - used to pass parameters to function zfcp_fsf_send_els
|
|
||||||
* @adapter: adapter where request is sent from
|
|
||||||
* @port: port where ELS is destinated (port reference count has to be increased)
|
|
||||||
* @d_id: destiniation id of port where request is sent to
|
|
||||||
* @req: scatter-gather list for request
|
|
||||||
* @resp: scatter-gather list for response
|
|
||||||
* @handler: handler function (called for response to the request)
|
|
||||||
* @handler_data: data passed to handler function
|
|
||||||
* @completion: completion for synchronization purposes
|
|
||||||
* @ls_code: hex code of ELS command
|
|
||||||
* @status: used to pass error status to calling function
|
|
||||||
*/
|
|
||||||
struct zfcp_send_els {
|
|
||||||
struct zfcp_adapter *adapter;
|
|
||||||
struct zfcp_port *port;
|
|
||||||
u32 d_id;
|
|
||||||
struct scatterlist *req;
|
|
||||||
struct scatterlist *resp;
|
|
||||||
void (*handler)(unsigned long);
|
|
||||||
unsigned long handler_data;
|
|
||||||
struct completion *completion;
|
|
||||||
int ls_code;
|
|
||||||
int status;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct zfcp_wka_port {
|
|
||||||
struct zfcp_adapter *adapter;
|
|
||||||
wait_queue_head_t completion_wq;
|
|
||||||
enum zfcp_wka_status status;
|
|
||||||
atomic_t refcount;
|
|
||||||
u32 d_id;
|
|
||||||
u32 handle;
|
|
||||||
struct mutex mutex;
|
|
||||||
struct delayed_work work;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct zfcp_wka_ports {
|
|
||||||
struct zfcp_wka_port ms; /* management service */
|
|
||||||
struct zfcp_wka_port ts; /* time service */
|
|
||||||
struct zfcp_wka_port ds; /* directory service */
|
|
||||||
struct zfcp_wka_port as; /* alias service */
|
|
||||||
struct zfcp_wka_port ks; /* key distribution service */
|
|
||||||
};
|
|
||||||
|
|
||||||
struct zfcp_qdio_queue {
|
struct zfcp_qdio_queue {
|
||||||
struct qdio_buffer *sbal[QDIO_MAX_BUFFERS_PER_Q];
|
struct qdio_buffer *sbal[QDIO_MAX_BUFFERS_PER_Q];
|
||||||
u8 first; /* index of next free bfr in queue */
|
u8 first; /* index of next free bfr in queue */
|
||||||
|
@ -446,9 +191,7 @@ struct zfcp_qdio {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct zfcp_adapter {
|
struct zfcp_adapter {
|
||||||
atomic_t refcount; /* reference count */
|
struct kref ref;
|
||||||
wait_queue_head_t remove_wq; /* can be used to wait for
|
|
||||||
refcount drop to zero */
|
|
||||||
u64 peer_wwnn; /* P2P peer WWNN */
|
u64 peer_wwnn; /* P2P peer WWNN */
|
||||||
u64 peer_wwpn; /* P2P peer WWPN */
|
u64 peer_wwpn; /* P2P peer WWPN */
|
||||||
u32 peer_d_id; /* P2P peer D_ID */
|
u32 peer_d_id; /* P2P peer D_ID */
|
||||||
|
@ -461,7 +204,8 @@ struct zfcp_adapter {
|
||||||
u32 hardware_version; /* of FCP channel */
|
u32 hardware_version; /* of FCP channel */
|
||||||
u16 timer_ticks; /* time int for a tick */
|
u16 timer_ticks; /* time int for a tick */
|
||||||
struct Scsi_Host *scsi_host; /* Pointer to mid-layer */
|
struct Scsi_Host *scsi_host; /* Pointer to mid-layer */
|
||||||
struct list_head port_list_head; /* remote port list */
|
struct list_head port_list; /* remote port list */
|
||||||
|
rwlock_t port_list_lock; /* port list lock */
|
||||||
unsigned long req_no; /* unique FSF req number */
|
unsigned long req_no; /* unique FSF req number */
|
||||||
struct list_head *req_list; /* list of pending reqs */
|
struct list_head *req_list; /* list of pending reqs */
|
||||||
spinlock_t req_list_lock; /* request list lock */
|
spinlock_t req_list_lock; /* request list lock */
|
||||||
|
@ -485,7 +229,7 @@ struct zfcp_adapter {
|
||||||
u32 erp_low_mem_count; /* nr of erp actions waiting
|
u32 erp_low_mem_count; /* nr of erp actions waiting
|
||||||
for memory */
|
for memory */
|
||||||
struct task_struct *erp_thread;
|
struct task_struct *erp_thread;
|
||||||
struct zfcp_wka_ports *gs; /* generic services */
|
struct zfcp_fc_wka_ports *gs; /* generic services */
|
||||||
struct zfcp_dbf *dbf; /* debug traces */
|
struct zfcp_dbf *dbf; /* debug traces */
|
||||||
struct zfcp_adapter_mempool pool; /* Adapter memory pools */
|
struct zfcp_adapter_mempool pool; /* Adapter memory pools */
|
||||||
struct fc_host_statistics *fc_stats;
|
struct fc_host_statistics *fc_stats;
|
||||||
|
@ -500,11 +244,9 @@ struct zfcp_port {
|
||||||
struct device sysfs_device; /* sysfs device */
|
struct device sysfs_device; /* sysfs device */
|
||||||
struct fc_rport *rport; /* rport of fc transport class */
|
struct fc_rport *rport; /* rport of fc transport class */
|
||||||
struct list_head list; /* list of remote ports */
|
struct list_head list; /* list of remote ports */
|
||||||
atomic_t refcount; /* reference count */
|
|
||||||
wait_queue_head_t remove_wq; /* can be used to wait for
|
|
||||||
refcount drop to zero */
|
|
||||||
struct zfcp_adapter *adapter; /* adapter used to access port */
|
struct zfcp_adapter *adapter; /* adapter used to access port */
|
||||||
struct list_head unit_list_head; /* head of logical unit list */
|
struct list_head unit_list; /* head of logical unit list */
|
||||||
|
rwlock_t unit_list_lock; /* unit list lock */
|
||||||
atomic_t status; /* status of this remote port */
|
atomic_t status; /* status of this remote port */
|
||||||
u64 wwnn; /* WWNN if known */
|
u64 wwnn; /* WWNN if known */
|
||||||
u64 wwpn; /* WWPN */
|
u64 wwpn; /* WWPN */
|
||||||
|
@ -523,9 +265,6 @@ struct zfcp_port {
|
||||||
struct zfcp_unit {
|
struct zfcp_unit {
|
||||||
struct device sysfs_device; /* sysfs device */
|
struct device sysfs_device; /* sysfs device */
|
||||||
struct list_head list; /* list of logical units */
|
struct list_head list; /* list of logical units */
|
||||||
atomic_t refcount; /* reference count */
|
|
||||||
wait_queue_head_t remove_wq; /* can be used to wait for
|
|
||||||
refcount drop to zero */
|
|
||||||
struct zfcp_port *port; /* remote port of unit */
|
struct zfcp_port *port; /* remote port of unit */
|
||||||
atomic_t status; /* status of this logical unit */
|
atomic_t status; /* status of this logical unit */
|
||||||
u64 fcp_lun; /* own FCP_LUN */
|
u64 fcp_lun; /* own FCP_LUN */
|
||||||
|
@ -601,14 +340,11 @@ struct zfcp_fsf_req {
|
||||||
struct zfcp_data {
|
struct zfcp_data {
|
||||||
struct scsi_host_template scsi_host_template;
|
struct scsi_host_template scsi_host_template;
|
||||||
struct scsi_transport_template *scsi_transport_template;
|
struct scsi_transport_template *scsi_transport_template;
|
||||||
rwlock_t config_lock; /* serialises changes
|
|
||||||
to adapter/port/unit
|
|
||||||
lists */
|
|
||||||
struct mutex config_mutex;
|
|
||||||
struct kmem_cache *gpn_ft_cache;
|
struct kmem_cache *gpn_ft_cache;
|
||||||
struct kmem_cache *qtcb_cache;
|
struct kmem_cache *qtcb_cache;
|
||||||
struct kmem_cache *sr_buffer_cache;
|
struct kmem_cache *sr_buffer_cache;
|
||||||
struct kmem_cache *gid_pn_cache;
|
struct kmem_cache *gid_pn_cache;
|
||||||
|
struct kmem_cache *adisc_cache;
|
||||||
};
|
};
|
||||||
|
|
||||||
/********************** ZFCP SPECIFIC DEFINES ********************************/
|
/********************** ZFCP SPECIFIC DEFINES ********************************/
|
||||||
|
@ -657,47 +393,4 @@ zfcp_reqlist_find_safe(struct zfcp_adapter *adapter, struct zfcp_fsf_req *req)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* functions needed for reference/usage counting
|
|
||||||
*/
|
|
||||||
|
|
||||||
static inline void
|
|
||||||
zfcp_unit_get(struct zfcp_unit *unit)
|
|
||||||
{
|
|
||||||
atomic_inc(&unit->refcount);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void
|
|
||||||
zfcp_unit_put(struct zfcp_unit *unit)
|
|
||||||
{
|
|
||||||
if (atomic_dec_return(&unit->refcount) == 0)
|
|
||||||
wake_up(&unit->remove_wq);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void
|
|
||||||
zfcp_port_get(struct zfcp_port *port)
|
|
||||||
{
|
|
||||||
atomic_inc(&port->refcount);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void
|
|
||||||
zfcp_port_put(struct zfcp_port *port)
|
|
||||||
{
|
|
||||||
if (atomic_dec_return(&port->refcount) == 0)
|
|
||||||
wake_up(&port->remove_wq);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void
|
|
||||||
zfcp_adapter_get(struct zfcp_adapter *adapter)
|
|
||||||
{
|
|
||||||
atomic_inc(&adapter->refcount);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void
|
|
||||||
zfcp_adapter_put(struct zfcp_adapter *adapter)
|
|
||||||
{
|
|
||||||
if (atomic_dec_return(&adapter->refcount) == 0)
|
|
||||||
wake_up(&adapter->remove_wq);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif /* ZFCP_DEF_H */
|
#endif /* ZFCP_DEF_H */
|
||||||
|
|
|
@ -99,9 +99,12 @@ static void zfcp_erp_action_dismiss_port(struct zfcp_port *port)
|
||||||
|
|
||||||
if (atomic_read(&port->status) & ZFCP_STATUS_COMMON_ERP_INUSE)
|
if (atomic_read(&port->status) & ZFCP_STATUS_COMMON_ERP_INUSE)
|
||||||
zfcp_erp_action_dismiss(&port->erp_action);
|
zfcp_erp_action_dismiss(&port->erp_action);
|
||||||
else
|
else {
|
||||||
list_for_each_entry(unit, &port->unit_list_head, list)
|
read_lock(&port->unit_list_lock);
|
||||||
|
list_for_each_entry(unit, &port->unit_list, list)
|
||||||
zfcp_erp_action_dismiss_unit(unit);
|
zfcp_erp_action_dismiss_unit(unit);
|
||||||
|
read_unlock(&port->unit_list_lock);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void zfcp_erp_action_dismiss_adapter(struct zfcp_adapter *adapter)
|
static void zfcp_erp_action_dismiss_adapter(struct zfcp_adapter *adapter)
|
||||||
|
@ -110,9 +113,12 @@ static void zfcp_erp_action_dismiss_adapter(struct zfcp_adapter *adapter)
|
||||||
|
|
||||||
if (atomic_read(&adapter->status) & ZFCP_STATUS_COMMON_ERP_INUSE)
|
if (atomic_read(&adapter->status) & ZFCP_STATUS_COMMON_ERP_INUSE)
|
||||||
zfcp_erp_action_dismiss(&adapter->erp_action);
|
zfcp_erp_action_dismiss(&adapter->erp_action);
|
||||||
else
|
else {
|
||||||
list_for_each_entry(port, &adapter->port_list_head, list)
|
read_lock(&adapter->port_list_lock);
|
||||||
|
list_for_each_entry(port, &adapter->port_list, list)
|
||||||
zfcp_erp_action_dismiss_port(port);
|
zfcp_erp_action_dismiss_port(port);
|
||||||
|
read_unlock(&adapter->port_list_lock);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int zfcp_erp_required_act(int want, struct zfcp_adapter *adapter,
|
static int zfcp_erp_required_act(int want, struct zfcp_adapter *adapter,
|
||||||
|
@ -168,7 +174,8 @@ static struct zfcp_erp_action *zfcp_erp_setup_act(int need,
|
||||||
|
|
||||||
switch (need) {
|
switch (need) {
|
||||||
case ZFCP_ERP_ACTION_REOPEN_UNIT:
|
case ZFCP_ERP_ACTION_REOPEN_UNIT:
|
||||||
zfcp_unit_get(unit);
|
if (!get_device(&unit->sysfs_device))
|
||||||
|
return NULL;
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_ERP_INUSE, &unit->status);
|
atomic_set_mask(ZFCP_STATUS_COMMON_ERP_INUSE, &unit->status);
|
||||||
erp_action = &unit->erp_action;
|
erp_action = &unit->erp_action;
|
||||||
if (!(atomic_read(&unit->status) & ZFCP_STATUS_COMMON_RUNNING))
|
if (!(atomic_read(&unit->status) & ZFCP_STATUS_COMMON_RUNNING))
|
||||||
|
@ -177,7 +184,8 @@ static struct zfcp_erp_action *zfcp_erp_setup_act(int need,
|
||||||
|
|
||||||
case ZFCP_ERP_ACTION_REOPEN_PORT:
|
case ZFCP_ERP_ACTION_REOPEN_PORT:
|
||||||
case ZFCP_ERP_ACTION_REOPEN_PORT_FORCED:
|
case ZFCP_ERP_ACTION_REOPEN_PORT_FORCED:
|
||||||
zfcp_port_get(port);
|
if (!get_device(&port->sysfs_device))
|
||||||
|
return NULL;
|
||||||
zfcp_erp_action_dismiss_port(port);
|
zfcp_erp_action_dismiss_port(port);
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_ERP_INUSE, &port->status);
|
atomic_set_mask(ZFCP_STATUS_COMMON_ERP_INUSE, &port->status);
|
||||||
erp_action = &port->erp_action;
|
erp_action = &port->erp_action;
|
||||||
|
@ -186,7 +194,7 @@ static struct zfcp_erp_action *zfcp_erp_setup_act(int need,
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case ZFCP_ERP_ACTION_REOPEN_ADAPTER:
|
case ZFCP_ERP_ACTION_REOPEN_ADAPTER:
|
||||||
zfcp_adapter_get(adapter);
|
kref_get(&adapter->ref);
|
||||||
zfcp_erp_action_dismiss_adapter(adapter);
|
zfcp_erp_action_dismiss_adapter(adapter);
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_ERP_INUSE, &adapter->status);
|
atomic_set_mask(ZFCP_STATUS_COMMON_ERP_INUSE, &adapter->status);
|
||||||
erp_action = &adapter->erp_action;
|
erp_action = &adapter->erp_action;
|
||||||
|
@ -264,11 +272,16 @@ void zfcp_erp_adapter_reopen(struct zfcp_adapter *adapter, int clear,
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
zfcp_erp_adapter_block(adapter, clear);
|
||||||
write_lock(&adapter->erp_lock);
|
zfcp_scsi_schedule_rports_block(adapter);
|
||||||
_zfcp_erp_adapter_reopen(adapter, clear, id, ref);
|
|
||||||
write_unlock(&adapter->erp_lock);
|
write_lock_irqsave(&adapter->erp_lock, flags);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
if (atomic_read(&adapter->status) & ZFCP_STATUS_COMMON_ERP_FAILED)
|
||||||
|
zfcp_erp_adapter_failed(adapter, "erareo1", NULL);
|
||||||
|
else
|
||||||
|
zfcp_erp_action_enqueue(ZFCP_ERP_ACTION_REOPEN_ADAPTER, adapter,
|
||||||
|
NULL, NULL, id, ref);
|
||||||
|
write_unlock_irqrestore(&adapter->erp_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -345,11 +358,9 @@ void zfcp_erp_port_forced_reopen(struct zfcp_port *port, int clear, char *id,
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
struct zfcp_adapter *adapter = port->adapter;
|
struct zfcp_adapter *adapter = port->adapter;
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
write_lock_irqsave(&adapter->erp_lock, flags);
|
||||||
write_lock(&adapter->erp_lock);
|
|
||||||
_zfcp_erp_port_forced_reopen(port, clear, id, ref);
|
_zfcp_erp_port_forced_reopen(port, clear, id, ref);
|
||||||
write_unlock(&adapter->erp_lock);
|
write_unlock_irqrestore(&adapter->erp_lock, flags);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _zfcp_erp_port_reopen(struct zfcp_port *port, int clear, char *id,
|
static int _zfcp_erp_port_reopen(struct zfcp_port *port, int clear, char *id,
|
||||||
|
@ -377,15 +388,13 @@ static int _zfcp_erp_port_reopen(struct zfcp_port *port, int clear, char *id,
|
||||||
*/
|
*/
|
||||||
int zfcp_erp_port_reopen(struct zfcp_port *port, int clear, char *id, void *ref)
|
int zfcp_erp_port_reopen(struct zfcp_port *port, int clear, char *id, void *ref)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
|
||||||
int retval;
|
int retval;
|
||||||
|
unsigned long flags;
|
||||||
struct zfcp_adapter *adapter = port->adapter;
|
struct zfcp_adapter *adapter = port->adapter;
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
write_lock_irqsave(&adapter->erp_lock, flags);
|
||||||
write_lock(&adapter->erp_lock);
|
|
||||||
retval = _zfcp_erp_port_reopen(port, clear, id, ref);
|
retval = _zfcp_erp_port_reopen(port, clear, id, ref);
|
||||||
write_unlock(&adapter->erp_lock);
|
write_unlock_irqrestore(&adapter->erp_lock, flags);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
|
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
@ -424,11 +433,9 @@ void zfcp_erp_unit_reopen(struct zfcp_unit *unit, int clear, char *id,
|
||||||
struct zfcp_port *port = unit->port;
|
struct zfcp_port *port = unit->port;
|
||||||
struct zfcp_adapter *adapter = port->adapter;
|
struct zfcp_adapter *adapter = port->adapter;
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
write_lock_irqsave(&adapter->erp_lock, flags);
|
||||||
write_lock(&adapter->erp_lock);
|
|
||||||
_zfcp_erp_unit_reopen(unit, clear, id, ref);
|
_zfcp_erp_unit_reopen(unit, clear, id, ref);
|
||||||
write_unlock(&adapter->erp_lock);
|
write_unlock_irqrestore(&adapter->erp_lock, flags);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int status_change_set(unsigned long mask, atomic_t *status)
|
static int status_change_set(unsigned long mask, atomic_t *status)
|
||||||
|
@ -540,8 +547,10 @@ static void _zfcp_erp_port_reopen_all(struct zfcp_adapter *adapter,
|
||||||
{
|
{
|
||||||
struct zfcp_port *port;
|
struct zfcp_port *port;
|
||||||
|
|
||||||
list_for_each_entry(port, &adapter->port_list_head, list)
|
read_lock(&adapter->port_list_lock);
|
||||||
|
list_for_each_entry(port, &adapter->port_list, list)
|
||||||
_zfcp_erp_port_reopen(port, clear, id, ref);
|
_zfcp_erp_port_reopen(port, clear, id, ref);
|
||||||
|
read_unlock(&adapter->port_list_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void _zfcp_erp_unit_reopen_all(struct zfcp_port *port, int clear,
|
static void _zfcp_erp_unit_reopen_all(struct zfcp_port *port, int clear,
|
||||||
|
@ -549,8 +558,10 @@ static void _zfcp_erp_unit_reopen_all(struct zfcp_port *port, int clear,
|
||||||
{
|
{
|
||||||
struct zfcp_unit *unit;
|
struct zfcp_unit *unit;
|
||||||
|
|
||||||
list_for_each_entry(unit, &port->unit_list_head, list)
|
read_lock(&port->unit_list_lock);
|
||||||
|
list_for_each_entry(unit, &port->unit_list, list)
|
||||||
_zfcp_erp_unit_reopen(unit, clear, id, ref);
|
_zfcp_erp_unit_reopen(unit, clear, id, ref);
|
||||||
|
read_unlock(&port->unit_list_lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void zfcp_erp_strategy_followup_failed(struct zfcp_erp_action *act)
|
static void zfcp_erp_strategy_followup_failed(struct zfcp_erp_action *act)
|
||||||
|
@ -590,16 +601,14 @@ static void zfcp_erp_wakeup(struct zfcp_adapter *adapter)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
read_lock_irqsave(&adapter->erp_lock, flags);
|
||||||
read_lock(&adapter->erp_lock);
|
|
||||||
if (list_empty(&adapter->erp_ready_head) &&
|
if (list_empty(&adapter->erp_ready_head) &&
|
||||||
list_empty(&adapter->erp_running_head)) {
|
list_empty(&adapter->erp_running_head)) {
|
||||||
atomic_clear_mask(ZFCP_STATUS_ADAPTER_ERP_PENDING,
|
atomic_clear_mask(ZFCP_STATUS_ADAPTER_ERP_PENDING,
|
||||||
&adapter->status);
|
&adapter->status);
|
||||||
wake_up(&adapter->erp_done_wqh);
|
wake_up(&adapter->erp_done_wqh);
|
||||||
}
|
}
|
||||||
read_unlock(&adapter->erp_lock);
|
read_unlock_irqrestore(&adapter->erp_lock, flags);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int zfcp_erp_adapter_strategy_open_qdio(struct zfcp_erp_action *act)
|
static int zfcp_erp_adapter_strategy_open_qdio(struct zfcp_erp_action *act)
|
||||||
|
@ -1170,28 +1179,28 @@ static void zfcp_erp_action_cleanup(struct zfcp_erp_action *act, int result)
|
||||||
switch (act->action) {
|
switch (act->action) {
|
||||||
case ZFCP_ERP_ACTION_REOPEN_UNIT:
|
case ZFCP_ERP_ACTION_REOPEN_UNIT:
|
||||||
if ((result == ZFCP_ERP_SUCCEEDED) && !unit->device) {
|
if ((result == ZFCP_ERP_SUCCEEDED) && !unit->device) {
|
||||||
zfcp_unit_get(unit);
|
get_device(&unit->sysfs_device);
|
||||||
if (scsi_queue_work(unit->port->adapter->scsi_host,
|
if (scsi_queue_work(unit->port->adapter->scsi_host,
|
||||||
&unit->scsi_work) <= 0)
|
&unit->scsi_work) <= 0)
|
||||||
zfcp_unit_put(unit);
|
put_device(&unit->sysfs_device);
|
||||||
}
|
}
|
||||||
zfcp_unit_put(unit);
|
put_device(&unit->sysfs_device);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case ZFCP_ERP_ACTION_REOPEN_PORT_FORCED:
|
case ZFCP_ERP_ACTION_REOPEN_PORT_FORCED:
|
||||||
case ZFCP_ERP_ACTION_REOPEN_PORT:
|
case ZFCP_ERP_ACTION_REOPEN_PORT:
|
||||||
if (result == ZFCP_ERP_SUCCEEDED)
|
if (result == ZFCP_ERP_SUCCEEDED)
|
||||||
zfcp_scsi_schedule_rport_register(port);
|
zfcp_scsi_schedule_rport_register(port);
|
||||||
zfcp_port_put(port);
|
put_device(&port->sysfs_device);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case ZFCP_ERP_ACTION_REOPEN_ADAPTER:
|
case ZFCP_ERP_ACTION_REOPEN_ADAPTER:
|
||||||
if (result == ZFCP_ERP_SUCCEEDED) {
|
if (result == ZFCP_ERP_SUCCEEDED) {
|
||||||
register_service_level(&adapter->service_level);
|
register_service_level(&adapter->service_level);
|
||||||
schedule_work(&adapter->scan_work);
|
queue_work(adapter->work_queue, &adapter->scan_work);
|
||||||
} else
|
} else
|
||||||
unregister_service_level(&adapter->service_level);
|
unregister_service_level(&adapter->service_level);
|
||||||
zfcp_adapter_put(adapter);
|
kref_put(&adapter->ref, zfcp_adapter_release);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1214,12 +1223,12 @@ static int zfcp_erp_strategy_do_action(struct zfcp_erp_action *erp_action)
|
||||||
static int zfcp_erp_strategy(struct zfcp_erp_action *erp_action)
|
static int zfcp_erp_strategy(struct zfcp_erp_action *erp_action)
|
||||||
{
|
{
|
||||||
int retval;
|
int retval;
|
||||||
struct zfcp_adapter *adapter = erp_action->adapter;
|
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
struct zfcp_adapter *adapter = erp_action->adapter;
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
kref_get(&adapter->ref);
|
||||||
write_lock(&adapter->erp_lock);
|
|
||||||
|
|
||||||
|
write_lock_irqsave(&adapter->erp_lock, flags);
|
||||||
zfcp_erp_strategy_check_fsfreq(erp_action);
|
zfcp_erp_strategy_check_fsfreq(erp_action);
|
||||||
|
|
||||||
if (erp_action->status & ZFCP_STATUS_ERP_DISMISSED) {
|
if (erp_action->status & ZFCP_STATUS_ERP_DISMISSED) {
|
||||||
|
@ -1231,11 +1240,9 @@ static int zfcp_erp_strategy(struct zfcp_erp_action *erp_action)
|
||||||
zfcp_erp_action_to_running(erp_action);
|
zfcp_erp_action_to_running(erp_action);
|
||||||
|
|
||||||
/* no lock to allow for blocking operations */
|
/* no lock to allow for blocking operations */
|
||||||
write_unlock(&adapter->erp_lock);
|
write_unlock_irqrestore(&adapter->erp_lock, flags);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
retval = zfcp_erp_strategy_do_action(erp_action);
|
retval = zfcp_erp_strategy_do_action(erp_action);
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
write_lock_irqsave(&adapter->erp_lock, flags);
|
||||||
write_lock(&adapter->erp_lock);
|
|
||||||
|
|
||||||
if (erp_action->status & ZFCP_STATUS_ERP_DISMISSED)
|
if (erp_action->status & ZFCP_STATUS_ERP_DISMISSED)
|
||||||
retval = ZFCP_ERP_CONTINUES;
|
retval = ZFCP_ERP_CONTINUES;
|
||||||
|
@ -1273,12 +1280,12 @@ static int zfcp_erp_strategy(struct zfcp_erp_action *erp_action)
|
||||||
zfcp_erp_strategy_followup_failed(erp_action);
|
zfcp_erp_strategy_followup_failed(erp_action);
|
||||||
|
|
||||||
unlock:
|
unlock:
|
||||||
write_unlock(&adapter->erp_lock);
|
write_unlock_irqrestore(&adapter->erp_lock, flags);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
|
|
||||||
if (retval != ZFCP_ERP_CONTINUES)
|
if (retval != ZFCP_ERP_CONTINUES)
|
||||||
zfcp_erp_action_cleanup(erp_action, retval);
|
zfcp_erp_action_cleanup(erp_action, retval);
|
||||||
|
|
||||||
|
kref_put(&adapter->ref, zfcp_adapter_release);
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1415,6 +1422,7 @@ void zfcp_erp_modify_adapter_status(struct zfcp_adapter *adapter, char *id,
|
||||||
void *ref, u32 mask, int set_or_clear)
|
void *ref, u32 mask, int set_or_clear)
|
||||||
{
|
{
|
||||||
struct zfcp_port *port;
|
struct zfcp_port *port;
|
||||||
|
unsigned long flags;
|
||||||
u32 common_mask = mask & ZFCP_COMMON_FLAGS;
|
u32 common_mask = mask & ZFCP_COMMON_FLAGS;
|
||||||
|
|
||||||
if (set_or_clear == ZFCP_SET) {
|
if (set_or_clear == ZFCP_SET) {
|
||||||
|
@ -1429,10 +1437,13 @@ void zfcp_erp_modify_adapter_status(struct zfcp_adapter *adapter, char *id,
|
||||||
atomic_set(&adapter->erp_counter, 0);
|
atomic_set(&adapter->erp_counter, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (common_mask)
|
if (common_mask) {
|
||||||
list_for_each_entry(port, &adapter->port_list_head, list)
|
read_lock_irqsave(&adapter->port_list_lock, flags);
|
||||||
|
list_for_each_entry(port, &adapter->port_list, list)
|
||||||
zfcp_erp_modify_port_status(port, id, ref, common_mask,
|
zfcp_erp_modify_port_status(port, id, ref, common_mask,
|
||||||
set_or_clear);
|
set_or_clear);
|
||||||
|
read_unlock_irqrestore(&adapter->port_list_lock, flags);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1449,6 +1460,7 @@ void zfcp_erp_modify_port_status(struct zfcp_port *port, char *id, void *ref,
|
||||||
u32 mask, int set_or_clear)
|
u32 mask, int set_or_clear)
|
||||||
{
|
{
|
||||||
struct zfcp_unit *unit;
|
struct zfcp_unit *unit;
|
||||||
|
unsigned long flags;
|
||||||
u32 common_mask = mask & ZFCP_COMMON_FLAGS;
|
u32 common_mask = mask & ZFCP_COMMON_FLAGS;
|
||||||
|
|
||||||
if (set_or_clear == ZFCP_SET) {
|
if (set_or_clear == ZFCP_SET) {
|
||||||
|
@ -1463,10 +1475,13 @@ void zfcp_erp_modify_port_status(struct zfcp_port *port, char *id, void *ref,
|
||||||
atomic_set(&port->erp_counter, 0);
|
atomic_set(&port->erp_counter, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (common_mask)
|
if (common_mask) {
|
||||||
list_for_each_entry(unit, &port->unit_list_head, list)
|
read_lock_irqsave(&port->unit_list_lock, flags);
|
||||||
|
list_for_each_entry(unit, &port->unit_list, list)
|
||||||
zfcp_erp_modify_unit_status(unit, id, ref, common_mask,
|
zfcp_erp_modify_unit_status(unit, id, ref, common_mask,
|
||||||
set_or_clear);
|
set_or_clear);
|
||||||
|
read_unlock_irqrestore(&port->unit_list_lock, flags);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1502,12 +1517,8 @@ void zfcp_erp_modify_unit_status(struct zfcp_unit *unit, char *id, void *ref,
|
||||||
*/
|
*/
|
||||||
void zfcp_erp_port_boxed(struct zfcp_port *port, char *id, void *ref)
|
void zfcp_erp_port_boxed(struct zfcp_port *port, char *id, void *ref)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
|
||||||
zfcp_erp_modify_port_status(port, id, ref,
|
zfcp_erp_modify_port_status(port, id, ref,
|
||||||
ZFCP_STATUS_COMMON_ACCESS_BOXED, ZFCP_SET);
|
ZFCP_STATUS_COMMON_ACCESS_BOXED, ZFCP_SET);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
zfcp_erp_port_reopen(port, ZFCP_STATUS_COMMON_ERP_FAILED, id, ref);
|
zfcp_erp_port_reopen(port, ZFCP_STATUS_COMMON_ERP_FAILED, id, ref);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1535,13 +1546,9 @@ void zfcp_erp_unit_boxed(struct zfcp_unit *unit, char *id, void *ref)
|
||||||
*/
|
*/
|
||||||
void zfcp_erp_port_access_denied(struct zfcp_port *port, char *id, void *ref)
|
void zfcp_erp_port_access_denied(struct zfcp_port *port, char *id, void *ref)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
|
||||||
zfcp_erp_modify_port_status(port, id, ref,
|
zfcp_erp_modify_port_status(port, id, ref,
|
||||||
ZFCP_STATUS_COMMON_ERP_FAILED |
|
ZFCP_STATUS_COMMON_ERP_FAILED |
|
||||||
ZFCP_STATUS_COMMON_ACCESS_DENIED, ZFCP_SET);
|
ZFCP_STATUS_COMMON_ACCESS_DENIED, ZFCP_SET);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1574,12 +1581,15 @@ static void zfcp_erp_port_access_changed(struct zfcp_port *port, char *id,
|
||||||
void *ref)
|
void *ref)
|
||||||
{
|
{
|
||||||
struct zfcp_unit *unit;
|
struct zfcp_unit *unit;
|
||||||
|
unsigned long flags;
|
||||||
int status = atomic_read(&port->status);
|
int status = atomic_read(&port->status);
|
||||||
|
|
||||||
if (!(status & (ZFCP_STATUS_COMMON_ACCESS_DENIED |
|
if (!(status & (ZFCP_STATUS_COMMON_ACCESS_DENIED |
|
||||||
ZFCP_STATUS_COMMON_ACCESS_BOXED))) {
|
ZFCP_STATUS_COMMON_ACCESS_BOXED))) {
|
||||||
list_for_each_entry(unit, &port->unit_list_head, list)
|
read_lock_irqsave(&port->unit_list_lock, flags);
|
||||||
|
list_for_each_entry(unit, &port->unit_list, list)
|
||||||
zfcp_erp_unit_access_changed(unit, id, ref);
|
zfcp_erp_unit_access_changed(unit, id, ref);
|
||||||
|
read_unlock_irqrestore(&port->unit_list_lock, flags);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1595,14 +1605,14 @@ static void zfcp_erp_port_access_changed(struct zfcp_port *port, char *id,
|
||||||
void zfcp_erp_adapter_access_changed(struct zfcp_adapter *adapter, char *id,
|
void zfcp_erp_adapter_access_changed(struct zfcp_adapter *adapter, char *id,
|
||||||
void *ref)
|
void *ref)
|
||||||
{
|
{
|
||||||
struct zfcp_port *port;
|
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
struct zfcp_port *port;
|
||||||
|
|
||||||
if (adapter->connection_features & FSF_FEATURE_NPIV_MODE)
|
if (adapter->connection_features & FSF_FEATURE_NPIV_MODE)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
read_lock_irqsave(&adapter->port_list_lock, flags);
|
||||||
list_for_each_entry(port, &adapter->port_list_head, list)
|
list_for_each_entry(port, &adapter->port_list, list)
|
||||||
zfcp_erp_port_access_changed(port, id, ref);
|
zfcp_erp_port_access_changed(port, id, ref);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
read_unlock_irqrestore(&adapter->port_list_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,26 +9,31 @@
|
||||||
#ifndef ZFCP_EXT_H
|
#ifndef ZFCP_EXT_H
|
||||||
#define ZFCP_EXT_H
|
#define ZFCP_EXT_H
|
||||||
|
|
||||||
|
#include <linux/types.h>
|
||||||
|
#include <scsi/fc/fc_els.h>
|
||||||
#include "zfcp_def.h"
|
#include "zfcp_def.h"
|
||||||
|
#include "zfcp_fc.h"
|
||||||
|
|
||||||
/* zfcp_aux.c */
|
/* zfcp_aux.c */
|
||||||
extern struct zfcp_unit *zfcp_get_unit_by_lun(struct zfcp_port *, u64);
|
extern struct zfcp_unit *zfcp_get_unit_by_lun(struct zfcp_port *, u64);
|
||||||
extern struct zfcp_port *zfcp_get_port_by_wwpn(struct zfcp_adapter *, u64);
|
extern struct zfcp_port *zfcp_get_port_by_wwpn(struct zfcp_adapter *, u64);
|
||||||
extern int zfcp_adapter_enqueue(struct ccw_device *);
|
extern struct zfcp_adapter *zfcp_adapter_enqueue(struct ccw_device *);
|
||||||
extern void zfcp_adapter_dequeue(struct zfcp_adapter *);
|
|
||||||
extern struct zfcp_port *zfcp_port_enqueue(struct zfcp_adapter *, u64, u32,
|
extern struct zfcp_port *zfcp_port_enqueue(struct zfcp_adapter *, u64, u32,
|
||||||
u32);
|
u32);
|
||||||
extern void zfcp_port_dequeue(struct zfcp_port *);
|
|
||||||
extern struct zfcp_unit *zfcp_unit_enqueue(struct zfcp_port *, u64);
|
extern struct zfcp_unit *zfcp_unit_enqueue(struct zfcp_port *, u64);
|
||||||
extern void zfcp_unit_dequeue(struct zfcp_unit *);
|
|
||||||
extern int zfcp_reqlist_isempty(struct zfcp_adapter *);
|
extern int zfcp_reqlist_isempty(struct zfcp_adapter *);
|
||||||
extern void zfcp_sg_free_table(struct scatterlist *, int);
|
extern void zfcp_sg_free_table(struct scatterlist *, int);
|
||||||
extern int zfcp_sg_setup_table(struct scatterlist *, int);
|
extern int zfcp_sg_setup_table(struct scatterlist *, int);
|
||||||
|
extern void zfcp_device_unregister(struct device *,
|
||||||
|
const struct attribute_group *);
|
||||||
|
extern void zfcp_adapter_release(struct kref *);
|
||||||
|
extern void zfcp_adapter_unregister(struct zfcp_adapter *);
|
||||||
|
|
||||||
/* zfcp_ccw.c */
|
/* zfcp_ccw.c */
|
||||||
extern int zfcp_ccw_register(void);
|
|
||||||
extern int zfcp_ccw_priv_sch(struct zfcp_adapter *);
|
extern int zfcp_ccw_priv_sch(struct zfcp_adapter *);
|
||||||
extern struct ccw_driver zfcp_ccw_driver;
|
extern struct ccw_driver zfcp_ccw_driver;
|
||||||
|
extern struct zfcp_adapter *zfcp_ccw_adapter_by_cdev(struct ccw_device *);
|
||||||
|
extern void zfcp_ccw_adapter_put(struct zfcp_adapter *);
|
||||||
|
|
||||||
/* zfcp_cfdc.c */
|
/* zfcp_cfdc.c */
|
||||||
extern struct miscdevice zfcp_cfdc_misc;
|
extern struct miscdevice zfcp_cfdc_misc;
|
||||||
|
@ -51,7 +56,7 @@ extern void _zfcp_dbf_hba_fsf_unsol(const char *, int level, struct zfcp_dbf *,
|
||||||
struct fsf_status_read_buffer *);
|
struct fsf_status_read_buffer *);
|
||||||
extern void zfcp_dbf_hba_qdio(struct zfcp_dbf *, unsigned int, int, int);
|
extern void zfcp_dbf_hba_qdio(struct zfcp_dbf *, unsigned int, int, int);
|
||||||
extern void zfcp_dbf_hba_berr(struct zfcp_dbf *, struct zfcp_fsf_req *);
|
extern void zfcp_dbf_hba_berr(struct zfcp_dbf *, struct zfcp_fsf_req *);
|
||||||
extern void zfcp_dbf_san_ct_request(struct zfcp_fsf_req *);
|
extern void zfcp_dbf_san_ct_request(struct zfcp_fsf_req *, u32);
|
||||||
extern void zfcp_dbf_san_ct_response(struct zfcp_fsf_req *);
|
extern void zfcp_dbf_san_ct_response(struct zfcp_fsf_req *);
|
||||||
extern void zfcp_dbf_san_els_request(struct zfcp_fsf_req *);
|
extern void zfcp_dbf_san_els_request(struct zfcp_fsf_req *);
|
||||||
extern void zfcp_dbf_san_els_response(struct zfcp_fsf_req *);
|
extern void zfcp_dbf_san_els_response(struct zfcp_fsf_req *);
|
||||||
|
@ -92,24 +97,22 @@ extern void zfcp_erp_adapter_access_changed(struct zfcp_adapter *, char *,
|
||||||
extern void zfcp_erp_timeout_handler(unsigned long);
|
extern void zfcp_erp_timeout_handler(unsigned long);
|
||||||
|
|
||||||
/* zfcp_fc.c */
|
/* zfcp_fc.c */
|
||||||
extern int zfcp_fc_scan_ports(struct zfcp_adapter *);
|
extern void zfcp_fc_scan_ports(struct work_struct *);
|
||||||
extern void _zfcp_fc_scan_ports_later(struct work_struct *);
|
|
||||||
extern void zfcp_fc_incoming_els(struct zfcp_fsf_req *);
|
extern void zfcp_fc_incoming_els(struct zfcp_fsf_req *);
|
||||||
extern void zfcp_fc_port_did_lookup(struct work_struct *);
|
extern void zfcp_fc_port_did_lookup(struct work_struct *);
|
||||||
extern void zfcp_fc_trigger_did_lookup(struct zfcp_port *);
|
extern void zfcp_fc_trigger_did_lookup(struct zfcp_port *);
|
||||||
extern void zfcp_fc_plogi_evaluate(struct zfcp_port *, struct fsf_plogi *);
|
extern void zfcp_fc_plogi_evaluate(struct zfcp_port *, struct fc_els_flogi *);
|
||||||
extern void zfcp_fc_test_link(struct zfcp_port *);
|
extern void zfcp_fc_test_link(struct zfcp_port *);
|
||||||
extern void zfcp_fc_link_test_work(struct work_struct *);
|
extern void zfcp_fc_link_test_work(struct work_struct *);
|
||||||
extern void zfcp_fc_wka_ports_force_offline(struct zfcp_wka_ports *);
|
extern void zfcp_fc_wka_ports_force_offline(struct zfcp_fc_wka_ports *);
|
||||||
extern int zfcp_fc_gs_setup(struct zfcp_adapter *);
|
extern int zfcp_fc_gs_setup(struct zfcp_adapter *);
|
||||||
extern void zfcp_fc_gs_destroy(struct zfcp_adapter *);
|
extern void zfcp_fc_gs_destroy(struct zfcp_adapter *);
|
||||||
extern int zfcp_fc_execute_els_fc_job(struct fc_bsg_job *);
|
extern int zfcp_fc_exec_bsg_job(struct fc_bsg_job *);
|
||||||
extern int zfcp_fc_execute_ct_fc_job(struct fc_bsg_job *);
|
|
||||||
|
|
||||||
/* zfcp_fsf.c */
|
/* zfcp_fsf.c */
|
||||||
extern int zfcp_fsf_open_port(struct zfcp_erp_action *);
|
extern int zfcp_fsf_open_port(struct zfcp_erp_action *);
|
||||||
extern int zfcp_fsf_open_wka_port(struct zfcp_wka_port *);
|
extern int zfcp_fsf_open_wka_port(struct zfcp_fc_wka_port *);
|
||||||
extern int zfcp_fsf_close_wka_port(struct zfcp_wka_port *);
|
extern int zfcp_fsf_close_wka_port(struct zfcp_fc_wka_port *);
|
||||||
extern int zfcp_fsf_close_port(struct zfcp_erp_action *);
|
extern int zfcp_fsf_close_port(struct zfcp_erp_action *);
|
||||||
extern int zfcp_fsf_close_physical_port(struct zfcp_erp_action *);
|
extern int zfcp_fsf_close_physical_port(struct zfcp_erp_action *);
|
||||||
extern int zfcp_fsf_open_unit(struct zfcp_erp_action *);
|
extern int zfcp_fsf_open_unit(struct zfcp_erp_action *);
|
||||||
|
@ -125,8 +128,10 @@ extern struct zfcp_fsf_req *zfcp_fsf_control_file(struct zfcp_adapter *,
|
||||||
extern void zfcp_fsf_req_dismiss_all(struct zfcp_adapter *);
|
extern void zfcp_fsf_req_dismiss_all(struct zfcp_adapter *);
|
||||||
extern int zfcp_fsf_status_read(struct zfcp_qdio *);
|
extern int zfcp_fsf_status_read(struct zfcp_qdio *);
|
||||||
extern int zfcp_status_read_refill(struct zfcp_adapter *adapter);
|
extern int zfcp_status_read_refill(struct zfcp_adapter *adapter);
|
||||||
extern int zfcp_fsf_send_ct(struct zfcp_send_ct *, mempool_t *);
|
extern int zfcp_fsf_send_ct(struct zfcp_fc_wka_port *, struct zfcp_fsf_ct_els *,
|
||||||
extern int zfcp_fsf_send_els(struct zfcp_send_els *);
|
mempool_t *);
|
||||||
|
extern int zfcp_fsf_send_els(struct zfcp_adapter *, u32,
|
||||||
|
struct zfcp_fsf_ct_els *);
|
||||||
extern int zfcp_fsf_send_fcp_command_task(struct zfcp_unit *,
|
extern int zfcp_fsf_send_fcp_command_task(struct zfcp_unit *,
|
||||||
struct scsi_cmnd *);
|
struct scsi_cmnd *);
|
||||||
extern void zfcp_fsf_req_free(struct zfcp_fsf_req *);
|
extern void zfcp_fsf_req_free(struct zfcp_fsf_req *);
|
||||||
|
@ -153,7 +158,6 @@ extern void zfcp_qdio_close(struct zfcp_qdio *);
|
||||||
extern struct zfcp_data zfcp_data;
|
extern struct zfcp_data zfcp_data;
|
||||||
extern int zfcp_adapter_scsi_register(struct zfcp_adapter *);
|
extern int zfcp_adapter_scsi_register(struct zfcp_adapter *);
|
||||||
extern void zfcp_adapter_scsi_unregister(struct zfcp_adapter *);
|
extern void zfcp_adapter_scsi_unregister(struct zfcp_adapter *);
|
||||||
extern char *zfcp_get_fcp_sns_info_ptr(struct fcp_rsp_iu *);
|
|
||||||
extern struct fc_function_template zfcp_transport_functions;
|
extern struct fc_function_template zfcp_transport_functions;
|
||||||
extern void zfcp_scsi_rport_work(struct work_struct *);
|
extern void zfcp_scsi_rport_work(struct work_struct *);
|
||||||
extern void zfcp_scsi_schedule_rport_register(struct zfcp_port *);
|
extern void zfcp_scsi_schedule_rport_register(struct zfcp_port *);
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,260 @@
|
||||||
|
/*
|
||||||
|
* zfcp device driver
|
||||||
|
*
|
||||||
|
* Fibre Channel related definitions and inline functions for the zfcp
|
||||||
|
* device driver
|
||||||
|
*
|
||||||
|
* Copyright IBM Corporation 2009
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef ZFCP_FC_H
|
||||||
|
#define ZFCP_FC_H
|
||||||
|
|
||||||
|
#include <scsi/fc/fc_els.h>
|
||||||
|
#include <scsi/fc/fc_fcp.h>
|
||||||
|
#include <scsi/fc/fc_ns.h>
|
||||||
|
#include <scsi/scsi_cmnd.h>
|
||||||
|
#include <scsi/scsi_tcq.h>
|
||||||
|
#include "zfcp_fsf.h"
|
||||||
|
|
||||||
|
#define ZFCP_FC_CT_SIZE_PAGE (PAGE_SIZE - sizeof(struct fc_ct_hdr))
|
||||||
|
#define ZFCP_FC_GPN_FT_ENT_PAGE (ZFCP_FC_CT_SIZE_PAGE \
|
||||||
|
/ sizeof(struct fc_gpn_ft_resp))
|
||||||
|
#define ZFCP_FC_GPN_FT_NUM_BUFS 4 /* memory pages */
|
||||||
|
|
||||||
|
#define ZFCP_FC_GPN_FT_MAX_SIZE (ZFCP_FC_GPN_FT_NUM_BUFS * PAGE_SIZE \
|
||||||
|
- sizeof(struct fc_ct_hdr))
|
||||||
|
#define ZFCP_FC_GPN_FT_MAX_ENT (ZFCP_FC_GPN_FT_NUM_BUFS * \
|
||||||
|
(ZFCP_FC_GPN_FT_ENT_PAGE + 1))
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct zfcp_fc_gid_pn_req - container for ct header plus gid_pn request
|
||||||
|
* @ct_hdr: FC GS common transport header
|
||||||
|
* @gid_pn: GID_PN request
|
||||||
|
*/
|
||||||
|
struct zfcp_fc_gid_pn_req {
|
||||||
|
struct fc_ct_hdr ct_hdr;
|
||||||
|
struct fc_ns_gid_pn gid_pn;
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct zfcp_fc_gid_pn_resp - container for ct header plus gid_pn response
|
||||||
|
* @ct_hdr: FC GS common transport header
|
||||||
|
* @gid_pn: GID_PN response
|
||||||
|
*/
|
||||||
|
struct zfcp_fc_gid_pn_resp {
|
||||||
|
struct fc_ct_hdr ct_hdr;
|
||||||
|
struct fc_gid_pn_resp gid_pn;
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct zfcp_fc_gid_pn - everything required in zfcp for gid_pn request
|
||||||
|
* @ct: data passed to zfcp_fsf for issuing fsf request
|
||||||
|
* @sg_req: scatterlist entry for request data
|
||||||
|
* @sg_resp: scatterlist entry for response data
|
||||||
|
* @gid_pn_req: GID_PN request data
|
||||||
|
* @gid_pn_resp: GID_PN response data
|
||||||
|
*/
|
||||||
|
struct zfcp_fc_gid_pn {
|
||||||
|
struct zfcp_fsf_ct_els ct;
|
||||||
|
struct scatterlist sg_req;
|
||||||
|
struct scatterlist sg_resp;
|
||||||
|
struct zfcp_fc_gid_pn_req gid_pn_req;
|
||||||
|
struct zfcp_fc_gid_pn_resp gid_pn_resp;
|
||||||
|
struct zfcp_port *port;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct zfcp_fc_gpn_ft - container for ct header plus gpn_ft request
|
||||||
|
* @ct_hdr: FC GS common transport header
|
||||||
|
* @gpn_ft: GPN_FT request
|
||||||
|
*/
|
||||||
|
struct zfcp_fc_gpn_ft_req {
|
||||||
|
struct fc_ct_hdr ct_hdr;
|
||||||
|
struct fc_ns_gid_ft gpn_ft;
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct zfcp_fc_gpn_ft_resp - container for ct header plus gpn_ft response
|
||||||
|
* @ct_hdr: FC GS common transport header
|
||||||
|
* @gpn_ft: Array of gpn_ft response data to fill one memory page
|
||||||
|
*/
|
||||||
|
struct zfcp_fc_gpn_ft_resp {
|
||||||
|
struct fc_ct_hdr ct_hdr;
|
||||||
|
struct fc_gpn_ft_resp gpn_ft[ZFCP_FC_GPN_FT_ENT_PAGE];
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct zfcp_fc_gpn_ft - zfcp data for gpn_ft request
|
||||||
|
* @ct: data passed to zfcp_fsf for issuing fsf request
|
||||||
|
* @sg_req: scatter list entry for gpn_ft request
|
||||||
|
* @sg_resp: scatter list entries for gpn_ft responses (per memory page)
|
||||||
|
*/
|
||||||
|
struct zfcp_fc_gpn_ft {
|
||||||
|
struct zfcp_fsf_ct_els ct;
|
||||||
|
struct scatterlist sg_req;
|
||||||
|
struct scatterlist sg_resp[ZFCP_FC_GPN_FT_NUM_BUFS];
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct zfcp_fc_els_adisc - everything required in zfcp for issuing ELS ADISC
|
||||||
|
* @els: data required for issuing els fsf command
|
||||||
|
* @req: scatterlist entry for ELS ADISC request
|
||||||
|
* @resp: scatterlist entry for ELS ADISC response
|
||||||
|
* @adisc_req: ELS ADISC request data
|
||||||
|
* @adisc_resp: ELS ADISC response data
|
||||||
|
*/
|
||||||
|
struct zfcp_fc_els_adisc {
|
||||||
|
struct zfcp_fsf_ct_els els;
|
||||||
|
struct scatterlist req;
|
||||||
|
struct scatterlist resp;
|
||||||
|
struct fc_els_adisc adisc_req;
|
||||||
|
struct fc_els_adisc adisc_resp;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* enum zfcp_fc_wka_status - FC WKA port status in zfcp
|
||||||
|
* @ZFCP_FC_WKA_PORT_OFFLINE: Port is closed and not in use
|
||||||
|
* @ZFCP_FC_WKA_PORT_CLOSING: The FSF "close port" request is pending
|
||||||
|
* @ZFCP_FC_WKA_PORT_OPENING: The FSF "open port" request is pending
|
||||||
|
* @ZFCP_FC_WKA_PORT_ONLINE: The port is open and the port handle is valid
|
||||||
|
*/
|
||||||
|
enum zfcp_fc_wka_status {
|
||||||
|
ZFCP_FC_WKA_PORT_OFFLINE,
|
||||||
|
ZFCP_FC_WKA_PORT_CLOSING,
|
||||||
|
ZFCP_FC_WKA_PORT_OPENING,
|
||||||
|
ZFCP_FC_WKA_PORT_ONLINE,
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct zfcp_fc_wka_port - representation of well-known-address (WKA) FC port
|
||||||
|
* @adapter: Pointer to adapter structure this WKA port belongs to
|
||||||
|
* @completion_wq: Wait for completion of open/close command
|
||||||
|
* @status: Current status of WKA port
|
||||||
|
* @refcount: Reference count to keep port open as long as it is in use
|
||||||
|
* @d_id: FC destination id or well-known-address
|
||||||
|
* @handle: FSF handle for the open WKA port
|
||||||
|
* @mutex: Mutex used during opening/closing state changes
|
||||||
|
* @work: For delaying the closing of the WKA port
|
||||||
|
*/
|
||||||
|
struct zfcp_fc_wka_port {
|
||||||
|
struct zfcp_adapter *adapter;
|
||||||
|
wait_queue_head_t completion_wq;
|
||||||
|
enum zfcp_fc_wka_status status;
|
||||||
|
atomic_t refcount;
|
||||||
|
u32 d_id;
|
||||||
|
u32 handle;
|
||||||
|
struct mutex mutex;
|
||||||
|
struct delayed_work work;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct zfcp_fc_wka_ports - Data structures for FC generic services
|
||||||
|
* @ms: FC Management service
|
||||||
|
* @ts: FC time service
|
||||||
|
* @ds: FC directory service
|
||||||
|
* @as: FC alias service
|
||||||
|
*/
|
||||||
|
struct zfcp_fc_wka_ports {
|
||||||
|
struct zfcp_fc_wka_port ms;
|
||||||
|
struct zfcp_fc_wka_port ts;
|
||||||
|
struct zfcp_fc_wka_port ds;
|
||||||
|
struct zfcp_fc_wka_port as;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* zfcp_fc_scsi_to_fcp - setup FCP command with data from scsi_cmnd
|
||||||
|
* @fcp: fcp_cmnd to setup
|
||||||
|
* @scsi: scsi_cmnd where to get LUN, task attributes/flags and CDB
|
||||||
|
*/
|
||||||
|
static inline
|
||||||
|
void zfcp_fc_scsi_to_fcp(struct fcp_cmnd *fcp, struct scsi_cmnd *scsi)
|
||||||
|
{
|
||||||
|
char tag[2];
|
||||||
|
|
||||||
|
int_to_scsilun(scsi->device->lun, (struct scsi_lun *) &fcp->fc_lun);
|
||||||
|
|
||||||
|
if (scsi_populate_tag_msg(scsi, tag)) {
|
||||||
|
switch (tag[0]) {
|
||||||
|
case MSG_ORDERED_TAG:
|
||||||
|
fcp->fc_pri_ta |= FCP_PTA_ORDERED;
|
||||||
|
break;
|
||||||
|
case MSG_SIMPLE_TAG:
|
||||||
|
fcp->fc_pri_ta |= FCP_PTA_SIMPLE;
|
||||||
|
break;
|
||||||
|
};
|
||||||
|
} else
|
||||||
|
fcp->fc_pri_ta = FCP_PTA_SIMPLE;
|
||||||
|
|
||||||
|
if (scsi->sc_data_direction == DMA_FROM_DEVICE)
|
||||||
|
fcp->fc_flags |= FCP_CFL_RDDATA;
|
||||||
|
if (scsi->sc_data_direction == DMA_TO_DEVICE)
|
||||||
|
fcp->fc_flags |= FCP_CFL_WRDATA;
|
||||||
|
|
||||||
|
memcpy(fcp->fc_cdb, scsi->cmnd, scsi->cmd_len);
|
||||||
|
|
||||||
|
fcp->fc_dl = scsi_bufflen(scsi);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* zfcp_fc_fcp_tm - setup FCP command as task management command
|
||||||
|
* @fcp: fcp_cmnd to setup
|
||||||
|
* @dev: scsi_device where to send the task management command
|
||||||
|
* @tm: task management flags to setup tm command
|
||||||
|
*/
|
||||||
|
static inline
|
||||||
|
void zfcp_fc_fcp_tm(struct fcp_cmnd *fcp, struct scsi_device *dev, u8 tm_flags)
|
||||||
|
{
|
||||||
|
int_to_scsilun(dev->lun, (struct scsi_lun *) &fcp->fc_lun);
|
||||||
|
fcp->fc_tm_flags |= tm_flags;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* zfcp_fc_evap_fcp_rsp - evaluate FCP RSP IU and update scsi_cmnd accordingly
|
||||||
|
* @fcp_rsp: FCP RSP IU to evaluate
|
||||||
|
* @scsi: SCSI command where to update status and sense buffer
|
||||||
|
*/
|
||||||
|
static inline
|
||||||
|
void zfcp_fc_eval_fcp_rsp(struct fcp_resp_with_ext *fcp_rsp,
|
||||||
|
struct scsi_cmnd *scsi)
|
||||||
|
{
|
||||||
|
struct fcp_resp_rsp_info *rsp_info;
|
||||||
|
char *sense;
|
||||||
|
u32 sense_len, resid;
|
||||||
|
u8 rsp_flags;
|
||||||
|
|
||||||
|
set_msg_byte(scsi, COMMAND_COMPLETE);
|
||||||
|
scsi->result |= fcp_rsp->resp.fr_status;
|
||||||
|
|
||||||
|
rsp_flags = fcp_rsp->resp.fr_flags;
|
||||||
|
|
||||||
|
if (unlikely(rsp_flags & FCP_RSP_LEN_VAL)) {
|
||||||
|
rsp_info = (struct fcp_resp_rsp_info *) &fcp_rsp[1];
|
||||||
|
if (rsp_info->rsp_code == FCP_TMF_CMPL)
|
||||||
|
set_host_byte(scsi, DID_OK);
|
||||||
|
else {
|
||||||
|
set_host_byte(scsi, DID_ERROR);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (unlikely(rsp_flags & FCP_SNS_LEN_VAL)) {
|
||||||
|
sense = (char *) &fcp_rsp[1];
|
||||||
|
if (rsp_flags & FCP_RSP_LEN_VAL)
|
||||||
|
sense += fcp_rsp->ext.fr_sns_len;
|
||||||
|
sense_len = min(fcp_rsp->ext.fr_sns_len,
|
||||||
|
(u32) SCSI_SENSE_BUFFERSIZE);
|
||||||
|
memcpy(scsi->sense_buffer, sense, sense_len);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (unlikely(rsp_flags & FCP_RESID_UNDER)) {
|
||||||
|
resid = fcp_rsp->ext.fr_resid;
|
||||||
|
scsi_set_resid(scsi, resid);
|
||||||
|
if (scsi_bufflen(scsi) - resid < scsi->underflow &&
|
||||||
|
!(rsp_flags & FCP_SNS_LEN_VAL) &&
|
||||||
|
fcp_rsp->resp.fr_status == SAM_STAT_GOOD)
|
||||||
|
set_host_byte(scsi, DID_ERROR);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -10,7 +10,9 @@
|
||||||
#define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
|
#define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
|
||||||
|
|
||||||
#include <linux/blktrace_api.h>
|
#include <linux/blktrace_api.h>
|
||||||
|
#include <scsi/fc/fc_els.h>
|
||||||
#include "zfcp_ext.h"
|
#include "zfcp_ext.h"
|
||||||
|
#include "zfcp_fc.h"
|
||||||
#include "zfcp_dbf.h"
|
#include "zfcp_dbf.h"
|
||||||
|
|
||||||
static void zfcp_fsf_request_timeout_handler(unsigned long data)
|
static void zfcp_fsf_request_timeout_handler(unsigned long data)
|
||||||
|
@ -122,36 +124,32 @@ void zfcp_fsf_req_free(struct zfcp_fsf_req *req)
|
||||||
|
|
||||||
static void zfcp_fsf_status_read_port_closed(struct zfcp_fsf_req *req)
|
static void zfcp_fsf_status_read_port_closed(struct zfcp_fsf_req *req)
|
||||||
{
|
{
|
||||||
|
unsigned long flags;
|
||||||
struct fsf_status_read_buffer *sr_buf = req->data;
|
struct fsf_status_read_buffer *sr_buf = req->data;
|
||||||
struct zfcp_adapter *adapter = req->adapter;
|
struct zfcp_adapter *adapter = req->adapter;
|
||||||
struct zfcp_port *port;
|
struct zfcp_port *port;
|
||||||
int d_id = sr_buf->d_id & ZFCP_DID_MASK;
|
int d_id = ntoh24(sr_buf->d_id);
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
read_lock_irqsave(&adapter->port_list_lock, flags);
|
||||||
list_for_each_entry(port, &adapter->port_list_head, list)
|
list_for_each_entry(port, &adapter->port_list, list)
|
||||||
if (port->d_id == d_id) {
|
if (port->d_id == d_id) {
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
zfcp_erp_port_reopen(port, 0, "fssrpc1", req);
|
zfcp_erp_port_reopen(port, 0, "fssrpc1", req);
|
||||||
return;
|
break;
|
||||||
}
|
}
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
read_unlock_irqrestore(&adapter->port_list_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void zfcp_fsf_link_down_info_eval(struct zfcp_fsf_req *req, char *id,
|
static void zfcp_fsf_link_down_info_eval(struct zfcp_fsf_req *req, char *id,
|
||||||
struct fsf_link_down_info *link_down)
|
struct fsf_link_down_info *link_down)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter = req->adapter;
|
struct zfcp_adapter *adapter = req->adapter;
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
if (atomic_read(&adapter->status) & ZFCP_STATUS_ADAPTER_LINK_UNPLUGGED)
|
if (atomic_read(&adapter->status) & ZFCP_STATUS_ADAPTER_LINK_UNPLUGGED)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
atomic_set_mask(ZFCP_STATUS_ADAPTER_LINK_UNPLUGGED, &adapter->status);
|
atomic_set_mask(ZFCP_STATUS_ADAPTER_LINK_UNPLUGGED, &adapter->status);
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
|
||||||
zfcp_scsi_schedule_rports_block(adapter);
|
zfcp_scsi_schedule_rports_block(adapter);
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
|
|
||||||
if (!link_down)
|
if (!link_down)
|
||||||
goto out;
|
goto out;
|
||||||
|
@ -291,7 +289,7 @@ static void zfcp_fsf_status_read_handler(struct zfcp_fsf_req *req)
|
||||||
zfcp_erp_adapter_access_changed(adapter, "fssrh_3",
|
zfcp_erp_adapter_access_changed(adapter, "fssrh_3",
|
||||||
req);
|
req);
|
||||||
if (sr_buf->status_subtype & FSF_STATUS_READ_SUB_INCOMING_ELS)
|
if (sr_buf->status_subtype & FSF_STATUS_READ_SUB_INCOMING_ELS)
|
||||||
schedule_work(&adapter->scan_work);
|
queue_work(adapter->work_queue, &adapter->scan_work);
|
||||||
break;
|
break;
|
||||||
case FSF_STATUS_READ_CFDC_UPDATED:
|
case FSF_STATUS_READ_CFDC_UPDATED:
|
||||||
zfcp_erp_adapter_access_changed(adapter, "fssrh_4", req);
|
zfcp_erp_adapter_access_changed(adapter, "fssrh_4", req);
|
||||||
|
@ -317,7 +315,6 @@ static void zfcp_fsf_fsfstatus_qual_eval(struct zfcp_fsf_req *req)
|
||||||
case FSF_SQ_ULP_DEPENDENT_ERP_REQUIRED:
|
case FSF_SQ_ULP_DEPENDENT_ERP_REQUIRED:
|
||||||
return;
|
return;
|
||||||
case FSF_SQ_COMMAND_ABORTED:
|
case FSF_SQ_COMMAND_ABORTED:
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ABORTED;
|
|
||||||
break;
|
break;
|
||||||
case FSF_SQ_NO_RECOM:
|
case FSF_SQ_NO_RECOM:
|
||||||
dev_err(&req->adapter->ccw_device->dev,
|
dev_err(&req->adapter->ccw_device->dev,
|
||||||
|
@ -358,8 +355,7 @@ static void zfcp_fsf_protstatus_eval(struct zfcp_fsf_req *req)
|
||||||
zfcp_dbf_hba_fsf_response(req);
|
zfcp_dbf_hba_fsf_response(req);
|
||||||
|
|
||||||
if (req->status & ZFCP_STATUS_FSFREQ_DISMISSED) {
|
if (req->status & ZFCP_STATUS_FSFREQ_DISMISSED) {
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR |
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
ZFCP_STATUS_FSFREQ_RETRY; /* only for SCSI cmnds. */
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -377,7 +373,7 @@ static void zfcp_fsf_protstatus_eval(struct zfcp_fsf_req *req)
|
||||||
case FSF_PROT_ERROR_STATE:
|
case FSF_PROT_ERROR_STATE:
|
||||||
case FSF_PROT_SEQ_NUMB_ERROR:
|
case FSF_PROT_SEQ_NUMB_ERROR:
|
||||||
zfcp_erp_adapter_reopen(adapter, 0, "fspse_2", req);
|
zfcp_erp_adapter_reopen(adapter, 0, "fspse_2", req);
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_RETRY;
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
break;
|
break;
|
||||||
case FSF_PROT_UNSUPP_QTCB_TYPE:
|
case FSF_PROT_UNSUPP_QTCB_TYPE:
|
||||||
dev_err(&adapter->ccw_device->dev,
|
dev_err(&adapter->ccw_device->dev,
|
||||||
|
@ -480,20 +476,27 @@ void zfcp_fsf_req_dismiss_all(struct zfcp_adapter *adapter)
|
||||||
|
|
||||||
static int zfcp_fsf_exchange_config_evaluate(struct zfcp_fsf_req *req)
|
static int zfcp_fsf_exchange_config_evaluate(struct zfcp_fsf_req *req)
|
||||||
{
|
{
|
||||||
struct fsf_qtcb_bottom_config *bottom;
|
struct fsf_qtcb_bottom_config *bottom = &req->qtcb->bottom.config;
|
||||||
struct zfcp_adapter *adapter = req->adapter;
|
struct zfcp_adapter *adapter = req->adapter;
|
||||||
struct Scsi_Host *shost = adapter->scsi_host;
|
struct Scsi_Host *shost = adapter->scsi_host;
|
||||||
|
struct fc_els_flogi *nsp, *plogi;
|
||||||
|
|
||||||
bottom = &req->qtcb->bottom.config;
|
/* adjust pointers for missing command code */
|
||||||
|
nsp = (struct fc_els_flogi *) ((u8 *)&bottom->nport_serv_param
|
||||||
|
- sizeof(u32));
|
||||||
|
plogi = (struct fc_els_flogi *) ((u8 *)&bottom->plogi_payload
|
||||||
|
- sizeof(u32));
|
||||||
|
|
||||||
if (req->data)
|
if (req->data)
|
||||||
memcpy(req->data, bottom, sizeof(*bottom));
|
memcpy(req->data, bottom, sizeof(*bottom));
|
||||||
|
|
||||||
fc_host_node_name(shost) = bottom->nport_serv_param.wwnn;
|
fc_host_port_name(shost) = nsp->fl_wwpn;
|
||||||
fc_host_port_name(shost) = bottom->nport_serv_param.wwpn;
|
fc_host_node_name(shost) = nsp->fl_wwnn;
|
||||||
fc_host_port_id(shost) = bottom->s_id & ZFCP_DID_MASK;
|
fc_host_port_id(shost) = ntoh24(bottom->s_id);
|
||||||
fc_host_speed(shost) = bottom->fc_link_speed;
|
fc_host_speed(shost) = bottom->fc_link_speed;
|
||||||
fc_host_supported_classes(shost) = FC_COS_CLASS2 | FC_COS_CLASS3;
|
fc_host_supported_classes(shost) = FC_COS_CLASS2 | FC_COS_CLASS3;
|
||||||
|
fc_host_supported_fc4s(shost)[2] = 1; /* FCP */
|
||||||
|
fc_host_active_fc4s(shost)[2] = 1; /* FCP */
|
||||||
|
|
||||||
adapter->hydra_version = bottom->adapter_type;
|
adapter->hydra_version = bottom->adapter_type;
|
||||||
adapter->timer_ticks = bottom->timer_interval;
|
adapter->timer_ticks = bottom->timer_interval;
|
||||||
|
@ -503,9 +506,9 @@ static int zfcp_fsf_exchange_config_evaluate(struct zfcp_fsf_req *req)
|
||||||
|
|
||||||
switch (bottom->fc_topology) {
|
switch (bottom->fc_topology) {
|
||||||
case FSF_TOPO_P2P:
|
case FSF_TOPO_P2P:
|
||||||
adapter->peer_d_id = bottom->peer_d_id & ZFCP_DID_MASK;
|
adapter->peer_d_id = ntoh24(bottom->peer_d_id);
|
||||||
adapter->peer_wwpn = bottom->plogi_payload.wwpn;
|
adapter->peer_wwpn = plogi->fl_wwpn;
|
||||||
adapter->peer_wwnn = bottom->plogi_payload.wwnn;
|
adapter->peer_wwnn = plogi->fl_wwnn;
|
||||||
fc_host_port_type(shost) = FC_PORTTYPE_PTP;
|
fc_host_port_type(shost) = FC_PORTTYPE_PTP;
|
||||||
break;
|
break;
|
||||||
case FSF_TOPO_FABRIC:
|
case FSF_TOPO_FABRIC:
|
||||||
|
@ -881,13 +884,11 @@ static void zfcp_fsf_abort_fcp_command_handler(struct zfcp_fsf_req *req)
|
||||||
break;
|
break;
|
||||||
case FSF_PORT_BOXED:
|
case FSF_PORT_BOXED:
|
||||||
zfcp_erp_port_boxed(unit->port, "fsafch3", req);
|
zfcp_erp_port_boxed(unit->port, "fsafch3", req);
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR |
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
ZFCP_STATUS_FSFREQ_RETRY;
|
|
||||||
break;
|
break;
|
||||||
case FSF_LUN_BOXED:
|
case FSF_LUN_BOXED:
|
||||||
zfcp_erp_unit_boxed(unit, "fsafch4", req);
|
zfcp_erp_unit_boxed(unit, "fsafch4", req);
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR |
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
ZFCP_STATUS_FSFREQ_RETRY;
|
|
||||||
break;
|
break;
|
||||||
case FSF_ADAPTER_STATUS_AVAILABLE:
|
case FSF_ADAPTER_STATUS_AVAILABLE:
|
||||||
switch (fsq->word[0]) {
|
switch (fsq->word[0]) {
|
||||||
|
@ -958,10 +959,10 @@ out:
|
||||||
static void zfcp_fsf_send_ct_handler(struct zfcp_fsf_req *req)
|
static void zfcp_fsf_send_ct_handler(struct zfcp_fsf_req *req)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter = req->adapter;
|
struct zfcp_adapter *adapter = req->adapter;
|
||||||
struct zfcp_send_ct *send_ct = req->data;
|
struct zfcp_fsf_ct_els *ct = req->data;
|
||||||
struct fsf_qtcb_header *header = &req->qtcb->header;
|
struct fsf_qtcb_header *header = &req->qtcb->header;
|
||||||
|
|
||||||
send_ct->status = -EINVAL;
|
ct->status = -EINVAL;
|
||||||
|
|
||||||
if (req->status & ZFCP_STATUS_FSFREQ_ERROR)
|
if (req->status & ZFCP_STATUS_FSFREQ_ERROR)
|
||||||
goto skip_fsfstatus;
|
goto skip_fsfstatus;
|
||||||
|
@ -969,7 +970,7 @@ static void zfcp_fsf_send_ct_handler(struct zfcp_fsf_req *req)
|
||||||
switch (header->fsf_status) {
|
switch (header->fsf_status) {
|
||||||
case FSF_GOOD:
|
case FSF_GOOD:
|
||||||
zfcp_dbf_san_ct_response(req);
|
zfcp_dbf_san_ct_response(req);
|
||||||
send_ct->status = 0;
|
ct->status = 0;
|
||||||
break;
|
break;
|
||||||
case FSF_SERVICE_CLASS_NOT_SUPPORTED:
|
case FSF_SERVICE_CLASS_NOT_SUPPORTED:
|
||||||
zfcp_fsf_class_not_supp(req);
|
zfcp_fsf_class_not_supp(req);
|
||||||
|
@ -985,8 +986,7 @@ static void zfcp_fsf_send_ct_handler(struct zfcp_fsf_req *req)
|
||||||
case FSF_ACCESS_DENIED:
|
case FSF_ACCESS_DENIED:
|
||||||
break;
|
break;
|
||||||
case FSF_PORT_BOXED:
|
case FSF_PORT_BOXED:
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR |
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
ZFCP_STATUS_FSFREQ_RETRY;
|
|
||||||
break;
|
break;
|
||||||
case FSF_PORT_HANDLE_NOT_VALID:
|
case FSF_PORT_HANDLE_NOT_VALID:
|
||||||
zfcp_erp_adapter_reopen(adapter, 0, "fsscth1", req);
|
zfcp_erp_adapter_reopen(adapter, 0, "fsscth1", req);
|
||||||
|
@ -1001,8 +1001,8 @@ static void zfcp_fsf_send_ct_handler(struct zfcp_fsf_req *req)
|
||||||
}
|
}
|
||||||
|
|
||||||
skip_fsfstatus:
|
skip_fsfstatus:
|
||||||
if (send_ct->handler)
|
if (ct->handler)
|
||||||
send_ct->handler(send_ct->handler_data);
|
ct->handler(ct->handler_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void zfcp_fsf_setup_ct_els_unchained(struct qdio_buffer_element *sbale,
|
static void zfcp_fsf_setup_ct_els_unchained(struct qdio_buffer_element *sbale,
|
||||||
|
@ -1071,15 +1071,17 @@ static int zfcp_fsf_setup_ct_els(struct zfcp_fsf_req *req,
|
||||||
int max_sbals)
|
int max_sbals)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
unsigned int fcp_chan_timeout;
|
||||||
|
|
||||||
ret = zfcp_fsf_setup_ct_els_sbals(req, sg_req, sg_resp, max_sbals);
|
ret = zfcp_fsf_setup_ct_els_sbals(req, sg_req, sg_resp, max_sbals);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
/* common settings for ct/gs and els requests */
|
/* common settings for ct/gs and els requests */
|
||||||
|
fcp_chan_timeout = 2 * FC_DEF_R_A_TOV / 1000;
|
||||||
req->qtcb->bottom.support.service_class = FSF_CLASS_3;
|
req->qtcb->bottom.support.service_class = FSF_CLASS_3;
|
||||||
req->qtcb->bottom.support.timeout = 2 * R_A_TOV;
|
req->qtcb->bottom.support.timeout = fcp_chan_timeout;
|
||||||
zfcp_fsf_start_timer(req, (2 * R_A_TOV + 10) * HZ);
|
zfcp_fsf_start_timer(req, (fcp_chan_timeout + 10) * HZ);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -1089,9 +1091,9 @@ static int zfcp_fsf_setup_ct_els(struct zfcp_fsf_req *req,
|
||||||
* @ct: pointer to struct zfcp_send_ct with data for request
|
* @ct: pointer to struct zfcp_send_ct with data for request
|
||||||
* @pool: if non-null this mempool is used to allocate struct zfcp_fsf_req
|
* @pool: if non-null this mempool is used to allocate struct zfcp_fsf_req
|
||||||
*/
|
*/
|
||||||
int zfcp_fsf_send_ct(struct zfcp_send_ct *ct, mempool_t *pool)
|
int zfcp_fsf_send_ct(struct zfcp_fc_wka_port *wka_port,
|
||||||
|
struct zfcp_fsf_ct_els *ct, mempool_t *pool)
|
||||||
{
|
{
|
||||||
struct zfcp_wka_port *wka_port = ct->wka_port;
|
|
||||||
struct zfcp_qdio *qdio = wka_port->adapter->qdio;
|
struct zfcp_qdio *qdio = wka_port->adapter->qdio;
|
||||||
struct zfcp_fsf_req *req;
|
struct zfcp_fsf_req *req;
|
||||||
int ret = -EIO;
|
int ret = -EIO;
|
||||||
|
@ -1117,7 +1119,7 @@ int zfcp_fsf_send_ct(struct zfcp_send_ct *ct, mempool_t *pool)
|
||||||
req->qtcb->header.port_handle = wka_port->handle;
|
req->qtcb->header.port_handle = wka_port->handle;
|
||||||
req->data = ct;
|
req->data = ct;
|
||||||
|
|
||||||
zfcp_dbf_san_ct_request(req);
|
zfcp_dbf_san_ct_request(req, wka_port->d_id);
|
||||||
|
|
||||||
ret = zfcp_fsf_req_send(req);
|
ret = zfcp_fsf_req_send(req);
|
||||||
if (ret)
|
if (ret)
|
||||||
|
@ -1134,7 +1136,7 @@ out:
|
||||||
|
|
||||||
static void zfcp_fsf_send_els_handler(struct zfcp_fsf_req *req)
|
static void zfcp_fsf_send_els_handler(struct zfcp_fsf_req *req)
|
||||||
{
|
{
|
||||||
struct zfcp_send_els *send_els = req->data;
|
struct zfcp_fsf_ct_els *send_els = req->data;
|
||||||
struct zfcp_port *port = send_els->port;
|
struct zfcp_port *port = send_els->port;
|
||||||
struct fsf_qtcb_header *header = &req->qtcb->header;
|
struct fsf_qtcb_header *header = &req->qtcb->header;
|
||||||
|
|
||||||
|
@ -1154,9 +1156,6 @@ static void zfcp_fsf_send_els_handler(struct zfcp_fsf_req *req)
|
||||||
case FSF_ADAPTER_STATUS_AVAILABLE:
|
case FSF_ADAPTER_STATUS_AVAILABLE:
|
||||||
switch (header->fsf_status_qual.word[0]){
|
switch (header->fsf_status_qual.word[0]){
|
||||||
case FSF_SQ_INVOKE_LINK_TEST_PROCEDURE:
|
case FSF_SQ_INVOKE_LINK_TEST_PROCEDURE:
|
||||||
if (port && (send_els->ls_code != ZFCP_LS_ADISC))
|
|
||||||
zfcp_fc_test_link(port);
|
|
||||||
/*fall through */
|
|
||||||
case FSF_SQ_ULP_DEPENDENT_ERP_REQUIRED:
|
case FSF_SQ_ULP_DEPENDENT_ERP_REQUIRED:
|
||||||
case FSF_SQ_RETRY_IF_POSSIBLE:
|
case FSF_SQ_RETRY_IF_POSSIBLE:
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
|
@ -1188,10 +1187,11 @@ skip_fsfstatus:
|
||||||
* zfcp_fsf_send_els - initiate an ELS command (FC-FS)
|
* zfcp_fsf_send_els - initiate an ELS command (FC-FS)
|
||||||
* @els: pointer to struct zfcp_send_els with data for the command
|
* @els: pointer to struct zfcp_send_els with data for the command
|
||||||
*/
|
*/
|
||||||
int zfcp_fsf_send_els(struct zfcp_send_els *els)
|
int zfcp_fsf_send_els(struct zfcp_adapter *adapter, u32 d_id,
|
||||||
|
struct zfcp_fsf_ct_els *els)
|
||||||
{
|
{
|
||||||
struct zfcp_fsf_req *req;
|
struct zfcp_fsf_req *req;
|
||||||
struct zfcp_qdio *qdio = els->adapter->qdio;
|
struct zfcp_qdio *qdio = adapter->qdio;
|
||||||
int ret = -EIO;
|
int ret = -EIO;
|
||||||
|
|
||||||
spin_lock_bh(&qdio->req_q_lock);
|
spin_lock_bh(&qdio->req_q_lock);
|
||||||
|
@ -1211,7 +1211,7 @@ int zfcp_fsf_send_els(struct zfcp_send_els *els)
|
||||||
if (ret)
|
if (ret)
|
||||||
goto failed_send;
|
goto failed_send;
|
||||||
|
|
||||||
req->qtcb->bottom.support.d_id = els->d_id;
|
hton24(req->qtcb->bottom.support.d_id, d_id);
|
||||||
req->handler = zfcp_fsf_send_els_handler;
|
req->handler = zfcp_fsf_send_els_handler;
|
||||||
req->data = els;
|
req->data = els;
|
||||||
|
|
||||||
|
@ -1422,7 +1422,7 @@ static void zfcp_fsf_open_port_handler(struct zfcp_fsf_req *req)
|
||||||
{
|
{
|
||||||
struct zfcp_port *port = req->data;
|
struct zfcp_port *port = req->data;
|
||||||
struct fsf_qtcb_header *header = &req->qtcb->header;
|
struct fsf_qtcb_header *header = &req->qtcb->header;
|
||||||
struct fsf_plogi *plogi;
|
struct fc_els_flogi *plogi;
|
||||||
|
|
||||||
if (req->status & ZFCP_STATUS_FSFREQ_ERROR)
|
if (req->status & ZFCP_STATUS_FSFREQ_ERROR)
|
||||||
goto out;
|
goto out;
|
||||||
|
@ -1472,23 +1472,10 @@ static void zfcp_fsf_open_port_handler(struct zfcp_fsf_req *req)
|
||||||
* another GID_PN straight after a port has been opened.
|
* another GID_PN straight after a port has been opened.
|
||||||
* Alternately, an ADISC/PDISC ELS should suffice, as well.
|
* Alternately, an ADISC/PDISC ELS should suffice, as well.
|
||||||
*/
|
*/
|
||||||
plogi = (struct fsf_plogi *) req->qtcb->bottom.support.els;
|
plogi = (struct fc_els_flogi *) req->qtcb->bottom.support.els;
|
||||||
if (req->qtcb->bottom.support.els1_length >=
|
if (req->qtcb->bottom.support.els1_length >=
|
||||||
FSF_PLOGI_MIN_LEN) {
|
FSF_PLOGI_MIN_LEN)
|
||||||
if (plogi->serv_param.wwpn != port->wwpn) {
|
|
||||||
port->d_id = 0;
|
|
||||||
dev_warn(&port->adapter->ccw_device->dev,
|
|
||||||
"A port opened with WWPN 0x%016Lx "
|
|
||||||
"returned data that identifies it as "
|
|
||||||
"WWPN 0x%016Lx\n",
|
|
||||||
(unsigned long long) port->wwpn,
|
|
||||||
(unsigned long long)
|
|
||||||
plogi->serv_param.wwpn);
|
|
||||||
} else {
|
|
||||||
port->wwnn = plogi->serv_param.wwnn;
|
|
||||||
zfcp_fc_plogi_evaluate(port, plogi);
|
zfcp_fc_plogi_evaluate(port, plogi);
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case FSF_UNKNOWN_OP_SUBTYPE:
|
case FSF_UNKNOWN_OP_SUBTYPE:
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
|
@ -1496,7 +1483,7 @@ static void zfcp_fsf_open_port_handler(struct zfcp_fsf_req *req)
|
||||||
}
|
}
|
||||||
|
|
||||||
out:
|
out:
|
||||||
zfcp_port_put(port);
|
put_device(&port->sysfs_device);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1530,18 +1517,18 @@ int zfcp_fsf_open_port(struct zfcp_erp_action *erp_action)
|
||||||
sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY;
|
sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY;
|
||||||
|
|
||||||
req->handler = zfcp_fsf_open_port_handler;
|
req->handler = zfcp_fsf_open_port_handler;
|
||||||
req->qtcb->bottom.support.d_id = port->d_id;
|
hton24(req->qtcb->bottom.support.d_id, port->d_id);
|
||||||
req->data = port;
|
req->data = port;
|
||||||
req->erp_action = erp_action;
|
req->erp_action = erp_action;
|
||||||
erp_action->fsf_req = req;
|
erp_action->fsf_req = req;
|
||||||
zfcp_port_get(port);
|
get_device(&port->sysfs_device);
|
||||||
|
|
||||||
zfcp_fsf_start_erp_timer(req);
|
zfcp_fsf_start_erp_timer(req);
|
||||||
retval = zfcp_fsf_req_send(req);
|
retval = zfcp_fsf_req_send(req);
|
||||||
if (retval) {
|
if (retval) {
|
||||||
zfcp_fsf_req_free(req);
|
zfcp_fsf_req_free(req);
|
||||||
erp_action->fsf_req = NULL;
|
erp_action->fsf_req = NULL;
|
||||||
zfcp_port_put(port);
|
put_device(&port->sysfs_device);
|
||||||
}
|
}
|
||||||
out:
|
out:
|
||||||
spin_unlock_bh(&qdio->req_q_lock);
|
spin_unlock_bh(&qdio->req_q_lock);
|
||||||
|
@ -1618,11 +1605,11 @@ out:
|
||||||
|
|
||||||
static void zfcp_fsf_open_wka_port_handler(struct zfcp_fsf_req *req)
|
static void zfcp_fsf_open_wka_port_handler(struct zfcp_fsf_req *req)
|
||||||
{
|
{
|
||||||
struct zfcp_wka_port *wka_port = req->data;
|
struct zfcp_fc_wka_port *wka_port = req->data;
|
||||||
struct fsf_qtcb_header *header = &req->qtcb->header;
|
struct fsf_qtcb_header *header = &req->qtcb->header;
|
||||||
|
|
||||||
if (req->status & ZFCP_STATUS_FSFREQ_ERROR) {
|
if (req->status & ZFCP_STATUS_FSFREQ_ERROR) {
|
||||||
wka_port->status = ZFCP_WKA_PORT_OFFLINE;
|
wka_port->status = ZFCP_FC_WKA_PORT_OFFLINE;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1635,13 +1622,13 @@ static void zfcp_fsf_open_wka_port_handler(struct zfcp_fsf_req *req)
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
/* fall through */
|
/* fall through */
|
||||||
case FSF_ACCESS_DENIED:
|
case FSF_ACCESS_DENIED:
|
||||||
wka_port->status = ZFCP_WKA_PORT_OFFLINE;
|
wka_port->status = ZFCP_FC_WKA_PORT_OFFLINE;
|
||||||
break;
|
break;
|
||||||
case FSF_GOOD:
|
case FSF_GOOD:
|
||||||
wka_port->handle = header->port_handle;
|
wka_port->handle = header->port_handle;
|
||||||
/* fall through */
|
/* fall through */
|
||||||
case FSF_PORT_ALREADY_OPEN:
|
case FSF_PORT_ALREADY_OPEN:
|
||||||
wka_port->status = ZFCP_WKA_PORT_ONLINE;
|
wka_port->status = ZFCP_FC_WKA_PORT_ONLINE;
|
||||||
}
|
}
|
||||||
out:
|
out:
|
||||||
wake_up(&wka_port->completion_wq);
|
wake_up(&wka_port->completion_wq);
|
||||||
|
@ -1649,10 +1636,10 @@ out:
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* zfcp_fsf_open_wka_port - create and send open wka-port request
|
* zfcp_fsf_open_wka_port - create and send open wka-port request
|
||||||
* @wka_port: pointer to struct zfcp_wka_port
|
* @wka_port: pointer to struct zfcp_fc_wka_port
|
||||||
* Returns: 0 on success, error otherwise
|
* Returns: 0 on success, error otherwise
|
||||||
*/
|
*/
|
||||||
int zfcp_fsf_open_wka_port(struct zfcp_wka_port *wka_port)
|
int zfcp_fsf_open_wka_port(struct zfcp_fc_wka_port *wka_port)
|
||||||
{
|
{
|
||||||
struct qdio_buffer_element *sbale;
|
struct qdio_buffer_element *sbale;
|
||||||
struct zfcp_qdio *qdio = wka_port->adapter->qdio;
|
struct zfcp_qdio *qdio = wka_port->adapter->qdio;
|
||||||
|
@ -1677,7 +1664,7 @@ int zfcp_fsf_open_wka_port(struct zfcp_wka_port *wka_port)
|
||||||
sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY;
|
sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY;
|
||||||
|
|
||||||
req->handler = zfcp_fsf_open_wka_port_handler;
|
req->handler = zfcp_fsf_open_wka_port_handler;
|
||||||
req->qtcb->bottom.support.d_id = wka_port->d_id;
|
hton24(req->qtcb->bottom.support.d_id, wka_port->d_id);
|
||||||
req->data = wka_port;
|
req->data = wka_port;
|
||||||
|
|
||||||
zfcp_fsf_start_timer(req, ZFCP_FSF_REQUEST_TIMEOUT);
|
zfcp_fsf_start_timer(req, ZFCP_FSF_REQUEST_TIMEOUT);
|
||||||
|
@ -1691,23 +1678,23 @@ out:
|
||||||
|
|
||||||
static void zfcp_fsf_close_wka_port_handler(struct zfcp_fsf_req *req)
|
static void zfcp_fsf_close_wka_port_handler(struct zfcp_fsf_req *req)
|
||||||
{
|
{
|
||||||
struct zfcp_wka_port *wka_port = req->data;
|
struct zfcp_fc_wka_port *wka_port = req->data;
|
||||||
|
|
||||||
if (req->qtcb->header.fsf_status == FSF_PORT_HANDLE_NOT_VALID) {
|
if (req->qtcb->header.fsf_status == FSF_PORT_HANDLE_NOT_VALID) {
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
zfcp_erp_adapter_reopen(wka_port->adapter, 0, "fscwph1", req);
|
zfcp_erp_adapter_reopen(wka_port->adapter, 0, "fscwph1", req);
|
||||||
}
|
}
|
||||||
|
|
||||||
wka_port->status = ZFCP_WKA_PORT_OFFLINE;
|
wka_port->status = ZFCP_FC_WKA_PORT_OFFLINE;
|
||||||
wake_up(&wka_port->completion_wq);
|
wake_up(&wka_port->completion_wq);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* zfcp_fsf_close_wka_port - create and send close wka port request
|
* zfcp_fsf_close_wka_port - create and send close wka port request
|
||||||
* @erp_action: pointer to struct zfcp_erp_action
|
* @wka_port: WKA port to open
|
||||||
* Returns: 0 on success, error otherwise
|
* Returns: 0 on success, error otherwise
|
||||||
*/
|
*/
|
||||||
int zfcp_fsf_close_wka_port(struct zfcp_wka_port *wka_port)
|
int zfcp_fsf_close_wka_port(struct zfcp_fc_wka_port *wka_port)
|
||||||
{
|
{
|
||||||
struct qdio_buffer_element *sbale;
|
struct qdio_buffer_element *sbale;
|
||||||
struct zfcp_qdio *qdio = wka_port->adapter->qdio;
|
struct zfcp_qdio *qdio = wka_port->adapter->qdio;
|
||||||
|
@ -1765,13 +1752,13 @@ static void zfcp_fsf_close_physical_port_handler(struct zfcp_fsf_req *req)
|
||||||
/* can't use generic zfcp_erp_modify_port_status because
|
/* can't use generic zfcp_erp_modify_port_status because
|
||||||
* ZFCP_STATUS_COMMON_OPEN must not be reset for the port */
|
* ZFCP_STATUS_COMMON_OPEN must not be reset for the port */
|
||||||
atomic_clear_mask(ZFCP_STATUS_PORT_PHYS_OPEN, &port->status);
|
atomic_clear_mask(ZFCP_STATUS_PORT_PHYS_OPEN, &port->status);
|
||||||
list_for_each_entry(unit, &port->unit_list_head, list)
|
read_lock(&port->unit_list_lock);
|
||||||
|
list_for_each_entry(unit, &port->unit_list, list)
|
||||||
atomic_clear_mask(ZFCP_STATUS_COMMON_OPEN,
|
atomic_clear_mask(ZFCP_STATUS_COMMON_OPEN,
|
||||||
&unit->status);
|
&unit->status);
|
||||||
|
read_unlock(&port->unit_list_lock);
|
||||||
zfcp_erp_port_boxed(port, "fscpph2", req);
|
zfcp_erp_port_boxed(port, "fscpph2", req);
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR |
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
ZFCP_STATUS_FSFREQ_RETRY;
|
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case FSF_ADAPTER_STATUS_AVAILABLE:
|
case FSF_ADAPTER_STATUS_AVAILABLE:
|
||||||
switch (header->fsf_status_qual.word[0]) {
|
switch (header->fsf_status_qual.word[0]) {
|
||||||
|
@ -1787,9 +1774,11 @@ static void zfcp_fsf_close_physical_port_handler(struct zfcp_fsf_req *req)
|
||||||
* ZFCP_STATUS_COMMON_OPEN must not be reset for the port
|
* ZFCP_STATUS_COMMON_OPEN must not be reset for the port
|
||||||
*/
|
*/
|
||||||
atomic_clear_mask(ZFCP_STATUS_PORT_PHYS_OPEN, &port->status);
|
atomic_clear_mask(ZFCP_STATUS_PORT_PHYS_OPEN, &port->status);
|
||||||
list_for_each_entry(unit, &port->unit_list_head, list)
|
read_lock(&port->unit_list_lock);
|
||||||
|
list_for_each_entry(unit, &port->unit_list, list)
|
||||||
atomic_clear_mask(ZFCP_STATUS_COMMON_OPEN,
|
atomic_clear_mask(ZFCP_STATUS_COMMON_OPEN,
|
||||||
&unit->status);
|
&unit->status);
|
||||||
|
read_unlock(&port->unit_list_lock);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1873,8 +1862,7 @@ static void zfcp_fsf_open_unit_handler(struct zfcp_fsf_req *req)
|
||||||
break;
|
break;
|
||||||
case FSF_PORT_BOXED:
|
case FSF_PORT_BOXED:
|
||||||
zfcp_erp_port_boxed(unit->port, "fsouh_2", req);
|
zfcp_erp_port_boxed(unit->port, "fsouh_2", req);
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR |
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
ZFCP_STATUS_FSFREQ_RETRY;
|
|
||||||
break;
|
break;
|
||||||
case FSF_LUN_SHARING_VIOLATION:
|
case FSF_LUN_SHARING_VIOLATION:
|
||||||
if (header->fsf_status_qual.word[0])
|
if (header->fsf_status_qual.word[0])
|
||||||
|
@ -2036,8 +2024,7 @@ static void zfcp_fsf_close_unit_handler(struct zfcp_fsf_req *req)
|
||||||
break;
|
break;
|
||||||
case FSF_PORT_BOXED:
|
case FSF_PORT_BOXED:
|
||||||
zfcp_erp_port_boxed(unit->port, "fscuh_3", req);
|
zfcp_erp_port_boxed(unit->port, "fscuh_3", req);
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR |
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
ZFCP_STATUS_FSFREQ_RETRY;
|
|
||||||
break;
|
break;
|
||||||
case FSF_ADAPTER_STATUS_AVAILABLE:
|
case FSF_ADAPTER_STATUS_AVAILABLE:
|
||||||
switch (req->qtcb->header.fsf_status_qual.word[0]) {
|
switch (req->qtcb->header.fsf_status_qual.word[0]) {
|
||||||
|
@ -2109,13 +2096,27 @@ static void zfcp_fsf_update_lat(struct fsf_latency_record *lat_rec, u32 lat)
|
||||||
lat_rec->max = max(lat_rec->max, lat);
|
lat_rec->max = max(lat_rec->max, lat);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void zfcp_fsf_req_latency(struct zfcp_fsf_req *req)
|
static void zfcp_fsf_req_trace(struct zfcp_fsf_req *req, struct scsi_cmnd *scsi)
|
||||||
{
|
{
|
||||||
struct fsf_qual_latency_info *lat_inf;
|
struct fsf_qual_latency_info *lat_in;
|
||||||
struct latency_cont *lat;
|
struct latency_cont *lat = NULL;
|
||||||
struct zfcp_unit *unit = req->unit;
|
struct zfcp_unit *unit = req->unit;
|
||||||
|
struct zfcp_blk_drv_data blktrc;
|
||||||
|
int ticks = req->adapter->timer_ticks;
|
||||||
|
|
||||||
lat_inf = &req->qtcb->prefix.prot_status_qual.latency_info;
|
lat_in = &req->qtcb->prefix.prot_status_qual.latency_info;
|
||||||
|
|
||||||
|
blktrc.flags = 0;
|
||||||
|
blktrc.magic = ZFCP_BLK_DRV_DATA_MAGIC;
|
||||||
|
if (req->status & ZFCP_STATUS_FSFREQ_ERROR)
|
||||||
|
blktrc.flags |= ZFCP_BLK_REQ_ERROR;
|
||||||
|
blktrc.inb_usage = req->queue_req.qdio_inb_usage;
|
||||||
|
blktrc.outb_usage = req->queue_req.qdio_outb_usage;
|
||||||
|
|
||||||
|
if (req->adapter->adapter_features & FSF_FEATURE_MEASUREMENT_DATA) {
|
||||||
|
blktrc.flags |= ZFCP_BLK_LAT_VALID;
|
||||||
|
blktrc.channel_lat = lat_in->channel_lat * ticks;
|
||||||
|
blktrc.fabric_lat = lat_in->fabric_lat * ticks;
|
||||||
|
|
||||||
switch (req->qtcb->bottom.io.data_direction) {
|
switch (req->qtcb->bottom.io.data_direction) {
|
||||||
case FSF_DATADIR_READ:
|
case FSF_DATADIR_READ:
|
||||||
|
@ -2127,54 +2128,25 @@ static void zfcp_fsf_req_latency(struct zfcp_fsf_req *req)
|
||||||
case FSF_DATADIR_CMND:
|
case FSF_DATADIR_CMND:
|
||||||
lat = &unit->latencies.cmd;
|
lat = &unit->latencies.cmd;
|
||||||
break;
|
break;
|
||||||
default:
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (lat) {
|
||||||
spin_lock(&unit->latencies.lock);
|
spin_lock(&unit->latencies.lock);
|
||||||
zfcp_fsf_update_lat(&lat->channel, lat_inf->channel_lat);
|
zfcp_fsf_update_lat(&lat->channel, lat_in->channel_lat);
|
||||||
zfcp_fsf_update_lat(&lat->fabric, lat_inf->fabric_lat);
|
zfcp_fsf_update_lat(&lat->fabric, lat_in->fabric_lat);
|
||||||
lat->counter++;
|
lat->counter++;
|
||||||
spin_unlock(&unit->latencies.lock);
|
spin_unlock(&unit->latencies.lock);
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef CONFIG_BLK_DEV_IO_TRACE
|
|
||||||
static void zfcp_fsf_trace_latency(struct zfcp_fsf_req *fsf_req)
|
|
||||||
{
|
|
||||||
struct fsf_qual_latency_info *lat_inf;
|
|
||||||
struct scsi_cmnd *scsi_cmnd = (struct scsi_cmnd *)fsf_req->data;
|
|
||||||
struct request *req = scsi_cmnd->request;
|
|
||||||
struct zfcp_blk_drv_data trace;
|
|
||||||
int ticks = fsf_req->adapter->timer_ticks;
|
|
||||||
|
|
||||||
trace.flags = 0;
|
|
||||||
trace.magic = ZFCP_BLK_DRV_DATA_MAGIC;
|
|
||||||
if (fsf_req->adapter->adapter_features & FSF_FEATURE_MEASUREMENT_DATA) {
|
|
||||||
trace.flags |= ZFCP_BLK_LAT_VALID;
|
|
||||||
lat_inf = &fsf_req->qtcb->prefix.prot_status_qual.latency_info;
|
|
||||||
trace.channel_lat = lat_inf->channel_lat * ticks;
|
|
||||||
trace.fabric_lat = lat_inf->fabric_lat * ticks;
|
|
||||||
}
|
}
|
||||||
if (fsf_req->status & ZFCP_STATUS_FSFREQ_ERROR)
|
}
|
||||||
trace.flags |= ZFCP_BLK_REQ_ERROR;
|
|
||||||
trace.inb_usage = fsf_req->queue_req.qdio_inb_usage;
|
|
||||||
trace.outb_usage = fsf_req->queue_req.qdio_outb_usage;
|
|
||||||
|
|
||||||
blk_add_driver_data(req->q, req, &trace, sizeof(trace));
|
blk_add_driver_data(scsi->request->q, scsi->request, &blktrc,
|
||||||
|
sizeof(blktrc));
|
||||||
}
|
}
|
||||||
#else
|
|
||||||
static inline void zfcp_fsf_trace_latency(struct zfcp_fsf_req *fsf_req)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
static void zfcp_fsf_send_fcp_command_task_handler(struct zfcp_fsf_req *req)
|
static void zfcp_fsf_send_fcp_command_task_handler(struct zfcp_fsf_req *req)
|
||||||
{
|
{
|
||||||
struct scsi_cmnd *scpnt;
|
struct scsi_cmnd *scpnt;
|
||||||
struct fcp_rsp_iu *fcp_rsp_iu = (struct fcp_rsp_iu *)
|
struct fcp_resp_with_ext *fcp_rsp;
|
||||||
&(req->qtcb->bottom.io.fcp_rsp);
|
|
||||||
u32 sns_len;
|
|
||||||
char *fcp_rsp_info = (unsigned char *) &fcp_rsp_iu[1];
|
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
read_lock_irqsave(&req->adapter->abort_lock, flags);
|
read_lock_irqsave(&req->adapter->abort_lock, flags);
|
||||||
|
@ -2185,50 +2157,16 @@ static void zfcp_fsf_send_fcp_command_task_handler(struct zfcp_fsf_req *req)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (unlikely(req->status & ZFCP_STATUS_FSFREQ_ABORTED)) {
|
|
||||||
set_host_byte(scpnt, DID_SOFT_ERROR);
|
|
||||||
goto skip_fsfstatus;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (unlikely(req->status & ZFCP_STATUS_FSFREQ_ERROR)) {
|
if (unlikely(req->status & ZFCP_STATUS_FSFREQ_ERROR)) {
|
||||||
set_host_byte(scpnt, DID_ERROR);
|
set_host_byte(scpnt, DID_TRANSPORT_DISRUPTED);
|
||||||
goto skip_fsfstatus;
|
goto skip_fsfstatus;
|
||||||
}
|
}
|
||||||
|
|
||||||
set_msg_byte(scpnt, COMMAND_COMPLETE);
|
fcp_rsp = (struct fcp_resp_with_ext *) &req->qtcb->bottom.io.fcp_rsp;
|
||||||
|
zfcp_fc_eval_fcp_rsp(fcp_rsp, scpnt);
|
||||||
|
|
||||||
scpnt->result |= fcp_rsp_iu->scsi_status;
|
zfcp_fsf_req_trace(req, scpnt);
|
||||||
|
|
||||||
if (req->adapter->adapter_features & FSF_FEATURE_MEASUREMENT_DATA)
|
|
||||||
zfcp_fsf_req_latency(req);
|
|
||||||
|
|
||||||
zfcp_fsf_trace_latency(req);
|
|
||||||
|
|
||||||
if (unlikely(fcp_rsp_iu->validity.bits.fcp_rsp_len_valid)) {
|
|
||||||
if (fcp_rsp_info[3] == RSP_CODE_GOOD)
|
|
||||||
set_host_byte(scpnt, DID_OK);
|
|
||||||
else {
|
|
||||||
set_host_byte(scpnt, DID_ERROR);
|
|
||||||
goto skip_fsfstatus;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (unlikely(fcp_rsp_iu->validity.bits.fcp_sns_len_valid)) {
|
|
||||||
sns_len = FSF_FCP_RSP_SIZE - sizeof(struct fcp_rsp_iu) +
|
|
||||||
fcp_rsp_iu->fcp_rsp_len;
|
|
||||||
sns_len = min(sns_len, (u32) SCSI_SENSE_BUFFERSIZE);
|
|
||||||
sns_len = min(sns_len, fcp_rsp_iu->fcp_sns_len);
|
|
||||||
|
|
||||||
memcpy(scpnt->sense_buffer,
|
|
||||||
zfcp_get_fcp_sns_info_ptr(fcp_rsp_iu), sns_len);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (unlikely(fcp_rsp_iu->validity.bits.fcp_resid_under)) {
|
|
||||||
scsi_set_resid(scpnt, fcp_rsp_iu->fcp_resid);
|
|
||||||
if (scsi_bufflen(scpnt) - scsi_get_resid(scpnt) <
|
|
||||||
scpnt->underflow)
|
|
||||||
set_host_byte(scpnt, DID_ERROR);
|
|
||||||
}
|
|
||||||
skip_fsfstatus:
|
skip_fsfstatus:
|
||||||
if (scpnt->result != 0)
|
if (scpnt->result != 0)
|
||||||
zfcp_dbf_scsi_result("erro", 3, req->adapter->dbf, scpnt, req);
|
zfcp_dbf_scsi_result("erro", 3, req->adapter->dbf, scpnt, req);
|
||||||
|
@ -2250,11 +2188,13 @@ skip_fsfstatus:
|
||||||
|
|
||||||
static void zfcp_fsf_send_fcp_ctm_handler(struct zfcp_fsf_req *req)
|
static void zfcp_fsf_send_fcp_ctm_handler(struct zfcp_fsf_req *req)
|
||||||
{
|
{
|
||||||
struct fcp_rsp_iu *fcp_rsp_iu = (struct fcp_rsp_iu *)
|
struct fcp_resp_with_ext *fcp_rsp;
|
||||||
&(req->qtcb->bottom.io.fcp_rsp);
|
struct fcp_resp_rsp_info *rsp_info;
|
||||||
char *fcp_rsp_info = (unsigned char *) &fcp_rsp_iu[1];
|
|
||||||
|
|
||||||
if ((fcp_rsp_info[3] != RSP_CODE_GOOD) ||
|
fcp_rsp = (struct fcp_resp_with_ext *) &req->qtcb->bottom.io.fcp_rsp;
|
||||||
|
rsp_info = (struct fcp_resp_rsp_info *) &fcp_rsp[1];
|
||||||
|
|
||||||
|
if ((rsp_info->rsp_code != FCP_TMF_CMPL) ||
|
||||||
(req->status & ZFCP_STATUS_FSFREQ_ERROR))
|
(req->status & ZFCP_STATUS_FSFREQ_ERROR))
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_TMFUNCFAILED;
|
req->status |= ZFCP_STATUS_FSFREQ_TMFUNCFAILED;
|
||||||
}
|
}
|
||||||
|
@ -2314,13 +2254,11 @@ static void zfcp_fsf_send_fcp_command_handler(struct zfcp_fsf_req *req)
|
||||||
break;
|
break;
|
||||||
case FSF_PORT_BOXED:
|
case FSF_PORT_BOXED:
|
||||||
zfcp_erp_port_boxed(unit->port, "fssfch5", req);
|
zfcp_erp_port_boxed(unit->port, "fssfch5", req);
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR |
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
ZFCP_STATUS_FSFREQ_RETRY;
|
|
||||||
break;
|
break;
|
||||||
case FSF_LUN_BOXED:
|
case FSF_LUN_BOXED:
|
||||||
zfcp_erp_unit_boxed(unit, "fssfch6", req);
|
zfcp_erp_unit_boxed(unit, "fssfch6", req);
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_ERROR |
|
req->status |= ZFCP_STATUS_FSFREQ_ERROR;
|
||||||
ZFCP_STATUS_FSFREQ_RETRY;
|
|
||||||
break;
|
break;
|
||||||
case FSF_ADAPTER_STATUS_AVAILABLE:
|
case FSF_ADAPTER_STATUS_AVAILABLE:
|
||||||
if (header->fsf_status_qual.word[0] ==
|
if (header->fsf_status_qual.word[0] ==
|
||||||
|
@ -2335,24 +2273,10 @@ skip_fsfstatus:
|
||||||
else {
|
else {
|
||||||
zfcp_fsf_send_fcp_command_task_handler(req);
|
zfcp_fsf_send_fcp_command_task_handler(req);
|
||||||
req->unit = NULL;
|
req->unit = NULL;
|
||||||
zfcp_unit_put(unit);
|
put_device(&unit->sysfs_device);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void zfcp_set_fcp_dl(struct fcp_cmnd_iu *fcp_cmd, u32 fcp_dl)
|
|
||||||
{
|
|
||||||
u32 *fcp_dl_ptr;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* fcp_dl_addr = start address of fcp_cmnd structure +
|
|
||||||
* size of fixed part + size of dynamically sized add_dcp_cdb field
|
|
||||||
* SEE FCP-2 documentation
|
|
||||||
*/
|
|
||||||
fcp_dl_ptr = (u32 *) ((unsigned char *) &fcp_cmd[1] +
|
|
||||||
(fcp_cmd->add_fcp_cdb_length << 2));
|
|
||||||
*fcp_dl_ptr = fcp_dl;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* zfcp_fsf_send_fcp_command_task - initiate an FCP command (for a SCSI command)
|
* zfcp_fsf_send_fcp_command_task - initiate an FCP command (for a SCSI command)
|
||||||
* @unit: unit where command is sent to
|
* @unit: unit where command is sent to
|
||||||
|
@ -2362,7 +2286,7 @@ int zfcp_fsf_send_fcp_command_task(struct zfcp_unit *unit,
|
||||||
struct scsi_cmnd *scsi_cmnd)
|
struct scsi_cmnd *scsi_cmnd)
|
||||||
{
|
{
|
||||||
struct zfcp_fsf_req *req;
|
struct zfcp_fsf_req *req;
|
||||||
struct fcp_cmnd_iu *fcp_cmnd_iu;
|
struct fcp_cmnd *fcp_cmnd;
|
||||||
unsigned int sbtype = SBAL_FLAGS0_TYPE_READ;
|
unsigned int sbtype = SBAL_FLAGS0_TYPE_READ;
|
||||||
int real_bytes, retval = -EIO;
|
int real_bytes, retval = -EIO;
|
||||||
struct zfcp_adapter *adapter = unit->port->adapter;
|
struct zfcp_adapter *adapter = unit->port->adapter;
|
||||||
|
@ -2387,23 +2311,21 @@ int zfcp_fsf_send_fcp_command_task(struct zfcp_unit *unit,
|
||||||
}
|
}
|
||||||
|
|
||||||
req->status |= ZFCP_STATUS_FSFREQ_CLEANUP;
|
req->status |= ZFCP_STATUS_FSFREQ_CLEANUP;
|
||||||
zfcp_unit_get(unit);
|
get_device(&unit->sysfs_device);
|
||||||
req->unit = unit;
|
req->unit = unit;
|
||||||
req->data = scsi_cmnd;
|
req->data = scsi_cmnd;
|
||||||
req->handler = zfcp_fsf_send_fcp_command_handler;
|
req->handler = zfcp_fsf_send_fcp_command_handler;
|
||||||
req->qtcb->header.lun_handle = unit->handle;
|
req->qtcb->header.lun_handle = unit->handle;
|
||||||
req->qtcb->header.port_handle = unit->port->handle;
|
req->qtcb->header.port_handle = unit->port->handle;
|
||||||
req->qtcb->bottom.io.service_class = FSF_CLASS_3;
|
req->qtcb->bottom.io.service_class = FSF_CLASS_3;
|
||||||
|
req->qtcb->bottom.io.fcp_cmnd_length = FCP_CMND_LEN;
|
||||||
|
|
||||||
scsi_cmnd->host_scribble = (unsigned char *) req->req_id;
|
scsi_cmnd->host_scribble = (unsigned char *) req->req_id;
|
||||||
|
|
||||||
fcp_cmnd_iu = (struct fcp_cmnd_iu *) &(req->qtcb->bottom.io.fcp_cmnd);
|
|
||||||
fcp_cmnd_iu->fcp_lun = unit->fcp_lun;
|
|
||||||
/*
|
/*
|
||||||
* set depending on data direction:
|
* set depending on data direction:
|
||||||
* data direction bits in SBALE (SB Type)
|
* data direction bits in SBALE (SB Type)
|
||||||
* data direction bits in QTCB
|
* data direction bits in QTCB
|
||||||
* data direction bits in FCP_CMND IU
|
|
||||||
*/
|
*/
|
||||||
switch (scsi_cmnd->sc_data_direction) {
|
switch (scsi_cmnd->sc_data_direction) {
|
||||||
case DMA_NONE:
|
case DMA_NONE:
|
||||||
|
@ -2411,32 +2333,17 @@ int zfcp_fsf_send_fcp_command_task(struct zfcp_unit *unit,
|
||||||
break;
|
break;
|
||||||
case DMA_FROM_DEVICE:
|
case DMA_FROM_DEVICE:
|
||||||
req->qtcb->bottom.io.data_direction = FSF_DATADIR_READ;
|
req->qtcb->bottom.io.data_direction = FSF_DATADIR_READ;
|
||||||
fcp_cmnd_iu->rddata = 1;
|
|
||||||
break;
|
break;
|
||||||
case DMA_TO_DEVICE:
|
case DMA_TO_DEVICE:
|
||||||
req->qtcb->bottom.io.data_direction = FSF_DATADIR_WRITE;
|
req->qtcb->bottom.io.data_direction = FSF_DATADIR_WRITE;
|
||||||
sbtype = SBAL_FLAGS0_TYPE_WRITE;
|
sbtype = SBAL_FLAGS0_TYPE_WRITE;
|
||||||
fcp_cmnd_iu->wddata = 1;
|
|
||||||
break;
|
break;
|
||||||
case DMA_BIDIRECTIONAL:
|
case DMA_BIDIRECTIONAL:
|
||||||
goto failed_scsi_cmnd;
|
goto failed_scsi_cmnd;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (likely((scsi_cmnd->device->simple_tags) ||
|
fcp_cmnd = (struct fcp_cmnd *) &req->qtcb->bottom.io.fcp_cmnd;
|
||||||
((atomic_read(&unit->status) & ZFCP_STATUS_UNIT_READONLY) &&
|
zfcp_fc_scsi_to_fcp(fcp_cmnd, scsi_cmnd);
|
||||||
(atomic_read(&unit->status) & ZFCP_STATUS_UNIT_SHARED))))
|
|
||||||
fcp_cmnd_iu->task_attribute = SIMPLE_Q;
|
|
||||||
else
|
|
||||||
fcp_cmnd_iu->task_attribute = UNTAGGED;
|
|
||||||
|
|
||||||
if (unlikely(scsi_cmnd->cmd_len > FCP_CDB_LENGTH))
|
|
||||||
fcp_cmnd_iu->add_fcp_cdb_length =
|
|
||||||
(scsi_cmnd->cmd_len - FCP_CDB_LENGTH) >> 2;
|
|
||||||
|
|
||||||
memcpy(fcp_cmnd_iu->fcp_cdb, scsi_cmnd->cmnd, scsi_cmnd->cmd_len);
|
|
||||||
|
|
||||||
req->qtcb->bottom.io.fcp_cmnd_length = sizeof(struct fcp_cmnd_iu) +
|
|
||||||
fcp_cmnd_iu->add_fcp_cdb_length + sizeof(u32);
|
|
||||||
|
|
||||||
real_bytes = zfcp_qdio_sbals_from_sg(qdio, &req->queue_req, sbtype,
|
real_bytes = zfcp_qdio_sbals_from_sg(qdio, &req->queue_req, sbtype,
|
||||||
scsi_sglist(scsi_cmnd),
|
scsi_sglist(scsi_cmnd),
|
||||||
|
@ -2454,8 +2361,6 @@ int zfcp_fsf_send_fcp_command_task(struct zfcp_unit *unit,
|
||||||
goto failed_scsi_cmnd;
|
goto failed_scsi_cmnd;
|
||||||
}
|
}
|
||||||
|
|
||||||
zfcp_set_fcp_dl(fcp_cmnd_iu, real_bytes);
|
|
||||||
|
|
||||||
retval = zfcp_fsf_req_send(req);
|
retval = zfcp_fsf_req_send(req);
|
||||||
if (unlikely(retval))
|
if (unlikely(retval))
|
||||||
goto failed_scsi_cmnd;
|
goto failed_scsi_cmnd;
|
||||||
|
@ -2463,7 +2368,7 @@ int zfcp_fsf_send_fcp_command_task(struct zfcp_unit *unit,
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
failed_scsi_cmnd:
|
failed_scsi_cmnd:
|
||||||
zfcp_unit_put(unit);
|
put_device(&unit->sysfs_device);
|
||||||
zfcp_fsf_req_free(req);
|
zfcp_fsf_req_free(req);
|
||||||
scsi_cmnd->host_scribble = NULL;
|
scsi_cmnd->host_scribble = NULL;
|
||||||
out:
|
out:
|
||||||
|
@ -2481,7 +2386,7 @@ struct zfcp_fsf_req *zfcp_fsf_send_fcp_ctm(struct zfcp_unit *unit, u8 tm_flags)
|
||||||
{
|
{
|
||||||
struct qdio_buffer_element *sbale;
|
struct qdio_buffer_element *sbale;
|
||||||
struct zfcp_fsf_req *req = NULL;
|
struct zfcp_fsf_req *req = NULL;
|
||||||
struct fcp_cmnd_iu *fcp_cmnd_iu;
|
struct fcp_cmnd *fcp_cmnd;
|
||||||
struct zfcp_qdio *qdio = unit->port->adapter->qdio;
|
struct zfcp_qdio *qdio = unit->port->adapter->qdio;
|
||||||
|
|
||||||
if (unlikely(!(atomic_read(&unit->status) &
|
if (unlikely(!(atomic_read(&unit->status) &
|
||||||
|
@ -2507,16 +2412,14 @@ struct zfcp_fsf_req *zfcp_fsf_send_fcp_ctm(struct zfcp_unit *unit, u8 tm_flags)
|
||||||
req->qtcb->header.port_handle = unit->port->handle;
|
req->qtcb->header.port_handle = unit->port->handle;
|
||||||
req->qtcb->bottom.io.data_direction = FSF_DATADIR_CMND;
|
req->qtcb->bottom.io.data_direction = FSF_DATADIR_CMND;
|
||||||
req->qtcb->bottom.io.service_class = FSF_CLASS_3;
|
req->qtcb->bottom.io.service_class = FSF_CLASS_3;
|
||||||
req->qtcb->bottom.io.fcp_cmnd_length = sizeof(struct fcp_cmnd_iu) +
|
req->qtcb->bottom.io.fcp_cmnd_length = FCP_CMND_LEN;
|
||||||
sizeof(u32);
|
|
||||||
|
|
||||||
sbale = zfcp_qdio_sbale_req(qdio, &req->queue_req);
|
sbale = zfcp_qdio_sbale_req(qdio, &req->queue_req);
|
||||||
sbale[0].flags |= SBAL_FLAGS0_TYPE_WRITE;
|
sbale[0].flags |= SBAL_FLAGS0_TYPE_WRITE;
|
||||||
sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY;
|
sbale[1].flags |= SBAL_FLAGS_LAST_ENTRY;
|
||||||
|
|
||||||
fcp_cmnd_iu = (struct fcp_cmnd_iu *) &req->qtcb->bottom.io.fcp_cmnd;
|
fcp_cmnd = (struct fcp_cmnd *) &req->qtcb->bottom.io.fcp_cmnd;
|
||||||
fcp_cmnd_iu->fcp_lun = unit->fcp_lun;
|
zfcp_fc_fcp_tm(fcp_cmnd, unit->device, tm_flags);
|
||||||
fcp_cmnd_iu->task_management_flags = tm_flags;
|
|
||||||
|
|
||||||
zfcp_fsf_start_timer(req, ZFCP_SCSI_ER_TIMEOUT);
|
zfcp_fsf_start_timer(req, ZFCP_SCSI_ER_TIMEOUT);
|
||||||
if (!zfcp_fsf_req_send(req))
|
if (!zfcp_fsf_req_send(req))
|
||||||
|
|
|
@ -11,6 +11,7 @@
|
||||||
|
|
||||||
#include <linux/pfn.h>
|
#include <linux/pfn.h>
|
||||||
#include <linux/scatterlist.h>
|
#include <linux/scatterlist.h>
|
||||||
|
#include <scsi/libfc.h>
|
||||||
|
|
||||||
#define FSF_QTCB_CURRENT_VERSION 0x00000001
|
#define FSF_QTCB_CURRENT_VERSION 0x00000001
|
||||||
|
|
||||||
|
@ -228,7 +229,8 @@ struct fsf_status_read_buffer {
|
||||||
u32 length;
|
u32 length;
|
||||||
u32 res1;
|
u32 res1;
|
||||||
struct fsf_queue_designator queue_designator;
|
struct fsf_queue_designator queue_designator;
|
||||||
u32 d_id;
|
u8 res2;
|
||||||
|
u8 d_id[3];
|
||||||
u32 class;
|
u32 class;
|
||||||
u64 fcp_lun;
|
u64 fcp_lun;
|
||||||
u8 res3[24];
|
u8 res3[24];
|
||||||
|
@ -309,22 +311,7 @@ struct fsf_qtcb_header {
|
||||||
u8 res4[16];
|
u8 res4[16];
|
||||||
} __attribute__ ((packed));
|
} __attribute__ ((packed));
|
||||||
|
|
||||||
struct fsf_nport_serv_param {
|
|
||||||
u8 common_serv_param[16];
|
|
||||||
u64 wwpn;
|
|
||||||
u64 wwnn;
|
|
||||||
u8 class1_serv_param[16];
|
|
||||||
u8 class2_serv_param[16];
|
|
||||||
u8 class3_serv_param[16];
|
|
||||||
u8 class4_serv_param[16];
|
|
||||||
u8 vendor_version_level[16];
|
|
||||||
} __attribute__ ((packed));
|
|
||||||
|
|
||||||
#define FSF_PLOGI_MIN_LEN 112
|
#define FSF_PLOGI_MIN_LEN 112
|
||||||
struct fsf_plogi {
|
|
||||||
u32 code;
|
|
||||||
struct fsf_nport_serv_param serv_param;
|
|
||||||
} __attribute__ ((packed));
|
|
||||||
|
|
||||||
#define FSF_FCP_CMND_SIZE 288
|
#define FSF_FCP_CMND_SIZE 288
|
||||||
#define FSF_FCP_RSP_SIZE 128
|
#define FSF_FCP_RSP_SIZE 128
|
||||||
|
@ -342,8 +329,8 @@ struct fsf_qtcb_bottom_io {
|
||||||
|
|
||||||
struct fsf_qtcb_bottom_support {
|
struct fsf_qtcb_bottom_support {
|
||||||
u32 operation_subtype;
|
u32 operation_subtype;
|
||||||
u8 res1[12];
|
u8 res1[13];
|
||||||
u32 d_id;
|
u8 d_id[3];
|
||||||
u32 option;
|
u32 option;
|
||||||
u64 fcp_lun;
|
u64 fcp_lun;
|
||||||
u64 res2;
|
u64 res2;
|
||||||
|
@ -372,18 +359,18 @@ struct fsf_qtcb_bottom_config {
|
||||||
u32 fc_topology;
|
u32 fc_topology;
|
||||||
u32 fc_link_speed;
|
u32 fc_link_speed;
|
||||||
u32 adapter_type;
|
u32 adapter_type;
|
||||||
u32 peer_d_id;
|
u8 res0;
|
||||||
|
u8 peer_d_id[3];
|
||||||
u8 res1[2];
|
u8 res1[2];
|
||||||
u16 timer_interval;
|
u16 timer_interval;
|
||||||
u8 res2[8];
|
u8 res2[9];
|
||||||
u32 s_id;
|
u8 s_id[3];
|
||||||
struct fsf_nport_serv_param nport_serv_param;
|
u8 nport_serv_param[128];
|
||||||
u8 reserved_nport_serv_param[16];
|
|
||||||
u8 res3[8];
|
u8 res3[8];
|
||||||
u32 adapter_ports;
|
u32 adapter_ports;
|
||||||
u32 hardware_version;
|
u32 hardware_version;
|
||||||
u8 serial_number[32];
|
u8 serial_number[32];
|
||||||
struct fsf_nport_serv_param plogi_payload;
|
u8 plogi_payload[112];
|
||||||
struct fsf_statistics_info stat_info;
|
struct fsf_statistics_info stat_info;
|
||||||
u8 res4[112];
|
u8 res4[112];
|
||||||
} __attribute__ ((packed));
|
} __attribute__ ((packed));
|
||||||
|
@ -450,4 +437,22 @@ struct zfcp_blk_drv_data {
|
||||||
u64 fabric_lat;
|
u64 fabric_lat;
|
||||||
} __attribute__ ((packed));
|
} __attribute__ ((packed));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct zfcp_fsf_ct_els - zfcp data for ct or els request
|
||||||
|
* @req: scatter-gather list for request
|
||||||
|
* @resp: scatter-gather list for response
|
||||||
|
* @handler: handler function (called for response to the request)
|
||||||
|
* @handler_data: data passed to handler function
|
||||||
|
* @port: Optional pointer to port for zfcp internal ELS (only test link ADISC)
|
||||||
|
* @status: used to pass error status to calling function
|
||||||
|
*/
|
||||||
|
struct zfcp_fsf_ct_els {
|
||||||
|
struct scatterlist *req;
|
||||||
|
struct scatterlist *resp;
|
||||||
|
void (*handler)(void *);
|
||||||
|
void *handler_data;
|
||||||
|
struct zfcp_port *port;
|
||||||
|
int status;
|
||||||
|
};
|
||||||
|
|
||||||
#endif /* FSF_H */
|
#endif /* FSF_H */
|
||||||
|
|
|
@ -9,29 +9,33 @@
|
||||||
#define KMSG_COMPONENT "zfcp"
|
#define KMSG_COMPONENT "zfcp"
|
||||||
#define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
|
#define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
|
||||||
|
|
||||||
|
#include <linux/types.h>
|
||||||
|
#include <scsi/fc/fc_fcp.h>
|
||||||
#include <asm/atomic.h>
|
#include <asm/atomic.h>
|
||||||
#include "zfcp_ext.h"
|
#include "zfcp_ext.h"
|
||||||
#include "zfcp_dbf.h"
|
#include "zfcp_dbf.h"
|
||||||
|
#include "zfcp_fc.h"
|
||||||
|
|
||||||
static unsigned int default_depth = 32;
|
static unsigned int default_depth = 32;
|
||||||
module_param_named(queue_depth, default_depth, uint, 0600);
|
module_param_named(queue_depth, default_depth, uint, 0600);
|
||||||
MODULE_PARM_DESC(queue_depth, "Default queue depth for new SCSI devices");
|
MODULE_PARM_DESC(queue_depth, "Default queue depth for new SCSI devices");
|
||||||
|
|
||||||
/* Find start of Sense Information in FCP response unit*/
|
static int zfcp_scsi_change_queue_depth(struct scsi_device *sdev, int depth,
|
||||||
char *zfcp_get_fcp_sns_info_ptr(struct fcp_rsp_iu *fcp_rsp_iu)
|
int reason)
|
||||||
{
|
|
||||||
char *fcp_sns_info_ptr;
|
|
||||||
|
|
||||||
fcp_sns_info_ptr = (unsigned char *) &fcp_rsp_iu[1];
|
|
||||||
if (fcp_rsp_iu->validity.bits.fcp_rsp_len_valid)
|
|
||||||
fcp_sns_info_ptr += fcp_rsp_iu->fcp_rsp_len;
|
|
||||||
|
|
||||||
return fcp_sns_info_ptr;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int zfcp_scsi_change_queue_depth(struct scsi_device *sdev, int depth)
|
|
||||||
{
|
{
|
||||||
|
switch (reason) {
|
||||||
|
case SCSI_QDEPTH_DEFAULT:
|
||||||
scsi_adjust_queue_depth(sdev, scsi_get_tag_type(sdev), depth);
|
scsi_adjust_queue_depth(sdev, scsi_get_tag_type(sdev), depth);
|
||||||
|
break;
|
||||||
|
case SCSI_QDEPTH_QFULL:
|
||||||
|
scsi_track_queue_full(sdev, depth);
|
||||||
|
break;
|
||||||
|
case SCSI_QDEPTH_RAMP_UP:
|
||||||
|
scsi_adjust_queue_depth(sdev, scsi_get_tag_type(sdev), depth);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
}
|
||||||
return sdev->queue_depth;
|
return sdev->queue_depth;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,7 +43,7 @@ static void zfcp_scsi_slave_destroy(struct scsi_device *sdpnt)
|
||||||
{
|
{
|
||||||
struct zfcp_unit *unit = (struct zfcp_unit *) sdpnt->hostdata;
|
struct zfcp_unit *unit = (struct zfcp_unit *) sdpnt->hostdata;
|
||||||
unit->device = NULL;
|
unit->device = NULL;
|
||||||
zfcp_unit_put(unit);
|
put_device(&unit->sysfs_device);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int zfcp_scsi_slave_configure(struct scsi_device *sdp)
|
static int zfcp_scsi_slave_configure(struct scsi_device *sdp)
|
||||||
|
@ -99,12 +103,26 @@ static int zfcp_scsi_queuecommand(struct scsi_cmnd *scpnt,
|
||||||
}
|
}
|
||||||
|
|
||||||
status = atomic_read(&unit->status);
|
status = atomic_read(&unit->status);
|
||||||
if (unlikely((status & ZFCP_STATUS_COMMON_ERP_FAILED) ||
|
if (unlikely(status & ZFCP_STATUS_COMMON_ERP_FAILED) &&
|
||||||
!(status & ZFCP_STATUS_COMMON_RUNNING))) {
|
!(atomic_read(&unit->port->status) &
|
||||||
|
ZFCP_STATUS_COMMON_ERP_FAILED)) {
|
||||||
|
/* only unit access denied, but port is good
|
||||||
|
* not covered by FC transport, have to fail here */
|
||||||
zfcp_scsi_command_fail(scpnt, DID_ERROR);
|
zfcp_scsi_command_fail(scpnt, DID_ERROR);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (unlikely(!(status & ZFCP_STATUS_COMMON_UNBLOCKED))) {
|
||||||
|
/* This could be either
|
||||||
|
* open unit pending: this is temporary, will result in
|
||||||
|
* open unit or ERP_FAILED, so retry command
|
||||||
|
* call to rport_delete pending: mimic retry from
|
||||||
|
* fc_remote_port_chkready until rport is BLOCKED
|
||||||
|
*/
|
||||||
|
zfcp_scsi_command_fail(scpnt, DID_IMM_RETRY);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
ret = zfcp_fsf_send_fcp_command_task(unit, scpnt);
|
ret = zfcp_fsf_send_fcp_command_task(unit, scpnt);
|
||||||
if (unlikely(ret == -EBUSY))
|
if (unlikely(ret == -EBUSY))
|
||||||
return SCSI_MLQUEUE_DEVICE_BUSY;
|
return SCSI_MLQUEUE_DEVICE_BUSY;
|
||||||
|
@ -115,49 +133,44 @@ static int zfcp_scsi_queuecommand(struct scsi_cmnd *scpnt,
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct zfcp_unit *zfcp_unit_lookup(struct zfcp_adapter *adapter,
|
static struct zfcp_unit *zfcp_unit_lookup(struct zfcp_adapter *adapter,
|
||||||
int channel, unsigned int id,
|
unsigned int id, u64 lun)
|
||||||
unsigned int lun)
|
|
||||||
{
|
{
|
||||||
|
unsigned long flags;
|
||||||
struct zfcp_port *port;
|
struct zfcp_port *port;
|
||||||
struct zfcp_unit *unit;
|
struct zfcp_unit *unit = NULL;
|
||||||
int scsi_lun;
|
|
||||||
|
|
||||||
list_for_each_entry(port, &adapter->port_list_head, list) {
|
read_lock_irqsave(&adapter->port_list_lock, flags);
|
||||||
|
list_for_each_entry(port, &adapter->port_list, list) {
|
||||||
if (!port->rport || (id != port->rport->scsi_target_id))
|
if (!port->rport || (id != port->rport->scsi_target_id))
|
||||||
continue;
|
continue;
|
||||||
list_for_each_entry(unit, &port->unit_list_head, list) {
|
unit = zfcp_get_unit_by_lun(port, lun);
|
||||||
scsi_lun = scsilun_to_int(
|
if (unit)
|
||||||
(struct scsi_lun *)&unit->fcp_lun);
|
break;
|
||||||
if (lun == scsi_lun)
|
|
||||||
return unit;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
read_unlock_irqrestore(&adapter->port_list_lock, flags);
|
||||||
|
|
||||||
return NULL;
|
return unit;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int zfcp_scsi_slave_alloc(struct scsi_device *sdp)
|
static int zfcp_scsi_slave_alloc(struct scsi_device *sdp)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter;
|
struct zfcp_adapter *adapter;
|
||||||
struct zfcp_unit *unit;
|
struct zfcp_unit *unit;
|
||||||
unsigned long flags;
|
u64 lun;
|
||||||
int retval = -ENXIO;
|
|
||||||
|
|
||||||
adapter = (struct zfcp_adapter *) sdp->host->hostdata[0];
|
adapter = (struct zfcp_adapter *) sdp->host->hostdata[0];
|
||||||
if (!adapter)
|
if (!adapter)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
read_lock_irqsave(&zfcp_data.config_lock, flags);
|
int_to_scsilun(sdp->lun, (struct scsi_lun *)&lun);
|
||||||
unit = zfcp_unit_lookup(adapter, sdp->channel, sdp->id, sdp->lun);
|
unit = zfcp_unit_lookup(adapter, sdp->id, lun);
|
||||||
if (unit) {
|
if (unit) {
|
||||||
sdp->hostdata = unit;
|
sdp->hostdata = unit;
|
||||||
unit->device = sdp;
|
unit->device = sdp;
|
||||||
zfcp_unit_get(unit);
|
return 0;
|
||||||
retval = 0;
|
|
||||||
}
|
}
|
||||||
read_unlock_irqrestore(&zfcp_data.config_lock, flags);
|
|
||||||
out:
|
out:
|
||||||
return retval;
|
return -ENXIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int zfcp_scsi_eh_abort_handler(struct scsi_cmnd *scpnt)
|
static int zfcp_scsi_eh_abort_handler(struct scsi_cmnd *scpnt)
|
||||||
|
@ -196,6 +209,7 @@ static int zfcp_scsi_eh_abort_handler(struct scsi_cmnd *scpnt)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
zfcp_erp_wait(adapter);
|
zfcp_erp_wait(adapter);
|
||||||
|
fc_block_scsi_eh(scpnt);
|
||||||
if (!(atomic_read(&adapter->status) &
|
if (!(atomic_read(&adapter->status) &
|
||||||
ZFCP_STATUS_COMMON_RUNNING)) {
|
ZFCP_STATUS_COMMON_RUNNING)) {
|
||||||
zfcp_dbf_scsi_abort("nres", adapter->dbf, scpnt, NULL,
|
zfcp_dbf_scsi_abort("nres", adapter->dbf, scpnt, NULL,
|
||||||
|
@ -235,6 +249,7 @@ static int zfcp_task_mgmt_function(struct scsi_cmnd *scpnt, u8 tm_flags)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
zfcp_erp_wait(adapter);
|
zfcp_erp_wait(adapter);
|
||||||
|
fc_block_scsi_eh(scpnt);
|
||||||
if (!(atomic_read(&adapter->status) &
|
if (!(atomic_read(&adapter->status) &
|
||||||
ZFCP_STATUS_COMMON_RUNNING)) {
|
ZFCP_STATUS_COMMON_RUNNING)) {
|
||||||
zfcp_dbf_scsi_devreset("nres", tm_flags, unit, scpnt);
|
zfcp_dbf_scsi_devreset("nres", tm_flags, unit, scpnt);
|
||||||
|
@ -249,9 +264,6 @@ static int zfcp_task_mgmt_function(struct scsi_cmnd *scpnt, u8 tm_flags)
|
||||||
if (fsf_req->status & ZFCP_STATUS_FSFREQ_TMFUNCFAILED) {
|
if (fsf_req->status & ZFCP_STATUS_FSFREQ_TMFUNCFAILED) {
|
||||||
zfcp_dbf_scsi_devreset("fail", tm_flags, unit, scpnt);
|
zfcp_dbf_scsi_devreset("fail", tm_flags, unit, scpnt);
|
||||||
retval = FAILED;
|
retval = FAILED;
|
||||||
} else if (fsf_req->status & ZFCP_STATUS_FSFREQ_TMFUNCNOTSUPP) {
|
|
||||||
zfcp_dbf_scsi_devreset("nsup", tm_flags, unit, scpnt);
|
|
||||||
retval = FAILED;
|
|
||||||
} else
|
} else
|
||||||
zfcp_dbf_scsi_devreset("okay", tm_flags, unit, scpnt);
|
zfcp_dbf_scsi_devreset("okay", tm_flags, unit, scpnt);
|
||||||
|
|
||||||
|
@ -261,12 +273,12 @@ static int zfcp_task_mgmt_function(struct scsi_cmnd *scpnt, u8 tm_flags)
|
||||||
|
|
||||||
static int zfcp_scsi_eh_device_reset_handler(struct scsi_cmnd *scpnt)
|
static int zfcp_scsi_eh_device_reset_handler(struct scsi_cmnd *scpnt)
|
||||||
{
|
{
|
||||||
return zfcp_task_mgmt_function(scpnt, FCP_LOGICAL_UNIT_RESET);
|
return zfcp_task_mgmt_function(scpnt, FCP_TMF_LUN_RESET);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int zfcp_scsi_eh_target_reset_handler(struct scsi_cmnd *scpnt)
|
static int zfcp_scsi_eh_target_reset_handler(struct scsi_cmnd *scpnt)
|
||||||
{
|
{
|
||||||
return zfcp_task_mgmt_function(scpnt, FCP_TARGET_RESET);
|
return zfcp_task_mgmt_function(scpnt, FCP_TMF_TGT_RESET);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int zfcp_scsi_eh_host_reset_handler(struct scsi_cmnd *scpnt)
|
static int zfcp_scsi_eh_host_reset_handler(struct scsi_cmnd *scpnt)
|
||||||
|
@ -276,6 +288,7 @@ static int zfcp_scsi_eh_host_reset_handler(struct scsi_cmnd *scpnt)
|
||||||
|
|
||||||
zfcp_erp_adapter_reopen(adapter, 0, "schrh_1", scpnt);
|
zfcp_erp_adapter_reopen(adapter, 0, "schrh_1", scpnt);
|
||||||
zfcp_erp_wait(adapter);
|
zfcp_erp_wait(adapter);
|
||||||
|
fc_block_scsi_eh(scpnt);
|
||||||
|
|
||||||
return SUCCESS;
|
return SUCCESS;
|
||||||
}
|
}
|
||||||
|
@ -303,7 +316,7 @@ int zfcp_adapter_scsi_register(struct zfcp_adapter *adapter)
|
||||||
adapter->scsi_host->max_lun = 1;
|
adapter->scsi_host->max_lun = 1;
|
||||||
adapter->scsi_host->max_channel = 0;
|
adapter->scsi_host->max_channel = 0;
|
||||||
adapter->scsi_host->unique_id = dev_id.devno;
|
adapter->scsi_host->unique_id = dev_id.devno;
|
||||||
adapter->scsi_host->max_cmd_len = 255;
|
adapter->scsi_host->max_cmd_len = 16; /* in struct fcp_cmnd */
|
||||||
adapter->scsi_host->transportt = zfcp_data.scsi_transport_template;
|
adapter->scsi_host->transportt = zfcp_data.scsi_transport_template;
|
||||||
|
|
||||||
adapter->scsi_host->hostdata[0] = (unsigned long) adapter;
|
adapter->scsi_host->hostdata[0] = (unsigned long) adapter;
|
||||||
|
@ -325,12 +338,11 @@ void zfcp_adapter_scsi_unregister(struct zfcp_adapter *adapter)
|
||||||
if (!shost)
|
if (!shost)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
read_lock_irq(&zfcp_data.config_lock);
|
read_lock_irq(&adapter->port_list_lock);
|
||||||
list_for_each_entry(port, &adapter->port_list_head, list)
|
list_for_each_entry(port, &adapter->port_list, list)
|
||||||
if (port->rport)
|
|
||||||
port->rport = NULL;
|
port->rport = NULL;
|
||||||
|
read_unlock_irq(&adapter->port_list_lock);
|
||||||
|
|
||||||
read_unlock_irq(&zfcp_data.config_lock);
|
|
||||||
fc_remove_host(shost);
|
fc_remove_host(shost);
|
||||||
scsi_remove_host(shost);
|
scsi_remove_host(shost);
|
||||||
scsi_host_put(shost);
|
scsi_host_put(shost);
|
||||||
|
@ -348,7 +360,7 @@ zfcp_init_fc_host_stats(struct zfcp_adapter *adapter)
|
||||||
fc_stats = kmalloc(sizeof(*fc_stats), GFP_KERNEL);
|
fc_stats = kmalloc(sizeof(*fc_stats), GFP_KERNEL);
|
||||||
if (!fc_stats)
|
if (!fc_stats)
|
||||||
return NULL;
|
return NULL;
|
||||||
adapter->fc_stats = fc_stats; /* freed in adater_dequeue */
|
adapter->fc_stats = fc_stats; /* freed in adapter_release */
|
||||||
}
|
}
|
||||||
memset(adapter->fc_stats, 0, sizeof(*adapter->fc_stats));
|
memset(adapter->fc_stats, 0, sizeof(*adapter->fc_stats));
|
||||||
return adapter->fc_stats;
|
return adapter->fc_stats;
|
||||||
|
@ -464,7 +476,7 @@ static void zfcp_reset_fc_host_stats(struct Scsi_Host *shost)
|
||||||
adapter->stats_reset = jiffies/HZ;
|
adapter->stats_reset = jiffies/HZ;
|
||||||
kfree(adapter->stats_reset_data);
|
kfree(adapter->stats_reset_data);
|
||||||
adapter->stats_reset_data = data; /* finally freed in
|
adapter->stats_reset_data = data; /* finally freed in
|
||||||
adapter_dequeue */
|
adapter_release */
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -495,7 +507,7 @@ static void zfcp_set_rport_dev_loss_tmo(struct fc_rport *rport, u32 timeout)
|
||||||
* @rport: The FC rport where to teminate I/O
|
* @rport: The FC rport where to teminate I/O
|
||||||
*
|
*
|
||||||
* Abort all pending SCSI commands for a port by closing the
|
* Abort all pending SCSI commands for a port by closing the
|
||||||
* port. Using a reopen for avoids a conflict with a shutdown
|
* port. Using a reopen avoiding a conflict with a shutdown
|
||||||
* overwriting a reopen.
|
* overwriting a reopen.
|
||||||
*/
|
*/
|
||||||
static void zfcp_scsi_terminate_rport_io(struct fc_rport *rport)
|
static void zfcp_scsi_terminate_rport_io(struct fc_rport *rport)
|
||||||
|
@ -505,15 +517,11 @@ static void zfcp_scsi_terminate_rport_io(struct fc_rport *rport)
|
||||||
struct zfcp_adapter *adapter =
|
struct zfcp_adapter *adapter =
|
||||||
(struct zfcp_adapter *)shost->hostdata[0];
|
(struct zfcp_adapter *)shost->hostdata[0];
|
||||||
|
|
||||||
write_lock_irq(&zfcp_data.config_lock);
|
|
||||||
port = zfcp_get_port_by_wwpn(adapter, rport->port_name);
|
port = zfcp_get_port_by_wwpn(adapter, rport->port_name);
|
||||||
if (port)
|
|
||||||
zfcp_port_get(port);
|
|
||||||
write_unlock_irq(&zfcp_data.config_lock);
|
|
||||||
|
|
||||||
if (port) {
|
if (port) {
|
||||||
zfcp_erp_port_reopen(port, 0, "sctrpi1", NULL);
|
zfcp_erp_port_reopen(port, 0, "sctrpi1", NULL);
|
||||||
zfcp_port_put(port);
|
put_device(&port->sysfs_device);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -555,31 +563,34 @@ static void zfcp_scsi_rport_block(struct zfcp_port *port)
|
||||||
|
|
||||||
void zfcp_scsi_schedule_rport_register(struct zfcp_port *port)
|
void zfcp_scsi_schedule_rport_register(struct zfcp_port *port)
|
||||||
{
|
{
|
||||||
zfcp_port_get(port);
|
get_device(&port->sysfs_device);
|
||||||
port->rport_task = RPORT_ADD;
|
port->rport_task = RPORT_ADD;
|
||||||
|
|
||||||
if (!queue_work(port->adapter->work_queue, &port->rport_work))
|
if (!queue_work(port->adapter->work_queue, &port->rport_work))
|
||||||
zfcp_port_put(port);
|
put_device(&port->sysfs_device);
|
||||||
}
|
}
|
||||||
|
|
||||||
void zfcp_scsi_schedule_rport_block(struct zfcp_port *port)
|
void zfcp_scsi_schedule_rport_block(struct zfcp_port *port)
|
||||||
{
|
{
|
||||||
zfcp_port_get(port);
|
get_device(&port->sysfs_device);
|
||||||
port->rport_task = RPORT_DEL;
|
port->rport_task = RPORT_DEL;
|
||||||
|
|
||||||
if (port->rport && queue_work(port->adapter->work_queue,
|
if (port->rport && queue_work(port->adapter->work_queue,
|
||||||
&port->rport_work))
|
&port->rport_work))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
zfcp_port_put(port);
|
put_device(&port->sysfs_device);
|
||||||
}
|
}
|
||||||
|
|
||||||
void zfcp_scsi_schedule_rports_block(struct zfcp_adapter *adapter)
|
void zfcp_scsi_schedule_rports_block(struct zfcp_adapter *adapter)
|
||||||
{
|
{
|
||||||
|
unsigned long flags;
|
||||||
struct zfcp_port *port;
|
struct zfcp_port *port;
|
||||||
|
|
||||||
list_for_each_entry(port, &adapter->port_list_head, list)
|
read_lock_irqsave(&adapter->port_list_lock, flags);
|
||||||
|
list_for_each_entry(port, &adapter->port_list, list)
|
||||||
zfcp_scsi_schedule_rport_block(port);
|
zfcp_scsi_schedule_rport_block(port);
|
||||||
|
read_unlock_irqrestore(&adapter->port_list_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
void zfcp_scsi_rport_work(struct work_struct *work)
|
void zfcp_scsi_rport_work(struct work_struct *work)
|
||||||
|
@ -597,7 +608,7 @@ void zfcp_scsi_rport_work(struct work_struct *work)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
zfcp_port_put(port);
|
put_device(&port->sysfs_device);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -615,21 +626,7 @@ void zfcp_scsi_scan(struct work_struct *work)
|
||||||
scsilun_to_int((struct scsi_lun *)
|
scsilun_to_int((struct scsi_lun *)
|
||||||
&unit->fcp_lun), 0);
|
&unit->fcp_lun), 0);
|
||||||
|
|
||||||
zfcp_unit_put(unit);
|
put_device(&unit->sysfs_device);
|
||||||
}
|
|
||||||
|
|
||||||
static int zfcp_execute_fc_job(struct fc_bsg_job *job)
|
|
||||||
{
|
|
||||||
switch (job->request->msgcode) {
|
|
||||||
case FC_BSG_RPT_ELS:
|
|
||||||
case FC_BSG_HST_ELS_NOLOGIN:
|
|
||||||
return zfcp_fc_execute_els_fc_job(job);
|
|
||||||
case FC_BSG_RPT_CT:
|
|
||||||
case FC_BSG_HST_CT:
|
|
||||||
return zfcp_fc_execute_ct_fc_job(job);
|
|
||||||
default:
|
|
||||||
return -EINVAL;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct fc_function_template zfcp_transport_functions = {
|
struct fc_function_template zfcp_transport_functions = {
|
||||||
|
@ -643,6 +640,7 @@ struct fc_function_template zfcp_transport_functions = {
|
||||||
.show_host_port_name = 1,
|
.show_host_port_name = 1,
|
||||||
.show_host_permanent_port_name = 1,
|
.show_host_permanent_port_name = 1,
|
||||||
.show_host_supported_classes = 1,
|
.show_host_supported_classes = 1,
|
||||||
|
.show_host_supported_fc4s = 1,
|
||||||
.show_host_supported_speeds = 1,
|
.show_host_supported_speeds = 1,
|
||||||
.show_host_maxframe_size = 1,
|
.show_host_maxframe_size = 1,
|
||||||
.show_host_serial_number = 1,
|
.show_host_serial_number = 1,
|
||||||
|
@ -652,13 +650,15 @@ struct fc_function_template zfcp_transport_functions = {
|
||||||
.get_host_port_state = zfcp_get_host_port_state,
|
.get_host_port_state = zfcp_get_host_port_state,
|
||||||
.terminate_rport_io = zfcp_scsi_terminate_rport_io,
|
.terminate_rport_io = zfcp_scsi_terminate_rport_io,
|
||||||
.show_host_port_state = 1,
|
.show_host_port_state = 1,
|
||||||
.bsg_request = zfcp_execute_fc_job,
|
.show_host_active_fc4s = 1,
|
||||||
|
.bsg_request = zfcp_fc_exec_bsg_job,
|
||||||
/* no functions registered for following dynamic attributes but
|
/* no functions registered for following dynamic attributes but
|
||||||
directly set by LLDD */
|
directly set by LLDD */
|
||||||
.show_host_port_type = 1,
|
.show_host_port_type = 1,
|
||||||
.show_host_speed = 1,
|
.show_host_speed = 1,
|
||||||
.show_host_port_id = 1,
|
.show_host_port_id = 1,
|
||||||
.disable_target_scan = 1,
|
.disable_target_scan = 1,
|
||||||
|
.dd_bsg_size = sizeof(struct zfcp_fsf_ct_els),
|
||||||
};
|
};
|
||||||
|
|
||||||
struct zfcp_data zfcp_data = {
|
struct zfcp_data zfcp_data = {
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
*
|
*
|
||||||
* sysfs attributes.
|
* sysfs attributes.
|
||||||
*
|
*
|
||||||
* Copyright IBM Corporation 2008
|
* Copyright IBM Corporation 2008, 2009
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define KMSG_COMPONENT "zfcp"
|
#define KMSG_COMPONENT "zfcp"
|
||||||
|
@ -19,29 +19,43 @@ static ssize_t zfcp_sysfs_##_feat##_##_name##_show(struct device *dev, \
|
||||||
struct device_attribute *at,\
|
struct device_attribute *at,\
|
||||||
char *buf) \
|
char *buf) \
|
||||||
{ \
|
{ \
|
||||||
struct _feat_def *_feat = dev_get_drvdata(dev); \
|
struct _feat_def *_feat = container_of(dev, struct _feat_def, \
|
||||||
|
sysfs_device); \
|
||||||
\
|
\
|
||||||
return sprintf(buf, _format, _value); \
|
return sprintf(buf, _format, _value); \
|
||||||
} \
|
} \
|
||||||
static ZFCP_DEV_ATTR(_feat, _name, S_IRUGO, \
|
static ZFCP_DEV_ATTR(_feat, _name, S_IRUGO, \
|
||||||
zfcp_sysfs_##_feat##_##_name##_show, NULL);
|
zfcp_sysfs_##_feat##_##_name##_show, NULL);
|
||||||
|
|
||||||
ZFCP_DEFINE_ATTR(zfcp_adapter, adapter, status, "0x%08x\n",
|
#define ZFCP_DEFINE_A_ATTR(_name, _format, _value) \
|
||||||
atomic_read(&adapter->status));
|
static ssize_t zfcp_sysfs_adapter_##_name##_show(struct device *dev, \
|
||||||
ZFCP_DEFINE_ATTR(zfcp_adapter, adapter, peer_wwnn, "0x%016llx\n",
|
struct device_attribute *at,\
|
||||||
|
char *buf) \
|
||||||
|
{ \
|
||||||
|
struct ccw_device *cdev = to_ccwdev(dev); \
|
||||||
|
struct zfcp_adapter *adapter = zfcp_ccw_adapter_by_cdev(cdev); \
|
||||||
|
int i; \
|
||||||
|
\
|
||||||
|
if (!adapter) \
|
||||||
|
return -ENODEV; \
|
||||||
|
\
|
||||||
|
i = sprintf(buf, _format, _value); \
|
||||||
|
zfcp_ccw_adapter_put(adapter); \
|
||||||
|
return i; \
|
||||||
|
} \
|
||||||
|
static ZFCP_DEV_ATTR(adapter, _name, S_IRUGO, \
|
||||||
|
zfcp_sysfs_adapter_##_name##_show, NULL);
|
||||||
|
|
||||||
|
ZFCP_DEFINE_A_ATTR(status, "0x%08x\n", atomic_read(&adapter->status));
|
||||||
|
ZFCP_DEFINE_A_ATTR(peer_wwnn, "0x%016llx\n",
|
||||||
(unsigned long long) adapter->peer_wwnn);
|
(unsigned long long) adapter->peer_wwnn);
|
||||||
ZFCP_DEFINE_ATTR(zfcp_adapter, adapter, peer_wwpn, "0x%016llx\n",
|
ZFCP_DEFINE_A_ATTR(peer_wwpn, "0x%016llx\n",
|
||||||
(unsigned long long) adapter->peer_wwpn);
|
(unsigned long long) adapter->peer_wwpn);
|
||||||
ZFCP_DEFINE_ATTR(zfcp_adapter, adapter, peer_d_id, "0x%06x\n",
|
ZFCP_DEFINE_A_ATTR(peer_d_id, "0x%06x\n", adapter->peer_d_id);
|
||||||
adapter->peer_d_id);
|
ZFCP_DEFINE_A_ATTR(card_version, "0x%04x\n", adapter->hydra_version);
|
||||||
ZFCP_DEFINE_ATTR(zfcp_adapter, adapter, card_version, "0x%04x\n",
|
ZFCP_DEFINE_A_ATTR(lic_version, "0x%08x\n", adapter->fsf_lic_version);
|
||||||
adapter->hydra_version);
|
ZFCP_DEFINE_A_ATTR(hardware_version, "0x%08x\n", adapter->hardware_version);
|
||||||
ZFCP_DEFINE_ATTR(zfcp_adapter, adapter, lic_version, "0x%08x\n",
|
ZFCP_DEFINE_A_ATTR(in_recovery, "%d\n", (atomic_read(&adapter->status) &
|
||||||
adapter->fsf_lic_version);
|
|
||||||
ZFCP_DEFINE_ATTR(zfcp_adapter, adapter, hardware_version, "0x%08x\n",
|
|
||||||
adapter->hardware_version);
|
|
||||||
ZFCP_DEFINE_ATTR(zfcp_adapter, adapter, in_recovery, "%d\n",
|
|
||||||
(atomic_read(&adapter->status) &
|
|
||||||
ZFCP_STATUS_COMMON_ERP_INUSE) != 0);
|
ZFCP_STATUS_COMMON_ERP_INUSE) != 0);
|
||||||
|
|
||||||
ZFCP_DEFINE_ATTR(zfcp_port, port, status, "0x%08x\n",
|
ZFCP_DEFINE_ATTR(zfcp_port, port, status, "0x%08x\n",
|
||||||
|
@ -73,7 +87,8 @@ static ssize_t zfcp_sysfs_##_feat##_failed_show(struct device *dev, \
|
||||||
struct device_attribute *attr, \
|
struct device_attribute *attr, \
|
||||||
char *buf) \
|
char *buf) \
|
||||||
{ \
|
{ \
|
||||||
struct _feat_def *_feat = dev_get_drvdata(dev); \
|
struct _feat_def *_feat = container_of(dev, struct _feat_def, \
|
||||||
|
sysfs_device); \
|
||||||
\
|
\
|
||||||
if (atomic_read(&_feat->status) & ZFCP_STATUS_COMMON_ERP_FAILED) \
|
if (atomic_read(&_feat->status) & ZFCP_STATUS_COMMON_ERP_FAILED) \
|
||||||
return sprintf(buf, "1\n"); \
|
return sprintf(buf, "1\n"); \
|
||||||
|
@ -84,15 +99,13 @@ static ssize_t zfcp_sysfs_##_feat##_failed_store(struct device *dev, \
|
||||||
struct device_attribute *attr,\
|
struct device_attribute *attr,\
|
||||||
const char *buf, size_t count)\
|
const char *buf, size_t count)\
|
||||||
{ \
|
{ \
|
||||||
struct _feat_def *_feat = dev_get_drvdata(dev); \
|
struct _feat_def *_feat = container_of(dev, struct _feat_def, \
|
||||||
|
sysfs_device); \
|
||||||
unsigned long val; \
|
unsigned long val; \
|
||||||
int retval = 0; \
|
int retval = 0; \
|
||||||
\
|
\
|
||||||
mutex_lock(&zfcp_data.config_mutex); \
|
if (!(_feat && get_device(&_feat->sysfs_device))) \
|
||||||
if (atomic_read(&_feat->status) & ZFCP_STATUS_COMMON_REMOVE) { \
|
return -EBUSY; \
|
||||||
retval = -EBUSY; \
|
|
||||||
goto out; \
|
|
||||||
} \
|
|
||||||
\
|
\
|
||||||
if (strict_strtoul(buf, 0, &val) || val != 0) { \
|
if (strict_strtoul(buf, 0, &val) || val != 0) { \
|
||||||
retval = -EINVAL; \
|
retval = -EINVAL; \
|
||||||
|
@ -105,29 +118,82 @@ static ssize_t zfcp_sysfs_##_feat##_failed_store(struct device *dev, \
|
||||||
_reopen_id, NULL); \
|
_reopen_id, NULL); \
|
||||||
zfcp_erp_wait(_adapter); \
|
zfcp_erp_wait(_adapter); \
|
||||||
out: \
|
out: \
|
||||||
mutex_unlock(&zfcp_data.config_mutex); \
|
put_device(&_feat->sysfs_device); \
|
||||||
return retval ? retval : (ssize_t) count; \
|
return retval ? retval : (ssize_t) count; \
|
||||||
} \
|
} \
|
||||||
static ZFCP_DEV_ATTR(_feat, failed, S_IWUSR | S_IRUGO, \
|
static ZFCP_DEV_ATTR(_feat, failed, S_IWUSR | S_IRUGO, \
|
||||||
zfcp_sysfs_##_feat##_failed_show, \
|
zfcp_sysfs_##_feat##_failed_show, \
|
||||||
zfcp_sysfs_##_feat##_failed_store);
|
zfcp_sysfs_##_feat##_failed_store);
|
||||||
|
|
||||||
ZFCP_SYSFS_FAILED(zfcp_adapter, adapter, adapter, "syafai1", "syafai2");
|
|
||||||
ZFCP_SYSFS_FAILED(zfcp_port, port, port->adapter, "sypfai1", "sypfai2");
|
ZFCP_SYSFS_FAILED(zfcp_port, port, port->adapter, "sypfai1", "sypfai2");
|
||||||
ZFCP_SYSFS_FAILED(zfcp_unit, unit, unit->port->adapter, "syufai1", "syufai2");
|
ZFCP_SYSFS_FAILED(zfcp_unit, unit, unit->port->adapter, "syufai1", "syufai2");
|
||||||
|
|
||||||
|
static ssize_t zfcp_sysfs_adapter_failed_show(struct device *dev,
|
||||||
|
struct device_attribute *attr,
|
||||||
|
char *buf)
|
||||||
|
{
|
||||||
|
struct ccw_device *cdev = to_ccwdev(dev);
|
||||||
|
struct zfcp_adapter *adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
|
int i;
|
||||||
|
|
||||||
|
if (!adapter)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
if (atomic_read(&adapter->status) & ZFCP_STATUS_COMMON_ERP_FAILED)
|
||||||
|
i = sprintf(buf, "1\n");
|
||||||
|
else
|
||||||
|
i = sprintf(buf, "0\n");
|
||||||
|
|
||||||
|
zfcp_ccw_adapter_put(adapter);
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t zfcp_sysfs_adapter_failed_store(struct device *dev,
|
||||||
|
struct device_attribute *attr,
|
||||||
|
const char *buf, size_t count)
|
||||||
|
{
|
||||||
|
struct ccw_device *cdev = to_ccwdev(dev);
|
||||||
|
struct zfcp_adapter *adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
|
unsigned long val;
|
||||||
|
int retval = 0;
|
||||||
|
|
||||||
|
if (!adapter)
|
||||||
|
return -ENODEV;
|
||||||
|
|
||||||
|
if (strict_strtoul(buf, 0, &val) || val != 0) {
|
||||||
|
retval = -EINVAL;
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
zfcp_erp_modify_adapter_status(adapter, "syafai1", NULL,
|
||||||
|
ZFCP_STATUS_COMMON_RUNNING, ZFCP_SET);
|
||||||
|
zfcp_erp_adapter_reopen(adapter, ZFCP_STATUS_COMMON_ERP_FAILED,
|
||||||
|
"syafai2", NULL);
|
||||||
|
zfcp_erp_wait(adapter);
|
||||||
|
out:
|
||||||
|
zfcp_ccw_adapter_put(adapter);
|
||||||
|
return retval ? retval : (ssize_t) count;
|
||||||
|
}
|
||||||
|
static ZFCP_DEV_ATTR(adapter, failed, S_IWUSR | S_IRUGO,
|
||||||
|
zfcp_sysfs_adapter_failed_show,
|
||||||
|
zfcp_sysfs_adapter_failed_store);
|
||||||
|
|
||||||
static ssize_t zfcp_sysfs_port_rescan_store(struct device *dev,
|
static ssize_t zfcp_sysfs_port_rescan_store(struct device *dev,
|
||||||
struct device_attribute *attr,
|
struct device_attribute *attr,
|
||||||
const char *buf, size_t count)
|
const char *buf, size_t count)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter = dev_get_drvdata(dev);
|
struct ccw_device *cdev = to_ccwdev(dev);
|
||||||
int ret;
|
struct zfcp_adapter *adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
|
|
||||||
if (atomic_read(&adapter->status) & ZFCP_STATUS_COMMON_REMOVE)
|
if (!adapter)
|
||||||
return -EBUSY;
|
return -ENODEV;
|
||||||
|
|
||||||
ret = zfcp_fc_scan_ports(adapter);
|
/* sync the user-space- with the kernel-invocation of scan_work */
|
||||||
return ret ? ret : (ssize_t) count;
|
queue_work(adapter->work_queue, &adapter->scan_work);
|
||||||
|
flush_work(&adapter->scan_work);
|
||||||
|
zfcp_ccw_adapter_put(adapter);
|
||||||
|
|
||||||
|
return (ssize_t) count;
|
||||||
}
|
}
|
||||||
static ZFCP_DEV_ATTR(adapter, port_rescan, S_IWUSR, NULL,
|
static ZFCP_DEV_ATTR(adapter, port_rescan, S_IWUSR, NULL,
|
||||||
zfcp_sysfs_port_rescan_store);
|
zfcp_sysfs_port_rescan_store);
|
||||||
|
@ -136,44 +202,34 @@ static ssize_t zfcp_sysfs_port_remove_store(struct device *dev,
|
||||||
struct device_attribute *attr,
|
struct device_attribute *attr,
|
||||||
const char *buf, size_t count)
|
const char *buf, size_t count)
|
||||||
{
|
{
|
||||||
struct zfcp_adapter *adapter = dev_get_drvdata(dev);
|
struct ccw_device *cdev = to_ccwdev(dev);
|
||||||
|
struct zfcp_adapter *adapter = zfcp_ccw_adapter_by_cdev(cdev);
|
||||||
struct zfcp_port *port;
|
struct zfcp_port *port;
|
||||||
u64 wwpn;
|
u64 wwpn;
|
||||||
int retval = 0;
|
int retval = -EINVAL;
|
||||||
LIST_HEAD(port_remove_lh);
|
|
||||||
|
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
if (!adapter)
|
||||||
if (atomic_read(&adapter->status) & ZFCP_STATUS_COMMON_REMOVE) {
|
return -ENODEV;
|
||||||
retval = -EBUSY;
|
|
||||||
|
if (strict_strtoull(buf, 0, (unsigned long long *) &wwpn))
|
||||||
goto out;
|
goto out;
|
||||||
}
|
|
||||||
|
|
||||||
if (strict_strtoull(buf, 0, (unsigned long long *) &wwpn)) {
|
|
||||||
retval = -EINVAL;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_lock_irq(&zfcp_data.config_lock);
|
|
||||||
port = zfcp_get_port_by_wwpn(adapter, wwpn);
|
port = zfcp_get_port_by_wwpn(adapter, wwpn);
|
||||||
if (port && (atomic_read(&port->refcount) == 0)) {
|
if (!port)
|
||||||
zfcp_port_get(port);
|
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_REMOVE, &port->status);
|
|
||||||
list_move(&port->list, &port_remove_lh);
|
|
||||||
} else
|
|
||||||
port = NULL;
|
|
||||||
write_unlock_irq(&zfcp_data.config_lock);
|
|
||||||
|
|
||||||
if (!port) {
|
|
||||||
retval = -ENXIO;
|
|
||||||
goto out;
|
goto out;
|
||||||
}
|
else
|
||||||
|
retval = 0;
|
||||||
|
|
||||||
|
write_lock_irq(&adapter->port_list_lock);
|
||||||
|
list_del(&port->list);
|
||||||
|
write_unlock_irq(&adapter->port_list_lock);
|
||||||
|
|
||||||
|
put_device(&port->sysfs_device);
|
||||||
|
|
||||||
zfcp_erp_port_shutdown(port, 0, "syprs_1", NULL);
|
zfcp_erp_port_shutdown(port, 0, "syprs_1", NULL);
|
||||||
zfcp_erp_wait(adapter);
|
zfcp_device_unregister(&port->sysfs_device, &zfcp_sysfs_port_attrs);
|
||||||
zfcp_port_put(port);
|
|
||||||
zfcp_port_dequeue(port);
|
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
zfcp_ccw_adapter_put(adapter);
|
||||||
return retval ? retval : (ssize_t) count;
|
return retval ? retval : (ssize_t) count;
|
||||||
}
|
}
|
||||||
static ZFCP_DEV_ATTR(adapter, port_remove, S_IWUSR, NULL,
|
static ZFCP_DEV_ATTR(adapter, port_remove, S_IWUSR, NULL,
|
||||||
|
@ -202,16 +258,14 @@ static ssize_t zfcp_sysfs_unit_add_store(struct device *dev,
|
||||||
struct device_attribute *attr,
|
struct device_attribute *attr,
|
||||||
const char *buf, size_t count)
|
const char *buf, size_t count)
|
||||||
{
|
{
|
||||||
struct zfcp_port *port = dev_get_drvdata(dev);
|
struct zfcp_port *port = container_of(dev, struct zfcp_port,
|
||||||
|
sysfs_device);
|
||||||
struct zfcp_unit *unit;
|
struct zfcp_unit *unit;
|
||||||
u64 fcp_lun;
|
u64 fcp_lun;
|
||||||
int retval = -EINVAL;
|
int retval = -EINVAL;
|
||||||
|
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
if (!(port && get_device(&port->sysfs_device)))
|
||||||
if (atomic_read(&port->status) & ZFCP_STATUS_COMMON_REMOVE) {
|
return -EBUSY;
|
||||||
retval = -EBUSY;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (strict_strtoull(buf, 0, (unsigned long long *) &fcp_lun))
|
if (strict_strtoull(buf, 0, (unsigned long long *) &fcp_lun))
|
||||||
goto out;
|
goto out;
|
||||||
|
@ -219,15 +273,14 @@ static ssize_t zfcp_sysfs_unit_add_store(struct device *dev,
|
||||||
unit = zfcp_unit_enqueue(port, fcp_lun);
|
unit = zfcp_unit_enqueue(port, fcp_lun);
|
||||||
if (IS_ERR(unit))
|
if (IS_ERR(unit))
|
||||||
goto out;
|
goto out;
|
||||||
|
else
|
||||||
retval = 0;
|
retval = 0;
|
||||||
|
|
||||||
zfcp_erp_unit_reopen(unit, 0, "syuas_1", NULL);
|
zfcp_erp_unit_reopen(unit, 0, "syuas_1", NULL);
|
||||||
zfcp_erp_wait(unit->port->adapter);
|
zfcp_erp_wait(unit->port->adapter);
|
||||||
flush_work(&unit->scsi_work);
|
flush_work(&unit->scsi_work);
|
||||||
zfcp_unit_put(unit);
|
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
put_device(&port->sysfs_device);
|
||||||
return retval ? retval : (ssize_t) count;
|
return retval ? retval : (ssize_t) count;
|
||||||
}
|
}
|
||||||
static DEVICE_ATTR(unit_add, S_IWUSR, NULL, zfcp_sysfs_unit_add_store);
|
static DEVICE_ATTR(unit_add, S_IWUSR, NULL, zfcp_sysfs_unit_add_store);
|
||||||
|
@ -236,54 +289,37 @@ static ssize_t zfcp_sysfs_unit_remove_store(struct device *dev,
|
||||||
struct device_attribute *attr,
|
struct device_attribute *attr,
|
||||||
const char *buf, size_t count)
|
const char *buf, size_t count)
|
||||||
{
|
{
|
||||||
struct zfcp_port *port = dev_get_drvdata(dev);
|
struct zfcp_port *port = container_of(dev, struct zfcp_port,
|
||||||
|
sysfs_device);
|
||||||
struct zfcp_unit *unit;
|
struct zfcp_unit *unit;
|
||||||
u64 fcp_lun;
|
u64 fcp_lun;
|
||||||
int retval = 0;
|
int retval = -EINVAL;
|
||||||
LIST_HEAD(unit_remove_lh);
|
|
||||||
|
|
||||||
mutex_lock(&zfcp_data.config_mutex);
|
if (!(port && get_device(&port->sysfs_device)))
|
||||||
if (atomic_read(&port->status) & ZFCP_STATUS_COMMON_REMOVE) {
|
return -EBUSY;
|
||||||
retval = -EBUSY;
|
|
||||||
|
if (strict_strtoull(buf, 0, (unsigned long long *) &fcp_lun))
|
||||||
goto out;
|
goto out;
|
||||||
}
|
|
||||||
|
|
||||||
if (strict_strtoull(buf, 0, (unsigned long long *) &fcp_lun)) {
|
|
||||||
retval = -EINVAL;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
write_lock_irq(&zfcp_data.config_lock);
|
|
||||||
unit = zfcp_get_unit_by_lun(port, fcp_lun);
|
unit = zfcp_get_unit_by_lun(port, fcp_lun);
|
||||||
if (unit) {
|
if (!unit)
|
||||||
write_unlock_irq(&zfcp_data.config_lock);
|
goto out;
|
||||||
|
else
|
||||||
|
retval = 0;
|
||||||
|
|
||||||
/* wait for possible timeout during SCSI probe */
|
/* wait for possible timeout during SCSI probe */
|
||||||
flush_work(&unit->scsi_work);
|
flush_work(&unit->scsi_work);
|
||||||
write_lock_irq(&zfcp_data.config_lock);
|
|
||||||
|
|
||||||
if (atomic_read(&unit->refcount) == 0) {
|
write_lock_irq(&port->unit_list_lock);
|
||||||
zfcp_unit_get(unit);
|
list_del(&unit->list);
|
||||||
atomic_set_mask(ZFCP_STATUS_COMMON_REMOVE,
|
write_unlock_irq(&port->unit_list_lock);
|
||||||
&unit->status);
|
|
||||||
list_move(&unit->list, &unit_remove_lh);
|
|
||||||
} else {
|
|
||||||
unit = NULL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
write_unlock_irq(&zfcp_data.config_lock);
|
put_device(&unit->sysfs_device);
|
||||||
|
|
||||||
if (!unit) {
|
|
||||||
retval = -ENXIO;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
zfcp_erp_unit_shutdown(unit, 0, "syurs_1", NULL);
|
zfcp_erp_unit_shutdown(unit, 0, "syurs_1", NULL);
|
||||||
zfcp_erp_wait(unit->port->adapter);
|
zfcp_device_unregister(&unit->sysfs_device, &zfcp_sysfs_unit_attrs);
|
||||||
zfcp_unit_put(unit);
|
|
||||||
zfcp_unit_dequeue(unit);
|
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&zfcp_data.config_mutex);
|
put_device(&port->sysfs_device);
|
||||||
return retval ? retval : (ssize_t) count;
|
return retval ? retval : (ssize_t) count;
|
||||||
}
|
}
|
||||||
static DEVICE_ATTR(unit_remove, S_IWUSR, NULL, zfcp_sysfs_unit_remove_store);
|
static DEVICE_ATTR(unit_remove, S_IWUSR, NULL, zfcp_sysfs_unit_remove_store);
|
||||||
|
|
|
@ -186,8 +186,12 @@ static ssize_t twa_show_stats(struct device *dev,
|
||||||
} /* End twa_show_stats() */
|
} /* End twa_show_stats() */
|
||||||
|
|
||||||
/* This function will set a devices queue depth */
|
/* This function will set a devices queue depth */
|
||||||
static int twa_change_queue_depth(struct scsi_device *sdev, int queue_depth)
|
static int twa_change_queue_depth(struct scsi_device *sdev, int queue_depth,
|
||||||
|
int reason)
|
||||||
{
|
{
|
||||||
|
if (reason != SCSI_QDEPTH_DEFAULT)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (queue_depth > TW_Q_LENGTH-2)
|
if (queue_depth > TW_Q_LENGTH-2)
|
||||||
queue_depth = TW_Q_LENGTH-2;
|
queue_depth = TW_Q_LENGTH-2;
|
||||||
scsi_adjust_queue_depth(sdev, MSG_ORDERED_TAG, queue_depth);
|
scsi_adjust_queue_depth(sdev, MSG_ORDERED_TAG, queue_depth);
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,396 @@
|
||||||
|
/*
|
||||||
|
3w-sas.h -- LSI 3ware SAS/SATA-RAID Controller device driver for Linux.
|
||||||
|
|
||||||
|
Written By: Adam Radford <linuxraid@lsi.com>
|
||||||
|
|
||||||
|
Copyright (C) 2009 LSI Corporation.
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
NO WARRANTY
|
||||||
|
THE PROGRAM IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR
|
||||||
|
CONDITIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED INCLUDING, WITHOUT
|
||||||
|
LIMITATION, ANY WARRANTIES OR CONDITIONS OF TITLE, NON-INFRINGEMENT,
|
||||||
|
MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. Each Recipient is
|
||||||
|
solely responsible for determining the appropriateness of using and
|
||||||
|
distributing the Program and assumes all risks associated with its
|
||||||
|
exercise of rights under this Agreement, including but not limited to
|
||||||
|
the risks and costs of program errors, damage to or loss of data,
|
||||||
|
programs or equipment, and unavailability or interruption of operations.
|
||||||
|
|
||||||
|
DISCLAIMER OF LIABILITY
|
||||||
|
NEITHER RECIPIENT NOR ANY CONTRIBUTORS SHALL HAVE ANY LIABILITY FOR ANY
|
||||||
|
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
DAMAGES (INCLUDING WITHOUT LIMITATION LOST PROFITS), HOWEVER CAUSED AND
|
||||||
|
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
|
||||||
|
TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
|
||||||
|
USE OR DISTRIBUTION OF THE PROGRAM OR THE EXERCISE OF ANY RIGHTS GRANTED
|
||||||
|
HEREUNDER, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGES
|
||||||
|
|
||||||
|
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
|
||||||
|
Bugs/Comments/Suggestions should be mailed to:
|
||||||
|
linuxraid@lsi.com
|
||||||
|
|
||||||
|
For more information, goto:
|
||||||
|
http://www.lsi.com
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _3W_SAS_H
|
||||||
|
#define _3W_SAS_H
|
||||||
|
|
||||||
|
/* AEN severity table */
|
||||||
|
static char *twl_aen_severity_table[] =
|
||||||
|
{
|
||||||
|
"None", "ERROR", "WARNING", "INFO", "DEBUG", NULL
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Liberator register offsets */
|
||||||
|
#define TWL_STATUS 0x0 /* Status */
|
||||||
|
#define TWL_HIBDB 0x20 /* Inbound doorbell */
|
||||||
|
#define TWL_HISTAT 0x30 /* Host interrupt status */
|
||||||
|
#define TWL_HIMASK 0x34 /* Host interrupt mask */
|
||||||
|
#define TWL_HOBDB 0x9C /* Outbound doorbell */
|
||||||
|
#define TWL_HOBDBC 0xA0 /* Outbound doorbell clear */
|
||||||
|
#define TWL_SCRPD3 0xBC /* Scratchpad */
|
||||||
|
#define TWL_HIBQPL 0xC0 /* Host inbound Q low */
|
||||||
|
#define TWL_HIBQPH 0xC4 /* Host inbound Q high */
|
||||||
|
#define TWL_HOBQPL 0xC8 /* Host outbound Q low */
|
||||||
|
#define TWL_HOBQPH 0xCC /* Host outbound Q high */
|
||||||
|
#define TWL_HISTATUS_VALID_INTERRUPT 0xC
|
||||||
|
#define TWL_HISTATUS_ATTENTION_INTERRUPT 0x4
|
||||||
|
#define TWL_HISTATUS_RESPONSE_INTERRUPT 0x8
|
||||||
|
#define TWL_STATUS_OVERRUN_SUBMIT 0x2000
|
||||||
|
#define TWL_ISSUE_SOFT_RESET 0x100
|
||||||
|
#define TWL_CONTROLLER_READY 0x2000
|
||||||
|
#define TWL_DOORBELL_CONTROLLER_ERROR 0x200000
|
||||||
|
#define TWL_DOORBELL_ATTENTION_INTERRUPT 0x40000
|
||||||
|
#define TWL_PULL_MODE 0x1
|
||||||
|
|
||||||
|
/* Command packet opcodes used by the driver */
|
||||||
|
#define TW_OP_INIT_CONNECTION 0x1
|
||||||
|
#define TW_OP_GET_PARAM 0x12
|
||||||
|
#define TW_OP_SET_PARAM 0x13
|
||||||
|
#define TW_OP_EXECUTE_SCSI 0x10
|
||||||
|
|
||||||
|
/* Asynchronous Event Notification (AEN) codes used by the driver */
|
||||||
|
#define TW_AEN_QUEUE_EMPTY 0x0000
|
||||||
|
#define TW_AEN_SOFT_RESET 0x0001
|
||||||
|
#define TW_AEN_SYNC_TIME_WITH_HOST 0x031
|
||||||
|
#define TW_AEN_SEVERITY_ERROR 0x1
|
||||||
|
#define TW_AEN_SEVERITY_DEBUG 0x4
|
||||||
|
#define TW_AEN_NOT_RETRIEVED 0x1
|
||||||
|
|
||||||
|
/* Command state defines */
|
||||||
|
#define TW_S_INITIAL 0x1 /* Initial state */
|
||||||
|
#define TW_S_STARTED 0x2 /* Id in use */
|
||||||
|
#define TW_S_POSTED 0x4 /* Posted to the controller */
|
||||||
|
#define TW_S_COMPLETED 0x8 /* Completed by isr */
|
||||||
|
#define TW_S_FINISHED 0x10 /* I/O completely done */
|
||||||
|
|
||||||
|
/* Compatibility defines */
|
||||||
|
#define TW_9750_ARCH_ID 10
|
||||||
|
#define TW_CURRENT_DRIVER_SRL 40
|
||||||
|
#define TW_CURRENT_DRIVER_BUILD 0
|
||||||
|
#define TW_CURRENT_DRIVER_BRANCH 0
|
||||||
|
|
||||||
|
/* Phase defines */
|
||||||
|
#define TW_PHASE_INITIAL 0
|
||||||
|
#define TW_PHASE_SGLIST 2
|
||||||
|
|
||||||
|
/* Misc defines */
|
||||||
|
#define TW_SECTOR_SIZE 512
|
||||||
|
#define TW_MAX_UNITS 32
|
||||||
|
#define TW_INIT_MESSAGE_CREDITS 0x100
|
||||||
|
#define TW_INIT_COMMAND_PACKET_SIZE 0x3
|
||||||
|
#define TW_INIT_COMMAND_PACKET_SIZE_EXTENDED 0x6
|
||||||
|
#define TW_EXTENDED_INIT_CONNECT 0x2
|
||||||
|
#define TW_BASE_FW_SRL 24
|
||||||
|
#define TW_BASE_FW_BRANCH 0
|
||||||
|
#define TW_BASE_FW_BUILD 1
|
||||||
|
#define TW_Q_LENGTH 256
|
||||||
|
#define TW_Q_START 0
|
||||||
|
#define TW_MAX_SLOT 32
|
||||||
|
#define TW_MAX_RESET_TRIES 2
|
||||||
|
#define TW_MAX_CMDS_PER_LUN 254
|
||||||
|
#define TW_MAX_AEN_DRAIN 255
|
||||||
|
#define TW_IN_RESET 2
|
||||||
|
#define TW_USING_MSI 3
|
||||||
|
#define TW_IN_ATTENTION_LOOP 4
|
||||||
|
#define TW_MAX_SECTORS 256
|
||||||
|
#define TW_MAX_CDB_LEN 16
|
||||||
|
#define TW_IOCTL_CHRDEV_TIMEOUT 60 /* 60 seconds */
|
||||||
|
#define TW_IOCTL_CHRDEV_FREE -1
|
||||||
|
#define TW_COMMAND_OFFSET 128 /* 128 bytes */
|
||||||
|
#define TW_VERSION_TABLE 0x0402
|
||||||
|
#define TW_TIMEKEEP_TABLE 0x040A
|
||||||
|
#define TW_INFORMATION_TABLE 0x0403
|
||||||
|
#define TW_PARAM_FWVER 3
|
||||||
|
#define TW_PARAM_FWVER_LENGTH 16
|
||||||
|
#define TW_PARAM_BIOSVER 4
|
||||||
|
#define TW_PARAM_BIOSVER_LENGTH 16
|
||||||
|
#define TW_PARAM_MODEL 8
|
||||||
|
#define TW_PARAM_MODEL_LENGTH 16
|
||||||
|
#define TW_PARAM_PHY_SUMMARY_TABLE 1
|
||||||
|
#define TW_PARAM_PHYCOUNT 2
|
||||||
|
#define TW_PARAM_PHYCOUNT_LENGTH 1
|
||||||
|
#define TW_IOCTL_FIRMWARE_PASS_THROUGH 0x108 // Used by smartmontools
|
||||||
|
#define TW_ALLOCATION_LENGTH 128
|
||||||
|
#define TW_SENSE_DATA_LENGTH 18
|
||||||
|
#define TW_ERROR_LOGICAL_UNIT_NOT_SUPPORTED 0x10a
|
||||||
|
#define TW_ERROR_INVALID_FIELD_IN_CDB 0x10d
|
||||||
|
#define TW_ERROR_UNIT_OFFLINE 0x128
|
||||||
|
#define TW_MESSAGE_SOURCE_CONTROLLER_ERROR 3
|
||||||
|
#define TW_MESSAGE_SOURCE_CONTROLLER_EVENT 4
|
||||||
|
#define TW_DRIVER 6
|
||||||
|
#ifndef PCI_DEVICE_ID_3WARE_9750
|
||||||
|
#define PCI_DEVICE_ID_3WARE_9750 0x1010
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Bitmask macros to eliminate bitfields */
|
||||||
|
|
||||||
|
/* opcode: 5, reserved: 3 */
|
||||||
|
#define TW_OPRES_IN(x,y) ((x << 5) | (y & 0x1f))
|
||||||
|
#define TW_OP_OUT(x) (x & 0x1f)
|
||||||
|
|
||||||
|
/* opcode: 5, sgloffset: 3 */
|
||||||
|
#define TW_OPSGL_IN(x,y) ((x << 5) | (y & 0x1f))
|
||||||
|
#define TW_SGL_OUT(x) ((x >> 5) & 0x7)
|
||||||
|
|
||||||
|
/* severity: 3, reserved: 5 */
|
||||||
|
#define TW_SEV_OUT(x) (x & 0x7)
|
||||||
|
|
||||||
|
/* not_mfa: 1, reserved: 7, status: 8, request_id: 16 */
|
||||||
|
#define TW_RESID_OUT(x) ((x >> 16) & 0xffff)
|
||||||
|
#define TW_NOTMFA_OUT(x) (x & 0x1)
|
||||||
|
|
||||||
|
/* request_id: 12, lun: 4 */
|
||||||
|
#define TW_REQ_LUN_IN(lun, request_id) (((lun << 12) & 0xf000) | (request_id & 0xfff))
|
||||||
|
#define TW_LUN_OUT(lun) ((lun >> 12) & 0xf)
|
||||||
|
|
||||||
|
/* Register access macros */
|
||||||
|
#define TWL_STATUS_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_STATUS)
|
||||||
|
#define TWL_HOBQPL_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_HOBQPL)
|
||||||
|
#define TWL_HOBQPH_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_HOBQPH)
|
||||||
|
#define TWL_HOBDB_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_HOBDB)
|
||||||
|
#define TWL_HOBDBC_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_HOBDBC)
|
||||||
|
#define TWL_HIMASK_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_HIMASK)
|
||||||
|
#define TWL_HISTAT_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_HISTAT)
|
||||||
|
#define TWL_HIBQPH_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_HIBQPH)
|
||||||
|
#define TWL_HIBQPL_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_HIBQPL)
|
||||||
|
#define TWL_HIBDB_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_HIBDB)
|
||||||
|
#define TWL_SCRPD3_REG_ADDR(x) ((unsigned char __iomem *)x->base_addr + TWL_SCRPD3)
|
||||||
|
#define TWL_MASK_INTERRUPTS(x) (writel(~0, TWL_HIMASK_REG_ADDR(tw_dev)))
|
||||||
|
#define TWL_UNMASK_INTERRUPTS(x) (writel(~TWL_HISTATUS_VALID_INTERRUPT, TWL_HIMASK_REG_ADDR(tw_dev)))
|
||||||
|
#define TWL_CLEAR_DB_INTERRUPT(x) (writel(~0, TWL_HOBDBC_REG_ADDR(tw_dev)))
|
||||||
|
#define TWL_SOFT_RESET(x) (writel(TWL_ISSUE_SOFT_RESET, TWL_HIBDB_REG_ADDR(tw_dev)))
|
||||||
|
|
||||||
|
/* Macros */
|
||||||
|
#define TW_PRINTK(h,a,b,c) { \
|
||||||
|
if (h) \
|
||||||
|
printk(KERN_WARNING "3w-sas: scsi%d: ERROR: (0x%02X:0x%04X): %s.\n",h->host_no,a,b,c); \
|
||||||
|
else \
|
||||||
|
printk(KERN_WARNING "3w-sas: ERROR: (0x%02X:0x%04X): %s.\n",a,b,c); \
|
||||||
|
}
|
||||||
|
#define TW_MAX_LUNS 16
|
||||||
|
#define TW_COMMAND_SIZE (sizeof(dma_addr_t) > 4 ? 6 : 4)
|
||||||
|
#define TW_LIBERATOR_MAX_SGL_LENGTH (sizeof(dma_addr_t) > 4 ? 46 : 92)
|
||||||
|
#define TW_LIBERATOR_MAX_SGL_LENGTH_OLD (sizeof(dma_addr_t) > 4 ? 47 : 94)
|
||||||
|
#define TW_PADDING_LENGTH_LIBERATOR 136
|
||||||
|
#define TW_PADDING_LENGTH_LIBERATOR_OLD 132
|
||||||
|
#define TW_CPU_TO_SGL(x) (sizeof(dma_addr_t) > 4 ? cpu_to_le64(x) : cpu_to_le32(x))
|
||||||
|
|
||||||
|
#pragma pack(1)
|
||||||
|
|
||||||
|
/* SGL entry */
|
||||||
|
typedef struct TAG_TW_SG_Entry_ISO {
|
||||||
|
dma_addr_t address;
|
||||||
|
dma_addr_t length;
|
||||||
|
} TW_SG_Entry_ISO;
|
||||||
|
|
||||||
|
/* Old Command Packet with ISO SGL */
|
||||||
|
typedef struct TW_Command {
|
||||||
|
unsigned char opcode__sgloffset;
|
||||||
|
unsigned char size;
|
||||||
|
unsigned char request_id;
|
||||||
|
unsigned char unit__hostid;
|
||||||
|
/* Second DWORD */
|
||||||
|
unsigned char status;
|
||||||
|
unsigned char flags;
|
||||||
|
union {
|
||||||
|
unsigned short block_count;
|
||||||
|
unsigned short parameter_count;
|
||||||
|
} byte6_offset;
|
||||||
|
union {
|
||||||
|
struct {
|
||||||
|
u32 lba;
|
||||||
|
TW_SG_Entry_ISO sgl[TW_LIBERATOR_MAX_SGL_LENGTH_OLD];
|
||||||
|
unsigned char padding[TW_PADDING_LENGTH_LIBERATOR_OLD];
|
||||||
|
} io;
|
||||||
|
struct {
|
||||||
|
TW_SG_Entry_ISO sgl[TW_LIBERATOR_MAX_SGL_LENGTH_OLD];
|
||||||
|
u32 padding;
|
||||||
|
unsigned char padding2[TW_PADDING_LENGTH_LIBERATOR_OLD];
|
||||||
|
} param;
|
||||||
|
} byte8_offset;
|
||||||
|
} TW_Command;
|
||||||
|
|
||||||
|
/* New Command Packet with ISO SGL */
|
||||||
|
typedef struct TAG_TW_Command_Apache {
|
||||||
|
unsigned char opcode__reserved;
|
||||||
|
unsigned char unit;
|
||||||
|
unsigned short request_id__lunl;
|
||||||
|
unsigned char status;
|
||||||
|
unsigned char sgl_offset;
|
||||||
|
unsigned short sgl_entries__lunh;
|
||||||
|
unsigned char cdb[16];
|
||||||
|
TW_SG_Entry_ISO sg_list[TW_LIBERATOR_MAX_SGL_LENGTH];
|
||||||
|
unsigned char padding[TW_PADDING_LENGTH_LIBERATOR];
|
||||||
|
} TW_Command_Apache;
|
||||||
|
|
||||||
|
/* New command packet header */
|
||||||
|
typedef struct TAG_TW_Command_Apache_Header {
|
||||||
|
unsigned char sense_data[TW_SENSE_DATA_LENGTH];
|
||||||
|
struct {
|
||||||
|
char reserved[4];
|
||||||
|
unsigned short error;
|
||||||
|
unsigned char padding;
|
||||||
|
unsigned char severity__reserved;
|
||||||
|
} status_block;
|
||||||
|
unsigned char err_specific_desc[98];
|
||||||
|
struct {
|
||||||
|
unsigned char size_header;
|
||||||
|
unsigned short request_id;
|
||||||
|
unsigned char size_sense;
|
||||||
|
} header_desc;
|
||||||
|
} TW_Command_Apache_Header;
|
||||||
|
|
||||||
|
/* This struct is a union of the 2 command packets */
|
||||||
|
typedef struct TAG_TW_Command_Full {
|
||||||
|
TW_Command_Apache_Header header;
|
||||||
|
union {
|
||||||
|
TW_Command oldcommand;
|
||||||
|
TW_Command_Apache newcommand;
|
||||||
|
} command;
|
||||||
|
} TW_Command_Full;
|
||||||
|
|
||||||
|
/* Initconnection structure */
|
||||||
|
typedef struct TAG_TW_Initconnect {
|
||||||
|
unsigned char opcode__reserved;
|
||||||
|
unsigned char size;
|
||||||
|
unsigned char request_id;
|
||||||
|
unsigned char res2;
|
||||||
|
unsigned char status;
|
||||||
|
unsigned char flags;
|
||||||
|
unsigned short message_credits;
|
||||||
|
u32 features;
|
||||||
|
unsigned short fw_srl;
|
||||||
|
unsigned short fw_arch_id;
|
||||||
|
unsigned short fw_branch;
|
||||||
|
unsigned short fw_build;
|
||||||
|
u32 result;
|
||||||
|
} TW_Initconnect;
|
||||||
|
|
||||||
|
/* Event info structure */
|
||||||
|
typedef struct TAG_TW_Event
|
||||||
|
{
|
||||||
|
unsigned int sequence_id;
|
||||||
|
unsigned int time_stamp_sec;
|
||||||
|
unsigned short aen_code;
|
||||||
|
unsigned char severity;
|
||||||
|
unsigned char retrieved;
|
||||||
|
unsigned char repeat_count;
|
||||||
|
unsigned char parameter_len;
|
||||||
|
unsigned char parameter_data[98];
|
||||||
|
} TW_Event;
|
||||||
|
|
||||||
|
typedef struct TAG_TW_Ioctl_Driver_Command {
|
||||||
|
unsigned int control_code;
|
||||||
|
unsigned int status;
|
||||||
|
unsigned int unique_id;
|
||||||
|
unsigned int sequence_id;
|
||||||
|
unsigned int os_specific;
|
||||||
|
unsigned int buffer_length;
|
||||||
|
} TW_Ioctl_Driver_Command;
|
||||||
|
|
||||||
|
typedef struct TAG_TW_Ioctl_Apache {
|
||||||
|
TW_Ioctl_Driver_Command driver_command;
|
||||||
|
char padding[488];
|
||||||
|
TW_Command_Full firmware_command;
|
||||||
|
char data_buffer[1];
|
||||||
|
} TW_Ioctl_Buf_Apache;
|
||||||
|
|
||||||
|
/* GetParam descriptor */
|
||||||
|
typedef struct {
|
||||||
|
unsigned short table_id;
|
||||||
|
unsigned short parameter_id;
|
||||||
|
unsigned short parameter_size_bytes;
|
||||||
|
unsigned short actual_parameter_size_bytes;
|
||||||
|
unsigned char data[1];
|
||||||
|
} TW_Param_Apache;
|
||||||
|
|
||||||
|
/* Compatibility information structure */
|
||||||
|
typedef struct TAG_TW_Compatibility_Info
|
||||||
|
{
|
||||||
|
char driver_version[32];
|
||||||
|
unsigned short working_srl;
|
||||||
|
unsigned short working_branch;
|
||||||
|
unsigned short working_build;
|
||||||
|
unsigned short driver_srl_high;
|
||||||
|
unsigned short driver_branch_high;
|
||||||
|
unsigned short driver_build_high;
|
||||||
|
unsigned short driver_srl_low;
|
||||||
|
unsigned short driver_branch_low;
|
||||||
|
unsigned short driver_build_low;
|
||||||
|
unsigned short fw_on_ctlr_srl;
|
||||||
|
unsigned short fw_on_ctlr_branch;
|
||||||
|
unsigned short fw_on_ctlr_build;
|
||||||
|
} TW_Compatibility_Info;
|
||||||
|
|
||||||
|
#pragma pack()
|
||||||
|
|
||||||
|
typedef struct TAG_TW_Device_Extension {
|
||||||
|
void __iomem *base_addr;
|
||||||
|
unsigned long *generic_buffer_virt[TW_Q_LENGTH];
|
||||||
|
dma_addr_t generic_buffer_phys[TW_Q_LENGTH];
|
||||||
|
TW_Command_Full *command_packet_virt[TW_Q_LENGTH];
|
||||||
|
dma_addr_t command_packet_phys[TW_Q_LENGTH];
|
||||||
|
TW_Command_Apache_Header *sense_buffer_virt[TW_Q_LENGTH];
|
||||||
|
dma_addr_t sense_buffer_phys[TW_Q_LENGTH];
|
||||||
|
struct pci_dev *tw_pci_dev;
|
||||||
|
struct scsi_cmnd *srb[TW_Q_LENGTH];
|
||||||
|
unsigned char free_queue[TW_Q_LENGTH];
|
||||||
|
unsigned char free_head;
|
||||||
|
unsigned char free_tail;
|
||||||
|
int state[TW_Q_LENGTH];
|
||||||
|
unsigned int posted_request_count;
|
||||||
|
unsigned int max_posted_request_count;
|
||||||
|
unsigned int max_sgl_entries;
|
||||||
|
unsigned int sgl_entries;
|
||||||
|
unsigned int num_resets;
|
||||||
|
unsigned int sector_count;
|
||||||
|
unsigned int max_sector_count;
|
||||||
|
unsigned int aen_count;
|
||||||
|
struct Scsi_Host *host;
|
||||||
|
long flags;
|
||||||
|
TW_Event *event_queue[TW_Q_LENGTH];
|
||||||
|
unsigned char error_index;
|
||||||
|
unsigned int error_sequence_id;
|
||||||
|
int chrdev_request_id;
|
||||||
|
wait_queue_head_t ioctl_wqueue;
|
||||||
|
struct mutex ioctl_lock;
|
||||||
|
TW_Compatibility_Info tw_compat_info;
|
||||||
|
char online;
|
||||||
|
} TW_Device_Extension;
|
||||||
|
|
||||||
|
#endif /* _3W_SAS_H */
|
||||||
|
|
|
@ -521,8 +521,12 @@ static ssize_t tw_show_stats(struct device *dev, struct device_attribute *attr,
|
||||||
} /* End tw_show_stats() */
|
} /* End tw_show_stats() */
|
||||||
|
|
||||||
/* This function will set a devices queue depth */
|
/* This function will set a devices queue depth */
|
||||||
static int tw_change_queue_depth(struct scsi_device *sdev, int queue_depth)
|
static int tw_change_queue_depth(struct scsi_device *sdev, int queue_depth,
|
||||||
|
int reason)
|
||||||
{
|
{
|
||||||
|
if (reason != SCSI_QDEPTH_DEFAULT)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (queue_depth > TW_Q_LENGTH-2)
|
if (queue_depth > TW_Q_LENGTH-2)
|
||||||
queue_depth = TW_Q_LENGTH-2;
|
queue_depth = TW_Q_LENGTH-2;
|
||||||
scsi_adjust_queue_depth(sdev, MSG_ORDERED_TAG, queue_depth);
|
scsi_adjust_queue_depth(sdev, MSG_ORDERED_TAG, queue_depth);
|
||||||
|
|
|
@ -175,7 +175,7 @@ STATIC void NCR_700_chip_reset(struct Scsi_Host *host);
|
||||||
STATIC int NCR_700_slave_alloc(struct scsi_device *SDpnt);
|
STATIC int NCR_700_slave_alloc(struct scsi_device *SDpnt);
|
||||||
STATIC int NCR_700_slave_configure(struct scsi_device *SDpnt);
|
STATIC int NCR_700_slave_configure(struct scsi_device *SDpnt);
|
||||||
STATIC void NCR_700_slave_destroy(struct scsi_device *SDpnt);
|
STATIC void NCR_700_slave_destroy(struct scsi_device *SDpnt);
|
||||||
static int NCR_700_change_queue_depth(struct scsi_device *SDpnt, int depth);
|
static int NCR_700_change_queue_depth(struct scsi_device *SDpnt, int depth, int reason);
|
||||||
static int NCR_700_change_queue_type(struct scsi_device *SDpnt, int depth);
|
static int NCR_700_change_queue_type(struct scsi_device *SDpnt, int depth);
|
||||||
|
|
||||||
STATIC struct device_attribute *NCR_700_dev_attrs[];
|
STATIC struct device_attribute *NCR_700_dev_attrs[];
|
||||||
|
@ -2082,8 +2082,11 @@ NCR_700_slave_destroy(struct scsi_device *SDp)
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
NCR_700_change_queue_depth(struct scsi_device *SDp, int depth)
|
NCR_700_change_queue_depth(struct scsi_device *SDp, int depth, int reason)
|
||||||
{
|
{
|
||||||
|
if (reason != SCSI_QDEPTH_DEFAULT)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (depth > NCR_700_MAX_TAGS)
|
if (depth > NCR_700_MAX_TAGS)
|
||||||
depth = NCR_700_MAX_TAGS;
|
depth = NCR_700_MAX_TAGS;
|
||||||
|
|
||||||
|
|
|
@ -399,6 +399,17 @@ config SCSI_3W_9XXX
|
||||||
Please read the comments at the top of
|
Please read the comments at the top of
|
||||||
<file:drivers/scsi/3w-9xxx.c>.
|
<file:drivers/scsi/3w-9xxx.c>.
|
||||||
|
|
||||||
|
config SCSI_3W_SAS
|
||||||
|
tristate "3ware 97xx SAS/SATA-RAID support"
|
||||||
|
depends on PCI && SCSI
|
||||||
|
help
|
||||||
|
This driver supports the LSI 3ware 9750 6Gb/s SAS/SATA-RAID cards.
|
||||||
|
|
||||||
|
<http://www.lsi.com>
|
||||||
|
|
||||||
|
Please read the comments at the top of
|
||||||
|
<file:drivers/scsi/3w-sas.c>.
|
||||||
|
|
||||||
config SCSI_7000FASST
|
config SCSI_7000FASST
|
||||||
tristate "7000FASST SCSI support"
|
tristate "7000FASST SCSI support"
|
||||||
depends on ISA && SCSI && ISA_DMA_API
|
depends on ISA && SCSI && ISA_DMA_API
|
||||||
|
@ -621,6 +632,14 @@ config SCSI_FLASHPOINT
|
||||||
substantial, so users of MultiMaster Host Adapters may not
|
substantial, so users of MultiMaster Host Adapters may not
|
||||||
wish to include it.
|
wish to include it.
|
||||||
|
|
||||||
|
config VMWARE_PVSCSI
|
||||||
|
tristate "VMware PVSCSI driver support"
|
||||||
|
depends on PCI && SCSI && X86
|
||||||
|
help
|
||||||
|
This driver supports VMware's para virtualized SCSI HBA.
|
||||||
|
To compile this driver as a module, choose M here: the
|
||||||
|
module will be called vmw_pvscsi.
|
||||||
|
|
||||||
config LIBFC
|
config LIBFC
|
||||||
tristate "LibFC module"
|
tristate "LibFC module"
|
||||||
select SCSI_FC_ATTRS
|
select SCSI_FC_ATTRS
|
||||||
|
@ -644,7 +663,7 @@ config FCOE
|
||||||
config FCOE_FNIC
|
config FCOE_FNIC
|
||||||
tristate "Cisco FNIC Driver"
|
tristate "Cisco FNIC Driver"
|
||||||
depends on PCI && X86
|
depends on PCI && X86
|
||||||
select LIBFC
|
select LIBFCOE
|
||||||
help
|
help
|
||||||
This is support for the Cisco PCI-Express FCoE HBA.
|
This is support for the Cisco PCI-Express FCoE HBA.
|
||||||
|
|
||||||
|
@ -1818,6 +1837,14 @@ config SCSI_PMCRAID
|
||||||
---help---
|
---help---
|
||||||
This driver supports the PMC SIERRA MaxRAID adapters.
|
This driver supports the PMC SIERRA MaxRAID adapters.
|
||||||
|
|
||||||
|
config SCSI_PM8001
|
||||||
|
tristate "PMC-Sierra SPC 8001 SAS/SATA Based Host Adapter driver"
|
||||||
|
depends on PCI && SCSI
|
||||||
|
select SCSI_SAS_LIBSAS
|
||||||
|
help
|
||||||
|
This driver supports PMC-Sierra PCIE SAS/SATA 8x6G SPC 8001 chip
|
||||||
|
based host adapters.
|
||||||
|
|
||||||
config SCSI_SRP
|
config SCSI_SRP
|
||||||
tristate "SCSI RDMA Protocol helper library"
|
tristate "SCSI RDMA Protocol helper library"
|
||||||
depends on SCSI && PCI
|
depends on SCSI && PCI
|
||||||
|
|
|
@ -70,6 +70,7 @@ obj-$(CONFIG_SCSI_AIC79XX) += aic7xxx/
|
||||||
obj-$(CONFIG_SCSI_AACRAID) += aacraid/
|
obj-$(CONFIG_SCSI_AACRAID) += aacraid/
|
||||||
obj-$(CONFIG_SCSI_AIC7XXX_OLD) += aic7xxx_old.o
|
obj-$(CONFIG_SCSI_AIC7XXX_OLD) += aic7xxx_old.o
|
||||||
obj-$(CONFIG_SCSI_AIC94XX) += aic94xx/
|
obj-$(CONFIG_SCSI_AIC94XX) += aic94xx/
|
||||||
|
obj-$(CONFIG_SCSI_PM8001) += pm8001/
|
||||||
obj-$(CONFIG_SCSI_IPS) += ips.o
|
obj-$(CONFIG_SCSI_IPS) += ips.o
|
||||||
obj-$(CONFIG_SCSI_FD_MCS) += fd_mcs.o
|
obj-$(CONFIG_SCSI_FD_MCS) += fd_mcs.o
|
||||||
obj-$(CONFIG_SCSI_FUTURE_DOMAIN)+= fdomain.o
|
obj-$(CONFIG_SCSI_FUTURE_DOMAIN)+= fdomain.o
|
||||||
|
@ -113,6 +114,7 @@ obj-$(CONFIG_SCSI_MESH) += mesh.o
|
||||||
obj-$(CONFIG_SCSI_MAC53C94) += mac53c94.o
|
obj-$(CONFIG_SCSI_MAC53C94) += mac53c94.o
|
||||||
obj-$(CONFIG_BLK_DEV_3W_XXXX_RAID) += 3w-xxxx.o
|
obj-$(CONFIG_BLK_DEV_3W_XXXX_RAID) += 3w-xxxx.o
|
||||||
obj-$(CONFIG_SCSI_3W_9XXX) += 3w-9xxx.o
|
obj-$(CONFIG_SCSI_3W_9XXX) += 3w-9xxx.o
|
||||||
|
obj-$(CONFIG_SCSI_3W_SAS) += 3w-sas.o
|
||||||
obj-$(CONFIG_SCSI_PPA) += ppa.o
|
obj-$(CONFIG_SCSI_PPA) += ppa.o
|
||||||
obj-$(CONFIG_SCSI_IMM) += imm.o
|
obj-$(CONFIG_SCSI_IMM) += imm.o
|
||||||
obj-$(CONFIG_JAZZ_ESP) += esp_scsi.o jazz_esp.o
|
obj-$(CONFIG_JAZZ_ESP) += esp_scsi.o jazz_esp.o
|
||||||
|
@ -133,6 +135,7 @@ obj-$(CONFIG_SCSI_CXGB3_ISCSI) += libiscsi.o libiscsi_tcp.o cxgb3i/
|
||||||
obj-$(CONFIG_SCSI_BNX2_ISCSI) += libiscsi.o bnx2i/
|
obj-$(CONFIG_SCSI_BNX2_ISCSI) += libiscsi.o bnx2i/
|
||||||
obj-$(CONFIG_BE2ISCSI) += libiscsi.o be2iscsi/
|
obj-$(CONFIG_BE2ISCSI) += libiscsi.o be2iscsi/
|
||||||
obj-$(CONFIG_SCSI_PMCRAID) += pmcraid.o
|
obj-$(CONFIG_SCSI_PMCRAID) += pmcraid.o
|
||||||
|
obj-$(CONFIG_VMWARE_PVSCSI) += vmw_pvscsi.o
|
||||||
|
|
||||||
obj-$(CONFIG_ARM) += arm/
|
obj-$(CONFIG_ARM) += arm/
|
||||||
|
|
||||||
|
|
|
@ -472,8 +472,12 @@ static int aac_slave_configure(struct scsi_device *sdev)
|
||||||
* total capacity and the queue depth supported by the target device.
|
* total capacity and the queue depth supported by the target device.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static int aac_change_queue_depth(struct scsi_device *sdev, int depth)
|
static int aac_change_queue_depth(struct scsi_device *sdev, int depth,
|
||||||
|
int reason)
|
||||||
{
|
{
|
||||||
|
if (reason != SCSI_QDEPTH_DEFAULT)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (sdev->tagged_supported && (sdev->type == TYPE_DISK) &&
|
if (sdev->tagged_supported && (sdev->type == TYPE_DISK) &&
|
||||||
(sdev_channel(sdev) == CONTAINER_CHANNEL)) {
|
(sdev_channel(sdev) == CONTAINER_CHANNEL)) {
|
||||||
struct scsi_device * dev;
|
struct scsi_device * dev;
|
||||||
|
|
|
@ -98,8 +98,11 @@ static void arcmsr_flush_hbb_cache(struct AdapterControlBlock *acb);
|
||||||
static const char *arcmsr_info(struct Scsi_Host *);
|
static const char *arcmsr_info(struct Scsi_Host *);
|
||||||
static irqreturn_t arcmsr_interrupt(struct AdapterControlBlock *acb);
|
static irqreturn_t arcmsr_interrupt(struct AdapterControlBlock *acb);
|
||||||
static int arcmsr_adjust_disk_queue_depth(struct scsi_device *sdev,
|
static int arcmsr_adjust_disk_queue_depth(struct scsi_device *sdev,
|
||||||
int queue_depth)
|
int queue_depth, int reason)
|
||||||
{
|
{
|
||||||
|
if (reason != SCSI_QDEPTH_DEFAULT)
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
if (queue_depth > ARCMSR_MAX_CMD_PERLUN)
|
if (queue_depth > ARCMSR_MAX_CMD_PERLUN)
|
||||||
queue_depth = ARCMSR_MAX_CMD_PERLUN;
|
queue_depth = ARCMSR_MAX_CMD_PERLUN;
|
||||||
scsi_adjust_queue_depth(sdev, MSG_ORDERED_TAG, queue_depth);
|
scsi_adjust_queue_depth(sdev, MSG_ORDERED_TAG, queue_depth);
|
||||||
|
|
|
@ -20,8 +20,10 @@
|
||||||
|
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
#include <linux/if_vlan.h>
|
#include <linux/if_vlan.h>
|
||||||
|
#include <linux/blk-iopoll.h>
|
||||||
#define FW_VER_LEN 32
|
#define FW_VER_LEN 32
|
||||||
|
#define MCC_Q_LEN 128
|
||||||
|
#define MCC_CQ_LEN 256
|
||||||
|
|
||||||
struct be_dma_mem {
|
struct be_dma_mem {
|
||||||
void *va;
|
void *va;
|
||||||
|
@ -74,18 +76,14 @@ static inline void queue_tail_inc(struct be_queue_info *q)
|
||||||
|
|
||||||
struct be_eq_obj {
|
struct be_eq_obj {
|
||||||
struct be_queue_info q;
|
struct be_queue_info q;
|
||||||
char desc[32];
|
struct beiscsi_hba *phba;
|
||||||
|
struct be_queue_info *cq;
|
||||||
/* Adaptive interrupt coalescing (AIC) info */
|
struct blk_iopoll iopoll;
|
||||||
bool enable_aic;
|
|
||||||
u16 min_eqd; /* in usecs */
|
|
||||||
u16 max_eqd; /* in usecs */
|
|
||||||
u16 cur_eqd; /* in usecs */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct be_mcc_obj {
|
struct be_mcc_obj {
|
||||||
struct be_queue_info *q;
|
struct be_queue_info q;
|
||||||
struct be_queue_info *cq;
|
struct be_queue_info cq;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct be_ctrl_info {
|
struct be_ctrl_info {
|
||||||
|
@ -176,8 +174,4 @@ static inline void swap_dws(void *wrb, int len)
|
||||||
} while (len);
|
} while (len);
|
||||||
#endif /* __BIG_ENDIAN */
|
#endif /* __BIG_ENDIAN */
|
||||||
}
|
}
|
||||||
|
|
||||||
extern void beiscsi_cq_notify(struct be_ctrl_info *ctrl, u16 qid, bool arm,
|
|
||||||
u16 num_popped);
|
|
||||||
|
|
||||||
#endif /* BEISCSI_H */
|
#endif /* BEISCSI_H */
|
||||||
|
|
|
@ -19,6 +19,16 @@
|
||||||
#include "be_mgmt.h"
|
#include "be_mgmt.h"
|
||||||
#include "be_main.h"
|
#include "be_main.h"
|
||||||
|
|
||||||
|
static void be_mcc_notify(struct beiscsi_hba *phba)
|
||||||
|
{
|
||||||
|
struct be_queue_info *mccq = &phba->ctrl.mcc_obj.q;
|
||||||
|
u32 val = 0;
|
||||||
|
|
||||||
|
val |= mccq->id & DB_MCCQ_RING_ID_MASK;
|
||||||
|
val |= 1 << DB_MCCQ_NUM_POSTED_SHIFT;
|
||||||
|
iowrite32(val, phba->db_va + DB_MCCQ_OFFSET);
|
||||||
|
}
|
||||||
|
|
||||||
static inline bool be_mcc_compl_is_new(struct be_mcc_compl *compl)
|
static inline bool be_mcc_compl_is_new(struct be_mcc_compl *compl)
|
||||||
{
|
{
|
||||||
if (compl->flags != 0) {
|
if (compl->flags != 0) {
|
||||||
|
@ -54,13 +64,56 @@ static int be_mcc_compl_process(struct be_ctrl_info *ctrl,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static inline bool is_link_state_evt(u32 trailer)
|
static inline bool is_link_state_evt(u32 trailer)
|
||||||
{
|
{
|
||||||
return (((trailer >> ASYNC_TRAILER_EVENT_CODE_SHIFT) &
|
return (((trailer >> ASYNC_TRAILER_EVENT_CODE_SHIFT) &
|
||||||
ASYNC_TRAILER_EVENT_CODE_MASK) == ASYNC_EVENT_CODE_LINK_STATE);
|
ASYNC_TRAILER_EVENT_CODE_MASK) ==
|
||||||
|
ASYNC_EVENT_CODE_LINK_STATE);
|
||||||
}
|
}
|
||||||
|
|
||||||
void beiscsi_cq_notify(struct be_ctrl_info *ctrl, u16 qid, bool arm,
|
static struct be_mcc_compl *be_mcc_compl_get(struct beiscsi_hba *phba)
|
||||||
|
{
|
||||||
|
struct be_queue_info *mcc_cq = &phba->ctrl.mcc_obj.cq;
|
||||||
|
struct be_mcc_compl *compl = queue_tail_node(mcc_cq);
|
||||||
|
|
||||||
|
if (be_mcc_compl_is_new(compl)) {
|
||||||
|
queue_tail_inc(mcc_cq);
|
||||||
|
return compl;
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void be2iscsi_fail_session(struct iscsi_cls_session *cls_session)
|
||||||
|
{
|
||||||
|
iscsi_session_failure(cls_session->dd_data, ISCSI_ERR_CONN_FAILED);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void beiscsi_async_link_state_process(struct beiscsi_hba *phba,
|
||||||
|
struct be_async_event_link_state *evt)
|
||||||
|
{
|
||||||
|
switch (evt->port_link_status) {
|
||||||
|
case ASYNC_EVENT_LINK_DOWN:
|
||||||
|
SE_DEBUG(DBG_LVL_1, "Link Down on Physical Port %d \n",
|
||||||
|
evt->physical_port);
|
||||||
|
phba->state |= BE_ADAPTER_LINK_DOWN;
|
||||||
|
break;
|
||||||
|
case ASYNC_EVENT_LINK_UP:
|
||||||
|
phba->state = BE_ADAPTER_UP;
|
||||||
|
SE_DEBUG(DBG_LVL_1, "Link UP on Physical Port %d \n",
|
||||||
|
evt->physical_port);
|
||||||
|
iscsi_host_for_each_session(phba->shost,
|
||||||
|
be2iscsi_fail_session);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
SE_DEBUG(DBG_LVL_1, "Unexpected Async Notification %d on"
|
||||||
|
"Physical Port %d \n",
|
||||||
|
evt->port_link_status,
|
||||||
|
evt->physical_port);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void beiscsi_cq_notify(struct beiscsi_hba *phba, u16 qid, bool arm,
|
||||||
u16 num_popped)
|
u16 num_popped)
|
||||||
{
|
{
|
||||||
u32 val = 0;
|
u32 val = 0;
|
||||||
|
@ -68,7 +121,66 @@ void beiscsi_cq_notify(struct be_ctrl_info *ctrl, u16 qid, bool arm,
|
||||||
if (arm)
|
if (arm)
|
||||||
val |= 1 << DB_CQ_REARM_SHIFT;
|
val |= 1 << DB_CQ_REARM_SHIFT;
|
||||||
val |= num_popped << DB_CQ_NUM_POPPED_SHIFT;
|
val |= num_popped << DB_CQ_NUM_POPPED_SHIFT;
|
||||||
iowrite32(val, ctrl->db + DB_CQ_OFFSET);
|
iowrite32(val, phba->db_va + DB_CQ_OFFSET);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int beiscsi_process_mcc(struct beiscsi_hba *phba)
|
||||||
|
{
|
||||||
|
struct be_mcc_compl *compl;
|
||||||
|
int num = 0, status = 0;
|
||||||
|
struct be_ctrl_info *ctrl = &phba->ctrl;
|
||||||
|
|
||||||
|
spin_lock_bh(&phba->ctrl.mcc_cq_lock);
|
||||||
|
while ((compl = be_mcc_compl_get(phba))) {
|
||||||
|
if (compl->flags & CQE_FLAGS_ASYNC_MASK) {
|
||||||
|
/* Interpret flags as an async trailer */
|
||||||
|
BUG_ON(!is_link_state_evt(compl->flags));
|
||||||
|
|
||||||
|
/* Interpret compl as a async link evt */
|
||||||
|
beiscsi_async_link_state_process(phba,
|
||||||
|
(struct be_async_event_link_state *) compl);
|
||||||
|
} else if (compl->flags & CQE_FLAGS_COMPLETED_MASK) {
|
||||||
|
status = be_mcc_compl_process(ctrl, compl);
|
||||||
|
atomic_dec(&phba->ctrl.mcc_obj.q.used);
|
||||||
|
}
|
||||||
|
be_mcc_compl_use(compl);
|
||||||
|
num++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (num)
|
||||||
|
beiscsi_cq_notify(phba, phba->ctrl.mcc_obj.cq.id, true, num);
|
||||||
|
|
||||||
|
spin_unlock_bh(&phba->ctrl.mcc_cq_lock);
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Wait till no more pending mcc requests are present */
|
||||||
|
static int be_mcc_wait_compl(struct beiscsi_hba *phba)
|
||||||
|
{
|
||||||
|
#define mcc_timeout 120000 /* 5s timeout */
|
||||||
|
int i, status;
|
||||||
|
for (i = 0; i < mcc_timeout; i++) {
|
||||||
|
status = beiscsi_process_mcc(phba);
|
||||||
|
if (status)
|
||||||
|
return status;
|
||||||
|
|
||||||
|
if (atomic_read(&phba->ctrl.mcc_obj.q.used) == 0)
|
||||||
|
break;
|
||||||
|
udelay(100);
|
||||||
|
}
|
||||||
|
if (i == mcc_timeout) {
|
||||||
|
dev_err(&phba->pcidev->dev, "mccq poll timed out\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Notify MCC requests and wait for completion */
|
||||||
|
int be_mcc_notify_wait(struct beiscsi_hba *phba)
|
||||||
|
{
|
||||||
|
be_mcc_notify(phba);
|
||||||
|
return be_mcc_wait_compl(phba);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int be_mbox_db_ready_wait(struct be_ctrl_info *ctrl)
|
static int be_mbox_db_ready_wait(struct be_ctrl_info *ctrl)
|
||||||
|
@ -142,6 +254,52 @@ int be_mbox_notify(struct be_ctrl_info *ctrl)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Insert the mailbox address into the doorbell in two steps
|
||||||
|
* Polls on the mbox doorbell till a command completion (or a timeout) occurs
|
||||||
|
*/
|
||||||
|
static int be_mbox_notify_wait(struct beiscsi_hba *phba)
|
||||||
|
{
|
||||||
|
int status;
|
||||||
|
u32 val = 0;
|
||||||
|
void __iomem *db = phba->ctrl.db + MPU_MAILBOX_DB_OFFSET;
|
||||||
|
struct be_dma_mem *mbox_mem = &phba->ctrl.mbox_mem;
|
||||||
|
struct be_mcc_mailbox *mbox = mbox_mem->va;
|
||||||
|
struct be_mcc_compl *compl = &mbox->compl;
|
||||||
|
struct be_ctrl_info *ctrl = &phba->ctrl;
|
||||||
|
|
||||||
|
val |= MPU_MAILBOX_DB_HI_MASK;
|
||||||
|
/* at bits 2 - 31 place mbox dma addr msb bits 34 - 63 */
|
||||||
|
val |= (upper_32_bits(mbox_mem->dma) >> 2) << 2;
|
||||||
|
iowrite32(val, db);
|
||||||
|
|
||||||
|
/* wait for ready to be set */
|
||||||
|
status = be_mbox_db_ready_wait(ctrl);
|
||||||
|
if (status != 0)
|
||||||
|
return status;
|
||||||
|
|
||||||
|
val = 0;
|
||||||
|
/* at bits 2 - 31 place mbox dma addr lsb bits 4 - 33 */
|
||||||
|
val |= (u32)(mbox_mem->dma >> 4) << 2;
|
||||||
|
iowrite32(val, db);
|
||||||
|
|
||||||
|
status = be_mbox_db_ready_wait(ctrl);
|
||||||
|
if (status != 0)
|
||||||
|
return status;
|
||||||
|
|
||||||
|
/* A cq entry has been made now */
|
||||||
|
if (be_mcc_compl_is_new(compl)) {
|
||||||
|
status = be_mcc_compl_process(ctrl, &mbox->compl);
|
||||||
|
be_mcc_compl_use(compl);
|
||||||
|
if (status)
|
||||||
|
return status;
|
||||||
|
} else {
|
||||||
|
dev_err(&phba->pcidev->dev, "invalid mailbox completion\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
void be_wrb_hdr_prepare(struct be_mcc_wrb *wrb, int payload_len,
|
void be_wrb_hdr_prepare(struct be_mcc_wrb *wrb, int payload_len,
|
||||||
bool embedded, u8 sge_cnt)
|
bool embedded, u8 sge_cnt)
|
||||||
{
|
{
|
||||||
|
@ -203,6 +361,20 @@ struct be_mcc_wrb *wrb_from_mbox(struct be_dma_mem *mbox_mem)
|
||||||
return &((struct be_mcc_mailbox *)(mbox_mem->va))->wrb;
|
return &((struct be_mcc_mailbox *)(mbox_mem->va))->wrb;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct be_mcc_wrb *wrb_from_mccq(struct beiscsi_hba *phba)
|
||||||
|
{
|
||||||
|
struct be_queue_info *mccq = &phba->ctrl.mcc_obj.q;
|
||||||
|
struct be_mcc_wrb *wrb;
|
||||||
|
|
||||||
|
BUG_ON(atomic_read(&mccq->used) >= mccq->len);
|
||||||
|
wrb = queue_head_node(mccq);
|
||||||
|
queue_head_inc(mccq);
|
||||||
|
atomic_inc(&mccq->used);
|
||||||
|
memset(wrb, 0, sizeof(*wrb));
|
||||||
|
return wrb;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
int beiscsi_cmd_eq_create(struct be_ctrl_info *ctrl,
|
int beiscsi_cmd_eq_create(struct be_ctrl_info *ctrl,
|
||||||
struct be_queue_info *eq, int eq_delay)
|
struct be_queue_info *eq, int eq_delay)
|
||||||
{
|
{
|
||||||
|
@ -212,6 +384,7 @@ int beiscsi_cmd_eq_create(struct be_ctrl_info *ctrl,
|
||||||
struct be_dma_mem *q_mem = &eq->dma_mem;
|
struct be_dma_mem *q_mem = &eq->dma_mem;
|
||||||
int status;
|
int status;
|
||||||
|
|
||||||
|
SE_DEBUG(DBG_LVL_8, "In beiscsi_cmd_eq_create\n");
|
||||||
spin_lock(&ctrl->mbox_lock);
|
spin_lock(&ctrl->mbox_lock);
|
||||||
memset(wrb, 0, sizeof(*wrb));
|
memset(wrb, 0, sizeof(*wrb));
|
||||||
|
|
||||||
|
@ -249,6 +422,7 @@ int be_cmd_fw_initialize(struct be_ctrl_info *ctrl)
|
||||||
int status;
|
int status;
|
||||||
u8 *endian_check;
|
u8 *endian_check;
|
||||||
|
|
||||||
|
SE_DEBUG(DBG_LVL_8, "In be_cmd_fw_initialize\n");
|
||||||
spin_lock(&ctrl->mbox_lock);
|
spin_lock(&ctrl->mbox_lock);
|
||||||
memset(wrb, 0, sizeof(*wrb));
|
memset(wrb, 0, sizeof(*wrb));
|
||||||
|
|
||||||
|
@ -282,6 +456,7 @@ int beiscsi_cmd_cq_create(struct be_ctrl_info *ctrl,
|
||||||
void *ctxt = &req->context;
|
void *ctxt = &req->context;
|
||||||
int status;
|
int status;
|
||||||
|
|
||||||
|
SE_DEBUG(DBG_LVL_8, "In beiscsi_cmd_cq_create \n");
|
||||||
spin_lock(&ctrl->mbox_lock);
|
spin_lock(&ctrl->mbox_lock);
|
||||||
memset(wrb, 0, sizeof(*wrb));
|
memset(wrb, 0, sizeof(*wrb));
|
||||||
|
|
||||||
|
@ -289,7 +464,6 @@ int beiscsi_cmd_cq_create(struct be_ctrl_info *ctrl,
|
||||||
|
|
||||||
be_cmd_hdr_prepare(&req->hdr, CMD_SUBSYSTEM_COMMON,
|
be_cmd_hdr_prepare(&req->hdr, CMD_SUBSYSTEM_COMMON,
|
||||||
OPCODE_COMMON_CQ_CREATE, sizeof(*req));
|
OPCODE_COMMON_CQ_CREATE, sizeof(*req));
|
||||||
|
|
||||||
if (!q_mem->va)
|
if (!q_mem->va)
|
||||||
SE_DEBUG(DBG_LVL_1, "uninitialized q_mem->va\n");
|
SE_DEBUG(DBG_LVL_1, "uninitialized q_mem->va\n");
|
||||||
|
|
||||||
|
@ -329,6 +503,53 @@ static u32 be_encoded_q_len(int q_len)
|
||||||
len_encoded = 0;
|
len_encoded = 0;
|
||||||
return len_encoded;
|
return len_encoded;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int beiscsi_cmd_mccq_create(struct beiscsi_hba *phba,
|
||||||
|
struct be_queue_info *mccq,
|
||||||
|
struct be_queue_info *cq)
|
||||||
|
{
|
||||||
|
struct be_mcc_wrb *wrb;
|
||||||
|
struct be_cmd_req_mcc_create *req;
|
||||||
|
struct be_dma_mem *q_mem = &mccq->dma_mem;
|
||||||
|
struct be_ctrl_info *ctrl;
|
||||||
|
void *ctxt;
|
||||||
|
int status;
|
||||||
|
|
||||||
|
spin_lock(&phba->ctrl.mbox_lock);
|
||||||
|
ctrl = &phba->ctrl;
|
||||||
|
wrb = wrb_from_mbox(&ctrl->mbox_mem);
|
||||||
|
req = embedded_payload(wrb);
|
||||||
|
ctxt = &req->context;
|
||||||
|
|
||||||
|
be_wrb_hdr_prepare(wrb, sizeof(*req), true, 0);
|
||||||
|
|
||||||
|
be_cmd_hdr_prepare(&req->hdr, CMD_SUBSYSTEM_COMMON,
|
||||||
|
OPCODE_COMMON_MCC_CREATE, sizeof(*req));
|
||||||
|
|
||||||
|
req->num_pages = PAGES_4K_SPANNED(q_mem->va, q_mem->size);
|
||||||
|
|
||||||
|
AMAP_SET_BITS(struct amap_mcc_context, fid, ctxt,
|
||||||
|
PCI_FUNC(phba->pcidev->devfn));
|
||||||
|
AMAP_SET_BITS(struct amap_mcc_context, valid, ctxt, 1);
|
||||||
|
AMAP_SET_BITS(struct amap_mcc_context, ring_size, ctxt,
|
||||||
|
be_encoded_q_len(mccq->len));
|
||||||
|
AMAP_SET_BITS(struct amap_mcc_context, cq_id, ctxt, cq->id);
|
||||||
|
|
||||||
|
be_dws_cpu_to_le(ctxt, sizeof(req->context));
|
||||||
|
|
||||||
|
be_cmd_page_addrs_prepare(req->pages, ARRAY_SIZE(req->pages), q_mem);
|
||||||
|
|
||||||
|
status = be_mbox_notify_wait(phba);
|
||||||
|
if (!status) {
|
||||||
|
struct be_cmd_resp_mcc_create *resp = embedded_payload(wrb);
|
||||||
|
mccq->id = le16_to_cpu(resp->id);
|
||||||
|
mccq->created = true;
|
||||||
|
}
|
||||||
|
spin_unlock(&phba->ctrl.mbox_lock);
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
int beiscsi_cmd_q_destroy(struct be_ctrl_info *ctrl, struct be_queue_info *q,
|
int beiscsi_cmd_q_destroy(struct be_ctrl_info *ctrl, struct be_queue_info *q,
|
||||||
int queue_type)
|
int queue_type)
|
||||||
{
|
{
|
||||||
|
@ -337,6 +558,7 @@ int beiscsi_cmd_q_destroy(struct be_ctrl_info *ctrl, struct be_queue_info *q,
|
||||||
u8 subsys = 0, opcode = 0;
|
u8 subsys = 0, opcode = 0;
|
||||||
int status;
|
int status;
|
||||||
|
|
||||||
|
SE_DEBUG(DBG_LVL_8, "In beiscsi_cmd_q_destroy \n");
|
||||||
spin_lock(&ctrl->mbox_lock);
|
spin_lock(&ctrl->mbox_lock);
|
||||||
memset(wrb, 0, sizeof(*wrb));
|
memset(wrb, 0, sizeof(*wrb));
|
||||||
be_wrb_hdr_prepare(wrb, sizeof(*req), true, 0);
|
be_wrb_hdr_prepare(wrb, sizeof(*req), true, 0);
|
||||||
|
@ -350,6 +572,10 @@ int beiscsi_cmd_q_destroy(struct be_ctrl_info *ctrl, struct be_queue_info *q,
|
||||||
subsys = CMD_SUBSYSTEM_COMMON;
|
subsys = CMD_SUBSYSTEM_COMMON;
|
||||||
opcode = OPCODE_COMMON_CQ_DESTROY;
|
opcode = OPCODE_COMMON_CQ_DESTROY;
|
||||||
break;
|
break;
|
||||||
|
case QTYPE_MCCQ:
|
||||||
|
subsys = CMD_SUBSYSTEM_COMMON;
|
||||||
|
opcode = OPCODE_COMMON_MCC_DESTROY;
|
||||||
|
break;
|
||||||
case QTYPE_WRBQ:
|
case QTYPE_WRBQ:
|
||||||
subsys = CMD_SUBSYSTEM_ISCSI;
|
subsys = CMD_SUBSYSTEM_ISCSI;
|
||||||
opcode = OPCODE_COMMON_ISCSI_WRBQ_DESTROY;
|
opcode = OPCODE_COMMON_ISCSI_WRBQ_DESTROY;
|
||||||
|
@ -377,30 +603,6 @@ int beiscsi_cmd_q_destroy(struct be_ctrl_info *ctrl, struct be_queue_info *q,
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
int be_cmd_get_mac_addr(struct be_ctrl_info *ctrl, u8 *mac_addr)
|
|
||||||
{
|
|
||||||
struct be_mcc_wrb *wrb = wrb_from_mbox(&ctrl->mbox_mem);
|
|
||||||
struct be_cmd_req_get_mac_addr *req = embedded_payload(wrb);
|
|
||||||
int status;
|
|
||||||
|
|
||||||
spin_lock(&ctrl->mbox_lock);
|
|
||||||
memset(wrb, 0, sizeof(*wrb));
|
|
||||||
be_wrb_hdr_prepare(wrb, sizeof(*req), true, 0);
|
|
||||||
be_cmd_hdr_prepare(&req->hdr, CMD_SUBSYSTEM_ISCSI,
|
|
||||||
OPCODE_COMMON_ISCSI_NTWK_GET_NIC_CONFIG,
|
|
||||||
sizeof(*req));
|
|
||||||
|
|
||||||
status = be_mbox_notify(ctrl);
|
|
||||||
if (!status) {
|
|
||||||
struct be_cmd_resp_get_mac_addr *resp = embedded_payload(wrb);
|
|
||||||
|
|
||||||
memcpy(mac_addr, resp->mac_address, ETH_ALEN);
|
|
||||||
}
|
|
||||||
|
|
||||||
spin_unlock(&ctrl->mbox_lock);
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
int be_cmd_create_default_pdu_queue(struct be_ctrl_info *ctrl,
|
int be_cmd_create_default_pdu_queue(struct be_ctrl_info *ctrl,
|
||||||
struct be_queue_info *cq,
|
struct be_queue_info *cq,
|
||||||
struct be_queue_info *dq, int length,
|
struct be_queue_info *dq, int length,
|
||||||
|
@ -412,6 +614,7 @@ int be_cmd_create_default_pdu_queue(struct be_ctrl_info *ctrl,
|
||||||
void *ctxt = &req->context;
|
void *ctxt = &req->context;
|
||||||
int status;
|
int status;
|
||||||
|
|
||||||
|
SE_DEBUG(DBG_LVL_8, "In be_cmd_create_default_pdu_queue\n");
|
||||||
spin_lock(&ctrl->mbox_lock);
|
spin_lock(&ctrl->mbox_lock);
|
||||||
memset(wrb, 0, sizeof(*wrb));
|
memset(wrb, 0, sizeof(*wrb));
|
||||||
|
|
||||||
|
@ -468,8 +671,10 @@ int be_cmd_wrbq_create(struct be_ctrl_info *ctrl, struct be_dma_mem *q_mem,
|
||||||
be_cmd_page_addrs_prepare(req->pages, ARRAY_SIZE(req->pages), q_mem);
|
be_cmd_page_addrs_prepare(req->pages, ARRAY_SIZE(req->pages), q_mem);
|
||||||
|
|
||||||
status = be_mbox_notify(ctrl);
|
status = be_mbox_notify(ctrl);
|
||||||
if (!status)
|
if (!status) {
|
||||||
wrbq->id = le16_to_cpu(resp->cid);
|
wrbq->id = le16_to_cpu(resp->cid);
|
||||||
|
wrbq->created = true;
|
||||||
|
}
|
||||||
spin_unlock(&ctrl->mbox_lock);
|
spin_unlock(&ctrl->mbox_lock);
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,6 +47,8 @@ struct be_mcc_wrb {
|
||||||
|
|
||||||
#define CQE_FLAGS_VALID_MASK (1 << 31)
|
#define CQE_FLAGS_VALID_MASK (1 << 31)
|
||||||
#define CQE_FLAGS_ASYNC_MASK (1 << 30)
|
#define CQE_FLAGS_ASYNC_MASK (1 << 30)
|
||||||
|
#define CQE_FLAGS_COMPLETED_MASK (1 << 28)
|
||||||
|
#define CQE_FLAGS_CONSUMED_MASK (1 << 27)
|
||||||
|
|
||||||
/* Completion Status */
|
/* Completion Status */
|
||||||
#define MCC_STATUS_SUCCESS 0x0
|
#define MCC_STATUS_SUCCESS 0x0
|
||||||
|
@ -173,7 +175,7 @@ struct be_cmd_req_hdr {
|
||||||
u8 domain; /* dword 0 */
|
u8 domain; /* dword 0 */
|
||||||
u32 timeout; /* dword 1 */
|
u32 timeout; /* dword 1 */
|
||||||
u32 request_length; /* dword 2 */
|
u32 request_length; /* dword 2 */
|
||||||
u32 rsvd; /* dword 3 */
|
u32 rsvd0; /* dword 3 */
|
||||||
};
|
};
|
||||||
|
|
||||||
struct be_cmd_resp_hdr {
|
struct be_cmd_resp_hdr {
|
||||||
|
@ -382,7 +384,6 @@ struct be_cmd_req_modify_eq_delay {
|
||||||
|
|
||||||
#define ETH_ALEN 6
|
#define ETH_ALEN 6
|
||||||
|
|
||||||
|
|
||||||
struct be_cmd_req_get_mac_addr {
|
struct be_cmd_req_get_mac_addr {
|
||||||
struct be_cmd_req_hdr hdr;
|
struct be_cmd_req_hdr hdr;
|
||||||
u32 nic_port_count;
|
u32 nic_port_count;
|
||||||
|
@ -417,14 +418,21 @@ int beiscsi_cmd_cq_create(struct be_ctrl_info *ctrl,
|
||||||
|
|
||||||
int beiscsi_cmd_q_destroy(struct be_ctrl_info *ctrl, struct be_queue_info *q,
|
int beiscsi_cmd_q_destroy(struct be_ctrl_info *ctrl, struct be_queue_info *q,
|
||||||
int type);
|
int type);
|
||||||
|
int beiscsi_cmd_mccq_create(struct beiscsi_hba *phba,
|
||||||
|
struct be_queue_info *mccq,
|
||||||
|
struct be_queue_info *cq);
|
||||||
|
|
||||||
int be_poll_mcc(struct be_ctrl_info *ctrl);
|
int be_poll_mcc(struct be_ctrl_info *ctrl);
|
||||||
unsigned char mgmt_check_supported_fw(struct be_ctrl_info *ctrl);
|
unsigned char mgmt_check_supported_fw(struct be_ctrl_info *ctrl,
|
||||||
int be_cmd_get_mac_addr(struct be_ctrl_info *ctrl, u8 *mac_addr);
|
struct beiscsi_hba *phba);
|
||||||
|
int be_cmd_get_mac_addr(struct beiscsi_hba *phba, u8 *mac_addr);
|
||||||
|
|
||||||
/*ISCSI Functuions */
|
/*ISCSI Functuions */
|
||||||
int be_cmd_fw_initialize(struct be_ctrl_info *ctrl);
|
int be_cmd_fw_initialize(struct be_ctrl_info *ctrl);
|
||||||
|
|
||||||
struct be_mcc_wrb *wrb_from_mbox(struct be_dma_mem *mbox_mem);
|
struct be_mcc_wrb *wrb_from_mbox(struct be_dma_mem *mbox_mem);
|
||||||
|
struct be_mcc_wrb *wrb_from_mccq(struct beiscsi_hba *phba);
|
||||||
|
int be_mcc_notify_wait(struct beiscsi_hba *phba);
|
||||||
|
|
||||||
int be_mbox_notify(struct be_ctrl_info *ctrl);
|
int be_mbox_notify(struct be_ctrl_info *ctrl);
|
||||||
|
|
||||||
|
@ -531,6 +539,23 @@ struct amap_sol_cqe {
|
||||||
u8 valid; /* dword 3 */
|
u8 valid; /* dword 3 */
|
||||||
} __packed;
|
} __packed;
|
||||||
|
|
||||||
|
#define SOL_ICD_INDEX_MASK 0x0003FFC0
|
||||||
|
struct amap_sol_cqe_ring {
|
||||||
|
u8 hw_sts[8]; /* dword 0 */
|
||||||
|
u8 i_sts[8]; /* dword 0 */
|
||||||
|
u8 i_resp[8]; /* dword 0 */
|
||||||
|
u8 i_flags[7]; /* dword 0 */
|
||||||
|
u8 s; /* dword 0 */
|
||||||
|
u8 i_exp_cmd_sn[32]; /* dword 1 */
|
||||||
|
u8 code[6]; /* dword 2 */
|
||||||
|
u8 icd_index[12]; /* dword 2 */
|
||||||
|
u8 rsvd[6]; /* dword 2 */
|
||||||
|
u8 i_cmd_wnd[8]; /* dword 2 */
|
||||||
|
u8 i_res_cnt[31]; /* dword 3 */
|
||||||
|
u8 valid; /* dword 3 */
|
||||||
|
} __packed;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Post WRB Queue Doorbell Register used by the host Storage
|
* Post WRB Queue Doorbell Register used by the host Storage
|
||||||
|
@ -664,8 +689,8 @@ struct be_fw_cfg {
|
||||||
#define OPCODE_COMMON_TCP_UPLOAD 56
|
#define OPCODE_COMMON_TCP_UPLOAD 56
|
||||||
#define OPCODE_COMMON_ISCSI_ERROR_RECOVERY_INVALIDATE_COMMANDS 1
|
#define OPCODE_COMMON_ISCSI_ERROR_RECOVERY_INVALIDATE_COMMANDS 1
|
||||||
/* --- CMD_ISCSI_INVALIDATE_CONNECTION_TYPE --- */
|
/* --- CMD_ISCSI_INVALIDATE_CONNECTION_TYPE --- */
|
||||||
#define CMD_ISCSI_CONNECTION_INVALIDATE 1
|
#define CMD_ISCSI_CONNECTION_INVALIDATE 0x8001
|
||||||
#define CMD_ISCSI_CONNECTION_ISSUE_TCP_RST 2
|
#define CMD_ISCSI_CONNECTION_ISSUE_TCP_RST 0x8002
|
||||||
#define OPCODE_ISCSI_INI_DRIVER_INVALIDATE_CONNECTION 42
|
#define OPCODE_ISCSI_INI_DRIVER_INVALIDATE_CONNECTION 42
|
||||||
|
|
||||||
#define INI_WR_CMD 1 /* Initiator write command */
|
#define INI_WR_CMD 1 /* Initiator write command */
|
||||||
|
|
|
@ -297,7 +297,7 @@ int beiscsi_get_host_param(struct Scsi_Host *shost,
|
||||||
|
|
||||||
switch (param) {
|
switch (param) {
|
||||||
case ISCSI_HOST_PARAM_HWADDRESS:
|
case ISCSI_HOST_PARAM_HWADDRESS:
|
||||||
be_cmd_get_mac_addr(&phba->ctrl, phba->mac_address);
|
be_cmd_get_mac_addr(phba, phba->mac_address);
|
||||||
len = sysfs_format_mac(buf, phba->mac_address, ETH_ALEN);
|
len = sysfs_format_mac(buf, phba->mac_address, ETH_ALEN);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
@ -377,16 +377,12 @@ int beiscsi_conn_start(struct iscsi_cls_conn *cls_conn)
|
||||||
struct beiscsi_conn *beiscsi_conn = conn->dd_data;
|
struct beiscsi_conn *beiscsi_conn = conn->dd_data;
|
||||||
struct beiscsi_endpoint *beiscsi_ep;
|
struct beiscsi_endpoint *beiscsi_ep;
|
||||||
struct beiscsi_offload_params params;
|
struct beiscsi_offload_params params;
|
||||||
struct iscsi_session *session = conn->session;
|
|
||||||
struct Scsi_Host *shost = iscsi_session_to_shost(session->cls_session);
|
|
||||||
struct beiscsi_hba *phba = iscsi_host_priv(shost);
|
|
||||||
|
|
||||||
memset(¶ms, 0, sizeof(struct beiscsi_offload_params));
|
memset(¶ms, 0, sizeof(struct beiscsi_offload_params));
|
||||||
beiscsi_ep = beiscsi_conn->ep;
|
beiscsi_ep = beiscsi_conn->ep;
|
||||||
if (!beiscsi_ep)
|
if (!beiscsi_ep)
|
||||||
SE_DEBUG(DBG_LVL_1, "In beiscsi_conn_start , no beiscsi_ep\n");
|
SE_DEBUG(DBG_LVL_1, "In beiscsi_conn_start , no beiscsi_ep\n");
|
||||||
|
|
||||||
free_mgmt_sgl_handle(phba, beiscsi_conn->plogin_sgl_handle);
|
|
||||||
beiscsi_conn->login_in_progress = 0;
|
beiscsi_conn->login_in_progress = 0;
|
||||||
beiscsi_set_params_for_offld(beiscsi_conn, ¶ms);
|
beiscsi_set_params_for_offld(beiscsi_conn, ¶ms);
|
||||||
beiscsi_offload_connection(beiscsi_conn, ¶ms);
|
beiscsi_offload_connection(beiscsi_conn, ¶ms);
|
||||||
|
@ -498,6 +494,13 @@ beiscsi_ep_connect(struct Scsi_Host *shost, struct sockaddr *dst_addr,
|
||||||
SE_DEBUG(DBG_LVL_1, "shost is NULL \n");
|
SE_DEBUG(DBG_LVL_1, "shost is NULL \n");
|
||||||
return ERR_PTR(ret);
|
return ERR_PTR(ret);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (phba->state) {
|
||||||
|
ret = -EBUSY;
|
||||||
|
SE_DEBUG(DBG_LVL_1, "The Adapet state is Not UP \n");
|
||||||
|
return ERR_PTR(ret);
|
||||||
|
}
|
||||||
|
|
||||||
ep = iscsi_create_endpoint(sizeof(struct beiscsi_endpoint));
|
ep = iscsi_create_endpoint(sizeof(struct beiscsi_endpoint));
|
||||||
if (!ep) {
|
if (!ep) {
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -21,11 +21,9 @@
|
||||||
#ifndef _BEISCSI_MAIN_
|
#ifndef _BEISCSI_MAIN_
|
||||||
#define _BEISCSI_MAIN_
|
#define _BEISCSI_MAIN_
|
||||||
|
|
||||||
|
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
#include <linux/in.h>
|
#include <linux/in.h>
|
||||||
#include <linux/blk-iopoll.h>
|
|
||||||
#include <scsi/scsi.h>
|
#include <scsi/scsi.h>
|
||||||
#include <scsi/scsi_cmnd.h>
|
#include <scsi/scsi_cmnd.h>
|
||||||
#include <scsi/scsi_device.h>
|
#include <scsi/scsi_device.h>
|
||||||
|
@ -35,12 +33,8 @@
|
||||||
#include <scsi/scsi_transport_iscsi.h>
|
#include <scsi/scsi_transport_iscsi.h>
|
||||||
|
|
||||||
#include "be.h"
|
#include "be.h"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#define DRV_NAME "be2iscsi"
|
#define DRV_NAME "be2iscsi"
|
||||||
#define BUILD_STR "2.0.527.0"
|
#define BUILD_STR "2.0.527.0"
|
||||||
|
|
||||||
#define BE_NAME "ServerEngines BladeEngine2" \
|
#define BE_NAME "ServerEngines BladeEngine2" \
|
||||||
"Linux iSCSI Driver version" BUILD_STR
|
"Linux iSCSI Driver version" BUILD_STR
|
||||||
#define DRV_DESC BE_NAME " " "Driver"
|
#define DRV_DESC BE_NAME " " "Driver"
|
||||||
|
@ -49,6 +43,8 @@
|
||||||
#define BE_DEVICE_ID1 0x212
|
#define BE_DEVICE_ID1 0x212
|
||||||
#define OC_DEVICE_ID1 0x702
|
#define OC_DEVICE_ID1 0x702
|
||||||
#define OC_DEVICE_ID2 0x703
|
#define OC_DEVICE_ID2 0x703
|
||||||
|
#define OC_DEVICE_ID3 0x712
|
||||||
|
#define OC_DEVICE_ID4 0x222
|
||||||
|
|
||||||
#define BE2_MAX_SESSIONS 64
|
#define BE2_MAX_SESSIONS 64
|
||||||
#define BE2_CMDS_PER_CXN 128
|
#define BE2_CMDS_PER_CXN 128
|
||||||
|
@ -63,6 +59,7 @@
|
||||||
#define BE2_IO_DEPTH \
|
#define BE2_IO_DEPTH \
|
||||||
(BE2_MAX_ICDS / 2 - (BE2_LOGOUTS + BE2_TMFS + BE2_NOPOUT_REQ))
|
(BE2_MAX_ICDS / 2 - (BE2_LOGOUTS + BE2_TMFS + BE2_NOPOUT_REQ))
|
||||||
|
|
||||||
|
#define MAX_CPUS 31
|
||||||
#define BEISCSI_SGLIST_ELEMENTS BE2_SGE
|
#define BEISCSI_SGLIST_ELEMENTS BE2_SGE
|
||||||
|
|
||||||
#define BEISCSI_MAX_CMNDS 1024 /* Max IO's per Ctrlr sht->can_queue */
|
#define BEISCSI_MAX_CMNDS 1024 /* Max IO's per Ctrlr sht->can_queue */
|
||||||
|
@ -79,7 +76,7 @@
|
||||||
#define BE_SENSE_INFO_SIZE 258
|
#define BE_SENSE_INFO_SIZE 258
|
||||||
#define BE_ISCSI_PDU_HEADER_SIZE 64
|
#define BE_ISCSI_PDU_HEADER_SIZE 64
|
||||||
#define BE_MIN_MEM_SIZE 16384
|
#define BE_MIN_MEM_SIZE 16384
|
||||||
|
#define MAX_CMD_SZ 65536
|
||||||
#define IIOC_SCSI_DATA 0x05 /* Write Operation */
|
#define IIOC_SCSI_DATA 0x05 /* Write Operation */
|
||||||
|
|
||||||
#define DBG_LVL 0x00000001
|
#define DBG_LVL 0x00000001
|
||||||
|
@ -100,6 +97,8 @@ do { \
|
||||||
} \
|
} \
|
||||||
} while (0);
|
} while (0);
|
||||||
|
|
||||||
|
#define BE_ADAPTER_UP 0x00000000
|
||||||
|
#define BE_ADAPTER_LINK_DOWN 0x00000001
|
||||||
/**
|
/**
|
||||||
* hardware needs the async PDU buffers to be posted in multiples of 8
|
* hardware needs the async PDU buffers to be posted in multiples of 8
|
||||||
* So have atleast 8 of them by default
|
* So have atleast 8 of them by default
|
||||||
|
@ -160,21 +159,19 @@ do { \
|
||||||
|
|
||||||
enum be_mem_enum {
|
enum be_mem_enum {
|
||||||
HWI_MEM_ADDN_CONTEXT,
|
HWI_MEM_ADDN_CONTEXT,
|
||||||
HWI_MEM_CQ,
|
|
||||||
HWI_MEM_EQ,
|
|
||||||
HWI_MEM_WRB,
|
HWI_MEM_WRB,
|
||||||
HWI_MEM_WRBH,
|
HWI_MEM_WRBH,
|
||||||
HWI_MEM_SGLH, /* 5 */
|
HWI_MEM_SGLH,
|
||||||
HWI_MEM_SGE,
|
HWI_MEM_SGE,
|
||||||
HWI_MEM_ASYNC_HEADER_BUF,
|
HWI_MEM_ASYNC_HEADER_BUF, /* 5 */
|
||||||
HWI_MEM_ASYNC_DATA_BUF,
|
HWI_MEM_ASYNC_DATA_BUF,
|
||||||
HWI_MEM_ASYNC_HEADER_RING,
|
HWI_MEM_ASYNC_HEADER_RING,
|
||||||
HWI_MEM_ASYNC_DATA_RING, /* 10 */
|
HWI_MEM_ASYNC_DATA_RING,
|
||||||
HWI_MEM_ASYNC_HEADER_HANDLE,
|
HWI_MEM_ASYNC_HEADER_HANDLE,
|
||||||
HWI_MEM_ASYNC_DATA_HANDLE,
|
HWI_MEM_ASYNC_DATA_HANDLE, /* 10 */
|
||||||
HWI_MEM_ASYNC_PDU_CONTEXT,
|
HWI_MEM_ASYNC_PDU_CONTEXT,
|
||||||
ISCSI_MEM_GLOBAL_HEADER,
|
ISCSI_MEM_GLOBAL_HEADER,
|
||||||
SE_MEM_MAX /* 15 */
|
SE_MEM_MAX
|
||||||
};
|
};
|
||||||
|
|
||||||
struct be_bus_address32 {
|
struct be_bus_address32 {
|
||||||
|
@ -212,6 +209,9 @@ struct be_mem_descriptor {
|
||||||
|
|
||||||
struct sgl_handle {
|
struct sgl_handle {
|
||||||
unsigned int sgl_index;
|
unsigned int sgl_index;
|
||||||
|
unsigned int type;
|
||||||
|
unsigned int cid;
|
||||||
|
struct iscsi_task *task;
|
||||||
struct iscsi_sge *pfrag;
|
struct iscsi_sge *pfrag;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -274,13 +274,17 @@ struct beiscsi_hba {
|
||||||
struct pci_dev *pcidev;
|
struct pci_dev *pcidev;
|
||||||
unsigned int state;
|
unsigned int state;
|
||||||
unsigned short asic_revision;
|
unsigned short asic_revision;
|
||||||
struct blk_iopoll iopoll;
|
unsigned int num_cpus;
|
||||||
|
unsigned int nxt_cqid;
|
||||||
|
struct msix_entry msix_entries[MAX_CPUS + 1];
|
||||||
|
bool msix_enabled;
|
||||||
struct be_mem_descriptor *init_mem;
|
struct be_mem_descriptor *init_mem;
|
||||||
|
|
||||||
unsigned short io_sgl_alloc_index;
|
unsigned short io_sgl_alloc_index;
|
||||||
unsigned short io_sgl_free_index;
|
unsigned short io_sgl_free_index;
|
||||||
unsigned short io_sgl_hndl_avbl;
|
unsigned short io_sgl_hndl_avbl;
|
||||||
struct sgl_handle **io_sgl_hndl_base;
|
struct sgl_handle **io_sgl_hndl_base;
|
||||||
|
struct sgl_handle **sgl_hndl_array;
|
||||||
|
|
||||||
unsigned short eh_sgl_alloc_index;
|
unsigned short eh_sgl_alloc_index;
|
||||||
unsigned short eh_sgl_free_index;
|
unsigned short eh_sgl_free_index;
|
||||||
|
@ -315,6 +319,7 @@ struct beiscsi_hba {
|
||||||
unsigned short cid_alloc;
|
unsigned short cid_alloc;
|
||||||
unsigned short cid_free;
|
unsigned short cid_free;
|
||||||
unsigned short avlbl_cids;
|
unsigned short avlbl_cids;
|
||||||
|
unsigned short iscsi_features;
|
||||||
spinlock_t cid_lock;
|
spinlock_t cid_lock;
|
||||||
} fw_config;
|
} fw_config;
|
||||||
|
|
||||||
|
@ -343,6 +348,7 @@ struct beiscsi_conn {
|
||||||
unsigned short login_in_progress;
|
unsigned short login_in_progress;
|
||||||
struct sgl_handle *plogin_sgl_handle;
|
struct sgl_handle *plogin_sgl_handle;
|
||||||
struct beiscsi_session *beiscsi_sess;
|
struct beiscsi_session *beiscsi_sess;
|
||||||
|
struct iscsi_task *task;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* This structure is used by the chip */
|
/* This structure is used by the chip */
|
||||||
|
@ -390,7 +396,7 @@ struct beiscsi_io_task {
|
||||||
unsigned int flags;
|
unsigned int flags;
|
||||||
unsigned short cid;
|
unsigned short cid;
|
||||||
unsigned short header_len;
|
unsigned short header_len;
|
||||||
|
itt_t libiscsi_itt;
|
||||||
struct be_cmd_bhs *cmd_bhs;
|
struct be_cmd_bhs *cmd_bhs;
|
||||||
struct be_bus_address bhs_pa;
|
struct be_bus_address bhs_pa;
|
||||||
unsigned short bhs_len;
|
unsigned short bhs_len;
|
||||||
|
@ -599,7 +605,6 @@ struct amap_cq_db {
|
||||||
|
|
||||||
void beiscsi_process_eq(struct beiscsi_hba *phba);
|
void beiscsi_process_eq(struct beiscsi_hba *phba);
|
||||||
|
|
||||||
|
|
||||||
struct iscsi_wrb {
|
struct iscsi_wrb {
|
||||||
u32 dw[16];
|
u32 dw[16];
|
||||||
} __packed;
|
} __packed;
|
||||||
|
@ -820,10 +825,12 @@ struct wrb_handle {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct hwi_context_memory {
|
struct hwi_context_memory {
|
||||||
struct be_eq_obj be_eq;
|
/* Adaptive interrupt coalescing (AIC) info */
|
||||||
struct be_queue_info be_cq;
|
u16 min_eqd; /* in usecs */
|
||||||
struct be_queue_info be_mcc_cq;
|
u16 max_eqd; /* in usecs */
|
||||||
struct be_queue_info be_mcc;
|
u16 cur_eqd; /* in usecs */
|
||||||
|
struct be_eq_obj be_eq[MAX_CPUS];
|
||||||
|
struct be_queue_info be_cq[MAX_CPUS];
|
||||||
|
|
||||||
struct be_queue_info be_def_hdrq;
|
struct be_queue_info be_def_hdrq;
|
||||||
struct be_queue_info be_def_dataq;
|
struct be_queue_info be_def_dataq;
|
||||||
|
|
|
@ -35,7 +35,6 @@ unsigned char mgmt_get_fw_config(struct be_ctrl_info *ctrl,
|
||||||
|
|
||||||
be_cmd_hdr_prepare(&req->hdr, CMD_SUBSYSTEM_COMMON,
|
be_cmd_hdr_prepare(&req->hdr, CMD_SUBSYSTEM_COMMON,
|
||||||
OPCODE_COMMON_QUERY_FIRMWARE_CONFIG, sizeof(*req));
|
OPCODE_COMMON_QUERY_FIRMWARE_CONFIG, sizeof(*req));
|
||||||
|
|
||||||
status = be_mbox_notify(ctrl);
|
status = be_mbox_notify(ctrl);
|
||||||
if (!status) {
|
if (!status) {
|
||||||
struct be_fw_cfg *pfw_cfg;
|
struct be_fw_cfg *pfw_cfg;
|
||||||
|
@ -58,7 +57,8 @@ unsigned char mgmt_get_fw_config(struct be_ctrl_info *ctrl,
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned char mgmt_check_supported_fw(struct be_ctrl_info *ctrl)
|
unsigned char mgmt_check_supported_fw(struct be_ctrl_info *ctrl,
|
||||||
|
struct beiscsi_hba *phba)
|
||||||
{
|
{
|
||||||
struct be_dma_mem nonemb_cmd;
|
struct be_dma_mem nonemb_cmd;
|
||||||
struct be_mcc_wrb *wrb = wrb_from_mbox(&ctrl->mbox_mem);
|
struct be_mcc_wrb *wrb = wrb_from_mbox(&ctrl->mbox_mem);
|
||||||
|
@ -85,7 +85,6 @@ unsigned char mgmt_check_supported_fw(struct be_ctrl_info *ctrl)
|
||||||
sge->pa_hi = cpu_to_le32(upper_32_bits(nonemb_cmd.dma));
|
sge->pa_hi = cpu_to_le32(upper_32_bits(nonemb_cmd.dma));
|
||||||
sge->pa_lo = cpu_to_le32(nonemb_cmd.dma & 0xFFFFFFFF);
|
sge->pa_lo = cpu_to_le32(nonemb_cmd.dma & 0xFFFFFFFF);
|
||||||
sge->len = cpu_to_le32(nonemb_cmd.size);
|
sge->len = cpu_to_le32(nonemb_cmd.size);
|
||||||
|
|
||||||
status = be_mbox_notify(ctrl);
|
status = be_mbox_notify(ctrl);
|
||||||
if (!status) {
|
if (!status) {
|
||||||
struct be_mgmt_controller_attributes_resp *resp = nonemb_cmd.va;
|
struct be_mgmt_controller_attributes_resp *resp = nonemb_cmd.va;
|
||||||
|
@ -95,21 +94,25 @@ unsigned char mgmt_check_supported_fw(struct be_ctrl_info *ctrl)
|
||||||
resp->params.hba_attribs.firmware_version_string);
|
resp->params.hba_attribs.firmware_version_string);
|
||||||
SE_DEBUG(DBG_LVL_8,
|
SE_DEBUG(DBG_LVL_8,
|
||||||
"Developer Build, not performing version check...\n");
|
"Developer Build, not performing version check...\n");
|
||||||
|
phba->fw_config.iscsi_features =
|
||||||
|
resp->params.hba_attribs.iscsi_features;
|
||||||
|
SE_DEBUG(DBG_LVL_8, " phba->fw_config.iscsi_features = %d\n",
|
||||||
|
phba->fw_config.iscsi_features);
|
||||||
} else
|
} else
|
||||||
SE_DEBUG(DBG_LVL_1, " Failed in mgmt_check_supported_fw\n");
|
SE_DEBUG(DBG_LVL_1, " Failed in mgmt_check_supported_fw\n");
|
||||||
|
spin_unlock(&ctrl->mbox_lock);
|
||||||
if (nonemb_cmd.va)
|
if (nonemb_cmd.va)
|
||||||
pci_free_consistent(ctrl->pdev, nonemb_cmd.size,
|
pci_free_consistent(ctrl->pdev, nonemb_cmd.size,
|
||||||
nonemb_cmd.va, nonemb_cmd.dma);
|
nonemb_cmd.va, nonemb_cmd.dma);
|
||||||
|
|
||||||
spin_unlock(&ctrl->mbox_lock);
|
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
unsigned char mgmt_epfw_cleanup(struct beiscsi_hba *phba, unsigned short chute)
|
unsigned char mgmt_epfw_cleanup(struct beiscsi_hba *phba, unsigned short chute)
|
||||||
{
|
{
|
||||||
struct be_ctrl_info *ctrl = &phba->ctrl;
|
struct be_ctrl_info *ctrl = &phba->ctrl;
|
||||||
struct be_mcc_wrb *wrb = wrb_from_mbox(&ctrl->mbox_mem);
|
struct be_mcc_wrb *wrb = wrb_from_mccq(phba);
|
||||||
struct iscsi_cleanup_req *req = embedded_payload(wrb);
|
struct iscsi_cleanup_req *req = embedded_payload(wrb);
|
||||||
int status = 0;
|
int status = 0;
|
||||||
|
|
||||||
|
@ -124,7 +127,7 @@ unsigned char mgmt_epfw_cleanup(struct beiscsi_hba *phba, unsigned short chute)
|
||||||
req->hdr_ring_id = 0;
|
req->hdr_ring_id = 0;
|
||||||
req->data_ring_id = 0;
|
req->data_ring_id = 0;
|
||||||
|
|
||||||
status = be_mbox_notify(ctrl);
|
status = be_mcc_notify_wait(phba);
|
||||||
if (status)
|
if (status)
|
||||||
shost_printk(KERN_WARNING, phba->shost,
|
shost_printk(KERN_WARNING, phba->shost,
|
||||||
" mgmt_epfw_cleanup , FAILED\n");
|
" mgmt_epfw_cleanup , FAILED\n");
|
||||||
|
@ -137,7 +140,7 @@ unsigned char mgmt_invalidate_icds(struct beiscsi_hba *phba,
|
||||||
{
|
{
|
||||||
struct be_dma_mem nonemb_cmd;
|
struct be_dma_mem nonemb_cmd;
|
||||||
struct be_ctrl_info *ctrl = &phba->ctrl;
|
struct be_ctrl_info *ctrl = &phba->ctrl;
|
||||||
struct be_mcc_wrb *wrb = wrb_from_mbox(&ctrl->mbox_mem);
|
struct be_mcc_wrb *wrb = wrb_from_mccq(phba);
|
||||||
struct be_sge *sge = nonembedded_sgl(wrb);
|
struct be_sge *sge = nonembedded_sgl(wrb);
|
||||||
struct invalidate_commands_params_in *req;
|
struct invalidate_commands_params_in *req;
|
||||||
int status = 0;
|
int status = 0;
|
||||||
|
@ -169,7 +172,7 @@ unsigned char mgmt_invalidate_icds(struct beiscsi_hba *phba,
|
||||||
sge->pa_lo = cpu_to_le32(nonemb_cmd.dma & 0xFFFFFFFF);
|
sge->pa_lo = cpu_to_le32(nonemb_cmd.dma & 0xFFFFFFFF);
|
||||||
sge->len = cpu_to_le32(nonemb_cmd.size);
|
sge->len = cpu_to_le32(nonemb_cmd.size);
|
||||||
|
|
||||||
status = be_mbox_notify(ctrl);
|
status = be_mcc_notify_wait(phba);
|
||||||
if (status)
|
if (status)
|
||||||
SE_DEBUG(DBG_LVL_1, "ICDS Invalidation Failed\n");
|
SE_DEBUG(DBG_LVL_1, "ICDS Invalidation Failed\n");
|
||||||
spin_unlock(&ctrl->mbox_lock);
|
spin_unlock(&ctrl->mbox_lock);
|
||||||
|
@ -186,7 +189,7 @@ unsigned char mgmt_invalidate_connection(struct beiscsi_hba *phba,
|
||||||
unsigned short savecfg_flag)
|
unsigned short savecfg_flag)
|
||||||
{
|
{
|
||||||
struct be_ctrl_info *ctrl = &phba->ctrl;
|
struct be_ctrl_info *ctrl = &phba->ctrl;
|
||||||
struct be_mcc_wrb *wrb = wrb_from_mbox(&ctrl->mbox_mem);
|
struct be_mcc_wrb *wrb = wrb_from_mccq(phba);
|
||||||
struct iscsi_invalidate_connection_params_in *req =
|
struct iscsi_invalidate_connection_params_in *req =
|
||||||
embedded_payload(wrb);
|
embedded_payload(wrb);
|
||||||
int status = 0;
|
int status = 0;
|
||||||
|
@ -205,7 +208,7 @@ unsigned char mgmt_invalidate_connection(struct beiscsi_hba *phba,
|
||||||
else
|
else
|
||||||
req->cleanup_type = CMD_ISCSI_CONNECTION_INVALIDATE;
|
req->cleanup_type = CMD_ISCSI_CONNECTION_INVALIDATE;
|
||||||
req->save_cfg = savecfg_flag;
|
req->save_cfg = savecfg_flag;
|
||||||
status = be_mbox_notify(ctrl);
|
status = be_mcc_notify_wait(phba);
|
||||||
if (status)
|
if (status)
|
||||||
SE_DEBUG(DBG_LVL_1, "Invalidation Failed\n");
|
SE_DEBUG(DBG_LVL_1, "Invalidation Failed\n");
|
||||||
|
|
||||||
|
@ -217,7 +220,7 @@ unsigned char mgmt_upload_connection(struct beiscsi_hba *phba,
|
||||||
unsigned short cid, unsigned int upload_flag)
|
unsigned short cid, unsigned int upload_flag)
|
||||||
{
|
{
|
||||||
struct be_ctrl_info *ctrl = &phba->ctrl;
|
struct be_ctrl_info *ctrl = &phba->ctrl;
|
||||||
struct be_mcc_wrb *wrb = wrb_from_mbox(&ctrl->mbox_mem);
|
struct be_mcc_wrb *wrb = wrb_from_mccq(phba);
|
||||||
struct tcp_upload_params_in *req = embedded_payload(wrb);
|
struct tcp_upload_params_in *req = embedded_payload(wrb);
|
||||||
int status = 0;
|
int status = 0;
|
||||||
|
|
||||||
|
@ -229,7 +232,7 @@ unsigned char mgmt_upload_connection(struct beiscsi_hba *phba,
|
||||||
OPCODE_COMMON_TCP_UPLOAD, sizeof(*req));
|
OPCODE_COMMON_TCP_UPLOAD, sizeof(*req));
|
||||||
req->id = (unsigned short)cid;
|
req->id = (unsigned short)cid;
|
||||||
req->upload_type = (unsigned char)upload_flag;
|
req->upload_type = (unsigned char)upload_flag;
|
||||||
status = be_mbox_notify(ctrl);
|
status = be_mcc_notify_wait(phba);
|
||||||
if (status)
|
if (status)
|
||||||
SE_DEBUG(DBG_LVL_1, "mgmt_upload_connection Failed\n");
|
SE_DEBUG(DBG_LVL_1, "mgmt_upload_connection Failed\n");
|
||||||
spin_unlock(&ctrl->mbox_lock);
|
spin_unlock(&ctrl->mbox_lock);
|
||||||
|
@ -245,13 +248,14 @@ int mgmt_open_connection(struct beiscsi_hba *phba,
|
||||||
struct sockaddr_in *daddr_in = (struct sockaddr_in *)dst_addr;
|
struct sockaddr_in *daddr_in = (struct sockaddr_in *)dst_addr;
|
||||||
struct sockaddr_in6 *daddr_in6 = (struct sockaddr_in6 *)dst_addr;
|
struct sockaddr_in6 *daddr_in6 = (struct sockaddr_in6 *)dst_addr;
|
||||||
struct be_ctrl_info *ctrl = &phba->ctrl;
|
struct be_ctrl_info *ctrl = &phba->ctrl;
|
||||||
struct be_mcc_wrb *wrb = wrb_from_mbox(&ctrl->mbox_mem);
|
struct be_mcc_wrb *wrb = wrb_from_mccq(phba);
|
||||||
struct tcp_connect_and_offload_in *req = embedded_payload(wrb);
|
struct tcp_connect_and_offload_in *req = embedded_payload(wrb);
|
||||||
unsigned short def_hdr_id;
|
unsigned short def_hdr_id;
|
||||||
unsigned short def_data_id;
|
unsigned short def_data_id;
|
||||||
struct phys_addr template_address = { 0, 0 };
|
struct phys_addr template_address = { 0, 0 };
|
||||||
struct phys_addr *ptemplate_address;
|
struct phys_addr *ptemplate_address;
|
||||||
int status = 0;
|
int status = 0;
|
||||||
|
unsigned int i;
|
||||||
unsigned short cid = beiscsi_ep->ep_cid;
|
unsigned short cid = beiscsi_ep->ep_cid;
|
||||||
|
|
||||||
phwi_ctrlr = phba->phwi_ctrlr;
|
phwi_ctrlr = phba->phwi_ctrlr;
|
||||||
|
@ -296,14 +300,18 @@ int mgmt_open_connection(struct beiscsi_hba *phba,
|
||||||
|
|
||||||
}
|
}
|
||||||
req->cid = cid;
|
req->cid = cid;
|
||||||
req->cq_id = phwi_context->be_cq.id;
|
i = phba->nxt_cqid++;
|
||||||
|
if (phba->nxt_cqid == phba->num_cpus)
|
||||||
|
phba->nxt_cqid = 0;
|
||||||
|
req->cq_id = phwi_context->be_cq[i].id;
|
||||||
|
SE_DEBUG(DBG_LVL_8, "i=%d cq_id=%d \n", i, req->cq_id);
|
||||||
req->defq_id = def_hdr_id;
|
req->defq_id = def_hdr_id;
|
||||||
req->hdr_ring_id = def_hdr_id;
|
req->hdr_ring_id = def_hdr_id;
|
||||||
req->data_ring_id = def_data_id;
|
req->data_ring_id = def_data_id;
|
||||||
req->do_offload = 1;
|
req->do_offload = 1;
|
||||||
req->dataout_template_pa.lo = ptemplate_address->lo;
|
req->dataout_template_pa.lo = ptemplate_address->lo;
|
||||||
req->dataout_template_pa.hi = ptemplate_address->hi;
|
req->dataout_template_pa.hi = ptemplate_address->hi;
|
||||||
status = be_mbox_notify(ctrl);
|
status = be_mcc_notify_wait(phba);
|
||||||
if (!status) {
|
if (!status) {
|
||||||
struct iscsi_endpoint *ep;
|
struct iscsi_endpoint *ep;
|
||||||
struct tcp_connect_and_offload_out *ptcpcnct_out =
|
struct tcp_connect_and_offload_out *ptcpcnct_out =
|
||||||
|
@ -311,7 +319,7 @@ int mgmt_open_connection(struct beiscsi_hba *phba,
|
||||||
|
|
||||||
ep = phba->ep_array[ptcpcnct_out->cid];
|
ep = phba->ep_array[ptcpcnct_out->cid];
|
||||||
beiscsi_ep = ep->dd_data;
|
beiscsi_ep = ep->dd_data;
|
||||||
beiscsi_ep->fw_handle = 0;
|
beiscsi_ep->fw_handle = ptcpcnct_out->connection_handle;
|
||||||
beiscsi_ep->cid_vld = 1;
|
beiscsi_ep->cid_vld = 1;
|
||||||
SE_DEBUG(DBG_LVL_8, "mgmt_open_connection Success\n");
|
SE_DEBUG(DBG_LVL_8, "mgmt_open_connection Success\n");
|
||||||
} else
|
} else
|
||||||
|
@ -319,3 +327,30 @@ int mgmt_open_connection(struct beiscsi_hba *phba,
|
||||||
spin_unlock(&ctrl->mbox_lock);
|
spin_unlock(&ctrl->mbox_lock);
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int be_cmd_get_mac_addr(struct beiscsi_hba *phba, u8 *mac_addr)
|
||||||
|
{
|
||||||
|
struct be_ctrl_info *ctrl = &phba->ctrl;
|
||||||
|
struct be_mcc_wrb *wrb = wrb_from_mccq(phba);
|
||||||
|
struct be_cmd_req_get_mac_addr *req = embedded_payload(wrb);
|
||||||
|
int status;
|
||||||
|
|
||||||
|
SE_DEBUG(DBG_LVL_8, "In be_cmd_get_mac_addr\n");
|
||||||
|
spin_lock(&ctrl->mbox_lock);
|
||||||
|
memset(wrb, 0, sizeof(*wrb));
|
||||||
|
be_wrb_hdr_prepare(wrb, sizeof(*req), true, 0);
|
||||||
|
be_cmd_hdr_prepare(&req->hdr, CMD_SUBSYSTEM_ISCSI,
|
||||||
|
OPCODE_COMMON_ISCSI_NTWK_GET_NIC_CONFIG,
|
||||||
|
sizeof(*req));
|
||||||
|
|
||||||
|
status = be_mcc_notify_wait(phba);
|
||||||
|
if (!status) {
|
||||||
|
struct be_cmd_resp_get_mac_addr *resp = embedded_payload(wrb);
|
||||||
|
|
||||||
|
memcpy(mac_addr, resp->mac_address, ETH_ALEN);
|
||||||
|
}
|
||||||
|
|
||||||
|
spin_unlock(&ctrl->mbox_lock);
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -175,7 +175,9 @@ struct mgmt_hba_attributes {
|
||||||
u8 phy_port;
|
u8 phy_port;
|
||||||
u32 firmware_post_status;
|
u32 firmware_post_status;
|
||||||
u32 hba_mtu[8];
|
u32 hba_mtu[8];
|
||||||
u32 future_u32[4];
|
u8 iscsi_features;
|
||||||
|
u8 future_u8[3];
|
||||||
|
u32 future_u32[3];
|
||||||
} __packed;
|
} __packed;
|
||||||
|
|
||||||
struct mgmt_controller_attributes {
|
struct mgmt_controller_attributes {
|
||||||
|
@ -246,4 +248,8 @@ unsigned char mgmt_invalidate_connection(struct beiscsi_hba *phba,
|
||||||
unsigned short cid,
|
unsigned short cid,
|
||||||
unsigned short issue_reset,
|
unsigned short issue_reset,
|
||||||
unsigned short savecfg_flag);
|
unsigned short savecfg_flag);
|
||||||
|
|
||||||
|
unsigned char mgmt_fw_cmd(struct be_ctrl_info *ctrl,
|
||||||
|
struct beiscsi_hba *phba,
|
||||||
|
char *buf, unsigned int len);
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -51,7 +51,7 @@ bfad_int_to_lun(u32 luno)
|
||||||
lun.bfa_lun = 0;
|
lun.bfa_lun = 0;
|
||||||
lun.scsi_lun[0] = bfa_os_htons(luno);
|
lun.scsi_lun[0] = bfa_os_htons(luno);
|
||||||
|
|
||||||
return (lun.bfa_lun);
|
return lun.bfa_lun;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -68,7 +68,7 @@ bfa_cb_ioim_get_cdb(struct bfad_ioim_s *dio)
|
||||||
{
|
{
|
||||||
struct scsi_cmnd *cmnd = (struct scsi_cmnd *)dio;
|
struct scsi_cmnd *cmnd = (struct scsi_cmnd *)dio;
|
||||||
|
|
||||||
return ((u8 *) cmnd->cmnd);
|
return (u8 *) cmnd->cmnd;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -97,7 +97,7 @@ bfa_cb_ioim_get_size(struct bfad_ioim_s *dio)
|
||||||
{
|
{
|
||||||
struct scsi_cmnd *cmnd = (struct scsi_cmnd *)dio;
|
struct scsi_cmnd *cmnd = (struct scsi_cmnd *)dio;
|
||||||
|
|
||||||
return (scsi_bufflen(cmnd));
|
return scsi_bufflen(cmnd);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -129,7 +129,7 @@ bfa_cb_ioim_get_sgaddr(struct bfad_ioim_s *dio, int sgeid)
|
||||||
sge = (struct scatterlist *)scsi_sglist(cmnd) + sgeid;
|
sge = (struct scatterlist *)scsi_sglist(cmnd) + sgeid;
|
||||||
addr = (u64) sg_dma_address(sge);
|
addr = (u64) sg_dma_address(sge);
|
||||||
|
|
||||||
return (*(union bfi_addr_u *) &addr);
|
return *((union bfi_addr_u *) &addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline u32
|
static inline u32
|
||||||
|
@ -197,7 +197,7 @@ bfa_cb_ioim_get_cdblen(struct bfad_ioim_s *dio)
|
||||||
{
|
{
|
||||||
struct scsi_cmnd *cmnd = (struct scsi_cmnd *)dio;
|
struct scsi_cmnd *cmnd = (struct scsi_cmnd *)dio;
|
||||||
|
|
||||||
return (cmnd->cmd_len);
|
return cmnd->cmd_len;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -228,7 +228,7 @@ bfa_cee_reset_stats_isr(struct bfa_cee_s *cee, bfa_status_t status)
|
||||||
u32
|
u32
|
||||||
bfa_cee_meminfo(void)
|
bfa_cee_meminfo(void)
|
||||||
{
|
{
|
||||||
return (bfa_cee_attr_meminfo() + bfa_cee_stats_meminfo());
|
return bfa_cee_attr_meminfo() + bfa_cee_stats_meminfo();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -47,12 +47,12 @@ bfa_q_is_on_q_func(struct list_head *q, struct list_head *qe)
|
||||||
tqe = bfa_q_next(q);
|
tqe = bfa_q_next(q);
|
||||||
while (tqe != q) {
|
while (tqe != q) {
|
||||||
if (tqe == qe)
|
if (tqe == qe)
|
||||||
return (1);
|
return 1;
|
||||||
tqe = bfa_q_next(tqe);
|
tqe = bfa_q_next(tqe);
|
||||||
if (tqe == NULL)
|
if (tqe == NULL)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
return (0);
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -131,7 +131,7 @@ bfa_fcpim_path_tov_get(struct bfa_s *bfa)
|
||||||
{
|
{
|
||||||
struct bfa_fcpim_mod_s *fcpim = BFA_FCPIM_MOD(bfa);
|
struct bfa_fcpim_mod_s *fcpim = BFA_FCPIM_MOD(bfa);
|
||||||
|
|
||||||
return (fcpim->path_tov / 1000);
|
return fcpim->path_tov / 1000;
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_status_t
|
bfa_status_t
|
||||||
|
@ -169,7 +169,7 @@ bfa_fcpim_qdepth_get(struct bfa_s *bfa)
|
||||||
{
|
{
|
||||||
struct bfa_fcpim_mod_s *fcpim = BFA_FCPIM_MOD(bfa);
|
struct bfa_fcpim_mod_s *fcpim = BFA_FCPIM_MOD(bfa);
|
||||||
|
|
||||||
return (fcpim->q_depth);
|
return fcpim->q_depth;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -35,7 +35,7 @@
|
||||||
#define BFA_FCPIM_PATHTOV_MAX (90 * 1000) /* in millisecs */
|
#define BFA_FCPIM_PATHTOV_MAX (90 * 1000) /* in millisecs */
|
||||||
|
|
||||||
#define bfa_fcpim_stats(__fcpim, __stats) \
|
#define bfa_fcpim_stats(__fcpim, __stats) \
|
||||||
(__fcpim)->stats.__stats ++
|
((__fcpim)->stats.__stats++)
|
||||||
|
|
||||||
struct bfa_fcpim_mod_s {
|
struct bfa_fcpim_mod_s {
|
||||||
struct bfa_s *bfa;
|
struct bfa_s *bfa;
|
||||||
|
@ -143,7 +143,7 @@ struct bfa_itnim_s {
|
||||||
struct bfa_itnim_hal_stats_s stats;
|
struct bfa_itnim_hal_stats_s stats;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define bfa_itnim_is_online(_itnim) (_itnim)->is_online
|
#define bfa_itnim_is_online(_itnim) ((_itnim)->is_online)
|
||||||
#define BFA_FCPIM_MOD(_hal) (&(_hal)->modules.fcpim_mod)
|
#define BFA_FCPIM_MOD(_hal) (&(_hal)->modules.fcpim_mod)
|
||||||
#define BFA_IOIM_FROM_TAG(_fcpim, _iotag) \
|
#define BFA_IOIM_FROM_TAG(_fcpim, _iotag) \
|
||||||
(&fcpim->ioim_arr[_iotag])
|
(&fcpim->ioim_arr[_iotag])
|
||||||
|
|
|
@ -388,32 +388,29 @@ bfa_pport_sm_linkup(struct bfa_pport_s *pport, enum bfa_pport_sm_event event)
|
||||||
bfa_pport_callback(pport, BFA_PPORT_LINKDOWN);
|
bfa_pport_callback(pport, BFA_PPORT_LINKDOWN);
|
||||||
bfa_plog_str(pport->bfa->plog, BFA_PL_MID_HAL,
|
bfa_plog_str(pport->bfa->plog, BFA_PL_MID_HAL,
|
||||||
BFA_PL_EID_PORT_ST_CHANGE, 0, "Port Linkdown");
|
BFA_PL_EID_PORT_ST_CHANGE, 0, "Port Linkdown");
|
||||||
if (BFA_PORT_IS_DISABLED(pport->bfa)) {
|
if (BFA_PORT_IS_DISABLED(pport->bfa))
|
||||||
bfa_pport_aen_post(pport, BFA_PORT_AEN_OFFLINE);
|
bfa_pport_aen_post(pport, BFA_PORT_AEN_OFFLINE);
|
||||||
} else {
|
else
|
||||||
bfa_pport_aen_post(pport, BFA_PORT_AEN_DISCONNECT);
|
bfa_pport_aen_post(pport, BFA_PORT_AEN_DISCONNECT);
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case BFA_PPORT_SM_STOP:
|
case BFA_PPORT_SM_STOP:
|
||||||
bfa_sm_set_state(pport, bfa_pport_sm_stopped);
|
bfa_sm_set_state(pport, bfa_pport_sm_stopped);
|
||||||
bfa_pport_reset_linkinfo(pport);
|
bfa_pport_reset_linkinfo(pport);
|
||||||
if (BFA_PORT_IS_DISABLED(pport->bfa)) {
|
if (BFA_PORT_IS_DISABLED(pport->bfa))
|
||||||
bfa_pport_aen_post(pport, BFA_PORT_AEN_OFFLINE);
|
bfa_pport_aen_post(pport, BFA_PORT_AEN_OFFLINE);
|
||||||
} else {
|
else
|
||||||
bfa_pport_aen_post(pport, BFA_PORT_AEN_DISCONNECT);
|
bfa_pport_aen_post(pport, BFA_PORT_AEN_DISCONNECT);
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case BFA_PPORT_SM_HWFAIL:
|
case BFA_PPORT_SM_HWFAIL:
|
||||||
bfa_sm_set_state(pport, bfa_pport_sm_iocdown);
|
bfa_sm_set_state(pport, bfa_pport_sm_iocdown);
|
||||||
bfa_pport_reset_linkinfo(pport);
|
bfa_pport_reset_linkinfo(pport);
|
||||||
bfa_pport_callback(pport, BFA_PPORT_LINKDOWN);
|
bfa_pport_callback(pport, BFA_PPORT_LINKDOWN);
|
||||||
if (BFA_PORT_IS_DISABLED(pport->bfa)) {
|
if (BFA_PORT_IS_DISABLED(pport->bfa))
|
||||||
bfa_pport_aen_post(pport, BFA_PORT_AEN_OFFLINE);
|
bfa_pport_aen_post(pport, BFA_PORT_AEN_OFFLINE);
|
||||||
} else {
|
else
|
||||||
bfa_pport_aen_post(pport, BFA_PORT_AEN_DISCONNECT);
|
bfa_pport_aen_post(pport, BFA_PORT_AEN_DISCONNECT);
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -999,10 +996,10 @@ bfa_pport_enable(struct bfa_s *bfa)
|
||||||
struct bfa_pport_s *pport = BFA_PORT_MOD(bfa);
|
struct bfa_pport_s *pport = BFA_PORT_MOD(bfa);
|
||||||
|
|
||||||
if (pport->diag_busy)
|
if (pport->diag_busy)
|
||||||
return (BFA_STATUS_DIAG_BUSY);
|
return BFA_STATUS_DIAG_BUSY;
|
||||||
else if (bfa_sm_cmp_state
|
else if (bfa_sm_cmp_state
|
||||||
(BFA_PORT_MOD(bfa), bfa_pport_sm_disabling_qwait))
|
(BFA_PORT_MOD(bfa), bfa_pport_sm_disabling_qwait))
|
||||||
return (BFA_STATUS_DEVBUSY);
|
return BFA_STATUS_DEVBUSY;
|
||||||
|
|
||||||
bfa_sm_send_event(BFA_PORT_MOD(bfa), BFA_PPORT_SM_ENABLE);
|
bfa_sm_send_event(BFA_PORT_MOD(bfa), BFA_PPORT_SM_ENABLE);
|
||||||
return BFA_STATUS_OK;
|
return BFA_STATUS_OK;
|
||||||
|
@ -1032,7 +1029,7 @@ bfa_pport_cfg_speed(struct bfa_s *bfa, enum bfa_pport_speed speed)
|
||||||
|
|
||||||
pport->cfg.speed = speed;
|
pport->cfg.speed = speed;
|
||||||
|
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1068,7 +1065,7 @@ bfa_pport_cfg_topology(struct bfa_s *bfa, enum bfa_pport_topology topology)
|
||||||
}
|
}
|
||||||
|
|
||||||
pport->cfg.topology = topology;
|
pport->cfg.topology = topology;
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1094,7 +1091,7 @@ bfa_pport_cfg_hardalpa(struct bfa_s *bfa, u8 alpa)
|
||||||
pport->cfg.cfg_hardalpa = BFA_TRUE;
|
pport->cfg.cfg_hardalpa = BFA_TRUE;
|
||||||
pport->cfg.hardalpa = alpa;
|
pport->cfg.hardalpa = alpa;
|
||||||
|
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_status_t
|
bfa_status_t
|
||||||
|
@ -1106,7 +1103,7 @@ bfa_pport_clr_hardalpa(struct bfa_s *bfa)
|
||||||
bfa_trc(bfa, pport->cfg.hardalpa);
|
bfa_trc(bfa, pport->cfg.hardalpa);
|
||||||
|
|
||||||
pport->cfg.cfg_hardalpa = BFA_FALSE;
|
pport->cfg.cfg_hardalpa = BFA_FALSE;
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_boolean_t
|
bfa_boolean_t
|
||||||
|
@ -1138,16 +1135,16 @@ bfa_pport_cfg_maxfrsize(struct bfa_s *bfa, u16 maxfrsize)
|
||||||
* with in range
|
* with in range
|
||||||
*/
|
*/
|
||||||
if ((maxfrsize > FC_MAX_PDUSZ) || (maxfrsize < FC_MIN_PDUSZ))
|
if ((maxfrsize > FC_MAX_PDUSZ) || (maxfrsize < FC_MIN_PDUSZ))
|
||||||
return (BFA_STATUS_INVLD_DFSZ);
|
return BFA_STATUS_INVLD_DFSZ;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* power of 2, if not the max frame size of 2112
|
* power of 2, if not the max frame size of 2112
|
||||||
*/
|
*/
|
||||||
if ((maxfrsize != FC_MAX_PDUSZ) && (maxfrsize & (maxfrsize - 1)))
|
if ((maxfrsize != FC_MAX_PDUSZ) && (maxfrsize & (maxfrsize - 1)))
|
||||||
return (BFA_STATUS_INVLD_DFSZ);
|
return BFA_STATUS_INVLD_DFSZ;
|
||||||
|
|
||||||
pport->cfg.maxfrsize = maxfrsize;
|
pport->cfg.maxfrsize = maxfrsize;
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1415,7 +1412,7 @@ bfa_pport_get_stats(struct bfa_s *bfa, union bfa_pport_stats_u *stats,
|
||||||
|
|
||||||
if (port->stats_busy) {
|
if (port->stats_busy) {
|
||||||
bfa_trc(bfa, port->stats_busy);
|
bfa_trc(bfa, port->stats_busy);
|
||||||
return (BFA_STATUS_DEVBUSY);
|
return BFA_STATUS_DEVBUSY;
|
||||||
}
|
}
|
||||||
|
|
||||||
port->stats_busy = BFA_TRUE;
|
port->stats_busy = BFA_TRUE;
|
||||||
|
@ -1427,7 +1424,7 @@ bfa_pport_get_stats(struct bfa_s *bfa, union bfa_pport_stats_u *stats,
|
||||||
|
|
||||||
bfa_timer_start(bfa, &port->timer, bfa_port_stats_timeout, port,
|
bfa_timer_start(bfa, &port->timer, bfa_port_stats_timeout, port,
|
||||||
BFA_PORT_STATS_TOV);
|
BFA_PORT_STATS_TOV);
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_status_t
|
bfa_status_t
|
||||||
|
@ -1437,7 +1434,7 @@ bfa_pport_clear_stats(struct bfa_s *bfa, bfa_cb_pport_t cbfn, void *cbarg)
|
||||||
|
|
||||||
if (port->stats_busy) {
|
if (port->stats_busy) {
|
||||||
bfa_trc(bfa, port->stats_busy);
|
bfa_trc(bfa, port->stats_busy);
|
||||||
return (BFA_STATUS_DEVBUSY);
|
return BFA_STATUS_DEVBUSY;
|
||||||
}
|
}
|
||||||
|
|
||||||
port->stats_busy = BFA_TRUE;
|
port->stats_busy = BFA_TRUE;
|
||||||
|
@ -1448,7 +1445,7 @@ bfa_pport_clear_stats(struct bfa_s *bfa, bfa_cb_pport_t cbfn, void *cbarg)
|
||||||
|
|
||||||
bfa_timer_start(bfa, &port->timer, bfa_port_stats_clr_timeout, port,
|
bfa_timer_start(bfa, &port->timer, bfa_port_stats_clr_timeout, port,
|
||||||
BFA_PORT_STATS_TOV);
|
BFA_PORT_STATS_TOV);
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_status_t
|
bfa_status_t
|
||||||
|
@ -1515,7 +1512,7 @@ bfa_pport_get_qos_stats(struct bfa_s *bfa, union bfa_pport_stats_u *stats,
|
||||||
/*
|
/*
|
||||||
* QoS stats is embedded in port stats
|
* QoS stats is embedded in port stats
|
||||||
*/
|
*/
|
||||||
return (bfa_pport_get_stats(bfa, stats, cbfn, cbarg));
|
return bfa_pport_get_stats(bfa, stats, cbfn, cbarg);
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_status_t
|
bfa_status_t
|
||||||
|
@ -1525,7 +1522,7 @@ bfa_pport_clear_qos_stats(struct bfa_s *bfa, bfa_cb_pport_t cbfn, void *cbarg)
|
||||||
|
|
||||||
if (port->stats_busy) {
|
if (port->stats_busy) {
|
||||||
bfa_trc(bfa, port->stats_busy);
|
bfa_trc(bfa, port->stats_busy);
|
||||||
return (BFA_STATUS_DEVBUSY);
|
return BFA_STATUS_DEVBUSY;
|
||||||
}
|
}
|
||||||
|
|
||||||
port->stats_busy = BFA_TRUE;
|
port->stats_busy = BFA_TRUE;
|
||||||
|
@ -1536,7 +1533,7 @@ bfa_pport_clear_qos_stats(struct bfa_s *bfa, bfa_cb_pport_t cbfn, void *cbarg)
|
||||||
|
|
||||||
bfa_timer_start(bfa, &port->timer, bfa_port_stats_clr_timeout, port,
|
bfa_timer_start(bfa, &port->timer, bfa_port_stats_clr_timeout, port,
|
||||||
BFA_PORT_STATS_TOV);
|
BFA_PORT_STATS_TOV);
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1545,7 +1542,7 @@ bfa_pport_clear_qos_stats(struct bfa_s *bfa, bfa_cb_pport_t cbfn, void *cbarg)
|
||||||
bfa_status_t
|
bfa_status_t
|
||||||
bfa_pport_trunk_disable(struct bfa_s *bfa)
|
bfa_pport_trunk_disable(struct bfa_s *bfa)
|
||||||
{
|
{
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_boolean_t
|
bfa_boolean_t
|
||||||
|
@ -1562,8 +1559,8 @@ bfa_pport_is_disabled(struct bfa_s *bfa)
|
||||||
{
|
{
|
||||||
struct bfa_pport_s *port = BFA_PORT_MOD(bfa);
|
struct bfa_pport_s *port = BFA_PORT_MOD(bfa);
|
||||||
|
|
||||||
return (bfa_sm_to_state(hal_pport_sm_table, port->sm) ==
|
return bfa_sm_to_state(hal_pport_sm_table, port->sm) ==
|
||||||
BFA_PPORT_ST_DISABLED);
|
BFA_PPORT_ST_DISABLED;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1572,7 +1569,7 @@ bfa_pport_is_ratelim(struct bfa_s *bfa)
|
||||||
{
|
{
|
||||||
struct bfa_pport_s *pport = BFA_PORT_MOD(bfa);
|
struct bfa_pport_s *pport = BFA_PORT_MOD(bfa);
|
||||||
|
|
||||||
return (pport->cfg.ratelimit ? BFA_TRUE : BFA_FALSE);
|
return pport->cfg.ratelimit ? BFA_TRUE : BFA_FALSE;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1620,7 +1617,7 @@ bfa_pport_cfg_ratelim_speed(struct bfa_s *bfa, enum bfa_pport_speed speed)
|
||||||
|
|
||||||
pport->cfg.trl_def_speed = speed;
|
pport->cfg.trl_def_speed = speed;
|
||||||
|
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1632,7 +1629,7 @@ bfa_pport_get_ratelim_speed(struct bfa_s *bfa)
|
||||||
struct bfa_pport_s *pport = BFA_PORT_MOD(bfa);
|
struct bfa_pport_s *pport = BFA_PORT_MOD(bfa);
|
||||||
|
|
||||||
bfa_trc(bfa, pport->cfg.trl_def_speed);
|
bfa_trc(bfa, pport->cfg.trl_def_speed);
|
||||||
return (pport->cfg.trl_def_speed);
|
return pport->cfg.trl_def_speed;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -568,11 +568,10 @@ bfa_fcs_port_offline_actions(struct bfa_fcs_port_s *port)
|
||||||
|
|
||||||
__port_action[port->fabric->fab_type].offline(port);
|
__port_action[port->fabric->fab_type].offline(port);
|
||||||
|
|
||||||
if (bfa_fcs_fabric_is_online(port->fabric) == BFA_TRUE) {
|
if (bfa_fcs_fabric_is_online(port->fabric) == BFA_TRUE)
|
||||||
bfa_fcs_port_aen_post(port, BFA_LPORT_AEN_DISCONNECT);
|
bfa_fcs_port_aen_post(port, BFA_LPORT_AEN_DISCONNECT);
|
||||||
} else {
|
else
|
||||||
bfa_fcs_port_aen_post(port, BFA_LPORT_AEN_OFFLINE);
|
bfa_fcs_port_aen_post(port, BFA_LPORT_AEN_OFFLINE);
|
||||||
}
|
|
||||||
bfa_fcb_port_offline(port->fcs->bfad, port->port_cfg.roles,
|
bfa_fcb_port_offline(port->fcs->bfad, port->port_cfg.roles,
|
||||||
port->fabric->vf_drv,
|
port->fabric->vf_drv,
|
||||||
(port->vport == NULL) ? NULL : port->vport->vport_drv);
|
(port->vport == NULL) ? NULL : port->vport->vport_drv);
|
||||||
|
@ -777,7 +776,7 @@ bfa_fcs_port_get_rport_by_pwwn(struct bfa_fcs_port_s *port, wwn_t pwwn)
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_trc(port->fcs, pwwn);
|
bfa_trc(port->fcs, pwwn);
|
||||||
return (NULL);
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -796,7 +795,7 @@ bfa_fcs_port_get_rport_by_nwwn(struct bfa_fcs_port_s *port, wwn_t nwwn)
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_trc(port->fcs, nwwn);
|
bfa_trc(port->fcs, nwwn);
|
||||||
return (NULL);
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -870,7 +869,7 @@ bfa_fcs_port_lip(struct bfa_fcs_port_s *port)
|
||||||
bfa_boolean_t
|
bfa_boolean_t
|
||||||
bfa_fcs_port_is_online(struct bfa_fcs_port_s *port)
|
bfa_fcs_port_is_online(struct bfa_fcs_port_s *port)
|
||||||
{
|
{
|
||||||
return (bfa_sm_cmp_state(port, bfa_fcs_port_sm_online));
|
return bfa_sm_cmp_state(port, bfa_fcs_port_sm_online);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -199,7 +199,7 @@ bfa_fcxp_get(struct bfa_fcxp_mod_s *fm)
|
||||||
if (fcxp)
|
if (fcxp)
|
||||||
list_add_tail(&fcxp->qe, &fm->fcxp_active_q);
|
list_add_tail(&fcxp->qe, &fm->fcxp_active_q);
|
||||||
|
|
||||||
return (fcxp);
|
return fcxp;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -503,7 +503,7 @@ bfa_fcxp_alloc(void *caller, struct bfa_s *bfa, int nreq_sgles,
|
||||||
|
|
||||||
fcxp = bfa_fcxp_get(BFA_FCXP_MOD(bfa));
|
fcxp = bfa_fcxp_get(BFA_FCXP_MOD(bfa));
|
||||||
if (fcxp == NULL)
|
if (fcxp == NULL)
|
||||||
return (NULL);
|
return NULL;
|
||||||
|
|
||||||
bfa_trc(bfa, fcxp->fcxp_tag);
|
bfa_trc(bfa, fcxp->fcxp_tag);
|
||||||
|
|
||||||
|
@ -568,7 +568,7 @@ bfa_fcxp_alloc(void *caller, struct bfa_s *bfa, int nreq_sgles,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (fcxp);
|
return fcxp;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -709,7 +709,7 @@ bfa_status_t
|
||||||
bfa_fcxp_abort(struct bfa_fcxp_s *fcxp)
|
bfa_fcxp_abort(struct bfa_fcxp_s *fcxp)
|
||||||
{
|
{
|
||||||
bfa_assert(0);
|
bfa_assert(0);
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
|
@ -59,7 +59,7 @@ bfa_intx(struct bfa_s *bfa)
|
||||||
qintr = intr & __HFN_INT_RME_MASK;
|
qintr = intr & __HFN_INT_RME_MASK;
|
||||||
bfa_reg_write(bfa->iocfc.bfa_regs.intr_status, qintr);
|
bfa_reg_write(bfa->iocfc.bfa_regs.intr_status, qintr);
|
||||||
|
|
||||||
for (queue = 0; queue < BFI_IOC_MAX_CQS_ASIC; queue ++) {
|
for (queue = 0; queue < BFI_IOC_MAX_CQS_ASIC; queue++) {
|
||||||
if (intr & (__HFN_INT_RME_Q0 << queue))
|
if (intr & (__HFN_INT_RME_Q0 << queue))
|
||||||
bfa_msix_rspq(bfa, queue & (BFI_IOC_MAX_CQS - 1));
|
bfa_msix_rspq(bfa, queue & (BFI_IOC_MAX_CQS - 1));
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,9 +26,9 @@ void bfa_isr_unhandled(struct bfa_s *bfa, struct bfi_msg_s *m);
|
||||||
void bfa_isr_bind(enum bfi_mclass mc, bfa_isr_func_t isr_func);
|
void bfa_isr_bind(enum bfi_mclass mc, bfa_isr_func_t isr_func);
|
||||||
|
|
||||||
|
|
||||||
#define bfa_reqq_pi(__bfa, __reqq) (__bfa)->iocfc.req_cq_pi[__reqq]
|
#define bfa_reqq_pi(__bfa, __reqq) ((__bfa)->iocfc.req_cq_pi[__reqq])
|
||||||
#define bfa_reqq_ci(__bfa, __reqq) \
|
#define bfa_reqq_ci(__bfa, __reqq) \
|
||||||
*(u32 *)((__bfa)->iocfc.req_cq_shadow_ci[__reqq].kva)
|
(*(u32 *)((__bfa)->iocfc.req_cq_shadow_ci[__reqq].kva))
|
||||||
|
|
||||||
#define bfa_reqq_full(__bfa, __reqq) \
|
#define bfa_reqq_full(__bfa, __reqq) \
|
||||||
(((bfa_reqq_pi(__bfa, __reqq) + 1) & \
|
(((bfa_reqq_pi(__bfa, __reqq) + 1) & \
|
||||||
|
@ -50,14 +50,16 @@ void bfa_isr_bind(enum bfi_mclass mc, bfa_isr_func_t isr_func);
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
#define bfa_rspq_pi(__bfa, __rspq) \
|
#define bfa_rspq_pi(__bfa, __rspq) \
|
||||||
*(u32 *)((__bfa)->iocfc.rsp_cq_shadow_pi[__rspq].kva)
|
(*(u32 *)((__bfa)->iocfc.rsp_cq_shadow_pi[__rspq].kva))
|
||||||
|
|
||||||
#define bfa_rspq_ci(__bfa, __rspq) (__bfa)->iocfc.rsp_cq_ci[__rspq]
|
#define bfa_rspq_ci(__bfa, __rspq) ((__bfa)->iocfc.rsp_cq_ci[__rspq])
|
||||||
#define bfa_rspq_elem(__bfa, __rspq, __ci) \
|
#define bfa_rspq_elem(__bfa, __rspq, __ci) \
|
||||||
&((struct bfi_msg_s *)((__bfa)->iocfc.rsp_cq_ba[__rspq].kva))[__ci]
|
(&((struct bfi_msg_s *)((__bfa)->iocfc.rsp_cq_ba[__rspq].kva))[__ci])
|
||||||
|
|
||||||
#define CQ_INCR(__index, __size) \
|
#define CQ_INCR(__index, __size) do { \
|
||||||
(__index)++; (__index) &= ((__size) - 1)
|
(__index)++; \
|
||||||
|
(__index) &= ((__size) - 1); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Queue element to wait for room in request queue. FIFO order is
|
* Queue element to wait for room in request queue. FIFO order is
|
||||||
|
@ -94,7 +96,7 @@ bfa_reqq_winit(struct bfa_reqq_wait_s *wqe, void (*qresume) (void *cbarg),
|
||||||
wqe->cbarg = cbarg;
|
wqe->cbarg = cbarg;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define bfa_reqq(__bfa, __reqq) &(__bfa)->reqq_waitq[__reqq]
|
#define bfa_reqq(__bfa, __reqq) (&(__bfa)->reqq_waitq[__reqq])
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* static inline void
|
* static inline void
|
||||||
|
|
|
@ -51,7 +51,7 @@ BFA_TRC_FILE(HAL, IOC);
|
||||||
(sizeof(struct bfa_trc_mod_s) - \
|
(sizeof(struct bfa_trc_mod_s) - \
|
||||||
BFA_TRC_MAX * sizeof(struct bfa_trc_s)))
|
BFA_TRC_MAX * sizeof(struct bfa_trc_s)))
|
||||||
#define BFA_DBG_FWTRC_OFF(_fn) (BFI_IOC_TRC_OFF + BFA_DBG_FWTRC_LEN * (_fn))
|
#define BFA_DBG_FWTRC_OFF(_fn) (BFI_IOC_TRC_OFF + BFA_DBG_FWTRC_LEN * (_fn))
|
||||||
#define bfa_ioc_stats(_ioc, _stats) (_ioc)->stats._stats ++
|
#define bfa_ioc_stats(_ioc, _stats) ((_ioc)->stats._stats++)
|
||||||
|
|
||||||
#define BFA_FLASH_CHUNK_NO(off) (off / BFI_FLASH_CHUNK_SZ_WORDS)
|
#define BFA_FLASH_CHUNK_NO(off) (off / BFI_FLASH_CHUNK_SZ_WORDS)
|
||||||
#define BFA_FLASH_OFFSET_IN_CHUNK(off) (off % BFI_FLASH_CHUNK_SZ_WORDS)
|
#define BFA_FLASH_OFFSET_IN_CHUNK(off) (off % BFI_FLASH_CHUNK_SZ_WORDS)
|
||||||
|
@ -1953,8 +1953,8 @@ bfa_ioc_error_isr(struct bfa_ioc_s *ioc)
|
||||||
bfa_boolean_t
|
bfa_boolean_t
|
||||||
bfa_ioc_is_disabled(struct bfa_ioc_s *ioc)
|
bfa_ioc_is_disabled(struct bfa_ioc_s *ioc)
|
||||||
{
|
{
|
||||||
return (bfa_fsm_cmp_state(ioc, bfa_ioc_sm_disabling)
|
return bfa_fsm_cmp_state(ioc, bfa_ioc_sm_disabling)
|
||||||
|| bfa_fsm_cmp_state(ioc, bfa_ioc_sm_disabled));
|
|| bfa_fsm_cmp_state(ioc, bfa_ioc_sm_disabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1963,9 +1963,9 @@ bfa_ioc_is_disabled(struct bfa_ioc_s *ioc)
|
||||||
bfa_boolean_t
|
bfa_boolean_t
|
||||||
bfa_ioc_fw_mismatch(struct bfa_ioc_s *ioc)
|
bfa_ioc_fw_mismatch(struct bfa_ioc_s *ioc)
|
||||||
{
|
{
|
||||||
return (bfa_fsm_cmp_state(ioc, bfa_ioc_sm_reset)
|
return bfa_fsm_cmp_state(ioc, bfa_ioc_sm_reset)
|
||||||
|| bfa_fsm_cmp_state(ioc, bfa_ioc_sm_fwcheck)
|
|| bfa_fsm_cmp_state(ioc, bfa_ioc_sm_fwcheck)
|
||||||
|| bfa_fsm_cmp_state(ioc, bfa_ioc_sm_mismatch));
|
|| bfa_fsm_cmp_state(ioc, bfa_ioc_sm_mismatch);
|
||||||
}
|
}
|
||||||
|
|
||||||
#define bfa_ioc_state_disabled(__sm) \
|
#define bfa_ioc_state_disabled(__sm) \
|
||||||
|
|
|
@ -179,16 +179,16 @@ struct bfa_ioc_s {
|
||||||
struct bfa_ioc_mbox_mod_s mbox_mod;
|
struct bfa_ioc_mbox_mod_s mbox_mod;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define bfa_ioc_pcifn(__ioc) (__ioc)->pcidev.pci_func
|
#define bfa_ioc_pcifn(__ioc) ((__ioc)->pcidev.pci_func)
|
||||||
#define bfa_ioc_devid(__ioc) (__ioc)->pcidev.device_id
|
#define bfa_ioc_devid(__ioc) ((__ioc)->pcidev.device_id)
|
||||||
#define bfa_ioc_bar0(__ioc) (__ioc)->pcidev.pci_bar_kva
|
#define bfa_ioc_bar0(__ioc) ((__ioc)->pcidev.pci_bar_kva)
|
||||||
#define bfa_ioc_portid(__ioc) ((__ioc)->port_id)
|
#define bfa_ioc_portid(__ioc) ((__ioc)->port_id)
|
||||||
#define bfa_ioc_fetch_stats(__ioc, __stats) \
|
#define bfa_ioc_fetch_stats(__ioc, __stats) \
|
||||||
((__stats)->drv_stats) = (__ioc)->stats
|
(((__stats)->drv_stats) = (__ioc)->stats)
|
||||||
#define bfa_ioc_clr_stats(__ioc) \
|
#define bfa_ioc_clr_stats(__ioc) \
|
||||||
bfa_os_memset(&(__ioc)->stats, 0, sizeof((__ioc)->stats))
|
bfa_os_memset(&(__ioc)->stats, 0, sizeof((__ioc)->stats))
|
||||||
#define bfa_ioc_maxfrsize(__ioc) (__ioc)->attr->maxfrsize
|
#define bfa_ioc_maxfrsize(__ioc) ((__ioc)->attr->maxfrsize)
|
||||||
#define bfa_ioc_rx_bbcredit(__ioc) (__ioc)->attr->rx_bbcredit
|
#define bfa_ioc_rx_bbcredit(__ioc) ((__ioc)->attr->rx_bbcredit)
|
||||||
#define bfa_ioc_speed_sup(__ioc) \
|
#define bfa_ioc_speed_sup(__ioc) \
|
||||||
BFI_ADAPTER_GETP(SPEED, (__ioc)->attr->adapter_prop)
|
BFI_ADAPTER_GETP(SPEED, (__ioc)->attr->adapter_prop)
|
||||||
|
|
||||||
|
|
|
@ -794,7 +794,7 @@ bfa_iocfc_get_stats(struct bfa_s *bfa, struct bfa_iocfc_stats_s *stats,
|
||||||
|
|
||||||
if (iocfc->stats_busy) {
|
if (iocfc->stats_busy) {
|
||||||
bfa_trc(bfa, iocfc->stats_busy);
|
bfa_trc(bfa, iocfc->stats_busy);
|
||||||
return (BFA_STATUS_DEVBUSY);
|
return BFA_STATUS_DEVBUSY;
|
||||||
}
|
}
|
||||||
|
|
||||||
iocfc->stats_busy = BFA_TRUE;
|
iocfc->stats_busy = BFA_TRUE;
|
||||||
|
@ -804,7 +804,7 @@ bfa_iocfc_get_stats(struct bfa_s *bfa, struct bfa_iocfc_stats_s *stats,
|
||||||
|
|
||||||
bfa_iocfc_stats_query(bfa);
|
bfa_iocfc_stats_query(bfa);
|
||||||
|
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_status_t
|
bfa_status_t
|
||||||
|
@ -814,7 +814,7 @@ bfa_iocfc_clear_stats(struct bfa_s *bfa, bfa_cb_ioc_t cbfn, void *cbarg)
|
||||||
|
|
||||||
if (iocfc->stats_busy) {
|
if (iocfc->stats_busy) {
|
||||||
bfa_trc(bfa, iocfc->stats_busy);
|
bfa_trc(bfa, iocfc->stats_busy);
|
||||||
return (BFA_STATUS_DEVBUSY);
|
return BFA_STATUS_DEVBUSY;
|
||||||
}
|
}
|
||||||
|
|
||||||
iocfc->stats_busy = BFA_TRUE;
|
iocfc->stats_busy = BFA_TRUE;
|
||||||
|
@ -822,7 +822,7 @@ bfa_iocfc_clear_stats(struct bfa_s *bfa, bfa_cb_ioc_t cbfn, void *cbarg)
|
||||||
iocfc->stats_cbarg = cbarg;
|
iocfc->stats_cbarg = cbarg;
|
||||||
|
|
||||||
bfa_iocfc_stats_clear(bfa);
|
bfa_iocfc_stats_clear(bfa);
|
||||||
return (BFA_STATUS_OK);
|
return BFA_STATUS_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -107,13 +107,13 @@ struct bfa_iocfc_s {
|
||||||
|
|
||||||
#define bfa_lpuid(__bfa) bfa_ioc_portid(&(__bfa)->ioc)
|
#define bfa_lpuid(__bfa) bfa_ioc_portid(&(__bfa)->ioc)
|
||||||
#define bfa_msix_init(__bfa, __nvecs) \
|
#define bfa_msix_init(__bfa, __nvecs) \
|
||||||
(__bfa)->iocfc.hwif.hw_msix_init(__bfa, __nvecs)
|
((__bfa)->iocfc.hwif.hw_msix_init(__bfa, __nvecs))
|
||||||
#define bfa_msix_install(__bfa) \
|
#define bfa_msix_install(__bfa) \
|
||||||
(__bfa)->iocfc.hwif.hw_msix_install(__bfa)
|
((__bfa)->iocfc.hwif.hw_msix_install(__bfa))
|
||||||
#define bfa_msix_uninstall(__bfa) \
|
#define bfa_msix_uninstall(__bfa) \
|
||||||
(__bfa)->iocfc.hwif.hw_msix_uninstall(__bfa)
|
((__bfa)->iocfc.hwif.hw_msix_uninstall(__bfa))
|
||||||
#define bfa_isr_mode_set(__bfa, __msix) \
|
#define bfa_isr_mode_set(__bfa, __msix) \
|
||||||
(__bfa)->iocfc.hwif.hw_isr_mode_set(__bfa, __msix)
|
((__bfa)->iocfc.hwif.hw_isr_mode_set(__bfa, __msix))
|
||||||
#define bfa_msix_getvecs(__bfa, __vecmap, __nvecs, __maxvec) \
|
#define bfa_msix_getvecs(__bfa, __vecmap, __nvecs, __maxvec) \
|
||||||
(__bfa)->iocfc.hwif.hw_msix_getvecs(__bfa, __vecmap, __nvecs, __maxvec)
|
(__bfa)->iocfc.hwif.hw_msix_getvecs(__bfa, __vecmap, __nvecs, __maxvec)
|
||||||
|
|
||||||
|
|
|
@ -1029,7 +1029,7 @@ bfa_itnim_create(struct bfa_s *bfa, struct bfa_rport_s *rport, void *ditn)
|
||||||
bfa_stats(itnim, creates);
|
bfa_stats(itnim, creates);
|
||||||
bfa_sm_send_event(itnim, BFA_ITNIM_SM_CREATE);
|
bfa_sm_send_event(itnim, BFA_ITNIM_SM_CREATE);
|
||||||
|
|
||||||
return (itnim);
|
return itnim;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -1061,7 +1061,7 @@ bfa_itnim_offline(struct bfa_itnim_s *itnim)
|
||||||
bfa_boolean_t
|
bfa_boolean_t
|
||||||
bfa_itnim_hold_io(struct bfa_itnim_s *itnim)
|
bfa_itnim_hold_io(struct bfa_itnim_s *itnim)
|
||||||
{
|
{
|
||||||
return (
|
return
|
||||||
itnim->fcpim->path_tov && itnim->iotov_active &&
|
itnim->fcpim->path_tov && itnim->iotov_active &&
|
||||||
(bfa_sm_cmp_state(itnim, bfa_itnim_sm_fwcreate) ||
|
(bfa_sm_cmp_state(itnim, bfa_itnim_sm_fwcreate) ||
|
||||||
bfa_sm_cmp_state(itnim, bfa_itnim_sm_sler) ||
|
bfa_sm_cmp_state(itnim, bfa_itnim_sm_sler) ||
|
||||||
|
@ -1069,7 +1069,7 @@ bfa_itnim_hold_io(struct bfa_itnim_s *itnim)
|
||||||
bfa_sm_cmp_state(itnim, bfa_itnim_sm_fwdelete) ||
|
bfa_sm_cmp_state(itnim, bfa_itnim_sm_fwdelete) ||
|
||||||
bfa_sm_cmp_state(itnim, bfa_itnim_sm_offline) ||
|
bfa_sm_cmp_state(itnim, bfa_itnim_sm_offline) ||
|
||||||
bfa_sm_cmp_state(itnim, bfa_itnim_sm_iocdisable))
|
bfa_sm_cmp_state(itnim, bfa_itnim_sm_iocdisable))
|
||||||
);
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
|
@ -231,9 +231,9 @@ bfa_log_get_level(struct bfa_log_mod_s *log_mod, int mod_id)
|
||||||
return BFA_LOG_INVALID;
|
return BFA_LOG_INVALID;
|
||||||
|
|
||||||
if (log_mod)
|
if (log_mod)
|
||||||
return (log_mod->log_level[mod_id]);
|
return log_mod->log_level[mod_id];
|
||||||
else
|
else
|
||||||
return (bfa_log_info[mod_id].level);
|
return bfa_log_info[mod_id].level;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum bfa_log_severity
|
enum bfa_log_severity
|
||||||
|
|
|
@ -59,8 +59,8 @@ struct bfa_pport_s {
|
||||||
u8 *stats_kva;
|
u8 *stats_kva;
|
||||||
u64 stats_pa;
|
u64 stats_pa;
|
||||||
union bfa_pport_stats_u *stats; /* pport stats */
|
union bfa_pport_stats_u *stats; /* pport stats */
|
||||||
u32 mypid : 24;
|
u32 mypid:24;
|
||||||
u32 rsvd_b : 8;
|
u32 rsvd_b:8;
|
||||||
struct bfa_timer_s timer; /* timer */
|
struct bfa_timer_s timer; /* timer */
|
||||||
union bfa_pport_stats_u *stats_ret;
|
union bfa_pport_stats_u *stats_ret;
|
||||||
/* driver stats location */
|
/* driver stats location */
|
||||||
|
|
|
@ -677,7 +677,7 @@ bfa_rport_alloc(struct bfa_rport_mod_s *mod)
|
||||||
if (rport)
|
if (rport)
|
||||||
list_add_tail(&rport->qe, &mod->rp_active_q);
|
list_add_tail(&rport->qe, &mod->rp_active_q);
|
||||||
|
|
||||||
return (rport);
|
return rport;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -834,7 +834,7 @@ bfa_rport_create(struct bfa_s *bfa, void *rport_drv)
|
||||||
rp = bfa_rport_alloc(BFA_RPORT_MOD(bfa));
|
rp = bfa_rport_alloc(BFA_RPORT_MOD(bfa));
|
||||||
|
|
||||||
if (rp == NULL)
|
if (rp == NULL)
|
||||||
return (NULL);
|
return NULL;
|
||||||
|
|
||||||
rp->bfa = bfa;
|
rp->bfa = bfa;
|
||||||
rp->rport_drv = rport_drv;
|
rp->rport_drv = rport_drv;
|
||||||
|
@ -843,7 +843,7 @@ bfa_rport_create(struct bfa_s *bfa, void *rport_drv)
|
||||||
bfa_assert(bfa_sm_cmp_state(rp, bfa_rport_sm_uninit));
|
bfa_assert(bfa_sm_cmp_state(rp, bfa_rport_sm_uninit));
|
||||||
bfa_sm_send_event(rp, BFA_RPORT_SM_CREATE);
|
bfa_sm_send_event(rp, BFA_RPORT_SM_CREATE);
|
||||||
|
|
||||||
return (rp);
|
return rp;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
|
@ -24,7 +24,8 @@ BFA_TRC_FILE(HAL, TSKIM);
|
||||||
* task management completion handling
|
* task management completion handling
|
||||||
*/
|
*/
|
||||||
#define bfa_tskim_qcomp(__tskim, __cbfn) do { \
|
#define bfa_tskim_qcomp(__tskim, __cbfn) do { \
|
||||||
bfa_cb_queue((__tskim)->bfa, &(__tskim)->hcb_qe, __cbfn, (__tskim)); \
|
bfa_cb_queue((__tskim)->bfa, &(__tskim)->hcb_qe, \
|
||||||
|
__cbfn, (__tskim)); \
|
||||||
bfa_tskim_notify_comp(__tskim); \
|
bfa_tskim_notify_comp(__tskim); \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
|
|
|
@ -185,7 +185,7 @@ bfa_uf_get(struct bfa_uf_mod_s *uf_mod)
|
||||||
struct bfa_uf_s *uf;
|
struct bfa_uf_s *uf;
|
||||||
|
|
||||||
bfa_q_deq(&uf_mod->uf_free_q, &uf);
|
bfa_q_deq(&uf_mod->uf_free_q, &uf);
|
||||||
return (uf);
|
return uf;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
|
|
@ -188,8 +188,8 @@ static struct bfad_port_s *
|
||||||
bfad_get_drv_port(struct bfad_s *bfad, struct bfad_vf_s *vf_drv,
|
bfad_get_drv_port(struct bfad_s *bfad, struct bfad_vf_s *vf_drv,
|
||||||
struct bfad_vport_s *vp_drv)
|
struct bfad_vport_s *vp_drv)
|
||||||
{
|
{
|
||||||
return ((vp_drv) ? (&(vp_drv)->drv_port)
|
return (vp_drv) ? (&(vp_drv)->drv_port)
|
||||||
: ((vf_drv) ? (&(vf_drv)->base_port) : (&(bfad)->pport)));
|
: ((vf_drv) ? (&(vf_drv)->base_port) : (&(bfad)->pport));
|
||||||
}
|
}
|
||||||
|
|
||||||
struct bfad_port_s *
|
struct bfad_port_s *
|
||||||
|
@ -716,7 +716,7 @@ bfad_drv_init(struct bfad_s *bfad)
|
||||||
if ((bfad->bfad_flags & BFAD_MSIX_ON)
|
if ((bfad->bfad_flags & BFAD_MSIX_ON)
|
||||||
&& bfad_install_msix_handler(bfad)) {
|
&& bfad_install_msix_handler(bfad)) {
|
||||||
printk(KERN_WARNING "%s: install_msix failed, bfad%d\n",
|
printk(KERN_WARNING "%s: install_msix failed, bfad%d\n",
|
||||||
__FUNCTION__, bfad->inst_no);
|
__func__, bfad->inst_no);
|
||||||
}
|
}
|
||||||
|
|
||||||
bfad_init_timer(bfad);
|
bfad_init_timer(bfad);
|
||||||
|
|
|
@ -65,10 +65,10 @@ bfad_read_firmware(struct pci_dev *pdev, u32 **bfi_image,
|
||||||
memcpy(*bfi_image, fw->data, fw->size);
|
memcpy(*bfi_image, fw->data, fw->size);
|
||||||
*bfi_image_size = fw->size/sizeof(u32);
|
*bfi_image_size = fw->size/sizeof(u32);
|
||||||
|
|
||||||
return(*bfi_image);
|
return *bfi_image;
|
||||||
|
|
||||||
error:
|
error:
|
||||||
return(NULL);
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
u32 *
|
u32 *
|
||||||
|
@ -78,12 +78,12 @@ bfad_get_firmware_buf(struct pci_dev *pdev)
|
||||||
if (bfi_image_ct_size == 0)
|
if (bfi_image_ct_size == 0)
|
||||||
bfad_read_firmware(pdev, &bfi_image_ct,
|
bfad_read_firmware(pdev, &bfi_image_ct,
|
||||||
&bfi_image_ct_size, BFAD_FW_FILE_CT);
|
&bfi_image_ct_size, BFAD_FW_FILE_CT);
|
||||||
return(bfi_image_ct);
|
return bfi_image_ct;
|
||||||
} else {
|
} else {
|
||||||
if (bfi_image_cb_size == 0)
|
if (bfi_image_cb_size == 0)
|
||||||
bfad_read_firmware(pdev, &bfi_image_cb,
|
bfad_read_firmware(pdev, &bfi_image_cb,
|
||||||
&bfi_image_cb_size, BFAD_FW_FILE_CB);
|
&bfi_image_cb_size, BFAD_FW_FILE_CB);
|
||||||
return(bfi_image_cb);
|
return bfi_image_cb;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1050,7 +1050,7 @@ bfad_im_itnim_work_handler(struct work_struct *work)
|
||||||
} else {
|
} else {
|
||||||
printk(KERN_WARNING
|
printk(KERN_WARNING
|
||||||
"%s: itnim %llx is already in online state\n",
|
"%s: itnim %llx is already in online state\n",
|
||||||
__FUNCTION__,
|
__func__,
|
||||||
bfa_fcs_itnim_get_pwwn(&itnim->fcs_itnim));
|
bfa_fcs_itnim_get_pwwn(&itnim->fcs_itnim));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -31,7 +31,7 @@ u32 *bfad_read_firmware(struct pci_dev *pdev, u32 **bfi_image,
|
||||||
static inline u32 *
|
static inline u32 *
|
||||||
bfad_load_fwimg(struct pci_dev *pdev)
|
bfad_load_fwimg(struct pci_dev *pdev)
|
||||||
{
|
{
|
||||||
return(bfad_get_firmware_buf(pdev));
|
return bfad_get_firmware_buf(pdev);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void
|
static inline void
|
||||||
|
|
|
@ -23,13 +23,12 @@ BFA_TRC_FILE(LDRV, INTR);
|
||||||
/**
|
/**
|
||||||
* bfa_isr BFA driver interrupt functions
|
* bfa_isr BFA driver interrupt functions
|
||||||
*/
|
*/
|
||||||
irqreturn_t bfad_intx(int irq, void *dev_id);
|
|
||||||
static int msix_disable;
|
static int msix_disable;
|
||||||
module_param(msix_disable, int, S_IRUGO | S_IWUSR);
|
module_param(msix_disable, int, S_IRUGO | S_IWUSR);
|
||||||
/**
|
/**
|
||||||
* Line based interrupt handler.
|
* Line based interrupt handler.
|
||||||
*/
|
*/
|
||||||
irqreturn_t
|
static irqreturn_t
|
||||||
bfad_intx(int irq, void *dev_id)
|
bfad_intx(int irq, void *dev_id)
|
||||||
{
|
{
|
||||||
struct bfad_s *bfad = dev_id;
|
struct bfad_s *bfad = dev_id;
|
||||||
|
|
|
@ -887,7 +887,7 @@ bfa_fcs_fabric_modsusp(struct bfa_fcs_s *fcs)
|
||||||
bfa_boolean_t
|
bfa_boolean_t
|
||||||
bfa_fcs_fabric_is_loopback(struct bfa_fcs_fabric_s *fabric)
|
bfa_fcs_fabric_is_loopback(struct bfa_fcs_fabric_s *fabric)
|
||||||
{
|
{
|
||||||
return (bfa_sm_cmp_state(fabric, bfa_fcs_fabric_sm_loopback));
|
return bfa_sm_cmp_state(fabric, bfa_fcs_fabric_sm_loopback);
|
||||||
}
|
}
|
||||||
|
|
||||||
enum bfa_pport_type
|
enum bfa_pport_type
|
||||||
|
@ -974,7 +974,7 @@ bfa_fcs_fabric_port_delete_comp(struct bfa_fcs_fabric_s *fabric)
|
||||||
int
|
int
|
||||||
bfa_fcs_fabric_is_online(struct bfa_fcs_fabric_s *fabric)
|
bfa_fcs_fabric_is_online(struct bfa_fcs_fabric_s *fabric)
|
||||||
{
|
{
|
||||||
return (bfa_sm_cmp_state(fabric, bfa_fcs_fabric_sm_online));
|
return bfa_sm_cmp_state(fabric, bfa_fcs_fabric_sm_online);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1015,7 +1015,7 @@ bfa_fcs_fabric_vport_lookup(struct bfa_fcs_fabric_s *fabric, wwn_t pwwn)
|
||||||
u16
|
u16
|
||||||
bfa_fcs_fabric_vport_count(struct bfa_fcs_fabric_s *fabric)
|
bfa_fcs_fabric_vport_count(struct bfa_fcs_fabric_s *fabric)
|
||||||
{
|
{
|
||||||
return (fabric->num_vports);
|
return fabric->num_vports;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -188,14 +188,14 @@ fc_els_rsp_parse(struct fchs_s *fchs, int len)
|
||||||
switch (els_cmd->els_code) {
|
switch (els_cmd->els_code) {
|
||||||
case FC_ELS_LS_RJT:
|
case FC_ELS_LS_RJT:
|
||||||
if (ls_rjt->reason_code == FC_LS_RJT_RSN_LOGICAL_BUSY)
|
if (ls_rjt->reason_code == FC_LS_RJT_RSN_LOGICAL_BUSY)
|
||||||
return (FC_PARSE_BUSY);
|
return FC_PARSE_BUSY;
|
||||||
else
|
else
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
case FC_ELS_ACC:
|
case FC_ELS_ACC:
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
}
|
}
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -228,7 +228,7 @@ fc_plogi_x_build(struct fchs_s *fchs, void *pld, u32 d_id, u32 s_id,
|
||||||
bfa_os_memcpy(&plogi->port_name, &port_name, sizeof(wwn_t));
|
bfa_os_memcpy(&plogi->port_name, &port_name, sizeof(wwn_t));
|
||||||
bfa_os_memcpy(&plogi->node_name, &node_name, sizeof(wwn_t));
|
bfa_os_memcpy(&plogi->node_name, &node_name, sizeof(wwn_t));
|
||||||
|
|
||||||
return (sizeof(struct fc_logi_s));
|
return sizeof(struct fc_logi_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -267,7 +267,7 @@ fc_flogi_build(struct fchs_s *fchs, struct fc_logi_s *flogi, u32 s_id,
|
||||||
flogi->csp.npiv_supp = 1; /* @todo. field name is not correct */
|
flogi->csp.npiv_supp = 1; /* @todo. field name is not correct */
|
||||||
vvl_info[0] = bfa_os_htonl(FLOGI_VVL_BRCD);
|
vvl_info[0] = bfa_os_htonl(FLOGI_VVL_BRCD);
|
||||||
|
|
||||||
return (sizeof(struct fc_logi_s));
|
return sizeof(struct fc_logi_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -287,7 +287,7 @@ fc_flogi_acc_build(struct fchs_s *fchs, struct fc_logi_s *flogi, u32 s_id,
|
||||||
|
|
||||||
flogi->csp.bbcred = bfa_os_htons(local_bb_credits);
|
flogi->csp.bbcred = bfa_os_htons(local_bb_credits);
|
||||||
|
|
||||||
return (sizeof(struct fc_logi_s));
|
return sizeof(struct fc_logi_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -306,7 +306,7 @@ fc_fdisc_build(struct fchs_s *fchs, struct fc_logi_s *flogi, u32 s_id,
|
||||||
flogi->port_name = port_name;
|
flogi->port_name = port_name;
|
||||||
flogi->node_name = node_name;
|
flogi->node_name = node_name;
|
||||||
|
|
||||||
return (sizeof(struct fc_logi_s));
|
return sizeof(struct fc_logi_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -338,26 +338,26 @@ fc_plogi_rsp_parse(struct fchs_s *fchs, int len, wwn_t port_name)
|
||||||
case FC_ELS_LS_RJT:
|
case FC_ELS_LS_RJT:
|
||||||
ls_rjt = (struct fc_ls_rjt_s *) (fchs + 1);
|
ls_rjt = (struct fc_ls_rjt_s *) (fchs + 1);
|
||||||
if (ls_rjt->reason_code == FC_LS_RJT_RSN_LOGICAL_BUSY)
|
if (ls_rjt->reason_code == FC_LS_RJT_RSN_LOGICAL_BUSY)
|
||||||
return (FC_PARSE_BUSY);
|
return FC_PARSE_BUSY;
|
||||||
else
|
else
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
case FC_ELS_ACC:
|
case FC_ELS_ACC:
|
||||||
plogi = (struct fc_logi_s *) (fchs + 1);
|
plogi = (struct fc_logi_s *) (fchs + 1);
|
||||||
if (len < sizeof(struct fc_logi_s))
|
if (len < sizeof(struct fc_logi_s))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (!wwn_is_equal(plogi->port_name, port_name))
|
if (!wwn_is_equal(plogi->port_name, port_name))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (!plogi->class3.class_valid)
|
if (!plogi->class3.class_valid)
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (bfa_os_ntohs(plogi->class3.rxsz) < (FC_MIN_PDUSZ))
|
if (bfa_os_ntohs(plogi->class3.rxsz) < (FC_MIN_PDUSZ))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
default:
|
default:
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -372,7 +372,7 @@ fc_plogi_parse(struct fchs_s *fchs)
|
||||||
if ((bfa_os_ntohs(plogi->class3.rxsz) < FC_MIN_PDUSZ)
|
if ((bfa_os_ntohs(plogi->class3.rxsz) < FC_MIN_PDUSZ)
|
||||||
|| (bfa_os_ntohs(plogi->class3.rxsz) > FC_MAX_PDUSZ)
|
|| (bfa_os_ntohs(plogi->class3.rxsz) > FC_MAX_PDUSZ)
|
||||||
|| (plogi->class3.rxsz == 0))
|
|| (plogi->class3.rxsz == 0))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
return FC_PARSE_OK;
|
return FC_PARSE_OK;
|
||||||
}
|
}
|
||||||
|
@ -393,7 +393,7 @@ fc_prli_build(struct fchs_s *fchs, void *pld, u32 d_id, u32 s_id,
|
||||||
prli->parampage.servparams.task_retry_id = 0;
|
prli->parampage.servparams.task_retry_id = 0;
|
||||||
prli->parampage.servparams.confirm = 1;
|
prli->parampage.servparams.confirm = 1;
|
||||||
|
|
||||||
return (sizeof(struct fc_prli_s));
|
return sizeof(struct fc_prli_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -414,41 +414,41 @@ fc_prli_acc_build(struct fchs_s *fchs, void *pld, u32 d_id, u32 s_id,
|
||||||
|
|
||||||
prli->parampage.rspcode = FC_PRLI_ACC_XQTD;
|
prli->parampage.rspcode = FC_PRLI_ACC_XQTD;
|
||||||
|
|
||||||
return (sizeof(struct fc_prli_s));
|
return sizeof(struct fc_prli_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
enum fc_parse_status
|
enum fc_parse_status
|
||||||
fc_prli_rsp_parse(struct fc_prli_s *prli, int len)
|
fc_prli_rsp_parse(struct fc_prli_s *prli, int len)
|
||||||
{
|
{
|
||||||
if (len < sizeof(struct fc_prli_s))
|
if (len < sizeof(struct fc_prli_s))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (prli->command != FC_ELS_ACC)
|
if (prli->command != FC_ELS_ACC)
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if ((prli->parampage.rspcode != FC_PRLI_ACC_XQTD)
|
if ((prli->parampage.rspcode != FC_PRLI_ACC_XQTD)
|
||||||
&& (prli->parampage.rspcode != FC_PRLI_ACC_PREDEF_IMG))
|
&& (prli->parampage.rspcode != FC_PRLI_ACC_PREDEF_IMG))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (prli->parampage.servparams.target != 1)
|
if (prli->parampage.servparams.target != 1)
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum fc_parse_status
|
enum fc_parse_status
|
||||||
fc_prli_parse(struct fc_prli_s *prli)
|
fc_prli_parse(struct fc_prli_s *prli)
|
||||||
{
|
{
|
||||||
if (prli->parampage.type != FC_TYPE_FCP)
|
if (prli->parampage.type != FC_TYPE_FCP)
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (!prli->parampage.imagepair)
|
if (!prli->parampage.imagepair)
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (!prli->parampage.servparams.initiator)
|
if (!prli->parampage.servparams.initiator)
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -462,7 +462,7 @@ fc_logo_build(struct fchs_s *fchs, struct fc_logo_s *logo, u32 d_id,
|
||||||
logo->nport_id = (s_id);
|
logo->nport_id = (s_id);
|
||||||
logo->orig_port_name = port_name;
|
logo->orig_port_name = port_name;
|
||||||
|
|
||||||
return (sizeof(struct fc_logo_s));
|
return sizeof(struct fc_logo_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
static u16
|
static u16
|
||||||
|
@ -484,7 +484,7 @@ fc_adisc_x_build(struct fchs_s *fchs, struct fc_adisc_s *adisc, u32 d_id,
|
||||||
adisc->orig_node_name = node_name;
|
adisc->orig_node_name = node_name;
|
||||||
adisc->nport_id = (s_id);
|
adisc->nport_id = (s_id);
|
||||||
|
|
||||||
return (sizeof(struct fc_adisc_s));
|
return sizeof(struct fc_adisc_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -511,15 +511,15 @@ fc_adisc_rsp_parse(struct fc_adisc_s *adisc, int len, wwn_t port_name,
|
||||||
{
|
{
|
||||||
|
|
||||||
if (len < sizeof(struct fc_adisc_s))
|
if (len < sizeof(struct fc_adisc_s))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (adisc->els_cmd.els_code != FC_ELS_ACC)
|
if (adisc->els_cmd.els_code != FC_ELS_ACC)
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (!wwn_is_equal(adisc->orig_port_name, port_name))
|
if (!wwn_is_equal(adisc->orig_port_name, port_name))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum fc_parse_status
|
enum fc_parse_status
|
||||||
|
@ -529,14 +529,14 @@ fc_adisc_parse(struct fchs_s *fchs, void *pld, u32 host_dap,
|
||||||
struct fc_adisc_s *adisc = (struct fc_adisc_s *) pld;
|
struct fc_adisc_s *adisc = (struct fc_adisc_s *) pld;
|
||||||
|
|
||||||
if (adisc->els_cmd.els_code != FC_ELS_ACC)
|
if (adisc->els_cmd.els_code != FC_ELS_ACC)
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if ((adisc->nport_id == (host_dap))
|
if ((adisc->nport_id == (host_dap))
|
||||||
&& wwn_is_equal(adisc->orig_port_name, port_name)
|
&& wwn_is_equal(adisc->orig_port_name, port_name)
|
||||||
&& wwn_is_equal(adisc->orig_node_name, node_name))
|
&& wwn_is_equal(adisc->orig_node_name, node_name))
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
|
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum fc_parse_status
|
enum fc_parse_status
|
||||||
|
@ -550,13 +550,13 @@ fc_pdisc_parse(struct fchs_s *fchs, wwn_t node_name, wwn_t port_name)
|
||||||
if ((bfa_os_ntohs(pdisc->class3.rxsz) <
|
if ((bfa_os_ntohs(pdisc->class3.rxsz) <
|
||||||
(FC_MIN_PDUSZ - sizeof(struct fchs_s)))
|
(FC_MIN_PDUSZ - sizeof(struct fchs_s)))
|
||||||
|| (pdisc->class3.rxsz == 0))
|
|| (pdisc->class3.rxsz == 0))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (!wwn_is_equal(pdisc->port_name, port_name))
|
if (!wwn_is_equal(pdisc->port_name, port_name))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
if (!wwn_is_equal(pdisc->node_name, node_name))
|
if (!wwn_is_equal(pdisc->node_name, node_name))
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
return FC_PARSE_OK;
|
return FC_PARSE_OK;
|
||||||
}
|
}
|
||||||
|
@ -570,7 +570,7 @@ fc_abts_build(struct fchs_s *fchs, u32 d_id, u32 s_id, u16 ox_id)
|
||||||
fchs->s_id = (s_id);
|
fchs->s_id = (s_id);
|
||||||
fchs->ox_id = bfa_os_htons(ox_id);
|
fchs->ox_id = bfa_os_htons(ox_id);
|
||||||
|
|
||||||
return (sizeof(struct fchs_s));
|
return sizeof(struct fchs_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
enum fc_parse_status
|
enum fc_parse_status
|
||||||
|
@ -578,9 +578,9 @@ fc_abts_rsp_parse(struct fchs_s *fchs, int len)
|
||||||
{
|
{
|
||||||
if ((fchs->cat_info == FC_CAT_BA_ACC)
|
if ((fchs->cat_info == FC_CAT_BA_ACC)
|
||||||
|| (fchs->cat_info == FC_CAT_BA_RJT))
|
|| (fchs->cat_info == FC_CAT_BA_RJT))
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
|
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -597,7 +597,7 @@ fc_rrq_build(struct fchs_s *fchs, struct fc_rrq_s *rrq, u32 d_id,
|
||||||
rrq->ox_id = bfa_os_htons(rrq_oxid);
|
rrq->ox_id = bfa_os_htons(rrq_oxid);
|
||||||
rrq->rx_id = FC_RXID_ANY;
|
rrq->rx_id = FC_RXID_ANY;
|
||||||
|
|
||||||
return (sizeof(struct fc_rrq_s));
|
return sizeof(struct fc_rrq_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -611,7 +611,7 @@ fc_logo_acc_build(struct fchs_s *fchs, void *pld, u32 d_id, u32 s_id,
|
||||||
memset(acc, 0, sizeof(struct fc_els_cmd_s));
|
memset(acc, 0, sizeof(struct fc_els_cmd_s));
|
||||||
acc->els_code = FC_ELS_ACC;
|
acc->els_code = FC_ELS_ACC;
|
||||||
|
|
||||||
return (sizeof(struct fc_els_cmd_s));
|
return sizeof(struct fc_els_cmd_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -627,7 +627,7 @@ fc_ls_rjt_build(struct fchs_s *fchs, struct fc_ls_rjt_s *ls_rjt, u32 d_id,
|
||||||
ls_rjt->reason_code_expl = reason_code_expl;
|
ls_rjt->reason_code_expl = reason_code_expl;
|
||||||
ls_rjt->vendor_unique = 0x00;
|
ls_rjt->vendor_unique = 0x00;
|
||||||
|
|
||||||
return (sizeof(struct fc_ls_rjt_s));
|
return sizeof(struct fc_ls_rjt_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -643,7 +643,7 @@ fc_ba_acc_build(struct fchs_s *fchs, struct fc_ba_acc_s *ba_acc, u32 d_id,
|
||||||
ba_acc->ox_id = fchs->ox_id;
|
ba_acc->ox_id = fchs->ox_id;
|
||||||
ba_acc->rx_id = fchs->rx_id;
|
ba_acc->rx_id = fchs->rx_id;
|
||||||
|
|
||||||
return (sizeof(struct fc_ba_acc_s));
|
return sizeof(struct fc_ba_acc_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -654,7 +654,7 @@ fc_ls_acc_build(struct fchs_s *fchs, struct fc_els_cmd_s *els_cmd,
|
||||||
memset(els_cmd, 0, sizeof(struct fc_els_cmd_s));
|
memset(els_cmd, 0, sizeof(struct fc_els_cmd_s));
|
||||||
els_cmd->els_code = FC_ELS_ACC;
|
els_cmd->els_code = FC_ELS_ACC;
|
||||||
|
|
||||||
return (sizeof(struct fc_els_cmd_s));
|
return sizeof(struct fc_els_cmd_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
|
@ -696,7 +696,7 @@ fc_tprlo_acc_build(struct fchs_s *fchs, struct fc_tprlo_acc_s *tprlo_acc,
|
||||||
tprlo_acc->tprlo_acc_params[page].orig_process_assc = 0;
|
tprlo_acc->tprlo_acc_params[page].orig_process_assc = 0;
|
||||||
tprlo_acc->tprlo_acc_params[page].resp_process_assc = 0;
|
tprlo_acc->tprlo_acc_params[page].resp_process_assc = 0;
|
||||||
}
|
}
|
||||||
return (bfa_os_ntohs(tprlo_acc->payload_len));
|
return bfa_os_ntohs(tprlo_acc->payload_len);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -721,7 +721,7 @@ fc_prlo_acc_build(struct fchs_s *fchs, struct fc_prlo_acc_s *prlo_acc,
|
||||||
prlo_acc->prlo_acc_params[page].resp_process_assc = 0;
|
prlo_acc->prlo_acc_params[page].resp_process_assc = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (bfa_os_ntohs(prlo_acc->payload_len));
|
return bfa_os_ntohs(prlo_acc->payload_len);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -735,7 +735,7 @@ fc_rnid_build(struct fchs_s *fchs, struct fc_rnid_cmd_s *rnid, u32 d_id,
|
||||||
rnid->els_cmd.els_code = FC_ELS_RNID;
|
rnid->els_cmd.els_code = FC_ELS_RNID;
|
||||||
rnid->node_id_data_format = data_format;
|
rnid->node_id_data_format = data_format;
|
||||||
|
|
||||||
return (sizeof(struct fc_rnid_cmd_s));
|
return sizeof(struct fc_rnid_cmd_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -759,10 +759,10 @@ fc_rnid_acc_build(struct fchs_s *fchs, struct fc_rnid_acc_s *rnid_acc,
|
||||||
rnid_acc->specific_id_data_length =
|
rnid_acc->specific_id_data_length =
|
||||||
sizeof(struct fc_rnid_general_topology_data_s);
|
sizeof(struct fc_rnid_general_topology_data_s);
|
||||||
bfa_os_assign(rnid_acc->gen_topology_data, *gen_topo_data);
|
bfa_os_assign(rnid_acc->gen_topology_data, *gen_topo_data);
|
||||||
return (sizeof(struct fc_rnid_acc_s));
|
return sizeof(struct fc_rnid_acc_s);
|
||||||
} else {
|
} else {
|
||||||
return (sizeof(struct fc_rnid_acc_s) -
|
return sizeof(struct fc_rnid_acc_s) -
|
||||||
sizeof(struct fc_rnid_general_topology_data_s));
|
sizeof(struct fc_rnid_general_topology_data_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -776,7 +776,7 @@ fc_rpsc_build(struct fchs_s *fchs, struct fc_rpsc_cmd_s *rpsc, u32 d_id,
|
||||||
memset(rpsc, 0, sizeof(struct fc_rpsc_cmd_s));
|
memset(rpsc, 0, sizeof(struct fc_rpsc_cmd_s));
|
||||||
|
|
||||||
rpsc->els_cmd.els_code = FC_ELS_RPSC;
|
rpsc->els_cmd.els_code = FC_ELS_RPSC;
|
||||||
return (sizeof(struct fc_rpsc_cmd_s));
|
return sizeof(struct fc_rpsc_cmd_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -797,8 +797,8 @@ fc_rpsc2_build(struct fchs_s *fchs, struct fc_rpsc2_cmd_s *rpsc2,
|
||||||
for (i = 0; i < npids; i++)
|
for (i = 0; i < npids; i++)
|
||||||
rpsc2->pid_list[i].pid = pid_list[i];
|
rpsc2->pid_list[i].pid = pid_list[i];
|
||||||
|
|
||||||
return (sizeof(struct fc_rpsc2_cmd_s) + ((npids - 1) *
|
return sizeof(struct fc_rpsc2_cmd_s) + ((npids - 1) *
|
||||||
(sizeof(u32))));
|
(sizeof(u32)));
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -819,7 +819,7 @@ fc_rpsc_acc_build(struct fchs_s *fchs, struct fc_rpsc_acc_s *rpsc_acc,
|
||||||
rpsc_acc->speed_info[0].port_op_speed =
|
rpsc_acc->speed_info[0].port_op_speed =
|
||||||
bfa_os_htons(oper_speed->port_op_speed);
|
bfa_os_htons(oper_speed->port_op_speed);
|
||||||
|
|
||||||
return (sizeof(struct fc_rpsc_acc_s));
|
return sizeof(struct fc_rpsc_acc_s);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -856,7 +856,7 @@ fc_pdisc_build(struct fchs_s *fchs, u32 d_id, u32 s_id,
|
||||||
pdisc->port_name = port_name;
|
pdisc->port_name = port_name;
|
||||||
pdisc->node_name = node_name;
|
pdisc->node_name = node_name;
|
||||||
|
|
||||||
return (sizeof(struct fc_logi_s));
|
return sizeof(struct fc_logi_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -865,21 +865,21 @@ fc_pdisc_rsp_parse(struct fchs_s *fchs, int len, wwn_t port_name)
|
||||||
struct fc_logi_s *pdisc = (struct fc_logi_s *) (fchs + 1);
|
struct fc_logi_s *pdisc = (struct fc_logi_s *) (fchs + 1);
|
||||||
|
|
||||||
if (len < sizeof(struct fc_logi_s))
|
if (len < sizeof(struct fc_logi_s))
|
||||||
return (FC_PARSE_LEN_INVAL);
|
return FC_PARSE_LEN_INVAL;
|
||||||
|
|
||||||
if (pdisc->els_cmd.els_code != FC_ELS_ACC)
|
if (pdisc->els_cmd.els_code != FC_ELS_ACC)
|
||||||
return (FC_PARSE_ACC_INVAL);
|
return FC_PARSE_ACC_INVAL;
|
||||||
|
|
||||||
if (!wwn_is_equal(pdisc->port_name, port_name))
|
if (!wwn_is_equal(pdisc->port_name, port_name))
|
||||||
return (FC_PARSE_PWWN_NOT_EQUAL);
|
return FC_PARSE_PWWN_NOT_EQUAL;
|
||||||
|
|
||||||
if (!pdisc->class3.class_valid)
|
if (!pdisc->class3.class_valid)
|
||||||
return (FC_PARSE_NWWN_NOT_EQUAL);
|
return FC_PARSE_NWWN_NOT_EQUAL;
|
||||||
|
|
||||||
if (bfa_os_ntohs(pdisc->class3.rxsz) < (FC_MIN_PDUSZ))
|
if (bfa_os_ntohs(pdisc->class3.rxsz) < (FC_MIN_PDUSZ))
|
||||||
return (FC_PARSE_RXSZ_INVAL);
|
return FC_PARSE_RXSZ_INVAL;
|
||||||
|
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -903,7 +903,7 @@ fc_prlo_build(struct fchs_s *fchs, u32 d_id, u32 s_id, u16 ox_id,
|
||||||
prlo->prlo_params[page].resp_process_assc = 0;
|
prlo->prlo_params[page].resp_process_assc = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (bfa_os_ntohs(prlo->payload_len));
|
return bfa_os_ntohs(prlo->payload_len);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -916,7 +916,7 @@ fc_prlo_rsp_parse(struct fchs_s *fchs, int len)
|
||||||
len = len;
|
len = len;
|
||||||
|
|
||||||
if (prlo->command != FC_ELS_ACC)
|
if (prlo->command != FC_ELS_ACC)
|
||||||
return (FC_PARSE_FAILURE);
|
return FC_PARSE_FAILURE;
|
||||||
|
|
||||||
num_pages = ((bfa_os_ntohs(prlo->payload_len)) - 4) / 16;
|
num_pages = ((bfa_os_ntohs(prlo->payload_len)) - 4) / 16;
|
||||||
|
|
||||||
|
@ -936,7 +936,7 @@ fc_prlo_rsp_parse(struct fchs_s *fchs, int len)
|
||||||
if (prlo->prlo_acc_params[page].resp_process_assc != 0)
|
if (prlo->prlo_acc_params[page].resp_process_assc != 0)
|
||||||
return FC_PARSE_FAILURE;
|
return FC_PARSE_FAILURE;
|
||||||
}
|
}
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -968,7 +968,7 @@ fc_tprlo_build(struct fchs_s *fchs, u32 d_id, u32 s_id,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return (bfa_os_ntohs(tprlo->payload_len));
|
return bfa_os_ntohs(tprlo->payload_len);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -981,23 +981,23 @@ fc_tprlo_rsp_parse(struct fchs_s *fchs, int len)
|
||||||
len = len;
|
len = len;
|
||||||
|
|
||||||
if (tprlo->command != FC_ELS_ACC)
|
if (tprlo->command != FC_ELS_ACC)
|
||||||
return (FC_PARSE_ACC_INVAL);
|
return FC_PARSE_ACC_INVAL;
|
||||||
|
|
||||||
num_pages = (bfa_os_ntohs(tprlo->payload_len) - 4) / 16;
|
num_pages = (bfa_os_ntohs(tprlo->payload_len) - 4) / 16;
|
||||||
|
|
||||||
for (page = 0; page < num_pages; page++) {
|
for (page = 0; page < num_pages; page++) {
|
||||||
if (tprlo->tprlo_acc_params[page].type != FC_TYPE_FCP)
|
if (tprlo->tprlo_acc_params[page].type != FC_TYPE_FCP)
|
||||||
return (FC_PARSE_NOT_FCP);
|
return FC_PARSE_NOT_FCP;
|
||||||
if (tprlo->tprlo_acc_params[page].opa_valid != 0)
|
if (tprlo->tprlo_acc_params[page].opa_valid != 0)
|
||||||
return (FC_PARSE_OPAFLAG_INVAL);
|
return FC_PARSE_OPAFLAG_INVAL;
|
||||||
if (tprlo->tprlo_acc_params[page].rpa_valid != 0)
|
if (tprlo->tprlo_acc_params[page].rpa_valid != 0)
|
||||||
return (FC_PARSE_RPAFLAG_INVAL);
|
return FC_PARSE_RPAFLAG_INVAL;
|
||||||
if (tprlo->tprlo_acc_params[page].orig_process_assc != 0)
|
if (tprlo->tprlo_acc_params[page].orig_process_assc != 0)
|
||||||
return (FC_PARSE_OPA_INVAL);
|
return FC_PARSE_OPA_INVAL;
|
||||||
if (tprlo->tprlo_acc_params[page].resp_process_assc != 0)
|
if (tprlo->tprlo_acc_params[page].resp_process_assc != 0)
|
||||||
return (FC_PARSE_RPA_INVAL);
|
return FC_PARSE_RPA_INVAL;
|
||||||
}
|
}
|
||||||
return (FC_PARSE_OK);
|
return FC_PARSE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
enum fc_parse_status
|
enum fc_parse_status
|
||||||
|
@ -1024,7 +1024,7 @@ fc_ba_rjt_build(struct fchs_s *fchs, u32 d_id, u32 s_id,
|
||||||
fchs->cat_info = FC_CAT_BA_RJT;
|
fchs->cat_info = FC_CAT_BA_RJT;
|
||||||
ba_rjt->reason_code = reason_code;
|
ba_rjt->reason_code = reason_code;
|
||||||
ba_rjt->reason_expl = reason_expl;
|
ba_rjt->reason_expl = reason_expl;
|
||||||
return (sizeof(struct fc_ba_rjt_s));
|
return sizeof(struct fc_ba_rjt_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -1073,7 +1073,7 @@ fc_gidpn_build(struct fchs_s *fchs, void *pyld, u32 s_id, u16 ox_id,
|
||||||
|
|
||||||
bfa_os_memset(gidpn, 0, sizeof(struct fcgs_gidpn_req_s));
|
bfa_os_memset(gidpn, 0, sizeof(struct fcgs_gidpn_req_s));
|
||||||
gidpn->port_name = port_name;
|
gidpn->port_name = port_name;
|
||||||
return (sizeof(struct fcgs_gidpn_req_s) + sizeof(struct ct_hdr_s));
|
return sizeof(struct fcgs_gidpn_req_s) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1090,7 +1090,7 @@ fc_gpnid_build(struct fchs_s *fchs, void *pyld, u32 s_id, u16 ox_id,
|
||||||
|
|
||||||
bfa_os_memset(gpnid, 0, sizeof(fcgs_gpnid_req_t));
|
bfa_os_memset(gpnid, 0, sizeof(fcgs_gpnid_req_t));
|
||||||
gpnid->dap = port_id;
|
gpnid->dap = port_id;
|
||||||
return (sizeof(fcgs_gpnid_req_t) + sizeof(struct ct_hdr_s));
|
return sizeof(fcgs_gpnid_req_t) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1107,7 +1107,7 @@ fc_gnnid_build(struct fchs_s *fchs, void *pyld, u32 s_id, u16 ox_id,
|
||||||
|
|
||||||
bfa_os_memset(gnnid, 0, sizeof(fcgs_gnnid_req_t));
|
bfa_os_memset(gnnid, 0, sizeof(fcgs_gnnid_req_t));
|
||||||
gnnid->dap = port_id;
|
gnnid->dap = port_id;
|
||||||
return (sizeof(fcgs_gnnid_req_t) + sizeof(struct ct_hdr_s));
|
return sizeof(fcgs_gnnid_req_t) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1137,7 +1137,7 @@ fc_scr_build(struct fchs_s *fchs, struct fc_scr_s *scr, u8 set_br_reg,
|
||||||
if (set_br_reg)
|
if (set_br_reg)
|
||||||
scr->vu_reg_func = FC_VU_SCR_REG_FUNC_FABRIC_NAME_CHANGE;
|
scr->vu_reg_func = FC_VU_SCR_REG_FUNC_FABRIC_NAME_CHANGE;
|
||||||
|
|
||||||
return (sizeof(struct fc_scr_s));
|
return sizeof(struct fc_scr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1157,7 +1157,7 @@ fc_rscn_build(struct fchs_s *fchs, struct fc_rscn_pl_s *rscn, u32 s_id,
|
||||||
rscn->event[0].format = FC_RSCN_FORMAT_PORTID;
|
rscn->event[0].format = FC_RSCN_FORMAT_PORTID;
|
||||||
rscn->event[0].portid = s_id;
|
rscn->event[0].portid = s_id;
|
||||||
|
|
||||||
return (sizeof(struct fc_rscn_pl_s));
|
return sizeof(struct fc_rscn_pl_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1188,7 +1188,7 @@ fc_rftid_build(struct fchs_s *fchs, void *pyld, u32 s_id, u16 ox_id,
|
||||||
rftid->fc4_type[index] |= bfa_os_htonl(type_value);
|
rftid->fc4_type[index] |= bfa_os_htonl(type_value);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (sizeof(struct fcgs_rftid_req_s) + sizeof(struct ct_hdr_s));
|
return sizeof(struct fcgs_rftid_req_s) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1210,7 +1210,7 @@ fc_rftid_build_sol(struct fchs_s *fchs, void *pyld, u32 s_id,
|
||||||
bfa_os_memcpy((void *)rftid->fc4_type, (void *)fc4_bitmap,
|
bfa_os_memcpy((void *)rftid->fc4_type, (void *)fc4_bitmap,
|
||||||
(bitmap_size < 32 ? bitmap_size : 32));
|
(bitmap_size < 32 ? bitmap_size : 32));
|
||||||
|
|
||||||
return (sizeof(struct fcgs_rftid_req_s) + sizeof(struct ct_hdr_s));
|
return sizeof(struct fcgs_rftid_req_s) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1231,7 +1231,7 @@ fc_rffid_build(struct fchs_s *fchs, void *pyld, u32 s_id, u16 ox_id,
|
||||||
rffid->fc4ftr_bits = fc4_ftrs;
|
rffid->fc4ftr_bits = fc4_ftrs;
|
||||||
rffid->fc4_type = fc4_type;
|
rffid->fc4_type = fc4_type;
|
||||||
|
|
||||||
return (sizeof(struct fcgs_rffid_req_s) + sizeof(struct ct_hdr_s));
|
return sizeof(struct fcgs_rffid_req_s) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1253,7 +1253,7 @@ fc_rspnid_build(struct fchs_s *fchs, void *pyld, u32 s_id, u16 ox_id,
|
||||||
rspnid->spn_len = (u8) strlen((char *)name);
|
rspnid->spn_len = (u8) strlen((char *)name);
|
||||||
strncpy((char *)rspnid->spn, (char *)name, rspnid->spn_len);
|
strncpy((char *)rspnid->spn, (char *)name, rspnid->spn_len);
|
||||||
|
|
||||||
return (sizeof(struct fcgs_rspnid_req_s) + sizeof(struct ct_hdr_s));
|
return sizeof(struct fcgs_rspnid_req_s) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1275,7 +1275,7 @@ fc_gid_ft_build(struct fchs_s *fchs, void *pyld, u32 s_id,
|
||||||
gidft->domain_id = 0;
|
gidft->domain_id = 0;
|
||||||
gidft->area_id = 0;
|
gidft->area_id = 0;
|
||||||
|
|
||||||
return (sizeof(struct fcgs_gidft_req_s) + sizeof(struct ct_hdr_s));
|
return sizeof(struct fcgs_gidft_req_s) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1294,7 +1294,7 @@ fc_rpnid_build(struct fchs_s *fchs, void *pyld, u32 s_id, u32 port_id,
|
||||||
rpnid->port_id = port_id;
|
rpnid->port_id = port_id;
|
||||||
rpnid->port_name = port_name;
|
rpnid->port_name = port_name;
|
||||||
|
|
||||||
return (sizeof(struct fcgs_rpnid_req_s) + sizeof(struct ct_hdr_s));
|
return sizeof(struct fcgs_rpnid_req_s) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1313,7 +1313,7 @@ fc_rnnid_build(struct fchs_s *fchs, void *pyld, u32 s_id, u32 port_id,
|
||||||
rnnid->port_id = port_id;
|
rnnid->port_id = port_id;
|
||||||
rnnid->node_name = node_name;
|
rnnid->node_name = node_name;
|
||||||
|
|
||||||
return (sizeof(struct fcgs_rnnid_req_s) + sizeof(struct ct_hdr_s));
|
return sizeof(struct fcgs_rnnid_req_s) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1332,7 +1332,7 @@ fc_rcsid_build(struct fchs_s *fchs, void *pyld, u32 s_id, u32 port_id,
|
||||||
rcsid->port_id = port_id;
|
rcsid->port_id = port_id;
|
||||||
rcsid->cos = cos;
|
rcsid->cos = cos;
|
||||||
|
|
||||||
return (sizeof(struct fcgs_rcsid_req_s) + sizeof(struct ct_hdr_s));
|
return sizeof(struct fcgs_rcsid_req_s) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1351,7 +1351,7 @@ fc_rptid_build(struct fchs_s *fchs, void *pyld, u32 s_id, u32 port_id,
|
||||||
rptid->port_id = port_id;
|
rptid->port_id = port_id;
|
||||||
rptid->port_type = port_type;
|
rptid->port_type = port_type;
|
||||||
|
|
||||||
return (sizeof(struct fcgs_rptid_req_s) + sizeof(struct ct_hdr_s));
|
return sizeof(struct fcgs_rptid_req_s) + sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
u16
|
u16
|
||||||
|
@ -1368,7 +1368,7 @@ fc_ganxt_build(struct fchs_s *fchs, void *pyld, u32 s_id, u32 port_id)
|
||||||
bfa_os_memset(ganxt, 0, sizeof(struct fcgs_ganxt_req_s));
|
bfa_os_memset(ganxt, 0, sizeof(struct fcgs_ganxt_req_s));
|
||||||
ganxt->port_id = port_id;
|
ganxt->port_id = port_id;
|
||||||
|
|
||||||
return (sizeof(struct ct_hdr_s) + sizeof(struct fcgs_ganxt_req_s));
|
return sizeof(struct ct_hdr_s) + sizeof(struct fcgs_ganxt_req_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1385,7 +1385,7 @@ fc_fdmi_reqhdr_build(struct fchs_s *fchs, void *pyld, u32 s_id,
|
||||||
fc_gs_fchdr_build(fchs, d_id, s_id, 0);
|
fc_gs_fchdr_build(fchs, d_id, s_id, 0);
|
||||||
fc_gs_fdmi_cthdr_build(cthdr, s_id, cmd_code);
|
fc_gs_fdmi_cthdr_build(cthdr, s_id, cmd_code);
|
||||||
|
|
||||||
return (sizeof(struct ct_hdr_s));
|
return sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1425,7 +1425,7 @@ fc_gmal_req_build(struct fchs_s *fchs, void *pyld, u32 s_id, wwn_t wwn)
|
||||||
bfa_os_memset(gmal, 0, sizeof(fcgs_gmal_req_t));
|
bfa_os_memset(gmal, 0, sizeof(fcgs_gmal_req_t));
|
||||||
gmal->wwn = wwn;
|
gmal->wwn = wwn;
|
||||||
|
|
||||||
return (sizeof(struct ct_hdr_s) + sizeof(fcgs_gmal_req_t));
|
return sizeof(struct ct_hdr_s) + sizeof(fcgs_gmal_req_t);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1445,5 +1445,5 @@ fc_gfn_req_build(struct fchs_s *fchs, void *pyld, u32 s_id, wwn_t wwn)
|
||||||
bfa_os_memset(gfn, 0, sizeof(fcgs_gfn_req_t));
|
bfa_os_memset(gfn, 0, sizeof(fcgs_gfn_req_t));
|
||||||
gfn->wwn = wwn;
|
gfn->wwn = wwn;
|
||||||
|
|
||||||
return (sizeof(struct ct_hdr_s) + sizeof(fcgs_gfn_req_t));
|
return sizeof(struct ct_hdr_s) + sizeof(fcgs_gfn_req_t);
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,8 +32,8 @@
|
||||||
* Utility Macros/functions
|
* Utility Macros/functions
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define fcif_sof_set(_ifhdr, _sof) (_ifhdr)->sof = FC_ ## _sof
|
#define fcif_sof_set(_ifhdr, _sof) ((_ifhdr)->sof = FC_ ## _sof)
|
||||||
#define fcif_eof_set(_ifhdr, _eof) (_ifhdr)->eof = FC_ ## _eof
|
#define fcif_eof_set(_ifhdr, _eof) ((_ifhdr)->eof = FC_ ## _eof)
|
||||||
|
|
||||||
#define wwn_is_equal(_wwn1, _wwn2) \
|
#define wwn_is_equal(_wwn1, _wwn2) \
|
||||||
(memcmp(&(_wwn1), &(_wwn2), sizeof(wwn_t)) == 0)
|
(memcmp(&(_wwn1), &(_wwn2), sizeof(wwn_t)) == 0)
|
||||||
|
@ -49,7 +49,7 @@
|
||||||
static inline u32
|
static inline u32
|
||||||
fc_get_ctresp_pyld_len(u32 resp_len)
|
fc_get_ctresp_pyld_len(u32 resp_len)
|
||||||
{
|
{
|
||||||
return (resp_len - sizeof(struct ct_hdr_s));
|
return resp_len - sizeof(struct ct_hdr_s);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -286,11 +286,10 @@ bfa_fcs_itnim_sm_online(struct bfa_fcs_itnim_s *itnim,
|
||||||
bfa_sm_set_state(itnim, bfa_fcs_itnim_sm_hcb_offline);
|
bfa_sm_set_state(itnim, bfa_fcs_itnim_sm_hcb_offline);
|
||||||
bfa_fcb_itnim_offline(itnim->itnim_drv);
|
bfa_fcb_itnim_offline(itnim->itnim_drv);
|
||||||
bfa_itnim_offline(itnim->bfa_itnim);
|
bfa_itnim_offline(itnim->bfa_itnim);
|
||||||
if (bfa_fcs_port_is_online(itnim->rport->port) == BFA_TRUE) {
|
if (bfa_fcs_port_is_online(itnim->rport->port) == BFA_TRUE)
|
||||||
bfa_fcs_itnim_aen_post(itnim, BFA_ITNIM_AEN_DISCONNECT);
|
bfa_fcs_itnim_aen_post(itnim, BFA_ITNIM_AEN_DISCONNECT);
|
||||||
} else {
|
else
|
||||||
bfa_fcs_itnim_aen_post(itnim, BFA_ITNIM_AEN_OFFLINE);
|
bfa_fcs_itnim_aen_post(itnim, BFA_ITNIM_AEN_OFFLINE);
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case BFA_FCS_ITNIM_SM_DELETE:
|
case BFA_FCS_ITNIM_SM_DELETE:
|
||||||
|
@ -732,7 +731,7 @@ bfa_fcs_itnim_lookup(struct bfa_fcs_port_s *port, wwn_t rpwwn)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
bfa_assert(rport->itnim != NULL);
|
bfa_assert(rport->itnim != NULL);
|
||||||
return (rport->itnim);
|
return rport->itnim;
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_status_t
|
bfa_status_t
|
||||||
|
|
|
@ -23,7 +23,7 @@
|
||||||
#ifndef __FCS_H__
|
#ifndef __FCS_H__
|
||||||
#define __FCS_H__
|
#define __FCS_H__
|
||||||
|
|
||||||
#define __fcs_min_cfg(__fcs) (__fcs)->min_cfg
|
#define __fcs_min_cfg(__fcs) ((__fcs)->min_cfg)
|
||||||
|
|
||||||
void bfa_fcs_modexit_comp(struct bfa_fcs_s *fcs);
|
void bfa_fcs_modexit_comp(struct bfa_fcs_s *fcs);
|
||||||
|
|
||||||
|
|
|
@ -72,9 +72,9 @@ static u16 bfa_fcs_port_fdmi_build_rpa_pyld(
|
||||||
struct bfa_fcs_port_fdmi_s *fdmi, u8 *pyld);
|
struct bfa_fcs_port_fdmi_s *fdmi, u8 *pyld);
|
||||||
static u16 bfa_fcs_port_fdmi_build_portattr_block(
|
static u16 bfa_fcs_port_fdmi_build_portattr_block(
|
||||||
struct bfa_fcs_port_fdmi_s *fdmi, u8 *pyld);
|
struct bfa_fcs_port_fdmi_s *fdmi, u8 *pyld);
|
||||||
void bfa_fcs_fdmi_get_hbaattr(struct bfa_fcs_port_fdmi_s *fdmi,
|
static void bfa_fcs_fdmi_get_hbaattr(struct bfa_fcs_port_fdmi_s *fdmi,
|
||||||
struct bfa_fcs_fdmi_hba_attr_s *hba_attr);
|
struct bfa_fcs_fdmi_hba_attr_s *hba_attr);
|
||||||
void bfa_fcs_fdmi_get_portattr(struct bfa_fcs_port_fdmi_s *fdmi,
|
static void bfa_fcs_fdmi_get_portattr(struct bfa_fcs_port_fdmi_s *fdmi,
|
||||||
struct bfa_fcs_fdmi_port_attr_s *port_attr);
|
struct bfa_fcs_fdmi_port_attr_s *port_attr);
|
||||||
/**
|
/**
|
||||||
* fcs_fdmi_sm FCS FDMI state machine
|
* fcs_fdmi_sm FCS FDMI state machine
|
||||||
|
@ -1091,7 +1091,7 @@ bfa_fcs_port_fdmi_timeout(void *arg)
|
||||||
bfa_sm_send_event(fdmi, FDMISM_EVENT_TIMEOUT);
|
bfa_sm_send_event(fdmi, FDMISM_EVENT_TIMEOUT);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
static void
|
||||||
bfa_fcs_fdmi_get_hbaattr(struct bfa_fcs_port_fdmi_s *fdmi,
|
bfa_fcs_fdmi_get_hbaattr(struct bfa_fcs_port_fdmi_s *fdmi,
|
||||||
struct bfa_fcs_fdmi_hba_attr_s *hba_attr)
|
struct bfa_fcs_fdmi_hba_attr_s *hba_attr)
|
||||||
{
|
{
|
||||||
|
@ -1145,7 +1145,7 @@ bfa_fcs_fdmi_get_hbaattr(struct bfa_fcs_port_fdmi_s *fdmi,
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
static void
|
||||||
bfa_fcs_fdmi_get_portattr(struct bfa_fcs_port_fdmi_s *fdmi,
|
bfa_fcs_fdmi_get_portattr(struct bfa_fcs_port_fdmi_s *fdmi,
|
||||||
struct bfa_fcs_fdmi_port_attr_s *port_attr)
|
struct bfa_fcs_fdmi_port_attr_s *port_attr)
|
||||||
{
|
{
|
||||||
|
|
|
@ -54,7 +54,7 @@ bfa_aen_get_max_cfg_entry(void)
|
||||||
static inline s32
|
static inline s32
|
||||||
bfa_aen_get_meminfo(void)
|
bfa_aen_get_meminfo(void)
|
||||||
{
|
{
|
||||||
return (sizeof(struct bfa_aen_entry_s) * bfa_aen_get_max_cfg_entry());
|
return sizeof(struct bfa_aen_entry_s) * bfa_aen_get_max_cfg_entry();
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline s32
|
static inline s32
|
||||||
|
|
|
@ -76,11 +76,11 @@ struct bfa_meminfo_s {
|
||||||
struct bfa_mem_elem_s meminfo[BFA_MEM_TYPE_MAX];
|
struct bfa_mem_elem_s meminfo[BFA_MEM_TYPE_MAX];
|
||||||
};
|
};
|
||||||
#define bfa_meminfo_kva(_m) \
|
#define bfa_meminfo_kva(_m) \
|
||||||
(_m)->meminfo[BFA_MEM_TYPE_KVA - 1].kva_curp
|
((_m)->meminfo[BFA_MEM_TYPE_KVA - 1].kva_curp)
|
||||||
#define bfa_meminfo_dma_virt(_m) \
|
#define bfa_meminfo_dma_virt(_m) \
|
||||||
(_m)->meminfo[BFA_MEM_TYPE_DMA - 1].kva_curp
|
((_m)->meminfo[BFA_MEM_TYPE_DMA - 1].kva_curp)
|
||||||
#define bfa_meminfo_dma_phys(_m) \
|
#define bfa_meminfo_dma_phys(_m) \
|
||||||
(_m)->meminfo[BFA_MEM_TYPE_DMA - 1].dma_curp
|
((_m)->meminfo[BFA_MEM_TYPE_DMA - 1].dma_curp)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generic Scatter Gather Element used by driver
|
* Generic Scatter Gather Element used by driver
|
||||||
|
@ -100,7 +100,7 @@ struct bfa_sge_s {
|
||||||
/*
|
/*
|
||||||
* bfa stats interfaces
|
* bfa stats interfaces
|
||||||
*/
|
*/
|
||||||
#define bfa_stats(_mod, _stats) (_mod)->stats._stats ++
|
#define bfa_stats(_mod, _stats) ((_mod)->stats._stats++)
|
||||||
|
|
||||||
#define bfa_ioc_get_stats(__bfa, __ioc_stats) \
|
#define bfa_ioc_get_stats(__bfa, __ioc_stats) \
|
||||||
bfa_ioc_fetch_stats(&(__bfa)->ioc, __ioc_stats)
|
bfa_ioc_fetch_stats(&(__bfa)->ioc, __ioc_stats)
|
||||||
|
@ -136,7 +136,7 @@ void bfa_isr_enable(struct bfa_s *bfa);
|
||||||
void bfa_isr_disable(struct bfa_s *bfa);
|
void bfa_isr_disable(struct bfa_s *bfa);
|
||||||
void bfa_msix_getvecs(struct bfa_s *bfa, u32 *msix_vecs_bmap,
|
void bfa_msix_getvecs(struct bfa_s *bfa, u32 *msix_vecs_bmap,
|
||||||
u32 *num_vecs, u32 *max_vec_bit);
|
u32 *num_vecs, u32 *max_vec_bit);
|
||||||
#define bfa_msix(__bfa, __vec) (__bfa)->msix.handler[__vec](__bfa, __vec)
|
#define bfa_msix(__bfa, __vec) ((__bfa)->msix.handler[__vec](__bfa, __vec))
|
||||||
|
|
||||||
void bfa_comp_deq(struct bfa_s *bfa, struct list_head *comp_q);
|
void bfa_comp_deq(struct bfa_s *bfa, struct list_head *comp_q);
|
||||||
void bfa_comp_process(struct bfa_s *bfa, struct list_head *comp_q);
|
void bfa_comp_process(struct bfa_s *bfa, struct list_head *comp_q);
|
||||||
|
|
|
@ -34,10 +34,10 @@ struct bfa_fcxp_s;
|
||||||
*/
|
*/
|
||||||
struct bfa_rport_info_s {
|
struct bfa_rport_info_s {
|
||||||
u16 max_frmsz; /* max rcv pdu size */
|
u16 max_frmsz; /* max rcv pdu size */
|
||||||
u32 pid : 24, /* remote port ID */
|
u32 pid:24, /* remote port ID */
|
||||||
lp_tag : 8;
|
lp_tag:8;
|
||||||
u32 local_pid : 24, /* local port ID */
|
u32 local_pid:24, /* local port ID */
|
||||||
cisc : 8; /* CIRO supported */
|
cisc:8; /* CIRO supported */
|
||||||
u8 fc_class; /* supported FC classes. enum fc_cos */
|
u8 fc_class; /* supported FC classes. enum fc_cos */
|
||||||
u8 vf_en; /* virtual fabric enable */
|
u8 vf_en; /* virtual fabric enable */
|
||||||
u16 vf_id; /* virtual fabric ID */
|
u16 vf_id; /* virtual fabric ID */
|
||||||
|
|
|
@ -93,13 +93,13 @@ union bfi_addr_u {
|
||||||
*/
|
*/
|
||||||
struct bfi_sge_s {
|
struct bfi_sge_s {
|
||||||
#ifdef __BIGENDIAN
|
#ifdef __BIGENDIAN
|
||||||
u32 flags : 2,
|
u32 flags:2,
|
||||||
rsvd : 2,
|
rsvd:2,
|
||||||
sg_len : 28;
|
sg_len:28;
|
||||||
#else
|
#else
|
||||||
u32 sg_len : 28,
|
u32 sg_len:28,
|
||||||
rsvd : 2,
|
rsvd:2,
|
||||||
flags : 2;
|
flags:2;
|
||||||
#endif
|
#endif
|
||||||
union bfi_addr_u sga;
|
union bfi_addr_u sga;
|
||||||
};
|
};
|
||||||
|
|
|
@ -142,7 +142,7 @@ enum {
|
||||||
BFI_ADAPTER_UNSUPP = 0x400000, /* unknown adapter type */
|
BFI_ADAPTER_UNSUPP = 0x400000, /* unknown adapter type */
|
||||||
};
|
};
|
||||||
|
|
||||||
#define BFI_ADAPTER_GETP(__prop,__adap_prop) \
|
#define BFI_ADAPTER_GETP(__prop, __adap_prop) \
|
||||||
(((__adap_prop) & BFI_ADAPTER_ ## __prop ## _MK) >> \
|
(((__adap_prop) & BFI_ADAPTER_ ## __prop ## _MK) >> \
|
||||||
BFI_ADAPTER_ ## __prop ## _SH)
|
BFI_ADAPTER_ ## __prop ## _SH)
|
||||||
#define BFI_ADAPTER_SETP(__prop, __val) \
|
#define BFI_ADAPTER_SETP(__prop, __val) \
|
||||||
|
|
|
@ -55,8 +55,8 @@ struct bfi_lps_login_rsp_s {
|
||||||
u16 bb_credit;
|
u16 bb_credit;
|
||||||
u8 f_port;
|
u8 f_port;
|
||||||
u8 npiv_en;
|
u8 npiv_en;
|
||||||
u32 lp_pid : 24;
|
u32 lp_pid:24;
|
||||||
u32 auth_req : 8;
|
u32 auth_req:8;
|
||||||
mac_t lp_mac;
|
mac_t lp_mac;
|
||||||
mac_t fcf_mac;
|
mac_t fcf_mac;
|
||||||
u8 ext_status;
|
u8 ext_status;
|
||||||
|
|
|
@ -38,10 +38,10 @@ struct bfi_rport_create_req_s {
|
||||||
struct bfi_mhdr_s mh; /* common msg header */
|
struct bfi_mhdr_s mh; /* common msg header */
|
||||||
u16 bfa_handle; /* host rport handle */
|
u16 bfa_handle; /* host rport handle */
|
||||||
u16 max_frmsz; /* max rcv pdu size */
|
u16 max_frmsz; /* max rcv pdu size */
|
||||||
u32 pid : 24, /* remote port ID */
|
u32 pid:24, /* remote port ID */
|
||||||
lp_tag : 8; /* local port tag */
|
lp_tag:8; /* local port tag */
|
||||||
u32 local_pid : 24, /* local port ID */
|
u32 local_pid:24, /* local port ID */
|
||||||
cisc : 8;
|
cisc:8;
|
||||||
u8 fc_class; /* supported FC classes */
|
u8 fc_class; /* supported FC classes */
|
||||||
u8 vf_en; /* virtual fabric enable */
|
u8 vf_en; /* virtual fabric enable */
|
||||||
u16 vf_id; /* virtual fabric ID */
|
u16 vf_id; /* virtual fabric ID */
|
||||||
|
|
|
@ -31,7 +31,7 @@ bfa_checksum_u32(u32 *buf, int sz)
|
||||||
for (i = 0; i < m; i++)
|
for (i = 0; i < m; i++)
|
||||||
sum ^= buf[i];
|
sum ^= buf[i];
|
||||||
|
|
||||||
return (sum);
|
return sum;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline u16
|
static inline u16
|
||||||
|
@ -43,7 +43,7 @@ bfa_checksum_u16(u16 *buf, int sz)
|
||||||
for (i = 0; i < m; i++)
|
for (i = 0; i < m; i++)
|
||||||
sum ^= buf[i];
|
sum ^= buf[i];
|
||||||
|
|
||||||
return (sum);
|
return sum;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline u8
|
static inline u8
|
||||||
|
@ -55,6 +55,6 @@ bfa_checksum_u8(u8 *buf, int sz)
|
||||||
for (i = 0; i < sz; i++)
|
for (i = 0; i < sz; i++)
|
||||||
sum ^= buf[i];
|
sum ^= buf[i];
|
||||||
|
|
||||||
return (sum);
|
return sum;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -24,8 +24,8 @@
|
||||||
|
|
||||||
typedef void (*bfa_sm_t)(void *sm, int event);
|
typedef void (*bfa_sm_t)(void *sm, int event);
|
||||||
|
|
||||||
#define bfa_sm_set_state(_sm, _state) (_sm)->sm = (bfa_sm_t)(_state)
|
#define bfa_sm_set_state(_sm, _state) ((_sm)->sm = (bfa_sm_t)(_state))
|
||||||
#define bfa_sm_send_event(_sm, _event) (_sm)->sm((_sm), (_event))
|
#define bfa_sm_send_event(_sm, _event) ((_sm)->sm((_sm), (_event)))
|
||||||
#define bfa_sm_get_state(_sm) ((_sm)->sm)
|
#define bfa_sm_get_state(_sm) ((_sm)->sm)
|
||||||
#define bfa_sm_cmp_state(_sm, _state) ((_sm)->sm == (bfa_sm_t)(_state))
|
#define bfa_sm_cmp_state(_sm, _state) ((_sm)->sm == (bfa_sm_t)(_state))
|
||||||
|
|
||||||
|
@ -62,7 +62,7 @@ typedef void (*bfa_fsm_t)(void *fsm, int event);
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
#define bfa_fsm_send_event(_fsm, _event) \
|
#define bfa_fsm_send_event(_fsm, _event) \
|
||||||
(_fsm)->fsm((_fsm), (_event))
|
((_fsm)->fsm((_fsm), (_event)))
|
||||||
#define bfa_fsm_cmp_state(_fsm, _state) \
|
#define bfa_fsm_cmp_state(_fsm, _state) \
|
||||||
((_fsm)->fsm == (bfa_fsm_t)(_state))
|
((_fsm)->fsm == (bfa_fsm_t)(_state))
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,7 @@
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef BFA_TRC_TS
|
#ifndef BFA_TRC_TS
|
||||||
#define BFA_TRC_TS(_trcm) ((_trcm)->ticks ++)
|
#define BFA_TRC_TS(_trcm) ((_trcm)->ticks++)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
struct bfa_trc_s {
|
struct bfa_trc_s {
|
||||||
|
|
|
@ -75,7 +75,7 @@ struct bfa_fcs_fabric_s {
|
||||||
*/
|
*/
|
||||||
};
|
};
|
||||||
|
|
||||||
#define bfa_fcs_fabric_npiv_capable(__f) (__f)->is_npiv
|
#define bfa_fcs_fabric_npiv_capable(__f) ((__f)->is_npiv)
|
||||||
#define bfa_fcs_fabric_is_switched(__f) \
|
#define bfa_fcs_fabric_is_switched(__f) \
|
||||||
((__f)->fab_type == BFA_FCS_FABRIC_SWITCHED)
|
((__f)->fab_type == BFA_FCS_FABRIC_SWITCHED)
|
||||||
|
|
||||||
|
|
|
@ -125,10 +125,10 @@ union bfa_fcs_port_topo_u {
|
||||||
struct bfa_fcs_port_s {
|
struct bfa_fcs_port_s {
|
||||||
struct list_head qe; /* used by port/vport */
|
struct list_head qe; /* used by port/vport */
|
||||||
bfa_sm_t sm; /* state machine */
|
bfa_sm_t sm; /* state machine */
|
||||||
struct bfa_fcs_fabric_s *fabric; /* parent fabric */
|
struct bfa_fcs_fabric_s *fabric;/* parent fabric */
|
||||||
struct bfa_port_cfg_s port_cfg; /* port configuration */
|
struct bfa_port_cfg_s port_cfg;/* port configuration */
|
||||||
struct bfa_timer_s link_timer; /* timer for link offline */
|
struct bfa_timer_s link_timer; /* timer for link offline */
|
||||||
u32 pid : 24; /* FC address */
|
u32 pid:24; /* FC address */
|
||||||
u8 lp_tag; /* lport tag */
|
u8 lp_tag; /* lport tag */
|
||||||
u16 num_rports; /* Num of r-ports */
|
u16 num_rports; /* Num of r-ports */
|
||||||
struct list_head rport_q; /* queue of discovered r-ports */
|
struct list_head rport_q; /* queue of discovered r-ports */
|
||||||
|
@ -188,13 +188,14 @@ bfa_fcs_port_get_drvport(struct bfa_fcs_port_s *port)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#define bfa_fcs_port_get_opertype(_lport) (_lport)->fabric->oper_type
|
#define bfa_fcs_port_get_opertype(_lport) ((_lport)->fabric->oper_type)
|
||||||
|
|
||||||
|
|
||||||
#define bfa_fcs_port_get_fabric_name(_lport) (_lport)->fabric->fabric_name
|
#define bfa_fcs_port_get_fabric_name(_lport) ((_lport)->fabric->fabric_name)
|
||||||
|
|
||||||
|
|
||||||
#define bfa_fcs_port_get_fabric_ipaddr(_lport) (_lport)->fabric->fabric_ip_addr
|
#define bfa_fcs_port_get_fabric_ipaddr(_lport) \
|
||||||
|
((_lport)->fabric->fabric_ip_addr)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* bfa fcs port public functions
|
* bfa fcs port public functions
|
||||||
|
|
|
@ -193,11 +193,11 @@ struct fcgs_rftid_req_s {
|
||||||
#define FC_GS_FCP_FC4_FEATURE_TARGET 0x01
|
#define FC_GS_FCP_FC4_FEATURE_TARGET 0x01
|
||||||
|
|
||||||
struct fcgs_rffid_req_s{
|
struct fcgs_rffid_req_s{
|
||||||
u32 rsvd :8;
|
u32 rsvd:8;
|
||||||
u32 dap :24; /* port identifier */
|
u32 dap:24; /* port identifier */
|
||||||
u32 rsvd1 :16;
|
u32 rsvd1:16;
|
||||||
u32 fc4ftr_bits :8; /* fc4 feature bits */
|
u32 fc4ftr_bits:8; /* fc4 feature bits */
|
||||||
u32 fc4_type :8; /* corresponding FC4 Type */
|
u32 fc4_type:8; /* corresponding FC4 Type */
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -486,14 +486,14 @@ struct fc_rsi_s {
|
||||||
* see FC-PH-X table 113 & 115 for explanation also FCP table 8
|
* see FC-PH-X table 113 & 115 for explanation also FCP table 8
|
||||||
*/
|
*/
|
||||||
struct fc_prli_params_s{
|
struct fc_prli_params_s{
|
||||||
u32 reserved: 16;
|
u32 reserved:16;
|
||||||
#ifdef __BIGENDIAN
|
#ifdef __BIGENDIAN
|
||||||
u32 reserved1: 5;
|
u32 reserved1:5;
|
||||||
u32 rec_support : 1;
|
u32 rec_support:1;
|
||||||
u32 task_retry_id : 1;
|
u32 task_retry_id:1;
|
||||||
u32 retry : 1;
|
u32 retry:1;
|
||||||
|
|
||||||
u32 confirm : 1;
|
u32 confirm:1;
|
||||||
u32 doverlay:1;
|
u32 doverlay:1;
|
||||||
u32 initiator:1;
|
u32 initiator:1;
|
||||||
u32 target:1;
|
u32 target:1;
|
||||||
|
@ -502,10 +502,10 @@ struct fc_prli_params_s{
|
||||||
u32 rxrdisab:1;
|
u32 rxrdisab:1;
|
||||||
u32 wxrdisab:1;
|
u32 wxrdisab:1;
|
||||||
#else
|
#else
|
||||||
u32 retry : 1;
|
u32 retry:1;
|
||||||
u32 task_retry_id : 1;
|
u32 task_retry_id:1;
|
||||||
u32 rec_support : 1;
|
u32 rec_support:1;
|
||||||
u32 reserved1: 5;
|
u32 reserved1:5;
|
||||||
|
|
||||||
u32 wxrdisab:1;
|
u32 wxrdisab:1;
|
||||||
u32 rxrdisab:1;
|
u32 rxrdisab:1;
|
||||||
|
@ -514,7 +514,7 @@ struct fc_prli_params_s{
|
||||||
u32 target:1;
|
u32 target:1;
|
||||||
u32 initiator:1;
|
u32 initiator:1;
|
||||||
u32 doverlay:1;
|
u32 doverlay:1;
|
||||||
u32 confirm : 1;
|
u32 confirm:1;
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -58,43 +58,10 @@ static const u8 port_loop_alpa_map[] = {
|
||||||
/*
|
/*
|
||||||
* Local Functions
|
* Local Functions
|
||||||
*/
|
*/
|
||||||
bfa_status_t bfa_fcs_port_loop_send_plogi(struct bfa_fcs_port_s *port,
|
static bfa_status_t bfa_fcs_port_loop_send_plogi(struct bfa_fcs_port_s *port,
|
||||||
u8 alpa);
|
u8 alpa);
|
||||||
|
|
||||||
void bfa_fcs_port_loop_plogi_response(void *fcsarg,
|
static void bfa_fcs_port_loop_plogi_response(void *fcsarg,
|
||||||
struct bfa_fcxp_s *fcxp,
|
|
||||||
void *cbarg,
|
|
||||||
bfa_status_t req_status,
|
|
||||||
u32 rsp_len,
|
|
||||||
u32 resid_len,
|
|
||||||
struct fchs_s *rsp_fchs);
|
|
||||||
|
|
||||||
bfa_status_t bfa_fcs_port_loop_send_adisc(struct bfa_fcs_port_s *port,
|
|
||||||
u8 alpa);
|
|
||||||
|
|
||||||
void bfa_fcs_port_loop_adisc_response(void *fcsarg,
|
|
||||||
struct bfa_fcxp_s *fcxp,
|
|
||||||
void *cbarg,
|
|
||||||
bfa_status_t req_status,
|
|
||||||
u32 rsp_len,
|
|
||||||
u32 resid_len,
|
|
||||||
struct fchs_s *rsp_fchs);
|
|
||||||
|
|
||||||
bfa_status_t bfa_fcs_port_loop_send_plogi_acc(struct bfa_fcs_port_s *port,
|
|
||||||
u8 alpa);
|
|
||||||
|
|
||||||
void bfa_fcs_port_loop_plogi_acc_response(void *fcsarg,
|
|
||||||
struct bfa_fcxp_s *fcxp,
|
|
||||||
void *cbarg,
|
|
||||||
bfa_status_t req_status,
|
|
||||||
u32 rsp_len,
|
|
||||||
u32 resid_len,
|
|
||||||
struct fchs_s *rsp_fchs);
|
|
||||||
|
|
||||||
bfa_status_t bfa_fcs_port_loop_send_adisc_acc(struct bfa_fcs_port_s *port,
|
|
||||||
u8 alpa);
|
|
||||||
|
|
||||||
void bfa_fcs_port_loop_adisc_acc_response(void *fcsarg,
|
|
||||||
struct bfa_fcxp_s *fcxp,
|
struct bfa_fcxp_s *fcxp,
|
||||||
void *cbarg,
|
void *cbarg,
|
||||||
bfa_status_t req_status,
|
bfa_status_t req_status,
|
||||||
|
@ -179,7 +146,7 @@ bfa_fcs_port_loop_lip(struct bfa_fcs_port_s *port)
|
||||||
/**
|
/**
|
||||||
* Local Functions.
|
* Local Functions.
|
||||||
*/
|
*/
|
||||||
bfa_status_t
|
static bfa_status_t
|
||||||
bfa_fcs_port_loop_send_plogi(struct bfa_fcs_port_s *port, u8 alpa)
|
bfa_fcs_port_loop_send_plogi(struct bfa_fcs_port_s *port, u8 alpa)
|
||||||
{
|
{
|
||||||
struct fchs_s fchs;
|
struct fchs_s fchs;
|
||||||
|
@ -208,7 +175,7 @@ bfa_fcs_port_loop_send_plogi(struct bfa_fcs_port_s *port, u8 alpa)
|
||||||
/**
|
/**
|
||||||
* Called by fcxp to notify the Plogi response
|
* Called by fcxp to notify the Plogi response
|
||||||
*/
|
*/
|
||||||
void
|
static void
|
||||||
bfa_fcs_port_loop_plogi_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
|
bfa_fcs_port_loop_plogi_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
|
||||||
void *cbarg, bfa_status_t req_status,
|
void *cbarg, bfa_status_t req_status,
|
||||||
u32 rsp_len, u32 resid_len,
|
u32 rsp_len, u32 resid_len,
|
||||||
|
@ -244,179 +211,3 @@ bfa_fcs_port_loop_plogi_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
|
||||||
bfa_assert(0);
|
bfa_assert(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_status_t
|
|
||||||
bfa_fcs_port_loop_send_plogi_acc(struct bfa_fcs_port_s *port, u8 alpa)
|
|
||||||
{
|
|
||||||
struct fchs_s fchs;
|
|
||||||
struct bfa_fcxp_s *fcxp;
|
|
||||||
int len;
|
|
||||||
|
|
||||||
bfa_trc(port->fcs, alpa);
|
|
||||||
|
|
||||||
fcxp = bfa_fcxp_alloc(NULL, port->fcs->bfa, 0, 0, NULL, NULL, NULL,
|
|
||||||
NULL);
|
|
||||||
bfa_assert(fcxp);
|
|
||||||
|
|
||||||
len = fc_plogi_acc_build(&fchs, bfa_fcxp_get_reqbuf(fcxp), alpa,
|
|
||||||
bfa_fcs_port_get_fcid(port), 0,
|
|
||||||
port->port_cfg.pwwn, port->port_cfg.nwwn,
|
|
||||||
bfa_pport_get_maxfrsize(port->fcs->bfa));
|
|
||||||
|
|
||||||
bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
|
|
||||||
FC_CLASS_3, len, &fchs,
|
|
||||||
bfa_fcs_port_loop_plogi_acc_response,
|
|
||||||
(void *)port, FC_MAX_PDUSZ, 0); /* No response
|
|
||||||
* expected
|
|
||||||
*/
|
|
||||||
|
|
||||||
return BFA_STATUS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Plogi Acc Response
|
|
||||||
* We donot do any processing here.
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
bfa_fcs_port_loop_plogi_acc_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
|
|
||||||
void *cbarg, bfa_status_t req_status,
|
|
||||||
u32 rsp_len, u32 resid_len,
|
|
||||||
struct fchs_s *rsp_fchs)
|
|
||||||
{
|
|
||||||
|
|
||||||
struct bfa_fcs_port_s *port = (struct bfa_fcs_port_s *) cbarg;
|
|
||||||
|
|
||||||
bfa_trc(port->fcs, port->pid);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Sanity Checks
|
|
||||||
*/
|
|
||||||
if (req_status != BFA_STATUS_OK) {
|
|
||||||
bfa_trc(port->fcs, req_status);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bfa_status_t
|
|
||||||
bfa_fcs_port_loop_send_adisc(struct bfa_fcs_port_s *port, u8 alpa)
|
|
||||||
{
|
|
||||||
struct fchs_s fchs;
|
|
||||||
struct bfa_fcxp_s *fcxp;
|
|
||||||
int len;
|
|
||||||
|
|
||||||
bfa_trc(port->fcs, alpa);
|
|
||||||
|
|
||||||
fcxp = bfa_fcxp_alloc(NULL, port->fcs->bfa, 0, 0, NULL, NULL, NULL,
|
|
||||||
NULL);
|
|
||||||
bfa_assert(fcxp);
|
|
||||||
|
|
||||||
len = fc_adisc_build(&fchs, bfa_fcxp_get_reqbuf(fcxp), alpa,
|
|
||||||
bfa_fcs_port_get_fcid(port), 0,
|
|
||||||
port->port_cfg.pwwn, port->port_cfg.nwwn);
|
|
||||||
|
|
||||||
bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
|
|
||||||
FC_CLASS_3, len, &fchs,
|
|
||||||
bfa_fcs_port_loop_adisc_response, (void *)port,
|
|
||||||
FC_MAX_PDUSZ, FC_RA_TOV);
|
|
||||||
|
|
||||||
return BFA_STATUS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Called by fcxp to notify the ADISC response
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
bfa_fcs_port_loop_adisc_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
|
|
||||||
void *cbarg, bfa_status_t req_status,
|
|
||||||
u32 rsp_len, u32 resid_len,
|
|
||||||
struct fchs_s *rsp_fchs)
|
|
||||||
{
|
|
||||||
struct bfa_fcs_port_s *port = (struct bfa_fcs_port_s *) cbarg;
|
|
||||||
struct bfa_fcs_rport_s *rport;
|
|
||||||
struct fc_adisc_s *adisc_resp;
|
|
||||||
struct fc_els_cmd_s *els_cmd;
|
|
||||||
u32 pid = rsp_fchs->s_id;
|
|
||||||
|
|
||||||
bfa_trc(port->fcs, req_status);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Sanity Checks
|
|
||||||
*/
|
|
||||||
if (req_status != BFA_STATUS_OK) {
|
|
||||||
/*
|
|
||||||
* TBD : we may need to retry certain requests
|
|
||||||
*/
|
|
||||||
bfa_fcxp_free(fcxp);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
els_cmd = (struct fc_els_cmd_s *) BFA_FCXP_RSP_PLD(fcxp);
|
|
||||||
adisc_resp = (struct fc_adisc_s *) els_cmd;
|
|
||||||
|
|
||||||
if (els_cmd->els_code == FC_ELS_ACC) {
|
|
||||||
} else {
|
|
||||||
bfa_trc(port->fcs, adisc_resp->els_cmd.els_code);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* TBD: we may need to check for reject codes and retry
|
|
||||||
*/
|
|
||||||
rport = bfa_fcs_port_get_rport_by_pid(port, pid);
|
|
||||||
if (rport) {
|
|
||||||
list_del(&rport->qe);
|
|
||||||
bfa_fcs_rport_delete(rport);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
bfa_status_t
|
|
||||||
bfa_fcs_port_loop_send_adisc_acc(struct bfa_fcs_port_s *port, u8 alpa)
|
|
||||||
{
|
|
||||||
struct fchs_s fchs;
|
|
||||||
struct bfa_fcxp_s *fcxp;
|
|
||||||
int len;
|
|
||||||
|
|
||||||
bfa_trc(port->fcs, alpa);
|
|
||||||
|
|
||||||
fcxp = bfa_fcxp_alloc(NULL, port->fcs->bfa, 0, 0, NULL, NULL, NULL,
|
|
||||||
NULL);
|
|
||||||
bfa_assert(fcxp);
|
|
||||||
|
|
||||||
len = fc_adisc_acc_build(&fchs, bfa_fcxp_get_reqbuf(fcxp), alpa,
|
|
||||||
bfa_fcs_port_get_fcid(port), 0,
|
|
||||||
port->port_cfg.pwwn, port->port_cfg.nwwn);
|
|
||||||
|
|
||||||
bfa_fcxp_send(fcxp, NULL, port->fabric->vf_id, port->lp_tag, BFA_FALSE,
|
|
||||||
FC_CLASS_3, len, &fchs,
|
|
||||||
bfa_fcs_port_loop_adisc_acc_response,
|
|
||||||
(void *)port, FC_MAX_PDUSZ, 0); /* no reponse
|
|
||||||
* expected
|
|
||||||
*/
|
|
||||||
|
|
||||||
return BFA_STATUS_OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Adisc Acc Response
|
|
||||||
* We donot do any processing here.
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
bfa_fcs_port_loop_adisc_acc_response(void *fcsarg, struct bfa_fcxp_s *fcxp,
|
|
||||||
void *cbarg, bfa_status_t req_status,
|
|
||||||
u32 rsp_len, u32 resid_len,
|
|
||||||
struct fchs_s *rsp_fchs)
|
|
||||||
{
|
|
||||||
|
|
||||||
struct bfa_fcs_port_s *port = (struct bfa_fcs_port_s *) cbarg;
|
|
||||||
|
|
||||||
bfa_trc(port->fcs, port->pid);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Sanity Checks
|
|
||||||
*/
|
|
||||||
if (req_status != BFA_STATUS_OK) {
|
|
||||||
bfa_trc(port->fcs, req_status);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -43,7 +43,7 @@ bfa_fcs_cfg_base_port(struct bfa_fcs_s *fcs, struct bfa_port_cfg_s *port_cfg)
|
||||||
struct bfa_fcs_port_s *
|
struct bfa_fcs_port_s *
|
||||||
bfa_fcs_get_base_port(struct bfa_fcs_s *fcs)
|
bfa_fcs_get_base_port(struct bfa_fcs_s *fcs)
|
||||||
{
|
{
|
||||||
return (&fcs->fabric.bport);
|
return &fcs->fabric.bport;
|
||||||
}
|
}
|
||||||
|
|
||||||
wwn_t
|
wwn_t
|
||||||
|
@ -88,11 +88,10 @@ bfa_fcs_port_get_rport(struct bfa_fcs_port_s *port, wwn_t wwn, int index,
|
||||||
}
|
}
|
||||||
|
|
||||||
bfa_trc(fcs, i);
|
bfa_trc(fcs, i);
|
||||||
if (rport) {
|
if (rport)
|
||||||
return rport->pwwn;
|
return rport->pwwn;
|
||||||
} else {
|
else
|
||||||
return (wwn_t) 0;
|
return (wwn_t) 0;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -198,17 +197,17 @@ bfa_fcs_lookup_port(struct bfa_fcs_s *fcs, u16 vf_id, wwn_t lpwwn)
|
||||||
vf = bfa_fcs_vf_lookup(fcs, vf_id);
|
vf = bfa_fcs_vf_lookup(fcs, vf_id);
|
||||||
if (vf == NULL) {
|
if (vf == NULL) {
|
||||||
bfa_trc(fcs, vf_id);
|
bfa_trc(fcs, vf_id);
|
||||||
return (NULL);
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!lpwwn || (vf->bport.port_cfg.pwwn == lpwwn))
|
if (!lpwwn || (vf->bport.port_cfg.pwwn == lpwwn))
|
||||||
return (&vf->bport);
|
return &vf->bport;
|
||||||
|
|
||||||
vport = bfa_fcs_fabric_vport_lookup(vf, lpwwn);
|
vport = bfa_fcs_fabric_vport_lookup(vf, lpwwn);
|
||||||
if (vport)
|
if (vport)
|
||||||
return (&vport->lport);
|
return &vport->lport;
|
||||||
|
|
||||||
return (NULL);
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -932,11 +932,10 @@ bfa_fcs_port_ns_send_rff_id(void *ns_cbarg, struct bfa_fcxp_s *fcxp_alloced)
|
||||||
}
|
}
|
||||||
ns->fcxp = fcxp;
|
ns->fcxp = fcxp;
|
||||||
|
|
||||||
if (BFA_FCS_VPORT_IS_INITIATOR_MODE(ns->port)) {
|
if (BFA_FCS_VPORT_IS_INITIATOR_MODE(ns->port))
|
||||||
fc4_ftrs = FC_GS_FCP_FC4_FEATURE_INITIATOR;
|
fc4_ftrs = FC_GS_FCP_FC4_FEATURE_INITIATOR;
|
||||||
} else if (BFA_FCS_VPORT_IS_TARGET_MODE(ns->port)) {
|
else if (BFA_FCS_VPORT_IS_TARGET_MODE(ns->port))
|
||||||
fc4_ftrs = FC_GS_FCP_FC4_FEATURE_TARGET;
|
fc4_ftrs = FC_GS_FCP_FC4_FEATURE_TARGET;
|
||||||
}
|
|
||||||
|
|
||||||
len = fc_rffid_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
|
len = fc_rffid_build(&fchs, bfa_fcxp_get_reqbuf(fcxp),
|
||||||
bfa_fcs_port_get_fcid(port), 0, FC_TYPE_FCP,
|
bfa_fcs_port_get_fcid(port), 0, FC_TYPE_FCP,
|
||||||
|
|
|
@ -180,5 +180,5 @@ bfa_plog_disable(struct bfa_plog_s *plog)
|
||||||
bfa_boolean_t
|
bfa_boolean_t
|
||||||
bfa_plog_get_setting(struct bfa_plog_s *plog)
|
bfa_plog_get_setting(struct bfa_plog_s *plog)
|
||||||
{
|
{
|
||||||
return((bfa_boolean_t)plog->plog_enabled);
|
return (bfa_boolean_t)plog->plog_enabled;
|
||||||
}
|
}
|
||||||
|
|
|
@ -79,7 +79,7 @@ bfa_fcs_rpf_sm_uninit(struct bfa_fcs_rpf_s *rpf, enum rpf_event event)
|
||||||
bfa_trc(rport->fcs, event);
|
bfa_trc(rport->fcs, event);
|
||||||
|
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case RPFSM_EVENT_RPORT_ONLINE :
|
case RPFSM_EVENT_RPORT_ONLINE:
|
||||||
if (!BFA_FCS_PID_IS_WKA(rport->pid)) {
|
if (!BFA_FCS_PID_IS_WKA(rport->pid)) {
|
||||||
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_rpsc_sending);
|
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_rpsc_sending);
|
||||||
rpf->rpsc_retries = 0;
|
rpf->rpsc_retries = 0;
|
||||||
|
@ -87,7 +87,7 @@ bfa_fcs_rpf_sm_uninit(struct bfa_fcs_rpf_s *rpf, enum rpf_event event)
|
||||||
break;
|
break;
|
||||||
};
|
};
|
||||||
|
|
||||||
case RPFSM_EVENT_RPORT_OFFLINE :
|
case RPFSM_EVENT_RPORT_OFFLINE:
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -107,7 +107,7 @@ bfa_fcs_rpf_sm_rpsc_sending(struct bfa_fcs_rpf_s *rpf, enum rpf_event event)
|
||||||
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_rpsc);
|
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_rpsc);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case RPFSM_EVENT_RPORT_OFFLINE :
|
case RPFSM_EVENT_RPORT_OFFLINE:
|
||||||
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_offline);
|
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_offline);
|
||||||
bfa_fcxp_walloc_cancel(rport->fcs->bfa, &rpf->fcxp_wqe);
|
bfa_fcxp_walloc_cancel(rport->fcs->bfa, &rpf->fcxp_wqe);
|
||||||
rpf->rpsc_retries = 0;
|
rpf->rpsc_retries = 0;
|
||||||
|
@ -130,11 +130,10 @@ bfa_fcs_rpf_sm_rpsc(struct bfa_fcs_rpf_s *rpf, enum rpf_event event)
|
||||||
case RPFSM_EVENT_RPSC_COMP:
|
case RPFSM_EVENT_RPSC_COMP:
|
||||||
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_online);
|
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_online);
|
||||||
/* Update speed info in f/w via BFA */
|
/* Update speed info in f/w via BFA */
|
||||||
if (rpf->rpsc_speed != BFA_PPORT_SPEED_UNKNOWN) {
|
if (rpf->rpsc_speed != BFA_PPORT_SPEED_UNKNOWN)
|
||||||
bfa_rport_speed(rport->bfa_rport, rpf->rpsc_speed);
|
bfa_rport_speed(rport->bfa_rport, rpf->rpsc_speed);
|
||||||
} else if (rpf->assigned_speed != BFA_PPORT_SPEED_UNKNOWN) {
|
else if (rpf->assigned_speed != BFA_PPORT_SPEED_UNKNOWN)
|
||||||
bfa_rport_speed(rport->bfa_rport, rpf->assigned_speed);
|
bfa_rport_speed(rport->bfa_rport, rpf->assigned_speed);
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case RPFSM_EVENT_RPSC_FAIL:
|
case RPFSM_EVENT_RPSC_FAIL:
|
||||||
|
@ -154,7 +153,7 @@ bfa_fcs_rpf_sm_rpsc(struct bfa_fcs_rpf_s *rpf, enum rpf_event event)
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case RPFSM_EVENT_RPORT_OFFLINE :
|
case RPFSM_EVENT_RPORT_OFFLINE:
|
||||||
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_offline);
|
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_offline);
|
||||||
bfa_fcxp_discard(rpf->fcxp);
|
bfa_fcxp_discard(rpf->fcxp);
|
||||||
rpf->rpsc_retries = 0;
|
rpf->rpsc_retries = 0;
|
||||||
|
@ -174,13 +173,13 @@ bfa_fcs_rpf_sm_rpsc_retry(struct bfa_fcs_rpf_s *rpf, enum rpf_event event)
|
||||||
bfa_trc(rport->fcs, event);
|
bfa_trc(rport->fcs, event);
|
||||||
|
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case RPFSM_EVENT_TIMEOUT :
|
case RPFSM_EVENT_TIMEOUT:
|
||||||
/* re-send the RPSC */
|
/* re-send the RPSC */
|
||||||
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_rpsc_sending);
|
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_rpsc_sending);
|
||||||
bfa_fcs_rpf_send_rpsc2(rpf, NULL);
|
bfa_fcs_rpf_send_rpsc2(rpf, NULL);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case RPFSM_EVENT_RPORT_OFFLINE :
|
case RPFSM_EVENT_RPORT_OFFLINE:
|
||||||
bfa_timer_stop(&rpf->timer);
|
bfa_timer_stop(&rpf->timer);
|
||||||
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_offline);
|
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_offline);
|
||||||
rpf->rpsc_retries = 0;
|
rpf->rpsc_retries = 0;
|
||||||
|
@ -201,7 +200,7 @@ bfa_fcs_rpf_sm_online(struct bfa_fcs_rpf_s *rpf, enum rpf_event event)
|
||||||
bfa_trc(rport->fcs, event);
|
bfa_trc(rport->fcs, event);
|
||||||
|
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case RPFSM_EVENT_RPORT_OFFLINE :
|
case RPFSM_EVENT_RPORT_OFFLINE:
|
||||||
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_offline);
|
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_offline);
|
||||||
rpf->rpsc_retries = 0;
|
rpf->rpsc_retries = 0;
|
||||||
break;
|
break;
|
||||||
|
@ -221,12 +220,12 @@ bfa_fcs_rpf_sm_offline(struct bfa_fcs_rpf_s *rpf, enum rpf_event event)
|
||||||
bfa_trc(rport->fcs, event);
|
bfa_trc(rport->fcs, event);
|
||||||
|
|
||||||
switch (event) {
|
switch (event) {
|
||||||
case RPFSM_EVENT_RPORT_ONLINE :
|
case RPFSM_EVENT_RPORT_ONLINE:
|
||||||
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_rpsc_sending);
|
bfa_sm_set_state(rpf, bfa_fcs_rpf_sm_rpsc_sending);
|
||||||
bfa_fcs_rpf_send_rpsc2(rpf, NULL);
|
bfa_fcs_rpf_send_rpsc2(rpf, NULL);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case RPFSM_EVENT_RPORT_OFFLINE :
|
case RPFSM_EVENT_RPORT_OFFLINE:
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -366,10 +365,9 @@ bfa_fcs_rpf_rpsc2_response(void *fcsarg, struct bfa_fcxp_s *fcxp, void *cbarg,
|
||||||
bfa_trc(rport->fcs, ls_rjt->reason_code);
|
bfa_trc(rport->fcs, ls_rjt->reason_code);
|
||||||
bfa_trc(rport->fcs, ls_rjt->reason_code_expl);
|
bfa_trc(rport->fcs, ls_rjt->reason_code_expl);
|
||||||
rport->stats.rpsc_rejects++;
|
rport->stats.rpsc_rejects++;
|
||||||
if (ls_rjt->reason_code == FC_LS_RJT_RSN_CMD_NOT_SUPP) {
|
if (ls_rjt->reason_code == FC_LS_RJT_RSN_CMD_NOT_SUPP)
|
||||||
bfa_sm_send_event(rpf, RPFSM_EVENT_RPSC_FAIL);
|
bfa_sm_send_event(rpf, RPFSM_EVENT_RPSC_FAIL);
|
||||||
} else {
|
else
|
||||||
bfa_sm_send_event(rpf, RPFSM_EVENT_RPSC_ERROR);
|
bfa_sm_send_event(rpf, RPFSM_EVENT_RPSC_ERROR);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -189,7 +189,7 @@ bfa_fcs_vf_lookup(struct bfa_fcs_s *fcs, u16 vf_id)
|
||||||
{
|
{
|
||||||
bfa_trc(fcs, vf_id);
|
bfa_trc(fcs, vf_id);
|
||||||
if (vf_id == FC_VF_ID_NULL)
|
if (vf_id == FC_VF_ID_NULL)
|
||||||
return (&fcs->fabric);
|
return &fcs->fabric;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @todo vf support
|
* @todo vf support
|
||||||
|
|
|
@ -31,13 +31,13 @@
|
||||||
|
|
||||||
BFA_TRC_FILE(FCS, VPORT);
|
BFA_TRC_FILE(FCS, VPORT);
|
||||||
|
|
||||||
#define __vport_fcs(__vp) (__vp)->lport.fcs
|
#define __vport_fcs(__vp) ((__vp)->lport.fcs)
|
||||||
#define __vport_pwwn(__vp) (__vp)->lport.port_cfg.pwwn
|
#define __vport_pwwn(__vp) ((__vp)->lport.port_cfg.pwwn)
|
||||||
#define __vport_nwwn(__vp) (__vp)->lport.port_cfg.nwwn
|
#define __vport_nwwn(__vp) ((__vp)->lport.port_cfg.nwwn)
|
||||||
#define __vport_bfa(__vp) (__vp)->lport.fcs->bfa
|
#define __vport_bfa(__vp) ((__vp)->lport.fcs->bfa)
|
||||||
#define __vport_fcid(__vp) (__vp)->lport.pid
|
#define __vport_fcid(__vp) ((__vp)->lport.pid)
|
||||||
#define __vport_fabric(__vp) (__vp)->lport.fabric
|
#define __vport_fabric(__vp) ((__vp)->lport.fabric)
|
||||||
#define __vport_vfid(__vp) (__vp)->lport.fabric->vf_id
|
#define __vport_vfid(__vp) ((__vp)->lport.fabric->vf_id)
|
||||||
|
|
||||||
#define BFA_FCS_VPORT_MAX_RETRIES 5
|
#define BFA_FCS_VPORT_MAX_RETRIES 5
|
||||||
/*
|
/*
|
||||||
|
@ -641,9 +641,9 @@ bfa_fcs_vport_get_max(struct bfa_fcs_s *fcs)
|
||||||
bfa_get_attr(fcs->bfa, &ioc_attr);
|
bfa_get_attr(fcs->bfa, &ioc_attr);
|
||||||
|
|
||||||
if (ioc_attr.pci_attr.device_id == BFA_PCI_DEVICE_ID_CT)
|
if (ioc_attr.pci_attr.device_id == BFA_PCI_DEVICE_ID_CT)
|
||||||
return (BFA_FCS_MAX_VPORTS_SUPP_CT);
|
return BFA_FCS_MAX_VPORTS_SUPP_CT;
|
||||||
else
|
else
|
||||||
return (BFA_FCS_MAX_VPORTS_SUPP_CB);
|
return BFA_FCS_MAX_VPORTS_SUPP_CB;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -675,7 +675,7 @@ bfa_fcs_vport_create(struct bfa_fcs_vport_s *vport, struct bfa_fcs_s *fcs,
|
||||||
struct bfad_vport_s *vport_drv)
|
struct bfad_vport_s *vport_drv)
|
||||||
{
|
{
|
||||||
if (vport_cfg->pwwn == 0)
|
if (vport_cfg->pwwn == 0)
|
||||||
return (BFA_STATUS_INVALID_WWN);
|
return BFA_STATUS_INVALID_WWN;
|
||||||
|
|
||||||
if (bfa_fcs_port_get_pwwn(&fcs->fabric.bport) == vport_cfg->pwwn)
|
if (bfa_fcs_port_get_pwwn(&fcs->fabric.bport) == vport_cfg->pwwn)
|
||||||
return BFA_STATUS_VPORT_WWN_BP;
|
return BFA_STATUS_VPORT_WWN_BP;
|
||||||
|
|
|
@ -232,7 +232,6 @@ struct bnx2i_conn {
|
||||||
struct iscsi_cls_conn *cls_conn;
|
struct iscsi_cls_conn *cls_conn;
|
||||||
struct bnx2i_hba *hba;
|
struct bnx2i_hba *hba;
|
||||||
struct completion cmd_cleanup_cmpl;
|
struct completion cmd_cleanup_cmpl;
|
||||||
int is_bound;
|
|
||||||
|
|
||||||
u32 iscsi_conn_cid;
|
u32 iscsi_conn_cid;
|
||||||
#define BNX2I_CID_RESERVED 0x5AFF
|
#define BNX2I_CID_RESERVED 0x5AFF
|
||||||
|
|
|
@ -1161,9 +1161,6 @@ static int bnx2i_task_xmit(struct iscsi_task *task)
|
||||||
struct bnx2i_cmd *cmd = task->dd_data;
|
struct bnx2i_cmd *cmd = task->dd_data;
|
||||||
struct iscsi_cmd *hdr = (struct iscsi_cmd *) task->hdr;
|
struct iscsi_cmd *hdr = (struct iscsi_cmd *) task->hdr;
|
||||||
|
|
||||||
if (!bnx2i_conn->is_bound)
|
|
||||||
return -ENOTCONN;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If there is no scsi_cmnd this must be a mgmt task
|
* If there is no scsi_cmnd this must be a mgmt task
|
||||||
*/
|
*/
|
||||||
|
@ -1371,7 +1368,6 @@ static int bnx2i_conn_bind(struct iscsi_cls_session *cls_session,
|
||||||
bnx2i_conn->ep = bnx2i_ep;
|
bnx2i_conn->ep = bnx2i_ep;
|
||||||
bnx2i_conn->iscsi_conn_cid = bnx2i_ep->ep_iscsi_cid;
|
bnx2i_conn->iscsi_conn_cid = bnx2i_ep->ep_iscsi_cid;
|
||||||
bnx2i_conn->fw_cid = bnx2i_ep->ep_cid;
|
bnx2i_conn->fw_cid = bnx2i_ep->ep_cid;
|
||||||
bnx2i_conn->is_bound = 1;
|
|
||||||
|
|
||||||
ret_code = bnx2i_bind_conn_to_iscsi_cid(hba, bnx2i_conn,
|
ret_code = bnx2i_bind_conn_to_iscsi_cid(hba, bnx2i_conn,
|
||||||
bnx2i_ep->ep_iscsi_cid);
|
bnx2i_ep->ep_iscsi_cid);
|
||||||
|
@ -1896,9 +1892,7 @@ static void bnx2i_ep_disconnect(struct iscsi_endpoint *ep)
|
||||||
conn = bnx2i_conn->cls_conn->dd_data;
|
conn = bnx2i_conn->cls_conn->dd_data;
|
||||||
session = conn->session;
|
session = conn->session;
|
||||||
|
|
||||||
spin_lock_bh(&session->lock);
|
iscsi_suspend_queue(conn);
|
||||||
bnx2i_conn->is_bound = 0;
|
|
||||||
spin_unlock_bh(&session->lock);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
hba = bnx2i_ep->hba;
|
hba = bnx2i_ep->hba;
|
||||||
|
@ -2034,7 +2028,7 @@ struct iscsi_transport bnx2i_iscsi_transport = {
|
||||||
ISCSI_USERNAME | ISCSI_PASSWORD |
|
ISCSI_USERNAME | ISCSI_PASSWORD |
|
||||||
ISCSI_USERNAME_IN | ISCSI_PASSWORD_IN |
|
ISCSI_USERNAME_IN | ISCSI_PASSWORD_IN |
|
||||||
ISCSI_FAST_ABORT | ISCSI_ABORT_TMO |
|
ISCSI_FAST_ABORT | ISCSI_ABORT_TMO |
|
||||||
ISCSI_LU_RESET_TMO |
|
ISCSI_LU_RESET_TMO | ISCSI_TGT_RESET_TMO |
|
||||||
ISCSI_PING_TMO | ISCSI_RECV_TMO |
|
ISCSI_PING_TMO | ISCSI_RECV_TMO |
|
||||||
ISCSI_IFACE_NAME | ISCSI_INITIATOR_NAME,
|
ISCSI_IFACE_NAME | ISCSI_INITIATOR_NAME,
|
||||||
.host_param_mask = ISCSI_HOST_HWADDRESS | ISCSI_HOST_NETDEV_NAME,
|
.host_param_mask = ISCSI_HOST_HWADDRESS | ISCSI_HOST_NETDEV_NAME,
|
||||||
|
|
|
@ -141,6 +141,7 @@ static const struct value_name_pair serv_out12_arr[] = {
|
||||||
static const struct value_name_pair serv_in16_arr[] = {
|
static const struct value_name_pair serv_in16_arr[] = {
|
||||||
{0x10, "Read capacity(16)"},
|
{0x10, "Read capacity(16)"},
|
||||||
{0x11, "Read long(16)"},
|
{0x11, "Read long(16)"},
|
||||||
|
{0x12, "Get LBA status"},
|
||||||
};
|
};
|
||||||
#define SERV_IN16_SZ ARRAY_SIZE(serv_in16_arr)
|
#define SERV_IN16_SZ ARRAY_SIZE(serv_in16_arr)
|
||||||
|
|
||||||
|
|
|
@ -937,7 +937,7 @@ static struct iscsi_transport cxgb3i_iscsi_transport = {
|
||||||
ISCSI_USERNAME | ISCSI_PASSWORD |
|
ISCSI_USERNAME | ISCSI_PASSWORD |
|
||||||
ISCSI_USERNAME_IN | ISCSI_PASSWORD_IN |
|
ISCSI_USERNAME_IN | ISCSI_PASSWORD_IN |
|
||||||
ISCSI_FAST_ABORT | ISCSI_ABORT_TMO |
|
ISCSI_FAST_ABORT | ISCSI_ABORT_TMO |
|
||||||
ISCSI_LU_RESET_TMO |
|
ISCSI_LU_RESET_TMO | ISCSI_TGT_RESET_TMO |
|
||||||
ISCSI_PING_TMO | ISCSI_RECV_TMO |
|
ISCSI_PING_TMO | ISCSI_RECV_TMO |
|
||||||
ISCSI_IFACE_NAME | ISCSI_INITIATOR_NAME,
|
ISCSI_IFACE_NAME | ISCSI_INITIATOR_NAME,
|
||||||
.host_param_mask = ISCSI_HOST_HWADDRESS | ISCSI_HOST_IPADDRESS |
|
.host_param_mask = ISCSI_HOST_HWADDRESS | ISCSI_HOST_IPADDRESS |
|
||||||
|
|
|
@ -226,7 +226,7 @@ store_dh_state(struct device *dev, struct device_attribute *attr,
|
||||||
* Activate a device handler
|
* Activate a device handler
|
||||||
*/
|
*/
|
||||||
if (scsi_dh->activate)
|
if (scsi_dh->activate)
|
||||||
err = scsi_dh->activate(sdev);
|
err = scsi_dh->activate(sdev, NULL, NULL);
|
||||||
else
|
else
|
||||||
err = 0;
|
err = 0;
|
||||||
}
|
}
|
||||||
|
@ -304,18 +304,15 @@ static int scsi_dh_notifier(struct notifier_block *nb,
|
||||||
sdev = to_scsi_device(dev);
|
sdev = to_scsi_device(dev);
|
||||||
|
|
||||||
if (action == BUS_NOTIFY_ADD_DEVICE) {
|
if (action == BUS_NOTIFY_ADD_DEVICE) {
|
||||||
devinfo = device_handler_match(NULL, sdev);
|
|
||||||
if (!devinfo)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
err = scsi_dh_handler_attach(sdev, devinfo);
|
|
||||||
if (!err)
|
|
||||||
err = device_create_file(dev, &scsi_dh_state_attr);
|
err = device_create_file(dev, &scsi_dh_state_attr);
|
||||||
|
/* don't care about err */
|
||||||
|
devinfo = device_handler_match(NULL, sdev);
|
||||||
|
if (devinfo)
|
||||||
|
err = scsi_dh_handler_attach(sdev, devinfo);
|
||||||
} else if (action == BUS_NOTIFY_DEL_DEVICE) {
|
} else if (action == BUS_NOTIFY_DEL_DEVICE) {
|
||||||
device_remove_file(dev, &scsi_dh_state_attr);
|
device_remove_file(dev, &scsi_dh_state_attr);
|
||||||
scsi_dh_handler_detach(sdev, NULL);
|
scsi_dh_handler_detach(sdev, NULL);
|
||||||
}
|
}
|
||||||
out:
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -423,10 +420,17 @@ EXPORT_SYMBOL_GPL(scsi_unregister_device_handler);
|
||||||
/*
|
/*
|
||||||
* scsi_dh_activate - activate the path associated with the scsi_device
|
* scsi_dh_activate - activate the path associated with the scsi_device
|
||||||
* corresponding to the given request queue.
|
* corresponding to the given request queue.
|
||||||
|
* Returns immediately without waiting for activation to be completed.
|
||||||
* @q - Request queue that is associated with the scsi_device to be
|
* @q - Request queue that is associated with the scsi_device to be
|
||||||
* activated.
|
* activated.
|
||||||
|
* @fn - Function to be called upon completion of the activation.
|
||||||
|
* Function fn is called with data (below) and the error code.
|
||||||
|
* Function fn may be called from the same calling context. So,
|
||||||
|
* do not hold the lock in the caller which may be needed in fn.
|
||||||
|
* @data - data passed to the function fn upon completion.
|
||||||
|
*
|
||||||
*/
|
*/
|
||||||
int scsi_dh_activate(struct request_queue *q)
|
int scsi_dh_activate(struct request_queue *q, activate_complete fn, void *data)
|
||||||
{
|
{
|
||||||
int err = 0;
|
int err = 0;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
@ -445,7 +449,7 @@ int scsi_dh_activate(struct request_queue *q)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
if (scsi_dh->activate)
|
if (scsi_dh->activate)
|
||||||
err = scsi_dh->activate(sdev);
|
err = scsi_dh->activate(sdev, fn, data);
|
||||||
put_device(&sdev->sdev_gendev);
|
put_device(&sdev->sdev_gendev);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
|
@ -60,11 +60,17 @@ struct alua_dh_data {
|
||||||
int bufflen;
|
int bufflen;
|
||||||
unsigned char sense[SCSI_SENSE_BUFFERSIZE];
|
unsigned char sense[SCSI_SENSE_BUFFERSIZE];
|
||||||
int senselen;
|
int senselen;
|
||||||
|
struct scsi_device *sdev;
|
||||||
|
activate_complete callback_fn;
|
||||||
|
void *callback_data;
|
||||||
};
|
};
|
||||||
|
|
||||||
#define ALUA_POLICY_SWITCH_CURRENT 0
|
#define ALUA_POLICY_SWITCH_CURRENT 0
|
||||||
#define ALUA_POLICY_SWITCH_ALL 1
|
#define ALUA_POLICY_SWITCH_ALL 1
|
||||||
|
|
||||||
|
static char print_alua_state(int);
|
||||||
|
static int alua_check_sense(struct scsi_device *, struct scsi_sense_hdr *);
|
||||||
|
|
||||||
static inline struct alua_dh_data *get_alua_data(struct scsi_device *sdev)
|
static inline struct alua_dh_data *get_alua_data(struct scsi_device *sdev)
|
||||||
{
|
{
|
||||||
struct scsi_dh_data *scsi_dh_data = sdev->scsi_dh_data;
|
struct scsi_dh_data *scsi_dh_data = sdev->scsi_dh_data;
|
||||||
|
@ -230,19 +236,72 @@ done:
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* alua_stpg - Evaluate SET TARGET GROUP STATES
|
||||||
|
* @sdev: the device to be evaluated
|
||||||
|
* @state: the new target group state
|
||||||
|
*
|
||||||
|
* Send a SET TARGET GROUP STATES command to the device.
|
||||||
|
* We only have to test here if we should resubmit the command;
|
||||||
|
* any other error is assumed as a failure.
|
||||||
|
*/
|
||||||
|
static void stpg_endio(struct request *req, int error)
|
||||||
|
{
|
||||||
|
struct alua_dh_data *h = req->end_io_data;
|
||||||
|
struct scsi_sense_hdr sense_hdr;
|
||||||
|
unsigned err = SCSI_DH_IO;
|
||||||
|
|
||||||
|
if (error || host_byte(req->errors) != DID_OK ||
|
||||||
|
msg_byte(req->errors) != COMMAND_COMPLETE)
|
||||||
|
goto done;
|
||||||
|
|
||||||
|
if (err == SCSI_DH_IO && h->senselen > 0) {
|
||||||
|
err = scsi_normalize_sense(h->sense, SCSI_SENSE_BUFFERSIZE,
|
||||||
|
&sense_hdr);
|
||||||
|
if (!err) {
|
||||||
|
err = SCSI_DH_IO;
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
err = alua_check_sense(h->sdev, &sense_hdr);
|
||||||
|
if (err == ADD_TO_MLQUEUE) {
|
||||||
|
err = SCSI_DH_RETRY;
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
sdev_printk(KERN_INFO, h->sdev,
|
||||||
|
"%s: stpg sense code: %02x/%02x/%02x\n",
|
||||||
|
ALUA_DH_NAME, sense_hdr.sense_key,
|
||||||
|
sense_hdr.asc, sense_hdr.ascq);
|
||||||
|
err = SCSI_DH_IO;
|
||||||
|
}
|
||||||
|
if (err == SCSI_DH_OK) {
|
||||||
|
h->state = TPGS_STATE_OPTIMIZED;
|
||||||
|
sdev_printk(KERN_INFO, h->sdev,
|
||||||
|
"%s: port group %02x switched to state %c\n",
|
||||||
|
ALUA_DH_NAME, h->group_id,
|
||||||
|
print_alua_state(h->state));
|
||||||
|
}
|
||||||
|
done:
|
||||||
|
blk_put_request(req);
|
||||||
|
if (h->callback_fn) {
|
||||||
|
h->callback_fn(h->callback_data, err);
|
||||||
|
h->callback_fn = h->callback_data = NULL;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* submit_stpg - Issue a SET TARGET GROUP STATES command
|
* submit_stpg - Issue a SET TARGET GROUP STATES command
|
||||||
* @sdev: sdev the command should be sent to
|
|
||||||
*
|
*
|
||||||
* Currently we're only setting the current target port group state
|
* Currently we're only setting the current target port group state
|
||||||
* to 'active/optimized' and let the array firmware figure out
|
* to 'active/optimized' and let the array firmware figure out
|
||||||
* the states of the remaining groups.
|
* the states of the remaining groups.
|
||||||
*/
|
*/
|
||||||
static unsigned submit_stpg(struct scsi_device *sdev, struct alua_dh_data *h)
|
static unsigned submit_stpg(struct alua_dh_data *h)
|
||||||
{
|
{
|
||||||
struct request *rq;
|
struct request *rq;
|
||||||
int err = SCSI_DH_RES_TEMP_UNAVAIL;
|
int err = SCSI_DH_RES_TEMP_UNAVAIL;
|
||||||
int stpg_len = 8;
|
int stpg_len = 8;
|
||||||
|
struct scsi_device *sdev = h->sdev;
|
||||||
|
|
||||||
/* Prepare the data buffer */
|
/* Prepare the data buffer */
|
||||||
memset(h->buff, 0, stpg_len);
|
memset(h->buff, 0, stpg_len);
|
||||||
|
@ -252,7 +311,7 @@ static unsigned submit_stpg(struct scsi_device *sdev, struct alua_dh_data *h)
|
||||||
|
|
||||||
rq = get_alua_req(sdev, h->buff, stpg_len, WRITE);
|
rq = get_alua_req(sdev, h->buff, stpg_len, WRITE);
|
||||||
if (!rq)
|
if (!rq)
|
||||||
goto done;
|
return SCSI_DH_RES_TEMP_UNAVAIL;
|
||||||
|
|
||||||
/* Prepare the command. */
|
/* Prepare the command. */
|
||||||
rq->cmd[0] = MAINTENANCE_OUT;
|
rq->cmd[0] = MAINTENANCE_OUT;
|
||||||
|
@ -266,17 +325,9 @@ static unsigned submit_stpg(struct scsi_device *sdev, struct alua_dh_data *h)
|
||||||
rq->sense = h->sense;
|
rq->sense = h->sense;
|
||||||
memset(rq->sense, 0, SCSI_SENSE_BUFFERSIZE);
|
memset(rq->sense, 0, SCSI_SENSE_BUFFERSIZE);
|
||||||
rq->sense_len = h->senselen = 0;
|
rq->sense_len = h->senselen = 0;
|
||||||
|
rq->end_io_data = h;
|
||||||
|
|
||||||
err = blk_execute_rq(rq->q, NULL, rq, 1);
|
blk_execute_rq_nowait(rq->q, NULL, rq, 1, stpg_endio);
|
||||||
if (err == -EIO) {
|
|
||||||
sdev_printk(KERN_INFO, sdev,
|
|
||||||
"%s: stpg failed with %x\n",
|
|
||||||
ALUA_DH_NAME, rq->errors);
|
|
||||||
h->senselen = rq->sense_len;
|
|
||||||
err = SCSI_DH_IO;
|
|
||||||
}
|
|
||||||
blk_put_request(rq);
|
|
||||||
done:
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -476,50 +527,6 @@ static int alua_check_sense(struct scsi_device *sdev,
|
||||||
return SCSI_RETURN_NOT_HANDLED;
|
return SCSI_RETURN_NOT_HANDLED;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* alua_stpg - Evaluate SET TARGET GROUP STATES
|
|
||||||
* @sdev: the device to be evaluated
|
|
||||||
* @state: the new target group state
|
|
||||||
*
|
|
||||||
* Send a SET TARGET GROUP STATES command to the device.
|
|
||||||
* We only have to test here if we should resubmit the command;
|
|
||||||
* any other error is assumed as a failure.
|
|
||||||
*/
|
|
||||||
static int alua_stpg(struct scsi_device *sdev, int state,
|
|
||||||
struct alua_dh_data *h)
|
|
||||||
{
|
|
||||||
struct scsi_sense_hdr sense_hdr;
|
|
||||||
unsigned err;
|
|
||||||
int retry = ALUA_FAILOVER_RETRIES;
|
|
||||||
|
|
||||||
retry:
|
|
||||||
err = submit_stpg(sdev, h);
|
|
||||||
if (err == SCSI_DH_IO && h->senselen > 0) {
|
|
||||||
err = scsi_normalize_sense(h->sense, SCSI_SENSE_BUFFERSIZE,
|
|
||||||
&sense_hdr);
|
|
||||||
if (!err)
|
|
||||||
return SCSI_DH_IO;
|
|
||||||
err = alua_check_sense(sdev, &sense_hdr);
|
|
||||||
if (retry > 0 && err == ADD_TO_MLQUEUE) {
|
|
||||||
retry--;
|
|
||||||
goto retry;
|
|
||||||
}
|
|
||||||
sdev_printk(KERN_INFO, sdev,
|
|
||||||
"%s: stpg sense code: %02x/%02x/%02x\n",
|
|
||||||
ALUA_DH_NAME, sense_hdr.sense_key,
|
|
||||||
sense_hdr.asc, sense_hdr.ascq);
|
|
||||||
err = SCSI_DH_IO;
|
|
||||||
}
|
|
||||||
if (err == SCSI_DH_OK) {
|
|
||||||
h->state = state;
|
|
||||||
sdev_printk(KERN_INFO, sdev,
|
|
||||||
"%s: port group %02x switched to state %c\n",
|
|
||||||
ALUA_DH_NAME, h->group_id,
|
|
||||||
print_alua_state(h->state) );
|
|
||||||
}
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* alua_rtpg - Evaluate REPORT TARGET GROUP STATES
|
* alua_rtpg - Evaluate REPORT TARGET GROUP STATES
|
||||||
* @sdev: the device to be evaluated.
|
* @sdev: the device to be evaluated.
|
||||||
|
@ -652,7 +659,8 @@ out:
|
||||||
* based on a certain policy. But until we actually encounter them it
|
* based on a certain policy. But until we actually encounter them it
|
||||||
* should be okay.
|
* should be okay.
|
||||||
*/
|
*/
|
||||||
static int alua_activate(struct scsi_device *sdev)
|
static int alua_activate(struct scsi_device *sdev,
|
||||||
|
activate_complete fn, void *data)
|
||||||
{
|
{
|
||||||
struct alua_dh_data *h = get_alua_data(sdev);
|
struct alua_dh_data *h = get_alua_data(sdev);
|
||||||
int err = SCSI_DH_OK;
|
int err = SCSI_DH_OK;
|
||||||
|
@ -663,11 +671,19 @@ static int alua_activate(struct scsi_device *sdev)
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (h->tpgs & TPGS_MODE_EXPLICIT && h->state != TPGS_STATE_OPTIMIZED)
|
if (h->tpgs & TPGS_MODE_EXPLICIT && h->state != TPGS_STATE_OPTIMIZED) {
|
||||||
err = alua_stpg(sdev, TPGS_STATE_OPTIMIZED, h);
|
h->callback_fn = fn;
|
||||||
|
h->callback_data = data;
|
||||||
|
err = submit_stpg(h);
|
||||||
|
if (err == SCSI_DH_OK)
|
||||||
|
return 0;
|
||||||
|
h->callback_fn = h->callback_data = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
out:
|
out:
|
||||||
return err;
|
if (fn)
|
||||||
|
fn(data, err);
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -745,6 +761,7 @@ static int alua_bus_attach(struct scsi_device *sdev)
|
||||||
h->rel_port = -1;
|
h->rel_port = -1;
|
||||||
h->buff = h->inq;
|
h->buff = h->inq;
|
||||||
h->bufflen = ALUA_INQUIRY_SIZE;
|
h->bufflen = ALUA_INQUIRY_SIZE;
|
||||||
|
h->sdev = sdev;
|
||||||
|
|
||||||
err = alua_initialize(sdev, h);
|
err = alua_initialize(sdev, h);
|
||||||
if (err != SCSI_DH_OK)
|
if (err != SCSI_DH_OK)
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue