# Introduction

by Erol Seke

For the course "FPGA Structures for Digital Comm."



### **Electronic Communication**



### **Digital Communication**

### Advantages :

- Mathematical/Logical Processing on the data is possible
- Therefore : higher protection against noise
- More flexible when performed using reconfigurable / reprogrammable elements
- ?

#### **Disadvantages :**

- Complexity is higher
- Higher speed devices are required
- Analog signals need to be converted/deconverted using ADC/DAC
- ?

against analog communication

### **General Communication System**



### **Simplest Waveforms**

**PAM**: Pulse Amplitude Modulation (Amplitude of the signal carries the information)



Simplest PAM (binary antipodal signaling)

PAM signaling is generally used in *baseband channels* 

Why?

It would have been called ASK otherwise :)

### **Binary PAM**

Example Data : 0100110100...

Transmitted signal is a sum of the corresponding waveforms at appropriate positions



Problem of the receiver : Determine the 1-0 data sequence from the noisy signal

### **M-ary PAM**

Instead of 2 levels (binary) M levels (M-ary) can be used. M is selected so that  $M = 2^{k}$ 



2 bits are transmitted at each T now, but the receiver's job is more complicated

### **Signal Generation for Driving Channel**



### Example

1 MHz BPSK signal is to be generated without using frequency upconversion.



Calculations

Sample rate = 10 x 1 MHz = 10 Msps (min for DAC and generator)

Serial Data rate = 1 MHz / samples\_per\_period / carrier\_periods\_per\_bit = 1MHz / (10x5) = 20 kbps (max)

Corollary : You need high speed digital equipment even for low data rates. Constraints are tighter at the receiver side.

### Primitive (not optimized) Sequence Generator



since 10 is not an integer power of 2, a 16 x B ROM is used (remaining 6 locations are not used) and the counter should count from 0 to 9 for the previous example (rethink efficiency/design)

### **BPSK on Sinusoidal Sequence**



All these circuits, except DAC/ADC, can be implemented by digital circuits.

FPGAs are full of digital circuit primitives. Therefore these circuits can be implemented on FPGAs.

### Advantages of FPGAs ? :

- reconfigurable
- small, power efficient
- short development time

### What are FPGAs ?

### Field Programmable Gate Array : We have a bunch of digital circuit primitives with user programmable connections *designer*

There are several ways to design digital circuits on FPGAs One option is to use a HDL

Hardware Description Language :

We describe the circuits in plain text just like a programming language. But it is not a programming language! it is a description language.



In this course, we will be using VHDL

### **Start with a Simple Digital Example**

Consider the following combinatorial digital circuit and truth table



We can describe the function as

X = A when S=0, B when S=1 or

X = (A and not S) or (B and S)







It is a 1-bit 2-to-1 multiplexer as we know



We can make other multiplexers using this basic mux.



Programmable / Configurable devices basically work just like that

In a device, we have a finite number of

- 1. Flip-Flops, Registers
- 2. RAMs
- 3. Look Up Tables (LUTs)
- 4. Gates
- 5. Arithmetic Units
- 6. MUXs
- 7. Other (clock managers, buses, I/O blocks etc)

that we can interconnect them as we wish and design the digital circuit needed

**or** we can use a **HDL** and let a compiler/synthesizer do the design and optimization for the resource/performance balance.

Example (Xilinx-Spartan3E structure)





### It may not be what it looks like

Combinatorial functions are usually implemented with look-up tables



with N being the number of variables in the function.

### **Steps of VHDL Design Flow**



Since we have Spartan3E kits in the lab. we will be referring Xilinx ISE tool from now on, remembering that other vendors/manufacturers provide similar tools too. Tools for the steps mentioned here are mostly device/vendor specific.

### **Hello World**



### How we do it using ISE tool (ver 14.7)



Click Next



### Select your device

If you cannot find your board in this list, select 'none specified' and just select your FPGA chip

These should be as shown here - -

#### New Project Wizard x Project Settings Specify device and project properties. -Select the device and design flow for the project Value. Property Name **Evaluation Development Board** 5partan-3E Starter Board Product Category Family Spartan3E XC35500E Device Package FG320 -4 -Speed Top-Level Source Type $\nabla$ Synthesis Tool XST (VHDL/Verilog) • Simulator • ISim (VHDL/Verilog) • Preferred Language VHDL Store all values • Property Specification in Project File Manual Compile Order • VHDL Source Analysis Standard VHDL-93 Enable Message Filtering More Info $\underline{N}ext >$ < <u>B</u>ack Cancel

