2  * rfd_ftl.c -- resident flash disk (flash translation layer)
 
   4  * Copyright (C) 2005  Sean Young <sean@mess.org>
 
   6  * $Id: rfd_ftl.c,v 1.5 2005/11/07 11:14:21 gleixner Exp $
 
   8  * This type of flash translation layer (FTL) is used by the Embedded BIOS
 
   9  * by General Software. It is known as the Resident Flash Disk (RFD), see:
 
  11  *      http://www.gensw.com/pages/prod/bios/rfd.htm
 
  16 #include <linux/hdreg.h>
 
  17 #include <linux/init.h>
 
  18 #include <linux/mtd/blktrans.h>
 
  19 #include <linux/mtd/mtd.h>
 
  20 #include <linux/vmalloc.h>
 
  21 #include <linux/jiffies.h>
 
  23 #include <asm/types.h>
 
  25 #define const_cpu_to_le16       __constant_cpu_to_le16
 
  27 static int block_size = 0;
 
  28 module_param(block_size, int, 0);
 
  29 MODULE_PARM_DESC(block_size, "Block size to use by RFD, defaults to erase unit size");
 
  31 #define PREFIX "rfd_ftl: "
 
  33 /* This major has been assigned by device@lanana.org */
 
  35 #define RFD_FTL_MAJOR           256
 
  38 /* Maximum number of partitions in an FTL region */
 
  41 /* An erase unit should start with this value */
 
  42 #define RFD_MAGIC               0x9193
 
  44 /* the second value is 0xffff or 0xffc8; function unknown */
 
  46 /* the third value is always 0xffff, ignored */
 
  48 /* next is an array of mapping for each corresponding sector */
 
  49 #define HEADER_MAP_OFFSET       3
 
  50 #define SECTOR_DELETED          0x0000
 
  51 #define SECTOR_ZERO             0xfffe
 
  52 #define SECTOR_FREE             0xffff
 
  54 #define SECTOR_SIZE             512
 
  56 #define SECTORS_PER_TRACK       63
 
  72         struct mtd_blktrans_dev mbd;
 
  74         u_int block_size;               /* size of erase unit */
 
  75         u_int total_blocks;             /* number of erase units */
 
  76         u_int header_sectors_per_block; /* header sectors in erase unit */
 
  77         u_int data_sectors_per_block;   /* data sectors in erase unit */
 
  78         u_int sector_count;             /* sectors in translated disk */
 
  79         u_int header_size;              /* bytes in header sector */
 
  80         int reserved_block;             /* block next up for reclaim */
 
  81         int current_block;              /* block to write to */
 
  82         u16 *header_cache;              /* cached header */
 
  91 static int rfd_ftl_writesect(struct mtd_blktrans_dev *dev, u_long sector, char *buf);
 
  93 static int build_block_map(struct partition *part, int block_no)
 
  95         struct block *block = &part->blocks[block_no];
 
  98         block->offset = part->block_size * block_no;
 
 100         if (le16_to_cpu(part->header_cache[0]) != RFD_MAGIC) {
 
 101                 block->state = BLOCK_ERASED; /* assumption */
 
 102                 block->free_sectors = part->data_sectors_per_block;
 
 103                 part->reserved_block = block_no;
 
 107         block->state = BLOCK_OK;
 
 109         for (i=0; i<part->data_sectors_per_block; i++) {
 
 112                 entry = le16_to_cpu(part->header_cache[HEADER_MAP_OFFSET + i]);
 
 114                 if (entry == SECTOR_DELETED)
 
 117                 if (entry == SECTOR_FREE) {
 
 118                         block->free_sectors++;
 
 122                 if (entry == SECTOR_ZERO)
 
 125                 if (entry >= part->sector_count) {
 
 126                         printk(KERN_NOTICE PREFIX
 
 127                                 "'%s': unit #%d: entry %d corrupt, "
 
 128                                 "sector %d out of range\n",
 
 129                                 part->mbd.mtd->name, block_no, i, entry);
 
 133                 if (part->sector_map[entry] != -1) {
 
 134                         printk(KERN_NOTICE PREFIX
 
 135                                 "'%s': more than one entry for sector %d\n",
 
 136                                 part->mbd.mtd->name, entry);
 
 141                 part->sector_map[entry] = block->offset +
 
 142                         (i + part->header_sectors_per_block) * SECTOR_SIZE;
 
 144                 block->used_sectors++;
 
 147         if (block->free_sectors == part->data_sectors_per_block)
 
 148                 part->reserved_block = block_no;
 
 153 static int scan_header(struct partition *part)
 
 155         int sectors_per_block;
 
 160         sectors_per_block = part->block_size / SECTOR_SIZE;
 
 161         part->total_blocks = part->mbd.mtd->size / part->block_size;
 
 163         if (part->total_blocks < 2)
 
 166         /* each erase block has three bytes header, followed by the map */
 
 167         part->header_sectors_per_block =
 
 168                         ((HEADER_MAP_OFFSET + sectors_per_block) *
 
 169                         sizeof(u16) + SECTOR_SIZE - 1) / SECTOR_SIZE;
 
 171         part->data_sectors_per_block = sectors_per_block -
 
 172                         part->header_sectors_per_block;
 
 174         part->header_size = (HEADER_MAP_OFFSET +
 
 175                         part->data_sectors_per_block) * sizeof(u16);
 
 177         part->cylinders = (part->data_sectors_per_block *
 
 178                         (part->total_blocks - 1) - 1) / SECTORS_PER_TRACK;
 
 180         part->sector_count = part->cylinders * SECTORS_PER_TRACK;
 
 182         part->current_block = -1;
 
 183         part->reserved_block = -1;
 
 184         part->is_reclaiming = 0;
 
 186         part->header_cache = kmalloc(part->header_size, GFP_KERNEL);
 
 187         if (!part->header_cache)
 
 190         part->blocks = kcalloc(part->total_blocks, sizeof(struct block),
 
 195         part->sector_map = vmalloc(part->sector_count * sizeof(u_long));
 
 196         if (!part->sector_map) {
 
 197                 printk(KERN_ERR PREFIX "'%s': unable to allocate memory for "
 
 198                         "sector map", part->mbd.mtd->name);
 
 202         for (i=0; i<part->sector_count; i++)
 
 203                 part->sector_map[i] = -1;
 
 205         for (i=0, blocks_found=0; i<part->total_blocks; i++) {
 
 206                 rc = part->mbd.mtd->read(part->mbd.mtd,
 
 207                                 i * part->block_size, part->header_size,
 
 208                                 &retlen, (u_char*)part->header_cache);
 
 210                 if (!rc && retlen != part->header_size)
 
 216                 if (!build_block_map(part, i))
 
 220         if (blocks_found == 0) {
 
 221                 printk(KERN_NOTICE PREFIX "no RFD magic found in '%s'\n",
 
 222                                 part->mbd.mtd->name);
 
 227         if (part->reserved_block == -1) {
 
 228                 printk(KERN_NOTICE PREFIX "'%s': no empty erase unit found\n",
 
 229                                 part->mbd.mtd->name);
 
 237         vfree(part->sector_map);
 
 238         kfree(part->header_cache);
 
 244 static int rfd_ftl_readsect(struct mtd_blktrans_dev *dev, u_long sector, char *buf)
 
 246         struct partition *part = (struct partition*)dev;
 
 251         if (sector >= part->sector_count)
 
 254         addr = part->sector_map[sector];
 
 256                 rc = part->mbd.mtd->read(part->mbd.mtd, addr, SECTOR_SIZE,
 
 257                                                 &retlen, (u_char*)buf);
 
 258                 if (!rc && retlen != SECTOR_SIZE)
 
 262                         printk(KERN_WARNING PREFIX "error reading '%s' at "
 
 263                                 "0x%lx\n", part->mbd.mtd->name, addr);
 
 267                 memset(buf, 0, SECTOR_SIZE);
 
 272 static void erase_callback(struct erase_info *erase)
 
 274         struct partition *part;
 
 279         part = (struct partition*)erase->priv;
 
 281         i = erase->addr / part->block_size;
 
 282         if (i >= part->total_blocks || part->blocks[i].offset != erase->addr) {
 
 283                 printk(KERN_ERR PREFIX "erase callback for unknown offset %x "
 
 284                                 "on '%s'\n", erase->addr, part->mbd.mtd->name);
 
 288         if (erase->state != MTD_ERASE_DONE) {
 
 289                 printk(KERN_WARNING PREFIX "erase failed at 0x%x on '%s', "
 
 290                                 "state %d\n", erase->addr,
 
 291                                 part->mbd.mtd->name, erase->state);
 
 293                 part->blocks[i].state = BLOCK_FAILED;
 
 294                 part->blocks[i].free_sectors = 0;
 
 295                 part->blocks[i].used_sectors = 0;
 
 302         magic = const_cpu_to_le16(RFD_MAGIC);
 
 304         part->blocks[i].state = BLOCK_ERASED;
 
 305         part->blocks[i].free_sectors = part->data_sectors_per_block;
 
 306         part->blocks[i].used_sectors = 0;
 
 307         part->blocks[i].erases++;
 
 309         rc = part->mbd.mtd->write(part->mbd.mtd,
 
 310                 part->blocks[i].offset, sizeof(magic), &retlen,
 
 313         if (!rc && retlen != sizeof(magic))
 
 317                 printk(KERN_NOTICE PREFIX "'%s': unable to write RFD "
 
 320                                 part->blocks[i].offset);
 
 321                 part->blocks[i].state = BLOCK_FAILED;
 
 324                 part->blocks[i].state = BLOCK_OK;
 
 329 static int erase_block(struct partition *part, int block)
 
 331         struct erase_info *erase;
 
 334         erase = kmalloc(sizeof(struct erase_info), GFP_KERNEL);
 
 338         erase->mtd = part->mbd.mtd;
 
 339         erase->callback = erase_callback;
 
 340         erase->addr = part->blocks[block].offset;
 
 341         erase->len = part->block_size;
 
 342         erase->priv = (u_long)part;
 
 344         part->blocks[block].state = BLOCK_ERASING;
 
 345         part->blocks[block].free_sectors = 0;
 
 347         rc = part->mbd.mtd->erase(part->mbd.mtd, erase);
 
 350                 printk(KERN_WARNING PREFIX "erase of region %x,%x on '%s' "
 
 351                                 "failed\n", erase->addr, erase->len,
 
 352                                 part->mbd.mtd->name);
 
 360 static int move_block_contents(struct partition *part, int block_no, u_long *old_sector)
 
 367         part->is_reclaiming = 1;
 
 369         sector_data = kmalloc(SECTOR_SIZE, GFP_KERNEL);
 
 373         map = kmalloc(part->header_size, GFP_KERNEL);
 
 377         rc = part->mbd.mtd->read(part->mbd.mtd,
 
 378                 part->blocks[block_no].offset, part->header_size,
 
 379                 &retlen, (u_char*)map);
 
 381         if (!rc && retlen != part->header_size)
 
 385                 printk(KERN_NOTICE PREFIX "error reading '%s' at "
 
 386                         "0x%lx\n", part->mbd.mtd->name,
 
 387                         part->blocks[block_no].offset);
 
 392         for (i=0; i<part->data_sectors_per_block; i++) {
 
 393                 u16 entry = le16_to_cpu(map[HEADER_MAP_OFFSET + i]);
 
 397                 if (entry == SECTOR_FREE || entry == SECTOR_DELETED)
 
 400                 if (entry == SECTOR_ZERO)
 
 403                 /* already warned about and ignored in build_block_map() */
 
 404                 if (entry >= part->sector_count)
 
 407                 addr = part->blocks[block_no].offset +
 
 408                         (i + part->header_sectors_per_block) * SECTOR_SIZE;
 
 410                 if (*old_sector == addr) {
 
 412                         if (!part->blocks[block_no].used_sectors--) {
 
 413                                 rc = erase_block(part, block_no);
 
 418                 rc = part->mbd.mtd->read(part->mbd.mtd, addr,
 
 419                         SECTOR_SIZE, &retlen, sector_data);
 
 421                 if (!rc && retlen != SECTOR_SIZE)
 
 425                         printk(KERN_NOTICE PREFIX "'%s': Unable to "
 
 426                                 "read sector for relocation\n",
 
 427                                 part->mbd.mtd->name);
 
 432                 rc = rfd_ftl_writesect((struct mtd_blktrans_dev*)part,
 
 444         part->is_reclaiming = 0;
 
 449 static int reclaim_block(struct partition *part, u_long *old_sector)
 
 451         int block, best_block, score, old_sector_block;
 
 454         /* we have a race if sync doesn't exist */
 
 455         if (part->mbd.mtd->sync)
 
 456                 part->mbd.mtd->sync(part->mbd.mtd);
 
 458         score = 0x7fffffff; /* MAX_INT */
 
 460         if (*old_sector != -1)
 
 461                 old_sector_block = *old_sector / part->block_size;
 
 463                 old_sector_block = -1;
 
 465         for (block=0; block<part->total_blocks; block++) {
 
 468                 if (block == part->reserved_block)
 
 472                  * Postpone reclaiming if there is a free sector as
 
 473                  * more removed sectors is more efficient (have to move
 
 476                 if (part->blocks[block].free_sectors)
 
 479                 this_score = part->blocks[block].used_sectors;
 
 481                 if (block == old_sector_block)
 
 484                         /* no point in moving a full block */
 
 485                         if (part->blocks[block].used_sectors ==
 
 486                                         part->data_sectors_per_block)
 
 490                 this_score += part->blocks[block].erases;
 
 492                 if (this_score < score) {
 
 498         if (best_block == -1)
 
 501         part->current_block = -1;
 
 502         part->reserved_block = best_block;
 
 504         pr_debug("reclaim_block: reclaiming block #%d with %d used "
 
 505                  "%d free sectors\n", best_block,
 
 506                  part->blocks[best_block].used_sectors,
 
 507                  part->blocks[best_block].free_sectors);
 
 509         if (part->blocks[best_block].used_sectors)
 
 510                 rc = move_block_contents(part, best_block, old_sector);
 
 512                 rc = erase_block(part, best_block);
 
 518  * IMPROVE: It would be best to choose the block with the most deleted sectors,
 
 519  * because if we fill that one up first it'll have the most chance of having
 
 520  * the least live sectors at reclaim.
 
 522 static int find_free_block(const struct partition *part)
 
 526         block = part->current_block == -1 ?
 
 527                         jiffies % part->total_blocks : part->current_block;
 
 531                 if (part->blocks[block].free_sectors &&
 
 532                                 block != part->reserved_block)
 
 535                 if (++block >= part->total_blocks)
 
 538         } while (block != stop);
 
 543 static int find_writeable_block(struct partition *part, u_long *old_sector)
 
 548         block = find_free_block(part);
 
 551                 if (!part->is_reclaiming) {
 
 552                         rc = reclaim_block(part, old_sector);
 
 556                         block = find_free_block(part);
 
 565         rc = part->mbd.mtd->read(part->mbd.mtd, part->blocks[block].offset,
 
 566                 part->header_size, &retlen, (u_char*)part->header_cache);
 
 568         if (!rc && retlen != part->header_size)
 
 572                 printk(KERN_NOTICE PREFIX "'%s': unable to read header at "
 
 573                                 "0x%lx\n", part->mbd.mtd->name,
 
 574                                 part->blocks[block].offset);
 
 578         part->current_block = block;
 
 584 static int mark_sector_deleted(struct partition *part, u_long old_addr)
 
 586         int block, offset, rc;
 
 589         u16 del = const_cpu_to_le16(SECTOR_DELETED);
 
 591         block = old_addr / part->block_size;
 
 592         offset = (old_addr % part->block_size) / SECTOR_SIZE -
 
 593                 part->header_sectors_per_block;
 
 595         addr = part->blocks[block].offset +
 
 596                         (HEADER_MAP_OFFSET + offset) * sizeof(u16);
 
 597         rc = part->mbd.mtd->write(part->mbd.mtd, addr,
 
 598                 sizeof(del), &retlen, (u_char*)&del);
 
 600         if (!rc && retlen != sizeof(del))
 
 604                 printk(KERN_WARNING PREFIX "error writing '%s' at "
 
 605                         "0x%lx\n", part->mbd.mtd->name, addr);
 
 609         if (block == part->current_block)
 
 610                 part->header_cache[offset + HEADER_MAP_OFFSET] = del;
 
 612         part->blocks[block].used_sectors--;
 
 614         if (!part->blocks[block].used_sectors &&
 
 615             !part->blocks[block].free_sectors)
 
 616                 rc = erase_block(part, block);
 
 622 static int find_free_sector(const struct partition *part, const struct block *block)
 
 626         i = stop = part->data_sectors_per_block - block->free_sectors;
 
 629                 if (le16_to_cpu(part->header_cache[HEADER_MAP_OFFSET + i])
 
 633                 if (++i == part->data_sectors_per_block)
 
 641 static int do_writesect(struct mtd_blktrans_dev *dev, u_long sector, char *buf, ulong *old_addr)
 
 643         struct partition *part = (struct partition*)dev;
 
 651         if (part->current_block == -1 ||
 
 652                 !part->blocks[part->current_block].free_sectors) {
 
 654                 rc = find_writeable_block(part, old_addr);
 
 659         block = &part->blocks[part->current_block];
 
 661         i = find_free_sector(part, block);
 
 668         addr = (i + part->header_sectors_per_block) * SECTOR_SIZE +
 
 670         rc = part->mbd.mtd->write(part->mbd.mtd,
 
 671                 addr, SECTOR_SIZE, &retlen, (u_char*)buf);
 
 673         if (!rc && retlen != SECTOR_SIZE)
 
 677                 printk(KERN_WARNING PREFIX "error writing '%s' at 0x%lx\n",
 
 678                                 part->mbd.mtd->name, addr);
 
 683         part->sector_map[sector] = addr;
 
 685         entry = cpu_to_le16(sector == 0 ? SECTOR_ZERO : sector);
 
 687         part->header_cache[i + HEADER_MAP_OFFSET] = entry;
 
 689         addr = block->offset + (HEADER_MAP_OFFSET + i) * sizeof(u16);
 
 690         rc = part->mbd.mtd->write(part->mbd.mtd, addr,
 
 691                         sizeof(entry), &retlen, (u_char*)&entry);
 
 693         if (!rc && retlen != sizeof(entry))
 
 697                 printk(KERN_WARNING PREFIX "error writing '%s' at 0x%lx\n",
 
 698                                 part->mbd.mtd->name, addr);
 
 702         block->used_sectors++;
 
 703         block->free_sectors--;
 
 709 static int rfd_ftl_writesect(struct mtd_blktrans_dev *dev, u_long sector, char *buf)
 
 711         struct partition *part = (struct partition*)dev;
 
 716         pr_debug("rfd_ftl_writesect(sector=0x%lx)\n", sector);
 
 718         if (part->reserved_block == -1) {
 
 723         if (sector >= part->sector_count) {
 
 728         old_addr = part->sector_map[sector];
 
 730         for (i=0; i<SECTOR_SIZE; i++) {
 
 734                 rc = do_writesect(dev, sector, buf, &old_addr);
 
 740         if (i == SECTOR_SIZE)
 
 741                 part->sector_map[sector] = -1;
 
 744                 rc = mark_sector_deleted(part, old_addr);
 
 750 static int rfd_ftl_getgeo(struct mtd_blktrans_dev *dev, struct hd_geometry *geo)
 
 752         struct partition *part = (struct partition*)dev;
 
 755         geo->sectors = SECTORS_PER_TRACK;
 
 756         geo->cylinders = part->cylinders;
 
 761 static void rfd_ftl_add_mtd(struct mtd_blktrans_ops *tr, struct mtd_info *mtd)
 
 763         struct partition *part;
 
 765         if (mtd->type != MTD_NORFLASH)
 
 768         part = kcalloc(1, sizeof(struct partition), GFP_KERNEL);
 
 775                 part->block_size = block_size;
 
 777                 if (!mtd->erasesize) {
 
 778                         printk(KERN_NOTICE PREFIX "please provide block_size");
 
 782                         part->block_size = mtd->erasesize;
 
 785         if (scan_header(part) == 0) {
 
 786                 part->mbd.size = part->sector_count;
 
 787                 part->mbd.blksize = SECTOR_SIZE;
 
 789                 part->mbd.devnum = -1;
 
 790                 if (!(mtd->flags & MTD_WRITEABLE))
 
 791                         part->mbd.readonly = 1;
 
 792                 else if (part->errors) {
 
 793                         printk(KERN_NOTICE PREFIX "'%s': errors found, "
 
 794                                         "setting read-only", mtd->name);
 
 795                         part->mbd.readonly = 1;
 
 798                 printk(KERN_INFO PREFIX "name: '%s' type: %d flags %x\n",
 
 799                                 mtd->name, mtd->type, mtd->flags);
 
 801                 if (!add_mtd_blktrans_dev((void*)part))
 
 808 static void rfd_ftl_remove_dev(struct mtd_blktrans_dev *dev)
 
 810         struct partition *part = (struct partition*)dev;
 
 813         for (i=0; i<part->total_blocks; i++) {
 
 814                 pr_debug("rfd_ftl_remove_dev:'%s': erase unit #%02d: %d erases\n",
 
 815                         part->mbd.mtd->name, i, part->blocks[i].erases);
 
 818         del_mtd_blktrans_dev(dev);
 
 819         vfree(part->sector_map);
 
 820         kfree(part->header_cache);
 
 825 struct mtd_blktrans_ops rfd_ftl_tr = {
 
 827         .major          = RFD_FTL_MAJOR,
 
 828         .part_bits      = PART_BITS,
 
 829         .readsect       = rfd_ftl_readsect,
 
 830         .writesect      = rfd_ftl_writesect,
 
 831         .getgeo         = rfd_ftl_getgeo,
 
 832         .add_mtd        = rfd_ftl_add_mtd,
 
 833         .remove_dev     = rfd_ftl_remove_dev,
 
 834         .owner          = THIS_MODULE,
 
 837 static int __init init_rfd_ftl(void)
 
 839         return register_mtd_blktrans(&rfd_ftl_tr);
 
 842 static void __exit cleanup_rfd_ftl(void)
 
 844         deregister_mtd_blktrans(&rfd_ftl_tr);
 
 847 module_init(init_rfd_ftl);
 
 848 module_exit(cleanup_rfd_ftl);
 
 850 MODULE_LICENSE("GPL");
 
 851 MODULE_AUTHOR("Sean Young <sean@mess.org>");
 
 852 MODULE_DESCRIPTION("Support code for RFD Flash Translation Layer, "
 
 853                 "used by General Software's Embedded BIOS");