#### FPGA Development for Radar, Radio-Astronomy and Communications MASTERS COURSE





Dept. Electrical Engineering, University of Cape Town Private Bag, Rondebosch, 7701, South Africa http://www.rrsg.uct.ac.za



Presented by John-Philip Taylor Convened by Dr Stephen Paine

Day 2 - 28 April 2022

Outline 1 of 33

**IP Library** 

Verilog Processes

Finite State Machines

**Timing Constraints** 

**JTAG** 





#### **Outline**

#### **IP Library**

Verilog Processes

Finite State Machines

**Timing Constraints** 

**JTAG** 





#### **IP Library**

- A combination of soft and hard IP
- Collectively known as "Megafunctions" or "IP Cores"







#### **IP Library**

- A combination of soft and hard IP
- Collectively known as "Megafunctions" or "IP Cores"







Wizard 3 of 33

 Generally easier to use the IPexpress wizard to generate wrapper modules

- Or one can instantiate the built-in modules directly
- ► Practical 06 Data Stream uses IPexpress to set up a EBR memory block





Wizard 3 of 33

 Generally easier to use the IPexpress wizard to generate wrapper modules

- ► Or one can instantiate the built-in modules directly
- ► Practical 06 Data Stream uses IPexpress to set up a EBR memory block





Wizard 3 of 33

 Generally easier to use the IPexpress wizard to generate wrapper modules

- ► Or one can instantiate the built-in modules directly
- ► Practical 06 Data Stream uses IPexpress to set up a EBR memory block





- ► RAM / ROM
- ▶ DSP blocks
- ▶ PLL / DLL blocks
- Processors / SoC (ARM) with bus infrastructure
- ► Interfaces (DDR Memory / PCIe / SerDes (JESD204) / etc.)







- ► RAM / ROM
- ▶ DSP blocks
- ▶ PLL / DLL blocks
- Processors / SoC (ARM) with bus infrastructure
- ► Interfaces (DDR Memory / PCIe / SerDes (JESD204) / etc.)







- ► RAM / ROM
- ▶ DSP blocks
- ► PLL / DLL blocks
- Processors / SoC (ARM) with bus infrastructure
- ► Interfaces (DDR Memory / PCIe / SerDes (JESD204) / etc.)







- ► RAM / ROM
- ▶ DSP blocks
- ► PLL / DLL blocks
- Processors / SoC (ARM) with bus infrastructure
- ► Interfaces (DDR Memory / PCIe / SerDes (JESD204) / etc.)







- ► RAM / ROM
- ▶ DSP blocks
- ► PLL / DLL blocks
- Processors / SoC (ARM) with bus infrastructure
- ► Interfaces (DDR Memory / PCIe / SerDes (JESD204) / etc.)







Simulation 5 of 33

- Modelsim can simulate IP modules
- ► Add the xp2 library in the "Start Simulation" dialogue box
- Also remember to compile the IP block
- Modelsim cannot understand the defparam style of parameters.





- ► Modelsim can simulate IP modules
- ► Add the xp2 library in the "Start Simulation" dialogue box
- ► Also remember to compile the IP block

► Modelsim cannot understand the defparam style of

parameters.







- ▶ Modelsim can simulate IP modules
- ► Add the xp2 library in the "Start Simulation" dialogue box
- ► Also remember to compile the IP block

► Modelsim cannot understand the defparam style of

parameters.







**5** of 33

- ▶ Modelsim can simulate IP modules
- ► Add the xp2 library in the "Start Simulation" dialogue box
- ► Also remember to compile the IP block
- ► Modelsim cannot understand the defparam style of parameters.

```
defparam Inst.Param1 = 8'h12;
defparam Inst.Param2 = 8'h23;
Mod Inst(
    // Port assignments
);
```





- ▶ Modelsim can simulate IP modules
- ► Add the xp2 library in the "Start Simulation" dialogue box
- ► Also remember to compile the IP block
- ► Modelsim cannot understand the defparam style of parameters.

```
Mod #(
   .Param1(8'h12),
   .Param2(8'h23)
) Inst(
   // Port assignments
);
```





