#include <linux/platform_device.h>
 #include <linux/pm_domain.h>
 #include <linux/mmc/host.h>
+#include <linux/mmc/mmc.h>
 #include <linux/mmc/slot-gpio.h>
 #include <linux/mfd/tmio.h>
 #include <linux/sh_dma.h>
 #define SDHI_VER_GEN3_SD       0xcc10
 #define SDHI_VER_GEN3_SDMMC    0xcd10
 
+#define SDHI_GEN3_MMC0_ADDR    0xee140000
+
 static void renesas_sdhi_sdbuf_width(struct tmio_mmc_host *host, int width)
 {
        u32 val;
 #define SH_MOBILE_SDHI_SCC_RVSREQ      0x00A
 #define SH_MOBILE_SDHI_SCC_SMPCMP       0x00C
 #define SH_MOBILE_SDHI_SCC_TMPPORT2    0x00E
+#define SH_MOBILE_SDHI_SCC_TMPPORT3    0x014
+#define SH_MOBILE_SDHI_SCC_TMPPORT4    0x016
+#define SH_MOBILE_SDHI_SCC_TMPPORT5    0x018
+#define SH_MOBILE_SDHI_SCC_TMPPORT6    0x01A
+#define SH_MOBILE_SDHI_SCC_TMPPORT7    0x01C
 
 #define SH_MOBILE_SDHI_SCC_DTCNTL_TAPEN                BIT(0)
 #define SH_MOBILE_SDHI_SCC_DTCNTL_TAPNUM_SHIFT 16
 #define SH_MOBILE_SDHI_SCC_TMPPORT2_HS400OSEL  BIT(4)
 #define SH_MOBILE_SDHI_SCC_TMPPORT2_HS400EN    BIT(31)
 
+/* Definitions for values the SH_MOBILE_SDHI_SCC_TMPPORT4 register */
+#define SH_MOBILE_SDHI_SCC_TMPPORT4_DLL_ACC_START      BIT(0)
+
+/* Definitions for values the SH_MOBILE_SDHI_SCC_TMPPORT5 register */
+#define SH_MOBILE_SDHI_SCC_TMPPORT5_DLL_RW_SEL_R       BIT(8)
+#define SH_MOBILE_SDHI_SCC_TMPPORT5_DLL_RW_SEL_W       (0 << 8)
+#define SH_MOBILE_SDHI_SCC_TMPPORT5_DLL_ADR_MASK       0x3F
+
+/* Definitions for values the SH_MOBILE_SDHI_SCC register */
+#define SH_MOBILE_SDHI_SCC_TMPPORT_DISABLE_WP_CODE     0xa5000000
+#define SH_MOBILE_SDHI_SCC_TMPPORT_CALIB_CODE_MASK     0x1f
+#define SH_MOBILE_SDHI_SCC_TMPPORT_MANUAL_MODE         BIT(7)
+
+static const u8 r8a7796_es13_calib_table[2][SDHI_CALIB_TABLE_MAX] = {
+       { 3,  3,  3,  3,  3,  3,  3,  4,  4,  5,  6,  7,  8,  9, 10, 15,
+        16, 16, 16, 16, 16, 16, 17, 18, 18, 19, 20, 21, 22, 23, 24, 25 },
+       { 5,  5,  5,  5,  5,  5,  5,  5,  5,  5,  5,  5,  6,  7,  8, 11,
+        12, 17, 18, 18, 18, 18, 18, 18, 18, 19, 20, 21, 22, 23, 25, 25 }
+};
+
+static const u8 r8a77965_calib_table[2][SDHI_CALIB_TABLE_MAX] = {
+       { 1,  2,  6,  6,  7,  8,  9, 10, 11, 12, 13, 14, 15, 15, 15, 16,
+        17, 18, 19, 20, 21, 22, 23, 24, 25, 25, 26, 27, 28, 29, 30, 31 },
+       { 2,  3,  4,  4,  5,  6,  7,  9, 10, 11, 12, 13, 14, 15, 16, 17,
+        17, 17, 20, 21, 22, 23, 24, 25, 27, 28, 29, 30, 31, 31, 31, 31 }
+};
+
+static const u8 r8a77990_calib_table[2][SDHI_CALIB_TABLE_MAX] = {
+       { 0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,
+         0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0,  0 },
+       { 0,  0,  0,  1,  2,  3,  3,  4,  4,  4,  5,  5,  6,  8,  9, 10,
+        11, 12, 13, 15, 16, 17, 17, 18, 18, 19, 20, 22, 24, 25, 26, 26 }
+};
+
 static inline u32 sd_scc_read32(struct tmio_mmc_host *host,
                                struct renesas_sdhi *priv, int addr)
 {
 
        sd_ctrl_write16(host, CTL_SD_CARD_CLK_CTL, CLK_CTL_SCLKEN |
                        sd_ctrl_read16(host, CTL_SD_CARD_CLK_CTL));
+
+       if (priv->adjust_hs400_calib_table)
+               priv->needs_adjust_hs400 = true;
 }
 
 static void renesas_sdhi_reset_scc(struct tmio_mmc_host *host,
                        sd_ctrl_read16(host, CTL_SD_CARD_CLK_CTL));
 }
 
