cregit-Linux how code gets into the kernel

Release 4.7 drivers/mtd/onenand/samsung.c

/*
 * Samsung S3C64XX/S5PC1XX OneNAND driver
 *
 *  Copyright © 2008-2010 Samsung Electronics
 *  Kyungmin Park <kyungmin.park@samsung.com>
 *  Marek Szyprowski <m.szyprowski@samsung.com>
 *
 * This program 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.
 *
 * Implementation:
 *      S3C64XX: emulate the pseudo BufferRAM
 *      S5PC110: use DMA
 */

#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/sched.h>
#include <linux/slab.h>
#include <linux/mtd/mtd.h>
#include <linux/mtd/onenand.h>
#include <linux/mtd/partitions.h>
#include <linux/dma-mapping.h>
#include <linux/interrupt.h>
#include <linux/io.h>

#include <asm/mach/flash.h>

#include "samsung.h"


enum soc_type {
	
TYPE_S3C6400,
	
TYPE_S3C6410,
	
TYPE_S5PC110,
};


#define ONENAND_ERASE_STATUS		0x00

#define ONENAND_MULTI_ERASE_SET		0x01

#define ONENAND_ERASE_START		0x03

#define ONENAND_UNLOCK_START		0x08

#define ONENAND_UNLOCK_END		0x09

#define ONENAND_LOCK_START		0x0A

#define ONENAND_LOCK_END		0x0B

#define ONENAND_LOCK_TIGHT_START	0x0C

#define ONENAND_LOCK_TIGHT_END		0x0D

#define ONENAND_UNLOCK_ALL		0x0E

#define ONENAND_OTP_ACCESS		0x12

#define ONENAND_SPARE_ACCESS_ONLY	0x13

#define ONENAND_MAIN_ACCESS_ONLY	0x14

#define ONENAND_ERASE_VERIFY		0x15

#define ONENAND_MAIN_SPARE_ACCESS	0x16

#define ONENAND_PIPELINE_READ		0x4000


#define MAP_00				(0x0)

#define MAP_01				(0x1)

#define MAP_10				(0x2)

#define MAP_11				(0x3)


#define S3C64XX_CMD_MAP_SHIFT		24


#define S3C6400_FBA_SHIFT		10

#define S3C6400_FPA_SHIFT		4

#define S3C6400_FSA_SHIFT		2


#define S3C6410_FBA_SHIFT		12

#define S3C6410_FPA_SHIFT		6

#define S3C6410_FSA_SHIFT		4

/* S5PC110 specific definitions */

#define S5PC110_DMA_SRC_ADDR		0x400

#define S5PC110_DMA_SRC_CFG		0x404

#define S5PC110_DMA_DST_ADDR		0x408

#define S5PC110_DMA_DST_CFG		0x40C

#define S5PC110_DMA_TRANS_SIZE		0x414

#define S5PC110_DMA_TRANS_CMD		0x418

#define S5PC110_DMA_TRANS_STATUS	0x41C

#define S5PC110_DMA_TRANS_DIR		0x420

#define S5PC110_INTC_DMA_CLR		0x1004

#define S5PC110_INTC_ONENAND_CLR	0x1008

#define S5PC110_INTC_DMA_MASK		0x1024

#define S5PC110_INTC_ONENAND_MASK	0x1028

#define S5PC110_INTC_DMA_PEND		0x1044

#define S5PC110_INTC_ONENAND_PEND	0x1048

#define S5PC110_INTC_DMA_STATUS		0x1064

#define S5PC110_INTC_ONENAND_STATUS	0x1068


#define S5PC110_INTC_DMA_TD		(1 << 24)

#define S5PC110_INTC_DMA_TE		(1 << 16)


#define S5PC110_DMA_CFG_SINGLE		(0x0 << 16)

#define S5PC110_DMA_CFG_4BURST		(0x2 << 16)

#define S5PC110_DMA_CFG_8BURST		(0x3 << 16)

#define S5PC110_DMA_CFG_16BURST		(0x4 << 16)


#define S5PC110_DMA_CFG_INC		(0x0 << 8)

#define S5PC110_DMA_CFG_CNT		(0x1 << 8)


#define S5PC110_DMA_CFG_8BIT		(0x0 << 0)

#define S5PC110_DMA_CFG_16BIT		(0x1 << 0)

#define S5PC110_DMA_CFG_32BIT		(0x2 << 0)


#define S5PC110_DMA_SRC_CFG_READ	(S5PC110_DMA_CFG_16BURST | \
                                        S5PC110_DMA_CFG_INC | \
                                        S5PC110_DMA_CFG_16BIT)

#define S5PC110_DMA_DST_CFG_READ	(S5PC110_DMA_CFG_16BURST | \
                                        S5PC110_DMA_CFG_INC | \
                                        S5PC110_DMA_CFG_32BIT)

#define S5PC110_DMA_SRC_CFG_WRITE	(S5PC110_DMA_CFG_16BURST | \
                                        S5PC110_DMA_CFG_INC | \
                                        S5PC110_DMA_CFG_32BIT)

