mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-08-05 16:54:27 +00:00

We no longer support Python 2 in the docs build chain at all, so we certainly do not need to admonish folks to keep this file working with it. Cc: Jani Nikula <jani.nikula@intel.com> Reviewed-by: Mauro Carvalho Chehab <mchehab+huawei@kernel.org> Acked-by: Jani Nikula <jani.nikula@intel.com> Tested-by: Akira Yokosawa <akiyks@gmail.com> Signed-off-by: Jonathan Corbet <corbet@lwn.net> Link: https://lore.kernel.org/r/20250703184403.274408-7-corbet@lwn.net
313 lines
9.8 KiB
Python
313 lines
9.8 KiB
Python
# coding=utf-8
|
|
# SPDX-License-Identifier: MIT
|
|
#
|
|
# Copyright © 2016 Intel Corporation
|
|
#
|
|
# Permission is hereby granted, free of charge, to any person obtaining a
|
|
# copy of this software and associated documentation files (the "Software"),
|
|
# to deal in the Software without restriction, including without limitation
|
|
# the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
# and/or sell copies of the Software, and to permit persons to whom the
|
|
# Software is furnished to do so, subject to the following conditions:
|
|
#
|
|
# The above copyright notice and this permission notice (including the next
|
|
# paragraph) shall be included in all copies or substantial portions of the
|
|
# Software.
|
|
#
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
|
# IN THE SOFTWARE.
|
|
#
|
|
# Authors:
|
|
# Jani Nikula <jani.nikula@intel.com>
|
|
#
|
|
|
|
import codecs
|
|
import os
|
|
import subprocess
|
|
import sys
|
|
import re
|
|
import glob
|
|
|
|
from docutils import nodes, statemachine
|
|
from docutils.statemachine import ViewList
|
|
from docutils.parsers.rst import directives, Directive
|
|
import sphinx
|
|
from sphinx.util.docutils import switch_source_input
|
|
from sphinx.util import logging
|
|
from pprint import pformat
|
|
|
|
srctree = os.path.abspath(os.environ["srctree"])
|
|
sys.path.insert(0, os.path.join(srctree, "scripts/lib/kdoc"))
|
|
|
|
from kdoc_files import KernelFiles
|
|
from kdoc_output import RestFormat
|
|
|
|
__version__ = '1.0'
|
|
kfiles = None
|
|
logger = logging.getLogger(__name__)
|
|
|
|
def cmd_str(cmd):
|
|
"""
|
|
Helper function to output a command line that can be used to produce
|
|
the same records via command line. Helpful to debug troubles at the
|
|
script.
|
|
"""
|
|
|
|
cmd_line = ""
|
|
|
|
for w in cmd:
|
|
if w == "" or " " in w:
|
|
esc_cmd = "'" + w + "'"
|
|
else:
|
|
esc_cmd = w
|
|
|
|
if cmd_line:
|
|
cmd_line += " " + esc_cmd
|
|
continue
|
|
else:
|
|
cmd_line = esc_cmd
|
|
|
|
return cmd_line
|
|
|
|
class KernelDocDirective(Directive):
|
|
"""Extract kernel-doc comments from the specified file"""
|
|
required_argument = 1
|
|
optional_arguments = 4
|
|
option_spec = {
|
|
'doc': directives.unchanged_required,
|
|
'export': directives.unchanged,
|
|
'internal': directives.unchanged,
|
|
'identifiers': directives.unchanged,
|
|
'no-identifiers': directives.unchanged,
|
|
'functions': directives.unchanged,
|
|
}
|
|
has_content = False
|
|
verbose = 0
|
|
|
|
parse_args = {}
|
|
msg_args = {}
|
|
|
|
def handle_args(self):
|
|
|
|
env = self.state.document.settings.env
|
|
cmd = [env.config.kerneldoc_bin, '-rst', '-enable-lineno']
|
|
|
|
filename = env.config.kerneldoc_srctree + '/' + self.arguments[0]
|
|
|
|
# Arguments used by KernelFiles.parse() function
|
|
self.parse_args = {
|
|
"file_list": [filename],
|
|
"export_file": []
|
|
}
|
|
|
|
# Arguments used by KernelFiles.msg() function
|
|
self.msg_args = {
|
|
"enable_lineno": True,
|
|
"export": False,
|
|
"internal": False,
|
|
"symbol": [],
|
|
"nosymbol": [],
|
|
"no_doc_sections": False
|
|
}
|
|
|
|
export_file_patterns = []
|
|
|
|
verbose = os.environ.get("V")
|
|
if verbose:
|
|
try:
|
|
self.verbose = int(verbose)
|
|
except ValueError:
|
|
pass
|
|
|
|
# Tell sphinx of the dependency
|
|
env.note_dependency(os.path.abspath(filename))
|
|
|
|
self.tab_width = self.options.get('tab-width',
|
|
self.state.document.settings.tab_width)
|
|
|
|
# 'function' is an alias of 'identifiers'
|
|
if 'functions' in self.options:
|
|
self.options['identifiers'] = self.options.get('functions')
|
|
|
|
# FIXME: make this nicer and more robust against errors
|
|
if 'export' in self.options:
|
|
cmd += ['-export']
|
|
self.msg_args["export"] = True
|
|
export_file_patterns = str(self.options.get('export')).split()
|
|
elif 'internal' in self.options:
|
|
cmd += ['-internal']
|
|
self.msg_args["internal"] = True
|
|
export_file_patterns = str(self.options.get('internal')).split()
|
|
elif 'doc' in self.options:
|
|
func = str(self.options.get('doc'))
|
|
cmd += ['-function', func]
|
|
self.msg_args["symbol"].append(func)
|
|
elif 'identifiers' in self.options:
|
|
identifiers = self.options.get('identifiers').split()
|
|
if identifiers:
|
|
for i in identifiers:
|
|
i = i.rstrip("\\").strip()
|
|
if not i:
|
|
continue
|
|
|
|
cmd += ['-function', i]
|
|
self.msg_args["symbol"].append(i)
|
|
else:
|
|
cmd += ['-no-doc-sections']
|
|
self.msg_args["no_doc_sections"] = True
|
|
|
|
if 'no-identifiers' in self.options:
|
|
no_identifiers = self.options.get('no-identifiers').split()
|
|
if no_identifiers:
|
|
for i in no_identifiers:
|
|
i = i.rstrip("\\").strip()
|
|
if not i:
|
|
continue
|
|
|
|
cmd += ['-nosymbol', i]
|
|
self.msg_args["nosymbol"].append(i)
|
|
|
|
for pattern in export_file_patterns:
|
|
pattern = pattern.rstrip("\\").strip()
|
|
if not pattern:
|
|
continue
|
|
|
|
for f in glob.glob(env.config.kerneldoc_srctree + '/' + pattern):
|
|
env.note_dependency(os.path.abspath(f))
|
|
cmd += ['-export-file', f]
|
|
self.parse_args["export_file"].append(f)
|
|
|
|
# Export file is needed by both parse and msg, as kernel-doc
|
|
# cache exports.
|
|
self.msg_args["export_file"] = self.parse_args["export_file"]
|
|
|
|
cmd += [filename]
|
|
|
|
return cmd
|
|
|
|
def run_cmd(self, cmd):
|
|
"""
|
|
Execute an external kernel-doc command.
|
|
"""
|
|
|
|
env = self.state.document.settings.env
|
|
node = nodes.section()
|
|
|
|
p = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
|
out, err = p.communicate()
|
|
|
|
out, err = codecs.decode(out, 'utf-8'), codecs.decode(err, 'utf-8')
|
|
|
|
if p.returncode != 0:
|
|
sys.stderr.write(err)
|
|
|
|
logger.warning("kernel-doc '%s' failed with return code %d"
|
|
% (" ".join(cmd), p.returncode))
|
|
return [nodes.error(None, nodes.paragraph(text = "kernel-doc missing"))]
|
|
elif env.config.kerneldoc_verbosity > 0:
|
|
sys.stderr.write(err)
|
|
|
|
filenames = self.parse_args["file_list"]
|
|
for filename in filenames:
|
|
self.parse_msg(filename, node, out, cmd)
|
|
|
|
return node.children
|
|
|
|
def parse_msg(self, filename, node, out, cmd):
|
|
"""
|
|
Handles a kernel-doc output for a given file
|
|
"""
|
|
|
|
env = self.state.document.settings.env
|
|
|
|
lines = statemachine.string2lines(out, self.tab_width,
|
|
convert_whitespace=True)
|
|
result = ViewList()
|
|
|
|
lineoffset = 0;
|
|
line_regex = re.compile(r"^\.\. LINENO ([0-9]+)$")
|
|
for line in lines:
|
|
match = line_regex.search(line)
|
|
if match:
|
|
# sphinx counts lines from 0
|
|
lineoffset = int(match.group(1)) - 1
|
|
# we must eat our comments since the upset the markup
|
|
else:
|
|
doc = str(env.srcdir) + "/" + env.docname + ":" + str(self.lineno)
|
|
result.append(line, doc + ": " + filename, lineoffset)
|
|
lineoffset += 1
|
|
|
|
self.do_parse(result, node)
|
|
|
|
def run_kdoc(self, cmd, kfiles):
|
|
"""
|
|
Execute kernel-doc classes directly instead of running as a separate
|
|
command.
|
|
"""
|
|
|
|
env = self.state.document.settings.env
|
|
|
|
node = nodes.section()
|
|
|
|
kfiles.parse(**self.parse_args)
|
|
filenames = self.parse_args["file_list"]
|
|
|
|
for filename, out in kfiles.msg(**self.msg_args, filenames=filenames):
|
|
self.parse_msg(filename, node, out, cmd)
|
|
|
|
return node.children
|
|
|
|
def run(self):
|
|
global kfiles
|
|
|
|
cmd = self.handle_args()
|
|
if self.verbose >= 1:
|
|
logger.info(cmd_str(cmd))
|
|
|
|
try:
|
|
if kfiles:
|
|
return self.run_kdoc(cmd, kfiles)
|
|
else:
|
|
return self.run_cmd(cmd)
|
|
|
|
except Exception as e: # pylint: disable=W0703
|
|
logger.warning("kernel-doc '%s' processing failed with: %s" %
|
|
(cmd_str(cmd), pformat(e)))
|
|
return [nodes.error(None, nodes.paragraph(text = "kernel-doc missing"))]
|
|
|
|
def do_parse(self, result, node):
|
|
with switch_source_input(self.state, result):
|
|
self.state.nested_parse(result, 0, node, match_titles=1)
|
|
|
|
def setup_kfiles(app):
|
|
global kfiles
|
|
|
|
kerneldoc_bin = app.env.config.kerneldoc_bin
|
|
|
|
if kerneldoc_bin and kerneldoc_bin.endswith("kernel-doc.py"):
|
|
print("Using Python kernel-doc")
|
|
out_style = RestFormat()
|
|
kfiles = KernelFiles(out_style=out_style, logger=logger)
|
|
else:
|
|
print(f"Using {kerneldoc_bin}")
|
|
|
|
|
|
def setup(app):
|
|
app.add_config_value('kerneldoc_bin', None, 'env')
|
|
app.add_config_value('kerneldoc_srctree', None, 'env')
|
|
app.add_config_value('kerneldoc_verbosity', 1, 'env')
|
|
|
|
app.add_directive('kernel-doc', KernelDocDirective)
|
|
|
|
app.connect('builder-inited', setup_kfiles)
|
|
|
|
return dict(
|
|
version = __version__,
|
|
parallel_read_safe = True,
|
|
parallel_write_safe = True
|
|
)
|