MinUk.Dev
verilog (베릴로그) - minuk dev wiki

verilog (베릴로그)

created : Thu, 25 Jun 2020 20:46:51 +0900
modified : Wed, 01 Jul 2020 20:53:18 +0900

개요

  • opennvm가 verilog가 짜져있어서 읽을수 있을 수준까지 공부하는게 목표

출처

기본 문법

module module_name(port_list);

port, reg, wire, parameter declaration
submodule instance
primitive gate
always, initial
assign,
funciton, task declaration
function, task definition

endmodule

Hello World

module main;
  initial
  begin
    $display("Hello World");
    $finish;
  end
endmodule

D flip-flop Code

module d_ff (d, clk, q, q_bar);
input d, clk;
output q, q_bar;
wire d, clk;
reg q, q_bar;

always @ (posedge clk)

begin
  q <= d;
  q_bar <= !d;
end

endmodule

Data Type (English)

  • Hardware does have two kinds of drivers.
  • A driver is a data type which can drive a load. Basically, in a physical circuit, a driver would be anything that electrons can move through/into.
    • Driver that can store a value (example: flip-flop).
    • Driver that can not store value, but connects two points (example: wire).
  • The first type of driver is called a reg in Verilog (short for “register”). The Second data type is called a wire (for… well, “wire”). You can refer to tidbits section to understand it better.

Data Type (한국어)

  • 하드웨어는 2가지 종류의 드라이버가 있으며, 기본적으로 물리적인 회로이며, 드라이버는 전자가 들어오거나 나갈수 있는 무엇이든 될 수 있다.
    • 드라이버는 값을 저장할 수 있다. (예를 들어 플립플롯) -> reg(register의 준말)
    • 드라이버는 값을 저장하지 못하는 대신, 두 지점을 연결 시킬수 있다. (예를 들어 wire) -> wire

Opertators

