# WGR-V Verilog Doku

### Inhaltsverzeichnis

# Datei: rtl/peripherals/debug\_module.v

Modul: debug\_module

Kurzbeschreibung: Debug-Modul für Host-Kommunikation und Speicherzugriff

Dieses Modul stellt eine einfache Debug-Schnittstelle über UART bereit, um mit einem Host-System zu kommunizieren. Es ermöglicht das Lesen und Schreiben von Speicherinhalten über serielle Kommandos. Über den UART-Eingang rx empfängt das Modul Befehle, die über interne Kontrollsignale (mem\_rd, mem\_wr, etc.) an den CPU-internen Speicher weitergegeben werden. Das Antwortsignal wird über tx gesendet.

## Eingänge:

- clk Systemtakt
- rst Reset-Signal
- rx Eingang vom seriellen Host (UART)
- mem\_data\_from\_cpu Daten aus dem Speicher (vom CPU-Interface)
- mem\_ready\_from\_cpu Bereitschaftssignal vom Speicher

# Ausgänge:

- tx Ausgang zur seriellen Schnittstelle
- mem\_addr\_to\_cpu Speicheradresse für den Zugriff
- mem\_data\_to\_cpu Daten, die geschrieben werden sollen
- mem\_wr Speicher-Schreibsignal
- mem\_rd Speicher-Lesesignal
- mem\_valid Gültigkeitssignal für Speicherzugriff

#### Source Code

`default\_nettype none

```
`timescale 1ns / 1ns
module debug_module (
    input wire
                       clk,
                       rst_n,
    input wire
    input wire [7:0] address,
    input wire [31:0] write_data,
    output wire [31:0] read_data,
    input wire
                       we,
    input wire
    output wire [31:0] debug_out
  );
 localparam DEBUG ADDR = 32'h00000000;
 reg [31:0] debug_reg;
```

```
assign read_data = debug_reg;
assign debug_out = debug_reg;

always @(posedge clk or negedge rst_n)
begin
   if (!rst_n)
begin
    debug_reg <= 32'h000000000;
end
else
   if (we)
begin
    debug_reg <= write_data;
    $display("DEBUG_REG UPDATED: 0x%08X (%d) at time %0t", write_data, write_data, $time);
end
end</pre>
```

endmodule

# Datei: rtl/peripherals/fifo.v

Modul: fifo

Kurzbeschreibung: Ein einfacher FIFO-Speicher.

Dieser FIFO (First-In-First-Out) Puffer speichert Daten mit einer konfigurierbaren Breite (DATA\_WIDTH) und Tiefe (DEPTH). Daten werden über das wr\_en Signal hineingeschrieben und über das rd\_en Signal ausgelesen. Das Modul liefert die Signale empty und full, um anzuzeigen, ob weitere Lese- oder Schreibvorgänge möglich sind.

#### Parameter:

- DATA\_WIDTH Breite der gespeicherten Daten in Bits.
- DEPTH Anzahl der Einträge im FIFO.

#### Lokale Parameter:

ADDR\_WIDTH Breite der Adresszeiger basierend auf DEPTH.

## Eingänge:

- clk Systemtakt.
- rst\_n Aktiv-low Reset.
- wr\_en Aktivierungssignal (Write-Enable) zum Schreiben in den FIFO.
- rd\_en Aktivierungssignal (Read-Enable) zum Lesen aus dem FIFO.
- din Eingangsdaten mit Breite DATA\_WIDTH.

## Ausgänge:

- empty Signal, das anzeigt, ob der FIFO leer ist.
- full Signal, das anzeigt, ob der FIFO voll ist.
- dout Ausgangsdaten mit Breite DATA WIDTH.

#### Source Code

