diff --git a/libraries/base/common/hdllib.cfg b/libraries/base/common/hdllib.cfg index 7e3305c36157f46417038c2e5a3f3b6c85aa4bd8..57c7d22f006785498e6b2ad8ccca0f056d8ba1d0 100644 --- a/libraries/base/common/hdllib.cfg +++ b/libraries/base/common/hdllib.cfg @@ -11,6 +11,7 @@ synth_files = $UNB/Firmware/modules/common/src/vhdl/common_mem_pkg.vhd $UNB/Firmware/modules/common/src/vhdl/common_field_pkg.vhd $UNB/Firmware/modules/common/src/vhdl/common_lfsr_sequences_pkg.vhd + $UNB/Firmware/modules/common/src/vhdl/common_network_layers_pkg.vhd $UNB/Firmware/modules/common/src/vhdl/common_components_pkg.vhd $UNB/Firmware/modules/MegaWizard/arith/lut_add_sub.vhd diff --git a/libraries/base/common/src/vhdl/common_network_layers_pkg.vhd b/libraries/base/common/src/vhdl/common_network_layers_pkg.vhd new file mode 100644 index 0000000000000000000000000000000000000000..c835020efd35ae0699d47e5c740bdf9ae68af782 --- /dev/null +++ b/libraries/base/common/src/vhdl/common_network_layers_pkg.vhd @@ -0,0 +1,739 @@ +------------------------------------------------------------------------------- +-- +-- Copyright (C) 2010 +-- ASTRON (Netherlands Institute for Radio Astronomy) <http://www.astron.nl/> +-- JIVE (Joint Institute for VLBI in Europe) <http://www.jive.nl/> +-- P.O.Box 2, 7990 AA Dwingeloo, The Netherlands +-- +-- This program is free software: you can redistribute it and/or modify +-- it under the terms of the GNU General Public License as published by +-- the Free Software Foundation, either version 3 of the License, or +-- (at your option) any later version. +-- +-- This program is distributed in the hope that it will be useful, +-- but WITHOUT ANY WARRANTY; without even the implied warranty of +-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +-- GNU General Public License for more details. +-- +-- You should have received a copy of the GNU General Public License +-- along with this program. If not, see <http://www.gnu.org/licenses/>. +-- +------------------------------------------------------------------------------- + +LIBRARY IEEE, common_lib; +USE IEEE.std_logic_1164.ALL; +USE IEEE.numeric_std.ALL; +USE common_lib.common_pkg.ALL; + + +PACKAGE eth_layers_pkg IS + + -- All *_len constants are in nof octets = nof bytes + CONSTANT c_8 : NATURAL := c_octet_w; -- = 8 + + -- The Ethernet packet is interpreted as words and word aligned + CONSTANT c_32 : NATURAL := c_word_w; -- = 32 + + ------------------------------------------------------------------------------ + -- Ethernet Packet (with payload word alignment!) + -- + -- 0 15 16 31 wi + -- |----------------------------------------------------------------------| + -- | Word Align | Destination MAC Address | 0 + -- |----------------------------------- | + -- | | 1 + -- |----------------------------------------------------------------------| + -- | Source MAC Address | 2 + -- | ------------------------------------| + -- | | EtherType | 3 + -- |----------------------------------|-----------------------------------| + -- | | + -- | Ethernet Payload | + -- | | + -- |------------------------------------------------------------ // ------| + -- | Frame Check Sequence | + -- |------------------------------------------------------------ // ------| + -- + + -- field widths in bits '_w' or in bytes '_len', '_min', '_max', '_sz' + CONSTANT c_eth_preamble_len : NATURAL := 8; + CONSTANT c_eth_mac_addr_len : NATURAL := 6; + CONSTANT c_eth_mac_addr_w : NATURAL := c_eth_mac_addr_len*c_8; + CONSTANT c_eth_type_len : NATURAL := 2; + CONSTANT c_eth_type_w : NATURAL := c_eth_type_len*c_8; + CONSTANT c_eth_header_len : NATURAL := 2*c_eth_mac_addr_len+c_eth_type_len; -- = 14 + CONSTANT c_eth_payload_min : NATURAL := 46; + CONSTANT c_eth_payload_max : NATURAL := 1500; + CONSTANT c_eth_payload_jumbo_max : NATURAL := 9000; + CONSTANT c_eth_crc_len : NATURAL := 4; + CONSTANT c_eth_crc_w : NATURAL := c_eth_crc_len*c_8; + CONSTANT c_eth_gap_len : NATURAL := 12; + CONSTANT c_eth_word_align_len : NATURAL := 2; -- c_eth_header_len + 2 = 16, align payload to 32 bit boundaries + CONSTANT c_eth_word_align_w : NATURAL := c_eth_word_align_len*c_8; + CONSTANT c_eth_frame_max : NATURAL := c_eth_header_len + c_eth_payload_max + c_eth_crc_len; -- = 1518 + CONSTANT c_eth_frame_jumbo_max : NATURAL := c_eth_header_len + c_eth_payload_jumbo_max + c_eth_crc_len; -- = 9018 + + -- default field values + CONSTANT c_eth_preamble : NATURAL := 5; -- nibble "0101" + CONSTANT c_eth_frame_delimiter : NATURAL := 13; -- nibble "1101" + CONSTANT c_eth_word_align : NATURAL := 0; + + -- useful field values + CONSTANT c_eth_mac_slv : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0) := (OTHERS=>'X'); -- Ethernet MAC slv RANGE + CONSTANT c_eth_bc_mac : STD_LOGIC_VECTOR(c_eth_mac_slv'RANGE) := (OTHERS=>'1'); -- Broadcast destination MAC + + CONSTANT c_eth_type_slv : STD_LOGIC_VECTOR(c_eth_type_w-1 DOWNTO 0) := (OTHERS=>'X'); -- Ethernet TYPE slv RANGE + CONSTANT c_eth_type_arp : NATURAL := 16#0806#; -- ARP = Address Resolution Prorotol + CONSTANT c_eth_type_ip : NATURAL := 16#0800#; -- IPv4 = Internet Protocol, Version 4 + + TYPE t_eth_header IS RECORD + word_align : STD_LOGIC_VECTOR(c_eth_word_align_w-1 DOWNTO 0); + dst_mac : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0); + src_mac : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0); + eth_type : STD_LOGIC_VECTOR(c_eth_type_w-1 DOWNTO 0); + END RECORD; + + -- field valid word indices + CONSTANT c_eth_lo_wi : NATURAL := 0; -- first word index + CONSTANT c_eth_dst_mac_wi : NATURAL := 1; + CONSTANT c_eth_src_mac_wi : NATURAL := 3; + CONSTANT c_eth_type_wi : NATURAL := 3; + CONSTANT c_eth_hi_wi : NATURAL := 3; -- last word index + CONSTANT c_eth_nof_words : NATURAL := c_eth_hi_wi - c_eth_lo_wi + 1; -- = (c_eth_word_align_len + c_eth_word_align_len) / c_word_sz + + + ------------------------------------------------------------------------------ + -- IPv4 Packet + -- + -- 0 3 4 7 8 15 16 18 19 31 wi + -- |----------------------------------------------------------------------| + -- | Version | HLEN | Services | Total Length | 4 + -- |----------------------------------------------------------------------| + -- | Identification | Flags | Fragment Offset | 5 + -- |----------------------------------------------------------------------| + -- | TTL | Protocol | Header Checksum | 6 + -- |----------------------------------------------------------------------| + -- | Source IP Address | 7 + -- |----------------------------------------------------------------------| + -- | Destination IP Address | 8 + -- |----------------------------------------------------------------------| + -- | | + -- | IP Payload | + -- | | + -- |------------------------------------------------------------ // ------| + -- + + -- field widths in bits '_w' or in bytes '_len' + CONSTANT c_ip_version_w : NATURAL := 4; -- 4-bit field + CONSTANT c_ip_header_length_w : NATURAL := 4; -- 4-bit field + CONSTANT c_ip_version_header_len : NATURAL := 1; + CONSTANT c_ip_version_header_w : NATURAL := c_ip_version_header_len*c_8; + CONSTANT c_ip_services_len : NATURAL := 1; + CONSTANT c_ip_services_w : NATURAL := c_ip_services_len*c_8; + CONSTANT c_ip_total_length_len : NATURAL := 2; + CONSTANT c_ip_total_length_w : NATURAL := c_ip_total_length_len*c_8; + CONSTANT c_ip_identification_len : NATURAL := 2; + CONSTANT c_ip_identification_w : NATURAL := c_ip_identification_len*c_8; + CONSTANT c_ip_flags_w : NATURAL := 3; -- 3-bit field + CONSTANT c_ip_fragment_offset_w : NATURAL := 13; -- 13-bit field + CONSTANT c_ip_flags_fragment_len : NATURAL := 2; + CONSTANT c_ip_flags_fragment_w : NATURAL := c_ip_flags_fragment_len*c_8; + CONSTANT c_ip_time_to_live_len : NATURAL := 1; + CONSTANT c_ip_time_to_live_w : NATURAL := c_ip_time_to_live_len*c_8; + CONSTANT c_ip_protocol_len : NATURAL := 1; + CONSTANT c_ip_protocol_w : NATURAL := c_ip_protocol_len*c_8; + CONSTANT c_ip_header_checksum_len : NATURAL := 2; + CONSTANT c_ip_header_checksum_w : NATURAL := c_ip_header_checksum_len*c_8; + CONSTANT c_ip_addr_len : NATURAL := 4; + CONSTANT c_ip_addr_w : NATURAL := c_ip_addr_len*c_8; + + -- [0:7] [8:15] [16:31] + CONSTANT c_ip_header_len : NATURAL := c_ip_version_header_len + c_ip_services_len + c_ip_total_length_len + + c_ip_identification_len + c_ip_flags_fragment_len + + c_ip_time_to_live_len + c_ip_protocol_len + c_ip_header_checksum_len + + c_ip_addr_len + + c_ip_addr_len; + -- = c_ip_header_length * c_word_sz = 20 + -- default field values + CONSTANT c_ip_version : NATURAL := 4; -- 4 = IPv4, + CONSTANT c_ip_header_length : NATURAL := 5; -- 5 = nof words in the header, no options field support + CONSTANT c_ip_services : NATURAL := 0; -- 0 = default, use default on transmit, ignore on receive, copy on reply + CONSTANT c_ip_total_length : NATURAL := 20; -- >= 20, nof bytes in entire datagram including header and data + CONSTANT c_ip_identification : NATURAL := 0; -- identification number, copy on reply + CONSTANT c_ip_flags : NATURAL := 2; -- 2 = don't fragment and this is the last fragment + CONSTANT c_ip_fragment_offset : NATURAL := 0; -- 0 = first fragment + CONSTANT c_ip_time_to_live : NATURAL := 127; -- number of hops until the packet will be discarded + CONSTANT c_ip_header_checksum : NATURAL := 0; -- init value + + -- useful field values + CONSTANT c_ip_protocol_slv : STD_LOGIC_VECTOR(c_ip_protocol_w-1 DOWNTO 0) := (OTHERS=>'X'); -- IP protocol slv RANGE + CONSTANT c_ip_protocol_udp : NATURAL := 17; -- UDP = User Datagram Protocol (for board control and streaming data) + CONSTANT c_ip_protocol_icmp : NATURAL := 1; -- ICMP = Internet Control Message Protocol (for ping) + + CONSTANT c_ip_addr_slv : STD_LOGIC_VECTOR(c_ip_addr_w-1 DOWNTO 0) := (OTHERS=>'X'); -- IP address slv RANGE + + TYPE t_ip_header IS RECORD + version : STD_LOGIC_VECTOR(c_ip_version_w-1 DOWNTO 0); -- 4 bit + header_length : STD_LOGIC_VECTOR(c_ip_header_length_w-1 DOWNTO 0); -- 4 bit + services : STD_LOGIC_VECTOR(c_ip_services_w-1 DOWNTO 0); -- 1 octet + total_length : STD_LOGIC_VECTOR(c_ip_total_length_w-1 DOWNTO 0); -- 2 octet + identification : STD_LOGIC_VECTOR(c_ip_identification_w-1 DOWNTO 0); -- 2 octet + flags : STD_LOGIC_VECTOR(c_ip_flags_w-1 DOWNTO 0); -- 3 bit + fragment_offset : STD_LOGIC_VECTOR(c_ip_fragment_offset_w-1 DOWNTO 0); -- 13 bit + time_to_live : STD_LOGIC_VECTOR(c_ip_time_to_live_w-1 DOWNTO 0); -- 1 octet + protocol : STD_LOGIC_VECTOR(c_ip_protocol_w-1 DOWNTO 0); -- 1 octet + header_checksum : STD_LOGIC_VECTOR(c_ip_header_checksum_w-1 DOWNTO 0); -- 2 octet + src_ip_addr : STD_LOGIC_VECTOR(c_ip_addr_w-1 DOWNTO 0); -- 4 octet + dst_ip_addr : STD_LOGIC_VECTOR(c_ip_addr_w-1 DOWNTO 0); -- 4 octet + END RECORD; + + -- field valid word indices + CONSTANT c_ip_lo_wi : NATURAL := 4; -- first word index + CONSTANT c_ip_version_wi : NATURAL := 4; + CONSTANT c_ip_header_length_wi : NATURAL := 4; + CONSTANT c_ip_services_wi : NATURAL := 4; + CONSTANT c_ip_total_length_wi : NATURAL := 4; + CONSTANT c_ip_identification_wi : NATURAL := 5; + CONSTANT c_ip_flags_wi : NATURAL := 5; + CONSTANT c_ip_fragment_offset_wi : NATURAL := 5; + CONSTANT c_ip_time_to_live_wi : NATURAL := 6; + CONSTANT c_ip_protocol_wi : NATURAL := 6; + CONSTANT c_ip_header_checksum_wi : NATURAL := 6; + CONSTANT c_ip_src_ip_addr_wi : NATURAL := 7; + CONSTANT c_ip_dst_ip_addr_wi : NATURAL := 8; + CONSTANT c_ip_hi_wi : NATURAL := 8; -- last word index + CONSTANT c_ip_nof_words : NATURAL := c_ip_hi_wi - c_ip_lo_wi + 1; -- = c_ip_header_len / c_word_sz + + + ------------------------------------------------------------------------------ + -- ARP Packet + -- + -- 0 7 8 15 16 31 wi + -- |----------------------------------------------------------------------| + -- | Hardware Type | Protocol Type | 4 + -- |----------------------------------------------------------------------| + -- | HW Addr Len | Prot Addr Len | Operation | 5 + -- |----------------------------------------------------------------------| + -- | Sender Hardware Address | 6 + -- | ------------------------------------| + -- | | | 7 + -- |---------------------------------/ /----------------------------------| + -- | Sender Protocol Address | | 8 + -- |----------------------------------- | + -- | Target Hardware Address | 9 + -- |----------------------------------------------------------------------| + -- | Target Protocol Address | 10 + -- |----------------------------------------------------------------------| + -- + + -- field widths in bits '_w' or in bytes '_len' + CONSTANT c_arp_htype_len : NATURAL := 2; + CONSTANT c_arp_htype_w : NATURAL := c_arp_htype_len*c_8; + CONSTANT c_arp_ptype_len : NATURAL := 2; + CONSTANT c_arp_ptype_w : NATURAL := c_arp_ptype_len*c_8; + CONSTANT c_arp_hlen_len : NATURAL := 1; + CONSTANT c_arp_hlen_w : NATURAL := c_arp_hlen_len*c_8; + CONSTANT c_arp_plen_len : NATURAL := 1; + CONSTANT c_arp_plen_w : NATURAL := c_arp_plen_len*c_8; + CONSTANT c_arp_oper_len : NATURAL := 2; + CONSTANT c_arp_oper_w : NATURAL := c_arp_oper_len*c_8; + + -- [0:15] [16:31] + CONSTANT c_arp_data_len : NATURAL := c_arp_htype_len + c_arp_ptype_len + + c_arp_hlen_len + c_arp_plen_len + c_arp_oper_len + + c_eth_mac_addr_len + c_ip_addr_len + + c_eth_mac_addr_len + c_ip_addr_len; + -- [0:47] [0:31] = 8 + 2*(6+4) = 28 + + -- default field values + CONSTANT c_arp_htype : NATURAL := 1; -- Hardware type, 1=ethernet + CONSTANT c_arp_ptype : NATURAL := c_eth_type_ip; -- Protocol type, do ARP for IPv4 + CONSTANT c_arp_hlen : NATURAL := c_eth_mac_addr_len; -- Hardware length = 6 + CONSTANT c_arp_plen : NATURAL := c_ip_addr_len; -- Protocol length = 4 + CONSTANT c_arp_oper_request : NATURAL := 1; -- Operator, 1=request + CONSTANT c_arp_oper_reply : NATURAL := 2; -- Operator, 2=reply + + -- useful field values + CONSTANT c_arp_dst_mac : STD_LOGIC_VECTOR(c_eth_mac_slv'RANGE) := c_eth_bc_mac; -- Broadcast destination MAC + CONSTANT c_arp_tha : STD_LOGIC_VECTOR(c_eth_mac_slv'RANGE) := c_eth_bc_mac; -- Broadcast target hardware address + + TYPE t_arp_packet IS RECORD + htype : STD_LOGIC_VECTOR(c_arp_htype_w-1 DOWNTO 0); -- 2 octet + ptype : STD_LOGIC_VECTOR(c_arp_ptype_w-1 DOWNTO 0); -- 2 octet + hlen : STD_LOGIC_VECTOR(c_arp_hlen_w-1 DOWNTO 0); -- 1 octet + plen : STD_LOGIC_VECTOR(c_arp_plen_w-1 DOWNTO 0); -- 1 octet + oper : STD_LOGIC_VECTOR(c_arp_oper_w-1 DOWNTO 0); -- 2 octet + sha : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0); -- 6 octet, Sender Hardware Address + spa : STD_LOGIC_VECTOR(c_ip_addr_w-1 DOWNTO 0); -- 4 octet, Sender Protocol Address + tha : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0); -- 6 octet, Target Hardware Address + tpa : STD_LOGIC_VECTOR(c_ip_addr_w-1 DOWNTO 0); -- 4 octet, Target Protocol Address + END RECORD; + + -- field valid word indices + CONSTANT c_arp_lo_wi : NATURAL := 4; -- first word index + CONSTANT c_arp_htype_wi : NATURAL := 4; + CONSTANT c_arp_ptype_wi : NATURAL := 4; + CONSTANT c_arp_hlen_wi : NATURAL := 5; + CONSTANT c_arp_plen_wi : NATURAL := 5; + CONSTANT c_arp_oper_wi : NATURAL := 5; + CONSTANT c_arp_sha_wi : NATURAL := 7; + CONSTANT c_arp_spa_wi : NATURAL := 8; + CONSTANT c_arp_tha_wi : NATURAL := 9; + CONSTANT c_arp_tpa_wi : NATURAL := 10; + CONSTANT c_arp_hi_wi : NATURAL := 10; -- last word index + CONSTANT c_arp_nof_words : NATURAL := c_arp_hi_wi - c_arp_lo_wi + 1; -- = c_arp_data_len / c_word_sz + + ------------------------------------------------------------------------------ + -- ICMP (for ping) + -- + -- 0 7 8 15 16 31 wi + -- |----------------------------------------------------------------------| + -- | Type | Code | Checksum | 9 + -- |----------------------------------------------------------------------| + -- | ID | Sequence | 10 + -- |----------------------------------------------------------------------| + -- | | + -- | ICMP Payload (padding data) | + -- | | + -- |------------------------------------------------------------ // ------| + -- + + -- field widths in bits '_w' or in bytes '_len' + CONSTANT c_icmp_msg_type_len : NATURAL := 1; + CONSTANT c_icmp_msg_type_w : NATURAL := c_icmp_msg_type_len*c_8; + CONSTANT c_icmp_code_len : NATURAL := 1; + CONSTANT c_icmp_code_w : NATURAL := c_icmp_code_len*c_8; + CONSTANT c_icmp_checksum_len : NATURAL := 2; + CONSTANT c_icmp_checksum_w : NATURAL := c_icmp_checksum_len*c_8; + CONSTANT c_icmp_id_len : NATURAL := 2; + CONSTANT c_icmp_id_w : NATURAL := c_icmp_id_len*c_8; + CONSTANT c_icmp_sequence_len : NATURAL := 2; + CONSTANT c_icmp_sequence_w : NATURAL := c_icmp_sequence_len*c_8; + CONSTANT c_icmp_header_len : NATURAL := c_icmp_msg_type_len + c_icmp_code_len + c_icmp_checksum_len + + c_icmp_id_len + c_icmp_sequence_len; + + -- default field values + CONSTANT c_icmp_msg_type_request : NATURAL := 8; -- 8 = echo request + CONSTANT c_icmp_msg_type_reply : NATURAL := 0; -- 8 = echo reply (ping) + CONSTANT c_icmp_checksum : NATURAL := 0; -- init value + + -- useful field values + CONSTANT c_icmp_code : NATURAL := 0; -- default + CONSTANT c_icmp_id : NATURAL := 3; -- arbitrary value + CONSTANT c_icmp_sequence : NATURAL := 4; -- arbitrary value + + TYPE t_icmp_header IS RECORD + msg_type : STD_LOGIC_VECTOR(c_icmp_msg_type_w-1 DOWNTO 0); -- 1 octet + code : STD_LOGIC_VECTOR(c_icmp_code_w-1 DOWNTO 0); -- 1 octet + checksum : STD_LOGIC_VECTOR(c_icmp_checksum_w-1 DOWNTO 0); -- 2 octet + id : STD_LOGIC_VECTOR(c_icmp_id_w-1 DOWNTO 0); -- 2 octet + sequence : STD_LOGIC_VECTOR(c_icmp_sequence_w-1 DOWNTO 0); -- 2 octet + END RECORD; + + -- field valid word indices + CONSTANT c_icmp_lo_wi : NATURAL := 9; -- first word index + CONSTANT c_icmp_msg_type_wi : NATURAL := 9; + CONSTANT c_icmp_code_wi : NATURAL := 9; + CONSTANT c_icmp_checksum_wi : NATURAL := 9; + CONSTANT c_icmp_id_wi : NATURAL := 10; + CONSTANT c_icmp_sequence_wi : NATURAL := 10; + CONSTANT c_icmp_hi_wi : NATURAL := 10; -- last word index + CONSTANT c_icmp_nof_words : NATURAL := c_icmp_hi_wi - c_icmp_lo_wi + 1; -- = c_icmp_header_len / c_word_sz + + + ------------------------------------------------------------------------------ + -- UDP Packet + -- + -- 0 15 16 31 wi + -- |----------------------------------------------------------------------| + -- | Source Port | Destination Port | 9 + -- |----------------------------------------------------------------------| + -- | Total Length | Checksum | 10 + -- |----------------------------------------------------------------------| + -- | | + -- | UDP Payload | + -- | | + -- |----------------------------------------------------------- // -------| + -- + + -- field widths in bits '_w' or in bytes '_len' + CONSTANT c_udp_port_len : NATURAL := 2; + CONSTANT c_udp_port_w : NATURAL := c_udp_port_len*c_8; + CONSTANT c_udp_total_length_len : NATURAL := 2; + CONSTANT c_udp_total_length_w : NATURAL := c_udp_total_length_len*c_8; + CONSTANT c_udp_checksum_len : NATURAL := 2; + CONSTANT c_udp_checksum_w : NATURAL := c_udp_checksum_len*c_8; + + -- [0:15] [16:31] + CONSTANT c_udp_header_len : NATURAL := c_udp_port_len + c_udp_port_len + + c_udp_total_length_len + c_udp_checksum_len; -- 8 + + -- default field values + CONSTANT c_udp_total_length : NATURAL := 8; -- >= 8, nof bytes in entire datagram including header and data + CONSTANT c_udp_checksum : NATURAL := 0; -- init value + + -- useful field values + CONSTANT c_udp_port_dhcp_in : NATURAL := 68; -- DHCP to client = Dynamic Host Configuration Protocol (for IP address assignment) + CONSTANT c_udp_port_dhcp_out : NATURAL := 67; -- DHCP to server + CONSTANT c_udp_port_slv : STD_LOGIC_VECTOR(c_udp_port_w-1 DOWNTO 0) := (OTHERS=>'X'); -- UDP port slv RANGE + + TYPE t_udp_header IS RECORD + src_port : STD_LOGIC_VECTOR(c_udp_port_w-1 DOWNTO 0); -- 2 octet + dst_port : STD_LOGIC_VECTOR(c_udp_port_w-1 DOWNTO 0); -- 2 octet + total_length : STD_LOGIC_VECTOR(c_udp_total_length_w-1 DOWNTO 0); -- 2 octet + checksum : STD_LOGIC_VECTOR(c_udp_checksum_w-1 DOWNTO 0); -- 2 octet + END RECORD; + + -- field valid word indices + CONSTANT c_udp_lo_wi : NATURAL := 9; -- first word index + CONSTANT c_udp_src_port_wi : NATURAL := 9; + CONSTANT c_udp_dst_port_wi : NATURAL := 9; + CONSTANT c_udp_total_length_wi : NATURAL := 10; + CONSTANT c_udp_checksum_wi : NATURAL := 10; + CONSTANT c_udp_hi_wi : NATURAL := 10; -- last word index + CONSTANT c_udp_nof_words : NATURAL := c_udp_hi_wi - c_udp_lo_wi + 1; -- = c_udp_header_len / c_word_sz + + + ------------------------------------------------------------------------------ + -- Total Ethernet Header + ------------------------------------------------------------------------------ + + CONSTANT c_eth_total_arp_len : NATURAL := c_eth_header_len + c_arp_data_len; -- = 14 + 28 = 42 + CONSTANT c_eth_total_icmp_len : NATURAL := c_eth_header_len + c_ip_header_len + c_icmp_header_len; -- = 14 + 20 + 8 = 42 + CONSTANT c_eth_total_udp_len : NATURAL := c_eth_header_len + c_ip_header_len + c_udp_header_len; -- = 14 + 20 + 8 = 42 + -- Hence it appears that all relevant packets have the same total header length + CONSTANT c_eth_total_header_len : NATURAL := 42; + CONSTANT c_eth_total_header_nof_words : NATURAL := (c_eth_word_align_len + c_eth_total_header_len)/4; -- = 44 / c_word_sz = 11 + + -- Aggregate all supported Ethernet headers into one record + TYPE t_eth_total_header IS RECORD + eth : t_eth_header; + arp : t_arp_packet; + ip : t_ip_header; + icmp : t_icmp_header; + udp : t_udp_header; + END RECORD; + + -- Ethernet frame word indices + TYPE t_eth_total_header_arr IS ARRAY(0 TO c_eth_total_header_nof_words-1) OF STD_LOGIC_VECTOR(c_32-1 DOWNTO 0); + + -- Combinatorial map of the total header on to the Ethernet header records + -- Type casting an array to a record is not possible, so therefore we need these functions + -- Define also the inverse mapping functions as overloaded functions + FUNCTION func_eth_map_eth_header( total_header : t_eth_total_header_arr) RETURN t_eth_header; + FUNCTION func_eth_map_eth_header( header : t_eth_header) RETURN t_eth_total_header_arr; + FUNCTION func_eth_map_ip_header( total_header : t_eth_total_header_arr) RETURN t_ip_header; + FUNCTION func_eth_map_ip_header( header : t_ip_header) RETURN t_eth_total_header_arr; + FUNCTION func_eth_map_arp_packet( total_header : t_eth_total_header_arr) RETURN t_arp_packet; + FUNCTION func_eth_map_arp_packet( header : t_arp_packet) RETURN t_eth_total_header_arr; + FUNCTION func_eth_map_icmp_header(total_header : t_eth_total_header_arr) RETURN t_icmp_header; + FUNCTION func_eth_map_icmp_header(header : t_icmp_header) RETURN t_eth_total_header_arr; + FUNCTION func_eth_map_udp_header( total_header : t_eth_total_header_arr) RETURN t_udp_header; + FUNCTION func_eth_map_udp_header( header : t_udp_header) RETURN t_eth_total_header_arr; + + -- Construct the total header array from the individual header records + FUNCTION func_eth_arp_total_header( eth : t_eth_header; arp : t_arp_packet) RETURN t_eth_total_header_arr; + FUNCTION func_eth_icmp_total_header(eth : t_eth_header; ip : t_ip_header; icmp : t_icmp_header) RETURN t_eth_total_header_arr; + FUNCTION func_eth_udp_total_header( eth : t_eth_header; ip : t_ip_header; udp : t_udp_header) RETURN t_eth_total_header_arr; + + -- Construct the response headers + FUNCTION func_eth_response_header( eth : t_eth_total_header_arr; mac_addr : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0)) RETURN t_eth_total_header_arr; + FUNCTION func_eth_arp_response_header( arp : t_eth_total_header_arr; mac_addr : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0); + ip_addr : STD_LOGIC_VECTOR(c_ip_addr_w-1 DOWNTO 0)) + RETURN t_eth_total_header_arr; + FUNCTION func_eth_ip_response_header( ip : t_eth_total_header_arr; mac_addr : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0)) RETURN t_eth_total_header_arr; + FUNCTION func_eth_icmp_response_header(icmp : t_eth_total_header_arr; mac_addr : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0)) RETURN t_eth_total_header_arr; + FUNCTION func_eth_udp_response_header( udp : t_eth_total_header_arr; mac_addr : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0)) RETURN t_eth_total_header_arr; + +END eth_layers_pkg; + + +PACKAGE BODY eth_layers_pkg IS + + ------------------------------------------------------------------------------ + -- Total Ethernet Header + ------------------------------------------------------------------------------ + + -- * Map the 11 words from the Ethernet header to the Ethernet field records + -- * Define also the inverse mapping functions, leave unused v_total words 'X' + -- * Outside this package use the c_*_lo_wi and c_*_hi_wi constants to select + -- the used words from the total header word array. + + FUNCTION func_eth_map_eth_header(total_header : t_eth_total_header_arr) RETURN t_eth_header IS + VARIABLE v_hdr : t_eth_header; + BEGIN + v_hdr.word_align := total_header(0)(31 DOWNTO 16); + v_hdr.dst_mac(47 DOWNTO 32) := total_header(0)(15 DOWNTO 0); + v_hdr.dst_mac(31 DOWNTO 0) := total_header(1); + v_hdr.src_mac(47 DOWNTO 16) := total_header(2); + v_hdr.src_mac(15 DOWNTO 0) := total_header(3)(31 DOWNTO 16); + v_hdr.eth_type := total_header(3)(15 DOWNTO 0); + RETURN v_hdr; + END; + + FUNCTION func_eth_map_eth_header(header : t_eth_header) RETURN t_eth_total_header_arr IS + VARIABLE v_total : t_eth_total_header_arr; + BEGIN + v_total(0)(31 DOWNTO 16) := header.word_align; + v_total(0)(15 DOWNTO 0) := header.dst_mac(47 DOWNTO 32); + v_total(1) := header.dst_mac(31 DOWNTO 0); + v_total(2) := header.src_mac(47 DOWNTO 16); + v_total(3)(31 DOWNTO 16) := header.src_mac(15 DOWNTO 0); + v_total(3)(15 DOWNTO 0) := header.eth_type; + RETURN v_total; + END; + + + FUNCTION func_eth_map_ip_header(total_header : t_eth_total_header_arr) RETURN t_ip_header IS + VARIABLE v_hdr : t_ip_header; + BEGIN + v_hdr.version := total_header(4)(31 DOWNTO 28); + v_hdr.header_length := total_header(4)(27 DOWNTO 24); + v_hdr.services := total_header(4)(23 DOWNTO 16); + v_hdr.total_length := total_header(4)(15 DOWNTO 0); + v_hdr.identification := total_header(5)(31 DOWNTO 16); + v_hdr.flags := total_header(5)(15 DOWNTO 13); + v_hdr.fragment_offset := total_header(5)(12 DOWNTO 0); + v_hdr.time_to_live := total_header(6)(31 DOWNTO 24); + v_hdr.protocol := total_header(6)(23 DOWNTO 16); + v_hdr.header_checksum := total_header(6)(15 DOWNTO 0); + v_hdr.src_ip_addr := total_header(7); + v_hdr.dst_ip_addr := total_header(8); + RETURN v_hdr; + END; + + FUNCTION func_eth_map_ip_header(header : t_ip_header) RETURN t_eth_total_header_arr IS + VARIABLE v_total : t_eth_total_header_arr; + BEGIN + v_total(4)(31 DOWNTO 28) := header.version; + v_total(4)(27 DOWNTO 24) := header.header_length; + v_total(4)(23 DOWNTO 16) := header.services; + v_total(4)(15 DOWNTO 0) := header.total_length; + v_total(5)(31 DOWNTO 16) := header.identification; + v_total(5)(15 DOWNTO 13) := header.flags; + v_total(5)(12 DOWNTO 0) := header.fragment_offset; + v_total(6)(31 DOWNTO 24) := header.time_to_live; + v_total(6)(23 DOWNTO 16) := header.protocol; + v_total(6)(15 DOWNTO 0) := header.header_checksum; + v_total(7) := header.src_ip_addr; + v_total(8) := header.dst_ip_addr; + RETURN v_total; + END; + + + FUNCTION func_eth_map_arp_packet(total_header : t_eth_total_header_arr) RETURN t_arp_packet IS + VARIABLE v_arp : t_arp_packet; + BEGIN + v_arp.htype := total_header(4)(31 DOWNTO 16); + v_arp.ptype := total_header(4)(15 DOWNTO 0); + v_arp.hlen := total_header(5)(31 DOWNTO 24); + v_arp.plen := total_header(5)(23 DOWNTO 16); + v_arp.oper := total_header(5)(15 DOWNTO 0); + v_arp.sha(47 DOWNTO 16) := total_header(6); + v_arp.sha(15 DOWNTO 0) := total_header(7)(31 DOWNTO 16); + v_arp.spa(31 DOWNTO 16) := total_header(7)(15 DOWNTO 0); + v_arp.spa(15 DOWNTO 0) := total_header(8)(31 DOWNTO 16); + v_arp.tha(47 DOWNTO 32) := total_header(8)(15 DOWNTO 0); + v_arp.tha(31 DOWNTO 0) := total_header(9); + v_arp.tpa := total_header(10); + RETURN v_arp; + END; + + FUNCTION func_eth_map_arp_packet(header : t_arp_packet) RETURN t_eth_total_header_arr IS + VARIABLE v_total : t_eth_total_header_arr; + BEGIN + v_total(4)(31 DOWNTO 16) := header.htype; + v_total(4)(15 DOWNTO 0) := header.ptype; + v_total(5)(31 DOWNTO 24) := header.hlen; + v_total(5)(23 DOWNTO 16) := header.plen; + v_total(5)(15 DOWNTO 0) := header.oper; + v_total(6) := header.sha(47 DOWNTO 16); + v_total(7)(31 DOWNTO 16) := header.sha(15 DOWNTO 0); + v_total(7)(15 DOWNTO 0) := header.spa(31 DOWNTO 16); + v_total(8)(31 DOWNTO 16) := header.spa(15 DOWNTO 0); + v_total(8)(15 DOWNTO 0) := header.tha(47 DOWNTO 32); + v_total(9) := header.tha(31 DOWNTO 0); + v_total(10) := header.tpa; + RETURN v_total; + END; + + + FUNCTION func_eth_map_icmp_header(total_header : t_eth_total_header_arr) RETURN t_icmp_header IS + VARIABLE v_hdr : t_icmp_header; + BEGIN + v_hdr.msg_type := total_header(9)(31 DOWNTO 24); + v_hdr.code := total_header(9)(23 DOWNTO 16); + v_hdr.checksum := total_header(9)(15 DOWNTO 0); + v_hdr.id := total_header(10)(31 DOWNTO 16); + v_hdr.sequence := total_header(10)(15 DOWNTO 0); + RETURN v_hdr; + END; + + FUNCTION func_eth_map_icmp_header(header : t_icmp_header) RETURN t_eth_total_header_arr IS + VARIABLE v_total : t_eth_total_header_arr; + BEGIN + v_total(9)(31 DOWNTO 24) := header.msg_type; + v_total(9)(23 DOWNTO 16) := header.code; + v_total(9)(15 DOWNTO 0) := header.checksum; + v_total(10)(31 DOWNTO 16) := header.id; + v_total(10)(15 DOWNTO 0) := header.sequence; + RETURN v_total; + END; + + + FUNCTION func_eth_map_udp_header(total_header : t_eth_total_header_arr) RETURN t_udp_header IS + VARIABLE v_hdr : t_udp_header; + BEGIN + v_hdr.src_port := total_header(9)(31 DOWNTO 16); + v_hdr.dst_port := total_header(9)(15 DOWNTO 0); + v_hdr.total_length := total_header(10)(31 DOWNTO 16); + v_hdr.checksum := total_header(10)(15 DOWNTO 0); + RETURN v_hdr; + END; + + + FUNCTION func_eth_map_udp_header(header : t_udp_header) RETURN t_eth_total_header_arr IS + VARIABLE v_total : t_eth_total_header_arr; + BEGIN + v_total(9)(31 DOWNTO 16) := header.src_port; + v_total(9)(15 DOWNTO 0) := header.dst_port; + v_total(10)(31 DOWNTO 16) := header.total_length; + v_total(10)(15 DOWNTO 0) := header.checksum; + RETURN v_total; + END; + + + -- Construct the total header array from the individual header records + FUNCTION func_eth_arp_total_header(eth : t_eth_header; arp : t_arp_packet) RETURN t_eth_total_header_arr IS + CONSTANT c_eth : t_eth_total_header_arr := func_eth_map_eth_header(eth); + CONSTANT c_arp : t_eth_total_header_arr := func_eth_map_arp_packet(arp); + VARIABLE v_total : t_eth_total_header_arr; + BEGIN + FOR I IN c_eth_lo_wi TO c_eth_hi_wi LOOP + v_total(I) := c_eth(I); + END LOOP; + FOR I IN c_arp_lo_wi TO c_arp_hi_wi LOOP + v_total(I) := c_arp(I); + END LOOP; + RETURN v_total; + END; + + FUNCTION func_eth_icmp_total_header(eth : t_eth_header; ip : t_ip_header; icmp : t_icmp_header) RETURN t_eth_total_header_arr IS + CONSTANT c_eth : t_eth_total_header_arr := func_eth_map_eth_header(eth); + CONSTANT c_ip : t_eth_total_header_arr := func_eth_map_ip_header(ip); + CONSTANT c_icmp : t_eth_total_header_arr := func_eth_map_icmp_header(icmp); + VARIABLE v_total : t_eth_total_header_arr; + BEGIN + FOR I IN c_eth_lo_wi TO c_eth_hi_wi LOOP + v_total(I) := c_eth(I); + END LOOP; + FOR I IN c_ip_lo_wi TO c_ip_hi_wi LOOP + v_total(I) := c_ip(I); + END LOOP; + FOR I IN c_icmp_lo_wi TO c_icmp_hi_wi LOOP + v_total(I) := c_icmp(I); + END LOOP; + RETURN v_total; + END; + + FUNCTION func_eth_udp_total_header( eth : t_eth_header; ip : t_ip_header; udp : t_udp_header) RETURN t_eth_total_header_arr IS + CONSTANT c_eth : t_eth_total_header_arr := func_eth_map_eth_header(eth); + CONSTANT c_ip : t_eth_total_header_arr := func_eth_map_ip_header(ip); + CONSTANT c_udp : t_eth_total_header_arr := func_eth_map_udp_header(udp); + VARIABLE v_total : t_eth_total_header_arr; + BEGIN + FOR I IN c_eth_lo_wi TO c_eth_hi_wi LOOP + v_total(I) := c_eth(I); + END LOOP; + FOR I IN c_ip_lo_wi TO c_ip_hi_wi LOOP + v_total(I) := c_ip(I); + END LOOP; + FOR I IN c_udp_lo_wi TO c_udp_hi_wi LOOP + v_total(I) := c_udp(I); + END LOOP; + RETURN v_total; + END; + + -- Construct the response headers + FUNCTION func_eth_response_header(eth : t_eth_total_header_arr; + mac_addr : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0)) RETURN t_eth_total_header_arr IS + VARIABLE v_response : t_eth_total_header_arr; + BEGIN + -- Default + v_response := eth; + -- ETH + -- . use input src mac for dst mac + v_response(0)(15 DOWNTO 0) := eth(2)(31 DOWNTO 16); + v_response(1) := eth(2)(15 DOWNTO 0) & eth(3)(31 DOWNTO 16); + -- . force eth src mac to this node mac address (because the input dst_mac can be via eth broadcast mac) + v_response(2) := mac_addr(47 DOWNTO 16); + v_response(3)(31 DOWNTO 16) := mac_addr(15 DOWNTO 0); + RETURN v_response; + END; + + FUNCTION func_eth_arp_response_header(arp : t_eth_total_header_arr; + mac_addr : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0); + ip_addr : STD_LOGIC_VECTOR(c_ip_addr_w-1 DOWNTO 0)) RETURN t_eth_total_header_arr IS + VARIABLE v_response : t_eth_total_header_arr; + BEGIN + -- ETH + v_response := func_eth_response_header(arp, mac_addr); + -- ARP + -- . force operation arp reply + v_response(5)(15 DOWNTO 0) := TO_UVEC(c_arp_oper_reply, 16); + -- . force sha to this node mac address + v_response(6) := mac_addr(47 DOWNTO 16); + v_response(7)(31 DOWNTO 16) := mac_addr(15 DOWNTO 0); + -- . force spa to this node ip address + v_response(7)(15 DOWNTO 0) := ip_addr(31 DOWNTO 16); + v_response(8)(31 DOWNTO 16) := ip_addr(15 DOWNTO 0); + -- . use input sha for tha + v_response(8)(15 DOWNTO 0) := arp(6)(31 DOWNTO 16); + v_response(9) := arp(6)(15 DOWNTO 0) & arp(7)(31 DOWNTO 16); + -- . use input spa for tpa + v_response(10) := arp(7)(15 DOWNTO 0) & arp(8)(31 DOWNTO 16); + RETURN v_response; + END; + + FUNCTION func_eth_ip_response_header(ip : t_eth_total_header_arr; + mac_addr : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0)) RETURN t_eth_total_header_arr IS + VARIABLE v_response : t_eth_total_header_arr; + BEGIN + -- ETH + v_response := func_eth_response_header(ip, mac_addr); + -- IP + -- . force ip header checksum to 0 + v_response(6)(15 DOWNTO 0) := TO_UVEC(0, 16); + -- . swap ip dst_addr and ip src_addr + v_response(7) := ip(8); + v_response(8) := ip(7); + RETURN v_response; + END; + + FUNCTION func_eth_icmp_response_header(icmp : t_eth_total_header_arr; + mac_addr : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0)) RETURN t_eth_total_header_arr IS + VARIABLE v_response : t_eth_total_header_arr; + BEGIN + -- ETH, IP + v_response := func_eth_ip_response_header(icmp, mac_addr); + -- ICMP : force type to icmp reply + v_response(9)(31 DOWNTO 24) := TO_UVEC(c_icmp_msg_type_reply, 8); + -- ICMP : force icmp checksum to 0 + v_response(9)(15 DOWNTO 0) := TO_UVEC(0, 16); + RETURN v_response; + END; + + FUNCTION func_eth_udp_response_header(udp : t_eth_total_header_arr; + mac_addr : STD_LOGIC_VECTOR(c_eth_mac_addr_w-1 DOWNTO 0)) RETURN t_eth_total_header_arr IS + VARIABLE v_response : t_eth_total_header_arr; + BEGIN + -- ETH, IP + v_response := func_eth_ip_response_header(udp, mac_addr); + -- UDP : swap udp dst port and udp src port + v_response(9) := udp(9)(15 DOWNTO 0) & udp(9)(31 DOWNTO 16); + -- UDP : force udp checksum to 0 + v_response(10)(15 DOWNTO 0) := TO_UVEC(0, 16); + RETURN v_response; + END; + +END eth_layers_pkg;