首页 程序笔记 使用STM32 USB转串口功能(自发自收回环测试、USB转TTL串口)

使用STM32 USB转串口功能(自发自收回环测试、USB转TTL串口)

前言

STM32单片机一般都有USB接口,用于实现各种类型的USB通信,典型的USB转串口功能,接下来就介绍一下,USB转串口的使用,先以自发自收回环测试为例,演示USB转串口的使用,然后以此为基础,修改为USB转TTL串口,其中主要代码使用STM32CubeMX生成。

开发环境

  • STM32F103RB
  • STM32CubeMX
  • STM32CUBEIDE

STM32CubeMx配置

新建工程

在STM32CubeMx中新建基于STM32F103RB的Project。

配置时钟

时钟的配置如下图所示:

开启USB

  1. 首先,使能USB Device,如下图:

  2. 开启USB设备:虚拟串口,如下图:

  3. 生成工程

修改代码实现回环收发数据测试

在​​usbd_cdc_if.c​​文件中新定义一个结构体:

USBD_CDC_LineCodingTypeDef USBD_CDC_LineCoding =
{
  115200,      // 默认波特率
  0X00,        // 1位停止位
  0X00,        // 无奇偶校
  0X08,        // 无流控,8bit数据位
};

找到​​CDC_Control_FS​​​函数,找到​​CDC_SET_LINE_CODING​​​和​​CDC_GET_LINE_CODING​​​分支,添加以下代码:
(CDC_SET_LINE_CODING:你在用串口助手选择波特率时候,STM32就会调用这个分支进行修改USB波特率)
(CDC_GET_LINE_CODING:获取STM32的USB波特率)

case CDC_SET_LINE_CODING:
        USBD_CDC_LineCoding.bitrate = (pbuf[3]  24) | (pbuf[2]  16) | (pbuf[1]  8) | pbuf[0];
        USBD_CDC_LineCoding.format = pbuf[4];
        USBD_CDC_LineCoding.paritytype = pbuf[5];
        USBD_CDC_LineCoding.datatype = pbuf[6];
    break;

    case CDC_GET_LINE_CODING:
        pbuf[0] = (uint8_t)(USBD_CDC_LineCoding.bitrate);
        pbuf[1] = (uint8_t)(USBD_CDC_LineCoding.bitrate  8);
        pbuf[2] = (uint8_t)(USBD_CDC_LineCoding.bitrate  16);
        pbuf[3] = (uint8_t)(USBD_CDC_LineCoding.bitrate  24);
        pbuf[4] = USBD_CDC_LineCoding.format;
        pbuf[5] = USBD_CDC_LineCoding.paritytype;
        pbuf[6] = USBD_CDC_LineCoding.datatype;
    break;

如下图所示:

找到​​CDC_Receive_FS​​​函数,这个函数如果USB虚拟串口数据收到就会被调用,我们在这个函数中将收到的数据在发回去,只需要添加​​CDC_Transmit_FS(Buf, *Len);​​这一句即可,如下图:

然后编译工程并下载,接上USB之后,设备管理器COM出现一个新的端口:

我们使用串口调试助手给它发数据:

实现USB转串口功能

发数据流程:串口调试助手发送数据-STM32的USB数据接收-STM32转发到串口3
收数据流程:STM32的串口3收到数据-转发到USB-STM32的USB发送到串口调试助手

第一步先在这里加入串口3的初始化操作:

贴一下串口3的初始化代码,这里我用到了队列,因为实际测试发现串口3接收数据量比较大的话,那么转发到USB虚拟串口的时候会丢数据,所以这里采用了缓存队列,当串口接收到的数据到达一定数据量之后才做一次转发到USB的操作,并且开启了空闲中断,作用是转发最后一包数据:
关于队列的使用可以查看我的另一篇博客:​​C/C++语言实现的一个缓存队列​​

/*
 * myusart.c
 *
 *  Created on: Mar 22, 2021
 *      Author: hello
 */
#include myusart.h
#include myqueue.h