```
input wire
                                 rd_en,
    output wire
                                 empty,
   output wire
                                 full,
   input wire [DATA_WIDTH-1:0] din,
   output wire [DATA_WIDTH-1:0] dout
 );
 localparam ADDR_WIDTH = $clog2(DEPTH);
 reg [ADDR_WIDTH :0]
                          rd_ptr;
 reg [ADDR_WIDTH :0]
                         wr_ptr;
 reg [DATA_WIDTH-1:0] mem[0:DEPTH-1];
 reg wr_en_prev;
 reg rd_en_prev;
 wire [ADDR_WIDTH: 0] next_wr;
 assign empty = (rd_ptr == wr_ptr);
 assign full
                = (wr ptr[ADDR WIDTH]
                                        != rd ptr[ADDR WIDTH]) &&
                   (wr_ptr[ADDR_WIDTH-1:0] == rd_ptr[ADDR_WIDTH-1:0]);
 assign next_wr = (wr_ptr + 1);
 always @(posedge clk or negedge rst_n)
 begin
   if (!rst_n)
   begin
     wr_ptr <= 0;
     rd_ptr <= 0;
     wr_en_prev <= 0;</pre>
     rd_en_prev <= 0;
    else
   begin
     wr_en_prev <= wr_en;</pre>
     rd_en_prev <= rd_en;
     if (wr_en && !wr_en_prev && !full)
       mem[wr_ptr[ADDR_WIDTH - 1: 0]] <= din;</pre>
       wr_ptr <= next_wr;</pre>
     end
      if (rd_en && !rd_en_prev && !empty)
     begin
       rd_ptr <= rd_ptr + 1;
      end
    end
  end
 assign dout = empty ? {DATA_WIDTH{1'b0}} : mem[rd_ptr[ADDR_WIDTH - 1 : 0]];
endmodule
```

Datei: rtl/peripherals/gpio.v

Modul: gpio

Kurzbeschreibung: GPIO Modul zur Steuerung allgemeiner I/O.

Dieses Modul implementiert einen einfachen GPIO-Controller. Es unterstützt das Auslesen von Eingangspins und das Schreiben auf Ausgangspins über definierte Adressen. Die Richtungssteuerung (gpio\_dir) ist derzeit nicht implementiert.

## Eingänge:

- clk Systemtakt.
- rst\_n Aktiv-low Reset-Signal.
- address Speicheradresse zur Auswahl der GPIO-Register.
- write\_data Daten, die in die angesprochenen GPIO-Register geschrieben werden.
- we Schreibaktivierungssignal (Write-Enable).
- re Leseaktivierungssignal (Read-Enable).
- gpio\_in Eingangssignale von den GPIO-Pins.

### Ausgänge:

- read\_data Ausgangsdaten basierend auf dem angesprochenen GPIO-Register.
- gpio\_out Register zur Steuerung des Ausgangszustands der GPIO-Pins.
- gpio\_dir GPIO-Richtungsregister (nicht implementiert).

#### Source Code

```
`default_nettype none
`timescale 1ns / 1ns
module gpio (
  input wire
                     clk,
  input wire rst_n,
  input wire [7:0] address,
  input wire [31:0] write_data,
  output wire [31:0] read_data,
  input wire
                    we,
  input wire
                     re,
  output reg [7:0] gpio_out,
  output reg [7:0] gpio_dir,
  input wire [7:0] gpio_in
  // No dir implemented at the moment
  localparam GPIO_DIR_BASE = 8'h00;
  localparam GPIO_IN_OFFSET = 8'h04;
  localparam GPIO_OUT_OFFSET = 8'h08;
  localparam GPIO OUT STEP
                              = 8'h04:
  wire [2:0] pin_index = (address - GPIO_OUT_OFFSET) >> 2;
  assign read_data = (address == GPIO_IN_OFFSET) ? {24'd0, gpio in} :
                     (address >= GPIO_OUT_OFFSET && address < (GPIO_OUT_OFFSET + (8 * GPIO_OUT_STEP)))
                     ? {31'd0, gpio_out[pin_index]} : 32'd0;
  always @(posedge clk or negedge rst_n)
  begin
    if (!rst_n)
    begin
      gpio_out <= 8'd0;</pre>
      gpio_dir <= 8'd0;</pre>
    end
    if (we \&\& address >= GPIO_OUT_OFFSET \&\& address < (GPIO_OUT_OFFSET + (8 * GPIO_OUT_STEP)))
      gpio_out[pin_index] <= write_data[0];</pre>
    end
  end
```

