



# **DIGITAL DESIGN VERIFICATION & TESTING**

0110  
0100 0101  
1001 1001 0110  
0101 0100  
0110  
0101 1001 0110  
1001 0101 1100  
**SLIDE SET 4**  
**SYSTEM VERILOG**

**CSE 313s**

**A y m a n   w a h b a**



## System Verilog Types

2

| System Verilog Types | Data type | 2-state/4-state | No. of bits | Signed/unsigned | Verilog Types |
|----------------------|-----------|-----------------|-------------|-----------------|---------------|
|                      | reg       | 4               | $\geq 1$    | unsigned        |               |
|                      | wire      | 4               | $\geq 1$    | unsigned        |               |
|                      | integer   | 4               | 32          | Signed          |               |
|                      | real      |                 |             |                 |               |
|                      | time      |                 |             |                 |               |
|                      | realtime  |                 |             |                 |               |
|                      | logic     | 4               | $\geq 1$    | unsigned        |               |
|                      | bit       | 2               | $\geq 1$    | unsigned        |               |
|                      | Byte      | 2               | 8           | signed          |               |
|                      | Shortint  | 2               | 16          | signed          |               |
|                      | Int       | 2               | 32          | signed          |               |
|                      | Longint   | 2               | 64          | signed          |               |
|                      | shortreal |                 |             |                 |               |

- So what are the 4 states mentioned above ?
- 0: variable/net is at 0 volts
- 1: variable/net is at some value  $> 0.7$  volts
- x or X: variable/net has either 0/1 - we just don't know
- z or Z: net has high impedance - may be the wire is not connected and is floating.



## The “Logic” data type

3

```

1 //-----
2 module logic_data_type();
3   parameter CYCLE = 20;
4   logic q, q_1, d, clk, rst_h, rst_l;
5
6   initial begin
7     clk = 0; // Procedural assignment
8     forever #(CYCLE/2) clk = ~clk;
9   end
10
11  initial begin
12    d = 1;
13    rst_h = 1;
14    #15 rst_h = 0;
15    #7 d = 0;
16    #30 d = 1;
17    #2 d = 0;
18    #6 d = 1;
19  end
20
21  assign rst_l = ~rst_h; // Continuous assignment
22  not n1(q_1, q); // q_1 is driven by gate
23  my_dff d1(q, d, clk, rst_l); // q is driven by module
24
25  initial
26  begin
27    $dumpfile("uvm.vcd");
28    $dumpvars;
29    #200 $finish;
30
31  endmodule
32

```

```

1 //-----
2 module my_dff(q, d, clk, rst_l);
3
4 //-----Input Ports-----
5 input d, clk, rst_l;
6 //-----Output Ports-----
7 output q ;
8 //-----Input ports Data Type-----
9 // By rule all the input ports should be wires
10 wire d,clk,rst_l;
11 //-----Output Ports Data Type-----
12 // Output port can be a storage element (reg) or a wire
13 reg q;
14
15 //-----Code Starts Here-----
16 // Since this f/f is a positive edge triggered one,
17 // We trigger the below block with respect to positive
18 // edge of the clock.
19
20 always @ (posedge clk)
21 begin
22   if (rst_l == 1'b0) begin
23     q <= 1'b0;
24   end
25   else q <= d;
26 end // end of Block
27 endmodule // end of module

```



- The one thing in verilog that always makes confusion is the difference between a *reg* and a *wire*.
- When driving a port, which should you use? How about when you are connecting blocks?
  - *reg <= reg* or *wire* (non-blocking assignment)
  - *reg = reg* or *wire* (blocking assignment)
  - *assign wire = reg* or *wire* (continuous)
- System Verilog improves the classic *reg* data type so that it can be driven by continuous assignments, gates, and modules, in addition to being a variable. It is given the synonym *logic* so that it does not look like a register declaration.
- A logic type is a 4-state type (1,0,x,z)
- A logic signal can be used anywhere a net is used,
- A logic variable cannot be driven by multiple structural drivers, such as when you are modeling a bidirectional bus. In this case, the variable needs to be a net-type such as *wire* so that System Verilog