#### **Outline**

IP Library

#### Verilog Processes

Finite State Machines

**Timing Constraints** 

**JTAG** 





```
reg A, B;
always @(posedge ipClk) begin
  A = C & B;
  B = A | C;
end
```



- Statements are evaluated in order, like a computer program
- ► Often results in unintentionally long combinational chains
- ▶ Note that all registers still change state on the clock edge





```
reg A, B;
always @(posedge ipClk) begin
  A = C & B;
  B = A | C;
end
```



- ► Statements are evaluated in order, like a computer program
- ► Often results in unintentionally long combinational chains
- ▶ Note that all registers still change state on the clock edge





```
reg A, B;
always @(posedge ipClk) begin
  A = C & B;
  B = A | C;
end
```



- ► Statements are evaluated in order, like a computer program
- ► Often results in unintentionally long combinational chains
- ► Note that all registers still change state on the clock edge





```
reg A, B;
always @(posedge ipClk) begin
  A = C & B;
  B = A | C;
end
```



- Statements are evaluated in order, like a computer program
- ► Often results in unintentionally long combinational chains
- ▶ Note that all registers still change state on the clock edge





```
reg A, B;
always @(posedge ipClk) begin
  A <= C & B;
  B <= A | C;
end</pre>
```



- All right-hand-side expressions are evaluated in parallel
- and then assigned to the left-hand-side on the clock edge
- ► The order of statements makes no difference to the functionality





```
reg A, B;
always @(posedge ipClk) begin
  A <= C & B;
  B <= A | C;
end</pre>
```



- ► All right-hand-side expressions are evaluated in parallel
- and then assigned to the left-hand-side on the clock edge
- ► The order of statements makes no difference to the functionality





```
reg A, B;
always @(posedge ipClk) begin
  A <= C & B;
  B <= A | C;
end</pre>
```



- ► All right-hand-side expressions are evaluated in parallel
- and then assigned to the left-hand-side on the clock edge
- ► The order of statements makes no difference to the functionality





```
reg A, B;
always @(posedge ipClk) begin
  A <= C & B;
  B <= A | C;
end</pre>
```



- ► All right-hand-side expressions are evaluated in parallel
- and then assigned to the left-hand-side on the clock edge
- ► The order of statements makes no difference to the functionality





# Never mix blocking and non-blocking statements in the same always block

Except inside test-benches, where it is sometimes useful...





# Never mix blocking and non-blocking statements in the same always block

Except inside test-benches, where it is sometimes useful...





- ▶ Use blocking assignments: allows algorithmic descriptions
- ► If not explicitly assigned a new value, the previous value is "remembered" in an "inferred latch" – to be avoided





- ► Use blocking assignments: allows algorithmic descriptions
- ► If not explicitly assigned a new value, the previous value is "remembered" in an "inferred latch" – to be avoided





#### **Look-up Tables**

```
always @(*) begin
 case (BCD)
    4 'h0:
             SevenSegment = 7'b0111111;
   4'h1:
             SevenSegment = 7'b0000110;
   4 h2:
             SevenSegment = 7'b1011011;
   4'h3:
             SevenSegment = 7'b1001111;
   4'h4:
             SevenSegment = 7'b1100110;
   4 h5:
             SevenSegment = 7'b1101101;
   4'h6:
             SevenSegment = 7'b11111101;
   4'h7:
             SevenSegment = 7'b0000111;
   4'h8:
             SevenSegment = 7'b1111111;
   4'h9:
             SevenSegment = 7'b1101111;
   default:; // This is bad: infers a latch
 endcase
end
```





#### **Look-up Tables**

```
always @(*) begin
  case (BCD)
    4 'h0:
             SevenSegment = 7'b0111111;
    4'h1:
             SevenSegment = 7'b0000110;
    4 h2:
             SevenSegment = 7'b1011011;
    4'h3:
             SevenSegment = 7'b1001111;
    4'h4:
             SevenSegment = 7'b1100110;
    4 h5:
             SevenSegment = 7'b1101101;
    4'h6:
             SevenSegment = 7'b1111101;
    4 h7:
             SevenSegment = 7'b0000111;
    4'h8:
             SevenSegment = 7'b1111111;
    4'h9:
             SevenSegment = 7'b1101111;
    default: SevenSegment = 0; // This is acceptable
  endcase
end
```





