前言
終于來到了 serialX 的實踐篇,期待很久了。
筆者曾經(jīng)在 [rt-thread 使用寶典(2022-0516更新)](https://club.rt-thread.org/ask/article/2460fcd7db4821ae.html) 這篇文章的“使用篇: Q1. 串口通訊數(shù)據(jù)被分多次接收了,怎么辦?”里貼了一段代碼,那段代碼有很強的適用性,稍作修改就能用到多種串口協(xié)議處理場合。今天我們嘗試在 finsh 上應(yīng)用 serialX,看看它能給我們帶來什么神奇效果。
打開控制臺
我們的 serialX 支持中斷收發(fā)、DMA 收發(fā)。所以我們可以隨意組合使用 中斷收、中斷發(fā)、DMA 收、DMA 發(fā),共四種組合(前提是對應(yīng)芯片底層驅(qū)動支持 DMA)。
if (rt_device_open(new_device, RT_DEVICE_OFLAG_RDWR | RT_DEVICE_FLAG_STREAM | RT_DEVICE_FLAG_INT_RX | RT_DEVICE_FLAG_INT_TX ) == RT_EOK) { }
或者
if (rt_device_open(new_device, RT_DEVICE_OFLAG_RDWR | RT_DEVICE_FLAG_STREAM | RT_DEVICE_FLAG_DMA_RX | RT_DEVICE_FLAG_DMA_TX ) == RT_EOK) { }
因為 serialX 自帶阻塞讀特性,所以它不需要執(zhí)行 `rt_device_set_rx_indicate(dev, finsh_rx_ind);` 這句代碼,我們接收數(shù)據(jù)自有同步妙法,請往下看。
finsh 線程
對 finsh 線程入口函數(shù)稍作修改:
void finsh_thread_entry(void *parameter){ int i, cnt; char istream[32]; ... 省略部分操作 while (1) { cnt = finsh_instream(istream, 32); for (i = 0; i < cnt; i++) { finsh_handle_onebyte(istream[i]); } }}
1. finsh 線程提供一個應(yīng)用層的數(shù)據(jù)緩存 `istream` ,這里只用的 32 個字節(jié)。
2. `finsh_instream` 函數(shù)代替 `finsh_getchar` ,它用來讀串口終端設(shè)備數(shù)據(jù)流,函數(shù)實現(xiàn)見下文。它可能返回多個字節(jié)數(shù)據(jù),返回值表示有效數(shù)據(jù)個數(shù)
3. 接下來對 `finsh_instream` 讀到的每字節(jié)數(shù)據(jù)進行處理
4. `finsh_handle_onebyte` 是對原來 `finsh_thread_entry` 函數(shù)中的 `while` 循環(huán)進行的改造
讀終端串口設(shè)備
如果 serialX 的阻塞模式打開的,同時串口接收緩存里是空的,執(zhí)行 `rt_device_read` 會永久等待下去,當(dāng)前線程進入睡眠態(tài)。
int finsh_instream(char *buf, int len){#ifdef RT_USING_DEVICE int i; RT_ASSERT(shell != RT_NULL); i = rt_device_read(shell->device, -1, buf, len); return i;#else extern char rt_hw_console_getchar(void); return rt_hw_console_getchar();#endif /* RT_USING_DEVICE */}
讀串口設(shè)備的數(shù)據(jù)放到 buf 指向的內(nèi)存中,最多 len 個字節(jié),最終返回實際讀到的數(shù)據(jù)量。
注:`rt_device_read` 的返回值可能是 0,也可能會是 -1。
處理命令行字符
這部分筆者把他們放到了一個單獨的函數(shù),不這么做也沒影響。
筆者做了一點兒小改進。
static void finsh_handle_onebyte(int ch){ static int last_ch = 0x20; ... /* handle end of line, break */ if (last_ch == '\r' && ch == '\n') { last_ch = ch; return; } if (ch == '\r' || ch == '\n') {#ifdef FINSH_USING_HISTORY shell_push_history(shell);#endif if (shell->echo_mode) rt_kprintf("\n"); msh_exec(shell->line, shell->line_position); rt_kprintf(FINSH_PROMPT); rt_memset(shell->line, 0, sizeof(shell->line)); shell->line_curpos = shell->line_position = 0; last_ch = ch; return; } ... last_ch = ch; // ch = 0; ...}
這樣一來,對以 '\r' '\n' "\r\n" 三種組合結(jié)束的命令都能識別,**更重要的是,它可以識別以 '\r' '\n' "\r\n" 分割的多條命令!!!**
如下命令列表,可以全復(fù)制,粘貼到終端,四條命令逐個被執(zhí)行。
lspslist_devicelist_thread
效果圖

結(jié)束語
在 rt-thread 的 finsh 終端串口設(shè)備使用 serialX 驅(qū)動。初戰(zhàn)告捷!
這次解決兩個問題:一個是, finsh 執(zhí)行 `rt_device_read` 時可以一次返回多個字節(jié)。另一個是,我們可以在終端里粘貼多條命令執(zhí)行啦。
審核編輯:湯梓紅
-
串口
+關(guān)注
關(guān)注
15文章
1618瀏覽量
82810 -
RT-Thread
+關(guān)注
關(guān)注
32文章
1613瀏覽量
44868 -
serialX
+關(guān)注
關(guān)注
0文章
7瀏覽量
875
發(fā)布評論請先 登錄
恩智浦亮相RT-Thread 20周年開發(fā)者大會
基于RT-Thread和兆易創(chuàng)新GD32F527系列MCU的健康監(jiān)測站 | 技術(shù)集結(jié)
首搭RT-Thread程翧車控平臺| RT-Thread程翧 S32K344 快速原型開發(fā)平臺正式上市!| 產(chǎn)品動態(tài)
基于RT-Thread的EK-RA2E2 設(shè)備驅(qū)動移植與應(yīng)用 | 技術(shù)集結(jié)
2025年RT-Thread開發(fā)者巡回培訓(xùn)報名正式啟動!
使用RT-Thread studio 開發(fā)RT-Thread Nano項目的時候 串口的驅(qū)動是要自己寫嗎?
rt-thread studio 2.2.9如何使用最新的RT-Thread v5.2.0 released?
【好書推薦】RT-Thread第20本相關(guān)書籍!《嵌入式實時操作系統(tǒng)RT-Thread原理與應(yīng)用》| 技術(shù)集結(jié)
RT-Thread Nano硬核移植指南:手把手實現(xiàn)VGLite圖形驅(qū)動適配 | 技術(shù)集結(jié)
用SPI玩轉(zhuǎn)WiFi,RT-Thread ESP-Hosted驅(qū)動深度適配指南 | 技術(shù)集結(jié)
RT-Thread榮獲2025優(yōu)秀開源項目 | 新聞速遞
深度剖析 RT-Thread 線程調(diào)度流程
揭秘RT-Thread上的AUTOSAR CP系統(tǒng)
2025 RT-Thread全球技術(shù)大會議程正式發(fā)布!
RT-Thread審核團招募: 深度參與開源RTOS社區(qū)治理與演進
rt-thread 驅(qū)動篇(五)serialX 小試牛刀
評論