#### endmodule

# Datei: rtl/peripherals/peripheral\_bus.v

## Modul: peripheral\_bus

Kurzbeschreibung: Einfacher Adressdecoder für Peripheriezugriffe

Dieses Modul implementiert einen einfachen Peripheriebus mit Adressdekodierung. Es leitet Speicherzugriffe basierend auf der übergebenen Adresse an die jeweils zuständige Peripherieeinheit weiter. Dabei werden Steuersignale (wr, rd, valid) sowie Datenleitungen entsprechend durchgeschaltet. Es unterstützt mehrere Peripheriegeräte mit jeweils eigenen Adressbereichen.

## Eingänge:

- clk Systemtakt
- rst Reset-Signal
- addr Adresse des Zugriffs
- wr Schreibbefehl
- rd Lesebefehl
- valid Gültigkeitssignal für Zugriff
- cpu\_data Daten vom CPU zur Peripherie

### Ausgänge:

- data\_out Rückgabewert von der aktiven Peripherie
- ready Gibt an, ob die Peripherie bereit ist
- gpio\_sel Auswahl für GPIO-Modul
- uart\_sel Auswahl für UART-Modul
- spi\_sel Auswahl für SPI-Modul
- ws2812b sel Auswahl für WS2812B-Modul
- pwm\_timer\_sel Auswahl für PWM-Timer
- system timer sel Auswahl für System-Timer
- debug\_module\_sel Auswahl für Debug-Modul

### Source Code

`include "../defines.v"
`default nettype none

```
`timescale 1ns / 1ns
module peripheral_bus (
  input wire
                     clk,
  input wire
                     rst_n,
  input wire [13:0] address,
  input wire [31:0] write_data,
  output reg [31:0] read_data,
  input wire
                     we,
  input wire
                     re,
  output wire [31:0] debug_out,
  output wire
                     uart_tx,
  input wire
                     uart_rx,
  output wire
                     pwm_out,
  output wire
                     ws out,
                     spi_mosi,
  output wire
  input wire
                     spi_miso,
  output wire
                     spi_clk,
  output wire
                     spi_cs,
  output wire [ 7:0] gpio_out,
  output wire [ 7:0] gpio_dir,
  input wire [ 7:0] gpio_in
);
```

```
localparam DEBUG_BASE = 6'h01;
 localparam UART_BASE = 6'h02;
 localparam TIME BASE = 6'h03;
 localparam PWM_BASE
                       = 6'h04;
 localparam MULT_BASE = 6'h05;
 localparam DIV_BASE = 6'h06;
 localparam SPI_BASE
                        = 6'h07;
 localparam GPIO_BASE = 6'h08;
 localparam WS_BASE
                        = 6'h09;
 wire [7:0] func_addr;
 assign func_addr = address[7:0];
  // Debug
`ifdef INCLUDE_DEBUG
 wire [31:0] debug_data;
 wire debug_sel;
 wire debug_we;
 wire debug_re;
 assign debug_sel = (address[12:8] == DEBUG_BASE);
 assign debug_we = we & debug_sel;
 assign debug_re = re & debug_sel;
 debug_module debug_inst (
           (clk),
   .clk
              (rst_n),
   .\mathtt{rst}_\mathtt{n}
   .address (func_addr),
    .write_data (write_data),
   .read_data (debug_data),
               (debug_we),
               (debug_re),
   .re
    .debug_out (debug_out)
 );
`else
 wire [31:0] debug_data = 32'h0;
               debug_sel = 1'b0;
 wire
               debug_out = 32'h0;
 assign
