parent
							
								
									7f1397b281
								
							
						
					
					
						commit
						5a17f94fdc
					
				|  | @ -1,8 +1,12 @@ | ||||||
| from collections import OrderedDict | from collections import OrderedDict | ||||||
|  | import warnings | ||||||
| 
 | 
 | ||||||
| from ..hdl.ast import * | from ..hdl.ast import * | ||||||
|  | with warnings.catch_warnings(): | ||||||
|  |     warnings.filterwarnings(action="ignore", category=DeprecationWarning) | ||||||
|     from ..hdl.rec import * |     from ..hdl.rec import * | ||||||
| from ..lib.io import * | from ..lib.io import * | ||||||
|  | from ..lib import wiring | ||||||
| 
 | 
 | ||||||
| from .dsl import * | from .dsl import * | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,9 +1,13 @@ | ||||||
|  | import warnings | ||||||
|  | 
 | ||||||
| from .ast import Shape, unsigned, signed | from .ast import Shape, unsigned, signed | ||||||
| from .ast import Value, Const, C, Mux, Cat, Repl, Array, Signal, ClockSignal, ResetSignal | from .ast import Value, Const, C, Mux, Cat, Repl, Array, Signal, ClockSignal, ResetSignal | ||||||
| from .dsl import Module | from .dsl import Module | ||||||
| from .cd import ClockDomain | from .cd import ClockDomain | ||||||
| from .ir import Elaboratable, Fragment, Instance | from .ir import Elaboratable, Fragment, Instance | ||||||
| from .mem import Memory | from .mem import Memory | ||||||
|  | with warnings.catch_warnings(): | ||||||
|  |     warnings.filterwarnings(action="ignore", category=DeprecationWarning) | ||||||
|     from .rec import Record |     from .rec import Record | ||||||
| from .xfrm import DomainRenamer, ResetInserter, EnableInserter | from .xfrm import DomainRenamer, ResetInserter, EnableInserter | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										280
									
								
								amaranth/hdl/_rec.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										280
									
								
								amaranth/hdl/_rec.py
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,280 @@ | ||||||
|  | # TODO(amaranth-0.6): remove module | ||||||
|  | 
 | ||||||
|  | from enum import Enum | ||||||
|  | from collections import OrderedDict | ||||||
|  | from functools import reduce, wraps | ||||||
|  | 
 | ||||||
|  | from .. import tracer | ||||||
|  | from .._utils import union | ||||||
|  | from .ast import * | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | __all__ = ["Direction", "DIR_NONE", "DIR_FANOUT", "DIR_FANIN", "Layout", "Record"] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | Direction = Enum('Direction', ('NONE', 'FANOUT', 'FANIN')) | ||||||
|  | 
 | ||||||
|  | DIR_NONE   = Direction.NONE | ||||||
|  | DIR_FANOUT = Direction.FANOUT | ||||||
|  | DIR_FANIN  = Direction.FANIN | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class Layout: | ||||||
|  |     @staticmethod | ||||||
|  |     def cast(obj, *, src_loc_at=0): | ||||||
|  |         if isinstance(obj, Layout): | ||||||
|  |             return obj | ||||||
|  |         return Layout(obj, src_loc_at=1 + src_loc_at) | ||||||
|  | 
 | ||||||
|  |     def __init__(self, fields, *, src_loc_at=0): | ||||||
|  |         self.fields = OrderedDict() | ||||||
|  |         for field in fields: | ||||||
|  |             if not isinstance(field, tuple) or len(field) not in (2, 3): | ||||||
|  |                 raise TypeError("Field {!r} has invalid layout: should be either " | ||||||
|  |                                 "(name, shape) or (name, shape, direction)" | ||||||
|  |                                 .format(field)) | ||||||
|  |             if len(field) == 2: | ||||||
|  |                 name, shape = field | ||||||
|  |                 direction = DIR_NONE | ||||||
|  |                 if isinstance(shape, list): | ||||||
|  |                     shape = Layout.cast(shape) | ||||||
|  |             else: | ||||||
|  |                 name, shape, direction = field | ||||||
|  |                 if not isinstance(direction, Direction): | ||||||
|  |                     raise TypeError("Field {!r} has invalid direction: should be a Direction " | ||||||
|  |                                     "instance like DIR_FANIN" | ||||||
|  |                                     .format(field)) | ||||||
|  |             if not isinstance(name, str): | ||||||
|  |                 raise TypeError("Field {!r} has invalid name: should be a string" | ||||||
|  |                                 .format(field)) | ||||||
|  |             if not isinstance(shape, Layout): | ||||||
|  |                 try: | ||||||
|  |                     # Check provided shape by calling Shape.cast and checking for exception | ||||||
|  |                     Shape.cast(shape, src_loc_at=1 + src_loc_at) | ||||||
|  |                 except Exception: | ||||||
|  |                     raise TypeError("Field {!r} has invalid shape: should be castable to Shape " | ||||||
|  |                                     "or a list of fields of a nested record" | ||||||
|  |                                     .format(field)) | ||||||
|  |             if name in self.fields: | ||||||
|  |                 raise NameError("Field {!r} has a name that is already present in the layout" | ||||||
|  |                                 .format(field)) | ||||||
|  |             self.fields[name] = (shape, direction) | ||||||
|  | 
 | ||||||
