单片机学习小组
直播中

张涛

7年用户 1328经验值
私信 关注

基于stm32CubeMX的流水灯和stm32是如何实现串口通信的

基于STM32CubeMX的流水灯和stm32是如何实现串口通信的?有哪些基本步骤?

回帖(2)

张彪其

2022-1-19 10:01:45
一.安装环境

stm32cubemx使用jave编程,需安装jdk。
1.安装jdk
安装完成。
2.安装STM32CubeMx

安装路径不要有中文
3.安装固件库

打开stm32cubeMX,在help下选择manage然后选择STM32F1然后1.8.4版本,然后install now
出现绿点表示安装成功。
二.流水灯点亮

1.建立工程


2. 选择F103C8芯片

3.修改debug

双击103C8芯片进入,点击system core 选择sys,将debug选择serial wire,右边是芯片的引脚图
4.修改HSI

将system clock mux从HSI设为PLLOCK
5.设置时钟RCC

在High Speed Clock选择Crystal/Ceramic Resonator 右边的引脚图选择自己选择芯片的引脚连线,我选择的是PA12,PB1,PC14,并且选择GPIO_OUT

6.GPIO output level

GPIO output level选择high

7.建立项目

在Toolchain/IDE选择MDK-ARM
进入code generate界面,选择生成初始化.c/.h文件,后面点击generate code,然后直接打开文件。

打开直接编写的代码

在while循环中加入亮灯熄灯代码,可直接调用亮灯熄灯函数,在stm32cubeMX已生成,延时函数也自动生成,500刚好对应0.5s延迟,1000则为1s

        HAL_GPIO_WritePin(GPIOA,GPIO_PIN_12,GPIO_PIN_SET);//PA12熄灯
                HAL_Delay(500);//延时0.5s
                HAL_GPIO_WritePin(GPIOA,GPIO_PIN_12,GPIO_PIN_RESET);//PA12亮灯
                HAL_Delay(500);//延时0.5s               
                HAL_GPIO_WritePin(GPIOB,GPIO_PIN_1,GPIO_PIN_SET);//PB1熄灯
                HAL_Delay(500);//延时0.5s
                HAL_GPIO_WritePin(GPIOB,GPIO_PIN_1,GPIO_PIN_RESET);//PB1亮灯
                HAL_Delay(500);//延时0.5s
               
                HAL_GPIO_WritePin(GPIOC,GPIO_PIN_14,GPIO_PIN_SET);//PC14熄灯
                HAL_Delay(500);//延时0.5s
               
                HAL_GPIO_WritePin(GPIOC,GPIO_PIN_14,GPIO_PIN_RESET);//PC14亮灯
                HAL_Delay(500);//延时0.5s


8.烧录

编译无错,然后生成hex文件进行烧录。

通电下将boot0置0,然后再次通电后可运行。
9.结果
三.串口通信

1.创建新工程


汇编语言不需要添加CMSIS和Device的starup 和core
2.添加汇编代码
代码

;RCC寄存器地址映像            
RCC_BASE            EQU    0x40021000
RCC_CR              EQU    (RCC_BASE + 0x00)
RCC_CFGR            EQU    (RCC_BASE + 0x04)
RCC_CIR             EQU    (RCC_BASE + 0x08)
RCC_APB2RSTR        EQU    (RCC_BASE + 0x0C)
RCC_APB1RSTR        EQU    (RCC_BASE + 0x10)
RCC_AHBENR          EQU    (RCC_BASE + 0x14)
RCC_APB2ENR         EQU    (RCC_BASE + 0x18)
RCC_APB1ENR         EQU    (RCC_BASE + 0x1C)
RCC_BDCR            EQU    (RCC_BASE + 0x20)
RCC_CSR             EQU    (RCC_BASE + 0x24)
                              
;AFIO寄存器地址映像            
AFIO_BASE           EQU    0x40010000
AFIO_EVCR           EQU    (AFIO_BASE + 0x00)
AFIO_MAPR           EQU    (AFIO_BASE + 0x04)
AFIO_EXTICR1        EQU    (AFIO_BASE + 0x08)
AFIO_EXTICR2        EQU    (AFIO_BASE + 0x0C)
AFIO_EXTICR3        EQU    (AFIO_BASE + 0x10)
AFIO_EXTICR4        EQU    (AFIO_BASE + 0x14)
                                                           
