Nand Flash 的Linux 驱动讲解

关于nand flash 硬件的操作请参考

https://www.cnblogs.com/shwzh1990/p/12132333.html

 

1 S3c2410_nand_inithw : 这个函数是初始化硬件设备

2.S3c2410_nand_init_chip:这个函数是初始化芯片

3.nand_scan: 扫描芯片里面的内容 再次函数中我们做一下的事情

    (1)nand_scan_ident : 扫描芯片并且认证芯片

    (2)nand_get_flash_type: 查出芯片的是什么类型

     (3)chip->select_chip(mtd,0) : 片选此芯片

     (4)chip->cmdfunc(mtd, NAND_CMD_READID, 0X00, -1): 发送读写ID的信号

       *maf_id = chip->read_byte(mtd)

        dev_id = chip_read_byte(mtd)

     nand_scan_tail

S3c2410_nand_add_partition

  add_mtd_paritions  添加分区

           add_mtd_device 增加驱动             

                  list_for_each(this, &mtd_notifiers) {
                  struct mtd_notifier *not = list_entry(this, struct mtd_notifier, list);
                   not->add(mtd);
                   }

注意nandflash 既可以被用作字符设备也可以被用作块设备,当被用作块设备的时候就可以使用文件系统了。

在字符设备中,nandflash 将会被生成两个分区一个是只读分区一个是正常的分区

字符设备使用mtd_notify_add 来实现的

在看块设备:blktrans_notify_add

里面会把链表blktrans_majors的内容注册到内核。

内核也会生成只读和正常的设备块。

 

    

 

 

 

注意 这里面的nand_scan是一个通用的函数。

nand flash在linux中的架构如图所示:

 

 

 在完成硬件相关的设置时我们需要首先分配一个nand_chip 结构体 

设置硬件相关的驱动 然后使用nand_scan 去得到芯片的信息 之后用add_mtd_partition来分配分区

 

 

编写内核的整体思路是:

1. 根据nand_chip 的底层操作函数识别nand flash 构造mtd_info

2. mtd_info 会生成字符设备和块设备

1. 分配了一个nand_chip 结构体 这个结构式里面需要实现一些函数

    1. select_chip 片选的信号

    2. cmd_ctrl: nand flash 的如何发送命令

    3. IO_ADDR_R 读的地址

    4. IO_ADDR_W  写的地址

    5. dev_ready  nand flash的等待信号

    6. ecc.mode: NAND ECC nand的位校验

  2. 设置一个硬件的结构体,并且设置时间的参数

  3. 使能 时钟

 分配mtd的结构体。

  mtd和nand chip的链接函数是 priv。

 5 之后 利用 add_mtd_partitions 进行分区。 

    

#include <linux/module.h>
#include <linux/types.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/string.h>
#include <linux/ioport.h>
#include <linux/platform_device.h>
#include <linux/delay.h>
#include <linux/err.h>
#include <linux/slab.h>
#include <linux/clk.h>
 
#include <linux/mtd/mtd.h>
#include <linux/mtd/nand.h>
#include <linux/mtd/nand_ecc.h>
#include <linux/mtd/partitions.h>
 
#include <asm/io.h>
 
#include <asm/arch/regs-nand.h>
#include <asm/arch/nand.h>

struct s3c_nand_regs {
    unsigned long nfconf  ;
    unsigned long nfcont  ;
    unsigned long nfcmd   ;
    unsigned long nfaddr  ;
    unsigned long nfdata  ;
    unsigned long nfeccd0 ;
    unsigned long nfeccd1 ;
    unsigned long nfeccd  ;
    unsigned long nfstat  ;
    unsigned long nfestat0;
    unsigned long nfestat1;
    unsigned long nfmecc0 ;
    unsigned long nfmecc1 ;
    unsigned long nfsecc  ;
    unsigned long nfsblk  ;
    unsigned long nfeblk  ;
};



/* NAND parititon from 2.4.18-swl5 */

