一、FinSH 的移植与使用问题

FinSH组件输入无反应的问题

  • 现象:当打开 finsh 组件后,控制台会打相应的信息,如下图说是:
 \ | /
- RT - Thread Operating System
/ | \ 3.1.5 build Jul 6 2022
2006 - 2020 Copyright by rt-thread team
do components initialization.
initialize rti_board_end:0 done
initialize finsh_system_init:0 done
RT-Thread Start......
msh >

但是输入回车或任意内容时,无反应。当然导致这样的现象有两种可能,一种可能是未正常开启相应的配置,第二种是未对接收字符的函数进行实现,具体看下面内容

  • FinSH 组件在 rtconfig.h 中的配置,如下所示:
/* --------------------------------- RT-Thread 内核部分 --------------------------------- */

/* 表示内核对象的名称的最大长度,若代码中对象名称的最大长度大于宏定义的长度,
* 多余的部分将被截掉。*/
#define RT_NAME_MAX 8 /* 字节对齐时设定对齐的字节个数。常使用 ALIGN(RT_ALIGN_SIZE) 进行字节对齐。*/
#define RT_ALIGN_SIZE 4 /* 定义系统线程优先级数;通常用 RT_THREAD_PRIORITY_MAX-1 定义空闲线程的优先级 */
#define RT_THREAD_PRIORITY_MAX 32 /* 定义时钟节拍,为 100 时表示 100 个 tick 每秒,一个 tick 为 10ms */
#define RT_TICK_PER_SECOND 1000 /* 检查栈是否溢出,未定义则关闭 */
#define RT_USING_OVERFLOW_CHECK /* 定义该宏开启 debug 模式,未定义则关闭 */
#define RT_DEBUG /* 开启 debug 模式时:该宏定义为 0 时表示关闭打印组件初始化信息,定义为 1 时表示启用 */
#define RT_DEBUG_INIT 1 /* 开启 debug 模式时:该宏定义为 0 时表示关闭打印线程切换信息,定义为 1 时表示启用 */
#define RT_DEBUG_THREAD 0 /* 定义该宏表示开启钩子函数的使用,未定义则关闭 */
//#define RT_USING_HOOK /* 定义了空闲线程的栈大小 */
#define IDLE_THREAD_STACK_SIZE 1024

FinSH 移植

FinSH 组件使用有三种种方式,如下:

  1. 通过 rt_hw_console_getchar() 函数获取控制台数据

    FinSH 线程的使用方式主要是通过实现rt_hw_console_getchar()函数,获取控制台输入的数据,具体方式看我之前的笔记,STM32 移植 RT-Thread 标准版的 FinSH 组件

  2. 通过外设驱动中的 数据流(stm32_getc函数) 获取控制台数据

    具体实现方式见UART外设的移植,稍后我也会将我移植的过程发出来,有需要的可以看我之后的笔记。

  3. 通过外设驱动中的 中断方式 获取控制台数据

    中断的方式包涵了DMA的方式获取控制台数据。

注意: 第二和第三中方式其实都是通过RT-Thread中的外设驱动获取的,这里我为啥会将 数据流和中断 分开说明,是因为他们之间会引入一个新的问题,具体见之后的流程

二、设备为空问题

  • 现象:msh >(dev != RT_NULL) assertion failed at function:rt_device_read, line number:320

  • 原因:出现这个现象主要是在注册设备的时候导致的,在注册设备的时候才用了数据流的方式回去了数据,如下所示:

    /* 配置串口设备 */
    result = rt_hw_serial_register(&uart_obj[i].serial, uart_obj[i].config->name,
    RT_DEVICE_OFLAG_RDWR
    | RT_DEVICE_FLAG_INT_RX
    | RT_DEVICE_FLAG_INT_TX
    , NULL);

    相应的配置宏如下所示:

    #define RT_DEVICE_FLAG_RDONLY       0x001 /* 只读 */
    #define RT_DEVICE_FLAG_WRONLY 0x002 /* 只写 */
    #define RT_DEVICE_FLAG_RDWR 0x003 /* 读写 */
    #define RT_DEVICE_FLAG_REMOVABLE 0x004 /* 可移除 */
    #define RT_DEVICE_FLAG_STANDALONE 0x008 /* 独立 */
    #define RT_DEVICE_FLAG_SUSPENDED 0x020 /* 挂起 */
    #define RT_DEVICE_FLAG_STREAM 0x040 /* 流模式 */
    #define RT_DEVICE_FLAG_INT_RX 0x100 /* 中断接收 */
    #define RT_DEVICE_FLAG_DMA_RX 0x200 /* DMA 接收 */
    #define RT_DEVICE_FLAG_INT_TX 0x400 /* 中断发送 */
    #define RT_DEVICE_FLAG_DMA_TX 0x800 /* DMA 发送 */

    认真思考的小伙伴就会存在一个疑问,为啥将数据接收注册为 流模式 会导致设备为空了,可以猜测在某处导致了设备丢失,我们仔细找一下代码就会发现在shell.c文件中,通过了中断的方式打开了设备,如下图所示:



    现在原因找到了,那么解决方式有两种,如下所示:

