CC2640R2F学习笔记(6)——UART串口使用_cc2640r2串口使用-程序员宅基地

技术标签: 串口  UART  BLE  CC2640R2F  

一、硬件连接

串口 引脚
TXD DIO_2
RXD DIO_3

二、移植文件

链接:https://pan.baidu.com/s/1fE8XWgvdzYiF1bdUzTDxUg 提取码:r5s2
uart.cuart.h 两个文件拖拽至CCS工程的Application文件夹下

添加文件过程中,选项选择如下

2.1 uart.c

/*********************************************************************
 * INCLUDES
 */
#include <xdc/runtime/Assert.h>
#include <xdc/runtime/Types.h>
#include <xdc/runtime/Log.h>
 
#include <ti/sysbios/BIOS.h>
#include <ti/drivers/power/PowerCC26XX.h>
#include <ti/drivers/uart/UARTCC26XX.h>
#include "board.h"
#include "util.h"
 
#include "board_uart.h"
 
/******************************************************************************
 * DEFINITIONS
 */
#define UART_RX                 IOID_2  // 接收引脚
#define UART_TX                 IOID_3  // 发送引脚
 
/*********************************************************************
 * LOCAL VARIABLES
 */
// 串口号
typedef enum{
    
    UART0 = 0,
 
    UARTCOUNT
} UARTName;
 
// 流控制所需的配置
static const UARTCC26XX_HWAttrsV2 s_UART_CC26XXHWAttrs[UARTCOUNT] = {
    
    {
    
        .baseAddr       = UART0_BASE,
        .powerMngrId    = PowerCC26XX_PERIPH_UART0,
        .intNum         = INT_UART0_COMB,
        .intPriority    = ~0,
        .swiPriority    = 0,
        .txPin          = UART_TX,
        .rxPin          = UART_RX,
        .ctsPin         = PIN_UNASSIGNED,
        .rtsPin         = PIN_UNASSIGNED
    }
};
 
// 串口目标
static UARTCC26XX_Object s_UART_CC26XXObject[UARTCOUNT];
static const UART_Config s_UART_Config[UARTCOUNT] = {
    
    {
    
        .fxnTablePtr = &UARTCC26XX_fxnTable,
        .object      = &s_UART_CC26XXObject[UART0],
        .hwAttrs     = &s_UART_CC26XXHWAttrs[UART0]
    },
};
 
// 串口句柄
static UART_Handle s_UART_Handle;
 
// 串口参数
static UART_Params s_UART_Params;
 
/*********************************************************************
 * GLOBAL VARIABLES
 */
uint8_t g_UART_RxBuf[UART_MAX_NUM_RX_BYTES] = {
    0};
uint8_t g_UART_TxBuf[UART_MAX_NUM_TX_BYTES] = {
    0};
uint32_t g_UART_WantedRxBytes;
uint8_t g_UART_Size;

// 串口定时器
Clock_Struct g_UART_Clock;

/*********************************************************************
 * PUBLIC FUNCTIONS
 */

/**
 @brief 串口驱动初始化
 @param none
 @return none
*/
void UART_Init(UART_Callback UART_ReadCallback)
{
    
    // 初始化为默认值
    UART_Params_init(&s_UART_Params);
 
    // 参数设置
    s_UART_Params.baudRate      = 115200;
    s_UART_Params.writeDataMode = UART_DATA_BINARY;
    s_UART_Params.readMode      = UART_MODE_CALLBACK;
    s_UART_Params.readDataMode  = UART_DATA_BINARY;
    s_UART_Params.readCallback  = UART_ReadCallback;
 
    // 打开串口
    s_UART_Handle = s_UART_Config[0].fxnTablePtr->openFxn((UART_Handle)&s_UART_Config[0], &s_UART_Params);
 
    // 启动串口局部返回功能,为了能在串口接收到数据时及时进回调函数。如果没有这条语句,将会在缓冲区满时才进入。
    s_UART_Config[0].fxnTablePtr->controlFxn(s_UART_Handle, UARTCC26XX_CMD_RETURN_PARTIAL_ENABLE, NULL);
 
    // 开始第一次的读取等待
    g_UART_WantedRxBytes = UART_MAX_NUM_RX_BYTES;
    UART_Receive(g_UART_RxBuf, g_UART_WantedRxBytes);
}

