ExtSpiMEmul.v 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205
  1. `timescale 1ns / 1ps
  2. module ExtSpiMEmul
  3. (
  4. input Rst_i,
  5. input Clk_i,
  6. input Start_i,
  7. output TxDone_o,
  8. output Sck_o,
  9. output reg Ss_o,
  10. output reg Mosi_o
  11. );
  12. //================================================================================
  13. // PARAMETERS
  14. localparam [1:0] IDLE = 0;
  15. localparam [1:0] CMD = 1;
  16. localparam [1:0] TX = 2;
  17. localparam [1:0] PAUSE = 3;
  18. parameter MODE = 1'h0;
  19. parameter [4:0] DEVID = 5'h1;
  20. parameter [16:0] WORDSNUM = 17'h3;
  21. parameter EOPBIT = 1'b1;
  22. //================================================================================
  23. // REG/WIRE
  24. reg [1:0] currState;
  25. reg [1:0] nextState;
  26. reg [6:0] txCnt;
  27. reg [6:0] cmdCnt;
  28. reg [3:0] pauseCnt;
  29. wire txStop = (cmdCnt >= WORDSNUM+1);
  30. reg [23:0] headerCmd = {MODE,DEVID,WORDSNUM,EOPBIT};
  31. reg [23:0] spiData;
  32. reg [23:0] dspSpiData;
  33. reg sckFlag;
  34. //================================================================================
  35. // ASSIGNMENTS
  36. assign Sck_o = (sckFlag)? ~Clk_i:1'b1;
  37. assign TxDone_o = (txStop & (currState== CMD));
  38. //================================================================================
  39. // CODING
  40. always @(posedge Clk_i) begin
  41. if (!Rst_i) begin
  42. if (currState == CMD) begin
  43. if (!txStop) begin
  44. cmdCnt <= cmdCnt+1;
  45. end else begin
  46. cmdCnt <= 0;
  47. end
  48. end
  49. end else begin
  50. cmdCnt <= 0;
  51. end
  52. end
  53. always @(posedge Clk_i) begin
  54. if (!Rst_i) begin
  55. if (currState == TX) begin
  56. txCnt <= txCnt+1;
  57. end else begin
  58. txCnt <= 0;
  59. end
  60. end else begin
  61. txCnt <= 0;
  62. end
  63. end
  64. always @(posedge Clk_i) begin
  65. if (!Rst_i) begin
  66. if (currState == PAUSE) begin
  67. pauseCnt <= pauseCnt+1;
  68. end else begin
  69. pauseCnt <= 0;
  70. end
  71. end else begin
  72. pauseCnt <= 0;
  73. end
  74. end
  75. always @(posedge Clk_i) begin
  76. if (!Rst_i) begin
  77. if (currState == CMD) begin
  78. spiData <= spiData+cmdCnt;
  79. end
  80. end else begin
  81. spiData <= 24'hab;
  82. end
  83. end
  84. always @(posedge Clk_i) begin
  85. if (currState == CMD) begin
  86. if (cmdCnt == 0) begin
  87. dspSpiData <= headerCmd;
  88. end else begin
  89. dspSpiData <= spiData;
  90. end
  91. end else if (currState == TX) begin
  92. dspSpiData <= dspSpiData<<1;
  93. end if (currState == IDLE) begin
  94. dspSpiData <= 0;
  95. end
  96. end
  97. always @(posedge Clk_i) begin
  98. if (currState == TX) begin
  99. if (txCnt >= 7'd0) begin
  100. Mosi_o <= dspSpiData[23];
  101. end else begin
  102. Mosi_o <= 1'b1;
  103. end
  104. end else begin
  105. Mosi_o <= 1'b1;
  106. end
  107. end
  108. always @(posedge Clk_i) begin
  109. if (currState == TX) begin
  110. Ss_o <= 1'b0;
  111. sckFlag <= 1'b1;
  112. end else begin
  113. Ss_o <= 1'b1;
  114. sckFlag <= 1'b0;
  115. end
  116. end
  117. always @(posedge Clk_i) begin
  118. if (Rst_i) begin
  119. currState <= IDLE;
  120. end else begin
  121. currState <= nextState;
  122. end
  123. end
  124. always @(*) begin
  125. nextState = IDLE;
  126. case(currState)
  127. IDLE : begin
  128. if (Start_i) begin
  129. nextState = CMD;
  130. end else begin
  131. nextState = IDLE;
  132. end
  133. end
  134. CMD : begin
  135. if (!txStop) begin
  136. nextState = TX;
  137. end else begin
  138. nextState = IDLE;
  139. end
  140. end
  141. TX : begin
  142. if (txCnt==6'd23) begin
  143. nextState = PAUSE;
  144. end else begin
  145. nextState = TX;
  146. end
  147. end
  148. PAUSE : begin
  149. if (pauseCnt==4'd2) begin
  150. nextState = CMD;
  151. end else begin
  152. nextState = PAUSE;
  153. end
  154. end
  155. endcase
  156. end
  157. endmodule