解决办法

  1. 将注册设备时,改为中断接收的方式注册设备

  2. 将shell.c文件中的发开方式改为流模式即可,只需要将 RT_DEVICE_FLAG_INT_RX 屏蔽,如下所示:

    void finsh_set_device(const char *device_name)
    {
    rt_device_t dev = RT_NULL; RT_ASSERT(shell != RT_NULL);
    dev = rt_device_find(device_name);
    if (dev == RT_NULL)
    {
    rt_kprintf("finsh: can not find device: %s\n", device_name);
    return;
    } /* check whether it's a same device */
    if (dev == shell->device) return;
    /* open this device and set the new device in finsh shell */
    if (rt_device_open(dev, RT_DEVICE_OFLAG_RDWR |
    // RT_DEVICE_FLAG_INT_RX |
    RT_DEVICE_FLAG_STREAM) == RT_EOK)
    {
    if (shell->device != RT_NULL)
    {
    /* close old finsh device */
    rt_device_close(shell->device);
    rt_device_set_rx_indicate(shell->device, RT_NULL);
    } /* clear line buffer before switch to new device */
    memset(shell->line, 0, sizeof(shell->line));
    shell->line_curpos = shell->line_position = 0; shell->device = dev;
    rt_device_set_rx_indicate(dev, finsh_rx_ind);
    }
    }

    ** 注意:** 改为流模式后,会发现一个奇怪的现象,就是当你使用调试模式时,可以正常接收指令,但是正常运行时,就无任何响应,遇到这样的现象不要慌,接着往下看。