Click Next

and click 'Finish' on the Project Summary dialog box



need to add the file to the project later.



### Define ports of the entity

The convention is to create a source file for each entity (circuit). Here you may define inputs/outputs of this entity.

Since source files are text files, many coders skip this step and insert/edit the port description by hand.

| >New Source Wizard                    |               |           |            |                |       |      |
|---------------------------------------|---------------|-----------|------------|----------------|-------|------|
| <b>Define Module</b><br>Specify ports | s for module. |           |            |                |       |      |
| Entity name                           | mux2to1       |           |            |                |       |      |
| Architecture name                     | Behavioral    |           |            |                |       |      |
|                                       | Port Name     | Direction | Bus        | MSB            | LSB   |      |
| A                                     |               | in 💌      |            |                |       |      |
| В                                     |               | in 💌      |            |                |       |      |
| 5                                     |               | in 💌      |            |                |       |      |
| X                                     |               | out       |            |                |       |      |
|                                       |               | in 💌      |            |                |       |      |
|                                       |               | in 💌      |            |                |       |      |
|                                       |               | in 💌      |            |                |       |      |
|                                       |               | in 💌      |            |                |       |      |
|                                       |               | in 👱      |            |                |       |      |
|                                       |               | in 👱      |            |                |       |      |
|                                       |               | in 🔻      |            |                |       |      |
| More Info                             |               | < [       | <u>ack</u> | <u>N</u> ext > | Cance | el 🗌 |

Default signal type is STD\_LOGIC.

```
entity mux2to1 is
    Port ( A : in STD_LOGIC;
        B : in STD_LOGIC;
        S : in STD_LOGIC;
        X : out STD_LOGIC);
end mux2to1;
```

Click Next

and click 'Finish' on the Summary dialog box

We now have a source file editor window with entity description, some comments and library definitions and an empty architecture section. Architecture section is where you describe your circuit's behaviour.

```
Edit/Insert VHDL Code
 6
                                     library IEEE;
                                     use IEEE.STD LOGIC 1164.ALL;
                                     entity mux2to1 is
                                         Port ( A : in STD LOGIC;
                                                 B : in STD LOGIC;
                                                 S : in STD LOGIC;
                                                 X : out STD LOGIC);
                                     end mux2to1:
                                     architecture Behavioral of mux2to1 is
                                     begin
Insert logical expressions here ---- X <= (A and not S) or (B and S)
(between begin and end keywords of
Architecture section)
                                     end Behavioral;
```

Click Save icon

When saving, automatic syntax check is performed. Watch Console for error messages



You can also check syntax by right clicking on the *Check Syntax* item in *Design Tab* and selecting *Run*, or double click on *Check Syntax* 





This time, the syntax error is caused by a missing; in  $X \le (A \text{ and } not S)$  or (B and S)





You should see the green checkmark on *Synthesize* item too after synthesizing

## Now we need to implement a physical circuit for our selected device from this workable circuit description

It is imperative to define actual input output pins for a correct implementation as our design is a complete circuit and wee need to test it by applying actual signals to the inputs and monitoring the outputs.

Therefore, we need to tell "which signal goes to which pin of the device" before this step. We do this by creating a constraint file.



Click Next

and click 'Finish' on the Summary dialog box

we will see a new editor window named as mux2to1.ucf (User Constraint File)

User Constraint File is a text file used for describing various constraints. There is a complete book on the possible contents of this file. This time we are just interested in pin connections.

Enter the following lines in the window and save it.

| ; | "H18" | = | LOC | "A" | NET |
|---|-------|---|-----|-----|-----|
| ; | "L14" | = | LOC | "B" | NET |
| ; | "N17" | = | LOC | "S" | NET |
| ; | "F12" | = | LOC | "X" | NET |

It tells the implementor to connect the I/Os of our multiplexer to the physical switches and LEDs on our Spartan 3E Evaluation Board.



X (LED)

For example: A is the signal name, H18 is the pin number of the FPGA which is physically connected to the second switch on the board

Note : Instead of editing UCF as described above, you may also enter the following into the declaration part of the architecture section of the VHDL file. Differences will be mentioned later.

attribute LOC: string; attribute LOC of "A" : signal is "H18"; attribute LOC of "B" : signal is "L14"; attribute LOC of "S" : signal is "N17"; attribute LOC of "X" : signal is "F12";



Programming File is a binary file with \*.BIT extension.

This file will be loaded onto FPGA through FPGA's programming pins.

Our Spartan 3E Starter Board has a USB programming feature

through which this file can be sent.

For this purpose, we will be using IMPACT program which can be

initiated by Configure Target Device item. -

(IMPACT can also be started externally)



### 11 Connect your board to your PC using its USB cable



Right Click on the blank window and select Initialize Chain to search for devices on the board

We should see the devices on the board in a chain configuration.

We also see a warning message about the configuration file(s). This time we will asssign the configuration file manually, therefore, dismiss this dialog box and the next one.



For the Spartan 3E Starter Board, there should be 3 programmable devices in the chain. The first one (xc3s500e) is the FPGA device and the one we are to program.

Click on xc3s500e to select device. Right click and select Assign New Configuration File...

| 🛃 Assign New Configuration                                     | File                              |                       | x           |         |                |                                    |
|----------------------------------------------------------------|-----------------------------------|-----------------------|-------------|---------|----------------|------------------------------------|
| VHDL_Proj                                                      | ects 🕶 mux2to1 👻 🔫                | Search mux2to1        | 2           |         | (SP178P1<br>2? |                                    |
| Organize 🔻 New folder                                          |                                   |                       | - 🗆 🕐       | трі     | E XILINX"      |                                    |
| 🔶 Favorites                                                    | Name *                            | Date modified         | Туре        |         |                | Get Device ID                      |
| Nesktop                                                        | 🐌 _ngo                            | 26.09.2016 10:54      | File folder |         |                | Get Device ID                      |
| Downloads                                                      | 鷆 _xmsgs                          | 26.09.2016 11:02      | File folder |         | xc3s500e       | Get Device Signature/Usercode      |
| 🔠 Recent Places                                                | 퉬 ipcore_dir                      | 26.09.2016 08:43      | File folder | TDO     | bypass         | Add SPI/BPI Flash                  |
| 🔁 Libraries                                                    | 퉬 iseconfig                       | 26.09.2016 09:08      | File folder | 100     |                | Assign New Configuration File      |
| Documents                                                      | 퉬 xlnx_auto_0_xdb                 | 26.09.2016 10:54      | File folder |         |                | Set Programming Properties         |
| 🌙 Music                                                        | 鷆 xst                             | 26.09.2016 10:16      | File folder |         |                |                                    |
| Pictures                                                       | mux2to1.bit                       | 26.09.2016 11:02      | BIT File    |         |                |                                    |
| Videos —                                                       | <b></b> Type:<br>Size: 2<br>Date: | BIT File              | ;           |         |                |                                    |
| Local Disk (C:)                                                |                                   |                       | -           | Find ar | nd sele        | ct mux2to1.bit file and click Open |
| File name: mux2to1.bit 🔹 All Design Files (*.bit *.rbt *.nky 💌 |                                   |                       |             |         |                |                                    |
|                                                                |                                   | <u>O</u> pen <b>v</b> | Cancel      |         |                |                                    |



Dismiss the dialog box about attachment of the SPI / PROM devices by clicking No

Select xc3s500e again, right click on the device and select Program



We should see the Program Succeeded message in the window

Program Succeeded

We can now test our multiplexer using switches and observing the LED

**warning :** Please try not to load files for xcf04s and xc2c64a devices and program them. This will destroy their original content and disables us to use simple test feature at the power up

### Reusability



### **Back to Code**



### STD\_LOGIC Type

STD LOGIC types can take the following values



Hmw : design a 4 to 1 mux using two 2 to 1 mux with oe inputs ③
#### STD\_LOGIC\_VECTOR Type

#### A collection of **STD** LOGIC types

Example: signal sel, sel2 : STD\_LOGIC\_VECTOR (0 to 3); signal LEDS : STD\_LOGIC\_VECTOR (7 downto 0);

```
sel <= "0110";
sel2(2 to 3) <= "01";
LEDS <= (7=>'1', 6=>'0', others=>'Z');
LEDS(4) <= '0'; -- notice single quotes
LEDS <= LEDS +1;
-- requires use IEEE.STD_LOGIC_ARITH.ALL;
```

LEDS(7 downto 4) <= sel;</pre>

#### Since Hi-Z based bus systems are not possible in FPGAs



#### **Another Combinatorial Example**



| LEDS <= | "00000001"  | when | SWS="000" | else |  |
|---------|-------------|------|-----------|------|--|
|         | "00000010"  | when | SWS="001" | else |  |
|         | "00000100"  | when | SWS="010" | else |  |
|         | "00001000"  | when | SWS="011" | else |  |
|         | "00010000"  | when | SWS="100" | else |  |
|         | "00100000"  | when | SWS="101" | else |  |
|         | "01000000"  | when | SWS="110" | else |  |
|         | "10000000", | ;    |           |      |  |

with SWS select LEDS <= "00000001" when "000", "00000010" when "001", "0000100" when "010", "0001000" when "011", "00100000" when "100", "00100000" when "101", "01000000" when "110",

| LEDS(0) | <= | '1' | when | SWS | = | "000" | else | '0'; |
|---------|----|-----|------|-----|---|-------|------|------|
| LEDS(1) | <= | '1' | when | SWS | = | "001" | else | '0'; |
| LEDS(2) | <= | '1' | when | SWS | = | "010" | else | '0'; |
| LEDS(3) | <= | '1' | when | SWS | = | "011" | else | '0'; |
| LEDS(4) | <= | '1' | when | SWS | = | "100" | else | '0'; |
| LEDS(5) | <= | '1' | when | SWS | = | "101" | else | '0'; |
| LEDS(6) | <= | '1' | when | SWS | = | "110" | else | '0'; |
| LEDS(7) | <= | '1' | when | SWS | = | "111" | else | '0'; |

do not forget to cover all possibilities
 when using select

if(SWS="000") LEDS<="00000001";
elsif(SWS="001") LEDS<="00000010";
...
else ...
end if;</pre>

classic if-elsif-else-end if; can be
used in processes

## 3 to 7 Decoder

```
library IEEE;
use IEEE.STD LOGIC VECTOR.ALL;
entity decoder3to7 is
    Port ( D : in STD LOGIC VECTOR(2 downto 0);
           Q : out STD LOGIC VECTOR(7 downto 0));
end decoder3to7;
architecture decoder3to7 of decoder3to7 is
begin
  Q <= "00000001" when D="000" else
       "00000010" when D="001" else
       "00000100" when D="010" else
       "00001000" when D="011" else
       "00010000" when D="100" else
       "00100000" when D="101" else
       "01000000" when D="110" else
       "1000000";
end decoder3to7 ;
```

# As simple as it gets





SPI is a modern way to communicate between master and slave IC's on a single PC-board

# **Initial Design (Synchronous Serial)**



Let us assume that data and clock line lengths differ by 10 cm



#### one of the signals arrive 0.5 picoseconds late.

(speed of e.m. wave on copper is about 2x10<sup>8</sup> m/s)

Problem is : for a 1GHz clock, 0.5 ps is about half a clock cycle.



# **Solution**

Solution is to generate clock from data at the receiver.

The data signal should necessarily be designed to perform such an operation



A Phase Locked Loop (PLL) can be used if there are enough transitions in the signal

# Use of Ground as signal return



This requires that grounds at both side must have the same potential which we cannot guarantee



# **Differential Signaling**



Receiver uses the voltage difference between two inputs.

The voltage between a signal line and the ground is not in the formula here, but we have another problem.

If there is a voltage difference between two grounds then there will be a *common mode current* on the signal lines returning from ground.

## **Capacitors to prevent CMC**



Problem is different this time.



if there are long runs of 0s or 1s in the signal the receiver might loose the synchronization and/or cannot read the data.

# Long runs of 0s and 1s is solved by coding



## What is a Sequential Circuit?

If a circuit needs some values calculated from previous input values, it has to have some way of remembering these values.



One would design such circuits using two well known models; Mealy (output depends on the inputs and stored values) and Moore (output depends only on the stored values)

Today, to make things simpler, almost all digital circuits are designed as clocked *sequential* machines (Moore). Things are done *synchronous*ly with the rising or falling (or both) edge of a clock signal.





"Sequential", for digital circuits, does not mean that circuit pieces described by each of our VHDL code lines do their stuff one after another. It means that, outputs of designed circuits are someway affected from the previous input sequences.

Remember that, VHDL is not a programming language. It is a description language. We are describing a digital circuit \* that does things *concurrent*ly (keeping in mind the delays caused by the electronics and the finite speed of EM waves on silicon of course).

## **Some Attributes**

```
Given
signal D : STD LOGIC VECTOR(7 downto 0);
signal X : STD LOGIC VECTOR(2 to 5);
                (lower array index)
D'LOW is 0
X'LOW is 2
D'HIGH is 7
                (upper array index)
X'HIGH is 5
D'LEFT is 7
                (leftmost array index)
X'LEFT is 2
                (rightmost array index)
D'RIGHT is 0
X'RIGHT is 5
D'LENGTH is 8 (size of the array)
X'LENGTH is 4
D'RANGE is (7 downto 0) (range of the array)
X'RANGE is (2 to 5)
D'REVERSE RANGE is (0 to 7) (range of the array in reverse order)
X'REVERSE RANGE is (5 downto 2)
```

```
signal Y: STD_LOGIC_VECTOR(D'RANGE);
...
D(D'RIGHT) <= '1'; -- set righmost bit to 1</pre>
```

#### **Some Attributes**

Given signal CLK : STD\_LOGIC;

| CLK'EVENT     | is TRUE if there is an event on CLK                                   |
|---------------|-----------------------------------------------------------------------|
| CLK'STABLE[t] | is TRUE if there is no event on CLK in last t time unit               |
| CLK'ACTIVE    | is TRUE when there is a transaction on (assignment) CLK               |
| CLK'QUIET[t]  | is TRUE if there is no transaction on CLK during the last t time unit |

'EVENT and 'STABLE attributes are synthesizable, others are for simulation only



#### processes



#### **Other Synthesizable Pre-Defined Simple Types**

**STD ULOGIC** : U stands for unresolved

- **BOOLEAN** : True or False
- **INTEGER** : 32 (max) bit integers (-2147483647 to +2147483647)
- **NATURAL** : non-negative integers

...and arrays of these...

```
signal OE : STD_LOGIC;
signal count : integer;
signal IsOK, DOIT : BOOLEAN;
```

```
OE <= 'Z';
count <= count + 1;
IsOK <= not DOIT ;
DOIT <= False;</pre>
```

we need to use related library for some types

#### **Vectors**

Notice that integer and natural are actually collections of bits. We have other collections too.

```
BIT VECTOR : collection of BITS
   STD LOGIC VECTOR : collection of STD LOGIC types
   STD ULOGIC VECTOR :
   SIGNED, UNSIGNED : kinda integerS
signal sel,sel2 : BIT VECTOR (0 to 3);
signal LEDS : STD LOGIC VECTOR (7 downto 0);
signal count : integer range 0 to 15;
                                           inherently creates a 4 bit signal
sel <= "0110";</pre>
sel2(2 to 3) <= "01";
LEDS <= (7=>'1', 6=>'0', others=>'Z');
count \leq count +1; -- counts up to 15
```

LEDS(7 downto 4) <= sel; -- error, incompatible types</pre>

Text Book : V.A. Pedroni, Circuit Design with VHDL, MIT Press.

Homework : Read sections 1, 2, 3, 4 Do problems 3.2, 3.4, 4.1, 4.2. Design a 7-segment decoder using a 10x7 ROM array. Design a press-on/press-off button controlled LED circuit

Text Book : B. Sklar, Digital Communications (2nd Ed.) Fundamentals and Applications, MIT Press. Homework : Read section 2.3, 2.3.1, 2.8.5

Do problems 2.2, 2.3.

#### An Up-Counter With Asynchronous Reset



# Model of the Simulation



# **VHDL Test Bench & Simulation**



Associate it with the implementation file and close the summary dialog

2

**Create Generators** 

```
ARCHITECTURE behavior OF cntr4 tb IS
  COMPONENT CntrWRst
    PORT (
         clk : IN std logic;
         Rst : IN std logic;
         Data : INOUT std logic vector(3 downto 0));
  END COMPONENT;
  signal clk : std logic := '0';
  signal Rst : std logic := '0';
  signal Data : std logic vector(3 downto 0);
BEGIN
 uut: CntrWRst PORT MAP (
    clk => clk,
    Rst => Rst,
    Data => Data
  );
  clk process :process begin
    clk <= '0';
    wait for 10ns;
    clk <= '1';
    wait for 10ns;
   end process;
  stim proc: process
   begin
     wait for 5ns;
     Rst <= '1';
     wait for 35ns; Rst <= '0';</pre>
     wait for 400ns; Rst <= '1';</pre>
     wait for 60ns; Rst <= '0';</pre>
     wait;
   end process;
END;
```





| 🧱 ISim (P.49d) -                         | [Default.wcfg]                    |                    |                  |         |           |       |         |       |           |       |       |        |      |              |          |        |
|------------------------------------------|-----------------------------------|--------------------|------------------|---------|-----------|-------|---------|-------|-----------|-------|-------|--------|------|--------------|----------|--------|
| 虅 <u>F</u> ile <u>E</u> dit <u>V</u> iew | <u>S</u> imulation <u>W</u> indow | Layo               | out <u>H</u> elp |         |           |       |         |       |           |       |       |        |      |              |          |        |
| ] 🗋 ờ 🖥 👗 📗                              | X 🗅 🗋 🗙 🕲 🕨                       | ଦ ଜା               | 🗛 🐹 🌡 🕆 🐼        | 5 H H 5 | ] 🎤 💦     | ] 🏓 🌶 | ° 🖉 🥕 🗟 | 12 A  | 1 1       | * 🖸   | • • × | 1.00us | ▼ ⁄≣ | <b> </b> 🗔 R | e-launch |        |
| Instanc 🕶 🗖 🗗 🗙                          | Objects ↔ 🗆 🗗 🗙                   | ۶                  |                  |         |           |       |         |       | 50.042 n: | S     |       |        |      |              |          |        |
| 🛛 🕥 🖃 🔏 🔹                                | Simulation Object                 | <i>&gt;</i> ⊃<br>⊛ | Name             | Value   | 10 ns     | 1     | 20 ns   | 40 ns |           | 60 ns |       | 80 ns  |      | 100 ns       |          | 120 ns |
| Instance and Pro-                        |                                   | 2                  | 🍓 clk            | 1       |           |       |         |       |           |       |       |        |      |              |          |        |
| ⊳ 📒 cntr4_tb                             | Object Name                       | ~                  | 🐚 rst            | 0       |           |       |         |       |           |       |       |        |      |              |          |        |
| 💿 std_logic_11                           | La ck                             |                    | 🕨 📑 data[3:0]    | 0001    | $\square$ |       | 0000    |       | 00        | 01    | 00    | 10 )   | 00   | 11 )         | 01       | 00     |
| 🔵 std_logic_ari                          | lla rst                           | $\odot$            |                  |         |           |       |         |       |           |       |       |        |      |              |          |        |
| 🔰 😒 std_logic_un:                        | 🕨 🏹 data[3:0]                     | 1                  |                  |         |           |       |         |       |           |       |       |        |      |              |          |        |
|                                          |                                   | ⇒r I               |                  |         |           |       |         |       |           |       |       |        |      |              |          |        |
|                                          |                                   |                    |                  |         |           |       |         |       |           |       |       |        |      |              |          |        |
| I                                        |                                   | Ť                  |                  |         |           |       |         |       |           |       |       |        |      |              |          |        |

