當(dāng)前位置:首頁(yè) > 單片機(jī) > 單片機(jī)
[導(dǎo)讀]接上一篇:s3c2440硬件篇之三:NandFlash(1)介紹s3c2440讀NAND Flash的步驟:①設(shè)置NFCONF 在HCLK=100Mhz的情況下,TACLS=0,TWRPH0=3,TWRPH1=0,則 NFCONF = 0x300 使能NAND Flash控制器、禁止控制引腳信號(hào)nFCE,

接上一篇:s3c2440硬件篇之三:NandFlash(1)介紹

s3c2440讀NAND Flash的步驟:

①設(shè)置NFCONF
在HCLK=100Mhz的情況下,TACLS=0,TWRPH0=3,TWRPH1=0,則
NFCONF = 0x300
使能NAND Flash控制器、禁止控制引腳信號(hào)nFCE,初始化ECC
NFCONT = (1<<4) | (1<<1) | (1<<0)
②操作NAND Flash前,復(fù)位
NFCONT &= ~(1<<1) 發(fā)出片選信號(hào)
NFCMD = 0xff reset命令
然后循環(huán)查詢NFSTAT位0,直到等于1,處于就緒態(tài)
最后禁止片選信號(hào),在實(shí)際使用時(shí)再使能
NFCONT |= 0x2 禁止NAND Flash
③發(fā)出讀命令
NFCONT &= ~(1<<1) 發(fā)出片選信號(hào)
NFCMD = 0 讀命令
④發(fā)出地址信號(hào)
⑤循環(huán)查詢NFSTAT,直到等于1
⑥連續(xù)讀NFDATA寄存器,得到一頁(yè)數(shù)據(jù)
⑦最后禁止NAND Flash片選信號(hào)
NFCONT |= (1<<1)

Nand硬件連接:

代碼詳解:(參考韋東山大哥代碼)

說(shuō)明:本實(shí)例的目的是把一部分代碼存放在NAND Flash地址4096(即4K)之后,當(dāng)程序啟動(dòng)后通過(guò)NAND Flash控制器讀出代碼,執(zhí)行。

(1)連接腳本nand.lds

SECTIONS {
firtst 0x00000000:{ head.o init.o nand.o}
second 0x30000000:AT(4096){ main.o }
}

(2)head.S

@******************************************************************************
@ File:head.s
@ 功能:設(shè)置SDRAM,將程序復(fù)制到SDRAM,然后跳到SDRAM繼續(xù)執(zhí)行
@******************************************************************************

.text
.global _start
_start:
@函數(shù)disable_watch_dog,memsetup,init_nand,nand_read_ll在init.c中定義
ldr sp,=4096 @設(shè)置堆棧
bldisable_watch_dog @關(guān)WATCH DOG
blmemsetup @初始化SDRAM
blnand_init @初始化NAND Flash

@將NAND Flash中地址4096開(kāi)始的1024字節(jié)代碼(main.c編譯得到)復(fù)制到SDRAM中
@nand_read_ll函數(shù)需要3個(gè)參數(shù):
ldr r0,=0x30000000 @1.目標(biāo)地址=0x30000000,這是SDRAM的起始地址
movr1,#4096 @2.源地址=4096,連接的時(shí)候,main.c中的代碼都存在NAND Flash地址4096開(kāi)始處
movr2,#2048 @3.復(fù)制長(zhǎng)度=2048(bytes),對(duì)于本實(shí)驗(yàn)的main.c,這是足夠了
blnand_read @調(diào)用C函數(shù)nand_read

ldr sp,=0x34000000 @設(shè)置棧
ldr lr,=halt_loop @設(shè)置返回地址
ldr pc,=main @b指令和bl指令只能前后跳轉(zhuǎn)32M的范圍,所以這里使用向pc賦值的方法進(jìn)行跳轉(zhuǎn)
halt_loop:
b halt_loop

(3)init.c

/* WOTCH DOG register */
#defineWTCON(*(volatileunsignedlong*)0x53000000)

/* SDRAM regisers */
#defineMEM_CTL_BASE0x48000000

voiddisable_watch_dog();
voidmemsetup();

/*上電后,WATCH DOG默認(rèn)是開(kāi)著的,要把它關(guān)掉 */
voiddisable_watch_dog()
{
WTCON=0;
}

