As we all know, there are two kinds of hardware description language (HDL) used for FPGA development: Verilog and VHDL. VHDL appeared earlier than Verilog. Verilog is widely used by major companies because of its simple grammar and the similarity of C language.

In fact, I learned VHDL language in college. Later, because the company was using Verilog, I relearned Verilog. Fortunately, WITH the FOUNDATION of C language, Verilog soon got started.

There are three versions of the Verilog standard document: VERILog-1995, VerILog-2001, and VerILog-2005, all of which are issued by IEEE. The latest version of the Verilog standard, version 2005, is simpler and more flexible than the previous two versions.

Although some of the official code, such as Xilinx IP core code, is still based on the Verilog-2001 standard in order to be compatible with previous comprehensive tools, I still strongly recommend that you use the latest Verilog-2005 standard.

Therefore, this paper is based on the IEEE-2005 grammar Standard, that is, IEEE P1364-2005/D3:Draft Standard for Verilog ® Hardware Description Language official Standard document. At the end of the article, see ieEE-2005 official standard document obtaining method.

Good code specification can improve the readability, reusability, and clarity of code, which is also a reflection of professional quality.

Our goal: Write Nowhere, Read Everywhere

What does it include?

  • named

    • The file name
    • Port name
    • Variable naming
    • Parameter named
  • structure

    • The overall structure
    • Port declaration
    • Whitespace and indentation make the code clearer
    • Parentheses increase readability
  • instantiated

  • annotation

  • other

  • Download the IEEE-2005 standard

named

Naming mainly includes file and module naming, port naming, variable naming, parameter naming and so on, in a nutshell: meaningful, short and easy to read, the important thing is not to use pinyin!

The file name

The file name must be the same as the module name. Only one module can be written into a file.

The name of the file must be meaningful, short and easy to read. Lowercase letters must be used in the file name and an underscore (_) must be used to separate the file name.

  • The underlying driver class module is named usingdrv_xxxType, such as:drv_led.v.drv_i2c.v.drv_ad7606.v
  • Serial port sends one byte:uart_tx_byte.v
  • Asynchronous and synchronous FIFO, and indicate depth and width:fifo_async_512_16.vandfifo_sync_256_64.v

The TestBench file is named as drv_led_tb.v, and the source file is named as drv_led_tb.v

V or top_project_name.v. For example, the top layer of the EEPROM read and write example project is named top_EEprom_demo.v

Port name

The name of a port is the same as that of a file. Use lowercase letters and use underscores to separate ports.

If the module is a top level module and connected to the actual FPGA pin, add _PAD_i, _PAD_o, _PAD_io to indicate the connection to the actual FPGA hardware pin.

Variable naming

  • Clock signals are used uniformlyclkIf it is a specific clock frequency, you can add the clock frequency after it, such asclk_50m
  • The reset signal is named RST. If it is active low, add _N after it, for examplerst_n
  • Flag bit name:flag_rise/flag_fall/flag_clr
  • Register beat signal name added_reg:reg rxd_reg
  • Add suffixes to shift register names_sreg:reg [3:0] busy_sreg

Some common abbreviations:

abbreviations Whole put together meaning
rst reset reset
clk clock The clock
rd read read
wr write write
addr address address
ack acknowledge The response

For more common port naming and variable naming abbreviations, see the following article:

Blog.csdn.net/heartdreamp…

Parameter named

The parameters in Verilog are similar to define in C language. There are mainly two types of localparam and parameter. The difference between the two is that the former cannot pass the parameters during the instantiation, while the latter can pass the parameters during the instantiation.

All other variables and file names are in lowercase, and only parameter definitions are in uppercase. When you need to define constants, you can specify a meaningful name through the parameter declaration. Such as:

parameter LED_ON     = 1'b0; parameter LED_OFF = ! LED_ON; parameter BAUD_RATE =115200;
parameter TIME_100MS = 25_000_000;
Copy the code

The state of a state machine is usually defined using localparam, and a meaningful name is specified. The Sn_NAME format is used uniformly, and the bitwidth is specified. Such as:

localparam S0_IDLE  = 4 'd0;
localparam S1_START = 4 'd1;
localparam S2_DOING = 4 'd2;
localparam S3_END   = 4 'd3;

localparam S_FINISH = 4 'd4;
localparam S_ERROR  = 'd5; //auto adaptiveCopy the code

structure

The overall structure

You are advised to write the Verilog module file in the following structure.

/* 1. File header: description of copyright information, file function, author, version history, etc. */
Port declaration: input/output/inout */
/* 3. Macro: 'define */
/* 4. Parameter definition: localparam/parameter */
/* 5. Register definition: reg */
/* 6. Wire network type definition: wire */
/* 7. Interconnect definition: assign */
/* 8. Timing logic description: always */
Copy the code

Example:

/* 1.file head */