# **A Simple Serial Communication System**

Problem : Design of a asynchronous serial comm. system between two boards with FPGA devices (Xilinx Spartan-3E starter kit)



www.xilinx.com

#### **Serial Peripheral Interface (SPI)**

Transmit: Master puts the bit to be transmitted on MOSI and raises the clock (SCLK) Slave latches the serial input on the rising edge of the incoming clock.



Receive: Master raises the SCLK and reads in MISO on the falling edge of SCLK.

SPI can not be used in long distance (a couple meters) communications, because of the timing problems. The length difference between clock and data lines must be much smaller than the clock wavelength.

SPI is used for board to board or chip to chip data transfers

#### Use of SPI in our Example

Transmiter: Master serializes the 4 bit data and puts each bit on the line.



Receiver: On the rising edge of SCLK slave reads in MOSI and deserializes every 4 bits.





# **Simple Transmitter**





signal cntr : integer range 0 to 3;



## **Simple Receiver**



```
signal reccntr : integer range 0 to 3;
signal recsig : STD LOGIC VECTOR(3 downto 0);
```



# **Simple Tranceiver**



| entity SI | PI IS | Port (                                                    |
|-----------|-------|-----------------------------------------------------------|
| clk       | : in  | <pre>STD_LOGIC;we need a clock for action</pre>           |
|           |       | transmitter part                                          |
| b         | : in  | <pre>STD_LOGIC_VECTOR(3 downto 0); parallel data in</pre> |
| SCK       | : out | t STD_LOGIC; SPI clock                                    |
| MOSI      | : out | t STD_LOGIC; serial data out                              |
|           |       | receiver part                                             |
| SCKin     | : in  | STD_LOGIC; SPI clock in                                   |
| MOSIin    | : in  | STD_LOGIC; serial data in                                 |
| LEDS      | : ou  | t STD_LOGIC_VECTOR(3 downto 0)); parallel data out        |
| end SPI;  |       |                                                           |

