/*! * Module:mul_saxi_wr_chn * @file mul_saxi_wr_chn.v * @date 2015-07-10 * @author Andrey Filippov * * @brief One channel of the mult_saxi_wr (read/write common buffer) * * @copyright Copyright (c) 2015 Elphel, Inc . * * License: * * mul_saxi_wr_chn.v 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. * * mul_saxi_wr_chn.v 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 . * * Additional permission under GNU GPL version 3 section 7: * If you modify this Program, or any covered work, by linking or combining it * with independent modules provided by the FPGA vendor only (this permission * does not extend to any 3-rd party modules, "soft cores" or macros) under * different license terms solely for the purpose of generating binary "bitstream" * files and/or simulating the code, the copyright holders of this Program give * you the right to distribute the covered work without those independent modules * as long as the source code for them is available from the FPGA vendor free of * charge, and there is no dependence on any encrypted modules for simulating of * the combined code. This permission applies to you if the distributed code * contains all the components and scripts required to completely simulate it * with at least one of the Free Software programs. */ `timescale 1ns/1ps module mult_saxi_wr_chn #( parameter MULT_SAXI_HALF_BRAM = 1, // 0 - use full 36Kb BRAM for the buffer, 1 - use just half parameter MULT_SAXI_BSLOG = 4, // number of bits to represent burst size (4 - b.s. = 16, 0 - b.s = 1) parameter MULT_SAXI_ADV_WR = 4, // number of clock cycles before end of write to genearte adv_wr_done parameter MULT_SAXI_ADV_RD = 3 // number of clock cycles before end of read to genearte wdata_busy (if !fifo_half_full) ) ( input mclk, // system clock input aclk, // global clock to run s_axi (@150MHz?) input en, // enable this channle ( 0 - reset) input has_burst, // channel has at least 1 burst (should go down immediately after read_burst if no more data) // use grant_wr to request reading external data // output read_burst, // request to read a burst of data from the channel input valid, // data valid (same latency) output rq_wr, // request to write to the buffer FIFO input grant_wr, // single-cycle output [(MULT_SAXI_HALF_BRAM?6:7):0] wa, // write buffer address (w/o 2 MSB - channel) output reg adv_wr_done, // outputs grant_wr for short bursts, or several clocks before end of wr // output pre_we, // will be registered after mux - use valid output reg rq_out, input grant_out, // single-cycle input fifo_half_full, // output fifo is half full - use it to suspend readout output [(MULT_SAXI_HALF_BRAM?6:7):0] ra, // read buffer address (w/o 2 MSB - channel) output pre_re, // will be registerd after the MUX output reg first_re, // reading first word (next cycle after corresponding pre_re) output reg last_re, // reading lastt word (next cycle after corresponding pre_re) output reg wdata_busy ); localparam BURSTS_BITS= (MULT_SAXI_HALF_BRAM ? 9 : 10 ) - MULT_SAXI_BSLOG - 2; // number of bits to count number of bursts in 0-th quarter of the buffer reg [BURSTS_BITS-1:0] wr_burst; reg [MULT_SAXI_BSLOG-1:0] wr_word; reg [BURSTS_BITS:0] wr_num_burst; // number of bursts in the buffer chn0, as seen from the write side reg [BURSTS_BITS-1:0] rd_burst; reg [MULT_SAXI_BSLOG-1:0] rd_word; reg [BURSTS_BITS:0] rd_num_burst; // number of bursts in the buffer chn0, as seen from the read side reg rq_wr_r; reg rq_wr_busy; // reg early_wr_done; // single-cycle pulse several clock before end of write busy // reg grant_wr_r; // wire grant_wr_sngl; // wire grant_wr_aclk; wire write_last_in_burst; wire burst_written_aclk; // reg grant_out_r; // wire grant_out_sngl; wire grant_out_mclk; reg en_aclk; wire last_word_busy; reg pre_re_r; // may be interrupted if fifo_half_full reg out_busy; // output data in progress assign wa = {wr_burst, wr_word}; assign ra = {rd_burst, rd_word}; assign rq_wr = rq_wr_r; // assign grant_wr_sngl = grant_wr && !grant_wr_r; // assign grant_out_sngl = grant_out && ~grant_out_r; assign last_word_busy = &wr_word ; // make it earlier, use BURSTS_BITS selection (& (word | (1 <= 0)? ADV_WR_COUNT : 0)); if (!en) rq_wr_busy <= 0; else if (grant_wr) rq_wr_busy <= 1; else if (valid && last_word_busy) rq_wr_busy <= 0; rq_wr_r <= has_burst & (~wr_num_burst[BURSTS_BITS] & ~(&wr_num_burst[BURSTS_BITS-1:0])) & ~grant_wr & ~rq_wr_busy; // Number of bursts in fifo as seen from the input if (!en) wr_num_burst <= 0; else if ( grant_wr && !grant_out_mclk) wr_num_burst <= wr_num_burst + 1; else if (!grant_wr && grant_out_mclk) wr_num_burst <= wr_num_burst - 1; if (!en || grant_wr) wr_word <= 0; else if (valid) wr_word <= wr_word + 1; if (!en) wr_burst <= 0; else if (write_last_in_burst) wr_burst <= wr_burst + 1; end reg early_busy; // output data in progress always @ (posedge aclk) begin en_aclk <= en; // Number of bursts in fifo as seen from the output if (!en_aclk) rd_num_burst <= 0; else if ( burst_written_aclk && !grant_out) rd_num_burst <= rd_num_burst + 1; else if (!burst_written_aclk && grant_out) rd_num_burst <= rd_num_burst - 1; if (!en_aclk) rq_out <= 0; else if ( burst_written_aclk && !grant_out) rq_out <= 1; else if (!burst_written_aclk && grant_out) rq_out <= |rd_num_burst[BURSTS_BITS:1]; // >=2 if (! en_aclk || grant_out) rd_word <= 0; else if (pre_re_r) rd_word <=rd_word +1; if (!en_aclk) rd_burst <= wr_burst; // <= 0 is OK too else if (pre_re_r && (&rd_word)) rd_burst <= rd_burst + 1; if (!en_aclk) out_busy <= 0; else if (grant_out) out_busy <= 1; else if ((&rd_word) && pre_re_r) out_busy <= 0; if (!en_aclk || fifo_half_full || ((&rd_word) && pre_re_r)) pre_re_r <= 0; else pre_re_r <= out_busy; first_re <= pre_re_r && !(|rd_word); // will be used to copy channel/axi_wid last_re <= pre_re_r && (&rd_word); // will be used to generate axi_wlast if (!en_aclk || (ADV_RD_COUNT > 0)) early_busy <= 0; // small counts will never get busy else if (grant_out) early_busy <= 1; else if (rd_word == ADV_RD_COUNT) early_busy <= 0; if (!en_aclk) wdata_busy <= 0; else if (grant_out) wdata_busy <= 1; else if ((!fifo_half_full && !early_busy) || (&rd_word) || !out_busy) wdata_busy <= 0; end pulse_cross_clock grant_out_mclk_i ( .rst (!en), // input .src_clk (aclk), // input .dst_clk (mclk), // input .in_pulse (grant_out), // input .out_pulse (grant_out_mclk), // output .busy() // output ); pulse_cross_clock write_last_in_burst_i ( .rst (!en_aclk), // input .src_clk (mclk), // input .dst_clk (aclk), // input .in_pulse (write_last_in_burst), // input .out_pulse (burst_written_aclk), // output .busy() // output ); endmodule