1 # SPDX-License-Identifier: GPL-2.0
2 # Copyright 2019 Jonathan Corbet <corbet@lwn.net>
4 # Apply kernel-specific tweaks after the initial document processing
7 from docutils
import nodes
9 from sphinx
import addnodes
10 if sphinx
.version_info
[0] < 2 or \
11 sphinx
.version_info
[0] == 2 and sphinx
.version_info
[1] < 1:
12 from sphinx
.environment
import NoUri
14 from sphinx
.errors
import NoUri
18 # Regex nastiness. Of course.
19 # Try to identify "function()" that's not already marked up some
20 # other way. Sphinx doesn't like a lot of stuff right after a
21 # :c:func: block (i.e. ":c:func:`mmap()`s" flakes out), so the last
22 # bit tries to restrict matches to things that won't create trouble.
24 RE_function
= re
.compile(r
'([\w_][\w\d_]+\(\))')
27 # Many places in the docs refer to common system calls. It is
28 # pointless to try to cross-reference them and, as has been known
29 # to happen, somebody defining a function by these names can lead
30 # to the creation of incorrect and confusing cross references. So
31 # just don't even try with these names.
33 Skipfuncs
= [ 'open', 'close', 'read', 'write', 'fcntl', 'mmap',
34 'select', 'poll', 'fork', 'execve', 'clone', 'ioctl',
38 # Find all occurrences of function() and try to replace them with
39 # appropriate cross references.
41 def markup_funcs(docname
, app
, node
):
42 cdom
= app
.env
.domains
['c']
46 for m
in RE_function
.finditer(t
):
48 # Include any text prior to function() as a normal text node.
51 repl
.append(nodes
.Text(t
[done
:m
.start()]))
53 # Go through the dance of getting an xref out of the C domain
55 target
= m
.group(1)[:-2]
56 target_text
= nodes
.Text(target
+ '()')
58 if target
not in Skipfuncs
:
59 lit_text
= nodes
.literal(classes
=['xref', 'c', 'c-func'])
60 lit_text
+= target_text
61 pxref
= addnodes
.pending_xref('', refdomain
= 'c',
63 reftarget
= target
, modname
= None,
66 # XXX The Latex builder will throw NoUri exceptions here,
67 # work around that by ignoring them.
70 xref
= cdom
.resolve_xref(app
.env
, docname
, app
.builder
,
71 'function', target
, pxref
, lit_text
)
75 # Toss the xref into the list if we got it; otherwise just put
81 repl
.append(target_text
)
84 repl
.append(nodes
.Text(t
[done
:]))
87 def auto_markup(app
, doctree
, name
):
89 # This loop could eventually be improved on. Someday maybe we
90 # want a proper tree traversal with a lot of awareness of which
91 # kinds of nodes to prune. But this works well for now.
93 # The nodes.literal test catches ``literal text``, its purpose is to
94 # avoid adding cross-references to functions that have been explicitly
95 # marked with cc:func:.
97 for para
in doctree
.traverse(nodes
.paragraph
):
98 for node
in para
.traverse(nodes
.Text
):
99 if not isinstance(node
.parent
, nodes
.literal
):
100 node
.parent
.replace(node
, markup_funcs(name
, app
, node
))
103 app
.connect('doctree-resolved', auto_markup
)
105 'parallel_read_safe': True,
106 'parallel_write_safe': True,