Blame view

Documentation/sphinx/cdomain.py 5.46 KB
81f7e3824   Eric Lee   Initial Release, ...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
  # -*- coding: utf-8; mode: python -*-
  # pylint: disable=W0141,C0113,C0103,C0325
  u"""
      cdomain
      ~~~~~~~
  
      Replacement for the sphinx c-domain.
  
      :copyright:  Copyright (C) 2016  Markus Heiser
      :license:    GPL Version 2, June 1991 see Linux/COPYING for details.
  
      List of customizations:
  
      * Moved the *duplicate C object description* warnings for function
        declarations in the nitpicky mode. See Sphinx documentation for
        the config values for ``nitpick`` and ``nitpick_ignore``.
  
      * Add option 'name' to the "c:function:" directive.  With option 'name' the
        ref-name of a function can be modified. E.g.::
  
            .. c:function:: int ioctl( int fd, int request )
               :name: VIDIOC_LOG_STATUS
  
        The func-name (e.g. ioctl) remains in the output but the ref-name changed
        from 'ioctl' to 'VIDIOC_LOG_STATUS'. The function is referenced by::
  
            * :c:func:`VIDIOC_LOG_STATUS` or
            * :any:`VIDIOC_LOG_STATUS` (``:any:`` needs sphinx 1.3)
  
       * Handle signatures of function-like macros well. Don't try to deduce
         arguments types of function-like macros.
  
  """
  
  from docutils import nodes
  from docutils.parsers.rst import directives
  
  import sphinx
  from sphinx import addnodes
  from sphinx.domains.c import c_funcptr_sig_re, c_sig_re
  from sphinx.domains.c import CObject as Base_CObject
  from sphinx.domains.c import CDomain as Base_CDomain
  
  __version__  = '1.0'
  
  # Get Sphinx version
  major, minor, patch = sphinx.version_info[:3]
  
  def setup(app):
  
      app.override_domain(CDomain)
  
      return dict(
          version = __version__,
          parallel_read_safe = True,
          parallel_write_safe = True
      )
  
  class CObject(Base_CObject):
  
      """
      Description of a C language object.
      """
      option_spec = {
          "name" : directives.unchanged
      }
  
      def handle_func_like_macro(self, sig, signode):
          u"""Handles signatures of function-like macros.
  
          If the objtype is 'function' and the the signature ``sig`` is a
          function-like macro, the name of the macro is returned. Otherwise
          ``False`` is returned.  """
  
          if not self.objtype == 'function':
              return False
  
          m = c_funcptr_sig_re.match(sig)
          if m is None:
              m = c_sig_re.match(sig)
              if m is None:
                  raise ValueError('no match')
  
          rettype, fullname, arglist, _const = m.groups()
          arglist = arglist.strip()
          if rettype or not arglist:
              return False
  
          arglist = arglist.replace('`', '').replace('\\ ', '') # remove markup
          arglist = [a.strip() for a in arglist.split(",")]
  
          # has the first argument a type?
          if len(arglist[0].split(" ")) > 1:
              return False
  
          # This is a function-like macro, it's arguments are typeless!
          signode  += addnodes.desc_name(fullname, fullname)
          paramlist = addnodes.desc_parameterlist()
          signode  += paramlist
  
          for argname in arglist:
              param = addnodes.desc_parameter('', '', noemph=True)
              # separate by non-breaking space in the output
              param += nodes.emphasis(argname, argname)
              paramlist += param
  
          return fullname
  
      def handle_signature(self, sig, signode):
          """Transform a C signature into RST nodes."""
  
          fullname = self.handle_func_like_macro(sig, signode)
          if not fullname:
              fullname = super(CObject, self).handle_signature(sig, signode)
  
          if "name" in self.options:
              if self.objtype == 'function':
                  fullname = self.options["name"]
              else:
                  # FIXME: handle :name: value of other declaration types?
                  pass
          return fullname
  
      def add_target_and_index(self, name, sig, signode):
          # for C API items we add a prefix since names are usually not qualified
          # by a module name and so easily clash with e.g. section titles
          targetname = 'c.' + name
          if targetname not in self.state.document.ids:
              signode['names'].append(targetname)
              signode['ids'].append(targetname)
              signode['first'] = (not self.names)
              self.state.document.note_explicit_target(signode)
              inv = self.env.domaindata['c']['objects']
              if (name in inv and self.env.config.nitpicky):
                  if self.objtype == 'function':
                      if ('c:func', name) not in self.env.config.nitpick_ignore:
                          self.state_machine.reporter.warning(
                              'duplicate C object description of %s, ' % name +
                              'other instance in ' + self.env.doc2path(inv[name][0]),
                              line=self.lineno)
              inv[name] = (self.env.docname, self.objtype)
  
          indextext = self.get_index_text(name)
          if indextext:
              if major == 1 and minor < 4:
                  # indexnode's tuple changed in 1.4
                  # https://github.com/sphinx-doc/sphinx/commit/e6a5a3a92e938fcd75866b4227db9e0524d58f7c
                  self.indexnode['entries'].append(
                      ('single', indextext, targetname, ''))
              else:
                  self.indexnode['entries'].append(
                      ('single', indextext, targetname, '', None))
  
  class CDomain(Base_CDomain):
  
      """C language domain."""
      name = 'c'
      label = 'C'
      directives = {
          'function': CObject,
          'member':   CObject,
          'macro':    CObject,
          'type':     CObject,
          'var':      CObject,
      }