-
Notifications
You must be signed in to change notification settings - Fork 64
/
Copy pathmii_gmii.vhd
208 lines (188 loc) · 5.99 KB
/
mii_gmii.vhd
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
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
-- This file is part of the ethernet_mac project.
--
-- For the full copyright and license information, please read the
-- LICENSE.md file that was distributed with this source code.
-- Adaption layer for data transfer with MII and GMII
library ieee;
use ieee.std_logic_1164.all;
use ieee.numeric_std.all;
use work.framing_common.all;
use work.ethernet_types.all;
entity mii_gmii is
port(
tx_reset_i : in std_ulogic;
tx_clock_i : in std_ulogic;
rx_reset_i : in std_ulogic;
rx_clock_i : in std_ulogic;
-- MII (Media-independent interface)
mii_tx_en_o : out std_ulogic;
mii_txd_o : out std_ulogic_vector(7 downto 0);
mii_rx_er_i : in std_ulogic;
mii_rx_dv_i : in std_ulogic;
mii_rxd_i : in std_ulogic_vector(7 downto 0);
-- RGMII (Reduced pin count gigabit media-independent interface)
-- Leave open if RGMII is not used
rgmii_tx_ctl_o : out std_ulogic;
rgmii_rx_ctl_i : in std_ulogic;
-- Other pins:
-- mii_gtx_clk_o is TXC
-- mii_txd_o[3:0] is TD[3:0]
-- mii_rx_clk_i is RXC
-- mii_rxd_i[3:0] is RD[3:0]
-- Interface control signals
-- Must stay stable after tx_reset_i or rx_reset_i is deasserted
speed_select_i : in t_ethernet_speed;
-- TX/RX control
-- TX signals synchronous to tx_clock
tx_enable_i : in std_ulogic;
-- When asserted together with tx_enable_i, tx_byte_sent_o works as normal, but no data is actually
-- put onto the media-independent interface (for IPG transmission)
tx_gap_i : in std_ulogic;
tx_data_i : in t_ethernet_data;
-- Put next data byte on tx_data_i when asserted
tx_byte_sent_o : out std_ulogic;
-- RX signals synchronous to rx_clock
-- Asserted as long as one continuous frame is being received
rx_frame_o : out std_ulogic;
-- Valid when rx_byte_received_o is asserted
rx_data_o : out t_ethernet_data;
rx_byte_received_o : out std_ulogic;
rx_error_o : out std_ulogic
);
end entity;
architecture rtl of mii_gmii is
-- Transmission
type t_mii_gmii_tx_state is (
TX_INIT,
TX_GMII,
TX_MII_LO_QUAD,
TX_MII_HI_QUAD
);
signal tx_state : t_mii_gmii_tx_state := TX_INIT;
-- Reception
type t_mii_gmii_rx_state is (
RX_INIT,
RX_GMII,
RX_MII_LO_QUAD,
RX_MII_HI_QUAD
);
signal rx_state : t_mii_gmii_rx_state := RX_INIT;
begin
-- TX FSM is split into this synchronous process and the output process for tx_byte_sent_o
-- A strictly one-process FSM is impractical for MII transmission: Wait states would be needed
-- to correctly generate tx_byte_sent_o for GMII.
mii_gmii_tx_sync : process(tx_reset_i, tx_clock_i)
begin
-- Use asynchronous reset, clock_tx is not guaranteed to be running during system initialization
if tx_reset_i = '1' then
tx_state <= TX_INIT;
mii_tx_en_o <= '0';
elsif rising_edge(tx_clock_i) then
mii_tx_en_o <= '0';
mii_txd_o <= (others => '0');
case tx_state is
when TX_INIT =>
case speed_select_i is
when SPEED_1000MBPS =>
tx_state <= TX_GMII;
when others =>
tx_state <= TX_MII_LO_QUAD;
end case;
when TX_GMII =>
-- GMII is very simple: Pass data through when
-- tx_enable_i is asserted
mii_tx_en_o <= tx_enable_i and not tx_gap_i;
mii_txd_o <= tx_data_i;
when TX_MII_LO_QUAD =>
mii_tx_en_o <= tx_enable_i and not tx_gap_i;
mii_txd_o <= "0000" & tx_data_i(3 downto 0);
if tx_enable_i = '1' then
-- Advance to high quad only when data was actually sent
tx_state <= TX_MII_HI_QUAD;
end if;
when TX_MII_HI_QUAD =>
-- tx_enable_i is not considered, a full byte always has to be sent
mii_tx_en_o <= not tx_gap_i;
mii_txd_o <= "0000" & tx_data_i(7 downto 4);
tx_state <= TX_MII_LO_QUAD;
end case;
end if;
end process;
-- TX output process
-- Generates only the tx_byte_sent_o output
mii_gmii_tx_output : process(tx_state, tx_enable_i, speed_select_i)
begin
-- Default output value
tx_byte_sent_o <= '0';
case tx_state is
when TX_INIT =>
-- Look ahead to have tx_byte_sent already set in the TX_GMII clock cycle
if tx_enable_i = '1' and speed_select_i = SPEED_1000MBPS then
tx_byte_sent_o <= '1';
end if;
when TX_GMII =>
-- Look ahead again
if tx_enable_i = '0' then
tx_byte_sent_o <= '0';
else
tx_byte_sent_o <= '1';
end if;
when TX_MII_LO_QUAD =>
null;
when TX_MII_HI_QUAD =>
-- MII is simpler, no look-ahead needed
tx_byte_sent_o <= '1';
end case;
end process;
-- MII/GMII packet reception
mii_gmii_rx_fsm : process(rx_clock_i, rx_reset_i)
begin
if rx_reset_i = '1' then
rx_state <= RX_INIT;
rx_byte_received_o <= '0';
elsif rising_edge(rx_clock_i) then
-- Default output values
rx_frame_o <= '0';
rx_byte_received_o <= '0';
rx_error_o <= '0';
if rx_state /= RX_INIT then
-- Hand indicators through
rx_error_o <= mii_rx_er_i;
rx_frame_o <= mii_rx_dv_i;
end if;
case rx_state is
when RX_INIT =>
-- Wait for a pause in reception
if mii_rx_dv_i = '0' then
case speed_select_i is
when SPEED_1000MBPS =>
rx_state <= RX_GMII;
when others =>
rx_state <= RX_MII_LO_QUAD;
end case;
end if;
when RX_GMII =>
-- Just pass the data through
rx_data_o <= mii_rxd_i;
rx_byte_received_o <= mii_rx_dv_i;
when RX_MII_LO_QUAD =>
-- Wait until start of reception
if mii_rx_dv_i = '1' then
rx_state <= RX_MII_HI_QUAD;
end if;
-- Capture low quad
rx_data_o(3 downto 0) <= mii_rxd_i(3 downto 0);
when RX_MII_HI_QUAD =>
-- Capture high quad and mark it valid
rx_data_o(7 downto 4) <= mii_rxd_i(3 downto 0);
rx_byte_received_o <= '1';
rx_frame_o <= '1';
if mii_rx_dv_i = '0' then
-- Frame ended prematurely on a half-byte
rx_error_o <= '1';
end if;
rx_state <= RX_MII_LO_QUAD;
end case;
end if;
end process;
end architecture;