axis_pulse_height_analyzer.v 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. `timescale 1 ns / 1 ps
  2. module axis_pulse_height_analyzer #
  3. (
  4. parameter integer AXIS_TDATA_WIDTH = 16,
  5. parameter AXIS_TDATA_SIGNED = "FALSE",
  6. parameter integer CNTR_WIDTH = 16
  7. )
  8. (
  9. // System signals
  10. input wire aclk,
  11. input wire aresetn,
  12. input wire [CNTR_WIDTH-1:0] cfg_data,
  13. input wire [AXIS_TDATA_WIDTH-1:0] min_data,
  14. input wire [AXIS_TDATA_WIDTH-1:0] max_data,
  15. // Slave side
  16. output wire s_axis_tready,
  17. input wire [AXIS_TDATA_WIDTH-1:0] s_axis_tdata,
  18. input wire s_axis_tvalid,
  19. // Master side
  20. input wire m_axis_tready,
  21. output wire [AXIS_TDATA_WIDTH-1:0] m_axis_tdata,
  22. output wire m_axis_tvalid
  23. );
  24. reg [AXIS_TDATA_WIDTH-1:0] int_data_reg[1:0], int_data_next[1:0];
  25. reg [AXIS_TDATA_WIDTH-1:0] int_min_reg, int_min_next;
  26. reg [AXIS_TDATA_WIDTH-1:0] int_tdata_reg, int_tdata_next;
  27. reg [CNTR_WIDTH-1:0] int_cntr_reg, int_cntr_next;
  28. reg int_enbl_reg, int_enbl_next;
  29. reg int_rising_reg, int_rising_next;
  30. reg int_tvalid_reg, int_tvalid_next;
  31. wire [AXIS_TDATA_WIDTH-1:0] int_tdata_wire;
  32. wire int_mincut_wire, int_maxcut_wire, int_rising_wire, int_delay_wire;
  33. assign int_delay_wire = int_cntr_reg < cfg_data;
  34. generate
  35. if(AXIS_TDATA_SIGNED == "TRUE")
  36. begin : SIGNED
  37. assign int_rising_wire = $signed(int_data_reg[1]) < $signed(s_axis_tdata);
  38. assign int_tdata_wire = $signed(int_data_reg[0]) - $signed(int_min_reg);
  39. assign int_mincut_wire = $signed(int_tdata_wire) > $signed(min_data);
  40. assign int_maxcut_wire = $signed(int_tdata_wire) < $signed(max_data);
  41. end
  42. else
  43. begin : UNSIGNED
  44. assign int_rising_wire = int_data_reg[1] < s_axis_tdata;
  45. assign int_tdata_wire = int_data_reg[0] - int_min_reg;
  46. assign int_mincut_wire = int_tdata_wire > min_data;
  47. assign int_maxcut_wire = int_tdata_wire < max_data;
  48. end
  49. endgenerate
  50. always @(posedge aclk)
  51. begin
  52. if(~aresetn)
  53. begin
  54. int_data_reg[0] <= {(AXIS_TDATA_WIDTH){1'b0}};
  55. int_data_reg[1] <= {(AXIS_TDATA_WIDTH){1'b0}};
  56. int_tdata_reg <= {(AXIS_TDATA_WIDTH){1'b0}};
  57. int_min_reg <= {(AXIS_TDATA_WIDTH){1'b0}};
  58. int_cntr_reg <= {(CNTR_WIDTH){1'b0}};
  59. int_enbl_reg <= 1'b0;
  60. int_rising_reg <= 1'b0;
  61. int_tvalid_reg <= 1'b0;
  62. end
  63. else
  64. begin
  65. int_data_reg[0] <= int_data_next[0];
  66. int_data_reg[1] <= int_data_next[1];
  67. int_tdata_reg <= int_tdata_next;
  68. int_min_reg <= int_min_next;
  69. int_cntr_reg <= int_cntr_next;
  70. int_enbl_reg <= int_enbl_next;
  71. int_rising_reg <= int_rising_next;
  72. int_tvalid_reg <= int_tvalid_next;
  73. end
  74. end
  75. always @*
  76. begin
  77. int_data_next[0] = int_data_reg[0];
  78. int_data_next[1] = int_data_reg[1];
  79. int_tdata_next = int_tdata_reg;
  80. int_min_next = int_min_reg;
  81. int_cntr_next = int_cntr_reg;
  82. int_enbl_next = int_enbl_reg;
  83. int_rising_next = int_rising_reg;
  84. int_tvalid_next = int_tvalid_reg;
  85. if(s_axis_tvalid)
  86. begin
  87. int_data_next[0] = s_axis_tdata;
  88. int_data_next[1] = int_data_reg[0];
  89. int_rising_next = int_rising_wire;
  90. end
  91. if(s_axis_tvalid & int_delay_wire)
  92. begin
  93. int_cntr_next = int_cntr_reg + 1'b1;
  94. end
  95. // minimum after delay
  96. if(s_axis_tvalid & ~int_delay_wire & ~int_rising_reg & int_rising_wire)
  97. begin
  98. int_min_next = int_data_reg[1];
  99. int_enbl_next = 1'b1;
  100. end
  101. // maximum after minimum
  102. if(s_axis_tvalid & int_enbl_reg & int_rising_reg & ~int_rising_wire & int_mincut_wire)
  103. begin
  104. int_tdata_next = int_maxcut_wire ? int_tdata_wire : {(AXIS_TDATA_WIDTH){1'b0}};
  105. int_tvalid_next = int_maxcut_wire;
  106. int_cntr_next = {(CNTR_WIDTH){1'b0}};
  107. int_enbl_next = 1'b0;
  108. end
  109. if(m_axis_tready & int_tvalid_reg)
  110. begin
  111. int_tdata_next = {(AXIS_TDATA_WIDTH){1'b0}};
  112. int_tvalid_next = 1'b0;
  113. end
  114. end
  115. assign s_axis_tready = 1'b1;
  116. assign m_axis_tdata = int_tdata_reg;
  117. assign m_axis_tvalid = int_tvalid_reg;
  118. endmodule