Operator TypeOperator SymbolOperation Performed
Arithmetic*Multiply
/Division
+Add
-Subtract
%Modulus
+Unary plus
-Unary minus
Logical!Logical negation
&&Logical and
`
>Greater than
<Less than
>=Greater than or equal
<=Less than or equal
Equality==Equality
!=inequality
Reduction~Bitwize negation
~&nand
``
`~`
^xor
^~xnor
~^xnor
Shift>>Right shift
<<Left shift
Concatenation{}Concatenation
Conditional?conditional

Control Statements

If-else

// begin and end act like curly braces in C/C++.
if (enable == 1'b1) begin
  data = 10;         // Decimal assigned
  address = 16'hDEAD; // Hexadecimal
  wr_enable = 1'b1; // Binary
end else begin
  data = 32'b0;
  wr_enable = 1'b0;
  address = address +1;
end

Case

case(address)
  0: $display ("It is 11:40PM");
  1: $display ("I am feeling sleepy");
  2: $display ("Let me skip this tutorial");
  default: $display ("Need to complete");
endcase

while

while (free_time) begin
  $display ("Continue with webpage development");
end

Counter example

module counter (clk, rst, enable, count);
input clk, rst, enabl;
output [3:0] count;
reg [3:0] count;

always @ (posedge clk or posedge rst)
if (rst) begin
  count <= 0;
end else begin: COUNT
  while (enable) begin
    count <= count + 1;
    disable COUNT;
  end
end

endmodule

For loop

for (i = 0; i < 15; i = i + 1) begin
  $display ("Current value of i is %d", i);
end

Repeat

repeat (16) begin
  $display ("Current value of i is %d", i);
  i = i + 1;
end

Initial Blocks

initial begin
  clk = 0;
  reset = 0;
  req_0 = 0;
  req_1 = 0;
end

Always Blocks

always @ (a or b or sel)
begin
  y = 0;
  if (sel == 0) begin
    y = a;
  end else begin
    y = b;
  end
end
  • ?? 이건 근데 mux랑 똑같은거 아닌가? mux 만들어 놓고, initial block 에서 wire 연결 해두면 안되는 건가?
  • always block의 쓰임을 잘 모르겠네, 어떻게 되는 거지?
  • => 주변에 있는 다른 분께 물어보니 같은 동작을 하는건 맞고, 실제 기판으로 나오는게 똑같으면 어떻게 코딩하든 상관 없다고 하심.
always begin
  #5 clk = ~clk;
end

Assign Statement

assign out = (enable) ? data : 1'bz;
assign out = data;

Task and Function

function parity;
input [31:0] data;
integer i;
begin
  parity = 0;
  for (i = 0; i< 32; i = i + 1) begin
    parity = parity ^ data[i];
  end
end
endfunction
  • 흠… for을 도는건 delay 없이 도는 거니 위에 내용은 실제로 컴파일 되면 data의 모든 bit를 한번에 xor하는 결과가 나오는건가?
  • 그래서 한번 만들고 compile 해봤다.
 
module parity (d, p);
  output reg p;
  input [31:0] d;

  function parity;
    input [31:0] data;
    integer i;

    begin
      parity = 0;
      for (i = 0; i< 32; i = i + 1) begin
        parity = parity ^ data[i];
      end
    end
  endfunction

  always @ (d)
    p = parity(d);

endmodule
  • 아래는 컴파일 된걸 연 결과다
#! /usr/bin/vvp
:ivl_version "10.1 (stable)";
:ivl_delay_selection "TYPICAL";
:vpi_time_precision + 0;
:vpi_module "system";
:vpi_module "vhdl_sys";
:vpi_module "v2005_math";
:vpi_module "va_math";
S_0x56211aafb800 .scope module, "parity" "parity" 2 1;
 .timescale 0 0;
    .port_info 0 /INPUT 32 "d"
    .port_info 1 /OUTPUT 1 "p"
o0x7f15420240a8 .functor BUFZ 32, C4<zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz>; HiZ drive
v0x56211ab1be30_0 .net "d", 31 0, o0x7f15420240a8;  0 drivers
v0x56211ab1bf30_0 .var "p", 0 0;
E_0x56211aac3cc0 .event edge, v0x56211ab1be30_0;
S_0x56211aafb9c0 .scope function, "parity" "parity" 2 5, 2 5 0, S_0x56211aafb800;
 .timescale 0 0;
v0x56211aafbbb0_0 .var "data", 31 0;
v0x56211ab1bcb0_0 .var/i "i", 31 0;
v0x56211ab1bd90_0 .var "parity", 0 0;
TD_parity.parity ;
    %pushi/vec4 0, 0, 1;
    %store/vec4 v0x56211ab1bd90_0, 0, 1;
    %pushi/vec4 0, 0, 32;
    %store/vec4 v0x56211ab1bcb0_0, 0, 32;
T_0.0 ;
    %load/vec4 v0x56211ab1bcb0_0;
    %cmpi/s 32, 0, 32;
    %jmp/0xz T_0.1, 5;
    %load/vec4 v0x56211ab1bd90_0;
    %load/vec4 v0x56211aafbbb0_0;
    %load/vec4 v0x56211ab1bcb0_0;
    %part/s 1;
    %xor;
    %store/vec4 v0x56211ab1bd90_0, 0, 1;
    %load/vec4 v0x56211ab1bcb0_0;
    %addi 1, 0, 32;
    %store/vec4 v0x56211ab1bcb0_0, 0, 32;
    %jmp T_0.0;
T_0.1 ;
    %end;
    .scope S_0x56211aafb800;
T_1 ;
    %wait E_0x56211aac3cc0;
    %load/vec4 v0x56211ab1be30_0;
    %store/vec4 v0x56211aafbbb0_0, 0, 32;
    %fork TD_parity.parity, S_0x56211aafb9c0;
    %join;
    %load/vec4  v0x56211ab1bd90_0;
    %store/vec4 v0x56211ab1bf30_0, 0, 1;
    %jmp T_1;
    .thread T_1, $push;
# The file index is used to find the file name in the following table.
:file_names 3;
    "N/A";
    "<interactive>";
    "parity.v";
  • 읽어보면, 실제로 loop돌고 jump 하는 것처럼 보인다.
  • 생각해보면 그럴수 밖에 없는 것 같기도 하고, 만약 1번에 끝내길 바랬으면 다르게 짜야지 싶기도 해서

Task Benches

module arbiter (
clock,
reset,
req_0,
req_1,
gnt_0,
gnt_1
);

input clock, reset, req_0, req_1;
output gnt_0, gnt_1;

reg gnt_0, gnt_1;

always @ (posedge clock or posedge reset)

if (reset) begin
  gnt_0 <= 0;
  gnt_1 <= 0;
end else if (req_0) begin
  gnt_0 <= 1;
  gnt_1 <= 0;
end else if (req_1) begin
  gnt_0 <= 0;
  gnt_1 <= 1;
end

endmodule
// Testbench Code Goes here
module arbiter_tb;

reg clock, reset, req0, req1;
wire gnt0, gnt1;

initial begin
  $monitor ("req0=%b,req1=%b,gnt0=%b,gnt1=%b", req0, req1, gnt0, gnt1);
  clock = 0;
  reset = 0;
  req0 = 0;
  req1 = 0;
  #5 reset = 1;
  #15 reset = 0;
  #10 req0 = 1;
  #10 req0 = 0;
  #10 req1 = 1;
  #10 req1 = 0;
  #10 {req0, req1} = 2'b11;
  #10 {req0, req1} = 2'b00;
  #10 $finish;
end

always begin
  #5 clock = !clock;
end

arbiter U0 (
.clock (clock),
.reset (reset),
.req_0 (req0),
.req_1 (req1),
.gnt_0 (gnt0),
.gnt_1 (gnt1)
);


endmodule
  • 처음에 initial block 에서 monitor로 변수 값 바뀔때 tracking 해주고, req0 = 0, req1 = 0 이 설정 되고, gnt는 설정이 안되서 0, 0, x, x 가 나오고 나머지는 다 delay 걸려서 실행 안되고 있고, arbiter 호출해준다.
  • 5 delays 뒤에 reset 이 1 이 되면서 gnt_0, gnt_1 이 되면서 0, 0, 0, 0 이 출력
  • 10 delays 뒤에 req0 가 1이 되면서, 1, 0, 0, 0
  • 그러면서 1, 0, 1, 0
  • 그 뒤 req0 이 0 되면서 0, 0, 1 , 0
  • req1 이 1 이 되면서 0, 1, 1, 0
  • 그 직후 0, 1, 0, 1
  • 0, 0, 0, 1
  • 1, 1, 0, 1
  • 1, 1, 1, 0
  • 0, 0, 1, 0
  • 순으으로 나오게 된다.
  • 처음에는 reset 이 0 으로 바뀌는걸 못봐서 한참 삽질했다.

Counter Design

//-----
// Function : This is a 4 bit up-counter with
// Synchronous active high reset and
// with active high enable signal
//-----
module first_counter (
clock,
reset,
enable,
counter_out
);

input clock;
input reset;
input enable;

output [3:0] counter_out;

wire clock;
wire reset;
wire enable;

reg [3:0] counter_out;

always @ (posedge clock)
begin: COUNTER // Block Name
  if (reset == 1'b1) begin
    counter_out <= # 4'b0000;
  end
  
  else if (enable == 1'b1) begin
    counter_out <= #1 counter_out + 1;
  end
end

endmodule
`include "first_counter.v"
module first_counter_tb();
// Declare inputs as regs and outputs as wires
reg clock, reset, enable;
wire [3:0] counter_out;