can resolve the multiple values to determine the final value.



4

## Arrays

**Static Arrays**

**Dynamic Arrays**

**Associative Arrays**

- Datatypes in Verilog are not sufficient to develop efficient testbenches and testcases. Hence SystemVerilog has extended Verilog by adding more data-types for better encapsulation and compactness.

# Fixed Size Arrays Declaration



5

```
1 module arrays();
2
3     int lo_hi[0:15]; // 16 ints [0]..[15]
4     int c_style[16]; // 16 ints [0]..[15]
5
6     int array2 [0:7][0:3]; // Verbose declaration
7     int array3 [8][4];    // Compact declaration
8
9     initial begin
10        lo_hi[3] = 234;   // Set one array element
11        array2[7][3] = 1; // Set last array element
12    end
13 endmodule
```

- SystemVerilog offers several flavors of arrays beyond the single-dimension, fixed size Verilog-1995 arrays, as many enhancements have been made to these classic arrays.
  - Verilog requires that the low and high array limits must be given in the declaration. Since almost all arrays use a low index of 0, SystemVerilog lets you use the shortcut of just giving the array size, which is similar to C's style.
  - You can create multidimensional fixed-size arrays by specifying the dimensions after the variable name.
  - If your code accidentally tries to read from an out-of-bounds address, SystemVerilog will return the default value for the array element type:
    - That just means that an array of 4-state types, such as logic, will return X's.
    - whereas an array of 2-state types, such as int or bit, will return 0.
- This applies also if your address has an X or Z.



# The Array Literal ' {}'

6

```
1 module arrays();
2
3     int ascend[4] = '{0,1,2,3}; // Initialize 4 elements
4     int descend[5];
5
6     initial begin
7         $display("1: ascend: ",ascend);
8         $display("2: descend: ",descend);
9
10    descend = '{4,3,2,1,0}; // Set 5 elements
11    $display("3: descend: ",descend);
12
13    descend[0:2] = '{5,6,7}; // Set first 3 elements {5,6,7,1,0}
14    $display("4: descend: ",descend);
15
16    ascend = '{4{8}};           // Four values of 8 {8,8,8,8}
17    descend = '{default:-1}; // {-1, -1, -1, -1, -1}
18    $display("5: ascend: ",ascend);
19    $display("6: descend",descend);
20
21 end
21 endmodule
```

```
1: ascend: '0, 1, 2, 3
2: descend: '0, 0, 0, 0, 0
3: descend: '4, 3, 2, 1, 0
4: descend: '5, 6, 7, 1, 0
5: ascend: '8, 8, 8, 8
6: descend'{-1, -1, -1, -1, -1}
```

- You can initialize an array using an array literal, which is an apostrophe followed by the values in curly braces. ' {}'
  - You can set some or all elements at once.
  - You are able to replicate values by putting a count before the curly braces.
  - Lastly, you might specify a default value for any element that does not have an explicit value.



## Array Operations for and foreach

7

```
1 module arrays();
2
3   initial begin
4     bit [31:0] src[5], dst[5];
5     for (int i=0; i<$size(src); i++)
6       src[i] = i;
7     foreach (dst[j])      // j is automatically declared
8       dst[j] = src[j] * 2; // dst doubles src values
9
10    foreach (src[j])
11      $display("src[%0d]= %b = %0d    dst[%0d]= %b = %0d", j, src[j],
12        src[j], j, dst[j], dst[j]);
13
14  end
15 endmodule
```

```
src[0]= 00000000000000000000000000000000 = 0    dst[0]= 00000000000000000000000000000000 = 0
src[1]= 00000000000000000000000000000001 = 1    dst[1]= 00000000000000000000000000000002 = 2
src[2]= 000000000000000000000000000000010 = 2   dst[2]= 0000000000000000000000000000000100 = 4
src[3]= 000000000000000000000000000000011 = 3   dst[3]= 0000000000000000000000000000000110 = 6
src[4]= 0000000000000000000000000000000100 = 4  dst[4]= 00000000000000000000000000000001000 = 8
V C S   S i m u l a t i o n   R e p o r t
```

