diff --git a/EF_SPI.yaml b/EF_SPI.yaml index 108da10..417f77c 100644 --- a/EF_SPI.yaml +++ b/EF_SPI.yaml @@ -7,7 +7,7 @@ info: license: APACHE 2.0 author: Mohamed Shalan email: mshalan@efabless.com - version: v1.0.10 + version: v1.0.11 date: 17-09-2024 category: digital tags: @@ -137,6 +137,14 @@ ports: width: 1 direction: input description: enable for spi master pulse generation +- name: done + width: 1 + direction: output + description: spi done flag. +- name: busy + width: 1 + direction: output + description: spi busy flag. external_interface: - name: miso @@ -144,7 +152,7 @@ external_interface: width: 1 direction: input description: SPI Master In Slave Out. - sync: True + sync: False - name: mosi port: mosi width: 1 @@ -236,7 +244,7 @@ registers: write_port: clk_divider description: SPI clock Prescaler; should have a value >= 2. SPI Clock Frequency = System Clock / PR. - name: STATUS - size: 6 + size: 8 mode: r fifo: no offset: 20 @@ -274,6 +282,16 @@ registers: bit_width: 1 read_port: rx_level_above description: Receive FIFO level is Above Threshold. + - name : busy + bit_offset: 6 + bit_width: 1 + read_port: busy + description: spi busy flag. + - name : done + bit_offset: 7 + bit_width: 1 + read_port: done + description: spi done flag. flags: diff --git a/hdl/rtl/EF_SPI.v b/hdl/rtl/EF_SPI.v index 418583b..036c0bd 100644 --- a/hdl/rtl/EF_SPI.v +++ b/hdl/rtl/EF_SPI.v @@ -49,8 +49,8 @@ module EF_SPI #(parameter output wire tx_level_below, output wire [FAW-1:0] tx_level, - //output busy, - //output done, + output wire busy, + output wire done, input wire miso, output wire mosi, @@ -61,9 +61,6 @@ module EF_SPI #(parameter localparam FDW = 8; - wire busy; - wire done; - // TX Side wire tx_wr = wr; wire tx_rd = !tx_empty & !busy; diff --git a/hdl/rtl/bus_wrappers/EF_SPI_AHBL.pp.v b/hdl/rtl/bus_wrappers/EF_SPI_AHBL.pp.v index d4e9a6b..e964c38 100644 --- a/hdl/rtl/bus_wrappers/EF_SPI_AHBL.pp.v +++ b/hdl/rtl/bus_wrappers/EF_SPI_AHBL.pp.v @@ -104,6 +104,10 @@ module EF_SPI_AHBL #( CDW = 8, FAW = 4 ) ( + + + + input wire HCLK, input wire HRESETn, input wire HWRITE, @@ -138,7 +142,21 @@ module EF_SPI_AHBL #( localparam MIS_REG_OFFSET = 16'hFF04; localparam RIS_REG_OFFSET = 16'hFF08; localparam IC_REG_OFFSET = 16'hFF0C; - wire clk = HCLK; + + reg [0:0] GCLK_REG; + wire clk_g; + wire clk_gated_en = GCLK_REG[0]; + ef_gating_cell clk_gate_cell( + + + + // USE_POWER_PINS + .clk(HCLK), + .clk_en(clk_gated_en), + .clk_o(clk_g) + ); + + wire clk = clk_g; wire rst_n = HRESETn; @@ -182,6 +200,8 @@ module EF_SPI_AHBL #( wire [FAW-1:0] tx_level; wire [1-1:0] ss; wire [1-1:0] enable; + wire [1-1:0] done; + wire [1-1:0] busy; // Register Definitions wire [8-1:0] RXDATA_WIRE; @@ -209,13 +229,15 @@ module EF_SPI_AHBL #( else if(ahbl_we & (last_HADDR[16-1:0]==PR_REG_OFFSET)) PR_REG <= HWDATA[CDW-1:0]; - wire [6-1:0] STATUS_WIRE; + wire [8-1:0] STATUS_WIRE; assign STATUS_WIRE[0 : 0] = tx_empty; assign STATUS_WIRE[1 : 1] = tx_full; assign STATUS_WIRE[2 : 2] = rx_empty; assign STATUS_WIRE[3 : 3] = rx_full; assign STATUS_WIRE[4 : 4] = tx_level_below; assign STATUS_WIRE[5 : 5] = rx_level_above; + assign STATUS_WIRE[6 : 6] = busy; + assign STATUS_WIRE[7 : 7] = done; wire [FAW-1:0] RX_FIFO_LEVEL_WIRE; assign RX_FIFO_LEVEL_WIRE[(FAW - 1) : 0] = rx_level; @@ -251,6 +273,11 @@ module EF_SPI_AHBL #( else TX_FIFO_FLUSH_REG <= 1'h0 & TX_FIFO_FLUSH_REG; + localparam GCLK_REG_OFFSET = 16'hFF10; + always @(posedge HCLK or negedge HRESETn) if(~HRESETn) GCLK_REG <= 0; + else if(ahbl_we & (last_HADDR[16-1:0]==GCLK_REG_OFFSET)) + GCLK_REG <= HWDATA[1-1:0]; + reg [5:0] IM_REG; reg [5:0] IC_REG; reg [5:0] RIS_REG; @@ -324,6 +351,8 @@ module EF_SPI_AHBL #( .tx_level(tx_level), .ss(ss), .enable(enable), + .done(done), + .busy(busy), .miso(miso), .mosi(mosi), .csb(csb), @@ -347,6 +376,7 @@ module EF_SPI_AHBL #( (last_HADDR[16-1:0] == MIS_REG_OFFSET) ? MIS_REG : (last_HADDR[16-1:0] == RIS_REG_OFFSET) ? RIS_REG : (last_HADDR[16-1:0] == IC_REG_OFFSET) ? IC_REG : + (last_HADDR[16-1:0] == GCLK_REG_OFFSET) ? GCLK_REG : 32'hDEADBEEF; assign HREADYOUT = 1'b1; diff --git a/hdl/rtl/bus_wrappers/EF_SPI_AHBL.v b/hdl/rtl/bus_wrappers/EF_SPI_AHBL.v index 0d05325..11824a2 100644 --- a/hdl/rtl/bus_wrappers/EF_SPI_AHBL.v +++ b/hdl/rtl/bus_wrappers/EF_SPI_AHBL.v @@ -31,6 +31,10 @@ module EF_SPI_AHBL #( CDW = 8, FAW = 4 ) ( +`ifdef USE_POWER_PINS + inout VPWR, + inout VGND, +`endif `AHBL_SLAVE_PORTS, input wire [1-1:0] miso, output wire [1-1:0] mosi, @@ -54,7 +58,21 @@ module EF_SPI_AHBL #( localparam MIS_REG_OFFSET = `AHBL_AW'hFF04; localparam RIS_REG_OFFSET = `AHBL_AW'hFF08; localparam IC_REG_OFFSET = `AHBL_AW'hFF0C; - wire clk = HCLK; + + reg [0:0] GCLK_REG; + wire clk_g; + wire clk_gated_en = GCLK_REG[0]; + ef_gating_cell clk_gate_cell( + `ifdef USE_POWER_PINS + .vpwr(VPWR), + .vgnd(VGND), + `endif // USE_POWER_PINS + .clk(HCLK), + .clk_en(clk_gated_en), + .clk_o(clk_g) + ); + + wire clk = clk_g; wire rst_n = HRESETn; @@ -82,6 +100,8 @@ module EF_SPI_AHBL #( wire [FAW-1:0] tx_level; wire [1-1:0] ss; wire [1-1:0] enable; + wire [1-1:0] done; + wire [1-1:0] busy; // Register Definitions wire [8-1:0] RXDATA_WIRE; @@ -103,13 +123,15 @@ module EF_SPI_AHBL #( assign clk_divider = PR_REG; `AHBL_REG(PR_REG, 'h2, CDW) - wire [6-1:0] STATUS_WIRE; + wire [8-1:0] STATUS_WIRE; assign STATUS_WIRE[0 : 0] = tx_empty; assign STATUS_WIRE[1 : 1] = tx_full; assign STATUS_WIRE[2 : 2] = rx_empty; assign STATUS_WIRE[3 : 3] = rx_full; assign STATUS_WIRE[4 : 4] = tx_level_below; assign STATUS_WIRE[5 : 5] = rx_level_above; + assign STATUS_WIRE[6 : 6] = busy; + assign STATUS_WIRE[7 : 7] = done; wire [FAW-1:0] RX_FIFO_LEVEL_WIRE; assign RX_FIFO_LEVEL_WIRE[(FAW - 1) : 0] = rx_level; @@ -133,6 +155,9 @@ module EF_SPI_AHBL #( assign tx_flush = TX_FIFO_FLUSH_REG[0 : 0]; `AHBL_REG_AC(TX_FIFO_FLUSH_REG, 0, 1, 1'h0) + localparam GCLK_REG_OFFSET = `AHBL_AW'hFF10; + `AHBL_REG(GCLK_REG, 0, 1) + reg [5:0] IM_REG; reg [5:0] IC_REG; reg [5:0] RIS_REG; @@ -201,6 +226,8 @@ module EF_SPI_AHBL #( .tx_level(tx_level), .ss(ss), .enable(enable), + .done(done), + .busy(busy), .miso(miso), .mosi(mosi), .csb(csb), @@ -224,6 +251,7 @@ module EF_SPI_AHBL #( (last_HADDR[`AHBL_AW-1:0] == MIS_REG_OFFSET) ? MIS_REG : (last_HADDR[`AHBL_AW-1:0] == RIS_REG_OFFSET) ? RIS_REG : (last_HADDR[`AHBL_AW-1:0] == IC_REG_OFFSET) ? IC_REG : + (last_HADDR[`AHBL_AW-1:0] == GCLK_REG_OFFSET) ? GCLK_REG : 32'hDEADBEEF; assign HREADYOUT = 1'b1; diff --git a/hdl/rtl/bus_wrappers/EF_SPI_APB.pp.v b/hdl/rtl/bus_wrappers/EF_SPI_APB.pp.v index 1b26145..3c929a9 100644 --- a/hdl/rtl/bus_wrappers/EF_SPI_APB.pp.v +++ b/hdl/rtl/bus_wrappers/EF_SPI_APB.pp.v @@ -102,10 +102,10 @@ module EF_SPI_APB #( CDW = 8, FAW = 4 ) ( -`ifdef USE_POWER_PINS - inout VPWR, - inout VGND, -`endif + + + + input wire PCLK, input wire PRESETn, input wire PWRITE, @@ -140,27 +140,20 @@ module EF_SPI_APB #( localparam RIS_REG_OFFSET = 16'hFF08; localparam IC_REG_OFFSET = 16'hFF0C; - reg [0:0] GCLK_REG; - wire clk_g; - wire clk_gated_en = GCLK_REG[0]; - - `ifdef FPGA - wire clk = PCLK; - `else - (* keep *) sky130_fd_sc_hd__dlclkp_4 clk_gate( - `ifdef USE_POWER_PINS - .VPWR(VPWR), - .VGND(VGND), - .VNB(VGND), - .VPB(VPWR), - `endif - .GCLK(clk_g), - .GATE(clk_gated_en), - .CLK(PCLK) - ); - - wire clk = clk_g; - `endif + reg [0:0] GCLK_REG; + wire clk_g; + wire clk_gated_en = GCLK_REG[0]; + ef_gating_cell clk_gate_cell( + + + + // USE_POWER_PINS + .clk(PCLK), + .clk_en(clk_gated_en), + .clk_o(clk_g) + ); + + wire clk = clk_g; wire rst_n = PRESETn; @@ -190,6 +183,8 @@ module EF_SPI_APB #( wire [FAW-1:0] tx_level; wire [1-1:0] ss; wire [1-1:0] enable; + wire [1-1:0] done; + wire [1-1:0] busy; // Register Definitions wire [8-1:0] RXDATA_WIRE; @@ -217,13 +212,15 @@ module EF_SPI_APB #( else if(apb_we & (PADDR[16-1:0]==PR_REG_OFFSET)) PR_REG <= PWDATA[CDW-1:0]; - wire [6-1:0] STATUS_WIRE; + wire [8-1:0] STATUS_WIRE; assign STATUS_WIRE[0 : 0] = tx_empty; assign STATUS_WIRE[1 : 1] = tx_full; assign STATUS_WIRE[2 : 2] = rx_empty; assign STATUS_WIRE[3 : 3] = rx_full; assign STATUS_WIRE[4 : 4] = tx_level_below; assign STATUS_WIRE[5 : 5] = rx_level_above; + assign STATUS_WIRE[6 : 6] = busy; + assign STATUS_WIRE[7 : 7] = done; wire [FAW-1:0] RX_FIFO_LEVEL_WIRE; assign RX_FIFO_LEVEL_WIRE[(FAW - 1) : 0] = rx_level; @@ -310,17 +307,6 @@ module EF_SPI_APB #( assign IRQ = |MIS_REG; - reg [0:0] _miso_reg_[1:0]; - wire _miso_w_ = _miso_reg_[1]; - always@(posedge PCLK or negedge PRESETn) - if(PRESETn == 0) begin - _miso_reg_[0] <= 'b0; - _miso_reg_[1] <= 'b0; - end - else begin - _miso_reg_[0] <= miso; - _miso_reg_[1] <= _miso_reg_[0]; - end EF_SPI #( .CDW(CDW), .FAW(FAW) @@ -349,7 +335,9 @@ module EF_SPI_APB #( .tx_level(tx_level), .ss(ss), .enable(enable), - .miso(_miso_w_), + .done(done), + .busy(busy), + .miso(miso), .mosi(mosi), .csb(csb), .sclk(sclk) diff --git a/hdl/rtl/bus_wrappers/EF_SPI_APB.v b/hdl/rtl/bus_wrappers/EF_SPI_APB.v index 0d3e274..54e83c6 100644 --- a/hdl/rtl/bus_wrappers/EF_SPI_APB.v +++ b/hdl/rtl/bus_wrappers/EF_SPI_APB.v @@ -59,27 +59,20 @@ module EF_SPI_APB #( localparam RIS_REG_OFFSET = `APB_AW'hFF08; localparam IC_REG_OFFSET = `APB_AW'hFF0C; - reg [0:0] GCLK_REG; - wire clk_g; - wire clk_gated_en = GCLK_REG[0]; - - `ifdef FPGA - wire clk = PCLK; - `else - (* keep *) sky130_fd_sc_hd__dlclkp_4 clk_gate( - `ifdef USE_POWER_PINS - .VPWR(VPWR), - .VGND(VGND), - .VNB(VGND), - .VPB(VPWR), - `endif - .GCLK(clk_g), - .GATE(clk_gated_en), - .CLK(PCLK) - ); - - wire clk = clk_g; - `endif + reg [0:0] GCLK_REG; + wire clk_g; + wire clk_gated_en = GCLK_REG[0]; + ef_gating_cell clk_gate_cell( + `ifdef USE_POWER_PINS + .vpwr(VPWR), + .vgnd(VGND), + `endif // USE_POWER_PINS + .clk(PCLK), + .clk_en(clk_gated_en), + .clk_o(clk_g) + ); + + wire clk = clk_g; wire rst_n = PRESETn; @@ -107,6 +100,8 @@ module EF_SPI_APB #( wire [FAW-1:0] tx_level; wire [1-1:0] ss; wire [1-1:0] enable; + wire [1-1:0] done; + wire [1-1:0] busy; // Register Definitions wire [8-1:0] RXDATA_WIRE; @@ -128,13 +123,15 @@ module EF_SPI_APB #( assign clk_divider = PR_REG; `APB_REG(PR_REG, 'h2, CDW) - wire [6-1:0] STATUS_WIRE; + wire [8-1:0] STATUS_WIRE; assign STATUS_WIRE[0 : 0] = tx_empty; assign STATUS_WIRE[1 : 1] = tx_full; assign STATUS_WIRE[2 : 2] = rx_empty; assign STATUS_WIRE[3 : 3] = rx_full; assign STATUS_WIRE[4 : 4] = tx_level_below; assign STATUS_WIRE[5 : 5] = rx_level_above; + assign STATUS_WIRE[6 : 6] = busy; + assign STATUS_WIRE[7 : 7] = done; wire [FAW-1:0] RX_FIFO_LEVEL_WIRE; assign RX_FIFO_LEVEL_WIRE[(FAW - 1) : 0] = rx_level; @@ -201,17 +198,6 @@ module EF_SPI_APB #( assign IRQ = |MIS_REG; - reg [0:0] _miso_reg_[1:0]; - wire _miso_w_ = _miso_reg_[1]; - always@(posedge PCLK or negedge PRESETn) - if(PRESETn == 0) begin - _miso_reg_[0] <= 'b0; - _miso_reg_[1] <= 'b0; - end - else begin - _miso_reg_[0] <= miso; - _miso_reg_[1] <= _miso_reg_[0]; - end EF_SPI #( .CDW(CDW), .FAW(FAW) @@ -240,7 +226,9 @@ module EF_SPI_APB #( .tx_level(tx_level), .ss(ss), .enable(enable), - .miso(_miso_w_), + .done(done), + .busy(busy), + .miso(miso), .mosi(mosi), .csb(csb), .sclk(sclk) diff --git a/hdl/rtl/bus_wrappers/EF_SPI_WB.pp.v b/hdl/rtl/bus_wrappers/EF_SPI_WB.pp.v index 3e16ec2..aceda61 100644 --- a/hdl/rtl/bus_wrappers/EF_SPI_WB.pp.v +++ b/hdl/rtl/bus_wrappers/EF_SPI_WB.pp.v @@ -81,6 +81,10 @@ module EF_SPI_WB #( CDW = 8, FAW = 4 ) ( + + + + input wire ext_clk, input wire clk_i, input wire rst_i, @@ -93,10 +97,10 @@ module EF_SPI_WB #( output reg ack_o, input wire we_i, output wire IRQ, - input wire [1-1:0] miso, - output wire [1-1:0] mosi, - output wire [1-1:0] csb, - output wire [1-1:0] sclk + input wire [1-1:0] miso, + output wire [1-1:0] mosi, + output wire [1-1:0] csb, + output wire [1-1:0] sclk ); localparam RXDATA_REG_OFFSET = 16'h0000; @@ -115,7 +119,21 @@ module EF_SPI_WB #( localparam MIS_REG_OFFSET = 16'hFF04; localparam RIS_REG_OFFSET = 16'hFF08; localparam IC_REG_OFFSET = 16'hFF0C; - wire clk = clk_i; + + reg [0:0] GCLK_REG; + wire clk_g; + wire clk_gated_en = GCLK_REG[0]; + ef_gating_cell clk_gate_cell( + + + + // USE_POWER_PINS + .clk(clk_i), + .clk_en(clk_gated_en), + .clk_o(clk_g) + ); + + wire clk = clk_g; wire rst_n = (~rst_i); @@ -146,6 +164,8 @@ module EF_SPI_WB #( wire [FAW-1:0] tx_level; wire [1-1:0] ss; wire [1-1:0] enable; + wire [1-1:0] done; + wire [1-1:0] busy; // Register Definitions wire [8-1:0] RXDATA_WIRE; @@ -167,13 +187,15 @@ module EF_SPI_WB #( assign clk_divider = PR_REG; always @(posedge clk_i or posedge rst_i) if(rst_i) PR_REG <= 'h2; else if(wb_we & (adr_i[16-1:0]==PR_REG_OFFSET)) PR_REG <= dat_i[CDW-1:0]; - wire [6-1:0] STATUS_WIRE; + wire [8-1:0] STATUS_WIRE; assign STATUS_WIRE[0 : 0] = tx_empty; assign STATUS_WIRE[1 : 1] = tx_full; assign STATUS_WIRE[2 : 2] = rx_empty; assign STATUS_WIRE[3 : 3] = rx_full; assign STATUS_WIRE[4 : 4] = tx_level_below; assign STATUS_WIRE[5 : 5] = rx_level_above; + assign STATUS_WIRE[6 : 6] = busy; + assign STATUS_WIRE[7 : 7] = done; wire [FAW-1:0] RX_FIFO_LEVEL_WIRE; assign RX_FIFO_LEVEL_WIRE[(FAW - 1) : 0] = rx_level; @@ -197,6 +219,9 @@ module EF_SPI_WB #( assign tx_flush = TX_FIFO_FLUSH_REG[0 : 0]; always @(posedge clk_i or posedge rst_i) if(rst_i) TX_FIFO_FLUSH_REG <= 0; else if(wb_we & (adr_i[16-1:0]==TX_FIFO_FLUSH_REG_OFFSET)) TX_FIFO_FLUSH_REG <= dat_i[1-1:0]; else TX_FIFO_FLUSH_REG <= 1'h0 & TX_FIFO_FLUSH_REG; + localparam GCLK_REG_OFFSET = 16'hFF10; + always @(posedge clk_i or posedge rst_i) if(rst_i) GCLK_REG <= 0; else if(wb_we & (adr_i[16-1:0]==GCLK_REG_OFFSET)) GCLK_REG <= dat_i[1-1:0]; + reg [5:0] IM_REG; reg [5:0] IC_REG; reg [5:0] RIS_REG; @@ -269,6 +294,8 @@ module EF_SPI_WB #( .tx_level(tx_level), .ss(ss), .enable(enable), + .done(done), + .busy(busy), .miso(miso), .mosi(mosi), .csb(csb), diff --git a/hdl/rtl/bus_wrappers/EF_SPI_WB.v b/hdl/rtl/bus_wrappers/EF_SPI_WB.v index 53c36bd..966bca6 100644 --- a/hdl/rtl/bus_wrappers/EF_SPI_WB.v +++ b/hdl/rtl/bus_wrappers/EF_SPI_WB.v @@ -31,6 +31,10 @@ module EF_SPI_WB #( CDW = 8, FAW = 4 ) ( +`ifdef USE_POWER_PINS + inout VPWR, + inout VGND, +`endif `WB_SLAVE_PORTS, input wire [1-1:0] miso, output wire [1-1:0] mosi, @@ -54,7 +58,21 @@ module EF_SPI_WB #( localparam MIS_REG_OFFSET = `WB_AW'hFF04; localparam RIS_REG_OFFSET = `WB_AW'hFF08; localparam IC_REG_OFFSET = `WB_AW'hFF0C; - wire clk = clk_i; + + reg [0:0] GCLK_REG; + wire clk_g; + wire clk_gated_en = GCLK_REG[0]; + ef_gating_cell clk_gate_cell( + `ifdef USE_POWER_PINS + .vpwr(VPWR), + .vgnd(VGND), + `endif // USE_POWER_PINS + .clk(clk_i), + .clk_en(clk_gated_en), + .clk_o(clk_g) + ); + + wire clk = clk_g; wire rst_n = (~rst_i); @@ -82,6 +100,8 @@ module EF_SPI_WB #( wire [FAW-1:0] tx_level; wire [1-1:0] ss; wire [1-1:0] enable; + wire [1-1:0] done; + wire [1-1:0] busy; // Register Definitions wire [8-1:0] RXDATA_WIRE; @@ -103,13 +123,15 @@ module EF_SPI_WB #( assign clk_divider = PR_REG; `WB_REG(PR_REG, 'h2, CDW) - wire [6-1:0] STATUS_WIRE; + wire [8-1:0] STATUS_WIRE; assign STATUS_WIRE[0 : 0] = tx_empty; assign STATUS_WIRE[1 : 1] = tx_full; assign STATUS_WIRE[2 : 2] = rx_empty; assign STATUS_WIRE[3 : 3] = rx_full; assign STATUS_WIRE[4 : 4] = tx_level_below; assign STATUS_WIRE[5 : 5] = rx_level_above; + assign STATUS_WIRE[6 : 6] = busy; + assign STATUS_WIRE[7 : 7] = done; wire [FAW-1:0] RX_FIFO_LEVEL_WIRE; assign RX_FIFO_LEVEL_WIRE[(FAW - 1) : 0] = rx_level; @@ -133,6 +155,9 @@ module EF_SPI_WB #( assign tx_flush = TX_FIFO_FLUSH_REG[0 : 0]; `WB_REG_AC(TX_FIFO_FLUSH_REG, 0, 1, 1'h0) + localparam GCLK_REG_OFFSET = `WB_AW'hFF10; + `WB_REG(GCLK_REG, 0, 1) + reg [5:0] IM_REG; reg [5:0] IC_REG; reg [5:0] RIS_REG; @@ -201,6 +226,8 @@ module EF_SPI_WB #( .tx_level(tx_level), .ss(ss), .enable(enable), + .done(done), + .busy(busy), .miso(miso), .mosi(mosi), .csb(csb), diff --git a/ip/.gitignore b/ip/.gitignore new file mode 100644 index 0000000..83fe78c --- /dev/null +++ b/ip/.gitignore @@ -0,0 +1,3 @@ +* +!dependencies.json +!.gitignore diff --git a/ip/dependencies.json b/ip/dependencies.json new file mode 100644 index 0000000..bc86f84 --- /dev/null +++ b/ip/dependencies.json @@ -0,0 +1,7 @@ +{ + "IP": [ + { + "IP_Utilities": "v1.0.0" + } + ] +} \ No newline at end of file diff --git a/verify/uvm-python/Makefile b/verify/uvm-python/Makefile index a9ad402..88d7085 100644 --- a/verify/uvm-python/Makefile +++ b/verify/uvm-python/Makefile @@ -4,7 +4,7 @@ MODULE ?= top_module AHB_FILES ?= $(PWD)/../../hdl/rtl/bus_wrappers/EF_SPI_AHBL.pp.v APB_FILES ?= $(PWD)/../../hdl/rtl/bus_wrappers/EF_SPI_APB.pp.v WB_FILES ?=$(PWD)/../../hdl/rtl/bus_wrappers/EF_SPI_WB.pp.v -HDL_FILES ?= $(PWD)/IP_Utilities/rtl/aucohl_lib.v $(PWD)/IP_Utilities/rtl/aucohl_rtl.vh $(PWD)/../../hdl/rtl/spi_master.v $(PWD)/../../hdl/rtl/EF_SPI.v +HDL_FILES ?= $(PWD)/../../ip/IP_Utilities/rtl/aucohl_lib.v $(PWD)/../../ip/IP_Utilities/rtl/aucohl_rtl.vh $(PWD)/../../hdl/rtl/spi_master.v $(PWD)/../../hdl/rtl/EF_SPI.v VERILOG_SOURCES ?= $(PWD)/top.v $(AHB_FILES) $(APB_FILES) $(WB_FILES) $(HDL_FILES) RTL_MACROS += "" @@ -48,10 +48,6 @@ PDK_DIR = $(HOME)/.volare/volare/sky130/versions/bdc9412b3e468c102d01b7cf6337be0 PDK_FILES = $(PDK_DIR)/libs.ref/sky130_fd_sc_hd/verilog/primitives.v $(PDK_DIR)/libs.ref/sky130_fd_sc_hd/verilog/sky130_fd_sc_hd.v # get this from openlane logs in the future POST_SYS_FILES = $(PWD)/top.v $(VIP_FILES) $(PWD)/../../hdl/gl/synthesis/nl/$(DESIGN_NAME).nl.v -clone_ip_util := $(shell if [ ! -d "IP_Utilities" ]; then \ - echo "Cloning the IP_Utilities repository..."; \ - git clone https://github.com/shalan/IP_Utilities.git; \ -fi;) clone_ef_uvm := $(shell if [ ! -d "EF_UVM" ]; then \ echo "Cloning the EF_UVM repository..."; \ diff --git a/verify/uvm-python/spi_ref_model/spi_ref_model.py b/verify/uvm-python/spi_ref_model/spi_ref_model.py index b7d2827..1208dd8 100644 --- a/verify/uvm-python/spi_ref_model/spi_ref_model.py +++ b/verify/uvm-python/spi_ref_model/spi_ref_model.py @@ -9,7 +9,7 @@ from EF_UVM.ref_model.ref_model import ref_model from EF_UVM.bus_env.bus_item import bus_item from cocotb.triggers import Event -from cocotb.queue import Queue, QueueFull +from cocotb.queue import Queue, QueueFull, QueueEmpty class spi_ref_model(ref_model): @@ -100,7 +100,11 @@ def write_bus(self, tr): # pass value as it is until logic of ris is implemented pass elif td.addr == self.regs.reg_name_to_address["RXDATA"]: - td.data = self.fifo_rx.get_nowait() + try: + td.data = self.fifo_rx.get_nowait() + except QueueEmpty: + uvm_warning(self.tag, f"reading from rx while fifo is empty") + td.data = "xxxxxxxxxxxxxxxxxxxxxxxx00000000" else: td.data = data self.bus_bus_export.write(td) # this is output to the scoreboard diff --git a/verify/uvm-python/spi_seq_lib/configure_spi_seq.py b/verify/uvm-python/spi_seq_lib/configure_spi_seq.py index 90e5fe6..178526e 100644 --- a/verify/uvm-python/spi_seq_lib/configure_spi_seq.py +++ b/verify/uvm-python/spi_seq_lib/configure_spi_seq.py @@ -20,11 +20,15 @@ async def body(self): # Add the sequqnce here # you could use method send_req to send a write or read using the register name # example for writing register by value > 5 + await self.send_req(is_write=True, reg="CLKGATE", data_condition=lambda data: data == 1) await self.send_req( is_write=True, reg="CFG", - data_condition=lambda data: data in [0b00, 0b01, 0b10, 0b11], + # data_condition=lambda data: data in [0b00, 0b01, 0b10, 0b11], + data_condition=lambda data: data in [0b00, 0b01, 0b11], ) + await self.send_nop() + await self.send_nop() rsp = [] await self.get_response(rsp) # wait until writing is done diff --git a/verify/uvm-python/spi_seq_lib/spi_base_seq.py b/verify/uvm-python/spi_seq_lib/spi_base_seq.py index 29adc3d..09151c5 100644 --- a/verify/uvm-python/spi_seq_lib/spi_base_seq.py +++ b/verify/uvm-python/spi_seq_lib/spi_base_seq.py @@ -6,6 +6,8 @@ from cocotb.triggers import Timer from uvm.macros.uvm_sequence_defines import uvm_do_with, uvm_do import random +from uvm.macros import uvm_component_utils, uvm_fatal, uvm_info, uvm_error, uvm_warning +from uvm.base.uvm_object_globals import UVM_HIGH, UVM_LOW, UVM_MEDIUM class spi_base_seq(bus_seq_base): @@ -16,7 +18,7 @@ def __init__(self, name="spi_base_seq"): super().__init__(name) async def wait_tx_fifo_empty(self): - # wait until tx is empty + # wait until tx is empty and not busy self.clear_response_queue() while True: rsp = [] @@ -25,10 +27,10 @@ async def wait_tx_fifo_empty(self): rsp = rsp[0] if ( rsp.addr == self.regs.reg_name_to_address["STATUS"] - and rsp.data & 0b1 == 0b1 + and rsp.data & 0b1 == 0b1 and rsp.data & 0b1000000 == 0b0 ): break - + # wait until not busy cycles_additional = 8 * 4 for _ in range(cycles_additional): await self.send_nop() @@ -36,22 +38,27 @@ async def wait_tx_fifo_empty(self): async def wait_rx_fifo_not_empty(self): # wait received fifo not empty self.clear_response_queue() + await self.send_req( + is_write=True, + reg="TXDATA", + data_condition=lambda data: data == 0, + ) while True: rsp = [] self.clear_response_queue() await self.send_req(is_write=False, reg="STATUS") - await self.get_response(rsp) + while self.response_queue.size() != 0: + await self.get_response(rsp) + uvm_info(self.get_full_name(), f"RSP: {rsp}", UVM_HIGH) + if rsp[0].addr == self.regs.reg_name_to_address["STATUS"]: + break rsp = rsp[0] if ( rsp.addr == self.regs.reg_name_to_address["STATUS"] and rsp.data & 0b100 == 0b0 ): break - await self.send_req( - is_write=True, - reg="TXDATA", - data_condition=lambda data: data == 0, - ) + uvm_object_utils(spi_base_seq) diff --git a/verify/uvm-python/spi_seq_lib/spi_send_MISO_seq.py b/verify/uvm-python/spi_seq_lib/spi_send_MISO_seq.py index c3eec89..66c9f68 100644 --- a/verify/uvm-python/spi_seq_lib/spi_send_MISO_seq.py +++ b/verify/uvm-python/spi_seq_lib/spi_send_MISO_seq.py @@ -9,7 +9,7 @@ from uvm.macros import uvm_component_utils, uvm_fatal, uvm_info import random from spi_seq_lib.spi_base_seq import spi_base_seq - +import cocotb class spi_send_MISO_seq(spi_base_seq): # use this sequence write or read from register by the bus interface @@ -35,8 +35,11 @@ async def body(self): ) for _ in range(self.num_data): await self.wait_rx_fifo_not_empty() + if cocotb.plusargs["BUS_TYPE"] == "AHB": # since apb is so fast + await self.wait_rx_fifo_not_empty()# to make sure it's empty if random.random() < 0.7: # 20% probability of reading await self.send_req(is_write=False, reg="RXDATA") + uvm_info(self.tag, f"interation number {_}", UVM_MEDIUM) await self.send_req( is_write=True, reg="CTRL", data_condition=lambda data: data == 0b0