|  | 
|  |  | 
 |  
|  |  |  |  
|  |  | 
 |  
|  |  | Assertion with PSL |  
|  |  | Now that we have seen the example of FIFO assertion using OVL, let's see the example of using PSL to build assertions for the FIFO. PSL assertions can be coded in two ways. |  
|  |  | 
 |  
|  |  | 
  inline Coding : In this method, psl assertions are coded into verilog code as comments.   External File : In this method, psl assertions are coded into a separate file with vunit as name. |  
|  |  | 
 |  
|  |  | inline Coding |  
|  |  | 
 |  
|  |  | 
  All assertions appear within a consecutive series of comments appropriate for the context.  The first assertion statement line must begin with the psl keyword.  Both the psl keyword and the keyword that follows it must be on the same line.  Specify a label followed by a colon.  Assert the behavior described in the property by using the assert or assume keyword.  Describe the behavior of the design.  Assertions cannot be embedded in Verilog tasks, functions, or UDPs.  Example : // psl label: assert behavior; |  
|  |  | 
 |  
|  |  | External File |  
|  |  | 
 |  
|  |  | 
  To add assertions to an existing design without modifying the source text, as with legacy IP.  To experiment with assertions before embedding them in the source file.  When you are working in teams where the assertions are not created by the HDL author. |  
|  |  | 
 |  
|  |  | vunit verification_unit_name (module_name) {
  default clock = clock_edge;
  property_name: assert behavior;
  property_name: cover {behavior};
}
 |  
|  |  | 
 |  
|  |  | For more details refer to PSL usage guide that comes with the simulator. |  
|  |  | 
 |  
|  |  |  |  
|  |  | 
 |  
|  |  | Assertion in RTL |  
|  |  | In the code below, we use a psl assertion to check if no write is done when FIFO is full and also check if no read is done when FIFO is empty.  |  
|  |  | 
 |  
|  |  | We can code psl assertions inline with code with // or /* */. Before we write any assertion, we need to declare the clock as in the example. |  
|  |  | 
 |  
|  |  | ncverilog +assert verilog_file1.v verilog_file2.v |  
|  |  | 
 |  