/**
 @brief 串口的写数据的函数
 @param pUART_TxBuf 写缓冲区
 @param UART_Len 数据长度
 @return none
*/
void UART_Send(uint8_t* pUART_TxBuf, uint8_t UART_Len)
{
    
    s_UART_Config[0].fxnTablePtr->writeFxn(s_UART_Handle, pUART_TxBuf, UART_Len);
}

/**
 @brief 串口的读数据的函数
 @param pUART_RxBuf 读缓冲区
 @param UART_Len 数据长度
 @return none
*/
void UART_Receive(uint8_t* pUART_RxBuf, uint8_t UART_Len)
{
    
    s_UART_Config[0].fxnTablePtr->readFxn(s_UART_Handle, pUART_RxBuf, UART_Len);
}

/**
 @brief 串口的读数据回调函数
 @param UART_Handle 串口句柄
 @param pUART_RxBuf 接收缓冲区
 @param UART_Size:接收数据大小
 @return none
*/
void UART_ReadCallback(UART_Handle UART_Handle, void *pUART_RxBuf, size_t UART_Size)
{
    
    // 保存数据
    g_UART_Size = UART_Size;
    memcpy(g_UART_RxBuf, pUART_RxBuf, g_UART_Size);

    // 启动定时器
    Util_startClock(&g_UART_Clock);
}

/*************************************END OF FILE*************************************/

2.2 uart.h

#ifndef _BOARD_UART_H_
#define _BOARD_UART_H_

/*********************************************************************
 * INCLUDES
 */
#include <ti/drivers/UART.h>
#include <ti/sysbios/knl/Clock.h>

/******************************************************************************
 * DEFINITIONS
 */
#define UART_MAX_NUM_RX_BYTES   60      // 最大接收字节
#define UART_MAX_NUM_TX_BYTES   60      // 最大发送字节
 
/*********************************************************************
 * EXTERNAL VARIABLES
 */
extern uint8_t g_UART_RxBuf[UART_MAX_NUM_RX_BYTES];
extern uint8_t g_UART_TxBuf[UART_MAX_NUM_TX_BYTES];
extern uint32_t g_UART_WantedRxBytes;
extern uint8_t g_UART_Size;

extern Clock_Struct g_UART_Clock;       // 串口定时器
 
/*********************************************************************
 * TYPEDEFS
 */
typedef void (*UART_Callback) (UART_Handle handle, void *pBuf, size_t count);

/*********************************************************************
 * API FUNCTIONS
 */
void UART_Init(UART_Callback UART_ReadCallback);
void UART_Send(uint8_t* npUART_TxBuf, uint8_t nUART_Len);
void UART_Receive(uint8_t* npUART_RxBuf, uint8_t nUART_Len);
void UART_ReadCallback(UART_Handle UART_Handle, void *pUART_RxBuf, size_t UART_Size);
 
#endif /* _BOARD_UART_H_ */

三、API调用

需包含头文件 uart.h

UART_Init

功能 初始化串口驱动
函数定义 void UART_Init(UART_Callback UART_ReadCallback)
参数 UART回调函数
返回

UART_Send

功能 串口的写数据的函数
函数定义 void UART_Send(uint8_t* pUART_TxBuf, uint8_t UART_Len)
参数1 pUART_TxBuf:写缓冲区
参数2 UART_Len:数据长度
返回

UART_Receive

功能 串口的读数据的函数
函数定义 void UART_Receive(uint8_t* pUART_RxBuf, uint8_t UART_Len)
参数1 pUART_RxBuf:读缓冲区
参数2 UART_Len:数据长度
返回

四、流程

① 串口收到数据
② 从底层发送数据到应用层的回调函数UART_ReadCallback中
③ 在回调函数UART_ReadCallback中保存数据
④ 启动串口处理事件的定时器
⑤ 在uart_performTask函数中做自定义串口数据处理
注:如果直接在串口回调函数中做处理,会导致“收发一段时间后无法发出数据”的现象。

五、使用例子

1)添加头文件(例multi_role.c中)

#include "uart.h"

2)添加初始化代码(multi_role.c的multi_role_init函数末尾中)
初始化时传入应用层的串口回调函数,以便串口接收到数据时能传到应用层中。