- You The most common way to manipulate an array is with a for or foreach loop.
- The SystemVerilog function \$size returns the size of the array.
- In the foreach-loop, you specify the array name and an index in square brackets, and SystemVerilog automatically steps through all the elements of the array. The index variable is automatically declared for you and is local to the loop.

# Array Operations for and foreach (2-D array)



8

```
1 module arrays();
2   initial begin
3     int two_d[3][5];           // 3 rows and 5 columns
4     foreach (two_d[i,j])
5       two_d[i][j] = 10*i+j;  // initialization
6
7     foreach (two_d[i]) begin // Step through the rows
8       $write("%2d:", i);    // write in a buffer
9       foreach(two_d[,j])   // Step through the columns
10         $write("%3d", two_d[i][j]);
11       $display;            // Display what is in the buffer
12     end
13   end
14 endmodule
```

```
0: 0 1 2 3 4
1: 10 11 12 13 14
2: 20 21 22 23 24
V C S  Simulation Report
```



## Array Operations copy and compare

9

```
1 module arrays();
2   initial begin
3     bit [31:0] src[5] = '{0,1,2,3,4},
4       dst[5] = '{5,4,3,2,1};
5   // Aggregate compare the two arrays
6   if (src==dst)
7     $display("src == dst");
8   else
9     $display("src != dst");
10  // Aggregate copy all src values to dst
11  dst = src;           // dst = src = {0,1,2,3,4}
12  // Change just one element
13  src[0] = 5;          // dst = {5,1,2,3,4}
14  // Are all values equal (no!)
15  $display("src %s dst", (src == dst) ? "==" : "!=");
16  // Use array slice to compare elements 1-4
17  $display("src[1:4] %s dst [1:4]",
18    (src[1:4] == dst[1:4]) ? "==" : "!=");
19  end
20 endmodule
```

```
src != dst
src != dst
src[1:4] == dst [1:4]
```

- You can perform aggregate *compare* and *copy* of arrays without loops. (An aggregate operation works on the entire array as opposed to working on just an individual element.)
- Comparisons are limited to just *equality* and *inequality*.
- In line 15, line 18, The “? :” conditional operator is a mini if-statement. In the above code it is used to choose between two strings.
- The final compare uses an array slice, src[1:4], which creates a temporary array with four elements.
- You cannot perform aggregate *arithmetic operations* such as addition on arrays. Instead, you can use loops. For *logical operations* such as xor, you have to either use a loop or use *packed arrays* as we will see later.



## Bit and Array Subscripts together

10

```
1 module arrays();
2   initial begin
3     bit [31:0] src[5] = '{5{5}};
4
5   foreach (src[j])
6     $displayb("src[%0d]= %b",j,src[j]);
7
8   $displayb(src[0],           // 'b101 or 'd5
9             src[0][0],       // 'b1
10            src[0][2:1]);  // 'b10
11
12 end
13 endmodule
```

```
src[0]= 00000000000000000000000000000000101
src[1]= 00000000000000000000000000000000101
src[2]= 00000000000000000000000000000000101
src[3]= 00000000000000000000000000000000101
src[4]= 00000000000000000000000000000000101
00000000000000000000000000000000101 1 10
```

- The above code prints the first array element (binary 101), its lowest bit (1), and the next two higher bits (binary 10).
- Remarks:
  - Note that \$displayb is used to display in binary format.
  - Not that you can use two commas “, ,” in the arguments of the display function to leave a blank space.



## Dynamic Arrays

11

```

1 module arrays();
2   int dyn[], d2[];
3   initial begin
4     dyn = new[5];           // Declare dynamic arrays
5     foreach (dyn[j]) dyn[j] = j; // A: Allocate 5 elements
6     d2 = dyn;             // B: Initialize the elements
7     d2[0] = 5;            // C: Copy a dynamic array
8     $display(dyn,d2);    // D: Modify the copy
9     dyn = new[20](dyn);   // E: See both values
10    $display(dyn);       // F: Allocate 20 ints & copy
11    dyn = new[40];       // G: Allocate 40 new ints
12    $display(dyn);       // Old values are lost
13    $display(dyn);
14    $display("Size= %0d", $size(dyn));
15    dyn.delete();        // H: Delete all elements
16    $display(dyn);
17  end
18 endmodule

```

