

# Documentație tema: Generator semnal PWM

Atanasiu Ștefan, Piloiu Raul-Ispas, Stoica Lorena-Elena

26 noiembrie 2025

## Rezumat

Acest document prezintă detaliile de implementare pentru un generator de semnal PWM (Pulse Width Modulation) controlat printr-o interfață serială SPI. Arhitectura este modulară, separând logica de comunicație, decodificarea instrucțiunilor, stocarea configurației și generarea efectivă a semnalului. Documentația se concentrează pe soluțiile tehnice adoptate pentru a asigura sincronizarea între domenii de ceas, gestionarea stărilor și manipularea semnalelor digitale.

## Cuprins

|          |                                                            |          |
|----------|------------------------------------------------------------|----------|
| <b>1</b> | <b>Introducere</b>                                         | <b>2</b> |
| <b>2</b> | <b>Descrierea Implementării Modulelor</b>                  | <b>2</b> |
| 2.1      | SPI Bridge ( <code>spi_bridge.v</code> ) . . . . .         | 2        |
| 2.1.1    | Sincronizarea Domeniilor de Ceas (CDC) . . . . .           | 2        |
| 2.1.2    | Logica de Transfer . . . . .                               | 2        |
| 2.2      | Instruction Decoder ( <code>instr_dcd.v</code> ) . . . . . | 3        |
| 2.2.1    | Particularități ale FSM . . . . .                          | 3        |
| 2.3      | Blocul de Rești (regs.v) . . . . .                         | 3        |
| 2.3.1    | Mecanismul Self-Clearing pentru Reset . . . . .            | 3        |
| 2.4      | Numărătorul (counter.v) . . . . .                          | 3        |
| 2.4.1    | Logica de Prescalare . . . . .                             | 3        |
| 2.4.2    | Numărare Bidirecțională . . . . .                          | 3        |
| 2.5      | Generatorul PWM ( <code>pwm_gen.v</code> ) . . . . .       | 4        |
| 2.5.1    | Moduri de Funcționare . . . . .                            | 4        |
| <b>3</b> | <b>Concluzii</b>                                           | <b>4</b> |

# 1 Introducere

Proiectul implementează un periferic hardware descris în limbajul Verilog, capabil să genereze semnale PWM configurabile. Sistemul este împărțit în cinci module principale interconectate prin modulul de top (`top.v`):

- **SPI Bridge:** Gestionează comunicația fizică serială.
- **Instruction Decoder:** Interpretează comenzi primite.
- **Registers:** Stochează configurația sistemului.
- **Counter:** Asigură baza de timp.
- **PWM Generator:** Formează semnalul de ieșire.

## 2 Descrierea Implementării Modulelor

### 2.1 SPI Bridge (`spi_bridge.v`)

Interfața SPI a fost implementată ținând cont de faptul că ceasul perifericului (`clk`) și ceasul SPI (`sclk`) sunt asincrone sau pot avea faze diferite.

#### 2.1.1 Sincronizarea Domeniilor de Ceas (CDC)

Pentru a evita problemele de setup/hold time, nu am utilizat `sclk` direct ca ceas pentru registrele interne. În schimb, am implementat un mecanism de *oversampling* și detectie de fronturi folosind ceasul sistemului (`clk`):

```
1 // Sincronizare double-flop
2 always @(posedge clk or negedge rst_n) begin
3     if (!rst_n) begin
4         sclk_sync_r1 <= 1'b0;
5         sclk_sync_r2 <= 1'b0;
6     end else begin
7         sclk_sync_r1 <= sclk;
8         sclk_sync_r2 <= sclk_sync_r1;
9     end
10 end
```

Astfel, generăm semnale interne `sclk_posedge` și `sclk_negedge` care sunt pulsuri de durată unui ciclu de `clk`. Aceasta garantează că toată logica internă rămâne sincronă cu `clk`.

#### 2.1.2 Logica de Transfer

Protocolul este implementat astfel:

