s3c2440裸机-norflash4-编程实现

发布者:upsilon30最新更新时间:2024-07-08 来源: elecfans关键字:编程 手机看文章 扫描二维码
随时随地手机看文章

1.识别norflash

我们知道要识别norflash属性,要让norflash进入cfi模式,然后按照手册上的表格发送一系列的命令就能获取norflash属性。


1)发送命令


那么我们需要实现一个cpu向nor发命令的一个函数nor_cmd()。我们的norflash是16bit位宽的,所以访问nor是以16位为单位访问的。


#define NOR_FLASH_BASE  0  /* jz2440, nor-->cs0, base addr = 0 */

/* 比如:   55H 98 

** 本意是: 往(0 + (0x55)<<1)写入0x98

*/

void nor_write_word(unsigned int base, unsigned int offset, unsigned int val)

{

    volatile unsigned short *p = (volatile unsigned short *)(base + (offset << 1));

    *p = val;

}


void nor_cmd(unsigned int offset, unsigned int cmd)

{

    nor_write_word(NOR_FLASH_BASE, offset, cmd);

}

我们通过调用nor_cmd就可以向flash的指定地址写入命令。


2)

unsigned int nor_read_word(unsigned int base, unsigned int offset)

{

    volatile unsigned short *p = (volatile unsigned short *)(base + (offset << 1));

    return *p;

}


unsigned int nor_dat(unsigned int offset)

{

    return nor_read_word(NOR_FLASH_BASE, offset);

}

这样我们调用nor_dat(addr)就可以获得norflash中addr地址处的数据,返回的数据是16bit(1 word)。


3)识别函数

那么现在有了发命令函数nor_cmd和读数据函数nor_dat,那么就就可以参考nor芯片手册的命令表进行操作norflash了。

/* 进入NOR FLASH的CFI模式

 * 读取flash属性

 */

void do_scan_nor_flash(void)

{

    char str[4];

    unsigned int size;

    int regions, i;

    int region_info_base = 0x2d; //第0块region的基地址2d,第1块region的基地址31,第2块region的基地址35......(参考手册表4-3)

    int block_addr=0, blocks, block_size, j;

    int cnt = 0;


    int vendor, device;


    /* 打印厂家ID、设备ID */

    nor_cmd(0x555, 0xaa);    /* 解锁 */

    nor_cmd(0x2aa, 0x55); 

    nor_cmd(0x555, 0x90);    /* read id */

    vendor = nor_dat(0);

    device = nor_dat(1);

    nor_cmd(0, 0xf0);        /* reset */


    nor_cmd(0x55, 0x98);  /* 进入cfi模式 */

    str[0] = nor_dat(0x10); //读地址10H得到0051('q')

    str[1] = nor_dat(0x11); //读地址11H得到0052('r')

    str[2] = nor_dat(0x12); //读地址12H得到0059('y')

    str[3] = '';

    printf('str = %snr', str);


    /* 打印容量 */

    size = 1<<(nor_dat(0x27));

    printf('vendor id = 0x%x, device id = 0x%x, nor size = 0x%x = %dMnr', vendor, device, size, size/(1024*1024));


    /* 打印各个扇区的起始地址 */

    /* 名词解释:

     * region : 一个nor flash含有1个或多个region, 一个region含有1个或多个block(扇区).

     * Erase block region[i] information:

     *    前2字节+1    : 表示该region有多少个block 

     *    后2字节*256  : 表示block的大小

    */


    printf('Block/Sector start Address:nr');

    regions = nor_dat(0x2c);  //读出region数量

    for (i = 0; i < regions; i++)

    {

        blocks = 1 + nor_dat(region_info_base) + (nor_dat(region_info_base+1)<<8);

        block_size = 256 * (nor_dat(region_info_base+2) + (nor_dat(region_info_base+3)<<8));


        printf('nrregion %d, blocks = %d, block_size = 0x%x, block_addr = 0x%xnr', i, blocks, block_size, block_addr);


        for (j = 0; j < blocks; j++)

        {

            /* 打印每个block的起始地址 */

            printHex(block_addr);

            putchar(' ');

            cnt++;

            if (cnt % 5 == 0)

                printf('nr');


            block_addr += block_size;

        }


        region_info_base += 4;  /*得到region[i]的基地址*/

    }

    printf('nr');

    /* 退出CFI模式 */

    nor_cmd(0, 0xf0);

}

测试结果如下:

从测试结果来看每个region的block个数和block_size不一定一样,像region[0]只有一个block,block_size为4*64K; region[1]有2个block,block_size=2*64K。


2.读NOR Flash

由于NOR Flash是内存类接口,可以像内存一样读取,那么do_read_nor_flash函数代码如下:


void do_read_nor_flash(void)

