2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2009-12-17 15:27:20 -08:00
/*
* Freescale eSDHC controller driver .
*
2012-02-14 14:05:37 +08:00
* Copyright ( c ) 2007 , 2010 , 2012 Freescale Semiconductor , Inc .
2009-12-17 15:27:20 -08:00
* Copyright ( c ) 2009 MontaVista Software , Inc .
2020-10-20 16:11:16 +08:00
* Copyright 2020 NXP
2009-12-17 15:27:20 -08:00
*
* Authors : Xiaobo Xie < X . Xie @ freescale . com >
* Anton Vorontsov < avorontsov @ ru . mvista . com >
*/
2013-06-27 12:00:05 -04:00
# include <linux/err.h>
2009-12-17 15:27:20 -08:00
# include <linux/io.h>
2012-02-14 14:05:37 +08:00
# include <linux/of.h>
2017-04-20 16:14:41 +08:00
# include <linux/of_address.h>
2009-12-17 15:27:20 -08:00
# include <linux/delay.h>
2011-07-03 15:15:51 -04:00
# include <linux/module.h>
2016-11-09 11:14:12 +08:00
# include <linux/sys_soc.h>
2017-04-20 16:14:40 +08:00
# include <linux/clk.h>
# include <linux/ktime.h>
2018-07-04 14:34:20 +03:00
# include <linux/dma-mapping.h>
2020-10-20 16:11:16 +08:00
# include <linux/iopoll.h>
2009-12-17 15:27:20 -08:00
# include <linux/mmc/host.h>
2019-03-11 02:16:47 +00:00
# include <linux/mmc/mmc.h>
2011-05-27 23:48:14 +08:00
# include "sdhci-pltfm.h"
2010-10-15 12:21:03 +02:00
# include "sdhci-esdhc.h"
2009-12-17 15:27:20 -08:00
2012-03-08 11:25:02 +08:00
# define VENDOR_V_22 0x12
2012-12-04 10:41:28 +08:00
# define VENDOR_V_23 0x13
2015-10-08 18:36:36 +08:00
2018-06-25 16:46:24 +08:00
# define MMC_TIMING_NUM (MMC_TIMING_MMC_HS400 + 1)
struct esdhc_clk_fixup {
const unsigned int sd_dflt_max_clk ;
const unsigned int max_clk [ MMC_TIMING_NUM ] ;
} ;
static const struct esdhc_clk_fixup ls1021a_esdhc_clk = {
. sd_dflt_max_clk = 25000000 ,
. max_clk [ MMC_TIMING_MMC_HS ] = 46500000 ,
. max_clk [ MMC_TIMING_SD_HS ] = 46500000 ,
} ;
static const struct esdhc_clk_fixup ls1046a_esdhc_clk = {
. sd_dflt_max_clk = 25000000 ,
. max_clk [ MMC_TIMING_UHS_SDR104 ] = 167000000 ,
. max_clk [ MMC_TIMING_MMC_HS200 ] = 167000000 ,
} ;
static const struct esdhc_clk_fixup ls1012a_esdhc_clk = {
. sd_dflt_max_clk = 25000000 ,
. max_clk [ MMC_TIMING_UHS_SDR104 ] = 125000000 ,
. max_clk [ MMC_TIMING_MMC_HS200 ] = 125000000 ,
} ;
static const struct esdhc_clk_fixup p1010_esdhc_clk = {
. sd_dflt_max_clk = 20000000 ,
. max_clk [ MMC_TIMING_LEGACY ] = 20000000 ,
. max_clk [ MMC_TIMING_MMC_HS ] = 42000000 ,
. max_clk [ MMC_TIMING_SD_HS ] = 40000000 ,
} ;
static const struct of_device_id sdhci_esdhc_of_match [ ] = {
{ . compatible = " fsl,ls1021a-esdhc " , . data = & ls1021a_esdhc_clk } ,
{ . compatible = " fsl,ls1046a-esdhc " , . data = & ls1046a_esdhc_clk } ,
{ . compatible = " fsl,ls1012a-esdhc " , . data = & ls1012a_esdhc_clk } ,
{ . compatible = " fsl,p1010-esdhc " , . data = & p1010_esdhc_clk } ,
{ . compatible = " fsl,mpc8379-esdhc " } ,
{ . compatible = " fsl,mpc8536-esdhc " } ,
{ . compatible = " fsl,esdhc " } ,
{ }
} ;
MODULE_DEVICE_TABLE ( of , sdhci_esdhc_of_match ) ;
2015-10-08 18:36:36 +08:00
struct sdhci_esdhc {
u8 vendor_ver ;
u8 spec_ver ;
2016-11-09 11:14:12 +08:00
bool quirk_incorrect_hostver ;
2018-11-23 11:15:35 +08:00
bool quirk_limited_clk_division ;
2018-11-23 11:15:37 +08:00
bool quirk_unreliable_pulse_detection ;
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
bool quirk_tuning_erratum_type1 ;
bool quirk_tuning_erratum_type2 ;
2019-03-11 02:16:51 +00:00
bool quirk_ignore_data_inhibit ;
2019-12-19 11:23:35 +08:00
bool quirk_delay_before_data_reset ;
2020-09-03 13:20:29 +12:00
bool quirk_trans_complete_erratum ;
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
bool in_sw_tuning ;
2017-04-20 16:14:40 +08:00
unsigned int peripheral_clock ;
2018-06-25 16:46:24 +08:00
const struct esdhc_clk_fixup * clk_fixup ;
2018-08-23 16:48:32 +08:00
u32 div_ratio ;
2015-10-08 18:36:36 +08:00
} ;
/**
* esdhc_read * _fixup - Fixup the value read from incompatible eSDHC register
* to make it compatible with SD spec .
*
* @ host : pointer to sdhci_host
* @ spec_reg : SD spec register address
* @ value : 32 bit eSDHC register value on spec_reg address
*
* In SD spec , there are 8 / 16 / 32 / 64 bits registers , while all of eSDHC
* registers are 32 bits . There are differences in register size , register
* address , register function , bit position and function between eSDHC spec
* and SD spec .
*
* Return a fixed up register value
*/
static u32 esdhc_readl_fixup ( struct sdhci_host * host ,
int spec_reg , u32 value )
2012-03-08 11:25:02 +08:00
{
2015-10-08 18:36:36 +08:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
2016-02-16 21:08:26 +08:00
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2012-03-08 11:25:02 +08:00
u32 ret ;
/*
* The bit of ADMA flag in eSDHC is not compatible with standard
* SDHC register , so set fake flag SDHCI_CAN_DO_ADMA2 when ADMA is
* supported by eSDHC .
* And for many FSL eSDHC controller , the reset value of field
2015-10-08 18:36:36 +08:00
* SDHCI_CAN_DO_ADMA1 is 1 , but some of them can ' t support ADMA ,
2012-03-08 11:25:02 +08:00
* only these vendor version is greater than 2.2 / 0x12 support ADMA .
*/
2015-10-08 18:36:36 +08:00
if ( ( spec_reg = = SDHCI_CAPABILITIES ) & & ( value & SDHCI_CAN_DO_ADMA1 ) ) {
if ( esdhc - > vendor_ver > VENDOR_V_22 ) {
ret = value | SDHCI_CAN_DO_ADMA2 ;
return ret ;
}
2012-03-08 11:25:02 +08:00
}
2016-11-15 11:13:16 +01:00
/*
* The DAT [ 3 : 0 ] line signal levels and the CMD line signal level are
* not compatible with standard SDHC register . The line signal levels
* DAT [ 7 : 0 ] are at bits 31 : 24 and the command line signal level is at
* bit 23. All other bits are the same as in the standard SDHC
* register .
*/
if ( spec_reg = = SDHCI_PRESENT_STATE ) {
ret = value & 0x000fffff ;
ret | = ( value > > 4 ) & SDHCI_DATA_LVL_MASK ;
ret | = ( value < < 1 ) & SDHCI_CMD_LVL ;
return ret ;
}
2017-08-15 10:17:03 +08:00
/*
* DTS properties of mmc host are used to enable each speed mode
* according to soc and board capability . So clean up
* SDR50 / SDR104 / DDR50 support bits here .
*/
if ( spec_reg = = SDHCI_CAPABILITIES_1 ) {
ret = value & ~ ( SDHCI_SUPPORT_SDR50 | SDHCI_SUPPORT_SDR104 |
SDHCI_SUPPORT_DDR50 ) ;
return ret ;
}
2019-03-11 02:16:51 +00:00
/*
* Some controllers have unreliable Data Line Active
* bit for commands with busy signal . This affects
* Command Inhibit ( data ) bit . Just ignore it since
* MMC core driver has already polled card status
* with CMD13 after any command with busy siganl .
*/
if ( ( spec_reg = = SDHCI_PRESENT_STATE ) & &
( esdhc - > quirk_ignore_data_inhibit = = true ) ) {
ret = value & ~ SDHCI_DATA_INHIBIT ;
return ret ;
}
2015-10-08 18:36:36 +08:00
ret = value ;
2012-03-08 11:25:02 +08:00
return ret ;
}
2015-10-08 18:36:36 +08:00
static u16 esdhc_readw_fixup ( struct sdhci_host * host ,
int spec_reg , u32 value )
2009-12-17 15:27:20 -08:00
{
2016-11-09 11:14:12 +08:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2009-12-17 15:27:20 -08:00
u16 ret ;
2015-10-08 18:36:36 +08:00
int shift = ( spec_reg & 0x2 ) * 8 ;
2009-12-17 15:27:20 -08:00
2020-01-17 14:38:58 +08:00
if ( spec_reg = = SDHCI_TRANSFER_MODE )
return pltfm_host - > xfer_mode_shadow ;
2015-10-08 18:36:36 +08:00
if ( spec_reg = = SDHCI_HOST_VERSION )
ret = value & 0xffff ;
2009-12-17 15:27:20 -08:00
else
2015-10-08 18:36:36 +08:00
ret = ( value > > shift ) & 0xffff ;
2016-11-09 11:14:12 +08:00
/* Workaround for T4240-R1.0-R2.0 eSDHC which has incorrect
* vendor version and spec version information .
*/
if ( ( spec_reg = = SDHCI_HOST_VERSION ) & &
( esdhc - > quirk_incorrect_hostver ) )
ret = ( VENDOR_V_23 < < SDHCI_VENDOR_VER_SHIFT ) | SDHCI_SPEC_200 ;
2011-09-09 20:05:46 +08:00
return ret ;
}
2015-10-08 18:36:36 +08:00
static u8 esdhc_readb_fixup ( struct sdhci_host * host ,
int spec_reg , u32 value )
2011-09-09 20:05:46 +08:00
{
2015-10-08 18:36:36 +08:00
u8 ret ;
u8 dma_bits ;
int shift = ( spec_reg & 0x3 ) * 8 ;
ret = ( value > > shift ) & 0xff ;
2012-01-13 15:02:01 +08:00
/*
* " DMA select " locates at offset 0x28 in SD specification , but on
* P5020 or P3041 , it locates at 0x29 .
*/
2015-10-08 18:36:36 +08:00
if ( spec_reg = = SDHCI_HOST_CONTROL ) {
2012-01-13 15:02:01 +08:00
/* DMA select is 22,23 bits in Protocol Control Register */
2015-10-08 18:36:36 +08:00
dma_bits = ( value > > 5 ) & SDHCI_CTRL_DMA_MASK ;
2012-01-13 15:02:01 +08:00
/* fixup the result */
ret & = ~ SDHCI_CTRL_DMA_MASK ;
ret | = dma_bits ;
}
2009-12-17 15:27:20 -08:00
return ret ;
}
2015-10-08 18:36:36 +08:00
/**
* esdhc_write * _fixup - Fixup the SD spec register value so that it could be
* written into eSDHC register .
*
* @ host : pointer to sdhci_host
* @ spec_reg : SD spec register address
* @ value : 8 / 16 / 32 bit SD spec register value that would be written
* @ old_value : 32 bit eSDHC register value on spec_reg address
*
* In SD spec , there are 8 / 16 / 32 / 64 bits registers , while all of eSDHC
* registers are 32 bits . There are differences in register size , register
* address , register function , bit position and function between eSDHC spec
* and SD spec .
*
* Return a fixed up register value
*/
static u32 esdhc_writel_fixup ( struct sdhci_host * host ,
int spec_reg , u32 value , u32 old_value )
2012-12-04 10:41:28 +08:00
{
2015-10-08 18:36:36 +08:00
u32 ret ;
2012-12-04 10:41:28 +08:00
/*
2015-10-08 18:36:36 +08:00
* Enabling IRQSTATEN [ BGESEN ] is just to set IRQSTAT [ BGE ]
* when SYSCTL [ RSTD ] is set for some special operations .
* No any impact on other operation .
2012-12-04 10:41:28 +08:00
*/
2015-10-08 18:36:36 +08:00
if ( spec_reg = = SDHCI_INT_ENABLE )
ret = value | SDHCI_INT_BLK_GAP ;
else
ret = value ;
return ret ;
2012-12-04 10:41:28 +08:00
}
2015-10-08 18:36:36 +08:00
static u32 esdhc_writew_fixup ( struct sdhci_host * host ,
int spec_reg , u16 value , u32 old_value )
2009-12-17 15:27:20 -08:00
{
2015-10-08 18:36:36 +08:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
int shift = ( spec_reg & 0x2 ) * 8 ;
u32 ret ;
switch ( spec_reg ) {
case SDHCI_TRANSFER_MODE :
/*
* Postpone this write , we must do it together with a
* command write that is down below . Return old value .
*/
pltfm_host - > xfer_mode_shadow = value ;
return old_value ;
case SDHCI_COMMAND :
ret = ( value < < 16 ) | pltfm_host - > xfer_mode_shadow ;
return ret ;
}
ret = old_value & ( ~ ( 0xffff < < shift ) ) ;
ret | = ( value < < shift ) ;
if ( spec_reg = = SDHCI_BLOCK_SIZE ) {
2009-12-17 15:27:20 -08:00
/*
* Two last DMA bits are reserved , and first one is used for
* non - standard blksz of 4096 bytes that we don ' t support
* yet . So clear the DMA boundary bits .
*/
2015-10-08 18:36:36 +08:00
ret & = ( ~ SDHCI_MAKE_BLKSZ ( 0x7 , 0 ) ) ;
2009-12-17 15:27:20 -08:00
}
2015-10-08 18:36:36 +08:00
return ret ;
2009-12-17 15:27:20 -08:00
}
2015-10-08 18:36:36 +08:00
static u32 esdhc_writeb_fixup ( struct sdhci_host * host ,
int spec_reg , u8 value , u32 old_value )
2009-12-17 15:27:20 -08:00
{
2015-10-08 18:36:36 +08:00
u32 ret ;
u32 dma_bits ;
u8 tmp ;
int shift = ( spec_reg & 0x3 ) * 8 ;
2015-10-16 15:44:03 +08:00
/*
* eSDHC doesn ' t have a standard power control register , so we do
* nothing here to avoid incorrect operation .
*/
if ( spec_reg = = SDHCI_POWER_CONTROL )
return old_value ;
2012-01-13 15:02:01 +08:00
/*
* " DMA select " location is offset 0x28 in SD specification , but on
* P5020 or P3041 , it ' s located at 0x29 .
*/
2015-10-08 18:36:36 +08:00
if ( spec_reg = = SDHCI_HOST_CONTROL ) {
2013-07-05 12:48:35 -04:00
/*
* If host control register is not standard , exit
* this function
*/
if ( host - > quirks2 & SDHCI_QUIRK2_BROKEN_HOST_CONTROL )
2015-10-08 18:36:36 +08:00
return old_value ;
2013-07-05 12:48:35 -04:00
2012-01-13 15:02:01 +08:00
/* DMA select is 22,23 bits in Protocol Control Register */
2015-10-08 18:36:36 +08:00
dma_bits = ( value & SDHCI_CTRL_DMA_MASK ) < < 5 ;
ret = ( old_value & ( ~ ( SDHCI_CTRL_DMA_MASK < < 5 ) ) ) | dma_bits ;
tmp = ( value & ( ~ SDHCI_CTRL_DMA_MASK ) ) |
( old_value & SDHCI_CTRL_DMA_MASK ) ;
ret = ( ret & ( ~ 0xff ) ) | tmp ;
/* Prevent SDHCI core from writing reserved bits (e.g. HISPD) */
ret & = ~ ESDHC_HOST_CONTROL_RES ;
return ret ;
2012-01-13 15:02:01 +08:00
}
2015-10-08 18:36:36 +08:00
ret = ( old_value & ( ~ ( 0xff < < shift ) ) ) | ( value < < shift ) ;
return ret ;
}
static u32 esdhc_be_readl ( struct sdhci_host * host , int reg )
{
u32 ret ;
u32 value ;
2017-08-15 10:17:03 +08:00
if ( reg = = SDHCI_CAPABILITIES_1 )
value = ioread32be ( host - > ioaddr + ESDHC_CAPABILITIES_1 ) ;
else
value = ioread32be ( host - > ioaddr + reg ) ;
2015-10-08 18:36:36 +08:00
ret = esdhc_readl_fixup ( host , reg , value ) ;
return ret ;
}
static u32 esdhc_le_readl ( struct sdhci_host * host , int reg )
{
u32 ret ;
u32 value ;
2017-08-15 10:17:03 +08:00
if ( reg = = SDHCI_CAPABILITIES_1 )
value = ioread32 ( host - > ioaddr + ESDHC_CAPABILITIES_1 ) ;
else
value = ioread32 ( host - > ioaddr + reg ) ;
2015-10-08 18:36:36 +08:00
ret = esdhc_readl_fixup ( host , reg , value ) ;
return ret ;
}
static u16 esdhc_be_readw ( struct sdhci_host * host , int reg )
{
u16 ret ;
u32 value ;
int base = reg & ~ 0x3 ;
value = ioread32be ( host - > ioaddr + base ) ;
ret = esdhc_readw_fixup ( host , reg , value ) ;
return ret ;
}
static u16 esdhc_le_readw ( struct sdhci_host * host , int reg )
{
u16 ret ;
u32 value ;
int base = reg & ~ 0x3 ;
value = ioread32 ( host - > ioaddr + base ) ;
ret = esdhc_readw_fixup ( host , reg , value ) ;
return ret ;
}
static u8 esdhc_be_readb ( struct sdhci_host * host , int reg )
{
u8 ret ;
u32 value ;
int base = reg & ~ 0x3 ;
value = ioread32be ( host - > ioaddr + base ) ;
ret = esdhc_readb_fixup ( host , reg , value ) ;
return ret ;
}
static u8 esdhc_le_readb ( struct sdhci_host * host , int reg )
{
u8 ret ;
u32 value ;
int base = reg & ~ 0x3 ;
value = ioread32 ( host - > ioaddr + base ) ;
ret = esdhc_readb_fixup ( host , reg , value ) ;
return ret ;
}
static void esdhc_be_writel ( struct sdhci_host * host , u32 val , int reg )
{
u32 value ;
value = esdhc_writel_fixup ( host , reg , val , 0 ) ;
iowrite32be ( value , host - > ioaddr + reg ) ;
}
static void esdhc_le_writel ( struct sdhci_host * host , u32 val , int reg )
{
u32 value ;
value = esdhc_writel_fixup ( host , reg , val , 0 ) ;
iowrite32 ( value , host - > ioaddr + reg ) ;
}
static void esdhc_be_writew ( struct sdhci_host * host , u16 val , int reg )
{
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2015-10-08 18:36:36 +08:00
int base = reg & ~ 0x3 ;
u32 value ;
u32 ret ;
value = ioread32be ( host - > ioaddr + base ) ;
ret = esdhc_writew_fixup ( host , reg , val , value ) ;
if ( reg ! = SDHCI_TRANSFER_MODE )
iowrite32be ( ret , host - > ioaddr + base ) ;
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
/* Starting SW tuning requires ESDHC_SMPCLKSEL to be set
* 1u s later after ESDHC_EXTN is set .
*/
if ( base = = ESDHC_SYSTEM_CONTROL_2 ) {
if ( ! ( value & ESDHC_EXTN ) & & ( ret & ESDHC_EXTN ) & &
esdhc - > in_sw_tuning ) {
udelay ( 1 ) ;
ret | = ESDHC_SMPCLKSEL ;
iowrite32be ( ret , host - > ioaddr + base ) ;
}
}
2015-10-08 18:36:36 +08:00
}
static void esdhc_le_writew ( struct sdhci_host * host , u16 val , int reg )
{
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2015-10-08 18:36:36 +08:00
int base = reg & ~ 0x3 ;
u32 value ;
u32 ret ;
value = ioread32 ( host - > ioaddr + base ) ;
ret = esdhc_writew_fixup ( host , reg , val , value ) ;
if ( reg ! = SDHCI_TRANSFER_MODE )
iowrite32 ( ret , host - > ioaddr + base ) ;
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
/* Starting SW tuning requires ESDHC_SMPCLKSEL to be set
* 1u s later after ESDHC_EXTN is set .
*/
if ( base = = ESDHC_SYSTEM_CONTROL_2 ) {
if ( ! ( value & ESDHC_EXTN ) & & ( ret & ESDHC_EXTN ) & &
esdhc - > in_sw_tuning ) {
udelay ( 1 ) ;
ret | = ESDHC_SMPCLKSEL ;
iowrite32 ( ret , host - > ioaddr + base ) ;
}
}
2015-10-08 18:36:36 +08:00
}
static void esdhc_be_writeb ( struct sdhci_host * host , u8 val , int reg )
{
int base = reg & ~ 0x3 ;
u32 value ;
u32 ret ;
value = ioread32be ( host - > ioaddr + base ) ;
ret = esdhc_writeb_fixup ( host , reg , val , value ) ;
iowrite32be ( ret , host - > ioaddr + base ) ;
}
static void esdhc_le_writeb ( struct sdhci_host * host , u8 val , int reg )
{
int base = reg & ~ 0x3 ;
u32 value ;
u32 ret ;
value = ioread32 ( host - > ioaddr + base ) ;
ret = esdhc_writeb_fixup ( host , reg , val , value ) ;
iowrite32 ( ret , host - > ioaddr + base ) ;
2009-12-17 15:27:20 -08:00
}
2012-12-04 10:41:28 +08:00
/*
* For Abort or Suspend after Stop at Block Gap , ignore the ADMA
* error ( IRQSTAT [ ADMAE ] ) if both Transfer Complete ( IRQSTAT [ TC ] )
* and Block Gap Event ( IRQSTAT [ BGE ] ) are also set .
* For Continue , apply soft reset for data ( SYSCTL [ RSTD ] ) ;
* and re - issue the entire read transaction from beginning .
*/
2015-10-08 18:36:36 +08:00
static void esdhc_of_adma_workaround ( struct sdhci_host * host , u32 intmask )
2012-12-04 10:41:28 +08:00
{
2015-10-08 18:36:36 +08:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
2016-02-16 21:08:26 +08:00
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2012-12-04 10:41:28 +08:00
bool applicable ;
dma_addr_t dmastart ;
dma_addr_t dmanow ;
applicable = ( intmask & SDHCI_INT_DATA_END ) & &
2015-10-08 18:36:36 +08:00
( intmask & SDHCI_INT_BLK_GAP ) & &
( esdhc - > vendor_ver = = VENDOR_V_23 ) ;
2012-12-04 10:41:28 +08:00
if ( ! applicable )
return ;
host - > data - > error = 0 ;
dmastart = sg_dma_address ( host - > data - > sg ) ;
dmanow = dmastart + host - > data - > bytes_xfered ;
/*
* Force update to the next DMA block boundary .
*/
dmanow = ( dmanow & ~ ( SDHCI_DEFAULT_BOUNDARY_SIZE - 1 ) ) +
SDHCI_DEFAULT_BOUNDARY_SIZE ;
host - > data - > bytes_xfered = dmanow - dmastart ;
sdhci_writel ( host , dmanow , SDHCI_DMA_ADDRESS ) ;
}
2010-10-15 12:21:03 +02:00
static int esdhc_of_enable_dma ( struct sdhci_host * host )
2009-12-17 15:27:20 -08:00
{
2015-10-08 18:36:36 +08:00
u32 value ;
2018-07-04 14:34:20 +03:00
struct device * dev = mmc_dev ( host - > mmc ) ;
if ( of_device_is_compatible ( dev - > of_node , " fsl,ls1043a-esdhc " ) | |
of_device_is_compatible ( dev - > of_node , " fsl,ls1046a-esdhc " ) )
dma_set_mask_and_coherent ( dev , DMA_BIT_MASK ( 40 ) ) ;
2015-10-08 18:36:36 +08:00
value = sdhci_readl ( host , ESDHC_DMA_SYSCTL ) ;
2019-09-22 11:26:58 +01:00
if ( of_dma_is_coherent ( dev - > of_node ) )
value | = ESDHC_DMA_SNOOP ;
else
value & = ~ ESDHC_DMA_SNOOP ;
2015-10-08 18:36:36 +08:00
sdhci_writel ( host , value , ESDHC_DMA_SYSCTL ) ;
2009-12-17 15:27:20 -08:00
return 0 ;
}
2010-10-15 12:21:03 +02:00
static unsigned int esdhc_of_get_max_clock ( struct sdhci_host * host )
2009-12-17 15:27:20 -08:00
{
2011-07-20 17:13:36 -04:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
2017-04-20 16:14:40 +08:00
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2009-12-17 15:27:20 -08:00
2017-04-20 16:14:40 +08:00
if ( esdhc - > peripheral_clock )
return esdhc - > peripheral_clock ;
else
return pltfm_host - > clock ;
2009-12-17 15:27:20 -08:00
}
2010-10-15 12:21:03 +02:00
static unsigned int esdhc_of_get_min_clock ( struct sdhci_host * host )
2009-12-17 15:27:20 -08:00
{
2011-07-20 17:13:36 -04:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
2017-04-20 16:14:40 +08:00
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
unsigned int clock ;
2009-12-17 15:27:20 -08:00
2017-04-20 16:14:40 +08:00
if ( esdhc - > peripheral_clock )
clock = esdhc - > peripheral_clock ;
else
clock = pltfm_host - > clock ;
return clock / 256 / 16 ;
2009-12-17 15:27:20 -08:00
}
2017-09-21 16:43:31 +08:00
static void esdhc_clock_enable ( struct sdhci_host * host , bool enable )
{
2020-01-08 12:07:13 +08:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2017-09-21 16:43:31 +08:00
ktime_t timeout ;
2020-01-08 12:07:13 +08:00
u32 val , clk_en ;
clk_en = ESDHC_CLOCK_SDCLKEN ;
/*
* IPGEN / HCKEN / PEREN bits exist on eSDHC whose vendor version
* is 2.2 or lower .
*/
if ( esdhc - > vendor_ver < = VENDOR_V_22 )
clk_en | = ( ESDHC_CLOCK_IPGEN | ESDHC_CLOCK_HCKEN |
ESDHC_CLOCK_PEREN ) ;
2017-09-21 16:43:31 +08:00
val = sdhci_readl ( host , ESDHC_SYSTEM_CONTROL ) ;
if ( enable )
2020-01-08 12:07:13 +08:00
val | = clk_en ;
2017-09-21 16:43:31 +08:00
else
2020-01-08 12:07:13 +08:00
val & = ~ clk_en ;
2017-09-21 16:43:31 +08:00
sdhci_writel ( host , val , ESDHC_SYSTEM_CONTROL ) ;
2020-01-08 12:07:13 +08:00
/*
* Wait max 20 ms . If vendor version is 2.2 or lower , do not
* wait clock stable bit which does not exist .
*/
2017-09-21 16:43:31 +08:00
timeout = ktime_add_ms ( ktime_get ( ) , 20 ) ;
2020-01-08 12:07:13 +08:00
while ( esdhc - > vendor_ver > VENDOR_V_22 ) {
2018-12-10 10:56:24 +02:00
bool timedout = ktime_after ( ktime_get ( ) , timeout ) ;
2020-01-08 12:07:13 +08:00
if ( sdhci_readl ( host , ESDHC_PRSSTAT ) & ESDHC_CLOCK_STABLE )
2018-12-10 10:56:24 +02:00
break ;
if ( timedout ) {
2017-09-21 16:43:31 +08:00
pr_err ( " %s: Internal clock never stabilised. \n " ,
mmc_hostname ( host - > mmc ) ) ;
break ;
}
2020-01-08 12:07:13 +08:00
usleep_range ( 10 , 20 ) ;
2017-09-21 16:43:31 +08:00
}
}
2019-10-09 15:41:39 +08:00
static void esdhc_flush_async_fifo ( struct sdhci_host * host )
{
ktime_t timeout ;
u32 val ;
val = sdhci_readl ( host , ESDHC_DMA_SYSCTL ) ;
val | = ESDHC_FLUSH_ASYNC_FIFO ;
sdhci_writel ( host , val , ESDHC_DMA_SYSCTL ) ;
/* Wait max 20 ms */
timeout = ktime_add_ms ( ktime_get ( ) , 20 ) ;
while ( 1 ) {
bool timedout = ktime_after ( ktime_get ( ) , timeout ) ;
if ( ! ( sdhci_readl ( host , ESDHC_DMA_SYSCTL ) &
ESDHC_FLUSH_ASYNC_FIFO ) )
break ;
if ( timedout ) {
pr_err ( " %s: flushing asynchronous FIFO timeout. \n " ,
mmc_hostname ( host - > mmc ) ) ;
break ;
}
usleep_range ( 10 , 20 ) ;
}
}
2012-02-14 14:05:37 +08:00
static void esdhc_of_set_clock ( struct sdhci_host * host , unsigned int clock )
{
2015-10-08 18:36:36 +08:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
2016-02-16 21:08:26 +08:00
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2020-01-08 12:07:13 +08:00
unsigned int pre_div = 1 , div = 1 ;
unsigned int clock_fixup = 0 ;
2017-04-26 10:45:49 +08:00
ktime_t timeout ;
2013-09-13 19:11:32 +08:00
u32 temp ;
2017-09-21 16:43:31 +08:00
if ( clock = = 0 ) {
2020-01-08 12:07:13 +08:00
host - > mmc - > actual_clock = 0 ;
2017-09-21 16:43:31 +08:00
esdhc_clock_enable ( host , false ) ;
2014-04-25 12:58:45 +01:00
return ;
2017-09-21 16:43:31 +08:00
}
2013-09-13 19:11:32 +08:00
2020-01-08 12:07:13 +08:00
/* Start pre_div at 2 for vendor version < 2.3. */
2015-10-08 18:36:36 +08:00
if ( esdhc - > vendor_ver < VENDOR_V_23 )
2015-08-11 10:53:34 +08:00
pre_div = 2 ;
2020-01-08 12:07:13 +08:00
/* Fix clock value. */
2018-06-25 16:46:24 +08:00
if ( host - > mmc - > card & & mmc_card_sd ( host - > mmc - > card ) & &
2020-01-08 12:07:13 +08:00
esdhc - > clk_fixup & & host - > mmc - > ios . timing = = MMC_TIMING_LEGACY )
clock_fixup = esdhc - > clk_fixup - > sd_dflt_max_clk ;
2018-06-25 16:46:24 +08:00
else if ( esdhc - > clk_fixup )
2020-01-08 12:07:13 +08:00
clock_fixup = esdhc - > clk_fixup - > max_clk [ host - > mmc - > ios . timing ] ;
2017-04-20 14:58:29 +08:00
2020-01-08 12:07:13 +08:00
if ( clock_fixup = = 0 | | clock < clock_fixup )
clock_fixup = clock ;
2012-02-14 14:05:37 +08:00
2020-01-08 12:07:13 +08:00
/* Calculate pre_div and div. */
while ( host - > max_clk / pre_div / 16 > clock_fixup & & pre_div < 256 )
2013-09-13 19:11:32 +08:00
pre_div * = 2 ;
2020-01-08 12:07:13 +08:00
while ( host - > max_clk / pre_div / div > clock_fixup & & div < 16 )
2013-09-13 19:11:32 +08:00
div + + ;
2020-01-08 12:07:13 +08:00
esdhc - > div_ratio = pre_div * div ;
/* Limit clock division for HS400 200MHz clock for quirk. */
2018-11-23 11:15:35 +08:00
if ( esdhc - > quirk_limited_clk_division & &
clock = = MMC_HS200_MAX_DTR & &
( host - > mmc - > ios . timing = = MMC_TIMING_MMC_HS400 | |
host - > flags & SDHCI_HS400_TUNING ) ) {
2020-01-08 12:07:13 +08:00
if ( esdhc - > div_ratio < = 4 ) {
2018-11-23 11:15:35 +08:00
pre_div = 4 ;
div = 1 ;
2020-01-08 12:07:13 +08:00
} else if ( esdhc - > div_ratio < = 8 ) {
2018-11-23 11:15:35 +08:00
pre_div = 4 ;
div = 2 ;
2020-01-08 12:07:13 +08:00
} else if ( esdhc - > div_ratio < = 12 ) {
2018-11-23 11:15:35 +08:00
pre_div = 4 ;
div = 3 ;
} else {
2018-12-06 09:24:11 +00:00
pr_warn ( " %s: using unsupported clock division. \n " ,
2018-11-23 11:15:35 +08:00
mmc_hostname ( host - > mmc ) ) ;
}
2020-01-08 12:07:13 +08:00
esdhc - > div_ratio = pre_div * div ;
2018-11-23 11:15:35 +08:00
}
2020-01-08 12:07:13 +08:00
host - > mmc - > actual_clock = host - > max_clk / esdhc - > div_ratio ;
2013-09-13 19:11:32 +08:00
dev_dbg ( mmc_dev ( host - > mmc ) , " desired SD clock: %d, actual: %d \n " ,
2020-01-08 12:07:13 +08:00
clock , host - > mmc - > actual_clock ) ;
/* Set clock division into register. */
2013-09-13 19:11:32 +08:00
pre_div > > = 1 ;
div - - ;
2020-01-08 12:07:13 +08:00
esdhc_clock_enable ( host , false ) ;
2013-09-13 19:11:32 +08:00
temp = sdhci_readl ( host , ESDHC_SYSTEM_CONTROL ) ;
2020-01-08 12:07:13 +08:00
temp & = ~ ESDHC_CLOCK_MASK ;
temp | = ( ( div < < ESDHC_DIVIDER_SHIFT ) |
( pre_div < < ESDHC_PREDIV_SHIFT ) ) ;
2013-09-13 19:11:32 +08:00
sdhci_writel ( host , temp , ESDHC_SYSTEM_CONTROL ) ;
2016-12-26 17:46:30 +08:00
2020-01-08 12:07:13 +08:00
/*
* Wait max 20 ms . If vendor version is 2.2 or lower , do not
* wait clock stable bit which does not exist .
*/
timeout = ktime_add_ms ( ktime_get ( ) , 20 ) ;
while ( esdhc - > vendor_ver > VENDOR_V_22 ) {
bool timedout = ktime_after ( ktime_get ( ) , timeout ) ;
if ( sdhci_readl ( host , ESDHC_PRSSTAT ) & ESDHC_CLOCK_STABLE )
break ;
if ( timedout ) {
pr_err ( " %s: Internal clock never stabilised. \n " ,
mmc_hostname ( host - > mmc ) ) ;
break ;
}
usleep_range ( 10 , 20 ) ;
}
/* Additional setting for HS400. */
2018-11-23 11:15:34 +08:00
if ( host - > mmc - > ios . timing = = MMC_TIMING_MMC_HS400 & &
clock = = MMC_HS200_MAX_DTR ) {
temp = sdhci_readl ( host , ESDHC_TBCTL ) ;
sdhci_writel ( host , temp | ESDHC_HS400_MODE , ESDHC_TBCTL ) ;
temp = sdhci_readl ( host , ESDHC_SDCLKCTL ) ;
sdhci_writel ( host , temp | ESDHC_CMD_CLK_CTL , ESDHC_SDCLKCTL ) ;
esdhc_clock_enable ( host , true ) ;
temp = sdhci_readl ( host , ESDHC_DLLCFG0 ) ;
2018-11-23 11:15:36 +08:00
temp | = ESDHC_DLL_ENABLE ;
if ( host - > mmc - > actual_clock = = MMC_HS200_MAX_DTR )
temp | = ESDHC_DLL_FREQ_SEL ;
2018-11-23 11:15:34 +08:00
sdhci_writel ( host , temp , ESDHC_DLLCFG0 ) ;
2020-10-20 16:11:16 +08:00
temp | = ESDHC_DLL_RESET ;
sdhci_writel ( host , temp , ESDHC_DLLCFG0 ) ;
udelay ( 1 ) ;
temp & = ~ ESDHC_DLL_RESET ;
sdhci_writel ( host , temp , ESDHC_DLLCFG0 ) ;
/* Wait max 20 ms */
if ( read_poll_timeout ( sdhci_readl , temp ,
temp & ESDHC_DLL_STS_SLV_LOCK ,
10 , 20000 , false ,
host , ESDHC_DLLSTAT0 ) )
pr_err ( " %s: timeout for delay chain lock. \n " ,
mmc_hostname ( host - > mmc ) ) ;
2018-11-23 11:15:34 +08:00
temp = sdhci_readl ( host , ESDHC_TBCTL ) ;
sdhci_writel ( host , temp | ESDHC_HS400_WNDW_ADJUST , ESDHC_TBCTL ) ;
esdhc_clock_enable ( host , false ) ;
2019-10-09 15:41:39 +08:00
esdhc_flush_async_fifo ( host ) ;
2018-11-23 11:15:34 +08:00
}
2020-01-20 17:48:35 +08:00
esdhc_clock_enable ( host , true ) ;
2012-02-14 14:05:37 +08:00
}
2014-04-25 12:57:07 +01:00
static void esdhc_pltfm_set_bus_width ( struct sdhci_host * host , int width )
2013-06-27 12:00:05 -04:00
{
u32 ctrl ;
2015-10-08 18:36:36 +08:00
ctrl = sdhci_readl ( host , ESDHC_PROCTL ) ;
ctrl & = ( ~ ESDHC_CTRL_BUSWIDTH_MASK ) ;
2013-06-27 12:00:05 -04:00
switch ( width ) {
case MMC_BUS_WIDTH_8 :
2015-10-08 18:36:36 +08:00
ctrl | = ESDHC_CTRL_8BITBUS ;
2013-06-27 12:00:05 -04:00
break ;
case MMC_BUS_WIDTH_4 :
2015-10-08 18:36:36 +08:00
ctrl | = ESDHC_CTRL_4BITBUS ;
2013-06-27 12:00:05 -04:00
break ;
default :
break ;
}
2015-10-08 18:36:36 +08:00
sdhci_writel ( host , ctrl , ESDHC_PROCTL ) ;
2013-06-27 12:00:05 -04:00
}
2014-12-09 09:40:38 +01:00
static void esdhc_reset ( struct sdhci_host * host , u8 mask )
{
2018-11-23 11:15:37 +08:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2020-01-08 12:07:12 +08:00
u32 val , bus_width = 0 ;
2017-12-01 15:09:34 +08:00
2020-01-08 12:07:12 +08:00
/*
* Add delay to make sure all the DMA transfers are finished
* for quirk .
*/
2019-12-19 11:23:35 +08:00
if ( esdhc - > quirk_delay_before_data_reset & &
( mask & SDHCI_RESET_DATA ) & &
( host - > flags & SDHCI_REQ_USE_DMA ) )
mdelay ( 5 ) ;
2020-01-08 12:07:12 +08:00
/*
* Save bus - width for eSDHC whose vendor version is 2.2
* or lower for data reset .
*/
if ( ( mask & SDHCI_RESET_DATA ) & &
( esdhc - > vendor_ver < = VENDOR_V_22 ) ) {
val = sdhci_readl ( host , ESDHC_PROCTL ) ;
bus_width = val & ESDHC_CTRL_BUSWIDTH_MASK ;
}
2014-12-09 09:40:38 +01:00
sdhci_reset ( host , mask ) ;
2020-01-08 12:07:12 +08:00
/*
* Restore bus - width setting and interrupt registers for eSDHC
* whose vendor version is 2.2 or lower for data reset .
*/
if ( ( mask & SDHCI_RESET_DATA ) & &
( esdhc - > vendor_ver < = VENDOR_V_22 ) ) {
val = sdhci_readl ( host , ESDHC_PROCTL ) ;
val & = ~ ESDHC_CTRL_BUSWIDTH_MASK ;
val | = bus_width ;
sdhci_writel ( host , val , ESDHC_PROCTL ) ;
sdhci_writel ( host , host - > ier , SDHCI_INT_ENABLE ) ;
sdhci_writel ( host , host - > ier , SDHCI_SIGNAL_ENABLE ) ;
}
2017-12-01 15:09:34 +08:00
2020-01-08 12:07:12 +08:00
/*
* Some bits have to be cleaned manually for eSDHC whose spec
* version is higher than 3.0 for all reset .
*/
if ( ( mask & SDHCI_RESET_ALL ) & &
( esdhc - > spec_ver > = SDHCI_SPEC_300 ) ) {
2017-12-01 15:09:34 +08:00
val = sdhci_readl ( host , ESDHC_TBCTL ) ;
val & = ~ ESDHC_TB_EN ;
sdhci_writel ( host , val , ESDHC_TBCTL ) ;
2018-11-23 11:15:37 +08:00
2020-01-08 12:07:12 +08:00
/*
* Initialize eSDHC_DLLCFG1 [ DLL_PD_PULSE_STRETCH_SEL ] to
* 0 for quirk .
*/
2018-11-23 11:15:37 +08:00
if ( esdhc - > quirk_unreliable_pulse_detection ) {
val = sdhci_readl ( host , ESDHC_DLLCFG1 ) ;
val & = ~ ESDHC_DLL_PD_PULSE_STRETCH_SEL ;
sdhci_writel ( host , val , ESDHC_DLLCFG1 ) ;
}
2017-12-01 15:09:34 +08:00
}
2014-12-09 09:40:38 +01:00
}
2017-04-20 16:14:41 +08:00
/* The SCFG, Supplemental Configuration Unit, provides SoC specific
* configuration and status registers for the device . There is a
* SDHC IO VSEL control register on SCFG for some platforms . It ' s
* used to support SDHC IO voltage switching .
*/
static const struct of_device_id scfg_device_ids [ ] = {
{ . compatible = " fsl,t1040-scfg " , } ,
{ . compatible = " fsl,ls1012a-scfg " , } ,
{ . compatible = " fsl,ls1046a-scfg " , } ,
{ }
} ;
/* SDHC IO VSEL control register definition */
# define SCFG_SDHCIOVSELCR 0x408
# define SDHCIOVSELCR_TGLEN 0x80000000
# define SDHCIOVSELCR_VSELVAL 0x60000000
# define SDHCIOVSELCR_SDHC_VS 0x00000001
static int esdhc_signal_voltage_switch ( struct mmc_host * mmc ,
struct mmc_ios * ios )
{
struct sdhci_host * host = mmc_priv ( mmc ) ;
struct device_node * scfg_node ;
void __iomem * scfg_base = NULL ;
u32 sdhciovselcr ;
u32 val ;
/*
* Signal Voltage Switching is only applicable for Host Controllers
* v3 .00 and above .
*/
if ( host - > version < SDHCI_SPEC_300 )
return 0 ;
val = sdhci_readl ( host , ESDHC_PROCTL ) ;
switch ( ios - > signal_voltage ) {
case MMC_SIGNAL_VOLTAGE_330 :
val & = ~ ESDHC_VOLT_SEL ;
sdhci_writel ( host , val , ESDHC_PROCTL ) ;
return 0 ;
case MMC_SIGNAL_VOLTAGE_180 :
scfg_node = of_find_matching_node ( NULL , scfg_device_ids ) ;
if ( scfg_node )
scfg_base = of_iomap ( scfg_node , 0 ) ;
if ( scfg_base ) {
sdhciovselcr = SDHCIOVSELCR_TGLEN |
SDHCIOVSELCR_VSELVAL ;
iowrite32be ( sdhciovselcr ,
scfg_base + SCFG_SDHCIOVSELCR ) ;
val | = ESDHC_VOLT_SEL ;
sdhci_writel ( host , val , ESDHC_PROCTL ) ;
mdelay ( 5 ) ;
sdhciovselcr = SDHCIOVSELCR_TGLEN |
SDHCIOVSELCR_SDHC_VS ;
iowrite32be ( sdhciovselcr ,
scfg_base + SCFG_SDHCIOVSELCR ) ;
iounmap ( scfg_base ) ;
} else {
val | = ESDHC_VOLT_SEL ;
sdhci_writel ( host , val , ESDHC_PROCTL ) ;
}
return 0 ;
default :
return 0 ;
}
}
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
static struct soc_device_attribute soc_tuning_erratum_type1 [ ] = {
2019-12-12 15:52:19 +08:00
{ . family = " QorIQ T1023 " , } ,
{ . family = " QorIQ T1040 " , } ,
{ . family = " QorIQ T2080 " , } ,
{ . family = " QorIQ LS1021A " , } ,
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
{ } ,
} ;
static struct soc_device_attribute soc_tuning_erratum_type2 [ ] = {
2019-12-12 15:52:19 +08:00
{ . family = " QorIQ LS1012A " , } ,
{ . family = " QorIQ LS1043A " , } ,
{ . family = " QorIQ LS1046A " , } ,
{ . family = " QorIQ LS1080A " , } ,
{ . family = " QorIQ LS2080A " , } ,
{ . family = " QorIQ LA1575A " , } ,
2018-08-23 16:48:32 +08:00
{ } ,
} ;
2018-11-23 11:15:34 +08:00
static void esdhc_tuning_block_enable ( struct sdhci_host * host , bool enable )
2017-04-20 16:14:42 +08:00
{
u32 val ;
esdhc_clock_enable ( host , false ) ;
2019-10-09 15:41:39 +08:00
esdhc_flush_async_fifo ( host ) ;
2017-04-20 16:14:42 +08:00
val = sdhci_readl ( host , ESDHC_TBCTL ) ;
2018-11-23 11:15:34 +08:00
if ( enable )
val | = ESDHC_TB_EN ;
else
val & = ~ ESDHC_TB_EN ;
2017-04-20 16:14:42 +08:00
sdhci_writel ( host , val , ESDHC_TBCTL ) ;
2018-11-23 11:15:34 +08:00
2017-04-20 16:14:42 +08:00
esdhc_clock_enable ( host , true ) ;
2018-11-23 11:15:34 +08:00
}
2019-12-12 15:52:18 +08:00
static void esdhc_tuning_window_ptr ( struct sdhci_host * host , u8 * window_start ,
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
u8 * window_end )
{
u32 val ;
/* Write TBCTL[11:8]=4'h8 */
val = sdhci_readl ( host , ESDHC_TBCTL ) ;
val & = ~ ( 0xf < < 8 ) ;
val | = 8 < < 8 ;
sdhci_writel ( host , val , ESDHC_TBCTL ) ;
mdelay ( 1 ) ;
/* Read TBCTL[31:0] register and rewrite again */
val = sdhci_readl ( host , ESDHC_TBCTL ) ;
sdhci_writel ( host , val , ESDHC_TBCTL ) ;
mdelay ( 1 ) ;
/* Read the TBSTAT[31:0] register twice */
val = sdhci_readl ( host , ESDHC_TBSTAT ) ;
val = sdhci_readl ( host , ESDHC_TBSTAT ) ;
2019-12-12 15:52:18 +08:00
* window_end = val & 0xff ;
* window_start = ( val > > 8 ) & 0xff ;
}
static void esdhc_prepare_sw_tuning ( struct sdhci_host * host , u8 * window_start ,
u8 * window_end )
{
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
u8 start_ptr , end_ptr ;
if ( esdhc - > quirk_tuning_erratum_type1 ) {
* window_start = 5 * esdhc - > div_ratio ;
* window_end = 3 * esdhc - > div_ratio ;
return ;
}
esdhc_tuning_window_ptr ( host , & start_ptr , & end_ptr ) ;
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
/* Reset data lines by setting ESDHCCTL[RSTD] */
sdhci_reset ( host , SDHCI_RESET_DATA ) ;
/* Write 32'hFFFF_FFFF to IRQSTAT register */
sdhci_writel ( host , 0xFFFFFFFF , SDHCI_INT_STATUS ) ;
2019-12-12 15:52:19 +08:00
/* If TBSTAT[15:8]-TBSTAT[7:0] > (4 * div_ratio) + 2
* or TBSTAT [ 7 : 0 ] - TBSTAT [ 15 : 8 ] > ( 4 * div_ratio ) + 2 ,
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
* then program TBPTR [ TB_WNDW_END_PTR ] = 4 * div_ratio
* and program TBPTR [ TB_WNDW_START_PTR ] = 8 * div_ratio .
*/
2019-12-12 15:52:19 +08:00
if ( abs ( start_ptr - end_ptr ) > ( 4 * esdhc - > div_ratio + 2 ) ) {
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
* window_start = 8 * esdhc - > div_ratio ;
* window_end = 4 * esdhc - > div_ratio ;
} else {
* window_start = 5 * esdhc - > div_ratio ;
* window_end = 3 * esdhc - > div_ratio ;
}
}
static int esdhc_execute_sw_tuning ( struct mmc_host * mmc , u32 opcode ,
u8 window_start , u8 window_end )
{
struct sdhci_host * host = mmc_priv ( mmc ) ;
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
u32 val ;
int ret ;
/* Program TBPTR[TB_WNDW_END_PTR] and TBPTR[TB_WNDW_START_PTR] */
val = ( ( u32 ) window_start < < ESDHC_WNDW_STRT_PTR_SHIFT ) &
ESDHC_WNDW_STRT_PTR_MASK ;
val | = window_end & ESDHC_WNDW_END_PTR_MASK ;
sdhci_writel ( host , val , ESDHC_TBPTR ) ;
/* Program the software tuning mode by setting TBCTL[TB_MODE]=2'h3 */
val = sdhci_readl ( host , ESDHC_TBCTL ) ;
val & = ~ ESDHC_TB_MODE_MASK ;
val | = ESDHC_TB_MODE_SW ;
sdhci_writel ( host , val , ESDHC_TBCTL ) ;
esdhc - > in_sw_tuning = true ;
ret = sdhci_execute_tuning ( mmc , opcode ) ;
esdhc - > in_sw_tuning = false ;
return ret ;
}
2018-11-23 11:15:34 +08:00
static int esdhc_execute_tuning ( struct mmc_host * mmc , u32 opcode )
{
struct sdhci_host * host = mmc_priv ( mmc ) ;
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
u8 window_start , window_end ;
int ret , retries = 1 ;
2018-11-23 11:15:34 +08:00
bool hs400_tuning ;
2019-06-14 16:29:53 +08:00
unsigned int clk ;
2018-11-23 11:15:34 +08:00
u32 val ;
2019-06-14 16:29:53 +08:00
/* For tuning mode, the sd clock divisor value
* must be larger than 3 according to reference manual .
*/
clk = esdhc - > peripheral_clock / 3 ;
if ( host - > clock > clk )
esdhc_of_set_clock ( host , clk ) ;
2018-11-23 11:15:34 +08:00
esdhc_tuning_block_enable ( host , true ) ;
2020-10-23 00:23:37 +02:00
/*
* The eSDHC controller takes the data timeout value into account
* during tuning . If the SD card is too slow sending the response , the
* timer will expire and a " Buffer Read Ready " interrupt without data
* is triggered . This leads to tuning errors .
*
* Just set the timeout to the maximum value because the core will
* already take care of it in sdhci_send_tuning ( ) .
*/
sdhci_writeb ( host , 0xe , SDHCI_TIMEOUT_CONTROL ) ;
2018-11-23 11:15:34 +08:00
hs400_tuning = host - > flags & SDHCI_HS400_TUNING ;
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
do {
if ( esdhc - > quirk_limited_clk_division & &
hs400_tuning )
esdhc_of_set_clock ( host , host - > clock ) ;
2017-04-20 16:14:42 +08:00
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
/* Do HW tuning */
val = sdhci_readl ( host , ESDHC_TBCTL ) ;
val & = ~ ESDHC_TB_MODE_MASK ;
val | = ESDHC_TB_MODE_3 ;
sdhci_writel ( host , val , ESDHC_TBCTL ) ;
2018-08-23 16:48:32 +08:00
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
ret = sdhci_execute_tuning ( mmc , opcode ) ;
if ( ret )
break ;
2018-08-23 16:48:32 +08:00
2019-12-12 15:52:19 +08:00
/* For type2 affected platforms of the tuning erratum,
* tuning may succeed although eSDHC might not have
* tuned properly . Need to check tuning window .
*/
if ( esdhc - > quirk_tuning_erratum_type2 & &
! host - > tuning_err ) {
esdhc_tuning_window_ptr ( host , & window_start ,
& window_end ) ;
if ( abs ( window_start - window_end ) >
( 4 * esdhc - > div_ratio + 2 ) )
host - > tuning_err = - EAGAIN ;
}
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
/* If HW tuning fails and triggers erratum,
* try workaround .
2018-08-23 16:48:32 +08:00
*/
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
ret = host - > tuning_err ;
if ( ret = = - EAGAIN & &
( esdhc - > quirk_tuning_erratum_type1 | |
esdhc - > quirk_tuning_erratum_type2 ) ) {
/* Recover HS400 tuning flag */
if ( hs400_tuning )
host - > flags | = SDHCI_HS400_TUNING ;
pr_info ( " %s: Hold on to use fixed sampling clock. Try SW tuning! \n " ,
mmc_hostname ( mmc ) ) ;
/* Do SW tuning */
esdhc_prepare_sw_tuning ( host , & window_start ,
& window_end ) ;
ret = esdhc_execute_sw_tuning ( mmc , opcode ,
window_start ,
window_end ) ;
if ( ret )
break ;
/* Retry both HW/SW tuning with reduced clock. */
ret = host - > tuning_err ;
if ( ret = = - EAGAIN & & retries ) {
/* Recover HS400 tuning flag */
if ( hs400_tuning )
host - > flags | = SDHCI_HS400_TUNING ;
clk = host - > max_clk / ( esdhc - > div_ratio + 1 ) ;
esdhc_of_set_clock ( host , clk ) ;
pr_info ( " %s: Hold on to use fixed sampling clock. Try tuning with reduced clock! \n " ,
mmc_hostname ( mmc ) ) ;
} else {
break ;
}
} else {
break ;
}
} while ( retries - - ) ;
if ( ret ) {
esdhc_tuning_block_enable ( host , false ) ;
} else if ( hs400_tuning ) {
val = sdhci_readl ( host , ESDHC_SDTIMNGCTL ) ;
val | = ESDHC_FLW_CTL_BG ;
sdhci_writel ( host , val , ESDHC_SDTIMNGCTL ) ;
2018-08-23 16:48:32 +08:00
}
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
2018-11-23 11:15:34 +08:00
return ret ;
}
static void esdhc_set_uhs_signaling ( struct sdhci_host * host ,
unsigned int timing )
{
2020-05-22 11:12:56 +08:00
u32 val ;
/*
* There are specific registers setting for HS400 mode .
* Clean all of them if controller is in HS400 mode to
* exit HS400 mode before re - setting any speed mode .
*/
val = sdhci_readl ( host , ESDHC_TBCTL ) ;
if ( val & ESDHC_HS400_MODE ) {
val = sdhci_readl ( host , ESDHC_SDTIMNGCTL ) ;
val & = ~ ESDHC_FLW_CTL_BG ;
sdhci_writel ( host , val , ESDHC_SDTIMNGCTL ) ;
val = sdhci_readl ( host , ESDHC_SDCLKCTL ) ;
val & = ~ ESDHC_CMD_CLK_CTL ;
sdhci_writel ( host , val , ESDHC_SDCLKCTL ) ;
esdhc_clock_enable ( host , false ) ;
val = sdhci_readl ( host , ESDHC_TBCTL ) ;
val & = ~ ESDHC_HS400_MODE ;
sdhci_writel ( host , val , ESDHC_TBCTL ) ;
esdhc_clock_enable ( host , true ) ;
val = sdhci_readl ( host , ESDHC_DLLCFG0 ) ;
val & = ~ ( ESDHC_DLL_ENABLE | ESDHC_DLL_FREQ_SEL ) ;
sdhci_writel ( host , val , ESDHC_DLLCFG0 ) ;
val = sdhci_readl ( host , ESDHC_TBCTL ) ;
val & = ~ ESDHC_HS400_WNDW_ADJUST ;
sdhci_writel ( host , val , ESDHC_TBCTL ) ;
esdhc_tuning_block_enable ( host , false ) ;
}
2018-11-23 11:15:34 +08:00
if ( timing = = MMC_TIMING_MMC_HS400 )
esdhc_tuning_block_enable ( host , true ) ;
else
sdhci_set_uhs_signaling ( host , timing ) ;
2017-04-20 16:14:42 +08:00
}
2019-03-11 02:16:47 +00:00
static u32 esdhc_irq ( struct sdhci_host * host , u32 intmask )
{
2020-09-03 13:20:29 +12:00
struct sdhci_pltfm_host * pltfm_host = sdhci_priv ( host ) ;
struct sdhci_esdhc * esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2019-03-11 02:16:47 +00:00
u32 command ;
2020-09-03 13:20:29 +12:00
if ( esdhc - > quirk_trans_complete_erratum ) {
2019-03-11 02:16:47 +00:00
command = SDHCI_GET_CMD ( sdhci_readw ( host ,
SDHCI_COMMAND ) ) ;
if ( command = = MMC_WRITE_MULTIPLE_BLOCK & &
sdhci_readw ( host , SDHCI_BLOCK_COUNT ) & &
intmask & SDHCI_INT_DATA_END ) {
intmask & = ~ SDHCI_INT_DATA_END ;
sdhci_writel ( host , SDHCI_INT_DATA_END ,
SDHCI_INT_STATUS ) ;
}
}
return intmask ;
}
2016-07-27 11:01:48 +02:00
# ifdef CONFIG_PM_SLEEP
2014-04-25 12:59:46 +01:00
static u32 esdhc_proctl ;
static int esdhc_of_suspend ( struct device * dev )
{
struct sdhci_host * host = dev_get_drvdata ( dev ) ;
2015-10-08 18:36:36 +08:00
esdhc_proctl = sdhci_readl ( host , SDHCI_HOST_CONTROL ) ;
2014-04-25 12:59:46 +01:00
2017-03-20 19:50:32 +02:00
if ( host - > tuning_mode ! = SDHCI_TUNING_MODE_3 )
mmc_retune_needed ( host - > mmc ) ;
2014-04-25 12:59:46 +01:00
return sdhci_suspend_host ( host ) ;
}
2014-05-23 10:36:44 +02:00
static int esdhc_of_resume ( struct device * dev )
2014-04-25 12:59:46 +01:00
{
struct sdhci_host * host = dev_get_drvdata ( dev ) ;
int ret = sdhci_resume_host ( host ) ;
if ( ret = = 0 ) {
/* Isn't this already done by sdhci_resume_host() ? --rmk */
esdhc_of_enable_dma ( host ) ;
2015-10-08 18:36:36 +08:00
sdhci_writel ( host , esdhc_proctl , SDHCI_HOST_CONTROL ) ;
2014-04-25 12:59:46 +01:00
}
return ret ;
}
# endif
2016-07-27 11:01:48 +02:00
static SIMPLE_DEV_PM_OPS ( esdhc_of_dev_pm_ops ,
esdhc_of_suspend ,
esdhc_of_resume ) ;
2015-10-08 18:36:36 +08:00
static const struct sdhci_ops sdhci_esdhc_be_ops = {
. read_l = esdhc_be_readl ,
. read_w = esdhc_be_readw ,
. read_b = esdhc_be_readb ,
. write_l = esdhc_be_writel ,
. write_w = esdhc_be_writew ,
. write_b = esdhc_be_writeb ,
. set_clock = esdhc_of_set_clock ,
. enable_dma = esdhc_of_enable_dma ,
. get_max_clock = esdhc_of_get_max_clock ,
. get_min_clock = esdhc_of_get_min_clock ,
. adma_workaround = esdhc_of_adma_workaround ,
. set_bus_width = esdhc_pltfm_set_bus_width ,
. reset = esdhc_reset ,
2018-11-23 11:15:34 +08:00
. set_uhs_signaling = esdhc_set_uhs_signaling ,
2019-03-11 02:16:47 +00:00
. irq = esdhc_irq ,
2015-10-08 18:36:36 +08:00
} ;
static const struct sdhci_ops sdhci_esdhc_le_ops = {
. read_l = esdhc_le_readl ,
. read_w = esdhc_le_readw ,
. read_b = esdhc_le_readb ,
. write_l = esdhc_le_writel ,
. write_w = esdhc_le_writew ,
. write_b = esdhc_le_writeb ,
. set_clock = esdhc_of_set_clock ,
. enable_dma = esdhc_of_enable_dma ,
. get_max_clock = esdhc_of_get_max_clock ,
. get_min_clock = esdhc_of_get_min_clock ,
. adma_workaround = esdhc_of_adma_workaround ,
. set_bus_width = esdhc_pltfm_set_bus_width ,
. reset = esdhc_reset ,
2018-11-23 11:15:34 +08:00
. set_uhs_signaling = esdhc_set_uhs_signaling ,
2019-03-11 02:16:47 +00:00
. irq = esdhc_irq ,
2015-10-08 18:36:36 +08:00
} ;
static const struct sdhci_pltfm_data sdhci_esdhc_be_pdata = {
2016-12-26 17:40:44 +08:00
. quirks = ESDHC_DEFAULT_QUIRKS |
# ifdef CONFIG_PPC
SDHCI_QUIRK_BROKEN_CARD_DETECTION |
# endif
SDHCI_QUIRK_NO_CARD_NO_RESET |
SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC ,
2015-10-08 18:36:36 +08:00
. ops = & sdhci_esdhc_be_ops ,
} ;
static const struct sdhci_pltfm_data sdhci_esdhc_le_pdata = {
2016-12-26 17:40:44 +08:00
. quirks = ESDHC_DEFAULT_QUIRKS |
SDHCI_QUIRK_NO_CARD_NO_RESET |
SDHCI_QUIRK_NO_ENDATTR_IN_NOPDESC ,
2015-10-08 18:36:36 +08:00
. ops = & sdhci_esdhc_le_ops ,
2009-12-17 15:27:20 -08:00
} ;
2011-05-27 23:48:14 +08:00
2016-11-09 11:14:12 +08:00
static struct soc_device_attribute soc_incorrect_hostver [ ] = {
{ . family = " QorIQ T4240 " , . revision = " 1.0 " , } ,
{ . family = " QorIQ T4240 " , . revision = " 2.0 " , } ,
{ } ,
} ;
2018-11-23 11:15:35 +08:00
static struct soc_device_attribute soc_fixup_sdhc_clkdivs [ ] = {
{ . family = " QorIQ LX2160A " , . revision = " 1.0 " , } ,
2019-03-07 02:32:44 +00:00
{ . family = " QorIQ LX2160A " , . revision = " 2.0 " , } ,
2019-08-14 15:26:49 +08:00
{ . family = " QorIQ LS1028A " , . revision = " 1.0 " , } ,
2018-11-23 11:15:35 +08:00
{ } ,
} ;
2018-11-23 11:15:37 +08:00
static struct soc_device_attribute soc_unreliable_pulse_detection [ ] = {
{ . family = " QorIQ LX2160A " , . revision = " 1.0 " , } ,
2020-11-10 15:13:14 +08:00
{ . family = " QorIQ LX2160A " , . revision = " 2.0 " , } ,
{ . family = " QorIQ LS1028A " , . revision = " 1.0 " , } ,
2018-11-23 11:15:37 +08:00
{ } ,
} ;
2015-10-08 18:36:36 +08:00
static void esdhc_init ( struct platform_device * pdev , struct sdhci_host * host )
{
2018-06-25 16:46:24 +08:00
const struct of_device_id * match ;
2015-10-08 18:36:36 +08:00
struct sdhci_pltfm_host * pltfm_host ;
struct sdhci_esdhc * esdhc ;
2017-04-20 16:14:40 +08:00
struct device_node * np ;
struct clk * clk ;
u32 val ;
2015-10-08 18:36:36 +08:00
u16 host_ver ;
pltfm_host = sdhci_priv ( host ) ;
2016-02-16 21:08:26 +08:00
esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
2015-10-08 18:36:36 +08:00
host_ver = sdhci_readw ( host , SDHCI_HOST_VERSION ) ;
esdhc - > vendor_ver = ( host_ver & SDHCI_VENDOR_VER_MASK ) > >
SDHCI_VENDOR_VER_SHIFT ;
esdhc - > spec_ver = host_ver & SDHCI_SPEC_VER_MASK ;
2016-11-09 11:14:12 +08:00
if ( soc_device_match ( soc_incorrect_hostver ) )
esdhc - > quirk_incorrect_hostver = true ;
else
esdhc - > quirk_incorrect_hostver = false ;
2017-04-20 16:14:40 +08:00
2018-11-23 11:15:35 +08:00
if ( soc_device_match ( soc_fixup_sdhc_clkdivs ) )
esdhc - > quirk_limited_clk_division = true ;
else
esdhc - > quirk_limited_clk_division = false ;
2018-11-23 11:15:37 +08:00
if ( soc_device_match ( soc_unreliable_pulse_detection ) )
esdhc - > quirk_unreliable_pulse_detection = true ;
else
esdhc - > quirk_unreliable_pulse_detection = false ;
2018-06-25 16:46:24 +08:00
match = of_match_node ( sdhci_esdhc_of_match , pdev - > dev . of_node ) ;
if ( match )
esdhc - > clk_fixup = match - > data ;
2017-04-20 16:14:40 +08:00
np = pdev - > dev . of_node ;
2019-12-19 11:23:35 +08:00
2020-09-03 13:20:29 +12:00
if ( of_device_is_compatible ( np , " fsl,p2020-esdhc " ) ) {
2019-12-19 11:23:35 +08:00
esdhc - > quirk_delay_before_data_reset = true ;
2020-09-03 13:20:29 +12:00
esdhc - > quirk_trans_complete_erratum = true ;
}
2019-12-19 11:23:35 +08:00
2017-04-20 16:14:40 +08:00
clk = of_clk_get ( np , 0 ) ;
if ( ! IS_ERR ( clk ) ) {
/*
* esdhc - > peripheral_clock would be assigned with a value
* which is eSDHC base clock when use periperal clock .
2019-06-14 16:29:54 +08:00
* For some platforms , the clock value got by common clk
* API is peripheral clock while the eSDHC base clock is
* 1 / 2 peripheral clock .
2017-04-20 16:14:40 +08:00
*/
2019-06-14 16:29:54 +08:00
if ( of_device_is_compatible ( np , " fsl,ls1046a-esdhc " ) | |
2019-12-16 17:19:11 +08:00
of_device_is_compatible ( np , " fsl,ls1028a-esdhc " ) | |
of_device_is_compatible ( np , " fsl,ls1088a-esdhc " ) )
2017-04-20 16:14:40 +08:00
esdhc - > peripheral_clock = clk_get_rate ( clk ) / 2 ;
else
esdhc - > peripheral_clock = clk_get_rate ( clk ) ;
clk_put ( clk ) ;
}
2020-09-27 16:23:04 +08:00
esdhc_clock_enable ( host , false ) ;
val = sdhci_readl ( host , ESDHC_DMA_SYSCTL ) ;
/*
* This bit is not able to be reset by SDHCI_RESET_ALL . Need to
* initialize it as 1 or 0 once , to override the different value
* which may be configured in bootloader .
*/
if ( esdhc - > peripheral_clock )
2017-04-20 16:14:40 +08:00
val | = ESDHC_PERIPHERAL_CLK_SEL ;
2020-09-27 16:23:04 +08:00
else
val & = ~ ESDHC_PERIPHERAL_CLK_SEL ;
sdhci_writel ( host , val , ESDHC_DMA_SYSCTL ) ;
esdhc_clock_enable ( host , true ) ;
2015-10-08 18:36:36 +08:00
}
2018-11-23 11:15:34 +08:00
static int esdhc_hs400_prepare_ddr ( struct mmc_host * mmc )
{
esdhc_tuning_block_enable ( mmc_priv ( mmc ) , false ) ;
return 0 ;
}
2012-11-19 13:23:06 -05:00
static int sdhci_esdhc_probe ( struct platform_device * pdev )
2011-05-27 23:48:14 +08:00
{
2013-06-27 12:00:05 -04:00
struct sdhci_host * host ;
2013-07-05 12:48:35 -04:00
struct device_node * np ;
2015-11-25 10:05:37 +08:00
struct sdhci_pltfm_host * pltfm_host ;
struct sdhci_esdhc * esdhc ;
2013-06-27 12:00:05 -04:00
int ret ;
2015-10-08 18:36:36 +08:00
np = pdev - > dev . of_node ;
2016-08-05 10:56:46 +02:00
if ( of_property_read_bool ( np , " little-endian " ) )
2016-02-16 21:08:26 +08:00
host = sdhci_pltfm_init ( pdev , & sdhci_esdhc_le_pdata ,
sizeof ( struct sdhci_esdhc ) ) ;
2015-10-08 18:36:36 +08:00
else
2016-02-16 21:08:26 +08:00
host = sdhci_pltfm_init ( pdev , & sdhci_esdhc_be_pdata ,
sizeof ( struct sdhci_esdhc ) ) ;
2015-10-08 18:36:36 +08:00
2013-06-27 12:00:05 -04:00
if ( IS_ERR ( host ) )
return PTR_ERR ( host ) ;
2017-04-20 16:14:41 +08:00
host - > mmc_host_ops . start_signal_voltage_switch =
esdhc_signal_voltage_switch ;
2017-04-20 16:14:42 +08:00
host - > mmc_host_ops . execute_tuning = esdhc_execute_tuning ;
2018-11-23 11:15:34 +08:00
host - > mmc_host_ops . hs400_prepare_ddr = esdhc_hs400_prepare_ddr ;
2017-04-20 16:14:44 +08:00
host - > tuning_delay = 1 ;
2017-04-20 16:14:41 +08:00
2015-10-08 18:36:36 +08:00
esdhc_init ( pdev , host ) ;
2013-06-27 12:00:05 -04:00
sdhci_get_of_property ( pdev ) ;
2015-11-25 10:05:37 +08:00
pltfm_host = sdhci_priv ( host ) ;
2016-02-16 21:08:26 +08:00
esdhc = sdhci_pltfm_priv ( pltfm_host ) ;
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
if ( soc_device_match ( soc_tuning_erratum_type1 ) )
esdhc - > quirk_tuning_erratum_type1 = true ;
else
esdhc - > quirk_tuning_erratum_type1 = false ;
if ( soc_device_match ( soc_tuning_erratum_type2 ) )
esdhc - > quirk_tuning_erratum_type2 = true ;
2018-08-23 16:48:32 +08:00
else
mmc: sdhci-of-esdhc: fix up erratum A-008171 workaround
A previous patch implemented an incomplete workaround of erratum
A-008171. The complete workaround is as below. This patch is to
implement the complete workaround which uses SW tuning if HW tuning
fails, and retries both HW/SW tuning once with reduced clock if
workaround fails. This is suggested by hardware team, and the patch
had been verified on LS1046A eSDHC + Phison 32G eMMC which could
trigger the erratum.
Workaround:
/* For T1040, T2080, LS1021A, T1023 Rev 1: */
1. Program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO.
2. Program TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
3. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
4. Set SYSCTL2[EXTN] and SYSCTL2[SAMPCLKSEL].
5. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
6. Wait for IRQSTAT[BRR], buffer read ready, to be set.
7. Clear IRQSTAT[BRR].
8. Check SYSCTL2[EXTN] to be cleared.
9. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0: */
1. Read the TBCTL[31:0] register. Write TBCTL[11:8]=4'h8 and wait for
1ms.
2. Read the TBCTL[31:0] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[31:0] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[15:8]-TBSTAT[7:0] > 4*DIV_RATIO or TBSTAT[7:0]-TBSTAT[15:8]
> 4*DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] = 4*DIV_RATIO and
program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1012A Rev1, LS1043A Rev 1.x, LS1046A 1.0: */
1. Read the TBCTL[0:31] register. Write TBCTL[20:23]=4'h8 and wait for
1ms.
2. Read the TBCTL[0:31] register and rewrite again. Wait for 1ms second.
3. Read the TBSTAT[0:31] register twice.
3.1 Reset data lines by setting ESDHCCTL[RSTD] bit.
3.2 Check ESDHCCTL[RSTD] bit.
3.3 If ESDHCCTL[RSTD] is 0, go to step 3.4 else go to step 3.2.
3.4 Write 32'hFFFF_FFFF to IRQSTAT register.
4. if TBSTAT[16:23]-TBSTAT[24:31] > 4*DIV_RATIO or TBSTAT[24:31]-
TBSTAT[16:23] > 4* DIV_RATIO , then program TBPTR[TB_WNDW_END_PTR] =
4*DIV_RATIO and program TBPTR[TB_WNDW_START_PTR] = 8*DIV_RATIO.
/* For LS1080A Rev 1, LS2088A Rev 1.0, LA1575A Rev 1.0 LS1012A Rev1,
* LS1043A Rev 1.x, LS1046A 1.0:
*/
5. else program TBPTR[TB_WNDW_END_PTR] = 3*DIV_RATIO and program
TBPTR[TB_WNDW_START_PTR] = 5*DIV_RATIO.
6. Program the software tuning mode by setting TBCTL[TB_MODE] = 2'h3.
7. Set SYSCTL2[EXTN], wait 1us and SYSCTL2[SAMPCLKSEL].
8. Issue SEND_TUNING_BLK Command (CMD19 for SD, CMD21 for MMC).
9. Wait for IRQSTAT[BRR], buffer read ready, to be set.
10. Clear IRQSTAT[BRR].
11. Check SYSCTL2[EXTN] to be cleared.
12. Check SYSCTL2[SAMPCLKSEL], Sampling Clock Select. It's set value
indicate tuning procedure success, and clear indicate failure.
In case of tuning failure, fixed sampling scheme could be used by
clearing TBCTL[TB_EN].
Fixes: b1f378ab5334 ("mmc: sdhci-of-esdhc: add erratum A008171 support")
Signed-off-by: Yinbo Zhu <yinbo.zhu@nxp.com>
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
Acked-by: Adrian Hunter <adrian.hunter@intel.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
2019-10-09 15:41:40 +08:00
esdhc - > quirk_tuning_erratum_type2 = false ;
2018-08-23 16:48:32 +08:00
2015-11-25 10:05:37 +08:00
if ( esdhc - > vendor_ver = = VENDOR_V_22 )
host - > quirks2 | = SDHCI_QUIRK2_HOST_NO_CMD23 ;
if ( esdhc - > vendor_ver > VENDOR_V_22 )
host - > quirks & = ~ SDHCI_QUIRK_NO_BUSY_IRQ ;
2019-03-11 02:16:40 +00:00
if ( of_find_compatible_node ( NULL , NULL , " fsl,p2020-esdhc " ) ) {
2019-12-16 11:18:42 +08:00
host - > quirks | = SDHCI_QUIRK_RESET_AFTER_REQUEST ;
host - > quirks | = SDHCI_QUIRK_BROKEN_TIMEOUT_VAL ;
2019-03-11 02:16:40 +00:00
}
2019-03-11 02:16:36 +00:00
2015-06-01 13:47:12 +08:00
if ( of_device_is_compatible ( np , " fsl,p5040-esdhc " ) | |
of_device_is_compatible ( np , " fsl,p5020-esdhc " ) | |
of_device_is_compatible ( np , " fsl,p4080-esdhc " ) | |
of_device_is_compatible ( np , " fsl,p1020-esdhc " ) | |
2016-12-26 17:40:44 +08:00
of_device_is_compatible ( np , " fsl,t1040-esdhc " ) )
2015-06-01 13:47:12 +08:00
host - > quirks & = ~ SDHCI_QUIRK_BROKEN_CARD_DETECTION ;
2015-10-08 18:36:57 +08:00
if ( of_device_is_compatible ( np , " fsl,ls1021a-esdhc " ) )
host - > quirks | = SDHCI_QUIRK_BROKEN_TIMEOUT_VAL ;
2019-03-11 02:16:51 +00:00
esdhc - > quirk_ignore_data_inhibit = false ;
2013-07-05 12:48:35 -04:00
if ( of_device_is_compatible ( np , " fsl,p2020-esdhc " ) ) {
/*
* Freescale messed up with P2020 as it has a non - standard
* host control register
*/
host - > quirks2 | = SDHCI_QUIRK2_BROKEN_HOST_CONTROL ;
2019-03-11 02:16:51 +00:00
esdhc - > quirk_ignore_data_inhibit = true ;
2013-07-05 12:48:35 -04:00
}
2013-06-27 12:00:05 -04:00
/* call to generic mmc_of_parse to support additional capabilities */
2014-12-18 10:41:41 +01:00
ret = mmc_of_parse ( host - > mmc ) ;
if ( ret )
goto err ;
2013-08-26 09:19:24 +08:00
mmc_of_parse_voltage ( np , & host - > ocr_mask ) ;
2013-06-27 12:00:05 -04:00
ret = sdhci_add_host ( host ) ;
if ( ret )
2014-12-18 10:41:41 +01:00
goto err ;
2013-06-27 12:00:05 -04:00
2014-12-18 10:41:41 +01:00
return 0 ;
err :
sdhci_pltfm_free ( pdev ) ;
2013-06-27 12:00:05 -04:00
return ret ;
2011-05-27 23:48:14 +08:00
}
static struct platform_driver sdhci_esdhc_driver = {
. driver = {
. name = " sdhci-esdhc " ,
2020-09-03 16:24:36 -07:00
. probe_type = PROBE_PREFER_ASYNCHRONOUS ,
2011-05-27 23:48:14 +08:00
. of_match_table = sdhci_esdhc_of_match ,
2016-07-27 11:01:48 +02:00
. pm = & esdhc_of_dev_pm_ops ,
2011-05-27 23:48:14 +08:00
} ,
. probe = sdhci_esdhc_probe ,
2015-02-27 15:47:31 +08:00
. remove = sdhci_pltfm_unregister ,
2011-05-27 23:48:14 +08:00
} ;
2011-11-26 12:55:43 +08:00
module_platform_driver ( sdhci_esdhc_driver ) ;
2011-05-27 23:48:14 +08:00
MODULE_DESCRIPTION ( " SDHCI OF driver for Freescale MPC eSDHC " ) ;
MODULE_AUTHOR ( " Xiaobo Xie <X.Xie@freescale.com>, "
" Anton Vorontsov <avorontsov@ru.mvista.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;