```

'{0, 1, 2, 3, 4} '{5, 1, 2, 3, 4}
'{0, 1, 2, 3, 4, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}
{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}
Size= 40
'{}

```

VCS Simulation Report

- A fixed-size array, has its size set a priori. What if the size of the array isn't known yet?
- For example, you may want to generate a random number of transactions at the start of simulation. If you stored the transactions in a fixed-size array, it would have to be large enough to hold the maximum number of transactions, but would typically hold fewer transactions, thus wasting memory.
- A dynamic array can be allocated and resized during simulation, so your simulation will consume a minimal amount of memory.
- A dynamic array is declared with empty word square brackets [ ] (see line 2). This means that you do not specify the array size in advance; instead, you give it at run-time.
- The array is initially empty, and so you must call the new[ ] constructor to allocate space, passing in the number of entries in the square brackets.
- When you copy a fixed-size array to a dynamic array, SystemVerilog calls the new[ ] constructor to allocate space, and then copies the values.
  - Line A calls new[5] to allocate 5 array elements.
  - Line B sets the value of each element of the array to its index value.
  - Line C allocates another array and copies the contents of dyn into it.
  - Lines D and E show that the arrays dyn and d2 are separate.
  - Line F allocates 20 new elements, and copies the existing 5 elements of dyn to the beginning of the array. The old 5-element array is deallocated. The result is that dyn points to a 20-element array.
  - Line G allocates 40 elements, but the existing values are not copied. The old 20-element array is deallocated.
  - Finally, line H deletes the dyn array.

- The `$size` function returns the size of an array. Dynamic arrays have several built-in routines, such as `delete` and `size`.



# Associative Arrays

12

- Imagine you are modeling a processor that has a multi-gigabyte address range.
- During a typical test, the processor may only touch a few locations containing executable code and data
- Allocating and initializing gigabytes of storage is wasteful.
- Associative arrays is a good solution in these cases.
- This means that while you can address a very large address space, SystemVerilog only allocates memory for an element when you write to it.



- Dynamic arrays are good if you want to occasionally create a big array, but what if you want something really large?
- Perhaps you are modeling a processor that has a multi-gigabyte address range.
- During a typical test, the processor may only touch a few hundred or thousand memory locations containing executable code and data, so allocating and initializing gigabytes of storage is wasteful.
- SystemVerilog offers associative arrays that store entries in a sparse matrix. This means that while you can address a very large address space, SystemVerilog only allocates memory for an element when you write to it.
- In the shown figure, the associative array holds values in locations 0, 3, 42, 1000, 4521, and 200,000. The memory used to store these is far less than would be needed to store a fixed or dynamic array with 200,000 entries.



## Associative Arrays (continued)

13

```
1 module arrays();
2     initial begin
3         //int idx = 1;
4         bit [39:0] assoc[int];
5         int idx = 1;
6         // Initialize widely scattered values
7         repeat (20) begin
8             assoc[idx] = idx;
9             idx = idx << 1;
10            end
11
12            // Step through all index values with foreach
13            foreach (assoc[i])
14                $display("assoc[%0d] = %0d", i, assoc[i]);
15
16            $display("xxxxxxxxxxxxxxxxxxxxxx");
17            // Step through all index values with functions
18            if assoc.first(idx)
19                begin // Get first index
20                    do
21                        $display("assoc[%0d]=%0d", idx, assoc[idx]);
22                        while (assoc.next(idx)); // Get next index
23                    end
24
25            // Find and delete the first element
26            assoc.first(idx);
27            assoc.delete(idx);
28            $display("The array now has %0d elements", assoc.num);
29        end
30    endmodule
```

