FreeBSD 中文社区 2025 第二季度问卷调查
FreeBSD 中文社区(CFC)
VitePress 镜像站QQ 群 787969044视频教程Ⅰ视频教程Ⅱ
  • FreeBSD 从入门到追忆
  • 中文期刊
  • 状态报告
  • 发行说明
  • 手册
  • 网络文章集锦
  • 笔记本支持报告
  • Port 开发者手册
  • 架构手册
  • 开发者手册
  • 中文 man 手册
  • 文章
  • 书籍
  • FreeBSD 中文期刊
  • 编辑日志
  • 2025-123 下游项目
    • FreeBSD 发布工程:新主管上任
    • GhostBSD:从易用到挣扎与重生
    • BSD Now 与将来
    • 字符设备驱动教程(第三部分)
    • 学会走路——连接 GPIO 系统
    • FreeBSD 中对 SYN 段的处理
    • FreeBSD 2024 年秋季峰会
  • 2024-1112 虚拟化
    • 字符设备驱动程序教程(第二部分)
    • 面向 Linux 和 Windows 用户的 bhyve
    • Xen 与 FreeBSD
    • Wifibox:一种嵌入式虚拟化无线路由器
    • 嵌入式 FreeBSD:Fabric——起步阶段
    • DGP:一种新的数据包控制方法
    • 会议报告:我在都柏林的 EuroBSDCon 体验
  • 2024-0910 内核开发
    • 字符设备驱动程序教程
    • VPP 移植到了 FreeBSD:基础用法
    • 利用 Kyua 的 Jail 功能提升 FreeBSD 测试套件的并行效率
    • FreeBSD 上的 Valgrind
    • 嵌入式 FreeBSD:探索 bhyve
    • TCP/IP 历险记:FreeBSD TCP 协议栈中的 Pacing
    • 实用软件:实现无纸化(Paperless)
  • 2024-0708 存储与文件系统
    • FreeBSD 中的 NVMe-oF
    • FreeBSD iSCSI 入门
    • 使用 ZFS 原生加密保护数据
    • 嵌入式 FreeBSD:打造自己的镜像
    • TCP LRO 简介
    • 基于 Samba 的时间机器备份
  • 2024-0506 配置管理对决
    • 基本系统中的 mfsBSD
    • rdist
    • Hashicorp Vault
    • 在 GitHub 上向 FreeBSD 提交 PR
    • 悼念 Mike Karels
    • 2024 年 5-6 月来信
    • 嵌入式 FreeBSD 面包板
    • TCP/IP 历险记:TCP BBLog
    • 实用软件:开发定制 Ansible 模块
  • 2024-0304 开发工作流与集成
    • FreeBSD 内核开发工作流程
    • FreeBSD 与 KDE 持续集成(CI)
    • 更现代的内核调试工具
    • 从零开始的 ZFS 镜像及 makefs -t zfs
    • 提升 Git 使用体验
  • 2024-0102 网络(十周年)
    • FreeBSD 中的 RACK 栈和替代 TCP 栈
    • FreeBSD 14 中有关 TCP 的更新
    • if_ovpn 还是 OpenVPN
    • SR-IOV 已成为 FreeBSD 的重要功能
    • FreeBSD 接口 API(IfAPI)
    • BATMAN:更优的可移动热点网络方式
    • 配置自己的 VPN——基于 FreeBSD、Wireguard、IPv6 和广告拦截
    • 实用软件:使用 Zabbix 监控主机
  • 2023-1112 FreeBSD 14.0
    • LinuxBoot:从 Linux 启动 FreeBSD
    • FreeBSD 容器镜像
    • 现在用 Webhook 触发我
    • 新的 Ports 提交者:oel Bodenmann (jbo@freebsd.org)
  • 2023-0910 Port 与软件包
    • 回忆录:与 Warner Losh(@imp)的访谈
    • 在你自己的仓库中定制 Poudriere 源
    • Wazuh 和 MITRE Caldera 在 FreeBSD Jail 中的使用
    • PEP 517
    • CCCamp 2023 旅行报告
  • 2023-0708 容器与云
    • 在 Firecracker 上的 FreeBSD
    • 使用 pot 和 nomad 管理 Jail
    • 会议报告:C 与 BSD 正如拉丁语与我们——一位神学家的旅程
    • 抒怀之旅:与 Doug Rabson 的访谈
    • 基于 Jail 的广告拦截教程
    • 我们收到的来信
  • 2023-0506 FreeBSD 三十周年纪念特刊
    • CheriBSD 近十多年的历程
    • AArch64:成为 FreeBSD 新的一级架构
    • 岁月如梭:我个人的时间线
    • 安装 FreeBSD 1.0:回顾 30 年前
    • ZFS 是如何进入 FreeBSD 的呢?
    • 我不是来自约克郡的,我保证!
    • 回忆录:采访 David Greenman Lawrence
    • FreeBSD 和早期的 Unix 社区
    • 早期的 FreeBSD 移植
    • FreeBSD 30 周年:成功的秘诀
    • FreeBSD 在日本:回忆之旅与今日之实
  • 2023-0304 嵌入式
    • CheriBSD port 和软件包
    • 让我们来试试 ChatGPT
    • GPU 直通
  • 2023-0102 构建 FreEBSD Web 服务器
    • ZFS 的原子 I/O 与 PostgreSQL
    • 虚拟实验室——BSD 编程研讨会
    • ZFS 简介
    • 会议报告:落基山庆祝女性计算机科学家
    • 进行中的工作/征求反馈:数据包批处理
    • 基金会与 FreeBSD 桌面
  • 2022-1112 可观测性和衡量标准
    • 在 FreeBSD 的 DDB 内核调试器中编写自定义命令
    • DTrace:老式跟踪系统的新扩展
    • 基于证书的 Icinga 监控
    • 活动监控脚本(activitymonitor.sh)
    • 实用 IPv6(第四部分)
    • EuroBSDCon 会议报道
    • 实用 Port:Prometheus 的安装与配置
    • 书评:《用火解决问题:管理老化的计算机系统(并为现代系统保驾护航)》Kill It with Fire: Manage Aging Computer Systems (and Future Proof Modern Ones)
  • 2022-0910 安全性
    • CARP 简介
    • 重构内核加密服务框架
    • PAM 小窍门
    • SSH 小窍门
    • 实用 IPv6(第三部分)
    • 书评:Understanding Software Dynamics(深入理解软件性能——一种动态视角)—— Richard L. Sites 著
    • 访谈:保障 FreeBSD 安全性
    • MCH 2022 会议报告
  • 2022-0708 科研、系统与 FreeBSD
    • 在 FreeBSD 上构建 Loom 框架
    • 教授本科生 Unix 课程
    • FreeBSD 入门研讨会
    • 实用 IPv6(第二部分)
    • 在 2022 年及以后推广 FreeBSD
    • 进行中的工作/征求反馈:Socket 缓冲区
    • FreeBSD 开发者峰会报告
    • 支持 Electromagnetic Field 2022
  • 2022-0506 灾难恢复
    • 使用 FreeBSD 构建高弹性的私有云
    • LLDB 14 —— FreeBSD 新调试器
    • 实用 IPv6(第一部分)
    • 利用 netdump(4) 进行事后内核调试
    • 进行中的工作/征求反馈:FreeBSD 启动性能
    • 实用 Port:在 OpenZFS 上设置 NFSv4 文件服务器
  • 2022-0304 ARM64 是一级架构
    • FreeBSD/ARM64 上的数据科学
    • Pinebook Pro 上的 FreeBSD
    • 嵌入式控制器的 ACPI 支持
    • 进行中的工作/征求反馈:Lumina 桌面征集开发人员
    • 实用 Port:如何设置 Apple 时间机器
  • 2022-0102 软件与系统管理
    • 为 FreeBSD Ports 做贡献
    • 使用 Git 贡献到 FreeBSD Ports
    • CBSD:第一部分——生产环境
    • 将 OpenBSD 的 pf syncookie 代码移植到 FreeBSD 的 pf
    • 进行中的工作/征求反馈:mkjail
    • 《编程智慧:编程鬼才的经验和思考》(The Kollected Kode Vicious)书评
    • 会议报告:EuroBSDCon 2021 我的第一次 EuroBSDCon:一位新组织者的视角
  • 2021-1112 存储
    • 开放通道 SSD
    • 构建 FreeBSD 社区
    • 与完美操作系统同行 27 年
    • 进行中的工作/征求反馈:OccamBSD
    • 通过 iSCSI 导入 ZFS ZIL——不要在工作中这样做——就像我做的那样
  • 2021-0910 FreeBSD 开发
    • FreeBSD 代码审查与 git-arc
    • 如何为 FreeBSD 实现简单的 USB 驱动程序
    • 内核开发技巧
    • 程序员编程杂谈
  • 2021-0708 桌面/无线网
    • 通往 FreeBSD 桌面的直线路径
    • FreeBSD 13 中的人机接口设备 (HID) 支持
    • Panfrost 驱动程序
    • 用 Git 更新 FreeBSD
    • FreeBSD 的新面孔
    • 想给你的桌面加点佐料?
  • 2021-0506 安全
    • 七种提升新安装 FreeBSD 安全性的方法
    • copyinout 框架
    • 使用 TLS 改善 NFS 安全性
    • Capsicum 案例研究:Got
    • 对 Jail 进行安全扫描
  • 2021-0304 FreeBSD 13.0
    • 展望未来
    • FreeBSD 13.0 工具链
    • FreeBSD 13.0 中有新加载器吗?
    • TCP Cubic 准备起飞
    • OpenZFS 中的 Zstandard 压缩
    • 会议报告:FreeBSD 供应商峰会
    • Git 不够吗?
  • 2021-0102 案例研究
    • Tarsnap 的 FreeBSD 集群
    • BALLY WULFF
    • Netflix Open Connect
    • FreeBSD 的新面孔
    • 写作学者的 FreeBSD
    • 在世界之巅
  • 2020-1112 工作流/持续集成(CI)
    • FreeBSD Git 快速入门
    • 使用 syzkaller 进行内核 Fuzzing
    • Mastering Vim Quickly 书评
    • 线上会议实用技巧
    • 在控制台上进行网络监控
  • 2020-0910 贡献与入门
    • 采访:Warner Losh,第 2 部分
    • 代码审查
    • 撰写良好的提交消息
    • 如何在不是程序员的情况下做出贡献——成为 FreeBSD 译者
    • 如何成为文档提交者
    • 谷歌编程之夏
    • 为 FreeBSD 期刊撰写文章
    • 你为什么使用 FreeBSD
    • FreeBSD 的新面孔
  • 2020-0708 基准测试/调优
    • FreeBSD Friday
    • 采访:Warner Losh,第 1 部分
    • 构建和运行开源社区
    • 在 FreeBSD 上轻松搭建我的世界(Minecraft)服务器
    • FreeBSD 的新面孔
  • 2020-0506 网络性能
    • 内核中的 TLS 卸载
    • 访谈:Michael W Lucas
    • FreeBSD 桌面发行版
    • 使用 Poudriere 进行 Port 批量管理
    • FreeBSD 的新面孔