`endif
 // UART
`ifdef INCLUDE_UART
 wire [31:0] uart_data;
 wire uart_sel;
 wire uart_we;
 wire uart_re;
 assign uart_sel = (address[12:8] == UART_BASE);
 assign uart_we = we & uart_sel;
 assign uart_re = re & uart_sel;
```

```
uart #(
   .FIFO_TX_DEPTH(`UART_FIFO_TX_DEPTH),
    .FIFO_RX_DEPTH(`UART_FIFO_RX_DEPTH)
 ) uart_inst (
   .clk (clk),
   .rst_n
             (rst n),
   .address (func_addr),
   .write_data (write_data),
   .read_data (uart_data),
          (uart_we),
   .we
           (uart_re),
   .re
   .uart_tx (uart_tx),
   .uart_rx (uart_rx)
 );
`else
 wire [31:0] uart_data = 32'h0;
             uart_sel = 1'b0;
 wire
            uart_tx = 1'b0;
 assign
`endif
 // System Timer
`ifdef INCLUDE_TIME
 wire [31:0] time_data;
 wire time_sel;
 wire time_we;
 wire time_re;
 assign time_sel = (address[12:8] == TIME_BASE);
 assign time_we = we & time_sel;
 assign time_re = re & time_sel;
 system_timer system_timer_inst (
   .clk (clk),
.rst_n (rst_n),
.address (func_addr),
   .write_data (write_data),
   .read_data (time_data),
   .we
           (time_we),
              (time_re)
   .re
 );
`else
 wire [31:0] time_data = 32'h0;
            time_sel = 1'b0;
 wire
`endif
 // PWM Timer
`ifdef INCLUDE_PWM
 wire [31:0] pwm_data;
 wire pwm_sel;
 wire pwm_we;
 wire pwm_re;
```

```
assign pwm_sel = (address[12:8] == PWM_BASE);
 assign pwm_we = we & pwm_sel;
 assign pwm_re = re & pwm_sel;
 pwm_timer pwm_timer_inst (
           (clk),
   .clk
             (rst_n),
   .rst_n
   .address (func_addr),
   .write_data (write_data),
   .read_data (pwm_data),
               (pwm_we),
   .we
   .re
             (pwm_re),
   .pwm_out (pwm_out)
 );
`else
 wire [31:0] pwm_data = 32'h0;
           pwm_sel = 1'b0;
 wire
            pwm_out = 1'b0;
 assign
`endif
 // Sequential Multiplier
`ifdef INCLUDE_MULT
 wire [31:0] mult_data;
 wire mult_sel;
 wire mult_we;
 wire mult_re;
 assign mult_sel = (address[12:8] == MULT_BASE);
 assign mult_we = we & mult_sel;
 assign mult_re = re & mult_sel;
 seq_multiplier seq_multiplier_inst (
   .clk (clk),
   .rst n
             (rst_n),
   .address (func_addr),
   .write_data (write_data),
   .read_data (mult_data),
   .we
           (mult_we),
             (mult_re)
   .re
 );
`else
 wire [31:0] mult_data = 32'h0;
            mult_sel = 1'b0;
 wire
`endif
 // Sequential Divider
`ifdef INCLUDE_DIV
 wire [31:0] div_data;
 wire div_sel;
 wire div_we;
 wire div_re;
```

```
assign div_sel = (address[12:8] == DIV_BASE);
 assign div_we = we & div_sel;
 assign div_re = re & div_sel;
 seq_divider seq_divider_inst (
   .clk
           (clk),
             (rst_n),
   .rst_n
   .address (func_addr),
    .write_data (write_data),
   .read_data (div_data),
               (div_we),
   .we
   .re
              (div_re)
 );
`else
 wire [31:0] div_data = 32'h0;
            div_sel = 1'b0;
`endif
 // SPI Peripheral
