2011-07-03 09:56:22 +04: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-05 05:45:05 +04:00
# include <scsi/sas.h>
2011-04-23 06:18:03 +04: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"
2011-04-28 03:32:45 +04:00
/**
2011-07-01 04:38:32 +04:00
* isci_remote_device_not_ready ( ) - This function is called by the ihost when
2011-04-28 03:32:45 +04:00
* 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
*
2011-07-01 06:14:33 +04:00
* sci_lock is held on entrance to this function .
2011-04-28 03:32:45 +04:00
*/
static void isci_remote_device_not_ready ( struct isci_host * ihost ,
struct isci_remote_device * idev , u32 reason )
{
2011-07-03 23:14:45 +04:00
struct isci_request * ireq ;
2011-06-24 04:09:02 +04:00
2011-04-28 03:32:45 +04:00
dev_dbg ( & ihost - > pdev - > dev ,
" %s: isci_device = %p \n " , __func__ , idev ) ;
2011-06-24 04:09:02 +04:00
switch ( reason ) {
case SCIC_REMOTE_DEVICE_NOT_READY_STOP_REQUESTED :
2011-06-14 04:39:44 +04:00
set_bit ( IDEV_GONE , & idev - > flags ) ;
2011-06-24 04:09:02 +04:00
break ;
case SCIC_REMOTE_DEVICE_NOT_READY_SATA_SDB_ERROR_FIS_RECEIVED :
set_bit ( IDEV_IO_NCQERROR , & idev - > flags ) ;
/* Kill all outstanding requests for the device. */
list_for_each_entry ( ireq , & idev - > reqs_in_process , dev_node ) {
dev_dbg ( & ihost - > pdev - > dev ,
" %s: isci_device = %p request = %p \n " ,
__func__ , idev , ireq ) ;
2011-07-01 06:14:33 +04:00
sci_controller_terminate_request ( ihost ,
2011-07-01 03:31:37 +04:00
idev ,
2011-06-28 01:57:03 +04:00
ireq ) ;
2011-06-24 04:09:02 +04:00
}
/* Fall through into the default case... */
default :
2011-06-16 22:26:12 +04:00
clear_bit ( IDEV_IO_READY , & idev - > flags ) ;
2011-06-24 04:09:02 +04:00
break ;
}
2011-04-28 03:32:45 +04:00
}
/**
2011-07-01 04:38:32 +04:00
* isci_remote_device_ready ( ) - This function is called by the ihost when the
2011-04-28 03:32:45 +04: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-24 04:09:02 +04:00
clear_bit ( IDEV_IO_NCQERROR , & idev - > flags ) ;
2011-06-16 22:26:12 +04:00
set_bit ( IDEV_IO_READY , & idev - > flags ) ;
2011-04-28 03:32:45 +04:00
if ( test_and_clear_bit ( IDEV_START_PENDING , & idev - > flags ) )
wake_up ( & ihost - > eventq ) ;
}
2011-05-02 01:19:25 +04: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-07-01 03:31:37 +04:00
struct isci_remote_device * idev = _dev ;
2011-04-28 03:32:45 +04:00
2011-07-01 03:31:37 +04:00
BUG_ON ( idev - > started_request_count ! = 0 ) ;
sci_change_state ( & idev - > sm , SCI_DEV_STOPPED ) ;
2011-05-02 01:19:25 +04:00
}
2011-04-28 03:32:45 +04:00
2011-07-01 06:14:33 +04:00
static enum sci_status sci_remote_device_terminate_requests ( struct isci_remote_device * idev )
2011-05-02 01:19:25 +04:00
{
2011-07-01 04:38:32 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-05-02 01:19:25 +04:00
enum sci_status status = SCI_SUCCESS ;
2011-06-29 20:45:48 +04:00
u32 i ;
2011-05-02 01:19:25 +04:00
2011-06-29 20:45:48 +04:00
for ( i = 0 ; i < SCI_MAX_IO_REQUESTS ; i + + ) {
2011-06-18 01:18:39 +04:00
struct isci_request * ireq = ihost - > reqs [ i ] ;
2011-05-02 01:19:25 +04:00
enum sci_status s ;
2011-06-18 01:18:39 +04:00
if ( ! test_bit ( IREQ_ACTIVE , & ireq - > flags ) | |
2011-07-01 03:31:37 +04:00
ireq - > target_device ! = idev )
2011-05-02 01:19:25 +04:00
continue ;
2011-06-18 01:18:39 +04:00
2011-07-01 06:14:33 +04:00
s = sci_controller_terminate_request ( ihost , idev , ireq ) ;
2011-05-02 01:19:25 +04:00
if ( s ! = SCI_SUCCESS )
status = s ;
}
return status ;
}
2011-07-01 06:14:33 +04:00
enum sci_status sci_remote_device_stop ( struct isci_remote_device * idev ,
2011-05-02 01:19:25 +04:00
u32 timeout )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-02 01:19:25 +04:00
switch ( state ) {
2011-06-02 04:10:43 +04:00
case SCI_DEV_INITIAL :
case SCI_DEV_FAILED :
case SCI_DEV_FINAL :
2011-05-02 01:19:25 +04:00
default :
2011-07-01 03:31:37 +04:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %d \n " ,
2011-05-02 01:19:25 +04:00
__func__ , state ) ;
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 04:10:43 +04:00
case SCI_DEV_STOPPED :
2011-05-02 01:19:25 +04:00
return SCI_SUCCESS ;
2011-06-02 04:10:43 +04:00
case SCI_DEV_STARTING :
2011-05-02 01:19:25 +04:00
/* device not started so there had better be no requests */
2011-07-01 03:31:37 +04:00
BUG_ON ( idev - > started_request_count ! = 0 ) ;
2011-07-01 06:14:33 +04:00
sci_remote_node_context_destruct ( & idev - > rnc ,
2011-07-01 03:31:37 +04:00
rnc_destruct_done , idev ) ;
2011-05-02 01:19:25 +04:00
/* Transition to the stopping state and wait for the
* remote node to complete being posted and invalidated .
*/
2011-06-02 04:10:43 +04:00
sci_change_state ( sm , SCI_DEV_STOPPING ) ;
2011-05-02 01:19:25 +04:00
return SCI_SUCCESS ;
2011-06-02 04:10:43 +04: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 ) ;
2011-07-01 03:31:37 +04:00
if ( idev - > started_request_count = = 0 ) {
2011-07-01 06:14:33 +04:00
sci_remote_node_context_destruct ( & idev - > rnc ,
2011-07-01 03:31:37 +04:00
rnc_destruct_done , idev ) ;
2011-05-02 01:19:25 +04:00
return SCI_SUCCESS ;
} else
2011-07-01 06:14:33 +04:00
return sci_remote_device_terminate_requests ( idev ) ;
2011-05-02 01:19:25 +04:00
break ;
2011-06-02 04:10:43 +04:00
case SCI_DEV_STOPPING :
2011-05-02 01:19:25 +04: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-07-01 06:14:33 +04:00
return sci_remote_device_terminate_requests ( idev ) ;
2011-06-02 04:10:43 +04:00
case SCI_DEV_RESETTING :
sci_change_state ( sm , SCI_DEV_STOPPING ) ;
2011-05-02 01:19:25 +04:00
return SCI_SUCCESS ;
}
2011-04-23 06:18:03 +04:00
}
2011-07-01 06:14:33 +04:00
enum sci_status sci_remote_device_reset ( struct isci_remote_device * idev )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-02 01:48:54 +04:00
switch ( state ) {
2011-06-02 04:10:43 +04: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-02 01:48:54 +04:00
default :
2011-07-01 03:31:37 +04:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %d \n " ,
2011-05-02 01:48:54 +04:00
__func__ , state ) ;
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 04:10:43 +04: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-02 01:48:54 +04:00
return SCI_SUCCESS ;
}
2011-04-23 06:18:03 +04:00
}
2011-07-01 06:14:33 +04:00
enum sci_status sci_remote_device_reset_complete ( struct isci_remote_device * idev )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-02 01:53:00 +04:00
2011-06-02 04:10:43 +04:00
if ( state ! = SCI_DEV_RESETTING ) {
2011-07-01 03:31:37 +04:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %d \n " ,
2011-05-02 01:53:00 +04:00
__func__ , state ) ;
return SCI_FAILURE_INVALID_STATE ;
}
2011-04-23 06:18:03 +04:00
2011-06-02 04:10:43 +04:00
sci_change_state ( sm , SCI_DEV_READY ) ;
2011-05-02 01:53:00 +04:00
return SCI_SUCCESS ;
}
2011-04-23 06:18:03 +04:00
2011-07-01 06:14:33 +04:00
enum sci_status sci_remote_device_suspend ( struct isci_remote_device * idev ,
2011-05-02 03:15:47 +04:00
u32 suspend_type )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-02 03:15:47 +04:00
2011-06-02 04:10:43 +04:00
if ( state ! = SCI_STP_DEV_CMD ) {
2011-07-01 03:31:37 +04:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %d \n " ,
2011-05-02 03:15:47 +04:00
__func__ , state ) ;
return SCI_FAILURE_INVALID_STATE ;
}
2011-07-01 06:14:33 +04:00
return sci_remote_node_context_suspend ( & idev - > rnc ,
2011-05-02 03:15:47 +04:00
suspend_type , NULL , NULL ) ;
2011-04-23 06:18:03 +04:00
}
2011-07-01 06:14:33 +04:00
enum sci_status sci_remote_device_frame_handler ( struct isci_remote_device * idev ,
2011-05-02 03:51:11 +04:00
u32 frame_index )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-07-01 04:38:32 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-05-02 03:51:11 +04:00
enum sci_status status ;
switch ( state ) {
2011-06-02 04:10:43 +04: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-02 03:51:11 +04:00
default :
2011-07-01 03:31:37 +04:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %d \n " ,
2011-05-02 03:51:11 +04:00
__func__ , state ) ;
/* Return the frame back to the controller */
2011-07-01 06:14:33 +04:00
sci_controller_release_frame ( ihost , frame_index ) ;
2011-05-02 03:51:11 +04:00
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 04:10:43 +04: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-28 01:57:03 +04:00
struct isci_request * ireq ;
2011-05-05 05:45:05 +04:00
struct ssp_frame_hdr hdr ;
void * frame_header ;
ssize_t word_cnt ;
2011-05-02 03:51:11 +04:00
2011-07-01 06:14:33 +04:00
status = sci_unsolicited_frame_control_get_header ( & ihost - > uf_control ,
2011-05-02 03:51:11 +04:00
frame_index ,
2011-05-05 05:45:05 +04:00
& frame_header ) ;
2011-05-02 03:51:11 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-05-05 05:45:05 +04:00
word_cnt = sizeof ( hdr ) / sizeof ( u32 ) ;
sci_swab32_cpy ( & hdr , frame_header , word_cnt ) ;
2011-07-01 06:14:33 +04:00
ireq = sci_request_by_tag ( ihost , be16_to_cpu ( hdr . tag ) ) ;
2011-07-01 03:31:37 +04:00
if ( ireq & & ireq - > target_device = = idev ) {
2011-05-02 03:51:11 +04:00
/* The IO request is now in charge of releasing the frame */
2011-07-01 06:14:33 +04:00
status = sci_io_request_frame_handler ( ireq , frame_index ) ;
2011-05-02 03:51:11 +04:00
} else {
/* We could not map this tag to a valid IO
* request Just toss the frame and continue
*/
2011-07-01 06:14:33 +04:00
sci_controller_release_frame ( ihost , frame_index ) ;
2011-05-02 03:51:11 +04:00
}
break ;
}
2011-06-02 04:10:43 +04:00
case SCI_STP_DEV_NCQ : {
2011-05-05 02:02:03 +04:00
struct dev_to_host_fis * hdr ;
2011-05-02 03:51:11 +04:00
2011-07-01 06:14:33 +04:00
status = sci_unsolicited_frame_control_get_header ( & ihost - > uf_control ,
2011-05-02 03:51:11 +04:00
frame_index ,
( void * * ) & hdr ) ;
if ( status ! = SCI_SUCCESS )
return status ;
2011-05-05 02:02:03 +04:00
if ( hdr - > fis_type = = FIS_SETDEVBITS & &
( hdr - > status & ATA_ERR ) ) {
2011-07-01 03:31:37 +04:00
idev - > not_ready_reason = SCIC_REMOTE_DEVICE_NOT_READY_SATA_SDB_ERROR_FIS_RECEIVED ;
2011-05-02 03:51:11 +04:00
/* TODO Check sactive and complete associated IO if any. */
2011-06-02 04:10:43 +04:00
sci_change_state ( sm , SCI_STP_DEV_NCQ_ERROR ) ;
2011-05-05 02:02:03 +04:00
} else if ( hdr - > fis_type = = FIS_REGD2H & &
( hdr - > status & ATA_ERR ) ) {
2011-05-02 03:51:11 +04:00
/*
* Some devices return D2H FIS when an NCQ error is detected .
* Treat this like an SDB error FIS ready reason .
*/
2011-07-01 03:31:37 +04: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-02 03:51:11 +04:00
} else
status = SCI_FAILURE ;
2011-07-01 06:14:33 +04:00
sci_controller_release_frame ( ihost , frame_index ) ;
2011-05-02 03:51:11 +04:00
break ;
}
2011-06-02 04:10:43 +04:00
case SCI_STP_DEV_CMD :
case SCI_SMP_DEV_CMD :
2011-05-02 03:51:11 +04: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-07-01 06:14:33 +04:00
status = sci_io_request_frame_handler ( idev - > working_request , frame_index ) ;
2011-05-02 03:51:11 +04:00
break ;
}
return status ;
2011-04-23 06:18:03 +04:00
}
2011-07-01 03:31:37 +04:00
static bool is_remote_device_ready ( struct isci_remote_device * idev )
2011-04-23 06:18:03 +04:00
{
2011-05-02 03:26:09 +04:00
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-02 03:26:09 +04:00
switch ( state ) {
2011-06-02 04:10:43 +04: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-02 03:26:09 +04:00
return true ;
default :
return false ;
}
}
2011-07-01 06:14:33 +04:00
enum sci_status sci_remote_device_event_handler ( struct isci_remote_device * idev ,
2011-05-02 03:26:09 +04:00
u32 event_code )
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-02 03:26:09 +04:00
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-07-01 06:14:33 +04:00
status = sci_remote_node_context_event_handler ( & idev - > rnc , event_code ) ;
2011-05-02 03:26:09 +04: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 */
2011-07-01 06:14:33 +04:00
sci_remote_node_context_suspend ( & idev - > rnc ,
2011-05-02 03:26:09 +04:00
SCI_SOFTWARE_SUSPENSION ,
NULL , NULL ) ;
2011-07-01 03:31:37 +04:00
dev_dbg ( scirdev_to_dev ( idev ) ,
2011-05-02 03:26:09 +04:00
" %s: device: %p event code: %x: %s \n " ,
2011-07-01 03:31:37 +04:00
__func__ , idev , event_code ,
is_remote_device_ready ( idev )
2011-05-02 03:26:09 +04:00
? " I_T_Nexus_Timeout event "
: " I_T_Nexus_Timeout event in wrong state " ) ;
break ;
}
/* Else, fall through and treat as unhandled... */
default :
2011-07-01 03:31:37 +04:00
dev_dbg ( scirdev_to_dev ( idev ) ,
2011-05-02 03:26:09 +04:00
" %s: device: %p event code: %x: %s \n " ,
2011-07-01 03:31:37 +04:00
__func__ , idev , event_code ,
is_remote_device_ready ( idev )
2011-05-02 03:26:09 +04:00
? " unexpected event "
: " unexpected event in wrong state " ) ;
status = SCI_FAILURE_INVALID_STATE ;
break ;
}
if ( status ! = SCI_SUCCESS )
return status ;
2011-06-02 04:10:43 +04:00
if ( state = = SCI_STP_DEV_IDLE ) {
2011-05-02 03:26:09 +04:00
/* We pick up suspension events to handle specifically to this
* state . We resume the RNC right away .
*/
if ( scu_get_event_type ( event_code ) = = SCU_EVENT_TYPE_RNC_SUSPEND_TX | |
scu_get_event_type ( event_code ) = = SCU_EVENT_TYPE_RNC_SUSPEND_TX_RX )
2011-07-01 06:14:33 +04:00
status = sci_remote_node_context_resume ( & idev - > rnc , NULL , NULL ) ;
2011-05-02 03:26:09 +04:00
}
return status ;
2011-04-23 06:18:03 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_remote_device_start_request ( struct isci_remote_device * idev ,
2011-06-28 01:57:03 +04:00
struct isci_request * ireq ,
2011-05-02 01:57:11 +04:00
enum sci_status status )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_port * iport = idev - > owning_port ;
2011-05-02 01:57:11 +04:00
/* cleanup requests that failed after starting on the port */
if ( status ! = SCI_SUCCESS )
2011-07-01 06:14:33 +04:00
sci_port_complete_io ( iport , idev , ireq ) ;
2011-06-14 04:39:44 +04:00
else {
2011-07-01 03:31:37 +04:00
kref_get ( & idev - > kref ) ;
2011-07-01 13:25:15 +04:00
idev - > started_request_count + + ;
2011-06-14 04:39:44 +04:00
}
2011-05-02 01:57:11 +04:00
}
2011-07-01 06:14:33 +04:00
enum sci_status sci_remote_device_start_io ( struct isci_host * ihost ,
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev ,
2011-06-28 01:57:03 +04:00
struct isci_request * ireq )
2011-05-02 01:57:11 +04:00
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-07-01 03:31:37 +04:00
struct isci_port * iport = idev - > owning_port ;
2011-05-02 01:57:11 +04:00
enum sci_status status ;
switch ( state ) {
2011-06-02 04:10:43 +04: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-02 01:57:11 +04:00
default :
2011-07-01 03:31:37 +04:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %d \n " ,
2011-05-02 01:57:11 +04:00
__func__ , state ) ;
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 04:10:43 +04:00
case SCI_DEV_READY :
2011-05-02 01:57:11 +04: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-07-01 06:14:33 +04:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-02 01:57:11 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-07-01 06:14:33 +04:00
status = sci_remote_node_context_start_io ( & idev - > rnc , ireq ) ;
2011-05-02 01:57:11 +04:00
if ( status ! = SCI_SUCCESS )
break ;
2011-07-01 06:14:33 +04:00
status = sci_request_start ( ireq ) ;
2011-05-02 01:57:11 +04:00
break ;
2011-06-02 04:10:43 +04:00
case SCI_STP_DEV_IDLE : {
2011-05-02 01:57:11 +04: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-07-01 06:14:33 +04:00
enum sci_remote_device_states new_state ;
2011-05-05 02:02:03 +04:00
struct sas_task * task = isci_request_access_task ( ireq ) ;
2011-05-02 01:57:11 +04:00
2011-07-01 06:14:33 +04:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-02 01:57:11 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-07-01 06:14:33 +04:00
status = sci_remote_node_context_start_io ( & idev - > rnc , ireq ) ;
2011-05-02 01:57:11 +04:00
if ( status ! = SCI_SUCCESS )
break ;
2011-07-01 06:14:33 +04:00
status = sci_request_start ( ireq ) ;
2011-05-02 01:57:11 +04:00
if ( status ! = SCI_SUCCESS )
break ;
2011-05-05 02:02:03 +04:00
if ( task - > ata_task . use_ncq )
2011-06-02 04:10:43 +04:00
new_state = SCI_STP_DEV_NCQ ;
2011-05-02 01:57:11 +04:00
else {
2011-07-01 03:31:37 +04:00
idev - > working_request = ireq ;
2011-06-02 04:10:43 +04:00
new_state = SCI_STP_DEV_CMD ;
2011-05-02 01:57:11 +04:00
}
2011-06-02 04:10:43 +04:00
sci_change_state ( sm , new_state ) ;
2011-05-02 01:57:11 +04:00
break ;
}
2011-06-02 04:10:43 +04:00
case SCI_STP_DEV_NCQ : {
2011-05-05 02:02:03 +04:00
struct sas_task * task = isci_request_access_task ( ireq ) ;
if ( task - > ata_task . use_ncq ) {
2011-07-01 06:14:33 +04:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-02 01:57:11 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-07-01 06:14:33 +04:00
status = sci_remote_node_context_start_io ( & idev - > rnc , ireq ) ;
2011-05-02 01:57:11 +04:00
if ( status ! = SCI_SUCCESS )
break ;
2011-07-01 06:14:33 +04:00
status = sci_request_start ( ireq ) ;
2011-05-02 01:57:11 +04:00
} else
return SCI_FAILURE_INVALID_STATE ;
break ;
2011-05-05 02:02:03 +04:00
}
2011-06-02 04:10:43 +04:00
case SCI_STP_DEV_AWAIT_RESET :
2011-05-02 01:57:11 +04:00
return SCI_FAILURE_REMOTE_DEVICE_RESET_REQUIRED ;
2011-06-02 04:10:43 +04:00
case SCI_SMP_DEV_IDLE :
2011-07-01 06:14:33 +04:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-02 01:57:11 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-07-01 06:14:33 +04:00
status = sci_remote_node_context_start_io ( & idev - > rnc , ireq ) ;
2011-05-02 01:57:11 +04:00
if ( status ! = SCI_SUCCESS )
break ;
2011-07-01 06:14:33 +04:00
status = sci_request_start ( ireq ) ;
2011-05-02 01:57:11 +04:00
if ( status ! = SCI_SUCCESS )
break ;
2011-07-01 03:31:37 +04:00
idev - > working_request = ireq ;
sci_change_state ( & idev - > sm , SCI_SMP_DEV_CMD ) ;
2011-05-02 01:57:11 +04:00
break ;
2011-06-02 04:10:43 +04:00
case SCI_STP_DEV_CMD :
case SCI_SMP_DEV_CMD :
2011-05-02 01:57:11 +04:00
/* device is already handling a command it can not accept new commands
* until this one is complete .
*/
return SCI_FAILURE_INVALID_STATE ;
}
2011-07-01 06:14:33 +04:00
sci_remote_device_start_request ( idev , ireq , status ) ;
2011-05-02 01:57:11 +04:00
return status ;
2011-04-23 06:18:03 +04:00
}
2011-06-30 00:09:25 +04:00
static enum sci_status common_complete_io ( struct isci_port * iport ,
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev ,
2011-06-28 01:57:03 +04:00
struct isci_request * ireq )
2011-04-23 06:18:03 +04:00
{
2011-05-02 02:33:43 +04:00
enum sci_status status ;
2011-07-01 06:14:33 +04:00
status = sci_request_complete ( ireq ) ;
2011-05-02 02:33:43 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-07-01 06:14:33 +04:00
status = sci_port_complete_io ( iport , idev , ireq ) ;
2011-05-02 02:33:43 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-07-01 06:14:33 +04:00
sci_remote_device_decrement_request_count ( idev ) ;
2011-05-02 02:33:43 +04:00
return status ;
}
2011-07-01 06:14:33 +04:00
enum sci_status sci_remote_device_complete_io ( struct isci_host * ihost ,
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev ,
2011-06-28 01:57:03 +04:00
struct isci_request * ireq )
2011-05-02 02:33:43 +04:00
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-07-01 03:31:37 +04:00
struct isci_port * iport = idev - > owning_port ;
2011-05-02 02:33:43 +04:00
enum sci_status status ;
switch ( state ) {
2011-06-02 04:10:43 +04: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-02 02:33:43 +04:00
default :
2011-07-01 03:31:37 +04:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %d \n " ,
2011-05-02 02:33:43 +04:00
__func__ , state ) ;
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 04:10:43 +04:00
case SCI_DEV_READY :
case SCI_STP_DEV_AWAIT_RESET :
case SCI_DEV_RESETTING :
2011-07-01 03:31:37 +04:00
status = common_complete_io ( iport , idev , ireq ) ;
2011-05-02 02:33:43 +04:00
break ;
2011-06-02 04:10:43 +04:00
case SCI_STP_DEV_CMD :
case SCI_STP_DEV_NCQ :
case SCI_STP_DEV_NCQ_ERROR :
2011-07-01 03:31:37 +04:00
status = common_complete_io ( iport , idev , ireq ) ;
2011-05-02 02:33:43 +04:00
if ( status ! = SCI_SUCCESS )
break ;
2011-06-28 01:57:03 +04:00
if ( ireq - > sci_status = = SCI_FAILURE_REMOTE_DEVICE_RESET_REQUIRED ) {
2011-05-02 02:33:43 +04: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 04:10:43 +04:00
sci_change_state ( sm , SCI_STP_DEV_AWAIT_RESET ) ;
2011-07-01 13:25:15 +04:00
} else if ( idev - > started_request_count = = 0 )
2011-06-02 04:10:43 +04:00
sci_change_state ( sm , SCI_STP_DEV_IDLE ) ;
2011-05-02 02:33:43 +04:00
break ;
2011-06-02 04:10:43 +04:00
case SCI_SMP_DEV_CMD :
2011-07-01 03:31:37 +04:00
status = common_complete_io ( iport , idev , ireq ) ;
2011-05-02 02:33:43 +04:00
if ( status ! = SCI_SUCCESS )
break ;
2011-06-02 04:10:43 +04:00
sci_change_state ( sm , SCI_SMP_DEV_IDLE ) ;
2011-05-02 02:33:43 +04:00
break ;
2011-06-02 04:10:43 +04:00
case SCI_DEV_STOPPING :
2011-07-01 03:31:37 +04:00
status = common_complete_io ( iport , idev , ireq ) ;
2011-05-02 02:33:43 +04:00
if ( status ! = SCI_SUCCESS )
break ;
2011-07-01 13:25:15 +04:00
if ( idev - > started_request_count = = 0 )
2011-07-01 06:14:33 +04:00
sci_remote_node_context_destruct ( & idev - > rnc ,
2011-07-01 13:25:15 +04:00
rnc_destruct_done ,
idev ) ;
2011-05-02 02:33:43 +04:00
break ;
}
if ( status ! = SCI_SUCCESS )
2011-07-01 03:31:37 +04:00
dev_err ( scirdev_to_dev ( idev ) ,
2011-05-02 02:33:43 +04:00
" %s: Port:0x%p Device:0x%p Request:0x%p Status:0x%x "
2011-06-30 00:09:25 +04:00
" could not complete \n " , __func__ , iport ,
2011-07-01 03:31:37 +04:00
idev , ireq , status ) ;
2011-06-14 04:39:44 +04:00
else
2011-07-01 03:31:37 +04:00
isci_put_device ( idev ) ;
2011-05-02 02:33:43 +04:00
return status ;
2011-04-23 06:18:03 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_remote_device_continue_request ( void * dev )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = dev ;
2011-05-02 02:53:25 +04:00
/* we need to check if this request is still valid to continue. */
2011-07-01 03:31:37 +04:00
if ( idev - > working_request )
2011-07-01 06:14:33 +04:00
sci_controller_continue_io ( idev - > working_request ) ;
2011-05-02 02:53:25 +04:00
}
2011-07-01 06:14:33 +04:00
enum sci_status sci_remote_device_start_task ( struct isci_host * ihost ,
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev ,
2011-06-28 01:57:03 +04:00
struct isci_request * ireq )
2011-05-02 02:53:25 +04:00
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-07-01 03:31:37 +04:00
struct isci_port * iport = idev - > owning_port ;
2011-05-02 02:53:25 +04:00
enum sci_status status ;
switch ( state ) {
2011-06-02 04:10:43 +04: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-02 02:53:25 +04:00
default :
2011-07-01 03:31:37 +04:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %d \n " ,
2011-05-02 02:53:25 +04:00
__func__ , state ) ;
return SCI_FAILURE_INVALID_STATE ;
2011-06-02 04:10:43 +04: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-07-01 06:14:33 +04:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-02 02:53:25 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-07-01 06:14:33 +04:00
status = sci_remote_node_context_start_task ( & idev - > rnc , ireq ) ;
2011-05-02 02:53:25 +04:00
if ( status ! = SCI_SUCCESS )
goto out ;
2011-07-01 06:14:33 +04:00
status = sci_request_start ( ireq ) ;
2011-05-02 02:53:25 +04: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-07-01 03:31:37 +04:00
idev - > working_request = ireq ;
2011-06-02 04:10:43 +04:00
sci_change_state ( sm , SCI_STP_DEV_CMD ) ;
2011-05-02 02:53:25 +04: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 .
*/
2011-07-01 06:14:33 +04:00
sci_remote_node_context_suspend ( & idev - > rnc ,
2011-05-02 02:53:25 +04:00
SCI_SOFTWARE_SUSPENSION , NULL , NULL ) ;
2011-07-01 06:14:33 +04:00
sci_remote_node_context_resume ( & idev - > rnc ,
sci_remote_device_continue_request ,
2011-07-01 03:31:37 +04:00
idev ) ;
2011-05-02 02:53:25 +04:00
out :
2011-07-01 06:14:33 +04:00
sci_remote_device_start_request ( idev , ireq , status ) ;
2011-05-02 02:53:25 +04: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 04:10:43 +04:00
case SCI_DEV_READY :
2011-07-01 06:14:33 +04:00
status = sci_port_start_io ( iport , idev , ireq ) ;
2011-05-02 02:53:25 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-07-01 06:14:33 +04:00
status = sci_remote_node_context_start_task ( & idev - > rnc , ireq ) ;
2011-05-02 02:53:25 +04:00
if ( status ! = SCI_SUCCESS )
break ;
2011-07-01 06:14:33 +04:00
status = sci_request_start ( ireq ) ;
2011-05-02 02:53:25 +04:00
break ;
}
2011-07-01 06:14:33 +04:00
sci_remote_device_start_request ( idev , ireq , status ) ;
2011-05-02 02:53:25 +04:00
return status ;
2011-04-23 06:18:03 +04:00
}
2011-07-01 13:25:15 +04:00
void sci_remote_device_post_request ( struct isci_remote_device * idev , u32 request )
2011-04-23 06:18:03 +04:00
{
2011-07-01 13:25:15 +04:00
struct isci_port * iport = idev - > owning_port ;
2011-04-23 06:18:03 +04:00
u32 context ;
2011-07-01 13:25:15 +04: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-23 06:18:03 +04:00
2011-07-01 13:25:15 +04:00
sci_controller_post_request ( iport - > owning_controller , context ) ;
2011-04-23 06:18:03 +04:00
}
2011-04-28 03:32:45 +04:00
/* called once the remote node context has transisitioned to a
2011-04-23 06:18:03 +04:00
* ready state . This is the indication that the remote device object can also
2011-04-28 03:32:45 +04:00
* transition to ready .
2011-04-23 06:18:03 +04:00
*/
2011-05-02 01:05:57 +04:00
static void remote_device_resume_done ( void * _dev )
2011-04-28 03:32:45 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = _dev ;
2011-04-28 03:32:45 +04:00
2011-07-01 03:31:37 +04:00
if ( is_remote_device_ready ( idev ) )
2011-05-02 03:26:09 +04:00
return ;
2011-04-23 06:18:03 +04:00
2011-05-02 03:26:09 +04:00
/* go 'ready' if we are not already in a ready state */
2011-07-01 03:31:37 +04:00
sci_change_state ( & idev - > sm , SCI_DEV_READY ) ;
2011-04-23 06:18:03 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_stp_remote_device_ready_idle_substate_resume_complete_handler ( void * _dev )
2011-04-28 03:32:45 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = _dev ;
2011-07-01 04:38:32 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-28 03:32:45 +04:00
/* For NCQ operation we do not issue a isci_remote_device_not_ready().
* As a result , avoid sending the ready notification .
*/
2011-07-01 03:31:37 +04:00
if ( idev - > sm . previous_state_id ! = SCI_STP_DEV_NCQ )
2011-07-01 04:38:32 +04:00
isci_remote_device_ready ( ihost , idev ) ;
2011-04-28 03:32:45 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_remote_device_initial_state_enter ( struct sci_base_state_machine * sm )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-04-23 06:18:03 +04:00
2011-04-28 03:32:45 +04:00
/* Initial state is a transitional state to the stopped state */
2011-07-01 03:31:37 +04:00
sci_change_state ( & idev - > sm , SCI_DEV_STOPPED ) ;
2011-04-23 06:18:03 +04:00
}
2011-07-03 09:56:22 +04:00
2011-04-23 06:18:03 +04:00
/**
2011-07-01 06:14:33 +04:00
* sci_remote_device_destruct ( ) - free remote node context and destruct
2011-04-23 06:18:03 +04: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-07-01 06:14:33 +04:00
static enum sci_status sci_remote_device_destruct ( struct isci_remote_device * idev )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-07-01 04:38:32 +04:00
struct isci_host * ihost ;
2011-05-02 01:38:26 +04:00
2011-06-02 04:10:43 +04:00
if ( state ! = SCI_DEV_STOPPED ) {
2011-07-01 03:31:37 +04:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %d \n " ,
2011-05-02 01:38:26 +04:00
__func__ , state ) ;
return SCI_FAILURE_INVALID_STATE ;
}
2011-07-01 04:38:32 +04:00
ihost = idev - > owning_port - > owning_controller ;
2011-07-01 06:14:33 +04:00
sci_controller_free_remote_node_context ( ihost , idev ,
2011-07-01 03:31:37 +04:00
idev - > rnc . remote_node_index ) ;
idev - > rnc . remote_node_index = SCIC_SDS_REMOTE_NODE_CONTEXT_INVALID_INDEX ;
2011-06-02 04:10:43 +04:00
sci_change_state ( sm , SCI_DEV_FINAL ) ;
2011-05-02 01:38:26 +04:00
return SCI_SUCCESS ;
2011-04-23 06:18:03 +04:00
}
2011-07-03 09:56:22 +04:00
/**
* isci_remote_device_deconstruct ( ) - This function frees an isci_remote_device .
2011-03-04 04:59:32 +03:00
* @ ihost : This parameter specifies the isci host object .
* @ idev : This parameter specifies the remote device to be freed .
2011-07-03 09:56:22 +04:00
*
*/
2011-03-04 04:59:32 +03:00
static void isci_remote_device_deconstruct ( struct isci_host * ihost , struct isci_remote_device * idev )
2011-07-03 09:56:22 +04:00
{
2011-03-04 04:59:32 +03:00
dev_dbg ( & ihost - > pdev - > dev ,
" %s: isci_device = %p \n " , __func__ , idev ) ;
2011-07-03 09:56:22 +04: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 */
2011-06-14 04:39:44 +04:00
BUG_ON ( ! list_empty ( & idev - > reqs_in_process ) ) ;
2011-07-03 09:56:22 +04:00
2011-07-01 06:14:33 +04:00
sci_remote_device_destruct ( idev ) ;
2011-03-04 04:59:32 +03:00
list_del_init ( & idev - > node ) ;
2011-06-14 04:39:44 +04:00
isci_put_device ( idev ) ;
2011-07-03 09:56:22 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_remote_device_stopped_state_enter ( struct sci_base_state_machine * sm )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 04:38:32 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-23 06:18:03 +04:00
u32 prev_state ;
/* If we are entering from the stopping state let the SCI User know that
* the stop operation has completed .
*/
2011-07-01 03:31:37 +04:00
prev_state = idev - > sm . previous_state_id ;
2011-06-02 04:10:43 +04:00
if ( prev_state = = SCI_DEV_STOPPING )
2011-07-01 04:38:32 +04:00
isci_remote_device_deconstruct ( ihost , idev ) ;
2011-04-23 06:18:03 +04:00
2011-07-01 06:14:33 +04:00
sci_controller_remote_device_stopped ( ihost , idev ) ;
2011-04-23 06:18:03 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_remote_device_starting_state_enter ( struct sci_base_state_machine * sm )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 13:25:15 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-23 06:18:03 +04:00
isci_remote_device_not_ready ( ihost , idev ,
SCIC_REMOTE_DEVICE_NOT_READY_START_REQUESTED ) ;
}
2011-07-01 06:14:33 +04:00
static void sci_remote_device_ready_state_enter ( struct sci_base_state_machine * sm )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 04:38:32 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-05-04 11:58:16 +04:00
struct domain_device * dev = idev - > domain_dev ;
2011-04-23 06:18:03 +04:00
2011-04-28 03:32:45 +04:00
if ( dev - > dev_type = = SATA_DEV | | ( dev - > tproto & SAS_PROTOCOL_SATA ) ) {
2011-07-01 03:31:37 +04:00
sci_change_state ( & idev - > sm , SCI_STP_DEV_IDLE ) ;
2011-04-28 03:32:45 +04:00
} else if ( dev_is_expander ( dev ) ) {
2011-07-01 03:31:37 +04:00
sci_change_state ( & idev - > sm , SCI_SMP_DEV_IDLE ) ;
2011-04-28 03:32:45 +04:00
} else
2011-07-01 04:38:32 +04:00
isci_remote_device_ready ( ihost , idev ) ;
2011-04-23 06:18:03 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_remote_device_ready_state_exit ( struct sci_base_state_machine * sm )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
struct domain_device * dev = idev - > domain_dev ;
2011-04-28 03:32:45 +04:00
if ( dev - > dev_type = = SAS_END_DEV ) {
2011-07-01 04:38:32 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-23 06:18:03 +04:00
2011-07-01 04:38:32 +04:00
isci_remote_device_not_ready ( ihost , idev ,
2011-04-23 06:18:03 +04:00
SCIC_REMOTE_DEVICE_NOT_READY_STOP_REQUESTED ) ;
}
}
2011-07-01 06:14:33 +04:00
static void sci_remote_device_resetting_state_enter ( struct sci_base_state_machine * sm )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-04-23 06:18:03 +04:00
2011-07-01 06:14:33 +04:00
sci_remote_node_context_suspend (
2011-07-01 03:31:37 +04:00
& idev - > rnc , SCI_SOFTWARE_SUSPENSION , NULL , NULL ) ;
2011-04-23 06:18:03 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_remote_device_resetting_state_exit ( struct sci_base_state_machine * sm )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-04-23 06:18:03 +04:00
2011-07-01 06:14:33 +04:00
sci_remote_node_context_resume ( & idev - > rnc , NULL , NULL ) ;
2011-04-23 06:18:03 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_stp_remote_device_ready_idle_substate_enter ( struct sci_base_state_machine * sm )
2011-04-28 03:32:45 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-04-28 03:32:45 +04:00
2011-07-01 03:31:37 +04:00
idev - > working_request = NULL ;
2011-07-01 06:14:33 +04:00
if ( sci_remote_node_context_is_ready ( & idev - > rnc ) ) {
2011-04-28 03:32:45 +04:00
/*
* Since the RNC is ready , it ' s alright to finish completion
* processing ( e . g . signal the remote device is ready ) . */
2011-07-01 06:14:33 +04:00
sci_stp_remote_device_ready_idle_substate_resume_complete_handler ( idev ) ;
2011-04-28 03:32:45 +04:00
} else {
2011-07-01 06:14:33 +04:00
sci_remote_node_context_resume ( & idev - > rnc ,
sci_stp_remote_device_ready_idle_substate_resume_complete_handler ,
2011-07-01 03:31:37 +04:00
idev ) ;
2011-04-28 03:32:45 +04:00
}
}
2011-07-01 06:14:33 +04:00
static void sci_stp_remote_device_ready_cmd_substate_enter ( struct sci_base_state_machine * sm )
2011-04-28 03:32:45 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 13:25:15 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-28 03:32:45 +04:00
2011-07-01 03:31:37 +04:00
BUG_ON ( idev - > working_request = = NULL ) ;
2011-04-28 03:32:45 +04:00
2011-07-01 04:38:32 +04:00
isci_remote_device_not_ready ( ihost , idev ,
2011-04-28 03:32:45 +04:00
SCIC_REMOTE_DEVICE_NOT_READY_SATA_REQUEST_STARTED ) ;
}
2011-07-01 06:14:33 +04:00
static void sci_stp_remote_device_ready_ncq_error_substate_enter ( struct sci_base_state_machine * sm )
2011-04-28 03:32:45 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 13:25:15 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-28 03:32:45 +04:00
2011-07-01 03:31:37 +04:00
if ( idev - > not_ready_reason = = SCIC_REMOTE_DEVICE_NOT_READY_SATA_SDB_ERROR_FIS_RECEIVED )
2011-07-01 04:38:32 +04:00
isci_remote_device_not_ready ( ihost , idev ,
2011-07-01 03:31:37 +04:00
idev - > not_ready_reason ) ;
2011-04-28 03:32:45 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_smp_remote_device_ready_idle_substate_enter ( struct sci_base_state_machine * sm )
2011-04-28 03:32:45 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 13:25:15 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-28 03:32:45 +04:00
2011-07-01 04:38:32 +04:00
isci_remote_device_ready ( ihost , idev ) ;
2011-04-28 03:32:45 +04:00
}
2011-07-01 06:14:33 +04:00
static void sci_smp_remote_device_ready_cmd_substate_enter ( struct sci_base_state_machine * sm )
2011-04-28 03:32:45 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-07-01 13:25:15 +04:00
struct isci_host * ihost = idev - > owning_port - > owning_controller ;
2011-04-28 03:32:45 +04:00
2011-07-01 03:31:37 +04:00
BUG_ON ( idev - > working_request = = NULL ) ;
2011-04-28 03:32:45 +04:00
2011-07-01 04:38:32 +04:00
isci_remote_device_not_ready ( ihost , idev ,
2011-04-28 03:32:45 +04:00
SCIC_REMOTE_DEVICE_NOT_READY_SMP_REQUEST_STARTED ) ;
}
2011-07-01 06:14:33 +04:00
static void sci_smp_remote_device_ready_cmd_substate_exit ( struct sci_base_state_machine * sm )
2011-04-28 03:32:45 +04:00
{
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev = container_of ( sm , typeof ( * idev ) , sm ) ;
2011-04-28 03:32:45 +04:00
2011-07-01 03:31:37 +04:00
idev - > working_request = NULL ;
2011-04-28 03:32:45 +04:00
}
2011-04-23 06:18:03 +04:00
2011-07-01 06:14:33 +04:00
static const struct sci_base_state sci_remote_device_state_table [ ] = {
2011-06-02 04:10:43 +04:00
[ SCI_DEV_INITIAL ] = {
2011-07-01 06:14:33 +04:00
. enter_state = sci_remote_device_initial_state_enter ,
2011-04-23 06:18:03 +04:00
} ,
2011-06-02 04:10:43 +04:00
[ SCI_DEV_STOPPED ] = {
2011-07-01 06:14:33 +04:00
. enter_state = sci_remote_device_stopped_state_enter ,
2011-04-23 06:18:03 +04:00
} ,
2011-06-02 04:10:43 +04:00
[ SCI_DEV_STARTING ] = {
2011-07-01 06:14:33 +04:00
. enter_state = sci_remote_device_starting_state_enter ,
2011-04-23 06:18:03 +04:00
} ,
2011-06-02 04:10:43 +04:00
[ SCI_DEV_READY ] = {
2011-07-01 06:14:33 +04:00
. enter_state = sci_remote_device_ready_state_enter ,
. exit_state = sci_remote_device_ready_state_exit
2011-04-23 06:18:03 +04:00
} ,
2011-06-02 04:10:43 +04:00
[ SCI_STP_DEV_IDLE ] = {
2011-07-01 06:14:33 +04:00
. enter_state = sci_stp_remote_device_ready_idle_substate_enter ,
2011-04-28 03:32:45 +04:00
} ,
2011-06-02 04:10:43 +04:00
[ SCI_STP_DEV_CMD ] = {
2011-07-01 06:14:33 +04:00
. enter_state = sci_stp_remote_device_ready_cmd_substate_enter ,
2011-04-28 03:32:45 +04:00
} ,
2011-06-02 04:10:43 +04:00
[ SCI_STP_DEV_NCQ ] = { } ,
[ SCI_STP_DEV_NCQ_ERROR ] = {
2011-07-01 06:14:33 +04:00
. enter_state = sci_stp_remote_device_ready_ncq_error_substate_enter ,
2011-04-28 03:32:45 +04:00
} ,
2011-06-02 04:10:43 +04:00
[ SCI_STP_DEV_AWAIT_RESET ] = { } ,
[ SCI_SMP_DEV_IDLE ] = {
2011-07-01 06:14:33 +04:00
. enter_state = sci_smp_remote_device_ready_idle_substate_enter ,
2011-04-28 03:32:45 +04:00
} ,
2011-06-02 04:10:43 +04:00
[ SCI_SMP_DEV_CMD ] = {
2011-07-01 06:14:33 +04:00
. enter_state = sci_smp_remote_device_ready_cmd_substate_enter ,
. exit_state = sci_smp_remote_device_ready_cmd_substate_exit ,
2011-04-28 03:32:45 +04:00
} ,
2011-06-02 04:10:43 +04:00
[ SCI_DEV_STOPPING ] = { } ,
[ SCI_DEV_FAILED ] = { } ,
[ SCI_DEV_RESETTING ] = {
2011-07-01 06:14:33 +04:00
. enter_state = sci_remote_device_resetting_state_enter ,
. exit_state = sci_remote_device_resetting_state_exit
2011-04-23 06:18:03 +04:00
} ,
2011-06-02 04:10:43 +04:00
[ SCI_DEV_FINAL ] = { } ,
2011-04-23 06:18:03 +04:00
} ;
/**
2011-07-01 06:14:33 +04:00
* sci_remote_device_construct ( ) - common construction
2011-04-23 06:18:03 +04:00
* @ sci_port : SAS / SATA port through which this device is accessed .
* @ sci_dev : remote device to construct
*
2011-04-25 22:48:29 +04:00
* This routine just performs benign initialization and does not
* allocate the remote_node_context which is left to
2011-07-01 06:14:33 +04:00
* sci_remote_device_ [ de ] a_construct ( ) . sci_remote_device_destruct ( )
2011-04-25 22:48:29 +04:00
* frees the remote_node_context ( s ) for the device .
2011-04-23 06:18:03 +04:00
*/
2011-07-01 06:14:33 +04:00
static void sci_remote_device_construct ( struct isci_port * iport ,
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
idev - > owning_port = iport ;
idev - > started_request_count = 0 ;
2011-04-23 06:18:03 +04:00
2011-07-01 06:14:33 +04:00
sci_init_sm ( & idev - > sm , sci_remote_device_state_table , SCI_DEV_INITIAL ) ;
2011-04-23 06:18:03 +04:00
2011-07-01 06:14:33 +04:00
sci_remote_node_context_construct ( & idev - > rnc ,
2011-04-23 06:18:03 +04:00
SCIC_SDS_REMOTE_NODE_CONTEXT_INVALID_INDEX ) ;
}
/**
2011-07-01 06:14:33 +04:00
* sci_remote_device_da_construct ( ) - construct direct attached device .
2011-04-25 22:48:29 +04: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-07-01 06:14:33 +04: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 22:48:29 +04: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-23 06:18:03 +04:00
*/
2011-07-01 06:14:33 +04:00
static enum sci_status sci_remote_device_da_construct ( struct isci_port * iport ,
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev )
2011-04-23 06:18:03 +04:00
{
enum sci_status status ;
2011-07-01 03:31:37 +04:00
struct domain_device * dev = idev - > domain_dev ;
2011-04-23 06:18:03 +04:00
2011-07-01 06:14:33 +04:00
sci_remote_device_construct ( iport , idev ) ;
2011-04-25 22:48:29 +04:00
2011-04-23 06:18:03 +04:00
/*
* This information is request to determine how many remote node context
* entries will be needed to store the remote node .
*/
2011-07-01 03:31:37 +04:00
idev - > is_direct_attached = true ;
2011-07-01 06:14:33 +04:00
status = sci_controller_allocate_remote_node_context ( iport - > owning_controller ,
2011-07-01 03:31:37 +04:00
idev ,
& idev - > rnc . remote_node_index ) ;
2011-04-23 06:18:03 +04:00
2011-04-22 05:44:45 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-04-23 06:18:03 +04:00
2011-04-28 03:32:45 +04:00
if ( dev - > dev_type = = SAS_END_DEV | | dev - > dev_type = = SATA_DEV | |
( dev - > tproto & SAS_PROTOCOL_STP ) | | dev_is_expander ( dev ) )
/* pass */ ;
else
2011-04-22 05:44:45 +04:00
return SCI_FAILURE_UNSUPPORTED_PROTOCOL ;
2011-04-23 06:18:03 +04:00
2011-07-01 06:14:33 +04:00
idev - > connection_rate = sci_port_get_max_allowed_speed ( iport ) ;
2011-04-23 06:18:03 +04:00
2011-04-22 05:44:45 +04:00
/* / @todo Should I assign the port width by reading all of the phys on the port? */
2011-07-01 03:31:37 +04:00
idev - > device_port_width = 1 ;
2011-04-23 06:18:03 +04:00
2011-04-22 05:44:45 +04:00
return SCI_SUCCESS ;
2011-04-23 06:18:03 +04:00
}
/**
2011-07-01 06:14:33 +04:00
* sci_remote_device_ea_construct ( ) - construct expander attached device
2011-04-25 22:48:29 +04:00
*
* Remote node context ( s ) is / are a global resource allocated by this
2011-07-01 06:14:33 +04:00
* routine , freed by sci_remote_device_destruct ( ) .
2011-04-25 22:48:29 +04: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-23 06:18:03 +04:00
*/
2011-07-01 06:14:33 +04:00
static enum sci_status sci_remote_device_ea_construct ( struct isci_port * iport ,
2011-07-01 03:31:37 +04:00
struct isci_remote_device * idev )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct domain_device * dev = idev - > domain_dev ;
2011-04-23 06:18:03 +04:00
enum sci_status status ;
2011-07-01 06:14:33 +04:00
sci_remote_device_construct ( iport , idev ) ;
2011-04-23 06:18:03 +04:00
2011-07-01 06:14:33 +04:00
status = sci_controller_allocate_remote_node_context ( iport - > owning_controller ,
2011-07-01 03:31:37 +04:00
idev ,
& idev - > rnc . remote_node_index ) ;
2011-04-22 05:44:45 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-04-23 06:18:03 +04:00
2011-04-28 03:32:45 +04:00
if ( dev - > dev_type = = SAS_END_DEV | | dev - > dev_type = = SATA_DEV | |
( dev - > tproto & SAS_PROTOCOL_STP ) | | dev_is_expander ( dev ) )
/* pass */ ;
else
return SCI_FAILURE_UNSUPPORTED_PROTOCOL ;
2011-04-23 06:18:03 +04:00
2011-04-22 05:44:45 +04:00
/*
* For SAS - 2 the physical link rate is actually a logical link
* 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
* one another , so this code works for both situations . */
2011-07-01 06:14:33 +04:00
idev - > connection_rate = min_t ( u16 , sci_port_get_max_allowed_speed ( iport ) ,
2011-04-26 01:29:29 +04:00
dev - > linkrate ) ;
2011-04-23 06:18:03 +04:00
2011-04-22 05:44:45 +04:00
/* / @todo Should I assign the port width by reading all of the phys on the port? */
2011-07-01 03:31:37 +04:00
idev - > device_port_width = 1 ;
2011-04-23 06:18:03 +04:00
2011-04-28 03:32:45 +04:00
return SCI_SUCCESS ;
2011-04-23 06:18:03 +04:00
}
/**
2011-07-01 06:14:33 +04:00
* sci_remote_device_start ( ) - This method will start the supplied remote
2011-04-23 06:18:03 +04: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-07-01 06:14:33 +04:00
static enum sci_status sci_remote_device_start ( struct isci_remote_device * idev ,
2011-05-02 01:05:57 +04:00
u32 timeout )
2011-04-23 06:18:03 +04:00
{
2011-07-01 03:31:37 +04:00
struct sci_base_state_machine * sm = & idev - > sm ;
2011-07-01 06:14:33 +04:00
enum sci_remote_device_states state = sm - > current_state_id ;
2011-05-02 01:05:57 +04:00
enum sci_status status ;
2011-06-02 04:10:43 +04:00
if ( state ! = SCI_DEV_STOPPED ) {
2011-07-01 03:31:37 +04:00
dev_warn ( scirdev_to_dev ( idev ) , " %s: in wrong state: %d \n " ,
2011-05-02 01:05:57 +04:00
__func__ , state ) ;
return SCI_FAILURE_INVALID_STATE ;
}
2011-07-01 06:14:33 +04:00
status = sci_remote_node_context_resume ( & idev - > rnc ,
2011-05-02 01:05:57 +04:00
remote_device_resume_done ,
2011-07-01 03:31:37 +04:00
idev ) ;
2011-05-02 01:05:57 +04:00
if ( status ! = SCI_SUCCESS )
return status ;
2011-06-02 04:10:43 +04:00
sci_change_state ( sm , SCI_DEV_STARTING ) ;
2011-05-02 01:05:57 +04:00
return SCI_SUCCESS ;
2011-04-23 06:18:03 +04:00
}
2011-07-03 09:56:22 +04:00
2011-04-26 01:29:29 +04:00
static enum sci_status isci_remote_device_construct ( struct isci_port * iport ,
struct isci_remote_device * idev )
2011-07-03 09:56:22 +04:00
{
2011-04-26 01:29:29 +04:00
struct isci_host * ihost = iport - > isci_host ;
struct domain_device * dev = idev - > domain_dev ;
enum sci_status status ;
2011-07-03 09:56:22 +04:00
2011-04-26 01:29:29 +04:00
if ( dev - > parent & & dev_is_expander ( dev - > parent ) )
2011-07-01 06:14:33 +04:00
status = sci_remote_device_ea_construct ( iport , idev ) ;
2011-04-26 01:29:29 +04:00
else
2011-07-01 06:14:33 +04:00
status = sci_remote_device_da_construct ( iport , idev ) ;
2011-07-03 09:56:22 +04:00
if ( status ! = SCI_SUCCESS ) {
2011-04-26 01:29:29 +04:00
dev_dbg ( & ihost - > pdev - > dev , " %s: construct failed: %d \n " ,
__func__ , status ) ;
2011-07-03 09:56:22 +04:00
return status ;
}
/* start the device. */
2011-07-01 06:14:33 +04:00
status = sci_remote_device_start ( idev , ISCI_REMOTE_DEVICE_START_TIMEOUT ) ;
2011-07-03 09:56:22 +04:00
2011-04-26 01:29:29 +04:00
if ( status ! = SCI_SUCCESS )
dev_warn ( & ihost - > pdev - > dev , " remote device start failed: %d \n " ,
status ) ;
2011-07-03 09:56:22 +04:00
return status ;
}
2011-04-01 00:10:44 +04:00
void isci_remote_device_nuke_requests ( struct isci_host * ihost , struct isci_remote_device * idev )
2011-07-03 09:56:22 +04:00
{
DECLARE_COMPLETION_ONSTACK ( aborted_task_completion ) ;
2011-04-01 00:10:44 +04:00
dev_dbg ( & ihost - > pdev - > dev ,
" %s: idev = %p \n " , __func__ , idev ) ;
2011-07-03 09:56:22 +04:00
/* Cleanup all requests pending for this device. */
2011-06-21 02:11:22 +04:00
isci_terminate_pending_requests ( ihost , idev ) ;
2011-07-03 09:56:22 +04:00
2011-04-01 00:10:44 +04:00
dev_dbg ( & ihost - > pdev - > dev ,
" %s: idev = %p, done \n " , __func__ , idev ) ;
2011-07-03 09:56:22 +04:00
}
/**
* 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-04 04:59:32 +03:00
isci_remote_device_alloc ( struct isci_host * ihost , struct isci_port * iport )
2011-07-03 09:56:22 +04:00
{
2011-03-04 04:59:32 +03:00
struct isci_remote_device * idev ;
int i ;
2011-07-03 09:56:22 +04:00
2011-03-04 04:59:32 +03:00
for ( i = 0 ; i < SCI_MAX_REMOTE_DEVICES ; i + + ) {
2011-04-22 05:14:45 +04:00
idev = & ihost - > devices [ i ] ;
2011-03-04 04:59:32 +03:00
if ( ! test_and_set_bit ( IDEV_ALLOCATED , & idev - > flags ) )
break ;
}
2011-07-03 09:56:22 +04:00
2011-03-04 04:59:32 +03:00
if ( i > = SCI_MAX_REMOTE_DEVICES ) {
dev_warn ( & ihost - > pdev - > dev , " %s: failed \n " , __func__ ) ;
2011-07-03 09:56:22 +04:00
return NULL ;
}
2011-04-13 04:28:43 +04:00
if ( WARN_ONCE ( ! list_empty ( & idev - > reqs_in_process ) , " found requests in process \n " ) )
return NULL ;
if ( WARN_ONCE ( ! list_empty ( & idev - > node ) , " found non-idle remote device \n " ) )
return NULL ;
2011-03-04 04:59:32 +03:00
return idev ;
2011-07-03 09:56:22 +04:00
}
2011-06-14 04:39:44 +04: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 22:26:12 +04:00
clear_bit ( IDEV_IO_READY , & idev - > flags ) ;
2011-06-14 04:39:44 +04:00
clear_bit ( IDEV_GONE , & idev - > flags ) ;
clear_bit ( IDEV_EH , & idev - > flags ) ;
smp_mb__before_clear_bit ( ) ;
clear_bit ( IDEV_ALLOCATED , & idev - > flags ) ;
wake_up ( & ihost - > eventq ) ;
}
2011-07-03 09:56:22 +04: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-07-01 04:38:32 +04:00
* The status of the ihost request to stop .
2011-07-03 09:56:22 +04:00
*/
2011-03-04 23:10:29 +03:00
enum sci_status isci_remote_device_stop ( struct isci_host * ihost , struct isci_remote_device * idev )
2011-07-03 09:56:22 +04:00
{
enum sci_status status ;
unsigned long flags ;
2011-03-04 23:10:29 +03:00
dev_dbg ( & ihost - > pdev - > dev ,
" %s: isci_device = %p \n " , __func__ , idev ) ;
2011-07-03 09:56:22 +04:00
2011-06-14 04:39:44 +04:00
spin_lock_irqsave ( & ihost - > scic_lock , flags ) ;
idev - > domain_dev - > lldd_dev = NULL ; /* disable new lookups */
set_bit ( IDEV_GONE , & idev - > flags ) ;
spin_unlock_irqrestore ( & ihost - > scic_lock , flags ) ;
2011-03-09 06:32:16 +03:00
/* Kill all outstanding requests. */
2011-04-01 00:10:44 +04:00
isci_remote_device_nuke_requests ( ihost , idev ) ;
2011-03-09 06:32:16 +03:00
2011-03-04 23:10:29 +03:00
set_bit ( IDEV_STOP_PENDING , & idev - > flags ) ;
2011-07-03 09:56:22 +04:00
2011-03-04 23:10:29 +03:00
spin_lock_irqsave ( & ihost - > scic_lock , flags ) ;
2011-07-01 06:14:33 +04:00
status = sci_remote_device_stop ( idev , 50 ) ;
2011-03-04 23:10:29 +03:00
spin_unlock_irqrestore ( & ihost - > scic_lock , flags ) ;
2011-07-03 09:56:22 +04:00
/* Wait for the stop complete callback. */
2011-06-14 04:39:44 +04:00
if ( WARN_ONCE ( status ! = SCI_SUCCESS , " failed to stop device \n " ) )
/* nothing to wait for */ ;
else
2011-03-04 23:10:29 +03:00
wait_for_device_stop ( ihost , idev ) ;
2011-07-03 09:56:22 +04: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 23:10:29 +03:00
void isci_remote_device_gone ( struct domain_device * dev )
2011-07-03 09:56:22 +04:00
{
2011-04-01 00:10:44 +04:00
struct isci_host * ihost = dev_to_ihost ( dev ) ;
2011-03-04 23:10:29 +03:00
struct isci_remote_device * idev = dev - > lldd_dev ;
2011-07-03 09:56:22 +04:00
2011-03-04 23:10:29 +03:00
dev_dbg ( & ihost - > pdev - > dev ,
2011-07-03 09:56:22 +04:00
" %s: domain_device = %p, isci_device = %p, isci_port = %p \n " ,
2011-03-04 23:10:29 +03:00
__func__ , dev , idev , idev - > isci_port ) ;
2011-07-03 09:56:22 +04:00
2011-03-04 23:10:29 +03:00
isci_remote_device_stop ( ihost , idev ) ;
2011-07-03 09:56:22 +04: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 .
*/
int isci_remote_device_found ( struct domain_device * domain_dev )
{
2011-04-01 00:10:44 +04:00
struct isci_host * isci_host = dev_to_ihost ( domain_dev ) ;
2011-07-03 09:56:22 +04:00
struct isci_port * isci_port ;
struct isci_phy * isci_phy ;
struct asd_sas_port * sas_port ;
struct asd_sas_phy * sas_phy ;
struct isci_remote_device * isci_device ;
enum sci_status status ;
dev_dbg ( & isci_host - > pdev - > dev ,
" %s: domain_device = %p \n " , __func__ , domain_dev ) ;
2011-02-18 20:25:07 +03:00
wait_for_start ( isci_host ) ;
2011-07-03 09:56:22 +04:00
sas_port = domain_dev - > port ;
sas_phy = list_first_entry ( & sas_port - > phy_list , struct asd_sas_phy ,
port_phy_el ) ;
2011-06-29 02:05:53 +04:00
isci_phy = to_iphy ( sas_phy ) ;
2011-07-03 09:56:22 +04:00
isci_port = isci_phy - > isci_port ;
/* we are being called for a device on this port,
* so it has to come up eventually
*/
wait_for_completion ( & isci_port - > start_complete ) ;
if ( ( isci_stopping = = isci_port_get_state ( isci_port ) ) | |
( isci_stopped = = isci_port_get_state ( isci_port ) ) )
return - ENODEV ;
isci_device = isci_remote_device_alloc ( isci_host , isci_port ) ;
2011-03-04 04:59:32 +03:00
if ( ! isci_device )
return - ENODEV ;
2011-07-03 09:56:22 +04:00
2011-06-14 04:39:44 +04:00
kref_init ( & isci_device - > kref ) ;
2011-07-03 09:56:22 +04:00
INIT_LIST_HEAD ( & isci_device - > node ) ;
2011-06-14 04:39:44 +04:00
spin_lock_irq ( & isci_host - > scic_lock ) ;
2011-07-03 09:56:22 +04:00
isci_device - > domain_dev = domain_dev ;
isci_device - > isci_port = isci_port ;
list_add_tail ( & isci_device - > node , & isci_port - > remote_dev_list ) ;
2011-03-04 23:10:29 +03:00
set_bit ( IDEV_START_PENDING , & isci_device - > flags ) ;
2011-07-03 09:56:22 +04: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-14 04:39:44 +04:00
if ( status = = SCI_SUCCESS ) {
/* device came up, advertise it to the world */
domain_dev - > lldd_dev = isci_device ;
} else
isci_put_device ( isci_device ) ;
spin_unlock_irq ( & isci_host - > scic_lock ) ;
2011-07-03 09:56:22 +04:00
2011-03-04 23:10:29 +03:00
/* wait for the device ready callback. */
wait_for_device_start ( isci_host , isci_device ) ;
2011-06-14 04:39:44 +04:00
return status = = SCI_SUCCESS ? 0 : - ENODEV ;
2011-07-03 09:56:22 +04:00
}
/**
* isci_device_is_reset_pending ( ) - This function will check if there is any
* pending reset condition on the device .
* @ request : This parameter is the isci_device object .
*
* true if there is a reset pending for the device .
*/
bool isci_device_is_reset_pending (
struct isci_host * isci_host ,
struct isci_remote_device * isci_device )
{
struct isci_request * isci_request ;
struct isci_request * tmp_req ;
bool reset_is_pending = false ;
unsigned long flags ;
dev_dbg ( & isci_host - > pdev - > dev ,
" %s: isci_device = %p \n " , __func__ , isci_device ) ;
spin_lock_irqsave ( & isci_host - > scic_lock , flags ) ;
/* Check for reset on all pending requests. */
list_for_each_entry_safe ( isci_request , tmp_req ,
& isci_device - > reqs_in_process , dev_node ) {
dev_dbg ( & isci_host - > pdev - > dev ,
" %s: isci_device = %p request = %p \n " ,
__func__ , isci_device , isci_request ) ;
if ( isci_request - > ttype = = io_task ) {
struct sas_task * task = isci_request_access_task (
isci_request ) ;
2011-04-13 04:28:41 +04:00
spin_lock ( & task - > task_state_lock ) ;
2011-07-03 09:56:22 +04:00
if ( task - > task_state_flags & SAS_TASK_NEED_DEV_RESET )
reset_is_pending = true ;
2011-04-13 04:28:41 +04:00
spin_unlock ( & task - > task_state_lock ) ;
2011-07-03 09:56:22 +04:00
}
}
spin_unlock_irqrestore ( & isci_host - > scic_lock , flags ) ;
dev_dbg ( & isci_host - > pdev - > dev ,
" %s: isci_device = %p reset_is_pending = %d \n " ,
__func__ , isci_device , reset_is_pending ) ;
return reset_is_pending ;
}
/**
* isci_device_clear_reset_pending ( ) - This function will clear if any pending
* reset condition flags on the device .
* @ request : This parameter is the isci_device object .
*
* true if there is a reset pending for the device .
*/
2011-04-01 00:10:44 +04:00
void isci_device_clear_reset_pending ( struct isci_host * ihost , struct isci_remote_device * idev )
2011-07-03 09:56:22 +04:00
{
struct isci_request * isci_request ;
struct isci_request * tmp_req ;
unsigned long flags = 0 ;
2011-04-01 00:10:44 +04:00
dev_dbg ( & ihost - > pdev - > dev , " %s: idev=%p, ihost=%p \n " ,
__func__ , idev , ihost ) ;
2011-07-03 09:56:22 +04:00
2011-04-01 00:10:44 +04:00
spin_lock_irqsave ( & ihost - > scic_lock , flags ) ;
2011-07-03 09:56:22 +04:00
/* Clear reset pending on all pending requests. */
list_for_each_entry_safe ( isci_request , tmp_req ,
2011-04-01 00:10:44 +04:00
& idev - > reqs_in_process , dev_node ) {
dev_dbg ( & ihost - > pdev - > dev , " %s: idev = %p request = %p \n " ,
__func__ , idev , isci_request ) ;
2011-07-03 09:56:22 +04:00
if ( isci_request - > ttype = = io_task ) {
unsigned long flags2 ;
struct sas_task * task = isci_request_access_task (
isci_request ) ;
spin_lock_irqsave ( & task - > task_state_lock , flags2 ) ;
task - > task_state_flags & = ~ SAS_TASK_NEED_DEV_RESET ;
spin_unlock_irqrestore ( & task - > task_state_lock , flags2 ) ;
}
}
2011-04-01 00:10:44 +04:00
spin_unlock_irqrestore ( & ihost - > scic_lock , flags ) ;
2011-07-03 09:56:22 +04:00
}