1294 lines
40 KiB
C
1294 lines
40 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (C) 2026 Qualcomm Technologies, Inc.
|
|
*
|
|
* Author:
|
|
* Can Guo <can.guo@oss.qualcomm.com>
|
|
*/
|
|
|
|
#include <linux/bitops.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/errno.h>
|
|
#include <linux/kernel.h>
|
|
#include <ufs/ufshcd.h>
|
|
#include <ufs/unipro.h>
|
|
#include "ufshcd-priv.h"
|
|
|
|
static bool use_adaptive_txeq;
|
|
module_param(use_adaptive_txeq, bool, 0644);
|
|
MODULE_PARM_DESC(use_adaptive_txeq, "Find and apply optimal TX Equalization settings before changing Power Mode (default: false)");
|
|
|
|
static int txeq_gear_set(const char *val, const struct kernel_param *kp)
|
|
{
|
|
return param_set_uint_minmax(val, kp, UFS_HS_G1, UFS_HS_GEAR_MAX);
|
|
}
|
|
|
|
static const struct kernel_param_ops txeq_gear_ops = {
|
|
.set = txeq_gear_set,
|
|
.get = param_get_uint,
|
|
};
|
|
|
|
static unsigned int adaptive_txeq_gear = UFS_HS_G6;
|
|
module_param_cb(adaptive_txeq_gear, &txeq_gear_ops, &adaptive_txeq_gear, 0644);
|
|
MODULE_PARM_DESC(adaptive_txeq_gear, "For HS-Gear[n] and above, adaptive txeq shall be used");
|
|
|
|
static bool use_txeq_presets;
|
|
module_param(use_txeq_presets, bool, 0644);
|
|
MODULE_PARM_DESC(use_txeq_presets, "Use only the 8 TX Equalization Presets (pre-defined Pre-Shoot & De-Emphasis combinations) for TX EQTR (default: false)");
|
|
|
|
static bool txeq_presets_selected[UFS_TX_EQ_PRESET_MAX] = {[0 ... (UFS_TX_EQ_PRESET_MAX - 1)] = 1};
|
|
module_param_array(txeq_presets_selected, bool, NULL, 0644);
|
|
MODULE_PARM_DESC(txeq_presets_selected, "Use only the selected Presets out of the 8 TX Equalization Presets for TX EQTR");
|
|
|
|
/*
|
|
* ufs_tx_eq_preset - Table of minimum required list of presets.
|
|
*
|
|
* A HS-G6 capable M-TX shall support the presets defined in M-PHY v6.0 spec.
|
|
* Preset Pre-Shoot(dB) De-Emphasis(dB)
|
|
* P0 0.0 0.0
|
|
* P1 0.0 0.8
|
|
* P2 0.0 1.6
|
|
* P3 0.8 0.0
|
|
* P4 1.6 0.0
|
|
* P5 0.8 0.8
|
|
* P6 0.8 1.6
|
|
* P7 1.6 0.8
|
|
*/
|
|
static const struct __ufs_tx_eq_preset {
|
|
u8 preshoot;
|
|
u8 deemphasis;
|
|
} ufs_tx_eq_preset[UFS_TX_EQ_PRESET_MAX] = {
|
|
[UFS_TX_EQ_PRESET_P0] = {UFS_TX_HS_PRESHOOT_DB_0P0, UFS_TX_HS_DEEMPHASIS_DB_0P0},
|
|
[UFS_TX_EQ_PRESET_P1] = {UFS_TX_HS_PRESHOOT_DB_0P0, UFS_TX_HS_DEEMPHASIS_DB_0P8},
|
|
[UFS_TX_EQ_PRESET_P2] = {UFS_TX_HS_PRESHOOT_DB_0P0, UFS_TX_HS_DEEMPHASIS_DB_1P6},
|
|
[UFS_TX_EQ_PRESET_P3] = {UFS_TX_HS_PRESHOOT_DB_0P8, UFS_TX_HS_DEEMPHASIS_DB_0P0},
|
|
[UFS_TX_EQ_PRESET_P4] = {UFS_TX_HS_PRESHOOT_DB_1P6, UFS_TX_HS_DEEMPHASIS_DB_0P0},
|
|
[UFS_TX_EQ_PRESET_P5] = {UFS_TX_HS_PRESHOOT_DB_0P8, UFS_TX_HS_DEEMPHASIS_DB_0P8},
|
|
[UFS_TX_EQ_PRESET_P6] = {UFS_TX_HS_PRESHOOT_DB_0P8, UFS_TX_HS_DEEMPHASIS_DB_1P6},
|
|
[UFS_TX_EQ_PRESET_P7] = {UFS_TX_HS_PRESHOOT_DB_1P6, UFS_TX_HS_DEEMPHASIS_DB_0P8},
|
|
};
|
|
|
|
/*
|
|
* pa_peer_rx_adapt_initial - Table of UniPro PA_PeerRxHSGnAdaptInitial
|
|
* attribute IDs for High Speed (HS) Gears.
|
|
*
|
|
* This table maps HS Gears to their respective UniPro PA_PeerRxHSGnAdaptInitial
|
|
* attribute IDs. Entries for Gears 1-3 are 0 (unsupported).
|
|
*/
|
|
static const u32 pa_peer_rx_adapt_initial[UFS_HS_GEAR_MAX] = {
|
|
0,
|
|
0,
|
|
0,
|
|
PA_PEERRXHSG4ADAPTINITIAL,
|
|
PA_PEERRXHSG5ADAPTINITIAL,
|
|
PA_PEERRXHSG6ADAPTINITIALL0L3
|
|
};
|
|
|
|
/*
|
|
* rx_adapt_initial_cap - Table of M-PHY RX_HS_Gn_ADAPT_INITIAL_Capability
|
|
* attribute IDs for High Speed (HS) Gears.
|
|
*
|
|
* This table maps HS Gears to their respective M-PHY
|
|
* RX_HS_Gn_ADAPT_INITIAL_Capability attribute IDs. Entries for Gears 1-3 are 0
|
|
* (unsupported).
|
|
*/
|
|
static const u32 rx_adapt_initial_cap[UFS_HS_GEAR_MAX] = {
|
|
0,
|
|
0,
|
|
0,
|
|
RX_HS_G4_ADAPT_INITIAL_CAP,
|
|
RX_HS_G5_ADAPT_INITIAL_CAP,
|
|
RX_HS_G6_ADAPT_INITIAL_CAP
|
|
};
|
|
|
|
/*
|
|
* pa_tx_eq_setting - Table of UniPro PA_TxEQGnSetting attribute IDs for High
|
|
* Speed (HS) Gears.
|
|
*
|
|
* This table maps HS Gears to their respective UniPro PA_TxEQGnSetting
|
|
* attribute IDs.
|
|
*/
|
|
static const u32 pa_tx_eq_setting[UFS_HS_GEAR_MAX] = {
|
|
PA_TXEQG1SETTING,
|
|
PA_TXEQG2SETTING,
|
|
PA_TXEQG3SETTING,
|
|
PA_TXEQG4SETTING,
|
|
PA_TXEQG5SETTING,
|
|
PA_TXEQG6SETTING
|
|
};
|
|
|
|
/**
|
|
* ufshcd_configure_precoding - Configure Pre-Coding for all active lanes
|
|
* @hba: per adapter instance
|
|
* @params: TX EQ parameters data structure
|
|
*
|
|
* Bit[7] in RX_FOM indicates that the receiver needs to enable Pre-Coding when
|
|
* set. Pre-Coding must be enabled on both the transmitter and receiver to
|
|
* ensure proper operation.
|
|
*
|
|
* Returns 0 on success, non-zero error code otherwise
|
|
*/
|
|
static int ufshcd_configure_precoding(struct ufs_hba *hba,
|
|
struct ufshcd_tx_eq_params *params)
|
|
{
|
|
struct ufs_pa_layer_attr *pwr_info = &hba->max_pwr_info.info;
|
|
u32 local_precode_en = 0;
|
|
u32 peer_precode_en = 0;
|
|
int lane, ret;
|
|
|
|
/* Enable Pre-Coding for Host's TX & Device's RX pair */
|
|
for (lane = 0; lane < pwr_info->lane_tx; lane++) {
|
|
if (params->host[lane].precode_en) {
|
|
local_precode_en |= PRECODEEN_TX_BIT(lane);
|
|
peer_precode_en |= PRECODEEN_RX_BIT(lane);
|
|
}
|
|
}
|
|
|
|
/* Enable Pre-Coding for Device's TX & Host's RX pair */
|
|
for (lane = 0; lane < pwr_info->lane_rx; lane++) {
|
|
if (params->device[lane].precode_en) {
|
|
peer_precode_en |= PRECODEEN_TX_BIT(lane);
|
|
local_precode_en |= PRECODEEN_RX_BIT(lane);
|
|
}
|
|
}
|
|
|
|
if (!local_precode_en && !peer_precode_en) {
|
|
dev_dbg(hba->dev, "Pre-Coding is not required for Host and Device\n");
|
|
return 0;
|
|
}
|
|
|
|
/* Set local PA_PreCodeEn */
|
|
ret = ufshcd_dme_set(hba, UIC_ARG_MIB(PA_PRECODEEN), local_precode_en);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to set local PA_PreCodeEn: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Set peer PA_PreCodeEn */
|
|
ret = ufshcd_dme_peer_set(hba, UIC_ARG_MIB(PA_PRECODEEN), peer_precode_en);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to set peer PA_PreCodeEn: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
dev_dbg(hba->dev, "Local PA_PreCodeEn: 0x%02x, Peer PA_PreCodeEn: 0x%02x\n",
|
|
local_precode_en, peer_precode_en);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void ufshcd_print_tx_eq_params(struct ufs_hba *hba)
|
|
{
|
|
struct ufs_pa_layer_attr *pwr_info = &hba->max_pwr_info.info;
|
|
struct ufshcd_tx_eq_params *params;
|
|
u32 gear = hba->pwr_info.gear_tx;
|
|
int lane;
|
|
|
|
if (!ufshcd_is_tx_eq_supported(hba))
|
|
return;
|
|
|
|
if (gear < UFS_HS_G1 || gear > UFS_HS_GEAR_MAX)
|
|
return;
|
|
|
|
params = &hba->tx_eq_params[gear - 1];
|
|
if (!params->is_valid || !params->is_applied)
|
|
return;
|
|
|
|
for (lane = 0; lane < pwr_info->lane_tx; lane++)
|
|
dev_dbg(hba->dev, "Host TX Lane %d: PreShoot %u, DeEmphasis %u, FOM %u, PreCodeEn %d\n",
|
|
lane, params->host[lane].preshoot,
|
|
params->host[lane].deemphasis,
|
|
params->host[lane].fom_val,
|
|
params->host[lane].precode_en);
|
|
|
|
for (lane = 0; lane < pwr_info->lane_rx; lane++)
|
|
dev_dbg(hba->dev, "Device TX Lane %d: PreShoot %u, DeEmphasis %u, FOM %u, PreCodeEn %d\n",
|
|
lane, params->device[lane].preshoot,
|
|
params->device[lane].deemphasis,
|
|
params->device[lane].fom_val,
|
|
params->device[lane].precode_en);
|
|
}
|
|
|
|
static inline u32
|
|
ufshcd_compose_tx_eq_setting(struct ufshcd_tx_eq_settings *settings,
|
|
int num_lanes)
|
|
{
|
|
u32 setting = 0;
|
|
int lane;
|
|
|
|
for (lane = 0; lane < num_lanes; lane++, settings++) {
|
|
setting |= TX_HS_PRESHOOT_BITS(lane, settings->preshoot);
|
|
setting |= TX_HS_DEEMPHASIS_BITS(lane, settings->deemphasis);
|
|
}
|
|
|
|
return setting;
|
|
}
|
|
|
|
/**
|
|
* ufshcd_apply_tx_eq_settings - Apply TX Equalization settings for target gear
|
|
* @hba: per adapter instance
|
|
* @params: TX EQ parameters data structure
|
|
* @gear: target gear
|
|
*
|
|
* Returns 0 on success, negative error code otherwise
|
|
*/
|
|
int ufshcd_apply_tx_eq_settings(struct ufs_hba *hba,
|
|
struct ufshcd_tx_eq_params *params, u32 gear)
|
|
{
|
|
struct ufs_pa_layer_attr *pwr_info = &hba->max_pwr_info.info;
|
|
u32 setting;
|
|
int ret;
|
|
|
|
/* Compose settings for Host's TX Lanes */
|
|
setting = ufshcd_compose_tx_eq_setting(params->host, pwr_info->lane_tx);
|
|
ret = ufshcd_dme_set(hba, UIC_ARG_MIB(pa_tx_eq_setting[gear - 1]), setting);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Compose settings for Device's TX Lanes */
|
|
setting = ufshcd_compose_tx_eq_setting(params->device, pwr_info->lane_rx);
|
|
ret = ufshcd_dme_peer_set(hba, UIC_ARG_MIB(pa_tx_eq_setting[gear - 1]), setting);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Configure Pre-Coding */
|
|
if (gear >= UFS_HS_G6) {
|
|
ret = ufshcd_configure_precoding(hba, params);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to configure pre-coding: %d\n", ret);
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(ufshcd_apply_tx_eq_settings);
|
|
|
|
/**
|
|
* ufshcd_evaluate_tx_eqtr_fom - Evaluate TX EQTR FOM results
|
|
* @hba: per adapter instance
|
|
* @pwr_mode: target power mode containing gear and rate information
|
|
* @eqtr_data: TX EQTR data structure
|
|
* @h_iter: host TX EQTR iterator data structure
|
|
* @d_iter: device TX EQTR iterator data structure
|
|
*
|
|
* Evaluate TX EQTR FOM results, update host and device TX EQTR data accordingy
|
|
* if FOM have been improved compared to previous iteration, and record TX EQTR
|
|
* FOM results.
|
|
*/
|
|
static void ufshcd_evaluate_tx_eqtr_fom(struct ufs_hba *hba,
|
|
struct ufs_pa_layer_attr *pwr_mode,
|
|
struct ufshcd_tx_eqtr_data *eqtr_data,
|
|
struct tx_eqtr_iter *h_iter,
|
|
struct tx_eqtr_iter *d_iter)
|
|
{
|
|
u8 preshoot, deemphasis, fom_value;
|
|
bool precode_en;
|
|
int lane;
|
|
|
|
for (lane = 0; h_iter->is_updated && lane < pwr_mode->lane_tx; lane++) {
|
|
preshoot = h_iter->preshoot;
|
|
deemphasis = h_iter->deemphasis;
|
|
fom_value = h_iter->fom[lane] & RX_FOM_VALUE_MASK;
|
|
precode_en = h_iter->fom[lane] & RX_FOM_PRECODING_EN_BIT;
|
|
|
|
/* Record host TX EQTR FOM */
|
|
eqtr_data->host_fom[lane][preshoot][deemphasis] = h_iter->fom[lane];
|
|
|
|
/* Check if FOM has been improved for host's TX Lanes */
|
|
if (fom_value > eqtr_data->host[lane].fom_val) {
|
|
eqtr_data->host[lane].preshoot = preshoot;
|
|
eqtr_data->host[lane].deemphasis = deemphasis;
|
|
eqtr_data->host[lane].fom_val = fom_value;
|
|
eqtr_data->host[lane].precode_en = precode_en;
|
|
}
|
|
|
|
dev_dbg(hba->dev, "TX EQTR: Host TX Lane %d: PreShoot %u, DeEmphasis %u, FOM value %u, PreCodeEn %d\n",
|
|
lane, preshoot, deemphasis, fom_value, precode_en);
|
|
}
|
|
|
|
for (lane = 0; d_iter->is_updated && lane < pwr_mode->lane_rx; lane++) {
|
|
preshoot = d_iter->preshoot;
|
|
deemphasis = d_iter->deemphasis;
|
|
fom_value = d_iter->fom[lane] & RX_FOM_VALUE_MASK;
|
|
precode_en = d_iter->fom[lane] & RX_FOM_PRECODING_EN_BIT;
|
|
|
|
/* Record device TX EQTR FOM */
|
|
eqtr_data->device_fom[lane][preshoot][deemphasis] = d_iter->fom[lane];
|
|
|
|
/* Check if FOM has been improved for Device's TX Lanes */
|
|
if (fom_value > eqtr_data->device[lane].fom_val) {
|
|
eqtr_data->device[lane].preshoot = preshoot;
|
|
eqtr_data->device[lane].deemphasis = deemphasis;
|
|
eqtr_data->device[lane].fom_val = fom_value;
|
|
eqtr_data->device[lane].precode_en = precode_en;
|
|
}
|
|
|
|
dev_dbg(hba->dev, "TX EQTR: Device TX Lane %d: PreShoot %u, DeEmphasis %u, FOM value %u, PreCodeEn %d\n",
|
|
lane, preshoot, deemphasis, fom_value, precode_en);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* ufshcd_get_rx_fom - Get Figure of Merit (FOM) for both sides
|
|
* @hba: per adapter instance
|
|
* @pwr_mode: target power mode containing gear and rate information
|
|
* @h_iter: host TX EQTR iterator data structure
|
|
* @d_iter: device TX EQTR iterator data structure
|
|
*
|
|
* Returns 0 on success, negative error code otherwise
|
|
*/
|
|
static int ufshcd_get_rx_fom(struct ufs_hba *hba,
|
|
struct ufs_pa_layer_attr *pwr_mode,
|
|
struct tx_eqtr_iter *h_iter,
|
|
struct tx_eqtr_iter *d_iter)
|
|
{
|
|
int lane, ret;
|
|
u32 fom;
|
|
|
|
/* Get FOM of host's TX lanes from device's RX_FOM. */
|
|
for (lane = 0; lane < pwr_mode->lane_tx; lane++) {
|
|
ret = ufshcd_dme_peer_get(hba, UIC_ARG_MIB_SEL(RX_FOM,
|
|
UIC_ARG_MPHY_RX_GEN_SEL_INDEX(lane)),
|
|
&fom);
|
|
if (ret)
|
|
return ret;
|
|
|
|
h_iter->fom[lane] = (u8)fom;
|
|
}
|
|
|
|
/* Get FOM of device's TX lanes from host's RX_FOM. */
|
|
for (lane = 0; lane < pwr_mode->lane_rx; lane++) {
|
|
ret = ufshcd_dme_get(hba, UIC_ARG_MIB_SEL(RX_FOM,
|
|
UIC_ARG_MPHY_RX_GEN_SEL_INDEX(lane)),
|
|
&fom);
|
|
if (ret)
|
|
return ret;
|
|
|
|
d_iter->fom[lane] = (u8)fom;
|
|
}
|
|
|
|
ret = ufshcd_vops_get_rx_fom(hba, pwr_mode, h_iter, d_iter);
|
|
if (ret)
|
|
dev_err(hba->dev, "Failed to get FOM via vops: %d\n", ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
bool ufshcd_is_txeq_presets_used(struct ufs_hba *hba)
|
|
{
|
|
return use_txeq_presets;
|
|
}
|
|
|
|
bool ufshcd_is_txeq_preset_selected(u8 preshoot, u8 deemphasis)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < UFS_TX_EQ_PRESET_MAX; i++) {
|
|
if (!txeq_presets_selected[i])
|
|
continue;
|
|
|
|
if (preshoot == ufs_tx_eq_preset[i].preshoot &&
|
|
deemphasis == ufs_tx_eq_preset[i].deemphasis)
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* tx_eqtr_iter_try_update - Try to update a TX EQTR iterator
|
|
* @iter: TX EQTR iterator data structure
|
|
* @preshoot: PreShoot value
|
|
* @deemphasis: DeEmphasis value
|
|
*
|
|
* This function validates whether the provided PreShoot and DeEmphasis
|
|
* combination can be used or not. If yes, it updates the TX EQTR iterator with
|
|
* the provided PreShoot and DeEmphasis, it also sets the is_updated flag
|
|
* to indicate the iterator has been updated.
|
|
*/
|
|
static void tx_eqtr_iter_try_update(struct tx_eqtr_iter *iter,
|
|
u8 preshoot, u8 deemphasis)
|
|
{
|
|
if (!test_bit(preshoot, &iter->preshoot_bitmap) ||
|
|
!test_bit(deemphasis, &iter->deemphasis_bitmap) ||
|
|
(use_txeq_presets && !ufshcd_is_txeq_preset_selected(preshoot, deemphasis))) {
|
|
iter->is_updated = false;
|
|
return;
|
|
}
|
|
|
|
iter->preshoot = preshoot;
|
|
iter->deemphasis = deemphasis;
|
|
iter->is_updated = true;
|
|
}
|
|
|
|
/**
|
|
* tx_eqtr_iter_update() - Update host and deviceTX EQTR iterators
|
|
* @preshoot: PreShoot value
|
|
* @deemphasis: DeEmphasis value
|
|
* @h_iter: Host TX EQTR iterator data structure
|
|
* @d_iter: Device TX EQTR iterator data structure
|
|
*
|
|
* Updates host and device TX Equalization training iterators with the
|
|
* provided PreShoot and DeEmphasis.
|
|
*
|
|
* Return: true if host and/or device TX Equalization training iterator has
|
|
* been updated to the provided PreShoot and DeEmphasis, false otherwise.
|
|
*/
|
|
static bool tx_eqtr_iter_update(u8 preshoot, u8 deemphasis,
|
|
struct tx_eqtr_iter *h_iter,
|
|
struct tx_eqtr_iter *d_iter)
|
|
{
|
|
tx_eqtr_iter_try_update(h_iter, preshoot, deemphasis);
|
|
tx_eqtr_iter_try_update(d_iter, preshoot, deemphasis);
|
|
|
|
return h_iter->is_updated || d_iter->is_updated;
|
|
}
|
|
|
|
/**
|
|
* ufshcd_tx_eqtr_iter_init - Initialize host and device TX EQTR iterators
|
|
* @hba: per adapter instance
|
|
* @h_iter: host TX EQTR iterator data structure
|
|
* @d_iter: device TX EQTR iterator data structure
|
|
*
|
|
* This function initializes the TX EQTR iterator structures for both host and
|
|
* device by reading their TX equalization capabilities. The capabilities are
|
|
* cached in the hba structure to avoid redundant DME operations in subsequent
|
|
* calls. In the TX EQTR procedure, the iterator structures are updated by
|
|
* tx_eqtr_iter_update() to systematically iterate through supported TX
|
|
* Equalization setting combinations.
|
|
*
|
|
* Returns 0 on success, negative error code otherwise
|
|
*/
|
|
static int ufshcd_tx_eqtr_iter_init(struct ufs_hba *hba,
|
|
struct tx_eqtr_iter *h_iter,
|
|
struct tx_eqtr_iter *d_iter)
|
|
{
|
|
u32 cap;
|
|
int ret;
|
|
|
|
if (!hba->host_preshoot_cap) {
|
|
ret = ufshcd_dme_get(hba, UIC_ARG_MIB(TX_HS_PRESHOOT_SETTING_CAP), &cap);
|
|
if (ret)
|
|
return ret;
|
|
|
|
hba->host_preshoot_cap = cap & TX_EQTR_CAP_MASK;
|
|
}
|
|
|
|
if (!hba->host_deemphasis_cap) {
|
|
ret = ufshcd_dme_get(hba, UIC_ARG_MIB(TX_HS_DEEMPHASIS_SETTING_CAP), &cap);
|
|
if (ret)
|
|
return ret;
|
|
|
|
hba->host_deemphasis_cap = cap & TX_EQTR_CAP_MASK;
|
|
}
|
|
|
|
if (!hba->device_preshoot_cap) {
|
|
ret = ufshcd_dme_peer_get(hba, UIC_ARG_MIB(TX_HS_PRESHOOT_SETTING_CAP), &cap);
|
|
if (ret)
|
|
return ret;
|
|
|
|
hba->device_preshoot_cap = cap & TX_EQTR_CAP_MASK;
|
|
}
|
|
|
|
if (!hba->device_deemphasis_cap) {
|
|
ret = ufshcd_dme_peer_get(hba, UIC_ARG_MIB(TX_HS_DEEMPHASIS_SETTING_CAP), &cap);
|
|
if (ret)
|
|
return ret;
|
|
|
|
hba->device_deemphasis_cap = cap & TX_EQTR_CAP_MASK;
|
|
}
|
|
|
|
/*
|
|
* Support PreShoot & DeEmphasis of value 0 is mandatory, hence they are
|
|
* not reflected in PreShoot/DeEmphasis capabilities. Left shift the
|
|
* capability bitmap by 1 and set bit[0] to reflect value 0 is
|
|
* supported, such that test_bit() can be used later for convenience.
|
|
*/
|
|
h_iter->preshoot_bitmap = (hba->host_preshoot_cap << 0x1) | 0x1;
|
|
h_iter->deemphasis_bitmap = (hba->host_deemphasis_cap << 0x1) | 0x1;
|
|
d_iter->preshoot_bitmap = (hba->device_preshoot_cap << 0x1) | 0x1;
|
|
d_iter->deemphasis_bitmap = (hba->device_deemphasis_cap << 0x1) | 0x1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* adapt_cap_to_t_adapt - Calculate TAdapt from adapt capability
|
|
* @adapt_cap: Adapt capability
|
|
*
|
|
* For NRZ:
|
|
* IF (ADAPT_range = FINE)
|
|
* TADAPT = 650 x (ADAPT_length + 1)
|
|
* ELSE (IF ADAPT_range = COARSE)
|
|
* TADAPT = 650 x 2^ADAPT_length
|
|
*
|
|
* Returns calculated TAdapt value in term of Unit Intervals (UI)
|
|
*/
|
|
static inline u64 adapt_cap_to_t_adapt(u32 adapt_cap)
|
|
{
|
|
u64 tadapt;
|
|
u8 adapt_length = adapt_cap & ADAPT_LENGTH_MASK;
|
|
|
|
if (!IS_ADAPT_RANGE_COARSE(adapt_cap))
|
|
tadapt = TADAPT_FACTOR * (adapt_length + 1);
|
|
else
|
|
tadapt = TADAPT_FACTOR * (1 << adapt_length);
|
|
|
|
return tadapt;
|
|
}
|
|
|
|
/**
|
|
* adapt_cap_to_t_adapt_l0l3 - Calculate TAdapt_L0_L3 from adapt capability
|
|
* @adapt_cap: Adapt capability
|
|
*
|
|
* For PAM-4:
|
|
* IF (ADAPT_range = FINE)
|
|
* TADAPT_L0_L3 = 2^9 x ADAPT_length
|
|
* ELSE IF (ADAPT_range = COARSE)
|
|
* TADAPT_L0_L3 = 2^9 x (2^ADAPT_length)
|
|
*
|
|
* Returns calculated TAdapt value in term of Unit Intervals (UI)
|
|
*/
|
|
static inline u64 adapt_cap_to_t_adapt_l0l3(u32 adapt_cap)
|
|
{
|
|
u64 tadapt;
|
|
u8 adapt_length = adapt_cap & ADAPT_LENGTH_MASK;
|
|
|
|
if (!IS_ADAPT_RANGE_COARSE(adapt_cap))
|
|
tadapt = TADAPT_L0L3_FACTOR * adapt_length;
|
|
else
|
|
tadapt = TADAPT_L0L3_FACTOR * (1 << adapt_length);
|
|
|
|
return tadapt;
|
|
}
|
|
|
|
/**
|
|
* adapt_cap_to_t_adapt_l0l1l2l3 - Calculate TAdapt_L0_L1_L2_L3 from adapt capability
|
|
* @adapt_cap: Adapt capability
|
|
*
|
|
* For PAM-4:
|
|
* IF (ADAPT_range_L0_L1_L2_L3 = FINE)
|
|
* TADAPT_L0_L1_L2_L3 = 2^15 x (ADAPT_length_L0_L1_L2_L3 + 1)
|
|
* ELSE IF (ADAPT_range_L0_L1_L2_L3 = COARSE)
|
|
* TADAPT_L0_L1_L2_L3 = 2^15 x 2^ADAPT_length_L0_L1_L2_L3
|
|
*
|
|
* Returns calculated TAdapt value in term of Unit Intervals (UI)
|
|
*/
|
|
static inline u64 adapt_cap_to_t_adapt_l0l1l2l3(u32 adapt_cap)
|
|
{
|
|
u64 tadapt;
|
|
u8 adapt_length = adapt_cap & ADAPT_LENGTH_MASK;
|
|
|
|
if (!IS_ADAPT_RANGE_COARSE(adapt_cap))
|
|
tadapt = TADAPT_L0L1L2L3_FACTOR * (adapt_length + 1);
|
|
else
|
|
tadapt = TADAPT_L0L1L2L3_FACTOR * (1 << adapt_length);
|
|
|
|
return tadapt;
|
|
}
|
|
|
|
/**
|
|
* ufshcd_setup_tx_eqtr_adapt_length - Setup TX adapt length for EQTR
|
|
* @hba: per adapter instance
|
|
* @params: TX EQ parameters data structure
|
|
* @gear: target gear for EQTR
|
|
*
|
|
* This function determines and configures the proper TX adapt length (TAdapt)
|
|
* for the TX EQTR procedure based on the target gear and RX adapt capabilities
|
|
* of both host and device.
|
|
*
|
|
* Guidelines from MIPI UniPro v3.0 spec - select the minimum Adapt Length for
|
|
* the Equalization Training procedure based on the following conditions:
|
|
*
|
|
* If the target High-Speed Gear n is HS-G4 or HS-G5:
|
|
* PA_TxAdaptLength_EQTR[7:0] >= Max (10us, RX_HS_Gn_ADAPT_INITIAL_Capability,
|
|
* PA_PeerRxHsGnAdaptInitial)
|
|
* PA_TxAdaptLength_EQTR[7:0] shall be shorter than PACP_REQUEST_TIMER (10ms)
|
|
* PA_TxAdaptLength_EQTR[15:8] is not relevant for HS-G4 and HS-G5. This field
|
|
* is set to 255 (reserved value).
|
|
*
|
|
* If the target High-Speed Gear n is HS-G6:
|
|
* PA_TxAdapthLength_EQTR >= 10us
|
|
* PA_TxAdapthLength_EQTR[7:0] >= Max (RX_HS_G6_ADAPT_INITIAL_Capability,
|
|
* PA_PeerRxHsG6AdaptInitialL0L3)
|
|
* PA_TxAdapthLength_EQTR[15:8] >= Max (RX_HS_G6_ADAPT_INITIAL_L0_L1_L2_L3_Capability,
|
|
* PA_PeerRxHsG6AdaptInitialL0L1L2L3)
|
|
* PA_TxAdaptLength_EQTR shall be shorter than PACP_REQUEST_TIMER value of 10ms.
|
|
*
|
|
* Since adapt capabilities encode both range (fine/coarse) and length values,
|
|
* direct comparison is not possible. This function converts adapt capabilities
|
|
* to actual time durations in Unit Intervals (UI) using the Adapt time
|
|
* calculation formular in M-PHY v6.0 spec (Table 8), then selects the maximum
|
|
* to ensure both host and device use adequate TX adapt length.
|
|
*
|
|
* Returns 0 on success, negative error code otherwise
|
|
*/
|
|
static int ufshcd_setup_tx_eqtr_adapt_length(struct ufs_hba *hba,
|
|
struct ufshcd_tx_eq_params *params,
|
|
u32 gear)
|
|
{
|
|
struct ufshcd_tx_eqtr_record *rec = params->eqtr_record;
|
|
u32 adapt_eqtr;
|
|
int ret;
|
|
|
|
if (rec && rec->saved_adapt_eqtr) {
|
|
adapt_eqtr = rec->saved_adapt_eqtr;
|
|
goto set_adapt_eqtr;
|
|
}
|
|
|
|
if (gear == UFS_HS_G4 || gear == UFS_HS_G5) {
|
|
u64 t_adapt, t_adapt_local, t_adapt_peer;
|
|
u32 adapt_cap_local, adapt_cap_peer, adapt_length;
|
|
|
|
ret = ufshcd_dme_get(hba, UIC_ARG_MIB_SEL(rx_adapt_initial_cap[gear - 1],
|
|
UIC_ARG_MPHY_RX_GEN_SEL_INDEX(0)),
|
|
&adapt_cap_local);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (adapt_cap_local > ADAPT_LENGTH_MAX) {
|
|
dev_err(hba->dev, "local RX_HS_G%u_ADAPT_INITIAL_CAP (0x%x) exceeds MAX\n",
|
|
gear, adapt_cap_local);
|
|
return -EINVAL;
|
|
}
|
|
|
|
ret = ufshcd_dme_get(hba, UIC_ARG_MIB(pa_peer_rx_adapt_initial[gear - 1]),
|
|
&adapt_cap_peer);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (adapt_cap_peer > ADAPT_LENGTH_MAX) {
|
|
dev_err(hba->dev, "local RX_HS_G%u_ADAPT_INITIAL_CAP (0x%x) exceeds MAX\n",
|
|
gear, adapt_cap_peer);
|
|
return -EINVAL;
|
|
}
|
|
|
|
t_adapt_local = adapt_cap_to_t_adapt(adapt_cap_local);
|
|
t_adapt_peer = adapt_cap_to_t_adapt(adapt_cap_peer);
|
|
t_adapt = max(t_adapt_local, t_adapt_peer);
|
|
|
|
dev_dbg(hba->dev, "local RX_HS_G%u_ADAPT_INITIAL_CAP = 0x%x\n",
|
|
gear, adapt_cap_local);
|
|
dev_dbg(hba->dev, "peer RX_HS_G%u_ADAPT_INITIAL_CAP = 0x%x\n",
|
|
gear, adapt_cap_peer);
|
|
dev_dbg(hba->dev, "t_adapt_local = %llu UI, t_adapt_peer = %llu UI\n",
|
|
t_adapt_local, t_adapt_peer);
|
|
dev_dbg(hba->dev, "TAdapt %llu UI selected for TX EQTR\n",
|
|
t_adapt);
|
|
|
|
adapt_length = (t_adapt_local >= t_adapt_peer) ?
|
|
adapt_cap_local : adapt_cap_peer;
|
|
|
|
if (gear == UFS_HS_G4 && t_adapt < TX_EQTR_HS_G4_MIN_T_ADAPT) {
|
|
dev_dbg(hba->dev, "TAdapt %llu UI is too short for TX EQTR for HS-G%u, use default Adapt 0x%x\n",
|
|
t_adapt, gear, TX_EQTR_HS_G4_ADAPT_DEFAULT);
|
|
adapt_length = TX_EQTR_HS_G4_ADAPT_DEFAULT;
|
|
} else if (gear == UFS_HS_G5 && t_adapt < TX_EQTR_HS_G5_MIN_T_ADAPT) {
|
|
dev_dbg(hba->dev, "TAdapt %llu UI is too short for TX EQTR for HS-G%u, use default Adapt 0x%x\n",
|
|
t_adapt, gear, TX_EQTR_HS_G5_ADAPT_DEFAULT);
|
|
adapt_length = TX_EQTR_HS_G5_ADAPT_DEFAULT;
|
|
}
|
|
|
|
adapt_eqtr = adapt_length |
|
|
(TX_EQTR_ADAPT_RESERVED << TX_EQTR_ADAPT_LENGTH_L0L1L2L3_SHIFT);
|
|
} else if (gear == UFS_HS_G6) {
|
|
u64 t_adapt, t_adapt_l0l3, t_adapt_l0l3_local, t_adapt_l0l3_peer;
|
|
u64 t_adapt_l0l1l2l3, t_adapt_l0l1l2l3_local, t_adapt_l0l1l2l3_peer;
|
|
u32 adapt_l0l3_cap_local, adapt_l0l3_cap_peer, adapt_length_l0l3;
|
|
u32 adapt_l0l1l2l3_cap_local, adapt_l0l1l2l3_cap_peer, adapt_length_l0l1l2l3;
|
|
|
|
ret = ufshcd_dme_get(hba, UIC_ARG_MIB_SEL(rx_adapt_initial_cap[gear - 1],
|
|
UIC_ARG_MPHY_RX_GEN_SEL_INDEX(0)),
|
|
&adapt_l0l3_cap_local);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (adapt_l0l3_cap_local > ADAPT_L0L3_LENGTH_MAX) {
|
|
dev_err(hba->dev, "local RX_HS_G%u_ADAPT_INITIAL_CAP (0x%x) exceeds MAX\n",
|
|
gear, adapt_l0l3_cap_local);
|
|
return -EINVAL;
|
|
}
|
|
|
|
ret = ufshcd_dme_get(hba, UIC_ARG_MIB(pa_peer_rx_adapt_initial[gear - 1]),
|
|
&adapt_l0l3_cap_peer);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (adapt_l0l3_cap_peer > ADAPT_L0L3_LENGTH_MAX) {
|
|
dev_err(hba->dev, "peer RX_HS_G%u_ADAPT_INITIAL_CAP (0x%x) exceeds MAX\n",
|
|
gear, adapt_l0l3_cap_peer);
|
|
return -EINVAL;
|
|
}
|
|
|
|
t_adapt_l0l3_local = adapt_cap_to_t_adapt_l0l3(adapt_l0l3_cap_local);
|
|
t_adapt_l0l3_peer = adapt_cap_to_t_adapt_l0l3(adapt_l0l3_cap_peer);
|
|
|
|
dev_dbg(hba->dev, "local RX_HS_G%u_ADAPT_INITIAL_CAP = 0x%x\n",
|
|
gear, adapt_l0l3_cap_local);
|
|
dev_dbg(hba->dev, "peer RX_HS_G%u_ADAPT_INITIAL_CAP = 0x%x\n",
|
|
gear, adapt_l0l3_cap_peer);
|
|
dev_dbg(hba->dev, "t_adapt_l0l3_local = %llu UI, t_adapt_l0l3_peer = %llu UI\n",
|
|
t_adapt_l0l3_local, t_adapt_l0l3_peer);
|
|
|
|
ret = ufshcd_dme_get(hba, UIC_ARG_MIB_SEL(RX_HS_G6_ADAPT_INITIAL_L0L1L2L3_CAP,
|
|
UIC_ARG_MPHY_RX_GEN_SEL_INDEX(0)),
|
|
&adapt_l0l1l2l3_cap_local);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (adapt_l0l1l2l3_cap_local > ADAPT_L0L1L2L3_LENGTH_MAX) {
|
|
dev_err(hba->dev, "local RX_HS_G%u_ADAPT_INITIAL_L0L1L2L3_CAP (0x%x) exceeds MAX\n",
|
|
gear, adapt_l0l1l2l3_cap_local);
|
|
return -EINVAL;
|
|
}
|
|
|
|
ret = ufshcd_dme_get(hba, UIC_ARG_MIB(PA_PEERRXHSG6ADAPTINITIALL0L1L2L3),
|
|
&adapt_l0l1l2l3_cap_peer);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (adapt_l0l1l2l3_cap_peer > ADAPT_L0L1L2L3_LENGTH_MAX) {
|
|
dev_err(hba->dev, "peer RX_HS_G%u_ADAPT_INITIAL_L0L1L2L3_CAP (0x%x) exceeds MAX\n",
|
|
gear, adapt_l0l1l2l3_cap_peer);
|
|
return -EINVAL;
|
|
}
|
|
|
|
t_adapt_l0l1l2l3_local = adapt_cap_to_t_adapt_l0l1l2l3(adapt_l0l1l2l3_cap_local);
|
|
t_adapt_l0l1l2l3_peer = adapt_cap_to_t_adapt_l0l1l2l3(adapt_l0l1l2l3_cap_peer);
|
|
|
|
dev_dbg(hba->dev, "local RX_HS_G%u_ADAPT_INITIAL_L0L1L2L3_CAP = 0x%x\n",
|
|
gear, adapt_l0l1l2l3_cap_local);
|
|
dev_dbg(hba->dev, "peer RX_HS_G%u_ADAPT_INITIAL_L0L1L2L3_CAP = 0x%x\n",
|
|
gear, adapt_l0l1l2l3_cap_peer);
|
|
dev_dbg(hba->dev, "t_adapt_l0l1l2l3_local = %llu UI, t_adapt_l0l1l2l3_peer = %llu UI\n",
|
|
t_adapt_l0l1l2l3_local, t_adapt_l0l1l2l3_peer);
|
|
|
|
t_adapt_l0l1l2l3 = max(t_adapt_l0l1l2l3_local, t_adapt_l0l1l2l3_peer);
|
|
t_adapt_l0l3 = max(t_adapt_l0l3_local, t_adapt_l0l3_peer);
|
|
t_adapt = t_adapt_l0l3 + t_adapt_l0l1l2l3;
|
|
|
|
dev_dbg(hba->dev, "TAdapt %llu PAM-4 UI selected for TX EQTR\n",
|
|
t_adapt);
|
|
|
|
adapt_length_l0l3 = (t_adapt_l0l3_local >= t_adapt_l0l3_peer) ?
|
|
adapt_l0l3_cap_local : adapt_l0l3_cap_peer;
|
|
adapt_length_l0l1l2l3 = (t_adapt_l0l1l2l3_local >= t_adapt_l0l1l2l3_peer) ?
|
|
adapt_l0l1l2l3_cap_local : adapt_l0l1l2l3_cap_peer;
|
|
|
|
if (t_adapt < TX_EQTR_HS_G6_MIN_T_ADAPT) {
|
|
dev_dbg(hba->dev, "TAdapt %llu UI is too short for TX EQTR for HS-G%u, use default Adapt 0x%x\n",
|
|
t_adapt, gear, TX_EQTR_HS_G6_ADAPT_DEFAULT);
|
|
adapt_length_l0l3 = TX_EQTR_HS_G6_ADAPT_DEFAULT;
|
|
}
|
|
|
|
adapt_eqtr = adapt_length_l0l3 |
|
|
(adapt_length_l0l1l2l3 << TX_EQTR_ADAPT_LENGTH_L0L1L2L3_SHIFT);
|
|
} else {
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (rec)
|
|
rec->saved_adapt_eqtr = (u16)adapt_eqtr;
|
|
|
|
set_adapt_eqtr:
|
|
ret = ufshcd_dme_set(hba, UIC_ARG_MIB(PA_TXADAPTLENGTH_EQTR), adapt_eqtr);
|
|
if (ret)
|
|
dev_err(hba->dev, "Failed to set adapt length for TX EQTR: %d\n", ret);
|
|
else
|
|
dev_dbg(hba->dev, "PA_TXADAPTLENGTH_EQTR configured to 0x%08x\n", adapt_eqtr);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* ufshcd_compose_tx_eqtr_setting - Compose TX EQTR setting
|
|
* @iter: TX EQTR iterator data structure
|
|
* @num_lanes: number of active lanes
|
|
*
|
|
* Returns composed TX EQTR setting, same setting is used for all active lanes
|
|
*/
|
|
static inline u32 ufshcd_compose_tx_eqtr_setting(struct tx_eqtr_iter *iter,
|
|
int num_lanes)
|
|
{
|
|
u32 setting = 0;
|
|
int lane;
|
|
|
|
for (lane = 0; lane < num_lanes; lane++) {
|
|
setting |= TX_HS_PRESHOOT_BITS(lane, iter->preshoot);
|
|
setting |= TX_HS_DEEMPHASIS_BITS(lane, iter->deemphasis);
|
|
}
|
|
|
|
return setting;
|
|
}
|
|
|
|
/**
|
|
* ufshcd_apply_tx_eqtr_settings - Apply TX EQTR setting
|
|
* @hba: per adapter instance
|
|
* @pwr_mode: target power mode containing gear and rate information
|
|
* @h_iter: host TX EQTR iterator data structure
|
|
* @d_iter: device TX EQTR iterator data structure
|
|
*
|
|
* Returns 0 on success, negative error code otherwise
|
|
*/
|
|
static int ufshcd_apply_tx_eqtr_settings(struct ufs_hba *hba,
|
|
struct ufs_pa_layer_attr *pwr_mode,
|
|
struct tx_eqtr_iter *h_iter,
|
|
struct tx_eqtr_iter *d_iter)
|
|
{
|
|
u32 setting;
|
|
int ret;
|
|
|
|
setting = ufshcd_compose_tx_eqtr_setting(h_iter, pwr_mode->lane_tx);
|
|
ret = ufshcd_dme_set(hba, UIC_ARG_MIB(PA_TXEQTRSETTING), setting);
|
|
if (ret)
|
|
return ret;
|
|
|
|
setting = ufshcd_compose_tx_eqtr_setting(d_iter, pwr_mode->lane_rx);
|
|
ret = ufshcd_dme_set(hba, UIC_ARG_MIB(PA_PEERTXEQTRSETTING), setting);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = ufshcd_vops_apply_tx_eqtr_settings(hba, pwr_mode, h_iter, d_iter);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* ufshcd_update_tx_eq_params - Update TX Equalization params
|
|
* @params: TX EQ parameters data structure
|
|
* @pwr_mode: target power mode containing gear and rate
|
|
* @eqtr_data: TX EQTR data structure
|
|
*
|
|
* Update TX Equalization params using results from TX EQTR data. Check also
|
|
* the TX EQTR FOM value for each TX lane in the TX EQTR data. If a TX lane got
|
|
* a FOM value of 0, restore the TX Equalization settings from the last known
|
|
* valid TX Equalization params for that specific TX lane.
|
|
*/
|
|
static inline void
|
|
ufshcd_update_tx_eq_params(struct ufshcd_tx_eq_params *params,
|
|
struct ufs_pa_layer_attr *pwr_mode,
|
|
struct ufshcd_tx_eqtr_data *eqtr_data)
|
|
{
|
|
struct ufshcd_tx_eqtr_record *rec = params->eqtr_record;
|
|
|
|
if (params->is_valid) {
|
|
int lane;
|
|
|
|
for (lane = 0; lane < pwr_mode->lane_tx; lane++)
|
|
if (eqtr_data->host[lane].fom_val == 0)
|
|
eqtr_data->host[lane] = params->host[lane];
|
|
|
|
for (lane = 0; lane < pwr_mode->lane_rx; lane++)
|
|
if (eqtr_data->device[lane].fom_val == 0)
|
|
eqtr_data->device[lane] = params->device[lane];
|
|
}
|
|
|
|
memcpy(params->host, eqtr_data->host, sizeof(params->host));
|
|
memcpy(params->device, eqtr_data->device, sizeof(params->device));
|
|
|
|
if (!rec)
|
|
return;
|
|
|
|
memcpy(rec->host_fom, eqtr_data->host_fom, sizeof(rec->host_fom));
|
|
memcpy(rec->device_fom, eqtr_data->device_fom, sizeof(rec->device_fom));
|
|
rec->last_record_ts = ktime_get();
|
|
rec->last_record_index++;
|
|
}
|
|
|
|
/**
|
|
* __ufshcd_tx_eqtr - TX Equalization Training (EQTR) procedure
|
|
* @hba: per adapter instance
|
|
* @params: TX EQ parameters data structure
|
|
* @pwr_mode: target power mode containing gear and rate information
|
|
*
|
|
* This function implements the complete TX EQTR procedure as defined in UFSHCI
|
|
* v5.0 specification. It iterates through all possible combinations of PreShoot
|
|
* and DeEmphasis settings to find the optimal TX Equalization settings for all
|
|
* active lanes.
|
|
*
|
|
* Returns 0 on success, negative error code otherwise
|
|
*/
|
|
static int __ufshcd_tx_eqtr(struct ufs_hba *hba,
|
|
struct ufshcd_tx_eq_params *params,
|
|
struct ufs_pa_layer_attr *pwr_mode)
|
|
{
|
|
struct ufshcd_tx_eqtr_data *eqtr_data __free(kfree) =
|
|
kzalloc(sizeof(*eqtr_data), GFP_KERNEL);
|
|
struct tx_eqtr_iter h_iter = {};
|
|
struct tx_eqtr_iter d_iter = {};
|
|
u32 gear = pwr_mode->gear_tx;
|
|
u8 preshoot, deemphasis;
|
|
ktime_t start;
|
|
int ret;
|
|
|
|
if (!eqtr_data)
|
|
return -ENOMEM;
|
|
|
|
dev_info(hba->dev, "Start TX EQTR procedure for HS-G%u, Rate-%s, RX Lanes: %u, TX Lanes: %u\n",
|
|
gear, ufs_hs_rate_to_str(pwr_mode->hs_rate),
|
|
pwr_mode->lane_rx, pwr_mode->lane_tx);
|
|
|
|
start = ktime_get();
|
|
|
|
/* Step 1 - Determine the TX Adapt Length for EQTR */
|
|
ret = ufshcd_setup_tx_eqtr_adapt_length(hba, params, gear);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to setup TX EQTR Adaptation length: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Step 2 - Determine TX Equalization setting capabilities */
|
|
ret = ufshcd_tx_eqtr_iter_init(hba, &h_iter, &d_iter);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to init TX EQTR data: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
/* TX EQTR main loop */
|
|
for (preshoot = 0; preshoot < TX_HS_NUM_PRESHOOT; preshoot++) {
|
|
for (deemphasis = 0; deemphasis < TX_HS_NUM_DEEMPHASIS; deemphasis++) {
|
|
if (!tx_eqtr_iter_update(preshoot, deemphasis, &h_iter, &d_iter))
|
|
continue;
|
|
|
|
/* Step 3 - Apply TX EQTR settings */
|
|
ret = ufshcd_apply_tx_eqtr_settings(hba, pwr_mode, &h_iter, &d_iter);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to apply TX EQTR settings (PreShoot %u, DeEmphasis %u): %d\n",
|
|
preshoot, deemphasis, ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Step 4 - Trigger UIC TX EQTR */
|
|
ret = ufshcd_uic_tx_eqtr(hba, gear);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to trigger UIC TX EQTR for target gear %u: %d\n",
|
|
gear, ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Step 5 - Get FOM */
|
|
ret = ufshcd_get_rx_fom(hba, pwr_mode, &h_iter, &d_iter);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to get RX_FOM: %d\n",
|
|
ret);
|
|
return ret;
|
|
}
|
|
|
|
ufshcd_evaluate_tx_eqtr_fom(hba, pwr_mode, eqtr_data, &h_iter, &d_iter);
|
|
}
|
|
}
|
|
|
|
dev_info(hba->dev, "TX EQTR procedure completed! Time elapsed: %llu ms\n",
|
|
ktime_to_ms(ktime_sub(ktime_get(), start)));
|
|
|
|
ufshcd_update_tx_eq_params(params, pwr_mode, eqtr_data);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* ufshcd_tx_eqtr_prepare - Prepare UFS link for TX EQTR procedure
|
|
* @hba: per adapter instance
|
|
* @pwr_mode: target power mode containing gear and rate
|
|
*
|
|
* This function prepares the UFS link for TX Equalization Training (EQTR) by
|
|
* establishing the proper initial conditions required by the EQTR procedure.
|
|
* It ensures that EQTR starts from the most reliable Power Mode (HS-G1) with
|
|
* all connected lanes activated and sets host TX HS Adapt Type to INITIAL.
|
|
*
|
|
* Returns 0 on successful preparation, negative error code on failure
|
|
*/
|
|
static int ufshcd_tx_eqtr_prepare(struct ufs_hba *hba,
|
|
struct ufs_pa_layer_attr *pwr_mode)
|
|
{
|
|
struct ufs_pa_layer_attr pwr_mode_hs_g1 = {
|
|
/* TX EQTR shall be initiated from the most reliable HS-G1 */
|
|
.gear_rx = UFS_HS_G1,
|
|
.gear_tx = UFS_HS_G1,
|
|
.lane_rx = pwr_mode->lane_rx,
|
|
.lane_tx = pwr_mode->lane_tx,
|
|
.pwr_rx = FAST_MODE,
|
|
.pwr_tx = FAST_MODE,
|
|
/* Use the target power mode's HS rate */
|
|
.hs_rate = pwr_mode->hs_rate,
|
|
};
|
|
u32 rate = pwr_mode->hs_rate;
|
|
int ret;
|
|
|
|
/* Change power mode to HS-G1, activate all connected lanes. */
|
|
ret = ufshcd_change_power_mode(hba, &pwr_mode_hs_g1,
|
|
UFSHCD_PMC_POLICY_DONT_FORCE);
|
|
if (ret) {
|
|
dev_err(hba->dev, "TX EQTR: Failed to change power mode to HS-G1, Rate-%s: %d\n",
|
|
ufs_hs_rate_to_str(rate), ret);
|
|
return ret;
|
|
}
|
|
|
|
ret = ufshcd_dme_set(hba, UIC_ARG_MIB(PA_TXHSADAPTTYPE),
|
|
PA_INITIAL_ADAPT);
|
|
if (ret)
|
|
dev_err(hba->dev, "TX EQTR: Failed to set Host Adapt type to INITIAL: %d\n",
|
|
ret);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void ufshcd_tx_eqtr_unprepare(struct ufs_hba *hba,
|
|
struct ufs_pa_layer_attr *pwr_mode)
|
|
{
|
|
int err;
|
|
|
|
if (pwr_mode->pwr_rx == SLOWAUTO_MODE || pwr_mode->hs_rate == 0)
|
|
return;
|
|
|
|
err = ufshcd_change_power_mode(hba, pwr_mode,
|
|
UFSHCD_PMC_POLICY_DONT_FORCE);
|
|
if (err)
|
|
dev_err(hba->dev, "%s: Failed to restore Power Mode: %d\n",
|
|
__func__, err);
|
|
}
|
|
|
|
/**
|
|
* ufshcd_tx_eqtr - Perform TX EQTR procedures with vops callbacks
|
|
* @hba: per adapter instance
|
|
* @params: TX EQ parameters data structure to populate
|
|
* @pwr_mode: target power mode containing gear and rate information
|
|
*
|
|
* This is the main entry point for performing TX Equalization Training (EQTR)
|
|
* procedure as defined in UFSCHI v5.0 specification. It serves as a wrapper
|
|
* around __ufshcd_tx_eqtr() to provide vops support through the variant
|
|
* operations framework.
|
|
*
|
|
* Returns 0 on success, negative error code on failure
|
|
*/
|
|
static int ufshcd_tx_eqtr(struct ufs_hba *hba,
|
|
struct ufshcd_tx_eq_params *params,
|
|
struct ufs_pa_layer_attr *pwr_mode)
|
|
{
|
|
struct ufs_pa_layer_attr old_pwr_info;
|
|
int ret;
|
|
|
|
if (!params->eqtr_record) {
|
|
params->eqtr_record = devm_kzalloc(hba->dev,
|
|
sizeof(*params->eqtr_record),
|
|
GFP_KERNEL);
|
|
if (!params->eqtr_record)
|
|
return -ENOMEM;
|
|
}
|
|
|
|
memcpy(&old_pwr_info, &hba->pwr_info, sizeof(struct ufs_pa_layer_attr));
|
|
|
|
ret = ufshcd_tx_eqtr_prepare(hba, pwr_mode);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to prepare TX EQTR: %d\n", ret);
|
|
goto out;
|
|
}
|
|
|
|
ret = ufshcd_vops_tx_eqtr_notify(hba, PRE_CHANGE, pwr_mode);
|
|
if (ret)
|
|
goto out;
|
|
|
|
ret = __ufshcd_tx_eqtr(hba, params, pwr_mode);
|
|
if (ret)
|
|
goto out;
|
|
|
|
ret = ufshcd_vops_tx_eqtr_notify(hba, POST_CHANGE, pwr_mode);
|
|
|
|
out:
|
|
if (ret)
|
|
ufshcd_tx_eqtr_unprepare(hba, &old_pwr_info);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* ufshcd_config_tx_eq_settings - Configure TX Equalization settings
|
|
* @hba: per adapter instance
|
|
* @pwr_mode: target power mode containing gear and rate information
|
|
* @force_tx_eqtr: execute the TX EQTR procedure
|
|
*
|
|
* This function finds and sets the TX Equalization settings for the given
|
|
* target power mode.
|
|
*
|
|
* Returns 0 on success, error code otherwise
|
|
*/
|
|
int ufshcd_config_tx_eq_settings(struct ufs_hba *hba,
|
|
struct ufs_pa_layer_attr *pwr_mode,
|
|
bool force_tx_eqtr)
|
|
{
|
|
struct ufshcd_tx_eq_params *params;
|
|
u32 gear, rate;
|
|
|
|
if (!ufshcd_is_tx_eq_supported(hba) || !use_adaptive_txeq)
|
|
return 0;
|
|
|
|
if (!hba->max_pwr_info.is_valid) {
|
|
dev_err(hba->dev, "Max power info is invalid\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!pwr_mode) {
|
|
dev_err(hba->dev, "Target power mode is NULL\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
gear = pwr_mode->gear_tx;
|
|
rate = pwr_mode->hs_rate;
|
|
|
|
if (gear < UFS_HS_G1 || gear > UFS_HS_GEAR_MAX) {
|
|
dev_err(hba->dev, "Invalid HS-Gear (%u) for TX Equalization\n",
|
|
gear);
|
|
return -EINVAL;
|
|
} else if (gear < max_t(u32, adaptive_txeq_gear, UFS_HS_G4)) {
|
|
/* TX EQTR is supported for HS-G4 and higher Gears */
|
|
return 0;
|
|
}
|
|
|
|
if (rate != PA_HS_MODE_A && rate != PA_HS_MODE_B) {
|
|
dev_err(hba->dev, "Invalid HS-Rate (%u) for TX Equalization\n",
|
|
rate);
|
|
return -EINVAL;
|
|
}
|
|
|
|
params = &hba->tx_eq_params[gear - 1];
|
|
if (!params->is_valid || force_tx_eqtr) {
|
|
int ret;
|
|
|
|
ret = ufshcd_tx_eqtr(hba, params, pwr_mode);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to train TX Equalization for HS-G%u, Rate-%s: %d\n",
|
|
gear, ufs_hs_rate_to_str(rate), ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Mark TX Equalization settings as valid */
|
|
params->is_valid = true;
|
|
params->is_applied = false;
|
|
}
|
|
|
|
if (params->is_valid && !params->is_applied) {
|
|
int ret;
|
|
|
|
ret = ufshcd_apply_tx_eq_settings(hba, params, gear);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to apply TX Equalization settings for HS-G%u, Rate-%s: %d\n",
|
|
gear, ufs_hs_rate_to_str(rate), ret);
|
|
return ret;
|
|
}
|
|
|
|
params->is_applied = true;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* ufshcd_apply_valid_tx_eq_settings - Apply valid TX Equalization settings
|
|
* @hba: per-adapter instance
|
|
*
|
|
* This function iterates through all supported High-Speed (HS) gears and
|
|
* applies valid TX Equalization settings to both Host and Device.
|
|
*/
|
|
void ufshcd_apply_valid_tx_eq_settings(struct ufs_hba *hba)
|
|
{
|
|
struct ufshcd_tx_eq_params *params;
|
|
int gear, err;
|
|
|
|
if (!ufshcd_is_tx_eq_supported(hba))
|
|
return;
|
|
|
|
if (!hba->max_pwr_info.is_valid) {
|
|
dev_err(hba->dev, "Max power info is invalid, cannot apply TX Equalization settings\n");
|
|
return;
|
|
}
|
|
|
|
for (gear = UFS_HS_G1; gear <= UFS_HS_GEAR_MAX; gear++) {
|
|
params = &hba->tx_eq_params[gear - 1];
|
|
|
|
if (params->is_valid) {
|
|
err = ufshcd_apply_tx_eq_settings(hba, params, gear);
|
|
if (err) {
|
|
params->is_applied = false;
|
|
dev_err(hba->dev, "Failed to apply TX Equalization settings for HS-G%u: %d\n",
|
|
gear, err);
|
|
} else {
|
|
params->is_applied = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* ufshcd_retrain_tx_eq - Retrain TX Equalization and apply new settings
|
|
* @hba: per-adapter instance
|
|
* @gear: target High-Speed (HS) gear for retraining
|
|
*
|
|
* This function initiates a refresh of the TX Equalization settings for a
|
|
* specific HS gear. It scales the clocks to maximum frequency, negotiates the
|
|
* power mode with the device, retrains TX EQ and applies new TX EQ settings
|
|
* by conducting a Power Mode change.
|
|
*
|
|
* Returns 0 on success, non-zero error code otherwise
|
|
*/
|
|
int ufshcd_retrain_tx_eq(struct ufs_hba *hba, u32 gear)
|
|
{
|
|
struct ufs_pa_layer_attr new_pwr_info, final_params = {};
|
|
int ret;
|
|
|
|
if (!ufshcd_is_tx_eq_supported(hba) || !use_adaptive_txeq)
|
|
return -EOPNOTSUPP;
|
|
|
|
if (gear < adaptive_txeq_gear)
|
|
return -ERANGE;
|
|
|
|
ufshcd_hold(hba);
|
|
|
|
ret = ufshcd_pause_command_processing(hba, 1 * USEC_PER_SEC);
|
|
if (ret) {
|
|
ufshcd_release(hba);
|
|
return ret;
|
|
}
|
|
|
|
/* scale up clocks to max frequency before TX EQTR */
|
|
if (ufshcd_is_clkscaling_supported(hba))
|
|
ufshcd_scale_clks(hba, ULONG_MAX, true);
|
|
|
|
new_pwr_info = hba->pwr_info;
|
|
new_pwr_info.gear_tx = gear;
|
|
new_pwr_info.gear_rx = gear;
|
|
|
|
ret = ufshcd_vops_negotiate_pwr_mode(hba, &new_pwr_info, &final_params);
|
|
if (ret)
|
|
memcpy(&final_params, &new_pwr_info, sizeof(final_params));
|
|
|
|
if (final_params.gear_tx != gear) {
|
|
dev_err(hba->dev, "Negotiated Gear (%u) does not match target Gear (%u)\n",
|
|
final_params.gear_tx, gear);
|
|
ret = -EINVAL;
|
|
goto out;
|
|
}
|
|
|
|
ret = ufshcd_config_tx_eq_settings(hba, &final_params, true);
|
|
if (ret) {
|
|
dev_err(hba->dev, "Failed to config TX Equalization for HS-G%u, Rate-%s: %d\n",
|
|
final_params.gear_tx,
|
|
ufs_hs_rate_to_str(final_params.hs_rate), ret);
|
|
goto out;
|
|
}
|
|
|
|
/* Change Power Mode to apply the new TX EQ settings */
|
|
ret = ufshcd_change_power_mode(hba, &final_params,
|
|
UFSHCD_PMC_POLICY_FORCE);
|
|
if (ret)
|
|
dev_err(hba->dev, "%s: Failed to change Power Mode to HS-G%u, Rate-%s: %d\n",
|
|
__func__, final_params.gear_tx,
|
|
ufs_hs_rate_to_str(final_params.hs_rate), ret);
|
|
|
|
out:
|
|
ufshcd_resume_command_processing(hba);
|
|
ufshcd_release(hba);
|
|
|
|
return ret;
|
|
}
|