ardupilot/Tools/autotest/logger_metadata/enum_parse.py
Simon Hancock 098a53e318 Tools: Updates to log message units and help text
parse.py: Remove lone bullet points rendered on replay messages
enum_parse.py: Tweak regex on enum parser to handle comments like: "FRED = 10, ///< text"
2024-01-21 14:26:54 +11:00

250 lines
9.6 KiB
Python
Executable File

#!/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",
"Blimp": "Blimp",
}
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 = FOO(17), // optional comment"
m = re.match("\s*([A-Z0-9_a-z]+) *= *(\w+) *\\( *(\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)