由 GitBook 提供支持
LogoLogo

FreeBSD 中文社区(CFC) 2025

在本页
  • DDB 执行上下文
  • 命令函数
  • 自定义命令表
  • 支持分页的命令
  • 结论
在GitHub上编辑
导出为 PDF
  1. 2022-1112 可观测性和衡量标准

在 FreeBSD 的 DDB 内核调试器中编写自定义命令

上一页基金会与 FreeBSD 桌面下一页DTrace:老式跟踪系统的新扩展

最后更新于1个月前

  • 原文链接:

  • 作者:JOHN BALDWIN

DDB 是一款交互式内核调试器,可用于检查系统状态并控制正在运行的内核。DDB 最初作为 Mach 操作系统的一部分开发,后来被移植到 386BSD,并由包括 FreeBSD、NetBSD 以及 OpenBSD 在内的多个操作系统继承。本文重点介绍了 FreeBSD 中 DDB 的实现。

DDB 在系统控制台上运行,当调试器激活时,系统执行将被暂停,从而能在一致的状态下检查系统。虽然可以手动进入 DDB,但通常是在内核崩溃后使用。FreeBSD 的内核可以配置为在内核崩溃后自动进入 DDB,从而让用户和系统管理员在重启之前检查系统状态。在默认情况下,从 FreeBSD 主分支构建的调试内核即如此。

