一、所需软件环境
我使用的是MDK5.10以及两个包
ARM.CMSIS.3.20.4.pack
Keil.STM32F1xx_DFP.1.0.4.pack
仿真实验所需软件Proteus 8 Professional
二、参考教程
手动配置环境点我跳转B站
MDK5内配置湖广午王代码编译可能会出现很多错误。我是网上随便找的一个代码。
Proteus 8 Professional入门介绍这里
三、实验及步骤
(一)MDK5的stm32开发
1.安装ARM和STM包
安装完MDK5.10后,安装所需的两个包,其它包以后有需要再安装。建议离线安装。双击即可安装。
2.新建工程
点击Project,再点击新建工程,选择路径。
3.选择芯片
理论上一类芯片都可以用以下代码。我选择的是STM32F103C8。
4.配置环境
选择以下环境。如果选择GPIO,会出现很多的问题,建议先不要选。
5.编写主程序
//宏定义,用于存放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 8 Professional仿真
打开软件,选择新建工程。
注:AT89C51芯片所需的编译包,需要到官网或者其它渠道获得。我是在网上找的.hex文件和代码,用来熟悉操作。
1.连接电路
2.编辑对应的.hex文件
左键点击AT89C51芯片,在Program File里选择.hex文件
也可以右键点击AT89C51芯片,点击Edit Source Code,选择环境和编译器,再编写代码进行编译。(参考这里)
3.运行效果
四、出现的问题与总结
编译stm32时出现以下问题
建议新手先不要用GPIO,如果有大佬能够帮忙解答,十分感谢!(已经百度过了,还是无法解决)
如果出现.\test2.axf: error: L6002U: Could not open file .\main.o: No such file问题可以解决,但是用户的一些数据会改变(仅修改用户)。
总结:新手用MDK来开发stm先要熟悉流程,然后再深入学习其中的一些原理和操作。