Commit 8204536207e99b3bbbd13d936aaa3b18449b9c00
Committed by
Artem Bityutskiy
1 parent
031da73fca
Exists in
smarc-l5.0.0_1.0.0-ga
and in
5 other branches
mtd: nand/gpio: use io{read,write}*_rep accessors
The {read,write}s{b,w,l} operations are not defined by all architectures and are being removed from the asm-generic/io.h interface. This patch replaces the usage of these string functions in the mtd gpio accessors with io{read,write}{8,16,32}_rep calls instead. Signed-off-by: Matthew Leach <matthew@mattleach.net> Signed-off-by: Will Deacon <will.deacon@arm.com> Signed-off-by: Artem Bityutskiy <artem.bityutskiy@linux.intel.com>
Showing 1 changed file with 4 additions and 4 deletions Inline Diff
drivers/mtd/nand/gpio.c
1 | /* | 1 | /* |
2 | * drivers/mtd/nand/gpio.c | 2 | * drivers/mtd/nand/gpio.c |
3 | * | 3 | * |
4 | * Updated, and converted to generic GPIO based driver by Russell King. | 4 | * Updated, and converted to generic GPIO based driver by Russell King. |
5 | * | 5 | * |
6 | * Written by Ben Dooks <ben@simtec.co.uk> | 6 | * Written by Ben Dooks <ben@simtec.co.uk> |
7 | * Based on 2.4 version by Mark Whittaker | 7 | * Based on 2.4 version by Mark Whittaker |
8 | * | 8 | * |
9 | * © 2004 Simtec Electronics | 9 | * © 2004 Simtec Electronics |
10 | * | 10 | * |
11 | * Device driver for NAND connected via GPIO | 11 | * Device driver for NAND connected via GPIO |
12 | * | 12 | * |
13 | * This program is free software; you can redistribute it and/or modify | 13 | * This program is free software; you can redistribute it and/or modify |
14 | * it under the terms of the GNU General Public License version 2 as | 14 | * it under the terms of the GNU General Public License version 2 as |
15 | * published by the Free Software Foundation. | 15 | * published by the Free Software Foundation. |
16 | * | 16 | * |
17 | */ | 17 | */ |
18 | 18 | ||
19 | #include <linux/kernel.h> | 19 | #include <linux/kernel.h> |
20 | #include <linux/init.h> | 20 | #include <linux/init.h> |
21 | #include <linux/slab.h> | 21 | #include <linux/slab.h> |
22 | #include <linux/module.h> | 22 | #include <linux/module.h> |
23 | #include <linux/platform_device.h> | 23 | #include <linux/platform_device.h> |
24 | #include <linux/gpio.h> | 24 | #include <linux/gpio.h> |
25 | #include <linux/io.h> | 25 | #include <linux/io.h> |
26 | #include <linux/mtd/mtd.h> | 26 | #include <linux/mtd/mtd.h> |
27 | #include <linux/mtd/nand.h> | 27 | #include <linux/mtd/nand.h> |
28 | #include <linux/mtd/partitions.h> | 28 | #include <linux/mtd/partitions.h> |
29 | #include <linux/mtd/nand-gpio.h> | 29 | #include <linux/mtd/nand-gpio.h> |
30 | #include <linux/of.h> | 30 | #include <linux/of.h> |
31 | #include <linux/of_address.h> | 31 | #include <linux/of_address.h> |
32 | #include <linux/of_gpio.h> | 32 | #include <linux/of_gpio.h> |
33 | 33 | ||
34 | struct gpiomtd { | 34 | struct gpiomtd { |
35 | void __iomem *io_sync; | 35 | void __iomem *io_sync; |
36 | struct mtd_info mtd_info; | 36 | struct mtd_info mtd_info; |
37 | struct nand_chip nand_chip; | 37 | struct nand_chip nand_chip; |
38 | struct gpio_nand_platdata plat; | 38 | struct gpio_nand_platdata plat; |
39 | }; | 39 | }; |
40 | 40 | ||
41 | #define gpio_nand_getpriv(x) container_of(x, struct gpiomtd, mtd_info) | 41 | #define gpio_nand_getpriv(x) container_of(x, struct gpiomtd, mtd_info) |
42 | 42 | ||
43 | 43 | ||
44 | #ifdef CONFIG_ARM | 44 | #ifdef CONFIG_ARM |
45 | /* gpio_nand_dosync() | 45 | /* gpio_nand_dosync() |
46 | * | 46 | * |
47 | * Make sure the GPIO state changes occur in-order with writes to NAND | 47 | * Make sure the GPIO state changes occur in-order with writes to NAND |
48 | * memory region. | 48 | * memory region. |
49 | * Needed on PXA due to bus-reordering within the SoC itself (see section on | 49 | * Needed on PXA due to bus-reordering within the SoC itself (see section on |
50 | * I/O ordering in PXA manual (section 2.3, p35) | 50 | * I/O ordering in PXA manual (section 2.3, p35) |
51 | */ | 51 | */ |
52 | static void gpio_nand_dosync(struct gpiomtd *gpiomtd) | 52 | static void gpio_nand_dosync(struct gpiomtd *gpiomtd) |
53 | { | 53 | { |
54 | unsigned long tmp; | 54 | unsigned long tmp; |
55 | 55 | ||
56 | if (gpiomtd->io_sync) { | 56 | if (gpiomtd->io_sync) { |
57 | /* | 57 | /* |
58 | * Linux memory barriers don't cater for what's required here. | 58 | * Linux memory barriers don't cater for what's required here. |
59 | * What's required is what's here - a read from a separate | 59 | * What's required is what's here - a read from a separate |
60 | * region with a dependency on that read. | 60 | * region with a dependency on that read. |
61 | */ | 61 | */ |
62 | tmp = readl(gpiomtd->io_sync); | 62 | tmp = readl(gpiomtd->io_sync); |
63 | asm volatile("mov %1, %0\n" : "=r" (tmp) : "r" (tmp)); | 63 | asm volatile("mov %1, %0\n" : "=r" (tmp) : "r" (tmp)); |
64 | } | 64 | } |
65 | } | 65 | } |
66 | #else | 66 | #else |
67 | static inline void gpio_nand_dosync(struct gpiomtd *gpiomtd) {} | 67 | static inline void gpio_nand_dosync(struct gpiomtd *gpiomtd) {} |
68 | #endif | 68 | #endif |
69 | 69 | ||
70 | static void gpio_nand_cmd_ctrl(struct mtd_info *mtd, int cmd, unsigned int ctrl) | 70 | static void gpio_nand_cmd_ctrl(struct mtd_info *mtd, int cmd, unsigned int ctrl) |
71 | { | 71 | { |
72 | struct gpiomtd *gpiomtd = gpio_nand_getpriv(mtd); | 72 | struct gpiomtd *gpiomtd = gpio_nand_getpriv(mtd); |
73 | 73 | ||
74 | gpio_nand_dosync(gpiomtd); | 74 | gpio_nand_dosync(gpiomtd); |
75 | 75 | ||
76 | if (ctrl & NAND_CTRL_CHANGE) { | 76 | if (ctrl & NAND_CTRL_CHANGE) { |
77 | gpio_set_value(gpiomtd->plat.gpio_nce, !(ctrl & NAND_NCE)); | 77 | gpio_set_value(gpiomtd->plat.gpio_nce, !(ctrl & NAND_NCE)); |
78 | gpio_set_value(gpiomtd->plat.gpio_cle, !!(ctrl & NAND_CLE)); | 78 | gpio_set_value(gpiomtd->plat.gpio_cle, !!(ctrl & NAND_CLE)); |
79 | gpio_set_value(gpiomtd->plat.gpio_ale, !!(ctrl & NAND_ALE)); | 79 | gpio_set_value(gpiomtd->plat.gpio_ale, !!(ctrl & NAND_ALE)); |
80 | gpio_nand_dosync(gpiomtd); | 80 | gpio_nand_dosync(gpiomtd); |
81 | } | 81 | } |
82 | if (cmd == NAND_CMD_NONE) | 82 | if (cmd == NAND_CMD_NONE) |
83 | return; | 83 | return; |
84 | 84 | ||
85 | writeb(cmd, gpiomtd->nand_chip.IO_ADDR_W); | 85 | writeb(cmd, gpiomtd->nand_chip.IO_ADDR_W); |
86 | gpio_nand_dosync(gpiomtd); | 86 | gpio_nand_dosync(gpiomtd); |
87 | } | 87 | } |
88 | 88 | ||
89 | static void gpio_nand_writebuf(struct mtd_info *mtd, const u_char *buf, int len) | 89 | static void gpio_nand_writebuf(struct mtd_info *mtd, const u_char *buf, int len) |
90 | { | 90 | { |
91 | struct nand_chip *this = mtd->priv; | 91 | struct nand_chip *this = mtd->priv; |
92 | 92 | ||
93 | writesb(this->IO_ADDR_W, buf, len); | 93 | iowrite8_rep(this->IO_ADDR_W, buf, len); |
94 | } | 94 | } |
95 | 95 | ||
96 | static void gpio_nand_readbuf(struct mtd_info *mtd, u_char *buf, int len) | 96 | static void gpio_nand_readbuf(struct mtd_info *mtd, u_char *buf, int len) |
97 | { | 97 | { |
98 | struct nand_chip *this = mtd->priv; | 98 | struct nand_chip *this = mtd->priv; |
99 | 99 | ||
100 | readsb(this->IO_ADDR_R, buf, len); | 100 | ioread8_rep(this->IO_ADDR_R, buf, len); |
101 | } | 101 | } |
102 | 102 | ||
103 | static void gpio_nand_writebuf16(struct mtd_info *mtd, const u_char *buf, | 103 | static void gpio_nand_writebuf16(struct mtd_info *mtd, const u_char *buf, |
104 | int len) | 104 | int len) |
105 | { | 105 | { |
106 | struct nand_chip *this = mtd->priv; | 106 | struct nand_chip *this = mtd->priv; |
107 | 107 | ||
108 | if (IS_ALIGNED((unsigned long)buf, 2)) { | 108 | if (IS_ALIGNED((unsigned long)buf, 2)) { |
109 | writesw(this->IO_ADDR_W, buf, len>>1); | 109 | iowrite16_rep(this->IO_ADDR_W, buf, len>>1); |
110 | } else { | 110 | } else { |
111 | int i; | 111 | int i; |
112 | unsigned short *ptr = (unsigned short *)buf; | 112 | unsigned short *ptr = (unsigned short *)buf; |
113 | 113 | ||
114 | for (i = 0; i < len; i += 2, ptr++) | 114 | for (i = 0; i < len; i += 2, ptr++) |
115 | writew(*ptr, this->IO_ADDR_W); | 115 | writew(*ptr, this->IO_ADDR_W); |
116 | } | 116 | } |
117 | } | 117 | } |
118 | 118 | ||
119 | static void gpio_nand_readbuf16(struct mtd_info *mtd, u_char *buf, int len) | 119 | static void gpio_nand_readbuf16(struct mtd_info *mtd, u_char *buf, int len) |
120 | { | 120 | { |
121 | struct nand_chip *this = mtd->priv; | 121 | struct nand_chip *this = mtd->priv; |
122 | 122 | ||
123 | if (IS_ALIGNED((unsigned long)buf, 2)) { | 123 | if (IS_ALIGNED((unsigned long)buf, 2)) { |
124 | readsw(this->IO_ADDR_R, buf, len>>1); | 124 | ioread16_rep(this->IO_ADDR_R, buf, len>>1); |
125 | } else { | 125 | } else { |
126 | int i; | 126 | int i; |
127 | unsigned short *ptr = (unsigned short *)buf; | 127 | unsigned short *ptr = (unsigned short *)buf; |
128 | 128 | ||
129 | for (i = 0; i < len; i += 2, ptr++) | 129 | for (i = 0; i < len; i += 2, ptr++) |
130 | *ptr = readw(this->IO_ADDR_R); | 130 | *ptr = readw(this->IO_ADDR_R); |
131 | } | 131 | } |
132 | } | 132 | } |
133 | 133 | ||
134 | static int gpio_nand_devready(struct mtd_info *mtd) | 134 | static int gpio_nand_devready(struct mtd_info *mtd) |
135 | { | 135 | { |
136 | struct gpiomtd *gpiomtd = gpio_nand_getpriv(mtd); | 136 | struct gpiomtd *gpiomtd = gpio_nand_getpriv(mtd); |
137 | 137 | ||
138 | if (gpio_is_valid(gpiomtd->plat.gpio_rdy)) | 138 | if (gpio_is_valid(gpiomtd->plat.gpio_rdy)) |
139 | return gpio_get_value(gpiomtd->plat.gpio_rdy); | 139 | return gpio_get_value(gpiomtd->plat.gpio_rdy); |
140 | 140 | ||
141 | return 1; | 141 | return 1; |
142 | } | 142 | } |
143 | 143 | ||
144 | #ifdef CONFIG_OF | 144 | #ifdef CONFIG_OF |
145 | static const struct of_device_id gpio_nand_id_table[] = { | 145 | static const struct of_device_id gpio_nand_id_table[] = { |
146 | { .compatible = "gpio-control-nand" }, | 146 | { .compatible = "gpio-control-nand" }, |
147 | {} | 147 | {} |
148 | }; | 148 | }; |
149 | MODULE_DEVICE_TABLE(of, gpio_nand_id_table); | 149 | MODULE_DEVICE_TABLE(of, gpio_nand_id_table); |
150 | 150 | ||
151 | static int gpio_nand_get_config_of(const struct device *dev, | 151 | static int gpio_nand_get_config_of(const struct device *dev, |
152 | struct gpio_nand_platdata *plat) | 152 | struct gpio_nand_platdata *plat) |
153 | { | 153 | { |
154 | u32 val; | 154 | u32 val; |
155 | 155 | ||
156 | if (!of_property_read_u32(dev->of_node, "bank-width", &val)) { | 156 | if (!of_property_read_u32(dev->of_node, "bank-width", &val)) { |
157 | if (val == 2) { | 157 | if (val == 2) { |
158 | plat->options |= NAND_BUSWIDTH_16; | 158 | plat->options |= NAND_BUSWIDTH_16; |
159 | } else if (val != 1) { | 159 | } else if (val != 1) { |
160 | dev_err(dev, "invalid bank-width %u\n", val); | 160 | dev_err(dev, "invalid bank-width %u\n", val); |
161 | return -EINVAL; | 161 | return -EINVAL; |
162 | } | 162 | } |
163 | } | 163 | } |
164 | 164 | ||
165 | plat->gpio_rdy = of_get_gpio(dev->of_node, 0); | 165 | plat->gpio_rdy = of_get_gpio(dev->of_node, 0); |
166 | plat->gpio_nce = of_get_gpio(dev->of_node, 1); | 166 | plat->gpio_nce = of_get_gpio(dev->of_node, 1); |
167 | plat->gpio_ale = of_get_gpio(dev->of_node, 2); | 167 | plat->gpio_ale = of_get_gpio(dev->of_node, 2); |
168 | plat->gpio_cle = of_get_gpio(dev->of_node, 3); | 168 | plat->gpio_cle = of_get_gpio(dev->of_node, 3); |
169 | plat->gpio_nwp = of_get_gpio(dev->of_node, 4); | 169 | plat->gpio_nwp = of_get_gpio(dev->of_node, 4); |
170 | 170 | ||
171 | if (!of_property_read_u32(dev->of_node, "chip-delay", &val)) | 171 | if (!of_property_read_u32(dev->of_node, "chip-delay", &val)) |
172 | plat->chip_delay = val; | 172 | plat->chip_delay = val; |
173 | 173 | ||
174 | return 0; | 174 | return 0; |
175 | } | 175 | } |
176 | 176 | ||
177 | static struct resource *gpio_nand_get_io_sync_of(struct platform_device *pdev) | 177 | static struct resource *gpio_nand_get_io_sync_of(struct platform_device *pdev) |
178 | { | 178 | { |
179 | struct resource *r = devm_kzalloc(&pdev->dev, sizeof(*r), GFP_KERNEL); | 179 | struct resource *r = devm_kzalloc(&pdev->dev, sizeof(*r), GFP_KERNEL); |
180 | u64 addr; | 180 | u64 addr; |
181 | 181 | ||
182 | if (!r || of_property_read_u64(pdev->dev.of_node, | 182 | if (!r || of_property_read_u64(pdev->dev.of_node, |
183 | "gpio-control-nand,io-sync-reg", &addr)) | 183 | "gpio-control-nand,io-sync-reg", &addr)) |
184 | return NULL; | 184 | return NULL; |
185 | 185 | ||
186 | r->start = addr; | 186 | r->start = addr; |
187 | r->end = r->start + 0x3; | 187 | r->end = r->start + 0x3; |
188 | r->flags = IORESOURCE_MEM; | 188 | r->flags = IORESOURCE_MEM; |
189 | 189 | ||
190 | return r; | 190 | return r; |
191 | } | 191 | } |
192 | #else /* CONFIG_OF */ | 192 | #else /* CONFIG_OF */ |
193 | #define gpio_nand_id_table NULL | 193 | #define gpio_nand_id_table NULL |
194 | static inline int gpio_nand_get_config_of(const struct device *dev, | 194 | static inline int gpio_nand_get_config_of(const struct device *dev, |
195 | struct gpio_nand_platdata *plat) | 195 | struct gpio_nand_platdata *plat) |
196 | { | 196 | { |
197 | return -ENOSYS; | 197 | return -ENOSYS; |
198 | } | 198 | } |
199 | 199 | ||
200 | static inline struct resource * | 200 | static inline struct resource * |
201 | gpio_nand_get_io_sync_of(struct platform_device *pdev) | 201 | gpio_nand_get_io_sync_of(struct platform_device *pdev) |
202 | { | 202 | { |
203 | return NULL; | 203 | return NULL; |
204 | } | 204 | } |
205 | #endif /* CONFIG_OF */ | 205 | #endif /* CONFIG_OF */ |
206 | 206 | ||
207 | static inline int gpio_nand_get_config(const struct device *dev, | 207 | static inline int gpio_nand_get_config(const struct device *dev, |
208 | struct gpio_nand_platdata *plat) | 208 | struct gpio_nand_platdata *plat) |
209 | { | 209 | { |
210 | int ret = gpio_nand_get_config_of(dev, plat); | 210 | int ret = gpio_nand_get_config_of(dev, plat); |
211 | 211 | ||
212 | if (!ret) | 212 | if (!ret) |
213 | return ret; | 213 | return ret; |
214 | 214 | ||
215 | if (dev->platform_data) { | 215 | if (dev->platform_data) { |
216 | memcpy(plat, dev->platform_data, sizeof(*plat)); | 216 | memcpy(plat, dev->platform_data, sizeof(*plat)); |
217 | return 0; | 217 | return 0; |
218 | } | 218 | } |
219 | 219 | ||
220 | return -EINVAL; | 220 | return -EINVAL; |
221 | } | 221 | } |
222 | 222 | ||
223 | static inline struct resource * | 223 | static inline struct resource * |
224 | gpio_nand_get_io_sync(struct platform_device *pdev) | 224 | gpio_nand_get_io_sync(struct platform_device *pdev) |
225 | { | 225 | { |
226 | struct resource *r = gpio_nand_get_io_sync_of(pdev); | 226 | struct resource *r = gpio_nand_get_io_sync_of(pdev); |
227 | 227 | ||
228 | if (r) | 228 | if (r) |
229 | return r; | 229 | return r; |
230 | 230 | ||
231 | return platform_get_resource(pdev, IORESOURCE_MEM, 1); | 231 | return platform_get_resource(pdev, IORESOURCE_MEM, 1); |
232 | } | 232 | } |
233 | 233 | ||
234 | static int gpio_nand_remove(struct platform_device *dev) | 234 | static int gpio_nand_remove(struct platform_device *dev) |
235 | { | 235 | { |
236 | struct gpiomtd *gpiomtd = platform_get_drvdata(dev); | 236 | struct gpiomtd *gpiomtd = platform_get_drvdata(dev); |
237 | struct resource *res; | 237 | struct resource *res; |
238 | 238 | ||
239 | nand_release(&gpiomtd->mtd_info); | 239 | nand_release(&gpiomtd->mtd_info); |
240 | 240 | ||
241 | res = gpio_nand_get_io_sync(dev); | 241 | res = gpio_nand_get_io_sync(dev); |
242 | iounmap(gpiomtd->io_sync); | 242 | iounmap(gpiomtd->io_sync); |
243 | if (res) | 243 | if (res) |
244 | release_mem_region(res->start, resource_size(res)); | 244 | release_mem_region(res->start, resource_size(res)); |
245 | 245 | ||
246 | res = platform_get_resource(dev, IORESOURCE_MEM, 0); | 246 | res = platform_get_resource(dev, IORESOURCE_MEM, 0); |
247 | iounmap(gpiomtd->nand_chip.IO_ADDR_R); | 247 | iounmap(gpiomtd->nand_chip.IO_ADDR_R); |
248 | release_mem_region(res->start, resource_size(res)); | 248 | release_mem_region(res->start, resource_size(res)); |
249 | 249 | ||
250 | if (gpio_is_valid(gpiomtd->plat.gpio_nwp)) | 250 | if (gpio_is_valid(gpiomtd->plat.gpio_nwp)) |
251 | gpio_set_value(gpiomtd->plat.gpio_nwp, 0); | 251 | gpio_set_value(gpiomtd->plat.gpio_nwp, 0); |
252 | gpio_set_value(gpiomtd->plat.gpio_nce, 1); | 252 | gpio_set_value(gpiomtd->plat.gpio_nce, 1); |
253 | 253 | ||
254 | gpio_free(gpiomtd->plat.gpio_cle); | 254 | gpio_free(gpiomtd->plat.gpio_cle); |
255 | gpio_free(gpiomtd->plat.gpio_ale); | 255 | gpio_free(gpiomtd->plat.gpio_ale); |
256 | gpio_free(gpiomtd->plat.gpio_nce); | 256 | gpio_free(gpiomtd->plat.gpio_nce); |
257 | if (gpio_is_valid(gpiomtd->plat.gpio_nwp)) | 257 | if (gpio_is_valid(gpiomtd->plat.gpio_nwp)) |
258 | gpio_free(gpiomtd->plat.gpio_nwp); | 258 | gpio_free(gpiomtd->plat.gpio_nwp); |
259 | if (gpio_is_valid(gpiomtd->plat.gpio_rdy)) | 259 | if (gpio_is_valid(gpiomtd->plat.gpio_rdy)) |
260 | gpio_free(gpiomtd->plat.gpio_rdy); | 260 | gpio_free(gpiomtd->plat.gpio_rdy); |
261 | 261 | ||
262 | kfree(gpiomtd); | 262 | kfree(gpiomtd); |
263 | 263 | ||
264 | return 0; | 264 | return 0; |
265 | } | 265 | } |
266 | 266 | ||
267 | static void __iomem *request_and_remap(struct resource *res, size_t size, | 267 | static void __iomem *request_and_remap(struct resource *res, size_t size, |
268 | const char *name, int *err) | 268 | const char *name, int *err) |
269 | { | 269 | { |
270 | void __iomem *ptr; | 270 | void __iomem *ptr; |
271 | 271 | ||
272 | if (!request_mem_region(res->start, resource_size(res), name)) { | 272 | if (!request_mem_region(res->start, resource_size(res), name)) { |
273 | *err = -EBUSY; | 273 | *err = -EBUSY; |
274 | return NULL; | 274 | return NULL; |
275 | } | 275 | } |
276 | 276 | ||
277 | ptr = ioremap(res->start, size); | 277 | ptr = ioremap(res->start, size); |
278 | if (!ptr) { | 278 | if (!ptr) { |
279 | release_mem_region(res->start, resource_size(res)); | 279 | release_mem_region(res->start, resource_size(res)); |
280 | *err = -ENOMEM; | 280 | *err = -ENOMEM; |
281 | } | 281 | } |
282 | return ptr; | 282 | return ptr; |
283 | } | 283 | } |
284 | 284 | ||
285 | static int gpio_nand_probe(struct platform_device *dev) | 285 | static int gpio_nand_probe(struct platform_device *dev) |
286 | { | 286 | { |
287 | struct gpiomtd *gpiomtd; | 287 | struct gpiomtd *gpiomtd; |
288 | struct nand_chip *this; | 288 | struct nand_chip *this; |
289 | struct resource *res0, *res1; | 289 | struct resource *res0, *res1; |
290 | struct mtd_part_parser_data ppdata = {}; | 290 | struct mtd_part_parser_data ppdata = {}; |
291 | int ret = 0; | 291 | int ret = 0; |
292 | 292 | ||
293 | if (!dev->dev.of_node && !dev->dev.platform_data) | 293 | if (!dev->dev.of_node && !dev->dev.platform_data) |
294 | return -EINVAL; | 294 | return -EINVAL; |
295 | 295 | ||
296 | res0 = platform_get_resource(dev, IORESOURCE_MEM, 0); | 296 | res0 = platform_get_resource(dev, IORESOURCE_MEM, 0); |
297 | if (!res0) | 297 | if (!res0) |
298 | return -EINVAL; | 298 | return -EINVAL; |
299 | 299 | ||
300 | gpiomtd = kzalloc(sizeof(*gpiomtd), GFP_KERNEL); | 300 | gpiomtd = kzalloc(sizeof(*gpiomtd), GFP_KERNEL); |
301 | if (gpiomtd == NULL) { | 301 | if (gpiomtd == NULL) { |
302 | dev_err(&dev->dev, "failed to create NAND MTD\n"); | 302 | dev_err(&dev->dev, "failed to create NAND MTD\n"); |
303 | return -ENOMEM; | 303 | return -ENOMEM; |
304 | } | 304 | } |
305 | 305 | ||
306 | this = &gpiomtd->nand_chip; | 306 | this = &gpiomtd->nand_chip; |
307 | this->IO_ADDR_R = request_and_remap(res0, 2, "NAND", &ret); | 307 | this->IO_ADDR_R = request_and_remap(res0, 2, "NAND", &ret); |
308 | if (!this->IO_ADDR_R) { | 308 | if (!this->IO_ADDR_R) { |
309 | dev_err(&dev->dev, "unable to map NAND\n"); | 309 | dev_err(&dev->dev, "unable to map NAND\n"); |
310 | goto err_map; | 310 | goto err_map; |
311 | } | 311 | } |
312 | 312 | ||
313 | res1 = gpio_nand_get_io_sync(dev); | 313 | res1 = gpio_nand_get_io_sync(dev); |
314 | if (res1) { | 314 | if (res1) { |
315 | gpiomtd->io_sync = request_and_remap(res1, 4, "NAND sync", &ret); | 315 | gpiomtd->io_sync = request_and_remap(res1, 4, "NAND sync", &ret); |
316 | if (!gpiomtd->io_sync) { | 316 | if (!gpiomtd->io_sync) { |
317 | dev_err(&dev->dev, "unable to map sync NAND\n"); | 317 | dev_err(&dev->dev, "unable to map sync NAND\n"); |
318 | goto err_sync; | 318 | goto err_sync; |
319 | } | 319 | } |
320 | } | 320 | } |
321 | 321 | ||
322 | ret = gpio_nand_get_config(&dev->dev, &gpiomtd->plat); | 322 | ret = gpio_nand_get_config(&dev->dev, &gpiomtd->plat); |
323 | if (ret) | 323 | if (ret) |
324 | goto err_nce; | 324 | goto err_nce; |
325 | 325 | ||
326 | ret = gpio_request(gpiomtd->plat.gpio_nce, "NAND NCE"); | 326 | ret = gpio_request(gpiomtd->plat.gpio_nce, "NAND NCE"); |
327 | if (ret) | 327 | if (ret) |
328 | goto err_nce; | 328 | goto err_nce; |
329 | gpio_direction_output(gpiomtd->plat.gpio_nce, 1); | 329 | gpio_direction_output(gpiomtd->plat.gpio_nce, 1); |
330 | if (gpio_is_valid(gpiomtd->plat.gpio_nwp)) { | 330 | if (gpio_is_valid(gpiomtd->plat.gpio_nwp)) { |
331 | ret = gpio_request(gpiomtd->plat.gpio_nwp, "NAND NWP"); | 331 | ret = gpio_request(gpiomtd->plat.gpio_nwp, "NAND NWP"); |
332 | if (ret) | 332 | if (ret) |
333 | goto err_nwp; | 333 | goto err_nwp; |
334 | gpio_direction_output(gpiomtd->plat.gpio_nwp, 1); | 334 | gpio_direction_output(gpiomtd->plat.gpio_nwp, 1); |
335 | } | 335 | } |
336 | ret = gpio_request(gpiomtd->plat.gpio_ale, "NAND ALE"); | 336 | ret = gpio_request(gpiomtd->plat.gpio_ale, "NAND ALE"); |
337 | if (ret) | 337 | if (ret) |
338 | goto err_ale; | 338 | goto err_ale; |
339 | gpio_direction_output(gpiomtd->plat.gpio_ale, 0); | 339 | gpio_direction_output(gpiomtd->plat.gpio_ale, 0); |
340 | ret = gpio_request(gpiomtd->plat.gpio_cle, "NAND CLE"); | 340 | ret = gpio_request(gpiomtd->plat.gpio_cle, "NAND CLE"); |
341 | if (ret) | 341 | if (ret) |
342 | goto err_cle; | 342 | goto err_cle; |
343 | gpio_direction_output(gpiomtd->plat.gpio_cle, 0); | 343 | gpio_direction_output(gpiomtd->plat.gpio_cle, 0); |
344 | if (gpio_is_valid(gpiomtd->plat.gpio_rdy)) { | 344 | if (gpio_is_valid(gpiomtd->plat.gpio_rdy)) { |
345 | ret = gpio_request(gpiomtd->plat.gpio_rdy, "NAND RDY"); | 345 | ret = gpio_request(gpiomtd->plat.gpio_rdy, "NAND RDY"); |
346 | if (ret) | 346 | if (ret) |
347 | goto err_rdy; | 347 | goto err_rdy; |
348 | gpio_direction_input(gpiomtd->plat.gpio_rdy); | 348 | gpio_direction_input(gpiomtd->plat.gpio_rdy); |
349 | } | 349 | } |
350 | 350 | ||
351 | 351 | ||
352 | this->IO_ADDR_W = this->IO_ADDR_R; | 352 | this->IO_ADDR_W = this->IO_ADDR_R; |
353 | this->ecc.mode = NAND_ECC_SOFT; | 353 | this->ecc.mode = NAND_ECC_SOFT; |
354 | this->options = gpiomtd->plat.options; | 354 | this->options = gpiomtd->plat.options; |
355 | this->chip_delay = gpiomtd->plat.chip_delay; | 355 | this->chip_delay = gpiomtd->plat.chip_delay; |
356 | 356 | ||
357 | /* install our routines */ | 357 | /* install our routines */ |
358 | this->cmd_ctrl = gpio_nand_cmd_ctrl; | 358 | this->cmd_ctrl = gpio_nand_cmd_ctrl; |
359 | this->dev_ready = gpio_nand_devready; | 359 | this->dev_ready = gpio_nand_devready; |
360 | 360 | ||
361 | if (this->options & NAND_BUSWIDTH_16) { | 361 | if (this->options & NAND_BUSWIDTH_16) { |
362 | this->read_buf = gpio_nand_readbuf16; | 362 | this->read_buf = gpio_nand_readbuf16; |
363 | this->write_buf = gpio_nand_writebuf16; | 363 | this->write_buf = gpio_nand_writebuf16; |
364 | } else { | 364 | } else { |
365 | this->read_buf = gpio_nand_readbuf; | 365 | this->read_buf = gpio_nand_readbuf; |
366 | this->write_buf = gpio_nand_writebuf; | 366 | this->write_buf = gpio_nand_writebuf; |
367 | } | 367 | } |
368 | 368 | ||
369 | /* set the mtd private data for the nand driver */ | 369 | /* set the mtd private data for the nand driver */ |
370 | gpiomtd->mtd_info.priv = this; | 370 | gpiomtd->mtd_info.priv = this; |
371 | gpiomtd->mtd_info.owner = THIS_MODULE; | 371 | gpiomtd->mtd_info.owner = THIS_MODULE; |
372 | 372 | ||
373 | if (nand_scan(&gpiomtd->mtd_info, 1)) { | 373 | if (nand_scan(&gpiomtd->mtd_info, 1)) { |
374 | dev_err(&dev->dev, "no nand chips found?\n"); | 374 | dev_err(&dev->dev, "no nand chips found?\n"); |
375 | ret = -ENXIO; | 375 | ret = -ENXIO; |
376 | goto err_wp; | 376 | goto err_wp; |
377 | } | 377 | } |
378 | 378 | ||
379 | if (gpiomtd->plat.adjust_parts) | 379 | if (gpiomtd->plat.adjust_parts) |
380 | gpiomtd->plat.adjust_parts(&gpiomtd->plat, | 380 | gpiomtd->plat.adjust_parts(&gpiomtd->plat, |
381 | gpiomtd->mtd_info.size); | 381 | gpiomtd->mtd_info.size); |
382 | 382 | ||
383 | ppdata.of_node = dev->dev.of_node; | 383 | ppdata.of_node = dev->dev.of_node; |
384 | ret = mtd_device_parse_register(&gpiomtd->mtd_info, NULL, &ppdata, | 384 | ret = mtd_device_parse_register(&gpiomtd->mtd_info, NULL, &ppdata, |
385 | gpiomtd->plat.parts, | 385 | gpiomtd->plat.parts, |
386 | gpiomtd->plat.num_parts); | 386 | gpiomtd->plat.num_parts); |
387 | if (ret) | 387 | if (ret) |
388 | goto err_wp; | 388 | goto err_wp; |
389 | platform_set_drvdata(dev, gpiomtd); | 389 | platform_set_drvdata(dev, gpiomtd); |
390 | 390 | ||
391 | return 0; | 391 | return 0; |
392 | 392 | ||
393 | err_wp: | 393 | err_wp: |
394 | if (gpio_is_valid(gpiomtd->plat.gpio_nwp)) | 394 | if (gpio_is_valid(gpiomtd->plat.gpio_nwp)) |
395 | gpio_set_value(gpiomtd->plat.gpio_nwp, 0); | 395 | gpio_set_value(gpiomtd->plat.gpio_nwp, 0); |
396 | if (gpio_is_valid(gpiomtd->plat.gpio_rdy)) | 396 | if (gpio_is_valid(gpiomtd->plat.gpio_rdy)) |
397 | gpio_free(gpiomtd->plat.gpio_rdy); | 397 | gpio_free(gpiomtd->plat.gpio_rdy); |
398 | err_rdy: | 398 | err_rdy: |
399 | gpio_free(gpiomtd->plat.gpio_cle); | 399 | gpio_free(gpiomtd->plat.gpio_cle); |
400 | err_cle: | 400 | err_cle: |
401 | gpio_free(gpiomtd->plat.gpio_ale); | 401 | gpio_free(gpiomtd->plat.gpio_ale); |
402 | err_ale: | 402 | err_ale: |
403 | if (gpio_is_valid(gpiomtd->plat.gpio_nwp)) | 403 | if (gpio_is_valid(gpiomtd->plat.gpio_nwp)) |
404 | gpio_free(gpiomtd->plat.gpio_nwp); | 404 | gpio_free(gpiomtd->plat.gpio_nwp); |
405 | err_nwp: | 405 | err_nwp: |
406 | gpio_free(gpiomtd->plat.gpio_nce); | 406 | gpio_free(gpiomtd->plat.gpio_nce); |
407 | err_nce: | 407 | err_nce: |
408 | iounmap(gpiomtd->io_sync); | 408 | iounmap(gpiomtd->io_sync); |
409 | if (res1) | 409 | if (res1) |
410 | release_mem_region(res1->start, resource_size(res1)); | 410 | release_mem_region(res1->start, resource_size(res1)); |
411 | err_sync: | 411 | err_sync: |
412 | iounmap(gpiomtd->nand_chip.IO_ADDR_R); | 412 | iounmap(gpiomtd->nand_chip.IO_ADDR_R); |
413 | release_mem_region(res0->start, resource_size(res0)); | 413 | release_mem_region(res0->start, resource_size(res0)); |
414 | err_map: | 414 | err_map: |
415 | kfree(gpiomtd); | 415 | kfree(gpiomtd); |
416 | return ret; | 416 | return ret; |
417 | } | 417 | } |
418 | 418 | ||
419 | static struct platform_driver gpio_nand_driver = { | 419 | static struct platform_driver gpio_nand_driver = { |
420 | .probe = gpio_nand_probe, | 420 | .probe = gpio_nand_probe, |
421 | .remove = gpio_nand_remove, | 421 | .remove = gpio_nand_remove, |
422 | .driver = { | 422 | .driver = { |
423 | .name = "gpio-nand", | 423 | .name = "gpio-nand", |
424 | .of_match_table = gpio_nand_id_table, | 424 | .of_match_table = gpio_nand_id_table, |
425 | }, | 425 | }, |
426 | }; | 426 | }; |
427 | 427 | ||
428 | module_platform_driver(gpio_nand_driver); | 428 | module_platform_driver(gpio_nand_driver); |
429 | 429 | ||
430 | MODULE_LICENSE("GPL"); | 430 | MODULE_LICENSE("GPL"); |
431 | MODULE_AUTHOR("Ben Dooks <ben@simtec.co.uk>"); | 431 | MODULE_AUTHOR("Ben Dooks <ben@simtec.co.uk>"); |
432 | MODULE_DESCRIPTION("GPIO NAND Driver"); | 432 | MODULE_DESCRIPTION("GPIO NAND Driver"); |
433 | 433 |