/* * Flash memory access on SA11x0 based devices * * (C) 2000 Nicolas Pitre * * $Id: sa1100-flash.c,v 1.22 2001/10/02 10:04:52 rmk Exp $ */ #include #include #include #include #include #include #include #include #include #ifndef CONFIG_ARCH_SA1100 #error This is for SA1100 architecture only #endif #define WINDOW_ADDR 0xe8000000 static __u8 sa1100_read8(struct map_info *map, unsigned long ofs) { return readb(map->map_priv_1 + ofs); } static __u16 sa1100_read16(struct map_info *map, unsigned long ofs) { return readw(map->map_priv_1 + ofs); } static __u32 sa1100_read32(struct map_info *map, unsigned long ofs) { return readl(map->map_priv_1 + ofs); } static void sa1100_copy_from(struct map_info *map, void *to, unsigned long from, ssize_t len) { memcpy(to, (void *)(map->map_priv_1 + from), len); } static void sa1100_write8(struct map_info *map, __u8 d, unsigned long adr) { writeb(d, map->map_priv_1 + adr); } static void sa1100_write16(struct map_info *map, __u16 d, unsigned long adr) { writew(d, map->map_priv_1 + adr); } static void sa1100_write32(struct map_info *map, __u32 d, unsigned long adr) { writel(d, map->map_priv_1 + adr); } static void sa1100_copy_to(struct map_info *map, unsigned long to, const void *from, ssize_t len) { memcpy((void *)(map->map_priv_1 + to), from, len); } #ifdef CONFIG_SA1100_H3600 static void h3600_set_vpp(struct map_info *map, int vpp) { if (vpp) set_h3600_egpio(EGPIO_H3600_VPP_ON); else clr_h3600_egpio(EGPIO_H3600_VPP_ON); } #endif #ifdef CONFIG_SA1100_JORNADA720 static void jornada720_set_vpp(int vpp) { if (vpp) PPSR |= 0x80; else PPSR &= ~0x80; PPDR |= 0x80; } #endif static struct map_info sa1100_map = { name: "SA1100 flash", read8: sa1100_read8, read16: sa1100_read16, read32: sa1100_read32, copy_from: sa1100_copy_from, write8: sa1100_write8, write16: sa1100_write16, write32: sa1100_write32, copy_to: sa1100_copy_to, map_priv_1: WINDOW_ADDR, }; /* * Here are partition information for all known SA1100-based devices. * See include/linux/mtd/partitions.h for definition of the mtd_partition * structure. * * The *_max_flash_size is the maximum possible mapped flash size which * is not necessarily the actual flash size. It must correspond to the * value specified in the mapping definition defined by the * "struct map_desc *_io_desc" for the corresponding machine. */ #ifdef CONFIG_SA1100_ASSABET /* Phase 4 Assabet has two 28F160B3 flash parts in bank 0: */ static unsigned long assabet4_max_flash_size = 0x00400000; static struct mtd_partition assabet4_partitions[] = { { name: "bootloader", size: 0x00020000, offset: 0, mask_flags: MTD_WRITEABLE },{ name: "bootloader params", size: 0x00020000, offset: MTDPART_OFS_APPEND, mask_flags: MTD_WRITEABLE },{ name: "jffs", size: MTDPART_SIZ_FULL, offset: MTDPART_OFS_APPEND } }; /* Phase 5 Assabet has two 28F128J3A flash parts in bank 0: */ static unsigned long assabet5_max_flash_size = 0x02000000; static struct mtd_partition assabet5_partitions[] = { { name: "bootloader", size: 0x00040000, offset: 0, mask_flags: MTD_WRITEABLE },{ name: "bootloader params", size: 0x00040000, offset: MTDPART_OFS_APPEND, mask_flags: MTD_WRITEABLE },{ name: "jffs", size: MTDPART_SIZ_FULL, offset: MTDPART_OFS_APPEND } }; #define assabet_max_flash_size assabet5_max_flash_size #define assabet_partitions assabet5_partitions #endif #ifdef CONFIG_SA1100_FLEXANET /* Flexanet has two 28F128J3A flash parts in bank 0: */ static unsigned long flexanet_max_flash_size = 0x02000000; static struct mtd_partition flexanet_partitions[] = { { name: "bootloader", size: 0x00040000, offset: 0, mask_flags: MTD_WRITEABLE },{ name: "bootloader params", size: 0x00040000, offset: MTDPART_OFS_APPEND, mask_flags: MTD_WRITEABLE },{ name: "kernel", size: 0x000C0000, offset: MTDPART_OFS_APPEND, mask_flags: MTD_WRITEABLE },{ name: "altkernel", size: 0x000C0000, offset: MTDPART_OFS_APPEND, mask_flags: MTD_WRITEABLE },{ name: "root", size: 0x00400000, offset: MTDPART_OFS_APPEND, mask_flags: MTD_WRITEABLE },{ name: "free1", size: 0x00300000, offset: MTDPART_OFS_APPEND, mask_flags: MTD_WRITEABLE },{ name: "free2", size: 0x00300000, offset: MTDPART_OFS_APPEND, mask_flags: MTD_WRITEABLE },{ name: "free3", size: MTDPART_SIZ_FULL, offset: MTDPART_OFS_APPEND, mask_flags: MTD_WRITEABLE } }; #endif #ifdef CONFIG_SA1100_HUW_WEBPANEL static unsigned long huw_webpanel_max_flash_size = 0x01000000; static struct mtd_partition huw_webpanel_partitions[] = { { name: "Loader", size: 0x00040000, offset: 0, },{ name: "Sector 1", size: 0x00040000, offset: MTDPART_OFS_APPEND, },{ size: MTDPART_SIZ_FULL, offset: MTDPART_OFS_APPEND, } }; #endif /* CONFIG_SA1100_HUW_WEBPANEL */ #ifdef CONFIG_SA1100_H3600 static unsigned long h3600_max_flash_size = 0x02000000; static struct mtd_partition h3600_partitions[] = { { name: "H3600 boot firmware", size: 0x00040000, offset: 0, mask_flags: MTD_WRITEABLE /* force read-only */ },{ name: "H3600 kernel", size: 0x00080000, offset: 0x40000 },{ name: "H3600 params", size: 0x00040000, offset: 0xC0000 },{ #ifdef CONFIG_JFFS2_FS name: "H3600 root jffs2", offset: 0x00100000, size: MTDPART_SIZ_FULL #else name: "H3600 initrd", size: 0x00100000, offset: 0x00100000 },{ name: "H3600 root cramfs", size: 0x00300000, offset: 0x00200000 },{ name: "H3600 usr cramfs", size: 0x00800000, offset: 0x00500000 },{ name: "H3600 usr local", offset: 0x00d00000, size: MTDPART_SIZ_FULL #endif } }; #endif #ifdef CONFIG_SA1100_FREEBIRD static unsigned long freebird_max_flash_size = 0x02000000; static struct mtd_partition freebird_partitions[] = { #if CONFIG_SA1100_FREEBIRD_NEW { name: "firmware", size: 0x00040000, offset: 0, mask_flags: MTD_WRITEABLE /* force read-only */ },{ name: "kernel", size: 0x00080000, offset: 0x40000 },{ name: "params", size: 0x00040000, offset: 0xC0000 },{ name: "initrd", size: 0x00100000, offset: 0x00100000 },{ name: "root cramfs", size: 0x00300000, offset: 0x00200000 },{ name: "usr cramfs", size: 0x00C00000, offset: 0x00500000 },{ name: "local", offset: 0x01100000, size: MTDPART_SIZ_FULL } #else { offset: 0, size: 0x00040000, }, { offset: MTDPART_OFS_APPEND, size: 0x000c0000, }, { offset: MTDPART_OFS_APPEND, size: 0x00400000, }, { offset: MTDPART_OFS_APPEND, size: MTDPART_SIZ_FULL } #endif }; #endif #ifdef CONFIG_SA1100_CERF static unsigned long cerf_max_flash_size = 0x01000000; static struct mtd_partition cerf_partitions[] = { { offset: 0, size: 0x00800000 }, { offset: MTDPART_OFS_APPEND, size: 0x00800000 } }; #endif #ifdef CONFIG_SA1100_GRAPHICSCLIENT static unsigned long graphicsclient_max_flash_size = 0x01000000; static struct mtd_partition graphicsclient_partitions[] = { { name: "zImage", offset: 0, size: 0x100000 }, { name: "ramdisk.gz", offset: MTDPART_OFS_APPEND, size: 0x300000 }, { name: "User FS", offset: MTDPART_OFS_APPEND, size: MTDPART_SIZ_FULL } }; #endif #ifdef CONFIG_SA1100_GRAPHICSMASTER static unsigned long graphicsmaster_max_flash_size = 0x01000000; static struct mtd_partition graphicsmaster_partitions[] = { { name: "zImage", offset: 0, size: 0x100000 }, { name: "ramdisk.gz", offset: MTDPART_OFS_APPEND, size: 0x300000 }, { name: "User FS", offset: MTDPART_OFS_APPEND, size: MTDPART_SIZ_FULL } }; #endif #ifdef CONFIG_SA1100_PANGOLIN static unsigned long pangolin_max_flash_size = 0x04000000; static struct mtd_partition pangolin_partitions[] = { { name: "boot firmware", offset: 0x00000000, size: 0x00080000, mask_flags: MTD_WRITEABLE, /* force read-only */ }, { name: "kernel", offset: 0x00080000, size: 0x00100000, }, { name: "initrd", offset: 0x00180000, size: 0x00280000, }, { name: "initrd-test", offset: 0x00400000, size: 0x03C00000, } }; #endif #ifdef CONFIG_SA1100_YOPY static unsigned long yopy_max_flash_size = 0x08000000; static struct mtd_partition yopy_partitions[] = { { name: "boot firmware", offset: 0x00000000, size: 0x00040000, mask_flags: MTD_WRITEABLE, /* force read-only */ }, { name: "kernel", offset: 0x00080000, size: 0x00080000, }, { name: "initrd", offset: 0x00100000, size: 0x00300000, }, { name: "root", offset: 0x00400000, size: 0x01000000, }, }; #endif #ifdef CONFIG_SA1100_JORNADA720 static unsigned long jornada720_max_flash_size = 0x02000000; static struct mtd_partition jornada720_partitions[] = { { name: "JORNADA720 boot firmware", size: 0x00040000, offset: 0, mask_flags: MTD_WRITEABLE /* force read-only */ },{ name: "JORNADA720 kernel", size: 0x000c0000, offset: 0x40000 },{ name: "JORNADA720 params", size: 0x00040000, offset: 0x100000 },{ name: "JORNADA720 initrd", size: 0x00100000, offset: 0x00140000 },{ name: "JORNADA720 root cramfs", size: 0x00300000, offset: 0x00240000 },{ name: "JORNADA720 usr cramfs", size: 0x00800000, offset: 0x00540000 },{ name: "JORNADA720 usr local", offset: 0x00d00000, size: 0 /* will expand to the end of the flash */ } }; #endif #ifdef CONFIG_SA1100_SHERMAN static unsigned long sherman_max_flash_size = 0x02000000; static struct mtd_partition sherman_partitions[] = { { offset: 0, size: 0x50000 }, { offset: MTDPART_OFS_APPEND, size: 0x70000 }, { offset: MTDPART_OFS_APPEND, size: 0x600000 }, { offset: MTDPART_OFS_APPEND, size: 0xA0000 } }; #endif #ifdef CONFIG_SA1100_STORK static unsigned long stork_max_flash_size = 0x02000000; static struct mtd_partition stork_partitions[] = { { name: "STORK boot firmware", size: 0x00040000, offset: 0, mask_flags: MTD_WRITEABLE /* force read-only */ },{ name: "STORK params", size: 0x00040000, offset: 0x40000 },{ name: "STORK kernel", size: 0x00100000, offset: 0x80000 },{ #ifdef CONFIG_JFFS2_FS name: "STORK root jffs2", offset: 0x00180000, size: MTDPART_SIZ_FULL #else name: "STORK initrd", size: 0x00100000, offset: 0x00180000 },{ name: "STORK root cramfs", size: 0x00300000, offset: 0x00280000 },{ name: "STORK usr cramfs", size: 0x00800000, offset: 0x00580000 },{ name: "STORK usr local", offset: 0x00d80000, size: MTDPART_SIZ_FULL #endif } }; #endif #define NB_OF(x) (sizeof(x)/sizeof(x[0])) extern int parse_redboot_partitions(struct mtd_info *master, struct mtd_partition **pparts); extern int parse_bootldr_partitions(struct mtd_info *master, struct mtd_partition **pparts); static struct mtd_partition *parsed_parts; static struct mtd_info *mymtd; int __init sa1100_mtd_init(void) { struct mtd_partition *parts; int nb_parts = 0; int parsed_nr_parts = 0; char *part_type; /* Default flash buswidth */ sa1100_map.buswidth = (MSC0 & MSC_RBW) ? 2 : 4; /* * Static partition definition selection */ part_type = "static"; #ifdef CONFIG_SA1100_ASSABET if (machine_is_assabet()) { parts = assabet_partitions; nb_parts = NB_OF(assabet_partitions); sa1100_map.size = assabet_max_flash_size; } #endif #ifdef CONFIG_SA1100_HUW_WEBPANEL if (machine_is_huw_webpanel()) { parts = huw_webpanel_partitions; nb_parts = NB_OF(huw_webpanel_partitions); sa1100_map.size = huw_webpanel_max_flash_size; } #endif #ifdef CONFIG_SA1100_H3600 if (machine_is_h3600()) { parts = h3600_partitions; nb_parts = NB_OF(h3600_partitions); sa1100_map.size = h3600_max_flash_size; sa1100_map.set_vpp = h3600_set_vpp; } #endif #ifdef CONFIG_SA1100_FREEBIRD if (machine_is_freebird()) { parts = freebird_partitions; nb_parts = NB_OF(freebird_partitions); sa1100_map.size = freebird_max_flash_size; } #endif #ifdef CONFIG_SA1100_CERF if (machine_is_cerf()) { parts = cerf_partitions; nb_parts = NB_OF(cerf_partitions); sa1100_map.size = cerf_max_flash_size; } #endif #ifdef CONFIG_SA1100_GRAPHICSCLIENT if (machine_is_graphicsclient()) { parts = graphicsclient_partitions; nb_parts = NB_OF(graphicsclient_partitions); sa1100_map.size = graphicsclient_max_flash_size; sa1100_map.buswidth = (MSC1 & MSC_RBW) ? 2:4; } #endif #ifdef CONFIG_SA1100_GRAPHICSMASTER if (machine_is_graphicsmaster()) { parts = graphicsmaster_partitions; nb_parts = NB_OF(graphicsmaster_partitions); sa1100_map.size = graphicsmaster_max_flash_size; sa1100_map.buswidth = (MSC1 & MSC_RBW) ? 2:4; } #endif #ifdef CONFIG_SA1100_PANGOLIN if (machine_is_pangolin()) { parts = pangolin_partitions; nb_parts = NB_OF(pangolin_partitions); sa1100_map.size = pangolin_max_flash_size; } #endif #ifdef CONFIG_SA1100_JORNADA720 if (machine_is_jornada720()) { parts = jornada720_partitions; nb_parts = NB_OF(jornada720_partitions); sa1100_map.size = jornada720_max_flash_size; sa1100_map.set_vpp = jornada720_set_vpp; } #endif #ifdef CONFIG_SA1100_YOPY if (machine_is_yopy()) { parts = yopy_partitions; nb_parts = NB_OF(yopy_partitions); sa1100_map.size = yopy_max_flash_size; } #endif #ifdef CONFIG_SA1100_SHERMAN if (machine_is_sherman()) { parts = sherman_partitions; nb_parts = NB_OF(sherman_partitions); sa1100_map.size = sherman_max_flash_size; } #endif #ifdef CONFIG_SA1100_FLEXANET if (machine_is_flexanet()) { parts = flexanet_partitions; nb_parts = NB_OF(flexanet_partitions); sa1100_map.size = flexanet_max_flash_size; } #endif #ifdef CONFIG_SA1100_STORK if (machine_is_stork()) { parts = stork_partitions; nb_parts = NB_OF(stork_partitions); sa1100_map.size = stork_max_flash_size; } #endif /* * Now let's probe for the actual flash. Do it here since * specific machine settings might have been set above. */ printk(KERN_NOTICE "SA1100 flash: probing %d-bit flash bus\n", sa1100_map.buswidth*8); mymtd = do_map_probe("cfi_probe", &sa1100_map); if (!mymtd) return -ENXIO; mymtd->module = THIS_MODULE; /* * Dynamic partition selection stuff (might override the static ones) */ #ifdef CONFIG_MTD_REDBOOT_PARTS if (parsed_nr_parts == 0) { int ret = parse_redboot_partitions(mymtd, &parsed_parts); if (ret > 0) { part_type = "RedBoot"; parsed_nr_parts = ret; } } #endif #ifdef CONFIG_MTD_BOOTLDR_PARTS if (parsed_nr_parts == 0) { int ret = parse_bootldr_partitions(mymtd, &parsed_parts); if (ret > 0) { part_type = "Compaq bootldr"; parsed_nr_parts = ret; } } #endif if (parsed_nr_parts > 0) { parts = parsed_parts; nb_parts = parsed_nr_parts; } if (nb_parts == 0) { printk(KERN_NOTICE "SA1100 flash: no partition info available, registering whole flash at once\n"); add_mtd_device(mymtd); } else { printk(KERN_NOTICE "Using %s partition definition\n", part_type); add_mtd_partitions(mymtd, parts, nb_parts); } return 0; } static void __exit sa1100_mtd_cleanup(void) { if (mymtd) { del_mtd_partitions(mymtd); map_destroy(mymtd); if (parsed_parts) kfree(parsed_parts); } } module_init(sa1100_mtd_init); module_exit(sa1100_mtd_cleanup); MODULE_AUTHOR("Nicolas Pitre"); MODULE_DESCRIPTION("SA1100 CFI map driver"); MODULE_LICENSE("GPL");