2006-01-08 16:34:27 -05:00
|
|
|
/*
|
2007-06-24 18:12:35 -04:00
|
|
|
* MTD SPI driver for ST M25Pxx (and similar) serial flash chips
|
2006-01-08 16:34:27 -05:00
|
|
|
*
|
|
|
|
* Author: Mike Lavender, mike@steroidmicros.com
|
|
|
|
*
|
|
|
|
* Copyright (c) 2005, Intec Automation Inc.
|
|
|
|
*
|
|
|
|
* Some parts are based on lart.c by Abraham Van Der Merwe
|
|
|
|
*
|
|
|
|
* Cleaned up and generalized based on mtd_dataflash.c
|
|
|
|
*
|
|
|
|
* This code is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/init.h>
|
2010-06-22 12:57:42 -04:00
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/errno.h>
|
2006-01-08 16:34:27 -05:00
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/device.h>
|
|
|
|
#include <linux/interrupt.h>
|
2007-06-24 18:09:13 -04:00
|
|
|
#include <linux/mutex.h>
|
2008-12-18 07:10:05 -05:00
|
|
|
#include <linux/math64.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 04:04:11 -04:00
|
|
|
#include <linux/slab.h>
|
2009-10-07 09:09:06 -04:00
|
|
|
#include <linux/sched.h>
|
2009-10-12 12:24:35 -04:00
|
|
|
#include <linux/mod_devicetable.h>
|
2007-06-24 18:09:13 -04:00
|
|
|
|
2006-01-08 16:34:27 -05:00
|
|
|
#include <linux/mtd/mtd.h>
|
|
|
|
#include <linux/mtd/partitions.h>
|
2007-06-24 18:09:13 -04:00
|
|
|
|
2006-01-08 16:34:27 -05:00
|
|
|
#include <linux/spi/spi.h>
|
|
|
|
#include <linux/spi/flash.h>
|
|
|
|
|
|
|
|
/* Flash opcodes. */
|
2007-06-24 18:12:35 -04:00
|
|
|
#define OPCODE_WREN 0x06 /* Write enable */
|
|
|
|
#define OPCODE_RDSR 0x05 /* Read status register */
|
2008-07-04 02:54:42 -04:00
|
|
|
#define OPCODE_WRSR 0x01 /* Write status register 1 byte */
|
2008-04-25 00:07:32 -04:00
|
|
|
#define OPCODE_NORM_READ 0x03 /* Read data bytes (low frequency) */
|
2007-06-24 18:12:35 -04:00
|
|
|
#define OPCODE_FAST_READ 0x0b /* Read data bytes (high frequency) */
|
|
|
|
#define OPCODE_PP 0x02 /* Page program (up to 256 bytes) */
|
2008-11-26 05:23:57 -05:00
|
|
|
#define OPCODE_BE_4K 0x20 /* Erase 4KiB block */
|
2007-06-28 17:38:38 -04:00
|
|
|
#define OPCODE_BE_32K 0x52 /* Erase 32KiB block */
|
2008-11-26 05:23:57 -05:00
|
|
|
#define OPCODE_CHIP_ERASE 0xc7 /* Erase whole flash chip */
|
2007-06-28 17:38:38 -04:00
|
|
|
#define OPCODE_SE 0xd8 /* Sector erase (usually 64KiB) */
|
2006-01-08 16:34:27 -05:00
|
|
|
#define OPCODE_RDID 0x9f /* Read JEDEC ID */
|
|
|
|
|
2009-06-15 04:23:41 -04:00
|
|
|
/* Used for SST flashes only. */
|
|
|
|
#define OPCODE_BP 0x02 /* Byte program */
|
|
|
|
#define OPCODE_WRDI 0x04 /* Write disable */
|
|
|
|
#define OPCODE_AAI_WP 0xad /* Auto address increment word program */
|
|
|
|
|
2006-01-08 16:34:27 -05:00
|
|
|
/* Status Register bits. */
|
|
|
|
#define SR_WIP 1 /* Write in progress */
|
|
|
|
#define SR_WEL 2 /* Write enable latch */
|
2007-06-24 18:12:35 -04:00
|
|
|
/* meaning of other SR_* bits may differ between vendors */
|
2006-01-08 16:34:27 -05:00
|
|
|
#define SR_BP0 4 /* Block protect 0 */
|
|
|
|
#define SR_BP1 8 /* Block protect 1 */
|
|
|
|
#define SR_BP2 0x10 /* Block protect 2 */
|
|
|
|
#define SR_SRWD 0x80 /* SR write protect */
|
|
|
|
|
|
|
|
/* Define max times to check status register before we give up. */
|
2009-06-26 12:42:47 -04:00
|
|
|
#define MAX_READY_WAIT_JIFFIES (40 * HZ) /* M25P16 specs 40s max chip erase */
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
#define MAX_CMD_SIZE 4
|
2006-01-08 16:34:27 -05:00
|
|
|
|
2008-04-25 00:07:32 -04:00
|
|
|
#ifdef CONFIG_M25PXX_USE_FAST_READ
|
|
|
|
#define OPCODE_READ OPCODE_FAST_READ
|
|
|
|
#define FAST_READ_DUMMY_BYTE 1
|
|
|
|
#else
|
|
|
|
#define OPCODE_READ OPCODE_NORM_READ
|
|
|
|
#define FAST_READ_DUMMY_BYTE 0
|
|
|
|
#endif
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/****************************************************************************/
|
|
|
|
|
|
|
|
struct m25p {
|
|
|
|
struct spi_device *spi;
|
2007-06-24 18:09:13 -04:00
|
|
|
struct mutex lock;
|
2006-01-08 16:34:27 -05:00
|
|
|
struct mtd_info mtd;
|
2007-06-24 18:12:35 -04:00
|
|
|
unsigned partitioned:1;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
u16 page_size;
|
|
|
|
u16 addr_width;
|
2007-06-24 18:12:35 -04:00
|
|
|
u8 erase_opcode;
|
2009-10-28 09:21:37 -04:00
|
|
|
u8 *command;
|
2006-01-08 16:34:27 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
static inline struct m25p *mtd_to_m25p(struct mtd_info *mtd)
|
|
|
|
{
|
|
|
|
return container_of(mtd, struct m25p, mtd);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Internal helper functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read the status register, returning its value in the location
|
|
|
|
* Return the status register value.
|
|
|
|
* Returns negative if error occurred.
|
|
|
|
*/
|
|
|
|
static int read_sr(struct m25p *flash)
|
|
|
|
{
|
|
|
|
ssize_t retval;
|
|
|
|
u8 code = OPCODE_RDSR;
|
|
|
|
u8 val;
|
|
|
|
|
|
|
|
retval = spi_write_then_read(flash->spi, &code, 1, &val, 1);
|
|
|
|
|
|
|
|
if (retval < 0) {
|
|
|
|
dev_err(&flash->spi->dev, "error %d reading SR\n",
|
|
|
|
(int) retval);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2008-07-04 02:54:42 -04:00
|
|
|
/*
|
|
|
|
* Write status register 1 byte
|
|
|
|
* Returns negative if error occurred.
|
|
|
|
*/
|
|
|
|
static int write_sr(struct m25p *flash, u8 val)
|
|
|
|
{
|
|
|
|
flash->command[0] = OPCODE_WRSR;
|
|
|
|
flash->command[1] = val;
|
|
|
|
|
|
|
|
return spi_write(flash->spi, flash->command, 2);
|
|
|
|
}
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Set write enable latch with Write Enable command.
|
|
|
|
* Returns negative if error occurred.
|
|
|
|
*/
|
|
|
|
static inline int write_enable(struct m25p *flash)
|
|
|
|
{
|
|
|
|
u8 code = OPCODE_WREN;
|
|
|
|
|
2008-10-20 04:26:16 -04:00
|
|
|
return spi_write_then_read(flash->spi, &code, 1, NULL, 0);
|
2006-01-08 16:34:27 -05:00
|
|
|
}
|
|
|
|
|
2009-06-15 04:23:41 -04:00
|
|
|
/*
|
|
|
|
* Send write disble instruction to the chip.
|
|
|
|
*/
|
|
|
|
static inline int write_disable(struct m25p *flash)
|
|
|
|
{
|
|
|
|
u8 code = OPCODE_WRDI;
|
|
|
|
|
|
|
|
return spi_write_then_read(flash->spi, &code, 1, NULL, 0);
|
|
|
|
}
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Service routine to read status register until ready, or timeout occurs.
|
|
|
|
* Returns non-zero if error.
|
|
|
|
*/
|
|
|
|
static int wait_till_ready(struct m25p *flash)
|
|
|
|
{
|
2009-05-08 08:51:53 -04:00
|
|
|
unsigned long deadline;
|
2006-01-08 16:34:27 -05:00
|
|
|
int sr;
|
|
|
|
|
2009-05-08 08:51:53 -04:00
|
|
|
deadline = jiffies + MAX_READY_WAIT_JIFFIES;
|
|
|
|
|
|
|
|
do {
|
2006-01-08 16:34:27 -05:00
|
|
|
if ((sr = read_sr(flash)) < 0)
|
|
|
|
break;
|
|
|
|
else if (!(sr & SR_WIP))
|
|
|
|
return 0;
|
|
|
|
|
2009-05-08 08:51:53 -04:00
|
|
|
cond_resched();
|
|
|
|
|
|
|
|
} while (!time_after_eq(jiffies, deadline));
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2008-08-11 04:59:13 -04:00
|
|
|
/*
|
|
|
|
* Erase the whole flash memory
|
|
|
|
*
|
|
|
|
* Returns 0 if successful, non-zero otherwise.
|
|
|
|
*/
|
2008-11-26 05:23:57 -05:00
|
|
|
static int erase_chip(struct m25p *flash)
|
2008-08-11 04:59:13 -04:00
|
|
|
{
|
2008-12-18 07:10:05 -05:00
|
|
|
DEBUG(MTD_DEBUG_LEVEL3, "%s: %s %lldKiB\n",
|
2008-12-23 05:00:14 -05:00
|
|
|
dev_name(&flash->spi->dev), __func__,
|
|
|
|
(long long)(flash->mtd.size >> 10));
|
2008-08-11 04:59:13 -04:00
|
|
|
|
|
|
|
/* Wait until finished previous write command. */
|
|
|
|
if (wait_till_ready(flash))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* Send write enable, then erase commands. */
|
|
|
|
write_enable(flash);
|
|
|
|
|
|
|
|
/* Set up command buffer. */
|
2008-11-26 05:23:57 -05:00
|
|
|
flash->command[0] = OPCODE_CHIP_ERASE;
|
2008-08-11 04:59:13 -04:00
|
|
|
|
|
|
|
spi_write(flash->spi, flash->command, 1);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2006-01-08 16:34:27 -05:00
|
|
|
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
static void m25p_addr2cmd(struct m25p *flash, unsigned int addr, u8 *cmd)
|
|
|
|
{
|
|
|
|
/* opcode is in cmd[0] */
|
|
|
|
cmd[1] = addr >> (flash->addr_width * 8 - 8);
|
|
|
|
cmd[2] = addr >> (flash->addr_width * 8 - 16);
|
|
|
|
cmd[3] = addr >> (flash->addr_width * 8 - 24);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int m25p_cmdsz(struct m25p *flash)
|
|
|
|
{
|
|
|
|
return 1 + flash->addr_width;
|
|
|
|
}
|
|
|
|
|
2006-01-08 16:34:27 -05:00
|
|
|
/*
|
|
|
|
* Erase one sector of flash memory at offset ``offset'' which is any
|
|
|
|
* address within the sector which should be erased.
|
|
|
|
*
|
|
|
|
* Returns 0 if successful, non-zero otherwise.
|
|
|
|
*/
|
|
|
|
static int erase_sector(struct m25p *flash, u32 offset)
|
|
|
|
{
|
2007-06-28 17:38:38 -04:00
|
|
|
DEBUG(MTD_DEBUG_LEVEL3, "%s: %s %dKiB at 0x%08x\n",
|
2008-12-23 05:00:14 -05:00
|
|
|
dev_name(&flash->spi->dev), __func__,
|
2007-06-24 18:12:35 -04:00
|
|
|
flash->mtd.erasesize / 1024, offset);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/* Wait until finished previous write command. */
|
|
|
|
if (wait_till_ready(flash))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
/* Send write enable, then erase commands. */
|
|
|
|
write_enable(flash);
|
|
|
|
|
|
|
|
/* Set up command buffer. */
|
2007-06-24 18:12:35 -04:00
|
|
|
flash->command[0] = flash->erase_opcode;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
m25p_addr2cmd(flash, offset, flash->command);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
spi_write(flash->spi, flash->command, m25p_cmdsz(flash));
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* MTD implementation
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Erase an address range on the flash chip. The address range may extend
|
|
|
|
* one or more erase sectors. Return an error is there is a problem erasing.
|
|
|
|
*/
|
|
|
|
static int m25p80_erase(struct mtd_info *mtd, struct erase_info *instr)
|
|
|
|
{
|
|
|
|
struct m25p *flash = mtd_to_m25p(mtd);
|
|
|
|
u32 addr,len;
|
2008-12-18 07:10:05 -05:00
|
|
|
uint32_t rem;
|
2006-01-08 16:34:27 -05:00
|
|
|
|
2008-12-18 07:10:05 -05:00
|
|
|
DEBUG(MTD_DEBUG_LEVEL2, "%s: %s %s 0x%llx, len %lld\n",
|
2008-12-23 05:00:14 -05:00
|
|
|
dev_name(&flash->spi->dev), __func__, "at",
|
|
|
|
(long long)instr->addr, (long long)instr->len);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/* sanity checks */
|
|
|
|
if (instr->addr + instr->len > flash->mtd.size)
|
|
|
|
return -EINVAL;
|
2008-12-18 07:10:05 -05:00
|
|
|
div_u64_rem(instr->len, mtd->erasesize, &rem);
|
|
|
|
if (rem)
|
2006-01-08 16:34:27 -05:00
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
addr = instr->addr;
|
|
|
|
len = instr->len;
|
|
|
|
|
2007-06-24 18:09:13 -04:00
|
|
|
mutex_lock(&flash->lock);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
2008-11-26 05:23:57 -05:00
|
|
|
/* whole-chip erase? */
|
2009-04-27 17:10:10 -04:00
|
|
|
if (len == flash->mtd.size) {
|
|
|
|
if (erase_chip(flash)) {
|
|
|
|
instr->state = MTD_ERASE_FAILED;
|
|
|
|
mutex_unlock(&flash->lock);
|
|
|
|
return -EIO;
|
|
|
|
}
|
2008-11-26 05:23:57 -05:00
|
|
|
|
|
|
|
/* REVISIT in some cases we could speed up erasing large regions
|
|
|
|
* by using OPCODE_SE instead of OPCODE_BE_4K. We may have set up
|
|
|
|
* to use "small sector erase", but that's not always optimal.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* "sector"-at-a-time erase */
|
2008-08-11 04:59:13 -04:00
|
|
|
} else {
|
|
|
|
while (len) {
|
|
|
|
if (erase_sector(flash, addr)) {
|
|
|
|
instr->state = MTD_ERASE_FAILED;
|
|
|
|
mutex_unlock(&flash->lock);
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
addr += mtd->erasesize;
|
|
|
|
len -= mtd->erasesize;
|
2006-01-08 16:34:27 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-06-24 18:09:13 -04:00
|
|
|
mutex_unlock(&flash->lock);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
instr->state = MTD_ERASE_DONE;
|
|
|
|
mtd_erase_callback(instr);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read an address range from the flash chip. The address range
|
|
|
|
* may be any size provided it is within the physical boundaries.
|
|
|
|
*/
|
|
|
|
static int m25p80_read(struct mtd_info *mtd, loff_t from, size_t len,
|
|
|
|
size_t *retlen, u_char *buf)
|
|
|
|
{
|
|
|
|
struct m25p *flash = mtd_to_m25p(mtd);
|
|
|
|
struct spi_transfer t[2];
|
|
|
|
struct spi_message m;
|
|
|
|
|
|
|
|
DEBUG(MTD_DEBUG_LEVEL2, "%s: %s %s 0x%08x, len %zd\n",
|
2008-12-23 05:00:14 -05:00
|
|
|
dev_name(&flash->spi->dev), __func__, "from",
|
2006-01-08 16:34:27 -05:00
|
|
|
(u32)from, len);
|
|
|
|
|
|
|
|
/* sanity checks */
|
|
|
|
if (!len)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (from + len > flash->mtd.size)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2006-01-08 16:34:28 -05:00
|
|
|
spi_message_init(&m);
|
|
|
|
memset(t, 0, (sizeof t));
|
|
|
|
|
2008-04-25 00:07:32 -04:00
|
|
|
/* NOTE:
|
|
|
|
* OPCODE_FAST_READ (if available) is faster.
|
|
|
|
* Should add 1 byte DUMMY_BYTE.
|
|
|
|
*/
|
2006-01-08 16:34:28 -05:00
|
|
|
t[0].tx_buf = flash->command;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
t[0].len = m25p_cmdsz(flash) + FAST_READ_DUMMY_BYTE;
|
2006-01-08 16:34:28 -05:00
|
|
|
spi_message_add_tail(&t[0], &m);
|
|
|
|
|
|
|
|
t[1].rx_buf = buf;
|
|
|
|
t[1].len = len;
|
|
|
|
spi_message_add_tail(&t[1], &m);
|
|
|
|
|
|
|
|
/* Byte count starts at zero. */
|
|
|
|
if (retlen)
|
|
|
|
*retlen = 0;
|
|
|
|
|
2007-06-24 18:09:13 -04:00
|
|
|
mutex_lock(&flash->lock);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/* Wait till previous write/erase is done. */
|
|
|
|
if (wait_till_ready(flash)) {
|
|
|
|
/* REVISIT status return?? */
|
2007-06-24 18:09:13 -04:00
|
|
|
mutex_unlock(&flash->lock);
|
2006-01-08 16:34:27 -05:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2007-06-24 18:12:35 -04:00
|
|
|
/* FIXME switch to OPCODE_FAST_READ. It's required for higher
|
|
|
|
* clocks; and at this writing, every chip this driver handles
|
|
|
|
* supports that opcode.
|
|
|
|
*/
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/* Set up the write data buffer. */
|
|
|
|
flash->command[0] = OPCODE_READ;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
m25p_addr2cmd(flash, from, flash->command);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
spi_sync(flash->spi, &m);
|
|
|
|
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
*retlen = m.actual_length - m25p_cmdsz(flash) - FAST_READ_DUMMY_BYTE;
|
2006-01-08 16:34:27 -05:00
|
|
|
|
2007-06-24 18:09:13 -04:00
|
|
|
mutex_unlock(&flash->lock);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Write an address range to the flash chip. Data must be written in
|
|
|
|
* FLASH_PAGESIZE chunks. The address range may be any size provided
|
|
|
|
* it is within the physical boundaries.
|
|
|
|
*/
|
|
|
|
static int m25p80_write(struct mtd_info *mtd, loff_t to, size_t len,
|
|
|
|
size_t *retlen, const u_char *buf)
|
|
|
|
{
|
|
|
|
struct m25p *flash = mtd_to_m25p(mtd);
|
|
|
|
u32 page_offset, page_size;
|
|
|
|
struct spi_transfer t[2];
|
|
|
|
struct spi_message m;
|
|
|
|
|
|
|
|
DEBUG(MTD_DEBUG_LEVEL2, "%s: %s %s 0x%08x, len %zd\n",
|
2008-12-23 05:00:14 -05:00
|
|
|
dev_name(&flash->spi->dev), __func__, "to",
|
2006-01-08 16:34:27 -05:00
|
|
|
(u32)to, len);
|
|
|
|
|
|
|
|
if (retlen)
|
|
|
|
*retlen = 0;
|
|
|
|
|
|
|
|
/* sanity checks */
|
|
|
|
if (!len)
|
|
|
|
return(0);
|
|
|
|
|
|
|
|
if (to + len > flash->mtd.size)
|
|
|
|
return -EINVAL;
|
|
|
|
|
2006-01-08 16:34:28 -05:00
|
|
|
spi_message_init(&m);
|
|
|
|
memset(t, 0, (sizeof t));
|
|
|
|
|
|
|
|
t[0].tx_buf = flash->command;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
t[0].len = m25p_cmdsz(flash);
|
2006-01-08 16:34:28 -05:00
|
|
|
spi_message_add_tail(&t[0], &m);
|
|
|
|
|
|
|
|
t[1].tx_buf = buf;
|
|
|
|
spi_message_add_tail(&t[1], &m);
|
|
|
|
|
2007-06-24 18:09:13 -04:00
|
|
|
mutex_lock(&flash->lock);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/* Wait until finished previous write command. */
|
2008-06-05 09:50:04 -04:00
|
|
|
if (wait_till_ready(flash)) {
|
|
|
|
mutex_unlock(&flash->lock);
|
2006-01-08 16:34:27 -05:00
|
|
|
return 1;
|
2008-06-05 09:50:04 -04:00
|
|
|
}
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
write_enable(flash);
|
|
|
|
|
|
|
|
/* Set up the opcode in the write buffer. */
|
|
|
|
flash->command[0] = OPCODE_PP;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
m25p_addr2cmd(flash, to, flash->command);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
page_offset = to & (flash->page_size - 1);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/* do all the bytes fit onto one page? */
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
if (page_offset + len <= flash->page_size) {
|
2006-01-08 16:34:27 -05:00
|
|
|
t[1].len = len;
|
|
|
|
|
|
|
|
spi_sync(flash->spi, &m);
|
|
|
|
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
*retlen = m.actual_length - m25p_cmdsz(flash);
|
2006-01-08 16:34:27 -05:00
|
|
|
} else {
|
|
|
|
u32 i;
|
|
|
|
|
|
|
|
/* the size of data remaining on the first page */
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
page_size = flash->page_size - page_offset;
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
t[1].len = page_size;
|
|
|
|
spi_sync(flash->spi, &m);
|
|
|
|
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
*retlen = m.actual_length - m25p_cmdsz(flash);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
/* write everything in flash->page_size chunks */
|
2006-01-08 16:34:27 -05:00
|
|
|
for (i = page_size; i < len; i += page_size) {
|
|
|
|
page_size = len - i;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
if (page_size > flash->page_size)
|
|
|
|
page_size = flash->page_size;
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/* write the next page to flash */
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
m25p_addr2cmd(flash, to + i, flash->command);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
t[1].tx_buf = buf + i;
|
|
|
|
t[1].len = page_size;
|
|
|
|
|
|
|
|
wait_till_ready(flash);
|
|
|
|
|
|
|
|
write_enable(flash);
|
|
|
|
|
|
|
|
spi_sync(flash->spi, &m);
|
|
|
|
|
2006-01-08 16:34:29 -05:00
|
|
|
if (retlen)
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
*retlen += m.actual_length - m25p_cmdsz(flash);
|
2007-06-24 18:09:13 -04:00
|
|
|
}
|
|
|
|
}
|
2006-01-08 16:34:27 -05:00
|
|
|
|
2007-06-24 18:09:13 -04:00
|
|
|
mutex_unlock(&flash->lock);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-15 04:23:41 -04:00
|
|
|
static int sst_write(struct mtd_info *mtd, loff_t to, size_t len,
|
|
|
|
size_t *retlen, const u_char *buf)
|
|
|
|
{
|
|
|
|
struct m25p *flash = mtd_to_m25p(mtd);
|
|
|
|
struct spi_transfer t[2];
|
|
|
|
struct spi_message m;
|
|
|
|
size_t actual;
|
|
|
|
int cmd_sz, ret;
|
|
|
|
|
|
|
|
if (retlen)
|
|
|
|
*retlen = 0;
|
|
|
|
|
|
|
|
/* sanity checks */
|
|
|
|
if (!len)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (to + len > flash->mtd.size)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
spi_message_init(&m);
|
|
|
|
memset(t, 0, (sizeof t));
|
|
|
|
|
|
|
|
t[0].tx_buf = flash->command;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
t[0].len = m25p_cmdsz(flash);
|
2009-06-15 04:23:41 -04:00
|
|
|
spi_message_add_tail(&t[0], &m);
|
|
|
|
|
|
|
|
t[1].tx_buf = buf;
|
|
|
|
spi_message_add_tail(&t[1], &m);
|
|
|
|
|
|
|
|
mutex_lock(&flash->lock);
|
|
|
|
|
|
|
|
/* Wait until finished previous write command. */
|
|
|
|
ret = wait_till_ready(flash);
|
|
|
|
if (ret)
|
|
|
|
goto time_out;
|
|
|
|
|
|
|
|
write_enable(flash);
|
|
|
|
|
|
|
|
actual = to % 2;
|
|
|
|
/* Start write from odd address. */
|
|
|
|
if (actual) {
|
|
|
|
flash->command[0] = OPCODE_BP;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
m25p_addr2cmd(flash, to, flash->command);
|
2009-06-15 04:23:41 -04:00
|
|
|
|
|
|
|
/* write one byte. */
|
|
|
|
t[1].len = 1;
|
|
|
|
spi_sync(flash->spi, &m);
|
|
|
|
ret = wait_till_ready(flash);
|
|
|
|
if (ret)
|
|
|
|
goto time_out;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
*retlen += m.actual_length - m25p_cmdsz(flash);
|
2009-06-15 04:23:41 -04:00
|
|
|
}
|
|
|
|
to += actual;
|
|
|
|
|
|
|
|
flash->command[0] = OPCODE_AAI_WP;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
m25p_addr2cmd(flash, to, flash->command);
|
2009-06-15 04:23:41 -04:00
|
|
|
|
|
|
|
/* Write out most of the data here. */
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
cmd_sz = m25p_cmdsz(flash);
|
2009-06-15 04:23:41 -04:00
|
|
|
for (; actual < len - 1; actual += 2) {
|
|
|
|
t[0].len = cmd_sz;
|
|
|
|
/* write two bytes. */
|
|
|
|
t[1].len = 2;
|
|
|
|
t[1].tx_buf = buf + actual;
|
|
|
|
|
|
|
|
spi_sync(flash->spi, &m);
|
|
|
|
ret = wait_till_ready(flash);
|
|
|
|
if (ret)
|
|
|
|
goto time_out;
|
|
|
|
*retlen += m.actual_length - cmd_sz;
|
|
|
|
cmd_sz = 1;
|
|
|
|
to += 2;
|
|
|
|
}
|
|
|
|
write_disable(flash);
|
|
|
|
ret = wait_till_ready(flash);
|
|
|
|
if (ret)
|
|
|
|
goto time_out;
|
|
|
|
|
|
|
|
/* Write out trailing byte if it exists. */
|
|
|
|
if (actual != len) {
|
|
|
|
write_enable(flash);
|
|
|
|
flash->command[0] = OPCODE_BP;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
m25p_addr2cmd(flash, to, flash->command);
|
|
|
|
t[0].len = m25p_cmdsz(flash);
|
2009-06-15 04:23:41 -04:00
|
|
|
t[1].len = 1;
|
|
|
|
t[1].tx_buf = buf + actual;
|
|
|
|
|
|
|
|
spi_sync(flash->spi, &m);
|
|
|
|
ret = wait_till_ready(flash);
|
|
|
|
if (ret)
|
|
|
|
goto time_out;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
*retlen += m.actual_length - m25p_cmdsz(flash);
|
2009-06-15 04:23:41 -04:00
|
|
|
write_disable(flash);
|
|
|
|
}
|
|
|
|
|
|
|
|
time_out:
|
|
|
|
mutex_unlock(&flash->lock);
|
|
|
|
return ret;
|
|
|
|
}
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
/****************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* SPI device driver setup and teardown
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct flash_info {
|
2007-06-24 18:12:35 -04:00
|
|
|
/* JEDEC id zero means "no ID" (most older chips); otherwise it has
|
|
|
|
* a high byte of zero plus three data bytes: the manufacturer id,
|
|
|
|
* then a two byte device id.
|
|
|
|
*/
|
|
|
|
u32 jedec_id;
|
2008-08-11 04:59:15 -04:00
|
|
|
u16 ext_id;
|
2007-06-24 18:12:35 -04:00
|
|
|
|
|
|
|
/* The size listed here is what works with OPCODE_SE, which isn't
|
|
|
|
* necessarily called a "sector" by the vendor.
|
|
|
|
*/
|
2006-01-08 16:34:27 -05:00
|
|
|
unsigned sector_size;
|
2007-06-24 18:12:35 -04:00
|
|
|
u16 n_sectors;
|
|
|
|
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
u16 page_size;
|
|
|
|
u16 addr_width;
|
|
|
|
|
2007-06-24 18:12:35 -04:00
|
|
|
u16 flags;
|
|
|
|
#define SECT_4K 0x01 /* OPCODE_BE_4K works uniformly */
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
#define M25P_NO_ERASE 0x02 /* No erase command needed */
|
2006-01-08 16:34:27 -05:00
|
|
|
};
|
|
|
|
|
2009-10-12 12:24:35 -04:00
|
|
|
#define INFO(_jedec_id, _ext_id, _sector_size, _n_sectors, _flags) \
|
|
|
|
((kernel_ulong_t)&(struct flash_info) { \
|
|
|
|
.jedec_id = (_jedec_id), \
|
|
|
|
.ext_id = (_ext_id), \
|
|
|
|
.sector_size = (_sector_size), \
|
|
|
|
.n_sectors = (_n_sectors), \
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
.page_size = 256, \
|
|
|
|
.addr_width = 3, \
|
2009-10-12 12:24:35 -04:00
|
|
|
.flags = (_flags), \
|
|
|
|
})
|
2007-06-24 18:12:35 -04:00
|
|
|
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
#define CAT25_INFO(_sector_size, _n_sectors, _page_size, _addr_width) \
|
|
|
|
((kernel_ulong_t)&(struct flash_info) { \
|
|
|
|
.sector_size = (_sector_size), \
|
|
|
|
.n_sectors = (_n_sectors), \
|
|
|
|
.page_size = (_page_size), \
|
|
|
|
.addr_width = (_addr_width), \
|
|
|
|
.flags = M25P_NO_ERASE, \
|
|
|
|
})
|
2007-06-24 18:12:35 -04:00
|
|
|
|
|
|
|
/* NOTE: double check command sets and memory organization when you add
|
|
|
|
* more flash chips. This current list focusses on newer chips, which
|
|
|
|
* have been converging on command sets which including JEDEC ID.
|
|
|
|
*/
|
2009-10-12 12:24:35 -04:00
|
|
|
static const struct spi_device_id m25p_ids[] = {
|
2007-06-24 18:12:35 -04:00
|
|
|
/* Atmel -- some are (confusingly) marketed as "DataFlash" */
|
2009-10-12 12:24:35 -04:00
|
|
|
{ "at25fs010", INFO(0x1f6601, 0, 32 * 1024, 4, SECT_4K) },
|
|
|
|
{ "at25fs040", INFO(0x1f6604, 0, 64 * 1024, 8, SECT_4K) },
|
2007-06-24 18:12:35 -04:00
|
|
|
|
2009-10-12 12:24:35 -04:00
|
|
|
{ "at25df041a", INFO(0x1f4401, 0, 64 * 1024, 8, SECT_4K) },
|
|
|
|
{ "at25df641", INFO(0x1f4800, 0, 64 * 1024, 128, SECT_4K) },
|
2007-06-24 18:12:35 -04:00
|
|
|
|
2009-10-12 12:24:35 -04:00
|
|
|
{ "at26f004", INFO(0x1f0400, 0, 64 * 1024, 8, SECT_4K) },
|
|
|
|
{ "at26df081a", INFO(0x1f4501, 0, 64 * 1024, 16, SECT_4K) },
|
|
|
|
{ "at26df161a", INFO(0x1f4601, 0, 64 * 1024, 32, SECT_4K) },
|
|
|
|
{ "at26df321", INFO(0x1f4701, 0, 64 * 1024, 64, SECT_4K) },
|
2007-06-24 18:12:35 -04:00
|
|
|
|
2010-08-04 15:14:25 -04:00
|
|
|
/* EON -- en25pxx */
|
|
|
|
{ "en25p32", INFO(0x1c2016, 0, 64 * 1024, 64, 0) },
|
|
|
|
{ "en25p64", INFO(0x1c2017, 0, 64 * 1024, 128, 0) },
|
|
|
|
|
2010-08-05 10:58:36 -04:00
|
|
|
/* Intel/Numonyx -- xxxs33b */
|
|
|
|
{ "160s33b", INFO(0x898911, 0, 64 * 1024, 32, 0) },
|
|
|
|
{ "320s33b", INFO(0x898912, 0, 64 * 1024, 64, 0) },
|
|
|
|
{ "640s33b", INFO(0x898913, 0, 64 * 1024, 128, 0) },
|
|
|
|
|
2009-05-20 07:07:11 -04:00
|
|
|
/* Macronix */
|
2009-12-05 09:28:00 -05:00
|
|
|
{ "mx25l4005a", INFO(0xc22013, 0, 64 * 1024, 8, SECT_4K) },
|
2010-06-07 14:31:01 -04:00
|
|
|
{ "mx25l8005", INFO(0xc22014, 0, 64 * 1024, 16, 0) },
|
2009-10-12 12:24:35 -04:00
|
|
|
{ "mx25l3205d", INFO(0xc22016, 0, 64 * 1024, 64, 0) },
|
|
|
|
{ "mx25l6405d", INFO(0xc22017, 0, 64 * 1024, 128, 0) },
|
|
|
|
{ "mx25l12805d", INFO(0xc22018, 0, 64 * 1024, 256, 0) },
|
|
|
|
{ "mx25l12855e", INFO(0xc22618, 0, 64 * 1024, 256, 0) },
|
2009-05-20 07:07:11 -04:00
|
|
|
|
2007-06-24 18:12:35 -04:00
|
|
|
/* Spansion -- single (large) sector size only, at least
|
|
|
|
* for the chips listed here (without boot sectors).
|
|
|
|
*/
|
2009-10-12 12:24:35 -04:00
|
|
|
{ "s25sl004a", INFO(0x010212, 0, 64 * 1024, 8, 0) },
|
|
|
|
{ "s25sl008a", INFO(0x010213, 0, 64 * 1024, 16, 0) },
|
|
|
|
{ "s25sl016a", INFO(0x010214, 0, 64 * 1024, 32, 0) },
|
|
|
|
{ "s25sl032a", INFO(0x010215, 0, 64 * 1024, 64, 0) },
|
|
|
|
{ "s25sl064a", INFO(0x010216, 0, 64 * 1024, 128, 0) },
|
|
|
|
{ "s25sl12800", INFO(0x012018, 0x0300, 256 * 1024, 64, 0) },
|
|
|
|
{ "s25sl12801", INFO(0x012018, 0x0301, 64 * 1024, 256, 0) },
|
|
|
|
{ "s25fl129p0", INFO(0x012018, 0x4d00, 256 * 1024, 64, 0) },
|
|
|
|
{ "s25fl129p1", INFO(0x012018, 0x4d01, 64 * 1024, 256, 0) },
|
2007-06-24 18:12:35 -04:00
|
|
|
|
|
|
|
/* SST -- large erase sizes are "overlays", "sectors" are 4K */
|
2009-10-12 12:24:35 -04:00
|
|
|
{ "sst25vf040b", INFO(0xbf258d, 0, 64 * 1024, 8, SECT_4K) },
|
|
|
|
{ "sst25vf080b", INFO(0xbf258e, 0, 64 * 1024, 16, SECT_4K) },
|
|
|
|
{ "sst25vf016b", INFO(0xbf2541, 0, 64 * 1024, 32, SECT_4K) },
|
|
|
|
{ "sst25vf032b", INFO(0xbf254a, 0, 64 * 1024, 64, SECT_4K) },
|
|
|
|
{ "sst25wf512", INFO(0xbf2501, 0, 64 * 1024, 1, SECT_4K) },
|
|
|
|
{ "sst25wf010", INFO(0xbf2502, 0, 64 * 1024, 2, SECT_4K) },
|
|
|
|
{ "sst25wf020", INFO(0xbf2503, 0, 64 * 1024, 4, SECT_4K) },
|
|
|
|
{ "sst25wf040", INFO(0xbf2504, 0, 64 * 1024, 8, SECT_4K) },
|
2007-06-24 18:12:35 -04:00
|
|
|
|
|
|
|
/* ST Microelectronics -- newer production may have feature updates */
|
2009-10-12 12:24:35 -04:00
|
|
|
{ "m25p05", INFO(0x202010, 0, 32 * 1024, 2, 0) },
|
|
|
|
{ "m25p10", INFO(0x202011, 0, 32 * 1024, 4, 0) },
|
|
|
|
{ "m25p20", INFO(0x202012, 0, 64 * 1024, 4, 0) },
|
|
|
|
{ "m25p40", INFO(0x202013, 0, 64 * 1024, 8, 0) },
|
|
|
|
{ "m25p80", INFO(0x202014, 0, 64 * 1024, 16, 0) },
|
|
|
|
{ "m25p16", INFO(0x202015, 0, 64 * 1024, 32, 0) },
|
|
|
|
{ "m25p32", INFO(0x202016, 0, 64 * 1024, 64, 0) },
|
|
|
|
{ "m25p64", INFO(0x202017, 0, 64 * 1024, 128, 0) },
|
|
|
|
{ "m25p128", INFO(0x202018, 0, 256 * 1024, 64, 0) },
|
|
|
|
|
2010-06-22 12:57:34 -04:00
|
|
|
{ "m25p05-nonjedec", INFO(0, 0, 32 * 1024, 2, 0) },
|
|
|
|
{ "m25p10-nonjedec", INFO(0, 0, 32 * 1024, 4, 0) },
|
|
|
|
{ "m25p20-nonjedec", INFO(0, 0, 64 * 1024, 4, 0) },
|
|
|
|
{ "m25p40-nonjedec", INFO(0, 0, 64 * 1024, 8, 0) },
|
|
|
|
{ "m25p80-nonjedec", INFO(0, 0, 64 * 1024, 16, 0) },
|
|
|
|
{ "m25p16-nonjedec", INFO(0, 0, 64 * 1024, 32, 0) },
|
|
|
|
{ "m25p32-nonjedec", INFO(0, 0, 64 * 1024, 64, 0) },
|
|
|
|
{ "m25p64-nonjedec", INFO(0, 0, 64 * 1024, 128, 0) },
|
|
|
|
{ "m25p128-nonjedec", INFO(0, 0, 256 * 1024, 64, 0) },
|
|
|
|
|
2009-10-12 12:24:35 -04:00
|
|
|
{ "m45pe10", INFO(0x204011, 0, 64 * 1024, 2, 0) },
|
|
|
|
{ "m45pe80", INFO(0x204014, 0, 64 * 1024, 16, 0) },
|
|
|
|
{ "m45pe16", INFO(0x204015, 0, 64 * 1024, 32, 0) },
|
|
|
|
|
|
|
|
{ "m25pe80", INFO(0x208014, 0, 64 * 1024, 16, 0) },
|
|
|
|
{ "m25pe16", INFO(0x208015, 0, 64 * 1024, 32, SECT_4K) },
|
2007-06-24 18:12:35 -04:00
|
|
|
|
2007-06-28 17:38:38 -04:00
|
|
|
/* Winbond -- w25x "blocks" are 64K, "sectors" are 4KiB */
|
2009-10-12 12:24:35 -04:00
|
|
|
{ "w25x10", INFO(0xef3011, 0, 64 * 1024, 2, SECT_4K) },
|
|
|
|
{ "w25x20", INFO(0xef3012, 0, 64 * 1024, 4, SECT_4K) },
|
|
|
|
{ "w25x40", INFO(0xef3013, 0, 64 * 1024, 8, SECT_4K) },
|
|
|
|
{ "w25x80", INFO(0xef3014, 0, 64 * 1024, 16, SECT_4K) },
|
|
|
|
{ "w25x16", INFO(0xef3015, 0, 64 * 1024, 32, SECT_4K) },
|
|
|
|
{ "w25x32", INFO(0xef3016, 0, 64 * 1024, 64, SECT_4K) },
|
2010-08-04 15:14:27 -04:00
|
|
|
{ "w25q32", INFO(0xef4016, 0, 64 * 1024, 64, SECT_4K) },
|
2009-10-12 12:24:35 -04:00
|
|
|
{ "w25x64", INFO(0xef3017, 0, 64 * 1024, 128, SECT_4K) },
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
|
|
|
|
/* Catalyst / On Semiconductor -- non-JEDEC */
|
|
|
|
{ "cat25c11", CAT25_INFO( 16, 8, 16, 1) },
|
|
|
|
{ "cat25c03", CAT25_INFO( 32, 8, 16, 2) },
|
|
|
|
{ "cat25c09", CAT25_INFO( 128, 8, 32, 2) },
|
|
|
|
{ "cat25c17", CAT25_INFO( 256, 8, 32, 2) },
|
|
|
|
{ "cat25128", CAT25_INFO(2048, 8, 64, 2) },
|
2009-10-12 12:24:35 -04:00
|
|
|
{ },
|
2006-01-08 16:34:27 -05:00
|
|
|
};
|
2009-10-12 12:24:35 -04:00
|
|
|
MODULE_DEVICE_TABLE(spi, m25p_ids);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
2009-10-12 12:24:35 -04:00
|
|
|
static const struct spi_device_id *__devinit jedec_probe(struct spi_device *spi)
|
2007-06-24 18:12:35 -04:00
|
|
|
{
|
|
|
|
int tmp;
|
|
|
|
u8 code = OPCODE_RDID;
|
2008-09-16 02:14:12 -04:00
|
|
|
u8 id[5];
|
2007-06-24 18:12:35 -04:00
|
|
|
u32 jedec;
|
2008-08-11 04:59:15 -04:00
|
|
|
u16 ext_jedec;
|
2007-06-24 18:12:35 -04:00
|
|
|
struct flash_info *info;
|
|
|
|
|
|
|
|
/* JEDEC also defines an optional "extended device information"
|
|
|
|
* string for after vendor-specific data, after the three bytes
|
|
|
|
* we use here. Supporting some chips might require using it.
|
|
|
|
*/
|
2008-09-16 02:14:12 -04:00
|
|
|
tmp = spi_write_then_read(spi, &code, 1, id, 5);
|
2007-06-24 18:12:35 -04:00
|
|
|
if (tmp < 0) {
|
|
|
|
DEBUG(MTD_DEBUG_LEVEL0, "%s: error %d reading JEDEC ID\n",
|
2008-12-23 05:00:14 -05:00
|
|
|
dev_name(&spi->dev), tmp);
|
2010-06-22 12:57:42 -04:00
|
|
|
return ERR_PTR(tmp);
|
2007-06-24 18:12:35 -04:00
|
|
|
}
|
|
|
|
jedec = id[0];
|
|
|
|
jedec = jedec << 8;
|
|
|
|
jedec |= id[1];
|
|
|
|
jedec = jedec << 8;
|
|
|
|
jedec |= id[2];
|
|
|
|
|
2008-08-11 04:59:15 -04:00
|
|
|
ext_jedec = id[3] << 8 | id[4];
|
|
|
|
|
2009-10-12 12:24:35 -04:00
|
|
|
for (tmp = 0; tmp < ARRAY_SIZE(m25p_ids) - 1; tmp++) {
|
|
|
|
info = (void *)m25p_ids[tmp].driver_data;
|
2008-11-26 05:23:25 -05:00
|
|
|
if (info->jedec_id == jedec) {
|
2008-11-26 05:23:35 -05:00
|
|
|
if (info->ext_id != 0 && info->ext_id != ext_jedec)
|
2008-08-11 04:59:15 -04:00
|
|
|
continue;
|
2009-10-12 12:24:35 -04:00
|
|
|
return &m25p_ids[tmp];
|
2008-11-26 05:23:25 -05:00
|
|
|
}
|
2007-06-24 18:12:35 -04:00
|
|
|
}
|
2010-06-22 12:57:42 -04:00
|
|
|
return ERR_PTR(-ENODEV);
|
2007-06-24 18:12:35 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-01-08 16:34:27 -05:00
|
|
|
/*
|
|
|
|
* board specific setup should have ensured the SPI clock used here
|
|
|
|
* matches what the READ command supports, at least until this driver
|
|
|
|
* understands FAST_READ (for clocks over 25 MHz).
|
|
|
|
*/
|
|
|
|
static int __devinit m25p_probe(struct spi_device *spi)
|
|
|
|
{
|
2009-10-12 12:24:38 -04:00
|
|
|
const struct spi_device_id *id = spi_get_device_id(spi);
|
2006-01-08 16:34:27 -05:00
|
|
|
struct flash_platform_data *data;
|
|
|
|
struct m25p *flash;
|
|
|
|
struct flash_info *info;
|
|
|
|
unsigned i;
|
|
|
|
|
|
|
|
/* Platform data helps sort out which chip type we have, as
|
2007-06-24 18:12:35 -04:00
|
|
|
* well as how this board partitions it. If we don't have
|
|
|
|
* a chip ID, try the JEDEC id commands; they'll work for most
|
|
|
|
* newer chips, even if we don't recognize the particular chip.
|
2006-01-08 16:34:27 -05:00
|
|
|
*/
|
|
|
|
data = spi->dev.platform_data;
|
2007-06-24 18:12:35 -04:00
|
|
|
if (data && data->type) {
|
2009-10-12 12:24:38 -04:00
|
|
|
const struct spi_device_id *plat_id;
|
2006-01-08 16:34:27 -05:00
|
|
|
|
2009-10-12 12:24:35 -04:00
|
|
|
for (i = 0; i < ARRAY_SIZE(m25p_ids) - 1; i++) {
|
2009-10-12 12:24:38 -04:00
|
|
|
plat_id = &m25p_ids[i];
|
|
|
|
if (strcmp(data->type, plat_id->name))
|
2009-10-12 12:24:35 -04:00
|
|
|
continue;
|
|
|
|
break;
|
2007-06-24 18:12:35 -04:00
|
|
|
}
|
|
|
|
|
2009-10-12 12:24:38 -04:00
|
|
|
if (plat_id)
|
|
|
|
id = plat_id;
|
|
|
|
else
|
|
|
|
dev_warn(&spi->dev, "unrecognized id %s\n", data->type);
|
2009-10-12 12:24:35 -04:00
|
|
|
}
|
2007-06-24 18:12:35 -04:00
|
|
|
|
2009-10-12 12:24:38 -04:00
|
|
|
info = (void *)id->driver_data;
|
|
|
|
|
|
|
|
if (info->jedec_id) {
|
|
|
|
const struct spi_device_id *jid;
|
|
|
|
|
|
|
|
jid = jedec_probe(spi);
|
2010-06-22 12:57:42 -04:00
|
|
|
if (IS_ERR(jid)) {
|
|
|
|
return PTR_ERR(jid);
|
2009-10-12 12:24:38 -04:00
|
|
|
} else if (jid != id) {
|
|
|
|
/*
|
|
|
|
* JEDEC knows better, so overwrite platform ID. We
|
|
|
|
* can't trust partitions any longer, but we'll let
|
|
|
|
* mtd apply them anyway, since some partitions may be
|
|
|
|
* marked read-only, and we don't want to lose that
|
|
|
|
* information, even if it's not 100% accurate.
|
|
|
|
*/
|
|
|
|
dev_warn(&spi->dev, "found %s, expected %s\n",
|
|
|
|
jid->name, id->name);
|
|
|
|
id = jid;
|
|
|
|
info = (void *)jid->driver_data;
|
|
|
|
}
|
|
|
|
}
|
2006-01-08 16:34:27 -05:00
|
|
|
|
2006-12-06 23:33:17 -05:00
|
|
|
flash = kzalloc(sizeof *flash, GFP_KERNEL);
|
2006-01-08 16:34:27 -05:00
|
|
|
if (!flash)
|
|
|
|
return -ENOMEM;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
flash->command = kmalloc(MAX_CMD_SIZE + FAST_READ_DUMMY_BYTE, GFP_KERNEL);
|
2009-10-28 09:21:37 -04:00
|
|
|
if (!flash->command) {
|
|
|
|
kfree(flash);
|
|
|
|
return -ENOMEM;
|
|
|
|
}
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
flash->spi = spi;
|
2007-06-24 18:09:13 -04:00
|
|
|
mutex_init(&flash->lock);
|
2006-01-08 16:34:27 -05:00
|
|
|
dev_set_drvdata(&spi->dev, flash);
|
|
|
|
|
2008-07-04 02:54:42 -04:00
|
|
|
/*
|
2010-08-05 10:58:36 -04:00
|
|
|
* Atmel, SST and Intel/Numonyx serial flash tend to power
|
2009-09-24 15:46:22 -04:00
|
|
|
* up with the software protection bits set
|
2008-07-04 02:54:42 -04:00
|
|
|
*/
|
|
|
|
|
2009-09-24 15:46:22 -04:00
|
|
|
if (info->jedec_id >> 16 == 0x1f ||
|
2010-08-05 10:58:36 -04:00
|
|
|
info->jedec_id >> 16 == 0x89 ||
|
2009-09-24 15:46:22 -04:00
|
|
|
info->jedec_id >> 16 == 0xbf) {
|
2008-07-04 02:54:42 -04:00
|
|
|
write_enable(flash);
|
|
|
|
write_sr(flash, 0);
|
|
|
|
}
|
|
|
|
|
2007-06-24 18:12:35 -04:00
|
|
|
if (data && data->name)
|
2006-01-08 16:34:27 -05:00
|
|
|
flash->mtd.name = data->name;
|
|
|
|
else
|
2008-12-23 05:00:14 -05:00
|
|
|
flash->mtd.name = dev_name(&spi->dev);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
flash->mtd.type = MTD_NORFLASH;
|
2006-06-14 11:53:44 -04:00
|
|
|
flash->mtd.writesize = 1;
|
2006-01-08 16:34:27 -05:00
|
|
|
flash->mtd.flags = MTD_CAP_NORFLASH;
|
|
|
|
flash->mtd.size = info->sector_size * info->n_sectors;
|
|
|
|
flash->mtd.erase = m25p80_erase;
|
|
|
|
flash->mtd.read = m25p80_read;
|
2009-06-15 04:23:41 -04:00
|
|
|
|
|
|
|
/* sst flash chips use AAI word program */
|
|
|
|
if (info->jedec_id >> 16 == 0xbf)
|
|
|
|
flash->mtd.write = sst_write;
|
|
|
|
else
|
|
|
|
flash->mtd.write = m25p80_write;
|
2006-01-08 16:34:27 -05:00
|
|
|
|
2007-06-24 18:12:35 -04:00
|
|
|
/* prefer "small sector" erase if possible */
|
|
|
|
if (info->flags & SECT_4K) {
|
|
|
|
flash->erase_opcode = OPCODE_BE_4K;
|
|
|
|
flash->mtd.erasesize = 4096;
|
|
|
|
} else {
|
|
|
|
flash->erase_opcode = OPCODE_SE;
|
|
|
|
flash->mtd.erasesize = info->sector_size;
|
|
|
|
}
|
|
|
|
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
if (info->flags & M25P_NO_ERASE)
|
|
|
|
flash->mtd.flags |= MTD_NO_ERASE;
|
|
|
|
|
2009-03-26 03:42:50 -04:00
|
|
|
flash->mtd.dev.parent = &spi->dev;
|
mtd: m25p80: Add support for CAT25xxx serial EEPROMs
CAT25 chips (as manufactured by On Semiconductor, previously Catalyst
Semiconductor) are similar to the original M25Px0 chips, except:
- Address width can vary (1-2 bytes, in contrast to 3 bytes in M25P
chips). So, implement convenient m25p_addr2cmd() and m25p_cmdsz()
calls, and place address width information into flash_info struct;
- Page size can vary, therefore we shouldn't hardcode it, so get rid
of FLASH_PAGESIZE definition, and place the page size information
into flash_info struct;
- CAT25 EEPROMs don't need to be erased, so add NO_ERASE flag, and
propagate it to the mtd subsystem.
[dwmw2: Fix up for conflicts with DMA safety patch]
Signed-off-by: Anton Vorontsov <avorontsov@ru.mvista.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
2009-10-12 12:24:40 -04:00
|
|
|
flash->page_size = info->page_size;
|
|
|
|
flash->addr_width = info->addr_width;
|
2009-03-26 03:42:50 -04:00
|
|
|
|
2009-10-12 12:24:35 -04:00
|
|
|
dev_info(&spi->dev, "%s (%lld Kbytes)\n", id->name,
|
2008-12-18 07:10:05 -05:00
|
|
|
(long long)flash->mtd.size >> 10);
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
DEBUG(MTD_DEBUG_LEVEL2,
|
2008-12-18 07:10:05 -05:00
|
|
|
"mtd .name = %s, .size = 0x%llx (%lldMiB) "
|
2007-06-28 17:38:38 -04:00
|
|
|
".erasesize = 0x%.8x (%uKiB) .numeraseregions = %d\n",
|
2006-01-08 16:34:27 -05:00
|
|
|
flash->mtd.name,
|
2008-12-18 07:10:05 -05:00
|
|
|
(long long)flash->mtd.size, (long long)(flash->mtd.size >> 20),
|
2006-01-08 16:34:27 -05:00
|
|
|
flash->mtd.erasesize, flash->mtd.erasesize / 1024,
|
|
|
|
flash->mtd.numeraseregions);
|
|
|
|
|
|
|
|
if (flash->mtd.numeraseregions)
|
|
|
|
for (i = 0; i < flash->mtd.numeraseregions; i++)
|
|
|
|
DEBUG(MTD_DEBUG_LEVEL2,
|
2008-12-18 07:10:05 -05:00
|
|
|
"mtd.eraseregions[%d] = { .offset = 0x%llx, "
|
2007-06-28 17:38:38 -04:00
|
|
|
".erasesize = 0x%.8x (%uKiB), "
|
2006-01-08 16:34:27 -05:00
|
|
|
".numblocks = %d }\n",
|
2008-12-18 07:10:05 -05:00
|
|
|
i, (long long)flash->mtd.eraseregions[i].offset,
|
2006-01-08 16:34:27 -05:00
|
|
|
flash->mtd.eraseregions[i].erasesize,
|
|
|
|
flash->mtd.eraseregions[i].erasesize / 1024,
|
|
|
|
flash->mtd.eraseregions[i].numblocks);
|
|
|
|
|
|
|
|
|
|
|
|
/* partitions should match sector boundaries; and it may be good to
|
|
|
|
* use readonly partitions for writeprotected sectors (BP2..BP0).
|
|
|
|
*/
|
|
|
|
if (mtd_has_partitions()) {
|
|
|
|
struct mtd_partition *parts = NULL;
|
|
|
|
int nr_parts = 0;
|
|
|
|
|
2009-03-04 15:01:41 -05:00
|
|
|
if (mtd_has_cmdlinepart()) {
|
|
|
|
static const char *part_probes[]
|
|
|
|
= { "cmdlinepart", NULL, };
|
2006-01-08 16:34:27 -05:00
|
|
|
|
2009-03-04 15:01:41 -05:00
|
|
|
nr_parts = parse_mtd_partitions(&flash->mtd,
|
|
|
|
part_probes, &parts, 0);
|
|
|
|
}
|
2006-01-08 16:34:27 -05:00
|
|
|
|
|
|
|
if (nr_parts <= 0 && data && data->parts) {
|
|
|
|
parts = data->parts;
|
|
|
|
nr_parts = data->nr_parts;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nr_parts > 0) {
|
2007-06-24 18:12:35 -04:00
|
|
|
for (i = 0; i < nr_parts; i++) {
|
2006-01-08 16:34:27 -05:00
|
|
|
DEBUG(MTD_DEBUG_LEVEL2, "partitions[%d] = "
|
2008-12-18 07:10:05 -05:00
|
|
|
"{.name = %s, .offset = 0x%llx, "
|
|
|
|
".size = 0x%llx (%lldKiB) }\n",
|
2007-06-24 18:12:35 -04:00
|
|
|
i, parts[i].name,
|
2008-12-18 07:10:05 -05:00
|
|
|
(long long)parts[i].offset,
|
|
|
|
(long long)parts[i].size,
|
|
|
|
(long long)(parts[i].size >> 10));
|
2006-01-08 16:34:27 -05:00
|
|
|
}
|
|
|
|
flash->partitioned = 1;
|
|
|
|
return add_mtd_partitions(&flash->mtd, parts, nr_parts);
|
|
|
|
}
|
2009-08-06 18:18:37 -04:00
|
|
|
} else if (data && data->nr_parts)
|
2006-01-08 16:34:27 -05:00
|
|
|
dev_warn(&spi->dev, "ignoring %d default partitions on %s\n",
|
|
|
|
data->nr_parts, data->name);
|
|
|
|
|
|
|
|
return add_mtd_device(&flash->mtd) == 1 ? -ENODEV : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int __devexit m25p_remove(struct spi_device *spi)
|
|
|
|
{
|
|
|
|
struct m25p *flash = dev_get_drvdata(&spi->dev);
|
|
|
|
int status;
|
|
|
|
|
|
|
|
/* Clean up MTD stuff. */
|
|
|
|
if (mtd_has_partitions() && flash->partitioned)
|
|
|
|
status = del_mtd_partitions(&flash->mtd);
|
|
|
|
else
|
|
|
|
status = del_mtd_device(&flash->mtd);
|
2009-10-28 09:21:37 -04:00
|
|
|
if (status == 0) {
|
|
|
|
kfree(flash->command);
|
2006-01-08 16:34:27 -05:00
|
|
|
kfree(flash);
|
2009-10-28 09:21:37 -04:00
|
|
|
}
|
2006-01-08 16:34:27 -05:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static struct spi_driver m25p80_driver = {
|
|
|
|
.driver = {
|
|
|
|
.name = "m25p80",
|
|
|
|
.bus = &spi_bus_type,
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
},
|
2009-10-12 12:24:35 -04:00
|
|
|
.id_table = m25p_ids,
|
2006-01-08 16:34:27 -05:00
|
|
|
.probe = m25p_probe,
|
|
|
|
.remove = __devexit_p(m25p_remove),
|
2007-06-24 18:12:35 -04:00
|
|
|
|
|
|
|
/* REVISIT: many of these chips have deep power-down modes, which
|
|
|
|
* should clearly be entered on suspend() to minimize power use.
|
|
|
|
* And also when they're otherwise idle...
|
|
|
|
*/
|
2006-01-08 16:34:27 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-06-10 20:23:33 -04:00
|
|
|
static int __init m25p80_init(void)
|
2006-01-08 16:34:27 -05:00
|
|
|
{
|
|
|
|
return spi_register_driver(&m25p80_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-06-10 20:23:33 -04:00
|
|
|
static void __exit m25p80_exit(void)
|
2006-01-08 16:34:27 -05:00
|
|
|
{
|
|
|
|
spi_unregister_driver(&m25p80_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
module_init(m25p80_init);
|
|
|
|
module_exit(m25p80_exit);
|
|
|
|
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
MODULE_AUTHOR("Mike Lavender");
|
|
|
|
MODULE_DESCRIPTION("MTD SPI driver for ST M25Pxx flash chips");
|