Before going into static and automatic variable let’s first go
through 2 important terminology and then will go through automatic and static.
Scope:
Scope is the region
or section of code where a variable can be accessed.
Variables declared inside a module, interface, program, or
checker, but outside a task, process, or function, are local in scope.
Lifetime:
Lifetime is the time
duration where an object/variable is in a valid state.
Variables declared outside a module, program, interface, checker,
task, or function are local to the
compilation unit and have a static lifetime (exist for the whole simulation).
compilation unit and have a static lifetime (exist for the whole simulation).
Variables declared inside a static task, function, or block are
local in scope and default to a static lifetime.
Specific variables within a static task, function, or block can be
explicitly declared as automatic. Such
variables have the lifetime of the call or block and are initialized on each entry to the call or block.
variables have the lifetime of the call or block and are initialized on each entry to the call or block.
---------------------------------------------------------------------
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
module top(); | |
// static int loop5; | |
int loop5; // same as above line, static keyword optional | |
// automatic int loop4; // illegal : All variables in module scope must have a static lifetime | |
// Automatic variable inside static function/task | |
function void print1(); | |
automatic int loop1; | |
//loop1 = 0; // executes on every call of print1 | |
for (int j=0; j<3; j++) begin | |
loop1 = loop1 + 1; | |
$write("%0d ", loop1); | |
end | |
$write("\t"); | |
endfunction : print1 | |
// Automatic variable inside block | |
function void print2(); | |
for (int i=0; i<3; i++) begin | |
automatic int loop1 = 0; // executes on every iteration of i loop | |
for (int j=0; j<3; j++) begin | |
loop1 = loop1 + 1; | |
$write("%0d ", loop1); | |
end | |
$write("\t"); | |
end | |
endfunction : print2 | |
// Static variable inside static function/task | |
function void print3(); | |
// static int loop2; | |
int loop2; // same as above line, static keyword optional | |
// loop2 = 0; // executes once before time 0 | |
for (int j=0; j<3; j++) begin | |
loop2 = loop2 + 1; | |
$write("%0d ", loop2); | |
end | |
$write("\t"); | |
endfunction : print3 | |
// Static variable inside block | |
function void print4(); | |
for (int i=0; i<3; i++) begin | |
int loop2 = 0; // executes once before time 0 | |
for (int j=0; j<3; j++) begin | |
loop2 = loop2 + 1; | |
$write("%0d ", loop2); | |
end | |
$write("\t"); | |
end | |
endfunction : print4 | |
// Static variable inside module | |
function void print5(); | |
for (int i=0; i<3; i++) begin | |
// loop5 = 0; // executes once before time 0 | |
for (int j=0; j<3; j++) begin | |
loop5 = loop5 + 1; | |
$write("%0d ", loop5); | |
end | |
$write("\t"); | |
end | |
endfunction : print5 | |
initial begin | |
$display ("Automatic variable inside static function/task"); | |
for (int i=0; i<3; i++) begin | |
print1(); | |
end | |
$display(); | |
$display ("Automatic variable inside block"); | |
print2(); | |
$display(); | |
$display(); | |
$display ("Static variable inside static function/task"); | |
for (int i=0; i<3; i++) begin | |
print3(); | |
end | |
$display(); | |
$display ("Static variable inside block"); | |
print4(); | |
$display(); | |
$display ("Static variable inside module"); | |
print5(); | |
$display(); | |
end | |
endmodule : top |
---------------------------------------------------------------------
Result:
---------------------------------------------------------------------
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//Output: | |
// Automatic variable inside static function/task | |
// 1 2 3 1 2 3 1 2 3 | |
// Automatic variable inside block | |
// 1 2 3 1 2 3 1 2 3 | |
// | |
// Static variable inside static function/task | |
// 1 2 3 4 5 6 7 8 9 | |
// Static variable inside block | |
// 1 2 3 4 5 6 7 8 9 | |
// Static variable inside module | |
// 1 2 3 4 5 6 7 8 9 |
---------------------------------------------------------------------
Automatic Function/Task (Function/Task with Automatic Lifetime):
In Verilog default
lifetime of all the task/function is static.
Tasks and functions may be declared as automatic. Variables
declared in an automatic task, function, or block are local in scope, default
to the lifetime of the call or block, and are initialized on each entry to the call
or block.
In other word, Automatic task/function variables cannot be
accessed by hierarchical references. Automatic variables are automatically destroyed once the scope (task, function, block) in which they are created ends
and regenerated when again enters into scope (task, function, block).
---------------------------------------------------------------------
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
module top(); | |
// Function with Static Lifetime | |
// By default lifetime of function/task is Static in Verilog/SV | |
function void print_s(); | |
int a; | |
a = a + 1; | |
$write("%0d ", a); | |
endfunction : print_s | |
// Function with Automatic Lifetime | |
function automatic void print_a(); | |
int a; | |
a = a + 1; | |
$write("%0d ", a); | |
endfunction : print_a | |
initial begin | |
$display("Static variable"); | |
for (int i=0; i<4; i++) begin | |
print_s(); | |
end | |
$display("\n"); | |
$display("Automatic variable"); | |
for (int i=0; i<4; i++) begin | |
print_a(); | |
end | |
$display(); | |
end | |
endmodule : top | |
//Output: | |
// Static variable | |
// 1 2 3 4 | |
// | |
// Automatic variable | |
// 1 1 1 1 |
---------------------------------------------------------------------
Class methods has Automatic lifetime by default:
Because of backward compatibility reason, SystemVerilog could not
change the default lifetime qualifier to ‘automatic’ but since class methods
were new, it changed the lifetime of all methods to be automatic.
---------------------------------------------------------------------
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class my_class; | |
// Function with Automatic Lifetime | |
// By default lifetime of function/task is Automatic in Class | |
function void print_a(); | |
int a; | |
a = a + 1; | |
$write("%0d ", a); | |
endfunction : print_a | |
// Function with Static Lifetime | |
function static void print_s(); | |
int a; | |
a = a + 1; | |
$write("%0d ", a); | |
endfunction : print_s | |
endclass : my_class | |
module top(); | |
my_class c; | |
initial begin | |
c = new(); | |
$display("Automatic variable"); | |
for (int i=0; i<4; i++) begin | |
c.print_a(); | |
end | |
$display("\n"); | |
$display("Static variable"); | |
for (int i=0; i<4; i++) begin | |
c.print_s(); | |
end | |
$display(); | |
end | |
endmodule : top | |
//Output: | |
// Automatic variable | |
// 1 1 1 1 | |
// | |
// Static variable | |
// 1 2 3 4 |
---------------------------------------------------------------------
The keyword ‘static’ is overloaded. There is a
semantic difference between the meaning of ‘static’ used to the left of a
function/task keyword and ‘static’ used to the right of the function/task
keyword.
When used to the left of the function or task,
‘static’ is a class qualifier and has the same as on any static method in
C++/Java. It is a method of the class type, not of an instance of a class
object. Other class qualifiers are ‘local’ and ‘protected’ and along with
‘static’ are only allowed in front of methods of a class. There is no
corresponding ‘automatic’ class qualifier.
When used to the right of a function or task, ‘static’ is
a lifetime qualifier, with ‘automatic’ being the corresponding qualifier.
For more details refer this link.