2012-02-29 12:11:50 +05:30
/*
2013-02-25 21:44:33 +05:30
* Universal Flash Storage Host controller driver Core
2012-02-29 12:11:50 +05:30
*
* This code is based on drivers / scsi / ufs / ufshcd . c
2013-02-25 21:44:32 +05:30
* Copyright ( C ) 2011 - 2013 Samsung India Software Operations
2014-09-25 15:32:21 +03:00
* Copyright ( c ) 2013 - 2014 , The Linux Foundation . All rights reserved .
2012-02-29 12:11:50 +05:30
*
2013-02-25 21:44:32 +05:30
* Authors :
* Santosh Yaraganavi < santosh . sy @ samsung . com >
* Vinayak Holikatti < h . vinayak @ samsung . com >
2012-02-29 12:11:50 +05:30
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version 2
* of the License , or ( at your option ) any later version .
2013-02-25 21:44:32 +05:30
* See the COPYING file in the top - level directory or visit
* < http : //www.gnu.org/licenses/gpl-2.0.html>
2012-02-29 12:11:50 +05:30
*
* 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 .
*
2013-02-25 21:44:32 +05:30
* This program is provided " AS IS " and " WITH ALL FAULTS " and
* without warranty of any kind . You are solely responsible for
* determining the appropriateness of using and distributing
* the program and assume all risks associated with your exercise
* of rights with respect to the program , including but not limited
* to infringement of third party rights , the risks and costs of
* program errors , damage to or loss of data , programs or equipment ,
* and unavailability or interruption of operations . Under no
* circumstances will the contributor of this Program be liable for
* any damages of any kind arising from your use or distribution of
* this program .
2014-09-25 15:32:21 +03:00
*
* The Linux Foundation chooses to take subject only to the GPLv2
* license terms , and distributes only under these terms .
2012-02-29 12:11:50 +05:30
*/
2013-06-26 22:39:29 +05:30
# include <linux/async.h>
2013-02-25 21:44:33 +05:30
# include "ufshcd.h"
2013-08-31 21:40:22 +05:30
# include "unipro.h"
2012-02-29 12:11:50 +05:30
2013-06-26 22:39:27 +05:30
# define UFSHCD_ENABLE_INTRS (UTP_TRANSFER_REQ_COMPL |\
UTP_TASK_REQ_COMPL | \
2013-08-31 21:40:22 +05:30
UIC_POWER_MODE | \
2013-06-26 22:39:27 +05:30
UFSHCD_ERROR_MASK )
2013-06-26 22:39:29 +05:30
/* UIC command timeout, unit: ms */
# define UIC_CMD_TIMEOUT 500
2013-06-26 22:39:27 +05:30
2013-07-30 00:35:57 +05:30
/* NOP OUT retries waiting for NOP IN response */
# define NOP_OUT_RETRIES 10
/* Timeout after 30 msecs if NOP OUT hangs without response */
# define NOP_OUT_TIMEOUT 30 /* msecs */
2013-07-30 00:35:58 +05:30
/* Query request retries */
# define QUERY_REQ_RETRIES 10
/* Query request timeout */
# define QUERY_REQ_TIMEOUT 30 /* msec */
2014-05-26 10:59:12 +05:30
/* Task management command timeout */
# define TM_CMD_TIMEOUT 100 /* msecs */
2014-09-25 15:32:26 +03:00
/* maximum number of link-startup retries */
# define DME_LINKSTARTUP_RETRIES 3
/* maximum number of reset retries before giving up */
# define MAX_HOST_RESET_RETRIES 5
2013-07-30 00:35:58 +05:30
/* Expose the flag value from utp_upiu_query.value */
# define MASK_QUERY_UPIU_FLAG_LOC 0xFF
2013-08-31 21:40:20 +05:30
/* Interrupt aggregation default timeout, unit: 40us */
# define INT_AGGR_DEF_TO 0x02
2014-09-25 15:32:22 +03:00
# define ufshcd_toggle_vreg(_dev, _vreg, _on) \
( { \
int _ret ; \
if ( _on ) \
_ret = ufshcd_enable_vreg ( _dev , _vreg ) ; \
else \
_ret = ufshcd_disable_vreg ( _dev , _vreg ) ; \
_ret ; \
} )
2014-09-25 15:32:25 +03:00
static u32 ufs_query_desc_max_size [ ] = {
QUERY_DESC_DEVICE_MAX_SIZE ,
QUERY_DESC_CONFIGURAION_MAX_SIZE ,
QUERY_DESC_UNIT_MAX_SIZE ,
QUERY_DESC_RFU_MAX_SIZE ,
QUERY_DESC_INTERCONNECT_MAX_SIZE ,
QUERY_DESC_STRING_MAX_SIZE ,
QUERY_DESC_RFU_MAX_SIZE ,
QUERY_DESC_GEOMETRY_MAZ_SIZE ,
QUERY_DESC_POWER_MAX_SIZE ,
QUERY_DESC_RFU_MAX_SIZE ,
} ;
2012-02-29 12:11:50 +05:30
enum {
UFSHCD_MAX_CHANNEL = 0 ,
UFSHCD_MAX_ID = 1 ,
UFSHCD_CMD_PER_LUN = 32 ,
UFSHCD_CAN_QUEUE = 32 ,
} ;
/* UFSHCD states */
enum {
UFSHCD_STATE_RESET ,
UFSHCD_STATE_ERROR ,
2014-05-26 10:59:14 +05:30
UFSHCD_STATE_OPERATIONAL ,
} ;
/* UFSHCD error handling flags */
enum {
UFSHCD_EH_IN_PROGRESS = ( 1 < < 0 ) ,
2012-02-29 12:11:50 +05:30
} ;
2014-05-26 10:59:15 +05:30
/* UFSHCD UIC layer error flags */
enum {
UFSHCD_UIC_DL_PA_INIT_ERROR = ( 1 < < 0 ) , /* Data link layer error */
UFSHCD_UIC_NL_ERROR = ( 1 < < 1 ) , /* Network layer error */
UFSHCD_UIC_TL_ERROR = ( 1 < < 2 ) , /* Transport Layer error */
UFSHCD_UIC_DME_ERROR = ( 1 < < 3 ) , /* DME error */
} ;
2012-02-29 12:11:50 +05:30
/* Interrupt configuration options */
enum {
UFSHCD_INT_DISABLE ,
UFSHCD_INT_ENABLE ,
UFSHCD_INT_CLEAR ,
} ;
2014-05-26 10:59:14 +05:30
# define ufshcd_set_eh_in_progress(h) \
( h - > eh_flags | = UFSHCD_EH_IN_PROGRESS )
# define ufshcd_eh_in_progress(h) \
( h - > eh_flags & UFSHCD_EH_IN_PROGRESS )
# define ufshcd_clear_eh_in_progress(h) \
( h - > eh_flags & = ~ UFSHCD_EH_IN_PROGRESS )
static void ufshcd_tmc_handler ( struct ufs_hba * hba ) ;
static void ufshcd_async_scan ( void * data , async_cookie_t cookie ) ;
2014-05-26 10:59:15 +05:30
static int ufshcd_reset_and_restore ( struct ufs_hba * hba ) ;
static int ufshcd_clear_tm_cmd ( struct ufs_hba * hba , int tag ) ;
2014-09-25 15:32:26 +03:00
static void ufshcd_hba_exit ( struct ufs_hba * hba ) ;
static int ufshcd_probe_hba ( struct ufs_hba * hba ) ;
2014-05-26 10:59:14 +05:30
2013-07-30 00:35:57 +05:30
/*
* ufshcd_wait_for_register - wait for register value to change
* @ hba - per - adapter interface
* @ reg - mmio register offset
* @ mask - mask to apply to read register value
* @ val - wait condition
* @ interval_us - polling interval in microsecs
* @ timeout_ms - timeout in millisecs
*
* Returns - ETIMEDOUT on error , zero on success
*/
static int ufshcd_wait_for_register ( struct ufs_hba * hba , u32 reg , u32 mask ,
u32 val , unsigned long interval_us , unsigned long timeout_ms )
{
int err = 0 ;
unsigned long timeout = jiffies + msecs_to_jiffies ( timeout_ms ) ;
/* ignore bits that we don't intend to wait on */
val = val & mask ;
while ( ( ufshcd_readl ( hba , reg ) & mask ) ! = val ) {
/* wakeup within 50us of expiry */
usleep_range ( interval_us , interval_us + 50 ) ;
if ( time_after ( jiffies , timeout ) ) {
if ( ( ufshcd_readl ( hba , reg ) & mask ) ! = val )
err = - ETIMEDOUT ;
break ;
}
}
return err ;
}
2013-06-26 22:39:27 +05:30
/**
* ufshcd_get_intr_mask - Get the interrupt bit mask
* @ hba - Pointer to adapter instance
*
* Returns interrupt bit mask per version
*/
static inline u32 ufshcd_get_intr_mask ( struct ufs_hba * hba )
{
if ( hba - > ufs_version = = UFSHCI_VERSION_10 )
return INTERRUPT_MASK_ALL_VER_10 ;
else
return INTERRUPT_MASK_ALL_VER_11 ;
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_get_ufs_version - Get the UFS version supported by the HBA
* @ hba - Pointer to adapter instance
*
* Returns UFSHCI version supported by the controller
*/
static inline u32 ufshcd_get_ufs_version ( struct ufs_hba * hba )
{
2013-06-26 22:39:26 +05:30
return ufshcd_readl ( hba , REG_UFS_VERSION ) ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_is_device_present - Check if any device connected to
* the host controller
2014-09-25 15:32:21 +03:00
* @ hba : pointer to adapter instance
2012-02-29 12:11:50 +05:30
*
2012-07-10 19:39:23 +05:30
* Returns 1 if device present , 0 if no device detected
2012-02-29 12:11:50 +05:30
*/
2014-09-25 15:32:21 +03:00
static inline int ufshcd_is_device_present ( struct ufs_hba * hba )
2012-02-29 12:11:50 +05:30
{
2014-09-25 15:32:21 +03:00
return ( ufshcd_readl ( hba , REG_CONTROLLER_STATUS ) &
DEVICE_PRESENT ) ? 1 : 0 ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_get_tr_ocs - Get the UTRD Overall Command Status
* @ lrb : pointer to local command reference block
*
* This function is used to get the OCS field from UTRD
* Returns the OCS field in the UTRD
*/
static inline int ufshcd_get_tr_ocs ( struct ufshcd_lrb * lrbp )
{
2014-05-26 10:59:10 +05:30
return le32_to_cpu ( lrbp - > utr_descriptor_ptr - > header . dword_2 ) & MASK_OCS ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_get_tmr_ocs - Get the UTMRD Overall Command Status
* @ task_req_descp : pointer to utp_task_req_desc structure
*
* This function is used to get the OCS field from UTMRD
* Returns the OCS field in the UTMRD
*/
static inline int
ufshcd_get_tmr_ocs ( struct utp_task_req_desc * task_req_descp )
{
2014-05-26 10:59:10 +05:30
return le32_to_cpu ( task_req_descp - > header . dword_2 ) & MASK_OCS ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_get_tm_free_slot - get a free slot for task management request
* @ hba : per adapter instance
2014-05-26 10:59:12 +05:30
* @ free_slot : pointer to variable with available slot value
2012-02-29 12:11:50 +05:30
*
2014-05-26 10:59:12 +05:30
* Get a free tag and lock it until ufshcd_put_tm_slot ( ) is called .
* Returns 0 if free slot is not available , else return 1 with tag value
* in @ free_slot .
2012-02-29 12:11:50 +05:30
*/
2014-05-26 10:59:12 +05:30
static bool ufshcd_get_tm_free_slot ( struct ufs_hba * hba , int * free_slot )
2012-02-29 12:11:50 +05:30
{
2014-05-26 10:59:12 +05:30
int tag ;
bool ret = false ;
if ( ! free_slot )
goto out ;
do {
tag = find_first_zero_bit ( & hba - > tm_slots_in_use , hba - > nutmrs ) ;
if ( tag > = hba - > nutmrs )
goto out ;
} while ( test_and_set_bit_lock ( tag , & hba - > tm_slots_in_use ) ) ;
* free_slot = tag ;
ret = true ;
out :
return ret ;
}
static inline void ufshcd_put_tm_slot ( struct ufs_hba * hba , int slot )
{
clear_bit_unlock ( slot , & hba - > tm_slots_in_use ) ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_utrl_clear - Clear a bit in UTRLCLR register
* @ hba : per adapter instance
* @ pos : position of the bit to be cleared
*/
static inline void ufshcd_utrl_clear ( struct ufs_hba * hba , u32 pos )
{
2013-06-26 22:39:26 +05:30
ufshcd_writel ( hba , ~ ( 1 < < pos ) , REG_UTP_TRANSFER_REQ_LIST_CLEAR ) ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_get_lists_status - Check UCRDY , UTRLRDY and UTMRLRDY
* @ reg : Register value of host controller status
*
* Returns integer , 0 on Success and positive value if failed
*/
static inline int ufshcd_get_lists_status ( u32 reg )
{
/*
* The mask 0xFF is for the following HCS register bits
* Bit Description
* 0 Device Present
* 1 UTRLRDY
* 2 UTMRLRDY
* 3 UCRDY
* 4 HEI
* 5 DEI
* 6 - 7 reserved
*/
return ( ( ( reg ) & ( 0xFF ) ) > > 1 ) ^ ( 0x07 ) ;
}
/**
* ufshcd_get_uic_cmd_result - Get the UIC command result
* @ hba : Pointer to adapter instance
*
* This function gets the result of UIC command completion
* Returns 0 on success , non zero value on error
*/
static inline int ufshcd_get_uic_cmd_result ( struct ufs_hba * hba )
{
2013-06-26 22:39:26 +05:30
return ufshcd_readl ( hba , REG_UIC_COMMAND_ARG_2 ) &
2012-02-29 12:11:50 +05:30
MASK_UIC_COMMAND_RESULT ;
}
2013-08-31 21:40:21 +05:30
/**
* ufshcd_get_dme_attr_val - Get the value of attribute returned by UIC command
* @ hba : Pointer to adapter instance
*
* This function gets UIC command argument3
* Returns 0 on success , non zero value on error
*/
static inline u32 ufshcd_get_dme_attr_val ( struct ufs_hba * hba )
{
return ufshcd_readl ( hba , REG_UIC_COMMAND_ARG_3 ) ;
}
2012-02-29 12:11:50 +05:30
/**
2013-07-30 00:35:57 +05:30
* ufshcd_get_req_rsp - returns the TR response transaction type
2012-02-29 12:11:50 +05:30
* @ ucd_rsp_ptr : pointer to response UPIU
*/
static inline int
2013-07-30 00:35:57 +05:30
ufshcd_get_req_rsp ( struct utp_upiu_rsp * ucd_rsp_ptr )
2012-02-29 12:11:50 +05:30
{
2013-07-30 00:35:57 +05:30
return be32_to_cpu ( ucd_rsp_ptr - > header . dword_0 ) > > 24 ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_get_rsp_upiu_result - Get the result from response UPIU
* @ ucd_rsp_ptr : pointer to response UPIU
*
* This function gets the response status and scsi_status from response UPIU
* Returns the response result code .
*/
static inline int
ufshcd_get_rsp_upiu_result ( struct utp_upiu_rsp * ucd_rsp_ptr )
{
return be32_to_cpu ( ucd_rsp_ptr - > header . dword_1 ) & MASK_RSP_UPIU_RESULT ;
}
2013-08-31 21:40:19 +05:30
/*
* ufshcd_get_rsp_upiu_data_seg_len - Get the data segment length
* from response UPIU
* @ ucd_rsp_ptr : pointer to response UPIU
*
* Return the data segment length .
*/
static inline unsigned int
ufshcd_get_rsp_upiu_data_seg_len ( struct utp_upiu_rsp * ucd_rsp_ptr )
{
return be32_to_cpu ( ucd_rsp_ptr - > header . dword_2 ) &
MASK_RSP_UPIU_DATA_SEG_LEN ;
}
2013-07-30 00:35:59 +05:30
/**
* ufshcd_is_exception_event - Check if the device raised an exception event
* @ ucd_rsp_ptr : pointer to response UPIU
*
* The function checks if the device raised an exception event indicated in
* the Device Information field of response UPIU .
*
* Returns true if exception is raised , false otherwise .
*/
static inline bool ufshcd_is_exception_event ( struct utp_upiu_rsp * ucd_rsp_ptr )
{
return be32_to_cpu ( ucd_rsp_ptr - > header . dword_2 ) &
MASK_RSP_EXCEPTION_EVENT ? true : false ;
}
2012-02-29 12:11:50 +05:30
/**
2013-08-31 21:40:20 +05:30
* ufshcd_reset_intr_aggr - Reset interrupt aggregation values .
2012-02-29 12:11:50 +05:30
* @ hba : per adapter instance
*/
static inline void
2013-08-31 21:40:20 +05:30
ufshcd_reset_intr_aggr ( struct ufs_hba * hba )
2012-02-29 12:11:50 +05:30
{
2013-08-31 21:40:20 +05:30
ufshcd_writel ( hba , INT_AGGR_ENABLE |
INT_AGGR_COUNTER_AND_TIMER_RESET ,
REG_UTP_TRANSFER_REQ_INT_AGG_CONTROL ) ;
}
/**
* ufshcd_config_intr_aggr - Configure interrupt aggregation values .
* @ hba : per adapter instance
* @ cnt : Interrupt aggregation counter threshold
* @ tmout : Interrupt aggregation timeout value
*/
static inline void
ufshcd_config_intr_aggr ( struct ufs_hba * hba , u8 cnt , u8 tmout )
{
ufshcd_writel ( hba , INT_AGGR_ENABLE | INT_AGGR_PARAM_WRITE |
INT_AGGR_COUNTER_THLD_VAL ( cnt ) |
INT_AGGR_TIMEOUT_VAL ( tmout ) ,
REG_UTP_TRANSFER_REQ_INT_AGG_CONTROL ) ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_enable_run_stop_reg - Enable run - stop registers ,
* When run - stop registers are set to 1 , it indicates the
* host controller that it can process the requests
* @ hba : per adapter instance
*/
static void ufshcd_enable_run_stop_reg ( struct ufs_hba * hba )
{
2013-06-26 22:39:26 +05:30
ufshcd_writel ( hba , UTP_TASK_REQ_LIST_RUN_STOP_BIT ,
REG_UTP_TASK_REQ_LIST_RUN_STOP ) ;
ufshcd_writel ( hba , UTP_TRANSFER_REQ_LIST_RUN_STOP_BIT ,
REG_UTP_TRANSFER_REQ_LIST_RUN_STOP ) ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_hba_start - Start controller initialization sequence
* @ hba : per adapter instance
*/
static inline void ufshcd_hba_start ( struct ufs_hba * hba )
{
2013-06-26 22:39:26 +05:30
ufshcd_writel ( hba , CONTROLLER_ENABLE , REG_CONTROLLER_ENABLE ) ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_is_hba_active - Get controller state
* @ hba : per adapter instance
*
* Returns zero if controller is active , 1 otherwise
*/
static inline int ufshcd_is_hba_active ( struct ufs_hba * hba )
{
2013-06-26 22:39:26 +05:30
return ( ufshcd_readl ( hba , REG_CONTROLLER_ENABLE ) & 0x1 ) ? 0 : 1 ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_send_command - Send SCSI or device management commands
* @ hba : per adapter instance
* @ task_tag : Task tag of the command
*/
static inline
void ufshcd_send_command ( struct ufs_hba * hba , unsigned int task_tag )
{
__set_bit ( task_tag , & hba - > outstanding_reqs ) ;
2013-06-26 22:39:26 +05:30
ufshcd_writel ( hba , 1 < < task_tag , REG_UTP_TRANSFER_REQ_DOOR_BELL ) ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_copy_sense_data - Copy sense data in case of check condition
* @ lrb - pointer to local reference block
*/
static inline void ufshcd_copy_sense_data ( struct ufshcd_lrb * lrbp )
{
int len ;
2013-08-31 21:40:19 +05:30
if ( lrbp - > sense_buffer & &
ufshcd_get_rsp_upiu_data_seg_len ( lrbp - > ucd_rsp_ptr ) ) {
2013-07-30 00:35:57 +05:30
len = be16_to_cpu ( lrbp - > ucd_rsp_ptr - > sr . sense_data_len ) ;
2012-02-29 12:11:50 +05:30
memcpy ( lrbp - > sense_buffer ,
2013-07-30 00:35:57 +05:30
lrbp - > ucd_rsp_ptr - > sr . sense_data ,
2012-02-29 12:11:50 +05:30
min_t ( int , len , SCSI_SENSE_BUFFERSIZE ) ) ;
}
}
2013-07-30 00:35:58 +05:30
/**
* ufshcd_copy_query_response ( ) - Copy the Query Response and the data
* descriptor
* @ hba : per adapter instance
* @ lrb - pointer to local reference block
*/
static
2014-06-29 09:40:18 +03:00
int ufshcd_copy_query_response ( struct ufs_hba * hba , struct ufshcd_lrb * lrbp )
2013-07-30 00:35:58 +05:30
{
struct ufs_query_res * query_res = & hba - > dev_cmd . query . response ;
memcpy ( & query_res - > upiu_res , & lrbp - > ucd_rsp_ptr - > qr , QUERY_OSF_SIZE ) ;
/* Get the descriptor */
if ( lrbp - > ucd_rsp_ptr - > qr . opcode = = UPIU_QUERY_OPCODE_READ_DESC ) {
2014-06-29 09:40:17 +03:00
u8 * descp = ( u8 * ) lrbp - > ucd_rsp_ptr +
2013-07-30 00:35:58 +05:30
GENERAL_UPIU_REQUEST_SIZE ;
2014-06-29 09:40:18 +03:00
u16 resp_len ;
u16 buf_len ;
2013-07-30 00:35:58 +05:30
/* data segment length */
2014-06-29 09:40:18 +03:00
resp_len = be32_to_cpu ( lrbp - > ucd_rsp_ptr - > header . dword_2 ) &
2013-07-30 00:35:58 +05:30
MASK_QUERY_DATA_SEG_LEN ;
2014-07-23 09:31:12 +03:00
buf_len = be16_to_cpu (
hba - > dev_cmd . query . request . upiu_req . length ) ;
2014-06-29 09:40:18 +03:00
if ( likely ( buf_len > = resp_len ) ) {
memcpy ( hba - > dev_cmd . query . descriptor , descp , resp_len ) ;
} else {
dev_warn ( hba - > dev ,
" %s: Response size is bigger than buffer " ,
__func__ ) ;
return - EINVAL ;
}
2013-07-30 00:35:58 +05:30
}
2014-06-29 09:40:18 +03:00
return 0 ;
2013-07-30 00:35:58 +05:30
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_hba_capabilities - Read controller capabilities
* @ hba : per adapter instance
*/
static inline void ufshcd_hba_capabilities ( struct ufs_hba * hba )
{
2013-06-26 22:39:26 +05:30
hba - > capabilities = ufshcd_readl ( hba , REG_CONTROLLER_CAPABILITIES ) ;
2012-02-29 12:11:50 +05:30
/* nutrs and nutmrs are 0 based values */
hba - > nutrs = ( hba - > capabilities & MASK_TRANSFER_REQUESTS_SLOTS ) + 1 ;
hba - > nutmrs =
( ( hba - > capabilities & MASK_TASK_MANAGEMENT_REQUEST_SLOTS ) > > 16 ) + 1 ;
}
/**
2013-06-26 22:39:29 +05:30
* ufshcd_ready_for_uic_cmd - Check if controller is ready
* to accept UIC commands
2012-02-29 12:11:50 +05:30
* @ hba : per adapter instance
2013-06-26 22:39:29 +05:30
* Return true on success , else false
*/
static inline bool ufshcd_ready_for_uic_cmd ( struct ufs_hba * hba )
{
if ( ufshcd_readl ( hba , REG_CONTROLLER_STATUS ) & UIC_COMMAND_READY )
return true ;
else
return false ;
}
2013-08-31 21:40:22 +05:30
/**
* ufshcd_get_upmcrs - Get the power mode change request status
* @ hba : Pointer to adapter instance
*
* This function gets the UPMCRS field of HCS register
* Returns value of UPMCRS field
*/
static inline u8 ufshcd_get_upmcrs ( struct ufs_hba * hba )
{
return ( ufshcd_readl ( hba , REG_CONTROLLER_STATUS ) > > 8 ) & 0x7 ;
}
2013-06-26 22:39:29 +05:30
/**
* ufshcd_dispatch_uic_cmd - Dispatch UIC commands to unipro layers
* @ hba : per adapter instance
* @ uic_cmd : UIC command
*
* Mutex must be held .
2012-02-29 12:11:50 +05:30
*/
static inline void
2013-06-26 22:39:29 +05:30
ufshcd_dispatch_uic_cmd ( struct ufs_hba * hba , struct uic_command * uic_cmd )
2012-02-29 12:11:50 +05:30
{
2013-06-26 22:39:29 +05:30
WARN_ON ( hba - > active_uic_cmd ) ;
hba - > active_uic_cmd = uic_cmd ;
2012-02-29 12:11:50 +05:30
/* Write Args */
2013-06-26 22:39:29 +05:30
ufshcd_writel ( hba , uic_cmd - > argument1 , REG_UIC_COMMAND_ARG_1 ) ;
ufshcd_writel ( hba , uic_cmd - > argument2 , REG_UIC_COMMAND_ARG_2 ) ;
ufshcd_writel ( hba , uic_cmd - > argument3 , REG_UIC_COMMAND_ARG_3 ) ;
2012-02-29 12:11:50 +05:30
/* Write UIC Cmd */
2013-06-26 22:39:29 +05:30
ufshcd_writel ( hba , uic_cmd - > command & COMMAND_OPCODE_MASK ,
2013-06-26 22:39:26 +05:30
REG_UIC_COMMAND ) ;
2012-02-29 12:11:50 +05:30
}
2013-06-26 22:39:29 +05:30
/**
* ufshcd_wait_for_uic_cmd - Wait complectioin of UIC command
* @ hba : per adapter instance
* @ uic_command : UIC command
*
* Must be called with mutex held .
* Returns 0 only if success .
*/
static int
ufshcd_wait_for_uic_cmd ( struct ufs_hba * hba , struct uic_command * uic_cmd )
{
int ret ;
unsigned long flags ;
if ( wait_for_completion_timeout ( & uic_cmd - > done ,
msecs_to_jiffies ( UIC_CMD_TIMEOUT ) ) )
ret = uic_cmd - > argument2 & MASK_UIC_COMMAND_RESULT ;
else
ret = - ETIMEDOUT ;
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
hba - > active_uic_cmd = NULL ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
return ret ;
}
/**
* __ufshcd_send_uic_cmd - Send UIC commands and retrieve the result
* @ hba : per adapter instance
* @ uic_cmd : UIC command
*
* Identical to ufshcd_send_uic_cmd ( ) expect mutex . Must be called
* with mutex held .
* Returns 0 only if success .
*/
static int
__ufshcd_send_uic_cmd ( struct ufs_hba * hba , struct uic_command * uic_cmd )
{
int ret ;
unsigned long flags ;
if ( ! ufshcd_ready_for_uic_cmd ( hba ) ) {
dev_err ( hba - > dev ,
" Controller not ready to accept UIC commands \n " ) ;
return - EIO ;
}
init_completion ( & uic_cmd - > done ) ;
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
ufshcd_dispatch_uic_cmd ( hba , uic_cmd ) ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
ret = ufshcd_wait_for_uic_cmd ( hba , uic_cmd ) ;
return ret ;
}
/**
* ufshcd_send_uic_cmd - Send UIC commands and retrieve the result
* @ hba : per adapter instance
* @ uic_cmd : UIC command
*
* Returns 0 only if success .
*/
static int
ufshcd_send_uic_cmd ( struct ufs_hba * hba , struct uic_command * uic_cmd )
{
int ret ;
mutex_lock ( & hba - > uic_cmd_mutex ) ;
ret = __ufshcd_send_uic_cmd ( hba , uic_cmd ) ;
mutex_unlock ( & hba - > uic_cmd_mutex ) ;
return ret ;
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_map_sg - Map scatter - gather list to prdt
* @ lrbp - pointer to local reference block
*
* Returns 0 in case of success , non - zero value in case of failure
*/
static int ufshcd_map_sg ( struct ufshcd_lrb * lrbp )
{
struct ufshcd_sg_entry * prd_table ;
struct scatterlist * sg ;
struct scsi_cmnd * cmd ;
int sg_segments ;
int i ;
cmd = lrbp - > cmd ;
sg_segments = scsi_dma_map ( cmd ) ;
if ( sg_segments < 0 )
return sg_segments ;
if ( sg_segments ) {
lrbp - > utr_descriptor_ptr - > prd_table_length =
cpu_to_le16 ( ( u16 ) ( sg_segments ) ) ;
prd_table = ( struct ufshcd_sg_entry * ) lrbp - > ucd_prdt_ptr ;
scsi_for_each_sg ( cmd , sg , sg_segments , i ) {
prd_table [ i ] . size =
cpu_to_le32 ( ( ( u32 ) sg_dma_len ( sg ) ) - 1 ) ;
prd_table [ i ] . base_addr =
cpu_to_le32 ( lower_32_bits ( sg - > dma_address ) ) ;
prd_table [ i ] . upper_addr =
cpu_to_le32 ( upper_32_bits ( sg - > dma_address ) ) ;
}
} else {
lrbp - > utr_descriptor_ptr - > prd_table_length = 0 ;
}
return 0 ;
}
/**
2013-06-26 22:39:27 +05:30
* ufshcd_enable_intr - enable interrupts
2012-02-29 12:11:50 +05:30
* @ hba : per adapter instance
2013-06-26 22:39:27 +05:30
* @ intrs : interrupt bits
2012-02-29 12:11:50 +05:30
*/
2013-06-26 22:39:27 +05:30
static void ufshcd_enable_intr ( struct ufs_hba * hba , u32 intrs )
2012-02-29 12:11:50 +05:30
{
2013-06-26 22:39:27 +05:30
u32 set = ufshcd_readl ( hba , REG_INTERRUPT_ENABLE ) ;
if ( hba - > ufs_version = = UFSHCI_VERSION_10 ) {
u32 rw ;
rw = set & INTERRUPT_MASK_RW_VER_10 ;
set = rw | ( ( set ^ intrs ) & intrs ) ;
} else {
set | = intrs ;
}
ufshcd_writel ( hba , set , REG_INTERRUPT_ENABLE ) ;
}
/**
* ufshcd_disable_intr - disable interrupts
* @ hba : per adapter instance
* @ intrs : interrupt bits
*/
static void ufshcd_disable_intr ( struct ufs_hba * hba , u32 intrs )
{
u32 set = ufshcd_readl ( hba , REG_INTERRUPT_ENABLE ) ;
if ( hba - > ufs_version = = UFSHCI_VERSION_10 ) {
u32 rw ;
rw = ( set & INTERRUPT_MASK_RW_VER_10 ) &
~ ( intrs & INTERRUPT_MASK_RW_VER_10 ) ;
set = rw | ( ( set & intrs ) & ~ INTERRUPT_MASK_RW_VER_10 ) ;
} else {
set & = ~ intrs ;
2012-02-29 12:11:50 +05:30
}
2013-06-26 22:39:27 +05:30
ufshcd_writel ( hba , set , REG_INTERRUPT_ENABLE ) ;
2012-02-29 12:11:50 +05:30
}
2013-07-30 00:35:57 +05:30
/**
* ufshcd_prepare_req_desc_hdr ( ) - Fills the requests header
* descriptor according to request
* @ lrbp : pointer to local reference block
* @ upiu_flags : flags required in the header
* @ cmd_dir : requests data direction
*/
static void ufshcd_prepare_req_desc_hdr ( struct ufshcd_lrb * lrbp ,
u32 * upiu_flags , enum dma_data_direction cmd_dir )
{
struct utp_transfer_req_desc * req_desc = lrbp - > utr_descriptor_ptr ;
u32 data_direction ;
u32 dword_0 ;
if ( cmd_dir = = DMA_FROM_DEVICE ) {
data_direction = UTP_DEVICE_TO_HOST ;
* upiu_flags = UPIU_CMD_FLAGS_READ ;
} else if ( cmd_dir = = DMA_TO_DEVICE ) {
data_direction = UTP_HOST_TO_DEVICE ;
* upiu_flags = UPIU_CMD_FLAGS_WRITE ;
} else {
data_direction = UTP_NO_DATA_TRANSFER ;
* upiu_flags = UPIU_CMD_FLAGS_NONE ;
}
dword_0 = data_direction | ( lrbp - > command_type
< < UPIU_COMMAND_TYPE_OFFSET ) ;
if ( lrbp - > intr_cmd )
dword_0 | = UTP_REQ_DESC_INT_CMD ;
/* Transfer request descriptor header fields */
req_desc - > header . dword_0 = cpu_to_le32 ( dword_0 ) ;
/*
* assigning invalid value for command status . Controller
* updates OCS on command completion , with the command
* status
*/
req_desc - > header . dword_2 =
cpu_to_le32 ( OCS_INVALID_COMMAND_STATUS ) ;
}
/**
* ufshcd_prepare_utp_scsi_cmd_upiu ( ) - fills the utp_transfer_req_desc ,
* for scsi commands
* @ lrbp - local reference block pointer
* @ upiu_flags - flags
*/
static
void ufshcd_prepare_utp_scsi_cmd_upiu ( struct ufshcd_lrb * lrbp , u32 upiu_flags )
{
struct utp_upiu_req * ucd_req_ptr = lrbp - > ucd_req_ptr ;
/* command descriptor fields */
ucd_req_ptr - > header . dword_0 = UPIU_HEADER_DWORD (
UPIU_TRANSACTION_COMMAND , upiu_flags ,
lrbp - > lun , lrbp - > task_tag ) ;
ucd_req_ptr - > header . dword_1 = UPIU_HEADER_DWORD (
UPIU_COMMAND_SET_TYPE_SCSI , 0 , 0 , 0 ) ;
/* Total EHS length and Data segment length will be zero */
ucd_req_ptr - > header . dword_2 = 0 ;
ucd_req_ptr - > sc . exp_data_transfer_len =
cpu_to_be32 ( lrbp - > cmd - > sdb . length ) ;
memcpy ( ucd_req_ptr - > sc . cdb , lrbp - > cmd - > cmnd ,
( min_t ( unsigned short , lrbp - > cmd - > cmd_len , MAX_CDB_SIZE ) ) ) ;
}
2013-07-30 00:35:58 +05:30
/**
* ufshcd_prepare_utp_query_req_upiu ( ) - fills the utp_transfer_req_desc ,
* for query requsts
* @ hba : UFS hba
* @ lrbp : local reference block pointer
* @ upiu_flags : flags
*/
static void ufshcd_prepare_utp_query_req_upiu ( struct ufs_hba * hba ,
struct ufshcd_lrb * lrbp , u32 upiu_flags )
{
struct utp_upiu_req * ucd_req_ptr = lrbp - > ucd_req_ptr ;
struct ufs_query * query = & hba - > dev_cmd . query ;
2014-05-26 10:59:10 +05:30
u16 len = be16_to_cpu ( query - > request . upiu_req . length ) ;
2013-07-30 00:35:58 +05:30
u8 * descp = ( u8 * ) lrbp - > ucd_req_ptr + GENERAL_UPIU_REQUEST_SIZE ;
/* Query request header */
ucd_req_ptr - > header . dword_0 = UPIU_HEADER_DWORD (
UPIU_TRANSACTION_QUERY_REQ , upiu_flags ,
lrbp - > lun , lrbp - > task_tag ) ;
ucd_req_ptr - > header . dword_1 = UPIU_HEADER_DWORD (
0 , query - > request . query_func , 0 , 0 ) ;
/* Data segment length */
ucd_req_ptr - > header . dword_2 = UPIU_HEADER_DWORD (
0 , 0 , len > > 8 , ( u8 ) len ) ;
/* Copy the Query Request buffer as is */
memcpy ( & ucd_req_ptr - > qr , & query - > request . upiu_req ,
QUERY_OSF_SIZE ) ;
/* Copy the Descriptor */
2014-06-29 09:40:18 +03:00
if ( query - > request . upiu_req . opcode = = UPIU_QUERY_OPCODE_WRITE_DESC )
memcpy ( descp , query - > descriptor , len ) ;
2013-07-30 00:35:58 +05:30
}
2013-07-30 00:35:57 +05:30
static inline void ufshcd_prepare_utp_nop_upiu ( struct ufshcd_lrb * lrbp )
{
struct utp_upiu_req * ucd_req_ptr = lrbp - > ucd_req_ptr ;
memset ( ucd_req_ptr , 0 , sizeof ( struct utp_upiu_req ) ) ;
/* command descriptor fields */
ucd_req_ptr - > header . dword_0 =
UPIU_HEADER_DWORD (
UPIU_TRANSACTION_NOP_OUT , 0 , 0 , lrbp - > task_tag ) ;
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_compose_upiu - form UFS Protocol Information Unit ( UPIU )
2013-07-30 00:35:57 +05:30
* @ hba - per adapter instance
2012-02-29 12:11:50 +05:30
* @ lrb - pointer to local reference block
*/
2013-07-30 00:35:57 +05:30
static int ufshcd_compose_upiu ( struct ufs_hba * hba , struct ufshcd_lrb * lrbp )
2012-02-29 12:11:50 +05:30
{
u32 upiu_flags ;
2013-07-30 00:35:57 +05:30
int ret = 0 ;
2012-02-29 12:11:50 +05:30
switch ( lrbp - > command_type ) {
case UTP_CMD_TYPE_SCSI :
2013-07-30 00:35:57 +05:30
if ( likely ( lrbp - > cmd ) ) {
ufshcd_prepare_req_desc_hdr ( lrbp , & upiu_flags ,
lrbp - > cmd - > sc_data_direction ) ;
ufshcd_prepare_utp_scsi_cmd_upiu ( lrbp , upiu_flags ) ;
2012-02-29 12:11:50 +05:30
} else {
2013-07-30 00:35:57 +05:30
ret = - EINVAL ;
2012-02-29 12:11:50 +05:30
}
break ;
case UTP_CMD_TYPE_DEV_MANAGE :
2013-07-30 00:35:57 +05:30
ufshcd_prepare_req_desc_hdr ( lrbp , & upiu_flags , DMA_NONE ) ;
2013-07-30 00:35:58 +05:30
if ( hba - > dev_cmd . type = = DEV_CMD_TYPE_QUERY )
ufshcd_prepare_utp_query_req_upiu (
hba , lrbp , upiu_flags ) ;
else if ( hba - > dev_cmd . type = = DEV_CMD_TYPE_NOP )
2013-07-30 00:35:57 +05:30
ufshcd_prepare_utp_nop_upiu ( lrbp ) ;
else
ret = - EINVAL ;
2012-02-29 12:11:50 +05:30
break ;
case UTP_CMD_TYPE_UFS :
/* For UFS native command implementation */
2013-07-30 00:35:57 +05:30
ret = - ENOTSUPP ;
dev_err ( hba - > dev , " %s: UFS native command are not supported \n " ,
__func__ ) ;
break ;
default :
ret = - ENOTSUPP ;
dev_err ( hba - > dev , " %s: unknown command type: 0x%x \n " ,
__func__ , lrbp - > command_type ) ;
2012-02-29 12:11:50 +05:30
break ;
} /* end of switch */
2013-07-30 00:35:57 +05:30
return ret ;
2012-02-29 12:11:50 +05:30
}
2014-09-25 15:32:29 +03:00
/*
* ufshcd_scsi_to_upiu_lun - maps scsi LUN to UPIU LUN
* @ scsi_lun : scsi LUN id
*
* Returns UPIU LUN id
*/
static inline u8 ufshcd_scsi_to_upiu_lun ( unsigned int scsi_lun )
{
if ( scsi_is_wlun ( scsi_lun ) )
return ( scsi_lun & UFS_UPIU_MAX_UNIT_NUM_ID )
| UFS_UPIU_WLUN_ID ;
else
return scsi_lun & UFS_UPIU_MAX_UNIT_NUM_ID ;
}
2014-09-25 15:32:28 +03:00
/**
* ufshcd_upiu_wlun_to_scsi_wlun - maps UPIU W - LUN id to SCSI W - LUN ID
* @ scsi_lun : UPIU W - LUN id
*
* Returns SCSI W - LUN id
*/
static inline u16 ufshcd_upiu_wlun_to_scsi_wlun ( u8 upiu_wlun_id )
{
return ( upiu_wlun_id & ~ UFS_UPIU_WLUN_ID ) | SCSI_W_LUN_BASE ;
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_queuecommand - main entry point for SCSI requests
* @ cmd : command from SCSI Midlayer
* @ done : call back function
*
* Returns 0 for success , non - zero in case of failure
*/
static int ufshcd_queuecommand ( struct Scsi_Host * host , struct scsi_cmnd * cmd )
{
struct ufshcd_lrb * lrbp ;
struct ufs_hba * hba ;
unsigned long flags ;
int tag ;
int err = 0 ;
hba = shost_priv ( host ) ;
tag = cmd - > request - > tag ;
2014-05-26 10:59:14 +05:30
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
switch ( hba - > ufshcd_state ) {
case UFSHCD_STATE_OPERATIONAL :
break ;
case UFSHCD_STATE_RESET :
2012-02-29 12:11:50 +05:30
err = SCSI_MLQUEUE_HOST_BUSY ;
2014-05-26 10:59:14 +05:30
goto out_unlock ;
case UFSHCD_STATE_ERROR :
set_host_byte ( cmd , DID_ERROR ) ;
cmd - > scsi_done ( cmd ) ;
goto out_unlock ;
default :
dev_WARN_ONCE ( hba - > dev , 1 , " %s: invalid state %d \n " ,
__func__ , hba - > ufshcd_state ) ;
set_host_byte ( cmd , DID_BAD_TARGET ) ;
cmd - > scsi_done ( cmd ) ;
goto out_unlock ;
2012-02-29 12:11:50 +05:30
}
2014-05-26 10:59:14 +05:30
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
2012-02-29 12:11:50 +05:30
2013-07-30 00:35:57 +05:30
/* acquire the tag to make sure device cmds don't use it */
if ( test_and_set_bit_lock ( tag , & hba - > lrb_in_use ) ) {
/*
* Dev manage command in progress , requeue the command .
* Requeuing the command helps in cases where the request * may *
* find different tag instead of waiting for dev manage command
* completion .
*/
err = SCSI_MLQUEUE_HOST_BUSY ;
goto out ;
}
2012-02-29 12:11:50 +05:30
lrbp = & hba - > lrb [ tag ] ;
2013-07-30 00:35:57 +05:30
WARN_ON ( lrbp - > cmd ) ;
2012-02-29 12:11:50 +05:30
lrbp - > cmd = cmd ;
lrbp - > sense_bufflen = SCSI_SENSE_BUFFERSIZE ;
lrbp - > sense_buffer = cmd - > sense_buffer ;
lrbp - > task_tag = tag ;
2014-09-25 15:32:29 +03:00
lrbp - > lun = ufshcd_scsi_to_upiu_lun ( cmd - > device - > lun ) ;
2013-07-30 00:35:57 +05:30
lrbp - > intr_cmd = false ;
2012-02-29 12:11:50 +05:30
lrbp - > command_type = UTP_CMD_TYPE_SCSI ;
/* form UPIU before issuing the command */
2013-07-30 00:35:57 +05:30
ufshcd_compose_upiu ( hba , lrbp ) ;
2012-02-29 12:11:50 +05:30
err = ufshcd_map_sg ( lrbp ) ;
2013-07-30 00:35:57 +05:30
if ( err ) {
lrbp - > cmd = NULL ;
clear_bit_unlock ( tag , & hba - > lrb_in_use ) ;
2012-02-29 12:11:50 +05:30
goto out ;
2013-07-30 00:35:57 +05:30
}
2012-02-29 12:11:50 +05:30
/* issue command to the controller */
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
ufshcd_send_command ( hba , tag ) ;
2014-05-26 10:59:14 +05:30
out_unlock :
2012-02-29 12:11:50 +05:30
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
out :
return err ;
}
2013-07-30 00:35:57 +05:30
static int ufshcd_compose_dev_cmd ( struct ufs_hba * hba ,
struct ufshcd_lrb * lrbp , enum dev_cmd_type cmd_type , int tag )
{
lrbp - > cmd = NULL ;
lrbp - > sense_bufflen = 0 ;
lrbp - > sense_buffer = NULL ;
lrbp - > task_tag = tag ;
lrbp - > lun = 0 ; /* device management cmd is not specific to any LUN */
lrbp - > command_type = UTP_CMD_TYPE_DEV_MANAGE ;
lrbp - > intr_cmd = true ; /* No interrupt aggregation */
hba - > dev_cmd . type = cmd_type ;
return ufshcd_compose_upiu ( hba , lrbp ) ;
}
static int
ufshcd_clear_cmd ( struct ufs_hba * hba , int tag )
{
int err = 0 ;
unsigned long flags ;
u32 mask = 1 < < tag ;
/* clear outstanding transaction before retry */
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
ufshcd_utrl_clear ( hba , tag ) ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
/*
* wait for for h / w to clear corresponding bit in door - bell .
* max . wait is 1 sec .
*/
err = ufshcd_wait_for_register ( hba ,
REG_UTP_TRANSFER_REQ_DOOR_BELL ,
mask , ~ mask , 1000 , 1000 ) ;
return err ;
}
2014-06-29 09:40:18 +03:00
static int
ufshcd_check_query_response ( struct ufs_hba * hba , struct ufshcd_lrb * lrbp )
{
struct ufs_query_res * query_res = & hba - > dev_cmd . query . response ;
/* Get the UPIU response */
query_res - > response = ufshcd_get_rsp_upiu_result ( lrbp - > ucd_rsp_ptr ) > >
UPIU_RSP_CODE_OFFSET ;
return query_res - > response ;
}
2013-07-30 00:35:57 +05:30
/**
* ufshcd_dev_cmd_completion ( ) - handles device management command responses
* @ hba : per adapter instance
* @ lrbp : pointer to local reference block
*/
static int
ufshcd_dev_cmd_completion ( struct ufs_hba * hba , struct ufshcd_lrb * lrbp )
{
int resp ;
int err = 0 ;
resp = ufshcd_get_req_rsp ( lrbp - > ucd_rsp_ptr ) ;
switch ( resp ) {
case UPIU_TRANSACTION_NOP_IN :
if ( hba - > dev_cmd . type ! = DEV_CMD_TYPE_NOP ) {
err = - EINVAL ;
dev_err ( hba - > dev , " %s: unexpected response %x \n " ,
__func__ , resp ) ;
}
break ;
2013-07-30 00:35:58 +05:30
case UPIU_TRANSACTION_QUERY_RSP :
2014-06-29 09:40:18 +03:00
err = ufshcd_check_query_response ( hba , lrbp ) ;
if ( ! err )
err = ufshcd_copy_query_response ( hba , lrbp ) ;
2013-07-30 00:35:58 +05:30
break ;
2013-07-30 00:35:57 +05:30
case UPIU_TRANSACTION_REJECT_UPIU :
/* TODO: handle Reject UPIU Response */
err = - EPERM ;
dev_err ( hba - > dev , " %s: Reject UPIU not fully implemented \n " ,
__func__ ) ;
break ;
default :
err = - EINVAL ;
dev_err ( hba - > dev , " %s: Invalid device management cmd response: %x \n " ,
__func__ , resp ) ;
break ;
}
return err ;
}
static int ufshcd_wait_for_dev_cmd ( struct ufs_hba * hba ,
struct ufshcd_lrb * lrbp , int max_timeout )
{
int err = 0 ;
unsigned long time_left ;
unsigned long flags ;
time_left = wait_for_completion_timeout ( hba - > dev_cmd . complete ,
msecs_to_jiffies ( max_timeout ) ) ;
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
hba - > dev_cmd . complete = NULL ;
if ( likely ( time_left ) ) {
err = ufshcd_get_tr_ocs ( lrbp ) ;
if ( ! err )
err = ufshcd_dev_cmd_completion ( hba , lrbp ) ;
}
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
if ( ! time_left ) {
err = - ETIMEDOUT ;
if ( ! ufshcd_clear_cmd ( hba , lrbp - > task_tag ) )
/* sucessfully cleared the command, retry if needed */
err = - EAGAIN ;
}
return err ;
}
/**
* ufshcd_get_dev_cmd_tag - Get device management command tag
* @ hba : per - adapter instance
* @ tag : pointer to variable with available slot value
*
* Get a free slot and lock it until device management command
* completes .
*
* Returns false if free slot is unavailable for locking , else
* return true with tag value in @ tag .
*/
static bool ufshcd_get_dev_cmd_tag ( struct ufs_hba * hba , int * tag_out )
{
int tag ;
bool ret = false ;
unsigned long tmp ;
if ( ! tag_out )
goto out ;
do {
tmp = ~ hba - > lrb_in_use ;
tag = find_last_bit ( & tmp , hba - > nutrs ) ;
if ( tag > = hba - > nutrs )
goto out ;
} while ( test_and_set_bit_lock ( tag , & hba - > lrb_in_use ) ) ;
* tag_out = tag ;
ret = true ;
out :
return ret ;
}
static inline void ufshcd_put_dev_cmd_tag ( struct ufs_hba * hba , int tag )
{
clear_bit_unlock ( tag , & hba - > lrb_in_use ) ;
}
/**
* ufshcd_exec_dev_cmd - API for sending device management requests
* @ hba - UFS hba
* @ cmd_type - specifies the type ( NOP , Query . . . )
* @ timeout - time in seconds
*
2013-07-30 00:35:58 +05:30
* NOTE : Since there is only one available tag for device management commands ,
* it is expected you hold the hba - > dev_cmd . lock mutex .
2013-07-30 00:35:57 +05:30
*/
static int ufshcd_exec_dev_cmd ( struct ufs_hba * hba ,
enum dev_cmd_type cmd_type , int timeout )
{
struct ufshcd_lrb * lrbp ;
int err ;
int tag ;
struct completion wait ;
unsigned long flags ;
/*
* Get free slot , sleep if slots are unavailable .
* Even though we use wait_event ( ) which sleeps indefinitely ,
* the maximum wait time is bounded by SCSI request timeout .
*/
wait_event ( hba - > dev_cmd . tag_wq , ufshcd_get_dev_cmd_tag ( hba , & tag ) ) ;
init_completion ( & wait ) ;
lrbp = & hba - > lrb [ tag ] ;
WARN_ON ( lrbp - > cmd ) ;
err = ufshcd_compose_dev_cmd ( hba , lrbp , cmd_type , tag ) ;
if ( unlikely ( err ) )
goto out_put_tag ;
hba - > dev_cmd . complete = & wait ;
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
ufshcd_send_command ( hba , tag ) ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
err = ufshcd_wait_for_dev_cmd ( hba , lrbp , timeout ) ;
out_put_tag :
ufshcd_put_dev_cmd_tag ( hba , tag ) ;
wake_up ( & hba - > dev_cmd . tag_wq ) ;
return err ;
}
2014-06-29 09:40:17 +03:00
/**
* ufshcd_init_query ( ) - init the query response and request parameters
* @ hba : per - adapter instance
* @ request : address of the request pointer to be initialized
* @ response : address of the response pointer to be initialized
* @ opcode : operation to perform
* @ idn : flag idn to access
* @ index : LU number to access
* @ selector : query / flag / descriptor further identification
*/
static inline void ufshcd_init_query ( struct ufs_hba * hba ,
struct ufs_query_req * * request , struct ufs_query_res * * response ,
enum query_opcode opcode , u8 idn , u8 index , u8 selector )
{
* request = & hba - > dev_cmd . query . request ;
* response = & hba - > dev_cmd . query . response ;
memset ( * request , 0 , sizeof ( struct ufs_query_req ) ) ;
memset ( * response , 0 , sizeof ( struct ufs_query_res ) ) ;
( * request ) - > upiu_req . opcode = opcode ;
( * request ) - > upiu_req . idn = idn ;
( * request ) - > upiu_req . index = index ;
( * request ) - > upiu_req . selector = selector ;
}
2013-07-30 00:35:58 +05:30
/**
* ufshcd_query_flag ( ) - API function for sending flag query requests
* hba : per - adapter instance
* query_opcode : flag query to perform
* idn : flag idn to access
* flag_res : the flag value after the query request completes
*
* Returns 0 for success , non - zero in case of failure
*/
static int ufshcd_query_flag ( struct ufs_hba * hba , enum query_opcode opcode ,
enum flag_idn idn , bool * flag_res )
{
2014-06-29 09:40:17 +03:00
struct ufs_query_req * request = NULL ;
struct ufs_query_res * response = NULL ;
int err , index = 0 , selector = 0 ;
2013-07-30 00:35:58 +05:30
BUG_ON ( ! hba ) ;
mutex_lock ( & hba - > dev_cmd . lock ) ;
2014-06-29 09:40:17 +03:00
ufshcd_init_query ( hba , & request , & response , opcode , idn , index ,
selector ) ;
2013-07-30 00:35:58 +05:30
switch ( opcode ) {
case UPIU_QUERY_OPCODE_SET_FLAG :
case UPIU_QUERY_OPCODE_CLEAR_FLAG :
case UPIU_QUERY_OPCODE_TOGGLE_FLAG :
request - > query_func = UPIU_QUERY_FUNC_STANDARD_WRITE_REQUEST ;
break ;
case UPIU_QUERY_OPCODE_READ_FLAG :
request - > query_func = UPIU_QUERY_FUNC_STANDARD_READ_REQUEST ;
if ( ! flag_res ) {
/* No dummy reads */
dev_err ( hba - > dev , " %s: Invalid argument for read request \n " ,
__func__ ) ;
err = - EINVAL ;
goto out_unlock ;
}
break ;
default :
dev_err ( hba - > dev ,
" %s: Expected query flag opcode but got = %d \n " ,
__func__ , opcode ) ;
err = - EINVAL ;
goto out_unlock ;
}
2014-06-29 09:40:17 +03:00
err = ufshcd_exec_dev_cmd ( hba , DEV_CMD_TYPE_QUERY , QUERY_REQ_TIMEOUT ) ;
2013-07-30 00:35:58 +05:30
if ( err ) {
dev_err ( hba - > dev ,
" %s: Sending flag query for idn %d failed, err = %d \n " ,
__func__ , idn , err ) ;
goto out_unlock ;
}
if ( flag_res )
2014-05-26 10:59:10 +05:30
* flag_res = ( be32_to_cpu ( response - > upiu_res . value ) &
2013-07-30 00:35:58 +05:30
MASK_QUERY_UPIU_FLAG_LOC ) & 0x1 ;
out_unlock :
mutex_unlock ( & hba - > dev_cmd . lock ) ;
return err ;
}
2013-07-30 00:35:59 +05:30
/**
* ufshcd_query_attr - API function for sending attribute requests
* hba : per - adapter instance
* opcode : attribute opcode
* idn : attribute idn to access
* index : index field
* selector : selector field
* attr_val : the attribute value after the query request completes
*
* Returns 0 for success , non - zero in case of failure
*/
2014-05-26 10:59:11 +05:30
static int ufshcd_query_attr ( struct ufs_hba * hba , enum query_opcode opcode ,
2013-07-30 00:35:59 +05:30
enum attr_idn idn , u8 index , u8 selector , u32 * attr_val )
{
2014-06-29 09:40:17 +03:00
struct ufs_query_req * request = NULL ;
struct ufs_query_res * response = NULL ;
2013-07-30 00:35:59 +05:30
int err ;
BUG_ON ( ! hba ) ;
if ( ! attr_val ) {
dev_err ( hba - > dev , " %s: attribute value required for opcode 0x%x \n " ,
__func__ , opcode ) ;
err = - EINVAL ;
goto out ;
}
mutex_lock ( & hba - > dev_cmd . lock ) ;
2014-06-29 09:40:17 +03:00
ufshcd_init_query ( hba , & request , & response , opcode , idn , index ,
selector ) ;
2013-07-30 00:35:59 +05:30
switch ( opcode ) {
case UPIU_QUERY_OPCODE_WRITE_ATTR :
request - > query_func = UPIU_QUERY_FUNC_STANDARD_WRITE_REQUEST ;
2014-05-26 10:59:10 +05:30
request - > upiu_req . value = cpu_to_be32 ( * attr_val ) ;
2013-07-30 00:35:59 +05:30
break ;
case UPIU_QUERY_OPCODE_READ_ATTR :
request - > query_func = UPIU_QUERY_FUNC_STANDARD_READ_REQUEST ;
break ;
default :
dev_err ( hba - > dev , " %s: Expected query attr opcode but got = 0x%.2x \n " ,
__func__ , opcode ) ;
err = - EINVAL ;
goto out_unlock ;
}
2014-06-29 09:40:17 +03:00
err = ufshcd_exec_dev_cmd ( hba , DEV_CMD_TYPE_QUERY , QUERY_REQ_TIMEOUT ) ;
2013-07-30 00:35:59 +05:30
if ( err ) {
dev_err ( hba - > dev , " %s: opcode 0x%.2x for idn %d failed, err = %d \n " ,
__func__ , opcode , idn , err ) ;
goto out_unlock ;
}
2014-05-26 10:59:10 +05:30
* attr_val = be32_to_cpu ( response - > upiu_res . value ) ;
2013-07-30 00:35:59 +05:30
out_unlock :
mutex_unlock ( & hba - > dev_cmd . lock ) ;
out :
return err ;
}
2014-06-29 09:40:17 +03:00
/**
* ufshcd_query_descriptor - API function for sending descriptor requests
* hba : per - adapter instance
* opcode : attribute opcode
* idn : attribute idn to access
* index : index field
* selector : selector field
* desc_buf : the buffer that contains the descriptor
* buf_len : length parameter passed to the device
*
* Returns 0 for success , non - zero in case of failure .
* The buf_len parameter will contain , on return , the length parameter
* received on the response .
*/
2014-07-23 09:31:11 +03:00
static int ufshcd_query_descriptor ( struct ufs_hba * hba ,
2014-06-29 09:40:17 +03:00
enum query_opcode opcode , enum desc_idn idn , u8 index ,
u8 selector , u8 * desc_buf , int * buf_len )
{
struct ufs_query_req * request = NULL ;
struct ufs_query_res * response = NULL ;
int err ;
BUG_ON ( ! hba ) ;
if ( ! desc_buf ) {
dev_err ( hba - > dev , " %s: descriptor buffer required for opcode 0x%x \n " ,
__func__ , opcode ) ;
err = - EINVAL ;
goto out ;
}
if ( * buf_len < = QUERY_DESC_MIN_SIZE | | * buf_len > QUERY_DESC_MAX_SIZE ) {
dev_err ( hba - > dev , " %s: descriptor buffer size (%d) is out of range \n " ,
__func__ , * buf_len ) ;
err = - EINVAL ;
goto out ;
}
mutex_lock ( & hba - > dev_cmd . lock ) ;
ufshcd_init_query ( hba , & request , & response , opcode , idn , index ,
selector ) ;
hba - > dev_cmd . query . descriptor = desc_buf ;
2014-07-23 09:31:12 +03:00
request - > upiu_req . length = cpu_to_be16 ( * buf_len ) ;
2014-06-29 09:40:17 +03:00
switch ( opcode ) {
case UPIU_QUERY_OPCODE_WRITE_DESC :
request - > query_func = UPIU_QUERY_FUNC_STANDARD_WRITE_REQUEST ;
break ;
case UPIU_QUERY_OPCODE_READ_DESC :
request - > query_func = UPIU_QUERY_FUNC_STANDARD_READ_REQUEST ;
break ;
default :
dev_err ( hba - > dev ,
" %s: Expected query descriptor opcode but got = 0x%.2x \n " ,
__func__ , opcode ) ;
err = - EINVAL ;
goto out_unlock ;
}
err = ufshcd_exec_dev_cmd ( hba , DEV_CMD_TYPE_QUERY , QUERY_REQ_TIMEOUT ) ;
if ( err ) {
dev_err ( hba - > dev , " %s: opcode 0x%.2x for idn %d failed, err = %d \n " ,
__func__ , opcode , idn , err ) ;
goto out_unlock ;
}
hba - > dev_cmd . query . descriptor = NULL ;
2014-07-23 09:31:12 +03:00
* buf_len = be16_to_cpu ( response - > upiu_res . length ) ;
2014-06-29 09:40:17 +03:00
out_unlock :
mutex_unlock ( & hba - > dev_cmd . lock ) ;
out :
return err ;
}
2014-09-25 15:32:25 +03:00
/**
* ufshcd_read_desc_param - read the specified descriptor parameter
* @ hba : Pointer to adapter instance
* @ desc_id : descriptor idn value
* @ desc_index : descriptor index
* @ param_offset : offset of the parameter to read
* @ param_read_buf : pointer to buffer where parameter would be read
* @ param_size : sizeof ( param_read_buf )
*
* Return 0 in case of success , non - zero otherwise
*/
static int ufshcd_read_desc_param ( struct ufs_hba * hba ,
enum desc_idn desc_id ,
int desc_index ,
u32 param_offset ,
u8 * param_read_buf ,
u32 param_size )
{
int ret ;
u8 * desc_buf ;
u32 buff_len ;
bool is_kmalloc = true ;
/* safety checks */
if ( desc_id > = QUERY_DESC_IDN_MAX )
return - EINVAL ;
buff_len = ufs_query_desc_max_size [ desc_id ] ;
if ( ( param_offset + param_size ) > buff_len )
return - EINVAL ;
if ( ! param_offset & & ( param_size = = buff_len ) ) {
/* memory space already available to hold full descriptor */
desc_buf = param_read_buf ;
is_kmalloc = false ;
} else {
/* allocate memory to hold full descriptor */
desc_buf = kmalloc ( buff_len , GFP_KERNEL ) ;
if ( ! desc_buf )
return - ENOMEM ;
}
ret = ufshcd_query_descriptor ( hba , UPIU_QUERY_OPCODE_READ_DESC ,
desc_id , desc_index , 0 , desc_buf ,
& buff_len ) ;
if ( ret | | ( buff_len < ufs_query_desc_max_size [ desc_id ] ) | |
( desc_buf [ QUERY_DESC_LENGTH_OFFSET ] ! =
ufs_query_desc_max_size [ desc_id ] )
| | ( desc_buf [ QUERY_DESC_DESC_TYPE_OFFSET ] ! = desc_id ) ) {
dev_err ( hba - > dev , " %s: Failed reading descriptor. desc_id %d param_offset %d buff_len %d ret %d " ,
__func__ , desc_id , param_offset , buff_len , ret ) ;
if ( ! ret )
ret = - EINVAL ;
goto out ;
}
if ( is_kmalloc )
memcpy ( param_read_buf , & desc_buf [ param_offset ] , param_size ) ;
out :
if ( is_kmalloc )
kfree ( desc_buf ) ;
return ret ;
}
static inline int ufshcd_read_desc ( struct ufs_hba * hba ,
enum desc_idn desc_id ,
int desc_index ,
u8 * buf ,
u32 size )
{
return ufshcd_read_desc_param ( hba , desc_id , desc_index , 0 , buf , size ) ;
}
static inline int ufshcd_read_power_desc ( struct ufs_hba * hba ,
u8 * buf ,
u32 size )
{
return ufshcd_read_desc ( hba , QUERY_DESC_IDN_POWER , 0 , buf , size ) ;
}
/**
* ufshcd_read_unit_desc_param - read the specified unit descriptor parameter
* @ hba : Pointer to adapter instance
* @ lun : lun id
* @ param_offset : offset of the parameter to read
* @ param_read_buf : pointer to buffer where parameter would be read
* @ param_size : sizeof ( param_read_buf )
*
* Return 0 in case of success , non - zero otherwise
*/
static inline int ufshcd_read_unit_desc_param ( struct ufs_hba * hba ,
int lun ,
enum unit_desc_param param_offset ,
u8 * param_read_buf ,
u32 param_size )
{
/*
* Unit descriptors are only available for general purpose LUs ( LUN id
* from 0 to 7 ) and RPMB Well known LU .
*/
2014-09-25 15:32:29 +03:00
if ( lun ! = UFS_UPIU_RPMB_WLUN & & ( lun > = UFS_UPIU_MAX_GENERAL_LUN ) )
2014-09-25 15:32:25 +03:00
return - EOPNOTSUPP ;
return ufshcd_read_desc_param ( hba , QUERY_DESC_IDN_UNIT , lun ,
param_offset , param_read_buf , param_size ) ;
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_memory_alloc - allocate memory for host memory space data structures
* @ hba : per adapter instance
*
* 1. Allocate DMA memory for Command Descriptor array
* Each command descriptor consist of Command UPIU , Response UPIU and PRDT
* 2. Allocate DMA memory for UTP Transfer Request Descriptor List ( UTRDL ) .
* 3. Allocate DMA memory for UTP Task Management Request Descriptor List
* ( UTMRDL )
* 4. Allocate memory for local reference block ( lrb ) .
*
* Returns 0 for success , non - zero in case of failure
*/
static int ufshcd_memory_alloc ( struct ufs_hba * hba )
{
size_t utmrdl_size , utrdl_size , ucdl_size ;
/* Allocate memory for UTP command descriptors */
ucdl_size = ( sizeof ( struct utp_transfer_cmd_desc ) * hba - > nutrs ) ;
2013-06-27 13:31:54 +09:00
hba - > ucdl_base_addr = dmam_alloc_coherent ( hba - > dev ,
ucdl_size ,
& hba - > ucdl_dma_addr ,
GFP_KERNEL ) ;
2012-02-29 12:11:50 +05:30
/*
* UFSHCI requires UTP command descriptor to be 128 byte aligned .
* make sure hba - > ucdl_dma_addr is aligned to PAGE_SIZE
* if hba - > ucdl_dma_addr is aligned to PAGE_SIZE , then it will
* be aligned to 128 bytes as well
*/
if ( ! hba - > ucdl_base_addr | |
WARN_ON ( hba - > ucdl_dma_addr & ( PAGE_SIZE - 1 ) ) ) {
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev ,
2012-02-29 12:11:50 +05:30
" Command Descriptor Memory allocation failed \n " ) ;
goto out ;
}
/*
* Allocate memory for UTP Transfer descriptors
* UFSHCI requires 1024 byte alignment of UTRD
*/
utrdl_size = ( sizeof ( struct utp_transfer_req_desc ) * hba - > nutrs ) ;
2013-06-27 13:31:54 +09:00
hba - > utrdl_base_addr = dmam_alloc_coherent ( hba - > dev ,
utrdl_size ,
& hba - > utrdl_dma_addr ,
GFP_KERNEL ) ;
2012-02-29 12:11:50 +05:30
if ( ! hba - > utrdl_base_addr | |
WARN_ON ( hba - > utrdl_dma_addr & ( PAGE_SIZE - 1 ) ) ) {
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev ,
2012-02-29 12:11:50 +05:30
" Transfer Descriptor Memory allocation failed \n " ) ;
goto out ;
}
/*
* Allocate memory for UTP Task Management descriptors
* UFSHCI requires 1024 byte alignment of UTMRD
*/
utmrdl_size = sizeof ( struct utp_task_req_desc ) * hba - > nutmrs ;
2013-06-27 13:31:54 +09:00
hba - > utmrdl_base_addr = dmam_alloc_coherent ( hba - > dev ,
utmrdl_size ,
& hba - > utmrdl_dma_addr ,
GFP_KERNEL ) ;
2012-02-29 12:11:50 +05:30
if ( ! hba - > utmrdl_base_addr | |
WARN_ON ( hba - > utmrdl_dma_addr & ( PAGE_SIZE - 1 ) ) ) {
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev ,
2012-02-29 12:11:50 +05:30
" Task Management Descriptor Memory allocation failed \n " ) ;
goto out ;
}
/* Allocate memory for local reference block */
2013-06-27 13:31:54 +09:00
hba - > lrb = devm_kzalloc ( hba - > dev ,
hba - > nutrs * sizeof ( struct ufshcd_lrb ) ,
GFP_KERNEL ) ;
2012-02-29 12:11:50 +05:30
if ( ! hba - > lrb ) {
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev , " LRB Memory allocation failed \n " ) ;
2012-02-29 12:11:50 +05:30
goto out ;
}
return 0 ;
out :
return - ENOMEM ;
}
/**
* ufshcd_host_memory_configure - configure local reference block with
* memory offsets
* @ hba : per adapter instance
*
* Configure Host memory space
* 1. Update Corresponding UTRD . UCDBA and UTRD . UCDBAU with UCD DMA
* address .
* 2. Update each UTRD with Response UPIU offset , Response UPIU length
* and PRDT offset .
* 3. Save the corresponding addresses of UTRD , UCD . CMD , UCD . RSP and UCD . PRDT
* into local reference block .
*/
static void ufshcd_host_memory_configure ( struct ufs_hba * hba )
{
struct utp_transfer_cmd_desc * cmd_descp ;
struct utp_transfer_req_desc * utrdlp ;
dma_addr_t cmd_desc_dma_addr ;
dma_addr_t cmd_desc_element_addr ;
u16 response_offset ;
u16 prdt_offset ;
int cmd_desc_size ;
int i ;
utrdlp = hba - > utrdl_base_addr ;
cmd_descp = hba - > ucdl_base_addr ;
response_offset =
offsetof ( struct utp_transfer_cmd_desc , response_upiu ) ;
prdt_offset =
offsetof ( struct utp_transfer_cmd_desc , prd_table ) ;
cmd_desc_size = sizeof ( struct utp_transfer_cmd_desc ) ;
cmd_desc_dma_addr = hba - > ucdl_dma_addr ;
for ( i = 0 ; i < hba - > nutrs ; i + + ) {
/* Configure UTRD with command descriptor base address */
cmd_desc_element_addr =
( cmd_desc_dma_addr + ( cmd_desc_size * i ) ) ;
utrdlp [ i ] . command_desc_base_addr_lo =
cpu_to_le32 ( lower_32_bits ( cmd_desc_element_addr ) ) ;
utrdlp [ i ] . command_desc_base_addr_hi =
cpu_to_le32 ( upper_32_bits ( cmd_desc_element_addr ) ) ;
/* Response upiu and prdt offset should be in double words */
utrdlp [ i ] . response_upiu_offset =
cpu_to_le16 ( ( response_offset > > 2 ) ) ;
utrdlp [ i ] . prd_table_offset =
cpu_to_le16 ( ( prdt_offset > > 2 ) ) ;
utrdlp [ i ] . response_upiu_length =
2013-06-26 22:39:30 +05:30
cpu_to_le16 ( ALIGNED_UPIU_SIZE > > 2 ) ;
2012-02-29 12:11:50 +05:30
hba - > lrb [ i ] . utr_descriptor_ptr = ( utrdlp + i ) ;
2013-07-30 00:35:57 +05:30
hba - > lrb [ i ] . ucd_req_ptr =
( struct utp_upiu_req * ) ( cmd_descp + i ) ;
2012-02-29 12:11:50 +05:30
hba - > lrb [ i ] . ucd_rsp_ptr =
( struct utp_upiu_rsp * ) cmd_descp [ i ] . response_upiu ;
hba - > lrb [ i ] . ucd_prdt_ptr =
( struct ufshcd_sg_entry * ) cmd_descp [ i ] . prd_table ;
}
}
/**
* ufshcd_dme_link_startup - Notify Unipro to perform link startup
* @ hba : per adapter instance
*
* UIC_CMD_DME_LINK_STARTUP command must be issued to Unipro layer ,
* in order to initialize the Unipro link startup procedure .
* Once the Unipro links are up , the device connected to the controller
* is detected .
*
* Returns 0 on success , non - zero value on failure
*/
static int ufshcd_dme_link_startup ( struct ufs_hba * hba )
{
2013-06-26 22:39:29 +05:30
struct uic_command uic_cmd = { 0 } ;
int ret ;
2012-02-29 12:11:50 +05:30
2013-06-26 22:39:29 +05:30
uic_cmd . command = UIC_CMD_DME_LINK_STARTUP ;
2012-02-29 12:11:50 +05:30
2013-06-26 22:39:29 +05:30
ret = ufshcd_send_uic_cmd ( hba , & uic_cmd ) ;
if ( ret )
dev_err ( hba - > dev ,
" dme-link-startup: error code %d \n " , ret ) ;
return ret ;
2012-02-29 12:11:50 +05:30
}
2013-08-31 21:40:21 +05:30
/**
* ufshcd_dme_set_attr - UIC command for DME_SET , DME_PEER_SET
* @ hba : per adapter instance
* @ attr_sel : uic command argument1
* @ attr_set : attribute set type as uic command argument2
* @ mib_val : setting value as uic command argument3
* @ peer : indicate whether peer or local
*
* Returns 0 on success , non - zero value on failure
*/
int ufshcd_dme_set_attr ( struct ufs_hba * hba , u32 attr_sel ,
u8 attr_set , u32 mib_val , u8 peer )
{
struct uic_command uic_cmd = { 0 } ;
static const char * const action [ ] = {
" dme-set " ,
" dme-peer-set "
} ;
const char * set = action [ ! ! peer ] ;
int ret ;
uic_cmd . command = peer ?
UIC_CMD_DME_PEER_SET : UIC_CMD_DME_SET ;
uic_cmd . argument1 = attr_sel ;
uic_cmd . argument2 = UIC_ARG_ATTR_TYPE ( attr_set ) ;
uic_cmd . argument3 = mib_val ;
ret = ufshcd_send_uic_cmd ( hba , & uic_cmd ) ;
if ( ret )
dev_err ( hba - > dev , " %s: attr-id 0x%x val 0x%x error code %d \n " ,
set , UIC_GET_ATTR_ID ( attr_sel ) , mib_val , ret ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( ufshcd_dme_set_attr ) ;
/**
* ufshcd_dme_get_attr - UIC command for DME_GET , DME_PEER_GET
* @ hba : per adapter instance
* @ attr_sel : uic command argument1
* @ mib_val : the value of the attribute as returned by the UIC command
* @ peer : indicate whether peer or local
*
* Returns 0 on success , non - zero value on failure
*/
int ufshcd_dme_get_attr ( struct ufs_hba * hba , u32 attr_sel ,
u32 * mib_val , u8 peer )
{
struct uic_command uic_cmd = { 0 } ;
static const char * const action [ ] = {
" dme-get " ,
" dme-peer-get "
} ;
const char * get = action [ ! ! peer ] ;
int ret ;
uic_cmd . command = peer ?
UIC_CMD_DME_PEER_GET : UIC_CMD_DME_GET ;
uic_cmd . argument1 = attr_sel ;
ret = ufshcd_send_uic_cmd ( hba , & uic_cmd ) ;
if ( ret ) {
dev_err ( hba - > dev , " %s: attr-id 0x%x error code %d \n " ,
get , UIC_GET_ATTR_ID ( attr_sel ) , ret ) ;
goto out ;
}
if ( mib_val )
* mib_val = uic_cmd . argument3 ;
out :
return ret ;
}
EXPORT_SYMBOL_GPL ( ufshcd_dme_get_attr ) ;
2013-08-31 21:40:22 +05:30
/**
* ufshcd_uic_change_pwr_mode - Perform the UIC power mode chage
* using DME_SET primitives .
* @ hba : per adapter instance
* @ mode : powr mode value
*
* Returns 0 on success , non - zero value on failure
*/
2014-05-26 10:59:11 +05:30
static int ufshcd_uic_change_pwr_mode ( struct ufs_hba * hba , u8 mode )
2013-08-31 21:40:22 +05:30
{
struct uic_command uic_cmd = { 0 } ;
struct completion pwr_done ;
unsigned long flags ;
u8 status ;
int ret ;
uic_cmd . command = UIC_CMD_DME_SET ;
uic_cmd . argument1 = UIC_ARG_MIB ( PA_PWRMODE ) ;
uic_cmd . argument3 = mode ;
init_completion ( & pwr_done ) ;
mutex_lock ( & hba - > uic_cmd_mutex ) ;
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
hba - > pwr_done = & pwr_done ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
ret = __ufshcd_send_uic_cmd ( hba , & uic_cmd ) ;
if ( ret ) {
dev_err ( hba - > dev ,
" pwr mode change with mode 0x%x uic error %d \n " ,
mode , ret ) ;
goto out ;
}
if ( ! wait_for_completion_timeout ( hba - > pwr_done ,
msecs_to_jiffies ( UIC_CMD_TIMEOUT ) ) ) {
dev_err ( hba - > dev ,
" pwr mode change with mode 0x%x completion timeout \n " ,
mode ) ;
ret = - ETIMEDOUT ;
goto out ;
}
status = ufshcd_get_upmcrs ( hba ) ;
if ( status ! = PWR_LOCAL ) {
dev_err ( hba - > dev ,
" pwr mode change failed, host umpcrs:0x%x \n " ,
status ) ;
ret = ( status ! = PWR_OK ) ? status : - 1 ;
}
out :
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
hba - > pwr_done = NULL ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
mutex_unlock ( & hba - > uic_cmd_mutex ) ;
return ret ;
}
2013-08-31 21:40:24 +05:30
/**
* ufshcd_config_max_pwr_mode - Set & Change power mode with
* maximum capability attribute information .
* @ hba : per adapter instance
*
* Returns 0 on success , non - zero value on failure
*/
static int ufshcd_config_max_pwr_mode ( struct ufs_hba * hba )
{
enum { RX = 0 , TX = 1 } ;
u32 lanes [ ] = { 1 , 1 } ;
u32 gear [ ] = { 1 , 1 } ;
u8 pwr [ ] = { FASTAUTO_MODE , FASTAUTO_MODE } ;
int ret ;
/* Get the connected lane count */
ufshcd_dme_get ( hba , UIC_ARG_MIB ( PA_CONNECTEDRXDATALANES ) , & lanes [ RX ] ) ;
ufshcd_dme_get ( hba , UIC_ARG_MIB ( PA_CONNECTEDTXDATALANES ) , & lanes [ TX ] ) ;
/*
* First , get the maximum gears of HS speed .
* If a zero value , it means there is no HSGEAR capability .
* Then , get the maximum gears of PWM speed .
*/
ufshcd_dme_get ( hba , UIC_ARG_MIB ( PA_MAXRXHSGEAR ) , & gear [ RX ] ) ;
if ( ! gear [ RX ] ) {
ufshcd_dme_get ( hba , UIC_ARG_MIB ( PA_MAXRXPWMGEAR ) , & gear [ RX ] ) ;
pwr [ RX ] = SLOWAUTO_MODE ;
}
ufshcd_dme_peer_get ( hba , UIC_ARG_MIB ( PA_MAXRXHSGEAR ) , & gear [ TX ] ) ;
if ( ! gear [ TX ] ) {
ufshcd_dme_peer_get ( hba , UIC_ARG_MIB ( PA_MAXRXPWMGEAR ) ,
& gear [ TX ] ) ;
pwr [ TX ] = SLOWAUTO_MODE ;
}
/*
* Configure attributes for power mode change with below .
* - PA_RXGEAR , PA_ACTIVERXDATALANES , PA_RXTERMINATION ,
* - PA_TXGEAR , PA_ACTIVETXDATALANES , PA_TXTERMINATION ,
* - PA_HSSERIES
*/
ufshcd_dme_set ( hba , UIC_ARG_MIB ( PA_RXGEAR ) , gear [ RX ] ) ;
ufshcd_dme_set ( hba , UIC_ARG_MIB ( PA_ACTIVERXDATALANES ) , lanes [ RX ] ) ;
if ( pwr [ RX ] = = FASTAUTO_MODE )
ufshcd_dme_set ( hba , UIC_ARG_MIB ( PA_RXTERMINATION ) , TRUE ) ;
ufshcd_dme_set ( hba , UIC_ARG_MIB ( PA_TXGEAR ) , gear [ TX ] ) ;
ufshcd_dme_set ( hba , UIC_ARG_MIB ( PA_ACTIVETXDATALANES ) , lanes [ TX ] ) ;
if ( pwr [ TX ] = = FASTAUTO_MODE )
ufshcd_dme_set ( hba , UIC_ARG_MIB ( PA_TXTERMINATION ) , TRUE ) ;
if ( pwr [ RX ] = = FASTAUTO_MODE | | pwr [ TX ] = = FASTAUTO_MODE )
ufshcd_dme_set ( hba , UIC_ARG_MIB ( PA_HSSERIES ) , PA_HS_MODE_B ) ;
ret = ufshcd_uic_change_pwr_mode ( hba , pwr [ RX ] < < 4 | pwr [ TX ] ) ;
if ( ret )
dev_err ( hba - > dev ,
" pwr_mode: power mode change failed %d \n " , ret ) ;
return ret ;
}
2013-07-30 00:35:58 +05:30
/**
* ufshcd_complete_dev_init ( ) - checks device readiness
* hba : per - adapter instance
*
* Set fDeviceInit flag and poll until device toggles it .
*/
static int ufshcd_complete_dev_init ( struct ufs_hba * hba )
{
int i , retries , err = 0 ;
bool flag_res = 1 ;
for ( retries = QUERY_REQ_RETRIES ; retries > 0 ; retries - - ) {
/* Set the fDeviceInit flag */
err = ufshcd_query_flag ( hba , UPIU_QUERY_OPCODE_SET_FLAG ,
QUERY_FLAG_IDN_FDEVICEINIT , NULL ) ;
if ( ! err | | err = = - ETIMEDOUT )
break ;
dev_dbg ( hba - > dev , " %s: error %d retrying \n " , __func__ , err ) ;
}
if ( err ) {
dev_err ( hba - > dev ,
" %s setting fDeviceInit flag failed with error %d \n " ,
__func__ , err ) ;
goto out ;
}
/* poll for max. 100 iterations for fDeviceInit flag to clear */
for ( i = 0 ; i < 100 & & ! err & & flag_res ; i + + ) {
for ( retries = QUERY_REQ_RETRIES ; retries > 0 ; retries - - ) {
err = ufshcd_query_flag ( hba ,
UPIU_QUERY_OPCODE_READ_FLAG ,
QUERY_FLAG_IDN_FDEVICEINIT , & flag_res ) ;
if ( ! err | | err = = - ETIMEDOUT )
break ;
dev_dbg ( hba - > dev , " %s: error %d retrying \n " , __func__ ,
err ) ;
}
}
if ( err )
dev_err ( hba - > dev ,
" %s reading fDeviceInit flag failed with error %d \n " ,
__func__ , err ) ;
else if ( flag_res )
dev_err ( hba - > dev ,
" %s fDeviceInit was not cleared by the device \n " ,
__func__ ) ;
out :
return err ;
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_make_hba_operational - Make UFS controller operational
* @ hba : per adapter instance
*
* To bring UFS host controller to operational state ,
2014-09-25 15:32:21 +03:00
* 1. Enable required interrupts
* 2. Configure interrupt aggregation
* 3. Program UTRL and UTMRL base addres
* 4. Configure run - stop - registers
2012-02-29 12:11:50 +05:30
*
* Returns 0 on success , non - zero value on failure
*/
static int ufshcd_make_hba_operational ( struct ufs_hba * hba )
{
int err = 0 ;
u32 reg ;
2013-06-26 22:39:29 +05:30
/* Enable required interrupts */
ufshcd_enable_intr ( hba , UFSHCD_ENABLE_INTRS ) ;
/* Configure interrupt aggregation */
2013-08-31 21:40:20 +05:30
ufshcd_config_intr_aggr ( hba , hba - > nutrs - 1 , INT_AGGR_DEF_TO ) ;
2013-06-26 22:39:29 +05:30
/* Configure UTRL and UTMRL base address registers */
ufshcd_writel ( hba , lower_32_bits ( hba - > utrdl_dma_addr ) ,
REG_UTP_TRANSFER_REQ_LIST_BASE_L ) ;
ufshcd_writel ( hba , upper_32_bits ( hba - > utrdl_dma_addr ) ,
REG_UTP_TRANSFER_REQ_LIST_BASE_H ) ;
ufshcd_writel ( hba , lower_32_bits ( hba - > utmrdl_dma_addr ) ,
REG_UTP_TASK_REQ_LIST_BASE_L ) ;
ufshcd_writel ( hba , upper_32_bits ( hba - > utmrdl_dma_addr ) ,
REG_UTP_TASK_REQ_LIST_BASE_H ) ;
2012-02-29 12:11:50 +05:30
/*
* UCRDY , UTMRLDY and UTRLRDY bits must be 1
* DEI , HEI bits must be 0
*/
2014-09-25 15:32:21 +03:00
reg = ufshcd_readl ( hba , REG_CONTROLLER_STATUS ) ;
2012-02-29 12:11:50 +05:30
if ( ! ( ufshcd_get_lists_status ( reg ) ) ) {
ufshcd_enable_run_stop_reg ( hba ) ;
} else {
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev ,
2012-02-29 12:11:50 +05:30
" Host controller not ready to process requests " ) ;
err = - EIO ;
goto out ;
}
out :
return err ;
}
/**
* ufshcd_hba_enable - initialize the controller
* @ hba : per adapter instance
*
* The controller resets itself and controller firmware initialization
* sequence kicks off . When controller is ready it will set
* the Host Controller Enable bit to 1.
*
* Returns 0 on success , non - zero value on failure
*/
static int ufshcd_hba_enable ( struct ufs_hba * hba )
{
int retry ;
/*
* msleep of 1 and 5 used in this function might result in msleep ( 20 ) ,
* but it was necessary to send the UFS FPGA to reset mode during
* development and testing of this driver . msleep can be changed to
* mdelay and retry count can be reduced based on the controller .
*/
if ( ! ufshcd_is_hba_active ( hba ) ) {
/* change controller state to "reset state" */
ufshcd_hba_stop ( hba ) ;
/*
* This delay is based on the testing done with UFS host
* controller FPGA . The delay can be changed based on the
* host controller used .
*/
msleep ( 5 ) ;
}
2014-09-25 15:32:21 +03:00
if ( hba - > vops & & hba - > vops - > hce_enable_notify )
hba - > vops - > hce_enable_notify ( hba , PRE_CHANGE ) ;
2012-02-29 12:11:50 +05:30
/* start controller initialization sequence */
ufshcd_hba_start ( hba ) ;
/*
* To initialize a UFS host controller HCE bit must be set to 1.
* During initialization the HCE bit value changes from 1 - > 0 - > 1.
* When the host controller completes initialization sequence
* it sets the value of HCE bit to 1. The same HCE bit is read back
* to check if the controller has completed initialization sequence .
* So without this delay the value HCE = 1 , set in the previous
* instruction might be read back .
* This delay can be changed based on the controller .
*/
msleep ( 1 ) ;
/* wait for the host controller to complete initialization */
retry = 10 ;
while ( ufshcd_is_hba_active ( hba ) ) {
if ( retry ) {
retry - - ;
} else {
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev ,
2012-02-29 12:11:50 +05:30
" Controller enable failed \n " ) ;
return - EIO ;
}
msleep ( 5 ) ;
}
2014-09-25 15:32:21 +03:00
2014-09-25 15:32:26 +03:00
/* enable UIC related interrupts */
ufshcd_enable_intr ( hba , UIC_COMMAND_COMPL ) ;
2014-09-25 15:32:21 +03:00
if ( hba - > vops & & hba - > vops - > hce_enable_notify )
hba - > vops - > hce_enable_notify ( hba , POST_CHANGE ) ;
2012-02-29 12:11:50 +05:30
return 0 ;
}
/**
2013-06-26 22:39:29 +05:30
* ufshcd_link_startup - Initialize unipro link startup
2012-02-29 12:11:50 +05:30
* @ hba : per adapter instance
*
2013-06-26 22:39:29 +05:30
* Returns 0 for success , non - zero in case of failure
2012-02-29 12:11:50 +05:30
*/
2013-06-26 22:39:29 +05:30
static int ufshcd_link_startup ( struct ufs_hba * hba )
2012-02-29 12:11:50 +05:30
{
2013-06-26 22:39:29 +05:30
int ret ;
2014-09-25 15:32:26 +03:00
int retries = DME_LINKSTARTUP_RETRIES ;
2012-02-29 12:11:50 +05:30
2014-09-25 15:32:26 +03:00
do {
if ( hba - > vops & & hba - > vops - > link_startup_notify )
hba - > vops - > link_startup_notify ( hba , PRE_CHANGE ) ;
2013-06-26 22:39:29 +05:30
2014-09-25 15:32:26 +03:00
ret = ufshcd_dme_link_startup ( hba ) ;
2014-09-25 15:32:21 +03:00
2014-09-25 15:32:26 +03:00
/* check if device is detected by inter-connect layer */
if ( ! ret & & ! ufshcd_is_device_present ( hba ) ) {
dev_err ( hba - > dev , " %s: Device not present \n " , __func__ ) ;
ret = - ENXIO ;
goto out ;
}
2013-06-26 22:39:29 +05:30
2014-09-25 15:32:26 +03:00
/*
* DME link lost indication is only received when link is up ,
* but we can ' t be sure if the link is up until link startup
* succeeds . So reset the local Uni - Pro and try again .
*/
if ( ret & & ufshcd_hba_enable ( hba ) )
goto out ;
} while ( ret & & retries - - ) ;
if ( ret )
/* failed to get the link up... retire */
2014-09-25 15:32:21 +03:00
goto out ;
/* Include any host controller configuration via UIC commands */
if ( hba - > vops & & hba - > vops - > link_startup_notify ) {
ret = hba - > vops - > link_startup_notify ( hba , POST_CHANGE ) ;
if ( ret )
goto out ;
}
2012-02-29 12:11:50 +05:30
2014-09-25 15:32:21 +03:00
ret = ufshcd_make_hba_operational ( hba ) ;
2013-06-26 22:39:29 +05:30
out :
if ( ret )
dev_err ( hba - > dev , " link startup failed %d \n " , ret ) ;
return ret ;
2012-02-29 12:11:50 +05:30
}
2013-07-30 00:35:57 +05:30
/**
* ufshcd_verify_dev_init ( ) - Verify device initialization
* @ hba : per - adapter instance
*
* Send NOP OUT UPIU and wait for NOP IN response to check whether the
* device Transport Protocol ( UTP ) layer is ready after a reset .
* If the UTP layer at the device side is not initialized , it may
* not respond with NOP IN UPIU within timeout of % NOP_OUT_TIMEOUT
* and we retry sending NOP OUT for % NOP_OUT_RETRIES iterations .
*/
static int ufshcd_verify_dev_init ( struct ufs_hba * hba )
{
int err = 0 ;
int retries ;
mutex_lock ( & hba - > dev_cmd . lock ) ;
for ( retries = NOP_OUT_RETRIES ; retries > 0 ; retries - - ) {
err = ufshcd_exec_dev_cmd ( hba , DEV_CMD_TYPE_NOP ,
NOP_OUT_TIMEOUT ) ;
if ( ! err | | err = = - ETIMEDOUT )
break ;
dev_dbg ( hba - > dev , " %s: error %d retrying \n " , __func__ , err ) ;
}
mutex_unlock ( & hba - > dev_cmd . lock ) ;
if ( err )
dev_err ( hba - > dev , " %s: NOP OUT failed %d \n " , __func__ , err ) ;
return err ;
}
2014-09-25 15:32:29 +03:00
/**
* ufshcd_set_queue_depth - set lun queue depth
* @ sdev : pointer to SCSI device
*
* Read bLUQueueDepth value and activate scsi tagged command
* queueing . For WLUN , queue depth is set to 1. For best - effort
* cases ( bLUQueueDepth = 0 ) the queue depth is set to a maximum
* value that host can queue .
*/
static void ufshcd_set_queue_depth ( struct scsi_device * sdev )
{
int ret = 0 ;
u8 lun_qdepth ;
struct ufs_hba * hba ;
hba = shost_priv ( sdev - > host ) ;
lun_qdepth = hba - > nutrs ;
ret = ufshcd_read_unit_desc_param ( hba ,
ufshcd_scsi_to_upiu_lun ( sdev - > lun ) ,
UNIT_DESC_PARAM_LU_Q_DEPTH ,
& lun_qdepth ,
sizeof ( lun_qdepth ) ) ;
/* Some WLUN doesn't support unit descriptor */
if ( ret = = - EOPNOTSUPP )
lun_qdepth = 1 ;
else if ( ! lun_qdepth )
/* eventually, we can figure out the real queue depth */
lun_qdepth = hba - > nutrs ;
else
lun_qdepth = min_t ( int , lun_qdepth , hba - > nutrs ) ;
dev_dbg ( hba - > dev , " %s: activate tcq with queue depth %d \n " ,
__func__ , lun_qdepth ) ;
scsi_activate_tcq ( sdev , lun_qdepth ) ;
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_slave_alloc - handle initial SCSI device configurations
* @ sdev : pointer to SCSI device
*
* Returns success
*/
static int ufshcd_slave_alloc ( struct scsi_device * sdev )
{
struct ufs_hba * hba ;
hba = shost_priv ( sdev - > host ) ;
sdev - > tagged_supported = 1 ;
/* Mode sense(6) is not supported by UFS, so use Mode sense(10) */
sdev - > use_10_for_ms = 1 ;
scsi_set_tag_type ( sdev , MSG_SIMPLE_TAG ) ;
2014-05-26 10:59:15 +05:30
/* allow SCSI layer to restart the device in case of errors */
sdev - > allow_restart = 1 ;
2014-06-29 09:40:20 +03:00
2014-07-01 12:22:38 +03:00
/* REPORT SUPPORTED OPERATION CODES is not supported */
sdev - > no_report_opcodes = 1 ;
2014-05-26 10:59:15 +05:30
2014-09-25 15:32:29 +03:00
ufshcd_set_queue_depth ( sdev ) ;
2014-06-29 09:40:20 +03:00
2012-02-29 12:11:50 +05:30
return 0 ;
}
2014-06-29 09:40:20 +03:00
/**
* ufshcd_change_queue_depth - change queue depth
* @ sdev : pointer to SCSI device
* @ depth : required depth to set
* @ reason : reason for changing the depth
*
* Change queue depth according to the reason and make sure
* the max . limits are not crossed .
*/
2014-07-23 09:31:11 +03:00
static int ufshcd_change_queue_depth ( struct scsi_device * sdev ,
int depth , int reason )
2014-06-29 09:40:20 +03:00
{
struct ufs_hba * hba = shost_priv ( sdev - > host ) ;
if ( depth > hba - > nutrs )
depth = hba - > nutrs ;
switch ( reason ) {
case SCSI_QDEPTH_DEFAULT :
case SCSI_QDEPTH_RAMP_UP :
if ( ! sdev - > tagged_supported )
depth = 1 ;
scsi_adjust_queue_depth ( sdev , scsi_get_tag_type ( sdev ) , depth ) ;
break ;
case SCSI_QDEPTH_QFULL :
scsi_track_queue_full ( sdev , depth ) ;
break ;
default :
return - EOPNOTSUPP ;
}
return depth ;
}
2014-07-01 23:00:32 +09:00
/**
* ufshcd_slave_configure - adjust SCSI device configurations
* @ sdev : pointer to SCSI device
*/
static int ufshcd_slave_configure ( struct scsi_device * sdev )
{
struct request_queue * q = sdev - > request_queue ;
blk_queue_update_dma_pad ( q , PRDT_DATA_BYTE_COUNT_PAD - 1 ) ;
blk_queue_max_segment_size ( q , PRDT_DATA_BYTE_COUNT_MAX ) ;
return 0 ;
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_slave_destroy - remove SCSI device configurations
* @ sdev : pointer to SCSI device
*/
static void ufshcd_slave_destroy ( struct scsi_device * sdev )
{
struct ufs_hba * hba ;
hba = shost_priv ( sdev - > host ) ;
scsi_deactivate_tcq ( sdev , hba - > nutrs ) ;
2014-09-25 15:32:29 +03:00
/* Drop the reference as it won't be needed anymore */
if ( ufshcd_scsi_to_upiu_lun ( sdev - > lun ) = = UFS_UPIU_UFS_DEVICE_WLUN )
hba - > sdev_ufs_device = NULL ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_task_req_compl - handle task management request completion
* @ hba : per adapter instance
* @ index : index of the completed request
2014-05-26 10:59:12 +05:30
* @ resp : task management service response
2012-02-29 12:11:50 +05:30
*
2014-05-26 10:59:12 +05:30
* Returns non - zero value on error , zero on success
2012-02-29 12:11:50 +05:30
*/
2014-05-26 10:59:12 +05:30
static int ufshcd_task_req_compl ( struct ufs_hba * hba , u32 index , u8 * resp )
2012-02-29 12:11:50 +05:30
{
struct utp_task_req_desc * task_req_descp ;
struct utp_upiu_task_rsp * task_rsp_upiup ;
unsigned long flags ;
int ocs_value ;
int task_result ;
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
/* Clear completed tasks from outstanding_tasks */
__clear_bit ( index , & hba - > outstanding_tasks ) ;
task_req_descp = hba - > utmrdl_base_addr ;
ocs_value = ufshcd_get_tmr_ocs ( & task_req_descp [ index ] ) ;
if ( ocs_value = = OCS_SUCCESS ) {
task_rsp_upiup = ( struct utp_upiu_task_rsp * )
task_req_descp [ index ] . task_rsp_upiu ;
task_result = be32_to_cpu ( task_rsp_upiup - > header . dword_1 ) ;
task_result = ( ( task_result & MASK_TASK_RESPONSE ) > > 8 ) ;
2014-05-26 10:59:12 +05:30
if ( resp )
* resp = ( u8 ) task_result ;
2012-02-29 12:11:50 +05:30
} else {
2014-05-26 10:59:12 +05:30
dev_err ( hba - > dev , " %s: failed, ocs = 0x%x \n " ,
__func__ , ocs_value ) ;
2012-02-29 12:11:50 +05:30
}
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
2014-05-26 10:59:12 +05:30
return ocs_value ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_scsi_cmd_status - Update SCSI command result based on SCSI status
* @ lrb : pointer to local reference block of completed command
* @ scsi_status : SCSI command status
*
* Returns value base on SCSI command status
*/
static inline int
ufshcd_scsi_cmd_status ( struct ufshcd_lrb * lrbp , int scsi_status )
{
int result = 0 ;
switch ( scsi_status ) {
case SAM_STAT_CHECK_CONDITION :
2013-08-31 21:40:19 +05:30
ufshcd_copy_sense_data ( lrbp ) ;
case SAM_STAT_GOOD :
2012-02-29 12:11:50 +05:30
result | = DID_OK < < 16 |
COMMAND_COMPLETE < < 8 |
2013-08-31 21:40:19 +05:30
scsi_status ;
2012-02-29 12:11:50 +05:30
break ;
case SAM_STAT_TASK_SET_FULL :
2013-08-31 21:40:19 +05:30
case SAM_STAT_BUSY :
2012-02-29 12:11:50 +05:30
case SAM_STAT_TASK_ABORTED :
2013-08-31 21:40:19 +05:30
ufshcd_copy_sense_data ( lrbp ) ;
result | = scsi_status ;
2012-02-29 12:11:50 +05:30
break ;
default :
result | = DID_ERROR < < 16 ;
break ;
} /* end of switch */
return result ;
}
/**
* ufshcd_transfer_rsp_status - Get overall status of the response
* @ hba : per adapter instance
* @ lrb : pointer to local reference block of completed command
*
* Returns result of the command to notify SCSI midlayer
*/
static inline int
ufshcd_transfer_rsp_status ( struct ufs_hba * hba , struct ufshcd_lrb * lrbp )
{
int result = 0 ;
int scsi_status ;
int ocs ;
/* overall command status of utrd */
ocs = ufshcd_get_tr_ocs ( lrbp ) ;
switch ( ocs ) {
case OCS_SUCCESS :
2013-07-30 00:35:57 +05:30
result = ufshcd_get_req_rsp ( lrbp - > ucd_rsp_ptr ) ;
2012-02-29 12:11:50 +05:30
2013-07-30 00:35:57 +05:30
switch ( result ) {
case UPIU_TRANSACTION_RESPONSE :
/*
* get the response UPIU result to extract
* the SCSI command status
*/
result = ufshcd_get_rsp_upiu_result ( lrbp - > ucd_rsp_ptr ) ;
/*
* get the result based on SCSI status response
* to notify the SCSI midlayer of the command status
*/
scsi_status = result & MASK_SCSI_STATUS ;
result = ufshcd_scsi_cmd_status ( lrbp , scsi_status ) ;
2013-07-30 00:35:59 +05:30
if ( ufshcd_is_exception_event ( lrbp - > ucd_rsp_ptr ) )
schedule_work ( & hba - > eeh_work ) ;
2013-07-30 00:35:57 +05:30
break ;
case UPIU_TRANSACTION_REJECT_UPIU :
/* TODO: handle Reject UPIU Response */
result = DID_ERROR < < 16 ;
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev ,
2013-07-30 00:35:57 +05:30
" Reject UPIU not fully implemented \n " ) ;
break ;
default :
result = DID_ERROR < < 16 ;
dev_err ( hba - > dev ,
" Unexpected request response code = %x \n " ,
result ) ;
2012-02-29 12:11:50 +05:30
break ;
}
break ;
case OCS_ABORTED :
result | = DID_ABORT < < 16 ;
break ;
2014-05-26 10:59:15 +05:30
case OCS_INVALID_COMMAND_STATUS :
result | = DID_REQUEUE < < 16 ;
break ;
2012-02-29 12:11:50 +05:30
case OCS_INVALID_CMD_TABLE_ATTR :
case OCS_INVALID_PRDT_ATTR :
case OCS_MISMATCH_DATA_BUF_SIZE :
case OCS_MISMATCH_RESP_UPIU_SIZE :
case OCS_PEER_COMM_FAILURE :
case OCS_FATAL_ERROR :
default :
result | = DID_ERROR < < 16 ;
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev ,
2012-02-29 12:11:50 +05:30
" OCS error from controller = %x \n " , ocs ) ;
break ;
} /* end of switch */
return result ;
}
2013-06-26 22:39:29 +05:30
/**
* ufshcd_uic_cmd_compl - handle completion of uic command
* @ hba : per adapter instance
2013-08-31 21:40:22 +05:30
* @ intr_status : interrupt status generated by the controller
2013-06-26 22:39:29 +05:30
*/
2013-08-31 21:40:22 +05:30
static void ufshcd_uic_cmd_compl ( struct ufs_hba * hba , u32 intr_status )
2013-06-26 22:39:29 +05:30
{
2013-08-31 21:40:22 +05:30
if ( ( intr_status & UIC_COMMAND_COMPL ) & & hba - > active_uic_cmd ) {
2013-06-26 22:39:29 +05:30
hba - > active_uic_cmd - > argument2 | =
ufshcd_get_uic_cmd_result ( hba ) ;
2013-08-31 21:40:21 +05:30
hba - > active_uic_cmd - > argument3 =
ufshcd_get_dme_attr_val ( hba ) ;
2013-06-26 22:39:29 +05:30
complete ( & hba - > active_uic_cmd - > done ) ;
}
2013-08-31 21:40:22 +05:30
if ( ( intr_status & UIC_POWER_MODE ) & & hba - > pwr_done )
complete ( hba - > pwr_done ) ;
2013-06-26 22:39:29 +05:30
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_transfer_req_compl - handle SCSI and query command completion
* @ hba : per adapter instance
*/
static void ufshcd_transfer_req_compl ( struct ufs_hba * hba )
{
2013-07-30 00:35:57 +05:30
struct ufshcd_lrb * lrbp ;
struct scsi_cmnd * cmd ;
2012-02-29 12:11:50 +05:30
unsigned long completed_reqs ;
u32 tr_doorbell ;
int result ;
int index ;
2014-07-01 12:22:37 +03:00
/* Resetting interrupt aggregation counters first and reading the
* DOOR_BELL afterward allows us to handle all the completed requests .
* In order to prevent other interrupts starvation the DB is read once
* after reset . The down side of this solution is the possibility of
* false interrupt if device completes another request after resetting
* aggregation and before reading the DB .
*/
ufshcd_reset_intr_aggr ( hba ) ;
2012-02-29 12:11:50 +05:30
2013-06-26 22:39:26 +05:30
tr_doorbell = ufshcd_readl ( hba , REG_UTP_TRANSFER_REQ_DOOR_BELL ) ;
2012-02-29 12:11:50 +05:30
completed_reqs = tr_doorbell ^ hba - > outstanding_reqs ;
2014-07-01 12:22:37 +03:00
for_each_set_bit ( index , & completed_reqs , hba - > nutrs ) {
lrbp = & hba - > lrb [ index ] ;
cmd = lrbp - > cmd ;
if ( cmd ) {
result = ufshcd_transfer_rsp_status ( hba , lrbp ) ;
scsi_dma_unmap ( cmd ) ;
cmd - > result = result ;
/* Mark completed command as NULL in LRB */
lrbp - > cmd = NULL ;
clear_bit_unlock ( index , & hba - > lrb_in_use ) ;
/* Do not touch lrbp after scsi done */
cmd - > scsi_done ( cmd ) ;
} else if ( lrbp - > command_type = = UTP_CMD_TYPE_DEV_MANAGE ) {
if ( hba - > dev_cmd . complete )
complete ( hba - > dev_cmd . complete ) ;
}
}
2012-02-29 12:11:50 +05:30
/* clear corresponding bits of completed commands */
hba - > outstanding_reqs ^ = completed_reqs ;
2013-07-30 00:35:57 +05:30
/* we might have free'd some tags above */
wake_up ( & hba - > dev_cmd . tag_wq ) ;
2012-02-29 12:11:50 +05:30
}
2013-07-30 00:35:59 +05:30
/**
* ufshcd_disable_ee - disable exception event
* @ hba : per - adapter instance
* @ mask : exception event to disable
*
* Disables exception event in the device so that the EVENT_ALERT
* bit is not set .
*
* Returns zero on success , non - zero error value on failure .
*/
static int ufshcd_disable_ee ( struct ufs_hba * hba , u16 mask )
{
int err = 0 ;
u32 val ;
if ( ! ( hba - > ee_ctrl_mask & mask ) )
goto out ;
val = hba - > ee_ctrl_mask & ~ mask ;
val & = 0xFFFF ; /* 2 bytes */
err = ufshcd_query_attr ( hba , UPIU_QUERY_OPCODE_WRITE_ATTR ,
QUERY_ATTR_IDN_EE_CONTROL , 0 , 0 , & val ) ;
if ( ! err )
hba - > ee_ctrl_mask & = ~ mask ;
out :
return err ;
}
/**
* ufshcd_enable_ee - enable exception event
* @ hba : per - adapter instance
* @ mask : exception event to enable
*
* Enable corresponding exception event in the device to allow
* device to alert host in critical scenarios .
*
* Returns zero on success , non - zero error value on failure .
*/
static int ufshcd_enable_ee ( struct ufs_hba * hba , u16 mask )
{
int err = 0 ;
u32 val ;
if ( hba - > ee_ctrl_mask & mask )
goto out ;
val = hba - > ee_ctrl_mask | mask ;
val & = 0xFFFF ; /* 2 bytes */
err = ufshcd_query_attr ( hba , UPIU_QUERY_OPCODE_WRITE_ATTR ,
QUERY_ATTR_IDN_EE_CONTROL , 0 , 0 , & val ) ;
if ( ! err )
hba - > ee_ctrl_mask | = mask ;
out :
return err ;
}
/**
* ufshcd_enable_auto_bkops - Allow device managed BKOPS
* @ hba : per - adapter instance
*
* Allow device to manage background operations on its own . Enabling
* this might lead to inconsistent latencies during normal data transfers
* as the device is allowed to manage its own way of handling background
* operations .
*
* Returns zero on success , non - zero on failure .
*/
static int ufshcd_enable_auto_bkops ( struct ufs_hba * hba )
{
int err = 0 ;
if ( hba - > auto_bkops_enabled )
goto out ;
err = ufshcd_query_flag ( hba , UPIU_QUERY_OPCODE_SET_FLAG ,
QUERY_FLAG_IDN_BKOPS_EN , NULL ) ;
if ( err ) {
dev_err ( hba - > dev , " %s: failed to enable bkops %d \n " ,
__func__ , err ) ;
goto out ;
}
hba - > auto_bkops_enabled = true ;
/* No need of URGENT_BKOPS exception from the device */
err = ufshcd_disable_ee ( hba , MASK_EE_URGENT_BKOPS ) ;
if ( err )
dev_err ( hba - > dev , " %s: failed to disable exception event %d \n " ,
__func__ , err ) ;
out :
return err ;
}
/**
* ufshcd_disable_auto_bkops - block device in doing background operations
* @ hba : per - adapter instance
*
* Disabling background operations improves command response latency but
* has drawback of device moving into critical state where the device is
* not - operable . Make sure to call ufshcd_enable_auto_bkops ( ) whenever the
* host is idle so that BKOPS are managed effectively without any negative
* impacts .
*
* Returns zero on success , non - zero on failure .
*/
static int ufshcd_disable_auto_bkops ( struct ufs_hba * hba )
{
int err = 0 ;
if ( ! hba - > auto_bkops_enabled )
goto out ;
/*
* If host assisted BKOPs is to be enabled , make sure
* urgent bkops exception is allowed .
*/
err = ufshcd_enable_ee ( hba , MASK_EE_URGENT_BKOPS ) ;
if ( err ) {
dev_err ( hba - > dev , " %s: failed to enable exception event %d \n " ,
__func__ , err ) ;
goto out ;
}
err = ufshcd_query_flag ( hba , UPIU_QUERY_OPCODE_CLEAR_FLAG ,
QUERY_FLAG_IDN_BKOPS_EN , NULL ) ;
if ( err ) {
dev_err ( hba - > dev , " %s: failed to disable bkops %d \n " ,
__func__ , err ) ;
ufshcd_disable_ee ( hba , MASK_EE_URGENT_BKOPS ) ;
goto out ;
}
hba - > auto_bkops_enabled = false ;
out :
return err ;
}
/**
* ufshcd_force_reset_auto_bkops - force enable of auto bkops
* @ hba : per adapter instance
*
* After a device reset the device may toggle the BKOPS_EN flag
* to default value . The s / w tracking variables should be updated
* as well . Do this by forcing enable of auto bkops .
*/
static void ufshcd_force_reset_auto_bkops ( struct ufs_hba * hba )
{
hba - > auto_bkops_enabled = false ;
hba - > ee_ctrl_mask | = MASK_EE_URGENT_BKOPS ;
ufshcd_enable_auto_bkops ( hba ) ;
}
static inline int ufshcd_get_bkops_status ( struct ufs_hba * hba , u32 * status )
{
return ufshcd_query_attr ( hba , UPIU_QUERY_OPCODE_READ_ATTR ,
QUERY_ATTR_IDN_BKOPS_STATUS , 0 , 0 , status ) ;
}
/**
* ufshcd_urgent_bkops - handle urgent bkops exception event
* @ hba : per - adapter instance
*
* Enable fBackgroundOpsEn flag in the device to permit background
* operations .
*/
static int ufshcd_urgent_bkops ( struct ufs_hba * hba )
{
int err ;
u32 status = 0 ;
err = ufshcd_get_bkops_status ( hba , & status ) ;
if ( err ) {
dev_err ( hba - > dev , " %s: failed to get BKOPS status %d \n " ,
__func__ , err ) ;
goto out ;
}
status = status & 0xF ;
/* handle only if status indicates performance impact or critical */
if ( status > = BKOPS_STATUS_PERF_IMPACT )
err = ufshcd_enable_auto_bkops ( hba ) ;
out :
return err ;
}
static inline int ufshcd_get_ee_status ( struct ufs_hba * hba , u32 * status )
{
return ufshcd_query_attr ( hba , UPIU_QUERY_OPCODE_READ_ATTR ,
QUERY_ATTR_IDN_EE_STATUS , 0 , 0 , status ) ;
}
/**
* ufshcd_exception_event_handler - handle exceptions raised by device
* @ work : pointer to work data
*
* Read bExceptionEventStatus attribute from the device and handle the
* exception event accordingly .
*/
static void ufshcd_exception_event_handler ( struct work_struct * work )
{
struct ufs_hba * hba ;
int err ;
u32 status = 0 ;
hba = container_of ( work , struct ufs_hba , eeh_work ) ;
2013-07-30 00:36:00 +05:30
pm_runtime_get_sync ( hba - > dev ) ;
2013-07-30 00:35:59 +05:30
err = ufshcd_get_ee_status ( hba , & status ) ;
if ( err ) {
dev_err ( hba - > dev , " %s: failed to get exception status %d \n " ,
__func__ , err ) ;
goto out ;
}
status & = hba - > ee_ctrl_mask ;
if ( status & MASK_EE_URGENT_BKOPS ) {
err = ufshcd_urgent_bkops ( hba ) ;
if ( err )
dev_err ( hba - > dev , " %s: failed to handle urgent bkops %d \n " ,
__func__ , err ) ;
}
out :
2013-07-30 00:36:00 +05:30
pm_runtime_put_sync ( hba - > dev ) ;
2013-07-30 00:35:59 +05:30
return ;
}
2012-02-29 12:11:50 +05:30
/**
2014-05-26 10:59:15 +05:30
* ufshcd_err_handler - handle UFS errors that require s / w attention
* @ work : pointer to work structure
2012-02-29 12:11:50 +05:30
*/
2014-05-26 10:59:15 +05:30
static void ufshcd_err_handler ( struct work_struct * work )
2012-02-29 12:11:50 +05:30
{
struct ufs_hba * hba ;
2014-05-26 10:59:15 +05:30
unsigned long flags ;
u32 err_xfer = 0 ;
u32 err_tm = 0 ;
int err = 0 ;
int tag ;
hba = container_of ( work , struct ufs_hba , eh_work ) ;
2012-02-29 12:11:50 +05:30
2013-07-30 00:36:00 +05:30
pm_runtime_get_sync ( hba - > dev ) ;
2014-05-26 10:59:15 +05:30
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
if ( hba - > ufshcd_state = = UFSHCD_STATE_RESET ) {
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
goto out ;
}
hba - > ufshcd_state = UFSHCD_STATE_RESET ;
ufshcd_set_eh_in_progress ( hba ) ;
/* Complete requests that have door-bell cleared by h/w */
ufshcd_transfer_req_compl ( hba ) ;
ufshcd_tmc_handler ( hba ) ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
/* Clear pending transfer requests */
for_each_set_bit ( tag , & hba - > outstanding_reqs , hba - > nutrs )
if ( ufshcd_clear_cmd ( hba , tag ) )
err_xfer | = 1 < < tag ;
/* Clear pending task management requests */
for_each_set_bit ( tag , & hba - > outstanding_tasks , hba - > nutmrs )
if ( ufshcd_clear_tm_cmd ( hba , tag ) )
err_tm | = 1 < < tag ;
/* Complete the requests that are cleared by s/w */
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
ufshcd_transfer_req_compl ( hba ) ;
ufshcd_tmc_handler ( hba ) ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
/* Fatal errors need reset */
if ( err_xfer | | err_tm | | ( hba - > saved_err & INT_FATAL_ERRORS ) | |
( ( hba - > saved_err & UIC_ERROR ) & &
( hba - > saved_uic_err & UFSHCD_UIC_DL_PA_INIT_ERROR ) ) ) {
err = ufshcd_reset_and_restore ( hba ) ;
if ( err ) {
dev_err ( hba - > dev , " %s: reset and restore failed \n " ,
__func__ ) ;
hba - > ufshcd_state = UFSHCD_STATE_ERROR ;
}
/*
* Inform scsi mid - layer that we did reset and allow to handle
* Unit Attention properly .
*/
scsi_report_bus_reset ( hba - > host , 0 ) ;
hba - > saved_err = 0 ;
hba - > saved_uic_err = 0 ;
}
ufshcd_clear_eh_in_progress ( hba ) ;
out :
scsi_unblock_requests ( hba - > host ) ;
2013-07-30 00:36:00 +05:30
pm_runtime_put_sync ( hba - > dev ) ;
2012-02-29 12:11:50 +05:30
}
/**
2014-05-26 10:59:15 +05:30
* ufshcd_update_uic_error - check and set fatal UIC error flags .
* @ hba : per - adapter instance
2012-02-29 12:11:50 +05:30
*/
2014-05-26 10:59:15 +05:30
static void ufshcd_update_uic_error ( struct ufs_hba * hba )
2012-02-29 12:11:50 +05:30
{
u32 reg ;
2014-05-26 10:59:15 +05:30
/* PA_INIT_ERROR is fatal and needs UIC reset */
reg = ufshcd_readl ( hba , REG_UIC_ERROR_CODE_DATA_LINK_LAYER ) ;
if ( reg & UIC_DATA_LINK_LAYER_ERROR_PA_INIT )
hba - > uic_error | = UFSHCD_UIC_DL_PA_INIT_ERROR ;
/* UIC NL/TL/DME errors needs software retry */
reg = ufshcd_readl ( hba , REG_UIC_ERROR_CODE_NETWORK_LAYER ) ;
if ( reg )
hba - > uic_error | = UFSHCD_UIC_NL_ERROR ;
reg = ufshcd_readl ( hba , REG_UIC_ERROR_CODE_TRANSPORT_LAYER ) ;
if ( reg )
hba - > uic_error | = UFSHCD_UIC_TL_ERROR ;
reg = ufshcd_readl ( hba , REG_UIC_ERROR_CODE_DME ) ;
if ( reg )
hba - > uic_error | = UFSHCD_UIC_DME_ERROR ;
dev_dbg ( hba - > dev , " %s: UIC error flags = 0x%08x \n " ,
__func__ , hba - > uic_error ) ;
}
/**
* ufshcd_check_errors - Check for errors that need s / w attention
* @ hba : per - adapter instance
*/
static void ufshcd_check_errors ( struct ufs_hba * hba )
{
bool queue_eh_work = false ;
2012-02-29 12:11:50 +05:30
if ( hba - > errors & INT_FATAL_ERRORS )
2014-05-26 10:59:15 +05:30
queue_eh_work = true ;
2012-02-29 12:11:50 +05:30
if ( hba - > errors & UIC_ERROR ) {
2014-05-26 10:59:15 +05:30
hba - > uic_error = 0 ;
ufshcd_update_uic_error ( hba ) ;
if ( hba - > uic_error )
queue_eh_work = true ;
2012-02-29 12:11:50 +05:30
}
2014-05-26 10:59:15 +05:30
if ( queue_eh_work ) {
/* handle fatal errors only when link is functional */
if ( hba - > ufshcd_state = = UFSHCD_STATE_OPERATIONAL ) {
/* block commands from scsi mid-layer */
scsi_block_requests ( hba - > host ) ;
/* transfer error masks to sticky bits */
hba - > saved_err | = hba - > errors ;
hba - > saved_uic_err | = hba - > uic_error ;
hba - > ufshcd_state = UFSHCD_STATE_ERROR ;
schedule_work ( & hba - > eh_work ) ;
}
2014-05-26 10:59:14 +05:30
}
2014-05-26 10:59:15 +05:30
/*
* if ( ! queue_eh_work ) -
* Other errors are either non - fatal where host recovers
* itself without s / w intervention or errors that will be
* handled by the SCSI core layer .
*/
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_tmc_handler - handle task management function completion
* @ hba : per adapter instance
*/
static void ufshcd_tmc_handler ( struct ufs_hba * hba )
{
u32 tm_doorbell ;
2013-06-26 22:39:26 +05:30
tm_doorbell = ufshcd_readl ( hba , REG_UTP_TASK_REQ_DOOR_BELL ) ;
2012-02-29 12:11:50 +05:30
hba - > tm_condition = tm_doorbell ^ hba - > outstanding_tasks ;
2014-05-26 10:59:12 +05:30
wake_up ( & hba - > tm_wq ) ;
2012-02-29 12:11:50 +05:30
}
/**
* ufshcd_sl_intr - Interrupt service routine
* @ hba : per adapter instance
* @ intr_status : contains interrupts generated by the controller
*/
static void ufshcd_sl_intr ( struct ufs_hba * hba , u32 intr_status )
{
hba - > errors = UFSHCD_ERROR_MASK & intr_status ;
if ( hba - > errors )
2014-05-26 10:59:15 +05:30
ufshcd_check_errors ( hba ) ;
2012-02-29 12:11:50 +05:30
2013-08-31 21:40:22 +05:30
if ( intr_status & UFSHCD_UIC_MASK )
ufshcd_uic_cmd_compl ( hba , intr_status ) ;
2012-02-29 12:11:50 +05:30
if ( intr_status & UTP_TASK_REQ_COMPL )
ufshcd_tmc_handler ( hba ) ;
if ( intr_status & UTP_TRANSFER_REQ_COMPL )
ufshcd_transfer_req_compl ( hba ) ;
}
/**
* ufshcd_intr - Main interrupt service routine
* @ irq : irq number
* @ __hba : pointer to adapter instance
*
* Returns IRQ_HANDLED - If interrupt is valid
* IRQ_NONE - If invalid interrupt
*/
static irqreturn_t ufshcd_intr ( int irq , void * __hba )
{
u32 intr_status ;
irqreturn_t retval = IRQ_NONE ;
struct ufs_hba * hba = __hba ;
spin_lock ( hba - > host - > host_lock ) ;
2013-06-26 22:39:26 +05:30
intr_status = ufshcd_readl ( hba , REG_INTERRUPT_STATUS ) ;
2012-02-29 12:11:50 +05:30
if ( intr_status ) {
2013-06-26 22:39:28 +05:30
ufshcd_writel ( hba , intr_status , REG_INTERRUPT_STATUS ) ;
2012-02-29 12:11:50 +05:30
ufshcd_sl_intr ( hba , intr_status ) ;
retval = IRQ_HANDLED ;
}
spin_unlock ( hba - > host - > host_lock ) ;
return retval ;
}
2014-05-26 10:59:12 +05:30
static int ufshcd_clear_tm_cmd ( struct ufs_hba * hba , int tag )
{
int err = 0 ;
u32 mask = 1 < < tag ;
unsigned long flags ;
if ( ! test_bit ( tag , & hba - > outstanding_tasks ) )
goto out ;
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
ufshcd_writel ( hba , ~ ( 1 < < tag ) , REG_UTP_TASK_REQ_LIST_CLEAR ) ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
/* poll for max. 1 sec to clear door bell register by h/w */
err = ufshcd_wait_for_register ( hba ,
REG_UTP_TASK_REQ_DOOR_BELL ,
mask , 0 , 1000 , 1000 ) ;
out :
return err ;
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_issue_tm_cmd - issues task management commands to controller
* @ hba : per adapter instance
2014-05-26 10:59:12 +05:30
* @ lun_id : LUN ID to which TM command is sent
* @ task_id : task ID to which the TM command is applicable
* @ tm_function : task management function opcode
* @ tm_response : task management service response return value
2012-02-29 12:11:50 +05:30
*
2014-05-26 10:59:12 +05:30
* Returns non - zero value on error , zero on success .
2012-02-29 12:11:50 +05:30
*/
2014-05-26 10:59:12 +05:30
static int ufshcd_issue_tm_cmd ( struct ufs_hba * hba , int lun_id , int task_id ,
u8 tm_function , u8 * tm_response )
2012-02-29 12:11:50 +05:30
{
struct utp_task_req_desc * task_req_descp ;
struct utp_upiu_task_req * task_req_upiup ;
struct Scsi_Host * host ;
unsigned long flags ;
2014-05-26 10:59:12 +05:30
int free_slot ;
2012-02-29 12:11:50 +05:30
int err ;
2014-05-26 10:59:12 +05:30
int task_tag ;
2012-02-29 12:11:50 +05:30
host = hba - > host ;
2014-05-26 10:59:12 +05:30
/*
* Get free slot , sleep if slots are unavailable .
* Even though we use wait_event ( ) which sleeps indefinitely ,
* the maximum wait time is bounded by % TM_CMD_TIMEOUT .
*/
wait_event ( hba - > tm_tag_wq , ufshcd_get_tm_free_slot ( hba , & free_slot ) ) ;
2012-02-29 12:11:50 +05:30
2014-05-26 10:59:12 +05:30
spin_lock_irqsave ( host - > host_lock , flags ) ;
2012-02-29 12:11:50 +05:30
task_req_descp = hba - > utmrdl_base_addr ;
task_req_descp + = free_slot ;
/* Configure task request descriptor */
task_req_descp - > header . dword_0 = cpu_to_le32 ( UTP_REQ_DESC_INT_CMD ) ;
task_req_descp - > header . dword_2 =
cpu_to_le32 ( OCS_INVALID_COMMAND_STATUS ) ;
/* Configure task request UPIU */
task_req_upiup =
( struct utp_upiu_task_req * ) task_req_descp - > task_req_upiu ;
2014-05-26 10:59:12 +05:30
task_tag = hba - > nutrs + free_slot ;
2012-02-29 12:11:50 +05:30
task_req_upiup - > header . dword_0 =
2013-07-30 00:35:57 +05:30
UPIU_HEADER_DWORD ( UPIU_TRANSACTION_TASK_REQ , 0 ,
2014-05-26 10:59:12 +05:30
lun_id , task_tag ) ;
2012-02-29 12:11:50 +05:30
task_req_upiup - > header . dword_1 =
2013-07-30 00:35:57 +05:30
UPIU_HEADER_DWORD ( 0 , tm_function , 0 , 0 ) ;
2014-09-25 15:32:29 +03:00
/*
* The host shall provide the same value for LUN field in the basic
* header and for Input Parameter .
*/
2014-05-26 10:59:12 +05:30
task_req_upiup - > input_param1 = cpu_to_be32 ( lun_id ) ;
task_req_upiup - > input_param2 = cpu_to_be32 ( task_id ) ;
2012-02-29 12:11:50 +05:30
/* send command to the controller */
__set_bit ( free_slot , & hba - > outstanding_tasks ) ;
2013-06-26 22:39:26 +05:30
ufshcd_writel ( hba , 1 < < free_slot , REG_UTP_TASK_REQ_DOOR_BELL ) ;
2012-02-29 12:11:50 +05:30
spin_unlock_irqrestore ( host - > host_lock , flags ) ;
/* wait until the task management command is completed */
2014-05-26 10:59:12 +05:30
err = wait_event_timeout ( hba - > tm_wq ,
test_bit ( free_slot , & hba - > tm_condition ) ,
msecs_to_jiffies ( TM_CMD_TIMEOUT ) ) ;
2012-02-29 12:11:50 +05:30
if ( ! err ) {
2014-05-26 10:59:12 +05:30
dev_err ( hba - > dev , " %s: task management cmd 0x%.2x timed-out \n " ,
__func__ , tm_function ) ;
if ( ufshcd_clear_tm_cmd ( hba , free_slot ) )
dev_WARN ( hba - > dev , " %s: unable clear tm cmd (slot %d) after timeout \n " ,
__func__ , free_slot ) ;
err = - ETIMEDOUT ;
} else {
err = ufshcd_task_req_compl ( hba , free_slot , tm_response ) ;
2012-02-29 12:11:50 +05:30
}
2014-05-26 10:59:12 +05:30
2012-02-29 12:11:50 +05:30
clear_bit ( free_slot , & hba - > tm_condition ) ;
2014-05-26 10:59:12 +05:30
ufshcd_put_tm_slot ( hba , free_slot ) ;
wake_up ( & hba - > tm_tag_wq ) ;
2012-02-29 12:11:50 +05:30
return err ;
}
/**
2014-05-26 10:59:14 +05:30
* ufshcd_eh_device_reset_handler - device reset handler registered to
* scsi layer .
2012-02-29 12:11:50 +05:30
* @ cmd : SCSI command pointer
*
* Returns SUCCESS / FAILED
*/
2014-05-26 10:59:14 +05:30
static int ufshcd_eh_device_reset_handler ( struct scsi_cmnd * cmd )
2012-02-29 12:11:50 +05:30
{
struct Scsi_Host * host ;
struct ufs_hba * hba ;
unsigned int tag ;
u32 pos ;
int err ;
2014-05-26 10:59:12 +05:30
u8 resp = 0xF ;
struct ufshcd_lrb * lrbp ;
2014-05-26 10:59:14 +05:30
unsigned long flags ;
2012-02-29 12:11:50 +05:30
host = cmd - > device - > host ;
hba = shost_priv ( host ) ;
tag = cmd - > request - > tag ;
2014-05-26 10:59:12 +05:30
lrbp = & hba - > lrb [ tag ] ;
err = ufshcd_issue_tm_cmd ( hba , lrbp - > lun , 0 , UFS_LOGICAL_RESET , & resp ) ;
if ( err | | resp ! = UPIU_TASK_MANAGEMENT_FUNC_COMPL ) {
2014-05-26 10:59:14 +05:30
if ( ! err )
err = resp ;
2012-02-29 12:11:50 +05:30
goto out ;
2014-05-26 10:59:12 +05:30
}
2012-02-29 12:11:50 +05:30
2014-05-26 10:59:14 +05:30
/* clear the commands that were pending for corresponding LUN */
for_each_set_bit ( pos , & hba - > outstanding_reqs , hba - > nutrs ) {
if ( hba - > lrb [ pos ] . lun = = lrbp - > lun ) {
err = ufshcd_clear_cmd ( hba , pos ) ;
if ( err )
break ;
2012-02-29 12:11:50 +05:30
}
2014-05-26 10:59:14 +05:30
}
spin_lock_irqsave ( host - > host_lock , flags ) ;
ufshcd_transfer_req_compl ( hba ) ;
spin_unlock_irqrestore ( host - > host_lock , flags ) ;
2012-02-29 12:11:50 +05:30
out :
2014-05-26 10:59:14 +05:30
if ( ! err ) {
err = SUCCESS ;
} else {
dev_err ( hba - > dev , " %s: failed with err %d \n " , __func__ , err ) ;
err = FAILED ;
}
2012-02-29 12:11:50 +05:30
return err ;
}
/**
* ufshcd_abort - abort a specific command
* @ cmd : SCSI command pointer
*
2014-05-26 10:59:13 +05:30
* Abort the pending command in device by sending UFS_ABORT_TASK task management
* command , and in host controller by clearing the door - bell register . There can
* be race between controller sending the command to the device while abort is
* issued . To avoid that , first issue UFS_QUERY_TASK to check if the command is
* really issued and then try to abort it .
*
2012-02-29 12:11:50 +05:30
* Returns SUCCESS / FAILED
*/
static int ufshcd_abort ( struct scsi_cmnd * cmd )
{
struct Scsi_Host * host ;
struct ufs_hba * hba ;
unsigned long flags ;
unsigned int tag ;
2014-05-26 10:59:13 +05:30
int err = 0 ;
int poll_cnt ;
2014-05-26 10:59:12 +05:30
u8 resp = 0xF ;
struct ufshcd_lrb * lrbp ;
2014-07-01 12:22:37 +03:00
u32 reg ;
2012-02-29 12:11:50 +05:30
host = cmd - > device - > host ;
hba = shost_priv ( host ) ;
tag = cmd - > request - > tag ;
2014-05-26 10:59:13 +05:30
/* If command is already aborted/completed, return SUCCESS */
if ( ! ( test_bit ( tag , & hba - > outstanding_reqs ) ) )
goto out ;
2012-02-29 12:11:50 +05:30
2014-07-01 12:22:37 +03:00
reg = ufshcd_readl ( hba , REG_UTP_TRANSFER_REQ_DOOR_BELL ) ;
if ( ! ( reg & ( 1 < < tag ) ) ) {
dev_err ( hba - > dev ,
" %s: cmd was completed, but without a notifying intr, tag = %d " ,
__func__ , tag ) ;
}
2014-05-26 10:59:13 +05:30
lrbp = & hba - > lrb [ tag ] ;
for ( poll_cnt = 100 ; poll_cnt ; poll_cnt - - ) {
err = ufshcd_issue_tm_cmd ( hba , lrbp - > lun , lrbp - > task_tag ,
UFS_QUERY_TASK , & resp ) ;
if ( ! err & & resp = = UPIU_TASK_MANAGEMENT_FUNC_SUCCEEDED ) {
/* cmd pending in the device */
break ;
} else if ( ! err & & resp = = UPIU_TASK_MANAGEMENT_FUNC_COMPL ) {
/*
* cmd not pending in the device , check if it is
* in transition .
*/
reg = ufshcd_readl ( hba , REG_UTP_TRANSFER_REQ_DOOR_BELL ) ;
if ( reg & ( 1 < < tag ) ) {
/* sleep for max. 200us to stabilize */
usleep_range ( 100 , 200 ) ;
continue ;
}
/* command completed already */
goto out ;
} else {
if ( ! err )
err = resp ; /* service response error */
goto out ;
}
}
if ( ! poll_cnt ) {
err = - EBUSY ;
2012-02-29 12:11:50 +05:30
goto out ;
}
2014-05-26 10:59:12 +05:30
err = ufshcd_issue_tm_cmd ( hba , lrbp - > lun , lrbp - > task_tag ,
UFS_ABORT_TASK , & resp ) ;
if ( err | | resp ! = UPIU_TASK_MANAGEMENT_FUNC_COMPL ) {
2014-05-26 10:59:13 +05:30
if ( ! err )
err = resp ; /* service response error */
2012-02-29 12:11:50 +05:30
goto out ;
2014-05-26 10:59:12 +05:30
}
2012-02-29 12:11:50 +05:30
2014-05-26 10:59:13 +05:30
err = ufshcd_clear_cmd ( hba , tag ) ;
if ( err )
goto out ;
2012-02-29 12:11:50 +05:30
scsi_dma_unmap ( cmd ) ;
spin_lock_irqsave ( host - > host_lock , flags ) ;
__clear_bit ( tag , & hba - > outstanding_reqs ) ;
hba - > lrb [ tag ] . cmd = NULL ;
spin_unlock_irqrestore ( host - > host_lock , flags ) ;
2013-07-30 00:35:57 +05:30
clear_bit_unlock ( tag , & hba - > lrb_in_use ) ;
wake_up ( & hba - > dev_cmd . tag_wq ) ;
2012-02-29 12:11:50 +05:30
out :
2014-05-26 10:59:13 +05:30
if ( ! err ) {
err = SUCCESS ;
} else {
dev_err ( hba - > dev , " %s: failed with err %d \n " , __func__ , err ) ;
err = FAILED ;
}
2012-02-29 12:11:50 +05:30
return err ;
}
2014-05-26 10:59:14 +05:30
/**
* ufshcd_host_reset_and_restore - reset and restore host controller
* @ hba : per - adapter instance
*
* Note that host controller reset may issue DME_RESET to
* local and remote ( device ) Uni - Pro stack and the attributes
* are reset to default state .
*
* Returns zero on success , non - zero on failure
*/
static int ufshcd_host_reset_and_restore ( struct ufs_hba * hba )
{
int err ;
unsigned long flags ;
/* Reset the host controller */
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
ufshcd_hba_stop ( hba ) ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
err = ufshcd_hba_enable ( hba ) ;
if ( err )
goto out ;
/* Establish the link again and restore the device */
2014-09-25 15:32:26 +03:00
err = ufshcd_probe_hba ( hba ) ;
if ( ! err & & ( hba - > ufshcd_state ! = UFSHCD_STATE_OPERATIONAL ) )
2014-05-26 10:59:14 +05:30
err = - EIO ;
out :
if ( err )
dev_err ( hba - > dev , " %s: Host init failed %d \n " , __func__ , err ) ;
return err ;
}
/**
* ufshcd_reset_and_restore - reset and re - initialize host / device
* @ hba : per - adapter instance
*
* Reset and recover device , host and re - establish link . This
* is helpful to recover the communication in fatal error conditions .
*
* Returns zero on success , non - zero on failure
*/
static int ufshcd_reset_and_restore ( struct ufs_hba * hba )
{
int err = 0 ;
unsigned long flags ;
2014-09-25 15:32:26 +03:00
int retries = MAX_HOST_RESET_RETRIES ;
2014-05-26 10:59:14 +05:30
2014-09-25 15:32:26 +03:00
do {
err = ufshcd_host_reset_and_restore ( hba ) ;
} while ( err & & - - retries ) ;
2014-05-26 10:59:14 +05:30
/*
* After reset the door - bell might be cleared , complete
* outstanding requests in s / w here .
*/
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
ufshcd_transfer_req_compl ( hba ) ;
ufshcd_tmc_handler ( hba ) ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
return err ;
}
/**
* ufshcd_eh_host_reset_handler - host reset handler registered to scsi layer
* @ cmd - SCSI command pointer
*
* Returns SUCCESS / FAILED
*/
static int ufshcd_eh_host_reset_handler ( struct scsi_cmnd * cmd )
{
int err ;
unsigned long flags ;
struct ufs_hba * hba ;
hba = shost_priv ( cmd - > device - > host ) ;
/*
* Check if there is any race with fatal error handling .
* If so , wait for it to complete . Even though fatal error
* handling does reset and restore in some cases , don ' t assume
* anything out of it . We are just avoiding race here .
*/
do {
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
2014-05-26 10:59:15 +05:30
if ( ! ( work_pending ( & hba - > eh_work ) | |
2014-05-26 10:59:14 +05:30
hba - > ufshcd_state = = UFSHCD_STATE_RESET ) )
break ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
dev_dbg ( hba - > dev , " %s: reset in progress \n " , __func__ ) ;
2014-05-26 10:59:15 +05:30
flush_work ( & hba - > eh_work ) ;
2014-05-26 10:59:14 +05:30
} while ( 1 ) ;
hba - > ufshcd_state = UFSHCD_STATE_RESET ;
ufshcd_set_eh_in_progress ( hba ) ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
err = ufshcd_reset_and_restore ( hba ) ;
spin_lock_irqsave ( hba - > host - > host_lock , flags ) ;
if ( ! err ) {
err = SUCCESS ;
hba - > ufshcd_state = UFSHCD_STATE_OPERATIONAL ;
} else {
err = FAILED ;
hba - > ufshcd_state = UFSHCD_STATE_ERROR ;
}
ufshcd_clear_eh_in_progress ( hba ) ;
spin_unlock_irqrestore ( hba - > host - > host_lock , flags ) ;
return err ;
}
2014-09-25 15:32:27 +03:00
/**
* ufshcd_get_max_icc_level - calculate the ICC level
* @ sup_curr_uA : max . current supported by the regulator
* @ start_scan : row at the desc table to start scan from
* @ buff : power descriptor buffer
*
* Returns calculated max ICC level for specific regulator
*/
static u32 ufshcd_get_max_icc_level ( int sup_curr_uA , u32 start_scan , char * buff )
{
int i ;
int curr_uA ;
u16 data ;
u16 unit ;
for ( i = start_scan ; i > = 0 ; i - - ) {
data = be16_to_cpu ( * ( ( u16 * ) ( buff + 2 * i ) ) ) ;
unit = ( data & ATTR_ICC_LVL_UNIT_MASK ) > >
ATTR_ICC_LVL_UNIT_OFFSET ;
curr_uA = data & ATTR_ICC_LVL_VALUE_MASK ;
switch ( unit ) {
case UFSHCD_NANO_AMP :
curr_uA = curr_uA / 1000 ;
break ;
case UFSHCD_MILI_AMP :
curr_uA = curr_uA * 1000 ;
break ;
case UFSHCD_AMP :
curr_uA = curr_uA * 1000 * 1000 ;
break ;
case UFSHCD_MICRO_AMP :
default :
break ;
}
if ( sup_curr_uA > = curr_uA )
break ;
}
if ( i < 0 ) {
i = 0 ;
pr_err ( " %s: Couldn't find valid icc_level = %d " , __func__ , i ) ;
}
return ( u32 ) i ;
}
/**
* ufshcd_calc_icc_level - calculate the max ICC level
* In case regulators are not initialized we ' ll return 0
* @ hba : per - adapter instance
* @ desc_buf : power descriptor buffer to extract ICC levels from .
* @ len : length of desc_buff
*
* Returns calculated ICC level
*/
static u32 ufshcd_find_max_sup_active_icc_level ( struct ufs_hba * hba ,
u8 * desc_buf , int len )
{
u32 icc_level = 0 ;
if ( ! hba - > vreg_info . vcc | | ! hba - > vreg_info . vccq | |
! hba - > vreg_info . vccq2 ) {
dev_err ( hba - > dev ,
" %s: Regulator capability was not set, actvIccLevel=%d " ,
__func__ , icc_level ) ;
goto out ;
}
if ( hba - > vreg_info . vcc )
icc_level = ufshcd_get_max_icc_level (
hba - > vreg_info . vcc - > max_uA ,
POWER_DESC_MAX_ACTV_ICC_LVLS - 1 ,
& desc_buf [ PWR_DESC_ACTIVE_LVLS_VCC_0 ] ) ;
if ( hba - > vreg_info . vccq )
icc_level = ufshcd_get_max_icc_level (
hba - > vreg_info . vccq - > max_uA ,
icc_level ,
& desc_buf [ PWR_DESC_ACTIVE_LVLS_VCCQ_0 ] ) ;
if ( hba - > vreg_info . vccq2 )
icc_level = ufshcd_get_max_icc_level (
hba - > vreg_info . vccq2 - > max_uA ,
icc_level ,
& desc_buf [ PWR_DESC_ACTIVE_LVLS_VCCQ2_0 ] ) ;
out :
return icc_level ;
}
static void ufshcd_init_icc_levels ( struct ufs_hba * hba )
{
int ret ;
int buff_len = QUERY_DESC_POWER_MAX_SIZE ;
u8 desc_buf [ QUERY_DESC_POWER_MAX_SIZE ] ;
ret = ufshcd_read_power_desc ( hba , desc_buf , buff_len ) ;
if ( ret ) {
dev_err ( hba - > dev ,
" %s: Failed reading power descriptor.len = %d ret = %d " ,
__func__ , buff_len , ret ) ;
return ;
}
hba - > init_prefetch_data . icc_level =
ufshcd_find_max_sup_active_icc_level ( hba ,
desc_buf , buff_len ) ;
dev_dbg ( hba - > dev , " %s: setting icc_level 0x%x " ,
__func__ , hba - > init_prefetch_data . icc_level ) ;
ret = ufshcd_query_attr ( hba , UPIU_QUERY_OPCODE_WRITE_ATTR ,
QUERY_ATTR_IDN_ACTIVE_ICC_LVL , 0 , 0 ,
& hba - > init_prefetch_data . icc_level ) ;
if ( ret )
dev_err ( hba - > dev ,
" %s: Failed configuring bActiveICCLevel = %d ret = %d " ,
__func__ , hba - > init_prefetch_data . icc_level , ret ) ;
}
2014-09-25 15:32:28 +03:00
/**
* ufshcd_scsi_add_wlus - Adds required W - LUs
* @ hba : per - adapter instance
*
* UFS device specification requires the UFS devices to support 4 well known
* logical units :
* " REPORT_LUNS " ( address : 01 h )
* " UFS Device " ( address : 50 h )
* " RPMB " ( address : 44 h )
* " BOOT " ( address : 30 h )
* UFS device ' s power management needs to be controlled by " POWER CONDITION "
* field of SSU ( START STOP UNIT ) command . But this " power condition " field
* will take effect only when its sent to " UFS device " well known logical unit
* hence we require the scsi_device instance to represent this logical unit in
* order for the UFS host driver to send the SSU command for power management .
* We also require the scsi_device instance for " RPMB " ( Replay Protected Memory
* Block ) LU so user space process can control this LU . User space may also
* want to have access to BOOT LU .
* This function adds scsi device instances for each of all well known LUs
* ( except " REPORT LUNS " LU ) .
*
* Returns zero on success ( all required W - LUs are added successfully ) ,
* non - zero error value on failure ( if failed to add any of the required W - LU ) .
*/
static int ufshcd_scsi_add_wlus ( struct ufs_hba * hba )
{
int ret = 0 ;
hba - > sdev_ufs_device = __scsi_add_device ( hba - > host , 0 , 0 ,
ufshcd_upiu_wlun_to_scsi_wlun ( UFS_UPIU_UFS_DEVICE_WLUN ) , NULL ) ;
if ( IS_ERR ( hba - > sdev_ufs_device ) ) {
ret = PTR_ERR ( hba - > sdev_ufs_device ) ;
hba - > sdev_ufs_device = NULL ;
goto out ;
}
hba - > sdev_boot = __scsi_add_device ( hba - > host , 0 , 0 ,
ufshcd_upiu_wlun_to_scsi_wlun ( UFS_UPIU_BOOT_WLUN ) , NULL ) ;
if ( IS_ERR ( hba - > sdev_boot ) ) {
ret = PTR_ERR ( hba - > sdev_boot ) ;
hba - > sdev_boot = NULL ;
goto remove_sdev_ufs_device ;
}
hba - > sdev_rpmb = __scsi_add_device ( hba - > host , 0 , 0 ,
ufshcd_upiu_wlun_to_scsi_wlun ( UFS_UPIU_RPMB_WLUN ) , NULL ) ;
if ( IS_ERR ( hba - > sdev_rpmb ) ) {
ret = PTR_ERR ( hba - > sdev_rpmb ) ;
hba - > sdev_rpmb = NULL ;
goto remove_sdev_boot ;
}
goto out ;
remove_sdev_boot :
scsi_remove_device ( hba - > sdev_boot ) ;
remove_sdev_ufs_device :
scsi_remove_device ( hba - > sdev_ufs_device ) ;
out :
return ret ;
}
/**
* ufshcd_scsi_remove_wlus - Removes the W - LUs which were added by
* ufshcd_scsi_add_wlus ( )
* @ hba : per - adapter instance
*
*/
static void ufshcd_scsi_remove_wlus ( struct ufs_hba * hba )
{
if ( hba - > sdev_ufs_device ) {
scsi_remove_device ( hba - > sdev_ufs_device ) ;
hba - > sdev_ufs_device = NULL ;
}
if ( hba - > sdev_boot ) {
scsi_remove_device ( hba - > sdev_boot ) ;
hba - > sdev_boot = NULL ;
}
if ( hba - > sdev_rpmb ) {
scsi_remove_device ( hba - > sdev_rpmb ) ;
hba - > sdev_rpmb = NULL ;
}
}
2013-06-26 22:39:29 +05:30
/**
2014-09-25 15:32:26 +03:00
* ufshcd_probe_hba - probe hba to detect device and initialize
* @ hba : per - adapter instance
*
* Execute link - startup and verify device initialization
2013-06-26 22:39:29 +05:30
*/
2014-09-25 15:32:26 +03:00
static int ufshcd_probe_hba ( struct ufs_hba * hba )
2013-06-26 22:39:29 +05:30
{
int ret ;
ret = ufshcd_link_startup ( hba ) ;
2013-07-30 00:35:57 +05:30
if ( ret )
goto out ;
2013-08-31 21:40:24 +05:30
ufshcd_config_max_pwr_mode ( hba ) ;
2013-07-30 00:35:57 +05:30
ret = ufshcd_verify_dev_init ( hba ) ;
if ( ret )
goto out ;
2013-07-30 00:35:58 +05:30
ret = ufshcd_complete_dev_init ( hba ) ;
if ( ret )
goto out ;
2013-07-30 00:35:57 +05:30
2013-07-30 00:35:59 +05:30
ufshcd_force_reset_auto_bkops ( hba ) ;
2014-05-26 10:59:14 +05:30
hba - > ufshcd_state = UFSHCD_STATE_OPERATIONAL ;
/* If we are in error handling context no need to scan the host */
if ( ! ufshcd_eh_in_progress ( hba ) ) {
2014-09-25 15:32:27 +03:00
if ( ! hba - > is_init_prefetch )
ufshcd_init_icc_levels ( hba ) ;
2014-09-25 15:32:28 +03:00
/* Add required well known logical units to scsi mid layer */
if ( ufshcd_scsi_add_wlus ( hba ) )
goto out ;
2014-05-26 10:59:14 +05:30
scsi_scan_host ( hba - > host ) ;
pm_runtime_put_sync ( hba - > dev ) ;
}
2014-09-25 15:32:27 +03:00
if ( ! hba - > is_init_prefetch )
hba - > is_init_prefetch = true ;
2013-07-30 00:35:57 +05:30
out :
2014-09-25 15:32:26 +03:00
/*
* If we failed to initialize the device or the device is not
* present , turn off the power / clocks etc .
*/
if ( ret & & ! ufshcd_eh_in_progress ( hba ) )
ufshcd_hba_exit ( hba ) ;
return ret ;
}
/**
* ufshcd_async_scan - asynchronous execution for probing hba
* @ data : data pointer to pass to this function
* @ cookie : cookie data
*/
static void ufshcd_async_scan ( void * data , async_cookie_t cookie )
{
struct ufs_hba * hba = ( struct ufs_hba * ) data ;
ufshcd_probe_hba ( hba ) ;
2013-06-26 22:39:29 +05:30
}
2012-02-29 12:11:50 +05:30
static struct scsi_host_template ufshcd_driver_template = {
. module = THIS_MODULE ,
. name = UFSHCD ,
. proc_name = UFSHCD ,
. queuecommand = ufshcd_queuecommand ,
. slave_alloc = ufshcd_slave_alloc ,
2014-07-01 23:00:32 +09:00
. slave_configure = ufshcd_slave_configure ,
2012-02-29 12:11:50 +05:30
. slave_destroy = ufshcd_slave_destroy ,
2014-06-29 09:40:20 +03:00
. change_queue_depth = ufshcd_change_queue_depth ,
2012-02-29 12:11:50 +05:30
. eh_abort_handler = ufshcd_abort ,
2014-05-26 10:59:14 +05:30
. eh_device_reset_handler = ufshcd_eh_device_reset_handler ,
. eh_host_reset_handler = ufshcd_eh_host_reset_handler ,
2012-02-29 12:11:50 +05:30
. this_id = - 1 ,
. sg_tablesize = SG_ALL ,
. cmd_per_lun = UFSHCD_CMD_PER_LUN ,
. can_queue = UFSHCD_CAN_QUEUE ,
} ;
2014-09-25 15:32:22 +03:00
static int ufshcd_config_vreg ( struct device * dev ,
struct ufs_vreg * vreg , bool on )
{
int ret = 0 ;
struct regulator * reg = vreg - > reg ;
const char * name = vreg - > name ;
int min_uV , uA_load ;
BUG_ON ( ! vreg ) ;
if ( regulator_count_voltages ( reg ) > 0 ) {
min_uV = on ? vreg - > min_uV : 0 ;
ret = regulator_set_voltage ( reg , min_uV , vreg - > max_uV ) ;
if ( ret ) {
dev_err ( dev , " %s: %s set voltage failed, err=%d \n " ,
__func__ , name , ret ) ;
goto out ;
}
uA_load = on ? vreg - > max_uA : 0 ;
ret = regulator_set_optimum_mode ( reg , uA_load ) ;
if ( ret > = 0 ) {
/*
* regulator_set_optimum_mode ( ) returns new regulator
* mode upon success .
*/
ret = 0 ;
} else {
dev_err ( dev , " %s: %s set optimum mode(uA_load=%d) failed, err=%d \n " ,
__func__ , name , uA_load , ret ) ;
goto out ;
}
}
out :
return ret ;
}
static int ufshcd_enable_vreg ( struct device * dev , struct ufs_vreg * vreg )
{
int ret = 0 ;
if ( ! vreg | | vreg - > enabled )
goto out ;
ret = ufshcd_config_vreg ( dev , vreg , true ) ;
if ( ! ret )
ret = regulator_enable ( vreg - > reg ) ;
if ( ! ret )
vreg - > enabled = true ;
else
dev_err ( dev , " %s: %s enable failed, err=%d \n " ,
__func__ , vreg - > name , ret ) ;
out :
return ret ;
}
static int ufshcd_disable_vreg ( struct device * dev , struct ufs_vreg * vreg )
{
int ret = 0 ;
if ( ! vreg | | ! vreg - > enabled )
goto out ;
ret = regulator_disable ( vreg - > reg ) ;
if ( ! ret ) {
/* ignore errors on applying disable config */
ufshcd_config_vreg ( dev , vreg , false ) ;
vreg - > enabled = false ;
} else {
dev_err ( dev , " %s: %s disable failed, err=%d \n " ,
__func__ , vreg - > name , ret ) ;
}
out :
return ret ;
}
static int ufshcd_setup_vreg ( struct ufs_hba * hba , bool on )
{
int ret = 0 ;
struct device * dev = hba - > dev ;
struct ufs_vreg_info * info = & hba - > vreg_info ;
if ( ! info )
goto out ;
ret = ufshcd_toggle_vreg ( dev , info - > vcc , on ) ;
if ( ret )
goto out ;
ret = ufshcd_toggle_vreg ( dev , info - > vccq , on ) ;
if ( ret )
goto out ;
ret = ufshcd_toggle_vreg ( dev , info - > vccq2 , on ) ;
if ( ret )
goto out ;
out :
if ( ret ) {
ufshcd_toggle_vreg ( dev , info - > vccq2 , false ) ;
ufshcd_toggle_vreg ( dev , info - > vccq , false ) ;
ufshcd_toggle_vreg ( dev , info - > vcc , false ) ;
}
return ret ;
}
2014-09-25 15:32:24 +03:00
static int ufshcd_setup_hba_vreg ( struct ufs_hba * hba , bool on )
{
struct ufs_vreg_info * info = & hba - > vreg_info ;
if ( info )
return ufshcd_toggle_vreg ( hba - > dev , info - > vdd_hba , on ) ;
return 0 ;
}
2014-09-25 15:32:22 +03:00
static int ufshcd_get_vreg ( struct device * dev , struct ufs_vreg * vreg )
{
int ret = 0 ;
if ( ! vreg )
goto out ;
vreg - > reg = devm_regulator_get ( dev , vreg - > name ) ;
if ( IS_ERR ( vreg - > reg ) ) {
ret = PTR_ERR ( vreg - > reg ) ;
dev_err ( dev , " %s: %s get failed, err=%d \n " ,
__func__ , vreg - > name , ret ) ;
}
out :
return ret ;
}
static int ufshcd_init_vreg ( struct ufs_hba * hba )
{
int ret = 0 ;
struct device * dev = hba - > dev ;
struct ufs_vreg_info * info = & hba - > vreg_info ;
if ( ! info )
goto out ;
ret = ufshcd_get_vreg ( dev , info - > vcc ) ;
if ( ret )
goto out ;
ret = ufshcd_get_vreg ( dev , info - > vccq ) ;
if ( ret )
goto out ;
ret = ufshcd_get_vreg ( dev , info - > vccq2 ) ;
out :
return ret ;
}
2014-09-25 15:32:24 +03:00
static int ufshcd_init_hba_vreg ( struct ufs_hba * hba )
{
struct ufs_vreg_info * info = & hba - > vreg_info ;
if ( info )
return ufshcd_get_vreg ( hba - > dev , info - > vdd_hba ) ;
return 0 ;
}
2014-09-25 15:32:23 +03:00
static int ufshcd_setup_clocks ( struct ufs_hba * hba , bool on )
{
int ret = 0 ;
struct ufs_clk_info * clki ;
struct list_head * head = & hba - > clk_list_head ;
if ( ! head | | list_empty ( head ) )
goto out ;
list_for_each_entry ( clki , head , list ) {
if ( ! IS_ERR_OR_NULL ( clki - > clk ) ) {
if ( on & & ! clki - > enabled ) {
ret = clk_prepare_enable ( clki - > clk ) ;
if ( ret ) {
dev_err ( hba - > dev , " %s: %s prepare enable failed, %d \n " ,
__func__ , clki - > name , ret ) ;
goto out ;
}
} else if ( ! on & & clki - > enabled ) {
clk_disable_unprepare ( clki - > clk ) ;
}
clki - > enabled = on ;
dev_dbg ( hba - > dev , " %s: clk: %s %sabled \n " , __func__ ,
clki - > name , on ? " en " : " dis " ) ;
}
}
out :
if ( ret ) {
list_for_each_entry ( clki , head , list ) {
if ( ! IS_ERR_OR_NULL ( clki - > clk ) & & clki - > enabled )
clk_disable_unprepare ( clki - > clk ) ;
}
}
return ret ;
}
static int ufshcd_init_clocks ( struct ufs_hba * hba )
{
int ret = 0 ;
struct ufs_clk_info * clki ;
struct device * dev = hba - > dev ;
struct list_head * head = & hba - > clk_list_head ;
if ( ! head | | list_empty ( head ) )
goto out ;
list_for_each_entry ( clki , head , list ) {
if ( ! clki - > name )
continue ;
clki - > clk = devm_clk_get ( dev , clki - > name ) ;
if ( IS_ERR ( clki - > clk ) ) {
ret = PTR_ERR ( clki - > clk ) ;
dev_err ( dev , " %s: %s clk get failed, %d \n " ,
__func__ , clki - > name , ret ) ;
goto out ;
}
if ( clki - > max_freq ) {
ret = clk_set_rate ( clki - > clk , clki - > max_freq ) ;
if ( ret ) {
dev_err ( hba - > dev , " %s: %s clk set rate(%dHz) failed, %d \n " ,
__func__ , clki - > name ,
clki - > max_freq , ret ) ;
goto out ;
}
}
dev_dbg ( dev , " %s: clk: %s, rate: %lu \n " , __func__ ,
clki - > name , clk_get_rate ( clki - > clk ) ) ;
}
out :
return ret ;
}
2014-09-25 15:32:21 +03:00
static int ufshcd_variant_hba_init ( struct ufs_hba * hba )
{
int err = 0 ;
if ( ! hba - > vops )
goto out ;
if ( hba - > vops - > init ) {
err = hba - > vops - > init ( hba ) ;
if ( err )
goto out ;
}
if ( hba - > vops - > setup_clocks ) {
err = hba - > vops - > setup_clocks ( hba , true ) ;
if ( err )
goto out_exit ;
}
if ( hba - > vops - > setup_regulators ) {
err = hba - > vops - > setup_regulators ( hba , true ) ;
if ( err )
goto out_clks ;
}
goto out ;
out_clks :
if ( hba - > vops - > setup_clocks )
hba - > vops - > setup_clocks ( hba , false ) ;
out_exit :
if ( hba - > vops - > exit )
hba - > vops - > exit ( hba ) ;
out :
if ( err )
dev_err ( hba - > dev , " %s: variant %s init failed err %d \n " ,
__func__ , hba - > vops ? hba - > vops - > name : " " , err ) ;
return err ;
}
static void ufshcd_variant_hba_exit ( struct ufs_hba * hba )
{
if ( ! hba - > vops )
return ;
if ( hba - > vops - > setup_clocks )
hba - > vops - > setup_clocks ( hba , false ) ;
if ( hba - > vops - > setup_regulators )
hba - > vops - > setup_regulators ( hba , false ) ;
if ( hba - > vops - > exit )
hba - > vops - > exit ( hba ) ;
}
2014-09-25 15:32:22 +03:00
static int ufshcd_hba_init ( struct ufs_hba * hba )
{
int err ;
2014-09-25 15:32:24 +03:00
/*
* Handle host controller power separately from the UFS device power
* rails as it will help controlling the UFS host controller power
* collapse easily which is different than UFS device power collapse .
* Also , enable the host controller power before we go ahead with rest
* of the initialization here .
*/
err = ufshcd_init_hba_vreg ( hba ) ;
2014-09-25 15:32:22 +03:00
if ( err )
goto out ;
2014-09-25 15:32:24 +03:00
err = ufshcd_setup_hba_vreg ( hba , true ) ;
2014-09-25 15:32:22 +03:00
if ( err )
goto out ;
2014-09-25 15:32:24 +03:00
err = ufshcd_init_clocks ( hba ) ;
if ( err )
goto out_disable_hba_vreg ;
err = ufshcd_setup_clocks ( hba , true ) ;
if ( err )
goto out_disable_hba_vreg ;
2014-09-25 15:32:23 +03:00
err = ufshcd_init_vreg ( hba ) ;
if ( err )
goto out_disable_clks ;
err = ufshcd_setup_vreg ( hba , true ) ;
if ( err )
goto out_disable_clks ;
2014-09-25 15:32:22 +03:00
err = ufshcd_variant_hba_init ( hba ) ;
if ( err )
goto out_disable_vreg ;
2014-09-25 15:32:26 +03:00
hba - > is_powered = true ;
2014-09-25 15:32:22 +03:00
goto out ;
out_disable_vreg :
ufshcd_setup_vreg ( hba , false ) ;
2014-09-25 15:32:23 +03:00
out_disable_clks :
ufshcd_setup_clocks ( hba , false ) ;
2014-09-25 15:32:24 +03:00
out_disable_hba_vreg :
ufshcd_setup_hba_vreg ( hba , false ) ;
2014-09-25 15:32:22 +03:00
out :
return err ;
}
static void ufshcd_hba_exit ( struct ufs_hba * hba )
{
2014-09-25 15:32:26 +03:00
if ( hba - > is_powered ) {
ufshcd_variant_hba_exit ( hba ) ;
ufshcd_setup_vreg ( hba , false ) ;
ufshcd_setup_clocks ( hba , false ) ;
ufshcd_setup_hba_vreg ( hba , false ) ;
hba - > is_powered = false ;
}
2014-09-25 15:32:22 +03:00
}
2012-02-29 12:11:50 +05:30
/**
* ufshcd_suspend - suspend power management function
2013-02-25 21:44:32 +05:30
* @ hba : per adapter instance
2012-02-29 12:11:50 +05:30
* @ state : power state
*
* Returns - ENOSYS
*/
2013-02-25 21:44:32 +05:30
int ufshcd_suspend ( struct ufs_hba * hba , pm_message_t state )
2012-02-29 12:11:50 +05:30
{
/*
* TODO :
* 1. Block SCSI requests from SCSI midlayer
* 2. Change the internal driver state to non operational
* 3. Set UTRLRSR and UTMRLRSR bits to zero
* 4. Wait until outstanding commands are completed
* 5. Set HCE to zero to send the UFS host controller to reset state
*/
return - ENOSYS ;
}
2013-02-25 21:44:32 +05:30
EXPORT_SYMBOL_GPL ( ufshcd_suspend ) ;
2012-02-29 12:11:50 +05:30
/**
* ufshcd_resume - resume power management function
2013-02-25 21:44:32 +05:30
* @ hba : per adapter instance
2012-02-29 12:11:50 +05:30
*
* Returns - ENOSYS
*/
2013-02-25 21:44:32 +05:30
int ufshcd_resume ( struct ufs_hba * hba )
2012-02-29 12:11:50 +05:30
{
/*
* TODO :
* 1. Set HCE to 1 , to start the UFS host controller
* initialization process
* 2. Set UTRLRSR and UTMRLRSR bits to 1
* 3. Change the internal driver state to operational
* 4. Unblock SCSI requests from SCSI midlayer
*/
return - ENOSYS ;
}
2013-02-25 21:44:32 +05:30
EXPORT_SYMBOL_GPL ( ufshcd_resume ) ;
2013-07-30 00:35:59 +05:30
int ufshcd_runtime_suspend ( struct ufs_hba * hba )
{
if ( ! hba )
return 0 ;
/*
* The device is idle with no requests in the queue ,
* allow background operations .
*/
return ufshcd_enable_auto_bkops ( hba ) ;
}
EXPORT_SYMBOL ( ufshcd_runtime_suspend ) ;
int ufshcd_runtime_resume ( struct ufs_hba * hba )
{
if ( ! hba )
return 0 ;
return ufshcd_disable_auto_bkops ( hba ) ;
}
EXPORT_SYMBOL ( ufshcd_runtime_resume ) ;
int ufshcd_runtime_idle ( struct ufs_hba * hba )
{
return 0 ;
}
EXPORT_SYMBOL ( ufshcd_runtime_idle ) ;
2012-02-29 12:11:50 +05:30
/**
2013-02-25 21:44:32 +05:30
* ufshcd_remove - de - allocate SCSI host and host memory space
2012-02-29 12:11:50 +05:30
* data structure memory
2013-02-25 21:44:32 +05:30
* @ hba - per adapter instance
2012-02-29 12:11:50 +05:30
*/
2013-02-25 21:44:32 +05:30
void ufshcd_remove ( struct ufs_hba * hba )
2012-02-29 12:11:50 +05:30
{
2013-07-30 00:36:03 +05:30
scsi_remove_host ( hba - > host ) ;
2014-09-25 15:32:28 +03:00
ufshcd_scsi_remove_wlus ( hba ) ;
2012-02-29 12:11:50 +05:30
/* disable interrupts */
2013-06-26 22:39:27 +05:30
ufshcd_disable_intr ( hba , hba - > intr_mask ) ;
2012-02-29 12:11:50 +05:30
ufshcd_hba_stop ( hba ) ;
scsi_host_put ( hba - > host ) ;
2014-09-25 15:32:21 +03:00
2014-09-25 15:32:22 +03:00
ufshcd_hba_exit ( hba ) ;
2013-02-25 21:44:32 +05:30
}
EXPORT_SYMBOL_GPL ( ufshcd_remove ) ;
2014-07-13 21:24:46 +09:00
/**
* ufshcd_set_dma_mask - Set dma mask based on the controller
* addressing capability
* @ hba : per adapter instance
*
* Returns 0 for success , non - zero for failure
*/
static int ufshcd_set_dma_mask ( struct ufs_hba * hba )
{
if ( hba - > capabilities & MASK_64_ADDRESSING_SUPPORT ) {
if ( ! dma_set_mask_and_coherent ( hba - > dev , DMA_BIT_MASK ( 64 ) ) )
return 0 ;
}
return dma_set_mask_and_coherent ( hba - > dev , DMA_BIT_MASK ( 32 ) ) ;
}
2012-02-29 12:11:50 +05:30
/**
2014-09-25 15:32:21 +03:00
* ufshcd_alloc_host - allocate Host Bus Adapter ( HBA )
2013-02-25 21:44:32 +05:30
* @ dev : pointer to device handle
* @ hba_handle : driver private handle
2012-02-29 12:11:50 +05:30
* Returns 0 on success , non - zero value on failure
*/
2014-09-25 15:32:21 +03:00
int ufshcd_alloc_host ( struct device * dev , struct ufs_hba * * hba_handle )
2012-02-29 12:11:50 +05:30
{
struct Scsi_Host * host ;
struct ufs_hba * hba ;
2014-09-25 15:32:21 +03:00
int err = 0 ;
2012-02-29 12:11:50 +05:30
2013-02-25 21:44:32 +05:30
if ( ! dev ) {
dev_err ( dev ,
" Invalid memory reference for dev is NULL \n " ) ;
err = - ENODEV ;
2012-02-29 12:11:50 +05:30
goto out_error ;
}
host = scsi_host_alloc ( & ufshcd_driver_template ,
sizeof ( struct ufs_hba ) ) ;
if ( ! host ) {
2013-02-25 21:44:32 +05:30
dev_err ( dev , " scsi_host_alloc failed \n " ) ;
2012-02-29 12:11:50 +05:30
err = - ENOMEM ;
2013-02-25 21:44:32 +05:30
goto out_error ;
2012-02-29 12:11:50 +05:30
}
hba = shost_priv ( host ) ;
hba - > host = host ;
2013-02-25 21:44:32 +05:30
hba - > dev = dev ;
2014-09-25 15:32:21 +03:00
* hba_handle = hba ;
out_error :
return err ;
}
EXPORT_SYMBOL ( ufshcd_alloc_host ) ;
/**
* ufshcd_init - Driver initialization routine
* @ hba : per - adapter instance
* @ mmio_base : base register address
* @ irq : Interrupt line of device
* Returns 0 on success , non - zero value on failure
*/
int ufshcd_init ( struct ufs_hba * hba , void __iomem * mmio_base , unsigned int irq )
{
int err ;
struct Scsi_Host * host = hba - > host ;
struct device * dev = hba - > dev ;
if ( ! mmio_base ) {
dev_err ( hba - > dev ,
" Invalid memory reference for mmio_base is NULL \n " ) ;
err = - ENODEV ;
goto out_error ;
}
2013-02-25 21:44:32 +05:30
hba - > mmio_base = mmio_base ;
hba - > irq = irq ;
2012-02-29 12:11:50 +05:30
2014-09-25 15:32:22 +03:00
err = ufshcd_hba_init ( hba ) ;
2014-09-25 15:32:21 +03:00
if ( err )
goto out_error ;
2012-02-29 12:11:50 +05:30
/* Read capabilities registers */
ufshcd_hba_capabilities ( hba ) ;
/* Get UFS version supported by the controller */
hba - > ufs_version = ufshcd_get_ufs_version ( hba ) ;
2013-06-26 22:39:27 +05:30
/* Get Interrupt bit mask per version */
hba - > intr_mask = ufshcd_get_intr_mask ( hba ) ;
2014-07-13 21:24:46 +09:00
err = ufshcd_set_dma_mask ( hba ) ;
if ( err ) {
dev_err ( hba - > dev , " set dma mask failed \n " ) ;
goto out_disable ;
}
2012-02-29 12:11:50 +05:30
/* Allocate memory for host memory space */
err = ufshcd_memory_alloc ( hba ) ;
if ( err ) {
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev , " Memory allocation failed \n " ) ;
goto out_disable ;
2012-02-29 12:11:50 +05:30
}
/* Configure LRB */
ufshcd_host_memory_configure ( hba ) ;
host - > can_queue = hba - > nutrs ;
host - > cmd_per_lun = hba - > nutrs ;
host - > max_id = UFSHCD_MAX_ID ;
2014-09-25 15:32:29 +03:00
host - > max_lun = UFS_MAX_LUNS ;
2012-02-29 12:11:50 +05:30
host - > max_channel = UFSHCD_MAX_CHANNEL ;
host - > unique_id = host - > host_no ;
host - > max_cmd_len = MAX_CDB_SIZE ;
/* Initailize wait queue for task management */
2014-05-26 10:59:12 +05:30
init_waitqueue_head ( & hba - > tm_wq ) ;
init_waitqueue_head ( & hba - > tm_tag_wq ) ;
2012-02-29 12:11:50 +05:30
/* Initialize work queues */
2014-05-26 10:59:15 +05:30
INIT_WORK ( & hba - > eh_work , ufshcd_err_handler ) ;
2013-07-30 00:35:59 +05:30
INIT_WORK ( & hba - > eeh_work , ufshcd_exception_event_handler ) ;
2012-02-29 12:11:50 +05:30
2013-06-26 22:39:29 +05:30
/* Initialize UIC command mutex */
mutex_init ( & hba - > uic_cmd_mutex ) ;
2013-07-30 00:35:57 +05:30
/* Initialize mutex for device management commands */
mutex_init ( & hba - > dev_cmd . lock ) ;
/* Initialize device management tag acquire wait queue */
init_waitqueue_head ( & hba - > dev_cmd . tag_wq ) ;
2012-02-29 12:11:50 +05:30
/* IRQ registration */
2013-06-27 13:31:54 +09:00
err = devm_request_irq ( dev , irq , ufshcd_intr , IRQF_SHARED , UFSHCD , hba ) ;
2012-02-29 12:11:50 +05:30
if ( err ) {
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev , " request irq failed \n " ) ;
2013-06-27 13:31:54 +09:00
goto out_disable ;
2012-02-29 12:11:50 +05:30
}
/* Enable SCSI tag mapping */
err = scsi_init_shared_tag_map ( host , host - > can_queue ) ;
if ( err ) {
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev , " init shared queue failed \n " ) ;
2013-06-27 13:31:54 +09:00
goto out_disable ;
2012-02-29 12:11:50 +05:30
}
2013-02-25 21:44:32 +05:30
err = scsi_add_host ( host , hba - > dev ) ;
2012-02-29 12:11:50 +05:30
if ( err ) {
2013-02-25 21:44:32 +05:30
dev_err ( hba - > dev , " scsi_add_host failed \n " ) ;
2013-06-27 13:31:54 +09:00
goto out_disable ;
2012-02-29 12:11:50 +05:30
}
2013-06-26 22:39:29 +05:30
/* Host controller enable */
err = ufshcd_hba_enable ( hba ) ;
2012-02-29 12:11:50 +05:30
if ( err ) {
2013-06-26 22:39:29 +05:30
dev_err ( hba - > dev , " Host controller enable failed \n " ) ;
2013-02-25 21:44:32 +05:30
goto out_remove_scsi_host ;
2012-02-29 12:11:50 +05:30
}
2013-06-26 22:39:29 +05:30
2013-07-30 00:36:00 +05:30
/* Hold auto suspend until async scan completes */
pm_runtime_get_sync ( dev ) ;
2013-06-26 22:39:29 +05:30
async_schedule ( ufshcd_async_scan , hba ) ;
2012-02-29 12:11:50 +05:30
return 0 ;
2013-02-25 21:44:32 +05:30
out_remove_scsi_host :
scsi_remove_host ( hba - > host ) ;
out_disable :
scsi_host_put ( host ) ;
2014-09-25 15:32:22 +03:00
ufshcd_hba_exit ( hba ) ;
2013-02-25 21:44:32 +05:30
out_error :
return err ;
}
EXPORT_SYMBOL_GPL ( ufshcd_init ) ;
MODULE_AUTHOR ( " Santosh Yaragnavi <santosh.sy@samsung.com> " ) ;
MODULE_AUTHOR ( " Vinayak Holikatti <h.vinayak@samsung.com> " ) ;
2013-02-25 21:44:33 +05:30
MODULE_DESCRIPTION ( " Generic UFS host controller driver Core " ) ;
2012-02-29 12:11:50 +05:30
MODULE_LICENSE ( " GPL " ) ;
MODULE_VERSION ( UFSHCD_DRIVER_VERSION ) ;