;GPIOA寄存器地址映像              
GPIOA_BASE          EQU    0x40010800
GPIOA_CRL           EQU    (GPIOA_BASE + 0x00)
GPIOA_CRH           EQU    (GPIOA_BASE + 0x04)
GPIOA_IDR           EQU    (GPIOA_BASE + 0x08)
GPIOA_ODR           EQU    (GPIOA_BASE + 0x0C)
GPIOA_BSRR          EQU    (GPIOA_BASE + 0x10)
GPIOA_BRR           EQU    (GPIOA_BASE + 0x14)
GPIOA_LCKR          EQU    (GPIOA_BASE + 0x18)
                                                      
;GPIO C口控制                  
GPIOC_BASE          EQU    0x40011000
GPIOC_CRL           EQU    (GPIOC_BASE + 0x00)
GPIOC_CRH           EQU    (GPIOC_BASE + 0x04)
GPIOC_IDR           EQU    (GPIOC_BASE + 0x08)
GPIOC_ODR           EQU    (GPIOC_BASE + 0x0C)
GPIOC_BSRR          EQU    (GPIOC_BASE + 0x10)
GPIOC_BRR           EQU    (GPIOC_BASE + 0x14)
GPIOC_LCKR          EQU    (GPIOC_BASE + 0x18)
                                                           
;串口1控制                       
USART1_BASE         EQU    0x40013800
USART1_SR           EQU    (USART1_BASE + 0x00)
USART1_DR           EQU    (USART1_BASE + 0x04)
USART1_BRR          EQU    (USART1_BASE + 0x08)
USART1_CR1          EQU    (USART1_BASE + 0x0c)
USART1_CR2          EQU    (USART1_BASE + 0x10)
USART1_CR3          EQU    (USART1_BASE + 0x14)
USART1_GTPR         EQU    (USART1_BASE + 0x18)
                           
;NVIC寄存器地址               
NVIC_BASE           EQU    0xE000E000
NVIC_SETEN          EQU    (NVIC_BASE + 0x0010)     
;SETENA寄存器阵列的起始地址
NVIC_IRQPRI         EQU    (NVIC_BASE + 0x0400)     
;中断优先级寄存器阵列的起始地址
NVIC_VECTTBL        EQU    (NVIC_BASE + 0x0D08)     
;向量表偏移寄存器的地址     
NVIC_AIRCR          EQU    (NVIC_BASE + 0x0D0C)     
;应用程序中断及复位控制寄存器的地址                                                
SETENA0             EQU    0xE000E100
SETENA1             EQU    0xE000E104
                           
                              
;SysTick寄存器地址            
SysTick_BASE        EQU    0xE000E010
SYSTICKCSR          EQU    (SysTick_BASE + 0x00)
SYSTICKRVR          EQU    (SysTick_BASE + 0x04)
                              
;FLASH缓冲寄存器地址映像     
FLASH_ACR           EQU    0x40022000
                             
;SCB_BASE           EQU    (SCS_BASE + 0x0D00)
                             
MSP_TOP             EQU    0x20005000               
;主堆栈起始值               
PSP_TOP             EQU    0x20004E00               
;进程堆栈起始值            
                           
BitAlias_BASE       EQU    0x22000000               
;位带别名区起始地址         
Flag1               EQU    0x20000200
b_flas              EQU    (BitAlias_BASE + (0x200*32) + (0*4))               
;位地址
b_05s               EQU    (BitAlias_BASE + (0x200*32) + (1*4))               
;位地址
DlyI                EQU    0x20000204
DlyJ                EQU    0x20000208
DlyK                EQU    0x2000020C
SysTim              EQU    0x20000210


