Tools: add script to extract enumeration as metadata

This commit is contained in:
Peter Barker 2020-03-21 17:56:15 +11:00 committed by Andrew Tridgell
parent 063a041d22
commit 3708ed7a45
5 changed files with 293 additions and 7 deletions

View File

@ -27,7 +27,7 @@ DO NOT EDIT
self.fh = open("LogMessages.html", mode='w')
print(self.preface(), file=self.fh)
def emit(self, doccos):
def emit(self, doccos, enumerations):
self.start()
for docco in doccos:
print(' <h1>%s</h1>' % docco.name, file=self.fh)

View File

@ -24,7 +24,7 @@ This is a list of log messages which may be present in logs produced and stored
self.fh = open("LogMessages.rst", mode='w')
print(self.preface(), file=self.fh)
def emit(self, doccos):
def emit(self, doccos, enumerations):
self.start()
for docco in doccos:
print('.. _%s:' % docco.name, file=self.fh)
@ -41,10 +41,29 @@ This is a list of log messages which may be present in logs produced and stored
fdesc = docco.fields[f]["description"]
else:
fdesc = ""
fieldnamething = None
if "bitmaskenum" in docco.fields[f]:
fieldnamething = "bitmaskenum"
table_label = "Bitmask values"
elif "valueenum" in docco.fields[f]:
fieldnamething = "valueenum"
table_label = "Values"
if fieldnamething is not None:
enum_name = docco.fields[f][fieldnamething]
if enum_name not in enumerations:
raise Exception("Unknown enum (%s) (have %s)" %
(enum_name, "\n".join(sorted(enumerations.keys()))))
enumeration = enumerations[enum_name]
bitmaskrows = []
for enumentry in enumeration.entries:
# print("enumentry: %s" % str(enumentry))
comment = enumentry.comment
if comment is None:
comment = ""
bitmaskrows.append([enumentry.name, str(enumentry.value), comment])
fdesc += "\n%s:\n%s" % (table_label, self.tablify(bitmaskrows))
rows.append([f, fdesc])
# if "bits" in docco.fields[f]:
# print(' <bits>%s</bits>' %
# docco.fields[f]["bits"], file=self.fh)
print(self.tablify(rows), file=self.fh)
print("", file=self.fh)

View File

@ -18,7 +18,7 @@ class XMLEmitter(emitter.Emitter):
print(self.preface(), file=self.fh)
self.loggermessagefile = etree.Element('loggermessagefile')
def emit(self, doccos):
def emit(self, doccos, enumerations):
self.start()
for docco in doccos:
xml_logformat = etree.SubElement(self.loggermessagefile, 'logformat', name=docco.name)

View File

