3
0
Fork 0
mirror of https://github.com/YosysHQ/sby.git synced 2025-10-26 10:44:37 +00:00
sby/sbysrc/sby_design.py
Jannis Harder 52184e5bf0 Initial support for a multi-task property status database
This adds initial support for an sqlite database that is shared across
multiple tasks of a single SBY file and that can track the status of
individual properties.

The amount of information tracked in the database is currently quite
minimal and depends on the engine and options used. This can be
incrementally extended in the future.

The ways in which the information in the database can be queries is even
more limited for this initial version, consisting of a single '--status'
option which lists all properties and their status.
2024-02-20 13:34:58 +01:00

275 lines
9.1 KiB
Python

#
# SymbiYosys (sby) -- Front-end for Yosys-based formal verification flows
#
# Copyright (C) 2022 N. Engelhardt <nak@yosyshq.com>
#
# Permission to use, copy, modify, and/or distribute this software for any
# purpose with or without fee is hereby granted, provided that the above
# copyright notice and this permission notice appear in all copies.
#
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
#
import json, re
from enum import Enum, auto
from dataclasses import dataclass, field
from typing import Optional, Tuple
addr_re = re.compile(r'\\\[[0-9]+\]$')
public_name_re = re.compile(r"\\([a-zA-Z_][a-zA-Z0-9_]*(\[[0-9]+\])?|\[[0-9]+\])$")
def pretty_name(id):
if public_name_re.match(id):
return id.lstrip("\\")
else:
return id
def pretty_path(path):
out = ""
for name in path:
name = pretty_name(name)
if name.startswith("["):
out += name
continue
if out:
out += "."
if name.startswith("\\") or name.startswith("$"):
out += name + " "
else:
out += name
return out
@dataclass
class SbyProperty:
class Type(Enum):
ASSUME = auto()
ASSERT = auto()
COVER = auto()
LIVE = auto()
FAIR = auto()
def __str__(self):
return self.name
@classmethod
def from_cell(c, name):
if name == "$assume":
return c.ASSUME
if name == "$assert":
return c.ASSERT
if name == "$cover":
return c.COVER
if name == "$live":
return c.LIVE
if name == "$fair":
return c.FAIR
raise ValueError("Unknown property type: " + name)
@classmethod
def from_flavor(c, name):
if name == "assume":
return c.ASSUME
if name == "assert":
return c.ASSERT
if name == "cover":
return c.COVER
if name == "live":
return c.LIVE
if name == "fair":
return c.FAIR
raise ValueError("Unknown property type: " + name)
@property
def assume_like(self):
return self in [self.ASSUME, self.FAIR]
name: str
path: Tuple[str, ...]
type: Type
location: str
hierarchy: str
status: str = field(default="UNKNOWN")
tracefiles: str = field(default_factory=list)
@property
def tracefile(self):
if self.tracefiles:
return self.tracefiles[0]
else:
return ""
@property
def celltype(self):
return f"${str(self.type).lower()}"
@property
def hdlname(self):
return pretty_path(self.path).rstrip()
def __repr__(self):
return f"SbyProperty<{self.type} {self.name} {self.path} at {self.location}: status={self.status}, tracefile=\"{self.tracefile}\">"
@dataclass
class SbyModule:
name: str
path: Tuple[str, ...]
type: str
submodules: dict = field(default_factory=dict)
properties: list = field(default_factory=list)
def __repr__(self):
return f"SbyModule<{self.name} : {self.type}, submodules={self.submodules}, properties={self.properties}>"
def __iter__(self):
for prop in self.properties:
yield prop
for submod in self.submodules.values():
yield from submod.__iter__()
def get_property_list(self):
return [p for p in self if p.type != p.Type.ASSUME]
def find_property(self, hierarchy, location):
# FIXME: use that RE that works with escaped paths from https://stackoverflow.com/questions/46207665/regex-pattern-to-split-verilog-path-in-different-instances-using-python
path = hierarchy.split('.')
mod = path.pop(0)
if self.name != mod:
raise ValueError(f"{self.name} is not the first module in hierarchical path {hierarchy}.")
try:
mod_hier = self
while path:
mod = path.pop(0)
mod_hier = mod_hier.submodules[mod]
except KeyError:
raise KeyError(f"Could not find {hierarchy} in design hierarchy!")
try:
prop = next(p for p in mod_hier.properties if location in p.location)
except StopIteration:
raise KeyError(f"Could not find assert at {location} in properties list!")
return prop
def find_property_by_cellname(self, cell_name, trans_dict=dict()):
# backends may need to mangle names irreversibly, so allow applying
# the same transformation here
for prop in self:
if cell_name == prop.name.translate(str.maketrans(trans_dict)):
return prop
raise KeyError(f"No such property: {cell_name}")
@dataclass
class SbyDesign:
hierarchy: SbyModule = None
memory_bits: int = 0
forall: bool = False
properties_by_path: dict = field(default_factory=dict)
def pass_unknown_asserts(self):
updated = []
for prop in self.hierarchy:
if prop.type == prop.Type.ASSERT and prop.status == "UNKNOWN":
prop.status = "PASS"
updated.append(prop)
return updated
def cell_path(cell):
path = cell["attributes"].get("hdlname")
if path is None:
if cell["name"].startswith('$'):
return (cell["name"],)
else:
return ("\\" + cell["name"],)
else:
return tuple(f"\\{segment}" for segment in path.split())
def design_hierarchy(filename):
design = SbyDesign(hierarchy=None)
design_json = json.load(filename)
def make_mod_hier(instance_name, module_name, hierarchy="", path=()):
# print(instance_name,":", module_name)
sub_hierarchy=f"{hierarchy}/{instance_name}" if hierarchy else instance_name
mod = SbyModule(name=instance_name, path=path, type=module_name)
for m in design_json["modules"]:
if m["name"] == module_name:
cell_sorts = m["cell_sorts"]
break
else:
raise ValueError(f"Cannot find module {module_name}")
for sort in cell_sorts:
if sort["type"] in ["$assume", "$assert", "$cover", "$live", "$fair"]:
for cell in sort["cells"]:
try:
location = cell["attributes"]["src"]
except KeyError:
location = ""
p = SbyProperty(
name=cell["name"],
path=(*path, *cell_path(cell)),
type=SbyProperty.Type.from_cell(sort["type"]),
location=location,
hierarchy=sub_hierarchy)
mod.properties.append(p)
if sort["type"] == "$check":
for cell in sort["cells"]:
try:
location = cell["attributes"]["src"]
except KeyError:
location = ""
p = SbyProperty(
name=cell["name"],
path=(*path, *cell_path(cell)),
type=SbyProperty.Type.from_flavor(cell["parameters"]["FLAVOR"]),
location=location,
hierarchy=sub_hierarchy)
mod.properties.append(p)
if sort["type"][0] != '$' or sort["type"].startswith("$paramod"):
for cell in sort["cells"]:
mod.submodules[cell["name"]] = make_mod_hier(
cell["name"], sort["type"], sub_hierarchy, (*path, *cell_path(cell)))
if sort["type"] in ["$mem", "$mem_v2"]:
for cell in sort["cells"]:
design.memory_bits += int(cell["parameters"]["WIDTH"], 2) * int(cell["parameters"]["SIZE"], 2)
if sort["type"] in ["$allconst", "$allseq"]:
design.forall = True
return mod
for m in design_json["modules"]:
attrs = m["attributes"]
if "top" in attrs and int(attrs["top"]) == 1:
design.hierarchy = make_mod_hier(m["name"], m["name"], "", (m["name"],))
for prop in design.hierarchy:
design.properties_by_path[prop.path[1:]] = prop
return design
else:
raise ValueError("Cannot find top module")
def main():
import sys
if len(sys.argv) != 2:
print(f"""Usage: {sys.argv[0]} design.json""")
with open(sys.argv[1]) as f:
design = design_hierarchy(f)
print("Design Hierarchy:", design.hierarchy)
for p in design.hierarchy.get_property_list():
print("Property:", p)
print("Memory Bits:", design.memory_bits)
if __name__ == '__main__':
main()