Received: by 2002:a05:6358:c692:b0:131:369:b2a3 with SMTP id fe18csp1020245rwb; Wed, 26 Jul 2023 06:33:11 -0700 (PDT) X-Google-Smtp-Source: APBJJlHE0AXymXr+Xpwzt8jM8IfyooUfoZg0y9tGRmpJxEjU+Tn011mS3Mk5BykpDn4k4AGKIwPC X-Received: by 2002:a17:907:7883:b0:994:4be4:a106 with SMTP id ku3-20020a170907788300b009944be4a106mr1631680ejc.10.1690378391476; Wed, 26 Jul 2023 06:33:11 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1690378391; cv=none; d=google.com; s=arc-20160816; b=L22I/Y5BSjcnVmONa/IJGvbsUQxzWVEDakgptlRtXoYCkgodFFwSfmqUU8aekjXJiK JwJlXgiWnYOoC7hELodRH4960k6ZrbwBiL5Z/EYk2PXsJRKWGSmGlk+Qa3l8iHvFS1ps y8TNT5gGPU3Aga8O8iv1uDosWzmOkwBJFnSzYhEux7dclVldMhd6SH6Mu5HfZsE5lojU mbYvC6SSo/OUpZGc86ImVqdNAVo2rfAABZFXVTR9PkChhYrVQrcZ1mX0b/61F0SgKXRW Ol3VbL4cUs6zJOv899WeRDjmTIOvpLCwlaykzCbpSr9PUIsI7F65+gWClSBDoFCgtkmb TeIQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:in-reply-to :organization:content-language:references:cc:to:subject:from :user-agent:mime-version:date:message-id:dkim-signature; bh=brXTQfmpfpmIfqTTmZrELgPbXyKXtpSbI0FZyGp+Bgs=; fh=el5piC8mKZJDr6E8Xybob2ZZOS9Pdfy5bWzwPWAeV0M=; b=T+ivFG3jf9+4lP8MhDgWHQ/isxnv1a3aqa+DLgBecVIg4yRZtUEcFzL2QdJpNKx6h/ LOU9Yuix/RUUXu4WxZgPGeImj7kkTMA3Q//jTd2FMIslVDms8R+SICXLRrcZGf1VdV5y WfQg4EzbU3FBvs+MAfyWzQmUK1rBfY6mGmcOoJGt6J6kR+N/DLG7g5Yp5bZgVfgZzl0T 2Gl9DhZACQiBbetFkAPytO1yh/VUSp/KGdNKMElbKaV0VIMyAJ4z3730qZakvfqLQCIv I2yhoAATaEi8B03Oa+sxwptv5xB6R+v+3n5JlkHryU+Kj5/pHUb55MnbZV1SPhmzn5XH AVCQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@intel.com header.s=Intel header.b=BQw5Dpmm; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=intel.com Return-Path: Received: from out1.vger.email (out1.vger.email. [2620:137:e000::1:20]) by mx.google.com with ESMTP id o4-20020a1709062e8400b009924806ccffsi9700609eji.488.2023.07.26.06.32.45; Wed, 26 Jul 2023 06:33:11 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) client-ip=2620:137:e000::1:20; Authentication-Results: mx.google.com; dkim=pass header.i=@intel.com header.s=Intel header.b=BQw5Dpmm; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:20 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=intel.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S233829AbjGZMmV (ORCPT + 99 others); Wed, 26 Jul 2023 08:42:21 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:46688 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S232199AbjGZMmS (ORCPT ); Wed, 26 Jul 2023 08:42:18 -0400 Received: from mga12.intel.com (mga12.intel.com [192.55.52.136]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 3C4D32723; Wed, 26 Jul 2023 05:42:08 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=intel.com; i=@intel.com; q=dns/txt; s=Intel; t=1690375328; x=1721911328; h=message-id:date:mime-version:from:subject:to:cc: references:in-reply-to:content-transfer-encoding; bh=TWtgsmghQDLwK7tptYOI7FgghGHT0sYpAbG3HffuvJo=; b=BQw5Dpmm0bPZugF8oO4GF7Xrnl6vzkWtOhcv81GvTQbwEzx8VNY8KpNb s/ElzRLndx4J6Xhl2BYv5ww63N/jdNyFJMpcyTgH9GOzU0vxKF9n9FynU lNsulxhWMDp6b4FdpVHFZ35bfovu1zmEB3aTI0JwBBBKfGwe5fLBwdqnO yC8CH2fx4Xh9wjoKUFM7bu8SHMfDQHgWuUh6GyXOVHHiLB+zxyL6WjALd +WJYpIH7SZkLHtKnSsABs8t1/sU7xfPO0HxyAEaZGCrezMMSV+MpEUeJz /NhlhMq238hERXZ92cAXlIAhxRCncXpQxgUDrV9dzDShWUOtiVZgQARWl w==; X-IronPort-AV: E=McAfee;i="6600,9927,10783"; a="347617559" X-IronPort-AV: E=Sophos;i="6.01,232,1684825200"; d="scan'208";a="347617559" Received: from orsmga003.jf.intel.com ([10.7.209.27]) by fmsmga106.fm.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 26 Jul 2023 05:42:07 -0700 X-ExtLoop1: 1 X-IronPort-AV: E=McAfee;i="6600,9927,10783"; a="676683382" X-IronPort-AV: E=Sophos;i="6.01,232,1684825200"; d="scan'208";a="676683382" Received: from ahunter6-mobl1.ger.corp.intel.com (HELO [10.0.2.15]) ([10.252.35.75]) by orsmga003-auth.jf.intel.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 26 Jul 2023 05:42:02 -0700 Message-ID: Date: Wed, 26 Jul 2023 15:41:58 +0300 MIME-Version: 1.0 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:102.0) Gecko/20100101 Firefox/102.0 Thunderbird/102.13.1 From: Adrian Hunter Subject: Re: [PATCH v4 3/6] mmc: sdhci-cadence: SD6 controller support To: Piyush Malgujar , linux-mmc@vger.kernel.org, linux-kernel@vger.kernel.org, ulf.hansson@linaro.org, p.zabel@pengutronix.de, robh+dt@kernel.org, krzysztof.kozlowski+dt@linaro.org, conor+dt@kernel.org, yamada.masahiro@socionext.com, devicetree@vger.kernel.org Cc: jannadurai@marvell.com, cchavva@marvell.com, Dhananjay Kangude References: <20230717125146.16791-1-pmalgujar@marvell.com> <20230717125146.16791-4-pmalgujar@marvell.com> Content-Language: en-US Organization: Intel Finland Oy, Registered Address: PL 281, 00181 Helsinki, Business Identity Code: 0357606 - 4, Domiciled in Helsinki In-Reply-To: <20230717125146.16791-4-pmalgujar@marvell.com> Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit X-Spam-Status: No, score=-4.5 required=5.0 tests=BAYES_00,DKIMWL_WL_HIGH, DKIM_SIGNED,DKIM_VALID,DKIM_VALID_AU,DKIM_VALID_EF,NICE_REPLY_A, RCVD_IN_DNSWL_MED,SPF_HELO_PASS,SPF_NONE,T_SCC_BODY_TEXT_LINE, URIBL_BLOCKED autolearn=ham autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 17/07/23 15:51, Piyush Malgujar wrote: > From: Dhananjay Kangude > > Add support for SD6 controller for Marvell CN10k SoCs and related ops > along with support for HS400 and HS400ES emmc modes. > Updated HS200 tuning values and support to read tune configuration > from FDT and support to configure and read host side drive strength, > slew from device tree. > > Signed-off-by: Dhananjay Kangude > Co-developed-by: Jayanthi Annadurai > Signed-off-by: Jayanthi Annadurai > Signed-off-by: Piyush Malgujar > --- > drivers/mmc/host/sdhci-cadence.c | 1256 +++++++++++++++++++++++++++++- > 1 file changed, 1253 insertions(+), 3 deletions(-) > > diff --git a/drivers/mmc/host/sdhci-cadence.c b/drivers/mmc/host/sdhci-cadence.c > index 98fe752bcf27a71607623f3cb1c36f1a16d688a4..8bcf585185053b0afaff2625d62316cec1824fa3 100644 > --- a/drivers/mmc/host/sdhci-cadence.c > +++ b/drivers/mmc/host/sdhci-cadence.c > @@ -16,8 +16,20 @@ > > #include "sdhci-pltfm.h" > > +#define SDMCLK_MAX_FREQ 200000000 > + > +#define DEFAULT_CMD_DELAY 16 > +#define SDHCI_CDNS_TUNE_START 16 > +#define SDHCI_CDNS_TUNE_STEP 6 > +#define SDHCI_CDNS_TUNE_ITERATIONS 40 > + > +#define SDHCI_CDNS_HRS00 0x00 > +#define SDHCI_CDNS_HRS00_SWR BIT(0) > + > +#define SDHCI_CDNS_HRS02 0x08 /* PHY access port */ > +#define SDHCI_CDNS_HRS04 0x10 /* PHY access port */ > + > /* SD 4.0 Controller HRS - Host Register Set (specific to Cadence) */ > -#define SDHCI_CDNS_HRS04 0x10 /* PHY access port */ > #define SDHCI_CDNS_SD4_HRS04_ACK BIT(26) > #define SDHCI_CDNS_SD4_HRS04_RD BIT(25) > #define SDHCI_CDNS_SD4_HRS04_WR BIT(24) > @@ -30,12 +42,89 @@ > #define SDHCI_CDNS_HRS06_TUNE GENMASK(13, 8) > #define SDHCI_CDNS_HRS06_MODE GENMASK(2, 0) > #define SDHCI_CDNS_HRS06_MODE_SD 0x0 > +#define SDHCI_CDNS_HRS06_MODE_LEGACY 0x1 The other field names have an extra 2 spaces before them, but not this one. One way or the other, please be consistent. > #define SDHCI_CDNS_HRS06_MODE_MMC_SDR 0x2 > #define SDHCI_CDNS_HRS06_MODE_MMC_DDR 0x3 > #define SDHCI_CDNS_HRS06_MODE_MMC_HS200 0x4 > #define SDHCI_CDNS_HRS06_MODE_MMC_HS400 0x5 > #define SDHCI_CDNS_HRS06_MODE_MMC_HS400ES 0x6 > > +/* SD 6.0 Controller HRS - Host Register Set (Specific to Cadence) */ > +#define SDHCI_CDNS_SD6_HRS04_ADDR GENMASK(15, 0) > + > +#define SDHCI_CDNS_HRS05 0x14 > + > +#define SDHCI_CDNS_HRS07 0x1C > +#define SDHCI_CDNS_HRS07_RW_COMPENSATE GENMASK(20, 16) > +#define SDHCI_CDNS_HRS07_IDELAY_VAL GENMASK(4, 0) There is a tab used above that expands to only 1 space (kernel expects tabs at 8 column intervals), which might as well have been a space, except the original style was to put 2 extra spaces before field names. One way or the other, please be consistent. > + > +#define SDHCI_CDNS_HRS09 0x24 > +#define SDHCI_CDNS_HRS09_RDDATA_EN BIT(16) > +#define SDHCI_CDNS_HRS09_RDCMD_EN BIT(15) > +#define SDHCI_CDNS_HRS09_EXTENDED_WR_MODE BIT(3) > +#define SDHCI_CDNS_HRS09_EXTENDED_RD_MODE BIT(2) > +#define SDHCI_CDNS_HRS09_PHY_INIT_COMPLETE BIT(1) > +#define SDHCI_CDNS_HRS09_PHY_SW_RESET BIT(0) > + > +#define SDHCI_CDNS_HRS10 0x28 > +#define SDHCI_CDNS_HRS10_HCSDCLKADJ GENMASK(19, 16) > + > +#define SDHCI_CDNS_HRS11 0x2c > +/*Reset related*/ > +#define SDHCI_CDNS_SRS11_SW_RESET_ALL BIT(24) > +#define SDHCI_CDNS_SRS11_SW_RESET_CMD BIT(25) > +#define SDHCI_CDNS_SRS11_SW_RESET_DAT BIT(26) > + > +#define SDHCI_CDNS_HRS16 0x40 > +#define SDHCI_CDNS_HRS16_WRDATA1_SDCLK_DLY GENMASK(31, 28) > +#define SDHCI_CDNS_HRS16_WRDATA0_SDCLK_DLY GENMASK(27, 24) > +#define SDHCI_CDNS_HRS16_WRCMD1_SDCLK_DLY GENMASK(23, 20) > +#define SDHCI_CDNS_HRS16_WRCMD0_SDCLK_DLY GENMASK(19, 16) > +#define SDHCI_CDNS_HRS16_WRDATA1_DLY GENMASK(15, 12) > +#define SDHCI_CDNS_HRS16_WRDATA0_DLY GENMASK(11, 8) > +#define SDHCI_CDNS_HRS16_WRCMD1_DLY GENMASK(7, 4) > +#define SDHCI_CDNS_HRS16_WRCMD0_DLY GENMASK(3, 0) > + > +/* PHY registers for SD6 controller */ > +#define SDHCI_CDNS_SD6_PHY_DQ_TIMING 0x2000 > +#define SDHCI_CDNS_SD6_PHY_DQ_TIMING_IO_MASK_ALWAYS_ON BIT(31) > +#define SDHCI_CDNS_SD6_PHY_DQ_TIMING_IO_MASK_END GENMASK(29, 27) > +#define SDHCI_CDNS_SD6_PHY_DQ_TIMING_IO_MASK_START GENMASK(26, 24) > +#define SDHCI_CDNS_SD6_PHY_DQ_TIMING_DATA_SELECT_OE_END GENMASK(2, 0) > + > +#define SDHCI_CDNS_SD6_PHY_DQS_TIMING 0x2004 > +#define SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_EXT_LPBK_DQS BIT(22) > +#define SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_LPBK_DQS BIT(21) > +#define SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_PHONY_DQS BIT(20) > +#define SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_PHONY_DQS_CMD BIT(19) > + > +#define SDHCI_CDNS_SD6_PHY_GATE_LPBK 0x2008 > +#define SDHCI_CDNS_SD6_PHY_GATE_LPBK_SYNC_METHOD BIT(31) > +#define SDHCI_CDNS_SD6_PHY_GATE_LPBK_SW_HALF_CYCLE_SHIFT BIT(28) > +#define SDHCI_CDNS_SD6_PHY_GATE_LPBK_RD_DEL_SEL GENMASK(24, 19) > +#define SDHCI_CDNS_SD6_PHY_GATE_LPBK_GATE_CFG_ALWAYS_ON BIT(6) > + > +#define SDHCI_CDNS_SD6_PHY_DLL_MASTER 0x200C > +#define SDHCI_CDNS_SD6_PHY_DLL_MASTER_BYPASS_MODE BIT(23) > +#define SDHCI_CDNS_SD6_PHY_DLL_MASTER_PHASE_DETECT_SEL GENMASK(22, 20) > +#define SDHCI_CDNS_SD6_PHY_DLL_MASTER_DLL_LOCK_NUM GENMASK(18, 16) > +#define SDHCI_CDNS_SD6_PHY_DLL_MASTER_DLL_START_POINT GENMASK(7, 0) > + > +#define SDHCI_CDNS_SD6_PHY_DLL_SLAVE 0x2010 > +#define SDHCI_CDNS_SD6_PHY_DLL_SLAVE_READ_DQS_CMD_DELAY GENMASK(31, 24) > +#define SDHCI_CDNS_SD6_PHY_DLL_SLAVE_CLK_WRDQS_DELAY GENMASK(23, 16) > +#define SDHCI_CDNS_SD6_PHY_DLL_SLAVE_CLK_WR_DELAY GENMASK(15, 8) > +#define SDHCI_CDNS_SD6_PHY_DLL_SLAVE_READ_DQS_DELAY GENMASK(7, 0) > + > +#define SDHCI_CDNS_SD6_PHY_CTRL 0x2080 > +#define SDHCI_CDNS_SD6_PHY_CTRL_PHONY_DQS_TIMING GENMASK(9, 4) > + > +#define SDHCI_CDNS_SD6_PHY_GPIO_CTRL0 0x2088 > +#define SDHCI_CDNS_SD6_PHY_GPIO_CTRL0_DRV GENMASK(6, 5) > +#define SDHCI_CDNS_SD6_PHY_GPIO_CTRL0_DRV_OVR_EN BIT(4) > +#define SDHCI_CDNS_SD6_PHY_GPIO_CTRL0_SLEW GENMASK(2, 1) > +#define SDHCI_CDNS_SD6_PHY_GPIO_CTRL0_SLEW_OVR_EN BIT(0) > + > /* SRS - Slot Register Set (SDHCI-compatible) */ > #define SDHCI_CDNS_SRS_BASE 0x200 > > @@ -60,6 +149,10 @@ > */ > #define SDHCI_CDNS_MAX_TUNING_LOOP 40 > > +static int tune_val_start = SDHCI_CDNS_TUNE_START; > +static int tune_val_step = SDHCI_CDNS_TUNE_STEP; These 2 need to be in one of the structs, not global. > +static int max_tune_iter = SDHCI_CDNS_TUNE_ITERATIONS; This never changes so might as well just use SDHCI_CDNS_TUNE_ITERATIONS instead. > + > struct sdhci_cdns_priv; > > struct sdhci_cdns_sd4_phy_param { > @@ -108,6 +201,558 @@ static const struct sdhci_cdns_sd4_phy_cfg sdhci_cdns_sd4_phy_cfgs[] = { > { "cdns,phy-dll-delay-strobe", SDHCI_CDNS_PHY_DLY_STROBE, }, > }; > > +enum sdhci_cdns_sd6_phy_lock_mode { > + SDHCI_CDNS_SD6_PHY_LOCK_MODE_FULL_CLK = 0, > + SDHCI_CDNS_SD6_PHY_LOCK_MODE_HALF_CLK = 2, > + SDHCI_CDNS_SD6_PHY_LOCK_MODE_SATURATION = 3, > +}; > + > +struct sdhci_cdns_sd6_phy_timings { > + u32 t_cmd_output_min; > + u32 t_cmd_output_max; > + u32 t_dat_output_min; > + u32 t_dat_output_max; > + u32 t_cmd_input_min; > + u32 t_cmd_input_max; > + u32 t_dat_input_min; > + u32 t_dat_input_max; > + u32 t_sdclk_min; > + u32 t_sdclk_max; > +}; > + > +struct sdhci_cdns_sd6_phy_delays { > + u32 phy_sdclk_delay; > + u32 phy_cmd_o_delay; > + u32 phy_dat_o_delay; > + u32 iocell_input_delay; > + u32 iocell_output_delay; > + u32 delay_element_org; > + u32 delay_element; > +}; > + > +struct sdhci_cdns_sd6_phy_settings { > + /* SDHCI_CDNS_SD6_PHY_DLL_SLAVE */ > + u32 cp_read_dqs_cmd_delay; > + u32 cp_read_dqs_delay; > + u32 cp_clk_wr_delay; > + u32 cp_clk_wrdqs_delay; > + > + /* SDHCI_CDNS_SD6_PHY_DLL_MASTER */ > + u32 cp_dll_bypass_mode; > + u32 cp_dll_start_point; > + > + /* SDHCI_CDNS_SD6_PHY_DLL_OBS_REG0 */ > + u32 cp_dll_locked_mode; > + > + /* SDHCI_CDNS_SD6_PHY_GATE_LPBK */ > + u32 cp_gate_cfg_always_on; > + u32 cp_sync_method; > + u32 cp_rd_del_sel; > + u32 cp_sw_half_cycle_shift; > + u32 cp_underrun_suppress; cp_underrun_suppress is not used. Please leave out anything that is not used. > + > + /* SDHCI_CDNS_SD6_PHY_DQ_TIMING */ > + u32 cp_io_mask_always_on; > + u32 cp_io_mask_end; > + u32 cp_io_mask_start; > + u32 cp_data_select_oe_end; > + > + /* SDHCI_CDNS_SD6_PHY_DQS_TIMING */ > + u32 cp_use_ext_lpbk_dqs; > + u32 cp_use_lpbk_dqs; > + u8 cp_use_phony_dqs; > + u8 cp_use_phony_dqs_cmd; > + > + /* HRS 09 */ > + u8 sdhc_extended_rd_mode; > + u8 sdhc_extended_wr_mode; > + u32 sdhc_rdcmd_en; > + u32 sdhc_rddata_en; > + > + /* HRS10 */ > + u32 sdhc_hcsdclkadj; > + > + /* HRS 07 */ > + u32 sdhc_idelay_val; > + u32 sdhc_rw_compensate; > + > + /* SRS 11 */ > + u32 sdhc_sdcfsh; > + u32 sdhc_sdcfsl; > + > + /* HRS 16 */ > + u32 sdhc_wrcmd0_dly; > + u32 sdhc_wrcmd0_sdclk_dly; > + u32 sdhc_wrcmd1_dly; > + u32 sdhc_wrcmd1_sdclk_dly; > + u32 sdhc_wrdata0_dly; > + u32 sdhc_wrdata0_sdclk_dly; > + u32 sdhc_wrdata1_dly; > + u32 sdhc_wrdata1_sdclk_dly; > + > + u32 hs200_tune_val; > + u32 drive; > + u32 slew; > +}; > + > +struct sdhci_cdns_sd6_phy_intermediate_results { > + u32 t_sdmclk_calc; > + u32 dll_max_value; > +}; > + > +struct sdhci_cdns_sd6_phy { > + struct sdhci_cdns_sd6_phy_timings t; > + struct sdhci_cdns_sd6_phy_delays d; Kernel style is not to have nested structures i.e. just put all the members of struct sdhci_cdns_sd6_phy_delays in struct sdhci_cdns_sd6_phy. > + u32 t_sdmclk; > + struct sdhci_cdns_sd6_phy_settings settings; Kernel style is not to have nested structures i.e. just put all the members of struct sdhci_cdns_sd6_phy_settings in struct sdhci_cdns_sd6_phy. > + struct sdhci_cdns_sd6_phy_intermediate_results vars; Kernel style is not to have nested structures i.e. just put all the members of struct sdhci_cdns_sd6_phy_intermediate_results in struct sdhci_cdns_sd6_phy. > + bool ddr; > + bool tune_cmd; > + bool tune_dat; > + bool strobe_cmd; > + bool strobe_dat; > + int mode; > + int t_sdclk; > +}; > + > +static void init_hs(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 2000, .t_cmd_output_max = t_sdclk - 6000, > + .t_dat_output_min = 2000, .t_dat_output_max = t_sdclk - 6000, > + .t_cmd_input_min = 14000, .t_cmd_input_max = t_sdclk + 2500, > + .t_dat_input_min = 14000, .t_dat_input_max = t_sdclk + 2500, > + .t_sdclk_min = 1000000 / 50, .t_sdclk_max = 1000000 / 0.4 > + }; > +} > + > +static void init_uhs_sdr12(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 800, .t_cmd_output_max = t_sdclk - 3000, > + .t_dat_output_min = 800, .t_dat_output_max = t_sdclk - 3000, > + .t_cmd_input_min = 14000, .t_cmd_input_max = t_sdclk + 1500, > + .t_dat_input_min = 14000, .t_dat_input_max = t_sdclk + 1500, > + .t_sdclk_min = 1000000 / 25, .t_sdclk_max = 1000000 / 0.4 > + }; > +} > + > +static void init_uhs_sdr25(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 800, .t_cmd_output_max = t_sdclk - 3000, > + .t_dat_output_min = 800, .t_dat_output_max = t_sdclk - 3000, > + .t_cmd_input_min = 14000, .t_cmd_input_max = t_sdclk + 1500, > + .t_dat_input_min = 14000, .t_dat_input_max = t_sdclk + 1500, > + .t_sdclk_min = 1000000 / 50, .t_sdclk_max = 1000000 / 0.4 > + }; > +} > + > +static void init_uhs_sdr50(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 800, .t_cmd_output_max = t_sdclk - 3000, > + .t_dat_output_min = 800, .t_dat_output_max = t_sdclk - 3000, > + .t_cmd_input_min = 7500, .t_cmd_input_max = t_sdclk + 1500, > + .t_dat_input_min = 7500, .t_dat_input_max = t_sdclk + 1500, > + .t_sdclk_min = 1000000 / 100, .t_sdclk_max = 1000000 / 0.4 > + }; > +} > + > +static void init_uhs_sdr104(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 800, .t_cmd_output_max = t_sdclk - 1400, > + .t_dat_output_min = 800, .t_dat_output_max = t_sdclk - 1400, > + .t_cmd_input_min = 1000, .t_cmd_input_max = t_sdclk + 1000, > + .t_dat_input_min = 1000, .t_dat_input_max = t_sdclk + 1000, > + .t_sdclk_min = 1000000 / 200, .t_sdclk_max = 1000000 / 100 > + }; > +} > + > +static void init_uhs_ddr50(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 800, .t_cmd_output_max = t_sdclk - 3000, > + .t_dat_output_min = 800, .t_dat_output_max = t_sdclk - 3000, > + .t_cmd_input_min = 13700, .t_cmd_input_max = t_sdclk + 1500, > + .t_dat_input_min = 7000, .t_dat_input_max = t_sdclk + 1500, > + .t_sdclk_min = 1000000 / 50, .t_sdclk_max = 1000000 / 0.4 > + }; > +} > + > +static void init_emmc_legacy(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 3000, .t_cmd_output_max = t_sdclk - 3000, > + .t_dat_output_min = 3000, .t_dat_output_max = t_sdclk - 3000, > + .t_cmd_input_min = 11700, .t_cmd_input_max = t_sdclk + 8300, > + .t_dat_input_min = 11700, .t_dat_input_max = t_sdclk + 8300, > + .t_sdclk_min = 1000000 / 25, .t_sdclk_max = 1000000 / 0.4 > + }; > +} > + > +static void init_emmc_sdr(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 3000, .t_cmd_output_max = t_sdclk - 3000, > + .t_dat_output_min = 3000, .t_dat_output_max = t_sdclk - 3000, > + .t_cmd_input_min = 13700, .t_cmd_input_max = t_sdclk + 2500, > + .t_dat_input_min = 13700, .t_dat_input_max = t_sdclk + 2500, > + .t_sdclk_min = 1000000 / 50, .t_sdclk_max = 1000000 / 0.4 > + }; > +} > + > +static void init_emmc_ddr(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 3000, .t_cmd_output_max = t_sdclk - 3000, > + .t_dat_output_min = 2500, .t_dat_output_max = t_sdclk - 2500, > + .t_cmd_input_min = 13700, .t_cmd_input_max = t_sdclk + 2500, > + .t_dat_input_min = 7000, .t_dat_input_max = t_sdclk + 1500, > + .t_sdclk_min = 1000000 / 50, .t_sdclk_max = 1000000 / 0.4 > + }; > +} > + > +static void init_emmc_hs200(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 800, .t_cmd_output_max = t_sdclk - 1400, > + .t_dat_output_min = 800, .t_dat_output_max = t_sdclk - 1400, > + .t_cmd_input_min = 1000, .t_cmd_input_max = t_sdclk + 1000, > + .t_dat_input_min = 1000, .t_dat_input_max = t_sdclk + 1000, > + .t_sdclk_min = 1000000 / 200, .t_sdclk_max = 1000000 / 100 > + }; > +} > + > +/* HS400 and HS400ES */ > +static void init_emmc_hs400(struct sdhci_cdns_sd6_phy_timings *t, int t_sdclk) > +{ > + *t = (struct sdhci_cdns_sd6_phy_timings){ > + .t_cmd_output_min = 800, .t_cmd_output_max = t_sdclk - 1400, > + .t_dat_output_min = 400, .t_dat_output_max = t_sdclk - 400, > + .t_cmd_input_min = 1000, .t_cmd_input_max = t_sdclk + 1000, > + .t_dat_input_min = 1000, .t_dat_input_max = t_sdclk + 1000, > + .t_sdclk_min = 1000000 / 200, .t_sdclk_max = 1000000 / 100 > + }; > +} > + > +static void (*init_timings[])(struct sdhci_cdns_sd6_phy_timings*, int) = { > + &init_hs, &init_emmc_legacy, &init_emmc_sdr, > + &init_emmc_ddr, &init_emmc_hs200, &init_emmc_hs400, > + &init_uhs_sdr12, &init_uhs_sdr25, &init_uhs_sdr50, > + &init_uhs_sdr104, &init_uhs_ddr50 > +}; Calling a function (sdhci_cdns_sd6_get_mode()) which uses a switch statement to get the index to this array of function pointers to call that function, begs the question: why not just call them from the switch statement in the first place? > + > +static u32 read_dqs_cmd_delay, clk_wrdqs_delay, clk_wr_delay, read_dqs_delay; These need to be in one of the structs, not global. > + > +static u32 sdhci_cdns_sd6_get_mode(struct sdhci_host *host, unsigned int timing); Avoid unnecessary forward declarations. Instead move the functions as needed. > + > +static int sdhci_cdns_sd6_phy_lock_dll(struct sdhci_cdns_sd6_phy *phy) > +{ > + u32 delay_element = phy->d.delay_element_org; > + u32 delay_elements_in_sdmclk; > + enum sdhci_cdns_sd6_phy_lock_mode mode; > + > + delay_elements_in_sdmclk = DIV_ROUND_UP(phy->t_sdmclk, delay_element); > + if (delay_elements_in_sdmclk > 256) { > + delay_element *= 2; > + delay_elements_in_sdmclk = DIV_ROUND_UP(phy->t_sdmclk, > + delay_element); > + > + if (delay_elements_in_sdmclk > 256) > + return -1; > + > + mode = SDHCI_CDNS_SD6_PHY_LOCK_MODE_HALF_CLK; > + phy->vars.dll_max_value = 127; > + } else { > + mode = SDHCI_CDNS_SD6_PHY_LOCK_MODE_FULL_CLK; > + phy->vars.dll_max_value = 255; > + } > + > + phy->vars.t_sdmclk_calc = delay_element * delay_elements_in_sdmclk; > + phy->d.delay_element = delay_element; > + phy->settings.cp_dll_locked_mode = mode; > + phy->settings.cp_dll_bypass_mode = 0; > + > + return 0; > +} > + > +static void sdhci_cdns_sd6_phy_dll_bypass(struct sdhci_cdns_sd6_phy *phy) > +{ > + phy->vars.dll_max_value = 256; > + phy->settings.cp_dll_bypass_mode = 1; > + phy->settings.cp_dll_locked_mode = > + SDHCI_CDNS_SD6_PHY_LOCK_MODE_SATURATION; > +} > + > +static void sdhci_cdns_sd6_phy_configure_dll(struct sdhci_cdns_sd6_phy *phy) > +{ > + if (phy->settings.sdhc_extended_wr_mode == 0) { > + if (sdhci_cdns_sd6_phy_lock_dll(phy) == 0) > + return; > + } > + sdhci_cdns_sd6_phy_dll_bypass(phy); > +} > + > +static void sdhci_cdns_sd6_phy_calc_out(struct sdhci_cdns_sd6_phy *phy, > + bool cmd_not_dat) > +{ > + u32 wr0_dly = 0, wr1_dly = 0, output_min, output_max, phy_o_delay, > + clk_wr_delay = 0, wr0_sdclk_dly = 0, wr1_sdclk_dly = 0; > + bool data_ddr = phy->ddr && !cmd_not_dat; > + int t; > + > + if (cmd_not_dat) { > + output_min = phy->t.t_cmd_output_min; > + output_max = phy->t.t_cmd_output_max; > + phy_o_delay = phy->d.phy_cmd_o_delay; > + } else { > + output_min = phy->t.t_dat_output_min; > + output_max = phy->t.t_dat_output_max; > + phy_o_delay = phy->d.phy_dat_o_delay; > + } > + > + if (data_ddr) { > + wr0_sdclk_dly = 1; > + wr1_sdclk_dly = 1; > + } > + > + t = phy_o_delay - phy->d.phy_sdclk_delay - output_min; > + if (t < 0 && phy->settings.sdhc_extended_wr_mode == 1) { > + u32 n_half_cycle = DIV_ROUND_UP(-t * 2, phy->t_sdmclk); > + > + wr0_dly = (n_half_cycle + 1) / 2; > + if (data_ddr) > + wr1_dly = (n_half_cycle + 1) / 2; > + else > + wr1_dly = (n_half_cycle + 1) % 2 + wr0_dly - 1; > + } > + > + if (phy->settings.sdhc_extended_wr_mode == 0) { > + u32 out_hold, out_setup, out_hold_margin; > + u32 n; > + > + if (!data_ddr) > + wr0_dly = 1; > + > + out_setup = output_max; > + out_hold = output_min; > + out_hold_margin = DIV_ROUND_UP(out_setup - out_hold, 4); > + out_hold += out_hold_margin; > + > + if (phy->settings.cp_dll_bypass_mode == 0) > + n = DIV_ROUND_UP(256 * out_hold, phy->vars.t_sdmclk_calc); > + else > + n = DIV_ROUND_UP(out_hold, phy->d.delay_element) - 1; > + > + if (n <= phy->vars.dll_max_value) > + clk_wr_delay = n; > + else > + clk_wr_delay = 255; > + } else { > + /* sdhc_extended_wr_mode = 1 - PHY IO cell work in SDR mode */ > + clk_wr_delay = 0; > + } > + > + if (cmd_not_dat) { > + phy->settings.sdhc_wrcmd0_dly = wr0_dly; > + phy->settings.sdhc_wrcmd1_dly = wr1_dly; > + phy->settings.cp_clk_wrdqs_delay = clk_wr_delay; > + phy->settings.sdhc_wrcmd0_sdclk_dly = wr0_sdclk_dly; > + phy->settings.sdhc_wrcmd1_sdclk_dly = wr1_sdclk_dly; > + } else { > + phy->settings.sdhc_wrdata0_dly = wr0_dly; > + phy->settings.sdhc_wrdata1_dly = wr1_dly; > + phy->settings.cp_clk_wr_delay = clk_wr_delay; > + phy->settings.sdhc_wrdata0_sdclk_dly = wr0_sdclk_dly; > + phy->settings.sdhc_wrdata1_sdclk_dly = wr1_sdclk_dly; > + } > +} > + > +static void sdhci_cdns_sd6_phy_calc_cmd_out(struct sdhci_cdns_sd6_phy *phy) > +{ > + sdhci_cdns_sd6_phy_calc_out(phy, true); > +} > + > +static void sdhci_cdns_sd6_phy_calc_cmd_in(struct sdhci_cdns_sd6_phy *phy) > +{ > + phy->settings.cp_io_mask_end = > + ((phy->d.iocell_output_delay + phy->d.iocell_input_delay) * 2) > + / phy->t_sdmclk; > + > + if (phy->settings.cp_io_mask_end >= 8) > + phy->settings.cp_io_mask_end = 7; > + > + if (phy->strobe_cmd && phy->settings.cp_io_mask_end > 0) > + phy->settings.cp_io_mask_end--; > + > + if (phy->strobe_cmd) { > + phy->settings.cp_use_phony_dqs_cmd = 0; > + phy->settings.cp_read_dqs_cmd_delay = 64; > + } else { > + phy->settings.cp_use_phony_dqs_cmd = 1; > + phy->settings.cp_read_dqs_cmd_delay = 0; > + } > + > + if ((phy->mode == MMC_TIMING_MMC_HS400 && !phy->strobe_cmd) || > + phy->mode == MMC_TIMING_MMC_HS200) > + phy->settings.cp_read_dqs_cmd_delay = > + phy->settings.hs200_tune_val; > +} > + > +static void sdhci_cdns_sd6_phy_calc_dat_in(struct sdhci_cdns_sd6_phy *phy) > +{ > + u32 hcsdclkadj = 0; > + > + if (phy->strobe_dat) { > + phy->settings.cp_use_phony_dqs = 0; > + phy->settings.cp_read_dqs_delay = 64; > + } else { > + phy->settings.cp_use_phony_dqs = 1; > + phy->settings.cp_read_dqs_delay = 0; > + } > + > + if (phy->mode == MMC_TIMING_MMC_HS200) > + phy->settings.cp_read_dqs_delay = > + phy->settings.hs200_tune_val; > + > + if (phy->strobe_dat) { > + /* dqs loopback input via IO cell */ > + hcsdclkadj += phy->d.iocell_input_delay; > + /* dfi_dqs_in: mem_dqs -> clean_dqs_mod; delay of hic_dll_dqs_nand2 */ > + hcsdclkadj += phy->d.delay_element / 2; > + /* delay line */ > + hcsdclkadj += phy->t_sdclk / 2; > + /* PHY FIFO write pointer */ > + hcsdclkadj += phy->t_sdclk / 2 + phy->d.delay_element; > + /* 1st synchronizer */ > + hcsdclkadj += DIV_ROUND_UP(hcsdclkadj, phy->t_sdmclk) > + * phy->t_sdmclk - hcsdclkadj; > + /* > + * 2nd synchronizer + PHY FIFO read pointer + PHY rddata > + * + PHY rddata registered, + FIFO 1st ciu_en > + */ > + hcsdclkadj += 5 * phy->t_sdmclk; > + /* FIFO 2st ciu_en */ > + hcsdclkadj += phy->t_sdclk; > + > + hcsdclkadj /= phy->t_sdclk; > + } else { > + u32 n; > + > + /* rebar PHY delay */ > + hcsdclkadj += 2 * phy->t_sdmclk; > + /* rebar output via IO cell */ > + hcsdclkadj += phy->d.iocell_output_delay; > + /* dqs loopback input via IO cell */ > + hcsdclkadj += phy->d.iocell_input_delay; > + /* dfi_dqs_in: mem_dqs -> clean_dqs_mod delay of hic_dll_dqs_nand2 */ > + hcsdclkadj += phy->d.delay_element / 2; > + /* dll: one delay element between SIGI_0 and SIGO_0 */ > + hcsdclkadj += phy->d.delay_element; > + /* dfi_dqs_in: mem_dqs_delayed -> clk_dqs delay of hic_dll_dqs_nand2 */ > + hcsdclkadj += phy->d.delay_element / 2; > + /* deskew DLL: clk_dqs -> clk_dqN: one delay element */ > + hcsdclkadj += phy->d.delay_element; > + > + if (phy->t_sdclk == phy->t_sdmclk) > + n = (hcsdclkadj - 2 * phy->t_sdmclk) / phy->t_sdclk; > + else > + n = hcsdclkadj / phy->t_sdclk; > + > + /* phase shift within one t_sdclk clock cycle caused by rebar - lbk dqs delay */ > + hcsdclkadj = hcsdclkadj % phy->t_sdclk; > + /* PHY FIFO write pointer */ > + hcsdclkadj += phy->t_sdclk / 2; > + /* 1st synchronizer */ > + hcsdclkadj += DIV_ROUND_UP(hcsdclkadj, phy->t_sdmclk) > + * phy->t_sdmclk - hcsdclkadj; > + /* > + * 2nd synchronizer + PHY FIFO read pointer + PHY rddata > + * + PHY rddata registered > + */ > + hcsdclkadj += 4 * phy->t_sdmclk; > + > + if ((phy->t_sdclk / phy->t_sdmclk) > 1) { > + u32 tmp1, tmp2; > + > + tmp1 = hcsdclkadj; > + tmp2 = (hcsdclkadj / phy->t_sdclk) * phy->t_sdclk > + + phy->t_sdclk - phy->t_sdmclk; > + if (tmp1 == tmp2) > + tmp2 += phy->t_sdclk; > + > + /* FIFO aligns to clock cycle before ciu_en */ > + hcsdclkadj += tmp2 - tmp1; > + } > + > + /* FIFO 1st ciu_en */ > + hcsdclkadj += phy->t_sdmclk; > + /* FIFO 2nd ciu_en */ > + hcsdclkadj += phy->t_sdclk; > + > + hcsdclkadj /= phy->t_sdclk; > + > + hcsdclkadj += n; > + > + if ((phy->t_sdclk / phy->t_sdmclk) >= 2) { > + if (phy->mode == MMC_TIMING_UHS_DDR50 || > + phy->mode == MMC_TIMING_MMC_DDR52) > + hcsdclkadj -= 2; > + else > + hcsdclkadj -= 1; > + } else if ((phy->t_sdclk / phy->t_sdmclk) == 1) { > + hcsdclkadj += 2; > + } > + > + if (phy->tune_dat) > + hcsdclkadj -= 1; > + } > + > + if (hcsdclkadj > 15) > + hcsdclkadj = 15; > + > + phy->settings.sdhc_hcsdclkadj = hcsdclkadj; > +} > + > +static void sdhci_cdns_sd6_phy_calc_dat_out(struct sdhci_cdns_sd6_phy *phy) > +{ > + sdhci_cdns_sd6_phy_calc_out(phy, false); > +} > + > +static void sdhci_cdns_sd6_phy_calc_io(struct sdhci_cdns_sd6_phy *phy) > +{ > + u32 rw_compensate; > + > + rw_compensate = (phy->d.iocell_input_delay + phy->d.iocell_output_delay) > + / phy->t_sdmclk + phy->settings.sdhc_wrdata0_dly + 5 + 3; > + > + phy->settings.sdhc_idelay_val = (2 * phy->d.iocell_input_delay) > + / phy->t_sdmclk; > + > + phy->settings.cp_io_mask_start = 0; > + if (phy->t_sdclk == phy->t_sdmclk && rw_compensate > 10) > + phy->settings.cp_io_mask_start = 2 * (rw_compensate - 10); > + > + if (phy->mode == MMC_TIMING_UHS_SDR104) > + phy->settings.cp_io_mask_start++; > + > + if (phy->t_sdclk == phy->t_sdmclk && phy->mode == MMC_TIMING_UHS_SDR50) > + phy->settings.cp_io_mask_start++; > + > + phy->settings.sdhc_rw_compensate = rw_compensate; > +} > + > +static void sdhci_cdns_sd6_phy_calc_settings(struct sdhci_cdns_sd6_phy *phy) > +{ > + sdhci_cdns_sd6_phy_calc_cmd_out(phy); > + sdhci_cdns_sd6_phy_calc_cmd_in(phy); > + sdhci_cdns_sd6_phy_calc_dat_out(phy); > + sdhci_cdns_sd6_phy_calc_dat_in(phy); > + sdhci_cdns_sd6_phy_calc_io(phy); > +} > + > static inline void cdns_writel(struct sdhci_cdns_priv *priv, u32 val, > void __iomem *reg) > { > @@ -188,7 +833,276 @@ static int sdhci_cdns_sd4_phy_init(struct sdhci_cdns_priv *priv) > if (ret) > return ret; > } A blank line was removed here and makes the diff look weird. > + return 0; > +} > > +static u32 sdhci_cdns_sd6_read_phy_reg(struct sdhci_cdns_priv *priv, > + u32 addr) > +{ > + writel(FIELD_PREP(SDHCI_CDNS_SD6_HRS04_ADDR, addr), > + priv->hrs_addr + SDHCI_CDNS_HRS04); > + return readl(priv->hrs_addr + SDHCI_CDNS_HRS05); > +} > + > +static void sdhci_cdns_sd6_write_phy_reg(struct sdhci_cdns_priv *priv, > + u32 addr, u32 data) > +{ > + writel(FIELD_PREP(SDHCI_CDNS_SD6_HRS04_ADDR, addr), > + priv->hrs_addr + SDHCI_CDNS_HRS04); > + writel(data, priv->hrs_addr + SDHCI_CDNS_HRS05); > +} > + > +static int sdhci_cdns_sd6_dll_reset(struct sdhci_cdns_priv *priv, bool reset) > +{ > + u32 reg; > + int ret = 0; > + > + reg = readl(priv->hrs_addr + SDHCI_CDNS_HRS09); > + if (reset) > + reg &= ~SDHCI_CDNS_HRS09_PHY_SW_RESET; > + else > + reg |= SDHCI_CDNS_HRS09_PHY_SW_RESET; > + > + writel(reg, priv->hrs_addr + SDHCI_CDNS_HRS09); > + > + if (!reset) > + ret = readl_poll_timeout(priv->hrs_addr + SDHCI_CDNS_HRS09, > + reg, > + (reg & > + SDHCI_CDNS_HRS09_PHY_INIT_COMPLETE), > + 0, 0); > + > + return ret; > +} > + > +static void sdhci_cdns_sd6_calc_phy(struct sdhci_cdns_sd6_phy *phy) > +{ > + if (phy->mode == MMC_TIMING_MMC_HS) { > + phy->settings.cp_clk_wr_delay = 0; > + phy->settings.cp_clk_wrdqs_delay = 0; > + phy->settings.cp_data_select_oe_end = 1; > + phy->settings.cp_dll_bypass_mode = 1; > + phy->settings.cp_dll_locked_mode = 3; > + phy->settings.cp_dll_start_point = 4; > + phy->settings.cp_gate_cfg_always_on = 1; > + phy->settings.cp_io_mask_always_on = 0; > + phy->settings.cp_io_mask_end = 0; > + phy->settings.cp_io_mask_start = 0; > + phy->settings.cp_rd_del_sel = 52; > + phy->settings.cp_read_dqs_cmd_delay = 0; > + phy->settings.cp_read_dqs_delay = 0; > + phy->settings.cp_sw_half_cycle_shift = 0; > + phy->settings.cp_sync_method = 1; > + phy->settings.cp_underrun_suppress = 1; > + phy->settings.cp_use_ext_lpbk_dqs = 1; > + phy->settings.cp_use_lpbk_dqs = 1; > + phy->settings.cp_use_phony_dqs = 1; > + phy->settings.cp_use_phony_dqs_cmd = 1; > + phy->settings.sdhc_extended_rd_mode = 1; > + phy->settings.sdhc_extended_wr_mode = 1; > + phy->settings.sdhc_hcsdclkadj = 2; > + phy->settings.sdhc_idelay_val = 0; > + phy->settings.sdhc_rdcmd_en = 1; > + phy->settings.sdhc_rddata_en = 1; > + phy->settings.sdhc_rw_compensate = 9; > + phy->settings.sdhc_sdcfsh = 0; > + phy->settings.sdhc_sdcfsl = 4; > + phy->settings.sdhc_wrcmd0_dly = 1; > + phy->settings.sdhc_wrcmd0_sdclk_dly = 0; > + phy->settings.sdhc_wrcmd1_dly = 0; > + phy->settings.sdhc_wrcmd1_sdclk_dly = 0; > + phy->settings.sdhc_wrdata0_dly = 1; > + phy->settings.sdhc_wrdata0_sdclk_dly = 0; > + phy->settings.sdhc_wrdata1_dly = 0; > + phy->settings.sdhc_wrdata1_sdclk_dly = 0; > + } > +} > + > +static > +int sdhci_cdns_sd6_get_delay_params(struct device *dev, > + struct sdhci_cdns_priv *priv) > +{ > + struct sdhci_cdns_sd6_phy *phy = priv->phy; > + int ret; > + > + of_property_read_u32(dev->of_node, "marvell,iocell-input-delay-ps", > + &phy->d.iocell_input_delay); > + of_property_read_u32(dev->of_node, "marvell,iocell-output-delay-ps", > + &phy->d.iocell_output_delay); > + of_property_read_u32(dev->of_node, "marvell,delay-element-ps", > + &phy->d.delay_element); > + ret = of_property_read_u32(dev->of_node, "marvell,read-dqs-cmd-delay-ps", > + &phy->settings.cp_read_dqs_cmd_delay); > + if (ret) > + phy->settings.cp_read_dqs_cmd_delay = DEFAULT_CMD_DELAY; > + > + ret = of_property_read_u32(dev->of_node, "marvell,tune-val-start-ps", > + &tune_val_start); > + if (ret) > + tune_val_start = SDHCI_CDNS_TUNE_START; > + > + ret = of_property_read_u32(dev->of_node, "marvell,tune-val-step-ps", > + &tune_val_step); > + if (ret) > + tune_val_step = SDHCI_CDNS_TUNE_STEP; > + > + read_dqs_cmd_delay = phy->settings.cp_read_dqs_cmd_delay; > + clk_wrdqs_delay = phy->settings.cp_clk_wrdqs_delay; > + clk_wr_delay = phy->settings.cp_clk_wr_delay; > + read_dqs_delay = phy->settings.cp_read_dqs_delay; > + return 0; > +} > + > +static int sdhci_cdns_sd6_phy_init(struct sdhci_cdns_priv *priv) > +{ > + int ret; > + u32 reg; > + struct sdhci_cdns_sd6_phy *phy = priv->phy; > + > + sdhci_cdns_sd6_dll_reset(priv, true); > + > + reg = sdhci_cdns_sd6_read_phy_reg(priv, SDHCI_CDNS_SD6_PHY_DQS_TIMING); > + reg &= ~SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_EXT_LPBK_DQS; > + reg &= ~SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_LPBK_DQS; > + reg &= ~SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_PHONY_DQS; > + reg &= ~SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_PHONY_DQS_CMD; > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_EXT_LPBK_DQS, > + phy->settings.cp_use_ext_lpbk_dqs); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_LPBK_DQS, > + phy->settings.cp_use_lpbk_dqs); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_PHONY_DQS, > + phy->settings.cp_use_phony_dqs); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DQS_TIMING_USE_PHONY_DQS_CMD, > + phy->settings.cp_use_phony_dqs_cmd); > + sdhci_cdns_sd6_write_phy_reg(priv, SDHCI_CDNS_SD6_PHY_DQS_TIMING, reg); > + > + reg = sdhci_cdns_sd6_read_phy_reg(priv, SDHCI_CDNS_SD6_PHY_GATE_LPBK); > + reg &= ~SDHCI_CDNS_SD6_PHY_GATE_LPBK_SYNC_METHOD; > + reg &= ~SDHCI_CDNS_SD6_PHY_GATE_LPBK_SW_HALF_CYCLE_SHIFT; > + reg &= ~SDHCI_CDNS_SD6_PHY_GATE_LPBK_RD_DEL_SEL; > + reg &= ~SDHCI_CDNS_SD6_PHY_GATE_LPBK_GATE_CFG_ALWAYS_ON; > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_GATE_LPBK_SYNC_METHOD, > + phy->settings.cp_sync_method); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_GATE_LPBK_SW_HALF_CYCLE_SHIFT, > + phy->settings.cp_sw_half_cycle_shift); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_GATE_LPBK_RD_DEL_SEL, > + phy->settings.cp_rd_del_sel); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_GATE_LPBK_GATE_CFG_ALWAYS_ON, > + phy->settings.cp_gate_cfg_always_on); > + sdhci_cdns_sd6_write_phy_reg(priv, SDHCI_CDNS_SD6_PHY_GATE_LPBK, reg); > + > + reg = 0x0; > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DLL_MASTER_BYPASS_MODE, > + phy->settings.cp_dll_bypass_mode); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DLL_MASTER_PHASE_DETECT_SEL, 2); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DLL_MASTER_DLL_LOCK_NUM, 0); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DLL_MASTER_DLL_START_POINT, > + phy->settings.cp_dll_start_point); > + sdhci_cdns_sd6_write_phy_reg(priv, SDHCI_CDNS_SD6_PHY_DLL_MASTER, reg); > + > + reg = 0x0; > + reg = FIELD_PREP(SDHCI_CDNS_SD6_PHY_DLL_SLAVE_READ_DQS_CMD_DELAY, > + phy->settings.cp_read_dqs_cmd_delay); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DLL_SLAVE_CLK_WRDQS_DELAY, > + phy->settings.cp_clk_wrdqs_delay); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DLL_SLAVE_CLK_WR_DELAY, > + phy->settings.cp_clk_wr_delay); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DLL_SLAVE_READ_DQS_DELAY, > + phy->settings.cp_read_dqs_delay); > + sdhci_cdns_sd6_write_phy_reg(priv, SDHCI_CDNS_SD6_PHY_DLL_SLAVE, reg); > + > + reg = sdhci_cdns_sd6_read_phy_reg(priv, SDHCI_CDNS_SD6_PHY_CTRL); > + reg &= ~SDHCI_CDNS_SD6_PHY_CTRL_PHONY_DQS_TIMING; > + sdhci_cdns_sd6_write_phy_reg(priv, SDHCI_CDNS_SD6_PHY_CTRL, reg); > + > + reg = sdhci_cdns_sd6_read_phy_reg(priv, SDHCI_CDNS_SD6_PHY_GPIO_CTRL0); > + reg &= ~0x77; > + reg |= SDHCI_CDNS_SD6_PHY_GPIO_CTRL0_DRV_OVR_EN | > + SDHCI_CDNS_SD6_PHY_GPIO_CTRL0_SLEW_OVR_EN; > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_GPIO_CTRL0_DRV, > + phy->settings.drive); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_GPIO_CTRL0_SLEW, > + phy->settings.slew); > + sdhci_cdns_sd6_write_phy_reg(priv, SDHCI_CDNS_SD6_PHY_GPIO_CTRL0, reg); > + > + ret = sdhci_cdns_sd6_dll_reset(priv, false); > + if (ret) > + return ret; > + > + reg = sdhci_cdns_sd6_read_phy_reg(priv, SDHCI_CDNS_SD6_PHY_DQ_TIMING); > + reg &= ~SDHCI_CDNS_SD6_PHY_DQ_TIMING_IO_MASK_ALWAYS_ON; > + reg &= ~SDHCI_CDNS_SD6_PHY_DQ_TIMING_IO_MASK_END; > + reg &= ~SDHCI_CDNS_SD6_PHY_DQ_TIMING_IO_MASK_START; > + reg &= ~SDHCI_CDNS_SD6_PHY_DQ_TIMING_DATA_SELECT_OE_END; > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DQ_TIMING_IO_MASK_ALWAYS_ON, > + phy->settings.cp_io_mask_always_on); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DQ_TIMING_IO_MASK_END, > + phy->settings.cp_io_mask_end); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DQ_TIMING_IO_MASK_START, > + phy->settings.cp_io_mask_start); > + reg |= FIELD_PREP(SDHCI_CDNS_SD6_PHY_DQ_TIMING_DATA_SELECT_OE_END, > + phy->settings.cp_data_select_oe_end); > + sdhci_cdns_sd6_write_phy_reg(priv, SDHCI_CDNS_SD6_PHY_DQ_TIMING, reg); > + > + reg = readl(priv->hrs_addr + SDHCI_CDNS_HRS09); > + if (phy->settings.sdhc_extended_wr_mode) > + reg |= SDHCI_CDNS_HRS09_EXTENDED_WR_MODE; > + else > + reg &= ~SDHCI_CDNS_HRS09_EXTENDED_WR_MODE; > + > + if (phy->settings.sdhc_extended_rd_mode) > + reg |= SDHCI_CDNS_HRS09_EXTENDED_RD_MODE; > + else > + reg &= ~SDHCI_CDNS_HRS09_EXTENDED_RD_MODE; > + > + if (phy->settings.sdhc_rddata_en) > + reg |= SDHCI_CDNS_HRS09_RDDATA_EN; > + else > + reg &= ~SDHCI_CDNS_HRS09_RDDATA_EN; > + > + if (phy->settings.sdhc_rdcmd_en) > + reg |= SDHCI_CDNS_HRS09_RDCMD_EN; > + else > + reg &= ~SDHCI_CDNS_HRS09_RDCMD_EN; > + > + writel(reg, priv->hrs_addr + SDHCI_CDNS_HRS09); > + > + writel(0x30004, priv->hrs_addr + SDHCI_CDNS_HRS02); > + > + reg = 0x0; > + reg = FIELD_PREP(SDHCI_CDNS_HRS10_HCSDCLKADJ, phy->settings.sdhc_hcsdclkadj); > + writel(reg, priv->hrs_addr + SDHCI_CDNS_HRS10); > + > + if (phy->mode != MMC_TIMING_MMC_HS && phy->mode != MMC_TIMING_MMC_DDR52) { > + reg = 0x0; > + reg = FIELD_PREP(SDHCI_CDNS_HRS16_WRDATA1_SDCLK_DLY, > + phy->settings.sdhc_wrdata1_sdclk_dly); > + reg |= FIELD_PREP(SDHCI_CDNS_HRS16_WRDATA0_SDCLK_DLY, > + phy->settings.sdhc_wrdata0_sdclk_dly); > + reg |= FIELD_PREP(SDHCI_CDNS_HRS16_WRCMD1_SDCLK_DLY, > + phy->settings.sdhc_wrcmd1_sdclk_dly); > + reg |= FIELD_PREP(SDHCI_CDNS_HRS16_WRCMD0_SDCLK_DLY, > + phy->settings.sdhc_wrcmd0_sdclk_dly); > + reg |= FIELD_PREP(SDHCI_CDNS_HRS16_WRDATA1_DLY, > + phy->settings.sdhc_wrdata1_dly); > + reg |= FIELD_PREP(SDHCI_CDNS_HRS16_WRDATA0_DLY, > + phy->settings.sdhc_wrdata0_dly); > + reg |= FIELD_PREP(SDHCI_CDNS_HRS16_WRCMD1_DLY, > + phy->settings.sdhc_wrcmd1_dly); > + reg |= FIELD_PREP(SDHCI_CDNS_HRS16_WRCMD0_DLY, > + phy->settings.sdhc_wrcmd0_dly); > + } else { > + reg = 0x202; > + } > + > + writel(reg, priv->hrs_addr + SDHCI_CDNS_HRS16); > + > + reg = 0x0; > + reg = FIELD_PREP(SDHCI_CDNS_HRS07_RW_COMPENSATE, > + phy->settings.sdhc_rw_compensate); > + reg |= FIELD_PREP(SDHCI_CDNS_HRS07_IDELAY_VAL, > + phy->settings.sdhc_idelay_val); > + writel(reg, priv->hrs_addr + SDHCI_CDNS_HRS07); > return 0; > } > > @@ -199,6 +1113,19 @@ static void *sdhci_cdns_priv(struct sdhci_host *host) > return sdhci_pltfm_priv(pltfm_host); > } > > +static int sdhci_cdns_sd6_set_tune_val(struct sdhci_host *host, > + unsigned int val) > +{ > + struct sdhci_cdns_priv *priv = sdhci_cdns_priv(host); > + struct sdhci_cdns_sd6_phy *phy = priv->phy; > + > + phy->settings.hs200_tune_val = val; > + phy->settings.cp_read_dqs_cmd_delay = val; > + phy->settings.cp_read_dqs_delay = val; > + > + return sdhci_cdns_sd6_phy_init(priv); > +} > + > static unsigned int sdhci_cdns_get_timeout_clock(struct sdhci_host *host) > { > /* > @@ -208,6 +1135,11 @@ static unsigned int sdhci_cdns_get_timeout_clock(struct sdhci_host *host) > return host->max_clk; > } > > +static unsigned int sdhci_cdns_get_max_clock(struct sdhci_host *host) > +{ > + return SDMCLK_MAX_FREQ; > +} > + > static void sdhci_cdns_set_emmc_mode(struct sdhci_cdns_priv *priv, u32 mode) > { > u32 tmp; > @@ -227,6 +1159,118 @@ static u32 sdhci_cdns_get_emmc_mode(struct sdhci_cdns_priv *priv) > return FIELD_GET(SDHCI_CDNS_HRS06_MODE, tmp); > } > > +static int sdhci_cdns_sd6_phy_update_timings(struct sdhci_host *host) > +{ > + struct sdhci_cdns_priv *priv = sdhci_cdns_priv(host); > + struct sdhci_cdns_sd6_phy *phy = priv->phy; > + int t_sdmclk = phy->t_sdmclk; > + int mode; > + > + mode = sdhci_cdns_sd6_get_mode(host, host->mmc->ios.timing); > + /* initialize input */ > + init_timings[mode](&phy->t, phy->t_sdclk); > + > + phy->mode = host->mmc->ios.timing; > + phy->strobe_dat = false; > + > + switch (phy->mode) { > + case MMC_TIMING_UHS_SDR104: > + phy->tune_cmd = true; > + phy->tune_dat = true; > + break; > + case MMC_TIMING_UHS_DDR50: > + phy->ddr = true; > + break; > + case MMC_TIMING_MMC_DDR52: > + phy->ddr = true; > + break; > + case MMC_TIMING_MMC_HS200: > + phy->tune_dat = true; > + phy->tune_cmd = true; > + break; > + case MMC_TIMING_MMC_HS400: > + phy->tune_cmd = true; > + phy->ddr = true; > + phy->strobe_dat = true; > + break; > + } > + > + if (priv->enhanced_strobe) > + phy->strobe_cmd = true; > + > + phy->d.phy_sdclk_delay = 2 * t_sdmclk; > + phy->d.phy_cmd_o_delay = 2 * t_sdmclk + t_sdmclk / 2; > + phy->d.phy_dat_o_delay = 2 * t_sdmclk + t_sdmclk / 2; > + > + if (phy->t_sdclk == phy->t_sdmclk) { > + phy->settings.sdhc_extended_wr_mode = 0; > + phy->settings.sdhc_extended_rd_mode = 0; > + } else { > + phy->settings.sdhc_extended_wr_mode = 1; > + phy->settings.sdhc_extended_rd_mode = 1; > + } > + > + phy->settings.cp_gate_cfg_always_on = 1; > + > + sdhci_cdns_sd6_phy_configure_dll(phy); > + > + sdhci_cdns_sd6_phy_calc_settings(phy); > + > + return 0; > +} > + > +static u32 sdhci_cdns_sd6_get_mode(struct sdhci_host *host, > + unsigned int timing) > +{ > + struct sdhci_cdns_priv *priv = sdhci_cdns_priv(host); > + u32 mode; > + > + switch (timing) { > + case MMC_TIMING_MMC_HS: > + mode = SDHCI_CDNS_HRS06_MODE_MMC_SDR; > + break; > + case MMC_TIMING_MMC_DDR52: > + mode = SDHCI_CDNS_HRS06_MODE_MMC_DDR; > + break; > + case MMC_TIMING_MMC_HS200: > + mode = SDHCI_CDNS_HRS06_MODE_MMC_HS200; > + break; > + case MMC_TIMING_MMC_HS400: > + if (priv->enhanced_strobe) > + mode = SDHCI_CDNS_HRS06_MODE_MMC_HS400ES; > + else > + mode = SDHCI_CDNS_HRS06_MODE_MMC_HS400; > + break; > + case MMC_TIMING_SD_HS: > + mode = SDHCI_CDNS_HRS06_MODE_SD; > + break; > + default: > + mode = SDHCI_CDNS_HRS06_MODE_MMC_SDR; > + break; > + } > + > + return mode; > +} > + > +static void sdhci_cdns_sd6_set_clock(struct sdhci_host *host, > + unsigned int clock) > +{ > + struct sdhci_cdns_priv *priv = sdhci_cdns_priv(host); > + struct sdhci_cdns_sd6_phy *phy = priv->phy; > + > + phy->t_sdclk = DIV_ROUND_DOWN_ULL(1e12, clock); > + > + pr_debug("%s %d %d\n", __func__, phy->mode, clock); > + > + if (sdhci_cdns_sd6_phy_update_timings(host)) > + pr_debug("%s: update timings failed\n", __func__); > + > + if (sdhci_cdns_sd6_phy_init(priv)) > + pr_debug("%s: phy init failed\n", __func__); > + > + sdhci_set_clock(host, clock); > +} > + > static int sdhci_cdns_sd4_phy_probe(struct platform_device *pdev, > struct sdhci_cdns_priv *priv) > { > @@ -248,6 +1292,106 @@ static int sdhci_cdns_sd4_phy_probe(struct platform_device *pdev, > return 0; > } > > +static int sdhci_cdns_sd6_phy_probe(struct platform_device *pdev, > + struct sdhci_cdns_priv *priv) > +{ > + struct device *dev = &pdev->dev; > + struct sdhci_cdns_sd6_phy *phy; > + u32 val; > + struct clk *clk; > + int ret; > + const char *mode_name; > + > + phy = devm_kzalloc(dev, sizeof(*phy), GFP_KERNEL); > + if (!phy) > + return -ENOMEM; > + > + clk = devm_clk_get(dev, "sdmclk"); > + if (IS_ERR(clk)) { > + dev_err(dev, "sdmclk get error\n"); > + return PTR_ERR(clk); > + } > + > + val = clk_get_rate(clk); > + phy->t_sdmclk = DIV_ROUND_DOWN_ULL(1e12, val); > + > + ret = of_property_read_u32(dev->of_node, "cdns,host_slew", > + &phy->settings.slew); > + if (ret) > + phy->settings.slew = 3; > + > + ret = of_property_read_u32(dev->of_node, "cdns,host_drive", > + &phy->settings.drive); > + if (ret) > + phy->settings.drive = 2; > + > + ret = of_property_read_u32(dev->of_node, "cdns,iocell-input-delay", > + &phy->d.iocell_input_delay); > + if (ret) > + phy->d.iocell_input_delay = 2500; > + > + ret = of_property_read_u32(dev->of_node, "cdns,iocell-output-delay", > + &phy->d.iocell_output_delay); > + if (ret) > + phy->d.iocell_output_delay = 2500; > + > + ret = of_property_read_u32(dev->of_node, "cdns,delay-element", > + &phy->d.delay_element); > + if (ret) > + phy->d.delay_element = 24; > + > + ret = of_property_read_string_index(dev->of_node, "cdns,mode", 0, > + &mode_name); > + if (!ret) { > + if (!strcmp("emmc_sdr", mode_name)) > + phy->mode = MMC_TIMING_MMC_HS; > + else if (!strcmp("emmc_ddr", mode_name)) > + phy->mode = MMC_TIMING_MMC_DDR52; > + else if (!strcmp("emmc_hs200", mode_name)) > + phy->mode = MMC_TIMING_MMC_HS200; > + else if (!strcmp("emmc_hs400", mode_name)) > + phy->mode = MMC_TIMING_MMC_HS400; > + else if (!strcmp("sd_hs", mode_name)) > + phy->mode = MMC_TIMING_SD_HS; > + else > + phy->mode = MMC_TIMING_MMC_HS; > + } else { > + phy->mode = MMC_TIMING_MMC_HS; > + } > + > + phy->d.delay_element_org = phy->d.delay_element; > + phy->d.iocell_input_delay = 650; > + phy->d.iocell_output_delay = 1800; > + > + switch (phy->mode) { > + case MMC_TIMING_MMC_HS: > + phy->t_sdclk = 10000; > + break; > + case MMC_TIMING_MMC_DDR52: > + phy->t_sdclk = 10000; > + break; > + case MMC_TIMING_MMC_HS200: > + phy->t_sdclk = 5000; > + break; > + case MMC_TIMING_MMC_HS400: > + phy->t_sdclk = 5000; > + break; > + case MMC_TIMING_SD_HS: > + phy->t_sdclk = 100000; > + break; > + default: > + phy->t_sdclk = 10000; > + break; > + } > + > + priv->phy = phy; > + > + sdhci_cdns_sd6_get_delay_params(dev, priv); > + > + sdhci_cdns_sd6_calc_phy(phy); > + return 0; > +} > + > static int sdhci_cdns_sd4_set_tune_val(struct sdhci_host *host, unsigned int val) > { > struct sdhci_cdns_priv *priv = sdhci_cdns_priv(host); > @@ -281,6 +1425,57 @@ static int sdhci_cdns_sd4_set_tune_val(struct sdhci_host *host, unsigned int val > return 0; > } > > +/* > + * In SD mode, software must not use the hardware tuning and instead perform > + * an almost identical procedure to eMMC. > + */ > +static int sdhci_cdns_sd6_execute_tuning(struct sdhci_host *host, u32 opcode) > +{ > + int cur_streak = 0; > + int max_streak = 0; > + int end_of_streak = 0; > + int i, midpoint, iter = 0; > + > + /* > + * Do not execute tuning for UHS_SDR50 or UHS_DDR50. > + * The delay is set by probe, based on the DT properties. > + */ > + if (host->timing != MMC_TIMING_MMC_HS200 && > + host->timing != MMC_TIMING_UHS_SDR104) > + return 0; > + > + for (i = tune_val_start; iter < max_tune_iter; iter++, i += tune_val_step) { > + if (sdhci_cdns_sd6_set_tune_val(host, i) || > + mmc_send_tuning(host->mmc, opcode, NULL)) { /* bad */ > + cur_streak = 0; > + } else { /* good */ > + cur_streak++; > + if (cur_streak > max_streak) { > + max_streak = cur_streak; > + end_of_streak = i; > + pr_debug("%s (%d-%d = %d)", __func__, > + end_of_streak - ((cur_streak - 1) * tune_val_step), > + end_of_streak, cur_streak); > + } else { > + pr_debug("%s (%d-%d)", __func__, > + i - ((cur_streak - 1) * tune_val_step), i); > + } > + } > + } > + > + if (!max_streak) { > + dev_err(mmc_dev(host->mmc), "no tuning point found\n"); > + return -EIO; > + } > + > + pr_debug("max_streak: %d-%d", end_of_streak - ((max_streak - 1) * tune_val_step), > + end_of_streak); > + > + midpoint = end_of_streak - (((max_streak - 1) * tune_val_step) / 2); > + > + return sdhci_cdns_sd6_set_tune_val(host, midpoint); > +} > + > /* > * In SD mode, software must not use the hardware tuning and instead perform > * an almost identical procedure to eMMC. > @@ -343,9 +1538,12 @@ static void sdhci_cdns_set_uhs_signaling(struct sdhci_host *host, > else > mode = SDHCI_CDNS_HRS06_MODE_MMC_HS400; > break; > - default: > + case MMC_TIMING_SD_HS: > mode = SDHCI_CDNS_HRS06_MODE_SD; > break; > + default: > + mode = SDHCI_CDNS_HRS06_MODE_LEGACY; > + break; This change affects all devices not just marvell,cdns-sd6hc If it is needed, it should be a separate patch. > } > > sdhci_cdns_set_emmc_mode(priv, mode); > @@ -440,6 +1638,24 @@ static int elba_drv_init(struct platform_device *pdev) > return 0; > } > > +static void sdhci_cdns_sd6_set_uhs_signaling(struct sdhci_host *host, > + unsigned int timing) > +{ > + struct sdhci_cdns_priv *priv = sdhci_cdns_priv(host); > + struct sdhci_cdns_sd6_phy *phy = priv->phy; > + > + sdhci_cdns_set_uhs_signaling(host, timing); > + > + if ((phy->mode == -1) || (phy->t_sdclk == -1)) > + return; > + > + if (sdhci_cdns_sd6_phy_update_timings(host)) > + pr_debug("%s: update timings failed\n", __func__); > + > + if (sdhci_cdns_sd6_phy_init(priv)) > + pr_debug("%s: phy init failed\n", __func__); > +} > + > static const struct sdhci_ops sdhci_cdns_sd4_ops = { > .set_clock = sdhci_set_clock, > .get_timeout_clock = sdhci_cdns_get_timeout_clock, > @@ -449,6 +1665,16 @@ static const struct sdhci_ops sdhci_cdns_sd4_ops = { > .set_uhs_signaling = sdhci_cdns_set_uhs_signaling, > }; > > +static const struct sdhci_ops sdhci_cdns_sd6_ops = { > + .get_max_clock = sdhci_cdns_get_max_clock, > + .set_clock = sdhci_cdns_sd6_set_clock, > + .get_timeout_clock = sdhci_cdns_get_timeout_clock, > + .set_bus_width = sdhci_set_bus_width, > + .reset = sdhci_reset, > + .platform_execute_tuning = sdhci_cdns_sd6_execute_tuning, > + .set_uhs_signaling = sdhci_cdns_sd6_set_uhs_signaling, > +}; > + > static const struct sdhci_cdns_drv_data sdhci_cdns_uniphier_drv_data = { > .phy_init = sdhci_cdns_sd4_phy_init, > .phy_probe = sdhci_cdns_sd4_phy_probe, > @@ -475,6 +1701,14 @@ static const struct sdhci_cdns_drv_data sdhci_cdns_sd4_drv_data = { > }, > }; > > +static const struct sdhci_cdns_drv_data sdhci_cdns_sd6_drv_data = { > + .phy_init = sdhci_cdns_sd6_phy_init, > + .phy_probe = sdhci_cdns_sd6_phy_probe, > + .pltfm_data = { > + .ops = &sdhci_cdns_sd6_ops, > + }, > +}; > + > static void sdhci_cdns_hs400_enhanced_strobe(struct mmc_host *mmc, > struct mmc_ios *ios) > { > @@ -518,8 +1752,11 @@ static int sdhci_cdns_probe(struct platform_device *pdev) > struct sdhci_pltfm_host *pltfm_host; > struct sdhci_cdns_priv *priv; > struct clk *clk; > + bool sd6_ctrl; > int ret; > struct device *dev = &pdev->dev; > + sd6_ctrl = of_device_is_compatible(dev->of_node, "marvell,cdns-sd6hc"); > + > static const u16 version = SDHCI_SPEC_400 << SDHCI_SPEC_VER_SHIFT; > > clk = devm_clk_get(dev, NULL); > @@ -545,6 +1782,12 @@ static int sdhci_cdns_probe(struct platform_device *pdev) > pltfm_host = sdhci_priv(host); > pltfm_host->clk = clk; > > + if (sd6_ctrl) { > + host->clk_mul = 0; host->clk_mul is going to get overwritten, so this doesn't do anything, but the value comes from caps. > + host->max_clk = SDMCLK_MAX_FREQ; host->max_clk is going to get overwritten, so this doesn't do anything, but the value comes from caps. > + host->quirks |= SDHCI_QUIRK_CAP_CLOCK_BASE_BROKEN; > + host->quirks2 |= SDHCI_QUIRK2_PRESET_VALUE_BROKEN; Implement data->init() and put these there. > + } > priv = sdhci_pltfm_priv(pltfm_host); > priv->hrs_addr = host->ioaddr; > priv->enhanced_strobe = false; > @@ -559,7 +1802,10 @@ static int sdhci_cdns_probe(struct platform_device *pdev) > goto free; > } > sdhci_enable_v4_mode(host); > - __sdhci_read_caps(host, &version, NULL, NULL); > + if (sd6_ctrl) > + __sdhci_read_caps(host, NULL, NULL, NULL); __sdhci_read_caps() is only being used here to set the version, otherwise probably it is not needed. > + else > + __sdhci_read_caps(host, &version, NULL, NULL); Should be dealt with by data->init() > > sdhci_get_of_property(pdev); > > @@ -645,6 +1891,10 @@ static const struct of_device_id sdhci_cdns_match[] = { > .compatible = "cdns,sd4hc", > .data = &sdhci_cdns_sd4_drv_data, > }, > + { > + .compatible = "marvell,cdns-sd6hc", > + .data = &sdhci_cdns_sd6_drv_data, > + }, > { /* sentinel */ } > }; > MODULE_DEVICE_TABLE(of, sdhci_cdns_match);