

#### Universidad de Alcalá

ESCUELA POLITÉCNICA SUPERIOR

# MÁSTER UNIVERSITARIO EN INGENIERÍA DE TELECOMUNICACIÓN

## Práctica Entregable 1 Transmisor 16 QAM

Diseño de Circuitos Electrónicos para Comunicaciones

Autor:

Paula Bartolomé Mora

16 de enero de 2024

# Índice general

| 1.         | Intr                                            | oducci  | ión                                                       | 1  |  |
|------------|-------------------------------------------------|---------|-----------------------------------------------------------|----|--|
| 2.         | Cap                                             | tura 🕽  | ADC y memoria FIFO                                        | 3  |  |
|            | 2.1. Obtención de los datos convertidos del ADC |         |                                                           |    |  |
|            |                                                 | 2.1.1.  | Configuración del XADC                                    | 4  |  |
|            |                                                 | 2.1.2.  | Simulación de la señal de entrada                         | 4  |  |
|            | 2.2.                                            | Lógica  | de control de lectura y escritura de los datos de la FIFO | 6  |  |
|            |                                                 | 2.2.1.  | Configuración del bloque FIFO                             | 6  |  |
|            |                                                 | 2.2.2.  | Comprobación del bloque FIFO                              | 8  |  |
| 3.         | Mo                                              | dulado  | r 16-QAM                                                  | 9  |  |
|            | 3.1.                                            | Mapea   | ado 16-QAM                                                | 10 |  |
|            |                                                 | 3.1.1.  | Configuración del mapeado 16-QAM                          | 10 |  |
|            | 3.2.                                            | Proces  | so de mapeado + Zero Padding                              | 12 |  |
|            |                                                 | 3.2.1.  | Configuración del proceso de mapeado + Zero Padding       | 12 |  |
|            |                                                 | 3.2.2.  | Comprobación del proceso de mapeado + Zero Padding        | 13 |  |
|            | 3.3.                                            | Bloque  | e de filtrado pulse shaping - RRC                         | 15 |  |
|            |                                                 | 3.3.1.  | Configuración del filtro RRC                              | 15 |  |
|            |                                                 | 3.3.2.  | Comprobación del filtrado RRC                             | 18 |  |
| 4.         | Mez                                             | zclador | de Transmisión                                            | 20 |  |
|            | 4.1.                                            | Gener   | ación de las sinusoides                                   | 21 |  |
|            | 4.2.                                            | Gener   | ación de las componentes I/Q y suma                       | 21 |  |
|            | 4.3.                                            | Comp    | robación de funcionamiento del mezclador                  | 22 |  |
| <b>5</b> . | Ger                                             | ieració | n de relojes                                              | 23 |  |
| 6          | Cor                                             | clusio  | nes                                                       | 25 |  |

### Introducción

El presente trabajo tiene como objetivo principal el desarrollo de un sistema de comunicaciones basado en 16-QAM (Modulación de Amplitud en Cuadratura de 16 niveles). Para ello, se propone el empleo en conjunto de los diferentes bloques electrónicos IP estudiados en la asignatura y se divide el desarrollo del entregable en una serie de fases de configuración:

- Captura XADC y Memoria FIFO: En esta primera etapa se realizará la captura de una señal triangular y se procederán a almacenar los datos de entrada en una memoria FIFO.
- Mapeado QAM: Se generará el mapeado de 16-QAM para obtener los símbolos a partir de los valores almacenados en la memoria FIFO.
- Zero Padding: Se implementará un upscaling o Zero-Padding con relación 1:32 previo al filtrado para conseguir una respuesta de frecuencia óptima y minimizar la interferencias entre símbolos.
- Filtrado Root Raised Cosine: Se deberá generar y aplicar a cada rama I/Q el filtrado pulse shaping del *Root Raised Cosine* (RRC).
- Mezclador DDS y Multiplicadores: Se generarán las componentes de transmisión de cada rama I/Q a través de la multiplicación de los valores filtrados por funciones coseno/-seno respectivamente. Estas funciones serán obtenidas a partir de un bloque sintetizador de señal (DDS).
- Sumador: Se incluirá un bloque final para la suma de las componentes de transmisión de los caminos I y Q.



Figura 1.1: Distribución de bloques del sistema de comunicaciones (I)

Para lograr tanto un correcto funcionamiento de cada bloque electrónico como la interoperabilidad entre ellos al integrarlos en del sistema, será preciso un estudio en profundidad de la documentación proporcionada por el fabricante y un análisis cuantitativo y cualitativo de cada uno de los resultados obtenidos en las simulaciones funcionales.



