SystemVerilog for循环中的fork join_none「建议收藏」

SystemVerilog for循环中的fork join_none「建议收藏」moduletest;initialbeginmain();endtaskmain();//forkjoin_anyblock1fork#5$display(“THREAD1%0t”,$time);#25$display(“THREAD2%0t”,$ti…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

 

module test;

   initial begin
      main();
   end

   task main();
      // fork join_any block1
      fork
         #5 $display("THREAD 1 %0t", $time);
         #25 $display("THREAD 2 %0t", $time);
      join_any
      dev_state();
      $display("After task call %0t", $time);
      #100 $finish;
   endtask

   task dev_state();
      // fork join_any block2
      fork
         #5 $display("THREAD 3 %0t", $time);
         #10 $display("THREAD 4 %0t", $time);
      join_any
      disable fork;
   endtask

   initial begin
      $display("debug point fork_1");
      fork_1();
      $display("debug point fork_2");
      fork_2();
      $display("debug point fork_3");
      fork_3();
      $display("debug point fork_4");
      fork_4();      
   end
   
   task fork_1();
      for(int i = 0; i < 16; i++) begin
         fork 
            // begin
            automatic int index =i;       
            send(index);
            // end 
         join_none 
      end
      wait fork;
   endtask

   task fork_2();
      for(int i = 0; i < 16; i++) begin
         fork 
            // begin
            automatic int index;
            index = i;          
            send(index);
            // end 
         join_none         
      end
      wait fork;
   endtask

   task fork_3();
      for(int i = 0; i < 16; i++) begin
         fork 
            begin
               automatic int index;
               index = i;         
               send(index);
            end 
         join_none 
      end
      wait fork;      
   endtask

   task fork_4();
      for(int i = 0; i < 16; i++) begin
         automatic int index;
         index = i;         
         fork 
            begin
               send(index);
            end 
         join_none 
      end
      wait fork;      
   endtask

   task send(int j);
      $display("driving port %0d" , j);
   endtask


endmodule
all : clean comp run sim.log 

clean:
	\rm -rf simv* csrc *.key
comp:
	vcs -full64 test.sv -sverilog
run:
	simv -l sim.log
debug point fork_1
driving port 0
driving port 1
driving port 2
driving port 3
driving port 4
driving port 5
driving port 6
driving port 7
driving port 8
driving port 9
driving port 10
driving port 11
driving port 12
driving port 13
driving port 14
driving port 15
debug point fork_2
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
debug point fork_3
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
driving port 16
debug point fork_4
driving port 0
driving port 1
driving port 2
driving port 3
driving port 4
driving port 5
driving port 6
driving port 7
driving port 8
driving port 9
driving port 10
driving port 11
driving port 12
driving port 13
driving port 14
driving port 15
THREAD 1 5
THREAD 3 10
After task call 10

 

使用以上Makefile和test.sv文件,运行仿真,仿真结果如下

可以看到虽然都是fork join_none wait fork但是每一个fork_x task的打印结果是不同的。这是为什么呢?

不同之处在于变量index的生存周期,以及何时为该变量分配变量i的值的时间。

意识到将有16个并发变量名为index,只有一个名为i。在情况1)和2)中,每次进入fork / join_none块时都会创建index变量。在派生fork / join_none中的任何进程之前发生。在情况1)中,变量初始化也发生在fork / join_none中的任何进程之前。您需要记住的是,自动变量是在输入时创建的,并在执行它们所在的块中的任何过程语句之前被初始化。因此,在情况1)中,每个索引变量在每次循环迭代中都获得i的当前值。
在情况2)中,您将初始化移到了单独的过程分配语句中。 fork / join_none中的每个语句将成为新的子进程,并且直到当前父线程挂起后,该子进程才开始执行。现在,for循环会生成16个线程,然后在i的值为16时在wait fork处挂起。(正如我之前说过的,如果send(index)看到单位值0或分配的值16,这是一个竞赛。
在情况3)中,现在在一个begin / end块内声明index变量,这是fork / join_none的单个语句。因此,直到所有16个进程都已生成并且i的值为16时,才创建索引变量。

