前言
終于來到了 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
效果圖
結束語
在 rt-thread 的 finsh 終端串口設備使用 serialX 驅動。初戰告捷!
這次解決兩個問題:一個是, finsh 執行 `rt_device_read` 時可以一次返回多個字節。另一個是,我們可以在終端里粘貼多條命令執行啦。
審核編輯:湯梓紅
-
串口
+關注
關注
14文章
1555瀏覽量
76679 -
RT-Thread
+關注
關注
31文章
1296瀏覽量
40242 -
serialX
+關注
關注
0文章
7瀏覽量
808
發布評論請先 登錄
相關推薦
評論