Commit a62d016c authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'for-linus-20150422' of git://git.infradead.org/linux-mtd

Pull MTD updates from Brian Norris:
 "Common MTD:

   - Add Kconfig option for keeping both the 'master' and 'partition'
     MTDs registered as devices.  This would really make a better
     default if we could do it over, as it allows a lot more flexibility
     in (1) determining the flash topology of the system from user-space
     and (2) adding temporary partitions at runtime (ioctl(BLKPG)).

     Unfortunately, this would possibly cause user-space breakage, as it
     will cause renumbering of the /dev/mtdX devices.  We'll see if we
     can change this in the future, as there have already been a few
     people looking for this feature, and I know others have just been
     working around our current limitations instead of fixing them this
     way.

   - Along with the previous change, add some additional information to
     sysfs, so user-space can read the offset of each partition within
     its master device

  SPI NOR:

   - add new device tree compatible binding to represent the
     mostly-compatible class of SPI NOR flash which can be detected by
     their extended JEDEC ID bytes, cutting down the duplication of our
     ID tables

   - misc.  new IDs

  Various other miscellaneous fixes and changes"

* tag 'for-linus-20150422' of git://git.infradead.org/linux-mtd: (53 commits)
  mtd: spi-nor: Add support for Macronix mx25u6435f serial flash
  mtd: spi-nor: Add support for Winbond w25q64dw serial flash
  mtd: spi-nor: add support for the Winbond W25X05 flash
  mtd: spi-nor: support en25s64 device
  mtd: m25p80: bind to "nor-jedec" ID, for auto-detection
  Documentation: devicetree: m25p80: add "nor-jedec" binding
  mtd: Make MTD tests cancelable
  mtd: mtd_oobtest: Fix bitflip_limit usage in test case 3
  mtd: docg3: remove invalid __exit annotations
  mtd: fsl_ifc_nand: use msecs_to_jiffies for time conversion
  mtd: atmel_nand: don't map the ROM table if no pmecc table offset in DT
  mtd: atmel_nand: add a definition for the oob reserved bytes
  mtd: part: Remove partition overlap checks
  mtd: part: Add sysfs variable for offset of partition
  mtd: part: Create the master device node when partitioned
  mtd: ts5500_flash: Fix typo in MODULE_DESCRIPTION in ts5500_flash.c
  mtd: denali: Disable sub-page writes in Denali NAND driver
  mtd: pxa3xx_nand: cleanup wait_for_completion handling
  mtd: nand: gpmi: Check for scan_bbt() error
  mtd: nand: gpmi: fixup return type of wait_for_completion_timeout
  ...