Figura 1.2: Distribución de bloques del sistema de comunicaciones (II)

Como paso adicional de esta práctica y, en relación a la generación de relojes, se creará un uso práctico de un bloque MMCM.

## Captura XADC y memoria FIFO

Como se ha introducido anteriormente, en este capítulo se definirá el proceso de captura y conversión de una señal de entrada triangular y el posterior almacenamiento de la misma en una memoria FIFO. En la Figura 2.5 se muestra el diseño de los bloques necesarios para esta fase y la definición de sus correspondientes puertos de entrada y salida.



Figura 2.1: Diseño del bloque XADC+FIFO

#### 2.1. Obtención de los datos convertidos del ADC

#### 2.1.1. Configuración del XADC

En primer lugar, teniendo en cuenta que mi puesto de laboratorio es el 8, se debe generar a la entrada una señal triangular con un valor de frecuencia igual a 13KHz y que opere en un rango de valores de tensión entre 0 y 0.596 voltios. Los valores que toma en cada instante temporal serán definidos en el fichero design.txt, el cual tendrá que ser importado al proyecto para llevar a cabo la conversión.

Como se puede ver en la Figura 2.2 se definen valores de tensión para dos entradas (VP y VN), que corresponden a las entradas analógicas diferenciales del XADC. Por simplificación, se configura el valor absoluto de tensión para la entrada positiva (VP) y un valor nulo para la negativa (VN).

```
1 TIME VP VN
2 0 0 0
3 7692 0.298 0
4 15385 0.596 0
5 23077 0.298 0
6 30769 0 0
7 38462 0.298 0
8 46154 0.596 0
9 53846 0.298 0
10 61538 0 0
```

Figura 2.2: Valores de tensión (V) de la señal triangular extraídos de design.txt

El XADC se configura en modo continuo y con un único canal de entrada para capturar las muestras, por lo que se ha seleccionado el canal 3 para registrar el valor convertido de tensión  $(s\_drp\_daddr)$ . La conversión se realizará a una velocidad de 1 MS/seg con un reloj de 52 MHz y se establecerá un tiempo de adquisición igual a 4 para que la señal de captura sea estable.

#### 2.1.2. Simulación de la señal de entrada

Una vez configurado el bloque IP XADC se procede a simular la conversión de la señal de entrada. En la figura 2.3 se puede visualizar cómo se obtiene la salida digital de 16 bits por el puerto  $do_-out$ .



Figura 2.3: Conversión analógico-digital de la señal triangular de entrada

Por otro lado, en la Figura 2.4 se aprecia de una forma más clara los valores que toman cada uno de los puertos del XADC en cada conversión. La señal de fin de conversión eoc\_out se activa y como consecuencia, también lo hará la de enable del Puerto de Reconfiguración Dinámico (DRP) s\_drp\_den para comenzar la conversión del siguiente dato.



Figura 2.4: Fin de conversión y dato preparado

La operación de lectura se completa cuando la señal de dato ready se activa  $s\_drp\_drdy$ , indicando que se el dato está preparado a la salida.

```
1 -- frec. conversion aprox 1MS -> frec 52 KHz (modo continuo)
2 for i in 0 to 200 -- a la espera de 200 datos
3 loop
4    -- End of Conversion signal
5    wait until eoc_out_0 <= '1';
6    -- Data ready signal for the dynamic reconfiguration port
7    wait until s_drp_0_drdy = '1';
8 end loop;</pre>
```

Listing 2.1: Bucle de conversión de los 200 datos

## 2.2. Lógica de control de lectura y escritura de los datos de la FIFO

#### 2.2.1. Configuración del bloque FIFO

El bloque de memoria FIFO almacenará cada uno de los datos capturados por el bloque XADC. Consta de una capacidad de 256 posiciones de 12 bits cada una, por lo que a la entrada del bloque FIFO será precisa la configuración de un bloque auxiliar Slice para transformar los 16 bits de la señal de salida del XADC en sus 12 bits de mayor peso.



Figura 2.5: Diseño del bloque XADC+FIFO

Se trata de un paso necesario porque el XADC en el fondo maneja una precisión de 12 bits a la salida. En la Figura 2.5 se puede apreciar esta conversión, además de la estructura completa del bloque a implementar.

