OpenCores
URL https://opencores.org/ocsvn/funbase_ip_library/funbase_ip_library/trunk

Subversion Repositories funbase_ip_library

[/] [funbase_ip_library/] [trunk/] [TUT/] [ip.hwp.interface/] [udp2hibi/] [1.0/] [tb/] [tb_rx_ctrl.vhd] - Rev 145

Compare with Previous | Blame | View Log

-------------------------------------------------------------------------------
-- Title      : Testbench for Rx ctrl block
-- Project    : 
-------------------------------------------------------------------------------
-- File       : tb_rx_ctrl.vhd
-- Author     : Jussi Nieminen
-- Last update: 2012-03-21
-- Platform   : Sim only
-------------------------------------------------------------------------------
-- Description: A couple of hard-coded test cases for ctrl-registers.
-------------------------------------------------------------------------------
-- Revisions  :
-- Date        Version  Author  Description
-- 2009/12/18  1.0      niemin95        Created
-------------------------------------------------------------------------------
 
library ieee;
use ieee.std_logic_1164.all;
use ieee.numeric_std.all;
use work.udp2hibi_pkg.all;
 
 
entity tb_rx_ctrl is
end tb_rx_ctrl;
 
 
architecture tb of tb_rx_ctrl is
 
  constant frequency_c     : integer   := 50000000;
  constant period_c        : time      := 20 ns;
  constant udp_ip_period_c : time      := 40 ns;
  signal   clk             : std_logic := '1';
  signal   clk_udp         : std_logic := '1';
  signal   rst_n           : std_logic := '0';
 
  constant rx_multiclk_fifo_depth_c : integer := 10;
  constant tx_fifo_depth_c          : integer := 10;
  constant hibi_data_width_c        : integer := 32;
 
  -- from UDP/IP
  signal rx_data_to_duv        : std_logic_vector(udp_block_data_w_c-1 downto 0) := (others => '0');
  signal rx_data_valid_to_duv  : std_logic                                       := '0';
  signal rx_re_from_duv        : std_logic;
  signal new_rx_to_duv         : std_logic                                       := '0';
  signal rx_len_to_duv         : std_logic_vector(tx_len_w_c-1 downto 0)         := (others => '0');
  signal source_ip_to_duv      : std_logic_vector(ip_addr_w_c-1 downto 0)        := (others => '0');
  signal dest_port_to_duv      : std_logic_vector(udp_port_w_c-1 downto 0)       := (others => '0');
  signal source_port_to_duv    : std_logic_vector(udp_port_w_c-1 downto 0)       := (others => '0');
  signal rx_erroneous_to_duv   : std_logic                                       := '0';
 
  -- to/from ctrl regs
  signal ip_from_duv           : std_logic_vector(ip_addr_w_c-1 downto 0);
  signal dest_port_from_duv    : std_logic_vector(udp_port_w_c-1 downto 0);
  signal source_port_from_duv  : std_logic_vector(udp_port_w_c-1 downto 0);
  signal rx_addr_valid_to_duv  : std_logic                                       := '0';
 
  -- to/from hibi_transmitter
  signal send_request_from_duv : std_logic;
  signal ready_for_tx_to_duv   : std_logic                                       := '0';
  signal rx_empty_from_duv     : std_logic;
  signal rx_data_from_duv      : std_logic_vector(hibi_data_width_c-1 downto 0);
  signal rx_re_to_duv          : std_logic                                       := '0';
 
 
  -- Testbench's state machines etc.
  type   send_state_type is (idle, sending);
  signal send_state      : send_state_type;
  signal send_data       : std_logic := '0';
  signal send_done       : std_logic;
  signal rx_data_valid_r : std_logic;
  signal current         : integer;
 
 
  -- Test constants
  constant test_data_amount_c : integer        := 16;  -- #words
  type     test_data_type is array (0 to test_data_amount_c-1) of std_logic_vector(udp_block_data_w_c-1 downto 0);
  constant test_data          : test_data_type :=
    (x"0100", x"0302", x"0504", x"0706", x"0908", x"0b0a", x"0d0c", x"0f0e",
      x"1110", x"1312", x"1514", x"1716", x"1918", x"1b1a", x"1d1c", x"1f1e");
 
  type test_txs_type is
  record
    source_ip         : std_logic_vector(ip_addr_w_c-1 downto 0);
    source_port       : std_logic_vector(udp_port_w_c-1 downto 0);
    dest_port         : std_logic_vector(udp_port_w_c-1 downto 0);
    rx_addr_valid     : std_logic;
    rx_len            : integer;         -- #bytes
    rx_erroneous      : std_logic;
    delay_before_next : time;
  end record;
 
  constant num_of_tests_c : integer := 5;
  type     test_txs_array is array (0 to num_of_tests_c-1) of test_txs_type;
 
  -- test cases:
  -- 0. Normal (not erroneus, rx address valid) 200 bytes long packet
  -- 1. Packet without an receiver (should be dumped)
  -- 2. Erroneous packet (should also be dumped)
  -- 3. Very short (1 byte) transfer with minimal delay
  -- 4. just something following the earlier short one
  constant test_txs : test_txs_array :=
    ((source_ip          => x"01234567",
       source_port       => x"1212",
       dest_port         => x"2121",
       rx_addr_valid     => '1',
       rx_len            => 200,
       rx_erroneous      => '0',
       delay_before_next => 10 * period_c),
     (source_ip          => x"12345678",
       source_port       => x"2323",
       dest_port         => x"3232",
       rx_addr_valid     => '0',
       rx_len            => 40,
       rx_erroneous      => '0',
       delay_before_next => 10 * period_c),
     (source_ip          => x"23456789",
       source_port       => x"3434",
       dest_port         => x"4343",
       rx_addr_valid     => '1',
       rx_len            => 30,
       rx_erroneous      => '1',
       delay_before_next => 10 * period_c),
     (source_ip          => x"3456789a",
       source_port       => x"4545",
       dest_port         => x"5454",
       rx_addr_valid     => '1',
       rx_len            => 1,
       rx_erroneous      => '0',
       delay_before_next => udp_ip_period_c),
     (source_ip          => x"456789ab",
       source_port       => x"5656",
       dest_port         => x"6565",
       rx_addr_valid     => '1',
       rx_len            => 20,
       rx_erroneous      => '0',
       delay_before_next => 10 * period_c)
     );
 
  signal test_id   : integer;
 
 