// 队列大小,定义为USB_CDC的发送包大小
#define QUEUE_SIZE APP_TX_DATA_SIZE

// 队列数据空间。请使用宏QALIGN4,目的是为了根据队列大小计算实际需要的队列存储空间大小并对齐4字节
uint8_t QueueBuffer[QALIGN4(QUEUE_SIZE)];

// 队列句柄
Queue Uart3QueueHandle = {0};
#define UART3_QUEUE_Handle (Uart3QueueHandle)

// 串口转发到USB的缓冲,定义为USB包的一半大小
static uint8_t buffer[APP_RX_DATA_SIZE  1];

void UART3_Init(const USBD_CDC_LineCodingTypeDef *USBD_CDC_LineCoding)
{
  __HAL_UART_DISABLE_IT(huart3, UART_IT_RXNE);

  __HAL_UART_DISABLE_IT(huart3, UART_IT_IDLE);

  HAL_UART_DeInit(huart3);

  huart3.Instance = USART3;
  huart3.Init.BaudRate = USBD_CDC_LineCoding-bitrate;
  huart3.Init.HwFlowCtl = UART_HWCONTROL_NONE;
  huart3.Init.Mode = UART_MODE_TX_RX;
  huart3.Init.OverSampling = UART_OVERSAMPLING_16;

  switch (USBD_CDC_LineCoding-paritytype)
  {
  case 0:
    huart3.Init.Parity = UART_PARITY_NONE;
    break;
  case 1:
    huart3.Init.Parity = UART_PARITY_ODD;
    break;
  case 2:
    huart3.Init.Parity = UART_PARITY_EVEN;
    break;
  default:
    huart3.Init.Parity = UART_PARITY_NONE;
    break;
  }

  switch (USBD_CDC_LineCoding-datatype)
  {
  case 0x07:
    huart3.Init.WordLength = UART_WORDLENGTH_8B;
    break;
  case 0x08:
    if (huart3.Init.Parity == UART_PARITY_NONE)
    {
      huart3.Init.WordLength = UART_WORDLENGTH_8B;
    }
    else
    {
      huart3.Init.WordLength = UART_WORDLENGTH_9B;
    }
    break;
  default:
    huart3.Init.WordLength = UART_WORDLENGTH_8B;
    break;
  }

  switch (USBD_CDC_LineCoding-format)
  {
  case 0:
    huart3.Init.StopBits = UART_STOPBITS_1;
    break;
  case 2:
    huart3.Init.StopBits = UART_STOPBITS_2;
    break;
  default:
    huart3.Init.StopBits = UART_STOPBITS_1;
    break;
  }

  HAL_UART_Init(huart3);

  Queue_Init(UART3_QUEUE_Handle, QUEUE_SIZE, QueueBuffer);

  __HAL_UART_ENABLE_IT(huart3, UART_IT_RXNE);

  __HAL_UART_ENABLE_IT(huart3, UART_IT_IDLE);
}

void USART3_IRQHandler(void)
{
  static uint32_t nsent = 0;
  static uint8_t dat = 0;

  if (__HAL_UART_GET_FLAG(huart3, UART_FLAG_RXNE) != RESET)
  {
    __HAL_UART_CLEAR_FLAG(huart3, UART_FLAG_RXNE);
    dat = huart3.Instance-DR  0XFF;
    Queue_PutByte(UART3_QUEUE_Handle, dat);                     // 入队一个字节的数据
    if (Queue_GetUsed(UART3_QUEUE_Handle) = sizeof(buffer))
    {
      Queue_Read(UART3_QUEUE_Handle, buffer, sizeof(buffer));
      CDC_Transmit_FS(buffer, sizeof(buffer));                // 转发到USB
    }
  }

  if (__HAL_UART_GET_FLAG(huart3, UART_FLAG_IDLE) != RESET)
  {
    nsent = Queue_GetUsed(UART3_QUEUE_Handle);
    if (nsent != 0)
    {
      Queue_Read(UART3_QUEUE_Handle, buffer, nsent);
      CDC_Transmit_FS(buffer, nsent);               // 转发到USB
    }
    __HAL_UART_CLEAR_IDLEFLAG(huart3);
  }
}