;常数定义
Bit0                EQU    0x00000001
Bit1                EQU    0x00000002
Bit2                EQU    0x00000004
Bit3                EQU    0x00000008
Bit4                EQU    0x00000010
Bit5                EQU    0x00000020
Bit6                EQU    0x00000040
Bit7                EQU    0x00000080
Bit8                EQU    0x00000100
Bit9                EQU    0x00000200
Bit10               EQU    0x00000400
Bit11               EQU    0x00000800
Bit12               EQU    0x00001000
Bit13               EQU    0x00002000
Bit14               EQU    0x00004000
Bit15               EQU    0x00008000
Bit16               EQU    0x00010000
Bit17               EQU    0x00020000
Bit18               EQU    0x00040000
Bit19               EQU    0x00080000
Bit20               EQU    0x00100000
Bit21               EQU    0x00200000
Bit22               EQU    0x00400000
Bit23               EQU    0x00800000
Bit24               EQU    0x01000000
Bit25               EQU    0x02000000
Bit26               EQU    0x04000000
Bit27               EQU    0x08000000
Bit28               EQU    0x10000000
Bit29               EQU    0x20000000
Bit30               EQU    0x40000000
Bit31               EQU    0x80000000


;向量表
    AREA RESET, DATA, READONLY
    DCD    MSP_TOP            ;初始化主堆栈
    DCD    Start              ;复位向量
    DCD    NMI_Handler        ;NMI Handler
    DCD    HardFault_Handler  ;Hard Fault Handler
    DCD    0                  
    DCD    0
    DCD    0
    DCD    0
    DCD    0
    DCD    0
    DCD    0
    DCD    0
    DCD    0
    DCD    0
    DCD    0
    DCD    SysTick_Handler    ;SysTick Handler
    SPACE  20                 ;预留空间20字节
举报

王志宏

2022-1-19 10:01:49
;代码段
    AREA |.text|, CODE, READONLY
    ;主程序开始
    ENTRY                           
    ;指示程序从这里开始执行
Start
    ;时钟系统设置
    ldr    r0, =RCC_CR
    ldr    r1, [r0]
    orr    r1, #Bit16
    str    r1, [r0]
    ;开启外部晶振使能  
    ;启动外部8M晶振
                                            
ClkOk           
    ldr    r1, [r0]
    ands   r1, #Bit17
    beq    ClkOk
    ;等待外部晶振就绪
    ldr    r1,[r0]
    orr    r1,#Bit17
    str    r1,[r0]
    ;FLASH缓冲器
    ldr    r0, =FLASH_ACR
    mov    r1, #0x00000032
    str    r1, [r0]
            
    ;设置PLL锁相环倍率为7,HSE输入不分频
    ldr    r0, =RCC_CFGR
    ldr    r1, [r0]
    orr    r1, #(Bit18 :OR: Bit19 :OR: Bit20 :OR: Bit16 :OR: Bit14)
    orr    r1, #Bit10
    str    r1, [r0]
    ;启动PLL锁相环
    ldr    r0, =RCC_CR
    ldr    r1, [r0]
    orr    r1, #Bit24
    str    r1, [r0]
PllOk
    ldr    r1, [r0]
    ands   r1, #Bit25
    beq    PllOk
    ;选择PLL时钟作为系统时钟
    ldr    r0, =RCC_CFGR
    ldr    r1, [r0]
    orr    r1, #(Bit18 :OR: Bit19 :OR: Bit20 :OR: Bit16 :OR: Bit14)
    orr    r1, #Bit10
    orr    r1, #Bit1
    str    r1, [r0]
    ;其它RCC相关设置
    ldr    r0, =RCC_APB2ENR
    mov    r1, #(Bit14 :OR: Bit4 :OR: Bit2)
    str    r1, [r0]      

            
    ;PA9串口0发射脚
    ldr    r0, =GPIOA_CRH
    ldr    r1, [r0]
    orr    r1, #(Bit4 :OR: Bit5)         
    ;PA.9输出模式,最大速度50MHz  
    orr    r1, #Bit7
    and    r1, #~Bit6
    ;10:复用功能推挽输出模式
    str    r1, [r0]   


    ldr    r0, =USART1_BRR   
    mov    r1, #0x271
    str    r1, [r0]
    ;配置波特率-> 115200
                  
    ldr    r0, =USART1_CR1   
    mov    r1, #0x200c
    str    r1, [r0]
    ;USART模块总使能 发送与接收使能
    ;71 02 00 00   2c 20 00 00
            
    ;AFIO 参数设置            
    ;Systick 参数设置
    ldr    r0, =SYSTICKRVR           
    ;Systick装初值
    mov    r1, #9000
    str    r1, [r0]
    ldr    r0, =SYSTICKCSR           
    ;设定,启动Systick
    mov    r1, #0x03
    str    r1, [r0]
              
    ;切换成用户级线程序模式
    ldr    r0, =PSP_TOP                  
    ;初始化线程堆栈
    msr    psp, r0
    mov    r0, #3
    msr    control, r0
              
    ;初始化SRAM寄存器
    mov    r1, #0
    ldr    r0, =Flag1
    str    r1, [r0]
    ldr    r0, =DlyI
    str    r1, [r0]
    ldr    r0, =DlyJ
    str    r1, [r0]
    ldr    r0, =DlyK
    str    r1, [r0]
    ldr    r0, =SysTim
    str    r1, [r0]
               
