mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-11-03 21:48:15 +00:00 
			
		
		
		
	When copying the build system for Linux we missed to replace some instances of 'Documentation' by 'doc'. Signed-off-by: Heinrich Schuchardt <heinrich.schuchardt@canonical.com>
		
			
				
	
	
		
			198 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			198 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
#!/usr/bin/env python
 | 
						|
# SPDX-License-Identifier: GPL-2.0
 | 
						|
# -*- coding: utf-8; mode: python -*-
 | 
						|
# pylint: disable=R0903, C0330, R0914, R0912, E0401
 | 
						|
 | 
						|
u"""
 | 
						|
    maintainers-include
 | 
						|
    ~~~~~~~~~~~~~~~~~~~
 | 
						|
 | 
						|
    Implementation of the ``maintainers-include`` reST-directive.
 | 
						|
 | 
						|
    :copyright:  Copyright (C) 2019  Kees Cook <keescook@chromium.org>
 | 
						|
    :license:    GPL Version 2, June 1991 see linux/COPYING for details.
 | 
						|
 | 
						|
    The ``maintainers-include`` reST-directive performs extensive parsing
 | 
						|
    specific to the Linux kernel's standard "MAINTAINERS" file, in an
 | 
						|
    effort to avoid needing to heavily mark up the original plain text.
 | 
						|
"""
 | 
						|
 | 
						|
import sys
 | 
						|
import re
 | 
						|
import os.path
 | 
						|
 | 
						|
from docutils import statemachine
 | 
						|
from docutils.utils.error_reporting import ErrorString
 | 
						|
from docutils.parsers.rst import Directive
 | 
						|
from docutils.parsers.rst.directives.misc import Include
 | 
						|
 | 
						|
__version__  = '1.0'
 | 
						|
 | 
						|
def setup(app):
 | 
						|
    app.add_directive("maintainers-include", MaintainersInclude)
 | 
						|
    return dict(
 | 
						|
        version = __version__,
 | 
						|
        parallel_read_safe = True,
 | 
						|
        parallel_write_safe = True
 | 
						|
    )
 | 
						|
 | 
						|
