3066 lines
111 KiB
Rust
3066 lines
111 KiB
Rust
// SPDX-License-Identifier: LGPL-3.0-or-later
|
|
// See Notices.txt for copyright information
|
|
use fayalite::{
|
|
annotations::{Annotation, CustomFirrtlAnnotation},
|
|
array::Array,
|
|
assert_export_firrtl,
|
|
clock::{Clock, ClockDomain},
|
|
hdl_module,
|
|
int::{DynUInt, DynUIntType, IntCmp, SInt, UInt},
|
|
intern::Intern,
|
|
memory::MemBuilder,
|
|
module::transform::simplify_enums::{simplify_enums, SimplifyEnumsKind},
|
|
reset::{SyncReset, ToReset},
|
|
source_location::SourceLocation,
|
|
Value,
|
|
};
|
|
use serde_json::json;
|
|
|
|
#[derive(Value, Hash, Eq, PartialEq, Debug, Clone)]
|
|
pub enum TestEnum {
|
|
A,
|
|
B(UInt<8>),
|
|
C(Array<[UInt<1>; 3]>),
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn my_module(width: usize) {
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
#[hdl]
|
|
let rst: SyncReset = m.input();
|
|
#[hdl]
|
|
let clock_domain: ClockDomain = m.wire();
|
|
m.connect(clock_domain.clock, clk);
|
|
m.connect(clock_domain.reset, rst.to_reset());
|
|
#[hdl]
|
|
let i: UInt<8> = m.input();
|
|
#[hdl]
|
|
let o: Array<[UInt<8>; 3]> = m.output();
|
|
#[hdl]
|
|
let i2: DynUInt = m.input(DynUIntType::new(width));
|
|
#[hdl]
|
|
let o2: DynUInt = m.output(DynUIntType::new(width));
|
|
#[hdl]
|
|
let o3: (UInt<32>, SInt<5>) = m.output();
|
|
m.connect(
|
|
o3,
|
|
#[hdl]
|
|
(5_hdl_u32, -3_hdl_i5),
|
|
);
|
|
#[hdl]
|
|
let m2 = m.instance(module2());
|
|
#[hdl]
|
|
let r: UInt<8> = m.reg_builder().clock_domain(clock_domain).reset(8_hdl_u8);
|
|
m.connect(m2.i, i);
|
|
m.connect(r, m2.o);
|
|
m.connect(
|
|
o,
|
|
#[hdl]
|
|
[r, r, 13_hdl_u8],
|
|
);
|
|
m.connect(o[1], 30_hdl_u8);
|
|
m.connect(o2, i2);
|
|
#[hdl]
|
|
let o4: TestEnum = m.output();
|
|
m.connect(
|
|
o4,
|
|
#[hdl]
|
|
TestEnum::C(Array::from([false.into(), true.into(), false.into()])),
|
|
);
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn module2() {
|
|
#[hdl]
|
|
let i: UInt<8> = m.input();
|
|
#[hdl]
|
|
let o: UInt<8> = m.output();
|
|
#[hdl]
|
|
let w: UInt<8> = m.wire();
|
|
m.connect(w, i);
|
|
m.connect(o, w);
|
|
}
|
|
|
|
#[test]
|
|
fn test_mymodule() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = my_module(3);
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/my_module.fir": r"FIRRTL version 3.2.0
|
|
circuit my_module:
|
|
type Ty0 = {`0`: UInt<32>, `1`: SInt<5>}
|
|
type Ty1 = {|A, B: UInt<8>, C: UInt<1>[3]|}
|
|
type Ty2 = {clock: Clock, `reset`: Reset}
|
|
type Ty3 = {flip i: UInt<8>, o: UInt<8>}
|
|
module my_module: @[module-XXXXXXXXXX.rs 1:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 2:1]
|
|
input rst: UInt<1> @[module-XXXXXXXXXX.rs 3:1]
|
|
input i: UInt<8> @[module-XXXXXXXXXX.rs 7:1]
|
|
output o: UInt<8>[3] @[module-XXXXXXXXXX.rs 8:1]
|
|
input i2: UInt<3> @[module-XXXXXXXXXX.rs 9:1]
|
|
output o2: UInt<3> @[module-XXXXXXXXXX.rs 10:1]
|
|
output o3: Ty0 @[module-XXXXXXXXXX.rs 11:1]
|
|
output o4: Ty1 @[module-XXXXXXXXXX.rs 20:1]
|
|
wire clock_domain: Ty2 @[module-XXXXXXXXXX.rs 4:1]
|
|
connect clock_domain.clock, clk @[module-XXXXXXXXXX.rs 5:1]
|
|
connect clock_domain.`reset`, rst @[module-XXXXXXXXXX.rs 6:1]
|
|
wire _bundle_literal_expr: Ty0
|
|
connect _bundle_literal_expr.`0`, UInt<32>(0h5)
|
|
connect _bundle_literal_expr.`1`, SInt<5>(-0h3)
|
|
connect o3, _bundle_literal_expr @[module-XXXXXXXXXX.rs 12:1]
|
|
inst m2 of module2 @[module-XXXXXXXXXX.rs 13:1]
|
|
regreset r: UInt<8>, clock_domain.clock, clock_domain.`reset`, UInt<8>(0h8) @[module-XXXXXXXXXX.rs 14:1]
|
|
connect m2.i, i @[module-XXXXXXXXXX.rs 15:1]
|
|
connect r, m2.o @[module-XXXXXXXXXX.rs 16:1]
|
|
wire _array_literal_expr: UInt<8>[3]
|
|
connect _array_literal_expr[0], r
|
|
connect _array_literal_expr[1], r
|
|
connect _array_literal_expr[2], UInt<8>(0hD)
|
|
connect o, _array_literal_expr @[module-XXXXXXXXXX.rs 17:1]
|
|
connect o[1], UInt<8>(0h1E) @[module-XXXXXXXXXX.rs 18:1]
|
|
connect o2, i2 @[module-XXXXXXXXXX.rs 19:1]
|
|
wire _literal_array: UInt<1>[3]
|
|
connect _literal_array[0], UInt<1>(0h0)
|
|
connect _literal_array[1], UInt<1>(0h1)
|
|
connect _literal_array[2], UInt<1>(0h0)
|
|
connect o4, {|A, B: UInt<8>, C: UInt<1>[3]|}(C, _literal_array) @[module-XXXXXXXXXX.rs 21:1]
|
|
module module2: @[module-XXXXXXXXXX-2.rs 1:1]
|
|
input i: UInt<8> @[module-XXXXXXXXXX-2.rs 2:1]
|
|
output o: UInt<8> @[module-XXXXXXXXXX-2.rs 3:1]
|
|
wire w: UInt<8> @[module-XXXXXXXXXX-2.rs 4:1]
|
|
connect w, i @[module-XXXXXXXXXX-2.rs 5:1]
|
|
connect o, w @[module-XXXXXXXXXX-2.rs 6:1]
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_partially_written() {
|
|
#[hdl]
|
|
let i: (UInt<8>, UInt<8>) = m.input();
|
|
#[hdl]
|
|
let o: (UInt<8>, UInt<8>) = m.output();
|
|
m.connect(o.0, i.0);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic(expected = "check_partially_written::o.1 is not connected to")]
|
|
fn test_partially_written() {
|
|
check_partially_written();
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_conditionally_written() {
|
|
#[hdl]
|
|
let i: UInt<8> = m.input();
|
|
#[hdl]
|
|
let cond: UInt<1> = m.input();
|
|
#[hdl]
|
|
let o: UInt<8> = m.output();
|
|
#[hdl]
|
|
if cond {
|
|
m.connect(o, i);
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic(expected = "check_conditionally_written::o is not always connected to")]
|
|
fn test_conditionally_written() {
|
|
check_conditionally_written();
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_written_inside_condition() {
|
|
#[hdl]
|
|
let i: UInt<8> = m.input();
|
|
#[hdl]
|
|
let cond: UInt<1> = m.input();
|
|
#[hdl]
|
|
if cond {
|
|
#[hdl]
|
|
let w: UInt<8> = m.wire();
|
|
m.connect(w, i);
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_written_inside_condition() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_written_inside_condition();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_written_inside_condition.fir": r"FIRRTL version 3.2.0
|
|
circuit check_written_inside_condition:
|
|
module check_written_inside_condition: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
input cond: UInt<1> @[module-XXXXXXXXXX.rs 3:1]
|
|
when cond: @[module-XXXXXXXXXX.rs 4:1]
|
|
wire w: UInt<8> @[module-XXXXXXXXXX.rs 5:1]
|
|
connect w, i @[module-XXXXXXXXXX.rs 6:1]
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_written_with_dynamic_index() {
|
|
#[hdl]
|
|
let i: UInt<8> = m.input();
|
|
#[hdl]
|
|
let idx: UInt<8> = m.input();
|
|
#[hdl]
|
|
let w: Array<[UInt<8>; 2]> = m.wire();
|
|
m.connect(w[idx], i);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "is not always connected to"]
|
|
fn test_written_with_dynamic_index() {
|
|
check_written_with_dynamic_index();
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_written_inside_both_if_else() {
|
|
#[hdl]
|
|
let i: UInt<8> = m.input();
|
|
#[hdl]
|
|
let cond: UInt<1> = m.input();
|
|
#[hdl]
|
|
let o: UInt<8> = m.output();
|
|
#[hdl]
|
|
if cond {
|
|
m.connect(o, i);
|
|
} else {
|
|
m.connect(o, 3_hdl_u8);
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_written_inside_both_if_else() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_written_inside_both_if_else();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_written_inside_both_if_else.fir": r"FIRRTL version 3.2.0
|
|
circuit check_written_inside_both_if_else:
|
|
module check_written_inside_both_if_else: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
input cond: UInt<1> @[module-XXXXXXXXXX.rs 3:1]
|
|
output o: UInt<8> @[module-XXXXXXXXXX.rs 4:1]
|
|
when cond: @[module-XXXXXXXXXX.rs 5:1]
|
|
connect o, i @[module-XXXXXXXXXX.rs 6:1]
|
|
else:
|
|
connect o, UInt<8>(0h3) @[module-XXXXXXXXXX.rs 7:1]
|
|
",
|
|
};
|
|
}
|
|
|
|
#[derive(Value, Clone, PartialEq, Eq, Hash, Debug)]
|
|
#[hdl(fixed_type, outline_generated)]
|
|
pub struct TestStruct<T> {
|
|
pub a: T,
|
|
pub b: UInt<8>,
|
|
}
|
|
|
|
#[derive(Value, Clone, PartialEq, Eq, Hash, Debug)]
|
|
#[hdl(fixed_type, outline_generated)]
|
|
pub struct TestStruct2(pub UInt<8>);
|
|
|
|
#[derive(Value, Clone, PartialEq, Eq, Hash, Debug)]
|
|
#[hdl(fixed_type, outline_generated)]
|
|
pub struct TestStruct3;
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_struct_literals() {
|
|
#[hdl]
|
|
let i: UInt<8> = m.input();
|
|
#[hdl]
|
|
let o: TestStruct<UInt<16>> = m.output();
|
|
#[hdl]
|
|
let o2: TestStruct2 = m.output();
|
|
#[hdl]
|
|
let o3: TestStruct3 = m.output();
|
|
m.connect(
|
|
o,
|
|
#[hdl]
|
|
TestStruct::<_> { a: 1234u16, b: i },
|
|
);
|
|
m.connect(
|
|
o2,
|
|
#[hdl]
|
|
TestStruct2(i),
|
|
);
|
|
m.connect(
|
|
o3,
|
|
#[hdl]
|
|
TestStruct3 {},
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn test_struct_literals() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_struct_literals();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_struct_literals.fir": r"FIRRTL version 3.2.0
|
|
circuit check_struct_literals:
|
|
type Ty0 = {a: UInt<16>, b: UInt<8>}
|
|
type Ty1 = {`0`: UInt<8>}
|
|
type Ty2 = {}
|
|
module check_struct_literals: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output o: Ty0 @[module-XXXXXXXXXX.rs 3:1]
|
|
output o2: Ty1 @[module-XXXXXXXXXX.rs 4:1]
|
|
output o3: Ty2 @[module-XXXXXXXXXX.rs 5:1]
|
|
wire _bundle_literal_expr: Ty0
|
|
connect _bundle_literal_expr.a, UInt<16>(0h4D2)
|
|
connect _bundle_literal_expr.b, i
|
|
connect o, _bundle_literal_expr @[module-XXXXXXXXXX.rs 6:1]
|
|
wire _bundle_literal_expr_1: Ty1
|
|
connect _bundle_literal_expr_1.`0`, i
|
|
connect o2, _bundle_literal_expr_1 @[module-XXXXXXXXXX.rs 7:1]
|
|
wire _bundle_literal_expr_2: Ty2
|
|
invalidate _bundle_literal_expr_2
|
|
connect o3, _bundle_literal_expr_2 @[module-XXXXXXXXXX.rs 8:1]
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_enum_literals() {
|
|
#[hdl]
|
|
let i: UInt<8> = m.input();
|
|
#[hdl]
|
|
let o: Option<UInt<8>> = m.output();
|
|
#[hdl]
|
|
let o2: TestEnum = m.output();
|
|
m.connect(
|
|
o,
|
|
#[hdl]
|
|
Some(i),
|
|
);
|
|
#[hdl]
|
|
if i.cmp_eq(0_hdl_u8) {
|
|
m.connect(
|
|
o2,
|
|
#[hdl]
|
|
TestEnum::A,
|
|
);
|
|
} else if i.cmp_gt(8_hdl_u8) {
|
|
m.connect(
|
|
o2,
|
|
#[hdl]
|
|
TestEnum::B(i),
|
|
);
|
|
} else {
|
|
m.connect(
|
|
o2,
|
|
#[hdl]
|
|
TestEnum::C(
|
|
#[hdl]
|
|
[i[0], i[2], i[1]],
|
|
),
|
|
);
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_enum_literals() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_enum_literals();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_enum_literals.fir": r"FIRRTL version 3.2.0
|
|
circuit check_enum_literals:
|
|
type Ty0 = {|None, Some: UInt<8>|}
|
|
type Ty1 = {|A, B: UInt<8>, C: UInt<1>[3]|}
|
|
module check_enum_literals: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output o: Ty0 @[module-XXXXXXXXXX.rs 3:1]
|
|
output o2: Ty1 @[module-XXXXXXXXXX.rs 4:1]
|
|
connect o, {|None, Some: UInt<8>|}(Some, i) @[module-XXXXXXXXXX.rs 5:1]
|
|
when eq(i, UInt<8>(0h0)): @[module-XXXXXXXXXX.rs 6:1]
|
|
connect o2, {|A, B: UInt<8>, C: UInt<1>[3]|}(A) @[module-XXXXXXXXXX.rs 7:1]
|
|
else when gt(i, UInt<8>(0h8)): @[module-XXXXXXXXXX.rs 8:1]
|
|
connect o2, {|A, B: UInt<8>, C: UInt<1>[3]|}(B, i) @[module-XXXXXXXXXX.rs 9:1]
|
|
else:
|
|
wire _array_literal_expr: UInt<1>[3]
|
|
connect _array_literal_expr[0], bits(i, 0, 0)
|
|
connect _array_literal_expr[1], bits(i, 2, 2)
|
|
connect _array_literal_expr[2], bits(i, 1, 1)
|
|
connect o2, {|A, B: UInt<8>, C: UInt<1>[3]|}(C, _array_literal_expr) @[module-XXXXXXXXXX.rs 10:1]
|
|
",
|
|
};
|
|
let orig_m = m.canonical().intern();
|
|
let m = simplify_enums(orig_m, SimplifyEnumsKind::SimplifyToEnumsWithNoBody).unwrap();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_enum_literals.fir": r"FIRRTL version 3.2.0
|
|
circuit check_enum_literals:
|
|
type Ty0 = {|None, Some|}
|
|
type Ty1 = {tag: Ty0, body: UInt<8>}
|
|
type Ty2 = {|A, B, C|}
|
|
type Ty3 = {tag: Ty2, body: UInt<8>}
|
|
type Ty4 = {tag: Ty2, body: UInt<3>}
|
|
module check_enum_literals: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output o: Ty1 @[module-XXXXXXXXXX.rs 3:1]
|
|
output o2: Ty3 @[module-XXXXXXXXXX.rs 4:1]
|
|
wire _bundle_literal_expr: Ty1
|
|
connect _bundle_literal_expr.tag, {|None, Some|}(Some)
|
|
connect _bundle_literal_expr.body, i
|
|
connect o, _bundle_literal_expr @[module-XXXXXXXXXX.rs 5:1]
|
|
wire _literal_bundle: Ty3
|
|
connect _literal_bundle.tag, {|A, B, C|}(A)
|
|
connect _literal_bundle.body, UInt<8>(0h0)
|
|
when eq(i, UInt<8>(0h0)): @[module-XXXXXXXXXX.rs 6:1]
|
|
connect o2, _literal_bundle @[module-XXXXXXXXXX.rs 7:1]
|
|
else when gt(i, UInt<8>(0h8)): @[module-XXXXXXXXXX.rs 8:1]
|
|
wire _bundle_literal_expr_1: Ty3
|
|
connect _bundle_literal_expr_1.tag, {|A, B, C|}(B)
|
|
connect _bundle_literal_expr_1.body, i
|
|
connect o2, _bundle_literal_expr_1 @[module-XXXXXXXXXX.rs 9:1]
|
|
else:
|
|
wire _bundle_literal_expr_2: Ty4
|
|
connect _bundle_literal_expr_2.tag, {|A, B, C|}(C)
|
|
wire _array_literal_expr: UInt<1>[3]
|
|
connect _array_literal_expr[0], bits(i, 0, 0)
|
|
connect _array_literal_expr[1], bits(i, 2, 2)
|
|
connect _array_literal_expr[2], bits(i, 1, 1)
|
|
wire _cast_array_to_bits_expr: UInt<1>[3]
|
|
connect _cast_array_to_bits_expr[0], _array_literal_expr[0]
|
|
connect _cast_array_to_bits_expr[1], _array_literal_expr[1]
|
|
connect _cast_array_to_bits_expr[2], _array_literal_expr[2]
|
|
wire _cast_to_bits_expr: UInt<3>
|
|
connect _cast_to_bits_expr, cat(_cast_array_to_bits_expr[2], cat(_cast_array_to_bits_expr[1], _cast_array_to_bits_expr[0]))
|
|
connect _bundle_literal_expr_2.body, _cast_to_bits_expr
|
|
connect o2, _bundle_literal_expr_2 @[module-XXXXXXXXXX.rs 10:1]
|
|
",
|
|
};
|
|
let m = simplify_enums(orig_m, SimplifyEnumsKind::ReplaceWithBundleOfUInts).unwrap();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_enum_literals.fir": r"FIRRTL version 3.2.0
|
|
circuit check_enum_literals:
|
|
type Ty0 = {tag: UInt<1>, body: UInt<8>}
|
|
type Ty1 = {tag: UInt<2>, body: UInt<8>}
|
|
type Ty2 = {tag: UInt<2>, body: UInt<3>}
|
|
module check_enum_literals: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output o: Ty0 @[module-XXXXXXXXXX.rs 3:1]
|
|
output o2: Ty1 @[module-XXXXXXXXXX.rs 4:1]
|
|
wire _bundle_literal_expr: Ty0
|
|
connect _bundle_literal_expr.tag, UInt<1>(0h1)
|
|
connect _bundle_literal_expr.body, i
|
|
connect o, _bundle_literal_expr @[module-XXXXXXXXXX.rs 5:1]
|
|
wire _literal_bundle: Ty1
|
|
connect _literal_bundle.tag, UInt<2>(0h0)
|
|
connect _literal_bundle.body, UInt<8>(0h0)
|
|
when eq(i, UInt<8>(0h0)): @[module-XXXXXXXXXX.rs 6:1]
|
|
connect o2, _literal_bundle @[module-XXXXXXXXXX.rs 7:1]
|
|
else when gt(i, UInt<8>(0h8)): @[module-XXXXXXXXXX.rs 8:1]
|
|
wire _bundle_literal_expr_1: Ty1
|
|
connect _bundle_literal_expr_1.tag, UInt<2>(0h1)
|
|
connect _bundle_literal_expr_1.body, i
|
|
connect o2, _bundle_literal_expr_1 @[module-XXXXXXXXXX.rs 9:1]
|
|
else:
|
|
wire _bundle_literal_expr_2: Ty2
|
|
connect _bundle_literal_expr_2.tag, UInt<2>(0h2)
|
|
wire _array_literal_expr: UInt<1>[3]
|
|
connect _array_literal_expr[0], bits(i, 0, 0)
|
|
connect _array_literal_expr[1], bits(i, 2, 2)
|
|
connect _array_literal_expr[2], bits(i, 1, 1)
|
|
wire _cast_array_to_bits_expr: UInt<1>[3]
|
|
connect _cast_array_to_bits_expr[0], _array_literal_expr[0]
|
|
connect _cast_array_to_bits_expr[1], _array_literal_expr[1]
|
|
connect _cast_array_to_bits_expr[2], _array_literal_expr[2]
|
|
wire _cast_to_bits_expr: UInt<3>
|
|
connect _cast_to_bits_expr, cat(_cast_array_to_bits_expr[2], cat(_cast_array_to_bits_expr[1], _cast_array_to_bits_expr[0]))
|
|
connect _bundle_literal_expr_2.body, _cast_to_bits_expr
|
|
connect o2, _bundle_literal_expr_2 @[module-XXXXXXXXXX.rs 10:1]
|
|
",
|
|
};
|
|
let m = simplify_enums(orig_m, SimplifyEnumsKind::ReplaceWithUInt).unwrap();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_enum_literals.fir": r"FIRRTL version 3.2.0
|
|
circuit check_enum_literals:
|
|
type Ty0 = {tag: UInt<1>, body: UInt<8>}
|
|
type Ty1 = {tag: UInt<2>, body: UInt<8>}
|
|
type Ty2 = {tag: UInt<2>, body: UInt<3>}
|
|
module check_enum_literals: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output o: UInt<9> @[module-XXXXXXXXXX.rs 3:1]
|
|
output o2: UInt<10> @[module-XXXXXXXXXX.rs 4:1]
|
|
wire _bundle_literal_expr: Ty0
|
|
connect _bundle_literal_expr.tag, UInt<1>(0h1)
|
|
connect _bundle_literal_expr.body, i
|
|
wire _cast_bundle_to_bits_expr: Ty0
|
|
connect _cast_bundle_to_bits_expr.tag, _bundle_literal_expr.tag
|
|
connect _cast_bundle_to_bits_expr.body, _bundle_literal_expr.body
|
|
wire _cast_to_bits_expr: UInt<9>
|
|
connect _cast_to_bits_expr, cat(_cast_bundle_to_bits_expr.body, _cast_bundle_to_bits_expr.tag)
|
|
connect o, _cast_to_bits_expr @[module-XXXXXXXXXX.rs 5:1]
|
|
when eq(i, UInt<8>(0h0)): @[module-XXXXXXXXXX.rs 6:1]
|
|
connect o2, UInt<10>(0h0) @[module-XXXXXXXXXX.rs 7:1]
|
|
else when gt(i, UInt<8>(0h8)): @[module-XXXXXXXXXX.rs 8:1]
|
|
wire _bundle_literal_expr_1: Ty1
|
|
connect _bundle_literal_expr_1.tag, UInt<2>(0h1)
|
|
connect _bundle_literal_expr_1.body, i
|
|
wire _cast_bundle_to_bits_expr_1: Ty1
|
|
connect _cast_bundle_to_bits_expr_1.tag, _bundle_literal_expr_1.tag
|
|
connect _cast_bundle_to_bits_expr_1.body, _bundle_literal_expr_1.body
|
|
wire _cast_to_bits_expr_1: UInt<10>
|
|
connect _cast_to_bits_expr_1, cat(_cast_bundle_to_bits_expr_1.body, _cast_bundle_to_bits_expr_1.tag)
|
|
connect o2, _cast_to_bits_expr_1 @[module-XXXXXXXXXX.rs 9:1]
|
|
else:
|
|
wire _bundle_literal_expr_2: Ty2
|
|
connect _bundle_literal_expr_2.tag, UInt<2>(0h2)
|
|
wire _array_literal_expr: UInt<1>[3]
|
|
connect _array_literal_expr[0], bits(i, 0, 0)
|
|
connect _array_literal_expr[1], bits(i, 2, 2)
|
|
connect _array_literal_expr[2], bits(i, 1, 1)
|
|
wire _cast_array_to_bits_expr: UInt<1>[3]
|
|
connect _cast_array_to_bits_expr[0], _array_literal_expr[0]
|
|
connect _cast_array_to_bits_expr[1], _array_literal_expr[1]
|
|
connect _cast_array_to_bits_expr[2], _array_literal_expr[2]
|
|
wire _cast_to_bits_expr_2: UInt<3>
|
|
connect _cast_to_bits_expr_2, cat(_cast_array_to_bits_expr[2], cat(_cast_array_to_bits_expr[1], _cast_array_to_bits_expr[0]))
|
|
connect _bundle_literal_expr_2.body, _cast_to_bits_expr_2
|
|
wire _cast_bundle_to_bits_expr_2: Ty2
|
|
connect _cast_bundle_to_bits_expr_2.tag, _bundle_literal_expr_2.tag
|
|
connect _cast_bundle_to_bits_expr_2.body, _bundle_literal_expr_2.body
|
|
wire _cast_to_bits_expr_3: UInt<5>
|
|
connect _cast_to_bits_expr_3, cat(_cast_bundle_to_bits_expr_2.body, _cast_bundle_to_bits_expr_2.tag)
|
|
connect o2, _cast_to_bits_expr_3 @[module-XXXXXXXXXX.rs 10:1]
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_struct_enum_match() {
|
|
#[hdl]
|
|
let i1: Option<UInt<8>> = m.input();
|
|
#[hdl]
|
|
let i2: TestEnum = m.input();
|
|
#[hdl]
|
|
let o: Array<[UInt<8>; 5]> = m.output();
|
|
#[hdl]
|
|
if let Some(v) = i1 {
|
|
m.connect(o[0], v);
|
|
} else if let TestEnum::B(v) = i2 {
|
|
m.connect_any(o[0], v + 2_hdl_u8);
|
|
} else {
|
|
m.connect(o[0], 23_hdl_u8);
|
|
}
|
|
#[hdl]
|
|
match i1 {
|
|
Some(_) => m.connect(o[1], 1_hdl_u8),
|
|
None => m.connect(o[1], 0_hdl_u8),
|
|
}
|
|
#[hdl]
|
|
match i2 {
|
|
TestEnum::A => m.connect(o[2], 0_hdl_u8),
|
|
TestEnum::B(v) => m.connect(o[2], v),
|
|
TestEnum::C(v) => m.connect_any(o[2], v[1]),
|
|
}
|
|
#[hdl]
|
|
match i2 {
|
|
TestEnum::A => m.connect(o[3], 0_hdl_u8),
|
|
TestEnum::B { 0: _ } => m.connect(o[3], 1_hdl_u8),
|
|
TestEnum::C(..) => m.connect(o[3], 2_hdl_u8),
|
|
}
|
|
#[hdl]
|
|
match i2 {
|
|
TestEnum::B { 0: _, .. } => m.connect(o[4], 1_hdl_u8),
|
|
TestEnum::C(v, ..) => m.connect_any(o[4], v[2]),
|
|
_ => m.connect(o[4], 0_hdl_u8),
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_struct_enum_match() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_struct_enum_match();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_struct_enum_match.fir": r"FIRRTL version 3.2.0
|
|
circuit check_struct_enum_match:
|
|
type Ty0 = {|None, Some: UInt<8>|}
|
|
type Ty1 = {|A, B: UInt<8>, C: UInt<1>[3]|}
|
|
module check_struct_enum_match: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i1: Ty0 @[module-XXXXXXXXXX.rs 2:1]
|
|
input i2: Ty1 @[module-XXXXXXXXXX.rs 3:1]
|
|
output o: UInt<8>[5] @[module-XXXXXXXXXX.rs 4:1]
|
|
match i1: @[module-XXXXXXXXXX.rs 5:1]
|
|
None:
|
|
match i2: @[module-XXXXXXXXXX.rs 6:1]
|
|
A:
|
|
connect o[0], UInt<8>(0h17) @[module-XXXXXXXXXX.rs 7:1]
|
|
B(_match_arm_value):
|
|
connect o[0], add(_match_arm_value, UInt<8>(0h2)) @[module-XXXXXXXXXX.rs 8:1]
|
|
C(_match_arm_value_1):
|
|
connect o[0], UInt<8>(0h17) @[module-XXXXXXXXXX.rs 7:1]
|
|
Some(_match_arm_value_2):
|
|
connect o[0], _match_arm_value_2 @[module-XXXXXXXXXX.rs 9:1]
|
|
match i1: @[module-XXXXXXXXXX.rs 10:1]
|
|
None:
|
|
connect o[1], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 11:1]
|
|
Some(_match_arm_value_3):
|
|
connect o[1], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 12:1]
|
|
match i2: @[module-XXXXXXXXXX.rs 13:1]
|
|
A:
|
|
connect o[2], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 14:1]
|
|
B(_match_arm_value_4):
|
|
connect o[2], _match_arm_value_4 @[module-XXXXXXXXXX.rs 15:1]
|
|
C(_match_arm_value_5):
|
|
connect o[2], _match_arm_value_5[1] @[module-XXXXXXXXXX.rs 16:1]
|
|
match i2: @[module-XXXXXXXXXX.rs 17:1]
|
|
A:
|
|
connect o[3], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 18:1]
|
|
B(_match_arm_value_6):
|
|
connect o[3], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 19:1]
|
|
C(_match_arm_value_7):
|
|
connect o[3], UInt<8>(0h2) @[module-XXXXXXXXXX.rs 20:1]
|
|
match i2: @[module-XXXXXXXXXX.rs 21:1]
|
|
A:
|
|
connect o[4], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 22:1]
|
|
B(_match_arm_value_8):
|
|
connect o[4], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 23:1]
|
|
C(_match_arm_value_9):
|
|
connect o[4], _match_arm_value_9[2] @[module-XXXXXXXXXX.rs 24:1]
|
|
",
|
|
};
|
|
let orig_m = m.canonical().intern();
|
|
let m = simplify_enums(orig_m, SimplifyEnumsKind::SimplifyToEnumsWithNoBody).unwrap();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_struct_enum_match.fir": r"FIRRTL version 3.2.0
|
|
circuit check_struct_enum_match:
|
|
type Ty0 = {|None, Some|}
|
|
type Ty1 = {tag: Ty0, body: UInt<8>}
|
|
type Ty2 = {|A, B, C|}
|
|
type Ty3 = {tag: Ty2, body: UInt<8>}
|
|
module check_struct_enum_match: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i1: Ty1 @[module-XXXXXXXXXX.rs 2:1]
|
|
input i2: Ty3 @[module-XXXXXXXXXX.rs 3:1]
|
|
output o: UInt<8>[5] @[module-XXXXXXXXXX.rs 4:1]
|
|
match i1.tag: @[module-XXXXXXXXXX.rs 5:1]
|
|
None:
|
|
match i2.tag: @[module-XXXXXXXXXX.rs 6:1]
|
|
A:
|
|
connect o[0], UInt<8>(0h17) @[module-XXXXXXXXXX.rs 7:1]
|
|
B:
|
|
connect o[0], add(bits(i2.body, 7, 0), UInt<8>(0h2)) @[module-XXXXXXXXXX.rs 8:1]
|
|
C:
|
|
connect o[0], UInt<8>(0h17) @[module-XXXXXXXXXX.rs 7:1]
|
|
Some:
|
|
connect o[0], bits(i1.body, 7, 0) @[module-XXXXXXXXXX.rs 9:1]
|
|
match i1.tag: @[module-XXXXXXXXXX.rs 10:1]
|
|
None:
|
|
connect o[1], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 11:1]
|
|
Some:
|
|
connect o[1], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 12:1]
|
|
match i2.tag: @[module-XXXXXXXXXX.rs 13:1]
|
|
A:
|
|
connect o[2], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 14:1]
|
|
B:
|
|
connect o[2], bits(i2.body, 7, 0) @[module-XXXXXXXXXX.rs 15:1]
|
|
C:
|
|
wire _cast_bits_to_array_expr: UInt<1>[3]
|
|
wire _cast_bits_to_array_expr_flattened: UInt<1>[3]
|
|
connect _cast_bits_to_array_expr_flattened[0], bits(bits(i2.body, 2, 0), 0, 0)
|
|
connect _cast_bits_to_array_expr[0], _cast_bits_to_array_expr_flattened[0]
|
|
connect _cast_bits_to_array_expr_flattened[1], bits(bits(i2.body, 2, 0), 1, 1)
|
|
connect _cast_bits_to_array_expr[1], _cast_bits_to_array_expr_flattened[1]
|
|
connect _cast_bits_to_array_expr_flattened[2], bits(bits(i2.body, 2, 0), 2, 2)
|
|
connect _cast_bits_to_array_expr[2], _cast_bits_to_array_expr_flattened[2]
|
|
connect o[2], _cast_bits_to_array_expr[1] @[module-XXXXXXXXXX.rs 16:1]
|
|
match i2.tag: @[module-XXXXXXXXXX.rs 17:1]
|
|
A:
|
|
connect o[3], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 18:1]
|
|
B:
|
|
connect o[3], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 19:1]
|
|
C:
|
|
connect o[3], UInt<8>(0h2) @[module-XXXXXXXXXX.rs 20:1]
|
|
match i2.tag: @[module-XXXXXXXXXX.rs 21:1]
|
|
A:
|
|
connect o[4], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 22:1]
|
|
B:
|
|
connect o[4], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 23:1]
|
|
C:
|
|
wire _cast_bits_to_array_expr_1: UInt<1>[3]
|
|
wire _cast_bits_to_array_expr_flattened_1: UInt<1>[3]
|
|
connect _cast_bits_to_array_expr_flattened_1[0], bits(bits(i2.body, 2, 0), 0, 0)
|
|
connect _cast_bits_to_array_expr_1[0], _cast_bits_to_array_expr_flattened_1[0]
|
|
connect _cast_bits_to_array_expr_flattened_1[1], bits(bits(i2.body, 2, 0), 1, 1)
|
|
connect _cast_bits_to_array_expr_1[1], _cast_bits_to_array_expr_flattened_1[1]
|
|
connect _cast_bits_to_array_expr_flattened_1[2], bits(bits(i2.body, 2, 0), 2, 2)
|
|
connect _cast_bits_to_array_expr_1[2], _cast_bits_to_array_expr_flattened_1[2]
|
|
connect o[4], _cast_bits_to_array_expr_1[2] @[module-XXXXXXXXXX.rs 24:1]
|
|
",
|
|
};
|
|
let m = simplify_enums(orig_m, SimplifyEnumsKind::ReplaceWithBundleOfUInts).unwrap();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_struct_enum_match.fir": r"FIRRTL version 3.2.0
|
|
circuit check_struct_enum_match:
|
|
type Ty0 = {tag: UInt<1>, body: UInt<8>}
|
|
type Ty1 = {tag: UInt<2>, body: UInt<8>}
|
|
module check_struct_enum_match: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i1: Ty0 @[module-XXXXXXXXXX.rs 2:1]
|
|
input i2: Ty1 @[module-XXXXXXXXXX.rs 3:1]
|
|
output o: UInt<8>[5] @[module-XXXXXXXXXX.rs 4:1]
|
|
when eq(i1.tag, UInt<1>(0h0)): @[module-XXXXXXXXXX.rs 5:1]
|
|
when eq(i2.tag, UInt<2>(0h0)): @[module-XXXXXXXXXX.rs 6:1]
|
|
connect o[0], UInt<8>(0h17) @[module-XXXXXXXXXX.rs 7:1]
|
|
else when eq(i2.tag, UInt<2>(0h1)): @[module-XXXXXXXXXX.rs 6:1]
|
|
connect o[0], add(bits(i2.body, 7, 0), UInt<8>(0h2)) @[module-XXXXXXXXXX.rs 8:1]
|
|
else:
|
|
connect o[0], UInt<8>(0h17) @[module-XXXXXXXXXX.rs 7:1]
|
|
else:
|
|
connect o[0], bits(i1.body, 7, 0) @[module-XXXXXXXXXX.rs 9:1]
|
|
when eq(i1.tag, UInt<1>(0h0)): @[module-XXXXXXXXXX.rs 10:1]
|
|
connect o[1], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 11:1]
|
|
else:
|
|
connect o[1], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 12:1]
|
|
when eq(i2.tag, UInt<2>(0h0)): @[module-XXXXXXXXXX.rs 13:1]
|
|
connect o[2], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 14:1]
|
|
else when eq(i2.tag, UInt<2>(0h1)): @[module-XXXXXXXXXX.rs 13:1]
|
|
connect o[2], bits(i2.body, 7, 0) @[module-XXXXXXXXXX.rs 15:1]
|
|
else:
|
|
wire _cast_bits_to_array_expr: UInt<1>[3]
|
|
wire _cast_bits_to_array_expr_flattened: UInt<1>[3]
|
|
connect _cast_bits_to_array_expr_flattened[0], bits(bits(i2.body, 2, 0), 0, 0)
|
|
connect _cast_bits_to_array_expr[0], _cast_bits_to_array_expr_flattened[0]
|
|
connect _cast_bits_to_array_expr_flattened[1], bits(bits(i2.body, 2, 0), 1, 1)
|
|
connect _cast_bits_to_array_expr[1], _cast_bits_to_array_expr_flattened[1]
|
|
connect _cast_bits_to_array_expr_flattened[2], bits(bits(i2.body, 2, 0), 2, 2)
|
|
connect _cast_bits_to_array_expr[2], _cast_bits_to_array_expr_flattened[2]
|
|
connect o[2], _cast_bits_to_array_expr[1] @[module-XXXXXXXXXX.rs 16:1]
|
|
when eq(i2.tag, UInt<2>(0h0)): @[module-XXXXXXXXXX.rs 17:1]
|
|
connect o[3], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 18:1]
|
|
else when eq(i2.tag, UInt<2>(0h1)): @[module-XXXXXXXXXX.rs 17:1]
|
|
connect o[3], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 19:1]
|
|
else:
|
|
connect o[3], UInt<8>(0h2) @[module-XXXXXXXXXX.rs 20:1]
|
|
when eq(i2.tag, UInt<2>(0h0)): @[module-XXXXXXXXXX.rs 21:1]
|
|
connect o[4], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 22:1]
|
|
else when eq(i2.tag, UInt<2>(0h1)): @[module-XXXXXXXXXX.rs 21:1]
|
|
connect o[4], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 23:1]
|
|
else:
|
|
wire _cast_bits_to_array_expr_1: UInt<1>[3]
|
|
wire _cast_bits_to_array_expr_flattened_1: UInt<1>[3]
|
|
connect _cast_bits_to_array_expr_flattened_1[0], bits(bits(i2.body, 2, 0), 0, 0)
|
|
connect _cast_bits_to_array_expr_1[0], _cast_bits_to_array_expr_flattened_1[0]
|
|
connect _cast_bits_to_array_expr_flattened_1[1], bits(bits(i2.body, 2, 0), 1, 1)
|
|
connect _cast_bits_to_array_expr_1[1], _cast_bits_to_array_expr_flattened_1[1]
|
|
connect _cast_bits_to_array_expr_flattened_1[2], bits(bits(i2.body, 2, 0), 2, 2)
|
|
connect _cast_bits_to_array_expr_1[2], _cast_bits_to_array_expr_flattened_1[2]
|
|
connect o[4], _cast_bits_to_array_expr_1[2] @[module-XXXXXXXXXX.rs 24:1]
|
|
",
|
|
};
|
|
let m = simplify_enums(orig_m, SimplifyEnumsKind::ReplaceWithUInt).unwrap();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_struct_enum_match.fir": r"FIRRTL version 3.2.0
|
|
circuit check_struct_enum_match:
|
|
module check_struct_enum_match: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i1: UInt<9> @[module-XXXXXXXXXX.rs 2:1]
|
|
input i2: UInt<10> @[module-XXXXXXXXXX.rs 3:1]
|
|
output o: UInt<8>[5] @[module-XXXXXXXXXX.rs 4:1]
|
|
when eq(bits(i1, 0, 0), UInt<1>(0h0)): @[module-XXXXXXXXXX.rs 5:1]
|
|
when eq(bits(i2, 1, 0), UInt<2>(0h0)): @[module-XXXXXXXXXX.rs 6:1]
|
|
connect o[0], UInt<8>(0h17) @[module-XXXXXXXXXX.rs 7:1]
|
|
else when eq(bits(i2, 1, 0), UInt<2>(0h1)): @[module-XXXXXXXXXX.rs 6:1]
|
|
connect o[0], add(bits(bits(i2, 9, 2), 7, 0), UInt<8>(0h2)) @[module-XXXXXXXXXX.rs 8:1]
|
|
else:
|
|
connect o[0], UInt<8>(0h17) @[module-XXXXXXXXXX.rs 7:1]
|
|
else:
|
|
connect o[0], bits(bits(i1, 8, 1), 7, 0) @[module-XXXXXXXXXX.rs 9:1]
|
|
when eq(bits(i1, 0, 0), UInt<1>(0h0)): @[module-XXXXXXXXXX.rs 10:1]
|
|
connect o[1], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 11:1]
|
|
else:
|
|
connect o[1], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 12:1]
|
|
when eq(bits(i2, 1, 0), UInt<2>(0h0)): @[module-XXXXXXXXXX.rs 13:1]
|
|
connect o[2], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 14:1]
|
|
else when eq(bits(i2, 1, 0), UInt<2>(0h1)): @[module-XXXXXXXXXX.rs 13:1]
|
|
connect o[2], bits(bits(i2, 9, 2), 7, 0) @[module-XXXXXXXXXX.rs 15:1]
|
|
else:
|
|
wire _cast_bits_to_array_expr: UInt<1>[3]
|
|
wire _cast_bits_to_array_expr_flattened: UInt<1>[3]
|
|
connect _cast_bits_to_array_expr_flattened[0], bits(bits(bits(i2, 9, 2), 2, 0), 0, 0)
|
|
connect _cast_bits_to_array_expr[0], _cast_bits_to_array_expr_flattened[0]
|
|
connect _cast_bits_to_array_expr_flattened[1], bits(bits(bits(i2, 9, 2), 2, 0), 1, 1)
|
|
connect _cast_bits_to_array_expr[1], _cast_bits_to_array_expr_flattened[1]
|
|
connect _cast_bits_to_array_expr_flattened[2], bits(bits(bits(i2, 9, 2), 2, 0), 2, 2)
|
|
connect _cast_bits_to_array_expr[2], _cast_bits_to_array_expr_flattened[2]
|
|
connect o[2], _cast_bits_to_array_expr[1] @[module-XXXXXXXXXX.rs 16:1]
|
|
when eq(bits(i2, 1, 0), UInt<2>(0h0)): @[module-XXXXXXXXXX.rs 17:1]
|
|
connect o[3], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 18:1]
|
|
else when eq(bits(i2, 1, 0), UInt<2>(0h1)): @[module-XXXXXXXXXX.rs 17:1]
|
|
connect o[3], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 19:1]
|
|
else:
|
|
connect o[3], UInt<8>(0h2) @[module-XXXXXXXXXX.rs 20:1]
|
|
when eq(bits(i2, 1, 0), UInt<2>(0h0)): @[module-XXXXXXXXXX.rs 21:1]
|
|
connect o[4], UInt<8>(0h0) @[module-XXXXXXXXXX.rs 22:1]
|
|
else when eq(bits(i2, 1, 0), UInt<2>(0h1)): @[module-XXXXXXXXXX.rs 21:1]
|
|
connect o[4], UInt<8>(0h1) @[module-XXXXXXXXXX.rs 23:1]
|
|
else:
|
|
wire _cast_bits_to_array_expr_1: UInt<1>[3]
|
|
wire _cast_bits_to_array_expr_flattened_1: UInt<1>[3]
|
|
connect _cast_bits_to_array_expr_flattened_1[0], bits(bits(bits(i2, 9, 2), 2, 0), 0, 0)
|
|
connect _cast_bits_to_array_expr_1[0], _cast_bits_to_array_expr_flattened_1[0]
|
|
connect _cast_bits_to_array_expr_flattened_1[1], bits(bits(bits(i2, 9, 2), 2, 0), 1, 1)
|
|
connect _cast_bits_to_array_expr_1[1], _cast_bits_to_array_expr_flattened_1[1]
|
|
connect _cast_bits_to_array_expr_flattened_1[2], bits(bits(bits(i2, 9, 2), 2, 0), 2, 2)
|
|
connect _cast_bits_to_array_expr_1[2], _cast_bits_to_array_expr_flattened_1[2]
|
|
connect o[4], _cast_bits_to_array_expr_1[2] @[module-XXXXXXXXXX.rs 24:1]
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_extern_module() {
|
|
#[hdl]
|
|
let i1: UInt<8> = m.input();
|
|
#[hdl]
|
|
let i2: Clock = m.input();
|
|
#[hdl]
|
|
let o: SInt<12> = m.output();
|
|
#[hdl_module(extern)]
|
|
fn extern_module() {
|
|
#[hdl]
|
|
let i1: UInt<8> = m.input();
|
|
#[hdl]
|
|
let i2: Clock = m.input();
|
|
#[hdl]
|
|
let o: SInt<12> = m.output();
|
|
m.verilog_name("verilog_module");
|
|
m.parameter_int("int_param", 0xFEDCBA9876543210u64 as i64);
|
|
m.parameter_str("str_param", "foo\nbar\n𝄞");
|
|
m.parameter_raw_verilog("raw_param", "12'hxzx");
|
|
}
|
|
#[hdl]
|
|
let submodule = m.instance(extern_module());
|
|
m.connect(submodule.i1, i1);
|
|
m.connect(submodule.i2, i2);
|
|
m.connect(o, submodule.o);
|
|
}
|
|
|
|
#[test]
|
|
fn test_extern_module() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_extern_module();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_extern_module.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_extern_module:
|
|
type Ty0 = {flip i1: UInt<8>, flip i2: Clock, o: SInt<12>}
|
|
module check_extern_module: @[module-XXXXXXXXXX.rs 1:1]
|
|
input i1: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
input i2: Clock @[module-XXXXXXXXXX.rs 3:1]
|
|
output o: SInt<12> @[module-XXXXXXXXXX.rs 4:1]
|
|
inst submodule of extern_module @[module-XXXXXXXXXX.rs 9:1]
|
|
connect submodule.i1, i1 @[module-XXXXXXXXXX.rs 10:1]
|
|
connect submodule.i2, i2 @[module-XXXXXXXXXX.rs 11:1]
|
|
connect o, submodule.o @[module-XXXXXXXXXX.rs 12:1]
|
|
extmodule extern_module: @[module-XXXXXXXXXX.rs 5:1]
|
|
input i1: UInt<8> @[module-XXXXXXXXXX.rs 6:1]
|
|
input i2: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
output o: SInt<12> @[module-XXXXXXXXXX.rs 8:1]
|
|
defname = verilog_module
|
|
parameter int_param = -81985529216486896
|
|
parameter str_param = "foo\nbar\n\F0\9D\84\9E"
|
|
parameter raw_param = '12\'hxzx'
|
|
"#,
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_memory() {
|
|
#[hdl]
|
|
let raddr: UInt<8> = m.input();
|
|
#[hdl]
|
|
let rdata: UInt<8> = m.output();
|
|
#[hdl]
|
|
let waddr: UInt<8> = m.input();
|
|
#[hdl]
|
|
let wdata: UInt<8> = m.input();
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
#[hdl]
|
|
let mut mem: MemBuilder<[UInt<8>; 0x100]> = m.memory_array();
|
|
let read_port = mem.new_read_port();
|
|
m.connect_any(read_port.addr, raddr);
|
|
m.connect(read_port.en, 1_hdl_u1);
|
|
m.connect(read_port.clk, clk);
|
|
m.connect(rdata, read_port.data);
|
|
let write_port = mem.new_write_port();
|
|
m.connect_any(write_port.addr, waddr);
|
|
m.connect(write_port.en, 1_hdl_u1);
|
|
m.connect(write_port.clk, clk);
|
|
m.connect(write_port.data, wdata);
|
|
m.connect(write_port.mask, 1_hdl_u1);
|
|
}
|
|
|
|
#[test]
|
|
fn test_memory() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_memory();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory.fir": r"FIRRTL version 3.2.0
|
|
circuit check_memory:
|
|
type Ty0 = {addr: UInt<8>, en: UInt<1>, clk: Clock, flip data: UInt<8>}
|
|
type Ty1 = {addr: UInt<8>, en: UInt<1>, clk: Clock, data: UInt<8>, mask: UInt<1>}
|
|
module check_memory: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: UInt<8> @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<8> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: UInt<8> @[module-XXXXXXXXXX.rs 5:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 6:1]
|
|
mem `mem`: @[module-XXXXXXXXXX.rs 7:1]
|
|
data-type => UInt<8>
|
|
depth => 256
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
connect `mem`.r0.addr, raddr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect `mem`.r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 10:1]
|
|
connect `mem`.r0.clk, clk @[module-XXXXXXXXXX.rs 11:1]
|
|
connect rdata, `mem`.r0.data @[module-XXXXXXXXXX.rs 12:1]
|
|
connect `mem`.w1.addr, waddr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect `mem`.w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 15:1]
|
|
connect `mem`.w1.clk, clk @[module-XXXXXXXXXX.rs 16:1]
|
|
connect `mem`.w1.data, wdata @[module-XXXXXXXXXX.rs 17:1]
|
|
connect `mem`.w1.mask, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 18:1]
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_memory_init() {
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
#[hdl]
|
|
let raddr1: UInt<8> = m.input();
|
|
#[hdl]
|
|
let rdata1: UInt<8> = m.output();
|
|
#[hdl]
|
|
let waddr1: UInt<8> = m.input();
|
|
#[hdl]
|
|
let wdata1: UInt<8> = m.input();
|
|
let mem_init1 = Vec::from_iter((0..0x100).map(|i| (i * i) as u8));
|
|
#[hdl]
|
|
let mut mem1 = m.memory_with_init(mem_init1);
|
|
let read_port1 = mem1.new_read_port();
|
|
m.connect_any(read_port1.addr, raddr1);
|
|
m.connect(read_port1.en, 1_hdl_u1);
|
|
m.connect(read_port1.clk, clk);
|
|
m.connect(rdata1, read_port1.data);
|
|
let write_port1 = mem1.new_write_port();
|
|
m.connect_any(write_port1.addr, waddr1);
|
|
m.connect(write_port1.en, 1_hdl_u1);
|
|
m.connect(write_port1.clk, clk);
|
|
m.connect(write_port1.data, wdata1);
|
|
m.connect(write_port1.mask, 1_hdl_u1);
|
|
|
|
#[hdl]
|
|
let raddr2: UInt<4> = m.input();
|
|
#[hdl]
|
|
let rdata2: UInt<31> = m.output();
|
|
#[hdl]
|
|
let waddr2: UInt<4> = m.input();
|
|
#[hdl]
|
|
let wdata2: UInt<31> = m.input();
|
|
let mem_init2 = Vec::from_iter((0..0x10).map(|i| UInt::new(i * i * i)));
|
|
#[hdl]
|
|
let mut mem2 = m.memory_with_init(mem_init2);
|
|
let read_port2 = mem2.new_read_port();
|
|
m.connect_any(read_port2.addr, raddr2);
|
|
m.connect(read_port2.en, 1_hdl_u1);
|
|
m.connect(read_port2.clk, clk);
|
|
m.connect(rdata2, read_port2.data);
|
|
let write_port2 = mem2.new_write_port();
|
|
m.connect_any(write_port2.addr, waddr2);
|
|
m.connect(write_port2.en, 1_hdl_u1);
|
|
m.connect(write_port2.clk, clk);
|
|
m.connect(write_port2.data, wdata2);
|
|
m.connect(write_port2.mask, 1_hdl_u1);
|
|
}
|
|
|
|
#[test]
|
|
fn test_memory_init() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_memory_init();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_init.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_memory_init: %[[
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_init/mem1.mem",
|
|
"hexOrBinary": "h",
|
|
"target": "~check_memory_init|check_memory_init>mem1"
|
|
},
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_init/mem2.mem",
|
|
"hexOrBinary": "b",
|
|
"target": "~check_memory_init|check_memory_init>mem2"
|
|
}
|
|
]]
|
|
type Ty0 = {addr: UInt<8>, en: UInt<1>, clk: Clock, flip data: UInt<8>}
|
|
type Ty1 = {addr: UInt<8>, en: UInt<1>, clk: Clock, data: UInt<8>, mask: UInt<1>}
|
|
type Ty2 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<31>}
|
|
type Ty3 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<31>, mask: UInt<1>}
|
|
module check_memory_init: @[module-XXXXXXXXXX.rs 1:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 2:1]
|
|
input raddr1: UInt<8> @[module-XXXXXXXXXX.rs 3:1]
|
|
output rdata1: UInt<8> @[module-XXXXXXXXXX.rs 4:1]
|
|
input waddr1: UInt<8> @[module-XXXXXXXXXX.rs 5:1]
|
|
input wdata1: UInt<8> @[module-XXXXXXXXXX.rs 6:1]
|
|
input raddr2: UInt<4> @[module-XXXXXXXXXX.rs 19:1]
|
|
output rdata2: UInt<31> @[module-XXXXXXXXXX.rs 20:1]
|
|
input waddr2: UInt<4> @[module-XXXXXXXXXX.rs 21:1]
|
|
input wdata2: UInt<31> @[module-XXXXXXXXXX.rs 22:1]
|
|
mem mem1: @[module-XXXXXXXXXX.rs 7:1]
|
|
data-type => UInt<8>
|
|
depth => 256
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
mem mem2: @[module-XXXXXXXXXX.rs 23:1]
|
|
data-type => UInt<31>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
connect mem1.r0.addr, raddr1 @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem1.r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 10:1]
|
|
connect mem1.r0.clk, clk @[module-XXXXXXXXXX.rs 11:1]
|
|
connect rdata1, mem1.r0.data @[module-XXXXXXXXXX.rs 12:1]
|
|
connect mem1.w1.addr, waddr1 @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem1.w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 15:1]
|
|
connect mem1.w1.clk, clk @[module-XXXXXXXXXX.rs 16:1]
|
|
connect mem1.w1.data, wdata1 @[module-XXXXXXXXXX.rs 17:1]
|
|
connect mem1.w1.mask, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 18:1]
|
|
connect mem2.r0.addr, raddr2 @[module-XXXXXXXXXX.rs 25:1]
|
|
connect mem2.r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 26:1]
|
|
connect mem2.r0.clk, clk @[module-XXXXXXXXXX.rs 27:1]
|
|
connect rdata2, mem2.r0.data @[module-XXXXXXXXXX.rs 28:1]
|
|
connect mem2.w1.addr, waddr2 @[module-XXXXXXXXXX.rs 30:1]
|
|
connect mem2.w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 31:1]
|
|
connect mem2.w1.clk, clk @[module-XXXXXXXXXX.rs 32:1]
|
|
connect mem2.w1.data, wdata2 @[module-XXXXXXXXXX.rs 33:1]
|
|
connect mem2.w1.mask, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 34:1]
|
|
"#,
|
|
"/test/check_memory_init/mem1.mem": r"00
|
|
01
|
|
04
|
|
09
|
|
10
|
|
19
|
|
24
|
|
31
|
|
40
|
|
51
|
|
64
|
|
79
|
|
90
|
|
a9
|
|
c4
|
|
e1
|
|
00
|
|
21
|
|
44
|
|
69
|
|
90
|
|
b9
|
|
e4
|
|
11
|
|
40
|
|
71
|
|
a4
|
|
d9
|
|
10
|
|
49
|
|
84
|
|
c1
|
|
00
|
|
41
|
|
84
|
|
c9
|
|
10
|
|
59
|
|
a4
|
|
f1
|
|
40
|
|
91
|
|
e4
|
|
39
|
|
90
|
|
e9
|
|
44
|
|
a1
|
|
00
|
|
61
|
|
c4
|
|
29
|
|
90
|
|
f9
|
|
64
|
|
d1
|
|
40
|
|
b1
|
|
24
|
|
99
|
|
10
|
|
89
|
|
04
|
|
81
|
|
00
|
|
81
|
|
04
|
|
89
|
|
10
|
|
99
|
|
24
|
|
b1
|
|
40
|
|
d1
|
|
64
|
|
f9
|
|
90
|
|
29
|
|
c4
|
|
61
|
|
00
|
|
a1
|
|
44
|
|
e9
|
|
90
|
|
39
|
|
e4
|
|
91
|
|
40
|
|
f1
|
|
a4
|
|
59
|
|
10
|
|
c9
|
|
84
|
|
41
|
|
00
|
|
c1
|
|
84
|
|
49
|
|
10
|
|
d9
|
|
a4
|
|
71
|
|
40
|
|
11
|
|
e4
|
|
b9
|
|
90
|
|
69
|
|
44
|
|
21
|
|
00
|
|
e1
|
|
c4
|
|
a9
|
|
90
|
|
79
|
|
64
|
|
51
|
|
40
|
|
31
|
|
24
|
|
19
|
|
10
|
|
09
|
|
04
|
|
01
|
|
00
|
|
01
|
|
04
|
|
09
|
|
10
|
|
19
|
|
24
|
|
31
|
|
40
|
|
51
|
|
64
|
|
79
|
|
90
|
|
a9
|
|
c4
|
|
e1
|
|
00
|
|
21
|
|
44
|
|
69
|
|
90
|
|
b9
|
|
e4
|
|
11
|
|
40
|
|
71
|
|
a4
|
|
d9
|
|
10
|
|
49
|
|
84
|
|
c1
|
|
00
|
|
41
|
|
84
|
|
c9
|
|
10
|
|
59
|
|
a4
|
|
f1
|
|
40
|
|
91
|
|
e4
|
|
39
|
|
90
|
|
e9
|
|
44
|
|
a1
|
|
00
|
|
61
|
|
c4
|
|
29
|
|
90
|
|
f9
|
|
64
|
|
d1
|
|
40
|
|
b1
|
|
24
|
|
99
|
|
10
|
|
89
|
|
04
|
|
81
|
|
00
|
|
81
|
|
04
|
|
89
|
|
10
|
|
99
|
|
24
|
|
b1
|
|
40
|
|
d1
|
|
64
|
|
f9
|
|
90
|
|
29
|
|
c4
|
|
61
|
|
00
|
|
a1
|
|
44
|
|
e9
|
|
90
|
|
39
|
|
e4
|
|
91
|
|
40
|
|
f1
|
|
a4
|
|
59
|
|
10
|
|
c9
|
|
84
|
|
41
|
|
00
|
|
c1
|
|
84
|
|
49
|
|
10
|
|
d9
|
|
a4
|
|
71
|
|
40
|
|
11
|
|
e4
|
|
b9
|
|
90
|
|
69
|
|
44
|
|
21
|
|
00
|
|
e1
|
|
c4
|
|
a9
|
|
90
|
|
79
|
|
64
|
|
51
|
|
40
|
|
31
|
|
24
|
|
19
|
|
10
|
|
09
|
|
04
|
|
01
|
|
",
|
|
"/test/check_memory_init/mem2.mem": r"0000000000000000000000000000000
|
|
0000000000000000000000000000001
|
|
0000000000000000000000000001000
|
|
0000000000000000000000000011011
|
|
0000000000000000000000001000000
|
|
0000000000000000000000001111101
|
|
0000000000000000000000011011000
|
|
0000000000000000000000101010111
|
|
0000000000000000000001000000000
|
|
0000000000000000000001011011001
|
|
0000000000000000000001111101000
|
|
0000000000000000000010100110011
|
|
0000000000000000000011011000000
|
|
0000000000000000000100010010101
|
|
0000000000000000000101010111000
|
|
0000000000000000000110100101111
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_memory_of_array() {
|
|
#[hdl]
|
|
let raddr: UInt<4> = m.input();
|
|
#[hdl]
|
|
let rdata: Array<[UInt<8>; 3]> = m.output();
|
|
#[hdl]
|
|
let waddr: UInt<4> = m.input();
|
|
#[hdl]
|
|
let wdata: Array<[UInt<8>; 3]> = m.input();
|
|
#[hdl]
|
|
let wmask: Array<[UInt<1>; 3]> = m.input();
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
let mem_init = Vec::from_iter((0..0x10).map(|i| {
|
|
#[hdl]
|
|
[UInt::new(i), UInt::new(i * i), UInt::new(i * i * i)]
|
|
}));
|
|
#[hdl]
|
|
let mut mem = m.memory_with_init(mem_init);
|
|
let read_port = mem.new_read_port();
|
|
m.connect_any(read_port.addr, raddr);
|
|
m.connect(read_port.en, 1_hdl_u1);
|
|
m.connect(read_port.clk, clk);
|
|
m.connect(rdata, read_port.data);
|
|
let write_port = mem.new_write_port();
|
|
m.connect_any(write_port.addr, waddr);
|
|
m.connect(write_port.en, 1_hdl_u1);
|
|
m.connect(write_port.clk, clk);
|
|
m.connect(write_port.data, wdata);
|
|
m.connect(write_port.mask, wmask);
|
|
}
|
|
|
|
#[test]
|
|
fn test_memory_of_array() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_memory_of_array();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_of_array.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_memory_of_array: %[[
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_array/mem.mem",
|
|
"hexOrBinary": "h",
|
|
"target": "~check_memory_of_array|check_memory_of_array>mem"
|
|
}
|
|
]]
|
|
type Ty0 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<8>[3]}
|
|
type Ty1 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<8>[3], mask: UInt<1>[3]}
|
|
module check_memory_of_array: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<4> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: UInt<8>[3] @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<4> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: UInt<8>[3] @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: UInt<1>[3] @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem `mem`: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<8>[3]
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
connect `mem`.r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect `mem`.r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect `mem`.r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, `mem`.r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect `mem`.w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect `mem`.w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect `mem`.w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect `mem`.w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect `mem`.w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
"#,
|
|
"/test/check_memory_of_array/mem.mem": r"000000
|
|
010101
|
|
080402
|
|
1b0903
|
|
401004
|
|
7d1905
|
|
d82406
|
|
573107
|
|
004008
|
|
d95109
|
|
e8640a
|
|
33790b
|
|
c0900c
|
|
95a90d
|
|
b8c40e
|
|
2fe10f
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_memory_of_arrays() {
|
|
#[hdl]
|
|
let raddr: UInt<4> = m.input();
|
|
#[hdl]
|
|
let rdata: Array<[Array<[UInt<8>; 2]>; 3]> = m.output();
|
|
#[hdl]
|
|
let waddr: UInt<4> = m.input();
|
|
#[hdl]
|
|
let wdata: Array<[Array<[UInt<8>; 2]>; 3]> = m.input();
|
|
#[hdl]
|
|
let wmask: Array<[Array<[UInt<1>; 2]>; 3]> = m.input();
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
let mem_init = Vec::from_iter((0..0x10).map(|i| {
|
|
#[hdl]
|
|
[
|
|
#[hdl]
|
|
[UInt::new(i), UInt::new(i * i)],
|
|
#[hdl]
|
|
[UInt::new(i * i * 2), UInt::new(i * i * 3)],
|
|
#[hdl]
|
|
[UInt::new(i * i * i), UInt::new(i * 2)],
|
|
]
|
|
}));
|
|
#[hdl]
|
|
let mut mem = m.memory_with_init(mem_init);
|
|
let read_port = mem.new_read_port();
|
|
m.connect_any(read_port.addr, raddr);
|
|
m.connect(read_port.en, 1_hdl_u1);
|
|
m.connect(read_port.clk, clk);
|
|
m.connect(rdata, read_port.data);
|
|
let write_port = mem.new_write_port();
|
|
m.connect_any(write_port.addr, waddr);
|
|
m.connect(write_port.en, 1_hdl_u1);
|
|
m.connect(write_port.clk, clk);
|
|
m.connect(write_port.data, wdata);
|
|
m.connect(write_port.mask, wmask);
|
|
}
|
|
|
|
#[test]
|
|
fn test_memory_of_arrays() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_memory_of_arrays();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_of_arrays.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_memory_of_arrays: %[[
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_arrays/mem_1.mem",
|
|
"hexOrBinary": "h",
|
|
"target": "~check_memory_of_arrays|check_memory_of_arrays>mem_1"
|
|
}
|
|
]]
|
|
type Ty0 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<8>[2][3]}
|
|
type Ty1 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<8>[2][3], mask: UInt<1>[2][3]}
|
|
type Ty2 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<8>[6]}
|
|
type Ty3 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<8>[6], mask: UInt<1>[6]}
|
|
module check_memory_of_arrays: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<4> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: UInt<8>[2][3] @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<4> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: UInt<8>[2][3] @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: UInt<1>[2][3] @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem mem_1: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<8>[6]
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
wire mem_r0: Ty0 @[module-XXXXXXXXXX.rs 9:1]
|
|
wire mem_w1: Ty1 @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data[0][0], mem_1.r0.data[0] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_r0.data[0][1], mem_1.r0.data[1] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_r0.data[1][0], mem_1.r0.data[2] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_r0.data[1][1], mem_1.r0.data[3] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_r0.data[2][0], mem_1.r0.data[4] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_r0.data[2][1], mem_1.r0.data[5] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.w1.data[0], mem_w1.data[0][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask[0], mem_w1.mask[0][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.data[1], mem_w1.data[0][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask[1], mem_w1.mask[0][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.data[2], mem_w1.data[1][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask[2], mem_w1.mask[1][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.data[3], mem_w1.data[1][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask[3], mem_w1.mask[1][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.data[4], mem_w1.data[2][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask[4], mem_w1.mask[2][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.data[5], mem_w1.data[2][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask[5], mem_w1.mask[2][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect mem_r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect mem_r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, mem_r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect mem_w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect mem_w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect mem_w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect mem_w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect mem_w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
"#,
|
|
"/test/check_memory_of_arrays/mem_1.mem": r"000000000000
|
|
020103020101
|
|
04080c080402
|
|
061b1b120903
|
|
084030201004
|
|
0a7d4b321905
|
|
0cd86c482406
|
|
0e5793623107
|
|
1000c0804008
|
|
12d9f3a25109
|
|
14e82cc8640a
|
|
16336bf2790b
|
|
18c0b020900c
|
|
1a95fb52a90d
|
|
1cb84c88c40e
|
|
1e2fa3c2e10f
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_memory_of_bundle_of_arrays() {
|
|
#[hdl]
|
|
let raddr: UInt<4> = m.input();
|
|
#[hdl]
|
|
let rdata: (Array<[Array<[UInt<8>; 2]>; 3]>, UInt<2>) = m.output();
|
|
#[hdl]
|
|
let waddr: UInt<4> = m.input();
|
|
#[hdl]
|
|
let wdata: (Array<[Array<[UInt<8>; 2]>; 3]>, UInt<2>) = m.input();
|
|
#[hdl]
|
|
let wmask: (Array<[Array<[UInt<1>; 2]>; 3]>, UInt<1>) = m.input();
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
let mem_init = Vec::from_iter((0..0x10).map(|i| {
|
|
#[hdl]
|
|
(
|
|
#[hdl]
|
|
[
|
|
#[hdl]
|
|
[UInt::new(i), UInt::new(i * i)],
|
|
#[hdl]
|
|
[UInt::new(i * i * 2), UInt::new(i * i * 3)],
|
|
#[hdl]
|
|
[UInt::new(i * i * i), UInt::new(i * 2)],
|
|
],
|
|
UInt::new(i),
|
|
)
|
|
}));
|
|
#[hdl]
|
|
let mut mem = m.memory_with_init(mem_init);
|
|
let read_port = mem.new_read_port();
|
|
m.connect_any(read_port.addr, raddr);
|
|
m.connect(read_port.en, 1_hdl_u1);
|
|
m.connect(read_port.clk, clk);
|
|
m.connect(rdata, read_port.data);
|
|
let write_port = mem.new_write_port();
|
|
m.connect_any(write_port.addr, waddr);
|
|
m.connect(write_port.en, 1_hdl_u1);
|
|
m.connect(write_port.clk, clk);
|
|
m.connect(write_port.data, wdata);
|
|
m.connect(write_port.mask, wmask);
|
|
}
|
|
|
|
#[test]
|
|
fn test_memory_of_bundle_of_arrays() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_memory_of_bundle_of_arrays();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_of_bundle_of_arrays.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_memory_of_bundle_of_arrays: %[[
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_bundle_of_arrays/mem_0.mem",
|
|
"hexOrBinary": "h",
|
|
"target": "~check_memory_of_bundle_of_arrays|check_memory_of_bundle_of_arrays>mem_0"
|
|
},
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_bundle_of_arrays/mem_1.mem",
|
|
"hexOrBinary": "b",
|
|
"target": "~check_memory_of_bundle_of_arrays|check_memory_of_bundle_of_arrays>mem_1"
|
|
}
|
|
]]
|
|
type Ty0 = {`0`: UInt<8>[2][3], `1`: UInt<2>}
|
|
type Ty1 = {`0`: UInt<1>[2][3], `1`: UInt<1>}
|
|
type Ty2 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: Ty0}
|
|
type Ty3 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: Ty0, mask: Ty1}
|
|
type Ty4 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<8>[6]}
|
|
type Ty5 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<8>[6], mask: UInt<1>[6]}
|
|
type Ty6 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<2>}
|
|
type Ty7 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<2>, mask: UInt<1>}
|
|
module check_memory_of_bundle_of_arrays: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<4> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: Ty0 @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<4> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: Ty0 @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: Ty1 @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem mem_0: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<8>[6]
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
mem mem_1: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<2>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
wire mem_r0: Ty2 @[module-XXXXXXXXXX.rs 9:1]
|
|
wire mem_w1: Ty3 @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data.`0`[0][0], mem_0.r0.data[0] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_r0.data.`0`[0][1], mem_0.r0.data[1] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_r0.data.`0`[1][0], mem_0.r0.data[2] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_r0.data.`0`[1][1], mem_0.r0.data[3] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_r0.data.`0`[2][0], mem_0.r0.data[4] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_r0.data.`0`[2][1], mem_0.r0.data[5] @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0.w1.data[0], mem_w1.data.`0`[0][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.mask[0], mem_w1.mask.`0`[0][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.data[1], mem_w1.data.`0`[0][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.mask[1], mem_w1.mask.`0`[0][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.data[2], mem_w1.data.`0`[1][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.mask[2], mem_w1.mask.`0`[1][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.data[3], mem_w1.data.`0`[1][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.mask[3], mem_w1.mask.`0`[1][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.data[4], mem_w1.data.`0`[2][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.mask[4], mem_w1.mask.`0`[2][0] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.data[5], mem_w1.data.`0`[2][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.mask[5], mem_w1.mask.`0`[2][1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data.`1`, mem_1.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.w1.data, mem_w1.data.`1` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask, mem_w1.mask.`1` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect mem_r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect mem_r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, mem_r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect mem_w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect mem_w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect mem_w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect mem_w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect mem_w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
"#,
|
|
"/test/check_memory_of_bundle_of_arrays/mem_0.mem": r"000000000000
|
|
020103020101
|
|
04080c080402
|
|
061b1b120903
|
|
084030201004
|
|
0a7d4b321905
|
|
0cd86c482406
|
|
0e5793623107
|
|
1000c0804008
|
|
12d9f3a25109
|
|
14e82cc8640a
|
|
16336bf2790b
|
|
18c0b020900c
|
|
1a95fb52a90d
|
|
1cb84c88c40e
|
|
1e2fa3c2e10f
|
|
",
|
|
"/test/check_memory_of_bundle_of_arrays/mem_1.mem": r"00
|
|
01
|
|
10
|
|
11
|
|
00
|
|
01
|
|
10
|
|
11
|
|
00
|
|
01
|
|
10
|
|
11
|
|
00
|
|
01
|
|
10
|
|
11
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_memory_of_array_of_bundle() {
|
|
#[hdl]
|
|
let raddr: UInt<4> = m.input();
|
|
#[hdl]
|
|
let rdata: Array<[(UInt<8>, SInt<1>); 3]> = m.output();
|
|
#[hdl]
|
|
let waddr: UInt<4> = m.input();
|
|
#[hdl]
|
|
let wdata: Array<[(UInt<8>, SInt<1>); 3]> = m.input();
|
|
#[hdl]
|
|
let wmask: Array<[(UInt<1>, UInt<1>); 3]> = m.input();
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
let mem_init = Vec::from_iter((0..0x10).map(|i| {
|
|
#[hdl]
|
|
[
|
|
#[hdl]
|
|
(UInt::new(i), SInt::new(i)),
|
|
#[hdl]
|
|
(UInt::new(i * i), SInt::new(i / 2)),
|
|
#[hdl]
|
|
(UInt::new(i * i * i), SInt::new(i / 4)),
|
|
]
|
|
}));
|
|
#[hdl]
|
|
let mut mem = m.memory_with_init(mem_init);
|
|
let read_port = mem.new_read_port();
|
|
m.connect_any(read_port.addr, raddr);
|
|
m.connect(read_port.en, 1_hdl_u1);
|
|
m.connect(read_port.clk, clk);
|
|
m.connect(rdata, read_port.data);
|
|
let write_port = mem.new_write_port();
|
|
m.connect_any(write_port.addr, waddr);
|
|
m.connect(write_port.en, 1_hdl_u1);
|
|
m.connect(write_port.clk, clk);
|
|
m.connect(write_port.data, wdata);
|
|
m.connect(write_port.mask, wmask);
|
|
}
|
|
|
|
#[test]
|
|
fn test_memory_of_array_of_bundle() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_memory_of_array_of_bundle();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_of_array_of_bundle.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_memory_of_array_of_bundle: %[[
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_array_of_bundle/mem_0_0.mem",
|
|
"hexOrBinary": "h",
|
|
"target": "~check_memory_of_array_of_bundle|check_memory_of_array_of_bundle>mem_0_0"
|
|
},
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_array_of_bundle/mem_0_1.mem",
|
|
"hexOrBinary": "b",
|
|
"target": "~check_memory_of_array_of_bundle|check_memory_of_array_of_bundle>mem_0_1"
|
|
},
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_array_of_bundle/mem_1_0.mem",
|
|
"hexOrBinary": "h",
|
|
"target": "~check_memory_of_array_of_bundle|check_memory_of_array_of_bundle>mem_1_0"
|
|
},
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_array_of_bundle/mem_1_1.mem",
|
|
"hexOrBinary": "b",
|
|
"target": "~check_memory_of_array_of_bundle|check_memory_of_array_of_bundle>mem_1_1"
|
|
},
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_array_of_bundle/mem_2_0.mem",
|
|
"hexOrBinary": "h",
|
|
"target": "~check_memory_of_array_of_bundle|check_memory_of_array_of_bundle>mem_2_0"
|
|
},
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_array_of_bundle/mem_2_1.mem",
|
|
"hexOrBinary": "b",
|
|
"target": "~check_memory_of_array_of_bundle|check_memory_of_array_of_bundle>mem_2_1"
|
|
}
|
|
]]
|
|
type Ty0 = {`0`: UInt<8>, `1`: SInt<1>}
|
|
type Ty1 = {`0`: UInt<1>, `1`: UInt<1>}
|
|
type Ty2 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: Ty0[3]}
|
|
type Ty3 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: Ty0[3], mask: Ty1[3]}
|
|
type Ty4 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<8>}
|
|
type Ty5 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<8>, mask: UInt<1>}
|
|
type Ty6 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: SInt<1>}
|
|
type Ty7 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: SInt<1>, mask: UInt<1>}
|
|
module check_memory_of_array_of_bundle: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<4> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: Ty0[3] @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<4> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: Ty0[3] @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: Ty1[3] @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem mem_0_0: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<8>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
mem mem_0_1: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => SInt<1>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
mem mem_1_0: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<8>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
mem mem_1_1: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => SInt<1>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
mem mem_2_0: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<8>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
mem mem_2_1: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => SInt<1>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
wire mem_r0: Ty2 @[module-XXXXXXXXXX.rs 9:1]
|
|
wire mem_w1: Ty3 @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data[0].`0`, mem_0_0.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0_0.w1.data, mem_w1.data[0].`0` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0_0.w1.mask, mem_w1.mask[0].`0` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0_0.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0_0.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0_0.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0_0.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0_0.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0_0.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data[0].`1`, mem_0_1.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0_1.w1.data, mem_w1.data[0].`1` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0_1.w1.mask, mem_w1.mask[0].`1` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0_1.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0_1.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0_1.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0_1.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0_1.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0_1.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data[1].`0`, mem_1_0.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1_0.w1.data, mem_w1.data[1].`0` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1_0.w1.mask, mem_w1.mask[1].`0` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1_0.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1_0.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1_0.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1_0.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1_0.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1_0.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data[1].`1`, mem_1_1.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1_1.w1.data, mem_w1.data[1].`1` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1_1.w1.mask, mem_w1.mask[1].`1` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1_1.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1_1.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1_1.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1_1.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1_1.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1_1.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data[2].`0`, mem_2_0.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_2_0.w1.data, mem_w1.data[2].`0` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_2_0.w1.mask, mem_w1.mask[2].`0` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_2_0.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_2_0.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_2_0.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_2_0.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_2_0.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_2_0.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data[2].`1`, mem_2_1.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_2_1.w1.data, mem_w1.data[2].`1` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_2_1.w1.mask, mem_w1.mask[2].`1` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_2_1.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_2_1.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_2_1.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_2_1.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_2_1.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_2_1.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect mem_r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect mem_r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, mem_r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect mem_w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect mem_w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect mem_w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect mem_w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect mem_w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
"#,
|
|
"/test/check_memory_of_array_of_bundle/mem_0_0.mem": r"00
|
|
01
|
|
02
|
|
03
|
|
04
|
|
05
|
|
06
|
|
07
|
|
08
|
|
09
|
|
0a
|
|
0b
|
|
0c
|
|
0d
|
|
0e
|
|
0f
|
|
",
|
|
"/test/check_memory_of_array_of_bundle/mem_0_1.mem": r"0
|
|
1
|
|
0
|
|
1
|
|
0
|
|
1
|
|
0
|
|
1
|
|
0
|
|
1
|
|
0
|
|
1
|
|
0
|
|
1
|
|
0
|
|
1
|
|
",
|
|
"/test/check_memory_of_array_of_bundle/mem_1_0.mem": r"00
|
|
01
|
|
04
|
|
09
|
|
10
|
|
19
|
|
24
|
|
31
|
|
40
|
|
51
|
|
64
|
|
79
|
|
90
|
|
a9
|
|
c4
|
|
e1
|
|
",
|
|
"/test/check_memory_of_array_of_bundle/mem_1_1.mem": r"0
|
|
0
|
|
1
|
|
1
|
|
0
|
|
0
|
|
1
|
|
1
|
|
0
|
|
0
|
|
1
|
|
1
|
|
0
|
|
0
|
|
1
|
|
1
|
|
",
|
|
"/test/check_memory_of_array_of_bundle/mem_2_0.mem": r"00
|
|
01
|
|
08
|
|
1b
|
|
40
|
|
7d
|
|
d8
|
|
57
|
|
00
|
|
d9
|
|
e8
|
|
33
|
|
c0
|
|
95
|
|
b8
|
|
2f
|
|
",
|
|
"/test/check_memory_of_array_of_bundle/mem_2_1.mem": r"0
|
|
0
|
|
0
|
|
0
|
|
1
|
|
1
|
|
1
|
|
1
|
|
0
|
|
0
|
|
0
|
|
0
|
|
1
|
|
1
|
|
1
|
|
1
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_memory_of_bundle() {
|
|
#[hdl]
|
|
let raddr: UInt<4> = m.input();
|
|
#[hdl]
|
|
let rdata: (UInt<8>, SInt<1>) = m.output();
|
|
#[hdl]
|
|
let waddr: UInt<4> = m.input();
|
|
#[hdl]
|
|
let wdata: (UInt<8>, SInt<1>) = m.input();
|
|
#[hdl]
|
|
let wmask: (UInt<1>, UInt<1>) = m.input();
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
let mem_init = Vec::from_iter((0..0x10).map(|i| {
|
|
#[hdl]
|
|
(UInt::new(i ^ 3), SInt::new(i ^ i / 2))
|
|
}));
|
|
#[hdl]
|
|
let mut mem = m.memory_with_init(mem_init);
|
|
let read_port = mem.new_read_port();
|
|
m.connect_any(read_port.addr, raddr);
|
|
m.connect(read_port.en, 1_hdl_u1);
|
|
m.connect(read_port.clk, clk);
|
|
m.connect(rdata, read_port.data);
|
|
let write_port = mem.new_write_port();
|
|
m.connect_any(write_port.addr, waddr);
|
|
m.connect(write_port.en, 1_hdl_u1);
|
|
m.connect(write_port.clk, clk);
|
|
m.connect(write_port.data, wdata);
|
|
m.connect(write_port.mask, wmask);
|
|
}
|
|
|
|
#[test]
|
|
fn test_memory_of_bundle() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_memory_of_bundle();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_of_bundle.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_memory_of_bundle: %[[
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_bundle/mem_0.mem",
|
|
"hexOrBinary": "h",
|
|
"target": "~check_memory_of_bundle|check_memory_of_bundle>mem_0"
|
|
},
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_bundle/mem_1.mem",
|
|
"hexOrBinary": "b",
|
|
"target": "~check_memory_of_bundle|check_memory_of_bundle>mem_1"
|
|
}
|
|
]]
|
|
type Ty0 = {`0`: UInt<8>, `1`: SInt<1>}
|
|
type Ty1 = {`0`: UInt<1>, `1`: UInt<1>}
|
|
type Ty2 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: Ty0}
|
|
type Ty3 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: Ty0, mask: Ty1}
|
|
type Ty4 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<8>}
|
|
type Ty5 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<8>, mask: UInt<1>}
|
|
type Ty6 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: SInt<1>}
|
|
type Ty7 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: SInt<1>, mask: UInt<1>}
|
|
module check_memory_of_bundle: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<4> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: Ty0 @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<4> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: Ty0 @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: Ty1 @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem mem_0: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<8>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
mem mem_1: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => SInt<1>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
wire mem_r0: Ty2 @[module-XXXXXXXXXX.rs 9:1]
|
|
wire mem_w1: Ty3 @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data.`0`, mem_0.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0.w1.data, mem_w1.data.`0` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.mask, mem_w1.mask.`0` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_0.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_0.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data.`1`, mem_1.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.w1.data, mem_w1.data.`1` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask, mem_w1.mask.`1` @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect mem_r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect mem_r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, mem_r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect mem_w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect mem_w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect mem_w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect mem_w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect mem_w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
"#,
|
|
"/test/check_memory_of_bundle/mem_0.mem": r"03
|
|
02
|
|
01
|
|
00
|
|
07
|
|
06
|
|
05
|
|
04
|
|
0b
|
|
0a
|
|
09
|
|
08
|
|
0f
|
|
0e
|
|
0d
|
|
0c
|
|
",
|
|
"/test/check_memory_of_bundle/mem_1.mem": r"0
|
|
1
|
|
1
|
|
0
|
|
0
|
|
1
|
|
1
|
|
0
|
|
0
|
|
1
|
|
1
|
|
0
|
|
0
|
|
1
|
|
1
|
|
0
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_memory_of_enum() {
|
|
#[hdl]
|
|
let raddr: UInt<8> = m.input();
|
|
#[hdl]
|
|
let rdata: TestEnum = m.output();
|
|
#[hdl]
|
|
let waddr: UInt<8> = m.input();
|
|
#[hdl]
|
|
let wdata: TestEnum = m.input();
|
|
#[hdl]
|
|
let wmask: UInt<1> = m.input();
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
let mem_init = Vec::from_iter((0..0x10).map(|i| match i {
|
|
0 =>
|
|
{
|
|
#[hdl]
|
|
TestEnum::A
|
|
}
|
|
1..=3 =>
|
|
{
|
|
#[hdl]
|
|
TestEnum::C(
|
|
#[hdl]
|
|
[UInt::new(i == 1), UInt::new(i == 2), UInt::new(i == 3)],
|
|
)
|
|
}
|
|
_ =>
|
|
{
|
|
#[hdl]
|
|
TestEnum::B(UInt::new(i))
|
|
}
|
|
}));
|
|
#[hdl]
|
|
let mut mem = m.memory_with_init(mem_init);
|
|
let read_port = mem.new_read_port();
|
|
m.connect_any(read_port.addr, raddr);
|
|
m.connect(read_port.en, 1_hdl_u1);
|
|
m.connect(read_port.clk, clk);
|
|
m.connect(rdata, read_port.data);
|
|
let write_port = mem.new_write_port();
|
|
m.connect_any(write_port.addr, waddr);
|
|
m.connect(write_port.en, 1_hdl_u1);
|
|
m.connect(write_port.clk, clk);
|
|
m.connect(write_port.data, wdata);
|
|
m.connect(write_port.mask, wmask);
|
|
}
|
|
|
|
#[test]
|
|
fn test_memory_of_enum() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_memory_of_enum();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_of_enum.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_memory_of_enum: %[[
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_enum/mem_1.mem",
|
|
"hexOrBinary": "b",
|
|
"target": "~check_memory_of_enum|check_memory_of_enum>mem_1"
|
|
}
|
|
]]
|
|
type Ty0 = {|A, B: UInt<8>, C: UInt<1>[3]|}
|
|
type Ty1 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: Ty0}
|
|
type Ty2 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: Ty0, mask: UInt<1>}
|
|
type Ty3 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<10>}
|
|
type Ty4 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<10>, mask: UInt<1>}
|
|
module check_memory_of_enum: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: Ty0 @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<8> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: Ty0 @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: UInt<1> @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem mem_1: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<10>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
wire mem_r0: Ty1 @[module-XXXXXXXXXX.rs 9:1]
|
|
wire mem_w1: Ty2 @[module-XXXXXXXXXX.rs 14:1]
|
|
wire _cast_bits_to_enum_expr: Ty0
|
|
wire _cast_bits_to_enum_expr_body: UInt<8>
|
|
connect _cast_bits_to_enum_expr_body, head(mem_1.r0.data, 8)
|
|
when eq(UInt<2>(0), tail(mem_1.r0.data, 8)):
|
|
connect _cast_bits_to_enum_expr, {|A, B: UInt<8>, C: UInt<1>[3]|}(A)
|
|
else when eq(UInt<2>(1), tail(mem_1.r0.data, 8)):
|
|
connect _cast_bits_to_enum_expr, {|A, B: UInt<8>, C: UInt<1>[3]|}(B, _cast_bits_to_enum_expr_body)
|
|
else:
|
|
wire _cast_bits_to_array_expr: UInt<1>[3]
|
|
wire _cast_bits_to_array_expr_flattened: UInt<1>[3]
|
|
connect _cast_bits_to_array_expr_flattened[0], bits(_cast_bits_to_enum_expr_body, 0, 0)
|
|
connect _cast_bits_to_array_expr[0], _cast_bits_to_array_expr_flattened[0]
|
|
connect _cast_bits_to_array_expr_flattened[1], bits(_cast_bits_to_enum_expr_body, 1, 1)
|
|
connect _cast_bits_to_array_expr[1], _cast_bits_to_array_expr_flattened[1]
|
|
connect _cast_bits_to_array_expr_flattened[2], bits(_cast_bits_to_enum_expr_body, 2, 2)
|
|
connect _cast_bits_to_array_expr[2], _cast_bits_to_array_expr_flattened[2]
|
|
connect _cast_bits_to_enum_expr, {|A, B: UInt<8>, C: UInt<1>[3]|}(C, _cast_bits_to_array_expr)
|
|
connect mem_r0.data, _cast_bits_to_enum_expr @[module-XXXXXXXXXX.rs 9:1]
|
|
wire _cast_enum_to_bits_expr: UInt<10>
|
|
match mem_w1.data:
|
|
A:
|
|
connect _cast_enum_to_bits_expr, UInt<10>(0)
|
|
B(_cast_enum_to_bits_expr_B):
|
|
connect _cast_enum_to_bits_expr, pad(cat(_cast_enum_to_bits_expr_B, UInt<2>(1)), 10)
|
|
C(_cast_enum_to_bits_expr_C):
|
|
wire _cast_array_to_bits_expr: UInt<1>[3]
|
|
connect _cast_array_to_bits_expr[0], _cast_enum_to_bits_expr_C[0]
|
|
connect _cast_array_to_bits_expr[1], _cast_enum_to_bits_expr_C[1]
|
|
connect _cast_array_to_bits_expr[2], _cast_enum_to_bits_expr_C[2]
|
|
wire _cast_to_bits_expr: UInt<3>
|
|
connect _cast_to_bits_expr, cat(_cast_array_to_bits_expr[2], cat(_cast_array_to_bits_expr[1], _cast_array_to_bits_expr[0]))
|
|
connect _cast_enum_to_bits_expr, pad(cat(_cast_to_bits_expr, UInt<2>(2)), 10)
|
|
connect mem_1.w1.data, _cast_enum_to_bits_expr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask, mem_w1.mask @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect mem_r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect mem_r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, mem_r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect mem_w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect mem_w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect mem_w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect mem_w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect mem_w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
"#,
|
|
"/test/check_memory_of_enum/mem_1.mem": r"0000000000
|
|
0000000110
|
|
0000001010
|
|
0000010010
|
|
0000010001
|
|
0000010101
|
|
0000011001
|
|
0000011101
|
|
0000100001
|
|
0000100101
|
|
0000101001
|
|
0000101101
|
|
0000110001
|
|
0000110101
|
|
0000111001
|
|
0000111101
|
|
",
|
|
};
|
|
let orig_m = m.canonical().intern();
|
|
let m = simplify_enums(orig_m, SimplifyEnumsKind::SimplifyToEnumsWithNoBody).unwrap();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_of_enum.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_memory_of_enum: %[[
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_enum/mem_tag.mem",
|
|
"hexOrBinary": "b",
|
|
"target": "~check_memory_of_enum|check_memory_of_enum>mem_tag"
|
|
},
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_enum/mem_body.mem",
|
|
"hexOrBinary": "h",
|
|
"target": "~check_memory_of_enum|check_memory_of_enum>mem_body"
|
|
}
|
|
]]
|
|
type Ty0 = {|A, B, C|}
|
|
type Ty1 = {tag: Ty0, body: UInt<8>}
|
|
type Ty2 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: Ty1}
|
|
type Ty3 = {tag: UInt<1>, body: UInt<1>}
|
|
type Ty4 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: Ty1, mask: Ty3}
|
|
type Ty5 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<2>}
|
|
type Ty6 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<2>, mask: UInt<1>}
|
|
type Ty7 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<8>}
|
|
type Ty8 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<8>, mask: UInt<1>}
|
|
type Ty9 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: Ty1, mask: UInt<1>}
|
|
module check_memory_of_enum: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: Ty1 @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<8> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: Ty1 @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: UInt<1> @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem mem_tag: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<2>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
mem mem_body: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<8>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
wire mem_r0: Ty2 @[module-XXXXXXXXXX.rs 9:1]
|
|
wire mem_w1_1: Ty4 @[module-XXXXXXXXXX.rs 14:1]
|
|
wire _cast_bits_to_enum_expr: Ty0
|
|
when eq(UInt<2>(0), tail(mem_tag.r0.data, 0)):
|
|
connect _cast_bits_to_enum_expr, {|A, B, C|}(A)
|
|
else when eq(UInt<2>(1), tail(mem_tag.r0.data, 0)):
|
|
connect _cast_bits_to_enum_expr, {|A, B, C|}(B)
|
|
else:
|
|
connect _cast_bits_to_enum_expr, {|A, B, C|}(C)
|
|
connect mem_r0.data.tag, _cast_bits_to_enum_expr @[module-XXXXXXXXXX.rs 9:1]
|
|
wire _cast_enum_to_bits_expr: UInt<2>
|
|
match mem_w1_1.data.tag:
|
|
A:
|
|
connect _cast_enum_to_bits_expr, UInt<2>(0)
|
|
B:
|
|
connect _cast_enum_to_bits_expr, UInt<2>(1)
|
|
C:
|
|
connect _cast_enum_to_bits_expr, UInt<2>(2)
|
|
connect mem_tag.w1.data, _cast_enum_to_bits_expr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_tag.w1.mask, mem_w1_1.mask.tag @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_tag.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_tag.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_tag.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_tag.w1.addr, mem_w1_1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_tag.w1.clk, mem_w1_1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_tag.w1.en, mem_w1_1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data.body, mem_body.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_body.w1.data, mem_w1_1.data.body @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_body.w1.mask, mem_w1_1.mask.body @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_body.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_body.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_body.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_body.w1.addr, mem_w1_1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_body.w1.clk, mem_w1_1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_body.w1.en, mem_w1_1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
wire mem_w1: Ty9 @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.data, mem_w1.data @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.mask.tag, mem_w1.mask @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.mask.body, mem_w1.mask @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect mem_r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect mem_r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, mem_r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect mem_w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect mem_w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect mem_w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect mem_w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect mem_w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
"#,
|
|
"/test/check_memory_of_enum/mem_body.mem": r"00
|
|
01
|
|
02
|
|
04
|
|
04
|
|
05
|
|
06
|
|
07
|
|
08
|
|
09
|
|
0a
|
|
0b
|
|
0c
|
|
0d
|
|
0e
|
|
0f
|
|
",
|
|
"/test/check_memory_of_enum/mem_tag.mem": r"00
|
|
10
|
|
10
|
|
10
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
",
|
|
};
|
|
let m = simplify_enums(orig_m, SimplifyEnumsKind::ReplaceWithBundleOfUInts).unwrap();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_of_enum.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_memory_of_enum: %[[
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_enum/mem_tag.mem",
|
|
"hexOrBinary": "b",
|
|
"target": "~check_memory_of_enum|check_memory_of_enum>mem_tag"
|
|
},
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_enum/mem_body.mem",
|
|
"hexOrBinary": "h",
|
|
"target": "~check_memory_of_enum|check_memory_of_enum>mem_body"
|
|
}
|
|
]]
|
|
type Ty0 = {tag: UInt<2>, body: UInt<8>}
|
|
type Ty1 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: Ty0}
|
|
type Ty2 = {tag: UInt<1>, body: UInt<1>}
|
|
type Ty3 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: Ty0, mask: Ty2}
|
|
type Ty4 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<2>}
|
|
type Ty5 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<2>, mask: UInt<1>}
|
|
type Ty6 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<8>}
|
|
type Ty7 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<8>, mask: UInt<1>}
|
|
type Ty8 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: Ty0, mask: UInt<1>}
|
|
module check_memory_of_enum: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: Ty0 @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<8> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: Ty0 @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: UInt<1> @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem mem_tag: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<2>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
mem mem_body: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<8>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
wire mem_r0: Ty1 @[module-XXXXXXXXXX.rs 9:1]
|
|
wire mem_w1_1: Ty3 @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data.tag, mem_tag.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_tag.w1.data, mem_w1_1.data.tag @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_tag.w1.mask, mem_w1_1.mask.tag @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_tag.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_tag.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_tag.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_tag.w1.addr, mem_w1_1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_tag.w1.clk, mem_w1_1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_tag.w1.en, mem_w1_1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.data.body, mem_body.r0.data @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_body.w1.data, mem_w1_1.data.body @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_body.w1.mask, mem_w1_1.mask.body @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_body.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_body.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_body.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_body.w1.addr, mem_w1_1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_body.w1.clk, mem_w1_1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_body.w1.en, mem_w1_1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
wire mem_w1: Ty8 @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.data, mem_w1.data @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.mask.tag, mem_w1.mask @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_w1_1.mask.body, mem_w1.mask @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect mem_r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect mem_r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, mem_r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect mem_w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect mem_w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect mem_w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect mem_w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect mem_w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
"#,
|
|
"/test/check_memory_of_enum/mem_body.mem": r"00
|
|
01
|
|
02
|
|
04
|
|
04
|
|
05
|
|
06
|
|
07
|
|
08
|
|
09
|
|
0a
|
|
0b
|
|
0c
|
|
0d
|
|
0e
|
|
0f
|
|
",
|
|
"/test/check_memory_of_enum/mem_tag.mem": r"00
|
|
10
|
|
10
|
|
10
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
01
|
|
",
|
|
};
|
|
let m = simplify_enums(orig_m, SimplifyEnumsKind::ReplaceWithUInt).unwrap();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_of_enum.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_memory_of_enum: %[[
|
|
{
|
|
"class": "firrtl.annotations.MemoryFileInlineAnnotation",
|
|
"filename": "/test/check_memory_of_enum/mem.mem",
|
|
"hexOrBinary": "b",
|
|
"target": "~check_memory_of_enum|check_memory_of_enum>mem"
|
|
}
|
|
]]
|
|
type Ty0 = {addr: UInt<4>, en: UInt<1>, clk: Clock, flip data: UInt<10>}
|
|
type Ty1 = {addr: UInt<4>, en: UInt<1>, clk: Clock, data: UInt<10>, mask: UInt<1>}
|
|
module check_memory_of_enum: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: UInt<10> @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<8> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: UInt<10> @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: UInt<1> @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem `mem`: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<10>
|
|
depth => 16
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
connect `mem`.r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect `mem`.r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect `mem`.r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, `mem`.r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect `mem`.w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect `mem`.w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect `mem`.w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect `mem`.w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect `mem`.w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
"#,
|
|
"/test/check_memory_of_enum/mem.mem": r"0000000000
|
|
0000000110
|
|
0000001010
|
|
0000010010
|
|
0000010001
|
|
0000010101
|
|
0000011001
|
|
0000011101
|
|
0000100001
|
|
0000100101
|
|
0000101001
|
|
0000101101
|
|
0000110001
|
|
0000110101
|
|
0000111001
|
|
0000111101
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_memory_of_array_of_enum() {
|
|
#[hdl]
|
|
let raddr: UInt<8> = m.input();
|
|
#[hdl]
|
|
let rdata: Array<[TestEnum; 2]> = m.output();
|
|
#[hdl]
|
|
let waddr: UInt<8> = m.input();
|
|
#[hdl]
|
|
let wdata: Array<[TestEnum; 2]> = m.input();
|
|
#[hdl]
|
|
let wmask: Array<[UInt<1>; 2]> = m.input();
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
#[hdl]
|
|
let mut mem: MemBuilder<[Array<[TestEnum; 2]>; 0x100]> = m.memory_array();
|
|
let read_port = mem.new_read_port();
|
|
m.connect_any(read_port.addr, raddr);
|
|
m.connect(read_port.en, 1_hdl_u1);
|
|
m.connect(read_port.clk, clk);
|
|
m.connect(rdata, read_port.data);
|
|
let write_port = mem.new_write_port();
|
|
m.connect_any(write_port.addr, waddr);
|
|
m.connect(write_port.en, 1_hdl_u1);
|
|
m.connect(write_port.clk, clk);
|
|
m.connect(write_port.data, wdata);
|
|
m.connect(write_port.mask, wmask);
|
|
}
|
|
|
|
#[test]
|
|
fn test_memory_of_array_of_enum() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_memory_of_array_of_enum();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_memory_of_array_of_enum.fir": r"FIRRTL version 3.2.0
|
|
circuit check_memory_of_array_of_enum:
|
|
type Ty0 = {|A, B: UInt<8>, C: UInt<1>[3]|}
|
|
type Ty1 = {addr: UInt<8>, en: UInt<1>, clk: Clock, flip data: Ty0[2]}
|
|
type Ty2 = {addr: UInt<8>, en: UInt<1>, clk: Clock, data: Ty0[2], mask: UInt<1>[2]}
|
|
type Ty3 = {addr: UInt<8>, en: UInt<1>, clk: Clock, flip data: UInt<10>[2]}
|
|
type Ty4 = {addr: UInt<8>, en: UInt<1>, clk: Clock, data: UInt<10>[2], mask: UInt<1>[2]}
|
|
module check_memory_of_array_of_enum: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: Ty0[2] @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<8> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: Ty0[2] @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: UInt<1>[2] @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem mem_1: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<10>[2]
|
|
depth => 256
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
wire mem_r0: Ty1 @[module-XXXXXXXXXX.rs 9:1]
|
|
wire mem_w1: Ty2 @[module-XXXXXXXXXX.rs 14:1]
|
|
wire _cast_bits_to_enum_expr: Ty0
|
|
wire _cast_bits_to_enum_expr_body: UInt<8>
|
|
connect _cast_bits_to_enum_expr_body, head(mem_1.r0.data[0], 8)
|
|
when eq(UInt<2>(0), tail(mem_1.r0.data[0], 8)):
|
|
connect _cast_bits_to_enum_expr, {|A, B: UInt<8>, C: UInt<1>[3]|}(A)
|
|
else when eq(UInt<2>(1), tail(mem_1.r0.data[0], 8)):
|
|
connect _cast_bits_to_enum_expr, {|A, B: UInt<8>, C: UInt<1>[3]|}(B, _cast_bits_to_enum_expr_body)
|
|
else:
|
|
wire _cast_bits_to_array_expr: UInt<1>[3]
|
|
wire _cast_bits_to_array_expr_flattened: UInt<1>[3]
|
|
connect _cast_bits_to_array_expr_flattened[0], bits(_cast_bits_to_enum_expr_body, 0, 0)
|
|
connect _cast_bits_to_array_expr[0], _cast_bits_to_array_expr_flattened[0]
|
|
connect _cast_bits_to_array_expr_flattened[1], bits(_cast_bits_to_enum_expr_body, 1, 1)
|
|
connect _cast_bits_to_array_expr[1], _cast_bits_to_array_expr_flattened[1]
|
|
connect _cast_bits_to_array_expr_flattened[2], bits(_cast_bits_to_enum_expr_body, 2, 2)
|
|
connect _cast_bits_to_array_expr[2], _cast_bits_to_array_expr_flattened[2]
|
|
connect _cast_bits_to_enum_expr, {|A, B: UInt<8>, C: UInt<1>[3]|}(C, _cast_bits_to_array_expr)
|
|
connect mem_r0.data[0], _cast_bits_to_enum_expr @[module-XXXXXXXXXX.rs 9:1]
|
|
wire _cast_bits_to_enum_expr_1: Ty0
|
|
wire _cast_bits_to_enum_expr_body_1: UInt<8>
|
|
connect _cast_bits_to_enum_expr_body_1, head(mem_1.r0.data[1], 8)
|
|
when eq(UInt<2>(0), tail(mem_1.r0.data[1], 8)):
|
|
connect _cast_bits_to_enum_expr_1, {|A, B: UInt<8>, C: UInt<1>[3]|}(A)
|
|
else when eq(UInt<2>(1), tail(mem_1.r0.data[1], 8)):
|
|
connect _cast_bits_to_enum_expr_1, {|A, B: UInt<8>, C: UInt<1>[3]|}(B, _cast_bits_to_enum_expr_body_1)
|
|
else:
|
|
wire _cast_bits_to_array_expr_1: UInt<1>[3]
|
|
wire _cast_bits_to_array_expr_flattened_1: UInt<1>[3]
|
|
connect _cast_bits_to_array_expr_flattened_1[0], bits(_cast_bits_to_enum_expr_body_1, 0, 0)
|
|
connect _cast_bits_to_array_expr_1[0], _cast_bits_to_array_expr_flattened_1[0]
|
|
connect _cast_bits_to_array_expr_flattened_1[1], bits(_cast_bits_to_enum_expr_body_1, 1, 1)
|
|
connect _cast_bits_to_array_expr_1[1], _cast_bits_to_array_expr_flattened_1[1]
|
|
connect _cast_bits_to_array_expr_flattened_1[2], bits(_cast_bits_to_enum_expr_body_1, 2, 2)
|
|
connect _cast_bits_to_array_expr_1[2], _cast_bits_to_array_expr_flattened_1[2]
|
|
connect _cast_bits_to_enum_expr_1, {|A, B: UInt<8>, C: UInt<1>[3]|}(C, _cast_bits_to_array_expr_1)
|
|
connect mem_r0.data[1], _cast_bits_to_enum_expr_1 @[module-XXXXXXXXXX.rs 9:1]
|
|
wire _cast_enum_to_bits_expr: UInt<10>
|
|
match mem_w1.data[0]:
|
|
A:
|
|
connect _cast_enum_to_bits_expr, UInt<10>(0)
|
|
B(_cast_enum_to_bits_expr_B):
|
|
connect _cast_enum_to_bits_expr, pad(cat(_cast_enum_to_bits_expr_B, UInt<2>(1)), 10)
|
|
C(_cast_enum_to_bits_expr_C):
|
|
wire _cast_array_to_bits_expr: UInt<1>[3]
|
|
connect _cast_array_to_bits_expr[0], _cast_enum_to_bits_expr_C[0]
|
|
connect _cast_array_to_bits_expr[1], _cast_enum_to_bits_expr_C[1]
|
|
connect _cast_array_to_bits_expr[2], _cast_enum_to_bits_expr_C[2]
|
|
wire _cast_to_bits_expr: UInt<3>
|
|
connect _cast_to_bits_expr, cat(_cast_array_to_bits_expr[2], cat(_cast_array_to_bits_expr[1], _cast_array_to_bits_expr[0]))
|
|
connect _cast_enum_to_bits_expr, pad(cat(_cast_to_bits_expr, UInt<2>(2)), 10)
|
|
connect mem_1.w1.data[0], _cast_enum_to_bits_expr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask[0], mem_w1.mask[0] @[module-XXXXXXXXXX.rs 14:1]
|
|
wire _cast_enum_to_bits_expr_1: UInt<10>
|
|
match mem_w1.data[1]:
|
|
A:
|
|
connect _cast_enum_to_bits_expr_1, UInt<10>(0)
|
|
B(_cast_enum_to_bits_expr_B_1):
|
|
connect _cast_enum_to_bits_expr_1, pad(cat(_cast_enum_to_bits_expr_B_1, UInt<2>(1)), 10)
|
|
C(_cast_enum_to_bits_expr_C_1):
|
|
wire _cast_array_to_bits_expr_1: UInt<1>[3]
|
|
connect _cast_array_to_bits_expr_1[0], _cast_enum_to_bits_expr_C_1[0]
|
|
connect _cast_array_to_bits_expr_1[1], _cast_enum_to_bits_expr_C_1[1]
|
|
connect _cast_array_to_bits_expr_1[2], _cast_enum_to_bits_expr_C_1[2]
|
|
wire _cast_to_bits_expr_1: UInt<3>
|
|
connect _cast_to_bits_expr_1, cat(_cast_array_to_bits_expr_1[2], cat(_cast_array_to_bits_expr_1[1], _cast_array_to_bits_expr_1[0]))
|
|
connect _cast_enum_to_bits_expr_1, pad(cat(_cast_to_bits_expr_1, UInt<2>(2)), 10)
|
|
connect mem_1.w1.data[1], _cast_enum_to_bits_expr_1 @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.mask[1], mem_w1.mask[1] @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.r0.addr, mem_r0.addr @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.r0.clk, mem_r0.clk @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.r0.en, mem_r0.en @[module-XXXXXXXXXX.rs 9:1]
|
|
connect mem_1.w1.addr, mem_w1.addr @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.clk, mem_w1.clk @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_1.w1.en, mem_w1.en @[module-XXXXXXXXXX.rs 14:1]
|
|
connect mem_r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect mem_r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect mem_r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, mem_r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect mem_w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect mem_w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect mem_w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect mem_w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect mem_w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
",
|
|
};
|
|
}
|
|
|
|
#[hdl_module(outline_generated)]
|
|
pub fn check_annotations() {
|
|
m.annotate_module(Annotation::CustomFirrtl(CustomFirrtlAnnotation {
|
|
class: "the.annotation.Example".intern(),
|
|
additional_fields: json!({
|
|
"bar": "a nice module!",
|
|
})
|
|
.try_into()
|
|
.unwrap(),
|
|
}));
|
|
#[hdl]
|
|
let raddr: UInt<8> = m.input();
|
|
m.annotate(raddr, Annotation::DontTouch);
|
|
#[hdl]
|
|
let rdata: Array<[UInt<4>; 2]> = m.output();
|
|
m.annotate(
|
|
rdata,
|
|
Annotation::CustomFirrtl(CustomFirrtlAnnotation {
|
|
class: "the.annotation.ExampleClass".intern(),
|
|
additional_fields: json!({
|
|
"foo": "bar",
|
|
"baz": [0, 1, "a", "b"],
|
|
})
|
|
.try_into()
|
|
.unwrap(),
|
|
}),
|
|
);
|
|
#[hdl]
|
|
let waddr: UInt<8> = m.input();
|
|
#[hdl]
|
|
let wdata: Array<[UInt<4>; 2]> = m.input();
|
|
#[hdl]
|
|
let wmask: Array<[UInt<1>; 2]> = m.input();
|
|
m.annotate(
|
|
wmask[1],
|
|
Annotation::CustomFirrtl(CustomFirrtlAnnotation {
|
|
class: "some.annotation.Class".intern(),
|
|
additional_fields: json!({
|
|
"baz": "second mask bit",
|
|
})
|
|
.try_into()
|
|
.unwrap(),
|
|
}),
|
|
);
|
|
#[hdl]
|
|
let clk: Clock = m.input();
|
|
#[hdl]
|
|
let mut mem: MemBuilder<[Array<[UInt<4>; 2]>; 0x100]> = m.memory_array();
|
|
mem.annotate(Annotation::CustomFirrtl(CustomFirrtlAnnotation {
|
|
class: "the.annotation.ExampleClass2".intern(),
|
|
additional_fields: json!({
|
|
"bar": "foo",
|
|
"baz": [0, 1, "a", "b"],
|
|
})
|
|
.try_into()
|
|
.unwrap(),
|
|
}));
|
|
let read_port = mem.new_read_port();
|
|
m.annotate(
|
|
read_port,
|
|
Annotation::CustomFirrtl(CustomFirrtlAnnotation {
|
|
class: "the.annotation.ExampleClass3".intern(),
|
|
additional_fields: json!({
|
|
"foo": "my read port",
|
|
})
|
|
.try_into()
|
|
.unwrap(),
|
|
}),
|
|
);
|
|
m.connect_any(read_port.addr, raddr);
|
|
m.connect(read_port.en, 1_hdl_u1);
|
|
m.connect(read_port.clk, clk);
|
|
m.connect(rdata, read_port.data);
|
|
let write_port = mem.new_write_port();
|
|
m.annotate(
|
|
write_port.data[0],
|
|
Annotation::CustomFirrtl(CustomFirrtlAnnotation {
|
|
class: "some.annotation.Class".intern(),
|
|
additional_fields: json!({
|
|
"baz": "first mask bit",
|
|
})
|
|
.try_into()
|
|
.unwrap(),
|
|
}),
|
|
);
|
|
m.connect_any(write_port.addr, waddr);
|
|
m.connect(write_port.en, 1_hdl_u1);
|
|
m.connect(write_port.clk, clk);
|
|
m.connect(write_port.data, wdata);
|
|
m.connect(write_port.mask, wmask);
|
|
}
|
|
|
|
#[test]
|
|
fn test_annotations() {
|
|
let _n = SourceLocation::normalize_files_for_tests();
|
|
let m = check_annotations();
|
|
dbg!(m);
|
|
#[rustfmt::skip] // work around https://github.com/rust-lang/rustfmt/issues/6161
|
|
assert_export_firrtl! {
|
|
m =>
|
|
"/test/check_annotations.fir": r#"FIRRTL version 3.2.0
|
|
circuit check_annotations: %[[
|
|
{
|
|
"class": "the.annotation.Example",
|
|
"bar": "a nice module!",
|
|
"target": "~check_annotations|check_annotations"
|
|
},
|
|
{
|
|
"class": "firrtl.transforms.DontTouchAnnotation",
|
|
"target": "~check_annotations|check_annotations>raddr"
|
|
},
|
|
{
|
|
"class": "the.annotation.ExampleClass",
|
|
"foo": "bar",
|
|
"baz": [
|
|
0,
|
|
1,
|
|
"a",
|
|
"b"
|
|
],
|
|
"target": "~check_annotations|check_annotations>rdata"
|
|
},
|
|
{
|
|
"class": "some.annotation.Class",
|
|
"baz": "second mask bit",
|
|
"target": "~check_annotations|check_annotations>wmask[1]"
|
|
},
|
|
{
|
|
"class": "the.annotation.ExampleClass2",
|
|
"bar": "foo",
|
|
"baz": [
|
|
0,
|
|
1,
|
|
"a",
|
|
"b"
|
|
],
|
|
"target": "~check_annotations|check_annotations>mem"
|
|
},
|
|
{
|
|
"class": "the.annotation.ExampleClass3",
|
|
"foo": "my read port",
|
|
"target": "~check_annotations|check_annotations>mem.r0"
|
|
},
|
|
{
|
|
"class": "some.annotation.Class",
|
|
"baz": "first mask bit",
|
|
"target": "~check_annotations|check_annotations>mem.w1.data[0]"
|
|
}
|
|
]]
|
|
type Ty0 = {addr: UInt<8>, en: UInt<1>, clk: Clock, data: UInt<4>[2], mask: UInt<1>[2]}
|
|
type Ty1 = {addr: UInt<8>, en: UInt<1>, clk: Clock, flip data: UInt<4>[2]}
|
|
module check_annotations: @[module-XXXXXXXXXX.rs 1:1]
|
|
input raddr: UInt<8> @[module-XXXXXXXXXX.rs 2:1]
|
|
output rdata: UInt<4>[2] @[module-XXXXXXXXXX.rs 3:1]
|
|
input waddr: UInt<8> @[module-XXXXXXXXXX.rs 4:1]
|
|
input wdata: UInt<4>[2] @[module-XXXXXXXXXX.rs 5:1]
|
|
input wmask: UInt<1>[2] @[module-XXXXXXXXXX.rs 6:1]
|
|
input clk: Clock @[module-XXXXXXXXXX.rs 7:1]
|
|
mem `mem`: @[module-XXXXXXXXXX.rs 8:1]
|
|
data-type => UInt<4>[2]
|
|
depth => 256
|
|
read-latency => 0
|
|
write-latency => 1
|
|
read-under-write => old
|
|
reader => r0
|
|
writer => w1
|
|
connect `mem`.r0.addr, raddr @[module-XXXXXXXXXX.rs 10:1]
|
|
connect `mem`.r0.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 11:1]
|
|
connect `mem`.r0.clk, clk @[module-XXXXXXXXXX.rs 12:1]
|
|
connect rdata, `mem`.r0.data @[module-XXXXXXXXXX.rs 13:1]
|
|
connect `mem`.w1.addr, waddr @[module-XXXXXXXXXX.rs 15:1]
|
|
connect `mem`.w1.en, UInt<1>(0h1) @[module-XXXXXXXXXX.rs 16:1]
|
|
connect `mem`.w1.clk, clk @[module-XXXXXXXXXX.rs 17:1]
|
|
connect `mem`.w1.data, wdata @[module-XXXXXXXXXX.rs 18:1]
|
|
connect `mem`.w1.mask, wmask @[module-XXXXXXXXXX.rs 19:1]
|
|
"#,
|
|
};
|
|
}
|