2019-05-19 15:51:48 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2005-04-16 15:20:36 -07:00
/*
* Adaptec AAC series RAID controller driver
2008-10-27 15:16:36 +00:00
* ( c ) Copyright 2001 Red Hat Inc .
2005-04-16 15:20:36 -07:00
*
* based on the old aacraid driver that is . .
* Adaptec aacraid device driver for Linux .
*
2011-03-17 02:10:32 -07:00
* Copyright ( c ) 2000 - 2010 Adaptec , Inc .
2017-02-02 15:53:36 -08:00
* 2010 - 2015 PMC - Sierra , Inc . ( aacraid @ pmc - sierra . com )
* 2016 - 2017 Microsemi Corp . ( aacraid @ microsemi . com )
2005-04-16 15:20:36 -07:00
*
* Module Name :
* commsup . c
*
* Abstract : Contain all routines that are required for FSA host / adapter
2005-05-16 18:28:42 -07:00
* communication .
2005-04-16 15:20:36 -07:00
*/
# include <linux/kernel.h>
# include <linux/init.h>
2017-12-26 20:34:48 -08:00
# include <linux/crash_dump.h>
2005-04-16 15:20:36 -07:00
# include <linux/types.h>
# include <linux/sched.h>
# include <linux/pci.h>
# include <linux/spinlock.h>
# include <linux/slab.h>
# include <linux/completion.h>
# include <linux/blkdev.h>
2005-11-30 23:47:05 -05:00
# include <linux/delay.h>
2006-02-14 18:45:06 +01:00
# include <linux/kthread.h>
2006-09-24 23:45:29 +01:00
# include <linux/interrupt.h>
2017-02-02 15:53:25 -08:00
# include <linux/bcd.h>
2006-08-03 08:03:30 -07:00
# include <scsi/scsi.h>
2005-05-16 18:28:42 -07:00
# include <scsi/scsi_host.h>
2005-09-26 13:04:56 -07:00
# include <scsi/scsi_device.h>
2006-08-03 08:03:30 -07:00
# include <scsi/scsi_cmnd.h>
2005-04-16 15:20:36 -07:00
# include "aacraid.h"
/**
* fib_map_alloc - allocate the fib objects
* @ dev : Adapter to allocate for
*
* Allocate and map the shared PCI space for the FIB blocks used to
* talk to the Adaptec firmware .
*/
2008-01-16 07:39:06 -08:00
2005-04-16 15:20:36 -07:00
static int fib_map_alloc ( struct aac_dev * dev )
{
2017-02-02 15:53:22 -08:00
if ( dev - > max_fib_size > AAC_MAX_NATIVE_SIZE )
dev - > max_cmd_size = AAC_MAX_NATIVE_SIZE ;
else
dev - > max_cmd_size = dev - > max_fib_size ;
2017-02-02 15:53:29 -08:00
if ( dev - > max_fib_size < AAC_MAX_NATIVE_SIZE ) {
dev - > max_cmd_size = AAC_MAX_NATIVE_SIZE ;
} else {
dev - > max_cmd_size = dev - > max_fib_size ;
}
2017-02-02 15:53:22 -08:00
2005-05-16 18:28:42 -07:00
dprintk ( ( KERN_INFO
2017-04-05 16:14:16 +05:30
" allocate hardware fibs dma_alloc_coherent(%p, %d * (%d + %d), %p) \n " ,
& dev - > pdev - > dev , dev - > max_cmd_size , dev - > scsi_host_ptr - > can_queue ,
2005-05-16 18:28:42 -07:00
AAC_NUM_MGT_FIB , & dev - > hw_fib_pa ) ) ;
2017-04-05 16:14:16 +05:30
dev - > hw_fib_va = dma_alloc_coherent ( & dev - > pdev - > dev ,
2017-02-02 15:53:22 -08:00
( dev - > max_cmd_size + sizeof ( struct aac_fib_xporthdr ) )
2011-03-17 02:10:32 -07:00
* ( dev - > scsi_host_ptr - > can_queue + AAC_NUM_MGT_FIB ) + ( ALIGN32 - 1 ) ,
2017-04-05 16:14:16 +05:30
& dev - > hw_fib_pa , GFP_KERNEL ) ;
2011-03-17 02:10:32 -07:00
if ( dev - > hw_fib_va = = NULL )
2005-04-16 15:20:36 -07:00
return - ENOMEM ;
return 0 ;
}
/**
2006-02-01 09:30:55 -08:00
* aac_fib_map_free - free the fib objects
2005-04-16 15:20:36 -07:00
* @ dev : Adapter to free
*
* Free the PCI mappings and the memory allocated for FIB blocks
* on this adapter .
*/
2006-02-01 09:30:55 -08:00
void aac_fib_map_free ( struct aac_dev * dev )
2005-04-16 15:20:36 -07:00
{
2017-02-16 12:51:14 -08:00
size_t alloc_size ;
size_t fib_size ;
int num_fibs ;
if ( ! dev - > hw_fib_va | | ! dev - > max_cmd_size )
return ;
num_fibs = dev - > scsi_host_ptr - > can_queue + AAC_NUM_MGT_FIB ;
fib_size = dev - > max_fib_size + sizeof ( struct aac_fib_xporthdr ) ;
alloc_size = fib_size * num_fibs + ALIGN32 - 1 ;
2017-04-05 16:14:16 +05:30
dma_free_coherent ( & dev - > pdev - > dev , alloc_size , dev - > hw_fib_va ,
dev - > hw_fib_pa ) ;
2017-02-16 12:51:14 -08:00
2007-07-17 11:15:08 -04:00
dev - > hw_fib_va = NULL ;
dev - > hw_fib_pa = 0 ;
2005-04-16 15:20:36 -07:00
}
2016-02-03 15:06:00 -08:00
void aac_fib_vector_assign ( struct aac_dev * dev )
{
u32 i = 0 ;
u32 vector = 1 ;
struct fib * fibptr = NULL ;
for ( i = 0 , fibptr = & dev - > fibs [ i ] ;
i < ( dev - > scsi_host_ptr - > can_queue + AAC_NUM_MGT_FIB ) ;
i + + , fibptr + + ) {
if ( ( dev - > max_msix = = 1 ) | |
( i > ( ( dev - > scsi_host_ptr - > can_queue + AAC_NUM_MGT_FIB - 1 )
- dev - > vector_cap ) ) ) {
fibptr - > vector_no = 0 ;
} else {
fibptr - > vector_no = vector ;
vector + + ;
if ( vector = = dev - > max_msix )
vector = 1 ;
}
}
}
2005-04-16 15:20:36 -07:00
/**
2006-02-01 09:30:55 -08:00
* aac_fib_setup - setup the fibs
2005-04-16 15:20:36 -07:00
* @ dev : Adapter to set up
*
tree-wide: fix comment/printk typos
"gadget", "through", "command", "maintain", "maintain", "controller", "address",
"between", "initiali[zs]e", "instead", "function", "select", "already",
"equal", "access", "management", "hierarchy", "registration", "interest",
"relative", "memory", "offset", "already",
Signed-off-by: Uwe Kleine-König <u.kleine-koenig@pengutronix.de>
Signed-off-by: Jiri Kosina <jkosina@suse.cz>
2010-11-01 15:38:34 -04:00
* Allocate the PCI space for the fibs , map it and then initialise the
2005-04-16 15:20:36 -07:00
* fib area , the unmapped fib data and also the free list
*/
2006-02-01 09:30:55 -08:00
int aac_fib_setup ( struct aac_dev * dev )
2005-04-16 15:20:36 -07:00
{
struct fib * fibptr ;
2007-03-15 10:26:22 -07:00
struct hw_fib * hw_fib ;
2005-04-16 15:20:36 -07:00
dma_addr_t hw_fib_pa ;
int i ;
2017-02-02 15:53:17 -08:00
u32 max_cmds ;
2005-05-16 18:28:42 -07:00
while ( ( ( i = fib_map_alloc ( dev ) ) = = - ENOMEM )
& & ( dev - > scsi_host_ptr - > can_queue > ( 64 - AAC_NUM_MGT_FIB ) ) ) {
2017-02-02 15:53:17 -08:00
max_cmds = ( dev - > scsi_host_ptr - > can_queue + AAC_NUM_MGT_FIB ) > > 1 ;
dev - > scsi_host_ptr - > can_queue = max_cmds - AAC_NUM_MGT_FIB ;
if ( dev - > comm_interface ! = AAC_COMM_MESSAGE_TYPE3 )
dev - > init - > r7 . max_io_commands = cpu_to_le32 ( max_cmds ) ;
2005-05-16 18:28:42 -07:00
}
if ( i < 0 )
2005-04-16 15:20:36 -07:00
return - ENOMEM ;
2008-01-16 07:39:06 -08:00
2011-03-17 02:10:32 -07:00
memset ( dev - > hw_fib_va , 0 ,
2017-02-02 15:53:29 -08:00
( dev - > max_cmd_size + sizeof ( struct aac_fib_xporthdr ) ) *
2011-03-17 02:10:32 -07:00
( dev - > scsi_host_ptr - > can_queue + AAC_NUM_MGT_FIB ) ) ;
2017-02-16 12:51:14 -08:00
/* 32 byte alignment for PMC */
hw_fib_pa = ( dev - > hw_fib_pa + ( ALIGN32 - 1 ) ) & ~ ( ALIGN32 - 1 ) ;
hw_fib = ( struct hw_fib * ) ( ( unsigned char * ) dev - > hw_fib_va +
( hw_fib_pa - dev - > hw_fib_pa ) ) ;
2011-03-17 02:10:32 -07:00
/* add Xport header */
2017-02-16 12:51:14 -08:00
hw_fib = ( struct hw_fib * ) ( ( unsigned char * ) hw_fib +
2011-03-17 02:10:32 -07:00
sizeof ( struct aac_fib_xporthdr ) ) ;
2017-02-16 12:51:14 -08:00
hw_fib_pa + = sizeof ( struct aac_fib_xporthdr ) ;
2011-03-17 02:10:32 -07:00
2005-04-16 15:20:36 -07:00
/*
* Initialise the fibs
*/
2008-01-16 07:39:06 -08:00
for ( i = 0 , fibptr = & dev - > fibs [ i ] ;
i < ( dev - > scsi_host_ptr - > can_queue + AAC_NUM_MGT_FIB ) ;
i + + , fibptr + + )
2005-04-16 15:20:36 -07:00
{
2012-07-14 18:18:51 +05:30
fibptr - > flags = 0 ;
2016-02-03 15:05:59 -08:00
fibptr - > size = sizeof ( struct fib ) ;
2005-04-16 15:20:36 -07:00
fibptr - > dev = dev ;
2007-03-15 10:26:22 -07:00
fibptr - > hw_fib_va = hw_fib ;
fibptr - > data = ( void * ) fibptr - > hw_fib_va - > data ;
2005-04-16 15:20:36 -07:00
fibptr - > next = fibptr + 1 ; /* Forward chain the fibs */
2018-12-10 22:32:41 +01:00
init_completion ( & fibptr - > event_wait ) ;
2005-04-16 15:20:36 -07:00
spin_lock_init ( & fibptr - > event_lock ) ;
2007-03-15 10:26:22 -07:00
hw_fib - > header . XferState = cpu_to_le32 ( 0xffffffff ) ;
2017-02-02 15:53:29 -08:00
hw_fib - > header . SenderSize =
cpu_to_le16 ( dev - > max_fib_size ) ; /* ?? max_cmd_size */
2005-04-16 15:20:36 -07:00
fibptr - > hw_fib_pa = hw_fib_pa ;
2017-02-02 15:53:29 -08:00
fibptr - > hw_sgl_pa = hw_fib_pa +
offsetof ( struct aac_hba_cmd_req , sge [ 2 ] ) ;
/*
* one element is for the ptr to the separate sg list ,
* second element for 32 byte alignment
*/
fibptr - > hw_error_pa = hw_fib_pa +
offsetof ( struct aac_native_hba , resp . resp_bytes [ 0 ] ) ;
2011-03-17 02:10:32 -07:00
hw_fib = ( struct hw_fib * ) ( ( unsigned char * ) hw_fib +
2017-02-02 15:53:22 -08:00
dev - > max_cmd_size + sizeof ( struct aac_fib_xporthdr ) ) ;
2011-03-17 02:10:32 -07:00
hw_fib_pa = hw_fib_pa +
2017-02-02 15:53:22 -08:00
dev - > max_cmd_size + sizeof ( struct aac_fib_xporthdr ) ;
2005-04-16 15:20:36 -07:00
}
2016-02-03 15:06:00 -08:00
/*
* Assign vector numbers to fibs
*/
aac_fib_vector_assign ( dev ) ;
2005-04-16 15:20:36 -07:00
/*
* Add the fib chain to the free list
*/
2005-05-16 18:28:42 -07:00
dev - > fibs [ dev - > scsi_host_ptr - > can_queue + AAC_NUM_MGT_FIB - 1 ] . next = NULL ;
2005-04-16 15:20:36 -07:00
/*
2016-02-03 15:05:59 -08:00
* Set 8 fibs aside for management tools
*/
dev - > free_fib = & dev - > fibs [ dev - > scsi_host_ptr - > can_queue ] ;
2005-04-16 15:20:36 -07:00
return 0 ;
}
2016-02-03 15:05:59 -08:00
/**
* aac_fib_alloc_tag - allocate a fib using tags
* @ dev : Adapter to allocate the fib for
2020-07-13 08:59:44 +01:00
* @ scmd : SCSI command
2016-02-03 15:05:59 -08:00
*
* Allocate a fib from the adapter fib pool using tags
* from the blk layer .
*/
struct fib * aac_fib_alloc_tag ( struct aac_dev * dev , struct scsi_cmnd * scmd )
{
struct fib * fibptr ;
fibptr = & dev - > fibs [ scmd - > request - > tag ] ;
/*
* Null out fields that depend on being zero at the start of
* each I / O
*/
fibptr - > hw_fib_va - > header . XferState = 0 ;
fibptr - > type = FSAFS_NTC_FIB_CONTEXT ;
fibptr - > callback_data = NULL ;
fibptr - > callback = NULL ;
2019-10-15 11:51:59 +05:30
fibptr - > flags = 0 ;
2016-02-03 15:05:59 -08:00
return fibptr ;
}
2005-04-16 15:20:36 -07:00
/**
2006-02-01 09:30:55 -08:00
* aac_fib_alloc - allocate a fib
2005-04-16 15:20:36 -07:00
* @ dev : Adapter to allocate the fib for
*
* Allocate a fib from the adapter fib pool . If the pool is empty we
2005-05-16 18:28:42 -07:00
* return NULL .
2005-04-16 15:20:36 -07:00
*/
2008-01-16 07:39:06 -08:00
2006-02-01 09:30:55 -08:00
struct fib * aac_fib_alloc ( struct aac_dev * dev )
2005-04-16 15:20:36 -07:00
{
struct fib * fibptr ;
unsigned long flags ;
spin_lock_irqsave ( & dev - > fib_lock , flags ) ;
2008-01-16 07:39:06 -08:00
fibptr = dev - > free_fib ;
2005-05-16 18:28:42 -07:00
if ( ! fibptr ) {
spin_unlock_irqrestore ( & dev - > fib_lock , flags ) ;
return fibptr ;
}
2005-04-16 15:20:36 -07:00
dev - > free_fib = fibptr - > next ;
spin_unlock_irqrestore ( & dev - > fib_lock , flags ) ;
/*
* Set the proper node type code and node byte size
*/
fibptr - > type = FSAFS_NTC_FIB_CONTEXT ;
fibptr - > size = sizeof ( struct fib ) ;
/*
* Null out fields that depend on being zero at the start of
* each I / O
*/
2007-03-15 10:26:22 -07:00
fibptr - > hw_fib_va - > header . XferState = 0 ;
2008-01-08 13:26:43 -08:00
fibptr - > flags = 0 ;
2005-04-16 15:20:36 -07:00
fibptr - > callback = NULL ;
fibptr - > callback_data = NULL ;
return fibptr ;
}
/**
2006-02-01 09:30:55 -08:00
* aac_fib_free - free a fib
2005-04-16 15:20:36 -07:00
* @ fibptr : fib to free up
*
* Frees up a fib and places it on the appropriate queue
*/
2008-01-16 07:39:06 -08:00
2006-02-01 09:30:55 -08:00
void aac_fib_free ( struct fib * fibptr )
2005-04-16 15:20:36 -07:00
{
2015-03-26 10:41:30 -04:00
unsigned long flags ;
2009-12-21 18:39:27 +05:30
2015-03-26 10:41:30 -04:00
if ( fibptr - > done = = 2 )
2009-12-21 18:39:27 +05:30
return ;
2005-04-16 15:20:36 -07:00
spin_lock_irqsave ( & fibptr - > dev - > fib_lock , flags ) ;
2007-03-15 10:27:45 -07:00
if ( unlikely ( fibptr - > flags & FIB_CONTEXT_FLAG_TIMED_OUT ) )
2005-04-16 15:20:36 -07:00
aac_config . fib_timeouts + + ;
2017-02-02 15:53:29 -08:00
if ( ! ( fibptr - > flags & FIB_CONTEXT_FLAG_NATIVE_HBA ) & &
fibptr - > hw_fib_va - > header . XferState ! = 0 ) {
2007-03-15 10:27:45 -07:00
printk ( KERN_WARNING " aac_fib_free, XferState != 0, fibptr = 0x%p, XferState = 0x%x \n " ,
( void * ) fibptr ,
le32_to_cpu ( fibptr - > hw_fib_va - > header . XferState ) ) ;
}
fibptr - > next = fibptr - > dev - > free_fib ;
fibptr - > dev - > free_fib = fibptr ;
2005-04-16 15:20:36 -07:00
spin_unlock_irqrestore ( & fibptr - > dev - > fib_lock , flags ) ;
}
/**
2006-02-01 09:30:55 -08:00
* aac_fib_init - initialise a fib
2005-04-16 15:20:36 -07:00
* @ fibptr : The fib to initialize
2008-01-16 07:39:06 -08:00
*
2005-04-16 15:20:36 -07:00
* Set up the generic fib fields ready for use
*/
2008-01-16 07:39:06 -08:00
2006-02-01 09:30:55 -08:00
void aac_fib_init ( struct fib * fibptr )
2005-04-16 15:20:36 -07:00
{
2007-03-15 10:26:22 -07:00
struct hw_fib * hw_fib = fibptr - > hw_fib_va ;
2005-04-16 15:20:36 -07:00
2012-07-14 18:18:51 +05:30
memset ( & hw_fib - > header , 0 , sizeof ( struct aac_fibhdr ) ) ;
2005-04-16 15:20:36 -07:00
hw_fib - > header . StructType = FIB_MAGIC ;
2005-05-16 18:28:42 -07:00
hw_fib - > header . Size = cpu_to_le16 ( fibptr - > dev - > max_fib_size ) ;
hw_fib - > header . XferState = cpu_to_le32 ( HostOwned | FibInitialized | FibEmpty | FastResponseCapable ) ;
2012-07-14 18:18:51 +05:30
hw_fib - > header . u . ReceiverFibAddress = cpu_to_le32 ( fibptr - > hw_fib_pa ) ;
2005-05-16 18:28:42 -07:00
hw_fib - > header . SenderSize = cpu_to_le16 ( fibptr - > dev - > max_fib_size ) ;
2005-04-16 15:20:36 -07:00
}
/**
2021-03-03 14:46:17 +00:00
* fib_dealloc - deallocate a fib
2005-04-16 15:20:36 -07:00
* @ fibptr : fib to deallocate
*
* Will deallocate and return to the free pool the FIB pointed to by the
* caller .
*/
2008-01-16 07:39:06 -08:00
2005-04-25 19:45:58 -07:00
static void fib_dealloc ( struct fib * fibptr )
2005-04-16 15:20:36 -07:00
{
2007-03-15 10:26:22 -07:00
struct hw_fib * hw_fib = fibptr - > hw_fib_va ;
2008-01-16 07:39:06 -08:00
hw_fib - > header . XferState = 0 ;
2005-04-16 15:20:36 -07:00
}
/*
* Commuication primitives define and support the queuing method we use to
* support host to adapter commuication . All queue accesses happen through
* these routines and are the only routines which have a knowledge of the
* how these queues are implemented .
*/
2008-01-16 07:39:06 -08:00
2005-04-16 15:20:36 -07:00
/**
* aac_get_entry - get a queue entry
* @ dev : Adapter
* @ qid : Queue Number
* @ entry : Entry return
* @ index : Index return
* @ nonotify : notification control
*
* With a priority the routine returns a queue entry if the queue has free entries . If the queue
* is full ( no free entries ) than no entry is returned and the function returns 0 otherwise 1 is
* returned .
*/
2008-01-16 07:39:06 -08:00
2005-04-16 15:20:36 -07:00
static int aac_get_entry ( struct aac_dev * dev , u32 qid , struct aac_entry * * entry , u32 * index , unsigned long * nonotify )
{
struct aac_queue * q ;
2005-08-03 15:38:51 -07:00
unsigned long idx ;
2005-04-16 15:20:36 -07:00
/*
* All of the queues wrap when they reach the end , so we check
* to see if they have reached the end and if they have we just
* set the index back to zero . This is a wrap . You could or off
* the high bits in all updates but this is a bit faster I think .
*/
q = & dev - > queues - > queue [ qid ] ;
2005-08-03 15:38:51 -07:00
idx = * index = le32_to_cpu ( * ( q - > headers . producer ) ) ;
/* Interrupt Moderation, only interrupt for first two entries */
if ( idx ! = le32_to_cpu ( * ( q - > headers . consumer ) ) ) {
if ( - - idx = = 0 ) {
2005-09-20 12:57:11 -07:00
if ( qid = = AdapNormCmdQueue )
2005-08-03 15:38:51 -07:00
idx = ADAP_NORM_CMD_ENTRIES ;
2005-09-20 12:57:11 -07:00
else
2005-08-03 15:38:51 -07:00
idx = ADAP_NORM_RESP_ENTRIES ;
}
if ( idx ! = le32_to_cpu ( * ( q - > headers . consumer ) ) )
2008-01-16 07:39:06 -08:00
* nonotify = 1 ;
2005-08-03 15:38:51 -07:00
}
2005-04-16 15:20:36 -07:00
2005-09-20 12:57:11 -07:00
if ( qid = = AdapNormCmdQueue ) {
2008-01-16 07:39:06 -08:00
if ( * index > = ADAP_NORM_CMD_ENTRIES )
2005-04-16 15:20:36 -07:00
* index = 0 ; /* Wrap to front of the Producer Queue. */
2005-09-20 12:57:11 -07:00
} else {
2008-01-16 07:39:06 -08:00
if ( * index > = ADAP_NORM_RESP_ENTRIES )
2005-04-16 15:20:36 -07:00
* index = 0 ; /* Wrap to front of the Producer Queue. */
}
2008-01-16 07:39:06 -08:00
/* Queue is full */
if ( ( * index + 1 ) = = le32_to_cpu ( * ( q - > headers . consumer ) ) ) {
2005-05-16 18:28:42 -07:00
printk ( KERN_WARNING " Queue %d full, %u outstanding. \n " ,
2015-03-26 10:41:30 -04:00
qid , atomic_read ( & q - > numpending ) ) ;
2005-04-16 15:20:36 -07:00
return 0 ;
} else {
2008-01-16 07:39:06 -08:00
* entry = q - > base + * index ;
2005-04-16 15:20:36 -07:00
return 1 ;
}
2008-01-16 07:39:06 -08:00
}
2005-04-16 15:20:36 -07:00
/**
* aac_queue_get - get the next free QE
* @ dev : Adapter
* @ index : Returned index
2020-07-13 08:59:44 +01:00
* @ qid : Queue number
* @ hw_fib : Fib to associate with the queue entry
2005-04-16 15:20:36 -07:00
* @ wait : Wait if queue full
* @ fibptr : Driver fib object to go with fib
* @ nonotify : Don ' t notify the adapter
*
* Gets the next free QE off the requested priorty adapter command
* queue and associates the Fib with the QE . The QE represented by
* index is ready to insert on the queue when this routine returns
* success .
*/
2007-01-23 14:59:20 -08:00
int aac_queue_get ( struct aac_dev * dev , u32 * index , u32 qid , struct hw_fib * hw_fib , int wait , struct fib * fibptr , unsigned long * nonotify )
2005-04-16 15:20:36 -07:00
{
struct aac_entry * entry = NULL ;
int map = 0 ;
2008-01-16 07:39:06 -08:00
2005-09-20 12:57:11 -07:00
if ( qid = = AdapNormCmdQueue ) {
2005-04-16 15:20:36 -07:00
/* if no entries wait for some if caller wants to */
2008-01-16 07:39:06 -08:00
while ( ! aac_get_entry ( dev , qid , & entry , index , nonotify ) ) {
2005-04-16 15:20:36 -07:00
printk ( KERN_ERR " GetEntries failed \n " ) ;
}
2008-01-16 07:39:06 -08:00
/*
* Setup queue entry with a command , status and fib mapped
*/
entry - > size = cpu_to_le32 ( le16_to_cpu ( hw_fib - > header . Size ) ) ;
map = 1 ;
2005-09-20 12:57:11 -07:00
} else {
2008-01-16 07:39:06 -08:00
while ( ! aac_get_entry ( dev , qid , & entry , index , nonotify ) ) {
2005-04-16 15:20:36 -07:00
/* if no entries wait for some if caller wants to */
}
2008-01-16 07:39:06 -08:00
/*
* Setup queue entry with command , status and fib mapped
*/
entry - > size = cpu_to_le32 ( le16_to_cpu ( hw_fib - > header . Size ) ) ;
entry - > addr = hw_fib - > header . SenderFibAddress ;
/* Restore adapters pointer to the FIB */
2012-07-14 18:18:51 +05:30
hw_fib - > header . u . ReceiverFibAddress = hw_fib - > header . SenderFibAddress ; /* Let the adapter now where to find its data */
2008-01-16 07:39:06 -08:00
map = 0 ;
2005-04-16 15:20:36 -07:00
}
/*
* If MapFib is true than we need to map the Fib and put pointers
* in the queue entry .
*/
if ( map )
entry - > addr = cpu_to_le32 ( fibptr - > hw_fib_pa ) ;
return 0 ;
}
/*
2008-01-16 07:39:06 -08:00
* Define the highest level of host to adapter communication routines .
* These routines will support host to adapter FS commuication . These
2005-04-16 15:20:36 -07:00
* routines have no knowledge of the commuication method used . This level
* sends and receives FIBs . This level has no knowledge of how these FIBs
* get passed back and forth .
*/
/**
2006-02-01 09:30:55 -08:00
* aac_fib_send - send a fib to the adapter
2005-04-16 15:20:36 -07:00
* @ command : Command to send
* @ fibptr : The fib
* @ size : Size of fib data area
* @ priority : Priority of Fib
* @ wait : Async / sync select
* @ reply : True if a reply is wanted
* @ callback : Called with reply
* @ callback_data : Passed to callback
*
* Sends the requested FIB to the adapter and optionally will wait for a
* response FIB . If the caller does not wish to wait for a response than
* an event to wait on must be supplied . This event will be set when a
* response FIB is received from the adapter .
*/
2008-01-16 07:39:06 -08:00
2006-02-01 09:30:55 -08:00
int aac_fib_send ( u16 command , struct fib * fibptr , unsigned long size ,
int priority , int wait , int reply , fib_callback callback ,
void * callback_data )
2005-04-16 15:20:36 -07:00
{
struct aac_dev * dev = fibptr - > dev ;
2007-03-15 10:26:22 -07:00
struct hw_fib * hw_fib = fibptr - > hw_fib_va ;
2005-04-16 15:20:36 -07:00
unsigned long flags = 0 ;
2009-12-21 18:39:27 +05:30
unsigned long mflags = 0 ;
2012-02-08 22:51:04 -08:00
unsigned long sflags = 0 ;
2009-12-21 18:39:27 +05:30
2005-04-16 15:20:36 -07:00
if ( ! ( hw_fib - > header . XferState & cpu_to_le32 ( HostOwned ) ) )
return - EBUSY ;
2017-02-16 12:51:13 -08:00
if ( hw_fib - > header . XferState & cpu_to_le32 ( AdapterProcessed ) )
return - EINVAL ;
2005-04-16 15:20:36 -07:00
/*
2011-03-30 22:57:33 -03:00
* There are 5 cases with the wait and response requested flags .
2005-04-16 15:20:36 -07:00
* The only invalid cases are if the caller requests to wait and
* does not request a response and if the caller does not want a
* response and the Fib is not allocated from pool . If a response
2018-08-07 01:14:29 -04:00
* is not requested the Fib will just be deallocaed by the DPC
2005-04-16 15:20:36 -07:00
* routine when the response comes back from the adapter . No
2008-01-16 07:39:06 -08:00
* further processing will be done besides deleting the Fib . We
2005-04-16 15:20:36 -07:00
* will have a debug mode where the adapter can notify the host
* it had a problem and the host can log that fact .
*/
2008-01-08 13:26:43 -08:00
fibptr - > flags = 0 ;
2005-04-16 15:20:36 -07:00
if ( wait & & ! reply ) {
return - EINVAL ;
} else if ( ! wait & & reply ) {
hw_fib - > header . XferState | = cpu_to_le32 ( Async | ResponseExpected ) ;
FIB_COUNTER_INCREMENT ( aac_config . AsyncSent ) ;
} else if ( ! wait & & ! reply ) {
hw_fib - > header . XferState | = cpu_to_le32 ( NoResponseExpected ) ;
FIB_COUNTER_INCREMENT ( aac_config . NoResponseSent ) ;
} else if ( wait & & reply ) {
hw_fib - > header . XferState | = cpu_to_le32 ( ResponseExpected ) ;
FIB_COUNTER_INCREMENT ( aac_config . NormalSent ) ;
2008-01-16 07:39:06 -08:00
}
2005-04-16 15:20:36 -07:00
/*
* Map the fib into 32 bits by using the fib number
*/
2017-02-02 15:53:29 -08:00
hw_fib - > header . SenderFibAddress =
cpu_to_le32 ( ( ( u32 ) ( fibptr - dev - > fibs ) ) < < 2 ) ;
/* use the same shifted value for handle to be compatible
* with the new native hba command handle
*/
hw_fib - > header . Handle =
cpu_to_le32 ( ( ( ( u32 ) ( fibptr - dev - > fibs ) ) < < 2 ) + 1 ) ;
2005-04-16 15:20:36 -07:00
/*
* Set FIB state to indicate where it came from and if we want a
* response from the adapter . Also load the command from the
* caller .
*
* Map the hw fib pointer as a 32 bit value
*/
hw_fib - > header . Command = cpu_to_le16 ( command ) ;
hw_fib - > header . XferState | = cpu_to_le32 ( SentFromHost ) ;
/*
* Set the size of the Fib we want to send to the adapter
*/
hw_fib - > header . Size = cpu_to_le16 ( sizeof ( struct aac_fibhdr ) + size ) ;
if ( le16_to_cpu ( hw_fib - > header . Size ) > le16_to_cpu ( hw_fib - > header . SenderSize ) ) {
return - EMSGSIZE ;
2008-01-16 07:39:06 -08:00
}
2005-04-16 15:20:36 -07:00
/*
* Get a queue entry connect the FIB to it and send an notify
* the adapter a command is ready .
*/
2005-09-20 12:57:11 -07:00
hw_fib - > header . XferState | = cpu_to_le32 ( NormalPriority ) ;
2005-04-16 15:20:36 -07:00
/*
* Fill in the Callback and CallbackContext if we are not
* going to wait .
*/
if ( ! wait ) {
fibptr - > callback = callback ;
fibptr - > callback_data = callback_data ;
2008-01-08 13:26:43 -08:00
fibptr - > flags = FIB_CONTEXT_FLAG ;
2005-04-16 15:20:36 -07:00
}
fibptr - > done = 0 ;
2005-09-20 12:57:11 -07:00
FIB_COUNTER_INCREMENT ( aac_config . FibsSent ) ;
dprintk ( ( KERN_DEBUG " Fib contents:. \n " ) ) ;
2005-10-24 10:52:22 -07:00
dprintk ( ( KERN_DEBUG " Command = %d. \n " , le32_to_cpu ( hw_fib - > header . Command ) ) ) ;
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 ) ) ) ;
2007-03-15 10:26:22 -07:00
dprintk ( ( KERN_DEBUG " hw_fib va being sent=%p \n " , fibptr - > hw_fib_va ) ) ;
2005-09-20 12:57:11 -07:00
dprintk ( ( KERN_DEBUG " hw_fib pa being sent=%lx \n " , ( ulong ) fibptr - > hw_fib_pa ) ) ;
dprintk ( ( KERN_DEBUG " fib being sent=%p \n " , fibptr ) ) ;
2006-08-03 08:02:24 -07:00
if ( ! dev - > queues )
2006-09-19 08:59:23 -07:00
return - EBUSY ;
2005-09-20 12:57:11 -07:00
2009-12-21 18:39:27 +05:30
if ( wait ) {
spin_lock_irqsave ( & dev - > manage_lock , mflags ) ;
if ( dev - > management_fib_count > = AAC_NUM_MGT_FIB ) {
printk ( KERN_INFO " No management Fibs Available:%d \n " ,
dev - > management_fib_count ) ;
spin_unlock_irqrestore ( & dev - > manage_lock , mflags ) ;
return - EBUSY ;
}
dev - > management_fib_count + + ;
spin_unlock_irqrestore ( & dev - > manage_lock , mflags ) ;
2005-09-20 12:57:11 -07:00
spin_lock_irqsave ( & fibptr - > event_lock , flags ) ;
2009-12-21 18:39:27 +05:30
}
2012-02-08 22:51:04 -08:00
if ( dev - > sync_mode ) {
if ( wait )
spin_unlock_irqrestore ( & fibptr - > event_lock , flags ) ;
spin_lock_irqsave ( & dev - > sync_lock , sflags ) ;
if ( dev - > sync_fib ) {
list_add_tail ( & fibptr - > fiblink , & dev - > sync_fib_list ) ;
spin_unlock_irqrestore ( & dev - > sync_lock , sflags ) ;
} else {
dev - > sync_fib = fibptr ;
spin_unlock_irqrestore ( & dev - > sync_lock , sflags ) ;
aac_adapter_sync_cmd ( dev , SEND_SYNCHRONOUS_FIB ,
( u32 ) fibptr - > hw_fib_pa , 0 , 0 , 0 , 0 , 0 ,
NULL , NULL , NULL , NULL , NULL ) ;
}
if ( wait ) {
fibptr - > flags | = FIB_CONTEXT_FLAG_WAIT ;
2018-12-10 22:32:41 +01:00
if ( wait_for_completion_interruptible ( & fibptr - > event_wait ) ) {
2012-02-08 22:51:04 -08:00
fibptr - > flags & = ~ FIB_CONTEXT_FLAG_WAIT ;
return - EFAULT ;
}
return 0 ;
}
return - EINPROGRESS ;
}
2009-12-21 18:39:27 +05:30
if ( aac_adapter_deliver ( fibptr ) ! = 0 ) {
printk ( KERN_ERR " aac_fib_send: returned -EBUSY \n " ) ;
if ( wait ) {
spin_unlock_irqrestore ( & fibptr - > event_lock , flags ) ;
spin_lock_irqsave ( & dev - > manage_lock , mflags ) ;
dev - > management_fib_count - - ;
spin_unlock_irqrestore ( & dev - > manage_lock , mflags ) ;
}
return - EBUSY ;
}
2005-10-24 10:52:22 -07:00
2005-04-16 15:20:36 -07:00
/*
2008-01-16 07:39:06 -08:00
* If the caller wanted us to wait for response wait now .
2005-04-16 15:20:36 -07:00
*/
2008-01-16 07:39:06 -08:00
2005-04-16 15:20:36 -07:00
if ( wait ) {
spin_unlock_irqrestore ( & fibptr - > event_lock , flags ) ;
2005-09-20 12:56:50 -07:00
/* Only set for first known interruptable command */
if ( wait < 0 ) {
/*
* * VERY * Dangerous to time out a command , the
* assumption is made that we have no hope of
* functioning because an interrupt routing or other
* hardware failure has occurred .
*/
2012-06-11 14:44:44 -04:00
unsigned long timeout = jiffies + ( 180 * HZ ) ; /* 3 minutes */
2018-12-10 22:32:41 +01:00
while ( ! try_wait_for_completion ( & fibptr - > event_wait ) ) {
2006-11-21 10:40:08 -08:00
int blink ;
2012-06-11 14:44:44 -04:00
if ( time_is_before_eq_jiffies ( timeout ) ) {
2007-01-23 14:59:20 -08:00
struct aac_queue * q = & dev - > queues - > queue [ AdapNormCmdQueue ] ;
2015-03-26 10:41:30 -04:00
atomic_dec ( & q - > numpending ) ;
2005-09-20 12:56:50 -07:00
if ( wait = = - 1 ) {
2006-02-01 09:30:55 -08:00
printk ( KERN_ERR " aacraid: aac_fib_send: first asynchronous command timed out. \n "
2005-09-20 12:56:50 -07:00
" Usually a result of a PCI interrupt routing problem; \n "
" update mother board BIOS or consider utilizing one of \n "
" the SAFE mode kernel options (acpi, apic etc) \n " ) ;
}
return - ETIMEDOUT ;
}
2017-02-16 12:51:12 -08:00
2019-03-22 12:16:03 -06:00
if ( unlikely ( aac_pci_offline ( dev ) ) )
2017-02-16 12:51:12 -08:00
return - EFAULT ;
2006-11-21 10:40:08 -08:00
if ( ( blink = aac_adapter_check_health ( dev ) ) > 0 ) {
if ( wait = = - 1 ) {
printk ( KERN_ERR " aacraid: aac_fib_send: adapter blinkLED 0x%x. \n "
" Usually a result of a serious unrecoverable hardware problem \n " ,
blink ) ;
}
return - EFAULT ;
}
2016-04-25 23:31:26 -07:00
/*
* Allow other processes / CPUS to use core
*/
schedule ( ) ;
2005-09-20 12:56:50 -07:00
}
2018-12-10 22:32:41 +01:00
} else if ( wait_for_completion_interruptible ( & fibptr - > event_wait ) ) {
2009-12-21 18:39:27 +05:30
/* Do nothing ... satisfy
2018-12-10 22:32:41 +01:00
* wait_for_completion_interruptible must_check */
2008-04-14 14:20:16 -04:00
}
2009-12-21 18:39:27 +05:30
2007-03-15 10:27:21 -07:00
spin_lock_irqsave ( & fibptr - > event_lock , flags ) ;
2009-12-21 18:39:27 +05:30
if ( fibptr - > done = = 0 ) {
2007-03-15 10:27:21 -07:00
fibptr - > done = 2 ; /* Tell interrupt we aborted */
2006-08-03 08:02:24 -07:00
spin_unlock_irqrestore ( & fibptr - > event_lock , flags ) ;
2009-12-21 18:39:27 +05:30
return - ERESTARTSYS ;
2006-08-03 08:02:24 -07:00
}
2007-03-15 10:27:21 -07:00
spin_unlock_irqrestore ( & fibptr - > event_lock , flags ) ;
2006-06-23 02:06:06 -07:00
BUG_ON ( fibptr - > done = = 0 ) ;
2008-01-16 07:39:06 -08:00
2007-03-26 09:21:14 -04:00
if ( unlikely ( fibptr - > flags & FIB_CONTEXT_FLAG_TIMED_OUT ) )
2005-04-16 15:20:36 -07:00
return - ETIMEDOUT ;
2007-03-26 09:21:14 -04:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
/*
* If the user does not want a response than return success otherwise
* return pending
*/
if ( reply )
return - EINPROGRESS ;
else
return 0 ;
}
2017-02-02 15:53:29 -08:00
int aac_hba_send ( u8 command , struct fib * fibptr , fib_callback callback ,
void * callback_data )
{
struct aac_dev * dev = fibptr - > dev ;
int wait ;
unsigned long flags = 0 ;
unsigned long mflags = 0 ;
2018-04-25 10:24:20 -06:00
struct aac_hba_cmd_req * hbacmd = ( struct aac_hba_cmd_req * )
fibptr - > hw_fib_va ;
2017-02-02 15:53:29 -08:00
fibptr - > flags = ( FIB_CONTEXT_FLAG | FIB_CONTEXT_FLAG_NATIVE_HBA ) ;
if ( callback ) {
wait = 0 ;
fibptr - > callback = callback ;
fibptr - > callback_data = callback_data ;
} else
wait = 1 ;
2018-04-25 10:24:20 -06:00
hbacmd - > iu_type = command ;
2017-02-02 15:53:29 -08:00
2018-04-25 10:24:20 -06:00
if ( command = = HBA_IU_TYPE_SCSI_CMD_REQ ) {
2017-02-02 15:53:29 -08:00
/* bit1 of request_id must be 0 */
hbacmd - > request_id =
cpu_to_le32 ( ( ( ( u32 ) ( fibptr - dev - > fibs ) ) < < 2 ) + 1 ) ;
2017-06-30 19:18:11 +02:00
fibptr - > flags | = FIB_CONTEXT_FLAG_SCSI_CMD ;
2020-02-12 16:29:31 -08:00
} else
2017-02-02 15:53:29 -08:00
return - EINVAL ;
if ( wait ) {
spin_lock_irqsave ( & dev - > manage_lock , mflags ) ;
if ( dev - > management_fib_count > = AAC_NUM_MGT_FIB ) {
spin_unlock_irqrestore ( & dev - > manage_lock , mflags ) ;
return - EBUSY ;
}
dev - > management_fib_count + + ;
spin_unlock_irqrestore ( & dev - > manage_lock , mflags ) ;
spin_lock_irqsave ( & fibptr - > event_lock , flags ) ;
}
if ( aac_adapter_deliver ( fibptr ) ! = 0 ) {
if ( wait ) {
spin_unlock_irqrestore ( & fibptr - > event_lock , flags ) ;
spin_lock_irqsave ( & dev - > manage_lock , mflags ) ;
dev - > management_fib_count - - ;
spin_unlock_irqrestore ( & dev - > manage_lock , mflags ) ;
}
return - EBUSY ;
}
FIB_COUNTER_INCREMENT ( aac_config . NativeSent ) ;
if ( wait ) {
2017-02-16 12:51:12 -08:00
2017-02-02 15:53:29 -08:00
spin_unlock_irqrestore ( & fibptr - > event_lock , flags ) ;
2017-02-16 12:51:12 -08:00
2019-03-22 12:16:03 -06:00
if ( unlikely ( aac_pci_offline ( dev ) ) )
2017-02-16 12:51:12 -08:00
return - EFAULT ;
2017-05-10 09:39:49 -07:00
fibptr - > flags | = FIB_CONTEXT_FLAG_WAIT ;
2018-12-10 22:32:41 +01:00
if ( wait_for_completion_interruptible ( & fibptr - > event_wait ) )
2017-02-02 15:53:29 -08:00
fibptr - > done = 2 ;
2017-05-10 09:39:49 -07:00
fibptr - > flags & = ~ ( FIB_CONTEXT_FLAG_WAIT ) ;
2017-02-02 15:53:29 -08:00
spin_lock_irqsave ( & fibptr - > event_lock , flags ) ;
if ( ( fibptr - > done = = 0 ) | | ( fibptr - > done = = 2 ) ) {
fibptr - > done = 2 ; /* Tell interrupt we aborted */
spin_unlock_irqrestore ( & fibptr - > event_lock , flags ) ;
return - ERESTARTSYS ;
}
spin_unlock_irqrestore ( & fibptr - > event_lock , flags ) ;
WARN_ON ( fibptr - > done = = 0 ) ;
if ( unlikely ( fibptr - > flags & FIB_CONTEXT_FLAG_TIMED_OUT ) )
return - ETIMEDOUT ;
return 0 ;
}
return - EINPROGRESS ;
}
2008-01-16 07:39:06 -08:00
/**
2005-04-16 15:20:36 -07:00
* aac_consumer_get - get the top of the queue
* @ dev : Adapter
* @ q : Queue
* @ entry : Return entry
*
* Will return a pointer to the entry on the top of the queue requested that
2008-01-16 07:39:06 -08:00
* we are a consumer of , and return the address of the queue entry . It does
* not change the state of the queue .
2005-04-16 15:20:36 -07:00
*/
int aac_consumer_get ( struct aac_dev * dev , struct aac_queue * q , struct aac_entry * * entry )
{
u32 index ;
int status ;
if ( le32_to_cpu ( * q - > headers . producer ) = = le32_to_cpu ( * q - > headers . consumer ) ) {
status = 0 ;
} else {
/*
* The consumer index must be wrapped if we have reached
* the end of the queue , else we just use the entry
* pointed to by the header index
*/
2008-01-16 07:39:06 -08:00
if ( le32_to_cpu ( * q - > headers . consumer ) > = q - > entries )
index = 0 ;
2005-04-16 15:20:36 -07:00
else
2008-01-16 07:39:06 -08:00
index = le32_to_cpu ( * q - > headers . consumer ) ;
2005-04-16 15:20:36 -07:00
* entry = q - > base + index ;
status = 1 ;
}
return ( status ) ;
}
/**
* aac_consumer_free - free consumer entry
* @ dev : Adapter
* @ q : Queue
* @ qid : Queue ident
*
* Frees up the current top of the queue we are a consumer of . If the
* queue was full notify the producer that the queue is no longer full .
*/
void aac_consumer_free ( struct aac_dev * dev , struct aac_queue * q , u32 qid )
{
int wasfull = 0 ;
u32 notify ;
if ( ( le32_to_cpu ( * q - > headers . producer ) + 1 ) = = le32_to_cpu ( * q - > headers . consumer ) )
wasfull = 1 ;
2008-01-16 07:39:06 -08:00
2005-04-16 15:20:36 -07:00
if ( le32_to_cpu ( * q - > headers . consumer ) > = q - > entries )
* q - > headers . consumer = cpu_to_le32 ( 1 ) ;
else
2008-03-28 14:48:35 -07:00
le32_add_cpu ( q - > headers . consumer , 1 ) ;
2008-01-16 07:39:06 -08:00
2005-04-16 15:20:36 -07:00
if ( wasfull ) {
switch ( qid ) {
case HostNormCmdQueue :
notify = HostNormCmdNotFull ;
break ;
case HostNormRespQueue :
notify = HostNormRespNotFull ;
break ;
default :
BUG ( ) ;
return ;
}
aac_adapter_notify ( dev , notify ) ;
}
2008-01-16 07:39:06 -08:00
}
2005-04-16 15:20:36 -07:00
/**
2006-02-01 09:30:55 -08:00
* aac_fib_adapter_complete - complete adapter issued fib
2005-04-16 15:20:36 -07:00
* @ fibptr : fib to complete
* @ size : size of fib
*
* Will do all necessary work to complete a FIB that was sent from
* the adapter .
*/
2006-02-01 09:30:55 -08:00
int aac_fib_adapter_complete ( struct fib * fibptr , unsigned short size )
2005-04-16 15:20:36 -07:00
{
2007-03-15 10:26:22 -07:00
struct hw_fib * hw_fib = fibptr - > hw_fib_va ;
2005-04-16 15:20:36 -07:00
struct aac_dev * dev = fibptr - > dev ;
2005-09-20 12:57:11 -07:00
struct aac_queue * q ;
2005-04-16 15:20:36 -07:00
unsigned long nointr = 0 ;
2005-09-20 12:57:11 -07:00
unsigned long qflags ;
2012-07-14 18:18:51 +05:30
if ( dev - > comm_interface = = AAC_COMM_MESSAGE_TYPE1 | |
2017-02-02 15:53:17 -08:00
dev - > comm_interface = = AAC_COMM_MESSAGE_TYPE2 | |
dev - > comm_interface = = AAC_COMM_MESSAGE_TYPE3 ) {
2011-03-17 02:10:32 -07:00
kfree ( hw_fib ) ;
return 0 ;
}
2005-09-20 12:57:11 -07:00
if ( hw_fib - > header . XferState = = 0 ) {
2007-01-23 14:59:20 -08:00
if ( dev - > comm_interface = = AAC_COMM_MESSAGE )
2011-03-17 02:10:32 -07:00
kfree ( hw_fib ) ;
2008-01-16 07:39:06 -08:00
return 0 ;
2005-09-20 12:57:11 -07:00
}
2005-04-16 15:20:36 -07:00
/*
* If we plan to do anything check the structure type first .
2008-01-16 07:39:06 -08:00
*/
2012-07-14 18:18:51 +05:30
if ( hw_fib - > header . StructType ! = FIB_MAGIC & &
hw_fib - > header . StructType ! = FIB_MAGIC2 & &
hw_fib - > header . StructType ! = FIB_MAGIC2_64 ) {
2007-01-23 14:59:20 -08:00
if ( dev - > comm_interface = = AAC_COMM_MESSAGE )
2011-03-17 02:10:32 -07:00
kfree ( hw_fib ) ;
2008-01-16 07:39:06 -08:00
return - EINVAL ;
2005-04-16 15:20:36 -07:00
}
/*
* This block handles the case where the adapter had sent us a
* command and we have finished processing the command . We
2008-01-16 07:39:06 -08:00
* call completeFib when we are done processing the command
* and want to send a response back to the adapter . This will
2005-04-16 15:20:36 -07:00
* send the completed cdb to the adapter .
*/
if ( hw_fib - > header . XferState & cpu_to_le32 ( SentFromAdapter ) ) {
2007-01-23 14:59:20 -08:00
if ( dev - > comm_interface = = AAC_COMM_MESSAGE ) {
2005-10-24 10:52:22 -07:00
kfree ( hw_fib ) ;
} else {
2008-01-16 07:39:06 -08:00
u32 index ;
hw_fib - > header . XferState | = cpu_to_le32 ( HostProcessed ) ;
2005-10-24 10:52:22 -07:00
if ( size ) {
size + = sizeof ( struct aac_fibhdr ) ;
2008-01-16 07:39:06 -08:00
if ( size > le16_to_cpu ( hw_fib - > header . SenderSize ) )
2005-10-24 10:52:22 -07:00
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 ) ;
2005-04-16 15:20:36 -07:00
}
2008-01-16 07:39:06 -08:00
} else {
printk ( KERN_WARNING " aac_fib_adapter_complete: "
" Unknown xferstate detected. \n " ) ;
BUG ( ) ;
2005-04-16 15:20:36 -07:00
}
return 0 ;
}
/**
2006-02-01 09:30:55 -08:00
* aac_fib_complete - fib completion handler
2020-07-13 08:59:44 +01:00
* @ fibptr : FIB to complete
2005-04-16 15:20:36 -07:00
*
* Will do all necessary work to complete a FIB .
*/
2008-01-16 07:39:06 -08:00
2006-02-01 09:30:55 -08:00
int aac_fib_complete ( struct fib * fibptr )
2005-04-16 15:20:36 -07:00
{
2007-03-15 10:26:22 -07:00
struct hw_fib * hw_fib = fibptr - > hw_fib_va ;
2005-04-16 15:20:36 -07:00
2017-02-02 15:53:29 -08:00
if ( fibptr - > flags & FIB_CONTEXT_FLAG_NATIVE_HBA ) {
fib_dealloc ( fibptr ) ;
return 0 ;
}
2005-04-16 15:20:36 -07:00
/*
2017-02-02 15:53:29 -08:00
* Check for a fib which has already been completed or with a
* status wait timeout
2005-04-16 15:20:36 -07:00
*/
2017-02-02 15:53:29 -08:00
if ( hw_fib - > header . XferState = = 0 | | fibptr - > done = = 2 )
2008-01-16 07:39:06 -08:00
return 0 ;
2005-04-16 15:20:36 -07:00
/*
* If we plan to do anything check the structure type first .
2008-01-16 07:39:06 -08:00
*/
2005-04-16 15:20:36 -07:00
2012-07-14 18:18:51 +05:30
if ( hw_fib - > header . StructType ! = FIB_MAGIC & &
hw_fib - > header . StructType ! = FIB_MAGIC2 & &
hw_fib - > header . StructType ! = FIB_MAGIC2_64 )
2008-01-16 07:39:06 -08:00
return - EINVAL ;
2005-04-16 15:20:36 -07:00
/*
2008-01-16 07:39:06 -08:00
* This block completes a cdb which orginated on the host and we
2005-04-16 15:20:36 -07:00
* just need to deallocate the cdb or reinit it . At this point the
* command is complete that we had sent to the adapter and this
* cdb could be reused .
*/
2009-12-21 18:39:27 +05:30
2005-04-16 15:20:36 -07:00
if ( ( hw_fib - > header . XferState & cpu_to_le32 ( SentFromHost ) ) & &
( hw_fib - > header . XferState & cpu_to_le32 ( AdapterProcessed ) ) )
{
fib_dealloc ( fibptr ) ;
}
else if ( hw_fib - > header . XferState & cpu_to_le32 ( SentFromHost ) )
{
/*
* This handles the case when the host has aborted the I / O
* to the adapter because the adapter is not responding
*/
fib_dealloc ( fibptr ) ;
} else if ( hw_fib - > header . XferState & cpu_to_le32 ( HostOwned ) ) {
fib_dealloc ( fibptr ) ;
} else {
BUG ( ) ;
2008-01-16 07:39:06 -08:00
}
2005-04-16 15:20:36 -07:00
return 0 ;
}
/**
* aac_printf - handle printf from firmware
* @ dev : Adapter
* @ val : Message info
*
* Print a message passed to us by the controller firmware on the
* Adaptec board
*/
void aac_printf ( struct aac_dev * dev , u32 val )
{
char * cp = dev - > printfbuf ;
2005-05-16 18:28:42 -07:00
if ( dev - > printf_enabled )
{
int length = val & 0xffff ;
int level = ( val > > 16 ) & 0xffff ;
2008-01-16 07:39:06 -08:00
2005-05-16 18:28:42 -07:00
/*
* The size of the printfbuf is set in port . c
* There is no variable or define for it
*/
if ( length > 255 )
length = 255 ;
if ( cp [ length ] ! = 0 )
cp [ length ] = 0 ;
if ( level = = LOG_AAC_HIGH_ERROR )
2006-03-27 09:44:23 -08:00
printk ( KERN_WARNING " %s:%s " , dev - > name , cp ) ;
2005-05-16 18:28:42 -07:00
else
2006-03-27 09:44:23 -08:00
printk ( KERN_INFO " %s:%s " , dev - > name , cp ) ;
2005-05-16 18:28:42 -07:00
}
2008-01-16 07:39:06 -08:00
memset ( cp , 0 , 256 ) ;
2005-04-16 15:20:36 -07:00
}
2016-04-25 23:32:09 -07:00
static inline int aac_aif_data ( struct aac_aifcmd * aifcmd , uint32_t index )
{
return le32_to_cpu ( ( ( __le32 * ) aifcmd - > data ) [ index ] ) ;
}
static void aac_handle_aif_bu ( struct aac_dev * dev , struct aac_aifcmd * aifcmd )
{
switch ( aac_aif_data ( aifcmd , 1 ) ) {
case AifBuCacheDataLoss :
if ( aac_aif_data ( aifcmd , 2 ) )
dev_info ( & dev - > pdev - > dev , " Backup unit had cache data loss - [%d] \n " ,
aac_aif_data ( aifcmd , 2 ) ) ;
else
dev_info ( & dev - > pdev - > dev , " Backup Unit had cache data loss \n " ) ;
break ;
case AifBuCacheDataRecover :
if ( aac_aif_data ( aifcmd , 2 ) )
dev_info ( & dev - > pdev - > dev , " DDR cache data recovered successfully - [%d] \n " ,
aac_aif_data ( aifcmd , 2 ) ) ;
else
dev_info ( & dev - > pdev - > dev , " DDR cache data recovered successfully \n " ) ;
break ;
}
}
2005-09-26 13:04:56 -07:00
2020-07-13 08:59:44 +01:00
# define AIF_SNIFF_TIMEOUT (500*HZ)
2005-09-26 13:04:56 -07:00
/**
* aac_handle_aif - Handle a message from the firmware
* @ dev : Which adapter this fib is from
* @ fibptr : Pointer to fibptr from adapter
*
* This routine handles a driver notify fib from the adapter and
* dispatches it to the appropriate routine for handling .
*/
static void aac_handle_aif ( struct aac_dev * dev , struct fib * fibptr )
{
2007-03-15 10:26:22 -07:00
struct hw_fib * hw_fib = fibptr - > hw_fib_va ;
2005-09-26 13:04:56 -07:00
struct aac_aifcmd * aifcmd = ( struct aac_aifcmd * ) hw_fib - > data ;
2008-01-11 11:56:07 -08:00
u32 channel , id , lun , container ;
2005-09-26 13:04:56 -07:00
struct scsi_device * device ;
enum {
NOTHING ,
DELETE ,
ADD ,
CHANGE
2008-01-11 11:56:07 -08:00
} device_config_needed = NOTHING ;
2005-09-26 13:04:56 -07:00
/* Sniff for container changes */
2006-08-03 08:02:24 -07:00
if ( ! dev | | ! dev - > fsa_dev )
2005-09-26 13:04:56 -07:00
return ;
2008-01-11 11:56:07 -08:00
container = channel = id = lun = ( u32 ) - 1 ;
2005-09-26 13:04:56 -07:00
/*
* We have set this up to try and minimize the number of
* re - configures that take place . As a result of this when
* certain AIF ' s come in we will set a flag waiting for another
* type of AIF before setting the re - config flag .
*/
switch ( le32_to_cpu ( aifcmd - > command ) ) {
case AifCmdDriverNotify :
2007-11-08 17:27:47 +00:00
switch ( le32_to_cpu ( ( ( __le32 * ) aifcmd - > data ) [ 0 ] ) ) {
2015-03-26 10:41:31 -04:00
case AifRawDeviceRemove :
container = le32_to_cpu ( ( ( __le32 * ) aifcmd - > data ) [ 1 ] ) ;
if ( ( container > > 28 ) ) {
container = ( u32 ) - 1 ;
break ;
}
channel = ( container > > 24 ) & 0xF ;
if ( channel > = dev - > maximum_num_channels ) {
container = ( u32 ) - 1 ;
break ;
}
id = container & 0xFFFF ;
if ( id > = dev - > maximum_num_physicals ) {
container = ( u32 ) - 1 ;
break ;
}
lun = ( container > > 16 ) & 0xFF ;
container = ( u32 ) - 1 ;
channel = aac_phys_to_logical ( channel ) ;
2017-02-02 15:53:29 -08:00
device_config_needed = DELETE ;
2015-03-26 10:41:31 -04:00
break ;
2017-02-02 15:53:29 -08:00
2005-09-26 13:04:56 -07:00
/*
* Morph or Expand complete
*/
case AifDenMorphComplete :
case AifDenVolumeExtendComplete :
2007-11-08 17:27:47 +00:00
container = le32_to_cpu ( ( ( __le32 * ) aifcmd - > data ) [ 1 ] ) ;
2005-09-26 13:04:56 -07:00
if ( container > = dev - > maximum_num_containers )
break ;
/*
2005-10-31 18:32:08 +01:00
* Find the scsi_device associated with the SCSI
2005-09-26 13:04:56 -07:00
* address . Make sure we have the right array , and if
* so set the flag to initiate a new re - config once we
* see an AifEnConfigChange AIF come through .
*/
if ( ( dev ! = NULL ) & & ( dev - > scsi_host_ptr ! = NULL ) ) {
2008-01-16 07:39:06 -08:00
device = scsi_device_lookup ( dev - > scsi_host_ptr ,
CONTAINER_TO_CHANNEL ( container ) ,
CONTAINER_TO_ID ( container ) ,
2005-09-26 13:04:56 -07:00
CONTAINER_TO_LUN ( container ) ) ;
if ( device ) {
dev - > fsa_dev [ container ] . config_needed = CHANGE ;
dev - > fsa_dev [ container ] . config_waiting_on = AifEnConfigChange ;
2006-03-27 09:43:55 -08:00
dev - > fsa_dev [ container ] . config_waiting_stamp = jiffies ;
2005-09-26 13:04:56 -07:00
scsi_device_put ( device ) ;
}
}
}
/*
* If we are waiting on something and this happens to be
* that thing then set the re - configure flag .
*/
if ( container ! = ( u32 ) - 1 ) {
if ( container > = dev - > maximum_num_containers )
break ;
2006-03-27 09:43:55 -08:00
if ( ( dev - > fsa_dev [ container ] . config_waiting_on = =
2007-11-08 17:27:47 +00:00
le32_to_cpu ( * ( __le32 * ) aifcmd - > data ) ) & &
2006-03-27 09:43:55 -08:00
time_before ( jiffies , dev - > fsa_dev [ container ] . config_waiting_stamp + AIF_SNIFF_TIMEOUT ) )
2005-09-26 13:04:56 -07:00
dev - > fsa_dev [ container ] . config_waiting_on = 0 ;
} else for ( container = 0 ;
container < dev - > maximum_num_containers ; + + container ) {
2006-03-27 09:43:55 -08:00
if ( ( dev - > fsa_dev [ container ] . config_waiting_on = =
2007-11-08 17:27:47 +00:00
le32_to_cpu ( * ( __le32 * ) aifcmd - > data ) ) & &
2006-03-27 09:43:55 -08:00
time_before ( jiffies , dev - > fsa_dev [ container ] . config_waiting_stamp + AIF_SNIFF_TIMEOUT ) )
2005-09-26 13:04:56 -07:00
dev - > fsa_dev [ container ] . config_waiting_on = 0 ;
}
break ;
case AifCmdEventNotify :
2007-11-08 17:27:47 +00:00
switch ( le32_to_cpu ( ( ( __le32 * ) aifcmd - > data ) [ 0 ] ) ) {
2008-01-08 12:01:07 -08:00
case AifEnBatteryEvent :
dev - > cache_protected =
( ( ( __le32 * ) aifcmd - > data ) [ 1 ] = = cpu_to_le32 ( 3 ) ) ;
break ;
2005-09-26 13:04:56 -07:00
/*
* Add an Array .
*/
case AifEnAddContainer :
2007-11-08 17:27:47 +00:00
container = le32_to_cpu ( ( ( __le32 * ) aifcmd - > data ) [ 1 ] ) ;
2005-09-26 13:04:56 -07:00
if ( container > = dev - > maximum_num_containers )
break ;
dev - > fsa_dev [ container ] . config_needed = ADD ;
dev - > fsa_dev [ container ] . config_waiting_on =
AifEnConfigChange ;
2006-03-27 09:43:55 -08:00
dev - > fsa_dev [ container ] . config_waiting_stamp = jiffies ;
2005-09-26 13:04:56 -07:00
break ;
/*
* Delete an Array .
*/
case AifEnDeleteContainer :
2007-11-08 17:27:47 +00:00
container = le32_to_cpu ( ( ( __le32 * ) aifcmd - > data ) [ 1 ] ) ;
2005-09-26 13:04:56 -07:00
if ( container > = dev - > maximum_num_containers )
break ;
dev - > fsa_dev [ container ] . config_needed = DELETE ;
dev - > fsa_dev [ container ] . config_waiting_on =
AifEnConfigChange ;
2006-03-27 09:43:55 -08:00
dev - > fsa_dev [ container ] . config_waiting_stamp = jiffies ;
2005-09-26 13:04:56 -07:00
break ;
/*
* Container change detected . If we currently are not
* waiting on something else , setup to wait on a Config Change .
*/
case AifEnContainerChange :
2007-11-08 17:27:47 +00:00
container = le32_to_cpu ( ( ( __le32 * ) aifcmd - > data ) [ 1 ] ) ;
2005-09-26 13:04:56 -07:00
if ( container > = dev - > maximum_num_containers )
break ;
2006-03-27 09:43:55 -08:00
if ( dev - > fsa_dev [ container ] . config_waiting_on & &
time_before ( jiffies , dev - > fsa_dev [ container ] . config_waiting_stamp + AIF_SNIFF_TIMEOUT ) )
2005-09-26 13:04:56 -07:00
break ;
dev - > fsa_dev [ container ] . config_needed = CHANGE ;
dev - > fsa_dev [ container ] . config_waiting_on =
AifEnConfigChange ;
2006-03-27 09:43:55 -08:00
dev - > fsa_dev [ container ] . config_waiting_stamp = jiffies ;
2005-09-26 13:04:56 -07:00
break ;
case AifEnConfigChange :
break ;
2008-01-17 09:25:07 -08:00
case AifEnAddJBOD :
case AifEnDeleteJBOD :
container = le32_to_cpu ( ( ( __le32 * ) aifcmd - > data ) [ 1 ] ) ;
2008-04-30 15:47:35 -04:00
if ( ( container > > 28 ) ) {
container = ( u32 ) - 1 ;
2008-01-17 09:25:07 -08:00
break ;
2008-04-30 15:47:35 -04:00
}
2008-01-17 09:25:07 -08:00
channel = ( container > > 24 ) & 0xF ;
2008-04-30 15:47:35 -04:00
if ( channel > = dev - > maximum_num_channels ) {
container = ( u32 ) - 1 ;
2008-01-17 09:25:07 -08:00
break ;
2008-04-30 15:47:35 -04:00
}
2008-01-17 09:25:07 -08:00
id = container & 0xFFFF ;
2008-04-30 15:47:35 -04:00
if ( id > = dev - > maximum_num_physicals ) {
container = ( u32 ) - 1 ;
2008-01-17 09:25:07 -08:00
break ;
2008-04-30 15:47:35 -04:00
}
2008-01-17 09:25:07 -08:00
lun = ( container > > 16 ) & 0xFF ;
2008-04-30 15:47:35 -04:00
container = ( u32 ) - 1 ;
2008-01-17 09:25:07 -08:00
channel = aac_phys_to_logical ( channel ) ;
device_config_needed =
( ( ( __le32 * ) aifcmd - > data ) [ 0 ] = =
cpu_to_le32 ( AifEnAddJBOD ) ) ? ADD : DELETE ;
2010-05-10 04:05:50 -07:00
if ( device_config_needed = = ADD ) {
device = scsi_device_lookup ( dev - > scsi_host_ptr ,
channel ,
id ,
lun ) ;
if ( device ) {
scsi_remove_device ( device ) ;
scsi_device_put ( device ) ;
}
}
2008-01-17 09:25:07 -08:00
break ;
2008-01-11 11:56:07 -08:00
case AifEnEnclosureManagement :
2008-01-17 09:25:07 -08:00
/*
* If in JBOD mode , automatic exposure of new
* physical target to be suppressed until configured .
*/
if ( dev - > jbod )
break ;
2008-01-11 11:56:07 -08:00
switch ( le32_to_cpu ( ( ( __le32 * ) aifcmd - > data ) [ 3 ] ) ) {
case EM_DRIVE_INSERTION :
case EM_DRIVE_REMOVAL :
2015-03-26 10:41:22 -04:00
case EM_SES_DRIVE_INSERTION :
case EM_SES_DRIVE_REMOVAL :
2008-01-11 11:56:07 -08:00
container = le32_to_cpu (
( ( __le32 * ) aifcmd - > data ) [ 2 ] ) ;
2008-04-30 15:47:35 -04:00
if ( ( container > > 28 ) ) {
container = ( u32 ) - 1 ;
2008-01-11 11:56:07 -08:00
break ;
2008-04-30 15:47:35 -04:00
}
2008-01-11 11:56:07 -08:00
channel = ( container > > 24 ) & 0xF ;
2008-04-30 15:47:35 -04:00
if ( channel > = dev - > maximum_num_channels ) {
container = ( u32 ) - 1 ;
2008-01-11 11:56:07 -08:00
break ;
2008-04-30 15:47:35 -04:00
}
2008-01-11 11:56:07 -08:00
id = container & 0xFFFF ;
lun = ( container > > 16 ) & 0xFF ;
2008-04-30 15:47:35 -04:00
container = ( u32 ) - 1 ;
2008-01-11 11:56:07 -08:00
if ( id > = dev - > maximum_num_physicals ) {
/* legacy dev_t ? */
if ( ( 0x2000 < = id ) | | lun | | channel | |
( ( channel = ( id > > 7 ) & 0x3F ) > =
dev - > maximum_num_channels ) )
break ;
lun = ( id > > 4 ) & 7 ;
id & = 0xF ;
}
channel = aac_phys_to_logical ( channel ) ;
device_config_needed =
2015-03-26 10:41:22 -04:00
( ( ( ( __le32 * ) aifcmd - > data ) [ 3 ]
= = cpu_to_le32 ( EM_DRIVE_INSERTION ) ) | |
( ( ( __le32 * ) aifcmd - > data ) [ 3 ]
= = cpu_to_le32 ( EM_SES_DRIVE_INSERTION ) ) ) ?
2008-01-11 11:56:07 -08:00
ADD : DELETE ;
break ;
}
2019-02-15 15:42:42 -06:00
break ;
case AifBuManagerEvent :
aac_handle_aif_bu ( dev , aifcmd ) ;
2008-01-11 11:56:07 -08:00
break ;
2005-09-26 13:04:56 -07:00
}
/*
* If we are waiting on something and this happens to be
* that thing then set the re - configure flag .
*/
if ( container ! = ( u32 ) - 1 ) {
if ( container > = dev - > maximum_num_containers )
break ;
2006-03-27 09:43:55 -08:00
if ( ( dev - > fsa_dev [ container ] . config_waiting_on = =
2007-11-08 17:27:47 +00:00
le32_to_cpu ( * ( __le32 * ) aifcmd - > data ) ) & &
2006-03-27 09:43:55 -08:00
time_before ( jiffies , dev - > fsa_dev [ container ] . config_waiting_stamp + AIF_SNIFF_TIMEOUT ) )
2005-09-26 13:04:56 -07:00
dev - > fsa_dev [ container ] . config_waiting_on = 0 ;
} else for ( container = 0 ;
container < dev - > maximum_num_containers ; + + container ) {
2006-03-27 09:43:55 -08:00
if ( ( dev - > fsa_dev [ container ] . config_waiting_on = =
2007-11-08 17:27:47 +00:00
le32_to_cpu ( * ( __le32 * ) aifcmd - > data ) ) & &
2006-03-27 09:43:55 -08:00
time_before ( jiffies , dev - > fsa_dev [ container ] . config_waiting_stamp + AIF_SNIFF_TIMEOUT ) )
2005-09-26 13:04:56 -07:00
dev - > fsa_dev [ container ] . config_waiting_on = 0 ;
}
break ;
case AifCmdJobProgress :
/*
* These are job progress AIF ' s . When a Clear is being
* done on a container it is initially created then hidden from
* the OS . When the clear completes we don ' t get a config
* change so we monitor the job status complete on a clear then
* wait for a container change .
*/
2007-11-08 17:27:47 +00:00
if ( ( ( __le32 * ) aifcmd - > data ) [ 1 ] = = cpu_to_le32 ( AifJobCtrZero ) & &
( ( ( __le32 * ) aifcmd - > data ) [ 6 ] = = ( ( __le32 * ) aifcmd - > data ) [ 5 ] | |
( ( __le32 * ) aifcmd - > data ) [ 4 ] = = cpu_to_le32 ( AifJobStsSuccess ) ) ) {
2005-09-26 13:04:56 -07:00
for ( container = 0 ;
container < dev - > maximum_num_containers ;
+ + container ) {
/*
* Stomp on all config sequencing for all
* containers ?
*/
dev - > fsa_dev [ container ] . config_waiting_on =
AifEnContainerChange ;
dev - > fsa_dev [ container ] . config_needed = ADD ;
2006-03-27 09:43:55 -08:00
dev - > fsa_dev [ container ] . config_waiting_stamp =
jiffies ;
2005-09-26 13:04:56 -07:00
}
}
2007-11-08 17:27:47 +00:00
if ( ( ( __le32 * ) aifcmd - > data ) [ 1 ] = = cpu_to_le32 ( AifJobCtrZero ) & &
( ( __le32 * ) aifcmd - > data ) [ 6 ] = = 0 & &
( ( __le32 * ) aifcmd - > data ) [ 4 ] = = cpu_to_le32 ( AifJobStsRunning ) ) {
2005-09-26 13:04:56 -07:00
for ( container = 0 ;
container < dev - > maximum_num_containers ;
+ + container ) {
/*
* Stomp on all config sequencing for all
* containers ?
*/
dev - > fsa_dev [ container ] . config_waiting_on =
AifEnContainerChange ;
dev - > fsa_dev [ container ] . config_needed = DELETE ;
2006-03-27 09:43:55 -08:00
dev - > fsa_dev [ container ] . config_waiting_stamp =
jiffies ;
2005-09-26 13:04:56 -07:00
}
}
break ;
}
2008-04-30 15:47:35 -04:00
container = 0 ;
retry_next :
2019-02-02 10:40:52 +00:00
if ( device_config_needed = = NOTHING ) {
for ( ; container < dev - > maximum_num_containers ; + + container ) {
if ( ( dev - > fsa_dev [ container ] . config_waiting_on = = 0 ) & &
( dev - > fsa_dev [ container ] . config_needed ! = NOTHING ) & &
time_before ( jiffies , dev - > fsa_dev [ container ] . config_waiting_stamp + AIF_SNIFF_TIMEOUT ) ) {
device_config_needed =
dev - > fsa_dev [ container ] . config_needed ;
dev - > fsa_dev [ container ] . config_needed = NOTHING ;
channel = CONTAINER_TO_CHANNEL ( container ) ;
id = CONTAINER_TO_ID ( container ) ;
lun = CONTAINER_TO_LUN ( container ) ;
break ;
}
2005-09-26 13:04:56 -07:00
}
}
if ( device_config_needed = = NOTHING )
return ;
/*
* If we decided that a re - configuration needs to be done ,
* schedule it here on the way out the door , please close the door
* behind you .
*/
/*
2005-10-31 18:32:08 +01:00
* Find the scsi_device associated with the SCSI address ,
2005-09-26 13:04:56 -07:00
* and mark it as changed , invalidating the cache . This deals
* with changes to existing device IDs .
*/
if ( ! dev | | ! dev - > scsi_host_ptr )
return ;
/*
2006-02-01 09:30:55 -08:00
* force reload of disk info via aac_probe_container
2005-09-26 13:04:56 -07:00
*/
2008-01-11 11:56:07 -08:00
if ( ( channel = = CONTAINER_CHANNEL ) & &
( device_config_needed ! = NOTHING ) ) {
if ( dev - > fsa_dev [ container ] . valid = = 1 )
dev - > fsa_dev [ container ] . valid = 2 ;
2006-02-01 09:30:55 -08:00
aac_probe_container ( dev , container ) ;
2008-01-11 11:56:07 -08:00
}
device = scsi_device_lookup ( dev - > scsi_host_ptr , channel , id , lun ) ;
2005-09-26 13:04:56 -07:00
if ( device ) {
switch ( device_config_needed ) {
case DELETE :
2010-05-10 04:29:25 -07:00
# if (defined(AAC_DEBUG_INSTRUMENT_AIF_DELETE))
scsi_remove_device ( device ) ;
# else
2008-01-11 11:56:07 -08:00
if ( scsi_device_online ( device ) ) {
scsi_device_set_state ( device , SDEV_OFFLINE ) ;
sdev_printk ( KERN_INFO , device ,
" Device offlined - %s \n " ,
( channel = = CONTAINER_CHANNEL ) ?
" array deleted " :
" enclosure services event " ) ;
}
2010-05-10 04:29:25 -07:00
# endif
2008-01-11 11:56:07 -08:00
break ;
case ADD :
if ( ! scsi_device_online ( device ) ) {
sdev_printk ( KERN_INFO , device ,
" Device online - %s \n " ,
( channel = = CONTAINER_CHANNEL ) ?
" array created " :
" enclosure services event " ) ;
scsi_device_set_state ( device , SDEV_RUNNING ) ;
}
2020-08-23 17:36:59 -05:00
fallthrough ;
2005-09-26 13:04:56 -07:00
case CHANGE :
2008-01-11 11:56:07 -08:00
if ( ( channel = = CONTAINER_CHANNEL )
& & ( ! dev - > fsa_dev [ container ] . valid ) ) {
2010-05-10 04:29:25 -07:00
# if (defined(AAC_DEBUG_INSTRUMENT_AIF_DELETE))
scsi_remove_device ( device ) ;
# else
2008-01-11 11:56:07 -08:00
if ( ! scsi_device_online ( device ) )
break ;
scsi_device_set_state ( device , SDEV_OFFLINE ) ;
sdev_printk ( KERN_INFO , device ,
" Device offlined - %s \n " ,
" array failed " ) ;
2010-05-10 04:29:25 -07:00
# endif
2008-01-11 11:56:07 -08:00
break ;
}
2005-09-26 13:04:56 -07:00
scsi_rescan_device ( & device - > sdev_gendev ) ;
2020-11-20 12:39:15 -06:00
break ;
2005-09-26 13:04:56 -07:00
default :
break ;
}
scsi_device_put ( device ) ;
2008-01-11 11:56:07 -08:00
device_config_needed = NOTHING ;
2005-09-26 13:04:56 -07:00
}
2008-01-11 11:56:07 -08:00
if ( device_config_needed = = ADD )
scsi_add_device ( dev - > scsi_host_ptr , channel , id , lun ) ;
2008-04-30 15:47:35 -04:00
if ( channel = = CONTAINER_CHANNEL ) {
container + + ;
device_config_needed = NOTHING ;
goto retry_next ;
}
2005-09-26 13:04:56 -07:00
}
2019-10-15 11:52:03 +05:30
static void aac_schedule_bus_scan ( struct aac_dev * aac )
{
if ( aac - > sa_firmware )
aac_schedule_safw_scan_worker ( aac ) ;
else
aac_schedule_src_reinit_aif_worker ( aac ) ;
}
2017-02-02 15:53:33 -08:00
static int _aac_reset_adapter ( struct aac_dev * aac , int forced , u8 reset_type )
2006-08-03 08:03:30 -07:00
{
int index , quirks ;
2015-08-28 06:38:38 -04:00
int retval ;
2020-02-28 08:53:12 +01:00
struct Scsi_Host * host = aac - > scsi_host_ptr ;
2007-06-12 09:33:54 -04:00
int jafo = 0 ;
2017-02-02 15:53:33 -08:00
int bled ;
2017-05-10 09:39:36 -07:00
u64 dmamask ;
2017-05-10 09:39:49 -07:00
int num_of_fibs = 0 ;
2006-08-03 08:03:30 -07:00
/*
* Assumptions :
2007-06-12 09:33:54 -04:00
* - host is locked , unless called by the aacraid thread .
* ( a matter of convenience , due to legacy issues surrounding
* eh_host_adapter_reset ) .
2006-08-03 08:03:30 -07:00
* - in_reset is asserted , so no new i / o is getting to the
* card .
2007-06-12 09:33:54 -04:00
* - The card is dead , or will be very shortly ; - / so no new
* commands are completing in the interrupt service .
2006-08-03 08:03:30 -07:00
*/
aac_adapter_disable_int ( aac ) ;
2018-04-03 15:50:42 -06:00
if ( aac - > thread & & aac - > thread - > pid ! = current - > pid ) {
2007-06-12 09:33:54 -04:00
spin_unlock_irq ( host - > host_lock ) ;
kthread_stop ( aac - > thread ) ;
2018-04-03 15:50:42 -06:00
aac - > thread = NULL ;
2007-06-12 09:33:54 -04:00
jafo = 1 ;
}
2006-08-03 08:03:30 -07:00
/*
* If a positive health , means in a known DEAD PANIC
* state and the adapter could be reset to ` try again ' .
*/
2017-02-02 15:53:33 -08:00
bled = forced ? 0 : aac_adapter_check_health ( aac ) ;
retval = aac_adapter_restart ( aac , bled , reset_type ) ;
2006-08-03 08:03:30 -07:00
if ( retval )
goto out ;
2006-11-21 10:40:31 -08:00
/*
* Loop through the fibs , close the synchronous FIBS
*/
2017-05-10 09:39:49 -07:00
retval = 1 ;
num_of_fibs = aac - > scsi_host_ptr - > can_queue + AAC_NUM_MGT_FIB ;
for ( index = 0 ; index < num_of_fibs ; index + + ) {
2006-11-21 10:40:31 -08:00
struct fib * fib = & aac - > fibs [ index ] ;
2017-05-10 09:39:49 -07:00
__le32 XferState = fib - > hw_fib_va - > header . XferState ;
bool is_response_expected = false ;
if ( ! ( XferState & cpu_to_le32 ( NoResponseExpected | Async ) ) & &
( XferState & cpu_to_le32 ( ResponseExpected ) ) )
is_response_expected = true ;
if ( is_response_expected
| | fib - > flags & FIB_CONTEXT_FLAG_WAIT ) {
2006-11-21 10:40:31 -08:00
unsigned long flagv ;
spin_lock_irqsave ( & fib - > event_lock , flagv ) ;
2018-12-10 22:32:41 +01:00
complete ( & fib - > event_wait ) ;
2006-11-21 10:40:31 -08:00
spin_unlock_irqrestore ( & fib - > event_lock , flagv ) ;
schedule ( ) ;
2007-03-15 10:27:21 -07:00
retval = 0 ;
2006-11-21 10:40:31 -08:00
}
}
2007-03-15 10:27:21 -07:00
/* Give some extra time for ioctls to complete. */
if ( retval = = 0 )
ssleep ( 2 ) ;
2006-08-03 08:03:30 -07:00
index = aac - > cardtype ;
/*
* Re - initialize the adapter , first free resources , then carefully
* apply the initialization sequence to come back again . Only risk
* is a change in Firmware dropping cache , it is assumed the caller
* will ensure that i / o is queisced and the card is flushed in that
* case .
*/
scsi: aacraid: Prevent crash in case of free interrupt during scsi EH path
As part of the scsi EH path, aacraid performs a reinitialization of the
adapter, which encompass freeing resources and IRQs, NULLifying lots of
pointers, and then initialize it all over again. We've identified a
problem during the free IRQ portion of this path if CONFIG_DEBUG_SHIRQ
is enabled on kernel config file.
Happens that, in case this flag was set, right after free_irq()
effectively clears the interrupt, it checks if it was requested as
IRQF_SHARED. In positive case, it performs another call to the IRQ
handler on driver. Problem is: since aacraid currently free some
resources *before* freeing the IRQ, once free_irq() path calls the
handler again (due to CONFIG_DEBUG_SHIRQ), aacraid crashes due to NULL
pointer dereference with the following trace:
aac_src_intr_message+0xf8/0x740 [aacraid]
__free_irq+0x33c/0x4a0
free_irq+0x78/0xb0
aac_free_irq+0x13c/0x150 [aacraid]
aac_reset_adapter+0x2e8/0x970 [aacraid]
aac_eh_reset+0x3a8/0x5d0 [aacraid]
scsi_try_host_reset+0x74/0x180
scsi_eh_ready_devs+0xc70/0x1510
scsi_error_handler+0x624/0xa20
This patch prevents the crash by changing the order of the
deinitialization in this path of aacraid: first we clear the IRQ, then
we free other resources. No functional change intended.
Signed-off-by: Guilherme G. Piccoli <gpiccoli@linux.vnet.ibm.com>
Reviewed-by: Raghava Aditya Renukunta <RaghavaAditya.Renukunta@microsemi.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
2017-11-17 19:14:55 -02:00
aac_free_irq ( aac ) ;
2006-08-03 08:03:30 -07:00
aac_fib_map_free ( aac ) ;
2017-04-05 16:14:16 +05:30
dma_free_coherent ( & aac - > pdev - > dev , aac - > comm_size , aac - > comm_addr ,
aac - > comm_phys ) ;
2020-09-26 17:00:15 +02:00
aac_adapter_ioremap ( aac , 0 ) ;
2006-08-03 08:03:30 -07:00
aac - > comm_addr = NULL ;
aac - > comm_phys = 0 ;
kfree ( aac - > queues ) ;
aac - > queues = NULL ;
kfree ( aac - > fsa_dev ) ;
aac - > fsa_dev = NULL ;
2017-05-10 09:39:36 -07:00
dmamask = DMA_BIT_MASK ( 32 ) ;
2007-12-13 16:14:18 -08:00
quirks = aac_get_driver_ident ( index ) - > quirks ;
2017-05-10 09:39:36 -07:00
if ( quirks & AAC_QUIRK_31BIT )
scsi: aacraid: Remove pci-dma-compat wrapper API
The legacy API wrappers in include/linux/pci-dma-compat.h should go away as
they create unnecessary midlayering for include/linux/dma-mapping.h API.
Instead use dma-mapping.h API directly.
The patch has been generated with the coccinelle script below.
Compile-tested.
@@@@
- PCI_DMA_BIDIRECTIONAL
+ DMA_BIDIRECTIONAL
@@@@
- PCI_DMA_TODEVICE
+ DMA_TO_DEVICE
@@@@
- PCI_DMA_FROMDEVICE
+ DMA_FROM_DEVICE
@@@@
- PCI_DMA_NONE
+ DMA_NONE
@@ expression E1, E2, E3; @@
- pci_alloc_consistent(E1, E2, E3)
+ dma_alloc_coherent(&E1->dev, E2, E3, GFP_)
@@ expression E1, E2, E3; @@
- pci_zalloc_consistent(E1, E2, E3)
+ dma_alloc_coherent(&E1->dev, E2, E3, GFP_)
@@ expression E1, E2, E3, E4; @@
- pci_free_consistent(E1, E2, E3, E4)
+ dma_free_coherent(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_map_single(E1, E2, E3, E4)
+ dma_map_single(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_single(E1, E2, E3, E4)
+ dma_unmap_single(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4, E5; @@
- pci_map_page(E1, E2, E3, E4, E5)
+ dma_map_page(&E1->dev, E2, E3, E4, E5)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_page(E1, E2, E3, E4)
+ dma_unmap_page(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_map_sg(E1, E2, E3, E4)
+ dma_map_sg(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_sg(E1, E2, E3, E4)
+ dma_unmap_sg(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_single_for_cpu(E1, E2, E3, E4)
+ dma_sync_single_for_cpu(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_single_for_device(E1, E2, E3, E4)
+ dma_sync_single_for_device(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_sg_for_cpu(E1, E2, E3, E4)
+ dma_sync_sg_for_cpu(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_sg_for_device(E1, E2, E3, E4)
+ dma_sync_sg_for_device(&E1->dev, E2, E3, E4)
@@ expression E1, E2; @@
- pci_dma_mapping_error(E1, E2)
+ dma_mapping_error(&E1->dev, E2)
@@ expression E1, E2; @@
- pci_set_consistent_dma_mask(E1, E2)
+ dma_set_coherent_mask(&E1->dev, E2)
@@ expression E1, E2; @@
- pci_set_dma_mask(E1, E2)
+ dma_set_mask(&E1->dev, E2)
Link: https://lore.kernel.org/r/f8d4778440d55ba26c04eef0f7d63fb211a39443.1596045683.git.usuraj35@gmail.com
Signed-off-by: Suraj Upadhyay <usuraj35@gmail.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
2020-07-29 23:36:27 +05:30
retval = dma_set_mask ( & aac - > pdev - > dev , dmamask ) ;
2017-05-10 09:39:36 -07:00
else if ( ! ( quirks & AAC_QUIRK_SRC ) )
scsi: aacraid: Remove pci-dma-compat wrapper API
The legacy API wrappers in include/linux/pci-dma-compat.h should go away as
they create unnecessary midlayering for include/linux/dma-mapping.h API.
Instead use dma-mapping.h API directly.
The patch has been generated with the coccinelle script below.
Compile-tested.
@@@@
- PCI_DMA_BIDIRECTIONAL
+ DMA_BIDIRECTIONAL
@@@@
- PCI_DMA_TODEVICE
+ DMA_TO_DEVICE
@@@@
- PCI_DMA_FROMDEVICE
+ DMA_FROM_DEVICE
@@@@
- PCI_DMA_NONE
+ DMA_NONE
@@ expression E1, E2, E3; @@
- pci_alloc_consistent(E1, E2, E3)
+ dma_alloc_coherent(&E1->dev, E2, E3, GFP_)
@@ expression E1, E2, E3; @@
- pci_zalloc_consistent(E1, E2, E3)
+ dma_alloc_coherent(&E1->dev, E2, E3, GFP_)
@@ expression E1, E2, E3, E4; @@
- pci_free_consistent(E1, E2, E3, E4)
+ dma_free_coherent(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_map_single(E1, E2, E3, E4)
+ dma_map_single(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_single(E1, E2, E3, E4)
+ dma_unmap_single(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4, E5; @@
- pci_map_page(E1, E2, E3, E4, E5)
+ dma_map_page(&E1->dev, E2, E3, E4, E5)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_page(E1, E2, E3, E4)
+ dma_unmap_page(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_map_sg(E1, E2, E3, E4)
+ dma_map_sg(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_sg(E1, E2, E3, E4)
+ dma_unmap_sg(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_single_for_cpu(E1, E2, E3, E4)
+ dma_sync_single_for_cpu(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_single_for_device(E1, E2, E3, E4)
+ dma_sync_single_for_device(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_sg_for_cpu(E1, E2, E3, E4)
+ dma_sync_sg_for_cpu(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_sg_for_device(E1, E2, E3, E4)
+ dma_sync_sg_for_device(&E1->dev, E2, E3, E4)
@@ expression E1, E2; @@
- pci_dma_mapping_error(E1, E2)
+ dma_mapping_error(&E1->dev, E2)
@@ expression E1, E2; @@
- pci_set_consistent_dma_mask(E1, E2)
+ dma_set_coherent_mask(&E1->dev, E2)
@@ expression E1, E2; @@
- pci_set_dma_mask(E1, E2)
+ dma_set_mask(&E1->dev, E2)
Link: https://lore.kernel.org/r/f8d4778440d55ba26c04eef0f7d63fb211a39443.1596045683.git.usuraj35@gmail.com
Signed-off-by: Suraj Upadhyay <usuraj35@gmail.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
2020-07-29 23:36:27 +05:30
retval = dma_set_mask ( & aac - > pdev - > dev , dmamask ) ;
2017-05-10 09:39:36 -07:00
else
scsi: aacraid: Remove pci-dma-compat wrapper API
The legacy API wrappers in include/linux/pci-dma-compat.h should go away as
they create unnecessary midlayering for include/linux/dma-mapping.h API.
Instead use dma-mapping.h API directly.
The patch has been generated with the coccinelle script below.
Compile-tested.
@@@@
- PCI_DMA_BIDIRECTIONAL
+ DMA_BIDIRECTIONAL
@@@@
- PCI_DMA_TODEVICE
+ DMA_TO_DEVICE
@@@@
- PCI_DMA_FROMDEVICE
+ DMA_FROM_DEVICE
@@@@
- PCI_DMA_NONE
+ DMA_NONE
@@ expression E1, E2, E3; @@
- pci_alloc_consistent(E1, E2, E3)
+ dma_alloc_coherent(&E1->dev, E2, E3, GFP_)
@@ expression E1, E2, E3; @@
- pci_zalloc_consistent(E1, E2, E3)
+ dma_alloc_coherent(&E1->dev, E2, E3, GFP_)
@@ expression E1, E2, E3, E4; @@
- pci_free_consistent(E1, E2, E3, E4)
+ dma_free_coherent(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_map_single(E1, E2, E3, E4)
+ dma_map_single(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_single(E1, E2, E3, E4)
+ dma_unmap_single(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4, E5; @@
- pci_map_page(E1, E2, E3, E4, E5)
+ dma_map_page(&E1->dev, E2, E3, E4, E5)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_page(E1, E2, E3, E4)
+ dma_unmap_page(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_map_sg(E1, E2, E3, E4)
+ dma_map_sg(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_sg(E1, E2, E3, E4)
+ dma_unmap_sg(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_single_for_cpu(E1, E2, E3, E4)
+ dma_sync_single_for_cpu(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_single_for_device(E1, E2, E3, E4)
+ dma_sync_single_for_device(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_sg_for_cpu(E1, E2, E3, E4)
+ dma_sync_sg_for_cpu(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_sg_for_device(E1, E2, E3, E4)
+ dma_sync_sg_for_device(&E1->dev, E2, E3, E4)
@@ expression E1, E2; @@
- pci_dma_mapping_error(E1, E2)
+ dma_mapping_error(&E1->dev, E2)
@@ expression E1, E2; @@
- pci_set_consistent_dma_mask(E1, E2)
+ dma_set_coherent_mask(&E1->dev, E2)
@@ expression E1, E2; @@
- pci_set_dma_mask(E1, E2)
+ dma_set_mask(&E1->dev, E2)
Link: https://lore.kernel.org/r/f8d4778440d55ba26c04eef0f7d63fb211a39443.1596045683.git.usuraj35@gmail.com
Signed-off-by: Suraj Upadhyay <usuraj35@gmail.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
2020-07-29 23:36:27 +05:30
retval = dma_set_coherent_mask ( & aac - > pdev - > dev , dmamask ) ;
2017-05-10 09:39:36 -07:00
if ( quirks & AAC_QUIRK_31BIT & & ! retval ) {
dmamask = DMA_BIT_MASK ( 31 ) ;
scsi: aacraid: Remove pci-dma-compat wrapper API
The legacy API wrappers in include/linux/pci-dma-compat.h should go away as
they create unnecessary midlayering for include/linux/dma-mapping.h API.
Instead use dma-mapping.h API directly.
The patch has been generated with the coccinelle script below.
Compile-tested.
@@@@
- PCI_DMA_BIDIRECTIONAL
+ DMA_BIDIRECTIONAL
@@@@
- PCI_DMA_TODEVICE
+ DMA_TO_DEVICE
@@@@
- PCI_DMA_FROMDEVICE
+ DMA_FROM_DEVICE
@@@@
- PCI_DMA_NONE
+ DMA_NONE
@@ expression E1, E2, E3; @@
- pci_alloc_consistent(E1, E2, E3)
+ dma_alloc_coherent(&E1->dev, E2, E3, GFP_)
@@ expression E1, E2, E3; @@
- pci_zalloc_consistent(E1, E2, E3)
+ dma_alloc_coherent(&E1->dev, E2, E3, GFP_)
@@ expression E1, E2, E3, E4; @@
- pci_free_consistent(E1, E2, E3, E4)
+ dma_free_coherent(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_map_single(E1, E2, E3, E4)
+ dma_map_single(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_single(E1, E2, E3, E4)
+ dma_unmap_single(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4, E5; @@
- pci_map_page(E1, E2, E3, E4, E5)
+ dma_map_page(&E1->dev, E2, E3, E4, E5)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_page(E1, E2, E3, E4)
+ dma_unmap_page(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_map_sg(E1, E2, E3, E4)
+ dma_map_sg(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_unmap_sg(E1, E2, E3, E4)
+ dma_unmap_sg(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_single_for_cpu(E1, E2, E3, E4)
+ dma_sync_single_for_cpu(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_single_for_device(E1, E2, E3, E4)
+ dma_sync_single_for_device(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_sg_for_cpu(E1, E2, E3, E4)
+ dma_sync_sg_for_cpu(&E1->dev, E2, E3, E4)
@@ expression E1, E2, E3, E4; @@
- pci_dma_sync_sg_for_device(E1, E2, E3, E4)
+ dma_sync_sg_for_device(&E1->dev, E2, E3, E4)
@@ expression E1, E2; @@
- pci_dma_mapping_error(E1, E2)
+ dma_mapping_error(&E1->dev, E2)
@@ expression E1, E2; @@
- pci_set_consistent_dma_mask(E1, E2)
+ dma_set_coherent_mask(&E1->dev, E2)
@@ expression E1, E2; @@
- pci_set_dma_mask(E1, E2)
+ dma_set_mask(&E1->dev, E2)
Link: https://lore.kernel.org/r/f8d4778440d55ba26c04eef0f7d63fb211a39443.1596045683.git.usuraj35@gmail.com
Signed-off-by: Suraj Upadhyay <usuraj35@gmail.com>
Signed-off-by: Martin K. Petersen <martin.petersen@oracle.com>
2020-07-29 23:36:27 +05:30
retval = dma_set_coherent_mask ( & aac - > pdev - > dev , dmamask ) ;
2006-08-03 08:03:30 -07:00
}
2017-05-10 09:39:36 -07:00
if ( retval )
goto out ;
2006-08-03 08:03:30 -07:00
if ( ( retval = ( * ( aac_get_driver_ident ( index ) - > init ) ) ( aac ) ) )
goto out ;
2017-05-10 09:39:36 -07:00
2007-06-12 09:33:54 -04:00
if ( jafo ) {
2013-07-03 15:04:58 -07:00
aac - > thread = kthread_run ( aac_command_thread , aac , " %s " ,
aac - > name ) ;
2007-06-12 09:33:54 -04:00
if ( IS_ERR ( aac - > thread ) ) {
retval = PTR_ERR ( aac - > thread ) ;
2018-04-03 15:50:42 -06:00
aac - > thread = NULL ;
2007-06-12 09:33:54 -04:00
goto out ;
}
2006-08-03 08:03:30 -07:00
}
( void ) aac_get_adapter_info ( aac ) ;
if ( ( quirks & AAC_QUIRK_34SG ) & & ( host - > sg_tablesize > 34 ) ) {
2008-01-16 07:39:06 -08:00
host - > sg_tablesize = 34 ;
host - > max_sectors = ( host - > sg_tablesize * 8 ) + 112 ;
}
if ( ( quirks & AAC_QUIRK_17SG ) & & ( host - > sg_tablesize > 17 ) ) {
host - > sg_tablesize = 17 ;
host - > max_sectors = ( host - > sg_tablesize * 8 ) + 112 ;
}
2006-08-03 08:03:30 -07:00
aac_get_config_status ( aac , 1 ) ;
aac_get_containers ( aac ) ;
/*
* This is where the assumption that the Adapter is quiesced
* is important .
*/
2020-02-28 08:53:10 +01:00
scsi_host_complete_all_commands ( host , DID_RESET ) ;
2006-08-03 08:03:30 -07:00
retval = 0 ;
out :
aac - > in_reset = 0 ;
2017-12-26 20:34:24 -08:00
2017-02-16 12:51:18 -08:00
/*
* Issue bus rescan to catch any configuration that might have
* occurred
*/
2017-12-26 20:34:48 -08:00
if ( ! retval & & ! is_kdump_kernel ( ) ) {
2017-12-26 20:34:47 -08:00
dev_info ( & aac - > pdev - > dev , " Scheduling bus rescan \n " ) ;
2019-10-15 11:52:03 +05:30
aac_schedule_bus_scan ( aac ) ;
2017-02-16 12:51:18 -08:00
}
2017-12-26 20:34:47 -08:00
2007-06-12 09:33:54 -04:00
if ( jafo ) {
spin_lock_irq ( host - > host_lock ) ;
}
return retval ;
}
2017-02-02 15:53:33 -08:00
int aac_reset_adapter ( struct aac_dev * aac , int forced , u8 reset_type )
2007-06-12 09:33:54 -04:00
{
unsigned long flagv = 0 ;
2020-03-31 10:41:11 +02:00
int retval , unblock_retval ;
2020-02-28 08:53:12 +01:00
struct Scsi_Host * host = aac - > scsi_host_ptr ;
2017-02-02 15:53:33 -08:00
int bled ;
2007-06-12 09:33:54 -04:00
if ( spin_trylock_irqsave ( & aac - > fib_lock , flagv ) = = 0 )
return - EBUSY ;
if ( aac - > in_reset ) {
spin_unlock_irqrestore ( & aac - > fib_lock , flagv ) ;
return - EBUSY ;
}
aac - > in_reset = 1 ;
spin_unlock_irqrestore ( & aac - > fib_lock , flagv ) ;
/*
* Wait for all commands to complete to this specific
* target ( block maximum 60 seconds ) . Although not necessary ,
* it does make us a good storage citizen .
*/
2020-02-28 08:53:14 +01:00
scsi_host_block ( host ) ;
2007-06-12 09:33:54 -04:00
/* Quiesce build, flush cache, write through mode */
2007-10-30 15:50:49 -04:00
if ( forced < 2 )
aac_send_shutdown ( aac ) ;
2007-06-12 09:33:54 -04:00
spin_lock_irqsave ( host - > host_lock , flagv ) ;
2017-02-02 15:53:33 -08:00
bled = forced ? forced :
( aac_check_reset ! = 0 & & aac_check_reset ! = 1 ) ;
retval = _aac_reset_adapter ( aac , bled , reset_type ) ;
2007-06-12 09:33:54 -04:00
spin_unlock_irqrestore ( host - > host_lock , flagv ) ;
2020-03-31 10:41:11 +02:00
unblock_retval = scsi_host_unblock ( host , SDEV_RUNNING ) ;
if ( ! retval )
retval = unblock_retval ;
2007-10-30 15:50:49 -04:00
if ( ( forced < 2 ) & & ( retval = = - ENODEV ) ) {
2007-06-12 09:33:54 -04:00
/* Unwind aac_send_shutdown() IOP_RESET unsupported/disabled */
struct fib * fibctx = aac_fib_alloc ( aac ) ;
if ( fibctx ) {
struct aac_pause * cmd ;
int status ;
aac_fib_init ( fibctx ) ;
cmd = ( struct aac_pause * ) fib_data ( fibctx ) ;
cmd - > command = cpu_to_le32 ( VM_ContainerConfig ) ;
cmd - > type = cpu_to_le32 ( CT_PAUSE_IO ) ;
cmd - > timeout = cpu_to_le32 ( 1 ) ;
cmd - > min = cpu_to_le32 ( 1 ) ;
cmd - > noRescan = cpu_to_le32 ( 1 ) ;
cmd - > count = cpu_to_le32 ( 0 ) ;
status = aac_fib_send ( ContainerCommand ,
fibctx ,
sizeof ( struct aac_pause ) ,
FsaNormal ,
- 2 /* Timeout silently */ , 1 ,
NULL , NULL ) ;
if ( status > = 0 )
aac_fib_complete ( fibctx ) ;
2009-12-21 18:39:27 +05:30
/* FIB should be freed only after getting
* the response from the F / W */
if ( status ! = - ERESTARTSYS )
aac_fib_free ( fibctx ) ;
2007-06-12 09:33:54 -04:00
}
}
2006-08-03 08:03:30 -07:00
return retval ;
}
int aac_check_health ( struct aac_dev * aac )
{
int BlinkLED ;
unsigned long time_now , flagv = 0 ;
struct list_head * entry ;
/* Extending the scope of fib_lock slightly to protect aac->in_reset */
if ( spin_trylock_irqsave ( & aac - > fib_lock , flagv ) = = 0 )
return 0 ;
if ( aac - > in_reset | | ! ( BlinkLED = aac_adapter_check_health ( aac ) ) ) {
spin_unlock_irqrestore ( & aac - > fib_lock , flagv ) ;
return 0 ; /* OK */
}
aac - > in_reset = 1 ;
/* Fake up an AIF:
* aac_aifcmd . command = AifCmdEventNotify = 1
* aac_aifcmd . seqnum = 0xFFFFFFFF
* aac_aifcmd . data [ 0 ] = AifEnExpEvent = 23
* aac_aifcmd . data [ 1 ] = AifExeFirmwarePanic = 3
* aac . aifcmd . data [ 2 ] = AifHighPriority = 3
* aac . aifcmd . data [ 3 ] = BlinkLED
*/
time_now = jiffies / HZ ;
entry = aac - > fib_list . next ;
/*
* For each Context that is on the
* fibctxList , make a copy of the
* fib , and then set the event to wake up the
* thread that is waiting for it .
*/
while ( entry ! = & aac - > fib_list ) {
/*
* Extract the fibctx
*/
struct aac_fib_context * fibctx = list_entry ( entry , struct aac_fib_context , next ) ;
struct hw_fib * hw_fib ;
struct fib * fib ;
/*
* Check if the queue is getting
* backlogged
*/
if ( fibctx - > count > 20 ) {
/*
* It ' s * not * jiffies folks ,
* but jiffies / HZ , so do not
* panic . . .
*/
u32 time_last = fibctx - > jiffies ;
/*
* Has it been > 2 minutes
* since the last read off
* the queue ?
*/
if ( ( time_now - time_last ) > aif_timeout ) {
entry = entry - > next ;
aac_close_fib_context ( aac , fibctx ) ;
continue ;
}
}
/*
* Warning : no sleep allowed while
* holding spinlock
*/
2007-04-16 11:21:50 -04:00
hw_fib = kzalloc ( sizeof ( struct hw_fib ) , GFP_ATOMIC ) ;
fib = kzalloc ( sizeof ( struct fib ) , GFP_ATOMIC ) ;
2006-08-03 08:03:30 -07:00
if ( fib & & hw_fib ) {
struct aac_aifcmd * aif ;
2007-03-15 10:26:22 -07:00
fib - > hw_fib_va = hw_fib ;
2006-08-03 08:03:30 -07:00
fib - > dev = aac ;
aac_fib_init ( fib ) ;
fib - > type = FSAFS_NTC_FIB_CONTEXT ;
fib - > size = sizeof ( struct fib ) ;
fib - > data = hw_fib - > data ;
aif = ( struct aac_aifcmd * ) hw_fib - > data ;
aif - > command = cpu_to_le32 ( AifCmdEventNotify ) ;
2008-01-08 12:48:25 -08:00
aif - > seqnum = cpu_to_le32 ( 0xFFFFFFFF ) ;
( ( __le32 * ) aif - > data ) [ 0 ] = cpu_to_le32 ( AifEnExpEvent ) ;
( ( __le32 * ) aif - > data ) [ 1 ] = cpu_to_le32 ( AifExeFirmwarePanic ) ;
( ( __le32 * ) aif - > data ) [ 2 ] = cpu_to_le32 ( AifHighPriority ) ;
( ( __le32 * ) aif - > data ) [ 3 ] = cpu_to_le32 ( BlinkLED ) ;
2006-08-03 08:03:30 -07:00
/*
* Put the FIB onto the
* fibctx ' s fibs
*/
list_add_tail ( & fib - > fiblink , & fibctx - > fib_list ) ;
fibctx - > count + + ;
/*
* Set the event to wake up the
* thread that will waiting .
*/
2018-12-10 22:32:40 +01:00
complete ( & fibctx - > completion ) ;
2006-08-03 08:03:30 -07:00
} else {
printk ( KERN_WARNING " aifd: didn't allocate NewFib. \n " ) ;
kfree ( fib ) ;
kfree ( hw_fib ) ;
}
entry = entry - > next ;
}
spin_unlock_irqrestore ( & aac - > fib_lock , flagv ) ;
if ( BlinkLED < 0 ) {
2017-04-06 18:12:09 -03:00
printk ( KERN_ERR " %s: Host adapter is dead (or got a PCI error) %d \n " ,
aac - > name , BlinkLED ) ;
2006-08-03 08:03:30 -07:00
goto out ;
}
printk ( KERN_ERR " %s: Host adapter BLINK LED 0x%x \n " , aac - > name , BlinkLED ) ;
out :
aac - > in_reset = 0 ;
return BlinkLED ;
}
2017-12-26 20:34:40 -08:00
static inline int is_safw_raid_volume ( struct aac_dev * aac , int bus , int target )
{
return bus = = CONTAINER_CHANNEL & & target < aac - > maximum_num_containers ;
}
2017-02-02 15:53:28 -08:00
2017-12-26 20:34:42 -08:00
static struct scsi_device * aac_lookup_safw_scsi_device ( struct aac_dev * dev ,
int bus ,
int target )
{
if ( bus ! = CONTAINER_CHANNEL )
bus = aac_phys_to_logical ( bus ) ;
return scsi_device_lookup ( dev - > scsi_host_ptr , bus , target , 0 ) ;
}
static int aac_add_safw_device ( struct aac_dev * dev , int bus , int target )
{
if ( bus ! = CONTAINER_CHANNEL )
bus = aac_phys_to_logical ( bus ) ;
return scsi_add_device ( dev - > scsi_host_ptr , bus , target , 0 ) ;
}
static void aac_put_safw_scsi_device ( struct scsi_device * sdev )
{
if ( sdev )
scsi_device_put ( sdev ) ;
}
static void aac_remove_safw_device ( struct aac_dev * dev , int bus , int target )
2017-02-02 15:53:28 -08:00
{
struct scsi_device * sdev ;
2017-12-26 20:34:42 -08:00
sdev = aac_lookup_safw_scsi_device ( dev , bus , target ) ;
scsi_remove_device ( sdev ) ;
aac_put_safw_scsi_device ( sdev ) ;
}
2017-02-02 15:53:28 -08:00
2017-12-26 20:34:40 -08:00
static inline int aac_is_safw_scan_count_equal ( struct aac_dev * dev ,
int bus , int target )
{
return dev - > hba_map [ bus ] [ target ] . scan_counter = = dev - > scan_counter ;
}
static int aac_is_safw_target_valid ( struct aac_dev * dev , int bus , int target )
{
if ( is_safw_raid_volume ( dev , bus , target ) )
return dev - > fsa_dev [ target ] . valid ;
else
return aac_is_safw_scan_count_equal ( dev , bus , target ) ;
}
2017-02-02 15:53:28 -08:00
2017-12-26 20:34:42 -08:00
static int aac_is_safw_device_exposed ( struct aac_dev * dev , int bus , int target )
{
int is_exposed = 0 ;
struct scsi_device * sdev ;
2017-02-02 15:53:28 -08:00
2017-12-26 20:34:42 -08:00
sdev = aac_lookup_safw_scsi_device ( dev , bus , target ) ;
if ( sdev )
is_exposed = 1 ;
aac_put_safw_scsi_device ( sdev ) ;
2017-02-02 15:53:28 -08:00
2017-12-26 20:34:42 -08:00
return is_exposed ;
}
2017-02-02 15:53:28 -08:00
2017-12-26 20:34:49 -08:00
static int aac_update_safw_host_devices ( struct aac_dev * dev )
2017-02-02 15:53:28 -08:00
{
2017-12-26 20:34:41 -08:00
int i ;
2017-12-26 20:34:43 -08:00
int bus ;
int target ;
2017-12-26 20:34:42 -08:00
int is_exposed = 0 ;
2017-12-26 20:34:43 -08:00
int rcode = 0 ;
2017-02-02 15:53:28 -08:00
2017-12-26 20:34:49 -08:00
rcode = aac_setup_safw_adapter ( dev ) ;
2017-12-26 20:34:43 -08:00
if ( unlikely ( rcode < 0 ) ) {
goto out ;
2017-02-02 15:53:28 -08:00
}
2017-12-26 20:34:41 -08:00
for ( i = 0 ; i < AAC_BUS_TARGET_LOOP ; i + + ) {
2017-02-02 15:53:28 -08:00
2017-12-26 20:34:41 -08:00
bus = get_bus_number ( i ) ;
target = get_target_number ( i ) ;
2017-02-02 15:53:28 -08:00
2017-12-26 20:34:42 -08:00
is_exposed = aac_is_safw_device_exposed ( dev , bus , target ) ;
2017-02-02 15:53:28 -08:00
2017-12-26 20:34:42 -08:00
if ( aac_is_safw_target_valid ( dev , bus , target ) & & ! is_exposed )
aac_add_safw_device ( dev , bus , target ) ;
else if ( ! aac_is_safw_target_valid ( dev , bus , target ) & &
is_exposed )
aac_remove_safw_device ( dev , bus , target ) ;
2017-02-02 15:53:28 -08:00
}
2017-12-26 20:34:43 -08:00
out :
return rcode ;
2017-02-02 15:53:28 -08:00
}
2017-12-26 20:34:49 -08:00
static int aac_scan_safw_host ( struct aac_dev * dev )
2017-12-26 20:34:46 -08:00
{
int rcode = 0 ;
2017-12-26 20:34:49 -08:00
rcode = aac_update_safw_host_devices ( dev ) ;
2017-12-26 20:34:46 -08:00
if ( rcode )
aac_schedule_safw_scan_worker ( dev ) ;
return rcode ;
}
2017-12-26 20:34:49 -08:00
int aac_scan_host ( struct aac_dev * dev )
2017-12-26 20:34:45 -08:00
{
int rcode = 0 ;
mutex_lock ( & dev - > scan_mutex ) ;
if ( dev - > sa_firmware )
2017-12-26 20:34:49 -08:00
rcode = aac_scan_safw_host ( dev ) ;
2017-12-26 20:34:45 -08:00
else
scsi_scan_host ( dev - > scsi_host_ptr ) ;
mutex_unlock ( & dev - > scan_mutex ) ;
2017-12-26 20:34:46 -08:00
2017-12-26 20:34:45 -08:00
return rcode ;
2017-02-02 15:53:28 -08:00
}
2019-10-15 11:52:03 +05:30
void aac_src_reinit_aif_worker ( struct work_struct * work )
{
struct aac_dev * dev = container_of ( to_delayed_work ( work ) ,
struct aac_dev , src_reinit_aif_worker ) ;
wait_event ( dev - > scsi_host_ptr - > host_wait ,
! scsi_host_in_recovery ( dev - > scsi_host_ptr ) ) ;
aac_reinit_aif ( dev , dev - > cardtype ) ;
}
2017-02-02 15:53:28 -08:00
/**
2021-03-03 14:46:17 +00:00
* aac_handle_sa_aif - Handle a message from the firmware
2017-02-02 15:53:28 -08:00
* @ dev : Which adapter this fib is from
* @ fibptr : Pointer to fibptr from adapter
*
* This routine handles a driver notify fib from the adapter and
* dispatches it to the appropriate routine for handling .
*/
static void aac_handle_sa_aif ( struct aac_dev * dev , struct fib * fibptr )
{
2017-12-26 20:34:40 -08:00
int i ;
2017-02-02 15:53:28 -08:00
u32 events = 0 ;
if ( fibptr - > hbacmd_size & SA_AIF_HOTPLUG )
events = SA_AIF_HOTPLUG ;
else if ( fibptr - > hbacmd_size & SA_AIF_HARDWARE )
events = SA_AIF_HARDWARE ;
else if ( fibptr - > hbacmd_size & SA_AIF_PDEV_CHANGE )
events = SA_AIF_PDEV_CHANGE ;
else if ( fibptr - > hbacmd_size & SA_AIF_LDEV_CHANGE )
events = SA_AIF_LDEV_CHANGE ;
else if ( fibptr - > hbacmd_size & SA_AIF_BPSTAT_CHANGE )
events = SA_AIF_BPSTAT_CHANGE ;
else if ( fibptr - > hbacmd_size & SA_AIF_BPCFG_CHANGE )
events = SA_AIF_BPCFG_CHANGE ;
switch ( events ) {
case SA_AIF_HOTPLUG :
case SA_AIF_HARDWARE :
case SA_AIF_PDEV_CHANGE :
case SA_AIF_LDEV_CHANGE :
case SA_AIF_BPCFG_CHANGE :
2017-12-26 20:34:49 -08:00
aac_scan_host ( dev ) ;
2017-12-26 20:34:45 -08:00
2017-02-02 15:53:28 -08:00
break ;
case SA_AIF_BPSTAT_CHANGE :
/* currently do nothing */
break ;
}
for ( i = 1 ; i < = 10 ; + + i ) {
events = src_readl ( dev , MUnit . IDR ) ;
if ( events & ( 1 < < 23 ) ) {
pr_warn ( " AIF not cleared by firmware - %d/%d) \n " ,
i , 10 ) ;
ssleep ( 1 ) ;
}
}
}
2017-02-02 15:53:24 -08:00
static int get_fib_count ( struct aac_dev * dev )
{
unsigned int num = 0 ;
struct list_head * entry ;
unsigned long flagv ;
/*
* Warning : no sleep allowed while
* holding spinlock . We take the estimate
* and pre - allocate a set of fibs outside the
* lock .
*/
num = le32_to_cpu ( dev - > init - > r7 . adapter_fibs_size )
/ sizeof ( struct hw_fib ) ; /* some extra */
spin_lock_irqsave ( & dev - > fib_lock , flagv ) ;
entry = dev - > fib_list . next ;
while ( entry ! = & dev - > fib_list ) {
entry = entry - > next ;
+ + num ;
}
spin_unlock_irqrestore ( & dev - > fib_lock , flagv ) ;
return num ;
}
static int fillup_pools ( struct aac_dev * dev , struct hw_fib * * hw_fib_pool ,
struct fib * * fib_pool ,
unsigned int num )
{
struct hw_fib * * hw_fib_p ;
struct fib * * fib_p ;
hw_fib_p = hw_fib_pool ;
fib_p = fib_pool ;
while ( hw_fib_p < & hw_fib_pool [ num ] ) {
* ( hw_fib_p ) = kmalloc ( sizeof ( struct hw_fib ) , GFP_KERNEL ) ;
if ( ! ( * ( hw_fib_p + + ) ) ) {
- - hw_fib_p ;
break ;
}
* ( fib_p ) = kmalloc ( sizeof ( struct fib ) , GFP_KERNEL ) ;
if ( ! ( * ( fib_p + + ) ) ) {
kfree ( * ( - - hw_fib_p ) ) ;
break ;
}
}
2017-03-14 09:20:19 -07:00
/*
* Get the actual number of allocated fibs
*/
2017-02-02 15:53:24 -08:00
num = hw_fib_p - hw_fib_pool ;
2017-03-14 09:20:19 -07:00
return num ;
2017-02-02 15:53:24 -08:00
}
static void wakeup_fibctx_threads ( struct aac_dev * dev ,
struct hw_fib * * hw_fib_pool ,
struct fib * * fib_pool ,
struct fib * fib ,
struct hw_fib * hw_fib ,
unsigned int num )
{
unsigned long flagv ;
struct list_head * entry ;
struct hw_fib * * hw_fib_p ;
struct fib * * fib_p ;
u32 time_now , time_last ;
struct hw_fib * hw_newfib ;
struct fib * newfib ;
struct aac_fib_context * fibctx ;
time_now = jiffies / HZ ;
spin_lock_irqsave ( & dev - > fib_lock , flagv ) ;
entry = dev - > fib_list . next ;
/*
* For each Context that is on the
* fibctxList , make a copy of the
* fib , and then set the event to wake up the
* thread that is waiting for it .
*/
hw_fib_p = hw_fib_pool ;
fib_p = fib_pool ;
while ( entry ! = & dev - > fib_list ) {
/*
* Extract the fibctx
*/
fibctx = list_entry ( entry , struct aac_fib_context ,
next ) ;
/*
* Check if the queue is getting
* backlogged
*/
if ( fibctx - > count > 20 ) {
/*
* It ' s * not * jiffies folks ,
* but jiffies / HZ so do not
* panic . . .
*/
time_last = fibctx - > jiffies ;
/*
* Has it been > 2 minutes
* since the last read off
* the queue ?
*/
if ( ( time_now - time_last ) > aif_timeout ) {
entry = entry - > next ;
aac_close_fib_context ( dev , fibctx ) ;
continue ;
}
}
/*
* Warning : no sleep allowed while
* holding spinlock
*/
if ( hw_fib_p > = & hw_fib_pool [ num ] ) {
pr_warn ( " aifd: didn't allocate NewFib \n " ) ;
entry = entry - > next ;
continue ;
}
hw_newfib = * hw_fib_p ;
* ( hw_fib_p + + ) = NULL ;
newfib = * fib_p ;
* ( fib_p + + ) = NULL ;
/*
* Make the copy of the FIB
*/
memcpy ( hw_newfib , hw_fib , sizeof ( struct hw_fib ) ) ;
memcpy ( newfib , fib , sizeof ( struct fib ) ) ;
newfib - > hw_fib_va = hw_newfib ;
/*
* Put the FIB onto the
* fibctx ' s fibs
*/
list_add_tail ( & newfib - > fiblink , & fibctx - > fib_list ) ;
fibctx - > count + + ;
/*
* Set the event to wake up the
* thread that is waiting .
*/
2018-12-10 22:32:40 +01:00
complete ( & fibctx - > completion ) ;
2017-02-02 15:53:24 -08:00
entry = entry - > next ;
}
/*
* Set the status of this FIB
*/
* ( __le32 * ) hw_fib - > data = cpu_to_le32 ( ST_OK ) ;
aac_fib_adapter_complete ( fib , sizeof ( u32 ) ) ;
spin_unlock_irqrestore ( & dev - > fib_lock , flagv ) ;
}
static void aac_process_events ( struct aac_dev * dev )
{
struct hw_fib * hw_fib ;
struct fib * fib ;
unsigned long flags ;
spinlock_t * t_lock ;
t_lock = dev - > queues - > queue [ HostNormCmdQueue ] . lock ;
spin_lock_irqsave ( t_lock , flags ) ;
while ( ! list_empty ( & ( dev - > queues - > queue [ HostNormCmdQueue ] . cmdq ) ) ) {
struct list_head * entry ;
struct aac_aifcmd * aifcmd ;
unsigned int num ;
struct hw_fib * * hw_fib_pool , * * hw_fib_p ;
struct fib * * fib_pool , * * fib_p ;
set_current_state ( TASK_RUNNING ) ;
entry = dev - > queues - > queue [ HostNormCmdQueue ] . cmdq . next ;
list_del ( entry ) ;
t_lock = dev - > queues - > queue [ HostNormCmdQueue ] . lock ;
spin_unlock_irqrestore ( t_lock , flags ) ;
fib = list_entry ( entry , struct fib , fiblink ) ;
hw_fib = fib - > hw_fib_va ;
2017-02-02 15:53:28 -08:00
if ( dev - > sa_firmware ) {
/* Thor AIF */
aac_handle_sa_aif ( dev , fib ) ;
aac_fib_adapter_complete ( fib , ( u16 ) sizeof ( u32 ) ) ;
2017-02-16 12:51:23 -08:00
goto free_fib ;
2017-02-02 15:53:28 -08:00
}
2017-02-02 15:53:24 -08:00
/*
* We will process the FIB here or pass it to a
* worker thread that is TBD . We Really can ' t
* do anything at this point since we don ' t have
* anything defined for this thread to do .
*/
memset ( fib , 0 , sizeof ( struct fib ) ) ;
fib - > type = FSAFS_NTC_FIB_CONTEXT ;
fib - > size = sizeof ( struct fib ) ;
fib - > hw_fib_va = hw_fib ;
fib - > data = hw_fib - > data ;
fib - > dev = dev ;
/*
* We only handle AifRequest fibs from the adapter .
*/
aifcmd = ( struct aac_aifcmd * ) hw_fib - > data ;
if ( aifcmd - > command = = cpu_to_le32 ( AifCmdDriverNotify ) ) {
/* Handle Driver Notify Events */
aac_handle_aif ( dev , fib ) ;
* ( __le32 * ) hw_fib - > data = cpu_to_le32 ( ST_OK ) ;
aac_fib_adapter_complete ( fib , ( u16 ) sizeof ( u32 ) ) ;
goto free_fib ;
}
/*
* The u32 here is important and intended . We are using
* 32 bit wrapping time to fit the adapter field
*/
/* Sniff events */
if ( aifcmd - > command = = cpu_to_le32 ( AifCmdEventNotify )
| | aifcmd - > command = = cpu_to_le32 ( AifCmdJobProgress ) ) {
aac_handle_aif ( dev , fib ) ;
}
/*
* get number of fibs to process
*/
num = get_fib_count ( dev ) ;
if ( ! num )
goto free_fib ;
hw_fib_pool = kmalloc_array ( num , sizeof ( struct hw_fib * ) ,
GFP_KERNEL ) ;
if ( ! hw_fib_pool )
goto free_fib ;
fib_pool = kmalloc_array ( num , sizeof ( struct fib * ) , GFP_KERNEL ) ;
if ( ! fib_pool )
goto free_hw_fib_pool ;
/*
* Fill up fib pointer pools with actual fibs
* and hw_fibs
*/
2017-03-14 09:20:19 -07:00
num = fillup_pools ( dev , hw_fib_pool , fib_pool , num ) ;
if ( ! num )
2017-02-02 15:53:24 -08:00
goto free_mem ;
/*
* wakeup the thread that is waiting for
* the response from fw ( ioctl )
*/
wakeup_fibctx_threads ( dev , hw_fib_pool , fib_pool ,
fib , hw_fib , num ) ;
free_mem :
/* Free up the remaining resources */
hw_fib_p = hw_fib_pool ;
fib_p = fib_pool ;
while ( hw_fib_p < & hw_fib_pool [ num ] ) {
kfree ( * hw_fib_p ) ;
kfree ( * fib_p ) ;
+ + fib_p ;
+ + hw_fib_p ;
}
kfree ( fib_pool ) ;
free_hw_fib_pool :
kfree ( hw_fib_pool ) ;
free_fib :
kfree ( fib ) ;
t_lock = dev - > queues - > queue [ HostNormCmdQueue ] . lock ;
spin_lock_irqsave ( t_lock , flags ) ;
}
/*
* There are no more AIF ' s
*/
t_lock = dev - > queues - > queue [ HostNormCmdQueue ] . lock ;
spin_unlock_irqrestore ( t_lock , flags ) ;
}
2006-08-03 08:03:30 -07:00
2017-02-02 15:53:25 -08:00
static int aac_send_wellness_command ( struct aac_dev * dev , char * wellness_str ,
u32 datasize )
{
struct aac_srb * srbcmd ;
struct sgmap64 * sg64 ;
dma_addr_t addr ;
char * dma_buf ;
struct fib * fibptr ;
int ret = - ENOMEM ;
u32 vbus , vid ;
fibptr = aac_fib_alloc ( dev ) ;
if ( ! fibptr )
goto out ;
2017-04-05 16:14:16 +05:30
dma_buf = dma_alloc_coherent ( & dev - > pdev - > dev , datasize , & addr ,
GFP_KERNEL ) ;
2017-02-02 15:53:25 -08:00
if ( ! dma_buf )
goto fib_free_out ;
aac_fib_init ( fibptr ) ;
2017-02-16 12:51:10 -08:00
vbus = ( u32 ) le16_to_cpu ( dev - > supplement_adapter_info . virt_device_bus ) ;
vid = ( u32 ) le16_to_cpu ( dev - > supplement_adapter_info . virt_device_target ) ;
2017-02-02 15:53:25 -08:00
srbcmd = ( struct aac_srb * ) fib_data ( fibptr ) ;
srbcmd - > function = cpu_to_le32 ( SRBF_ExecuteScsi ) ;
srbcmd - > channel = cpu_to_le32 ( vbus ) ;
srbcmd - > id = cpu_to_le32 ( vid ) ;
srbcmd - > lun = 0 ;
srbcmd - > flags = cpu_to_le32 ( SRB_DataOut ) ;
srbcmd - > timeout = cpu_to_le32 ( 10 ) ;
srbcmd - > retry_limit = 0 ;
srbcmd - > cdb_size = cpu_to_le32 ( 12 ) ;
srbcmd - > count = cpu_to_le32 ( datasize ) ;
memset ( srbcmd - > cdb , 0 , sizeof ( srbcmd - > cdb ) ) ;
srbcmd - > cdb [ 0 ] = BMIC_OUT ;
srbcmd - > cdb [ 6 ] = WRITE_HOST_WELLNESS ;
memcpy ( dma_buf , ( char * ) wellness_str , datasize ) ;
sg64 = ( struct sgmap64 * ) & srbcmd - > sg ;
sg64 - > count = cpu_to_le32 ( 1 ) ;
sg64 - > sg [ 0 ] . addr [ 1 ] = cpu_to_le32 ( ( u32 ) ( ( ( addr ) > > 16 ) > > 16 ) ) ;
sg64 - > sg [ 0 ] . addr [ 0 ] = cpu_to_le32 ( ( u32 ) ( addr & 0xffffffff ) ) ;
sg64 - > sg [ 0 ] . count = cpu_to_le32 ( datasize ) ;
ret = aac_fib_send ( ScsiPortCommand64 , fibptr , sizeof ( struct aac_srb ) ,
FsaNormal , 1 , 1 , NULL , NULL ) ;
2017-04-05 16:14:16 +05:30
dma_free_coherent ( & dev - > pdev - > dev , datasize , dma_buf , addr ) ;
2017-02-02 15:53:25 -08:00
/*
* Do not set XferState to zero unless
* receives a response from F / W
*/
if ( ret > = 0 )
aac_fib_complete ( fibptr ) ;
/*
* FIB should be freed only after
* getting the response from the F / W
*/
if ( ret ! = - ERESTARTSYS )
goto fib_free_out ;
out :
return ret ;
fib_free_out :
aac_fib_free ( fibptr ) ;
goto out ;
}
2020-04-30 18:02:12 +08:00
static int aac_send_safw_hostttime ( struct aac_dev * dev , struct timespec64 * now )
2017-02-02 15:53:25 -08:00
{
struct tm cur_tm ;
char wellness_str [ ] = " <HW>TD \010 \0 \0 \0 \0 \0 \0 \0 \0 \0 DW \0 \0 ZZ " ;
u32 datasize = sizeof ( wellness_str ) ;
2017-11-07 11:46:05 +01:00
time64_t local_time ;
2017-02-02 15:53:25 -08:00
int ret = - ENODEV ;
if ( ! dev - > sa_firmware )
goto out ;
2017-11-07 11:46:05 +01:00
local_time = ( now - > tv_sec - ( sys_tz . tz_minuteswest * 60 ) ) ;
time64_to_tm ( local_time , 0 , & cur_tm ) ;
2017-02-02 15:53:25 -08:00
cur_tm . tm_mon + = 1 ;
cur_tm . tm_year + = 1900 ;
wellness_str [ 8 ] = bin2bcd ( cur_tm . tm_hour ) ;
wellness_str [ 9 ] = bin2bcd ( cur_tm . tm_min ) ;
wellness_str [ 10 ] = bin2bcd ( cur_tm . tm_sec ) ;
wellness_str [ 12 ] = bin2bcd ( cur_tm . tm_mon ) ;
wellness_str [ 13 ] = bin2bcd ( cur_tm . tm_mday ) ;
wellness_str [ 14 ] = bin2bcd ( cur_tm . tm_year / 100 ) ;
wellness_str [ 15 ] = bin2bcd ( cur_tm . tm_year % 100 ) ;
ret = aac_send_wellness_command ( dev , wellness_str , datasize ) ;
out :
return ret ;
}
2020-04-30 18:02:12 +08:00
static int aac_send_hosttime ( struct aac_dev * dev , struct timespec64 * now )
2017-02-02 15:53:25 -08:00
{
int ret = - ENOMEM ;
struct fib * fibptr ;
__le32 * info ;
fibptr = aac_fib_alloc ( dev ) ;
if ( ! fibptr )
goto out ;
aac_fib_init ( fibptr ) ;
info = ( __le32 * ) fib_data ( fibptr ) ;
2017-11-07 11:46:05 +01:00
* info = cpu_to_le32 ( now - > tv_sec ) ; /* overflow in y2106 */
2017-02-02 15:53:25 -08:00
ret = aac_fib_send ( SendHostTime , fibptr , sizeof ( * info ) , FsaNormal ,
1 , 1 , NULL , NULL ) ;
/*
* Do not set XferState to zero unless
* receives a response from F / W
*/
if ( ret > = 0 )
aac_fib_complete ( fibptr ) ;
/*
* FIB should be freed only after
* getting the response from the F / W
*/
if ( ret ! = - ERESTARTSYS )
aac_fib_free ( fibptr ) ;
out :
return ret ;
}
2005-04-16 15:20:36 -07:00
/**
* aac_command_thread - command processing thread
2020-07-13 08:59:44 +01:00
* @ data : Adapter to monitor
2005-04-16 15:20:36 -07:00
*
* Waits on the commandready event in it ' s queue . When the event gets set
* it will pull FIBs off it ' s queue . It will continue to pull FIBs off
* until the queue is empty . When the queue is empty it will wait for
* more FIBs .
*/
2008-01-16 07:39:06 -08:00
2006-02-14 18:45:06 +01:00
int aac_command_thread ( void * data )
2005-04-16 15:20:36 -07:00
{
2006-02-14 18:45:06 +01:00
struct aac_dev * dev = data ;
2005-04-16 15:20:36 -07:00
DECLARE_WAITQUEUE ( wait , current ) ;
2007-06-12 09:33:54 -04:00
unsigned long next_jiffies = jiffies + HZ ;
unsigned long next_check_jiffies = next_jiffies ;
long difference = HZ ;
2005-04-16 15:20:36 -07:00
/*
* We can only have one thread per adapter for AIF ' s .
*/
if ( dev - > aif_thread )
return - EINVAL ;
2006-02-14 18:45:06 +01:00
2005-04-16 15:20:36 -07:00
/*
* Let the DPC know it has a place to send the AIF ' s to .
*/
dev - > aif_thread = 1 ;
2005-09-26 13:02:15 -07:00
add_wait_queue ( & dev - > queues - > queue [ HostNormCmdQueue ] . cmdready , & wait ) ;
2005-04-16 15:20:36 -07:00
set_current_state ( TASK_INTERRUPTIBLE ) ;
2005-09-26 13:02:15 -07:00
dprintk ( ( KERN_INFO " aac_command_thread start \n " ) ) ;
2008-01-16 07:39:06 -08:00
while ( 1 ) {
2005-04-16 15:20:36 -07:00
2017-02-02 15:53:24 -08:00
aac_process_events ( dev ) ;
2007-06-12 09:33:54 -04:00
/*
* Background activity
*/
if ( ( time_before ( next_check_jiffies , next_jiffies ) )
& & ( ( difference = next_check_jiffies - jiffies ) < = 0 ) ) {
next_check_jiffies = next_jiffies ;
2017-05-10 09:39:48 -07:00
if ( aac_adapter_check_health ( dev ) = = 0 ) {
2007-06-12 09:33:54 -04:00
difference = ( ( long ) ( unsigned ) check_interval )
* HZ ;
next_check_jiffies = jiffies + difference ;
} else if ( ! dev - > queues )
break ;
}
if ( ! time_before ( next_check_jiffies , next_jiffies )
& & ( ( difference = next_jiffies - jiffies ) < = 0 ) ) {
2017-11-07 11:46:05 +01:00
struct timespec64 now ;
2007-06-12 09:33:54 -04:00
int ret ;
/* Don't even try to talk to adapter if its sick */
2017-05-10 09:39:48 -07:00
ret = aac_adapter_check_health ( dev ) ;
2017-02-16 12:51:17 -08:00
if ( ret | | ! dev - > queues )
2007-06-12 09:33:54 -04:00
break ;
next_check_jiffies = jiffies
+ ( ( long ) ( unsigned ) check_interval )
* HZ ;
2017-11-07 11:46:05 +01:00
ktime_get_real_ts64 ( & now ) ;
2007-06-12 09:33:54 -04:00
/* Synchronize our watches */
2017-11-07 11:46:05 +01:00
if ( ( ( NSEC_PER_SEC - ( NSEC_PER_SEC / HZ ) ) > now . tv_nsec )
& & ( now . tv_nsec > ( NSEC_PER_SEC / HZ ) ) )
2017-11-28 14:25:25 +01:00
difference = HZ + HZ / 2 -
now . tv_nsec / ( NSEC_PER_SEC / HZ ) ;
2017-02-24 14:43:30 +00:00
else {
2017-11-07 11:46:05 +01:00
if ( now . tv_nsec > NSEC_PER_SEC / 2 )
2017-02-02 15:53:25 -08:00
+ + now . tv_sec ;
if ( dev - > sa_firmware )
ret =
aac_send_safw_hostttime ( dev , & now ) ;
else
ret = aac_send_hosttime ( dev , & now ) ;
2007-06-12 09:33:54 -04:00
difference = ( long ) ( unsigned ) update_interval * HZ ;
}
next_jiffies = jiffies + difference ;
if ( time_before ( next_check_jiffies , next_jiffies ) )
difference = next_check_jiffies - jiffies ;
}
if ( difference < = 0 )
difference = 1 ;
set_current_state ( TASK_INTERRUPTIBLE ) ;
2016-04-25 23:31:57 -07:00
if ( kthread_should_stop ( ) )
break ;
2017-11-28 14:25:25 +01:00
/*
* we probably want usleep_range ( ) here instead of the
* jiffies computation
*/
2007-06-12 09:33:54 -04:00
schedule_timeout ( difference ) ;
2005-04-16 15:20:36 -07:00
2006-02-14 18:45:06 +01:00
if ( kthread_should_stop ( ) )
2005-04-16 15:20:36 -07:00
break ;
}
2005-09-26 13:02:15 -07:00
if ( dev - > queues )
remove_wait_queue ( & dev - > queues - > queue [ HostNormCmdQueue ] . cmdready , & wait ) ;
2005-04-16 15:20:36 -07:00
dev - > aif_thread = 0 ;
2005-09-26 13:02:15 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2015-08-28 06:38:38 -04:00
int aac_acquire_irq ( struct aac_dev * dev )
{
int i ;
int j ;
int ret = 0 ;
if ( ! dev - > sync_mode & & dev - > msi_enabled & & dev - > max_msix > 1 ) {
for ( i = 0 ; i < dev - > max_msix ; i + + ) {
dev - > aac_msix [ i ] . vector_no = i ;
dev - > aac_msix [ i ] . dev = dev ;
2016-11-08 08:11:30 +01:00
if ( request_irq ( pci_irq_vector ( dev - > pdev , i ) ,
2015-08-28 06:38:38 -04:00
dev - > a_ops . adapter_intr ,
0 , " aacraid " , & ( dev - > aac_msix [ i ] ) ) ) {
printk ( KERN_ERR " %s%d: Failed to register IRQ for vector %d. \n " ,
dev - > name , dev - > id , i ) ;
for ( j = 0 ; j < i ; j + + )
2016-11-08 08:11:30 +01:00
free_irq ( pci_irq_vector ( dev - > pdev , j ) ,
2015-08-28 06:38:38 -04:00
& ( dev - > aac_msix [ j ] ) ) ;
pci_disable_msix ( dev - > pdev ) ;
ret = - 1 ;
}
}
} else {
dev - > aac_msix [ 0 ] . vector_no = 0 ;
dev - > aac_msix [ 0 ] . dev = dev ;
if ( request_irq ( dev - > pdev - > irq , dev - > a_ops . adapter_intr ,
IRQF_SHARED , " aacraid " ,
& ( dev - > aac_msix [ 0 ] ) ) < 0 ) {
if ( dev - > msi )
pci_disable_msi ( dev - > pdev ) ;
printk ( KERN_ERR " %s%d: Interrupt unavailable. \n " ,
dev - > name , dev - > id ) ;
ret = - 1 ;
}
}
return ret ;
}
void aac_free_irq ( struct aac_dev * dev )
{
int i ;
2017-05-10 09:39:52 -07:00
if ( aac_is_src ( dev ) ) {
2015-08-28 06:38:38 -04:00
if ( dev - > max_msix > 1 ) {
2016-11-08 08:11:30 +01:00
for ( i = 0 ; i < dev - > max_msix ; i + + )
free_irq ( pci_irq_vector ( dev - > pdev , i ) ,
& ( dev - > aac_msix [ i ] ) ) ;
2015-08-28 06:38:38 -04:00
} else {
free_irq ( dev - > pdev - > irq , & ( dev - > aac_msix [ 0 ] ) ) ;
}
} else {
free_irq ( dev - > pdev - > irq , dev ) ;
}
if ( dev - > msi )
pci_disable_msi ( dev - > pdev ) ;
else if ( dev - > max_msix > 1 )
pci_disable_msix ( dev - > pdev ) ;
}