VerilogHDL基础j教程之:程序基本结构

分享到:
361
下一篇 >

 Verilog HDL是一种用于数字逻辑电路设计的语言。用Verilog HDL描述的电路设计就是该电路的Verilog HDL模型。Verilog HDL既是一种行为描述的语言,也是一种结构描述的语言。也就是说,既可以用电路的功用**语言结构实现设计模块的外部性用**语言结构实现设计算法的模型。

  • RTL级(Register Transfer Level):描述数据在寄存器之间流动和如何处理这些数据的模型。
  • 门级(gate-level):描述逻辑门以及逻辑门之间的连接的模型。
  • 开关级(switch-level):描述器件中三极管和储存节点以及它们之间连接的模型。

       

      一个复杂电路系统的完整Verilog HDL模型是由若干个Verilog HDL模块构成的,每一个模块又可以由若干个子模块构成。其中有些模块需要综合成具体电路,而有些模块只是与用户所设计的模块交互的现存电路或激励信号程序结构。

    • 提供了可定义。
    • 提供了用于建立表达式的算术运算符、逻辑运算符、位运算符。
    • Verilog HDL语言作为一种结构化的语言也非常适合于门级和开关级的模型设计。因其结构化的特点又使它具有以下功;
    • 提供了双向通路和电阻器件的原语;
    • 可建立MOS器件的电荷分享和电荷衰减动态模型。

           

          Verilog HDL的构造性语句可以**地建立信号的模型。这是因为在Verilog HDL中,提供了延迟和输出强度的原语来建立**程度很高的信号模型。信号值可以有不同的强度,可以通过设定宽范围的模糊值来降低不确定条件的影响。

          Verilog HDL作为一种**的硬件描述编程语言,有着类似C语言的风格。其中if语句、case语句等和C语言中的对应语句十分相似。如果读者已经掌握C语言编程的基础,那么学习Verilog HDL并不困难,只要对Verilog HDL某些语句的特殊方面着重理解,并加强上机练习就加法器。

          module  adder ( count,sum,a,b,cin );   //加法器模块端口声明

                input [2:0] a,b;                         //端口说明

                input   cin;

                output  count;

                output [2:0] sum;

                assign {count,sum} = a + b + cin;      //加法器算法实现

          endmodule

           

          这个例子通过连续赋值语句描述了一个名为adder的三位加法器可以根据两个三比特数a、b和进位(cin)计算出和(sum)和进位(count)。从例子中可以看出整个Verilog HDL程序是嵌套在module和endmodule声明语句里的。

           

          例2:比较器。

          module  compare ( equal,a,b );     //比较器模块端口声明

                   output  equal;                 //输出信号equal

                   input [1:0] a,b;             //输入信号a、b

                   assign  equal=(a==b)?1:0; //如果a、b 两个输入信号相等,输出为1,否则为0

          endmodule

           

          这个程序通过连续赋值语句描述了一个名为compare的比较器。对两比特数a、b进行比较,如a与b相等,则输出equal为高电平,否则为低电平。在这个程序中,“/*........*/”和“//.........”表示注释部分,注释只是为了方便程序员理解程序,对编译是不起作用的。

           

          例3:使用原语的三态驱动器。

          module  trist2(out,in,enable);     //三态启动器模块端口声明

               output  out;                      //端口说明

               input   in, enable;

               bufif1  mybuf(out,in,enable);  //实例化宏模块bufif1

          endmodule

           

          这个例子描述了一个名为trist2的三态驱动器。程序通过调用一个在Verilog语言库中现存的三态驱动器实例元件bufif1来实现其功自行设计的三态驱动器。

          module  trist1(out,in,enable);     //三态启动器模块端口声明

               output  out;                       //端口说明

               input  in, enable;

               mytri  tri_inst(out,in,enable);//实例化由mytri模块定义的实例元件tri_inst

          endmodule

               module  mytri(out,in,enable); //三态启动器模块端口声明

               output  out;                            //端口说明

               input  in, enable;

               assign  out = enable? in : 'bz;    //三态启动器算法描述

          endmodule

           

          这个例子通过另一种方法描述了一个三态门。在这个例子中存在着两个模块。模块trist1调用由模块mytri定义的实例元件tri_inst。模块trist1是顶层模块。模块mytri则被称为子模块。

          通过上面的例子可以看到。

          • Verilog HDL程序是由模块构成的。每个模块的内容都是嵌在module和endmodule两个语句之间。每个模块实现特定的功

            Module     模块名(端口1,端口2,端口3,端口4, …);

             

            2.I/O说明

            I/O说明的格式如下:

             

            输入口:input  端口名1,端口名2,…,端口名i;   //(共有i个输入口)

            输出口:output 端口名1,端口名2,…,端口名j;       //(共有j个输出口)

             

            I/O说明也可以写在端口声明语句里,其格式如下:

             

            module  module_name(input port1,input port2,…,output port1,output port2… )

             

            3.内部信号声明

            在模块内用到的和与端口有关的wire和reg变量的声明,如下所示:

             

            reg [width-1 : 0] R变量1,R变量2 …;

            wire [width-1 : 0] W变量1,W变量2 …;

             

            4.功用“assign”声明语句。

            assign a = b & c;

            这种方法的句法很简单,只需写一个“assign”,后面再加一个方程式即可。例子中的方程式描述了一个有两个输入的与门。

             

            (2)用实例元件。

            and and_inst( q, a, b );

            采用实例元件的方法在电路图输入方式下,调入库元件。键入元件的名字和相连的引脚即可,表示在设计中用到一个跟与门(and)一样的名为and_inst的与门,其输入端为a、b,输出为q。要求每个实例元件的名字必须是惟一的,以避免与其他调用与门(and)的实例混淆。

             

            (3)用“always”块。

            always @(posedge clk or posedge clr) begin         //时钟上升沿触发,异步清零

                if(clr)  q <= 0;                               //清零

                else  if(en) q <= d;                           //使能赋值

                end

             

            采用“assign”语句是描述组合逻辑*常用的方法之一,而“always”块既可用于描述组合逻辑,也可描述时序逻辑。上面的例子用“always”块生成了一个带有异步**端的D触发器。

            “always”块可用很多种描述手段来表达逻辑,例如上例中就用了“if...else”语句来表达逻辑关系。如按一定的风格来编写“always”块,可以通过综合工具把源代码自动综合成用门级结构表示的组合或时序逻辑电路。需要注意的是,如果用Verilog模块实现一定的功能,首先应该清楚哪些是同时发生的,哪些是顺序发生的。

            上面3个例子分别采用了“assign”语句、实例元件和“always”块。这3个例子描述的逻辑功能是同时执行的。也就是说,如果把这3项写到一个Verilog模块文件中去,它们的次序不会影响逻辑实现的功能。这3项是同时执行的,也就是并发的。

            然而,在“always”模块内,逻辑是按照指定的顺序执行的。“always”块中的语句称为“顺序语句”,因为它们是顺序执行的。请注意,两个或更多的“always”模块也是同时执行的,但是模块内部的语句是顺序执行的。

            看一下“always”内的语句,就会明白它是如何实现功能的。“if…else… if”必须顺序执行,否则其功能就没有任何意义。如果else语句在if语句之前执行,功能就会不符合要求。为了能实现上述描述的功能,“always”模块内部的语句将按照书写的顺序执行。

             

             

        • 你可能感兴趣: 设计应用 三极管 器件 硬件
          无觅相关文章插件,快速提升流量