void UART3_SendData(const void *buf, uint32_t len)
{
  const uint8_t *p = (const uint8_t*) buf;
  while (len--)
  {
    while (__HAL_UART_GET_FLAG(huart3, UART_FLAG_TXE) != SET);
    huart3.Instance-DR = (uint8_t) (*p++  0XFF);
    while (__HAL_UART_GET_FLAG(huart3, UART_FLAG_TC) != SET);
  }
}

//#ifdef __GNUC__
//#define PUTCHAR_PROTOTYPE int __io_putchar(int ch)
//#else
//#define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
//#endif
//PUTCHAR_PROTOTYPE
//{
//  while (__HAL_UART_GET_FLAG(huart3, UART_FLAG_TXE) != SET);
//  huart3.Instance-DR = (uint8_t) (ch  0XFF);
//  while (__HAL_UART_GET_FLAG(huart3, UART_FLAG_TC) != SET);
//  return ch;
//}

然后在添加USB转发到串口的操作:

这样就实现了一个USB转TTL串口的功能!

原文链接(转载有改动,感谢原作者):https://blog.51cto.com/u_15950551/6032374

2

站心网

前言 STM32单片机一般都有USB接口,用于实现各种类型的USB通信,典型的USB转串口功能,接下来就介绍一下,U..

为您推荐

如何设计功能测试用例

最近招聘面试测试工程师,发现很多测试工程师都是半路出家,对一些基本的测试概念,思路与技术没有认知。在设计测试用例的时候只是基于直觉与经验去测试。这就导致测试用例的设计既不全面,也不科学。一般功能测试指..

DISC测试,让你轻松读懂领导

我们常说“性格决定命运”,企业家的性格决定着企业的命运,而身处企业中的我们,如何和不同性格的老板相处,则直接关系到我们的职场命运。企业家作为掌舵者,他的行为自然决定了整条大船如何行进。行为来自于习惯,..

使用 html2canvas 实现截图功能

html2canvas 是一个开源的 JavaScript 库,用于将网页上的 HTML 元素渲染成图像。它通过遍历页面的 DOM 树和计算样式,然后将其绘制到 <canvas> 元素上,最终生成图片。该库不依赖服务器端,而是通过浏览器端的 Java..

使用SuperWebSocket实现Web消息推送

在大部分Web系统中,我们可能遇到需要向客户端推送消息的需求。SuperWebSocket第三方库能让我们轻松的完成任务。SuperWebSocket第三方库可以从网上下载,不过通过Visual Studio Nuget安装更快。引用SuperWebSocket相..

.NET C# 使用Hook钩子实现全局监听键盘和鼠标

C# 是一种面向对象的编程语言,具有丰富的类库和工具支持,适用于各种类型的应用程序开发。Windows 提供了一种称为"钩子"(Hook)的机制,允许拦截并处理系统级别的事件,如键盘按键和鼠标移动。通过结合 C# 和 Hook..

C#使用 Attribute 实现 AOP 功能

在 C# 中,通过自定义 Attribute 并结合一些技术(如动态代理、反射等)可以实现 AOP(面向切面编程)。AOP 通常用于日志记录、性能监控、权限验证等横切关注点。以下是一个使用 C# Attribute 实现 AOP 功能的示例。..

ABP.Net Core使用教程(一)启动模版项目

只需要简单的3步:1,到官网下载模版项目 https://aspnetboilerplate.com/Templates2,用VS2017打开,将Web.Host设置为启动项3,在程序包管理器控制台(Nuget控制台)里设定默认项目为EntityFrameworkCore,执行命令..

C#中的线程安全的集合ConcurrentQueue使用示例

在多线程编程中,如何安全地在不同线程之间共享数据是一个非常重要的问题。C# 为我们提供了一些专门设计的线程安全集合,其中之一就是 ConcurrentQueue<T>。它是一种先进先出(FIFO)的数据结构,专门为多线程环境设..

CSS砌体布局示例和使用场景