// Initialize all variables

initial begin
  $display ("time\t clk reset enable counter");
  $monitor ("%g\t %b %b %b %b");
    $time, clock, reset, enable, counter_out;
    
  clock = 1;
  reset = 0;
  enable = 0;
  #5 reset = 1;
  #10 reset = 0;
  #10 enable = 1;
  #100 enable = 0;
  #5 $finish;
end

always begin
  #5 clock = ~clock;
end

first_counter U_counter (
clock,
reset,
enable,
counter_out
);

endmoudle
  • 다른 module 을 부를때 `include 라는 문법을 쓰는걸 알게 되었다.
  • 나머진 예전에 배운 논리회로랑 똑같아서 스킵

Comments

/* This is a
  Multi line comment
  example */
module addbit (
a,
b,
ci,
sum,
co);

input a;
input b;
input ci;
output sum;
output co;
wire a;
wire b;
wire ci;
wire sum;
wire co;

endmodule

Numbers in Verilog

Integer Number

  • Syntax : ’;
IntegerStored as
100000000000000000000000000000001
8’hAA10101010
6’b10_0011100011
‘hF00000000000000000000000000001111

Real Numbers

  • Syntax : ., E

Modules

Ports

  • Syntax
input [range_val:range_var] list_of_identifier;
output [range_val:range_var] list_of_identifier;
inout [range_val:range_var] list_of_identifier;
  • Examples
input clk;
input [15:0] data_in;
output [&;0] count;
inout data_bi;
module addbit(
a,
b,
ci, sum,
co
);
input a;
input b;
input ci;
output sum;
output co;

wire a;
wire b;
wire ci;
wire sum;
wire co;

assign {co, sum} = a + b + ci;

endmodule
Modules connected by port order (implicit)
module adder_implicit (
result,
carry,
r1,
r2,
ci
);

input [3:0] r1;
input [3:0] r2;
input ci;

output [3:0] result;
ouput carry;

wire [3:0] r1;
wire [3:0] r2;
wire ci;
wire [3:0] result;
wire crarry;

wire c1;
wire c2;
wire c3;

addbit u0 (
r1[0],
r2[0],
ci ,
result[0],
c1
);

addbit u1 (
r1[1] ,
r2[1] ,
c1,
result[1],
c2
);

addbit u2 (
r1[2] ,
r2[2], 
c2,
result[2],
c3
);

addbit u3(
r1[3],
r2[3],
c3,
result[3],
carry
);

endmodule
Modules connected by name
module adder_explicit (
result,
carry,
r1,
r2,
ci
);

input [3:0] r1;
input [3:0] r2;
input ci;

ouput [3:0] result;
ouput carry;

wire [3:0] r1;
wire [3:0] r2;
wire ci;
wire [3:0] result;
wire carry;

wire c1;
wire c2;
wire c3;

addbit u0 (
.a (r1[0]),
.b (r2[0]),
.ci (ci),
.sum (result[0]),
.sum (result[0]),
co (c1)
);

addbit u1 (
.a (r1[1]),
.b (r2[1]),
.ci (c1),
.sum (result[1]),
.co (c2)
);

addbit u2 (
.a (r1[2]),
.b (r2[2]),
.ci (c2),
.sum (result[2]),
.co (3)
);

addbit u3 (
.a (r1[3]),
.b (r2[3]),
.ci (c3),
.sum (result[3]),
.co (carry)
);

endmodule
Instantiating a module
module parity (
a,
b,
c,
d,
y
);

input a;
input b;
input c;
input d;

ouput y;

wire a;
wire b;
wire c;
wire d;
wire y;

wire out_0;
wire out_1;

xor u0 (out_0, a, b);
xor u1 (out_1, c, d);

xor u2 (y, out_0, out_1);

endmodule

Port Connection Rules

  • Inputs : internally must always be of type net, externally the inputs can be connected to a variable of type reg or net.
  • Outputs : internally can be of type net or reg, externally the outputs must be connected to a variable of type net.
  • Inouts : internally or externally must always be type net, can only be connected to a variable net type.
  • Width matching : it is legal to connect internal and external ports of different sizes. But beware, synthesis tools could report problems.
  • Unconnected ports : unconnected ports are allowed by using a “,”.
  • The net data types are used to connect structure.
  • A net data type is required if a signal can be driven a structural connection.
Example - Implicit Unconnected Port
module implicit();
reg clk, d, rst, pre;
wire q;

dff u0 (q,, clk, d, rst, pre);

end module

module dff (q, q_bar, clk, d, rst, pre);
input clk, d, rst, pre;
output q, q_bar;
reg q;

assign a_bar = ~q;

always @ (posedge clk)

if 9rst ==1'b1) gegin
  q <= 0;
end else if (pre == 1'b1) begin
  q <= 1;
end else being
  q <= d;
end

endmodule
Example - Explicit Unconnected Port
module explicit();
reg clk, d, rst, pre;
wire q;

dff u0 (
.q (q),
.d (d),
.clk (clk),
.q_bar (),
.rst (rst),
.pre (pre)
);

endmodule


module dff (q, q_bar, clk, d, rst, pre);
input clk, d, rst, pe;
output q, q-bar;
reg q;

assign q_bar = ~q;

always @ (posedge clk)

if (rst == 1'b1) begin
  q <= 0;
end else if (pre == 1'b1) begin
  q <= 1;
end else begin
  q <= d;
end

endmodule

Hierarchical Identifiers

`include "addbit.v"
module adder_hier (
result,
carry,
r1,
r2,
ci
);

