# Verilog HDL **Testing & Verification** Dept. of Computer Science & Engg, IIT Kharagpur ### Pallab Dasgupta Professor, Dept. of Computer Science & Engg., Professor-in-charge, AVLSI Design Lab, Indian Institute of Technology Kharagpur ### **Agenda** - Structural Hardware Models - 4-Valued Logic - Delay - Instantiation - Wiring - Test Benches - Behavioral Models - Concurrency - Summary Source: *The Verilog Hardware Description Language*, By Thomas and Moorby, Kluwer Academic Publishers ### Representation: Structural Models #### □ Structural models - Are built from gate primitives and/or other modules - They describe the circuit using logic gates much as you would see in an implementation of a circuit. #### **□** Identify: - Gate instances, wire names, delay from a or b to f. - This is a multiplexor it selects one of n inputs (2 here) and passes it on to the output ### **Representation: Gate-Level Models** - Need to model the gate's: - **■** Function - Delay - □ Function - Generally, HDLs have built-in gate-level primitives - Verilog has NAND, NOR, AND, OR, XOR, XNOR, BUF, NOT, and some others - The gates operate on input values producing an output value - Typical Verilog gate instantiation is: optional and #delay instance-name (out, in1, in2, in3, ...); and #5 g1 (f1, a, nsel); a comma here let's you list other instance names and their port lists. ### **Four-Valued Logic** #### □ Verilog Logic Values - The underlying data representation allows for any bit to have one of four values - 1, 0, x (unknown), z (high impedance) - x one of: 1, 0, z, or in the state of change - z the high impedance output of a tri-state gate. ### **Four-Valued Logic** - What basis do these have in reality? - 0, 1 ... no question - z... A *tri-state* gate drives either a zero or one on its output ...and if it's not doing that, its output is high impedance (z). Tri-state gates are real devices and z is a *real* electrical affect. - x ... not a real value. There is no real gate that drives an x on to a wire. x is used as a debugging aid. x means the simulator can't determine the answer and so maybe you should worry! All values in a simulation start as x. #### □ BTW ... Verilog keeps track of more values than these in some situations. ### **Four-Valued Logic** - Logic with multi-level logic values - Logic with these four values make sense - Nand anything with a 0, and you get a 1. This includes having an x or z on the other input. That's the nature of the nand gate - Nand two x's and you get an x makes sense! - Note: z treated as an x on input. Their rows and columns are the same - If you forget to connect an input ... it will be seen as an z. - At the start of simulation, everything is an x. Innut D | | | input 6 | | | | |---------|------|---------|---|---|---| | | Nand | 0 | 1 | X | Z | | Input A | 0 | 1 | 1 | 1 | 1 | | | 1 | 1 | 0 | X | X | | | X | 1 | X | X | X | | | Z | 1 | X | X | X | A 4-valued truth table for a Nand gate with two inputs ### **Delay** - □ Transport delay input to output delay - "nand #35 (f1, a, b, c);" #35 is the transport delay - □ What if the input changes during that time? - i.e., how wide must an input spike be to affect the output? - Think of the gate as having inertia. The input change must be present long enough to get the output to change. (That "long enough" time is called inertial delay) - in Verilog, this time is equal to the transport delay ### Let's Build a Wider 2-bit MUX #### ☐ Build a 2-bit 2:1 MUX - OK, let's put two 1-bit 2:1 MUXes in the same module with a common select line - What would it look like? #### Reuse! #### □ Reuse of smaller objects - Can we use the MUX module that we already designed? - A big idea *instantiation* - Modules and primitive gates can be instantiated copied to many sites in a design - Previously, two ANDs, one OR, and a NOT gate were instantiated into module MUX - Now we instantiate two copies of module MUX into module wideMux ``` module wideMux (f1, f0, a1, a0, b1, b0, sel); input a1, a0, b1, b0, sel; output f1, f0; MUX bit1 (f1, a1, b1, sel), bit0 (f0, a0, b0, sel); endmodule ``` Instantiate two MUX modules, name them, and specify connections (the order is important). ## **Instantiation** — Copies - Modules and gate primitives are instantiated == copied - Note the word "copies" - The copies (also called *instances*) share the module (or primitive) definition - If we ever change a module definition, the copies will all change too - However, the internal entities (gate names, internal port names, and other things to come) are all private, separate copies ## **Instantiation** — Copies - Modules and gate primitives are instantiated == copied - Don't think of module instantiations as subroutines that are called - They are copies there are 2 MUX modules in wideMux with a total of: ``` 4 AND gates, ``` 2 OR gates, 2 NOT gates ### Why Is This Cool? - □ In Verilog - "Primitive" gates are predefined (NAND, NOR, ...) - Other modules are built by instantiating these gates - Other modules are built by instantiating other modules, ... - ☐ The design *hierarchy* of modules is built using instantiation - Bigger modules of useful functionality are defined - You can then design with these bigger modules - You can reuse modules that you've already built and tested - You can hide the detail why show a bunch of gates and their interconnection when you know it's a mux! - ☐ Instantiation & hierarchy control complexity. - No one designs 1M+ random gates they use hierarchy. - What are the software analogies? ## **How to Wire Modules Together** ### ☐ Real designs have many modules and gates ``` module putTogether (); wire w1, w2, w3, w4; bbb lucy (w1, w2, w3, w4); aaa ricky (w3, w2, w1); ... ``` what happens when out1 is set to 1? ``` module bbb (i1, i2, o1, clk); input i1, i2, clk; output o1; xor (o1, i2, ...); ... ``` Each module has it's own namespace. Wires connect elements of namespaces. ``` module aaa (in1, out1, out2); input in1; output out1, out2; ... nand #2 (out1, in1, b); nand #6 (out2, in1, b); ... ``` ## **Implicit Wires** - How come there were no wires declared in some of these modules? - Gate instantiations implicitly declare wires for their outputs. - All other connections must be explicitly declared as wires for instance, connections between module ports - Output and input declarations are wires ``` module putTogether (); wire w1, w2, w3, w4; mux inst1 (w1, w2, w3, w4); duh (w3, w2, w1); .... wires explicitly declared ``` ``` module mux (f, a, b, sel); output f; input a, b, sel; and #5 g1 (f1, a, nsel), g2 (f2, b, sel); or #5 g3 (f, f1, f2); not g4 (nsel, sel); endmodule wires implicitly declared (f1, f2, nsel) ``` ### **How to Build and Test a Module** - □ Construct a "test bench" for your design - Develop your hierarchical system within a module that has input and output ports (called "design" here) - Develop a separate module to generate tests for the module ("test") - Connect these together within another module ("testbench") ``` module testbench (); wire I, m, n; design d (I, m, n); test t (I, m); initial begin //monitor and display ... ``` ``` module design (a, b, c); input a, b; output c; ... ``` ``` module test (q, r); output q, r; initial begin //drive the outputs with signals ``` #### **Another View of This** **□** 3 chunks of Verilog, one for each of: ### **An Example** Module testAdd generated inputs for module halfAdd and displayed changes. Module halfAdd was the *design* ``` module tBench; wire su, co, a, b; halfAdd ad (su, co, a, b); testAdd tb (a, b, su, co); endmodule ``` ``` module halfAdd (sum, cOut, a, b); output sum, cOut; input a, b; xor #2 (sum, a, b); and #2 (cOut, a, b); endmodule ``` ``` module testAdd (a, b, sum, cOut); input sum, cOut; output a, b; a, b; reg initial begin $monitor ($time,, "a=%b, b=%b, sum=%b, cOut=%b", a, b, sum, cOut); a = 0; b = 0; #10 b = 1: #10 a = 1; #10 b = 0; #10 $finish; end endmodule ``` #### **The Test Module** - ☐ It's the test generator - □ \$monitor - prints its string when executed. - after that, the string is printed when one of the listed values changes. - only one monitor can be active at any time - prints at end of current simulation time - □ Function of this tester - at time zero, print values and set a=b=0 - after 10 time units, set b=1 - after another 10, set a=1 - after another 10 set b=0 - then another 10 and finish ``` module testAdd(a, b, sum, cOut); input sum, cOut; output a, b; a, b; req initial begin $monitor ($time., "a=%b, b=%b, sum=%b, cOut=%b", a, b, sum, cOut); a = 0; b = 0; #10 b = 1; #10 a = 1; #10 b = 0; #10 $finish; end endmodule ``` ### **Another Version of a Test Module** #### Multi-bit "thingies" - test is a two-bit register and output - It acts as a two-bit number (counts 00-01-10-11-00...) - Module tBench needs to connect it correctly mod halfAdd has 1-bit ports. ``` module tBench; wire su, co; wire [1:0] t; halfAdd ad (su, co, t[1], t[0]); testAdd tb (t, su, co); endmodule ``` ``` module testAdd (test, sum, cOut); input sum, cOut; output [1:0] test; [1:0] test; reg initial begin $monitor ($time., "test=%b, sum=%b, cOut=%b", test, sum, cOut); test = 0; #10 \text{ test} = \text{test} + 1; #10 \text{ test} = \text{test} + 1; #10 \text{ test} = \text{test} + 1; #10 $finish; end endmodule ``` Connects bit 0 or wire t to this port (b of the module halfAdder) #### Another Version of testAdd - Other procedural statements - You can use "for", "while", "if-thenelse" and others here. - This makes it easier to write if you have lots of input bits. ``` module tBench; wire su, co; wire [1:0] t; halfAdd ad (su, co, t[1], t[0]); testAdd tb (t, su, co); endmodule ``` ``` module testAdd (test, sum, cOut); input sum, cOut; output [1:0] test: reg [1:0] test: initial begin $monitor ($time,, "test=%b, sum=%b, cOut=%b", test, sum, cOut); for (test = 0; test < 3; test = test + 1) #10; #10 $finish; end endmodule ``` hmm... "<3" ... ? ### Structural Vs. Behavioral Models - □ Structural model - Just specifies primitive gates and wires - i.e., the structure of a logical netlist - You basically know how to do this now. - □ Behavioral model - More like a procedure in a programming language - Still specify a module in Verilog with inputs and outputs... - ...but inside the module you write code to tell what you want to have happen, NOT what gates to connect to make it happen - i.e., you specify the behavior you want, not the structure to do it - Why use behavioral models - For testbench modules to test structural designs - For high-level specs to drive logic synthesis tools ### **How Do Behavioral Models Fit In?** - How do they work with the event list and scheduler? - Initial (and always) begin executing at time 0 in arbitrary order - They execute until they come to a "#delay" operator - They then suspend, putting themselves in the event list 10 time units in the future (for the case at the right) - At 10 time units in the future, they resume executing where they left off. - Some details omitted - ...more to come ``` module testAdd (a, b, sum, cOut); input sum, cOut; output a, b; rea a, b; initial begin $monitor ($time,, "a=%b, b=%b, sum=%b, cOut=%b", a, b, sum, cOut); a = 0; b = 0; #10 b = 1; #10 a = 1; #10 b = 0; #10 $finish; end endmodule ``` ## **Concurrent Activity** Eval g2, g3 - □ Do these two evaluations happen at the same time? - Yes and No! Yes and No! - ☐ Yes ... - They happen at the same *simulated* (or virtual) time - After all, the time when they occur is 27 - □ No ... - We all know the processor is only doing one thing at any given time - So, which is done first? - That's undefined. We can't assume anything except that the order is arbitrary. ## **Concurrent Activity** - □ The point is - In the real implementation, all activity will be concurrent - Thus the simulator models the elements of the system as being concurrent in simulated time - The simulator stands on its head trying to do this! - ☐ Thus, - Even though the simulator executes each element of the design one at a time ... - ... we'll call it concurrent ### **Behavioral Verilog HDL codes** ``` module module_name(port_names); input [port_size] input_port_names; output [port_size] output_port_names; wire [wire_size] wire_names; reg [reg_size] reg_names; always @(sensitivity list) ``` endmodule #### <u>Multiplexer</u> ``` module mux (f, a, b, sel); input [3:0] a, b; input sel; output [3:0] f; reg [3:0] f; always @(a or b or sel) if (sel) f=b; else f=a; endmodule ``` behavioral statements ## Flip-flop Design: An Example ``` module DFF(d, q, qbar, clk, reset); input d, clk, reset; output q, qbar; reg q, qbar; always @(posedge clk or posedge reset) begin if (sel) begin q = 1'b0; qbar = 1'b1; end else begin q = d; qbar = \sim d; end end endmodule ``` ### **Behavioral Statements** - □ Continuous assignment statements - using assign - □ Procedural assignment statements - Blocking assignment (using =) - Non blocking assignment (using <=)</p> #### **Blocks Statements** - Sequential Block Statements: - Sequential block is a group of statements between a *begin* and an *end*. - A sequential block, in an always statement executes repeatedly - Inside an *initial* statement, it operates only once - Parallel Block Statements: - Statements are enclosed within fork \_\_\_\_ \_\_\_\_ join ### **Block statements:** *Examples* ``` always @(a or b or c); begin #5 d = a+b; #10 e = a-c; #15 f = b+c; end initial begin #5 d = a+b; #10 e = a-c; #15 f = b+c; end ``` ``` always @(a or b or c); fork #5 d = a+b; #10 e = a-c; #15 f = b+c; join ``` ## **Examples** □ Blocking: ``` always @(A1 or B1 or C1 or M1) begin M1 = #3 (A1 & B1); Y1 = #1 (M1 | C1); end ``` - ☐ Statement executed at time tcausing M1 to be assigned at t+3 - □ Statement executed at time t+3 causing Y1 to be assigned at time t+4 ■ Non-blocking: always @(A2 or B2 or C2 or M2) begin M2 <= #3 (A2 & B2); Y2 <= #1 (M1 | C1); end - Statement executed at time t causing M2 to be assigned at t+3 - □ Statement executed at time t causing Y2 to be assigned at time t+1. Uses old values. ### **Example:** *Implementation* #### ■ Blocking Assignment ``` module BA(clk, a, b, c) input clk, a, b; output c; reg b, c; always @(posedge clk) begin b=a; c=b; end endmodule ``` ### **Example:** *Implementation* #### Non Blocking Assignment ``` module NBA(clk, a, b, c) input clk, a, b; output c; reg b, c; always @(posedge clk) begin b<=a; c<=b; end endmodule ``` ## **Design using Functions and Tasks** ``` Function module m_name( port_declarations) begin ret_val = func_name(arguments); end function func_name; input declaration variable declaration begin <statements> end endfunction endmodule ``` ``` □ Task module m_name( port_declarations) begin task_name(arguments); end task task_name; input declaration output declaration variable declaration begin <statements> end endtask endmodule ``` ## **FSM Design using Verilog HDL** ``` module parity (clk, reset, i, o); input clk, reset, i; output 0; reg st, next st, o; parameter st_even = 0, st_odd = 1; always @ (posedge clk or posedge reset) begin if (reset == 1) st <= st even: else st <= next_st; end /* State transitions */ /* Output computation */ endmodule ``` ### **State Transitions & O/P computations** ``` // State Transitions always @ (i or st) begin if (i==1) begin if (st == st_even) next st = st odd; else next_st = st_even; end else next_st = st; end // Output Computation always @(st) begin if (st == st_even) o=0; else o=1; end ```