

# **SoundPlayer: Lettore walkman di file Wav da scheda SD**

Progettazione di Sistemi Digitali

*Luca Ceragioli, Marco Ferrini*

*Anno Accademico 2022/2023*



# Indice

|                                           |           |
|-------------------------------------------|-----------|
| <b>Indice.....</b>                        | <b>1</b>  |
| <b>Introduzione.....</b>                  | <b>2</b>  |
| Caratteristiche principali.....           | 2         |
| Sfide tecniche.....                       | 2         |
| Quick start guide.....                    | 2         |
| <b>Dettagli implementativi.....</b>       | <b>3</b>  |
| Schede SD.....                            | 3         |
| File system FAT32.....                    | 3         |
| File Audio WAV.....                       | 3         |
| Buffer Audio.....                         | 4         |
| Debug con Logic Analyzer.....             | 4         |
| <b>Architettura.....</b>                  | <b>5</b>  |
| Schema a Blocchi.....                     | 5         |
| SDcard Reader.....                        | 6         |
| SD Card main fsm.....                     | 8         |
| SD Card cmd fsm.....                      | 11        |
| SPI master.....                           | 13        |
| FAT32 Reader.....                         | 15        |
| RAM dualport.....                         | 18        |
| Codec Interface.....                      | 19        |
| Codec Config.....                         | 21        |
| I2S Master.....                           | 22        |
| Main PLL.....                             | 23        |
| Codec audio WM8731.....                   | 23        |
| <b>Debug e funzionamento.....</b>         | <b>24</b> |
| Protocollo SPI scheda SD.....             | 24        |
| Protocollo I2C codec.....                 | 24        |
| Protocollo I2S codec.....                 | 25        |
| Lettura di un blocco della scheda SD..... | 25        |
| Lettura di un cluster FAT32.....          | 26        |
| Funzionamento a regime.....               | 27        |
| <b>Conclusioni.....</b>                   | <b>28</b> |

# Introduzione

Il progetto consiste nel prototipo di un riproduttore portatile di file musicali, comunemente chiamato “walkman”. I file audio sono memorizzati su scheda SD, utilizzando la scheda di sviluppo TERASIC DE2. Il lettore permette di ascoltare brani audio, di metterli in pausa e di saltare da un brano all’altro scorrendoli in ordine.