input [3:0] r1;
input [3:0] r2;
input ci;

output [3:0] result;
output carry;

wire [3:0] r1;
wire [3:0] r2;
wire ci;
wire [3:0] result;
wire carry;

wire c1;
wire c2;
wire c3;


addbit u0 (r1[0], r2[0],ci,result[0],c1);
addbit u1 (r1[1], r2[1],c1,result[1],c2);
addbit u2 (r1[2], r2[2],c2,result[2],c3);
addbit u3 (r1[3], r2[3],c3,result[3],carry);

endmodule

module tb();

reg [3:0] r1, r2;
reg ci;
wire [3:0] result;
wire carry;

initial begin
  r1 = 0;
  r2 = 0;
  ci = 0;
  #10 r1 = 10;
  #10 r2 = 2;
  #10 ci = 1;
  #10 $display("+-----------------+");
  $finish;
end


adder_hier U (result, carry, r1, r2, ci);

initial begin
  $display("+---------------+");
  $display("|r1|r2|ci|u0.sum|u1.sum|u2.sum|u3.sum|");
  $display("+---------------+");
  $monitoer("|%h|%h|%h|%h|h|%h|%h|",
    r1, r2, ci, tb.U.u0.sum, tb.U.u1.sum, tb.U.u2.sum, tb.U.u3.sum);
