首页 > 其他分享 >【RTT-Studio】详细使用教程十三:UART的DMA 接收及轮询发送

【RTT-Studio】详细使用教程十三:UART的DMA 接收及轮询发送

时间:2024-08-21 20:24:41浏览次数:17  
标签:rt DMA UART 轮询 rx RT msg serial define

文章目录

一、简介

  串口是指数据一位一位地顺序传送,其特点是通讯线路简单,只要一对传输线就可以实现双向通信(可以直接利用电话线作为传输线),从而大大降低了成本,特别适用于远距离通信,但传送速度较慢。一条信息的各位数据被逐位按顺序传送的通讯方式称为串行通讯。串行通讯的特点是:数据位的传送,按位顺序进行,最少只需一根传输线即可完成;成本低但传送速度慢。串行通讯的距离可以从几米到几千米;根据信息的传送方向,串行通讯可以进一步分为单工、半双工和全双工三种。

  运行序列图如下图所示:
在这里插入图片描述
  当串口接收到一批数据后会调用接收回调函数,接收回调函数会把此时缓冲区的数据大小通过消息队列发送给等待的数据处理线程。线程获取到消息后被激活,并读取数据。一般情况下 DMA 接收模式会结合 DMA 接收完成中断和串口空闲中断完成数据接收。


二、RTT配置

(1)在board.h文件中进行配置,具体操作在board.h文件中有介绍:

/** After configuring corresponding UART or UART DMA, you can use it.
 *
 * STEP 1, define macro define related to the serial port opening based on the serial port number
 *                 such as     #define BSP_USING_UART1
 *
 * STEP 2, according to the corresponding pin of serial port, define the related serial port information macro
 *                 such as     #define BSP_UART1_TX_PIN       "PA9"
 *                             #define BSP_UART1_RX_PIN       "PA10"
 *
 * STEP 3, if you want using SERIAL DMA, you must open it in the RT-Thread Settings.
 *                 RT-Thread Setting -> Components -> Device Drivers -> Serial Device Drivers -> Enable Serial DMA Mode
 *
 * STEP 4, according to serial port number to define serial port tx/rx DMA function in the board.h file
 *                 such as     #define BSP_UART1_RX_USING_DMA
 *
 */

#define BSP_USING_UART1
#define BSP_UART1_TX_PIN       "PA9"
#define BSP_UART1_RX_PIN       "PA10"

#define BSP_USING_UART3
#define BSP_UART3_TX_PIN       "PD8"
#define BSP_UART3_RX_PIN       "PD9"
#define BSP_UART3_RX_USING_DMA

(2)打开可视化界面上驱动使能,使能串口DMA模式,并且可以根据自己需要设置缓冲区的大小:
在这里插入图片描述

(3)打开DMA所需要的HAL库文件,只需要将相应的宏定义打开即可。
在这里插入图片描述


三、使用信号量接收

1.uart3.h文件

#ifndef APPLICATIONS_UART3_H_
#define APPLICATIONS_UART3_H_

#include <rtthread.h>
#include <drv_common.h>

#define SAMPLE_UART_NAME        "uart3"

rt_size_t rx_len;

/* 串口设备句柄 */
static rt_device_t serial;

/* 信号量控制变量 */
rt_sem_t  rx_sem;

#endif /* APPLICATIONS_UART3_H_ */

2.uart3.c文件

#include "uart3.h"

/* 接收数据回调函数 */
static rt_err_t uart_input(rt_device_t dev, rt_size_t size)
{
    rx_len = size;
    rt_sem_release(rx_sem);

    return RT_EOK;
}

static void serial_thread_entry(void *parameter)
{
    rt_size_t len = 0;
    char buffer[256] = {0};

    while (1)
    {
        rt_memset(&buffer, 0, sizeof(buffer));
        /* 从信号量中读取消息 */
        rt_sem_take(rx_sem, RT_WAITING_FOREVER);
        len = rt_device_read(serial, 0, buffer, 256);
        buffer[len] = '\0';
        rt_device_write(serial, 0, buffer, len);
        rt_kprintf("%s\n", buffer);
    }
}