@ -0,0 +1,243 @@
#!/usr/bin/env python
from __future__ import print_function
import argparse
import os
import re
import sys
topdir = os.path.join(os.path.dirname(os.path.realpath(__file__)), '../../../')
topdir = os.path.realpath(topdir)
class EnumDocco(object):
vehicle_map = {
"Rover": "Rover",
"Sub": "ArduSub",
"Copter": "ArduCopter",
"Plane": "ArduPlane",
"Tracker": "AntennaTracker",
}
def __init__(self, vehicle):
self.vehicle = vehicle
self.enumerations = []
class EnumEntry(object):
def __init__(self, name, value, comment):
self.name = name
self.value = value
self.comment = comment
def match_enum_line(self, line):
# attempts to extract name, value and comment from line.
# Match: " FRED, // optional comment"
m = re.match("\s*([A-Z0-9_a-z]+)\s*,? *(?:// *(.*) *)?$", line)
if m is not None:
return (m.group(1), None, m.group(2))
# Match: " FRED, /* optional comment */"
m = re.match("\s*([A-Z0-9_a-z]+)\s*,? *(?:/[*] *(.*) *[*]/ *)?$", line)
if m is not None:
return (m.group(1), None, m.group(2))
# Match: " FRED = 17, // optional comment"
m = re.match("\s*([A-Z0-9_a-z]+)\s*=\s*([-0-9]+)\s*,?(?:\s*//\s*(.*) *)?$",
line)
if m is not None:
return (m.group(1), m.group(2), m.group(3))
# Match: " FRED = 17, // optional comment"
m = re.match("\s*([A-Z0-9_a-z]+) *= *([-0-9]+) *,?(?: */* *(.*) *)? *[*]/ *$",
line)
if m is not None:
return (m.group(1), m.group(2), m.group(3))
# Match: " FRED = 1U<<0, // optional comment"
m = re.match("\s*([A-Z0-9_a-z]+) *= *[(]?1U? *[<][<] *(\d+)(?:, *// *(.*) *)?",
line)
if m is not None:
return (m.group(1), 1 << int(m.group(2)), m.group(3))
# Match: " FRED = 0xabc, // optional comment"
m = re.match("\s*([A-Z0-9_a-z]+) *= *(?:0[xX]([0-9A-Fa-f]+))(?:, *// *(.*) *)?",
line)
if m is not None:
return (m.group(1), int(m.group(2), 16), m.group(3))
'''start discarded matches - lines we understand but can't do anything
with:'''
# Match: " FRED = 17, // optional comment"
m = re.match("\s*([A-Z0-9_a-z]+) *= *(\w+) *,?(?: *// *(.*) *)?$",
line)
if m is not None:
return (None, None, None)
# Match: " FRED = 1U<<0, // optional comment"
m = re.match("\s*([A-Z0-9_a-z]+) *= *[(]?3U? *[<][<] *(\d+)(?:, *// *(.*) *)?",
line)
if m is not None:
return (m.group(1), 1 << int(m.group(2)), m.group(3))
if m is None:
raise ValueError("Failed to match (%s)" % line)
def enumerations_from_file(self, source_file):
state_outside = "outside"
state_inside = "inside"
state = state_outside
enumerations = []
with open(source_file) as f:
enum_name = None
in_class = None
while True:
line = f.readline()
if line == "":
break
line = line.rstrip()
# print("state=%s line: %s" % (state, line))
if re.match("\s*//.*", line):
continue
if state == "outside":
if re.match("class .*;", line) is not None:
# forward-declaration of a class
continue
m = re.match("class *(\w+)", line)
if m is not None:
in_class = m.group(1)
continue
m = re.match("namespace *(\w+)", line)
if m is not None:
in_class = m.group(1)
continue
m = re.match(".*enum\s*(class)? *([\w]+)\s*(?::.*_t)? *{(.*)};", line)
if m is not None:
# all one one line! Thanks!
enum_name = m.group(2)
entries_string = m.group(3)
entry_names = [x.strip() for x in entries_string.split(",")]
count = 0
entries = []
for entry in entry_names:
entries.append(EnumDocco.EnumEntry(enum_name, count, None))
count += 1
new_enumeration = EnumDocco.Enumeration(enum_name, entries)
enumerations.append(new_enumeration)
continue
m = re.match(".*enum\s*(class)? *([\w]+)\s*(?::.*_t)? *{", line)
if m is not None:
enum_name = m.group(2)
# print("%s: %s" % (source_file, enum_name))
entries = []
last_value = None
state = state_inside
skip_enumeration = False
continue
if state == "inside":
if re.match("\s*$", line):
continue
if re.match("#if", line):
continue
if re.match("#endif", line):
continue
if re.match("#else", line):
continue
if re.match(".*}\s*\w*(\s*=\s*[\w:]+)?;", line):
# potential end of enumeration
if not skip_enumeration:
if enum_name is None:
raise Exception("WT??")
if in_class is not None:
enum_name = "::".join([in_class, enum_name])
new_enumeration = EnumDocco.Enumeration(enum_name, entries)
enumerations.append(new_enumeration)
# print("Got enum (%s)" % enum_name)
# for entry in new_enumeration.entries:
# print(" %s: %u (%s)" % (entry.name, entry.value, entry.comment))
state = state_outside
continue
(name, value, comment) = self.match_enum_line(line)
if name is None:
skip_enumeration = True
continue
# print(" name=(%s) value=(%s) comment=(%s)\n" % (name, value, comment))
if value is None:
if last_value is None:
value = 0
last_value = 0
else:
last_value += 1
value = last_value
else:
value = int(value)
last_value = value
# print("entry=%s value=%s comment=%s" % (name, value, comment))
entries.append(EnumDocco.EnumEntry(name, value, comment))
return enumerations
class Enumeration(object):
def __init__(self, name, entries):
self.name = name
self.entries = entries
def search_for_files(self, dirs_to_search):
_next = []
for _dir in dirs_to_search:
for entry in os.listdir(_dir):
if "AP_Scripting/lua" in _dir:
continue
if "modules" in _dir:
continue
if "examples" in _dir:
continue
filepath = os.path.join(_dir, entry)
if os.path.isdir(filepath):
_next.append(filepath)
continue
(name, extension) = os.path.splitext(filepath)
if extension not in [".cpp", ".h"]:
continue
if filepath.endswith("libraries/AP_HAL/utility/getopt_cpp.h"):
continue
self.files.append(filepath)
if len(_next):
self.search_for_files(_next)
def parse_files(self):
for _file in self.files:
self.enumerations.extend(self.enumerations_from_file(_file))
def get_enumerations(self):
self.files = []
self.search_for_files([os.path.join(topdir, x) for x in [
self.vehicle_map[self.vehicle],
"libraries"]])
self.parse_files()
return self.enumerations
def run(self):
self.get_enumerations()
if __name__ == '__main__':
parser = argparse.ArgumentParser(description="Parse parameters.")
parser.add_argument("-v", "--verbose", dest='verbose', action='store_true', default=False, help="show debugging output")
parser.add_argument("--vehicle", required=True, help="Vehicle type to generate for")
args = parser.parse_args()
s = EnumDocco(args.vehicle)
if args.vehicle not in s.vehicle_map:
print("Invalid vehicle (choose from: %s)" % str(s.vehicle_map.keys()))
sys.exit(1)
s.run()
print("Enumerations: %s" % s.enumerations)

