subpagetest.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (C) 2006-2007 Nokia Corporation
  4. *
  5. * Test sub-page read and write on MTD device.
  6. * Author: Adrian Hunter <[email protected]>
  7. */
  8. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  9. #include <linux/init.h>
  10. #include <linux/module.h>
  11. #include <linux/moduleparam.h>
  12. #include <linux/err.h>
  13. #include <linux/mtd/mtd.h>
  14. #include <linux/slab.h>
  15. #include <linux/sched.h>
  16. #include <linux/random.h>
  17. #include "mtd_test.h"
  18. static int dev = -EINVAL;
  19. module_param(dev, int, S_IRUGO);
  20. MODULE_PARM_DESC(dev, "MTD device number to use");
  21. static struct mtd_info *mtd;
  22. static unsigned char *writebuf;
  23. static unsigned char *readbuf;
  24. static unsigned char *bbt;
  25. static int subpgsize;
  26. static int bufsize;
  27. static int ebcnt;
  28. static int pgcnt;
  29. static int errcnt;
  30. static struct rnd_state rnd_state;
  31. static inline void clear_data(unsigned char *buf, size_t len)
  32. {
  33. memset(buf, 0, len);
  34. }
  35. static int write_eraseblock(int ebnum)
  36. {
  37. size_t written;
  38. int err = 0;
  39. loff_t addr = (loff_t)ebnum * mtd->erasesize;
  40. prandom_bytes_state(&rnd_state, writebuf, subpgsize);
  41. err = mtd_write(mtd, addr, subpgsize, &written, writebuf);
  42. if (unlikely(err || written != subpgsize)) {
  43. pr_err("error: write failed at %#llx\n",
  44. (long long)addr);
  45. if (written != subpgsize) {
  46. pr_err(" write size: %#x\n", subpgsize);
  47. pr_err(" written: %#zx\n", written);
  48. }
  49. return err ? err : -1;
  50. }
  51. addr += subpgsize;
  52. prandom_bytes_state(&rnd_state, writebuf, subpgsize);
  53. err = mtd_write(mtd, addr, subpgsize, &written, writebuf);
  54. if (unlikely(err || written != subpgsize)) {
  55. pr_err("error: write failed at %#llx\n",
  56. (long long)addr);
  57. if (written != subpgsize) {
  58. pr_err(" write size: %#x\n", subpgsize);
  59. pr_err(" written: %#zx\n", written);
  60. }
  61. return err ? err : -1;
  62. }
  63. return err;
  64. }
  65. static int write_eraseblock2(int ebnum)
  66. {
  67. size_t written;
  68. int err = 0, k;
  69. loff_t addr = (loff_t)ebnum * mtd->erasesize;
  70. for (k = 1; k < 33; ++k) {
  71. if (addr + (subpgsize * k) > (loff_t)(ebnum + 1) * mtd->erasesize)
  72. break;
  73. prandom_bytes_state(&rnd_state, writebuf, subpgsize * k);
  74. err = mtd_write(mtd, addr, subpgsize * k, &written, writebuf);
  75. if (unlikely(err || written != subpgsize * k)) {
  76. pr_err("error: write failed at %#llx\n",
  77. (long long)addr);
  78. if (written != subpgsize * k) {
  79. pr_err(" write size: %#x\n",
  80. subpgsize * k);
  81. pr_err(" written: %#08zx\n",
  82. written);
  83. }
  84. return err ? err : -1;
  85. }
  86. addr += subpgsize * k;
  87. }
  88. return err;
  89. }
  90. static void print_subpage(unsigned char *p)
  91. {
  92. int i, j;
  93. for (i = 0; i < subpgsize; ) {
  94. for (j = 0; i < subpgsize && j < 32; ++i, ++j)
  95. printk("%02x", *p++);
  96. printk("\n");
  97. }
  98. }
  99. static int verify_eraseblock(int ebnum)
  100. {
  101. size_t read;
  102. int err = 0;
  103. loff_t addr = (loff_t)ebnum * mtd->erasesize;
  104. prandom_bytes_state(&rnd_state, writebuf, subpgsize);
  105. clear_data(readbuf, subpgsize);
  106. err = mtd_read(mtd, addr, subpgsize, &read, readbuf);
  107. if (unlikely(err || read != subpgsize)) {
  108. if (mtd_is_bitflip(err) && read == subpgsize) {
  109. pr_info("ECC correction at %#llx\n",
  110. (long long)addr);
  111. err = 0;
  112. } else {
  113. pr_err("error: read failed at %#llx\n",
  114. (long long)addr);
  115. return err ? err : -1;
  116. }
  117. }
  118. if (unlikely(memcmp(readbuf, writebuf, subpgsize))) {
  119. pr_err("error: verify failed at %#llx\n",
  120. (long long)addr);
  121. pr_info("------------- written----------------\n");
  122. print_subpage(writebuf);
  123. pr_info("------------- read ------------------\n");
  124. print_subpage(readbuf);
  125. pr_info("-------------------------------------\n");
  126. errcnt += 1;
  127. }
  128. addr += subpgsize;
  129. prandom_bytes_state(&rnd_state, writebuf, subpgsize);
  130. clear_data(readbuf, subpgsize);
  131. err = mtd_read(mtd, addr, subpgsize, &read, readbuf);
  132. if (unlikely(err || read != subpgsize)) {
  133. if (mtd_is_bitflip(err) && read == subpgsize) {
  134. pr_info("ECC correction at %#llx\n",
  135. (long long)addr);
  136. err = 0;
  137. } else {
  138. pr_err("error: read failed at %#llx\n",
  139. (long long)addr);
  140. return err ? err : -1;
  141. }
  142. }
  143. if (unlikely(memcmp(readbuf, writebuf, subpgsize))) {
  144. pr_info("error: verify failed at %#llx\n",
  145. (long long)addr);
  146. pr_info("------------- written----------------\n");
  147. print_subpage(writebuf);
  148. pr_info("------------- read ------------------\n");
  149. print_subpage(readbuf);
  150. pr_info("-------------------------------------\n");
  151. errcnt += 1;
  152. }
  153. return err;
  154. }
  155. static int verify_eraseblock2(int ebnum)
  156. {
  157. size_t read;
  158. int err = 0, k;
  159. loff_t addr = (loff_t)ebnum * mtd->erasesize;
  160. for (k = 1; k < 33; ++k) {
  161. if (addr + (subpgsize * k) > (loff_t)(ebnum + 1) * mtd->erasesize)
  162. break;
  163. prandom_bytes_state(&rnd_state, writebuf, subpgsize * k);
  164. clear_data(readbuf, subpgsize * k);
  165. err = mtd_read(mtd, addr, subpgsize * k, &read, readbuf);
  166. if (unlikely(err || read != subpgsize * k)) {
  167. if (mtd_is_bitflip(err) && read == subpgsize * k) {
  168. pr_info("ECC correction at %#llx\n",
  169. (long long)addr);
  170. err = 0;
  171. } else {
  172. pr_err("error: read failed at "
  173. "%#llx\n", (long long)addr);
  174. return err ? err : -1;
  175. }
  176. }
  177. if (unlikely(memcmp(readbuf, writebuf, subpgsize * k))) {
  178. pr_err("error: verify failed at %#llx\n",
  179. (long long)addr);
  180. errcnt += 1;
  181. }
  182. addr += subpgsize * k;
  183. }
  184. return err;
  185. }
  186. static int verify_eraseblock_ff(int ebnum)
  187. {
  188. uint32_t j;
  189. size_t read;
  190. int err = 0;
  191. loff_t addr = (loff_t)ebnum * mtd->erasesize;
  192. memset(writebuf, 0xff, subpgsize);
  193. for (j = 0; j < mtd->erasesize / subpgsize; ++j) {
  194. clear_data(readbuf, subpgsize);
  195. err = mtd_read(mtd, addr, subpgsize, &read, readbuf);
  196. if (unlikely(err || read != subpgsize)) {
  197. if (mtd_is_bitflip(err) && read == subpgsize) {
  198. pr_info("ECC correction at %#llx\n",
  199. (long long)addr);
  200. err = 0;
  201. } else {
  202. pr_err("error: read failed at "
  203. "%#llx\n", (long long)addr);
  204. return err ? err : -1;
  205. }
  206. }
  207. if (unlikely(memcmp(readbuf, writebuf, subpgsize))) {
  208. pr_err("error: verify 0xff failed at "
  209. "%#llx\n", (long long)addr);
  210. errcnt += 1;
  211. }
  212. addr += subpgsize;
  213. }
  214. return err;
  215. }
  216. static int verify_all_eraseblocks_ff(void)
  217. {
  218. int err;
  219. unsigned int i;
  220. pr_info("verifying all eraseblocks for 0xff\n");
  221. for (i = 0; i < ebcnt; ++i) {
  222. if (bbt[i])
  223. continue;
  224. err = verify_eraseblock_ff(i);
  225. if (err)
  226. return err;
  227. if (i % 256 == 0)
  228. pr_info("verified up to eraseblock %u\n", i);
  229. err = mtdtest_relax();
  230. if (err)
  231. return err;
  232. }
  233. pr_info("verified %u eraseblocks\n", i);
  234. return 0;
  235. }
  236. static int __init mtd_subpagetest_init(void)
  237. {
  238. int err = 0;
  239. uint32_t i;
  240. uint64_t tmp;
  241. printk(KERN_INFO "\n");
  242. printk(KERN_INFO "=================================================\n");
  243. if (dev < 0) {
  244. pr_info("Please specify a valid mtd-device via module parameter\n");
  245. pr_crit("CAREFUL: This test wipes all data on the specified MTD device!\n");
  246. return -EINVAL;
  247. }
  248. pr_info("MTD device: %d\n", dev);
  249. mtd = get_mtd_device(NULL, dev);
  250. if (IS_ERR(mtd)) {
  251. err = PTR_ERR(mtd);
  252. pr_err("error: cannot get MTD device\n");
  253. return err;
  254. }
  255. if (!mtd_type_is_nand(mtd)) {
  256. pr_info("this test requires NAND flash\n");
  257. goto out;
  258. }
  259. subpgsize = mtd->writesize >> mtd->subpage_sft;
  260. tmp = mtd->size;
  261. do_div(tmp, mtd->erasesize);
  262. ebcnt = tmp;
  263. pgcnt = mtd->erasesize / mtd->writesize;
  264. pr_info("MTD device size %llu, eraseblock size %u, "
  265. "page size %u, subpage size %u, count of eraseblocks %u, "
  266. "pages per eraseblock %u, OOB size %u\n",
  267. (unsigned long long)mtd->size, mtd->erasesize,
  268. mtd->writesize, subpgsize, ebcnt, pgcnt, mtd->oobsize);
  269. err = -ENOMEM;
  270. bufsize = subpgsize * 32;
  271. writebuf = kmalloc(bufsize, GFP_KERNEL);
  272. if (!writebuf)
  273. goto out;
  274. readbuf = kmalloc(bufsize, GFP_KERNEL);
  275. if (!readbuf)
  276. goto out;
  277. bbt = kzalloc(ebcnt, GFP_KERNEL);
  278. if (!bbt)
  279. goto out;
  280. err = mtdtest_scan_for_bad_eraseblocks(mtd, bbt, 0, ebcnt);
  281. if (err)
  282. goto out;
  283. err = mtdtest_erase_good_eraseblocks(mtd, bbt, 0, ebcnt);
  284. if (err)
  285. goto out;
  286. pr_info("writing whole device\n");
  287. prandom_seed_state(&rnd_state, 1);
  288. for (i = 0; i < ebcnt; ++i) {
  289. if (bbt[i])
  290. continue;
  291. err = write_eraseblock(i);
  292. if (unlikely(err))
  293. goto out;
  294. if (i % 256 == 0)
  295. pr_info("written up to eraseblock %u\n", i);
  296. err = mtdtest_relax();
  297. if (err)
  298. goto out;
  299. }
  300. pr_info("written %u eraseblocks\n", i);
  301. prandom_seed_state(&rnd_state, 1);
  302. pr_info("verifying all eraseblocks\n");
  303. for (i = 0; i < ebcnt; ++i) {
  304. if (bbt[i])
  305. continue;
  306. err = verify_eraseblock(i);
  307. if (unlikely(err))
  308. goto out;
  309. if (i % 256 == 0)
  310. pr_info("verified up to eraseblock %u\n", i);
  311. err = mtdtest_relax();
  312. if (err)
  313. goto out;
  314. }
  315. pr_info("verified %u eraseblocks\n", i);
  316. err = mtdtest_erase_good_eraseblocks(mtd, bbt, 0, ebcnt);
  317. if (err)
  318. goto out;
  319. err = verify_all_eraseblocks_ff();
  320. if (err)
  321. goto out;
  322. /* Write all eraseblocks */
  323. prandom_seed_state(&rnd_state, 3);
  324. pr_info("writing whole device\n");
  325. for (i = 0; i < ebcnt; ++i) {
  326. if (bbt[i])
  327. continue;
  328. err = write_eraseblock2(i);
  329. if (unlikely(err))
  330. goto out;
  331. if (i % 256 == 0)
  332. pr_info("written up to eraseblock %u\n", i);
  333. err = mtdtest_relax();
  334. if (err)
  335. goto out;
  336. }
  337. pr_info("written %u eraseblocks\n", i);
  338. /* Check all eraseblocks */
  339. prandom_seed_state(&rnd_state, 3);
  340. pr_info("verifying all eraseblocks\n");
  341. for (i = 0; i < ebcnt; ++i) {
  342. if (bbt[i])
  343. continue;
  344. err = verify_eraseblock2(i);
  345. if (unlikely(err))
  346. goto out;
  347. if (i % 256 == 0)
  348. pr_info("verified up to eraseblock %u\n", i);
  349. err = mtdtest_relax();
  350. if (err)
  351. goto out;
  352. }
  353. pr_info("verified %u eraseblocks\n", i);
  354. err = mtdtest_erase_good_eraseblocks(mtd, bbt, 0, ebcnt);
  355. if (err)
  356. goto out;
  357. err = verify_all_eraseblocks_ff();
  358. if (err)
  359. goto out;
  360. pr_info("finished with %d errors\n", errcnt);
  361. out:
  362. kfree(bbt);
  363. kfree(readbuf);
  364. kfree(writebuf);
  365. put_mtd_device(mtd);
  366. if (err)
  367. pr_info("error %d occurred\n", err);
  368. printk(KERN_INFO "=================================================\n");
  369. return err;
  370. }
  371. module_init(mtd_subpagetest_init);
  372. static void __exit mtd_subpagetest_exit(void)
  373. {
  374. return;
  375. }
  376. module_exit(mtd_subpagetest_exit);
  377. MODULE_DESCRIPTION("Subpage test module");
  378. MODULE_AUTHOR("Adrian Hunter");
  379. MODULE_LICENSE("GPL");