Introduction to SystemVerilog
()
About this ebook
This book provides a hands-on, application-oriented guide to the entire IEEE standard 1800 SystemVerilog language. Readers will benefit from the step-by-step approach to learning the language and methodology nuances, which will enable them to design and verify complex ASIC/SoC and CPU chips. The author covers the entire spectrum of the language, including random constraints, SystemVerilog Assertions, Functional Coverage, Class, checkers, interfaces, and Data Types, among other features of the language. Written by an experienced, professional end-user of ASIC/SoC/CPU and FPGA designs, this book explains each concept with easy to understand examples, simulation logs and applications derived from real projects. Readers will be empowered to tackle the complex task of multi-million gate ASIC designs.
- Provides comprehensive coverage of the entire IEEE standard SystemVerilog language;
- Covers important topics such as constrained random verification, SystemVerilog Class, Assertions, Functional coverage, data types, checkers, interfaces, processes and procedures, among other language features;
- Uses easy to understand examples and simulation logs; examples are simulatable and will be provided online;
- Written by an experienced, professional end-user of ASIC/SoC/CPU and FPGA designs.
This is quite a comprehensive work. It must have taken a long time to write it. I really like that the author has taken apart each of the SystemVerilog constructs and talks about them in great detail, including example code and simulation logs. For example, there is a chapter dedicated to arrays, and another dedicated to queues - that is great to have!
The Language Reference Manual (LRM) is quite dense and difficult to use as a text for learning the language. This book explains semantics at a level of detail that is not possible in an LRM. This is the strength of the book. This will be an excellent book for novice users and as a handy reference for experienced programmers.Mark Glasser
Cerebras Systems
Related to Introduction to SystemVerilog
Related ebooks
Learn Rust Programming: Safe Code, Supports Low Level and Embedded Systems Programming with a Strong Ecosystem (English Edition) Rating: 0 out of 5 stars0 ratingsImplementing Domain-Specific Languages with Xtext and Xtend - Second Edition Rating: 0 out of 5 stars0 ratingsSystemVerilog for Hardware Description: RTL Design and Verification Rating: 0 out of 5 stars0 ratingsData Structures and Algorithms with Go: Create efficient solutions and optimize your Go coding skills (English Edition) Rating: 0 out of 5 stars0 ratingsIntroducing Vala Programming: A Language and Techniques to Boost Productivity Rating: 0 out of 5 stars0 ratingsProtocol Buffers Handbook: Getting deeper into Protobuf internals and its usage Rating: 0 out of 5 stars0 ratingsVisual Word: Unlocking the Power of Image Understanding Rating: 0 out of 5 stars0 ratingsWeb Applications with Elm: Functional Programming for the Web Rating: 0 out of 5 stars0 ratingsCore Objective-C in 24 Hours Rating: 5 out of 5 stars5/5Offensive Shellcode from Scratch.: Get to grips with shellcode countermeasures and discover how to bypass them Rating: 0 out of 5 stars0 ratingsIntroduction to Programming Languages Rating: 4 out of 5 stars4/5The Rust Guide to Generative AI Rating: 0 out of 5 stars0 ratingsSwift Programming Nuts and bolts Rating: 0 out of 5 stars0 ratingsiOS Programming Nuts and bolts Rating: 4 out of 5 stars4/5Objective-C Programming Nuts and bolts Rating: 0 out of 5 stars0 ratingsKnowledge Reasoning: Fundamentals and Applications Rating: 0 out of 5 stars0 ratingsLearn Kotlin for Android Development: The Next Generation Language for Modern Android Apps Programming Rating: 0 out of 5 stars0 ratingsSystemC: From the Ground Up, Second Edition Rating: 0 out of 5 stars0 ratingsSoftware Architecture for Busy Developers: Talk and act like a software architect in one weekend Rating: 0 out of 5 stars0 ratingsPro C# 8 with .NET Core 3: Foundational Principles and Practices in Programming Rating: 0 out of 5 stars0 ratingsSubject Oriented Programming Rating: 0 out of 5 stars0 ratingsDevSecOps for .NET Core: Securing Modern Software Applications Rating: 0 out of 5 stars0 ratingsRegex Quick Syntax Reference: Understanding and Using Regular Expressions Rating: 0 out of 5 stars0 ratings.NET Mastery: The .NET Interview Questions and Answers Rating: 0 out of 5 stars0 ratingsAssembly Language: From Basics to Expert Proficiency Rating: 0 out of 5 stars0 ratingsDebugging Systems-on-Chip: Communication-centric and Abstraction-based Techniques Rating: 0 out of 5 stars0 ratings
Electrical Engineering & Electronics For You
How to Diagnose and Fix Everything Electronic, Second Edition Rating: 4 out of 5 stars4/5Beginner's Guide to Reading Schematics, Fourth Edition Rating: 4 out of 5 stars4/5Practical Electrical Wiring: Residential, Farm, Commercial, and Industrial Rating: 4 out of 5 stars4/5The Innovators: How a Group of Hackers, Geniuses, and Geeks Created the Digital Revolution Rating: 4 out of 5 stars4/5Electrician's Pocket Manual Rating: 0 out of 5 stars0 ratingsUpcycled Technology: Clever Projects You Can Do With Your Discarded Tech (Tech gift) Rating: 5 out of 5 stars5/5The Homeowner's DIY Guide to Electrical Wiring Rating: 5 out of 5 stars5/5Off-Grid Projects: Step-by-Step Guide to Building Your Own Off-Grid System Rating: 0 out of 5 stars0 ratingsProgramming the Raspberry Pi, Third Edition: Getting Started with Python Rating: 5 out of 5 stars5/5Ramblings of a Mad Scientist: 100 Ideas for a Stranger Tomorrow Rating: 0 out of 5 stars0 ratingsBeginner's Guide to Reading Schematics, Third Edition Rating: 0 out of 5 stars0 ratingsThe Illustrated Tesla (Rediscovered Books): With linked Table of Contents Rating: 5 out of 5 stars5/5Understanding Electricity Rating: 4 out of 5 stars4/5Two-Stroke Engine Repair and Maintenance Rating: 0 out of 5 stars0 ratingsDIY Lithium Battery Rating: 3 out of 5 stars3/5Electricity for Beginners Rating: 4 out of 5 stars4/5Basic Electricity Rating: 4 out of 5 stars4/5Electric Circuits Essentials Rating: 5 out of 5 stars5/5The Electrician's Trade Demystified Rating: 0 out of 5 stars0 ratingsRaspberry Pi Projects for the Evil Genius Rating: 0 out of 5 stars0 ratingsSchaum's Outline of Basic Electricity, Second Edition Rating: 5 out of 5 stars5/5Digital Filmmaking for Beginners A Practical Guide to Video Production Rating: 0 out of 5 stars0 ratingsSoldering electronic circuits: Beginner's guide Rating: 4 out of 5 stars4/5Programming Arduino: Getting Started with Sketches Rating: 4 out of 5 stars4/5Making Everyday Electronics Work: A Do-It-Yourself Guide: A Do-It-Yourself Guide Rating: 4 out of 5 stars4/5Circuitbuilding Do-It-Yourself For Dummies Rating: 0 out of 5 stars0 ratingsTHE Amateur Radio Dictionary: The Most Complete Glossary of Ham Radio Terms Ever Compiled Rating: 4 out of 5 stars4/5Theory on DC Electric Circuits Rating: 0 out of 5 stars0 ratings
Reviews for Introduction to SystemVerilog
0 ratings0 reviews
Book preview
Introduction to SystemVerilog - Ashok B. Mehta
© The Author(s), under exclusive license to Springer Nature Switzerland AG 2021
A. B. MehtaIntroduction to SystemVeriloghttps://doi.org/10.1007/978-3-030-71319-5_1
1. Introduction
Ashok B. Mehta¹
(1)
DefineView Consulting, Los Gatos, CA, USA
Introduction
This chapter introduces the evolution of the IEEE standard SystemVerilog language. It describes how the SystemVerilog for Design and Verification, SystemVerilog Assertions, and SystemVerilog Functional Coverage language subsets fold into a single unified language.
SystemVerilog is the IEEE standard 1800 unified language for functional design and verification. It has its roots in the Verilog language that was invented way back in the mid-1980s by Phil Moorby of Gateway Design Automation. This book covers the IEEE 1800–2017 version of the standard. The standard was developed to meet the increasing usage of the language in specification, design, and verification of hardware.
SystemVerilog language has four distinct languages under a single simulation kernel. They are:
1.
SystemVerilog for Design (synthesizable subset)
2.
SystemVerilog for Verification (which includes the OOP subset)
3.
SystemVerilog Assertions
4.
SystemVerilog Functional Coverage
The SystemVerilog for Design and Verification subsets, SystemVerilog Assertions (SVA) subsets, and SystemVerilog Functional Coverage subsets are orthogonal to each other. Different features have separate syntax, but they all share a common set of data types, expression operators, etc. And all these subsets work under a unified simulation kernel (simulation time tick).
Figure 1.1 shows the language subset for testbench constructs (class, constrained random, etc.), the subset for SystemVerilog Assertions (SVA), and the subset for Functional Coverage. Each has a unique role to play in the language. It is the combination of all these subsets that makes the language ever so powerful. No need to have a multi-language simulation environment (one language for design, one for verification, etc.). The unified language allows you to tackle the entire domain of design and verification of hardware:
Design the DUT (SystemVerilog subset for design).
Provide stimulus/check response from the testbench (SystemVerilog OOP subset for Verification).
Check combinatorial/sequential logic response from the DUT (SystemVerilog Assertions).
Measure Functional Coverage of the design (SystemVerilog Functional Coverage).
../images/497959_1_En_1_Chapter/497959_1_En_1_Fig1_HTML.pngFig. 1.1
SystemVerilog language
This integrated whole created by SystemVerilog greatly exceeds the sum of its individual components, creating a new type of engineering language, HDVL (hardware design and verification language). Using a single unified language enables a unified environment for the engineers to model large, complex designs and verify that these designs are functionally correct.
1.1 SystemVerilog Language Evolution
It all started with the Verilog language as the first generation of hardware design and verification language around 1985. It became an IEEE standard in 1995. The original Verilog was developed by Phil Moorby of Gateway Design Automation. It was geared more toward hardware design but had sufficient behavioral constructs to develop testbenches. Note that the designs in late 1980s were in 10 to 100,000 gates; range and verification entailed quite a bit of gate-level verification and some RTL level. So, the language, at that time, had sufficient features to tackle both design and verification.
Verilog-2001 was a major update to the standard and added features such as multi-dimensional arrays, auto variables, etc. Higher-level constructs for design were added. By this time, the designs were already in multimillion gates, and it became evident that the language needed a major overhaul when it came to functional verification capabilities. Since the Verilog language lacked advanced constructs for verification, other languages popped up, namely, Vera and e,
to augment Verilog with OOP (object-oriented programming) subsets. But now the user had to create a multi-language environment with support from different EDA vendors, which is a very cumbersome, error-prone and time-consuming task.
Enter SystemVerilog 3.0 (June of 2002). It added advanced Verilog and C
data types. It was a step forward to making a robust language for both design and verification. It added extensions to synthesizable constructs of Verilog and enabled modeling hardware at higher levels of abstraction. But still it lacked language constructs that would allow for a reusable/modular code/environment for verification.
Enter SystemVerilog 3.1 (May of 2003). This version completely overhauled the language subset for verification. It added C++-style class
construct with methods, properties, inheritance, etc. It also added features to allow constrained random verification. It also added the SystemVerilog Assertions (SVA) subset with enhanced semantics for sequential temporal domain expressions, sequence and property generation, etc. It also added the Functional Coverage subset to the language. It allowed one to objectively measure the functional coverage of a design using coverpoints, covergroups, bins, etc.
Enter SystemVerilog 3.1a (May of 2004). Accellera continued to refine the SystemVerilog 3.1 standard by working closely with major electronic design automation (EDA) companies to ensure that the SystemVerilog specification could be implemented as intended. A few additional modeling and verification constructs were also defined. In May of 2004, a final Accellera SystemVerilog draft was ratified by Accellera and called SystemVerilog 3.1a.
So far, SystemVerilog was not an IEEE standard. It was an Accellera standard. In June of 2004, after SystemVerilog 3.1a was released, Accellera donated SystemVerilog standard to IEEE standards association, which oversaw the Verilog 1364 standard. Accellera worked with the IEEE to form a new standards request, to review and standardize the SystemVerilog extensions to Verilog. IEEE assigned the project number 1800 to SystemVerilog, and hence SystemVerilog is IEEE 1800 standard.
SystemVerilog IEEE 1800 contains many extensions for the verification of large designs, integrating features from SUPERLOG, Vera, C, C++, and VHDL languages. The primary technology donations that make up SystemVerilog include:
The SUPERLOG Extended Synthesizable Subset (SUPERLOG ESS), from Co-Design Automation:
SUPERLOG (Co-Design Automation) was the brainchild of Peter Flake, Phil Moorby, and Simon Davidmann. In 2001, Co-Design Automation (which was acquired by Synopsys in 2002) donated to Accellera the SUPERLOG Extended Synthesizable Subset. This subset included a standard set of enhancements, including enhancements for verification. You can say, SystemVerilog started with the donation of the SUPERLOG language to Accellera in 2002 by the startup company Co-Design Automation.
PSL Assertions (which began as a donation of Sugar language from IBM)
OpenVera Assertions (OVA) from Synopsys:
OpenVera Assertions (OVA) and DirectC features were donated to Accellera. These donations significantly extended the verification capabilities of Verilog. The bulk of the verification functionality is based on the OpenVera language donated by Synopsys.
The DirectC and coverage API from Synopsys
Tagged unions and high-level language features from Bluespec
Then came the next revision of the standard SystemVerilog – 2012. And then SystemVerilog – 2017. This book is based on the 2017 LRM.
SystemVerilog Assertions (SVA) subset/sub-language is also influenced by many different languages as shown in Fig. 1.2.
../images/497959_1_En_1_Chapter/497959_1_En_1_Fig2_HTML.pngFig. 1.2
SystemVerilog Assertions (SVA) evolution
SystemVerilog Assertions language is derived from many different languages. Features from these languages either influenced the language or were directly used as part of the language syntax/semantic.
Sugar from IBM led to PSL. Both contributed to SVA. The other languages that contributed are Vera, e,
CBV from Motorola, and ForSpec from Intel.
In short, when we use SystemVerilog Assertions language, we have the benefit of using the latest evolution of an assertion language that benefited from many other robust assertion languages.
© The Author(s), under exclusive license to Springer Nature Switzerland AG 2021
A. B. MehtaIntroduction to SystemVeriloghttps://doi.org/10.1007/978-3-030-71319-5_2
2. Data Types
Ashok B. Mehta¹
(1)
DefineView Consulting, Los Gatos, CA, USA
Electronic Supplementary Material The online version of this chapter (https://doi.org/10.1007/978-3-030-71319-5_2) contains supplementary material, which is available to authorized users.
Introduction
This chapter describes the rich set of data types that SystemVerilog offers. Integer datatypes and real datatypes are discussed. In addition, use-defined types; static, local, automatic, and global variables; enumerated types; string data types; and event data types are discussed. Each data type is explained with examples and simulation logs.
SystemVerilog offers a rich variety of integer and real data types and nets. Verilog used to provide reg
and wire,
but these were not sufficient especially for functional verification. Verilog 2001 introduced the concept of variables
to emphasize that reg
is a data type of a variable. SV takes it a step further in saying that a net is a signal with a logic data, and you can apply other data types to nets. SystemVerilog also offers object-oriented feature set that allows for modular and reusable verification environment. Many of the data types are geared toward supporting these new features.
The categories under which data types can be broadly summarized are the integer
data type and the real
data type and nets.
Note: We will use the term integral data type throughout the book. It refers to the data types that can represent a single basic integer data type, packed array, packed structure, packed union, enum variable, or time variable.
2.1 Integer Data Types
The integer data types are shown in Table 2.1. The integer data types can be broadly classified into two types, the 2-state vs. 4-state types and signed vs. unsigned types. 4-states are 0,
1,
x,
and z
and 2-states are 0
and 1.
Table 2.1
Integer data types
2.1.1 Integer, int, longint, shortint, logic, byte, reg
The int, longint, and shortint are 2-state signed integer data types, while the integer
is a 4-state signed integer data type. Legacy reg
and the new logic
types are identical. They are both 4-state unsigned.
Let us look at an example to see how these work:
module datatype1;
integer a; //4 state - 32 bit signed
int b; //2 state - 32 bit signed
shortint c; //2 state - 16 bit signed
longint d; //2 state - 64 bit signed
logic [7:0] A1; //4-state - unsigned ‘logic’
logic signed [7:0] sl1; //4-state - signed ‘logic’
byte bl1; //2-state signed ‘byte’
reg [7:0] r1; //4-state - unsigned ‘reg’
initial
begin
a = 'h xxzz_ffff; //integer - 4 state - 32 bit signed
b = -1; //int - 2 state - 32 bit signed
c = 'h fxfx; //shortint - 2 state - 16 bit signed
d = 'h ffff_xxxx_ffff_zzzz;
//longint - 2 state - 64 bit signed
A1 = -1 ; //signed assignment to unsigned 'logic’
sl1 = -1; //signed assignment to signed 'logic'
bl1 = -1; //signed byte
r1 = 8'b xzxz_0101; //'reg' - unsigned 4-state
end
initial
begin #10;
$display(a = %h b = %h c = %h d = %h
, a, b, c, d);
$display(A1 = %0d sl1=%0d bl1 = %0d r1 = %b
,A1,sl1,bl1,r1);
#10 $finish(2);
end
endmodule
In this example, we define a, b, c, and d to be of type integer, int, shortint, and longint, respectively. Then we define A1
as an 8-bit unsigned logic type. Next, we define sl1
as of type logic
– but – signed. Yes, you can take an unsigned type and make it a signed type by explicitly declaring it as a signed type. Finally, we define a signed byte bl1
and an unsigned 8-bit reg r1.
In the testbench, we assign different numbers to each of these variables. Some of these assigned values have x
(unknown) in them to show how a 2-state vs. a 4-state variable treats an x.
We also assign both positive and negative values to some of the variables to see how signed vs. unsigned values work:
Simulation log:
a = xxzzffff b = ffffffff c = f0f0 d = ffff0000ffff0000
A1 = 255 sl1= -1 bl1 = -1 r1 = xzxz0101
V C S S i m u l a t i o n R e p o r t
First, we assign a = ‘h xxzz_ffff, where a
is of type integer which is unsigned 4-state. So, the simulation log shows that a
retains x
as an x
and z
as z
(4-states are 0, 1, X, and Z).
Next, we assign b = −1, where b
is int
signed 2-state type. We have displayed b
as a hex value. Hence, b
shows an assignment of ffffffff
which is the equivalent of decimal −1. Point is that since b
is signed, it retains its signed value assignment as signed.
Next, we assign c = ‘h fxfx, where c
is a shortint 2-state signed type. Hence, the display shows that assigned value to c
is f0f0.
This is because x
is converted to 0
since there is no x
state in a 2-state variable.
Next, we assign d = ‘h ffff_xxxx_ffff_zzzz, where d
is a longint 2-state signed type. Since d
is a 2-state variable, it converts x
to 0
as well as z
to 0.
Hence the displayed value is ffff0000ffff0000.
Next, we assign A1 = −1, where A1
is unsigned logic type. Note that A1
is unsigned, but we are assigning a negative value to it. So, it will convert the negative 1 to positive 255 which is the equivalent of −1 in unsigned logic. Hence, the display shows A1 = 255.
Next, we assign sl1 = −1, where sl1
is of type logic but is explicitly declared as signed.
So, even though logic
is of type unsigned, sl1
is converted to of type signed. Hence, the assigned value of -1
remains as -1
as shown in the display (sl1 = −1).
Next, we assign bl1 = −1, where bl1 is a byte of type 2-state signed. Hence, it also retains the assigned negative value as negative, as shown in the simulation log.
Finally, we assign r1 = 8’b xzxz_0101, where r1 is reg
which is unsigned 4-state type. Since it is a 4-state type, it will retain x
as an x
and a z
as a z
as shown in the simulation log. Note that unsigned 4-state logic
and unsigned 4-state reg
are equivalent. There is no difference between them. reg
is kept around for legacy reasons.
Also, you can explicitly assign a signed number to a variable. So, for example, 1’sb1 is signed number assignment, while 1’b1 is unsigned assignment. Here is a simple example:
logic [7:0] L1; //unsigned logic type
L1 = 4’sb1001; //= 8’b11111001 //Sign extension
L1 = 1’sb1; //= 8’b1111_1111 //Sign extension
L1 = 8’sb1; //= 8’b0000_0001 //NO sign extension because of
//explicit width being same as vector declaration
L1 = 8’sbX; //=8’bxxxx_xxxx
2.1.2 Signed Types
Let us see how signed and unsigned variables work. You can explicitly describe a logic
type as signed, as we saw in previous section. Let us further explore that in seeing how assignments to signed vs. unsigned variables are evaluated. Here is an example:
module top;
logic [7:0] r1;
logic signed [7:0] sr1;
initial begin
r1 = -2;
$display($stime,,,r1=%d
,r1);
sr1 = -2;
$display($stime,,,sr1=%d
,sr1);
r1 = r1+1;
$display($stime,,,r1=%d
,r1);
sr1 = sr1+1;
$display($stime,,,sr1=%d
,sr1);
end
endmodule
Simulation log:
# run –all
# 0 r1=254
# 0 sr1= -2
# 0 r1=255
# 0 sr1= -1
# exit
r1
is declared as default unsigned 8-bit vector, while sr1
is declared as signed 8-bit vector. When we assign r1 = −2, since r1
is unsigned, it will have the value 254 (decimal equivalent of −2). But sr1
will evaluate as −2 since it is signed. When we add a 1 to r1,
it evaluates to 255 (254 + 1). But when we add a 1 to sr1,
it will be −1 (−2 + 1). This exemplifies how signed and unsigned variables are evaluated and interpreted.
Here’s how signed and unsigned variables are interpreted.
../images/497959_1_En_2_Chapter/497959_1_En_2_Figa_HTML.pngYou can also declare wires and ports as signed. Not that logic, reg, wire, inputs, and outputs are unsigned by default:
wire signed [7:0] w;
module sm (input signed [7:0] iBus, output logic signed [7:0] oBus);
Here are some more examples:
logic signed [3:0] sr = -1; ( sr = 4’sb1111)
logic signed [7:0] sr1 = 1; (sr1 = 8’sb00000001)
logic [7:0] adds = sr + sr1; ( adds = 8’b00000000)
adds = sr; (adds = 8’b11111111);
logic [7:0] usr = 1;
logic signed [7:0] s_add;
s_add = sr + usr; (s_add = 15+1 = 8’sb00010000) (signed + unsigned = unsigned; sr is treated as unsigned 15)
2.1.3 Bits vs. Bytes
As we know, a bit
is unsigned, while a byte
is signed. So, do you think the following two declarations are equivalent?
bit [7:0] aBit; // Note ‘bit’ is 2-state, unsigned
byte bByte; // Note ‘byte’ is 2-state, 8-bit signed integer
Answer is no because:
bit [7:0] aBit; // = 0 to 255
byte bByte; // = -128 to 127
So, you need to be careful in mixing bits with bytes because they have different polarities.
Similarly, do you think the following two statements are equivalent?
byte MEM_BYTES [256];
bit signed [7:0] MY_MEM_BYTES [256];
The answer is yes. This is because we explicitly declared bit
to be signed. So, bit signed [7:0]
is equivalent to a byte.
2.2 Real Data Types
The real
data types are shown in Table 2.2. Variables of these three types are commonly called real
variables. Note that the following is not allowed on these variables:
Edge event controls (posedge, negedge, edge) applied to real variables
Bit-select or part-select references of variables declared as real
Real number index expressions of bit-select or part-select references of vectors
Table 2.2
real
data types
Also, note the following when converting real to integer or vice versa.
Conversion from real
to integer
:
Real numbers are converted to integers by truncating the real number to the nearest integer.
Conversion from integer
to real
:
Individual bits that are x
or z
in the net or the variable are treated as zero upon conversion.
2.2.1 real
Data-Type Conversion Functions
There are also system functions available that allow for conversion from real
and shortreal
to integer,
bit,
and vice versa. This is shown in Table 2.3. The table describes each conversion function.
Table 2.3
real
data-type conversion functions
Let us look at an example of how these functions work:
module datatype1;
real real1, real2, real3;
integer i1;
bit [63:0] bit1;
initial begin
real1 = 123.45;
i1 = $rtoi(real1);
real2 = $itor(i1);
bit1 = $realtobits ( real1);
real3 = $bitstoreal(bit1);
end
initial begin
#10;
$display(real1 = %f real2 = %f i1=%0d
,real1,real2,i1);
$display(bit1 = %b real3=%f
,bit1,real3);
#10 $finish(2);
end
endmodule
In this example, we assign 123.45 to real1
which is of type real.
Then we use conversion functions on that value.
Here is the simulation log:
real1 = 123.450000
i1=123
real2 = 123.000000
bit1 = 0100000001011110110111001100110011001100110011001100110011001101
real3=123.450000
V C S S i m u l a t i o n R e p o r t
First, we take real1 = 123.45
and convert it to an integer using the $rtoi conversion function. Since $rtoi truncates the real value to integer, we see in the simulation log that i1
is equal to 123 (0.45 is truncated).
Next, we take the converted integer value and reconvert it to real (real2 = $itor(i1);). This produces real2 = 123.000000.
Next, we take the real1
value and convert it to bits using bit1 = $realtobits ( real1);.
This conversion function produces a 64-bit vector representation of the real value. This is shown in simulation log as bit1 = 0100000001011110110111001100110011001100110011001100110011001101.
Next we take this 64-bit converted value and convert it back to real type (real3 = $bitstoreal(bit1);). This produces real3 = 123.45. So, no precision was lost when going from $realtobits and then back to $bitstoreal.
2.3 Nets
Nets are used to connect elements such as logic gates. As such, they do not store any value. They simply make a connection from the driving logic to the receiving logic or simply pull a net high or low- or high-impedance state.
Let us look at the truth table of each of this type that will explain their functionality. Table 2.4 shows net data types.
Table 2.4
Nets
2.3.1 wire
and tri
The wire and tri nets connect elements. They do not store any value.
The net types wire and tri are identical in their syntax and functions.
Two names are provided so that the name of a net can indicate the purpose of the net in that model.
A wire net can be used for nets that are driven by a single gate or continuous assignment. But note that this is simulator/methodology dependent. A wire
that is driven by multiple nets may be caught by a lint type tool to catch a multiply driven wire.
Most methodologies restrict wire
to have a single driver.
The tri net type can be used where multiple drivers drive a net.
Here is a simple example showing how tristate logic works.
In Fig. 2.1, we show a typical use of tri
net. There are two drivers, namely, buif0 and bufif1. They both drive a single tri
net called Znet. Bufif1 drives the net A
when Aenb is 1,
and bufif0 drives the net B
when Benb is 0.
So, when A
= 1 and B
= 0, there will be conflict on the net Znet, and the result will be X
(unknown). But when A
= 0 and B
= 1, both drivers will be disabled and Znet will be equal to Z.
In other words, Znet is a resolved type net. It will resolve the driven values by two (or more) drives and produce the correct result. This is in contrast to reg
or logic
type which are unresolved type (they will cause a race condition when multiple drivers drive from concurrently running logic processes).
Fig. 2.1
Tristate logic
Here is the SystemVerilog code and testbench for the circuit in Fig. 2.1:
module trinet;
logic A, B, Aenb, Benb;
tri Znet;
bufif1 if1(Znet, A, Aenb); //bufif1 driver
bufif0 if0(Znet, B, Benb); //bufif0 driver
initial begin
A = 1; Aenb = 1; //Drive bufif1 with 1
B = 0; Benb = 0; //Drive bufif0 with 0
#10;
Aenb = 0; //disable bufif1
Benb = 1; //disable bufif0
end
initial begin
$monitor ($stime ,,,A=%0d Aenb=%0d B=%0d Benb = %0d Znet=%0d
, A, Aenb, B, Benb, Znet);
end
endmodule
Simulation log:
../images/497959_1_En_2_Chapter/497959_1_En_2_Figb_HTML.pngAs you notice from the code, when both bufif0 and bufif1 drive opposite values, Znet = x.
But when both are disabled, Znet = z.
Note that in the preceding example, you can replace tri
with wire,
and you will get the same results. But, from a methodology point of view, you want to use wire
only for singly driven nets.
Here is the truth table for wire/tri.
They both have the same functionality when they are driven by multiple drivers. A strongly driven 0
or 1
will always win over a high impedance z
state, and whenever there are multiple conflicting value drivers, the result will be an x.
x
always wins over any other driven value. The truth table is shown in Table 2.5.
Table 2.5
wire/tri truth table
2.3.2 Unresolved wire
Type: uwire
As we saw, both wire
and tri
allow multiple drivers. But what if you want to restrict wire
to only single driver? What if you want to see an error when multiple drivers drive a wire
? Well, that is where unresolved uwire
type comes into picture.
The keyword to declare an unresolved wire is uwire.
It is an unresolved or unidriver wire and is used to model nets that allow only a single driver. It is an error to connect any bit of a uwire net to more than one driver. It is also an error to connect a uwire net to a bidirectional terminal of a bidirectional pass switch.
Here is a simple example:
module init;
uwire w1;
logic enable, a, b;
assign w1 = enable ? a:1'bz;
assign w1 = !enable ? b:1'bz;
endmodule
In this example, uwire w1
is driven by multiple drivers. If w1
was declared as wire,
this would be perfectly normal and accepted. But since w1
is declared as uwire, you will get the following error from the simulator:
# SLP: Elaboration phase ...
# SLP: Error: testbench.sv (5): Net 'w1' of the uwire type cannot be driven by multiple drivers.
# SLP: Fatal Error: Cannot continue elaboration due to previous errors
uwire
is an excellent way to enforce a single driver wire
methodology:
2.3.3 Resolved vs. Unresolved Type
Also, note that, in contrast to net wire,
variables are neither resolved nor unresolved, and net types are resolved (except when you declare a wire
as a uwire
as we just saw). Here is a simple example to show how the code will behave differently for a variable vs. a net. In short, a variable will cause a race condition when driven from more than one procedural block, while a wire
will behave with correct net resolution. Here is a simple code snippet:
module init;
wire w1; //resolved type
logic enable, a, b, varC; //unresolved type
assign w1 = enable ? a:1'bz; //w1 is resolved type
assign w1 = enable ? b:1'bz;
initial begin
a = 1'b1; b=1'b0;
enable = 1'b1;
end
always @(enable or b) begin
if (enable) varC = b; //race condition
else varC = 1'bz;
end
always @* begin
if (enable) varC = a; //race condition
else varC = 1'bz;
end
endmodule
As you notice from the above example, varC
is driven from two procedural always blocks. There is no guarantee which assignment will win! This is considered a race condition and should be avoided at all cost. Different simulators will give you different results and your logic will be unstable. w1
is also driven by two nets, but it will resolve according to Table 2.5.
2.3.4 wand
and triand
The wand and triand nets will create wired and configurations so that if any driver is 0, the value of the net is 0.
Table 2.6 shows the truth table for wand
/triand.
Table 2.6
wand/triand truth table
2.3.5 wor
and trior
The wor and trior nets will create wired or configurations so that when any of the drivers is 1, the resulting value of the net is 1. Table 2.7 shows the truth table for wor
/trior.
Table 2.7
wor
/trior
truth table
2.3.6 tri0
and tri1
The tri0 and tri1 nets model nets with resistive pulldown and resistive pullup devices on them:
When no driver drives a tri0 net, its value is 0 with strength pull.
When no driver drives a tri1 net, its value is 1 with strength pull.
Following truth tables model multiple drivers of strength strong on tri0 (Table 2.8) and tri1 (Table 2.9) nets. The resulting value on the net has strength strong, unless both drivers are z, in which case the net has strength pull.
Table 2.8
tri0 truth table
Table 2.9
tri1 truth table
Let us take the example that we discussed before but replace tri
with tri0.
The results will be as per the truth table in Table 2.8:
module trinet;
logic A, B, Aenb, Benb;
tri0 Znet; //Note Znet is declared as 'tri0'
bufif1 if1(Znet, A, Aenb);
bufif0 if0(Znet, B, Benb);
initial begin
A = 1; Aenb = 1;
B = 0; Benb = 0;
#10;
Aenb = 0; //disable bufif1
Benb = 1; //disable bufif0
end
initial begin
$monitor ($stime ,,,A=%0d Aenb=%0d B=%0d Benb = %0d Znet=%0d
, A, Aenb, B, Benb, Znet);
end
endmodule
Simulation log:
../images/497959_1_En_2_Chapter/497959_1_En_2_Figc_HTML.pngAs you notice, at time 10, when both drivers are disabled (meaning they both drive z
on Znet), the resultant value on Znet is 0
(and not z
as we saw when Znet was declared as tri
).
2.4 Drive Strengths
The nets and primitives can be assigned drive strengths. They are defined as follows:
The strength with Level 7 has the highest strength and then in decreasing order. Srong0 and strong1 are the default strengths. So, for example, you have the following assign driving the same net wr1
:
assign (strong0, weak1) wr1 = a;
assign (pull0, pull1) wr1 = b;
If a
is 0 and b
is 1, then strong0 (in the first assignment) will win over pull1 (in the second assignment), and the final result value on wr1
will be strong0 since strong0 has higher strength than pull1.
Here is an example:
module top ();
tri0 t0;
tri t2;
trireg (large) #(0,0,25) trg;
wand w1; //same as triand
wire w2;
supply0 s1;
supply1 s2;
reg a, b, c, d, e, f, enb;
buf b1 (w1, a);
buf b2 (w1, b);
buf (weak1, strong0) (w2, a);
buf (supply0, pull1) (w2, b);
bufif1 (t0, b, enb);
bufif1 (t2, b, enb);
bufif1 (trg, b, enb);
initial begin
a = 0; b = 1; enb=0;
#10; enb = 1;
#10; enb=0;
#10;
end
initial $monitor($stime ,,,a=%b b=%b enb=%b wand-w1=%v wor-w2=%v tri0-t0=%v tri-t2=%v trg=%v s1=%v s2=%v
,a,b,enb,w1,w2,t0,t2,trg,s1,s2);
endmodule
Here is the pictorial representation of the above model.
../images/497959_1_En_2_Chapter/497959_1_En_2_Figd_HTML.pngSimulation log:
# run –all
# 0 a=0 b=1 enb=0wand-w1=St0 w2=St0 tri0-t0=Pu0 tri-t2=HiZ trg=LaX s1=Su0 s2=Su1
# 10 a=0b=1 enb=1wand-w1=St0 w2=St0tri0-t0=St1 tri-t2=St1 trg=St1s1=Su0 s2=Su1
# 20 a=0 b=1enb=0wand-w1=St0 w2=St0tri0-t0=Pu0 tri-t2=HiZ trg=La1s1=Su0 s2=Su1
# ** Warning: (vsim-3466) [DECAY] - Charge on node '/top/trg' has decayed.
# Time: 45 ns Iteration: 0 Instance: /top File: testbench.sv Line: 5
# 45 a=0 b=1 enb=0 wand-w1=St0 w2=St0 tri0-t0=Pu0 tri-t2=HiZtrg=LaXs1=Su0 s2=Su1
w1
is declared as wand
(which is the same as triand
). We drive w1
as follows:
buf b1 (w1, a);
buf b2 (w1, b);
a
is driven with 0 and b
is driven with 1. Since this is a wand, 0 will win over 1, and w1
will be St0 as shown in the simulation log at time 0. Note that it is St0 which is the default strength.
Then we drive w2
as follows:
buf (weak1, strong0) (w2, a);
buf (supply0, pull1) (w2, b);
and again a = 0 and b = 1. So, strong0 wins over pull1 and the resultant value on w2
will be St0.
There are three bufif1 in the model and their corresponding nets are declared as follows:
bufif1 (t0, b, enb);
bufif1 (t2, b, enb);
bufif1 (trg, b, enb);
tri0 t0;
tri t2;
trireg (large) #(0,0,25) trg;
First we drive enb = 1, meaning the tristate buffers are enabled. Since b
(input to tristate buffers) is 1, t0 will be St1. t2 will also be St1
and trg will also be St1.
Then, we disable the tristate buffers (enb = 0). Now, t0 will be Pu0 since tri0
when disabled will pull down the net with a strength of pull. t2 will be HighZ because it is a tristate net, and trg will be La1 since it is a capacitor with large
strength.
Note that for trg
the charge decay time is 25. So, once trg
is driven to La1, it will decay out in 25 time units. That is shown as a warning in the simulation log. After 25 time units, trg
will be driven to an unknown state with strength of large
(LaX).
2.5 Variable vs. Net
There is a subtle different between variables and nets. Variables are assigned through procedural blocks such as initial,
always,
task,
and function
or through continuous assignments. They are the left-hand side of the assignment as shown below:
logic [31:0] bus;
bit dataEnb;
always @(dataEnb)
begin
bus = 32'h0;
end
Here, bus
is a variable of type logic.
It can only be assigned through a procedural block (always
block in this case) or as the left-hand side of a continuous assignment. Variables can be written by one or more procedural statements, including procedural continuous assignments. The last write determines the value. You can also write a variable from different procedural blocks. But be careful not to write to the same variable from multiple procedural blocks, you may end up creating a race condition.
The rule is that it is an error to have multiple continuous assignments or a mixture of procedural and continuous assignments writing to a variable. Again, you will create a race condition.
In contrast, a net is a wire and can only be driven by continuous assignment or as output of gates (i.e., they are connecting nets between gates, output connected to input). Nets cannot be driven from a procedural block. A net can be written by one or more continuous assignments, by primitive outputs, or through module ports. The resultant value of multiple drivers is determined by the resolution function of the net (we will see that in upcoming sections). If a net on one side of a port is driven by a variable on the other side, a continuous assignment is implied. Note that a force statement can override the value of a net from a procedural block. When released, the net returns to the resolved value. I highly recommend that you do – NOT – use a force
for any kind of assignment in your code (nets or variables). Believe me, it will come back to haunt you.
The following is driven by continuous assignment:
wire [31:0] wBus;
assign wBus = wBusInput; //continuous assignment driving a net
OR as output of gate logic:
wire y;
and g1 (y, a, b); //net type driven by output of a gate
Note that multiple continuous assignments to the same net are illegal. For example, the following is illegal:
assign C = sel ? p1 : p2;
assign C = sel ? p3 : p4;
2.6 var
Note that there is also a keyword var
available when declaring variables. If a var
is declared, the data type is optional and will be implicitly declared as of type logic.
For example:
var byte vByte; // same as byte vByte
var A; //same as var logic A
//var logic A
is recommended coding practice
var [31:0] address; //same as var logic [31:0] address
2.7 Variable and Net Initialization
This section described how variables and nets behave at time 0
with their default values as well as assigned values in their declaration.
Here are the default initial values of variables and nets:
You must be careful on how these default initial values affect your testbench logic. Let us look at an example:
module init;
logic clk1; //uninitialized = 'x
wire w1 = '0; //Continuous assignment
initial begin
clk1 = 1; //NO 'x to '1 transition at time 0
@(posedge clk1);
$display(posedge clk1 detected
);
end
initial begin
@(negedge w1); //'z to '0 transition detected
$display(negedge w1 detected
);
end
endmodule
Simulation log:
ncsim> run
negedge w1 detected
ncsim: *W,RNQUIE: Simulation is complete.
Here is what is going on.
We have not initialized clk1 with logic clk1
declaration – so it takes on the default value of ‘x. With ‘wire w1 = 0,
we have initialized wire w1
to an initial value of 0.
Now, in the procedural code, let us try to see how these initial values trigger (or not) an event at time 0.
In the first initial block, we assign a 1
to clk1. This is to see if we get a posedge event at time 0 – because clk1 would transition from ‘x to ‘1 at time 0. In other words, is there a time 0 posedge event on clk1? The answer is no. The simulator does not detect a posedge event even though we may think that clk1 goes from default ‘x to initial value ‘1. This is shown in the simulation log. Note that clk1
is uninitialized. What would happen if it was explicitly initialized to value 0
? Will you get posedge event on clk1
? Try it out.
Now, wire w1
is initialized to 0.
The default value of a wire
is ‘z. So, do we see a transition from ‘z to ‘0 at time 0? The answer is yes. The nets seem to behave differently from the variables when it comes to time 0 event. So, keep this in mind in order to understand initial time events.
Disclaimer: You may get different results from different simulators for time 0 events. The above simulation was done using Cadence’s NCSIM. So, take my example with a grain of salt.
Static variable initializations happen at time 0 before any processes begin (initial/always/continuous assignments). A net declaration assignment is just a shortcut for a declaration and a separate continuous assignment.
Unlike nets, a variable cannot have an implicit continuous assignment as part of its declaration. An assignment as part of the declaration of a variable is a variable initialization, not a continuous assignment.
For example:
wire w = a & b; // net with a continuous assignment
logic v = consta & constb; // variable with initialization
logic vw; // no initial assignment
assign vw = vara & varb; // continuous assignment to a variable
2.8 Static, Automatic, and Local Variables
This section is mainly to focus on different forms of variables available in the language.
Fundamentally, three forms of variable are available, namely, static, automatic, and local. These are described in Table 2.10.
Table 2.10
Variables
2.8.1 Static vs. Local Variables
Let us see the difference between a static variable and a local one. Here is an example to illustrate that:
static int n; // Static variable – outside ‘module’ –
// globally declared
//visible to all modules/scopes that follow.
module vars;
int n;
//Local variable - Module level - visible to all scopes below
initial begin
n = 2;
$display(module level ‘n’ = %0d
,n);
end
initial begin : init2
int n; //Local - Block level
n = 3;
$display(block level ‘n’ = %0d
,n);
$unit::n = 4; //Static Global
$display(Statically declared ‘n’ = %0d
,$unit::n);
end
initial begin //hierarchical reference to local variable
$display(init2.n = %0d
, init2.n);
end
endmodule
module next;
//Static variable 'n' is visible in the module 'next'
initial begin
$display(Statically declared 'n' in module 'next' = %0d
,$unit::n);
end
endmodule
Simulation log:
module level ‘n’ = 2
block level ‘n’ = 3
Statically declared ‘n’ = 4
init2.n = 3
Statically declared 'n' in module 'next' = 4
V C S S i m u l a t i o n R e p o r t
First, we declare a static var. int n
outside of the module ‘vars’.
This makes it visible to all the modules that follow (note that when I use the term global
it does not mean a global variable. It is just a static variable visible to all the modules that follow. It is a static variable declared outside of a module. There are no global variables in SystemVerilog). You can have variables declared in the compilation unit scope $unit, but there can be multiple $units. You can never access things in one compilation unit from another.
Then, we declare another variable int n
inside the module vars
and assign it a value 2. This makes it a module-level var., visible to all the blocks under the module scope. Then we declare another variable int n
within the named block init2
and assign it the value 3. This makes it local only to that named block (except for hierarchical reference to it). Then we display the module level n
vs. the block (init2
block)-level n.
We get the following in simulation log. The module-level and block-level n
are in different scopes, and one does not clobber/override another:
module level n = 2
block level n = 3
In order to assign a value to the statically declared n,
we need to use $unit. This is done using $unit::n = 4;.
Note that this assignment will now be visible to all
modules that follow the declaration of the statically declared variable.
First, we display this variable using $unit:: and get the required result Global n = 4
as shown in the simulation log.
Then, we show that the init2
block-level n
can indeed be accessed hierarchically from other procedural blocks. This is shown by the following code:
initial begin //hierarchical reference to local variable
$display(init2.n = %0d
, init2.n);
end
And the simulation log shows this display as init2.n = 3.
Finally, we declare another module next.
This is to show that the globally declared n
(preceding the module vars
) is indeed accessible across module boundaries. When we display n
from module next,
we get the desired result (i.e., the value assigned to $unit::n in module vars
). This is shown in the simulation log as
Statically declared 'n' in module 'next' = 4
2.8.2 Automatic vs. Static Variable
Automatic variables are reallocated and initialized each time entering the block. This allows for reentrancy in SystemVerilog. In contrast, static variables are allocated and initialized once and do not reset to their value each time you enter a block. Here is an example to illustrate this point:
module autovars;
initial begin
for (int i=0; i<2; i++) begin
automatic int loop3 = 0; // executes every loop
for (int j=0; j<2; j++) begin
loop3++;
$display(loop3=%0d
,loop3);
end
end // loop3 = 1 2 1 2
for (int i=0; i<2; i++) begin
static int loop2 = 0; // executes once at time zero
for (int j=0; j<2; j++) begin
loop2++;
$display(loop2=%0d
,loop2);
end
end // loop2 = 1 2 3 4
end
endmodule : autovars
In this example, there are two for loops. The first for loop declares an automatic
variable called loop3
initialized to 0. Note the keyword automatic in its declaration. The second for loop declares a static
variable called loop2 initialized to 0. Note the keyword static
in its declaration.
Here is the simulation log. I will explain it right after the log:
loop3=1
loop3=2
loop3=1
loop3=2
loop2=1
loop2=2
loop2=3
loop2=4
In the first for loop, the automatic variable will reinitialize itself every time you reenter the loop. It is automatic and allows for reentrant code. The variable will reinitialize to 0 every time you enter the external for loop. This way when loop3 count is incremented, it will start from 0 and increment by 1 and 2. It will not clobber the initial value. That is why you see the following in the simulation log:
loop3=1
loop3=2
loop3=1
loop3=2
In contrast, the second for loop’s static variable will – not – reinitialize itself every time you enter the for loop. It’s initialized once and then increment successively until both for loops are over. Hence, it will increment as follows:
loop2=1
loop2=2
loop2=3
loop2=4
This example illustrates the difference between an automatic and a static variable. I will further explain the nuances of SystemVerilog’s reentrant (or lack thereof) behavior when it comes to reentering the procedural always
block. Is an always
block reentrant (my favorite interview question)? Hold on to your hats.
Note also that we have not yet discussed automatic
tasks or functions. That is another topic. But in brief, variables declared in an automatic task, function, or block are local in scope, are default to the lifetime of the call or block, and are initialized on each entry to the call or block. An automatic block is one in which declarations are automatic by default. Also, specific variables within an automatic task, function, or block can be explicitly declared as static. We will see examples on how automatic tasks and functions work in later sections.
2.8.3 Variable Lifetimes
SystemVerilog has three different kinds of variable lifetimes:
Static – exists for the entire life of the simulation. Initialized once at time 0. Can be referenced from outside the scope of where it is declared.
Automatic – a new instance gets created and initialized for each entry to the scope where it is declared (must be a procedural scope). Its lifetime ends when exiting the scope, and all nested scopes exit (to handle fork/join_none). Can only be referenced from within the scope where it is declared.
Dynamic – created by the execution of a procedural statement. Its lifetime can end a number of ways but normally by executing a procedural statement.
Dynamically sized arrays have a compound concept of lifetimes. Individual elements have dynamic lifetimes, but the array as a whole aggregate can have any of the above lifetimes. If we consider the array as an aggregate, it means that whenever the lifetime of an array variable ends, all the dynamically allocated elements are reclaimed.
Please refer to Chap. 8 on class to better understand the following statement.
Class objects have dynamic lifetimes, but the class variables that hold handles referencing class objects can have any of the above lifetimes. But since more than one class variable can reference the same class object, the lifetime of class object ends when there are no more class variables referencing that object. So, if that class object contains dynamic array variables, those variables lifetime end when the object lifetime ends.
SystemVerilog does not specify how garbage collection works. When the lifetime of something ends, you can no longer access it. There is no way to know when the memory actually gets reclaimed.
2.9 Enumerated Types
An enumerated type defines a set of named values. Enumerated variables can be declared without either a data type or data value(s). In the absence of a data-type declaration, the default data type will be int .
Let us look at some examples:
enum {red, green, blue} light1, light2;
In this example, no data type is specified, so the default int
data type is assumed. In this example, light1 and light2 are defined to be variables of the default type int that includes three members: red, green, and blue. No values are specified. So, red = 0, green = 1, and blue = 2. Values are assigned in an ascending order:
enum integer {IDLE, XX='x, S1='b01, S2='b10} state, next;
Here the data type is integer
which means you can assign x
and z
values to the enum members. Note that when values are assigned to the members of an enum type, they must be in ascending order. Here, since XX = ‘x, you have to explicitly assign values to S1 and S2 since the number system does not follow an x
(you can’t go from x
to 0
or 1
).
Here is an example that will cause an error since after XX = ‘x, you did not assign explicitly incrementing number to S1 (or S2). What values can follow x
?
enum integer {IDLE, XX='x, S1, S2} state, next;
Simulator will give you following type syntax error (Synopsys – VCS):
Syntax ERROR: Values assigned must be in ascending order (or assign explicit value to each member).
Let us look at more examples:
enum {bronze=3, silver, gold} medal;
In this example, we start off with bronze = 3 as the starting value. Hence, silver will be equal to 4 and gold will be equal to 5:
enum {a=3, b=7, c} alphabet;
In this example we are assigning explicit values to a
and b.
But we do not assign any value to c.
So, c
will take the value that follows b = 7. So, c
will be equal to 8:
enum {a=0, b=7, c, d=8} alphabet;
This will cause an error since c
will take on value 8 because it follows b = 7. But then you assign d = 8. So, both c
and d
will have value 8. You cannot have duplicate values and hence the error. Here is the error report by Synopsys – VCS:
Error-[ENUMDUPL] Duplicate labels in enum
The enum label 'd' has the value 4'd8 which is duplicate of enum label
'c' in the declared enum.
enum {a, b=7, c} alphabet;
In this example, a
is the first member and it does not have a value. So, it will start with 0.
So, the values are a = 0, b = 7, and c = 8:
enum bit [3:0] {red=‘h13, green, blue} color;
This will give an error because the enum is of type bit [3:0]
and the value assigned to red (=‘h13) is outside the range allowed by bit [3:0].
Here is the error report by Synopsys – VCS:
Error-[ENUMRANGE] Enum label outside value range
The enum label 'red' has the value 'h00000013 which is outside the range of
the base type of the declared enum, which is 4 bit unsigned.
enum bit [3:0] {red=‘d13, green, blue} color;
Similar to above example, but all the values fall in range of bit [3:0].
So, red = 13, green = 14, and blue = 15:
enum bit [3:0] {bronze=10, silver, gold=5} medal4;
Here, bronze = 10, silver = 11, and gold = 5.
Point in this example is that when you explicitly assign values, they may not be in an ascending order:
enum bit [3:0] {red, green, blue=3'h5} color;
Here, red = 0, green = 1, and blue = 5:
enum {color[4]} color_set;
Here, color0 = 0, color1 = 1, color2 = 2, and color3 = 3:
enum {color[3] = 5} color_set;
Here, color0 = 5, color1 = 6, and color2 = 7:
enum {color[3:5]} color_set;
Here, color3 = 0, color4 = 1, and color5 = 2.
Enums can be especially useful in design of finite-state machines. For example, you can define an enum as follows to describe the states of a state machine:
enum logic [1:0] { IDLE = 2’b00,
READ = 2’b01,
WRITE = 2’b10,
RMW = 2’b11,
ILLEGAL = ‘x } current_state, next_state;
Since the type is logic,
you can assign an unknown (‘x) value to an enum member. Such ‘x assignment is very useful for simulation debug, and synthesis doesn’t care optimization. This enum can then be used in state machine coding as follows:
always @(posedge clk, negedge reset)
if (!reset) current_state <= IDLE;
else current_state = next_state;
always @* begin
…
case (current_state)
IDLE : if (rdy) next_state = READ;
READ : if (go) next_state = WRITE;
…
endcase
end
2.9.1 Enumerated-Type Methods
Enumerated-type methods are offered by the language which facilitates extracting values of the members of the enum type. Table 2.11 describes these methods.
Table 2.11
Enumerated-type methods
Let us look at an example on how these methods work:
module enum_methods;
typedef enum { red, green, blue, yellow } Colors;
Colors c;
initial begin
$display(Number of members in Colors = %0d
,c.num);
c = c.first( );
$display(First member # = %0d
,c);
c = c.next(2);
$display(c = %0d
,c);
c = c.last ( );
$display(Last member # = %0d
,c);
$display( %s : %0d
, c.name, c );
end
endmodule
Simulation log:
run -all;
# KERNEL: Number of members in Colors = 4
# KERNEL: First member # = 0
# KERNEL: c = 2
# KERNEL: Last member # = 3
# KERNEL: yellow : 3
# KERNEL: Simulation has finished. There are no more test vectors to simulate.
exit
In this example, we define a typedef enum { red, green, blue, yellow } Colors;.
It has four members, with values, red = 0, green = 1, blue = 2, and yellow = 3.
We first display the number of members in the enum using the method num ().
There are four members in the enum. So, the display log shows Number of members in Colors = 4.
We then get the value of the first member using the method first().
As we see that the first member is red
with a value 0, we see that in the display First member = 0.
We then get the next
value of the enum. But we are using next(2).
We are at the first member red
which has a value of 0. So, the next(2) will give us the value 2, as shown in the simulation log.
Next, we get the value of the last member of the enum. That is value 3, as shown in the simulation log Last member # = 3.
Now, we are at the last member yellow.
So, when we display c.name,
we get the name of the member as yellow
and its value as 3, as shown in the simulation log yellow : 3.
Here is another simple example. Comments describe the functionality:
typedef enum { red, green, blue, yellow, white, black } Colors;
Colors col;
integer a, b;
a = blue * 3; // a = 6 (since blue is in 3rd position; blue= 2)
//(2*3=6)
a = yellow; //a = 3;
b = a + green; //b = 3 + 1 = 4
One more example:
module datatype1;
enum bit [3:0] {red, green, blue=5} color;
typedef enum { read=10, write[5], intr[6:8] } E1;
E1 e1 = write0; //initialize with a member of the enum
int i1;
initial
begin
i1 = green;
$display(i1 = %0d
,i1);
$display (e1.name=%s
,e1.name);
e1 = e1.last ( );
$display (e1.last=%0d e1.name=%s
,e1, e1.name);
$display (color.name = %s
, color.name);
$display (red=%s green=%d blue=%d
,color.name,green,blue); //OK
$display (red=%d green=%d blue=%d
,red,green,blue); //OK
$display (write0=%0d write1=%0d write2=%0d write3=%0d write4=%0d
, write0, write1, write2, write3, write4);
end
endmodule
Simulationlog:
i1 = 1
e1.name=write0
e1.last =18 e1.name=intr8
color.name = red
red=red green= 1 blue= 5
red= 0 green= 1 blue= 5
write0=11 write1=12 write2=13 write3=14 write4=15
V C S S i m u l a t i o n R e p o r t
A few things to note. We can initialize an enum type with a member of the enum, as in:
E1 e1 = write0; //initialize with a member of the enum
Then, when we display the name of the enum, we’ll get the initialized value as the name:
$display (e1.name=%s
,e1.name);
Simulation log shows:
e1.name=write0
Let us look at the following code:
typedef enum { read=10, write[5], intr[6:8] } E1;
e1 = e1.last ( );
$display (e1.last=%0d e1.name=%s
,e1,e1.name);
The value of the last member of the enum E1 is 18. That’s because the first member read = 10
is followed by write[5]
(five elements); so, 10 + 5 = 15. Then, intr[6:8]
(three elements), so 15 + 3 = 18. That is shown in the simulation log. And the last member is intr8.
In the simulation log:
e1.last =18 e1.name=intr8
Study the rest of the simulation log carefully to see what is going on.
2.9.2 Enumerated Type with Ranges
Table 2.12 shows that you can specify a range with a member of the enumerated type. Instead of declaring each member of a range separately, you can simply specify the required range with the member name. The name
in Table 2.12 means the name of a member of the enumerated type. We will understand this better with an example right after we declare the table.
Table 2.12
Enumeration element ranges
Here is an example of how these ranges with enumerated member work. We have so far seen member-assigned constants without any range:
module datatype1;
typedef enum { read=10, write[5], intr[6:8] } cycle;
enum { readreg[2] = 1, writereg[2:4] = 10 } reg0;
initial
begin
$display (read=%0d\n
, read);
$display (write0=%0d write1=%0d write2=%0d write3=%0d write4=%0d\n
, write0,write1,write2,write3,write4);
$display (intr6=%0d, intr7=%0d intr8=%0d\n
,intr6, intr7, intr8);
$display (readreg0=%0d readreg1=%0d\n
,readreg0, readreg1);
$display (writereg2=%0d writereg3=%0d writereg4=%0d\n
,writereg2, writereg3, writereg4);
end
endmodule
Here is how to interpret the example:
typedef enum { read=10, write[5], intr[6:8] } cycle;
defines an enum cycle
which has three members. First read
; then a range of five writes, namely, write0,
write1,
write2,
write3,
and write4
; and then a range of three intr,
namely, intr6,
intr7,
and intr8.
Similarly:
enum { readreg[2] = 1, writereg[2:4] = 10 } reg0;
defines an enum reg0
with two members: first, readreg0 and readreg1 and second writereg2, writereg3, and writereg4.
Here is the simulation log that shows how the values are assigned to members with a range.
Simulation log:
read=10
write0=11 write1=12 write2=13 write3=14 write4=15
intr6=16, intr7=17 intr8=18
readreg0=1 readreg1=2
writereg2=10 writereg3=11 writereg4=12
V C