

# Computer Architecture MP3 Report

Chang Jun Park

March 9, 2025

## 1 Introduction

This report describes the design and operation of a pipelined sine wave generator using a quarter-sine lookup table (LUT) approach on an FPGA. By storing only 128 samples corresponding to the first quarter of the sine wave and performing mirroring and inversion, the design reconstructs the full 512-sample (9-bit phase) sine wave with minimal memory usage. The quarter-sine data is read from a small memory module and transformed in real time to produce a 10-bit output.

## 2 Design Overview

The design includes two main modules:

1. **Top-level (top.sv)**: Manages the 9-bit phase counter, calculates addresses for the quarter-sine LUT, inverts the samples as needed for the 3rd/4th quarter, and drives the 10-bit output.
2. **Memory (memory.sv)**: Stores 128 samples of a quarter sine wave in a synchronous memory array. Data is loaded at synthesis or simulation time using `$readmemh`.

## 3 Circuit Operation

### 3.1 Phase Counter and Quarters

A 9-bit counter (`counter`) increments every clock cycle, giving a 512-step cycle for the sine wave. The two most significant bits `counter[8:7]` identify which quarter of the sine wave is currently in use:

|                          |
|--------------------------|
| 00 → Quarter 1 (0–127)   |
| 01 → Quarter 2 (128–255) |
| 10 → Quarter 3 (256–383) |
| 11 → Quarter 4 (384–511) |

### 3.2 Memory Address Generation

To minimize storage, only the first quarter of the sine wave (128 samples) is stored in `quarter_sine.txt`. The `mem_address` logic reuses those samples for quarters 2, 3, and 4 by selectively mirroring:

---

```
1 Quarter 1 (00): mem_address = counter[6:0]
2 Quarter 2 (01): mem_address = 127 - counter[6:0]
3 Quarter 3 (10): mem_address = counter[6:0]
4 Quarter 4 (11): mem_address = 127 - counter[6:0]
```

---

This gives the correct “shapes” for the first and second quarters by reversing addresses for the second. The identical logic applies for the third and fourth quarters.

### 3.3 Data Inversion

For the third and fourth quarters, the sine wave goes negative. Instead of storing negative values, the design inverts the data in real time:

$$\text{output\_data} = \begin{cases} \text{mem\_data}, & \text{for quarters 0 and 1 (00, 01),} \\ (1023 - \text{mem\_data}), & \text{for quarters 2 and 3 (10, 11).} \end{cases}$$

Because each sample is 10 bits, 1023 (decimal 10'd1023) is the maximum value, so `1023 - mem_data` provides a symmetric inversion.

### 3.4 Output Pins

The final 10-bit value is mapped to output pins in a specific bit order:

$$\{-48b, -45a, -49a, -3b, -5a, -0a, -2a, -4a, -6a, -9b\} = \text{output\_data}$$

## 4 Simulation Results



Figure 1: Sine Wave Simulation in GTKWave

The green digital signals (top) show the 10 output pins, while the red analog trace (bottom) is a digital-to-analog plot of the same 10-bit output. We can see how the amplitude follows the sine curve through each quarter-cycle, reflecting our read-and-inversion logic.

## 5 Oscilloscope Test



Figure 2: Testing Setup



Figure 3: Oscilloscope Output of Sine Wave

## 6 Source Code

### 6.1 Top-level Module (top.sv)

---

```
1 `include "memory.sv"
2
3 // MP3 top level module - Sine Wave Generator
4
5 module top(
6     input logic    clk,
7     output logic   _9b,      // D0
8     output logic   _6a,      // D1
9     output logic   _4a,      // D2
10    output logic   _2a,      // D3
11    output logic   _0a,      // D4
12    output logic   _5a,      // D5
13    output logic   _3b,      // D6
14    output logic   _49a,     // D7
15    output logic   _45a,     // D8
16    output logic   _48b      // D9
17 );
18 // 9 bits for 512 positions of the sine wave
19 logic [8:0] counter = 0;
20
21 logic [6:0] mem_address;
22 logic [9:0] mem_data;
23 logic [9:0] output_data;
24
25 always_ff @(posedge clk) begin
26     counter <= counter + 1;
27 end
28
29 always_ff @(posedge clk) begin
30     case (counter[8:7])
31         2'b00: mem_address <= counter[6:0];           // Quarter 1
32         2'b01: mem_address <= 7'd127 - counter[6:0]; // Quarter 2
33         2'b10: mem_address <= counter[6:0];           // Quarter 3
34         2'b11: mem_address <= 7'd127 - counter[6:0]; // Quarter 4
35     endcase
36 end
37
38 memory #(
39     .INIT_FILE("quarter_sine.txt")
40 ) sine_mem (
41     .clk          (clk),
42     .read_address (mem_address),
43     .read_data    (mem_data)
44 );
45
46 always_ff @(posedge clk) begin
47     case (counter[8:7])
48         2'b00: output_data <= mem_data;
49         2'b01: output_data <= mem_data;
50         2'b10: output_data <= 10'd1023 - mem_data;
51         2'b11: output_data <= 10'd1023 - mem_data;
52     endcase
53 end
54
55 assign {_48b, _45a, _49a, _3b, _5a, _0a, _2a, _4a, _6a, _9b} = output_data;
56 endmodule
```

---

## 6.2 Memory Module (memory.sv)

---

```
1 // Memory
2 module memory #(
3     parameter INIT_FILE = ""
4 )(
5     input logic      clk,
6     input logic [6:0] read_address,
7     output logic [9:0] read_data
8 );
9
10 // store 128 10-bit samples (first quarter of sine)
11 logic [9:0] sample_memory [0:127];
12
13 initial if (INIT_FILE) begin
14     $readmemh(INIT_FILE, sample_memory);
15 end
16
17 always_ff @(posedge clk) begin
18     read_data <= sample_memory[read_address];
19 end
20
21 endmodule
```

---