class MaintainersInclude(Include):
 | 
						|
    u"""MaintainersInclude (``maintainers-include``) directive"""
 | 
						|
    required_arguments = 0
 | 
						|
 | 
						|
    def parse_maintainers(self, path):
 | 
						|
        """Parse all the MAINTAINERS lines into ReST for human-readability"""
 | 
						|
 | 
						|
        result = list()
 | 
						|
        result.append(".. _maintainers:")
 | 
						|
        result.append("")
 | 
						|
 | 
						|
        # Poor man's state machine.
 | 
						|
        descriptions = False
 | 
						|
        maintainers = False
 | 
						|
        subsystems = False
 | 
						|
 | 
						|
        # Field letter to field name mapping.
 | 
						|
        field_letter = None
 | 
						|
        fields = dict()
 | 
						|
 | 
						|
        prev = None
 | 
						|
        field_prev = ""
 | 
						|
        field_content = ""
 | 
						|
 | 
						|
        for line in open(path):
 | 
						|
            if sys.version_info.major == 2:
 | 
						|
                line = unicode(line, 'utf-8')
 | 
						|
            # Have we reached the end of the preformatted Descriptions text?
 | 
						|
            if descriptions and line.startswith('Maintainers'):
 | 
						|
                descriptions = False
 | 
						|
                # Ensure a blank line following the last "|"-prefixed line.
 | 
						|
                result.append("")
 | 
						|
 | 
						|
            # Start subsystem processing? This is to skip processing the text
 | 
						|
            # between the Maintainers heading and the first subsystem name.
 | 
						|
            if maintainers and not subsystems:
 | 
						|
                if re.search('^[A-Z0-9]', line):
 | 
						|
                    subsystems = True
 | 
						|
 | 
						|
            # Drop needless input whitespace.
 | 
						|
            line = line.rstrip()
 | 
						|
 | 
						|
            # Linkify all non-wildcard refs to ReST files in doc/.
 | 
						|
            pat = r'(doc/([^\s\?\*]*)\.rst)'
 | 
						|
            m = re.search(pat, line)
 | 
						|
            if m:
 | 
						|
                # maintainers.rst is in a subdirectory, so include "../".
 | 
						|
                line = re.sub(pat, ':doc:`%s <../%s>`' % (m.group(2), m.group(2)), line)
 | 
						|
 | 
						|
            # Check state machine for output rendering behavior.
 | 
						|
            output = None
 | 
						|
            if descriptions:
 | 
						|
                # Escape the escapes in preformatted text.
 | 
						|
                output = "| %s" % (line.replace("\\", "\\\\"))
 | 
						|
                # Look for and record field letter to field name mappings:
 | 
						|
                #   R: Designated *reviewer*: FullName <address@domain>
 | 
						|
                m = re.search(r"\s(\S):\s", line)
 | 
						|
                if m:
 | 
						|
                    field_letter = m.group(1)
 | 
						|
                if field_letter and not field_letter in fields:
 | 
						|
                    m = re.search(r"\*([^\*]+)\*", line)
 | 
						|
                    if m:
 | 
						|
                        fields[field_letter] = m.group(1)
 | 
						|
            elif subsystems:
 | 
						|
                # Skip empty lines: subsystem parser adds them as needed.
 | 
						|
                if len(line) == 0:
 | 
						|
                    continue
 | 
						|
                # Subsystem fields are batched into "field_content"
 | 
						|
                if line[1] != ':':
 | 
						|
                    # Render a subsystem entry as:
 | 
						|
                    #   SUBSYSTEM NAME
 | 
						|
                    #   ~~~~~~~~~~~~~~
 | 
						|
 | 
						|
                    # Flush pending field content.
 | 
						|
                    output = field_content + "\n\n"
 | 
						|
                    field_content = ""
 | 
						|
 | 
						|
                    # Collapse whitespace in subsystem name.
 | 
						|
                    heading = re.sub(r"\s+", " ", line)
 | 
						|
                    output = output + "%s\n%s" % (heading, "~" * len(heading))
 | 
						|
                    field_prev = ""
 | 
						|
                else:
 | 
						|
                    # Render a subsystem field as:
 | 
						|
                    #   :Field: entry
 | 
						|
                    #           entry...
 | 
						|
                    field, details = line.split(':', 1)
 | 
						|
                    details = details.strip()
 | 
						|
 | 
						|
                    # Mark paths (and regexes) as literal text for improved
 | 
						|
                    # readability and to escape any escapes.
 | 
						|
                    if field in ['F', 'N', 'X', 'K']:
 | 
						|
                        # But only if not already marked :)
 | 
						|
                        if not ':doc:' in details:
 | 
						|
                            details = '``%s``' % (details)
 | 
						|
 | 
						|
                    # Comma separate email field continuations.
 | 
						|
                    if field == field_prev and field_prev in ['M', 'R', 'L']:
 | 
						|
                        field_content = field_content + ","
 | 
						|
 | 
						|
                    # Do not repeat field names, so that field entries
 | 
						|
                    # will be collapsed together.
 | 
						|
                    if field != field_prev:
 | 
						|
                        output = field_content + "\n"
 | 
						|
                        field_content = ":%s:" % (fields.get(field, field))
 | 
						|
                    field_content = field_content + "\n\t%s" % (details)
 | 
						|
                    field_prev = field
 | 
						|
            else:
 | 
						|
                output = line
 | 
						|
 | 
						|
            # Re-split on any added newlines in any above parsing.
 | 
						|
            if output != None:
 | 
						|
                for separated in output.split('\n'):
 | 
						|
                    result.append(separated)
 | 
						|
 | 
						|
            # Update the state machine when we find heading separators.
 | 
						|
            if line.startswith('----------'):
 | 
						|
                if prev.startswith('Descriptions'):
 | 
						|
                    descriptions = True
 | 
						|
                if prev.startswith('Maintainers'):
 | 
						|
                    maintainers = True
 | 
						|
 | 
						|
            # Retain previous line for state machine transitions.
 | 
						|
            prev = line
 | 
						|
 | 
						|
        # Flush pending field contents.
 | 
						|
        if field_content != "":
 | 
						|
            for separated in field_content.split('\n'):
 | 
						|
                result.append(separated)
 | 
						|
 | 
						|
        output = "\n".join(result)
 | 
						|
        # For debugging the pre-rendered results...
 | 
						|
        #print(output, file=open("/tmp/MAINTAINERS.rst", "w"))
 | 
						|
 | 
						|
        self.state_machine.insert_input(
 | 
						|
          statemachine.string2lines(output), path)
 | 
						|
 | 
						|
    def run(self):
 | 
						|
        """Include the MAINTAINERS file as part of this reST file."""
 | 
						|
        if not self.state.document.settings.file_insertion_enabled:
 | 
						|
            raise self.warning('"%s" directive disabled.' % self.name)
 | 
						|
 | 
						|
        # Walk up source path directories to find doc/../
 | 
						|
        path = self.state_machine.document.attributes['source']
 | 
						|
        path = os.path.realpath(path)
 | 
						|
        tail = path
 | 
						|
        while tail != "doc" and tail != "":
 | 
						|
            (path, tail) = os.path.split(path)
 | 
						|
 | 
						|
        # Append "MAINTAINERS"
 | 
						|
        path = os.path.join(path, "MAINTAINERS")
 | 
						|
 | 
						|
        try:
 | 
						|
            self.state.document.settings.record_dependencies.add(path)
 | 
						|
            lines = self.parse_maintainers(path)
 | 
						|
        except IOError as error:
 | 
						|
            raise self.severe('Problems with "%s" directive path:\n%s.' %
 | 
						|
                      (self.name, ErrorString(error)))
 | 
						|
 | 
						|
        return []
 |