|  |     def __getitem__(self, item): | ||||||
|  |         if isinstance(item, tuple): | ||||||
|  |             return Layout([ | ||||||
|  |                 (name, shape, dir) | ||||||
|  |                 for (name, (shape, dir)) in self.fields.items() | ||||||
|  |                 if name in item | ||||||
|  |             ]) | ||||||
|  | 
 | ||||||
|  |         return self.fields[item] | ||||||
|  | 
 | ||||||
|  |     def __iter__(self): | ||||||
|  |         for name, (shape, dir) in self.fields.items(): | ||||||
|  |             yield (name, shape, dir) | ||||||
|  | 
 | ||||||
|  |     def __eq__(self, other): | ||||||
|  |         return self.fields == other.fields | ||||||
|  | 
 | ||||||
|  |     def __repr__(self): | ||||||
|  |         field_reprs = [] | ||||||
|  |         for name, shape, dir in self: | ||||||
|  |             if dir == DIR_NONE: | ||||||
|  |                 field_reprs.append("({!r}, {!r})".format(name, shape)) | ||||||
|  |             else: | ||||||
|  |                 field_reprs.append("({!r}, {!r}, Direction.{})".format(name, shape, dir.name)) | ||||||
|  |         return "Layout([{}])".format(", ".join(field_reprs)) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class Record(ValueCastable): | ||||||
|  |     @staticmethod | ||||||
|  |     def like(other, *, name=None, name_suffix=None, src_loc_at=0): | ||||||
|  |         if name is not None: | ||||||
|  |             new_name = str(name) | ||||||
|  |         elif name_suffix is not None: | ||||||
|  |             new_name = other.name + str(name_suffix) | ||||||
|  |         else: | ||||||
|  |             new_name = tracer.get_var_name(depth=2 + src_loc_at, default=None) | ||||||
|  | 
 | ||||||
|  |         def concat(a, b): | ||||||
|  |             if a is None: | ||||||
|  |                 return b | ||||||
|  |             return "{}__{}".format(a, b) | ||||||
|  | 
 | ||||||
|  |         fields = {} | ||||||
|  |         for field_name in other.fields: | ||||||
|  |             field = other[field_name] | ||||||
|  |             if isinstance(field, Record): | ||||||
|  |                 fields[field_name] = Record.like(field, name=concat(new_name, field_name), | ||||||
|  |                                                  src_loc_at=1 + src_loc_at) | ||||||
|  |             else: | ||||||
|  |                 fields[field_name] = Signal.like(field, name=concat(new_name, field_name), | ||||||
|  |                                                  src_loc_at=1 + src_loc_at) | ||||||
|  | 
 | ||||||
|  |         return Record(other.layout, name=new_name, fields=fields, src_loc_at=1) | ||||||
|  | 
 | ||||||
|  |     def __init__(self, layout, *, name=None, fields=None, src_loc_at=0): | ||||||
|  |         if name is None: | ||||||
|  |             name = tracer.get_var_name(depth=2 + src_loc_at, default=None) | ||||||
|  | 
 | ||||||
|  |         self.name    = name | ||||||
|  |         self.src_loc = tracer.get_src_loc(src_loc_at) | ||||||
|  | 
 | ||||||
|  |         def concat(a, b): | ||||||
|  |             if a is None: | ||||||
|  |                 return b | ||||||
|  |             return "{}__{}".format(a, b) | ||||||
|  | 
 | ||||||