#### **Look-up Tables**

```
always @(*) begin
  case (BCD)
    4 'h0:
             SevenSegment = 7'b0111111;
    4'h1:
             SevenSegment = 7'b0000110;
    4 h2:
             SevenSegment = 7'b1011011;
    4'h3:
             SevenSegment = 7'b1001111;
    4'h4:
             SevenSegment = 7'b1100110;
    4 h5:
             SevenSegment = 7'b1101101;
    4'h6:
             SevenSegment = 7'b1111101;
    4 h7:
             SevenSegment = 7'b0000111;
    4'h8:
             SevenSegment = 7'b1111111;
    4'h9:
             SevenSegment = 7'b1101111;
    default: SevenSegment = 7'bXXXXXXX; // This is better
  endcase
end
```





## **Sparse Case Statements**

```
always @(*) begin
 case (Address) // 8-bit address, 16-bit data
   8'h00: Data = FirmwareVersion;
   8'h01: Data = BuildDate;
   8'h02: Data = BuildTime;
   8'h10: Data = { 6'd0, LED };
   8'h11: Data = { 6'd0, Switches};
   8'h12: Data = {14'd0, Buttons };
   8'h20: Data = Accelerometer_X;
   8'h21: Data = Accelerometer Y:
   8'h22: Data = Accelerometer_Z;
   default: Data = 0; // This is OK
 endcase
end
```





## **Sparse Case Statements**

```
always @(*) begin
 case (Address) // 8-bit address, 16-bit data
   8'h00: Data = FirmwareVersion;
   8'h01: Data = BuildDate;
   8'h02: Data = BuildTime;
   8'h10: Data = { 6'd0, LED };
   8'h11: Data = { 6'd0, Switches};
   8'h12: Data = {14'd0, Buttons };
   8'h20: Data = Accelerometer_X;
   8'h21: Data = Accelerometer Y:
   8'h22: Data = Accelerometer_Z;
   default: Data = 16'hXXXX; // This is better
 endcase
end
```





### **Register Transfer Logic**







## **Register Transfer Logic**

Use non-blocking assignments: easier to relate all calculations to the clock edge

```
reg Reset;
always @ (posedge ipClk) begin
  Reset <= ipReset; // Localise the reset

if (Reset) begin
  // Reset stuff here

end else if (ipEnabled) begin
  // RTL code goes here
end
end</pre>
```





## **Register Transfer Logic**

► If not explicitly assigned a new value, the previous value is "remembered" in the register – very useful

```
req Reset;
reg [11:0]Count;
always @(posedge ipClk) begin
  Reset <= ipReset; // Localise the reset
  if(Reset) begin
    Count. \leq 0:
  end else if(ipEnabled) begin
   Count <= Count + 1'b1;
  end
end
```





# **Synchronous and Local Resets**

14 of 33







# **Synchronous and Local Resets**

14 of 33







### **Outline**

**IP** Library

Verilog Processes

Finite State Machines

**Timing Constraints** 

**JTAG** 





#### **Finite State Machines**







#### **Named States**

#### ► Example using Gray code:





#### **Named States**

► Example using one-hot encoding:





► Or let the compiler select the encoding:

```
typedef enum{ // SystemVerilog only
  Off, On,
  EspressoReady, SteamerReady,
  MakeEspresso, MakeSteam
} tState;
tState State;
```





16 of 33

# **FSM Template**

```
always @(posedge ipClk) begin
  if(Reset) begin
    State <= Off;
  end else begin
    case (State)
      Off:
                     begin ... end
      On:
                     begin ... end
      EspressoReady: begin ... end
      SteamerReady:
                     begin ... end
      MakeEspresso:
                     begin ... end
      MakeSteam:
                     begin ... end
      default:;
    endcase
  end
end
```





### **FSM Example**