`ifdef INCLUDE_SPI
 wire [31:0] spi_data;
 wire spi_sel;
 wire spi_we;
 wire spi_re;
 assign spi_sel = (address[12:8] == SPI_BASE);
 assign spi_we = we & spi_sel;
 assign spi_re = re & spi_sel;
 spi #(
    .FIFO_TX_DEPTH(`SPI_FIFO_TX_DEPTH),
    .FIFO_RX_DEPTH(`SPI_FIFO_RX_DEPTH)
 ) spi_inst (
   .clk
              (clk),
   .rst_n (rst_n),
   .address (func_addr),
   .write_data (write_data),
   .read_data (spi_data),
   .we
          (spi_we),
             (spi_re),
   .re
   .spi_clk (spi_clk),
   .spi_mosi (spi_mosi),
   .spi_miso (spi_miso),
    .spi_cs
             (spi_cs)
 );
`else
 wire [31:0] spi_data = 32'h0;
 wire spi_sel = 1'b0;
 assign
           spi_mosi = 1'b0;
           spi_clk = 1'b0;
 assign
 assign
           spi_cs = 1'b1;
`endif
```

```
`ifdef INCLUDE_GPIO
 wire [31:0] gpio_data;
 wire gpio_sel;
 wire gpio_we;
 wire gpio_re;
 assign gpio_sel = (address[12:8] == GPIO_BASE);
 assign gpio_we = we & gpio_sel;
  assign gpio_re
                  = re & gpio_sel;
 gpio gpio_inst (
   .clk
            (clk),
    .\mathtt{rst}_\mathtt{n}
               (rst_n),
    .address (func_addr),
    .write_data (write_data),
    .read_data (gpio_data),
    .we
                (gpio_we),
    .re
               (gpio_re),
    .gpio_out (gpio_out),
    .gpio_dir (gpio_dir),
    .gpio_in (gpio_in)
 );
`else
 wire [31:0] gpio_data = 32'h0;
 wire
          gpio_sel = 1'b0;
 assign
              gpio_out = 8'b0;
 assign
              gpio_dir = 8'b0;
`endif
  // WS2812B
`ifdef INCLUDE_WS
 wire [31:0] ws_data;
 wire ws_sel;
 wire ws_we;
 wire ws_re;
  assign ws_sel = (address[12:8] == WS_BASE);
 assign ws_we = we & ws_sel;
  assign ws_re = re & ws_sel;
 ws2812b ws2812b_inst (
              (clk),
    .clk
    .rst_n
               (rst_n),
    .address
               (func_addr),
    .write_data (write_data),
    .read_data (ws_data),
    .we
                (ws_we),
               (ws_re),
    .re
    .ws_out
               (ws_out)
 );
`else
 wire [31:0] ws_data = 32'h0;
             ws_sel = 1'b0;
 wire
```

```
ws_out = 1'b0;
 assign
`endif
 always @(posedge clk or negedge rst_n)
    if (!rst_n)
      read_data <= 32'h0;</pre>
    else
    if (re)
    begin
`ifdef INCLUDE_DEBUG
      if
               (debug_sel) read_data <= debug_data;</pre>
`endif
`ifdef INCLUDE UART
      else if (uart_sel) read_data <= uart_data;</pre>
`endif
`ifdef INCLUDE_TIME
      else if (time_sel) read_data <= time_data;</pre>
`endif
`ifdef INCLUDE PWM
      else if (pwm_sel)
                            read_data <= pwm_data;</pre>
`endif
`ifdef INCLUDE MULT
      else if (mult_sel) read_data <= mult_data;</pre>
`ifdef INCLUDE_DIV
      else if (div_sel)
                            read_data <= div_data;</pre>
`endif
`ifdef INCLUDE_SPI
      else if (spi_sel)
                            read_data <= spi_data;</pre>
`endif
`ifdef INCLUDE_GPIO
      else if (gpio_sel)
                           read_data <= gpio_data;
`endif
`ifdef INCLUDE_WS
      else if (ws sel)
                            read data <= ws data;
`endif
      else
        read data <= 32'h0;
    end
  end
```

Datei: rtl/peripherals/seq\_multiplier.v

Modul: seq\_multiplier

endmodule

Kurzbeschreibung: Sequentieller Multiplikator.

Dieses Modul implementiert eine sequentielle Multiplikation zweier 32-Bit-Werte. Die Multiplikation erfolgt durch aufeinanderfolgende Additionen basierend auf den Bits des Multiplikators. Das Modul ist speicherabbildbasiert und kann über Adressen gesteuert werden: - MUL1\_OFFSET: Erster Multiplikand. - MUL2\_OFFSET: Zweiter Multiplikator (startet die Berechnung). - RESH\_OFFSET: Höhere 32 Bits des Ergebnisses. - INFO\_OFFSET: Gibt an, ob die Berechnung noch läuft (busy).

## Lokale Parameter:

- INFO\_OFFSET Adresse für den Status (busy-Bit).
- MUL1\_OFFSET Adresse für den ersten Multiplikanden.
- MUL2\_OFFSET Adresse für den zweiten Multiplikator (startet Berechnung).
- RESH\_OFFSET Adresse für die höheren 32 Bit des Ergebnisses.
- RESL\_OFFSET Adresse für die niedrigeren 32 Bit des Ergebnisses.

## Eingänge:

- clk Systemtakt.
- rst n Aktiv-low Reset.
- address Speicheradresse für den Zugriff auf Register.
- write\_data Daten, die in die ausgewählten Register geschrieben werden sollen.
- we Schreibaktivierungssignal (Write-Enable).
- re Leseaktivierungssignal (Read-Enable).

## Ausgänge:

• read\_data Zu lesende Daten basierend auf der Adresse.

```
Source Code
```

```
`default_nettype none
`timescale 1ns / 1ns
module seq_multiplier (
  input wire clk,
  input wire
              rst_n,
  input wire [ 7:0] address,
  input wire [31:0] write_data,
  output wire [31:0] read_data,
  input wire
                    we,
  input wire
                     re
);
  localparam INFO_OFFSET = 8'h00;
  localparam MUL1_OFFSET = 8'h04;
  localparam MUL2_OFFSET = 8'h08;
  localparam RESH_OFFSET = 8'hOC;
  localparam RESL_OFFSET = 8'h10;
  reg [31:0] multiplicand;
  reg [31:0] multiplier;
  reg [63:0] product;
  reg [ 5:0] bit_index;
            busy;
  reg
  assign read_data = (address == INFO_OFFSET) ? {31'd0, busy} :
                     (address == MUL1_OFFSET) ? multiplicand
                     (address == MUL2_OFFSET) ? multiplier
                     (address == RESH_OFFSET) ? product[63:32]:
                     (address == RESL_OFFSET) ? product[31: 0]:
                     32'd0;
  always @(posedge clk or negedge rst_n)
  begin
    if (!rst_n) begin
     multiplicand <= 32'd0;
     multiplier <= 32'd0;</pre>
      product <= 64'd0;</pre>
     bit_index <= 6'd0;</pre>
                 <= 1'b0;
      busy
    end
```

```
else
  begin
    if (we)
    begin
      case (address)
        MUL1_OFFSET: begin
          multiplicand <= write_data;</pre>
         end
        MUL2_OFFSET: begin
           multiplier <= write_data;</pre>
           product <= 64'd0;</pre>
          bit_index <= 6'd31;</pre>
           busy
                   <= 1'b1;
         end
      endcase
    end
    if (busy)
    begin
      if (multiplier[bit_index])
        product <= product + ((64'd1 << bit_index) * multiplicand);</pre>
      end
      if (bit_index == 0)
      begin
        busy <= 1'b0;
      \quad \text{end} \quad
      else
      begin
        bit_index <= bit_index - 1;</pre>
    end
  end
end
```

endmodule