2 * YAFFS: Yet another FFS. A NAND-flash specific file system.
3 * mkyaffs.c Format a chunk of NAND for YAFFS.
5 * Copyright (C) 2002 Aleph One Ltd.
6 * for Toby Churchill Ltd and Brightstar Engineering
8 * Created by Charles Manning <charles@aleph1.co.uk>
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License version 2 as
12 * published by the Free Software Foundation.
15 * This file is crafted from nandtest.c by Miguel Freitas (miguel@cetuc.puc-rio.br)
16 * and Steven J. Hill (sjhill@cotw.com)
19 * Formatting a YAFFS device is very simple. Just erase all undamaged blocks.
20 * NB Don't erase blocks maked as damaged.
31 #include <sys/ioctl.h>
32 #include <sys/types.h>
33 #include <asm/types.h>
34 #include <linux/config.h>
35 #include <linux/mtd/mtd.h>
37 const char *mkyaffs_c_version = "$Id: mkyaffs.c,v 1.6 2003-03-11 05:16:53 charles Exp $";
39 // countBits is a quick way of counting the number of bits in a byte.
40 // ie. countBits[n] holds the number of 1 bits in a byte with the value n.
42 static const char countBits[256] =
44 0,1,1,2,1,2,2,3,1,2,2,3,2,3,3,4,
45 1,2,2,3,2,3,3,4,2,3,3,4,3,4,4,5,
46 1,2,2,3,2,3,3,4,2,3,3,4,3,4,4,5,
47 2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,
48 1,2,2,3,2,3,3,4,2,3,3,4,3,4,4,5,
49 2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,
50 2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,
51 3,4,4,5,4,5,5,6,4,5,5,6,5,6,6,7,
52 1,2,2,3,2,3,3,4,2,3,3,4,3,4,4,5,
53 2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,
54 2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,
55 3,4,4,5,4,5,5,6,4,5,5,6,5,6,6,7,
56 2,3,3,4,3,4,4,5,3,4,4,5,4,5,5,6,
57 3,4,4,5,4,5,5,6,4,5,5,6,5,6,6,7,
58 3,4,4,5,4,5,5,6,4,5,5,6,5,6,6,7,
59 4,5,5,6,5,6,6,7,5,6,6,7,6,7,7,8
63 * Buffer arrays used for running tests
66 unsigned char oobbuf[16];
67 unsigned char imgpage[528];
73 struct nand_oobinfo yaffs_oobinfo = {
75 eccpos: {8, 9, 10, 13, 14, 15}
78 struct nand_oobinfo yaffs_noeccinfo = {
86 int main(int argc, char **argv)
95 struct mtd_oob_buf oob = {0, 16, (unsigned char *) &oobbuf};
98 struct nand_oobinfo oobsel;
100 if (strcmp (argv[optcnt], "-e") == 0) {
105 /* Make sure a device was specified */
106 if(argc < (optcnt + 2)) {
107 printf("usage: %s -e <mtdname> <image name>\n", argv[0]);
111 if((img = open(argv[optcnt + 1],O_RDONLY)) == -1) {
112 perror("opening image file");
117 imglen = lseek(img,0,SEEK_END);
119 printf("Image not a multiple of 528 bytes\n");
124 lseek(img,0,SEEK_SET);
126 /* Open the device */
127 if((fd = open(argv[optcnt], O_RDWR)) == -1) {
128 perror("opening flash");
132 /* Fill in MTD device capability structure */
133 if(ioctl(fd, MEMGETINFO, &meminfo) != 0) {
134 perror("MEMGETINFO");
139 // set the appropriate oob layout selector
140 oobsel = usemtdecc ? yaffs_oobinfo : yaffs_noeccinfo;
141 if (ioctl (fd, MEMSETOOBSEL, &oobsel) != 0) {
142 perror ("MEMSETOOBSEL");
147 /* Make sure device page sizes are valid */
148 if( !(meminfo.oobsize == 16 && meminfo.oobblock == 512))
150 printf("Unknown flash (not normal NAND)\n");
156 (imglen/528 +32)*512 > meminfo.size){
157 printf("Image is too big for NAND\n");
162 printf("Erasing and programming NAND\n");
163 for(addr = 0; addr < meminfo.size; addr += meminfo.erasesize)
165 /* Read the OOB data to determine if the block is valid.
166 * If the block is damaged, then byte 5 of the OOB data will
167 * have at least 2 zero bits.
172 if (ioctl(fd, MEMREADOOB, &oob) != 0)
174 perror("ioctl(MEMREADOOB)");
179 if(countBits[oobbuf[5]] < 7)
181 printf("Block at 0x08%lx is damaged and is not being formatted\n",addr);
185 /* Erase this block */
187 erase.length = meminfo.erasesize;
188 printf("Erasing block at 0x08%lx\n",addr);
189 if(ioctl(fd, MEMERASE, &erase) != 0)
191 perror("\nMTD Erase failure\n");
196 /* Do some programming, but not in the first block */
199 for(offset = 0; offset <meminfo.erasesize; offset+=512)
201 if(read(img,imgpage,528) == 528){
203 imgpage[512+8] = 0xff;
204 imgpage[512+9] = 0xff;
205 imgpage[512+10] = 0xff;
206 imgpage[512+13] = 0xff;
207 imgpage[512+14] = 0xff;
208 imgpage[512+15] = 0xff;
210 oob.start = addr+offset;
212 oob.ptr=&imgpage[512];
213 ioctl(fd,MEMWRITEOOB,&oob);
215 lseek(fd,addr+offset,SEEK_SET);
216 write(fd,imgpage,512);
227 /* All the tests succeeded */