/* 設(shè)置控制SDRAM的13個(gè)寄存器 */
voidmemsetup()
{
inti=0;
unsignedlong*p=(unsignedlong*)MEM_CTL_BASE;

/* SDRAM 13個(gè)寄存器的值 */
unsignedlongconstmem_cfg_val[]={0x22011110,//BWSCON

0x00000700,//BANKCON0

0x00000700,//BANKCON1

0x00000700,//BANKCON2

0x00000700,//BANKCON3

0x00000700,//BANKCON4

0x00000700,//BANKCON5

0x00018005,//BANKCON6

0x00018005,//BANKCON7

0x008C07A3,//REFRESH

0x000000B1,//BANKSIZE

0x00000030,//MRSRB6

0x00000030,//MRSRB7

};

for(;i<13;i++)
p[i]=mem_cfg_val[i];
}


(4)nand.c ,定義一個(gè)宏#defineLARGER_NAND_PAGE來(lái)區(qū)分大頁(yè)小頁(yè)。


#defineLARGER_NAND_PAGE

#defineGSTATUS1(*(volatileunsignedint*)0x560000B0)
#defineBUSY 1

#defineNAND_SECTOR_SIZE 512
#defineNAND_BLOCK_MASK(NAND_SECTOR_SIZE-1)

#defineNAND_SECTOR_SIZE_LP 2048
#defineNAND_BLOCK_MASK_LP(NAND_SECTOR_SIZE_LP-1)

typedefunsignedintS3C24X0_REG32;


/* NAND FLASH (see S3C2410 manual chapter 6) */
typedefstruct{
S3C24X0_REG32 NFCONF;
S3C24X0_REG32 NFCMD;
S3C24X0_REG32 NFADDR;
S3C24X0_REG32 NFDATA;
S3C24X0_REG32 NFSTAT;
S3C24X0_REG32 NFECC;
}S3C2410_NAND;

/* NAND FLASH (see S3C2440 manual chapter 6, www.100ask.net) */
typedefstruct{
S3C24X0_REG32 NFCONF;
S3C24X0_REG32 NFCONT;
S3C24X0_REG32 NFCMD;
S3C24X0_REG32 NFADDR;
S3C24X0_REG32 NFDATA;
S3C24X0_REG32 NFMECCD0;
S3C24X0_REG32 NFMECCD1;
S3C24X0_REG32 NFSECCD;
S3C24X0_REG32 NFSTAT;
S3C24X0_REG32 NFESTAT0;
S3C24X0_REG32 NFESTAT1;
S3C24X0_REG32 NFMECC0;
S3C24X0_REG32 NFMECC1;
S3C24X0_REG32 NFSECC;
S3C24X0_REG32 NFSBLK;
S3C24X0_REG32 NFEBLK;
}S3C2440_NAND;


typedefstruct{
void(*nand_reset)(void);
void(*wait_idle)(void);
void(*nand_select_chip)(void);
void(*nand_deselect_chip)(void);
void(*write_cmd)(intcmd);
void(*write_addr)(unsignedintaddr);
unsignedchar(*read_data)(void);
}t_nand_chip;

staticS3C2410_NAND*s3c2410nand=(S3C2410_NAND*)0x4e000000;
staticS3C2440_NAND*s3c2440nand=(S3C2440_NAND*)0x4e000000;

statict_nand_chip nand_chip;

/* 供外部調(diào)用的函數(shù) */
voidnand_init(void);
voidnand_read(unsignedchar*buf,unsignedlongstart_addr,intsize);

/* NAND Flash操作的總?cè)肟? 它們將調(diào)用S3C2410或S3C2440的相應(yīng)函數(shù) */
staticvoidnand_reset(void);
staticvoidwait_idle(void);
staticvoidnand_select_chip(void);
staticvoidnand_deselect_chip(void);
staticvoidwrite_cmd(intcmd);
staticvoidwrite_addr(unsignedintaddr);
staticunsignedcharread_data(void);

/* S3C2410的NAND Flash處理函數(shù) */
staticvoids3c2410_nand_reset(void);
staticvoids3c2410_wait_idle(void);
staticvoids3c2410_nand_select_chip(void);
staticvoids3c2410_nand_deselect_chip(void);
staticvoids3c2410_write_cmd(intcmd);
staticvoids3c2410_write_addr(unsignedintaddr);
staticunsignedchars3c2410_read_data();

/* S3C2440的NAND Flash處理函數(shù) */
staticvoids3c2440_nand_reset(void);
staticvoids3c2440_wait_idle(void);
staticvoids3c2440_nand_select_chip(void);
staticvoids3c2440_nand_deselect_chip(void);
staticvoids3c2440_write_cmd(intcmd);
staticvoids3c2440_write_addr(unsignedintaddr);
staticunsignedchars3c2440_read_data(void);