```
process(wr_clk)
 variable reg : std_logic; -- variable para almacenar dato de salida del
     xadc
3 begin
  if rising_edge(wr_clk) then
      if reset_in_0 = '1' then -- reset XADC = 1
          reg := '0';
          -- Enable signal for the dynamic reconfiguration port
          s_drp_0_den <= '0';
      else
          s_drp_0_den <= reg ; -- ENABLE = 1 solo durante un ciclo
10
          reg := eoc_out_0; -- Se habilita la salida de un nuevo dato
      end if;
12
13 end if;
14 end process;
wr_en <= s_drp_0_drdy; -- se escribe en la FIFO nuevo dato
```

Listing 2.2: Proceso de escritura

Entrando en el proceso de escritura, el funcionamiento sería el siguiente: cada vez que exista un nuevo dato proporcionado por el XADC se escribirá en la memoria FIFO de forma continua a 1 MS/s y con un reloj de 52 MHz  $(wr\_clk)$  hasta llegar a la capacidad máxima.

Una vez la memoria FIFO está llena, comenzará el proceso de lectura de 200 datos a una frecuencia de lectura de 2MHz  $(rd\_clk)$  mediante la activación de la señal  $(rd\_enable)$ . En este caso, al desear leer un número de datos menor a la capacidad máxima (256), será necesario configurar una señal auxiliar programable  $(programable\_full)$ .

```
process(rd_clk)
2 file outfile: text is out "FIFO_output.txt";
3 variable line_num: line;
variable opened, finished: std_logic := '0';
5 begin
      file_open (outfile, "FIFO_output.txt", write_mode);
6
      if rising_edge(rd_clk) then
          if reset_in_0 = '1' then -- reset XADC = 1
               rd_en <= '0';
9
          else
               --se comienza a leer la FIFO
11
               if prog_full = '1' then
12
                   rd_en <= '1';
13
               end if:
14
               if empty = '1' then
                   rd_en <= '0';
16
               end if;
17
               --proceso de escritura del .txt
18
               if rd_en = '1' and finished = '0' then
19
                   write (line_num, FIFO_out);
20
                   writeline(outfile, line_num);
21
                   opened := '1';
22
               end if;
23
               --Se finaliza la escritura de .txt tras el primer empty
24
               if opened = '1' and empty = '1' then
                   finished := '1';
26
               end if;
27
          end if;
28
      end if;
30 end process;
```

Listing 2.3: Proceso de lectura

Adicionalmente, es necesario implementar la escritura de un fichero de salida con los datos almacenados en la FIFO, en el cual cada dato de 12 bits será una nueva línea. Este servirá para simular la lectura de la FIFO desde otro proyecto independiente de Vivado, suponiendo de este modo la entrada del siguiente bloque de mapeado. Se debe limitar la escritura a los primeros 200 datos de salida mediante el uso de variables adicionales (opened, finished).

#### 2.2.2. Comprobación del bloque FIFO

En las Figuras 2.6 y 2.7 se pueden visualizar los procesos de escritura y de lectura en la memoria FIFO como se ha descrito en el apartado anterior. A modo de depuración de que se está produciendo un correcto funcionamiento según los tiempos configurados, se añaden a la simulación dos señales de contadores de ciclos de lectura y de escritura.



Figura 2.6: Simulación del proceso de escritura en la memoria FIFO



Figura 2.7: Simulación del proceso de lectura en la memoria FIFO

## Modulador 16-QAM

En este Capítulo se muestra el bloque encargado de la modulación 16-QAM. Este bloque se compone de tres procesos: mapeado 16-QAM, aplicación de Zero Padding y filtrado *Root Raised Cosine*. En la Figura 3.1 se puede visualizar la estructura de los bloques IP que se han incluido en este segundo proyecto de Vivado.



Figura 3.1: Diseño de los bloques mapeado 16-QAM + ZP y filtrado RRC

#### 3.1. Mapeado 16-QAM

#### 3.1.1. Configuración del mapeado 16-QAM

Para implementar una modulación 16-QAM (Modulación de Amplitud en Cuadratura) lo primero que se debe realizar es el bloque de mapeado, con el fin de asignar patrones de amplitud y fase a cada símbolo y proporcionar una mayor eficiencia espectral.