#### F9 E9 D11 C11 F11 E11 E12 F12



SW3 SW2 SW1 SW0 (N17) (H18) (L14) (L13)

#### Q1 : What if We Use Two Transmitter & Receiver ?



warning : do not connect any power line other than the Gnd

Do we have any problems having correct LEDs lid?

Q2: We directly connected SCK to internal 50 MHz oscillator. How about not transmitting SCK and using internal 50 MHz on each board?

# **Asynchronous Transmitter & Receiver**



MOSIin : in STD\_LOGIC; -- serial data in LEDS : out STD\_LOGIC\_VECTOR(3 downto 0));-- parallel data out end SPI;

> Q1: Do we have any problems having correct LEDs lid? Q2: Are LEDs stable ?

# **Solutions**



solution to bit number ambiguity

use frame/word synchronization markers



#### When the rest of the receiver works with another clock



#### When the frame/word marker is actually a data word

#### That happens.!

A careful frame design and transmit/receive protocol are required. Sometimes upper level sync words are inserted to the stream.

**Q**: Did you notice that problems are almost always on the receiver side?
### **Symbol Synchronization**

The aim is to locally generate a signal that is synchronous to the incoming stream



## Correlation

is a measure of similarity of two signals (in signal processing)

 $\psi(t)$  and r(t)



