From 460c70d9d48becbd98e45fd080adf85be578f31f Mon Sep 17 00:00:00 2001 From: auphelia Date: Wed, 15 May 2024 17:31:05 +0100 Subject: [PATCH] [RTL Thresholding] UPdate code generation to allow for independent input and threshold values --- .../hdl/thresholding_template_wrapper.v | 2 +- .../fpgadataflow/rtl/thresholding_rtl.py | 11 ++++---- .../fpgadataflow/convert_to_hw_layers.py | 10 ++++++-- .../test_fpgadataflow_thresholding.py | 25 +++++++++++++------ 4 files changed, 33 insertions(+), 15 deletions(-) diff --git a/finn-rtllib/thresholding/hdl/thresholding_template_wrapper.v b/finn-rtllib/thresholding/hdl/thresholding_template_wrapper.v index 62d92362dc..49a1f2bd8b 100644 --- a/finn-rtllib/thresholding/hdl/thresholding_template_wrapper.v +++ b/finn-rtllib/thresholding/hdl/thresholding_template_wrapper.v @@ -88,7 +88,7 @@ module $MODULE_NAME_AXI_WRAPPER$ #( //- AXI Stream - Input -------------- output in0_V_TREADY, input in0_V_TVALID, - input [((PE*K+7)/8)*8-1:0] in0_V_TDATA, + input [((PE*WI+7)/8)*8-1:0] in0_V_TDATA, //- AXI Stream - Output ------------- input out_V_TREADY, diff --git a/src/finn/custom_op/fpgadataflow/rtl/thresholding_rtl.py b/src/finn/custom_op/fpgadataflow/rtl/thresholding_rtl.py index ec875858ff..9584c3ae5f 100644 --- a/src/finn/custom_op/fpgadataflow/rtl/thresholding_rtl.py +++ b/src/finn/custom_op/fpgadataflow/rtl/thresholding_rtl.py @@ -180,15 +180,15 @@ def prepare_codegen_rtl_values(self, model): # Additionally, increase number of threshold steps to reflect new shape expected_thresholds = 2**o_bitwidth - 1 n_thres_steps = self.get_nodeattr("numSteps") + wdt = self.get_weight_datatype() if expected_thresholds != n_thres_steps: - min_val = DataType[input_data_type].min() + min_val = wdt.min() thresholds = np.insert(thresholds, 0, min_val, axis=1) bias = bias - 1 n_thres_steps += 1 # add dummy dimension as final dimension (that's what gets packed with next call) t_expand = np.expand_dims(thresholds, axis=-1) - wdt = self.get_weight_datatype() bw_hexdigit = roundup_to_integer_multiple(wdt.bitwidth(), 4) t_packed = pack_innermost_dim_as_hex_string( t_expand, @@ -242,9 +242,10 @@ def prepare_codegen_rtl_values(self, model): i_bitwidth = DataType[input_data_type].bitwidth() code_gen_dict["$N$"] = [str(o_bitwidth)] # output precision - convert bitwidth to string - code_gen_dict["$M$"] = [ - str(i_bitwidth) - ] # input/threshold precision - convert bitwidth to string + code_gen_dict["$WT$"] = [ + str(wdt.bitwidth()) + ] # threshold precision - convert bitwidth to string + code_gen_dict["$WI$"] = [str(i_bitwidth)] # input precision - convert bitwidth to string code_gen_dict["$C$"] = [str(num_channels)] # number of channels code_gen_dict["$BIAS$"] = [str(bias)] # activation bias value code_gen_dict["$PE$"] = [str(pe)] # requires C = M*PE diff --git a/src/finn/transformation/fpgadataflow/convert_to_hw_layers.py b/src/finn/transformation/fpgadataflow/convert_to_hw_layers.py index 897d714bf8..e14181b140 100644 --- a/src/finn/transformation/fpgadataflow/convert_to_hw_layers.py +++ b/src/finn/transformation/fpgadataflow/convert_to_hw_layers.py @@ -199,10 +199,16 @@ def apply(self, model): thl_in_shape = model.get_tensor_shape(thl_input) thl_thres_shape = model.get_tensor_shape(thl_threshold) idt = model.get_tensor_datatype(thl_input) - + tdt = model.get_tensor_datatype(thl_threshold) # skip conversion for layers with float input if not idt.is_integer(): continue + assert tdt.is_integer(), ( + node.name + + """: MultiThreshold cannot be converted + because thresholds are float type. Input data type is integer, + please run RoundAndClipThresholds to convert thresholds to integer.""" + ) # check layout of inputs/outputs, and convert if needed # check layout and convert if necessary @@ -253,7 +259,7 @@ def apply(self, model): PE=pe, numSteps=thl_thres_shape[1], inputDataType=idt.name, - weightDataType=idt.name, + weightDataType=tdt.name, outputDataType=odt.name, numInputVectors=list(thl_in_shape[:-1]), ActVal=actval, diff --git a/tests/fpgadataflow/test_fpgadataflow_thresholding.py b/tests/fpgadataflow/test_fpgadataflow_thresholding.py index 88e4247c2a..6501dba33e 100644 --- a/tests/fpgadataflow/test_fpgadataflow_thresholding.py +++ b/tests/fpgadataflow/test_fpgadataflow_thresholding.py @@ -55,7 +55,7 @@ def generate_random_threshold_values( - input_data_type, num_input_channels, num_steps, narrow=False, per_tensor=False + data_type, num_input_channels, num_steps, narrow=False, per_tensor=False ): if per_tensor: num_input_channels = 1 @@ -63,8 +63,8 @@ def generate_random_threshold_values( num_steps -= 1 return np.random.randint( - input_data_type.min(), - input_data_type.max() + 1, + data_type.min(), + data_type.max() + 1, (num_input_channels, num_steps), ).astype(np.float32) @@ -76,6 +76,7 @@ def sort_thresholds_increasing(thresholds): def make_single_multithresholding_modelwrapper( thresholds, input_data_type, + threshold_data_type, output_data_type, activation_bias, num_input_vecs, @@ -115,7 +116,7 @@ def make_single_multithresholding_modelwrapper( model.set_tensor_datatype("inp", input_data_type) model.set_tensor_datatype("outp", output_data_type) - model.set_tensor_datatype("thresh", input_data_type) + model.set_tensor_datatype("thresh", threshold_data_type) model.set_initializer("thresh", thresholds) return model @@ -129,7 +130,15 @@ def make_single_multithresholding_modelwrapper( ], ) @pytest.mark.parametrize("activation", [DataType["INT4"], DataType["BIPOLAR"]]) -@pytest.mark.parametrize("input_data_type", [DataType["INT8"], DataType["UINT8"]]) +@pytest.mark.parametrize( + "idt_tdt_cfg", + [ + (DataType["INT8"], DataType["INT8"]), + (DataType["INT8"], DataType["INT9"]), + (DataType["UINT8"], DataType["UINT8"]), + (DataType["UINT8"], DataType["UINT9"]), + ], +) @pytest.mark.parametrize("fold", [-1, 1, 2]) @pytest.mark.parametrize("narrow", [True, False]) @pytest.mark.parametrize("per_tensor", [True, False]) @@ -143,7 +152,7 @@ def test_fpgadataflow_thresholding( num_input_channels, num_input_vecs, activation, - input_data_type, + idt_tdt_cfg, fold, narrow, per_tensor, @@ -161,6 +170,7 @@ def test_fpgadataflow_thresholding( ) if narrow and activation == DataType["BIPOLAR"]: pytest.skip("Narrow needs to be false with biploar activation.") + input_data_type, threshold_data_type = idt_tdt_cfg num_steps = activation.get_num_possible_values() - 1 if fold == -1: @@ -179,7 +189,7 @@ def test_fpgadataflow_thresholding( # Generate random thresholds and sort in ascending order thresholds = generate_random_threshold_values( - input_data_type, num_input_channels, num_steps, narrow, per_tensor + threshold_data_type, num_input_channels, num_steps, narrow, per_tensor ) # provide non-decreasing/ascending thresholds @@ -189,6 +199,7 @@ def test_fpgadataflow_thresholding( model = make_single_multithresholding_modelwrapper( thresholds, input_data_type, + threshold_data_type, output_data_type, activation_bias, num_input_vecs,