// 串口初始化
UART_Init(UART_ReadCallback); 
UART_Send("TEST\r\n", 6); 
// 串口处理定时器初始化
Util_constructClock(&g_UART_Clock, multi_role_clockHandler, 0, 0, false, UART_EVT);

3)修改串口处理事件的宏
以multi_role工程为例,在multi_role.c的CONSTANTS常量定义中,加入UART_EVT,id号递增。

// Internal Events for RTOS application
#define MR_ICALL_EVT                         ICALL_MSG_EVENT_ID // Event_Id_31
#define MR_QUEUE_EVT                         UTIL_QUEUE_EVENT_ID // Event_Id_30
#define MR_STATE_CHANGE_EVT                  Event_Id_00
#define MR_CHAR_CHANGE_EVT                   Event_Id_01
#define MR_CONN_EVT_END_EVT                  Event_Id_02
#define MR_KEY_CHANGE_EVT                    Event_Id_03
#define MR_PAIRING_STATE_EVT                 Event_Id_04
#define MR_PASSCODE_NEEDED_EVT               Event_Id_05
#define MR_PERIODIC_EVT                      Event_Id_06
#define UART_EVT                             Event_Id_07    // 串口事件

在MR_ALL_EVENTS事件集合定义中,加入刚刚的自定义串口事件。

#define MR_ALL_EVENTS                        (MR_ICALL_EVT           | \
                                             MR_QUEUE_EVT            | \
                                             MR_STATE_CHANGE_EVT     | \
                                             MR_CHAR_CHANGE_EVT      | \
                                             MR_CONN_EVT_END_EVT     | \
                                             MR_KEY_CHANGE_EVT       | \
                                             MR_PAIRING_STATE_EVT    | \
                                             MR_PERIODIC_EVT         | \
                                             MR_PASSCODE_NEEDED_EVT  | \
                                             UART_EVT)

4)添加串口处理事件的处理
在multi_role.c的multi_role_taskFxn函数中尾部加入。

/*----------------- 串口处理事件 ------------------*/
if (events & UART_EVT)
{
    
    // 串口处理函数
    uart_performTask();
}

5)添加串口事件处理函数
在multi_role.c尾部添加

/*********************************************************************
* @fn     uart_performTask
*
* @brief   串口事件的处理函数,在此函数自定义串口处理功能
*
* @param   none
*
* @return  none
*/
static void uart_performTask(void)
{
    
    // 发送数据     
    UART_Send(g_UART_RxBuf, g_UART_Size);

    // 开始新一轮的读取等待
    UART_Receive(g_UART_RxBuf, g_UART_WantedRxBytes);
}

6)声明串口事件处理函数
在multi_role.c的LOCAL FUNCTIONS局部函数中加入

static void uart_performTask(void);

7)修改配置中的预编译
默认 BOARD_DISPLAY_USE_UARTBOARD_DISPLAY_USE_UART_ANSI 两个宏是为1的,本来TI工程的串口使用,现在要把这两个宏修改为0。


• 由 Leung 写于 2019 年 3 月 7 日

• 参考:【CC2640R2F】香瓜CC2640R2F之串口

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_36347513/article/details/88292705

智能推荐

c# 调用c++ lib静态库_c#调用lib-程序员宅基地

文章浏览阅读2w次,点赞7次,收藏51次。四个步骤1.创建C++ Win32项目动态库dll 2.在Win32项目动态库中添加 外部依赖项 lib头文件和lib库3.导出C接口4.c#调用c++动态库开始你的表演...①创建一个空白的解决方案,在解决方案中添加 Visual C++ , Win32 项目空白解决方案的创建:添加Visual C++ , Win32 项目这......_c#调用lib

deepin/ubuntu安装苹方字体-程序员宅基地

文章浏览阅读4.6k次。苹方字体是苹果系统上的黑体,挺好看的。注重颜值的网站都会使用,例如知乎:font-family: -apple-system, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Microsoft YaHei, Source Han Sans SC, Noto Sans CJK SC, W..._ubuntu pingfang

html表单常见操作汇总_html表单的处理程序有那些-程序员宅基地