DDB 提供了许多调试器常见的功能。它支持诸如单步执行和断点等运行控制,并在硬件支持的平臺上支持硬件监视点。DDB 包含多个命令,用于显示系统信息,包括堆栈跟踪和内存转储。

与许多其他调试器不同,DDB 无法理解类型信息,也无法美化结构体的输出或在表达式中计算结构体或联合体成员。不过,DDB 可以通过定义新命令来扩展,新命令甚至可以在内核模块中实现,这些模块可以在启动后加载。

DDB 执行上下文

DDB 在一种与普通内核执行上下文有几处不同的特殊上下文中执行:

  • 当 DDB 激活时,系统处于暂停状态,并借用了每个 CPU 上当前执行线程的执行上下文。在此上下文中,正常的内核调度器不会运行,并且不允许借用的每个 CPU 上的线程进行上下文切换。这意味着在此上下文中的代码执行不得进入休眠状态或因等待锁而阻塞。

  • 如果在执行 DDB 命令期间发生故障或陷阱,当前线程将使用 longjmp() 返回到 DDB 的主循环继续执行。

  • DDB 直接访问控制台设备以实现控制台的输入和输出。

由于这些独特的行为,DDB 命令的实现应遵循以下准则:

  • 命令应避免副作用。 如果在命令执行期间发生错误,则无法撤销任何已产生的副作用。因此,最安全的方法是在可能的情况下避免副作用。

  • 命令不应使用锁。 由于所有 CPU 的执行均被暂停,系统中大多数数据结构的状态不会发生变化,因此不需要通过锁与其他 CPU 进行同步。此外,获取锁本身也是一种副作用,如果在持锁状态下命令出错,这种副作用无法回退。在特殊情况下,如果命令希望以安全的方式修改系统状态,可以使用尝试锁(try locks)。目前,使用这种方式的命令之一是 kill 命令,该命令可以向进程发送信号。

  • 命令应避免使用复杂的 API。 高级 API 往往会修改系统状态或包含其他副作用(例如获取锁)。

  • 大多数 DDB 命令仅检查系统状态而不进行修改,并输出系统某一部分状态的易读描述。 其中许多命令是用于美化输出的打印程序,它们会打印关于特定数据结构或数据结构列表的信息。

  • 命令必须使用 DDB 的 API 进行控制台输入和输出。 这主要意味着在输出时应使用 db_printf() 而不是 printf()。DDB 提供了一个简单的控制台输出 API。db_printf() 函数类似于普通内核中的 printf(),并支持所有相同的格式说明符。该函数直接写入控制台设备,绕过系统日志设备。此外,db_printf() 还包含简单的分页支持。每当向控制台输出一个换行符时,db_printf() 会检查是否应暂停输出。如果需要暂停,db_printf() 会在控制台上输出提示,允许用户控制在下一次暂停前显示多少行内容。待用户对提示作出响应,db_printf() 就会返回。如果用户请求当前命令退出(停止生成输出),db_printf() 会将全局变量 db_pager_quit 设置为非零值。如果命令在循环中生成输出(例如,使用循环遍历一个链表数据结构),则该命令应在每次循环迭代中检查 db_pager_quit,如果其被设置,则提前退出循环。