#define S5PC110_DMA_DST_CFG_WRITE	(S5PC110_DMA_CFG_16BURST | \
                                        S5PC110_DMA_CFG_INC | \
                                        S5PC110_DMA_CFG_16BIT)


#define S5PC110_DMA_TRANS_CMD_TDC	(0x1 << 18)

#define S5PC110_DMA_TRANS_CMD_TEC	(0x1 << 16)

#define S5PC110_DMA_TRANS_CMD_TR	(0x1 << 0)


#define S5PC110_DMA_TRANS_STATUS_TD	(0x1 << 18)

#define S5PC110_DMA_TRANS_STATUS_TB	(0x1 << 17)

#define S5PC110_DMA_TRANS_STATUS_TE	(0x1 << 16)


#define S5PC110_DMA_DIR_READ		0x0

#define S5PC110_DMA_DIR_WRITE		0x1


struct s3c_onenand {
	
struct mtd_info	*mtd;
	
struct platform_device	*pdev;
	
enum soc_type	type;
	
void __iomem	*base;
	
struct resource *base_res;
	
void __iomem	*ahb_addr;
	
struct resource *ahb_res;
	
int		bootram_command;
	
void __iomem	*page_buf;
	
void __iomem	*oob_buf;
	
unsigned int	(*mem_addr)(int fba, int fpa, int fsa);
	
unsigned int	(*cmd_map)(unsigned int type, unsigned int val);
	
void __iomem	*dma_addr;
	
struct resource *dma_res;
	
unsigned long	phys_base;
	
struct completion	complete;
};


#define CMD_MAP_00(dev, addr)		(dev->cmd_map(MAP_00, ((addr) << 1)))

#define CMD_MAP_01(dev, mem_addr)	(dev->cmd_map(MAP_01, (mem_addr)))

#define CMD_MAP_10(dev, mem_addr)	(dev->cmd_map(MAP_10, (mem_addr)))

#define CMD_MAP_11(dev, addr)		(dev->cmd_map(MAP_11, ((addr) << 2)))


static struct s3c_onenand *onenand;


static inline int s3c_read_reg(int offset) { return readl(onenand->base + offset); }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park20100.00%1100.00%
Total20100.00%1100.00%


static inline void s3c_write_reg(int value, int offset) { writel(value, onenand->base + offset); }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park24100.00%1100.00%
Total24100.00%1100.00%


static inline int s3c_read_cmd(unsigned int cmd) { return readl(onenand->ahb_addr + cmd); }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park21100.00%1100.00%
Total21100.00%1100.00%


static inline void s3c_write_cmd(int value, unsigned int cmd) { writel(value, onenand->ahb_addr + cmd); }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park25100.00%1100.00%
Total25100.00%1100.00%

#ifdef SAMSUNG_DEBUG
static void s3c_dump_reg(void) { int i; for (i = 0; i < 0x400; i += 0x40) { printk(KERN_INFO "0x%08X: 0x%08x 0x%08x 0x%08x 0x%08x\n", (unsigned int) onenand->base + i, s3c_read_reg(i), s3c_read_reg(i + 0x10), s3c_read_reg(i + 0x20), s3c_read_reg(i + 0x30)); } }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park69100.00%1100.00%
Total69100.00%1100.00%

