2 * Flash memory access on SA11x0 based devices
4 * (C) 2000 Nicolas Pitre <nico@fluxnic.net>
6 #include <linux/module.h>
7 #include <linux/types.h>
8 #include <linux/ioport.h>
9 #include <linux/kernel.h>
10 #include <linux/init.h>
11 #include <linux/errno.h>
12 #include <linux/slab.h>
13 #include <linux/platform_device.h>
14 #include <linux/err.h>
17 #include <linux/mtd/mtd.h>
18 #include <linux/mtd/map.h>
19 #include <linux/mtd/partitions.h>
20 #include <linux/mtd/concat.h>
22 #include <mach/hardware.h>
23 #include <asm/sizes.h>
24 #include <asm/mach/flash.h>
28 * This is here for documentation purposes only - until these people
29 * submit their machine types. It will be gone January 2005.
31 static struct mtd_partition consus_partitions
[] = {
33 .name
= "Consus boot firmware",
36 .mask_flags
= MTD_WRITABLE
, /* force read-only */
38 .name
= "Consus kernel",
43 .name
= "Consus disk",
45 /* The rest (up to 16M) for jffs. We could put 0 and
46 make it find the size automatically, but right now
47 i have 32 megs. jffs will use all 32 megs if given
48 the chance, and this leads to horrible problems
49 when you try to re-flash the image because blob
50 won't erase the whole partition. */
51 .size
= 0x01000000 - 0x00140000,
54 /* this disk is a secondary disk, which can be used as
55 needed, for simplicity, make it the size of the other
56 consus partition, although realistically it could be
57 the remainder of the disk (depending on the file
59 .name
= "Consus disk2",
61 .size
= 0x01000000 - 0x00140000,
66 /* Frodo has 2 x 16M 28F128J3A flash chips in bank 0: */
67 static struct mtd_partition frodo_partitions
[] =
73 .mask_flags
= MTD_WRITEABLE
75 .name
= "bootloader params",
77 .offset
= MTDPART_OFS_APPEND
,
78 .mask_flags
= MTD_WRITEABLE
82 .offset
= MTDPART_OFS_APPEND
,
83 .mask_flags
= MTD_WRITEABLE
87 .offset
= MTDPART_OFS_APPEND
,
88 .mask_flags
= MTD_WRITEABLE
90 .name
= "file system",
91 .size
= MTDPART_SIZ_FULL
,
92 .offset
= MTDPART_OFS_APPEND
96 static struct mtd_partition jornada56x_partitions
[] = {
101 .mask_flags
= MTD_WRITEABLE
,
104 .size
= MTDPART_SIZ_FULL
,
105 .offset
= MTDPART_OFS_APPEND
,
109 static void jornada56x_set_vpp(int vpp
)
119 * Machine Phys Size set_vpp
120 * Consus : SA1100_CS0_PHYS SZ_32M
121 * Frodo : SA1100_CS0_PHYS SZ_32M
122 * Jornada56x: SA1100_CS0_PHYS SZ_32M jornada56x_set_vpp
126 struct sa_subdev_info
{
129 struct mtd_info
*mtd
;
130 struct flash_platform_data
*plat
;
134 struct mtd_info
*mtd
;
136 struct sa_subdev_info subdev
[0];
139 static void sa1100_set_vpp(struct map_info
*map
, int on
)
141 struct sa_subdev_info
*subdev
= container_of(map
, struct sa_subdev_info
, map
);
142 subdev
->plat
->set_vpp(on
);
145 static void sa1100_destroy_subdev(struct sa_subdev_info
*subdev
)
148 map_destroy(subdev
->mtd
);
149 if (subdev
->map
.virt
)
150 iounmap(subdev
->map
.virt
);
151 release_mem_region(subdev
->map
.phys
, subdev
->map
.size
);
154 static int sa1100_probe_subdev(struct sa_subdev_info
*subdev
, struct resource
*res
)
161 size
= res
->end
- phys
+ 1;
164 * Retrieve the bankwidth from the MSC registers.
165 * We currently only implement CS0 and CS1 here.
169 printk(KERN_WARNING
"SA1100 flash: unknown base address "
170 "0x%08lx, assuming CS0\n", phys
);
172 case SA1100_CS0_PHYS
:
173 subdev
->map
.bankwidth
= (MSC0
& MSC_RBW
) ? 2 : 4;
176 case SA1100_CS1_PHYS
:
177 subdev
->map
.bankwidth
= ((MSC0
>> 16) & MSC_RBW
) ? 2 : 4;
181 if (!request_mem_region(phys
, size
, subdev
->name
)) {
186 if (subdev
->plat
->set_vpp
)
187 subdev
->map
.set_vpp
= sa1100_set_vpp
;
189 subdev
->map
.phys
= phys
;
190 subdev
->map
.size
= size
;
191 subdev
->map
.virt
= ioremap(phys
, size
);
192 if (!subdev
->map
.virt
) {
197 simple_map_init(&subdev
->map
);
200 * Now let's probe for the actual flash. Do it here since
201 * specific machine settings might have been set above.
203 subdev
->mtd
= do_map_probe(subdev
->plat
->map_name
, &subdev
->map
);
204 if (subdev
->mtd
== NULL
) {
208 subdev
->mtd
->owner
= THIS_MODULE
;
210 printk(KERN_INFO
"SA1100 flash: CFI device at 0x%08lx, %uMiB, %d-bit\n",
211 phys
, (unsigned)(subdev
->mtd
->size
>> 20),
212 subdev
->map
.bankwidth
* 8);
217 sa1100_destroy_subdev(subdev
);
222 static void sa1100_destroy(struct sa_info
*info
, struct flash_platform_data
*plat
)
227 mtd_device_unregister(info
->mtd
);
228 if (info
->mtd
!= info
->subdev
[0].mtd
)
229 mtd_concat_destroy(info
->mtd
);
232 for (i
= info
->num_subdev
- 1; i
>= 0; i
--)
233 sa1100_destroy_subdev(&info
->subdev
[i
]);
240 static struct sa_info
*__devinit
241 sa1100_setup_mtd(struct platform_device
*pdev
, struct flash_platform_data
*plat
)
243 struct sa_info
*info
;
244 int nr
, size
, i
, ret
= 0;
247 * Count number of devices.
250 if (!platform_get_resource(pdev
, IORESOURCE_MEM
, nr
))
258 size
= sizeof(struct sa_info
) + sizeof(struct sa_subdev_info
) * nr
;
261 * Allocate the map_info structs in one go.
263 info
= kzalloc(size
, GFP_KERNEL
);
276 * Claim and then map the memory regions.
278 for (i
= 0; i
< nr
; i
++) {
279 struct sa_subdev_info
*subdev
= &info
->subdev
[i
];
280 struct resource
*res
;
282 res
= platform_get_resource(pdev
, IORESOURCE_MEM
, i
);
286 subdev
->map
.name
= subdev
->name
;
287 sprintf(subdev
->name
, "%s-%d", plat
->name
, i
);
290 ret
= sa1100_probe_subdev(subdev
, res
);
295 info
->num_subdev
= i
;
298 * ENXIO is special. It means we didn't find a chip when we probed.
300 if (ret
!= 0 && !(ret
== -ENXIO
&& info
->num_subdev
> 0))
304 * If we found one device, don't bother with concat support. If
305 * we found multiple devices, use concat if we have it available,
306 * otherwise fail. Either way, it'll be called "sa1100".
308 if (info
->num_subdev
== 1) {
309 strcpy(info
->subdev
[0].name
, plat
->name
);
310 info
->mtd
= info
->subdev
[0].mtd
;
312 } else if (info
->num_subdev
> 1) {
313 struct mtd_info
*cdev
[nr
];
315 * We detected multiple devices. Concatenate them together.
317 for (i
= 0; i
< info
->num_subdev
; i
++)
318 cdev
[i
] = info
->subdev
[i
].mtd
;
320 info
->mtd
= mtd_concat_create(cdev
, info
->num_subdev
,
322 if (info
->mtd
== NULL
)
330 sa1100_destroy(info
, plat
);
335 static const char *part_probes
[] = { "cmdlinepart", "RedBoot", NULL
};
337 static int __devinit
sa1100_mtd_probe(struct platform_device
*pdev
)
339 struct flash_platform_data
*plat
= pdev
->dev
.platform_data
;
340 struct sa_info
*info
;
346 info
= sa1100_setup_mtd(pdev
, plat
);
353 * Partition selection stuff.
355 mtd_device_parse_register(info
->mtd
, part_probes
, 0,
356 plat
->parts
, plat
->nr_parts
);
358 platform_set_drvdata(pdev
, info
);
365 static int __exit
sa1100_mtd_remove(struct platform_device
*pdev
)
367 struct sa_info
*info
= platform_get_drvdata(pdev
);
368 struct flash_platform_data
*plat
= pdev
->dev
.platform_data
;
370 platform_set_drvdata(pdev
, NULL
);
371 sa1100_destroy(info
, plat
);
377 static void sa1100_mtd_shutdown(struct platform_device
*dev
)
379 struct sa_info
*info
= platform_get_drvdata(dev
);
380 if (info
&& mtd_suspend(info
->mtd
) == 0)
381 mtd_resume(info
->mtd
);
384 #define sa1100_mtd_shutdown NULL
387 static struct platform_driver sa1100_mtd_driver
= {
388 .probe
= sa1100_mtd_probe
,
389 .remove
= __exit_p(sa1100_mtd_remove
),
390 .shutdown
= sa1100_mtd_shutdown
,
392 .name
= "sa1100-mtd",
393 .owner
= THIS_MODULE
,
397 module_platform_driver(sa1100_mtd_driver
);
399 MODULE_AUTHOR("Nicolas Pitre");
400 MODULE_DESCRIPTION("SA1100 CFI map driver");
401 MODULE_LICENSE("GPL");
402 MODULE_ALIAS("platform:sa1100-mtd");