1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
|
/*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; version 2.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*/
#include <common.h>
#include <command.h>
#include <malloc.h>
#include <getopt.h>
#include <fs.h>
#include <fcntl.h>
#include <ioctl.h>
#include <linux/mtd/mtd.h>
#include <mtd/mtd-peb.h>
static int do_nand_bitflip(int argc, char *argv[])
{
int opt, ret, fd;
static struct mtd_info_user meminfo;
int block = 0;
int random = 0;
int num_bitflips = 1;
loff_t offset = 0, roffset;
int check = 0;
size_t r;
void *buf;
while ((opt = getopt(argc, argv, "b:rn:o:c")) > 0) {
switch (opt) {
case 'r':
random = 1;
break;
case 'n':
num_bitflips = simple_strtoul(optarg, NULL, 0);
break;
case 'b':
block = simple_strtoul(optarg, NULL, 0);
break;
case 'o':
offset = simple_strtoull(optarg, NULL, 0);
break;
case 'c':
check = 1;
break;
default:
return COMMAND_ERROR_USAGE;
}
}
if (optind >= argc)
return COMMAND_ERROR_USAGE;
fd = open(argv[optind], O_RDWR);
if (fd < 0)
return fd;
ret = ioctl(fd, MEMGETINFO, &meminfo);
close(fd);
if (ret)
return ret;
block += mtd_div_by_eb(offset, meminfo.mtd);
offset = mtd_mod_by_eb(offset, meminfo.mtd);
if ((int)offset % meminfo.mtd->writesize) {
printf("offset has to be pagesize aligned\n");
return 1;
}
if (!check) {
ret = mtd_peb_create_bitflips(meminfo.mtd, block, offset, meminfo.writesize,
num_bitflips, random, 1);
if (ret) {
printf("Creating bitflips failed with: %s\n", strerror(-ret));
return ret;
}
}
buf = xzalloc(meminfo.writesize);
roffset = (loff_t)block * meminfo.mtd->erasesize + offset;
ret = meminfo.mtd->read(meminfo.mtd, roffset, meminfo.writesize, &r, buf);
if (ret > 0) {
printf("page at block %d, offset 0x%08llx has %d bitflips%s\n",
block, offset, ret,
ret >= meminfo.mtd->bitflip_threshold ? ", needs cleanup" : "");
} else if (!ret) {
printf("No bitflips found on block %d, offset 0x%08llx\n", block, offset);
} else {
printf("Reading block %d, offset 0x%08llx failed with: %s\n", block, offset,
strerror(-ret));
}
free(buf);
return 0;
}
BAREBOX_CMD_HELP_START(nand_bitflip)
BAREBOX_CMD_HELP_TEXT("This command creates bitflips on Nand pages.")
BAREBOX_CMD_HELP_TEXT("Options:")
BAREBOX_CMD_HELP_OPT ("-b <block>", "block to work on")
BAREBOX_CMD_HELP_OPT ("-c\t", "Check only for bitflips")
BAREBOX_CMD_HELP_OPT ("-o <offset>", "offset in Nand")
BAREBOX_CMD_HELP_OPT ("-r\t", "flip random bits")
BAREBOX_CMD_HELP_OPT ("-n <numbitflips>", "Specify maximum number of bitflips to generate")
BAREBOX_CMD_HELP_END
BAREBOX_CMD_START(nand_bitflip)
.cmd = do_nand_bitflip,
BAREBOX_CMD_DESC("Create bitflips on Nand pages")
BAREBOX_CMD_OPTS("NANDDEV")
BAREBOX_CMD_GROUP(CMD_GRP_HWMANIP)
BAREBOX_CMD_HELP(cmd_nand_bitflip_help)
BAREBOX_CMD_END
|