|  |         self.layout = Layout.cast(layout, src_loc_at=1 + src_loc_at) | ||||||
|  |         self.fields = OrderedDict() | ||||||
|  |         for field_name, field_shape, field_dir in self.layout: | ||||||
|  |             if fields is not None and field_name in fields: | ||||||
|  |                 field = fields[field_name] | ||||||
|  |                 if isinstance(field_shape, Layout): | ||||||
|  |                     assert isinstance(field, Record) and field_shape == field.layout | ||||||
|  |                 else: | ||||||
|  |                     assert isinstance(field, Signal) and Shape.cast(field_shape) == field.shape() | ||||||
|  |                 self.fields[field_name] = field | ||||||
|  |             else: | ||||||
|  |                 if isinstance(field_shape, Layout): | ||||||
|  |                     self.fields[field_name] = Record(field_shape, name=concat(name, field_name), | ||||||
|  |                                                      src_loc_at=1 + src_loc_at) | ||||||
|  |                 else: | ||||||
|  |                     self.fields[field_name] = Signal(field_shape, name=concat(name, field_name), | ||||||
|  |                                                      src_loc_at=1 + src_loc_at) | ||||||
|  | 
 | ||||||
|  |     def __getattr__(self, name): | ||||||
|  |         return self[name] | ||||||
|  | 
 | ||||||
|  |     def __getitem__(self, item): | ||||||
|  |         if isinstance(item, str): | ||||||
|  |             try: | ||||||
|  |                 return self.fields[item] | ||||||
|  |             except KeyError: | ||||||
|  |                 if self.name is None: | ||||||
|  |                     reference = "Unnamed record" | ||||||
|  |                 else: | ||||||
|  |                     reference = "Record '{}'".format(self.name) | ||||||
|  |                 raise AttributeError("{} does not have a field '{}'. Did you mean one of: {}?" | ||||||
|  |                                      .format(reference, item, ", ".join(self.fields))) from None | ||||||
|  |         elif isinstance(item, tuple): | ||||||
|  |             return Record(self.layout[item], fields={ | ||||||
|  |                 field_name: field_value | ||||||
|  |                 for field_name, field_value in self.fields.items() | ||||||
|  |                 if field_name in item | ||||||
|  |             }) | ||||||
|  |         else: | ||||||
|  |             try: | ||||||
|  |                 return Value.__getitem__(self, item) | ||||||
|  |             except KeyError: | ||||||
|  |                 if self.name is None: | ||||||
|  |                     reference = "Unnamed record" | ||||||
|  |                 else: | ||||||
|  |                     reference = "Record '{}'".format(self.name) | ||||||
|  |                 raise AttributeError("{} does not have a field '{}'. Did you mean one of: {}?" | ||||||
|  |                                      .format(reference, item, ", ".join(self.fields))) from None | ||||||
|  | 
 | ||||||
|  |     @ValueCastable.lowermethod | ||||||
|  |     def as_value(self): | ||||||
|  |         return Cat(self.fields.values()) | ||||||
|  | 
 | ||||||
|  |     def __len__(self): | ||||||
|  |         return len(self.as_value()) | ||||||
|  | 
 | ||||||
|  |     def _lhs_signals(self): | ||||||
|  |         return union((f._lhs_signals() for f in self.fields.values()), start=SignalSet()) | ||||||
|  | 
 | ||||||
|  |     def _rhs_signals(self): | ||||||
|  |         return union((f._rhs_signals() for f in self.fields.values()), start=SignalSet()) | ||||||
|  | 
 | ||||||
|  |     def __repr__(self): | ||||||
|  |         fields = [] | ||||||
|  |         for field_name, field in self.fields.items(): | ||||||
|  |             if isinstance(field, Signal): | ||||||
|  |                 fields.append(field_name) | ||||||
|  |             else: | ||||||
|  |                 fields.append(repr(field)) | ||||||
|  |         name = self.name | ||||||
|  |         if name is None: | ||||||
|  |             name = "<unnamed>" | ||||||
|  |         return "(rec {} {})".format(name, " ".join(fields)) | ||||||
|  | 
 | ||||||
|  |     def shape(self): | ||||||
|  |         return self.as_value().shape() | ||||||
|  | 
 | ||||||
|  |     def connect(self, *subordinates, include=None, exclude=None): | ||||||
|  |         def rec_name(record): | ||||||
|  |             if record.name is None: | ||||||
|  |                 return "unnamed record" | ||||||
|  |             else: | ||||||
|  |                 return "record '{}'".format(record.name) | ||||||
|  | 
 | ||||||
|  |         for field in include or {}: | ||||||
|  |             if field not in self.fields: | ||||||
|  |                 raise AttributeError("Cannot include field '{}' because it is not present in {}" | ||||||
|  |                                      .format(field, rec_name(self))) | ||||||
|  |         for field in exclude or {}: | ||||||
|  |             if field not in self.fields: | ||||||
|  |                 raise AttributeError("Cannot exclude field '{}' because it is not present in {}" | ||||||
|  |                                      .format(field, rec_name(self))) | ||||||
|  | 
 | ||||||
