blob: b6c5800321cc95b71d5e4ff6f3d55060afff6e5d (
plain)
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
|
library ieee;
use ieee.std_logic_1164.all;
entity uart_tx is
generic (
-- TODO: range in submodules is not yet supported by synthesis
-- it would be useful to limit between 5 to 8
C_BITS : integer := 8;
C_CYCLES_PER_BIT : integer := 104
);
port (
isl_clk : in std_logic;
isl_valid : in std_logic;
islv_data : in std_logic_vector(C_BITS-1 downto 0);
osl_ready : out std_logic;
osl_data_n : out std_logic
);
end entity uart_tx;
architecture rtl of uart_tx is
signal int_cycle_cnt : integer range 0 to C_CYCLES_PER_BIT-1 := 0;
signal int_bit_cnt : integer range 0 to C_BITS+2 := 0;
signal slv_data : std_logic_vector(C_BITS downto 0) := (others => '0');
type t_state is (IDLE, INIT, SEND);
signal state : t_state;
begin
process(isl_clk)
begin
if rising_edge(isl_clk) then
case state is
when IDLE =>
if isl_valid = '1' then
state <= INIT;
end if;
when INIT =>
int_cycle_cnt <= 0;
int_bit_cnt <= 0;
slv_data <= islv_data & '1';
state <= SEND;
when SEND =>
if int_cycle_cnt < C_CYCLES_PER_BIT-1 then
int_cycle_cnt <= int_cycle_cnt+1;
elsif int_bit_cnt < C_BITS+1 then
int_cycle_cnt <= 0;
int_bit_cnt <= int_bit_cnt+1;
slv_data <= '0' & slv_data(slv_data'LEFT downto 1);
else
state <= IDLE;
end if;
end case;
end if;
end process;
osl_ready <= '1' when state = IDLE else '0';
osl_data_n <= not slv_data(0); -- low active
end architecture rtl;
|