```
case (State)
  Off: begin
    if(Switch On) State <= On;</pre>
  end
  On: begin
          if(Switch_Off) State <= Off;</pre>
    else if(Heat_Up ) State <= EspressoReady;</pre>
  end
  EspressoReady: begin
          if(Switch_Off ) State <= Off;</pre>
    else if(Steam_On ) State <= SteamerReady;</pre>
    else if(Espresso_On) State <= MakeEspresso;</pre>
  end
```







- ► For the ADXL345 Digital Accelerometer:
  - ► Maximum SClk frequency is 5 MHz (200 ns period)
  - ► SDI sample and hold is 5 ns (sampled on rising edge)
  - ► SClk falling edge to SDO delay is 40 ns





#### The Abstraction

```
module ADXI345 #(
 parameter Clock_Div = 5 // 5 MHz SClk on a 50 MHz Clk
  input ipClk, ipReset,
  output reg [15:0]X, // 2's Complement
  output reg [15:0]Y,
  output reg [15:0]Z,
  // Physical device interface
  output reg nCS, SClk, SDI,
  input SDO
);
endmodule
```





#### **General Structure**

```
req Reset;
reg [3:0]Clock_Count;
wire Clock Enable = (Clock Count == Clock Div);
always @(posedge ipClk) begin
 Reset <= ipReset;
  if(Clock_Enable) Clock_Count <= 4'd1;</pre>
  else
                 Clock Count <= Clock Count + 1'b1;
  if(Reset) begin
    // Reset the machine here
  end else if(Clock_Enable) begin
   // State machine goes here
  end
end
```









```
if(Reset) begin
 nCS <= 1'b1;
 SClk <= 1'b1;
 SDI <= 1'b1;
 State <= Setup;
end else if(Clock_Enable) begin
 case (State)
    Setup: begin
      // SPI 4-wire; Full-res; Right-justify; 4g Range
     WriteData <= {2'b00, 6'h31, 8'b0000_1001};
     Count <= 5'd16;
     State <= Transaction;
     RetState <= ReadX;</pre>
   end
```





```
ReadX: begin
  Z <= {ReadData[7:0], ReadData[15:8]};</pre>
  WriteData <= {2'b11, 6'h32, 8'd0};
  Count <= 5'd24;
  State <= Transaction;</pre>
 RetState <= ReadY;</pre>
end
ReadY: begin
  X <= {ReadData[7:0], ReadData[15:8]};</pre>
  WriteData <= {2'b11, 6'h34, 8'd0};
  Count <= 5'd24;
  State <= Transaction;</pre>
  RetState <= ReadZ;</pre>
end
```









### **Reading Data**

```
Transaction: begin
  if(nCS) begin
    nCS <= 1'b0;
  end else begin
    if(SClk) begin
      if(Count == 0) begin
        nCS <= 1'b1; State <= RetState;</pre>
      end else begin
        SClk <= 1'b0;
        {SDI, WriteData[15:1]} <= WriteData;
      end
      Count <= Count - 1'b1;
      ReadData <= {ReadData[14:0], SDO};</pre>
    end else begin
      SClk <= 1'b1;
end end end
```











### **Outline**

**IP** Library

Verilog Processes

Finite State Machines

**Timing Constraints** 

**JTAG** 











- ▶ De facto industry standard
- ► TCL based, so one can use TCL scripting within the SDC file
- Only specify what Quartus does not already know:
  - Clock frequencies
  - Asynchronous paths
  - ▶ PGB trace delays
  - External device parameters
  - Multi-cycle paths
  - etc.





26 of 33

- De facto industry standard
- ► TCL based, so one can use TCL scripting within the SDC file
- Only specify what Quartus does not already know:
  - Clock frequencies
  - Asvnchronous paths
  - PCB trace delays
  - External device parameters
  - Multi-cycle paths
  - etc





- ► De facto industry standard
- ► TCL based, so one can use TCL scripting within the SDC file
- ► Only specify what Quartus does not already know:
  - ▶ Clock frequencies
  - Asynchronous paths
  - ▶ PCB trace delays
  - External device parameters
  - Multi-cycle paths
  - etc.