|  |         stmts = [] | ||||||
|  |         for field in self.fields: | ||||||
|  |             if include is not None and field not in include: | ||||||
|  |                 continue | ||||||
|  |             if exclude is not None and field in exclude: | ||||||
|  |                 continue | ||||||
|  | 
 | ||||||
|  |             shape, direction = self.layout[field] | ||||||
|  |             if not isinstance(shape, Layout) and direction == DIR_NONE: | ||||||
|  |                 raise TypeError("Cannot connect field '{}' of {} because it does not have " | ||||||
|  |                                 "a direction" | ||||||
|  |                                 .format(field, rec_name(self))) | ||||||
|  | 
 | ||||||
|  |             item = self.fields[field] | ||||||
|  |             subord_items = [] | ||||||
|  |             for subord in subordinates: | ||||||
|  |                 if field not in subord.fields: | ||||||
|  |                     raise AttributeError("Cannot connect field '{}' of {} to subordinate {} " | ||||||
|  |                                          "because the subordinate record does not have this field" | ||||||
|  |                                          .format(field, rec_name(self), rec_name(subord))) | ||||||
|  |                 subord_items.append(subord.fields[field]) | ||||||
|  | 
 | ||||||
|  |             if isinstance(shape, Layout): | ||||||
|  |                 sub_include = include[field] if include and field in include else None | ||||||
|  |                 sub_exclude = exclude[field] if exclude and field in exclude else None | ||||||
|  |                 stmts += item.connect(*subord_items, include=sub_include, exclude=sub_exclude) | ||||||
|  |             else: | ||||||
|  |                 if direction == DIR_FANOUT: | ||||||
|  |                     stmts += [sub_item.eq(item) for sub_item in subord_items] | ||||||
|  |                 if direction == DIR_FANIN: | ||||||
|  |                     stmts += [item.eq(reduce(lambda a, b: a | b, subord_items))] | ||||||
|  | 
 | ||||||
|  |         return stmts | ||||||
|  | 
 | ||||||
|  | def _valueproxy(name): | ||||||
|  |     value_func = getattr(Value, name) | ||||||
|  |     @wraps(value_func) | ||||||
|  |     def _wrapper(self, *args, **kwargs): | ||||||
|  |         return value_func(Value.cast(self), *args, **kwargs) | ||||||
|  |     return _wrapper | ||||||
|  | 
 | ||||||
|  | for name in [ | ||||||
|  |         "__bool__", | ||||||
|  |         "__invert__", "__neg__", | ||||||
|  |         "__add__", "__radd__", "__sub__", "__rsub__", | ||||||
|  |         "__mul__", "__rmul__", | ||||||
|  |         "__mod__", "__rmod__", "__floordiv__", "__rfloordiv__", | ||||||
|  |         "__lshift__", "__rlshift__", "__rshift__", "__rrshift__", | ||||||
|  |         "__and__", "__rand__", "__xor__", "__rxor__", "__or__", "__ror__", | ||||||
|  |         "__eq__", "__ne__", "__lt__", "__le__", "__gt__", "__ge__", | ||||||
|  |         "__abs__", "__len__", | ||||||
|  |         "as_unsigned", "as_signed", "bool", "any", "all", "xor", "implies", | ||||||
|  |         "bit_select", "word_select", "matches", | ||||||
|  |         "shift_left", "shift_right", "rotate_left", "rotate_right", "eq" | ||||||
|  |         ]: | ||||||
|  |     setattr(Record, name, _valueproxy(name)) | ||||||
|  | 
 | ||||||
|  | del _valueproxy | ||||||
|  | del name | ||||||
|  | @ -1,278 +1,22 @@ | ||||||
| from enum import Enum | # TODO(amaranth-0.6): remove module | ||||||
| from collections import OrderedDict |  | ||||||
| from functools import reduce, wraps |  | ||||||
| 
 | 
 | ||||||
| from .. import tracer | import warnings | ||||||
| from .._utils import union | import importlib | ||||||
| from .ast import * | from . import _rec | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| __all__ = ["Direction", "DIR_NONE", "DIR_FANOUT", "DIR_FANIN", "Layout", "Record"] | __all__ = ["Direction", "DIR_NONE", "DIR_FANOUT", "DIR_FANIN", "Layout", "Record"] | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| Direction = Enum('Direction', ('NONE', 'FANOUT', 'FANIN')) | def __dir__(): | ||||||
| 
 |     return list({*globals(), *__all__}) | ||||||