;主循环            
main            
    ldr    r0, =Flag1
    ldr    r1, [r0]
    tst    r1, #Bit1                 
    ;SysTick产生0.5s,置位bit 1
    beq    main                  ;0.5s标志还没有置位      
     
    ;0.5s标志已经置位
    ldr    r0, =b_05s               
    ;位带操作清零0.5s标志
    mov    r1, #0
    str    r1, [r0]

    mov    r0, #'H'
    bl     send_a_char
       
        mov    r0, #'e'
    bl     send_a_char
       
        mov    r0, #'l'
    bl     send_a_char
       
        mov    r0, #'l'
    bl     send_a_char
       
        mov    r0, #'o'
    bl     send_a_char
       
        mov    r0, #' '
    bl     send_a_char
       
        mov    r0, #'W'
    bl     send_a_char
       
        mov    r0, #'o'
    bl     send_a_char
       
        mov    r0, #'r'
    bl     send_a_char
       
        mov    r0, #'l'
    bl     send_a_char
       
        mov    r0, #'d'
    bl     send_a_char
       
        mov    r0, #'n'
    bl     send_a_char
       
        b      main
            
              
            
;子程序 串口1发送一个字符
send_a_char
    push   {r0 - r3}
    ldr    r2, =USART1_DR   
    str    r0, [r2]
b1
    ldr    r2, =USART1_SR  
    ldr    r2, [r2]
    tst    r2, #0x40
    beq    b1
    ;发送完成(Transmission complete)等待
    pop    {r0 - r3}
    bx     lr
                                
;异常程序
NMI_Handler
    bx     lr


HardFault_Handler
    bx     lr
              
SysTick_Handler
    ldr    r0, =SysTim
    ldr    r1, [r0]
    add    r1, #1
    str    r1, [r0]
    cmp    r1, #500
    bcc    TickExit
    mov    r1, #0
    str    r1, [r0]
    ldr    r0, =b_05s  
    ;大于等于500次 清零时钟滴答计数器 设置0.5s标志位
    ;位带操作置1
    mov    r1, #1
    str    r1, [r0]
TickExit   
    bx     lr
                                                                           
    ALIGN            
    ;通过用零或空指令NOP填充,来使当前位置与一个指定的边界对齐
    END
编译生成hex文件

烧录与接受
首先使用mcuisp将芯片给清空,然后将上面生成的hex烧录进去,然后在使用串口调试助手
此时是在boot0为1情况下烧录。开串口调试助手,首先打开刚刚生成的hex文件,再点击发送文件,波特率默认是115200,1位停止位,无校验位。接着将boot0置0,点击reset即可接收到hello world.
四.keil5观测波形

1.环境设置


  设置Debug的参数,该程序是在无硬件条件下进行的仿真注意Dialog和Parameter,根据stm32使用的不同参数也不同。
2.观察波形


三个引脚在周期为0.5s交替出现,从而实现点灯。

串口调试分析波形

环境配置跟上面一致
引脚改变一下

观察波形

烧录的hello world程序的周期为0.5s,每相隔0.5s电平就会变化,从而周期性的输出。

五.总结

本次实验相比于上一个实验来说感觉没有那么复杂,有了前面的基础,做起来也比较的快,没有之前疯狂看代码的那种焦虑,直接生成就很棒,只需要加一点点。
举报

更多回帖

发帖
×
20
完善资料,
赚取积分