System verilog

PACKED vs UNPACKED ARRAY

logic [3:0][7:0] register [0:9];

PACKED UNPACKED


PACKED - Assigned in contiguous location in memory. Part select is supported. (restricted to bit, logic, int etc)

UNPACKED - Can be assigned in random location in memory. No part select support.

ENUMERATION and TYPEDEF

Enumerations allow you to define a data type whose values have names. Such data types are appropriate and useful for representing state values, opcodes and other such non-numeric or symbolic data.

Typedef allows user to create custom data type.

TYPEDEF example:

typedef logic [7:0] octet;

typedef octet [3:0] quadOctet;

quadOctet qBytes [1:10];


or


logic [7:0] [3:0] qBytes [1:10];


ENUM:

enum { circle, ellipse, freeform } c;


typedef enum { circle, ellipse, freeform } ClosedCurve;

ClosedCurve c;

The named values of an enumeration type act like constants. The default type is int. You can copy them to and from variables of the enumeration type, compare them with one another and so on. Enumerations are strongly typed. You can’t copy a numeric value into a variable of enumeration type, unless you use a type-cast:

c = 2; // ERROR

c = ClosedCurve'(2); // Casting – okay


However, when you use an enumeration in an expression, the value you are working with is the literal’s integer equivalent; so, for example, it’s okay to compare an enumeration variable with an integer; and it’s okay to use an enumeration value in an integer expression.

Sample param using struct

module regSample;

typedef struct packed {

logic [7:0] offset;

logic [7:0] bits;

}bitDescr;

parameter bitDescr regF [0:1] = '{'{8'h1,8'h4}, '{8'h2,8'h8}};

logic [regF[0].bits -1 : 0] sampleReg0;

logic [regF[1].bits -1 : 0] sampleReg1;

initial

begin

sampleReg0 = regF[0].offset;

$display ("Sample0 = %d",sampleReg0);

sampleReg1 = 8'hFF;

$display ("Sample1 = %d",sampleReg1);

end

endmodule

Issue with nested struct

A field name of a struct is not another type or a variable.

This is an issue as getting the width of a field is not possible (since $bits only take "type" as input)


For e.g.

typedef struct packed {

logic [1:0] f1;

logic [2:0] f2;

} sample_t;


parameter STRUCT_WIDTH = $bits(sample_t);

parameter FIELD_WIDTH = $bits(sample_t.f1); // <- *Errors out


An alternate solution is to create a typedef for the field being used.

i.e.

typedef logic [1:0] f1;

typedef struct packed {

f1_t f1;

logic [2:0] f2;

} sample_t;


parameter STRUCT_WIDTH = $bits(sample_t);

parameter FIELD_WIDTH = $bits(f1_t);


Note: There is a suggestion on forum (from Dave rich) --> "Declare an automatic variable local to the scope where you need the $bits. It will probably get optimized away, but not before the type information gets extracted for $bits."


Struct debug

To debug struct use the %p format specifier. This will print the entire struct in a clean formatted way.

e.g.:

// Struct definition

typedef struct packed {

bit [7:0] addr;

bit valid;

bit [31:0] data;

} mem_pkt;


// Struct declaration

mem_pkt pkt;


// Initializing Struct

pkt = '{8'h6, 1'b1, 32'hC001_0FAB};

$display ("pkt = %p", pkt);


// This will print.

pkt = '{addr:6, valid:1, data:3221295019}

Format specifiers

Systemverilog supports a list of new format specifiers.

%p -> Print struct in a formatted way

%m -> Print the hierarchy of the design element, named block etc.

Other tips n tricks:

  1. Size of a design elements, $bits operator can be used in combination with concatenation operator.

    • Example:

$display("Port width = %0d", $bits({port1,port2,....}));