end

endmodule

Data Types

  • Verilog Language has two primary data types:
    • Nets - Present structural connections between components.
    • Registers - represent variables used to store data.
  • Every signal has a data type associated with it:
    • Explicitly declared with a declaration in your Verilog code.
    • Implicitly declared with no declaration when used to connect structural building blocks in your code. Implicit declaration is always a net type “wire” and is one bit wide.
Types of Nets
Net Data TypeFunctionality간략 번역
wire, triInterconnecting wire - nospecial resolution function상호 연결
wor, triorWired outputs Or together (models ECL)연결된 것들끼리 or한 값
wand, triandWired outputs And together (models open-collector)연결된 것들끼리 and 한 값
tri0, tri1Net pulls-down or pulls-up when not driventri와 동일한데, z 일때 0이거나 1
supply0, supply1Net has a constant logic 0 or logic 1 (supply strenth)상수 ground, vdd
triregRetains last value, when driven by z(tristate).z일때 예전 값을 기억
Example - wor
module test_wor();
wor a;
reg b, c;
assign a = b;
assign a = c;

initial begin
  $monitor("%g a = %b b = %b c = %b", $time, a, b, c);
  #1 b = 0;
  #1 c = 0;
  #1 b = 1;
  #1 b = 0;
  #1 c = 1;
  #1 b = 1;
  #1 b = 0;
  #1 $finish;
end

endmoudle
Example - tri
module test_tri();

tri a;
reg b, c;


assign a = (b) ? c : 1'bz;

initial begin
  $monitor("%g a = %b b = %b c= %b", $time, a, b, c);
  b = 0;
  c = 0;
  #1 b = 1;
  #1 b = 0;
  #1 c = 1;
  #1 b = 1;
  #1 b = 0;
  #1 $finish;
end

endmodule
Example - trireg
module test_trireg();
trireg a;
reg b, c;

assign a = (b) ? c : 1'bz;

initial begin
  $monitor("%g a = %b b = %b c= %b", $time, a, b, c);
  b = 0;
  c = 0;
  #1 b = 1;
  #1 b = 0;
  #1 c = 1;
  #1 b = 1;
  #1 b = 0;
  #1 $finish;
end

endmodule

Register Data Types

  • Registers store the last value assigned to them until another assignment statement changes their value. (새로운 할당 전까지 그 값이 변하지 않고 저장한다.)
  • Registers represent data storage constructs. (register는 data구성을 표현한다?)
  • You can create regs arrays called memories. (memories 라고도 불리는 regs array를 만들수도 있다.)
  • register data types are used as variables in procedural blocks. (register의 data type은 procedural block들 에서 변수로써 쓰인다.)
  • A register data type is required if a signal is assigned a value within a procedural block (procedural block에서 변수의 값을 할당하려면 register의 data type이 필요하다.)
  • Procedural blocks begin with keyword initial and always. (Procedural block 은 initial 과 always 키워드로 시작한다.)
Data typesFunctionality
regUnsigned variable
integerSigned variable - 32 bits
timeUnsigned integer - 64 bits
realDouble precision floating point variable

Strings

CharacterDescription
\nNew line character
\tTab character
\Backslash () character
"Double quote (") Character
\dddA character specified in 1-3 octal digits (0 <= d <= 7)
%%Percent (%) character

Gate Primitives

module gates();

wire out0;
wire out1;
wire out2;
reg in1, in2, in3, in4;

not U1(out0, in1);
and U2(out1, in1, in2, in3, in4);
xor U3(out2, in1, in2, in3);