Para ello, se recibe una señal de entrada de 12 bits, obtenida a partir de la lectura de los 200 valores almacenados en la memoria FIFO. Se debe crear un proceso en el testbench que se ejecute en cada ciclo del reloj global, configurado a 192MHz, que realice la lectura del fichero FIFO\_out.txt. Como se había descrito en el Apartado 2.2.1, se escriben en dicho fichero los valores almacenados en la memoria FIFO para emplearlo como la entrada del bloque de mapeado 16-QAM y optimizar así los proyectos de Vivado.

```
process(clk) --192MHz
      variable line_buffer : line;
      variable nuevo_valor : STD_LOGIC_VECTOR(11 DOWNTO 0);
      begin
           if rst_0 = '1' then
6
               eof <= false; --se reinicia fin de archivo
               cont_in <= 0;
           elsif rising_edge(clk) then
               cont_in <= cont_in + 1;</pre>
               if cont_in = 96 then --cada 500 ns se lee entrada
11
                   cont_in <= 0;
13
                   if not eof then
                        if endfile(file_handle) then
14
                            eof <= true;
                        else
                            readline(file_handle, line_buffer);
17
                            read(line_buffer, nuevo_valor);
18
19
                            entrada <= nuevo_valor;</pre>
                        end if;
20
                   end if;
21
               end if;
22
           end if:
23
24 end process;
```

Listing 3.1: Proceso de lectura del fichero FIFO\_out.txt

Es importante tener en cuenta que la lectura se produce a una frecuencia de 2MHz, por lo que se debe añadir un contador de 96 ciclos en el proceso, además de comprobar si el fichero ha llegado al final. Según las especificaciones de este proyecto, la salida del mapeado debe funcionar a una frecuencia de 6MHz, es decir, el triple de la anterior porque por cada valor a la entrada de 12 bits se obtendrán 3 símbolos 16-QAM. En otros términos, se tratarán los bits de entrada de 4 en 4 para mapear cada símbolo 16-QAM.

El mapeado de cada símbolo 16-QAM supone la generación a la salida de dos caminos independientes: uno para la componente en Fase (I) y otro para la de Cuadratura (Q). Cada símbolo es de 3 bits con signo y puede tomar los valores (-3,-1,+1,+3) como se puede visualizar en la Figura 3.2, procedente del ejemplo simulado en Matlab.



Figura 3.2: Constelación 16-QAM

En este caso, se han decidido ajustar las tablas Q e I para establecer una codificación Gray. Esta se trata de una técnica específica que garantiza que solo un bit cambie entre dos valores consecutivos. Además, consigue minimizar la posibilidad de errores producidos por cambios de múltiples bits en la transmisión. A continuación, se puede visualizar la definición de cada una de las tablas:

Listing 3.2: Definición de la tabla de mapeado Q

```
type Array3Bit is array (0 to 15) of STD_LOGIC_VECTOR(2 downto 0);
constant tabla_mapeado_I: Array3Bit :=
    ("000", "000", "001", "001",
    "011", "011", "010", "010",
    "110", "110", "111", "111",
    "101", "101", "100", "100");
```

Listing 3.3: Definición de la tabla de mapeado I

#### 3.2. Proceso de mapeado + Zero Padding

#### 3.2.1. Configuración del proceso de mapeado + Zero Padding

Además de implementar el mapeado 16-QAM, se añade en el bloque el zero-padding para optimizar el sistema de comunicación. Se insertan ceros a la señal original y se extiende su duración, aplicando en este caso una relación 1:32. Por lo tanto, por cada flanco en el que se transmita información, le seguirán 31 flancos en los que se transmitan bits con valor cero.

Por tanto, siguiendo las especificaciones anteriores, se decide combinar el mapeado 16-QAM con el zero-padding en un mismo bloque, para así obtener a la salida del mismo los símbolos 16-QAM con el Zero Padding ya aplicado. Así, se garantiza que la señal tenga una duración suficiente para conseguir una respuesta de frecuencia óptima, además de minimizar la interferencia entre símbolos. De forma adicional, se ajusta la salida a una longitud de 8 bits para dejarla preparada para la entrada del filtro pulse shaping del *Root Raised Cosine* (RRC).

