s3c2440裸机-内存控制器(三-4、norflash编程之实现)

发布者:EtherealGrace最新更新时间:2023-08-09 来源: elecfans关键字:s3c2440  裸机  内存控制器 手机看文章 扫描二维码
随时随地手机看文章

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  裸机  内存控制器 引用地址:s3c2440裸机-内存控制器(三-4、norflash编程之实现)

上一篇:s3c2440裸机-nandflash编程(一. nandflash原理及结构简介)
下一篇:s3c2440裸机-内存控制器(三-3、norflash编程之uboot中操作norflash)

推荐阅读最新更新时间:2024-11-10 16:22

s3c2440裸机开发调试环境(MDK4.6,Jlink v8,mini2440)
用于arm裸机程序开发的IDE基本有 以下3个:MDK,IAR,还有ADS。具体它们的具体情况在这里我就不多说了,百度一下就明白了。由于之前开发c51,stm32时候都使用了MDK开发环境,而且MDK的界面确实看起来舒服多了,所以我选择了MDK作为我的s3c2440裸机开发的IDE。以下主要介绍一下如何使用MDK配合J-link来调试基于s3c2440的开发板。 首先,我们需要下载的有以下2样: MDK J-link 驱动 我的开发环境:windows 7 64位,J-linkv8,mini2440开发板一块。最后选择了:MDK4.6版本,J-ink v8的当前最新版本J-linkARM v4.56。 MDK的大部
[单片机]
<font color='red'>s3c2440</font><font color='red'>裸机</font>开发调试环境(MDK4.6,Jlink v8,mini2440)
S3C2440裸板程序设计(无操作系统)的抽象
在嵌入式领域中,几乎所有的设备控制和各种协议控制都在同一个嵌入式CPU当中,非常有利于对CPU Core和设备进行抽象。如果能对CPU Core和设备的各种控制进行抽象,人们在移植OS或者开发驱动程序时就没有必要对CPU进行非常深入的了解,不必要了解某个寄存器的某个位是控制什么的,也没有必要了解怎样初始化某个控制寄存器等等。 在利用控制器编写裸板应用程序时,编程人员只需要了解该控制器的初始化顺序以及初始化的内容而不需要了解初始化的具体细节就能完成应用程序。显然可以大大的提高工作效率,并且对于硬件的具体细节设置是在应用程序中最容易出错的地方,而利用控制器则可以大大的减少出错的可能性。 ARM是功能更加强大的单
[单片机]
S3C2440之ADC
S3C2440包含一个8通道A/D转换器,有10位分辨率下面简要介绍一下S3C2440中ADC的用法: 用到的寄存器: ADCCON:ADC控制寄存器 ADCDAT0:ADC转换数据寄存器 SUBSRCPND:次级源挂起寄存器 INTSUBMSK:中断次级屏蔽寄存器 SRCPND:源挂起寄存器 INTPND:中断挂起寄存器 INTMSK:中断屏蔽寄存器 ADCCON:AD控制寄存器 ADCDAT0:AD转换数据寄存器 1,首先设置控制寄存器ADCCON的相应位来选择频率和通道: view plain copy rADCCON = (1 14)|(preScale
[单片机]
<font color='red'>S3C2440</font>之ADC
S3C2440中断体系结构
一、S3C2440中断体系结构 1)ARM体系CPU的7种工作模式 用户模式(usr):ARM处理器正常的程序执行状态 快速中断模式(fiq):用于高速数据传输或通道处理 中断模式(irq):用于通用的中断处理 管理模式(svc):操作系统使用的保护模式 数据访问终止模式(abt):当数据或指令预取终止时进入该模式,可用于虚拟存储及存储保护 系统模式(sys):运行具有特权的操作系统任务 未定义指令中止模式(und):当未定义的指令执行时进入该模式,可用于支持硬件协处理器的软件仿真 除用户模式外,其他6种工作模式都属于特权模式,大多数程序运行于用户模式,进入特权模式是为了处理中断、异常,或者访问被保护的系统资源。
[单片机]
<font color='red'>S3C2440</font>中断体系结构
OK6410裸机学习之串口实验
start.S文件: .globl _start _start: // 硬件相关的设置 // Peri port setup ldr r0, =0x70000000 orr r0, r0, #0x13 mcr p15,0,r0,c15,c2,4 @ 256M(0x70000000-0x7fffffff) // 关看门狗 // 往WTCON(0x7E004000)写0 ldr r0, =0x7E004000 mov r1, #0 str r1, // 设置栈 ldr sp, =8*1024 // 设置时钟 bl clock_init bl main
[单片机]
OK6410裸机之串口发送中断
start.S源码: .globl _start _start: // 0 地址 b reset // 复位时,cpu跳到0地址 ldr pc, =undefined_instruction // cpu遇到不能识别的指令时 ldr pc, _vector_swi // 当执行swi指令时, 进入swi模 式 b halt @ldr pc, _prefetch_abort // 预取中止异常 b halt @ldr pc, _data_abort // 数据访问异常 b halt @ld
[单片机]
S3C2440学习笔记一(系统频率以及一些入门级问题)
①MPLL和UPLL有何不同? 对于MPLL使用到MPLLCON,而UPLL使用到UPLLCON寄存器,同属于时钟电源管理单元。MPLLCON地址是0X4C000004,UPLL地址是0X4C000008。MPLL用于CPU及其他外围器件,这里把他理解成单片机的主频就对了,UPLL用于USB。产生FCLK,HCLK,PCLK三种频率。 CLKDIVN时钟分配控制寄存器,地址是0X4C000014,对FCLK分频给UCLK,PCLK,不分频或2分频,CAMDIVN照相机时钟分配寄存器,地址是0X4C000018,也是做一个时钟分配。 ②2440slib.s中出现mrc,bic,mcr,MOV_PC_LR是什么意思,有什么资料学习和
[单片机]
<font color='red'>S3C2440</font>学习笔记一(系统频率以及一些入门级问题)
S3C2440 LED驱动
//头文件 #include linux/miscdevice.h #include linux/delay.h #include asm/irq.h #include mach/regs-gpio.h #include mach/hardware.h #include linux/kernel.h #include linux/module.h #include linux/init.h #include linux/mm.h #include linux/fs.h #include linux/types.h #include linux/delay.h #include linux/moduleparam.h #includ
[单片机]
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

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

换一换 更多 相关热搜器件
随便看看

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

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