最简单的STM32入门教程----闪烁LED

本文讲述的是如何从零开始,使用keil建立一个简单的STM32的工程,并闪烁LED灯,给小白看。

第零步,当然首先你得有一个STM32的板子,其IO口上接了一个LED。。。

第一步,建立一个文件夹0.0

第二步,打开keil,建立工程

在弹出来的对话框中选择你所用的STM32的芯片。

在接下来弹出来的对话框中选择是,这样keil就帮我们建立好了启动文件。

第三步,新建一个main.c文件,并添加到工程中。

点击New按钮,建立一个文本文件。

在建立的文本文件中输入C中的main函数

点击保存

保存后,将文件添加到工程中

第四步,点击编译

可以看到keil有报错

错误信息为:没有定义的符号SystemInit,这是因为在启动文件中有调用SystemInit函数,但是我们没有定义它,如下图:

暂时不用理会上述启动文件中汇编的含义,只需在main.c中添加该函数即可消除该错误。

修改后再编译,程序没有报错了。至此,一个STM32的工程就建立完成了。

 第五步,将下面的代码复制粘贴

#define PERIPH_BASE           ((unsigned int)0x40000000)
#define APB2PERIPH_BASE       (PERIPH_BASE + 0x10000)
#define GPIOA_BASE            (APB2PERIPH_BASE + 0x0800)
#define GPIOB_BASE            (APB2PERIPH_BASE + 0x0C00)
#define GPIOC_BASE            (APB2PERIPH_BASE + 0x1000)
#define GPIOD_BASE            (APB2PERIPH_BASE + 0x1400)
#define GPIOE_BASE            (APB2PERIPH_BASE + 0x1800)
#define GPIOF_BASE            (APB2PERIPH_BASE + 0x1C00)
#define GPIOG_BASE            (APB2PERIPH_BASE + 0x2000)
#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 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 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;

#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;
		Delay_ms(500);
		LED0=1;
		Delay_ms(500);
	}
}

void SystemInit(void)
{

}

  

下面一段是对代码的简单讲解,可不用太深入。

/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

STM32 GPIO输出高低电平的完整流程如下:

配置时钟→配置IO口→IO口输出高地电平

1,STM32的时钟是可以PLL的。但是我这里为了简化操作,就不配置,只使用默认的时钟----内部8MZH振荡器。

2,硬件上,我的LED灯是接到PA8。那么,我要怎么控制PA8呢?

STM32不能像单片机一样,一上来就能操作IO口。要操作STM32的IO口,首先要配置IO口。

void LEDInit(void)

{

RCC->APB2ENR|=1<<2;

GPIOA->CRH&=0XFFFFFFF0;

GPIOA->CRH|=0X00000003;

}

其中RCC->APB2ENR|=1<<2;是使能GPIOA的时钟。若是要使能GPIOB的时钟则是RCC->APB2ENR|=1<<3;其他的以此类推。

GPIOA->CRH&=0XFFFFFFF0;

GPIOA->CRH|=0X00000003;

是配置PA8为推挽输出,50MHZ。

若是要设置PA9则是:

GPIOA->CRH&=0XFFFFFF0F;

GPIOA->CRH|=0X00000030;

其他的以此类推,若是PA0~PA7则将CRH改为CRL就行。

若是要配置其他GPIO口,将GPIOA改成GPIOB,GPIOC。。。就行。,

3,我们知道51单片机是可以单独控制每个IO口的,STM32也可以做到,这个机制叫做Bit-Bond。

参考《Cortex-M3权威指南》可知道,只要找到PA8输出寄存器在Bit-Bond的地址,即可操作PA8的输出。地址的计算方法如下图:

《Cortex-M3权威指南》还给出了C语言宏定义的方法,我们可以直接使用。

/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

将上述代码输入后,点开配置选项,配置如下:

勾上 Create HEX File

选择J-Link下载工具

再次build,然后下载就大功告成啦。

综上可以看出,若是不配置STM32的PLL,那么相对于51单片机,STM32闪烁LED灯只不过多了一个步骤----配置GPIO口为输出而已。

时间: 2024-12-21 04:32:34

最简单的STM32入门教程----闪烁LED的相关文章

STM32使用TIM闪烁LED——输出比较方式

STM32定时器输出比较器可以直接操作对应的GPIO,在计数器值等于比较寄存器的值时,对应的GPIO可以有以下四种动作 无动作 激活 取消激活 翻转 激活电平由输出极性寄存器决定 将LED置于定时器输出比较对应的GPIO上,对它的操作将非常方便,完全由硬件完成,不消耗CPU时间,无需中断,每个定时器都对应有4个输出比较,可以轻松实现4个LED的流水灯效果 宏定义部分如下 #define USER_TIM_PSC 36000-1 #define USER_TIM_PERIOD 1000-1 #de

AVR单片机教程——闪烁LED

