219 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			219 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* vi: set sw=4 ts=4: */
 | 
						|
/*
 | 
						|
 * eraseall.c -- erase the whole of a MTD device
 | 
						|
 *
 | 
						|
 * Ported to busybox from mtd-utils.
 | 
						|
 *
 | 
						|
 * Copyright (C) 2000 Arcom Control System Ltd
 | 
						|
 *
 | 
						|
 * Renamed to flash_eraseall.c
 | 
						|
 *
 | 
						|
 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
 | 
						|
 */
 | 
						|
//config:config FLASH_ERASEALL
 | 
						|
//config:	bool "flash_eraseall (5.9 kb)"
 | 
						|
//config:	default n  # doesn't build on Ubuntu 8.04
 | 
						|
//config:	help
 | 
						|
//config:	The flash_eraseall binary from mtd-utils as of git head c4c6a59eb.
 | 
						|
//config:	This utility is used to erase the whole MTD device.
 | 
						|
 | 
						|
//applet:IF_FLASH_ERASEALL(APPLET(flash_eraseall, BB_DIR_USR_SBIN, BB_SUID_DROP))
 | 
						|
/* not NOEXEC: if flash operation stalls, use less memory in "hung" process */
 | 
						|
 | 
						|
//kbuild:lib-$(CONFIG_FLASH_ERASEALL) += flash_eraseall.o
 | 
						|
 | 
						|
//usage:#define flash_eraseall_trivial_usage
 | 
						|
//usage:       "[-jNq] MTD_DEVICE"
 | 
						|
//usage:#define flash_eraseall_full_usage "\n\n"
 | 
						|
//usage:       "Erase an MTD device\n"
 | 
						|
//usage:     "\n	-j	Format the device for jffs2"
 | 
						|
//usage:     "\n	-N	Don't skip bad blocks"
 | 
						|
//usage:     "\n	-q	Don't display progress messages"
 | 
						|
 | 
						|
#include "libbb.h"
 | 
						|
#include <mtd/mtd-user.h>
 | 
						|
#include <linux/jffs2.h>
 | 
						|
 | 
						|
#define OPTION_J  (1 << 0)
 | 
						|
#define OPTION_N  (1 << 1)
 | 
						|
#define OPTION_Q  (1 << 2)
 | 
						|
#define IS_NAND   (1 << 3)
 | 
						|
 | 
						|
/* mtd/jffs2-user.h used to have this atrocity:
 | 
						|
extern int target_endian;
 | 
						|
 | 
						|
#define t16(x) ({ __u16 __b = (x); (target_endian==__BYTE_ORDER)?__b:bswap_16(__b); })
 | 
						|
#define t32(x) ({ __u32 __b = (x); (target_endian==__BYTE_ORDER)?__b:bswap_32(__b); })
 | 
						|
 | 
						|
#define cpu_to_je16(x) ((jint16_t){t16(x)})
 | 
						|
#define cpu_to_je32(x) ((jint32_t){t32(x)})
 | 
						|
#define cpu_to_jemode(x) ((jmode_t){t32(x)})
 | 
						|
 | 
						|
#define je16_to_cpu(x) (t16((x).v16))
 | 
						|
#define je32_to_cpu(x) (t32((x).v32))
 | 
						|
#define jemode_to_cpu(x) (t32((x).m))
 | 
						|
 | 
						|
but mtd/jffs2-user.h is gone now (at least 2.6.31.6 does not have it anymore)
 | 
						|
*/
 | 
						|
 | 
						|
/* We always use native endianness */
 | 
						|
#undef cpu_to_je16
 | 
						|
#undef cpu_to_je32
 | 
						|
#define cpu_to_je16(v) ((jint16_t){(v)})
 | 
						|
#define cpu_to_je32(v) ((jint32_t){(v)})
 | 
						|
 | 
						|
