isci: bypass scic_controller_get_handler_methods()
The indirection is unecessary and broken in the current case that assigns the handlers based on a not up-to-date pdev->msix_enabled value. Route the handlers directly to the requisite core routines. Todo: hook up error interrupt handling Reported-by: Jeff Garzik <jeff@garzik.org> Cc: Christoph Hellwig <hch@infradead.org> Signed-off-by: Edmund Nadolski <edmund.nadolski@intel.com> Signed-off-by: Dan Williams <dan.j.williams@intel.com>
This commit is contained in:
@@ -1898,8 +1898,7 @@ static void scic_sds_controller_single_vector_completion_handler(
|
|||||||
*
|
*
|
||||||
* bool true if an interrupt is processed false if no interrupt was processed
|
* bool true if an interrupt is processed false if no interrupt was processed
|
||||||
*/
|
*/
|
||||||
static bool scic_sds_controller_normal_vector_interrupt_handler(
|
bool scic_sds_controller_isr(struct scic_sds_controller *scic)
|
||||||
struct scic_sds_controller *scic)
|
|
||||||
{
|
{
|
||||||
if (scic_sds_controller_completion_queue_has_entries(scic)) {
|
if (scic_sds_controller_completion_queue_has_entries(scic)) {
|
||||||
return true;
|
return true;
|
||||||
@@ -1925,8 +1924,7 @@ static bool scic_sds_controller_normal_vector_interrupt_handler(
|
|||||||
* This is the method provided to handle the completions for a normal MSIX
|
* This is the method provided to handle the completions for a normal MSIX
|
||||||
* message.
|
* message.
|
||||||
*/
|
*/
|
||||||
static void scic_sds_controller_normal_vector_completion_handler(
|
void scic_sds_controller_completion_handler(struct scic_sds_controller *scic)
|
||||||
struct scic_sds_controller *scic)
|
|
||||||
{
|
{
|
||||||
/* Empty out the completion queue */
|
/* Empty out the completion queue */
|
||||||
if (scic_sds_controller_completion_queue_has_entries(scic))
|
if (scic_sds_controller_completion_queue_has_entries(scic))
|
||||||
@@ -2582,9 +2580,9 @@ enum sci_status scic_controller_get_handler_methods(
|
|||||||
status = SCI_SUCCESS;
|
status = SCI_SUCCESS;
|
||||||
} else if (message_count == 2) {
|
} else if (message_count == 2) {
|
||||||
handler_methods[0].interrupt_handler
|
handler_methods[0].interrupt_handler
|
||||||
= scic_sds_controller_normal_vector_interrupt_handler;
|
= scic_sds_controller_isr;
|
||||||
handler_methods[0].completion_handler
|
handler_methods[0].completion_handler
|
||||||
= scic_sds_controller_normal_vector_completion_handler;
|
= scic_sds_controller_completion_handler;
|
||||||
|
|
||||||
handler_methods[1].interrupt_handler
|
handler_methods[1].interrupt_handler
|
||||||
= scic_sds_controller_error_vector_interrupt_handler;
|
= scic_sds_controller_error_vector_interrupt_handler;
|
||||||
|
@@ -62,80 +62,49 @@
|
|||||||
#include "request.h"
|
#include "request.h"
|
||||||
#include "host.h"
|
#include "host.h"
|
||||||
|
|
||||||
/**
|
irqreturn_t isci_msix_isr(int vec, void *data)
|
||||||
* isci_isr() - This function is the interrupt service routine for the
|
|
||||||
* controller. It schedules the tasklet and returns.
|
|
||||||
* @vec: This parameter specifies the interrupt vector.
|
|
||||||
* @data: This parameter specifies the ISCI host object.
|
|
||||||
*
|
|
||||||
* IRQ_HANDLED if out interrupt otherwise, IRQ_NONE
|
|
||||||
*/
|
|
||||||
irqreturn_t isci_isr(int vec, void *data)
|
|
||||||
{
|
{
|
||||||
struct isci_host *isci_host
|
struct isci_host *ihost = data;
|
||||||
= (struct isci_host *)data;
|
struct scic_sds_controller *scic = ihost->core_controller;
|
||||||
struct scic_controller_handler_methods *handlers
|
|
||||||
= &isci_host->scic_irq_handlers[SCI_MSIX_NORMAL_VECTOR];
|
|
||||||
irqreturn_t ret = IRQ_NONE;
|
|
||||||
|
|
||||||
if (isci_host_get_state(isci_host) != isci_starting
|
if (isci_host_get_state(ihost) != isci_starting) {
|
||||||
&& handlers->interrupt_handler) {
|
if (scic_sds_controller_isr(scic)) {
|
||||||
|
if (isci_host_get_state(ihost) != isci_stopped)
|
||||||
if (handlers->interrupt_handler(isci_host->core_controller)) {
|
tasklet_schedule(&ihost->completion_tasklet);
|
||||||
if (isci_host_get_state(isci_host) != isci_stopped) {
|
else
|
||||||
tasklet_schedule(
|
dev_dbg(&ihost->pdev->dev,
|
||||||
&isci_host->completion_tasklet);
|
"%s: controller stopped\n", __func__);
|
||||||
} else
|
}
|
||||||
dev_dbg(&isci_host->pdev->dev,
|
|
||||||
"%s: controller stopped\n",
|
|
||||||
__func__);
|
|
||||||
ret = IRQ_HANDLED;
|
|
||||||
}
|
}
|
||||||
} else
|
|
||||||
dev_warn(&isci_host->pdev->dev,
|
|
||||||
"%s: get_handler_methods failed, "
|
|
||||||
"isci_host->status = 0x%x\n",
|
|
||||||
__func__,
|
|
||||||
isci_host_get_state(isci_host));
|
|
||||||
|
|
||||||
return ret;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
|
||||||
irqreturn_t isci_legacy_isr(int vec, void *data)
|
irqreturn_t isci_intx_isr(int vec, void *data)
|
||||||
{
|
{
|
||||||
struct pci_dev *pdev = data;
|
struct pci_dev *pdev = data;
|
||||||
struct isci_host *isci_host;
|
struct isci_host *ihost;
|
||||||
struct scic_controller_handler_methods *handlers;
|
|
||||||
irqreturn_t ret = IRQ_NONE;
|
irqreturn_t ret = IRQ_NONE;
|
||||||
|
|
||||||
/*
|
for_each_isci_host(ihost, pdev) {
|
||||||
* Since this is a legacy interrupt, either or both
|
struct scic_sds_controller *scic = ihost->core_controller;
|
||||||
* controllers could have triggered it. Thus, we have to call
|
|
||||||
* the legacy interrupt handler for all controllers on the
|
|
||||||
* PCI function.
|
|
||||||
*/
|
|
||||||
for_each_isci_host(isci_host, pdev) {
|
|
||||||
handlers = &isci_host->scic_irq_handlers[SCI_MSIX_NORMAL_VECTOR];
|
|
||||||
|
|
||||||
if (isci_host_get_state(isci_host) != isci_starting
|
if (isci_host_get_state(ihost) != isci_starting) {
|
||||||
&& handlers->interrupt_handler) {
|
if (scic_sds_controller_isr(scic)) {
|
||||||
|
if (isci_host_get_state(ihost) != isci_stopped)
|
||||||
if (handlers->interrupt_handler(isci_host->core_controller)) {
|
tasklet_schedule(&ihost->completion_tasklet);
|
||||||
if (isci_host_get_state(isci_host) != isci_stopped) {
|
else
|
||||||
tasklet_schedule(
|
dev_dbg(&ihost->pdev->dev,
|
||||||
&isci_host->completion_tasklet);
|
|
||||||
} else
|
|
||||||
dev_dbg(&isci_host->pdev->dev,
|
|
||||||
"%s: controller stopped\n",
|
"%s: controller stopped\n",
|
||||||
__func__);
|
__func__);
|
||||||
ret = IRQ_HANDLED;
|
ret = IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
} else
|
} else
|
||||||
dev_warn(&isci_host->pdev->dev,
|
dev_warn(&ihost->pdev->dev,
|
||||||
"%s: get_handler_methods failed, "
|
"%s: get_handler_methods failed, "
|
||||||
"isci_host->status = 0x%x\n",
|
"ihost->status = 0x%x\n",
|
||||||
__func__,
|
__func__,
|
||||||
isci_host_get_state(isci_host));
|
isci_host_get_state(ihost));
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@@ -166,34 +135,9 @@ void isci_host_start_complete(
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int isci_host_scan_finished(struct Scsi_Host *shost, unsigned long time)
|
||||||
|
|
||||||
/**
|
|
||||||
* isci_host_scan_finished() - This function is one of the SCSI Host Template
|
|
||||||
* functions. The SCSI midlayer calls this function during a target scan,
|
|
||||||
* approx. once every 10 millisecs.
|
|
||||||
* @shost: This parameter specifies the SCSI host being scanned
|
|
||||||
* @time: This parameter specifies the number of ticks since the scan started.
|
|
||||||
*
|
|
||||||
* scan status, zero indicates the SCSI midlayer should continue to poll,
|
|
||||||
* otherwise assume controller is ready.
|
|
||||||
*/
|
|
||||||
int isci_host_scan_finished(
|
|
||||||
struct Scsi_Host *shost,
|
|
||||||
unsigned long time)
|
|
||||||
{
|
{
|
||||||
struct isci_host *isci_host
|
struct isci_host *isci_host = isci_host_from_sas_ha(SHOST_TO_SAS_HA(shost));
|
||||||
= isci_host_from_sas_ha(SHOST_TO_SAS_HA(shost));
|
|
||||||
|
|
||||||
struct scic_controller_handler_methods *handlers
|
|
||||||
= &isci_host->scic_irq_handlers[SCI_MSIX_NORMAL_VECTOR];
|
|
||||||
|
|
||||||
if (handlers->interrupt_handler == NULL) {
|
|
||||||
dev_err(&isci_host->pdev->dev,
|
|
||||||
"%s: scic_controller_get_handler_methods failed\n",
|
|
||||||
__func__);
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* check interrupt_handler's status and call completion_handler if true,
|
* check interrupt_handler's status and call completion_handler if true,
|
||||||
@@ -204,8 +148,8 @@ int isci_host_scan_finished(
|
|||||||
* continue to return zero from thee scan_finished routine until
|
* continue to return zero from thee scan_finished routine until
|
||||||
* the scic_cb_controller_start_complete() call comes from the core.
|
* the scic_cb_controller_start_complete() call comes from the core.
|
||||||
**/
|
**/
|
||||||
if (handlers->interrupt_handler(isci_host->core_controller))
|
if (scic_sds_controller_isr(isci_host->core_controller))
|
||||||
handlers->completion_handler(isci_host->core_controller);
|
scic_sds_controller_completion_handler(isci_host->core_controller);
|
||||||
|
|
||||||
if (isci_starting == isci_host_get_state(isci_host)
|
if (isci_starting == isci_host_get_state(isci_host)
|
||||||
&& time < (HZ * 10)) {
|
&& time < (HZ * 10)) {
|
||||||
@@ -363,8 +307,6 @@ static int isci_host_mdl_allocate_coherent(
|
|||||||
static void isci_host_completion_routine(unsigned long data)
|
static void isci_host_completion_routine(unsigned long data)
|
||||||
{
|
{
|
||||||
struct isci_host *isci_host = (struct isci_host *)data;
|
struct isci_host *isci_host = (struct isci_host *)data;
|
||||||
struct scic_controller_handler_methods *handlers
|
|
||||||
= &isci_host->scic_irq_handlers[SCI_MSIX_NORMAL_VECTOR];
|
|
||||||
struct list_head completed_request_list;
|
struct list_head completed_request_list;
|
||||||
struct list_head aborted_request_list;
|
struct list_head aborted_request_list;
|
||||||
struct list_head *current_position;
|
struct list_head *current_position;
|
||||||
@@ -378,11 +320,8 @@ static void isci_host_completion_routine(unsigned long data)
|
|||||||
|
|
||||||
spin_lock_irq(&isci_host->scic_lock);
|
spin_lock_irq(&isci_host->scic_lock);
|
||||||
|
|
||||||
if (handlers->completion_handler) {
|
scic_sds_controller_completion_handler(isci_host->core_controller);
|
||||||
handlers->completion_handler(
|
|
||||||
isci_host->core_controller
|
|
||||||
);
|
|
||||||
}
|
|
||||||
/* Take the lists of completed I/Os from the host. */
|
/* Take the lists of completed I/Os from the host. */
|
||||||
list_splice_init(&isci_host->requests_to_complete,
|
list_splice_init(&isci_host->requests_to_complete,
|
||||||
&completed_request_list);
|
&completed_request_list);
|
||||||
@@ -544,8 +483,6 @@ int isci_host_init(struct isci_host *isci_host)
|
|||||||
enum sci_status status;
|
enum sci_status status;
|
||||||
struct scic_sds_controller *controller;
|
struct scic_sds_controller *controller;
|
||||||
struct scic_sds_port *scic_port;
|
struct scic_sds_port *scic_port;
|
||||||
struct scic_controller_handler_methods *handlers
|
|
||||||
= &isci_host->scic_irq_handlers[0];
|
|
||||||
union scic_oem_parameters scic_oem_params;
|
union scic_oem_parameters scic_oem_params;
|
||||||
union scic_user_parameters scic_user_params;
|
union scic_user_parameters scic_user_params;
|
||||||
const struct firmware *fw = NULL;
|
const struct firmware *fw = NULL;
|
||||||
@@ -691,35 +628,8 @@ int isci_host_init(struct isci_host *isci_host)
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* @todo: use both MSI-X interrupts, and don't do indirect
|
|
||||||
* calls to the handlers just register direct calls
|
|
||||||
*/
|
|
||||||
if (isci_host->pdev->msix_enabled) {
|
|
||||||
status = scic_controller_get_handler_methods(
|
|
||||||
SCIC_MSIX_INTERRUPT_TYPE,
|
|
||||||
SCI_MSIX_DOUBLE_VECTOR,
|
|
||||||
handlers
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
status = scic_controller_get_handler_methods(
|
|
||||||
SCIC_LEGACY_LINE_INTERRUPT_TYPE,
|
|
||||||
0,
|
|
||||||
handlers
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (status != SCI_SUCCESS) {
|
|
||||||
handlers->interrupt_handler = NULL;
|
|
||||||
handlers->completion_handler = NULL;
|
|
||||||
dev_err(&isci_host->pdev->dev,
|
|
||||||
"%s: scic_controller_get_handler_methods failed\n",
|
|
||||||
__func__);
|
|
||||||
}
|
|
||||||
|
|
||||||
tasklet_init(&isci_host->completion_tasklet,
|
tasklet_init(&isci_host->completion_tasklet,
|
||||||
isci_host_completion_routine,
|
isci_host_completion_routine, (unsigned long)isci_host);
|
||||||
(unsigned long)isci_host
|
|
||||||
);
|
|
||||||
|
|
||||||
INIT_LIST_HEAD(&(isci_host->mdl_struct_list));
|
INIT_LIST_HEAD(&(isci_host->mdl_struct_list));
|
||||||
|
|
||||||
|
@@ -53,13 +53,6 @@
|
|||||||
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
|
||||||
* This file contains the isci_module initialization routines.
|
|
||||||
*
|
|
||||||
* host.h
|
|
||||||
*/
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#if !defined(_SCI_HOST_H_)
|
#if !defined(_SCI_HOST_H_)
|
||||||
#define _SCI_HOST_H_
|
#define _SCI_HOST_H_
|
||||||
@@ -79,10 +72,6 @@
|
|||||||
#define SCI_SCU_BAR_SIZE (4*1024*1024)
|
#define SCI_SCU_BAR_SIZE (4*1024*1024)
|
||||||
#define SCI_IO_SPACE_BAR0 2
|
#define SCI_IO_SPACE_BAR0 2
|
||||||
#define SCI_IO_SPACE_BAR1 3
|
#define SCI_IO_SPACE_BAR1 3
|
||||||
#define SCI_MSIX_NORMAL_VECTOR 0
|
|
||||||
#define SCI_MSIX_ERROR_VECTOR 1
|
|
||||||
#define SCI_MSIX_SINGLE_VECTOR 1
|
|
||||||
#define SCI_MSIX_DOUBLE_VECTOR 2
|
|
||||||
#define ISCI_CAN_QUEUE_VAL 250 /* < SCI_MAX_IO_REQUESTS ? */
|
#define ISCI_CAN_QUEUE_VAL 250 /* < SCI_MAX_IO_REQUESTS ? */
|
||||||
#define SCIC_CONTROLLER_STOP_TIMEOUT 5000
|
#define SCIC_CONTROLLER_STOP_TIMEOUT 5000
|
||||||
|
|
||||||
@@ -96,7 +85,6 @@ struct coherent_memory_info {
|
|||||||
|
|
||||||
struct isci_host {
|
struct isci_host {
|
||||||
struct scic_sds_controller *core_controller;
|
struct scic_sds_controller *core_controller;
|
||||||
struct scic_controller_handler_methods scic_irq_handlers[SCI_NUM_MSI_X_INT];
|
|
||||||
union scic_oem_parameters oem_parameters;
|
union scic_oem_parameters oem_parameters;
|
||||||
|
|
||||||
int id; /* unique within a given pci device */
|
int id; /* unique within a given pci device */
|
||||||
|
@@ -334,7 +334,7 @@ static int isci_setup_interrupts(struct pci_dev *pdev)
|
|||||||
BUG_ON(!isci_host);
|
BUG_ON(!isci_host);
|
||||||
|
|
||||||
/* @todo: need to handle error case. */
|
/* @todo: need to handle error case. */
|
||||||
err = devm_request_irq(&pdev->dev, msix->vector, isci_isr, 0,
|
err = devm_request_irq(&pdev->dev, msix->vector, isci_msix_isr, 0,
|
||||||
DRV_NAME"-msix", isci_host);
|
DRV_NAME"-msix", isci_host);
|
||||||
if (!err)
|
if (!err)
|
||||||
continue;
|
continue;
|
||||||
@@ -353,7 +353,7 @@ static int isci_setup_interrupts(struct pci_dev *pdev)
|
|||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
intx:
|
intx:
|
||||||
err = devm_request_irq(&pdev->dev, pdev->irq, isci_legacy_isr,
|
err = devm_request_irq(&pdev->dev, pdev->irq, isci_intx_isr,
|
||||||
IRQF_SHARED, DRV_NAME"-intx", pdev);
|
IRQF_SHARED, DRV_NAME"-intx", pdev);
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
|
@@ -113,8 +113,11 @@ struct isci_firmware {
|
|||||||
u8 sas_addrs_size;
|
u8 sas_addrs_size;
|
||||||
};
|
};
|
||||||
|
|
||||||
irqreturn_t isci_isr(int vec, void *data);
|
irqreturn_t isci_msix_isr(int vec, void *data);
|
||||||
irqreturn_t isci_legacy_isr(int vec, void *data);
|
irqreturn_t isci_intx_isr(int vec, void *data);
|
||||||
|
|
||||||
|
bool scic_sds_controller_isr(struct scic_sds_controller *scic);
|
||||||
|
void scic_sds_controller_completion_handler(struct scic_sds_controller *scic);
|
||||||
|
|
||||||
enum sci_status isci_parse_oem_parameters(
|
enum sci_status isci_parse_oem_parameters(
|
||||||
union scic_oem_parameters *oem_params,
|
union scic_oem_parameters *oem_params,
|
||||||
|
Reference in New Issue
Block a user