Hi, On Thursday 15 October 2015 08:38 AM, Alim Akhtar wrote: > +CCing kishon Vijay, > > On 10/14/2015 06:25 PM, Alim Akhtar wrote: >> From: Seungwon Jeon <essuuj@xxxxxxxxx> >> >> This patch introduces Exynos UFS host controller driver, >> which mainly handles vendor-specific operations including >> link startup, power mode change and hibernation/unhibernation. >> >> Signed-off-by: Seungwon Jeon <essuuj@xxxxxxxxx> >> Signed-off-by: Alim Akhtar <alim.akhtar@xxxxxxxxxxx> >> --- >> drivers/scsi/ufs/Kconfig | 12 + >> drivers/scsi/ufs/Makefile | 1 + >> drivers/scsi/ufs/ufs-exynos-hw.c | 131 ++++ >> drivers/scsi/ufs/ufs-exynos-hw.h | 43 ++ >> drivers/scsi/ufs/ufs-exynos.c | 1317 >> ++++++++++++++++++++++++++++++++++++++ >> drivers/scsi/ufs/ufs-exynos.h | 247 +++++++ >> drivers/scsi/ufs/ufshci.h | 26 +- >> drivers/scsi/ufs/unipro.h | 47 ++ >> 8 files changed, 1823 insertions(+), 1 deletion(-) >> create mode 100644 drivers/scsi/ufs/ufs-exynos-hw.c >> create mode 100644 drivers/scsi/ufs/ufs-exynos-hw.h >> create mode 100644 drivers/scsi/ufs/ufs-exynos.c >> create mode 100644 drivers/scsi/ufs/ufs-exynos.h >> . . <snip> . . >> diff --git a/drivers/scsi/ufs/ufs-exynos-hw.c >> b/drivers/scsi/ufs/ufs-exynos-hw.c >> new file mode 100644 >> index 000000000000..be6c61541a8f >> --- /dev/null >> +++ b/drivers/scsi/ufs/ufs-exynos-hw.c >> @@ -0,0 +1,131 @@ . . <snip> . . >> + >> +#define PWR_MODE_STR_LEN 64 >> +static int exynos_ufs_post_pwr_mode(struct ufs_hba *hba, >> + struct ufs_pa_layer_attr *pwr_max, >> + struct ufs_pa_layer_attr *pwr_req) >> +{ >> + struct exynos_ufs *ufs = to_exynos_ufs(hba); >> + struct exynos_ufs_phy_info *phy_info = phy_get_drvdata(ufs->phy); This is abusing the interface. phy_get_drvdata is meant to be used only by the PHY driver. >> + struct exynos_ufs_phy_specific_ops *phy_ops = >> + phy_info->phy_specific_ops; I'm really not happy about having platform specific ops for PHY. We have to see if existing PHY ops can be used for this or in worst case add new PHY ops. >> + struct uic_pwr_mode *pwr = &ufs->pwr_act; >> + char pwr_str[PWR_MODE_STR_LEN] = ""; >> + int ret = 0; >> + >> + if (ufs->drv_data->post_pwr_change) >> + ufs->drv_data->post_pwr_change(ufs, pwr); >> + >> + if (IS_UFS_PWR_MODE_HS(pwr->mode)) { >> + switch (pwr->hs_series) { >> + case PA_HS_MODE_A: >> + case PA_HS_MODE_B: >> + phy_ops->calibrate_phy(ufs->phy, CFG_POST_PWR_HS, >> + PWR_MODE_HS(pwr->gear, pwr->hs_series)); >> + break; >> + } >> + >> + ret = phy_ops->wait_for_lock_acq(ufs->phy); >> + snprintf(pwr_str, sizeof(pwr_str), "Fast%s series_%s G_%d L_%d", >> + pwr->mode == FASTAUTO_MODE ? "_Auto" : "", >> + pwr->hs_series == PA_HS_MODE_A ? "A" : "B", >> + pwr->gear, pwr->lane); >> + } else if (IS_UFS_PWR_MODE_PWM(pwr->mode)) { >> + snprintf(pwr_str, sizeof(pwr_str), "Slow%s G_%d L_%d", >> + pwr->mode == SLOWAUTO_MODE ? "_Auto" : "", >> + pwr->gear, pwr->lane); >> + } >> + >> + dev_info(hba->dev, "Power mode change %d : %s\n", ret, pwr_str); >> + return ret; >> +} >> + >> +static void exynos_ufs_specify_nexus_t_xfer_req(struct ufs_hba *hba, >> + int tag, struct scsi_cmnd *cmd) >> +{ >> + struct exynos_ufs *ufs = to_exynos_ufs(hba); >> + u32 type; >> + >> + type = hci_readl(ufs, HCI_UTRL_NEXUS_TYPE); >> + >> + if (cmd) >> + hci_writel(ufs, type | (1 << tag), HCI_UTRL_NEXUS_TYPE); >> + else >> + hci_writel(ufs, type & ~(1 << tag), HCI_UTRL_NEXUS_TYPE); >> +} >> + >> +static void exynos_ufs_specify_nexus_t_tm_req(struct ufs_hba *hba, >> + int tag, u8 func) >> +{ >> + struct exynos_ufs *ufs = to_exynos_ufs(hba); >> + u32 type; >> + >> + type = hci_readl(ufs, HCI_UTMRL_NEXUS_TYPE); >> + >> + switch (func) { >> + case UFS_ABORT_TASK: >> + case UFS_QUERY_TASK: >> + hci_writel(ufs, type | (1 << tag), HCI_UTMRL_NEXUS_TYPE); >> + break; >> + case UFS_ABORT_TASK_SET: >> + case UFS_CLEAR_TASK_SET: >> + case UFS_LOGICAL_RESET: >> + case UFS_QUERY_TASK_SET: >> + hci_writel(ufs, type & ~(1 << tag), HCI_UTMRL_NEXUS_TYPE); >> + break; >> + } >> +} >> + >> +static void exynos_ufs_phy_init(struct exynos_ufs *ufs) >> +{ >> + struct ufs_hba *hba = ufs->hba; >> + struct exynos_ufs_phy_info *phy_info = phy_get_drvdata(ufs->phy); >> + struct exynos_ufs_phy_specific_ops *phy_ops = >> + phy_info->phy_specific_ops; >> + >> + if (ufs->avail_ln_rx == 0 || ufs->avail_ln_tx == 0) { >> + ufshcd_dme_get(hba, UIC_ARG_MIB(PA_AVAILRXDATALANES), >> + &ufs->avail_ln_rx); >> + ufshcd_dme_get(hba, UIC_ARG_MIB(PA_AVAILTXDATALANES), >> + &ufs->avail_ln_tx); >> + WARN(ufs->avail_ln_rx != ufs->avail_ln_tx, >> + "available data lane is not equal(rx:%d, tx:%d)\n", >> + ufs->avail_ln_rx, ufs->avail_ln_tx); >> + } >> + >> + phy_ops->set_lane_cnt(ufs->phy, ufs->avail_ln_rx); can't bus_width attribute in phy core be reused for this? >> + phy_ops->calibrate_phy(ufs->phy, CFG_PRE_INIT, PWR_MODE_ANY); Why can't calibrate PHY be directly done in phy_init? Thanks Kishon -- To unsubscribe from this list: send the line "unsubscribe linux-scsi" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html