命令函数

大多数 DDB 命令遵循 ddb(4) 中描述的简单语法:

command[/modifier] [address[,count]]

当用户在 DDB 提示符下输入命令时,DDB 会解析该命令行。其中,地址(address)和计数(count)字段被视为表达式,可以包含对命名符号的引用以及许多 C 语言算术运算符;而命令(command)和修饰符(modifier)字段则被视为简单字符串。DDB 使用命令字段来查找一个 C 函数指针,并调用该 C 函数来执行命令。

实现 DDB 命令的函数使用如下签名:

void fn(db_expr_t addr, bool have_addr, db_expr_t count, char *modif)

这里的 addr 参数包含命令要操作的地址。该地址可以是用户显式提供的地址,也可以是上一次命令使用的地址; have_addr 参数为 true 时表示地址是显式提供的;count 参数包含计数字段的值。如果计数字段未指定,则 count 的值设为 -1;modif 参数是一个指向包含修饰符字段的 C 字符串的指针。如果未指定修饰符,则 modif 将指向一个空字符串。

命令函数通过 DDB 维护的内部表与命令名称关联。DDB 提供了一些辅助宏,用于抽象出注册新命令的大部分细节。每个宏接受两个参数:第一个参数是命令名称,第二个参数是与该命令关联的 C 函数名称。除了在表中注册链接之外,这些宏还提供了 C 函数的声明,并应紧跟着函数体。每个宏都与特定的命令表相关联。

  • DB_COMMAND 宏用于定义一个新的顶级命令;

  • DB_SHOW_COMMAND 宏用于在“show”表中定义一个新命令;

  • DB_SHOW_ALL_COMMAND 宏用于在“show all”表中定义一个新命令。

