lpc2000系列proteus仿真+代碼,新手自取
菜鳥(niǎo)的ARM學(xué)習(xí)筆記 下面就是我學(xué)習(xí)ARM的第一階段的記錄,這段時(shí)間的學(xué)習(xí)基本上是使用Proteus配合KEIL做簡(jiǎn)單的實(shí)驗(yàn)(最后有實(shí)驗(yàn)的目錄以及下載地址)。通過(guò)該階段的學(xué)習(xí),算是對(duì)ARM的基本結(jié)構(gòu)有了了解。 該階段主要學(xué)習(xí)資料是《基于PROTEUS的ARM虛擬開(kāi)發(fā)技術(shù)》,以及另外一本ARM體系結(jié)構(gòu)的書(shū)籍,感覺(jué)這類(lèi)書(shū)都差不多。 學(xué)習(xí)ARM前需要的基礎(chǔ) - 前輩學(xué)習(xí)ARM的經(jīng)驗(yàn)! (我是在嵌入式開(kāi)發(fā)聯(lián)盟的新人區(qū)看的帖子。)
- 掌握C語(yǔ)言編程。
- 了解簡(jiǎn)單的微機(jī)算計(jì)原理知識(shí),例如二進(jìn)制,計(jì)算機(jī)程序的執(zhí)行過(guò)程,總線(數(shù)據(jù)、地址、控制),軟件系統(tǒng)(系統(tǒng)軟件與應(yīng)用軟件)。
- 聽(tīng)說(shuō)過(guò)RISC與CISC,高級(jí)語(yǔ)言與低級(jí)語(yǔ)言的區(qū)別。
- 最好聽(tīng)說(shuō)過(guò)串行傳輸與并行傳輸。
- 普林斯頓(ARM7)和哈佛結(jié)構(gòu)(ARM9、10、11—)。
什么是ARM? 學(xué)ARM,自然要理解ARM是什么,也好明確學(xué)習(xí)目標(biāo)。網(wǎng)上的資料很多,“ARM是一家公司,也是一個(gè)處理器體系”……我將學(xué)ARM分為以下幾類(lèi): - 做ARM的核心研發(fā)。也就是進(jìn)ARM公司做IP核,應(yīng)該是學(xué)電子之類(lèi)的東西吧。
- 買(mǎi)ARM的IP核,做具體的嵌入式處理器、核心板,例如三星和NXP。
- 買(mǎi)ARM核心板,連接外圍電路制作教育用或開(kāi)發(fā)用的開(kāi)發(fā)板,或者直接開(kāi)發(fā)其它中斷產(chǎn)品。
- 買(mǎi)ARM開(kāi)發(fā)板做產(chǎn)品,要做系統(tǒng)軟件和應(yīng)用軟件。
3和4基本并列了。 ARM基礎(chǔ) 任何一本介紹ARM體系結(jié)構(gòu)書(shū)籍都應(yīng)該有這些內(nèi)容。 處理器模式 用戶模式、特權(quán)模式又分為系統(tǒng)模式、管理模式、快中斷模式、中斷模式、終止模式、未定義指令終止模式。 2. 寄存器 R0-R7、R15和CPSR是所有模式共享的。
R8-R12出快中斷模式有RX-fiq外所有模式共享。
R13、R14和SPSR只有用戶模式和系統(tǒng)模式共享,其它都有似有SPSR。 R15(PC)程序計(jì)數(shù)器
R16(CPSR)程序轉(zhuǎn)臺(tái)寄存器
R13(SP)堆棧指針
P14(LR)鏈接寄存器 ARM指令集 匯編程序設(shè)計(jì) 略了,我看了,但是做Proteus仿真實(shí)驗(yàn)沒(méi)用上,兩天就忘了。 LPC2000 我買(mǎi)的3本ARM入門(mén)書(shū)籍中有兩本都是以LPC2000系列為例的,其實(shí)從網(wǎng)上可以下載到具體LPC2XXX處理器的datasheet,上面的資料是最權(quán)威和詳盡的。 引腳選擇 PINSEL0、PINSEL1設(shè)置各個(gè)引腳的功能。 中斷 中斷的寄存器太多了,沒(méi)記。 GPIO 做輸入輸出。以P0口為例,寄存器有IO0PIN、IO0SET、IO0DIR、IO0CLR。 存儲(chǔ)器 - LPC2000可用地址為4GB,內(nèi)部2GB(0~0x7FFFFFFF),外部2GB(0x80000000~0xDFFFFFFF),高0.5GB是I/O設(shè)備地址空間(0xE0000000~0xFFFFFFFF)。
- 內(nèi)部最低128KB或256KB為Flash。
- 高1GB(0x40000000~0x7FFFFFFF)為SRAM。其中0x40000000~0x40001FFF為片內(nèi)SRAM。
- I/O部分,低2MB(0xE0000000~0xE001FFFFF)為VPB外設(shè)空間,高2MB(0xFFE00000~0xFFFFFFFF)為AHB外設(shè)空間。
- FLASH加速模塊。
MAMCR設(shè)置是否允許加速、MAMTIM設(shè)置預(yù)取處理器時(shí)鐘。 時(shí)鐘 cclk、pclk
分振蕩器模式和從屬模式,振蕩器Fosc經(jīng)PLL升頻為cclk,cclk經(jīng)過(guò)VPB分頻后為pclk。
設(shè)置cclk
PLLCFG 設(shè)置PLL倍頻M,PLL分頻器值P
PLLCON PLL的允許與連接
PLLSTAT 讀取PLL狀態(tài)
PLLFEED 使PLL設(shè)置生效
例Fosc=12MHz,cclk=60MHz,則M=60/12-1=4,因?yàn)镕cco=P*cclk*2(要求156M~320MHz)。
設(shè)置pclk
由VPBDIV設(shè)置00為4分頻、01為不分頻、10為二分頻。 定時(shí)器 pclk定時(shí),定時(shí)器為32位,從0計(jì)數(shù)到0xFFFFFFFF。以T0為例
T0TC,計(jì)數(shù)器初值
T0PR,定時(shí)計(jì)數(shù)器分頻,pclk/(PR+1)
T0MR0~3,匹配值,當(dāng)計(jì)數(shù)帶到時(shí)候,按照T0MCR的設(shè)置觸發(fā)不同動(dòng)作。
T0MCR,計(jì)數(shù)器到達(dá)匹配值的動(dòng)作(復(fù)位、中斷、停止)
T0EMR,外部匹配寄存器,到達(dá)匹配值時(shí)候外部引腳的操作(MAT0.0~3)
T0CCR,外部引腳有特定動(dòng)作時(shí)候,計(jì)數(shù)值存入T0CR0~3,設(shè)置是否觸發(fā)中斷
T0CR0~3,在T0CCR控制下存TC值。
T0TCR,復(fù)位與使能
T0IR,對(duì)應(yīng)MR與CR中斷 PWM 看門(mén)狗 Pclk四分頻后控看門(mén)狗的32為計(jì)數(shù)器減一。
WDTC,看門(mén)狗計(jì)數(shù)器初值。
WDMOD,看門(mén)狗工作模式,可以開(kāi)啟和復(fù)位看門(mén)狗。
WDFEED,喂狗寄存器。
WDTV,看門(mén)狗計(jì)數(shù)器當(dāng)前值。 UART 引腳RxD0,TxD0
U0RBR,暫存接受數(shù)據(jù)。
U0THR,暫存發(fā)送數(shù)據(jù)。訪問(wèn)它時(shí),U0LCR的DLAB位為0。
U0IER,串口個(gè)狀態(tài)的中斷允許。
U0IIR,中斷標(biāo)志。
U0FCR,控制UART的FIFO(暫時(shí)沒(méi)明白)。
U0LCR,傳輸模式。
U0LSR,當(dāng)前狀態(tài)(錯(cuò)誤指示)。
U0DLL、U0DLM,pclk/( U0DLL U0DLM),訪問(wèn)時(shí)UOLCR的DLAB位為1。 SPI 全雙工同步串行接口
引腳:
SCK0,串行時(shí)鐘。
SSEL0,從機(jī)選擇。
MISO,主機(jī)輸入,從機(jī)輸出。
MOSI,主機(jī)輸出,從機(jī)輸入。
寄存器:
S0SPCR,SPI控制。
S0SPSR,SPI狀態(tài)。
S0SPDR,SPI數(shù)據(jù)。
S0SPCCR,控制SCK的頻率。必須為偶數(shù)且大于等于8。(指示一個(gè)SCK周期中的pclk周期)
S0SPINT,SPI中斷。 I2C 引腳:SDA,SCL
寄存器:
I2CONSET
I2CONCLR
I2CON
上面三個(gè)寄存器控制應(yīng)答標(biāo)志位,中斷標(biāo)志、停止和起始以及I2C使能。
I2STAT,I2C狀態(tài)。
I2DAT,I2C數(shù)據(jù)。
I2ADR,I2C從模式地址。
I2STAT,I2C狀態(tài)。
I2SCLH,高電平占空比占pclk周期個(gè)數(shù)。
I2SCLL,低電平占空比占pclk周期個(gè)數(shù)。
分頻fpclk/( I2SCLH+I2SCLL)。 AD轉(zhuǎn)換 引腳:AIN0~3
寄存器:
ADCR,工作模式選擇。
ADDR,轉(zhuǎn)換數(shù)據(jù)以及標(biāo)志的暫存。 基于Proteus的ARM實(shí)驗(yàn)?zāi)夸?/strong> 菜鳥(niǎo)的ARM學(xué)習(xí)筆記(第一階段) - LED閃爍——ARM的Proteus實(shí)驗(yàn)
- 開(kāi)關(guān)控制LED——ARM的Proteus實(shí)驗(yàn)
- LCD——ARM的Proteus實(shí)驗(yàn)
- UART——ARM的Proteus實(shí)驗(yàn)
- Eint1外部中斷——ARM的Proteus實(shí)驗(yàn)
- 多個(gè)外部中斷——ARM的Proteus實(shí)驗(yàn)
- 中斷結(jié)合串口——ARM的Proteus實(shí)驗(yàn)
- 定時(shí)器——ARM的Proteus實(shí)驗(yàn)
- SPI通信——ARM的Proteus實(shí)驗(yàn)
- SPI通信(多從設(shè)備)——ARM的Proteus實(shí)驗(yàn)
- ADC數(shù)模轉(zhuǎn)換——ARM的Proteus實(shí)驗(yàn)
1、LED閃爍——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 ARM(LPC21XX)的一個(gè)I/O口接LED,通過(guò)給它送0和1來(lái)設(shè)置LED的亮和滅。 Proteus仿真電路圖 步驟 KEIL - 創(chuàng)建新工程
- 選擇ARM型號(hào)(KEIL會(huì)自動(dòng)生成啟動(dòng)代碼startup.s)
- 添加源文件,編寫(xiě)程序
- 設(shè)置項(xiàng)目選項(xiàng)(是否輸出hex、lst文件,設(shè)置linker script)
Proteus C語(yǔ)言源程序 /******************************************************************************/
/* */
/* led.c: 用ARM點(diǎn)亮一個(gè)led并閃爍,有點(diǎn)浪費(fèi)…… */
/* */
/******************************************************************************/
#include <LPC21xx.H>
/*******************************************************************************
**函數(shù)名: delay()
**描述: 軟件延時(shí)
********************************************************************************/
void delay (void) {
unsigned volatile long i,j;
for(i=0;i<60000;i++)
for(j=0;j<5;j++)
;
}
int main(void) {
PINSEL0 = 0; /*設(shè)置引腳為GPIO */
IO0DIR = 0x000001; /*將P0.0設(shè)置為輸出 */
IO0SET = 0x000001; /*將P0.0置1,也就是讓led滅 */
while (1) {
IO0CLR = 0x000001;
delay();
IO0SET = 0x000001;
delay();
}
} 2、開(kāi)關(guān)控制LED——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 ARM的P0.1口接按鈕,再通過(guò)P0.0控制LED的亮、滅。本實(shí)驗(yàn)的電路圖以及實(shí)驗(yàn)均在上一個(gè)實(shí)驗(yàn)基礎(chǔ)之上修改。其中電路圖只多了一個(gè)開(kāi)關(guān)。 Proteus仿真電路圖 實(shí)驗(yàn)步驟略(與上一實(shí)驗(yàn)相同) C語(yǔ)言源程序 /******************************************************************************/
/* */
/* led.c: 用ARM實(shí)現(xiàn)開(kāi)關(guān)控制led并亮滅,還是有點(diǎn)浪費(fèi)…… */
/* */
/******************************************************************************/
#include <LPC21xx.H>
#define P0_1 0x02; /*P0.1*/
/*******************************************************************************
**函數(shù)名: delay()
**描述: 軟件延時(shí)
********************************************************************************/
void delay (void) {
unsigned volatile long i;
for(i=0;i<10000;i++)
;
}
int main(void) {
int p01State;
PINSEL0 = 0; /*設(shè)置引腳為GPIO */
IO0DIR = 0x000001; /*將P0.0設(shè)置為輸出 */
IO0SET = 0x000001; /*將P0.0置1,也就是讓led滅 */
while (1) {
p01State = IO0PIN&P0_1; /*讀取開(kāi)關(guān)狀態(tài)*/
if(p01State == 0){
IO0CLR = 0x000001;
delay();
}
else{
IO0SET = 0x000001;
delay();
}
}
} 3、LCD——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 ARM的P0.0口到P0.10口接LCD,P0.11接LED。每過(guò)一段時(shí)間LED狀態(tài)改變,LCD顯示LED的狀態(tài)。 Proteus仿真電路圖 C語(yǔ)言源程序 #include <LPC21XX.H>
#define rs (1<<8)
#define rw (1<<9)
#define en (1<<10)
#define busy (1<<7) //P0.7
typedef unsigned char uint8;
uint8 ledDown[]={"The LED is down!"};
uint8 ledUp[]={"The LED is up!"};
void waitLCD() /*等待LCD*/
{
IO0DIR=0xf00;
while(1)
{
IO0CLR=rs;
IO0SET=rw;
IO0SET=en;
if(!(IO0PIN & busy))break;
IO0CLR = en;
}
IO0DIR=0xfff;
}
void lcdOp(uint8 dat)/*送LCD控制碼*/
{
waitLCD();
IO0CLR=rs;
IO0CLR=rw;
IO0CLR=0xff;
IO0SET=dat;
IO0SET=en;
IO0CLR=en;
}
void lcdData(uint8 dat)/*送LCD顯示數(shù)據(jù)*/
{
waitLCD();
IO0SET=rs;
IO0CLR=rw;
IO0CLR=0xff;
IO0SET=dat;
IO0SET=en;
IO0CLR=en;
}
void lcdInit(void)/*初始化LCD,DataSheet里有建議的初始化代碼*/
{
/* LCD配置為兩行,5*7字體 */
lcdOp(0x38);
lcdOp(0x38);
lcdOp(0x06);
lcdOp(0x0E);
lcdOp(0x01);
/* LCD配置為一行,5*10字體
lcdOp(0x34);
lcdOp(0x34);
lcdOp(0x06);
lcdOp(0x0E);
lcdOp(0x01);
*/
}
void lcdDisplay(uint8 addr,uint8 *p)/*LCD顯示字符串*/
{
lcdOp(addr);
while(*p !='\0'){
lcdData(*(p++));
}
}
void lcdClear(void)/*LCD清屏*/
{
lcdOp(0x01);
}
void delay (void) {
unsigned volatile long i,j;
for(i=0;i<60000;i++)
for(j=0;j<10;j++)
;
}
int main(void)
{
lcdInit();/*初始化LCD顯示*/
IO0DIR=0xfff;//設(shè)置為輸出口
IO0CLR=0xfff;
while (1) {
IO0CLR = 0x000800;
lcdDisplay(0x80,ledUp);
delay();
lcdClear();
IO0SET = 0x000800;
lcdDisplay(0x80,ledDown);
delay();
lcdClear();
}
} 4、UART——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 ARM的P0.0口接LED,串口接Proteus的虛擬終端。每隔一段時(shí)間改變一次LED的狀態(tài),并且在串口上輸出LED的當(dāng)前狀態(tài)。 Proteus仿真電路圖 C語(yǔ)言源程序 #include <LPC21xx.H>
#include "uart.h"
#define CR 0x0D
char ledDown[]={"The LED is down!\n"};
char ledUp[]={"The LED is up!\n"};
int putchar (int ch) {/* 向串口輸出一個(gè)字符 */
if (ch == '\n') {
while (!(U1LSR & 0x20));
U1THR = CR; a
}
while (!(U1LSR & 0x20));
return (U1THR = ch);
}
void delay (void) {
unsigned volatile long i,j;
for(i=0;i<60000;i++)
for(j=0;j<5;j++)
;
}
void serialPuts(char *p){/* 向串口輸出字符串 */
while (*p != '\0'){
putchar(*p++);
}
}
int main (void) {
/* 開(kāi)始初始化串口 */
PINSEL0 = 0x00050000;/* 設(shè)置引腳,開(kāi)串口功能 */
U1LCR = 0x83;/* 8位數(shù)據(jù),無(wú)效驗(yàn),一個(gè)停止位 */
U1DLL = 97;/* VPB 15MHz的時(shí)候波特率為9600 */
U1LCR = 0x03;/* DLAB = 0 */
/* 結(jié)束初始化串口 */
IO0DIR = 0x000001;/*將P0.0設(shè)置為輸出 */
while (1) {
IO0CLR = 0x000001;
serialPuts(ledUp);
delay();
IO0SET = 0x000001;
serialPuts(ledDown);
delay();
}
} 5、Eint1外部中斷——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 ARM的P0.25接一個(gè)LED,引腳設(shè)置時(shí)連接EINT1功能,按鈕觸發(fā)中斷。中斷服務(wù)程序另LED快速閃爍。 Proteus仿真電路圖 C語(yǔ)言源程序 #include <LPC21XX.H>
#define LEDCON 0x02000000/*LED接在P0.25上*/
typedef unsigned int uint32;
void IRQ_Eint1(void) __attribute__ ((interrupt));/*聲明某函數(shù)為中斷服務(wù)子程序的方法*/
uint32 times = 100;/*循環(huán)次數(shù)默認(rèn)為100*/
void IRQ_Eint1(void){
times = 5;
while((EXTINT&0x02)!=0){
EXTINT=0x02; //清除EINT1中斷標(biāo)志
}
VICVectAddr=0;
}
void delay100(void) {
unsigned volatile long i,j;
for(i=0;i<10000;i++)
for(j=0;j<times;j++)
;
if(times > 100){
times--;
}else if(times <100){
times++;
}
}
int main(void)
{
IO0DIR = LEDCON;
PINSEL0 = 0x20000000;/*引腳選中EINT1功能*/
PINSEL1 = 0x00000000;
/*以下為中斷控制部分*/
VICIntSelect=0;/*全部中斷設(shè)置為IRQ,若某位為1是FIQ*/
VICIntEnable=0x00008000;/*使能EINT1,EINT為第15位*/
VICVectCntl1=0x2F;/*0xF,15號(hào)中斷*/
VICVectAddr1=(int)IRQ_Eint1;/*設(shè)置中斷服務(wù)子程序*/
EXTINT=0x07;
while (1) {
IO0CLR = LEDCON;
delay100();
IO0SET = LEDCON;
delay100();
}
} 6、多個(gè)外部中斷——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 ARM開(kāi)啟兩個(gè)中斷源Eint1與Eint2,分別用一個(gè)按鈕來(lái)控制。在沒(méi)有中斷的時(shí)候兩個(gè)LED都緩慢閃爍,當(dāng)任何一個(gè)中斷被出發(fā)的時(shí)候,對(duì)應(yīng)的LED會(huì)急促閃爍,逐漸回復(fù)正常。 Proteus仿真電路圖 C語(yǔ)言源程序 #include <LPC21XX.H>
#define LED1 0x02000000/*LED1接在P0.25上*/
#define LED0 0x01000000/*LED0接在P0.24上*/
typedef unsigned int uint32;
void Eint1_ISR(void) __attribute__ ((interrupt));/*聲明某函數(shù)為中斷服務(wù)子程序的方法*/
void Eint0_ISR(void) __attribute__ ((interrupt));
uint32 times = 40;/*循環(huán)次數(shù)默認(rèn)為40*/
void delay40(void) {
unsigned volatile long i,j;
for(i=0;i<10000;i++)
for(j=0;j<times;j++)
;
if(times > 40){
times-=2;
}else if(times <40){
times+=2;
}
}
void Eint0_ISR(void){
times = 0;
while(times!=40){
IO0CLR = LED0;
delay40();
IO0SET = LED0;
delay40();
}
while((EXTINT&0x01)!=0){
EXTINT=0x01;/*清除EINT0中斷標(biāo)志*/
}
VICVectAddr=0x00;
}
void Eint1_ISR(void){
times = 0;
while(times!=40){
IO0CLR = LED1;
delay40();
IO0SET = LED1;
delay40();
}
while((EXTINT&0x02)!=0){
EXTINT=0x02;/*清除EINT1中斷標(biāo)志*/
}
VICVectAddr=0;
}
int main(void)
{
IO0DIR = LED1|LED0;
PINSEL0 = 0x20000000;/*引腳選中EINT1功能*/
PINSEL1 = 0x00000001;/*引腳選中EINT0功能*/
/*以下為中斷控制部分*/
VICIntSelect=0;/*全部中斷設(shè)置為IRQ,若某位為1是FIQ*/
VICIntEnable=0x0000C000;/*使能EINT1、0,EINT1為第15位,0為14位*/
VICVectCntl0=0x2E;/*EINT0最高優(yōu)先級(jí)*/
VICVectAddr0=(int)Eint0_ISR;/*設(shè)置EINT0向量地址*/
VICVectCntl1=0x2F;/*0xF,15號(hào)中斷*/
VICVectAddr1=(int)Eint1_ISR;/*設(shè)置中斷服務(wù)子程序*/
EXTINT=0x07;
while (1) {/*無(wú)中斷時(shí),兩燈一起緩慢閃爍*/
IO0CLR = LED1|LED0;
delay40();
IO0SET = LED1|LED0;
delay40();
}
} 7、中斷結(jié)合串口——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 同上多中斷源實(shí)驗(yàn)相同,ARM開(kāi)啟兩個(gè)中斷源Eint1與Eint2,分別用一個(gè)按鈕來(lái)控制。在沒(méi)有中斷的時(shí)候兩個(gè)LED都緩慢閃爍,當(dāng)任何一個(gè)中斷被出發(fā)的時(shí)候,對(duì)應(yīng)的LED會(huì)急促閃爍,逐漸回復(fù)正常。 此外,使用一個(gè)串口來(lái)發(fā)送當(dāng)前狀態(tài),在無(wú)中斷的時(shí)候發(fā)送正常狀態(tài)報(bào)告,當(dāng)有中斷的時(shí)候,串口發(fā)送中斷源。 Proteus仿真電路圖 C語(yǔ)言源程序 文件一:main.c #include <LPC21XX.H>
#include "uart0.h"
#define LED1 0x02000000/*LED1接在P0.25上*/
#define LED0 0x01000000/*LED0接在P0.24上*/
typedef unsigned int uint32;
void Eint1_ISR(void) __attribute__ ((interrupt));/*聲明某函數(shù)為中斷服務(wù)子程序的方法*/
void Eint0_ISR(void) __attribute__ ((interrupt));
uint32 times = 40;/*循環(huán)次數(shù)默認(rèn)為40*/
char status[] = "Everytnig is fine.";
char eint0Str[] = "Interruption EINT0 activated!!!";
char eint1Str[] = "Interruption EINT1 activated!!!";
void delay40(void) {
unsigned volatile long i,j;
for(i=0;i<10000;i++)
for(j=0;j<times;j++)
;
if(times > 40){
times-=2;
}else if(times <40){
times+=2;
}
}
void Eint0_ISR(void){
times = 0;
while(times!=40){
IO0CLR = LED0;
delay40();
IO0SET = LED0;
delay40();
serialPuts(eint0Str);
}
while((EXTINT&0x01)!=0){
EXTINT=0x01;/*清除EINT0中斷標(biāo)志*/
}
VICVectAddr=0x00;
}
void Eint1_ISR(void){
times = 0;
while(times!=40){
IO0CLR = LED1;
delay40();
IO0SET = LED1;
delay40();
serialPuts(eint1Str);
}
while((EXTINT&0x02)!=0){
EXTINT=0x02;/*清除EINT1中斷標(biāo)志*/
}
VICVectAddr=0;
}
int main(void)
{
IO0DIR = LED1|LED0;
PINSEL0 = 0x20000005;/*引腳選中EINT1功能,開(kāi)串口UART0*/
PINSEL1 = 0x00000001;/*引腳選中EINT0功能*/
/*以下為中斷控制部分*/
VICIntSelect=0;/*全部中斷設(shè)置為IRQ,若某位為1是FIQ*/
VICIntEnable=0x0000C000;/*使能EINT1、0,EINT1為第15位,0為14位*/
VICVectCntl0=0x2E;/*EINT0最高優(yōu)先級(jí)*/
VICVectAddr0=(int)Eint0_ISR;/*設(shè)置EINT0向量地址*/
VICVectCntl1=0x2F;/*0xF,15號(hào)中斷*/
VICVectAddr1=(int)Eint1_ISR;/*設(shè)置中斷服務(wù)子程序*/
EXTINT=0x07;
uart0Init();
while (1) {/*無(wú)中斷時(shí),兩燈一起緩慢閃爍*/
IO0CLR = LED1|LED0;
delay40();
IO0SET = LED1|LED0;
delay40();
serialPuts(status);
}
}
文件二:uart0.c #include <LPC21XX.H>
#include "uart0.h"
#define CR 0x0D
int putchar (int ch) {/* 向串口輸出一個(gè)字符 */
if (ch == '\n') {
while (!(U0LSR & 0x20));
U0THR = CR;
}
while (!(U0LSR & 0x20));
return (U0THR = ch);
}
void serialPuts(char *p){/* 向串口輸出字符串 */
while (*p != '\0'){
putchar(*p++);
}
putchar('\n');
}
void uart0Init(void){
U0LCR = 0x83;/* 8位數(shù)據(jù),無(wú)效驗(yàn),一個(gè)停止位 */
U0DLL = 97;/* VPB 15MHz的時(shí)候波特率為9600 */
U0LCR = 0x03;/* DLAB = 0 */
} 8、定時(shí)器——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 ARM的定時(shí)器實(shí)驗(yàn),定時(shí)改變LED的狀態(tài)。 Proteus仿真電路圖 C語(yǔ)言源程序 #include <LPC21xx.H>
#define LED 0x000001
typedef unsigned int uint32;
typedef unsigned char uint8;
void timer0_ISR (void) __attribute__ ((interrupt));
uint8 timer0Times = 0;
void timer0Init (void) {
T0MR0 = 119999;/*匹配寄存器,120000-1,12000000為1秒*/
T0MCR = 3;/*產(chǎn)生中斷,重置TC*/
T0TCR = 1;/*使能定時(shí)計(jì)數(shù)器0*/
VICVectAddr0 = (unsigned long)timer0_ISR;
VICVectCntl0 = 0x20 | 4;/*定時(shí)器計(jì)數(shù)器0為4號(hào)中斷*/
VICIntEnable = 0x00000010;/*開(kāi)定時(shí)計(jì)數(shù)器0中斷*/
}
void timer0_ISR (void) {
timer0Times++;
uint32 i;
if(timer0Times == 10){
i=IO0SET; //讀出當(dāng)前LED2控制值
if((i&LED)==0){
IO0SET=LED;
}else{
IO0CLR=LED;
}
timer0Times = 0;
}
T0IR = 1;/*清除定時(shí)器0中斷*/
VICVectAddr = 0;
}
int main(void) {
PINSEL0 = 0;/*設(shè)置引腳為GPIO */
IO0DIR = LED;/*將P0.0設(shè)置為輸出 */
IO0SET = LED;/*將P0.0置1,也就是讓led滅 */
timer0Init();
while (1) {
}
} 9、SPI通信——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 使用SPI協(xié)議,利用ARM控制數(shù)碼顯示管顯示1到F,用串口檢驗(yàn)發(fā)送的數(shù)據(jù)是否正確。 Proteus仿真電路圖 C語(yǔ)言源程序 下面是主程序文件,串口部分的程序代碼與“中斷結(jié)合串口”相同,所以省略。 #include <LPC21XX.H>
#include "uart0.h"
#define HC595_CS 0x00000100/*P0.8口為74HC595的片選*/
typedef unsigned int uint32;
typedef unsigned char uint8;
char status[] = "Everytnig is fine.";
uint8 const DISP_TAB[16]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82, 0xF8,0x80,0x90,0x88,0x83,0xC6,0xA1,0x86,0x8E};
uint8 rcv_data;
void delay(void) {
unsigned volatile long i,j;
for(i=0;i<10000;i++)
for(j=0;j<30;j++)
;
}
void spiInit(void){
S0SPCCR=0x52; //設(shè)置SPI時(shí)鐘分頻
S0SPCR=0x30; //設(shè)置SPI接口模式
}
uint8 spiChar(uint8 data){
IO0CLR=HC595_CS; //片選
S0SPDR=data;
while(0==(S0SPSR&0x80)); //等待SPIF置位,即等待數(shù)據(jù)發(fā)送完畢
IO0SET=HC595_CS;
return(S0SPDR);
}
int main(void){
uint8 i;
PINSEL0 = 0x00005505;/*引腳開(kāi)串口SPI、UART0*/
IO0DIR=HC595_CS;
spiInit();/*初始化SPI接口*/
uart0Init();
while(1){
for (i=0;i<16;i++) {
if(i<10){putchar(i+'0');}
else{putchar(i+7+'0');}
rcv_data=spiChar(DISP_TAB[ i]);/*發(fā)送顯示數(shù)據(jù)*/
delay();
}
serialPuts(status);
}
} [ i]
10、SPI通信(多從設(shè)備)——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 使用SPI協(xié)議,利用ARM控制數(shù)碼顯示管1顯示1到F,數(shù)碼顯示管2則從F到1倒計(jì)數(shù),數(shù)碼顯示管則一直顯示0。該連接方法使用片選來(lái)激活不同的SPI從設(shè)備(HC595)。 Proteus仿真電路圖 C語(yǔ)言源程序 下面是主程序文件,串口部分的程序代碼與“中斷結(jié)合串口”相同,所以省略。 #include <LPC21XX.H>
#include "uart0.h"
#define HC595_CS 0x00000100/*P0.8口為第一個(gè)74HC595的片選*/
#define HC595_CS2 0x00000200/*P0.9口為第二個(gè)74HC595的片選*/
#define HC595_CS3 0x00000400/*P0.10口為第三個(gè)74HC595的片選*/
typedef unsigned int uint32;
typedef unsigned char uint8;
char status[] = "Everytnig is fine.";
uint8 const DISP_TAB[16]={0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8, 0x80,0x90,0x88,0x83,0xC6,0xA1,0x86,0x8E};
uint8 rcv_data;
void delay(void) {
unsigned volatile long i,j;
for(i=0;i<10000;i++)
for(j=0;j<30;j++)
;
}
void spiInit(void){
S0SPCCR=0x52;/*設(shè)置SPI時(shí)鐘分頻*/
S0SPCR=0x30;/*設(shè)置SPI接口模式*/
}
uint8 spiChar(uint8 data){
IO0CLR=HC595_CS;/*SPI1片選*/
S0SPDR=data;
while(0==(S0SPSR&0x80));/*等待SPIF置位,即等待數(shù)據(jù)發(fā)送完畢*/
IO0SET=HC595_CS;
return(S0SPDR);
}
uint8 spi2Char(uint8 data){
IO0CLR=HC595_CS2;/*SPI2片選*/
S0SPDR=data;
while(0==(S0SPSR&0x80));
IO0SET=HC595_CS2;
return(S0SPDR);
}
uint8 spi3Char(uint8 data){
IO0CLR=HC595_CS3;/*SPI3片選*/
S0SPDR=data;
while(0==(S0SPSR&0x80));
IO0SET=HC595_CS3;
return(S0SPDR);
}
int main(void){
uint8 i;
PINSEL0 = 0x00005505;/*引腳開(kāi)SPI、UART0*/
IO0DIR=HC595_CS|HC595_CS2|HC595_CS3;
spiInit();/*初始化SPI接口*/
uart0Init();
while(1){
for (i=0;i<16;i++) {
if(i<10){putchar(i+'0');}
else{putchar(i+7+'0');}
rcv_data=spiChar(DISP_TAB[ i]);/*發(fā)送顯示數(shù)據(jù)到數(shù)碼顯示管1*/
rcv_data=spi2Char(DISP_TAB[15-i]);/*數(shù)碼顯示管2*/
rcv_data=spi3Char(DISP_TAB[0]);/*數(shù)碼顯示管2*/
delay();
}
serialPuts(status);
}
} [ i]
11、ADC數(shù)模轉(zhuǎn)換——ARM的Proteus實(shí)驗(yàn) 實(shí)驗(yàn)原理 使用ARM微處理器內(nèi)置的AD轉(zhuǎn)換,將電壓值轉(zhuǎn)換為數(shù)字量后直接輸出到串口UART0。 Proteus仿真電路圖 C語(yǔ)言源程序 main.c #include <LPC21XX.H>
#include "uart0.h"
typedef unsigned int uint32;
void delay(void) {
unsigned volatile long i,j;
for(i=0;i<10000;i++)
for(j=0;j<50;j++)
;
}
void adcRead (void) {
unsigned int val;
ADCR |= 0x01000000;/* 開(kāi)始AD轉(zhuǎn)換 */
do {
val = ADDR;/* 讀取AD轉(zhuǎn)換數(shù)據(jù)寄存器 */
} while ((val & 0x80000000) == 0);/* 等待AD轉(zhuǎn)換結(jié)束 */
ADCR &= ~0x01000000;/* 結(jié)束AD轉(zhuǎn)換 */
val = (val >> 6) & 0x03FF;/* 設(shè)置數(shù)據(jù)格式并且按照16進(jìn)制輸出 */
putstr ("\nAIN0 Result = 0x");
puthex((val >> 8) & 0x0F);
puthex((val >> 4) & 0x0F);
puthex (val & 0x0F);
}
int main(void)
{
ADCR = 0x002E0401;/* Setup A/D: 10-bit AIN0 @ 3MHz */
PINSEL0 = 0x20000005;/*引腳選中EINT1功能,開(kāi)串口UART0*/
PINSEL1 = 0x00000001;/*引腳選中EINT0功能*/
uart0Init();
while (1) {
adcRead();
delay();
}
}
uart0.c #include <LPC21XX.H>
#include "uart0.h"
#define CR 0x0D
int putchar (int ch) {/* 向串口輸出一個(gè)字符 */
if (ch == '\n') {
while (!(U0LSR & 0x20));
U0THR = CR;
}
while (!(U0LSR & 0x20));
return (U0THR = ch);
}
void serialPuts(char *p){/* 向串口輸出字符串 */
while (*p != '\0'){
putchar(*p++);
}
putchar('\n');
}
void uart0Init(void){
U0LCR = 0x83;/* 8位數(shù)據(jù),無(wú)效驗(yàn),一個(gè)停止位 */
U0DLL = 97;/* VPB 15MHz的時(shí)候波特率為9600 */
U0LCR = 0x03;/* DLAB = 0 */
}
void puthex (int hex) {/* Write Hex Digit to Serial Port */
if (hex > 9) putchar('A' + (hex - 10));
else putchar('0' + hex);
}
void putstr (char *p) {/* Write string */
while (*p) {
putchar(*p++);
}
}
完整的Word格式文檔51黑下載地址:
菜鳥(niǎo)的ARM學(xué)習(xí)筆記proteus仿真.zip
(183.92 KB, 下載次數(shù): 118)
2018-11-15 17:50 上傳
點(diǎn)擊文件名下載附件
ARM筆記
|