```
process(clk)
    variable contador : integer := 0; -- 4*contador+3 downto 4*contador
    variable cont_32 : integer := 0; -- 6MHz
    variable captured_bits : std_logic_vector(3 downto 0) := "0000";
    variable aux_salida : std_logic_vector(2 downto 0):= "000";
6 begin
    if rising_edge(clk) then
      if rst = '1' then
        contador := 0;
9
        cont_32 := 0;
        salida <= (others => '0');
11
      else
        if cont_32 = 32 then -- se saca simbolo por la salida
13
          cont_32 := 0;
          captured_bits := entrada(4*contador+3 downto 4*contador);
15
          aux_salida := tabla_mapeado_I(to_integer(unsigned(captured_bits)
     ));
          --salida extendida a 8 bits
17
          salida(2 downto 0) <= aux_salida;</pre>
18
          salida(7 downto 3) <= (others => aux_salida(2));
19
          contador := contador + 1;
20
          if contador = 3 then
            contador := 0;
22
          end if;
        else --Zero Padding
24
          cont_32 := cont_32 + 1;
          salida <= (others => '0');
26
        end if;
      end if;
28
    end if;
29
30 end process;
```

Listing 3.4: Proceso de mapeado (Camino I) + Zero Padding

Como se puede visualizar, el proceso emplea dos contadores: uno para extraer en cada flanco los bits capturados de 4 en 4 e implementar el mapeado QAM para obtener los símbolos de 3 bits y otro para implementar la relación 1:32 respectiva al Zero Padding.

#### 3.2.2. Comprobación del proceso de mapeado + Zero Padding

En el testbench se incluye el proceso de estimulación de las señales de habilitación (*clock enable*) y de reset del bloque de mapeado. También, se pueden apreciar las líneas de código dedicadas a la activación de las señales de validación de datos del filtro (*s\_axis\_data\_tvalid*), pero esto se detallará en el Apartado 3.3.2.

Además, se deben abrir los ficheros correspondientes a la lectura y escritura de datos para cada uno de los bloques: para lectura de las muestras almacenadas en la FIFO y para escritura de las salidas de las ramas de filtrado.

```
1 process
  begin
     file_open(file_handle, ".\FIFO_output.txt", READ_MODE);
     file_open(file_handle_I, ".\FIR_output_I.txt", WRITE_MODE);
     file_open(file_handle_Q, ".\FIR_output_Q.txt", WRITE_MODE);
6
     ce <= '0';
     rst_0 <= '1';
     s_axis_data_tvalid_0 <= '0';</pre>
10
     s_axis_data_tvalid_1 <= '0';</pre>
11
12
     wait for 32*clk_period;
13
     rst_0 <= '0';
14
15
     wait for 32*clk_period;
16
     ce <= '1';
17
     s_axis_data_tvalid_0 <= '1';</pre>
18
     s_axis_data_tvalid_1 <= '1';</pre>
19
20
     wait;
21
22 end process;
```

Listing 3.5: Proceso de estimulación de señales de reset y clock enable

A continuación, en las Figuras 3.3, 3.4 y 3.5 se puede apreciar cómo se implementa el bloque de mapeado 16-QAM + zero-padding a través del proceso expuesto anteriormente.



Figura 3.3: Simulación completa del proceso de mapeado 16-QAM + ZP



Figura 3.4: Simulación del proceso de mapeado 16-QAM + ZP (II) (zoom medio)



Figura 3.5: Simulación del proceso de mapeado 16-QAM + ZP (III) (zoom alto)

#### 3.3. Bloque de filtrado pulse shaping - RRC

#### 3.3.1. Configuración del filtro RRC

El filtro pulse shaping del *Root Raised Cosine* (RRC) es un tipo de filtro denominado de coseno elevado. Se emplea en sistemas de comunicación digital para dar forma de onda a los pulsos que se transmiten. Minimiza la interferencia entre símbolos adyacentes y maximiza la eficiencia del espectro.

El primer paso a seguir para crear el filtro RRC es obtener los coeficientes del mismo. Para ello, se va a emplear Matlab, estableciendo de la siguiente forma las especificaciones del filtro a diseñar:

Listing 3.6: Diseño del filtro RRC en Matlab

Tras ejecutar el código, se obtendrán en total 193 coeficientes. Esto tiene como consecuencia que, para aplicar el filtro a la señal de entrada se irá ajustando un enventanado de longitud igual a 193 datos (dato actual + 192 datos siguientes) hasta finalizar la etapa.



Figura 3.6: Respuesta al impulso del filtro RRC

Mediante la herramienta gráfica *fvtool* que proporciona Matlab se puede obtener la respuesta impulsiva del filtro RRC tal y como se muestra en la Figura 3.6.

Para añadir el bloque del filtro en Vivado se requiere conocer el número de bits de los coeficientes, tanto de la parte entera como de la parte fraccionaria. Para ello, se emplea el comando de MatLab sfi(rrcFilter) y se obtiene la salida de la Figura 3.8