static struct mtd_partition smdk_default_nand_part[] = {
    [0] = {
        .name    = "Bootloader",
        .size    = SZ_16K,
        .offset    = 0,
    },
    [1] = {
        .name    = "params",
        .offset = MTDPART_OFS_APPEND,
        .size    = 0x00020000,
    },
    [2] = {
        .name    = "kernel",
        .offset = MTDPART_OFS_APPEND,
        .size    = 0x00200000,
    },
    [3] = {
        .name    = "root",
        .offset    = MTDPART_OFS_APPEND,
        .size    = MTDPART_SIZ_FULL,
    },

};


static struct nand_chip *s3c_nand;
static struct mtd_info *s3c_mtd;
static struct s3c_nand_regs *s3c_nand_regs;

static void s3c2440_select_chip(struct mtd_info *mtd, int chipnr)
{
    if (chipnr == -1)
    {
        /* È¡ÏûÑ¡ÖÐ: NFCONT[1]ÉèΪ1 */
        s3c_nand_regs->nfcont |= (1<<1);        
    }
    else
    {
        /* Ñ¡ÖÐ: NFCONT[1]ÉèΪ0 */
        s3c_nand_regs->nfcont &= ~(1<<1);
    }
}

static void s3c2440_cmd_ctrl(struct mtd_info *mtd, int dat, unsigned int ctrl)
{
    if (ctrl & NAND_CLE)
    {
        /* ·¢ÃüÁî: NFCMMD=dat */
        s3c_nand_regs->nfcmd = dat;
    }
    else
    {
        /* ·¢µØÖ·: NFADDR=dat */
        s3c_nand_regs->nfaddr = dat;
    }
}

static int s3c2440_dev_ready(struct mtd_info *mtd)
{
    return (s3c_nand_regs->nfstat & (1<<0));
}


static int s3c_nand_init(void)
{
    struct clk *clk;
    
    /* 1. ·ÖÅäÒ»¸önand_chip½á¹¹Ìå */
    s3c_nand = kzalloc(sizeof(struct nand_chip), GFP_KERNEL);

    s3c_nand_regs = ioremap(0x4E000000, sizeof(struct s3c_nand_regs));
    
    /* 2. ÉèÖÃnand_chip */
    /* ÉèÖÃnand_chipÊǸønand_scanº¯ÊýʹÓõÄ, Èç¹û²»ÖªµÀÔõôÉèÖÃ, ÏÈ¿´nand_scanÔõôʹÓà 
     * ËüÓ¦¸ÃÌṩ:Ñ¡ÖÐ,·¢ÃüÁî,·¢µØÖ·,·¢Êý¾Ý,¶ÁÊý¾Ý,ÅжÏ״̬µÄ¹¦ÄÜ
     */
    s3c_nand->select_chip = s3c2440_select_chip;
    s3c_nand->cmd_ctrl    = s3c2440_cmd_ctrl;
    s3c_nand->IO_ADDR_R   = &s3c_nand_regs->nfdata;
    s3c_nand->IO_ADDR_W   = &s3c_nand_regs->nfdata;
    s3c_nand->dev_ready   = s3c2440_dev_ready;
    s3c_nand->ecc.mode    = NAND_ECC_SOFT;
    /* 3. Ó²¼þÏà¹ØµÄÉèÖÃ: ¸ù¾ÝNAND FLASHµÄÊÖ²áÉèÖÃʱ¼ä²ÎÊý */
    /* ʹÄÜNAND FLASH¿ØÖÆÆ÷µÄʱÖÓ */
    clk = clk_get(NULL, "nand");
    clk_enable(clk);              /* CLKCON'bit[4] */
    
    /* HCLK=100MHz
     * TACLS:  ·¢³öCLE/ALEÖ®ºó¶à³¤Ê±¼ä²Å·¢³önWEÐźÅ, ´ÓNANDÊÖ²á¿ÉÖªCLE/ALEÓënWE¿ÉÒÔͬʱ·¢³ö,ËùÒÔTACLS=0
     * TWRPH0: nWEµÄÂö³å¿í¶È, HCLK x ( TWRPH0 + 1 ), ´ÓNANDÊÖ²á¿ÉÖªËüÒª>=12ns, ËùÒÔTWRPH0>=1
     * TWRPH1: nWE±äΪ¸ßµçƽºó¶à³¤Ê±¼äCLE/ALE²ÅÄܱäΪµÍµçƽ, ´ÓNANDÊÖ²á¿ÉÖªËüÒª>=5ns, ËùÒÔTWRPH1>=0
     */
#define TACLS    0
#define TWRPH0   1
#define TWRPH1   0
    s3c_nand_regs->nfconf = (TACLS<<12) | (TWRPH0<<8) | (TWRPH1<<4);

    /* NFCONT: 
     * BIT1-ÉèΪ1, È¡ÏûƬѡ 
     * BIT0-ÉèΪ1, ʹÄÜNAND FLASH¿ØÖÆÆ÷
     */
    s3c_nand_regs->nfcont = (1<<1) | (1<<0);
    
    /* 4. ʹÓÃ: nand_scan */
    s3c_mtd = kzalloc(sizeof(struct mtd_info), GFP_KERNEL);
    s3c_mtd->owner = THIS_MODULE;
    s3c_mtd->priv  = s3c_nand;
    
    nand_scan(s3c_mtd, 1);  /* ʶ±ðNAND FLASH, ¹¹Ôìmtd_info */
    
    /* 5. add_mtd_partitions */
    add_mtd_partitions(s3c_mtd, smdk_default_nand_part, 4);
    return 0;
}

