Merge master.kernel.org:/pub/scm/linux/kernel/git/jejb/scsi-for-linus-2.6
This commit is contained in:
commit
fecb4a0c87
|
@ -0,0 +1,45 @@
|
||||||
|
Copyright (c) 2003-2005 QLogic Corporation
|
||||||
|
QLogic Linux Fibre Channel HBA Driver
|
||||||
|
|
||||||
|
This program includes a device driver for Linux 2.6 that may be
|
||||||
|
distributed with QLogic hardware specific firmware binary file.
|
||||||
|
You may modify and redistribute the device driver code under the
|
||||||
|
GNU General Public License as published by the Free Software
|
||||||
|
Foundation (version 2 or a later version).
|
||||||
|
|
||||||
|
You may redistribute the hardware specific firmware binary file
|
||||||
|
under the following terms:
|
||||||
|
|
||||||
|
1. Redistribution of source code (only if applicable),
|
||||||
|
must retain the above copyright notice, this list of
|
||||||
|
conditions and the following disclaimer.
|
||||||
|
|
||||||
|
2. Redistribution in binary form must reproduce the above
|
||||||
|
copyright notice, this list of conditions and the
|
||||||
|
following disclaimer in the documentation and/or other
|
||||||
|
materials provided with the distribution.
|
||||||
|
|
||||||
|
3. The name of QLogic Corporation may not be used to
|
||||||
|
endorse or promote products derived from this software
|
||||||
|
without specific prior written permission
|
||||||
|
|
||||||
|
REGARDLESS OF WHAT LICENSING MECHANISM IS USED OR APPLICABLE,
|
||||||
|
THIS PROGRAM IS PROVIDED BY QLOGIC CORPORATION "AS IS'' AND ANY
|
||||||
|
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR
|
||||||
|
BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
||||||
|
EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
|
||||||
|
TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||||
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 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 OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||||||
|
POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
USER ACKNOWLEDGES AND AGREES THAT USE OF THIS PROGRAM WILL NOT
|
||||||
|
CREATE OR GIVE GROUNDS FOR A LICENSE BY IMPLICATION, ESTOPPEL, OR
|
||||||
|
OTHERWISE IN ANY INTELLECTUAL PROPERTY RIGHTS (PATENT, COPYRIGHT,
|
||||||
|
TRADE SECRET, MASK WORK, OR OTHER PROPRIETARY RIGHT) EMBODIED IN
|
||||||
|
ANY OTHER QLOGIC HARDWARE OR SOFTWARE EITHER SOLELY OR IN
|
||||||
|
COMBINATION WITH THIS PROGRAM.
|
|
@ -77,8 +77,8 @@
|
||||||
#define COPYRIGHT "Copyright (c) 1999-2005 " MODULEAUTHOR
|
#define COPYRIGHT "Copyright (c) 1999-2005 " MODULEAUTHOR
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define MPT_LINUX_VERSION_COMMON "3.03.03"
|
#define MPT_LINUX_VERSION_COMMON "3.03.04"
|
||||||
#define MPT_LINUX_PACKAGE_NAME "@(#)mptlinux-3.03.03"
|
#define MPT_LINUX_PACKAGE_NAME "@(#)mptlinux-3.03.04"
|
||||||
#define WHAT_MAGIC_STRING "@" "(" "#" ")"
|
#define WHAT_MAGIC_STRING "@" "(" "#" ")"
|
||||||
|
|
||||||
#define show_mptmod_ver(s,ver) \
|
#define show_mptmod_ver(s,ver) \
|
||||||
|
@ -421,6 +421,17 @@ typedef struct _MPT_IOCTL {
|
||||||
struct semaphore sem_ioc;
|
struct semaphore sem_ioc;
|
||||||
} MPT_IOCTL;
|
} MPT_IOCTL;
|
||||||
|
|
||||||
|
#define MPT_SAS_MGMT_STATUS_RF_VALID 0x02 /* The Reply Frame is VALID */
|
||||||
|
#define MPT_SAS_MGMT_STATUS_COMMAND_GOOD 0x10 /* Command Status GOOD */
|
||||||
|
#define MPT_SAS_MGMT_STATUS_TM_FAILED 0x40 /* User TM request failed */
|
||||||
|
|
||||||
|
typedef struct _MPT_SAS_MGMT {
|
||||||
|
struct semaphore mutex;
|
||||||
|
struct completion done;
|
||||||
|
u8 reply[MPT_DEFAULT_FRAME_SIZE]; /* reply frame data */
|
||||||
|
u8 status; /* current command status */
|
||||||
|
}MPT_SAS_MGMT;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Event Structure and define
|
* Event Structure and define
|
||||||
*/
|
*/
|
||||||
|
@ -604,6 +615,7 @@ typedef struct _MPT_ADAPTER
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
struct net_device *netdev;
|
struct net_device *netdev;
|
||||||
struct list_head sas_topology;
|
struct list_head sas_topology;
|
||||||
|
MPT_SAS_MGMT sas_mgmt;
|
||||||
} MPT_ADAPTER;
|
} MPT_ADAPTER;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
@ -83,6 +83,7 @@ MODULE_PARM_DESC(mpt_pt_clear,
|
||||||
static int mptsasDoneCtx = -1;
|
static int mptsasDoneCtx = -1;
|
||||||
static int mptsasTaskCtx = -1;
|
static int mptsasTaskCtx = -1;
|
||||||
static int mptsasInternalCtx = -1; /* Used only for internal commands */
|
static int mptsasInternalCtx = -1; /* Used only for internal commands */
|
||||||
|
static int mptsasMgmtCtx = -1;
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -123,6 +124,104 @@ struct mptsas_portinfo {
|
||||||
struct mptsas_phyinfo *phy_info;
|
struct mptsas_phyinfo *phy_info;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
#ifdef SASDEBUG
|
||||||
|
static void mptsas_print_phy_data(MPI_SAS_IO_UNIT0_PHY_DATA *phy_data)
|
||||||
|
{
|
||||||
|
printk("---- IO UNIT PAGE 0 ------------\n");
|
||||||
|
printk("Handle=0x%X\n",
|
||||||
|
le16_to_cpu(phy_data->AttachedDeviceHandle));
|
||||||
|
printk("Controller Handle=0x%X\n",
|
||||||
|
le16_to_cpu(phy_data->ControllerDevHandle));
|
||||||
|
printk("Port=0x%X\n", phy_data->Port);
|
||||||
|
printk("Port Flags=0x%X\n", phy_data->PortFlags);
|
||||||
|
printk("PHY Flags=0x%X\n", phy_data->PhyFlags);
|
||||||
|
printk("Negotiated Link Rate=0x%X\n", phy_data->NegotiatedLinkRate);
|
||||||
|
printk("Controller PHY Device Info=0x%X\n",
|
||||||
|
le32_to_cpu(phy_data->ControllerPhyDeviceInfo));
|
||||||
|
printk("DiscoveryStatus=0x%X\n",
|
||||||
|
le32_to_cpu(phy_data->DiscoveryStatus));
|
||||||
|
printk("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mptsas_print_phy_pg0(SasPhyPage0_t *pg0)
|
||||||
|
{
|
||||||
|
__le64 sas_address;
|
||||||
|
|
||||||
|
memcpy(&sas_address, &pg0->SASAddress, sizeof(__le64));
|
||||||
|
|
||||||
|
printk("---- SAS PHY PAGE 0 ------------\n");
|
||||||
|
printk("Attached Device Handle=0x%X\n",
|
||||||
|
le16_to_cpu(pg0->AttachedDevHandle));
|
||||||
|
printk("SAS Address=0x%llX\n",
|
||||||
|
(unsigned long long)le64_to_cpu(sas_address));
|
||||||
|
printk("Attached PHY Identifier=0x%X\n", pg0->AttachedPhyIdentifier);
|
||||||
|
printk("Attached Device Info=0x%X\n",
|
||||||
|
le32_to_cpu(pg0->AttachedDeviceInfo));
|
||||||
|
printk("Programmed Link Rate=0x%X\n", pg0->ProgrammedLinkRate);
|
||||||
|
printk("Change Count=0x%X\n", pg0->ChangeCount);
|
||||||
|
printk("PHY Info=0x%X\n", le32_to_cpu(pg0->PhyInfo));
|
||||||
|
printk("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mptsas_print_phy_pg1(SasPhyPage1_t *pg1)
|
||||||
|
{
|
||||||
|
printk("---- SAS PHY PAGE 1 ------------\n");
|
||||||
|
printk("Invalid Dword Count=0x%x\n", pg1->InvalidDwordCount);
|
||||||
|
printk("Running Disparity Error Count=0x%x\n",
|
||||||
|
pg1->RunningDisparityErrorCount);
|
||||||
|
printk("Loss Dword Synch Count=0x%x\n", pg1->LossDwordSynchCount);
|
||||||
|
printk("PHY Reset Problem Count=0x%x\n", pg1->PhyResetProblemCount);
|
||||||
|
printk("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mptsas_print_device_pg0(SasDevicePage0_t *pg0)
|
||||||
|
{
|
||||||
|
__le64 sas_address;
|
||||||
|
|
||||||
|
memcpy(&sas_address, &pg0->SASAddress, sizeof(__le64));
|
||||||
|
|
||||||
|
printk("---- SAS DEVICE PAGE 0 ---------\n");
|
||||||
|
printk("Handle=0x%X\n" ,le16_to_cpu(pg0->DevHandle));
|
||||||
|
printk("Enclosure Handle=0x%X\n", le16_to_cpu(pg0->EnclosureHandle));
|
||||||
|
printk("Slot=0x%X\n", le16_to_cpu(pg0->Slot));
|
||||||
|
printk("SAS Address=0x%llX\n", le64_to_cpu(sas_address));
|
||||||
|
printk("Target ID=0x%X\n", pg0->TargetID);
|
||||||
|
printk("Bus=0x%X\n", pg0->Bus);
|
||||||
|
/* The PhyNum field specifies the PHY number of the parent
|
||||||
|
* device this device is linked to
|
||||||
|
*/
|
||||||
|
printk("Parent Phy Num=0x%X\n", pg0->PhyNum);
|
||||||
|
printk("Access Status=0x%X\n", le16_to_cpu(pg0->AccessStatus));
|
||||||
|
printk("Device Info=0x%X\n", le32_to_cpu(pg0->DeviceInfo));
|
||||||
|
printk("Flags=0x%X\n", le16_to_cpu(pg0->Flags));
|
||||||
|
printk("Physical Port=0x%X\n", pg0->PhysicalPort);
|
||||||
|
printk("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static void mptsas_print_expander_pg1(SasExpanderPage1_t *pg1)
|
||||||
|
{
|
||||||
|
printk("---- SAS EXPANDER PAGE 1 ------------\n");
|
||||||
|
|
||||||
|
printk("Physical Port=0x%X\n", pg1->PhysicalPort);
|
||||||
|
printk("PHY Identifier=0x%X\n", pg1->PhyIdentifier);
|
||||||
|
printk("Negotiated Link Rate=0x%X\n", pg1->NegotiatedLinkRate);
|
||||||
|
printk("Programmed Link Rate=0x%X\n", pg1->ProgrammedLinkRate);
|
||||||
|
printk("Hardware Link Rate=0x%X\n", pg1->HwLinkRate);
|
||||||
|
printk("Owner Device Handle=0x%X\n",
|
||||||
|
le16_to_cpu(pg1->OwnerDevHandle));
|
||||||
|
printk("Attached Device Handle=0x%X\n",
|
||||||
|
le16_to_cpu(pg1->AttachedDevHandle));
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
#define mptsas_print_phy_data(phy_data) do { } while (0)
|
||||||
|
#define mptsas_print_phy_pg0(pg0) do { } while (0)
|
||||||
|
#define mptsas_print_phy_pg1(pg1) do { } while (0)
|
||||||
|
#define mptsas_print_device_pg0(pg0) do { } while (0)
|
||||||
|
#define mptsas_print_expander_pg1(pg1) do { } while (0)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This is pretty ugly. We will be able to seriously clean it up
|
* This is pretty ugly. We will be able to seriously clean it up
|
||||||
* once the DV code in mptscsih goes away and we can properly
|
* once the DV code in mptscsih goes away and we can properly
|
||||||
|
@ -200,92 +299,160 @@ static struct scsi_host_template mptsas_driver_template = {
|
||||||
.use_clustering = ENABLE_CLUSTERING,
|
.use_clustering = ENABLE_CLUSTERING,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static inline MPT_ADAPTER *phy_to_ioc(struct sas_phy *phy)
|
||||||
|
{
|
||||||
|
struct Scsi_Host *shost = dev_to_shost(phy->dev.parent);
|
||||||
|
return ((MPT_SCSI_HOST *)shost->hostdata)->ioc;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mptsas_get_linkerrors(struct sas_phy *phy)
|
||||||
|
{
|
||||||
|
MPT_ADAPTER *ioc = phy_to_ioc(phy);
|
||||||
|
ConfigExtendedPageHeader_t hdr;
|
||||||
|
CONFIGPARMS cfg;
|
||||||
|
SasPhyPage1_t *buffer;
|
||||||
|
dma_addr_t dma_handle;
|
||||||
|
int error;
|
||||||
|
|
||||||
|
hdr.PageVersion = MPI_SASPHY1_PAGEVERSION;
|
||||||
|
hdr.ExtPageLength = 0;
|
||||||
|
hdr.PageNumber = 1 /* page number 1*/;
|
||||||
|
hdr.Reserved1 = 0;
|
||||||
|
hdr.Reserved2 = 0;
|
||||||
|
hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
|
||||||
|
hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_SAS_PHY;
|
||||||
|
|
||||||
|
cfg.cfghdr.ehdr = &hdr;
|
||||||
|
cfg.physAddr = -1;
|
||||||
|
cfg.pageAddr = phy->identify.phy_identifier;
|
||||||
|
cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
|
||||||
|
cfg.dir = 0; /* read */
|
||||||
|
cfg.timeout = 10;
|
||||||
|
|
||||||
|
error = mpt_config(ioc, &cfg);
|
||||||
|
if (error)
|
||||||
|
return error;
|
||||||
|
if (!hdr.ExtPageLength)
|
||||||
|
return -ENXIO;
|
||||||
|
|
||||||
|
buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
|
||||||
|
&dma_handle);
|
||||||
|
if (!buffer)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
cfg.physAddr = dma_handle;
|
||||||
|
cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
|
||||||
|
|
||||||
|
error = mpt_config(ioc, &cfg);
|
||||||
|
if (error)
|
||||||
|
goto out_free_consistent;
|
||||||
|
|
||||||
|
mptsas_print_phy_pg1(buffer);
|
||||||
|
|
||||||
|
phy->invalid_dword_count = le32_to_cpu(buffer->InvalidDwordCount);
|
||||||
|
phy->running_disparity_error_count =
|
||||||
|
le32_to_cpu(buffer->RunningDisparityErrorCount);
|
||||||
|
phy->loss_of_dword_sync_count =
|
||||||
|
le32_to_cpu(buffer->LossDwordSynchCount);
|
||||||
|
phy->phy_reset_problem_count =
|
||||||
|
le32_to_cpu(buffer->PhyResetProblemCount);
|
||||||
|
|
||||||
|
out_free_consistent:
|
||||||
|
pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
|
||||||
|
buffer, dma_handle);
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mptsas_mgmt_done(MPT_ADAPTER *ioc, MPT_FRAME_HDR *req,
|
||||||
|
MPT_FRAME_HDR *reply)
|
||||||
|
{
|
||||||
|
ioc->sas_mgmt.status |= MPT_SAS_MGMT_STATUS_COMMAND_GOOD;
|
||||||
|
if (reply != NULL) {
|
||||||
|
ioc->sas_mgmt.status |= MPT_SAS_MGMT_STATUS_RF_VALID;
|
||||||
|
memcpy(ioc->sas_mgmt.reply, reply,
|
||||||
|
min(ioc->reply_sz, 4 * reply->u.reply.MsgLength));
|
||||||
|
}
|
||||||
|
complete(&ioc->sas_mgmt.done);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int mptsas_phy_reset(struct sas_phy *phy, int hard_reset)
|
||||||
|
{
|
||||||
|
MPT_ADAPTER *ioc = phy_to_ioc(phy);
|
||||||
|
SasIoUnitControlRequest_t *req;
|
||||||
|
SasIoUnitControlReply_t *reply;
|
||||||
|
MPT_FRAME_HDR *mf;
|
||||||
|
MPIHeader_t *hdr;
|
||||||
|
unsigned long timeleft;
|
||||||
|
int error = -ERESTARTSYS;
|
||||||
|
|
||||||
|
/* not implemented for expanders */
|
||||||
|
if (phy->identify.target_port_protocols & SAS_PROTOCOL_SMP)
|
||||||
|
return -ENXIO;
|
||||||
|
|
||||||
|
if (down_interruptible(&ioc->sas_mgmt.mutex))
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
mf = mpt_get_msg_frame(mptsasMgmtCtx, ioc);
|
||||||
|
if (!mf) {
|
||||||
|
error = -ENOMEM;
|
||||||
|
goto out_unlock;
|
||||||
|
}
|
||||||
|
|
||||||
|
hdr = (MPIHeader_t *) mf;
|
||||||
|
req = (SasIoUnitControlRequest_t *)mf;
|
||||||
|
memset(req, 0, sizeof(SasIoUnitControlRequest_t));
|
||||||
|
req->Function = MPI_FUNCTION_SAS_IO_UNIT_CONTROL;
|
||||||
|
req->MsgContext = hdr->MsgContext;
|
||||||
|
req->Operation = hard_reset ?
|
||||||
|
MPI_SAS_OP_PHY_HARD_RESET : MPI_SAS_OP_PHY_LINK_RESET;
|
||||||
|
req->PhyNum = phy->identify.phy_identifier;
|
||||||
|
|
||||||
|
mpt_put_msg_frame(mptsasMgmtCtx, ioc, mf);
|
||||||
|
|
||||||
|
timeleft = wait_for_completion_timeout(&ioc->sas_mgmt.done,
|
||||||
|
10 * HZ);
|
||||||
|
if (!timeleft) {
|
||||||
|
/* On timeout reset the board */
|
||||||
|
mpt_free_msg_frame(ioc, mf);
|
||||||
|
mpt_HardResetHandler(ioc, CAN_SLEEP);
|
||||||
|
error = -ETIMEDOUT;
|
||||||
|
goto out_unlock;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* a reply frame is expected */
|
||||||
|
if ((ioc->sas_mgmt.status &
|
||||||
|
MPT_IOCTL_STATUS_RF_VALID) == 0) {
|
||||||
|
error = -ENXIO;
|
||||||
|
goto out_unlock;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* process the completed Reply Message Frame */
|
||||||
|
reply = (SasIoUnitControlReply_t *)ioc->sas_mgmt.reply;
|
||||||
|
if (reply->IOCStatus != MPI_IOCSTATUS_SUCCESS) {
|
||||||
|
printk("%s: IOCStatus=0x%X IOCLogInfo=0x%X\n",
|
||||||
|
__FUNCTION__,
|
||||||
|
reply->IOCStatus,
|
||||||
|
reply->IOCLogInfo);
|
||||||
|
error = -ENXIO;
|
||||||
|
goto out_unlock;
|
||||||
|
}
|
||||||
|
|
||||||
|
error = 0;
|
||||||
|
|
||||||
|
out_unlock:
|
||||||
|
up(&ioc->sas_mgmt.mutex);
|
||||||
|
out:
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
static struct sas_function_template mptsas_transport_functions = {
|
static struct sas_function_template mptsas_transport_functions = {
|
||||||
|
.get_linkerrors = mptsas_get_linkerrors,
|
||||||
|
.phy_reset = mptsas_phy_reset,
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct scsi_transport_template *mptsas_transport_template;
|
static struct scsi_transport_template *mptsas_transport_template;
|
||||||
|
|
||||||
#ifdef SASDEBUG
|
|
||||||
static void mptsas_print_phy_data(MPI_SAS_IO_UNIT0_PHY_DATA *phy_data)
|
|
||||||
{
|
|
||||||
printk("---- IO UNIT PAGE 0 ------------\n");
|
|
||||||
printk("Handle=0x%X\n",
|
|
||||||
le16_to_cpu(phy_data->AttachedDeviceHandle));
|
|
||||||
printk("Controller Handle=0x%X\n",
|
|
||||||
le16_to_cpu(phy_data->ControllerDevHandle));
|
|
||||||
printk("Port=0x%X\n", phy_data->Port);
|
|
||||||
printk("Port Flags=0x%X\n", phy_data->PortFlags);
|
|
||||||
printk("PHY Flags=0x%X\n", phy_data->PhyFlags);
|
|
||||||
printk("Negotiated Link Rate=0x%X\n", phy_data->NegotiatedLinkRate);
|
|
||||||
printk("Controller PHY Device Info=0x%X\n",
|
|
||||||
le32_to_cpu(phy_data->ControllerPhyDeviceInfo));
|
|
||||||
printk("DiscoveryStatus=0x%X\n",
|
|
||||||
le32_to_cpu(phy_data->DiscoveryStatus));
|
|
||||||
printk("\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
static void mptsas_print_phy_pg0(SasPhyPage0_t *pg0)
|
|
||||||
{
|
|
||||||
__le64 sas_address;
|
|
||||||
|
|
||||||
memcpy(&sas_address, &pg0->SASAddress, sizeof(__le64));
|
|
||||||
|
|
||||||
printk("---- SAS PHY PAGE 0 ------------\n");
|
|
||||||
printk("Attached Device Handle=0x%X\n",
|
|
||||||
le16_to_cpu(pg0->AttachedDevHandle));
|
|
||||||
printk("SAS Address=0x%llX\n",
|
|
||||||
(unsigned long long)le64_to_cpu(sas_address));
|
|
||||||
printk("Attached PHY Identifier=0x%X\n", pg0->AttachedPhyIdentifier);
|
|
||||||
printk("Attached Device Info=0x%X\n",
|
|
||||||
le32_to_cpu(pg0->AttachedDeviceInfo));
|
|
||||||
printk("Programmed Link Rate=0x%X\n", pg0->ProgrammedLinkRate);
|
|
||||||
printk("Change Count=0x%X\n", pg0->ChangeCount);
|
|
||||||
printk("PHY Info=0x%X\n", le32_to_cpu(pg0->PhyInfo));
|
|
||||||
printk("\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
static void mptsas_print_device_pg0(SasDevicePage0_t *pg0)
|
|
||||||
{
|
|
||||||
__le64 sas_address;
|
|
||||||
|
|
||||||
memcpy(&sas_address, &pg0->SASAddress, sizeof(__le64));
|
|
||||||
|
|
||||||
printk("---- SAS DEVICE PAGE 0 ---------\n");
|
|
||||||
printk("Handle=0x%X\n" ,le16_to_cpu(pg0->DevHandle));
|
|
||||||
printk("Enclosure Handle=0x%X\n", le16_to_cpu(pg0->EnclosureHandle));
|
|
||||||
printk("Slot=0x%X\n", le16_to_cpu(pg0->Slot));
|
|
||||||
printk("SAS Address=0x%llX\n", le64_to_cpu(sas_address));
|
|
||||||
printk("Target ID=0x%X\n", pg0->TargetID);
|
|
||||||
printk("Bus=0x%X\n", pg0->Bus);
|
|
||||||
printk("Parent Phy Num=0x%X\n", pg0->PhyNum);
|
|
||||||
printk("Access Status=0x%X\n", le16_to_cpu(pg0->AccessStatus));
|
|
||||||
printk("Device Info=0x%X\n", le32_to_cpu(pg0->DeviceInfo));
|
|
||||||
printk("Flags=0x%X\n", le16_to_cpu(pg0->Flags));
|
|
||||||
printk("Physical Port=0x%X\n", pg0->PhysicalPort);
|
|
||||||
printk("\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
static void mptsas_print_expander_pg1(SasExpanderPage1_t *pg1)
|
|
||||||
{
|
|
||||||
printk("---- SAS EXPANDER PAGE 1 ------------\n");
|
|
||||||
|
|
||||||
printk("Physical Port=0x%X\n", pg1->PhysicalPort);
|
|
||||||
printk("PHY Identifier=0x%X\n", pg1->PhyIdentifier);
|
|
||||||
printk("Negotiated Link Rate=0x%X\n", pg1->NegotiatedLinkRate);
|
|
||||||
printk("Programmed Link Rate=0x%X\n", pg1->ProgrammedLinkRate);
|
|
||||||
printk("Hardware Link Rate=0x%X\n", pg1->HwLinkRate);
|
|
||||||
printk("Owner Device Handle=0x%X\n",
|
|
||||||
le16_to_cpu(pg1->OwnerDevHandle));
|
|
||||||
printk("Attached Device Handle=0x%X\n",
|
|
||||||
le16_to_cpu(pg1->AttachedDevHandle));
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
#define mptsas_print_phy_data(phy_data) do { } while (0)
|
|
||||||
#define mptsas_print_phy_pg0(pg0) do { } while (0)
|
|
||||||
#define mptsas_print_device_pg0(pg0) do { } while (0)
|
|
||||||
#define mptsas_print_expander_pg1(pg1) do { } while (0)
|
|
||||||
#endif
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
mptsas_sas_io_unit_pg0(MPT_ADAPTER *ioc, struct mptsas_portinfo *port_info)
|
mptsas_sas_io_unit_pg0(MPT_ADAPTER *ioc, struct mptsas_portinfo *port_info)
|
||||||
{
|
{
|
||||||
|
@ -680,7 +847,7 @@ mptsas_parse_device_info(struct sas_identify *identify,
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mptsas_probe_one_phy(struct device *dev,
|
static int mptsas_probe_one_phy(struct device *dev,
|
||||||
struct mptsas_phyinfo *phy_info, int index)
|
struct mptsas_phyinfo *phy_info, int index, int local)
|
||||||
{
|
{
|
||||||
struct sas_phy *port;
|
struct sas_phy *port;
|
||||||
int error;
|
int error;
|
||||||
|
@ -773,6 +940,9 @@ static int mptsas_probe_one_phy(struct device *dev,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (local)
|
||||||
|
port->local_attached = 1;
|
||||||
|
|
||||||
error = sas_phy_add(port);
|
error = sas_phy_add(port);
|
||||||
if (error) {
|
if (error) {
|
||||||
sas_phy_free(port);
|
sas_phy_free(port);
|
||||||
|
@ -838,7 +1008,7 @@ mptsas_probe_hba_phys(MPT_ADAPTER *ioc, int *index)
|
||||||
}
|
}
|
||||||
|
|
||||||
mptsas_probe_one_phy(&ioc->sh->shost_gendev,
|
mptsas_probe_one_phy(&ioc->sh->shost_gendev,
|
||||||
&port_info->phy_info[i], *index);
|
&port_info->phy_info[i], *index, 1);
|
||||||
(*index)++;
|
(*index)++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -909,7 +1079,8 @@ mptsas_probe_expander_phys(MPT_ADAPTER *ioc, u32 *handle, int *index)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
mptsas_probe_one_phy(parent, &port_info->phy_info[i], *index);
|
mptsas_probe_one_phy(parent, &port_info->phy_info[i],
|
||||||
|
*index, 0);
|
||||||
(*index)++;
|
(*index)++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1021,6 +1192,8 @@ mptsas_probe(struct pci_dev *pdev, const struct pci_device_id *id)
|
||||||
sh->unique_id = ioc->id;
|
sh->unique_id = ioc->id;
|
||||||
|
|
||||||
INIT_LIST_HEAD(&ioc->sas_topology);
|
INIT_LIST_HEAD(&ioc->sas_topology);
|
||||||
|
init_MUTEX(&ioc->sas_mgmt.mutex);
|
||||||
|
init_completion(&ioc->sas_mgmt.done);
|
||||||
|
|
||||||
/* Verify that we won't exceed the maximum
|
/* Verify that we won't exceed the maximum
|
||||||
* number of chain buffers
|
* number of chain buffers
|
||||||
|
@ -1207,6 +1380,7 @@ mptsas_init(void)
|
||||||
mptsasTaskCtx = mpt_register(mptscsih_taskmgmt_complete, MPTSAS_DRIVER);
|
mptsasTaskCtx = mpt_register(mptscsih_taskmgmt_complete, MPTSAS_DRIVER);
|
||||||
mptsasInternalCtx =
|
mptsasInternalCtx =
|
||||||
mpt_register(mptscsih_scandv_complete, MPTSAS_DRIVER);
|
mpt_register(mptscsih_scandv_complete, MPTSAS_DRIVER);
|
||||||
|
mptsasMgmtCtx = mpt_register(mptsas_mgmt_done, MPTSAS_DRIVER);
|
||||||
|
|
||||||
if (mpt_event_register(mptsasDoneCtx, mptscsih_event_process) == 0) {
|
if (mpt_event_register(mptsasDoneCtx, mptscsih_event_process) == 0) {
|
||||||
devtprintk((KERN_INFO MYNAM
|
devtprintk((KERN_INFO MYNAM
|
||||||
|
@ -1230,6 +1404,7 @@ mptsas_exit(void)
|
||||||
mpt_reset_deregister(mptsasDoneCtx);
|
mpt_reset_deregister(mptsasDoneCtx);
|
||||||
mpt_event_deregister(mptsasDoneCtx);
|
mpt_event_deregister(mptsasDoneCtx);
|
||||||
|
|
||||||
|
mpt_deregister(mptsasMgmtCtx);
|
||||||
mpt_deregister(mptsasInternalCtx);
|
mpt_deregister(mptsasInternalCtx);
|
||||||
mpt_deregister(mptsasTaskCtx);
|
mpt_deregister(mptsasTaskCtx);
|
||||||
mpt_deregister(mptsasDoneCtx);
|
mpt_deregister(mptsasDoneCtx);
|
||||||
|
|
|
@ -1732,7 +1732,9 @@ static int twa_scsi_eh_reset(struct scsi_cmnd *SCpnt)
|
||||||
|
|
||||||
tw_dev->num_resets++;
|
tw_dev->num_resets++;
|
||||||
|
|
||||||
printk(KERN_WARNING "3w-9xxx: scsi%d: WARNING: (0x%02X:0x%04X): Unit #%d: Command (0x%x) timed out, resetting card.\n", tw_dev->host->host_no, TW_DRIVER, 0x2c, SCpnt->device->id, SCpnt->cmnd[0]);
|
sdev_printk(KERN_WARNING, SCpnt->device,
|
||||||
|
"WARNING: (0x%02X:0x%04X): Command (0x%x) timed out, resetting card.\n",
|
||||||
|
TW_DRIVER, 0x2c, SCpnt->cmnd[0]);
|
||||||
|
|
||||||
/* Now reset the card and some of the device extension data */
|
/* Now reset the card and some of the device extension data */
|
||||||
if (twa_reset_device_extension(tw_dev, 0)) {
|
if (twa_reset_device_extension(tw_dev, 0)) {
|
||||||
|
|
|
@ -1432,7 +1432,9 @@ static int tw_scsi_eh_reset(struct scsi_cmnd *SCpnt)
|
||||||
|
|
||||||
tw_dev->num_resets++;
|
tw_dev->num_resets++;
|
||||||
|
|
||||||
printk(KERN_WARNING "3w-xxxx: scsi%d: WARNING: Unit #%d: Command (0x%x) timed out, resetting card.\n", tw_dev->host->host_no, SCpnt->device->id, SCpnt->cmnd[0]);
|
sdev_printk(KERN_WARNING, SCpnt->device,
|
||||||
|
"WARNING: Command (0x%x) timed out, resetting card.\n",
|
||||||
|
SCpnt->cmnd[0]);
|
||||||
|
|
||||||
/* Now reset the card and some of the device extension data */
|
/* Now reset the card and some of the device extension data */
|
||||||
if (tw_reset_device_extension(tw_dev, 0)) {
|
if (tw_reset_device_extension(tw_dev, 0)) {
|
||||||
|
|
|
@ -128,6 +128,7 @@
|
||||||
#include <linux/blkdev.h>
|
#include <linux/blkdev.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/device.h>
|
||||||
#include <asm/dma.h>
|
#include <asm/dma.h>
|
||||||
#include <asm/system.h>
|
#include <asm/system.h>
|
||||||
#include <asm/io.h>
|
#include <asm/io.h>
|
||||||
|
@ -831,8 +832,8 @@ process_extended_message(struct Scsi_Host *host,
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
/* SDTR message out of the blue, reject it */
|
/* SDTR message out of the blue, reject it */
|
||||||
printk(KERN_WARNING "scsi%d Unexpected SDTR msg\n",
|
shost_printk(KERN_WARNING, host,
|
||||||
host->host_no);
|
"Unexpected SDTR msg\n");
|
||||||
hostdata->msgout[0] = A_REJECT_MSG;
|
hostdata->msgout[0] = A_REJECT_MSG;
|
||||||
dma_cache_sync(hostdata->msgout, 1, DMA_TO_DEVICE);
|
dma_cache_sync(hostdata->msgout, 1, DMA_TO_DEVICE);
|
||||||
script_patch_16(hostdata->script, MessageCount, 1);
|
script_patch_16(hostdata->script, MessageCount, 1);
|
||||||
|
@ -906,15 +907,17 @@ process_message(struct Scsi_Host *host, struct NCR_700_Host_Parameters *hostdata
|
||||||
NCR_700_clear_flag(SCp->device, NCR_700_DEV_BEGIN_SYNC_NEGOTIATION);
|
NCR_700_clear_flag(SCp->device, NCR_700_DEV_BEGIN_SYNC_NEGOTIATION);
|
||||||
} else if(SCp != NULL && NCR_700_get_tag_neg_state(SCp->device) == NCR_700_DURING_TAG_NEGOTIATION) {
|
} else if(SCp != NULL && NCR_700_get_tag_neg_state(SCp->device) == NCR_700_DURING_TAG_NEGOTIATION) {
|
||||||
/* rejected our first simple tag message */
|
/* rejected our first simple tag message */
|
||||||
printk(KERN_WARNING "scsi%d (%d:%d) Rejected first tag queue attempt, turning off tag queueing\n", host->host_no, pun, lun);
|
scmd_printk(KERN_WARNING, SCp,
|
||||||
|
"Rejected first tag queue attempt, turning off tag queueing\n");
|
||||||
/* we're done negotiating */
|
/* we're done negotiating */
|
||||||
NCR_700_set_tag_neg_state(SCp->device, NCR_700_FINISHED_TAG_NEGOTIATION);
|
NCR_700_set_tag_neg_state(SCp->device, NCR_700_FINISHED_TAG_NEGOTIATION);
|
||||||
hostdata->tag_negotiated &= ~(1<<SCp->device->id);
|
hostdata->tag_negotiated &= ~(1<<scmd_id(SCp));
|
||||||
SCp->device->tagged_supported = 0;
|
SCp->device->tagged_supported = 0;
|
||||||
scsi_deactivate_tcq(SCp->device, host->cmd_per_lun);
|
scsi_deactivate_tcq(SCp->device, host->cmd_per_lun);
|
||||||
} else {
|
} else {
|
||||||
printk(KERN_WARNING "scsi%d (%d:%d) Unexpected REJECT Message %s\n",
|
shost_printk(KERN_WARNING, host,
|
||||||
host->host_no, pun, lun,
|
"(%d:%d) Unexpected REJECT Message %s\n",
|
||||||
|
pun, lun,
|
||||||
NCR_700_phase[(dsps & 0xf00) >> 8]);
|
NCR_700_phase[(dsps & 0xf00) >> 8]);
|
||||||
/* however, just ignore it */
|
/* however, just ignore it */
|
||||||
}
|
}
|
||||||
|
@ -983,7 +986,8 @@ process_script_interrupt(__u32 dsps, __u32 dsp, struct scsi_cmnd *SCp,
|
||||||
if(SCp->cmnd[0] == REQUEST_SENSE) {
|
if(SCp->cmnd[0] == REQUEST_SENSE) {
|
||||||
/* OOPS: bad device, returning another
|
/* OOPS: bad device, returning another
|
||||||
* contingent allegiance condition */
|
* contingent allegiance condition */
|
||||||
printk(KERN_ERR "scsi%d (%d:%d) broken device is looping in contingent allegiance: ignoring\n", host->host_no, pun, lun);
|
scmd_printk(KERN_ERR, SCp,
|
||||||
|
"broken device is looping in contingent allegiance: ignoring\n");
|
||||||
NCR_700_scsi_done(hostdata, SCp, hostdata->status[0]);
|
NCR_700_scsi_done(hostdata, SCp, hostdata->status[0]);
|
||||||
} else {
|
} else {
|
||||||
#ifdef NCR_DEBUG
|
#ifdef NCR_DEBUG
|
||||||
|
@ -1047,12 +1051,13 @@ process_script_interrupt(__u32 dsps, __u32 dsp, struct scsi_cmnd *SCp,
|
||||||
// SCp->request_bufflen,
|
// SCp->request_bufflen,
|
||||||
// DMA_FROM_DEVICE);
|
// DMA_FROM_DEVICE);
|
||||||
// if(((char *)SCp->request_buffer)[7] & 0x02) {
|
// if(((char *)SCp->request_buffer)[7] & 0x02) {
|
||||||
// printk(KERN_INFO "scsi%d: (%d:%d) Enabling Tag Command Queuing\n", host->host_no, pun, lun);
|
// scmd_printk(KERN_INFO, SCp,
|
||||||
// hostdata->tag_negotiated |= (1<<SCp->device->id);
|
// "Enabling Tag Command Queuing\n");
|
||||||
|
// hostdata->tag_negotiated |= (1<<scmd_id(SCp));
|
||||||
// NCR_700_set_flag(SCp->device, NCR_700_DEV_BEGIN_TAG_QUEUEING);
|
// NCR_700_set_flag(SCp->device, NCR_700_DEV_BEGIN_TAG_QUEUEING);
|
||||||
// } else {
|
// } else {
|
||||||
// NCR_700_clear_flag(SCp->device, NCR_700_DEV_BEGIN_TAG_QUEUEING);
|
// NCR_700_clear_flag(SCp->device, NCR_700_DEV_BEGIN_TAG_QUEUEING);
|
||||||
// hostdata->tag_negotiated &= ~(1<<SCp->device->id);
|
// hostdata->tag_negotiated &= ~(1<<scmd_id(SCp));
|
||||||
// }
|
// }
|
||||||
//}
|
//}
|
||||||
NCR_700_scsi_done(hostdata, SCp, hostdata->status[0]);
|
NCR_700_scsi_done(hostdata, SCp, hostdata->status[0]);
|
||||||
|
@ -1060,11 +1065,11 @@ process_script_interrupt(__u32 dsps, __u32 dsp, struct scsi_cmnd *SCp,
|
||||||
} else if((dsps & 0xfffff0f0) == A_UNEXPECTED_PHASE) {
|
} else if((dsps & 0xfffff0f0) == A_UNEXPECTED_PHASE) {
|
||||||
__u8 i = (dsps & 0xf00) >> 8;
|
__u8 i = (dsps & 0xf00) >> 8;
|
||||||
|
|
||||||
printk(KERN_ERR "scsi%d: (%d:%d), UNEXPECTED PHASE %s (%s)\n",
|
scmd_printk(KERN_ERR, SCp, "UNEXPECTED PHASE %s (%s)\n",
|
||||||
host->host_no, pun, lun,
|
|
||||||
NCR_700_phase[i],
|
NCR_700_phase[i],
|
||||||
sbcl_to_string(NCR_700_readb(host, SBCL_REG)));
|
sbcl_to_string(NCR_700_readb(host, SBCL_REG)));
|
||||||
printk(KERN_ERR " len = %d, cmd =", SCp->cmd_len);
|
scmd_printk(KERN_ERR, SCp, " len = %d, cmd =",
|
||||||
|
SCp->cmd_len);
|
||||||
scsi_print_command(SCp);
|
scsi_print_command(SCp);
|
||||||
|
|
||||||
NCR_700_internal_bus_reset(host);
|
NCR_700_internal_bus_reset(host);
|
||||||
|
@ -1115,14 +1120,14 @@ process_script_interrupt(__u32 dsps, __u32 dsp, struct scsi_cmnd *SCp,
|
||||||
}
|
}
|
||||||
|
|
||||||
slot = (struct NCR_700_command_slot *)SCp->host_scribble;
|
slot = (struct NCR_700_command_slot *)SCp->host_scribble;
|
||||||
DEBUG(("53c700: %d:%d:%d, reselection is tag %d, slot %p(%d)\n",
|
DDEBUG(KERN_DEBUG, SDp,
|
||||||
host->host_no, SDp->id, SDp->lun,
|
"reselection is tag %d, slot %p(%d)\n",
|
||||||
hostdata->msgin[2], slot, slot->tag));
|
hostdata->msgin[2], slot, slot->tag);
|
||||||
} else {
|
} else {
|
||||||
struct scsi_cmnd *SCp = scsi_find_tag(SDp, SCSI_NO_TAG);
|
struct scsi_cmnd *SCp = scsi_find_tag(SDp, SCSI_NO_TAG);
|
||||||
if(unlikely(SCp == NULL)) {
|
if(unlikely(SCp == NULL)) {
|
||||||
printk(KERN_ERR "scsi%d: (%d:%d) no saved request for untagged cmd\n",
|
sdev_printk(KERN_ERR, SDp,
|
||||||
host->host_no, reselection_id, lun);
|
"no saved request for untagged cmd\n");
|
||||||
BUG();
|
BUG();
|
||||||
}
|
}
|
||||||
slot = (struct NCR_700_command_slot *)SCp->host_scribble;
|
slot = (struct NCR_700_command_slot *)SCp->host_scribble;
|
||||||
|
@ -1422,7 +1427,7 @@ NCR_700_start_command(struct scsi_cmnd *SCp)
|
||||||
* If a contingent allegiance condition exists, the device
|
* If a contingent allegiance condition exists, the device
|
||||||
* will refuse all tags, so send the request sense as untagged
|
* will refuse all tags, so send the request sense as untagged
|
||||||
* */
|
* */
|
||||||
if((hostdata->tag_negotiated & (1<<SCp->device->id))
|
if((hostdata->tag_negotiated & (1<<scmd_id(SCp)))
|
||||||
&& (slot->tag != SCSI_NO_TAG && SCp->cmnd[0] != REQUEST_SENSE)) {
|
&& (slot->tag != SCSI_NO_TAG && SCp->cmnd[0] != REQUEST_SENSE)) {
|
||||||
count += scsi_populate_tag_msg(SCp, &hostdata->msgout[count]);
|
count += scsi_populate_tag_msg(SCp, &hostdata->msgout[count]);
|
||||||
}
|
}
|
||||||
|
@ -1441,7 +1446,7 @@ NCR_700_start_command(struct scsi_cmnd *SCp)
|
||||||
|
|
||||||
|
|
||||||
script_patch_ID(hostdata->script,
|
script_patch_ID(hostdata->script,
|
||||||
Device_ID, 1<<SCp->device->id);
|
Device_ID, 1<<scmd_id(SCp));
|
||||||
|
|
||||||
script_patch_32_abs(hostdata->script, CommandAddress,
|
script_patch_32_abs(hostdata->script, CommandAddress,
|
||||||
slot->pCmd);
|
slot->pCmd);
|
||||||
|
@ -1764,17 +1769,15 @@ NCR_700_queuecommand(struct scsi_cmnd *SCp, void (*done)(struct scsi_cmnd *))
|
||||||
* - The blk layer sent and untagged command
|
* - The blk layer sent and untagged command
|
||||||
*/
|
*/
|
||||||
if(NCR_700_get_depth(SCp->device) != 0
|
if(NCR_700_get_depth(SCp->device) != 0
|
||||||
&& (!(hostdata->tag_negotiated & (1<<SCp->device->id))
|
&& (!(hostdata->tag_negotiated & (1<<scmd_id(SCp)))
|
||||||
|| !blk_rq_tagged(SCp->request))) {
|
|| !blk_rq_tagged(SCp->request))) {
|
||||||
DEBUG((KERN_ERR "scsi%d (%d:%d) has non zero depth %d\n",
|
CDEBUG(KERN_ERR, SCp, "has non zero depth %d\n",
|
||||||
SCp->device->host->host_no, SCp->device->id, SCp->device->lun,
|
NCR_700_get_depth(SCp->device));
|
||||||
NCR_700_get_depth(SCp->device)));
|
|
||||||
return SCSI_MLQUEUE_DEVICE_BUSY;
|
return SCSI_MLQUEUE_DEVICE_BUSY;
|
||||||
}
|
}
|
||||||
if(NCR_700_get_depth(SCp->device) >= SCp->device->queue_depth) {
|
if(NCR_700_get_depth(SCp->device) >= SCp->device->queue_depth) {
|
||||||
DEBUG((KERN_ERR "scsi%d (%d:%d) has max tag depth %d\n",
|
CDEBUG(KERN_ERR, SCp, "has max tag depth %d\n",
|
||||||
SCp->device->host->host_no, SCp->device->id, SCp->device->lun,
|
NCR_700_get_depth(SCp->device));
|
||||||
NCR_700_get_depth(SCp->device)));
|
|
||||||
return SCSI_MLQUEUE_DEVICE_BUSY;
|
return SCSI_MLQUEUE_DEVICE_BUSY;
|
||||||
}
|
}
|
||||||
NCR_700_set_depth(SCp->device, NCR_700_get_depth(SCp->device) + 1);
|
NCR_700_set_depth(SCp->device, NCR_700_get_depth(SCp->device) + 1);
|
||||||
|
@ -1796,10 +1799,10 @@ NCR_700_queuecommand(struct scsi_cmnd *SCp, void (*done)(struct scsi_cmnd *))
|
||||||
scsi_print_command(SCp);
|
scsi_print_command(SCp);
|
||||||
#endif
|
#endif
|
||||||
if(blk_rq_tagged(SCp->request)
|
if(blk_rq_tagged(SCp->request)
|
||||||
&& (hostdata->tag_negotiated &(1<<SCp->device->id)) == 0
|
&& (hostdata->tag_negotiated &(1<<scmd_id(SCp))) == 0
|
||||||
&& NCR_700_get_tag_neg_state(SCp->device) == NCR_700_START_TAG_NEGOTIATION) {
|
&& NCR_700_get_tag_neg_state(SCp->device) == NCR_700_START_TAG_NEGOTIATION) {
|
||||||
printk(KERN_ERR "scsi%d: (%d:%d) Enabling Tag Command Queuing\n", SCp->device->host->host_no, SCp->device->id, SCp->device->lun);
|
scmd_printk(KERN_ERR, SCp, "Enabling Tag Command Queuing\n");
|
||||||
hostdata->tag_negotiated |= (1<<SCp->device->id);
|
hostdata->tag_negotiated |= (1<<scmd_id(SCp));
|
||||||
NCR_700_set_tag_neg_state(SCp->device, NCR_700_DURING_TAG_NEGOTIATION);
|
NCR_700_set_tag_neg_state(SCp->device, NCR_700_DURING_TAG_NEGOTIATION);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1810,17 +1813,16 @@ NCR_700_queuecommand(struct scsi_cmnd *SCp, void (*done)(struct scsi_cmnd *))
|
||||||
* FIXME: This will royally screw up on multiple LUN devices
|
* FIXME: This will royally screw up on multiple LUN devices
|
||||||
* */
|
* */
|
||||||
if(!blk_rq_tagged(SCp->request)
|
if(!blk_rq_tagged(SCp->request)
|
||||||
&& (hostdata->tag_negotiated &(1<<SCp->device->id))) {
|
&& (hostdata->tag_negotiated &(1<<scmd_id(SCp)))) {
|
||||||
printk(KERN_INFO "scsi%d: (%d:%d) Disabling Tag Command Queuing\n", SCp->device->host->host_no, SCp->device->id, SCp->device->lun);
|
scmd_printk(KERN_INFO, SCp, "Disabling Tag Command Queuing\n");
|
||||||
hostdata->tag_negotiated &= ~(1<<SCp->device->id);
|
hostdata->tag_negotiated &= ~(1<<scmd_id(SCp));
|
||||||
}
|
}
|
||||||
|
|
||||||
if((hostdata->tag_negotiated &(1<<SCp->device->id))
|
if((hostdata->tag_negotiated &(1<<scmd_id(SCp)))
|
||||||
&& scsi_get_tag_type(SCp->device)) {
|
&& scsi_get_tag_type(SCp->device)) {
|
||||||
slot->tag = SCp->request->tag;
|
slot->tag = SCp->request->tag;
|
||||||
DEBUG(("53c700 %d:%d:%d, sending out tag %d, slot %p\n",
|
CDEBUG(KERN_DEBUG, SCp, "sending out tag %d, slot %p\n",
|
||||||
SCp->device->host->host_no, SCp->device->id, SCp->device->lun, slot->tag,
|
slot->tag, slot);
|
||||||
slot));
|
|
||||||
} else {
|
} else {
|
||||||
slot->tag = SCSI_NO_TAG;
|
slot->tag = SCSI_NO_TAG;
|
||||||
/* must populate current_cmnd for scsi_find_tag to work */
|
/* must populate current_cmnd for scsi_find_tag to work */
|
||||||
|
@ -1920,8 +1922,8 @@ NCR_700_abort(struct scsi_cmnd * SCp)
|
||||||
{
|
{
|
||||||
struct NCR_700_command_slot *slot;
|
struct NCR_700_command_slot *slot;
|
||||||
|
|
||||||
printk(KERN_INFO "scsi%d (%d:%d) New error handler wants to abort command\n\t",
|
scmd_printk(KERN_INFO, SCp,
|
||||||
SCp->device->host->host_no, SCp->device->id, SCp->device->lun);
|
"New error handler wants to abort command\n\t");
|
||||||
scsi_print_command(SCp);
|
scsi_print_command(SCp);
|
||||||
|
|
||||||
slot = (struct NCR_700_command_slot *)SCp->host_scribble;
|
slot = (struct NCR_700_command_slot *)SCp->host_scribble;
|
||||||
|
@ -1954,8 +1956,8 @@ NCR_700_bus_reset(struct scsi_cmnd * SCp)
|
||||||
struct NCR_700_Host_Parameters *hostdata =
|
struct NCR_700_Host_Parameters *hostdata =
|
||||||
(struct NCR_700_Host_Parameters *)SCp->device->host->hostdata[0];
|
(struct NCR_700_Host_Parameters *)SCp->device->host->hostdata[0];
|
||||||
|
|
||||||
printk(KERN_INFO "scsi%d (%d:%d) New error handler wants BUS reset, cmd %p\n\t",
|
scmd_printk(KERN_INFO, SCp,
|
||||||
SCp->device->host->host_no, SCp->device->id, SCp->device->lun, SCp);
|
"New error handler wants BUS reset, cmd %p\n\t", SCp);
|
||||||
scsi_print_command(SCp);
|
scsi_print_command(SCp);
|
||||||
|
|
||||||
/* In theory, eh_complete should always be null because the
|
/* In theory, eh_complete should always be null because the
|
||||||
|
@ -1987,8 +1989,7 @@ NCR_700_bus_reset(struct scsi_cmnd * SCp)
|
||||||
STATIC int
|
STATIC int
|
||||||
NCR_700_host_reset(struct scsi_cmnd * SCp)
|
NCR_700_host_reset(struct scsi_cmnd * SCp)
|
||||||
{
|
{
|
||||||
printk(KERN_INFO "scsi%d (%d:%d) New error handler wants HOST reset\n\t",
|
scmd_printk(KERN_INFO, SCp, "New error handler wants HOST reset\n\t");
|
||||||
SCp->device->host->host_no, SCp->device->id, SCp->device->lun);
|
|
||||||
scsi_print_command(SCp);
|
scsi_print_command(SCp);
|
||||||
|
|
||||||
spin_lock_irq(SCp->device->host->host_lock);
|
spin_lock_irq(SCp->device->host->host_lock);
|
||||||
|
@ -2110,7 +2111,7 @@ static int NCR_700_change_queue_type(struct scsi_device *SDp, int tag_type)
|
||||||
/* shift back to the default unqueued number of commands
|
/* shift back to the default unqueued number of commands
|
||||||
* (the user can still raise this) */
|
* (the user can still raise this) */
|
||||||
scsi_deactivate_tcq(SDp, SDp->host->cmd_per_lun);
|
scsi_deactivate_tcq(SDp, SDp->host->cmd_per_lun);
|
||||||
hostdata->tag_negotiated &= ~(1 << SDp->id);
|
hostdata->tag_negotiated &= ~(1 << sdev_id(SDp));
|
||||||
} else {
|
} else {
|
||||||
/* Here, we cleared the negotiation flag above, so this
|
/* Here, we cleared the negotiation flag above, so this
|
||||||
* will force the driver to renegotiate */
|
* will force the driver to renegotiate */
|
||||||
|
|
|
@ -22,8 +22,14 @@
|
||||||
|
|
||||||
#ifdef NCR_700_DEBUG
|
#ifdef NCR_700_DEBUG
|
||||||
#define DEBUG(x) printk x
|
#define DEBUG(x) printk x
|
||||||
|
#define DDEBUG(prefix, sdev, fmt, a...) \
|
||||||
|
sdev_printk(prefix, sdev, fmt, ##a)
|
||||||
|
#define CDEBUG(prefix, scmd, fmt, a...) \
|
||||||
|
scmd_printk(prefix, scmd, fmt, ##a)
|
||||||
#else
|
#else
|
||||||
#define DEBUG(x)
|
#define DEBUG(x) do {} while (0)
|
||||||
|
#define DDEBUG(prefix, scmd, fmt, a...) do {} while (0)
|
||||||
|
#define CDEBUG(prefix, scmd, fmt, a...) do {} while (0)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* The number of available command slots */
|
/* The number of available command slots */
|
||||||
|
|
|
@ -229,7 +229,7 @@ config SCSI_FC_ATTRS
|
||||||
|
|
||||||
config SCSI_ISCSI_ATTRS
|
config SCSI_ISCSI_ATTRS
|
||||||
tristate "iSCSI Transport Attributes"
|
tristate "iSCSI Transport Attributes"
|
||||||
depends on SCSI
|
depends on SCSI && NET
|
||||||
help
|
help
|
||||||
If you wish to export transport-specific information about
|
If you wish to export transport-specific information about
|
||||||
each attached iSCSI device to sysfs, say Y.
|
each attached iSCSI device to sysfs, say Y.
|
||||||
|
@ -247,6 +247,30 @@ endmenu
|
||||||
menu "SCSI low-level drivers"
|
menu "SCSI low-level drivers"
|
||||||
depends on SCSI!=n
|
depends on SCSI!=n
|
||||||
|
|
||||||
|
config ISCSI_TCP
|
||||||
|
tristate "iSCSI Initiator over TCP/IP"
|
||||||
|
depends on SCSI && INET
|
||||||
|
select CRYPTO
|
||||||
|
select CRYPTO_MD5
|
||||||
|
select CRYPTO_CRC32C
|
||||||
|
select SCSI_ISCSI_ATTRS
|
||||||
|
help
|
||||||
|
The iSCSI Driver provides a host with the ability to access storage
|
||||||
|
through an IP network. The driver uses the iSCSI protocol to transport
|
||||||
|
SCSI requests and responses over a TCP/IP network between the host
|
||||||
|
(the "initiator") and "targets". Architecturally, the iSCSI driver
|
||||||
|
combines with the host's TCP/IP stack, network drivers, and Network
|
||||||
|
Interface Card (NIC) to provide the same functions as a SCSI or a
|
||||||
|
Fibre Channel (FC) adapter driver with a Host Bus Adapter (HBA).
|
||||||
|
|
||||||
|
To compile this driver as a module, choose M here: the
|
||||||
|
module will be called iscsi_tcp.
|
||||||
|
|
||||||
|
The userspace component needed to initialize the driver, documentation,
|
||||||
|
and sample configuration files can be found here:
|
||||||
|
|
||||||
|
http://linux-iscsi.sf.net
|
||||||
|
|
||||||
config SGIWD93_SCSI
|
config SGIWD93_SCSI
|
||||||
tristate "SGI WD93C93 SCSI Driver"
|
tristate "SGI WD93C93 SCSI Driver"
|
||||||
depends on SGI_IP22 && SCSI
|
depends on SGI_IP22 && SCSI
|
||||||
|
@ -596,19 +620,6 @@ config SCSI_OMIT_FLASHPOINT
|
||||||
substantial, so users of MultiMaster Host Adapters may wish to omit
|
substantial, so users of MultiMaster Host Adapters may wish to omit
|
||||||
it.
|
it.
|
||||||
|
|
||||||
#
|
|
||||||
# This is marked broken because it uses over 4kB of stack in
|
|
||||||
# just two routines:
|
|
||||||
# 2076 CpqTsProcessIMQEntry
|
|
||||||
# 2052 PeekIMQEntry
|
|
||||||
#
|
|
||||||
config SCSI_CPQFCTS
|
|
||||||
tristate "Compaq Fibre Channel 64-bit/66Mhz HBA support"
|
|
||||||
depends on PCI && SCSI && BROKEN
|
|
||||||
help
|
|
||||||
Say Y here to compile in support for the Compaq StorageWorks Fibre
|
|
||||||
Channel 64-bit/66Mhz Host Bus Adapter.
|
|
||||||
|
|
||||||
config SCSI_DMX3191D
|
config SCSI_DMX3191D
|
||||||
tristate "DMX3191D SCSI support"
|
tristate "DMX3191D SCSI support"
|
||||||
depends on PCI && SCSI
|
depends on PCI && SCSI
|
||||||
|
|
|
@ -33,6 +33,7 @@ obj-$(CONFIG_SCSI_FC_ATTRS) += scsi_transport_fc.o
|
||||||
obj-$(CONFIG_SCSI_ISCSI_ATTRS) += scsi_transport_iscsi.o
|
obj-$(CONFIG_SCSI_ISCSI_ATTRS) += scsi_transport_iscsi.o
|
||||||
obj-$(CONFIG_SCSI_SAS_ATTRS) += scsi_transport_sas.o
|
obj-$(CONFIG_SCSI_SAS_ATTRS) += scsi_transport_sas.o
|
||||||
|
|
||||||
|
obj-$(CONFIG_ISCSI_TCP) += iscsi_tcp.o
|
||||||
obj-$(CONFIG_SCSI_AMIGA7XX) += amiga7xx.o 53c7xx.o
|
obj-$(CONFIG_SCSI_AMIGA7XX) += amiga7xx.o 53c7xx.o
|
||||||
obj-$(CONFIG_A3000_SCSI) += a3000.o wd33c93.o
|
obj-$(CONFIG_A3000_SCSI) += a3000.o wd33c93.o
|
||||||
obj-$(CONFIG_A2091_SCSI) += a2091.o wd33c93.o
|
obj-$(CONFIG_A2091_SCSI) += a2091.o wd33c93.o
|
||||||
|
@ -119,7 +120,6 @@ obj-$(CONFIG_JAZZ_ESP) += NCR53C9x.o jazz_esp.o
|
||||||
obj-$(CONFIG_SUN3X_ESP) += NCR53C9x.o sun3x_esp.o
|
obj-$(CONFIG_SUN3X_ESP) += NCR53C9x.o sun3x_esp.o
|
||||||
obj-$(CONFIG_SCSI_DEBUG) += scsi_debug.o
|
obj-$(CONFIG_SCSI_DEBUG) += scsi_debug.o
|
||||||
obj-$(CONFIG_SCSI_FCAL) += fcal.o
|
obj-$(CONFIG_SCSI_FCAL) += fcal.o
|
||||||
obj-$(CONFIG_SCSI_CPQFCTS) += cpqfc.o
|
|
||||||
obj-$(CONFIG_SCSI_LASI700) += 53c700.o lasi700.o
|
obj-$(CONFIG_SCSI_LASI700) += 53c700.o lasi700.o
|
||||||
obj-$(CONFIG_SCSI_NSP32) += nsp32.o
|
obj-$(CONFIG_SCSI_NSP32) += nsp32.o
|
||||||
obj-$(CONFIG_SCSI_IPR) += ipr.o
|
obj-$(CONFIG_SCSI_IPR) += ipr.o
|
||||||
|
@ -164,8 +164,6 @@ ncr53c8xx-flags-$(CONFIG_SCSI_ZALON) \
|
||||||
CFLAGS_ncr53c8xx.o := $(ncr53c8xx-flags-y) $(ncr53c8xx-flags-m)
|
CFLAGS_ncr53c8xx.o := $(ncr53c8xx-flags-y) $(ncr53c8xx-flags-m)
|
||||||
zalon7xx-objs := zalon.o ncr53c8xx.o
|
zalon7xx-objs := zalon.o ncr53c8xx.o
|
||||||
NCR_Q720_mod-objs := NCR_Q720.o ncr53c8xx.o
|
NCR_Q720_mod-objs := NCR_Q720.o ncr53c8xx.o
|
||||||
cpqfc-objs := cpqfcTSinit.o cpqfcTScontrol.o cpqfcTSi2c.o \
|
|
||||||
cpqfcTSworker.o cpqfcTStrigger.o
|
|
||||||
libata-objs := libata-core.o libata-scsi.o
|
libata-objs := libata-core.o libata-scsi.o
|
||||||
|
|
||||||
# Files generated that shall be removed upon make clean
|
# Files generated that shall be removed upon make clean
|
||||||
|
|
|
@ -1247,13 +1247,13 @@ static void collect_stats(struct NCR5380_hostdata *hostdata, Scsi_Cmnd * cmd)
|
||||||
case WRITE:
|
case WRITE:
|
||||||
case WRITE_6:
|
case WRITE_6:
|
||||||
case WRITE_10:
|
case WRITE_10:
|
||||||
hostdata->time_write[cmd->device->id] += (jiffies - hostdata->timebase);
|
hostdata->time_write[scmd_id(cmd)] += (jiffies - hostdata->timebase);
|
||||||
hostdata->pendingw--;
|
hostdata->pendingw--;
|
||||||
break;
|
break;
|
||||||
case READ:
|
case READ:
|
||||||
case READ_6:
|
case READ_6:
|
||||||
case READ_10:
|
case READ_10:
|
||||||
hostdata->time_read[cmd->device->id] += (jiffies - hostdata->timebase);
|
hostdata->time_read[scmd_id(cmd)] += (jiffies - hostdata->timebase);
|
||||||
hostdata->pendingr--;
|
hostdata->pendingr--;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -1385,7 +1385,7 @@ static int NCR5380_select(struct Scsi_Host *instance, Scsi_Cmnd * cmd, int tag)
|
||||||
* the host and target ID's on the SCSI bus.
|
* the host and target ID's on the SCSI bus.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
NCR5380_write(OUTPUT_DATA_REG, (hostdata->id_mask | (1 << cmd->device->id)));
|
NCR5380_write(OUTPUT_DATA_REG, (hostdata->id_mask | (1 << scmd_id(cmd))));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Raise ATN while SEL is true before BSY goes false from arbitration,
|
* Raise ATN while SEL is true before BSY goes false from arbitration,
|
||||||
|
@ -1430,7 +1430,7 @@ static int NCR5380_select(struct Scsi_Host *instance, Scsi_Cmnd * cmd, int tag)
|
||||||
|
|
||||||
udelay(1);
|
udelay(1);
|
||||||
|
|
||||||
dprintk(NDEBUG_SELECTION, ("scsi%d : selecting target %d\n", instance->host_no, cmd->device->id));
|
dprintk(NDEBUG_SELECTION, ("scsi%d : selecting target %d\n", instance->host_no, scmd_id(cmd)));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The SCSI specification calls for a 250 ms timeout for the actual
|
* The SCSI specification calls for a 250 ms timeout for the actual
|
||||||
|
@ -1483,7 +1483,7 @@ part2:
|
||||||
|
|
||||||
if (!(NCR5380_read(STATUS_REG) & SR_BSY)) {
|
if (!(NCR5380_read(STATUS_REG) & SR_BSY)) {
|
||||||
NCR5380_write(INITIATOR_COMMAND_REG, ICR_BASE);
|
NCR5380_write(INITIATOR_COMMAND_REG, ICR_BASE);
|
||||||
if (hostdata->targets_present & (1 << cmd->device->id)) {
|
if (hostdata->targets_present & (1 << scmd_id(cmd))) {
|
||||||
printk(KERN_DEBUG "scsi%d : weirdness\n", instance->host_no);
|
printk(KERN_DEBUG "scsi%d : weirdness\n", instance->host_no);
|
||||||
if (hostdata->restart_select)
|
if (hostdata->restart_select)
|
||||||
printk(KERN_DEBUG "\trestart select\n");
|
printk(KERN_DEBUG "\trestart select\n");
|
||||||
|
@ -1499,7 +1499,7 @@ part2:
|
||||||
NCR5380_write(SELECT_ENABLE_REG, hostdata->id_mask);
|
NCR5380_write(SELECT_ENABLE_REG, hostdata->id_mask);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
hostdata->targets_present |= (1 << cmd->device->id);
|
hostdata->targets_present |= (1 << scmd_id(cmd));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Since we followed the SCSI spec, and raised ATN while SEL
|
* Since we followed the SCSI spec, and raised ATN while SEL
|
||||||
|
@ -2190,7 +2190,8 @@ static void NCR5380_information_transfer(struct Scsi_Host *instance) {
|
||||||
* If the watchdog timer fires, all future accesses to this
|
* If the watchdog timer fires, all future accesses to this
|
||||||
* device will use the polled-IO.
|
* device will use the polled-IO.
|
||||||
*/
|
*/
|
||||||
printk("scsi%d : switching target %d lun %d to slow handshake\n", instance->host_no, cmd->device->id, cmd->device->lun);
|
scmd_printk(KERN_INFO, cmd,
|
||||||
|
"switching to slow handshake\n");
|
||||||
cmd->device->borken = 1;
|
cmd->device->borken = 1;
|
||||||
NCR5380_write(INITIATOR_COMMAND_REG, ICR_BASE | ICR_ASSERT_ATN);
|
NCR5380_write(INITIATOR_COMMAND_REG, ICR_BASE | ICR_ASSERT_ATN);
|
||||||
sink = 1;
|
sink = 1;
|
||||||
|
@ -2429,9 +2430,11 @@ static void NCR5380_information_transfer(struct Scsi_Host *instance) {
|
||||||
scsi_print_msg(extended_msg);
|
scsi_print_msg(extended_msg);
|
||||||
printk("\n");
|
printk("\n");
|
||||||
} else if (tmp != EXTENDED_MESSAGE)
|
} else if (tmp != EXTENDED_MESSAGE)
|
||||||
printk("scsi%d: rejecting unknown message %02x from target %d, lun %d\n", instance->host_no, tmp, cmd->device->id, cmd->device->lun);
|
scmd_printk(KERN_INFO, cmd,
|
||||||
|
"rejecting unknown message %02x\n",tmp);
|
||||||
else
|
else
|
||||||
printk("scsi%d: rejecting unknown extended message code %02x, length %d from target %d, lun %d\n", instance->host_no, extended_msg[1], extended_msg[0], cmd->device->id, cmd->device->lun);
|
scmd_printk(KERN_INFO, cmd,
|
||||||
|
"rejecting unknown extended message code %02x, length %d\n", extended_msg[1], extended_msg[0]);
|
||||||
|
|
||||||
msgout = MESSAGE_REJECT;
|
msgout = MESSAGE_REJECT;
|
||||||
NCR5380_write(INITIATOR_COMMAND_REG, ICR_BASE | ICR_ASSERT_ATN);
|
NCR5380_write(INITIATOR_COMMAND_REG, ICR_BASE | ICR_ASSERT_ATN);
|
||||||
|
|
|
@ -936,7 +936,7 @@ static void esp_release_dmabufs(struct NCR_ESP *esp, Scsi_Cmnd *sp)
|
||||||
|
|
||||||
static void esp_restore_pointers(struct NCR_ESP *esp, Scsi_Cmnd *sp)
|
static void esp_restore_pointers(struct NCR_ESP *esp, Scsi_Cmnd *sp)
|
||||||
{
|
{
|
||||||
struct esp_pointers *ep = &esp->data_pointers[sp->device->id];
|
struct esp_pointers *ep = &esp->data_pointers[scmd_id(sp)];
|
||||||
|
|
||||||
sp->SCp.ptr = ep->saved_ptr;
|
sp->SCp.ptr = ep->saved_ptr;
|
||||||
sp->SCp.buffer = ep->saved_buffer;
|
sp->SCp.buffer = ep->saved_buffer;
|
||||||
|
@ -946,7 +946,7 @@ static void esp_restore_pointers(struct NCR_ESP *esp, Scsi_Cmnd *sp)
|
||||||
|
|
||||||
static void esp_save_pointers(struct NCR_ESP *esp, Scsi_Cmnd *sp)
|
static void esp_save_pointers(struct NCR_ESP *esp, Scsi_Cmnd *sp)
|
||||||
{
|
{
|
||||||
struct esp_pointers *ep = &esp->data_pointers[sp->device->id];
|
struct esp_pointers *ep = &esp->data_pointers[scmd_id(sp)];
|
||||||
|
|
||||||
ep->saved_ptr = sp->SCp.ptr;
|
ep->saved_ptr = sp->SCp.ptr;
|
||||||
ep->saved_buffer = sp->SCp.buffer;
|
ep->saved_buffer = sp->SCp.buffer;
|
||||||
|
@ -1693,13 +1693,13 @@ static inline void esp_connect(struct NCR_ESP *esp, struct ESP_regs *eregs,
|
||||||
if(esp->prev_soff != esp_dev->sync_max_offset ||
|
if(esp->prev_soff != esp_dev->sync_max_offset ||
|
||||||
esp->prev_stp != esp_dev->sync_min_period ||
|
esp->prev_stp != esp_dev->sync_min_period ||
|
||||||
(esp->erev > esp100a &&
|
(esp->erev > esp100a &&
|
||||||
esp->prev_cfg3 != esp->config3[sp->device->id])) {
|
esp->prev_cfg3 != esp->config3[scmd_id(sp)])) {
|
||||||
esp->prev_soff = esp_dev->sync_max_offset;
|
esp->prev_soff = esp_dev->sync_max_offset;
|
||||||
esp_write(eregs->esp_soff, esp->prev_soff);
|
esp_write(eregs->esp_soff, esp->prev_soff);
|
||||||
esp->prev_stp = esp_dev->sync_min_period;
|
esp->prev_stp = esp_dev->sync_min_period;
|
||||||
esp_write(eregs->esp_stp, esp->prev_stp);
|
esp_write(eregs->esp_stp, esp->prev_stp);
|
||||||
if(esp->erev > esp100a) {
|
if(esp->erev > esp100a) {
|
||||||
esp->prev_cfg3 = esp->config3[sp->device->id];
|
esp->prev_cfg3 = esp->config3[scmd_id(sp)];
|
||||||
esp_write(eregs->esp_cfg3, esp->prev_cfg3);
|
esp_write(eregs->esp_cfg3, esp->prev_cfg3);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2205,7 +2205,7 @@ static int esp_do_freebus(struct NCR_ESP *esp, struct ESP_regs *eregs)
|
||||||
|
|
||||||
if(SCptr->SCp.Status != GOOD &&
|
if(SCptr->SCp.Status != GOOD &&
|
||||||
SCptr->SCp.Status != CONDITION_GOOD &&
|
SCptr->SCp.Status != CONDITION_GOOD &&
|
||||||
((1<<SCptr->device->id) & esp->targets_present) &&
|
((1<<scmd_id(SCptr)) & esp->targets_present) &&
|
||||||
esp_dev->sync && esp_dev->sync_max_offset) {
|
esp_dev->sync && esp_dev->sync_max_offset) {
|
||||||
/* SCSI standard says that the synchronous capabilities
|
/* SCSI standard says that the synchronous capabilities
|
||||||
* should be renegotiated at this point. Most likely
|
* should be renegotiated at this point. Most likely
|
||||||
|
@ -2597,7 +2597,7 @@ static int esp_select_complete(struct NCR_ESP *esp, struct ESP_regs *eregs)
|
||||||
*/
|
*/
|
||||||
if(esp->ireg == (ESP_INTR_FDONE | ESP_INTR_BSERV)) {
|
if(esp->ireg == (ESP_INTR_FDONE | ESP_INTR_BSERV)) {
|
||||||
/* target speaks... */
|
/* target speaks... */
|
||||||
esp->targets_present |= (1<<SCptr->device->id);
|
esp->targets_present |= (1<<scmd_id(SCptr));
|
||||||
|
|
||||||
/* What if the target ignores the sdtr? */
|
/* What if the target ignores the sdtr? */
|
||||||
if(esp->snip)
|
if(esp->snip)
|
||||||
|
@ -3064,7 +3064,7 @@ static int check_multibyte_msg(struct NCR_ESP *esp,
|
||||||
ESPSDTR(("soff=%2x stp=%2x cfg3=%2x\n",
|
ESPSDTR(("soff=%2x stp=%2x cfg3=%2x\n",
|
||||||
esp_dev->sync_max_offset,
|
esp_dev->sync_max_offset,
|
||||||
esp_dev->sync_min_period,
|
esp_dev->sync_min_period,
|
||||||
esp->config3[SCptr->device->id]));
|
esp->config3[scmd_id(SCptr)]));
|
||||||
|
|
||||||
esp->snip = 0;
|
esp->snip = 0;
|
||||||
} else if(esp_dev->sync_max_offset) {
|
} else if(esp_dev->sync_max_offset) {
|
||||||
|
@ -3621,7 +3621,7 @@ void esp_slave_destroy(Scsi_Device *SDptr)
|
||||||
{
|
{
|
||||||
struct NCR_ESP *esp = (struct NCR_ESP *) SDptr->host->hostdata;
|
struct NCR_ESP *esp = (struct NCR_ESP *) SDptr->host->hostdata;
|
||||||
|
|
||||||
esp->targets_present &= ~(1 << SDptr->id);
|
esp->targets_present &= ~(1 << sdev_id(SDptr));
|
||||||
kfree(SDptr->hostdata);
|
kfree(SDptr->hostdata);
|
||||||
SDptr->hostdata = NULL;
|
SDptr->hostdata = NULL;
|
||||||
}
|
}
|
||||||
|
|
|
@ -710,7 +710,7 @@ static int NCR53c406a_queue(Scsi_Cmnd * SCpnt, void (*done) (Scsi_Cmnd *))
|
||||||
|
|
||||||
/* We are locked here already by the mid layer */
|
/* We are locked here already by the mid layer */
|
||||||
REG0;
|
REG0;
|
||||||
outb(SCpnt->device->id, DEST_ID); /* set destination */
|
outb(scmd_id(SCpnt), DEST_ID); /* set destination */
|
||||||
outb(FLUSH_FIFO, CMD_REG); /* reset the fifos */
|
outb(FLUSH_FIFO, CMD_REG); /* reset the fifos */
|
||||||
|
|
||||||
for (i = 0; i < SCpnt->cmd_len; i++) {
|
for (i = 0; i < SCpnt->cmd_len; i++) {
|
||||||
|
|
|
@ -923,7 +923,7 @@ static int inia100_device_reset(struct scsi_cmnd * SCpnt)
|
||||||
{ /* I need Host Control Block Information */
|
{ /* I need Host Control Block Information */
|
||||||
ORC_HCS *pHCB;
|
ORC_HCS *pHCB;
|
||||||
pHCB = (ORC_HCS *) SCpnt->device->host->hostdata;
|
pHCB = (ORC_HCS *) SCpnt->device->host->hostdata;
|
||||||
return orc_device_reset(pHCB, SCpnt, SCpnt->device->id);
|
return orc_device_reset(pHCB, SCpnt, scmd_id(SCpnt));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -57,7 +57,7 @@ Deanna Bonds (non-DASD support, PAE fibs and 64 bit,
|
||||||
(fixed 64bit and 64G memory model, changed confusing naming convention
|
(fixed 64bit and 64G memory model, changed confusing naming convention
|
||||||
where fibs that go to the hardware are consistently called hw_fibs and
|
where fibs that go to the hardware are consistently called hw_fibs and
|
||||||
not just fibs like the name of the driver tracking structure)
|
not just fibs like the name of the driver tracking structure)
|
||||||
Mark Salyzyn <Mark_Salyzyn@adaptec.com> Fixed panic issues and added some new product ids for upcoming hbas.
|
Mark Salyzyn <Mark_Salyzyn@adaptec.com> Fixed panic issues and added some new product ids for upcoming hbas. Performance tuning, card failover and bug mitigations.
|
||||||
|
|
||||||
Original Driver
|
Original Driver
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
o Testing
|
o Testing
|
||||||
o More testing
|
o More testing
|
||||||
o Drop irq_mask, basically unused
|
|
||||||
o I/O size increase
|
o I/O size increase
|
||||||
|
|
|
@ -359,15 +359,6 @@ int aac_get_containers(struct aac_dev *dev)
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void aac_io_done(struct scsi_cmnd * scsicmd)
|
|
||||||
{
|
|
||||||
unsigned long cpu_flags;
|
|
||||||
struct Scsi_Host *host = scsicmd->device->host;
|
|
||||||
spin_lock_irqsave(host->host_lock, cpu_flags);
|
|
||||||
scsicmd->scsi_done(scsicmd);
|
|
||||||
spin_unlock_irqrestore(host->host_lock, cpu_flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void aac_internal_transfer(struct scsi_cmnd *scsicmd, void *data, unsigned int offset, unsigned int len)
|
static void aac_internal_transfer(struct scsi_cmnd *scsicmd, void *data, unsigned int offset, unsigned int len)
|
||||||
{
|
{
|
||||||
void *buf;
|
void *buf;
|
||||||
|
@ -424,7 +415,7 @@ static void get_container_name_callback(void *context, struct fib * fibptr)
|
||||||
|
|
||||||
fib_complete(fibptr);
|
fib_complete(fibptr);
|
||||||
fib_free(fibptr);
|
fib_free(fibptr);
|
||||||
aac_io_done(scsicmd);
|
scsicmd->scsi_done(scsicmd);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -608,17 +599,43 @@ static char *container_types[] = {
|
||||||
* files instead of in OS dependant driver source.
|
* files instead of in OS dependant driver source.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static void setinqstr(int devtype, void *data, int tindex)
|
static void setinqstr(struct aac_dev *dev, void *data, int tindex)
|
||||||
{
|
{
|
||||||
struct scsi_inq *str;
|
struct scsi_inq *str;
|
||||||
struct aac_driver_ident *mp;
|
|
||||||
|
|
||||||
mp = aac_get_driver_ident(devtype);
|
|
||||||
|
|
||||||
str = (struct scsi_inq *)(data); /* cast data to scsi inq block */
|
str = (struct scsi_inq *)(data); /* cast data to scsi inq block */
|
||||||
|
memset(str, ' ', sizeof(*str));
|
||||||
|
|
||||||
inqstrcpy (mp->vname, str->vid);
|
if (dev->supplement_adapter_info.AdapterTypeText[0]) {
|
||||||
inqstrcpy (mp->model, str->pid); /* last six chars reserved for vol type */
|
char * cp = dev->supplement_adapter_info.AdapterTypeText;
|
||||||
|
int c = sizeof(str->vid);
|
||||||
|
while (*cp && *cp != ' ' && --c)
|
||||||
|
++cp;
|
||||||
|
c = *cp;
|
||||||
|
*cp = '\0';
|
||||||
|
inqstrcpy (dev->supplement_adapter_info.AdapterTypeText,
|
||||||
|
str->vid);
|
||||||
|
*cp = c;
|
||||||
|
while (*cp && *cp != ' ')
|
||||||
|
++cp;
|
||||||
|
while (*cp == ' ')
|
||||||
|
++cp;
|
||||||
|
/* last six chars reserved for vol type */
|
||||||
|
c = 0;
|
||||||
|
if (strlen(cp) > sizeof(str->pid)) {
|
||||||
|
c = cp[sizeof(str->pid)];
|
||||||
|
cp[sizeof(str->pid)] = '\0';
|
||||||
|
}
|
||||||
|
inqstrcpy (cp, str->pid);
|
||||||
|
if (c)
|
||||||
|
cp[sizeof(str->pid)] = c;
|
||||||
|
} else {
|
||||||
|
struct aac_driver_ident *mp = aac_get_driver_ident(dev->cardtype);
|
||||||
|
|
||||||
|
inqstrcpy (mp->vname, str->vid);
|
||||||
|
/* last six chars reserved for vol type */
|
||||||
|
inqstrcpy (mp->model, str->pid);
|
||||||
|
}
|
||||||
|
|
||||||
if (tindex < (sizeof(container_types)/sizeof(char *))){
|
if (tindex < (sizeof(container_types)/sizeof(char *))){
|
||||||
char *findit = str->pid;
|
char *findit = str->pid;
|
||||||
|
@ -627,7 +644,9 @@ static void setinqstr(int devtype, void *data, int tindex)
|
||||||
/* RAID is superfluous in the context of a RAID device */
|
/* RAID is superfluous in the context of a RAID device */
|
||||||
if (memcmp(findit-4, "RAID", 4) == 0)
|
if (memcmp(findit-4, "RAID", 4) == 0)
|
||||||
*(findit -= 4) = ' ';
|
*(findit -= 4) = ' ';
|
||||||
inqstrcpy (container_types[tindex], findit + 1);
|
if (((findit - str->pid) + strlen(container_types[tindex]))
|
||||||
|
< (sizeof(str->pid) + sizeof(str->prl)))
|
||||||
|
inqstrcpy (container_types[tindex], findit + 1);
|
||||||
}
|
}
|
||||||
inqstrcpy ("V1.0", str->prl);
|
inqstrcpy ("V1.0", str->prl);
|
||||||
}
|
}
|
||||||
|
@ -822,12 +841,12 @@ int aac_get_adapter_info(struct aac_dev* dev)
|
||||||
dev->dac_support = (dacmode!=0);
|
dev->dac_support = (dacmode!=0);
|
||||||
}
|
}
|
||||||
if(dev->dac_support != 0) {
|
if(dev->dac_support != 0) {
|
||||||
if (!pci_set_dma_mask(dev->pdev, 0xFFFFFFFFFFFFFFFFULL) &&
|
if (!pci_set_dma_mask(dev->pdev, DMA_64BIT_MASK) &&
|
||||||
!pci_set_consistent_dma_mask(dev->pdev, 0xFFFFFFFFFFFFFFFFULL)) {
|
!pci_set_consistent_dma_mask(dev->pdev, DMA_64BIT_MASK)) {
|
||||||
printk(KERN_INFO"%s%d: 64 Bit DAC enabled\n",
|
printk(KERN_INFO"%s%d: 64 Bit DAC enabled\n",
|
||||||
dev->name, dev->id);
|
dev->name, dev->id);
|
||||||
} else if (!pci_set_dma_mask(dev->pdev, 0xFFFFFFFFULL) &&
|
} else if (!pci_set_dma_mask(dev->pdev, DMA_32BIT_MASK) &&
|
||||||
!pci_set_consistent_dma_mask(dev->pdev, 0xFFFFFFFFULL)) {
|
!pci_set_consistent_dma_mask(dev->pdev, DMA_32BIT_MASK)) {
|
||||||
printk(KERN_INFO"%s%d: DMA mask set failed, 64 Bit DAC disabled\n",
|
printk(KERN_INFO"%s%d: DMA mask set failed, 64 Bit DAC disabled\n",
|
||||||
dev->name, dev->id);
|
dev->name, dev->id);
|
||||||
dev->dac_support = 0;
|
dev->dac_support = 0;
|
||||||
|
@ -960,7 +979,7 @@ static void io_callback(void *context, struct fib * fibptr)
|
||||||
fib_complete(fibptr);
|
fib_complete(fibptr);
|
||||||
fib_free(fibptr);
|
fib_free(fibptr);
|
||||||
|
|
||||||
aac_io_done(scsicmd);
|
scsicmd->scsi_done(scsicmd);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int aac_read(struct scsi_cmnd * scsicmd, int cid)
|
static int aac_read(struct scsi_cmnd * scsicmd, int cid)
|
||||||
|
@ -1139,7 +1158,7 @@ static int aac_read(struct scsi_cmnd * scsicmd, int cid)
|
||||||
* For some reason, the Fib didn't queue, return QUEUE_FULL
|
* For some reason, the Fib didn't queue, return QUEUE_FULL
|
||||||
*/
|
*/
|
||||||
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_TASK_SET_FULL;
|
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_TASK_SET_FULL;
|
||||||
aac_io_done(scsicmd);
|
scsicmd->scsi_done(scsicmd);
|
||||||
fib_complete(cmd_fibcontext);
|
fib_complete(cmd_fibcontext);
|
||||||
fib_free(cmd_fibcontext);
|
fib_free(cmd_fibcontext);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1211,7 +1230,7 @@ static int aac_write(struct scsi_cmnd * scsicmd, int cid)
|
||||||
*/
|
*/
|
||||||
if (!(cmd_fibcontext = fib_alloc(dev))) {
|
if (!(cmd_fibcontext = fib_alloc(dev))) {
|
||||||
scsicmd->result = DID_ERROR << 16;
|
scsicmd->result = DID_ERROR << 16;
|
||||||
aac_io_done(scsicmd);
|
scsicmd->scsi_done(scsicmd);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
fib_init(cmd_fibcontext);
|
fib_init(cmd_fibcontext);
|
||||||
|
@ -1308,7 +1327,7 @@ static int aac_write(struct scsi_cmnd * scsicmd, int cid)
|
||||||
* For some reason, the Fib didn't queue, return QUEUE_FULL
|
* For some reason, the Fib didn't queue, return QUEUE_FULL
|
||||||
*/
|
*/
|
||||||
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_TASK_SET_FULL;
|
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_TASK_SET_FULL;
|
||||||
aac_io_done(scsicmd);
|
scsicmd->scsi_done(scsicmd);
|
||||||
|
|
||||||
fib_complete(cmd_fibcontext);
|
fib_complete(cmd_fibcontext);
|
||||||
fib_free(cmd_fibcontext);
|
fib_free(cmd_fibcontext);
|
||||||
|
@ -1352,7 +1371,7 @@ static void synchronize_callback(void *context, struct fib *fibptr)
|
||||||
|
|
||||||
fib_complete(fibptr);
|
fib_complete(fibptr);
|
||||||
fib_free(fibptr);
|
fib_free(fibptr);
|
||||||
aac_io_done(cmd);
|
cmd->scsi_done(cmd);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int aac_synchronize(struct scsi_cmnd *scsicmd, int cid)
|
static int aac_synchronize(struct scsi_cmnd *scsicmd, int cid)
|
||||||
|
@ -1438,7 +1457,6 @@ int aac_scsi_cmd(struct scsi_cmnd * scsicmd)
|
||||||
struct Scsi_Host *host = scsicmd->device->host;
|
struct Scsi_Host *host = scsicmd->device->host;
|
||||||
struct aac_dev *dev = (struct aac_dev *)host->hostdata;
|
struct aac_dev *dev = (struct aac_dev *)host->hostdata;
|
||||||
struct fsa_dev_info *fsa_dev_ptr = dev->fsa_dev;
|
struct fsa_dev_info *fsa_dev_ptr = dev->fsa_dev;
|
||||||
int cardtype = dev->cardtype;
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -1446,7 +1464,7 @@ int aac_scsi_cmd(struct scsi_cmnd * scsicmd)
|
||||||
* Test does not apply to ID 16, the pseudo id for the controller
|
* Test does not apply to ID 16, the pseudo id for the controller
|
||||||
* itself.
|
* itself.
|
||||||
*/
|
*/
|
||||||
if (scsicmd->device->id != host->this_id) {
|
if (scmd_id(scsicmd) != host->this_id) {
|
||||||
if ((scsicmd->device->channel == 0) ){
|
if ((scsicmd->device->channel == 0) ){
|
||||||
if( (scsicmd->device->id >= dev->maximum_num_containers) || (scsicmd->device->lun != 0)){
|
if( (scsicmd->device->id >= dev->maximum_num_containers) || (scsicmd->device->lun != 0)){
|
||||||
scsicmd->result = DID_NO_CONNECT << 16;
|
scsicmd->result = DID_NO_CONNECT << 16;
|
||||||
|
@ -1541,15 +1559,15 @@ int aac_scsi_cmd(struct scsi_cmnd * scsicmd)
|
||||||
* Set the Vendor, Product, and Revision Level
|
* Set the Vendor, Product, and Revision Level
|
||||||
* see: <vendor>.c i.e. aac.c
|
* see: <vendor>.c i.e. aac.c
|
||||||
*/
|
*/
|
||||||
if (scsicmd->device->id == host->this_id) {
|
if (scmd_id(scsicmd) == host->this_id) {
|
||||||
setinqstr(cardtype, (void *) (inq_data.inqd_vid), (sizeof(container_types)/sizeof(char *)));
|
setinqstr(dev, (void *) (inq_data.inqd_vid), (sizeof(container_types)/sizeof(char *)));
|
||||||
inq_data.inqd_pdt = INQD_PDT_PROC; /* Processor device */
|
inq_data.inqd_pdt = INQD_PDT_PROC; /* Processor device */
|
||||||
aac_internal_transfer(scsicmd, &inq_data, 0, sizeof(inq_data));
|
aac_internal_transfer(scsicmd, &inq_data, 0, sizeof(inq_data));
|
||||||
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD;
|
scsicmd->result = DID_OK << 16 | COMMAND_COMPLETE << 8 | SAM_STAT_GOOD;
|
||||||
scsicmd->scsi_done(scsicmd);
|
scsicmd->scsi_done(scsicmd);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
setinqstr(cardtype, (void *) (inq_data.inqd_vid), fsa_dev_ptr[cid].type);
|
setinqstr(dev, (void *) (inq_data.inqd_vid), fsa_dev_ptr[cid].type);
|
||||||
inq_data.inqd_pdt = INQD_PDT_DA; /* Direct/random access device */
|
inq_data.inqd_pdt = INQD_PDT_DA; /* Direct/random access device */
|
||||||
aac_internal_transfer(scsicmd, &inq_data, 0, sizeof(inq_data));
|
aac_internal_transfer(scsicmd, &inq_data, 0, sizeof(inq_data));
|
||||||
return aac_get_container_name(scsicmd, cid);
|
return aac_get_container_name(scsicmd, cid);
|
||||||
|
@ -1931,7 +1949,7 @@ static void aac_srb_callback(void *context, struct fib * fibptr)
|
||||||
* the channel is 2
|
* the channel is 2
|
||||||
*/
|
*/
|
||||||
} else if ((dev->raid_scsi_mode) &&
|
} else if ((dev->raid_scsi_mode) &&
|
||||||
(scsicmd->device->channel == 2)) {
|
(scmd_channel(scsicmd) == 2)) {
|
||||||
scsicmd->result = DID_OK << 16 |
|
scsicmd->result = DID_OK << 16 |
|
||||||
COMMAND_COMPLETE << 8;
|
COMMAND_COMPLETE << 8;
|
||||||
} else {
|
} else {
|
||||||
|
@ -1975,7 +1993,7 @@ static void aac_srb_callback(void *context, struct fib * fibptr)
|
||||||
* the channel is 2
|
* the channel is 2
|
||||||
*/
|
*/
|
||||||
} else if ((dev->raid_scsi_mode) &&
|
} else if ((dev->raid_scsi_mode) &&
|
||||||
(scsicmd->device->channel == 2)) {
|
(scmd_channel(scsicmd) == 2)) {
|
||||||
scsicmd->result = DID_OK << 16 |
|
scsicmd->result = DID_OK << 16 |
|
||||||
COMMAND_COMPLETE << 8;
|
COMMAND_COMPLETE << 8;
|
||||||
} else {
|
} else {
|
||||||
|
@ -2070,7 +2088,7 @@ static void aac_srb_callback(void *context, struct fib * fibptr)
|
||||||
|
|
||||||
fib_complete(fibptr);
|
fib_complete(fibptr);
|
||||||
fib_free(fibptr);
|
fib_free(fibptr);
|
||||||
aac_io_done(scsicmd);
|
scsicmd->scsi_done(scsicmd);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -481,6 +481,7 @@ enum aac_log_level {
|
||||||
#define FSAFS_NTC_FIB_CONTEXT 0x030c
|
#define FSAFS_NTC_FIB_CONTEXT 0x030c
|
||||||
|
|
||||||
struct aac_dev;
|
struct aac_dev;
|
||||||
|
struct fib;
|
||||||
|
|
||||||
struct adapter_ops
|
struct adapter_ops
|
||||||
{
|
{
|
||||||
|
@ -489,6 +490,7 @@ struct adapter_ops
|
||||||
void (*adapter_disable_int)(struct aac_dev *dev);
|
void (*adapter_disable_int)(struct aac_dev *dev);
|
||||||
int (*adapter_sync_cmd)(struct aac_dev *dev, u32 command, u32 p1, u32 p2, u32 p3, u32 p4, u32 p5, u32 p6, u32 *status, u32 *r1, u32 *r2, u32 *r3, u32 *r4);
|
int (*adapter_sync_cmd)(struct aac_dev *dev, u32 command, u32 p1, u32 p2, u32 p3, u32 p4, u32 p5, u32 p6, u32 *status, u32 *r1, u32 *r2, u32 *r3, u32 *r4);
|
||||||
int (*adapter_check_health)(struct aac_dev *dev);
|
int (*adapter_check_health)(struct aac_dev *dev);
|
||||||
|
int (*adapter_send)(struct fib * fib);
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -659,6 +661,10 @@ struct rx_mu_registers {
|
||||||
Status Register */
|
Status Register */
|
||||||
__le32 OIMR; /* 1334h | 34h | Outbound Interrupt
|
__le32 OIMR; /* 1334h | 34h | Outbound Interrupt
|
||||||
Mask Register */
|
Mask Register */
|
||||||
|
__le32 reserved2; /* 1338h | 38h | Reserved */
|
||||||
|
__le32 reserved3; /* 133Ch | 3Ch | Reserved */
|
||||||
|
__le32 InboundQueue;/* 1340h | 40h | Inbound Queue Port relative to firmware */
|
||||||
|
__le32 OutboundQueue;/*1344h | 44h | Outbound Queue Port relative to firmware */
|
||||||
/* * Must access through ATU Inbound
|
/* * Must access through ATU Inbound
|
||||||
Translation Window */
|
Translation Window */
|
||||||
};
|
};
|
||||||
|
@ -693,8 +699,8 @@ struct rx_inbound {
|
||||||
#define OutboundDoorbellReg MUnit.ODR
|
#define OutboundDoorbellReg MUnit.ODR
|
||||||
|
|
||||||
struct rx_registers {
|
struct rx_registers {
|
||||||
struct rx_mu_registers MUnit; /* 1300h - 1334h */
|
struct rx_mu_registers MUnit; /* 1300h - 1344h */
|
||||||
__le32 reserved1[6]; /* 1338h - 134ch */
|
__le32 reserved1[2]; /* 1348h - 134ch */
|
||||||
struct rx_inbound IndexRegs;
|
struct rx_inbound IndexRegs;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -711,8 +717,8 @@ struct rx_registers {
|
||||||
#define rkt_inbound rx_inbound
|
#define rkt_inbound rx_inbound
|
||||||
|
|
||||||
struct rkt_registers {
|
struct rkt_registers {
|
||||||
struct rkt_mu_registers MUnit; /* 1300h - 1334h */
|
struct rkt_mu_registers MUnit; /* 1300h - 1344h */
|
||||||
__le32 reserved1[1010]; /* 1338h - 22fch */
|
__le32 reserved1[1006]; /* 1348h - 22fch */
|
||||||
struct rkt_inbound IndexRegs; /* 2300h - */
|
struct rkt_inbound IndexRegs; /* 2300h - */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -721,8 +727,6 @@ struct rkt_registers {
|
||||||
#define rkt_writeb(AEP, CSR, value) writeb(value, &((AEP)->regs.rkt->CSR))
|
#define rkt_writeb(AEP, CSR, value) writeb(value, &((AEP)->regs.rkt->CSR))
|
||||||
#define rkt_writel(AEP, CSR, value) writel(value, &((AEP)->regs.rkt->CSR))
|
#define rkt_writel(AEP, CSR, value) writel(value, &((AEP)->regs.rkt->CSR))
|
||||||
|
|
||||||
struct fib;
|
|
||||||
|
|
||||||
typedef void (*fib_callback)(void *ctxt, struct fib *fibctx);
|
typedef void (*fib_callback)(void *ctxt, struct fib *fibctx);
|
||||||
|
|
||||||
struct aac_fib_context {
|
struct aac_fib_context {
|
||||||
|
@ -937,7 +941,6 @@ struct aac_dev
|
||||||
const char *name;
|
const char *name;
|
||||||
int id;
|
int id;
|
||||||
|
|
||||||
u16 irq_mask;
|
|
||||||
/*
|
/*
|
||||||
* negotiated FIB settings
|
* negotiated FIB settings
|
||||||
*/
|
*/
|
||||||
|
@ -972,6 +975,7 @@ struct aac_dev
|
||||||
struct adapter_ops a_ops;
|
struct adapter_ops a_ops;
|
||||||
unsigned long fsrev; /* Main driver's revision number */
|
unsigned long fsrev; /* Main driver's revision number */
|
||||||
|
|
||||||
|
unsigned base_size; /* Size of mapped in region */
|
||||||
struct aac_init *init; /* Holds initialization info to communicate with adapter */
|
struct aac_init *init; /* Holds initialization info to communicate with adapter */
|
||||||
dma_addr_t init_pa; /* Holds physical address of the init struct */
|
dma_addr_t init_pa; /* Holds physical address of the init struct */
|
||||||
|
|
||||||
|
@ -992,6 +996,9 @@ struct aac_dev
|
||||||
/*
|
/*
|
||||||
* The following is the device specific extension.
|
* The following is the device specific extension.
|
||||||
*/
|
*/
|
||||||
|
#if (!defined(AAC_MIN_FOOTPRINT_SIZE))
|
||||||
|
# define AAC_MIN_FOOTPRINT_SIZE 8192
|
||||||
|
#endif
|
||||||
union
|
union
|
||||||
{
|
{
|
||||||
struct sa_registers __iomem *sa;
|
struct sa_registers __iomem *sa;
|
||||||
|
@ -1012,6 +1019,7 @@ struct aac_dev
|
||||||
u8 nondasd_support;
|
u8 nondasd_support;
|
||||||
u8 dac_support;
|
u8 dac_support;
|
||||||
u8 raid_scsi_mode;
|
u8 raid_scsi_mode;
|
||||||
|
u8 new_comm_interface;
|
||||||
/* macro side-effects BEWARE */
|
/* macro side-effects BEWARE */
|
||||||
# define raw_io_interface \
|
# define raw_io_interface \
|
||||||
init->InitStructRevision==cpu_to_le32(ADAPTER_INIT_STRUCT_REVISION_4)
|
init->InitStructRevision==cpu_to_le32(ADAPTER_INIT_STRUCT_REVISION_4)
|
||||||
|
@ -1034,6 +1042,8 @@ struct aac_dev
|
||||||
#define aac_adapter_check_health(dev) \
|
#define aac_adapter_check_health(dev) \
|
||||||
(dev)->a_ops.adapter_check_health(dev)
|
(dev)->a_ops.adapter_check_health(dev)
|
||||||
|
|
||||||
|
#define aac_adapter_send(fib) \
|
||||||
|
((fib)->dev)->a_ops.adapter_send(fib)
|
||||||
|
|
||||||
#define FIB_CONTEXT_FLAG_TIMED_OUT (0x00000001)
|
#define FIB_CONTEXT_FLAG_TIMED_OUT (0x00000001)
|
||||||
|
|
||||||
|
@ -1560,7 +1570,7 @@ struct fib_ioctl
|
||||||
|
|
||||||
struct revision
|
struct revision
|
||||||
{
|
{
|
||||||
__le32 compat;
|
u32 compat;
|
||||||
__le32 version;
|
__le32 version;
|
||||||
__le32 build;
|
__le32 build;
|
||||||
};
|
};
|
||||||
|
@ -1779,6 +1789,7 @@ int aac_rkt_init(struct aac_dev *dev);
|
||||||
int aac_sa_init(struct aac_dev *dev);
|
int aac_sa_init(struct aac_dev *dev);
|
||||||
unsigned int aac_response_normal(struct aac_queue * q);
|
unsigned int aac_response_normal(struct aac_queue * q);
|
||||||
unsigned int aac_command_normal(struct aac_queue * q);
|
unsigned int aac_command_normal(struct aac_queue * q);
|
||||||
|
unsigned int aac_intr_normal(struct aac_dev * dev, u32 Index);
|
||||||
int aac_command_thread(struct aac_dev * dev);
|
int aac_command_thread(struct aac_dev * dev);
|
||||||
int aac_close_fib_context(struct aac_dev * dev, struct aac_fib_context *fibctx);
|
int aac_close_fib_context(struct aac_dev * dev, struct aac_fib_context *fibctx);
|
||||||
int fib_adapter_complete(struct fib * fibptr, unsigned short size);
|
int fib_adapter_complete(struct fib * fibptr, unsigned short size);
|
||||||
|
|
|
@ -408,7 +408,7 @@ static int check_revision(struct aac_dev *dev, void __user *arg)
|
||||||
char *driver_version = aac_driver_version;
|
char *driver_version = aac_driver_version;
|
||||||
u32 version;
|
u32 version;
|
||||||
|
|
||||||
response.compat = cpu_to_le32(1);
|
response.compat = 1;
|
||||||
version = (simple_strtol(driver_version,
|
version = (simple_strtol(driver_version,
|
||||||
&driver_version, 10) << 24) | 0x00000400;
|
&driver_version, 10) << 24) | 0x00000400;
|
||||||
version += simple_strtol(driver_version + 1, &driver_version, 10) << 16;
|
version += simple_strtol(driver_version + 1, &driver_version, 10) << 16;
|
||||||
|
@ -574,7 +574,7 @@ static int aac_send_raw_srb(struct aac_dev* dev, void __user * arg)
|
||||||
rcode = -ENOMEM;
|
rcode = -ENOMEM;
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
sg_user[i] = (void __user *)usg->sg[i].addr;
|
sg_user[i] = (void __user *)(long)usg->sg[i].addr;
|
||||||
sg_list[i] = p; // save so we can clean up later
|
sg_list[i] = p; // save so we can clean up later
|
||||||
sg_indx = i;
|
sg_indx = i;
|
||||||
|
|
||||||
|
@ -624,7 +624,7 @@ static int aac_send_raw_srb(struct aac_dev* dev, void __user * arg)
|
||||||
rcode = -ENOMEM;
|
rcode = -ENOMEM;
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
sg_user[i] = (void __user *)upsg->sg[i].addr;
|
sg_user[i] = (void __user *)(long)upsg->sg[i].addr;
|
||||||
sg_list[i] = p; // save so we can clean up later
|
sg_list[i] = p; // save so we can clean up later
|
||||||
sg_indx = i;
|
sg_indx = i;
|
||||||
|
|
||||||
|
|
|
@ -116,6 +116,10 @@ static int aac_alloc_comm(struct aac_dev *dev, void **commaddr, unsigned long co
|
||||||
}
|
}
|
||||||
|
|
||||||
init->InitFlags = 0;
|
init->InitFlags = 0;
|
||||||
|
if (dev->new_comm_interface) {
|
||||||
|
init->InitFlags = cpu_to_le32(INITFLAGS_NEW_COMM_SUPPORTED);
|
||||||
|
dprintk((KERN_WARNING"aacraid: New Comm Interface enabled\n"));
|
||||||
|
}
|
||||||
init->MaxIoCommands = cpu_to_le32(dev->scsi_host_ptr->can_queue + AAC_NUM_MGT_FIB);
|
init->MaxIoCommands = cpu_to_le32(dev->scsi_host_ptr->can_queue + AAC_NUM_MGT_FIB);
|
||||||
init->MaxIoSize = cpu_to_le32(dev->scsi_host_ptr->max_sectors << 9);
|
init->MaxIoSize = cpu_to_le32(dev->scsi_host_ptr->max_sectors << 9);
|
||||||
init->MaxFibSize = cpu_to_le32(dev->max_fib_size);
|
init->MaxFibSize = cpu_to_le32(dev->max_fib_size);
|
||||||
|
@ -315,12 +319,33 @@ struct aac_dev *aac_init_adapter(struct aac_dev *dev)
|
||||||
- sizeof(struct aac_fibhdr)
|
- sizeof(struct aac_fibhdr)
|
||||||
- sizeof(struct aac_write) + sizeof(struct sgentry))
|
- sizeof(struct aac_write) + sizeof(struct sgentry))
|
||||||
/ sizeof(struct sgentry);
|
/ sizeof(struct sgentry);
|
||||||
|
dev->new_comm_interface = 0;
|
||||||
dev->raw_io_64 = 0;
|
dev->raw_io_64 = 0;
|
||||||
if ((!aac_adapter_sync_cmd(dev, GET_ADAPTER_PROPERTIES,
|
if ((!aac_adapter_sync_cmd(dev, GET_ADAPTER_PROPERTIES,
|
||||||
0, 0, 0, 0, 0, 0, status+0, status+1, status+2, NULL, NULL)) &&
|
0, 0, 0, 0, 0, 0, status+0, status+1, status+2, NULL, NULL)) &&
|
||||||
(status[0] == 0x00000001)) {
|
(status[0] == 0x00000001)) {
|
||||||
if (status[1] & AAC_OPT_NEW_COMM_64)
|
if (status[1] & AAC_OPT_NEW_COMM_64)
|
||||||
dev->raw_io_64 = 1;
|
dev->raw_io_64 = 1;
|
||||||
|
if (status[1] & AAC_OPT_NEW_COMM)
|
||||||
|
dev->new_comm_interface = dev->a_ops.adapter_send != 0;
|
||||||
|
if (dev->new_comm_interface && (status[2] > dev->base_size)) {
|
||||||
|
iounmap(dev->regs.sa);
|
||||||
|
dev->base_size = status[2];
|
||||||
|
dprintk((KERN_DEBUG "ioremap(%lx,%d)\n",
|
||||||
|
host->base, status[2]));
|
||||||
|
dev->regs.sa = ioremap(host->base, status[2]);
|
||||||
|
if (dev->regs.sa == NULL) {
|
||||||
|
/* remap failed, go back ... */
|
||||||
|
dev->new_comm_interface = 0;
|
||||||
|
dev->regs.sa = ioremap(host->base,
|
||||||
|
AAC_MIN_FOOTPRINT_SIZE);
|
||||||
|
if (dev->regs.sa == NULL) {
|
||||||
|
printk(KERN_WARNING
|
||||||
|
"aacraid: unable to map adapter.\n");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if ((!aac_adapter_sync_cmd(dev, GET_COMM_PREFERRED_SETTINGS,
|
if ((!aac_adapter_sync_cmd(dev, GET_COMM_PREFERRED_SETTINGS,
|
||||||
0, 0, 0, 0, 0, 0,
|
0, 0, 0, 0, 0, 0,
|
||||||
|
|
|
@ -212,7 +212,7 @@ void fib_init(struct fib *fibptr)
|
||||||
hw_fib->header.StructType = FIB_MAGIC;
|
hw_fib->header.StructType = FIB_MAGIC;
|
||||||
hw_fib->header.Size = cpu_to_le16(fibptr->dev->max_fib_size);
|
hw_fib->header.Size = cpu_to_le16(fibptr->dev->max_fib_size);
|
||||||
hw_fib->header.XferState = cpu_to_le32(HostOwned | FibInitialized | FibEmpty | FastResponseCapable);
|
hw_fib->header.XferState = cpu_to_le32(HostOwned | FibInitialized | FibEmpty | FastResponseCapable);
|
||||||
hw_fib->header.SenderFibAddress = cpu_to_le32(fibptr->hw_fib_pa);
|
hw_fib->header.SenderFibAddress = 0; /* Filled in later if needed */
|
||||||
hw_fib->header.ReceiverFibAddress = cpu_to_le32(fibptr->hw_fib_pa);
|
hw_fib->header.ReceiverFibAddress = cpu_to_le32(fibptr->hw_fib_pa);
|
||||||
hw_fib->header.SenderSize = cpu_to_le16(fibptr->dev->max_fib_size);
|
hw_fib->header.SenderSize = cpu_to_le16(fibptr->dev->max_fib_size);
|
||||||
}
|
}
|
||||||
|
@ -380,9 +380,7 @@ static int aac_queue_get(struct aac_dev * dev, u32 * index, u32 qid, struct hw_f
|
||||||
|
|
||||||
int fib_send(u16 command, struct fib * fibptr, unsigned long size, int priority, int wait, int reply, fib_callback callback, void * callback_data)
|
int fib_send(u16 command, struct fib * fibptr, unsigned long size, int priority, int wait, int reply, fib_callback callback, void * callback_data)
|
||||||
{
|
{
|
||||||
u32 index;
|
|
||||||
struct aac_dev * dev = fibptr->dev;
|
struct aac_dev * dev = fibptr->dev;
|
||||||
unsigned long nointr = 0;
|
|
||||||
struct hw_fib * hw_fib = fibptr->hw_fib;
|
struct hw_fib * hw_fib = fibptr->hw_fib;
|
||||||
struct aac_queue * q;
|
struct aac_queue * q;
|
||||||
unsigned long flags = 0;
|
unsigned long flags = 0;
|
||||||
|
@ -417,7 +415,7 @@ int fib_send(u16 command, struct fib * fibptr, unsigned long size, int priority
|
||||||
* Map the fib into 32bits by using the fib number
|
* Map the fib into 32bits by using the fib number
|
||||||
*/
|
*/
|
||||||
|
|
||||||
hw_fib->header.SenderFibAddress = cpu_to_le32(((u32)(fibptr-dev->fibs)) << 1);
|
hw_fib->header.SenderFibAddress = cpu_to_le32(((u32)(fibptr - dev->fibs)) << 2);
|
||||||
hw_fib->header.SenderData = (u32)(fibptr - dev->fibs);
|
hw_fib->header.SenderData = (u32)(fibptr - dev->fibs);
|
||||||
/*
|
/*
|
||||||
* Set FIB state to indicate where it came from and if we want a
|
* Set FIB state to indicate where it came from and if we want a
|
||||||
|
@ -456,10 +454,10 @@ int fib_send(u16 command, struct fib * fibptr, unsigned long size, int priority
|
||||||
|
|
||||||
FIB_COUNTER_INCREMENT(aac_config.FibsSent);
|
FIB_COUNTER_INCREMENT(aac_config.FibsSent);
|
||||||
|
|
||||||
dprintk((KERN_DEBUG "fib_send: inserting a queue entry at index %d.\n",index));
|
|
||||||
dprintk((KERN_DEBUG "Fib contents:.\n"));
|
dprintk((KERN_DEBUG "Fib contents:.\n"));
|
||||||
dprintk((KERN_DEBUG " Command = %d.\n", hw_fib->header.Command));
|
dprintk((KERN_DEBUG " Command = %d.\n", le32_to_cpu(hw_fib->header.Command)));
|
||||||
dprintk((KERN_DEBUG " XferState = %x.\n", hw_fib->header.XferState));
|
dprintk((KERN_DEBUG " SubCommand = %d.\n", le32_to_cpu(((struct aac_query_mount *)fib_data(fibptr))->command)));
|
||||||
|
dprintk((KERN_DEBUG " XferState = %x.\n", le32_to_cpu(hw_fib->header.XferState)));
|
||||||
dprintk((KERN_DEBUG " hw_fib va being sent=%p\n",fibptr->hw_fib));
|
dprintk((KERN_DEBUG " hw_fib va being sent=%p\n",fibptr->hw_fib));
|
||||||
dprintk((KERN_DEBUG " hw_fib pa being sent=%lx\n",(ulong)fibptr->hw_fib_pa));
|
dprintk((KERN_DEBUG " hw_fib pa being sent=%lx\n",(ulong)fibptr->hw_fib_pa));
|
||||||
dprintk((KERN_DEBUG " fib being sent=%p\n",fibptr));
|
dprintk((KERN_DEBUG " fib being sent=%p\n",fibptr));
|
||||||
|
@ -469,14 +467,37 @@ int fib_send(u16 command, struct fib * fibptr, unsigned long size, int priority
|
||||||
if(wait)
|
if(wait)
|
||||||
spin_lock_irqsave(&fibptr->event_lock, flags);
|
spin_lock_irqsave(&fibptr->event_lock, flags);
|
||||||
spin_lock_irqsave(q->lock, qflags);
|
spin_lock_irqsave(q->lock, qflags);
|
||||||
aac_queue_get( dev, &index, AdapNormCmdQueue, hw_fib, 1, fibptr, &nointr);
|
if (dev->new_comm_interface) {
|
||||||
|
unsigned long count = 10000000L; /* 50 seconds */
|
||||||
|
list_add_tail(&fibptr->queue, &q->pendingq);
|
||||||
|
q->numpending++;
|
||||||
|
spin_unlock_irqrestore(q->lock, qflags);
|
||||||
|
while (aac_adapter_send(fibptr) != 0) {
|
||||||
|
if (--count == 0) {
|
||||||
|
if (wait)
|
||||||
|
spin_unlock_irqrestore(&fibptr->event_lock, flags);
|
||||||
|
spin_lock_irqsave(q->lock, qflags);
|
||||||
|
q->numpending--;
|
||||||
|
list_del(&fibptr->queue);
|
||||||
|
spin_unlock_irqrestore(q->lock, qflags);
|
||||||
|
return -ETIMEDOUT;
|
||||||
|
}
|
||||||
|
udelay(5);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
u32 index;
|
||||||
|
unsigned long nointr = 0;
|
||||||
|
aac_queue_get( dev, &index, AdapNormCmdQueue, hw_fib, 1, fibptr, &nointr);
|
||||||
|
|
||||||
|
list_add_tail(&fibptr->queue, &q->pendingq);
|
||||||
|
q->numpending++;
|
||||||
|
*(q->headers.producer) = cpu_to_le32(index + 1);
|
||||||
|
spin_unlock_irqrestore(q->lock, qflags);
|
||||||
|
dprintk((KERN_DEBUG "fib_send: inserting a queue entry at index %d.\n",index));
|
||||||
|
if (!(nointr & aac_config.irq_mod))
|
||||||
|
aac_adapter_notify(dev, AdapNormCmdQueue);
|
||||||
|
}
|
||||||
|
|
||||||
list_add_tail(&fibptr->queue, &q->pendingq);
|
|
||||||
q->numpending++;
|
|
||||||
*(q->headers.producer) = cpu_to_le32(index + 1);
|
|
||||||
spin_unlock_irqrestore(q->lock, qflags);
|
|
||||||
if (!(nointr & aac_config.irq_mod))
|
|
||||||
aac_adapter_notify(dev, AdapNormCmdQueue);
|
|
||||||
/*
|
/*
|
||||||
* If the caller wanted us to wait for response wait now.
|
* If the caller wanted us to wait for response wait now.
|
||||||
*/
|
*/
|
||||||
|
@ -492,7 +513,6 @@ int fib_send(u16 command, struct fib * fibptr, unsigned long size, int priority
|
||||||
* hardware failure has occurred.
|
* hardware failure has occurred.
|
||||||
*/
|
*/
|
||||||
unsigned long count = 36000000L; /* 3 minutes */
|
unsigned long count = 36000000L; /* 3 minutes */
|
||||||
unsigned long qflags;
|
|
||||||
while (down_trylock(&fibptr->event_wait)) {
|
while (down_trylock(&fibptr->event_wait)) {
|
||||||
if (--count == 0) {
|
if (--count == 0) {
|
||||||
spin_lock_irqsave(q->lock, qflags);
|
spin_lock_irqsave(q->lock, qflags);
|
||||||
|
@ -621,12 +641,16 @@ int fib_adapter_complete(struct fib * fibptr, unsigned short size)
|
||||||
unsigned long qflags;
|
unsigned long qflags;
|
||||||
|
|
||||||
if (hw_fib->header.XferState == 0) {
|
if (hw_fib->header.XferState == 0) {
|
||||||
|
if (dev->new_comm_interface)
|
||||||
|
kfree (hw_fib);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
* If we plan to do anything check the structure type first.
|
* If we plan to do anything check the structure type first.
|
||||||
*/
|
*/
|
||||||
if ( hw_fib->header.StructType != FIB_MAGIC ) {
|
if ( hw_fib->header.StructType != FIB_MAGIC ) {
|
||||||
|
if (dev->new_comm_interface)
|
||||||
|
kfree (hw_fib);
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
|
@ -637,21 +661,25 @@ int fib_adapter_complete(struct fib * fibptr, unsigned short size)
|
||||||
* send the completed cdb to the adapter.
|
* send the completed cdb to the adapter.
|
||||||
*/
|
*/
|
||||||
if (hw_fib->header.XferState & cpu_to_le32(SentFromAdapter)) {
|
if (hw_fib->header.XferState & cpu_to_le32(SentFromAdapter)) {
|
||||||
u32 index;
|
if (dev->new_comm_interface) {
|
||||||
hw_fib->header.XferState |= cpu_to_le32(HostProcessed);
|
kfree (hw_fib);
|
||||||
if (size) {
|
} else {
|
||||||
size += sizeof(struct aac_fibhdr);
|
u32 index;
|
||||||
if (size > le16_to_cpu(hw_fib->header.SenderSize))
|
hw_fib->header.XferState |= cpu_to_le32(HostProcessed);
|
||||||
return -EMSGSIZE;
|
if (size) {
|
||||||
hw_fib->header.Size = cpu_to_le16(size);
|
size += sizeof(struct aac_fibhdr);
|
||||||
|
if (size > le16_to_cpu(hw_fib->header.SenderSize))
|
||||||
|
return -EMSGSIZE;
|
||||||
|
hw_fib->header.Size = cpu_to_le16(size);
|
||||||
|
}
|
||||||
|
q = &dev->queues->queue[AdapNormRespQueue];
|
||||||
|
spin_lock_irqsave(q->lock, qflags);
|
||||||
|
aac_queue_get(dev, &index, AdapNormRespQueue, hw_fib, 1, NULL, &nointr);
|
||||||
|
*(q->headers.producer) = cpu_to_le32(index + 1);
|
||||||
|
spin_unlock_irqrestore(q->lock, qflags);
|
||||||
|
if (!(nointr & (int)aac_config.irq_mod))
|
||||||
|
aac_adapter_notify(dev, AdapNormRespQueue);
|
||||||
}
|
}
|
||||||
q = &dev->queues->queue[AdapNormRespQueue];
|
|
||||||
spin_lock_irqsave(q->lock, qflags);
|
|
||||||
aac_queue_get(dev, &index, AdapNormRespQueue, hw_fib, 1, NULL, &nointr);
|
|
||||||
*(q->headers.producer) = cpu_to_le32(index + 1);
|
|
||||||
spin_unlock_irqrestore(q->lock, qflags);
|
|
||||||
if (!(nointr & (int)aac_config.irq_mod))
|
|
||||||
aac_adapter_notify(dev, AdapNormRespQueue);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|
|
@ -73,7 +73,7 @@ unsigned int aac_response_normal(struct aac_queue * q)
|
||||||
int fast;
|
int fast;
|
||||||
u32 index = le32_to_cpu(entry->addr);
|
u32 index = le32_to_cpu(entry->addr);
|
||||||
fast = index & 0x01;
|
fast = index & 0x01;
|
||||||
fib = &dev->fibs[index >> 1];
|
fib = &dev->fibs[index >> 2];
|
||||||
hwfib = fib->hw_fib;
|
hwfib = fib->hw_fib;
|
||||||
|
|
||||||
aac_consumer_free(dev, q, HostNormRespQueue);
|
aac_consumer_free(dev, q, HostNormRespQueue);
|
||||||
|
@ -213,3 +213,116 @@ unsigned int aac_command_normal(struct aac_queue *q)
|
||||||
spin_unlock_irqrestore(q->lock, flags);
|
spin_unlock_irqrestore(q->lock, flags);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* aac_intr_normal - Handle command replies
|
||||||
|
* @dev: Device
|
||||||
|
* @index: completion reference
|
||||||
|
*
|
||||||
|
* This DPC routine will be run when the adapter interrupts us to let us
|
||||||
|
* know there is a response on our normal priority queue. We will pull off
|
||||||
|
* all QE there are and wake up all the waiters before exiting.
|
||||||
|
*/
|
||||||
|
|
||||||
|
unsigned int aac_intr_normal(struct aac_dev * dev, u32 Index)
|
||||||
|
{
|
||||||
|
u32 index = le32_to_cpu(Index);
|
||||||
|
|
||||||
|
dprintk((KERN_INFO "aac_intr_normal(%p,%x)\n", dev, Index));
|
||||||
|
if ((index & 0x00000002L)) {
|
||||||
|
struct hw_fib * hw_fib;
|
||||||
|
struct fib * fib;
|
||||||
|
struct aac_queue *q = &dev->queues->queue[HostNormCmdQueue];
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
if (index == 0xFFFFFFFEL) /* Special Case */
|
||||||
|
return 0; /* Do nothing */
|
||||||
|
/*
|
||||||
|
* Allocate a FIB. For non queued stuff we can just use
|
||||||
|
* the stack so we are happy. We need a fib object in order to
|
||||||
|
* manage the linked lists.
|
||||||
|
*/
|
||||||
|
if ((!dev->aif_thread)
|
||||||
|
|| (!(fib = kmalloc(sizeof(struct fib),GFP_ATOMIC))))
|
||||||
|
return 1;
|
||||||
|
if (!(hw_fib = kmalloc(sizeof(struct hw_fib),GFP_ATOMIC))) {
|
||||||
|
kfree (fib);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
memset(hw_fib, 0, sizeof(struct hw_fib));
|
||||||
|
memcpy(hw_fib, (struct hw_fib *)(((unsigned long)(dev->regs.sa)) + (index & ~0x00000002L)), sizeof(struct hw_fib));
|
||||||
|
memset(fib, 0, sizeof(struct fib));
|
||||||
|
INIT_LIST_HEAD(&fib->fiblink);
|
||||||
|
fib->type = FSAFS_NTC_FIB_CONTEXT;
|
||||||
|
fib->size = sizeof(struct fib);
|
||||||
|
fib->hw_fib = hw_fib;
|
||||||
|
fib->data = hw_fib->data;
|
||||||
|
fib->dev = dev;
|
||||||
|
|
||||||
|
spin_lock_irqsave(q->lock, flags);
|
||||||
|
list_add_tail(&fib->fiblink, &q->cmdq);
|
||||||
|
wake_up_interruptible(&q->cmdready);
|
||||||
|
spin_unlock_irqrestore(q->lock, flags);
|
||||||
|
return 1;
|
||||||
|
} else {
|
||||||
|
int fast = index & 0x01;
|
||||||
|
struct fib * fib = &dev->fibs[index >> 2];
|
||||||
|
struct hw_fib * hwfib = fib->hw_fib;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Remove this fib from the Outstanding I/O queue.
|
||||||
|
* But only if it has not already been timed out.
|
||||||
|
*
|
||||||
|
* If the fib has been timed out already, then just
|
||||||
|
* continue. The caller has already been notified that
|
||||||
|
* the fib timed out.
|
||||||
|
*/
|
||||||
|
if ((fib->flags & FIB_CONTEXT_FLAG_TIMED_OUT)) {
|
||||||
|
printk(KERN_WARNING "aacraid: FIB timeout (%x).\n", fib->flags);
|
||||||
|
printk(KERN_DEBUG"aacraid: hwfib=%p index=%i fib=%p\n",hwfib, hwfib->header.SenderData,fib);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
list_del(&fib->queue);
|
||||||
|
dev->queues->queue[AdapNormCmdQueue].numpending--;
|
||||||
|
|
||||||
|
if (fast) {
|
||||||
|
/*
|
||||||
|
* Doctor the fib
|
||||||
|
*/
|
||||||
|
*(__le32 *)hwfib->data = cpu_to_le32(ST_OK);
|
||||||
|
hwfib->header.XferState |= cpu_to_le32(AdapterProcessed);
|
||||||
|
}
|
||||||
|
|
||||||
|
FIB_COUNTER_INCREMENT(aac_config.FibRecved);
|
||||||
|
|
||||||
|
if (hwfib->header.Command == cpu_to_le16(NuFileSystem))
|
||||||
|
{
|
||||||
|
u32 *pstatus = (u32 *)hwfib->data;
|
||||||
|
if (*pstatus & cpu_to_le32(0xffff0000))
|
||||||
|
*pstatus = cpu_to_le32(ST_OK);
|
||||||
|
}
|
||||||
|
if (hwfib->header.XferState & cpu_to_le32(NoResponseExpected | Async))
|
||||||
|
{
|
||||||
|
if (hwfib->header.XferState & cpu_to_le32(NoResponseExpected))
|
||||||
|
FIB_COUNTER_INCREMENT(aac_config.NoResponseRecved);
|
||||||
|
else
|
||||||
|
FIB_COUNTER_INCREMENT(aac_config.AsyncRecved);
|
||||||
|
/*
|
||||||
|
* NOTE: we cannot touch the fib after this
|
||||||
|
* call, because it may have been deallocated.
|
||||||
|
*/
|
||||||
|
fib->callback(fib->callback_data, fib);
|
||||||
|
} else {
|
||||||
|
unsigned long flagv;
|
||||||
|
dprintk((KERN_INFO "event_wait up\n"));
|
||||||
|
spin_lock_irqsave(&fib->event_lock, flagv);
|
||||||
|
fib->done = 1;
|
||||||
|
up(&fib->event_wait);
|
||||||
|
spin_unlock_irqrestore(&fib->event_lock, flagv);
|
||||||
|
FIB_COUNTER_INCREMENT(aac_config.NormalRecved);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -752,8 +752,8 @@ static int __devinit aac_probe_one(struct pci_dev *pdev,
|
||||||
if (error)
|
if (error)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
if (pci_set_dma_mask(pdev, 0xFFFFFFFFULL) ||
|
if (pci_set_dma_mask(pdev, DMA_32BIT_MASK) ||
|
||||||
pci_set_consistent_dma_mask(pdev, 0xFFFFFFFFULL))
|
pci_set_consistent_dma_mask(pdev, DMA_32BIT_MASK))
|
||||||
goto out;
|
goto out;
|
||||||
/*
|
/*
|
||||||
* If the quirk31 bit is set, the adapter needs adapter
|
* If the quirk31 bit is set, the adapter needs adapter
|
||||||
|
@ -788,8 +788,29 @@ static int __devinit aac_probe_one(struct pci_dev *pdev,
|
||||||
goto out_free_host;
|
goto out_free_host;
|
||||||
spin_lock_init(&aac->fib_lock);
|
spin_lock_init(&aac->fib_lock);
|
||||||
|
|
||||||
if ((*aac_drivers[index].init)(aac))
|
/*
|
||||||
|
* Map in the registers from the adapter.
|
||||||
|
*/
|
||||||
|
aac->base_size = AAC_MIN_FOOTPRINT_SIZE;
|
||||||
|
if ((aac->regs.sa = ioremap(
|
||||||
|
(unsigned long)aac->scsi_host_ptr->base, AAC_MIN_FOOTPRINT_SIZE))
|
||||||
|
== NULL) {
|
||||||
|
printk(KERN_WARNING "%s: unable to map adapter.\n",
|
||||||
|
AAC_DRIVERNAME);
|
||||||
goto out_free_fibs;
|
goto out_free_fibs;
|
||||||
|
}
|
||||||
|
if ((*aac_drivers[index].init)(aac))
|
||||||
|
goto out_unmap;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Start any kernel threads needed
|
||||||
|
*/
|
||||||
|
aac->thread_pid = kernel_thread((int (*)(void *))aac_command_thread,
|
||||||
|
aac, 0);
|
||||||
|
if (aac->thread_pid < 0) {
|
||||||
|
printk(KERN_ERR "aacraid: Unable to create command thread.\n");
|
||||||
|
goto out_deinit;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If we had set a smaller DMA mask earlier, set it to 4gig
|
* If we had set a smaller DMA mask earlier, set it to 4gig
|
||||||
|
@ -797,9 +818,9 @@ static int __devinit aac_probe_one(struct pci_dev *pdev,
|
||||||
* address space.
|
* address space.
|
||||||
*/
|
*/
|
||||||
if (aac_drivers[index].quirks & AAC_QUIRK_31BIT)
|
if (aac_drivers[index].quirks & AAC_QUIRK_31BIT)
|
||||||
if (pci_set_dma_mask(pdev, 0xFFFFFFFFULL))
|
if (pci_set_dma_mask(pdev, DMA_32BIT_MASK))
|
||||||
goto out_free_fibs;
|
goto out_deinit;
|
||||||
|
|
||||||
aac->maximum_num_channels = aac_drivers[index].channels;
|
aac->maximum_num_channels = aac_drivers[index].channels;
|
||||||
error = aac_get_adapter_info(aac);
|
error = aac_get_adapter_info(aac);
|
||||||
if (error < 0)
|
if (error < 0)
|
||||||
|
@ -866,10 +887,11 @@ static int __devinit aac_probe_one(struct pci_dev *pdev,
|
||||||
|
|
||||||
aac_send_shutdown(aac);
|
aac_send_shutdown(aac);
|
||||||
aac_adapter_disable_int(aac);
|
aac_adapter_disable_int(aac);
|
||||||
|
free_irq(pdev->irq, aac);
|
||||||
|
out_unmap:
|
||||||
fib_map_free(aac);
|
fib_map_free(aac);
|
||||||
pci_free_consistent(aac->pdev, aac->comm_size, aac->comm_addr, aac->comm_phys);
|
pci_free_consistent(aac->pdev, aac->comm_size, aac->comm_addr, aac->comm_phys);
|
||||||
kfree(aac->queues);
|
kfree(aac->queues);
|
||||||
free_irq(pdev->irq, aac);
|
|
||||||
iounmap(aac->regs.sa);
|
iounmap(aac->regs.sa);
|
||||||
out_free_fibs:
|
out_free_fibs:
|
||||||
kfree(aac->fibs);
|
kfree(aac->fibs);
|
||||||
|
@ -910,6 +932,7 @@ static void __devexit aac_remove_one(struct pci_dev *pdev)
|
||||||
iounmap(aac->regs.sa);
|
iounmap(aac->regs.sa);
|
||||||
|
|
||||||
kfree(aac->fibs);
|
kfree(aac->fibs);
|
||||||
|
kfree(aac->fsa_dev);
|
||||||
|
|
||||||
list_del(&aac->entry);
|
list_del(&aac->entry);
|
||||||
scsi_host_put(shost);
|
scsi_host_put(shost);
|
||||||
|
|
|
@ -49,40 +49,57 @@
|
||||||
static irqreturn_t aac_rkt_intr(int irq, void *dev_id, struct pt_regs *regs)
|
static irqreturn_t aac_rkt_intr(int irq, void *dev_id, struct pt_regs *regs)
|
||||||
{
|
{
|
||||||
struct aac_dev *dev = dev_id;
|
struct aac_dev *dev = dev_id;
|
||||||
unsigned long bellbits;
|
|
||||||
u8 intstat, mask;
|
if (dev->new_comm_interface) {
|
||||||
intstat = rkt_readb(dev, MUnit.OISR);
|
u32 Index = rkt_readl(dev, MUnit.OutboundQueue);
|
||||||
/*
|
if (Index == 0xFFFFFFFFL)
|
||||||
* Read mask and invert because drawbridge is reversed.
|
Index = rkt_readl(dev, MUnit.OutboundQueue);
|
||||||
* This allows us to only service interrupts that have
|
if (Index != 0xFFFFFFFFL) {
|
||||||
* been enabled.
|
do {
|
||||||
*/
|
if (aac_intr_normal(dev, Index)) {
|
||||||
mask = ~(dev->OIMR);
|
rkt_writel(dev, MUnit.OutboundQueue, Index);
|
||||||
/* Check to see if this is our interrupt. If it isn't just return */
|
rkt_writel(dev, MUnit.ODR, DoorBellAdapterNormRespReady);
|
||||||
if (intstat & mask)
|
}
|
||||||
{
|
Index = rkt_readl(dev, MUnit.OutboundQueue);
|
||||||
bellbits = rkt_readl(dev, OutboundDoorbellReg);
|
} while (Index != 0xFFFFFFFFL);
|
||||||
if (bellbits & DoorBellPrintfReady) {
|
return IRQ_HANDLED;
|
||||||
aac_printf(dev, rkt_readl(dev, IndexRegs.Mailbox[5]));
|
|
||||||
rkt_writel(dev, MUnit.ODR,DoorBellPrintfReady);
|
|
||||||
rkt_writel(dev, InboundDoorbellReg,DoorBellPrintfDone);
|
|
||||||
}
|
}
|
||||||
else if (bellbits & DoorBellAdapterNormCmdReady) {
|
} else {
|
||||||
rkt_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdReady);
|
unsigned long bellbits;
|
||||||
aac_command_normal(&dev->queues->queue[HostNormCmdQueue]);
|
u8 intstat;
|
||||||
|
intstat = rkt_readb(dev, MUnit.OISR);
|
||||||
|
/*
|
||||||
|
* Read mask and invert because drawbridge is reversed.
|
||||||
|
* This allows us to only service interrupts that have
|
||||||
|
* been enabled.
|
||||||
|
* Check to see if this is our interrupt. If it isn't just return
|
||||||
|
*/
|
||||||
|
if (intstat & ~(dev->OIMR))
|
||||||
|
{
|
||||||
|
bellbits = rkt_readl(dev, OutboundDoorbellReg);
|
||||||
|
if (bellbits & DoorBellPrintfReady) {
|
||||||
|
aac_printf(dev, rkt_readl (dev, IndexRegs.Mailbox[5]));
|
||||||
|
rkt_writel(dev, MUnit.ODR,DoorBellPrintfReady);
|
||||||
|
rkt_writel(dev, InboundDoorbellReg,DoorBellPrintfDone);
|
||||||
|
}
|
||||||
|
else if (bellbits & DoorBellAdapterNormCmdReady) {
|
||||||
|
rkt_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdReady);
|
||||||
|
aac_command_normal(&dev->queues->queue[HostNormCmdQueue]);
|
||||||
|
// rkt_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdReady);
|
||||||
|
}
|
||||||
|
else if (bellbits & DoorBellAdapterNormRespReady) {
|
||||||
|
rkt_writel(dev, MUnit.ODR,DoorBellAdapterNormRespReady);
|
||||||
|
aac_response_normal(&dev->queues->queue[HostNormRespQueue]);
|
||||||
|
}
|
||||||
|
else if (bellbits & DoorBellAdapterNormCmdNotFull) {
|
||||||
|
rkt_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdNotFull);
|
||||||
|
}
|
||||||
|
else if (bellbits & DoorBellAdapterNormRespNotFull) {
|
||||||
|
rkt_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdNotFull);
|
||||||
|
rkt_writel(dev, MUnit.ODR, DoorBellAdapterNormRespNotFull);
|
||||||
|
}
|
||||||
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
else if (bellbits & DoorBellAdapterNormRespReady) {
|
|
||||||
aac_response_normal(&dev->queues->queue[HostNormRespQueue]);
|
|
||||||
rkt_writel(dev, MUnit.ODR,DoorBellAdapterNormRespReady);
|
|
||||||
}
|
|
||||||
else if (bellbits & DoorBellAdapterNormCmdNotFull) {
|
|
||||||
rkt_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdNotFull);
|
|
||||||
}
|
|
||||||
else if (bellbits & DoorBellAdapterNormRespNotFull) {
|
|
||||||
rkt_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdNotFull);
|
|
||||||
rkt_writel(dev, MUnit.ODR, DoorBellAdapterNormRespNotFull);
|
|
||||||
}
|
|
||||||
return IRQ_HANDLED;
|
|
||||||
}
|
}
|
||||||
return IRQ_NONE;
|
return IRQ_NONE;
|
||||||
}
|
}
|
||||||
|
@ -173,7 +190,10 @@ static int rkt_sync_cmd(struct aac_dev *dev, u32 command,
|
||||||
/*
|
/*
|
||||||
* Restore interrupt mask even though we timed out
|
* Restore interrupt mask even though we timed out
|
||||||
*/
|
*/
|
||||||
rkt_writeb(dev, MUnit.OIMR, dev->OIMR = 0xfb);
|
if (dev->new_comm_interface)
|
||||||
|
rkt_writeb(dev, MUnit.OIMR, dev->OIMR = 0xf7);
|
||||||
|
else
|
||||||
|
rkt_writeb(dev, MUnit.OIMR, dev->OIMR = 0xfb);
|
||||||
return -ETIMEDOUT;
|
return -ETIMEDOUT;
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
|
@ -196,7 +216,10 @@ static int rkt_sync_cmd(struct aac_dev *dev, u32 command,
|
||||||
/*
|
/*
|
||||||
* Restore interrupt mask
|
* Restore interrupt mask
|
||||||
*/
|
*/
|
||||||
rkt_writeb(dev, MUnit.OIMR, dev->OIMR = 0xfb);
|
if (dev->new_comm_interface)
|
||||||
|
rkt_writeb(dev, MUnit.OIMR, dev->OIMR = 0xf7);
|
||||||
|
else
|
||||||
|
rkt_writeb(dev, MUnit.OIMR, dev->OIMR = 0xfb);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -268,15 +291,6 @@ static void aac_rkt_start_adapter(struct aac_dev *dev)
|
||||||
|
|
||||||
init = dev->init;
|
init = dev->init;
|
||||||
init->HostElapsedSeconds = cpu_to_le32(get_seconds());
|
init->HostElapsedSeconds = cpu_to_le32(get_seconds());
|
||||||
/*
|
|
||||||
* First clear out all interrupts. Then enable the one's that we
|
|
||||||
* can handle.
|
|
||||||
*/
|
|
||||||
rkt_writeb(dev, MUnit.OIMR, 0xff);
|
|
||||||
rkt_writel(dev, MUnit.ODR, 0xffffffff);
|
|
||||||
// rkt_writeb(dev, MUnit.OIMR, ~(u8)OUTBOUND_DOORBELL_INTERRUPT_MASK);
|
|
||||||
rkt_writeb(dev, MUnit.OIMR, dev->OIMR = 0xfb);
|
|
||||||
|
|
||||||
// We can only use a 32 bit address here
|
// We can only use a 32 bit address here
|
||||||
rkt_sync_cmd(dev, INIT_STRUCT_BASE_ADDRESS, (u32)(ulong)dev->init_pa,
|
rkt_sync_cmd(dev, INIT_STRUCT_BASE_ADDRESS, (u32)(ulong)dev->init_pa,
|
||||||
0, 0, 0, 0, 0, NULL, NULL, NULL, NULL, NULL);
|
0, 0, 0, 0, 0, NULL, NULL, NULL, NULL, NULL);
|
||||||
|
@ -349,6 +363,39 @@ static int aac_rkt_check_health(struct aac_dev *dev)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* aac_rkt_send
|
||||||
|
* @fib: fib to issue
|
||||||
|
*
|
||||||
|
* Will send a fib, returning 0 if successful.
|
||||||
|
*/
|
||||||
|
static int aac_rkt_send(struct fib * fib)
|
||||||
|
{
|
||||||
|
u64 addr = fib->hw_fib_pa;
|
||||||
|
struct aac_dev *dev = fib->dev;
|
||||||
|
volatile void __iomem *device = dev->regs.rkt;
|
||||||
|
u32 Index;
|
||||||
|
|
||||||
|
dprintk((KERN_DEBUG "%p->aac_rkt_send(%p->%llx)\n", dev, fib, addr));
|
||||||
|
Index = rkt_readl(dev, MUnit.InboundQueue);
|
||||||
|
if (Index == 0xFFFFFFFFL)
|
||||||
|
Index = rkt_readl(dev, MUnit.InboundQueue);
|
||||||
|
dprintk((KERN_DEBUG "Index = 0x%x\n", Index));
|
||||||
|
if (Index == 0xFFFFFFFFL)
|
||||||
|
return Index;
|
||||||
|
device += Index;
|
||||||
|
dprintk((KERN_DEBUG "entry = %x %x %u\n", (u32)(addr & 0xffffffff),
|
||||||
|
(u32)(addr >> 32), (u32)le16_to_cpu(fib->hw_fib->header.Size)));
|
||||||
|
writel((u32)(addr & 0xffffffff), device);
|
||||||
|
device += sizeof(u32);
|
||||||
|
writel((u32)(addr >> 32), device);
|
||||||
|
device += sizeof(u32);
|
||||||
|
writel(le16_to_cpu(fib->hw_fib->header.Size), device);
|
||||||
|
rkt_writel(dev, MUnit.InboundQueue, Index);
|
||||||
|
dprintk((KERN_DEBUG "aac_rkt_send - return 0\n"));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* aac_rkt_init - initialize an i960 based AAC card
|
* aac_rkt_init - initialize an i960 based AAC card
|
||||||
* @dev: device to configure
|
* @dev: device to configure
|
||||||
|
@ -369,13 +416,8 @@ int aac_rkt_init(struct aac_dev *dev)
|
||||||
name = dev->name;
|
name = dev->name;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Map in the registers from the adapter.
|
* Check to see if the board panic'd while booting.
|
||||||
*/
|
*/
|
||||||
if((dev->regs.rkt = ioremap((unsigned long)dev->scsi_host_ptr->base, 8192))==NULL)
|
|
||||||
{
|
|
||||||
printk(KERN_WARNING "aacraid: unable to map i960.\n" );
|
|
||||||
goto error_iounmap;
|
|
||||||
}
|
|
||||||
/*
|
/*
|
||||||
* Check to see if the board failed any self tests.
|
* Check to see if the board failed any self tests.
|
||||||
*/
|
*/
|
||||||
|
@ -426,6 +468,7 @@ int aac_rkt_init(struct aac_dev *dev)
|
||||||
dev->a_ops.adapter_notify = aac_rkt_notify_adapter;
|
dev->a_ops.adapter_notify = aac_rkt_notify_adapter;
|
||||||
dev->a_ops.adapter_sync_cmd = rkt_sync_cmd;
|
dev->a_ops.adapter_sync_cmd = rkt_sync_cmd;
|
||||||
dev->a_ops.adapter_check_health = aac_rkt_check_health;
|
dev->a_ops.adapter_check_health = aac_rkt_check_health;
|
||||||
|
dev->a_ops.adapter_send = aac_rkt_send;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* First clear out all interrupts. Then enable the one's that we
|
* First clear out all interrupts. Then enable the one's that we
|
||||||
|
@ -437,15 +480,24 @@ int aac_rkt_init(struct aac_dev *dev)
|
||||||
|
|
||||||
if (aac_init_adapter(dev) == NULL)
|
if (aac_init_adapter(dev) == NULL)
|
||||||
goto error_irq;
|
goto error_irq;
|
||||||
/*
|
if (dev->new_comm_interface) {
|
||||||
* Start any kernel threads needed
|
/*
|
||||||
*/
|
* FIB Setup has already been done, but we can minimize the
|
||||||
dev->thread_pid = kernel_thread((int (*)(void *))aac_command_thread, dev, 0);
|
* damage by at least ensuring the OS never issues more
|
||||||
if(dev->thread_pid < 0)
|
* commands than we can handle. The Rocket adapters currently
|
||||||
{
|
* can only handle 246 commands and 8 AIFs at the same time,
|
||||||
printk(KERN_ERR "aacraid: Unable to create rkt thread.\n");
|
* and in fact do notify us accordingly if we negotiate the
|
||||||
goto error_kfree;
|
* FIB size. The problem that causes us to add this check is
|
||||||
}
|
* to ensure that we do not overdo it with the adapter when a
|
||||||
|
* hard coded FIB override is being utilized. This special
|
||||||
|
* case warrants this half baked, but convenient, check here.
|
||||||
|
*/
|
||||||
|
if (dev->scsi_host_ptr->can_queue > (246 - AAC_NUM_MGT_FIB)) {
|
||||||
|
dev->init->MaxIoCommands = cpu_to_le32(246);
|
||||||
|
dev->scsi_host_ptr->can_queue = 246 - AAC_NUM_MGT_FIB;
|
||||||
|
}
|
||||||
|
rkt_writeb(dev, MUnit.OIMR, dev->OIMR = 0xf7);
|
||||||
|
}
|
||||||
/*
|
/*
|
||||||
* Tell the adapter that all is configured, and it can start
|
* Tell the adapter that all is configured, and it can start
|
||||||
* accepting requests
|
* accepting requests
|
||||||
|
@ -453,15 +505,11 @@ int aac_rkt_init(struct aac_dev *dev)
|
||||||
aac_rkt_start_adapter(dev);
|
aac_rkt_start_adapter(dev);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
error_kfree:
|
|
||||||
kfree(dev->queues);
|
|
||||||
|
|
||||||
error_irq:
|
error_irq:
|
||||||
rkt_writeb(dev, MUnit.OIMR, dev->OIMR = 0xff);
|
rkt_writeb(dev, MUnit.OIMR, dev->OIMR = 0xff);
|
||||||
free_irq(dev->scsi_host_ptr->irq, (void *)dev);
|
free_irq(dev->scsi_host_ptr->irq, (void *)dev);
|
||||||
|
|
||||||
error_iounmap:
|
error_iounmap:
|
||||||
iounmap(dev->regs.rkt);
|
|
||||||
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
|
@ -49,40 +49,57 @@
|
||||||
static irqreturn_t aac_rx_intr(int irq, void *dev_id, struct pt_regs *regs)
|
static irqreturn_t aac_rx_intr(int irq, void *dev_id, struct pt_regs *regs)
|
||||||
{
|
{
|
||||||
struct aac_dev *dev = dev_id;
|
struct aac_dev *dev = dev_id;
|
||||||
unsigned long bellbits;
|
|
||||||
u8 intstat, mask;
|
dprintk((KERN_DEBUG "aac_rx_intr(%d,%p,%p)\n", irq, dev_id, regs));
|
||||||
intstat = rx_readb(dev, MUnit.OISR);
|
if (dev->new_comm_interface) {
|
||||||
/*
|
u32 Index = rx_readl(dev, MUnit.OutboundQueue);
|
||||||
* Read mask and invert because drawbridge is reversed.
|
if (Index == 0xFFFFFFFFL)
|
||||||
* This allows us to only service interrupts that have
|
Index = rx_readl(dev, MUnit.OutboundQueue);
|
||||||
* been enabled.
|
if (Index != 0xFFFFFFFFL) {
|
||||||
*/
|
do {
|
||||||
mask = ~(dev->OIMR);
|
if (aac_intr_normal(dev, Index)) {
|
||||||
/* Check to see if this is our interrupt. If it isn't just return */
|
rx_writel(dev, MUnit.OutboundQueue, Index);
|
||||||
if (intstat & mask)
|
rx_writel(dev, MUnit.ODR, DoorBellAdapterNormRespReady);
|
||||||
{
|
}
|
||||||
bellbits = rx_readl(dev, OutboundDoorbellReg);
|
Index = rx_readl(dev, MUnit.OutboundQueue);
|
||||||
if (bellbits & DoorBellPrintfReady) {
|
} while (Index != 0xFFFFFFFFL);
|
||||||
aac_printf(dev, rx_readl(dev, IndexRegs.Mailbox[5]));
|
return IRQ_HANDLED;
|
||||||
rx_writel(dev, MUnit.ODR,DoorBellPrintfReady);
|
|
||||||
rx_writel(dev, InboundDoorbellReg,DoorBellPrintfDone);
|
|
||||||
}
|
}
|
||||||
else if (bellbits & DoorBellAdapterNormCmdReady) {
|
} else {
|
||||||
rx_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdReady);
|
unsigned long bellbits;
|
||||||
aac_command_normal(&dev->queues->queue[HostNormCmdQueue]);
|
u8 intstat;
|
||||||
|
intstat = rx_readb(dev, MUnit.OISR);
|
||||||
|
/*
|
||||||
|
* Read mask and invert because drawbridge is reversed.
|
||||||
|
* This allows us to only service interrupts that have
|
||||||
|
* been enabled.
|
||||||
|
* Check to see if this is our interrupt. If it isn't just return
|
||||||
|
*/
|
||||||
|
if (intstat & ~(dev->OIMR))
|
||||||
|
{
|
||||||
|
bellbits = rx_readl(dev, OutboundDoorbellReg);
|
||||||
|
if (bellbits & DoorBellPrintfReady) {
|
||||||
|
aac_printf(dev, rx_readl (dev, IndexRegs.Mailbox[5]));
|
||||||
|
rx_writel(dev, MUnit.ODR,DoorBellPrintfReady);
|
||||||
|
rx_writel(dev, InboundDoorbellReg,DoorBellPrintfDone);
|
||||||
|
}
|
||||||
|
else if (bellbits & DoorBellAdapterNormCmdReady) {
|
||||||
|
rx_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdReady);
|
||||||
|
aac_command_normal(&dev->queues->queue[HostNormCmdQueue]);
|
||||||
|
}
|
||||||
|
else if (bellbits & DoorBellAdapterNormRespReady) {
|
||||||
|
rx_writel(dev, MUnit.ODR,DoorBellAdapterNormRespReady);
|
||||||
|
aac_response_normal(&dev->queues->queue[HostNormRespQueue]);
|
||||||
|
}
|
||||||
|
else if (bellbits & DoorBellAdapterNormCmdNotFull) {
|
||||||
|
rx_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdNotFull);
|
||||||
|
}
|
||||||
|
else if (bellbits & DoorBellAdapterNormRespNotFull) {
|
||||||
|
rx_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdNotFull);
|
||||||
|
rx_writel(dev, MUnit.ODR, DoorBellAdapterNormRespNotFull);
|
||||||
|
}
|
||||||
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
else if (bellbits & DoorBellAdapterNormRespReady) {
|
|
||||||
aac_response_normal(&dev->queues->queue[HostNormRespQueue]);
|
|
||||||
rx_writel(dev, MUnit.ODR,DoorBellAdapterNormRespReady);
|
|
||||||
}
|
|
||||||
else if (bellbits & DoorBellAdapterNormCmdNotFull) {
|
|
||||||
rx_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdNotFull);
|
|
||||||
}
|
|
||||||
else if (bellbits & DoorBellAdapterNormRespNotFull) {
|
|
||||||
rx_writel(dev, MUnit.ODR, DoorBellAdapterNormCmdNotFull);
|
|
||||||
rx_writel(dev, MUnit.ODR, DoorBellAdapterNormRespNotFull);
|
|
||||||
}
|
|
||||||
return IRQ_HANDLED;
|
|
||||||
}
|
}
|
||||||
return IRQ_NONE;
|
return IRQ_NONE;
|
||||||
}
|
}
|
||||||
|
@ -173,7 +190,10 @@ static int rx_sync_cmd(struct aac_dev *dev, u32 command,
|
||||||
/*
|
/*
|
||||||
* Restore interrupt mask even though we timed out
|
* Restore interrupt mask even though we timed out
|
||||||
*/
|
*/
|
||||||
rx_writeb(dev, MUnit.OIMR, dev->OIMR &= 0xfb);
|
if (dev->new_comm_interface)
|
||||||
|
rx_writeb(dev, MUnit.OIMR, dev->OIMR = 0xf7);
|
||||||
|
else
|
||||||
|
rx_writeb(dev, MUnit.OIMR, dev->OIMR = 0xfb);
|
||||||
return -ETIMEDOUT;
|
return -ETIMEDOUT;
|
||||||
}
|
}
|
||||||
/*
|
/*
|
||||||
|
@ -196,7 +216,10 @@ static int rx_sync_cmd(struct aac_dev *dev, u32 command,
|
||||||
/*
|
/*
|
||||||
* Restore interrupt mask
|
* Restore interrupt mask
|
||||||
*/
|
*/
|
||||||
rx_writeb(dev, MUnit.OIMR, dev->OIMR &= 0xfb);
|
if (dev->new_comm_interface)
|
||||||
|
rx_writeb(dev, MUnit.OIMR, dev->OIMR = 0xf7);
|
||||||
|
else
|
||||||
|
rx_writeb(dev, MUnit.OIMR, dev->OIMR = 0xfb);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -267,15 +290,6 @@ static void aac_rx_start_adapter(struct aac_dev *dev)
|
||||||
|
|
||||||
init = dev->init;
|
init = dev->init;
|
||||||
init->HostElapsedSeconds = cpu_to_le32(get_seconds());
|
init->HostElapsedSeconds = cpu_to_le32(get_seconds());
|
||||||
/*
|
|
||||||
* First clear out all interrupts. Then enable the one's that we
|
|
||||||
* can handle.
|
|
||||||
*/
|
|
||||||
rx_writeb(dev, MUnit.OIMR, 0xff);
|
|
||||||
rx_writel(dev, MUnit.ODR, 0xffffffff);
|
|
||||||
// rx_writeb(dev, MUnit.OIMR, ~(u8)OUTBOUND_DOORBELL_INTERRUPT_MASK);
|
|
||||||
rx_writeb(dev, MUnit.OIMR, dev->OIMR = 0xfb);
|
|
||||||
|
|
||||||
// We can only use a 32 bit address here
|
// We can only use a 32 bit address here
|
||||||
rx_sync_cmd(dev, INIT_STRUCT_BASE_ADDRESS, (u32)(ulong)dev->init_pa,
|
rx_sync_cmd(dev, INIT_STRUCT_BASE_ADDRESS, (u32)(ulong)dev->init_pa,
|
||||||
0, 0, 0, 0, 0, NULL, NULL, NULL, NULL, NULL);
|
0, 0, 0, 0, 0, NULL, NULL, NULL, NULL, NULL);
|
||||||
|
@ -348,6 +362,39 @@ static int aac_rx_check_health(struct aac_dev *dev)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* aac_rx_send
|
||||||
|
* @fib: fib to issue
|
||||||
|
*
|
||||||
|
* Will send a fib, returning 0 if successful.
|
||||||
|
*/
|
||||||
|
static int aac_rx_send(struct fib * fib)
|
||||||
|
{
|
||||||
|
u64 addr = fib->hw_fib_pa;
|
||||||
|
struct aac_dev *dev = fib->dev;
|
||||||
|
volatile void __iomem *device = dev->regs.rx;
|
||||||
|
u32 Index;
|
||||||
|
|
||||||
|
dprintk((KERN_DEBUG "%p->aac_rx_send(%p->%llx)\n", dev, fib, addr));
|
||||||
|
Index = rx_readl(dev, MUnit.InboundQueue);
|
||||||
|
if (Index == 0xFFFFFFFFL)
|
||||||
|
Index = rx_readl(dev, MUnit.InboundQueue);
|
||||||
|
dprintk((KERN_DEBUG "Index = 0x%x\n", Index));
|
||||||
|
if (Index == 0xFFFFFFFFL)
|
||||||
|
return Index;
|
||||||
|
device += Index;
|
||||||
|
dprintk((KERN_DEBUG "entry = %x %x %u\n", (u32)(addr & 0xffffffff),
|
||||||
|
(u32)(addr >> 32), (u32)le16_to_cpu(fib->hw_fib->header.Size)));
|
||||||
|
writel((u32)(addr & 0xffffffff), device);
|
||||||
|
device += sizeof(u32);
|
||||||
|
writel((u32)(addr >> 32), device);
|
||||||
|
device += sizeof(u32);
|
||||||
|
writel(le16_to_cpu(fib->hw_fib->header.Size), device);
|
||||||
|
rx_writel(dev, MUnit.InboundQueue, Index);
|
||||||
|
dprintk((KERN_DEBUG "aac_rx_send - return 0\n"));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* aac_rx_init - initialize an i960 based AAC card
|
* aac_rx_init - initialize an i960 based AAC card
|
||||||
* @dev: device to configure
|
* @dev: device to configure
|
||||||
|
@ -368,13 +415,8 @@ int aac_rx_init(struct aac_dev *dev)
|
||||||
name = dev->name;
|
name = dev->name;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Map in the registers from the adapter.
|
* Check to see if the board panic'd while booting.
|
||||||
*/
|
*/
|
||||||
if((dev->regs.rx = ioremap((unsigned long)dev->scsi_host_ptr->base, 8192))==NULL)
|
|
||||||
{
|
|
||||||
printk(KERN_WARNING "aacraid: unable to map i960.\n" );
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
/*
|
/*
|
||||||
* Check to see if the board failed any self tests.
|
* Check to see if the board failed any self tests.
|
||||||
*/
|
*/
|
||||||
|
@ -426,6 +468,7 @@ int aac_rx_init(struct aac_dev *dev)
|
||||||
dev->a_ops.adapter_notify = aac_rx_notify_adapter;
|
dev->a_ops.adapter_notify = aac_rx_notify_adapter;
|
||||||
dev->a_ops.adapter_sync_cmd = rx_sync_cmd;
|
dev->a_ops.adapter_sync_cmd = rx_sync_cmd;
|
||||||
dev->a_ops.adapter_check_health = aac_rx_check_health;
|
dev->a_ops.adapter_check_health = aac_rx_check_health;
|
||||||
|
dev->a_ops.adapter_send = aac_rx_send;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* First clear out all interrupts. Then enable the one's that we
|
* First clear out all interrupts. Then enable the one's that we
|
||||||
|
@ -437,15 +480,9 @@ int aac_rx_init(struct aac_dev *dev)
|
||||||
|
|
||||||
if (aac_init_adapter(dev) == NULL)
|
if (aac_init_adapter(dev) == NULL)
|
||||||
goto error_irq;
|
goto error_irq;
|
||||||
/*
|
if (dev->new_comm_interface)
|
||||||
* Start any kernel threads needed
|
rx_writeb(dev, MUnit.OIMR, dev->OIMR = 0xf7);
|
||||||
*/
|
|
||||||
dev->thread_pid = kernel_thread((int (*)(void *))aac_command_thread, dev, 0);
|
|
||||||
if(dev->thread_pid < 0)
|
|
||||||
{
|
|
||||||
printk(KERN_ERR "aacraid: Unable to create rx thread.\n");
|
|
||||||
goto error_kfree;
|
|
||||||
}
|
|
||||||
/*
|
/*
|
||||||
* Tell the adapter that all is configured, and it can start
|
* Tell the adapter that all is configured, and it can start
|
||||||
* accepting requests
|
* accepting requests
|
||||||
|
@ -453,15 +490,11 @@ int aac_rx_init(struct aac_dev *dev)
|
||||||
aac_rx_start_adapter(dev);
|
aac_rx_start_adapter(dev);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
error_kfree:
|
|
||||||
kfree(dev->queues);
|
|
||||||
|
|
||||||
error_irq:
|
error_irq:
|
||||||
rx_writeb(dev, MUnit.OIMR, dev->OIMR = 0xff);
|
rx_writeb(dev, MUnit.OIMR, dev->OIMR = 0xff);
|
||||||
free_irq(dev->scsi_host_ptr->irq, (void *)dev);
|
free_irq(dev->scsi_host_ptr->irq, (void *)dev);
|
||||||
|
|
||||||
error_iounmap:
|
error_iounmap:
|
||||||
iounmap(dev->regs.rx);
|
|
||||||
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
|
@ -237,29 +237,16 @@ static void aac_sa_interrupt_adapter (struct aac_dev *dev)
|
||||||
|
|
||||||
static void aac_sa_start_adapter(struct aac_dev *dev)
|
static void aac_sa_start_adapter(struct aac_dev *dev)
|
||||||
{
|
{
|
||||||
u32 ret;
|
|
||||||
struct aac_init *init;
|
struct aac_init *init;
|
||||||
/*
|
/*
|
||||||
* Fill in the remaining pieces of the init.
|
* Fill in the remaining pieces of the init.
|
||||||
*/
|
*/
|
||||||
init = dev->init;
|
init = dev->init;
|
||||||
init->HostElapsedSeconds = cpu_to_le32(get_seconds());
|
init->HostElapsedSeconds = cpu_to_le32(get_seconds());
|
||||||
|
|
||||||
/*
|
|
||||||
* Tell the adapter we are back and up and running so it will scan its command
|
|
||||||
* queues and enable our interrupts
|
|
||||||
*/
|
|
||||||
dev->irq_mask = (PrintfReady | DOORBELL_1 | DOORBELL_2 | DOORBELL_3 | DOORBELL_4);
|
|
||||||
/*
|
|
||||||
* First clear out all interrupts. Then enable the one's that
|
|
||||||
* we can handle.
|
|
||||||
*/
|
|
||||||
sa_writew(dev, SaDbCSR.PRISETIRQMASK, 0xffff);
|
|
||||||
sa_writew(dev, SaDbCSR.PRICLEARIRQMASK, (PrintfReady | DOORBELL_1 | DOORBELL_2 | DOORBELL_3 | DOORBELL_4));
|
|
||||||
/* We can only use a 32 bit address here */
|
/* We can only use a 32 bit address here */
|
||||||
sa_sync_cmd(dev, INIT_STRUCT_BASE_ADDRESS,
|
sa_sync_cmd(dev, INIT_STRUCT_BASE_ADDRESS,
|
||||||
(u32)(ulong)dev->init_pa, 0, 0, 0, 0, 0,
|
(u32)(ulong)dev->init_pa, 0, 0, 0, 0, 0,
|
||||||
&ret, NULL, NULL, NULL, NULL);
|
NULL, NULL, NULL, NULL, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -313,15 +300,6 @@ int aac_sa_init(struct aac_dev *dev)
|
||||||
instance = dev->id;
|
instance = dev->id;
|
||||||
name = dev->name;
|
name = dev->name;
|
||||||
|
|
||||||
/*
|
|
||||||
* Map in the registers from the adapter.
|
|
||||||
*/
|
|
||||||
|
|
||||||
if((dev->regs.sa = ioremap((unsigned long)dev->scsi_host_ptr->base, 8192))==NULL)
|
|
||||||
{
|
|
||||||
printk(KERN_WARNING "aacraid: unable to map ARM.\n" );
|
|
||||||
goto error_iounmap;
|
|
||||||
}
|
|
||||||
/*
|
/*
|
||||||
* Check to see if the board failed any self tests.
|
* Check to see if the board failed any self tests.
|
||||||
*/
|
*/
|
||||||
|
@ -377,15 +355,6 @@ int aac_sa_init(struct aac_dev *dev)
|
||||||
if(aac_init_adapter(dev) == NULL)
|
if(aac_init_adapter(dev) == NULL)
|
||||||
goto error_irq;
|
goto error_irq;
|
||||||
|
|
||||||
/*
|
|
||||||
* Start any kernel threads needed
|
|
||||||
*/
|
|
||||||
dev->thread_pid = kernel_thread((int (*)(void *))aac_command_thread, dev, 0);
|
|
||||||
if (dev->thread_pid < 0) {
|
|
||||||
printk(KERN_ERR "aacraid: Unable to create command thread.\n");
|
|
||||||
goto error_kfree;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Tell the adapter that all is configure, and it can start
|
* Tell the adapter that all is configure, and it can start
|
||||||
* accepting requests
|
* accepting requests
|
||||||
|
@ -393,16 +362,11 @@ int aac_sa_init(struct aac_dev *dev)
|
||||||
aac_sa_start_adapter(dev);
|
aac_sa_start_adapter(dev);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
|
||||||
error_kfree:
|
|
||||||
kfree(dev->queues);
|
|
||||||
|
|
||||||
error_irq:
|
error_irq:
|
||||||
sa_writew(dev, SaDbCSR.PRISETIRQMASK, 0xffff);
|
sa_writew(dev, SaDbCSR.PRISETIRQMASK, 0xffff);
|
||||||
free_irq(dev->scsi_host_ptr->irq, (void *)dev);
|
free_irq(dev->scsi_host_ptr->irq, (void *)dev);
|
||||||
|
|
||||||
error_iounmap:
|
error_iounmap:
|
||||||
iounmap(dev->regs.sa);
|
|
||||||
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
|
@ -2921,8 +2921,7 @@ static void disp_enintr(struct Scsi_Host *shpnt)
|
||||||
*/
|
*/
|
||||||
static void show_command(Scsi_Cmnd *ptr)
|
static void show_command(Scsi_Cmnd *ptr)
|
||||||
{
|
{
|
||||||
printk(KERN_DEBUG "0x%08x: target=%d; lun=%d; cmnd=(",
|
scmd_printk(KERN_DEBUG, ptr, "%p: cmnd=(", ptr);
|
||||||
(unsigned int) ptr, ptr->device->id, ptr->device->lun);
|
|
||||||
|
|
||||||
__scsi_print_command(ptr->cmnd);
|
__scsi_print_command(ptr->cmnd);
|
||||||
|
|
||||||
|
|
|
@ -1405,7 +1405,8 @@ static int aha1542_dev_reset(Scsi_Cmnd * SCpnt)
|
||||||
*/
|
*/
|
||||||
aha1542_out(SCpnt->device->host->io_port, &ahacmd, 1);
|
aha1542_out(SCpnt->device->host->io_port, &ahacmd, 1);
|
||||||
|
|
||||||
printk(KERN_WARNING "aha1542.c: Trying device reset for target %d\n", SCpnt->device->id);
|
scmd_printk(KERN_WARNING, SCpnt,
|
||||||
|
"Trying device reset for target\n");
|
||||||
|
|
||||||
return SUCCESS;
|
return SUCCESS;
|
||||||
|
|
||||||
|
|
|
@ -347,7 +347,7 @@ static int aha1740_queuecommand(Scsi_Cmnd * SCpnt, void (*done)(Scsi_Cmnd *))
|
||||||
{
|
{
|
||||||
unchar direction;
|
unchar direction;
|
||||||
unchar *cmd = (unchar *) SCpnt->cmnd;
|
unchar *cmd = (unchar *) SCpnt->cmnd;
|
||||||
unchar target = SCpnt->device->id;
|
unchar target = scmd_id(SCpnt);
|
||||||
struct aha1740_hostdata *host = HOSTDATA(SCpnt->device->host);
|
struct aha1740_hostdata *host = HOSTDATA(SCpnt->device->host);
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
void *buff = SCpnt->request_buffer;
|
void *buff = SCpnt->request_buffer;
|
||||||
|
|
|
@ -52,6 +52,7 @@ static struct scsi_transport_template *ahd_linux_transport_template = NULL;
|
||||||
#include <linux/mm.h> /* For fetching system memory size */
|
#include <linux/mm.h> /* For fetching system memory size */
|
||||||
#include <linux/blkdev.h> /* For block_size() */
|
#include <linux/blkdev.h> /* For block_size() */
|
||||||
#include <linux/delay.h> /* For ssleep/msleep */
|
#include <linux/delay.h> /* For ssleep/msleep */
|
||||||
|
#include <linux/device.h>
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Bucket size for counting good commands in between bad ones.
|
* Bucket size for counting good commands in between bad ones.
|
||||||
|
@ -397,7 +398,7 @@ ahd_linux_unmap_scb(struct ahd_softc *ahd, struct scb *scb)
|
||||||
|
|
||||||
/******************************** Macros **************************************/
|
/******************************** Macros **************************************/
|
||||||
#define BUILD_SCSIID(ahd, cmd) \
|
#define BUILD_SCSIID(ahd, cmd) \
|
||||||
((((cmd)->device->id << TID_SHIFT) & TID) | (ahd)->our_id)
|
(((scmd_id(cmd) << TID_SHIFT) & TID) | (ahd)->our_id)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Return a string describing the driver.
|
* Return a string describing the driver.
|
||||||
|
@ -565,7 +566,7 @@ ahd_linux_slave_configure(struct scsi_device *sdev)
|
||||||
|
|
||||||
ahd = *((struct ahd_softc **)sdev->host->hostdata);
|
ahd = *((struct ahd_softc **)sdev->host->hostdata);
|
||||||
if (bootverbose)
|
if (bootverbose)
|
||||||
printf("%s: Slave Configure %d\n", ahd_name(ahd), sdev->id);
|
sdev_printk(KERN_INFO, sdev, "Slave Configure\n");
|
||||||
|
|
||||||
ahd_linux_device_queue_depth(sdev);
|
ahd_linux_device_queue_depth(sdev);
|
||||||
|
|
||||||
|
@ -684,7 +685,7 @@ ahd_linux_bus_reset(struct scsi_cmnd *cmd)
|
||||||
ahd_name(ahd), cmd);
|
ahd_name(ahd), cmd);
|
||||||
#endif
|
#endif
|
||||||
ahd_lock(ahd, &s);
|
ahd_lock(ahd, &s);
|
||||||
found = ahd_reset_channel(ahd, cmd->device->channel + 'A',
|
found = ahd_reset_channel(ahd, scmd_channel(cmd) + 'A',
|
||||||
/*initiate reset*/TRUE);
|
/*initiate reset*/TRUE);
|
||||||
ahd_unlock(ahd, &s);
|
ahd_unlock(ahd, &s);
|
||||||
|
|
||||||
|
@ -2067,9 +2068,8 @@ ahd_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
wait = FALSE;
|
wait = FALSE;
|
||||||
ahd = *(struct ahd_softc **)cmd->device->host->hostdata;
|
ahd = *(struct ahd_softc **)cmd->device->host->hostdata;
|
||||||
|
|
||||||
printf("%s:%d:%d:%d: Attempting to queue a%s message:",
|
scmd_printk(KERN_INFO, cmd,
|
||||||
ahd_name(ahd), cmd->device->channel,
|
"Attempting to queue a%s message:",
|
||||||
cmd->device->id, cmd->device->lun,
|
|
||||||
flag == SCB_ABORT ? "n ABORT" : " TARGET RESET");
|
flag == SCB_ABORT ? "n ABORT" : " TARGET RESET");
|
||||||
|
|
||||||
printf("CDB:");
|
printf("CDB:");
|
||||||
|
@ -2093,9 +2093,7 @@ ahd_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
* No target device for this command exists,
|
* No target device for this command exists,
|
||||||
* so we must not still own the command.
|
* so we must not still own the command.
|
||||||
*/
|
*/
|
||||||
printf("%s:%d:%d:%d: Is not an active device\n",
|
scmd_printk(KERN_INFO, cmd, "Is not an active device\n");
|
||||||
ahd_name(ahd), cmd->device->channel, cmd->device->id,
|
|
||||||
cmd->device->lun);
|
|
||||||
retval = SUCCESS;
|
retval = SUCCESS;
|
||||||
goto no_cmd;
|
goto no_cmd;
|
||||||
}
|
}
|
||||||
|
@ -2112,8 +2110,9 @@ ahd_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
|
|
||||||
/* Any SCB for this device will do for a target reset */
|
/* Any SCB for this device will do for a target reset */
|
||||||
LIST_FOREACH(pending_scb, &ahd->pending_scbs, pending_links) {
|
LIST_FOREACH(pending_scb, &ahd->pending_scbs, pending_links) {
|
||||||
if (ahd_match_scb(ahd, pending_scb, cmd->device->id,
|
if (ahd_match_scb(ahd, pending_scb,
|
||||||
cmd->device->channel + 'A',
|
scmd_id(cmd),
|
||||||
|
scmd_channel(cmd) + 'A',
|
||||||
CAM_LUN_WILDCARD,
|
CAM_LUN_WILDCARD,
|
||||||
SCB_LIST_NULL, ROLE_INITIATOR) == 0)
|
SCB_LIST_NULL, ROLE_INITIATOR) == 0)
|
||||||
break;
|
break;
|
||||||
|
@ -2121,9 +2120,7 @@ ahd_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pending_scb == NULL) {
|
if (pending_scb == NULL) {
|
||||||
printf("%s:%d:%d:%d: Command not found\n",
|
scmd_printk(KERN_INFO, cmd, "Command not found\n");
|
||||||
ahd_name(ahd), cmd->device->channel, cmd->device->id,
|
|
||||||
cmd->device->lun);
|
|
||||||
goto no_cmd;
|
goto no_cmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2146,9 +2143,7 @@ ahd_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
paused = TRUE;
|
paused = TRUE;
|
||||||
|
|
||||||
if ((pending_scb->flags & SCB_ACTIVE) == 0) {
|
if ((pending_scb->flags & SCB_ACTIVE) == 0) {
|
||||||
printf("%s:%d:%d:%d: Command already completed\n",
|
scmd_printk(KERN_INFO, cmd, "Command already completed\n");
|
||||||
ahd_name(ahd), cmd->device->channel, cmd->device->id,
|
|
||||||
cmd->device->lun);
|
|
||||||
goto no_cmd;
|
goto no_cmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2204,7 +2199,7 @@ ahd_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
if (last_phase != P_BUSFREE
|
if (last_phase != P_BUSFREE
|
||||||
&& (SCB_GET_TAG(pending_scb) == active_scbptr
|
&& (SCB_GET_TAG(pending_scb) == active_scbptr
|
||||||
|| (flag == SCB_DEVICE_RESET
|
|| (flag == SCB_DEVICE_RESET
|
||||||
&& SCSIID_TARGET(ahd, saved_scsiid) == cmd->device->id))) {
|
&& SCSIID_TARGET(ahd, saved_scsiid) == scmd_id(cmd)))) {
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We're active on the bus, so assert ATN
|
* We're active on the bus, so assert ATN
|
||||||
|
@ -2214,9 +2209,7 @@ ahd_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
pending_scb->flags |= SCB_RECOVERY_SCB|flag;
|
pending_scb->flags |= SCB_RECOVERY_SCB|flag;
|
||||||
ahd_outb(ahd, MSG_OUT, HOST_MSG);
|
ahd_outb(ahd, MSG_OUT, HOST_MSG);
|
||||||
ahd_outb(ahd, SCSISIGO, last_phase|ATNO);
|
ahd_outb(ahd, SCSISIGO, last_phase|ATNO);
|
||||||
printf("%s:%d:%d:%d: Device is active, asserting ATN\n",
|
scmd_printk(KERN_INFO, cmd, "Device is active, asserting ATN\n");
|
||||||
ahd_name(ahd), cmd->device->channel,
|
|
||||||
cmd->device->id, cmd->device->lun);
|
|
||||||
wait = TRUE;
|
wait = TRUE;
|
||||||
} else if (disconnected) {
|
} else if (disconnected) {
|
||||||
|
|
||||||
|
@ -2277,9 +2270,7 @@ ahd_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
printf("Device is disconnected, re-queuing SCB\n");
|
printf("Device is disconnected, re-queuing SCB\n");
|
||||||
wait = TRUE;
|
wait = TRUE;
|
||||||
} else {
|
} else {
|
||||||
printf("%s:%d:%d:%d: Unable to deliver message\n",
|
scmd_printk(KERN_INFO, cmd, "Unable to deliver message\n");
|
||||||
ahd_name(ahd), cmd->device->channel,
|
|
||||||
cmd->device->id, cmd->device->lun);
|
|
||||||
retval = FAILED;
|
retval = FAILED;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
|
@ -641,7 +641,7 @@ ahc_linux_slave_configure(struct scsi_device *sdev)
|
||||||
ahc = *((struct ahc_softc **)sdev->host->hostdata);
|
ahc = *((struct ahc_softc **)sdev->host->hostdata);
|
||||||
|
|
||||||
if (bootverbose)
|
if (bootverbose)
|
||||||
printf("%s: Slave Configure %d\n", ahc_name(ahc), sdev->id);
|
sdev_printk(KERN_INFO, sdev, "Slave Configure\n");
|
||||||
|
|
||||||
ahc_linux_device_queue_depth(sdev);
|
ahc_linux_device_queue_depth(sdev);
|
||||||
|
|
||||||
|
@ -686,7 +686,7 @@ ahc_linux_biosparam(struct scsi_device *sdev, struct block_device *bdev,
|
||||||
u_int channel;
|
u_int channel;
|
||||||
|
|
||||||
ahc = *((struct ahc_softc **)sdev->host->hostdata);
|
ahc = *((struct ahc_softc **)sdev->host->hostdata);
|
||||||
channel = sdev->channel;
|
channel = sdev_channel(sdev);
|
||||||
|
|
||||||
bh = scsi_bios_ptable(bdev);
|
bh = scsi_bios_ptable(bdev);
|
||||||
if (bh) {
|
if (bh) {
|
||||||
|
@ -759,7 +759,7 @@ ahc_linux_bus_reset(struct scsi_cmnd *cmd)
|
||||||
ahc = *(struct ahc_softc **)cmd->device->host->hostdata;
|
ahc = *(struct ahc_softc **)cmd->device->host->hostdata;
|
||||||
|
|
||||||
ahc_lock(ahc, &flags);
|
ahc_lock(ahc, &flags);
|
||||||
found = ahc_reset_channel(ahc, cmd->device->channel + 'A',
|
found = ahc_reset_channel(ahc, scmd_channel(cmd) + 'A',
|
||||||
/*initiate reset*/TRUE);
|
/*initiate reset*/TRUE);
|
||||||
ahc_unlock(ahc, &flags);
|
ahc_unlock(ahc, &flags);
|
||||||
|
|
||||||
|
@ -2117,9 +2117,7 @@ ahc_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
wait = FALSE;
|
wait = FALSE;
|
||||||
ahc = *(struct ahc_softc **)cmd->device->host->hostdata;
|
ahc = *(struct ahc_softc **)cmd->device->host->hostdata;
|
||||||
|
|
||||||
printf("%s:%d:%d:%d: Attempting to queue a%s message\n",
|
scmd_printk(KERN_INFO, cmd, "Attempting to queue a%s message\n",
|
||||||
ahc_name(ahc), cmd->device->channel,
|
|
||||||
cmd->device->id, cmd->device->lun,
|
|
||||||
flag == SCB_ABORT ? "n ABORT" : " TARGET RESET");
|
flag == SCB_ABORT ? "n ABORT" : " TARGET RESET");
|
||||||
|
|
||||||
printf("CDB:");
|
printf("CDB:");
|
||||||
|
@ -2174,8 +2172,8 @@ ahc_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
|
|
||||||
/* Any SCB for this device will do for a target reset */
|
/* Any SCB for this device will do for a target reset */
|
||||||
LIST_FOREACH(pending_scb, &ahc->pending_scbs, pending_links) {
|
LIST_FOREACH(pending_scb, &ahc->pending_scbs, pending_links) {
|
||||||
if (ahc_match_scb(ahc, pending_scb, cmd->device->id,
|
if (ahc_match_scb(ahc, pending_scb, scmd_id(cmd),
|
||||||
cmd->device->channel + 'A',
|
scmd_channel(cmd) + 'A',
|
||||||
CAM_LUN_WILDCARD,
|
CAM_LUN_WILDCARD,
|
||||||
SCB_LIST_NULL, ROLE_INITIATOR) == 0)
|
SCB_LIST_NULL, ROLE_INITIATOR) == 0)
|
||||||
break;
|
break;
|
||||||
|
@ -2183,9 +2181,7 @@ ahc_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pending_scb == NULL) {
|
if (pending_scb == NULL) {
|
||||||
printf("%s:%d:%d:%d: Command not found\n",
|
scmd_printk(KERN_INFO, cmd, "Command not found\n");
|
||||||
ahc_name(ahc), cmd->device->channel, cmd->device->id,
|
|
||||||
cmd->device->lun);
|
|
||||||
goto no_cmd;
|
goto no_cmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2207,9 +2203,7 @@ ahc_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
paused = TRUE;
|
paused = TRUE;
|
||||||
|
|
||||||
if ((pending_scb->flags & SCB_ACTIVE) == 0) {
|
if ((pending_scb->flags & SCB_ACTIVE) == 0) {
|
||||||
printf("%s:%d:%d:%d: Command already completed\n",
|
scmd_printk(KERN_INFO, cmd, "Command already completed\n");
|
||||||
ahc_name(ahc), cmd->device->channel, cmd->device->id,
|
|
||||||
cmd->device->lun);
|
|
||||||
goto no_cmd;
|
goto no_cmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2266,7 +2260,7 @@ ahc_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
if (last_phase != P_BUSFREE
|
if (last_phase != P_BUSFREE
|
||||||
&& (pending_scb->hscb->tag == active_scb_index
|
&& (pending_scb->hscb->tag == active_scb_index
|
||||||
|| (flag == SCB_DEVICE_RESET
|
|| (flag == SCB_DEVICE_RESET
|
||||||
&& SCSIID_TARGET(ahc, saved_scsiid) == cmd->device->id))) {
|
&& SCSIID_TARGET(ahc, saved_scsiid) == scmd_id(cmd)))) {
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We're active on the bus, so assert ATN
|
* We're active on the bus, so assert ATN
|
||||||
|
@ -2276,9 +2270,7 @@ ahc_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
pending_scb->flags |= SCB_RECOVERY_SCB|flag;
|
pending_scb->flags |= SCB_RECOVERY_SCB|flag;
|
||||||
ahc_outb(ahc, MSG_OUT, HOST_MSG);
|
ahc_outb(ahc, MSG_OUT, HOST_MSG);
|
||||||
ahc_outb(ahc, SCSISIGO, last_phase|ATNO);
|
ahc_outb(ahc, SCSISIGO, last_phase|ATNO);
|
||||||
printf("%s:%d:%d:%d: Device is active, asserting ATN\n",
|
scmd_printk(KERN_INFO, cmd, "Device is active, asserting ATN\n");
|
||||||
ahc_name(ahc), cmd->device->channel, cmd->device->id,
|
|
||||||
cmd->device->lun);
|
|
||||||
wait = TRUE;
|
wait = TRUE;
|
||||||
} else if (disconnected) {
|
} else if (disconnected) {
|
||||||
|
|
||||||
|
@ -2344,9 +2336,7 @@ ahc_linux_queue_recovery_cmd(struct scsi_cmnd *cmd, scb_flag flag)
|
||||||
printf("Device is disconnected, re-queuing SCB\n");
|
printf("Device is disconnected, re-queuing SCB\n");
|
||||||
wait = TRUE;
|
wait = TRUE;
|
||||||
} else {
|
} else {
|
||||||
printf("%s:%d:%d:%d: Unable to deliver message\n",
|
scmd_printk(KERN_INFO, cmd, "Unable to deliver message\n");
|
||||||
ahc_name(ahc), cmd->device->channel, cmd->device->id,
|
|
||||||
cmd->device->lun);
|
|
||||||
retval = FAILED;
|
retval = FAILED;
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
|
@ -297,11 +297,10 @@ stop_dma:
|
||||||
}
|
}
|
||||||
workreq = dev->id[c][target_id].curr_req;
|
workreq = dev->id[c][target_id].curr_req;
|
||||||
#ifdef ED_DBGP
|
#ifdef ED_DBGP
|
||||||
printk(KERN_DEBUG "Channel = %d ID = %d LUN = %d CDB",c,workreq->device->id,workreq->device->lun);
|
scmd_printk(KERN_DEBUG, workreq, "CDB");
|
||||||
for(l=0;l<workreq->cmd_len;l++)
|
for (l = 0; l < workreq->cmd_len; l++)
|
||||||
{
|
|
||||||
printk(KERN_DEBUG " %x",workreq->cmnd[l]);
|
printk(KERN_DEBUG " %x",workreq->cmnd[l]);
|
||||||
}
|
printk("\n");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
tmport = workport + 0x0f;
|
tmport = workport + 0x0f;
|
||||||
|
@ -622,10 +621,10 @@ static int atp870u_queuecommand(struct scsi_cmnd * req_p,
|
||||||
struct atp_unit *dev;
|
struct atp_unit *dev;
|
||||||
struct Scsi_Host *host;
|
struct Scsi_Host *host;
|
||||||
|
|
||||||
c = req_p->device->channel;
|
c = scmd_channel(req_p);
|
||||||
req_p->sense_buffer[0]=0;
|
req_p->sense_buffer[0]=0;
|
||||||
req_p->resid = 0;
|
req_p->resid = 0;
|
||||||
if (req_p->device->channel > 1) {
|
if (scmd_channel(req_p) > 1) {
|
||||||
req_p->result = 0x00040000;
|
req_p->result = 0x00040000;
|
||||||
done(req_p);
|
done(req_p);
|
||||||
#ifdef ED_DBGP
|
#ifdef ED_DBGP
|
||||||
|
@ -640,7 +639,7 @@ static int atp870u_queuecommand(struct scsi_cmnd * req_p,
|
||||||
|
|
||||||
|
|
||||||
m = 1;
|
m = 1;
|
||||||
m = m << req_p->device->id;
|
m = m << scmd_id(req_p);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Fake a timeout for missing targets
|
* Fake a timeout for missing targets
|
||||||
|
@ -758,9 +757,9 @@ static void send_s870(struct atp_unit *dev,unsigned char c)
|
||||||
dev->quhd[c] = 0;
|
dev->quhd[c] = 0;
|
||||||
}
|
}
|
||||||
workreq = dev->quereq[c][dev->quhd[c]];
|
workreq = dev->quereq[c][dev->quhd[c]];
|
||||||
if (dev->id[c][workreq->device->id].curr_req == 0) {
|
if (dev->id[c][scmd_id(workreq)].curr_req == 0) {
|
||||||
dev->id[c][workreq->device->id].curr_req = workreq;
|
dev->id[c][scmd_id(workreq)].curr_req = workreq;
|
||||||
dev->last_cmd[c] = workreq->device->id;
|
dev->last_cmd[c] = scmd_id(workreq);
|
||||||
goto cmd_subp;
|
goto cmd_subp;
|
||||||
}
|
}
|
||||||
dev->quhd[c] = j;
|
dev->quhd[c] = j;
|
||||||
|
@ -787,16 +786,16 @@ abortsnd:
|
||||||
oktosend:
|
oktosend:
|
||||||
#ifdef ED_DBGP
|
#ifdef ED_DBGP
|
||||||
printk("OK to Send\n");
|
printk("OK to Send\n");
|
||||||
printk("CDB");
|
scmd_printk(KERN_DEBUG, workreq, "CDB");
|
||||||
for(i=0;i<workreq->cmd_len;i++) {
|
for(i=0;i<workreq->cmd_len;i++) {
|
||||||
printk(" %x",workreq->cmnd[i]);
|
printk(" %x",workreq->cmnd[i]);
|
||||||
}
|
}
|
||||||
printk("\nChannel = %d ID = %d LUN = %d\n",c,workreq->device->id,workreq->device->lun);
|
printk("\n");
|
||||||
#endif
|
#endif
|
||||||
if (dev->dev_id == ATP885_DEVID) {
|
if (dev->dev_id == ATP885_DEVID) {
|
||||||
j = inb(dev->baseport + 0x29) & 0xfe;
|
j = inb(dev->baseport + 0x29) & 0xfe;
|
||||||
outb(j, dev->baseport + 0x29);
|
outb(j, dev->baseport + 0x29);
|
||||||
dev->r1f[c][workreq->device->id] = 0;
|
dev->r1f[c][scmd_id(workreq)] = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (workreq->cmnd[0] == READ_CAPACITY) {
|
if (workreq->cmnd[0] == READ_CAPACITY) {
|
||||||
|
@ -810,7 +809,7 @@ oktosend:
|
||||||
|
|
||||||
tmport = workport + 0x1b;
|
tmport = workport + 0x1b;
|
||||||
j = 0;
|
j = 0;
|
||||||
target_id = workreq->device->id;
|
target_id = scmd_id(workreq);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Wide ?
|
* Wide ?
|
||||||
|
@ -3109,7 +3108,7 @@ static int atp870u_abort(struct scsi_cmnd * SCpnt)
|
||||||
host = SCpnt->device->host;
|
host = SCpnt->device->host;
|
||||||
|
|
||||||
dev = (struct atp_unit *)&host->hostdata;
|
dev = (struct atp_unit *)&host->hostdata;
|
||||||
c=SCpnt->device->channel;
|
c = scmd_channel(SCpnt);
|
||||||
printk(" atp870u: abort Channel = %x \n", c);
|
printk(" atp870u: abort Channel = %x \n", c);
|
||||||
printk("working=%x last_cmd=%x ", dev->working[c], dev->last_cmd[c]);
|
printk("working=%x last_cmd=%x ", dev->working[c], dev->last_cmd[c]);
|
||||||
printk(" quhdu=%x quendu=%x ", dev->quhd[c], dev->quend[c]);
|
printk(" quhdu=%x quendu=%x ", dev->quhd[c], dev->quend[c]);
|
||||||
|
|
|
@ -940,9 +940,7 @@ static int ch_probe(struct device *dev)
|
||||||
MKDEV(SCSI_CHANGER_MAJOR,ch->minor),
|
MKDEV(SCSI_CHANGER_MAJOR,ch->minor),
|
||||||
dev, "s%s", ch->name);
|
dev, "s%s", ch->name);
|
||||||
|
|
||||||
printk(KERN_INFO "Attached scsi changer %s "
|
sdev_printk(KERN_INFO, sd, "Attached scsi changer %s\n", ch->name);
|
||||||
"at scsi%d, channel %d, id %d, lun %d\n",
|
|
||||||
ch->name, sd->host->host_no, sd->channel, sd->id, sd->lun);
|
|
||||||
|
|
||||||
spin_lock(&ch_devlist_lock);
|
spin_lock(&ch_devlist_lock);
|
||||||
list_add_tail(&ch->list,&ch_devlist);
|
list_add_tail(&ch->list,&ch_devlist);
|
||||||
|
|
|
@ -1389,10 +1389,7 @@ EXPORT_SYMBOL(scsi_print_msg);
|
||||||
void scsi_print_command(struct scsi_cmnd *cmd)
|
void scsi_print_command(struct scsi_cmnd *cmd)
|
||||||
{
|
{
|
||||||
/* Assume appended output (i.e. not at start of line) */
|
/* Assume appended output (i.e. not at start of line) */
|
||||||
printk("scsi%d : destination target %d, lun %d\n",
|
sdev_printk("", cmd->device, "\n");
|
||||||
cmd->device->host->host_no,
|
|
||||||
cmd->device->id,
|
|
||||||
cmd->device->lun);
|
|
||||||
printk(KERN_INFO " command: ");
|
printk(KERN_INFO " command: ");
|
||||||
scsi_print_cdb(cmd->cmnd, cmd->cmd_len, 0);
|
scsi_print_cdb(cmd->cmnd, cmd->cmd_len, 0);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,19 +0,0 @@
|
||||||
#ifndef CPQFCTS_H
|
|
||||||
#define CPQFCTS_H
|
|
||||||
#include "cpqfcTSstructs.h"
|
|
||||||
|
|
||||||
// These functions are required by the Linux SCSI layers
|
|
||||||
extern int cpqfcTS_detect(Scsi_Host_Template *);
|
|
||||||
extern int cpqfcTS_release(struct Scsi_Host *);
|
|
||||||
extern const char * cpqfcTS_info(struct Scsi_Host *);
|
|
||||||
extern int cpqfcTS_proc_info(struct Scsi_Host *, char *, char **, off_t, int, int);
|
|
||||||
extern int cpqfcTS_queuecommand(Scsi_Cmnd *, void (* done)(Scsi_Cmnd *));
|
|
||||||
extern int cpqfcTS_abort(Scsi_Cmnd *);
|
|
||||||
extern int cpqfcTS_reset(Scsi_Cmnd *, unsigned int);
|
|
||||||
extern int cpqfcTS_eh_abort(Scsi_Cmnd *Cmnd);
|
|
||||||
extern int cpqfcTS_eh_device_reset(Scsi_Cmnd *);
|
|
||||||
extern int cpqfcTS_biosparam(struct scsi_device *, struct block_device *,
|
|
||||||
sector_t, int[]);
|
|
||||||
extern int cpqfcTS_ioctl( Scsi_Device *ScsiDev, int Cmnd, void *arg);
|
|
||||||
|
|
||||||
#endif /* CPQFCTS_H */
|
|
|
@ -1,238 +0,0 @@
|
||||||
/* Copyright(c) 2000, Compaq Computer Corporation
|
|
||||||
* Fibre Channel Host Bus Adapter
|
|
||||||
* 64-bit, 66MHz PCI
|
|
||||||
* Originally developed and tested on:
|
|
||||||
* (front): [chip] Tachyon TS HPFC-5166A/1.2 L2C1090 ...
|
|
||||||
* SP# P225CXCBFIEL6T, Rev XC
|
|
||||||
* SP# 161290-001, Rev XD
|
|
||||||
* (back): Board No. 010008-001 A/W Rev X5, FAB REV X5
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
* Written by Don Zimmerman
|
|
||||||
*/
|
|
||||||
#ifndef CPQFCTSCHIP_H
|
|
||||||
#define CPQFCTSCHIP_H
|
|
||||||
#ifndef TACHYON_CHIP_INC
|
|
||||||
|
|
||||||
// FC-PH (Physical) specification levels for Login payloads
|
|
||||||
// NOTE: These are NOT strictly complied with by any FC vendors
|
|
||||||
|
|
||||||
#define FC_PH42 0x08
|
|
||||||
#define FC_PH43 0x09
|
|
||||||
#define FC_PH3 0x20
|
|
||||||
|
|
||||||
#define TACHLITE_TS_RX_SIZE 1024 // max inbound frame size
|
|
||||||
// "I" prefix is for Include
|
|
||||||
|
|
||||||
#define IVENDID 0x00 // word
|
|
||||||
#define IDEVID 0x02
|
|
||||||
#define ITLCFGCMD 0x04
|
|
||||||
#define IMEMBASE 0x18 // Tachyon
|
|
||||||
#define ITLMEMBASE 0x1C // Tachlite
|
|
||||||
#define IIOBASEL 0x10 // Tachyon I/O base address, lower 256 bytes
|
|
||||||
#define IIOBASEU 0x14 // Tachyon I/O base address, upper 256 bytes
|
|
||||||
#define ITLIOBASEL 0x14 // TachLite I/O base address, lower 256 bytes
|
|
||||||
#define ITLIOBASEU 0x18 // TachLite I/O base address, upper 256 bytes
|
|
||||||
#define ITLRAMBASE 0x20 // TL on-board RAM start
|
|
||||||
#define ISROMBASE 0x24
|
|
||||||
#define IROMBASE 0x30
|
|
||||||
|
|
||||||
#define ICFGCMD 0x04 // PCI config - PCI config access (word)
|
|
||||||
#define ICFGSTAT 0x06 // PCI status (R - word)
|
|
||||||
#define IRCTR_WCTR 0x1F2 // ROM control / pre-fetch wait counter
|
|
||||||
#define IPCIMCTR 0x1F3 // PCI master control register
|
|
||||||
#define IINTPEND 0x1FD // Interrupt pending (I/O Upper - Tachyon & TL)
|
|
||||||
#define IINTEN 0x1FE // Interrupt enable (I/O Upper - Tachyon & TL)
|
|
||||||
#define IINTSTAT 0x1FF // Interrupt status (I/O Upper - Tachyon & TL)
|
|
||||||
|
|
||||||
#define IMQ_BASE 0x80
|
|
||||||
#define IMQ_LENGTH 0x84
|
|
||||||
#define IMQ_CONSUMER_INDEX 0x88
|
|
||||||
#define IMQ_PRODUCER_INDEX 0x8C // Tach copies its INDX to bits 0-7 of value
|
|
||||||
|
|
||||||
/*
|
|
||||||
// IOBASE UPPER
|
|
||||||
#define SFSBQ_BASE 0x00 // single-frame sequences
|
|
||||||
#define SFSBQ_LENGTH 0x04
|
|
||||||
#define SFSBQ_PRODUCER_INDEX 0x08
|
|
||||||
#define SFSBQ_CONSUMER_INDEX 0x0C // (R)
|
|
||||||
#define SFS_BUFFER_LENGTH 0X10
|
|
||||||
// SCSI-FCP hardware assists
|
|
||||||
#define SEST_BASE 0x40 // SSCI Exchange State Table
|
|
||||||
#define SEST_LENGTH 0x44
|
|
||||||
#define SCSI_BUFFER_LENGTH 0x48
|
|
||||||
#define SEST_LINKED_LIST 0x4C
|
|
||||||
|
|
||||||
#define TACHYON_My_ID 0x6C
|
|
||||||
#define TACHYON_CONFIGURATION 0x84 // (R/W) reset val 2
|
|
||||||
#define TACHYON_CONTROL 0x88
|
|
||||||
#define TACHYON_STATUS 0x8C // (R)
|
|
||||||
#define TACHYON_FLUSH_SEST 0x90 // (R/W)
|
|
||||||
#define TACHYON_EE_CREDIT_TMR 0x94 // (R)
|
|
||||||
#define TACHYON_BB_CREDIT_TMR 0x98 // (R)
|
|
||||||
#define TACHYON_RCV_FRAME_ERR 0x9C // (R)
|
|
||||||
#define FRAME_MANAGER_CONFIG 0xC0 // (R/W)
|
|
||||||
#define FRAME_MANAGER_CONTROL 0xC4
|
|
||||||
#define FRAME_MANAGER_STATUS 0xC8 // (R)
|
|
||||||
#define FRAME_MANAGER_ED_TOV 0xCC
|
|
||||||
#define FRAME_MANAGER_LINK_ERR1 0xD0 // (R)
|
|
||||||
#define FRAME_MANAGER_LINK_ERR2 0xD4 // (R)
|
|
||||||
#define FRAME_MANAGER_TIMEOUT2 0xD8 // (W)
|
|
||||||
#define FRAME_MANAGER_BB_CREDIT 0xDC // (R)
|
|
||||||
#define FRAME_MANAGER_WWN_HI 0xE0 // (R/W)
|
|
||||||
#define FRAME_MANAGER_WWN_LO 0xE4 // (R/W)
|
|
||||||
#define FRAME_MANAGER_RCV_AL_PA 0xE8 // (R)
|
|
||||||
#define FRAME_MANAGER_PRIMITIVE 0xEC // {K28.5} byte1 byte2 byte3
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define TL_MEM_ERQ_BASE 0x0 //ERQ Base
|
|
||||||
#define TL_IO_ERQ_BASE 0x0 //ERQ base
|
|
||||||
|
|
||||||
#define TL_MEM_ERQ_LENGTH 0x4 //ERQ Length
|
|
||||||
#define TL_IO_ERQ_LENGTH 0x4 //ERQ Length
|
|
||||||
|
|
||||||
#define TL_MEM_ERQ_PRODUCER_INDEX 0x8 //ERQ Producer Index register
|
|
||||||
#define TL_IO_ERQ_PRODUCER_INDEX 0x8 //ERQ Producer Index register
|
|
||||||
|
|
||||||
#define TL_MEM_ERQ_CONSUMER_INDEX_ADR 0xC //ERQ Consumer Index address register
|
|
||||||
#define TL_IO_ERQ_CONSUMER_INDEX_ADR 0xC //ERQ Consumer Index address register
|
|
||||||
|
|
||||||
#define TL_MEM_ERQ_CONSUMER_INDEX 0xC //ERQ Consumer Index
|
|
||||||
#define TL_IO_ERQ_CONSUMER_INDEX 0xC //ERQ Consumer Index
|
|
||||||
|
|
||||||
#define TL_MEM_SFQ_BASE 0x50 //SFQ Base
|
|
||||||
#define TL_IO_SFQ_BASE 0x50 //SFQ base
|
|
||||||
|
|
||||||
#define TL_MEM_SFQ_LENGTH 0x54 //SFQ Length
|
|
||||||
#define TL_IO_SFQ_LENGTH 0x54 //SFQ Length
|
|
||||||
|
|
||||||
#define TL_MEM_SFQ_CONSUMER_INDEX 0x58 //SFQ Consumer Index
|
|
||||||
#define TL_IO_SFQ_CONSUMER_INDEX 0x58 //SFQ Consumer Index
|
|
||||||
|
|
||||||
#define TL_MEM_IMQ_BASE 0x80 //IMQ Base
|
|
||||||
#define TL_IO_IMQ_BASE 0x80 //IMQ base
|
|
||||||
|
|
||||||
#define TL_MEM_IMQ_LENGTH 0x84 //IMQ Length
|
|
||||||
#define TL_IO_IMQ_LENGTH 0x84 //IMQ Length
|
|
||||||
|
|
||||||
#define TL_MEM_IMQ_CONSUMER_INDEX 0x88 //IMQ Consumer Index
|
|
||||||
#define TL_IO_IMQ_CONSUMER_INDEX 0x88 //IMQ Consumer Index
|
|
||||||
|
|
||||||
#define TL_MEM_IMQ_PRODUCER_INDEX_ADR 0x8C //IMQ Producer Index address register
|
|
||||||
#define TL_IO_IMQ_PRODUCER_INDEX_ADR 0x8C //IMQ Producer Index address register
|
|
||||||
|
|
||||||
#define TL_MEM_SEST_BASE 0x140 //SFQ Base
|
|
||||||
#define TL_IO_SEST_BASE 0x40 //SFQ base
|
|
||||||
|
|
||||||
#define TL_MEM_SEST_LENGTH 0x144 //SFQ Length
|
|
||||||
#define TL_IO_SEST_LENGTH 0x44 //SFQ Length
|
|
||||||
|
|
||||||
#define TL_MEM_SEST_LINKED_LIST 0x14C
|
|
||||||
|
|
||||||
#define TL_MEM_SEST_SG_PAGE 0x168 // Extended Scatter/Gather page size
|
|
||||||
|
|
||||||
#define TL_MEM_TACH_My_ID 0x16C
|
|
||||||
#define TL_IO_TACH_My_ID 0x6C //My AL_PA ID
|
|
||||||
|
|
||||||
#define TL_MEM_TACH_CONFIG 0x184 //Tachlite Configuration register
|
|
||||||
#define TL_IO_CONFIG 0x84 //Tachlite Configuration register
|
|
||||||
|
|
||||||
#define TL_MEM_TACH_CONTROL 0x188 //Tachlite Control register
|
|
||||||
#define TL_IO_CTR 0x88 //Tachlite Control register
|
|
||||||
|
|
||||||
#define TL_MEM_TACH_STATUS 0x18C //Tachlite Status register
|
|
||||||
#define TL_IO_STAT 0x8C //Tachlite Status register
|
|
||||||
|
|
||||||
#define TL_MEM_FM_CONFIG 0x1C0 //Frame Manager Configuration register
|
|
||||||
#define TL_IO_FM_CONFIG 0xC0 //Frame Manager Configuration register
|
|
||||||
|
|
||||||
#define TL_MEM_FM_CONTROL 0x1C4 //Frame Manager Control
|
|
||||||
#define TL_IO_FM_CTL 0xC4 //Frame Manager Control
|
|
||||||
|
|
||||||
#define TL_MEM_FM_STATUS 0x1C8 //Frame Manager Status
|
|
||||||
#define TL_IO_FM_STAT 0xC8 //Frame Manager Status
|
|
||||||
|
|
||||||
#define TL_MEM_FM_LINK_STAT1 0x1D0 //Frame Manager Link Status 1
|
|
||||||
#define TL_IO_FM_LINK_STAT1 0xD0 //Frame Manager Link Status 1
|
|
||||||
|
|
||||||
#define TL_MEM_FM_LINK_STAT2 0x1D4 //Frame Manager Link Status 2
|
|
||||||
#define TL_IO_FM_LINK_STAT2 0xD4 //Frame Manager Link Status 2
|
|
||||||
|
|
||||||
#define TL_MEM_FM_TIMEOUT2 0x1D8 // (W)
|
|
||||||
|
|
||||||
#define TL_MEM_FM_BB_CREDIT0 0x1DC
|
|
||||||
|
|
||||||
#define TL_MEM_FM_WWN_HI 0x1E0 //Frame Manager World Wide Name High
|
|
||||||
#define TL_IO_FM_WWN_HI 0xE0 //Frame Manager World Wide Name High
|
|
||||||
|
|
||||||
#define TL_MEM_FM_WWN_LO 0x1E4 //Frame Manager World Wide Name LOW
|
|
||||||
#define TL_IO_FM_WWN_LO 0xE4 //Frame Manager World Wide Name Low
|
|
||||||
|
|
||||||
#define TL_MEM_FM_RCV_AL_PA 0x1E8 //Frame Manager AL_PA Received register
|
|
||||||
#define TL_IO_FM_ALPA 0xE8 //Frame Manager AL_PA Received register
|
|
||||||
|
|
||||||
#define TL_MEM_FM_ED_TOV 0x1CC
|
|
||||||
|
|
||||||
#define TL_IO_ROMCTR 0xFA //TL PCI ROM Control Register
|
|
||||||
#define TL_IO_PCIMCTR 0xFB //TL PCI Master Control Register
|
|
||||||
#define TL_IO_SOFTRST 0xFC //Tachlite Configuration register
|
|
||||||
#define TL_MEM_SOFTRST 0x1FC //Tachlite Configuration register
|
|
||||||
|
|
||||||
// completion message types (bit 8 set means Interrupt generated)
|
|
||||||
// CM_Type
|
|
||||||
#define OUTBOUND_COMPLETION 0
|
|
||||||
#define ERROR_IDLE_COMPLETION 0x01
|
|
||||||
#define OUT_HI_PRI_COMPLETION 0x01
|
|
||||||
#define INBOUND_MFS_COMPLETION 0x02
|
|
||||||
#define INBOUND_000_COMPLETION 0x03
|
|
||||||
#define INBOUND_SFS_COMPLETION 0x04 // Tachyon & TachLite
|
|
||||||
#define ERQ_FROZEN_COMPLETION 0x06 // TachLite
|
|
||||||
#define INBOUND_C1_TIMEOUT 0x05
|
|
||||||
#define INBOUND_BUSIED_FRAME 0x06
|
|
||||||
#define SFS_BUF_WARN 0x07
|
|
||||||
#define FCP_FROZEN_COMPLETION 0x07 // TachLite
|
|
||||||
#define MFS_BUF_WARN 0x08
|
|
||||||
#define IMQ_BUF_WARN 0x09
|
|
||||||
#define FRAME_MGR_INTERRUPT 0x0A
|
|
||||||
#define READ_STATUS 0x0B
|
|
||||||
#define INBOUND_SCSI_DATA_COMPLETION 0x0C
|
|
||||||
#define INBOUND_FCP_XCHG_COMPLETION 0x0C // TachLite
|
|
||||||
#define INBOUND_SCSI_DATA_COMMAND 0x0D
|
|
||||||
#define BAD_SCSI_FRAME 0x0E
|
|
||||||
#define INB_SCSI_STATUS_COMPLETION 0x0F
|
|
||||||
#define BUFFER_PROCESSED_COMPLETION 0x11
|
|
||||||
|
|
||||||
// FC-AL (Tachyon) Loop Port State Machine defs
|
|
||||||
// (loop "Up" states)
|
|
||||||
#define MONITORING 0x0
|
|
||||||
#define ARBITRATING 0x1
|
|
||||||
#define ARBITRAT_WON 0x2
|
|
||||||
#define OPEN 0x3
|
|
||||||
#define OPENED 0x4
|
|
||||||
#define XMITTD_CLOSE 0x5
|
|
||||||
#define RCVD_CLOSE 0x6
|
|
||||||
#define TRANSFER 0x7
|
|
||||||
|
|
||||||
// (loop "Down" states)
|
|
||||||
#define INITIALIZING 0x8
|
|
||||||
#define O_I_INIT 0x9
|
|
||||||
#define O_I_PROTOCOL 0xa
|
|
||||||
#define O_I_LIP_RCVD 0xb
|
|
||||||
#define HOST_CONTROL 0xc
|
|
||||||
#define LOOP_FAIL 0xd
|
|
||||||
// (no 0xe)
|
|
||||||
#define OLD_PORT 0xf
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#define TACHYON_CHIP_INC
|
|
||||||
#endif
|
|
||||||
#endif /* CPQFCTSCHIP_H */
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,493 +0,0 @@
|
||||||
/* Copyright(c) 2000, Compaq Computer Corporation
|
|
||||||
* Fibre Channel Host Bus Adapter
|
|
||||||
* 64-bit, 66MHz PCI
|
|
||||||
* Originally developed and tested on:
|
|
||||||
* (front): [chip] Tachyon TS HPFC-5166A/1.2 L2C1090 ...
|
|
||||||
* SP# P225CXCBFIEL6T, Rev XC
|
|
||||||
* SP# 161290-001, Rev XD
|
|
||||||
* (back): Board No. 010008-001 A/W Rev X5, FAB REV X5
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
* Written by Don Zimmerman
|
|
||||||
*/
|
|
||||||
// These functions control the NVRAM I2C hardware on
|
|
||||||
// non-intelligent Fibre Host Adapters.
|
|
||||||
// The primary purpose is to read the HBA's NVRAM to get adapter's
|
|
||||||
// manufactured WWN to copy into Tachyon chip registers
|
|
||||||
// Orignal source author unknown
|
|
||||||
|
|
||||||
#include <linux/types.h>
|
|
||||||
enum boolean { FALSE, TRUE } ;
|
|
||||||
|
|
||||||
|
|
||||||
#ifndef UCHAR
|
|
||||||
typedef __u8 UCHAR;
|
|
||||||
#endif
|
|
||||||
#ifndef BOOLEAN
|
|
||||||
typedef __u8 BOOLEAN;
|
|
||||||
#endif
|
|
||||||
#ifndef USHORT
|
|
||||||
typedef __u16 USHORT;
|
|
||||||
#endif
|
|
||||||
#ifndef ULONG
|
|
||||||
typedef __u32 ULONG;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
|
|
||||||
#include <linux/string.h>
|
|
||||||
#include <linux/pci.h>
|
|
||||||
#include <linux/delay.h>
|
|
||||||
#include <linux/sched.h>
|
|
||||||
#include <asm/io.h> // struct pt_regs for IRQ handler & Port I/O
|
|
||||||
|
|
||||||
#include "cpqfcTSchip.h"
|
|
||||||
|
|
||||||
static void tl_i2c_tx_byte( void* GPIOout, UCHAR data );
|
|
||||||
/*static BOOLEAN tl_write_i2c_page_portion( void* GPIOin, void* GPIOout,
|
|
||||||
USHORT startOffset, // e.g. 0x2f for WWN start
|
|
||||||
USHORT count,
|
|
||||||
UCHAR *buf );
|
|
||||||
*/
|
|
||||||
|
|
||||||
//
|
|
||||||
// Tachlite GPIO2, GPIO3 (I2C) DEFINES
|
|
||||||
// The NVRAM chip NM24C03 defines SCL (serial clock) and SDA (serial data)
|
|
||||||
// GPIO2 drives SDA, and GPIO3 drives SCL
|
|
||||||
//
|
|
||||||
// Since Tachlite inverts the state of the GPIO 0-3 outputs, SET writes 0
|
|
||||||
// and clear writes 1. The input lines (read in TL status) is NOT inverted
|
|
||||||
// This really helps confuse the code and debugging.
|
|
||||||
|
|
||||||
#define SET_DATA_HI 0x0
|
|
||||||
#define SET_DATA_LO 0x8
|
|
||||||
#define SET_CLOCK_HI 0x0
|
|
||||||
#define SET_CLOCK_LO 0x4
|
|
||||||
|
|
||||||
#define SENSE_DATA_HI 0x8
|
|
||||||
#define SENSE_DATA_LO 0x0
|
|
||||||
#define SENSE_CLOCK_HI 0x4
|
|
||||||
#define SENSE_CLOCK_LO 0x0
|
|
||||||
|
|
||||||
#define SLAVE_READ_ADDRESS 0xA1
|
|
||||||
#define SLAVE_WRITE_ADDRESS 0xA0
|
|
||||||
|
|
||||||
|
|
||||||
static void i2c_delay(ULONG mstime);
|
|
||||||
static void tl_i2c_clock_pulse( UCHAR , void* GPIOout);
|
|
||||||
static UCHAR tl_read_i2c_data( void* );
|
|
||||||
|
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// Name: I2C_RX_ACK
|
|
||||||
//
|
|
||||||
// This routine receives an acknowledge over the I2C bus.
|
|
||||||
//
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
static unsigned short tl_i2c_rx_ack( void* GPIOin, void* GPIOout )
|
|
||||||
{
|
|
||||||
unsigned long value;
|
|
||||||
|
|
||||||
// do clock pulse, let data line float high
|
|
||||||
tl_i2c_clock_pulse( SET_DATA_HI, GPIOout );
|
|
||||||
|
|
||||||
// slave must drive data low for acknowledge
|
|
||||||
value = tl_read_i2c_data( GPIOin);
|
|
||||||
if (value & SENSE_DATA_HI )
|
|
||||||
return( FALSE );
|
|
||||||
|
|
||||||
return( TRUE );
|
|
||||||
}
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// Name: READ_I2C_REG
|
|
||||||
//
|
|
||||||
// This routine reads the I2C control register using the global
|
|
||||||
// IO address stored in gpioreg.
|
|
||||||
//
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
static UCHAR tl_read_i2c_data( void* gpioreg )
|
|
||||||
{
|
|
||||||
return( (UCHAR)(readl( gpioreg ) & 0x08L) ); // GPIO3
|
|
||||||
}
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// Name: WRITE_I2C_REG
|
|
||||||
//
|
|
||||||
// This routine writes the I2C control register using the global
|
|
||||||
// IO address stored in gpioreg.
|
|
||||||
// In Tachlite, we don't want to modify other bits in TL Control reg.
|
|
||||||
//
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
static void tl_write_i2c_reg( void* gpioregOUT, UCHAR value )
|
|
||||||
{
|
|
||||||
ULONG temp;
|
|
||||||
|
|
||||||
// First read the register and clear out the old bits
|
|
||||||
temp = readl( gpioregOUT ) & 0xfffffff3L;
|
|
||||||
|
|
||||||
// Now or in the new data and send it back out
|
|
||||||
writel( temp | value, gpioregOUT);
|
|
||||||
}
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// Name: I2C_TX_START
|
|
||||||
//
|
|
||||||
// This routine transmits a start condition over the I2C bus.
|
|
||||||
// 1. Set SCL (clock, GPIO2) HIGH, set SDA (data, GPIO3) HIGH,
|
|
||||||
// wait 5us to stabilize.
|
|
||||||
// 2. With SCL still HIGH, drive SDA low. The low transition marks
|
|
||||||
// the start condition to NM24Cxx (the chip)
|
|
||||||
// NOTE! In TL control reg., output 1 means chip sees LOW
|
|
||||||
//
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
static unsigned short tl_i2c_tx_start( void* GPIOin, void* GPIOout )
|
|
||||||
{
|
|
||||||
unsigned short i;
|
|
||||||
ULONG value;
|
|
||||||
|
|
||||||
if ( !(tl_read_i2c_data(GPIOin) & SENSE_DATA_HI))
|
|
||||||
{
|
|
||||||
// start with clock high, let data float high
|
|
||||||
tl_write_i2c_reg( GPIOout, SET_DATA_HI | SET_CLOCK_HI );
|
|
||||||
|
|
||||||
// keep sending clock pulses if slave is driving data line
|
|
||||||
for (i = 0; i < 10; i++)
|
|
||||||
{
|
|
||||||
tl_i2c_clock_pulse( SET_DATA_HI, GPIOout );
|
|
||||||
|
|
||||||
if ( tl_read_i2c_data(GPIOin) & SENSE_DATA_HI )
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// if he's still driving data low after 10 clocks, abort
|
|
||||||
value = tl_read_i2c_data( GPIOin ); // read status
|
|
||||||
if (!(value & 0x08) )
|
|
||||||
return( FALSE );
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// To START, bring data low while clock high
|
|
||||||
tl_write_i2c_reg( GPIOout, SET_CLOCK_HI | SET_DATA_LO );
|
|
||||||
|
|
||||||
i2c_delay(0);
|
|
||||||
|
|
||||||
return( TRUE ); // TX start successful
|
|
||||||
}
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// Name: I2C_TX_STOP
|
|
||||||
//
|
|
||||||
// This routine transmits a stop condition over the I2C bus.
|
|
||||||
//
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
|
|
||||||
static unsigned short tl_i2c_tx_stop( void* GPIOin, void* GPIOout )
|
|
||||||
{
|
|
||||||
int i;
|
|
||||||
|
|
||||||
for (i = 0; i < 10; i++)
|
|
||||||
{
|
|
||||||
// Send clock pulse, drive data line low
|
|
||||||
tl_i2c_clock_pulse( SET_DATA_LO, GPIOout );
|
|
||||||
|
|
||||||
// To STOP, bring data high while clock high
|
|
||||||
tl_write_i2c_reg( GPIOout, SET_DATA_HI | SET_CLOCK_HI );
|
|
||||||
|
|
||||||
// Give the data line time to float high
|
|
||||||
i2c_delay(0);
|
|
||||||
|
|
||||||
// If slave is driving data line low, there's a problem; retry
|
|
||||||
if ( tl_read_i2c_data(GPIOin) & SENSE_DATA_HI )
|
|
||||||
return( TRUE ); // TX STOP successful!
|
|
||||||
}
|
|
||||||
|
|
||||||
return( FALSE ); // error
|
|
||||||
}
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// Name: I2C_TX_uchar
|
|
||||||
//
|
|
||||||
// This routine transmits a byte across the I2C bus.
|
|
||||||
//
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
static void tl_i2c_tx_byte( void* GPIOout, UCHAR data )
|
|
||||||
{
|
|
||||||
UCHAR bit;
|
|
||||||
|
|
||||||
for (bit = 0x80; bit; bit >>= 1)
|
|
||||||
{
|
|
||||||
if( data & bit )
|
|
||||||
tl_i2c_clock_pulse( (UCHAR)SET_DATA_HI, GPIOout);
|
|
||||||
else
|
|
||||||
tl_i2c_clock_pulse( (UCHAR)SET_DATA_LO, GPIOout);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
//
|
|
||||||
// Name: I2C_RX_uchar
|
|
||||||
//
|
|
||||||
// This routine receives a byte across the I2C bus.
|
|
||||||
//
|
|
||||||
//-----------------------------------------------------------------------------
|
|
||||||
static UCHAR tl_i2c_rx_byte( void* GPIOin, void* GPIOout )
|
|
||||||
{
|
|
||||||
UCHAR bit;
|
|
||||||
UCHAR data = 0;
|
|
||||||
|
|
||||||
|
|
||||||
for (bit = 0x80; bit; bit >>= 1) {
|
|
||||||
// do clock pulse, let data line float high
|
|
||||||
tl_i2c_clock_pulse( SET_DATA_HI, GPIOout );
|
|
||||||
|
|
||||||
// read data line
|
|
||||||
if ( tl_read_i2c_data( GPIOin) & 0x08 )
|
|
||||||
data |= bit;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (data);
|
|
||||||
}
|
|
||||||
//*****************************************************************************
|
|
||||||
//*****************************************************************************
|
|
||||||
// Function: read_i2c_nvram
|
|
||||||
// Arguments: UCHAR count number of bytes to read
|
|
||||||
// UCHAR *buf area to store the bytes read
|
|
||||||
// Returns: 0 - failed
|
|
||||||
// 1 - success
|
|
||||||
//*****************************************************************************
|
|
||||||
//*****************************************************************************
|
|
||||||
unsigned long cpqfcTS_ReadNVRAM( void* GPIOin, void* GPIOout , USHORT count,
|
|
||||||
UCHAR *buf )
|
|
||||||
{
|
|
||||||
unsigned short i;
|
|
||||||
|
|
||||||
if( !( tl_i2c_tx_start(GPIOin, GPIOout) ))
|
|
||||||
return FALSE;
|
|
||||||
|
|
||||||
// Select the NVRAM for "dummy" write, to set the address
|
|
||||||
tl_i2c_tx_byte( GPIOout , SLAVE_WRITE_ADDRESS );
|
|
||||||
if ( !tl_i2c_rx_ack(GPIOin, GPIOout ) )
|
|
||||||
return( FALSE );
|
|
||||||
|
|
||||||
// Now send the address where we want to start reading
|
|
||||||
tl_i2c_tx_byte( GPIOout , 0 );
|
|
||||||
if ( !tl_i2c_rx_ack(GPIOin, GPIOout ) )
|
|
||||||
return( FALSE );
|
|
||||||
|
|
||||||
// Send a repeated start condition and select the
|
|
||||||
// slave for reading now.
|
|
||||||
if( tl_i2c_tx_start(GPIOin, GPIOout) )
|
|
||||||
tl_i2c_tx_byte( GPIOout, SLAVE_READ_ADDRESS );
|
|
||||||
|
|
||||||
if ( !tl_i2c_rx_ack(GPIOin, GPIOout) )
|
|
||||||
return( FALSE );
|
|
||||||
|
|
||||||
// this loop will now read out the data and store it
|
|
||||||
// in the buffer pointed to by buf
|
|
||||||
for ( i=0; i<count; i++)
|
|
||||||
{
|
|
||||||
*buf++ = tl_i2c_rx_byte(GPIOin, GPIOout);
|
|
||||||
|
|
||||||
// Send ACK by holding data line low for 1 clock
|
|
||||||
if ( i < (count-1) )
|
|
||||||
tl_i2c_clock_pulse( 0x08, GPIOout );
|
|
||||||
else {
|
|
||||||
// Don't send ack for final byte
|
|
||||||
tl_i2c_clock_pulse( SET_DATA_HI, GPIOout );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
tl_i2c_tx_stop(GPIOin, GPIOout);
|
|
||||||
|
|
||||||
return( TRUE );
|
|
||||||
}
|
|
||||||
|
|
||||||
//****************************************************************
|
|
||||||
//
|
|
||||||
//
|
|
||||||
//
|
|
||||||
// routines to set and clear the data and clock bits
|
|
||||||
//
|
|
||||||
//
|
|
||||||
//
|
|
||||||
//****************************************************************
|
|
||||||
|
|
||||||
static void tl_set_clock(void* gpioreg)
|
|
||||||
{
|
|
||||||
ULONG ret_val;
|
|
||||||
|
|
||||||
ret_val = readl( gpioreg );
|
|
||||||
ret_val &= 0xffffffFBL; // clear GPIO2 (SCL)
|
|
||||||
writel( ret_val, gpioreg);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void tl_clr_clock(void* gpioreg)
|
|
||||||
{
|
|
||||||
ULONG ret_val;
|
|
||||||
|
|
||||||
ret_val = readl( gpioreg );
|
|
||||||
ret_val |= SET_CLOCK_LO;
|
|
||||||
writel( ret_val, gpioreg);
|
|
||||||
}
|
|
||||||
|
|
||||||
//*****************************************************************
|
|
||||||
//
|
|
||||||
//
|
|
||||||
// This routine will advance the clock by one period
|
|
||||||
//
|
|
||||||
//
|
|
||||||
//*****************************************************************
|
|
||||||
static void tl_i2c_clock_pulse( UCHAR value, void* GPIOout )
|
|
||||||
{
|
|
||||||
ULONG ret_val;
|
|
||||||
|
|
||||||
// clear the clock bit
|
|
||||||
tl_clr_clock( GPIOout );
|
|
||||||
|
|
||||||
i2c_delay(0);
|
|
||||||
|
|
||||||
|
|
||||||
// read the port to preserve non-I2C bits
|
|
||||||
ret_val = readl( GPIOout );
|
|
||||||
|
|
||||||
// clear the data & clock bits
|
|
||||||
ret_val &= 0xFFFFFFf3;
|
|
||||||
|
|
||||||
// write the value passed in...
|
|
||||||
// data can only change while clock is LOW!
|
|
||||||
ret_val |= value; // the data
|
|
||||||
ret_val |= SET_CLOCK_LO; // the clock
|
|
||||||
writel( ret_val, GPIOout );
|
|
||||||
|
|
||||||
i2c_delay(0);
|
|
||||||
|
|
||||||
|
|
||||||
//set clock bit
|
|
||||||
tl_set_clock( GPIOout);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//*****************************************************************
|
|
||||||
//
|
|
||||||
//
|
|
||||||
// This routine returns the 64-bit WWN
|
|
||||||
//
|
|
||||||
//
|
|
||||||
//*****************************************************************
|
|
||||||
int cpqfcTS_GetNVRAM_data( UCHAR *wwnbuf, UCHAR *buf )
|
|
||||||
{
|
|
||||||
ULONG len;
|
|
||||||
ULONG sub_len;
|
|
||||||
ULONG ptr_inc;
|
|
||||||
ULONG i;
|
|
||||||
ULONG j;
|
|
||||||
UCHAR *data_ptr;
|
|
||||||
UCHAR z;
|
|
||||||
UCHAR name;
|
|
||||||
UCHAR sub_name;
|
|
||||||
UCHAR done;
|
|
||||||
int iReturn=0; // def. 0 offset is failure to find WWN field
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
data_ptr = (UCHAR *)buf;
|
|
||||||
|
|
||||||
done = FALSE;
|
|
||||||
i = 0;
|
|
||||||
|
|
||||||
while ( (i < 128) && (!done) )
|
|
||||||
{
|
|
||||||
z = data_ptr[i];\
|
|
||||||
if ( !(z & 0x80) )
|
|
||||||
{
|
|
||||||
len = 1 + (z & 0x07);
|
|
||||||
|
|
||||||
name = (z & 0x78) >> 3;
|
|
||||||
if (name == 0x0F)
|
|
||||||
done = TRUE;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
name = z & 0x7F;
|
|
||||||
len = 3 + data_ptr[i+1] + (data_ptr[i+2] << 8);
|
|
||||||
|
|
||||||
switch (name)
|
|
||||||
{
|
|
||||||
case 0x0D:
|
|
||||||
//
|
|
||||||
j = i + 3;
|
|
||||||
//
|
|
||||||
if ( data_ptr[j] == 0x3b ) {
|
|
||||||
len = 6;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
while ( j<(i+len) ) {
|
|
||||||
sub_name = (data_ptr[j] & 0x3f);
|
|
||||||
sub_len = data_ptr[j+1] +
|
|
||||||
(data_ptr[j+2] << 8);
|
|
||||||
ptr_inc = sub_len + 3;
|
|
||||||
switch (sub_name)
|
|
||||||
{
|
|
||||||
case 0x3C:
|
|
||||||
memcpy( wwnbuf, &data_ptr[j+3], 8);
|
|
||||||
iReturn = j+3;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
j += ptr_inc;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
//
|
|
||||||
i += len;
|
|
||||||
} // end while
|
|
||||||
return iReturn;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// define a short 5 micro sec delay, and longer (ms) delay
|
|
||||||
|
|
||||||
static void i2c_delay(ULONG mstime)
|
|
||||||
{
|
|
||||||
ULONG i;
|
|
||||||
|
|
||||||
// NOTE: we only expect to use these delays when reading
|
|
||||||
// our adapter's NVRAM, which happens only during adapter reset.
|
|
||||||
// Delay technique from "Linux Device Drivers", A. Rubini
|
|
||||||
// (1st Ed.) pg 137.
|
|
||||||
|
|
||||||
// printk(" delay %lx ", mstime);
|
|
||||||
if( mstime ) // ms delay?
|
|
||||||
{
|
|
||||||
// delay technique
|
|
||||||
for( i=0; i < mstime; i++)
|
|
||||||
udelay(1000); // 1ms per loop
|
|
||||||
|
|
||||||
}
|
|
||||||
else // 5 micro sec delay
|
|
||||||
|
|
||||||
udelay( 5 ); // micro secs
|
|
||||||
|
|
||||||
// printk("done\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,94 +0,0 @@
|
||||||
// for user apps, make sure data size types are defined
|
|
||||||
// with
|
|
||||||
|
|
||||||
|
|
||||||
#define CCPQFCTS_IOC_MAGIC 'Z'
|
|
||||||
|
|
||||||
typedef struct
|
|
||||||
{
|
|
||||||
__u8 bus;
|
|
||||||
__u8 dev_fn;
|
|
||||||
__u32 board_id;
|
|
||||||
} cpqfc_pci_info_struct;
|
|
||||||
|
|
||||||
typedef __u32 DriverVer_type;
|
|
||||||
/*
|
|
||||||
typedef union
|
|
||||||
{
|
|
||||||
struct // Peripheral Unit Device
|
|
||||||
{
|
|
||||||
__u8 Bus:6;
|
|
||||||
__u8 Mode:2; // b00
|
|
||||||
__u8 Dev;
|
|
||||||
} PeripDev;
|
|
||||||
struct // Volume Set Address
|
|
||||||
{
|
|
||||||
__u8 DevMSB:6;
|
|
||||||
__u8 Mode:2; // b01
|
|
||||||
__u8 DevLSB;
|
|
||||||
} LogDev;
|
|
||||||
struct // Logical Unit Device (SCSI-3, SCC-2 defined)
|
|
||||||
{
|
|
||||||
__u8 Targ:6;
|
|
||||||
__u8 Mode:2; // b10
|
|
||||||
__u8 Dev:5;
|
|
||||||
__u8 Bus:3;
|
|
||||||
|
|
||||||
} LogUnit;
|
|
||||||
} SCSI3Addr_struct;
|
|
||||||
|
|
||||||
|
|
||||||
typedef struct
|
|
||||||
{
|
|
||||||
SCSI3Addr_struct FCP_Nexus;
|
|
||||||
__u8 cdb[16];
|
|
||||||
} PassThru_Command_struct;
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* this is nearly duplicated in idashare.h */
|
|
||||||
typedef struct {
|
|
||||||
int lc; /* Controller number */
|
|
||||||
int node; /* Node (box) number */
|
|
||||||
int ld; /* Logical Drive on this box, if required */
|
|
||||||
__u32 nexus; /* SCSI Nexus */
|
|
||||||
void *argp; /* Argument pointer */
|
|
||||||
} VENDOR_IOCTL_REQ;
|
|
||||||
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
char cdb[16]; /* SCSI CDB for the pass-through */
|
|
||||||
ushort bus; /* Target bus on the box */
|
|
||||||
ushort pdrive; /* Physical drive on the box */
|
|
||||||
int len; /* Length of the data area of the CDB */
|
|
||||||
int sense_len; /* Length of the sense data */
|
|
||||||
char sense_data[40]; /* Sense data */
|
|
||||||
void *bufp; /* Data area for the CDB */
|
|
||||||
char rw_flag; /* Read CDB or Write CDB */
|
|
||||||
} cpqfc_passthru_t;
|
|
||||||
|
|
||||||
/*
|
|
||||||
** Defines for the IOCTLS.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#define VENDOR_READ_OPCODE 0x26
|
|
||||||
#define VENDOR_WRITE_OPCODE 0x27
|
|
||||||
|
|
||||||
#define CPQFCTS_GETPCIINFO _IOR( CCPQFCTS_IOC_MAGIC, 1, cpqfc_pci_info_struct)
|
|
||||||
#define CPQFCTS_GETDRIVVER _IOR( CCPQFCTS_IOC_MAGIC, 9, DriverVer_type)
|
|
||||||
|
|
||||||
#define CPQFCTS_SCSI_PASSTHRU _IOWR( CCPQFCTS_IOC_MAGIC,11, VENDOR_IOCTL_REQ)
|
|
||||||
|
|
||||||
/* We would rather have equivalent generic, low-level driver agnostic
|
|
||||||
ioctls that do what CPQFC_IOCTL_FC_TARGET_ADDRESS and
|
|
||||||
CPQFC_IOCTL_FC_TDR 0x5388 do, but currently, we do not have them,
|
|
||||||
consequently applications would have to know they are talking to cpqfc. */
|
|
||||||
|
|
||||||
/* Used to get Fibre Channel WWN and port_id from device */
|
|
||||||
// #define CPQFC_IOCTL_FC_TARGET_ADDRESS 0x5387
|
|
||||||
#define CPQFC_IOCTL_FC_TARGET_ADDRESS \
|
|
||||||
_IOR( CCPQFCTS_IOC_MAGIC, 13, Scsi_FCTargAddress)
|
|
||||||
|
|
||||||
/* Used to invoke Target Defice Reset for Fibre Channel */
|
|
||||||
// #define CPQFC_IOCTL_FC_TDR 0x5388
|
|
||||||
#define CPQFC_IOCTL_FC_TDR _IO( CCPQFCTS_IOC_MAGIC, 15)
|
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,33 +0,0 @@
|
||||||
// Routine to trigger Finisar GTA analyzer. Runs of GPIO2
|
|
||||||
// NOTE: DEBUG ONLY! Could interfere with FCMNGR/Miniport operation
|
|
||||||
// since it writes directly to the Tachyon board. This function
|
|
||||||
// developed for Compaq HBA Tachyon TS v1.2 (Rev X5 PCB)
|
|
||||||
|
|
||||||
#include "cpqfcTStrigger.h"
|
|
||||||
#if TRIGGERABLE_HBA
|
|
||||||
|
|
||||||
#include <linux/kernel.h>
|
|
||||||
#include <linux/ioport.h>
|
|
||||||
#include <linux/types.h>
|
|
||||||
#include <linux/pci.h>
|
|
||||||
#include <asm/io.h>
|
|
||||||
|
|
||||||
void TriggerHBA( void* IOBaseUpper, int Print)
|
|
||||||
{
|
|
||||||
__u32 long value;
|
|
||||||
|
|
||||||
// get initial value in hopes of not modifying any other GPIO line
|
|
||||||
IOBaseUpper += 0x188; // TachTL/TS Control reg
|
|
||||||
|
|
||||||
value = readl( IOBaseUpper);
|
|
||||||
// set HIGH to trigger external analyzer (tested on Dolche Finisar 1Gb GTA)
|
|
||||||
// The Finisar anaylzer triggers on low-to-high TTL transition
|
|
||||||
value |= 0x01; // set bit 0
|
|
||||||
|
|
||||||
writel( value, IOBaseUpper);
|
|
||||||
|
|
||||||
if( Print)
|
|
||||||
printk( " -GPIO0 set- ");
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -1,8 +0,0 @@
|
||||||
// don't do this unless you have the right hardware!
|
|
||||||
#define TRIGGERABLE_HBA 0
|
|
||||||
#if TRIGGERABLE_HBA
|
|
||||||
void TriggerHBA( void*, int);
|
|
||||||
#else
|
|
||||||
#define TriggerHBA(x, y)
|
|
||||||
#endif
|
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -976,6 +976,16 @@ static void send_srb(struct AdapterCtlBlk *acb, struct ScsiReqBlk *srb)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void pio_trigger(void)
|
||||||
|
{
|
||||||
|
static int feedback_requested;
|
||||||
|
|
||||||
|
if (!feedback_requested) {
|
||||||
|
feedback_requested = 1;
|
||||||
|
printk(KERN_WARNING "%s: Please, contact <linux-scsi@vger.kernel.org> "
|
||||||
|
"to help improve support for your system.\n", __FILE__);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Prepare SRB for being sent to Device DCB w/ command *cmd */
|
/* Prepare SRB for being sent to Device DCB w/ command *cmd */
|
||||||
static void build_srb(struct scsi_cmnd *cmd, struct DeviceCtlBlk *dcb,
|
static void build_srb(struct scsi_cmnd *cmd, struct DeviceCtlBlk *dcb,
|
||||||
|
@ -2320,6 +2330,7 @@ static void data_in_phase0(struct AdapterCtlBlk *acb, struct ScsiReqBlk *srb,
|
||||||
CFG2_WIDEFIFO);
|
CFG2_WIDEFIFO);
|
||||||
while (DC395x_read8(acb, TRM_S1040_SCSI_FIFOCNT) != 0x40) {
|
while (DC395x_read8(acb, TRM_S1040_SCSI_FIFOCNT) != 0x40) {
|
||||||
u8 byte = DC395x_read8(acb, TRM_S1040_SCSI_FIFO);
|
u8 byte = DC395x_read8(acb, TRM_S1040_SCSI_FIFO);
|
||||||
|
pio_trigger();
|
||||||
*(srb->virt_addr)++ = byte;
|
*(srb->virt_addr)++ = byte;
|
||||||
if (debug_enabled(DBG_PIO))
|
if (debug_enabled(DBG_PIO))
|
||||||
printk(" %02x", byte);
|
printk(" %02x", byte);
|
||||||
|
@ -2331,6 +2342,7 @@ static void data_in_phase0(struct AdapterCtlBlk *acb, struct ScsiReqBlk *srb,
|
||||||
/* Read the last byte ... */
|
/* Read the last byte ... */
|
||||||
if (srb->total_xfer_length > 0) {
|
if (srb->total_xfer_length > 0) {
|
||||||
u8 byte = DC395x_read8(acb, TRM_S1040_SCSI_FIFO);
|
u8 byte = DC395x_read8(acb, TRM_S1040_SCSI_FIFO);
|
||||||
|
pio_trigger();
|
||||||
*(srb->virt_addr)++ = byte;
|
*(srb->virt_addr)++ = byte;
|
||||||
srb->total_xfer_length--;
|
srb->total_xfer_length--;
|
||||||
if (debug_enabled(DBG_PIO))
|
if (debug_enabled(DBG_PIO))
|
||||||
|
@ -2507,6 +2519,7 @@ static void data_io_transfer(struct AdapterCtlBlk *acb,
|
||||||
if (debug_enabled(DBG_PIO))
|
if (debug_enabled(DBG_PIO))
|
||||||
printk(" %02x", (unsigned char) *(srb->virt_addr));
|
printk(" %02x", (unsigned char) *(srb->virt_addr));
|
||||||
|
|
||||||
|
pio_trigger();
|
||||||
DC395x_write8(acb, TRM_S1040_SCSI_FIFO,
|
DC395x_write8(acb, TRM_S1040_SCSI_FIFO,
|
||||||
*(srb->virt_addr)++);
|
*(srb->virt_addr)++);
|
||||||
|
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
* 20001005 - Initialization fixes for 2.4.0-test9
|
* 20001005 - Initialization fixes for 2.4.0-test9
|
||||||
* Florian Lohoff <flo@rfc822.org>
|
* Florian Lohoff <flo@rfc822.org>
|
||||||
*
|
*
|
||||||
* Copyright (C) 2002, 2003 Maciej W. Rozycki
|
* Copyright (C) 2002, 2003, 2005 Maciej W. Rozycki
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
|
@ -41,6 +41,7 @@
|
||||||
#include <asm/dec/ioasic_addrs.h>
|
#include <asm/dec/ioasic_addrs.h>
|
||||||
#include <asm/dec/ioasic_ints.h>
|
#include <asm/dec/ioasic_ints.h>
|
||||||
#include <asm/dec/machtype.h>
|
#include <asm/dec/machtype.h>
|
||||||
|
#include <asm/dec/system.h>
|
||||||
#include <asm/dec/tc.h>
|
#include <asm/dec/tc.h>
|
||||||
|
|
||||||
#define DEC_SCSI_SREG 0
|
#define DEC_SCSI_SREG 0
|
||||||
|
@ -183,7 +184,8 @@ static int dec_esp_detect(Scsi_Host_Template * tpnt)
|
||||||
esp->dregs = 0;
|
esp->dregs = 0;
|
||||||
|
|
||||||
/* ESP register base */
|
/* ESP register base */
|
||||||
esp->eregs = (struct ESP_regs *) (system_base + IOASIC_SCSI);
|
esp->eregs = (void *)CKSEG1ADDR(dec_kn_slot_base +
|
||||||
|
IOASIC_SCSI);
|
||||||
|
|
||||||
/* Set the command buffer */
|
/* Set the command buffer */
|
||||||
esp->esp_command = (volatile unsigned char *) cmd_buffer;
|
esp->esp_command = (volatile unsigned char *) cmd_buffer;
|
||||||
|
@ -231,7 +233,8 @@ static int dec_esp_detect(Scsi_Host_Template * tpnt)
|
||||||
esp->slot = CPHYSADDR(mem_start);
|
esp->slot = CPHYSADDR(mem_start);
|
||||||
|
|
||||||
esp->dregs = 0;
|
esp->dregs = 0;
|
||||||
esp->eregs = (struct ESP_regs *) (mem_start + DEC_SCSI_SREG);
|
esp->eregs = (void *)CKSEG1ADDR(mem_start +
|
||||||
|
DEC_SCSI_SREG);
|
||||||
esp->do_pio_cmds = 1;
|
esp->do_pio_cmds = 1;
|
||||||
|
|
||||||
/* Set the command buffer */
|
/* Set the command buffer */
|
||||||
|
@ -513,14 +516,15 @@ static void dma_advance_sg(struct scsi_cmnd * sp)
|
||||||
static void pmaz_dma_drain(struct NCR_ESP *esp)
|
static void pmaz_dma_drain(struct NCR_ESP *esp)
|
||||||
{
|
{
|
||||||
memcpy(phys_to_virt(esp_virt_buffer),
|
memcpy(phys_to_virt(esp_virt_buffer),
|
||||||
(void *)KSEG1ADDR(esp->slot + DEC_SCSI_SRAM + ESP_TGT_DMA_SIZE),
|
(void *)CKSEG1ADDR(esp->slot + DEC_SCSI_SRAM +
|
||||||
scsi_current_length);
|
ESP_TGT_DMA_SIZE),
|
||||||
|
scsi_current_length);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void pmaz_dma_init_read(struct NCR_ESP *esp, u32 vaddress, int length)
|
static void pmaz_dma_init_read(struct NCR_ESP *esp, u32 vaddress, int length)
|
||||||
{
|
{
|
||||||
volatile u32 *dmareg =
|
volatile u32 *dmareg =
|
||||||
(volatile u32 *)KSEG1ADDR(esp->slot + DEC_SCSI_DMAREG);
|
(volatile u32 *)CKSEG1ADDR(esp->slot + DEC_SCSI_DMAREG);
|
||||||
|
|
||||||
if (length > ESP_TGT_DMA_SIZE)
|
if (length > ESP_TGT_DMA_SIZE)
|
||||||
length = ESP_TGT_DMA_SIZE;
|
length = ESP_TGT_DMA_SIZE;
|
||||||
|
@ -536,9 +540,10 @@ static void pmaz_dma_init_read(struct NCR_ESP *esp, u32 vaddress, int length)
|
||||||
static void pmaz_dma_init_write(struct NCR_ESP *esp, u32 vaddress, int length)
|
static void pmaz_dma_init_write(struct NCR_ESP *esp, u32 vaddress, int length)
|
||||||
{
|
{
|
||||||
volatile u32 *dmareg =
|
volatile u32 *dmareg =
|
||||||
(volatile u32 *)KSEG1ADDR(esp->slot + DEC_SCSI_DMAREG);
|
(volatile u32 *)CKSEG1ADDR(esp->slot + DEC_SCSI_DMAREG);
|
||||||
|
|
||||||
memcpy((void *)KSEG1ADDR(esp->slot + DEC_SCSI_SRAM + ESP_TGT_DMA_SIZE),
|
memcpy((void *)CKSEG1ADDR(esp->slot + DEC_SCSI_SRAM +
|
||||||
|
ESP_TGT_DMA_SIZE),
|
||||||
phys_to_virt(vaddress), length);
|
phys_to_virt(vaddress), length);
|
||||||
|
|
||||||
wmb();
|
wmb();
|
||||||
|
|
|
@ -941,8 +941,6 @@ static int eata2x_slave_configure(struct scsi_device *dev)
|
||||||
{
|
{
|
||||||
int tqd, utqd;
|
int tqd, utqd;
|
||||||
char *tag_suffix, *link_suffix;
|
char *tag_suffix, *link_suffix;
|
||||||
struct Scsi_Host *shost = dev->host;
|
|
||||||
struct hostdata *ha = (struct hostdata *)shost->hostdata;
|
|
||||||
|
|
||||||
utqd = MAX_CMD_PER_LUN;
|
utqd = MAX_CMD_PER_LUN;
|
||||||
tqd = max_queue_depth;
|
tqd = max_queue_depth;
|
||||||
|
@ -973,8 +971,8 @@ static int eata2x_slave_configure(struct scsi_device *dev)
|
||||||
else
|
else
|
||||||
link_suffix = "";
|
link_suffix = "";
|
||||||
|
|
||||||
printk("%s: scsi%d, channel %d, id %d, lun %d, cmds/lun %d%s%s.\n",
|
sdev_printk(KERN_INFO, dev,
|
||||||
ha->board_name, shost->host_no, dev->channel, dev->id, dev->lun,
|
"cmds/lun %d%s%s.\n",
|
||||||
dev->queue_depth, link_suffix, tag_suffix);
|
dev->queue_depth, link_suffix, tag_suffix);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -1813,9 +1811,8 @@ static int eata2x_queuecommand(struct scsi_cmnd *SCpnt,
|
||||||
SCpnt->host_scribble = (unsigned char *)&cpp->cpp_index;
|
SCpnt->host_scribble = (unsigned char *)&cpp->cpp_index;
|
||||||
|
|
||||||
if (do_trace)
|
if (do_trace)
|
||||||
printk("%s: qcomm, mbox %d, target %d.%d:%d, pid %ld.\n",
|
scmd_printk(KERN_INFO, SCpnt,
|
||||||
ha->board_name, i, SCpnt->device->channel, SCpnt->device->id,
|
"qcomm, mbox %d, pid %ld.\n", i, SCpnt->pid);
|
||||||
SCpnt->device->lun, SCpnt->pid);
|
|
||||||
|
|
||||||
cpp->reqsen = 1;
|
cpp->reqsen = 1;
|
||||||
cpp->dispri = 1;
|
cpp->dispri = 1;
|
||||||
|
@ -1847,9 +1844,8 @@ static int eata2x_queuecommand(struct scsi_cmnd *SCpnt,
|
||||||
if (do_dma(shost->io_port, cpp->cp_dma_addr, SEND_CP_DMA)) {
|
if (do_dma(shost->io_port, cpp->cp_dma_addr, SEND_CP_DMA)) {
|
||||||
unmap_dma(i, ha);
|
unmap_dma(i, ha);
|
||||||
SCpnt->host_scribble = NULL;
|
SCpnt->host_scribble = NULL;
|
||||||
printk("%s: qcomm, target %d.%d:%d, pid %ld, adapter busy.\n",
|
scmd_printk(KERN_INFO, SCpnt,
|
||||||
ha->board_name, SCpnt->device->channel, SCpnt->device->id,
|
"qcomm, pid %ld, adapter busy.\n", SCpnt->pid);
|
||||||
SCpnt->device->lun, SCpnt->pid);
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1864,16 +1860,14 @@ static int eata2x_eh_abort(struct scsi_cmnd *SCarg)
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
|
|
||||||
if (SCarg->host_scribble == NULL) {
|
if (SCarg->host_scribble == NULL) {
|
||||||
printk("%s: abort, target %d.%d:%d, pid %ld inactive.\n",
|
scmd_printk(KERN_INFO, SCarg,
|
||||||
ha->board_name, SCarg->device->channel, SCarg->device->id,
|
"abort, pid %ld inactive.\n", SCarg->pid);
|
||||||
SCarg->device->lun, SCarg->pid);
|
|
||||||
return SUCCESS;
|
return SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
i = *(unsigned int *)SCarg->host_scribble;
|
i = *(unsigned int *)SCarg->host_scribble;
|
||||||
printk("%s: abort, mbox %d, target %d.%d:%d, pid %ld.\n",
|
scmd_printk(KERN_WARNING, SCarg,
|
||||||
ha->board_name, i, SCarg->device->channel, SCarg->device->id,
|
"abort, mbox %d, pid %ld.\n", i, SCarg->pid);
|
||||||
SCarg->device->lun, SCarg->pid);
|
|
||||||
|
|
||||||
if (i >= shost->can_queue)
|
if (i >= shost->can_queue)
|
||||||
panic("%s: abort, invalid SCarg->host_scribble.\n", ha->board_name);
|
panic("%s: abort, invalid SCarg->host_scribble.\n", ha->board_name);
|
||||||
|
@ -1934,9 +1928,8 @@ static int eata2x_eh_host_reset(struct scsi_cmnd *SCarg)
|
||||||
struct Scsi_Host *shost = SCarg->device->host;
|
struct Scsi_Host *shost = SCarg->device->host;
|
||||||
struct hostdata *ha = (struct hostdata *)shost->hostdata;
|
struct hostdata *ha = (struct hostdata *)shost->hostdata;
|
||||||
|
|
||||||
printk("%s: reset, enter, target %d.%d:%d, pid %ld.\n",
|
scmd_printk(KERN_INFO, SCarg,
|
||||||
ha->board_name, SCarg->device->channel, SCarg->device->id,
|
"reset, enter, pid %ld.\n", SCarg->pid);
|
||||||
SCarg->device->lun, SCarg->pid);
|
|
||||||
|
|
||||||
spin_lock_irq(shost->host_lock);
|
spin_lock_irq(shost->host_lock);
|
||||||
|
|
||||||
|
@ -2253,12 +2246,11 @@ static int reorder(struct hostdata *ha, unsigned long cursec,
|
||||||
k = il[n];
|
k = il[n];
|
||||||
cpp = &ha->cp[k];
|
cpp = &ha->cp[k];
|
||||||
SCpnt = cpp->SCpnt;
|
SCpnt = cpp->SCpnt;
|
||||||
printk
|
scmd_printk(KERN_INFO, SCpnt,
|
||||||
("%s %d.%d:%d pid %ld mb %d fc %d nr %d sec %ld ns %ld"
|
"%s pid %ld mb %d fc %d nr %d sec %ld ns %ld"
|
||||||
" cur %ld s:%c r:%c rev:%c in:%c ov:%c xd %d.\n",
|
" cur %ld s:%c r:%c rev:%c in:%c ov:%c xd %d.\n",
|
||||||
(ihdlr ? "ihdlr" : "qcomm"),
|
(ihdlr ? "ihdlr" : "qcomm"),
|
||||||
SCpnt->device->channel, SCpnt->device->id,
|
SCpnt->pid, k, flushcount,
|
||||||
SCpnt->device->lun, SCpnt->pid, k, flushcount,
|
|
||||||
n_ready, SCpnt->request->sector,
|
n_ready, SCpnt->request->sector,
|
||||||
SCpnt->request->nr_sectors, cursec, YESNO(s),
|
SCpnt->request->nr_sectors, cursec, YESNO(s),
|
||||||
YESNO(r), YESNO(rev), YESNO(input_only),
|
YESNO(r), YESNO(rev), YESNO(input_only),
|
||||||
|
@ -2301,12 +2293,11 @@ static void flush_dev(struct scsi_device *dev, unsigned long cursec,
|
||||||
SCpnt = cpp->SCpnt;
|
SCpnt = cpp->SCpnt;
|
||||||
|
|
||||||
if (do_dma(dev->host->io_port, cpp->cp_dma_addr, SEND_CP_DMA)) {
|
if (do_dma(dev->host->io_port, cpp->cp_dma_addr, SEND_CP_DMA)) {
|
||||||
printk
|
scmd_printk(KERN_INFO, SCpnt,
|
||||||
("%s: %s, target %d.%d:%d, pid %ld, mbox %d, adapter"
|
"%s, pid %ld, mbox %d, adapter"
|
||||||
" busy, will abort.\n", ha->board_name,
|
" busy, will abort.\n",
|
||||||
(ihdlr ? "ihdlr" : "qcomm"),
|
(ihdlr ? "ihdlr" : "qcomm"),
|
||||||
SCpnt->device->channel, SCpnt->device->id,
|
SCpnt->pid, k);
|
||||||
SCpnt->device->lun, SCpnt->pid, k);
|
|
||||||
ha->cp_stat[k] = ABORTING;
|
ha->cp_stat[k] = ABORTING;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -2542,11 +2533,10 @@ static irqreturn_t ihdlr(int irq, struct Scsi_Host *shost)
|
||||||
spp->adapter_status != ASST && ha->iocount <= 1000) ||
|
spp->adapter_status != ASST && ha->iocount <= 1000) ||
|
||||||
do_trace || msg_byte(spp->target_status))
|
do_trace || msg_byte(spp->target_status))
|
||||||
#endif
|
#endif
|
||||||
printk("%s: ihdlr, mbox %2d, err 0x%x:%x,"
|
scmd_printk(KERN_INFO, SCpnt, "ihdlr, mbox %2d, err 0x%x:%x,"
|
||||||
" target %d.%d:%d, pid %ld, reg 0x%x, count %d.\n",
|
" pid %ld, reg 0x%x, count %d.\n",
|
||||||
ha->board_name, i, spp->adapter_status, spp->target_status,
|
i, spp->adapter_status, spp->target_status,
|
||||||
SCpnt->device->channel, SCpnt->device->id,
|
SCpnt->pid, reg, ha->iocount);
|
||||||
SCpnt->device->lun, SCpnt->pid, reg, ha->iocount);
|
|
||||||
|
|
||||||
unmap_dma(i, ha);
|
unmap_dma(i, ha);
|
||||||
|
|
||||||
|
|
|
@ -384,7 +384,9 @@ static int eata_pio_queue(struct scsi_cmnd *cmd,
|
||||||
|
|
||||||
cp->status = USED; /* claim free slot */
|
cp->status = USED; /* claim free slot */
|
||||||
|
|
||||||
DBG(DBG_QUEUE, printk(KERN_DEBUG "eata_pio_queue pid %ld, target: %x, lun:" " %x, y %d\n", cmd->pid, cmd->device->id, cmd->device->lun, y));
|
DBG(DBG_QUEUE, scmd_printk(KERN_DEBUG, cmd,
|
||||||
|
"eata_pio_queue pid %ld, y %d\n",
|
||||||
|
cmd->pid, y));
|
||||||
|
|
||||||
cmd->scsi_done = (void *) done;
|
cmd->scsi_done = (void *) done;
|
||||||
|
|
||||||
|
@ -427,7 +429,9 @@ static int eata_pio_queue(struct scsi_cmnd *cmd,
|
||||||
|
|
||||||
if (eata_pio_send_command(base, EATA_CMD_PIO_SEND_CP)) {
|
if (eata_pio_send_command(base, EATA_CMD_PIO_SEND_CP)) {
|
||||||
cmd->result = DID_BUS_BUSY << 16;
|
cmd->result = DID_BUS_BUSY << 16;
|
||||||
printk(KERN_NOTICE "eata_pio_queue target %d, pid %ld, HBA busy, " "returning DID_BUS_BUSY, done.\n", cmd->device->id, cmd->pid);
|
scmd_printk(KERN_NOTICE, cmd,
|
||||||
|
"eata_pio_queue pid %ld, HBA busy, "
|
||||||
|
"returning DID_BUS_BUSY, done.\n", cmd->pid);
|
||||||
done(cmd);
|
done(cmd);
|
||||||
cp->status = FREE;
|
cp->status = FREE;
|
||||||
return (0);
|
return (0);
|
||||||
|
@ -440,7 +444,9 @@ static int eata_pio_queue(struct scsi_cmnd *cmd,
|
||||||
for (x = 0; x < hd->cppadlen; x++)
|
for (x = 0; x < hd->cppadlen; x++)
|
||||||
outw(0, base + HA_RDATA);
|
outw(0, base + HA_RDATA);
|
||||||
|
|
||||||
DBG(DBG_QUEUE, printk(KERN_DEBUG "Queued base %#.4lx pid: %ld target: %x " "lun: %x slot %d irq %d\n", (long) sh->base, cmd->pid, cmd->device->id, cmd->device->lun, y, sh->irq));
|
DBG(DBG_QUEUE, scmd_printk(KERN_DEBUG, cmd,
|
||||||
|
"Queued base %#.4lx pid: %ld "
|
||||||
|
"slot %d irq %d\n", (long) sh->base, cmd->pid, y, sh->irq));
|
||||||
|
|
||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
@ -449,8 +455,9 @@ static int eata_pio_abort(struct scsi_cmnd *cmd)
|
||||||
{
|
{
|
||||||
uint loop = HZ;
|
uint loop = HZ;
|
||||||
|
|
||||||
DBG(DBG_ABNORM, printk(KERN_WARNING "eata_pio_abort called pid: %ld " "target: %x lun: %x\n", cmd->pid, cmd->device->id, cmd->device->lun));
|
DBG(DBG_ABNORM, scmd_printk(KERN_WARNING, cmd,
|
||||||
|
"eata_pio_abort called pid: %ld\n",
|
||||||
|
cmd->pid));
|
||||||
|
|
||||||
while (inb(cmd->device->host->base + HA_RAUXSTAT) & HA_ABUSY)
|
while (inb(cmd->device->host->base + HA_RAUXSTAT) & HA_ABUSY)
|
||||||
if (--loop == 0) {
|
if (--loop == 0) {
|
||||||
|
@ -484,7 +491,9 @@ static int eata_pio_host_reset(struct scsi_cmnd *cmd)
|
||||||
struct scsi_cmnd *sp;
|
struct scsi_cmnd *sp;
|
||||||
struct Scsi_Host *host = cmd->device->host;
|
struct Scsi_Host *host = cmd->device->host;
|
||||||
|
|
||||||
DBG(DBG_ABNORM, printk(KERN_WARNING "eata_pio_reset called pid:%ld target:" " %x lun: %x\n", cmd->pid, cmd->device->id, cmd->device->lun));
|
DBG(DBG_ABNORM, scmd_printk(KERN_WARNING, cmd,
|
||||||
|
"eata_pio_reset called pid:%ld\n",
|
||||||
|
cmd->pid));
|
||||||
|
|
||||||
spin_lock_irq(host->host_lock);
|
spin_lock_irq(host->host_lock);
|
||||||
|
|
||||||
|
|
|
@ -671,7 +671,7 @@ static irqreturn_t fd_mcs_intr(int irq, void *dev_id, struct pt_regs *regs)
|
||||||
outb(0x40 | FIFO_COUNT, Interrupt_Cntl_port);
|
outb(0x40 | FIFO_COUNT, Interrupt_Cntl_port);
|
||||||
|
|
||||||
outb(0x82, SCSI_Cntl_port); /* Bus Enable + Select */
|
outb(0x82, SCSI_Cntl_port); /* Bus Enable + Select */
|
||||||
outb(adapter_mask | (1 << current_SC->device->id), SCSI_Data_NoACK_port);
|
outb(adapter_mask | (1 << scmd_id(current_SC)), SCSI_Data_NoACK_port);
|
||||||
|
|
||||||
/* Stop arbitration and enable parity */
|
/* Stop arbitration and enable parity */
|
||||||
outb(0x10 | PARITY_MASK, TMC_Cntl_port);
|
outb(0x10 | PARITY_MASK, TMC_Cntl_port);
|
||||||
|
@ -683,7 +683,7 @@ static irqreturn_t fd_mcs_intr(int irq, void *dev_id, struct pt_regs *regs)
|
||||||
status = inb(SCSI_Status_port);
|
status = inb(SCSI_Status_port);
|
||||||
if (!(status & 0x01)) {
|
if (!(status & 0x01)) {
|
||||||
/* Try again, for slow devices */
|
/* Try again, for slow devices */
|
||||||
if (fd_mcs_select(shpnt, current_SC->device->id)) {
|
if (fd_mcs_select(shpnt, scmd_id(current_SC))) {
|
||||||
#if EVERY_ACCESS
|
#if EVERY_ACCESS
|
||||||
printk(" SFAIL ");
|
printk(" SFAIL ");
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1154,7 +1154,7 @@ static irqreturn_t do_fdomain_16x0_intr(int irq, void *dev_id,
|
||||||
outb(0x40 | FIFO_COUNT, port_base + Interrupt_Cntl);
|
outb(0x40 | FIFO_COUNT, port_base + Interrupt_Cntl);
|
||||||
|
|
||||||
outb(0x82, port_base + SCSI_Cntl); /* Bus Enable + Select */
|
outb(0x82, port_base + SCSI_Cntl); /* Bus Enable + Select */
|
||||||
outb(adapter_mask | (1 << current_SC->device->id), port_base + SCSI_Data_NoACK);
|
outb(adapter_mask | (1 << scmd_id(current_SC)), port_base + SCSI_Data_NoACK);
|
||||||
|
|
||||||
/* Stop arbitration and enable parity */
|
/* Stop arbitration and enable parity */
|
||||||
outb(0x10 | PARITY_MASK, port_base + TMC_Cntl);
|
outb(0x10 | PARITY_MASK, port_base + TMC_Cntl);
|
||||||
|
@ -1166,7 +1166,7 @@ static irqreturn_t do_fdomain_16x0_intr(int irq, void *dev_id,
|
||||||
status = inb(port_base + SCSI_Status);
|
status = inb(port_base + SCSI_Status);
|
||||||
if (!(status & 0x01)) {
|
if (!(status & 0x01)) {
|
||||||
/* Try again, for slow devices */
|
/* Try again, for slow devices */
|
||||||
if (fdomain_select( current_SC->device->id )) {
|
if (fdomain_select( scmd_id(current_SC) )) {
|
||||||
#if EVERY_ACCESS
|
#if EVERY_ACCESS
|
||||||
printk( " SFAIL " );
|
printk( " SFAIL " );
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -140,11 +140,11 @@ int scsi_host_set_state(struct Scsi_Host *shost, enum scsi_host_state state)
|
||||||
|
|
||||||
illegal:
|
illegal:
|
||||||
SCSI_LOG_ERROR_RECOVERY(1,
|
SCSI_LOG_ERROR_RECOVERY(1,
|
||||||
dev_printk(KERN_ERR, &shost->shost_gendev,
|
shost_printk(KERN_ERR, shost,
|
||||||
"Illegal host state transition"
|
"Illegal host state transition"
|
||||||
"%s->%s\n",
|
"%s->%s\n",
|
||||||
scsi_host_state_name(oldstate),
|
scsi_host_state_name(oldstate),
|
||||||
scsi_host_state_name(state)));
|
scsi_host_state_name(state)));
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL(scsi_host_set_state);
|
EXPORT_SYMBOL(scsi_host_set_state);
|
||||||
|
|
|
@ -1860,7 +1860,10 @@ static int ibmmca_queuecommand(Scsi_Cmnd * cmd, void (*done) (Scsi_Cmnd *))
|
||||||
next_ldn(host_index) = 7;
|
next_ldn(host_index) = 7;
|
||||||
if (current_ldn == next_ldn(host_index)) { /* One circle done ? */
|
if (current_ldn == next_ldn(host_index)) { /* One circle done ? */
|
||||||
/* no non-processing ldn found */
|
/* no non-processing ldn found */
|
||||||
printk("IBM MCA SCSI: Cannot assign SCSI-device dynamically!\n" " On ldn 7-14 SCSI-commands everywhere in progress.\n" " Reporting DID_NO_CONNECT for device (%d,%d).\n", target, cmd->device->lun);
|
scmd_printk(KERN_WARNING, cmd,
|
||||||
|
"IBM MCA SCSI: Cannot assign SCSI-device dynamically!\n"
|
||||||
|
" On ldn 7-14 SCSI-commands everywhere in progress.\n"
|
||||||
|
" Reporting DID_NO_CONNECT for device.\n");
|
||||||
cmd->result = DID_NO_CONNECT << 16; /* return no connect */
|
cmd->result = DID_NO_CONNECT << 16; /* return no connect */
|
||||||
if (done)
|
if (done)
|
||||||
done(cmd);
|
done(cmd);
|
||||||
|
|
|
@ -899,7 +899,7 @@ static int idescsi_queue (struct scsi_cmnd *cmd,
|
||||||
idescsi_pc_t *pc = NULL;
|
idescsi_pc_t *pc = NULL;
|
||||||
|
|
||||||
if (!drive) {
|
if (!drive) {
|
||||||
printk (KERN_ERR "ide-scsi: drive id %d not present\n", cmd->device->id);
|
scmd_printk (KERN_ERR, cmd, "drive not present\n");
|
||||||
goto abort;
|
goto abort;
|
||||||
}
|
}
|
||||||
scsi = drive_to_idescsi(drive);
|
scsi = drive_to_idescsi(drive);
|
||||||
|
|
|
@ -830,7 +830,7 @@ static int imm_engine(imm_struct *dev, struct scsi_cmnd *cmd)
|
||||||
|
|
||||||
/* Phase 2 - We are now talking to the scsi bus */
|
/* Phase 2 - We are now talking to the scsi bus */
|
||||||
case 2:
|
case 2:
|
||||||
if (!imm_select(dev, cmd->device->id)) {
|
if (!imm_select(dev, scmd_id(cmd))) {
|
||||||
imm_fail(dev, DID_NO_CONNECT);
|
imm_fail(dev, DID_NO_CONNECT);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -343,7 +343,7 @@ static int in2000_queuecommand(Scsi_Cmnd * cmd, void (*done) (Scsi_Cmnd *))
|
||||||
instance = cmd->device->host;
|
instance = cmd->device->host;
|
||||||
hostdata = (struct IN2000_hostdata *) instance->hostdata;
|
hostdata = (struct IN2000_hostdata *) instance->hostdata;
|
||||||
|
|
||||||
DB(DB_QUEUE_COMMAND, printk("Q-%d-%02x-%ld(", cmd->device->id, cmd->cmnd[0], cmd->pid))
|
DB(DB_QUEUE_COMMAND, scmd_printk(KERN_DEBUG, cmd, "Q-%02x-%ld(", cmd->cmnd[0], cmd->pid))
|
||||||
|
|
||||||
/* Set up a few fields in the Scsi_Cmnd structure for our own use:
|
/* Set up a few fields in the Scsi_Cmnd structure for our own use:
|
||||||
* - host_scribble is the pointer to the next cmd in the input queue
|
* - host_scribble is the pointer to the next cmd in the input queue
|
||||||
|
|
|
@ -1114,9 +1114,8 @@ struct ipr_ucode_image_header {
|
||||||
#define ipr_warn(...) printk(KERN_WARNING IPR_NAME": "__VA_ARGS__)
|
#define ipr_warn(...) printk(KERN_WARNING IPR_NAME": "__VA_ARGS__)
|
||||||
#define ipr_dbg(...) IPR_DBG_CMD(printk(KERN_INFO IPR_NAME ": "__VA_ARGS__))
|
#define ipr_dbg(...) IPR_DBG_CMD(printk(KERN_INFO IPR_NAME ": "__VA_ARGS__))
|
||||||
|
|
||||||
#define ipr_sdev_printk(level, sdev, fmt, ...) \
|
#define ipr_sdev_printk(level, sdev, fmt, args...) \
|
||||||
printk(level IPR_NAME ": %d:%d:%d:%d: " fmt, sdev->host->host_no, \
|
sdev_printk(level, sdev, fmt, ## args)
|
||||||
sdev->channel, sdev->id, sdev->lun, ##__VA_ARGS__)
|
|
||||||
|
|
||||||
#define ipr_sdev_err(sdev, fmt, ...) \
|
#define ipr_sdev_err(sdev, fmt, ...) \
|
||||||
ipr_sdev_printk(KERN_ERR, sdev, fmt, ##__VA_ARGS__)
|
ipr_sdev_printk(KERN_ERR, sdev, fmt, ##__VA_ARGS__)
|
||||||
|
|
|
@ -219,15 +219,12 @@ module_param(ips, charp, 0);
|
||||||
#if LINUX_VERSION_CODE <= KERNEL_VERSION(2,5,0)
|
#if LINUX_VERSION_CODE <= KERNEL_VERSION(2,5,0)
|
||||||
#include <linux/blk.h>
|
#include <linux/blk.h>
|
||||||
#include "sd.h"
|
#include "sd.h"
|
||||||
#define IPS_SG_ADDRESS(sg) ((sg)->address)
|
|
||||||
#define IPS_LOCK_SAVE(lock,flags) spin_lock_irqsave(&io_request_lock,flags)
|
#define IPS_LOCK_SAVE(lock,flags) spin_lock_irqsave(&io_request_lock,flags)
|
||||||
#define IPS_UNLOCK_RESTORE(lock,flags) spin_unlock_irqrestore(&io_request_lock,flags)
|
#define IPS_UNLOCK_RESTORE(lock,flags) spin_unlock_irqrestore(&io_request_lock,flags)
|
||||||
#ifndef __devexit_p
|
#ifndef __devexit_p
|
||||||
#define __devexit_p(x) x
|
#define __devexit_p(x) x
|
||||||
#endif
|
#endif
|
||||||
#else
|
#else
|
||||||
#define IPS_SG_ADDRESS(sg) (page_address((sg)->page) ? \
|
|
||||||
page_address((sg)->page)+(sg)->offset : NULL)
|
|
||||||
#define IPS_LOCK_SAVE(lock,flags) do{spin_lock(lock);(void)flags;}while(0)
|
#define IPS_LOCK_SAVE(lock,flags) do{spin_lock(lock);(void)flags;}while(0)
|
||||||
#define IPS_UNLOCK_RESTORE(lock,flags) do{spin_unlock(lock);(void)flags;}while(0)
|
#define IPS_UNLOCK_RESTORE(lock,flags) do{spin_unlock(lock);(void)flags;}while(0)
|
||||||
#endif
|
#endif
|
||||||
|
@ -358,6 +355,9 @@ static int ips_init_phase2(int index);
|
||||||
static int ips_init_phase1(struct pci_dev *pci_dev, int *indexPtr);
|
static int ips_init_phase1(struct pci_dev *pci_dev, int *indexPtr);
|
||||||
static int ips_register_scsi(int index);
|
static int ips_register_scsi(int index);
|
||||||
|
|
||||||
|
static int ips_poll_for_flush_complete(ips_ha_t * ha);
|
||||||
|
static void ips_flush_and_reset(ips_ha_t *ha);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* global variables
|
* global variables
|
||||||
*/
|
*/
|
||||||
|
@ -1125,8 +1125,8 @@ ips_queue(Scsi_Cmnd * SC, void (*done) (Scsi_Cmnd *))
|
||||||
SC->device->channel, SC->device->id, SC->device->lun);
|
SC->device->channel, SC->device->id, SC->device->lun);
|
||||||
|
|
||||||
/* Check for command to initiator IDs */
|
/* Check for command to initiator IDs */
|
||||||
if ((SC->device->channel > 0)
|
if ((scmd_channel(SC) > 0)
|
||||||
&& (SC->device->id == ha->ha_id[SC->device->channel])) {
|
&& (scmd_id(SC) == ha->ha_id[scmd_channel(SC)])) {
|
||||||
SC->result = DID_NO_CONNECT << 16;
|
SC->result = DID_NO_CONNECT << 16;
|
||||||
done(SC);
|
done(SC);
|
||||||
|
|
||||||
|
@ -1605,6 +1605,8 @@ ips_proc_info(struct Scsi_Host *host, char *buffer, char **start, off_t offset,
|
||||||
static int
|
static int
|
||||||
ips_is_passthru(Scsi_Cmnd * SC)
|
ips_is_passthru(Scsi_Cmnd * SC)
|
||||||
{
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
METHOD_TRACE("ips_is_passthru", 1);
|
METHOD_TRACE("ips_is_passthru", 1);
|
||||||
|
|
||||||
if (!SC)
|
if (!SC)
|
||||||
|
@ -1622,10 +1624,20 @@ ips_is_passthru(Scsi_Cmnd * SC)
|
||||||
return 1;
|
return 1;
|
||||||
else if (SC->use_sg) {
|
else if (SC->use_sg) {
|
||||||
struct scatterlist *sg = SC->request_buffer;
|
struct scatterlist *sg = SC->request_buffer;
|
||||||
char *buffer = IPS_SG_ADDRESS(sg);
|
char *buffer;
|
||||||
|
|
||||||
|
/* kmap_atomic() ensures addressability of the user buffer.*/
|
||||||
|
/* local_irq_save() protects the KM_IRQ0 address slot. */
|
||||||
|
local_irq_save(flags);
|
||||||
|
buffer = kmap_atomic(sg->page, KM_IRQ0) + sg->offset;
|
||||||
if (buffer && buffer[0] == 'C' && buffer[1] == 'O' &&
|
if (buffer && buffer[0] == 'C' && buffer[1] == 'O' &&
|
||||||
buffer[2] == 'P' && buffer[3] == 'P')
|
buffer[2] == 'P' && buffer[3] == 'P') {
|
||||||
|
kunmap_atomic(buffer - sg->offset, KM_IRQ0);
|
||||||
|
local_irq_restore(flags);
|
||||||
return 1;
|
return 1;
|
||||||
|
}
|
||||||
|
kunmap_atomic(buffer - sg->offset, KM_IRQ0);
|
||||||
|
local_irq_restore(flags);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -2830,10 +2842,10 @@ ips_next(ips_ha_t * ha, int intr)
|
||||||
|
|
||||||
p = ha->scb_waitlist.head;
|
p = ha->scb_waitlist.head;
|
||||||
while ((p) && (scb = ips_getscb(ha))) {
|
while ((p) && (scb = ips_getscb(ha))) {
|
||||||
if ((p->device->channel > 0)
|
if ((scmd_channel(p) > 0)
|
||||||
&& (ha->
|
&& (ha->
|
||||||
dcdb_active[p->device->channel -
|
dcdb_active[scmd_channel(p) -
|
||||||
1] & (1 << p->device->id))) {
|
1] & (1 << scmd_id(p)))) {
|
||||||
ips_freescb(ha, scb);
|
ips_freescb(ha, scb);
|
||||||
p = (Scsi_Cmnd *) p->host_scribble;
|
p = (Scsi_Cmnd *) p->host_scribble;
|
||||||
continue;
|
continue;
|
||||||
|
@ -3656,14 +3668,21 @@ ips_scmd_buf_write(Scsi_Cmnd * scmd, void *data, unsigned
|
||||||
int i;
|
int i;
|
||||||
unsigned int min_cnt, xfer_cnt;
|
unsigned int min_cnt, xfer_cnt;
|
||||||
char *cdata = (char *) data;
|
char *cdata = (char *) data;
|
||||||
|
unsigned char *buffer;
|
||||||
|
unsigned long flags;
|
||||||
struct scatterlist *sg = scmd->request_buffer;
|
struct scatterlist *sg = scmd->request_buffer;
|
||||||
for (i = 0, xfer_cnt = 0;
|
for (i = 0, xfer_cnt = 0;
|
||||||
(i < scmd->use_sg) && (xfer_cnt < count); i++) {
|
(i < scmd->use_sg) && (xfer_cnt < count); i++) {
|
||||||
if (!IPS_SG_ADDRESS(&sg[i]))
|
|
||||||
return;
|
|
||||||
min_cnt = min(count - xfer_cnt, sg[i].length);
|
min_cnt = min(count - xfer_cnt, sg[i].length);
|
||||||
memcpy(IPS_SG_ADDRESS(&sg[i]), &cdata[xfer_cnt],
|
|
||||||
min_cnt);
|
/* kmap_atomic() ensures addressability of the data buffer.*/
|
||||||
|
/* local_irq_save() protects the KM_IRQ0 address slot. */
|
||||||
|
local_irq_save(flags);
|
||||||
|
buffer = kmap_atomic(sg[i].page, KM_IRQ0) + sg[i].offset;
|
||||||
|
memcpy(buffer, &cdata[xfer_cnt], min_cnt);
|
||||||
|
kunmap_atomic(buffer - sg[i].offset, KM_IRQ0);
|
||||||
|
local_irq_restore(flags);
|
||||||
|
|
||||||
xfer_cnt += min_cnt;
|
xfer_cnt += min_cnt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3688,14 +3707,21 @@ ips_scmd_buf_read(Scsi_Cmnd * scmd, void *data, unsigned
|
||||||
int i;
|
int i;
|
||||||
unsigned int min_cnt, xfer_cnt;
|
unsigned int min_cnt, xfer_cnt;
|
||||||
char *cdata = (char *) data;
|
char *cdata = (char *) data;
|
||||||
|
unsigned char *buffer;
|
||||||
|
unsigned long flags;
|
||||||
struct scatterlist *sg = scmd->request_buffer;
|
struct scatterlist *sg = scmd->request_buffer;
|
||||||
for (i = 0, xfer_cnt = 0;
|
for (i = 0, xfer_cnt = 0;
|
||||||
(i < scmd->use_sg) && (xfer_cnt < count); i++) {
|
(i < scmd->use_sg) && (xfer_cnt < count); i++) {
|
||||||
if (!IPS_SG_ADDRESS(&sg[i]))
|
|
||||||
return;
|
|
||||||
min_cnt = min(count - xfer_cnt, sg[i].length);
|
min_cnt = min(count - xfer_cnt, sg[i].length);
|
||||||
memcpy(&cdata[xfer_cnt], IPS_SG_ADDRESS(&sg[i]),
|
|
||||||
min_cnt);
|
/* kmap_atomic() ensures addressability of the data buffer.*/
|
||||||
|
/* local_irq_save() protects the KM_IRQ0 address slot. */
|
||||||
|
local_irq_save(flags);
|
||||||
|
buffer = kmap_atomic(sg[i].page, KM_IRQ0) + sg[i].offset;
|
||||||
|
memcpy(&cdata[xfer_cnt], buffer, min_cnt);
|
||||||
|
kunmap_atomic(buffer - sg[i].offset, KM_IRQ0);
|
||||||
|
local_irq_restore(flags);
|
||||||
|
|
||||||
xfer_cnt += min_cnt;
|
xfer_cnt += min_cnt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4807,6 +4833,9 @@ ips_isinit_morpheus(ips_ha_t * ha)
|
||||||
uint32_t bits;
|
uint32_t bits;
|
||||||
|
|
||||||
METHOD_TRACE("ips_is_init_morpheus", 1);
|
METHOD_TRACE("ips_is_init_morpheus", 1);
|
||||||
|
|
||||||
|
if (ips_isintr_morpheus(ha))
|
||||||
|
ips_flush_and_reset(ha);
|
||||||
|
|
||||||
post = readl(ha->mem_ptr + IPS_REG_I960_MSG0);
|
post = readl(ha->mem_ptr + IPS_REG_I960_MSG0);
|
||||||
bits = readl(ha->mem_ptr + IPS_REG_I2O_HIR);
|
bits = readl(ha->mem_ptr + IPS_REG_I2O_HIR);
|
||||||
|
@ -4819,6 +4848,94 @@ ips_isinit_morpheus(ips_ha_t * ha)
|
||||||
return (1);
|
return (1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/****************************************************************************/
|
||||||
|
/* */
|
||||||
|
/* Routine Name: ips_flush_and_reset */
|
||||||
|
/* */
|
||||||
|
/* Routine Description: */
|
||||||
|
/* */
|
||||||
|
/* Perform cleanup ( FLUSH and RESET ) when the adapter is in an unknown */
|
||||||
|
/* state ( was trying to INIT and an interrupt was already pending ) ... */
|
||||||
|
/* */
|
||||||
|
/****************************************************************************/
|
||||||
|
static void
|
||||||
|
ips_flush_and_reset(ips_ha_t *ha)
|
||||||
|
{
|
||||||
|
ips_scb_t *scb;
|
||||||
|
int ret;
|
||||||
|
int time;
|
||||||
|
int done;
|
||||||
|
dma_addr_t command_dma;
|
||||||
|
|
||||||
|
/* Create a usuable SCB */
|
||||||
|
scb = pci_alloc_consistent(ha->pcidev, sizeof(ips_scb_t), &command_dma);
|
||||||
|
if (scb) {
|
||||||
|
memset(scb, 0, sizeof(ips_scb_t));
|
||||||
|
ips_init_scb(ha, scb);
|
||||||
|
scb->scb_busaddr = command_dma;
|
||||||
|
|
||||||
|
scb->timeout = ips_cmd_timeout;
|
||||||
|
scb->cdb[0] = IPS_CMD_FLUSH;
|
||||||
|
|
||||||
|
scb->cmd.flush_cache.op_code = IPS_CMD_FLUSH;
|
||||||
|
scb->cmd.flush_cache.command_id = IPS_MAX_CMDS; /* Use an ID that would otherwise not exist */
|
||||||
|
scb->cmd.flush_cache.state = IPS_NORM_STATE;
|
||||||
|
scb->cmd.flush_cache.reserved = 0;
|
||||||
|
scb->cmd.flush_cache.reserved2 = 0;
|
||||||
|
scb->cmd.flush_cache.reserved3 = 0;
|
||||||
|
scb->cmd.flush_cache.reserved4 = 0;
|
||||||
|
|
||||||
|
ret = ips_send_cmd(ha, scb); /* Send the Flush Command */
|
||||||
|
|
||||||
|
if (ret == IPS_SUCCESS) {
|
||||||
|
time = 60 * IPS_ONE_SEC; /* Max Wait time is 60 seconds */
|
||||||
|
done = 0;
|
||||||
|
|
||||||
|
while ((time > 0) && (!done)) {
|
||||||
|
done = ips_poll_for_flush_complete(ha);
|
||||||
|
/* This may look evil, but it's only done during extremely rare start-up conditions ! */
|
||||||
|
udelay(1000);
|
||||||
|
time--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Now RESET and INIT the adapter */
|
||||||
|
(*ha->func.reset) (ha);
|
||||||
|
|
||||||
|
pci_free_consistent(ha->pcidev, sizeof(ips_scb_t), scb, command_dma);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/****************************************************************************/
|
||||||
|
/* */
|
||||||
|
/* Routine Name: ips_poll_for_flush_complete */
|
||||||
|
/* */
|
||||||
|
/* Routine Description: */
|
||||||
|
/* */
|
||||||
|
/* Poll for the Flush Command issued by ips_flush_and_reset() to complete */
|
||||||
|
/* All other responses are just taken off the queue and ignored */
|
||||||
|
/* */
|
||||||
|
/****************************************************************************/
|
||||||
|
static int
|
||||||
|
ips_poll_for_flush_complete(ips_ha_t * ha)
|
||||||
|
{
|
||||||
|
IPS_STATUS cstatus;
|
||||||
|
|
||||||
|
while (TRUE) {
|
||||||
|
cstatus.value = (*ha->func.statupd) (ha);
|
||||||
|
|
||||||
|
if (cstatus.value == 0xffffffff) /* If No Interrupt to process */
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* Success is when we see the Flush Command ID */
|
||||||
|
if (cstatus.fields.command_id == IPS_MAX_CMDS )
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/****************************************************************************/
|
/****************************************************************************/
|
||||||
/* */
|
/* */
|
||||||
/* Routine Name: ips_enable_int_copperhead */
|
/* Routine Name: ips_enable_int_copperhead */
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,322 @@
|
||||||
|
/*
|
||||||
|
* iSCSI Initiator TCP Transport
|
||||||
|
* Copyright (C) 2004 Dmitry Yusupov
|
||||||
|
* Copyright (C) 2004 Alex Aizman
|
||||||
|
* Copyright (C) 2005 Mike Christie
|
||||||
|
* maintained by open-iscsi@googlegroups.com
|
||||||
|
*
|
||||||
|
* 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; either version 2 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*
|
||||||
|
* See the file COPYING included with this distribution for more details.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef ISCSI_TCP_H
|
||||||
|
#define ISCSI_TCP_H
|
||||||
|
|
||||||
|
/* Session's states */
|
||||||
|
#define ISCSI_STATE_FREE 1
|
||||||
|
#define ISCSI_STATE_LOGGED_IN 2
|
||||||
|
#define ISCSI_STATE_FAILED 3
|
||||||
|
#define ISCSI_STATE_TERMINATE 4
|
||||||
|
|
||||||
|
/* Connection's states */
|
||||||
|
#define ISCSI_CONN_INITIAL_STAGE 0
|
||||||
|
#define ISCSI_CONN_STARTED 1
|
||||||
|
#define ISCSI_CONN_STOPPED 2
|
||||||
|
#define ISCSI_CONN_CLEANUP_WAIT 3
|
||||||
|
|
||||||
|
/* Connection suspend "bit" */
|
||||||
|
#define SUSPEND_BIT 1
|
||||||
|
|
||||||
|
/* Socket's Receive state machine */
|
||||||
|
#define IN_PROGRESS_WAIT_HEADER 0x0
|
||||||
|
#define IN_PROGRESS_HEADER_GATHER 0x1
|
||||||
|
#define IN_PROGRESS_DATA_RECV 0x2
|
||||||
|
#define IN_PROGRESS_DDIGEST_RECV 0x3
|
||||||
|
|
||||||
|
/* Task Mgmt states */
|
||||||
|
#define TMABORT_INITIAL 0x0
|
||||||
|
#define TMABORT_SUCCESS 0x1
|
||||||
|
#define TMABORT_FAILED 0x2
|
||||||
|
#define TMABORT_TIMEDOUT 0x3
|
||||||
|
|
||||||
|
/* xmit state machine */
|
||||||
|
#define XMSTATE_IDLE 0x0
|
||||||
|
#define XMSTATE_R_HDR 0x1
|
||||||
|
#define XMSTATE_W_HDR 0x2
|
||||||
|
#define XMSTATE_IMM_HDR 0x4
|
||||||
|
#define XMSTATE_IMM_DATA 0x8
|
||||||
|
#define XMSTATE_UNS_INIT 0x10
|
||||||
|
#define XMSTATE_UNS_HDR 0x20
|
||||||
|
#define XMSTATE_UNS_DATA 0x40
|
||||||
|
#define XMSTATE_SOL_HDR 0x80
|
||||||
|
#define XMSTATE_SOL_DATA 0x100
|
||||||
|
#define XMSTATE_W_PAD 0x200
|
||||||
|
#define XMSTATE_DATA_DIGEST 0x400
|
||||||
|
|
||||||
|
#define ISCSI_CONN_MAX 1
|
||||||
|
#define ISCSI_CONN_RCVBUF_MIN 262144
|
||||||
|
#define ISCSI_CONN_SNDBUF_MIN 262144
|
||||||
|
#define ISCSI_PAD_LEN 4
|
||||||
|
#define ISCSI_R2T_MAX 16
|
||||||
|
#define ISCSI_XMIT_CMDS_MAX 128 /* must be power of 2 */
|
||||||
|
#define ISCSI_MGMT_CMDS_MAX 32 /* must be power of 2 */
|
||||||
|
#define ISCSI_MGMT_ITT_OFFSET 0xa00
|
||||||
|
#define ISCSI_SG_TABLESIZE SG_ALL
|
||||||
|
#define ISCSI_CMD_PER_LUN 128
|
||||||
|
#define ISCSI_TCP_MAX_CMD_LEN 16
|
||||||
|
|
||||||
|
#define ITT_MASK (0xfff)
|
||||||
|
#define CID_SHIFT 12
|
||||||
|
#define CID_MASK (0xffff<<CID_SHIFT)
|
||||||
|
#define AGE_SHIFT 28
|
||||||
|
#define AGE_MASK (0xf<<AGE_SHIFT)
|
||||||
|
|
||||||
|
struct iscsi_queue {
|
||||||
|
struct kfifo *queue; /* FIFO Queue */
|
||||||
|
void **pool; /* Pool of elements */
|
||||||
|
int max; /* Max number of elements */
|
||||||
|
};
|
||||||
|
|
||||||
|
struct iscsi_session;
|
||||||
|
struct iscsi_cmd_task;
|
||||||
|
struct iscsi_mgmt_task;
|
||||||
|
|
||||||
|
/* Socket connection recieve helper */
|
||||||
|
struct iscsi_tcp_recv {
|
||||||
|
struct iscsi_hdr *hdr;
|
||||||
|
struct sk_buff *skb;
|
||||||
|
int offset;
|
||||||
|
int len;
|
||||||
|
int hdr_offset;
|
||||||
|
int copy;
|
||||||
|
int copied;
|
||||||
|
int padding;
|
||||||
|
struct iscsi_cmd_task *ctask; /* current cmd in progress */
|
||||||
|
|
||||||
|
/* copied and flipped values */
|
||||||
|
int opcode;
|
||||||
|
int flags;
|
||||||
|
int cmd_status;
|
||||||
|
int ahslen;
|
||||||
|
int datalen;
|
||||||
|
uint32_t itt;
|
||||||
|
int datadgst;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct iscsi_conn {
|
||||||
|
struct iscsi_hdr hdr; /* header placeholder */
|
||||||
|
char hdrext[4*sizeof(__u16) +
|
||||||
|
sizeof(__u32)];
|
||||||
|
int data_copied;
|
||||||
|
char *data; /* data placeholder */
|
||||||
|
struct socket *sock; /* TCP socket */
|
||||||
|
int data_size; /* actual recv_dlength */
|
||||||
|
int stop_stage; /* conn_stop() flag: *
|
||||||
|
* stop to recover, *
|
||||||
|
* stop to terminate */
|
||||||
|
/* iSCSI connection-wide sequencing */
|
||||||
|
uint32_t exp_statsn;
|
||||||
|
int hdr_size; /* PDU header size */
|
||||||
|
unsigned long suspend_rx; /* suspend Rx */
|
||||||
|
|
||||||
|
struct crypto_tfm *rx_tfm; /* CRC32C (Rx) */
|
||||||
|
struct crypto_tfm *data_rx_tfm; /* CRC32C (Rx) for data */
|
||||||
|
|
||||||
|
/* control data */
|
||||||
|
int senselen; /* scsi sense length */
|
||||||
|
int id; /* CID */
|
||||||
|
struct iscsi_tcp_recv in; /* TCP receive context */
|
||||||
|
struct iscsi_session *session; /* parent session */
|
||||||
|
struct list_head item; /* maintains list of conns */
|
||||||
|
int in_progress; /* connection state machine */
|
||||||
|
int c_stage; /* connection state */
|
||||||
|
struct iscsi_mgmt_task *login_mtask; /* mtask used for login/text */
|
||||||
|
struct iscsi_mgmt_task *mtask; /* xmit mtask in progress */
|
||||||
|
struct iscsi_cmd_task *ctask; /* xmit ctask in progress */
|
||||||
|
spinlock_t lock; /* FIXME: to be removed */
|
||||||
|
|
||||||
|
/* old values for socket callbacks */
|
||||||
|
void (*old_data_ready)(struct sock *, int);
|
||||||
|
void (*old_state_change)(struct sock *);
|
||||||
|
void (*old_write_space)(struct sock *);
|
||||||
|
|
||||||
|
/* xmit */
|
||||||
|
struct crypto_tfm *tx_tfm; /* CRC32C (Tx) */
|
||||||
|
struct crypto_tfm *data_tx_tfm; /* CRC32C (Tx) for data */
|
||||||
|
struct kfifo *writequeue; /* write cmds for Data-Outs */
|
||||||
|
struct kfifo *immqueue; /* immediate xmit queue */
|
||||||
|
struct kfifo *mgmtqueue; /* mgmt (control) xmit queue */
|
||||||
|
struct kfifo *xmitqueue; /* data-path cmd queue */
|
||||||
|
struct work_struct xmitwork; /* per-conn. xmit workqueue */
|
||||||
|
struct semaphore xmitsema; /* serializes connection xmit,
|
||||||
|
* access to kfifos: *
|
||||||
|
* xmitqueue, writequeue, *
|
||||||
|
* immqueue, mgmtqueue */
|
||||||
|
unsigned long suspend_tx; /* suspend Tx */
|
||||||
|
|
||||||
|
/* abort */
|
||||||
|
wait_queue_head_t ehwait; /* used in eh_abort() */
|
||||||
|
struct iscsi_tm tmhdr;
|
||||||
|
struct timer_list tmabort_timer; /* abort timer */
|
||||||
|
int tmabort_state; /* see TMABORT_INITIAL, etc.*/
|
||||||
|
|
||||||
|
/* negotiated params */
|
||||||
|
int max_recv_dlength;
|
||||||
|
int max_xmit_dlength;
|
||||||
|
int hdrdgst_en;
|
||||||
|
int datadgst_en;
|
||||||
|
|
||||||
|
/* MIB-statistics */
|
||||||
|
uint64_t txdata_octets;
|
||||||
|
uint64_t rxdata_octets;
|
||||||
|
uint32_t scsicmd_pdus_cnt;
|
||||||
|
uint32_t dataout_pdus_cnt;
|
||||||
|
uint32_t scsirsp_pdus_cnt;
|
||||||
|
uint32_t datain_pdus_cnt;
|
||||||
|
uint32_t r2t_pdus_cnt;
|
||||||
|
uint32_t tmfcmd_pdus_cnt;
|
||||||
|
int32_t tmfrsp_pdus_cnt;
|
||||||
|
|
||||||
|
/* custom statistics */
|
||||||
|
uint32_t sendpage_failures_cnt;
|
||||||
|
uint32_t discontiguous_hdr_cnt;
|
||||||
|
uint32_t eh_abort_cnt;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct iscsi_session {
|
||||||
|
/* iSCSI session-wide sequencing */
|
||||||
|
uint32_t cmdsn;
|
||||||
|
uint32_t exp_cmdsn;
|
||||||
|
uint32_t max_cmdsn;
|
||||||
|
|
||||||
|
/* configuration */
|
||||||
|
int initial_r2t_en;
|
||||||
|
int max_r2t;
|
||||||
|
int imm_data_en;
|
||||||
|
int first_burst;
|
||||||
|
int max_burst;
|
||||||
|
int time2wait;
|
||||||
|
int time2retain;
|
||||||
|
int pdu_inorder_en;
|
||||||
|
int dataseq_inorder_en;
|
||||||
|
int erl;
|
||||||
|
int ifmarker_en;
|
||||||
|
int ofmarker_en;
|
||||||
|
|
||||||
|
/* control data */
|
||||||
|
struct Scsi_Host *host;
|
||||||
|
int id;
|
||||||
|
struct iscsi_conn *leadconn; /* leading connection */
|
||||||
|
spinlock_t lock; /* protects session state, *
|
||||||
|
* sequence numbers, *
|
||||||
|
* session resources: *
|
||||||
|
* - cmdpool, *
|
||||||
|
* - mgmtpool, *
|
||||||
|
* - r2tpool */
|
||||||
|
int state; /* session state */
|
||||||
|
struct list_head item;
|
||||||
|
void *auth_client;
|
||||||
|
int conn_cnt;
|
||||||
|
int age; /* counts session re-opens */
|
||||||
|
|
||||||
|
struct list_head connections; /* list of connections */
|
||||||
|
int cmds_max; /* size of cmds array */
|
||||||
|
struct iscsi_cmd_task **cmds; /* Original Cmds arr */
|
||||||
|
struct iscsi_queue cmdpool; /* PDU's pool */
|
||||||
|
int mgmtpool_max; /* size of mgmt array */
|
||||||
|
struct iscsi_mgmt_task **mgmt_cmds; /* Original mgmt arr */
|
||||||
|
struct iscsi_queue mgmtpool; /* Mgmt PDU's pool */
|
||||||
|
};
|
||||||
|
|
||||||
|
struct iscsi_buf {
|
||||||
|
struct scatterlist sg;
|
||||||
|
struct kvec iov;
|
||||||
|
unsigned int sent;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct iscsi_data_task {
|
||||||
|
struct iscsi_data hdr; /* PDU */
|
||||||
|
char hdrext[sizeof(__u32)]; /* Header-Digest */
|
||||||
|
struct list_head item; /* data queue item */
|
||||||
|
struct iscsi_buf digestbuf; /* digest buffer */
|
||||||
|
uint32_t digest; /* data digest */
|
||||||
|
};
|
||||||
|
#define ISCSI_DTASK_DEFAULT_MAX ISCSI_SG_TABLESIZE * PAGE_SIZE / 512
|
||||||
|
|
||||||
|
struct iscsi_mgmt_task {
|
||||||
|
struct iscsi_hdr hdr; /* mgmt. PDU */
|
||||||
|
char hdrext[sizeof(__u32)]; /* Header-Digest */
|
||||||
|
char *data; /* mgmt payload */
|
||||||
|
int xmstate; /* mgmt xmit progress */
|
||||||
|
int data_count; /* counts data to be sent */
|
||||||
|
struct iscsi_buf headbuf; /* header buffer */
|
||||||
|
struct iscsi_buf sendbuf; /* in progress buffer */
|
||||||
|
int sent;
|
||||||
|
uint32_t itt; /* this ITT */
|
||||||
|
};
|
||||||
|
|
||||||
|
struct iscsi_r2t_info {
|
||||||
|
__be32 ttt; /* copied from R2T */
|
||||||
|
__be32 exp_statsn; /* copied from R2T */
|
||||||
|
uint32_t data_length; /* copied from R2T */
|
||||||
|
uint32_t data_offset; /* copied from R2T */
|
||||||
|
struct iscsi_buf headbuf; /* Data-Out Header Buffer */
|
||||||
|
struct iscsi_buf sendbuf; /* Data-Out in progress buffer*/
|
||||||
|
int sent; /* R2T sequence progress */
|
||||||
|
int data_count; /* DATA-Out payload progress */
|
||||||
|
struct scatterlist *sg; /* per-R2T SG list */
|
||||||
|
int solicit_datasn;
|
||||||
|
struct iscsi_data_task *dtask; /* which data task */
|
||||||
|
};
|
||||||
|
|
||||||
|
struct iscsi_cmd_task {
|
||||||
|
struct iscsi_cmd hdr; /* iSCSI PDU header */
|
||||||
|
char hdrext[4*sizeof(__u16)+ /* AHS */
|
||||||
|
sizeof(__u32)]; /* HeaderDigest */
|
||||||
|
char pad[ISCSI_PAD_LEN];
|
||||||
|
int itt; /* this ITT */
|
||||||
|
int datasn; /* DataSN */
|
||||||
|
struct iscsi_buf headbuf; /* header buf (xmit) */
|
||||||
|
struct iscsi_buf sendbuf; /* in progress buffer*/
|
||||||
|
int sent;
|
||||||
|
struct scatterlist *sg; /* per-cmd SG list */
|
||||||
|
struct scatterlist *bad_sg; /* assert statement */
|
||||||
|
int sg_count; /* SG's to process */
|
||||||
|
uint32_t unsol_datasn;
|
||||||
|
uint32_t exp_r2tsn;
|
||||||
|
int xmstate; /* xmit xtate machine */
|
||||||
|
int imm_count; /* imm-data (bytes) */
|
||||||
|
int unsol_count; /* unsolicited (bytes)*/
|
||||||
|
int r2t_data_count; /* R2T Data-Out bytes */
|
||||||
|
int data_count; /* remaining Data-Out */
|
||||||
|
int pad_count; /* padded bytes */
|
||||||
|
struct scsi_cmnd *sc; /* associated SCSI cmd*/
|
||||||
|
int total_length;
|
||||||
|
int data_offset;
|
||||||
|
struct iscsi_conn *conn; /* used connection */
|
||||||
|
struct iscsi_mgmt_task *mtask; /* tmf mtask in progr */
|
||||||
|
|
||||||
|
struct iscsi_r2t_info *r2t; /* in progress R2T */
|
||||||
|
struct iscsi_queue r2tpool;
|
||||||
|
struct kfifo *r2tqueue;
|
||||||
|
struct iscsi_r2t_info **r2ts;
|
||||||
|
struct list_head dataqueue; /* Data-Out dataqueue */
|
||||||
|
mempool_t *datapool;
|
||||||
|
uint32_t datadigest; /* for recover digest */
|
||||||
|
int digest_count;
|
||||||
|
uint32_t immdigest; /* for imm data */
|
||||||
|
struct iscsi_buf immbuf; /* for imm data digest */
|
||||||
|
struct iscsi_data_task *dtask; /* data task in progress*/
|
||||||
|
int digest_offset; /* for partial buff digest */
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif /* ISCSI_H */
|
|
@ -267,10 +267,6 @@ struct lpfc_hba {
|
||||||
struct lpfc_nodelist fc_fcpnodev; /* nodelist entry for no device */
|
struct lpfc_nodelist fc_fcpnodev; /* nodelist entry for no device */
|
||||||
uint32_t nport_event_cnt; /* timestamp for nlplist entry */
|
uint32_t nport_event_cnt; /* timestamp for nlplist entry */
|
||||||
|
|
||||||
#define LPFC_RPI_HASH_SIZE 64
|
|
||||||
#define LPFC_RPI_HASH_FUNC(x) ((x) & (0x3f))
|
|
||||||
/* ptr to active D_ID / RPIs */
|
|
||||||
struct lpfc_nodelist *fc_nlplookup[LPFC_RPI_HASH_SIZE];
|
|
||||||
uint32_t wwnn[2];
|
uint32_t wwnn[2];
|
||||||
uint32_t RandomData[7];
|
uint32_t RandomData[7];
|
||||||
|
|
||||||
|
|
|
@ -200,19 +200,13 @@ lpfc_num_discovered_ports_show(struct class_device *cdev, char *buf)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static ssize_t
|
static int
|
||||||
lpfc_issue_lip (struct class_device *cdev, const char *buf, size_t count)
|
lpfc_issue_lip(struct Scsi_Host *host)
|
||||||
{
|
{
|
||||||
struct Scsi_Host *host = class_to_shost(cdev);
|
|
||||||
struct lpfc_hba *phba = (struct lpfc_hba *) host->hostdata[0];
|
struct lpfc_hba *phba = (struct lpfc_hba *) host->hostdata[0];
|
||||||
int val = 0;
|
|
||||||
LPFC_MBOXQ_t *pmboxq;
|
LPFC_MBOXQ_t *pmboxq;
|
||||||
int mbxstatus = MBXERR_ERROR;
|
int mbxstatus = MBXERR_ERROR;
|
||||||
|
|
||||||
if ((sscanf(buf, "%d", &val) != 1) ||
|
|
||||||
(val != 1))
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
if ((phba->fc_flag & FC_OFFLINE_MODE) ||
|
if ((phba->fc_flag & FC_OFFLINE_MODE) ||
|
||||||
(phba->hba_state != LPFC_HBA_READY))
|
(phba->hba_state != LPFC_HBA_READY))
|
||||||
return -EPERM;
|
return -EPERM;
|
||||||
|
@ -229,12 +223,12 @@ lpfc_issue_lip (struct class_device *cdev, const char *buf, size_t count)
|
||||||
if (mbxstatus == MBX_TIMEOUT)
|
if (mbxstatus == MBX_TIMEOUT)
|
||||||
pmboxq->mbox_cmpl = lpfc_sli_def_mbox_cmpl;
|
pmboxq->mbox_cmpl = lpfc_sli_def_mbox_cmpl;
|
||||||
else
|
else
|
||||||
mempool_free( pmboxq, phba->mbox_mem_pool);
|
mempool_free(pmboxq, phba->mbox_mem_pool);
|
||||||
|
|
||||||
if (mbxstatus == MBXERR_ERROR)
|
if (mbxstatus == MBXERR_ERROR)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
return strlen(buf);
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static ssize_t
|
static ssize_t
|
||||||
|
@ -251,8 +245,6 @@ lpfc_board_online_show(struct class_device *cdev, char *buf)
|
||||||
struct Scsi_Host *host = class_to_shost(cdev);
|
struct Scsi_Host *host = class_to_shost(cdev);
|
||||||
struct lpfc_hba *phba = (struct lpfc_hba*)host->hostdata[0];
|
struct lpfc_hba *phba = (struct lpfc_hba*)host->hostdata[0];
|
||||||
|
|
||||||
if (!phba) return 0;
|
|
||||||
|
|
||||||
if (phba->fc_flag & FC_OFFLINE_MODE)
|
if (phba->fc_flag & FC_OFFLINE_MODE)
|
||||||
return snprintf(buf, PAGE_SIZE, "0\n");
|
return snprintf(buf, PAGE_SIZE, "0\n");
|
||||||
else
|
else
|
||||||
|
@ -269,7 +261,7 @@ lpfc_board_online_store(struct class_device *cdev, const char *buf,
|
||||||
int val=0, status=0;
|
int val=0, status=0;
|
||||||
|
|
||||||
if (sscanf(buf, "%d", &val) != 1)
|
if (sscanf(buf, "%d", &val) != 1)
|
||||||
return 0;
|
return -EINVAL;
|
||||||
|
|
||||||
init_completion(&online_compl);
|
init_completion(&online_compl);
|
||||||
|
|
||||||
|
@ -283,7 +275,7 @@ lpfc_board_online_store(struct class_device *cdev, const char *buf,
|
||||||
if (!status)
|
if (!status)
|
||||||
return strlen(buf);
|
return strlen(buf);
|
||||||
else
|
else
|
||||||
return 0;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -294,47 +286,83 @@ lpfc_##attr##_show(struct class_device *cdev, char *buf) \
|
||||||
struct Scsi_Host *host = class_to_shost(cdev);\
|
struct Scsi_Host *host = class_to_shost(cdev);\
|
||||||
struct lpfc_hba *phba = (struct lpfc_hba*)host->hostdata[0];\
|
struct lpfc_hba *phba = (struct lpfc_hba*)host->hostdata[0];\
|
||||||
int val = 0;\
|
int val = 0;\
|
||||||
if (phba){\
|
val = phba->cfg_##attr;\
|
||||||
val = phba->cfg_##attr;\
|
return snprintf(buf, PAGE_SIZE, "%d\n",\
|
||||||
return snprintf(buf, PAGE_SIZE, "%d\n",\
|
phba->cfg_##attr);\
|
||||||
phba->cfg_##attr);\
|
|
||||||
}\
|
|
||||||
return 0;\
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#define lpfc_param_store(attr, minval, maxval) \
|
#define lpfc_param_hex_show(attr) \
|
||||||
|
static ssize_t \
|
||||||
|
lpfc_##attr##_show(struct class_device *cdev, char *buf) \
|
||||||
|
{ \
|
||||||
|
struct Scsi_Host *host = class_to_shost(cdev);\
|
||||||
|
struct lpfc_hba *phba = (struct lpfc_hba*)host->hostdata[0];\
|
||||||
|
int val = 0;\
|
||||||
|
val = phba->cfg_##attr;\
|
||||||
|
return snprintf(buf, PAGE_SIZE, "%#x\n",\
|
||||||
|
phba->cfg_##attr);\
|
||||||
|
}
|
||||||
|
|
||||||
|
#define lpfc_param_init(attr, default, minval, maxval) \
|
||||||
|
static int \
|
||||||
|
lpfc_##attr##_init(struct lpfc_hba *phba, int val) \
|
||||||
|
{ \
|
||||||
|
if (val >= minval && val <= maxval) {\
|
||||||
|
phba->cfg_##attr = val;\
|
||||||
|
return 0;\
|
||||||
|
}\
|
||||||
|
lpfc_printf_log(phba, KERN_ERR, LOG_INIT, \
|
||||||
|
"%d:0449 lpfc_"#attr" attribute cannot be set to %d, "\
|
||||||
|
"allowed range is ["#minval", "#maxval"]\n", \
|
||||||
|
phba->brd_no, val); \
|
||||||
|
phba->cfg_##attr = default;\
|
||||||
|
return -EINVAL;\
|
||||||
|
}
|
||||||
|
|
||||||
|
#define lpfc_param_set(attr, default, minval, maxval) \
|
||||||
|
static int \
|
||||||
|
lpfc_##attr##_set(struct lpfc_hba *phba, int val) \
|
||||||
|
{ \
|
||||||
|
if (val >= minval && val <= maxval) {\
|
||||||
|
phba->cfg_##attr = val;\
|
||||||
|
return 0;\
|
||||||
|
}\
|
||||||
|
lpfc_printf_log(phba, KERN_ERR, LOG_INIT, \
|
||||||
|
"%d:0450 lpfc_"#attr" attribute cannot be set to %d, "\
|
||||||
|
"allowed range is ["#minval", "#maxval"]\n", \
|
||||||
|
phba->brd_no, val); \
|
||||||
|
return -EINVAL;\
|
||||||
|
}
|
||||||
|
|
||||||
|
#define lpfc_param_store(attr) \
|
||||||
static ssize_t \
|
static ssize_t \
|
||||||
lpfc_##attr##_store(struct class_device *cdev, const char *buf, size_t count) \
|
lpfc_##attr##_store(struct class_device *cdev, const char *buf, size_t count) \
|
||||||
{ \
|
{ \
|
||||||
struct Scsi_Host *host = class_to_shost(cdev);\
|
struct Scsi_Host *host = class_to_shost(cdev);\
|
||||||
struct lpfc_hba *phba = (struct lpfc_hba*)host->hostdata[0];\
|
struct lpfc_hba *phba = (struct lpfc_hba*)host->hostdata[0];\
|
||||||
int val = 0;\
|
int val=0;\
|
||||||
if (!isdigit(buf[0]))\
|
if (!isdigit(buf[0]))\
|
||||||
return -EINVAL;\
|
return -EINVAL;\
|
||||||
if (sscanf(buf, "0x%x", &val) != 1)\
|
if (sscanf(buf, "%i", &val) != 1)\
|
||||||
if (sscanf(buf, "%d", &val) != 1)\
|
return -EINVAL;\
|
||||||
return -EINVAL;\
|
if (lpfc_##attr##_set(phba, val) == 0) \
|
||||||
if (phba){\
|
return strlen(buf);\
|
||||||
if (val >= minval && val <= maxval) {\
|
else \
|
||||||
phba->cfg_##attr = val;\
|
return -EINVAL;\
|
||||||
return strlen(buf);\
|
|
||||||
}\
|
|
||||||
}\
|
|
||||||
return 0;\
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#define LPFC_ATTR_R_NOINIT(name, desc) \
|
#define LPFC_ATTR(name, defval, minval, maxval, desc) \
|
||||||
extern int lpfc_##name;\
|
static int lpfc_##name = defval;\
|
||||||
module_param(lpfc_##name, int, 0);\
|
module_param(lpfc_##name, int, 0);\
|
||||||
MODULE_PARM_DESC(lpfc_##name, desc);\
|
MODULE_PARM_DESC(lpfc_##name, desc);\
|
||||||
lpfc_param_show(name)\
|
lpfc_param_init(name, defval, minval, maxval)
|
||||||
static CLASS_DEVICE_ATTR(lpfc_##name, S_IRUGO , lpfc_##name##_show, NULL)
|
|
||||||
|
|
||||||
#define LPFC_ATTR_R(name, defval, minval, maxval, desc) \
|
#define LPFC_ATTR_R(name, defval, minval, maxval, desc) \
|
||||||
static int lpfc_##name = defval;\
|
static int lpfc_##name = defval;\
|
||||||
module_param(lpfc_##name, int, 0);\
|
module_param(lpfc_##name, int, 0);\
|
||||||
MODULE_PARM_DESC(lpfc_##name, desc);\
|
MODULE_PARM_DESC(lpfc_##name, desc);\
|
||||||
lpfc_param_show(name)\
|
lpfc_param_show(name)\
|
||||||
|
lpfc_param_init(name, defval, minval, maxval)\
|
||||||
static CLASS_DEVICE_ATTR(lpfc_##name, S_IRUGO , lpfc_##name##_show, NULL)
|
static CLASS_DEVICE_ATTR(lpfc_##name, S_IRUGO , lpfc_##name##_show, NULL)
|
||||||
|
|
||||||
#define LPFC_ATTR_RW(name, defval, minval, maxval, desc) \
|
#define LPFC_ATTR_RW(name, defval, minval, maxval, desc) \
|
||||||
|
@ -342,7 +370,28 @@ static int lpfc_##name = defval;\
|
||||||
module_param(lpfc_##name, int, 0);\
|
module_param(lpfc_##name, int, 0);\
|
||||||
MODULE_PARM_DESC(lpfc_##name, desc);\
|
MODULE_PARM_DESC(lpfc_##name, desc);\
|
||||||
lpfc_param_show(name)\
|
lpfc_param_show(name)\
|
||||||
lpfc_param_store(name, minval, maxval)\
|
lpfc_param_init(name, defval, minval, maxval)\
|
||||||
|
lpfc_param_set(name, defval, minval, maxval)\
|
||||||
|
lpfc_param_store(name)\
|
||||||
|
static CLASS_DEVICE_ATTR(lpfc_##name, S_IRUGO | S_IWUSR,\
|
||||||
|
lpfc_##name##_show, lpfc_##name##_store)
|
||||||
|
|
||||||
|
#define LPFC_ATTR_HEX_R(name, defval, minval, maxval, desc) \
|
||||||
|
static int lpfc_##name = defval;\
|
||||||
|
module_param(lpfc_##name, int, 0);\
|
||||||
|
MODULE_PARM_DESC(lpfc_##name, desc);\
|
||||||
|
lpfc_param_hex_show(name)\
|
||||||
|
lpfc_param_init(name, defval, minval, maxval)\
|
||||||
|
static CLASS_DEVICE_ATTR(lpfc_##name, S_IRUGO , lpfc_##name##_show, NULL)
|
||||||
|
|
||||||
|
#define LPFC_ATTR_HEX_RW(name, defval, minval, maxval, desc) \
|
||||||
|
static int lpfc_##name = defval;\
|
||||||
|
module_param(lpfc_##name, int, 0);\
|
||||||
|
MODULE_PARM_DESC(lpfc_##name, desc);\
|
||||||
|
lpfc_param_hex_show(name)\
|
||||||
|
lpfc_param_init(name, defval, minval, maxval)\
|
||||||
|
lpfc_param_set(name, defval, minval, maxval)\
|
||||||
|
lpfc_param_store(name)\
|
||||||
static CLASS_DEVICE_ATTR(lpfc_##name, S_IRUGO | S_IWUSR,\
|
static CLASS_DEVICE_ATTR(lpfc_##name, S_IRUGO | S_IWUSR,\
|
||||||
lpfc_##name##_show, lpfc_##name##_store)
|
lpfc_##name##_show, lpfc_##name##_store)
|
||||||
|
|
||||||
|
@ -364,7 +413,6 @@ static CLASS_DEVICE_ATTR(lpfc_drvr_version, S_IRUGO, lpfc_drvr_version_show,
|
||||||
NULL);
|
NULL);
|
||||||
static CLASS_DEVICE_ATTR(management_version, S_IRUGO, management_version_show,
|
static CLASS_DEVICE_ATTR(management_version, S_IRUGO, management_version_show,
|
||||||
NULL);
|
NULL);
|
||||||
static CLASS_DEVICE_ATTR(issue_lip, S_IWUSR, NULL, lpfc_issue_lip);
|
|
||||||
static CLASS_DEVICE_ATTR(board_online, S_IRUGO | S_IWUSR,
|
static CLASS_DEVICE_ATTR(board_online, S_IRUGO | S_IWUSR,
|
||||||
lpfc_board_online_show, lpfc_board_online_store);
|
lpfc_board_online_show, lpfc_board_online_store);
|
||||||
|
|
||||||
|
@ -388,7 +436,7 @@ static CLASS_DEVICE_ATTR(board_online, S_IRUGO | S_IWUSR,
|
||||||
# LOG_LIBDFC 0x2000 LIBDFC events
|
# LOG_LIBDFC 0x2000 LIBDFC events
|
||||||
# LOG_ALL_MSG 0xffff LOG all messages
|
# LOG_ALL_MSG 0xffff LOG all messages
|
||||||
*/
|
*/
|
||||||
LPFC_ATTR_RW(log_verbose, 0x0, 0x0, 0xffff, "Verbose logging bit-mask");
|
LPFC_ATTR_HEX_RW(log_verbose, 0x0, 0x0, 0xffff, "Verbose logging bit-mask");
|
||||||
|
|
||||||
/*
|
/*
|
||||||
# lun_queue_depth: This parameter is used to limit the number of outstanding
|
# lun_queue_depth: This parameter is used to limit the number of outstanding
|
||||||
|
@ -419,7 +467,7 @@ LPFC_ATTR_R(scan_down, 1, 0, 1,
|
||||||
|
|
||||||
/*
|
/*
|
||||||
# lpfc_nodev_tmo: If set, it will hold all I/O errors on devices that disappear
|
# lpfc_nodev_tmo: If set, it will hold all I/O errors on devices that disappear
|
||||||
# until the timer expires. Value range is [0,255]. Default value is 20.
|
# until the timer expires. Value range is [0,255]. Default value is 30.
|
||||||
# NOTE: this MUST be less then the SCSI Layer command timeout - 1.
|
# NOTE: this MUST be less then the SCSI Layer command timeout - 1.
|
||||||
*/
|
*/
|
||||||
LPFC_ATTR_RW(nodev_tmo, 30, 0, 255,
|
LPFC_ATTR_RW(nodev_tmo, 30, 0, 255,
|
||||||
|
@ -475,14 +523,10 @@ LPFC_ATTR_R(ack0, 0, 0, 1, "Enable ACK0 support");
|
||||||
# is 0. Default value of cr_count is 1. The cr_count feature is disabled if
|
# is 0. Default value of cr_count is 1. The cr_count feature is disabled if
|
||||||
# cr_delay is set to 0.
|
# cr_delay is set to 0.
|
||||||
*/
|
*/
|
||||||
static int lpfc_cr_delay = 0;
|
LPFC_ATTR(cr_delay, 0, 0, 63, "A count of milliseconds after which an"
|
||||||
module_param(lpfc_cr_delay, int , 0);
|
|
||||||
MODULE_PARM_DESC(lpfc_cr_delay, "A count of milliseconds after which an "
|
|
||||||
"interrupt response is generated");
|
"interrupt response is generated");
|
||||||
|
|
||||||
static int lpfc_cr_count = 1;
|
LPFC_ATTR(cr_count, 1, 1, 255, "A count of I/O completions after which an"
|
||||||
module_param(lpfc_cr_count, int, 0);
|
|
||||||
MODULE_PARM_DESC(lpfc_cr_count, "A count of I/O completions after which an "
|
|
||||||
"interrupt response is generated");
|
"interrupt response is generated");
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -498,9 +542,7 @@ LPFC_ATTR_RW(fdmi_on, 0, 0, 2, "Enable FDMI support");
|
||||||
# Specifies the maximum number of ELS cmds we can have outstanding (for
|
# Specifies the maximum number of ELS cmds we can have outstanding (for
|
||||||
# discovery). Value range is [1,64]. Default value = 32.
|
# discovery). Value range is [1,64]. Default value = 32.
|
||||||
*/
|
*/
|
||||||
static int lpfc_discovery_threads = 32;
|
LPFC_ATTR(discovery_threads, 32, 1, 64, "Maximum number of ELS commands"
|
||||||
module_param(lpfc_discovery_threads, int, 0);
|
|
||||||
MODULE_PARM_DESC(lpfc_discovery_threads, "Maximum number of ELS commands "
|
|
||||||
"during discovery");
|
"during discovery");
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -537,7 +579,6 @@ struct class_device_attribute *lpfc_host_attrs[] = {
|
||||||
&class_device_attr_lpfc_max_luns,
|
&class_device_attr_lpfc_max_luns,
|
||||||
&class_device_attr_nport_evt_cnt,
|
&class_device_attr_nport_evt_cnt,
|
||||||
&class_device_attr_management_version,
|
&class_device_attr_management_version,
|
||||||
&class_device_attr_issue_lip,
|
|
||||||
&class_device_attr_board_online,
|
&class_device_attr_board_online,
|
||||||
NULL,
|
NULL,
|
||||||
};
|
};
|
||||||
|
@ -992,7 +1033,7 @@ lpfc_get_stats(struct Scsi_Host *shost)
|
||||||
struct fc_host_statistics *hs = &phba->link_stats;
|
struct fc_host_statistics *hs = &phba->link_stats;
|
||||||
LPFC_MBOXQ_t *pmboxq;
|
LPFC_MBOXQ_t *pmboxq;
|
||||||
MAILBOX_t *pmb;
|
MAILBOX_t *pmb;
|
||||||
int rc=0;
|
int rc = 0;
|
||||||
|
|
||||||
pmboxq = mempool_alloc(phba->mbox_mem_pool, GFP_KERNEL);
|
pmboxq = mempool_alloc(phba->mbox_mem_pool, GFP_KERNEL);
|
||||||
if (!pmboxq)
|
if (!pmboxq)
|
||||||
|
@ -1005,18 +1046,16 @@ lpfc_get_stats(struct Scsi_Host *shost)
|
||||||
pmboxq->context1 = NULL;
|
pmboxq->context1 = NULL;
|
||||||
|
|
||||||
if ((phba->fc_flag & FC_OFFLINE_MODE) ||
|
if ((phba->fc_flag & FC_OFFLINE_MODE) ||
|
||||||
(!(psli->sli_flag & LPFC_SLI2_ACTIVE))){
|
(!(psli->sli_flag & LPFC_SLI2_ACTIVE)))
|
||||||
rc = lpfc_sli_issue_mbox(phba, pmboxq, MBX_POLL);
|
rc = lpfc_sli_issue_mbox(phba, pmboxq, MBX_POLL);
|
||||||
} else
|
else
|
||||||
rc = lpfc_sli_issue_mbox_wait(phba, pmboxq, phba->fc_ratov * 2);
|
rc = lpfc_sli_issue_mbox_wait(phba, pmboxq, phba->fc_ratov * 2);
|
||||||
|
|
||||||
if (rc != MBX_SUCCESS) {
|
if (rc != MBX_SUCCESS) {
|
||||||
if (pmboxq) {
|
if (rc == MBX_TIMEOUT)
|
||||||
if (rc == MBX_TIMEOUT)
|
pmboxq->mbox_cmpl = lpfc_sli_def_mbox_cmpl;
|
||||||
pmboxq->mbox_cmpl = lpfc_sli_def_mbox_cmpl;
|
else
|
||||||
else
|
mempool_free(pmboxq, phba->mbox_mem_pool);
|
||||||
mempool_free( pmboxq, phba->mbox_mem_pool);
|
|
||||||
}
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1027,24 +1066,22 @@ lpfc_get_stats(struct Scsi_Host *shost)
|
||||||
hs->rx_frames = pmb->un.varRdStatus.rcvFrameCnt;
|
hs->rx_frames = pmb->un.varRdStatus.rcvFrameCnt;
|
||||||
hs->rx_words = (pmb->un.varRdStatus.rcvByteCnt * 256);
|
hs->rx_words = (pmb->un.varRdStatus.rcvByteCnt * 256);
|
||||||
|
|
||||||
memset((void *)pmboxq, 0, sizeof (LPFC_MBOXQ_t));
|
memset(pmboxq, 0, sizeof (LPFC_MBOXQ_t));
|
||||||
pmb->mbxCommand = MBX_READ_LNK_STAT;
|
pmb->mbxCommand = MBX_READ_LNK_STAT;
|
||||||
pmb->mbxOwner = OWN_HOST;
|
pmb->mbxOwner = OWN_HOST;
|
||||||
pmboxq->context1 = NULL;
|
pmboxq->context1 = NULL;
|
||||||
|
|
||||||
if ((phba->fc_flag & FC_OFFLINE_MODE) ||
|
if ((phba->fc_flag & FC_OFFLINE_MODE) ||
|
||||||
(!(psli->sli_flag & LPFC_SLI2_ACTIVE))) {
|
(!(psli->sli_flag & LPFC_SLI2_ACTIVE)))
|
||||||
rc = lpfc_sli_issue_mbox(phba, pmboxq, MBX_POLL);
|
rc = lpfc_sli_issue_mbox(phba, pmboxq, MBX_POLL);
|
||||||
} else
|
else
|
||||||
rc = lpfc_sli_issue_mbox_wait(phba, pmboxq, phba->fc_ratov * 2);
|
rc = lpfc_sli_issue_mbox_wait(phba, pmboxq, phba->fc_ratov * 2);
|
||||||
|
|
||||||
if (rc != MBX_SUCCESS) {
|
if (rc != MBX_SUCCESS) {
|
||||||
if (pmboxq) {
|
if (rc == MBX_TIMEOUT)
|
||||||
if (rc == MBX_TIMEOUT)
|
pmboxq->mbox_cmpl = lpfc_sli_def_mbox_cmpl;
|
||||||
pmboxq->mbox_cmpl = lpfc_sli_def_mbox_cmpl;
|
else
|
||||||
else
|
mempool_free( pmboxq, phba->mbox_mem_pool);
|
||||||
mempool_free( pmboxq, phba->mbox_mem_pool);
|
|
||||||
}
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1234,25 +1271,27 @@ struct fc_function_template lpfc_transport_functions = {
|
||||||
|
|
||||||
.get_starget_port_name = lpfc_get_starget_port_name,
|
.get_starget_port_name = lpfc_get_starget_port_name,
|
||||||
.show_starget_port_name = 1,
|
.show_starget_port_name = 1,
|
||||||
|
|
||||||
|
.issue_fc_host_lip = lpfc_issue_lip,
|
||||||
};
|
};
|
||||||
|
|
||||||
void
|
void
|
||||||
lpfc_get_cfgparam(struct lpfc_hba *phba)
|
lpfc_get_cfgparam(struct lpfc_hba *phba)
|
||||||
{
|
{
|
||||||
phba->cfg_log_verbose = lpfc_log_verbose;
|
lpfc_log_verbose_init(phba, lpfc_log_verbose);
|
||||||
phba->cfg_cr_delay = lpfc_cr_delay;
|
lpfc_cr_delay_init(phba, lpfc_cr_delay);
|
||||||
phba->cfg_cr_count = lpfc_cr_count;
|
lpfc_cr_count_init(phba, lpfc_cr_count);
|
||||||
phba->cfg_lun_queue_depth = lpfc_lun_queue_depth;
|
lpfc_lun_queue_depth_init(phba, lpfc_lun_queue_depth);
|
||||||
phba->cfg_fcp_class = lpfc_fcp_class;
|
lpfc_fcp_class_init(phba, lpfc_fcp_class);
|
||||||
phba->cfg_use_adisc = lpfc_use_adisc;
|
lpfc_use_adisc_init(phba, lpfc_use_adisc);
|
||||||
phba->cfg_ack0 = lpfc_ack0;
|
lpfc_ack0_init(phba, lpfc_ack0);
|
||||||
phba->cfg_topology = lpfc_topology;
|
lpfc_topology_init(phba, lpfc_topology);
|
||||||
phba->cfg_scan_down = lpfc_scan_down;
|
lpfc_scan_down_init(phba, lpfc_scan_down);
|
||||||
phba->cfg_nodev_tmo = lpfc_nodev_tmo;
|
lpfc_nodev_tmo_init(phba, lpfc_nodev_tmo);
|
||||||
phba->cfg_link_speed = lpfc_link_speed;
|
lpfc_link_speed_init(phba, lpfc_link_speed);
|
||||||
phba->cfg_fdmi_on = lpfc_fdmi_on;
|
lpfc_fdmi_on_init(phba, lpfc_fdmi_on);
|
||||||
phba->cfg_discovery_threads = lpfc_discovery_threads;
|
lpfc_discovery_threads_init(phba, lpfc_discovery_threads);
|
||||||
phba->cfg_max_luns = lpfc_max_luns;
|
lpfc_max_luns_init(phba, lpfc_max_luns);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The total number of segments is the configuration value plus 2
|
* The total number of segments is the configuration value plus 2
|
||||||
|
|
|
@ -62,10 +62,6 @@ void lpfc_disc_timeout(unsigned long);
|
||||||
void lpfc_scan_timeout(unsigned long);
|
void lpfc_scan_timeout(unsigned long);
|
||||||
|
|
||||||
struct lpfc_nodelist *lpfc_findnode_rpi(struct lpfc_hba * phba, uint16_t rpi);
|
struct lpfc_nodelist *lpfc_findnode_rpi(struct lpfc_hba * phba, uint16_t rpi);
|
||||||
struct lpfc_nodelist *lpfc_findnode_remove_rpi(struct lpfc_hba * phba,
|
|
||||||
uint16_t rpi);
|
|
||||||
void lpfc_addnode_rpi(struct lpfc_hba * phba, struct lpfc_nodelist * ndlp,
|
|
||||||
uint16_t rpi);
|
|
||||||
|
|
||||||
int lpfc_workq_post_event(struct lpfc_hba *, void *, void *, uint32_t);
|
int lpfc_workq_post_event(struct lpfc_hba *, void *, void *, uint32_t);
|
||||||
int lpfc_do_work(void *);
|
int lpfc_do_work(void *);
|
||||||
|
@ -147,6 +143,9 @@ LPFC_MBOXQ_t *lpfc_mbox_get(struct lpfc_hba *);
|
||||||
int lpfc_mem_alloc(struct lpfc_hba *);
|
int lpfc_mem_alloc(struct lpfc_hba *);
|
||||||
void lpfc_mem_free(struct lpfc_hba *);
|
void lpfc_mem_free(struct lpfc_hba *);
|
||||||
|
|
||||||
|
struct lpfc_iocbq * lpfc_sli_get_iocbq(struct lpfc_hba *);
|
||||||
|
void lpfc_sli_release_iocbq(struct lpfc_hba * phba, struct lpfc_iocbq * iocb);
|
||||||
|
uint16_t lpfc_sli_next_iotag(struct lpfc_hba * phba, struct lpfc_iocbq * iocb);
|
||||||
int lpfc_sli_hba_setup(struct lpfc_hba *);
|
int lpfc_sli_hba_setup(struct lpfc_hba *);
|
||||||
int lpfc_sli_hba_down(struct lpfc_hba *);
|
int lpfc_sli_hba_down(struct lpfc_hba *);
|
||||||
int lpfc_sli_issue_mbox(struct lpfc_hba *, LPFC_MBOXQ_t *, uint32_t);
|
int lpfc_sli_issue_mbox(struct lpfc_hba *, LPFC_MBOXQ_t *, uint32_t);
|
||||||
|
@ -182,15 +181,11 @@ struct lpfc_nodelist *lpfc_findnode_did(struct lpfc_hba * phba, uint32_t order,
|
||||||
int lpfc_sli_issue_mbox_wait(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmboxq,
|
int lpfc_sli_issue_mbox_wait(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmboxq,
|
||||||
uint32_t timeout);
|
uint32_t timeout);
|
||||||
|
|
||||||
int lpfc_sli_issue_iocb_wait_high_priority(struct lpfc_hba * phba,
|
int lpfc_sli_issue_iocb_wait(struct lpfc_hba * phba,
|
||||||
struct lpfc_sli_ring * pring,
|
struct lpfc_sli_ring * pring,
|
||||||
struct lpfc_iocbq * piocb,
|
struct lpfc_iocbq * piocb,
|
||||||
uint32_t flag,
|
struct lpfc_iocbq * prspiocbq,
|
||||||
struct lpfc_iocbq * prspiocbq,
|
uint32_t timeout);
|
||||||
uint32_t timeout);
|
|
||||||
void lpfc_sli_wake_iocb_high_priority(struct lpfc_hba * phba,
|
|
||||||
struct lpfc_iocbq * queue1,
|
|
||||||
struct lpfc_iocbq * queue2);
|
|
||||||
void lpfc_sli_abort_fcp_cmpl(struct lpfc_hba * phba,
|
void lpfc_sli_abort_fcp_cmpl(struct lpfc_hba * phba,
|
||||||
struct lpfc_iocbq * cmdiocb,
|
struct lpfc_iocbq * cmdiocb,
|
||||||
struct lpfc_iocbq * rspiocb);
|
struct lpfc_iocbq * rspiocb);
|
||||||
|
|
|
@ -224,18 +224,16 @@ lpfc_gen_req(struct lpfc_hba *phba, struct lpfc_dmabuf *bmp,
|
||||||
|
|
||||||
struct lpfc_sli *psli = &phba->sli;
|
struct lpfc_sli *psli = &phba->sli;
|
||||||
struct lpfc_sli_ring *pring = &psli->ring[LPFC_ELS_RING];
|
struct lpfc_sli_ring *pring = &psli->ring[LPFC_ELS_RING];
|
||||||
struct list_head *lpfc_iocb_list = &phba->lpfc_iocb_list;
|
|
||||||
IOCB_t *icmd;
|
IOCB_t *icmd;
|
||||||
struct lpfc_iocbq *geniocb = NULL;
|
struct lpfc_iocbq *geniocb;
|
||||||
|
|
||||||
/* Allocate buffer for command iocb */
|
/* Allocate buffer for command iocb */
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_remove_head(lpfc_iocb_list, geniocb, struct lpfc_iocbq, list);
|
geniocb = lpfc_sli_get_iocbq(phba);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
|
|
||||||
if (geniocb == NULL)
|
if (geniocb == NULL)
|
||||||
return 1;
|
return 1;
|
||||||
memset(geniocb, 0, sizeof (struct lpfc_iocbq));
|
|
||||||
|
|
||||||
icmd = &geniocb->iocb;
|
icmd = &geniocb->iocb;
|
||||||
icmd->un.genreq64.bdl.ulpIoTag32 = 0;
|
icmd->un.genreq64.bdl.ulpIoTag32 = 0;
|
||||||
|
@ -279,7 +277,7 @@ lpfc_gen_req(struct lpfc_hba *phba, struct lpfc_dmabuf *bmp,
|
||||||
geniocb->drvrTimeout = icmd->ulpTimeout + LPFC_DRVR_TIMEOUT;
|
geniocb->drvrTimeout = icmd->ulpTimeout + LPFC_DRVR_TIMEOUT;
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
if (lpfc_sli_issue_iocb(phba, pring, geniocb, 0) == IOCB_ERROR) {
|
if (lpfc_sli_issue_iocb(phba, pring, geniocb, 0) == IOCB_ERROR) {
|
||||||
list_add_tail(&geniocb->list, lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, geniocb);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -487,7 +485,7 @@ out:
|
||||||
kfree(inp);
|
kfree(inp);
|
||||||
kfree(bmp);
|
kfree(bmp);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_add_tail(&cmdiocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, cmdiocb);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -526,7 +524,7 @@ lpfc_cmpl_ct_cmd_rft_id(struct lpfc_hba * phba, struct lpfc_iocbq * cmdiocb,
|
||||||
kfree(inp);
|
kfree(inp);
|
||||||
kfree(bmp);
|
kfree(bmp);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_add_tail(&cmdiocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, cmdiocb);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -735,7 +733,7 @@ lpfc_cmpl_ct_cmd_fdmi(struct lpfc_hba * phba,
|
||||||
kfree(inp);
|
kfree(inp);
|
||||||
kfree(bmp);
|
kfree(bmp);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_add_tail(&cmdiocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, cmdiocb);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
|
@ -70,7 +70,6 @@ struct lpfc_nodelist {
|
||||||
struct timer_list nlp_tmofunc; /* Used for nodev tmo */
|
struct timer_list nlp_tmofunc; /* Used for nodev tmo */
|
||||||
struct fc_rport *rport; /* Corresponding FC transport
|
struct fc_rport *rport; /* Corresponding FC transport
|
||||||
port structure */
|
port structure */
|
||||||
struct lpfc_nodelist *nlp_rpi_hash_next;
|
|
||||||
struct lpfc_hba *nlp_phba;
|
struct lpfc_hba *nlp_phba;
|
||||||
struct lpfc_work_evt nodev_timeout_evt;
|
struct lpfc_work_evt nodev_timeout_evt;
|
||||||
struct lpfc_work_evt els_retry_evt;
|
struct lpfc_work_evt els_retry_evt;
|
||||||
|
|
|
@ -102,9 +102,8 @@ lpfc_prep_els_iocb(struct lpfc_hba * phba,
|
||||||
uint16_t cmdSize,
|
uint16_t cmdSize,
|
||||||
uint8_t retry, struct lpfc_nodelist * ndlp, uint32_t elscmd)
|
uint8_t retry, struct lpfc_nodelist * ndlp, uint32_t elscmd)
|
||||||
{
|
{
|
||||||
struct list_head *lpfc_iocb_list = &phba->lpfc_iocb_list;
|
|
||||||
struct lpfc_sli_ring *pring;
|
struct lpfc_sli_ring *pring;
|
||||||
struct lpfc_iocbq *elsiocb = NULL;
|
struct lpfc_iocbq *elsiocb;
|
||||||
struct lpfc_dmabuf *pcmd, *prsp, *pbuflist;
|
struct lpfc_dmabuf *pcmd, *prsp, *pbuflist;
|
||||||
struct ulp_bde64 *bpl;
|
struct ulp_bde64 *bpl;
|
||||||
IOCB_t *icmd;
|
IOCB_t *icmd;
|
||||||
|
@ -114,15 +113,13 @@ lpfc_prep_els_iocb(struct lpfc_hba * phba,
|
||||||
if (phba->hba_state < LPFC_LINK_UP)
|
if (phba->hba_state < LPFC_LINK_UP)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
|
|
||||||
/* Allocate buffer for command iocb */
|
/* Allocate buffer for command iocb */
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_remove_head(lpfc_iocb_list, elsiocb, struct lpfc_iocbq, list);
|
elsiocb = lpfc_sli_get_iocbq(phba);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
|
|
||||||
if (elsiocb == NULL)
|
if (elsiocb == NULL)
|
||||||
return NULL;
|
return NULL;
|
||||||
memset(elsiocb, 0, sizeof (struct lpfc_iocbq));
|
|
||||||
icmd = &elsiocb->iocb;
|
icmd = &elsiocb->iocb;
|
||||||
|
|
||||||
/* fill in BDEs for command */
|
/* fill in BDEs for command */
|
||||||
|
@ -133,7 +130,9 @@ lpfc_prep_els_iocb(struct lpfc_hba * phba,
|
||||||
if (pcmd)
|
if (pcmd)
|
||||||
kfree(pcmd);
|
kfree(pcmd);
|
||||||
|
|
||||||
list_add_tail(&elsiocb->list, lpfc_iocb_list);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
|
lpfc_sli_release_iocbq(phba, elsiocb);
|
||||||
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -150,7 +149,9 @@ lpfc_prep_els_iocb(struct lpfc_hba * phba,
|
||||||
kfree(prsp);
|
kfree(prsp);
|
||||||
lpfc_mbuf_free(phba, pcmd->virt, pcmd->phys);
|
lpfc_mbuf_free(phba, pcmd->virt, pcmd->phys);
|
||||||
kfree(pcmd);
|
kfree(pcmd);
|
||||||
list_add_tail(&elsiocb->list, lpfc_iocb_list);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
|
lpfc_sli_release_iocbq(phba, elsiocb);
|
||||||
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
INIT_LIST_HEAD(&prsp->list);
|
INIT_LIST_HEAD(&prsp->list);
|
||||||
|
@ -164,7 +165,9 @@ lpfc_prep_els_iocb(struct lpfc_hba * phba,
|
||||||
pbuflist->virt = lpfc_mbuf_alloc(phba, MEM_PRI,
|
pbuflist->virt = lpfc_mbuf_alloc(phba, MEM_PRI,
|
||||||
&pbuflist->phys);
|
&pbuflist->phys);
|
||||||
if (pbuflist == 0 || pbuflist->virt == 0) {
|
if (pbuflist == 0 || pbuflist->virt == 0) {
|
||||||
list_add_tail(&elsiocb->list, lpfc_iocb_list);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
|
lpfc_sli_release_iocbq(phba, elsiocb);
|
||||||
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
lpfc_mbuf_free(phba, pcmd->virt, pcmd->phys);
|
lpfc_mbuf_free(phba, pcmd->virt, pcmd->phys);
|
||||||
lpfc_mbuf_free(phba, prsp->virt, prsp->phys);
|
lpfc_mbuf_free(phba, prsp->virt, prsp->phys);
|
||||||
kfree(pcmd);
|
kfree(pcmd);
|
||||||
|
@ -596,10 +599,8 @@ lpfc_els_abort_flogi(struct lpfc_hba * phba)
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
} else {
|
} else
|
||||||
list_add_tail(&iocb->list,
|
lpfc_sli_release_iocbq(phba, iocb);
|
||||||
&phba->lpfc_iocb_list);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1713,7 +1714,7 @@ lpfc_els_free_iocb(struct lpfc_hba * phba, struct lpfc_iocbq * elsiocb)
|
||||||
kfree(buf_ptr);
|
kfree(buf_ptr);
|
||||||
}
|
}
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_add_tail(&elsiocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, elsiocb);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -2929,9 +2930,8 @@ lpfc_els_timeout_handler(struct lpfc_hba *phba)
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
(piocb->iocb_cmpl) (phba, piocb, piocb);
|
(piocb->iocb_cmpl) (phba, piocb, piocb);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
} else {
|
} else
|
||||||
list_add_tail(&piocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, piocb);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (phba->sli.ring[LPFC_ELS_RING].txcmplq_cnt) {
|
if (phba->sli.ring[LPFC_ELS_RING].txcmplq_cnt) {
|
||||||
phba->els_tmofunc.expires = jiffies + HZ * timeout;
|
phba->els_tmofunc.expires = jiffies + HZ * timeout;
|
||||||
|
@ -2996,7 +2996,7 @@ lpfc_els_flush_cmd(struct lpfc_hba * phba)
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
list_add_tail(&piocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, piocb);
|
||||||
}
|
}
|
||||||
|
|
||||||
list_for_each_entry_safe(piocb, tmp_iocb, &pring->txcmplq, list) {
|
list_for_each_entry_safe(piocb, tmp_iocb, &pring->txcmplq, list) {
|
||||||
|
@ -3033,7 +3033,7 @@ lpfc_els_flush_cmd(struct lpfc_hba * phba)
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
list_add_tail(&piocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, piocb);
|
||||||
}
|
}
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -890,10 +890,7 @@ lpfc_mbx_cmpl_fabric_reg_login(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmb)
|
||||||
|
|
||||||
pmb->context1 = NULL;
|
pmb->context1 = NULL;
|
||||||
|
|
||||||
if (ndlp->nlp_rpi != 0)
|
|
||||||
lpfc_findnode_remove_rpi(phba, ndlp->nlp_rpi);
|
|
||||||
ndlp->nlp_rpi = mb->un.varWords[0];
|
ndlp->nlp_rpi = mb->un.varWords[0];
|
||||||
lpfc_addnode_rpi(phba, ndlp, ndlp->nlp_rpi);
|
|
||||||
ndlp->nlp_type |= NLP_FABRIC;
|
ndlp->nlp_type |= NLP_FABRIC;
|
||||||
ndlp->nlp_state = NLP_STE_UNMAPPED_NODE;
|
ndlp->nlp_state = NLP_STE_UNMAPPED_NODE;
|
||||||
lpfc_nlp_list(phba, ndlp, NLP_UNMAPPED_LIST);
|
lpfc_nlp_list(phba, ndlp, NLP_UNMAPPED_LIST);
|
||||||
|
@ -981,10 +978,7 @@ lpfc_mbx_cmpl_ns_reg_login(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmb)
|
||||||
|
|
||||||
pmb->context1 = NULL;
|
pmb->context1 = NULL;
|
||||||
|
|
||||||
if (ndlp->nlp_rpi != 0)
|
|
||||||
lpfc_findnode_remove_rpi(phba, ndlp->nlp_rpi);
|
|
||||||
ndlp->nlp_rpi = mb->un.varWords[0];
|
ndlp->nlp_rpi = mb->un.varWords[0];
|
||||||
lpfc_addnode_rpi(phba, ndlp, ndlp->nlp_rpi);
|
|
||||||
ndlp->nlp_type |= NLP_FABRIC;
|
ndlp->nlp_type |= NLP_FABRIC;
|
||||||
ndlp->nlp_state = NLP_STE_UNMAPPED_NODE;
|
ndlp->nlp_state = NLP_STE_UNMAPPED_NODE;
|
||||||
lpfc_nlp_list(phba, ndlp, NLP_UNMAPPED_LIST);
|
lpfc_nlp_list(phba, ndlp, NLP_UNMAPPED_LIST);
|
||||||
|
@ -1028,6 +1022,7 @@ lpfc_register_remote_port(struct lpfc_hba * phba,
|
||||||
if (ndlp->nlp_type & NLP_FCP_INITIATOR)
|
if (ndlp->nlp_type & NLP_FCP_INITIATOR)
|
||||||
rport_ids.roles |= FC_RPORT_ROLE_FCP_INITIATOR;
|
rport_ids.roles |= FC_RPORT_ROLE_FCP_INITIATOR;
|
||||||
|
|
||||||
|
scsi_block_requests(phba->host);
|
||||||
ndlp->rport = rport = fc_remote_port_add(phba->host, 0, &rport_ids);
|
ndlp->rport = rport = fc_remote_port_add(phba->host, 0, &rport_ids);
|
||||||
if (!rport) {
|
if (!rport) {
|
||||||
dev_printk(KERN_WARNING, &phba->pcidev->dev,
|
dev_printk(KERN_WARNING, &phba->pcidev->dev,
|
||||||
|
@ -1044,6 +1039,23 @@ lpfc_register_remote_port(struct lpfc_hba * phba,
|
||||||
}
|
}
|
||||||
rdata = rport->dd_data;
|
rdata = rport->dd_data;
|
||||||
rdata->pnode = ndlp;
|
rdata->pnode = ndlp;
|
||||||
|
scsi_unblock_requests(phba->host);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
lpfc_unregister_remote_port(struct lpfc_hba * phba,
|
||||||
|
struct lpfc_nodelist * ndlp)
|
||||||
|
{
|
||||||
|
struct fc_rport *rport = ndlp->rport;
|
||||||
|
struct lpfc_rport_data *rdata = rport->dd_data;
|
||||||
|
|
||||||
|
ndlp->rport = NULL;
|
||||||
|
rdata->pnode = NULL;
|
||||||
|
scsi_block_requests(phba->host);
|
||||||
|
fc_remote_port_delete(rport);
|
||||||
|
scsi_unblock_requests(phba->host);
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -1260,7 +1272,7 @@ lpfc_nlp_list(struct lpfc_hba * phba, struct lpfc_nodelist * nlp, int list)
|
||||||
* may have removed the remote port.
|
* may have removed the remote port.
|
||||||
*/
|
*/
|
||||||
if ((rport_del != none) && nlp->rport)
|
if ((rport_del != none) && nlp->rport)
|
||||||
fc_remote_port_block(nlp->rport);
|
lpfc_unregister_remote_port(phba, nlp);
|
||||||
|
|
||||||
if (rport_add != none) {
|
if (rport_add != none) {
|
||||||
/*
|
/*
|
||||||
|
@ -1270,8 +1282,6 @@ lpfc_nlp_list(struct lpfc_hba * phba, struct lpfc_nodelist * nlp, int list)
|
||||||
*/
|
*/
|
||||||
if (!nlp->rport)
|
if (!nlp->rport)
|
||||||
lpfc_register_remote_port(phba, nlp);
|
lpfc_register_remote_port(phba, nlp);
|
||||||
else
|
|
||||||
fc_remote_port_unblock(nlp->rport);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* if we added to Mapped list, but the remote port
|
* if we added to Mapped list, but the remote port
|
||||||
|
@ -1435,10 +1445,9 @@ lpfc_no_rpi(struct lpfc_hba * phba, struct lpfc_nodelist * ndlp)
|
||||||
iocb, iocb);
|
iocb, iocb);
|
||||||
spin_lock_irq(phba->host->
|
spin_lock_irq(phba->host->
|
||||||
host_lock);
|
host_lock);
|
||||||
} else {
|
} else
|
||||||
list_add_tail(&iocb->list,
|
lpfc_sli_release_iocbq(phba,
|
||||||
&phba->lpfc_iocb_list);
|
iocb);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
|
@ -1472,7 +1481,6 @@ lpfc_unreg_rpi(struct lpfc_hba * phba, struct lpfc_nodelist * ndlp)
|
||||||
if (rc == MBX_NOT_FINISHED)
|
if (rc == MBX_NOT_FINISHED)
|
||||||
mempool_free( mbox, phba->mbox_mem_pool);
|
mempool_free( mbox, phba->mbox_mem_pool);
|
||||||
}
|
}
|
||||||
lpfc_findnode_remove_rpi(phba, ndlp->nlp_rpi);
|
|
||||||
lpfc_no_rpi(phba, ndlp);
|
lpfc_no_rpi(phba, ndlp);
|
||||||
ndlp->nlp_rpi = 0;
|
ndlp->nlp_rpi = 0;
|
||||||
return 1;
|
return 1;
|
||||||
|
@ -1490,7 +1498,6 @@ lpfc_freenode(struct lpfc_hba * phba, struct lpfc_nodelist * ndlp)
|
||||||
LPFC_MBOXQ_t *mb;
|
LPFC_MBOXQ_t *mb;
|
||||||
LPFC_MBOXQ_t *nextmb;
|
LPFC_MBOXQ_t *nextmb;
|
||||||
struct lpfc_dmabuf *mp;
|
struct lpfc_dmabuf *mp;
|
||||||
struct fc_rport *rport;
|
|
||||||
|
|
||||||
/* Cleanup node for NPort <nlp_DID> */
|
/* Cleanup node for NPort <nlp_DID> */
|
||||||
lpfc_printf_log(phba, KERN_INFO, LOG_NODE,
|
lpfc_printf_log(phba, KERN_INFO, LOG_NODE,
|
||||||
|
@ -1507,10 +1514,7 @@ lpfc_freenode(struct lpfc_hba * phba, struct lpfc_nodelist * ndlp)
|
||||||
* and flush cache's w/o generating flush errors.
|
* and flush cache's w/o generating flush errors.
|
||||||
*/
|
*/
|
||||||
if ((ndlp->rport) && !(phba->fc_flag & FC_UNLOADING)) {
|
if ((ndlp->rport) && !(phba->fc_flag & FC_UNLOADING)) {
|
||||||
rport = ndlp->rport;
|
lpfc_unregister_remote_port(phba, ndlp);
|
||||||
ndlp->rport = NULL;
|
|
||||||
fc_remote_port_unblock(rport);
|
|
||||||
fc_remote_port_delete(rport);
|
|
||||||
ndlp->nlp_sid = NLP_NO_SID;
|
ndlp->nlp_sid = NLP_NO_SID;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2422,10 +2426,7 @@ lpfc_mbx_cmpl_fdmi_reg_login(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmb)
|
||||||
|
|
||||||
pmb->context1 = NULL;
|
pmb->context1 = NULL;
|
||||||
|
|
||||||
if (ndlp->nlp_rpi != 0)
|
|
||||||
lpfc_findnode_remove_rpi(phba, ndlp->nlp_rpi);
|
|
||||||
ndlp->nlp_rpi = mb->un.varWords[0];
|
ndlp->nlp_rpi = mb->un.varWords[0];
|
||||||
lpfc_addnode_rpi(phba, ndlp, ndlp->nlp_rpi);
|
|
||||||
ndlp->nlp_type |= NLP_FABRIC;
|
ndlp->nlp_type |= NLP_FABRIC;
|
||||||
ndlp->nlp_state = NLP_STE_UNMAPPED_NODE;
|
ndlp->nlp_state = NLP_STE_UNMAPPED_NODE;
|
||||||
lpfc_nlp_list(phba, ndlp, NLP_UNMAPPED_LIST);
|
lpfc_nlp_list(phba, ndlp, NLP_UNMAPPED_LIST);
|
||||||
|
@ -2451,75 +2452,28 @@ lpfc_mbx_cmpl_fdmi_reg_login(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmb)
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This routine looks up the ndlp hash
|
* This routine looks up the ndlp lists
|
||||||
* table for the given RPI. If rpi found
|
* for the given RPI. If rpi found
|
||||||
* it return the node list pointer
|
* it return the node list pointer
|
||||||
* else return 0.
|
* else return NULL.
|
||||||
*/
|
*/
|
||||||
struct lpfc_nodelist *
|
struct lpfc_nodelist *
|
||||||
lpfc_findnode_rpi(struct lpfc_hba * phba, uint16_t rpi)
|
lpfc_findnode_rpi(struct lpfc_hba * phba, uint16_t rpi)
|
||||||
{
|
{
|
||||||
struct lpfc_nodelist *ret;
|
struct lpfc_nodelist *ndlp;
|
||||||
|
struct list_head * lists[]={&phba->fc_nlpunmap_list,
|
||||||
|
&phba->fc_nlpmap_list,
|
||||||
|
&phba->fc_plogi_list,
|
||||||
|
&phba->fc_adisc_list,
|
||||||
|
&phba->fc_reglogin_list};
|
||||||
|
int i;
|
||||||
|
|
||||||
ret = phba->fc_nlplookup[LPFC_RPI_HASH_FUNC(rpi)];
|
for (i = 0; i < ARRAY_SIZE(lists); i++ )
|
||||||
while ((ret != 0) && (ret->nlp_rpi != rpi)) {
|
list_for_each_entry(ndlp, lists[i], nlp_listp)
|
||||||
ret = ret->nlp_rpi_hash_next;
|
if (ndlp->nlp_rpi == rpi)
|
||||||
}
|
return (ndlp);
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
return NULL;
|
||||||
* This routine looks up the ndlp hash table for the
|
|
||||||
* given RPI. If rpi found it return the node list
|
|
||||||
* pointer else return 0 after deleting the entry
|
|
||||||
* from hash table.
|
|
||||||
*/
|
|
||||||
struct lpfc_nodelist *
|
|
||||||
lpfc_findnode_remove_rpi(struct lpfc_hba * phba, uint16_t rpi)
|
|
||||||
{
|
|
||||||
struct lpfc_nodelist *ret, *temp;;
|
|
||||||
|
|
||||||
ret = phba->fc_nlplookup[LPFC_RPI_HASH_FUNC(rpi)];
|
|
||||||
if (ret == 0)
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
if (ret->nlp_rpi == rpi) {
|
|
||||||
phba->fc_nlplookup[LPFC_RPI_HASH_FUNC(rpi)] =
|
|
||||||
ret->nlp_rpi_hash_next;
|
|
||||||
ret->nlp_rpi_hash_next = NULL;
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
while ((ret->nlp_rpi_hash_next != 0) &&
|
|
||||||
(ret->nlp_rpi_hash_next->nlp_rpi != rpi)) {
|
|
||||||
ret = ret->nlp_rpi_hash_next;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ret->nlp_rpi_hash_next != 0) {
|
|
||||||
temp = ret->nlp_rpi_hash_next;
|
|
||||||
ret->nlp_rpi_hash_next = temp->nlp_rpi_hash_next;
|
|
||||||
temp->nlp_rpi_hash_next = NULL;
|
|
||||||
return temp;
|
|
||||||
} else {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This routine adds the node list entry to the
|
|
||||||
* ndlp hash table.
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
lpfc_addnode_rpi(struct lpfc_hba * phba, struct lpfc_nodelist * ndlp,
|
|
||||||
uint16_t rpi)
|
|
||||||
{
|
|
||||||
|
|
||||||
uint32_t index;
|
|
||||||
|
|
||||||
index = LPFC_RPI_HASH_FUNC(rpi);
|
|
||||||
ndlp->nlp_rpi_hash_next = phba->fc_nlplookup[index];
|
|
||||||
phba->fc_nlplookup[index] = ndlp;
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
|
@ -537,12 +537,6 @@ lpfc_handle_eratt(struct lpfc_hba * phba)
|
||||||
|
|
||||||
lpfc_offline(phba);
|
lpfc_offline(phba);
|
||||||
|
|
||||||
/*
|
|
||||||
* Restart all traffic to this host. Since the fc_transport
|
|
||||||
* block functions (future) were not called in lpfc_offline,
|
|
||||||
* don't call them here.
|
|
||||||
*/
|
|
||||||
scsi_unblock_requests(phba->host);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -772,10 +766,12 @@ lpfc_get_hba_model_desc(struct lpfc_hba * phba, uint8_t * mdp, uint8_t * descp)
|
||||||
{
|
{
|
||||||
lpfc_vpd_t *vp;
|
lpfc_vpd_t *vp;
|
||||||
uint32_t id;
|
uint32_t id;
|
||||||
|
uint8_t hdrtype;
|
||||||
char str[16];
|
char str[16];
|
||||||
|
|
||||||
vp = &phba->vpd;
|
vp = &phba->vpd;
|
||||||
pci_read_config_dword(phba->pcidev, PCI_VENDOR_ID, &id);
|
pci_read_config_dword(phba->pcidev, PCI_VENDOR_ID, &id);
|
||||||
|
pci_read_config_byte(phba->pcidev, PCI_HEADER_TYPE, &hdrtype);
|
||||||
|
|
||||||
switch ((id >> 16) & 0xffff) {
|
switch ((id >> 16) & 0xffff) {
|
||||||
case PCI_DEVICE_ID_FIREFLY:
|
case PCI_DEVICE_ID_FIREFLY:
|
||||||
|
@ -803,7 +799,10 @@ lpfc_get_hba_model_desc(struct lpfc_hba * phba, uint8_t * mdp, uint8_t * descp)
|
||||||
strcpy(str, "LP9802 2");
|
strcpy(str, "LP9802 2");
|
||||||
break;
|
break;
|
||||||
case PCI_DEVICE_ID_THOR:
|
case PCI_DEVICE_ID_THOR:
|
||||||
strcpy(str, "LP10000 2");
|
if (hdrtype == 0x80)
|
||||||
|
strcpy(str, "LP10000DC 2");
|
||||||
|
else
|
||||||
|
strcpy(str, "LP10000 2");
|
||||||
break;
|
break;
|
||||||
case PCI_DEVICE_ID_VIPER:
|
case PCI_DEVICE_ID_VIPER:
|
||||||
strcpy(str, "LPX1000 10");
|
strcpy(str, "LPX1000 10");
|
||||||
|
@ -812,10 +811,16 @@ lpfc_get_hba_model_desc(struct lpfc_hba * phba, uint8_t * mdp, uint8_t * descp)
|
||||||
strcpy(str, "LP982 2");
|
strcpy(str, "LP982 2");
|
||||||
break;
|
break;
|
||||||
case PCI_DEVICE_ID_TFLY:
|
case PCI_DEVICE_ID_TFLY:
|
||||||
strcpy(str, "LP1050 2");
|
if (hdrtype == 0x80)
|
||||||
|
strcpy(str, "LP1050DC 2");
|
||||||
|
else
|
||||||
|
strcpy(str, "LP1050 2");
|
||||||
break;
|
break;
|
||||||
case PCI_DEVICE_ID_HELIOS:
|
case PCI_DEVICE_ID_HELIOS:
|
||||||
strcpy(str, "LP11000 4");
|
if (hdrtype == 0x80)
|
||||||
|
strcpy(str, "LP11002 4");
|
||||||
|
else
|
||||||
|
strcpy(str, "LP11000 4");
|
||||||
break;
|
break;
|
||||||
case PCI_DEVICE_ID_BMID:
|
case PCI_DEVICE_ID_BMID:
|
||||||
strcpy(str, "LP1150 4");
|
strcpy(str, "LP1150 4");
|
||||||
|
@ -824,13 +829,16 @@ lpfc_get_hba_model_desc(struct lpfc_hba * phba, uint8_t * mdp, uint8_t * descp)
|
||||||
strcpy(str, "LP111 4");
|
strcpy(str, "LP111 4");
|
||||||
break;
|
break;
|
||||||
case PCI_DEVICE_ID_ZEPHYR:
|
case PCI_DEVICE_ID_ZEPHYR:
|
||||||
strcpy(str, "LP11000e 4");
|
if (hdrtype == 0x80)
|
||||||
|
strcpy(str, "LPe11002 4");
|
||||||
|
else
|
||||||
|
strcpy(str, "LPe11000 4");
|
||||||
break;
|
break;
|
||||||
case PCI_DEVICE_ID_ZMID:
|
case PCI_DEVICE_ID_ZMID:
|
||||||
strcpy(str, "LP1150e 4");
|
strcpy(str, "LPe1150 4");
|
||||||
break;
|
break;
|
||||||
case PCI_DEVICE_ID_ZSMB:
|
case PCI_DEVICE_ID_ZSMB:
|
||||||
strcpy(str, "LP111e 4");
|
strcpy(str, "LPe111 4");
|
||||||
break;
|
break;
|
||||||
case PCI_DEVICE_ID_LP101:
|
case PCI_DEVICE_ID_LP101:
|
||||||
strcpy(str, "LP101 2");
|
strcpy(str, "LP101 2");
|
||||||
|
@ -862,8 +870,7 @@ lpfc_post_buffer(struct lpfc_hba * phba, struct lpfc_sli_ring * pring, int cnt,
|
||||||
int type)
|
int type)
|
||||||
{
|
{
|
||||||
IOCB_t *icmd;
|
IOCB_t *icmd;
|
||||||
struct list_head *lpfc_iocb_list = &phba->lpfc_iocb_list;
|
struct lpfc_iocbq *iocb;
|
||||||
struct lpfc_iocbq *iocb = NULL;
|
|
||||||
struct lpfc_dmabuf *mp1, *mp2;
|
struct lpfc_dmabuf *mp1, *mp2;
|
||||||
|
|
||||||
cnt += pring->missbufcnt;
|
cnt += pring->missbufcnt;
|
||||||
|
@ -872,13 +879,12 @@ lpfc_post_buffer(struct lpfc_hba * phba, struct lpfc_sli_ring * pring, int cnt,
|
||||||
while (cnt > 0) {
|
while (cnt > 0) {
|
||||||
/* Allocate buffer for command iocb */
|
/* Allocate buffer for command iocb */
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_remove_head(lpfc_iocb_list, iocb, struct lpfc_iocbq, list);
|
iocb = lpfc_sli_get_iocbq(phba);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
if (iocb == NULL) {
|
if (iocb == NULL) {
|
||||||
pring->missbufcnt = cnt;
|
pring->missbufcnt = cnt;
|
||||||
return cnt;
|
return cnt;
|
||||||
}
|
}
|
||||||
memset(iocb, 0, sizeof (struct lpfc_iocbq));
|
|
||||||
icmd = &iocb->iocb;
|
icmd = &iocb->iocb;
|
||||||
|
|
||||||
/* 2 buffers can be posted per command */
|
/* 2 buffers can be posted per command */
|
||||||
|
@ -891,7 +897,7 @@ lpfc_post_buffer(struct lpfc_hba * phba, struct lpfc_sli_ring * pring, int cnt,
|
||||||
if (mp1)
|
if (mp1)
|
||||||
kfree(mp1);
|
kfree(mp1);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_add_tail(&iocb->list, lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, iocb);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
pring->missbufcnt = cnt;
|
pring->missbufcnt = cnt;
|
||||||
return cnt;
|
return cnt;
|
||||||
|
@ -910,7 +916,7 @@ lpfc_post_buffer(struct lpfc_hba * phba, struct lpfc_sli_ring * pring, int cnt,
|
||||||
lpfc_mbuf_free(phba, mp1->virt, mp1->phys);
|
lpfc_mbuf_free(phba, mp1->virt, mp1->phys);
|
||||||
kfree(mp1);
|
kfree(mp1);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_add_tail(&iocb->list, lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, iocb);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
pring->missbufcnt = cnt;
|
pring->missbufcnt = cnt;
|
||||||
return cnt;
|
return cnt;
|
||||||
|
@ -947,7 +953,7 @@ lpfc_post_buffer(struct lpfc_hba * phba, struct lpfc_sli_ring * pring, int cnt,
|
||||||
kfree(mp2);
|
kfree(mp2);
|
||||||
cnt++;
|
cnt++;
|
||||||
}
|
}
|
||||||
list_add_tail(&iocb->list, lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, iocb);
|
||||||
pring->missbufcnt = cnt;
|
pring->missbufcnt = cnt;
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return cnt;
|
return cnt;
|
||||||
|
@ -1226,12 +1232,6 @@ lpfc_online(struct lpfc_hba * phba)
|
||||||
phba->fc_flag &= ~FC_OFFLINE_MODE;
|
phba->fc_flag &= ~FC_OFFLINE_MODE;
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
|
|
||||||
/*
|
|
||||||
* Restart all traffic to this host. Since the fc_transport block
|
|
||||||
* functions (future) were not called in lpfc_offline, don't call them
|
|
||||||
* here.
|
|
||||||
*/
|
|
||||||
scsi_unblock_requests(phba->host);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1249,13 +1249,6 @@ lpfc_offline(struct lpfc_hba * phba)
|
||||||
if (phba->fc_flag & FC_OFFLINE_MODE)
|
if (phba->fc_flag & FC_OFFLINE_MODE)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/*
|
|
||||||
* Don't call the fc_transport block api (future). The device is
|
|
||||||
* going offline and causing a timer to fire in the midlayer is
|
|
||||||
* unproductive. Just block all new requests until the driver
|
|
||||||
* comes back online.
|
|
||||||
*/
|
|
||||||
scsi_block_requests(phba->host);
|
|
||||||
psli = &phba->sli;
|
psli = &phba->sli;
|
||||||
pring = &psli->ring[psli->fcp_ring];
|
pring = &psli->ring[psli->fcp_ring];
|
||||||
|
|
||||||
|
@ -1333,6 +1326,7 @@ lpfc_pci_probe_one(struct pci_dev *pdev, const struct pci_device_id *pid)
|
||||||
unsigned long bar0map_len, bar2map_len;
|
unsigned long bar0map_len, bar2map_len;
|
||||||
int error = -ENODEV, retval;
|
int error = -ENODEV, retval;
|
||||||
int i;
|
int i;
|
||||||
|
uint16_t iotag;
|
||||||
|
|
||||||
if (pci_enable_device(pdev))
|
if (pci_enable_device(pdev))
|
||||||
goto out;
|
goto out;
|
||||||
|
@ -1434,6 +1428,7 @@ lpfc_pci_probe_one(struct pci_dev *pdev, const struct pci_device_id *pid)
|
||||||
if (!phba->slim2p)
|
if (!phba->slim2p)
|
||||||
goto out_iounmap;
|
goto out_iounmap;
|
||||||
|
|
||||||
|
memset(phba->slim2p, 0, SLI2_SLIM_SIZE);
|
||||||
|
|
||||||
/* Initialize the SLI Layer to run with lpfc HBAs. */
|
/* Initialize the SLI Layer to run with lpfc HBAs. */
|
||||||
lpfc_sli_setup(phba);
|
lpfc_sli_setup(phba);
|
||||||
|
@ -1456,6 +1451,15 @@ lpfc_pci_probe_one(struct pci_dev *pdev, const struct pci_device_id *pid)
|
||||||
}
|
}
|
||||||
|
|
||||||
memset(iocbq_entry, 0, sizeof(struct lpfc_iocbq));
|
memset(iocbq_entry, 0, sizeof(struct lpfc_iocbq));
|
||||||
|
iotag = lpfc_sli_next_iotag(phba, iocbq_entry);
|
||||||
|
if (iotag == 0) {
|
||||||
|
kfree (iocbq_entry);
|
||||||
|
printk(KERN_ERR "%s: failed to allocate IOTAG. "
|
||||||
|
"Unloading driver.\n",
|
||||||
|
__FUNCTION__);
|
||||||
|
error = -ENOMEM;
|
||||||
|
goto out_free_iocbq;
|
||||||
|
}
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_add(&iocbq_entry->list, &phba->lpfc_iocb_list);
|
list_add(&iocbq_entry->list, &phba->lpfc_iocb_list);
|
||||||
phba->total_iocbq_bufs++;
|
phba->total_iocbq_bufs++;
|
||||||
|
@ -1702,6 +1706,7 @@ MODULE_DEVICE_TABLE(pci, lpfc_id_table);
|
||||||
|
|
||||||
static struct pci_driver lpfc_driver = {
|
static struct pci_driver lpfc_driver = {
|
||||||
.name = LPFC_DRIVER_NAME,
|
.name = LPFC_DRIVER_NAME,
|
||||||
|
.owner = THIS_MODULE,
|
||||||
.id_table = lpfc_id_table,
|
.id_table = lpfc_id_table,
|
||||||
.probe = lpfc_pci_probe_one,
|
.probe = lpfc_pci_probe_one,
|
||||||
.remove = __devexit_p(lpfc_pci_remove_one),
|
.remove = __devexit_p(lpfc_pci_remove_one),
|
||||||
|
|
|
@ -531,6 +531,7 @@ lpfc_config_port(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmb)
|
||||||
size_t offset;
|
size_t offset;
|
||||||
struct lpfc_hgp hgp;
|
struct lpfc_hgp hgp;
|
||||||
void __iomem *to_slim;
|
void __iomem *to_slim;
|
||||||
|
int i;
|
||||||
|
|
||||||
memset(pmb, 0, sizeof(LPFC_MBOXQ_t));
|
memset(pmb, 0, sizeof(LPFC_MBOXQ_t));
|
||||||
mb->mbxCommand = MBX_CONFIG_PORT;
|
mb->mbxCommand = MBX_CONFIG_PORT;
|
||||||
|
@ -587,7 +588,11 @@ lpfc_config_port(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmb)
|
||||||
/* write HGP data to SLIM at the required longword offset */
|
/* write HGP data to SLIM at the required longword offset */
|
||||||
memset(&hgp, 0, sizeof(struct lpfc_hgp));
|
memset(&hgp, 0, sizeof(struct lpfc_hgp));
|
||||||
to_slim = phba->MBslimaddr + (SLIMOFF*sizeof (uint32_t));
|
to_slim = phba->MBslimaddr + (SLIMOFF*sizeof (uint32_t));
|
||||||
lpfc_memcpy_to_slim(to_slim, &hgp, sizeof(struct lpfc_hgp));
|
|
||||||
|
for (i=0; i < phba->sli.num_rings; i++) {
|
||||||
|
lpfc_memcpy_to_slim(to_slim, &hgp, sizeof(struct lpfc_hgp));
|
||||||
|
to_slim += sizeof (struct lpfc_hgp);
|
||||||
|
}
|
||||||
|
|
||||||
/* Setup Port Group ring pointer */
|
/* Setup Port Group ring pointer */
|
||||||
offset = (uint8_t *)&phba->slim2p->mbx.us.s2.port -
|
offset = (uint8_t *)&phba->slim2p->mbx.us.s2.port -
|
||||||
|
|
|
@ -187,10 +187,8 @@ lpfc_els_abort(struct lpfc_hba * phba, struct lpfc_nodelist * ndlp,
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
} else {
|
} else
|
||||||
list_add_tail(&iocb->list,
|
lpfc_sli_release_iocbq(phba, iocb);
|
||||||
&phba->lpfc_iocb_list);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -232,10 +230,8 @@ lpfc_els_abort(struct lpfc_hba * phba, struct lpfc_nodelist * ndlp,
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
} else {
|
} else
|
||||||
list_add_tail(&iocb->list,
|
lpfc_sli_release_iocbq(phba, iocb);
|
||||||
&phba->lpfc_iocb_list);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1086,11 +1082,7 @@ lpfc_cmpl_reglogin_reglogin_issue(struct lpfc_hba * phba,
|
||||||
return (ndlp->nlp_state);
|
return (ndlp->nlp_state);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ndlp->nlp_rpi != 0)
|
|
||||||
lpfc_findnode_remove_rpi(phba, ndlp->nlp_rpi);
|
|
||||||
|
|
||||||
ndlp->nlp_rpi = mb->un.varWords[0];
|
ndlp->nlp_rpi = mb->un.varWords[0];
|
||||||
lpfc_addnode_rpi(phba, ndlp, ndlp->nlp_rpi);
|
|
||||||
|
|
||||||
/* Only if we are not a fabric nport do we issue PRLI */
|
/* Only if we are not a fabric nport do we issue PRLI */
|
||||||
if (!(ndlp->nlp_type & NLP_FABRIC)) {
|
if (!(ndlp->nlp_type & NLP_FABRIC)) {
|
||||||
|
@ -1593,12 +1585,7 @@ lpfc_cmpl_reglogin_npr_node(struct lpfc_hba * phba,
|
||||||
pmb = (LPFC_MBOXQ_t *) arg;
|
pmb = (LPFC_MBOXQ_t *) arg;
|
||||||
mb = &pmb->mb;
|
mb = &pmb->mb;
|
||||||
|
|
||||||
/* save rpi */
|
|
||||||
if (ndlp->nlp_rpi != 0)
|
|
||||||
lpfc_findnode_remove_rpi(phba, ndlp->nlp_rpi);
|
|
||||||
|
|
||||||
ndlp->nlp_rpi = mb->un.varWords[0];
|
ndlp->nlp_rpi = mb->un.varWords[0];
|
||||||
lpfc_addnode_rpi(phba, ndlp, ndlp->nlp_rpi);
|
|
||||||
|
|
||||||
return (ndlp->nlp_state);
|
return (ndlp->nlp_state);
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,12 +50,13 @@
|
||||||
* and the BPL BDE is setup in the IOCB.
|
* and the BPL BDE is setup in the IOCB.
|
||||||
*/
|
*/
|
||||||
static struct lpfc_scsi_buf *
|
static struct lpfc_scsi_buf *
|
||||||
lpfc_get_scsi_buf(struct lpfc_hba * phba)
|
lpfc_new_scsi_buf(struct lpfc_hba * phba)
|
||||||
{
|
{
|
||||||
struct lpfc_scsi_buf *psb;
|
struct lpfc_scsi_buf *psb;
|
||||||
struct ulp_bde64 *bpl;
|
struct ulp_bde64 *bpl;
|
||||||
IOCB_t *iocb;
|
IOCB_t *iocb;
|
||||||
dma_addr_t pdma_phys;
|
dma_addr_t pdma_phys;
|
||||||
|
uint16_t iotag;
|
||||||
|
|
||||||
psb = kmalloc(sizeof(struct lpfc_scsi_buf), GFP_KERNEL);
|
psb = kmalloc(sizeof(struct lpfc_scsi_buf), GFP_KERNEL);
|
||||||
if (!psb)
|
if (!psb)
|
||||||
|
@ -79,6 +80,16 @@ lpfc_get_scsi_buf(struct lpfc_hba * phba)
|
||||||
/* Initialize virtual ptrs to dma_buf region. */
|
/* Initialize virtual ptrs to dma_buf region. */
|
||||||
memset(psb->data, 0, phba->cfg_sg_dma_buf_size);
|
memset(psb->data, 0, phba->cfg_sg_dma_buf_size);
|
||||||
|
|
||||||
|
/* Allocate iotag for psb->cur_iocbq. */
|
||||||
|
iotag = lpfc_sli_next_iotag(phba, &psb->cur_iocbq);
|
||||||
|
if (iotag == 0) {
|
||||||
|
pci_pool_free(phba->lpfc_scsi_dma_buf_pool,
|
||||||
|
psb->data, psb->dma_handle);
|
||||||
|
kfree (psb);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
psb->cur_iocbq.iocb_flag |= LPFC_IO_FCP;
|
||||||
|
|
||||||
psb->fcp_cmnd = psb->data;
|
psb->fcp_cmnd = psb->data;
|
||||||
psb->fcp_rsp = psb->data + sizeof(struct fcp_cmnd);
|
psb->fcp_rsp = psb->data + sizeof(struct fcp_cmnd);
|
||||||
psb->fcp_bpl = psb->data + sizeof(struct fcp_cmnd) +
|
psb->fcp_bpl = psb->data + sizeof(struct fcp_cmnd) +
|
||||||
|
@ -125,11 +136,19 @@ lpfc_get_scsi_buf(struct lpfc_hba * phba)
|
||||||
return psb;
|
return psb;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
struct lpfc_scsi_buf*
|
||||||
lpfc_free_scsi_buf(struct lpfc_scsi_buf * psb)
|
lpfc_sli_get_scsi_buf(struct lpfc_hba * phba)
|
||||||
{
|
{
|
||||||
struct lpfc_hba *phba = psb->scsi_hba;
|
struct lpfc_scsi_buf * lpfc_cmd = NULL;
|
||||||
|
struct list_head *scsi_buf_list = &phba->lpfc_scsi_buf_list;
|
||||||
|
|
||||||
|
list_remove_head(scsi_buf_list, lpfc_cmd, struct lpfc_scsi_buf, list);
|
||||||
|
return lpfc_cmd;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
lpfc_release_scsi_buf(struct lpfc_hba * phba, struct lpfc_scsi_buf * psb)
|
||||||
|
{
|
||||||
/*
|
/*
|
||||||
* There are only two special cases to consider. (1) the scsi command
|
* There are only two special cases to consider. (1) the scsi command
|
||||||
* requested scatter-gather usage or (2) the scsi command allocated
|
* requested scatter-gather usage or (2) the scsi command allocated
|
||||||
|
@ -147,6 +166,7 @@ lpfc_free_scsi_buf(struct lpfc_scsi_buf * psb)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
psb->pCmd = NULL;
|
||||||
list_add_tail(&psb->list, &phba->lpfc_scsi_buf_list);
|
list_add_tail(&psb->list, &phba->lpfc_scsi_buf_list);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -403,14 +423,9 @@ lpfc_scsi_cmd_iocb_cmpl(struct lpfc_hba *phba, struct lpfc_iocbq *pIocbIn,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pnode) {
|
if ((pnode == NULL )
|
||||||
if (pnode->nlp_state != NLP_STE_MAPPED_NODE)
|
|| (pnode->nlp_state != NLP_STE_MAPPED_NODE))
|
||||||
cmd->result = ScsiResult(DID_BUS_BUSY,
|
cmd->result = ScsiResult(DID_BUS_BUSY, SAM_STAT_BUSY);
|
||||||
SAM_STAT_BUSY);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
cmd->result = ScsiResult(DID_NO_CONNECT, 0);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
cmd->result = ScsiResult(DID_OK, 0);
|
cmd->result = ScsiResult(DID_OK, 0);
|
||||||
}
|
}
|
||||||
|
@ -426,12 +441,11 @@ lpfc_scsi_cmd_iocb_cmpl(struct lpfc_hba *phba, struct lpfc_iocbq *pIocbIn,
|
||||||
*lp, *(lp + 3), cmd->retries, cmd->resid);
|
*lp, *(lp + 3), cmd->retries, cmd->resid);
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_lock_irqsave(phba->host->host_lock, iflag);
|
|
||||||
lpfc_free_scsi_buf(lpfc_cmd);
|
|
||||||
cmd->host_scribble = NULL;
|
|
||||||
spin_unlock_irqrestore(phba->host->host_lock, iflag);
|
|
||||||
|
|
||||||
cmd->scsi_done(cmd);
|
cmd->scsi_done(cmd);
|
||||||
|
|
||||||
|
spin_lock_irqsave(phba->host->host_lock, iflag);
|
||||||
|
lpfc_release_scsi_buf(phba, lpfc_cmd);
|
||||||
|
spin_unlock_irqrestore(phba->host->host_lock, iflag);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -539,7 +553,7 @@ lpfc_scsi_prep_task_mgmt_cmd(struct lpfc_hba *phba,
|
||||||
struct lpfc_rport_data *rdata = scsi_dev->hostdata;
|
struct lpfc_rport_data *rdata = scsi_dev->hostdata;
|
||||||
struct lpfc_nodelist *ndlp = rdata->pnode;
|
struct lpfc_nodelist *ndlp = rdata->pnode;
|
||||||
|
|
||||||
if ((ndlp == 0) || (ndlp->nlp_state != NLP_STE_MAPPED_NODE)) {
|
if ((ndlp == NULL) || (ndlp->nlp_state != NLP_STE_MAPPED_NODE)) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -618,8 +632,7 @@ static int
|
||||||
lpfc_scsi_tgt_reset(struct lpfc_scsi_buf * lpfc_cmd, struct lpfc_hba * phba)
|
lpfc_scsi_tgt_reset(struct lpfc_scsi_buf * lpfc_cmd, struct lpfc_hba * phba)
|
||||||
{
|
{
|
||||||
struct lpfc_iocbq *iocbq;
|
struct lpfc_iocbq *iocbq;
|
||||||
struct lpfc_iocbq *iocbqrsp = NULL;
|
struct lpfc_iocbq *iocbqrsp;
|
||||||
struct list_head *lpfc_iocb_list = &phba->lpfc_iocb_list;
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ret = lpfc_scsi_prep_task_mgmt_cmd(phba, lpfc_cmd, FCP_TARGET_RESET);
|
ret = lpfc_scsi_prep_task_mgmt_cmd(phba, lpfc_cmd, FCP_TARGET_RESET);
|
||||||
|
@ -628,17 +641,14 @@ lpfc_scsi_tgt_reset(struct lpfc_scsi_buf * lpfc_cmd, struct lpfc_hba * phba)
|
||||||
|
|
||||||
lpfc_cmd->scsi_hba = phba;
|
lpfc_cmd->scsi_hba = phba;
|
||||||
iocbq = &lpfc_cmd->cur_iocbq;
|
iocbq = &lpfc_cmd->cur_iocbq;
|
||||||
list_remove_head(lpfc_iocb_list, iocbqrsp, struct lpfc_iocbq, list);
|
iocbqrsp = lpfc_sli_get_iocbq(phba);
|
||||||
|
|
||||||
if (!iocbqrsp)
|
if (!iocbqrsp)
|
||||||
return FAILED;
|
return FAILED;
|
||||||
memset(iocbqrsp, 0, sizeof (struct lpfc_iocbq));
|
|
||||||
|
|
||||||
iocbq->iocb_flag |= LPFC_IO_POLL;
|
ret = lpfc_sli_issue_iocb_wait(phba,
|
||||||
ret = lpfc_sli_issue_iocb_wait_high_priority(phba,
|
&phba->sli.ring[phba->sli.fcp_ring],
|
||||||
&phba->sli.ring[phba->sli.fcp_ring],
|
iocbq, iocbqrsp, lpfc_cmd->timeout);
|
||||||
iocbq, SLI_IOCB_HIGH_PRIORITY,
|
|
||||||
iocbqrsp,
|
|
||||||
lpfc_cmd->timeout);
|
|
||||||
if (ret != IOCB_SUCCESS) {
|
if (ret != IOCB_SUCCESS) {
|
||||||
lpfc_cmd->status = IOSTAT_DRIVER_REJECT;
|
lpfc_cmd->status = IOSTAT_DRIVER_REJECT;
|
||||||
ret = FAILED;
|
ret = FAILED;
|
||||||
|
@ -651,45 +661,10 @@ lpfc_scsi_tgt_reset(struct lpfc_scsi_buf * lpfc_cmd, struct lpfc_hba * phba)
|
||||||
lpfc_cmd->status = IOSTAT_DRIVER_REJECT;
|
lpfc_cmd->status = IOSTAT_DRIVER_REJECT;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
lpfc_sli_release_iocbq(phba, iocbqrsp);
|
||||||
* All outstanding txcmplq I/Os should have been aborted by the target.
|
|
||||||
* Unfortunately, some targets do not abide by this forcing the driver
|
|
||||||
* to double check.
|
|
||||||
*/
|
|
||||||
lpfc_sli_abort_iocb(phba, &phba->sli.ring[phba->sli.fcp_ring],
|
|
||||||
lpfc_cmd->pCmd->device->id,
|
|
||||||
lpfc_cmd->pCmd->device->lun, 0, LPFC_CTX_TGT);
|
|
||||||
|
|
||||||
/* Return response IOCB to free list. */
|
|
||||||
list_add_tail(&iocbqrsp->list, lpfc_iocb_list);
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
lpfc_scsi_cmd_iocb_cleanup (struct lpfc_hba *phba, struct lpfc_iocbq *pIocbIn,
|
|
||||||
struct lpfc_iocbq *pIocbOut)
|
|
||||||
{
|
|
||||||
unsigned long iflag;
|
|
||||||
struct lpfc_scsi_buf *lpfc_cmd =
|
|
||||||
(struct lpfc_scsi_buf *) pIocbIn->context1;
|
|
||||||
|
|
||||||
spin_lock_irqsave(phba->host->host_lock, iflag);
|
|
||||||
lpfc_free_scsi_buf(lpfc_cmd);
|
|
||||||
spin_unlock_irqrestore(phba->host->host_lock, iflag);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
lpfc_scsi_cmd_iocb_cmpl_aborted(struct lpfc_hba *phba,
|
|
||||||
struct lpfc_iocbq *pIocbIn,
|
|
||||||
struct lpfc_iocbq *pIocbOut)
|
|
||||||
{
|
|
||||||
struct scsi_cmnd *ml_cmd =
|
|
||||||
((struct lpfc_scsi_buf *) pIocbIn->context1)->pCmd;
|
|
||||||
|
|
||||||
lpfc_scsi_cmd_iocb_cleanup (phba, pIocbIn, pIocbOut);
|
|
||||||
ml_cmd->host_scribble = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
const char *
|
const char *
|
||||||
lpfc_info(struct Scsi_Host *host)
|
lpfc_info(struct Scsi_Host *host)
|
||||||
{
|
{
|
||||||
|
@ -726,43 +701,25 @@ lpfc_queuecommand(struct scsi_cmnd *cmnd, void (*done) (struct scsi_cmnd *))
|
||||||
struct lpfc_sli *psli = &phba->sli;
|
struct lpfc_sli *psli = &phba->sli;
|
||||||
struct lpfc_rport_data *rdata = cmnd->device->hostdata;
|
struct lpfc_rport_data *rdata = cmnd->device->hostdata;
|
||||||
struct lpfc_nodelist *ndlp = rdata->pnode;
|
struct lpfc_nodelist *ndlp = rdata->pnode;
|
||||||
struct lpfc_scsi_buf *lpfc_cmd = NULL;
|
struct lpfc_scsi_buf *lpfc_cmd;
|
||||||
struct list_head *scsi_buf_list = &phba->lpfc_scsi_buf_list;
|
struct fc_rport *rport = starget_to_rport(scsi_target(cmnd->device));
|
||||||
int err = 0;
|
int err;
|
||||||
|
|
||||||
/*
|
err = fc_remote_port_chkready(rport);
|
||||||
* The target pointer is guaranteed not to be NULL because the driver
|
if (err) {
|
||||||
* only clears the device->hostdata field in lpfc_slave_destroy. This
|
cmnd->result = err;
|
||||||
* approach guarantees no further IO calls on this target.
|
|
||||||
*/
|
|
||||||
if (!ndlp) {
|
|
||||||
cmnd->result = ScsiResult(DID_NO_CONNECT, 0);
|
|
||||||
goto out_fail_command;
|
goto out_fail_command;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* A Fibre Channel target is present and functioning only when the node
|
* Catch race where our node has transitioned, but the
|
||||||
* state is MAPPED. Any other state is a failure.
|
* transport is still transitioning.
|
||||||
*/
|
*/
|
||||||
if (ndlp->nlp_state != NLP_STE_MAPPED_NODE) {
|
if (!ndlp) {
|
||||||
if ((ndlp->nlp_state == NLP_STE_UNMAPPED_NODE) ||
|
cmnd->result = ScsiResult(DID_BUS_BUSY, 0);
|
||||||
(ndlp->nlp_state == NLP_STE_UNUSED_NODE)) {
|
goto out_fail_command;
|
||||||
cmnd->result = ScsiResult(DID_NO_CONNECT, 0);
|
|
||||||
goto out_fail_command;
|
|
||||||
}
|
|
||||||
else if (ndlp->nlp_state == NLP_STE_NPR_NODE) {
|
|
||||||
cmnd->result = ScsiResult(DID_BUS_BUSY, 0);
|
|
||||||
goto out_fail_command;
|
|
||||||
}
|
|
||||||
/*
|
|
||||||
* The device is most likely recovered and the driver
|
|
||||||
* needs a bit more time to finish. Ask the midlayer
|
|
||||||
* to retry.
|
|
||||||
*/
|
|
||||||
goto out_host_busy;
|
|
||||||
}
|
}
|
||||||
|
lpfc_cmd = lpfc_sli_get_scsi_buf (phba);
|
||||||
list_remove_head(scsi_buf_list, lpfc_cmd, struct lpfc_scsi_buf, list);
|
|
||||||
if (lpfc_cmd == NULL) {
|
if (lpfc_cmd == NULL) {
|
||||||
printk(KERN_WARNING "%s: No buffer available - list empty, "
|
printk(KERN_WARNING "%s: No buffer available - list empty, "
|
||||||
"total count %d\n", __FUNCTION__, phba->total_scsi_bufs);
|
"total count %d\n", __FUNCTION__, phba->total_scsi_bufs);
|
||||||
|
@ -792,7 +749,7 @@ lpfc_queuecommand(struct scsi_cmnd *cmnd, void (*done) (struct scsi_cmnd *))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
out_host_busy_free_buf:
|
out_host_busy_free_buf:
|
||||||
lpfc_free_scsi_buf(lpfc_cmd);
|
lpfc_release_scsi_buf(phba, lpfc_cmd);
|
||||||
cmnd->host_scribble = NULL;
|
cmnd->host_scribble = NULL;
|
||||||
out_host_busy:
|
out_host_busy:
|
||||||
return SCSI_MLQUEUE_HOST_BUSY;
|
return SCSI_MLQUEUE_HOST_BUSY;
|
||||||
|
@ -808,119 +765,92 @@ __lpfc_abort_handler(struct scsi_cmnd *cmnd)
|
||||||
struct lpfc_hba *phba =
|
struct lpfc_hba *phba =
|
||||||
(struct lpfc_hba *)cmnd->device->host->hostdata[0];
|
(struct lpfc_hba *)cmnd->device->host->hostdata[0];
|
||||||
struct lpfc_sli_ring *pring = &phba->sli.ring[phba->sli.fcp_ring];
|
struct lpfc_sli_ring *pring = &phba->sli.ring[phba->sli.fcp_ring];
|
||||||
struct lpfc_iocbq *iocb, *next_iocb;
|
struct lpfc_iocbq *iocb;
|
||||||
struct lpfc_iocbq *abtsiocb = NULL;
|
struct lpfc_iocbq *abtsiocb;
|
||||||
struct lpfc_scsi_buf *lpfc_cmd;
|
struct lpfc_scsi_buf *lpfc_cmd;
|
||||||
struct list_head *lpfc_iocb_list = &phba->lpfc_iocb_list;
|
|
||||||
IOCB_t *cmd, *icmd;
|
IOCB_t *cmd, *icmd;
|
||||||
unsigned long snum;
|
|
||||||
unsigned int id, lun;
|
|
||||||
unsigned int loop_count = 0;
|
unsigned int loop_count = 0;
|
||||||
int ret = IOCB_SUCCESS;
|
int ret = SUCCESS;
|
||||||
|
|
||||||
|
|
||||||
|
lpfc_cmd = (struct lpfc_scsi_buf *)cmnd->host_scribble;
|
||||||
|
BUG_ON(!lpfc_cmd);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the host_scribble data area is NULL, then the driver has already
|
* If pCmd field of the corresponding lpfc_scsi_buf structure
|
||||||
* completed this command, but the midlayer did not see the completion
|
* points to a different SCSI command, then the driver has
|
||||||
* before the eh fired. Just return SUCCESS.
|
* already completed this command, but the midlayer did not
|
||||||
|
* see the completion before the eh fired. Just return
|
||||||
|
* SUCCESS.
|
||||||
*/
|
*/
|
||||||
lpfc_cmd = (struct lpfc_scsi_buf *)cmnd->host_scribble;
|
iocb = &lpfc_cmd->cur_iocbq;
|
||||||
if (!lpfc_cmd)
|
if (lpfc_cmd->pCmd != cmnd)
|
||||||
return SUCCESS;
|
goto out;
|
||||||
|
|
||||||
/* save these now since lpfc_cmd can be freed */
|
BUG_ON(iocb->context1 != lpfc_cmd);
|
||||||
id = lpfc_cmd->pCmd->device->id;
|
|
||||||
lun = lpfc_cmd->pCmd->device->lun;
|
|
||||||
snum = lpfc_cmd->pCmd->serial_number;
|
|
||||||
|
|
||||||
list_for_each_entry_safe(iocb, next_iocb, &pring->txq, list) {
|
|
||||||
cmd = &iocb->iocb;
|
|
||||||
if (iocb->context1 != lpfc_cmd)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
list_del_init(&iocb->list);
|
|
||||||
pring->txq_cnt--;
|
|
||||||
if (!iocb->iocb_cmpl) {
|
|
||||||
list_add_tail(&iocb->list, lpfc_iocb_list);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
cmd->ulpStatus = IOSTAT_LOCAL_REJECT;
|
|
||||||
cmd->un.ulpWord[4] = IOERR_SLI_ABORTED;
|
|
||||||
lpfc_scsi_cmd_iocb_cmpl_aborted(phba, iocb, iocb);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
abtsiocb = lpfc_sli_get_iocbq(phba);
|
||||||
|
if (abtsiocb == NULL) {
|
||||||
|
ret = FAILED;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
list_remove_head(lpfc_iocb_list, abtsiocb, struct lpfc_iocbq, list);
|
|
||||||
if (abtsiocb == NULL)
|
|
||||||
return FAILED;
|
|
||||||
|
|
||||||
memset(abtsiocb, 0, sizeof (struct lpfc_iocbq));
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The scsi command was not in the txq. Check the txcmplq and if it is
|
* The scsi command can not be in txq and it is in flight because the
|
||||||
* found, send an abort to the FW.
|
* pCmd is still pointig at the SCSI command we have to abort. There
|
||||||
|
* is no need to search the txcmplq. Just send an abort to the FW.
|
||||||
*/
|
*/
|
||||||
list_for_each_entry_safe(iocb, next_iocb, &pring->txcmplq, list) {
|
|
||||||
if (iocb->context1 != lpfc_cmd)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
iocb->iocb_cmpl = lpfc_scsi_cmd_iocb_cmpl_aborted;
|
cmd = &iocb->iocb;
|
||||||
cmd = &iocb->iocb;
|
icmd = &abtsiocb->iocb;
|
||||||
icmd = &abtsiocb->iocb;
|
icmd->un.acxri.abortType = ABORT_TYPE_ABTS;
|
||||||
icmd->un.acxri.abortType = ABORT_TYPE_ABTS;
|
icmd->un.acxri.abortContextTag = cmd->ulpContext;
|
||||||
icmd->un.acxri.abortContextTag = cmd->ulpContext;
|
icmd->un.acxri.abortIoTag = cmd->ulpIoTag;
|
||||||
icmd->un.acxri.abortIoTag = cmd->ulpIoTag;
|
|
||||||
|
|
||||||
icmd->ulpLe = 1;
|
icmd->ulpLe = 1;
|
||||||
icmd->ulpClass = cmd->ulpClass;
|
icmd->ulpClass = cmd->ulpClass;
|
||||||
if (phba->hba_state >= LPFC_LINK_UP)
|
if (phba->hba_state >= LPFC_LINK_UP)
|
||||||
icmd->ulpCommand = CMD_ABORT_XRI_CN;
|
icmd->ulpCommand = CMD_ABORT_XRI_CN;
|
||||||
else
|
else
|
||||||
icmd->ulpCommand = CMD_CLOSE_XRI_CN;
|
icmd->ulpCommand = CMD_CLOSE_XRI_CN;
|
||||||
|
|
||||||
abtsiocb->iocb_cmpl = lpfc_sli_abort_fcp_cmpl;
|
abtsiocb->iocb_cmpl = lpfc_sli_abort_fcp_cmpl;
|
||||||
if (lpfc_sli_issue_iocb(phba, pring, abtsiocb, 0) ==
|
if (lpfc_sli_issue_iocb(phba, pring, abtsiocb, 0) == IOCB_ERROR) {
|
||||||
IOCB_ERROR) {
|
lpfc_sli_release_iocbq(phba, abtsiocb);
|
||||||
list_add_tail(&abtsiocb->list, lpfc_iocb_list);
|
ret = FAILED;
|
||||||
ret = IOCB_ERROR;
|
goto out;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Wait for abort to complete */
|
||||||
|
while (lpfc_cmd->pCmd == cmnd)
|
||||||
|
{
|
||||||
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
|
set_current_state(TASK_UNINTERRUPTIBLE);
|
||||||
|
schedule_timeout(LPFC_ABORT_WAIT*HZ);
|
||||||
|
spin_lock_irq(phba->host->host_lock);
|
||||||
|
if (++loop_count
|
||||||
|
> (2 * phba->cfg_nodev_tmo)/LPFC_ABORT_WAIT)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Wait for abort to complete */
|
if (lpfc_cmd->pCmd == cmnd) {
|
||||||
while (cmnd->host_scribble)
|
ret = FAILED;
|
||||||
{
|
lpfc_printf_log(phba, KERN_ERR, LOG_FCP,
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
"%d:0748 abort handler timed out waiting for "
|
||||||
set_current_state(TASK_UNINTERRUPTIBLE);
|
"abort to complete: ret %#x, ID %d, LUN %d, "
|
||||||
schedule_timeout(LPFC_ABORT_WAIT*HZ);
|
"snum %#lx\n",
|
||||||
spin_lock_irq(phba->host->host_lock);
|
phba->brd_no, ret, cmnd->device->id,
|
||||||
if (++loop_count
|
cmnd->device->lun, cmnd->serial_number);
|
||||||
> (2 * phba->cfg_nodev_tmo)/LPFC_ABORT_WAIT)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if(cmnd->host_scribble) {
|
|
||||||
lpfc_printf_log(phba, KERN_ERR, LOG_FCP,
|
|
||||||
"%d:0748 abort handler timed "
|
|
||||||
"out waiting for abort to "
|
|
||||||
"complete. Data: "
|
|
||||||
"x%x x%x x%x x%lx\n",
|
|
||||||
phba->brd_no, ret, id, lun, snum);
|
|
||||||
cmnd->host_scribble = NULL;
|
|
||||||
iocb->iocb_cmpl = lpfc_scsi_cmd_iocb_cleanup;
|
|
||||||
ret = IOCB_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
out:
|
out:
|
||||||
lpfc_printf_log(phba, KERN_WARNING, LOG_FCP,
|
lpfc_printf_log(phba, KERN_WARNING, LOG_FCP,
|
||||||
"%d:0749 SCSI layer issued abort device "
|
"%d:0749 SCSI layer issued abort device: ret %#x, "
|
||||||
"Data: x%x x%x x%x x%lx\n",
|
"ID %d, LUN %d, snum %#lx\n",
|
||||||
phba->brd_no, ret, id, lun, snum);
|
phba->brd_no, ret, cmnd->device->id,
|
||||||
|
cmnd->device->lun, cmnd->serial_number);
|
||||||
|
|
||||||
return ret == IOCB_SUCCESS ? SUCCESS : FAILED;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
|
@ -938,11 +868,8 @@ __lpfc_reset_lun_handler(struct scsi_cmnd *cmnd)
|
||||||
{
|
{
|
||||||
struct Scsi_Host *shost = cmnd->device->host;
|
struct Scsi_Host *shost = cmnd->device->host;
|
||||||
struct lpfc_hba *phba = (struct lpfc_hba *)shost->hostdata[0];
|
struct lpfc_hba *phba = (struct lpfc_hba *)shost->hostdata[0];
|
||||||
struct lpfc_sli *psli = &phba->sli;
|
struct lpfc_scsi_buf *lpfc_cmd;
|
||||||
struct lpfc_scsi_buf *lpfc_cmd = NULL;
|
struct lpfc_iocbq *iocbq, *iocbqrsp;
|
||||||
struct list_head *scsi_buf_list = &phba->lpfc_scsi_buf_list;
|
|
||||||
struct list_head *lpfc_iocb_list = &phba->lpfc_iocb_list;
|
|
||||||
struct lpfc_iocbq *iocbq, *iocbqrsp = NULL;
|
|
||||||
struct lpfc_rport_data *rdata = cmnd->device->hostdata;
|
struct lpfc_rport_data *rdata = cmnd->device->hostdata;
|
||||||
struct lpfc_nodelist *pnode = rdata->pnode;
|
struct lpfc_nodelist *pnode = rdata->pnode;
|
||||||
int ret = FAILED;
|
int ret = FAILED;
|
||||||
|
@ -966,7 +893,7 @@ __lpfc_reset_lun_handler(struct scsi_cmnd *cmnd)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
list_remove_head(scsi_buf_list, lpfc_cmd, struct lpfc_scsi_buf, list);
|
lpfc_cmd = lpfc_sli_get_scsi_buf (phba);
|
||||||
if (lpfc_cmd == NULL)
|
if (lpfc_cmd == NULL)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
@ -981,18 +908,13 @@ __lpfc_reset_lun_handler(struct scsi_cmnd *cmnd)
|
||||||
iocbq = &lpfc_cmd->cur_iocbq;
|
iocbq = &lpfc_cmd->cur_iocbq;
|
||||||
|
|
||||||
/* get a buffer for this IOCB command response */
|
/* get a buffer for this IOCB command response */
|
||||||
list_remove_head(lpfc_iocb_list, iocbqrsp, struct lpfc_iocbq, list);
|
iocbqrsp = lpfc_sli_get_iocbq(phba);
|
||||||
if (iocbqrsp == NULL)
|
if (iocbqrsp == NULL)
|
||||||
goto out_free_scsi_buf;
|
goto out_free_scsi_buf;
|
||||||
|
|
||||||
memset(iocbqrsp, 0, sizeof (struct lpfc_iocbq));
|
ret = lpfc_sli_issue_iocb_wait(phba,
|
||||||
|
&phba->sli.ring[phba->sli.fcp_ring],
|
||||||
iocbq->iocb_flag |= LPFC_IO_POLL;
|
iocbq, iocbqrsp, lpfc_cmd->timeout);
|
||||||
iocbq->iocb_cmpl = lpfc_sli_wake_iocb_high_priority;
|
|
||||||
|
|
||||||
ret = lpfc_sli_issue_iocb_wait_high_priority(phba,
|
|
||||||
&phba->sli.ring[psli->fcp_ring],
|
|
||||||
iocbq, 0, iocbqrsp, 60);
|
|
||||||
if (ret == IOCB_SUCCESS)
|
if (ret == IOCB_SUCCESS)
|
||||||
ret = SUCCESS;
|
ret = SUCCESS;
|
||||||
|
|
||||||
|
@ -1027,12 +949,13 @@ __lpfc_reset_lun_handler(struct scsi_cmnd *cmnd)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cnt) {
|
if (cnt) {
|
||||||
lpfc_printf_log(phba, KERN_INFO, LOG_FCP,
|
lpfc_printf_log(phba, KERN_ERR, LOG_FCP,
|
||||||
"%d:0719 LUN Reset I/O flush failure: cnt x%x\n",
|
"%d:0719 LUN Reset I/O flush failure: cnt x%x\n",
|
||||||
phba->brd_no, cnt);
|
phba->brd_no, cnt);
|
||||||
|
ret = FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
list_add_tail(&iocbqrsp->list, lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, iocbqrsp);
|
||||||
|
|
||||||
out_free_scsi_buf:
|
out_free_scsi_buf:
|
||||||
lpfc_printf_log(phba, KERN_ERR, LOG_FCP,
|
lpfc_printf_log(phba, KERN_ERR, LOG_FCP,
|
||||||
|
@ -1041,7 +964,7 @@ out_free_scsi_buf:
|
||||||
phba->brd_no, lpfc_cmd->pCmd->device->id,
|
phba->brd_no, lpfc_cmd->pCmd->device->id,
|
||||||
lpfc_cmd->pCmd->device->lun, ret, lpfc_cmd->status,
|
lpfc_cmd->pCmd->device->lun, ret, lpfc_cmd->status,
|
||||||
lpfc_cmd->result);
|
lpfc_cmd->result);
|
||||||
lpfc_free_scsi_buf(lpfc_cmd);
|
lpfc_release_scsi_buf(phba, lpfc_cmd);
|
||||||
out:
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@ -1069,10 +992,9 @@ __lpfc_reset_bus_handler(struct scsi_cmnd *cmnd)
|
||||||
int ret = FAILED, i, err_count = 0;
|
int ret = FAILED, i, err_count = 0;
|
||||||
int cnt, loopcnt;
|
int cnt, loopcnt;
|
||||||
unsigned int midlayer_id = 0;
|
unsigned int midlayer_id = 0;
|
||||||
struct lpfc_scsi_buf * lpfc_cmd = NULL;
|
struct lpfc_scsi_buf * lpfc_cmd;
|
||||||
struct list_head *scsi_buf_list = &phba->lpfc_scsi_buf_list;
|
|
||||||
|
|
||||||
list_remove_head(scsi_buf_list, lpfc_cmd, struct lpfc_scsi_buf, list);
|
lpfc_cmd = lpfc_sli_get_scsi_buf (phba);
|
||||||
if (lpfc_cmd == NULL)
|
if (lpfc_cmd == NULL)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
@ -1136,10 +1058,12 @@ __lpfc_reset_bus_handler(struct scsi_cmnd *cmnd)
|
||||||
phba->brd_no, cnt, i);
|
phba->brd_no, cnt, i);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!err_count)
|
if (cnt == 0)
|
||||||
ret = SUCCESS;
|
ret = SUCCESS;
|
||||||
|
else
|
||||||
|
ret = FAILED;
|
||||||
|
|
||||||
lpfc_free_scsi_buf(lpfc_cmd);
|
lpfc_release_scsi_buf(phba, lpfc_cmd);
|
||||||
lpfc_printf_log(phba,
|
lpfc_printf_log(phba,
|
||||||
KERN_ERR,
|
KERN_ERR,
|
||||||
LOG_FCP,
|
LOG_FCP,
|
||||||
|
@ -1163,66 +1087,47 @@ static int
|
||||||
lpfc_slave_alloc(struct scsi_device *sdev)
|
lpfc_slave_alloc(struct scsi_device *sdev)
|
||||||
{
|
{
|
||||||
struct lpfc_hba *phba = (struct lpfc_hba *)sdev->host->hostdata[0];
|
struct lpfc_hba *phba = (struct lpfc_hba *)sdev->host->hostdata[0];
|
||||||
struct lpfc_nodelist *ndlp = NULL;
|
|
||||||
int match = 0;
|
|
||||||
struct lpfc_scsi_buf *scsi_buf = NULL;
|
struct lpfc_scsi_buf *scsi_buf = NULL;
|
||||||
|
struct fc_rport *rport = starget_to_rport(scsi_target(sdev));
|
||||||
uint32_t total = 0, i;
|
uint32_t total = 0, i;
|
||||||
uint32_t num_to_alloc = 0;
|
uint32_t num_to_alloc = 0;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
struct list_head *listp;
|
|
||||||
struct list_head *node_list[6];
|
|
||||||
|
|
||||||
/*
|
if (!rport || fc_remote_port_chkready(rport))
|
||||||
* Store the target pointer in the scsi_device hostdata pointer provided
|
|
||||||
* the driver has already discovered the target id.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/* Search the nlp lists other than unmap_list for this target ID */
|
|
||||||
node_list[0] = &phba->fc_npr_list;
|
|
||||||
node_list[1] = &phba->fc_nlpmap_list;
|
|
||||||
node_list[2] = &phba->fc_prli_list;
|
|
||||||
node_list[3] = &phba->fc_reglogin_list;
|
|
||||||
node_list[4] = &phba->fc_adisc_list;
|
|
||||||
node_list[5] = &phba->fc_plogi_list;
|
|
||||||
|
|
||||||
for (i = 0; i < 6 && !match; i++) {
|
|
||||||
listp = node_list[i];
|
|
||||||
if (list_empty(listp))
|
|
||||||
continue;
|
|
||||||
list_for_each_entry(ndlp, listp, nlp_listp) {
|
|
||||||
if ((sdev->id == ndlp->nlp_sid) && ndlp->rport) {
|
|
||||||
match = 1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!match)
|
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
|
||||||
sdev->hostdata = ndlp->rport->dd_data;
|
sdev->hostdata = rport->dd_data;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Populate the cmds_per_lun count scsi_bufs into this host's globally
|
* Populate the cmds_per_lun count scsi_bufs into this host's globally
|
||||||
* available list of scsi buffers. Don't allocate more than the
|
* available list of scsi buffers. Don't allocate more than the
|
||||||
* HBA limit conveyed to the midlayer via the host structure. Note
|
* HBA limit conveyed to the midlayer via the host structure. The
|
||||||
* that this list of scsi bufs exists for the lifetime of the driver.
|
* formula accounts for the lun_queue_depth + error handlers + 1
|
||||||
|
* extra. This list of scsi bufs exists for the lifetime of the driver.
|
||||||
*/
|
*/
|
||||||
total = phba->total_scsi_bufs;
|
total = phba->total_scsi_bufs;
|
||||||
num_to_alloc = LPFC_CMD_PER_LUN;
|
num_to_alloc = phba->cfg_lun_queue_depth + 2;
|
||||||
if (total >= phba->cfg_hba_queue_depth) {
|
if (total >= phba->cfg_hba_queue_depth) {
|
||||||
printk(KERN_WARNING "%s, At config limitation of "
|
lpfc_printf_log(phba, KERN_WARNING, LOG_FCP,
|
||||||
"%d allocated scsi_bufs\n", __FUNCTION__, total);
|
"%d:0704 At limitation of %d preallocated "
|
||||||
|
"command buffers\n", phba->brd_no, total);
|
||||||
return 0;
|
return 0;
|
||||||
} else if (total + num_to_alloc > phba->cfg_hba_queue_depth) {
|
} else if (total + num_to_alloc > phba->cfg_hba_queue_depth) {
|
||||||
|
lpfc_printf_log(phba, KERN_WARNING, LOG_FCP,
|
||||||
|
"%d:0705 Allocation request of %d command "
|
||||||
|
"buffers will exceed max of %d. Reducing "
|
||||||
|
"allocation request to %d.\n", phba->brd_no,
|
||||||
|
num_to_alloc, phba->cfg_hba_queue_depth,
|
||||||
|
(phba->cfg_hba_queue_depth - total));
|
||||||
num_to_alloc = phba->cfg_hba_queue_depth - total;
|
num_to_alloc = phba->cfg_hba_queue_depth - total;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < num_to_alloc; i++) {
|
for (i = 0; i < num_to_alloc; i++) {
|
||||||
scsi_buf = lpfc_get_scsi_buf(phba);
|
scsi_buf = lpfc_new_scsi_buf(phba);
|
||||||
if (!scsi_buf) {
|
if (!scsi_buf) {
|
||||||
printk(KERN_ERR "%s, failed to allocate "
|
lpfc_printf_log(phba, KERN_ERR, LOG_FCP,
|
||||||
"scsi_buf\n", __FUNCTION__);
|
"%d:0706 Failed to allocate command "
|
||||||
|
"buffer\n", phba->brd_no);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -65,6 +65,28 @@ typedef enum _lpfc_iocb_type {
|
||||||
LPFC_ABORT_IOCB
|
LPFC_ABORT_IOCB
|
||||||
} lpfc_iocb_type;
|
} lpfc_iocb_type;
|
||||||
|
|
||||||
|
struct lpfc_iocbq *
|
||||||
|
lpfc_sli_get_iocbq(struct lpfc_hba * phba)
|
||||||
|
{
|
||||||
|
struct list_head *lpfc_iocb_list = &phba->lpfc_iocb_list;
|
||||||
|
struct lpfc_iocbq * iocbq = NULL;
|
||||||
|
|
||||||
|
list_remove_head(lpfc_iocb_list, iocbq, struct lpfc_iocbq, list);
|
||||||
|
return iocbq;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
lpfc_sli_release_iocbq(struct lpfc_hba * phba, struct lpfc_iocbq * iocbq)
|
||||||
|
{
|
||||||
|
size_t start_clean = (size_t)(&((struct lpfc_iocbq *)NULL)->iocb);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Clean all volatile data fields, preserve iotag and node struct.
|
||||||
|
*/
|
||||||
|
memset((char*)iocbq + start_clean, 0, sizeof(*iocbq) - start_clean);
|
||||||
|
list_add_tail(&iocbq->list, &phba->lpfc_iocb_list);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Translate the iocb command to an iocb command type used to decide the final
|
* Translate the iocb command to an iocb command type used to decide the final
|
||||||
* disposition of each completed IOCB.
|
* disposition of each completed IOCB.
|
||||||
|
@ -265,41 +287,69 @@ lpfc_sli_next_iocb_slot (struct lpfc_hba *phba, struct lpfc_sli_ring *pring)
|
||||||
return iocb;
|
return iocb;
|
||||||
}
|
}
|
||||||
|
|
||||||
static uint32_t
|
uint16_t
|
||||||
lpfc_sli_next_iotag(struct lpfc_hba * phba, struct lpfc_sli_ring * pring)
|
lpfc_sli_next_iotag(struct lpfc_hba * phba, struct lpfc_iocbq * iocbq)
|
||||||
{
|
{
|
||||||
uint32_t search_start;
|
struct lpfc_iocbq ** new_arr;
|
||||||
|
struct lpfc_iocbq ** old_arr;
|
||||||
|
size_t new_len;
|
||||||
|
struct lpfc_sli *psli = &phba->sli;
|
||||||
|
uint16_t iotag;
|
||||||
|
|
||||||
if (pring->fast_lookup == NULL) {
|
spin_lock_irq(phba->host->host_lock);
|
||||||
pring->iotag_ctr++;
|
iotag = psli->last_iotag;
|
||||||
if (pring->iotag_ctr >= pring->iotag_max)
|
if(++iotag < psli->iocbq_lookup_len) {
|
||||||
pring->iotag_ctr = 1;
|
psli->last_iotag = iotag;
|
||||||
return pring->iotag_ctr;
|
psli->iocbq_lookup[iotag] = iocbq;
|
||||||
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
|
iocbq->iotag = iotag;
|
||||||
|
return iotag;
|
||||||
|
}
|
||||||
|
else if (psli->iocbq_lookup_len < (0xffff
|
||||||
|
- LPFC_IOCBQ_LOOKUP_INCREMENT)) {
|
||||||
|
new_len = psli->iocbq_lookup_len + LPFC_IOCBQ_LOOKUP_INCREMENT;
|
||||||
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
|
new_arr = kmalloc(new_len * sizeof (struct lpfc_iocbq *),
|
||||||
|
GFP_KERNEL);
|
||||||
|
if (new_arr) {
|
||||||
|
memset((char *)new_arr, 0,
|
||||||
|
new_len * sizeof (struct lpfc_iocbq *));
|
||||||
|
spin_lock_irq(phba->host->host_lock);
|
||||||
|
old_arr = psli->iocbq_lookup;
|
||||||
|
if (new_len <= psli->iocbq_lookup_len) {
|
||||||
|
/* highly unprobable case */
|
||||||
|
kfree(new_arr);
|
||||||
|
iotag = psli->last_iotag;
|
||||||
|
if(++iotag < psli->iocbq_lookup_len) {
|
||||||
|
psli->last_iotag = iotag;
|
||||||
|
psli->iocbq_lookup[iotag] = iocbq;
|
||||||
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
|
iocbq->iotag = iotag;
|
||||||
|
return iotag;
|
||||||
|
}
|
||||||
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (psli->iocbq_lookup)
|
||||||
|
memcpy(new_arr, old_arr,
|
||||||
|
((psli->last_iotag + 1) *
|
||||||
|
sizeof (struct lpfc_iocbq *)));
|
||||||
|
psli->iocbq_lookup = new_arr;
|
||||||
|
psli->iocbq_lookup_len = new_len;
|
||||||
|
psli->last_iotag = iotag;
|
||||||
|
psli->iocbq_lookup[iotag] = iocbq;
|
||||||
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
|
iocbq->iotag = iotag;
|
||||||
|
kfree(old_arr);
|
||||||
|
return iotag;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
search_start = pring->iotag_ctr;
|
lpfc_printf_log(phba, KERN_ERR,LOG_SLI,
|
||||||
|
"%d:0318 Failed to allocate IOTAG.last IOTAG is %d\n",
|
||||||
|
phba->brd_no, psli->last_iotag);
|
||||||
|
|
||||||
do {
|
return 0;
|
||||||
pring->iotag_ctr++;
|
|
||||||
if (pring->iotag_ctr >= pring->fast_iotag)
|
|
||||||
pring->iotag_ctr = 1;
|
|
||||||
|
|
||||||
if (*(pring->fast_lookup + pring->iotag_ctr) == NULL)
|
|
||||||
return pring->iotag_ctr;
|
|
||||||
|
|
||||||
} while (pring->iotag_ctr != search_start);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Outstanding I/O count for ring <ringno> is at max <fast_iotag>
|
|
||||||
*/
|
|
||||||
lpfc_printf_log(phba,
|
|
||||||
KERN_ERR,
|
|
||||||
LOG_SLI,
|
|
||||||
"%d:0318 Outstanding I/O count for ring %d is at max x%x\n",
|
|
||||||
phba->brd_no,
|
|
||||||
pring->ringno,
|
|
||||||
pring->fast_iotag);
|
|
||||||
return (0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -307,10 +357,9 @@ lpfc_sli_submit_iocb(struct lpfc_hba *phba, struct lpfc_sli_ring *pring,
|
||||||
IOCB_t *iocb, struct lpfc_iocbq *nextiocb)
|
IOCB_t *iocb, struct lpfc_iocbq *nextiocb)
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
* Allocate and set up an iotag
|
* Set up an iotag
|
||||||
*/
|
*/
|
||||||
nextiocb->iocb.ulpIoTag =
|
nextiocb->iocb.ulpIoTag = (nextiocb->iocb_cmpl) ? nextiocb->iotag : 0;
|
||||||
lpfc_sli_next_iotag(phba, &phba->sli.ring[phba->sli.fcp_ring]);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Issue iocb command to adapter
|
* Issue iocb command to adapter
|
||||||
|
@ -326,16 +375,15 @@ lpfc_sli_submit_iocb(struct lpfc_hba *phba, struct lpfc_sli_ring *pring,
|
||||||
*/
|
*/
|
||||||
if (nextiocb->iocb_cmpl)
|
if (nextiocb->iocb_cmpl)
|
||||||
lpfc_sli_ringtxcmpl_put(phba, pring, nextiocb);
|
lpfc_sli_ringtxcmpl_put(phba, pring, nextiocb);
|
||||||
else {
|
else
|
||||||
list_add_tail(&nextiocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, nextiocb);
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Let the HBA know what IOCB slot will be the next one the
|
* Let the HBA know what IOCB slot will be the next one the
|
||||||
* driver will put a command into.
|
* driver will put a command into.
|
||||||
*/
|
*/
|
||||||
pring->cmdidx = pring->next_cmdidx;
|
pring->cmdidx = pring->next_cmdidx;
|
||||||
writeb(pring->cmdidx, phba->MBslimaddr
|
writel(pring->cmdidx, phba->MBslimaddr
|
||||||
+ (SLIMOFF + (pring->ringno * 2)) * 4);
|
+ (SLIMOFF + (pring->ringno * 2)) * 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -752,80 +800,28 @@ lpfc_sli_process_unsol_iocb(struct lpfc_hba *phba, struct lpfc_sli_ring *pring,
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct lpfc_iocbq *
|
static struct lpfc_iocbq *
|
||||||
lpfc_sli_txcmpl_ring_search_slow(struct lpfc_sli_ring * pring,
|
lpfc_sli_iocbq_lookup(struct lpfc_hba * phba,
|
||||||
struct lpfc_iocbq * prspiocb)
|
struct lpfc_sli_ring * pring,
|
||||||
|
struct lpfc_iocbq * prspiocb)
|
||||||
{
|
{
|
||||||
IOCB_t *icmd = NULL;
|
|
||||||
IOCB_t *irsp = NULL;
|
|
||||||
struct lpfc_iocbq *cmd_iocb;
|
|
||||||
struct lpfc_iocbq *iocb, *next_iocb;
|
|
||||||
uint16_t iotag;
|
|
||||||
|
|
||||||
irsp = &prspiocb->iocb;
|
|
||||||
iotag = irsp->ulpIoTag;
|
|
||||||
cmd_iocb = NULL;
|
|
||||||
|
|
||||||
/* Search through txcmpl from the begining */
|
|
||||||
list_for_each_entry_safe(iocb, next_iocb, &(pring->txcmplq), list) {
|
|
||||||
icmd = &iocb->iocb;
|
|
||||||
if (iotag == icmd->ulpIoTag) {
|
|
||||||
/* Found a match. */
|
|
||||||
cmd_iocb = iocb;
|
|
||||||
list_del(&iocb->list);
|
|
||||||
pring->txcmplq_cnt--;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return (cmd_iocb);
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct lpfc_iocbq *
|
|
||||||
lpfc_sli_txcmpl_ring_iotag_lookup(struct lpfc_hba * phba,
|
|
||||||
struct lpfc_sli_ring * pring,
|
|
||||||
struct lpfc_iocbq * prspiocb)
|
|
||||||
{
|
|
||||||
IOCB_t *irsp = NULL;
|
|
||||||
struct lpfc_iocbq *cmd_iocb = NULL;
|
struct lpfc_iocbq *cmd_iocb = NULL;
|
||||||
uint16_t iotag;
|
uint16_t iotag;
|
||||||
|
|
||||||
if (unlikely(pring->fast_lookup == NULL))
|
iotag = prspiocb->iocb.ulpIoTag;
|
||||||
return NULL;
|
|
||||||
|
|
||||||
/* Use fast lookup based on iotag for completion */
|
if (iotag != 0 && iotag <= phba->sli.last_iotag) {
|
||||||
irsp = &prspiocb->iocb;
|
cmd_iocb = phba->sli.iocbq_lookup[iotag];
|
||||||
iotag = irsp->ulpIoTag;
|
list_del(&cmd_iocb->list);
|
||||||
if (iotag < pring->fast_iotag) {
|
pring->txcmplq_cnt--;
|
||||||
cmd_iocb = *(pring->fast_lookup + iotag);
|
return cmd_iocb;
|
||||||
*(pring->fast_lookup + iotag) = NULL;
|
|
||||||
if (cmd_iocb) {
|
|
||||||
list_del(&cmd_iocb->list);
|
|
||||||
pring->txcmplq_cnt--;
|
|
||||||
return cmd_iocb;
|
|
||||||
} else {
|
|
||||||
/*
|
|
||||||
* This is clearly an error. A ring that uses iotags
|
|
||||||
* should never have a interrupt for a completion that
|
|
||||||
* is not on the ring. Return NULL and log a error.
|
|
||||||
*/
|
|
||||||
lpfc_printf_log(phba, KERN_ERR, LOG_SLI,
|
|
||||||
"%d:0327 Rsp ring %d error - command "
|
|
||||||
"completion for iotag x%x not found\n",
|
|
||||||
phba->brd_no, pring->ringno, iotag);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Rsp ring <ringno> get: iotag <iotag> greater then
|
|
||||||
* configured max <fast_iotag> wd0 <irsp>. This is an
|
|
||||||
* error. Just return NULL.
|
|
||||||
*/
|
|
||||||
lpfc_printf_log(phba, KERN_ERR, LOG_SLI,
|
lpfc_printf_log(phba, KERN_ERR, LOG_SLI,
|
||||||
"%d:0317 Rsp ring %d get: iotag x%x greater then "
|
"%d:0317 iotag x%x is out off "
|
||||||
"configured max x%x wd0 x%x\n",
|
"range: max iotag x%x wd0 x%x\n",
|
||||||
phba->brd_no, pring->ringno, iotag, pring->fast_iotag,
|
phba->brd_no, iotag,
|
||||||
*(((uint32_t *) irsp) + 7));
|
phba->sli.last_iotag,
|
||||||
|
*(((uint32_t *) &prspiocb->iocb) + 7));
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -839,7 +835,7 @@ lpfc_sli_process_sol_iocb(struct lpfc_hba * phba, struct lpfc_sli_ring * pring,
|
||||||
|
|
||||||
/* Based on the iotag field, get the cmd IOCB from the txcmplq */
|
/* Based on the iotag field, get the cmd IOCB from the txcmplq */
|
||||||
spin_lock_irqsave(phba->host->host_lock, iflag);
|
spin_lock_irqsave(phba->host->host_lock, iflag);
|
||||||
cmdiocbp = lpfc_sli_txcmpl_ring_search_slow(pring, saveq);
|
cmdiocbp = lpfc_sli_iocbq_lookup(phba, pring, saveq);
|
||||||
if (cmdiocbp) {
|
if (cmdiocbp) {
|
||||||
if (cmdiocbp->iocb_cmpl) {
|
if (cmdiocbp->iocb_cmpl) {
|
||||||
/*
|
/*
|
||||||
|
@ -853,17 +849,13 @@ lpfc_sli_process_sol_iocb(struct lpfc_hba * phba, struct lpfc_sli_ring * pring,
|
||||||
spin_lock_irqsave(phba->host->host_lock, iflag);
|
spin_lock_irqsave(phba->host->host_lock, iflag);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (cmdiocbp->iocb_flag & LPFC_IO_POLL)
|
|
||||||
rc = 0;
|
|
||||||
|
|
||||||
spin_unlock_irqrestore(phba->host->host_lock,
|
spin_unlock_irqrestore(phba->host->host_lock,
|
||||||
iflag);
|
iflag);
|
||||||
(cmdiocbp->iocb_cmpl) (phba, cmdiocbp, saveq);
|
(cmdiocbp->iocb_cmpl) (phba, cmdiocbp, saveq);
|
||||||
spin_lock_irqsave(phba->host->host_lock, iflag);
|
spin_lock_irqsave(phba->host->host_lock, iflag);
|
||||||
}
|
}
|
||||||
} else {
|
} else
|
||||||
list_add_tail(&cmdiocbp->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, cmdiocbp);
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
/*
|
/*
|
||||||
* Unknown initiating command based on the response iotag.
|
* Unknown initiating command based on the response iotag.
|
||||||
|
@ -889,6 +881,7 @@ lpfc_sli_process_sol_iocb(struct lpfc_hba * phba, struct lpfc_sli_ring * pring,
|
||||||
saveq->iocb.ulpContext);
|
saveq->iocb.ulpContext);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_unlock_irqrestore(phba->host->host_lock, iflag);
|
spin_unlock_irqrestore(phba->host->host_lock, iflag);
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
@ -953,7 +946,7 @@ lpfc_sli_handle_fast_ring_event(struct lpfc_hba * phba,
|
||||||
* structure. The copy involves a byte-swap since the
|
* structure. The copy involves a byte-swap since the
|
||||||
* network byte order and pci byte orders are different.
|
* network byte order and pci byte orders are different.
|
||||||
*/
|
*/
|
||||||
entry = (IOCB_t *) IOCB_ENTRY(pring->rspringaddr, pring->rspidx);
|
entry = IOCB_ENTRY(pring->rspringaddr, pring->rspidx);
|
||||||
lpfc_sli_pcimem_bcopy((uint32_t *) entry,
|
lpfc_sli_pcimem_bcopy((uint32_t *) entry,
|
||||||
(uint32_t *) &rspiocbq.iocb,
|
(uint32_t *) &rspiocbq.iocb,
|
||||||
sizeof (IOCB_t));
|
sizeof (IOCB_t));
|
||||||
|
@ -990,9 +983,8 @@ lpfc_sli_handle_fast_ring_event(struct lpfc_hba * phba,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
cmdiocbq = lpfc_sli_txcmpl_ring_iotag_lookup(phba,
|
cmdiocbq = lpfc_sli_iocbq_lookup(phba, pring,
|
||||||
pring,
|
&rspiocbq);
|
||||||
&rspiocbq);
|
|
||||||
if ((cmdiocbq) && (cmdiocbq->iocb_cmpl)) {
|
if ((cmdiocbq) && (cmdiocbq->iocb_cmpl)) {
|
||||||
spin_unlock_irqrestore(
|
spin_unlock_irqrestore(
|
||||||
phba->host->host_lock, iflag);
|
phba->host->host_lock, iflag);
|
||||||
|
@ -1033,7 +1025,7 @@ lpfc_sli_handle_fast_ring_event(struct lpfc_hba * phba,
|
||||||
|
|
||||||
to_slim = phba->MBslimaddr +
|
to_slim = phba->MBslimaddr +
|
||||||
(SLIMOFF + (pring->ringno * 2) + 1) * 4;
|
(SLIMOFF + (pring->ringno * 2) + 1) * 4;
|
||||||
writeb(pring->rspidx, to_slim);
|
writel(pring->rspidx, to_slim);
|
||||||
|
|
||||||
if (pring->rspidx == portRspPut)
|
if (pring->rspidx == portRspPut)
|
||||||
portRspPut = le32_to_cpu(pgp->rspPutInx);
|
portRspPut = le32_to_cpu(pgp->rspPutInx);
|
||||||
|
@ -1073,7 +1065,6 @@ lpfc_sli_handle_slow_ring_event(struct lpfc_hba * phba,
|
||||||
struct lpfc_iocbq *next_iocb;
|
struct lpfc_iocbq *next_iocb;
|
||||||
struct lpfc_iocbq *cmdiocbp;
|
struct lpfc_iocbq *cmdiocbp;
|
||||||
struct lpfc_iocbq *saveq;
|
struct lpfc_iocbq *saveq;
|
||||||
struct list_head *lpfc_iocb_list = &phba->lpfc_iocb_list;
|
|
||||||
struct lpfc_pgp *pgp = &phba->slim2p->mbx.us.s2.port[pring->ringno];
|
struct lpfc_pgp *pgp = &phba->slim2p->mbx.us.s2.port[pring->ringno];
|
||||||
uint8_t iocb_cmd_type;
|
uint8_t iocb_cmd_type;
|
||||||
lpfc_iocb_type type;
|
lpfc_iocb_type type;
|
||||||
|
@ -1115,7 +1106,6 @@ lpfc_sli_handle_slow_ring_event(struct lpfc_hba * phba,
|
||||||
}
|
}
|
||||||
|
|
||||||
rmb();
|
rmb();
|
||||||
lpfc_iocb_list = &phba->lpfc_iocb_list;
|
|
||||||
while (pring->rspidx != portRspPut) {
|
while (pring->rspidx != portRspPut) {
|
||||||
/*
|
/*
|
||||||
* Build a completion list and call the appropriate handler.
|
* Build a completion list and call the appropriate handler.
|
||||||
|
@ -1131,8 +1121,7 @@ lpfc_sli_handle_slow_ring_event(struct lpfc_hba * phba,
|
||||||
* received.
|
* received.
|
||||||
*/
|
*/
|
||||||
entry = IOCB_ENTRY(pring->rspringaddr, pring->rspidx);
|
entry = IOCB_ENTRY(pring->rspringaddr, pring->rspidx);
|
||||||
list_remove_head(lpfc_iocb_list, rspiocbp, struct lpfc_iocbq,
|
rspiocbp = lpfc_sli_get_iocbq(phba);
|
||||||
list);
|
|
||||||
if (rspiocbp == NULL) {
|
if (rspiocbp == NULL) {
|
||||||
printk(KERN_ERR "%s: out of buffers! Failing "
|
printk(KERN_ERR "%s: out of buffers! Failing "
|
||||||
"completion.\n", __FUNCTION__);
|
"completion.\n", __FUNCTION__);
|
||||||
|
@ -1147,7 +1136,7 @@ lpfc_sli_handle_slow_ring_event(struct lpfc_hba * phba,
|
||||||
|
|
||||||
to_slim = phba->MBslimaddr + (SLIMOFF + (pring->ringno * 2)
|
to_slim = phba->MBslimaddr + (SLIMOFF + (pring->ringno * 2)
|
||||||
+ 1) * 4;
|
+ 1) * 4;
|
||||||
writeb(pring->rspidx, to_slim);
|
writel(pring->rspidx, to_slim);
|
||||||
|
|
||||||
if (list_empty(&(pring->iocb_continueq))) {
|
if (list_empty(&(pring->iocb_continueq))) {
|
||||||
list_add(&rspiocbp->list, &(pring->iocb_continueq));
|
list_add(&rspiocbp->list, &(pring->iocb_continueq));
|
||||||
|
@ -1213,8 +1202,8 @@ lpfc_sli_handle_slow_ring_event(struct lpfc_hba * phba,
|
||||||
} else if (type == LPFC_ABORT_IOCB) {
|
} else if (type == LPFC_ABORT_IOCB) {
|
||||||
if ((irsp->ulpCommand != CMD_XRI_ABORTED_CX) &&
|
if ((irsp->ulpCommand != CMD_XRI_ABORTED_CX) &&
|
||||||
((cmdiocbp =
|
((cmdiocbp =
|
||||||
lpfc_sli_txcmpl_ring_search_slow(pring,
|
lpfc_sli_iocbq_lookup(phba, pring,
|
||||||
saveq)))) {
|
saveq)))) {
|
||||||
/* Call the specified completion
|
/* Call the specified completion
|
||||||
routine */
|
routine */
|
||||||
if (cmdiocbp->iocb_cmpl) {
|
if (cmdiocbp->iocb_cmpl) {
|
||||||
|
@ -1226,10 +1215,9 @@ lpfc_sli_handle_slow_ring_event(struct lpfc_hba * phba,
|
||||||
spin_lock_irqsave(
|
spin_lock_irqsave(
|
||||||
phba->host->host_lock,
|
phba->host->host_lock,
|
||||||
iflag);
|
iflag);
|
||||||
} else {
|
} else
|
||||||
list_add_tail(&cmdiocbp->list,
|
lpfc_sli_release_iocbq(phba,
|
||||||
lpfc_iocb_list);
|
cmdiocbp);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else if (type == LPFC_UNKNOWN_IOCB) {
|
} else if (type == LPFC_UNKNOWN_IOCB) {
|
||||||
if (irsp->ulpCommand == CMD_ADAPTER_MSG) {
|
if (irsp->ulpCommand == CMD_ADAPTER_MSG) {
|
||||||
|
@ -1264,12 +1252,12 @@ lpfc_sli_handle_slow_ring_event(struct lpfc_hba * phba,
|
||||||
next_iocb,
|
next_iocb,
|
||||||
&saveq->list,
|
&saveq->list,
|
||||||
list) {
|
list) {
|
||||||
list_add_tail(&rspiocbp->list,
|
lpfc_sli_release_iocbq(phba,
|
||||||
lpfc_iocb_list);
|
rspiocbp);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
list_add_tail(&saveq->list, lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, saveq);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1314,7 +1302,6 @@ lpfc_sli_abort_iocb_ring(struct lpfc_hba *phba, struct lpfc_sli_ring *pring)
|
||||||
struct lpfc_iocbq *iocb, *next_iocb;
|
struct lpfc_iocbq *iocb, *next_iocb;
|
||||||
IOCB_t *icmd = NULL, *cmd = NULL;
|
IOCB_t *icmd = NULL, *cmd = NULL;
|
||||||
int errcnt;
|
int errcnt;
|
||||||
uint16_t iotag;
|
|
||||||
|
|
||||||
errcnt = 0;
|
errcnt = 0;
|
||||||
|
|
||||||
|
@ -1331,9 +1318,8 @@ lpfc_sli_abort_iocb_ring(struct lpfc_hba *phba, struct lpfc_sli_ring *pring)
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
} else {
|
} else
|
||||||
list_add_tail(&iocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, iocb);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
pring->txq_cnt = 0;
|
pring->txq_cnt = 0;
|
||||||
INIT_LIST_HEAD(&(pring->txq));
|
INIT_LIST_HEAD(&(pring->txq));
|
||||||
|
@ -1343,13 +1329,8 @@ lpfc_sli_abort_iocb_ring(struct lpfc_hba *phba, struct lpfc_sli_ring *pring)
|
||||||
cmd = &iocb->iocb;
|
cmd = &iocb->iocb;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Imediate abort of IOCB, clear fast_lookup entry,
|
* Imediate abort of IOCB, deque and call compl
|
||||||
* if any, deque and call compl
|
|
||||||
*/
|
*/
|
||||||
iotag = cmd->ulpIoTag;
|
|
||||||
if (iotag && pring->fast_lookup &&
|
|
||||||
(iotag < pring->fast_iotag))
|
|
||||||
pring->fast_lookup[iotag] = NULL;
|
|
||||||
|
|
||||||
list_del_init(&iocb->list);
|
list_del_init(&iocb->list);
|
||||||
pring->txcmplq_cnt--;
|
pring->txcmplq_cnt--;
|
||||||
|
@ -1360,9 +1341,8 @@ lpfc_sli_abort_iocb_ring(struct lpfc_hba *phba, struct lpfc_sli_ring *pring)
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
} else {
|
} else
|
||||||
list_add_tail(&iocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, iocb);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
INIT_LIST_HEAD(&pring->txcmplq);
|
INIT_LIST_HEAD(&pring->txcmplq);
|
||||||
|
@ -2147,6 +2127,10 @@ lpfc_sli_setup(struct lpfc_hba *phba)
|
||||||
psli->next_ring = LPFC_FCP_NEXT_RING;
|
psli->next_ring = LPFC_FCP_NEXT_RING;
|
||||||
psli->ip_ring = LPFC_IP_RING;
|
psli->ip_ring = LPFC_IP_RING;
|
||||||
|
|
||||||
|
psli->iocbq_lookup = NULL;
|
||||||
|
psli->iocbq_lookup_len = 0;
|
||||||
|
psli->last_iotag = 0;
|
||||||
|
|
||||||
for (i = 0; i < psli->num_rings; i++) {
|
for (i = 0; i < psli->num_rings; i++) {
|
||||||
pring = &psli->ring[i];
|
pring = &psli->ring[i];
|
||||||
switch (i) {
|
switch (i) {
|
||||||
|
@ -2222,7 +2206,7 @@ lpfc_sli_queue_setup(struct lpfc_hba * phba)
|
||||||
{
|
{
|
||||||
struct lpfc_sli *psli;
|
struct lpfc_sli *psli;
|
||||||
struct lpfc_sli_ring *pring;
|
struct lpfc_sli_ring *pring;
|
||||||
int i, cnt;
|
int i;
|
||||||
|
|
||||||
psli = &phba->sli;
|
psli = &phba->sli;
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
|
@ -2238,19 +2222,6 @@ lpfc_sli_queue_setup(struct lpfc_hba * phba)
|
||||||
INIT_LIST_HEAD(&pring->txcmplq);
|
INIT_LIST_HEAD(&pring->txcmplq);
|
||||||
INIT_LIST_HEAD(&pring->iocb_continueq);
|
INIT_LIST_HEAD(&pring->iocb_continueq);
|
||||||
INIT_LIST_HEAD(&pring->postbufq);
|
INIT_LIST_HEAD(&pring->postbufq);
|
||||||
cnt = pring->fast_iotag;
|
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
|
||||||
if (cnt) {
|
|
||||||
pring->fast_lookup =
|
|
||||||
kmalloc(cnt * sizeof (struct lpfc_iocbq *),
|
|
||||||
GFP_KERNEL);
|
|
||||||
if (pring->fast_lookup == 0) {
|
|
||||||
return (0);
|
|
||||||
}
|
|
||||||
memset((char *)pring->fast_lookup, 0,
|
|
||||||
cnt * sizeof (struct lpfc_iocbq *));
|
|
||||||
}
|
|
||||||
spin_lock_irq(phba->host->host_lock);
|
|
||||||
}
|
}
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return (1);
|
return (1);
|
||||||
|
@ -2292,10 +2263,8 @@ lpfc_sli_hba_down(struct lpfc_hba * phba)
|
||||||
flags);
|
flags);
|
||||||
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
(iocb->iocb_cmpl) (phba, iocb, iocb);
|
||||||
spin_lock_irqsave(phba->host->host_lock, flags);
|
spin_lock_irqsave(phba->host->host_lock, flags);
|
||||||
} else {
|
} else
|
||||||
list_add_tail(&iocb->list,
|
lpfc_sli_release_iocbq(phba, iocb);
|
||||||
&phba->lpfc_iocb_list);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
INIT_LIST_HEAD(&(pring->txq));
|
INIT_LIST_HEAD(&(pring->txq));
|
||||||
|
@ -2436,7 +2405,7 @@ lpfc_sli_abort_elsreq_cmpl(struct lpfc_hba * phba, struct lpfc_iocbq * cmdiocb,
|
||||||
kfree(buf_ptr);
|
kfree(buf_ptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
list_add_tail(&cmdiocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, cmdiocb);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2445,16 +2414,14 @@ lpfc_sli_issue_abort_iotag32(struct lpfc_hba * phba,
|
||||||
struct lpfc_sli_ring * pring,
|
struct lpfc_sli_ring * pring,
|
||||||
struct lpfc_iocbq * cmdiocb)
|
struct lpfc_iocbq * cmdiocb)
|
||||||
{
|
{
|
||||||
struct list_head *lpfc_iocb_list = &phba->lpfc_iocb_list;
|
struct lpfc_iocbq *abtsiocbp;
|
||||||
struct lpfc_iocbq *abtsiocbp = NULL;
|
|
||||||
IOCB_t *icmd = NULL;
|
IOCB_t *icmd = NULL;
|
||||||
IOCB_t *iabt = NULL;
|
IOCB_t *iabt = NULL;
|
||||||
|
|
||||||
/* issue ABTS for this IOCB based on iotag */
|
/* issue ABTS for this IOCB based on iotag */
|
||||||
list_remove_head(lpfc_iocb_list, abtsiocbp, struct lpfc_iocbq, list);
|
abtsiocbp = lpfc_sli_get_iocbq(phba);
|
||||||
if (abtsiocbp == NULL)
|
if (abtsiocbp == NULL)
|
||||||
return 0;
|
return 0;
|
||||||
memset(abtsiocbp, 0, sizeof (struct lpfc_iocbq));
|
|
||||||
|
|
||||||
iabt = &abtsiocbp->iocb;
|
iabt = &abtsiocbp->iocb;
|
||||||
icmd = &cmdiocb->iocb;
|
icmd = &cmdiocb->iocb;
|
||||||
|
@ -2473,7 +2440,7 @@ lpfc_sli_issue_abort_iotag32(struct lpfc_hba * phba,
|
||||||
abtsiocbp->iocb_cmpl = lpfc_sli_abort_elsreq_cmpl;
|
abtsiocbp->iocb_cmpl = lpfc_sli_abort_elsreq_cmpl;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
list_add_tail(&abtsiocbp->list, lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, abtsiocbp);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2485,7 +2452,7 @@ lpfc_sli_issue_abort_iotag32(struct lpfc_hba * phba,
|
||||||
iabt->ulpCommand = CMD_ABORT_MXRI64_CN;
|
iabt->ulpCommand = CMD_ABORT_MXRI64_CN;
|
||||||
|
|
||||||
if (lpfc_sli_issue_iocb(phba, pring, abtsiocbp, 0) == IOCB_ERROR) {
|
if (lpfc_sli_issue_iocb(phba, pring, abtsiocbp, 0) == IOCB_ERROR) {
|
||||||
list_add_tail(&abtsiocbp->list, lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, abtsiocbp);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2493,28 +2460,37 @@ lpfc_sli_issue_abort_iotag32(struct lpfc_hba * phba,
|
||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
lpfc_sli_validate_iocb_cmd(struct lpfc_scsi_buf *lpfc_cmd, uint16_t tgt_id,
|
lpfc_sli_validate_fcp_iocb(struct lpfc_iocbq *iocbq, uint16_t tgt_id,
|
||||||
uint64_t lun_id, struct lpfc_iocbq *iocb,
|
uint64_t lun_id, uint32_t ctx,
|
||||||
uint32_t ctx, lpfc_ctx_cmd ctx_cmd)
|
lpfc_ctx_cmd ctx_cmd)
|
||||||
{
|
{
|
||||||
|
struct lpfc_scsi_buf *lpfc_cmd;
|
||||||
|
struct scsi_cmnd *cmnd;
|
||||||
int rc = 1;
|
int rc = 1;
|
||||||
|
|
||||||
if (lpfc_cmd == NULL)
|
if (!(iocbq->iocb_flag & LPFC_IO_FCP))
|
||||||
|
return rc;
|
||||||
|
|
||||||
|
lpfc_cmd = container_of(iocbq, struct lpfc_scsi_buf, cur_iocbq);
|
||||||
|
cmnd = lpfc_cmd->pCmd;
|
||||||
|
|
||||||
|
if (cmnd == NULL)
|
||||||
return rc;
|
return rc;
|
||||||
|
|
||||||
switch (ctx_cmd) {
|
switch (ctx_cmd) {
|
||||||
case LPFC_CTX_LUN:
|
case LPFC_CTX_LUN:
|
||||||
if ((lpfc_cmd->pCmd->device->id == tgt_id) &&
|
if ((cmnd->device->id == tgt_id) &&
|
||||||
(lpfc_cmd->pCmd->device->lun == lun_id))
|
(cmnd->device->lun == lun_id))
|
||||||
rc = 0;
|
rc = 0;
|
||||||
break;
|
break;
|
||||||
case LPFC_CTX_TGT:
|
case LPFC_CTX_TGT:
|
||||||
if (lpfc_cmd->pCmd->device->id == tgt_id)
|
if (cmnd->device->id == tgt_id)
|
||||||
rc = 0;
|
rc = 0;
|
||||||
break;
|
break;
|
||||||
case LPFC_CTX_CTX:
|
case LPFC_CTX_CTX:
|
||||||
if (iocb->iocb.ulpContext == ctx)
|
if (iocbq->iocb.ulpContext == ctx)
|
||||||
rc = 0;
|
rc = 0;
|
||||||
|
break;
|
||||||
case LPFC_CTX_HOST:
|
case LPFC_CTX_HOST:
|
||||||
rc = 0;
|
rc = 0;
|
||||||
break;
|
break;
|
||||||
|
@ -2531,30 +2507,17 @@ int
|
||||||
lpfc_sli_sum_iocb(struct lpfc_hba *phba, struct lpfc_sli_ring *pring,
|
lpfc_sli_sum_iocb(struct lpfc_hba *phba, struct lpfc_sli_ring *pring,
|
||||||
uint16_t tgt_id, uint64_t lun_id, lpfc_ctx_cmd ctx_cmd)
|
uint16_t tgt_id, uint64_t lun_id, lpfc_ctx_cmd ctx_cmd)
|
||||||
{
|
{
|
||||||
struct lpfc_iocbq *iocb, *next_iocb;
|
struct lpfc_iocbq *iocbq;
|
||||||
IOCB_t *cmd = NULL;
|
int sum, i;
|
||||||
struct lpfc_scsi_buf *lpfc_cmd;
|
|
||||||
int sum = 0, ret_val = 0;
|
|
||||||
|
|
||||||
/* Next check the txcmplq */
|
for (i = 1, sum = 0; i <= phba->sli.last_iotag; i++) {
|
||||||
list_for_each_entry_safe(iocb, next_iocb, &pring->txcmplq, list) {
|
iocbq = phba->sli.iocbq_lookup[i];
|
||||||
cmd = &iocb->iocb;
|
|
||||||
|
|
||||||
/* Must be a FCP command */
|
if (lpfc_sli_validate_fcp_iocb (iocbq, tgt_id, lun_id,
|
||||||
if ((cmd->ulpCommand != CMD_FCP_ICMND64_CR) &&
|
0, ctx_cmd) == 0)
|
||||||
(cmd->ulpCommand != CMD_FCP_IWRITE64_CR) &&
|
sum++;
|
||||||
(cmd->ulpCommand != CMD_FCP_IREAD64_CR)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* context1 MUST be a struct lpfc_scsi_buf */
|
|
||||||
lpfc_cmd = (struct lpfc_scsi_buf *) (iocb->context1);
|
|
||||||
ret_val = lpfc_sli_validate_iocb_cmd(lpfc_cmd, tgt_id, lun_id,
|
|
||||||
NULL, 0, ctx_cmd);
|
|
||||||
if (ret_val != 0)
|
|
||||||
continue;
|
|
||||||
sum++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return sum;
|
return sum;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2563,7 +2526,7 @@ lpfc_sli_abort_fcp_cmpl(struct lpfc_hba * phba, struct lpfc_iocbq * cmdiocb,
|
||||||
struct lpfc_iocbq * rspiocb)
|
struct lpfc_iocbq * rspiocb)
|
||||||
{
|
{
|
||||||
spin_lock_irq(phba->host->host_lock);
|
spin_lock_irq(phba->host->host_lock);
|
||||||
list_add_tail(&cmdiocb->list, &phba->lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, cmdiocb);
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -2573,39 +2536,27 @@ lpfc_sli_abort_iocb(struct lpfc_hba *phba, struct lpfc_sli_ring *pring,
|
||||||
uint16_t tgt_id, uint64_t lun_id, uint32_t ctx,
|
uint16_t tgt_id, uint64_t lun_id, uint32_t ctx,
|
||||||
lpfc_ctx_cmd abort_cmd)
|
lpfc_ctx_cmd abort_cmd)
|
||||||
{
|
{
|
||||||
struct lpfc_iocbq *iocb, *next_iocb;
|
struct lpfc_iocbq *iocbq;
|
||||||
struct lpfc_iocbq *abtsiocb = NULL;
|
struct lpfc_iocbq *abtsiocb;
|
||||||
struct list_head *lpfc_iocb_list = &phba->lpfc_iocb_list;
|
|
||||||
IOCB_t *cmd = NULL;
|
IOCB_t *cmd = NULL;
|
||||||
struct lpfc_scsi_buf *lpfc_cmd;
|
|
||||||
int errcnt = 0, ret_val = 0;
|
int errcnt = 0, ret_val = 0;
|
||||||
|
int i;
|
||||||
|
|
||||||
list_for_each_entry_safe(iocb, next_iocb, &pring->txcmplq, list) {
|
for (i = 1; i <= phba->sli.last_iotag; i++) {
|
||||||
cmd = &iocb->iocb;
|
iocbq = phba->sli.iocbq_lookup[i];
|
||||||
|
|
||||||
/* Must be a FCP command */
|
if (lpfc_sli_validate_fcp_iocb (iocbq, tgt_id, lun_id,
|
||||||
if ((cmd->ulpCommand != CMD_FCP_ICMND64_CR) &&
|
0, abort_cmd) != 0)
|
||||||
(cmd->ulpCommand != CMD_FCP_IWRITE64_CR) &&
|
|
||||||
(cmd->ulpCommand != CMD_FCP_IREAD64_CR)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* context1 MUST be a struct lpfc_scsi_buf */
|
|
||||||
lpfc_cmd = (struct lpfc_scsi_buf *) (iocb->context1);
|
|
||||||
ret_val = lpfc_sli_validate_iocb_cmd(lpfc_cmd, tgt_id, lun_id,
|
|
||||||
iocb, ctx, abort_cmd);
|
|
||||||
if (ret_val != 0)
|
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
/* issue ABTS for this IOCB based on iotag */
|
/* issue ABTS for this IOCB based on iotag */
|
||||||
list_remove_head(lpfc_iocb_list, abtsiocb, struct lpfc_iocbq,
|
abtsiocb = lpfc_sli_get_iocbq(phba);
|
||||||
list);
|
|
||||||
if (abtsiocb == NULL) {
|
if (abtsiocb == NULL) {
|
||||||
errcnt++;
|
errcnt++;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
memset(abtsiocb, 0, sizeof (struct lpfc_iocbq));
|
|
||||||
|
|
||||||
|
cmd = &iocbq->iocb;
|
||||||
abtsiocb->iocb.un.acxri.abortType = ABORT_TYPE_ABTS;
|
abtsiocb->iocb.un.acxri.abortType = ABORT_TYPE_ABTS;
|
||||||
abtsiocb->iocb.un.acxri.abortContextTag = cmd->ulpContext;
|
abtsiocb->iocb.un.acxri.abortContextTag = cmd->ulpContext;
|
||||||
abtsiocb->iocb.un.acxri.abortIoTag = cmd->ulpIoTag;
|
abtsiocb->iocb.un.acxri.abortIoTag = cmd->ulpIoTag;
|
||||||
|
@ -2621,7 +2572,7 @@ lpfc_sli_abort_iocb(struct lpfc_hba *phba, struct lpfc_sli_ring *pring,
|
||||||
abtsiocb->iocb_cmpl = lpfc_sli_abort_fcp_cmpl;
|
abtsiocb->iocb_cmpl = lpfc_sli_abort_fcp_cmpl;
|
||||||
ret_val = lpfc_sli_issue_iocb(phba, pring, abtsiocb, 0);
|
ret_val = lpfc_sli_issue_iocb(phba, pring, abtsiocb, 0);
|
||||||
if (ret_val == IOCB_ERROR) {
|
if (ret_val == IOCB_ERROR) {
|
||||||
list_add_tail(&abtsiocb->list, lpfc_iocb_list);
|
lpfc_sli_release_iocbq(phba, abtsiocb);
|
||||||
errcnt++;
|
errcnt++;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -2630,83 +2581,99 @@ lpfc_sli_abort_iocb(struct lpfc_hba *phba, struct lpfc_sli_ring *pring,
|
||||||
return errcnt;
|
return errcnt;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
static void
|
||||||
lpfc_sli_wake_iocb_high_priority(struct lpfc_hba * phba,
|
lpfc_sli_wake_iocb_wait(struct lpfc_hba *phba,
|
||||||
struct lpfc_iocbq * queue1,
|
struct lpfc_iocbq *cmdiocbq,
|
||||||
struct lpfc_iocbq * queue2)
|
struct lpfc_iocbq *rspiocbq)
|
||||||
{
|
{
|
||||||
if (queue1->context2 && queue2)
|
wait_queue_head_t *pdone_q;
|
||||||
memcpy(queue1->context2, queue2, sizeof (struct lpfc_iocbq));
|
unsigned long iflags;
|
||||||
|
|
||||||
/* The waiter is looking for LPFC_IO_HIPRI bit to be set
|
spin_lock_irqsave(phba->host->host_lock, iflags);
|
||||||
as a signal to wake up */
|
cmdiocbq->iocb_flag |= LPFC_IO_WAKE;
|
||||||
queue1->iocb_flag |= LPFC_IO_HIPRI;
|
if (cmdiocbq->context2 && rspiocbq)
|
||||||
|
memcpy(&((struct lpfc_iocbq *)cmdiocbq->context2)->iocb,
|
||||||
|
&rspiocbq->iocb, sizeof(IOCB_t));
|
||||||
|
|
||||||
|
pdone_q = cmdiocbq->context_un.wait_queue;
|
||||||
|
spin_unlock_irqrestore(phba->host->host_lock, iflags);
|
||||||
|
if (pdone_q)
|
||||||
|
wake_up(pdone_q);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Issue the caller's iocb and wait for its completion, but no longer than the
|
||||||
|
* caller's timeout. Note that iocb_flags is cleared before the
|
||||||
|
* lpfc_sli_issue_call since the wake routine sets a unique value and by
|
||||||
|
* definition this is a wait function.
|
||||||
|
*/
|
||||||
int
|
int
|
||||||
lpfc_sli_issue_iocb_wait_high_priority(struct lpfc_hba * phba,
|
lpfc_sli_issue_iocb_wait(struct lpfc_hba * phba,
|
||||||
struct lpfc_sli_ring * pring,
|
struct lpfc_sli_ring * pring,
|
||||||
struct lpfc_iocbq * piocb,
|
struct lpfc_iocbq * piocb,
|
||||||
uint32_t flag,
|
struct lpfc_iocbq * prspiocbq,
|
||||||
struct lpfc_iocbq * prspiocbq,
|
uint32_t timeout)
|
||||||
uint32_t timeout)
|
|
||||||
{
|
{
|
||||||
int j, delay_time, retval = IOCB_ERROR;
|
DECLARE_WAIT_QUEUE_HEAD(done_q);
|
||||||
|
long timeleft, timeout_req = 0;
|
||||||
/* The caller must left context1 empty. */
|
int retval = IOCB_SUCCESS;
|
||||||
if (piocb->context_un.hipri_wait_queue != 0) {
|
|
||||||
return IOCB_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If the caller has provided a response iocbq buffer, context2 must
|
* If the caller has provided a response iocbq buffer, then context2
|
||||||
* be NULL or its an error.
|
* is NULL or its an error.
|
||||||
*/
|
*/
|
||||||
if (prspiocbq && piocb->context2) {
|
if (prspiocbq) {
|
||||||
return IOCB_ERROR;
|
if (piocb->context2)
|
||||||
|
return IOCB_ERROR;
|
||||||
|
piocb->context2 = prspiocbq;
|
||||||
}
|
}
|
||||||
|
|
||||||
piocb->context2 = prspiocbq;
|
piocb->iocb_cmpl = lpfc_sli_wake_iocb_wait;
|
||||||
|
piocb->context_un.wait_queue = &done_q;
|
||||||
|
piocb->iocb_flag &= ~LPFC_IO_WAKE;
|
||||||
|
|
||||||
/* Setup callback routine and issue the command. */
|
retval = lpfc_sli_issue_iocb(phba, pring, piocb, 0);
|
||||||
piocb->iocb_cmpl = lpfc_sli_wake_iocb_high_priority;
|
if (retval == IOCB_SUCCESS) {
|
||||||
retval = lpfc_sli_issue_iocb(phba, pring, piocb,
|
timeout_req = timeout * HZ;
|
||||||
flag | SLI_IOCB_HIGH_PRIORITY);
|
spin_unlock_irq(phba->host->host_lock);
|
||||||
if (retval != IOCB_SUCCESS) {
|
timeleft = wait_event_timeout(done_q,
|
||||||
piocb->context2 = NULL;
|
piocb->iocb_flag & LPFC_IO_WAKE,
|
||||||
return IOCB_ERROR;
|
timeout_req);
|
||||||
}
|
spin_lock_irq(phba->host->host_lock);
|
||||||
|
|
||||||
/*
|
if (timeleft == 0) {
|
||||||
* This high-priority iocb was sent out-of-band. Poll for its
|
lpfc_printf_log(phba, KERN_ERR, LOG_SLI,
|
||||||
* completion rather than wait for a signal. Note that the host_lock
|
"%d:0329 IOCB wait timeout error - no "
|
||||||
* is held by the midlayer and must be released here to allow the
|
"wake response Data x%x\n",
|
||||||
* interrupt handlers to complete the IO and signal this routine via
|
phba->brd_no, timeout);
|
||||||
* the iocb_flag.
|
retval = IOCB_TIMEDOUT;
|
||||||
* Also, the delay_time is computed to be one second longer than
|
} else if (!(piocb->iocb_flag & LPFC_IO_WAKE)) {
|
||||||
* the scsi command timeout to give the FW time to abort on
|
lpfc_printf_log(phba, KERN_ERR, LOG_SLI,
|
||||||
* timeout rather than the driver just giving up. Typically,
|
"%d:0330 IOCB wake NOT set, "
|
||||||
* the midlayer does not specify a time for this command so the
|
"Data x%x x%lx\n", phba->brd_no,
|
||||||
* driver is free to enforce its own timeout.
|
timeout, (timeleft / jiffies));
|
||||||
*/
|
retval = IOCB_TIMEDOUT;
|
||||||
|
} else {
|
||||||
delay_time = ((timeout + 1) * 1000) >> 6;
|
lpfc_printf_log(phba, KERN_INFO, LOG_SLI,
|
||||||
retval = IOCB_ERROR;
|
"%d:0331 IOCB wake signaled\n",
|
||||||
spin_unlock_irq(phba->host->host_lock);
|
phba->brd_no);
|
||||||
for (j = 0; j < 64; j++) {
|
|
||||||
msleep(delay_time);
|
|
||||||
if (piocb->iocb_flag & LPFC_IO_HIPRI) {
|
|
||||||
piocb->iocb_flag &= ~LPFC_IO_HIPRI;
|
|
||||||
retval = IOCB_SUCCESS;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
lpfc_printf_log(phba, KERN_INFO, LOG_SLI,
|
||||||
|
"%d:0332 IOCB wait issue failed, Data x%x\n",
|
||||||
|
phba->brd_no, retval);
|
||||||
|
retval = IOCB_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_lock_irq(phba->host->host_lock);
|
if (prspiocbq)
|
||||||
piocb->context2 = NULL;
|
piocb->context2 = NULL;
|
||||||
|
|
||||||
|
piocb->context_un.wait_queue = NULL;
|
||||||
|
piocb->iocb_cmpl = NULL;
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
lpfc_sli_issue_mbox_wait(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmboxq,
|
lpfc_sli_issue_mbox_wait(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmboxq,
|
||||||
uint32_t timeout)
|
uint32_t timeout)
|
||||||
|
|
|
@ -33,13 +33,15 @@ typedef enum _lpfc_ctx_cmd {
|
||||||
struct lpfc_iocbq {
|
struct lpfc_iocbq {
|
||||||
/* lpfc_iocbqs are used in double linked lists */
|
/* lpfc_iocbqs are used in double linked lists */
|
||||||
struct list_head list;
|
struct list_head list;
|
||||||
|
uint16_t iotag; /* pre-assigned IO tag */
|
||||||
|
uint16_t rsvd1;
|
||||||
|
|
||||||
IOCB_t iocb; /* IOCB cmd */
|
IOCB_t iocb; /* IOCB cmd */
|
||||||
uint8_t retry; /* retry counter for IOCB cmd - if needed */
|
uint8_t retry; /* retry counter for IOCB cmd - if needed */
|
||||||
uint8_t iocb_flag;
|
uint8_t iocb_flag;
|
||||||
#define LPFC_IO_POLL 1 /* Polling mode iocb */
|
#define LPFC_IO_LIBDFC 1 /* libdfc iocb */
|
||||||
#define LPFC_IO_LIBDFC 2 /* libdfc iocb */
|
#define LPFC_IO_WAKE 2 /* High Priority Queue signal flag */
|
||||||
#define LPFC_IO_WAIT 4
|
#define LPFC_IO_FCP 4 /* FCP command -- iocbq in scsi_buf */
|
||||||
#define LPFC_IO_HIPRI 8 /* High Priority Queue signal flag */
|
|
||||||
|
|
||||||
uint8_t abort_count;
|
uint8_t abort_count;
|
||||||
uint8_t rsvd2;
|
uint8_t rsvd2;
|
||||||
|
@ -48,8 +50,7 @@ struct lpfc_iocbq {
|
||||||
void *context2; /* caller context information */
|
void *context2; /* caller context information */
|
||||||
void *context3; /* caller context information */
|
void *context3; /* caller context information */
|
||||||
union {
|
union {
|
||||||
wait_queue_head_t *hipri_wait_queue; /* High Priority Queue wait
|
wait_queue_head_t *wait_queue;
|
||||||
queue */
|
|
||||||
struct lpfc_iocbq *rsp_iocb;
|
struct lpfc_iocbq *rsp_iocb;
|
||||||
struct lpfcMboxq *mbox;
|
struct lpfcMboxq *mbox;
|
||||||
} context_un;
|
} context_un;
|
||||||
|
@ -125,10 +126,10 @@ struct lpfc_sli_ring {
|
||||||
|
|
||||||
uint32_t local_getidx; /* last available cmd index (from cmdGetInx) */
|
uint32_t local_getidx; /* last available cmd index (from cmdGetInx) */
|
||||||
uint32_t next_cmdidx; /* next_cmd index */
|
uint32_t next_cmdidx; /* next_cmd index */
|
||||||
|
uint32_t rspidx; /* current index in response ring */
|
||||||
|
uint32_t cmdidx; /* current index in command ring */
|
||||||
uint8_t rsvd;
|
uint8_t rsvd;
|
||||||
uint8_t ringno; /* ring number */
|
uint8_t ringno; /* ring number */
|
||||||
uint8_t rspidx; /* current index in response ring */
|
|
||||||
uint8_t cmdidx; /* current index in command ring */
|
|
||||||
uint16_t numCiocb; /* number of command iocb's per ring */
|
uint16_t numCiocb; /* number of command iocb's per ring */
|
||||||
uint16_t numRiocb; /* number of rsp iocb's per ring */
|
uint16_t numRiocb; /* number of rsp iocb's per ring */
|
||||||
|
|
||||||
|
@ -200,6 +201,11 @@ struct lpfc_sli {
|
||||||
cmd */
|
cmd */
|
||||||
|
|
||||||
uint32_t *MBhostaddr; /* virtual address for mbox cmds */
|
uint32_t *MBhostaddr; /* virtual address for mbox cmds */
|
||||||
|
|
||||||
|
#define LPFC_IOCBQ_LOOKUP_INCREMENT 1024
|
||||||
|
struct lpfc_iocbq ** iocbq_lookup; /* array to lookup IOCB by IOTAG */
|
||||||
|
size_t iocbq_lookup_len; /* current lengs of the array */
|
||||||
|
uint16_t last_iotag; /* last allocated IOTAG */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Given a pointer to the start of the ring, and the slot number of
|
/* Given a pointer to the start of the ring, and the slot number of
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
* included with this package. *
|
* included with this package. *
|
||||||
*******************************************************************/
|
*******************************************************************/
|
||||||
|
|
||||||
#define LPFC_DRIVER_VERSION "8.0.30"
|
#define LPFC_DRIVER_VERSION "8.1.0"
|
||||||
|
|
||||||
#define LPFC_DRIVER_NAME "lpfc"
|
#define LPFC_DRIVER_NAME "lpfc"
|
||||||
|
|
||||||
|
|
|
@ -758,9 +758,8 @@ static int megasas_generic_reset(struct scsi_cmnd *scmd)
|
||||||
|
|
||||||
instance = (struct megasas_instance *)scmd->device->host->hostdata;
|
instance = (struct megasas_instance *)scmd->device->host->hostdata;
|
||||||
|
|
||||||
printk(KERN_NOTICE "megasas: RESET -%ld cmd=%x <c=%d t=%d l=%d>\n",
|
scmd_printk(KERN_NOTICE, scmd, "megasas: RESET -%ld cmd=%x\n",
|
||||||
scmd->serial_number, scmd->cmnd[0], scmd->device->channel,
|
scmd->serial_number, scmd->cmnd[0]);
|
||||||
scmd->device->id, scmd->device->lun);
|
|
||||||
|
|
||||||
if (instance->hw_crit_error) {
|
if (instance->hw_crit_error) {
|
||||||
printk(KERN_ERR "megasas: cannot recover from previous reset "
|
printk(KERN_ERR "megasas: cannot recover from previous reset "
|
||||||
|
|
|
@ -3481,8 +3481,8 @@ static int ncr_queue_command (struct ncb *np, struct scsi_cmnd *cmd)
|
||||||
**----------------------------------------------------
|
**----------------------------------------------------
|
||||||
*/
|
*/
|
||||||
if (np->settle_time && cmd->timeout_per_command >= HZ) {
|
if (np->settle_time && cmd->timeout_per_command >= HZ) {
|
||||||
u_long tlimit = ktime_get(cmd->timeout_per_command - HZ);
|
u_long tlimit = jiffies + cmd->timeout_per_command - HZ;
|
||||||
if (ktime_dif(np->settle_time, tlimit) > 0)
|
if (time_after(np->settle_time, tlimit))
|
||||||
np->settle_time = tlimit;
|
np->settle_time = tlimit;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3516,7 +3516,7 @@ static int ncr_queue_command (struct ncb *np, struct scsi_cmnd *cmd)
|
||||||
** Force ordered tag if necessary to avoid timeouts
|
** Force ordered tag if necessary to avoid timeouts
|
||||||
** and to preserve interactivity.
|
** and to preserve interactivity.
|
||||||
*/
|
*/
|
||||||
if (lp && ktime_exp(lp->tags_stime)) {
|
if (lp && time_after(jiffies, lp->tags_stime)) {
|
||||||
if (lp->tags_smap) {
|
if (lp->tags_smap) {
|
||||||
order = M_ORDERED_TAG;
|
order = M_ORDERED_TAG;
|
||||||
if ((DEBUG_FLAGS & DEBUG_TAGS)||bootverbose>2){
|
if ((DEBUG_FLAGS & DEBUG_TAGS)||bootverbose>2){
|
||||||
|
@ -3524,7 +3524,7 @@ static int ncr_queue_command (struct ncb *np, struct scsi_cmnd *cmd)
|
||||||
"ordered tag forced.\n");
|
"ordered tag forced.\n");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
lp->tags_stime = ktime_get(3*HZ);
|
lp->tags_stime = jiffies + 3*HZ;
|
||||||
lp->tags_smap = lp->tags_umap;
|
lp->tags_smap = lp->tags_umap;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3669,7 +3669,7 @@ static int ncr_queue_command (struct ncb *np, struct scsi_cmnd *cmd)
|
||||||
/*
|
/*
|
||||||
** select
|
** select
|
||||||
*/
|
*/
|
||||||
cp->phys.select.sel_id = sdev->id;
|
cp->phys.select.sel_id = sdev_id(sdev);
|
||||||
cp->phys.select.sel_scntl3 = tp->wval;
|
cp->phys.select.sel_scntl3 = tp->wval;
|
||||||
cp->phys.select.sel_sxfer = tp->sval;
|
cp->phys.select.sel_sxfer = tp->sval;
|
||||||
/*
|
/*
|
||||||
|
@ -3792,7 +3792,7 @@ static int ncr_reset_scsi_bus(struct ncb *np, int enab_int, int settle_delay)
|
||||||
u32 term;
|
u32 term;
|
||||||
int retv = 0;
|
int retv = 0;
|
||||||
|
|
||||||
np->settle_time = ktime_get(settle_delay * HZ);
|
np->settle_time = jiffies + settle_delay * HZ;
|
||||||
|
|
||||||
if (bootverbose > 1)
|
if (bootverbose > 1)
|
||||||
printk("%s: resetting, "
|
printk("%s: resetting, "
|
||||||
|
@ -4820,7 +4820,7 @@ static void ncr_set_sync_wide_status (struct ncb *np, u_char target)
|
||||||
*/
|
*/
|
||||||
for (cp = np->ccb; cp; cp = cp->link_ccb) {
|
for (cp = np->ccb; cp; cp = cp->link_ccb) {
|
||||||
if (!cp->cmd) continue;
|
if (!cp->cmd) continue;
|
||||||
if (cp->cmd->device->id != target) continue;
|
if (scmd_id(cp->cmd) != target) continue;
|
||||||
#if 0
|
#if 0
|
||||||
cp->sync_status = tp->sval;
|
cp->sync_status = tp->sval;
|
||||||
cp->wide_status = tp->wval;
|
cp->wide_status = tp->wval;
|
||||||
|
@ -4844,7 +4844,7 @@ static void ncr_setsync (struct ncb *np, struct ccb *cp, u_char scntl3, u_char s
|
||||||
u_char target = INB (nc_sdid) & 0x0f;
|
u_char target = INB (nc_sdid) & 0x0f;
|
||||||
u_char idiv;
|
u_char idiv;
|
||||||
|
|
||||||
BUG_ON(target != (cmd->device->id & 0xf));
|
BUG_ON(target != (scmd_id(cmd) & 0xf));
|
||||||
|
|
||||||
tp = &np->target[target];
|
tp = &np->target[target];
|
||||||
|
|
||||||
|
@ -4902,7 +4902,7 @@ static void ncr_setwide (struct ncb *np, struct ccb *cp, u_char wide, u_char ack
|
||||||
u_char scntl3;
|
u_char scntl3;
|
||||||
u_char sxfer;
|
u_char sxfer;
|
||||||
|
|
||||||
BUG_ON(target != (cmd->device->id & 0xf));
|
BUG_ON(target != (scmd_id(cmd) & 0xf));
|
||||||
|
|
||||||
tp = &np->target[target];
|
tp = &np->target[target];
|
||||||
tp->widedone = wide+1;
|
tp->widedone = wide+1;
|
||||||
|
@ -5044,7 +5044,7 @@ static void ncr_setup_tags (struct ncb *np, struct scsi_device *sdev)
|
||||||
|
|
||||||
static void ncr_timeout (struct ncb *np)
|
static void ncr_timeout (struct ncb *np)
|
||||||
{
|
{
|
||||||
u_long thistime = ktime_get(0);
|
u_long thistime = jiffies;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
** If release process in progress, let's go
|
** If release process in progress, let's go
|
||||||
|
@ -5057,7 +5057,7 @@ static void ncr_timeout (struct ncb *np)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
np->timer.expires = ktime_get(SCSI_NCR_TIMER_INTERVAL);
|
np->timer.expires = jiffies + SCSI_NCR_TIMER_INTERVAL;
|
||||||
add_timer(&np->timer);
|
add_timer(&np->timer);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -5336,8 +5336,8 @@ void ncr_exception (struct ncb *np)
|
||||||
**=========================================================
|
**=========================================================
|
||||||
*/
|
*/
|
||||||
|
|
||||||
if (ktime_exp(np->regtime)) {
|
if (time_after(jiffies, np->regtime)) {
|
||||||
np->regtime = ktime_get(10*HZ);
|
np->regtime = jiffies + 10*HZ;
|
||||||
for (i = 0; i<sizeof(np->regdump); i++)
|
for (i = 0; i<sizeof(np->regdump); i++)
|
||||||
((char*)&np->regdump)[i] = INB_OFF(i);
|
((char*)&np->regdump)[i] = INB_OFF(i);
|
||||||
np->regdump.nc_dstat = dstat;
|
np->regdump.nc_dstat = dstat;
|
||||||
|
@ -5453,7 +5453,7 @@ static int ncr_int_sbmc (struct ncb *np)
|
||||||
** Suspend command processing for 1 second and
|
** Suspend command processing for 1 second and
|
||||||
** reinitialize all except the chip.
|
** reinitialize all except the chip.
|
||||||
*/
|
*/
|
||||||
np->settle_time = ktime_get(1*HZ);
|
np->settle_time = jiffies + HZ;
|
||||||
ncr_init (np, 0, bootverbose ? "scsi mode change" : NULL, HS_RESET);
|
ncr_init (np, 0, bootverbose ? "scsi mode change" : NULL, HS_RESET);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -6923,7 +6923,7 @@ static struct lcb *ncr_setup_lcb (struct ncb *np, struct scsi_device *sdev)
|
||||||
for (i = 0 ; i < MAX_TAGS ; i++)
|
for (i = 0 ; i < MAX_TAGS ; i++)
|
||||||
lp->cb_tags[i] = i;
|
lp->cb_tags[i] = i;
|
||||||
lp->maxnxs = MAX_TAGS;
|
lp->maxnxs = MAX_TAGS;
|
||||||
lp->tags_stime = ktime_get(3*HZ);
|
lp->tags_stime = jiffies + 3*HZ;
|
||||||
ncr_setup_tags (np, sdev);
|
ncr_setup_tags (np, sdev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -481,7 +481,7 @@ static int nsp32_selection_autopara(struct scsi_cmnd *SCpnt)
|
||||||
nsp32_hw_data *data = (nsp32_hw_data *)SCpnt->device->host->hostdata;
|
nsp32_hw_data *data = (nsp32_hw_data *)SCpnt->device->host->hostdata;
|
||||||
unsigned int base = SCpnt->device->host->io_port;
|
unsigned int base = SCpnt->device->host->io_port;
|
||||||
unsigned int host_id = SCpnt->device->host->this_id;
|
unsigned int host_id = SCpnt->device->host->this_id;
|
||||||
unsigned char target = SCpnt->device->id;
|
unsigned char target = scmd_id(SCpnt);
|
||||||
nsp32_autoparam *param = data->autoparam;
|
nsp32_autoparam *param = data->autoparam;
|
||||||
unsigned char phase;
|
unsigned char phase;
|
||||||
int i, ret;
|
int i, ret;
|
||||||
|
@ -612,7 +612,7 @@ static int nsp32_selection_autoscsi(struct scsi_cmnd *SCpnt)
|
||||||
nsp32_hw_data *data = (nsp32_hw_data *)SCpnt->device->host->hostdata;
|
nsp32_hw_data *data = (nsp32_hw_data *)SCpnt->device->host->hostdata;
|
||||||
unsigned int base = SCpnt->device->host->io_port;
|
unsigned int base = SCpnt->device->host->io_port;
|
||||||
unsigned int host_id = SCpnt->device->host->this_id;
|
unsigned int host_id = SCpnt->device->host->this_id;
|
||||||
unsigned char target = SCpnt->device->id;
|
unsigned char target = scmd_id(SCpnt);
|
||||||
unsigned char phase;
|
unsigned char phase;
|
||||||
int status;
|
int status;
|
||||||
unsigned short command = 0;
|
unsigned short command = 0;
|
||||||
|
@ -973,7 +973,7 @@ static int nsp32_queuecommand(struct scsi_cmnd *SCpnt, void (*done)(struct scsi_
|
||||||
}
|
}
|
||||||
|
|
||||||
/* check target ID is not same as this initiator ID */
|
/* check target ID is not same as this initiator ID */
|
||||||
if (SCpnt->device->id == SCpnt->device->host->this_id) {
|
if (scmd_id(SCpnt) == SCpnt->device->host->this_id) {
|
||||||
nsp32_dbg(NSP32_DEBUG_QUEUECOMMAND, "terget==host???");
|
nsp32_dbg(NSP32_DEBUG_QUEUECOMMAND, "terget==host???");
|
||||||
SCpnt->result = DID_BAD_TARGET << 16;
|
SCpnt->result = DID_BAD_TARGET << 16;
|
||||||
done(SCpnt);
|
done(SCpnt);
|
||||||
|
@ -1028,7 +1028,7 @@ static int nsp32_queuecommand(struct scsi_cmnd *SCpnt, void (*done)(struct scsi_
|
||||||
* (target don't have SDTR_DONE and SDTR_INITIATOR), sync
|
* (target don't have SDTR_DONE and SDTR_INITIATOR), sync
|
||||||
* message SDTR is needed to do synchronous transfer.
|
* message SDTR is needed to do synchronous transfer.
|
||||||
*/
|
*/
|
||||||
target = &data->target[SCpnt->device->id];
|
target = &data->target[scmd_id(SCpnt)];
|
||||||
data->cur_target = target;
|
data->cur_target = target;
|
||||||
|
|
||||||
if (!(target->sync_flag & (SDTR_DONE | SDTR_INITIATOR | SDTR_TARGET))) {
|
if (!(target->sync_flag & (SDTR_DONE | SDTR_INITIATOR | SDTR_TARGET))) {
|
||||||
|
|
|
@ -5819,9 +5819,9 @@ static int osst_probe(struct device *dev)
|
||||||
}
|
}
|
||||||
drive->number = devfs_register_tape(SDp->devfs_name);
|
drive->number = devfs_register_tape(SDp->devfs_name);
|
||||||
|
|
||||||
printk(KERN_INFO
|
sdev_printk(KERN_INFO, SDp,
|
||||||
"osst :I: Attached OnStream %.5s tape at scsi%d, channel %d, id %d, lun %d as %s\n",
|
"osst :I: Attached OnStream %.5s tape as %s\n",
|
||||||
SDp->model, SDp->host->host_no, SDp->channel, SDp->id, SDp->lun, tape_name(tpnt));
|
SDp->model, tape_name(tpnt));
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
|
|
@ -201,7 +201,7 @@ static int nsp_queuecommand(Scsi_Cmnd *SCpnt, void (*done)(Scsi_Cmnd *))
|
||||||
#ifdef NSP_DEBUG
|
#ifdef NSP_DEBUG
|
||||||
/*unsigned int host_id = SCpnt->device->host->this_id;*/
|
/*unsigned int host_id = SCpnt->device->host->this_id;*/
|
||||||
/*unsigned int base = SCpnt->device->host->io_port;*/
|
/*unsigned int base = SCpnt->device->host->io_port;*/
|
||||||
unsigned char target = SCpnt->device->id;
|
unsigned char target = scmd_id(SCpnt);
|
||||||
#endif
|
#endif
|
||||||
nsp_hw_data *data = (nsp_hw_data *)SCpnt->device->host->hostdata;
|
nsp_hw_data *data = (nsp_hw_data *)SCpnt->device->host->hostdata;
|
||||||
|
|
||||||
|
@ -373,7 +373,7 @@ static int nsphw_start_selection(Scsi_Cmnd *SCpnt)
|
||||||
{
|
{
|
||||||
unsigned int host_id = SCpnt->device->host->this_id;
|
unsigned int host_id = SCpnt->device->host->this_id;
|
||||||
unsigned int base = SCpnt->device->host->io_port;
|
unsigned int base = SCpnt->device->host->io_port;
|
||||||
unsigned char target = SCpnt->device->id;
|
unsigned char target = scmd_id(SCpnt);
|
||||||
nsp_hw_data *data = (nsp_hw_data *)SCpnt->device->host->hostdata;
|
nsp_hw_data *data = (nsp_hw_data *)SCpnt->device->host->hostdata;
|
||||||
int time_out;
|
int time_out;
|
||||||
unsigned char phase, arbit;
|
unsigned char phase, arbit;
|
||||||
|
@ -452,7 +452,7 @@ static struct nsp_sync_table nsp_sync_table_20M[] = {
|
||||||
*/
|
*/
|
||||||
static int nsp_analyze_sdtr(Scsi_Cmnd *SCpnt)
|
static int nsp_analyze_sdtr(Scsi_Cmnd *SCpnt)
|
||||||
{
|
{
|
||||||
unsigned char target = SCpnt->device->id;
|
unsigned char target = scmd_id(SCpnt);
|
||||||
// unsigned char lun = SCpnt->device->lun;
|
// unsigned char lun = SCpnt->device->lun;
|
||||||
nsp_hw_data *data = (nsp_hw_data *)SCpnt->device->host->hostdata;
|
nsp_hw_data *data = (nsp_hw_data *)SCpnt->device->host->hostdata;
|
||||||
sync_data *sync = &(data->Sync[target]);
|
sync_data *sync = &(data->Sync[target]);
|
||||||
|
@ -677,7 +677,7 @@ static int nsp_reselected(Scsi_Cmnd *SCpnt)
|
||||||
target++;
|
target++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (SCpnt->device->id != target) {
|
if (scmd_id(SCpnt) != target) {
|
||||||
nsp_msg(KERN_ERR, "XXX: reselect ID must be %d in this implementation.", target);
|
nsp_msg(KERN_ERR, "XXX: reselect ID must be %d in this implementation.", target);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -912,7 +912,7 @@ static void nsp_pio_write(Scsi_Cmnd *SCpnt)
|
||||||
static int nsp_nexus(Scsi_Cmnd *SCpnt)
|
static int nsp_nexus(Scsi_Cmnd *SCpnt)
|
||||||
{
|
{
|
||||||
unsigned int base = SCpnt->device->host->io_port;
|
unsigned int base = SCpnt->device->host->io_port;
|
||||||
unsigned char target = SCpnt->device->id;
|
unsigned char target = scmd_id(SCpnt);
|
||||||
// unsigned char lun = SCpnt->device->lun;
|
// unsigned char lun = SCpnt->device->lun;
|
||||||
nsp_hw_data *data = (nsp_hw_data *)SCpnt->device->host->hostdata;
|
nsp_hw_data *data = (nsp_hw_data *)SCpnt->device->host->hostdata;
|
||||||
sync_data *sync = &(data->Sync[target]);
|
sync_data *sync = &(data->Sync[target]);
|
||||||
|
|
|
@ -610,7 +610,7 @@ SYM53C500_queue(struct scsi_cmnd *SCpnt, void (*done)(struct scsi_cmnd *))
|
||||||
|
|
||||||
/* We are locked here already by the mid layer */
|
/* We are locked here already by the mid layer */
|
||||||
REG0(port_base);
|
REG0(port_base);
|
||||||
outb(SCpnt->device->id, port_base + DEST_ID); /* set destination */
|
outb(scmd_id(SCpnt), port_base + DEST_ID); /* set destination */
|
||||||
outb(FLUSH_FIFO, port_base + CMD_REG); /* reset the fifos */
|
outb(FLUSH_FIFO, port_base + CMD_REG); /* reset the fifos */
|
||||||
|
|
||||||
for (i = 0; i < SCpnt->cmd_len; i++) {
|
for (i = 0; i < SCpnt->cmd_len; i++) {
|
||||||
|
|
|
@ -740,7 +740,7 @@ static int ppa_engine(ppa_struct *dev, struct scsi_cmnd *cmd)
|
||||||
}
|
}
|
||||||
|
|
||||||
case 2: /* Phase 2 - We are now talking to the scsi bus */
|
case 2: /* Phase 2 - We are now talking to the scsi bus */
|
||||||
if (!ppa_select(dev, cmd->device->id)) {
|
if (!ppa_select(dev, scmd_id(cmd))) {
|
||||||
ppa_fail(dev, DID_NO_CONNECT);
|
ppa_fail(dev, DID_NO_CONNECT);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -659,7 +659,7 @@ static int Psi240i_BiosParam (struct scsi_device *sdev, struct block_device *dev
|
||||||
{
|
{
|
||||||
POUR_DEVICE pdev;
|
POUR_DEVICE pdev;
|
||||||
|
|
||||||
pdev = &(HOSTDATA(sdev->host)->device[sdev->id]);
|
pdev = &(HOSTDATA(sdev->host)->device[sdev_id(sdev)]);
|
||||||
|
|
||||||
geom[0] = pdev->heads;
|
geom[0] = pdev->heads;
|
||||||
geom[1] = pdev->sectors;
|
geom[1] = pdev->sectors;
|
||||||
|
|
|
@ -1,11 +1,10 @@
|
||||||
/*
|
/*
|
||||||
* QLogic ISP2100 device driver for Linux 2.6.x
|
* QLogic Fibre Channel HBA Driver
|
||||||
* Copyright (C) 2003 Christoph Hellwig.
|
* Copyright (C) 2003 Christoph Hellwig.
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation (www.qlogic.com)
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
*
|
*
|
||||||
* Released under GPL v2.
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/init.h>
|
#include <linux/init.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
|
|
|
@ -1,21 +1,9 @@
|
||||||
/******************************************************************************
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
*
|
*
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
*/
|
||||||
* (www.qlogic.com)
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*
|
|
||||||
*************************************************************************/
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Firmware Version 1.19.25 (13:12 Dec 10, 2003)
|
* Firmware Version 1.19.25 (13:12 Dec 10, 2003)
|
||||||
|
|
|
@ -1,11 +1,10 @@
|
||||||
/*
|
/*
|
||||||
* QLogic ISP2200 device driver for Linux 2.6.x
|
* QLogic Fibre Channel HBA Driver
|
||||||
* Copyright (C) 2003 Christoph Hellwig.
|
* Copyright (C) 2003 Christoph Hellwig.
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation (www.qlogic.com)
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
*
|
*
|
||||||
* Released under GPL v2.
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/init.h>
|
#include <linux/init.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
|
|
|
@ -1,21 +1,9 @@
|
||||||
/******************************************************************************
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
*
|
*
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
*/
|
||||||
* (www.qlogic.com)
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*
|
|
||||||
*************************************************************************/
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Firmware Version 2.02.08 (17:06 Mar 22, 2005)
|
* Firmware Version 2.02.08 (17:06 Mar 22, 2005)
|
||||||
|
|
|
@ -1,11 +1,10 @@
|
||||||
/*
|
/*
|
||||||
* QLogic ISP2300 device driver for Linux 2.6.x
|
* QLogic Fibre Channel HBA Driver
|
||||||
* Copyright (C) 2003 Christoph Hellwig.
|
* Copyright (C) 2003 Christoph Hellwig.
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation (www.qlogic.com)
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
*
|
*
|
||||||
* Released under GPL v2.
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/init.h>
|
#include <linux/init.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,10 +1,9 @@
|
||||||
/*
|
/*
|
||||||
* QLogic ISP2322 device driver for Linux 2.6.x
|
* QLogic Fibre Channel HBA Driver
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation (www.qlogic.com)
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
*
|
*
|
||||||
* Released under GPL v2.
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/init.h>
|
#include <linux/init.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,10 +1,9 @@
|
||||||
/*
|
/*
|
||||||
* QLogic ISP6312 device driver for Linux 2.6.x
|
* QLogic Fibre Channel HBA Driver
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation (www.qlogic.com)
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
*
|
*
|
||||||
* Released under GPL v2.
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/init.h>
|
#include <linux/init.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,20 +1,8 @@
|
||||||
/*
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
*
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
|
||||||
* (www.qlogic.com)
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*
|
*
|
||||||
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
*/
|
*/
|
||||||
#include "qla_def.h"
|
#include "qla_def.h"
|
||||||
|
|
||||||
|
@ -319,6 +307,83 @@ qla2x00_state_show(struct class_device *cdev, char *buf)
|
||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
qla2x00_zio_show(struct class_device *cdev, char *buf)
|
||||||
|
{
|
||||||
|
scsi_qla_host_t *ha = to_qla_host(class_to_shost(cdev));
|
||||||
|
int len = 0;
|
||||||
|
|
||||||
|
switch (ha->zio_mode) {
|
||||||
|
case QLA_ZIO_MODE_5:
|
||||||
|
len += snprintf(buf + len, PAGE_SIZE-len, "Mode 5\n");
|
||||||
|
break;
|
||||||
|
case QLA_ZIO_MODE_6:
|
||||||
|
len += snprintf(buf + len, PAGE_SIZE-len, "Mode 6\n");
|
||||||
|
break;
|
||||||
|
case QLA_ZIO_DISABLED:
|
||||||
|
len += snprintf(buf + len, PAGE_SIZE-len, "Disabled\n");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
qla2x00_zio_store(struct class_device *cdev, const char *buf, size_t count)
|
||||||
|
{
|
||||||
|
scsi_qla_host_t *ha = to_qla_host(class_to_shost(cdev));
|
||||||
|
int val = 0;
|
||||||
|
uint16_t zio_mode;
|
||||||
|
|
||||||
|
if (sscanf(buf, "%d", &val) != 1)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
switch (val) {
|
||||||
|
case 1:
|
||||||
|
zio_mode = QLA_ZIO_MODE_5;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
zio_mode = QLA_ZIO_MODE_6;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
zio_mode = QLA_ZIO_DISABLED;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Update per-hba values and queue a reset. */
|
||||||
|
if (zio_mode != QLA_ZIO_DISABLED || ha->zio_mode != QLA_ZIO_DISABLED) {
|
||||||
|
ha->zio_mode = zio_mode;
|
||||||
|
set_bit(ISP_ABORT_NEEDED, &ha->dpc_flags);
|
||||||
|
}
|
||||||
|
return strlen(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
qla2x00_zio_timer_show(struct class_device *cdev, char *buf)
|
||||||
|
{
|
||||||
|
scsi_qla_host_t *ha = to_qla_host(class_to_shost(cdev));
|
||||||
|
|
||||||
|
return snprintf(buf, PAGE_SIZE, "%d us\n", ha->zio_timer * 100);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ssize_t
|
||||||
|
qla2x00_zio_timer_store(struct class_device *cdev, const char *buf,
|
||||||
|
size_t count)
|
||||||
|
{
|
||||||
|
scsi_qla_host_t *ha = to_qla_host(class_to_shost(cdev));
|
||||||
|
int val = 0;
|
||||||
|
uint16_t zio_timer;
|
||||||
|
|
||||||
|
if (sscanf(buf, "%d", &val) != 1)
|
||||||
|
return -EINVAL;
|
||||||
|
if (val > 25500 || val < 100)
|
||||||
|
return -ERANGE;
|
||||||
|
|
||||||
|
zio_timer = (uint16_t)(val / 100);
|
||||||
|
ha->zio_timer = zio_timer;
|
||||||
|
|
||||||
|
return strlen(buf);
|
||||||
|
}
|
||||||
|
|
||||||
static CLASS_DEVICE_ATTR(driver_version, S_IRUGO, qla2x00_drvr_version_show,
|
static CLASS_DEVICE_ATTR(driver_version, S_IRUGO, qla2x00_drvr_version_show,
|
||||||
NULL);
|
NULL);
|
||||||
static CLASS_DEVICE_ATTR(fw_version, S_IRUGO, qla2x00_fw_version_show, NULL);
|
static CLASS_DEVICE_ATTR(fw_version, S_IRUGO, qla2x00_fw_version_show, NULL);
|
||||||
|
@ -329,6 +394,10 @@ static CLASS_DEVICE_ATTR(model_name, S_IRUGO, qla2x00_model_name_show, NULL);
|
||||||
static CLASS_DEVICE_ATTR(model_desc, S_IRUGO, qla2x00_model_desc_show, NULL);
|
static CLASS_DEVICE_ATTR(model_desc, S_IRUGO, qla2x00_model_desc_show, NULL);
|
||||||
static CLASS_DEVICE_ATTR(pci_info, S_IRUGO, qla2x00_pci_info_show, NULL);
|
static CLASS_DEVICE_ATTR(pci_info, S_IRUGO, qla2x00_pci_info_show, NULL);
|
||||||
static CLASS_DEVICE_ATTR(state, S_IRUGO, qla2x00_state_show, NULL);
|
static CLASS_DEVICE_ATTR(state, S_IRUGO, qla2x00_state_show, NULL);
|
||||||
|
static CLASS_DEVICE_ATTR(zio, S_IRUGO | S_IWUSR, qla2x00_zio_show,
|
||||||
|
qla2x00_zio_store);
|
||||||
|
static CLASS_DEVICE_ATTR(zio_timer, S_IRUGO | S_IWUSR, qla2x00_zio_timer_show,
|
||||||
|
qla2x00_zio_timer_store);
|
||||||
|
|
||||||
struct class_device_attribute *qla2x00_host_attrs[] = {
|
struct class_device_attribute *qla2x00_host_attrs[] = {
|
||||||
&class_device_attr_driver_version,
|
&class_device_attr_driver_version,
|
||||||
|
@ -340,6 +409,8 @@ struct class_device_attribute *qla2x00_host_attrs[] = {
|
||||||
&class_device_attr_model_desc,
|
&class_device_attr_model_desc,
|
||||||
&class_device_attr_pci_info,
|
&class_device_attr_pci_info,
|
||||||
&class_device_attr_state,
|
&class_device_attr_state,
|
||||||
|
&class_device_attr_zio,
|
||||||
|
&class_device_attr_zio_timer,
|
||||||
NULL,
|
NULL,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -432,6 +503,15 @@ qla2x00_set_rport_loss_tmo(struct fc_rport *rport, uint32_t timeout)
|
||||||
rport->dev_loss_tmo = ha->port_down_retry_count + 5;
|
rport->dev_loss_tmo = ha->port_down_retry_count + 5;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
qla2x00_issue_lip(struct Scsi_Host *shost)
|
||||||
|
{
|
||||||
|
scsi_qla_host_t *ha = to_qla_host(shost);
|
||||||
|
|
||||||
|
set_bit(LOOP_RESET_NEEDED, &ha->dpc_flags);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
struct fc_function_template qla2xxx_transport_functions = {
|
struct fc_function_template qla2xxx_transport_functions = {
|
||||||
|
|
||||||
.show_host_node_name = 1,
|
.show_host_node_name = 1,
|
||||||
|
@ -455,6 +535,7 @@ struct fc_function_template qla2xxx_transport_functions = {
|
||||||
.set_rport_dev_loss_tmo = qla2x00_set_rport_loss_tmo,
|
.set_rport_dev_loss_tmo = qla2x00_set_rport_loss_tmo,
|
||||||
.show_rport_dev_loss_tmo = 1,
|
.show_rport_dev_loss_tmo = 1,
|
||||||
|
|
||||||
|
.issue_fc_host_lip = qla2x00_issue_lip,
|
||||||
};
|
};
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
|
@ -1,20 +1,8 @@
|
||||||
/*
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
*
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
|
||||||
* (www.qlogic.com)
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*
|
*
|
||||||
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
*/
|
*/
|
||||||
#include "qla_def.h"
|
#include "qla_def.h"
|
||||||
|
|
||||||
|
|
|
@ -1,22 +1,9 @@
|
||||||
/******************************************************************************
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
*
|
*
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
*/
|
||||||
* (www.qlogic.com)
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*
|
|
||||||
******************************************************************************/
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Driver debug definitions.
|
* Driver debug definitions.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -1,22 +1,9 @@
|
||||||
/********************************************************************************
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
*
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
*
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
* (www.qlogic.com)
|
*/
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
**
|
|
||||||
******************************************************************************/
|
|
||||||
|
|
||||||
#ifndef __QLA_DEF_H
|
#ifndef __QLA_DEF_H
|
||||||
#define __QLA_DEF_H
|
#define __QLA_DEF_H
|
||||||
|
|
||||||
|
@ -34,6 +21,7 @@
|
||||||
#include <linux/spinlock.h>
|
#include <linux/spinlock.h>
|
||||||
#include <linux/completion.h>
|
#include <linux/completion.h>
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/workqueue.h>
|
||||||
#include <asm/semaphore.h>
|
#include <asm/semaphore.h>
|
||||||
|
|
||||||
#include <scsi/scsi.h>
|
#include <scsi/scsi.h>
|
||||||
|
@ -823,6 +811,11 @@ typedef struct {
|
||||||
#define PD_STATE_WAIT_PORT_LOGOUT_ACK 11
|
#define PD_STATE_WAIT_PORT_LOGOUT_ACK 11
|
||||||
|
|
||||||
|
|
||||||
|
#define QLA_ZIO_MODE_5 (BIT_2 | BIT_0)
|
||||||
|
#define QLA_ZIO_MODE_6 (BIT_2 | BIT_1)
|
||||||
|
#define QLA_ZIO_DISABLED 0
|
||||||
|
#define QLA_ZIO_DEFAULT_TIMER 2
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* ISP Initialization Control Block.
|
* ISP Initialization Control Block.
|
||||||
* Little endian except where noted.
|
* Little endian except where noted.
|
||||||
|
@ -1673,6 +1666,8 @@ typedef struct fc_port {
|
||||||
|
|
||||||
struct fc_rport *rport;
|
struct fc_rport *rport;
|
||||||
u32 supported_classes;
|
u32 supported_classes;
|
||||||
|
struct work_struct rport_add_work;
|
||||||
|
struct work_struct rport_del_work;
|
||||||
} fc_port_t;
|
} fc_port_t;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -2470,6 +2465,9 @@ typedef struct scsi_qla_host {
|
||||||
/* Needed for BEACON */
|
/* Needed for BEACON */
|
||||||
uint16_t beacon_blink_led;
|
uint16_t beacon_blink_led;
|
||||||
uint16_t beacon_green_on;
|
uint16_t beacon_green_on;
|
||||||
|
|
||||||
|
uint16_t zio_mode;
|
||||||
|
uint16_t zio_timer;
|
||||||
} scsi_qla_host_t;
|
} scsi_qla_host_t;
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -1,23 +1,9 @@
|
||||||
|
/*
|
||||||
/********************************************************************************
|
* QLogic Fibre Channel HBA Driver
|
||||||
* QLOGIC LINUX SOFTWARE
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
*
|
*
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
*/
|
||||||
* (www.qlogic.com)
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
**
|
|
||||||
******************************************************************************/
|
|
||||||
|
|
||||||
#ifndef __QLA_FW_H
|
#ifndef __QLA_FW_H
|
||||||
#define __QLA_FW_H
|
#define __QLA_FW_H
|
||||||
|
|
||||||
|
@ -394,7 +380,7 @@ struct cmd_type_6 {
|
||||||
|
|
||||||
uint16_t fcp_rsp_dsd_len; /* FCP_RSP DSD length. */
|
uint16_t fcp_rsp_dsd_len; /* FCP_RSP DSD length. */
|
||||||
|
|
||||||
uint8_t lun[8]; /* FCP LUN (BE). */
|
struct scsi_lun lun; /* FCP LUN (BE). */
|
||||||
|
|
||||||
uint16_t control_flags; /* Control flags. */
|
uint16_t control_flags; /* Control flags. */
|
||||||
#define CF_DATA_SEG_DESCR_ENABLE BIT_2
|
#define CF_DATA_SEG_DESCR_ENABLE BIT_2
|
||||||
|
@ -432,7 +418,7 @@ struct cmd_type_7 {
|
||||||
uint16_t dseg_count; /* Data segment count. */
|
uint16_t dseg_count; /* Data segment count. */
|
||||||
uint16_t reserved_1;
|
uint16_t reserved_1;
|
||||||
|
|
||||||
uint8_t lun[8]; /* FCP LUN (BE). */
|
struct scsi_lun lun; /* FCP LUN (BE). */
|
||||||
|
|
||||||
uint16_t task_mgmt_flags; /* Task management flags. */
|
uint16_t task_mgmt_flags; /* Task management flags. */
|
||||||
#define TMF_CLEAR_ACA BIT_14
|
#define TMF_CLEAR_ACA BIT_14
|
||||||
|
|
|
@ -1,25 +1,9 @@
|
||||||
/********************************************************************************
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
*
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
*
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
* (www.qlogic.com)
|
*/
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*
|
|
||||||
******************************************************************************
|
|
||||||
* Global include file.
|
|
||||||
******************************************************************************/
|
|
||||||
|
|
||||||
|
|
||||||
#ifndef __QLA_GBL_H
|
#ifndef __QLA_GBL_H
|
||||||
#define __QLA_GBL_H
|
#define __QLA_GBL_H
|
||||||
|
|
||||||
|
@ -76,8 +60,6 @@ extern char qla2x00_version_str[];
|
||||||
extern int ql2xlogintimeout;
|
extern int ql2xlogintimeout;
|
||||||
extern int qlport_down_retry;
|
extern int qlport_down_retry;
|
||||||
extern int ql2xplogiabsentdevice;
|
extern int ql2xplogiabsentdevice;
|
||||||
extern int ql2xenablezio;
|
|
||||||
extern int ql2xintrdelaytimer;
|
|
||||||
extern int ql2xloginretrycount;
|
extern int ql2xloginretrycount;
|
||||||
extern int ql2xfdmienable;
|
extern int ql2xfdmienable;
|
||||||
|
|
||||||
|
@ -223,6 +205,7 @@ extern irqreturn_t qla2100_intr_handler(int, void *, struct pt_regs *);
|
||||||
extern irqreturn_t qla2300_intr_handler(int, void *, struct pt_regs *);
|
extern irqreturn_t qla2300_intr_handler(int, void *, struct pt_regs *);
|
||||||
extern irqreturn_t qla24xx_intr_handler(int, void *, struct pt_regs *);
|
extern irqreturn_t qla24xx_intr_handler(int, void *, struct pt_regs *);
|
||||||
extern void qla2x00_process_response_queue(struct scsi_qla_host *);
|
extern void qla2x00_process_response_queue(struct scsi_qla_host *);
|
||||||
|
extern void qla24xx_process_response_queue(struct scsi_qla_host *);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Global Function Prototypes in qla_sup.c source file.
|
* Global Function Prototypes in qla_sup.c source file.
|
||||||
|
|
|
@ -1,20 +1,8 @@
|
||||||
/*
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
*
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
|
||||||
* (www.qlogic.com)
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*
|
*
|
||||||
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
*/
|
*/
|
||||||
#include "qla_def.h"
|
#include "qla_def.h"
|
||||||
|
|
||||||
|
|
|
@ -1,20 +1,8 @@
|
||||||
/*
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
*
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
|
||||||
* (www.qlogic.com)
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*
|
*
|
||||||
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
*/
|
*/
|
||||||
#include "qla_def.h"
|
#include "qla_def.h"
|
||||||
|
|
||||||
|
@ -1372,7 +1360,6 @@ qla2x00_nvram_config(scsi_qla_host_t *ha)
|
||||||
nvram_t *nv = (nvram_t *)ha->request_ring;
|
nvram_t *nv = (nvram_t *)ha->request_ring;
|
||||||
uint8_t *ptr = (uint8_t *)ha->request_ring;
|
uint8_t *ptr = (uint8_t *)ha->request_ring;
|
||||||
struct device_reg_2xxx __iomem *reg = &ha->iobase->isp;
|
struct device_reg_2xxx __iomem *reg = &ha->iobase->isp;
|
||||||
uint8_t timer_mode;
|
|
||||||
|
|
||||||
rval = QLA_SUCCESS;
|
rval = QLA_SUCCESS;
|
||||||
|
|
||||||
|
@ -1650,22 +1637,26 @@ qla2x00_nvram_config(scsi_qla_host_t *ha)
|
||||||
|
|
||||||
ha->flags.process_response_queue = 1;
|
ha->flags.process_response_queue = 1;
|
||||||
} else {
|
} else {
|
||||||
/* Enable ZIO -- Support mode 5 only. */
|
/* Enable ZIO. */
|
||||||
timer_mode = icb->add_firmware_options[0] &
|
if (!ha->flags.init_done) {
|
||||||
(BIT_3 | BIT_2 | BIT_1 | BIT_0);
|
ha->zio_mode = icb->add_firmware_options[0] &
|
||||||
|
(BIT_3 | BIT_2 | BIT_1 | BIT_0);
|
||||||
|
ha->zio_timer = icb->interrupt_delay_timer ?
|
||||||
|
icb->interrupt_delay_timer: 2;
|
||||||
|
}
|
||||||
icb->add_firmware_options[0] &=
|
icb->add_firmware_options[0] &=
|
||||||
~(BIT_3 | BIT_2 | BIT_1 | BIT_0);
|
~(BIT_3 | BIT_2 | BIT_1 | BIT_0);
|
||||||
if (ql2xenablezio)
|
ha->flags.process_response_queue = 0;
|
||||||
timer_mode = BIT_2 | BIT_0;
|
if (ha->zio_mode != QLA_ZIO_DISABLED) {
|
||||||
if (timer_mode == (BIT_2 | BIT_0)) {
|
DEBUG2(printk("scsi(%ld): ZIO mode %d enabled; timer "
|
||||||
DEBUG2(printk("scsi(%ld): ZIO enabled; timer delay "
|
"delay (%d us).\n", ha->host_no, ha->zio_mode,
|
||||||
"(%d).\n", ha->host_no, ql2xintrdelaytimer));
|
ha->zio_timer * 100));
|
||||||
qla_printk(KERN_INFO, ha,
|
qla_printk(KERN_INFO, ha,
|
||||||
"ZIO enabled; timer delay (%d).\n",
|
"ZIO mode %d enabled; timer delay (%d us).\n",
|
||||||
ql2xintrdelaytimer);
|
ha->zio_mode, ha->zio_timer * 100);
|
||||||
|
|
||||||
icb->add_firmware_options[0] |= timer_mode;
|
icb->add_firmware_options[0] |= (uint8_t)ha->zio_mode;
|
||||||
icb->interrupt_delay_timer = ql2xintrdelaytimer;
|
icb->interrupt_delay_timer = (uint8_t)ha->zio_timer;
|
||||||
ha->flags.process_response_queue = 1;
|
ha->flags.process_response_queue = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1677,6 +1668,24 @@ qla2x00_nvram_config(scsi_qla_host_t *ha)
|
||||||
return (rval);
|
return (rval);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
qla2x00_rport_add(void *data)
|
||||||
|
{
|
||||||
|
fc_port_t *fcport = data;
|
||||||
|
|
||||||
|
qla2x00_reg_remote_port(fcport->ha, fcport);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
qla2x00_rport_del(void *data)
|
||||||
|
{
|
||||||
|
fc_port_t *fcport = data;
|
||||||
|
|
||||||
|
if (fcport->rport)
|
||||||
|
fc_remote_port_delete(fcport->rport);
|
||||||
|
fcport->rport = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* qla2x00_alloc_fcport() - Allocate a generic fcport.
|
* qla2x00_alloc_fcport() - Allocate a generic fcport.
|
||||||
* @ha: HA context
|
* @ha: HA context
|
||||||
|
@ -1702,6 +1711,8 @@ qla2x00_alloc_fcport(scsi_qla_host_t *ha, gfp_t flags)
|
||||||
atomic_set(&fcport->state, FCS_UNCONFIGURED);
|
atomic_set(&fcport->state, FCS_UNCONFIGURED);
|
||||||
fcport->flags = FCF_RLC_SUPPORT;
|
fcport->flags = FCF_RLC_SUPPORT;
|
||||||
fcport->supported_classes = FC_COS_UNSPECIFIED;
|
fcport->supported_classes = FC_COS_UNSPECIFIED;
|
||||||
|
INIT_WORK(&fcport->rport_add_work, qla2x00_rport_add, fcport);
|
||||||
|
INIT_WORK(&fcport->rport_del_work, qla2x00_rport_del, fcport);
|
||||||
|
|
||||||
return (fcport);
|
return (fcport);
|
||||||
}
|
}
|
||||||
|
@ -2065,8 +2076,8 @@ qla2x00_reg_remote_port(scsi_qla_host_t *ha, fc_port_t *fcport)
|
||||||
struct fc_rport *rport;
|
struct fc_rport *rport;
|
||||||
|
|
||||||
if (fcport->rport) {
|
if (fcport->rport) {
|
||||||
fc_remote_port_unblock(fcport->rport);
|
fc_remote_port_delete(fcport->rport);
|
||||||
return;
|
fcport->rport = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
rport_ids.node_name = wwn_to_u64(fcport->node_name);
|
rport_ids.node_name = wwn_to_u64(fcport->node_name);
|
||||||
|
@ -2080,7 +2091,7 @@ qla2x00_reg_remote_port(scsi_qla_host_t *ha, fc_port_t *fcport)
|
||||||
"Unable to allocate fc remote port!\n");
|
"Unable to allocate fc remote port!\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
rport->dd_data = fcport;
|
*((fc_port_t **)rport->dd_data) = fcport;
|
||||||
rport->supported_classes = fcport->supported_classes;
|
rport->supported_classes = fcport->supported_classes;
|
||||||
|
|
||||||
rport_ids.roles = FC_RPORT_ROLE_UNKNOWN;
|
rport_ids.roles = FC_RPORT_ROLE_UNKNOWN;
|
||||||
|
@ -2858,7 +2869,7 @@ qla2x00_fabric_login(scsi_qla_host_t *ha, fc_port_t *fcport,
|
||||||
fcport->d_id.b.domain, fcport->d_id.b.area,
|
fcport->d_id.b.domain, fcport->d_id.b.area,
|
||||||
fcport->d_id.b.al_pa);
|
fcport->d_id.b.al_pa);
|
||||||
fcport->loop_id = FC_NO_LOOP_ID;
|
fcport->loop_id = FC_NO_LOOP_ID;
|
||||||
atomic_set(&fcport->state, FCS_DEVICE_DEAD);
|
fcport->login_retry = 0;
|
||||||
|
|
||||||
rval = 3;
|
rval = 3;
|
||||||
break;
|
break;
|
||||||
|
@ -3442,6 +3453,30 @@ qla24xx_nvram_config(scsi_qla_host_t *ha)
|
||||||
if (ql2xloginretrycount)
|
if (ql2xloginretrycount)
|
||||||
ha->login_retry_count = ql2xloginretrycount;
|
ha->login_retry_count = ql2xloginretrycount;
|
||||||
|
|
||||||
|
/* Enable ZIO. */
|
||||||
|
if (!ha->flags.init_done) {
|
||||||
|
ha->zio_mode = le32_to_cpu(icb->firmware_options_2) &
|
||||||
|
(BIT_3 | BIT_2 | BIT_1 | BIT_0);
|
||||||
|
ha->zio_timer = le16_to_cpu(icb->interrupt_delay_timer) ?
|
||||||
|
le16_to_cpu(icb->interrupt_delay_timer): 2;
|
||||||
|
}
|
||||||
|
icb->firmware_options_2 &= __constant_cpu_to_le32(
|
||||||
|
~(BIT_3 | BIT_2 | BIT_1 | BIT_0));
|
||||||
|
ha->flags.process_response_queue = 0;
|
||||||
|
if (ha->zio_mode != QLA_ZIO_DISABLED) {
|
||||||
|
DEBUG2(printk("scsi(%ld): ZIO mode %d enabled; timer delay "
|
||||||
|
"(%d us).\n", ha->host_no, ha->zio_mode,
|
||||||
|
ha->zio_timer * 100));
|
||||||
|
qla_printk(KERN_INFO, ha,
|
||||||
|
"ZIO mode %d enabled; timer delay (%d us).\n",
|
||||||
|
ha->zio_mode, ha->zio_timer * 100);
|
||||||
|
|
||||||
|
icb->firmware_options_2 |= cpu_to_le32(
|
||||||
|
(uint32_t)ha->zio_mode);
|
||||||
|
icb->interrupt_delay_timer = cpu_to_le16(ha->zio_timer);
|
||||||
|
ha->flags.process_response_queue = 1;
|
||||||
|
}
|
||||||
|
|
||||||
if (rval) {
|
if (rval) {
|
||||||
DEBUG2_3(printk(KERN_WARNING
|
DEBUG2_3(printk(KERN_WARNING
|
||||||
"scsi(%ld): NVRAM configuration failed!\n", ha->host_no));
|
"scsi(%ld): NVRAM configuration failed!\n", ha->host_no));
|
||||||
|
|
|
@ -1,23 +1,10 @@
|
||||||
/*
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
*
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
|
||||||
* (www.qlogic.com)
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*
|
*
|
||||||
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
static __inline__ uint16_t qla2x00_debounce_register(volatile uint16_t __iomem *);
|
static __inline__ uint16_t qla2x00_debounce_register(volatile uint16_t __iomem *);
|
||||||
/*
|
/*
|
||||||
* qla2x00_debounce_register
|
* qla2x00_debounce_register
|
||||||
|
|
|
@ -1,22 +1,9 @@
|
||||||
/******************************************************************************
|
/*
|
||||||
* QLOGIC LINUX SOFTWARE
|
* QLogic Fibre Channel HBA Driver
|
||||||
|
* Copyright (c) 2003-2005 QLogic Corporation
|
||||||
*
|
*
|
||||||
* QLogic ISP2x00 device driver for Linux 2.6.x
|
* See LICENSE.qla2xxx for copyright and licensing details.
|
||||||
* Copyright (C) 2003-2005 QLogic Corporation
|
*/
|
||||||
* (www.qlogic.com)
|
|
||||||
*
|
|
||||||
* 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; either version 2, or (at your option) any
|
|
||||||
* later version.
|
|
||||||
*
|
|
||||||
* 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.
|
|
||||||
*
|
|
||||||
******************************************************************************/
|
|
||||||
|
|
||||||
#include "qla_def.h"
|
#include "qla_def.h"
|
||||||
|
|
||||||
#include <linux/blkdev.h>
|
#include <linux/blkdev.h>
|
||||||
|
@ -316,7 +303,6 @@ qla2x00_start_scsi(srb_t *sp)
|
||||||
uint16_t req_cnt;
|
uint16_t req_cnt;
|
||||||
uint16_t tot_dsds;
|
uint16_t tot_dsds;
|
||||||
struct device_reg_2xxx __iomem *reg;
|
struct device_reg_2xxx __iomem *reg;
|
||||||
char tag[2];
|
|
||||||
|
|
||||||
/* Setup device pointers. */
|
/* Setup device pointers. */
|
||||||
ret = 0;
|
ret = 0;
|
||||||
|
@ -401,18 +387,6 @@ qla2x00_start_scsi(srb_t *sp)
|
||||||
|
|
||||||
/* Update tagged queuing modifier */
|
/* Update tagged queuing modifier */
|
||||||
cmd_pkt->control_flags = __constant_cpu_to_le16(CF_SIMPLE_TAG);
|
cmd_pkt->control_flags = __constant_cpu_to_le16(CF_SIMPLE_TAG);
|
||||||
if (scsi_populate_tag_msg(cmd, tag)) {
|
|
||||||
switch (tag[0]) {
|
|
||||||
case MSG_HEAD_TAG:
|
|
||||||
cmd_pkt->control_flags =
|
|
||||||
__constant_cpu_to_le16(CF_HEAD_TAG);
|
|
||||||
break;
|
|
||||||
case MSG_ORDERED_TAG:
|
|
||||||
cmd_pkt->control_flags =
|
|
||||||
__constant_cpu_to_le16(CF_ORDERED_TAG);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Load SCSI command packet. */
|
/* Load SCSI command packet. */
|
||||||
memcpy(cmd_pkt->scsi_cdb, cmd->cmnd, cmd->cmd_len);
|
memcpy(cmd_pkt->scsi_cdb, cmd->cmnd, cmd->cmd_len);
|
||||||
|
@ -440,6 +414,11 @@ qla2x00_start_scsi(srb_t *sp)
|
||||||
WRT_REG_WORD(ISP_REQ_Q_IN(ha, reg), ha->req_ring_index);
|
WRT_REG_WORD(ISP_REQ_Q_IN(ha, reg), ha->req_ring_index);
|
||||||
RD_REG_WORD_RELAXED(ISP_REQ_Q_IN(ha, reg)); /* PCI Posting. */
|
RD_REG_WORD_RELAXED(ISP_REQ_Q_IN(ha, reg)); /* PCI Posting. */
|
||||||
|
|
||||||
|
/* Manage unprocessed RIO/ZIO commands in response queue. */
|
||||||
|
if (ha->flags.process_response_queue &&
|
||||||
|
ha->response_ring_ptr->signature != RESPONSE_PROCESSED)
|
||||||
|
qla2x00_process_response_queue(ha);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&ha->hardware_lock, flags);
|
spin_unlock_irqrestore(&ha->hardware_lock, flags);
|
||||||
return (QLA_SUCCESS);
|
return (QLA_SUCCESS);
|
||||||
|
|
||||||
|
@ -749,7 +728,6 @@ qla24xx_start_scsi(srb_t *sp)
|
||||||
uint16_t req_cnt;
|
uint16_t req_cnt;
|
||||||
uint16_t tot_dsds;
|
uint16_t tot_dsds;
|
||||||
struct device_reg_24xx __iomem *reg;
|
struct device_reg_24xx __iomem *reg;
|
||||||
char tag[2];
|
|
||||||
|
|
||||||
/* Setup device pointers. */
|
/* Setup device pointers. */
|
||||||
ret = 0;
|
ret = 0;
|
||||||
|
@ -824,6 +802,7 @@ qla24xx_start_scsi(srb_t *sp)
|
||||||
cmd_pkt->handle = handle;
|
cmd_pkt->handle = handle;
|
||||||
|
|
||||||
/* Zero out remaining portion of packet. */
|
/* Zero out remaining portion of packet. */
|
||||||
|
/* tagged queuing modifier -- default is TSK_SIMPLE (0). */
|
||||||
clr_ptr = (uint32_t *)cmd_pkt + 2;
|
clr_ptr = (uint32_t *)cmd_pkt + 2;
|
||||||
memset(clr_ptr, 0, REQUEST_ENTRY_SIZE - 8);
|
memset(clr_ptr, 0, REQUEST_ENTRY_SIZE - 8);
|
||||||
cmd_pkt->dseg_count = cpu_to_le16(tot_dsds);
|
cmd_pkt->dseg_count = cpu_to_le16(tot_dsds);
|
||||||
|
@ -834,20 +813,7 @@ qla24xx_start_scsi(srb_t *sp)
|
||||||
cmd_pkt->port_id[1] = sp->fcport->d_id.b.area;
|
cmd_pkt->port_id[1] = sp->fcport->d_id.b.area;
|
||||||
cmd_pkt->port_id[2] = sp->fcport->d_id.b.domain;
|
cmd_pkt->port_id[2] = sp->fcport->d_id.b.domain;
|
||||||
|
|
||||||
cmd_pkt->lun[1] = LSB(sp->cmd->device->lun);
|
int_to_scsilun(sp->cmd->device->lun, &cmd_pkt->lun);
|
||||||
cmd_pkt->lun[2] = MSB(sp->cmd->device->lun);
|
|
||||||
|
|
||||||
/* Update tagged queuing modifier -- default is TSK_SIMPLE (0). */
|
|
||||||
if (scsi_populate_tag_msg(cmd, tag)) {
|
|
||||||
switch (tag[0]) {
|
|
||||||
case MSG_HEAD_TAG:
|
|
||||||
cmd_pkt->task = TSK_HEAD_OF_QUEUE;
|
|
||||||
break;
|
|
||||||
case MSG_ORDERED_TAG:
|
|
||||||
cmd_pkt->task = TSK_ORDERED;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Load SCSI command packet. */
|
/* Load SCSI command packet. */
|
||||||
memcpy(cmd_pkt->fcp_cdb, cmd->cmnd, cmd->cmd_len);
|
memcpy(cmd_pkt->fcp_cdb, cmd->cmnd, cmd->cmd_len);
|
||||||
|
@ -877,6 +843,11 @@ qla24xx_start_scsi(srb_t *sp)
|
||||||
WRT_REG_DWORD(®->req_q_in, ha->req_ring_index);
|
WRT_REG_DWORD(®->req_q_in, ha->req_ring_index);
|
||||||
RD_REG_DWORD_RELAXED(®->req_q_in); /* PCI Posting. */
|
RD_REG_DWORD_RELAXED(®->req_q_in); /* PCI Posting. */
|
||||||
|
|
||||||
|
/* Manage unprocessed RIO/ZIO commands in response queue. */
|
||||||
|
if (ha->flags.process_response_queue &&
|
||||||
|
ha->response_ring_ptr->signature != RESPONSE_PROCESSED)
|
||||||
|
qla24xx_process_response_queue(ha);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&ha->hardware_lock, flags);
|
spin_unlock_irqrestore(&ha->hardware_lock, flags);
|
||||||
return QLA_SUCCESS;
|
return QLA_SUCCESS;
|
||||||
|
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue