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

要移植操作系统,汇编是道不得不跨过去的坎。所以承接上篇的思路,我准备用汇编写一个简单的闪烁LED灯的程式。以此练习汇编,为操作系统做准备。

第一步,还是和上篇一样,建立一个空的文件夹。

第二步,因为是要用汇编来写程式,所以不需要启动代码,这里选择否。

第三步,建立一个.s文件,并把文件添加到工程中。

第四步,在LED.s文件中添加如下代码。

LED0 EQU 0x422101a0
RCC_APB2ENR EQU 0x40021018
GPIOA_CRH EQU 0x40010804

Stack_Size      EQU     0x00000400

                AREA    STACK, NOINIT, READWRITE, ALIGN=3
Stack_Mem       SPACE   Stack_Size
__initial_sp

                AREA    RESET, DATA, READONLY

__Vectors       DCD     __initial_sp               ; Top of Stack
                DCD     Reset_Handler              ; Reset Handler

                AREA    |.text|, CODE, READONLY

                THUMB
                REQUIRE8
                PRESERVE8

                ENTRY
Reset_Handler
                BL LED_Init
MainLoop        BL LED_ON
                BL Delay
                BL LED_OFF
                BL Delay

                B MainLoop

LED_Init
                PUSH {R0,R1, LR}

                LDR R0,=RCC_APB2ENR
                ORR R0,R0,#0x04
                LDR R1,=RCC_APB2ENR
                STR R0,[R1]

                LDR R0,=GPIOA_CRH
                BIC R0,R0,#0x0F
                LDR R1,=GPIOA_CRH
                STR R0,[R1]

                LDR R0,=GPIOA_CRH
                ORR R0,R0,#0x03
                LDR R1,=GPIOA_CRH
                STR R0,[R1]

                MOV R0,#1
                LDR R1,=LED0
                STR R0,[R1]

                POP {R0,R1,PC}

LED_ON
                PUSH {R0,R1, LR}    

                MOV R0,#0
                LDR R1,=LED0
                STR R0,[R1]

                POP {R0,R1,PC}

LED_OFF
                PUSH {R0,R1, LR}    

                MOV R0,#1
                LDR R1,=LED0
                STR R0,[R1]

                POP {R0,R1,PC}             

Delay
                PUSH {R0,R1, LR}

                MOVS R0,#0
                MOVS R1,#0
                MOVS R2,#0

DelayLoop0
                ADDS R0,R0,#1

                CMP R0,#330
                BCC DelayLoop0

                MOVS R0,#0
                ADDS R1,R1,#1
                CMP R1,#330
                BCC DelayLoop0

                MOVS R0,#0
                MOVS R1,#0
                ADDS R2,R2,#1
                CMP R2,#15
                BCC DelayLoop0

                POP {R0,R1,PC}    

    ;         NOP
             END

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

代码的简单讲解

1,预定义

LED0 EQU 0x422101a0 ;PA8的Bit-Bond地址。

RCC_APB2ENR EQU 0x40021018

GPIOA_CRH EQU 0x40010804

为方便操作,给每个需要用到的寄存器地址定义一个名字,类似于C语言的#define。PA8的Bit-Bond地址的计算方法可按上篇文章中C语言的算法算出。后面的两个地址时固定的,可从STM32的手册查询,或者根据ST官方的库文件查找计算。

2,分配栈空间

Stack_Size EQU 0x00000400

AREA STACK, NOINIT, READWRITE, ALIGN=3

Stack_Mem SPACE Stack_Size

__initial_sp

这一段摘自启动文件。要读懂这段代码,首先要了解两个命令。

AREA命令:AREA 命令指示汇编器汇编一个新的代码段或数据段。段是独立的、指定的、不可见的代码或数据块,它们由链接器处理。格式如下:

AREA 段名,段属性1,段属性2,段属性3。。。

AREA STACK, NOINIT, READWRITE, ALIGN=3

NOINIT: = NO Init,不初始化。

READWRITE : 可读,可写。

ALIGN =3 : 2^3 对齐,即8字节对齐。

SPACE命令:SPACE 命令保留一个用零填充的存储器块。

所以整段的意思为:分配一个STACK段,该段不初始化,可读写,按8字节对齐。分配一个大小为Stack_Size的存储空间,并使栈顶的地址为__initial_sp。

3,分配向量表

AREA RESET, DATA, READONLY

__Vectors DCD __initial_sp ; Top of Stack

DCD Reset_Handler ; Reset Handler

这里的向量可参考我之前写的《STM32向量表详细分析》。

4,开始代码段

AREA |.text|, CODE, READONLY

通知汇编器,开始代码段。

THUMB

REQUIRE8

PRESERVE8

这段的意思是,汇编器支持THUMB指令,代码段按8字节对齐。

ENTRY命令:声明整个程式的入口点,入口点有且仅有一个。不管哪种语言,编译器都得有个入口点,这没什么好说的。

5,程序正式开始。

后面的代码,皆用标准的THUMB2汇编指令。首先了解下代码中用到的指令。

BL:带链接的跳转指令。当使用该指令跳转时,当前地址(PC)会自动送入LR寄存器。

B:无条件跳转。

PUSH和POP:可以看到,所有的子程序都是由PUSH和POP包起来的。借用一张图解释下这两个指令。

据上可知,PUSH {R0,R1, LR}的意思即把R0,R1,LR中的值放入堆栈中。由于主程式中使用BL跳转指令,所以LR中的值实际上就是当前PC的值。而POP {R0,R1,PC}的意思即是将栈中之前存的R0,R1,LR的值返还给R0,R1,PC。这样就完成了子程序的返回。

LDR和STR:寄存器的装载和存储指令。

LDR是把地址装载到寄存器中(比如R0)。

STR是把值存储到寄存器所指的地址中。

举个例子:

MOV R0,#1 ;将立即数1送入R0.

LDR R1,=LED0;将PA8 bit-bond的地址送入R1.

STR R0,[R1];将R0的值,也就是1,送给R1中的值所指向的地址中,也就是PA8的bit-bond地址。

上面三句话的意思即是将PA8置1。

ORR和BIC:

ORR 按位或操作。ORR R0,R0,#0x04意思即将R0中的数或上0x04,再将结果送往R0中。实际意思就是将R0的第二位置1,其他位不变。

BIC 先把立即数取反,再按位与。

CMP和BCC:CMP是比较两个数,相等或大于则将标志位C置位,否则将C清零。BCC是个组合指令,实际为B+CC,意思是如果C=0则跳转。

CMP R2,#15; 计算R2-15的值,若是R2<15,则C=0;若是R2>=15,则C=1。

BCC DelayLoop0;若是C=0,则跳到DelayLoop0,若是c=1,则不跳转。

以上就是代码段相关指令的介绍,相信了解了这些指令的含义,要理解代码并不困难。

整个代码的结构和上篇用C语言写的基本是一样的。可参照理解

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

第五步,编译,下载。

编译后,会有一个警告 No section matches pattern……可不用管。下载后,LED灯正常闪烁。

时间: 2024-10-21 13:38:17

简单的STM32 汇编程序—闪烁LED的相关文章

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

本文讲述的是如何从零开始,使用keil建立一个简单的STM32的工程,并闪烁LED灯,给小白看. 第零步,当然首先你得有一个STM32的板子,其IO口上接了一个LED... 第一步,建立一个文件夹0.0 第二步,打开keil,建立工程 在弹出来的对话框中选择你所用的STM32的芯片. 在接下来弹出来的对话框中选择是,这样keil就帮我们建立好了启动文件. 第三步,新建一个main.c文件,并添加到工程中. 点击New按钮,建立一个文本文件. 在建立的文本文件中输入C中的main函数 点击保存 保

引脚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

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

bb_black_学习笔记&mdash;&mdash;(4)闪烁LED之shell命令

上一篇笔记在终端输入shell命令实现了LED灯的点亮与熄灭,作为初学者,已经兴奋了一阵,因为终于有了零的突破.手动点亮LED总会比较麻烦,还是得通过程序让它自己去点亮与熄灭,这样才更好玩.这篇笔记里,笔者通过学习shell脚本,实现LED的闪烁,这样才好玩.本篇笔记还是先从现象看起,然后在分析其实现原理. 第一步:实现LED灯闪烁 1.先来个简单的shell脚本,了解一下shell脚本的使用流程.下图为创建一个名为hello_sh的shell脚本编辑执行的过程. 几点说明: (1)shell脚

AVR单片机教程——闪烁LED

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

iOS简单动画效果:闪烁、移动、旋转、路径、组合

#define kDegreesToRadian(x) (M_PI * (x) / 180.0) #define kRadianToDegrees(radian) (radian*180.0)/(M_PI) - (void)viewDidLoad { [superviewDidLoad]; self.title = @"测试动画"; self.view.backgroundColor = [UIColorlightGrayColor]; myTest1 = [[UILabelalloc

CC2640R2F&amp;TI-RTOS 创建 一个任务 和 使用 信号量 超时来闪烁 LED灯

/* * data_process.c * * Created on: 2018年7月5日 * Author: admin */ #include <ti/sysbios/knl/Task.h> #include <ti/sysbios/knl/Clock.h> #include <ti/sysbios/knl/Event.h> #include <ti/sysbios/knl/Queue.h> #include <ti/sysbios/knl/Sem

Keil新建STM32工程(LED灯)

1.新建项目 2.选择芯片,若没有就下载 3.导入文件及文件夹 4.导入文件路径 5.编写LED程序和main 原文地址:https://www.cnblogs.com/LcVong/p/12549345.html

[stm32][ucos][ucgui] 2、LED闪烁、串口、滑块、文本编辑框简单例程

上一篇:[stm32][ucos] 1.基于ucos操作系统的LED闪烁.串口通信简单例程 * 内容简述: 本例程操作系统采用ucos2.86a版本, 建立了7个任务            任务名                                             优先级            APP_TASK_START_PRIO                               2            主任务            APP_TASK_USER_