2019-05-01 12:14:10 -04:00
// SPDX-License-Identifier: GPL-2.0
2006-08-29 09:22:51 -05:00
/*
* Serial Attached SCSI ( SAS ) Event processing
*
* Copyright ( C ) 2005 Adaptec , Inc . All rights reserved .
* Copyright ( C ) 2005 Luben Tuikov < luben_tuikov @ adaptec . com >
*/
2011-12-19 16:42:34 -08:00
# include <linux/export.h>
2006-08-29 09:22:51 -05:00
# include <scsi/scsi_host.h>
# include "sas_internal.h"
2017-06-14 13:52:43 +02:00
int sas_queue_work ( struct sas_ha_struct * ha , struct sas_work * sw )
2011-12-19 16:42:34 -08:00
{
2017-12-08 17:42:04 +08:00
/* it's added to the defer_q when draining so return succeed */
int rc = 1 ;
2017-06-14 13:52:43 +02:00
2011-12-19 16:42:34 -08:00
if ( ! test_bit ( SAS_HA_REGISTERED , & ha - > state ) )
2017-06-14 13:52:43 +02:00
return 0 ;
2011-12-19 16:42:34 -08:00
2012-03-09 11:00:06 -08:00
if ( test_bit ( SAS_HA_DRAINING , & ha - > state ) ) {
/* add it to the defer list, if not already pending */
if ( list_empty ( & sw - > drain_node ) )
2017-09-06 17:15:14 +08:00
list_add_tail ( & sw - > drain_node , & ha - > defer_q ) ;
2012-03-09 11:00:06 -08:00
} else
2017-12-08 17:42:07 +08:00
rc = queue_work ( ha - > event_q , & sw - > work ) ;
2017-06-14 13:52:43 +02:00
return rc ;
2011-12-19 16:42:34 -08:00
}
2017-12-08 17:42:04 +08:00
static int sas_queue_event ( int event , struct sas_work * work ,
2011-12-19 16:42:34 -08:00
struct sas_ha_struct * ha )
{
2017-12-08 17:42:04 +08:00
unsigned long flags ;
int rc ;
2017-06-14 13:52:43 +02:00
2017-12-08 17:42:04 +08:00
spin_lock_irqsave ( & ha - > lock , flags ) ;
rc = sas_queue_work ( ha , work ) ;
spin_unlock_irqrestore ( & ha - > lock , flags ) ;
2017-06-14 13:52:43 +02:00
return rc ;
2011-12-19 16:42:34 -08:00
}
2012-01-12 11:47:24 -08:00
void __sas_drain_work ( struct sas_ha_struct * ha )
2011-12-19 16:42:34 -08:00
{
2012-03-09 11:00:06 -08:00
struct sas_work * sw , * _sw ;
2017-12-08 17:42:04 +08:00
int ret ;
2011-12-19 16:42:34 -08:00
set_bit ( SAS_HA_DRAINING , & ha - > state ) ;
/* flush submitters */
2012-06-21 23:25:27 -07:00
spin_lock_irq ( & ha - > lock ) ;
spin_unlock_irq ( & ha - > lock ) ;
2011-12-19 16:42:34 -08:00
2017-12-08 17:42:07 +08:00
drain_workqueue ( ha - > event_q ) ;
drain_workqueue ( ha - > disco_q ) ;
2011-12-19 16:42:34 -08:00
2012-06-21 23:25:27 -07:00
spin_lock_irq ( & ha - > lock ) ;
2011-12-19 16:42:34 -08:00
clear_bit ( SAS_HA_DRAINING , & ha - > state ) ;
2012-03-09 11:00:06 -08:00
list_for_each_entry_safe ( sw , _sw , & ha - > defer_q , drain_node ) {
list_del_init ( & sw - > drain_node ) ;
2017-12-08 17:42:04 +08:00
ret = sas_queue_work ( ha , sw ) ;
if ( ret ! = 1 )
sas_free_event ( to_asd_sas_event ( & sw - > work ) ) ;
2011-12-19 16:42:34 -08:00
}
2012-06-21 23:25:27 -07:00
spin_unlock_irq ( & ha - > lock ) ;
2012-01-12 11:47:24 -08:00
}
int sas_drain_work ( struct sas_ha_struct * ha )
{
int err ;
err = mutex_lock_interruptible ( & ha - > drain_mutex ) ;
if ( err )
return err ;
if ( test_bit ( SAS_HA_REGISTERED , & ha - > state ) )
__sas_drain_work ( ha ) ;
2011-12-19 16:42:34 -08:00
mutex_unlock ( & ha - > drain_mutex ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( sas_drain_work ) ;
2011-11-17 17:59:51 -08:00
void sas_disable_revalidation ( struct sas_ha_struct * ha )
{
mutex_lock ( & ha - > disco_mutex ) ;
set_bit ( SAS_HA_ATA_EH_ACTIVE , & ha - > state ) ;
mutex_unlock ( & ha - > disco_mutex ) ;
}
void sas_enable_revalidation ( struct sas_ha_struct * ha )
{
int i ;
mutex_lock ( & ha - > disco_mutex ) ;
clear_bit ( SAS_HA_ATA_EH_ACTIVE , & ha - > state ) ;
for ( i = 0 ; i < ha - > num_phys ; i + + ) {
struct asd_sas_port * port = ha - > sas_port [ i ] ;
const int ev = DISCE_REVALIDATE_DOMAIN ;
struct sas_discovery * d = & port - > disc ;
2017-12-08 17:42:10 +08:00
struct asd_sas_phy * sas_phy ;
2011-11-17 17:59:51 -08:00
if ( ! test_and_clear_bit ( ev , & d - > pending ) )
continue ;
2021-12-20 19:21:27 +08:00
spin_lock ( & port - > phy_list_lock ) ;
if ( list_empty ( & port - > phy_list ) ) {
spin_unlock ( & port - > phy_list_lock ) ;
2017-12-08 17:42:10 +08:00
continue ;
2021-12-20 19:21:27 +08:00
}
2017-12-08 17:42:10 +08:00
sas_phy = container_of ( port - > phy_list . next , struct asd_sas_phy ,
port_phy_el ) ;
2021-12-20 19:21:27 +08:00
spin_unlock ( & port - > phy_list_lock ) ;
2021-01-18 11:09:52 +01:00
sas_notify_port_event ( sas_phy ,
2021-01-18 11:09:44 +01:00
PORTE_BROADCAST_RCVD , GFP_KERNEL ) ;
2011-11-17 17:59:51 -08:00
}
mutex_unlock ( & ha - > disco_mutex ) ;
}
2017-12-08 17:42:04 +08:00
static void sas_port_event_worker ( struct work_struct * work )
{
struct asd_sas_event * ev = to_asd_sas_event ( work ) ;
sas_port_event_fns [ ev - > event ] ( work ) ;
sas_free_event ( ev ) ;
}
static void sas_phy_event_worker ( struct work_struct * work )
{
struct asd_sas_event * ev = to_asd_sas_event ( work ) ;
sas_phy_event_fns [ ev - > event ] ( work ) ;
sas_free_event ( ev ) ;
}
2021-01-18 11:09:48 +01:00
int sas_notify_port_event ( struct asd_sas_phy * phy , enum port_event event ,
gfp_t gfp_flags )
2006-08-29 09:22:51 -05:00
{
struct sas_ha_struct * ha = phy - > ha ;
2021-01-18 11:09:48 +01:00
struct asd_sas_event * ev ;
2017-12-08 17:42:04 +08:00
int ret ;
2006-08-29 09:22:51 -05:00
BUG_ON ( event > = PORT_NUM_EVENTS ) ;
2021-01-18 11:09:52 +01:00
ev = sas_alloc_event ( phy , gfp_flags ) ;
2021-01-18 11:09:48 +01:00
if ( ! ev )
return - ENOMEM ;
2017-12-08 17:42:04 +08:00
INIT_SAS_EVENT ( ev , sas_port_event_worker , phy , event ) ;
ret = sas_queue_event ( event , & ev - > work , ha ) ;
if ( ret ! = 1 )
sas_free_event ( ev ) ;
return ret ;
2006-08-29 09:22:51 -05:00
}
2021-01-18 11:09:48 +01:00
EXPORT_SYMBOL_GPL ( sas_notify_port_event ) ;
2006-08-29 09:22:51 -05:00
2021-01-18 11:09:48 +01:00
int sas_notify_phy_event ( struct asd_sas_phy * phy , enum phy_event event ,
gfp_t gfp_flags )
2021-01-18 11:09:39 +01:00
{
struct sas_ha_struct * ha = phy - > ha ;
2021-01-18 11:09:48 +01:00
struct asd_sas_event * ev ;
2021-01-18 11:09:39 +01:00
int ret ;
BUG_ON ( event > = PHY_NUM_EVENTS ) ;
2021-01-18 11:09:52 +01:00
ev = sas_alloc_event ( phy , gfp_flags ) ;
2021-01-18 11:09:48 +01:00
if ( ! ev )
return - ENOMEM ;
2017-12-08 17:42:04 +08:00
INIT_SAS_EVENT ( ev , sas_phy_event_worker , phy , event ) ;
ret = sas_queue_event ( event , & ev - > work , ha ) ;
if ( ret ! = 1 )
sas_free_event ( ev ) ;
return ret ;
2006-08-29 09:22:51 -05:00
}
2021-01-18 11:09:48 +01:00
EXPORT_SYMBOL_GPL ( sas_notify_phy_event ) ;