static void show_progress(mtd_info_t *meminfo, erase_info_t *erase)
 | 
						|
{
 | 
						|
	printf("\rErasing %u Kibyte @ %x - %2u%% complete.",
 | 
						|
		(unsigned)meminfo->erasesize / 1024,
 | 
						|
		erase->start,
 | 
						|
		(unsigned) ((unsigned long long) erase->start * 100 / meminfo->size)
 | 
						|
	);
 | 
						|
	fflush_all();
 | 
						|
}
 | 
						|
 | 
						|
int flash_eraseall_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
 | 
						|
int flash_eraseall_main(int argc UNUSED_PARAM, char **argv)
 | 
						|
{
 | 
						|
	struct jffs2_unknown_node cleanmarker;
 | 
						|
	mtd_info_t meminfo;
 | 
						|
	int fd, clmpos, clmlen;
 | 
						|
	erase_info_t erase;
 | 
						|
	struct stat st;
 | 
						|
	unsigned int flags;
 | 
						|
	char *mtd_name;
 | 
						|
 | 
						|
	flags = getopt32(argv, "^" "jNq" "\0" "=1");
 | 
						|
 | 
						|
	mtd_name = argv[optind];
 | 
						|
	fd = xopen(mtd_name, O_RDWR);
 | 
						|
	fstat(fd, &st);
 | 
						|
	if (!S_ISCHR(st.st_mode))
 | 
						|
		bb_error_msg_and_die("%s: not a char device", mtd_name);
 | 
						|
 | 
						|
	xioctl(fd, MEMGETINFO, &meminfo);
 | 
						|
	erase.length = meminfo.erasesize;
 | 
						|
	if (meminfo.type == MTD_NANDFLASH)
 | 
						|
		flags |= IS_NAND;
 | 
						|
 | 
						|
	clmpos = 0;
 | 
						|
	clmlen = 8;
 | 
						|
	if (flags & OPTION_J) {
 | 
						|
		uint32_t *crc32_table;
 | 
						|
 | 
						|
		crc32_table = crc32_new_table_le();
 | 
						|
 | 
						|
		cleanmarker.magic = cpu_to_je16(JFFS2_MAGIC_BITMASK);
 | 
						|
		cleanmarker.nodetype = cpu_to_je16(JFFS2_NODETYPE_CLEANMARKER);
 | 
						|
		if (!(flags & IS_NAND))
 | 
						|
			cleanmarker.totlen = cpu_to_je32(sizeof(struct jffs2_unknown_node));
 | 
						|
		else {
 | 
						|
			struct nand_oobinfo oobinfo;
 | 
						|
 | 
						|
			xioctl(fd, MEMGETOOBSEL, &oobinfo);
 | 
						|
 | 
						|
			/* Check for autoplacement */
 | 
						|
			if (oobinfo.useecc == MTD_NANDECC_AUTOPLACE) {
 | 
						|
				/* Get the position of the free bytes */
 | 
						|
				clmpos = oobinfo.oobfree[0][0];
 | 
						|
				clmlen = oobinfo.oobfree[0][1];
 | 
						|
				if (clmlen > 8)
 | 
						|
					clmlen = 8;
 | 
						|
				if (clmlen == 0)
 | 
						|
					bb_error_msg_and_die("autoplacement selected and no empty space in oob");
 | 
						|
			} else {
 | 
						|
				/* Legacy mode */
 | 
						|
				switch (meminfo.oobsize) {
 | 
						|
				case 8:
 | 
						|
					clmpos = 6;
 | 
						|
					clmlen = 2;
 | 
						|
					break;
 | 
						|
				case 16:
 | 
						|
					clmpos = 8;
 | 
						|
					/*clmlen = 8;*/
 | 
						|
					break;
 | 
						|
				case 64:
 | 
						|
					clmpos = 16;
 | 
						|
					/*clmlen = 8;*/
 | 
						|
					break;
 | 
						|
				}
 | 
						|
			}
 | 
						|
			cleanmarker.totlen = cpu_to_je32(8);
 | 
						|
		}
 | 
						|
 | 
						|
		cleanmarker.hdr_crc = cpu_to_je32(
 | 
						|
			crc32_block_endian0(0, &cleanmarker, sizeof(struct jffs2_unknown_node) - 4, crc32_table)
 | 
						|
		);
 | 
						|
	}
 | 
						|
 | 
						|
	/* Don't want to destroy progress indicator by bb_error_msg's */
 | 
						|
	applet_name = xasprintf("\n%s: %s", applet_name, mtd_name);
 | 
						|
 | 
						|
	for (erase.start = 0; erase.start < meminfo.size;
 | 
						|
	     erase.start += meminfo.erasesize) {
 | 
						|
		if (!(flags & OPTION_N)) {
 | 
						|
			int ret;
 | 
						|
			loff_t offset = erase.start;
 | 
						|
 | 
						|
			ret = ioctl(fd, MEMGETBADBLOCK, &offset);
 | 
						|
			if (ret > 0) {
 | 
						|
				if (!(flags & OPTION_Q))
 | 
						|
					printf("\nSkipping bad block at 0x%08x\n", erase.start);
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			if (ret < 0) {
 | 
						|
				/* Black block table is not available on certain flash
 | 
						|
				 * types e.g. NOR
 | 
						|
				 */
 | 
						|
				if (errno == EOPNOTSUPP) {
 | 
						|
					flags |= OPTION_N;
 | 
						|
					if (flags & IS_NAND)
 | 
						|
						bb_error_msg_and_die("bad block check not available");
 | 
						|
				} else {
 | 
						|
					bb_perror_msg_and_die("MEMGETBADBLOCK error");
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (!(flags & OPTION_Q))
 | 
						|
			show_progress(&meminfo, &erase);
 | 
						|
 | 
						|
		xioctl(fd, MEMERASE, &erase);
 | 
						|
 | 
						|
		/* format for JFFS2 ? */
 | 
						|
		if (!(flags & OPTION_J))
 | 
						|
			continue;
 | 
						|
 | 
						|
		/* write cleanmarker */
 | 
						|
		if (flags & IS_NAND) {
 | 
						|
			struct mtd_oob_buf oob;
 | 
						|
 | 
						|
			oob.ptr = (unsigned char *) &cleanmarker;
 | 
						|
			oob.start = erase.start + clmpos;
 | 
						|
			oob.length = clmlen;
 | 
						|
			xioctl(fd, MEMWRITEOOB, &oob);
 | 
						|
		} else {
 | 
						|
			xlseek(fd, erase.start, SEEK_SET);
 | 
						|
			/* if (lseek(fd, erase.start, SEEK_SET) < 0) {
 | 
						|
				bb_perror_msg("MTD %s failure", "seek");
 | 
						|
				continue;
 | 
						|
			} */
 | 
						|
			xwrite(fd, &cleanmarker, sizeof(cleanmarker));
 | 
						|
			/* if (write(fd, &cleanmarker, sizeof(cleanmarker)) != sizeof(cleanmarker)) {
 | 
						|
				bb_perror_msg("MTD %s failure", "write");
 | 
						|
				continue;
 | 
						|
			} */
 | 
						|
		}
 | 
						|
		if (!(flags & OPTION_Q))
 | 
						|
			printf(" Cleanmarker written at %x.", erase.start);
 | 
						|
	}
 | 
						|
	if (!(flags & OPTION_Q)) {
 | 
						|
		show_progress(&meminfo, &erase);
 | 
						|
		bb_putchar('\n');
 | 
						|
	}
 | 
						|
 | 
						|
	if (ENABLE_FEATURE_CLEAN_UP)
 | 
						|
		close(fd);
 | 
						|
	return EXIT_SUCCESS;
 | 
						|
}
 |