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-07 21:11:43 +04:00
# ifndef _ISCI_PORT_H_
2011-07-03 09:56:22 +04:00
# define _ISCI_PORT_H_
2011-05-09 02:49:15 +04:00
# include <scsi/libsas.h>
# include "isci.h"
2011-05-10 13:28:46 +04:00
# include "sas.h"
# include "phy.h"
# define SCIC_SDS_DUMMY_PORT 0xFF
2011-07-03 09:56:22 +04:00
2012-01-04 13:33:41 +04:00
# define PF_NOTIFY (1 << 0)
# define PF_RESUME (1 << 1)
2011-07-03 09:56:22 +04:00
struct isci_phy ;
struct isci_host ;
enum isci_status {
isci_freed = 0x00 ,
isci_starting = 0x01 ,
isci_ready = 0x02 ,
isci_ready_for_io = 0x03 ,
isci_stopping = 0x04 ,
isci_stopped = 0x05 ,
} ;
2011-05-10 13:28:46 +04:00
/**
2011-06-30 00:09:25 +04:00
* struct isci_port - isci direct attached sas port object
* @ ready_exit : several states constitute ' ready ' . When exiting ready we
* need to take extra port - teardown actions that are
* skipped when exiting to another ' ready ' state .
* @ logical_port_index : software port index
* @ physical_port_index : hardware port index
* @ active_phy_mask : identifies phy members
2012-01-04 13:33:41 +04:00
* @ enabled_phy_mask : phy mask for the port
* that are already part of the port
2011-06-30 00:09:25 +04:00
* @ reserved_tag :
* @ reserved_rni : reserver for port task scheduler workaround
* @ started_request_count : reference count for outstanding commands
* @ not_ready_reason : set during state transitions and notified
* @ timer : timeout start / stop operations
2011-07-03 09:56:22 +04:00
*/
struct isci_port {
struct isci_host * isci_host ;
struct list_head remote_dev_list ;
2011-11-30 23:57:34 +04:00
# define IPORT_RESET_PENDING 0
unsigned long state ;
2011-07-03 09:56:22 +04:00
enum sci_status hard_reset_status ;
2011-06-30 00:09:25 +04:00
struct sci_base_state_machine sm ;
bool ready_exit ;
u8 logical_port_index ;
u8 physical_port_index ;
u8 active_phy_mask ;
2012-01-04 13:33:41 +04:00
u8 enabled_phy_mask ;
2011-10-28 02:05:32 +04:00
u8 last_active_phy ;
2011-06-30 00:09:25 +04:00
u16 reserved_rni ;
u16 reserved_tag ;
u32 started_request_count ;
u32 assigned_device_count ;
2012-03-09 10:41:48 +04:00
u32 hang_detect_users ;
2011-06-30 00:09:25 +04:00
u32 not_ready_reason ;
struct isci_phy * phy_table [ SCI_MAX_PHYS ] ;
2011-07-01 04:38:32 +04:00
struct isci_host * owning_controller ;
2011-06-30 00:09:25 +04:00
struct sci_timer timer ;
struct scu_port_task_scheduler_registers __iomem * port_task_scheduler_registers ;
/* XXX rework: only one register, no need to replicate per-port */
u32 __iomem * port_pe_configuration_register ;
struct scu_viit_entry __iomem * viit_registers ;
2011-07-03 09:56:22 +04:00
} ;
2011-07-01 06:14:33 +04:00
enum sci_port_not_ready_reason_code {
2011-05-10 13:28:46 +04:00
SCIC_PORT_NOT_READY_NO_ACTIVE_PHYS ,
SCIC_PORT_NOT_READY_HARD_RESET_REQUESTED ,
SCIC_PORT_NOT_READY_INVALID_PORT_CONFIGURATION ,
SCIC_PORT_NOT_READY_RECONFIGURING ,
SCIC_PORT_NOT_READY_REASON_CODE_MAX
} ;
2011-07-01 06:14:33 +04:00
struct sci_port_end_point_properties {
2011-05-10 13:28:46 +04:00
struct sci_sas_address sas_address ;
2011-07-01 06:14:33 +04:00
struct sci_phy_proto protocols ;
2011-05-10 13:28:46 +04:00
} ;
2011-07-01 06:14:33 +04:00
struct sci_port_properties {
2011-05-10 13:28:46 +04:00
u32 index ;
2011-07-01 06:14:33 +04:00
struct sci_port_end_point_properties local ;
struct sci_port_end_point_properties remote ;
2011-05-10 13:28:46 +04:00
u32 phy_mask ;
} ;
/**
2012-02-10 13:18:44 +04:00
* enum sci_port_states - port state machine states
* @ SCI_PORT_STOPPED : port has successfully been stopped . In this state
* no new IO operations are permitted . This state is
* entered from the STOPPING state .
* @ SCI_PORT_STOPPING : port is in the process of stopping . In this
* state no new IO operations are permitted , but
* existing IO operations are allowed to complete .
* This state is entered from the READY state .
* @ SCI_PORT_READY : port is now ready . Thus , the user is able to
* perform IO operations on this port . This state is
* entered from the STARTING state .
* @ SCI_PORT_SUB_WAITING : port is started and ready but has no active
* phys .
* @ SCI_PORT_SUB_OPERATIONAL : port is started and ready and there is at
* least one phy operational .
* @ SCI_PORT_SUB_CONFIGURING : port is started and there was an
* add / remove phy event . This state is only
* used in Automatic Port Configuration Mode
* ( APC )
* @ SCI_PORT_RESETTING : port is in the process of performing a hard
* reset . Thus , the user is unable to perform IO
* operations on this port . This state is entered
* from the READY state .
* @ SCI_PORT_FAILED : port has failed a reset request . This state is
* entered when a port reset request times out . This
* state is entered from the RESETTING state .
2011-05-10 13:28:46 +04:00
*/
2012-02-10 13:18:44 +04:00
# define PORT_STATES {\
C ( PORT_STOPPED ) , \
C ( PORT_STOPPING ) , \
C ( PORT_READY ) , \
C ( PORT_SUB_WAITING ) , \
C ( PORT_SUB_OPERATIONAL ) , \
C ( PORT_SUB_CONFIGURING ) , \
C ( PORT_RESETTING ) , \
C ( PORT_FAILED ) , \
}
# undef C
# define C(a) SCI_##a
enum sci_port_states PORT_STATES ;
# undef C
2011-05-10 13:28:46 +04:00
2011-07-01 06:14:33 +04:00
static inline void sci_port_decrement_request_count ( struct isci_port * iport )
2011-05-10 13:28:46 +04:00
{
2011-06-30 00:09:25 +04:00
if ( WARN_ONCE ( iport - > started_request_count = = 0 ,
2011-05-10 13:28:46 +04:00
" %s: tried to decrement started_request_count past 0!? " ,
__func__ ) )
/* pass */ ;
else
2011-06-30 00:09:25 +04:00
iport - > started_request_count - - ;
2011-05-10 13:28:46 +04:00
}
2011-07-01 06:14:33 +04:00
# define sci_port_active_phy(port, phy) \
2011-05-10 13:28:46 +04:00
( ( ( port ) - > active_phy_mask & ( 1 < < ( phy ) - > phy_index ) ) ! = 0 )
2011-07-01 06:14:33 +04:00
void sci_port_construct (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ,
2011-05-10 13:28:46 +04:00
u8 port_index ,
2011-07-01 04:38:32 +04:00
struct isci_host * ihost ) ;
2011-05-10 13:28:46 +04:00
2011-07-01 06:14:33 +04:00
enum sci_status sci_port_start ( struct isci_port * iport ) ;
enum sci_status sci_port_stop ( struct isci_port * iport ) ;
2011-05-12 03:52:26 +04:00
2011-07-01 06:14:33 +04:00
enum sci_status sci_port_add_phy (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ,
2011-06-29 02:05:53 +04:00
struct isci_phy * iphy ) ;
2011-05-10 13:28:46 +04:00
2011-07-01 06:14:33 +04:00
enum sci_status sci_port_remove_phy (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ,
2011-06-29 02:05:53 +04:00
struct isci_phy * iphy ) ;
2011-05-10 13:28:46 +04:00
2011-07-01 06:14:33 +04:00
void sci_port_setup_transports (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ,
2011-05-10 13:28:46 +04:00
u32 device_id ) ;
2011-06-21 23:16:33 +04:00
void isci_port_bcn_enable ( struct isci_host * , struct isci_port * ) ;
2011-05-10 13:28:46 +04:00
2011-07-01 06:14:33 +04:00
void sci_port_deactivate_phy (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ,
2011-06-29 02:05:53 +04:00
struct isci_phy * iphy ,
2011-05-10 13:28:46 +04:00
bool do_notify_user ) ;
2011-07-01 06:14:33 +04:00
bool sci_port_link_detected (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ,
2011-06-29 02:05:53 +04:00
struct isci_phy * iphy ) ;
2011-05-10 13:28:46 +04:00
2012-01-04 13:33:20 +04:00
enum sci_status sci_port_get_properties (
struct isci_port * iport ,
struct sci_port_properties * prop ) ;
2011-07-01 06:14:33 +04:00
enum sci_status sci_port_link_up ( struct isci_port * iport ,
2011-06-29 02:05:53 +04:00
struct isci_phy * iphy ) ;
2011-07-01 06:14:33 +04:00
enum sci_status sci_port_link_down ( struct isci_port * iport ,
2011-06-29 02:05:53 +04:00
struct isci_phy * iphy ) ;
2011-05-10 13:28:46 +04:00
2011-06-28 01:57:03 +04:00
struct isci_request ;
2011-07-01 03:31:37 +04:00
struct isci_remote_device ;
2011-07-01 06:14:33 +04:00
enum sci_status sci_port_start_io (
2011-06-30 00:09:25 +04:00
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-05-10 13:28:46 +04:00
2011-07-01 06:14:33 +04:00
enum sci_status sci_port_complete_io (
2011-06-30 00:09:25 +04:00
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-05-10 13:28:46 +04:00
2011-07-01 06:14:33 +04:00
enum sas_linkrate sci_port_get_max_allowed_speed (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ) ;
2011-05-10 13:28:46 +04:00
2011-07-01 06:14:33 +04:00
void sci_port_broadcast_change_received (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ,
2011-06-29 02:05:53 +04:00
struct isci_phy * iphy ) ;
2011-05-10 13:28:46 +04:00
2011-07-01 06:14:33 +04:00
bool sci_port_is_valid_phy_assignment (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ,
2011-05-10 13:28:46 +04:00
u32 phy_index ) ;
2011-07-01 06:14:33 +04:00
void sci_port_get_sas_address (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ,
2011-05-10 13:28:46 +04:00
struct sci_sas_address * sas_address ) ;
2011-07-01 06:14:33 +04:00
void sci_port_get_attached_sas_address (
2011-06-30 00:09:25 +04:00
struct isci_port * iport ,
2011-05-10 13:28:46 +04:00
struct sci_sas_address * sas_address ) ;
2012-03-09 10:41:48 +04:00
void sci_port_set_hang_detection_timeout (
struct isci_port * isci_port ,
u32 timeout ) ;
2011-05-10 13:28:46 +04:00
void isci_port_formed ( struct asd_sas_phy * ) ;
void isci_port_deformed ( struct asd_sas_phy * ) ;
2011-07-03 09:56:22 +04:00
2011-04-01 00:10:44 +04:00
int isci_port_perform_hard_reset ( struct isci_host * ihost , struct isci_port * iport ,
struct isci_phy * iphy ) ;
2011-11-18 06:01:38 +04:00
int isci_ata_check_ready ( struct domain_device * dev ) ;
2011-07-03 09:56:22 +04:00
# endif /* !defined(_ISCI_PORT_H_) */