STM32串口通讯实现——USB转串口

一、RS232,TTL简介

RS232是个人计算机的通讯接口之一,一般会有两组RS323接口,分别为COM1和COM2,电平标准为+12V为逻辑负,-12为逻辑正。
TTL电平为5V为逻辑正,0为逻辑负,这样的数据通信及电平规定方式,被称做TTL(晶体管-晶体管逻辑电平)信号系统。这是计算机处理器控制的设备内部各部分之间通信的标准技术。

二、串口通讯原理

了解了RS232和TTL两种通信标准后,就可以开始了解串口通信USART了。
UART :通用异步收发传输器,它将要传输的资料在串行通信与并行通信之间加以转换。作为把并行输入信号转成串行输出信号的芯片。同样它是串行通信接口 UART只有数据线收和发,并无时钟线,故为异步串行通信接口,可以实现全双工传输和接收;在嵌入式中,常用与上位机与外设通信。
串口通信的概念非常简单,串口按位发送和接收字节。尽管比按字节的并行通信慢,但是串口可以在使用一根线发送数据的同时用另一根线接收数据。它很简单并且能够实现远距离通信。一般来说有3种串口通讯方式。

  1. 两设备通过232标准通讯
    用控制器发出TTL电平,然后经过电平转换芯片转换为RS232电平,然后通过D89接口进行通讯

  2. 用USB转串口进行通讯
    用控制器发出TTL电平,然后经过电平转换芯片转换,然后通过USB接口和USB转串口进行通讯,需要安装CH340驱动。本博客就主要针对这种方式进行串口通讯。

  3. TTL电平之间直接通讯
    不需要控制器和转换芯片,直接使用TTL电平进行通讯。

三、生成文件

1.建立工程

打开Keil5软件,新建一个工程,然后添加一个.s文件,点击魔法棒选择Output选项,勾选Creat Hex File。
在这里插入图片描述

2.输入代码

在.s文件当中输入以下代码,代码是在别人的博客当中复制的。

;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字节 








                 
;代码段 
    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]      


    ;IO端口设置 
    ldr    r0, =GPIOC_CRL 
    ldr    r1, [r0] 
    orr    r1, #(Bit28 :OR: Bit29)          
    ;PC.7输出模式,最大速度50MHz  
    and    r1, #(~Bit30 & ~Bit31)   
    ;PC.7通用推挽输出模式 
    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] 
            
    ;NVIC                     
    ;ldr   r0, =SETENA0 
    ;mov   r1, 0x00800000 
    ;str   r1, [r0] 
    ;ldr   r0, =SETENA1 
    ;mov   r1, #0x00000100 
    ;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] 
    bl     LedFlas 


    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 


                 
;子程序 led闪烁 
LedFlas      
    push   {r0 - r3} 
    ldr    r0, =Flag1 
    ldr    r1, [r0] 
    tst    r1, #Bit0 
    ;bit0 闪烁标志位 
    beq    ONLED        ;0 打开led灯 
    ;1 关闭led灯 
    ldr    r0, =b_flas 
    mov    r1, #0 
    str    r1, [r0] 
    ;闪烁标志位置为0,下一状态为打开灯 
    ;PC.7输出0 
    ldr    r0, =GPIOC_BRR 
    ldr    r1, [r0] 
    orr    r1, #Bit7 
    str    r1, [r0] 
    b      LedEx 
ONLED       
    ;0 打开led灯 
    ldr    r0, =b_flas 
    mov    r1, #1 
    str    r1, [r0] 
    ;闪烁标志位置为1,下一状态为关闭灯 
    ;PC.7输出1 
    ldr    r0, =GPIOC_BSRR 
    ldr    r1, [r0] 
    orr    r1, #Bit7 
    str    r1, [r0] 
LedEx        
    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

3.生成.hex文件

输入代码编译后就会在工程目录下生成一个.hex文件
在这里插入图片描述

四、烧录程序

打开mcuisp软件,然后选择生成的.hex文件进行烧录,记得进行设置,如图所示
在这里插入图片描述

五、运行调试

烧录完成后复位,打开串口调试助手,找到生成的.hex文件,然后将参数设置好,因为都是默认的,所以这里不需要设置,然后点击打开串口就完成了
在这里插入图片描述

六、波形观测

同样的,我们可以通过Keil5进行仿真,用逻辑分析仪观测波形,不需要接入硬件。具体打开方法可以参考我的上一篇博客链接: 使用STM32CubeMX生成代码点亮流水灯——基于HAL库.
点击Setup Logic Analyzer添加引脚,在也可以选择View里面选择Symbol window然后在Special Funtion Register里面选择USART1_SR引脚,然后拖到逻辑分析仪窗口,点击Setup Logic Analyzer设置一下,Type设置为Bit
在这里插入图片描述
最后点击run运行即可
在这里插入图片描述

七、总结