- Link ai **sorgenti** del progetto: <https://github.com/VirtuContraFurore/SoundPlayer>
- Video dimostrativo: [https://www.youtube.com/watch?v=giQW\\_ChssqA](https://www.youtube.com/watch?v=giQW_ChssqA)

## Caratteristiche principali:

- Compatibile con schede SDHC o SDXC formattate FAT32
- Riproduce file audio di tipo WAV a 44.1 kHz sia mono che stereo
- Uscita audio su jack stereo da 3.5 mm
- Bottone “**Play/Pause**”
- Bottone “**Next Track**”
- Bottone “**Restart Track/Previous Track**”<sup>1</sup>
- Permette di navigare fino a 80 brani diversi



## Sfide tecniche:

- Implementare i protocolli di comunicazione con periferiche **I2C**, **I2S** e **SPI**
- Realizzare la sequenza di inizializzazione di schede SD di nuova generazione
- Rilevamento e lettura di una partizione FAT32 valida
- Lettura dei file attraverso la *File Allocation Table* del file system **FAT32**
- Sincronizzazione del **buffer audio** con la lettura del file dalla scheda SD
- Debug real-time attraverso l’uso di un **logic analyzer**

## Quick start guide

1. Formattare una scheda SD da almeno 4GB col file system FAT32, scegliendo, se possibile, *allocation unit* di 8 kbytes o superiore.
2. Ottenere dei file audio WAV. Se si possiede il file in un altro formato, per convertirlo si può usare lo script allegato al progetto, che sfrutta il programma open source `ffmpeg` per ottenere file audio compatibili.<sup>2</sup>
3. Copiare i file nella scheda SD. Non è necessario riformattare la scheda ogni volta che si aggiunge o rimuove un file: questo perchè il file system viene effettivamente letto e non si sfrutta l’allocazione contigua tipica delle partizioni vergini.
4. Inserire la scheda SD, programmare la board DE2 e collegare delle cuffie all’uscita audio!

<sup>1</sup> Se premuto dopo 5 secondi dall’inizio della traccia fa ripartire la stessa canzone, altrimenti, se premuto entro 5 secondi dall’inizio della traccia, va alla canzone precedente.

<sup>2</sup> File compatibili: file con estensione WAV, struttura RIFF, encoding pcm16s con sampling frequency di 44.1 kHz, traccia mono o stereo.

# Dettagli implementativi

## Schede SD

Non sono supportate le schede di vecchia generazione (quelle con capacità minore od uguale a 2GB) in quanto la sequenza di startup legacy non è stata implementata. Le schede SD più recenti, ossia quelle SDHC o SDXC hanno una nuova sequenza di inizializzazione come riportato dalle specifiche *Physical Layer* disponibili sul sito <https://www.sdcards.org>.

La lettura delle schede SD avviene tramite interfaccia SPI e con blocchi di 512 bytes. Da quando viene richiesto un blocco a quando esso viene mandato al master passa un certo intervallo di tempo; per questo è meglio, quando possibile, leggere blocchi contigui: in questo caso il tempo di attesa è nullo.

Si riportano alcune risorse che di fatto riassumono le specifiche e ne agevolano la comprensione:

1. [http://elm-chan.org/docs/mmc/mmc\\_e.html](http://elm-chan.org/docs/mmc/mmc_e.html)
2. <http://chlazza.nfshost.com/sdcardinfo.html>
3. <https://www.pjrc.com/tech/8051/ide/fat32.html>

## File system FAT32

Probabilmente il punto centrale di tutto il progetto, dato che, assieme alle caratteristiche di lettura della scheda SD, definisce le modalità e le tempistiche per accedere ai file al suo interno, il file system FAT32 è stato scelto in quanto diffusissimo e largamente supportato da diversi sistemi operativi. Va notato che una partizione FAT32 può essere grande al massimo 32 GB, per questo alcuni programmi potrebbero rifiutarsi di formattare una scheda SD più grande con questo tipo di file system. Il problema è facilmente risolvibile accettando di avere parte della scheda inutilizzata – anche se, nella parte inutilizzata, vi si può mettere agilmente un'altra partizione.

Brevemente, il file system FAT32 possiede una tabella - detta ‘allocation table’, appunto - che permette di frammentare lo spazio di archiviazione in *cluster* di dimensione fissata (raccomandati 8 kbyte o superiore) il quale ordine di lettura viene ricostruito leggendo le catene di cluster dalla tabella stessa.

Il nostro progetto riproduce solo i file WAV contenuti della cartella “root” della scheda SD<sup>3</sup>. Per i dettagli sul sistema FAT32 si rimanda alla pagina:

[https://it.wikipedia.org/wiki/File\\_Allocation\\_Table](https://it.wikipedia.org/wiki/File_Allocation_Table)

## File Audio WAV

I file audio supportati devono avere le seguenti caratteristiche:

- Container di tipo “RIFF”
- Nessun metadato
- Codifica pcm16s

<sup>3</sup> In realtà, riproduce solo i file contenuti nel primo cluster della directory “root”, che corrispondono ad un massimo di circa 80 canzoni. La lettura dei cluster successivi al primo è implementabile estendendo la macchina a stati di lettura dei file; per i nostri scopi abbiamo ritenuto 80 canzoni sufficienti a dimostrare la funzionalità del progetto.

- Sampling rate 44.1 kHz
- Canali stereo o mono

Non tutti i file di tipo “WAV” vanno bene, in quanto ci sono diversi container che possono ospitare i campioni in codifica pcm. Si raccomanda di usare lo script “convert\_audio\_file.sh” allegato al progetto per avere un file compatibile. È uno script bash, quindi occorre avere una shell UNIX. Su windows è possibile usare *Cygwin* o *WSL* per avere una shell Linux e installare il programma *ffmpeg*.

Risorse utili:

- <http://soundfile.sapp.org/doc/WaveFormat/>

## Buffer Audio

Viene impiegata la tecnica del *double buffering*: sono presenti due buffer di ugual dimensione: mentre uno viene riprodotto sull’uscita audio, l’altro viene riempito con i dati letti dalla scheda SD. È importante notare che, data la lettura a blocchi da 512 bytes della scheda SD, il buffer dovrà essere grande quanto un multiplo intero della dimensione del blocco di lettura. La memoria del buffer sfrutta i blocchi di tipo M4K disponibili dentro lo FPGA Altera.

## Debug con Logic Analyzer

È stato utilizzato un fondamentale strumento in grado di farci vedere le forme d’onda digitali sui pin dello FPGA. Sono stati connessi, attraverso il verilog, i segnali di interesse ai GPIO della board DE2. In questo modo è stato possibile visualizzare ogni protocollo e controllarne il corretto funzionamento.

Lo strumento è un Saleae Logic a 8 canali. Con interfaccia USB, si adopera col programma messo a disposizione dall’azienda.

Sotto, in figura: analizzatore logico e cattura di segnali col tool *Logic2*.



# Architettura

In questa sezione viene spiegata l'architettura di alto livello del progetto e, nelle sottosezioni seguenti, viene descritto ciascun sottoblocco.

## Schema a Blocchi



In figura è illustrato lo schema a blocchi generale del sistema, ponendo l'attenzione sul flusso di dati:

- **SD Reader:** Preleva i dati dal lettore di schede SD attraverso il protocollo SPI.
- **FAT32 Reader:** Riceve i dati da SD Reader guidandolo nel file system della scheda SD.
- **Buffer:** Memoria RAM dual-channel, permette la sincronizzazione delle parte destra dello schema con la parte sinistra.
- **Codec Interface:** wrapper che contiene tutti i moduli responsabili della comunicazione con il Codec.
  - I2S module: Preleva i dati dal buffer e li invia al codec secondo il protocollo I2S.
  - Codec Configurator (I2C): Configura il Codec audio tramite il protocollo I2C.
- **Codec Audio WM8731:** Riceve il flusso di dati digitali dal blocco I2S e genera un segnale analogico per le cuffie.

Nel file Verilog `globals.v` ci sono le configurazioni del progetto, ossia:

- frequenze del clock globale e di ogni protocollo usato
- mapping dei bottoni alle loro funzioni

in questa maniera è possibile modificare agevolmente le frequenze di lavoro dei vari blocchi senza dover aprire file differenti.

## SDcard Reader

SDcard Reader gestisce l'inizializzazione e la comunicazione con la scheda SD.

Esso è costituito in realtà da 3 sottomoduli ovvero:

- SDCard\_main\_fsm: macchina a stati principale che gestisce inizializzazione e comunicazione con la scheda SD.
- SDCard\_cmd\_fsm: macchina a stati che sia durante l'inizializzazione che poi nella successiva comunicazione con la scheda SD, decodifica le informazioni riguardanti i comandi da inviare alla scheda SD e ne gestisce la trasmissione.
- SPI\_master: macchina a stati finiti che gestisce la comunicazione SPI con la scheda SD implementando il physical layer del protocollo.

Iniziamo un'analisi approfondita del modulo in questione partendo dal suo pinout, riferimento figura 3.2.1.

Input:

- **clk, rst\_n**: clock e reset attivo basso.
- **block\_read\_block\_addr[31:0]**: bus a 32 bit per specificare l'indirizzo del blocco da leggere nel file system della scheda SD.
- **block\_read\_trigger**: segnale che avvia la lettura dei blocchi della scheda SD.
- **block\_read\_continuous\_mode**: flag che avvia la modalità di lettura continua della scheda SD.
- **sd\_do**: SPI miso.

Output:

- **sd\_clk**: clock inviato da SPI master alla scheda SD.
- **sd\_di**: SPI mosi
- **sd\_cs\_n**: chip select relativo alla scheda SD
- **block\_read\_data\_new\_flag**: flag che si attiva quando viene ricevuto un nuovo byte relativo ad un blocco.
- **card\_configured**: segnale che viene messo ad 1 quando il processo di inizializzazione, della scheda SD è terminato.
- **block\_read\_card\_ready**: segnale che viene attivato quando SD card reader è pronto a leggere un nuovo blocco.
- **block\_read\_data\_out[7:0]**: Bus che trasmette i dati ricevuti dalla scheda al FAT32\_reader.
- **block\_read\_data\_idx[8:0]**: Contatore che indica il numero di byte ricevuti relativi ad un blocco.

Procediamo con l'analisi dei sottomoduli.



Figura 3.2.1: Schema a blocchi, SDCard Reader.

*Table 3.*

## SD Card main fsm

Analisi del pinout del modulo.

Input:

- **clk, rst\_n**: clock e reset attivo basso.
- **block\_read\_trigger**: vedi pinout SD reader
- **block\_read\_continuos\_mode**: vedi pinout SD reader
- **block\_read\_block\_addr**: vedi pinout SD reader
- **cmd\_ready**: segnale di uscita della SD Card cmd fsm che viene attivato quando quest'ultima è nello stato di riposo e dunque quando è pronta a decodificare un comando
- **spi\_ready**: segnale di uscita dell'SPI master che viene attivato quando quest'ultimo non è impegnato in trasmissione/ricezione.
- **spi\_rx\_data**: bus dei dati ricevuti da SPI master
- **cmd\_response\_bytes**: bus dati in uscita dall'SD Card cmd fsm in cui viene riportata la risposta ad un comando inviato, ricevuta tramite la periferica SPI.

Output:

- **block\_read\_card\_ready**: vedi pinout SD reader
- **sd\_cs\_n**: vedi pinout SD\_reader
- **spi\_req**: segnale con il quale viene richiesto l'ultimo dell'interfaccia SPI.
- **cmd\_req\_idx**: bus dati in cui viene riportato il codice identificativo del comando da inviare
- **block\_read\_data\_idx**: registro di uscita in cui viene riportato il numero di bytes inviati relativi al blocco in lettura
- **cmd\_block\_addr**: registro di uscita in cui viene riportato l'indirizzo del blocco della scheda SD da leggere ovvero block\_read\_block\_addr.
- **block\_read\_data\_out[7:0]**: vedi pinout SD reader
- **card\_configured**: vedi pinout SD reader
- **block\_read\_data\_new\_flag**: vedi pinout SD reader
- **spi\_prescaler**: bus in cui viene indicato il codice che identifica il fattore di divisione in frequenza per ottenere il clock di SPI master a partire dal master clock

La macchina a stati è organizzata in un unico blocco always e gli stati possono essere divisi in due macrogruppi:

- Relativi all'inizializzazione della scheda
- Relativi alla lettura dei blocchi della scheda.

Nel processo di inizializzazione lo stato di partenza è FSM\_CARD\_NOT\_READY, stato in cui si approda a seguito di reset sincrono, qui viene settato cmd\_req\_idx con il codice relativo al comando NOCMD, ovvero un comando fittizio che serve a mantenere la macchina a stati SD Card cmd fsm nello stato di idle. Viene impostato spi\_prescaler in modo che, durante la fase di inizializzazione, il clock di SPI master abbia l'opportuna frequenza. Infine si passa allo stato successivo: FSM\_CONFIGURING\_CARD\_1.

Questo è uno stato di setup, si disattiva sd\_cs\_n, si setta spi\_req e successivamente si aspetta che il SPI master risponda ad spi\_req per passare allo stato successivo: FSM\_CONFIGURING\_CARD\_2.

In FSM\_CONFIGURING\_CARD\_2 vengono inviati 80 impulsi di clock con il CS disattivato, per permettere il reset della scheda successivamente viene attivato sd\_cs\_n.

Successivamente inizia la configurazione vera e propria della scheda SD con il susseguirsi di 5 stati, (da FSM\_CONFIGURING\_CARD\_3 a FSM\_CONFIGURING\_CARD\_7) nel quale vengono inviati opportuni comandi alla scheda SD. Ciò viene ottenuto settando opportunamente cmd\_req\_idx e passando da due stadi (FSM\_RUN\_CMD\_0 e FSM\_RUN\_CMD\_1) nel quale si aspetta l'esecuzione del comando.

I 5 comandi inviati sono, in ordine: CMD0, CMD8, CMD58, CMD55, ACMD41, per capirne la funzione si faccia riferimento alla sezione relativa alle schede SD a pagine 3.

Terminato questa prima fase di configurazione, si approda nello stato FSM\_CONFIGURING\_CARD\_8, dove viene valutato il contenuto del LSB del registro R0, nel quale viene posto il segnale di ingresso costituito da cmd\_response\_bytes. Se il bit è nullo si può procedere con la configurazione altrimenti vengono rinvolti i comandi CMD55, e ACMD41.

Propriamente l'inizializzazione della scheda SD termina con lo stato FSM\_CONFIGURING\_CARD\_9 nel quale, in modo analogo a prima, viene inviato il comando CMD58.

Infine la macchina evolve nello stato FSM\_CONFIGURING\_CARD\_10 nel quale viene settato card\_configured, e viene impostato il valore di spi\_prescaler opportuno per quando la scheda è configurata.

Con l'evoluzione verso lo stato FSM\_CARD\_READY la macchina a stati inizia ad occuparsi della lettura del contenuto della scheda SD. Questo è lo stato da cui inizia la lettura di un nuovo blocco, infatti viene azzerato il contenuto di byte counter, contatore usato per tenere traccia del numero di bytes inviati relativi al blocco in lettura, e si attende l'attivazione di block\_read\_trigger. Giunta quest'ultima, si pone in cmd\_block\_addr l'indirizzo del blocco dati da leggere e si invia, dipendentemente dal valore di block\_read\_continuos\_mode, il comando CMD18 o CMD17 che impongono rispettivamente lettura di più blocchi dati o di blocchi singoli.

Nello stato FSM\_CARD\_BUSY\_0 la macchina valuta la corretta esecuzione dell'ultimo comando inviato, per poi eventualmente passare al successivo.

Negli stati FSM\_CARD\_BUSY\_1 e FSM\_CARD\_BUSY\_2 la macchina a stati attiva il flag spi\_reg e, una volta che l'interfaccia SPI è pronta, aspetta che sul bus dati in ingresso siano presenti dati validi, per poi evolvere nello successivo stato.

FSM\_CARD\_BUSY\_3 è uno stato di attesa.

Iniziano ora una serie di stati dediti alla gestione dei dati in ingresso partendo da FSM\_CARD\_BUSY\_4, nel quale i bytes in ingresso (spi\_rx\_data) vengono riportati sul bus di uscita block\_read\_data\_out, viene assegnato il contenuto di byte\_counter a block\_read\_data\_idx per poi essere incrementato. Nel frattempo viene settato il flag block\_read\_data\_new\_flag. Infine abbiamo uno statement condizionale che discrimina il caso in cui la ricezione del blocco dati è incompleta, dunque si torna allo stato

FSM\_CARD\_BUSY\_3 per ricevere i bytes successivi, e il caso in cui la ricezione è completa, in quel caso la macchina evolve verso gli stati FSM\_CARD\_BUSY\_CRC\_1 e FSM\_CARD\_BUSY\_CRC\_2 in cui gestisce la ricezione dei due byte CRC.

Nello stato FSM\_CARD\_BUSY\_CRC\_3 abbiamo un nested conditional statement per determinare l'evoluzione della macchina, in cui se è non stata selezionata la continuos read mode si torna nello stato FMS\_CARD\_READY in cui si aspetta un nuovo segnale di trigger per la lettura di un nuovo blocco dati. Se invece la suddetta modalità di funzionamento è stata selezionata, se è richiesta la lettura di un nuovo blocco dati si torna allo stato FSM\_CARD\_BUSY\_1 procedendo con la lettura del blocco contiguo o, viceversa, la macchina evolve verso lo stato FSM\_CARD\_BUSY\_STOP\_1.

Gli stati FSM\_CARD\_BUSY\_STOP\_1, FSM\_CARD\_BUSY\_STOP\_2, FSM\_CARD\_BUSY\_STOP\_3 sono dediti all'interruzione della modalità di lettura continua, la macchina, infatti, invia il relativo comando CMD12, e aspetta l'opportuna risposta da parte della scheda SD per poi evolvere in FSM\_CARD\_READY.

## SD Card cmd fsm

Analisi pinout del modulo:

Input:

- **clk, rst\_n**: clock e reset sincrono
- **cmd\_req\_idx**: bus di ingresso nel quale viene riportato il codice relativo al comando da inviare.
- **spi\_ready**: vedi pinout SD Card main fsm.
- **spi\_rx\_data**: vedi pinout SD Card main fsm.
- **cmd\_block\_addr**: bus in cui viene riportato l'indirizzo del blocco dati da leggere.

Output:

- **cmd\_ready**: flag con il quale la macchina a stati segnala che è nello stato di riposo e quindi pronta a ricevere nuove istruzioni.
- **cmd\_response\_bytes**: bus multi byte nel quale viene riportata la risposta della scheda ad un comando precedentemente inviato.
- **spi\_tx\_data**: bus dei dati da inviare alla scheda tramite l'interfaccia SPI.
- **spi\_req**: vedi pinout SD Card main fsm.
- **spi\_tx\_en**: flag che abilita la trasmissione dati dell'SPI master.

La macchina a stati è organizzata in due blocchi always entrambi che descrivono logica sincrona e gli stati possono essere divisi in due macrogruppi:

- dediti alla trasmissione dei comandi.
- dediti alla ricezione della risposta da parte della scheda SD.

Il primo blocco always, quando la macchina stati è riposo, decodifica segnale riportato in cmd\_req\_idx inizializzando in modo opportuno la struttura dati del comando da inviare. Quest'ultima è card\_cmd\_packet, ovvero 6 bytes in cui i primi 5 sono dedicati agli argomenti del comando da inviare mentre l'ultimo byte è dedicato al codice di Cyclic Redundancy Check. Inoltre per ciascun comando decodificato viene indicato il numero di bytes della risposta corrispondente e viene attivato il flag pending\_request che segnale la presenza di un comando da inviare.

Il secondo blocco always, invece, gestisce trasmissione dei comandi e ricezione delle risposte.

CMD\_FSM\_IDLE costituisce lo stato di riposo, viene inizializzato cmd\_counter, ovvero un contatore con triplice funzione: tiene traccia del numero di byte che sono stati inviati relativi ad un comando, indica il byte da inviare e segnala il numero di bytes ricevuti relativi alla risposta ad un comando. Nello stato, inoltre, vengono settati a zero spi\_req e spi\_tx\_en. Successivamente si aspetta che sia presente un comando da inviare ed eventualmente si procede verso lo stato CMD\_FSM\_SEND\_CMD\_1.

Questo costituisce uno stato di setup della trasmissione, infatti vengono settati spi\_tx\_en e spi\_req per poi evolvere in CMD\_FSM\_SEND\_CMD\_2 nel quale si ha l'effettiva trasmissione dei dati. Qui, finché non è stato trasmesso l'intero comando viene mantenuto spi\_tx\_en alto e quando spi-ready è alto viene incrementato il numero di bytes inviati e si controlla di non aver terminato la trasmissione per poi eventualmente evolvere nello stato CMD\_FSM\_RECEIVE\_CMD\_RESP\_2.

CMD\_FSM\_RECEIVE\_CMD\_RESP\_2 e CMD\_FSM\_RECEIVE\_CMD\_RESP\_1 costituiscono due stati di attesa nel quale la macchina aspetta di ricevere qualcosa dalla scheda SD per poi evolvere in CMD\_FSM\_RECEIVE\_CMD\_RESP\_3 nel quale effettivamente si gestisce la ricezione. Qui infatti i bytes ricevuti in risposta vengono posti in uscita al blocco tramite il bus cmd\_response\_bytes, quando poi il numero di bytes egualgia il numero di bytes che costituiscono la risposta ad un comando si torna allo stato di riposo CMD\_FSM\_IDLE.

## SPI master

Analisi del pinout del modulo:

Input

- **clk, rst\_n**: clock e reset asincrono attivo basso.
- **spi\_miso**: MISO dell'interfaccia SPI
- **req**: flag che segnala la presenza di una richiesta di utilizzo dell'interfaccia SPI, è controllato, tramite un OR dai due segnali spi\_req delle due macchine a stati del SD Card reader.
- **tx\_en**: segnale che abilita la trasmissione dati.
- **data\_tx**: bus dei dati da trasmettere.
- **prescaler**: bus che indica il modulo del prescaler da utilizzare per ottenere il clock da inviare a SPI slave.

Output:

- **spi\_mosi**: MOSI dell'interfaccia SPI.
- **spi\_clk**: slave clock dell'interfaccia SPI.
- **busy**: segnale che indica quando il modulo non è nello stato di riposo.
- **data\_rx**: bus dei dati ricevuti.
- **\_spi\_rising\_edge**: vedi capitolo debug.
- **\_spi\_clock\_counter**: vedi capitolo debug.
- **\_spi\_sub\_counter**: vedi capitolo debug.
- **\_spi\_done**: vedi capitolo debug.
- **\_spi\_fsm**: vedi capitolo debug.

Inoltre sono presenti dei parametri da settare quando istanziamo il modulo:

- **SPI\_PACKET\_SIZE**: dimensione dei bus data\_rx e data\_tx.
- **SPI\_MOSI\_IDLE**: livello logico di riposo del MOSI.
- **SPI\_CPOL**: livello logico di riposo di SPI clock.
- **SPI\_CPHA**: regola la fase del clock, ovvero il fronte di clock in cui il ricevente campiona il segnale in ingresso.

il modulo è costituito da una funzione e due blocchi always che descrivono logica sincrona.

La funzione opera la decodifica del codice prescaler.

Un blocco always genera il clock per l'interfaccia SPI e i segnali di fronte in salita dello stesso clock. Se il flag interno spi\_clk\_en è attivo, il contatore sub\_counter viene usato come divisore in frequenza del master clock, quando questo è uguale a sub\_counter\_cap, ovvero il fattore di divisione in frequenza decodificato dalla funzione sopra citata, allora viene decrementato il contatore clock\_counter il cui LSB, concordemente a i valori di SPI\_CPOL e SPI\_CPHA, costituisce il clock del SPI. Quando invece spi\_clk\_en è disattivato vengono resettati entrambi i contatori.

Si nota che clock\_counter viene inizializzato con il doppio del valore di SPI\_PACKET\_SIZE in modo tale che quando il contatore si azzerà si è completato trasmissione e ricezione e si può attivare il segnale done e \_spi\_done utile per il debug.

Il secondo blocco always invece costituisce la macchina a stati vera e propria che gestisce trasmissione e ricezione. Qui oltre allo stato di reset abbiamo lo stato di riposo FSM\_IDLE nel quale se è richiesto l'utilizzo del SPI si attiva spi\_clk settando l'opportuno valore del prescaler e se necessario si abilita la trasmissione.

Lo stato FMS\_START gestisce la ricezione dati campionando i bit su spi\_miso nei fronti in salita di spi\_clk che vengono messi nel registro rx\_buffer. Quest'ultimo è un registro di dimensione  $2 \times \text{SPI_PACKET_SIZE}$  così facendo posso scrivere in una metà mentre il byte precedentemente ricevuto è disponibile in uscita tramite il bus data\_rx. Quando poi il flag done è attivo si disabilita spi\_clk si invertono le metà di rx\_buffer destinate a lettura e scrittura e si ritorna in FSM\_IDLE.

## FAT32 Reader

Macchina a stati che gestisce la lettura del file WAV al livello di astrazione più alto. Quest'ultima offre un'interfaccia utente molto semplice attraverso il quale è possibile gestire la riproduzione dei vari file WAV presenti nella SD, e ricevere dei feedback sullo stato di lettura del file.

Analisi del pinout del modulo.

Input:

- **clk, rst\_n**: master clock e reset sincrono attivo basso.
- **block\_read\_card\_ready**: vedi pinout SD card reader.
- **block\_read\_data\_new\_flag**: vedi pinout SD card reader.
- **block\_read\_data\_in**: bus dei dati letti dalla scheda SD.
- **block\_read\_data\_idx**: vedi pinout SD card reader.
- **audio\_buffer\_empty\_i**: Segnale per la sincronizzazione con il modulo Codec che legge dal buffer dati, viene messo a 1 quando il buffer è vuoto.
- **player\_next\_song\_req\_i**: richiesta di riproduzione di una nuova canzone, proveniente dall'interfaccia utente.
- **player\_next\_song\_forward\_i**: segnale che discrimina il caso in cui la nuova canzone riprodotta deve essere la precedente o la successiva nel file system.

Output:

- **block\_read\_trigger**: vedi pinout SD card reader.
- **block\_read\_continous\_mode**: vedi pinout SD card reader.
- **block\_read\_block\_addr**: vedi pinout SD card reader.
- **error\_no\_fat\_found**: segnale di errore attivato quando il tipo di partizione utilizzato nella scheda SD non è FAT32.
- **audio\_buffer\_data\_o**: bus dei dati da scrivere nel buffer.
- **audio\_buffer\_addr\_o**: bus degli indirizzi in scrittura del buffer.
- **audio\_buffer\_wren\_o**: segnale di write enable.
- **audio\_buffer\_filled\_o**: Segnale per la sincronizzazione con il modulo Codec che legge dal buffer dati, viene messo a 1 quando il buffer è pieno.
- **audio\_buffer\_empty\_ack\_o**: segnale di acknowledgement relativo al segnale audio\_buffer\_empty\_i.
- **wav\_info\_sampling\_rate**: bus dati che trasmette le informazioni sul sampling rate del file WAV.
- **wav\_info\_audio\_channels**: bus dati che specifica se il file WAV è mono o stereo.
- **player\_next\_song\_req\_ack\_o**: segnale di acknowledgement relativo al segnale audio\_buffer\_empty\_i.

La macchina a stati è costituita da un unico blocco always che descrive logica sincrona.

Gli stati di quest'ultima possono essere divisi in quattro macro gruppi .

- **FSM\_READ\_MBR**: dediti alla lettura del Master Boot Record della scheda.
- **FSM\_READ\_FAT**: dediti alla lettura del Boot Sector del FAT32.
- **FSM\_PARSE\_FILE\_ENTRY**: dediti alla lettura della Root Directory Region.

- **FSM\_PARSE\_WAV\_FILE**: dediti alla lettura della Data Region e in particolar modo di file WAV trovati.

Analizzando il primo macro blocco, nello stato **FSM\_READ\_MBR\_1** si richiede la lettura del primo settore della memoria, ovvero il master boot record, si passa allo stato successivo, **FSM\_READ\_MBR\_2**, e quando sono presenti dati letti dalla scheda si controlla che il file system sia il FAT32 e si salva nel registro interno `far32_start` il numero di byte presenti tra MBR e il primo settore del file system. Se il tipo di partizione utilizzata non è FAT32 la macchina si blocca e viene segnalato un errore tramite il segnale `error_no_fat_found`.

Analogamente al precedente, il secondo macro blocco inizia con **FSM\_READ\_FAT\_0** che richiede la lettura del primo settore del file system, ovvero il boot sector. Di questo settore vengono salvati in appositi registri interni le informazioni contenute nel BIOS parameter block, ovvero informazioni sul file system del tipo:

- Il numero di Bytes per settore.
- Il numero di settori per cluster.
- Il numero di settori riservati all'inizio del file system.
- Il numero di File Allocation Tables.
- Il numero di settori per ogni FAT.
- Il numero di root cluster.

Una volta lette tali informazioni, la macchina evolve verso nello stato **FSM\_READ\_FAT\_2** dove si aspetta che il buffer sia vuoto tramite gli stati **FSM\_WAIT\_BUFFER\_0** e **FSM\_WAIT\_BUFFER\_1** in cui, rispettivamente, la macchina aspetta il flag `audio_buffer_empty_i`, manda il corrispondente acknowledgement, aspetta che il flag sopracitato si disattivato e infine evolve verso lo stato **FSM\_PARSE\_FILE\_ENTRY\_0**.

La macchina evolve, in questo modo, nel macro gruppo dedito alla lettura della Root Directory Region. Qui si scansiona le entries della regione ricercando quella corrispondente ad un file WAV riproducibile. Per farlo, nello stato **FSM\_PARSE\_FILE\_ENTRY\_0**, è richiesta la lettura del settore corrispondente ad una entry e nello stato **FSM\_PARSE\_FILE\_ENTRY\_1** si salvano, in appositi registri, le informazioni riportate in tale settore, ovvero:

- Il nome del file.
- L'estensione del file.
- Gli attributi del file
- L'indirizzo del primo cluster del file.
- La dimensione del file in bytes.

Successivamente si controlla che il filename non sia 0x00, in quanto sta ad indicare che è l'ultimo file entry nella Root Directory Region, in quel caso si riparte dall'inizio della regione stessa con la scansione. Al contrario se il filename è diverso si incrementa o decrementa `dir_entry_idx` (il contatore che guida la scansione) dipendentemente dal segnale `search_backwards`. Infine si esegue un ulteriore controllo sulle informazioni ottenute sul file, il cui risultato confluiscce sul flag `file_good` tramite una porta AND. Questi controlli consistono nella verifica sulla corretta estensione del file (.wav), un controllo sugli attributi del file (si controlla che non si tratti di una subdirectory) e un'ulteriore verifica sul file name (si controlla che sia diverso da 0xE5 in quanto corrispondente a file precedentemente cancellati).

Se `file_good` è alto, è stato trovato un file WAV valido e dunque possiamo procedere

con la lettura.

La lettura del file WAV inizia con la richiesta dei settori della scheda SD relativi al primo cluster del file WAV, per fare ciò viene abilitata la modalità di lettura continua tramite block\_read\_continuos\_mode successivamente, nello stato FSM\_PARSE\_WAV\_FILE\_1 si procede con la lettura dell'header del file ovvero 44 bytes, dal quale si ricavano informazioni sul file stesso:

- ID del file, 4 byte che contrassegnano il file come RIFF file.
- File type header: nel nostro caso deve essere WAVE.
- Format chunk header: nel nostro caso “fmt”
- Type of audio format.
- Number of channels: sono supportati solo file mono o stereo.
- Frequenza di campionamento: supportati segnali campionati a 44.1Khz.
- Data chunk header: nel nostro caso “data”.
- Dimensione del file.

Si verificano le informazioni ricevute affinché siano coerenti con quanto sopra riportato, se il controllo fallisce, la macchina evolve in FSM\_PARSE\_WAV\_FILE\_7 e da lì ritorna alla fase di parsing dei file, in caso contrario si continua con la lettura. Durante la ricezione dei dati relativi all'header del file nel buffer vengono scritti degli zeri.

Nello stato FSM\_PARSE\_WAV\_FILE\_2 avviene l'effettiva lettura della regione dati del file e la scrittura nel buffer. Quest'ultima è accompagnata da una serie di controlli: Quando è stato ricevuto un intero blocco della scheda SD la lettura continua solo se non sono stati ricevuti tutti i file relativi ad un cluster del file system e se il buffer non è pieno. Se il buffer è pieno viene attivato l'apposito flag audio\_buffer\_filled\_o e passando per lo stato FSM\_WAIT\_BUFFER\_0 si aspetta che il buffer sia svuotato. Se invece sono stati letti tutti i settori relativi ad un cluster la macchina evolve nello stato FSM\_PARSE\_WAV\_FILE\_3.

In questo stato si ha la richiesta settore della file allocation table relativo al cluster attualmente in lettura ottenendo in questo modo l'indirizzo del cluster successivo.

Se tale indirizzo corrisponde all'EOF si ritorna al parsing dei file, se così non è si procede con la lettura del file richiedendo i settori del cluster selezionato e si torna allo stato FSM\_PARSE\_WAV\_FILE\_2.

Infine si nota che tutte le volte che la macchina evolve nello stato FSM\_WAIT\_BUFFER\_0 vengono letti i segnali di ingresso player\_next\_song\_req\_i e player\_next\_song\_forward\_i che implementano le funzionalità di “Next Track” e “Restart Track/Previous Track”.

## RAM dualport

Modulo ottenuto per tramite MegaWizard Plug-In Manager, per questo motivo verrà illustrato esclusivamente il pinout, rimandando per maggiori informazioni alla documentazione Altera.

Input:

- **clock**.
- **data**: bus dati in scrittura
- **raddress**: bus indirizzi in lettura
- **wraddress**: bus indirizzi in lettura
- **wren**: segnale di abilitazione lettura

Output:

- **q**: dati in lettura.

Il clock di questo modulo è una versione sfasata di 180° rispetto al clock di sistema per garantire un corretto campionamento dei dati in ingresso.

Inoltre la gestione degli indirizzi è stata realizzata in modo tale che la RAM è come fosse divisa in due buffer distinti: in uno scrive FAT32\_reader mentre nell'altro legge Codec, quando uno è pieno e l'altro è vuoto si invertono. Per ottenere questa funzionalità il MSB di entrambi raddress e wraddress è controllato dal modulo codec.

## Codec Interface

Macchina a stati che costituisce l'interfaccia con il codec audio gestendo inizializzazione dello stesso e trasmissione dati. Quest'ultimo presenta dei sottomoduli istanziati al suo interno:

- **Codec\_config**: modulo I2C dedicato alla corretta configurazione del Codec audio.
- **I2S\_master**: modulo che gestisce la trasmissione dati tramite il protocollo I2S.

Analisi pinout del modulo.

Input:

- **clk, rst\_n**: clock e reset sincrono attivo basso.
- **codec\_pause\_i**: segnale proveniente dall'interfaccia utente che permette di mettere in pausa la riproduzione del brano.
- **codec\_buffer\_filled\_i**: Segnale per la sincronizzazione con il modulo FAT32 che scrive nel buffer dati, viene messo a 1 quando il buffer è pieno.
- **codec\_buffer\_empty\_ack\_i**: acknowledgement del segnale **codec\_buffer\_empty\_o**.
- **codec\_buffer\_data\_i**: bus dati provenienti dal buffer.
- **wav\_info\_audio\_channels\_i**: bus dati che specifica se il file audio è mono o stereo.
- **wav\_info\_sample\_rate\_i**: bus dati che specifica il sample rate del file audio.

Inout:

- **codec\_i2c\_sdat\_io**: linea dati in/out proveniente dall'interfaccia I2C.

Output:

- **codec\_aud\_xck\_o**: master clock line dell'interfaccia I2S.
- **codec\_aud\_bclk\_o**: bit clock line dell'interfaccia I2S
- **codec\_aud\_dacdat\_o**: data line dell'interfaccia I2S
- **codec\_aud\_daclrck\_o**: word clock line dell'interfaccia I2S
- **codec\_i2c\_sclk\_o**: slave clock interfaccia I2C.
- **codec\_buffer\_addr\_o**: indirizzo in lettura del buffer dati.
- **codec\_buffer\_sel\_o**: flag di selezione del buffer dati in cui leggere e di conseguenza anche di quello in cui scrivere.
- **codec\_buffer\_empty\_o**: Segnale per la sincronizzazione con il modulo FAT32 che scrive nel buffer dati, viene messo a 1 quando il buffer è vuoto.

La macchina a stati è descritta tramite un unico blocco always che implementa logica sincrona.

Oltre allo stato di reset abbiamo due stati di wait: **FSM\_WAIT\_BUFFER\_0** e **FSM\_WAIT\_BUFFER\_1**. Qui la macchina azzerà il segnale **codec\_buffer\_empty\_o** in caso sia stato ricevuto il corrispondente ack, se il modulo I2S invia il segnale **I2S\_done** se ne azzerà il trigger e successivamente si aspetta che il buffer dati attualmente in scrittura sia pieno e si inverte con quello in lettura alzando il flag **codec\_buffer\_empty\_o** per poi evolvere nello stato **FSM\_CONSUMING\_BUFFER**.

Qui si ha lo switch dei buffer se il buffer in scrittura è pieno e quello in lettura è vuoto (informazione contenuta nel flag **swap\_buffer**) e si inizia la lettura aggiornando gli indirizzi.

Il registro interno sample\_count è un contatore che ha un valore massimo pari a 1 se il file è mono, mentre se è stereo è pari a 3 questo permette di inviare opportunamente i campioni letti all'interfaccia I2S. Quindi in pratica in questo stato si inviano al codec 1 o 2 sample dipendentemente se il file è mono o stereo, si valuta se il buffer è stato letto completamente e si continua con la lettura o si torna allo stato FSM\_WAIT\_BUFFER\_1.

Lo stato FSM\_WAIT\_I2S serve per aspettare che il modulo I2S abbia terminato la trasmissione dei sample prima di continuare con la lettura. Questo inoltre è lo stato in cui si aspetta nel caso in cui dall'interfaccia utente arrivi il comando di “Pause”

Si procede con l'analisi dei sottomoduli.

## Codec Config

Analisi pinout del modulo.

Input:

- **clk, rst\_n:** clock e reset sincrono attivo basso

Inout:

- **i2c\_sdat:** data line interfaccia I2C

Output:

- **i2c\_sclk:** slave clock.

Modulo che istanzia al suo interno il sottomodulo I2C\_controller che implementa il physical layer del protocollo I2C, quest'ultimo è stato ripreso dagli esempi forniti da Altera per questo motivo se ne analizza solo il pinout e si rimanda alla documentazione Altera per maggiori informazioni.

Input:

- **CLOCK, RESET.**
- **I2C\_DATA:** registro di ingresso che conterrà l'indirizzo del dispositivo slave, l'indirizzo di un registro interno dello slave e i dati da scriverci dentro.
- **GO:** variabile di sblocco della scrittura dei registri interni di un certo dispositivo slave.

Inout:

- **I2C\_SDAT:** linea dati.

Output:

- **I2C\_SCLK:** linea per il clock del dispositivo slave.
- **END:** variabile che indica, se attiva, il termine della trasmissione
- **ACK:** variabile che indica, se a'0', se la comunicazione è avvenuta correttamente

Il modulo Codec Config è costituito da 3 blocchi always, il primo presenta nella sensitivity list il clock di sistema e genera il clock di per il modulo I2C\_controller.

Il secondo, invece, implementa una LUT nel quale sono salvati i dati da scrivere nei registri per l'opportuna configurazione del codec e gli indirizzi degli stessi.

Il terzo e ultimo blocco always è sincrono con I2C controller e semplicemente invia i dati da scrivere e aspetta una risposta positiva dal I2C master per inviare il successivo.

## I2S Master

Analisi pinout del modulo.

Input:

- **clk, rst\_n**: clock e reset sincrono attivo basso.
- **i2s\_sample\_data\_L\_i**: bus dati in ingresso relativo al canale sinistro.
- **i2s\_sample\_data\_R\_i**: bus dati in ingresso relativo al canale destro.
- **i2s\_send\_i**: Segnale che fa evolvere la macchina a stati dalla condizione di riposo.

Output:

- **codec\_aud\_xck\_o**: vedi pinout di Codec.
- **codec\_aud\_bclk\_o**: vedi pinout di Codec.
- **codec\_aud\_dacdat\_o**: vedi pinout di Codec.
- **codec\_aud\_daclrck\_o**: vedi pinout di Codec.
- **i2s\_done\_o**: flag che segnala il termine della trasmissione dei campioni ricevuti.

Il modulo è costituito da due tre blocchi always.

Il primo, sincrono con il clk, è dedicato alla generazione di codec\_aud\_xck\_o e codec\_aud\_bclk\_o a partire dal clock di sistema. Per farlo utilizza due contatori utilizzati da divisorì in frequenza, ovvero xck\_counter e bclk\_counter. Il primo divide il clock di un fattore XCK\_CNT\_TOP mentre l'altro di un ulteriore BCLK\_CNT\_TOP in modo tale che alla fine codec\_aud\_bclk\_o abbia una frequenza sufficiente affinché sia possibile campionare tutti i bit di due sample del file audio (nel caso sia un file stereo) i quali devono essere inviati con la frequenza più possibile simile a quella di campionamento.

Il secondo blocco always, triggerato nei fronti in discesa di codec\_aud\_bclk\_o invia i bit (MSB First) dei campioni ricevuti in ingresso.

Il terzo blocco always, sensibile ai fronti in salita del clock di sistema, costituisce la funzione di stato del modulo. Oltre che lo stato di reset abbiamo uno stato di riposo FSM\_IDLE nel quale si aspetta il segnale i2s\_send\_i per evolvere nello stato FSM\_SEND.

In questo stato si commuta il segnale codec\_aud\_daclrck\_o quando sono stati inviati tutti i bit relativi ad un sample, tale segnale, ricordiamolo, deve presentare una frequenza pari alla frequenza di campionamento del file audio e quando è alto determina l'invio del campione sinistro mentre quando è basso di quello destro.

Quando sono stati inviati sia un campione destro e sinistro, se i2s\_send\_i è ancora attivo, si continua ad inviare i nuovi campioni ricevuti. Mentre se il segnale i2s\_send\_i viene disattivato la macchina ritorna nello stato di IDLE

Il segnale i2s\_done\_o viene attivato quando la macchina è nello stato FSM\_IDLE o quando sono stati inviati sia il campione destro e sinistro.

## Main PLL

Modulo ottenuto dalle librerie Altera, si faccia dunque riferimento all'apposita documentazione per maggiori informazioni.

Il PLL è usato per ottenere un clock di sistema con frequenza di 100MHz a partire da un clock di 50MHz.

## Codec audio WM8731

Il codec audio WM8731 si occupa della conversione A/D e D/A e dell'amplificazione del segnale. Lo schema a blocchi è mostrato in figura:



Figura 3.7.1

Caratteristiche:

- Stereo Audio Codec with Integrated Headphone Driver.
- Stereo ADC di tipo - con SNR pari a 90 dB.
- Frequenza di campionamento impostabile da 8 kHz a 96 kHz.
- Master or slave clocking mode.
- Word Length impostabile da 8 bit a 32 bit.
- Programmabile tramite interfaccia I2C o SPI. La scelta tra le due è determinata dallo stato logico di un pin esterno.
- Programmable Audio Data Interface: I2S, Left/Right Justified o DSP. Output Volume and Mute Controls.

Datasheet: <http://cdn.sparkfun.com/datasheets/Dev/Arduino/Shields/WolfsonWM8731.pdf>.

# Debug e funzionamento

In questa sezione vengono mostrate waveforms acquisite con l'analizzatore logico, per mostrare il funzionamento *live* - ossia non simulato - del sistema.

## Protocollo SPI scheda SD

Dopo il reset, avvenuto a +0.6 ms, si osserva il chip select andare alto per 80 cicli di clock. Questa procedura segnala alla scheda SD che deve uscire dal suo stato di power down (in cui riduce praticamente a zero i consumi) e prepararsi a comunicare.



Viene poi mandato il primo comando, ossia il CMD0, a cui la scheda risponde con una parola di 8 bit “0x01”. L'ultimo bit segnala che la scheda non è ancora pronta a leggere/scrivere i dati al suo interno e necessita prima di terminare la sua configurazione.

## Protocollo I2C codec

Dopo il reset vengono anche configurati i registri del codec attraverso il protocollo I2C.



In figura, sono evidenziati con pallini verdi i bit di *start* e in rosso i bit di *stop* del protocollo I2C. È mostrata la scrittura del registro 0x0: viene prima mandato l'indirizzo del codec e poi due bytes, contenenti i 7 bit di indirizzo del registro e i 9 bit di dati del registro; in questo caso entrambi zero.

## Protocollo I2S codec

Il protocollo I2S si compone di tre segnali, mostrati in figura.



I campioni, in formato pcm16s, sono rappresentati come interi con segno in complemento a due. Possiamo quindi concludere che i dati in figura, che iniziano con '0xF', sono numeri negativi.

## Lettura di un blocco della scheda SD

La lettura di un blocco inizia mandando il comando di lettura, contenente l'indirizzo del blocco da 512 bytes da leggere. Si aspetta poi che i dati siano pronti. Con questa scheda SD il tempo di attesa è di circa 308 us, e viene misurato con la coppia di marker rossi in figura.



Quando i dati sono pronti, vengono trasmessi i 512 bytes del blocco uno dopo l'altro. Al termine del blocco in figura si vede che viene immediatamente mandato il comando di lettura successivo.

Mostriamo ora un zoom sul comando di lettura:



Il comando si compone di 6 bytes. Il primo byte è il numero del comando. I bytes 2-5 contengono invece il blocco da leggere: in figura viene richiesto il blocco '0x00001000'.

I dati sono pronti quando la scheda risponde '0xFE'



Il byte successivo, ossia '0xF8', è il primo byte valido del blocco da 512 bytes richiesto.

## Lettura di un cluster FAT32

Per la lettura di un cluster bisogna fare alcune operazioni:

1. leggere la FAT per scoprire in quale settore inizia il cluster successivo
  2. legger tutto il cluster mandando il comando 'continuous block read'
  3. dopo aver letto 16 blocchi, ossia la dimensione di un cluster, mandare il comando di 'stop continuous block read'



Si notano due intervalli in cui il MISO rimane in *idle* sul livello logico alto: sono gli intervalli in cui si aspetta che siano pronti rispettivamente il blocco della fat e il primo blocco del cluster.

La lettura sequenziale offre il vantaggio di essere più rapida: non c'è da aspettare il tempo di fetch del blocco poiché la scheda SD lavora con la pipeline piena.



Mostriamo ora l'invio del comando di stop:



Notare che la scheda SD ha già iniziato a mandare il blocco successivo, che inizia col byte '0x69', poiché '0xFF' e '0xFE' sono rispettivamente il byte di idle e di start data block.

Dopo aver ricevuto il comando di stop, la scheda risponde con '0x00' e cessa di inviare dati.

## Funzionamento a regime

In figura sono mostrate le ripetute letture della scheda SD, per riempire il buffer coi campioni audio da mandare al codec.



La frequenza con cui si ripetono le letture coincide dipende dal tempo impiegato dal codec per riprodurre tutti i campioni dentro il buffer.

La traccia gialla mostra il segnale 'buffer full': esso viene messo a 1 quando il buffer viene riempito leggendo i dati dalla scheda SD. Scende poi a 0 quando il codec segnala di avere bisogno di un altro buffer pieno; inizia così il processo di riempimento del buffer successivo. Ricordiamo che viene impiegato il double buffering: in ogni istante c'è un buffer che viene letto e uno che viene riempito.

# Conclusioni

Il sistema illustrato implementa un lettore walkman di file WAV funzionante, ciò nonostante sarebbe comunque possibile apportare delle migliorie, come ad esempio un'interfaccia utente più completa, sfruttando anche il display LCD di cui è fornita la scheda di sviluppo per mostrare il nome della canzone in riproduzione.

È stato molto interessante il debug con l'analizzatore logico: sfruttare i GPIO per portare 'nel mondo esterno' i segnali che normalmente sono sepolti dentro il chip offre possibilità di debug estremamente flessibili e immediate, che non sono assolutamente disponibili quando si lavora invece con i microcontrollori, a cui eravamo abituati.

In conclusione, il progetto si è rivelato molto interessante e gratificante nella sua realizzazione.