| DIR_NONE   = Direction.NONE |  | ||||||
| DIR_FANOUT = Direction.FANOUT |  | ||||||
| DIR_FANIN  = Direction.FANIN |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class Layout: | def __getattr__(name): | ||||||
|     @staticmethod |     if name in __all__: | ||||||
|     def cast(obj, *, src_loc_at=0): |         warnings.warn(f"instead of `{__name__}.{name}`, use the `amaranth.lib.data` and " | ||||||
|         if isinstance(obj, Layout): |                       f"`amaranth.lib.wiring` libraries as appropriate for the application; " | ||||||
|             return obj |                       f"`{__name__}` will be removed in Amaranth 0.6", | ||||||
|         return Layout(obj, src_loc_at=1 + src_loc_at) |                       DeprecationWarning, stacklevel=2) | ||||||
| 
 |         return getattr(_rec, name) | ||||||
|     def __init__(self, fields, *, src_loc_at=0): |     raise AttributeError(f"module {__name__!r} has no attribute {name!r}") | ||||||
|         self.fields = OrderedDict() |  | ||||||
|         for field in fields: |  | ||||||
|             if not isinstance(field, tuple) or len(field) not in (2, 3): |  | ||||||
|                 raise TypeError("Field {!r} has invalid layout: should be either " |  | ||||||
|                                 "(name, shape) or (name, shape, direction)" |  | ||||||
|                                 .format(field)) |  | ||||||
|             if len(field) == 2: |  | ||||||
|                 name, shape = field |  | ||||||
|                 direction = DIR_NONE |  | ||||||
|                 if isinstance(shape, list): |  | ||||||
|                     shape = Layout.cast(shape) |  | ||||||
|             else: |  | ||||||
|                 name, shape, direction = field |  | ||||||
|                 if not isinstance(direction, Direction): |  | ||||||
|                     raise TypeError("Field {!r} has invalid direction: should be a Direction " |  | ||||||
|                                     "instance like DIR_FANIN" |  | ||||||
|                                     .format(field)) |  | ||||||
|             if not isinstance(name, str): |  | ||||||
|                 raise TypeError("Field {!r} has invalid name: should be a string" |  | ||||||
|                                 .format(field)) |  | ||||||
|             if not isinstance(shape, Layout): |  | ||||||
|                 try: |  | ||||||
|                     # Check provided shape by calling Shape.cast and checking for exception |  | ||||||
|                     Shape.cast(shape, src_loc_at=1 + src_loc_at) |  | ||||||
|                 except Exception: |  | ||||||
|                     raise TypeError("Field {!r} has invalid shape: should be castable to Shape " |  | ||||||
|                                     "or a list of fields of a nested record" |  | ||||||
|                                     .format(field)) |  | ||||||
|             if name in self.fields: |  | ||||||
|                 raise NameError("Field {!r} has a name that is already present in the layout" |  | ||||||
|                                 .format(field)) |  | ||||||
|             self.fields[name] = (shape, direction) |  | ||||||
| 
 |  | ||||||
|     def __getitem__(self, item): |  | ||||||
|         if isinstance(item, tuple): |  | ||||||
|             return Layout([ |  | ||||||
|                 (name, shape, dir) |  | ||||||
|                 for (name, (shape, dir)) in self.fields.items() |  | ||||||
|                 if name in item |  | ||||||
|             ]) |  | ||||||
| 
 |  | ||||||
|         return self.fields[item] |  | ||||||
| 
 |  | ||||||
|     def __iter__(self): |  | ||||||
|         for name, (shape, dir) in self.fields.items(): |  | ||||||
|             yield (name, shape, dir) |  | ||||||
| 
 |  | ||||||
|     def __eq__(self, other): |  | ||||||
|         return self.fields == other.fields |  | ||||||
| 
 |  | ||||||
|     def __repr__(self): |  | ||||||
|         field_reprs = [] |  | ||||||
|         for name, shape, dir in self: |  | ||||||
|             if dir == DIR_NONE: |  | ||||||
|                 field_reprs.append("({!r}, {!r})".format(name, shape)) |  | ||||||
|             else: |  | ||||||
|                 field_reprs.append("({!r}, {!r}, Direction.{})".format(name, shape, dir.name)) |  | ||||||
|         return "Layout([{}])".format(", ".join(field_reprs)) |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| class Record(ValueCastable): |  | ||||||
|     @staticmethod |  | ||||||
|     def like(other, *, name=None, name_suffix=None, src_loc_at=0): |  | ||||||
|         if name is not None: |  | ||||||
|             new_name = str(name) |  | ||||||
|         elif name_suffix is not None: |  | ||||||
|             new_name = other.name + str(name_suffix) |  | ||||||
|         else: |  | ||||||
|             new_name = tracer.get_var_name(depth=2 + src_loc_at, default=None) |  | ||||||
| 
 |  | ||||||