- De facto industry standard
- ► TCL based, so one can use TCL scripting within the SDC file
- ► Only specify what Quartus does not already know:
  - ► Clock frequencies
  - Asynchronous paths
  - ▶ PCB trace delays
  - External device parameters
  - Multi-cycle paths
  - etc.





- ► De facto industry standard
- ► TCL based, so one can use TCL scripting within the SDC file
- ► Only specify what Quartus does not already know:
  - ► Clock frequencies
  - Asynchronous paths
  - ▶ PCB trace delays
  - External device parameters
  - Multi-cycle paths
  - etc.





- De facto industry standard
- ► TCL based, so one can use TCL scripting within the SDC file
- ► Only specify what Quartus does not already know:
  - ► Clock frequencies
  - Asynchronous paths
  - ► PCB trace delays
  - External device parameters
  - Multi-cycle paths
  - etc.





- ► De facto industry standard
- ► TCL based, so one can use TCL scripting within the SDC file
- ► Only specify what Quartus does not already know:
  - ► Clock frequencies
  - Asynchronous paths
  - PCB trace delays
  - External device parameters
  - Multi-cycle paths
  - etc.





26 of 33

- De facto industry standard
- ► TCL based, so one can use TCL scripting within the SDC file
- ► Only specify what Quartus does not already know:
  - ► Clock frequencies
  - Asynchronous paths
  - ► PCB trace delays
  - External device parameters
  - Multi-cycle paths
  - etc.





- ► De facto industry standard
- ► TCL based, so one can use TCL scripting within the SDC file
- ► Only specify what Quartus does not already know:
  - ► Clock frequencies
  - Asynchronous paths
  - ► PCB trace delays
  - External device parameters
  - Multi-cycle paths
  - etc.





# **Asynchronous Pins**

- ▶ Pins such as LEDs, buttons, RS-232 signals, etc. does not belong to a clock domain
- ► The compiler must not try to meet timing on these:





# **Asynchronous Pins**

- ▶ Pins such as LEDs, buttons, RS-232 signals, etc. does not belong to a clock domain
- ► The compiler must not try to meet timing on these:





## **Clock Specification**

```
create_clock -period 100 [get_ports ADC_Clk]
create_clock -period 20 [get_ports Clk1]
create_clock -period 20 [get_ports Clk2]

derive_pll_clocks
derive_clock_uncertainty
```





# **Clock Groups**

- ► Unless specified otherwise, the compiler assumes that all clocks are related and in the same clock domain
- When clocks are unrelated, all paths between them must be marked as "false paths"
- ► Do this with clock groups:

```
set_clock_groups -logically_exclusive \
  -group [get_clocks ADC_Clk] \
  -group [get_clocks Clk1] \
  -group [get_clocks Clk2] \
  -group [get_clocks {SRAM_CLK *altpll_0*}]
```





## **Clock Groups**

- ► Unless specified otherwise, the compiler assumes that all clocks are related and in the same clock domain
- ► When clocks are unrelated, all paths between them must be marked as "false paths"
- ► Do this with clock groups:

```
set_clock_groups -logically_exclusive \
  -group [get_clocks ADC_Clk] \
  -group [get_clocks Clk1] \
  -group [get_clocks Clk2] \
  -group [get_clocks {SRAM_CLK *altpll_0*}]
```





## **Clock Groups**

- ► Unless specified otherwise, the compiler assumes that all clocks are related and in the same clock domain
- ► When clocks are unrelated, all paths between them must be marked as "false paths"
- ► Do this with clock groups:

```
set_clock_groups -logically_exclusive \
  -group [get_clocks ADC_Clk] \
  -group [get_clocks Clk1] \
  -group [get_clocks Clk2] \
  -group [get_clocks {SRAM_CLK *altpll_0*}]
```





#### **Outline**

**IP** Library

Verilog Processes

Finite State Machines

**Timing Constraints** 

**JTAG** 







- ▶ Connects to the PC over USB / Ethernet / etc.
- ► Connected devices form a long chain of shift-registers







