前言
終于來到了 serialX 的實踐篇,期待很久了。
筆者曾經在 [rt-thread 使用寶典(2022-0516更新)](https://club.rt-thread.org/ask/article/2460fcd7db4821ae.html) 這篇文章的“使用篇: Q1. 串口通訊數據被分多次接收了,怎么辦?”里貼了一段代碼,那段代碼有很強的適用性,稍作修改就能用到多種串口協議處理場合。今天我們嘗試在 finsh 上應用 serialX,看看它能給我們帶來什么神奇效果。
打開控制臺
我們的 serialX 支持中斷收發、DMA 收發。所以我們可以隨意組合使用 中斷收、中斷發、DMA 收、DMA 發,共四種組合(前提是對應芯片底層驅動支持 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 自帶阻塞讀特性,所以它不需要執行 `rt_device_set_rx_indicate(dev, finsh_rx_ind);` 這句代碼,我們接收數據自有同步妙法,請往下看。
finsh 線程
對 finsh 線程入口函數稍作修改:
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 線程提供一個應用層的數據緩存 `istream` ,這里只用的 32 個字節。
2. `finsh_instream` 函數代替 `finsh_getchar` ,它用來讀串口終端設備數據流,函數實現見下文。它可能返回多個字節數據,返回值表示有效數據個數
3. 接下來對 `finsh_instream` 讀到的每字節數據進行處理
4. `finsh_handle_onebyte` 是對原來 `finsh_thread_entry` 函數中的 `while` 循環進行的改造
讀終端串口設備
如果 serialX 的阻塞模式打開的,同時串口接收緩存里是空的,執行 `rt_device_read` 會永久等待下去,當前線程進入睡眠態。
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 */}
讀串口設備的數據放到 buf 指向的內存中,最多 len 個字節,最終返回實際讀到的數據量。
注:`rt_device_read` 的返回值可能是 0,也可能會是 -1。
處理命令行字符
這部分筆者把他們放到了一個單獨的函數,不這么做也沒影響。
筆者做了一點兒小改進。
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" 三種組合結束的命令都能識別,**更重要的是,它可以識別以 '\r' '\n' "\r\n" 分割的多條命令!!!**
如下命令列表,可以全復制,粘貼到終端,四條命令逐個被執行。
lspslist_devicelist_thread
效果圖
![pYYBAGKoI1yASW4hAAONcCR-Fko872.png](https://file.elecfans.com/web2/M00/4B/54/pYYBAGKoI1yASW4hAAONcCR-Fko872.png)
結束語
在 rt-thread 的 finsh 終端串口設備使用 serialX 驅動。初戰告捷!
這次解決兩個問題:一個是, finsh 執行 `rt_device_read` 時可以一次返回多個字節。另一個是,我們可以在終端里粘貼多條命令執行啦。
審核編輯:湯梓紅
-
串口
+關注
關注
14文章
1560瀏覽量
77149 -
RT-Thread
+關注
關注
31文章
1306瀏覽量
40434 -
serialX
+關注
關注
0文章
7瀏覽量
811
發布評論請先 登錄
相關推薦
RT-Thread上CAN實踐
![<b class='flag-5'>RT-Thread</b>上CAN實踐](https://file1.elecfans.com/web2/M00/C4/8A/wKgZomX0EhWACv8DAAAUet8ikhs451.png)
新書發布——《RT-Thread嵌入式實時操作系統內核、驅動和應用開發技術》
![新書發布——《<b class='flag-5'>RT-Thread</b>嵌入式實時操作系統內核、<b class='flag-5'>驅動</b>和應用開發技術》](https://file1.elecfans.com/web2/M00/C4/8A/wKgZomX0EhWACv8DAAAUet8ikhs451.png)
2024 RT-Thread全球巡回 線下培訓火熱來襲!
![2024 <b class='flag-5'>RT-Thread</b>全球巡回 線下培訓火熱來襲!](https://file1.elecfans.com/web2/M00/C4/8A/wKgZomX0EhWACv8DAAAUet8ikhs451.png)
【好書推薦】RT-Thread設備驅動開發指南
![【好書推薦】<b class='flag-5'>RT-Thread</b>設備<b class='flag-5'>驅動</b>開發指南](https://file1.elecfans.com/web2/M00/C4/8A/wKgZomX0EhWACv8DAAAUet8ikhs451.png)
RT-Thread 新里程碑達成——GitHub Star 破萬!
![<b class='flag-5'>RT-Thread</b> 新里程碑達成——GitHub Star 破萬!](https://file1.elecfans.com/web2/M00/C4/8A/wKgZomX0EhWACv8DAAAUet8ikhs451.png)
6月6日杭州站RT-Thread線下workshop,探索RT-Thread混合部署新模式!
![6月6日杭州站<b class='flag-5'>RT-Thread</b>線下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新模式!](https://file1.elecfans.com/web2/M00/C4/8A/wKgZomX0EhWACv8DAAAUet8ikhs451.png)
4月25日北京站RT-Thread線下workshop,探索RT-Thread混合部署新模式
![4月25日北京站<b class='flag-5'>RT-Thread</b>線下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新模式](https://file1.elecfans.com/web2/M00/C4/8A/wKgZomX0EhWACv8DAAAUet8ikhs451.png)
4月10日深圳場RT-Thread線下workshop,探索RT-Thread混合部署新模式!
![4月10日深圳場<b class='flag-5'>RT-Thread</b>線下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新模式!](https://file1.elecfans.com/web2/M00/C6/D0/wKgaomYDlJyAKUBmAAAgR-TqYwc187.png)
4月10日深圳場RT-Thread線下workshop,探索RT-Thread混合部署新模式!
![4月10日深圳場<b class='flag-5'>RT-Thread</b>線下workshop,探索<b class='flag-5'>RT-Thread</b>混合部署新模式!](https://file1.elecfans.com/web2/M00/C4/8A/wKgZomX0EhWACv8DAAAUet8ikhs451.png)
恩智浦半導體正式加入RT-Thread全球合作伙伴計劃!
![恩智浦半導體正式加入<b class='flag-5'>RT-Thread</b>全球合作伙伴計劃!](https://file1.elecfans.com/web2/M00/C5/31/wKgaomXyY_eAHGhqAAAkpBZI8PE901.png)
RT-Thread驅動開發指南進階篇-動手驅動先楫未適配的外設LCD
![<b class='flag-5'>RT-Thread</b><b class='flag-5'>驅動</b>開發指南進階<b class='flag-5'>篇</b>-動手<b class='flag-5'>驅動</b>先楫未適配的外設LCD](https://file1.elecfans.com/web2/M00/C1/D4/wKgaomXarr-AKhdfAAAcu6ZeWvU306.png)
《RT-Thread設備驅動開發指南》基礎篇--以先楫bsp的hwtimer設備為例
![《<b class='flag-5'>RT-Thread</b>設備<b class='flag-5'>驅動</b>開發指南》基礎<b class='flag-5'>篇</b>--以先楫bsp的hwtimer設備為例](https://file.elecfans.com/web2/M00/37/D7/pYYBAGI9l9uAOwALAAAmFmqVYdg094.png)
RT-Thread設備驅動開發指南基礎篇—以先楫bsp的hwtimer設備為例
![<b class='flag-5'>RT-Thread</b>設備<b class='flag-5'>驅動</b>開發指南基礎<b class='flag-5'>篇</b>—以先楫bsp的hwtimer設備為例](https://file1.elecfans.com/web2/M00/C1/40/wKgaomXUXOGAMhesAAAk3OUcwHA076.png)
評論