2018-12-17 15:55:30 -07:00
|
|
|
from collections import OrderedDict
|
|
|
|
|
2018-12-15 07:23:42 -07:00
|
|
|
from ..hdl.ast import *
|
|
|
|
from ..hdl.cd import *
|
|
|
|
from ..hdl.ir import *
|
2018-12-22 14:43:46 -07:00
|
|
|
from ..hdl.mem import *
|
2018-12-13 02:19:16 -07:00
|
|
|
from .tools import *
|
2018-12-13 01:09:39 -07:00
|
|
|
|
|
|
|
|
2019-02-14 13:52:42 -07:00
|
|
|
class FragmentGetTestCase(FHDLTestCase):
|
|
|
|
def test_get_wrong(self):
|
|
|
|
with self.assertRaises(AttributeError,
|
|
|
|
msg="Object 'None' cannot be elaborated"):
|
|
|
|
Fragment.get(None, platform=None)
|
|
|
|
|
|
|
|
|
2018-12-26 05:35:27 -07:00
|
|
|
class FragmentGeneratedTestCase(FHDLTestCase):
|
|
|
|
def test_find_subfragment(self):
|
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f1.add_subfragment(f2, "f2")
|
|
|
|
|
|
|
|
self.assertEqual(f1.find_subfragment(0), f2)
|
|
|
|
self.assertEqual(f1.find_subfragment("f2"), f2)
|
|
|
|
|
|
|
|
def test_find_subfragment_wrong(self):
|
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f1.add_subfragment(f2, "f2")
|
|
|
|
|
|
|
|
with self.assertRaises(NameError,
|
|
|
|
msg="No subfragment at index #1"):
|
|
|
|
f1.find_subfragment(1)
|
|
|
|
with self.assertRaises(NameError,
|
|
|
|
msg="No subfragment with name 'fx'"):
|
|
|
|
f1.find_subfragment("fx")
|
|
|
|
|
|
|
|
def test_find_generated(self):
|
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f2.generated["sig"] = sig = Signal()
|
|
|
|
f1.add_subfragment(f2, "f2")
|
|
|
|
|
|
|
|
self.assertEqual(SignalKey(f1.find_generated("f2", "sig")),
|
|
|
|
SignalKey(sig))
|
|
|
|
|
|
|
|
|
2018-12-15 02:26:36 -07:00
|
|
|
class FragmentDriversTestCase(FHDLTestCase):
|
|
|
|
def test_empty(self):
|
|
|
|
f = Fragment()
|
|
|
|
self.assertEqual(list(f.iter_comb()), [])
|
|
|
|
self.assertEqual(list(f.iter_sync()), [])
|
|
|
|
|
|
|
|
|
2018-12-13 02:19:16 -07:00
|
|
|
class FragmentPortsTestCase(FHDLTestCase):
|
2018-12-13 01:09:39 -07:00
|
|
|
def setUp(self):
|
|
|
|
self.s1 = Signal()
|
|
|
|
self.s2 = Signal()
|
|
|
|
self.s3 = Signal()
|
|
|
|
self.c1 = Signal()
|
|
|
|
self.c2 = Signal()
|
|
|
|
self.c3 = Signal()
|
|
|
|
|
2018-12-13 04:25:49 -07:00
|
|
|
def test_empty(self):
|
|
|
|
f = Fragment()
|
2018-12-15 02:26:36 -07:00
|
|
|
self.assertEqual(list(f.iter_ports()), [])
|
2018-12-13 04:25:49 -07:00
|
|
|
|
2018-12-13 06:12:31 -07:00
|
|
|
f._propagate_ports(ports=())
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f.ports, SignalDict([]))
|
2018-12-13 04:25:49 -07:00
|
|
|
|
2018-12-15 02:26:36 -07:00
|
|
|
def test_iter_signals(self):
|
|
|
|
f = Fragment()
|
2018-12-17 15:55:30 -07:00
|
|
|
f.add_ports(self.s1, self.s2, dir="io")
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(SignalSet((self.s1, self.s2)), f.iter_signals())
|
2018-12-15 02:26:36 -07:00
|
|
|
|
2018-12-13 01:09:39 -07:00
|
|
|
def test_self_contained(self):
|
|
|
|
f = Fragment()
|
|
|
|
f.add_statements(
|
|
|
|
self.c1.eq(self.s1),
|
|
|
|
self.s1.eq(self.c1)
|
|
|
|
)
|
2018-12-13 04:01:03 -07:00
|
|
|
|
2018-12-13 06:12:31 -07:00
|
|
|
f._propagate_ports(ports=())
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f.ports, SignalDict([]))
|
2018-12-13 01:09:39 -07:00
|
|
|
|
|
|
|
def test_infer_input(self):
|
|
|
|
f = Fragment()
|
|
|
|
f.add_statements(
|
|
|
|
self.c1.eq(self.s1)
|
|
|
|
)
|
2018-12-13 04:01:03 -07:00
|
|
|
|
2018-12-13 06:12:31 -07:00
|
|
|
f._propagate_ports(ports=())
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f.ports, SignalDict([
|
2018-12-13 06:12:31 -07:00
|
|
|
(self.s1, "i")
|
|
|
|
]))
|
2018-12-13 01:09:39 -07:00
|
|
|
|
|
|
|
def test_request_output(self):
|
|
|
|
f = Fragment()
|
|
|
|
f.add_statements(
|
|
|
|
self.c1.eq(self.s1)
|
|
|
|
)
|
2018-12-13 04:01:03 -07:00
|
|
|
|
2018-12-13 06:12:31 -07:00
|
|
|
f._propagate_ports(ports=(self.c1,))
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f.ports, SignalDict([
|
2018-12-13 06:12:31 -07:00
|
|
|
(self.s1, "i"),
|
|
|
|
(self.c1, "o")
|
|
|
|
]))
|
2018-12-13 01:09:39 -07:00
|
|
|
|
|
|
|
def test_input_in_subfragment(self):
|
|
|
|
f1 = Fragment()
|
|
|
|
f1.add_statements(
|
|
|
|
self.c1.eq(self.s1)
|
|
|
|
)
|
|
|
|
f2 = Fragment()
|
|
|
|
f2.add_statements(
|
|
|
|
self.s1.eq(0)
|
|
|
|
)
|
|
|
|
f1.add_subfragment(f2)
|
2018-12-13 06:12:31 -07:00
|
|
|
f1._propagate_ports(ports=())
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f1.ports, SignalDict())
|
|
|
|
self.assertEqual(f2.ports, SignalDict([
|
2018-12-13 06:12:31 -07:00
|
|
|
(self.s1, "o"),
|
|
|
|
]))
|
2018-12-13 01:09:39 -07:00
|
|
|
|
2018-12-13 04:50:56 -07:00
|
|
|
def test_input_only_in_subfragment(self):
|
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f2.add_statements(
|
|
|
|
self.c1.eq(self.s1)
|
|
|
|
)
|
|
|
|
f1.add_subfragment(f2)
|
2018-12-13 06:12:31 -07:00
|
|
|
f1._propagate_ports(ports=())
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f1.ports, SignalDict([
|
2018-12-13 06:12:31 -07:00
|
|
|
(self.s1, "i"),
|
|
|
|
]))
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f2.ports, SignalDict([
|
2018-12-13 06:12:31 -07:00
|
|
|
(self.s1, "i"),
|
|
|
|
]))
|
2018-12-13 04:50:56 -07:00
|
|
|
|
2018-12-13 01:09:39 -07:00
|
|
|
def test_output_from_subfragment(self):
|
|
|
|
f1 = Fragment()
|
|
|
|
f1.add_statements(
|
|
|
|
self.c1.eq(0)
|
|
|
|
)
|
|
|
|
f2 = Fragment()
|
|
|
|
f2.add_statements(
|
|
|
|
self.c2.eq(1)
|
|
|
|
)
|
|
|
|
f1.add_subfragment(f2)
|
2018-12-13 04:01:03 -07:00
|
|
|
|
2018-12-13 06:12:31 -07:00
|
|
|
f1._propagate_ports(ports=(self.c2,))
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f1.ports, SignalDict([
|
2018-12-13 06:12:31 -07:00
|
|
|
(self.c2, "o"),
|
|
|
|
]))
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f2.ports, SignalDict([
|
2018-12-13 06:12:31 -07:00
|
|
|
(self.c2, "o"),
|
|
|
|
]))
|
2018-12-13 04:01:03 -07:00
|
|
|
|
2018-12-21 16:53:18 -07:00
|
|
|
def test_input_output_sibling(self):
|
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f2.add_statements(
|
|
|
|
self.c1.eq(self.c2)
|
|
|
|
)
|
|
|
|
f1.add_subfragment(f2)
|
|
|
|
f3 = Fragment()
|
|
|
|
f3.add_statements(
|
|
|
|
self.c2.eq(0)
|
|
|
|
)
|
|
|
|
f3.add_driver(self.c2)
|
|
|
|
f1.add_subfragment(f3)
|
|
|
|
|
|
|
|
f1._propagate_ports(ports=())
|
|
|
|
self.assertEqual(f1.ports, SignalDict())
|
|
|
|
|
2018-12-21 17:31:31 -07:00
|
|
|
def test_output_input_sibling(self):
|
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f2.add_statements(
|
|
|
|
self.c2.eq(0)
|
|
|
|
)
|
|
|
|
f2.add_driver(self.c2)
|
|
|
|
f1.add_subfragment(f2)
|
|
|
|
f3 = Fragment()
|
|
|
|
f3.add_statements(
|
|
|
|
self.c1.eq(self.c2)
|
|
|
|
)
|
|
|
|
f1.add_subfragment(f3)
|
|
|
|
|
|
|
|
f1._propagate_ports(ports=())
|
|
|
|
self.assertEqual(f1.ports, SignalDict())
|
|
|
|
|
2018-12-13 04:01:03 -07:00
|
|
|
def test_input_cd(self):
|
|
|
|
sync = ClockDomain()
|
|
|
|
f = Fragment()
|
|
|
|
f.add_statements(
|
|
|
|
self.c1.eq(self.s1)
|
|
|
|
)
|
|
|
|
f.add_domains(sync)
|
2018-12-14 13:58:29 -07:00
|
|
|
f.add_driver(self.c1, "sync")
|
2018-12-13 04:01:03 -07:00
|
|
|
|
2018-12-13 06:12:31 -07:00
|
|
|
f._propagate_ports(ports=())
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f.ports, SignalDict([
|
2018-12-13 06:12:31 -07:00
|
|
|
(self.s1, "i"),
|
|
|
|
(sync.clk, "i"),
|
|
|
|
(sync.rst, "i"),
|
|
|
|
]))
|
2018-12-13 04:01:03 -07:00
|
|
|
|
|
|
|
def test_input_cd_reset_less(self):
|
|
|
|
sync = ClockDomain(reset_less=True)
|
|
|
|
f = Fragment()
|
|
|
|
f.add_statements(
|
|
|
|
self.c1.eq(self.s1)
|
|
|
|
)
|
|
|
|
f.add_domains(sync)
|
2018-12-14 13:58:29 -07:00
|
|
|
f.add_driver(self.c1, "sync")
|
2018-12-13 04:01:03 -07:00
|
|
|
|
2018-12-13 06:12:31 -07:00
|
|
|
f._propagate_ports(ports=())
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(f.ports, SignalDict([
|
2018-12-13 06:12:31 -07:00
|
|
|
(self.s1, "i"),
|
|
|
|
(sync.clk, "i"),
|
|
|
|
]))
|
2018-12-13 04:01:03 -07:00
|
|
|
|
2018-12-17 15:55:30 -07:00
|
|
|
def test_inout(self):
|
|
|
|
s = Signal()
|
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f2.add_ports(s, dir="io")
|
|
|
|
f1.add_subfragment(f2)
|
|
|
|
|
|
|
|
f1._propagate_ports(ports=())
|
|
|
|
self.assertEqual(f1.ports, SignalDict([
|
|
|
|
(s, "io")
|
|
|
|
]))
|
|
|
|
|
2018-12-13 04:01:03 -07:00
|
|
|
|
|
|
|
class FragmentDomainsTestCase(FHDLTestCase):
|
2018-12-15 02:26:36 -07:00
|
|
|
def test_iter_signals(self):
|
|
|
|
cd1 = ClockDomain()
|
|
|
|
cd2 = ClockDomain(reset_less=True)
|
|
|
|
s1 = Signal()
|
|
|
|
s2 = Signal()
|
|
|
|
|
|
|
|
f = Fragment()
|
|
|
|
f.add_domains(cd1, cd2)
|
|
|
|
f.add_driver(s1, "cd1")
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(SignalSet((cd1.clk, cd1.rst, s1)), f.iter_signals())
|
2018-12-15 02:26:36 -07:00
|
|
|
f.add_driver(s2, "cd2")
|
2018-12-17 10:21:12 -07:00
|
|
|
self.assertEqual(SignalSet((cd1.clk, cd1.rst, cd2.clk, s1, s2)), f.iter_signals())
|
2018-12-15 02:26:36 -07:00
|
|
|
|
2018-12-13 04:01:03 -07:00
|
|
|
def test_propagate_up(self):
|
|
|
|
cd = ClockDomain()
|
|
|
|
|
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f1.add_subfragment(f2)
|
|
|
|
f2.add_domains(cd)
|
|
|
|
|
|
|
|
f1._propagate_domains_up()
|
|
|
|
self.assertEqual(f1.domains, {"cd": cd})
|
|
|
|
|
|
|
|
def test_domain_conflict(self):
|
|
|
|
cda = ClockDomain("sync")
|
|
|
|
cdb = ClockDomain("sync")
|
|
|
|
|
|
|
|
fa = Fragment()
|
|
|
|
fa.add_domains(cda)
|
|
|
|
fb = Fragment()
|
|
|
|
fb.add_domains(cdb)
|
|
|
|
f = Fragment()
|
|
|
|
f.add_subfragment(fa, "a")
|
|
|
|
f.add_subfragment(fb, "b")
|
|
|
|
|
|
|
|
f._propagate_domains_up()
|
|
|
|
self.assertEqual(f.domains, {"a_sync": cda, "b_sync": cdb})
|
|
|
|
(fa, _), (fb, _) = f.subfragments
|
|
|
|
self.assertEqual(fa.domains, {"a_sync": cda})
|
|
|
|
self.assertEqual(fb.domains, {"b_sync": cdb})
|
|
|
|
|
|
|
|
def test_domain_conflict_anon(self):
|
|
|
|
cda = ClockDomain("sync")
|
|
|
|
cdb = ClockDomain("sync")
|
|
|
|
|
|
|
|
fa = Fragment()
|
|
|
|
fa.add_domains(cda)
|
|
|
|
fb = Fragment()
|
|
|
|
fb.add_domains(cdb)
|
|
|
|
f = Fragment()
|
|
|
|
f.add_subfragment(fa, "a")
|
|
|
|
f.add_subfragment(fb)
|
|
|
|
|
|
|
|
with self.assertRaises(DomainError,
|
|
|
|
msg="Domain 'sync' is defined by subfragments 'a', <unnamed #1> of fragment "
|
|
|
|
"'top'; it is necessary to either rename subfragment domains explicitly, "
|
|
|
|
"or give names to subfragments"):
|
|
|
|
f._propagate_domains_up()
|
|
|
|
|
|
|
|
def test_domain_conflict_name(self):
|
|
|
|
cda = ClockDomain("sync")
|
|
|
|
cdb = ClockDomain("sync")
|
|
|
|
|
|
|
|
fa = Fragment()
|
|
|
|
fa.add_domains(cda)
|
|
|
|
fb = Fragment()
|
|
|
|
fb.add_domains(cdb)
|
|
|
|
f = Fragment()
|
|
|
|
f.add_subfragment(fa, "x")
|
|
|
|
f.add_subfragment(fb, "x")
|
|
|
|
|
|
|
|
with self.assertRaises(DomainError,
|
|
|
|
msg="Domain 'sync' is defined by subfragments #0, #1 of fragment 'top', some "
|
|
|
|
"of which have identical names; it is necessary to either rename subfragment "
|
|
|
|
"domains explicitly, or give distinct names to subfragments"):
|
|
|
|
f._propagate_domains_up()
|
|
|
|
|
|
|
|
def test_propagate_down(self):
|
|
|
|
cd = ClockDomain()
|
|
|
|
|
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f1.add_domains(cd)
|
|
|
|
f1.add_subfragment(f2)
|
|
|
|
|
|
|
|
f1._propagate_domains_down()
|
|
|
|
self.assertEqual(f2.domains, {"cd": cd})
|
|
|
|
|
|
|
|
def test_propagate_down_idempotent(self):
|
|
|
|
cd = ClockDomain()
|
|
|
|
|
|
|
|
f1 = Fragment()
|
|
|
|
f1.add_domains(cd)
|
|
|
|
f2 = Fragment()
|
|
|
|
f2.add_domains(cd)
|
|
|
|
f1.add_subfragment(f2)
|
|
|
|
|
|
|
|
f1._propagate_domains_down()
|
|
|
|
self.assertEqual(f1.domains, {"cd": cd})
|
|
|
|
self.assertEqual(f2.domains, {"cd": cd})
|
|
|
|
|
|
|
|
def test_propagate(self):
|
|
|
|
cd = ClockDomain()
|
|
|
|
|
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f1.add_domains(cd)
|
|
|
|
f1.add_subfragment(f2)
|
|
|
|
|
2018-12-13 07:33:39 -07:00
|
|
|
f1._propagate_domains(ensure_sync_exists=False)
|
2018-12-13 04:01:03 -07:00
|
|
|
self.assertEqual(f1.domains, {"cd": cd})
|
|
|
|
self.assertEqual(f2.domains, {"cd": cd})
|
|
|
|
|
2018-12-13 07:33:39 -07:00
|
|
|
def test_propagate_ensure_sync(self):
|
2018-12-13 04:01:03 -07:00
|
|
|
f1 = Fragment()
|
|
|
|
f2 = Fragment()
|
|
|
|
f1.add_subfragment(f2)
|
|
|
|
|
|
|
|
f1._propagate_domains(ensure_sync_exists=True)
|
|
|
|
self.assertEqual(f1.domains.keys(), {"sync"})
|
|
|
|
self.assertEqual(f2.domains.keys(), {"sync"})
|
|
|
|
self.assertEqual(f1.domains["sync"], f2.domains["sync"])
|
2018-12-14 15:47:58 -07:00
|
|
|
|
|
|
|
|
2018-12-22 14:43:46 -07:00
|
|
|
class FragmentHierarchyConflictTestCase(FHDLTestCase):
|
2018-12-14 15:47:58 -07:00
|
|
|
def setUp_self_sub(self):
|
|
|
|
self.s1 = Signal()
|
|
|
|
self.c1 = Signal()
|
|
|
|
self.c2 = Signal()
|
|
|
|
|
|
|
|
self.f1 = Fragment()
|
|
|
|
self.f1.add_statements(self.c1.eq(0))
|
|
|
|
self.f1.add_driver(self.s1)
|
|
|
|
self.f1.add_driver(self.c1, "sync")
|
|
|
|
|
|
|
|
self.f1a = Fragment()
|
|
|
|
self.f1.add_subfragment(self.f1a, "f1a")
|
|
|
|
|
|
|
|
self.f2 = Fragment()
|
|
|
|
self.f2.add_statements(self.c2.eq(1))
|
|
|
|
self.f2.add_driver(self.s1)
|
|
|
|
self.f2.add_driver(self.c2, "sync")
|
|
|
|
self.f1.add_subfragment(self.f2)
|
|
|
|
|
|
|
|
self.f1b = Fragment()
|
|
|
|
self.f1.add_subfragment(self.f1b, "f1b")
|
|
|
|
|
|
|
|
self.f2a = Fragment()
|
|
|
|
self.f2.add_subfragment(self.f2a, "f2a")
|
|
|
|
|
|
|
|
def test_conflict_self_sub(self):
|
|
|
|
self.setUp_self_sub()
|
|
|
|
|
2018-12-22 14:43:46 -07:00
|
|
|
self.f1._resolve_hierarchy_conflicts(mode="silent")
|
2018-12-14 15:47:58 -07:00
|
|
|
self.assertEqual(self.f1.subfragments, [
|
|
|
|
(self.f1a, "f1a"),
|
|
|
|
(self.f1b, "f1b"),
|
|
|
|
(self.f2a, "f2a"),
|
|
|
|
])
|
|
|
|
self.assertRepr(self.f1.statements, """
|
|
|
|
(
|
|
|
|
(eq (sig c1) (const 1'd0))
|
|
|
|
(eq (sig c2) (const 1'd1))
|
|
|
|
)
|
|
|
|
""")
|
|
|
|
self.assertEqual(self.f1.drivers, {
|
2018-12-17 10:21:12 -07:00
|
|
|
None: SignalSet((self.s1,)),
|
|
|
|
"sync": SignalSet((self.c1, self.c2)),
|
2018-12-14 15:47:58 -07:00
|
|
|
})
|
|
|
|
|
|
|
|
def test_conflict_self_sub_error(self):
|
|
|
|
self.setUp_self_sub()
|
|
|
|
|
|
|
|
with self.assertRaises(DriverConflict,
|
|
|
|
msg="Signal '(sig s1)' is driven from multiple fragments: top, top.<unnamed #1>"):
|
2018-12-22 14:43:46 -07:00
|
|
|
self.f1._resolve_hierarchy_conflicts(mode="error")
|
2018-12-14 15:47:58 -07:00
|
|
|
|
|
|
|
def test_conflict_self_sub_warning(self):
|
|
|
|
self.setUp_self_sub()
|
|
|
|
|
|
|
|
with self.assertWarns(DriverConflict,
|
|
|
|
msg="Signal '(sig s1)' is driven from multiple fragments: top, top.<unnamed #1>; "
|
|
|
|
"hierarchy will be flattened"):
|
2018-12-22 14:43:46 -07:00
|
|
|
self.f1._resolve_hierarchy_conflicts(mode="warn")
|
2018-12-14 15:47:58 -07:00
|
|
|
|
|
|
|
def setUp_sub_sub(self):
|
|
|
|
self.s1 = Signal()
|
|
|
|
self.c1 = Signal()
|
|
|
|
self.c2 = Signal()
|
|
|
|
|
|
|
|
self.f1 = Fragment()
|
|
|
|
|
|
|
|
self.f2 = Fragment()
|
|
|
|
self.f2.add_driver(self.s1)
|
|
|
|
self.f2.add_statements(self.c1.eq(0))
|
|
|
|
self.f1.add_subfragment(self.f2)
|
|
|
|
|
|
|
|
self.f3 = Fragment()
|
|
|
|
self.f3.add_driver(self.s1)
|
|
|
|
self.f3.add_statements(self.c2.eq(1))
|
|
|
|
self.f1.add_subfragment(self.f3)
|
|
|
|
|
|
|
|
def test_conflict_sub_sub(self):
|
|
|
|
self.setUp_sub_sub()
|
|
|
|
|
2018-12-22 14:43:46 -07:00
|
|
|
self.f1._resolve_hierarchy_conflicts(mode="silent")
|
2018-12-14 15:47:58 -07:00
|
|
|
self.assertEqual(self.f1.subfragments, [])
|
|
|
|
self.assertRepr(self.f1.statements, """
|
|
|
|
(
|
|
|
|
(eq (sig c1) (const 1'd0))
|
|
|
|
(eq (sig c2) (const 1'd1))
|
|
|
|
)
|
|
|
|
""")
|
|
|
|
|
|
|
|
def setUp_self_subsub(self):
|
|
|
|
self.s1 = Signal()
|
|
|
|
self.c1 = Signal()
|
|
|
|
self.c2 = Signal()
|
|
|
|
|
|
|
|
self.f1 = Fragment()
|
|
|
|
self.f1.add_driver(self.s1)
|
|
|
|
|
|
|
|
self.f2 = Fragment()
|
|
|
|
self.f2.add_statements(self.c1.eq(0))
|
|
|
|
self.f1.add_subfragment(self.f2)
|
|
|
|
|
|
|
|
self.f3 = Fragment()
|
|
|
|
self.f3.add_driver(self.s1)
|
|
|
|
self.f3.add_statements(self.c2.eq(1))
|
|
|
|
self.f2.add_subfragment(self.f3)
|
|
|
|
|
|
|
|
def test_conflict_self_subsub(self):
|
|
|
|
self.setUp_self_subsub()
|
|
|
|
|
2018-12-22 14:43:46 -07:00
|
|
|
self.f1._resolve_hierarchy_conflicts(mode="silent")
|
2018-12-14 15:47:58 -07:00
|
|
|
self.assertEqual(self.f1.subfragments, [])
|
|
|
|
self.assertRepr(self.f1.statements, """
|
|
|
|
(
|
|
|
|
(eq (sig c1) (const 1'd0))
|
|
|
|
(eq (sig c2) (const 1'd1))
|
|
|
|
)
|
|
|
|
""")
|
2018-12-17 15:55:30 -07:00
|
|
|
|
2018-12-22 14:43:46 -07:00
|
|
|
def setUp_memory(self):
|
|
|
|
self.m = Memory(width=8, depth=4)
|
2019-01-25 19:31:12 -07:00
|
|
|
self.fr = self.m.read_port().elaborate(platform=None)
|
|
|
|
self.fw = self.m.write_port().elaborate(platform=None)
|
2018-12-22 14:43:46 -07:00
|
|
|
self.f1 = Fragment()
|
|
|
|
self.f2 = Fragment()
|
|
|
|
self.f2.add_subfragment(self.fr)
|
|
|
|
self.f1.add_subfragment(self.f2)
|
|
|
|
self.f3 = Fragment()
|
|
|
|
self.f3.add_subfragment(self.fw)
|
|
|
|
self.f1.add_subfragment(self.f3)
|
|
|
|
|
|
|
|
def test_conflict_memory(self):
|
|
|
|
self.setUp_memory()
|
|
|
|
|
|
|
|
self.f1._resolve_hierarchy_conflicts(mode="silent")
|
|
|
|
self.assertEqual(self.f1.subfragments, [
|
|
|
|
(self.fr, None),
|
|
|
|
(self.fw, None),
|
|
|
|
])
|
|
|
|
|
|
|
|
def test_conflict_memory_error(self):
|
|
|
|
self.setUp_memory()
|
|
|
|
|
|
|
|
with self.assertRaises(DriverConflict,
|
|
|
|
msg="Memory 'm' is accessed from multiple fragments: top.<unnamed #0>, "
|
|
|
|
"top.<unnamed #1>"):
|
|
|
|
self.f1._resolve_hierarchy_conflicts(mode="error")
|
|
|
|
|
|
|
|
def test_conflict_memory_warning(self):
|
|
|
|
self.setUp_memory()
|
|
|
|
|
|
|
|
with self.assertWarns(DriverConflict,
|
|
|
|
msg="Memory 'm' is accessed from multiple fragments: top.<unnamed #0>, "
|
|
|
|
"top.<unnamed #1>; hierarchy will be flattened"):
|
|
|
|
self.f1._resolve_hierarchy_conflicts(mode="warn")
|
|
|
|
|
2019-01-14 10:04:23 -07:00
|
|
|
def test_explicit_flatten(self):
|
|
|
|
self.f1 = Fragment()
|
|
|
|
self.f2 = Fragment()
|
|
|
|
self.f2.flatten = True
|
|
|
|
self.f1.add_subfragment(self.f2)
|
|
|
|
|
|
|
|
self.f1._resolve_hierarchy_conflicts(mode="silent")
|
|
|
|
self.assertEqual(self.f1.subfragments, [])
|
|
|
|
|
2018-12-17 15:55:30 -07:00
|
|
|
|
|
|
|
class InstanceTestCase(FHDLTestCase):
|
2018-12-20 21:03:03 -07:00
|
|
|
def setUp_cpu(self):
|
|
|
|
self.rst = Signal()
|
|
|
|
self.stb = Signal()
|
|
|
|
self.pins = Signal(8)
|
|
|
|
self.inst = Instance("cpu",
|
2018-12-20 16:38:01 -07:00
|
|
|
p_RESET=0x1234,
|
|
|
|
i_clk=ClockSignal(),
|
2018-12-20 21:03:03 -07:00
|
|
|
i_rst=self.rst,
|
|
|
|
o_stb=self.stb,
|
|
|
|
io_pins=self.pins
|
2018-12-20 16:38:01 -07:00
|
|
|
)
|
2018-12-20 21:03:03 -07:00
|
|
|
|
|
|
|
def test_init(self):
|
|
|
|
self.setUp_cpu()
|
|
|
|
f = self.inst
|
|
|
|
self.assertEqual(f.type, "cpu")
|
|
|
|
self.assertEqual(f.parameters, OrderedDict([("RESET", 0x1234)]))
|
|
|
|
self.assertEqual(list(f.named_ports.keys()), ["clk", "rst", "stb", "pins"])
|
|
|
|
self.assertEqual(f.ports, SignalDict([
|
|
|
|
(self.stb, "o"),
|
|
|
|
(self.pins, "io"),
|
|
|
|
]))
|
|
|
|
|
|
|
|
def test_prepare(self):
|
|
|
|
self.setUp_cpu()
|
|
|
|
f = self.inst.prepare()
|
|
|
|
clk = f.domains["sync"].clk
|
|
|
|
self.assertEqual(f.type, "cpu")
|
|
|
|
self.assertEqual(f.parameters, OrderedDict([("RESET", 0x1234)]))
|
|
|
|
self.assertEqual(list(f.named_ports.keys()), ["clk", "rst", "stb", "pins"])
|
|
|
|
self.assertEqual(f.ports, SignalDict([
|
|
|
|
(clk, "i"),
|
|
|
|
(self.rst, "i"),
|
|
|
|
(self.stb, "o"),
|
|
|
|
(self.pins, "io"),
|
2018-12-17 15:55:30 -07:00
|
|
|
]))
|