- Connects to the PC over USB / Ethernet / etc.
- Connected devices form a long chain of shift-registers







- ► Connects to the PC over USB / Ethernet / etc.
- Connected devices form a long chain of shift-registers







- ► Connects to the PC over USB / Ethernet / etc.
- ► Connected devices form a long chain of shift-registers





















































# **JTAG Debugging**

FPGA IDEs includes powerful JTAG-based debugging tools. Practical 04 - JTAG Debugging introduces the Lattice Diamond Reveal Analyzer.

| Bus/Signal                 | Data       | 3710<br>0:896 | 3774<br>0:960 | 3838<br>0:1024 | 3902<br>0:1088 | 3988<br>0:1152 | 4030<br>0:1216 | 4094<br>0:1280 | 0:1344 | 0:1908 | 0:1472 |
|----------------------------|------------|---------------|---------------|----------------|----------------|----------------|----------------|----------------|--------|--------|--------|
| € stimstate                | pk_ready   |               | startpk       | ХX             | ·              |                |                | pk_finish      | _end   |        |        |
| start                      | 1          |               |               |                |                |                |                |                |        |        |        |
| ⊞ hand_type                | zero 💌     |               |               | zero           |                |                | Х              |                | 01     | e      |        |
| pb_done                    | 0          |               |               |                |                |                |                |                |        |        |        |
| sort_done                  | 0          |               |               |                |                |                | 1              |                |        |        |        |
| stimcount                  | 1 "        |               |               | N.             |                |                |                |                |        |        |        |
| ⊕ gmstate                  | wait_cards | Wo            | it_hand2      |                | X              |                |                |                |        |        |        |
| chstate                    | idle       | sort          | XX            | , idle         | XXXX           | sort           | - A            | X              |        | idle   |        |
| uut/gamesm/nxstate         | 0101       |               | 0100          | X 010          | CX.            |                |                | 1111           | 1      |        |        |
| uut/ch/sort/sort_loop_done | 1 ~        | CKORORORO     | Х             | 1              | 25             | KK NOKOKOKO    | 0              |                | 1      |        |        |
| uut/gamesm/bet_amount_int  |            |               |               |                |                |                |                |                |        |        |        |
| game_ready                 | 1          |               |               |                |                |                |                |                |        |        |        |
| bet                        | 0          |               |               |                |                |                |                |                |        |        |        |
| deal                       | 0          |               |               |                |                |                |                |                |        |        |        |
| - finish                   | 0          |               |               |                | T              |                |                |                |        |        |        |
| discard                    | 0          |               |               | 1              | 1              |                |                |                |        |        |        |
| uut/ch/bj_mux_done         | 0          |               |               |                |                |                |                |                |        |        |        |
| uut/ch/phm_ready           | 1          |               |               |                |                |                |                |                |        |        |        |
| uut/ch/dhm_ready           | 1          |               |               |                |                |                |                |                |        |        |        |
| uut/hand_ready             | 1          |               |               | nn             |                |                |                |                |        |        |        |
| ()                         |            |               |               |                |                | 11.0           |                |                |        |        |        |





#### **Select References**

- Stephen Brown and Zvonko Vranesic Fundamentals of Digital Logic with Verilog Design, 2<sup>nd</sup> Edition ISBN 978-0-07-721164-6
- Merrill L Skolnik Introduction to RADAR Systems ISBN 978-0-07-288138-7
- Mark A. Richards and James A. Scheer Principles of Modern Radar: Basic Principles ISBN 978-1-89-112152-4
- Deepak Kumar Tala
  World of ASIC
  http://www.asic-world.com/
- Jean P. Nicolle
  FPGA 4 Fun
  http://www.fpga4fun.com/





#### FPGA Development for Radar, Radio-Astronomy and Communications MASTERS COURSE





Dept. Electrical Engineering, University of Cape Town Private Bag, Rondebosch, 7701, South Africa http://www.rrsg.uct.ac.za



Presented by John-Philip Taylor Convened by Dr Stephen Paine

Day 2 - 28 April 2022