0001 .. include:: ../disclaimer-zh_CN.rst
0002
0003 :Original: Documentation/doc-guide/kernel-doc.rst
0004
0005 :译者: 吴想成 Wu XiangCheng <bobwxc@email.cn>
0006
0007 编写kernel-doc注释
0008 ==================
0009
0010 Linux内核源文件可以包含kernel-doc格式的结构化文档注释,用以描述代码的函数、
0011 类型和设计。将文档嵌入源文件更容易保持文档最新。
0012
0013 .. note:: 内核文档格式与javadoc、gtk-doc或Doxygen看似很相似,但由于历史原因,
0014 实际有着明显的不同。内核源包含成千上万个kernel-doc注释。请坚持遵循
0015 此处描述的风格。
0016
0017 .. note:: kernel-doc无法包含Rust代码:请参考 Documentation/rust/general-information.rst 。
0018
0019 从注释中提取kernel-doc结构,并从中生成适当的 `Sphinx C 域`_ 函数和带有锚点的
0020 类型描述。这些注释将被过滤以生成特殊kernel-doc高亮和交叉引用。详见下文。
0021
0022 .. _Sphinx C 域: http://www.sphinx-doc.org/en/stable/domains.html
0023
0024 使用 ``EXPORT_SYMBOL`` 或 ``EXPORT_SYMBOL_GPL`` 导出到可加载模块的每个函数都
0025 应该有一个kernel-doc注释。模块使用的头文件中的函数和数据结构也应该有
0026 kernel-doc注释。
0027
0028 对于其他内核文件(未标记为 ``static`` )中外部可见的函数,提供kernel-doc格式
0029 的文档是一个很好的实践。我们也建议为私有(文件 ``static`` )程序提供kernel-doc
0030 格式的文档,以确保内核源代码布局的一致性。此建议优先级较低,由内核源文件的
0031 维护者自行决定。
0032
0033 如何格式化kernel-doc注释
0034 ------------------------
0035
0036 kernel-doc注释用 ``/**`` 作为开始标记。 ``kernel-doc`` 工具将提取以这种方式
0037 标记的注释。注释其余部分的格式类似于一个普通的多行注释,左侧有一列星号,以
0038 ``*/`` 行结束。
0039
0040 函数和类型的kernel-doc注释应该放在所描述的函数或类型之前,以便最大限度地提高
0041 更改代码的人同时更改文档的可能性。概述kernel-doc注释可以放在最顶部的任何地方。
0042
0043 用详细模式和不生成实际输出来运行 ``kernel-doc`` 工具,可以验证文档注释的格式
0044 是否正确。例如::
0045
0046 scripts/kernel-doc -v -none drivers/foo/bar.c
0047
0048 当请求执行额外的gcc检查时,内核构建将验证文档格式::
0049
0050 make W=n
0051
0052 函数文档
0053 --------
0054
0055 函数和函数式宏的kernel-doc注释的一般格式是::
0056
0057 /**
0058 * 函数名() - 函数简要说明.
0059 * @参数1: 描述第一个参数.
0060 * @参数2: 描述第二个参数.
0061 * 可以为参数提供一段
0062 * 多行描述.
0063 *
0064 * 更详细的描述,进一步讨论函数 函数名(), 这可能对使用或修改它的人有用.
0065 * 以空注释行开始, 内部可以包含空注释行.
0066 *
0067 * 详细描述可以有多个段落.
0068 *
0069 * Context: 描述函数是否可以休眠, 它需要、释放或期望持有什么锁.
0070 * 可以写多行.
0071 * Return: 描述函数返回值.
0072 *
0073 * 返回值描述也可以有多个段落,
0074 * 并且应该放在注释块的末尾.
0075 */
0076
0077 函数名后面的简短描述可以跨多行,并以参数描述、空注释行或注释块结尾结束。
0078
0079 函数参数
0080 ~~~~~~~~
0081
0082 每个函数参数都应该按照顺序描述,紧跟在函数简要说明之后。不要在函数描述和参数
0083 之间,也不要在参数之间留空。
0084
0085 每个 ``@参数:`` 描述可以跨多行。
0086
0087 .. note::
0088
0089 如果 ``@参数`` 描述有多行,则说明的续行应该从上一行的同一列开始::
0090
0091 * @参数: 较长说明
0092 * 的续行
0093
0094 或::
0095
0096 * @参数:
0097 * 较长说明
0098 * 的续行
0099
0100 如果函数的参数数目可变,则需用kernel-doc格式对其进行描述::
0101
0102 * @...: 描述
0103
0104 函数上下文
0105 ~~~~~~~~~~
0106
0107 可调用函数的上下文应该在 ``Context`` 节中描述。此节应该包括函数是休眠的还是
0108 可以从中断上下文调用的,以及它需要什么锁、释放什么锁和期望它的调用者持有什么
0109 锁。
0110
0111 例如::
0112
0113 * Context: Any context.
0114 * Context: Any context. Takes and releases the RCU lock.
0115 * Context: Any context. Expects <lock> to be held by caller.
0116 * Context: Process context. May sleep if @gfp flags permit.
0117 * Context: Process context. Takes and releases <mutex>.
0118 * Context: Softirq or process context. Takes and releases <lock>, BH-safe.
0119 * Context: Interrupt context.
0120
0121 返回值
0122 ~~~~~~
0123
0124 如有返回值,应在 ``Return`` 节中描述。
0125
0126 .. note::
0127
0128 #) 您提供的多行描述文本 *不会* 识别换行符,因此如果您想将某些文本预格式化,
0129 如::
0130
0131 * Return:
0132 * 0 - OK
0133 * -EINVAL - invalid argument
0134 * -ENOMEM - out of memory
0135
0136 它们在最终文档中变成一行::
0137
0138 Return: 0 - OK -EINVAL - invalid argument -ENOMEM - out of memory
0139
0140 因此,为了在需要的地方换行,您需要使用ReST列表,例如::
0141
0142 * Return:
0143 * * 0 - OK to runtime suspend the device
0144 * * -EBUSY - Device should not be runtime suspended
0145
0146 #) 如果您提供的描述性文本中的行以某个后跟冒号的短语开头,则每一个这种短语
0147 都将被视为新的节标题,可能会产生意料不到的效果。
0148
0149 结构体、共用体、枚举类型文档
0150 ----------------------------
0151
0152 结构体(struct)、共用体(union)、枚举(enum)类型kernel-doc注释的一般格式为::
0153
0154 /**
0155 * struct 结构体名 - 简要描述.
0156 * @成员1: 成员1描述.
0157 * @成员2: 成员2描述.
0158 * 可以为成员提供
0159 * 多行描述.
0160 *
0161 * 结构体的描述.
0162 */
0163
0164 可以用 ``union`` 或 ``enum`` 替换上面示例中的 ``struct`` ,以描述共用体或枚举。
0165 ``成员`` 用于表示枚举中的元素或共用体成员。
0166
0167 结构体名称后面的简要说明可以跨多行,并以成员说明、空白注释行或注释块结尾结束。
0168
0169 成员
0170 ~~~~
0171
0172 结构体、共用体和枚举的成员应以与函数参数相同的方式记录;它们后紧跟简短的描述,
0173 并且为多行。
0174
0175 在结构体或共用体描述中,可以使用 ``private:`` 和 ``public:`` 注释标签。
0176 ``private:`` 域内的字段不会列在生成的文档中。
0177
0178 ``private:`` 和 ``public:`` 标签必须紧跟在 ``/*`` 注释标记之后。可以选择是否
0179 在 ``:`` 和 ``*/`` 结束标记之间包含注释。
0180
0181 例子::
0182
0183 /**
0184 * struct 张三 - 简短描述
0185 * @a: 第一个成员
0186 * @b: 第二个成员
0187 * @d: 第三个成员
0188 *
0189 * 详细描述
0190 */
0191 struct 张三 {
0192 int a;
0193 int b;
0194 /* private: 仅内部使用 */
0195 int c;
0196 /* public: 下一个是公有的 */
0197 int d;
0198 };
0199
0200 嵌套的结构体/共用体
0201 ~~~~~~~~~~~~~~~~~~~
0202
0203 嵌套的结构体/共用体可像这样记录::
0204
0205 /**
0206 * struct nested_foobar - a struct with nested unions and structs
0207 * @memb1: first member of anonymous union/anonymous struct
0208 * @memb2: second member of anonymous union/anonymous struct
0209 * @memb3: third member of anonymous union/anonymous struct
0210 * @memb4: fourth member of anonymous union/anonymous struct
0211 * @bar: non-anonymous union
0212 * @bar.st1: struct st1 inside @bar
0213 * @bar.st2: struct st2 inside @bar
0214 * @bar.st1.memb1: first member of struct st1 on union bar
0215 * @bar.st1.memb2: second member of struct st1 on union bar
0216 * @bar.st2.memb1: first member of struct st2 on union bar
0217 * @bar.st2.memb2: second member of struct st2 on union bar
0218 */
0219 struct nested_foobar {
0220 /* Anonymous union/struct*/
0221 union {
0222 struct {
0223 int memb1;
0224 int memb2;
0225 };
0226 struct {
0227 void *memb3;
0228 int memb4;
0229 };
0230 };
0231 union {
0232 struct {
0233 int memb1;
0234 int memb2;
0235 } st1;
0236 struct {
0237 void *memb1;
0238 int memb2;
0239 } st2;
0240 } bar;
0241 };
0242
0243 .. note::
0244
0245 #) 在记录嵌套结构体或共用体时,如果结构体/共用体 ``张三`` 已命名,则其中
0246 的成员 ``李四`` 应记录为 ``@张三.李四:``
0247
0248 #) 当嵌套结构体/共用体是匿名的时,其中的成员 ``李四`` 应记录为 ``@李四:``
0249
0250 行间注释文档
0251 ~~~~~~~~~~~~
0252
0253 结构成员也可在定义时以行间注释形式记录。有两种样式,一种是单行注释,其中开始
0254 ``/**`` 和结束 ``*/`` 位于同一行;另一种是多行注释,开头结尾各自位于一行,就
0255 像所有其他核心文档注释一样::
0256
0257 /**
0258 * struct 张三 - 简短描述.
0259 * @张三: 成员张三.
0260 */
0261 struct 张三 {
0262 int 张三;
0263 /**
0264 * @李四: 成员李四.
0265 */
0266 int 李四;
0267 /**
0268 * @王五: 成员王五.
0269 *
0270 * 此处,成员描述可以为好几段.
0271 */
0272 int 王五;
0273 union {
0274 /** @儿子: 单行描述. */
0275 int 儿子;
0276 };
0277 /** @赵六: 描述@张三里面的结构体@赵六 */
0278 struct {
0279 /**
0280 * @赵六.女儿: 描述@张三.赵六里面的@女儿
0281 */
0282 int 女儿;
0283 } 赵六;
0284 };
0285
0286 Typedef文档
0287 -----------
0288
0289 Typedef的kernel-doc文档注释的一般格式为::
0290
0291 /**
0292 * typedef 类型名称 - 简短描述.
0293 *
0294 * 类型描述.
0295 */
0296
0297 还可以记录带有函数原型的typedef::
0298
0299 /**
0300 * typedef 类型名称 - 简短描述.
0301 * @参数1: 参数1的描述
0302 * @参数2: 参数2的描述
0303 *
0304 * 类型描述.
0305 *
0306 * Context: 锁(Locking)上下文.
0307 * Return: 返回值的意义.
0308 */
0309 typedef void (*类型名称)(struct v4l2_ctrl *参数1, void *参数2);
0310
0311 高亮与交叉引用
0312 --------------
0313
0314 在kernel-doc注释的描述文本中可以识别以下特殊模式,并将其转换为正确的
0315 reStructuredText标记和 `Sphinx C 域`_ 引用。
0316
0317 .. attention:: 以下内容 **仅** 在kernel-doc注释中识别, **不会** 在普通的
0318 reStructuredText文档中识别。
0319
0320 ``funcname()``
0321 函数引用。
0322
0323 ``@parameter``
0324 函数参数的名称(未交叉引用,仅格式化)。
0325
0326 ``%CONST``
0327 常量的名称(未交叉引用,仅格式化)。
0328
0329 ````literal````
0330 预格式化文本块。输出将使用等距字体。
0331
0332 若你需要使用在kernel-doc脚本或reStructuredText中有特殊含义的字符,则此功能
0333 非常有用。
0334
0335 若你需要在函数描述中使用类似于 ``%ph`` 的东西,这特别有用。
0336
0337 ``$ENVVAR``
0338 环境变量名称(未交叉引用,仅格式化)。
0339
0340 ``&struct name``
0341 结构体引用。
0342
0343 ``&enum name``
0344 枚举引用。
0345
0346 ``&typedef name``
0347 Typedef引用。
0348
0349 ``&struct_name->member`` or ``&struct_name.member``
0350 结构体或共用体成员引用。交叉引用将链接到结构体或共用体定义,而不是直接到成员。
0351
0352 ``&name``
0353 泛类型引用。请首选上面描述的完整引用方式。此法主要是为了可能未描述的注释。
0354
0355 从reStructuredText交叉引用
0356 ~~~~~~~~~~~~~~~~~~~~~~~~~~
0357
0358 无需额外的语法来从reStructuredText文档交叉引用kernel-do注释中定义的函数和类型。
0359 只需以 ``()`` 结束函数名,并在类型之前写上 ``struct`` , ``union`` , ``enum``
0360 或 ``typedef`` 。
0361 例如::
0362
0363 See foo().
0364 See struct foo.
0365 See union bar.
0366 See enum baz.
0367 See typedef meh.
0368
0369 若要在交叉引用链接中使用自定义文本,可以通过以下语法进行::
0370
0371 See :c:func:`my custom link text for function foo <foo>`.
0372 See :c:type:`my custom link text for struct bar <bar>`.
0373
0374 有关更多详细信息,请参阅 `Sphinx C 域`_ 文档。
0375
0376 总述性文档注释
0377 --------------
0378
0379 为了促进源代码和注释紧密联合,可以将kernel-doc文档块作为自由形式的注释,而
0380 不是函数、结构、联合、枚举或typedef的绑定kernel-doc。例如,这可以用于解释
0381 驱动程序或库代码的操作理论。
0382
0383 这是通过使用带有节标题的 ``DOC:`` 节关键字来实现的。
0384
0385 总述或高层级文档注释的一般格式为::
0386
0387 /**
0388 * DOC: Theory of Operation
0389 *
0390 * The whizbang foobar is a dilly of a gizmo. It can do whatever you
0391 * want it to do, at any time. It reads your mind. Here's how it works.
0392 *
0393 * foo bar splat
0394 *
0395 * The only drawback to this gizmo is that is can sometimes damage
0396 * hardware, software, or its subject(s).
0397 */
0398
0399 ``DOC:`` 后面的标题用作源文件中的标题,但也用作提取文档注释的标识符。因此,
0400 文件中的标题必须是唯一的。
0401
0402 包含kernel-doc注释
0403 ==================
0404
0405 文档注释可以被包含在任何使用专用kernel-doc Sphinx指令扩展的reStructuredText
0406 文档中。
0407
0408 kernel-doc指令的格式如下::
0409
0410 .. kernel-doc:: source
0411 :option:
0412
0413 *source* 是相对于内核源代码树的源文件路径。
0414 支持以下指令选项:
0415
0416 export: *[source-pattern ...]*
0417 包括 *source* 中使用 ``EXPORT_SYMBOL`` 或 ``EXPORT_SYMBOL_GPL`` 导出的所有
0418 函数的文档,无论是在 *source* 中还是在 *source-pattern* 指定的任何文件中。
0419
0420 当kernel-doc注释被放置在头文件中,而 ``EXPORT_SYMBOL`` 和 ``EXPORT_SYMBOL_GPL``
0421 位于函数定义旁边时, *source-pattern* 非常有用。
0422
0423 例子::
0424
0425 .. kernel-doc:: lib/bitmap.c
0426 :export:
0427
0428 .. kernel-doc:: include/net/mac80211.h
0429 :export: net/mac80211/*.c
0430
0431 internal: *[source-pattern ...]*
0432 包括 *source* 中所有在 *source* 或 *source-pattern* 的任何文件中都没有使用
0433 ``EXPORT_SYMBOL`` 或 ``EXPORT_SYMBOL_GPL`` 导出的函数和类型的文档。
0434
0435 例子::
0436
0437 .. kernel-doc:: drivers/gpu/drm/i915/intel_audio.c
0438 :internal:
0439
0440 identifiers: *[ function/type ...]*
0441 在 *source* 中包含每个 *function* 和 *type* 的文档。如果没有指定 *function* ,
0442 则 *source* 中所有函数和类型的文档都将包含在内。
0443
0444 例子::
0445
0446 .. kernel-doc:: lib/bitmap.c
0447 :identifiers: bitmap_parselist bitmap_parselist_user
0448
0449 .. kernel-doc:: lib/idr.c
0450 :identifiers:
0451
0452 no-identifiers: *[ function/type ...]*
0453 排除 *source* 中所有 *function* 和 *type* 的文档。
0454
0455 例子::
0456
0457 .. kernel-doc:: lib/bitmap.c
0458 :no-identifiers: bitmap_parselist
0459
0460 functions: *[ function/type ...]*
0461 这是“identifiers”指令的别名,已弃用。
0462
0463 doc: *title*
0464 包含 *source* 中由 *title* 标题标识的 ``DOC:`` 文档段落。 *title* 中允许
0465 空格;不要在 *title* 上加引号。 *title* 仅用作段落的标识符,不包含在输出中。
0466 请确保在所附的reStructuredText文档中有适当的标题。
0467
0468 例子::
0469
0470 .. kernel-doc:: drivers/gpu/drm/i915/intel_audio.c
0471 :doc: High Definition Audio over HDMI and Display Port
0472
0473 如果没有选项,kernel-doc指令将包含源文件中的所有文档注释。
0474
0475 kernel-doc扩展包含在内核源代码树中,位于 ``Documentation/sphinx/kerneldoc.py`` 。
0476 在内部,它使用 ``scripts/kernel-doc`` 脚本从源代码中提取文档注释。
0477
0478 .. _kernel_doc_zh:
0479
0480 如何使用kernel-doc生成手册(man)页
0481 -----------------------------------
0482
0483 如果您只想使用kernel-doc生成手册页,可以从内核git树这样做::
0484
0485 $ scripts/kernel-doc -man \
0486 $(git grep -l '/\*\*' -- :^Documentation :^tools) \
0487 | scripts/split-man.pl /tmp/man
0488
0489 一些旧版本的git不支持路径排除语法的某些变体。
0490 以下命令之一可能适用于这些版本::
0491
0492 $ scripts/kernel-doc -man \
0493 $(git grep -l '/\*\*' -- . ':!Documentation' ':!tools') \
0494 | scripts/split-man.pl /tmp/man
0495
0496 $ scripts/kernel-doc -man \
0497 $(git grep -l '/\*\*' -- . ":(exclude)Documentation" ":(exclude)tools") \
0498 | scripts/split-man.pl /tmp/man
0499