|  |  | 
   1 //=============================================
   2 // Function  : Synchronous (single clock) FIFO
   3 //             With Assertion
   4 // Coder     : Deepak Kumar Tala
   5 // Date      : 31-October-2002
   6 //=============================================
   7 module syn_fifo (
   8 clk      , // Clock input
   9 rst      , // Active high reset
  10 wr_cs    , // Write chip select
  11 rd_cs    , // Read chipe select
  12 data_in  , // Data input
  13 rd_en    , // Read enable
  14 wr_en    , // Write Enable
  15 data_out , // Data Output
  16 empty    , // FIFO empty
  17 full       // FIFO full
  18 );    
  19  
  20 // FIFO constants
  21 parameter DATA_WIDTH = 8;
  22 parameter ADDR_WIDTH = 8;
  23 parameter RAM_DEPTH = (1 << ADDR_WIDTH);
  24 // Port Declarations
  25 input clk ;
  26 input rst ;
  27 input wr_cs ;
  28 input rd_cs ;
  29 input rd_en ;
  30 input wr_en ;
  31 input [DATA_WIDTH-1:0] data_in ;
  32 output full ;
  33 output empty ;
  34 output [DATA_WIDTH-1:0] data_out ;
  35 
  36 //-----------Internal variables-------------------
  37 reg [ADDR_WIDTH-1:0] wr_pointer;
  38 reg [ADDR_WIDTH-1:0] rd_pointer;
  39 reg [ADDR_WIDTH :0] status_cnt;
  40 reg [DATA_WIDTH-1:0] data_out ;
  41 wire [DATA_WIDTH-1:0] data_ram ;
  42 
  43 //-----------Variable assignments---------------
  44 assign full = (status_cnt == (RAM_DEPTH-1));
  45 assign empty = (status_cnt == 0);
  46 
  47 //-----------Code Start---------------------------
  48 always @ (posedge clk or posedge rst)
  49 begin : WRITE_POINTER
  50   if (rst) begin
  51    wr_pointer <= 0;
  52   end else if (wr_cs && wr_en ) begin
  53    wr_pointer <= wr_pointer + 1;
  54   end
  55 end
  56 
  57 always @ (posedge clk or posedge rst)
  58 begin : READ_POINTER
  59   if (rst) begin
  60     rd_pointer <= 0;
  61     data_out <= 0;
  62   end else if (rd_cs && rd_en ) begin
  63     rd_pointer <= rd_pointer + 1;
  64     data_out <= data_ram;
  65   end
  66 end
  67 
  68 always  @ (posedge clk or posedge rst)
  69 begin : READ_DATA
  70   if (rst) begin
  71     data_out <= 0;
  72   end else if (rd_cs && rd_en ) begin
  73     data_out <= data_ram;
  74   end
  75 end
  76 
  77 always @ (posedge clk or posedge rst)
  78 begin : STATUS_COUNTER
  79   if (rst) begin
  80     status_cnt <= 0;
  81   // Read but no write.
  82   end else if ((rd_cs && rd_en) &&  ! (wr_cs && wr_en) 
  83                 && (status_cnt  ! = 0)) begin
  84     status_cnt <= status_cnt - 1;
  85   // Write but no read.
  86   end else if ((wr_cs && wr_en) &&  ! (rd_cs && rd_en) 
  87                && (status_cnt  ! = RAM_DEPTH)) begin
  88     status_cnt <= status_cnt + 1;
  89   end
  90 end 
  91    
  92 ram_dp_ar_aw #(DATA_WIDTH,ADDR_WIDTH) DP_RAM (
  93 .address_0 (wr_pointer) , // address_0 input 
  94 .data_0    (data_in)    , // data_0 bi-directional
  95 .cs_0      (wr_cs)      , // chip select
  96 .we_0      (wr_en)      , // write enable
  97 .oe_0      (1'b0)       , // output enable
  98 .address_1 (rd_pointer) , // address_q input
  99 .data_1    (data_ram)   , // data_1 bi-directional
 100 .cs_1      (rd_cs)      , // chip select
 101 .we_1      (1'b0)       , // Read enable
 102 .oe_1      (rd_en)        // output enable
 103 );  
 104 
 105 // Add assertion here
 106 // psl default clock = (posedge clk);
 107 // psl ERRORwritefull: assert never {full && wr_en && wr_cs};
 108 // psl ERRORreadempty: assert never {empty && rd_en && rd_cs};
 109 
 110 endmodule
You could download file syn_fifo_psl.v here |  
|  |  | 
 |  
|  |  | Simulator Output |  
|  |  | Whenever there is a violation, an assertion is printed. |  
|  |  |  0 wr:0 wr_data:00 rd:0 rd_data:xx
 5 wr:0 wr_data:00 rd:0 rd_data:00
 10 wr:1 wr_data:00 rd:0 rd_data:00
 12 wr:1 wr_data:01 rd:0 rd_data:00
 14 wr:1 wr_data:02 rd:0 rd_data:00
 16 wr:1 wr_data:03 rd:0 rd_data:00
 18 wr:1 wr_data:04 rd:0 rd_data:00
 20 wr:1 wr_data:05 rd:0 rd_data:00
 22 wr:1 wr_data:06 rd:0 rd_data:00
 24 wr:1 wr_data:07 rd:0 rd_data:00
 ncsim: *E,ASRTST (syn_fifo_psl.v,107): (time 25 NS) 
 Assertion fifo_tb.fifo.ERRORwritefull has failed 
 26 wr:1 wr_data:08 rd:0 rd_data:00
 28 wr:1 wr_data:09 rd:0 rd_data:00
 30 wr:0 wr_data:09 rd:0 rd_data:00
 32 wr:0 wr_data:09 rd:1 rd_data:00
 33 wr:0 wr_data:09 rd:1 rd_data:08
 35 wr:0 wr_data:09 rd:1 rd_data:09
 39 wr:0 wr_data:09 rd:1 rd_data:03
 41 wr:0 wr_data:09 rd:1 rd_data:04
 43 wr:0 wr_data:09 rd:1 rd_data:05
 45 wr:0 wr_data:09 rd:1 rd_data:06
 47 wr:0 wr_data:09 rd:1 rd_data:07
 ncsim: *E,ASRTST (syn_fifo_psl.v,108): (time 49 NS) 
 Assertion fifo_tb.fifo.ERRORreadempty has failed 
 49 wr:0 wr_data:09 rd:1 rd_data:08
 ncsim: *E,ASRTST (syn_fifo_psl.v,108): (time 51 NS) 
 Assertion fifo_tb.fifo.ERRORreadempty has failed 
 51 wr:0 wr_data:09 rd:1 rd_data:09
 52 wr:0 wr_data:09 rd:0 rd_data:09
 Simulation complete via $finish(1) at time 152 NS + 0
 fifo_tb.v:36   #100 $finish;
 |  
|  |  | 
 |  
|  |  | Post Processing |  
|  |  | Like with any other simulation, we need to have post processing scripts to process the messages that are printed by the simulator to declare if the simulation has passed or failed. |  
|  |  | 
 |  
|  |  | 
 |  
|  |  | 
 |  
|  |  |  |  
|  |  | 
 |  |  | 
|  
 |  
 |  
 | 
| 
 | 
|    |  
| Copyright © 1998-2025 |  
| Deepak Kumar Tala - All rights reserved |  
| Do you have any Comment? mail me at:deepak@asic-world.com
 |  |