We know that the difference between blocking statements and non-blocking statements is: blocking statements executes sequentially (execution of next statement is blocked until present one completes) and are used to perform in combinational circuits.
Example:
always@(*) begin
c = A & B;
D = C/A;
end
Here all the statements executes sequentially (blocking statement)
Whereas non-blocking statements execute parallelly (execution of next statement is not blocked) and are used to perform in sequential circuits.
Example: Take an example of shift register
always@posedge(clk) begin
A<=B;
B<=C;
C<=D;
end
Here all the statements executes parallely because it is non-blocking and we have used posedge clk
Now if you see the difference between begin end
and fork join
, the difference is: in begin end
, statements are executed in the order they are listed (i.e. sequentially), whereas in fork join
, statements are executed parallelly.
My question here is, in the above example of non-blocking statement, we have used begin end
but the statements are been executed parallelly not sequentially, but if you see in the difference between begin end
and fork join
it says begin end
executes the statements one after another.
Could someone explain with a clear answer to this?
Serge is correct. You can think of the fork .. join as splitting each expression into its own action block. So if you recoded your eample:
as
A simulator would interpret the above as if it had been written as:
Which in this case would happen to behave exactly as if it were in the begin .. end pair.
As others have noted, a synthesis tool would typically reject the fork .. join construct, as while you are not using them here, the fork join provides additional features that are designed for efficient test bench coding. As an example, code execution stops at the join until each of the the forked blocks has completed execution.