任何自动变量的生命周期都将在其块及其所有嵌套块的生命周期结束时结束。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/193010.html原文链接:https://javaforall.net

(0)
上一篇 2025年12月3日 下午9:22
下一篇 2025年12月3日 下午10:01


相关推荐

  • originpro 2021 附安装教程

    originpro 2021 附安装教程近日新推出了origin系列的最新版本:origin2021,是一款非常实用的科学绘图与数据分析软件,并且该版本可以和2018——2021版本共享设置,若你拥有这些版本中的任何一个,则只需安装并运行新版本即可。不仅如此,它为了带给用户最佳的使用体验,进行了全方面的新增和优化,现如今能够使用新的颜色管理器创建自己的颜色列表或调色板,其中包括通过颜色选择和颜色插值,还在工作表上添加了新的公式栏,轻松编辑复杂的公式,具有调整公式栏字体大小的选项,以便于阅读,而且Origin中的嵌入式Python环境也得到了极大的

    2022年6月11日
    1.9K
  • FAST角点检测算法

    FAST角点检测算法FAST 是一种角点检测算法 其思想源于 corner 的定义 也就是检测候选点周围像素点的像素值 如果候选点周围邻域内有足够多的像素点与该候选点的灰度值差别够大 则认为该候选点为一个特征点 nbsp nbsp nbsp nbsp FAST 检测角点的过程 nbsp nbsp nbsp nbsp 1 初步筛选 如下图所示 将像素点 P1 P5 P9 P13 的像素值与中心像素点 P 的像素值进行比较 如果至少有三个像素点的像素值都大于 Ip nbsp t

    2026年3月26日
    2
  • 计算机网络协议(四)——HTTP、HTTPS、P2P协议

    计算机网络协议(四)——HTTP、HTTPS、P2P协议底层网络知识详解 最常用的应用层概述一 HTTP 协议 1 1 HTTP1 11 2 HTTP2 01 3QUIC 协议概述这个专栏的计算机网络协议 我是在极客时间上学习已经有三万多人购买的刘超老师的趣谈网络协议专栏 讲的特别好 像看小说一样学习到了平时很枯燥的知识点 计算机网络的书籍太枯燥 感兴趣的同学可以去付费购买 绝对物超所值 本文就是对自己学习专栏的总结 评论区可以留下你的问题

    2026年3月26日
    2
  • 前端开发技术(vscode怎么下载)

    前言   在前端开发中,有一个非常好用的工具,VisualStudioCode,简称VScode。   都不用我安利VScode,大家就会乖乖的去用,无数个大言不惭的攻城狮,都被VScode比德芙还丝滑的强大功能所折服。   我是来给大家安利插件的,想做个比较全面的插件集合给大家。网上的我也看过一些,但是都比较零散,时间也久了一些,我结合最近的情况,总结一下,造福大家,才是我…

    2022年4月10日
    57
  • 终于,我感受到了IDEA的强大[通俗易懂]

    Java开发者千千万,开发者用的开发工具目前主流却只有2种:eclipse和IDEA,我入行以来一直用的eclipse,听过IDEA很好很强大,但是也只是处于听说的阶段,基本没用过,自然没怎么体会过。

    2022年2月16日
    47
  • java 调试js_JavaScript 调试

    java 调试js_JavaScript 调试原标题 JavaScript 调试在编写 Java 时 如果没有调试工具将是一件很痛苦的事情 Java 调试没有调试工具是很难去编写 Java 程序的 你的代码可能包含语法错误 逻辑错误 如果没有调试工具 这些错误比较难于发现 通常 如果 Java 出现错误 是不会有提示信息 这样你就无法找到代码错误的位置 通常 你在编写一个新的 Java 代码过程中都会发生错误 Java 调试工具在程序代码中

    2026年3月17日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号