```
assoc[1]=1
assoc[2]=2
assoc[4]=4
assoc[8]=8
assoc[16]=16
assoc[32]=32
assoc[64]=64
assoc[128]=128
assoc[256]=256
assoc[512]=512
assoc[1024]=1024
assoc[2048]=2048
assoc[4096]=4096
assoc[8192]=8192
assoc[16384]=16384
assoc[32768]=32768
assoc[65536]=65536
assoc[131072]=131072
assoc[262144]=262144
assoc[524288]=524288
The array now has 19 elements
```

- An associative array is declared with a data type in square brackets, such as [int].
- The shown code shows declaring, initializing, and stepping through an associative array.
- In the above code we have an associative array, “assoc”, with very scattered elements: 1, 2, 4, 8, 16, etc.
- A simple for-loop cannot step through them; you need to use a foreach loop.
- If you want finer control, you can use the first and next functions in a do...while loop. These functions modify the index argument, and return 0 or 1 depending on whether any elements are left in the array.
- Used methods in the above code: first(<index>), next(<index>), delete(<index>).



## Associative Arrays (continued)

14

```
1 module arrays();
2 /*
3 Input file contains:
4 42 min_address
5 1492 max_address
6 */
7
8 int switch[string]; // Associative array searchable by strings
9 int min_address, max_address;
10 initial begin
11     int i, file;
12     string s;
13     myfile = $fopen("switch.txt", "r"); // create a file handler
14     while (! $feof(myfile)) begin
15         $fscanf(myfile, "%d %s", i, s);
16         switch[s] = i; // switch is an associative array
17     end
18     $fclose(myfile);
19
20 // Get the min address, default is 0
21 min_address = switch["min_address"];
22 // Get the max address, default = 1000
23 if (switch.exists("max_address"))
24     max_address = switch["max_address"];
25 else
26     max_address = 1000;
27 // Print all switches
28 foreach (switch[s])
29     $display("switch['%s']=%0d", s, switch[s]);
30 end
31 endmodule
```

switch['max\_address']=1492  
switch['min\_address']=42

- Associative arrays can also be addressed with a string index, similar to Perl's hash arrays.
- The above code reads a file with strings and builds the associative array so that you can quickly map from a string value to a number.
- Strings are explained later. You can use the function `exists()` to check if an element exists.
- If you try to read an element that has not been written, SystemVerilog returns the default value for the array type, such as 0 for 2-state types, or X for 4-state types.
- Used methods in the above code: `exists(<index>)`.



# Queues

15

```
1 module arrays();
2   int j = 1,
3   q2[$] = {3,4}, // Queue literals do not use '
4   q[$] = {0,2,5}; // {0,2,5}
5   initial begin
6     q.insert(1, j); // {0,1,2,5} Insert 1 before 2
7     $display(q);
8     q.delete(1); // {0,2,5} Delete elem. #1
9     // These operations are fast
10    q.push_front(6); // {6,0,2,5} Insert at front
11    j = q.pop_back(); // {6,0,2} j = 5
12    q.push_back(8); // {6,0,2,8} Insert at back
13    j = q.pop_front(); // {0,2,8} j = 6
14    q.delete(); // {} Delete entire queue
15    $display(q);
16  end
17 endmodule
```

{0, 1, 2, 5}  
{}

- SystemVerilog introduces a new data type, the queue, which combines the best of a linked list and array.
- You can add or remove elements anywhere in a queue, without the performance hit of a dynamic array that has to allocate a new array and copy the entire contents.
- Like an array, you can directly access any element with an index, without linked list's overhead of stepping through the preceding elements.
- A queue is declared with word subscripts containing a dollar sign: `[$]`. (lines 3, 4)
- The code above shows how you can add and remove values from a queue using methods.
- Note that queue literals only have curly braces, and are missing the initial apostrophe of array literals.
- If you add enough elements that the queue runs out of that extra space, SystemVerilog automatically allocates more. As a result, you can grow and shrink a queue without the performance penalty of a dynamic array, and SystemVerilog keeps track of the free space for you.
- Note that you never call the `new[]` constructor for a queue.
- Used methods in the above code: `insert(<index>, <value>)`, `delete(<index>)`, `delete()`, `push_front(<value>)`, `push_back(<value>)`,