|         def concat(a, b): |  | ||||||
|             if a is None: |  | ||||||
|                 return b |  | ||||||
|             return "{}__{}".format(a, b) |  | ||||||
| 
 |  | ||||||
|         fields = {} |  | ||||||
|         for field_name in other.fields: |  | ||||||
|             field = other[field_name] |  | ||||||
|             if isinstance(field, Record): |  | ||||||
|                 fields[field_name] = Record.like(field, name=concat(new_name, field_name), |  | ||||||
|                                                  src_loc_at=1 + src_loc_at) |  | ||||||
|             else: |  | ||||||
|                 fields[field_name] = Signal.like(field, name=concat(new_name, field_name), |  | ||||||
|                                                  src_loc_at=1 + src_loc_at) |  | ||||||
| 
 |  | ||||||
|         return Record(other.layout, name=new_name, fields=fields, src_loc_at=1) |  | ||||||
| 
 |  | ||||||
|     def __init__(self, layout, *, name=None, fields=None, src_loc_at=0): |  | ||||||
|         if name is None: |  | ||||||
|             name = tracer.get_var_name(depth=2 + src_loc_at, default=None) |  | ||||||
| 
 |  | ||||||
|         self.name    = name |  | ||||||
|         self.src_loc = tracer.get_src_loc(src_loc_at) |  | ||||||
| 
 |  | ||||||
|         def concat(a, b): |  | ||||||
|             if a is None: |  | ||||||
|                 return b |  | ||||||
|             return "{}__{}".format(a, b) |  | ||||||
| 
 |  | ||||||
|         self.layout = Layout.cast(layout, src_loc_at=1 + src_loc_at) |  | ||||||
|         self.fields = OrderedDict() |  | ||||||
|         for field_name, field_shape, field_dir in self.layout: |  | ||||||
|             if fields is not None and field_name in fields: |  | ||||||
|                 field = fields[field_name] |  | ||||||
|                 if isinstance(field_shape, Layout): |  | ||||||
|                     assert isinstance(field, Record) and field_shape == field.layout |  | ||||||
|                 else: |  | ||||||
|                     assert isinstance(field, Signal) and Shape.cast(field_shape) == field.shape() |  | ||||||
|                 self.fields[field_name] = field |  | ||||||
|             else: |  | ||||||
|                 if isinstance(field_shape, Layout): |  | ||||||
|                     self.fields[field_name] = Record(field_shape, name=concat(name, field_name), |  | ||||||
|                                                      src_loc_at=1 + src_loc_at) |  | ||||||
|                 else: |  | ||||||
|                     self.fields[field_name] = Signal(field_shape, name=concat(name, field_name), |  | ||||||
|                                                      src_loc_at=1 + src_loc_at) |  | ||||||
| 
 |  | ||||||
|     def __getattr__(self, name): |  | ||||||
|         return self[name] |  | ||||||
| 
 |  | ||||||
|     def __getitem__(self, item): |  | ||||||
|         if isinstance(item, str): |  | ||||||
|             try: |  | ||||||
|                 return self.fields[item] |  | ||||||
|             except KeyError: |  | ||||||
|                 if self.name is None: |  | ||||||
|                     reference = "Unnamed record" |  | ||||||
|                 else: |  | ||||||
|                     reference = "Record '{}'".format(self.name) |  | ||||||
|                 raise AttributeError("{} does not have a field '{}'. Did you mean one of: {}?" |  | ||||||
|                                      .format(reference, item, ", ".join(self.fields))) from None |  | ||||||
|         elif isinstance(item, tuple): |  | ||||||
|             return Record(self.layout[item], fields={ |  | ||||||
|                 field_name: field_value |  | ||||||
|                 for field_name, field_value in self.fields.items() |  | ||||||
|                 if field_name in item |  | ||||||
|             }) |  | ||||||
|         else: |  | ||||||
|             try: |  | ||||||
|                 return Value.__getitem__(self, item) |  | ||||||
|             except KeyError: |  | ||||||
|                 if self.name is None: |  | ||||||
|                     reference = "Unnamed record" |  | ||||||
|                 else: |  | ||||||
|                     reference = "Record '{}'".format(self.name) |  | ||||||
|                 raise AttributeError("{} does not have a field '{}'. Did you mean one of: {}?" |  | ||||||
|                                      .format(reference, item, ", ".join(self.fields))) from None |  | ||||||
| 
 |  | ||||||
|     @ValueCastable.lowermethod |  | ||||||
|     def as_value(self): |  | ||||||
|         return Cat(self.fields.values()) |  | ||||||
| 
 |  | ||||||
|     def __len__(self): |  | ||||||
|         return len(self.as_value()) |  | ||||||
| 
 |  | ||||||