+static u32 sd_scc_tmpport_read32(struct tmio_mmc_host *host,
+                                struct renesas_sdhi *priv, u32 addr)
+{
+       /* read mode */
+       sd_scc_write32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT5,
+                      SH_MOBILE_SDHI_SCC_TMPPORT5_DLL_RW_SEL_R |
+                      (SH_MOBILE_SDHI_SCC_TMPPORT5_DLL_ADR_MASK & addr));
+
+       /* access start and stop */
+       sd_scc_write32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT4,
+                      SH_MOBILE_SDHI_SCC_TMPPORT4_DLL_ACC_START);
+       sd_scc_write32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT4, 0);
+
+       return sd_scc_read32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT7);
+}
+
+static void sd_scc_tmpport_write32(struct tmio_mmc_host *host,
+                                  struct renesas_sdhi *priv, u32 addr, u32 val)
+{
+       /* write mode */
+       sd_scc_write32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT5,
+                      SH_MOBILE_SDHI_SCC_TMPPORT5_DLL_RW_SEL_W |
+                      (SH_MOBILE_SDHI_SCC_TMPPORT5_DLL_ADR_MASK & addr));
+
+       sd_scc_write32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT6, val);
+
+       /* access start and stop */
+       sd_scc_write32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT4,
+                      SH_MOBILE_SDHI_SCC_TMPPORT4_DLL_ACC_START);
+       sd_scc_write32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT4, 0);
+}
+
+static void renesas_sdhi_adjust_hs400_mode_enable(struct tmio_mmc_host *host)
+{
+       struct renesas_sdhi *priv = host_to_priv(host);
+       u32 calib_code;
+
+       /* disable write protect */
+       sd_scc_tmpport_write32(host, priv, 0x00,
+                              SH_MOBILE_SDHI_SCC_TMPPORT_DISABLE_WP_CODE);
+       /* read calibration code and adjust */
+       calib_code = sd_scc_tmpport_read32(host, priv, 0x26);
+       calib_code &= SH_MOBILE_SDHI_SCC_TMPPORT_CALIB_CODE_MASK;
+
+       sd_scc_tmpport_write32(host, priv, 0x22,
+                              SH_MOBILE_SDHI_SCC_TMPPORT_MANUAL_MODE |
+                              priv->adjust_hs400_calib_table[calib_code]);
+
+       /* set offset value to TMPPORT3, hardcoded to OFFSET0 (= 0x3) for now */
+       sd_scc_write32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT3, 0x3);
+
+       /* adjustment done, clear flag */
+       priv->needs_adjust_hs400 = false;
+}
+
+static void renesas_sdhi_adjust_hs400_mode_disable(struct tmio_mmc_host *host)
+{
+       struct renesas_sdhi *priv = host_to_priv(host);
+
+       /* disable write protect */
+       sd_scc_tmpport_write32(host, priv, 0x00,
+                              SH_MOBILE_SDHI_SCC_TMPPORT_DISABLE_WP_CODE);
+       /* disable manual calibration */
+       sd_scc_tmpport_write32(host, priv, 0x22, 0);
+       /* clear offset value of TMPPORT3 */
+       sd_scc_write32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT3, 0);
+}
+
 static void renesas_sdhi_reset_hs400_mode(struct tmio_mmc_host *host,
                                          struct renesas_sdhi *priv)
 {
                         SH_MOBILE_SDHI_SCC_TMPPORT2_HS400OSEL) &
                        sd_scc_read32(host, priv, SH_MOBILE_SDHI_SCC_TMPPORT2));
 
