OpenOCD 代码学习(1)注册命令

概述

  • 1)OpenOCD 与调试器、MCU 的关系:

    • (1)OpenOCD 运行在 PC 机上,用户可以通过 GDB 调试、Telnet 连接 Socket 以及 CMD 命令行的方式执行命令

    • (2)OpenOCD 将命令发送到 MCU 端需要双方约定协议。目前常用的协议为 SWD 和 JTAG,实现这些协议的工具有 DAPLink、ST-Link、JLink 等

    • (3)在 MCU 端内置了 DAP(Debug Access Port)模块,它接收到命令,通过 AHB 总线控制 CPU 内核。(因此,通过 DAP 我们可以访问挂载在 AHB 总线上所有外设,尤其是 FLASH 外设,它是实现芯片烧录程序的关键。)

  • 2)在 《Windows 下编译 OpenOCD》一节,我们已经可以通过 JetBrains CLion 打开 OpenOCD 源码(也可以使用 VSCode 打开)。一般程序都是从 main() 函数开始,那么让我们打开 /openocd/src 目录,从 main.c 文件开始:

1 main()

  • 从 main() 函数到 openocd_main() 函数,最后再到 setup_command_handler() 函数,OpenOCD 进行命令注册操作。

  • 那么接下来,我们重点看一下 setup_command_handler() 函数。

2 setup_command_handler()

  • 1)setup_command_handler() 函数主要进行了以下两个操作,我们一个一个来看:

    • command_init()
    • (*command_registrants[i])(cmd_ctx)
  • 2)对于 command_init() 函数,我们略去 Jim 框架的代码,可以看到以下内容:

    register_commands(context, NULL, command_builtin_handlers);
    
    if (Jim_Eval_Named(interp, startup_tcl, "embedded:startup.tcl", 1) == JIM_ERR) {
    
    • 第一行代码注册一些 OpenOCD 的内置命令,如 ocd_find、capture、sleep 等等
    • 第二行则是解析了 startup.tcl 文件。

    这里要注意,.tcl 文件里有一些调用过程,当你无法在源码中找到某些命令的实现时,可以搜索一下 .tcl 文件。如 CLion 通过 program 命令烧录,而该命令实际上是一个定义在 src/flash/startup.tcl 中的一个 proc,并在其内部转换成 flash write_image erase 组合命令来进行烧录。

  • 3)对于 (*command_registrants[i])(cmd_ctx) 函数调用,我们结合上下文来看:

/* 1 定义名为 command_registrant_t,参数类型为 struct command_context 的函数指针 */
typedef int (*command_registrant_t)(struct command_context *cmd_ctx_value);
/* 2 声明一个 command_registrant_t 类型的函数指针数组 */
static const command_registrant_t command_registrants[] = {
    &workaround_for_jimtcl_expr,
    &openocd_register_commands,
    &server_register_commands,
    &gdb_register_commands,
    &log_register_commands,
    &rtt_server_register_commands,
    &transport_register_commands,
    &adapter_register_commands,
    &target_register_commands,
    &flash_register_commands,
    &nand_register_commands,
    &pld_register_commands,
    &cti_register_commands,
    &dap_register_commands,
    &arm_tpiu_swo_register_commands,
    NULL
};
/* 3 遍历并执行声明的函数指针 */
for (unsigned i = 0; command_registrants[i]; i++) {
    int retval = (*command_registrants[i])(cmd_ctx);
    if (retval != ERROR_OK) {
        command_done(cmd_ctx);
        return NULL;
    }
}
  • 也就是说,(*command_registrants[i])(cmd_ctx) 其实是调用那些注册命令的函数。下面我们来看一下命令注册的实现逻辑。

3 register_commands()

3.1 server_register_commands()

  • 1)这里我们以 server_register_commands() 函数为例,如下图:

    • telnet_register_commands():注册 telnet 相关的命令,如 exit、telnet_port 等,且在这里指定 telnet 端口号为 4444

    • tcl_register_commands() 和 jsp_register_commands() 分别指定了 6666 和 7777 两个端口号(暂时不知道这个两个是干吗的,不过不影响使用)

    • register_commands():注册 server 相关的命令,如 shutdown 等。

  • 2)然后我们看一下命令的具体声明内容:

    static const struct command_registration telnet_command_handlers[] = {
    	{
    		.name = "exit",
    		.handler = handle_exit_command,
    		.mode = COMMAND_EXEC,
    		.usage = "",
    		.help = "exit telnet session",
    	},
    	{
    		.name = "telnet_port",
    		.handler = handle_telnet_port_command,
    		.mode = COMMAND_CONFIG,
    		.help = "Specify port on which to listen "
    			"for incoming telnet connections.  "
    			"Read help on 'gdb_port'.",
    		.usage = "[port_num]",
    	},
    	COMMAND_REGISTRATION_DONE
    };
    
    • (1)其中的 .name 是处理器名称,Jim 模块可根据该名称查找到对应的 handler
    • (2)mode 表示该 handler 的模式(我愿称之为生命周期):
        1. COMMAND_EXEC,表示该 handler 在命令行输入时才会触发
        1. COMMAND_CONFIG,表示该 handler 在 OpenOCD 启动时解析配置文件时触发
        1. COMMAND_ANY,表示该 handler 以上两种情况均会触发

3.2 register command

  • 1)以下为注册命令的实现逻辑:

  • 2)每个函数的作用已经在流程图中注明,需要注意有两点:

    • 将 jim_command_dispatch() 函数赋值给 cmdProc 指针的 u.native.cmdProc 属性
    • 将 command_new() 函数构造的 command 实例,赋值给 cmdProc 指针的 u.native.privData 属性。
    Jim_Cmd *cmdPtr = Jim_Alloc(sizeof(*cmdPtr));
    
    /* Store the new details for this command */
    memset(cmdPtr, 0, sizeof(*cmdPtr));
    cmdPtr->inUse = 1;
    cmdPtr->u.native.delProc = delProc;
    cmdPtr->u.native.cmdProc = cmdProc;     // jim_command_dispatch
    cmdPtr->u.native.privData = privData;   // command 实例
    
  • 3)最终所有命令会缓存到 Jim 框架的 Hash 数组结构中(查找时间复杂度 O(1))。

  • 4)通过对各个 xx__register_commands() 函数的遍历,所有的命令都将注册到此。注册完命令后要执行命令,下一篇文章我们来看一下执行命令的逻辑。

posted @ 2024-09-28 13:23  送南阳马生序  阅读(238)  评论(0编辑  收藏  举报