From aa8e78a0b942585e520929b6fc1ecdcb7d052b38 Mon Sep 17 00:00:00 2001 From: Eric Kooistra <kooistra@astron.nl> Date: Mon, 18 Sep 2023 15:09:26 +0200 Subject: [PATCH] Use separate sdp_bdo_pkg.vhd for BDO multiple destinations. --- applications/lofar2/libraries/sdp/hdllib.cfg | 1 + .../libraries/sdp/src/vhdl/sdp_bdo_pkg.vhd | 164 ++++++++++++++++++ .../lofar2/libraries/sdp/src/vhdl/sdp_pkg.vhd | 62 ------- 3 files changed, 165 insertions(+), 62 deletions(-) create mode 100644 applications/lofar2/libraries/sdp/src/vhdl/sdp_bdo_pkg.vhd diff --git a/applications/lofar2/libraries/sdp/hdllib.cfg b/applications/lofar2/libraries/sdp/hdllib.cfg index 25f8a6fd36..e092129598 100644 --- a/applications/lofar2/libraries/sdp/hdllib.cfg +++ b/applications/lofar2/libraries/sdp/hdllib.cfg @@ -14,6 +14,7 @@ synth_files = src/vhdl/sdp_beamformer_remote.vhd src/vhdl/sdp_info_reg.vhd src/vhdl/sdp_info.vhd + src/vhdl/sdp_bdo_pkg.vhd src/vhdl/sdp_bdo_destinations_reg.vhd src/vhdl/sdp_beamformer_output.vhd src/vhdl/sdp_statistics_offload.vhd diff --git a/applications/lofar2/libraries/sdp/src/vhdl/sdp_bdo_pkg.vhd b/applications/lofar2/libraries/sdp/src/vhdl/sdp_bdo_pkg.vhd new file mode 100644 index 0000000000..3142ce9dd4 --- /dev/null +++ b/applications/lofar2/libraries/sdp/src/vhdl/sdp_bdo_pkg.vhd @@ -0,0 +1,164 @@ +------------------------------------------------------------------------------- +-- +-- Copyright 2023 +-- ASTRON (Netherlands Institute for Radio Astronomy) <http://www.astron.nl/> +-- P.O.Box 2, 7990 AA Dwingeloo, The Netherlands +-- +-- Licensed under the Apache License, Version 2.0 (the "License"); +-- you may not use this file except in compliance with the License. +-- You may obtain a copy of the License at +-- +-- http://www.apache.org/licenses/LICENSE-2.0 +-- +-- Unless required by applicable law or agreed to in writing, software +-- distributed under the License is distributed on an "AS IS" BASIS, +-- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +-- See the License for the specific language governing permissions and +-- limitations under the License. +-- +------------------------------------------------------------------------------- + +------------------------------------------------------------------------------- +-- +-- Author: E. Kooistra +-- Purpose: +-- . This package contains sdp beamlet data output (BDO) specific constants. +-- Description: See [1] +-- References: +-- . [1] https://support.astron.nl/confluence/display/L2M/L4+SDPFW+Decision%3A+Multiple+beamlet+output+destinations +------------------------------------------------------------------------------- +library IEEE, common_lib; +use IEEE.std_logic_1164.all; +use common_lib.common_pkg.all; +use work.sdp_pkg.all; + +package sdp_bdo_pkg is + -- Beamlet data output (BDO) for multiple destinations + constant c_sdp_bdo_nof_destinations_max : natural := 16; + constant c_sdp_bdo_reorder_nof_blocks_max : natural := largest(16, c_sdp_cep_nof_blocks_per_packet); + + constant c_sdp_bdo_destinations_info_nof_hdr_fields : natural := c_sdp_bdo_nof_destinations_max * 3 + 3; -- = 52 fields + + type t_sdp_bdo_destinations_info is record + eth_destination_mac_arr : t_slv_48_arr(c_sdp_bdo_nof_destinations_max - 1 downto 0); + ip_destination_address_arr : t_slv_32_arr(c_sdp_bdo_nof_destinations_max - 1 downto 0); + udp_destination_port_arr : t_slv_16_arr(c_sdp_bdo_nof_destinations_max - 1 downto 0); + nof_destinations : natural; + nof_destinations_act : natural; + nof_blocks_per_packet_act : natural; + end record; + + constant t_sdp_bdo_destinations_info_rst : t_sdp_bdo_destinations_info := + ( (others => (others => '0')), + (others => (others => '0')), + (others => (others => '0')), 0, 0, 0 ); + + function func_sdp_bdo_parse_nof_destinations(nof_destinations : natural) return natural; + function func_sdp_bdo_nof_blocks_per_packet_look_up_table return t_natural_arr; + function func_sdp_nof_beamlets_per_block_look_up_table return t_natural_arr; + function func_sdp_nof_beamlets_per_block_look_up_matrix return t_natural_matrix; +end package sdp_bdo_pkg; + +package body sdp_bdo_pkg is + function func_sdp_bdo_parse_nof_destinations(nof_destinations : natural) return natural is + begin + if nof_destinations = 0 then + return 1; + elsif nof_destinations > c_sdp_bdo_nof_destinations_max then + return c_sdp_bdo_nof_destinations_max; + else + return nof_destinations; + end if; + end func_sdp_bdo_parse_nof_destinations; + + function func_sdp_bdo_nof_blocks_per_packet_look_up_table return t_natural_arr is + variable v_arr : t_natural_arr(1 to c_sdp_bdo_nof_destinations_max); + begin + -- Determine nof_blocks_per_packet as function of number of destinations D. Use + -- look up table to precalculate the values as constants. + -- . Use same number of nof_blocks_per_packet for each destination. + -- . In total there are maximum c_sdp_bdo_reorder_nof_blocks_max = 16 blocks to + -- distribute over D destinations, so floor(16 / D) per destination yields: + -- D = 1:16 --> 16, 8, 5, 4, 3, 2, 2, 2, 1, 1, 1, 1, 1, 1, 1, 1 + for D in 1 to c_sdp_bdo_nof_destinations_max loop + v_arr(D) := c_sdp_bdo_reorder_nof_blocks_max / D; + end loop; + -- . With 1 destination c_sdp_cep_nof_blocks_per_packet = 4 can fit in a jumbo frame. + -- . With D destinations D * c_sdp_cep_nof_blocks_per_packet can fit in a jumbo frame, + -- because the number of beamlets per destination reduces by D. + -- . Taking smallest yields the actual maximum number of blocks per packet + -- per destination, as function of number of destinations D: + -- D = 1:16 --> 4, 8, 5, 4, 3, 2, 2, 2, 1, 1, 1, 1, 1, 1, 1, 1 + for D in 1 to c_sdp_bdo_nof_destinations_max loop + v_arr(D) := smallest(v_arr(D), D * c_sdp_cep_nof_blocks_per_packet); + end loop; + return v_arr; + end func_sdp_bdo_nof_blocks_per_packet_look_up_table; + + function func_sdp_nof_beamlets_per_block_look_up_table return t_natural_arr is + variable v_arr : t_natural_arr(1 to c_sdp_bdo_nof_destinations_max); + begin + -- Determine nof_beamlets_per_block as function of number of destinations D. Use + -- look up table to precalculate the values as constants. + -- . In total there are c_sdp_S_sub_bf = 488 dual polarization beamlets to + -- distribute over D destinations, so ceil(488 / D) per destination yields: + -- D = 1:16 --> 488, 244, 163, 122, 98, 82, 70, 61, 55, 49, 45, 41, 38, 35, 33, 31 + for D in 1 to c_sdp_bdo_nof_destinations_max loop + v_arr(D) := ceil_div(c_sdp_S_sub_bf, D); + end loop; + return v_arr; + end func_sdp_nof_beamlets_per_block_look_up_table; + + function func_sdp_nof_beamlets_per_block_look_up_matrix return t_natural_matrix is + constant c_arr : t_natural_arr(1 to c_sdp_bdo_nof_destinations_max) := + func_sdp_nof_beamlets_per_block_look_up_table; + variable v_mat : t_natural_matrix(1 to c_sdp_bdo_nof_destinations_max, + 1 to c_sdp_bdo_nof_destinations_max); + variable v_hi : natural; + variable v_lo : natural; + begin + -- Determine nof_beamlets_per_block as function of number of destinations DN and + -- destination index DI. Use look up table to precalculate the values as constants. + -- + -- * vertical : nof_destinations DN + -- * horizontal : destination index DI + -- + -- DI: 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 + -- DN: + -- 1 488, 0, .................................................................0 + -- 2 244, 244, 0 . + -- 3 163, 163, 162, 0 . + -- 4 122, 122, 122, 122, 0 . + -- 5 98, 98, 98, 97, 97, 0 . + -- 6 82, 82, 81, 81, 81, 81, 0 . + -- 7 70, 70, 70, 70, 70, 69, 69, 0 . + -- 8 61, 61, 61, 61, 61, 61, 61, 61, 0 . + -- 9 55, 55, 57, 57, 57, 57, 57, 57, 57, 0 . + -- 10 49, 49, 49, 49, 49, 49, 49, 49, 48, 48, 0 . + -- 11 45, 45, 45, 45, 44, 44, 44, 44, 44, 44, 44, 0 . + -- 12 41, 41, 41, 41, 41, 41, 41, 41, 40, 40, 40, 40, 0 . + -- 13 38, 38, 38, 38, 38, 38, 38, 37, 37, 37, 37, 37, 37, 0 . + -- 14 35, 35, 35, 35, 35, 35, 35, 35, 35, 35, 35, 35, 34, 34, 0 . + -- 15 33, 33, 33, 33, 33, 33, 33, 33, 32, 32, 32, 32, 32, 32, 32, 0 + -- 16 31, 31, 31, 31, 31, 31, 31, 31, 30, 30, 30, 30, 30, 30, 30, 30 + for DN in 1 to c_sdp_bdo_nof_destinations_max loop + v_hi := c_arr(DN); + v_lo := v_hi - 1; + for DI in 1 to c_sdp_bdo_nof_destinations_max loop + -- Default initialize to zero for unused elements + v_mat(DN, DI) := 0; + -- Determine number of destinations in DN with v_hi value and with + -- v_lo value, to distribute in total c_sdp_S_sub_bf beamlets to DN + -- destinations. + if DI * v_hi + (DN - DI) * v_lo <= c_sdp_S_sub_bf then + -- Use v_hi for first destinations DI + v_mat(DN, DI) := v_hi; + else + -- Use v_lo for remaining destinations, if there are any remaining + v_mat(DN, DI) := v_lo; + end if; + end loop; + end loop; + return v_mat; + end func_sdp_nof_beamlets_per_block_look_up_matrix; +end sdp_bdo_pkg; diff --git a/applications/lofar2/libraries/sdp/src/vhdl/sdp_pkg.vhd b/applications/lofar2/libraries/sdp/src/vhdl/sdp_pkg.vhd index 3d44aca5ca..f4c052fb91 100644 --- a/applications/lofar2/libraries/sdp/src/vhdl/sdp_pkg.vhd +++ b/applications/lofar2/libraries/sdp/src/vhdl/sdp_pkg.vhd @@ -113,8 +113,6 @@ package sdp_pkg is constant c_sdp_W_local_oscillator_fraction : natural := 15; -- = p in s(w, p) constant c_sdp_W_local_oscillator_magnitude : natural := c_sdp_W_local_oscillator - c_sdp_W_local_oscillator_fraction - 1; -- = 0 constant c_sdp_N_ring_nof_mac10g : natural := 3; -- for sdp_station_xsub_ring design. - constant c_sdp_N_bdo_nof_destinations_max : natural := 16; - constant c_sdp_N_bdo_reorder_nof_blocks_max : natural := 16; -- Derived constants constant c_sdp_FS_adc : natural := 2**(c_sdp_W_adc - 1); -- full scale FS corresponds to amplitude 1.0, will just cause clipping of +FS to +FS-1 @@ -426,42 +424,6 @@ package sdp_pkg is constant c_sdp_cep_payload_nof_longwords : natural := c_sdp_cep_nof_beamlets_per_packet / c_sdp_nof_beamlets_per_longword; -- = 976 constant c_sdp_cep_packet_nof_longwords : natural := ceil_div(c_sdp_cep_header_len, c_longword_sz) + c_sdp_cep_payload_nof_longwords; -- without tail CRC, the CRC is applied by 10GbE MAC - -- Beamlet data output (BDO) multiple destinations - constant c_sdp_bdo_destinations_info_nof_hdr_fields : natural := c_sdp_N_bdo_nof_destinations_max * 3 + 4; -- = 52 fields - - type t_sdp_bdo_destinations_info is record - eth_destination_mac_arr : t_slv_48_arr(c_sdp_N_bdo_nof_destinations_max - 1 downto 0); - ip_destination_address_arr : t_slv_32_arr(c_sdp_N_bdo_nof_destinations_max - 1 downto 0); - udp_destination_port_arr : t_slv_16_arr(c_sdp_N_bdo_nof_destinations_max - 1 downto 0); - nof_destinations : natural; - nof_destinations_act : natural; - nof_blocks_per_packet : natural; - nof_blocks_per_packet_act : natural; - end record; - - constant t_sdp_bdo_destinations_info_rst : t_sdp_bdo_destinations_info := - ( (others => (others => '0')), - (others => (others => '0')), - (others => (others => '0')), 0, 0, 0, 0 ); - - -- Determine maximum nof_blocks_per_packet as function of nof_destinations: - -- . Use same number of nof_blocks_per_packet for each destination. - -- . In total there are maximum c_sdp_N_bdo_reorder_nof_blocks_max = 16 blocks to - -- distribute over N destinations, so floor(16 / N) per destination yields: - -- N = 1:16 --> 16, 8, 5, 4, 3, 2, 2, 2, 1, 1, 1, 1, 1, 1, 1, 1 - -- . With 1 destination c_sdp_cep_nof_blocks_per_packet = 4 can fit in a jumbo frame. - -- . With N destinations N * c_sdp_cep_nof_blocks_per_packet can fit in a jumbo frame, - -- because the number of beamlets per destination reduces by N. - -- N = 1:16 --> 4 * [1, 2, 3, 4, 5 ,6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16] - -- . Taking smallest yields the actual maximum number of blocks per packet - -- per destination, as function of nof_destinations N: - -- N = 1:16 --> 4, 8, 5, 4, 3, 2, 2, 2, 1, 1, 1, 1, 1, 1, 1, 1 - constant c_sdp_nof_blocks_per_packet_max_per_destination_arr : t_natural_arr(1 to c_sdp_N_bdo_nof_destinations_max) := - (4, 8, 5, 4, 3, 2, 2, 2, 1, 1, 1, 1, 1, 1, 1, 1); - - function func_sdp_parse_nof_destinations(nof_destinations : natural) return natural; - function func_sdp_parse_nof_blocks_per_packet(nof_blocks_per_packet, nof_destinations : natural) return natural; - -- CEP packet header constant c_sdp_cep_nof_hdr_fields : natural := 3 + 12 + 4 + 4 + 9 + 6 + 1; -- = 39 fields -- c_sdp_cep_header_len / c_longword_sz = 74 / 8 = 9.25 64b words = 592b @@ -1044,28 +1006,4 @@ package body sdp_pkg is end loop; return v_info; end func_sdp_step_crosslets_info; - - function func_sdp_parse_nof_destinations(nof_destinations : natural) return natural is - begin - if nof_destinations = 0 then - return 1; - elsif nof_destinations > c_sdp_N_bdo_nof_destinations_max then - return c_sdp_N_bdo_nof_destinations_max; - else - return nof_destinations; - end if; - end func_sdp_parse_nof_destinations; - - function func_sdp_parse_nof_blocks_per_packet(nof_blocks_per_packet, nof_destinations : natural) return natural is - constant c_nof_blocks_per_packet_max : natural := c_sdp_nof_blocks_per_packet_max_per_destination_arr(nof_destinations); - begin - if nof_blocks_per_packet = 0 then - return 1; - elsif nof_blocks_per_packet > c_nof_blocks_per_packet_max then - return c_nof_blocks_per_packet_max; - else - return nof_blocks_per_packet; - end if; - end func_sdp_parse_nof_blocks_per_packet; - end sdp_pkg; -- GitLab