mirror of
				https://github.com/espressif/esp-idf.git
				synced 2025-10-26 03:37:51 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			306 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			306 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # Extension to generate Doxygen XML include files, with IDF config & soc macros included
 | |
| from __future__ import print_function, unicode_literals
 | |
| 
 | |
| import os
 | |
| import os.path
 | |
| import re
 | |
| import subprocess
 | |
| from io import open
 | |
| 
 | |
| from .util import copy_if_modified
 | |
| 
 | |
| ALL_KINDS = [
 | |
|     ('function', 'Functions'),
 | |
|     ('union', 'Unions'),
 | |
|     ('struct', 'Structures'),
 | |
|     ('define', 'Macros'),
 | |
|     ('typedef', 'Type Definitions'),
 | |
|     ('enum', 'Enumerations')
 | |
| ]
 | |
| """list of items that will be generated for a single API file
 | |
| """
 | |
| 
 | |
| 
 | |
| def setup(app):
 | |
|     # The idf_build_system extension will emit this event once it has generated documentation macro definitions
 | |
|     app.connect('idf-defines-generated', generate_doxygen)
 | |
|     return {'parallel_read_safe': True, 'parallel_write_safe': True, 'version': '0.2'}
 | |
| 
 | |
| 
 | |
| def generate_doxygen(app, defines):
 | |
|     build_dir = os.path.dirname(app.doctreedir.rstrip(os.sep))
 | |
| 
 | |
|     # Call Doxygen to get XML files from the header files
 | |
|     print('Calling Doxygen to generate latest XML files')
 | |
|     doxy_env = os.environ
 | |
|     doxy_env.update({
 | |
|         'ENV_DOXYGEN_DEFINES': ' '.join('{}={}'.format(key, value) for key, value in defines.items()),
 | |
|         'IDF_PATH': app.config.idf_path,
 | |
|         'IDF_TARGET': app.config.idf_target,
 | |
|     })
 | |
|     doxyfile_dir = os.path.join(app.config.docs_root, 'doxygen')
 | |
|     doxyfile_main = os.path.join(doxyfile_dir, 'Doxyfile_common')
 | |
|     doxyfile_target = os.path.join(doxyfile_dir, 'Doxyfile_' + app.config.idf_target)
 | |
|     print('Running doxygen with doxyfiles {} and {}'.format(doxyfile_main, doxyfile_target))
 | |
| 
 | |
|     # It's possible to have doxygen log warnings to a file using WARN_LOGFILE directive,
 | |
|     # but in some cases it will still log an error to stderr and return success!
 | |
|     #
 | |
|     # So take all of stderr and redirect it to a logfile (will contain warnings and errors)
 | |
|     logfile = os.path.join(build_dir, 'doxygen-warning-log.txt')
 | |
| 
 | |
|     with open(logfile, 'w') as f:
 | |
|         # note: run Doxygen in the build directory, so the xml & xml_in files end up in there
 | |
|         subprocess.check_call(['doxygen', doxyfile_main], env=doxy_env, cwd=build_dir, stderr=f)
 | |
| 
 | |
|     # Doxygen has generated XML files in 'xml' directory.
 | |
|     # Copy them to 'xml_in', only touching the files which have changed.
 | |
|     copy_if_modified(os.path.join(build_dir, 'xml/'), os.path.join(build_dir, 'xml_in/'))
 | |
| 
 | |
|     # Generate 'api_name.inc' files from the Doxygen XML files
 | |
|     doxygen_paths = [doxyfile_main, doxyfile_target]
 | |
|     convert_api_xml_to_inc(app, doxygen_paths)
 | |
| 
 | |
| 
 | |
| def convert_api_xml_to_inc(app, doxyfiles):
 | |
|     """ Generate header_file.inc files
 | |
|     with API reference made of doxygen directives
 | |
|     for each header file
 | |
|     specified in the 'INPUT' statement of the Doxyfile.
 | |
|     """
 | |
|     build_dir = app.config.build_dir
 | |
| 
 | |
|     xml_directory_path = '{}/xml'.format(build_dir)
 | |
|     inc_directory_path = '{}/inc'.format(build_dir)
 | |
| 
 | |
|     if not os.path.isdir(xml_directory_path):
 | |
|         raise RuntimeError('Directory {} does not exist!'.format(xml_directory_path))
 | |
| 
 | |
|     if not os.path.exists(inc_directory_path):
 | |
|         os.makedirs(inc_directory_path)
 | |
| 
 | |
|     header_paths = [p for d in doxyfiles for p in get_doxyfile_input_paths(app, d)]
 | |
| 
 | |