void Create_Uart3_Test_Entry(void)
{
    char str[] = "hello, rt-thread!";
    /* 查找串口设备 */
    serial = rt_device_find(SAMPLE_UART_NAME);
    if (!serial)
    {
        rt_kprintf("find %s failed!\n", SAMPLE_UART_NAME);
    }

    /* 以 DMA 接收及轮询发送方式打开串口设备 */
    rt_device_open(serial, RT_DEVICE_FLAG_DMA_RX);

    /* 初始化配置参数 */
    struct serial_configure uart3_config = RT_SERIAL_CONFIG_DEFAULT;
    uart3_config.baud_rate = BAUD_RATE_19200;
    rt_device_control(serial, RT_DEVICE_CTRL_CONFIG, (void *) &uart3_config);

    /* 创建信号量 */
    rx_sem = rt_sem_create("rx_sem", 0, RT_IPC_FLAG_PRIO);

    /* 设置接收回调函数 */
    rt_device_set_rx_indicate(serial, uart_input);

    /* 发送字符串 */
    rt_device_write(serial, 0, str, (sizeof(str) - 1));

    /* 创建 serial 线程 */
    rt_thread_t thread = rt_thread_create("serial", serial_thread_entry, RT_NULL, 1024, 25, 10);
    /* 创建成功则启动线程 */
    if (thread != RT_NULL)
    {
        rt_thread_startup(thread);
    }
}
MSH_CMD_EXPORT_ALIAS(Create_Uart3_Test_Entry, test, Create Uart3 Test Entry);

四、使用消息队列接收

1.uart3.h文件

#ifndef APPLICATIONS_UART3_H_
#define APPLICATIONS_UART3_H_

#include <rtthread.h>
#include <drv_common.h>

#define SAMPLE_UART_NAME        "uart3"

/* 串口接收消息结构 */
struct rx_msg
{
    rt_device_t dev;
    rt_size_t   size;
};

/* 串口设备句柄 */
static rt_device_t serial;

/* 消息队列控制块 */
static struct rt_messagequeue rx_mq;

#endif /* APPLICATIONS_UART3_H_ */

2.uart3.c文件

#include "uart3.h"

/* 接收数据回调函数 */
static rt_err_t uart_input(rt_device_t dev, rt_size_t size)
{
    struct rx_msg msg;
    rt_err_t result;

    msg.dev = dev;
    msg.size = size;

    result = rt_mq_send(&rx_mq, &msg, sizeof(msg));
    if ( result == -RT_EFULL)
    {
        /* 消息队列满 */
        rt_kprintf("message queue full!\n");
    }
    return result;
}

static void serial_thread_entry(void *parameter)
{
    struct rx_msg msg;
    rt_err_t result;
    rt_uint32_t rx_length;
    static char rx_buffer[RT_SERIAL_RB_BUFSZ + 1];

    while (1)
    {
        rt_memset(&msg, 0, sizeof(msg));
        /* 从消息队列中读取消息 */
        result = rt_mq_recv(&rx_mq, &msg, sizeof(msg), RT_WAITING_FOREVER);
        if (result == RT_EOK)
        {
            /* 从串口读取数据 */
            rx_length = rt_device_read(msg.dev, 0, rx_buffer, msg.size);
            rx_buffer[rx_length] = '\0';
            /* 通过串口设备 serial 输出读取到的消息  */
            rt_device_write(serial, 0, rx_buffer, rx_length);
            /* 打印数据 */
            rt_kprintf("%s\n", rx_buffer);
        }
    }
}

void Create_Uart3_Test_Entry(void)
{
    static char msg_pool[256];
    char str[] = "hello RT-Thread!\r\n";

    /* 查找串口设备 */
    serial = rt_device_find(SAMPLE_UART_NAME);
    if (!serial)
    {
        rt_kprintf("find %s failed!\n", SAMPLE_UART_NAME);
    }

    /* 初始化消息队列 */
    rt_mq_init(&rx_mq, "rx_mq",
               msg_pool,                /* 存放消息的缓冲区  */
               sizeof(struct rx_msg),   /* 一条消息的最大长度  */
               sizeof(msg_pool),        /* 存放消息的缓冲区大小  */
               RT_IPC_FLAG_FIFO);       /* 如果有多个线程等待,按照先来先得到的方法分配消息  */

    /* 以 DMA 接收及轮询发送方式打开串口设备 */
    rt_device_open(serial, RT_DEVICE_FLAG_DMA_RX);

    /* 初始化配置参数 */
    struct serial_configure uart3_config = RT_SERIAL_CONFIG_DEFAULT;
    uart3_config.baud_rate = BAUD_RATE_19200;
    rt_device_control(serial, RT_DEVICE_CTRL_CONFIG, (void *) &uart3_config);

    /* 设置接收回调函数 */
    rt_device_set_rx_indicate(serial, uart_input);

    /* 发送字符串 */
    rt_device_write(serial, 0, str, (sizeof(str) - 1));

    /* 创建 serial 线程 */
    rt_thread_t thread = rt_thread_create("serial", serial_thread_entry, RT_NULL, 1024, 25, 10);
    /* 创建成功则启动线程 */
    if (thread != RT_NULL)
    {
        rt_thread_startup(thread);
    }
}
MSH_CMD_EXPORT_ALIAS(Create_Uart3_Test_Entry, test, Create Uart3 Test Entry);

