CC2640R2F学习笔记(6)——UART串口使用

一、硬件连接

串口 引脚
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之串口

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容

  • 大学的时候,帮朋友写的操作系统调研的作业,最近整理过去的文档时候偶然发现,遂作为博客发出来。 从串口驱动到Linu...
    free_will阅读 7,358评论 7 59
  • ORA-00001: 违反唯一约束条件 (.) 错误说明:当在唯一索引所对应的列上键入重复值时,会触发此异常。 O...
    我想起个好名字阅读 5,141评论 0 9
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,082评论 1 32
  • “串体系”的日子里,也认识了许多牛人!比如天津一姑娘,研究生,讲奖金制度讲的最好,来自北京的乔阿姨,公务员,价值远...
    大張冰阅读 139评论 4 4
  • 现实生活中,交易费用永远不是零,只要有人的地方,只要有两个人以上的社会,它就存在着大量的交易费用.正是因为有大量的...
    尘世书童阅读 158评论 0 1