例如,DB_SHOW_COMMAND(bar, db_show_bar_func) 定义了一个新的“show bar”命令,同时也定义了一个新的 C 函数 db_show_bar_func,用于实现该命令。按照最佳实践(但不是强制要求),建议使用 db__cmd 这种命名模式来命名与命令关联的 C 函数。

清单 1 展示了一个名为“double”的简单命令的源码。该命令将用户提供的地址乘以 2,并输出结果。

清单 2 则展示了该命令的一些使用案例。第三个案例的输出可能会让人感到意外,因为 32 乘以 2 显然不等于 100。造成这种现象的原因是 DDB 解析整数值时默认使用 16 进制(这一默认进制由 DDB 内部的 $radix 变量控制)。在 16 进制中,32 的计算结果对应的十进制值为 50。

DB_COMMAND(double, db_double_cmd)
{
if (have_addr)
 db_printf("%u\n", (u_int)addr * 2);
else
 db_printf("no address\n");
}

清单 1:“double”命令的源代码

db> double
no address
db> double 4
8
db> double 32
100

清单 2: “double”命令的示例输出

具有自定义语法的命令

DDB 命令不必仅使用上述简单语法。命令函数可以选择支持其他语法。命令在注册时通过传递额外的标志来请求这种功能。另一组宏接受命令标志作为第三个参数,分别为:DB_COMMAND_FLAGS、DB_SHOW_COMMAND_FLAGS 和 DB_SHOW_ALL_COMMAND_FLAGS。

有两个标志可用于控制命令行解析。

  • CS_MORE 表示一个命令大体上遵循简单语法,但该命令支持多个地址。当指定此标志时,DDB 的主循环仍然会按正常方式解析命令行,但在调用命令函数之前不会丢弃词法分析器中剩余的所有标记。这允许命令函数在命令行中解析额外的选项。

  • 第二个标志 CS_OWN 表示命令函数将自行完成所有解析工作。当指定此标志时,DDB 的主循环在读取命令名称后就停止对命令行进行解析,接下来的解析由命令函数通过 DDB 的词法分析器完成。

无论指定哪种标志,命令函数在返回之前都必须调用 db_skip_to_eol() 来丢弃当前命令行中剩余的标记。

