1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
//! Mux.
use super::*;
/// Extension trait for `mux`.
pub trait MuxExt<const N: usize>: Interface
where [(); clog2(N)]:
{
/// Egress interface.
type E: Interface;
/// Mux.
fn mux(self, cntl: Valid<U<{ clog2(N) }>>) -> Self::E;
}
impl<P: Copy, R: Copy, const N: usize, const D: Dep> MuxExt<N> for [I<ValidH<P, R>, D>; N]
where [(); clog2(N)]:
{
type E = I<ValidH<P, R>, D>;
/// Muxes `N` `ValidH` hazard interfaces based on `cntl`.
///
/// `cntl` selects which ingress interface to connect to the egress interface.
///
/// - Payloads: Outputs the payload of the interface selected by `cntl`.
/// - Resolver: Duplicated to multiple interfaces.
///
/// | Interface | Ingress | Egress |
/// | :-------: | ---------------------- | ------------ |
/// | **Fwd** | `Array<HOption<P>, N>` | `HOption<P>` |
/// | **Bwd** | `Array<R, N>` | `R` |
fn mux(self, cntl: Valid<U<{ clog2(N) }>>) -> I<ValidH<P, R>, D> {
unsafe {
(self, cntl).fsm::<I<ValidH<P, R>, D>, ()>((), |(ip, sel), er, s| {
let ep = sel.and_then(|sel| ip[sel]);
let ir = er.repeat::<N>();
(ep, (ir, ()), s)
})
}
}
}
impl<P: Copy, R: Copy, const N: usize, const D: Dep> MuxExt<N> for [I<VrH<P, R>, D>; N]
where [(); clog2(N)]:
{
type E = I<VrH<P, R>, D>;
/// Muxes `N` `VrH` hazard interfaces based on `cntl`.
///
/// `cntl` selects which ingress interface to connect to the egress interface.
///
/// - Payloads: Outputs the payload of the interface selected by `cntl`.
/// - Resolver: The selected interface's resolver follows the egress resolver. All the other resolvers are invalid.
///
/// | Interface | Ingress | Egress |
/// | :-------: | ---------------------- | ------------ |
/// | **Fwd** | `Array<HOption<P>, N>` | `HOption<P>` |
/// | **Bwd** | `Array<Ready<R>, N>` | `Ready<R>` |
fn mux(self, cntl: Valid<U<{ clog2(N) }>>) -> I<VrH<P, R>, D> {
unsafe {
(self, cntl).fsm::<I<VrH<P, R>, D>, ()>((), |(ip, sel), er, s| {
let ep = sel.and_then(|sel| ip[sel]);
let ir = if let Some(sel) = sel {
Ready::invalid().repeat::<N>().set(sel, er)
} else {
Ready::invalid().repeat::<N>()
};
(ep, (ir, ()), s)
})
}
}
}
impl<P: Copy, R1: Copy, R2: Copy, const D: Dep> MuxExt<2> for (I<VrH<P, R1>, D>, I<VrH<P, R2>, D>) {
type E = I<VrH<P, (R1, R2)>, D>;
/// Muxes two `VrH` hazard interfaces based on `cntl`.
///
/// `cntl` selects which ingress interface to connect to the egress interface.
///
/// - Payloads: Outputs the payload of the interface selected by `cntl`.
/// - Resolver: The selected interface's resolver follows the egress resolver. All the other resolvers are invalid.
///
/// | Interface | Ingress | Egress |
/// | :-------: | -------------------------- | ----------------- |
/// | **Fwd** | `(HOption<P>, HOption<P>)` | `HOption<P>` |
/// | **Bwd** | `(Ready<R1>, Ready<R2>)` | `Ready<(R1, R2)>` |
fn mux(self, cntl: Valid<U<{ clog2(2) }>>) -> I<VrH<P, (R1, R2)>, D> {
unsafe {
(self, cntl).fsm::<I<VrH<P, (R1, R2)>, D>, ()>((), |((ip1, ip2), sel), er, s| {
let ep = sel.and_then(|sel| if sel == 0.into_u() { ip1 } else { ip2 });
let ir1 = if sel.filter(|p| p == 0.into_u()).is_some() {
er.map(|r| r.0)
} else {
Ready::new(false, er.inner.0)
};
let ir2 = if sel.filter(|p| p == 1.into_u()).is_some() {
er.map(|r| r.1)
} else {
Ready::new(false, er.inner.1)
};
(ep, ((ir1, ir2), ()), s)
})
}
}
}