SourceConstraint ChallengesSV Verification Directory

Problem Statement

Generate a sequence of non-repeating random numbers stored in an array. The solution should ensure that each number appears only once and should cover all possible values in the specified range. Do not use the randc function in SystemVerilog.

Randc

Randc is a built-in command in SystemVerilog designed to generate random variables in a cyclic manner. Unlike standard randomization methods, randc ensures that all possible values within the specified range are generated exactly once before any value repeats.

Problem Breakdown

  • Create an array to hold all the possible numbers. This array serves as a container for storing numbers.
  • Use a random number generator to produce numbers within the specified range.
  • If the generated random number is present in the array, do not add it to the array.
  • If the generated random number is not present in the array, add it to the array.
  • When the array is full, delete the array and repeat the process.

Implementation: Version 1

In this implementation, a constraint ensures that the generated random number is not already present in the array. To manage the array dynamically, the post_randomize function is utilized.

  • Constraint Check: The constraint validates that the newly generated random number is unique and does not exist in the array.
  • Updating the Array: After each successful randomization, the post_randomize function updates the array with the new number.
  • Handling Full Arrays: Within the post_randomize function, the array is cleared once it is completely full, allowing for a fresh sequence generation if needed.
class cstrs_challenge1 #(parameter N = 2);

   // class properties
   rand bit [N-1:0] x;
   bit      [N-1:0] q [$];

   constraint c {
      !(x inside {q});
   }

   // post randomization
   function void post_randomize();
      q.push_front(x);
      // if the is == to the number of combinations
      if(q.size() == 2**N)
        q.delete();
   endfunction // post_randomize

   // display function
   function void disp();
      $display("x = %d", x);
   endfunction // disp

endclass // cstrs_challenge1

Implementation: Version 2

In the second implementation, the process uses both pre_randomize and post_randomize functions alongside a constraint to achieve unique random number generation.

  • Index-Based Randomization: Instead of directly randomizing the numbers, the randomized value represents the index of the array.
  • Constraint: A constraint ensures that the randomized index value is within the valid range, i.e., the current size of the array.

How It Works:

  • pre_randomize Function:

    • If the array is empty, this function fills the array with sequential values from 0 to the specified range.
    • This initializes the array as the source of numbers to be randomized.
  • post_randomize Function:

    • The random index generated during randomization is used to fetch the corresponding value from the array.
    • After the random number is output, the element at the random index is replaced by the last element in the array.
    • The array size is then reduced by one, effectively “removing” the used number.
class cstrs_challenge2 #(int N = 2);

   // class properties
   bit [N-1:0] x;
   rand int  rand_idx;

   bit [N-1:0] q[$];

   constraint c {
      rand_idx >= 0;
      rand_idx < q.size();
   }

   // pre randomization
   function void pre_randomize();
      if(q.size() == 0) begin
         for(int i = 0; i < 2**N; i++) begin
            q.push_back(i);
         end
      end
   endfunction // pre_randomize

   // post randomization
   function void post_randomize();
      x = q[rand_idx];

      q[rand_idx] = q[q.size()-1];

      q.delete(q.size()-1);
   endfunction // post_randomization

   // display function
   function void disp();
      $display("x = %d", x);
   endfunction // disp

endclass // cstrs_challenge2