961 lines
27 KiB
C
Executable File
961 lines
27 KiB
C
Executable File
/*
|
|
* device driver for Serial NOR Flash on Novatek platform
|
|
* The serial nor interface is largely based on drivers/mtd/m25p80.c,
|
|
* however the SPI interface has been replaced by Novatek.
|
|
*
|
|
* Copyright © 2017 Novatek Microelectronics.
|
|
*
|
|
* This file is licensed under the terms of the GNU General Public
|
|
* License version 2. This program is licensed "as is" without any
|
|
* warranty of any kind, whether express or implied.
|
|
*/
|
|
|
|
#include <linux/clk.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/device.h>
|
|
#include <linux/err.h>
|
|
#include <linux/errno.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/io.h>
|
|
#include <linux/ioport.h>
|
|
#include <linux/jiffies.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/param.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/pm.h>
|
|
#include <linux/mtd/mtd.h>
|
|
#include <linux/mtd/partitions.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/wait.h>
|
|
#include <linux/of.h>
|
|
#include <linux/of_address.h>
|
|
#include <plat/nvt_flash.h>
|
|
#include <plat/nvt-sramctl.h>
|
|
#include "../nvt_flash_spi/nvt_flash_spi_reg.h"
|
|
#include "../nvt_flash_spi/nvt_flash_spi_int.h"
|
|
#include "ext_flash_table.h"
|
|
|
|
#define NOR_VERSION "1.07.122"
|
|
|
|
/* SMI clock rate */
|
|
#define NVT_FLASH_MAX_CLOCK_FREQ 50000000 /* 50 MHz */
|
|
|
|
/* MAX time out to safely come out of a erase or write busy conditions */
|
|
#define NVT_FLASH_PROBE_TIMEOUT (HZ / 10)
|
|
#define NVT_FLASH_MAX_TIME_OUT (3 * HZ)
|
|
|
|
/* timeout for command completion */
|
|
#define NVT_FLASH_CMD_TIMEOUT (HZ / 10)
|
|
|
|
#define ERASE_MASK_64K 0xFFFF
|
|
#define ERASE_64K 0x10000
|
|
|
|
#ifndef CONFIG_MTD_EXTERNAL_FLASH_TABLE
|
|
static struct flash_device flash_devices[] = {
|
|
FLASH_ID("st m25p16" , 0xd8, 0x00152020, 0x100, 0x10000, 0x200000, 0, 0),
|
|
FLASH_ID("st m25p32" , 0xd8, 0x00162020, 0x100, 0x10000, 0x400000, 0, 0),
|
|
FLASH_ID("st m25p64" , 0xd8, 0x00172020, 0x100, 0x10000, 0x800000, 0, 0),
|
|
FLASH_ID("st m25p128" , 0xd8, 0x00182020, 0x100, 0x40000, 0x1000000, 0, 0),
|
|
FLASH_ID("st m25p05" , 0xd8, 0x00102020, 0x80 , 0x8000 , 0x10000, 0, 0),
|
|
FLASH_ID("st m25p10" , 0xd8, 0x00112020, 0x80 , 0x8000 , 0x20000, 0, 0),
|
|
FLASH_ID("st m25p20" , 0xd8, 0x00122020, 0x100, 0x10000, 0x40000, 0, 0),
|
|
FLASH_ID("st m25p40" , 0xd8, 0x00132020, 0x100, 0x10000, 0x80000, 0, 0),
|
|
FLASH_ID("st m25p80" , 0xd8, 0x00142020, 0x100, 0x10000, 0x100000, 0, 0),
|
|
FLASH_ID("st m45pe10" , 0xd8, 0x00114020, 0x100, 0x10000, 0x20000, 0, 0),
|
|
FLASH_ID("st m45pe20" , 0xd8, 0x00124020, 0x100, 0x10000, 0x40000, 0, 0),
|
|
FLASH_ID("st m45pe40" , 0xd8, 0x00134020, 0x100, 0x10000, 0x80000, 0, 0),
|
|
FLASH_ID("st m45pe80" , 0xd8, 0x00144020, 0x100, 0x10000, 0x100000, 0, 0),
|
|
FLASH_ID("sp s25fl004" , 0xd8, 0x00120201, 0x100, 0x10000, 0x80000, 0, 0),
|
|
FLASH_ID("sp s25fl008" , 0xd8, 0x00130201, 0x100, 0x10000, 0x100000, 0, 0),
|
|
FLASH_ID("sp s25fl016" , 0xd8, 0x00140201, 0x100, 0x10000, 0x200000, 0, 0),
|
|
FLASH_ID("sp s25fl032" , 0xd8, 0x00150201, 0x100, 0x10000, 0x400000, 0, 0),
|
|
FLASH_ID("sp s25fl064" , 0xd8, 0x00160201, 0x100, 0x10000, 0x800000, 0, 0),
|
|
FLASH_ID("sp s25fl128" , 0xd8, 0x00182001, 0x100, 0x10000, 0x1000000, 0, 0),
|
|
FLASH_ID("atmel 25f512" , 0x52, 0x0065001F, 0x80 , 0x8000 , 0x10000, 0, 0),
|
|
FLASH_ID("atmel 25f1024" , 0x52, 0x0060001F, 0x100, 0x8000 , 0x20000, 0, 0),
|
|
FLASH_ID("atmel 25f2048" , 0x52, 0x0063001F, 0x100, 0x10000, 0x40000, 0, 0),
|
|
FLASH_ID("atmel 25f4096" , 0x52, 0x0064001F, 0x100, 0x10000, 0x80000, 0, 0),
|
|
FLASH_ID("atmel 25fs040" , 0xd7, 0x0004661F, 0x100, 0x10000, 0x80000, 0, 0),
|
|
FLASH_ID("mac 25l512" , 0xd8, 0x001020C2, 0x010, 0x10000, 0x10000, 0, 0),
|
|
FLASH_ID("mac 25l1005" , 0xd8, 0x001120C2, 0x010, 0x10000, 0x20000, 0, 0),
|
|
FLASH_ID("mac 25l2005" , 0xd8, 0x001220C2, 0x010, 0x10000, 0x40000, 0, 0),
|
|
FLASH_ID("mac 25l4005" , 0xd8, 0x001320C2, 0x010, 0x10000, 0x80000, 0, 0),
|
|
FLASH_ID("mac 25l4005a" , 0xd8, 0x001320C2, 0x010, 0x10000, 0x80000, 0, 0),
|
|
FLASH_ID("mac 25l8005" , 0xd8, 0x001420C2, 0x010, 0x10000, 0x100000, 0, 0),
|
|
FLASH_ID("mac 25l1605" , 0xd8, 0x001520C2, 0x100, 0x10000, 0x200000, 0, 0),
|
|
FLASH_ID("mac 25l1605a" , 0xd8, 0x001520C2, 0x010, 0x10000, 0x200000, 0, 0),
|
|
FLASH_ID("mac 25l3205" , 0xd8, 0x001620C2, 0x100, 0x10000, 0x400000, 0, 0),
|
|
FLASH_ID("mac 25l3205a" , 0xd8, 0x001620C2, 0x100, 0x10000, 0x400000, 0, 0),
|
|
FLASH_ID("mac 25l6405" , 0xd8, 0x001720C2, 0x100, 0x10000, 0x800000, 0, SPI_NOR_DUAL_READ),
|
|
FLASH_ID("mx 25l12835f" , 0xd8, 0x001820C2, 0x100, 0x10000, 0x1000000, 0, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("mx 25l25645g" , 0xd8, 0x001920C2, 0x100, 0x10000, 0x2000000, 0, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("mx 25l51245g" , 0xd8, 0x001A20C2, 0x100, 0x10000, 0x4000000, 0, 0),
|
|
FLASH_ID("wb W25Q32FV" , 0xd8, 0x001640EF, 0x100, 0x10000, 0x400000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("wb W25Q64FV" , 0xd8, 0x001740EF, 0x100, 0x10000, 0x800000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("wb W25Q128BV" , 0xd8, 0x001840EF, 0x100, 0x10000, 0x1000000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("wb W25Q256BV" , 0xd8, 0x001940EF, 0x100, 0x10000, 0x2000000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("wb W25Q256BV" , 0xd8, 0x001970EF, 0x100, 0x10000, 0x2000000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("wb W25Q512JV" , 0xd8, 0x002040EF, 0x100, 0x10000, 0x4000000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("gd W25Q64C" , 0xd8, 0x001740C8, 0x100, 0x10000, 0x800000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("gd W25Q127C" , 0xd8, 0x001840C8, 0x100, 0x10000, 0x1000000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("gd W25Q256C" , 0xd8, 0x001940C8, 0x100, 0x10000, 0x2000000, 0, SPI_NOR_DUAL_READ),
|
|
FLASH_ID("eon en25qh64" , 0xd8, 0x0017701C, 0x100, 0x10000, 0x800000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("eon en25qh128" , 0xd8, 0x0018701C, 0x100, 0x10000, 0x1000000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
FLASH_ID("xmc xm25qh128a", 0xd8, 0x00187020, 0x100, 0x10000, 0x1000000, WR_QPP, SPI_NOR_QUAD_READ),
|
|
};
|
|
#endif
|
|
|
|
static struct flash_device flash_std_devices[] = {
|
|
FLASH_ID("STDR04FW" , 0xd8, 0x0013FFFF, 0x100, 0x10000, 0x80000, 0, SPI_NOR_DUAL_READ),
|
|
FLASH_ID("STDR08FW" , 0xd8, 0x0014FFFF, 0x100, 0x10000, 0x100000, 0, SPI_NOR_DUAL_READ),
|
|
FLASH_ID("STDR16FW" , 0xd8, 0x0015FFFF, 0x100, 0x10000, 0x200000, 0, SPI_NOR_DUAL_READ),
|
|
FLASH_ID("STDR32FW" , 0xd8, 0x0016FFFF, 0x100, 0x10000, 0x400000, 0, SPI_NOR_DUAL_READ),
|
|
FLASH_ID("STDR64FW" , 0xd8, 0x0017FFFF, 0x100, 0x10000, 0x800000, 0, SPI_NOR_DUAL_READ),
|
|
FLASH_ID("STDR128FW" , 0xd8, 0x0018FFFF, 0x100, 0x10000, 0x1000000, 0, SPI_NOR_DUAL_READ),
|
|
FLASH_ID("STDR256FW" , 0xd8, 0x0019FFFF, 0x100, 0x10000, 0x2000000, 0, SPI_NOR_DUAL_READ),
|
|
FLASH_ID("STDR512FW" , 0xd8, 0x001AFFFF, 0x100, 0x10000, 0x4000000, 0, SPI_NOR_DUAL_READ),
|
|
};
|
|
|
|
|
|
/* Define spear specific structures */
|
|
|
|
struct nvt_spinor_flash;
|
|
u8 std_path = 0;
|
|
|
|
/**
|
|
* struct nvt_flash - Structure for NVT Flash Device
|
|
*
|
|
* @clk: functional clock
|
|
* @status: current status register of NVT Flash.
|
|
* @clk_rate: functional clock rate of NVT Flash (default: NVT_FLASH_MAX_CLOCK_FREQ)
|
|
* @lock: lock to prevent parallel access of NVT Flash.
|
|
* @info: drv_nand_dev_info of NVT Flash.
|
|
* @pdev: platform device
|
|
* @flash: separate structure for each Serial NOR-flash attached to NVT Flash.
|
|
*/
|
|
struct nvt_flash {
|
|
struct clk *clk;
|
|
unsigned long clk_rate;
|
|
struct mutex lock;
|
|
struct drv_nand_dev_info *info;
|
|
struct platform_device *pdev;
|
|
struct nvt_spinor_flash *flash;
|
|
};
|
|
|
|
/**
|
|
* struct nvt_spinor_flash - Structure for Serial NOR Flash
|
|
*
|
|
* @dev_id: Device ID of NOR-flash.
|
|
* @mtd: MTD info for each NOR-flash.
|
|
* @parts: Partition info of NOR-flash.
|
|
* @page_size: Page size of NOR-flash.
|
|
* @base_addr: Base address of NOR-flash.
|
|
* @erase_cmd: erase command may vary on different flash types
|
|
*/
|
|
struct nvt_spinor_flash {
|
|
u32 dev_id;
|
|
struct mtd_info mtd;
|
|
struct mtd_partition *parts;
|
|
u32 page_size;
|
|
u8 erase_cmd;
|
|
void __iomem *base_addr;
|
|
};
|
|
|
|
static void nvt_spinor_check_fastboot(void)
|
|
{
|
|
u32 m_fastboot = 0x0, preload = 0x0;
|
|
struct device_node* of_node = of_find_node_by_path("/fastboot");
|
|
int ret = 0;
|
|
|
|
if (of_node) {
|
|
of_property_read_u32(of_node, "enable", &m_fastboot);
|
|
}
|
|
|
|
of_node = of_find_node_by_path("/fastboot/spi-nor");
|
|
|
|
if (of_node) {
|
|
of_property_read_u32(of_node, "preload", &preload);
|
|
}
|
|
|
|
if (m_fastboot && preload) {
|
|
ret = nvt_check_preload_finish();
|
|
if (ret != 1)
|
|
pr_err("error with waiting preload %d\n", ret);
|
|
}
|
|
}
|
|
|
|
static inline struct nvt_spinor_flash *get_flash_data(struct mtd_info *mtd)
|
|
{
|
|
return container_of(mtd, struct nvt_spinor_flash, mtd);
|
|
}
|
|
|
|
/**
|
|
* nvt_flash_int_handler - NVT Flash Interrupt Handler.
|
|
* @irq: irq number
|
|
* @dev_id: structure of NVT Flash device, embedded in dev_id.
|
|
*
|
|
* The handler clears all interrupt conditions and records the status in
|
|
* dev->status which is used by the driver later.
|
|
*/
|
|
static irqreturn_t nvt_flash_int_handler(int irq, void *dev_id)
|
|
{
|
|
struct nvt_flash *dev = dev_id;
|
|
|
|
dev->info->nand_int_status = NAND_GETREG(dev->info, NAND_CTRL_STS_REG_OFS);
|
|
|
|
if (dev->info->nand_int_status) {
|
|
NAND_SETREG(dev->info, NAND_CTRL_STS_REG_OFS, dev->info->nand_int_status);
|
|
complete(&dev->info->cmd_complete);
|
|
return IRQ_HANDLED;
|
|
} else
|
|
return IRQ_NONE;
|
|
}
|
|
|
|
/**
|
|
* nvt_flash_hw_init - initializes the NVT Flash controller.
|
|
* @dev: structure of NVT Flash device
|
|
*
|
|
* this routine initializes the nvt flash controller wit the default values
|
|
*/
|
|
static void nvt_flash_hw_init(struct nvt_flash *dev)
|
|
{
|
|
mutex_lock(&dev->lock);
|
|
|
|
NAND_SETREG(dev->info, NAND_TIME0_REG_OFS, 0x06002222);
|
|
NAND_SETREG(dev->info, NAND_TIME1_REG_OFS, 0x7f0f);
|
|
|
|
clk_set_rate(dev->clk, dev->info->flash_freq);
|
|
|
|
/* Release SRAM */
|
|
nvt_disable_sram_shutdown(NAND_SD);
|
|
|
|
nand_host_set_nand_type(dev->info, NANDCTRL_SPI_NOR_TYPE);
|
|
|
|
nand_host_settiming2(dev->info, 0x5F51);
|
|
|
|
nand_phy_config(dev->info);
|
|
|
|
nand_dll_reset(dev->info);
|
|
|
|
mutex_unlock(&dev->lock);
|
|
}
|
|
|
|
/**
|
|
* get_flash_index - match chip id from a flash list.
|
|
* @flash_id: a valid nor flash chip id obtained from board.
|
|
*
|
|
* try to validate the chip id by matching from a list, if not found then simply
|
|
* returns negative. In case of success returns index in to the flash devices
|
|
* array.
|
|
*/
|
|
static int get_flash_index(struct nvt_flash *dev, u32 flash_id)
|
|
{
|
|
int index;
|
|
u32 std_id = flash_id & 0xFF0000;
|
|
|
|
if (dev->info->trace_stdtable) {
|
|
/* Matches chip-id to entire list of standard table*/
|
|
for (index = 0; index < ARRAY_SIZE(flash_std_devices); index++) {
|
|
if ((flash_std_devices[index].device_id & 0xFF0000) == std_id) {
|
|
std_path = 1;
|
|
return index;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Matches chip-id to entire list of 'serial-nor flash' ids */
|
|
for (index = 0; index < ARRAY_SIZE(flash_devices); index++) {
|
|
if (flash_devices[index].device_id == flash_id)
|
|
return index;
|
|
}
|
|
|
|
/* Memory chip is not listed and not supported */
|
|
return -ENODEV;
|
|
}
|
|
|
|
/**
|
|
* nvt_flash_erase_sector - erase one sector of flash
|
|
* @dev: structure of nvt flash information
|
|
* @command: erase command to be send
|
|
* @bytes: size of command
|
|
*
|
|
* 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 nvt_flash_erase_sector(struct nvt_flash *dev, u8 cmd, u32 addr)
|
|
{
|
|
return spinor_erase_sector(dev->info, cmd, addr);
|
|
}
|
|
|
|
/**
|
|
* nvt_flash_mtd_erase - perform flash erase operation as requested by user
|
|
* @mtd: Provides the memory characteristics
|
|
* @e_info: Provides the erase information
|
|
*
|
|
* 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 nvt_flash_mtd_erase(struct mtd_info *mtd, struct erase_info *e_info)
|
|
{
|
|
struct nvt_spinor_flash *flash = get_flash_data(mtd);
|
|
struct nvt_flash *dev = mtd->priv;
|
|
u32 addr;
|
|
int len, ret, blk_erase;
|
|
|
|
if (!flash || !dev)
|
|
return -ENODEV;
|
|
|
|
addr = e_info->addr;
|
|
len = e_info->len;
|
|
|
|
if (len & ERASE_MASK_64K)
|
|
blk_erase = 0;
|
|
else
|
|
blk_erase = 1;
|
|
|
|
mutex_lock(&dev->lock);
|
|
|
|
/* now erase sectors in loop */
|
|
while (len) {
|
|
/* preparing the command for flash */
|
|
if (blk_erase)
|
|
ret = nvt_flash_erase_sector(dev, flash->erase_cmd, addr);
|
|
else {
|
|
if (flash->mtd.size > SPI_FLASH_16MB_BOUN)
|
|
if (dev->info->enter_4byte_addr_mode == 1)
|
|
ret = nvt_flash_erase_sector(dev, FLASH_CMD_SECTOR_ERASE, addr);
|
|
else
|
|
ret = nvt_flash_erase_sector(dev, FLASH_CMD_SECTOR_ERASE_4BYTE, addr);
|
|
else
|
|
ret = nvt_flash_erase_sector(dev, FLASH_CMD_SECTOR_ERASE, addr);
|
|
}
|
|
|
|
if (ret) {
|
|
mutex_unlock(&dev->lock);
|
|
return ret;
|
|
}
|
|
if (blk_erase) {
|
|
addr += ERASE_64K;
|
|
len -= ERASE_64K;
|
|
} else {
|
|
addr += mtd->erasesize;
|
|
len -= mtd->erasesize;
|
|
}
|
|
}
|
|
|
|
mutex_unlock(&dev->lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* nvt_flash_mtd_read - performs flash read operation as requested by the user
|
|
* @mtd: MTD information
|
|
* @from: Address from which to start read
|
|
* @len: Number of bytes to be read
|
|
* @retlen: Fills the Number of bytes actually read
|
|
* @buf: Fills this after reading
|
|
*
|
|
* Read an address range from the flash chip. The address range
|
|
* may be any size provided it is within the physical boundaries.
|
|
* Returns 0 on success, non zero otherwise
|
|
*/
|
|
static int nvt_flash_mtd_read(struct mtd_info *mtd, loff_t from, size_t len,
|
|
size_t *retlen, u8 *buf)
|
|
{
|
|
struct nvt_spinor_flash *flash = get_flash_data(mtd);
|
|
struct nvt_flash *dev = mtd->priv;
|
|
int ret;
|
|
void *src;
|
|
|
|
if (!flash || !dev)
|
|
return -ENODEV;
|
|
|
|
src = flash->base_addr;
|
|
|
|
mutex_lock(&dev->lock);
|
|
|
|
/*Read operation*/
|
|
|
|
do {
|
|
if (len > PAGE_SIZE)
|
|
ret = spinor_read_operation(dev->info, from, PAGE_SIZE, (u8 *)src);
|
|
else
|
|
ret = spinor_read_operation(dev->info, from, len, (u8 *)src);
|
|
|
|
if (ret) {
|
|
mutex_unlock(&dev->lock);
|
|
return ret;
|
|
}
|
|
|
|
if (len > PAGE_SIZE) {
|
|
memcpy_fromio(buf, (u8 *)src, PAGE_SIZE);
|
|
buf += PAGE_SIZE;
|
|
*retlen += PAGE_SIZE;
|
|
from += PAGE_SIZE;
|
|
len -= PAGE_SIZE;
|
|
} else {
|
|
memcpy_fromio(buf, (u8 *)src, len);
|
|
*retlen += len;
|
|
len = 0;
|
|
}
|
|
} while (len > 0);
|
|
|
|
mutex_unlock(&dev->lock);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static inline int nvt_flash_cpy_toio(struct nvt_flash *dev,
|
|
loff_t dest, const void *src, size_t len)
|
|
{
|
|
return spinor_program_operation(dev->info, dest, len, (u8 *)src);
|
|
}
|
|
|
|
/**
|
|
* nvt_flash_mtd_write - performs write operation as requested by the user.
|
|
* @mtd: MTD information.
|
|
* @to: Address to write.
|
|
* @len: Number of bytes to be written.
|
|
* @retlen: Number of bytes actually wrote.
|
|
* @buf: Buffer from which the data to be taken.
|
|
*
|
|
* Write an address range to the flash chip. Data must be written in
|
|
* flash_page_size chunks. The address range may be any size provided
|
|
* it is within the physical boundaries.
|
|
* Returns 0 on success, non zero otherwise
|
|
*/
|
|
static int nvt_flash_mtd_write(struct mtd_info *mtd, loff_t to, size_t len,
|
|
size_t *retlen, const u8 *sr_buf)
|
|
{
|
|
struct nvt_spinor_flash *flash = get_flash_data(mtd);
|
|
struct nvt_flash *dev = mtd->priv;
|
|
u32 page_offset, page_size;
|
|
int ret;
|
|
void *buf;
|
|
|
|
if (!flash || !dev)
|
|
return -ENODEV;
|
|
|
|
buf = flash->base_addr;
|
|
|
|
mutex_lock(&dev->lock);
|
|
|
|
page_offset = (u32)to % flash->page_size;
|
|
|
|
// /* do if all the bytes fit onto one page */
|
|
if (page_offset + len <= flash->page_size) {
|
|
memcpy_toio(buf, sr_buf, len);
|
|
ret = nvt_flash_cpy_toio(dev, to, buf, len);
|
|
if (!ret)
|
|
*retlen += len;
|
|
} else {
|
|
u32 i;
|
|
|
|
/* the size of data remaining on the first page */
|
|
page_size = flash->page_size - page_offset;
|
|
|
|
memcpy_toio(buf, sr_buf, page_size);
|
|
|
|
ret = nvt_flash_cpy_toio(dev, to, buf,
|
|
page_size);
|
|
if (ret)
|
|
goto err_write;
|
|
else
|
|
*retlen += page_size;
|
|
|
|
/* write everything in pagesize chunks */
|
|
for (i = page_size; i < len; i += page_size) {
|
|
page_size = len - i;
|
|
if (page_size > flash->page_size)
|
|
page_size = flash->page_size;
|
|
|
|
memcpy_toio(buf, sr_buf + i, page_size);
|
|
|
|
ret = nvt_flash_cpy_toio(dev, to + i,
|
|
buf, page_size);
|
|
if (ret)
|
|
break;
|
|
else
|
|
*retlen += page_size;
|
|
}
|
|
}
|
|
|
|
err_write:
|
|
mutex_unlock(&dev->lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int nvt_spinor_read_id(struct drv_nand_dev_info *info, uint32_t *id)
|
|
{
|
|
uint8_t card_id[8] = {0};
|
|
|
|
if (nand_cmd_read_id(card_id, info) != 0) {
|
|
printk("NOR cmd timeout\r\n");
|
|
return -1;
|
|
} else {
|
|
printk("id = 0x%02x 0x%02x 0x%02x 0x%02x\n",
|
|
card_id[0], card_id[1], card_id[2], card_id[3]);
|
|
|
|
*id = card_id[0] | (card_id[1] << 8) | (card_id[2] << 16);
|
|
if (*id == WINBOND_W25Q256FV) {
|
|
info->enter_4byte_addr_mode = 1;
|
|
} else {
|
|
info->enter_4byte_addr_mode = 0;
|
|
}
|
|
return 0;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* nvt_flash_detect - Detects the NOR Flash chip.
|
|
* @dev: structure of NVT Flash information.
|
|
*
|
|
* This routine will check whether there exists a flash chip
|
|
* Return index of the probed flash in flash devices structure
|
|
*/
|
|
static int nvt_flash_detect(struct nvt_flash *dev)
|
|
{
|
|
int ret;
|
|
u32 val = 0;
|
|
|
|
mutex_lock(&dev->lock);
|
|
|
|
/* send readid command in sw mode */
|
|
ret = nvt_spinor_read_id(dev->info, (uint32_t *)&val);
|
|
if (ret)
|
|
goto err_probe;
|
|
|
|
/* get memory chip id */
|
|
val &= 0x00ffffff;
|
|
ret = get_flash_index(dev, val);
|
|
|
|
err_probe:
|
|
mutex_unlock(&dev->lock);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
#ifdef CONFIG_OF
|
|
static int nvt_flash_probe_config_dt(struct platform_device *pdev,
|
|
struct device_node *np)
|
|
{
|
|
struct nvt_flash_plat_data *pdata = dev_get_platdata(&pdev->dev);
|
|
struct device_node *pp = NULL;
|
|
const __be32 *addr;
|
|
u32 val;
|
|
int len;
|
|
|
|
if (!np)
|
|
return -ENODEV;
|
|
|
|
of_property_read_u32(np, "clock-frequency", &val);
|
|
pdata->clk_rate = val;
|
|
|
|
pdata->board_flash_info = devm_kzalloc(&pdev->dev,
|
|
sizeof(*pdata->board_flash_info),
|
|
GFP_KERNEL);
|
|
|
|
/* Fill structs for each subnode (flash device) */
|
|
while ((pp = of_get_next_child(np, pp))) {
|
|
struct nvt_flash_info *flash_info;
|
|
|
|
flash_info = (void*) &pdata->board_flash_info;
|
|
pdata->np = pp;
|
|
|
|
/* Read base-addr and size from DT */
|
|
addr = of_get_property(pp, "reg", &len);
|
|
if (addr) {
|
|
pdata->board_flash_info->mem_base = be32_to_cpup(&addr[1]);
|
|
pdata->board_flash_info->size = be32_to_cpup(&addr[3]);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#else
|
|
static int nvt_flash_probe_config_dt(struct platform_device *pdev,
|
|
struct device_node *np)
|
|
{
|
|
return -ENOSYS;
|
|
}
|
|
#endif
|
|
|
|
static int nvt_read_sfdp(struct drv_nand_dev_info *info)
|
|
{
|
|
int ret;
|
|
u8 sfdp_data;
|
|
u32 address = 0x32;
|
|
|
|
ret = spinor_read_sfdp(info, address, 1, &sfdp_data);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (sfdp_data == MXIC25L25635F_SFDP)
|
|
info->hspeed_dummy_cycle = 10;
|
|
|
|
return E_OK;
|
|
}
|
|
|
|
static int nvt_flash_setup(struct platform_device *pdev,
|
|
struct device_node *np)
|
|
{
|
|
struct nvt_flash *dev = platform_get_drvdata(pdev);
|
|
struct nvt_flash_info *flash_info;
|
|
struct nvt_flash_plat_data *pdata;
|
|
struct nvt_spinor_flash *flash;
|
|
#ifndef CONFIG_OF
|
|
struct mtd_part_parser_data ppdata = {};
|
|
struct mtd_partition *parts = NULL;
|
|
int count = 0;
|
|
#endif
|
|
int flash_index;
|
|
int ret = 0;
|
|
struct flash_device *trace_flash_devices;
|
|
|
|
pdata = dev_get_platdata(&pdev->dev);
|
|
|
|
flash_info = pdata->board_flash_info;
|
|
if (!flash_info)
|
|
return -ENODEV;
|
|
|
|
flash = devm_kzalloc(&pdev->dev, sizeof(*flash), GFP_ATOMIC);
|
|
if (!flash)
|
|
return -ENOMEM;
|
|
|
|
/* verify whether nor flash is really present on board */
|
|
flash_index = nvt_flash_detect(dev);
|
|
if (flash_index < 0) {
|
|
dev_info(&pdev->dev, "spinor not found\n");
|
|
return flash_index;
|
|
}
|
|
|
|
/* map the memory for nor flash chip */
|
|
flash->base_addr = kzalloc(PAGE_SIZE, GFP_KERNEL);
|
|
if (!flash->base_addr)
|
|
return -EIO;
|
|
|
|
dev->flash = flash;
|
|
flash->mtd.priv = dev;
|
|
|
|
#ifdef CONFIG_OF
|
|
of_property_read_string_index(pdev->dev.of_node, "nvt-devname", 0, (void *)&flash_info->name);
|
|
#endif
|
|
|
|
if (std_path)
|
|
trace_flash_devices = flash_std_devices;
|
|
else
|
|
trace_flash_devices = flash_devices;
|
|
|
|
|
|
if (flash_info->name)
|
|
flash->mtd.name = flash_info->name;
|
|
else
|
|
flash->mtd.name = trace_flash_devices[flash_index].name;
|
|
|
|
flash->mtd.type = MTD_NORFLASH;
|
|
flash->mtd.writesize = 1;
|
|
flash->mtd.flags = MTD_CAP_NORFLASH;
|
|
flash->mtd.size = trace_flash_devices[flash_index].size_in_bytes;;
|
|
flash->mtd.erasesize = trace_flash_devices[flash_index].sectorsize;
|
|
flash->page_size = trace_flash_devices[flash_index].pagesize;
|
|
flash->mtd.writebufsize = flash->page_size;
|
|
if (flash->mtd.size > SPI_FLASH_16MB_BOUN) {
|
|
if(trace_flash_devices[flash_index].erase_cmd == FLASH_CMD_BLOCK_ERASE)
|
|
if (dev->info->enter_4byte_addr_mode != 1)
|
|
trace_flash_devices[flash_index].erase_cmd = FLASH_CMD_BLOCK_ERASE_4BYTE;
|
|
|
|
if(trace_flash_devices[flash_index].erase_cmd == FLASH_CMD_SECTOR_ERASE)
|
|
trace_flash_devices[flash_index].erase_cmd = FLASH_CMD_SECTOR_ERASE_4BYTE;
|
|
}
|
|
|
|
flash->erase_cmd = trace_flash_devices[flash_index].erase_cmd;
|
|
flash->mtd._erase = nvt_flash_mtd_erase;
|
|
flash->mtd._read = nvt_flash_mtd_read;
|
|
flash->mtd._write = nvt_flash_mtd_write;
|
|
flash->dev_id = trace_flash_devices[flash_index].device_id;
|
|
/*Copy flash info*/
|
|
dev->info->flash_info->page_size = flash->page_size;
|
|
dev->info->flash_info->device_size = flash->mtd.size;
|
|
dev->info->flash_info->block_size = flash->mtd.erasesize;
|
|
dev->info->flash_info->chip_sel = 0;
|
|
#ifndef CONFIG_FLASH_ONLY_DUAL
|
|
dev->info->flash_info->nor_quad_support = trace_flash_devices[flash_index].quad_mode;
|
|
dev->info->flash_info->nor_read_mode = trace_flash_devices[flash_index].read_mode;
|
|
#else
|
|
if (trace_flash_devices[flash_index].quad_mode == WR_QPP)
|
|
dev->info->flash_info->nor_quad_support = 0;
|
|
|
|
if (trace_flash_devices[flash_index].read_mode == SPI_NOR_QUAD_READ)
|
|
dev->info->flash_info->nor_read_mode = SPI_NOR_DUAL_READ;
|
|
#endif
|
|
dev->info->flash_info->phy_page_ratio = 0;
|
|
dev->info->flash_info->chip_id = flash->dev_id;
|
|
|
|
dev_info(&pdev->dev, "mtd .name=%s .size=%llx(%lluM) .erasesize = 0x%x(%uK)\n",
|
|
flash->mtd.name, flash->mtd.size,
|
|
flash->mtd.size / (1024 * 1024),
|
|
flash->mtd.erasesize, flash->mtd.erasesize / 1024);
|
|
|
|
dev_info(&pdev->dev, "%d-bit mode @ %d Hz\n",
|
|
(dev->info->flash_info->nor_quad_support == WR_QPP) ? \
|
|
4 : 1, dev->info->flash_freq);
|
|
|
|
#ifndef CONFIG_OF
|
|
if (flash_info->partitions) {
|
|
parts = flash_info->partitions;
|
|
count = flash_info->nr_partitions;
|
|
}
|
|
ret = mtd_device_parse_register(&flash->mtd, NULL, &ppdata, parts, count);
|
|
#else
|
|
flash->mtd.dev.parent = &pdev->dev;
|
|
|
|
flash->mtd.dev.of_node = pdev->dev.of_node;
|
|
|
|
ret = mtd_device_register(&flash->mtd, NULL, 0);
|
|
#endif
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "Err MTD partition=%d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
dev->info->hspeed_dummy_cycle = 8;
|
|
|
|
nvt_read_sfdp(dev->info);
|
|
|
|
flash_copy_info(dev->info);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* nvt_flash_probe - Entry routine
|
|
* @pdev: platform device structure
|
|
*
|
|
* This is the first routine which gets invoked during booting and does all
|
|
* initialization/allocation work. The routine looks for available memory banks,
|
|
* and do proper init for any found one.
|
|
* Returns 0 on success, non zero otherwise
|
|
*/
|
|
static int nvt_flash_probe(struct platform_device *pdev)
|
|
{
|
|
struct device_node *np = pdev->dev.of_node;
|
|
struct nvt_flash_plat_data *pdata = NULL;
|
|
struct nvt_flash *dev;
|
|
struct resource *nvt_flash_base;
|
|
int ret = 0;
|
|
|
|
nvt_spinor_check_fastboot();
|
|
|
|
if (np) {
|
|
pdata = devm_kzalloc(&pdev->dev, sizeof(*pdata), GFP_KERNEL);
|
|
if (!pdata) {
|
|
pr_err("%s: ERROR: no memory", __func__);
|
|
ret = -ENOMEM;
|
|
goto err;
|
|
}
|
|
pdev->dev.platform_data = pdata;
|
|
ret = nvt_flash_probe_config_dt(pdev, np);
|
|
if (ret) {
|
|
ret = -ENODEV;
|
|
dev_err(&pdev->dev, "no platform data\n");
|
|
goto err;
|
|
}
|
|
} else {
|
|
pdata = dev_get_platdata(&pdev->dev);
|
|
if (!pdata) {
|
|
ret = -ENODEV;
|
|
dev_err(&pdev->dev, "no platform data\n");
|
|
goto err;
|
|
}
|
|
}
|
|
|
|
dev = devm_kzalloc(&pdev->dev, sizeof(*dev), GFP_KERNEL);
|
|
if (!dev) {
|
|
ret = -ENOMEM;
|
|
dev_err(&pdev->dev, "mem alloc fail\n");
|
|
goto err;
|
|
}
|
|
|
|
dev->info = kzalloc(sizeof(struct drv_nand_dev_info), GFP_KERNEL);
|
|
if (!dev->info) {
|
|
dev_err(&pdev->dev, "failed to allocate drv_nand_dev_info\n");
|
|
return -ENOMEM;
|
|
}
|
|
|
|
dev->info->flash_info = kzalloc(sizeof(struct nvt_nand_flash), GFP_KERNEL);
|
|
if (!dev->info->flash_info) {
|
|
dev_err(&pdev->dev, "failed to allocate nvt_nand_flash\n");
|
|
return -ENOMEM;
|
|
}
|
|
|
|
dev->info->irq = platform_get_irq(pdev, 0);
|
|
if (dev->info->irq < 0) {
|
|
ret = -ENODEV;
|
|
dev_err(&pdev->dev, "invalid smi irq\n");
|
|
goto err;
|
|
}
|
|
|
|
nvt_flash_base = platform_get_resource(pdev, IORESOURCE_MEM, 0);
|
|
|
|
dev->info->mmio_base = devm_ioremap_resource(&pdev->dev, nvt_flash_base);
|
|
if (IS_ERR(dev->info->mmio_base)) {
|
|
ret = PTR_ERR(dev->info->mmio_base);
|
|
goto err;
|
|
}
|
|
|
|
dev->info->pdev = pdev;
|
|
dev->clk = clk_get(&pdev->dev, dev_name(&pdev->dev));
|
|
if (IS_ERR(dev->clk)) {
|
|
ret = PTR_ERR(dev->clk);
|
|
goto err;
|
|
} else {
|
|
clk_prepare(dev->clk);
|
|
clk_enable(dev->clk);
|
|
}
|
|
dev->info->clk = dev->clk;
|
|
|
|
of_property_read_u32(pdev->dev.of_node, "clock-frequency",
|
|
&dev->info->flash_freq);
|
|
|
|
of_property_read_u32(pdev->dev.of_node, "trace-stdtable",
|
|
&dev->info->trace_stdtable);
|
|
|
|
mutex_init(&dev->lock);
|
|
|
|
init_completion(&dev->info->cmd_complete);
|
|
|
|
ret = devm_request_irq(&pdev->dev, dev->info->irq, nvt_flash_int_handler,
|
|
0, pdev->name, dev);
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "NVT SPIFLASH IRQ allocation failed\n");
|
|
goto err_irq;
|
|
}
|
|
|
|
nvt_flash_hw_init(dev);
|
|
|
|
platform_set_drvdata(pdev, dev);
|
|
|
|
ret = nvt_flash_setup(pdev, pdata->np);
|
|
if (ret) {
|
|
dev_err(&pdev->dev, "setup failed\n");
|
|
goto err_setup;
|
|
}
|
|
|
|
return 0;
|
|
|
|
err_setup:
|
|
platform_set_drvdata(pdev, NULL);
|
|
err_irq:
|
|
clk_disable_unprepare(dev->clk);
|
|
err:
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* nvt_flash_remove - Exit routine
|
|
* @pdev: platform device structure
|
|
*
|
|
* free all allocations and delete the partitions.
|
|
*/
|
|
static int nvt_flash_remove(struct platform_device *pdev)
|
|
{
|
|
struct nvt_flash *dev;
|
|
struct nvt_spinor_flash *flash;
|
|
int ret;
|
|
|
|
dev = platform_get_drvdata(pdev);
|
|
if (!dev) {
|
|
dev_err(&pdev->dev, "dev is null\n");
|
|
return -ENODEV;
|
|
}
|
|
|
|
/* clean up for nor flash */
|
|
flash = dev->flash;
|
|
|
|
/* clean up mtd stuff */
|
|
ret = mtd_device_unregister(&flash->mtd);
|
|
if (ret)
|
|
dev_err(&pdev->dev, "error removing mtd\n");
|
|
|
|
clk_disable_unprepare(dev->clk);
|
|
platform_set_drvdata(pdev, NULL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
#ifdef CONFIG_PM
|
|
static int nvt_flash_suspend(struct device *dev)
|
|
{
|
|
struct nvt_flash *sdev = dev_get_drvdata(dev);
|
|
|
|
if (sdev && sdev->clk)
|
|
clk_disable_unprepare(sdev->clk);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int nvt_flash_resume(struct device *dev)
|
|
{
|
|
struct nvt_flash *sdev = dev_get_drvdata(dev);
|
|
int ret = -EPERM;
|
|
|
|
if (sdev && sdev->clk)
|
|
ret = clk_prepare_enable(sdev->clk);
|
|
|
|
if (!ret)
|
|
nvt_flash_hw_init(sdev);
|
|
return ret;
|
|
}
|
|
|
|
static SIMPLE_DEV_PM_OPS(nvt_flash_pm_ops, nvt_flash_suspend, nvt_flash_resume);
|
|
#endif
|
|
|
|
#ifdef CONFIG_OF
|
|
static const struct of_device_id nvt_flash_id_table[] = {
|
|
{ .compatible = "nvt,nvt_spinor" },
|
|
{}
|
|
};
|
|
MODULE_DEVICE_TABLE(of, nvt_flash_id_table);
|
|
#endif
|
|
|
|
static struct platform_driver nvt_flash_driver = {
|
|
.driver = {
|
|
.name = "spi_nor",
|
|
.bus = &platform_bus_type,
|
|
.owner = THIS_MODULE,
|
|
#ifdef CONFIG_OF
|
|
.of_match_table = of_match_ptr(nvt_flash_id_table),
|
|
#endif
|
|
#ifdef CONFIG_PM
|
|
.pm = &nvt_flash_pm_ops,
|
|
#endif
|
|
},
|
|
.probe = nvt_flash_probe,
|
|
.remove = nvt_flash_remove,
|
|
};
|
|
|
|
static int __init spinor_init(void)
|
|
{
|
|
return platform_driver_register(&nvt_flash_driver);
|
|
}
|
|
|
|
static void __exit spinor_exit(void)
|
|
{
|
|
platform_driver_unregister(&nvt_flash_driver);
|
|
}
|
|
|
|
module_init(spinor_init);
|
|
module_exit(spinor_exit);
|
|
|
|
MODULE_LICENSE("GPL v2");
|
|
MODULE_AUTHOR("Howard Chang");
|
|
MODULE_DESCRIPTION("MTD Flash driver for serial nor flash chips");
|
|
MODULE_VERSION("NOR_VERSION");
|