negative high correlation

we are to measure similarity for the symbol period (so that we can change it)

$$R_{\psi} = \int_{\alpha}^{\alpha + T_{b}} r(t)\psi(t)dt$$



#### **Implementation with Digital Circuits**



accumulator (discrete integrator) is as simple as a summation in VHDL

```
ACC: process(clkH) is begin
  if(rising_edge(clkH)) then
    R <= R + u;
  end if;
end process;</pre>
```

#### **Integration for a Symbol Period**



```
INTEGRATOR: process(clkH) is begin
if(rising_edge(clkH)) then
R <= R + ux - D[N-1];
D[N-1] <= D[N-2];
...
D[1] <= D[0];
D[0] <= ux;
end if;
end process;</pre>
```

## An Ad-Hoc Method for Synchronization



Cases for  $s_i$ 's:

- 1. All  $s_i$  are the same : No problem, continue at the same rate.
- 2.  $s_0$  is different : Clock is fast, so slow down (ignore next clock pulse)
- 3.  $s_2$  is different : Clock is slow, so move faster (use  $s_2$  as  $s_0$  for the next bit period)

Take  $s_1$  as the bit value for all three cases.

Notes : Technique is simple and easy to design for digital systems like FPGAs. It requires that the input to be very clean. With a little more effort a two sample per bit method can be developed.

