1#!/usr/bin/env python3 2# SPDX-License-Identifier: GPL-2.0 3# Copyright(c) 2026: Mauro Carvalho Chehab <mchehab@kernel.org>. 4 5import re 6 7from kdoc.kdoc_re import KernRe 8from kdoc.c_lex import CMatch, CTokenizer 9 10struct_args_pattern = r"([^,)]+)" 11 12 13class CTransforms: 14 """ 15 Data class containing a long set of transformations to turn 16 structure member prefixes, and macro invocations and variables 17 into something we can parse and generate kdoc for. 18 """ 19 20 # 21 # NOTE: 22 # Due to performance reasons, place CMatch rules before KernRe, 23 # as this avoids running the C parser every time. 24 # 25 26 #: Transforms for structs and unions. 27 struct_xforms = [ 28 (CMatch("__attribute__"), ""), 29 (CMatch("__aligned"), ""), 30 (CMatch("__counted_by"), ""), 31 (CMatch("__counted_by_(le|be)"), ""), 32 (CMatch("__counted_by_ptr"), ""), 33 (CMatch("__guarded_by"), ""), 34 (CMatch("__pt_guarded_by"), ""), 35 (CMatch("__packed"), ""), 36 (CMatch("CRYPTO_MINALIGN_ATTR"), ""), 37 (CMatch("__private"), ""), 38 (CMatch("__rcu"), ""), 39 (CMatch("____cacheline_aligned_in_smp"), ""), 40 (CMatch("____cacheline_aligned"), ""), 41 (CMatch("__cacheline_group_(?:begin|end)"), ""), 42 (CMatch("__ETHTOOL_DECLARE_LINK_MODE_MASK"), r"DECLARE_BITMAP(\1, __ETHTOOL_LINK_MODE_MASK_NBITS)"), 43 (CMatch("DECLARE_PHY_INTERFACE_MASK",),r"DECLARE_BITMAP(\1, PHY_INTERFACE_MODE_MAX)"), 44 (CMatch("DECLARE_BITMAP"), r"unsigned long \1[BITS_TO_LONGS(\2)]"), 45 (CMatch("DECLARE_HASHTABLE"), r"unsigned long \1[1 << ((\2) - 1)]"), 46 (CMatch("DECLARE_KFIFO"), r"\2 *\1"), 47 (CMatch("DECLARE_KFIFO_PTR"), r"\2 *\1"), 48 (CMatch("(?:__)?DECLARE_FLEX_ARRAY"), r"\1 \2[]"), 49 (CMatch("DEFINE_DMA_UNMAP_ADDR"), r"dma_addr_t \1"), 50 (CMatch("DEFINE_DMA_UNMAP_LEN"), r"__u32 \1"), 51 (CMatch("VIRTIO_DECLARE_FEATURES"), r"union { u64 \1; u64 \1_array[VIRTIO_FEATURES_U64S]; }"), 52 (CMatch("__cond_acquires"), ""), 53 (CMatch("__cond_releases"), ""), 54 (CMatch("__acquires"), ""), 55 (CMatch("__releases"), ""), 56 (CMatch("__must_hold"), ""), 57 (CMatch("__must_not_hold"), ""), 58 (CMatch("__must_hold_shared"), ""), 59 (CMatch("__cond_acquires_shared"), ""), 60 (CMatch("__acquires_shared"), ""), 61 (CMatch("__releases_shared"), ""), 62 (CMatch("__attribute__"), ""), 63 64 # 65 # Macro __struct_group() creates an union with an anonymous 66 # and a non-anonymous struct, depending on the parameters. We only 67 # need one of those at kernel-doc, as we won't be documenting the same 68 # members twice. 69 # 70 (CMatch("struct_group"), r"struct { \2+ };"), 71 (CMatch("struct_group_attr"), r"struct { \3+ };"), 72 (CMatch("struct_group_tagged"), r"struct { \3+ };"), 73 (CMatch("__struct_group"), r"struct { \4+ };"), 74 ] 75 76 #: Transforms for function prototypes. 77 function_xforms = [ 78 (CMatch("static"), ""), 79 (CMatch("extern"), ""), 80 (CMatch("asmlinkage"), ""), 81 (CMatch("inline"), ""), 82 (CMatch("__inline__"), ""), 83 (CMatch("__inline"), ""), 84 (CMatch("__always_inline"), ""), 85 (CMatch("noinline"), ""), 86 (CMatch("__FORTIFY_INLINE"), ""), 87 (CMatch("__init"), ""), 88 (CMatch("__init_or_module"), ""), 89 (CMatch("__exit"), ""), 90 (CMatch("__deprecated"), ""), 91 (CMatch("__flatten"), ""), 92 (CMatch("__meminit"), ""), 93 (CMatch("__must_check"), ""), 94 (CMatch("__weak"), ""), 95 (CMatch("__sched"), ""), 96 (CMatch("__always_unused"), ""), 97 (CMatch("__maybe_unused"), ""), 98 (CMatch("__printf"), ""), 99 (CMatch("__(?:re)?alloc_size"), ""), 100 (CMatch("__diagnose_as"), ""), 101 (CMatch("DECL_BUCKET_PARAMS"), r"\1, \2"), 102 (CMatch("__no_context_analysis"), ""), 103 (CMatch("__attribute_const__"), ""), 104 (CMatch("__attribute__"), ""), 105 106 # 107 # HACK: this is similar to process_export() hack. It is meant to 108 # drop _noproof from function name. See for instance: 109 # ahash_request_alloc kernel-doc declaration at include/crypto/hash.h. 110 # 111 (KernRe("_noprof"), ""), 112 ] 113 114 #: Transforms for variable prototypes. 115 var_xforms = [ 116 (CMatch("__read_mostly"), ""), 117 (CMatch("__ro_after_init"), ""), 118 (CMatch("__guarded_by"), ""), 119 (CMatch("__pt_guarded_by"), ""), 120 (CMatch("LIST_HEAD"), r"struct list_head \1"), 121 122 (KernRe(r"(?://.*)$"), ""), 123 (KernRe(r"(?:/\*.*\*/)"), ""), 124 (KernRe(r";$"), ""), 125 ] 126 127 #: Transforms main dictionary used at apply_transforms(). 128 xforms = { 129 "struct": struct_xforms, 130 "func": function_xforms, 131 "var": var_xforms, 132 } 133 134 def apply(self, xforms_type, source): 135 """ 136 Apply a set of transforms to a block of source. 137 138 As tokenizer is used here, this function also remove comments 139 at the end. 140 """ 141 if xforms_type not in self.xforms: 142 return source 143 144 if isinstance(source, str): 145 source = CTokenizer(source) 146 147 for search, subst in self.xforms[xforms_type]: 148 # 149 # KernRe only accept strings. 150 # 151 if isinstance(search, KernRe): 152 source = str(source) 153 154 source = search.sub(subst, source) 155 return str(source) 156