ARM开发环境入门

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档


一.stm32简单程序编译(LED闪烁)

1)创建新文件

1>打开Keil uVision5,点击Project,然后点击New uVision Project

在这里插入图片描述
在这里插入图片描述

2>选择使用的stm32芯片
在这里插入图片描述
3>对Run-Time Environment进行设置,点击ok,完成工程创建
在这里插入图片描述

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);//延时时间
     }
}

3)编译结果

在这里插入图片描述

4)调试

1>点击Options for target
2>选择Debug,左侧选择Use Simulator,右侧选择ULINK2/ME Cortex Debugger,最后点击Settings
在这里插入图片描述

在这里插入图片描述
3>弹出该框点ok
4>选择Start Debug,然后利用调试工具开始进行调试操作
在这里插入图片描述
5>结束调制,再次点击Start Debug,调制结束

二.51程序设计和仿真

1.创建工程

1>在proteus中创建一个新的工程
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
2>在protues中绘图,连接元器件
在这里插入图片描述

2)在keil中创建文件

1>创建一个新工程,建立一个新文件,在新文件中输入代码

#include<reg51.h>
#define uchar unsigned char
#define uint unsigned int
sbit rs=P1^0;
sbit rw=P1^1;
sbit e=P1^2;

sbit FMQ=P2^3;

unsigned char year=20,month=4,day=24,week=5,sec=55,min=59,hour=7;

unsigned char DL_hour1=8;
unsigned char DL_min1=0;

unsigned char DL_hour2=8;
unsigned char DL_min2=45;

unsigned char DL_hour3=8;
unsigned char DL_min3=55;

unsigned char DL_hour4=9;
unsigned char DL_min4=40;

unsigned char DL_hour5=10;
unsigned char DL_min5=10;

unsigned char DL_hour6=10;
unsigned char DL_min6=55;

unsigned char DL_hour7=11;
unsigned char DL_min7=5;

unsigned char DL_hour8=11;
unsigned char DL_min8=50;

unsigned char DL_hour9=14;
unsigned char DL_min9=0;

unsigned char DL_hour10=14;
unsigned char DL_min10=40;

unsigned char DL_hour11=14;
unsigned char DL_min11=55;

unsigned char DL_hour12=15;
unsigned char DL_min12=40;

unsigned char DL_hour13=16;
unsigned char DL_min13=10;

unsigned char DL_hour14=16;
unsigned char DL_min14=55;

unsigned char DL_hour15=17;
unsigned char DL_min15=5;

unsigned char DL_hour16=17;
unsigned char DL_min16=50;

void LCD_YANSHI(unsigned int i)
{
while(i–);
}
void Lcd1602_W_C(unsigned char c)
{
LCD_YANSHI(500);
rs=0;
rw=0;
e=0;
P0=c;
e=1;
LCD_YANSHI(10);
e=0;
}
void Lcd1602_W_D(unsigned char dat)
{
LCD_YANSHI(500);
rs=1;
rw=0;
e=0;
P0=dat;
e=1;
LCD_YANSHI(10);
e=0;
rs=0;
}
void Lcd1602_Init()
{
LCD_YANSHI(500);
Lcd1602_W_C(0x38);
Lcd1602_W_C(0x06);
Lcd1602_W_C(0x0c);
Lcd1602_W_C(0x01);
}

void DISPLAY();
void Timer_INIT();

void main()
{
Lcd1602_Init();
Timer_INIT();
while(1)
{
DISPLAY();
}
}

void Timer_INIT()
{
TMOD = 0x01; //选择工作方式1
TH0=0Xfc;
TL0=0X18; //1ms
EA = 1; //打开总中断
ET0 = 1; //打开定时器0中断
TR0 = 1; //启动定时器0
}

void DISPLAY()
{
Lcd1602_W_C(0x80+3);
Lcd1602_W_D(2);
Lcd1602_W_D(0);
Lcd1602_W_D(0+year/10);
Lcd1602_W_D(0+year%10);
Lcd1602_W_D(/);
Lcd1602_W_D(0+month/10);
Lcd1602_W_D(0+month%10);
Lcd1602_W_D(/);
Lcd1602_W_D(0+day/10);
Lcd1602_W_D(0+day%10);
//–显示时钟–//
Lcd1602_W_C(0xC0+4);
Lcd1602_W_D(0+hour/10);
Lcd1602_W_D(0+hour%10);
Lcd1602_W_D(:);
Lcd1602_W_D(0+min/10);
Lcd1602_W_D(0+min%10);
Lcd1602_W_D(:);
Lcd1602_W_D(0+sec/10);
Lcd1602_W_D(0+sec%10);
}

void time0() interrupt 1
{
static unsigned int j;
TH0=0Xfc;
TL0=0X18; //1ms
j++;
if(j1000) //1s
{
j=0;
sec++;
if(sec60) //秒满60清零
{
sec=0;
min++;
if(min60) //分满60清零
{
min=0;
hour++;
if(hour24)
{
hour=0;
day++;
week++;
if(week>7)
{
week=0;
}
if(((month1)||(month3)||(month5)||(month7)||(month8)||(month10)||(month12))
&&(day32)
)
{
day=1;
month++;
if(month13)
{
year++;
month=0;
}
}
if(((month4)||(month6)||(month9)||(month11))
&&(day31)
)
{
day=1;
month++;
}
if((((year%4)0)&&(month2))
&&(day30)
)
{
day=1;
month++;
}
if((((year%4)!=0)&&(month2))
&&(day==29)
)
{
day=1;
month++;
}
}
}
}

	if((((hour==DL_hour1)&&(min==DL_min1))||		
		((hour==DL_hour2)&&(min==DL_min2))||	    
		((hour==DL_hour3)&&(min==DL_min3))||	    
		((hour==DL_hour4)&&(min==DL_min4))||
		((hour==DL_hour5)&&(min==DL_min5))||		
		((hour==DL_hour6)&&(min==DL_min6))||	    
		((hour==DL_hour7)&&(min==DL_min7))||	    
		((hour==DL_hour8)&&(min==DL_min8))||
		((hour==DL_hour9)&&(min==DL_min9))||		
		((hour==DL_hour10)&&(min==DL_min10))||	    
		((hour==DL_hour11)&&(min==DL_min11))||	    
		((hour==DL_hour12)&&(min==DL_min12))||
		((hour==DL_hour13)&&(min==DL_min13))||		
		((hour==DL_hour14)&&(min==DL_min14))||	    
		((hour==DL_hour15)&&(min==DL_min15))||	    
		((hour==DL_hour16)&&(min==DL_min16)))&&(week<=5))		
	{
		FMQ=0;
	}
	else
	{
		FMQ=1;
	}
}
}

2>然后进行编译和仿真

在这里插入图片描述
调试结果如下

在这里插入图片描述
说明过程与结果是正确的

3)将文件与protues建立联系

1>在protues中双击单片机,在弹出框中选择文件的路径,点击ok
2>点击右下角运行标识,运行结果如下

在这里插入图片描述

三.总结

本次作业是对单片机的软硬件设计,不但要使用keil来进行编程,还要使用protues对该实验的硬件进行仿真。不但要学会对程序的编写,而且要懂得对protues的使用,还要将keil编写的程序导入到protues当中,所以可以通过实验来熟悉简单的Proteus操作,达到更加熟练的程度。

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值