mirror of https://gitee.com/openkylin/linux.git
scsi: ufs: introduce UFSHCD_QUIRK_DME_PEER_ACCESS_AUTO_MODE quirk
Some UFS host controllers may only allow accessing the peer DME attribute in AUTO mode (FAST AUTO or SLOW AUTO) hence we had added a quirk for switching to AUTO power mode before accessing the peer DME attribute. Signed-off-by: Yaniv Gardi <ygardi@codeaurora.org> Reviewed-by: Akinobu Mita <akinobu.mita@gmail.com> Signed-off-by: James Bottomley <JBottomley@Odin.com>
This commit is contained in:
parent
8163743e23
commit
874237f7f2
|
@ -188,6 +188,8 @@ static int ufshcd_host_reset_and_restore(struct ufs_hba *hba);
|
||||||
static irqreturn_t ufshcd_intr(int irq, void *__hba);
|
static irqreturn_t ufshcd_intr(int irq, void *__hba);
|
||||||
static int ufshcd_config_pwr_mode(struct ufs_hba *hba,
|
static int ufshcd_config_pwr_mode(struct ufs_hba *hba,
|
||||||
struct ufs_pa_layer_attr *desired_pwr_mode);
|
struct ufs_pa_layer_attr *desired_pwr_mode);
|
||||||
|
static int ufshcd_change_power_mode(struct ufs_hba *hba,
|
||||||
|
struct ufs_pa_layer_attr *pwr_mode);
|
||||||
|
|
||||||
static inline int ufshcd_enable_irq(struct ufs_hba *hba)
|
static inline int ufshcd_enable_irq(struct ufs_hba *hba)
|
||||||
{
|
{
|
||||||
|
@ -2156,6 +2158,31 @@ int ufshcd_dme_get_attr(struct ufs_hba *hba, u32 attr_sel,
|
||||||
};
|
};
|
||||||
const char *get = action[!!peer];
|
const char *get = action[!!peer];
|
||||||
int ret;
|
int ret;
|
||||||
|
struct ufs_pa_layer_attr orig_pwr_info;
|
||||||
|
struct ufs_pa_layer_attr temp_pwr_info;
|
||||||
|
bool pwr_mode_change = false;
|
||||||
|
|
||||||
|
if (peer && (hba->quirks & UFSHCD_QUIRK_DME_PEER_ACCESS_AUTO_MODE)) {
|
||||||
|
orig_pwr_info = hba->pwr_info;
|
||||||
|
temp_pwr_info = orig_pwr_info;
|
||||||
|
|
||||||
|
if (orig_pwr_info.pwr_tx == FAST_MODE ||
|
||||||
|
orig_pwr_info.pwr_rx == FAST_MODE) {
|
||||||
|
temp_pwr_info.pwr_tx = FASTAUTO_MODE;
|
||||||
|
temp_pwr_info.pwr_rx = FASTAUTO_MODE;
|
||||||
|
pwr_mode_change = true;
|
||||||
|
} else if (orig_pwr_info.pwr_tx == SLOW_MODE ||
|
||||||
|
orig_pwr_info.pwr_rx == SLOW_MODE) {
|
||||||
|
temp_pwr_info.pwr_tx = SLOWAUTO_MODE;
|
||||||
|
temp_pwr_info.pwr_rx = SLOWAUTO_MODE;
|
||||||
|
pwr_mode_change = true;
|
||||||
|
}
|
||||||
|
if (pwr_mode_change) {
|
||||||
|
ret = ufshcd_change_power_mode(hba, &temp_pwr_info);
|
||||||
|
if (ret)
|
||||||
|
goto out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
uic_cmd.command = peer ?
|
uic_cmd.command = peer ?
|
||||||
UIC_CMD_DME_PEER_GET : UIC_CMD_DME_GET;
|
UIC_CMD_DME_PEER_GET : UIC_CMD_DME_GET;
|
||||||
|
@ -2170,6 +2197,10 @@ int ufshcd_dme_get_attr(struct ufs_hba *hba, u32 attr_sel,
|
||||||
|
|
||||||
if (mib_val)
|
if (mib_val)
|
||||||
*mib_val = uic_cmd.argument3;
|
*mib_val = uic_cmd.argument3;
|
||||||
|
|
||||||
|
if (peer && (hba->quirks & UFSHCD_QUIRK_DME_PEER_ACCESS_AUTO_MODE)
|
||||||
|
&& pwr_mode_change)
|
||||||
|
ufshcd_change_power_mode(hba, &orig_pwr_info);
|
||||||
out:
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
|
@ -442,6 +442,13 @@ struct ufs_hba {
|
||||||
*/
|
*/
|
||||||
#define UFSHCD_QUIRK_BROKEN_PA_RXHSUNTERMCAP UFS_BIT(3)
|
#define UFSHCD_QUIRK_BROKEN_PA_RXHSUNTERMCAP UFS_BIT(3)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This quirk needs to be enabled if the host contoller only allows
|
||||||
|
* accessing the peer dme attributes in AUTO mode (FAST AUTO or
|
||||||
|
* SLOW AUTO).
|
||||||
|
*/
|
||||||
|
#define UFSHCD_QUIRK_DME_PEER_ACCESS_AUTO_MODE UFS_BIT(4)
|
||||||
|
|
||||||
unsigned int quirks; /* Deviations from standard UFSHCI spec. */
|
unsigned int quirks; /* Deviations from standard UFSHCI spec. */
|
||||||
|
|
||||||
wait_queue_head_t tm_wq;
|
wait_queue_head_t tm_wq;
|
||||||
|
|
Loading…
Reference in New Issue