五、测试验证

  通过将上面的两个程序烧录到控制板中,通过RTT特有的控制台进行指令操作,通过大宋质量test来执行串口函数,从而实现对串口3的参数配置以及DMA配置,通过信号量或者消息队列的方式来接收数据,从而将接收到的数据重新发送到串口3,实际的实验效果如下所示:
在这里插入图片描述


标签:rt,DMA,UART,轮询,rx,RT,msg,serial,define
From: https://blog.csdn.net/Hei_se_meng_yan/article/details/141401850

相关文章

  • 两种形式的dma 实现memory copy代码
    在飞思卡尔的时候,需要用SDMA实现内存到内存memorycopy的功能,需要做两部分的工作:1:在DMAcontroller中加入M2M的支持。2:写一个驱动来调用DMAcontroller的M2M功能。上面的2实际上对于不同的SoC来讲,思路是一样的,有通用性,在这里总结下。当时在实现的时候,用了两种方法:1:cyclic,用dma_a......
  • 配置stm32cubemx采集stm32H743IIT6,通过DMA实现多通道和多模块ADC的采集,亲测有效!
     之前写到stm32cubemx通过阻塞实现单通道和多通道的ADC的采集。本文分享通过DMA实现单模块多通道和多模块多通道的ADC采集。stm32cubemx的版本6.10.0。一、DMA采集多通道ADC数据阻塞采集是每次采集adc数据,cpu死等,直到采集完或者在设定时间超时没能采集,返回到cpu。DMA采集......
  • CADMATIC许可证续费
    在数字化时代,CADMATIC软件已成为工程设计和制造业领域的核心工具。为了保持竞争优势,企业需要确保CADMATIC软件的持续使用。而CADMATIC许可证续费,则是确保软件持续稳定运行的关键。本文将为您揭示CADMATIC许可证续费的必要性,以及如何顺利完成许可证续费,帮助您在竞争激烈的市场中保......
  • Spring Boot整合Quartz框架
    说明:Quartz是一个定时器框架,可以实现定时任务,本文介绍如何在SpringBoot项目中整合Quartz框架,Quartz介绍参看下面这篇文章:【Quartz】Quartz定时任务框架创建Demo首先,创建一个SpringBoot项目,pom文件如下:<?xmlversion="1.0"encoding="UTF-8"?><projectxmlns="http://......
  • CADMATIC许可证配置文件
    在数字化时代,CADMATIC软件已成为工程设计和制造业领域的核心工具。为了充分发挥CADMATIC软件的性能并确保合规性,合理的许可证配置文件设置至关重要。本文将为您揭示CADMATIC许可证配置文件的奥秘,帮助您优化软件性能并确保合规性,提升企业的竞争力。一、CADMATIC许可证配置文件的重......
  • 利用Quartz实现复杂的任务调度
    原文地址:java-利用Quartz实现复杂的任务调度-宋小黑-SegmentFault思否  第一章:引言大家好,我是小黑,任务调度,简而言之,就是按照预定计划自动执行任务的过程。不管是数据库备份、报表生成还是发送定时邮件,它们都需要一个可靠的任务调度系统来保证按时完成。那么,为什么......
  • UART\SPI\I2C的区别与联系
    UART全双工(两根线tx,rx),无时钟线,只能两个设备SPI全双工(两根线tx、rx+时钟线+片选),一主多从,扩展了接入的设备,同步传输,速度更快I2C半双工(一根数据线+时钟线),多主一从或者多主多从UART(UniversalAsynchronousReceiver/Transmitter)全双工:意味着数据可以同时在两个方向上......
  • UART串行通信协议
    UART串行通信协议UART(UniversalAsynchronousReceiver/Transmitter,通用异步接收器/发送器)是一种串行通信协议,用于异步通信,即发送和接收设备不需要共享一个时钟信号来同步数据传输。以下是UART的详细介绍:UART的基本特性异步通信:UART不使用时钟信号来同步数据,而是依赖于数据包......
  • UART 通信协议详解
    目录一、概述二、UART详解1、数据通信的基本概念1.1数据通信方式1.2数据传输方向1.3数据同步方式1.4通信速率2、UART协议2.1串口连接2.2串口协议帧一、概述UART(UniversalAsynchronousReceiver/Transmitter,通用异步收发器)是一种常用的串行通信协议,......
  • 解锁CADMATIC许可证的奥秘
    解锁CADMATIC许可证的奥秘:选择最适合您的许可证类型在数字化时代,CADMATIC软件已成为工程设计和制造业领域的核心工具。为了充分发挥CADMATIC软件的功能和优势,选择合适的许可证类型至关重要。本文将为您揭示CADMATIC许可证的奥秘,帮助您选择最适合您的许可证类型,提升企业的竞争力。......