CSS砌体布局(Masonry Layout)CSS砌体布局是一种网页布局技术,它的灵感来源于砖石墙的排列方式,类似于“拼图”或“拼砖”的效果。在砌体布局中,元素的排列并不完全遵循传统的网格布局规则,..

使用CSS columns-visibility实现砌体布局

CSS的 columns 属性(如 columns、column-count 和 column-width)通常用于多列文本布局,而不是直接用于砌体布局。然而,结合 columns 和 visibility 属性,可以在某些情况下实现类似砌体布局的效果,虽然它并不完..

使用System.Linq.Dynamic.Core扩展库动态构建 LINQ 查询

System.Linq.Dynamic.Core 是一个扩展库,用于在运行时动态构建 LINQ 查询,支持字符串形式的表达式解析和动态查询操作。它是 .NET 的一个强大工具,适合处理需要灵活定义查询逻辑的场景,例如动态过滤、排序、投影..

小米开源智能家居平台 ha_xiaomi_home 使用示例

小米近期在 GitHub 上开源了名为“ha_xiaomi_home”的项目,即 Home Assistant 米家集成组件。该组件由小米官方支持,旨在让用户在 Home Assistant 中集成和控制小米 IoT 智能设备。主要特点:官方支持:..

C#13新特性 使用System.Threading.Lock简化线程同步

C# 13 引入了新的线程同步类型 System.Threading.Lock,它通过作用域管理的方式简化了锁的使用,使代码更加清晰可靠。本文将全面介绍 System.Threading.Lock 的功能、适用场景,并提供完整的运行示例程序。1. 什么是..

微软官方Microsoft.Extensions.AI库使用示例

Microsoft.Extensions.AI 库介绍Microsoft.Extensions.AI 是一个扩展库,用于在 .NET 应用程序中轻松集成人工智能(AI)服务,例如 OpenAI、Azure OpenAI 和其他支持文本生成或语言模型的 API。通过与 Microsoft.Ext..

.Net Core中Dapper的使用详解

1.安装Dapper这里直接使用Nuget安装。安装版本是1.50.5安装完成之后,发现Nuget下已经有了Dapper。2.创建DapperHelper接下来创建一个DapperHelper帮助类,来进行读取数据库连接字符串,打开数据库等操作。public cla..

最新CentOS7安装搭建shadowsocks服务端+客户端使用图文教程

使用的CentOS版本是7.9,其他版本也可以。超级推荐的是搭建shadowsocks服务端,安装配置都很简单,几分钟就搞定,客户端支持PC移动端,下面是安装shadowsocks的过程,只要复制粘贴命令就行了,文件夹路径都不需要改..

ASP.NET 使用Entity Framework (EF) 创建迁移修改SQLite数据库表结构

在 ASP.NET 中,使用 Entity Framework (EF) 创建并连接 SQLite 数据库是一种轻量级、高效的数据库管理方式。以下是详细步骤:安装必要的 NuGet 包安装EntityFrameworkCore.Sqlite包:Install-Package Microsoft.Ent..

使用shields.io来实时显示GitHub项目star、watch和fork的数量

如何获取GitHub repo实时的star,watch和fork数量呢?这里推荐一个Shields.io工具,可以实时生成GitHub徽章,同时显示star数。显示效果如下:什么是 Shields.io?Shields.io 是一个开源项目,用于生成各种类型的徽章..

.NET 开源 ORM FreeSql 使用教程

什么是 FreeSql?FreeSql 是一个高性能、灵活且易用的 .NET 开源 ORM(对象关系映射工具),提供数据库操作的强大功能,包括实体类映射、链式查询、表达式树支持、数据库迁移等。它可以帮助开发者快速、高效地操作数..

SQL Server EF使用Sequence全局自增ID

在使用 Entity Framework (EF) 时,如果需要在 SQL Server 中实现一个 全局自增 ID,可以通过以下方法来实现。全局自增 ID 的需求通常是为了在多表之间实现唯一性递增 ID。实现方式 1:使用 SQL Server 的 SequenceS..

发表回复

返回顶部