#endif
static unsigned int s3c64xx_cmd_map(unsigned type, unsigned val) { return (type << S3C64XX_CMD_MAP_SHIFT) | val; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park22100.00%1100.00%
Total22100.00%1100.00%


static unsigned int s3c6400_mem_addr(int fba, int fpa, int fsa) { return (fba << S3C6400_FBA_SHIFT) | (fpa << S3C6400_FPA_SHIFT) | (fsa << S3C6400_FSA_SHIFT); }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park35100.00%1100.00%
Total35100.00%1100.00%


static unsigned int s3c6410_mem_addr(int fba, int fpa, int fsa) { return (fba << S3C6410_FBA_SHIFT) | (fpa << S3C6410_FPA_SHIFT) | (fsa << S3C6410_FSA_SHIFT); }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park35100.00%1100.00%
Total35100.00%1100.00%


static void s3c_onenand_reset(void) { unsigned long timeout = 0x10000; int stat; s3c_write_reg(ONENAND_MEM_RESET_COLD, MEM_RESET_OFFSET); while (1 && timeout--) { stat = s3c_read_reg(INT_ERR_STAT_OFFSET); if (stat & RST_CMP) break; } stat = s3c_read_reg(INT_ERR_STAT_OFFSET); s3c_write_reg(stat, INT_ERR_ACK_OFFSET); /* Clear interrupt */ s3c_write_reg(0x0, INT_ERR_ACK_OFFSET); /* Clear the ECC status */ s3c_write_reg(0x0, ECC_ERR_STAT_OFFSET); }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park77100.00%1100.00%
Total77100.00%1100.00%


static unsigned short s3c_onenand_readw(void __iomem *addr) { struct onenand_chip *this = onenand->mtd->priv; struct device *dev = &onenand->pdev->dev; int reg = addr - this->base; int word_addr = reg >> 1; int value; /* It's used for probing time */ switch (reg) { case ONENAND_REG_MANUFACTURER_ID: return s3c_read_reg(MANUFACT_ID_OFFSET); case ONENAND_REG_DEVICE_ID: return s3c_read_reg(DEVICE_ID_OFFSET); case ONENAND_REG_VERSION_ID: return s3c_read_reg(FLASH_VER_ID_OFFSET); case ONENAND_REG_DATA_BUFFER_SIZE: return s3c_read_reg(DATA_BUF_SIZE_OFFSET); case ONENAND_REG_TECHNOLOGY: return s3c_read_reg(TECH_OFFSET); case ONENAND_REG_SYS_CFG1: return s3c_read_reg(MEM_CFG_OFFSET); /* Used at unlock all status */ case ONENAND_REG_CTRL_STATUS: return 0; case ONENAND_REG_WP_STATUS: return ONENAND_WP_US; default: break; } /* BootRAM access control */ if ((unsigned int) addr < ONENAND_DATARAM && onenand->bootram_command) { if (word_addr == 0) return s3c_read_reg(MANUFACT_ID_OFFSET); if (word_addr == 1) return s3c_read_reg(DEVICE_ID_OFFSET); if (word_addr == 2) return s3c_read_reg(FLASH_VER_ID_OFFSET); } value = s3c_read_cmd(CMD_MAP_11(onenand, word_addr)) & 0xffff; dev_info(dev, "%s: Illegal access at reg 0x%x, value 0x%x\n", __func__, word_addr, value); return value; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park213100.00%1100.00%
Total213100.00%1100.00%


static void s3c_onenand_writew(unsigned short value, void __iomem *addr) { struct onenand_chip *this = onenand->mtd->priv; struct device *dev = &onenand->pdev->dev; unsigned int reg = addr - this->base; unsigned int word_addr = reg >> 1; /* It's used for probing time */ switch (reg) { case ONENAND_REG_SYS_CFG1: s3c_write_reg(value, MEM_CFG_OFFSET); return; case ONENAND_REG_START_ADDRESS1: case ONENAND_REG_START_ADDRESS2: return; /* Lock/lock-tight/unlock/unlock_all */ case ONENAND_REG_START_BLOCK_ADDRESS: return; default: break; } /* BootRAM access control */ if ((unsigned int)addr < ONENAND_DATARAM) { if (value == ONENAND_CMD_READID) { onenand->bootram_command = 1; return; } if (value == ONENAND_CMD_RESET) { s3c_write_reg(ONENAND_MEM_RESET_COLD, MEM_RESET_OFFSET); onenand->bootram_command = 0; return; } } dev_info(dev, "%s: Illegal access at reg 0x%x, value 0x%x\n", __func__, word_addr, value); s3c_write_cmd(value, CMD_MAP_11(onenand, word_addr)); }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park163100.00%1100.00%
Total163100.00%1100.00%


static int s3c_onenand_wait(struct mtd_info *mtd, int state) { struct device *dev = &onenand->pdev->dev; unsigned int flags = INT_ACT; unsigned int stat, ecc; unsigned long timeout; switch (state) { case FL_READING: flags |= BLK_RW_CMP | LOAD_CMP; break; case FL_WRITING: flags |= BLK_RW_CMP | PGM_CMP; break; case FL_ERASING: flags |= BLK_RW_CMP | ERS_CMP; break; case FL_LOCKING: flags |= BLK_RW_CMP; break; default: break; } /* The 20 msec is enough */ timeout = jiffies + msecs_to_jiffies(20); while (time_before(jiffies, timeout)) { stat = s3c_read_reg(INT_ERR_STAT_OFFSET); if (stat & flags) break; if (state != FL_READING) cond_resched(); } /* To get correct interrupt status in timeout case */ stat = s3c_read_reg(INT_ERR_STAT_OFFSET); s3c_write_reg(stat, INT_ERR_ACK_OFFSET); /* * In the Spec. it checks the controller status first * However if you get the correct information in case of * power off recovery (POR) test, it should read ECC status first */ if (stat & LOAD_CMP) { ecc = s3c_read_reg(ECC_ERR_STAT_OFFSET); if (ecc & ONENAND_ECC_4BIT_UNCORRECTABLE) { dev_info(dev, "%s: ECC error = 0x%04x\n", __func__, ecc); mtd->ecc_stats.failed++; return -EBADMSG; } } if (stat & (LOCKED_BLK | ERS_FAIL | PGM_FAIL | LD_FAIL_ECC_ERR)) { dev_info(dev, "%s: controller error = 0x%04x\n", __func__, stat); if (stat & LOCKED_BLK) dev_info(dev, "%s: it's locked error = 0x%04x\n", __func__, stat); return -EIO; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park244100.00%1100.00%
Total244100.00%1100.00%


static int s3c_onenand_command(struct mtd_info *mtd, int cmd, loff_t addr, size_t len) { struct onenand_chip *this = mtd->priv; unsigned int *m, *s; int fba, fpa, fsa = 0; unsigned int mem_addr, cmd_map_01, cmd_map_10; int i, mcount, scount; int index; fba = (int) (addr >> this->erase_shift); fpa = (int) (addr >> this->page_shift); fpa &= this->page_mask; mem_addr = onenand->mem_addr(fba, fpa, fsa); cmd_map_01 = CMD_MAP_01(onenand, mem_addr); cmd_map_10 = CMD_MAP_10(onenand, mem_addr); switch (cmd) { case ONENAND_CMD_READ: case ONENAND_CMD_READOOB: case ONENAND_CMD_BUFFERRAM: ONENAND_SET_NEXT_BUFFERRAM(this); default: break; } index = ONENAND_CURRENT_BUFFERRAM(this); /* * Emulate Two BufferRAMs and access with 4 bytes pointer */ m = (unsigned int *) onenand->page_buf; s = (unsigned int *) onenand->oob_buf; if (index) { m += (this->writesize >> 2); s += (mtd->oobsize >> 2); } mcount = mtd->writesize >> 2; scount = mtd->oobsize >> 2; switch (cmd) { case ONENAND_CMD_READ: /* Main */ for (i = 0; i < mcount; i++) *m++ = s3c_read_cmd(cmd_map_01); return 0; case ONENAND_CMD_READOOB: s3c_write_reg(TSRF, TRANS_SPARE_OFFSET); /* Main */ for (i = 0; i < mcount; i++) *m++ = s3c_read_cmd(cmd_map_01); /* Spare */ for (i = 0; i < scount; i++) *s++ = s3c_read_cmd(cmd_map_01); s3c_write_reg(0, TRANS_SPARE_OFFSET); return 0; case ONENAND_CMD_PROG: /* Main */ for (i = 0; i < mcount; i++) s3c_write_cmd(*m++, cmd_map_01); return 0; case ONENAND_CMD_PROGOOB: s3c_write_reg(TSRF, TRANS_SPARE_OFFSET); /* Main - dummy write */ for (i = 0; i < mcount; i++) s3c_write_cmd(0xffffffff, cmd_map_01); /* Spare */ for (i = 0; i < scount; i++) s3c_write_cmd(*s++, cmd_map_01); s3c_write_reg(0, TRANS_SPARE_OFFSET); return 0; case ONENAND_CMD_UNLOCK_ALL: s3c_write_cmd(ONENAND_UNLOCK_ALL, cmd_map_10); return 0; case ONENAND_CMD_ERASE: s3c_write_cmd(ONENAND_ERASE_START, cmd_map_10); return 0; default: break; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park446100.00%1100.00%
Total446100.00%1100.00%


static unsigned char *s3c_get_bufferram(struct mtd_info *mtd, int area) { struct onenand_chip *this = mtd->priv; int index = ONENAND_CURRENT_BUFFERRAM(this); unsigned char *p; if (area == ONENAND_DATARAM) { p = (unsigned char *) onenand->page_buf; if (index == 1) p += this->writesize; } else { p = (unsigned char *) onenand->oob_buf; if (index == 1) p += mtd->oobsize; } return p; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park98100.00%1100.00%
Total98100.00%1100.00%


static int onenand_read_bufferram(struct mtd_info *mtd, int area, unsigned char *buffer, int offset, size_t count) { unsigned char *p; p = s3c_get_bufferram(mtd, area); memcpy(buffer, p + offset, count); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park53100.00%1100.00%
Total53100.00%1100.00%


static int onenand_write_bufferram(struct mtd_info *mtd, int area, const unsigned char *buffer, int offset, size_t count) { unsigned char *p; p = s3c_get_bufferram(mtd, area); memcpy(p + offset, buffer, count); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park54100.00%1100.00%
Total54100.00%1100.00%

static int (*s5pc110_dma_ops)(dma_addr_t dst, dma_addr_t src, size_t count, int direction);
static int s5pc110_dma_poll(dma_addr_t dst, dma_addr_t src, size_t count, int direction) { void __iomem *base = onenand->dma_addr; int status; unsigned long timeout; writel(src, base + S5PC110_DMA_SRC_ADDR); writel(dst, base + S5PC110_DMA_DST_ADDR); if (direction == S5PC110_DMA_DIR_READ) { writel(S5PC110_DMA_SRC_CFG_READ, base + S5PC110_DMA_SRC_CFG); writel(S5PC110_DMA_DST_CFG_READ, base + S5PC110_DMA_DST_CFG); } else { writel(S5PC110_DMA_SRC_CFG_WRITE, base + S5PC110_DMA_SRC_CFG); writel(S5PC110_DMA_DST_CFG_WRITE, base + S5PC110_DMA_DST_CFG); } writel(count, base + S5PC110_DMA_TRANS_SIZE); writel(direction, base + S5PC110_DMA_TRANS_DIR); writel(S5PC110_DMA_TRANS_CMD_TR, base + S5PC110_DMA_TRANS_CMD); /* * There's no exact timeout values at Spec. * In real case it takes under 1 msec. * So 20 msecs are enough. */ timeout = jiffies + msecs_to_jiffies(20); do { status = readl(base + S5PC110_DMA_TRANS_STATUS); if (status & S5PC110_DMA_TRANS_STATUS_TE) { writel(S5PC110_DMA_TRANS_CMD_TEC, base + S5PC110_DMA_TRANS_CMD); return -EIO; } } while (!(status & S5PC110_DMA_TRANS_STATUS_TD) && time_before(jiffies, timeout)); writel(S5PC110_DMA_TRANS_CMD_TDC, base + S5PC110_DMA_TRANS_CMD); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park19698.99%480.00%
arnd bergmannarnd bergmann21.01%120.00%
Total198100.00%5100.00%


static irqreturn_t s5pc110_onenand_irq(int irq, void *data) { void __iomem *base = onenand->dma_addr; int status, cmd = 0; status = readl(base + S5PC110_INTC_DMA_STATUS); if (likely(status & S5PC110_INTC_DMA_TD)) cmd = S5PC110_DMA_TRANS_CMD_TDC; if (unlikely(status & S5PC110_INTC_DMA_TE)) cmd = S5PC110_DMA_TRANS_CMD_TEC; writel(cmd, base + S5PC110_DMA_TRANS_CMD); writel(status, base + S5PC110_INTC_DMA_CLR); if (!onenand->complete.done) complete(&onenand->complete); return IRQ_HANDLED; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park102100.00%1100.00%
Total102100.00%1100.00%


static int s5pc110_dma_irq(dma_addr_t dst, dma_addr_t src, size_t count, int direction) { void __iomem *base = onenand->dma_addr; int status; status = readl(base + S5PC110_INTC_DMA_MASK); if (status) { status &= ~(S5PC110_INTC_DMA_TD | S5PC110_INTC_DMA_TE); writel(status, base + S5PC110_INTC_DMA_MASK); } writel(src, base + S5PC110_DMA_SRC_ADDR); writel(dst, base + S5PC110_DMA_DST_ADDR); if (direction == S5PC110_DMA_DIR_READ) { writel(S5PC110_DMA_SRC_CFG_READ, base + S5PC110_DMA_SRC_CFG); writel(S5PC110_DMA_DST_CFG_READ, base + S5PC110_DMA_DST_CFG); } else { writel(S5PC110_DMA_SRC_CFG_WRITE, base + S5PC110_DMA_SRC_CFG); writel(S5PC110_DMA_DST_CFG_WRITE, base + S5PC110_DMA_DST_CFG); } writel(count, base + S5PC110_DMA_TRANS_SIZE); writel(direction, base + S5PC110_DMA_TRANS_DIR); writel(S5PC110_DMA_TRANS_CMD_TR, base + S5PC110_DMA_TRANS_CMD); wait_for_completion_timeout(&onenand->complete, msecs_to_jiffies(20)); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park16998.83%150.00%
arnd bergmannarnd bergmann21.17%150.00%
Total171100.00%2100.00%


static int s5pc110_read_bufferram(struct mtd_info *mtd, int area, unsigned char *buffer, int offset, size_t count) { struct onenand_chip *this = mtd->priv; void __iomem *p; void *buf = (void *) buffer; dma_addr_t dma_src, dma_dst; int err, ofs, page_dma = 0; struct device *dev = &onenand->pdev->dev; p = this->base + area; if (ONENAND_CURRENT_BUFFERRAM(this)) { if (area == ONENAND_DATARAM) p += this->writesize; else p += mtd->oobsize; } if (offset & 3 || (size_t) buf & 3 || !onenand->dma_addr || count != mtd->writesize) goto normal; /* Handle vmalloc address */ if (buf >= high_memory) { struct page *page; if (((size_t) buf & PAGE_MASK) != ((size_t) (buf + count - 1) & PAGE_MASK)) goto normal; page = vmalloc_to_page(buf); if (!page) goto normal; /* Page offset */ ofs = ((size_t) buf & ~PAGE_MASK); page_dma = 1; /* DMA routine */ dma_src = onenand->phys_base + (p - this->base); dma_dst = dma_map_page(dev, page, ofs, count, DMA_FROM_DEVICE); } else { /* DMA routine */ dma_src = onenand->phys_base + (p - this->base); dma_dst = dma_map_single(dev, buf, count, DMA_FROM_DEVICE); } if (dma_mapping_error(dev, dma_dst)) { dev_err(dev, "Couldn't map a %d byte buffer for DMA\n", count); goto normal; } err = s5pc110_dma_ops(dma_dst, dma_src, count, S5PC110_DMA_DIR_READ); if (page_dma) dma_unmap_page(dev, dma_dst, count, DMA_FROM_DEVICE); else dma_unmap_single(dev, dma_dst, count, DMA_FROM_DEVICE); if (!err) return 0; normal: if (count != mtd->writesize) { /* Copy the bufferram to memory to prevent unaligned access */ memcpy(this->page_buf, p, mtd->writesize); p = this->page_buf + offset; } memcpy(buffer, p, count); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park391100.00%4100.00%
Total391100.00%4100.00%


static int s5pc110_chip_probe(struct mtd_info *mtd) { /* Now just return 0 */ return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park15100.00%1100.00%
Total15100.00%1100.00%


static int s3c_onenand_bbt_wait(struct mtd_info *mtd, int state) { unsigned int flags = INT_ACT | LOAD_CMP; unsigned int stat; unsigned long timeout; /* The 20 msec is enough */ timeout = jiffies + msecs_to_jiffies(20); while (time_before(jiffies, timeout)) { stat = s3c_read_reg(INT_ERR_STAT_OFFSET); if (stat & flags) break; } /* To get correct interrupt status in timeout case */ stat = s3c_read_reg(INT_ERR_STAT_OFFSET); s3c_write_reg(stat, INT_ERR_ACK_OFFSET); if (stat & LD_FAIL_ECC_ERR) { s3c_onenand_reset(); return ONENAND_BBT_READ_ERROR; } if (stat & LOAD_CMP) { int ecc = s3c_read_reg(ECC_ERR_STAT_OFFSET); if (ecc & ONENAND_ECC_4BIT_UNCORRECTABLE) { s3c_onenand_reset(); return ONENAND_BBT_READ_ERROR; } } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park127100.00%1100.00%
Total127100.00%1100.00%


static void s3c_onenand_check_lock_status(struct mtd_info *mtd) { struct onenand_chip *this = mtd->priv; struct device *dev = &onenand->pdev->dev; unsigned int block, end; int tmp; end = this->chipsize >> this->erase_shift; for (block = 0; block < end; block++) { unsigned int mem_addr = onenand->mem_addr(block, 0, 0); tmp = s3c_read_cmd(CMD_MAP_01(onenand, mem_addr)); if (s3c_read_reg(INT_ERR_STAT_OFFSET) & LOCKED_BLK) { dev_err(dev, "block %d is write-protected!\n", block); s3c_write_reg(LOCKED_BLK, INT_ERR_ACK_OFFSET); } } }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park120100.00%1100.00%
Total120100.00%1100.00%


static void s3c_onenand_do_lock_cmd(struct mtd_info *mtd, loff_t ofs, size_t len, int cmd) { struct onenand_chip *this = mtd->priv; int start, end, start_mem_addr, end_mem_addr; start = ofs >> this->erase_shift; start_mem_addr = onenand->mem_addr(start, 0, 0); end = start + (len >> this->erase_shift) - 1; end_mem_addr = onenand->mem_addr(end, 0, 0); if (cmd == ONENAND_CMD_LOCK) { s3c_write_cmd(ONENAND_LOCK_START, CMD_MAP_10(onenand, start_mem_addr)); s3c_write_cmd(ONENAND_LOCK_END, CMD_MAP_10(onenand, end_mem_addr)); } else { s3c_write_cmd(ONENAND_UNLOCK_START, CMD_MAP_10(onenand, start_mem_addr)); s3c_write_cmd(ONENAND_UNLOCK_END, CMD_MAP_10(onenand, end_mem_addr)); } this->wait(mtd, FL_LOCKING); }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park154100.00%1100.00%
Total154100.00%1100.00%


static void s3c_unlock_all(struct mtd_info *mtd) { struct onenand_chip *this = mtd->priv; loff_t ofs = 0; size_t len = this->chipsize; if (this->options & ONENAND_HAS_UNLOCK_ALL) { /* Write unlock command */ this->command(mtd, ONENAND_CMD_UNLOCK_ALL, 0, 0); /* No need to check return value */ this->wait(mtd, FL_LOCKING); /* Workaround for all block unlock in DDP */ if (!ONENAND_IS_DDP(this)) { s3c_onenand_check_lock_status(mtd); return; } /* All blocks on another chip */ ofs = this->chipsize >> 1; len = this->chipsize >> 1; } s3c_onenand_do_lock_cmd(mtd, ofs, len, ONENAND_CMD_UNLOCK); s3c_onenand_check_lock_status(mtd); }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park116100.00%1100.00%
Total116100.00%1100.00%


static void s3c_onenand_setup(struct mtd_info *mtd) { struct onenand_chip *this = mtd->priv; onenand->mtd = mtd; if (onenand->type == TYPE_S3C6400) { onenand->mem_addr = s3c6400_mem_addr; onenand->cmd_map = s3c64xx_cmd_map; } else if (onenand->type == TYPE_S3C6410) { onenand->mem_addr = s3c6410_mem_addr; onenand->cmd_map = s3c64xx_cmd_map; } else if (onenand->type == TYPE_S5PC110) { /* Use generic onenand functions */ this->read_bufferram = s5pc110_read_bufferram; this->chip_probe = s5pc110_chip_probe; return; } else { BUG(); } this->read_word = s3c_onenand_readw; this->write_word = s3c_onenand_writew; this->wait = s3c_onenand_wait; this->bbt_wait = s3c_onenand_bbt_wait; this->unlock_all = s3c_unlock_all; this->command = s3c_onenand_command; this->read_bufferram = onenand_read_bufferram; this->write_bufferram = onenand_write_bufferram; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park150100.00%2100.00%
Total150100.00%2100.00%


static int s3c_onenand_probe(struct platform_device *pdev) { struct onenand_platform_data *pdata; struct onenand_chip *this; struct mtd_info *mtd; struct resource *r; int size, err; pdata = dev_get_platdata(&pdev->dev); /* No need to check pdata. the platform data is optional */ size = sizeof(struct mtd_info) + sizeof(struct onenand_chip); mtd = kzalloc(size, GFP_KERNEL); if (!mtd) return -ENOMEM; onenand = kzalloc(sizeof(struct s3c_onenand), GFP_KERNEL); if (!onenand) { err = -ENOMEM; goto onenand_fail; } this = (struct onenand_chip *) &mtd[1]; mtd->priv = this; mtd->dev.parent = &pdev->dev; onenand->pdev = pdev; onenand->type = platform_get_device_id(pdev)->driver_data; s3c_onenand_setup(mtd); r = platform_get_resource(pdev, IORESOURCE_MEM, 0); if (!r) { dev_err(&pdev->dev, "no memory resource defined\n"); return -ENOENT; goto ahb_resource_failed; } onenand->base_res = request_mem_region(r->start, resource_size(r), pdev->name); if (!onenand->base_res) { dev_err(&pdev->dev, "failed to request memory resource\n"); err = -EBUSY; goto resource_failed; } onenand->base = ioremap(r->start, resource_size(r)); if (!onenand->base) { dev_err(&pdev->dev, "failed to map memory resource\n"); err = -EFAULT; goto ioremap_failed; } /* Set onenand_chip also */ this->base = onenand->base; /* Use runtime badblock check */ this->options |= ONENAND_SKIP_UNLOCK_CHECK; if (onenand->type != TYPE_S5PC110) { r = platform_get_resource(pdev, IORESOURCE_MEM, 1); if (!r) { dev_err(&pdev->dev, "no buffer memory resource defined\n"); err = -ENOENT; goto ahb_resource_failed; } onenand->ahb_res = request_mem_region(r->start, resource_size(r), pdev->name); if (!onenand->ahb_res) { dev_err(&pdev->dev, "failed to request buffer memory resource\n"); err = -EBUSY; goto ahb_resource_failed; } onenand->ahb_addr = ioremap(r->start, resource_size(r)); if (!onenand->ahb_addr) { dev_err(&pdev->dev, "failed to map buffer memory resource\n"); err = -EINVAL; goto ahb_ioremap_failed; } /* Allocate 4KiB BufferRAM */ onenand->page_buf = kzalloc(SZ_4K, GFP_KERNEL); if (!onenand->page_buf) { err = -ENOMEM; goto page_buf_fail; } /* Allocate 128 SpareRAM */ onenand->oob_buf = kzalloc(128, GFP_KERNEL); if (!onenand->oob_buf) { err = -ENOMEM; goto oob_buf_fail; } /* S3C doesn't handle subpage write */ mtd->subpage_sft = 0; this->subpagesize = mtd->writesize; } else { /* S5PC110 */ r = platform_get_resource(pdev, IORESOURCE_MEM, 1); if (!r) { dev_err(&pdev->dev, "no dma memory resource defined\n"); err = -ENOENT; goto dma_resource_failed; } onenand->dma_res = request_mem_region(r->start, resource_size(r), pdev->name); if (!onenand->dma_res) { dev_err(&pdev->dev, "failed to request dma memory resource\n"); err = -EBUSY; goto dma_resource_failed; } onenand->dma_addr = ioremap(r->start, resource_size(r)); if (!onenand->dma_addr) { dev_err(&pdev->dev, "failed to map dma memory resource\n"); err = -EINVAL; goto dma_ioremap_failed; } onenand->phys_base = onenand->base_res->start; s5pc110_dma_ops = s5pc110_dma_poll; /* Interrupt support */ r = platform_get_resource(pdev, IORESOURCE_IRQ, 0); if (r) { init_completion(&onenand->complete); s5pc110_dma_ops = s5pc110_dma_irq; err = request_irq(r->start, s5pc110_onenand_irq, IRQF_SHARED, "onenand", &onenand); if (err) { dev_err(&pdev->dev, "failed to get irq\n"); goto scan_failed; } } } if (onenand_scan(mtd, 1)) { err = -EFAULT; goto scan_failed; } if (onenand->type != TYPE_S5PC110) { /* S3C doesn't handle subpage write */ mtd->subpage_sft = 0; this->subpagesize = mtd->writesize; } if (s3c_read_reg(MEM_CFG_OFFSET) & ONENAND_SYS_CFG1_SYNC_READ) dev_info(&onenand->pdev->dev, "OneNAND Sync. Burst Read enabled\n"); err = mtd_device_parse_register(mtd, NULL, NULL, pdata ? pdata->parts : NULL, pdata ? pdata->nr_parts : 0); platform_set_drvdata(pdev, mtd); return 0; scan_failed: if (onenand->dma_addr) iounmap(onenand->dma_addr); dma_ioremap_failed: if (onenand->dma_res) release_mem_region(onenand->dma_res->start, resource_size(onenand->dma_res)); kfree(onenand->oob_buf); oob_buf_fail: kfree(onenand->page_buf); page_buf_fail: if (onenand->ahb_addr) iounmap(onenand->ahb_addr); ahb_ioremap_failed: if (onenand->ahb_res) release_mem_region(onenand->ahb_res->start, resource_size(onenand->ahb_res)); dma_resource_failed: ahb_resource_failed: iounmap(onenand->base); ioremap_failed: if (onenand->base_res) release_mem_region(onenand->base_res->start, resource_size(onenand->base_res)); resource_failed: kfree(onenand); onenand_fail: kfree(mtd); return err; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park94998.04%333.33%
dmitry eremin-baryshkovdmitry eremin-baryshkov70.72%222.22%
julia lawalljulia lawall60.62%111.11%
jingoo hanjingoo han40.41%111.11%
artem bityutskiyartem bityutskiy10.10%111.11%
jamie ilesjamie iles10.10%111.11%
Total968100.00%9100.00%


static int s3c_onenand_remove(struct platform_device *pdev) { struct mtd_info *mtd = platform_get_drvdata(pdev); onenand_release(mtd); if (onenand->ahb_addr) iounmap(onenand->ahb_addr); if (onenand->ahb_res) release_mem_region(onenand->ahb_res->start, resource_size(onenand->ahb_res)); if (onenand->dma_addr) iounmap(onenand->dma_addr); if (onenand->dma_res) release_mem_region(onenand->dma_res->start, resource_size(onenand->dma_res)); iounmap(onenand->base); release_mem_region(onenand->base_res->start, resource_size(onenand->base_res)); kfree(onenand->oob_buf); kfree(onenand->page_buf); kfree(onenand); kfree(mtd); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park146100.00%1100.00%
Total146100.00%1100.00%


static int s3c_pm_ops_suspend(struct device *dev) { struct platform_device *pdev = to_platform_device(dev); struct mtd_info *mtd = platform_get_drvdata(pdev); struct onenand_chip *this = mtd->priv; this->wait(mtd, FL_PM_SUSPENDED); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park52100.00%2100.00%
Total52100.00%2100.00%


static int s3c_pm_ops_resume(struct device *dev) { struct platform_device *pdev = to_platform_device(dev); struct mtd_info *mtd = platform_get_drvdata(pdev); struct onenand_chip *this = mtd->priv; this->unlock_all(mtd); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park50100.00%1100.00%
Total50100.00%1100.00%

static const struct dev_pm_ops s3c_pm_ops = { .suspend = s3c_pm_ops_suspend, .resume = s3c_pm_ops_resume, }; static const struct platform_device_id s3c_onenand_driver_ids[] = { { .name = "s3c6400-onenand", .driver_data = TYPE_S3C6400, }, { .name = "s3c6410-onenand", .driver_data = TYPE_S3C6410, }, { .name = "s5pc110-onenand", .driver_data = TYPE_S5PC110, }, { }, }; MODULE_DEVICE_TABLE(platform, s3c_onenand_driver_ids); static struct platform_driver s3c_onenand_driver = { .driver = { .name = "samsung-onenand", .pm = &s3c_pm_ops, }, .id_table = s3c_onenand_driver_ids, .probe = s3c_onenand_probe, .remove = s3c_onenand_remove, }; module_platform_driver(s3c_onenand_driver); MODULE_LICENSE("GPL"); MODULE_AUTHOR("Kyungmin Park <kyungmin.park@samsung.com>"); MODULE_DESCRIPTION("Samsung OneNAND controller support");

Overall Contributors

PersonTokensPropCommitsCommitProp
kyungmin parkkyungmin park493999.36%945.00%
arnd bergmannarnd bergmann90.18%210.00%
dmitry eremin-baryshkovdmitry eremin-baryshkov70.14%210.00%
julia lawalljulia lawall60.12%15.00%
jingoo hanjingoo han40.08%15.00%
axel linaxel lin20.04%15.00%
artem bityutskiyartem bityutskiy10.02%15.00%
krzysztof kozlowskikrzysztof kozlowski10.02%15.00%
kukjin kimkukjin kim10.02%15.00%
jamie ilesjamie iles10.02%15.00%
Total4971100.00%20100.00%
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
{% endraw %}