```
DataTypeMode: Fixed-point: binary point scaling
Signedness: Signed
WordLength: 16
FractionLength: 17
```

Figura 3.7: Información sobre la matriz de coeficientes del filtro RRC

Como los coeficientes se encuentran dentro de un rango de valores entre 0 y 1, se establecen 16 bits para la logitud total y 17 bits, para la parte fraccionaria. Por otro lado, se configura el modo de truncamiento, una salida de filtrado de 24 bits y una entrada de 8 bits, que ya ha sido previamente ajustada en el bloque anterior, por lo que no se requieren procesos adicionales.

| Coefficient Options              |               |          |  |  |  |
|----------------------------------|---------------|----------|--|--|--|
| Coefficient Type                 | Signed        | ~        |  |  |  |
| Quantization                     | Quantize Only | ~        |  |  |  |
| Coefficient Width                | 16            | [2 - 49] |  |  |  |
| ✓ Best Precision Fraction Length |               |          |  |  |  |
| Coefficient Fractional Bits      | 17            | [0 - 17] |  |  |  |
| Coefficient Structure            |               |          |  |  |  |
| O Inferred                       |               |          |  |  |  |
| O Non Symmetric                  |               |          |  |  |  |
| <ul><li>Symmetric</li></ul>      |               |          |  |  |  |
|                                  |               |          |  |  |  |
| Data Path Options                |               |          |  |  |  |
| Input Data Type                  | Signed        | ~        |  |  |  |
| Input Data Width                 | 8             | [2 - 47] |  |  |  |
| Input Data Fractional Bits       | 0             | [8 - 0]  |  |  |  |
| Output Rounding Mode             | Truncate LSBs | ~        |  |  |  |
| Output Width                     | 24            | [1 - 28] |  |  |  |
| Output Fractional Bits : 12      |               |          |  |  |  |

Figura 3.8: Configuración del filtro en Vivado

Entrando en el código, para que el filtrado comience a operar es necesario primero resetear las entradas de validación de datos  $s\_axis\_data\_tvalid$  de ambos filtros (camino I y Q) y después, activarlas cuando la señal de  $clock\ enable$  se pone a 1.

Por otro lado, como se había realizado con la salida de la memoria FIFO, para el bloque de filtrado también se van a almacenar los valores de salida en un fichero .txt. Para ello, se emplea el proceso definido anteriormente para la lectura de la FIFO y se añaden las líneas necesarias de escritura de los nuevos ficheros.

```
process(clk) --192MHz
      --lectura datos FIFO
      variable line_buffer : line;
      variable nuevo_valor : STD_LOGIC_VECTOR(11 DOWNTO 0);
      --escritura datos FIR
6
      variable line_buffer_I, line_buffer_Q : line;
      variable nuevo_valor_I, nuevo_valor_Q : integer;
      begin
          if rst_0 = '1' then
11
               eof <= false; --se reinicia fin de archivo
12
               cont_in <= 0;
13
          elsif rising_edge(clk) then
14
               cont_in <= cont_in + 1;</pre>
15
               if cont_in = 96 then --cada 500 ns se lee entrada
                   cont_in <= 0;
17
                   if not eof then
18
                       if endfile(file_handle) then
19
                            eof <= true;
21
                            readline(file_handle, line_buffer);
22
                            read(line_buffer, nuevo_valor);
23
                            entrada <= nuevo_valor;</pre>
                       end if;
25
                   end if;
               end if;
27
               if s_axis_data_tvalid_0 = '1' and s_axis_data_tvalid_1 = '1'
      and not eof then
                   --escritura de salida FIR I a 192MHz
                   nuevo_valor_I:=to_integer(signed(m_axis_data_tdata_0));
30
                   write(line_buffer_I, nuevo_valor_I);
31
                   writeline(file_handle_I, line_buffer_I);
32
                   --escritura de salida FIR Q a 192MHz
33
                   nuevo_valor_Q:=to_integer(signed(m_axis_data_tdata_1));
34
                   write(line_buffer_Q, nuevo_valor_Q);
35
                   writeline(file_handle_Q, line_buffer_Q);
36
37
          end if;
  end process;
```

Listing 3.7: Proceso de escritura de los ficheros FIR\_output\_I.txt y FIR\_output\_Q.txt

#### 3.3.2. Comprobación del filtrado RRC

