Hi,

On Sunday 25 October 2015 05:34 PM, Alim Akhtar wrote:
> Hi Kishon
> Thanks again for you review.
> 
> On Fri, Oct 23, 2015 at 8:48 PM, Kishon Vijay Abraham I <kis...@ti.com> wrote:
>> 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 <ess...@gmail.com>
>>>>
>>>> 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 <ess...@gmail.com>
>>>> Signed-off-by: Alim Akhtar <alim.akh...@samsung.com>
>>>> ---
>>>>   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.
> Well you said you like the controller driver to use only PHY ops[1], I
> am sorry If I misunderstood that point, can you please help me to
> understand that?

I meant PHY generic ops and not PHY ops.
> [1]-> https://lkml.org/lkml/2015/9/18/29
> 
>>>> +    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?
>>
> I will take a look on it.
> 
>>>> +    phy_ops->calibrate_phy(ufs->phy, CFG_PRE_INIT, PWR_MODE_ANY);
>>
>> Why can't calibrate PHY be directly done in phy_init?
>>
> This is just one instance, need to calibrate PHY when the ufs pwr mode 
> changes.

Then maybe we should check when the power mode changes and see if it
makes sense to calibrate PHY in other generic PHY ops like phy_power_on
etc..

Thanks
Kishon
--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to