xref: /linux/tools/docs/kernel-doc (revision eba6ffd126cd52358181ed5a179644a161f9c65f)
1*eba6ffd1SJonathan Corbet#!/usr/bin/env python3
2*eba6ffd1SJonathan Corbet# SPDX-License-Identifier: GPL-2.0
3*eba6ffd1SJonathan Corbet# Copyright(c) 2025: Mauro Carvalho Chehab <mchehab@kernel.org>.
4*eba6ffd1SJonathan Corbet#
5*eba6ffd1SJonathan Corbet# pylint: disable=C0103,R0912,R0914,R0915
6*eba6ffd1SJonathan Corbet#
7*eba6ffd1SJonathan Corbet# NOTE: While kernel-doc requires at least version 3.6 to run, the
8*eba6ffd1SJonathan Corbet#       command line should work with Python 3.2+ (tested with 3.4).
9*eba6ffd1SJonathan Corbet#       The rationale is that it shall fail gracefully during Kernel
10*eba6ffd1SJonathan Corbet#       compilation with older Kernel versions. Due to that:
11*eba6ffd1SJonathan Corbet#       - encoding line is needed here;
12*eba6ffd1SJonathan Corbet#       - f-strings cannot be used in this file.
13*eba6ffd1SJonathan Corbet#       - libraries that require newer versions can only be included
14*eba6ffd1SJonathan Corbet#         after the Python version has been checked.
15*eba6ffd1SJonathan Corbet#
16*eba6ffd1SJonathan Corbet# Converted from the kernel-doc script originally written in Perl
17*eba6ffd1SJonathan Corbet# under GPLv2, copyrighted since 1998 by the following authors:
18*eba6ffd1SJonathan Corbet#
19*eba6ffd1SJonathan Corbet#    Aditya Srivastava <yashsri421@gmail.com>
20*eba6ffd1SJonathan Corbet#    Akira Yokosawa <akiyks@gmail.com>
21*eba6ffd1SJonathan Corbet#    Alexander A. Klimov <grandmaster@al2klimov.de>
22*eba6ffd1SJonathan Corbet#    Alexander Lobakin <aleksander.lobakin@intel.com>
23*eba6ffd1SJonathan Corbet#    André Almeida <andrealmeid@igalia.com>
24*eba6ffd1SJonathan Corbet#    Andy Shevchenko <andriy.shevchenko@linux.intel.com>
25*eba6ffd1SJonathan Corbet#    Anna-Maria Behnsen <anna-maria@linutronix.de>
26*eba6ffd1SJonathan Corbet#    Armin Kuster <akuster@mvista.com>
27*eba6ffd1SJonathan Corbet#    Bart Van Assche <bart.vanassche@sandisk.com>
28*eba6ffd1SJonathan Corbet#    Ben Hutchings <ben@decadent.org.uk>
29*eba6ffd1SJonathan Corbet#    Borislav Petkov <bbpetkov@yahoo.de>
30*eba6ffd1SJonathan Corbet#    Chen-Yu Tsai <wenst@chromium.org>
31*eba6ffd1SJonathan Corbet#    Coco Li <lixiaoyan@google.com>
32*eba6ffd1SJonathan Corbet#    Conchúr Navid <conchur@web.de>
33*eba6ffd1SJonathan Corbet#    Daniel Santos <daniel.santos@pobox.com>
34*eba6ffd1SJonathan Corbet#    Danilo Cesar Lemes de Paula <danilo.cesar@collabora.co.uk>
35*eba6ffd1SJonathan Corbet#    Dan Luedtke <mail@danrl.de>
36*eba6ffd1SJonathan Corbet#    Donald Hunter <donald.hunter@gmail.com>
37*eba6ffd1SJonathan Corbet#    Gabriel Krisman Bertazi <krisman@collabora.co.uk>
38*eba6ffd1SJonathan Corbet#    Greg Kroah-Hartman <gregkh@linuxfoundation.org>
39*eba6ffd1SJonathan Corbet#    Harvey Harrison <harvey.harrison@gmail.com>
40*eba6ffd1SJonathan Corbet#    Horia Geanta <horia.geanta@freescale.com>
41*eba6ffd1SJonathan Corbet#    Ilya Dryomov <idryomov@gmail.com>
42*eba6ffd1SJonathan Corbet#    Jakub Kicinski <kuba@kernel.org>
43*eba6ffd1SJonathan Corbet#    Jani Nikula <jani.nikula@intel.com>
44*eba6ffd1SJonathan Corbet#    Jason Baron <jbaron@redhat.com>
45*eba6ffd1SJonathan Corbet#    Jason Gunthorpe <jgg@nvidia.com>
46*eba6ffd1SJonathan Corbet#    Jérémy Bobbio <lunar@debian.org>
47*eba6ffd1SJonathan Corbet#    Johannes Berg <johannes.berg@intel.com>
48*eba6ffd1SJonathan Corbet#    Johannes Weiner <hannes@cmpxchg.org>
49*eba6ffd1SJonathan Corbet#    Jonathan Cameron <Jonathan.Cameron@huawei.com>
50*eba6ffd1SJonathan Corbet#    Jonathan Corbet <corbet@lwn.net>
51*eba6ffd1SJonathan Corbet#    Jonathan Neuschäfer <j.neuschaefer@gmx.net>
52*eba6ffd1SJonathan Corbet#    Kamil Rytarowski <n54@gmx.com>
53*eba6ffd1SJonathan Corbet#    Kees Cook <kees@kernel.org>
54*eba6ffd1SJonathan Corbet#    Laurent Pinchart <laurent.pinchart@ideasonboard.com>
55*eba6ffd1SJonathan Corbet#    Levin, Alexander (Sasha Levin) <alexander.levin@verizon.com>
56*eba6ffd1SJonathan Corbet#    Linus Torvalds <torvalds@linux-foundation.org>
57*eba6ffd1SJonathan Corbet#    Lucas De Marchi <lucas.demarchi@profusion.mobi>
58*eba6ffd1SJonathan Corbet#    Mark Rutland <mark.rutland@arm.com>
59*eba6ffd1SJonathan Corbet#    Markus Heiser <markus.heiser@darmarit.de>
60*eba6ffd1SJonathan Corbet#    Martin Waitz <tali@admingilde.org>
61*eba6ffd1SJonathan Corbet#    Masahiro Yamada <masahiroy@kernel.org>
62*eba6ffd1SJonathan Corbet#    Matthew Wilcox <willy@infradead.org>
63*eba6ffd1SJonathan Corbet#    Mauro Carvalho Chehab <mchehab+huawei@kernel.org>
64*eba6ffd1SJonathan Corbet#    Michal Wajdeczko <michal.wajdeczko@intel.com>
65*eba6ffd1SJonathan Corbet#    Michael Zucchi
66*eba6ffd1SJonathan Corbet#    Mike Rapoport <rppt@linux.ibm.com>
67*eba6ffd1SJonathan Corbet#    Niklas Söderlund <niklas.soderlund@corigine.com>
68*eba6ffd1SJonathan Corbet#    Nishanth Menon <nm@ti.com>
69*eba6ffd1SJonathan Corbet#    Paolo Bonzini <pbonzini@redhat.com>
70*eba6ffd1SJonathan Corbet#    Pavan Kumar Linga <pavan.kumar.linga@intel.com>
71*eba6ffd1SJonathan Corbet#    Pavel Pisa <pisa@cmp.felk.cvut.cz>
72*eba6ffd1SJonathan Corbet#    Peter Maydell <peter.maydell@linaro.org>
73*eba6ffd1SJonathan Corbet#    Pierre-Louis Bossart <pierre-louis.bossart@linux.intel.com>
74*eba6ffd1SJonathan Corbet#    Randy Dunlap <rdunlap@infradead.org>
75*eba6ffd1SJonathan Corbet#    Richard Kennedy <richard@rsk.demon.co.uk>
76*eba6ffd1SJonathan Corbet#    Rich Walker <rw@shadow.org.uk>
77*eba6ffd1SJonathan Corbet#    Rolf Eike Beer <eike-kernel@sf-tec.de>
78*eba6ffd1SJonathan Corbet#    Sakari Ailus <sakari.ailus@linux.intel.com>
79*eba6ffd1SJonathan Corbet#    Silvio Fricke <silvio.fricke@gmail.com>
80*eba6ffd1SJonathan Corbet#    Simon Huggins
81*eba6ffd1SJonathan Corbet#    Tim Waugh <twaugh@redhat.com>
82*eba6ffd1SJonathan Corbet#    Tomasz Warniełło <tomasz.warniello@gmail.com>
83*eba6ffd1SJonathan Corbet#    Utkarsh Tripathi <utripathi2002@gmail.com>
84*eba6ffd1SJonathan Corbet#    valdis.kletnieks@vt.edu <valdis.kletnieks@vt.edu>
85*eba6ffd1SJonathan Corbet#    Vegard Nossum <vegard.nossum@oracle.com>
86*eba6ffd1SJonathan Corbet#    Will Deacon <will.deacon@arm.com>
87*eba6ffd1SJonathan Corbet#    Yacine Belkadi <yacine.belkadi.1@gmail.com>
88*eba6ffd1SJonathan Corbet#    Yujie Liu <yujie.liu@intel.com>
89*eba6ffd1SJonathan Corbet
90*eba6ffd1SJonathan Corbet"""
91*eba6ffd1SJonathan CorbetPrint formatted kernel documentation to stdout.
92*eba6ffd1SJonathan Corbet
93*eba6ffd1SJonathan CorbetRead C language source or header FILEs, extract embedded
94*eba6ffd1SJonathan Corbetdocumentation comments, and print formatted documentation
95*eba6ffd1SJonathan Corbetto standard output.
96*eba6ffd1SJonathan Corbet
97*eba6ffd1SJonathan CorbetThe documentation comments are identified by the ``/**``
98*eba6ffd1SJonathan Corbetopening comment mark.
99*eba6ffd1SJonathan Corbet
100*eba6ffd1SJonathan CorbetSee Documentation/doc-guide/kernel-doc.rst for the
101*eba6ffd1SJonathan Corbetdocumentation comment syntax.
102*eba6ffd1SJonathan Corbet"""
103*eba6ffd1SJonathan Corbet
104*eba6ffd1SJonathan Corbetimport argparse
105*eba6ffd1SJonathan Corbetimport logging
106*eba6ffd1SJonathan Corbetimport os
107*eba6ffd1SJonathan Corbetimport sys
108*eba6ffd1SJonathan Corbet
109*eba6ffd1SJonathan Corbet# Import Python modules
110*eba6ffd1SJonathan Corbet
111*eba6ffd1SJonathan CorbetLIB_DIR = "../lib/python"
112*eba6ffd1SJonathan CorbetSRC_DIR = os.path.dirname(os.path.realpath(__file__))
113*eba6ffd1SJonathan Corbet
114*eba6ffd1SJonathan Corbetsys.path.insert(0, os.path.join(SRC_DIR, LIB_DIR))
115*eba6ffd1SJonathan Corbet
116*eba6ffd1SJonathan CorbetWERROR_RETURN_CODE = 3
117*eba6ffd1SJonathan Corbet
118*eba6ffd1SJonathan CorbetDESC = """
119*eba6ffd1SJonathan CorbetRead C language source or header FILEs, extract embedded documentation comments,
120*eba6ffd1SJonathan Corbetand print formatted documentation to standard output.
121*eba6ffd1SJonathan Corbet
122*eba6ffd1SJonathan CorbetThe documentation comments are identified by the "/**" opening comment mark.
123*eba6ffd1SJonathan Corbet
124*eba6ffd1SJonathan CorbetSee Documentation/doc-guide/kernel-doc.rst for the documentation comment syntax.
125*eba6ffd1SJonathan Corbet"""
126*eba6ffd1SJonathan Corbet
127*eba6ffd1SJonathan CorbetEXPORT_FILE_DESC = """
128*eba6ffd1SJonathan CorbetSpecify an additional FILE in which to look for EXPORT_SYMBOL information.
129*eba6ffd1SJonathan Corbet
130*eba6ffd1SJonathan CorbetMay be used multiple times.
131*eba6ffd1SJonathan Corbet"""
132*eba6ffd1SJonathan Corbet
133*eba6ffd1SJonathan CorbetEXPORT_DESC = """
134*eba6ffd1SJonathan CorbetOnly output documentation for symbols that have been
135*eba6ffd1SJonathan Corbetexported using EXPORT_SYMBOL() and related macros in any input
136*eba6ffd1SJonathan CorbetFILE or -export-file FILE.
137*eba6ffd1SJonathan Corbet"""
138*eba6ffd1SJonathan Corbet
139*eba6ffd1SJonathan CorbetINTERNAL_DESC = """
140*eba6ffd1SJonathan CorbetOnly output documentation for symbols that have NOT been
141*eba6ffd1SJonathan Corbetexported using EXPORT_SYMBOL() and related macros in any input
142*eba6ffd1SJonathan CorbetFILE or -export-file FILE.
143*eba6ffd1SJonathan Corbet"""
144*eba6ffd1SJonathan Corbet
145*eba6ffd1SJonathan CorbetFUNCTION_DESC = """
146*eba6ffd1SJonathan CorbetOnly output documentation for the given function or DOC: section
147*eba6ffd1SJonathan Corbettitle. All other functions and DOC: sections are ignored.
148*eba6ffd1SJonathan Corbet
149*eba6ffd1SJonathan CorbetMay be used multiple times.
150*eba6ffd1SJonathan Corbet"""
151*eba6ffd1SJonathan Corbet
152*eba6ffd1SJonathan CorbetNOSYMBOL_DESC = """
153*eba6ffd1SJonathan CorbetExclude the specified symbol from the output documentation.
154*eba6ffd1SJonathan Corbet
155*eba6ffd1SJonathan CorbetMay be used multiple times.
156*eba6ffd1SJonathan Corbet"""
157*eba6ffd1SJonathan Corbet
158*eba6ffd1SJonathan CorbetFILES_DESC = """
159*eba6ffd1SJonathan CorbetHeader and C source files to be parsed.
160*eba6ffd1SJonathan Corbet"""
161*eba6ffd1SJonathan Corbet
162*eba6ffd1SJonathan CorbetWARN_CONTENTS_BEFORE_SECTIONS_DESC = """
163*eba6ffd1SJonathan CorbetWarn if there are contents before sections (deprecated).
164*eba6ffd1SJonathan Corbet
165*eba6ffd1SJonathan CorbetThis option is kept just for backward-compatibility, but it does nothing,
166*eba6ffd1SJonathan Corbetneither here nor at the original Perl script.
167*eba6ffd1SJonathan Corbet"""
168*eba6ffd1SJonathan Corbet
169*eba6ffd1SJonathan Corbet
170*eba6ffd1SJonathan Corbetclass MsgFormatter(logging.Formatter):
171*eba6ffd1SJonathan Corbet    """Helper class to format warnings in a similar way to kernel-doc.pl."""
172*eba6ffd1SJonathan Corbet
173*eba6ffd1SJonathan Corbet    def format(self, record):
174*eba6ffd1SJonathan Corbet        record.levelname = record.levelname.capitalize()
175*eba6ffd1SJonathan Corbet        return logging.Formatter.format(self, record)
176*eba6ffd1SJonathan Corbet
177*eba6ffd1SJonathan Corbetdef main():
178*eba6ffd1SJonathan Corbet    """
179*eba6ffd1SJonathan Corbet    Main program.
180*eba6ffd1SJonathan Corbet
181*eba6ffd1SJonathan Corbet    By default, the return value is:
182*eba6ffd1SJonathan Corbet
183*eba6ffd1SJonathan Corbet    - 0: success or Python version is not compatible with
184*eba6ffd1SJonathan Corbet      kernel-doc.  If -Werror is not used, it will also
185*eba6ffd1SJonathan Corbet      return 0 if there are issues at kernel-doc markups;
186*eba6ffd1SJonathan Corbet
187*eba6ffd1SJonathan Corbet    - 1: an abnormal condition happened;
188*eba6ffd1SJonathan Corbet
189*eba6ffd1SJonathan Corbet    - 2: argparse issued an error;
190*eba6ffd1SJonathan Corbet
191*eba6ffd1SJonathan Corbet    - 3: -Werror is used, and one or more unfiltered parse warnings happened.
192*eba6ffd1SJonathan Corbet    """
193*eba6ffd1SJonathan Corbet
194*eba6ffd1SJonathan Corbet    parser = argparse.ArgumentParser(formatter_class=argparse.RawTextHelpFormatter,
195*eba6ffd1SJonathan Corbet                                     description=DESC)
196*eba6ffd1SJonathan Corbet
197*eba6ffd1SJonathan Corbet    #
198*eba6ffd1SJonathan Corbet    # Normal arguments
199*eba6ffd1SJonathan Corbet    #
200*eba6ffd1SJonathan Corbet    parser.add_argument("-v", "-verbose", "--verbose", action="store_true",
201*eba6ffd1SJonathan Corbet                        help="Verbose output, more warnings and other information.")
202*eba6ffd1SJonathan Corbet
203*eba6ffd1SJonathan Corbet    parser.add_argument("-d", "-debug", "--debug", action="store_true",
204*eba6ffd1SJonathan Corbet                        help="Enable debug messages")
205*eba6ffd1SJonathan Corbet
206*eba6ffd1SJonathan Corbet    parser.add_argument("-M", "-modulename", "--modulename",
207*eba6ffd1SJonathan Corbet                        default="Kernel API",
208*eba6ffd1SJonathan Corbet                        help="Allow setting a module name at the output.")
209*eba6ffd1SJonathan Corbet
210*eba6ffd1SJonathan Corbet    parser.add_argument("-l", "-enable-lineno", "--enable_lineno",
211*eba6ffd1SJonathan Corbet                        action="store_true",
212*eba6ffd1SJonathan Corbet                        help="Enable line number output (only in ReST mode)")
213*eba6ffd1SJonathan Corbet
214*eba6ffd1SJonathan Corbet    #
215*eba6ffd1SJonathan Corbet    # Arguments to control the warning behavior
216*eba6ffd1SJonathan Corbet    #
217*eba6ffd1SJonathan Corbet    parser.add_argument("-Wreturn", "--wreturn", action="store_true",
218*eba6ffd1SJonathan Corbet                        help="Warns about the lack of a return markup on functions.")
219*eba6ffd1SJonathan Corbet
220*eba6ffd1SJonathan Corbet    parser.add_argument("-Wshort-desc", "-Wshort-description", "--wshort-desc",
221*eba6ffd1SJonathan Corbet                        action="store_true",
222*eba6ffd1SJonathan Corbet                        help="Warns if initial short description is missing")
223*eba6ffd1SJonathan Corbet
224*eba6ffd1SJonathan Corbet    parser.add_argument("-Wcontents-before-sections",
225*eba6ffd1SJonathan Corbet                        "--wcontents-before-sections", action="store_true",
226*eba6ffd1SJonathan Corbet                        help=WARN_CONTENTS_BEFORE_SECTIONS_DESC)
227*eba6ffd1SJonathan Corbet
228*eba6ffd1SJonathan Corbet    parser.add_argument("-Wall", "--wall", action="store_true",
229*eba6ffd1SJonathan Corbet                        help="Enable all types of warnings")
230*eba6ffd1SJonathan Corbet
231*eba6ffd1SJonathan Corbet    parser.add_argument("-Werror", "--werror", action="store_true",
232*eba6ffd1SJonathan Corbet                        help="Treat warnings as errors.")
233*eba6ffd1SJonathan Corbet
234*eba6ffd1SJonathan Corbet    parser.add_argument("-export-file", "--export-file", action='append',
235*eba6ffd1SJonathan Corbet                        help=EXPORT_FILE_DESC)
236*eba6ffd1SJonathan Corbet
237*eba6ffd1SJonathan Corbet    #
238*eba6ffd1SJonathan Corbet    # Output format mutually-exclusive group
239*eba6ffd1SJonathan Corbet    #
240*eba6ffd1SJonathan Corbet    out_group = parser.add_argument_group("Output format selection (mutually exclusive)")
241*eba6ffd1SJonathan Corbet
242*eba6ffd1SJonathan Corbet    out_fmt = out_group.add_mutually_exclusive_group()
243*eba6ffd1SJonathan Corbet
244*eba6ffd1SJonathan Corbet    out_fmt.add_argument("-m", "-man", "--man", action="store_true",
245*eba6ffd1SJonathan Corbet                         help="Output troff manual page format.")
246*eba6ffd1SJonathan Corbet    out_fmt.add_argument("-r", "-rst", "--rst", action="store_true",
247*eba6ffd1SJonathan Corbet                         help="Output reStructuredText format (default).")
248*eba6ffd1SJonathan Corbet    out_fmt.add_argument("-N", "-none", "--none", action="store_true",
249*eba6ffd1SJonathan Corbet                         help="Do not output documentation, only warnings.")
250*eba6ffd1SJonathan Corbet
251*eba6ffd1SJonathan Corbet    #
252*eba6ffd1SJonathan Corbet    # Output selection mutually-exclusive group
253*eba6ffd1SJonathan Corbet    #
254*eba6ffd1SJonathan Corbet    sel_group = parser.add_argument_group("Output selection (mutually exclusive)")
255*eba6ffd1SJonathan Corbet    sel_mut = sel_group.add_mutually_exclusive_group()
256*eba6ffd1SJonathan Corbet
257*eba6ffd1SJonathan Corbet    sel_mut.add_argument("-e", "-export", "--export", action='store_true',
258*eba6ffd1SJonathan Corbet                         help=EXPORT_DESC)
259*eba6ffd1SJonathan Corbet
260*eba6ffd1SJonathan Corbet    sel_mut.add_argument("-i", "-internal", "--internal", action='store_true',
261*eba6ffd1SJonathan Corbet                         help=INTERNAL_DESC)
262*eba6ffd1SJonathan Corbet
263*eba6ffd1SJonathan Corbet    sel_mut.add_argument("-s", "-function", "--symbol", action='append',
264*eba6ffd1SJonathan Corbet                         help=FUNCTION_DESC)
265*eba6ffd1SJonathan Corbet
266*eba6ffd1SJonathan Corbet    #
267*eba6ffd1SJonathan Corbet    # Those are valid for all 3 types of filter
268*eba6ffd1SJonathan Corbet    #
269*eba6ffd1SJonathan Corbet    parser.add_argument("-n", "-nosymbol", "--nosymbol", action='append',
270*eba6ffd1SJonathan Corbet                        help=NOSYMBOL_DESC)
271*eba6ffd1SJonathan Corbet
272*eba6ffd1SJonathan Corbet    parser.add_argument("-D", "-no-doc-sections", "--no-doc-sections",
273*eba6ffd1SJonathan Corbet                        action='store_true', help="Don't output DOC sections")
274*eba6ffd1SJonathan Corbet
275*eba6ffd1SJonathan Corbet    parser.add_argument("files", metavar="FILE",
276*eba6ffd1SJonathan Corbet                        nargs="+", help=FILES_DESC)
277*eba6ffd1SJonathan Corbet
278*eba6ffd1SJonathan Corbet    args = parser.parse_args()
279*eba6ffd1SJonathan Corbet
280*eba6ffd1SJonathan Corbet    if args.wall:
281*eba6ffd1SJonathan Corbet        args.wreturn = True
282*eba6ffd1SJonathan Corbet        args.wshort_desc = True
283*eba6ffd1SJonathan Corbet        args.wcontents_before_sections = True
284*eba6ffd1SJonathan Corbet
285*eba6ffd1SJonathan Corbet    logger = logging.getLogger()
286*eba6ffd1SJonathan Corbet
287*eba6ffd1SJonathan Corbet    if not args.debug:
288*eba6ffd1SJonathan Corbet        logger.setLevel(logging.INFO)
289*eba6ffd1SJonathan Corbet    else:
290*eba6ffd1SJonathan Corbet        logger.setLevel(logging.DEBUG)
291*eba6ffd1SJonathan Corbet
292*eba6ffd1SJonathan Corbet    formatter = MsgFormatter('%(levelname)s: %(message)s')
293*eba6ffd1SJonathan Corbet
294*eba6ffd1SJonathan Corbet    handler = logging.StreamHandler()
295*eba6ffd1SJonathan Corbet    handler.setFormatter(formatter)
296*eba6ffd1SJonathan Corbet
297*eba6ffd1SJonathan Corbet    logger.addHandler(handler)
298*eba6ffd1SJonathan Corbet
299*eba6ffd1SJonathan Corbet    python_ver = sys.version_info[:2]
300*eba6ffd1SJonathan Corbet    if python_ver < (3,6):
301*eba6ffd1SJonathan Corbet        #
302*eba6ffd1SJonathan Corbet        # Depending on the Kernel configuration, kernel-doc --none is called at
303*eba6ffd1SJonathan Corbet        # build time. As we don't want to break compilation due to the
304*eba6ffd1SJonathan Corbet        # usage of an old Python version, return 0 here.
305*eba6ffd1SJonathan Corbet        #
306*eba6ffd1SJonathan Corbet        if args.none:
307*eba6ffd1SJonathan Corbet            logger.error("Python 3.6 or later is required by kernel-doc. Skipping checks")
308*eba6ffd1SJonathan Corbet            sys.exit(0)
309*eba6ffd1SJonathan Corbet
310*eba6ffd1SJonathan Corbet        sys.exit("Python 3.6 or later is required by kernel-doc. Aborting.")
311*eba6ffd1SJonathan Corbet
312*eba6ffd1SJonathan Corbet    if python_ver < (3,7):
313*eba6ffd1SJonathan Corbet        logger.warning("Python 3.7 or later is required for correct results")
314*eba6ffd1SJonathan Corbet
315*eba6ffd1SJonathan Corbet    #
316*eba6ffd1SJonathan Corbet    # Import kernel-doc libraries only after checking the Python version
317*eba6ffd1SJonathan Corbet    #
318*eba6ffd1SJonathan Corbet    from kdoc.kdoc_files import KernelFiles             # pylint: disable=C0415
319*eba6ffd1SJonathan Corbet    from kdoc.kdoc_output import RestFormat, ManFormat  # pylint: disable=C0415
320*eba6ffd1SJonathan Corbet
321*eba6ffd1SJonathan Corbet    if args.man:
322*eba6ffd1SJonathan Corbet        out_style = ManFormat(modulename=args.modulename)
323*eba6ffd1SJonathan Corbet    elif args.none:
324*eba6ffd1SJonathan Corbet        out_style = None
325*eba6ffd1SJonathan Corbet    else:
326*eba6ffd1SJonathan Corbet        out_style = RestFormat()
327*eba6ffd1SJonathan Corbet
328*eba6ffd1SJonathan Corbet    kfiles = KernelFiles(verbose=args.verbose,
329*eba6ffd1SJonathan Corbet                         out_style=out_style, werror=args.werror,
330*eba6ffd1SJonathan Corbet                         wreturn=args.wreturn, wshort_desc=args.wshort_desc,
331*eba6ffd1SJonathan Corbet                         wcontents_before_sections=args.wcontents_before_sections)
332*eba6ffd1SJonathan Corbet
333*eba6ffd1SJonathan Corbet    kfiles.parse(args.files, export_file=args.export_file)
334*eba6ffd1SJonathan Corbet
335*eba6ffd1SJonathan Corbet    for t in kfiles.msg(enable_lineno=args.enable_lineno, export=args.export,
336*eba6ffd1SJonathan Corbet                        internal=args.internal, symbol=args.symbol,
337*eba6ffd1SJonathan Corbet                        nosymbol=args.nosymbol, export_file=args.export_file,
338*eba6ffd1SJonathan Corbet                        no_doc_sections=args.no_doc_sections):
339*eba6ffd1SJonathan Corbet        msg = t[1]
340*eba6ffd1SJonathan Corbet        if msg:
341*eba6ffd1SJonathan Corbet            print(msg)
342*eba6ffd1SJonathan Corbet
343*eba6ffd1SJonathan Corbet    error_count = kfiles.errors
344*eba6ffd1SJonathan Corbet    if not error_count:
345*eba6ffd1SJonathan Corbet        sys.exit(0)
346*eba6ffd1SJonathan Corbet
347*eba6ffd1SJonathan Corbet    if args.werror:
348*eba6ffd1SJonathan Corbet        print("%s warnings as errors" % error_count)    # pylint: disable=C0209
349*eba6ffd1SJonathan Corbet        sys.exit(WERROR_RETURN_CODE)
350*eba6ffd1SJonathan Corbet
351*eba6ffd1SJonathan Corbet    if args.verbose:
352*eba6ffd1SJonathan Corbet        print("%s errors" % error_count)                # pylint: disable=C0209
353*eba6ffd1SJonathan Corbet
354*eba6ffd1SJonathan Corbet    sys.exit(0)
355*eba6ffd1SJonathan Corbet
356*eba6ffd1SJonathan Corbet#
357*eba6ffd1SJonathan Corbet# Call main method
358*eba6ffd1SJonathan Corbet#
359*eba6ffd1SJonathan Corbetif __name__ == "__main__":
360*eba6ffd1SJonathan Corbet    main()
361