|     print("Generating 'api_name.inc' files with Doxygen directives")
 | |
|     for header_file_path in header_paths:
 | |
|         api_name = get_api_name(header_file_path)
 | |
|         inc_file_path = inc_directory_path + '/' + api_name + '.inc'
 | |
|         rst_output = generate_directives(header_file_path, xml_directory_path)
 | |
| 
 | |
|         previous_rst_output = ''
 | |
|         if os.path.isfile(inc_file_path):
 | |
|             with open(inc_file_path, 'r', encoding='utf-8') as inc_file_old:
 | |
|                 previous_rst_output = inc_file_old.read()
 | |
| 
 | |
|         if previous_rst_output != rst_output:
 | |
|             with open(inc_file_path, 'w', encoding='utf-8') as inc_file:
 | |
|                 inc_file.write(rst_output)
 | |
| 
 | |
| 
 | |
| def get_doxyfile_input_paths(app, doxyfile_path):
 | |
|     """Get contents of Doxyfile's INPUT statement.
 | |
| 
 | |
|     Returns:
 | |
|         Contents of Doxyfile's INPUT.
 | |
| 
 | |
|     """
 | |
|     if not os.path.isfile(doxyfile_path):
 | |
|         raise RuntimeError("Doxyfile '{}' does not exist!".format(doxyfile_path))
 | |
| 
 | |
|     print("Getting Doxyfile's INPUT")
 | |
| 
 | |
|     with open(doxyfile_path, 'r', encoding='utf-8') as input_file:
 | |
|         line = input_file.readline()
 | |
|         # read contents of Doxyfile until 'INPUT' statement
 | |
|         while line:
 | |
|             if line.find('INPUT') == 0:
 | |
|                 break
 | |
|             line = input_file.readline()
 | |
| 
 | |
|         doxyfile_INPUT = []
 | |
|         line = input_file.readline()
 | |
|         # skip input_file contents until end of 'INPUT' statement
 | |
|         while line:
 | |
|             if line.isspace():
 | |
|                 # we have reached the end of 'INPUT' statement
 | |
|                 break
 | |
|             # process only lines that are not comments
 | |
|             if line.find('#') == -1:
 | |
|                 # extract header file path inside components folder
 | |
|                 m = re.search('components/(.*\.h)', line)  # noqa: W605 - regular expression
 | |
|                 header_file_path = m.group(1)
 | |
| 
 | |
|                 # Replace env variable used for multi target header
 | |
|                 header_file_path = header_file_path.replace('$(IDF_TARGET)', app.config.idf_target)
 | |
| 
 | |
|                 doxyfile_INPUT.append(header_file_path)
 | |
| 
 | |
|             # proceed reading next line
 | |
|             line = input_file.readline()
 | |
| 
 | |
|     return doxyfile_INPUT
 | |
| 
 | |
| 
 | |
| def get_api_name(header_file_path):
 | |
|     """Get name of API from header file path.
 | |
| 
 | |
|     Args:
 | |
|         header_file_path: path to the header file.
 | |
| 
 | |
|     Returns:
 | |
|         The name of API.
 | |
| 
 | |
|     """
 | |
|     api_name = ''
 | |
|     regex = r'.*/(.*)\.h'
 | |
|     m = re.search(regex, header_file_path)
 | |
|     if m:
 | |
|         api_name = m.group(1)
 | |
| 
 | |
|     return api_name
 | |
| 
 | |
| 
 | |
| def generate_directives(header_file_path, xml_directory_path):
 | |
|     """Generate API reference with Doxygen directives for a header file.
 | |
| 
 | |
|     Args:
 | |
|         header_file_path: a path to the header file with API.
 | |
| 
 | |
|     Returns:
 | |
|         Doxygen directives for the header file.
 | |
| 
 | |
|     """
 | |
| 
 | |
|     api_name = get_api_name(header_file_path)
 | |
| 
 | |
|     # in XLT file name each "_" in the api name is expanded by Doxygen to "__"
 | |
|     xlt_api_name = api_name.replace('_', '__')
 | |
|     xml_file_path = '%s/%s_8h.xml' % (xml_directory_path, xlt_api_name)
 | |
| 
 | |
|     rst_output = ''
 | |
|     rst_output = ".. File automatically generated by 'gen-dxd.py'\n"
 | |
|     rst_output += '\n'
 | |
|     rst_output += get_rst_header('Header File')
 | |
|     rst_output += '* :component_file:`' + header_file_path + '`\n'
 | |
|     rst_output += '\n'
 | |
| 
 | |
|     try:
 | |
|         import xml.etree.cElementTree as ET
 | |
|     except ImportError:
 | |