- **Recepție (MOSI):** Datele sunt eșantionate pe frontul descrescător detectat al SCLK.
- **Transmisie (MISO):** Datele sunt schimbatе pe frontul crescător detectat al SCLK.

La finalul a 8 biți transferați, se generează semnalul `byte_sync`, care activează decodorul de instrucțiuni.

## 2.2 Instruction Decoder (instr\_dcd.v)

Decodorul funcționează ca o mașină de stări finite (FSM) cu două stări principale: S\_INSTR (faza de comandă) și S\_DATA (faza de date).

### 2.2.1 Particularități ale FSM

- **Anticiparea Citirii:** În starea S\_INSTR, dacă bitul de R/W indică o citire, semnalul `read` este activat imediat combinatoric, iar adresa este setată. Acest lucru permite modulului de registri să furnizeze datele valide (`data_read`) înainte de următorul front de ceas SPI, asigurând că `spi_bridge` are datele corecte pregătite pentru transmisie în faza S\_DATA.
- **Adresarea:** Se extrage adresa de bază din instrucțiune, iar logica internă decide dacă acceseează octetul superior sau inferior în funcție de bitul 6 al comenzi.

## 2.3 Blocul de Regiștri (regs.v)

Acest modul gestionează stocarea parametrilor. O particularitate importantă a implementării este gestionarea resetării software a contorului.

### 2.3.1 Mecanismul Self-Clearing pentru Reset

Registrul COUNTER\_RESET (adresa 0x07) este de tip "write-only" și declanșează o acțiune, nu o stare persistentă. Am implementat un mecanism de auto-curățare:

```
1 if (write && addr == 6'h07) begin
2     count_reset_pulse_r <= 1'b1;
3 end else if (count_reset_pulse_r == 1'b1) begin
4     count_reset_pulse_r <= 1'b0;
5 end
```

Această logică asigură că semnalul de reset către contor este activ timp de exact un ciclu de ceas, prevenind blocarea contorului în starea de reset dacă software-ul nu scrie explicit '0' înapoi.

## 2.4 Numărătorul (counter.v)

Numărătorul este inima sistemului și include un etaj de prescalare pentru divizarea frecvenței.

### 2.4.1 Logica de Prescalare

În loc să folosim un ceas divizat (ceea ce nu este recomandat în FPGA/ASIC), am folosit un semnal de "Enable" derivat intern. Variabila `prescaler_count` incrementează la fiecare ciclu de ceas. Doar când aceasta atinge valoarea din registrul `prescale`, contorul principal (`count_val`) este actualizat. Formula implicită este:  $T_{update} = T_{clk} \times (PRESCALE + 1)$ .

### 2.4.2 Numărare Bidirectională

S-a implementat logica de *overflow* și *underflow*:

- **Up:** Când atinge valoarea `period`, se resetează la 0.
- **Down:** Când atinge 0, se încarcă cu valoarea `period`.

## 2.5 Generatorul PWM (pwm\_gen.v)

Acest modul este pur combinațional la nivel logic, dar ieșirea este înregistrată (registered output) pentru a evita glitch-urile pe linia de ieșire.

### 2.5.1 Moduri de Funcționare

Implementarea folosește biții din registrul `functions` pentru a multiplexa comportamentul:

- **Mod Nealiniat (Standard):** Compara `count_val` doar cu `compare1`. Ieșirea comută când contorul depășește pragul. Polaritatea este decisă de bitul de aliniere.
- **Mod Dual-Compare:** Ignoră biții de aliniere clasice și generează pulsul activ strict în intervalul [`compare1`, `compare2`). Această implementare permite generarea unor pulsuri poziționate arbitrar în cadrul perioadei, nu doar la începutul sau sfârșitul acesteia.

## 3 Concluzii

Soluția propusă respectă arhitectura impusă, aducând contribuții specifice în zona de robustețe a semnalelor (sincronizarea SPI cu ceasul sistemului) și ușurință în utilizare (reset-ul contorului cu auto-clear). Codul este modular, parametrizabil prin registri și pregătit pentru integrarea într-un sistem embedded mai larg.