static void s3c_nand_exit(void)
{ 
    del_mtd_partitions(s3c_mtd);
    kfree(s3c_mtd);
    iounmap(s3c_nand_regs);
    kfree(s3c_nand);
}

module_init(s3c_nand_init);
module_exit(s3c_nand_exit);

MODULE_LICENSE("GPL");

 

原文地址:https://www.cnblogs.com/shwzh1990/p/14134696.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


  译序:JWMediaPlayer是开源的网页使用的Flash播放器。本文采摘于JWPlayer的官方文档,讲解了JWPlayer对于RTMP的使用方法,我们可以从JWPlayer客户端的角度来了解RTMP协议。以下是官方原文:      简介    RTMP(RealTimeMessagingProtocol
    Flash编程原理都是只能将1写为0,而不能将0写成1.所以在Flash编程之前,必须将对应的块擦除,而擦除的过程就是将所有位都写为1的过程,块内的所有字节变为0xFF.因此可以说,编程是将相应位写0的过程,而擦除是将相应位写1的过程,两者的执行过程完全相反.一、Nor和NandFlash
 上传setenvgatewayip192.168.1.1;setenvserverip192.168.1.7;setenvipaddr192.168.1.156;mw.b0x820000000xff0x1000000sfprobe0sfread0x8200000000x1000000tftp0x82000000test.bin0x1000000 下载mw.b82000000ff1000000tftp82000000test.bi
Error:FlashDownloadFailed-"Cortex-M3"出现一般有两种情况:1.SWD模式下,Debug菜单中,Reset菜单选项(Autodetect/HWreset/sysresetReq/Vectreset)默认是AutoDetect,改成SysResetReq即可。2.Jtag模式下,主要是芯片大小选错。Flash->ConfigureFalshTools配置窗口,切换到“Utilities"
jPlayer是一个用于控制和播放mp3文件的jQuery插件。它在后台使用Flash来播放mp3文件,前台播放器外观完全可以使用XHML/CSS自定义。支持:有一点比较好的是,在支持html5的浏览器上会使用html5的标签audio或者video,而不支持的浏览器上使用swf来播放选择需要播放的Mp3文件。播放、暂停
#ifndef__FONTUPD_H__#define__FONTUPD_H__#include"sys.h" //字库信息结构体定义33字节__packedtypedefstruct{u8fontok;//字库存在标志,0XAA,字库正常;其他,字库不存在u32ugbkaddr;//unigbk的地址u32ugbksize;//unigbk的大小u32f12addr;//gbk12地址u32g
ROM(ReadOnlyMemory)和RAM(RandomAccessMemory)指的都是半导体存储器。ROM在系统停止供电的时候仍然可以保持数据,而RAM通常都是在掉电之后就丢失数据,但是访问速度快。典型的RAM就是计算机的内存。RAM有两大类,一种称为静态RAM(StaticRAM/SRAM),SRAM速度非常快,是目前读写最快的存储
JSpc端和移动端实现复制到剪贴板功能实现在网页上复制文本到剪切板,一般是使用JS+Flash结合的方法,网上有很多相关文章介绍。随着HTML5技术的发展,Flash已经在很多场合不适用了,甚至被屏蔽。本文介绍的一款JS插件,实现了纯JS方法复制文本到剪切板。插件名是Clipboard.js,该插件不依
例子:R0=1R1=1R2=10R3=e000ed10R12=0LR=fffffff9(中断返回值)PC=0PSR=60000013或60000016或60000036(Z、C、EXCEPT_NUM:RTC_WKUP_IRQn、EXTI0_IRQn、USART2_IRQn)BFAR=e000ed38(不关心)CFSR=20000(INVSTATE:Invalidstateusagefault thePCvaluestackedf
 内存接口概念首先来分析下操作GPIO控制器和操作UART控制器两者的区别如图是S3C2440是个片上系统,有GPIO控制器(接有GPIO管脚),有串口控制器(接有TXDRXD引脚)配置GPIO控制器相应的寄存器,即可让引脚输出高低电平;配置UART控制器相应的寄存器,即可让引脚输出波形。前者相对简单,类
小编导语:    近几年来,网页游戏成为了游戏界关注的焦点,由于其制作简单,成本低并且收益率较高,因此成为了众多游戏厂商追逐的对象,但是除了商家夸张的炒作宣传外,很少有页游佳作出现。然而,随着Unity3D游戏引擎的出现,网页游戏的3D化成了页游冲出重围的杀手锏,那么在flash网页游戏称
1.指定数组到特定的Flash单元#pragmalocation=0x000FFF00 __rootconstcharFlash_config[]={0x0,0x1,0x2,0x3,0x4,0x5,0x6,0x7,0x8,0x9,0xA,0xB,0xC,0xD,0xE,0xF,0x10,0x11,0x12,0x13,0x14,0x15,0x16,0x17,0x18,0x19,0x1A,0x1B,0x1C,0x1D,0x1E,0x1F,0x20,0x21,0x22,0x23,0
继续研究发现,计算机的固件真的很有趣。参考了一些重要的资料,比如http://donovan6000.blogspot.com/2013/06/insyde-bios-modding-advanced-and-power-tabs.html等,对于IDA的使用也了解了一些。最后,总结一下目前看来可行性的方案:0.基础知识储备,包括UEFIBIOS的概念,InsydeBIOS的
<!DOCTYPEhtml><htmllang="en"><head><metacharset="UTF-8"><title>navigator对象<itle></head><body><buttononclick="checkFlash()">检测</button>
修改网上流传的flash-marker.js(function(global,factory){typeofexports==='object'&&typeofmodule!=='undefined'?module.exports=factory():typeofdefine==='function'&&define.amd?define(factory
shareObject本地缓存存储位置:win7系统用户到C:\Users\[你的用户名]\AppData\Roaming\Macromedia\FlashPlayer\#SharedObjects\XP或2003用户到:C:\DocumentsandSettings\用户名\ApplicationData\Macromedia\FlashPlayer\#SharedObjects\ ---------------------作者:iteye_
安装谷歌浏览器之后经常遇到Flash崩溃或者浏览器在浏览Flash内容时卡死的情况。在网上查找资料大多都认为应该是浏览器自带的Flash插件工作模式引起的问题,解决方法如下:首先在地址栏输入chrome://plugins/显示浏览器使用的插件。点击右上角的详细信息,可以看到Flash插件为进程外
之前一直使用的W25Q16spiflash都没问题,换了一款W25Q80后发现工作不正常,经过测试,初步定位到问题在于初始化SPI后是否将CS拉高。于是又去查看了一下原厂代码:发现原厂的代码初始化SPI接口时是专门拉高CS的。结论:网上很多代码初始化SPI接口时没有专门拉高CS,对某些型号可能确实
======================================================NANDFlash最小存储单元:写数据操作:通过对控制闸(ControlGate)施加高电压,然后允许源极(SOURCE)和汲极(RRAIN)间的N信道(N-Channel)流入电子,等到电流够强,电子获得足够能量时,便会越过浮置闸(FloatingGate)底下的二氧化硅层(S
安装CnarioPlayer3.8.1.156或其他版本时,有时会出现如下提示:Warning4154.AdobeFlashPlayer13...notcorrectlyinstalled:请前往AdobeFlash网站,并选择下图示的版本下载安装: