1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun * Allwinner NAND randomizer and image builder implementation:
3*4882a593Smuzhiyun *
4*4882a593Smuzhiyun * Copyright © 2016 NextThing Co.
5*4882a593Smuzhiyun * Copyright © 2016 Free Electrons
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Author: Boris Brezillon <boris.brezillon@free-electrons.com>
8*4882a593Smuzhiyun *
9*4882a593Smuzhiyun */
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun #include <linux/bch.h>
12*4882a593Smuzhiyun
13*4882a593Smuzhiyun #include <getopt.h>
14*4882a593Smuzhiyun #include <version.h>
15*4882a593Smuzhiyun
16*4882a593Smuzhiyun #define BCH_PRIMITIVE_POLY 0x5803
17*4882a593Smuzhiyun
18*4882a593Smuzhiyun #define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0]))
19*4882a593Smuzhiyun #define DIV_ROUND_UP(n,d) (((n) + (d) - 1) / (d))
20*4882a593Smuzhiyun
21*4882a593Smuzhiyun struct image_info {
22*4882a593Smuzhiyun int ecc_strength;
23*4882a593Smuzhiyun int ecc_step_size;
24*4882a593Smuzhiyun int page_size;
25*4882a593Smuzhiyun int oob_size;
26*4882a593Smuzhiyun int usable_page_size;
27*4882a593Smuzhiyun int eraseblock_size;
28*4882a593Smuzhiyun int scramble;
29*4882a593Smuzhiyun int boot0;
30*4882a593Smuzhiyun off_t offset;
31*4882a593Smuzhiyun const char *source;
32*4882a593Smuzhiyun const char *dest;
33*4882a593Smuzhiyun };
34*4882a593Smuzhiyun
swap_bits(uint8_t * buf,int len)35*4882a593Smuzhiyun static void swap_bits(uint8_t *buf, int len)
36*4882a593Smuzhiyun {
37*4882a593Smuzhiyun int i, j;
38*4882a593Smuzhiyun
39*4882a593Smuzhiyun for (j = 0; j < len; j++) {
40*4882a593Smuzhiyun uint8_t byte = buf[j];
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun buf[j] = 0;
43*4882a593Smuzhiyun for (i = 0; i < 8; i++) {
44*4882a593Smuzhiyun if (byte & (1 << i))
45*4882a593Smuzhiyun buf[j] |= (1 << (7 - i));
46*4882a593Smuzhiyun }
47*4882a593Smuzhiyun }
48*4882a593Smuzhiyun }
49*4882a593Smuzhiyun
lfsr_step(uint16_t state,int count)50*4882a593Smuzhiyun static uint16_t lfsr_step(uint16_t state, int count)
51*4882a593Smuzhiyun {
52*4882a593Smuzhiyun state &= 0x7fff;
53*4882a593Smuzhiyun while (count--)
54*4882a593Smuzhiyun state = ((state >> 1) |
55*4882a593Smuzhiyun ((((state >> 0) ^ (state >> 1)) & 1) << 14)) & 0x7fff;
56*4882a593Smuzhiyun
57*4882a593Smuzhiyun return state;
58*4882a593Smuzhiyun }
59*4882a593Smuzhiyun
60*4882a593Smuzhiyun static uint16_t default_scrambler_seeds[] = {
61*4882a593Smuzhiyun 0x2b75, 0x0bd0, 0x5ca3, 0x62d1, 0x1c93, 0x07e9, 0x2162, 0x3a72,
62*4882a593Smuzhiyun 0x0d67, 0x67f9, 0x1be7, 0x077d, 0x032f, 0x0dac, 0x2716, 0x2436,
63*4882a593Smuzhiyun 0x7922, 0x1510, 0x3860, 0x5287, 0x480f, 0x4252, 0x1789, 0x5a2d,
64*4882a593Smuzhiyun 0x2a49, 0x5e10, 0x437f, 0x4b4e, 0x2f45, 0x216e, 0x5cb7, 0x7130,
65*4882a593Smuzhiyun 0x2a3f, 0x60e4, 0x4dc9, 0x0ef0, 0x0f52, 0x1bb9, 0x6211, 0x7a56,
66*4882a593Smuzhiyun 0x226d, 0x4ea7, 0x6f36, 0x3692, 0x38bf, 0x0c62, 0x05eb, 0x4c55,
67*4882a593Smuzhiyun 0x60f4, 0x728c, 0x3b6f, 0x2037, 0x7f69, 0x0936, 0x651a, 0x4ceb,
68*4882a593Smuzhiyun 0x6218, 0x79f3, 0x383f, 0x18d9, 0x4f05, 0x5c82, 0x2912, 0x6f17,
69*4882a593Smuzhiyun 0x6856, 0x5938, 0x1007, 0x61ab, 0x3e7f, 0x57c2, 0x542f, 0x4f62,
70*4882a593Smuzhiyun 0x7454, 0x2eac, 0x7739, 0x42d4, 0x2f90, 0x435a, 0x2e52, 0x2064,
71*4882a593Smuzhiyun 0x637c, 0x66ad, 0x2c90, 0x0bad, 0x759c, 0x0029, 0x0986, 0x7126,
72*4882a593Smuzhiyun 0x1ca7, 0x1605, 0x386a, 0x27f5, 0x1380, 0x6d75, 0x24c3, 0x0f8e,
73*4882a593Smuzhiyun 0x2b7a, 0x1418, 0x1fd1, 0x7dc1, 0x2d8e, 0x43af, 0x2267, 0x7da3,
74*4882a593Smuzhiyun 0x4e3d, 0x1338, 0x50db, 0x454d, 0x764d, 0x40a3, 0x42e6, 0x262b,
75*4882a593Smuzhiyun 0x2d2e, 0x1aea, 0x2e17, 0x173d, 0x3a6e, 0x71bf, 0x25f9, 0x0a5d,
76*4882a593Smuzhiyun 0x7c57, 0x0fbe, 0x46ce, 0x4939, 0x6b17, 0x37bb, 0x3e91, 0x76db,
77*4882a593Smuzhiyun };
78*4882a593Smuzhiyun
79*4882a593Smuzhiyun static uint16_t brom_scrambler_seeds[] = { 0x4a80 };
80*4882a593Smuzhiyun
scramble(const struct image_info * info,int page,uint8_t * data,int datalen)81*4882a593Smuzhiyun static void scramble(const struct image_info *info,
82*4882a593Smuzhiyun int page, uint8_t *data, int datalen)
83*4882a593Smuzhiyun {
84*4882a593Smuzhiyun uint16_t state;
85*4882a593Smuzhiyun int i;
86*4882a593Smuzhiyun
87*4882a593Smuzhiyun /* Boot0 is always scrambled no matter the command line option. */
88*4882a593Smuzhiyun if (info->boot0) {
89*4882a593Smuzhiyun state = brom_scrambler_seeds[0];
90*4882a593Smuzhiyun } else {
91*4882a593Smuzhiyun unsigned seedmod = info->eraseblock_size / info->page_size;
92*4882a593Smuzhiyun
93*4882a593Smuzhiyun /* Bail out earlier if the user didn't ask for scrambling. */
94*4882a593Smuzhiyun if (!info->scramble)
95*4882a593Smuzhiyun return;
96*4882a593Smuzhiyun
97*4882a593Smuzhiyun if (seedmod > ARRAY_SIZE(default_scrambler_seeds))
98*4882a593Smuzhiyun seedmod = ARRAY_SIZE(default_scrambler_seeds);
99*4882a593Smuzhiyun
100*4882a593Smuzhiyun state = default_scrambler_seeds[page % seedmod];
101*4882a593Smuzhiyun }
102*4882a593Smuzhiyun
103*4882a593Smuzhiyun /* Prepare the initial state... */
104*4882a593Smuzhiyun state = lfsr_step(state, 15);
105*4882a593Smuzhiyun
106*4882a593Smuzhiyun /* and start scrambling data. */
107*4882a593Smuzhiyun for (i = 0; i < datalen; i++) {
108*4882a593Smuzhiyun data[i] ^= state;
109*4882a593Smuzhiyun state = lfsr_step(state, 8);
110*4882a593Smuzhiyun }
111*4882a593Smuzhiyun }
112*4882a593Smuzhiyun
write_page(const struct image_info * info,uint8_t * buffer,FILE * src,FILE * rnd,FILE * dst,struct bch_control * bch,int page)113*4882a593Smuzhiyun static int write_page(const struct image_info *info, uint8_t *buffer,
114*4882a593Smuzhiyun FILE *src, FILE *rnd, FILE *dst,
115*4882a593Smuzhiyun struct bch_control *bch, int page)
116*4882a593Smuzhiyun {
117*4882a593Smuzhiyun int steps = info->usable_page_size / info->ecc_step_size;
118*4882a593Smuzhiyun int eccbytes = DIV_ROUND_UP(info->ecc_strength * 14, 8);
119*4882a593Smuzhiyun off_t pos = ftell(dst);
120*4882a593Smuzhiyun size_t pad, cnt;
121*4882a593Smuzhiyun int i;
122*4882a593Smuzhiyun
123*4882a593Smuzhiyun if (eccbytes % 2)
124*4882a593Smuzhiyun eccbytes++;
125*4882a593Smuzhiyun
126*4882a593Smuzhiyun memset(buffer, 0xff, info->page_size + info->oob_size);
127*4882a593Smuzhiyun cnt = fread(buffer, 1, info->usable_page_size, src);
128*4882a593Smuzhiyun if (!cnt) {
129*4882a593Smuzhiyun if (!feof(src)) {
130*4882a593Smuzhiyun fprintf(stderr,
131*4882a593Smuzhiyun "Failed to read data from the source\n");
132*4882a593Smuzhiyun return -1;
133*4882a593Smuzhiyun } else {
134*4882a593Smuzhiyun return 0;
135*4882a593Smuzhiyun }
136*4882a593Smuzhiyun }
137*4882a593Smuzhiyun
138*4882a593Smuzhiyun fwrite(buffer, info->page_size + info->oob_size, 1, dst);
139*4882a593Smuzhiyun
140*4882a593Smuzhiyun for (i = 0; i < info->usable_page_size; i++) {
141*4882a593Smuzhiyun if (buffer[i] != 0xff)
142*4882a593Smuzhiyun break;
143*4882a593Smuzhiyun }
144*4882a593Smuzhiyun
145*4882a593Smuzhiyun /* We leave empty pages at 0xff. */
146*4882a593Smuzhiyun if (i == info->usable_page_size)
147*4882a593Smuzhiyun return 0;
148*4882a593Smuzhiyun
149*4882a593Smuzhiyun /* Restore the source pointer to read it again. */
150*4882a593Smuzhiyun fseek(src, -cnt, SEEK_CUR);
151*4882a593Smuzhiyun
152*4882a593Smuzhiyun /* Randomize unused space if scrambling is required. */
153*4882a593Smuzhiyun if (info->scramble) {
154*4882a593Smuzhiyun int offs;
155*4882a593Smuzhiyun
156*4882a593Smuzhiyun if (info->boot0) {
157*4882a593Smuzhiyun size_t ret;
158*4882a593Smuzhiyun
159*4882a593Smuzhiyun offs = steps * (info->ecc_step_size + eccbytes + 4);
160*4882a593Smuzhiyun cnt = info->page_size + info->oob_size - offs;
161*4882a593Smuzhiyun ret = fread(buffer + offs, 1, cnt, rnd);
162*4882a593Smuzhiyun if (!ret && !feof(rnd)) {
163*4882a593Smuzhiyun fprintf(stderr,
164*4882a593Smuzhiyun "Failed to read random data\n");
165*4882a593Smuzhiyun return -1;
166*4882a593Smuzhiyun }
167*4882a593Smuzhiyun } else {
168*4882a593Smuzhiyun offs = info->page_size + (steps * (eccbytes + 4));
169*4882a593Smuzhiyun cnt = info->page_size + info->oob_size - offs;
170*4882a593Smuzhiyun memset(buffer + offs, 0xff, cnt);
171*4882a593Smuzhiyun scramble(info, page, buffer + offs, cnt);
172*4882a593Smuzhiyun }
173*4882a593Smuzhiyun fseek(dst, pos + offs, SEEK_SET);
174*4882a593Smuzhiyun fwrite(buffer + offs, cnt, 1, dst);
175*4882a593Smuzhiyun }
176*4882a593Smuzhiyun
177*4882a593Smuzhiyun for (i = 0; i < steps; i++) {
178*4882a593Smuzhiyun int ecc_offs, data_offs;
179*4882a593Smuzhiyun uint8_t *ecc;
180*4882a593Smuzhiyun
181*4882a593Smuzhiyun memset(buffer, 0xff, info->ecc_step_size + eccbytes + 4);
182*4882a593Smuzhiyun ecc = buffer + info->ecc_step_size + 4;
183*4882a593Smuzhiyun if (info->boot0) {
184*4882a593Smuzhiyun data_offs = i * (info->ecc_step_size + eccbytes + 4);
185*4882a593Smuzhiyun ecc_offs = data_offs + info->ecc_step_size + 4;
186*4882a593Smuzhiyun } else {
187*4882a593Smuzhiyun data_offs = i * info->ecc_step_size;
188*4882a593Smuzhiyun ecc_offs = info->page_size + 4 + (i * (eccbytes + 4));
189*4882a593Smuzhiyun }
190*4882a593Smuzhiyun
191*4882a593Smuzhiyun cnt = fread(buffer, 1, info->ecc_step_size, src);
192*4882a593Smuzhiyun if (!cnt && !feof(src)) {
193*4882a593Smuzhiyun fprintf(stderr,
194*4882a593Smuzhiyun "Failed to read data from the source\n");
195*4882a593Smuzhiyun return -1;
196*4882a593Smuzhiyun }
197*4882a593Smuzhiyun
198*4882a593Smuzhiyun pad = info->ecc_step_size - cnt;
199*4882a593Smuzhiyun if (pad) {
200*4882a593Smuzhiyun if (info->scramble && info->boot0) {
201*4882a593Smuzhiyun size_t ret;
202*4882a593Smuzhiyun
203*4882a593Smuzhiyun ret = fread(buffer + cnt, 1, pad, rnd);
204*4882a593Smuzhiyun if (!ret && !feof(rnd)) {
205*4882a593Smuzhiyun fprintf(stderr,
206*4882a593Smuzhiyun "Failed to read random data\n");
207*4882a593Smuzhiyun return -1;
208*4882a593Smuzhiyun }
209*4882a593Smuzhiyun } else {
210*4882a593Smuzhiyun memset(buffer + cnt, 0xff, pad);
211*4882a593Smuzhiyun }
212*4882a593Smuzhiyun }
213*4882a593Smuzhiyun
214*4882a593Smuzhiyun memset(ecc, 0, eccbytes);
215*4882a593Smuzhiyun swap_bits(buffer, info->ecc_step_size + 4);
216*4882a593Smuzhiyun encode_bch(bch, buffer, info->ecc_step_size + 4, ecc);
217*4882a593Smuzhiyun swap_bits(buffer, info->ecc_step_size + 4);
218*4882a593Smuzhiyun swap_bits(ecc, eccbytes);
219*4882a593Smuzhiyun scramble(info, page, buffer, info->ecc_step_size + 4 + eccbytes);
220*4882a593Smuzhiyun
221*4882a593Smuzhiyun fseek(dst, pos + data_offs, SEEK_SET);
222*4882a593Smuzhiyun fwrite(buffer, info->ecc_step_size, 1, dst);
223*4882a593Smuzhiyun fseek(dst, pos + ecc_offs - 4, SEEK_SET);
224*4882a593Smuzhiyun fwrite(ecc - 4, eccbytes + 4, 1, dst);
225*4882a593Smuzhiyun }
226*4882a593Smuzhiyun
227*4882a593Smuzhiyun /* Fix BBM. */
228*4882a593Smuzhiyun fseek(dst, pos + info->page_size, SEEK_SET);
229*4882a593Smuzhiyun memset(buffer, 0xff, 2);
230*4882a593Smuzhiyun fwrite(buffer, 2, 1, dst);
231*4882a593Smuzhiyun
232*4882a593Smuzhiyun /* Make dst pointer point to the next page. */
233*4882a593Smuzhiyun fseek(dst, pos + info->page_size + info->oob_size, SEEK_SET);
234*4882a593Smuzhiyun
235*4882a593Smuzhiyun return 0;
236*4882a593Smuzhiyun }
237*4882a593Smuzhiyun
create_image(const struct image_info * info)238*4882a593Smuzhiyun static int create_image(const struct image_info *info)
239*4882a593Smuzhiyun {
240*4882a593Smuzhiyun off_t page = info->offset / info->page_size;
241*4882a593Smuzhiyun struct bch_control *bch;
242*4882a593Smuzhiyun FILE *src, *dst, *rnd;
243*4882a593Smuzhiyun uint8_t *buffer;
244*4882a593Smuzhiyun
245*4882a593Smuzhiyun bch = init_bch(14, info->ecc_strength, BCH_PRIMITIVE_POLY);
246*4882a593Smuzhiyun if (!bch) {
247*4882a593Smuzhiyun fprintf(stderr, "Failed to init the BCH engine\n");
248*4882a593Smuzhiyun return -1;
249*4882a593Smuzhiyun }
250*4882a593Smuzhiyun
251*4882a593Smuzhiyun buffer = malloc(info->page_size + info->oob_size);
252*4882a593Smuzhiyun if (!buffer) {
253*4882a593Smuzhiyun fprintf(stderr, "Failed to allocate the NAND page buffer\n");
254*4882a593Smuzhiyun return -1;
255*4882a593Smuzhiyun }
256*4882a593Smuzhiyun
257*4882a593Smuzhiyun memset(buffer, 0xff, info->page_size + info->oob_size);
258*4882a593Smuzhiyun
259*4882a593Smuzhiyun src = fopen(info->source, "r");
260*4882a593Smuzhiyun if (!src) {
261*4882a593Smuzhiyun fprintf(stderr, "Failed to open source file (%s)\n",
262*4882a593Smuzhiyun info->source);
263*4882a593Smuzhiyun return -1;
264*4882a593Smuzhiyun }
265*4882a593Smuzhiyun
266*4882a593Smuzhiyun dst = fopen(info->dest, "w");
267*4882a593Smuzhiyun if (!dst) {
268*4882a593Smuzhiyun fprintf(stderr, "Failed to open dest file (%s)\n", info->dest);
269*4882a593Smuzhiyun return -1;
270*4882a593Smuzhiyun }
271*4882a593Smuzhiyun
272*4882a593Smuzhiyun rnd = fopen("/dev/urandom", "r");
273*4882a593Smuzhiyun if (!rnd) {
274*4882a593Smuzhiyun fprintf(stderr, "Failed to open /dev/urandom\n");
275*4882a593Smuzhiyun return -1;
276*4882a593Smuzhiyun }
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun while (!feof(src)) {
279*4882a593Smuzhiyun int ret;
280*4882a593Smuzhiyun
281*4882a593Smuzhiyun ret = write_page(info, buffer, src, rnd, dst, bch, page++);
282*4882a593Smuzhiyun if (ret)
283*4882a593Smuzhiyun return ret;
284*4882a593Smuzhiyun }
285*4882a593Smuzhiyun
286*4882a593Smuzhiyun return 0;
287*4882a593Smuzhiyun }
288*4882a593Smuzhiyun
display_help(int status)289*4882a593Smuzhiyun static void display_help(int status)
290*4882a593Smuzhiyun {
291*4882a593Smuzhiyun fprintf(status == EXIT_SUCCESS ? stdout : stderr,
292*4882a593Smuzhiyun "sunxi-nand-image-builder %s\n"
293*4882a593Smuzhiyun "\n"
294*4882a593Smuzhiyun "Usage: sunxi-nand-image-builder [OPTIONS] source-image output-image\n"
295*4882a593Smuzhiyun "\n"
296*4882a593Smuzhiyun "Creates a raw NAND image that can be read by the sunxi NAND controller.\n"
297*4882a593Smuzhiyun "\n"
298*4882a593Smuzhiyun "-h --help Display this help and exit\n"
299*4882a593Smuzhiyun "-c <str>/<step> --ecc=<str>/<step> ECC config (strength/step-size)\n"
300*4882a593Smuzhiyun "-p <size> --page=<size> Page size\n"
301*4882a593Smuzhiyun "-o <size> --oob=<size> OOB size\n"
302*4882a593Smuzhiyun "-u <size> --usable=<size> Usable page size\n"
303*4882a593Smuzhiyun "-e <size> --eraseblock=<size> Erase block size\n"
304*4882a593Smuzhiyun "-b --boot0 Build a boot0 image.\n"
305*4882a593Smuzhiyun "-s --scramble Scramble data\n"
306*4882a593Smuzhiyun "-a <offset> --address=<offset> Where the image will be programmed.\n"
307*4882a593Smuzhiyun "\n"
308*4882a593Smuzhiyun "Notes:\n"
309*4882a593Smuzhiyun "All the information you need to pass to this tool should be part of\n"
310*4882a593Smuzhiyun "the NAND datasheet.\n"
311*4882a593Smuzhiyun "\n"
312*4882a593Smuzhiyun "The NAND controller only supports the following ECC configs\n"
313*4882a593Smuzhiyun " Valid ECC strengths: 16, 24, 28, 32, 40, 48, 56, 60 and 64\n"
314*4882a593Smuzhiyun " Valid ECC step size: 512 and 1024\n"
315*4882a593Smuzhiyun "\n"
316*4882a593Smuzhiyun "If you are building a boot0 image, you'll have specify extra options.\n"
317*4882a593Smuzhiyun "These options should be chosen based on the layouts described here:\n"
318*4882a593Smuzhiyun " http://linux-sunxi.org/NAND#More_information_on_BROM_NAND\n"
319*4882a593Smuzhiyun "\n"
320*4882a593Smuzhiyun " --usable should be assigned the 'Hardware page' value\n"
321*4882a593Smuzhiyun " --ecc should be assigned the 'ECC capacity'/'ECC page' values\n"
322*4882a593Smuzhiyun " --usable should be smaller than --page\n"
323*4882a593Smuzhiyun "\n"
324*4882a593Smuzhiyun "The --address option is only required for non-boot0 images that are \n"
325*4882a593Smuzhiyun "meant to be programmed at a non eraseblock aligned offset.\n"
326*4882a593Smuzhiyun "\n"
327*4882a593Smuzhiyun "Examples:\n"
328*4882a593Smuzhiyun " The H27UCG8T2BTR-BC NAND exposes\n"
329*4882a593Smuzhiyun " * 16k pages\n"
330*4882a593Smuzhiyun " * 1280 OOB bytes per page\n"
331*4882a593Smuzhiyun " * 4M eraseblocks\n"
332*4882a593Smuzhiyun " * requires data scrambling\n"
333*4882a593Smuzhiyun " * expects a minimum ECC of 40bits/1024bytes\n"
334*4882a593Smuzhiyun "\n"
335*4882a593Smuzhiyun " A normal image can be generated with\n"
336*4882a593Smuzhiyun " sunxi-nand-image-builder -p 16384 -o 1280 -e 0x400000 -s -c 40/1024\n"
337*4882a593Smuzhiyun " A boot0 image can be generated with\n"
338*4882a593Smuzhiyun " sunxi-nand-image-builder -p 16384 -o 1280 -e 0x400000 -s -b -u 4096 -c 64/1024\n",
339*4882a593Smuzhiyun PLAIN_VERSION);
340*4882a593Smuzhiyun exit(status);
341*4882a593Smuzhiyun }
342*4882a593Smuzhiyun
check_image_info(struct image_info * info)343*4882a593Smuzhiyun static int check_image_info(struct image_info *info)
344*4882a593Smuzhiyun {
345*4882a593Smuzhiyun static int valid_ecc_strengths[] = { 16, 24, 28, 32, 40, 48, 56, 60, 64 };
346*4882a593Smuzhiyun int eccbytes, eccsteps;
347*4882a593Smuzhiyun unsigned i;
348*4882a593Smuzhiyun
349*4882a593Smuzhiyun if (!info->page_size) {
350*4882a593Smuzhiyun fprintf(stderr, "--page is missing\n");
351*4882a593Smuzhiyun return -EINVAL;
352*4882a593Smuzhiyun }
353*4882a593Smuzhiyun
354*4882a593Smuzhiyun if (!info->page_size) {
355*4882a593Smuzhiyun fprintf(stderr, "--oob is missing\n");
356*4882a593Smuzhiyun return -EINVAL;
357*4882a593Smuzhiyun }
358*4882a593Smuzhiyun
359*4882a593Smuzhiyun if (!info->eraseblock_size) {
360*4882a593Smuzhiyun fprintf(stderr, "--eraseblock is missing\n");
361*4882a593Smuzhiyun return -EINVAL;
362*4882a593Smuzhiyun }
363*4882a593Smuzhiyun
364*4882a593Smuzhiyun if (info->ecc_step_size != 512 && info->ecc_step_size != 1024) {
365*4882a593Smuzhiyun fprintf(stderr, "Invalid ECC step argument: %d\n",
366*4882a593Smuzhiyun info->ecc_step_size);
367*4882a593Smuzhiyun return -EINVAL;
368*4882a593Smuzhiyun }
369*4882a593Smuzhiyun
370*4882a593Smuzhiyun for (i = 0; i < ARRAY_SIZE(valid_ecc_strengths); i++) {
371*4882a593Smuzhiyun if (valid_ecc_strengths[i] == info->ecc_strength)
372*4882a593Smuzhiyun break;
373*4882a593Smuzhiyun }
374*4882a593Smuzhiyun
375*4882a593Smuzhiyun if (i == ARRAY_SIZE(valid_ecc_strengths)) {
376*4882a593Smuzhiyun fprintf(stderr, "Invalid ECC strength argument: %d\n",
377*4882a593Smuzhiyun info->ecc_strength);
378*4882a593Smuzhiyun return -EINVAL;
379*4882a593Smuzhiyun }
380*4882a593Smuzhiyun
381*4882a593Smuzhiyun eccbytes = DIV_ROUND_UP(info->ecc_strength * 14, 8);
382*4882a593Smuzhiyun if (eccbytes % 2)
383*4882a593Smuzhiyun eccbytes++;
384*4882a593Smuzhiyun eccbytes += 4;
385*4882a593Smuzhiyun
386*4882a593Smuzhiyun eccsteps = info->usable_page_size / info->ecc_step_size;
387*4882a593Smuzhiyun
388*4882a593Smuzhiyun if (info->page_size + info->oob_size <
389*4882a593Smuzhiyun info->usable_page_size + (eccsteps * eccbytes)) {
390*4882a593Smuzhiyun fprintf(stderr,
391*4882a593Smuzhiyun "ECC bytes do not fit in the NAND page, choose a weaker ECC\n");
392*4882a593Smuzhiyun return -EINVAL;
393*4882a593Smuzhiyun }
394*4882a593Smuzhiyun
395*4882a593Smuzhiyun return 0;
396*4882a593Smuzhiyun }
397*4882a593Smuzhiyun
main(int argc,char ** argv)398*4882a593Smuzhiyun int main(int argc, char **argv)
399*4882a593Smuzhiyun {
400*4882a593Smuzhiyun struct image_info info;
401*4882a593Smuzhiyun
402*4882a593Smuzhiyun memset(&info, 0, sizeof(info));
403*4882a593Smuzhiyun /*
404*4882a593Smuzhiyun * Process user arguments
405*4882a593Smuzhiyun */
406*4882a593Smuzhiyun for (;;) {
407*4882a593Smuzhiyun int option_index = 0;
408*4882a593Smuzhiyun char *endptr = NULL;
409*4882a593Smuzhiyun static const struct option long_options[] = {
410*4882a593Smuzhiyun {"help", no_argument, 0, 'h'},
411*4882a593Smuzhiyun {"ecc", required_argument, 0, 'c'},
412*4882a593Smuzhiyun {"page", required_argument, 0, 'p'},
413*4882a593Smuzhiyun {"oob", required_argument, 0, 'o'},
414*4882a593Smuzhiyun {"usable", required_argument, 0, 'u'},
415*4882a593Smuzhiyun {"eraseblock", required_argument, 0, 'e'},
416*4882a593Smuzhiyun {"boot0", no_argument, 0, 'b'},
417*4882a593Smuzhiyun {"scramble", no_argument, 0, 's'},
418*4882a593Smuzhiyun {"address", required_argument, 0, 'a'},
419*4882a593Smuzhiyun {0, 0, 0, 0},
420*4882a593Smuzhiyun };
421*4882a593Smuzhiyun
422*4882a593Smuzhiyun int c = getopt_long(argc, argv, "c:p:o:u:e:ba:sh",
423*4882a593Smuzhiyun long_options, &option_index);
424*4882a593Smuzhiyun if (c == EOF)
425*4882a593Smuzhiyun break;
426*4882a593Smuzhiyun
427*4882a593Smuzhiyun switch (c) {
428*4882a593Smuzhiyun case 'h':
429*4882a593Smuzhiyun display_help(0);
430*4882a593Smuzhiyun break;
431*4882a593Smuzhiyun case 's':
432*4882a593Smuzhiyun info.scramble = 1;
433*4882a593Smuzhiyun break;
434*4882a593Smuzhiyun case 'c':
435*4882a593Smuzhiyun info.ecc_strength = strtol(optarg, &endptr, 0);
436*4882a593Smuzhiyun if (*endptr == '/')
437*4882a593Smuzhiyun info.ecc_step_size = strtol(endptr + 1, NULL, 0);
438*4882a593Smuzhiyun break;
439*4882a593Smuzhiyun case 'p':
440*4882a593Smuzhiyun info.page_size = strtol(optarg, NULL, 0);
441*4882a593Smuzhiyun break;
442*4882a593Smuzhiyun case 'o':
443*4882a593Smuzhiyun info.oob_size = strtol(optarg, NULL, 0);
444*4882a593Smuzhiyun break;
445*4882a593Smuzhiyun case 'u':
446*4882a593Smuzhiyun info.usable_page_size = strtol(optarg, NULL, 0);
447*4882a593Smuzhiyun break;
448*4882a593Smuzhiyun case 'e':
449*4882a593Smuzhiyun info.eraseblock_size = strtol(optarg, NULL, 0);
450*4882a593Smuzhiyun break;
451*4882a593Smuzhiyun case 'b':
452*4882a593Smuzhiyun info.boot0 = 1;
453*4882a593Smuzhiyun break;
454*4882a593Smuzhiyun case 'a':
455*4882a593Smuzhiyun info.offset = strtoull(optarg, NULL, 0);
456*4882a593Smuzhiyun break;
457*4882a593Smuzhiyun case '?':
458*4882a593Smuzhiyun display_help(-1);
459*4882a593Smuzhiyun break;
460*4882a593Smuzhiyun }
461*4882a593Smuzhiyun }
462*4882a593Smuzhiyun
463*4882a593Smuzhiyun if ((argc - optind) != 2)
464*4882a593Smuzhiyun display_help(-1);
465*4882a593Smuzhiyun
466*4882a593Smuzhiyun info.source = argv[optind];
467*4882a593Smuzhiyun info.dest = argv[optind + 1];
468*4882a593Smuzhiyun
469*4882a593Smuzhiyun if (!info.boot0) {
470*4882a593Smuzhiyun info.usable_page_size = info.page_size;
471*4882a593Smuzhiyun } else if (!info.usable_page_size) {
472*4882a593Smuzhiyun if (info.page_size > 8192)
473*4882a593Smuzhiyun info.usable_page_size = 8192;
474*4882a593Smuzhiyun else if (info.page_size > 4096)
475*4882a593Smuzhiyun info.usable_page_size = 4096;
476*4882a593Smuzhiyun else
477*4882a593Smuzhiyun info.usable_page_size = 1024;
478*4882a593Smuzhiyun }
479*4882a593Smuzhiyun
480*4882a593Smuzhiyun if (check_image_info(&info))
481*4882a593Smuzhiyun display_help(-1);
482*4882a593Smuzhiyun
483*4882a593Smuzhiyun return create_image(&info);
484*4882a593Smuzhiyun }
485