stm32的简单程序的编译和Proteus 电路仿真软件

    科技2024-06-25  71

    ARM开发环境入门

    一、安装mdk5软件和stm32包 二、stm32程序——闪烁LED 1.先建立工程选择需要的芯片 并进行相关设置 2.编写程序代码

    //宏定义,用于存放stm32寄存器映射 #define PERIPH_BASE ((unsigned int)0x40000000)//AHB #define APB2PERIPH_BASE (PERIPH_BASE + 0x10000) #define GPIOA_BASE (APB2PERIPH_BASE + 0x0800) //GPIOA_BASE=0x40000000+0x10000+0x0800=0x40010800,该地址为GPIOA的基地址 #define GPIOB_BASE (APB2PERIPH_BASE + 0x0C00) //GPIOB_BASE=0x40000000+0x10000+0x0C00=0x40010C00,该地址为GPIOB的基地址 #define GPIOC_BASE (APB2PERIPH_BASE + 0x1000) //GPIOC_BASE=0x40000000+0x10000+0x1000=0x40011000,该地址为GPIOC的基地址 #define GPIOD_BASE (APB2PERIPH_BASE + 0x1400) //GPIOD_BASE=0x40000000+0x10000+0x1400=0x40011400,该地址为GPIOD的基地址 #define GPIOE_BASE (APB2PERIPH_BASE + 0x1800) //GPIOE_BASE=0x40000000+0x10000+0x0800=0x40011800,该地址为GPIOE的基地址 #define GPIOF_BASE (APB2PERIPH_BASE + 0x1C00) //GPIOF_BASE=0x40000000+0x10000+0x0800=0x40011C00,该地址为GPIOF的基地址 #define GPIOG_BASE (APB2PERIPH_BASE + 0x2000) //GPIOG_BASE=0x40000000+0x10000+0x0800=0x40012000,该地址为GPIOG的基地址 #define GPIOA_ODR_Addr (GPIOA_BASE+12) //0x4001080C #define GPIOB_ODR_Addr (GPIOB_BASE+12) //0x40010C0C #define GPIOC_ODR_Addr (GPIOC_BASE+12) //0x4001100C #define GPIOD_ODR_Addr (GPIOD_BASE+12) //0x4001140C #define GPIOE_ODR_Addr (GPIOE_BASE+12) //0x4001180C #define GPIOF_ODR_Addr (GPIOF_BASE+12) //0x40011A0C #define GPIOG_ODR_Addr (GPIOG_BASE+12) //0x40011E0C #define BITBAND(addr, bitnum) ((addr & 0xF0000000)+0x2000000+((addr &0xFFFFF)<<5)+(bitnum<<2)) #define MEM_ADDR(addr) *((volatile unsigned long *)(addr)) #define LED0 MEM_ADDR(BITBAND(GPIOA_ODR_Addr,8)) //#define LED0 *((volatile unsigned long *)(0x422101a0)) //PA8 //定义typedef类型别名 typedef struct { volatile unsigned int CR; volatile unsigned int CFGR; volatile unsigned int CIR; volatile unsigned int APB2RSTR; volatile unsigned int APB1RSTR; volatile unsigned int AHBENR; volatile unsigned int APB2ENR; volatile unsigned int APB1ENR; volatile unsigned int BDCR; volatile unsigned int CSR; } RCC_TypeDef; #define RCC ((RCC_TypeDef *)0x40021000) //定义typedef类型别名 typedef struct { volatile unsigned int CRL; volatile unsigned int CRH; volatile unsigned int IDR; volatile unsigned int ODR; volatile unsigned int BSRR; volatile unsigned int BRR; volatile unsigned int LCKR; } GPIO_TypeDef; //GPIOA指向地址GPIOA_BASE,GPIOA_BASE地址存放的数据类型为GPIO_TypeDef #define GPIOA ((GPIO_TypeDef *)GPIOA_BASE) void LEDInit( void ) { RCC->APB2ENR|=1<<2; //GPIOA 时钟开启 GPIOA->CRH&=0XFFFFFFF0; GPIOA->CRH|=0X00000003; } //粗略延时 void Delay_ms( volatile unsigned int t) { unsigned int i,n; for (n=0;n<t;n++) for (i=0;i<800;i++); } int main(void) { LEDInit(); while (1) { LED0=0;//LED熄灭 Delay_ms(500);//延时时间 LED0=1;//LED亮 Delay_ms(500);//延时时间 } }

    如下图 编译成功后:

    三、Proteus 电路仿真软件

    1.将本次仿真需要的软件添加到元件列表中:AT89C51、电阻、数码管、开关、喇叭等等,如下图所示 2.仿真 单击 开始仿真。 3.源代码调试 双击Keil uvision图标,进入KeilC Vision集成开发环境,创建一个新项目(Project),并为该项目选定合适的单片机CPU器件(如:Atmel公司的AT89C51)。并为该项目加入Keil C51或ASM51源程序。 源程序代码如下:

    #include"reg51.h" #include"intrins.h" #define uchar unsigned char #define uint unsigned int uchar code DSY_CODE[]= {0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0xff}; uchar DSY_BUFFER[]={0,0,0xBF,0,0,0xff,0,0}; uchar DSY_BUFFER1[]={0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff}; uchar Scan_BIT; uchar DSY_IDX; uchar Key_State; uchar s,s100,h1 ; uchar h,m; sbit dd=P1^7; sbit cc=P1^6; sbit ss=P1^5; void DelayMS(uchar x) { uchar i; while(x--) for(i = 0;i<=120;i++); } void Increase_Hour() { if (++h>=23) h=0; DSY_BUFFER[0] =DSY_CODE[h/10]; DSY_BUFFER[1] =DSY_CODE[h]; } void Decrease_Hour() { h--; if (h<=0)h=23; DSY_BUFFER[0] =DSY_CODE[h/10]; DSY_BUFFER[1] =DSY_CODE[h]; } void Increase_Minute() { if(++m>=59) { m=0;Increase_Hour(); } DSY_BUFFER[3] =DSY_CODE[m/10]; DSY_BUFFER[4]=DSY_CODE[m]; } void Decrease_Minute() { m--; if(m<=0) m=59; DSY_BUFFER[3] =DSY_CODE[m/10]; DSY_BUFFER[4] =DSY_CODE[m]; } void Increase_Hour1() { if (++h1>7) h1=1; DSY_BUFFER[6]=DSY_CODE[h1/10]; DSY_BUFFER[7] =DSY_CODE[h1]; } void Increase_Second() { if(++s>=59) { s=0;Increase_Minute(); } DSY_BUFFER1[0] =DSY_CODE[s/10]; DSY_BUFFER1[1] =DSY_CODE[s]; } void T0_INT() interrupt 1 { TH0=(65536-1000)/256; TL0=(65536-1000)%6; if(ss==1) { P3=Scan_BIT; P2=~DSY_BUFFER[DSY_IDX]; } else { P3=Scan_BIT; P2=~DSY_BUFFER[DSY_IDX]; } Scan_BIT=_crol_(Scan_BIT,1); DSY_IDX=(DSY_IDX+1)%8; if(h1==3) { if(h==6&m==0)cc=dd=0;else cc=dd=1; if(h==6&m==15)cc=dd=0; else cc=dd=1; if(h==7&m==0)cc=dd=0; else cc=dd=1; if(h==12&m==30)cc=dd=0; else cc=dd=1; if(h==14&m==0)cc=dd=0; else cc=dd=1; if(h==16&m==30)cc=dd=0; else cc=dd=1; if(h==18&m==0)cc=dd=0; else cc=dd=1; if(h==18&m==45)cc=dd=0; else cc=dd=1; if(h==19&m==30)cc=dd=0; else cc=dd=1; if(h==21&m==30)cc=dd=0; else cc=dd=1; if(h==22&m==0)cc=dd=0;else cc=dd=1; } else cc=dd=1; if(h1==1|h1==4|h1==2|h1==5) { if(h==6&m==0)cc=dd=0;else cc=dd=1; if(h==7&m==0)cc=dd=0; else cc=dd=1; if(h==12&m==30)cc=dd=0; else cc=dd=1; if(h==14&m==0)cc=dd=0; else cc=dd=1; if(h==16&m==30)cc=dd=0; else cc=dd=1; if(h==18&m==0)cc=dd=0; else cc=dd=1; if(h==18&m==45)cc=dd=0; else cc=dd=1; if(h==19&m==30)cc=dd=0; else cc=dd=1; if(h==21&m==30)cc=dd=0; else cc=dd=1; if(h==22&m==0)cc=dd=0;else cc=dd=1; } else cc=dd=1; if(h1==6) { if(h==6&m==0)cc=dd=0;else cc=dd=1; if(h==7&m==0)cc=dd=0; else cc=dd=1; if(h==12&m==30)cc=dd=0; else cc=dd=1; if(h==14&m==30)cc=dd=0; else cc=dd=1; if(h==18&m==0)cc=dd=0; else cc=dd=1; if(h==21&m==0)cc=dd=0; else cc=dd=1; if(h==21&m==30)cc=dd=0; else cc=dd=1; if(h==22&m==30)cc=dd=0;else cc=dd=1; } else cc=dd=1; if(h1==7) { if(h==6&m==30)cc=dd=0;else cc=dd=1; if(h==7&m==0)cc=dd=0; else cc=dd=1; if(h==11&m==30)cc=dd=0; else cc=dd=1; if(h==12&m==30)cc=dd=0; else cc=dd=1; if(h==15&m==30)cc=dd=0; else cc=dd=1; if(h==16&m==30)cc=dd=0; else cc=dd=1; if(h==18&m==0)cc=dd=0; else cc=dd=1; if(h==18&m==40)cc=dd=0; else cc=dd=1; if(h==20&m==0)cc=dd=0; else cc=dd=1; if(h==20&m==30)cc=dd=0; else cc=dd=1; if(h==21&m==30)cc=dd=0; else cc=dd=1; if(h==22&m==0)cc=dd=0;else cc=dd=1; } else cc=dd=1; } void T1_INT() interrupt 3 { TH1=(65536-50000)/256; TL1=(65536-50000)%6; if (s100==20) { s100=0; Increase_Second(); } else s100++; } void main() { P2=P3=0xFF; TMOD=0x11; TH0=(65536-10000)/256; TL0=(65536-1000)%6; TH1=0XDC; TL1=0; TCON=0x01; EA=1; ET0=1; ET1=1; h=05,m=55,s=s100,h1=1; DSY_BUFFER[0]=DSY_CODE[h/10]; DSY_BUFFER[1]=DSY_CODE[h]; DSY_BUFFER[3]=DSY_CODE[m/10]; DSY_BUFFER[4]=DSY_CODE[m]; DSY_BUFFER[6]=DSY_CODE[h1/10]; DSY_BUFFER[7]=DSY_CODE[h1]; DSY_BUFFER1[0]=DSY_CODE[h1/10]; DSY_BUFFER1[1]=DSY_CODE[h1]; Scan_BIT=0xFE; DSY_IDX=0; TR0=TR1=1; Key_State=0xFF; while(1) { if (P1^Key_State) { DelayMS(10); if (P1^Key_State) { Key_State=P1; EA=0; if((Key_State&0x01)==0)Increase_Hour(); else if ((Key_State&0x02)==0) Decrease_Hour(); else if ((Key_State&0x04)==0)Increase_Minute(); else if ((Key_State&0x08)==0)Decrease_Minute(); else if ((Key_State&0x10)==0) {h=05,m=55,s=0,h1=1; DSY_BUFFER[0]=DSY_CODE[h/10]; DSY_BUFFER[1]=DSY_CODE[h]; DSY_BUFFER[3]=DSY_CODE[m/10]; DSY_BUFFER[4]=DSY_CODE[m]; DSY_BUFFER[6]=DSY_CODE[h1/10]; DSY_BUFFER[7]=DSY_CODE[h1]; } else if ((Key_State&0x20)==0)Increase_Hour1(); EA=1; } } } }

    4.单击“Project菜单/Options for Target”选项或者点击工具栏的“option for target”按钮,弹出窗口,点击Output将Create HEX File打勾 5.KeilC与Proteus连接仿真调试 点击仿真后,得到如下图的结果: 随着仿真的运行,能听到喇叭的声音,也能看到显示屏的数字,就算仿真成功了。

    Processed: 0.010, SQL: 8