aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/mmc/host/sdhci-esdhc-imx.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/mmc/host/sdhci-esdhc-imx.c')
-rw-r--r--drivers/mmc/host/sdhci-esdhc-imx.c44
1 files changed, 37 insertions, 7 deletions
diff --git a/drivers/mmc/host/sdhci-esdhc-imx.c b/drivers/mmc/host/sdhci-esdhc-imx.c
index 3b5248567973..67a2da961bb5 100644
--- a/drivers/mmc/host/sdhci-esdhc-imx.c
+++ b/drivers/mmc/host/sdhci-esdhc-imx.c
@@ -16,6 +16,7 @@
16#include <linux/err.h> 16#include <linux/err.h>
17#include <linux/clk.h> 17#include <linux/clk.h>
18#include <linux/gpio.h> 18#include <linux/gpio.h>
19#include <linux/slab.h>
19#include <linux/mmc/host.h> 20#include <linux/mmc/host.h>
20#include <linux/mmc/sdhci-pltfm.h> 21#include <linux/mmc/sdhci-pltfm.h>
21#include <mach/hardware.h> 22#include <mach/hardware.h>
@@ -24,6 +25,13 @@
24#include "sdhci-pltfm.h" 25#include "sdhci-pltfm.h"
25#include "sdhci-esdhc.h" 26#include "sdhci-esdhc.h"
26 27
28#define ESDHC_FLAG_GPIO_FOR_CD_WP (1 << 0)
29
30struct pltfm_imx_data {
31 int flags;
32 u32 scratchpad;
33};
34
27static inline void esdhc_clrset_le(struct sdhci_host *host, u32 mask, u32 val, int reg) 35static inline void esdhc_clrset_le(struct sdhci_host *host, u32 mask, u32 val, int reg)
28{ 36{
29 void __iomem *base = host->ioaddr + (reg & ~0x3); 37 void __iomem *base = host->ioaddr + (reg & ~0x3);
@@ -34,10 +42,14 @@ static inline void esdhc_clrset_le(struct sdhci_host *host, u32 mask, u32 val, i
34 42
35static u32 esdhc_readl_le(struct sdhci_host *host, int reg) 43static u32 esdhc_readl_le(struct sdhci_host *host, int reg)
36{ 44{
45 struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
46 struct pltfm_imx_data *imx_data = pltfm_host->priv;
47
37 /* fake CARD_PRESENT flag on mx25/35 */ 48 /* fake CARD_PRESENT flag on mx25/35 */
38 u32 val = readl(host->ioaddr + reg); 49 u32 val = readl(host->ioaddr + reg);
39 50
40 if (unlikely(reg == SDHCI_PRESENT_STATE)) { 51 if (unlikely((reg == SDHCI_PRESENT_STATE)
52 && (imx_data->flags & ESDHC_FLAG_GPIO_FOR_CD_WP))) {
41 struct esdhc_platform_data *boarddata = 53 struct esdhc_platform_data *boarddata =
42 host->mmc->parent->platform_data; 54 host->mmc->parent->platform_data;
43 55
@@ -55,7 +67,11 @@ static u32 esdhc_readl_le(struct sdhci_host *host, int reg)
55 67
56static void esdhc_writel_le(struct sdhci_host *host, u32 val, int reg) 68static void esdhc_writel_le(struct sdhci_host *host, u32 val, int reg)
57{ 69{
58 if (unlikely(reg == SDHCI_INT_ENABLE || reg == SDHCI_SIGNAL_ENABLE)) 70 struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
71 struct pltfm_imx_data *imx_data = pltfm_host->priv;
72
73 if (unlikely((reg == SDHCI_INT_ENABLE || reg == SDHCI_SIGNAL_ENABLE)
74 && (imx_data->flags & ESDHC_FLAG_GPIO_FOR_CD_WP)))
59 /* 75 /*
60 * these interrupts won't work with a custom card_detect gpio 76 * these interrupts won't work with a custom card_detect gpio
61 * (only applied to mx25/35) 77 * (only applied to mx25/35)
@@ -76,6 +92,7 @@ static u16 esdhc_readw_le(struct sdhci_host *host, int reg)
76static void esdhc_writew_le(struct sdhci_host *host, u16 val, int reg) 92static void esdhc_writew_le(struct sdhci_host *host, u16 val, int reg)
77{ 93{
78 struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host); 94 struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
95 struct pltfm_imx_data *imx_data = pltfm_host->priv;
79 96
80 switch (reg) { 97 switch (reg) {
81 case SDHCI_TRANSFER_MODE: 98 case SDHCI_TRANSFER_MODE:
@@ -83,10 +100,10 @@ static void esdhc_writew_le(struct sdhci_host *host, u16 val, int reg)
83 * Postpone this write, we must do it together with a 100 * Postpone this write, we must do it together with a
84 * command write that is down below. 101 * command write that is down below.
85 */ 102 */
86 pltfm_host->scratchpad = val; 103 imx_data->scratchpad = val;
87 return; 104 return;
88 case SDHCI_COMMAND: 105 case SDHCI_COMMAND:
89 writel(val << 16 | pltfm_host->scratchpad, 106 writel(val << 16 | imx_data->scratchpad,
90 host->ioaddr + SDHCI_TRANSFER_MODE); 107 host->ioaddr + SDHCI_TRANSFER_MODE);
91 return; 108 return;
92 case SDHCI_BLOCK_SIZE: 109 case SDHCI_BLOCK_SIZE:
@@ -146,7 +163,9 @@ static unsigned int esdhc_pltfm_get_ro(struct sdhci_host *host)
146} 163}
147 164
148static struct sdhci_ops sdhci_esdhc_ops = { 165static struct sdhci_ops sdhci_esdhc_ops = {
166 .read_l = esdhc_readl_le,
149 .read_w = esdhc_readw_le, 167 .read_w = esdhc_readw_le,
168 .write_l = esdhc_writel_le,
150 .write_w = esdhc_writew_le, 169 .write_w = esdhc_writew_le,
151 .write_b = esdhc_writeb_le, 170 .write_b = esdhc_writeb_le,
152 .set_clock = esdhc_set_clock, 171 .set_clock = esdhc_set_clock,
@@ -168,6 +187,7 @@ static int esdhc_pltfm_init(struct sdhci_host *host, struct sdhci_pltfm_data *pd
168 struct esdhc_platform_data *boarddata = host->mmc->parent->platform_data; 187 struct esdhc_platform_data *boarddata = host->mmc->parent->platform_data;
169 struct clk *clk; 188 struct clk *clk;
170 int err; 189 int err;
190 struct pltfm_imx_data *imx_data;
171 191
172 clk = clk_get(mmc_dev(host->mmc), NULL); 192 clk = clk_get(mmc_dev(host->mmc), NULL);
173 if (IS_ERR(clk)) { 193 if (IS_ERR(clk)) {
@@ -177,7 +197,15 @@ static int esdhc_pltfm_init(struct sdhci_host *host, struct sdhci_pltfm_data *pd
177 clk_enable(clk); 197 clk_enable(clk);
178 pltfm_host->clk = clk; 198 pltfm_host->clk = clk;
179 199
180 if (cpu_is_mx35() || cpu_is_mx51()) 200 imx_data = kzalloc(sizeof(struct pltfm_imx_data), GFP_KERNEL);
201 if (!imx_data) {
202 clk_disable(pltfm_host->clk);
203 clk_put(pltfm_host->clk);
204 return -ENOMEM;
205 }
206 pltfm_host->priv = imx_data;
207
208 if (!cpu_is_mx25())
181 host->quirks |= SDHCI_QUIRK_BROKEN_TIMEOUT_VAL; 209 host->quirks |= SDHCI_QUIRK_BROKEN_TIMEOUT_VAL;
182 210
183 if (cpu_is_mx25() || cpu_is_mx35()) { 211 if (cpu_is_mx25() || cpu_is_mx35()) {
@@ -214,8 +242,7 @@ static int esdhc_pltfm_init(struct sdhci_host *host, struct sdhci_pltfm_data *pd
214 goto no_card_detect_irq; 242 goto no_card_detect_irq;
215 } 243 }
216 244
217 sdhci_esdhc_ops.write_l = esdhc_writel_le; 245 imx_data->flags |= ESDHC_FLAG_GPIO_FOR_CD_WP;
218 sdhci_esdhc_ops.read_l = esdhc_readl_le;
219 /* Now we have a working card_detect again */ 246 /* Now we have a working card_detect again */
220 host->quirks &= ~SDHCI_QUIRK_BROKEN_CARD_DETECTION; 247 host->quirks &= ~SDHCI_QUIRK_BROKEN_CARD_DETECTION;
221 } 248 }
@@ -227,6 +254,7 @@ static int esdhc_pltfm_init(struct sdhci_host *host, struct sdhci_pltfm_data *pd
227 no_card_detect_pin: 254 no_card_detect_pin:
228 boarddata->cd_gpio = err; 255 boarddata->cd_gpio = err;
229 not_supported: 256 not_supported:
257 kfree(imx_data);
230 return 0; 258 return 0;
231} 259}
232 260
@@ -234,6 +262,7 @@ static void esdhc_pltfm_exit(struct sdhci_host *host)
234{ 262{
235 struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host); 263 struct sdhci_pltfm_host *pltfm_host = sdhci_priv(host);
236 struct esdhc_platform_data *boarddata = host->mmc->parent->platform_data; 264 struct esdhc_platform_data *boarddata = host->mmc->parent->platform_data;
265 struct pltfm_imx_data *imx_data = pltfm_host->priv;
237 266
238 if (boarddata && gpio_is_valid(boarddata->wp_gpio)) 267 if (boarddata && gpio_is_valid(boarddata->wp_gpio))
239 gpio_free(boarddata->wp_gpio); 268 gpio_free(boarddata->wp_gpio);
@@ -247,6 +276,7 @@ static void esdhc_pltfm_exit(struct sdhci_host *host)
247 276
248 clk_disable(pltfm_host->clk); 277 clk_disable(pltfm_host->clk);
249 clk_put(pltfm_host->clk); 278 clk_put(pltfm_host->clk);
279 kfree(imx_data);
250} 280}
251 281
252struct sdhci_pltfm_data sdhci_esdhc_imx_pdata = { 282struct sdhci_pltfm_data sdhci_esdhc_imx_pdata = {