/*********************************************************** Copyright 2021 'wechat:mcu149'. All rights reserved. FileName: drv_led.v Function: led driver Author : mcu149 SVN : SVN Revision: 1490 SVN Date: 2021-06-05 19:49:49 Revision: 2020-09-09: Rev 1.0 2020-10-01: Rev 1.1 * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * /

/* 2.input/output/inout */
module drv_led(
    //Inputs
    input rst_n,
    input clk_50m,
    input en,
    
    //Outputs
    output led1,
    output reg led2 = 0      //define initial value is 0
);

/* 3.define */

/* 4.parameter/localparam */
parameter TIME_500MS = 32 'd25_000_000;

/* 5.reg */
reg [31:0] cnt = 0;

/* 6.wire */
wire flag_toggle = (cnt == TIME_500MS);
    
/* 7.assign */
assign led1 = (en == 0)?0 : cnt[20];

/* 8.always block */
always @ (posedge clk_50m) begin
    if(! rst_n)
        cnt <= 0;
    else if(flag_toggle)
        cnt <= 0;
    else 
        cnt <= cnt + 1;
end

always @ (posedge clk_50m) begin
    if(! rst_n)
        led2 <= 0;
    else if(! en) led2 <=0;
    else if(en) begin
        if(flag_toggle) led2 <= ! led2; end end endmoduleCopy the code

Port declaration

The input ports are put together, the output ports are put together, the bidirectional ports are put together. A new feature of Verilog-2005 is that if an output signal requires an initial value, it can be specified directly at the port definition.

This point some friends may be divided by function, such as connected to the same chip together.

The advantage of the input-output separation is that it is easy to distinguish between inputs and outputs when instantiating.

Whitespace and indentation make the code clearer

  • Add a space at either end of the operator to make the program structure clearer and more readable
  • The indentation style is KR. That is, begin is at the end of a line and end is on a single line
  • Use four Spaces for indentation instead of TAB
  • If there is only one line in an if/else statement, begin-end can be omitted
  • Add blank lines to separate blocks. Separate always blocks by newlines

The following are two code writing specifications, reasonable indentation, reasonable white space greatly increases readability.

Parentheses increase readability

In school, some tests give anti-human questions to test students’ mastery of the priority of various operators.

When working on a real project, unlike an exam, the goal is readability and ease of use, so when using multiple operators, don’t be stingy with parentheses to indicate the priority of the operation in order to increase readability and avoid ambiguity.

instantiated

Instantiation can be regarded as the soul of FPGA development. The process of instantiation is actually the process of calling hardware modules. For example, we use Verilog to describe a 3-8 decoder module, which can be used (instantiated) in different places, and named UT0 / UT1 / UT2, etc., and can specify parameters when instantiating. For example, the serial port sending and receiving module, by specifying different parameters to achieve different baud rate compatibility.

  • The order of the instantiation and port declaration is the same, putting the input ports together and the output ports together
  • Multibit signals, which need to specify bit width when instantiated to increase readability
  • The top-level module only performs module instantiation and does not write any control statements

Example:

wire [7:0] rx_data;
wire rx_done;
wire rx_err;

/* The serial port receives 1 byte */
uart_rx_byte #(
    .BAUD_RATE(32 'd115200),
    .EN_PARITY(2 'd0),  //0: no check bit, 1: odd check, 2: even check
    .EN_STOP_2(1'b0)   //1: enables two stop bits
)uart_rx_byte_0(
    // Inputs
    .clk(clk_32m),
    .rst_n(rst_n),
    .rxd(uart_rxd),

    // Outputs
    .data_rx(rx_data[7:0]),	// Specify the bit width
    .done(rx_done),
    .err(rx_err)
);
Copy the code

annotation

Don’t believe that code is the best comment! I don’t deny that some people write code that is very standard, well named, and clearly formatted.

But I don’t think you’ve gotten to the point where everyone can read uncommented code. Annotation is not only for others to see, but also for yourself to see, a good memory is better than a bad writing.

/**/ comments are used in the same way, or mixed with //, depending on personal habits!

  • After each variable is defined, you need to annotate the function of the variable
  • Each always block feature requires comments
  • State machine State Meaning requires comments
  • Comments need to be added after conditional statements
  • As the code changes, so do the comments

other

  • When used properly, Generate for can be used to define and instantiate modules in batches, reducing the amount of code and improving readability
  • Using tasks and functions in Testbench can improve efficiency
  • The shift operation is replaced by the concatenation complement zero operation, which is easier to read
  • Sequential logic uniformly uses non-blocking assignments, i.e< =symbol
  • A line contains no more than 80 characters. If the line length is too long, it is handled by wrapping the line
  • There is a top-level design, and then a molecular module function, or a bottom-up approach
  • Modular design, improve the degree of module reuse
  • Try not to use if/else if you can use case
  • Variable of type reg, depending on the need to see whether latched

Eiee-2005 Verilog standard download

【Verilog standard 】 To obtain ieEE_Verilog_1364_2005.pdf standard documents.

For a negative example of the Verilog code specification, see: How do you write Verilog code that your colleagues cannot maintain?