# amaranth: UnusedElaboratable=no import unittest from types import SimpleNamespace as NS from amaranth import * from amaranth.lib import data, enum from amaranth.lib.wiring import Flow, In, Out, Member from amaranth.lib.wiring import SignatureError, SignatureMembers, FlippedSignatureMembers from amaranth.lib.wiring import Signature, FlippedSignature, PureInterface, FlippedInterface from amaranth.lib.wiring import Component, ComponentMetadata, InvalidMetadata from amaranth.lib.wiring import ConnectionError, connect, flipped from amaranth.lib.meta import Annotation class FlowTestCase(unittest.TestCase): def test_flow_call(self): self.assertEqual(In(unsigned(1)), Member(Flow.In, unsigned(1))) self.assertEqual(Out(5), Member(Flow.Out, 5)) def test_flow_repr(self): self.assertEqual(repr(Flow.In), "In") self.assertEqual(repr(Flow.Out), "Out") def test_flow_str(self): self.assertEqual(str(Flow.In), "In") self.assertEqual(str(Flow.Out), "Out") def test_flow_value(self): self.assertEqual(Flow.In.value, "In") self.assertEqual(Flow.Out.value, "Out") class MemberTestCase(unittest.TestCase): def test_port_member(self): member = Member(In, unsigned(1)) self.assertEqual(member.flow, In) self.assertEqual(member.is_port, True) self.assertEqual(member.shape, unsigned(1)) self.assertEqual(member.init, None) self.assertEqual(member.is_signature, False) with self.assertRaisesRegex(AttributeError, r"^A port member does not have a signature$"): member.signature self.assertEqual(member.dimensions, ()) self.assertEqual(repr(member), "In(unsigned(1))") def test_port_member_wrong(self): with self.assertRaisesRegex(TypeError, r"^Port member description must be a shape-castable object or a signature, " r"not 'whatever'$"): Member(In, "whatever") def test_port_member_init(self): member = Member(Out, unsigned(1), init=1) self.assertEqual(member.flow, Out) self.assertEqual(member.shape, unsigned(1)) self.assertEqual(member.init, 1) self.assertEqual(repr(member._init_as_const), repr(Const(1, 1))) self.assertEqual(repr(member), "Out(unsigned(1), init=1)") def test_port_member_init_wrong(self): with self.assertRaisesRegex(TypeError, r"^Port member initial value 'no' is not a valid constant initializer " r"for unsigned\(1\)$"): Member(In, 1, init="no") def test_port_member_init_shape_castable(self): layout = data.StructLayout({"a": 32}) member = Member(In, layout, init={"a": 1}) self.assertEqual(member.flow, In) self.assertEqual(member.shape, layout) self.assertEqual(member.init, {"a": 1}) self.assertEqual(repr(member), "In(StructLayout({'a': 32}), init={'a': 1})") def test_port_member_init_shape_castable_wrong(self): with self.assertRaisesRegex(TypeError, r"^Port member initial value 'no' is not a valid constant initializer " r"for StructLayout\({'a': 32}\)$"): Member(In, data.StructLayout({"a": 32}), init="no") def test_port_member_reset(self): with self.assertWarnsRegex(DeprecationWarning, r"^`reset=` is deprecated, use `init=` instead$"): member = Member(Out, unsigned(1), reset=1) self.assertEqual(member.flow, Out) self.assertEqual(member.shape, unsigned(1)) self.assertEqual(member.init, 1) self.assertEqual(repr(member._init_as_const), repr(Const(1, 1))) self.assertEqual(repr(member), "Out(unsigned(1), init=1)") with self.assertWarnsRegex(DeprecationWarning, r"^`Member.reset` is deprecated, use `Member.init` instead$"): self.assertEqual(member.reset, 1) with self.assertWarnsRegex(DeprecationWarning, r"^`reset=` is deprecated, use `init=` instead$"): member = Out(unsigned(1), reset=1) self.assertEqual(member.init,1) def test_port_member_reset_wrong(self): with self.assertRaisesRegex(ValueError, r"^Cannot specify both `reset` and `init`$"): Member(Out, unsigned(1), reset=1, init=1) with self.assertRaisesRegex(ValueError, r"^Cannot specify both `reset` and `init`$"): Out(unsigned(1), reset=1, init=1) def test_signature_member_out(self): sig = Signature({"data": Out(unsigned(32))}) member = Member(Out, sig) self.assertEqual(member.flow, Out) self.assertEqual(member.is_port, False) with self.assertRaisesRegex(AttributeError, r"^A signature member does not have a shape$"): member.shape with self.assertRaisesRegex(AttributeError, r"^A signature member does not have an initial value$"): member.init self.assertEqual(member.is_signature, True) self.assertEqual(member.signature, sig) self.assertEqual(member.dimensions, ()) self.assertEqual(repr(member), "Out(Signature({'data': Out(unsigned(32))}))") def test_signature_member_in(self): sig = Signature({"data": In(unsigned(32))}) member = Member(In, sig) self.assertEqual(member.flow, In) self.assertEqual(member.is_port, False) with self.assertRaisesRegex(AttributeError, r"^A signature member does not have a shape$"): member.shape with self.assertRaisesRegex(AttributeError, r"^A signature member does not have an initial value$"): member.init self.assertEqual(member.is_signature, True) self.assertEqual(member.signature, sig.flip()) self.assertEqual(member.dimensions, ()) self.assertEqual(repr(member), "In(Signature({'data': In(unsigned(32))}))") def test_signature_member_wrong(self): with self.assertRaisesRegex(ValueError, r"^A signature member cannot have an initial value$"): Member(In, Signature({}), init=1) def test_array(self): array_2 = Member(In, unsigned(1)).array(2) self.assertEqual(array_2.dimensions, (2,)) self.assertEqual(repr(array_2), "In(unsigned(1)).array(2)") array_2_3 = Member(In, unsigned(1)).array(2, 3) self.assertEqual(array_2_3.dimensions, (2, 3)) self.assertEqual(repr(array_2_3), "In(unsigned(1)).array(2, 3)") array_2_3_chained = Member(In, unsigned(1)).array(3).array(2) self.assertEqual(array_2_3_chained.dimensions, (2, 3)) self.assertEqual(repr(array_2_3_chained), "In(unsigned(1)).array(2, 3)") def test_array_wrong(self): with self.assertRaisesRegex(TypeError, r"^Member array dimensions must be non-negative integers, not -1$"): Member(In, unsigned(1)).array(-1) with self.assertRaisesRegex(TypeError, r"^Member array dimensions must be non-negative integers, not 'what'$"): Member(In, unsigned(1)).array("what") def test_flip(self): self.assertEqual(In(1).flip(), Out(1)) self.assertEqual(Out(1).flip(), In(1)) def test_equality(self): self.assertEqual(In(1), In(1)) self.assertNotEqual(In(1), Out(1)) self.assertNotEqual(In(1), In(1, init=1)) self.assertNotEqual(In(1), In(1, init=0)) self.assertEqual(In(1), In(1).array()) self.assertNotEqual(In(1), In(1).array(1)) sig = Signature({}) self.assertEqual(In(sig), In(sig)) self.assertNotEqual(In(1), In(Signature({}))) class SignatureMembersTestCase(unittest.TestCase): def test_contains(self): self.assertNotIn("a", SignatureMembers()) self.assertIn("a", SignatureMembers({"a": In(1)})) def test_getitem(self): members = SignatureMembers({"a": In(1)}) self.assertEqual(members["a"], In(1)) def test_getitem_missing(self): members = SignatureMembers({"a": In(1)}) with self.assertRaisesRegex(SignatureError, r"^Member 'b' is not a part of the signature$"): members["b"] def test_getitem_wrong(self): members = SignatureMembers({"a": In(1)}) with self.assertRaisesRegex(TypeError, r"^Member name must be a string, not 1$"): members[1] with self.assertRaisesRegex(NameError, r"^Member name '_a' must be a valid, public Python attribute name$"): members["_a"] with self.assertRaisesRegex(NameError, r"^Member name cannot be 'signature'$"): members["signature"] def test_setitem(self): members = SignatureMembers() with self.assertRaisesRegex(SignatureError, r"^Members cannot be added to a signature once constructed$"): members["a"] = In(1) def test_delitem(self): members = SignatureMembers() with self.assertRaisesRegex(SignatureError, r"^Members cannot be removed from a signature$"): del members["a"] def test_iter_len(self): members = SignatureMembers() self.assertEqual(list(iter(members)), []) self.assertEqual(len(members), 0) members = SignatureMembers({"a": In(1)}) self.assertEqual(list(iter(members)), ["a"]) self.assertEqual(len(members), 1) def test_iter_insertion_order(self): self.assertEqual(list(iter(SignatureMembers({"a": In(1), "b": Out(1)}))), ["a", "b"]) self.assertEqual(list(iter(SignatureMembers({"b": In(1), "a": Out(1)}))), ["b", "a"]) def test_flatten(self): sig = Signature({ "b": Out(1), "c": In(2) }) members = SignatureMembers({ "a": In(1), "s": Out(sig) }) self.assertEqual(list(members.flatten()), [ (("a",), In(1)), (("s",), Out(sig)), (("s", "b"), Out(1)), (("s", "c"), In(2)), ]) def test_create(self): sig = Signature({ "b": Out(2) }) members = SignatureMembers({ "a": In(1), "s": Out(sig) }) attrs = members.create() self.assertEqual(list(attrs.keys()), ["a", "s"]) self.assertIsInstance(attrs["a"], Signal) self.assertEqual(attrs["a"].shape(), unsigned(1)) self.assertEqual(attrs["a"].name, "attrs__a") self.assertEqual(attrs["s"].b.shape(), unsigned(2)) self.assertEqual(attrs["s"].b.name, "attrs__s__b") def test_create_init(self): members = SignatureMembers({ "a": In(1, init=1), }) attrs = members.create() self.assertEqual(attrs["a"].init, 1) def test_create_tuple(self): sig = SignatureMembers({ "a": Out(1).array(2, 3) }) members = sig.create() self.assertEqual(len(members["a"]), 2) self.assertEqual(len(members["a"][0]), 3) self.assertEqual(len(members["a"][1]), 3) for x in members["a"]: for y in x: self.assertIsInstance(y, Signal) self.assertEqual(members["a"][1][2].name, "members__a__1__2") def test_create_wrong(self): with self.assertRaisesRegex(TypeError, r"^Value 1 must be a member; did you mean In\(1\) or Out\(1\)\?$"): SignatureMembers({"a": 1}) def test_repr(self): self.assertEqual(repr(SignatureMembers({})), "SignatureMembers({})") self.assertEqual(repr(SignatureMembers({"a": In(1)})), "SignatureMembers({'a': In(1)})") members = SignatureMembers({"b": Out(2)}) self.assertEqual(repr(members), "SignatureMembers({'b': Out(2)})") class FlippedSignatureMembersTestCase(unittest.TestCase): def test_basic(self): members = SignatureMembers({"a": In(1)}) fmembers = members.flip() self.assertIsInstance(fmembers, FlippedSignatureMembers) self.assertIn("a", fmembers) self.assertEqual(fmembers["a"], Out(1)) members = SignatureMembers({"a": In(1), "b": In(2)}) fmembers = members.flip() self.assertEqual(len(fmembers), 2) self.assertEqual(fmembers["b"], Out(2)) self.assertEqual(list(fmembers), ["a", "b"]) members = SignatureMembers({"a": In(1), "b": In(2), "c": Out(2)}) fmembers = members.flip() self.assertEqual(fmembers["c"], In(2)) self.assertIs(fmembers.flip(), members) def test_eq(self): self.assertEqual(SignatureMembers({"a": In(1)}).flip(), SignatureMembers({"a": In(1)}).flip()) self.assertEqual(SignatureMembers({"a": In(1)}).flip(), SignatureMembers({"a": Out(1)})) def test_delitem(self): fmembers = SignatureMembers().flip() with self.assertRaisesRegex(SignatureError, r"^Members cannot be removed from a signature$"): del fmembers["a"] def test_repr(self): fmembers = SignatureMembers({"a": In(1)}).flip() self.assertEqual(repr(fmembers), "SignatureMembers({'a': In(1)}).flip()") class SignatureTestCase(unittest.TestCase): def test_create(self): sig = Signature({"a": In(1)}) self.assertEqual(sig.members, SignatureMembers({"a": In(1)})) def test_annotations_empty(self): sig = Signature({"a": In(1)}) iface = PureInterface(sig) self.assertEqual(sig.annotations(iface), ()) def test_eq(self): self.assertEqual(Signature({"a": In(1)}), Signature({"a": In(1)})) self.assertNotEqual(Signature({"a": In(1)}), Signature({"a": Out(1)})) def test_members_equal_wrong(self): sig = Signature({}) with self.assertRaises(AttributeError): sig.members = SignatureMembers({}) def assertFlattenedSignature(self, actual, expected): for (a_path, a_member, a_value), (b_path, b_member, b_value) in zip(actual, expected): self.assertEqual(a_path, b_path) self.assertEqual(a_member, b_member) self.assertIs(a_value, b_value) def test_flatten(self): sig = Signature({"a": In(1), "b": Out(2).array(2)}) intf = sig.create() self.assertFlattenedSignature(sig.flatten(intf), [ (("a",), In(1), intf.a), (("b", 0), Out(2), intf.b[0]), (("b", 1), Out(2), intf.b[1]) ]) def test_flatten_sig(self): sig = Signature({ "a": Out(Signature({"p": Out(1)})), "b": Out(Signature({"q": In (1)})), "c": In( Signature({"r": Out(1)})), "d": In( Signature({"s": In (1)})), }) intf = sig.create() self.assertFlattenedSignature(sig.flatten(intf), [ (("a", "p"), Out(1), intf.a.p), (("b", "q"), In (1), intf.b.q), (("c", "r"), In (1), intf.c.r), (("d", "s"), Out(1), intf.d.s), ]) def test_is_compliant_signature(self): sig = Signature({}) obj1 = NS() self.assertFalse(sig.is_compliant(obj1)) reasons = [] self.assertFalse(sig.is_compliant(obj1, reasons=reasons)) self.assertEqual(reasons, ["'obj' does not have an attribute 'signature'"]) obj = NS(signature=1) self.assertFalse(sig.is_compliant(obj)) reasons = [] self.assertFalse(sig.is_compliant(obj, reasons=reasons)) self.assertEqual(reasons, ["'obj.signature' is expected to be a signature, but it is a 1"]) obj = NS(signature=Signature({"a": In(1)})) self.assertFalse(sig.is_compliant(obj)) reasons = [] self.assertFalse(sig.is_compliant(obj, reasons=reasons)) self.assertEqual(reasons, [ "'obj.signature' is expected to be equal to this signature, " "Signature({}), but it is a Signature({'a': In(1)})" ]) def assertNotCompliant(self, reason_regex, sig, obj): obj.signature = sig self.assertFalse(sig.is_compliant(obj)) reasons = [] self.assertFalse(sig.is_compliant(obj, reasons=reasons)) self.assertEqual(len(reasons), 1) self.assertRegex(reasons[0], reason_regex) def test_is_compliant(self): self.assertNotCompliant( r"^'obj' does not have an attribute 'a'$", sig=Signature({"a": In(1)}), obj=NS()) self.assertNotCompliant( r"^'obj\.a' is expected to be a tuple or a list, but it is a \(sig \$signal\)$", sig=Signature({"a": In(1).array(2)}), obj=NS(a=Signal())) self.assertNotCompliant( r"^'obj\.a' is expected to have dimension 2, but its length is 1$", sig=Signature({"a": In(1).array(2)}), obj=NS(a=[Signal()])) self.assertNotCompliant( r"^'obj\.a\[0\]' is expected to have dimension 2, but its length is 1$", sig=Signature({"a": In(1).array(1, 2)}), obj=NS(a=[[Signal()]])) self.assertNotCompliant( r"^'obj\.a' is not a value-castable object, but 'foo'$", sig=Signature({"a": In(1)}), obj=NS(a="foo")) self.assertNotCompliant( r"^'obj\.a' is neither a signal nor a constant, but " r"\(\+ \(const 1'd1\) \(const 1'd1\)\)$", sig=Signature({"a": In(1)}), obj=NS(a=Const(1)+1)) self.assertNotCompliant( r"^'obj\.a' is expected to have the shape unsigned\(1\), but " r"it has the shape unsigned\(2\)$", sig=Signature({"a": In(1)}), obj=NS(a=Signal(2))) self.assertNotCompliant( r"^'obj\.a' is expected to have the shape unsigned\(1\), but " r"it has the shape signed\(1\)$", sig=Signature({"a": In(unsigned(1))}), obj=NS(a=Signal(signed(1)))) self.assertNotCompliant( r"^'obj\.a' is expected to have the initial value None, but it has the initial value 1$", sig=Signature({"a": In(1)}), obj=NS(a=Signal(init=1))) self.assertNotCompliant( r"^'obj\.a' is expected to have the initial value 1, but it has the initial value 0$", sig=Signature({"a": In(1, init=1)}), obj=NS(a=Signal(1))) self.assertTrue( Signature({"a": In(1)}).is_compliant( NS(signature=Signature({"a": In(1)}), a=Signal(1, reset_less=True)))) self.assertNotCompliant( r"^'obj\.a' does not have an attribute 'b'$", sig=Signature({"a": Out(Signature({"b": In(1)}))}), obj=NS(a=NS(signature=Signature({"b": In(1)})))) self.assertTrue( Signature({"a": In(1)}).is_compliant( NS(signature=Signature({"a": In(1)}), a=Signal()))) self.assertTrue( Signature({"a": In(1)}).is_compliant( NS(signature=Signature({"a": In(1)}), a=Const(1)))) self.assertTrue( # list Signature({"a": In(1).array(2, 2)}).is_compliant( NS(signature=Signature({"a": In(1).array(2, 2)}), a=[[Const(1), Const(1)], [Signal(), Signal()]]))) self.assertTrue( # tuple Signature({"a": In(1).array(2, 2)}).is_compliant( NS(signature=Signature({"a": In(1).array(2, 2)}), a=((Const(1), Const(1)), (Signal(), Signal()))))) self.assertTrue( # mixed list and tuple Signature({"a": In(1).array(2, 2)}).is_compliant( NS(signature=Signature({"a": In(1).array(2, 2)}), a=[[Const(1), Const(1)], (Signal(), Signal())]))) self.assertTrue( Signature({"a": Out(Signature({"b": In(1)}))}).is_compliant( NS(signature=Signature({"a": Out(Signature({"b": In(1)}))}), a=NS(signature=Signature({"b": In(1)}), b=Signal())))) def test_repr(self): sig = Signature({"a": In(1)}) self.assertEqual(repr(sig), "Signature({'a': In(1)})") def test_repr_subclass(self): class S(Signature): def __init__(self): super().__init__({"a": In(1)}) sig = S() self.assertRegex(repr(sig), r"^<.+\.S object at .+?>$") def test_subclasscheck(self): class S(Signature): pass self.assertTrue(issubclass(FlippedSignature, Signature)) self.assertTrue(issubclass(Signature, Signature)) self.assertTrue(issubclass(FlippedSignature, S)) self.assertTrue(not issubclass(Signature, S)) def test_instancecheck(self): class S(Signature): pass sig = Signature({}) sig2 = S({}) self.assertTrue(isinstance(sig.flip(), Signature)) self.assertTrue(isinstance(sig2.flip(), Signature)) self.assertTrue(not isinstance(sig.flip(), S)) self.assertTrue(isinstance(sig2.flip(), S)) class FlippedSignatureTestCase(unittest.TestCase): def test_create(self): sig = Signature({"a": In(1)}) fsig = sig.flip() self.assertIsInstance(fsig, FlippedSignature) self.assertIsInstance(fsig.members, FlippedSignatureMembers) self.assertIs(fsig.flip(), sig) def test_eq(self): self.assertEqual(Signature({"a": In(1)}).flip(), Signature({"a": In(1)}).flip()) self.assertEqual(Signature({"a": In(1)}).flip(), Signature({"a": Out(1)})) def test_repr(self): sig = Signature({"a": In(1)}).flip() self.assertEqual(repr(sig), "Signature({'a': In(1)}).flip()") def test_getsetdelattr(self): class S(Signature): def __init__(self): super().__init__({}) self.x = 1 def f(self2): self.assertIsInstance(self2, FlippedSignature) return "f()" sig = S() fsig = sig.flip() self.assertEqual(fsig.x, 1) self.assertEqual(fsig.f(), "f()") fsig.y = 2 self.assertEqual(sig.y, 2) del fsig.y self.assertFalse(hasattr(sig, "y")) def test_getsetdelattr_property(self): class S(Signature): def __init__(self): super().__init__({}) self.x_get_type = None self.x_set_type = None self.x_set_val = None self.x_del_type = None @property def x(self): self.x_get_type = type(self) @x.setter def x(self, val): self.x_set_type = type(self) self.x_set_val = val @x.deleter def x(self): self.x_del_type = type(self) sig = S() fsig = sig.flip() fsig.x fsig.x = 1 del fsig.x # Tests both attribute access through the descriptor, and attribute setting without one! self.assertEqual(sig.x_get_type, type(fsig)) self.assertEqual(sig.x_set_type, type(fsig)) self.assertEqual(sig.x_set_val, 1) self.assertEqual(sig.x_del_type, type(fsig)) def test_classmethod(self): x_type = None class S(Signature): @classmethod def x(cls): nonlocal x_type x_type = cls sig = S({}) fsig = sig.flip() fsig.x() self.assertEqual(x_type, S) def test_members_equal_wrong(self): sig = Signature({}) with self.assertRaises(AttributeError): sig.flip().members = SignatureMembers({}) class PureInterfaceTestCase(unittest.TestCase): def test_construct(self): sig = Signature({ "a": In(4), "b": Out(signed(2)), }) intf = PureInterface(sig, path=("test",)) self.assertIs(intf.signature, sig) self.assertIsInstance(intf.a, Signal) self.assertIsInstance(intf.b, Signal) def test_repr(self): sig = Signature({ "a": In(4), "b": Out(signed(2)), }) intf = PureInterface(sig, path=("test",)) self.assertEqual(repr(intf), "") def test_repr_inherit(self): class CustomInterface(PureInterface): pass intf = CustomInterface(Signature({}), path=()) self.assertRegex(repr(intf), r"^$") class FlippedInterfaceTestCase(unittest.TestCase): def test_basic(self): sig = Signature({"a": In(1)}) intf = sig.create() self.assertTrue(sig.is_compliant(intf)) self.assertIs(intf.signature, sig) tintf = flipped(intf) self.assertEqual(tintf.signature, intf.signature.flip()) self.assertEqual(tintf, flipped(intf)) self.assertRegex(repr(tintf), r"^flipped\(\)$") self.assertIs(flipped(tintf), intf) def test_getsetdelattr(self): class I: signature = Signature({}) def __init__(self): self.x = 1 def f(self2): self.assertIsInstance(self2, FlippedInterface) return "f()" intf = I() fintf = flipped(intf) self.assertEqual(fintf.x, 1) self.assertEqual(fintf.f(), "f()") fintf.y = 2 self.assertEqual(intf.y, 2) del fintf.y self.assertFalse(hasattr(intf, "y")) def test_getsetdelattr_property(self): class I: signature = Signature({}) def __init__(self): self.x_get_type = None self.x_set_type = None self.x_set_val = None self.x_del_type = None @property def x(self): self.x_get_type = type(self) @x.setter def x(self, val): self.x_set_type = type(self) self.x_set_val = val @x.deleter def x(self): self.x_del_type = type(self) intf = I() fintf = flipped(intf) fintf.x fintf.x = 1 del fintf.x # Tests both attribute access through the descriptor, and attribute setting without one! self.assertEqual(intf.x_get_type, type(fintf)) self.assertEqual(intf.x_set_type, type(fintf)) self.assertEqual(intf.x_set_val, 1) self.assertEqual(intf.x_del_type, type(fintf)) def test_classmethod(self): x_type = None class I: signature = Signature({}) def __init__(self): pass @classmethod def x(cls): nonlocal x_type x_type = cls intf = I() fintf = flipped(intf) fintf.x() self.assertEqual(x_type, I) def test_flipped_wrong(self): with self.assertRaisesRegex(TypeError, r"^flipped\(\) can only flip an interface object, not Signature\({}\)$"): flipped(Signature({})) def test_create_subclass_flipped(self): class CustomInterface(PureInterface): def custom_method(self): return 69 class CustomSignature(Signature): def create(self, *, path=None, src_loc_at=0): return CustomInterface(self, path=path, src_loc_at=1 + src_loc_at) flipped_interface = CustomSignature({}).flip().create() self.assertTrue(hasattr(flipped_interface, "custom_method")) def test_propagate_flipped(self): class InterfaceWithFlippedSub(Component): a: In(Signature({ "b": Out(Signature({ "c": Out(1) })), "d": In(Signature({ "e": Out(1) })), "f": Out(1) })) def __init__(self): super().__init__() self.g = Signature({"h": In(1)}) ifsub = InterfaceWithFlippedSub() self.assertIsInstance(ifsub.a.b.signature, FlippedSignature) self.assertIsInstance(ifsub.a.d.signature, Signature) self.assertIsInstance(ifsub.signature.members["a"].signature. members["b"].signature, FlippedSignature) self.assertIsInstance(ifsub.signature.members["a"].signature. members["d"].signature, Signature) self.assertIsInstance(ifsub.a.f, Signal) self.assertEqual(ifsub.signature.members["a"].signature. members["f"].flow, In) self.assertIsInstance(flipped(ifsub).g, Signature) self.assertEqual(ifsub.g.members["h"].flow, In) self.assertEqual(flipped(ifsub).g.members["h"].flow, In) # This should be a no-op! That requires hooking ``__setattr__``. flipped(ifsub).a = flipped(ifsub).a self.assertEqual(ifsub.a.signature.members["f"].flow, In) class ConnectTestCase(unittest.TestCase): def test_arg_handles_and_signature_attr(self): m = Module() with self.assertRaisesRegex(AttributeError, r"^Argument 0 must have a 'signature' attribute$"): connect(m, object()) with self.assertRaisesRegex(AttributeError, r"^Argument 'x' must have a 'signature' attribute$"): connect(m, x=object()) def test_signature_type(self): m = Module() with self.assertRaisesRegex(TypeError, r"^Signature of argument 0 must be a signature, not 1$"): connect(m, NS(signature=1)) def test_signature_compliant(self): m = Module() with self.assertRaisesRegex(ConnectionError, r"^Argument 0 does not match its signature:\n" r"- 'arg0' does not have an attribute 'a'$"): connect(m, NS(signature=Signature({"a": In(1)}))) def test_member_missing(self): m = Module() with self.assertRaisesRegex(ConnectionError, r"^Member 'b' is present in 'q', but not in 'p'$"): connect(m, p=NS(signature=Signature({"a": In(1)}), a=Signal()), q=NS(signature=Signature({"a": In(1), "b": Out(1)}), a=Signal(), b=Signal())) with self.assertRaisesRegex(ConnectionError, r"^Member 'b' is present in 'p', but not in 'q'$"): connect(m, p=NS(signature=Signature({"a": In(1), "b": Out(1)}), a=Signal(), b=Signal()), q=NS(signature=Signature({"a": In(1)}), a=Signal())) def test_signature_to_port(self): m = Module() with self.assertRaisesRegex(ConnectionError, r"^Cannot connect signature member\(s\) 'p\.a' with port member\(s\) 'q\.a'$"): connect(m, p=NS(signature=Signature({"a": Out(Signature({}))}), a=NS(signature=Signature({}))), q=NS(signature=Signature({"a": In(1)}), a=Signal())) def test_shape_mismatch(self): m = Module() with self.assertRaisesRegex(ConnectionError, r"^Cannot connect the member 'q\.a' with shape unsigned\(2\) to the member 'p\.a' " r"with shape unsigned\(1\) because the shape widths \(2 and 1\) do not match$"): connect(m, p=NS(signature=Signature({"a": Out(1)}), a=Signal()), q=NS(signature=Signature({"a": In(2)}), a=Signal(2))) def test_shape_mismatch_enum(self): class Cycle(enum.Enum, shape=2): READ = 0 WRITE = 1 m = Module() with self.assertRaisesRegex(ConnectionError, r"^Cannot connect the member 'q\.a' with shape unsigned\(2\) \(\) " r"to the member 'p\.a' with shape unsigned\(1\) because the shape widths " r"\(2 and 1\) do not match$"): connect(m, p=NS(signature=Signature({"a": Out(1)}), a=Signal()), q=NS(signature=Signature({"a": In(Cycle)}), a=Signal(Cycle))) def test_init_mismatch(self): m = Module() with self.assertRaisesRegex(ConnectionError, r"^Cannot connect together the member 'q\.a' with initial value 1 and the member " r"'p\.a' with initial value 0 because the initial values do not match$"): connect(m, p=NS(signature=Signature({"a": Out(1, init=0)}), a=Signal()), q=NS(signature=Signature({"a": In(1, init=1)}), a=Signal(init=1))) def test_init_none_match(self): m = Module() connect(m, p=NS(signature=Signature({"a": Out(1, init=0)}), a=Signal()), q=NS(signature=Signature({"a": In(1)}), a=Signal())) def test_out_to_out(self): m = Module() with self.assertRaisesRegex(ConnectionError, r"^Cannot connect several output members 'p\.a', 'q\.a' together$"): connect(m, p=NS(signature=Signature({"a": Out(1)}), a=Signal()), q=NS(signature=Signature({"a": Out(1)}), a=Signal())) def test_out_to_const_in(self): m = Module() with self.assertRaisesRegex(ConnectionError, r"^Cannot connect input member 'q\.a' that has a constant value 0 to an output " r"member 'p\.a' that has a varying value$"): connect(m, p=NS(signature=Signature({"a": Out(1)}), a=Signal()), q=NS(signature=Signature({"a": In(1)}), a=Const(0))) def test_const_out_to_const_in_value_mismatch(self): m = Module() with self.assertRaisesRegex(ConnectionError, r"^Cannot connect input member 'q\.a' that has a constant value 0 to an output " r"member 'p\.a' that has a different constant value 1$"): connect(m, p=NS(signature=Signature({"a": Out(1)}), a=Const(1)), q=NS(signature=Signature({"a": In(1)}), a=Const(0))) def test_simple_bus(self): class Cycle(enum.Enum): IDLE = 0 READ = 1 WRITE = 2 sig = Signature({ "cycle": Out(Cycle), "addr": Out(16), "r_data": In(32), "w_data": Out(32), }) src = sig.create(path=('src',)) snk = sig.flip().create(path=('snk',)) m = Module() connect(m, src=src, snk=snk) self.assertEqual([repr(stmt) for stmt in m._statements["comb"]], [ '(eq (sig snk__addr) (sig src__addr))', '(eq (sig snk__cycle) (sig src__cycle))', '(eq (sig src__r_data) (sig snk__r_data))', '(eq (sig snk__w_data) (sig src__w_data))' ]) def test_const_in_out(self): m = Module() connect(m, p=NS(signature=Signature({"a": Out(1)}), a=Const(1)), q=NS(signature=Signature({"a": In(1)}), a=Const(1))) self.assertEqual(m._statements, {}) def test_nested(self): m = Module() connect(m, p=NS(signature=Signature({"a": Out(Signature({"f": Out(1)}))}), a=NS(signature=Signature({"f": Out(1)}), f=Signal(name='p__a'))), q=NS(signature=Signature({"a": In(Signature({"f": Out(1)}))}), a=NS(signature=Signature({"f": Out(1)}).flip(), f=Signal(name='q__a')))) self.assertEqual([repr(stmt) for stmt in m._statements["comb"]], [ '(eq (sig q__a) (sig p__a))' ]) def test_unordered(self): m = Module() connect(m, p=NS(signature=Signature({"a": Out(1), "b": Out(Signature({"f": Out(1), "g": Out(1)}))}), a=Signal(name="p__a"), b=NS(signature=Signature({"f": Out(1), "g": Out(1)}), f=Signal(name="p__b__f"), g=Signal(name="p__b__g"))), q=NS(signature=Signature({"b": In(Signature({"g": Out(1), "f": Out(1)})), "a": In(1)}), b=NS(signature=Signature({"g": Out(1), "f": Out(1)}).flip(), g=Signal(name="q__b__g"), f=Signal(name="q__b__f")), a=Signal(name="q__a"))) self.assertEqual([repr(stmt) for stmt in m._statements["comb"]], [ '(eq (sig q__a) (sig p__a))', '(eq (sig q__b__f) (sig p__b__f))', '(eq (sig q__b__g) (sig p__b__g))', ]) def test_dimension(self): sig = Signature({"a": Out(1).array(2)}) m = Module() connect(m, p=sig.create(path=('p',)), q=sig.flip().create(path=('q',))) self.assertEqual([repr(stmt) for stmt in m._statements["comb"]], [ '(eq (sig q__a__0) (sig p__a__0))', '(eq (sig q__a__1) (sig p__a__1))' ]) def test_dimension_multi(self): sig = Signature({"a": Out(1).array(1).array(1)}) m = Module() connect(m, p=sig.create(path=('p',)), q=sig.flip().create(path=('q',))) self.assertEqual([repr(stmt) for stmt in m._statements["comb"]], [ '(eq (sig q__a__0__0) (sig p__a__0__0))', ]) def test_connect_none(self): # Connecting zero or more empty signatures is permitted as (a) it's not # something you can write mistakenly out by hand, and (b) it permits # generic code to expand to nothing without errors around edges. m = Module() connect(m) def test_connect_empty(self): m = Module() connect(m, p=NS(signature=Signature({}))) def test_connect_empty_multi(self): m = Module() connect(m, p=NS(signature=Signature({})), q=NS(signature=Signature({}))) def test_connect_one(self): # Connecting just one signature should be allowed for the same reasons # as above. (It's possible to forget an argument, but that stands out.) m = Module() connect(m, p=NS(signature=Signature({"a": Out(1), "b": In(1)}), a=Signal(), b=Signal())) def test_connect_one_in_only(self): # As above, even if there's only inputs. m = Module() connect(m, p=NS(signature=Signature({"a": In(1)}), a=Signal())) def test_connect_multi_in_only_fails(self): # If we're only attempting to connect multiple inputs, we're not # actually doing anything and it's most likely a mistake. m = Module() with self.assertRaisesRegex(ConnectionError, r"^Only input to input connections have been made between several interfaces; " r"should one of them have been flipped\?$"): connect(m, p=NS(signature=Signature({"a": In(1), "b": In(8)}), a=Signal(), b=Signal(8)), q=NS(signature=Signature({"a": In(1), "b": In(8)}), a=Signal(), b=Signal(8))) def test_connect_multi_some_in_pairs(self): # Connecting matching inputs is an allowed no-op if there are also # actual input-output connections to be made. See # https://github.com/amaranth-lang/amaranth/pull/1153#issuecomment-1962810678 # for more discussion. m = Module() connect(m, p=NS(signature=Signature({"a": In(1), "b": In(1)}), a=Signal(), b=Signal()), q=NS(signature=Signature({"a": Out(1), "b": In(1)}), a=Signal(), b=Signal())) class ComponentTestCase(unittest.TestCase): def test_basic(self): class C(Component): sig : Out(2) c = C() self.assertEqual(c.signature, Signature({"sig": Out(2)})) self.assertIsInstance(c.sig, Signal) self.assertEqual(c.sig.shape(), unsigned(2)) def test_non_member_annotations(self): class C(Component): sig : Out(2) foo : int c = C() self.assertEqual(c.signature, Signature({"sig": Out(2)})) def test_private_member_annotations(self): class C(Component): sig_pub : Out(2) _sig_priv : Out(2) c = C() self.assertEqual(c.signature, Signature({"sig_pub": Out(2)})) def test_no_annotations(self): class C(Component): pass with self.assertRaisesRegex(TypeError, r"^Component '.+?\.C' does not have signature member annotations$"): C() def test_would_overwrite_field(self): class C(Component): sig : Out(2) def __init__(self): self.sig = 1 super().__init__() with self.assertRaisesRegex(NameError, r"^Cannot initialize attribute for signature member 'sig' because an attribute " r"with the same name already exists$"): C() def test_inherit(self): class A(Component): clk: In(1) class B(A): rst: In(1) class C(B): pass c = C() self.assertEqual(c.signature, Signature({"clk": In(1), "rst": In(1)})) def test_inherit_wrong(self): class A(Component): a: In(1) class B(A): a: Out(1) with self.assertRaisesRegex(NameError, r"^Member 'a' is redefined in .*.B$"): B() def test_create(self): class C(Component): def __init__(self, width): super().__init__(Signature({ "a": In(width) })) c = C(2) self.assertEqual(c.signature, Signature({"a": In(2)})) self.assertIsInstance(c.a, Signal) self.assertEqual(c.a.shape(), unsigned(2)) def test_create_dict(self): class C(Component): def __init__(self, width): super().__init__({ "a": In(width) }) c = C(2) self.assertEqual(c.signature, Signature({"a": In(2)})) self.assertIsInstance(c.a, Signal) self.assertEqual(c.a.shape(), unsigned(2)) def test_create_wrong(self): class C(Component): a: In(2) def __init__(self, width): super().__init__(Signature({ "a": In(width) })) with self.assertRaisesRegex(TypeError, r"^Signature was passed as an argument, but component '.*.C' already has signature member annotations$"): C(2) def test_create_wrong_type(self): class C(Component): def __init__(self, width): super().__init__(4) with self.assertRaisesRegex(TypeError, r"^Object 4 is not a signature nor a dict$"): C(2) def test_metadata_origin(self): class A(Component): clk: In(1) a = A() self.assertIsInstance(a.metadata, ComponentMetadata) self.assertIs(a.metadata.origin, a) class ComponentMetadataTestCase(unittest.TestCase): def test_as_json(self): class Annotation1(Annotation): schema = { "$schema": "https://json-schema.org/draft/2020-12/schema", "$id": "https://example.com/schema/foo/0.1/bar.json", "type": "object", "properties": { "hello": { "type": "boolean" }, }, } def origin(self): return object() def as_json(self): instance = { "hello": True } self.validate(instance) return instance class Signature1(Signature): def __init__(self): super().__init__({ "i": In(unsigned(8), init=42).array(2).array(3), "o": Out(signed(4)) }) def annotations(self, obj): return (*Signature.annotations(self, obj), Annotation1()) class Signature2(Signature): def __init__(self): super().__init__({ "clk": In(1), "foo": Out(Signature1()).array(4), "oof": In(Signature1()) }) def annotations(self, obj): return (*Signature.annotations(self, obj), Annotation1()) class A(Component): def __init__(self): super().__init__(Signature2()) metadata = ComponentMetadata(A()) self.assertEqual(metadata.as_json(), { "interface": { "members": { "clk": { "type": "port", "name": "clk", "dir": "in", "width": 1, "signed": False, "init": "0", }, "foo": [{ "type": "interface", "members": { "i": [[{ "type": "port", "name": f"foo__{x}__i__{y}__{z}", "dir": "in", "width": 8, "signed": False, "init": "42", } for z in range(2)] for y in range(3)], "o": { "type": "port", "name": f"foo__{x}__o", "dir": "out", "width": 4, "signed": True, "init": "0", }, }, "annotations": { "https://example.com/schema/foo/0.1/bar.json": { "hello": True, }, }, } for x in range(4)], "oof": { "type": "interface", "members": { "i": [[{ "type": "port", "name": f"oof__i__{y}__{z}", "dir": "out", "width": 8, "signed": False, "init": "42", } for z in range(2)] for y in range(3)], "o": { "type": "port", "name": "oof__o", "dir": "in", "width": 4, "signed": True, "init": "0", }, }, "annotations": { "https://example.com/schema/foo/0.1/bar.json": { "hello": True, }, }, }, }, "annotations": { "https://example.com/schema/foo/0.1/bar.json": { "hello": True, }, }, }, }) def test_validate(self): ComponentMetadata.validate({ "interface": { "members": { "i": { "type": "port", "name": "i", "dir": "in", "width": 1, "signed": False, "init": "0", }, "o": { "type": "port", "name": "o", "dir": "out", "width": 1, "signed": False, "init": "0", }, }, "annotations": { "https://example.com/schema/foo/0/foo.json": { "foo": True, }, }, }, }) def test_validate_error(self): with self.assertRaises(InvalidMetadata): ComponentMetadata.validate({ "interface": { "members": { "foo": True, }, "annotations": {}, }, }) def test_wrong_origin(self): with self.assertRaisesRegex(TypeError, r"Origin must be a component, not 'foo'"): ComponentMetadata("foo")