En las Figuras 3.9 y 3.10 se visualiza la simulación completa del proyecto que incluye los bloques de mapeado 16-QAM + zero-padding y filtrado RRC. Se puede comprobar cómo el filtro da forma de onda a los pulsos que se transmiten y que provienen del proceso de mapeado previo.

Como se había adelantado en el Apartado 3.2.2, en el testbench se deben activar las señales de validación de datos del filtro (s\_axis\_data\_tvalid) a la vez que se produce la habilitación del clock enable del mapeado. Esto es imprescindible añadirlo para que no se realice el filtrado si no existen nuevas muestras de entrada.



Figura 3.9: Simulación de los bloques de mapeado 16-QAM + ZP y filtrado RRC (I)



Figura 3.10: Simulación de los bloques de mapeado 16-QAM + ZP y filtrado RRC (II)

### Mezclador de Transmisión

En este Capítulo se describe el último bloque del sistema, correspondiente al mezclador de transmisión. Este bloque se compone de tres procesos: la generación de señales sinusoidales mediante un sintetizador de señal (DDS), la multiplicación de los datos I/Q por estas señales sinusoidales y la suma final de las componentes I/Q de transmisión. En la Figura 4.1 se visualiza la estructura de los bloques IP que se han incluido en este tercer proyecto de Vivado.



Figura 4.1: Diseño del bloque mezclador de transmisión

#### 4.1. Generación de las sinusoides

Se deben generar dos señales sinusoidales, en particular una función coseno y otra-seno, a partir de un bloque DDS (*Direct Digital Synthesis*). Este bloque es capaz de generar señales analógicas de forma directa mediante un oscilador controlado numéricamente (NCO) y para ello, es preciso configurar internamente la frecuencia de salida deseada de las señales. En el caso de este proyecto será de 100MHz y el reloj del sistema tendrá una frecuencia de 576MHz (3x192MHz), por lo que se obtendrán 5/6 muestras por ciclo.

La salida del bloque DDS ( $m_axis_data_tdata$ ) esta constituida por 16 bits, de los cuales 8 corresponderán a la señal de seno y los otros 8, a la de coseno como se puede apreciar en la Figura 4.2. Es por ello que se requiere dividir la salida y definir los bits que se dirigirán a la entrada de cada multiplicador mediante dos bloques slice.

| AXI4-Stream Port Structure |                    |        |  |  |  |
|----------------------------|--------------------|--------|--|--|--|
| M_AXIS_DATA - TDATA        |                    |        |  |  |  |
| Transaction                | <u>Field</u>       | Type   |  |  |  |
| 0                          | CHAN_0_SINE(15:8)  | fix8_7 |  |  |  |
|                            | CHAN_0_COSINE(7:0) | fix8_7 |  |  |  |

Figura 4.2: Configuración de salida de las señales sinusoidales en el bloque DDS

#### 4.2. Generación de las componentes I/Q y suma

Para generar las componentes I/Q de transmisión se deben multiplicar las salidas I/Q del filtrado realizado anteriormente por las funciones coseno/-seno generadas en el bloque DDS. Como se ha introducido, el procesamiento de los datos del filtro y el bloque de multiplicación trabajarán a una frecuencia de 576 MHz, siendo el triple de la empleada en los bloques anteriores.

Para las salidas I/Q del filtrado (Salida\_RRC\_I y Salida\_RRC\_Q), es preciso extraer los datos a partir de la lectura de los ficheros FIR\_output\_I.txt y FIR\_output\_Q.txt, generados anteriormente. A continuación, se muestra el proceso de lectura en el caso de la rama I.

```
process(clk)
      --lectura datos salidas filtros
      variable line_buffer : line;
      variable nuevo_valor : integer;
      begin
6
          if rst = '0' then
               eof_I <= false; --se reinicia fin de archivo</pre>
          elsif rising_edge(clk) then
               if not eof_I then
10
                   if endfile(file_handle_I) then
11
                       eof_I <= true;</pre>
                   elsif m_axis_data_tvalid_0 = '1' then
13
                       readline(file_handle_I, line_buffer);
14
                       read(line_buffer, nuevo_valor);
                       Salida_RRC_I <= std_logic_vector(to_signed(
16
     nuevo_valor, Salida_RRC_I'length));
                   end if;
17
               end if;
18
          end if;
19
20 end process;
```

Listing 4.1: Proceso de lectura del fichero de salida del filtrado (rama I)