/* S3C2410的NAND Flash操作函數(shù) */

/* 復(fù)位 */
staticvoids3c2410_nand_reset(void)
{
s3c2410_nand_select_chip();
s3c2410_write_cmd(0xff);// 復(fù)位命令

s3c2410_wait_idle();
s3c2410_nand_deselect_chip();
}

/* 等待NAND Flash就緒 */
staticvoids3c2410_wait_idle(void)
{
inti;
volatileunsignedchar*p=(volatileunsignedchar*)&s3c2410nand->NFSTAT;
while(!(*p&BUSY))
for(i=0;i<10;i++);
}

/* 發(fā)出片選信號(hào) */
staticvoids3c2410_nand_select_chip(void)
{
inti;
s3c2410nand->NFCONF&=~(1<<11);
for(i=0;i<10;i++);
}

/* 取消片選信號(hào) */
staticvoids3c2410_nand_deselect_chip(void)
{
s3c2410nand->NFCONF|=(1<<11);
}

/* 發(fā)出命令 */
staticvoids3c2410_write_cmd(intcmd)
{
volatileunsignedchar*p=(volatileunsignedchar*)&s3c2410nand->NFCMD;
*p=cmd;
}

/* 發(fā)出地址 */
staticvoids3c2410_write_addr(unsignedintaddr)
{
inti;
volatileunsignedchar*p=(volatileunsignedchar*)&s3c2410nand->NFADDR;

*p=addr&0xff;
for(i=0;i<10;i++);
*p=(addr>>9)&0xff;
for(i=0;i<10;i++);
*p=(addr>>17)&0xff;
for(i=0;i<10;i++);
*p=(addr>>25)&0xff;
for(i=0;i<10;i++);
}

/* 讀取數(shù)據(jù) */
staticunsignedchars3c2410_read_data(void)
{
volatileunsignedchar*p=(volatileunsignedchar*)&s3c2410nand->NFDATA;
return*p;
}

/* S3C2440的NAND Flash操作函數(shù) */

/* 復(fù)位 */
staticvoids3c2440_nand_reset(void)
{
s3c2440_nand_select_chip();
s3c2440_write_cmd(0xff);// 復(fù)位命令

s3c2440_wait_idle();
s3c2440_nand_deselect_chip();
}

/* 等待NAND Flash就緒 */
staticvoids3c2440_wait_idle(void)
{
inti;
volatileunsignedchar*p=(volatileunsignedchar*)&s3c2440nand->NFSTAT;
while(!(*p&BUSY))
for(i=0;i<10;i++);
}

/* 發(fā)出片選信號(hào) */
staticvoids3c2440_nand_select_chip(void)
{
inti;
s3c2440nand->NFCONT&=~(1<<1);
for(i=0;i<10;i++);
}

/* 取消片選信號(hào) */
staticvoids3c2440_nand_deselect_chip(void)
{
s3c2440nand->NFCONT|=(1<<1);
}

/* 發(fā)出命令 */
staticvoids3c2440_write_cmd(intcmd)
{
volatileunsignedchar*p=(volatileunsignedchar*)&s3c2440nand->NFCMD;
*p=cmd;
}

/* 發(fā)出地址 */
staticvoids3c2440_write_addr(unsignedintaddr)
{
inti;
volatileunsignedchar*p=(volatileunsignedchar*)&s3c2440nand->NFADDR;

*p=addr&0xff;
for(i=0;i<10;i++);
*p=(addr>>9)&0xff;
for(i=0;i<10;i++);
*p=(addr>>17)&0xff;
for(i=0;i<10;i++);
*p=(addr>>25)&0xff;
for(i=0;i<10;i++);
}


staticvoids3c2440_write_addr_lp(unsignedintaddr)
{
inti;
volatileunsignedchar*p=(volatileunsignedchar*)&s3c2440nand->NFADDR;
intcol,page;

col=addr&NAND_BLOCK_MASK_LP;
page=addr/NAND_SECTOR_SIZE_LP;

*p=col&0xff;/* Column Address A0~A7 */
for(i=0;i<10;i++);
*p=(col>>8)&0x0f;/* Column Address A8~A11 */
for(i=0;i<10;i++);
*p=page&0xff;/* Row Address A12~A19 */
for(i=0;i<10;i++);
*p=(page>>8)&0xff;/* Row Address A20~A27 */
for(i=0;i<10;i++);
*p=(page>>16)&0x03;/* Row Address A28~A29 */
for(i=0;i<10;i++);
}


