2011-07-02 22:56:22 -07:00
/*
* This file is provided under a dual BSD / GPLv2 license . When using or
* redistributing this file , you may do so under either license .
*
* GPL LICENSE SUMMARY
*
* Copyright ( c ) 2008 - 2011 Intel Corporation . All rights reserved .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of version 2 of the GNU General Public License as
* published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin St - Fifth Floor , Boston , MA 02110 - 1301 USA .
* The full GNU General Public License is included in this distribution
* in the file called LICENSE . GPL .
*
* BSD LICENSE
*
* Copyright ( c ) 2008 - 2011 Intel Corporation . All rights reserved .
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
*
* * Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
* * Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in
* the documentation and / or other materials provided with the
* distribution .
* * Neither the name of Intel Corporation nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission .
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* " AS IS " AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT
* LIMITED TO , THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT , INDIRECT , INCIDENTAL ,
* SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT NOT
* LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*/
2011-05-04 18:45:05 -07:00
# include <scsi/sas.h>
2012-01-04 01:33:20 -08:00
# include <linux/bitops.h>
2011-04-22 19:18:03 -07:00
# include "isci.h"
# include "port.h"
# include "remote_device.h"
# include "request.h"
# include "remote_node_context.h"
# include "scu_event_codes.h"
# include "task.h"
2012-02-10 01:18:44 -08:00
# undef C
# define C(a) (#a)
const char * dev_state_name ( enum sci_remote_device_states state )
{
static const char * const strings [ ] = REMOTE_DEV_STATES ;
return strings [ state ] ;
}
# undef C
2012-03-08 22:41:58 -08:00
static enum sci_status sci_remote_device_suspend ( struct isci_remote_device * idev ,
enum sci_remote_node_suspension_reasons reason )
2012-03-08 22:41:52 -08:00
{
2012-03-08 22:41:59 -08:00
return sci_remote_node_context_suspend ( & idev - > rnc , reason ,
SCI_SOFTWARE_SUSPEND_EXPECTED_EVENT ) ;
2012-03-08 22:41:52 -08:00
}
2011-04-27 16:32:45 -07:00
/**
2011-06-30 17:38:32 -07:00
* isci_remote_device_ready ( ) - This function is called by the ihost when the
2011-04-27 16:32:45 -07:00
* remote device is ready . We mark the isci device as ready and signal the
* waiting proccess .
* @ ihost : our valid isci_host
* @ idev : remote device
*
*/
static void isci_remote_device_ready ( struct isci_host * ihost , struct isci_remote_device * idev )
{
dev_dbg ( & ihost - > pdev - > dev ,
" %s: idev = %p \n " , __func__ , idev ) ;
2011-06-23 17:09:02 -07:00
clear_bit ( IDEV_IO_NCQERROR , & idev - > flags ) ;
2011-06-16 11:26:12 -07:00
set_bit ( IDEV_IO_READY , & idev - > flags ) ;
2011-04-27 16:32:45 -07:00
if ( test_and_clear_bit ( IDEV_START_PENDING , & idev - > flags ) )
wake_up ( & ihost - > eventq ) ;
}
2012-03-08 22:41:54 -08:00
static enum sci_status sci_remote_device_terminate_req (
struct isci_host * ihost ,
struct isci_remote_device * idev ,
int check_abort ,
struct isci_request * ireq )
{
dev_dbg ( & ihost - > pdev - > dev ,
" %s: idev=%p; flags=%lx; req=%p; req target=%p \n " ,
__func__ , idev , idev - > flags , ireq , ireq - > target_device ) ;
if ( ! test_bit ( IREQ_ACTIVE , & ireq - > flags ) | |
( ireq - > target_device ! = idev ) | |
( check_abort & & ! test_bit ( IREQ_PENDING_ABORT , & ireq - > flags ) ) )
return SCI_SUCCESS ;
set_bit ( IREQ_ABORT_PATH_ACTIVE , & ireq - > flags ) ;
return sci_controller_terminate_request ( ihost , idev , ireq ) ;
}
static enum sci_status sci_remote_device_terminate_reqs_checkabort (
struct isci_remote_device * idev ,
int chk )
2012-03-08 22:41:51 -08:00
{
2012-03-08 22:41:54 -08:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
enum sci_status status = SCI_SUCCESS ;
u32 i ;
for ( i = 0 ; i < SCI_MAX_IO_REQUESTS ; i + + ) {
struct isci_request * ireq = ihost - > reqs [ i ] ;
enum sci_status s ;
s = sci_remote_device_terminate_req ( ihost , idev , chk , ireq ) ;
if ( s ! = SCI_SUCCESS )
status = s ;
}
return status ;
2012-03-08 22:41:51 -08:00
}
2012-03-08 22:41:54 -08:00
enum sci_status isci_remote_device_terminate_requests (
2012-03-08 22:41:50 -08:00
struct isci_host * ihost ,
2012-03-08 22:41:54 -08:00
struct isci_remote_device * idev ,
struct isci_request * ireq )
2012-03-08 22:41:50 -08:00
{
2012-03-08 22:41:54 -08:00
enum sci_status status = SCI_SUCCESS ;
2012-03-08 22:41:50 -08:00
unsigned long flags ;
spin_lock_irqsave ( & ihost - > scic_lock , flags ) ;
2012-03-08 22:41:54 -08:00
if ( isci_get_device ( idev ) = = NULL ) {
dev_dbg ( & ihost - > pdev - > dev , " %s: failed isci_get_device(idev=%p) \n " ,
__func__ , idev ) ;
2012-03-08 22:41:50 -08:00
spin_unlock_irqrestore ( & ihost - > scic_lock , flags ) ;
status = SCI_FAILURE ;
} else {
2012-03-08 22:41:54 -08:00
dev_dbg ( & ihost - > pdev - > dev ,
" %s: idev=%p, ireq=%p; started_request_count=%d, "
" about to wait \n " ,
__func__ , idev , ireq , idev - > started_request_count ) ;
if ( ireq ) {
/* Terminate a specific TC. */
sci_remote_device_terminate_req ( ihost , idev , 0 , ireq ) ;
spin_unlock_irqrestore ( & ihost - > scic_lock , flags ) ;
wait_event ( ihost - > eventq , ! test_bit ( IREQ_ACTIVE ,
& ireq - > flags ) ) ;
2012-03-08 22:41:50 -08:00
2012-03-08 22:41:54 -08:00
} else {
/* Terminate all TCs. */
sci_remote_device_terminate_requests ( idev ) ;
spin_unlock_irqrestore ( & ihost - > scic_lock , flags ) ;
wait_event ( ihost - > eventq ,
idev - > started_request_count = = 0 ) ;
}
dev_dbg ( & ihost - > pdev - > dev , " %s: idev=%p, wait done \n " ,
__func__ , idev ) ;
2012-03-08 22:41:50 -08:00
isci_put_device ( idev ) ;
}
return status ;
}
2012-03-08 22:41:54 -08:00
/**
* isci_remote_device_not_ready ( ) - This function is called by the ihost when
* the remote device is not ready . We mark the isci device as ready ( not
* " ready_for_io " ) and signal the waiting proccess .
* @ isci_host : This parameter specifies the isci host object .
* @ isci_device : This parameter specifies the remote device
*
* sci_lock is held on entrance to this function .
*/
static void isci_remote_device_not_ready ( struct isci_host * ihost ,
struct isci_remote_device * idev ,
u32 reason )
{
dev_dbg ( & ihost - > pdev - > dev ,
2012-03-08 22:41:54 -08:00
" %s: isci_device = %p; reason = %d \n " , __func__ , idev , reason ) ;
2012-03-08 22:41:54 -08:00
switch ( reason ) {
case SCIC_REMOTE_DEVICE_NOT_READY_SATA_SDB_ERROR_FIS_RECEIVED :
set_bit ( IDEV_IO_NCQERROR , & idev - > flags ) ;
/* Suspend the remote device so the I/O can be terminated. */
2012-03-08 22:41:58 -08:00
sci_remote_device_suspend ( idev , SCI_SW_SUSPEND_NORMAL ) ;
2012-03-08 22:41:54 -08:00
/* Kill all outstanding requests for the device. */
sci_remote_device_terminate_requests ( idev ) ;
/* Fall through into the default case... */
default :
clear_bit ( IDEV_IO_READY , & idev - > flags ) ;
break ;
}
}
2011-05-01 14:19:25 -07:00
/* called once the remote node context is ready to be freed.
* The remote device can now report that its stop operation is complete . none
*/
static void rnc_destruct_done ( void * _dev )
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = _dev ;
2011-04-27 16:32:45 -07:00
2011-06-30 16:31:37 -07:00
BUG_ON ( idev - > started_request_count ! = 0 ) ;
sci_change_state ( & idev - > sm , SCI_DEV_STOPPED ) ;
2011-05-01 14:19:25 -07:00
}
2011-04-27 16:32:45 -07:00
2012-03-08 22:41:50 -08:00
enum sci_status sci_remote_device_terminate_requests (
struct isci_remote_device * idev )
{
2012-03-08 22:41:54 -08:00
return sci_remote_device_terminate_reqs_checkabort ( idev , 0 ) ;
2012-03-08 22:41:50 -08:00
}
2011-06-30 19:14:33 -07:00
enum sci_status sci_remote_device_stop ( struct isci_remote_device * idev ,
2011-05-01 14:19:25 -07:00
u32 timeout )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-01 14:19:25 -07:00
switch ( state ) {
2011-06-02 00:10:43 +00:00
case SCI_DEV_INITIAL :
case SCI_DEV_FAILED :
case SCI_DEV_FINAL :
2011-05-01 14:19:25 -07:00
default :
2012-02-10 01:18:44 -08:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %s \n " ,
__func__ , dev_state_name ( state ) ) ;
2011-05-01 14:19:25 -07:00
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_STOPPED :
2011-05-01 14:19:25 -07:00
return SCI_SUCCESS ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_STARTING :
2011-05-01 14:19:25 -07:00
/* device not started so there had better be no requests */
2011-06-30 16:31:37 -07:00
BUG_ON ( idev - > started_request_count ! = 0 ) ;
2011-06-30 19:14:33 -07:00
sci_remote_node_context_destruct ( & idev - > rnc ,
2011-06-30 16:31:37 -07:00
rnc_destruct_done , idev ) ;
2011-05-01 14:19:25 -07:00
/* Transition to the stopping state and wait for the
* remote node to complete being posted and invalidated .
*/
2011-06-02 00:10:43 +00:00
sci_change_state ( sm , SCI_DEV_STOPPING ) ;
2011-05-01 14:19:25 -07:00
return SCI_SUCCESS ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_READY :
case SCI_STP_DEV_IDLE :
case SCI_STP_DEV_CMD :
case SCI_STP_DEV_NCQ :
case SCI_STP_DEV_NCQ_ERROR :
case SCI_STP_DEV_AWAIT_RESET :
case SCI_SMP_DEV_IDLE :
case SCI_SMP_DEV_CMD :
sci_change_state ( sm , SCI_DEV_STOPPING ) ;
2012-03-08 22:41:52 -08:00
if ( idev - > started_request_count = = 0 )
2011-06-30 19:14:33 -07:00
sci_remote_node_context_destruct ( & idev - > rnc ,
2012-03-08 22:41:52 -08:00
rnc_destruct_done ,
idev ) ;
else {
2012-03-08 22:41:58 -08:00
sci_remote_device_suspend (
idev , SCI_SW_SUSPEND_LINKHANG_DETECT ) ;
2012-03-08 22:41:52 -08:00
sci_remote_device_terminate_requests ( idev ) ;
}
return SCI_SUCCESS ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_STOPPING :
2011-05-01 14:19:25 -07:00
/* All requests should have been terminated, but if there is an
* attempt to stop a device already in the stopping state , then
* try again to terminate .
*/
2011-06-30 19:14:33 -07:00
return sci_remote_device_terminate_requests ( idev ) ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_RESETTING :
sci_change_state ( sm , SCI_DEV_STOPPING ) ;
2011-05-01 14:19:25 -07:00
return SCI_SUCCESS ;
}
2011-04-22 19:18:03 -07:00
}
2011-06-30 19:14:33 -07:00
enum sci_status sci_remote_device_reset ( struct isci_remote_device * idev )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-01 14:48:54 -07:00
switch ( state ) {
2011-06-02 00:10:43 +00:00
case SCI_DEV_INITIAL :
case SCI_DEV_STOPPED :
case SCI_DEV_STARTING :
case SCI_SMP_DEV_IDLE :
case SCI_SMP_DEV_CMD :
case SCI_DEV_STOPPING :
case SCI_DEV_FAILED :
case SCI_DEV_RESETTING :
case SCI_DEV_FINAL :
2011-05-01 14:48:54 -07:00
default :
2012-02-10 01:18:44 -08:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %s \n " ,
__func__ , dev_state_name ( state ) ) ;
2011-05-01 14:48:54 -07:00
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_READY :
case SCI_STP_DEV_IDLE :
case SCI_STP_DEV_CMD :
case SCI_STP_DEV_NCQ :
case SCI_STP_DEV_NCQ_ERROR :
case SCI_STP_DEV_AWAIT_RESET :
sci_change_state ( sm , SCI_DEV_RESETTING ) ;
2011-05-01 14:48:54 -07:00
return SCI_SUCCESS ;
}
2011-04-22 19:18:03 -07:00
}
2011-06-30 19:14:33 -07:00
enum sci_status sci_remote_device_reset_complete ( struct isci_remote_device * idev )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-01 14:53:00 -07:00
2011-06-02 00:10:43 +00:00
if ( state ! = SCI_DEV_RESETTING ) {
2012-02-10 01:18:44 -08:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %s \n " ,
__func__ , dev_state_name ( state ) ) ;
2011-05-01 14:53:00 -07:00
return SCI_FAILURE_INVALID_STATE ;
}
2011-04-22 19:18:03 -07:00
2011-06-02 00:10:43 +00:00
sci_change_state ( sm , SCI_DEV_READY ) ;
2011-05-01 14:53:00 -07:00
return SCI_SUCCESS ;
}
2011-04-22 19:18:03 -07:00
2011-06-30 19:14:33 -07:00
enum sci_status sci_remote_device_frame_handler ( struct isci_remote_device * idev ,
2011-05-01 16:51:11 -07:00
u32 frame_index )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-06-30 17:38:32 -07:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-05-01 16:51:11 -07:00
enum sci_status status ;
switch ( state ) {
2011-06-02 00:10:43 +00:00
case SCI_DEV_INITIAL :
case SCI_DEV_STOPPED :
case SCI_DEV_STARTING :
case SCI_STP_DEV_IDLE :
case SCI_SMP_DEV_IDLE :
case SCI_DEV_FINAL :
2011-05-01 16:51:11 -07:00
default :
2012-02-10 01:18:44 -08:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %s \n " ,
__func__ , dev_state_name ( state ) ) ;
2011-05-01 16:51:11 -07:00
/* Return the frame back to the controller */
2011-06-30 19:14:33 -07:00
sci_controller_release_frame ( ihost , frame_index ) ;
2011-05-01 16:51:11 -07:00
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_READY :
case SCI_STP_DEV_NCQ_ERROR :
case SCI_STP_DEV_AWAIT_RESET :
case SCI_DEV_STOPPING :
case SCI_DEV_FAILED :
case SCI_DEV_RESETTING : {
2011-06-27 14:57:03 -07:00
struct isci_request * ireq ;
2011-05-04 18:45:05 -07:00
struct ssp_frame_hdr hdr ;
void * frame_header ;
ssize_t word_cnt ;
2011-05-01 16:51:11 -07:00
2011-06-30 19:14:33 -07:00
status = sci_unsolicited_frame_control_get_header ( & ihost - > uf_control ,
2011-05-01 16:51:11 -07:00
frame_index ,
2011-05-04 18:45:05 -07:00
& frame_header ) ;
2011-05-01 16:51:11 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-05-04 18:45:05 -07:00
word_cnt = sizeof ( hdr ) / sizeof ( u32 ) ;
sci_swab32_cpy ( & hdr , frame_header , word_cnt ) ;
2011-06-30 19:14:33 -07:00
ireq = sci_request_by_tag ( ihost , be16_to_cpu ( hdr . tag ) ) ;
2011-06-30 16:31:37 -07:00
if ( ireq & & ireq - > target_device = = idev ) {
2011-05-01 16:51:11 -07:00
/* The IO request is now in charge of releasing the frame */
2011-06-30 19:14:33 -07:00
status = sci_io_request_frame_handler ( ireq , frame_index ) ;
2011-05-01 16:51:11 -07:00
} else {
/* We could not map this tag to a valid IO
* request Just toss the frame and continue
*/
2011-06-30 19:14:33 -07:00
sci_controller_release_frame ( ihost , frame_index ) ;
2011-05-01 16:51:11 -07:00
}
break ;
}
2011-06-02 00:10:43 +00:00
case SCI_STP_DEV_NCQ : {
2011-05-04 15:02:03 -07:00
struct dev_to_host_fis * hdr ;
2011-05-01 16:51:11 -07:00
2011-06-30 19:14:33 -07:00
status = sci_unsolicited_frame_control_get_header ( & ihost - > uf_control ,
2011-05-01 16:51:11 -07:00
frame_index ,
( void * * ) & hdr ) ;
if ( status ! = SCI_SUCCESS )
return status ;
2011-05-04 15:02:03 -07:00
if ( hdr - > fis_type = = FIS_SETDEVBITS & &
( hdr - > status & ATA_ERR ) ) {
2011-06-30 16:31:37 -07:00
idev - > not_ready_reason = SCIC_REMOTE_DEVICE_NOT_READY_SATA_SDB_ERROR_FIS_RECEIVED ;
2011-05-01 16:51:11 -07:00
/* TODO Check sactive and complete associated IO if any. */
2011-06-02 00:10:43 +00:00
sci_change_state ( sm , SCI_STP_DEV_NCQ_ERROR ) ;
2011-05-04 15:02:03 -07:00
} else if ( hdr - > fis_type = = FIS_REGD2H & &
( hdr - > status & ATA_ERR ) ) {
2011-05-01 16:51:11 -07:00
/*
* Some devices return D2H FIS when an NCQ error is detected .
* Treat this like an SDB error FIS ready reason .
*/
2011-06-30 16:31:37 -07:00
idev - > not_ready_reason = SCIC_REMOTE_DEVICE_NOT_READY_SATA_SDB_ERROR_FIS_RECEIVED ;
sci_change_state ( & idev - > sm , SCI_STP_DEV_NCQ_ERROR ) ;
2011-05-01 16:51:11 -07:00
} else
status = SCI_FAILURE ;
2011-06-30 19:14:33 -07:00
sci_controller_release_frame ( ihost , frame_index ) ;
2011-05-01 16:51:11 -07:00
break ;
}
2011-06-02 00:10:43 +00:00
case SCI_STP_DEV_CMD :
case SCI_SMP_DEV_CMD :
2011-05-01 16:51:11 -07:00
/* The device does not process any UF received from the hardware while
* in this state . All unsolicited frames are forwarded to the io request
* object .
*/
2011-06-30 19:14:33 -07:00
status = sci_io_request_frame_handler ( idev - > working_request , frame_index ) ;
2011-05-01 16:51:11 -07:00
break ;
}
return status ;
2011-04-22 19:18:03 -07:00
}
2011-06-30 16:31:37 -07:00
static bool is_remote_device_ready ( struct isci_remote_device * idev )
2011-04-22 19:18:03 -07:00
{
2011-05-01 16:26:09 -07:00
2011-06-30 16:31:37 -07:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-01 16:26:09 -07:00
switch ( state ) {
2011-06-02 00:10:43 +00:00
case SCI_DEV_READY :
case SCI_STP_DEV_IDLE :
case SCI_STP_DEV_CMD :
case SCI_STP_DEV_NCQ :
case SCI_STP_DEV_NCQ_ERROR :
case SCI_STP_DEV_AWAIT_RESET :
case SCI_SMP_DEV_IDLE :
case SCI_SMP_DEV_CMD :
2011-05-01 16:26:09 -07:00
return true ;
default :
return false ;
}
}
2011-09-30 18:52:19 -07:00
/*
* called once the remote node context has transisitioned to a ready
* state ( after suspending RX and / or TX due to early D2H fis )
*/
static void atapi_remote_device_resume_done ( void * _dev )
{
struct isci_remote_device * idev = _dev ;
struct isci_request * ireq = idev - > working_request ;
sci_change_state ( & ireq - > sm , SCI_REQ_COMPLETED ) ;
}
2011-06-30 19:14:33 -07:00
enum sci_status sci_remote_device_event_handler ( struct isci_remote_device * idev ,
2011-05-01 16:26:09 -07:00
u32 event_code )
{
enum sci_status status ;
switch ( scu_get_event_type ( event_code ) ) {
case SCU_EVENT_TYPE_RNC_OPS_MISC :
case SCU_EVENT_TYPE_RNC_SUSPEND_TX :
case SCU_EVENT_TYPE_RNC_SUSPEND_TX_RX :
2011-06-30 19:14:33 -07:00
status = sci_remote_node_context_event_handler ( & idev - > rnc , event_code ) ;
2011-05-01 16:26:09 -07:00
break ;
case SCU_EVENT_TYPE_PTX_SCHEDULE_EVENT :
if ( scu_get_event_code ( event_code ) = = SCU_EVENT_IT_NEXUS_TIMEOUT ) {
status = SCI_SUCCESS ;
/* Suspend the associated RNC */
2012-03-08 22:41:58 -08:00
sci_remote_device_suspend ( idev , SCI_SW_SUSPEND_NORMAL ) ;
2011-05-01 16:26:09 -07:00
2011-06-30 16:31:37 -07:00
dev_dbg ( scirdev_to_dev ( idev ) ,
2011-05-01 16:26:09 -07:00
" %s: device: %p event code: %x: %s \n " ,
2011-06-30 16:31:37 -07:00
__func__ , idev , event_code ,
is_remote_device_ready ( idev )
2011-05-01 16:26:09 -07:00
? " I_T_Nexus_Timeout event "
: " I_T_Nexus_Timeout event in wrong state " ) ;
break ;
}
/* Else, fall through and treat as unhandled... */
default :
2011-06-30 16:31:37 -07:00
dev_dbg ( scirdev_to_dev ( idev ) ,
2011-05-01 16:26:09 -07:00
" %s: device: %p event code: %x: %s \n " ,
2011-06-30 16:31:37 -07:00
__func__ , idev , event_code ,
is_remote_device_ready ( idev )
2011-05-01 16:26:09 -07:00
? " unexpected event "
: " unexpected event in wrong state " ) ;
status = SCI_FAILURE_INVALID_STATE ;
break ;
}
if ( status ! = SCI_SUCCESS )
return status ;
return status ;
2011-04-22 19:18:03 -07:00
}
2011-06-30 19:14:33 -07:00
static void sci_remote_device_start_request ( struct isci_remote_device * idev ,
2011-06-27 14:57:03 -07:00
struct isci_request * ireq ,
2011-05-01 14:57:11 -07:00
enum sci_status status )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_port * iport = idev - > owning_port ;
2011-05-01 14:57:11 -07:00
/* cleanup requests that failed after starting on the port */
if ( status ! = SCI_SUCCESS )
2011-06-30 19:14:33 -07:00
sci_port_complete_io ( iport , idev , ireq ) ;
2011-06-13 17:39:44 -07:00
else {
2011-06-30 16:31:37 -07:00
kref_get ( & idev - > kref ) ;
2011-07-01 02:25:15 -07:00
idev - > started_request_count + + ;
2011-06-13 17:39:44 -07:00
}
2011-05-01 14:57:11 -07:00
}
2011-06-30 19:14:33 -07:00
enum sci_status sci_remote_device_start_io ( struct isci_host * ihost ,
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev ,
2011-06-27 14:57:03 -07:00
struct isci_request * ireq )
2011-05-01 14:57:11 -07:00
{
2011-06-30 16:31:37 -07:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-06-30 16:31:37 -07:00
struct isci_port * iport = idev - > owning_port ;
2011-05-01 14:57:11 -07:00
enum sci_status status ;
switch ( state ) {
2011-06-02 00:10:43 +00:00
case SCI_DEV_INITIAL :
case SCI_DEV_STOPPED :
case SCI_DEV_STARTING :
case SCI_STP_DEV_NCQ_ERROR :
case SCI_DEV_STOPPING :
case SCI_DEV_FAILED :
case SCI_DEV_RESETTING :
case SCI_DEV_FINAL :
2011-05-01 14:57:11 -07:00
default :
2012-02-10 01:18:44 -08:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %s \n " ,
__func__ , dev_state_name ( state ) ) ;
2011-05-01 14:57:11 -07:00
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_READY :
2011-05-01 14:57:11 -07:00
/* attempt to start an io request for this device object. The remote
* device object will issue the start request for the io and if
* successful it will start the request for the port object then
* increment its own request count .
*/
2011-06-30 19:14:33 -07:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-01 14:57:11 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-06-30 19:14:33 -07:00
status = sci_remote_node_context_start_io ( & idev - > rnc , ireq ) ;
2011-05-01 14:57:11 -07:00
if ( status ! = SCI_SUCCESS )
break ;
2011-06-30 19:14:33 -07:00
status = sci_request_start ( ireq ) ;
2011-05-01 14:57:11 -07:00
break ;
2011-06-02 00:10:43 +00:00
case SCI_STP_DEV_IDLE : {
2011-05-01 14:57:11 -07:00
/* handle the start io operation for a sata device that is in
* the command idle state . - Evalute the type of IO request to
* be started - If its an NCQ request change to NCQ substate -
* If its any other command change to the CMD substate
*
* If this is a softreset we may want to have a different
* substate .
*/
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states new_state ;
2011-05-04 15:02:03 -07:00
struct sas_task * task = isci_request_access_task ( ireq ) ;
2011-05-01 14:57:11 -07:00
2011-06-30 19:14:33 -07:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-01 14:57:11 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-06-30 19:14:33 -07:00
status = sci_remote_node_context_start_io ( & idev - > rnc , ireq ) ;
2011-05-01 14:57:11 -07:00
if ( status ! = SCI_SUCCESS )
break ;
2011-06-30 19:14:33 -07:00
status = sci_request_start ( ireq ) ;
2011-05-01 14:57:11 -07:00
if ( status ! = SCI_SUCCESS )
break ;
2011-05-04 15:02:03 -07:00
if ( task - > ata_task . use_ncq )
2011-06-02 00:10:43 +00:00
new_state = SCI_STP_DEV_NCQ ;
2011-05-01 14:57:11 -07:00
else {
2011-06-30 16:31:37 -07:00
idev - > working_request = ireq ;
2011-06-02 00:10:43 +00:00
new_state = SCI_STP_DEV_CMD ;
2011-05-01 14:57:11 -07:00
}
2011-06-02 00:10:43 +00:00
sci_change_state ( sm , new_state ) ;
2011-05-01 14:57:11 -07:00
break ;
}
2011-06-02 00:10:43 +00:00
case SCI_STP_DEV_NCQ : {
2011-05-04 15:02:03 -07:00
struct sas_task * task = isci_request_access_task ( ireq ) ;
if ( task - > ata_task . use_ncq ) {
2011-06-30 19:14:33 -07:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-01 14:57:11 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-06-30 19:14:33 -07:00
status = sci_remote_node_context_start_io ( & idev - > rnc , ireq ) ;
2011-05-01 14:57:11 -07:00
if ( status ! = SCI_SUCCESS )
break ;
2011-06-30 19:14:33 -07:00
status = sci_request_start ( ireq ) ;
2011-05-01 14:57:11 -07:00
} else
return SCI_FAILURE_INVALID_STATE ;
break ;
2011-05-04 15:02:03 -07:00
}
2011-06-02 00:10:43 +00:00
case SCI_STP_DEV_AWAIT_RESET :
2011-05-01 14:57:11 -07:00
return SCI_FAILURE_REMOTE_DEVICE_RESET_REQUIRED ;
2011-06-02 00:10:43 +00:00
case SCI_SMP_DEV_IDLE :
2011-06-30 19:14:33 -07:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-01 14:57:11 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-06-30 19:14:33 -07:00
status = sci_remote_node_context_start_io ( & idev - > rnc , ireq ) ;
2011-05-01 14:57:11 -07:00
if ( status ! = SCI_SUCCESS )
break ;
2011-06-30 19:14:33 -07:00
status = sci_request_start ( ireq ) ;
2011-05-01 14:57:11 -07:00
if ( status ! = SCI_SUCCESS )
break ;
2011-06-30 16:31:37 -07:00
idev - > working_request = ireq ;
sci_change_state ( & idev - > sm , SCI_SMP_DEV_CMD ) ;
2011-05-01 14:57:11 -07:00
break ;
2011-06-02 00:10:43 +00:00
case SCI_STP_DEV_CMD :
case SCI_SMP_DEV_CMD :
2011-05-01 14:57:11 -07:00
/* device is already handling a command it can not accept new commands
* until this one is complete .
*/
return SCI_FAILURE_INVALID_STATE ;
}
2011-06-30 19:14:33 -07:00
sci_remote_device_start_request ( idev , ireq , status ) ;
2011-05-01 14:57:11 -07:00
return status ;
2011-04-22 19:18:03 -07:00
}
2011-06-29 13:09:25 -07:00
static enum sci_status common_complete_io ( struct isci_port * iport ,
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev ,
2011-06-27 14:57:03 -07:00
struct isci_request * ireq )
2011-04-22 19:18:03 -07:00
{
2011-05-01 15:33:43 -07:00
enum sci_status status ;
2011-06-30 19:14:33 -07:00
status = sci_request_complete ( ireq ) ;
2011-05-01 15:33:43 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-06-30 19:14:33 -07:00
status = sci_port_complete_io ( iport , idev , ireq ) ;
2011-05-01 15:33:43 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-06-30 19:14:33 -07:00
sci_remote_device_decrement_request_count ( idev ) ;
2011-05-01 15:33:43 -07:00
return status ;
}
2011-06-30 19:14:33 -07:00
enum sci_status sci_remote_device_complete_io ( struct isci_host * ihost ,
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev ,
2011-06-27 14:57:03 -07:00
struct isci_request * ireq )
2011-05-01 15:33:43 -07:00
{
2011-06-30 16:31:37 -07:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-06-30 16:31:37 -07:00
struct isci_port * iport = idev - > owning_port ;
2011-05-01 15:33:43 -07:00
enum sci_status status ;
switch ( state ) {
2011-06-02 00:10:43 +00:00
case SCI_DEV_INITIAL :
case SCI_DEV_STOPPED :
case SCI_DEV_STARTING :
case SCI_STP_DEV_IDLE :
case SCI_SMP_DEV_IDLE :
case SCI_DEV_FAILED :
case SCI_DEV_FINAL :
2011-05-01 15:33:43 -07:00
default :
2012-02-10 01:18:44 -08:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %s \n " ,
__func__ , dev_state_name ( state ) ) ;
2011-05-01 15:33:43 -07:00
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_READY :
case SCI_STP_DEV_AWAIT_RESET :
case SCI_DEV_RESETTING :
2011-06-30 16:31:37 -07:00
status = common_complete_io ( iport , idev , ireq ) ;
2011-05-01 15:33:43 -07:00
break ;
2011-06-02 00:10:43 +00:00
case SCI_STP_DEV_CMD :
case SCI_STP_DEV_NCQ :
case SCI_STP_DEV_NCQ_ERROR :
2011-09-30 18:52:19 -07:00
case SCI_STP_DEV_ATAPI_ERROR :
2011-06-30 16:31:37 -07:00
status = common_complete_io ( iport , idev , ireq ) ;
2011-05-01 15:33:43 -07:00
if ( status ! = SCI_SUCCESS )
break ;
2011-06-27 14:57:03 -07:00
if ( ireq - > sci_status = = SCI_FAILURE_REMOTE_DEVICE_RESET_REQUIRED ) {
2011-05-01 15:33:43 -07:00
/* This request causes hardware error, device needs to be Lun Reset.
* So here we force the state machine to IDLE state so the rest IOs
* can reach RNC state handler , these IOs will be completed by RNC with
* status of " DEVICE_RESET_REQUIRED " , instead of " INVALID STATE " .
*/
2011-06-02 00:10:43 +00:00
sci_change_state ( sm , SCI_STP_DEV_AWAIT_RESET ) ;
2011-07-01 02:25:15 -07:00
} else if ( idev - > started_request_count = = 0 )
2011-06-02 00:10:43 +00:00
sci_change_state ( sm , SCI_STP_DEV_IDLE ) ;
2011-05-01 15:33:43 -07:00
break ;
2011-06-02 00:10:43 +00:00
case SCI_SMP_DEV_CMD :
2011-06-30 16:31:37 -07:00
status = common_complete_io ( iport , idev , ireq ) ;
2011-05-01 15:33:43 -07:00
if ( status ! = SCI_SUCCESS )
break ;
2011-06-02 00:10:43 +00:00
sci_change_state ( sm , SCI_SMP_DEV_IDLE ) ;
2011-05-01 15:33:43 -07:00
break ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_STOPPING :
2011-06-30 16:31:37 -07:00
status = common_complete_io ( iport , idev , ireq ) ;
2011-05-01 15:33:43 -07:00
if ( status ! = SCI_SUCCESS )
break ;
2011-07-01 02:25:15 -07:00
if ( idev - > started_request_count = = 0 )
2011-06-30 19:14:33 -07:00
sci_remote_node_context_destruct ( & idev - > rnc ,
2011-07-01 02:25:15 -07:00
rnc_destruct_done ,
idev ) ;
2011-05-01 15:33:43 -07:00
break ;
}
if ( status ! = SCI_SUCCESS )
2011-06-30 16:31:37 -07:00
dev_err ( scirdev_to_dev ( idev ) ,
2011-05-01 15:33:43 -07:00
" %s: Port:0x%p Device:0x%p Request:0x%p Status:0x%x "
2011-06-29 13:09:25 -07:00
" could not complete \n " , __func__ , iport ,
2011-06-30 16:31:37 -07:00
idev , ireq , status ) ;
2011-06-13 17:39:44 -07:00
else
2011-06-30 16:31:37 -07:00
isci_put_device ( idev ) ;
2011-05-01 15:33:43 -07:00
return status ;
2011-04-22 19:18:03 -07:00
}
2011-06-30 19:14:33 -07:00
static void sci_remote_device_continue_request ( void * dev )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = dev ;
2011-05-01 15:53:25 -07:00
/* we need to check if this request is still valid to continue. */
2011-06-30 16:31:37 -07:00
if ( idev - > working_request )
2011-06-30 19:14:33 -07:00
sci_controller_continue_io ( idev - > working_request ) ;
2011-05-01 15:53:25 -07:00
}
2011-06-30 19:14:33 -07:00
enum sci_status sci_remote_device_start_task ( struct isci_host * ihost ,
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev ,
2011-06-27 14:57:03 -07:00
struct isci_request * ireq )
2011-05-01 15:53:25 -07:00
{
2011-06-30 16:31:37 -07:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-06-30 16:31:37 -07:00
struct isci_port * iport = idev - > owning_port ;
2011-05-01 15:53:25 -07:00
enum sci_status status ;
switch ( state ) {
2011-06-02 00:10:43 +00:00
case SCI_DEV_INITIAL :
case SCI_DEV_STOPPED :
case SCI_DEV_STARTING :
case SCI_SMP_DEV_IDLE :
case SCI_SMP_DEV_CMD :
case SCI_DEV_STOPPING :
case SCI_DEV_FAILED :
case SCI_DEV_RESETTING :
case SCI_DEV_FINAL :
2011-05-01 15:53:25 -07:00
default :
2012-02-10 01:18:44 -08:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %s \n " ,
__func__ , dev_state_name ( state ) ) ;
2011-05-01 15:53:25 -07:00
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 00:10:43 +00:00
case SCI_STP_DEV_IDLE :
case SCI_STP_DEV_CMD :
case SCI_STP_DEV_NCQ :
case SCI_STP_DEV_NCQ_ERROR :
case SCI_STP_DEV_AWAIT_RESET :
2011-06-30 19:14:33 -07:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-01 15:53:25 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-06-30 19:14:33 -07:00
status = sci_request_start ( ireq ) ;
2011-05-01 15:53:25 -07:00
if ( status ! = SCI_SUCCESS )
goto out ;
/* Note: If the remote device state is not IDLE this will
* replace the request that probably resulted in the task
* management request .
*/
2011-06-30 16:31:37 -07:00
idev - > working_request = ireq ;
2011-06-02 00:10:43 +00:00
sci_change_state ( sm , SCI_STP_DEV_CMD ) ;
2011-05-01 15:53:25 -07:00
/* The remote node context must cleanup the TCi to NCQ mapping
* table . The only way to do this correctly is to either write
* to the TLCR register or to invalidate and repost the RNC . In
* either case the remote node context state machine will take
* the correct action when the remote node context is suspended
* and later resumed .
*/
2012-03-08 22:41:58 -08:00
sci_remote_device_suspend ( idev ,
SCI_SW_SUSPEND_LINKHANG_DETECT ) ;
2012-03-08 22:41:54 -08:00
status = sci_remote_node_context_start_task ( & idev - > rnc , ireq ,
sci_remote_device_continue_request , idev ) ;
2011-05-01 15:53:25 -07:00
out :
2011-06-30 19:14:33 -07:00
sci_remote_device_start_request ( idev , ireq , status ) ;
2011-05-01 15:53:25 -07:00
/* We need to let the controller start request handler know that
* it can ' t post TC yet . We will provide a callback function to
* post TC when RNC gets resumed .
*/
return SCI_FAILURE_RESET_DEVICE_PARTIAL_SUCCESS ;
2011-06-02 00:10:43 +00:00
case SCI_DEV_READY :
2011-06-30 19:14:33 -07:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-01 15:53:25 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2012-03-08 22:41:54 -08:00
/* Resume the RNC as needed: */
status = sci_remote_node_context_start_task ( & idev - > rnc , ireq ,
NULL , NULL ) ;
2011-05-01 15:53:25 -07:00
if ( status ! = SCI_SUCCESS )
break ;
2011-06-30 19:14:33 -07:00
status = sci_request_start ( ireq ) ;
2011-05-01 15:53:25 -07:00
break ;
}
2011-06-30 19:14:33 -07:00
sci_remote_device_start_request ( idev , ireq , status ) ;
2011-05-01 15:53:25 -07:00
return status ;
2011-04-22 19:18:03 -07:00
}
2011-07-01 02:25:15 -07:00
void sci_remote_device_post_request ( struct isci_remote_device * idev , u32 request )
2011-04-22 19:18:03 -07:00
{
2011-07-01 02:25:15 -07:00
struct isci_port * iport = idev - > owning_port ;
2011-04-22 19:18:03 -07:00
u32 context ;
2011-07-01 02:25:15 -07:00
context = request |
( ISCI_PEG < < SCU_CONTEXT_COMMAND_PROTOCOL_ENGINE_GROUP_SHIFT ) |
( iport - > physical_port_index < < SCU_CONTEXT_COMMAND_LOGICAL_PORT_SHIFT ) |
idev - > rnc . remote_node_index ;
2011-04-22 19:18:03 -07:00
2011-07-01 02:25:15 -07:00
sci_controller_post_request ( iport - > owning_controller , context ) ;
2011-04-22 19:18:03 -07:00
}
2011-04-27 16:32:45 -07:00
/* called once the remote node context has transisitioned to a
2011-04-22 19:18:03 -07:00
* ready state . This is the indication that the remote device object can also
2011-04-27 16:32:45 -07:00
* transition to ready .
2011-04-22 19:18:03 -07:00
*/
2011-05-01 14:05:57 -07:00
static void remote_device_resume_done ( void * _dev )
2011-04-27 16:32:45 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = _dev ;
2011-04-27 16:32:45 -07:00
2011-06-30 16:31:37 -07:00
if ( is_remote_device_ready ( idev ) )
2011-05-01 16:26:09 -07:00
return ;
2011-04-22 19:18:03 -07:00
2011-05-01 16:26:09 -07:00
/* go 'ready' if we are not already in a ready state */
2011-06-30 16:31:37 -07:00
sci_change_state ( & idev - > sm , SCI_DEV_READY ) ;
2011-04-22 19:18:03 -07:00
}
2011-06-30 19:14:33 -07:00
static void sci_stp_remote_device_ready_idle_substate_resume_complete_handler ( void * _dev )
2011-04-27 16:32:45 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = _dev ;
2011-06-30 17:38:32 -07:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-27 16:32:45 -07:00
/* For NCQ operation we do not issue a isci_remote_device_not_ready().
* As a result , avoid sending the ready notification .
*/
2011-06-30 16:31:37 -07:00
if ( idev - > sm . previous_state_id ! = SCI_STP_DEV_NCQ )
2011-06-30 17:38:32 -07:00
isci_remote_device_ready ( ihost , idev ) ;
2011-04-27 16:32:45 -07:00
}
2011-06-30 19:14:33 -07:00
static void sci_remote_device_initial_state_enter ( struct sci_base_state_machine * sm )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-04-22 19:18:03 -07:00
2011-04-27 16:32:45 -07:00
/* Initial state is a transitional state to the stopped state */
2011-06-30 16:31:37 -07:00
sci_change_state ( & idev - > sm , SCI_DEV_STOPPED ) ;
2011-04-22 19:18:03 -07:00
}
2011-07-02 22:56:22 -07:00
2011-04-22 19:18:03 -07:00
/**
2011-06-30 19:14:33 -07:00
* sci_remote_device_destruct ( ) - free remote node context and destruct
2011-04-22 19:18:03 -07:00
* @ remote_device : This parameter specifies the remote device to be destructed .
*
* Remote device objects are a limited resource . As such , they must be
* protected . Thus calls to construct and destruct are mutually exclusive and
* non - reentrant . The return value shall indicate if the device was
* successfully destructed or if some failure occurred . enum sci_status This value
* is returned if the device is successfully destructed .
* SCI_FAILURE_INVALID_REMOTE_DEVICE This value is returned if the supplied
* device isn ' t valid ( e . g . it ' s already been destoryed , the handle isn ' t
* valid , etc . ) .
*/
2011-06-30 19:14:33 -07:00
static enum sci_status sci_remote_device_destruct ( struct isci_remote_device * idev )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-06-30 17:38:32 -07:00
struct isci_host * ihost ;
2011-05-01 14:38:26 -07:00
2011-06-02 00:10:43 +00:00
if ( state ! = SCI_DEV_STOPPED ) {
2012-02-10 01:18:44 -08:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %s \n " ,
__func__ , dev_state_name ( state ) ) ;
2011-05-01 14:38:26 -07:00
return SCI_FAILURE_INVALID_STATE ;
}
2011-06-30 17:38:32 -07:00
ihost = idev - > owning_port - > owning_controller ;
2011-06-30 19:14:33 -07:00
sci_controller_free_remote_node_context ( ihost , idev ,
2011-06-30 16:31:37 -07:00
idev - > rnc . remote_node_index ) ;
idev - > rnc . remote_node_index = SCIC_SDS_REMOTE_NODE_CONTEXT_INVALID_INDEX ;
2011-06-02 00:10:43 +00:00
sci_change_state ( sm , SCI_DEV_FINAL ) ;
2011-05-01 14:38:26 -07:00
return SCI_SUCCESS ;
2011-04-22 19:18:03 -07:00
}
2011-07-02 22:56:22 -07:00
/**
* isci_remote_device_deconstruct ( ) - This function frees an isci_remote_device .
2011-03-03 17:59:32 -08:00
* @ ihost : This parameter specifies the isci host object .
* @ idev : This parameter specifies the remote device to be freed .
2011-07-02 22:56:22 -07:00
*
*/
2011-03-03 17:59:32 -08:00
static void isci_remote_device_deconstruct ( struct isci_host * ihost , struct isci_remote_device * idev )
2011-07-02 22:56:22 -07:00
{
2011-03-03 17:59:32 -08:00
dev_dbg ( & ihost - > pdev - > dev ,
" %s: isci_device = %p \n " , __func__ , idev ) ;
2011-07-02 22:56:22 -07:00
/* There should not be any outstanding io's. All paths to
* here should go through isci_remote_device_nuke_requests .
* If we hit this condition , we will need a way to complete
* io requests in process */
2012-03-08 22:41:57 -08:00
BUG_ON ( idev - > started_request_count > 0 ) ;
2011-07-02 22:56:22 -07:00
2011-06-30 19:14:33 -07:00
sci_remote_device_destruct ( idev ) ;
2011-03-03 17:59:32 -08:00
list_del_init ( & idev - > node ) ;
2011-06-13 17:39:44 -07:00
isci_put_device ( idev ) ;
2011-07-02 22:56:22 -07:00
}
2011-06-30 19:14:33 -07:00
static void sci_remote_device_stopped_state_enter ( struct sci_base_state_machine * sm )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-06-30 17:38:32 -07:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-22 19:18:03 -07:00
u32 prev_state ;
/* If we are entering from the stopping state let the SCI User know that
* the stop operation has completed .
*/
2011-06-30 16:31:37 -07:00
prev_state = idev - > sm . previous_state_id ;
2011-06-02 00:10:43 +00:00
if ( prev_state = = SCI_DEV_STOPPING )
2011-06-30 17:38:32 -07:00
isci_remote_device_deconstruct ( ihost , idev ) ;
2011-04-22 19:18:03 -07:00
2011-06-30 19:14:33 -07:00
sci_controller_remote_device_stopped ( ihost , idev ) ;
2011-04-22 19:18:03 -07:00
}
2011-06-30 19:14:33 -07:00
static void sci_remote_device_starting_state_enter ( struct sci_base_state_machine * sm )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 02:25:15 -07:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-22 19:18:03 -07:00
isci_remote_device_not_ready ( ihost , idev ,
SCIC_REMOTE_DEVICE_NOT_READY_START_REQUESTED ) ;
}
2011-06-30 19:14:33 -07:00
static void sci_remote_device_ready_state_enter ( struct sci_base_state_machine * sm )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-06-30 17:38:32 -07:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-05-04 07:58:16 +00:00
struct domain_device * dev = idev - > domain_dev ;
2011-04-22 19:18:03 -07:00
2011-04-27 16:32:45 -07:00
if ( dev - > dev_type = = SATA_DEV | | ( dev - > tproto & SAS_PROTOCOL_SATA ) ) {
2011-06-30 16:31:37 -07:00
sci_change_state ( & idev - > sm , SCI_STP_DEV_IDLE ) ;
2011-04-27 16:32:45 -07:00
} else if ( dev_is_expander ( dev ) ) {
2011-06-30 16:31:37 -07:00
sci_change_state ( & idev - > sm , SCI_SMP_DEV_IDLE ) ;
2011-04-27 16:32:45 -07:00
} else
2011-06-30 17:38:32 -07:00
isci_remote_device_ready ( ihost , idev ) ;
2011-04-22 19:18:03 -07:00
}
2011-06-30 19:14:33 -07:00
static void sci_remote_device_ready_state_exit ( struct sci_base_state_machine * sm )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
struct domain_device * dev = idev - > domain_dev ;
2011-04-27 16:32:45 -07:00
if ( dev - > dev_type = = SAS_END_DEV ) {
2011-06-30 17:38:32 -07:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-22 19:18:03 -07:00
2011-06-30 17:38:32 -07:00
isci_remote_device_not_ready ( ihost , idev ,
2011-04-22 19:18:03 -07:00
SCIC_REMOTE_DEVICE_NOT_READY_STOP_REQUESTED ) ;
}
}
2011-06-30 19:14:33 -07:00
static void sci_remote_device_resetting_state_enter ( struct sci_base_state_machine * sm )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2012-03-08 22:41:50 -08:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
dev_dbg ( & ihost - > pdev - > dev ,
" %s: isci_device = %p \n " , __func__ , idev ) ;
2011-04-22 19:18:03 -07:00
2012-03-08 22:41:58 -08:00
sci_remote_device_suspend ( idev , SCI_SW_SUSPEND_LINKHANG_DETECT ) ;
2011-04-22 19:18:03 -07:00
}
2011-06-30 19:14:33 -07:00
static void sci_remote_device_resetting_state_exit ( struct sci_base_state_machine * sm )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2012-03-08 22:41:50 -08:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
dev_dbg ( & ihost - > pdev - > dev ,
" %s: isci_device = %p \n " , __func__ , idev ) ;
2011-04-22 19:18:03 -07:00
2011-06-30 19:14:33 -07:00
sci_remote_node_context_resume ( & idev - > rnc , NULL , NULL ) ;
2011-04-22 19:18:03 -07:00
}
2011-06-30 19:14:33 -07:00
static void sci_stp_remote_device_ready_idle_substate_enter ( struct sci_base_state_machine * sm )
2011-04-27 16:32:45 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-04-27 16:32:45 -07:00
2011-06-30 16:31:37 -07:00
idev - > working_request = NULL ;
2011-06-30 19:14:33 -07:00
if ( sci_remote_node_context_is_ready ( & idev - > rnc ) ) {
2011-04-27 16:32:45 -07:00
/*
* Since the RNC is ready , it ' s alright to finish completion
* processing ( e . g . signal the remote device is ready ) . */
2011-06-30 19:14:33 -07:00
sci_stp_remote_device_ready_idle_substate_resume_complete_handler ( idev ) ;
2011-04-27 16:32:45 -07:00
} else {
2011-06-30 19:14:33 -07:00
sci_remote_node_context_resume ( & idev - > rnc ,
sci_stp_remote_device_ready_idle_substate_resume_complete_handler ,
2011-06-30 16:31:37 -07:00
idev ) ;
2011-04-27 16:32:45 -07:00
}
}
2011-06-30 19:14:33 -07:00
static void sci_stp_remote_device_ready_cmd_substate_enter ( struct sci_base_state_machine * sm )
2011-04-27 16:32:45 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 02:25:15 -07:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-27 16:32:45 -07:00
2011-06-30 16:31:37 -07:00
BUG_ON ( idev - > working_request = = NULL ) ;
2011-04-27 16:32:45 -07:00
2011-06-30 17:38:32 -07:00
isci_remote_device_not_ready ( ihost , idev ,
2011-04-27 16:32:45 -07:00
SCIC_REMOTE_DEVICE_NOT_READY_SATA_REQUEST_STARTED ) ;
}
2011-06-30 19:14:33 -07:00
static void sci_stp_remote_device_ready_ncq_error_substate_enter ( struct sci_base_state_machine * sm )
2011-04-27 16:32:45 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 02:25:15 -07:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-27 16:32:45 -07:00
2011-06-30 16:31:37 -07:00
if ( idev - > not_ready_reason = = SCIC_REMOTE_DEVICE_NOT_READY_SATA_SDB_ERROR_FIS_RECEIVED )
2011-06-30 17:38:32 -07:00
isci_remote_device_not_ready ( ihost , idev ,
2011-06-30 16:31:37 -07:00
idev - > not_ready_reason ) ;
2011-04-27 16:32:45 -07:00
}
2012-03-08 22:41:50 -08:00
static void sci_stp_remote_device_atapi_error_substate_enter (
struct sci_base_state_machine * sm )
{
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
/* This state is entered when an I/O is decoded with an error
* condition . By this point the RNC expected suspension state is set .
* The error conditions suspend the device , so unsuspend here if
* possible .
*/
sci_remote_node_context_resume ( & idev - > rnc ,
atapi_remote_device_resume_done ,
idev ) ;
}
2011-06-30 19:14:33 -07:00
static void sci_smp_remote_device_ready_idle_substate_enter ( struct sci_base_state_machine * sm )
2011-04-27 16:32:45 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 02:25:15 -07:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-27 16:32:45 -07:00
2011-06-30 17:38:32 -07:00
isci_remote_device_ready ( ihost , idev ) ;
2011-04-27 16:32:45 -07:00
}
2011-06-30 19:14:33 -07:00
static void sci_smp_remote_device_ready_cmd_substate_enter ( struct sci_base_state_machine * sm )
2011-04-27 16:32:45 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 02:25:15 -07:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-27 16:32:45 -07:00
2011-06-30 16:31:37 -07:00
BUG_ON ( idev - > working_request = = NULL ) ;
2011-04-27 16:32:45 -07:00
2011-06-30 17:38:32 -07:00
isci_remote_device_not_ready ( ihost , idev ,
2011-04-27 16:32:45 -07:00
SCIC_REMOTE_DEVICE_NOT_READY_SMP_REQUEST_STARTED ) ;
}
2011-06-30 19:14:33 -07:00
static void sci_smp_remote_device_ready_cmd_substate_exit ( struct sci_base_state_machine * sm )
2011-04-27 16:32:45 -07:00
{
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-04-27 16:32:45 -07:00
2011-06-30 16:31:37 -07:00
idev - > working_request = NULL ;
2011-04-27 16:32:45 -07:00
}
2011-04-22 19:18:03 -07:00
2011-06-30 19:14:33 -07:00
static const struct sci_base_state sci_remote_device_state_table [ ] = {
2011-06-02 00:10:43 +00:00
[ SCI_DEV_INITIAL ] = {
2011-06-30 19:14:33 -07:00
. enter_state = sci_remote_device_initial_state_enter ,
2011-04-22 19:18:03 -07:00
} ,
2011-06-02 00:10:43 +00:00
[ SCI_DEV_STOPPED ] = {
2011-06-30 19:14:33 -07:00
. enter_state = sci_remote_device_stopped_state_enter ,
2011-04-22 19:18:03 -07:00
} ,
2011-06-02 00:10:43 +00:00
[ SCI_DEV_STARTING ] = {
2011-06-30 19:14:33 -07:00
. enter_state = sci_remote_device_starting_state_enter ,
2011-04-22 19:18:03 -07:00
} ,
2011-06-02 00:10:43 +00:00
[ SCI_DEV_READY ] = {
2011-06-30 19:14:33 -07:00
. enter_state = sci_remote_device_ready_state_enter ,
. exit_state = sci_remote_device_ready_state_exit
2011-04-22 19:18:03 -07:00
} ,
2011-06-02 00:10:43 +00:00
[ SCI_STP_DEV_IDLE ] = {
2011-06-30 19:14:33 -07:00
. enter_state = sci_stp_remote_device_ready_idle_substate_enter ,
2011-04-27 16:32:45 -07:00
} ,
2011-06-02 00:10:43 +00:00
[ SCI_STP_DEV_CMD ] = {
2011-06-30 19:14:33 -07:00
. enter_state = sci_stp_remote_device_ready_cmd_substate_enter ,
2011-04-27 16:32:45 -07:00
} ,
2011-06-02 00:10:43 +00:00
[ SCI_STP_DEV_NCQ ] = { } ,
[ SCI_STP_DEV_NCQ_ERROR ] = {
2011-06-30 19:14:33 -07:00
. enter_state = sci_stp_remote_device_ready_ncq_error_substate_enter ,
2011-04-27 16:32:45 -07:00
} ,
2012-03-08 22:41:50 -08:00
[ SCI_STP_DEV_ATAPI_ERROR ] = {
. enter_state = sci_stp_remote_device_atapi_error_substate_enter ,
} ,
2011-06-02 00:10:43 +00:00
[ SCI_STP_DEV_AWAIT_RESET ] = { } ,
[ SCI_SMP_DEV_IDLE ] = {
2011-06-30 19:14:33 -07:00
. enter_state = sci_smp_remote_device_ready_idle_substate_enter ,
2011-04-27 16:32:45 -07:00
} ,
2011-06-02 00:10:43 +00:00
[ SCI_SMP_DEV_CMD ] = {
2011-06-30 19:14:33 -07:00
. enter_state = sci_smp_remote_device_ready_cmd_substate_enter ,
. exit_state = sci_smp_remote_device_ready_cmd_substate_exit ,
2011-04-27 16:32:45 -07:00
} ,
2011-06-02 00:10:43 +00:00
[ SCI_DEV_STOPPING ] = { } ,
[ SCI_DEV_FAILED ] = { } ,
[ SCI_DEV_RESETTING ] = {
2011-06-30 19:14:33 -07:00
. enter_state = sci_remote_device_resetting_state_enter ,
. exit_state = sci_remote_device_resetting_state_exit
2011-04-22 19:18:03 -07:00
} ,
2011-06-02 00:10:43 +00:00
[ SCI_DEV_FINAL ] = { } ,
2011-04-22 19:18:03 -07:00
} ;
/**
2011-06-30 19:14:33 -07:00
* sci_remote_device_construct ( ) - common construction
2011-04-22 19:18:03 -07:00
* @ sci_port : SAS / SATA port through which this device is accessed .
* @ sci_dev : remote device to construct
*
2011-04-25 11:48:29 -07:00
* This routine just performs benign initialization and does not
* allocate the remote_node_context which is left to
2011-06-30 19:14:33 -07:00
* sci_remote_device_ [ de ] a_construct ( ) . sci_remote_device_destruct ( )
2011-04-25 11:48:29 -07:00
* frees the remote_node_context ( s ) for the device .
2011-04-22 19:18:03 -07:00
*/
2011-06-30 19:14:33 -07:00
static void sci_remote_device_construct ( struct isci_port * iport ,
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
idev - > owning_port = iport ;
idev - > started_request_count = 0 ;
2011-04-22 19:18:03 -07:00
2011-06-30 19:14:33 -07:00
sci_init_sm ( & idev - > sm , sci_remote_device_state_table , SCI_DEV_INITIAL ) ;
2011-04-22 19:18:03 -07:00
2011-06-30 19:14:33 -07:00
sci_remote_node_context_construct ( & idev - > rnc ,
2011-04-22 19:18:03 -07:00
SCIC_SDS_REMOTE_NODE_CONTEXT_INVALID_INDEX ) ;
}
/**
2011-06-30 19:14:33 -07:00
* sci_remote_device_da_construct ( ) - construct direct attached device .
2011-04-25 11:48:29 -07:00
*
* The information ( e . g . IAF , Signature FIS , etc . ) necessary to build
* the device is known to the SCI Core since it is contained in the
2011-06-30 19:14:33 -07:00
* sci_phy object . Remote node context ( s ) is / are a global resource
* allocated by this routine , freed by sci_remote_device_destruct ( ) .
2011-04-25 11:48:29 -07:00
*
* Returns :
* SCI_FAILURE_DEVICE_EXISTS - device has already been constructed .
* SCI_FAILURE_UNSUPPORTED_PROTOCOL - e . g . sas device attached to
* sata - only controller instance .
* SCI_FAILURE_INSUFFICIENT_RESOURCES - remote node contexts exhausted .
2011-04-22 19:18:03 -07:00
*/
2011-06-30 19:14:33 -07:00
static enum sci_status sci_remote_device_da_construct ( struct isci_port * iport ,
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev )
2011-04-22 19:18:03 -07:00
{
enum sci_status status ;
2012-01-04 01:33:20 -08:00
struct sci_port_properties properties ;
2011-04-22 19:18:03 -07:00
2011-06-30 19:14:33 -07:00
sci_remote_device_construct ( iport , idev ) ;
2011-04-25 11:48:29 -07:00
2012-01-04 01:33:20 -08:00
sci_port_get_properties ( iport , & properties ) ;
/* Get accurate port width from port's phy mask for a DA device. */
idev - > device_port_width = hweight32 ( properties . phy_mask ) ;
2011-06-30 19:14:33 -07:00
status = sci_controller_allocate_remote_node_context ( iport - > owning_controller ,
2012-02-01 00:23:10 -08:00
idev ,
& idev - > rnc . remote_node_index ) ;
2011-04-22 19:18:03 -07:00
2011-04-21 18:44:45 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-04-22 19:18:03 -07:00
2011-06-30 19:14:33 -07:00
idev - > connection_rate = sci_port_get_max_allowed_speed ( iport ) ;
2011-04-22 19:18:03 -07:00
2011-04-21 18:44:45 -07:00
return SCI_SUCCESS ;
2011-04-22 19:18:03 -07:00
}
/**
2011-06-30 19:14:33 -07:00
* sci_remote_device_ea_construct ( ) - construct expander attached device
2011-04-25 11:48:29 -07:00
*
* Remote node context ( s ) is / are a global resource allocated by this
2011-06-30 19:14:33 -07:00
* routine , freed by sci_remote_device_destruct ( ) .
2011-04-25 11:48:29 -07:00
*
* Returns :
* SCI_FAILURE_DEVICE_EXISTS - device has already been constructed .
* SCI_FAILURE_UNSUPPORTED_PROTOCOL - e . g . sas device attached to
* sata - only controller instance .
* SCI_FAILURE_INSUFFICIENT_RESOURCES - remote node contexts exhausted .
2011-04-22 19:18:03 -07:00
*/
2011-06-30 19:14:33 -07:00
static enum sci_status sci_remote_device_ea_construct ( struct isci_port * iport ,
2011-06-30 16:31:37 -07:00
struct isci_remote_device * idev )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct domain_device * dev = idev - > domain_dev ;
2011-04-22 19:18:03 -07:00
enum sci_status status ;
2011-06-30 19:14:33 -07:00
sci_remote_device_construct ( iport , idev ) ;
2011-04-22 19:18:03 -07:00
2011-06-30 19:14:33 -07:00
status = sci_controller_allocate_remote_node_context ( iport - > owning_controller ,
2011-06-30 16:31:37 -07:00
idev ,
& idev - > rnc . remote_node_index ) ;
2011-04-21 18:44:45 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-04-22 19:18:03 -07:00
2012-02-01 00:23:10 -08:00
/* For SAS-2 the physical link rate is actually a logical link
2011-04-21 18:44:45 -07:00
* rate that incorporates multiplexing . The SCU doesn ' t
* incorporate multiplexing and for the purposes of the
* connection the logical link rate is that same as the
* physical . Furthermore , the SAS - 2 and SAS - 1.1 fields overlay
2012-02-01 00:23:10 -08:00
* one another , so this code works for both situations .
*/
2011-06-30 19:14:33 -07:00
idev - > connection_rate = min_t ( u16 , sci_port_get_max_allowed_speed ( iport ) ,
2011-04-25 14:29:29 -07:00
dev - > linkrate ) ;
2011-04-22 19:18:03 -07:00
2011-04-21 18:44:45 -07:00
/* / @todo Should I assign the port width by reading all of the phys on the port? */
2011-06-30 16:31:37 -07:00
idev - > device_port_width = 1 ;
2011-04-22 19:18:03 -07:00
2011-04-27 16:32:45 -07:00
return SCI_SUCCESS ;
2011-04-22 19:18:03 -07:00
}
2012-03-08 22:41:51 -08:00
enum sci_status sci_remote_device_resume (
struct isci_remote_device * idev ,
scics_sds_remote_node_context_callback cb_fn ,
void * cb_p )
{
enum sci_status status ;
status = sci_remote_node_context_resume ( & idev - > rnc , cb_fn , cb_p ) ;
if ( status ! = SCI_SUCCESS )
dev_dbg ( scirdev_to_dev ( idev ) , " %s: failed to resume: %d \n " ,
__func__ , status ) ;
return status ;
}
enum sci_status isci_remote_device_resume (
struct isci_host * ihost ,
struct isci_remote_device * idev ,
scics_sds_remote_node_context_callback cb_fn ,
void * cb_p )
{
unsigned long flags ;
enum sci_status status ;
spin_lock_irqsave ( & ihost - > scic_lock , flags ) ;
status = sci_remote_device_resume ( idev , cb_fn , cb_p ) ;
spin_unlock_irqrestore ( & ihost - > scic_lock , flags ) ;
return status ;
}
2011-04-22 19:18:03 -07:00
/**
2011-06-30 19:14:33 -07:00
* sci_remote_device_start ( ) - This method will start the supplied remote
2011-04-22 19:18:03 -07:00
* device . This method enables normal IO requests to flow through to the
* remote device .
* @ remote_device : This parameter specifies the device to be started .
* @ timeout : This parameter specifies the number of milliseconds in which the
* start operation should complete .
*
* An indication of whether the device was successfully started . SCI_SUCCESS
* This value is returned if the device was successfully started .
* SCI_FAILURE_INVALID_PHY This value is returned if the user attempts to start
* the device when there have been no phys added to it .
*/
2011-06-30 19:14:33 -07:00
static enum sci_status sci_remote_device_start ( struct isci_remote_device * idev ,
2012-03-08 22:41:50 -08:00
u32 timeout )
2011-04-22 19:18:03 -07:00
{
2011-06-30 16:31:37 -07:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-06-30 19:14:33 -07:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-01 14:05:57 -07:00
enum sci_status status ;
2011-06-02 00:10:43 +00:00
if ( state ! = SCI_DEV_STOPPED ) {
2012-02-10 01:18:44 -08:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %s \n " ,
__func__ , dev_state_name ( state ) ) ;
2011-05-01 14:05:57 -07:00
return SCI_FAILURE_INVALID_STATE ;
}
2012-03-08 22:41:51 -08:00
status = sci_remote_device_resume ( idev , remote_device_resume_done ,
idev ) ;
2011-05-01 14:05:57 -07:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-06-02 00:10:43 +00:00
sci_change_state ( sm , SCI_DEV_STARTING ) ;
2011-05-01 14:05:57 -07:00
return SCI_SUCCESS ;
2011-04-22 19:18:03 -07:00
}
2011-07-02 22:56:22 -07:00
2011-04-25 14:29:29 -07:00
static enum sci_status isci_remote_device_construct ( struct isci_port * iport ,
struct isci_remote_device * idev )
2011-07-02 22:56:22 -07:00
{
2011-04-25 14:29:29 -07:00
struct isci_host * ihost = iport - > isci_host ;
struct domain_device * dev = idev - > domain_dev ;
enum sci_status status ;
2011-07-02 22:56:22 -07:00
2011-04-25 14:29:29 -07:00
if ( dev - > parent & & dev_is_expander ( dev - > parent ) )
2011-06-30 19:14:33 -07:00
status = sci_remote_device_ea_construct ( iport , idev ) ;
2011-04-25 14:29:29 -07:00
else
2011-06-30 19:14:33 -07:00
status = sci_remote_device_da_construct ( iport , idev ) ;
2011-07-02 22:56:22 -07:00
if ( status ! = SCI_SUCCESS ) {
2011-04-25 14:29:29 -07:00
dev_dbg ( & ihost - > pdev - > dev , " %s: construct failed: %d \n " ,
__func__ , status ) ;
2011-07-02 22:56:22 -07:00
return status ;
}
/* start the device. */
2011-06-30 19:14:33 -07:00
status = sci_remote_device_start ( idev , ISCI_REMOTE_DEVICE_START_TIMEOUT ) ;
2011-07-02 22:56:22 -07:00
2011-04-25 14:29:29 -07:00
if ( status ! = SCI_SUCCESS )
dev_warn ( & ihost - > pdev - > dev , " remote device start failed: %d \n " ,
status ) ;
2011-07-02 22:56:22 -07:00
return status ;
}
/**
* This function builds the isci_remote_device when a libsas dev_found message
* is received .
* @ isci_host : This parameter specifies the isci host object .
* @ port : This parameter specifies the isci_port conected to this device .
*
* pointer to new isci_remote_device .
*/
static struct isci_remote_device *
2011-03-03 17:59:32 -08:00
isci_remote_device_alloc ( struct isci_host * ihost , struct isci_port * iport )
2011-07-02 22:56:22 -07:00
{
2011-03-03 17:59:32 -08:00
struct isci_remote_device * idev ;
int i ;
2011-07-02 22:56:22 -07:00
2011-03-03 17:59:32 -08:00
for ( i = 0 ; i < SCI_MAX_REMOTE_DEVICES ; i + + ) {
2011-04-21 18:14:45 -07:00
idev = & ihost - > devices [ i ] ;
2011-03-03 17:59:32 -08:00
if ( ! test_and_set_bit ( IDEV_ALLOCATED , & idev - > flags ) )
break ;
}
2011-07-02 22:56:22 -07:00
2011-03-03 17:59:32 -08:00
if ( i > = SCI_MAX_REMOTE_DEVICES ) {
dev_warn ( & ihost - > pdev - > dev , " %s: failed \n " , __func__ ) ;
2011-07-02 22:56:22 -07:00
return NULL ;
}
2011-04-12 17:28:43 -07:00
if ( WARN_ONCE ( ! list_empty ( & idev - > node ) , " found non-idle remote device \n " ) )
return NULL ;
2011-03-03 17:59:32 -08:00
return idev ;
2011-07-02 22:56:22 -07:00
}
2011-06-13 17:39:44 -07:00
void isci_remote_device_release ( struct kref * kref )
{
struct isci_remote_device * idev = container_of ( kref , typeof ( * idev ) , kref ) ;
struct isci_host * ihost = idev - > isci_port - > isci_host ;
idev - > domain_dev = NULL ;
idev - > isci_port = NULL ;
clear_bit ( IDEV_START_PENDING , & idev - > flags ) ;
clear_bit ( IDEV_STOP_PENDING , & idev - > flags ) ;
2011-06-16 11:26:12 -07:00
clear_bit ( IDEV_IO_READY , & idev - > flags ) ;
2011-06-13 17:39:44 -07:00
clear_bit ( IDEV_GONE , & idev - > flags ) ;
smp_mb__before_clear_bit ( ) ;
clear_bit ( IDEV_ALLOCATED , & idev - > flags ) ;
wake_up ( & ihost - > eventq ) ;
}
2011-07-02 22:56:22 -07:00
/**
* isci_remote_device_stop ( ) - This function is called internally to stop the
* remote device .
* @ isci_host : This parameter specifies the isci host object .
* @ isci_device : This parameter specifies the remote device .
*
2011-06-30 17:38:32 -07:00
* The status of the ihost request to stop .
2011-07-02 22:56:22 -07:00
*/
2011-03-04 12:10:29 -08:00
enum sci_status isci_remote_device_stop ( struct isci_host * ihost , struct isci_remote_device * idev )
2011-07-02 22:56:22 -07:00
{
enum sci_status status ;
unsigned long flags ;
2011-03-04 12:10:29 -08:00
dev_dbg ( & ihost - > pdev - > dev ,
" %s: isci_device = %p \n " , __func__ , idev ) ;
2011-07-02 22:56:22 -07:00
2011-06-13 17:39:44 -07:00
spin_lock_irqsave ( & ihost - > scic_lock , flags ) ;
idev - > domain_dev - > lldd_dev = NULL ; /* disable new lookups */
set_bit ( IDEV_GONE , & idev - > flags ) ;
2011-03-08 20:32:16 -07:00
2011-03-04 12:10:29 -08:00
set_bit ( IDEV_STOP_PENDING , & idev - > flags ) ;
2011-06-30 19:14:33 -07:00
status = sci_remote_device_stop ( idev , 50 ) ;
2011-03-04 12:10:29 -08:00
spin_unlock_irqrestore ( & ihost - > scic_lock , flags ) ;
2011-07-02 22:56:22 -07:00
/* Wait for the stop complete callback. */
2011-06-13 17:39:44 -07:00
if ( WARN_ONCE ( status ! = SCI_SUCCESS , " failed to stop device \n " ) )
/* nothing to wait for */ ;
else
2011-03-04 12:10:29 -08:00
wait_for_device_stop ( ihost , idev ) ;
2011-07-02 22:56:22 -07:00
2012-03-08 22:41:52 -08:00
dev_dbg ( & ihost - > pdev - > dev ,
" %s: isci_device = %p, waiting done. \n " , __func__ , idev ) ;
2011-07-02 22:56:22 -07:00
return status ;
}
/**
* isci_remote_device_gone ( ) - This function is called by libsas when a domain
* device is removed .
* @ domain_device : This parameter specifies the libsas domain device .
*
*/
2011-03-04 12:10:29 -08:00
void isci_remote_device_gone ( struct domain_device * dev )
2011-07-02 22:56:22 -07:00
{
2011-03-31 13:10:44 -07:00
struct isci_host * ihost = dev_to_ihost ( dev ) ;
2011-03-04 12:10:29 -08:00
struct isci_remote_device * idev = dev - > lldd_dev ;
2011-07-02 22:56:22 -07:00
2011-03-04 12:10:29 -08:00
dev_dbg ( & ihost - > pdev - > dev ,
2011-07-02 22:56:22 -07:00
" %s: domain_device = %p, isci_device = %p, isci_port = %p \n " ,
2011-03-04 12:10:29 -08:00
__func__ , dev , idev , idev - > isci_port ) ;
2011-07-02 22:56:22 -07:00
2011-03-04 12:10:29 -08:00
isci_remote_device_stop ( ihost , idev ) ;
2011-07-02 22:56:22 -07:00
}
/**
* isci_remote_device_found ( ) - This function is called by libsas when a remote
* device is discovered . A remote device object is created and started . the
* function then sleeps until the sci core device started message is
* received .
* @ domain_device : This parameter specifies the libsas domain device .
*
* status , zero indicates success .
*/
2012-01-03 23:26:08 -08:00
int isci_remote_device_found ( struct domain_device * dev )
2011-07-02 22:56:22 -07:00
{
2012-01-03 23:26:08 -08:00
struct isci_host * isci_host = dev_to_ihost ( dev ) ;
struct isci_port * isci_port = dev - > port - > lldd_port ;
2011-07-02 22:56:22 -07:00
struct isci_remote_device * isci_device ;
enum sci_status status ;
dev_dbg ( & isci_host - > pdev - > dev ,
2012-01-03 23:26:08 -08:00
" %s: domain_device = %p \n " , __func__ , dev ) ;
2011-07-02 22:56:22 -07:00
2012-01-03 23:26:08 -08:00
if ( ! isci_port )
return - ENODEV ;
2011-07-02 22:56:22 -07:00
isci_device = isci_remote_device_alloc ( isci_host , isci_port ) ;
2011-03-03 17:59:32 -08:00
if ( ! isci_device )
return - ENODEV ;
2011-07-02 22:56:22 -07:00
2011-06-13 17:39:44 -07:00
kref_init ( & isci_device - > kref ) ;
2011-07-02 22:56:22 -07:00
INIT_LIST_HEAD ( & isci_device - > node ) ;
2011-06-13 17:39:44 -07:00
spin_lock_irq ( & isci_host - > scic_lock ) ;
2012-01-03 23:26:08 -08:00
isci_device - > domain_dev = dev ;
2011-07-02 22:56:22 -07:00
isci_device - > isci_port = isci_port ;
list_add_tail ( & isci_device - > node , & isci_port - > remote_dev_list ) ;
2011-03-04 12:10:29 -08:00
set_bit ( IDEV_START_PENDING , & isci_device - > flags ) ;
2011-07-02 22:56:22 -07:00
status = isci_remote_device_construct ( isci_port , isci_device ) ;
dev_dbg ( & isci_host - > pdev - > dev ,
" %s: isci_device = %p \n " ,
__func__ , isci_device ) ;
2011-06-13 17:39:44 -07:00
if ( status = = SCI_SUCCESS ) {
/* device came up, advertise it to the world */
2012-01-03 23:26:08 -08:00
dev - > lldd_dev = isci_device ;
2011-06-13 17:39:44 -07:00
} else
isci_put_device ( isci_device ) ;
spin_unlock_irq ( & isci_host - > scic_lock ) ;
2011-07-02 22:56:22 -07:00
2011-03-04 12:10:29 -08:00
/* wait for the device ready callback. */
wait_for_device_start ( isci_host , isci_device ) ;
2011-06-13 17:39:44 -07:00
return status = = SCI_SUCCESS ? 0 : - ENODEV ;
2011-07-02 22:56:22 -07:00
}
2012-03-08 22:41:50 -08:00
2012-03-08 22:41:54 -08:00
enum sci_status isci_remote_device_suspend_terminate (
2012-03-08 22:41:51 -08:00
struct isci_host * ihost ,
2012-03-08 22:41:54 -08:00
struct isci_remote_device * idev ,
struct isci_request * ireq )
2012-03-08 22:41:50 -08:00
{
unsigned long flags ;
enum sci_status status ;
2012-03-08 22:41:54 -08:00
/* Put the device into suspension. */
2012-03-08 22:41:51 -08:00
spin_lock_irqsave ( & ihost - > scic_lock , flags ) ;
2012-03-08 22:41:58 -08:00
sci_remote_device_suspend ( idev , SCI_SW_SUSPEND_LINKHANG_DETECT ) ;
2012-03-08 22:41:51 -08:00
spin_unlock_irqrestore ( & ihost - > scic_lock , flags ) ;
2012-03-08 22:41:54 -08:00
/* Terminate and wait for the completions. */
status = isci_remote_device_terminate_requests ( ihost , idev , ireq ) ;
if ( status ! = SCI_SUCCESS )
2012-03-08 22:41:50 -08:00
dev_dbg ( & ihost - > pdev - > dev ,
2012-03-08 22:41:54 -08:00
" %s: isci_remote_device_terminate_requests(%p) "
" returned %d! \n " ,
2012-03-08 22:41:50 -08:00
__func__ , idev , status ) ;
2012-03-08 22:41:54 -08:00
/* NOTE: RNC resumption is left to the caller! */
2012-03-08 22:41:50 -08:00
return status ;
}
int isci_remote_device_is_safe_to_abort (
struct isci_remote_device * idev )
{
return sci_remote_node_context_is_safe_to_abort ( & idev - > rnc ) ;
}
enum sci_status sci_remote_device_abort_requests_pending_abort (
struct isci_remote_device * idev )
{
2012-03-08 22:41:54 -08:00
return sci_remote_device_terminate_reqs_checkabort ( idev , 1 ) ;
2012-03-08 22:41:50 -08:00
}
2012-03-08 22:41:51 -08:00
enum sci_status isci_remote_device_reset_complete (
struct isci_host * ihost ,
struct isci_remote_device * idev )
{
unsigned long flags ;
enum sci_status status ;
spin_lock_irqsave ( & ihost - > scic_lock , flags ) ;
status = sci_remote_device_reset_complete ( idev ) ;
spin_unlock_irqrestore ( & ihost - > scic_lock , flags ) ;
return status ;
}
2012-03-08 22:42:00 -08:00
void isci_dev_set_hang_detection_timeout (
struct isci_remote_device * idev ,
u32 timeout )
{
if ( dev_is_sata ( idev - > domain_dev ) ) {
if ( timeout ) {
if ( test_and_set_bit ( IDEV_RNC_LLHANG_ENABLED ,
& idev - > flags ) )
return ; /* Already enabled. */
} else if ( ! test_and_clear_bit ( IDEV_RNC_LLHANG_ENABLED ,
& idev - > flags ) )
return ; /* Not enabled. */
sci_port_set_hang_detection_timeout ( idev - > owning_port ,
timeout ) ;
}
}