上次我们把LED点亮了.你可能已经试过把 LED_RED 换成其他灯,也可能已经用 led_on() 把所有LED一起点亮了.但是LED点亮以后,程序就退出了,之后LED一直没有暗,直到没有供电.这一次,我们用程序来控制LED的亮和暗. 新建一个C executable项目,选择ATmega324PA单片机,在项目属性中添加库libee1,将配置改为Release.这是本教程现阶段中每一次新建项目都要做的.我刚才尝试用project template简化,然而设置无法导入. 默认生成的main.

简单的STM32 汇编程序—闪烁LED

要移植操作系统,汇编是道不得不跨过去的坎.所以承接上篇的思路,我准备用汇编写一个简单的闪烁LED灯的程式.以此练习汇编,为操作系统做准备. 第一步,还是和上篇一样,建立一个空的文件夹. 第二步,因为是要用汇编来写程式,所以不需要启动代码,这里选择否. 第三步,建立一个.s文件,并把文件添加到工程中. 第四步,在LED.s文件中添加如下代码. LED0 EQU 0x422101a0 RCC_APB2ENR EQU 0x40021018 GPIOA_CRH EQU 0x40010804 Stack_

程序员,一起玩转GitHub版本控制,超简单入门教程 干货2

本GitHub教程旨在能够帮助大家快速入门学习使用GitHub,进行版本控制.帮助大家摆脱命令行工具,简单快速的使用GitHub. 做全栈攻城狮-写代码也要读书,爱全栈,更爱生活. 更多原创教程请关注头条号.每日更新.也可以添加小编微信:fullstackCourse.一起交流,获取最新全栈教程信息.因为FQ原因,不能下载客户端的同仁,可以关注后回复“GitHub客户端”获取安装软件. 上篇教程:GitHub这么火,程序员你不学学吗? 超简单入门教程 干货 GitHub概念部分出现了一丝纰漏.为

iBatis简单入门教程

iBatis 简介: iBatis 是apache 的一个开源项目,一个O/R Mapping 解决方案,iBatis 最大的特点就是小巧,上手很快.如果不需要太多复杂的功能,iBatis 是能够满足你的要求又足够灵活的最简单的解决方案,现在的iBatis 已经改名为Mybatis 了. 官网为:http://www.mybatis.org/ 搭建iBatis 开发环境: 1 .导入相关的jar 包,ibatis-2.3.0.677.jar .mysql-connector-java-5.1.6

开源TinyXML 最简单的入门教程

TinyXML是目前非常流行的一款基于DOM模型的XML解析器,简单易用且小巧玲珑,非常适合存储简单数据,配置文件. 该项目属于开源项目,在sourceforge上边的链接是:http://sourceforge.net/projects/tinyxml/ 当前最新版本是2.6.2 先看一下源码文档的结构: Docs是帮助文档,里边有非常多的使用说明,仅仅截一张图看一下: 具体根据需要再看 我们使用的是它的库,可以是静态的也可以是动态库,我就用静态库了,将这里边的几个头文件和源文件一起创建一个工

引脚5接GUN简单的闪烁LED程序

简单的闪烁LED程序,直接上传运行,代码贴在这里做个笔记 int ledPin=5; //设定控制LED的数字IO脚 void setup(){ pinMode(ledPin,OUTPUT);//设定数字IO口的模式,OUTPUT 为输出 } void loop(){ digitalWrite(ledPin,HIGH); //设定PIN5脚为HIGH = 5V左右 delay(5000); //延时时间,5000 = 5秒,即亮5秒 digitalWrite(ledPin,LOW); //设定P

【转】WCF入门教程六[一个简单的Demo]

一.前言 前面的几个章节介绍了很多理论基础,如:什么是WCF.WCF中的A.B.C.WCF的传输模式.本文从零开始和大家一起写一个小的WCF应用程序Demo. 大多框架的学习都是从增.删.改.查开始来学习的,我们学习WCF也是一样的.从简单来看(不包括安全.优化等相关问题),WCF的增删改查和WebForm相差无几.WCF只是把具体"实现"写在"Service端",而"调用"放在了"Client端".觉得有帮助别忘了点个赞哈,

无废话WCF入门教程六[一个简单的Demo]

wcf技术交流,同学习,同进步. 群号:89718412 一.前言 前面的几个章节介绍了很多理论基础,如:什么是WCF.WCF中的A.B.C.WCF的传输模式.本文从零开始和大家一起写一个小的WCF应用程序Demo. 大多框架的学习都是从增.删.改.查开始来学习的,我们学习WCF也是一样的.从简单来看(不包括安全.优化等相关问题),WCF的增删改查和WebForm相差无几.WCF只是把具体“实现”写在“Service端”,而“调用”放在了“Client端”.觉得有帮助别忘了点个赞哈,谢谢哦~ 二