DDB 提供了一些函数来解析命令行参数:

  • db_expression() 用于解析算术表达式。它可以消耗多组输入,并支持完整的 DDB 表达式语法,包括符号解析以及各种 C 语言运算符。如果没有更多命令行参数可供解析,db_expression() 返回 0;如果表达式解析成功,则返回非零值,并将表达式的结果存储在其唯一参数所指向的变量中。如果 db_expression() 在解析表达式时遇到语法错误,它会打印一条信息,并通过 longjmp() 中止当前命令。命令函数在调用 db_expression() 时应避免任何副作用,因为如果用户提供无效输入,这些副作用将无法回退。

  • 另外还有两个函数提供了对 DDB 词法分析器的低级接口:

    • db_read_token() 解析命令行中的下一个标记,并返回一个常量,该常量标识所解析的标记类型。这些常量命名为 tXXX,并定义在相关头文件中。大多数常量与 C 语言运算符和其他特殊标记相关,但有几个常量对自定义命令很有用:

      • tEOL 在遇到命令行末尾时返回。

      • tEOF 用于表示无效输入,例如包含无效字符的数字。

      • tIDENT 在解析到单词(标识符)时返回。该单词的副本保存在全局变量 db_tok_string 中。

      • tNUMBER 在解析到数字时返回。该数值以整数形式保存在全局变量 db_tok_number 中。需要注意的是,DDB 的词法分析器假定任何以十进制数字开头的单词都是数字,而任何以字母、下划线或反斜杠开头的单词都是标识符。

    • db_unread_token() 将单个标记插入到下一次调用 db_read_token() 时返回。传递给 db_unread_token() 的值应为上述 t 常量之一。通常,该函数用于在 db_read_token() 返回了无效或意外标记时,将刚读取的标记放回去。

DDB 还提供了两个额外的函数来处理解析错误:

  • db_error() 会打印出调用者提供的消息,刷新词法分析器状态,并调用 longjmp() 来中止当前命令并返回到 DDB 的主循环。

  • db_flush_lex() 则仅刷新词法分析器状态,丢弃当前命令行。如果需要更详细的错误消息或在 longjmp() 不合适的情况下撤销额外状态,可以使用 db_flush_lex()。

清单 3 展示了一个名为“sum”的命令的源代码。该命令计算命令行上给出的所有表达式的和。它使用了标志 CS_MORE,并在循环中使用 db_expression() 来解析命令行中的其他表达式。

清单 4 显示了该命令的一些示例输出。请注意,在第三个示例中,db_expression() 解析了表达式“9 * 3”,并将值 27 返回给了 db_sum_cmd() 中的循环。

DB_COMMAND_FLAGS(sum, db_sum_cmd, CS_MORE)
{
long total;
db_expr_t value;
if (!have_addr)
 db_error("no values to sum\n");
total = addr;
while (db_expression(&value))
 total += value;
db_skip_to_eol();
db_printf("Total is %lu\n", total);
}

清单 3:“sum”命令的源代码

db> sum 1
Total is 1
db> sum 1 2 3
Total is 6
db> sum 9 * 3 4
Total is 31

清单 4: “sum”命令的示例输出

清单 5 包含了“show softc”命令的源代码。该命令接受设备名称作为单个命令行参数。如果找到该设备,命令将打印出指向该设备 softc 结构的指针值。该结构包含设备驱动程序维护的每个设备的信息。此命令使用 CS_OWN 标志以请求对命令行解析的完全控制。它使用 db_read_token() 从命令行中获取设备名称。如果给出了有效的设备名称,将返回一个 tIDENT 标记,并在 db_tok_string 中保存该设备名称。

清单 6 显示了此命令的一些示例输出。

DB_SHOW_COMMAND_FLAGS(softc, db_show_softc_cmd, CS_OWN)
{
device_t dev;
int token;
token = db_read_token();
if (token != tIDENT)
 db_error("Missing or invalid device name");
dev = device_lookup_by_name(db_tok_string);
db_skip_to_eol();
if (dev == NULL)
 db_error("device not found\n");
db_printf("%p\n", device_get_softc(dev));
}

清单 5:“show softc”命令的源代码

db> show softc 4
Missing or invalid device name
db> show softc foo0
device not found
db> show softc pci0
0xfffff800039380f0

清单 6: “show softc”命令的示例输出

自定义命令表

DDB 命令表包含一组命令。可以通过在现有表中定义一个特殊命令来创建额外的命令表,从而构建命令表的层级结构。

定义新表的命令不会指定用于处理该命令的函数。相反,表必须定义并初始化一个 struct db_command_table 类型的变量,该变量包含一个链接列表,用于存储属于该表的命令。然后,父表中的命令条目会关联到这个新表的指针。按照惯例,该变量的命名格式应为 db__table。