文章浏览阅读159次。表单表单概述表单标签表单域按钮控件demo表单标签表单标签基本语法结构<form action="处理数据程序的url地址“ method=”get|post“ name="表单名称”></form><!--action,当提交表单时,向何处发送表单中的数据,地址可以是相对地址也可以是绝对地址--><!--method将表单中的数据传送给服务器处理,get方式直接显示在url地址中,数据可以被缓存,且长度有限制;而post方式数据隐藏传输,_html表单的处理程序有那些

PHP设置谷歌验证器(Google Authenticator)实现操作二步验证_php otp 验证器-程序员宅基地

文章浏览阅读1.2k次。使用说明:开启Google的登陆二步验证(即Google Authenticator服务)后用户登陆时需要输入额外由手机客户端生成的一次性密码。实现Google Authenticator功能需要服务器端和客户端的支持。服务器端负责密钥的生成、验证一次性密码是否正确。客户端记录密钥后生成一次性密码。下载谷歌验证类库文件放到项目合适位置(我这边放在项目Vender下面)https://github.com/PHPGangsta/GoogleAuthenticatorPHP代码示例://引入谷_php otp 验证器

【Python】matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距-程序员宅基地

文章浏览阅读4.3k次,点赞5次,收藏11次。matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距

docker — 容器存储_docker 保存容器-程序员宅基地

文章浏览阅读2.2k次。①Storage driver 处理各镜像层及容器层的处理细节,实现了多层数据的堆叠,为用户 提供了多层数据合并后的统一视图②所有 Storage driver 都使用可堆叠图像层和写时复制(CoW)策略③docker info 命令可查看当系统上的 storage driver主要用于测试目的,不建议用于生成环境。_docker 保存容器

随便推点

网络拓扑结构_网络拓扑csdn-程序员宅基地

文章浏览阅读834次,点赞27次,收藏13次。网络拓扑结构是指计算机网络中各组件(如计算机、服务器、打印机、路由器、交换机等设备)及其连接线路在物理布局或逻辑构型上的排列形式。这种布局不仅描述了设备间的实际物理连接方式,也决定了数据在网络中流动的路径和方式。不同的网络拓扑结构影响着网络的性能、可靠性、可扩展性及管理维护的难易程度。_网络拓扑csdn

JS重写Date函数,兼容IOS系统_date.prototype 将所有 ios-程序员宅基地

文章浏览阅读1.8k次,点赞5次,收藏8次。IOS系统Date的坑要创建一个指定时间的new Date对象时,通常的做法是:new Date("2020-09-21 11:11:00")这行代码在 PC 端和安卓端都是正常的,而在 iOS 端则会提示 Invalid Date 无效日期。在IOS年月日中间的横岗许换成斜杠,也就是new Date("2020/09/21 11:11:00")通常为了兼容IOS的这个坑,需要做一些额外的特殊处理,笔者在开发的时候经常会忘了兼容IOS系统。所以就想试着重写Date函数,一劳永逸,避免每次ne_date.prototype 将所有 ios

如何将EXCEL表导入plsql数据库中-程序员宅基地

文章浏览阅读5.3k次。方法一:用PLSQL Developer工具。 1 在PLSQL Developer的sql window里输入select * from test for update; 2 按F8执行 3 打开锁, 再按一下加号. 鼠标点到第一列的列头,使全列成选中状态,然后粘贴,最后commit提交即可。(前提..._excel导入pl/sql

Git常用命令速查手册-程序员宅基地

文章浏览阅读83次。Git常用命令速查手册1、初始化仓库git init2、将文件添加到仓库git add 文件名 # 将工作区的某个文件添加到暂存区 git add -u # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,不处理untracked的文件git add -A # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,包括untracked的文件...

分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120-程序员宅基地

文章浏览阅读202次。分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120

【C++缺省函数】 空类默认产生的6个类成员函数_空类默认产生哪些类成员函数-程序员宅基地

文章浏览阅读1.8k次。版权声明:转载请注明出处 http://blog.csdn.net/irean_lau。目录(?)[+]1、缺省构造函数。2、缺省拷贝构造函数。3、 缺省析构函数。4、缺省赋值运算符。5、缺省取址运算符。6、 缺省取址运算符 const。[cpp] view plain copy_空类默认产生哪些类成员函数

推荐文章

热门文章

相关标签