提交 2355b66e 编写于 作者: C Can Guo 提交者: Martin K. Petersen

scsi: ufs: Handle LINERESET indication in err handler

PA Layer issues a LINERESET to the PHY at the recovery step in the Power
Mode change operation. If it happens during auto or manual hibern8 enter,
even if hibern8 enter succeeds, UFS power mode shall be set to PWM-G1 mode
and kept in that mode after exit from hibern8, leading to bad performance.
Handle the LINERESET in the eh_work by restoring power mode to HS mode
after all pending reqs and tasks are cleared from doorbell.

Link: https://lore.kernel.org/r/1598321228-21093-3-git-send-email-cang@codeaurora.orgSigned-off-by: NCan Guo <cang@codeaurora.org>
Signed-off-by: NMartin K. Petersen <martin.petersen@oracle.com>
上级 307348f6
...@@ -146,6 +146,7 @@ enum { ...@@ -146,6 +146,7 @@ enum {
UFSHCD_UIC_NL_ERROR = (1 << 3), /* Network layer error */ UFSHCD_UIC_NL_ERROR = (1 << 3), /* Network layer error */
UFSHCD_UIC_TL_ERROR = (1 << 4), /* Transport Layer error */ UFSHCD_UIC_TL_ERROR = (1 << 4), /* Transport Layer error */
UFSHCD_UIC_DME_ERROR = (1 << 5), /* DME error */ UFSHCD_UIC_DME_ERROR = (1 << 5), /* DME error */
UFSHCD_UIC_PA_GENERIC_ERROR = (1 << 6), /* Generic PA error */
}; };
#define ufshcd_set_eh_in_progress(h) \ #define ufshcd_set_eh_in_progress(h) \
...@@ -4072,7 +4073,8 @@ static int ufshcd_change_power_mode(struct ufs_hba *hba, ...@@ -4072,7 +4073,8 @@ static int ufshcd_change_power_mode(struct ufs_hba *hba,
int ret; int ret;
/* if already configured to the requested pwr_mode */ /* if already configured to the requested pwr_mode */
if (pwr_mode->gear_rx == hba->pwr_info.gear_rx && if (!hba->force_pmc &&
pwr_mode->gear_rx == hba->pwr_info.gear_rx &&
pwr_mode->gear_tx == hba->pwr_info.gear_tx && pwr_mode->gear_tx == hba->pwr_info.gear_tx &&
pwr_mode->lane_rx == hba->pwr_info.lane_rx && pwr_mode->lane_rx == hba->pwr_info.lane_rx &&
pwr_mode->lane_tx == hba->pwr_info.lane_tx && pwr_mode->lane_tx == hba->pwr_info.lane_tx &&
...@@ -4506,6 +4508,8 @@ static int ufshcd_link_startup(struct ufs_hba *hba) ...@@ -4506,6 +4508,8 @@ static int ufshcd_link_startup(struct ufs_hba *hba)
if (ret) if (ret)
goto out; goto out;
/* Clear UECPA once due to LINERESET has happened during LINK_STARTUP */
ufshcd_readl(hba, REG_UIC_ERROR_CODE_PHY_ADAPTER_LAYER);
ret = ufshcd_make_hba_operational(hba); ret = ufshcd_make_hba_operational(hba);
out: out:
if (ret) { if (ret) {
...@@ -5665,6 +5669,22 @@ static inline void ufshcd_recover_pm_error(struct ufs_hba *hba) ...@@ -5665,6 +5669,22 @@ static inline void ufshcd_recover_pm_error(struct ufs_hba *hba)
} }
#endif #endif
static bool ufshcd_is_pwr_mode_restore_needed(struct ufs_hba *hba)
{
struct ufs_pa_layer_attr *pwr_info = &hba->pwr_info;
u32 mode;
ufshcd_dme_get(hba, UIC_ARG_MIB(PA_PWRMODE), &mode);
if (pwr_info->pwr_rx != ((mode >> PWRMODE_RX_OFFSET) & PWRMODE_MASK))
return true;
if (pwr_info->pwr_tx != (mode & PWRMODE_MASK))
return true;
return false;
}
/** /**
* ufshcd_err_handler - handle UFS errors that require s/w attention * ufshcd_err_handler - handle UFS errors that require s/w attention
* @work: pointer to work structure * @work: pointer to work structure
...@@ -5675,9 +5695,9 @@ static void ufshcd_err_handler(struct work_struct *work) ...@@ -5675,9 +5695,9 @@ static void ufshcd_err_handler(struct work_struct *work)
unsigned long flags; unsigned long flags;
bool err_xfer = false; bool err_xfer = false;
bool err_tm = false; bool err_tm = false;
int err = 0; int err = 0, pmc_err;
int tag; int tag;
bool needs_reset = false; bool needs_reset = false, needs_restore = false;
hba = container_of(work, struct ufs_hba, eh_work); hba = container_of(work, struct ufs_hba, eh_work);
...@@ -5725,8 +5745,9 @@ static void ufshcd_err_handler(struct work_struct *work) ...@@ -5725,8 +5745,9 @@ static void ufshcd_err_handler(struct work_struct *work)
UFSHCD_UIC_DL_TCx_REPLAY_ERROR)))) UFSHCD_UIC_DL_TCx_REPLAY_ERROR))))
needs_reset = true; needs_reset = true;
if (hba->saved_err & (INT_FATAL_ERRORS | UIC_ERROR | if ((hba->saved_err & (INT_FATAL_ERRORS | UFSHCD_UIC_HIBERN8_MASK)) ||
UFSHCD_UIC_HIBERN8_MASK)) { (hba->saved_uic_err &&
(hba->saved_uic_err != UFSHCD_UIC_PA_GENERIC_ERROR))) {
bool pr_prdt = !!(hba->saved_err & SYSTEM_BUS_FATAL_ERROR); bool pr_prdt = !!(hba->saved_err & SYSTEM_BUS_FATAL_ERROR);
spin_unlock_irqrestore(hba->host->host_lock, flags); spin_unlock_irqrestore(hba->host->host_lock, flags);
...@@ -5744,8 +5765,25 @@ static void ufshcd_err_handler(struct work_struct *work) ...@@ -5744,8 +5765,25 @@ static void ufshcd_err_handler(struct work_struct *work)
* host reset and restore * host reset and restore
*/ */
if (needs_reset) if (needs_reset)
goto skip_pending_xfer_clear; goto do_reset;
/*
* If LINERESET was caught, UFS might have been put to PWM mode,
* check if power mode restore is needed.
*/
if (hba->saved_uic_err & UFSHCD_UIC_PA_GENERIC_ERROR) {
hba->saved_uic_err &= ~UFSHCD_UIC_PA_GENERIC_ERROR;
if (!hba->saved_uic_err)
hba->saved_err &= ~UIC_ERROR;
spin_unlock_irqrestore(hba->host->host_lock, flags);
if (ufshcd_is_pwr_mode_restore_needed(hba))
needs_restore = true;
spin_lock_irqsave(hba->host->host_lock, flags);
if (!hba->saved_err && !needs_restore)
goto skip_err_handling;
}
hba->silence_err_logs = true;
/* release lock as clear command might sleep */ /* release lock as clear command might sleep */
spin_unlock_irqrestore(hba->host->host_lock, flags); spin_unlock_irqrestore(hba->host->host_lock, flags);
/* Clear pending transfer requests */ /* Clear pending transfer requests */
...@@ -5769,11 +5807,38 @@ static void ufshcd_err_handler(struct work_struct *work) ...@@ -5769,11 +5807,38 @@ static void ufshcd_err_handler(struct work_struct *work)
/* Complete the requests that are cleared by s/w */ /* Complete the requests that are cleared by s/w */
ufshcd_complete_requests(hba); ufshcd_complete_requests(hba);
hba->silence_err_logs = false;
if (err_xfer || err_tm) if (err_xfer || err_tm) {
needs_reset = true; needs_reset = true;
goto do_reset;
}
skip_pending_xfer_clear: /*
* After all reqs and tasks are cleared from doorbell,
* now it is safe to retore power mode.
*/
if (needs_restore) {
spin_unlock_irqrestore(hba->host->host_lock, flags);
/*
* Hold the scaling lock just in case dev cmds
* are sent via bsg and/or sysfs.
*/
down_write(&hba->clk_scaling_lock);
hba->force_pmc = true;
pmc_err = ufshcd_config_pwr_mode(hba, &(hba->pwr_info));
if (pmc_err) {
needs_reset = true;
dev_err(hba->dev, "%s: Failed to restore power mode, err = %d\n",
__func__, pmc_err);
}
hba->force_pmc = false;
ufshcd_print_pwr_info(hba);
up_write(&hba->clk_scaling_lock);
spin_lock_irqsave(hba->host->host_lock, flags);
}
do_reset:
/* Fatal errors need reset */ /* Fatal errors need reset */
if (needs_reset) { if (needs_reset) {
unsigned long max_doorbells = (1UL << hba->nutrs) - 1; unsigned long max_doorbells = (1UL << hba->nutrs) - 1;
...@@ -5829,17 +5894,33 @@ static irqreturn_t ufshcd_update_uic_error(struct ufs_hba *hba) ...@@ -5829,17 +5894,33 @@ static irqreturn_t ufshcd_update_uic_error(struct ufs_hba *hba)
u32 reg; u32 reg;
irqreturn_t retval = IRQ_NONE; irqreturn_t retval = IRQ_NONE;
/* PHY layer lane error */ /* PHY layer error */
reg = ufshcd_readl(hba, REG_UIC_ERROR_CODE_PHY_ADAPTER_LAYER); reg = ufshcd_readl(hba, REG_UIC_ERROR_CODE_PHY_ADAPTER_LAYER);
/* Ignore LINERESET indication, as this is not an error */
if ((reg & UIC_PHY_ADAPTER_LAYER_ERROR) && if ((reg & UIC_PHY_ADAPTER_LAYER_ERROR) &&
(reg & UIC_PHY_ADAPTER_LAYER_LANE_ERR_MASK)) { (reg & UIC_PHY_ADAPTER_LAYER_ERROR_CODE_MASK)) {
ufshcd_update_reg_hist(&hba->ufs_stats.pa_err, reg);
/* /*
* To know whether this error is fatal or not, DB timeout * To know whether this error is fatal or not, DB timeout
* must be checked but this error is handled separately. * must be checked but this error is handled separately.
*/ */
dev_dbg(hba->dev, "%s: UIC Lane error reported\n", __func__); if (reg & UIC_PHY_ADAPTER_LAYER_LANE_ERR_MASK)
ufshcd_update_reg_hist(&hba->ufs_stats.pa_err, reg); dev_dbg(hba->dev, "%s: UIC Lane error reported\n",
__func__);
/* Got a LINERESET indication. */
if (reg & UIC_PHY_ADAPTER_LAYER_GENERIC_ERROR) {
struct uic_command *cmd = NULL;
hba->uic_error |= UFSHCD_UIC_PA_GENERIC_ERROR;
if (hba->uic_async_done && hba->active_uic_cmd)
cmd = hba->active_uic_cmd;
/*
* Ignore the LINERESET during power mode change
* operation via DME_SET command.
*/
if (cmd && (cmd->command == UIC_CMD_DME_SET))
hba->uic_error &= ~UFSHCD_UIC_PA_GENERIC_ERROR;
}
retval |= IRQ_HANDLED; retval |= IRQ_HANDLED;
} }
...@@ -5956,7 +6037,9 @@ static irqreturn_t ufshcd_check_errors(struct ufs_hba *hba) ...@@ -5956,7 +6037,9 @@ static irqreturn_t ufshcd_check_errors(struct ufs_hba *hba)
hba->saved_uic_err |= hba->uic_error; hba->saved_uic_err |= hba->uic_error;
/* dump controller state before resetting */ /* dump controller state before resetting */
if (hba->saved_err & (INT_FATAL_ERRORS | UIC_ERROR)) { if ((hba->saved_err & (INT_FATAL_ERRORS)) ||
(hba->saved_uic_err &&
(hba->saved_uic_err != UFSHCD_UIC_PA_GENERIC_ERROR))) {
dev_err(hba->dev, "%s: saved_err 0x%x saved_uic_err 0x%x\n", dev_err(hba->dev, "%s: saved_err 0x%x saved_uic_err 0x%x\n",
__func__, hba->saved_err, __func__, hba->saved_err,
hba->saved_uic_err); hba->saved_uic_err);
......
...@@ -634,6 +634,7 @@ struct ufs_hba_variant_params { ...@@ -634,6 +634,7 @@ struct ufs_hba_variant_params {
* @saved_err: sticky error mask * @saved_err: sticky error mask
* @saved_uic_err: sticky UIC error mask * @saved_uic_err: sticky UIC error mask
* @force_reset: flag to force eh_work perform a full reset * @force_reset: flag to force eh_work perform a full reset
* @force_pmc: flag to force a power mode change
* @silence_err_logs: flag to silence error logs * @silence_err_logs: flag to silence error logs
* @dev_cmd: ufs device management command information * @dev_cmd: ufs device management command information
* @last_dme_cmd_tstamp: time stamp of the last completed DME command * @last_dme_cmd_tstamp: time stamp of the last completed DME command
...@@ -733,6 +734,7 @@ struct ufs_hba { ...@@ -733,6 +734,7 @@ struct ufs_hba {
u32 saved_uic_err; u32 saved_uic_err;
struct ufs_stats ufs_stats; struct ufs_stats ufs_stats;
bool force_reset; bool force_reset;
bool force_pmc;
bool silence_err_logs; bool silence_err_logs;
/* Device management request data */ /* Device management request data */
......
...@@ -171,6 +171,7 @@ enum { ...@@ -171,6 +171,7 @@ enum {
#define UIC_PHY_ADAPTER_LAYER_ERROR 0x80000000 #define UIC_PHY_ADAPTER_LAYER_ERROR 0x80000000
#define UIC_PHY_ADAPTER_LAYER_ERROR_CODE_MASK 0x1F #define UIC_PHY_ADAPTER_LAYER_ERROR_CODE_MASK 0x1F
#define UIC_PHY_ADAPTER_LAYER_LANE_ERR_MASK 0xF #define UIC_PHY_ADAPTER_LAYER_LANE_ERR_MASK 0xF
#define UIC_PHY_ADAPTER_LAYER_GENERIC_ERROR 0x10
/* UECDL - Host UIC Error Code Data Link Layer 3Ch */ /* UECDL - Host UIC Error Code Data Link Layer 3Ch */
#define UIC_DATA_LINK_LAYER_ERROR 0x80000000 #define UIC_DATA_LINK_LAYER_ERROR 0x80000000
......
...@@ -205,6 +205,9 @@ enum { ...@@ -205,6 +205,9 @@ enum {
UNCHANGED = 7, UNCHANGED = 7,
}; };
#define PWRMODE_MASK 0xF
#define PWRMODE_RX_OFFSET 4
/* PA TX/RX Frequency Series */ /* PA TX/RX Frequency Series */
enum { enum {
PA_HS_MODE_A = 1, PA_HS_MODE_A = 1,
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册