|     def _lhs_signals(self): |  | ||||||
|         return union((f._lhs_signals() for f in self.fields.values()), start=SignalSet()) |  | ||||||
| 
 |  | ||||||
|     def _rhs_signals(self): |  | ||||||
|         return union((f._rhs_signals() for f in self.fields.values()), start=SignalSet()) |  | ||||||
| 
 |  | ||||||
|     def __repr__(self): |  | ||||||
|         fields = [] |  | ||||||
|         for field_name, field in self.fields.items(): |  | ||||||
|             if isinstance(field, Signal): |  | ||||||
|                 fields.append(field_name) |  | ||||||
|             else: |  | ||||||
|                 fields.append(repr(field)) |  | ||||||
|         name = self.name |  | ||||||
|         if name is None: |  | ||||||
|             name = "<unnamed>" |  | ||||||
|         return "(rec {} {})".format(name, " ".join(fields)) |  | ||||||
| 
 |  | ||||||
|     def shape(self): |  | ||||||
|         return self.as_value().shape() |  | ||||||
| 
 |  | ||||||
|     def connect(self, *subordinates, include=None, exclude=None): |  | ||||||
|         def rec_name(record): |  | ||||||
|             if record.name is None: |  | ||||||
|                 return "unnamed record" |  | ||||||
|             else: |  | ||||||
|                 return "record '{}'".format(record.name) |  | ||||||
| 
 |  | ||||||
|         for field in include or {}: |  | ||||||
|             if field not in self.fields: |  | ||||||
|                 raise AttributeError("Cannot include field '{}' because it is not present in {}" |  | ||||||
|                                      .format(field, rec_name(self))) |  | ||||||
|         for field in exclude or {}: |  | ||||||
|             if field not in self.fields: |  | ||||||
|                 raise AttributeError("Cannot exclude field '{}' because it is not present in {}" |  | ||||||
|                                      .format(field, rec_name(self))) |  | ||||||
| 
 |  | ||||||
|         stmts = [] |  | ||||||
|         for field in self.fields: |  | ||||||
|             if include is not None and field not in include: |  | ||||||
|                 continue |  | ||||||
|             if exclude is not None and field in exclude: |  | ||||||
|                 continue |  | ||||||
| 
 |  | ||||||
|             shape, direction = self.layout[field] |  | ||||||
|             if not isinstance(shape, Layout) and direction == DIR_NONE: |  | ||||||
|                 raise TypeError("Cannot connect field '{}' of {} because it does not have " |  | ||||||
|                                 "a direction" |  | ||||||
|                                 .format(field, rec_name(self))) |  | ||||||
| 
 |  | ||||||
|             item = self.fields[field] |  | ||||||
|             subord_items = [] |  | ||||||
|             for subord in subordinates: |  | ||||||
|                 if field not in subord.fields: |  | ||||||
|                     raise AttributeError("Cannot connect field '{}' of {} to subordinate {} " |  | ||||||
|                                          "because the subordinate record does not have this field" |  | ||||||
|                                          .format(field, rec_name(self), rec_name(subord))) |  | ||||||
|                 subord_items.append(subord.fields[field]) |  | ||||||
| 
 |  | ||||||
|             if isinstance(shape, Layout): |  | ||||||
|                 sub_include = include[field] if include and field in include else None |  | ||||||
|                 sub_exclude = exclude[field] if exclude and field in exclude else None |  | ||||||
|                 stmts += item.connect(*subord_items, include=sub_include, exclude=sub_exclude) |  | ||||||
|             else: |  | ||||||
|                 if direction == DIR_FANOUT: |  | ||||||
|                     stmts += [sub_item.eq(item) for sub_item in subord_items] |  | ||||||
|                 if direction == DIR_FANIN: |  | ||||||
|                     stmts += [item.eq(reduce(lambda a, b: a | b, subord_items))] |  | ||||||
| 
 |  | ||||||
|         return stmts |  | ||||||
| 
 |  | ||||||
| def _valueproxy(name): |  | ||||||
|     value_func = getattr(Value, name) |  | ||||||
|     @wraps(value_func) |  | ||||||
|     def _wrapper(self, *args, **kwargs): |  | ||||||
|         return value_func(Value.cast(self), *args, **kwargs) |  | ||||||
|     return _wrapper |  | ||||||
| 
 |  | ||||||