parents 7c034dfd 3e550d23
...@@ -222,3 +222,13 @@ Description: ...@@ -222,3 +222,13 @@ Description:
The number of blocks that are marked as reserved, if any, in The number of blocks that are marked as reserved, if any, in
this partition. These are typically used to store the in-flash this partition. These are typically used to store the in-flash
bad block table (BBT). bad block table (BBT).
What: /sys/class/mtd/mtdX/offset
Date: March 2015
KernelVersion: 4.1
Contact: linux-mtd@lists.infradead.org
Description:
For a partition, the offset of that partition from the start
of the master device in bytes. This attribute is absent on
main devices, so it can be used to distinguish between
partitions and devices that aren't partitions.
...@@ -3,10 +3,13 @@ ...@@ -3,10 +3,13 @@
Required properties: Required properties:
- #address-cells, #size-cells : Must be present if the device has sub-nodes - #address-cells, #size-cells : Must be present if the device has sub-nodes
representing partitions. representing partitions.
- compatible : Should be the manufacturer and the name of the chip. Bear in mind - compatible : May include a device-specific string consisting of the
the DT binding is not Linux-only, but in case of Linux, see the manufacturer and name of the chip. Bear in mind the DT binding
"spi_nor_ids" table in drivers/mtd/spi-nor/spi-nor.c for the list is not Linux-only, but in case of Linux, see the "m25p_ids"
of supported chips. table in drivers/mtd/devices/m25p80.c for the list of supported
chips.
Must also include "nor-jedec" for any SPI NOR flash that can be
identified by the JEDEC READ ID opcode (0x9F).
- reg : Chip-Select number - reg : Chip-Select number
- spi-max-frequency : Maximum frequency of the SPI bus the chip can operate at - spi-max-frequency : Maximum frequency of the SPI bus the chip can operate at
...@@ -22,7 +25,7 @@ Example: ...@@ -22,7 +25,7 @@ Example:
flash: m25p80@0 { flash: m25p80@0 {
#address-cells = <1>; #address-cells = <1>;
#size-cells = <1>; #size-cells = <1>;
compatible = "spansion,m25p80"; compatible = "spansion,m25p80", "nor-jedec";
reg = <0>; reg = <0>;
spi-max-frequency = <40000000>; spi-max-frequency = <40000000>;
m25p,fast-read; m25p,fast-read;
......
...@@ -14,7 +14,7 @@ Optional properties: ...@@ -14,7 +14,7 @@ Optional properties:
- marvell,nand-enable-arbiter: Set to enable the bus arbiter - marvell,nand-enable-arbiter: Set to enable the bus arbiter
- marvell,nand-keep-config: Set to keep the NAND controller config as set - marvell,nand-keep-config: Set to keep the NAND controller config as set
by the bootloader by the bootloader
- num-cs: Number of chipselect lines to usw - num-cs: Number of chipselect lines to use
- nand-on-flash-bbt: boolean to enable on flash bbt option if - nand-on-flash-bbt: boolean to enable on flash bbt option if
not present false not present false
- nand-ecc-strength: number of bits to correct per ECC step - nand-ecc-strength: number of bits to correct per ECC step
......
...@@ -21,7 +21,7 @@ Optional properties: ...@@ -21,7 +21,7 @@ Optional properties:
- nand-ecc-mode : one of the supported ECC modes ("hw", "hw_syndrome", "soft", - nand-ecc-mode : one of the supported ECC modes ("hw", "hw_syndrome", "soft",
"soft_bch" or "none") "soft_bch" or "none")
see Documentation/devicetree/mtd/nand.txt for generic bindings. see Documentation/devicetree/bindings/mtd/nand.txt for generic bindings.
Examples: Examples:
......
...@@ -309,6 +309,19 @@ config MTD_SWAP ...@@ -309,6 +309,19 @@ config MTD_SWAP
The driver provides wear leveling by storing erase counter into the The driver provides wear leveling by storing erase counter into the
OOB. OOB.
config MTD_PARTITIONED_MASTER
bool "Retain master device when partitioned"
default n
depends on MTD
help
For historical reasons, by default, either a master is present or
several partitions are present, but not both. The concern was that
data listed in multiple partitions was dangerous; however, SCSI does
this and it is frequently useful for applications. This config option
leaves the master in even if the device is partitioned. It also makes
the parent of the partition device be the master device, rather than
what lies behind the master.
source "drivers/mtd/chips/Kconfig" source "drivers/mtd/chips/Kconfig"
source "drivers/mtd/maps/Kconfig" source "drivers/mtd/maps/Kconfig"
......
...@@ -9,7 +9,15 @@ ...@@ -9,7 +9,15 @@
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
/*
* When the first attempt at device initialization fails, we may need to
* wait a little bit and retry. This timeout, by default 3 seconds, gives
* device time to start up. Required on BCM2708 and a few other chipsets.
*/
#define MTD_DEFAULT_TIMEOUT 3
#include <linux/module.h> #include <linux/module.h>
#include <linux/delay.h>
#include <linux/fs.h> #include <linux/fs.h>
#include <linux/blkdev.h> #include <linux/blkdev.h>
#include <linux/bio.h> #include <linux/bio.h>
...@@ -209,10 +217,14 @@ static void block2mtd_free_device(struct block2mtd_dev *dev) ...@@ -209,10 +217,14 @@ static void block2mtd_free_device(struct block2mtd_dev *dev)
} }
static struct block2mtd_dev *add_device(char *devname, int erase_size) static struct block2mtd_dev *add_device(char *devname, int erase_size,
int timeout)
{ {
#ifndef MODULE
int i;
#endif
const fmode_t mode = FMODE_READ | FMODE_WRITE | FMODE_EXCL; const fmode_t mode = FMODE_READ | FMODE_WRITE | FMODE_EXCL;
struct block_device *bdev; struct block_device *bdev = ERR_PTR(-ENODEV);
struct block2mtd_dev *dev; struct block2mtd_dev *dev;
char *name; char *name;
...@@ -225,14 +237,27 @@ static struct block2mtd_dev *add_device(char *devname, int erase_size) ...@@ -225,14 +237,27 @@ static struct block2mtd_dev *add_device(char *devname, int erase_size)
/* Get a handle on the device */ /* Get a handle on the device */
bdev = blkdev_get_by_path(devname, mode, dev); bdev = blkdev_get_by_path(devname, mode, dev);
#ifndef MODULE
if (IS_ERR(bdev)) {
/* We might not have rootfs mounted at this point. Try #ifndef MODULE
to resolve the device name by other means. */ /*
* We might not have the root device mounted at this point.
* Try to resolve the device name by other means.
*/
for (i = 0; IS_ERR(bdev) && i <= timeout; i++) {
dev_t devt;
if (i)
/*
* Calling wait_for_device_probe in the first loop
* was not enough, sleep for a bit in subsequent
* go-arounds.
*/
msleep(1000);
wait_for_device_probe();
dev_t devt = name_to_dev_t(devname); devt = name_to_dev_t(devname);
if (devt) if (!devt)
continue;
bdev = blkdev_get_by_dev(devt, mode, dev); bdev = blkdev_get_by_dev(devt, mode, dev);
} }
#endif #endif
...@@ -280,6 +305,7 @@ static struct block2mtd_dev *add_device(char *devname, int erase_size) ...@@ -280,6 +305,7 @@ static struct block2mtd_dev *add_device(char *devname, int erase_size)
/* Device didn't get added, so free the entry */ /* Device didn't get added, so free the entry */
goto err_destroy_mutex; goto err_destroy_mutex;
} }
list_add(&dev->list, &blkmtd_device_list); list_add(&dev->list, &blkmtd_device_list);
pr_info("mtd%d: [%s] erase_size = %dKiB [%d]\n", pr_info("mtd%d: [%s] erase_size = %dKiB [%d]\n",
dev->mtd.index, dev->mtd.index,
...@@ -348,16 +374,19 @@ static inline void kill_final_newline(char *str) ...@@ -348,16 +374,19 @@ static inline void kill_final_newline(char *str)
#ifndef MODULE #ifndef MODULE
static int block2mtd_init_called = 0; static int block2mtd_init_called = 0;
static char block2mtd_paramline[80 + 12]; /* 80 for device, 12 for erase size */ /* 80 for device, 12 for erase size */
static char block2mtd_paramline[80 + 12];
#endif #endif
static int block2mtd_setup2(const char *val) static int block2mtd_setup2(const char *val)
{ {
char buf[80 + 12]; /* 80 for device, 12 for erase size */ /* 80 for device, 12 for erase size, 80 for name, 8 for timeout */
char buf[80 + 12 + 80 + 8];
char *str = buf; char *str = buf;
char *token[2]; char *token[2];
char *name; char *name;
size_t erase_size = PAGE_SIZE; size_t erase_size = PAGE_SIZE;
unsigned long timeout = MTD_DEFAULT_TIMEOUT;
int i, ret; int i, ret;
if (strnlen(val, sizeof(buf)) >= sizeof(buf)) { if (strnlen(val, sizeof(buf)) >= sizeof(buf)) {
...@@ -395,7 +424,7 @@ static int block2mtd_setup2(const char *val) ...@@ -395,7 +424,7 @@ static int block2mtd_setup2(const char *val)
} }
} }
add_device(name, erase_size); add_device(name, erase_size, timeout);
return 0; return 0;
} }
...@@ -463,8 +492,7 @@ static void block2mtd_exit(void) ...@@ -463,8 +492,7 @@ static void block2mtd_exit(void)
} }
} }
late_initcall(block2mtd_init);
module_init(block2mtd_init);
module_exit(block2mtd_exit); module_exit(block2mtd_exit);
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
......
...@@ -1805,7 +1805,7 @@ static int __init doc_dbg_register(struct docg3 *docg3) ...@@ -1805,7 +1805,7 @@ static int __init doc_dbg_register(struct docg3 *docg3)
} }
} }
static void __exit doc_dbg_unregister(struct docg3 *docg3) static void doc_dbg_unregister(struct docg3 *docg3)
{ {
debugfs_remove_recursive(docg3->debugfs_root); debugfs_remove_recursive(docg3->debugfs_root);
} }
...@@ -2033,7 +2033,7 @@ static int __init docg3_probe(struct platform_device *pdev) ...@@ -2033,7 +2033,7 @@ static int __init docg3_probe(struct platform_device *pdev)
struct mtd_info *mtd; struct mtd_info *mtd;
struct resource *ress; struct resource *ress;
void __iomem *base; void __iomem *base;
int ret, floor, found = 0; int ret, floor;
struct docg3_cascade *cascade; struct docg3_cascade *cascade;
ret = -ENXIO; ret = -ENXIO;
...@@ -2073,14 +2073,11 @@ static int __init docg3_probe(struct platform_device *pdev) ...@@ -2073,14 +2073,11 @@ static int __init docg3_probe(struct platform_device *pdev)
0); 0);
if (ret) if (ret)
goto err_probe; goto err_probe;
found++;
} }
ret = doc_register_sysfs(pdev, cascade); ret = doc_register_sysfs(pdev, cascade);
if (ret) if (ret)
goto err_probe; goto err_probe;
if (!found)
goto notfound;
platform_set_drvdata(pdev, cascade); platform_set_drvdata(pdev, cascade);
doc_dbg_register(cascade->floors[0]->priv); doc_dbg_register(cascade->floors[0]->priv);
...@@ -2103,7 +2100,7 @@ static int __init docg3_probe(struct platform_device *pdev) ...@@ -2103,7 +2100,7 @@ static int __init docg3_probe(struct platform_device *pdev)
* *
* Returns 0 * Returns 0
*/ */
static int __exit docg3_release(struct platform_device *pdev) static int docg3_release(struct platform_device *pdev)
{ {
struct docg3_cascade *cascade = platform_get_drvdata(pdev); struct docg3_cascade *cascade = platform_get_drvdata(pdev);
struct docg3 *docg3 = cascade->floors[0]->priv; struct docg3 *docg3 = cascade->floors[0]->priv;
...@@ -2134,7 +2131,7 @@ static struct platform_driver g3_driver = { ...@@ -2134,7 +2131,7 @@ static struct platform_driver g3_driver = {
}, },
.suspend = docg3_suspend, .suspend = docg3_suspend,
.resume = docg3_resume, .resume = docg3_resume,
.remove = __exit_p(docg3_release), .remove = docg3_release,
}; };
module_platform_driver_probe(g3_driver, docg3_probe); module_platform_driver_probe(g3_driver, docg3_probe);
......
...@@ -223,6 +223,8 @@ static int m25p_probe(struct spi_device *spi) ...@@ -223,6 +223,8 @@ static int m25p_probe(struct spi_device *spi)
*/ */
if (data && data->type) if (data && data->type)
flash_name = data->type; flash_name = data->type;
else if (!strcmp(spi->modalias, "nor-jedec"))
flash_name = NULL; /* auto-detect */
else else
flash_name = spi->modalias; flash_name = spi->modalias;
...@@ -247,9 +249,16 @@ static int m25p_remove(struct spi_device *spi) ...@@ -247,9 +249,16 @@ static int m25p_remove(struct spi_device *spi)
} }
/* /*
* XXX This needs to be kept in sync with spi_nor_ids. We can't share * Do NOT add to this array without reading the following:
* it with spi-nor, because if this is built as a module then modpost *
* won't be able to read it and add appropriate aliases. * Historically, many flash devices are bound to this driver by their name. But
* since most of these flash are compatible to some extent, and their
* differences can often be differentiated by the JEDEC read-ID command, we
* encourage new users to add support to the spi-nor library, and simply bind
* against a generic string here (e.g., "nor-jedec").
*
* Many flash names are kept here in this list (as well as in spi-nor.c) to
* keep them available as module aliases for existing platforms.
*/ */
static const struct spi_device_id m25p_ids[] = { static const struct spi_device_id m25p_ids[] = {
{"at25fs010"}, {"at25fs040"}, {"at25df041a"}, {"at25df321a"}, {"at25fs010"}, {"at25fs040"}, {"at25df041a"}, {"at25df321a"},
...@@ -291,6 +300,12 @@ static const struct spi_device_id m25p_ids[] = { ...@@ -291,6 +300,12 @@ static const struct spi_device_id m25p_ids[] = {
{"w25x64"}, {"w25q64"}, {"w25q80"}, {"w25q80bl"}, {"w25x64"}, {"w25q64"}, {"w25q80"}, {"w25q80bl"},
{"w25q128"}, {"w25q256"}, {"cat25c11"}, {"w25q128"}, {"w25q256"}, {"cat25c11"},
{"cat25c03"}, {"cat25c09"}, {"cat25c17"}, {"cat25128"}, {"cat25c03"}, {"cat25c09"}, {"cat25c17"}, {"cat25128"},
/*
* Generic support for SPI NOR that can be identified by the JEDEC READ
* ID opcode (0x9F). Use this, if possible.
*/
{"nor-jedec"},
{ }, { },
}; };
MODULE_DEVICE_TABLE(spi, m25p_ids); MODULE_DEVICE_TABLE(spi, m25p_ids);
......
...@@ -242,7 +242,7 @@ config MTD_L440GX ...@@ -242,7 +242,7 @@ config MTD_L440GX
config MTD_CFI_FLAGADM config MTD_CFI_FLAGADM
tristate "CFI Flash device mapping on FlagaDM" tristate "CFI Flash device mapping on FlagaDM"
depends on 8xx && MTD_CFI depends on PPC_8xx && MTD_CFI
help help
Mapping for the Flaga digital module. If you don't have one, ignore Mapping for the Flaga digital module. If you don't have one, ignore
this setting. this setting.
......
...@@ -274,7 +274,7 @@ static int sa1100_mtd_probe(struct platform_device *pdev) ...@@ -274,7 +274,7 @@ static int sa1100_mtd_probe(struct platform_device *pdev)
return err; return err;
} }
static int __exit sa1100_mtd_remove(struct platform_device *pdev) static int sa1100_mtd_remove(struct platform_device *pdev)
{ {
struct sa_info *info = platform_get_drvdata(pdev); struct sa_info *info = platform_get_drvdata(pdev);
struct flash_platform_data *plat = dev_get_platdata(&pdev->dev); struct flash_platform_data *plat = dev_get_platdata(&pdev->dev);
...@@ -286,7 +286,7 @@ static int __exit sa1100_mtd_remove(struct platform_device *pdev) ...@@ -286,7 +286,7 @@ static int __exit sa1100_mtd_remove(struct platform_device *pdev)
static struct platform_driver sa1100_mtd_driver = { static struct platform_driver sa1100_mtd_driver = {
.probe = sa1100_mtd_probe, .probe = sa1100_mtd_probe,
.remove = __exit_p(sa1100_mtd_remove), .remove = sa1100_mtd_remove,
.driver = { .driver = {
.name = "sa1100-mtd", .name = "sa1100-mtd",
}, },
......
...@@ -117,5 +117,5 @@ module_exit(cleanup_ts5500_map); ...@@ -117,5 +117,5 @@ module_exit(cleanup_ts5500_map);
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_AUTHOR("Sean Young <sean@mess.org>"); MODULE_AUTHOR("Sean Young <sean@mess.org>");
MODULE_DESCRIPTION("MTD map driver for Techology Systems TS-5500 board"); MODULE_DESCRIPTION("MTD map driver for Technology Systems TS-5500 board");
...@@ -171,9 +171,6 @@ static void mtd_blktrans_work(struct work_struct *work) ...@@ -171,9 +171,6 @@ static void mtd_blktrans_work(struct work_struct *work)
background_done = 0; background_done = 0;
} }
if (req)
__blk_end_request_all(req, -EIO);
spin_unlock_irq(rq->queue_lock); spin_unlock_irq(rq->queue_lock);
} }
......
...@@ -38,6 +38,7 @@ ...@@ -38,6 +38,7 @@
#include <linux/gfp.h> #include <linux/gfp.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/reboot.h> #include <linux/reboot.h>
#include <linux/kconfig.h>
#include <linux/mtd/mtd.h> #include <linux/mtd/mtd.h>
#include <linux/mtd/partitions.h> #include <linux/mtd/partitions.h>
...@@ -501,6 +502,29 @@ int del_mtd_device(struct mtd_info *mtd) ...@@ -501,6 +502,29 @@ int del_mtd_device(struct mtd_info *mtd)
return ret; return ret;
} }
static int mtd_add_device_partitions(struct mtd_info *mtd,
struct mtd_partition *real_parts,
int nbparts)
{
int ret;
if (nbparts == 0 || IS_ENABLED(CONFIG_MTD_PARTITIONED_MASTER)) {
ret = add_mtd_device(mtd);
if (ret == 1)
return -ENODEV;
}
if (nbparts > 0) {
ret = add_mtd_partitions(mtd, real_parts, nbparts);
if (ret && IS_ENABLED(CONFIG_MTD_PARTITIONED_MASTER))
del_mtd_device(mtd);
return ret;
}
return 0;
}
/** /**
* mtd_device_parse_register - parse partitions and register an MTD device. * mtd_device_parse_register - parse partitions and register an MTD device.
* *
...@@ -523,7 +547,8 @@ int del_mtd_device(struct mtd_info *mtd) ...@@ -523,7 +547,8 @@ int del_mtd_device(struct mtd_info *mtd)
* found this functions tries to fallback to information specified in * found this functions tries to fallback to information specified in
* @parts/@nr_parts. * @parts/@nr_parts.
* * If any partitioning info was found, this function registers the found * * If any partitioning info was found, this function registers the found
* partitions. * partitions. If the MTD_PARTITIONED_MASTER option is set, then the device
* as a whole is registered first.
* * If no partitions were found this function just registers the MTD device * * If no partitions were found this function just registers the MTD device
* @mtd and exits. * @mtd and exits.
* *
...@@ -534,27 +559,21 @@ int mtd_device_parse_register(struct mtd_info *mtd, const char * const *types, ...@@ -534,27 +559,21 @@ int mtd_device_parse_register(struct mtd_info *mtd, const char * const *types,
const struct mtd_partition *parts, const struct mtd_partition *parts,
int nr_parts) int nr_parts)
{ {
int err; int ret;
struct mtd_partition *real_parts; struct mtd_partition *real_parts = NULL;
err = parse_mtd_partitions(mtd, types, &real_parts, parser_data); ret = parse_mtd_partitions(mtd, types, &real_parts, parser_data);
if (err <= 0 && nr_parts && parts) { if (ret <= 0 && nr_parts && parts) {
real_parts = kmemdup(parts, sizeof(*parts) * nr_parts, real_parts = kmemdup(parts, sizeof(*parts) * nr_parts,
GFP_KERNEL); GFP_KERNEL);
if (!real_parts) if (!real_parts)
err = -ENOMEM; ret = -ENOMEM;
else else
err = nr_parts; ret = nr_parts;
} }
if (err > 0) { if (ret >= 0)
err = add_mtd_partitions(mtd, real_parts, err); ret = mtd_add_device_partitions(mtd, real_parts, ret);
kfree(real_parts);
} else if (err == 0) {
err = add_mtd_device(mtd);
if (err == 1)
err = -ENODEV;
}
/* /*
* FIXME: some drivers unfortunately call this function more than once. * FIXME: some drivers unfortunately call this function more than once.
...@@ -569,7 +588,8 @@ int mtd_device_parse_register(struct mtd_info *mtd, const char * const *types, ...@@ -569,7 +588,8 @@ int mtd_device_parse_register(struct mtd_info *mtd, const char * const *types,
register_reboot_notifier(&mtd->reboot_notifier); register_reboot_notifier(&mtd->reboot_notifier);
} }
return err; kfree(real_parts);
return ret;
} }
EXPORT_SYMBOL_GPL(mtd_device_parse_register); EXPORT_SYMBOL_GPL(mtd_device_parse_register);
......
...@@ -30,6 +30,7 @@ ...@@ -30,6 +30,7 @@
#include <linux/mtd/mtd.h> #include <linux/mtd/mtd.h>
#include <linux/mtd/partitions.h> #include <linux/mtd/partitions.h>
#include <linux/err.h> #include <linux/err.h>
#include <linux/kconfig.h>
#include "mtdcore.h" #include "mtdcore.h"
...@@ -379,10 +380,17 @@ static struct mtd_part *allocate_partition(struct mtd_info *master, ...@@ -379,10 +380,17 @@ static struct mtd_part *allocate_partition(struct mtd_info *master,
slave->mtd.name = name; slave->mtd.name = name;
slave->mtd.owner = master->owner; slave->mtd.owner = master->owner;
/* NOTE: we don't arrange MTDs as a tree; it'd be error-prone /* NOTE: Historically, we didn't arrange MTDs as a tree out of
* to have the same data be in two different partitions. * concern for showing the same data in multiple partitions.
* However, it is very useful to have the master node present,
* so the MTD_PARTITIONED_MASTER option allows that. The master
* will have device nodes etc only if this is set, so make the
* parent conditional on that option. Note, this is a way to
* distinguish between the master and the partition in sysfs.
*/ */
slave->mtd.dev.parent = master->dev.parent; slave->mtd.dev.parent = IS_ENABLED(CONFIG_MTD_PARTITIONED_MASTER) ?
&master->dev :
master->dev.parent;
slave->mtd._read = part_read; slave->mtd._read = part_read;
slave->mtd._write = part_write; slave->mtd._write = part_write;
...@@ -546,12 +554,35 @@ static struct mtd_part *allocate_partition(struct mtd_info *master, ...@@ -546,12 +554,35 @@ static struct mtd_part *allocate_partition(struct mtd_info *master,
return slave; return slave;
} }
static ssize_t mtd_partition_offset_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct mtd_info *mtd = dev_get_drvdata(dev);
struct mtd_part *part = PART(mtd);
return snprintf(buf, PAGE_SIZE, "%lld\n", part->offset);
}
static DEVICE_ATTR(offset, S_IRUGO, mtd_partition_offset_show, NULL);
static const struct attribute *mtd_partition_attrs[] = {
&dev_attr_offset.attr,
NULL
};
static int mtd_add_partition_attrs(struct mtd_part *new)
{
int ret = sysfs_create_files(&new->mtd.dev.kobj, mtd_partition_attrs);
if (ret)
printk(KERN_WARNING
"mtd: failed to create partition attrs, err=%d\n", ret);
return ret;
}
int mtd_add_partition(struct mtd_info *master, const char *name, int mtd_add_partition(struct mtd_info *master, const char *name,
long long offset, long long length) long long offset, long long length)
{ {
struct mtd_partition part; struct mtd_partition part;
struct mtd_part *p, *new; struct mtd_part *new;
uint64_t start, end;
int ret = 0; int ret = 0;
/* the direct offset is expected */ /* the direct offset is expected */
...@@ -575,31 +606,15 @@ int mtd_add_partition(struct mtd_info *master, const char *name, ...@@ -575,31 +606,15 @@ int mtd_add_partition(struct mtd_info *master, const char *name,
if (IS_ERR(new)) if (IS_ERR(new))
return PTR_ERR(new); return PTR_ERR(new);
start = offset;
end = offset + length;
mutex_lock(&mtd_partitions_mutex); mutex_lock(&mtd_partitions_mutex);
list_for_each_entry(p, &mtd_partitions, list)
if (p->master == master) {
if ((start >= p->offset) &&
(start < (p->offset + p->mtd.size)))
goto err_inv;
if ((end >= p->offset) &&
(end < (p->offset + p->mtd.size)))
goto err_inv;
}
list_add(&new->list, &mtd_partitions); list_add(&new->list, &mtd_partitions);
mutex_unlock(&mtd_partitions_mutex); mutex_unlock(&mtd_partitions_mutex);
add_mtd_device(&new->mtd); add_mtd_device(&new->mtd);
mtd_add_partition_attrs(new);
return ret; return ret;
err_inv:
mutex_unlock(&mtd_partitions_mutex);
free_partition(new);
return -EINVAL;
} }
EXPORT_SYMBOL_GPL(mtd_add_partition); EXPORT_SYMBOL_GPL(mtd_add_partition);
...@@ -612,6 +627,8 @@ int mtd_del_partition(struct mtd_info *master, int partno) ...@@ -612,6 +627,8 @@ int mtd_del_partition(struct mtd_info *master, int partno)
list_for_each_entry_safe(slave, next, &mtd_partitions, list) list_for_each_entry_safe(slave, next, &mtd_partitions, list)
if ((slave->master == master) && if ((slave->master == master) &&
(slave->mtd.index == partno)) { (slave->mtd.index == partno)) {
sysfs_remove_files(&slave->mtd.dev.kobj,
mtd_partition_attrs);
ret = del_mtd_device(&slave->mtd); ret = del_mtd_device(&slave->mtd);
if (ret < 0) if (ret < 0)
break; break;
...@@ -631,8 +648,8 @@ EXPORT_SYMBOL_GPL(mtd_del_partition); ...@@ -631,8 +648,8 @@ EXPORT_SYMBOL_GPL(mtd_del_partition);
* and registers slave MTD objects which are bound to the master according to * and registers slave MTD objects which are bound to the master according to
* the partition definitions. * the partition definitions.
* *
* We don't register the master, or expect the caller to have done so, * For historical reasons, this function's caller only registers the master
* for reasons of data integrity. * if the MTD_PARTITIONED_MASTER config option is set.
*/ */
int add_mtd_partitions(struct mtd_info *master, int add_mtd_partitions(struct mtd_info *master,
...@@ -655,6 +672,7 @@ int add_mtd_partitions(struct mtd_info *master, ...@@ -655,6 +672,7 @@ int add_mtd_partitions(struct mtd_info *master,
mutex_unlock(&mtd_partitions_mutex); mutex_unlock(&mtd_partitions_mutex);
add_mtd_device(&slave->mtd); add_mtd_device(&slave->mtd);
mtd_add_partition_attrs(slave);
cur_offset = slave->offset + slave->mtd.size; cur_offset = slave->offset + slave->mtd.size;
} }
......
...@@ -485,7 +485,7 @@ static void pmecc_config_ecc_layout(struct nand_ecclayout *layout, ...@@ -485,7 +485,7 @@ static void pmecc_config_ecc_layout(struct nand_ecclayout *layout,
for (i = 0; i < ecc_len; i++) for (i = 0; i < ecc_len; i++)
layout->eccpos[i] = oobsize - ecc_len + i; layout->eccpos[i] = oobsize - ecc_len + i;
layout->oobfree[0].offset = 2; layout->oobfree[0].offset = PMECC_OOB_RESERVED_BYTES;
layout->oobfree[0].length = layout->oobfree[0].length =
oobsize - ecc_len - layout->oobfree[0].offset; oobsize - ecc_len - layout->oobfree[0].offset;
} }
...@@ -1204,15 +1204,15 @@ static int atmel_pmecc_nand_init_params(struct platform_device *pdev, ...@@ -1204,15 +1204,15 @@ static int atmel_pmecc_nand_init_params(struct platform_device *pdev,
goto err; goto err;
} }
if (!host->has_no_lookup_table) {
regs_rom = platform_get_resource(pdev, IORESOURCE_MEM, 3); regs_rom = platform_get_resource(pdev, IORESOURCE_MEM, 3);
host->pmecc_rom_base = devm_ioremap_resource(&pdev->dev, regs_rom); host->pmecc_rom_base = devm_ioremap_resource(&pdev->dev,
regs_rom);
if (IS_ERR(host->pmecc_rom_base)) { if (IS_ERR(host->pmecc_rom_base)) {
if (!host->has_no_lookup_table)
/* Don't display the information again */
dev_err(host->dev, "Can not get I/O resource for ROM, will build a lookup table in runtime!\n"); dev_err(host->dev, "Can not get I/O resource for ROM, will build a lookup table in runtime!\n");
host->has_no_lookup_table = true; host->has_no_lookup_table = true;
} }
}
if (host->has_no_lookup_table) { if (host->has_no_lookup_table) {
/* Build the look-up table in runtime */ /* Build the look-up table in runtime */
...@@ -1254,7 +1254,8 @@ static int atmel_pmecc_nand_init_params(struct platform_device *pdev, ...@@ -1254,7 +1254,8 @@ static int atmel_pmecc_nand_init_params(struct platform_device *pdev,
nand_chip->ecc.steps = mtd->writesize / sector_size; nand_chip->ecc.steps = mtd->writesize / sector_size;
nand_chip->ecc.total = nand_chip->ecc.bytes * nand_chip->ecc.total = nand_chip->ecc.bytes *
nand_chip->ecc.steps; nand_chip->ecc.steps;
if (nand_chip->ecc.total > mtd->oobsize - 2) { if (nand_chip->ecc.total >
mtd->oobsize - PMECC_OOB_RESERVED_BYTES) {
dev_err(host->dev, "No room for ECC bytes\n"); dev_err(host->dev, "No room for ECC bytes\n");
err_no = -EINVAL; err_no = -EINVAL;
goto err; goto err;
...@@ -1719,7 +1720,7 @@ static int nfc_wait_interrupt(struct atmel_nand_host *host, u32 flag) ...@@ -1719,7 +1720,7 @@ static int nfc_wait_interrupt(struct atmel_nand_host *host, u32 flag)
comp[index++] = &host->nfc->comp_cmd_done; comp[index++] = &host->nfc->comp_cmd_done;
if (index == 0) { if (index == 0) {
dev_err(host->dev, "Unkown interrupt flag: 0x%08x\n", flag); dev_err(host->dev, "Unknown interrupt flag: 0x%08x\n", flag);
return -EINVAL; return -EINVAL;
} }
...@@ -1752,11 +1753,10 @@ static int nfc_send_command(struct atmel_nand_host *host, ...@@ -1752,11 +1753,10 @@ static int nfc_send_command(struct atmel_nand_host *host,
cmd, addr, cycle0); cmd, addr, cycle0);
timeout = jiffies + msecs_to_jiffies(NFC_TIME_OUT_MS); timeout = jiffies + msecs_to_jiffies(NFC_TIME_OUT_MS);
while (nfc_cmd_readl(NFCADDR_CMD_NFCBUSY, host->nfc->base_cmd_regs) while (nfc_readl(host->nfc->hsmc_regs, SR) & NFC_SR_BUSY) {
& NFCADDR_CMD_NFCBUSY) {
if (time_after(jiffies, timeout)) { if (time_after(jiffies, timeout)) {
dev_err(host->dev, dev_err(host->dev,
"Time out to wait CMD_NFCBUSY ready!\n"); "Time out to wait for NFC ready!\n");
return -ETIMEDOUT; return -ETIMEDOUT;
} }
} }
......
...@@ -152,4 +152,7 @@ ...@@ -152,4 +152,7 @@
/* Time out value for reading PMECC status register */ /* Time out value for reading PMECC status register */
#define PMECC_MAX_TIMEOUT_MS 100 #define PMECC_MAX_TIMEOUT_MS 100
/* Reserved bytes in oob area */
#define PMECC_OOB_RESERVED_BYTES 2
#endif #endif
...@@ -35,6 +35,7 @@ ...@@ -35,6 +35,7 @@
#define NFC_CTRL_DISABLE (1 << 1) #define NFC_CTRL_DISABLE (1 << 1)
#define ATMEL_HSMC_NFC_SR 0x08 /* NFC Status Register */ #define ATMEL_HSMC_NFC_SR 0x08 /* NFC Status Register */
#define NFC_SR_BUSY (1 << 8)
#define NFC_SR_XFR_DONE (1 << 16) #define NFC_SR_XFR_DONE (1 << 16)
#define NFC_SR_CMD_DONE (1 << 17) #define NFC_SR_CMD_DONE (1 << 17)
#define NFC_SR_DTOE (1 << 20) #define NFC_SR_DTOE (1 << 20)
......
...@@ -225,7 +225,6 @@ static void nand_onfi_timing_set(struct denali_nand_info *denali, ...@@ -225,7 +225,6 @@ static void nand_onfi_timing_set(struct denali_nand_info *denali,
uint16_t Twhr[6] = {120, 80, 80, 60, 60, 60}; uint16_t Twhr[6] = {120, 80, 80, 60, 60, 60};
uint16_t Tcs[6] = {70, 35, 25, 25, 20, 15}; uint16_t Tcs[6] = {70, 35, 25, 25, 20, 15};
uint16_t TclsRising = 1;
uint16_t data_invalid_rhoh, data_invalid_rloh, data_invalid; uint16_t data_invalid_rhoh, data_invalid_rloh, data_invalid;
uint16_t dv_window = 0; uint16_t dv_window = 0;
uint16_t en_lo, en_hi; uint16_t en_lo, en_hi;
...@@ -276,8 +275,6 @@ static void nand_onfi_timing_set(struct denali_nand_info *denali, ...@@ -276,8 +275,6 @@ static void nand_onfi_timing_set(struct denali_nand_info *denali,
re_2_re = CEIL_DIV(Trhz[mode], CLK_X); re_2_re = CEIL_DIV(Trhz[mode], CLK_X);
we_2_re = CEIL_DIV(Twhr[mode], CLK_X); we_2_re = CEIL_DIV(Twhr[mode], CLK_X);
cs_cnt = CEIL_DIV((Tcs[mode] - Trp[mode]), CLK_X); cs_cnt = CEIL_DIV((Tcs[mode] - Trp[mode]), CLK_X);
if (!TclsRising)
cs_cnt = CEIL_DIV(Tcs[mode], CLK_X);
if (cs_cnt == 0) if (cs_cnt == 0)
cs_cnt = 1; cs_cnt = 1;
...@@ -1536,6 +1533,9 @@ int denali_init(struct denali_nand_info *denali) ...@@ -1536,6 +1533,9 @@ int denali_init(struct denali_nand_info *denali)
denali->nand.options |= NAND_SKIP_BBTSCAN; denali->nand.options |= NAND_SKIP_BBTSCAN;
denali->nand.ecc.mode = NAND_ECC_HW_SYNDROME; denali->nand.ecc.mode = NAND_ECC_HW_SYNDROME;
/* no subpage writes on denali */
denali->nand.options |= NAND_NO_SUBPAGE_WRITE;
/* /*
* Denali Controller only support 15bit and 8bit ECC in MRST, * Denali Controller only support 15bit and 8bit ECC in MRST,
* so just let controller do 15bit ECC for MLC and 8bit ECC for * so just let controller do 15bit ECC for MLC and 8bit ECC for
......
...@@ -317,7 +317,7 @@ static void fsl_ifc_run_command(struct mtd_info *mtd) ...@@ -317,7 +317,7 @@ static void fsl_ifc_run_command(struct mtd_info *mtd)
/* wait for command complete flag or timeout */ /* wait for command complete flag or timeout */
wait_event_timeout(ctrl->nand_wait, ctrl->nand_stat, wait_event_timeout(ctrl->nand_wait, ctrl->nand_stat,
IFC_TIMEOUT_MSECS * HZ/1000); msecs_to_jiffies(IFC_TIMEOUT_MSECS));
/* ctrl->nand_stat will be updated from IRQ context */ /* ctrl->nand_stat will be updated from IRQ context */
if (!ctrl->nand_stat) if (!ctrl->nand_stat)
...@@ -860,7 +860,7 @@ static void fsl_ifc_sram_init(struct fsl_ifc_mtd *priv) ...@@ -860,7 +860,7 @@ static void fsl_ifc_sram_init(struct fsl_ifc_mtd *priv)
/* wait for command complete flag or timeout */ /* wait for command complete flag or timeout */
wait_event_timeout(ctrl->nand_wait, ctrl->nand_stat, wait_event_timeout(ctrl->nand_wait, ctrl->nand_stat,
IFC_TIMEOUT_MSECS * HZ/1000); msecs_to_jiffies(IFC_TIMEOUT_MSECS));
if (ctrl->nand_stat != IFC_NAND_EVTER_STAT_OPC) if (ctrl->nand_stat != IFC_NAND_EVTER_STAT_OPC)
printk(KERN_ERR "fsl-ifc: Failed to Initialise SRAM\n"); printk(KERN_ERR "fsl-ifc: Failed to Initialise SRAM\n");
......
...@@ -873,6 +873,7 @@ static int fsmc_nand_probe_config_dt(struct platform_device *pdev, ...@@ -873,6 +873,7 @@ static int fsmc_nand_probe_config_dt(struct platform_device *pdev,
{ {
struct fsmc_nand_platform_data *pdata = dev_get_platdata(&pdev->dev); struct fsmc_nand_platform_data *pdata = dev_get_platdata(&pdev->dev);
u32 val; u32 val;
int ret;
/* Set default NAND width to 8 bits */ /* Set default NAND width to 8 bits */
pdata->width = 8; pdata->width = 8;
...@@ -891,8 +892,12 @@ static int fsmc_nand_probe_config_dt(struct platform_device *pdev, ...@@ -891,8 +892,12 @@ static int fsmc_nand_probe_config_dt(struct platform_device *pdev,
sizeof(*pdata->nand_timings), GFP_KERNEL); sizeof(*pdata->nand_timings), GFP_KERNEL);
if (!pdata->nand_timings) if (!pdata->nand_timings)
return -ENOMEM; return -ENOMEM;
of_property_read_u8_array(np, "timings", (u8 *)pdata->nand_timings, ret = of_property_read_u8_array(np, "timings", (u8 *)pdata->nand_timings,
sizeof(*pdata->nand_timings)); sizeof(*pdata->nand_timings));
if (ret) {
dev_info(&pdev->dev, "No timings in dts specified, using default timings!\n");
pdata->nand_timings = NULL;
}
/* Set default NAND bank to 0 */ /* Set default NAND bank to 0 */
pdata->bank = 0; pdata->bank = 0;
......
...@@ -446,7 +446,7 @@ int start_dma_without_bch_irq(struct gpmi_nand_data *this, ...@@ -446,7 +446,7 @@ int start_dma_without_bch_irq(struct gpmi_nand_data *this,
struct dma_async_tx_descriptor *desc) struct dma_async_tx_descriptor *desc)
{ {
struct completion *dma_c = &this->dma_done; struct completion *dma_c = &this->dma_done;
int err; unsigned long timeout;
init_completion(dma_c); init_completion(dma_c);
...@@ -456,8 +456,8 @@ int start_dma_without_bch_irq(struct gpmi_nand_data *this, ...@@ -456,8 +456,8 @@ int start_dma_without_bch_irq(struct gpmi_nand_data *this,
dma_async_issue_pending(get_dma_chan(this)); dma_async_issue_pending(get_dma_chan(this));
/* Wait for the interrupt from the DMA block. */ /* Wait for the interrupt from the DMA block. */
err = wait_for_completion_timeout(dma_c, msecs_to_jiffies(1000)); timeout = wait_for_completion_timeout(dma_c, msecs_to_jiffies(1000));
if (!err) { if (!timeout) {
dev_err(this->dev, "DMA timeout, last DMA :%d\n", dev_err(this->dev, "DMA timeout, last DMA :%d\n",
this->last_dma_type); this->last_dma_type);
gpmi_dump_info(this); gpmi_dump_info(this);
...@@ -477,7 +477,7 @@ int start_dma_with_bch_irq(struct gpmi_nand_data *this, ...@@ -477,7 +477,7 @@ int start_dma_with_bch_irq(struct gpmi_nand_data *this,
struct dma_async_tx_descriptor *desc) struct dma_async_tx_descriptor *desc)
{ {
struct completion *bch_c = &this->bch_done; struct completion *bch_c = &this->bch_done;
int err; unsigned long timeout;
/* Prepare to receive an interrupt from the BCH block. */ /* Prepare to receive an interrupt from the BCH block. */
init_completion(bch_c); init_completion(bch_c);
...@@ -486,8 +486,8 @@ int start_dma_with_bch_irq(struct gpmi_nand_data *this, ...@@ -486,8 +486,8 @@ int start_dma_with_bch_irq(struct gpmi_nand_data *this,
start_dma_without_bch_irq(this, desc); start_dma_without_bch_irq(this, desc);
/* Wait for the interrupt from the BCH block. */ /* Wait for the interrupt from the BCH block. */
err = wait_for_completion_timeout(bch_c, msecs_to_jiffies(1000)); timeout = wait_for_completion_timeout(bch_c, msecs_to_jiffies(1000));
if (!err) { if (!timeout) {
dev_err(this->dev, "BCH timeout, last DMA :%d\n", dev_err(this->dev, "BCH timeout, last DMA :%d\n",
this->last_dma_type); this->last_dma_type);
gpmi_dump_info(this); gpmi_dump_info(this);
...@@ -1950,7 +1950,9 @@ static int gpmi_nand_init(struct gpmi_nand_data *this) ...@@ -1950,7 +1950,9 @@ static int gpmi_nand_init(struct gpmi_nand_data *this)
ret = nand_boot_init(this); ret = nand_boot_init(this);
if (ret) if (ret)
goto err_out; goto err_out;
chip->scan_bbt(mtd); ret = chip->scan_bbt(mtd);
if (ret)
goto err_out;
ppdata.of_node = this->pdev->dev.of_node; ppdata.of_node = this->pdev->dev.of_node;
ret = mtd_device_parse_register(mtd, NULL, &ppdata, NULL, 0); ret = mtd_device_parse_register(mtd, NULL, &ppdata, NULL, 0);
......
...@@ -386,26 +386,51 @@ static irqreturn_t mxc_nfc_irq(int irq, void *dev_id) ...@@ -386,26 +386,51 @@ static irqreturn_t mxc_nfc_irq(int irq, void *dev_id)
/* This function polls the NANDFC to wait for the basic operation to /* This function polls the NANDFC to wait for the basic operation to
* complete by checking the INT bit of config2 register. * complete by checking the INT bit of config2 register.
*/ */
static void wait_op_done(struct mxc_nand_host *host, int useirq) static int wait_op_done(struct mxc_nand_host *host, int useirq)
{ {
int max_retries = 8000; int ret = 0;
/*
* If operation is already complete, don't bother to setup an irq or a
* loop.
*/
if (host->devtype_data->check_int(host))
return 0;
if (useirq) { if (useirq) {
if (!host->devtype_data->check_int(host)) { unsigned long timeout;
reinit_completion(&host->op_completion); reinit_completion(&host->op_completion);
irq_control(host, 1); irq_control(host, 1);
wait_for_completion(&host->op_completion);
timeout = wait_for_completion_timeout(&host->op_completion, HZ);
if (!timeout && !host->devtype_data->check_int(host)) {
dev_dbg(host->dev, "timeout waiting for irq\n");
ret = -ETIMEDOUT;
} }
} else { } else {
while (max_retries-- > 0) { int max_retries = 8000;
if (host->devtype_data->check_int(host)) int done;
break;
do {
udelay(1); udelay(1);
done = host->devtype_data->check_int(host);
if (done)
break;
} while (--max_retries);
if (!done) {
dev_dbg(host->dev, "timeout polling for completion\n");
ret = -ETIMEDOUT;
} }
if (max_retries < 0)
pr_debug("%s: INT not set\n", __func__);
} }
WARN_ONCE(ret < 0, "timeout! useirq=%d\n", useirq);
return ret;
} }
static void send_cmd_v3(struct mxc_nand_host *host, uint16_t cmd, int useirq) static void send_cmd_v3(struct mxc_nand_host *host, uint16_t cmd, int useirq)
...@@ -527,30 +552,17 @@ static void send_page_v1(struct mtd_info *mtd, unsigned int ops) ...@@ -527,30 +552,17 @@ static void send_page_v1(struct mtd_info *mtd, unsigned int ops)
static void send_read_id_v3(struct mxc_nand_host *host) static void send_read_id_v3(struct mxc_nand_host *host)
{ {
struct nand_chip *this = &host->nand;
/* Read ID into main buffer */ /* Read ID into main buffer */
writel(NFC_ID, NFC_V3_LAUNCH); writel(NFC_ID, NFC_V3_LAUNCH);
wait_op_done(host, true); wait_op_done(host, true);
memcpy32_fromio(host->data_buf, host->main_area0, 16); memcpy32_fromio(host->data_buf, host->main_area0, 16);
if (this->options & NAND_BUSWIDTH_16) {
/* compress the ID info */
host->data_buf[1] = host->data_buf[2];
host->data_buf[2] = host->data_buf[4];
host->data_buf[3] = host->data_buf[6];
host->data_buf[4] = host->data_buf[8];
host->data_buf[5] = host->data_buf[10];
}
} }
/* Request the NANDFC to perform a read of the NAND device ID. */ /* Request the NANDFC to perform a read of the NAND device ID. */
static void send_read_id_v1_v2(struct mxc_nand_host *host) static void send_read_id_v1_v2(struct mxc_nand_host *host)
{ {
struct nand_chip *this = &host->nand;
/* NANDFC buffer 0 is used for device ID output */ /* NANDFC buffer 0 is used for device ID output */
writew(host->active_cs << 4, NFC_V1_V2_BUF_ADDR); writew(host->active_cs << 4, NFC_V1_V2_BUF_ADDR);
...@@ -560,15 +572,6 @@ static void send_read_id_v1_v2(struct mxc_nand_host *host) ...@@ -560,15 +572,6 @@ static void send_read_id_v1_v2(struct mxc_nand_host *host)
wait_op_done(host, true); wait_op_done(host, true);
memcpy32_fromio(host->data_buf, host->main_area0, 16); memcpy32_fromio(host->data_buf, host->main_area0, 16);
if (this->options & NAND_BUSWIDTH_16) {
/* compress the ID info */
host->data_buf[1] = host->data_buf[2];
host->data_buf[2] = host->data_buf[4];
host->data_buf[3] = host->data_buf[6];
host->data_buf[4] = host->data_buf[8];
host->data_buf[5] = host->data_buf[10];
}
} }
static uint16_t get_dev_status_v3(struct mxc_nand_host *host) static uint16_t get_dev_status_v3(struct mxc_nand_host *host)
...@@ -694,9 +697,17 @@ static u_char mxc_nand_read_byte(struct mtd_info *mtd) ...@@ -694,9 +697,17 @@ static u_char mxc_nand_read_byte(struct mtd_info *mtd)
if (host->status_request) if (host->status_request)
return host->devtype_data->get_dev_status(host) & 0xFF; return host->devtype_data->get_dev_status(host) & 0xFF;
if (nand_chip->options & NAND_BUSWIDTH_16) {
/* only take the lower byte of each word */
ret = *(uint16_t *)(host->data_buf + host->buf_start);
host->buf_start += 2;
} else {
ret = *(uint8_t *)(host->data_buf + host->buf_start); ret = *(uint8_t *)(host->data_buf + host->buf_start);
host->buf_start++; host->buf_start++;
}
pr_debug("%s: ret=0x%hhx (start=%u)\n", __func__, ret, host->buf_start);
return ret; return ret;
} }
...@@ -825,6 +836,12 @@ static void copy_spare(struct mtd_info *mtd, bool bfrom) ...@@ -825,6 +836,12 @@ static void copy_spare(struct mtd_info *mtd, bool bfrom)
} }
} }
/*
* MXC NANDFC can only perform full page+spare or spare-only read/write. When
* the upper layers perform a read/write buf operation, the saved column address
* is used to index into the full page. So usually this function is called with
* column == 0 (unless no column cycle is needed indicated by column == -1)
*/
static void mxc_do_addr_cycle(struct mtd_info *mtd, int column, int page_addr) static void mxc_do_addr_cycle(struct mtd_info *mtd, int column, int page_addr)
{ {
struct nand_chip *nand_chip = mtd->priv; struct nand_chip *nand_chip = mtd->priv;
...@@ -832,16 +849,13 @@ static void mxc_do_addr_cycle(struct mtd_info *mtd, int column, int page_addr) ...@@ -832,16 +849,13 @@ static void mxc_do_addr_cycle(struct mtd_info *mtd, int column, int page_addr)
/* Write out column address, if necessary */ /* Write out column address, if necessary */
if (column != -1) { if (column != -1) {
/* host->devtype_data->send_addr(host, column & 0xff,
* MXC NANDFC can only perform full page+spare or page_addr == -1);
* spare-only read/write. When the upper layers
* perform a read/write buf operation, the saved column
* address is used to index into the full page.
*/
host->devtype_data->send_addr(host, 0, page_addr == -1);
if (mtd->writesize > 512) if (mtd->writesize > 512)
/* another col addr cycle for 2k page */ /* another col addr cycle for 2k page */
host->devtype_data->send_addr(host, 0, false); host->devtype_data->send_addr(host,
(column >> 8) & 0xff,
false);
} }
/* Write out page address, if necessary */ /* Write out page address, if necessary */
...@@ -903,7 +917,7 @@ static void preset_v1(struct mtd_info *mtd) ...@@ -903,7 +917,7 @@ static void preset_v1(struct mtd_info *mtd)
struct mxc_nand_host *host = nand_chip->priv; struct mxc_nand_host *host = nand_chip->priv;
uint16_t config1 = 0; uint16_t config1 = 0;
if (nand_chip->ecc.mode == NAND_ECC_HW) if (nand_chip->ecc.mode == NAND_ECC_HW && mtd->writesize)
config1 |= NFC_V1_V2_CONFIG1_ECC_EN; config1 |= NFC_V1_V2_CONFIG1_ECC_EN;
if (!host->devtype_data->irqpending_quirk) if (!host->devtype_data->irqpending_quirk)
...@@ -931,9 +945,6 @@ static void preset_v2(struct mtd_info *mtd) ...@@ -931,9 +945,6 @@ static void preset_v2(struct mtd_info *mtd)
struct mxc_nand_host *host = nand_chip->priv; struct mxc_nand_host *host = nand_chip->priv;
uint16_t config1 = 0; uint16_t config1 = 0;
if (nand_chip->ecc.mode == NAND_ECC_HW)
config1 |= NFC_V1_V2_CONFIG1_ECC_EN;
config1 |= NFC_V2_CONFIG1_FP_INT; config1 |= NFC_V2_CONFIG1_FP_INT;
if (!host->devtype_data->irqpending_quirk) if (!host->devtype_data->irqpending_quirk)
...@@ -942,6 +953,9 @@ static void preset_v2(struct mtd_info *mtd) ...@@ -942,6 +953,9 @@ static void preset_v2(struct mtd_info *mtd)
if (mtd->writesize) { if (mtd->writesize) {
uint16_t pages_per_block = mtd->erasesize / mtd->writesize; uint16_t pages_per_block = mtd->erasesize / mtd->writesize;
if (nand_chip->ecc.mode == NAND_ECC_HW)
config1 |= NFC_V1_V2_CONFIG1_ECC_EN;
host->eccsize = get_eccsize(mtd); host->eccsize = get_eccsize(mtd);
if (host->eccsize == 4) if (host->eccsize == 4)
config1 |= NFC_V2_CONFIG1_ECC_MODE_4; config1 |= NFC_V2_CONFIG1_ECC_MODE_4;
...@@ -999,9 +1013,6 @@ static void preset_v3(struct mtd_info *mtd) ...@@ -999,9 +1013,6 @@ static void preset_v3(struct mtd_info *mtd)
NFC_V3_CONFIG2_INT_MSK | NFC_V3_CONFIG2_INT_MSK |
NFC_V3_CONFIG2_NUM_ADDR_PHASE0; NFC_V3_CONFIG2_NUM_ADDR_PHASE0;
if (chip->ecc.mode == NAND_ECC_HW)
config2 |= NFC_V3_CONFIG2_ECC_EN;
addr_phases = fls(chip->pagemask) >> 3; addr_phases = fls(chip->pagemask) >> 3;
if (mtd->writesize == 2048) { if (mtd->writesize == 2048) {
...@@ -1016,6 +1027,9 @@ static void preset_v3(struct mtd_info *mtd) ...@@ -1016,6 +1027,9 @@ static void preset_v3(struct mtd_info *mtd)
} }
if (mtd->writesize) { if (mtd->writesize) {
if (chip->ecc.mode == NAND_ECC_HW)
config2 |= NFC_V3_CONFIG2_ECC_EN;
config2 |= NFC_V3_CONFIG2_PPB( config2 |= NFC_V3_CONFIG2_PPB(
ffs(mtd->erasesize / mtd->writesize) - 6, ffs(mtd->erasesize / mtd->writesize) - 6,
host->devtype_data->ppb_shift); host->devtype_data->ppb_shift);
...@@ -1066,6 +1080,9 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command, ...@@ -1066,6 +1080,9 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command,
host->status_request = true; host->status_request = true;
host->devtype_data->send_cmd(host, command, true); host->devtype_data->send_cmd(host, command, true);
WARN_ONCE(column != -1 || page_addr != -1,
"Unexpected column/row value (cmd=%u, col=%d, row=%d)\n",
command, column, page_addr);
mxc_do_addr_cycle(mtd, column, page_addr); mxc_do_addr_cycle(mtd, column, page_addr);
break; break;
...@@ -1079,7 +1096,10 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command, ...@@ -1079,7 +1096,10 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command,
command = NAND_CMD_READ0; /* only READ0 is valid */ command = NAND_CMD_READ0; /* only READ0 is valid */
host->devtype_data->send_cmd(host, command, false); host->devtype_data->send_cmd(host, command, false);
mxc_do_addr_cycle(mtd, column, page_addr); WARN_ONCE(column < 0,
"Unexpected column/row value (cmd=%u, col=%d, row=%d)\n",
command, column, page_addr);
mxc_do_addr_cycle(mtd, 0, page_addr);
if (mtd->writesize > 512) if (mtd->writesize > 512)
host->devtype_data->send_cmd(host, host->devtype_data->send_cmd(host,
...@@ -1100,7 +1120,10 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command, ...@@ -1100,7 +1120,10 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command,
host->buf_start = column; host->buf_start = column;
host->devtype_data->send_cmd(host, command, false); host->devtype_data->send_cmd(host, command, false);
mxc_do_addr_cycle(mtd, column, page_addr); WARN_ONCE(column < -1,
"Unexpected column/row value (cmd=%u, col=%d, row=%d)\n",
command, column, page_addr);
mxc_do_addr_cycle(mtd, 0, page_addr);
break; break;
case NAND_CMD_PAGEPROG: case NAND_CMD_PAGEPROG:
...@@ -1108,6 +1131,9 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command, ...@@ -1108,6 +1131,9 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command,
copy_spare(mtd, false); copy_spare(mtd, false);
host->devtype_data->send_page(mtd, NFC_INPUT); host->devtype_data->send_page(mtd, NFC_INPUT);
host->devtype_data->send_cmd(host, command, true); host->devtype_data->send_cmd(host, command, true);
WARN_ONCE(column != -1 || page_addr != -1,
"Unexpected column/row value (cmd=%u, col=%d, row=%d)\n",
command, column, page_addr);
mxc_do_addr_cycle(mtd, column, page_addr); mxc_do_addr_cycle(mtd, column, page_addr);
break; break;
...@@ -1115,15 +1141,29 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command, ...@@ -1115,15 +1141,29 @@ static void mxc_nand_command(struct mtd_info *mtd, unsigned command,
host->devtype_data->send_cmd(host, command, true); host->devtype_data->send_cmd(host, command, true);
mxc_do_addr_cycle(mtd, column, page_addr); mxc_do_addr_cycle(mtd, column, page_addr);
host->devtype_data->send_read_id(host); host->devtype_data->send_read_id(host);
host->buf_start = column; host->buf_start = 0;
break; break;
case NAND_CMD_ERASE1: case NAND_CMD_ERASE1:
case NAND_CMD_ERASE2: case NAND_CMD_ERASE2:
host->devtype_data->send_cmd(host, command, false); host->devtype_data->send_cmd(host, command, false);
WARN_ONCE(column != -1,
"Unexpected column value (cmd=%u, col=%d)\n",
command, column);
mxc_do_addr_cycle(mtd, column, page_addr); mxc_do_addr_cycle(mtd, column, page_addr);
break; break;
case NAND_CMD_PARAM:
host->devtype_data->send_cmd(host, command, false);
mxc_do_addr_cycle(mtd, column, page_addr);
host->devtype_data->send_page(mtd, NFC_OUTPUT);
memcpy32_fromio(host->data_buf, host->main_area0, 512);
host->buf_start = 0;
break;
default:
WARN_ONCE(1, "Unimplemented command (cmd=%u)\n",
command);
break;
} }
} }
......
...@@ -386,7 +386,7 @@ static int nand_default_block_markbad(struct mtd_info *mtd, loff_t ofs) ...@@ -386,7 +386,7 @@ static int nand_default_block_markbad(struct mtd_info *mtd, loff_t ofs)
uint8_t buf[2] = { 0, 0 }; uint8_t buf[2] = { 0, 0 };
int ret = 0, res, i = 0; int ret = 0, res, i = 0;
ops.datbuf = NULL; memset(&ops, 0, sizeof(ops));
ops.oobbuf = buf; ops.oobbuf = buf;
ops.ooboffs = chip->badblockpos; ops.ooboffs = chip->badblockpos;
if (chip->options & NAND_BUSWIDTH_16) { if (chip->options & NAND_BUSWIDTH_16) {
...@@ -565,6 +565,25 @@ void nand_wait_ready(struct mtd_info *mtd) ...@@ -565,6 +565,25 @@ void nand_wait_ready(struct mtd_info *mtd)
} }
EXPORT_SYMBOL_GPL(nand_wait_ready); EXPORT_SYMBOL_GPL(nand_wait_ready);
/**
* nand_wait_status_ready - [GENERIC] Wait for the ready status after commands.
* @mtd: MTD device structure
* @timeo: Timeout in ms
*
* Wait for status ready (i.e. command done) or timeout.
*/
static void nand_wait_status_ready(struct mtd_info *mtd, unsigned long timeo)
{
register struct nand_chip *chip = mtd->priv;
timeo = jiffies + msecs_to_jiffies(timeo);
do {
if ((chip->read_byte(mtd) & NAND_STATUS_READY))
break;
touch_softlockup_watchdog();
} while (time_before(jiffies, timeo));
};
/** /**
* nand_command - [DEFAULT] Send command to NAND device * nand_command - [DEFAULT] Send command to NAND device
* @mtd: MTD device structure * @mtd: MTD device structure
...@@ -643,8 +662,8 @@ static void nand_command(struct mtd_info *mtd, unsigned int command, ...@@ -643,8 +662,8 @@ static void nand_command(struct mtd_info *mtd, unsigned int command,
NAND_CTRL_CLE | NAND_CTRL_CHANGE); NAND_CTRL_CLE | NAND_CTRL_CHANGE);
chip->cmd_ctrl(mtd, chip->cmd_ctrl(mtd,
NAND_CMD_NONE, NAND_NCE | NAND_CTRL_CHANGE); NAND_CMD_NONE, NAND_NCE | NAND_CTRL_CHANGE);
while (!(chip->read_byte(mtd) & NAND_STATUS_READY)) /* EZ-NAND can take upto 250ms as per ONFi v4.0 */
; nand_wait_status_ready(mtd, 250);
return; return;
/* This applies to read commands */ /* This applies to read commands */
...@@ -740,8 +759,8 @@ static void nand_command_lp(struct mtd_info *mtd, unsigned int command, ...@@ -740,8 +759,8 @@ static void nand_command_lp(struct mtd_info *mtd, unsigned int command,
NAND_NCE | NAND_CLE | NAND_CTRL_CHANGE); NAND_NCE | NAND_CLE | NAND_CTRL_CHANGE);
chip->cmd_ctrl(mtd, NAND_CMD_NONE, chip->cmd_ctrl(mtd, NAND_CMD_NONE,
NAND_NCE | NAND_CTRL_CHANGE); NAND_NCE | NAND_CTRL_CHANGE);
while (!(chip->read_byte(mtd) & NAND_STATUS_READY)) /* EZ-NAND can take upto 250ms as per ONFi v4.0 */
; nand_wait_status_ready(mtd, 250);
return; return;
case NAND_CMD_RNDOUT: case NAND_CMD_RNDOUT:
...@@ -968,7 +987,7 @@ int nand_unlock(struct mtd_info *mtd, loff_t ofs, uint64_t len) ...@@ -968,7 +987,7 @@ int nand_unlock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
__func__, (unsigned long long)ofs, len); __func__, (unsigned long long)ofs, len);
if (check_offs_len(mtd, ofs, len)) if (check_offs_len(mtd, ofs, len))
ret = -EINVAL; return -EINVAL;
/* Align to last block address if size addresses end of the device */ /* Align to last block address if size addresses end of the device */
if (ofs + len == mtd->size) if (ofs + len == mtd->size)
...@@ -1031,7 +1050,7 @@ int nand_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len) ...@@ -1031,7 +1050,7 @@ int nand_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
__func__, (unsigned long long)ofs, len); __func__, (unsigned long long)ofs, len);
if (check_offs_len(mtd, ofs, len)) if (check_offs_len(mtd, ofs, len))
ret = -EINVAL; return -EINVAL;
nand_get_device(mtd, FL_LOCKING); nand_get_device(mtd, FL_LOCKING);
...@@ -1716,9 +1735,9 @@ static int nand_read(struct mtd_info *mtd, loff_t from, size_t len, ...@@ -1716,9 +1735,9 @@ static int nand_read(struct mtd_info *mtd, loff_t from, size_t len,
int ret; int ret;
nand_get_device(mtd, FL_READING); nand_get_device(mtd, FL_READING);
memset(&ops, 0, sizeof(ops));
ops.len = len; ops.len = len;
ops.datbuf = buf; ops.datbuf = buf;
ops.oobbuf = NULL;
ops.mode = MTD_OPS_PLACE_OOB; ops.mode = MTD_OPS_PLACE_OOB;
ret = nand_do_read_ops(mtd, from, &ops); ret = nand_do_read_ops(mtd, from, &ops);
*retlen = ops.retlen; *retlen = ops.retlen;
...@@ -2124,7 +2143,7 @@ static int nand_write_page_hwecc(struct mtd_info *mtd, struct nand_chip *chip, ...@@ -2124,7 +2143,7 @@ static int nand_write_page_hwecc(struct mtd_info *mtd, struct nand_chip *chip,
/** /**
* nand_write_subpage_hwecc - [REPLACABLE] hardware ECC based subpage write * nand_write_subpage_hwecc - [REPLACEABLE] hardware ECC based subpage write
* @mtd: mtd info structure * @mtd: mtd info structure
* @chip: nand chip info structure * @chip: nand chip info structure
* @offset: column address of subpage within the page * @offset: column address of subpage within the page
...@@ -2508,9 +2527,9 @@ static int panic_nand_write(struct mtd_info *mtd, loff_t to, size_t len, ...@@ -2508,9 +2527,9 @@ static int panic_nand_write(struct mtd_info *mtd, loff_t to, size_t len,
/* Grab the device */ /* Grab the device */
panic_nand_get_device(chip, mtd, FL_WRITING); panic_nand_get_device(chip, mtd, FL_WRITING);
memset(&ops, 0, sizeof(ops));
ops.len = len; ops.len = len;
ops.datbuf = (uint8_t *)buf; ops.datbuf = (uint8_t *)buf;
ops.oobbuf = NULL;
ops.mode = MTD_OPS_PLACE_OOB; ops.mode = MTD_OPS_PLACE_OOB;
ret = nand_do_write_ops(mtd, to, &ops); ret = nand_do_write_ops(mtd, to, &ops);
...@@ -2536,9 +2555,9 @@ static int nand_write(struct mtd_info *mtd, loff_t to, size_t len, ...@@ -2536,9 +2555,9 @@ static int nand_write(struct mtd_info *mtd, loff_t to, size_t len,
int ret; int ret;
nand_get_device(mtd, FL_WRITING); nand_get_device(mtd, FL_WRITING);
memset(&ops, 0, sizeof(ops));
ops.len = len; ops.len = len;
ops.datbuf = (uint8_t *)buf; ops.datbuf = (uint8_t *)buf;
ops.oobbuf = NULL;
ops.mode = MTD_OPS_PLACE_OOB; ops.mode = MTD_OPS_PLACE_OOB;
ret = nand_do_write_ops(mtd, to, &ops); ret = nand_do_write_ops(mtd, to, &ops);
*retlen = ops.retlen; *retlen = ops.retlen;
......
...@@ -38,8 +38,8 @@ ...@@ -38,8 +38,8 @@
#include <linux/platform_data/mtd-nand-pxa3xx.h> #include <linux/platform_data/mtd-nand-pxa3xx.h>
#define CHIP_DELAY_TIMEOUT (2 * HZ/10) #define CHIP_DELAY_TIMEOUT msecs_to_jiffies(200)
#define NAND_STOP_DELAY (2 * HZ/50) #define NAND_STOP_DELAY msecs_to_jiffies(40)
#define PAGE_CHUNK_SIZE (2048) #define PAGE_CHUNK_SIZE (2048)
/* /*
...@@ -605,11 +605,24 @@ static void start_data_dma(struct pxa3xx_nand_info *info) ...@@ -605,11 +605,24 @@ static void start_data_dma(struct pxa3xx_nand_info *info)
{} {}
#endif #endif
static irqreturn_t pxa3xx_nand_irq_thread(int irq, void *data)
{
struct pxa3xx_nand_info *info = data;
handle_data_pio(info);
info->state = STATE_CMD_DONE;
nand_writel(info, NDSR, NDSR_WRDREQ | NDSR_RDDREQ);
return IRQ_HANDLED;
}
static irqreturn_t pxa3xx_nand_irq(int irq, void *devid) static irqreturn_t pxa3xx_nand_irq(int irq, void *devid)
{ {
struct pxa3xx_nand_info *info = devid; struct pxa3xx_nand_info *info = devid;
unsigned int status, is_completed = 0, is_ready = 0; unsigned int status, is_completed = 0, is_ready = 0;
unsigned int ready, cmd_done; unsigned int ready, cmd_done;
irqreturn_t ret = IRQ_HANDLED;
if (info->cs == 0) { if (info->cs == 0) {
ready = NDSR_FLASH_RDY; ready = NDSR_FLASH_RDY;
...@@ -651,7 +664,8 @@ static irqreturn_t pxa3xx_nand_irq(int irq, void *devid) ...@@ -651,7 +664,8 @@ static irqreturn_t pxa3xx_nand_irq(int irq, void *devid)
} else { } else {
info->state = (status & NDSR_RDDREQ) ? info->state = (status & NDSR_RDDREQ) ?
STATE_PIO_READING : STATE_PIO_WRITING; STATE_PIO_READING : STATE_PIO_WRITING;
handle_data_pio(info); ret = IRQ_WAKE_THREAD;
goto NORMAL_IRQ_EXIT;
} }
} }
if (status & cmd_done) { if (status & cmd_done) {
...@@ -692,7 +706,7 @@ static irqreturn_t pxa3xx_nand_irq(int irq, void *devid) ...@@ -692,7 +706,7 @@ static irqreturn_t pxa3xx_nand_irq(int irq, void *devid)
if (is_ready) if (is_ready)
complete(&info->dev_ready); complete(&info->dev_ready);
NORMAL_IRQ_EXIT: NORMAL_IRQ_EXIT:
return IRQ_HANDLED; return ret;
} }
static inline int is_buf_blank(uint8_t *buf, size_t len) static inline int is_buf_blank(uint8_t *buf, size_t len)
...@@ -951,7 +965,7 @@ static void nand_cmdfunc(struct mtd_info *mtd, unsigned command, ...@@ -951,7 +965,7 @@ static void nand_cmdfunc(struct mtd_info *mtd, unsigned command,
{ {
struct pxa3xx_nand_host *host = mtd->priv; struct pxa3xx_nand_host *host = mtd->priv;
struct pxa3xx_nand_info *info = host->info_data; struct pxa3xx_nand_info *info = host->info_data;
int ret, exec_cmd; int exec_cmd;
/* /*
* if this is a x16 device ,then convert the input * if this is a x16 device ,then convert the input
...@@ -983,9 +997,8 @@ static void nand_cmdfunc(struct mtd_info *mtd, unsigned command, ...@@ -983,9 +997,8 @@ static void nand_cmdfunc(struct mtd_info *mtd, unsigned command,
info->need_wait = 1; info->need_wait = 1;
pxa3xx_nand_start(info); pxa3xx_nand_start(info);
ret = wait_for_completion_timeout(&info->cmd_complete, if (!wait_for_completion_timeout(&info->cmd_complete,
CHIP_DELAY_TIMEOUT); CHIP_DELAY_TIMEOUT)) {
if (!ret) {
dev_err(&info->pdev->dev, "Wait time out!!!\n"); dev_err(&info->pdev->dev, "Wait time out!!!\n");
/* Stop State Machine for next command cycle */ /* Stop State Machine for next command cycle */
pxa3xx_nand_stop(info); pxa3xx_nand_stop(info);
...@@ -1000,7 +1013,7 @@ static void nand_cmdfunc_extended(struct mtd_info *mtd, ...@@ -1000,7 +1013,7 @@ static void nand_cmdfunc_extended(struct mtd_info *mtd,
{ {
struct pxa3xx_nand_host *host = mtd->priv; struct pxa3xx_nand_host *host = mtd->priv;
struct pxa3xx_nand_info *info = host->info_data; struct pxa3xx_nand_info *info = host->info_data;
int ret, exec_cmd, ext_cmd_type; int exec_cmd, ext_cmd_type;
/* /*
* if this is a x16 device then convert the input * if this is a x16 device then convert the input
...@@ -1063,9 +1076,8 @@ static void nand_cmdfunc_extended(struct mtd_info *mtd, ...@@ -1063,9 +1076,8 @@ static void nand_cmdfunc_extended(struct mtd_info *mtd,
init_completion(&info->cmd_complete); init_completion(&info->cmd_complete);
pxa3xx_nand_start(info); pxa3xx_nand_start(info);
ret = wait_for_completion_timeout(&info->cmd_complete, if (!wait_for_completion_timeout(&info->cmd_complete,
CHIP_DELAY_TIMEOUT); CHIP_DELAY_TIMEOUT)) {
if (!ret) {
dev_err(&info->pdev->dev, "Wait time out!!!\n"); dev_err(&info->pdev->dev, "Wait time out!!!\n");
/* Stop State Machine for next command cycle */ /* Stop State Machine for next command cycle */
pxa3xx_nand_stop(info); pxa3xx_nand_stop(info);
...@@ -1198,13 +1210,11 @@ static int pxa3xx_nand_waitfunc(struct mtd_info *mtd, struct nand_chip *this) ...@@ -1198,13 +1210,11 @@ static int pxa3xx_nand_waitfunc(struct mtd_info *mtd, struct nand_chip *this)
{ {
struct pxa3xx_nand_host *host = mtd->priv; struct pxa3xx_nand_host *host = mtd->priv;
struct pxa3xx_nand_info *info = host->info_data; struct pxa3xx_nand_info *info = host->info_data;
int ret;
if (info->need_wait) { if (info->need_wait) {
ret = wait_for_completion_timeout(&info->dev_ready,
CHIP_DELAY_TIMEOUT);
info->need_wait = 0; info->need_wait = 0;
if (!ret) { if (!wait_for_completion_timeout(&info->dev_ready,
CHIP_DELAY_TIMEOUT)) {
dev_err(&info->pdev->dev, "Ready time out!!!\n"); dev_err(&info->pdev->dev, "Ready time out!!!\n");
return NAND_STATUS_FAIL; return NAND_STATUS_FAIL;
} }
...@@ -1508,6 +1518,8 @@ static int pxa3xx_nand_scan(struct mtd_info *mtd) ...@@ -1508,6 +1518,8 @@ static int pxa3xx_nand_scan(struct mtd_info *mtd)
return ret; return ret;
} }
memset(pxa3xx_flash_ids, 0, sizeof(pxa3xx_flash_ids));
pxa3xx_flash_ids[0].name = f->name; pxa3xx_flash_ids[0].name = f->name;
pxa3xx_flash_ids[0].dev_id = (f->chip_id >> 8) & 0xffff; pxa3xx_flash_ids[0].dev_id = (f->chip_id >> 8) & 0xffff;
pxa3xx_flash_ids[0].pagesize = f->page_size; pxa3xx_flash_ids[0].pagesize = f->page_size;
...@@ -1710,7 +1722,9 @@ static int alloc_nand_resource(struct platform_device *pdev) ...@@ -1710,7 +1722,9 @@ static int alloc_nand_resource(struct platform_device *pdev)
/* initialize all interrupts to be disabled */ /* initialize all interrupts to be disabled */
disable_int(info, NDSR_MASK); disable_int(info, NDSR_MASK);
ret = request_irq(irq, pxa3xx_nand_irq, 0, pdev->name, info); ret = request_threaded_irq(irq, pxa3xx_nand_irq,
pxa3xx_nand_irq_thread, IRQF_ONESHOT,
pdev->name, info);
if (ret < 0) { if (ret < 0) {
dev_err(&pdev->dev, "failed to request IRQ\n"); dev_err(&pdev->dev, "failed to request IRQ\n");
goto fail_free_buf; goto fail_free_buf;
......
...@@ -948,8 +948,6 @@ static int s3c24xx_nand_probe(struct platform_device *pdev) ...@@ -948,8 +948,6 @@ static int s3c24xx_nand_probe(struct platform_device *pdev)
cpu_type = platform_get_device_id(pdev)->driver_data; cpu_type = platform_get_device_id(pdev)->driver_data;
pr_debug("s3c2410_nand_probe(%p)\n", pdev);
info = devm_kzalloc(&pdev->dev, sizeof(*info), GFP_KERNEL); info = devm_kzalloc(&pdev->dev, sizeof(*info), GFP_KERNEL);
if (info == NULL) { if (info == NULL) {
err = -ENOMEM; err = -ENOMEM;
...@@ -1045,7 +1043,6 @@ static int s3c24xx_nand_probe(struct platform_device *pdev) ...@@ -1045,7 +1043,6 @@ static int s3c24xx_nand_probe(struct platform_device *pdev)
s3c2410_nand_clk_set_state(info, CLOCK_SUSPEND); s3c2410_nand_clk_set_state(info, CLOCK_SUSPEND);
} }
pr_debug("initialised ok\n");
return 0; return 0;
exit_error: exit_error:
......
...@@ -1743,7 +1743,6 @@ static int onenand_panic_write(struct mtd_info *mtd, loff_t to, size_t len, ...@@ -1743,7 +1743,6 @@ static int onenand_panic_write(struct mtd_info *mtd, loff_t to, size_t len,
struct onenand_chip *this = mtd->priv; struct onenand_chip *this = mtd->priv;
int column, subpage; int column, subpage;
int written = 0; int written = 0;
int ret = 0;
if (this->state == FL_PM_SUSPENDED) if (this->state == FL_PM_SUSPENDED)
return -EBUSY; return -EBUSY;
...@@ -1786,15 +1785,10 @@ static int onenand_panic_write(struct mtd_info *mtd, loff_t to, size_t len, ...@@ -1786,15 +1785,10 @@ static int onenand_panic_write(struct mtd_info *mtd, loff_t to, size_t len,
onenand_panic_wait(mtd); onenand_panic_wait(mtd);
/* In partial page write we don't update bufferram */ /* In partial page write we don't update bufferram */
onenand_update_bufferram(mtd, to, !ret && !subpage); onenand_update_bufferram(mtd, to, !subpage);
if (ONENAND_IS_2PLANE(this)) { if (ONENAND_IS_2PLANE(this)) {
ONENAND_SET_BUFFERRAM1(this); ONENAND_SET_BUFFERRAM1(this);
onenand_update_bufferram(mtd, to + this->writesize, !ret && !subpage); onenand_update_bufferram(mtd, to + this->writesize, !subpage);
}
if (ret) {
printk(KERN_ERR "%s: write failed %d\n", __func__, ret);
break;
} }
written += thislen; written += thislen;
...@@ -1808,7 +1802,7 @@ static int onenand_panic_write(struct mtd_info *mtd, loff_t to, size_t len, ...@@ -1808,7 +1802,7 @@ static int onenand_panic_write(struct mtd_info *mtd, loff_t to, size_t len,
} }
*retlen = written; *retlen = written;
return ret; return 0;
} }
/** /**
......
...@@ -460,8 +460,7 @@ fsl_qspi_runcmd(struct fsl_qspi *q, u8 cmd, unsigned int addr, int len) ...@@ -460,8 +460,7 @@ fsl_qspi_runcmd(struct fsl_qspi *q, u8 cmd, unsigned int addr, int len)
writel((seqid << QUADSPI_IPCR_SEQID_SHIFT) | len, base + QUADSPI_IPCR); writel((seqid << QUADSPI_IPCR_SEQID_SHIFT) | len, base + QUADSPI_IPCR);
/* Wait for the interrupt. */ /* Wait for the interrupt. */
err = wait_for_completion_timeout(&q->c, msecs_to_jiffies(1000)); if (!wait_for_completion_timeout(&q->c, msecs_to_jiffies(1000))) {
if (!err) {
dev_err(q->dev, dev_err(q->dev,
"cmd 0x%.2x timeout, addr@%.8x, FR:0x%.8x, SR:0x%.8x\n", "cmd 0x%.2x timeout, addr@%.8x, FR:0x%.8x, SR:0x%.8x\n",
cmd, addr, readl(base + QUADSPI_FR), cmd, addr, readl(base + QUADSPI_FR),
...@@ -830,27 +829,27 @@ static int fsl_qspi_probe(struct platform_device *pdev) ...@@ -830,27 +829,27 @@ static int fsl_qspi_probe(struct platform_device *pdev)
ret = clk_prepare_enable(q->clk_en); ret = clk_prepare_enable(q->clk_en);
if (ret) { if (ret) {
dev_err(dev, "can not enable the qspi_en clock\n"); dev_err(dev, "cannot enable the qspi_en clock: %d\n", ret);
return ret; return ret;
} }
ret = clk_prepare_enable(q->clk); ret = clk_prepare_enable(q->clk);
if (ret) { if (ret) {
dev_err(dev, "can not enable the qspi clock\n"); dev_err(dev, "cannot enable the qspi clock: %d\n", ret);
goto clk_failed; goto clk_failed;
} }
/* find the irq */ /* find the irq */
ret = platform_get_irq(pdev, 0); ret = platform_get_irq(pdev, 0);
if (ret < 0) { if (ret < 0) {
dev_err(dev, "failed to get the irq\n"); dev_err(dev, "failed to get the irq: %d\n", ret);
goto irq_failed; goto irq_failed;
} }
ret = devm_request_irq(dev, ret, ret = devm_request_irq(dev, ret,
fsl_qspi_irq_handler, 0, pdev->name, q); fsl_qspi_irq_handler, 0, pdev->name, q);
if (ret) { if (ret) {
dev_err(dev, "failed to request irq.\n"); dev_err(dev, "failed to request irq: %d\n", ret);
goto irq_failed; goto irq_failed;
} }
......
...@@ -369,17 +369,13 @@ static int spi_nor_erase(struct mtd_info *mtd, struct erase_info *instr) ...@@ -369,17 +369,13 @@ static int spi_nor_erase(struct mtd_info *mtd, struct erase_info *instr)
return ret; return ret;
} }
static int spi_nor_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len) static int stm_lock(struct spi_nor *nor, loff_t ofs, uint64_t len)
{ {
struct spi_nor *nor = mtd_to_spi_nor(mtd); struct mtd_info *mtd = nor->mtd;
uint32_t offset = ofs; uint32_t offset = ofs;
uint8_t status_old, status_new; uint8_t status_old, status_new;
int ret = 0; int ret = 0;
ret = spi_nor_lock_and_prep(nor, SPI_NOR_OPS_LOCK);
if (ret)
return ret;
status_old = read_sr(nor); status_old = read_sr(nor);
if (offset < mtd->size - (mtd->size / 2)) if (offset < mtd->size - (mtd->size / 2))
...@@ -402,26 +398,18 @@ static int spi_nor_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len) ...@@ -402,26 +398,18 @@ static int spi_nor_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
(status_old & (SR_BP2 | SR_BP1 | SR_BP0))) { (status_old & (SR_BP2 | SR_BP1 | SR_BP0))) {
write_enable(nor); write_enable(nor);
ret = write_sr(nor, status_new); ret = write_sr(nor, status_new);
if (ret)
goto err;
} }
err:
spi_nor_unlock_and_unprep(nor, SPI_NOR_OPS_LOCK);
return ret; return ret;
} }
static int spi_nor_unlock(struct mtd_info *mtd, loff_t ofs, uint64_t len) static int stm_unlock(struct spi_nor *nor, loff_t ofs, uint64_t len)
{ {
struct spi_nor *nor = mtd_to_spi_nor(mtd); struct mtd_info *mtd = nor->mtd;
uint32_t offset = ofs; uint32_t offset = ofs;
uint8_t status_old, status_new; uint8_t status_old, status_new;
int ret = 0; int ret = 0;
ret = spi_nor_lock_and_prep(nor, SPI_NOR_OPS_UNLOCK);
if (ret)
return ret;
status_old = read_sr(nor); status_old = read_sr(nor);
if (offset+len > mtd->size - (mtd->size / 64)) if (offset+len > mtd->size - (mtd->size / 64))
...@@ -444,15 +432,41 @@ static int spi_nor_unlock(struct mtd_info *mtd, loff_t ofs, uint64_t len) ...@@ -444,15 +432,41 @@ static int spi_nor_unlock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
(status_old & (SR_BP2 | SR_BP1 | SR_BP0))) { (status_old & (SR_BP2 | SR_BP1 | SR_BP0))) {
write_enable(nor); write_enable(nor);
ret = write_sr(nor, status_new); ret = write_sr(nor, status_new);
if (ret)
goto err;
} }
err: return ret;
}
static int spi_nor_lock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
{
struct spi_nor *nor = mtd_to_spi_nor(mtd);
int ret;
ret = spi_nor_lock_and_prep(nor, SPI_NOR_OPS_LOCK);
if (ret)
return ret;
ret = nor->flash_lock(nor, ofs, len);
spi_nor_unlock_and_unprep(nor, SPI_NOR_OPS_UNLOCK); spi_nor_unlock_and_unprep(nor, SPI_NOR_OPS_UNLOCK);
return ret; return ret;
} }
static int spi_nor_unlock(struct mtd_info *mtd, loff_t ofs, uint64_t len)
{
struct spi_nor *nor = mtd_to_spi_nor(mtd);
int ret;
ret = spi_nor_lock_and_prep(nor, SPI_NOR_OPS_UNLOCK);
if (ret)
return ret;
ret = nor->flash_unlock(nor, ofs, len);
spi_nor_unlock_and_unprep(nor, SPI_NOR_OPS_LOCK);
return ret;
}
/* Used when the "_ext_id" is two bytes at most */ /* Used when the "_ext_id" is two bytes at most */
#define INFO(_jedec_id, _ext_id, _sector_size, _n_sectors, _flags) \ #define INFO(_jedec_id, _ext_id, _sector_size, _n_sectors, _flags) \
((kernel_ulong_t)&(struct flash_info) { \ ((kernel_ulong_t)&(struct flash_info) { \
...@@ -524,6 +538,7 @@ static const struct spi_device_id spi_nor_ids[] = { ...@@ -524,6 +538,7 @@ static const struct spi_device_id spi_nor_ids[] = {
{ "en25q64", INFO(0x1c3017, 0, 64 * 1024, 128, SECT_4K) }, { "en25q64", INFO(0x1c3017, 0, 64 * 1024, 128, SECT_4K) },
{ "en25qh128", INFO(0x1c7018, 0, 64 * 1024, 256, 0) }, { "en25qh128", INFO(0x1c7018, 0, 64 * 1024, 256, 0) },
{ "en25qh256", INFO(0x1c7019, 0, 64 * 1024, 512, 0) }, { "en25qh256", INFO(0x1c7019, 0, 64 * 1024, 512, 0) },
{ "en25s64", INFO(0x1c3817, 0, 64 * 1024, 128, 0) },
/* ESMT */ /* ESMT */
{ "f25l32pa", INFO(0x8c2016, 0, 64 * 1024, 64, SECT_4K) }, { "f25l32pa", INFO(0x8c2016, 0, 64 * 1024, 64, SECT_4K) },
...@@ -553,6 +568,7 @@ static const struct spi_device_id spi_nor_ids[] = { ...@@ -553,6 +568,7 @@ static const struct spi_device_id spi_nor_ids[] = {
{ "mx25l3205d", INFO(0xc22016, 0, 64 * 1024, 64, 0) }, { "mx25l3205d", INFO(0xc22016, 0, 64 * 1024, 64, 0) },
{ "mx25l3255e", INFO(0xc29e16, 0, 64 * 1024, 64, SECT_4K) }, { "mx25l3255e", INFO(0xc29e16, 0, 64 * 1024, 64, SECT_4K) },
{ "mx25l6405d", INFO(0xc22017, 0, 64 * 1024, 128, 0) }, { "mx25l6405d", INFO(0xc22017, 0, 64 * 1024, 128, 0) },
{ "mx25u6435f", INFO(0xc22537, 0, 64 * 1024, 128, SECT_4K) },
{ "mx25l12805d", INFO(0xc22018, 0, 64 * 1024, 256, 0) }, { "mx25l12805d", INFO(0xc22018, 0, 64 * 1024, 256, 0) },
{ "mx25l12855e", INFO(0xc22618, 0, 64 * 1024, 256, 0) }, { "mx25l12855e", INFO(0xc22618, 0, 64 * 1024, 256, 0) },
{ "mx25l25635e", INFO(0xc22019, 0, 64 * 1024, 512, 0) }, { "mx25l25635e", INFO(0xc22019, 0, 64 * 1024, 512, 0) },
...@@ -648,6 +664,7 @@ static const struct spi_device_id spi_nor_ids[] = { ...@@ -648,6 +664,7 @@ static const struct spi_device_id spi_nor_ids[] = {
{ "m25px80", INFO(0x207114, 0, 64 * 1024, 16, 0) }, { "m25px80", INFO(0x207114, 0, 64 * 1024, 16, 0) },
/* Winbond -- w25x "blocks" are 64K, "sectors" are 4KiB */ /* Winbond -- w25x "blocks" are 64K, "sectors" are 4KiB */
{ "w25x05", INFO(0xef3010, 0, 64 * 1024, 1, SECT_4K) },
{ "w25x10", INFO(0xef3011, 0, 64 * 1024, 2, SECT_4K) }, { "w25x10", INFO(0xef3011, 0, 64 * 1024, 2, SECT_4K) },
{ "w25x20", INFO(0xef3012, 0, 64 * 1024, 4, SECT_4K) }, { "w25x20", INFO(0xef3012, 0, 64 * 1024, 4, SECT_4K) },
{ "w25x40", INFO(0xef3013, 0, 64 * 1024, 8, SECT_4K) }, { "w25x40", INFO(0xef3013, 0, 64 * 1024, 8, SECT_4K) },
...@@ -658,6 +675,7 @@ static const struct spi_device_id spi_nor_ids[] = { ...@@ -658,6 +675,7 @@ static const struct spi_device_id spi_nor_ids[] = {
{ "w25q32dw", INFO(0xef6016, 0, 64 * 1024, 64, SECT_4K) }, { "w25q32dw", INFO(0xef6016, 0, 64 * 1024, 64, SECT_4K) },
{ "w25x64", INFO(0xef3017, 0, 64 * 1024, 128, SECT_4K) }, { "w25x64", INFO(0xef3017, 0, 64 * 1024, 128, SECT_4K) },
{ "w25q64", INFO(0xef4017, 0, 64 * 1024, 128, SECT_4K) }, { "w25q64", INFO(0xef4017, 0, 64 * 1024, 128, SECT_4K) },
{ "w25q64dw", INFO(0xef6017, 0, 64 * 1024, 128, SECT_4K) },
{ "w25q80", INFO(0xef5014, 0, 64 * 1024, 16, SECT_4K) }, { "w25q80", INFO(0xef5014, 0, 64 * 1024, 16, SECT_4K) },
{ "w25q80bl", INFO(0xef4014, 0, 64 * 1024, 16, SECT_4K) }, { "w25q80bl", INFO(0xef4014, 0, 64 * 1024, 16, SECT_4K) },
{ "w25q128", INFO(0xef4018, 0, 64 * 1024, 256, SECT_4K) }, { "w25q128", INFO(0xef4018, 0, 64 * 1024, 256, SECT_4K) },
...@@ -1045,6 +1063,11 @@ int spi_nor_scan(struct spi_nor *nor, const char *name, enum read_mode mode) ...@@ -1045,6 +1063,11 @@ int spi_nor_scan(struct spi_nor *nor, const char *name, enum read_mode mode)
/* nor protection support for STmicro chips */ /* nor protection support for STmicro chips */
if (JEDEC_MFR(info) == CFI_MFR_ST) { if (JEDEC_MFR(info) == CFI_MFR_ST) {
nor->flash_lock = stm_lock;
nor->flash_unlock = stm_unlock;
}
if (nor->flash_lock && nor->flash_unlock) {
mtd->_lock = spi_nor_lock; mtd->_lock = spi_nor_lock;
mtd->_unlock = spi_nor_unlock; mtd->_unlock = spi_nor_unlock;
} }
......
...@@ -9,6 +9,8 @@ ...@@ -9,6 +9,8 @@
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/mtd/nand_ecc.h> #include <linux/mtd/nand_ecc.h>
#include "mtd_test.h"
/* /*
* Test the implementation for software ECC * Test the implementation for software ECC
* *
...@@ -274,6 +276,10 @@ static int nand_ecc_test_run(const size_t size) ...@@ -274,6 +276,10 @@ static int nand_ecc_test_run(const size_t size)
} }
pr_info("ok - %s-%zd\n", pr_info("ok - %s-%zd\n",
nand_ecc_test[i].name, size); nand_ecc_test[i].name, size);
err = mtdtest_relax();
if (err)
break;
} }
error: error:
kfree(error_data); kfree(error_data);
......
#include <linux/mtd/mtd.h> #include <linux/mtd/mtd.h>
#include <linux/sched.h>
static inline int mtdtest_relax(void)
{
cond_resched();
if (signal_pending(current)) {
pr_info("aborting test due to pending signal!\n");
return -EINTR;
}
return 0;
}
int mtdtest_erase_eraseblock(struct mtd_info *mtd, unsigned int ebnum); int mtdtest_erase_eraseblock(struct mtd_info *mtd, unsigned int ebnum);
int mtdtest_scan_for_bad_eraseblocks(struct mtd_info *mtd, unsigned char *bbt, int mtdtest_scan_for_bad_eraseblocks(struct mtd_info *mtd, unsigned char *bbt,
......
...@@ -320,6 +320,10 @@ static int overwrite_test(void) ...@@ -320,6 +320,10 @@ static int overwrite_test(void)
break; break;
} }
err = mtdtest_relax();
if (err)
break;
opno++; opno++;
} }
......
...@@ -70,7 +70,7 @@ static int write_eraseblock(int ebnum) ...@@ -70,7 +70,7 @@ static int write_eraseblock(int ebnum)
int i; int i;
struct mtd_oob_ops ops; struct mtd_oob_ops ops;
int err = 0; int err = 0;
loff_t addr = ebnum * mtd->erasesize; loff_t addr = (loff_t)ebnum * mtd->erasesize;
prandom_bytes_state(&rnd_state, writebuf, use_len_max * pgcnt); prandom_bytes_state(&rnd_state, writebuf, use_len_max * pgcnt);
for (i = 0; i < pgcnt; ++i, addr += mtd->writesize) { for (i = 0; i < pgcnt; ++i, addr += mtd->writesize) {
...@@ -112,7 +112,10 @@ static int write_whole_device(void) ...@@ -112,7 +112,10 @@ static int write_whole_device(void)
return err; return err;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("written up to eraseblock %u\n", i); pr_info("written up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
return err;
} }
pr_info("written %u eraseblocks\n", i); pr_info("written %u eraseblocks\n", i);
return 0; return 0;
...@@ -141,6 +144,31 @@ static size_t memcmpshow(loff_t addr, const void *cs, const void *ct, size_t cou ...@@ -141,6 +144,31 @@ static size_t memcmpshow(loff_t addr, const void *cs, const void *ct, size_t cou
return bitflips; return bitflips;
} }
/*
* Compare with 0xff and show the address, offset and data bytes at
* comparison failure. Return number of bitflips encountered.
*/
static size_t memffshow(loff_t addr, loff_t offset, const void *cs,
size_t count)
{
const unsigned char *su1;
int res;
size_t i = 0;
size_t bitflips = 0;
for (su1 = cs; 0 < count; ++su1, count--, i++) {
res = *su1 ^ 0xff;
if (res) {
pr_info("error @addr[0x%lx:0x%lx] 0x%x -> 0xff diff 0x%x\n",
(unsigned long)addr, (unsigned long)offset + i,
*su1, res);
bitflips += hweight8(res);
}
}
return bitflips;
}
static int verify_eraseblock(int ebnum) static int verify_eraseblock(int ebnum)
{ {
int i; int i;
...@@ -203,6 +231,15 @@ static int verify_eraseblock(int ebnum) ...@@ -203,6 +231,15 @@ static int verify_eraseblock(int ebnum)
bitflips = memcmpshow(addr, readbuf + use_offset, bitflips = memcmpshow(addr, readbuf + use_offset,
writebuf + (use_len_max * i) + use_offset, writebuf + (use_len_max * i) + use_offset,
use_len); use_len);
/* verify pre-offset area for 0xff */
bitflips += memffshow(addr, 0, readbuf, use_offset);
/* verify post-(use_offset + use_len) area for 0xff */
k = use_offset + use_len;
bitflips += memffshow(addr, k, readbuf + k,
mtd->ecclayout->oobavail - k);
if (bitflips > bitflip_limit) { if (bitflips > bitflip_limit) {
pr_err("error: verify failed at %#llx\n", pr_err("error: verify failed at %#llx\n",
(long long)addr); (long long)addr);
...@@ -212,33 +249,7 @@ static int verify_eraseblock(int ebnum) ...@@ -212,33 +249,7 @@ static int verify_eraseblock(int ebnum)
return -1; return -1;
} }
} else if (bitflips) { } else if (bitflips) {
pr_info("ignoring error as within bitflip_limit\n"); pr_info("ignoring errors as within bitflip limit\n");
}
for (k = 0; k < use_offset; ++k)
if (readbuf[k] != 0xff) {
pr_err("error: verify 0xff "
"failed at %#llx\n",
(long long)addr);
errcnt += 1;
if (errcnt > 1000) {
pr_err("error: too "
"many errors\n");
return -1;
}
}
for (k = use_offset + use_len;
k < mtd->ecclayout->oobavail; ++k)
if (readbuf[k] != 0xff) {
pr_err("error: verify 0xff "
"failed at %#llx\n",
(long long)addr);
errcnt += 1;
if (errcnt > 1000) {
pr_err("error: too "
"many errors\n");
return -1;
}
} }
} }
if (vary_offset) if (vary_offset)
...@@ -310,7 +321,10 @@ static int verify_all_eraseblocks(void) ...@@ -310,7 +321,10 @@ static int verify_all_eraseblocks(void)
return err; return err;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("verified up to eraseblock %u\n", i); pr_info("verified up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
return err;
} }
pr_info("verified %u eraseblocks\n", i); pr_info("verified %u eraseblocks\n", i);
return 0; return 0;
...@@ -421,7 +435,10 @@ static int __init mtd_oobtest_init(void) ...@@ -421,7 +435,10 @@ static int __init mtd_oobtest_init(void)
goto out; goto out;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("verified up to eraseblock %u\n", i); pr_info("verified up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
pr_info("verified %u eraseblocks\n", i); pr_info("verified %u eraseblocks\n", i);
...@@ -634,7 +651,11 @@ static int __init mtd_oobtest_init(void) ...@@ -634,7 +651,11 @@ static int __init mtd_oobtest_init(void)
goto out; goto out;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("written up to eraseblock %u\n", i); pr_info("written up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
addr += mtd->writesize; addr += mtd->writesize;
} }
} }
...@@ -672,7 +693,10 @@ static int __init mtd_oobtest_init(void) ...@@ -672,7 +693,10 @@ static int __init mtd_oobtest_init(void)
} }
if (i % 256 == 0) if (i % 256 == 0)
pr_info("verified up to eraseblock %u\n", i); pr_info("verified up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
pr_info("verified %u eraseblocks\n", i); pr_info("verified %u eraseblocks\n", i);
......
...@@ -407,7 +407,10 @@ static int __init mtd_pagetest_init(void) ...@@ -407,7 +407,10 @@ static int __init mtd_pagetest_init(void)
goto out; goto out;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("written up to eraseblock %u\n", i); pr_info("written up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
pr_info("written %u eraseblocks\n", i); pr_info("written %u eraseblocks\n", i);
...@@ -422,7 +425,10 @@ static int __init mtd_pagetest_init(void) ...@@ -422,7 +425,10 @@ static int __init mtd_pagetest_init(void)
goto out; goto out;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("verified up to eraseblock %u\n", i); pr_info("verified up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
pr_info("verified %u eraseblocks\n", i); pr_info("verified %u eraseblocks\n", i);
......
...@@ -190,7 +190,10 @@ static int __init mtd_readtest_init(void) ...@@ -190,7 +190,10 @@ static int __init mtd_readtest_init(void)
if (!err) if (!err)
err = ret; err = ret;
} }
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
if (err) if (err)
......
...@@ -185,7 +185,7 @@ static long calc_speed(void) ...@@ -185,7 +185,7 @@ static long calc_speed(void)
(finish.tv_usec - start.tv_usec) / 1000; (finish.tv_usec - start.tv_usec) / 1000;
if (ms == 0) if (ms == 0)
return 0; return 0;
k = goodebcnt * (mtd->erasesize / 1024) * 1000; k = (uint64_t)goodebcnt * (mtd->erasesize / 1024) * 1000;
do_div(k, ms); do_div(k, ms);
return k; return k;
} }
...@@ -269,7 +269,10 @@ static int __init mtd_speedtest_init(void) ...@@ -269,7 +269,10 @@ static int __init mtd_speedtest_init(void)
err = write_eraseblock(i); err = write_eraseblock(i);
if (err) if (err)
goto out; goto out;
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
stop_timing(); stop_timing();
speed = calc_speed(); speed = calc_speed();
...@@ -284,7 +287,10 @@ static int __init mtd_speedtest_init(void) ...@@ -284,7 +287,10 @@ static int __init mtd_speedtest_init(void)
err = read_eraseblock(i); err = read_eraseblock(i);
if (err) if (err)
goto out; goto out;
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
stop_timing(); stop_timing();
speed = calc_speed(); speed = calc_speed();
...@@ -303,7 +309,10 @@ static int __init mtd_speedtest_init(void) ...@@ -303,7 +309,10 @@ static int __init mtd_speedtest_init(void)
err = write_eraseblock_by_page(i); err = write_eraseblock_by_page(i);
if (err) if (err)
goto out; goto out;
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
stop_timing(); stop_timing();
speed = calc_speed(); speed = calc_speed();
...@@ -318,7 +327,10 @@ static int __init mtd_speedtest_init(void) ...@@ -318,7 +327,10 @@ static int __init mtd_speedtest_init(void)
err = read_eraseblock_by_page(i); err = read_eraseblock_by_page(i);
if (err) if (err)
goto out; goto out;
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
stop_timing(); stop_timing();
speed = calc_speed(); speed = calc_speed();
...@@ -337,7 +349,10 @@ static int __init mtd_speedtest_init(void) ...@@ -337,7 +349,10 @@ static int __init mtd_speedtest_init(void)
err = write_eraseblock_by_2pages(i); err = write_eraseblock_by_2pages(i);
if (err) if (err)
goto out; goto out;
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
stop_timing(); stop_timing();
speed = calc_speed(); speed = calc_speed();
...@@ -352,7 +367,10 @@ static int __init mtd_speedtest_init(void) ...@@ -352,7 +367,10 @@ static int __init mtd_speedtest_init(void)
err = read_eraseblock_by_2pages(i); err = read_eraseblock_by_2pages(i);
if (err) if (err)
goto out; goto out;
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
stop_timing(); stop_timing();
speed = calc_speed(); speed = calc_speed();
...@@ -385,7 +403,11 @@ static int __init mtd_speedtest_init(void) ...@@ -385,7 +403,11 @@ static int __init mtd_speedtest_init(void)
err = multiblock_erase(i, j); err = multiblock_erase(i, j);
if (err) if (err)
goto out; goto out;
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
i += j; i += j;
} }
stop_timing(); stop_timing();
......
...@@ -96,7 +96,7 @@ static int do_read(void) ...@@ -96,7 +96,7 @@ static int do_read(void)
if (offs + len > mtd->erasesize) if (offs + len > mtd->erasesize)
len = mtd->erasesize - offs; len = mtd->erasesize - offs;
} }
addr = eb * mtd->erasesize + offs; addr = (loff_t)eb * mtd->erasesize + offs;
return mtdtest_read(mtd, addr, len, readbuf); return mtdtest_read(mtd, addr, len, readbuf);
} }
...@@ -124,7 +124,7 @@ static int do_write(void) ...@@ -124,7 +124,7 @@ static int do_write(void)
offsets[eb + 1] = 0; offsets[eb + 1] = 0;
} }
} }
addr = eb * mtd->erasesize + offs; addr = (loff_t)eb * mtd->erasesize + offs;
err = mtdtest_write(mtd, addr, len, writebuf); err = mtdtest_write(mtd, addr, len, writebuf);
if (unlikely(err)) if (unlikely(err))
return err; return err;
...@@ -221,7 +221,10 @@ static int __init mtd_stresstest_init(void) ...@@ -221,7 +221,10 @@ static int __init mtd_stresstest_init(void)
err = do_operation(); err = do_operation();
if (err) if (err)
goto out; goto out;
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
pr_info("finished, %d operations done\n", op); pr_info("finished, %d operations done\n", op);
......
...@@ -95,7 +95,7 @@ static int write_eraseblock2(int ebnum) ...@@ -95,7 +95,7 @@ static int write_eraseblock2(int ebnum)
loff_t addr = (loff_t)ebnum * mtd->erasesize; loff_t addr = (loff_t)ebnum * mtd->erasesize;
for (k = 1; k < 33; ++k) { for (k = 1; k < 33; ++k) {
if (addr + (subpgsize * k) > (ebnum + 1) * mtd->erasesize) if (addr + (subpgsize * k) > (loff_t)(ebnum + 1) * mtd->erasesize)
break; break;
prandom_bytes_state(&rnd_state, writebuf, subpgsize * k); prandom_bytes_state(&rnd_state, writebuf, subpgsize * k);
err = mtd_write(mtd, addr, subpgsize * k, &written, writebuf); err = mtd_write(mtd, addr, subpgsize * k, &written, writebuf);
...@@ -195,7 +195,7 @@ static int verify_eraseblock2(int ebnum) ...@@ -195,7 +195,7 @@ static int verify_eraseblock2(int ebnum)
loff_t addr = (loff_t)ebnum * mtd->erasesize; loff_t addr = (loff_t)ebnum * mtd->erasesize;
for (k = 1; k < 33; ++k) { for (k = 1; k < 33; ++k) {
if (addr + (subpgsize * k) > (ebnum + 1) * mtd->erasesize) if (addr + (subpgsize * k) > (loff_t)(ebnum + 1) * mtd->erasesize)
break; break;
prandom_bytes_state(&rnd_state, writebuf, subpgsize * k); prandom_bytes_state(&rnd_state, writebuf, subpgsize * k);
clear_data(readbuf, subpgsize * k); clear_data(readbuf, subpgsize * k);
...@@ -269,7 +269,10 @@ static int verify_all_eraseblocks_ff(void) ...@@ -269,7 +269,10 @@ static int verify_all_eraseblocks_ff(void)
return err; return err;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("verified up to eraseblock %u\n", i); pr_info("verified up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
return err;
} }
pr_info("verified %u eraseblocks\n", i); pr_info("verified %u eraseblocks\n", i);
return 0; return 0;
...@@ -346,7 +349,10 @@ static int __init mtd_subpagetest_init(void) ...@@ -346,7 +349,10 @@ static int __init mtd_subpagetest_init(void)
goto out; goto out;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("written up to eraseblock %u\n", i); pr_info("written up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
pr_info("written %u eraseblocks\n", i); pr_info("written %u eraseblocks\n", i);
...@@ -360,7 +366,10 @@ static int __init mtd_subpagetest_init(void) ...@@ -360,7 +366,10 @@ static int __init mtd_subpagetest_init(void)
goto out; goto out;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("verified up to eraseblock %u\n", i); pr_info("verified up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
pr_info("verified %u eraseblocks\n", i); pr_info("verified %u eraseblocks\n", i);
...@@ -383,7 +392,10 @@ static int __init mtd_subpagetest_init(void) ...@@ -383,7 +392,10 @@ static int __init mtd_subpagetest_init(void)
goto out; goto out;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("written up to eraseblock %u\n", i); pr_info("written up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
pr_info("written %u eraseblocks\n", i); pr_info("written %u eraseblocks\n", i);
...@@ -398,7 +410,10 @@ static int __init mtd_subpagetest_init(void) ...@@ -398,7 +410,10 @@ static int __init mtd_subpagetest_init(void)
goto out; goto out;
if (i % 256 == 0) if (i % 256 == 0)
pr_info("verified up to eraseblock %u\n", i); pr_info("verified up to eraseblock %u\n", i);
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
pr_info("verified %u eraseblocks\n", i); pr_info("verified %u eraseblocks\n", i);
......
...@@ -101,11 +101,11 @@ static inline int check_eraseblock(int ebnum, unsigned char *buf) ...@@ -101,11 +101,11 @@ static inline int check_eraseblock(int ebnum, unsigned char *buf)
{ {
int err, retries = 0; int err, retries = 0;
size_t read; size_t read;
loff_t addr = ebnum * mtd->erasesize; loff_t addr = (loff_t)ebnum * mtd->erasesize;
size_t len = mtd->erasesize; size_t len = mtd->erasesize;
if (pgcnt) { if (pgcnt) {
addr = (ebnum + 1) * mtd->erasesize - pgcnt * pgsize; addr = (loff_t)(ebnum + 1) * mtd->erasesize - pgcnt * pgsize;
len = pgcnt * pgsize; len = pgcnt * pgsize;
} }
...@@ -155,11 +155,11 @@ static inline int write_pattern(int ebnum, void *buf) ...@@ -155,11 +155,11 @@ static inline int write_pattern(int ebnum, void *buf)
{ {
int err; int err;
size_t written; size_t written;
loff_t addr = ebnum * mtd->erasesize; loff_t addr = (loff_t)ebnum * mtd->erasesize;
size_t len = mtd->erasesize; size_t len = mtd->erasesize;
if (pgcnt) { if (pgcnt) {
addr = (ebnum + 1) * mtd->erasesize - pgcnt * pgsize; addr = (loff_t)(ebnum + 1) * mtd->erasesize - pgcnt * pgsize;
len = pgcnt * pgsize; len = pgcnt * pgsize;
} }
err = mtd_write(mtd, addr, len, &written, buf); err = mtd_write(mtd, addr, len, &written, buf);
...@@ -279,7 +279,10 @@ static int __init tort_init(void) ...@@ -279,7 +279,10 @@ static int __init tort_init(void)
" for 0xFF... pattern\n"); " for 0xFF... pattern\n");
goto out; goto out;
} }
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
} }
...@@ -294,7 +297,10 @@ static int __init tort_init(void) ...@@ -294,7 +297,10 @@ static int __init tort_init(void)
err = write_pattern(i, patt); err = write_pattern(i, patt);
if (err) if (err)
goto out; goto out;
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
/* Verify what we wrote */ /* Verify what we wrote */
...@@ -314,7 +320,10 @@ static int __init tort_init(void) ...@@ -314,7 +320,10 @@ static int __init tort_init(void)
"0x55AA55..." : "0xAA55AA..."); "0x55AA55..." : "0xAA55AA...");
goto out; goto out;
} }
cond_resched();
err = mtdtest_relax();
if (err)
goto out;
} }
} }
......
...@@ -1266,7 +1266,6 @@ int jffs2_garbage_collect_xattr_ref(struct jffs2_sb_info *c, struct jffs2_xattr_ ...@@ -1266,7 +1266,6 @@ int jffs2_garbage_collect_xattr_ref(struct jffs2_sb_info *c, struct jffs2_xattr_
if (rc) { if (rc) {
JFFS2_WARNING("%s: jffs2_reserve_space_gc() = %d, request = %u\n", JFFS2_WARNING("%s: jffs2_reserve_space_gc() = %d, request = %u\n",
__func__, rc, totlen); __func__, rc, totlen);
rc = rc ? rc : -EBADFD;
goto out; goto out;
} }
rc = save_xattr_ref(c, ref); rc = save_xattr_ref(c, ref);
......
...@@ -77,7 +77,7 @@ ...@@ -77,7 +77,7 @@
/* ensure we never evaluate anything shorted than an unsigned long /* ensure we never evaluate anything shorted than an unsigned long
* to zero, and ensure we'll never miss the end of an comparison (bjd) */ * to zero, and ensure we'll never miss the end of an comparison (bjd) */
#define map_calc_words(map) ((map_bankwidth(map) + (sizeof(unsigned long)-1))/ sizeof(unsigned long)) #define map_calc_words(map) ((map_bankwidth(map) + (sizeof(unsigned long)-1)) / sizeof(unsigned long))
#ifdef CONFIG_MTD_MAP_BANK_WIDTH_8 #ifdef CONFIG_MTD_MAP_BANK_WIDTH_8
# ifdef map_bankwidth # ifdef map_bankwidth
...@@ -181,7 +181,7 @@ static inline int map_bankwidth_supported(int w) ...@@ -181,7 +181,7 @@ static inline int map_bankwidth_supported(int w)
} }
} }
#define MAX_MAP_LONGS ( ((MAX_MAP_BANKWIDTH*8) + BITS_PER_LONG - 1) / BITS_PER_LONG ) #define MAX_MAP_LONGS (((MAX_MAP_BANKWIDTH * 8) + BITS_PER_LONG - 1) / BITS_PER_LONG)
typedef union { typedef union {
unsigned long x[MAX_MAP_LONGS]; unsigned long x[MAX_MAP_LONGS];
...@@ -264,20 +264,22 @@ void unregister_mtd_chip_driver(struct mtd_chip_driver *); ...@@ -264,20 +264,22 @@ void unregister_mtd_chip_driver(struct mtd_chip_driver *);
struct mtd_info *do_map_probe(const char *name, struct map_info *map); struct mtd_info *do_map_probe(const char *name, struct map_info *map);
void map_destroy(struct mtd_info *mtd); void map_destroy(struct mtd_info *mtd);
#define ENABLE_VPP(map) do { if(map->set_vpp) map->set_vpp(map, 1); } while(0) #define ENABLE_VPP(map) do { if (map->set_vpp) map->set_vpp(map, 1); } while (0)
#define DISABLE_VPP(map) do { if(map->set_vpp) map->set_vpp(map, 0); } while(0) #define DISABLE_VPP(map) do { if (map->set_vpp) map->set_vpp(map, 0); } while (0)
#define INVALIDATE_CACHED_RANGE(map, from, size) \ #define INVALIDATE_CACHED_RANGE(map, from, size) \
do { if(map->inval_cache) map->inval_cache(map, from, size); } while(0) do { if (map->inval_cache) map->inval_cache(map, from, size); } while (0)
static inline int map_word_equal(struct map_info *map, map_word val1, map_word val2) static inline int map_word_equal(struct map_info *map, map_word val1, map_word val2)
{ {
int i; int i;
for (i=0; i<map_words(map); i++) {
for (i = 0; i < map_words(map); i++) {
if (val1.x[i] != val2.x[i]) if (val1.x[i] != val2.x[i])
return 0; return 0;
} }
return 1; return 1;
} }
...@@ -286,9 +288,9 @@ static inline map_word map_word_and(struct map_info *map, map_word val1, map_wor ...@@ -286,9 +288,9 @@ static inline map_word map_word_and(struct map_info *map, map_word val1, map_wor
map_word r; map_word r;
int i; int i;
for (i=0; i<map_words(map); i++) { for (i = 0; i < map_words(map); i++)
r.x[i] = val1.x[i] & val2.x[i]; r.x[i] = val1.x[i] & val2.x[i];
}
return r; return r;
} }
...@@ -297,9 +299,9 @@ static inline map_word map_word_clr(struct map_info *map, map_word val1, map_wor ...@@ -297,9 +299,9 @@ static inline map_word map_word_clr(struct map_info *map, map_word val1, map_wor
map_word r; map_word r;
int i; int i;
for (i=0; i<map_words(map); i++) { for (i = 0; i < map_words(map); i++)
r.x[i] = val1.x[i] & ~val2.x[i]; r.x[i] = val1.x[i] & ~val2.x[i];
}
return r; return r;
} }
...@@ -308,22 +310,33 @@ static inline map_word map_word_or(struct map_info *map, map_word val1, map_word ...@@ -308,22 +310,33 @@ static inline map_word map_word_or(struct map_info *map, map_word val1, map_word
map_word r; map_word r;
int i; int i;
for (i=0; i<map_words(map); i++) { for (i = 0; i < map_words(map); i++)
r.x[i] = val1.x[i] | val2.x[i]; r.x[i] = val1.x[i] | val2.x[i];
}
return r; return r;
} }
#define map_word_andequal(m, a, b, z) map_word_equal(m, z, map_word_and(m, a, b)) static inline int map_word_andequal(struct map_info *map, map_word val1, map_word val2, map_word val3)
{
int i;
for (i = 0; i < map_words(map); i++) {
if ((val1.x[i] & val2.x[i]) != val3.x[i])
return 0;
}
return 1;
}
static inline int map_word_bitsset(struct map_info *map, map_word val1, map_word val2) static inline int map_word_bitsset(struct map_info *map, map_word val1, map_word val2)
{ {
int i; int i;
for (i=0; i<map_words(map); i++) { for (i = 0; i < map_words(map); i++) {
if (val1.x[i] & val2.x[i]) if (val1.x[i] & val2.x[i])
return 1; return 1;
} }
return 0; return 0;
} }
...@@ -355,14 +368,16 @@ static inline map_word map_word_load_partial(struct map_info *map, map_word orig ...@@ -355,14 +368,16 @@ static inline map_word map_word_load_partial(struct map_info *map, map_word orig
if (map_bankwidth_is_large(map)) { if (map_bankwidth_is_large(map)) {
char *dest = (char *)&orig; char *dest = (char *)&orig;
memcpy(dest+start, buf, len); memcpy(dest+start, buf, len);
} else { } else {
for (i=start; i < start+len; i++) { for (i = start; i < start+len; i++) {
int bitpos; int bitpos;
#ifdef __LITTLE_ENDIAN #ifdef __LITTLE_ENDIAN
bitpos = i*8; bitpos = i * 8;
#else /* __BIG_ENDIAN */ #else /* __BIG_ENDIAN */
bitpos = (map_bankwidth(map)-1-i)*8; bitpos = (map_bankwidth(map) - 1 - i) * 8;
#endif #endif
orig.x[0] &= ~(0xff << bitpos); orig.x[0] &= ~(0xff << bitpos);
orig.x[0] |= (unsigned long)buf[i-start] << bitpos; orig.x[0] |= (unsigned long)buf[i-start] << bitpos;
...@@ -384,9 +399,10 @@ static inline map_word map_word_ff(struct map_info *map) ...@@ -384,9 +399,10 @@ static inline map_word map_word_ff(struct map_info *map)
if (map_bankwidth(map) < MAP_FF_LIMIT) { if (map_bankwidth(map) < MAP_FF_LIMIT) {
int bw = 8 * map_bankwidth(map); int bw = 8 * map_bankwidth(map);
r.x[0] = (1UL << bw) - 1; r.x[0] = (1UL << bw) - 1;
} else { } else {
for (i=0; i<map_words(map); i++) for (i = 0; i < map_words(map); i++)
r.x[i] = ~0UL; r.x[i] = ~0UL;
} }
return r; return r;
...@@ -407,7 +423,7 @@ static inline map_word inline_map_read(struct map_info *map, unsigned long ofs) ...@@ -407,7 +423,7 @@ static inline map_word inline_map_read(struct map_info *map, unsigned long ofs)
r.x[0] = __raw_readq(map->virt + ofs); r.x[0] = __raw_readq(map->virt + ofs);
#endif #endif
else if (map_bankwidth_is_large(map)) else if (map_bankwidth_is_large(map))
memcpy_fromio(r.x, map->virt+ofs, map->bankwidth); memcpy_fromio(r.x, map->virt + ofs, map->bankwidth);
else else
BUG(); BUG();
......
...@@ -155,6 +155,8 @@ enum spi_nor_option_flags { ...@@ -155,6 +155,8 @@ enum spi_nor_option_flags {
* @write: [DRIVER-SPECIFIC] write data to the SPI NOR * @write: [DRIVER-SPECIFIC] write data to the SPI NOR
* @erase: [DRIVER-SPECIFIC] erase a sector of the SPI NOR * @erase: [DRIVER-SPECIFIC] erase a sector of the SPI NOR
* at the offset @offs * at the offset @offs
* @lock: [FLASH-SPECIFIC] lock a region of the SPI NOR
* @unlock: [FLASH-SPECIFIC] unlock a region of the SPI NOR
* @priv: the private data * @priv: the private data
*/ */
struct spi_nor { struct spi_nor {
...@@ -189,6 +191,9 @@ struct spi_nor { ...@@ -189,6 +191,9 @@ struct spi_nor {
size_t len, size_t *retlen, const u_char *write_buf); size_t len, size_t *retlen, const u_char *write_buf);
int (*erase)(struct spi_nor *nor, loff_t offs); int (*erase)(struct spi_nor *nor, loff_t offs);
int (*flash_lock)(struct spi_nor *nor, loff_t ofs, uint64_t len);
int (*flash_unlock)(struct spi_nor *nor, loff_t ofs, uint64_t len);
void *priv; void *priv;
}; };
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment