// // Copyright 2013 Ettus Research LLC // Copyright 2018 Ettus Research, a National Instruments Company // // SPDX-License-Identifier: LGPL-3.0-or-later // module chdr_16sc_to_12sc #(parameter BASE=0) ( // Clocks and resets input clk, input reset, // Settings bus input set_stb, input [7:0] set_addr, input [31:0] set_data, // Input CHDR bus input [63:0] i_tdata, input i_tlast, input i_tvalid, output i_tready, // Output CHDR bus output [63:0] o_tdata, output o_tlast, output o_tvalid, input o_tready, // Debug output [31:0] debug ); wire chdr_has_time = i_tdata[61]; wire [11:0] q0; wire [11:0] i0; wire [11:0] q1; wire [11:0] i1; wire [11:0] q2; wire [11:0] i2; wire [16:0] round_q0; wire [16:0] round_i0; wire [16:0] round_q1; wire [16:0] round_i1; wire [16:0] round_q2; wire [16:0] round_i2; reg [63:0] curr_word; // Pipeline register reg [63:0] buff; reg buff_tvalid; reg buff_tlast; // CHDR has either 8 bytes of header or 16 if VITA time is included. wire [15:0] chdr_header_bytes = chdr_has_time? 16 : 8; // Calculate size of samples input in bytes by taking CHDR size filed and subtracting header length. wire [15:0] sample_byte_count_in = i_tdata[47:32] - chdr_header_bytes; // Calculate size of samples to be output by taking input size and scaling by 3/4 wire [15:0] sample_byte_count_out = (sample_byte_count_in*3) >> 2; // Calculate size of output CHDR packet by adding back header size to new payload size. wire [15:0] output_chdr_pkt_size = sample_byte_count_out + chdr_header_bytes; reg odd; wire set_sid; wire [15:0] new_sid_dst; setting_reg #(.my_addr(BASE), .width(17)) new_destination (.clk(clk), .rst(reset), .strobe(set_stb), .addr(set_addr), .in(set_data), .out({set_sid, new_sid_dst[15:0]})); // state machine localparam HEADER = 3'd0; localparam TIME = 3'd1; localparam SAMPLE1 = 3'd2; localparam SAMPLE2 = 3'd3; localparam SAMPLE3 = 3'd4; localparam SAMPLE4 = 3'd5; reg [2:0] state; always @(posedge clk) if (reset) begin state <= HEADER; buff <= 64'd0; buff_tvalid <= 1'd0; buff_tlast <= 1'd0; end else begin case(state) // // Process header // Check for timestamp. Byte count conversion is done above. // If there is residual data in the buffer, store the header and // output the line in the buffer. If not, output the header. // HEADER: begin if (i_tvalid & i_tready) begin odd <= sample_byte_count_in [2]; if (buff_tvalid) begin buff <= curr_word; buff_tvalid <= i_tvalid; buff_tlast <= i_tlast; end else begin buff <= 64'd0; buff_tvalid <= 1'd0; buff_tlast <= 1'd0; end state <= i_tlast ? HEADER : (i_tdata[61]) ? TIME : SAMPLE1; end else if (buff_tvalid & o_tready) begin buff <= 64'd0; buff_tvalid <= 1'd0; buff_tlast <= 1'd0; end end // // Process time field // If the header is in the buffer, output the header and // store the timestamp. If not, output the timestamp. // TIME: begin if (i_tvalid & i_tready) begin if (buff_tvalid) begin buff <= curr_word; buff_tvalid <= i_tvalid; buff_tlast <= i_tlast; end else begin buff <= 64'd0; buff_tvalid <= 1'd0; buff_tlast <= 1'd0; end state <= i_tlast ? HEADER: SAMPLE1; end end // // There are 3 lines of output data for each 4 lines of input data. // The 4 sample states below represent the 4 lines of input. // They are repeatedly cycled until all data is consumed. // // Process first line // The 8 bytes are converted to 6 bytes, so there is not enough for an // 8-byte output line. Store the data unless this is the last line in // the packet. If the timestamp is in the buffer, output it. // SAMPLE1: begin if (i_tvalid & i_tready) begin buff <= curr_word; buff_tvalid <= i_tlast; buff_tlast <= i_tlast; state <= i_tlast ? HEADER : SAMPLE2; end end // // Process second line // Output a line comprised of the 6 bytes from the fist line and // 2 bytes from this line. Store the remaining 4 bytes. // SAMPLE2: begin if (i_tvalid & i_tready) begin buff <= {i0[7:0],q1,i1,32'd0}; buff_tvalid <= i_tlast; buff_tlast <= i_tlast; state <= i_tlast ? HEADER : SAMPLE3; end end // // Process third line // Output line comprised of the 4 remaining bytes from the second line // and 4 bytes from this line. Store the remaining 2 bytes unless this // is the last line in the packet and the number of samples is odd. // SAMPLE3: begin if (i_tvalid & i_tready) begin buff <= {q1[3:0],i1,48'd0}; buff_tvalid <= i_tlast & ~odd; buff_tlast <= i_tlast & ~odd; state <= i_tlast ? HEADER : SAMPLE4; end end // // Process fourth line // Output line comprised of the remaining 2 bytes from the third line // and the 6 bytes from this line. // SAMPLE4: begin if (i_tvalid & i_tready) begin buff <= 64'd0; buff_tvalid <= 1'd0; buff_tlast <= 1'd0; state <= i_tlast ? HEADER : SAMPLE1; end end // // Should never get here. // default: state <= HEADER; endcase end // Add rounding value into 16bit samples before trunctaion assign round_q0 = ({i_tdata[63],i_tdata[63:48]} + 'h0008); assign round_i0 = ({i_tdata[47],i_tdata[47:32]} + 'h0008); // Truncate with saturation to 12bits precision. assign q0 = (round_q0[16:15] == 2'b01) ? 12'h7FF : ((round_q0[16:15] == 2'b10) ? 12'h800 : round_q0[15:4]); assign i0 = (round_i0[16:15] == 2'b01) ? 12'h7FF : ((round_i0[16:15] == 2'b10) ? 12'h800 : round_i0[15:4]); // Add rounding value into 16bit samples before trunctaion assign round_q1 = ({i_tdata[31],i_tdata[31:16]} + 'h0008); assign round_i1 = ({i_tdata[15],i_tdata[15:0]} + 'h0008); // Truncate with saturation to 12bits precision. assign q1 = (round_q1[16:15] == 2'b01) ? 12'h3FF : ((round_q1[16:15] == 2'b10) ? 12'h800 : round_q1[15:4]); assign i1 = (round_i1[16:15] == 2'b01) ? 12'h3FF : ((round_i1[16:15] == 2'b10) ? 12'h800 : round_i1[15:4]); // // Mux for current word // always @(*) case(state) HEADER: curr_word <= {i_tdata[63:48], output_chdr_pkt_size, set_sid ? {i_tdata[15:0], new_sid_dst[15:0]}:i_tdata[31:0]}; TIME: curr_word <= i_tdata; SAMPLE1: curr_word <= {q0,i0,q1, i1, 16'b0}; SAMPLE2: curr_word <= {buff[63:16], q0, i0[11:8]}; SAMPLE3: curr_word <= {buff[63:32], q0, i0,q1[11:4]}; SAMPLE4: curr_word <= {buff[63:48], q0, i0, q1, i1}; endcase assign o_tdata = buff_tvalid ? buff : curr_word; assign o_tvalid = (state == HEADER && buff_tvalid) || (i_tvalid && (state != SAMPLE1 || (state == SAMPLE1 && buff_tvalid))); assign o_tlast = buff_tvalid ? buff_tlast : i_tlast && (state == HEADER || state == TIME || (state == SAMPLE3 && odd) || state == SAMPLE4); assign i_tready = o_tready; endmodule