2018-12-13 02:19:16 -07:00
|
|
|
from ..fhdl.ast import *
|
2018-12-13 04:01:03 -07:00
|
|
|
from ..fhdl.cd import *
|
2018-12-13 02:19:16 -07:00
|
|
|
from ..fhdl.ir import *
|
|
|
|
from .tools import *
|
2018-12-13 01:09:39 -07:00
|
|
|
|
|
|
|
|
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-13 06:12:31 -07:00
|
|
|
f._propagate_ports(ports=())
|
|
|
|
self.assertEqual(f.ports, ValueDict([]))
|
2018-12-13 04:25:49 -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=())
|
|
|
|
self.assertEqual(f.ports, ValueDict([]))
|
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=())
|
|
|
|
self.assertEqual(f.ports, ValueDict([
|
|
|
|
(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,))
|
|
|
|
self.assertEqual(f.ports, ValueDict([
|
|
|
|
(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=())
|
|
|
|
self.assertEqual(f1.ports, ValueDict())
|
|
|
|
self.assertEqual(f2.ports, ValueDict([
|
|
|
|
(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=())
|
|
|
|
self.assertEqual(f1.ports, ValueDict([
|
|
|
|
(self.s1, "i"),
|
|
|
|
]))
|
|
|
|
self.assertEqual(f2.ports, ValueDict([
|
|
|
|
(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,))
|
|
|
|
self.assertEqual(f1.ports, ValueDict([
|
|
|
|
(self.c2, "o"),
|
|
|
|
]))
|
|
|
|
self.assertEqual(f2.ports, ValueDict([
|
|
|
|
(self.c2, "o"),
|
|
|
|
]))
|
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)
|
|
|
|
f.drive(self.c1, "sync")
|
|
|
|
|
2018-12-13 06:12:31 -07:00
|
|
|
f._propagate_ports(ports=())
|
|
|
|
self.assertEqual(f.ports, ValueDict([
|
|
|
|
(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)
|
|
|
|
f.drive(self.c1, "sync")
|
|
|
|
|
2018-12-13 06:12:31 -07:00
|
|
|
f._propagate_ports(ports=())
|
|
|
|
self.assertEqual(f.ports, ValueDict([
|
|
|
|
(self.s1, "i"),
|
|
|
|
(sync.clk, "i"),
|
|
|
|
]))
|
2018-12-13 04:01:03 -07:00
|
|
|
|
|
|
|
|
|
|
|
class FragmentDomainsTestCase(FHDLTestCase):
|
|
|
|
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"])
|