Como proceso final, se deben combinar las dos componentes I/Q generadas a la salida de los multiplicadores en un bloque sumador. Este se configurará con dos entradas de 32 bits (C y CONCAT) y una salida de 33 bits.

#### 4.3. Comprobación de funcionamiento del mezclador

En el testbench se añade el proceso principal para activar y desactivar la señal de reset, que irá conectada al bloque DDS y de la cual también dependerá el comienzo de lectura de los ficheros de datos del filtro. Es imprescindible configurarlo de esta manera para que se realice tanto la generación de las ondas sinusoidales como la lectura del filtrado de forma simultánea para poder operar.

```
process
begin

file_open(file_handle_I, ".\FIR_output_I.txt", READ_MODE);

file_open(file_handle_Q, ".\FIR_output_Q.txt", READ_MODE);

rst <= '0'; -- negativo
    wait for 100 ns;
    rst <= '1';

wait;
end process;</pre>
```

Listing 4.2: Proceso de estimulación

## Generación de relojes

En relación a la generación de señales de reloj, se ha decidido implementar un bloque MMCM en el último proyecto de Vivado, perteneciente al bloque del mezclador (ver Capítulo 4).

El objetivo es hacer un uso práctico de este bloque en uno de los módulos del sistema para validar su funcionamiento, mientras que en el resto se trasladaría directamente la señal de reloj correspondiente. Se considera partir de un reloj externo de frecuencia igual a 36 MHz para generar a la salida tres relojes: de 576 MHz (DDS/Multiplicadores), 192 MHz (RRC) y 52 MHz (XADC).

Para ello, se añaden al proyecto dos bloques IP: Simulation Clock Generation, para generar el reloj externo y Clocking Wizard, para obtener las tres salidas de reloj con diferentes frecuencias. En la Figura 5.1 se puede observar la configuración del bloque IP para la generación de relojes Clocking Wizard.

#### **Primary Input Clock Attributes**

| Input Clock Frequency (MHz) | 36.000                         |  |
|-----------------------------|--------------------------------|--|
| Clock Source                | Single_ended_clock_capable_pin |  |
| Jitter                      | 0.010                          |  |

#### **Clocking Primitive Attributes**

Primitive Instantiated: MMCM
Divide Counter: 1
Mult Counter: 32.000
Clock Phase Shift: None

| Clock Wiz O/p Pins | Source       | Divider Value | Tspread (ps) | Pk-to-Pk Jitter (ps) | Phase Error (ps) |
|--------------------|--------------|---------------|--------------|----------------------|------------------|
| clk_out1           | MMCM CLKOUT0 | 2.000         | OFF          | 129.300              | 173.591          |
| clk_out2           | MMCM CLKOUT1 | 6             | OFF          | 151.084              | 173.591          |
| clk_out3           | MMCM CLKOUT2 | 22            | OFF          | 193.692              | 173.591          |

Figura 5.1: Resumen de configuración del bloque IP Clocking Wizard

Como se implementa en el módulo del mezclador se cogerá la salida con frecuencia 576 MHz y se conectará la misma a todos los bloques.



Figura 5.2: Comprobación del funcionamiento del bloque IP Clocking Wizard

### **Conclusiones**

Con este trabajo se ha conseguido aportar de forma significativa un mayor entendimiento sobre los sistemas de comunicaciones avanzados, especialmente sobre los basados en 16-QAM. Mediante la integración de los conceptos estudiados teóricamente en la asignatura y la aplicación de las tecnologías explicadas en el laboratorio, se han conseguido implementar todos los módulos del sistema propuesto, logrando el objetivo principal del proyecto.

Para ello, ha sido necesario un análisis en profundidad de la documentación dada por Xilinx en referencia con el funcionamiento de cada uno de los bloques a desarrollar. Este paso ha sido de gran importancia para garantizar una correcta configuración de los mismos.

Por otro lado, la validación del funcionamiento de los módulos se ha conseguido a partir del análisis cuantitativo y cualitativo de los resultados obtenidos en las simulaciones funcionales. Por ello, se han diseñado ficheros de testbench para cada módulo implementado. La continua evaluación del código ha permitido en ciertos momentos el reajuste de las configuraciones realizadas con el fin de corregir errores y optimizar el sistema.

Finalmente, se puede expresar que se ha logrado una integración completa de todos los módulos de forma exitosa mediante el desarrollo de procesos de lectura y escritura de ficheros de entrada y salida en el testbench para conseguir la cohesión en todo el sistema.

## Bibliografía