Section 9.2

LRM-11

Changes (change in red):

 

The always_comb procedure provides functionality that is different than a normal always procedure:

 

      There is an inferred sensitivity list that includes the expressions defined in Section 9.2.1 every variable read by the procedure.

      The variables written on the left-hand side of assignments shall not be written to by any other process.

      The procedure is automatically triggered once at time zero, after all initial and always blocks have been started, so that the outputs of the procedure are consistent with the inputs.

 

Section 9.2.1 (new)

LRM-11

Add the following section (change in red):

 

9.2.1 Implicit always_comb sensitivies

 

The expansion of longest static prefix “P” is defined to be:

 

a) P itself if the P is not a memory or indexing select or if P is a legal word or bit select.

b) if P is a memory or indexing select, the expansion is every possible legal memory word select with a static prefix that matches P.

 

The implicit sensitivity list of an always_comb includes the expansions of the longest static prefix of each variable or select expression that is read within the block or within any function called within the block with the following exceptions:

 

1) any expansion of a variable declared within the block or within any function called within the block.

2) any expression that is also written within the block or within any function called within the block

 

Section 9.9 (new)

LRM-24

Add the following section (change in red):

 

9.9 Fine-grain process control

 

A process is a built-in class that allows one process to access and control another process once it has started. Users can declare variables of type process and safely pass them through tasks or incorporate them into other objects. The prototype for the process class is:

 

class process;

enum state { FINISHED, RUNNING, WAITING, SUSPENDED, KILLED };

 

static function process self();

function state status();

task kill();

task await();

task suspend();

task resume();

endclass

 

Objects of type process are created internally when processes are spawned. Users may not create objects of type process; attempts to call new shall not create a new process, and instead result in an error. The process class cannot be extended. Attempts to extend it shall result in a compilation error. Objects of type process are unique; they become available for reuse once the underlying process terminates and all references to the object are discarded.

 

The self()function returns a handle to the current process, that is, a handle to the process making the call.

 

The status() function returns the process status, as defined by the state enumeration:

 

      FINISHED Process terminated normally.

      RUNNING Process is currently running (not in a blocking statement).

      WAITING Process is waiting in a blocking statement.

      SUSPENDED Process is stopped awaiting a resume.

      KILLED Process was forcibly killed (via kill or disable).

 

The kill() task terminates the given process and all its sub-processes, that is, processes spawned using fork statements by the process being killed. If the process to be terminated is not blocked waiting on some other condition, such as an event, wait expression, or a delay then the process shall be terminated at some unspecified time in the current time step.

 

The await() task allows one process to wait for the completion of another process. It shall be an error to call this task on the current process, i.e., a process may not wait for its own completion.

 

The suspend() task allows a process to suspend either its own execution or that of another process. If the process to be suspended is not blocked waiting on some other condition, such as an event, wait expression, or a delay then the process shall be suspended at some unspecified time in the current time step. Calling this method more than once, on the same (suspended) process, has no effect.

 

The resume() task restarts a previously suspended process. Calling resume on a process that was suspended while blocked on another condition shall re-sensitize the process to the event expression, or wait for the wait condition to become true, or for the delay to expire. If the wait condition is now true or the original delay has transpired, the process is scheduled onto the Active or Reactive region, so as to continue its execution in the current time step. Calling resume on a process that suspends itself causes the process to continue to execute at the statement following the call to suspend.

 

The example below starts an arbitrary number of processes, as specified by the task argument N. Next, the task waits for the last process to start executing, and then waits for the first process to terminate. At that point the parent process forcibly terminates all forked processes that have not completed yet.

 

task do_n_way( int N );

process job[1:N];

 

for( int j = 1; j <= N; j++ )

fork

begin job[j] = process::self(); ... ; end

join_none

 

wait( job[N] != null );           // wait for last process to start

 

job[1].await();                         // wait for first process to finish

 

for( int k = 1; k <= N; k++ ) begin

if( job[k].status != process::FINISHED )

job[k].kill();

end

endtask