| for name in [ |  | ||||||
|         "__bool__", |  | ||||||
|         "__invert__", "__neg__", |  | ||||||
|         "__add__", "__radd__", "__sub__", "__rsub__", |  | ||||||
|         "__mul__", "__rmul__", |  | ||||||
|         "__mod__", "__rmod__", "__floordiv__", "__rfloordiv__", |  | ||||||
|         "__lshift__", "__rlshift__", "__rshift__", "__rrshift__", |  | ||||||
|         "__and__", "__rand__", "__xor__", "__rxor__", "__or__", "__ror__", |  | ||||||
|         "__eq__", "__ne__", "__lt__", "__le__", "__gt__", "__ge__", |  | ||||||
|         "__abs__", "__len__", |  | ||||||
|         "as_unsigned", "as_signed", "bool", "any", "all", "xor", "implies", |  | ||||||
|         "bit_select", "word_select", "matches", |  | ||||||
|         "shift_left", "shift_right", "rotate_left", "rotate_right", "eq" |  | ||||||
|         ]: |  | ||||||
|     setattr(Record, name, _valueproxy(name)) |  | ||||||
| 
 |  | ||||||
| del _valueproxy |  | ||||||
| del name |  | ||||||
|  |  | ||||||
|  | @ -8,7 +8,6 @@ from .ast import * | ||||||
| from .ast import _StatementList | from .ast import _StatementList | ||||||
| from .cd import * | from .cd import * | ||||||
| from .ir import * | from .ir import * | ||||||
| from .rec import * |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| __all__ = ["ValueVisitor", "ValueTransformer", | __all__ = ["ValueVisitor", "ValueTransformer", | ||||||
|  |  | ||||||
|  | @ -1,4 +1,8 @@ | ||||||
|  | import warnings | ||||||
|  | 
 | ||||||
| from .. import * | from .. import * | ||||||
|  | with warnings.catch_warnings(): | ||||||
|  |     warnings.filterwarnings(action="ignore", category=DeprecationWarning) | ||||||
|     from ..hdl.rec import * |     from ..hdl.rec import * | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,6 +1,10 @@ | ||||||
| # amaranth: UnusedElaboratable=no | # amaranth: UnusedElaboratable=no | ||||||
| 
 | 
 | ||||||
|  | import warnings | ||||||
|  | 
 | ||||||
| from amaranth import * | from amaranth import * | ||||||
|  | with warnings.catch_warnings(): | ||||||
|  |     warnings.filterwarnings(action="ignore", category=DeprecationWarning) | ||||||
|     from amaranth.hdl.rec import * |     from amaranth.hdl.rec import * | ||||||
| from amaranth.lib.io import * | from amaranth.lib.io import * | ||||||
| from amaranth.build.dsl import * | from amaranth.build.dsl import * | ||||||
|  |  | ||||||
|  | @ -1,6 +1,10 @@ | ||||||
|  | import warnings | ||||||
|  | 
 | ||||||
| from enum import Enum | from enum import Enum | ||||||
| 
 | 
 | ||||||
| from amaranth.hdl.ast import * | from amaranth.hdl.ast import * | ||||||
|  | with warnings.catch_warnings(): | ||||||
|  |     warnings.filterwarnings(action="ignore", category=DeprecationWarning) | ||||||
|     from amaranth.hdl.rec import * |     from amaranth.hdl.rec import * | ||||||
| 
 | 
 | ||||||
| from .utils import * | from .utils import * | ||||||
|  |  | ||||||
|  | @ -1,4 +1,8 @@ | ||||||
|  | import warnings | ||||||
|  | 
 | ||||||
| from amaranth.hdl import * | from amaranth.hdl import * | ||||||
|  | with warnings.catch_warnings(): | ||||||
|  |     warnings.filterwarnings(action="ignore", category=DeprecationWarning) | ||||||
|     from amaranth.hdl.rec import * |     from amaranth.hdl.rec import * | ||||||
| from amaranth.sim import * | from amaranth.sim import * | ||||||
| from amaranth.lib.io import * | from amaranth.lib.io import * | ||||||
|  |  | ||||||
|  | @ -6,6 +6,8 @@ from amaranth._utils import flatten | ||||||
| from amaranth.hdl.ast import * | from amaranth.hdl.ast import * | ||||||
| from amaranth.hdl.cd import  * | from amaranth.hdl.cd import  * | ||||||
| from amaranth.hdl.mem import * | from amaranth.hdl.mem import * | ||||||
|  | with warnings.catch_warnings(): | ||||||
|  |     warnings.filterwarnings(action="ignore", category=DeprecationWarning) | ||||||
|     from amaranth.hdl.rec import * |     from amaranth.hdl.rec import * | ||||||
| from amaranth.hdl.dsl import  * | from amaranth.hdl.dsl import  * | ||||||
| from amaranth.hdl.ir import * | from amaranth.hdl.ir import * | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Catherine
						Catherine