三、FinSH 卡死问题

  • 现象:FinSH线程卡死,明显的发现就是,使用调试模式时,可以正常接收指令,但是正常运行时,就无任何响应。

  • 原因:深入分析后,会在 shell.c 文件中的 finsh_getchar 函数中看到信号量的使用,如下图所示:

    现在原因找到了,那么我们只需要在适当的时候释放信号量即可,那我们在找找看看有么有信号释放的函数,接下来我们会发现在 shell.c 文件中 finsh_rx_ind 函数就是释放信号量的,如图所示:

    那么新的问题又来了,怎么调用这个函数了,因为在shell.h文件中也没有这个函数的定义, 不要怕我们接着找,最后在shell.c 文件中的 finsh_set_device 函数中,会将释放信号量的函数指针放入 rt_device 结构体中,如下图所示:

    那么问题就变得简单了,解决办法如下

  • 解决办法:我们已经知道怎么释放信号量了,所以只需要在 数据接收函数(stm32_getc)中,完成数据接收后,释放信号量即可,如下所示:

    /**
    * 接收一个字符数据
    */
    static int stm32_getc(struct rt_serial_device *serial)
    {
    int ch;
    struct stm32_uart *uart;
    RT_ASSERT(serial != RT_NULL);
    uart = rt_container_of(serial, struct stm32_uart, serial); ch = -1;
    if (USART_GetFlagStatus(uart->handle.Instance, USART_FLAG_RXNE) != RESET)
    {
    #if defined(SOC_SERIES_STM32L4) || defined(SOC_SERIES_STM32F7) || defined(SOC_SERIES_STM32F0) \
    || defined(SOC_SERIES_STM32L0) || defined(SOC_SERIES_STM32G0) || defined(SOC_SERIES_STM32H7) \
    || defined(SOC_SERIES_STM32G4)
    ch = uart->handle.Instance->RDR & 0xff;
    #else
    // ch = (uint16_t)uart->handle.Instance->DR & (uint16_t)0x00ff;
    ch = (char)USART_ReceiveData(uart->handle.Instance);
    #endif } /* 调用设备接收数据回调,释放信号量 */
    uart->serial.parent.rx_indicate(&serial->parent, 0);
    return ch;
    }

四、测试

解决完问题后,在控制台输入回车有相应的回应就没问题,如下图所示:

RT-Thread 组件 FinSH 使用时遇到的问题的更多相关文章

  1. 小程序scroll-view组件使用时,子元素虽设置样式display:inline-flex;whit-space:nowrap

    小程序scroll-view组件使用时,子元素虽设置样式display:inline-flex;whit-space:nowrap

  2. Android插件化(三):OpenAtlas的插件重建以及使用时安装

    Android插件化(三):OpenAtlas的插件重建以及使用时安装 转 https://www.300168.com/yidong/show-2778.html    核心提示:在上一篇博客 An ...

  3. Android Loader使用时,屏幕解锁后,重复加载

    在使用AsyncTaskLoader时,当手机解锁后,会重复加载数据,代码如下: static class CouponShopQueryLoader extends AsyncTaskLoader& ...

  4. Cookie使用时需要注意个数及大小限制

    各浏览器对Cookie有一定的限制,在使用时需要格外注意. 各浏览器之间对cookie的不同限制:   IE6.0 IE7.0/8.0/9.0+ Opera FF Safari Chrome cook ...

  5. EntityFrameWork 使用时碰到的小问题

    EntityFrameWork 使用时碰到的小问题 1,在使用orm访问数据库的相目里,也要引用EntityFrameWork.dll,否则无法使用orm 否则,编译错误 错误 5 "Sys ...

  6. MySQL 安装和启动服务,“本地计算机 上的 MySQL 服务启动后停止。某些服务在未由其他服务或程序使用时将自动停止。”

    MySQL 安装和启动服务,以及遇到的问题 MySQL版本: mysql-5.7.13-winx64.zip (免安装,解压放到程序文件夹即可,比如 C:\Program Files\mysql-5. ...

  7. MaterialCalendarView使用时遇到的问题

    一.概述 MaterialCalendarView是一个开源项目.功能强大支持多选.单选.标注等. 二.问题 1.其继承自ViewGroup,故与CalendarView半毛钱关系都没有,完全是一个新 ...

  8. [备忘][转]rsync使用时的常见问题

    sync使用时的常见问题: 错误1: rsync: read error: Connection reset by peer (104) rsync error: error in rsync pro ...

  9. 小白学数据分析----->移动游戏的使用时长分析

    写下该文章,是因为之前看到了几款游戏一个典型的玩家刺激活动,在<多塔联盟>,<萌江湖>等多款游戏的设计中都有体现,如下图所示: 这个功能点的设计,今天在这里讲的更多的还是跟数据 ...

随机推荐

  1. GO语言学习——基本数据类型字符串

    字符串 Go语言中的字符串以原生数据类型出现. Go 语言里的字符串的内部实现使用UTF-8编码. 字符串的值为双引号(")中的内容,可以在Go语言的源码中直接添加非ASCII码字符 GO语 ...

  2. Objective-C 基础教程第九章,内存管理

    目录 Object-C 基础教程第九章,内存管理 前言: 对象生命周期 引用计数 RetainCount1项目例子 对象所有权 访问方法中的保留和释放 自动释放 所有对象放入池中 自动释放池的销毁时间 ...

  3. 用python爬虫,对12306网站进行模拟登陆

    from selenium import webdriver from time import sleep from PIL import Image from selenium.webdriver ...

  4. 深度优先搜索 DFS 学习笔记

    深度优先搜索 学习笔记 引入 深度优先搜索 DFS 是图论中最基础,最重要的算法之一.DFS 是一种盲目搜寻法,也就是在每个点 \(u\) 上,任选一条边 DFS,直到回溯到 \(u\) 时才选择别的 ...

  5. ubuntu 16.04,ros kinetic 使用husy_gazebo

    我当前使用的是ubuntu 16.04,ros kinetic ,Gazebo版本为7.0.protoc需要确保版本为2.6.1,而我当前的为3.4.0,因此需要将系统中的protoc替换为2.6.1 ...

  6. 修复Arch Linux和Manjaro Linux无法显示emoji的问题

    安装好Arch Linux或Manjaro Linux系统后默认没办法正常显示emoji,通常会变成方框或者带有unicode码的方块: 这是因为缺失字体以及相关的字体配置导致的. 当然也有一小部分应 ...

  7. Ping原理详解

    关注「开源Linux」,选择"设为星标" 回复「学习」,有我为您特别筛选的学习资料~ 前言 Ping是排除设备访问故障的常见方法.它使用Internet控制消息协议ICMP(Int ...

  8. shell脚本实现MySQL全量备份+异地备份

    一.知识储备工作: Mysql导出数据库语法: mysqldump -u用户名 -p密码 数据库名 > 数据库名.sql shell脚本for循环及if条件判断基本语法 gzip压缩文件用法 r ...

  9. Google Summer of Code谷歌编程之夏活动流程全解析(上)

    本期由尔等同学来对话Casbin罗杨老师,为大家介绍开源及GSoC活动流程. > 罗杨:GSoC 2013.2015学生.GSoC期间在Nmap开源社区作为主力开发了Windows平台网络抓包工 ...

  10. 真香警告!JitPack 开源库集成平台

    前言: 请各大网友尊重本人原创知识分享,谨记本人博客:南国以南i 简介 官方介绍: JitPack 是一个用于 JVM 和 Android 项目的新颖的包存储库.它按需构建 Git 项目并为您提供即用 ...