|         import xml.etree.ElementTree as ET
 | |
| 
 | |
|     tree = ET.ElementTree(file=xml_file_path)
 | |
|     for kind, label in ALL_KINDS:
 | |
|         rst_output += get_directives(tree, kind)
 | |
| 
 | |
|     return rst_output
 | |
| 
 | |
| 
 | |
| def get_rst_header(header_name):
 | |
|     """Get rst formatted code with a header.
 | |
| 
 | |
|     Args:
 | |
|         header_name: name of header.
 | |
| 
 | |
|     Returns:
 | |
|         Formatted rst code with the header.
 | |
| 
 | |
|     """
 | |
| 
 | |
|     rst_output = ''
 | |
|     rst_output += header_name + '\n'
 | |
|     rst_output += '^' * len(header_name) + '\n'
 | |
|     rst_output += '\n'
 | |
| 
 | |
|     return rst_output
 | |
| 
 | |
| 
 | |
| def select_unions(innerclass_list):
 | |
|     """Select unions from innerclass list.
 | |
| 
 | |
|     Args:
 | |
|         innerclass_list: raw list with unions and structures
 | |
|                          extracted from Dogygen's xml file.
 | |
| 
 | |
|     Returns:
 | |
|         Doxygen directives with unions selected from the list.
 | |
| 
 | |
|     """
 | |
| 
 | |
|     rst_output = ''
 | |
|     for line in innerclass_list.splitlines():
 | |
|         # union is denoted by "union" at the beginning of line
 | |
|         if line.find('union') == 0:
 | |
|             union_id, union_name = re.split(r'\t+', line)
 | |
|             rst_output += '.. doxygenunion:: '
 | |
|             rst_output += union_name
 | |
|             rst_output += '\n'
 | |
| 
 | |
|     return rst_output
 | |
| 
 | |
| 
 | |
| def select_structs(innerclass_list):
 | |
|     """Select structures from innerclass list.
 | |
| 
 | |
|     Args:
 | |
|         innerclass_list: raw list with unions and structures
 | |
|                          extracted from Dogygen's xml file.
 | |
| 
 | |
|     Returns:
 | |
|         Doxygen directives with structures selected from the list.
 | |
|         Note: some structures are excluded as described on code below.
 | |
| 
 | |
|     """
 | |
| 
 | |
|     rst_output = ''
 | |
|     for line in innerclass_list.splitlines():
 | |
|         # structure is denoted by "struct" at the beginning of line
 | |
|         if line.find('struct') == 0:
 | |
|             # skip structures that are part of union
 | |
|             # they are documented by 'doxygenunion' directive
 | |
|             if line.find('::') > 0:
 | |
|                 continue
 | |
|             struct_id, struct_name = re.split(r'\t+', line)
 | |
|             rst_output += '.. doxygenstruct:: '
 | |
|             rst_output += struct_name
 | |
|             rst_output += '\n'
 | |
|             rst_output += '    :members:\n'
 | |
|             rst_output += '\n'
 | |
| 
 | |
|     return rst_output
 | |
| 
 | |
| 
 | |
| def get_directives(tree, kind):
 | |
|     """Get directives for specific 'kind'.
 | |
| 
 | |
|     Args:
 | |
|         tree: the ElementTree 'tree' of XML by Doxygen
 | |
|         kind: name of API "kind" to be generated
 | |
| 
 | |
|     Returns:
 | |
|         Doxygen directives for selected 'kind'.
 | |
|         Note: the header with "kind" name is included.
 | |
| 
 | |
|     """
 | |
| 
 | |
|     rst_output = ''
 | |
|     if kind in ['union', 'struct']:
 | |
|         innerclass_list = ''
 | |
|         for elem in tree.iterfind('compounddef/innerclass'):
 | |
|             innerclass_list += elem.attrib['refid'] + '\t' + elem.text + '\n'
 | |
|         if kind == 'union':
 | |
|             rst_output += select_unions(innerclass_list)
 | |
|         else:
 | |
|             rst_output += select_structs(innerclass_list)
 | |
|     else:
 | |
|         for elem in tree.iterfind(
 | |
|                 'compounddef/sectiondef/memberdef[@kind="%s"]' % kind):
 | |
|             name = elem.find('name')
 | |
|             rst_output += '.. doxygen%s:: ' % kind
 | |
|             rst_output += name.text + '\n'
 | |
|     if rst_output:
 | |
|         all_kinds_dict = dict(ALL_KINDS)
 | |
|         rst_output = get_rst_header(all_kinds_dict[kind]) + rst_output + '\n'
 | |
| 
 | |
|     return rst_output
 | 
