[topgen] Cleanup PLIC Source/Interrupt Mapping Logic
Signed-off-by: Sam Elliott <selliott@lowrisc.org>
diff --git a/util/topgen/c.py b/util/topgen/c.py
index 3d202c0..9455a68 100644
--- a/util/topgen/c.py
+++ b/util/topgen/c.py
@@ -5,11 +5,13 @@
`top_{name}.h`.
"""
+from collections import OrderedDict
+
from mako.template import Template
class Name(object):
- """We often need to format names in specific ways, this class does so."""
+ """We often need to format names in specific ways; this class does so."""
def __add__(self, other):
return Name(self.parts + other.parts)
@@ -77,6 +79,8 @@
self.constants.append((full_name, value, docstring))
+ return full_name
+
def add_last_constant(self, docstring=""):
assert not self.finalized
@@ -96,11 +100,114 @@
return Template(template).render(enum=self)
+class CArrayMapping(object):
+ def __init__(self, name, output_type_name):
+ self.name = name
+ self.output_type_name = output_type_name
+
+ self.mapping = OrderedDict()
+
+ def add_entry(self, in_name, out_name):
+ self.mapping[in_name] = out_name
+
+ def render_declaration(self):
+ template = (
+ "extern const ${mapping.output_type_name.as_c_type()}\n"
+ " ${mapping.name.as_snake_case()}[${len(mapping.mapping)}];")
+ return Template(template).render(mapping=self)
+
+ def render_definition(self):
+ template = (
+ "const ${mapping.output_type_name.as_c_type()}\n"
+ " ${mapping.name.as_snake_case()}[${len(mapping.mapping)}] = {\n"
+ "% for in_name, out_name in mapping.mapping.items():\n"
+ " [${in_name.as_c_enum()}] = ${out_name.as_c_enum()},\n"
+ "% endfor\n"
+ "};\n")
+ return Template(template).render(mapping=self)
+
+
class TopGenC(object):
def __init__(self, top_info):
self.top = top_info
self._top_name = Name(["top"]) + Name.from_snake_case(top_info["name"])
+ self._init_plic_targets()
+ self._init_plic_mapping()
+
+ def _init_plic_targets(self):
+ enum = CEnum(self._top_name + Name(["plic", "target"]))
+
+ for core_id in range(int(self.top["num_cores"])):
+ enum.add_constant(Name(["ibex", str(core_id)]),
+ docstring="Ibex Core {}".format(core_id))
+
+ enum.add_last_constant("Final PLIC target")
+
+ self.plic_targets = enum
+
+ def _init_plic_mapping(self):
+ """We eventually want to generate a mapping from interrupt id to the
+ source peripheral.
+
+ In order to do so, we generate two enums, and store the generated names
+ in a dictionary that represents the mapping.
+
+ Plic Interrupt ID 0 corresponds to no interrupt, and so no peripheral,
+ so we encode that in the enum as "unknown".
+
+ The interrupts have to be added in order, with "none" first, to ensure
+ that they get the correct mapping to their PLIC id, which is used for
+ addressing the right registers and bits.
+ """
+ sources = CEnum(self._top_name + Name(["plic", "peripheral"]))
+ interrupts = CEnum(self._top_name + Name(["plic", "irq", "id"]))
+ plic_mapping = CArrayMapping(
+ self._top_name + Name(["plic", "interrupt", "for", "peripheral"]),
+ sources.name)
+
+ unknown_source = sources.add_constant(Name(["unknown"]),
+ docstring="Unknown Peripheral")
+ none_irq_id = interrupts.add_constant(Name(["none"]),
+ docstring="No Interrupt")
+ plic_mapping.add_entry(none_irq_id, unknown_source)
+
+ # When we generate the `interrupts` enum, the only info we have about
+ # the source is the module name. We'll use `source_name_map` to map a
+ # short module name to the full name object used for the enum constant.
+ source_name_map = {}
+
+ for name in self.top["interrupt_module"]:
+ source_name = sources.add_constant(Name.from_snake_case(name),
+ docstring=name)
+ source_name_map[name] = source_name
+
+ sources.add_last_constant("Final PLIC peripheral")
+
+ for intr in self.top["interrupt"]:
+ # Some interrupts are multiple bits wide. Here we deal with that by
+ # adding a bit-index suffix
+ if "width" in intr and int(intr["width"]) != 1:
+ for i in range(int(intr["width"])):
+ name = Name.from_snake_case(
+ intr["name"]) + Name([str(i)])
+ irq_id = interrupts.add_constant(name,
+ docstring="{} {}".format(
+ intr["name"], i))
+ source_name = source_name_map[intr["module_name"]]
+ plic_mapping.add_entry(irq_id, source_name)
+ else:
+ name = Name.from_snake_case(intr["name"])
+ irq_id = interrupts.add_constant(name, docstring=intr["name"])
+ source_name = source_name_map[intr["module_name"]]
+ plic_mapping.add_entry(irq_id, source_name)
+
+ interrupts.add_last_constant("The Last Valid Interrupt ID.")
+
+ self.plic_sources = sources
+ self.plic_interrupts = interrupts
+ self.plic_mapping = plic_mapping
+
def modules(self):
return [(m["name"],
MemoryRegion(self._top_name + Name.from_snake_case(m["name"]),
@@ -112,14 +219,3 @@
MemoryRegion(self._top_name + Name.from_snake_case(m["name"]),
m["base_addr"], m["size"]))
for m in self.top["memory"]]
-
- def plic_targets(self):
- enum = CEnum(self._top_name + Name(["plic", "target"]))
-
- for core_id in range(int(self.top["num_cores"])):
- enum.add_constant(Name(["ibex", str(core_id)]),
- docstring="Ibex Core {}".format(core_id))
-
- enum.add_last_constant("Final PLIC target")
-
- return enum