Lines Matching +full:avg +full:- +full:pos +full:- +full:strength

1 // SPDX-License-Identifier: GPL-2.0-or-later
142 MODULE_PARM_DESC(bus_width, "Chip's bus width (8- or 16-bit)");
143 MODULE_PARM_DESC(do_delays, "Simulate NAND delays using busy-waits if not zero");
165 "be correctable in 512-byte blocks");
182 /* Busy-wait delay macros (microseconds, milliseconds) */
189 #define NS_IS_INITIALIZED(ns) ((ns)->geom.totsz != 0)
192 #define NS_STATUS_OK(ns) (NAND_STATUS_READY | (NAND_STATUS_WP * ((ns)->lines.wp == 0)))
199 (((ns)->regs.row * (ns)->geom.pgszoob) + (ns)->regs.column)
202 #define NS_RAW_OFFSET_OOB(ns) (NS_RAW_OFFSET(ns) + ns->geom.pgsz)
205 #define NS_PAGE_BYTE_SHIFT(ns) ((ns)->regs.column + (ns)->regs.off)
258 #define OPT_PAGE512 0x00000002 /* 512-byte page chips */
259 #define OPT_PAGE2048 0x00000008 /* 2048-byte page chips */
260 #define OPT_PAGE512_8BIT 0x00000040 /* 512-byte page chips with 8-bit bus width */
261 #define OPT_PAGE4096 0x00000080 /* 4096-byte page chips */
262 #define OPT_LARGEPAGE (OPT_PAGE2048 | OPT_PAGE4096) /* 2048 & 4096-byte page chips */
263 #define OPT_SMALLPAGE (OPT_PAGE512) /* 512-byte page chips */
271 * (which is only valid for 512-byte pages).
283 uint16_t *word; /* for 16-bit word access */
442 unsigned long wmin = -1, wmax = 0, avg; in ns_show() local
471 avg = tot / wear_eb_count; in ns_show()
476 seq_printf(m, "Average number of erases: %lu\n", avg); in ns_show()
480 unsigned long from = (i ? decile_max[i - 1] + 1 : 0); in ns_show()
494 * ns_debugfs_create - initialize debugfs
502 struct dentry *root = nsmtd->dbg.dfs_dir; in ns_debugfs_create()
515 ns->dent = debugfs_create_file("nandsim_wear_report", 0400, root, ns, in ns_debugfs_create()
517 if (IS_ERR_OR_NULL(ns->dent)) { in ns_debugfs_create()
519 return -1; in ns_debugfs_create()
527 debugfs_remove_recursive(ns->dent); in ns_debugfs_remove()
534 * RETURNS: 0 if success, -ENOMEM if memory alloc fails.
545 if (!(cfile->f_mode & FMODE_CAN_READ)) { in ns_alloc_device()
547 err = -EINVAL; in ns_alloc_device()
550 if (!(cfile->f_mode & FMODE_CAN_WRITE)) { in ns_alloc_device()
552 err = -EINVAL; in ns_alloc_device()
555 ns->pages_written = in ns_alloc_device()
557 BITS_TO_LONGS(ns->geom.pgnum))); in ns_alloc_device()
558 if (!ns->pages_written) { in ns_alloc_device()
560 err = -ENOMEM; in ns_alloc_device()
563 ns->file_buf = kmalloc(ns->geom.pgszoob, GFP_KERNEL); in ns_alloc_device()
564 if (!ns->file_buf) { in ns_alloc_device()
566 err = -ENOMEM; in ns_alloc_device()
569 ns->cfile = cfile; in ns_alloc_device()
574 vfree(ns->pages_written); in ns_alloc_device()
581 ns->pages = vmalloc(array_size(sizeof(union ns_mem), ns->geom.pgnum)); in ns_alloc_device()
582 if (!ns->pages) { in ns_alloc_device()
584 return -ENOMEM; in ns_alloc_device()
586 for (i = 0; i < ns->geom.pgnum; i++) { in ns_alloc_device()
587 ns->pages[i].byte = NULL; in ns_alloc_device()
589 ns->nand_pages_slab = kmem_cache_create("nandsim", in ns_alloc_device()
590 ns->geom.pgszoob, 0, 0, NULL); in ns_alloc_device()
591 if (!ns->nand_pages_slab) { in ns_alloc_device()
593 err = -ENOMEM; in ns_alloc_device()
600 vfree(ns->pages); in ns_alloc_device()
612 if (ns->cfile) { in ns_free_device()
613 kfree(ns->file_buf); in ns_free_device()
614 vfree(ns->pages_written); in ns_free_device()
615 filp_close(ns->cfile, NULL); in ns_free_device()
619 if (ns->pages) { in ns_free_device()
620 for (i = 0; i < ns->geom.pgnum; i++) { in ns_free_device()
621 if (ns->pages[i].byte) in ns_free_device()
622 kmem_cache_free(ns->nand_pages_slab, in ns_free_device()
623 ns->pages[i].byte); in ns_free_device()
625 kmem_cache_destroy(ns->nand_pages_slab); in ns_free_device()
626 vfree(ns->pages); in ns_free_device()
638 * RETURNS: 0 if success, -ERRNO if failure.
650 return -EIO; in ns_init()
654 ns->busw = chip->options & NAND_BUSWIDTH_16 ? 16 : 8; in ns_init()
655 ns->geom.totsz = mtd->size; in ns_init()
656 ns->geom.pgsz = mtd->writesize; in ns_init()
657 ns->geom.oobsz = mtd->oobsize; in ns_init()
658 ns->geom.secsz = mtd->erasesize; in ns_init()
659 ns->geom.pgszoob = ns->geom.pgsz + ns->geom.oobsz; in ns_init()
660 ns->geom.pgnum = div_u64(ns->geom.totsz, ns->geom.pgsz); in ns_init()
661 ns->geom.totszoob = ns->geom.totsz + (uint64_t)ns->geom.pgnum * ns->geom.oobsz; in ns_init()
662 ns->geom.secshift = ffs(ns->geom.secsz) - 1; in ns_init()
663 ns->geom.pgshift = chip->page_shift; in ns_init()
664 ns->geom.pgsec = ns->geom.secsz / ns->geom.pgsz; in ns_init()
665 ns->geom.secszoob = ns->geom.secsz + ns->geom.oobsz * ns->geom.pgsec; in ns_init()
666 ns->options = 0; in ns_init()
668 if (ns->geom.pgsz == 512) { in ns_init()
669 ns->options |= OPT_PAGE512; in ns_init()
670 if (ns->busw == 8) in ns_init()
671 ns->options |= OPT_PAGE512_8BIT; in ns_init()
672 } else if (ns->geom.pgsz == 2048) { in ns_init()
673 ns->options |= OPT_PAGE2048; in ns_init()
674 } else if (ns->geom.pgsz == 4096) { in ns_init()
675 ns->options |= OPT_PAGE4096; in ns_init()
677 NS_ERR("init_nandsim: unknown page size %u\n", ns->geom.pgsz); in ns_init()
678 return -EIO; in ns_init()
681 if (ns->options & OPT_SMALLPAGE) { in ns_init()
682 if (ns->geom.totsz <= (32 << 20)) { in ns_init()
683 ns->geom.pgaddrbytes = 3; in ns_init()
684 ns->geom.secaddrbytes = 2; in ns_init()
686 ns->geom.pgaddrbytes = 4; in ns_init()
687 ns->geom.secaddrbytes = 3; in ns_init()
690 if (ns->geom.totsz <= (128 << 20)) { in ns_init()
691 ns->geom.pgaddrbytes = 4; in ns_init()
692 ns->geom.secaddrbytes = 2; in ns_init()
694 ns->geom.pgaddrbytes = 5; in ns_init()
695 ns->geom.secaddrbytes = 3; in ns_init()
700 if (parts_num > ARRAY_SIZE(ns->partitions)) { in ns_init()
702 return -EINVAL; in ns_init()
704 remains = ns->geom.totsz; in ns_init()
707 uint64_t part_sz = (uint64_t)parts[i] * ns->geom.secsz; in ns_init()
711 return -EINVAL; in ns_init()
713 ns->partitions[i].name = ns_get_partition_name(i); in ns_init()
714 if (!ns->partitions[i].name) { in ns_init()
716 return -ENOMEM; in ns_init()
718 ns->partitions[i].offset = next_offset; in ns_init()
719 ns->partitions[i].size = part_sz; in ns_init()
720 next_offset += ns->partitions[i].size; in ns_init()
721 remains -= ns->partitions[i].size; in ns_init()
723 ns->nbparts = parts_num; in ns_init()
725 if (parts_num + 1 > ARRAY_SIZE(ns->partitions)) { in ns_init()
727 ret = -EINVAL; in ns_init()
730 ns->partitions[i].name = ns_get_partition_name(i); in ns_init()
731 if (!ns->partitions[i].name) { in ns_init()
733 ret = -ENOMEM; in ns_init()
736 ns->partitions[i].offset = next_offset; in ns_init()
737 ns->partitions[i].size = remains; in ns_init()
738 ns->nbparts += 1; in ns_init()
741 if (ns->busw == 16) in ns_init()
742 NS_WARN("16-bit flashes support wasn't tested\n"); in ns_init()
745 (unsigned long long)ns->geom.totsz >> 20); in ns_init()
746 printk("page size: %u bytes\n", ns->geom.pgsz); in ns_init()
747 printk("OOB area size: %u bytes\n", ns->geom.oobsz); in ns_init()
748 printk("sector size: %u KiB\n", ns->geom.secsz >> 10); in ns_init()
749 printk("pages number: %u\n", ns->geom.pgnum); in ns_init()
750 printk("pages per sector: %u\n", ns->geom.pgsec); in ns_init()
751 printk("bus width: %u\n", ns->busw); in ns_init()
752 printk("bits in sector size: %u\n", ns->geom.secshift); in ns_init()
753 printk("bits in page size: %u\n", ns->geom.pgshift); in ns_init()
754 printk("bits in OOB size: %u\n", ffs(ns->geom.oobsz) - 1); in ns_init()
756 (unsigned long long)ns->geom.totszoob >> 10); in ns_init()
757 printk("page address bytes: %u\n", ns->geom.pgaddrbytes); in ns_init()
758 printk("sector address bytes: %u\n", ns->geom.secaddrbytes); in ns_init()
759 printk("options: %#x\n", ns->options); in ns_init()
766 ns->buf.byte = kmalloc(ns->geom.pgszoob, GFP_KERNEL); in ns_init()
767 if (!ns->buf.byte) { in ns_init()
769 ns->geom.pgszoob); in ns_init()
770 ret = -ENOMEM; in ns_init()
773 memset(ns->buf.byte, 0xFF, ns->geom.pgszoob); in ns_init()
780 for (i = 0; i < ARRAY_SIZE(ns->partitions); ++i) in ns_init()
781 kfree(ns->partitions[i].name); in ns_init()
793 for (i = 0; i < ARRAY_SIZE(ns->partitions); ++i) in ns_free()
794 kfree(ns->partitions[i].name); in ns_free()
796 kfree(ns->buf.byte); in ns_free()
817 return -EINVAL; in ns_parse_badblocks()
819 offset = (loff_t)erase_block_no * ns->geom.secsz; in ns_parse_badblocks()
822 return -EINVAL; in ns_parse_badblocks()
846 return -EINVAL; in ns_parse_weakblocks()
858 return -ENOMEM; in ns_parse_weakblocks()
860 wb->erase_block_no = erase_block_no; in ns_parse_weakblocks()
861 wb->max_erases = max_erases; in ns_parse_weakblocks()
862 list_add(&wb->list, &weak_blocks); in ns_parse_weakblocks()
872 if (wb->erase_block_no == erase_block_no) { in ns_erase_error()
873 if (wb->erases_done >= wb->max_erases) in ns_erase_error()
875 wb->erases_done += 1; in ns_erase_error()
897 return -EINVAL; in ns_parse_weakpages()
909 return -ENOMEM; in ns_parse_weakpages()
911 wp->page_no = page_no; in ns_parse_weakpages()
912 wp->max_writes = max_writes; in ns_parse_weakpages()
913 list_add(&wp->list, &weak_pages); in ns_parse_weakpages()
923 if (wp->page_no == page_no) { in ns_write_error()
924 if (wp->writes_done >= wp->max_writes) in ns_write_error()
926 wp->writes_done += 1; in ns_write_error()
948 return -EINVAL; in ns_parse_gravepages()
960 return -ENOMEM; in ns_parse_gravepages()
962 gp->page_no = page_no; in ns_parse_gravepages()
963 gp->max_reads = max_reads; in ns_parse_gravepages()
964 list_add(&gp->list, &grave_pages); in ns_parse_gravepages()
974 if (gp->page_no == page_no) { in ns_read_error()
975 if (gp->reads_done >= gp->max_reads) in ns_read_error()
977 gp->reads_done += 1; in ns_read_error()
985 wear_eb_count = div_u64(mtd->size, mtd->erasesize); in ns_setup_wear_reporting()
989 return -ENOMEM; in ns_setup_wear_reporting()
1142 if (ns->regs.count < (ns->geom.pgaddrbytes - ns->geom.secaddrbytes)) in ns_accept_addr_byte()
1143 ns->regs.column |= (byte << 8 * ns->regs.count); in ns_accept_addr_byte()
1145 ns->regs.row |= (byte << 8 * (ns->regs.count - in ns_accept_addr_byte()
1146 ns->geom.pgaddrbytes + in ns_accept_addr_byte()
1147 ns->geom.secaddrbytes)); in ns_accept_addr_byte()
1161 ns->state = STATE_READY; in ns_switch_to_ready_state()
1162 ns->nxstate = STATE_UNKNOWN; in ns_switch_to_ready_state()
1163 ns->op = NULL; in ns_switch_to_ready_state()
1164 ns->npstates = 0; in ns_switch_to_ready_state()
1165 ns->stateidx = 0; in ns_switch_to_ready_state()
1166 ns->regs.num = 0; in ns_switch_to_ready_state()
1167 ns->regs.count = 0; in ns_switch_to_ready_state()
1168 ns->regs.off = 0; in ns_switch_to_ready_state()
1169 ns->regs.row = 0; in ns_switch_to_ready_state()
1170 ns->regs.column = 0; in ns_switch_to_ready_state()
1171 ns->regs.status = status; in ns_switch_to_ready_state()
1180 * correspondent states chain. In this case ns->npstates = 0;
1184 * case the ns->pstates[] array contains previous states.
1188 * ns->pstates[0], ... ns->pstates[ns->npstates], ns->state
1191 * ns->ops, ns->state, ns->nxstate are initialized, ns->npstate is
1195 * ns->pstates.
1201 * ns->pstates[0], ... ns->pstates[ns->npstates], <address input>
1212 * RETURNS: -2 - no matched operations found.
1213 * -1 - several matches.
1214 * 0 - operation is found.
1225 if (!(ns->options & ops[i].reqopts)) in ns_find_operation()
1230 if (!(ops[i].states[ns->npstates] & STATE_ADDR_MASK)) in ns_find_operation()
1233 if (NS_STATE(ns->state) != NS_STATE(ops[i].states[ns->npstates])) in ns_find_operation()
1237 for (j = 0; j < ns->npstates; j++) in ns_find_operation()
1238 if (NS_STATE(ops[i].states[j]) != NS_STATE(ns->pstates[j]) in ns_find_operation()
1239 && (ns->options & ops[idx].reqopts)) { in ns_find_operation()
1252 ns->op = &ops[idx].states[0]; in ns_find_operation()
1259 * state must be the next state (ns->nxstate). in ns_find_operation()
1261 ns->stateidx = ns->npstates - 1; in ns_find_operation()
1263 ns->stateidx = ns->npstates; in ns_find_operation()
1265 ns->npstates = 0; in ns_find_operation()
1266 ns->state = ns->op[ns->stateidx]; in ns_find_operation()
1267 ns->nxstate = ns->op[ns->stateidx + 1]; in ns_find_operation()
1269 idx, ns_get_state_name(ns->state), in ns_find_operation()
1270 ns_get_state_name(ns->nxstate)); in ns_find_operation()
1276 if (ns->npstates != 0) { in ns_find_operation()
1278 ns_get_state_name(ns->state)); in ns_find_operation()
1279 ns->npstates = 0; in ns_find_operation()
1285 return -2; in ns_find_operation()
1291 return -2; in ns_find_operation()
1296 ns->pstates[ns->npstates++] = ns->state; in ns_find_operation()
1298 return -1; in ns_find_operation()
1305 for (i = 0; i < ns->held_cnt; i++) in ns_put_pages()
1306 put_page(ns->held_pages[i]); in ns_put_pages()
1311 loff_t pos) in ns_get_pages() argument
1315 struct address_space *mapping = file->f_mapping; in ns_get_pages()
1317 start_index = pos >> PAGE_SHIFT; in ns_get_pages()
1318 end_index = (pos + count - 1) >> PAGE_SHIFT; in ns_get_pages()
1319 if (end_index - start_index + 1 > NS_MAX_HELD_PAGES) in ns_get_pages()
1320 return -EINVAL; in ns_get_pages()
1321 ns->held_cnt = 0; in ns_get_pages()
1327 write_inode_now(mapping->host, 1); in ns_get_pages()
1332 return -ENOMEM; in ns_get_pages()
1336 ns->held_pages[ns->held_cnt++] = page; in ns_get_pages()
1342 size_t count, loff_t pos) in ns_read_file() argument
1348 err = ns_get_pages(ns, file, count, pos); in ns_read_file()
1352 tx = kernel_read(file, buf, count, &pos); in ns_read_file()
1359 size_t count, loff_t pos) in ns_write_file() argument
1365 err = ns_get_pages(ns, file, count, pos); in ns_write_file()
1369 tx = kernel_write(file, buf, count, &pos); in ns_write_file()
1380 return &(ns->pages[ns->regs.row]); in NS_GET_PAGE()
1388 return NS_GET_PAGE(ns)->byte + NS_PAGE_BYTE_SHIFT(ns); in NS_PAGE_BYTE_OFF()
1393 unsigned int page_no = ns->regs.row; in ns_do_read_error()
1396 get_random_bytes(ns->buf.byte, num); in ns_do_read_error()
1409 while (flips--) { in ns_do_bit_flips()
1410 int pos = get_random_u32_below(num * 8); in ns_do_bit_flips() local
1411 ns->buf.byte[pos / 8] ^= (1 << (pos % 8)); in ns_do_bit_flips()
1414 pos, ns->regs.row, NS_PAGE_BYTE_SHIFT(ns), in ns_do_bit_flips()
1415 nsmtd->ecc_stats.corrected, nsmtd->ecc_stats.failed); in ns_do_bit_flips()
1427 if (ns->cfile) { in ns_read_page()
1428 if (!test_bit(ns->regs.row, ns->pages_written)) { in ns_read_page()
1429 NS_DBG("read_page: page %d not written\n", ns->regs.row); in ns_read_page()
1430 memset(ns->buf.byte, 0xFF, num); in ns_read_page()
1432 loff_t pos; in ns_read_page() local
1436 ns->regs.row, NS_PAGE_BYTE_SHIFT(ns)); in ns_read_page()
1439 pos = (loff_t)NS_RAW_OFFSET(ns) + ns->regs.off; in ns_read_page()
1440 tx = ns_read_file(ns, ns->cfile, ns->buf.byte, num, in ns_read_page()
1441 pos); in ns_read_page()
1443 NS_ERR("read_page: read error for page %d ret %ld\n", ns->regs.row, (long)tx); in ns_read_page()
1452 if (mypage->byte == NULL) { in ns_read_page()
1453 NS_DBG("read_page: page %d not allocated\n", ns->regs.row); in ns_read_page()
1454 memset(ns->buf.byte, 0xFF, num); in ns_read_page()
1457 ns->regs.row, NS_PAGE_BYTE_SHIFT(ns)); in ns_read_page()
1460 memcpy(ns->buf.byte, NS_PAGE_BYTE_OFF(ns), num); in ns_read_page()
1473 if (ns->cfile) { in ns_erase_sector()
1474 for (i = 0; i < ns->geom.pgsec; i++) in ns_erase_sector()
1475 if (__test_and_clear_bit(ns->regs.row + i, in ns_erase_sector()
1476 ns->pages_written)) { in ns_erase_sector()
1477 NS_DBG("erase_sector: freeing page %d\n", ns->regs.row + i); in ns_erase_sector()
1483 for (i = 0; i < ns->geom.pgsec; i++) { in ns_erase_sector()
1484 if (mypage->byte != NULL) { in ns_erase_sector()
1485 NS_DBG("erase_sector: freeing page %d\n", ns->regs.row+i); in ns_erase_sector()
1486 kmem_cache_free(ns->nand_pages_slab, mypage->byte); in ns_erase_sector()
1487 mypage->byte = NULL; in ns_erase_sector()
1502 if (ns->cfile) { in ns_prog_page()
1507 NS_DBG("prog_page: writing page %d\n", ns->regs.row); in ns_prog_page()
1508 pg_off = ns->file_buf + NS_PAGE_BYTE_SHIFT(ns); in ns_prog_page()
1509 off = (loff_t)NS_RAW_OFFSET(ns) + ns->regs.off; in ns_prog_page()
1510 if (!test_bit(ns->regs.row, ns->pages_written)) { in ns_prog_page()
1512 memset(ns->file_buf, 0xff, ns->geom.pgszoob); in ns_prog_page()
1515 tx = ns_read_file(ns, ns->cfile, pg_off, num, off); in ns_prog_page()
1517 NS_ERR("prog_page: read error for page %d ret %ld\n", ns->regs.row, (long)tx); in ns_prog_page()
1518 return -1; in ns_prog_page()
1522 pg_off[i] &= ns->buf.byte[i]; in ns_prog_page()
1524 loff_t pos = (loff_t)ns->regs.row * ns->geom.pgszoob; in ns_prog_page() local
1525 tx = ns_write_file(ns, ns->cfile, ns->file_buf, in ns_prog_page()
1526 ns->geom.pgszoob, pos); in ns_prog_page()
1527 if (tx != ns->geom.pgszoob) { in ns_prog_page()
1528 NS_ERR("prog_page: write error for page %d ret %ld\n", ns->regs.row, (long)tx); in ns_prog_page()
1529 return -1; in ns_prog_page()
1531 __set_bit(ns->regs.row, ns->pages_written); in ns_prog_page()
1533 tx = ns_write_file(ns, ns->cfile, pg_off, num, off); in ns_prog_page()
1535 NS_ERR("prog_page: write error for page %d ret %ld\n", ns->regs.row, (long)tx); in ns_prog_page()
1536 return -1; in ns_prog_page()
1543 if (mypage->byte == NULL) { in ns_prog_page()
1544 NS_DBG("prog_page: allocating page %d\n", ns->regs.row); in ns_prog_page()
1551 mypage->byte = kmem_cache_alloc(ns->nand_pages_slab, GFP_NOFS); in ns_prog_page()
1552 if (mypage->byte == NULL) { in ns_prog_page()
1553 NS_ERR("prog_page: error allocating memory for page %d\n", ns->regs.row); in ns_prog_page()
1554 return -1; in ns_prog_page()
1556 memset(mypage->byte, 0xFF, ns->geom.pgszoob); in ns_prog_page()
1561 pg_off[i] &= ns->buf.byte[i]; in ns_prog_page()
1569 * RETURNS: 0 if success, -1 if error.
1574 int busdiv = ns->busw == 8 ? 1 : 2; in ns_do_state_action()
1580 if (action != ACTION_SECERASE && ns->regs.row >= ns->geom.pgnum) { in ns_do_state_action()
1581 NS_WARN("do_state_action: wrong page number (%#x)\n", ns->regs.row); in ns_do_state_action()
1582 return -1; in ns_do_state_action()
1593 if (ns->regs.column >= (ns->geom.pgszoob - ns->regs.off)) { in ns_do_state_action()
1597 num = ns->geom.pgszoob - NS_PAGE_BYTE_SHIFT(ns); in ns_do_state_action()
1601 num, NS_RAW_OFFSET(ns) + ns->regs.off); in ns_do_state_action()
1603 if (ns->regs.off == 0) in ns_do_state_action()
1604 NS_LOG("read page %d\n", ns->regs.row); in ns_do_state_action()
1605 else if (ns->regs.off < ns->geom.pgsz) in ns_do_state_action()
1606 NS_LOG("read page %d (second half)\n", ns->regs.row); in ns_do_state_action()
1608 NS_LOG("read OOB of page %d\n", ns->regs.row); in ns_do_state_action()
1611 NS_UDELAY(input_cycle * ns->geom.pgsz / 1000 / busdiv); in ns_do_state_action()
1620 if (ns->lines.wp) { in ns_do_state_action()
1621 NS_ERR("do_state_action: device is write-protected, ignore sector erase\n"); in ns_do_state_action()
1622 return -1; in ns_do_state_action()
1625 if (ns->regs.row >= ns->geom.pgnum - ns->geom.pgsec in ns_do_state_action()
1626 || (ns->regs.row & ~(ns->geom.secsz - 1))) { in ns_do_state_action()
1627 NS_ERR("do_state_action: wrong sector address (%#x)\n", ns->regs.row); in ns_do_state_action()
1628 return -1; in ns_do_state_action()
1631 ns->regs.row = (ns->regs.row << in ns_do_state_action()
1632 8 * (ns->geom.pgaddrbytes - ns->geom.secaddrbytes)) | ns->regs.column; in ns_do_state_action()
1633 ns->regs.column = 0; in ns_do_state_action()
1635 erase_block_no = ns->regs.row >> (ns->geom.secshift - ns->geom.pgshift); in ns_do_state_action()
1638 ns->regs.row, NS_RAW_OFFSET(ns)); in ns_do_state_action()
1650 return -1; in ns_do_state_action()
1657 * Program page - move internal buffer data to the page. in ns_do_state_action()
1660 if (ns->lines.wp) { in ns_do_state_action()
1661 NS_WARN("do_state_action: device is write-protected, programm\n"); in ns_do_state_action()
1662 return -1; in ns_do_state_action()
1665 num = ns->geom.pgszoob - NS_PAGE_BYTE_SHIFT(ns); in ns_do_state_action()
1666 if (num != ns->regs.count) { in ns_do_state_action()
1668 ns->regs.count, num); in ns_do_state_action()
1669 return -1; in ns_do_state_action()
1672 if (ns_prog_page(ns, num) == -1) in ns_do_state_action()
1673 return -1; in ns_do_state_action()
1675 page_no = ns->regs.row; in ns_do_state_action()
1678 num, ns->regs.row, ns->regs.column, NS_RAW_OFFSET(ns) + ns->regs.off); in ns_do_state_action()
1679 NS_LOG("programm page %d\n", ns->regs.row); in ns_do_state_action()
1682 NS_UDELAY(output_cycle * ns->geom.pgsz / 1000 / busdiv); in ns_do_state_action()
1686 return -1; in ns_do_state_action()
1693 ns->regs.off = 0; in ns_do_state_action()
1697 if (!(ns->options & OPT_PAGE512_8BIT)) { in ns_do_state_action()
1698 NS_ERR("do_state_action: BUG! can't skip half of page for non-512" in ns_do_state_action()
1700 return -1; in ns_do_state_action()
1702 NS_DBG("do_state_action: set internal offset to %d\n", ns->geom.pgsz/2); in ns_do_state_action()
1703 ns->regs.off = ns->geom.pgsz/2; in ns_do_state_action()
1707 NS_DBG("do_state_action: set internal offset to %d\n", ns->geom.pgsz); in ns_do_state_action()
1708 ns->regs.off = ns->geom.pgsz; in ns_do_state_action()
1723 if (ns->op) { in ns_switch_state()
1729 ns->stateidx += 1; in ns_switch_state()
1730 ns->state = ns->nxstate; in ns_switch_state()
1731 ns->nxstate = ns->op[ns->stateidx + 1]; in ns_switch_state()
1735 ns_get_state_name(ns->state), in ns_switch_state()
1736 ns_get_state_name(ns->nxstate)); in ns_switch_state()
1747 ns->state = ns_get_state_by_command(ns->regs.command); in ns_switch_state()
1756 if ((ns->state & ACTION_MASK) && in ns_switch_state()
1757 ns_do_state_action(ns, ns->state) < 0) { in ns_switch_state()
1763 if ((ns->nxstate & STATE_ADDR_MASK) && ns->busw == 16) { in ns_switch_state()
1765 ns->regs.column <<= 1; in ns_switch_state()
1768 if (NS_STATE(ns->nxstate) == STATE_READY) { in ns_switch_state()
1776 if ((ns->state & (STATE_DATAIN_MASK | STATE_DATAOUT_MASK)) in ns_switch_state()
1777 && ns->regs.count != ns->regs.num) { in ns_switch_state()
1779 ns->regs.num - ns->regs.count); in ns_switch_state()
1788 } else if (ns->nxstate & (STATE_DATAIN_MASK | STATE_DATAOUT_MASK)) { in ns_switch_state()
1793 ns->state = ns->nxstate; in ns_switch_state()
1794 ns->nxstate = ns->op[++ns->stateidx + 1]; in ns_switch_state()
1795 ns->regs.num = ns->regs.count = 0; in ns_switch_state()
1799 ns_get_state_name(ns->state), in ns_switch_state()
1800 ns_get_state_name(ns->nxstate)); in ns_switch_state()
1806 switch (NS_STATE(ns->state)) { in ns_switch_state()
1809 ns->regs.num = ns->geom.pgszoob - NS_PAGE_BYTE_SHIFT(ns); in ns_switch_state()
1813 ns->regs.num = ns->geom.idbytes; in ns_switch_state()
1817 ns->regs.count = ns->regs.num = 0; in ns_switch_state()
1824 } else if (ns->nxstate & STATE_ADDR_MASK) { in ns_switch_state()
1830 ns->regs.count = 0; in ns_switch_state()
1832 switch (NS_STATE(ns->nxstate)) { in ns_switch_state()
1834 ns->regs.num = ns->geom.pgaddrbytes; in ns_switch_state()
1838 ns->regs.num = ns->geom.secaddrbytes; in ns_switch_state()
1842 ns->regs.num = 1; in ns_switch_state()
1847 ns->regs.num = ns->geom.pgaddrbytes - ns->geom.secaddrbytes; in ns_switch_state()
1858 ns->regs.num = 0; in ns_switch_state()
1859 ns->regs.count = 0; in ns_switch_state()
1869 if (!ns->lines.ce) { in ns_nand_read_byte()
1873 if (ns->lines.ale || ns->lines.cle) { in ns_nand_read_byte()
1877 if (!(ns->state & STATE_DATAOUT_MASK)) { in ns_nand_read_byte()
1879 ns_get_state_name(ns->state), (uint)outb); in ns_nand_read_byte()
1884 if (NS_STATE(ns->state) == STATE_DATAOUT_STATUS) { in ns_nand_read_byte()
1885 NS_DBG("read_byte: return %#x status\n", ns->regs.status); in ns_nand_read_byte()
1886 return ns->regs.status; in ns_nand_read_byte()
1890 if (ns->regs.count == ns->regs.num) { in ns_nand_read_byte()
1895 switch (NS_STATE(ns->state)) { in ns_nand_read_byte()
1897 if (ns->busw == 8) { in ns_nand_read_byte()
1898 outb = ns->buf.byte[ns->regs.count]; in ns_nand_read_byte()
1899 ns->regs.count += 1; in ns_nand_read_byte()
1901 outb = (u_char)cpu_to_le16(ns->buf.word[ns->regs.count >> 1]); in ns_nand_read_byte()
1902 ns->regs.count += 2; in ns_nand_read_byte()
1906 NS_DBG("read_byte: read ID byte %d, total = %d\n", ns->regs.count, ns->regs.num); in ns_nand_read_byte()
1907 outb = ns->ids[ns->regs.count]; in ns_nand_read_byte()
1908 ns->regs.count += 1; in ns_nand_read_byte()
1914 if (ns->regs.count == ns->regs.num) { in ns_nand_read_byte()
1917 if (NS_STATE(ns->nxstate) == STATE_READY) in ns_nand_read_byte()
1929 if (!ns->lines.ce) { in ns_nand_write_byte()
1933 if (ns->lines.ale && ns->lines.cle) { in ns_nand_write_byte()
1938 if (ns->lines.cle == 1) { in ns_nand_write_byte()
1955 if (NS_STATE(ns->state) == STATE_DATAOUT_STATUS in ns_nand_write_byte()
1956 || NS_STATE(ns->state) == STATE_DATAOUT) { in ns_nand_write_byte()
1957 int row = ns->regs.row; in ns_nand_write_byte()
1961 ns->regs.row = row; in ns_nand_write_byte()
1965 if (NS_STATE(ns->nxstate) != STATE_UNKNOWN && !(ns->nxstate & STATE_CMD_MASK)) { in ns_nand_write_byte()
1967 if (!(ns->regs.command == NAND_CMD_READID && in ns_nand_write_byte()
1968 NS_STATE(ns->state) == STATE_DATAOUT_ID && ns->regs.count == 2)) { in ns_nand_write_byte()
1976 ns_get_state_name(ns->nxstate)); in ns_nand_write_byte()
1983 ns->regs.command = byte; in ns_nand_write_byte()
1986 } else if (ns->lines.ale == 1) { in ns_nand_write_byte()
1991 if (NS_STATE(ns->nxstate) == STATE_UNKNOWN) { in ns_nand_write_byte()
1998 if ((ns->state & ACTION_MASK) && in ns_nand_write_byte()
1999 ns_do_state_action(ns, ns->state) < 0) { in ns_nand_write_byte()
2005 ns->regs.count = 0; in ns_nand_write_byte()
2006 switch (NS_STATE(ns->nxstate)) { in ns_nand_write_byte()
2008 ns->regs.num = ns->geom.pgaddrbytes; in ns_nand_write_byte()
2011 ns->regs.num = ns->geom.secaddrbytes; in ns_nand_write_byte()
2014 ns->regs.num = 1; in ns_nand_write_byte()
2022 if (!(ns->nxstate & STATE_ADDR_MASK)) { in ns_nand_write_byte()
2024 (uint)byte, ns_get_state_name(ns->nxstate)); in ns_nand_write_byte()
2030 if (ns->regs.count == ns->regs.num) { in ns_nand_write_byte()
2038 ns->regs.count += 1; in ns_nand_write_byte()
2041 (uint)byte, ns->regs.count, ns->regs.num); in ns_nand_write_byte()
2043 if (ns->regs.count == ns->regs.num) { in ns_nand_write_byte()
2044 NS_DBG("address (%#x, %#x) is accepted\n", ns->regs.row, ns->regs.column); in ns_nand_write_byte()
2054 if (!(ns->state & STATE_DATAIN_MASK)) { in ns_nand_write_byte()
2056 (uint)byte, ns_get_state_name(ns->state), in ns_nand_write_byte()
2063 if (ns->regs.count == ns->regs.num) { in ns_nand_write_byte()
2065 ns->regs.num); in ns_nand_write_byte()
2069 if (ns->busw == 8) { in ns_nand_write_byte()
2070 ns->buf.byte[ns->regs.count] = byte; in ns_nand_write_byte()
2071 ns->regs.count += 1; in ns_nand_write_byte()
2073 ns->buf.word[ns->regs.count >> 1] = cpu_to_le16((uint16_t)byte); in ns_nand_write_byte()
2074 ns->regs.count += 2; in ns_nand_write_byte()
2087 if (!(ns->state & STATE_DATAIN_MASK)) { in ns_nand_write_buf()
2089 ns_get_state_name(ns->state)); in ns_nand_write_buf()
2095 if (ns->regs.count + len > ns->regs.num) { in ns_nand_write_buf()
2101 memcpy(ns->buf.byte + ns->regs.count, buf, len); in ns_nand_write_buf()
2102 ns->regs.count += len; in ns_nand_write_buf()
2104 if (ns->regs.count == ns->regs.num) { in ns_nand_write_buf()
2105 NS_DBG("write_buf: %d bytes were written\n", ns->regs.count); in ns_nand_write_buf()
2114 if (!ns->lines.ce) { in ns_nand_read_buf()
2118 if (ns->lines.ale || ns->lines.cle) { in ns_nand_read_buf()
2122 if (!(ns->state & STATE_DATAOUT_MASK)) { in ns_nand_read_buf()
2124 ns_get_state_name(ns->state)); in ns_nand_read_buf()
2128 if (NS_STATE(ns->state) != STATE_DATAOUT) { in ns_nand_read_buf()
2138 if (ns->regs.count + len > ns->regs.num) { in ns_nand_read_buf()
2144 memcpy(buf, ns->buf.byte + ns->regs.count, len); in ns_nand_read_buf()
2145 ns->regs.count += len; in ns_nand_read_buf()
2147 if (ns->regs.count == ns->regs.num) { in ns_nand_read_buf()
2148 if (NS_STATE(ns->nxstate) == STATE_READY) in ns_nand_read_buf()
2170 for (op_id = 0; op_id < op->ninstrs; op_id++) { in ns_exec_op()
2171 instr = &op->instrs[op_id]; in ns_exec_op()
2172 if (instr->type == NAND_OP_CMD_INSTR && in ns_exec_op()
2173 (instr->ctx.cmd.opcode == NAND_CMD_READCACHEEND || in ns_exec_op()
2174 instr->ctx.cmd.opcode == NAND_CMD_READCACHESEQ)) in ns_exec_op()
2175 return -EOPNOTSUPP; in ns_exec_op()
2181 ns->lines.ce = 1; in ns_exec_op()
2183 for (op_id = 0; op_id < op->ninstrs; op_id++) { in ns_exec_op()
2184 instr = &op->instrs[op_id]; in ns_exec_op()
2185 ns->lines.cle = 0; in ns_exec_op()
2186 ns->lines.ale = 0; in ns_exec_op()
2188 switch (instr->type) { in ns_exec_op()
2190 ns->lines.cle = 1; in ns_exec_op()
2191 ns_nand_write_byte(chip, instr->ctx.cmd.opcode); in ns_exec_op()
2194 ns->lines.ale = 1; in ns_exec_op()
2195 for (i = 0; i < instr->ctx.addr.naddrs; i++) in ns_exec_op()
2196 ns_nand_write_byte(chip, instr->ctx.addr.addrs[i]); in ns_exec_op()
2199 ns_nand_read_buf(chip, instr->ctx.data.buf.in, instr->ctx.data.len); in ns_exec_op()
2202 ns_nand_write_buf(chip, instr->ctx.data.buf.out, instr->ctx.data.len); in ns_exec_op()
2217 chip->ecc.engine_type = NAND_ECC_ENGINE_TYPE_SOFT; in ns_attach_chip()
2218 chip->ecc.algo = bch ? NAND_ECC_ALGO_BCH : NAND_ECC_ALGO_HAMMING; in ns_attach_chip()
2225 return -EINVAL; in ns_attach_chip()
2228 /* Use 512-byte ecc blocks */ in ns_attach_chip()
2229 eccsteps = nsmtd->writesize / 512; in ns_attach_chip()
2233 if (nsmtd->oobsize < 64 || !eccsteps) { in ns_attach_chip()
2235 return -EINVAL; in ns_attach_chip()
2238 if (((eccbytes * eccsteps) + 2) > nsmtd->oobsize) { in ns_attach_chip()
2240 return -EINVAL; in ns_attach_chip()
2243 chip->ecc.size = 512; in ns_attach_chip()
2244 chip->ecc.strength = bch; in ns_attach_chip()
2245 chip->ecc.bytes = eccbytes; in ns_attach_chip()
2247 NS_INFO("Using %u-bit/%u bytes BCH ECC\n", bch, chip->ecc.size); in ns_attach_chip()
2262 struct list_head *pos, *n; in ns_init_module() local
2269 return -EINVAL; in ns_init_module()
2275 return -ENOMEM; in ns_init_module()
2277 chip = &ns->chip; in ns_init_module()
2283 chip->options |= NAND_SKIP_BBTSCAN; in ns_init_module()
2287 chip->bbt_options |= NAND_BBT_NO_OOB; in ns_init_module()
2290 chip->bbt_options |= NAND_BBT_USE_FLASH; in ns_init_module()
2296 ret = -EINVAL; in ns_init_module()
2304 ns->geom.idbytes = 8; in ns_init_module()
2306 ns->geom.idbytes = 6; in ns_init_module()
2308 ns->geom.idbytes = 4; in ns_init_module()
2310 ns->geom.idbytes = 2; in ns_init_module()
2311 ns->regs.status = NS_STATUS_OK(ns); in ns_init_module()
2312 ns->nxstate = STATE_UNKNOWN; in ns_init_module()
2313 ns->options |= OPT_PAGE512; /* temporary value */ in ns_init_module()
2314 memcpy(ns->ids, id_bytes, sizeof(ns->ids)); in ns_init_module()
2316 ns->busw = 16; in ns_init_module()
2317 chip->options |= NAND_BUSWIDTH_16; in ns_init_module()
2320 nsmtd->owner = THIS_MODULE; in ns_init_module()
2334 nand_controller_init(&ns->base); in ns_init_module()
2335 ns->base.ops = &ns_controller_ops; in ns_init_module()
2336 chip->controller = &ns->base; in ns_init_module()
2345 uint64_t new_size = (uint64_t)nsmtd->erasesize << overridesize; in ns_init_module()
2349 memorg = nanddev_get_memorg(&chip->base); in ns_init_module()
2351 if (new_size >> overridesize != nsmtd->erasesize) { in ns_init_module()
2353 ret = -EINVAL; in ns_init_module()
2358 nsmtd->size = new_size; in ns_init_module()
2359 memorg->eraseblocks_per_lun = 1 << overridesize; in ns_init_module()
2360 targetsize = nanddev_target_size(&chip->base); in ns_init_module()
2361 chip->chip_shift = ffs(nsmtd->erasesize) + overridesize - 1; in ns_init_module()
2362 chip->pagemask = (targetsize >> chip->page_shift) - 1; in ns_init_module()
2382 ret = mtd_device_register(nsmtd, &ns->partitions[0], ns->nbparts); in ns_init_module()
2401 list_for_each_safe(pos, n, &grave_pages) { in ns_init_module()
2402 list_del(pos); in ns_init_module()
2403 kfree(list_entry(pos, struct grave_page, list)); in ns_init_module()
2406 list_for_each_safe(pos, n, &weak_pages) { in ns_init_module()
2407 list_del(pos); in ns_init_module()
2408 kfree(list_entry(pos, struct weak_page, list)); in ns_init_module()
2411 list_for_each_safe(pos, n, &weak_blocks) { in ns_init_module()
2412 list_del(pos); in ns_init_module()
2413 kfree(list_entry(pos, struct weak_block, list)); in ns_init_module()
2424 * Module clean-up function
2430 struct list_head *pos, *n; in ns_cleanup_module() local
2438 list_for_each_safe(pos, n, &grave_pages) { in ns_cleanup_module()
2439 list_del(pos); in ns_cleanup_module()
2440 kfree(list_entry(pos, struct grave_page, list)); in ns_cleanup_module()
2443 list_for_each_safe(pos, n, &weak_pages) { in ns_cleanup_module()
2444 list_del(pos); in ns_cleanup_module()
2445 kfree(list_entry(pos, struct weak_page, list)); in ns_cleanup_module()
2448 list_for_each_safe(pos, n, &weak_blocks) { in ns_cleanup_module()
2449 list_del(pos); in ns_cleanup_module()
2450 kfree(list_entry(pos, struct weak_block, list)); in ns_cleanup_module()