initial begin
  $monitor("in1 = %b in2= %b in3=%b out0=%b out1=%b out2=%b",
    in1, in2, in3, in4, out0, out1, out2);
  in1 = 0;
  in2 = 0;
  in3 = 0;
  in4 = 0;
  
  #1 in1 = 1;
  #1 in2 = 1;
  #1 in3 = 1;
  #1 in4 = 1;
  #1 $finish;
end

endmodule

Transmission Gate Primitives

module transmission_gates();

reg data_enable_low, in;
wire data_bus, out1, out2;

bufif0 u1(data_bus, in, data_enable_low);
buf U2(out1, in);
not U3(out2, in);

initial begin
  $monitor(
    "@%g in = %b data_enable_low=%b out1=%b out2=%b data_bus=%b",
    $time, in, data_enable_low, out1, out2, data_bus);
  data_enable_low = 0;
  in = 0;
  #4 data_enable_low = 1;
  #8 $finish;
end

always #2 in = ~in;

endmodule

Switch Primitives

module switch_primitives();

wire net1, net2, net3;
wire n4, net5, net6;

tranif0 my_gate1 (net1, net2, net3);
rtranif1 my_gate2 (net4, net5, net6);

endmodule

Logic Values and signal Strengths

Logic ValueDescription
0zero, low, false
1one, high, true
z or Zhigh impedance, floating
x or Xunknown, uninitialized, contention

Verilog Strength Levels

Strengtth LevelSpecification Keyword
7 Supply Drivesupply0 supply1
6 Strong Pullstrong0 strong1
5 Pull Drivepull0 pull1
4 Large Capacitancelarge
3 Weak Driveweak0 weak1
2 Medium Capacitancemedium
1 Small Capacitancesmall
0 Hi Impedancehighz0 highz1

User Defined Primitives

Syntax

primitive udp_syntax (
a,
b,
c,
d
);
output a;
input b,c,d;

endprimitive

UDP ports rules

  • An UDP can contain only one output and up to 10 inputs.
  • Output port should be the first port followed by one or more input ports.
  • All UDP ports are scalar, i.e. Vector ports are not allowed.
  • UDPs can not have bidirectional ports.
  • The output terminal of a sequetial UDP requires an additional declaration as type reg.
  • It is illegal to declare a reg for the output terminal of a combinational UDP

Body

primitive udp_body (
a,
b,
c
);

ouput a;
input b, c;

// A = B | C;
table
  ? 1 : 1;
  1 ? : 1;
  0 0 : 0;
endtable