{

    unsigned int addr;

    volatile unsigned char *p;

    int i, j;

    unsigned char c;

    unsigned char str[16];


    /* 获得地址 */

    printf('Enter the address to read: ');

    addr = get_uint();


    p = (volatile unsigned char *)addr;


    printf('Data : nr');


    for (i = 0; i < 4; i++)

    {

        for (j = 0; j < 16; j++)

        {

            c = *p++;

            str[j] = c;

            printf('%02x ', c);

        }


        printf('   ; ');


        for (j = 0; j < 16; j++)

        {

            if (str[j] < 0x20 || str[j] > 0x7e)  /* 不可视字符 */

                putchar('.');

            else

                putchar(str[j]);

        }

        printf('nr');

    }

}

3.擦norflash

norflash擦写都是需要一定时间的,那么当我执行擦除或者写入动作后什么时候代表一次擦写动作已经完成了呢?


芯片手册提供了一个方法,每次擦除或者烧写过程中都可以查询数据总线上的第6位(Q6),当它保持稳定的时候表示一次擦除或者烧写动作完成,如下图:

void wait_ready(unsigned int addr)

{

    unsigned int val;

    unsigned int pre;


    pre = nor_dat(addr>>1);

    val = nor_dat(addr>>1);

    while ((val & (1<<6)) != (pre & (1<<6)))

    {

        pre = val;

        val = nor_dat(addr>>1);     

    }

}


void do_erase_nor_flash(void)

{

    unsigned int addr;


    printf('Enter the address of sector to erase: ');

    addr = get_uint();


    printf('erasing ...nr');

    nor_cmd(0x555, 0xaa);    /* 解锁 */

    nor_cmd(0x2aa, 0x55); 

    nor_cmd(0x555, 0x80);    /* erase sector */


    nor_cmd(0x555, 0xaa);    /* 解锁 */

    nor_cmd(0x2aa, 0x55); 

    nor_cmd(addr>>1, 0x30);  /* 发出扇区地址 */

    wait_ready(addr);

}


4.写norflash

void do_write_nor_flash(void)

{

    unsigned int addr;

    unsigned char str[100];

    int i, j;

    unsigned int val;


    /* 获得地址 */

    printf('Enter the address of sector to write: ');

    addr = get_uint();


    printf('Enter the string to write: ');

    gets(str);


    printf('writing ...nr');


    /* str[0],str[1]==>16bit 

     * str[2],str[3]==>16bit 

     */

    i = 0;

    j = 1;

    while (str[i] && str[j])

    {

        val = str[i] + (str[j]<<8);


        /* 烧写 */

        nor_cmd(0x555, 0xaa);    /* 解锁 */

        nor_cmd(0x2aa, 0x55); 

        nor_cmd(0x555, 0xa0);    /* program */

        nor_cmd(addr>>1, val);

        /* 等待烧写完成 : 读数据, Q6无变化时表示结束 */

        wait_ready(addr);


        i += 2;

        j += 2;

        addr += 2;

    }


    val = str[i];

    /* 烧写 */

    nor_cmd(0x555, 0xaa);    /* 解锁 */

    nor_cmd(0x2aa, 0x55); 

    nor_cmd(0x555, 0xa0);    /* program */

    nor_cmd(addr>>1, val);

    /* 等待烧写完成 : 读数据, Q6无变化时表示结束 */

    wait_ready(addr);

}

由于我的norflash是位宽为16bit的,所以我们上面代码do_write_nor_flash进行写入时是以2byte(wold)为单位进行写入的。


关键字:编程 引用地址:s3c2440裸机-norflash4-编程实现

上一篇:s3c2440裸机-代码重定位-1-重定位的引入
下一篇:s3c2440裸机-norflash3-uboot中操作norflash

推荐阅读最新更新时间:2026-03-20 15:08