View File

@ -12,6 +12,8 @@ import emit_html
import emit_rst
import emit_xml
import enum_parse
topdir = os.path.join(os.path.dirname(os.path.realpath(__file__)), '../../../')
topdir = os.path.realpath(topdir)
@ -22,6 +24,8 @@ re_url = re.compile(r"\s*//\s*@URL\s*:\s*(.*)")
re_field = re.compile(r"\s*//\s*@Field\s*:\s*(\w+):\s*(.*)")
re_fieldbits = re.compile(r"\s*//\s*@FieldBits\s*:\s*(\w+):\s*(.*)")
re_fieldbits = re.compile(r"\s*//\s*@FieldBits\s*:\s*(\w+):\s*(.*)")
re_fieldbitmaskenum = re.compile(r"\s*//\s*@FieldBitmaskEnum\s*:\s*(\w+):\s*(.*)")
re_fieldvalueenum = re.compile(r"\s*//\s*@FieldValueEnum\s*:\s*(\w+):\s*(.*)")
re_vehicles = re.compile(r"\s*//\s*@Vehicles\s*:\s*(.*)")
# TODO: validate URLS actually return 200
@ -80,6 +84,14 @@ class LoggerDocco(object):
self.ensure_field(field)
self.fields[field]["bits"] = bits
def set_fieldbitmaskenum(self, field, bits):
self.ensure_field(field)
self.fields[field]["bitmaskenum"] = bits
def set_fieldbitmaskvalue(self, field, bits):
self.ensure_field(field)
self.fields[field]["valueenum"] = bits
def set_vehicles(self, vehicles):
self.vehicles = vehicles
@ -140,6 +152,14 @@ class LoggerDocco(object):
if m is not None:
docco.set_field_bits(m.group(1), m.group(2))
continue
m = re_fieldbitmaskenum.match(line)
if m is not None:
docco.set_fieldbitmaskenum(m.group(1), m.group(2))
continue
m = re_fieldvalueenum.match(line)
if m is not None:
docco.set_fieldbitmaskvalue(m.group(1), m.group(2))
continue
m = re_vehicles.match(line)
if m is not None:
docco.set_vehicles([x.strip() for x in m.group(1).split(',')])
@ -164,10 +184,14 @@ class LoggerDocco(object):
new_doccos.append(docco)
new_doccos = sorted(new_doccos, key=lambda x : x.name)
enums_by_name = {}
for enum in self.enumerations:
enums_by_name[enum.name] = enum
for emitter in self.emitters:
emitter.emit(new_doccos)
emitter.emit(new_doccos, enums_by_name)
def run(self):
self.enumerations = enum_parse.EnumDocco(self.vehicle).get_enumerations()
self.files = []
self.search_for_files([self.vehicle_map[self.vehicle], "libraries"])
self.parse_files()