zoukankan      html  css  js  c++  java
  • Zephyr的Shell

    1 前言

    通过Shell可以跟子系统打交道,子系统也可以提供很多接口供外部设置和读取信息。

    下面就Shell的Kconfig配置、Shell的使用以及如何新建一个Shell命令展开。

    可以说Shell是一窥内核究竟的管道,有了这个管道能使开发事半功倍。

    有时为了开发和调试需求,还需要新增命令。

    2 Shell相关配置

    CONFIG_CONSOLE_SHELL是Shell子系统的开关,打开某一个模块的Shell,比如NET可以通过CONFIG_NET_SHELL=y。

    子模块的使能必须基于CONFIG_CONSOLE_SHELL。

    如果要新增一个Shell配置,需要在使能CONFIG_CONSOLE_SHELL的情况下,在对应子模块中的Kconfig新增一个选项。

     3 新增一个Shell命令

    新增一个Shell命令,有如下步骤。

    1. 新增一个选项,当然必须在定义CONFIG_CONSOLE_SHELL情况下才有效。

    2. 通过SHELL_REGISTER.注册一个新命令,包括命令名称和回调函数。

    shell命令结构体:

    struct shell_cmd {
        const char *cmd_name;---------------子命令名称
        shell_cmd_function_t cb;------------子命令回调
        const char *help;-------------------子命令帮助信息
    };

    3.1 kernel命令走读

    kernel命令提供了获取

    #if defined(CONFIG_INIT_STACKS)
    static int shell_cmd_stack(int argc, char *argv[])
    {
        k_call_stacks_analyze();
        return 0;
    }
    #endif
    
    struct shell_cmd kernel_commands[] = {-----------------------------------------shell_cmd类型的结构体数组
        { "version", shell_cmd_version, "show kernel version" },--------------------
        { "uptime", shell_cmd_uptime, "show system uptime in milliseconds" },
        { "cycles", shell_cmd_cycles, "show system hardware cycles" },
    #if defined(CONFIG_OBJECT_TRACING) && defined(CONFIG_THREAD_MONITOR)
        { "tasks", shell_cmd_tasks, "show running tasks" },
    #endif
    #if defined(CONFIG_INIT_STACKS)
        { "stacks", shell_cmd_stack, "show system stacks" },
    #endif
        { NULL, NULL, NULL }
    };
    
    SHELL_REGISTER(SHELL_KERNEL, kernel_commands);---------------------------------通过SHELL_REGISTER注册shell这个命令,处理函数是kernel_commands。

    4 Shell相关API

    位于Shell API Functions.

    Shell子系统初始化SYS_INIT(shell_run, APPLICATION, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT)(shell_service.c)。

    #define SHELL_PROMPT "shell> "
    
    int shell_run(struct device *dev)
    {
        ARG_UNUSED(dev);
    
        shell_init(SHELL_PROMPT);
        return 0;
    }
    
    SYS_INIT(shell_run, APPLICATION, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);---------------Shell子系统初始化
    
    
    void shell_init(const char *str)
    {
        k_fifo_init(&cmds_queue);
        k_fifo_init(&avail_queue);
    
        line_queue_init();
    
        prompt = str ? str : "";
    
        k_thread_create(&shell_thread, stack, STACKSIZE, shell, NULL, NULL,---------------创建一个shell线程处理shell命令
                NULL, K_PRIO_COOP(7), 0, K_NO_WAIT);
    
        /* Register serial console handler */
    #ifdef CONFIG_UART_CONSOLE
        uart_register_input(&avail_queue, &cmds_queue, completion);
    #endif
    #ifdef CONFIG_TELNET_CONSOLE
        telnet_register_input(&avail_queue, &cmds_queue, completion);
    #endif
    }
  • 相关阅读:
    setTimeout()和setInterval()的区别
    iOS开发小技巧
    iOS应用跳转到App Store评分
    前端小技巧-定位的活学活用之仿淘宝列表
    前端CSS
    用c# 开发html5的尝试,试用bridge.net
    Faster数据库研习,一
    五一劳动节,讲讲NEO智能合约的调试
    NEO GUI 多方签名使用
    NEO智能合约开发(二)再续不可能的任务
  • 原文地址:https://www.cnblogs.com/arnoldlu/p/7667371.html
Copyright © 2011-2022 走看看