s3c2440裸机-时钟编程-2-配置时钟寄存器
1.2440时钟时序 下图是2440时钟配置时序: 1.上电后,nRESET复位信号拉低,此时cpu还无法取指令工作。 2.nRESET复位信号结束后变为高电平,此时cpu开始工作。此时cpu主频FCLK=osc。 3.此时可以配置PLL,经过lock time后,FCLK倍频成新的时钟。 2.如何配置时钟 在参考手册的特性里介绍了S3C2440的工作频率,Fclk最高400MHz,Hclk最高136MHz,Pclk最高68MHz。那么 我们干脆配置FCLK:HCLK:PCLK= 400:100:50 (MHz). 1,先配置lock time 我们取芯片手册上的推荐值。
[单片机]
s3c2440裸机-UART编程-2-UART编程实现
UART编程 1.初始化 我们的2440支持3个UART串口,以uart0为例讲解。 那么我们需要实现以下这几个函数完成串口的最基本功能: (1)uart0_init()用于初始化串口 (2)putchar()用于发送一个字符 (3)getchar()用于接收一个字符 (4)puts()用于发送一串字符 1.uart0_init() 1.配置uart0引脚 (1)根据原理图GPH2,3用于TxD0, RxD0。 (2)查看dataset,配置GPH控制寄存器,让GPH2,3配成uart模式;为了将其保持为高电平,先设置其为上拉。 GPHCON &= ~((3 4) | (3 6)); G
[单片机]
s3c2440裸机-LCD编程-1-LCD硬件原理
1.LCD示意图 下图是LCD示意图,里面的每个点就是一个像素点。 它里面有一个电子枪,一边移动,一边发出各种颜色的光。用动态图表示如下: 电子枪是如何移动的? 有一条CLK时钟线与LCD相连,每发出一次CLK(高低电平),电子枪就移动一个像素。 颜色如何确定? 由连接LCD的三组线RGB三原色混合而成:R(Red)、G(Green)、B(Blue)确定。 电子枪如何得知应跳到下一行? 有一条HSYNC信号线与LCD相连,每发出一次脉冲(高低电平),电子枪就跳到下一行,该信号叫做行同步信号。 电子枪如何得知应跳到原点? 有一条VSYNC信号线与LCD相连,每发出一次脉冲(高低电平),电子枪就跳到原点,该信
[单片机]
<font color='red'>s3c2440</font><font color='red'>裸机</font>-LCD<font color='red'>编程</font>-1-LCD硬件原理
s3c2440裸机-LCD编程-3-框架准备和LCD初始化
1.准备框架 为了让程序更加好扩展,体现出”高内聚、低耦合 的特点,能够兼容各种不同型号的lcd,假如有两款尺寸大小的lcd,如何快速的在两个lcd上切换? 首先我们抽象出lcd_3.5.c和lcd_4.3.c的共同点,比如都有初始化函数init(),我们可以新建一个lcd.c,然后定义一个结构体: struct lcd_opr{ void (*init)(void); }; 用户不接触lcd_3.5.c和lcd_4.3.c,只需要在lcd.c里通过指针访问对应的结构体的函数,也就调用了不同init(),如下图所示: 我们的目的是在LCD显示屏上画线、画圆(geomentry.c)和写字(font.c)其核心是画点(
[单片机]
s3c2440裸机-LCD编程-5-LCD上实现画点线圆
1.画点 无论是何种图形,都是基于点来构成的,因此我们需要先实现画点,其他的都是上层的一些数据处理了,像各种图形、甚至色彩鲜艳的图片无非都是一些由点构造出的数据而已。 我们在在farmebuffer.c实现画点,在geomentry.c实现画线、画圆等几何图形,font.c实现画字。 那么一个像素点要显示到lcd上,我们要知道它的位置坐标,然后还要知道它的颜色值,假设该像素点的坐标为(x,y),那么该像素的地址为: (x,y)= fb_base + (xres*(bpp/8))*y +x*bpp/8; 那么所以在画点前需要先获取lcd参数:fb_base、xres、yres、bpp; static unsigned
[单片机]
s3c2440裸机-时钟编程(二、配置时钟寄存器)
1.2440时钟时序 下图是2440时钟配置时序: 1.上电后,nRESET复位信号拉低,此时cpu还无法取指令工作。 2.nRESET复位信号结束后变为高电平,此时cpu开始工作。此时cpu主频FCLK=osc。 3.此时可以配置PLL,经过lock time后,FCLK倍频成新的时钟。 2.如何配置时钟 在参考手册的特性里介绍了S3C2440的工作频率,Fclk最高400MHz,Hclk最高136MHz,Pclk最高68MHz。那么 我们干脆配置FCLK:HCLK:PCLK= 400:100:50 (MHz). 1,先配置lock time 我们取芯片手册上的推荐值。 /* LOCKTIME(0x4C000000)
[单片机]
<font color='red'>s3c2440</font><font color='red'>裸机</font>-时钟<font color='red'>编程</font>(二、配置时钟寄存器)
s3c2440裸机-LCD编程(四、实现显示功能)
1.让LCD显示满屏红色 我们想要在LCD上显示出数据,所需步骤如下: a. 初始化LCD b. 使能LCD c. 获取LCD参数: fb_base, xres, yres, bpp d. 往framebuffer中写数据 1.初始化LCD工作我们在前面一节框架准备和LCD初始化已经详细介绍了。 2.使能LCD void lcd_enable() { lcd_controller_enalbe(); //会间接调用s3c2440_lcd_controller_enalbe } 3.获取LCD参数 void get_lcd_params(unsigned int *fb_base, int *xres, int
[单片机]
s3c2440裸机-电阻触摸屏编程(2.触摸屏TS控制器)
触摸屏接口模式 Normal Conversion Mode: 正常转换模式,一般情况下可以配置ADCCON和ADCDAT0来读取数据。 Separate X/Y position conversion Mode: x,y坐标分离转换格式,x坐标会写入ADCDAT0, y坐标会写入ADCDAT1,所以会产生2次中断开分开完成x,y的坐标转换。 Auto(Sequential) X/Y Position Conversion Mode 自动转换模式,当触摸屏按下后,会一次性对x,y方向的坐标进行转换,x坐标会写入ADCDAT0, x坐标会写入ADCDAT1。会产生一次中断进行x,y坐标的自动转换。 Waiting
[单片机]
<font color='red'>s3c2440</font><font color='red'>裸机</font>-电阻触摸屏<font color='red'>编程</font>(2.触摸屏TS控制器)
小广播
最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

厂商技术中心

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

 
机器人开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2026 EEWORLD.com.cn, Inc. All rights reserved