If the signal is noisy, multisample digital or analog integrator may be required.

#### An Ad-Hoc Method for Synchronization

```
architecture SerialRec of SerialRec is
                            signal s0 : STD LOGIC :='0';
                            signal cntr : integer range 0 to 2 :=0;
                            signal dummypulse : STD LOGIC := '0';
                          begin
                            process(clk) is begin
                              if (RISING EDGE (clk)) then
                                if(dummypulse='1') then
                                  dummypulse <= '0'; cntr <= 0;</pre>
                                else
                                  if(cntr=0) then
                                                    oclk <= '0'; cntr <= 1;</pre>
                                    s0 \leq SDin;
                                  elsif(cntr=1) then
entity SerialRec is Port (
                                    SDout <= SDin; oclk <= '0'; cntr <= 2;</pre>
   -- three clock per bit
                                  else
 clk : in STD LOGIC;
                                    if (s0/=SDout) then -- Data is slow (clock is fast
 SDin : in STD LOGIC;
                                      dummypulse <= '1'; -- ignore next clock pulse
 SDout : inout STD LOGIC;
                                    elsif(SDin/=SDout) then -- Data is fast (clock is slow)
   -- one pulse per output bit
                                      s0 \leq SDin; cntr \leq 1;
 oclk : out STD LOGIC );
                                    else
end SerialRec;
                                      cntr <= 0;
                                    end if;
                                    oclk <= '1';
                                  end if;
                                end if; --dummypulse
                              end if; -- clk
                            end process;
                          end SerialRec;
```

