2020-04-30 16:55:12 +08:00
|
|
|
# Sphinx extension to integrate defines into the Sphinx Build
|
|
|
|
#
|
|
|
|
# Runs after the IDF dummy project has been built
|
|
|
|
#
|
|
|
|
# Then emits the new 'idf-defines-generated' event which has a dictionary of raw text define values
|
|
|
|
# that other extensions can use to generate relevant data.
|
|
|
|
|
|
|
|
import glob
|
|
|
|
import os
|
2020-06-02 10:44:18 +10:00
|
|
|
import pprint
|
2020-04-30 16:55:12 +08:00
|
|
|
import re
|
2021-01-26 10:49:01 +08:00
|
|
|
import subprocess
|
2020-04-30 16:55:12 +08:00
|
|
|
|
|
|
|
|
|
|
|
def generate_defines(app, project_description):
|
2021-01-26 10:49:01 +08:00
|
|
|
sdk_config_path = os.path.join(project_description['build_dir'], 'config')
|
2020-04-30 16:55:12 +08:00
|
|
|
|
|
|
|
# Parse kconfig macros to pass into doxygen
|
|
|
|
#
|
|
|
|
# TODO: this should use the set of "config which can't be changed" eventually,
|
|
|
|
# not the header
|
2021-01-26 10:49:01 +08:00
|
|
|
defines = get_defines(os.path.join(project_description['build_dir'],
|
|
|
|
'config', 'sdkconfig.h'), sdk_config_path)
|
2020-04-30 16:55:12 +08:00
|
|
|
|
|
|
|
# Add all SOC _caps.h headers and kconfig macros to the defines
|
|
|
|
#
|
|
|
|
# kind of a hack, be nicer to add a component info dict in project_description.json
|
2021-01-26 10:49:01 +08:00
|
|
|
soc_path = [p for p in project_description['build_component_paths'] if p.endswith('/soc')][0]
|
|
|
|
soc_headers = glob.glob(os.path.join(soc_path, project_description['target'],
|
|
|
|
'include', 'soc', '*_caps.h'))
|
2020-04-30 16:55:12 +08:00
|
|
|
assert len(soc_headers) > 0
|
|
|
|
|
|
|
|
for soc_header in soc_headers:
|
|
|
|
defines.update(get_defines(soc_header, sdk_config_path))
|
|
|
|
|
2020-06-02 10:44:18 +10:00
|
|
|
# write a list of definitions to make debugging easier
|
2021-01-26 10:49:01 +08:00
|
|
|
with open(os.path.join(app.config.build_dir, 'macro-definitions.txt'), 'w') as f:
|
2020-06-02 10:44:18 +10:00
|
|
|
pprint.pprint(defines, f)
|
2021-01-26 10:49:01 +08:00
|
|
|
print('Saved macro list to %s' % f.name)
|
2020-06-02 10:44:18 +10:00
|
|
|
|
2020-04-30 16:55:12 +08:00
|
|
|
add_tags(app, defines)
|
|
|
|
|
|
|
|
app.emit('idf-defines-generated', defines)
|
|
|
|
|
|
|
|
|
|
|
|
def get_defines(header_path, sdk_config_path):
|
|
|
|
defines = {}
|
|
|
|
# Note: we run C preprocessor here without any -I arguments (except "sdkconfig.h"), so assumption is
|
|
|
|
# that these headers are all self-contained and don't include any other headers
|
|
|
|
# not in the same directory
|
2021-01-26 10:49:01 +08:00
|
|
|
print('Reading macros from %s...' % (header_path))
|
|
|
|
processed_output = subprocess.check_output(['xtensa-esp32-elf-gcc', '-I', sdk_config_path,
|
|
|
|
'-dM', '-E', header_path]).decode()
|
|
|
|
for line in processed_output.split('\n'):
|
2020-04-30 16:55:12 +08:00
|
|
|
line = line.strip()
|
2021-01-26 10:49:01 +08:00
|
|
|
m = re.search('#define ([^ ]+) ?(.*)', line)
|
2020-06-02 10:44:18 +10:00
|
|
|
if m:
|
|
|
|
name = m.group(1)
|
|
|
|
value = m.group(2)
|
2021-01-26 10:49:01 +08:00
|
|
|
if name.startswith('_'):
|
2020-06-02 10:44:18 +10:00
|
|
|
continue # toolchain macro
|
2021-01-26 10:49:01 +08:00
|
|
|
if (' ' in value) or ('=' in value):
|
|
|
|
value = '' # macros that expand to multiple tokens (ie function macros) cause doxygen errors, so just mark as 'defined'
|
2020-06-02 10:44:18 +10:00
|
|
|
defines[name] = value
|
2020-04-30 16:55:12 +08:00
|
|
|
|
|
|
|
return defines
|
|
|
|
|
|
|
|
|
|
|
|
def add_tags(app, defines):
|
|
|
|
# try to parse define values as ints and add to tags
|
|
|
|
for name, value in defines.items():
|
|
|
|
try:
|
2021-01-26 10:49:01 +08:00
|
|
|
define_value = int(value.strip('()'))
|
2020-04-30 16:55:12 +08:00
|
|
|
if define_value > 0:
|
|
|
|
app.tags.add(name)
|
|
|
|
except ValueError:
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
|
|
def setup(app):
|
|
|
|
app.connect('idf-info', generate_defines)
|
|
|
|
app.add_event('idf-defines-generated')
|
|
|
|
|
2020-06-02 10:44:18 +10:00
|
|
|
return {'parallel_read_safe': True, 'parallel_write_safe': True, 'version': '0.2'}
|