+       if (priv->adjust_hs400_calib_table)
+               renesas_sdhi_adjust_hs400_mode_disable(host);
+
        sd_ctrl_write16(host, CTL_SD_CARD_CLK_CTL, CLK_CTL_SCLKEN |
                        sd_ctrl_read16(host, CTL_SD_CARD_CLK_CTL));
 }
 
        renesas_sdhi_reset_scc(host, priv);
        renesas_sdhi_reset_hs400_mode(host, priv);
+       priv->needs_adjust_hs400 = false;
 
        sd_ctrl_write16(host, CTL_SD_CARD_CLK_CTL, CLK_CTL_SCLKEN |
                        sd_ctrl_read16(host, CTL_SD_CARD_CLK_CTL));
        return blk_size;
 }
 
+static void renesas_sdhi_fixup_request(struct tmio_mmc_host *host, struct mmc_request *mrq)
+{
+       struct renesas_sdhi *priv = host_to_priv(host);
+
+       if (priv->needs_adjust_hs400 && mrq->cmd->opcode == MMC_SEND_STATUS)
+               renesas_sdhi_adjust_hs400_mode_enable(host);
+}
 static void renesas_sdhi_enable_dma(struct tmio_mmc_host *host, bool enable)
 {
        /* Iff regs are 8 byte apart, sdbuf is 64 bit. Otherwise always 32. */
        .hs400_bad_taps = BIT(2) | BIT(3) | BIT(6) | BIT(7),
 };
 
+static const struct renesas_sdhi_quirks sdhi_quirks_r8a7796_es13 = {
+       .hs400_4taps = true,
+       .hs400_bad_taps = BIT(2) | BIT(3) | BIT(6) | BIT(7),
+       .hs400_calib_table = r8a7796_es13_calib_table,
+};
+
+static const struct renesas_sdhi_quirks sdhi_quirks_r8a77965 = {
+       .hs400_bad_taps = BIT(2) | BIT(3) | BIT(6) | BIT(7),
+       .hs400_calib_table = r8a77965_calib_table,
+};
+
+static const struct renesas_sdhi_quirks sdhi_quirks_r8a77990 = {
+       .hs400_calib_table = r8a77990_calib_table,
+};
+
 /*
  * Note for r8a7796 / r8a774a1: we can't distinguish ES1.1 and 1.2 as of now.
  * So, we want to treat them equally and only have a match for ES1.2 to enforce
        { .soc_id = "r8a7795", .revision = "ES2.0", .data = &sdhi_quirks_4tap },
        { .soc_id = "r8a7795", .revision = "ES3.*", .data = &sdhi_quirks_bad_taps2367 },
        { .soc_id = "r8a7796", .revision = "ES1.[012]", .data = &sdhi_quirks_4tap_nohs400 },
-       { .soc_id = "r8a7796", .revision = "ES1.*", .data = &sdhi_quirks_4tap },
+       { .soc_id = "r8a7796", .revision = "ES1.*", .data = &sdhi_quirks_r8a7796_es13 },
        { .soc_id = "r8a7796", .revision = "ES3.*", .data = &sdhi_quirks_bad_taps1357 },
-       { .soc_id = "r8a77965", .data = &sdhi_quirks_bad_taps2367 },
+       { .soc_id = "r8a77965", .data = &sdhi_quirks_r8a77965 },
        { .soc_id = "r8a77980", .data = &sdhi_quirks_nohs400 },
+       { .soc_id = "r8a77990", .data = &sdhi_quirks_r8a77990 },
        { /* Sentinel. */ },
 };
 
        if (ver == SDHI_VER_GEN2_SDR50)
                mmc_data->flags &= ~TMIO_MMC_HAVE_CBSY;
 
+       if (ver == SDHI_VER_GEN3_SDMMC && quirks && quirks->hs400_calib_table) {
+               host->fixup_request = renesas_sdhi_fixup_request;
+               priv->adjust_hs400_calib_table = *(
+                       res->start == SDHI_GEN3_MMC0_ADDR ?
+                       quirks->hs400_calib_table :
+                       quirks->hs400_calib_table + 1);
+       }
+
        ret = tmio_mmc_host_probe(host);
        if (ret < 0)
                goto edisclk;