begin  -- tb
 
 
  duv : entity work.rx_ctrl
    generic map (
      rx_multiclk_fifo_depth_g => rx_multiclk_fifo_depth_c,
      tx_fifo_depth_g          => tx_fifo_depth_c,
      hibi_data_width_g        => hibi_data_width_c,
      frequency_g              => frequency_c
      )
    port map (
      clk              => clk,
      clk_udp          => clk_udp,
      rst_n            => rst_n,
      rx_data_in       => rx_data_to_duv,
      rx_data_valid_in => rx_data_valid_to_duv,
      rx_re_out        => rx_re_from_duv,
      new_rx_in        => new_rx_to_duv,
      rx_len_in        => rx_len_to_duv,
      source_ip_in     => source_ip_to_duv,
      dest_port_in     => dest_port_to_duv,
      source_port_in   => source_port_to_duv,
      rx_erroneous_in  => rx_erroneous_to_duv,
      ip_out           => ip_from_duv,
      dest_port_out    => dest_port_from_duv,
      source_port_out  => source_port_from_duv,
      rx_addr_valid_in => rx_addr_valid_to_duv,
      send_request_out => send_request_from_duv,
      ready_for_tx_in  => ready_for_tx_to_duv,
      rx_empty_out     => rx_empty_from_duv,
      rx_data_out      => rx_data_from_duv,
      rx_re_in         => rx_re_to_duv
      );
 
 
  -- clk generation:
  clk     <= not clk     after period_c/2;
  clk_udp <= not clk_udp after udp_ip_period_c/2;
  rst_n   <= '1'         after 4*period_c;
 
  test_id <= current;                   -- ES
 
 
  -----------------------------------------------------------------------------
  -- Three processes
  --  - main gives commands to others, (behav) process with wait statements
  --  - sender provides stimulues when requested, seq. process
  --  - reader checks response, seq. process
  -----------------------------------------------------------------------------
 
  main_ctrl : process
  begin  -- process main_ctrl
 
    if rst_n = '0' then
      wait until rst_n = '1';
    end if;
 
    wait for period_c*4;
 
    -- Start the test transfers
    for n in 0 to num_of_tests_c-1 loop
 
      current <= n;
 
      -- Give parameters to duv
      new_rx_to_duv       <= '1';
      source_ip_to_duv    <= test_txs(n).source_ip;
      source_port_to_duv  <= test_txs(n).source_port;
      dest_port_to_duv    <= test_txs(n).dest_port;
      rx_len_to_duv       <= std_logic_vector(to_unsigned(test_txs(n).rx_len, tx_len_w_c));
      rx_erroneous_to_duv <= test_txs(n).rx_erroneous;
 
      -- Request other process to provide the data
      send_data <= '1';
      wait for udp_ip_period_c;
      send_data <= '0';
 
      rx_addr_valid_to_duv <= test_txs(n).rx_addr_valid;
 
      -- Obsolete check, perhaps?
      if ready_for_tx_to_duv = '0' then
        -- Let this be down for a while. Assertion inside the  reader process
        -- will check that no
        -- send requests are made before this is up
        wait for period_c*30;
        ready_for_tx_to_duv <= '1';
      end if;
 
      -- Wait until duv starts reading and then clear the params
      if rx_re_from_duv = '0' then
        wait until rx_re_from_duv = '1';
      end if;
      wait for period_c;
 
      new_rx_to_duv       <= '0';
      source_ip_to_duv    <= (others => 'Z');
      source_port_to_duv  <= (others => 'Z');
      dest_port_to_duv    <= (others => 'Z');
      rx_erroneous_to_duv <= 'Z';
 
 
      -- Wait until other process completes
      if send_done = '0' then
        wait until send_done = '1';
      end if;
 
      wait for test_txs(n).delay_before_next;
 
    end loop;  -- n
 
 
    wait for period_c*30;
 
    report "Simulation ended." severity failure;
 
  end process main_ctrl;
 
 
  -----------------------------------------------------------------------------
  --
  -----------------------------------------------------------------------------
  sender : process (clk_udp, rst_n)
    variable send_cnt : integer := 0;
  begin  -- process sender
    if rst_n = '0' then                 -- asynchronous reset (active low)
 
      rx_data_to_duv  <= (others => '0');
      rx_data_valid_r <= '0';
      send_state      <= idle;
      send_done       <= '0';
 
    elsif clk_udp'event and clk_udp = '1' then  -- rising clock edge
 
      case send_state is
        when idle =>
          -- Wait for other process' request
 
          if send_data = '1' then
            send_state      <= sending;
            send_done       <= '0';
            rx_data_valid_r <= '1';
            rx_data_to_duv  <= test_data(0);
          end if;
 
        when sending =>
          -- Provide the data to duv
 
          if rx_re_from_duv = '1' and rx_data_valid_r = '1' then
            rx_data_valid_r <= '0';
 
            if test_txs(current).rx_len - send_cnt*2 <= 2 then
              send_done  <= '1';
              send_cnt   := 0;
              send_state <= idle;
 
            else
              rx_data_to_duv <= test_data((send_cnt+1) mod test_data_amount_c);
              send_cnt       := send_cnt + 1;
 
            end if;
 
          else
            rx_data_valid_r <= '1';
          end if;
 
        when others => null;
      end case;
 
    end if;
  end process sender;
 
  -- this is done to make the valid signal go up at the same time with new_rx
  rx_data_valid_to_duv <= rx_data_valid_r or send_data;
 
 
 
  -----------------------------------------------------------------------------
  -- 
  -----------------------------------------------------------------------------
  reader : process (clk, rst_n)
    variable read_cnt : integer := 0;
  begin  -- process reader
    if rst_n = '0' then                 -- asynchronous reset (active low)
 
    elsif clk'event and clk = '1' then  -- rising clock edge
 
      -- Check the handshake
      if send_request_from_duv = '1' then
        assert ready_for_tx_to_duv = '1'
          report "Failure in test: Send request when not ready." severity failure;
        -- reset read_cnt
        read_cnt := 0;
      end if;
 
      -- Read the data
      if rx_empty_from_duv = '0' then
        rx_re_to_duv <= '1';
      else
        rx_re_to_duv <= '0';
      end if;
 
      -- Check the data
      if rx_re_to_duv = '1' and rx_empty_from_duv = '0' then
        -- We are reading, check that data is correct
 
        assert test_txs(current).rx_addr_valid = '1' and
          test_txs(current).rx_erroneous = '0'
          report "Failure in test: Rx not dumped." severity failure;
 
        assert rx_data_from_duv =
          test_data((2*read_cnt+1) mod test_data_amount_c) & test_data((2*read_cnt) mod test_data_amount_c)
          report "Warning: Either invalid data or last word from odd length rx. Check manually!" severity warning;
 
        read_cnt := read_cnt + 1;
      end if;
 
    end if;
  end process reader;
 
end tb;
 

Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.