# Verilog Prof. Usagi # Turn a design into Verilog ## Verilog - Verilog is a Hardware Description Language (HDL) - Used to describe & model the operation of digital circuits. - Specify simulation procedure for the circuit and check its response simulation requires a logic simulator. - Synthesis: transformation of the HDL description into a physical implementation (transistors, gates) - · When a human does this, it is called logic design. - When a machine does this, it is called synthesis. - In this class, we use Verilog to implement and verify your processor. - C/Java like syntax #### Data types in Verilog - Bit vector is the only data type in Verilog - A bit can be one of the following - 0: logic zero - 1: logic one - X: unknown logic value, don't care - Z: high impedance, floating - Bit vectors expressed in multiple ways - 4-bit binary: 4 'b11\_10 (\_ is just for readability) - 16-bit hex: 16 ' h034f - 32-bit decimal: 32 'd270 ## **Operators** | Arithmetic | | Logical | | Bitwise | | Relational | | | |---------------|----------------|--------------------------------------------|-----|-------------|-------------|--------------------------------|---------------------------------------|--| | + | addition | ! | not | ~ | not | > | greater than | | | - | substraction | && | and | & | and | < | less than | | | * | multiplication | П | or | I | or | >= | greater or equal | | | / | division | | | ^ | xor | <= | less or equal | | | % | Don'truse | | | ~^ | xnor | == | equal (doesn't work if there is x, z) | | | ** | power | | | << | shift left | != | not equal | | | | | | | >> | shift right | === | really equal | | | Concatenation | | {} (e.g., {1b'1,1b'0} is 2b'10) | | Replication | | {{}} (e.g., {4{1b'0}} is 4b'0) | | | | | Conditional | condition ? value_if_true : value_if_false | | | | | | | #### Wire and Reg - wire is used to denote a hardware net "continuously assigned" values and do not store - single wire wire my\_wire; - array of wires wire[7:0] my\_wire; - reg is used for procedural assignments values that store information until the next value assignment is made. - again, can either have a single reg or an array reg[7:0] result; // 8-bit reg - reg is not necessarily a hardware register - you may consider it as a variable in C #### Revisit the 4-bit adder #### Half adder | Inp | out | Output | | | |-----|-----|--------|------|--| | Α | В | Out | Cout | | | 0 | 0 | 0 | 0 | | | 0 | 1 | 1 | 0 | | | 1 | 0 | 1 | 0 | | | 1 | 1 | 0 | 1 | | ``` Out = A'B + AB' Cout = AB ``` ``` module HA( input a, input b, input b, input b, output cout, output out ); assign out = (~a & b)|(a & ~b); assign cout = a&b; endmodule ``` #### Full adder | | npu | t | Output | | | |---|-----|-----|--------|------|--| | A | В | Cin | Out | Cout | | | 0 | 0 | 0 | 0 | 0 | | | 0 | 1 | 0 | 1 | 0 | | | 1 | 0 | 0 | 1 | 0 | | | 1 | 1 | 0 | 0 | 1 | | | 0 | 0 | 1 | 1 | 0 | | | 0 | 1 | 1 | 0 | 1 | | | 1 | 0 | 1 | 0 | 1 | | | 1 | 1 | 1 | 1 | 1 | | #### The Adder ``` module FA( input a, A_0B_0 A_2B_2 A<sub>3</sub>B<sub>3</sub> A_1B_1 input b, input cin, output cout, output out ); assign out = (~a\&b\&~cin)|(a\&~b\&~cin)|(~a\&~b\&cin)|(a\&b\&cin); Half Full Full assign cout = (a\&b\&\sim cin) | (\sim a\&b\&cin) | (a\&\sim b\&cin) | (a\&b\&cin);; Full endmodule Adder Adder Adder Adder module HA input a, input b, output cout, O_2 O<sub>1</sub> O_0 output out ); assign out = (~a \& b)|(a \& ~b); assign cout = a\&b; endmodule module adder( input[3:0] A; input[3:0] B, Connecting ports by name yields output[3:0] 0, clearer and less buggy code. output cout); wire [2:0] carries; HA ha0(.a(A[0]), .b(B[0]), .out(O[0]), .cout(carries[0])); FA fa1(.a(A[1]), .b(B[1]), .cin(carries[0]), .out(O[1]), .cout(carries[1])); FA fa2(.a(A[2]), .b(B[2]), .cin(carries[1]), .out(O[2]), .cout(carries[2])); FA fa3(.a(A[3]), .b(B[3]), .cin(carries[2]), .out(O[2]), .cout(cout); ``` endmodule ## Always block — combinational logic Executes when the condition in the sensitivity list occurs ## Always block — sequential logic Executes when the condition in the sensitivity list occurs ``` always@(posedge clk)// the following block only triggered by a positive clock begin ... end ``` #### Blocking and non-blocking - Inside an always block, = is a blocking assignment - assignment happens immediately and affect the subsequent statements in the always block - <= is a non-blocking assignment</p> - All the assignments happens at the end of the block - Assignment rules: - The left hand side, LHS, must be a reg. - The right hand side, RHS, may be a wire, a reg, a constant, or expressions with operators using one or more wires, regs, and constants. #### Initially, a = 2, b = 3 ``` reg a[3:0]; reg a[3:0]; reg b[3:0]; reg b[3:0]; reg c[3:0]; reg c[3:0]; always 0(*) always @(posedge clock) begin begin a = b; a <= b; c = a; c <= a; end end Afterwards: a = 3 and c = 3 Afterwards: a = 3 and c = 2 ``` #### "Always blocks" permit more advanced sequential idioms ``` module mux4( input a,b,c,d, input [1:0] sel, output out ); reg out; always 0(*) begin if ( sel == 2'd0 ) out = a; else if ( sel == 2'd1 ) out = b else if ( sel == 2'd2 ) out = c else if ( sel == 2'd3 ) out = d else out = 1'bx; end endmodule ``` ``` module mux4( input a,b,c,d, input [1:0] sel, output out ); reg out; always 0(*) begin case ( sel ) 2'd0 : out = a; 2'd1 : out = b; 2'd2 : out = c; 2'd3 : out = d; default : out = 1'bx; endcase end endmodule ``` #### **Initial block** Executes only once in beginning of the code initial begin • • • end ## Testing the adder! ``` `timescale 1ns/1ns // Add this to the top of your file to set time scale module testbench(); reg [3:0] A, B; reg C0; wire [3:0] S; wire C4; adder uut (.B(B), .A(A), .sum(S), .cout(C4)); // instantiate adder initial begin A = 4'd0; B = 4'd0; C0 = 1'b0; \#50 A = 4'd3; B = 4'd4; // wait 50 ns before next assignment \#50 A = 4'b0001; B = 4'b0010; // don't use \#n outside of testbenches end endmodule ``` ## Parameterize your module endmodule ## Coding guides - When modeling sequential logic, use nonblocking assignments. - When modeling latches, use nonblocking assignments. - When modeling combinational logic with an always block, use blocking assignments. - When modeling both sequential and combinational logic within the same always block, use nonblocking assignments. - Do not mix blocking and nonblocking assignments in the same always block. - Do not make assignments to the same variable from more than one always block. - Use \$strobe to display values that have been assigned using nonblocking assignments. - Do not make assignments using #0 delays.