通过此次实验,了解到了串口通讯的方式之一——USB转串口进行通讯。具体的操作其实很简单,但是需要了解背后的原理,才能进一步掌握串口通讯的方式。并且通过Keil5的逻辑分析仪的仿真波形也是非常有用的观测实验结果的方式,并且能够迅速的查找错误,如果程序出错,根据波形能够直观的了解程序运行错误的步骤。
参考资料: 基于 MDK 创建 STM32 汇编程序:串口输出 Hello world.

  • 3
    点赞
  • 34
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: STM32 USB串口源码指的是基于STM32芯片实现USB接口和多串口功能的源代码。 对于STM32系列芯片而言,它们内置了USB模块和多个串口模块。要实现USB串口功能,我们可以使用STM32Cube软件包来为该芯片生成相应的源码。 首先,我们需要使用STM32CubeMX软件来配置USB接口和多串口功能。在软件中选择目标芯片型号,然后配置相应的时钟和引脚。接下来,我们在"Configuration"选项卡中启用USB Device模块,并选择需要的USB功能,例如CDC(通用串行数据通信)模式。此外,我们还需要启用相应数量的串口模块,并配置它们的参数,如波特率、数据位、停止位和校验位等。 完成配置后,我们导出代码并打开生成的工程。在工程中,我们可以找到对应USB串口功能的源代码文件。USB部分的源码主要包括USB设备驱动和相关的USB类库函数。串口部分的源码主要包括串口初始化函数、发送和接收函数等。 我们可以在USB设备驱动中找到USB的初始化代码,并根据需要配置和定制USB功能。例如,在CDC模式下,我们可以配置串口的数量和每个串口的设置。在串口初始化函数中,我们可以设置波特率、数据位、停止位、校验位等参数,以及处理接收和发送缓冲区的相关代码。 通过对源代码的深入研究和修改,我们可以根据需求来实现特定的USB串口功能。当然,在开发过程中还需要注意USB串口的中断处理和状态机的设计,以确保数据的正常传输和处理。 综上所述,STM32 USB串口源码是一种基于STM32芯片实现USB接口和多串口功能的源代码。通过配置和定制相关的USB设备驱动和串口模块的源代码,我们可以实现特定的USB串口功能。 ### 回答2: STM32 USB串口源码可以通过STM32Cube软件包中的USB虚拟串口示例来实现。该软件包提供了STM32系列微控制器的USB功能驱动和示例代码。 首先,我们需要在STM32CubeMX中配置USB虚拟串口功能。打开STM32CubeMX,选择对应的微控制器型号,然后在"Middleware"选项卡中找到"USB_DEVICE",勾选"Class for Virtual COM Port"。接下来,在"Configuration"选项卡中设置串口的参数,如波特率、数据位、停止位等。 然后,点击"Project"菜单中的"Generate Code"按钮,生成相应的工程代码。得到的工程代码中将包含USB设备驱动程序和示例程序。 接下来,我们使用开发环境(如Keil或IAR)打开生成的工程代码。在示例程序中,可以找到有关USB虚拟串口的源码。源码中将包含初始化USB设备的代码、处理接收和发送数据的代码以及与串口相关的配置代码。 在源码中,我们可以根据自己的需求修改配置,例如增加新的虚拟串口、设置串口的参数等。具体的修改方式可以参考源码中的注释和相关文档。 最后,使用编译工具编译源码,并将生成的可执行文件烧录到STM32微控制器中。在微控制器运行时,它将作为一个USB设备接入到计算机中,并将虚拟串口与计算机上的串口进行通信。 总结起来,实现STM32 USB串口的源码开发可以通过使用STM32Cube软件包中的USB虚拟串口示例,根据自己的需求进行配置和修改,最后生成可执行文件并烧录到STM32微控制器中实现。 ### 回答3: STM32 USB串口是一种基于STM32微控制器的串口通信方式,通过USB接口连接电脑与STM32微控制器,实现电脑与STM32之间的串口通信。 在使用STM32 USB串口功能时,需要借助STM32串口库函数,并对其进行相应的配置和初始化。首先,在STM32的外设时钟配置中,使能对应的USB外设时钟,并进行相应的复位和初始化设置。然后,通过设置GPIO引脚的工作模式和输出模式,将USB通信所需的引脚连接到STM32微控制器的对应引脚上。 之后,在主程序中,使用STM32串口库函数对串口进行配置,例如设置波特率、数据位、停止位和校验位等。同时,编相应的接收和发送数据的函数来实现数据的接收和发送功能。 对于USB串口功能的源码实现,则需要借助STM32提供的USB库函数进行相关的配置和初始化。具体来说,可以通过USB CDC (Communications Device Class)协议实现USB串口功能。在编源码时,首先需要配置USB外设并进行初始化设置,然后通过USB CDC协议提供的库函数,实现数据的接收和发送。 在数据的接收端,可以使用中断方式或者DMA方式实现数据的接收。中断方式下,当有数据到达时,触发相应的中断,并在中断服务函数中读取数据。而DMA方式则是通过设置DMA传输通道,将接收到的数据直接存储到指定的内存区域中。 在数据的发送端,可以通过调用库函数实现发送数据的功能。可以发送单个字符或者一段数据,发送时可以选择使用中断方式或者轮询方式,在发送完成后进行相应的处理。 需要注意的是,STM32 USB串口源码的实现会涉及到一些底层的配置和操作,具体实现过程和代码可以参考STM32官方提供的开发文档和示例代码。同时,不同的STM32型号和不同的开发环境可能会有所不同,具体的实现方式还需根据具体情况进行相应的调整和修改。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值