截至目前,还没有类似 DB_COMMAND 这样高度抽象的宏可用于定义新表。因此,定义新表必须使用“内部”宏 _DB_SET。属于该表的命令必须使用“内部”宏 _DB_FUNC 定义,或者通过定义一个类似 DB_SHOW_COMMAND 的辅助宏来封装 _DB_FUNC。

清单 7 包含了一个名为“demo”的命令表的源代码,以及属于该表的两个命令。

代码首先定义了 db_demo_table 变量,以存储属于该新表的 DDB 命令列表。 _DB_SET 的调用将“demo”命令添加到顶级命令表(类似于 DB_COMMAND)。需要注意的是,_DB_SET 的第三个参数(通常用于存放函数处理程序的指针)被设为 NULL,而最后一个参数指向新定义的表。

代码的其余部分定义了属于该新表的两个简单命令。 _DB_FUNC 的第二个和第三个参数类似于 DB_COMMAND 的两个参数。第四个参数用于指定新命令所属的父表。第五个参数用于标识诸如 CS_MORE 或 CS_OWN 之类的标志,最后一个参数应设为 NULL。

_DB_SET 和 _DB_FUNC 的第一个参数应为父表的名称,前面加上下划线,并将空格替换为下划线。如果父表是主命令表,则应使用 "_cmd"。

清单 8 显示了这些命令的一些示例输出。

/* 保存“demo *”命令的列表。*/
static struct db_command_table db_demo_table = LIST_HEAD_INITIALIZER(db_demo_table);
6 of 8
FreeBSD Journal • November/December 2022 11
/* 定义了一个“demo”顶级命令。 */
_DB_SET(_cmd, demo, NULL, db_cmd_table, 0, &db_demo_table);
_DB_FUNC(_demo, one, db_demo_one_cmd, db_demo_table, 0, NULL)
{
db_printf("one\n");
}
_DB_FUNC(_demo, two, db_demo_two_cmd, db_demo_table, 0, NULL)
{
db_printf("two\n");
}

清单 7:“deom”命令的源代码

db> demo
Subcommand required; available subcommands:
one two
db> demo one
one
db> demo two
two

支持分页的命令

我们的最后一个示例命令展示了如何遵循 DDB 的输出分页机制。

大多数分页操作(例如继续输出一页或单行)都由 db_printf() 内部的分页实现处理。然而,如果用户请求停止分页,db_pager_quit 这个全局变量会被设置为非零值(如前文所述)。因此,任何在循环中生成输出的命令都应检查该变量,并在其被设置时终止循环。

清单 9 提供了一个精简的示例命令,该命令会检查 db_pager_quit 的状态。该命令实现了 Internet 上的“chargen”(字符生成)服务,它会在屏幕上持续生成输出行,直到用户通过分页机制请求退出,从而终止循环。

这个示例的关键之处在于主循环的最后两行:如果 db_pager_quit 被设置,循环将立即退出。

DB_COMMAND(chargen, db_chargen_cmd)
{
char *rs;
int len;
for (rs = ring;;) {
 …
 db_printf("\n");
 if (db_pager_quit)

Break;
  }
}

清单 8:“chargen”命令的简略源代码

结论

DDB 提供了一个相对简单的框架,用于添加新命令。新命令甚至可以在系统引导后通过加载包含新命令的内核模块来添加。


John Baldwin 是一名系统软件开发人员。在过去二十多年里,他在 FreeBSD 操作系统的多个内核组件(包括 x86 平台支持、SMP、各类设备驱动程序以及虚拟内存子系统)和用户空间程序中直接提交了代码修改。除了编写代码,他还曾担任 FreeBSD 核心团队和发布工程团队的成员,并为 GDB 调试器做出了贡献。John 与他的妻子 Kimberly 及三个孩子 Janelle、Evan 和 Bella 现居住在加利福尼亚州康科德市。

FreeBSD 的源码树中包含许多自定义命令的示例,可作为开发新命令的参考。可以通过搜索 DB.*_COMMAND 或 db_printf 来查找这些示例。此外,一个包含本文所有命令的内核模块可以在以下地址找到:

Writing Custom Commands in FreeBSD’s DDB Kernel Debugger
https://github.com/bsdjhb/ddb_commands_demo