## **Another Ad-Hoc Method**

11



| abc | decision | cntr correction |
|-----|----------|-----------------|
| 000 | 0        | -               |
| 001 | 0        | increment cntr  |
| 010 | 0        | Х               |
| 011 | 1        | decrement cntr  |
| 100 | 0        | decrement cntr  |
| 101 | 1        | Х               |
| 110 | 1        | increment cntr  |
| 111 | 1        | -               |

# **Simple Problem**



We need to generate required waveforms locally

## Start with Binary PAM (no waveform generation)



Channel signal is composed of these





# **Noisy Signal Case**



Having synchronization is equivalent to correct detection of symbols

## **Early-Late Gating**



### Waveform generation



#### **BRAM Primitives in FPGAs**

There are several RAM blocks in Xilinx FPGAs including Spartan-3E (and in other vendors' too)



## **Available Configurations in Spartan-3E**

#### Block RAM instantiation templates can be seen using Language Templates







# **Combining / Extending BRAMs**



Data Bus (Horizontal) Extension

| BRAM | BRAM |
|------|------|
|------|------|

additional address line



Vertical Extension (more addresses)



#### **Defining RAMs with Inference**

```
entity DualPortRAM is
    Port ( ADDR_A : in STD_LOGIC_VECTOR (6 downto 0);
        DATAIN : in STD_LOGIC_VECTOR (7 downto 0);
        WE : in STD_LOGIC;
        CLK : in STD_LOGIC;
        DATAO_A : out STD_LOGIC_VECTOR (7 downto 0);
        ADDR_B : in STD_LOGIC_VECTOR (6 downto 0);
        DATAO_B : out STD_LOGIC_VECTOR (7 downto 0));
end DualPortRAM;
architecture Behavioral of DualPortRAM is
   type TDPRAM is array (0 to 127) of std_logic_vector (7 downto 0);
        signal MRAM : TDPRAM;
....
```

```
process(CLK) begin

if (rising_edge(CLK)) then

if (WE = '1') then

MRAM(conv_integer(ADDR_A)) <= DATAIN ;

end if;

DATAO_A <= MRAM(conv_integer(ADDR_A));

DATAO_B <= MRAM(conv_integer(ADDR_B));

end if;

end process;
```

### Sample ROM

```
entity SinSamples is
    Port ( CLK : in STD LOGIC;
           DATA : out STD LOGIC VECTOR (7 downto 0));
end SinSamples ;
architecture Behavioral of SinSamples is
  type TDPRAM is array (0 to 31) of integer range -127 to 127;
  constant LUT: TDPRAM := {0,25,49,72,91,107,118,125,127,124,115,
                103,86,65,43,18,-7,-32,-56,-78,-96,-111,-121,-126,
                -127, -122, -112, -98, -80, -59, -35, -11\};
  signal ADDR: integer range 0 to 31;
begin
  process(CLK) begin
    if (rising edge(CLK)) then
      DATA <= conv std logic vector(LUT(ADDR));
      ADDR \leq ADDR+1;
    end if;
  end process;
end;
```

#### **Variable Frequency**

```
entity SinSamples is
    Port ( Incr : in STD LOGIC VECTOR (4 downto 0);
           CLK : in STD LOGIC;
           DATA : out STD LOGIC VECTOR (7 downto 0));
end SinSamples ;
architecture Behavioral of SinSamples is
  type TDPRAM is array (0 to 31) of integer range -127 to 127;
  constant LUT: TDPRAM := {0,25,49,72,91,107,118,125,127,124,115,
                103,86,65,43,18,-7,-32,-56,-78,-96,-111,-121,-126,
                -127, -122, -112, -98, -80, -59, -35, -11\};
  signal ADDR: integer range 0 to 31;
begin
 process(CLK) begin
    if (rising edge(CLK)) then
      DATA <= conv std logic vector(LUT(ADDR));
      ADDR <= ADDR + conv integer(Incr);
    end if;
  end process;
end;
```

#### LUT Based Controlled Frequency Waveform Generator