/* 讀取數(shù)據(jù) */
staticunsignedchars3c2440_read_data(void)
{
volatileunsignedchar*p=(volatileunsignedchar*)&s3c2440nand->NFDATA;
return*p;
}


/* 在第一次使用NAND Flash前,復(fù)位一下NAND Flash */
staticvoidnand_reset(void)
{
nand_chip.nand_reset();
}

staticvoidwait_idle(void)
{
nand_chip.wait_idle();
}

staticvoidnand_select_chip(void)
{
inti;
nand_chip.nand_select_chip();
for(i=0;i<10;i++);
}

staticvoidnand_deselect_chip(void)
{
nand_chip.nand_deselect_chip();
}

staticvoidwrite_cmd(intcmd)
{
nand_chip.write_cmd(cmd);
}
staticvoidwrite_addr(unsignedintaddr)
{
nand_chip.write_addr(addr);
}

staticunsignedcharread_data(void)
{
returnnand_chip.read_data();
}


/* 初始化NAND Flash */
voidnand_init(void)
{
#defineTACLS 0
#defineTWRPH0 3
#defineTWRPH1 0

/* 判斷是S3C2410還是S3C2440 */
if((GSTATUS1==0x32410000)||(GSTATUS1==0x32410002))
{
nand_chip.nand_reset=s3c2410_nand_reset;
nand_chip.wait_idle=s3c2410_wait_idle;
nand_chip.nand_select_chip=s3c2410_nand_select_chip;
nand_chip.nand_deselect_chip=s3c2410_nand_deselect_chip;
nand_chip.write_cmd=s3c2410_write_cmd;
nand_chip.write_addr=s3c2410_write_addr;
nand_chip.read_data=s3c2410_read_data;

/* 使能NAND Flash控制器, 初始化ECC, 禁止片選, 設(shè)置時(shí)序 */
s3c2410nand->NFCONF=(1<<15)|(1<<12)|(1<<11)|(TACLS<<8)|(TWRPH0<<4)|(TWRPH1<<0);
}
else
{
nand_chip.nand_reset=s3c2440_nand_reset;
nand_chip.wait_idle=s3c2440_wait_idle;
nand_chip.nand_select_chip=s3c2440_nand_select_chip;
nand_chip.nand_deselect_chip=s3c2440_nand_deselect_chip;
nand_chip.write_cmd=s3c2440_write_cmd;
#ifdefLARGER_NAND_PAGE
nand_chip.write_addr=s3c2440_write_addr_lp;
#else
nand_chip.write_addr=s3c2440_write_addr;
#endif
nand_chip.read_data=s3c2440_read_data;

/* 設(shè)置時(shí)序 */
s3c2440nand->NFCONF=(TACLS<<12)|(TWRPH0<<8)|(TWRPH1<<4);
/* 使能NAND Flash控制器, 初始化ECC, 禁止片選 */
s3c2440nand->NFCONT=(1<<4)|(1<<1)|(1<<0);
}

/* 復(fù)位NAND Flash */
nand_reset();
}


/* 讀函數(shù) */
voidnand_read(unsignedchar*buf,unsignedlongstart_addr,intsize)
{
inti,j;

#ifdefLARGER_NAND_PAGE
if((start_addr&NAND_BLOCK_MASK_LP)||(size&NAND_BLOCK_MASK_LP)){
return;/* 地址或長(zhǎng)度不對(duì)齊 */
}
#else
if((start_addr&NAND_BLOCK_MASK)||(size&NAND_BLOCK_MASK)){
return;/* 地址或長(zhǎng)度不對(duì)齊 */
}
#endif

/* 選中芯片 */
nand_select_chip();

for(i=start_addr;i<(start_addr+size);){
/* 發(fā)出READ0命令 */
write_cmd(0);

/* Write Address */
write_addr(i);
#ifdefLARGER_NAND_PAGE
write_cmd(0x30);
#endif
wait_idle();

#ifdefLARGER_NAND_PAGE
for(j=0;j#else
for(j=0;j#endif
*buf=read_data();
buf++;
}
}

/* 取消片選信號(hào) */
nand_deselect_chip();

return;
}

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開(kāi)發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來(lái)越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來(lái)越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開(kāi)幕式在貴陽(yáng)舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語(yǔ)權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