endprimitive
`include "udp_body.v"
module udp_body_tb();

reg b,c;
wire a;

udp_body udp (a, b, c);

initial begin
  $monitor(" B = %b C = %b A = %b", b, c, a);
  b = 0;
  c = 0;
  #1 b = 1;
  #1 b = 0;
  #1 c = 1;
  #1 b = 1'bx;
  #1 c = 0;
  #1 b = 1;
  #1 c = 1'bx;
  #1 b = 0;
  #1 $finish;
end

endmodule

Symbols

SymbolinterpretationExplanation
?0 or 1 or X? means the variable can be 0 or 1 or x
b0 or 1Same as ?, but x is not included
f(10)Falling edge on an input
r(01)Rising edge on an input
p(01) or (0x) or (x1) or (1z0 or (z1))Rising edge including x and z
n(10) or (1x) or (x0) or (0z) or (z0)Falling edge including x and z
*(??)All transitions
-No changeNo Change

System Task and Function

  • $display("format_string", par_1, par_2, ...);
  • $strobe("format_string", par_1, par2, ...);
  • $monitor("format_string", par_1, par_2, ...);
  • $displayb (as above but defaults to binary..);
  • $strobeh (as above but defaults to hex..);
  • $monitoro (as above but defaults to octal..);

  • $time, $stime, $realtime : the current simulation time as a 64-bit integer, a 32-bit integer, and a real number
  • $reset, $stop, $finish : $reset resets the simulation back to time 0; $stop halts the simulator and puts it in interactive mode where the user can enter commands; $finish exits the simulator back to the operating system.
  • $scope, $showscope : $scope(hierarchy_name) sets the current hierarchical scope to hierarchy_name. $showscopes(n) lists all modules, tasks and block names in (and below, if n is set to 1) the current scope.
  • $random generates a random integer every tie it is called. If the sequence is to be repeatable, teh first time one invokes random giving it a numerical argument (a seed). Otherwise the seed is derived from the computer clock.

  • $dumpfile, $dumpvar, $dumpon, $dumpoff, $dumpall
  • $dumpfile("filename.vcd")
  • $dumpvar : dumps all variables in the design.
  • $dumpvar(1, top) : dumps all the varaibles in module top and below, but not modules instantiated in top.
  • $dumpvar(2, top) : dumps all the variables in module top and 1 level below.
  • $dumpvar(n, top) : dumps all the variables in module top and n-1 levels below.
  • $dumpvar(0, top) : dumps all the variables in module top and all level below.
  • $dumpon : initiates the dump.
  • $dumpoff : stop dumping.

  • $fopen, $fdisplay, $fstrobe, $fmonitor and $fwrite
  • $fopen opens an output file and gives the open file a handle for use by the other commands.
  • $fclose : closes the file and lests other programs access it.
  • $fdisplay and $fwrite write formatted data to a file whenever they are executed. They are the same except $fdisplay inserts a new line after every execution and $write does not.
  • $disrobe also writes to file when executed, but its waits until all other operations in the time step are complete before writing. Thus initial #1 a = 1; b= 0; $fstrobe(hand1, a,b); b=1; will write write 1 1 for a and b.
  • $fmonitor writes to a file whenever any of tis arguments changes.
  • handle1 = $fopen("filenam1.suffix")
  • handle2=$fopen("filename2.suffix")
  • $fstrobe(handle1, format, variable list)
  • $fdisplay(handle2, format, variable list)
  • $fwrite(handle2, format, variable list)

실습

  • 단순히 문법만 배우니까 먼가 언어를 배우는 느낌이 안나서 코딩해보기로 했다.
  • 해보니까 새롭게 알게 된게 있어서 해보길 잘했다고 생각한다.
/* heap.v */
module heap (
  d,
  is_insert,
  ret,
  enable,
  clock,
  reset
);

parameter MAX_SIZE = 128;

input wire [31:0] d;
input wire is_insert;
input wire enable;
input wire clock;
input wire reset;

output reg signed [31:0] ret;

integer node[MAX_SIZE - 1:0];
integer count;
integer tmp1, tmp2, tmp3;
integer i;

initial
begin
  for (i = 0; i < MAX_SIZE; i = i + 1) begin
    node[i] = 0;
  end
  $display("time | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 ");
  $monitor("%g | %d | %d | %d | %d | %d | %d | %d | %d",
      $time, node[1], node[2], node[3], node[4], node[5], node[6], node[7], node[8]);
  count = 0;
end

always @ (posedge clock)
begin
  if (enable) begin
    /* insert */
    if (is_insert) begin
      count = count + 1;
      node[count] = d;
      
      tmp1 = count;
      while (tmp1 > 1 && node[tmp1 / 2] < node[tmp1]) begin
        tmp2 = node [tmp1/2];
        node [tmp1/2] = node[tmp1];
        node [tmp1] = tmp2;

        tmp1 = tmp1 / 2;
      end
    end
    /* pop */
    else begin
      ret = node[1];
      
      node[1] = node[count];
      node[count] = node[1];
      count = count - 1;

      tmp1 = 1;
      tmp2 = tmp1 * 2;
      if (tmp2 + 1 <= count) begin
      tmp2 = (node[tmp2] > node[tmp2 + 1]) ? tmp2 : tmp2 + 1;
      end

      while (tmp2 <= count && node[tmp1] < node[tmp2]) begin
        tmp3 = node[tmp1];
        node[tmp1] = node[tmp2];
        node[tmp2] = tmp3;

        tmp1 = tmp2;
        tmp2 = tmp2 * 2;

        if (tmp2 + 1 <= count) begin
          tmp2 = (node[tmp2] > node[tmp2 +1]) ? tmp2 : tmp2 + 1;
        end
      end

      node[count + 1] = 0; /* to display */
    end
  end
end

endmodule
/* test.v */
`include "heap.v" // to fix syntax highlight`

module heap_test();

reg clock, reset, enable, is_insert;
integer data;
wire [31:0] ret;

initial begin
  clock = 0;
  reset = 0;
  enable = 1;
  is_insert = 1;
  data = 0;
  #5 data = 5;
  #10 data = 10;
  #10 data = 7;
  #10 is_insert = 0;
  #5 $finish;
end

always begin 
  #5 clock = ~clock;
end

heap U(data, is_insert, ret, enable, clock, reset);

endmodule