## Queues (continued)

16

```
1 module arrays();
2   int j = 1,
3     q2[$] = {3,4}, // Queue literals do not use '
4     q[$] = {0,2,5}; // {0,2,5}
5   initial begin
6     q = {q[0], j, q[1:$]}; // {0,1,2,5} Insert 1 before 2
7     q = {q[0:2], q2, q[3:$]}; // {0,1,2,3,4,5} Insert queue in q
8     q = {q[0], q[2:$]}; // {0,2,3,4,5} Delete elem. #1
9     // These operations are fast
10    q = {6, q}; // {6,0,2,3,4,5} Insert at front
11    j = q[$]; // j = 5
12    q = q[0:$-1]; // {6,0,2,3,4} pop_back } Equivalent to pop_back
13    q = {q, 8}; // {6,0,2,3,4,8} Insert at back
14    j = q[0]; // j = 6
15    q = q[1:$]; // {0,2,3,4,8} pop_front } Equivalent to pop_front
16    q = {};// {} Delete entire queue
17 end
18 endmodule
```

- You can use concatenation instead of methods.
- As a shortcut, if you put a \$ on the left side of a range, such as [:\$:2], the \$ stands for the minimum value, [0:2].
- If you put \$ on the right side, as in [1:\$], stands for the maximum value, [1:2], in first line of the initial block shown above.
- The queue elements are stored in contiguous locations, and so it is efficient to push and pop elements from the front and back. This takes a fixed amount of time no matter how large the queue.
- Adding and deleting elements in the middle of a queue requires shifting the existing data to make room. The time to do this grows linearly with the size of the queue.
- You can also copy the contents of a fixed or dynamic array into a queue.



## Array Methods

17

- **Array Reduction Methods**
- **Array Locator methods**
- **Array sorting and ordering methods**

- There are many array methods that you can use on any unpacked array types: fixed, dynamic, queue, and associative. These routines can be as simple as giving the current array size or as complex as sorting the elements.

# Array Reduction Methods

(sum, product, and, or, xor)



18

```
1 module arrays();
2   bit on[10]; // Array of single bits
3   int total;
4   initial begin
5     foreach (on[i])
6       on[i] = i; // on[i] gets 0 or 1
7   // Print the single-bit sum
8   $display("on.sum = %0d", on.sum); // on.sum 1
9   // Compute with 32-bit signed arithmetic
10  $display("int sum=%0d", on.sum with (int'(item)));
11 end
12 endmodule
```

on.sum = 1

int sum=5

- A basic array reduction method takes an array and reduces it to a single value.
- The most common reduction method is sum, which adds together all the values in an array.
- Be careful of SystemVerilog's rules for handling the width of operations. By default, if you add the values of a single-bit array, the result is a single bit.
- However, if you use the proper with expression, System Verilog uses 32-bits when adding up the values. The with expression is described in full details later.
- Other array reduction methods are product, and, or, and xor.



## Array Reduction Methods

(sum ... with (condition on item))

19

```
1 module arrays();
2   int count, total, d[] = '{9,1,8,3,4,4};
3   initial begin
4     count = d.sum with (int'(item > 7));           // 2: {9, 8}
5     $display(count);
6     total = d.sum with ((item > 7) * item);        // 17= 9+8
7     $display(total);
8     count = d.sum with (int'(item < 8));           // 4: {1, 3, 4, 4}
9     $display(count);
10    total = d.sum with (item < 8 ? item : 0);      // 12=1+3+4+4
11    $display(total);
12    count = d.sum with (int'(item == 4));           // 2: {4, 4}
13    $display(count);
14  end
15 endmodule
```

2  
17  
4  
12  
2

