spi_master.sv 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. //
  2. // spi_master.sv
  3. //
  4. // Simple byte-oriented SPI master unit with optional multiwidth support
  5. // (1, 2, 4, 8).
  6. //
  7. // The output SPI clock equals the system clock /2 unless clk_en is used
  8. // to throttle the output clock.
  9. //
  10. // spi_io[0] = DI, spi_io[1] = DO in single bit mode.
  11. //
  12. // All unused spi_io are driven to 1.
  13. //
  14. //
  15. // XXX: add clock polarity, MSB/LSB options
  16. //
  17. module spi_master
  18. #(
  19. parameter width = 1, // Width of SPI data
  20. parameter n_cs = 1, // Number of CS# outputs
  21. parameter cs_delay = 1,
  22. parameter io_max = max(ilog2c(width)-1, 1)
  23. )
  24. (
  25. input rst_n, // Unit reset
  26. input clk, // System clock
  27. input clk_en, // SPI clock enable
  28. input [7:0] d, // System data in
  29. output [7:0] q, // System data out
  30. input req, // Session request
  31. input dir, // Session is write (for multibit)
  32. input [1:0] iowidth, // Session width (lg2)
  33. output sack, // Session started
  34. output eack, // Session ended
  35. input [n_cs-1:0] cs, // Device select (active high)
  36. output spi_sck, // SPI clock
  37. inout [min(ilog2c(width)-1, 1):0] spi_io, // SPI data
  38. output [n_cs-1:0] spi_cs_n // SPI CS# lines
  39. );
  40. localparam ctr_max = max(ilog2c(cs_delay)-1,2);
  41. reg spi_clk;
  42. reg spi_sck_q;
  43. reg spi_active;
  44. reg [ilog2c(width)-1:0] spi_width;
  45. reg [ctr_max:0] spi_ctr;
  46. reg [n_cs-1:0] spi_cs_q;
  47. reg [7:0] spi_out_q;
  48. reg [1:0] spi_oe_q;
  49. reg d_dir;
  50. reg [7:0] d_out;
  51. reg [7:0] d_in;
  52. reg [7:0] q_q;
  53. reg sack_q;
  54. reg eack_q;
  55. assign spi_cs_n = ~spi_cs_q;
  56. assign spi_sck = spi_sck_q;
  57. wire spi_cs_changed = |(spi_cs_q ^ cs);
  58. assign spi_io[0] = spi_oe_q[0] ? spi_out_q[0] : 1'bz;
  59. assign spi_io[io_max:1] = spi_oe_q[1] ? spi_out_q[io_max:1] : {io_max{1'bz}};
  60. always @(negedge rst_n or posedge clk)
  61. if (~rst_n)
  62. begin
  63. spi_clk <= 1'b0;
  64. spi_active <= 1'b0;
  65. spi_width <= 4'b0001;
  66. spi_ctr <= 1'b0;
  67. spi_cs_q <= 1'b0;
  68. spi_out_q <= 8'hFF;
  69. spi_oe_q <= 2'b10;
  70. sack_q <= 1'b0;
  71. eack_q <= 1'b0;
  72. d_out <= 8'hFF;
  73. d_in <= 8'hxx;
  74. q_q <= 8'hxx;
  75. end
  76. else
  77. begin
  78. sack_q <= 1'b0;
  79. eack_q <= 1'b0;
  80. if (clk_en)
  81. begin
  82. spi_ctr <= spi_ctr - 1'b1;
  83. spi_clk <= spi_ctr[0] & spi_active;
  84. if (~spi_ctr[0])
  85. begin
  86. case (spi_width)
  87. 4'd1:
  88. d_in <= { d_in[6:0], spi_io[0] };
  89. 4'd2:
  90. d_in <= { d_in[5:0], spi_io[1:0] };
  91. 4'd4:
  92. d_in <= { d_in[3:0], spi_io[3:0] };
  93. 4'd8:
  94. d_in <= spi_io;
  95. default:
  96. d_in <= 8'hxx;
  97. endcase // case 2'd3
  98. end // if (~spi_ctr[0])
  99. else
  100. begin
  101. if (spi_active)
  102. begin
  103. spi_out_q <= 8'hFF;
  104. if (~|spi_ctr[ctr_max:1])
  105. begin
  106. eack_q <= 1'b1;
  107. q_q <= d_in;
  108. spi_active <= 1'b0;
  109. end
  110. case (spi_width)
  111. 4'd1:
  112. begin
  113. d_out <= { d_out[6:0], 1'b1 };
  114. spi_out_q[1] <= d_out[7];
  115. end
  116. 4'd2:
  117. begin
  118. d_out <= { d_out[5:0], 2'b11 };
  119. spi_out_q[1:0] <= d_out[7:6];
  120. end
  121. 4'd4:
  122. begin
  123. d_out <= { d_out[3:0], 4'b1111 };
  124. spi_out_q[3:0] <= d_out[7:4];
  125. end
  126. 4'd8:
  127. begin
  128. d_out <= 8'hFF;
  129. spi_out_q <= d_out;
  130. end
  131. default:
  132. begin
  133. d_out <= 8'hxx;
  134. spi_out_q <= 8'hxx;
  135. end
  136. endcase // case (spi_width)
  137. end // if (spi_active)
  138. else
  139. begin
  140. spi_cs_q <= cs;
  141. d_out <= d;
  142. spi_out_q <= 8'hFF;
  143. if (cs_delay != 0 &&
  144. (spi_cs_changed | ~|spi_ctr[ctr_max:1]))
  145. begin
  146. if (spi_cs_changed)
  147. spi_ctr[ctr_max:1] <= cs_delay;
  148. spi_oe_q <= 2'b10; // As for 1-bit mode
  149. end
  150. else if (cs_delay == 0 || ~|spi_ctr[ctr_max:1])
  151. begin
  152. if (req)
  153. begin
  154. case (iowidth)
  155. 2'd0:
  156. begin
  157. spi_width <= 4'b0001;
  158. spi_ctr[ctr_max:1] <= 3'd8;
  159. spi_oe_q <= 2'b10;
  160. end
  161. 2'd1:
  162. if (width < 2)
  163. begin
  164. spi_width <= 4'b000x;
  165. spi_ctr[ctr_max:1] <= 1'bx;
  166. spi_oe_q <= 2'bxx;
  167. end
  168. else
  169. begin
  170. spi_width <= 4'b0010;
  171. spi_ctr[ctr_max:1] <= 3'd4;
  172. spi_oe_q <= {2{dir}};
  173. end
  174. 2'd2:
  175. if (width < 4)
  176. begin
  177. spi_width <= 4'b00xx;
  178. spi_ctr[ctr_max:1] <= 1'bx;
  179. spi_oe_q <= 2'bxx;
  180. end
  181. else
  182. begin
  183. spi_width <= 4'b0100;
  184. spi_ctr[ctr_max:1] <= 3'd2;
  185. spi_oe_q <= {2{dir}};
  186. end
  187. 2'd3:
  188. if (width < 8)
  189. begin
  190. spi_width <= 4'b0xxx;
  191. spi_ctr[ctr_max:1] <= 1'bx;
  192. spi_oe_q <= 2'bxx;
  193. end
  194. else
  195. begin
  196. spi_width <= 4'b1000;
  197. spi_ctr[ctr_max:1] <= 3'd1;
  198. spi_oe_q <= {2{dir}};
  199. end
  200. endcase // case (iowidth)
  201. spi_active <= 1'b1;
  202. sack_q <= 1'b1;
  203. end // if (req)
  204. end // if (cs_delay == 0 || ~|spi_ctr[ctr_max:1])
  205. end // else: !if(spi_active)
  206. end // else: !if(~spi_ctr[0])
  207. end // if (clk_en)
  208. end // else: !if(~rst_n)
  209. endmodule // spi_master