- The code above shows various ways to total up a subset of the values in the array.
- When you combine an array reduction such as sum using the with clause, the results may surprise you.
- The sum operator totals the number of times that the expression is true. For the first statement there are two array elements that are greater than 7 (9 and 8) and so count is set to 2.
- The first total compares the item with 7. This relational returns a 1 (true) or 0 (false) and multiplies this with the array. So the sum of {9,0,8,0,0,0} is 17.
- The second total is computed using the ? : conditional operator.



## Array Locator Methods

(min, max, unique)

20

```
1 module arrays();
2   int f[6] = '{1,6,2,6,8,6};      // fixed array
3   int d[] = '{2,4,6,8,10};       // dynamic array
4   int q[$] = {1,3,5,7}, tq[$]; // queue
5   initial begin
6     tq = q.min;                // {1}
7     $display(tq);
8     tq = d.max;                // {10}
9     $display(tq);
10    tq = f.unique;             // {1,6,2,8}
11    $display(tq);
12  end
13 endmodule
```

{1}  
{10}  
{1, 6, 2, 8}

- What is the largest value in an array?
- Does an array contain a certain value?
- The array locator methods find data in an array.
- These methods always **return a queue**.
- The shown code uses:
  - a fixed-size array, f [6],
  - a dynamic array, d [ ],
  - and a queue, q [ \$ ].
- The min and max functions find the smallest and largest elements in an array.
- Note that they return a queue, not a scalar as you might expect.
- These methods also work for associative arrays.
- The unique method returns a queue of the unique values from the array - duplicate values are not included.

## Array Locator Methods

(find ... with (condition on item))



21

```
1 module arrays();
2   initial begin
3     int d[] = '{9,1,8,3,4,4}, tq[$];
4 // Find all elements greater than 3
5   tq = d.find with (item > 3); // {9,8,4,4}
6 // Equivalent code
7   tq.delete();
8   foreach (d[i])
9     if (d[i] > 3)
10      tq.push_back(d[i]);
11 // with expression
12   tq = d.find_index with (item > 3); // {0,2,4,5}
13   tq = d.find_first with (item > 99); // {} Ø none found
14   tq = d.find_first_index with (item==8); // {2} d[2]=8
15   tq = d.find_last with (item==4); // {4}
16   tq = d.find_last_index with (item==4); // {5} d[5]=4
17 end
18 endmodule
```

- You could search through an array using a foreach-loop.
- SystemVerilog can do this in one operation with a locator method.
- The with expression tells SystemVerilog how to perform the search, as shown above.
- In a with clause, the name item is called the iterator argument and represents a single element of the array.
- You can specify your own name by putting it in the argument list of the array method. For example:

tq = d.find\_first(x) with (x>4);

- The array locator methods that return an index, such as find\_index, return a queue of type int, not integer. Your code may not compile if you use the wrong queue type with these statements.



## Array Sorting & Ordering

(reverse, sort, rsort, shuffle)

22

```
1 module arrays();
2   initial begin
3     int d[] = '{9,1,8,3,4,4};
4     $display("Original Array: ", d);
5
6     d.reverse(); // reverses the order of the array
7     $display("Reversed Array: ", d);
8
9     d.sort(); // sort the array
10    $display("Sorted Array: ", d);
11
12    d.rsort(); // reverse sort the array
13    $display("Reverse Ordered Array: ", d);
14
15    d.shuffle();
16    $display("Shuffled Array 1: ", d);
17
18    d.shuffle();
19    $display("Shuffled Array 2: ", d);
20
21  end
22 endmodule
```

```
Original Array: '{9, 1, 8, 3, 4, 4}
Reversed Array: '{4, 4, 3, 8, 1, 9}
Sorted Array: '{1, 3, 4, 4, 8, 9}
Reverse Ordered Array: '{9, 8, 4, 4, 3, 1}
Shuffled Array 1: '{4, 8, 4, 9, 1, 3}
Shuffled Array 2: '{9, 1, 3, 4, 4, 8}
```

- SystemVerilog has several methods for changing the order of elements in an array.
- You can sort the elements, reverse their order, or shuffle the order.
- Notice that these methods change the original array, unlike the array locator methods which create a queue to hold the results.