

# Audio core for Intel<sup>®</sup> DE-Series Boards

For Quartus® Prime 18.0

#### 1 Core Overview

The Audio core interacts with the Audio CODEC (enCOder/DECoder) on the Intel® DE-series boards and provides an interface for audio input and output.

## **2 Functional Description**

The Audio core facilitates the transfer of audio data with the Audio CODEC chip on the Intel DE-series boards. Data is transferred serially between the FPGA and the CODEC. However, data written/read to/from the Audio core is transferred in a parallel manner. The Audio core automatically serializes/deserializes the data.

The Audio core contains four FIFOs to buffer the In and Out audio data, both having the right and left audio channels. Each FIFO can store up to 128 32-bit words. To guarantee that the left and right audio output channels are synchronized, data will not play until both channels are received. If only one channel is to be played, the other channel must have zeros written to it.

The Audio core can be connected to a system using one of two different modes: Memory-Mapped or Streaming. Figure 1 shows a block diagram of the Audio core with a memory-mapped interface. This mode allows access to the core's FIFOs through memory-mapped registers. This is suitable when connecting the core to a processor, such as the Nios<sup>®</sup> II processor, using the Platform Designer System Integration tool. Figure 2 shows a block diagram of the Audio core with a streaming interface. This mode allows direct access to the core's FIFOs. This is suitable when connecting to custom hardware using the Platform Designer System Integration tool or as a standalone component using the IP Catalog.

The Audio core requires certain clock frequencies based on the sample rate of the audio. The University Program's IP core, *Audio Clock for DE-series Boards*, can be use to provide those required clock frequencies. The Audio core also requires that the audio chip be initialized with some default values. The University Program's IO core, *Audio and Video Config*, provides the functionality required to initialize the audio chip. Refer to the specific documentation for those cores for more information.



Figure 1. Block diagram for Audio core with Memory-Mapped Interface

## 3 Instantiating the Core

The Audio core can be instantiated in a system using Platform Designer or as a standalone component from the IP Catalog within the Quartus<sup>®</sup> Prime software. Designers use the Audio core's **Configuration wizard** to specify the desired features. In the configuration wizard, the user can choose the interface type, whether it be memory-mapped or streaming. Also, the user can select whether to included only one or both of the audio in and/or audio out channels. In addition, the Data Width per Channel can be specified. Data widths of 16, 20, 24, and 32 bits are supported. Remember to export the external\_interface connection to connect the core with the audio CODEC chip. It is recommended to set the Avalon Type to Memory-Mapped when connecting to a processor, otherwise set it to Streaming.

- Image: Intel recommends also instantiating the *Audio and Video Config* core. This core automatically configures some required settings of the audio CODEC chip on the DE2/DE1 boards. Refer to the *Audio and Video Config* documentation for more information on properly intializing the audio codec.
- The user *must* also instantiate the *Audio Clock for DE-series Boards* core and choose the proper audio clock setting for the Audio core. See Wolfson\* WM8731 audio CODEC Datasheet in the "Audio Data Sampling Rates" section on page 37 for details on the relationship between sampling rate and clock frequency. Note that the Audio and Video Config core provides settings for these values.
- When using the Memory-Mapped Avalon Type in Platform Designer, Intel recommends that the Audio core be used with the standard or fast versions of the Intel Nios II processor, so that a program running on the processor



Figure 2. Block diagram for Audio core with Streaming Interface

can keep up with the generation of audio data. If the economic version of the processor is used, then the program may run too slowly, and the audio may not be clear. In such, cases, it may be possible to improve the audio clarity by selecting a lower sampling rate in the audio chip.

## 4 Software Programming Model

### 4.1 Register Map

Software can control and communicate with the Audio core through four 32-bit registers when using the Memory-Mapped Avalon Type. By writing or reading these registers, data can be fetched from the CODEC's Analog-Digital Converter (ADC) or sent to the Digital-Analog Converter (DAC). Table 1 shows the format of the registers.

| Table 1. Audio core register map |           |        |                 |       |       |    |       |     |    |    |    |    |  |
|----------------------------------|-----------|--------|-----------------|-------|-------|----|-------|-----|----|----|----|----|--|
| Offset                           | Register  | R/W    | Bit Description |       |       |    |       |     |    |    |    |    |  |
| in bytes                         | Name      | IX/ VV | 3124            | 2316  | 1510  | 9  | 8     | 74  | 3  | 2  | 1  | 0  |  |
| 0                                | control   | RW     |                 | (1)   |       | WI | RI    | (1) | CW | CR | WE | RE |  |
| 4                                | fifospace | R      | WS LC           | WS RC | RA LC |    | RA RC |     |    |    |    |    |  |
| 8                                | leftdata  | RW (2) | Left Data       |       |       |    |       |     |    |    |    |    |  |
| 12                               | rightdata | RW (2) | Right Data      |       |       |    |       |     |    |    |    |    |  |

Notes on Table 1:

- (1) Reserved. Read values are undefined. Write zero.
- (2) Only reads incoming audio data and writes outgoing audio data.

#### 4.1.1 *Control* Register

| Table 2. Control register bits |          |            |                                                           |  |  |  |  |
|--------------------------------|----------|------------|-----------------------------------------------------------|--|--|--|--|
| Bit number                     | Bit name | Read/Write | Description                                               |  |  |  |  |
| 0                              | RE       | R/W        | Interrupt-enable bit for read interrupts. If the RE bit   |  |  |  |  |
|                                |          |            | is set to 1 and both the left and right channel read      |  |  |  |  |
|                                |          |            | FIFOs contain data, the Audio core generates an in-       |  |  |  |  |
|                                |          |            | terrupt request (IRQ).                                    |  |  |  |  |
| 1                              | WE       | R/W        | Interrupt-enable bit for write interrupts. If the WE      |  |  |  |  |
|                                |          |            | bit is set to 1 and both the left and right channel write |  |  |  |  |
|                                |          |            | FIFOs have space available for more data, the Audio       |  |  |  |  |
|                                |          |            | core generates an interrupt request (IRQ).                |  |  |  |  |
| 2                              | CR       | R/W        | Clears the Audio core's Input FIFOs, when the bit is      |  |  |  |  |
|                                |          |            | 1. Clear remains active until specifically set to zero.   |  |  |  |  |
| 3                              | CW       | R/W        | Clears the Audio core's Output FIFOs, when the bit        |  |  |  |  |
|                                |          |            | is 1. Clear remains active until specifically set to      |  |  |  |  |
|                                |          |            | zero.                                                     |  |  |  |  |
| 8                              | RI       | R          | Indicates that a read interrupt is pending.               |  |  |  |  |
| 9                              | WI       | R          | Indicates that a write interrupt is pending.              |  |  |  |  |

#### 4.1.2 Fifospace Register

The fifospace register fields WSLC ( $b_{31-24}$ ) and WSRC ( $b_{23-16}$ ) indicate the number of words available (i.e., the amount of empty space) for outgoing data in the left and right channel FIFOs, respectively, while RALC ( $b_{15-8}$ ) and RARC ( $b_{7-0}$ ) indicate the number of words of incoming audio data in the left and right channel FIFOs, respectively. When all of the outgoing and incoming FIFOs are empty, the fifospace register will hold WSLC = WSRC = 128, and RALC = RARC = 0.

#### 4.1.3 Leftdata Register

The leftdata register is readable only for Audio In and writable only for Audio Out. It stores the data coming from or going to the left channel. The data is always flush right, i.e., the LSB is  $b_0$  of the leftdata register.

#### 4.1.4 Rightdata Register

The rightdata register is readable only for Audio In and writable only for Audio Out. It stores the data coming from or going to the right channel. The data is always flush right, i.e., the LSB is b<sub>0</sub> of the right data register.

#### 4.2 **Interrupt Behavior**

The Audio core produces a read interrupt when either of the read FIFOs are filled to 75% or more. The interrupt is cleared when the FIFO becomes less than 75% full. Also, it produces a write interrupt when either of the write FIFOs have available space of 75% or more. The interrupt is cleared when the FIFO becomes less than 75% empty. The Audio core generates an interrupt when either of these individual interrupt conditions are pending and enabled.

#### Device Driver for the Nios® II Processor

For use with the Nios II processor, the Audio core is packaged with C-language functions accessible through the hardware abstraction layer (HAL). These functions implement basic operations for the Audio core.

To use the functions, the C code must include the statement:

```
#include "altera_up_avalon_audio.h"
```

An example of C code that uses the Audio core is given at the end of this section.

### 4.3.1 alt\_up\_audio\_open\_dev

**Prototype:** alt\_up\_audio\_dev\* alt\_up\_audio\_open\_dev(const

char \*name)

Include: <altera\_up\_avalon\_audio.h> **Parameters:** name – the audio component name in Qsys.

**Returns:** The corresponding device structure, or NULL if the device is not found Opens the audio device specified by name (default "/dev/audio/"). **Description:** 

#### 4.3.2 alt\_up\_audio\_enable\_read\_interrupt

**Prototype:** void alt\_up\_audio\_enable\_read\_interrupt(alt\_up\_audio\_dev

\*audio)

**Include:** <altera\_up\_avalon\_audio.h> **Parameters:** audio - the audio device structure

**Returns:** 

**Description:** Enable read interrupts for the Audio Core.

#### 4.3.3 alt\_up\_audio\_disable\_read\_interrupt

Prototype: void alt\_up\_audio\_disable\_read\_interrupt(alt\_up\_audio\_dev

\*audio)

**Returns:** nothing

**Description:** Disable read interrupts for the Audio Core.

#### 4.3.4 alt\_up\_audio\_enable\_write\_interrupt

Prototype: void alt\_up\_audio\_enable\_write\_interrupt(alt\_up\_audio\_dev

\*audio)

**Returns:** nothing

**Description:** Enable write interrupts for the Audio Core.

### 4.3.5 alt\_up\_audio\_disable\_write\_interrupt

**Prototype:** void alt\_up\_audio\_disable\_write\_interrupt(alt\_up\_audio\_dev

\*audio)

**Returns:** nothing

**Description:** Disable the read interrupts for the Audio Core.

#### 4.3.6 alt\_up\_audio\_read\_interrupt\_pending

Prototype: int alt up audio read interrupt pending (alt up audio dev

\*audio)

Include: <altera\_up\_avalon\_audio.h>
Parameters: audio – the audio device structure
Returns: 1 if read interrupt is pending, else 0

**Description:** Check if read interrupt pending for the Audio Core.

## 4.3.7 alt\_up\_audio\_write\_interrupt\_pending

Prototype: int alt\_up\_audio\_write\_interrupt\_pending(alt\_up\_audio\_dev

\*audio)

Include: <altera\_up\_avalon\_audio.h>
Parameters: audio – the audio device structure
Returns: 1 if write interrupt is pending, else 0

**Description:** Check if write interrupt pending for the Audio Core.

#### 4.3.8 alt\_up\_audio\_reset\_audio\_core

Prototype: void alt\_up\_audio\_reset\_audio\_core(alt\_up\_audio\_dev

\*audio)

**Returns:** nothing

**Description:** Reset the Audio Core by clearing read and write FIFOs for left and right

channels.

#### 4.3.9 alt\_up\_audio\_read\_fifo\_avail

**Prototype:** unsigned int alt\_up\_audio\_read\_fifo\_avail(alt\_up\_audio\_dev

\*audio, int channel)

channel – left or right channel selection

**Returns:** number of words available

**Description:** provides number of words of data available in the incoming FIFO for

channel

#### 4.3.10 alt\_up\_audio\_record\_r

Prototype: unsigned int alt\_up\_audio\_record\_r(alt\_up\_audio\_dev

\*audio, unsigned int \*buf, int len)

buf – the pointer to the allocated memory for storing audio data. Size

of buf should be no smaller than len words.

len – the number of data in words to read from the input FIFO

**Returns:** The total number of words read.

**Description:** Read len words of data from right input FIFO, if the FIFO is above a

threshold, and store data to where buf points.

#### 4.3.11 alt\_up\_audio\_record\_l

Prototype: unsigned int alt\_up\_audio\_record\_l(alt\_up\_audio\_dev

\*audio, unsigned int \*buf, int len)

buf – the pointer to the allocated memory for storing audio data. Size

of buf should be no smaller than len words.

len - the number of data in words to read from the input FIFO

**Returns:** The total number of words read.

**Description:** Read len words of data from left input FIFO, if the FIFO is above a

threshold, and store data to where buf points.

#### 4.3.12 alt\_up\_audio\_write\_fifo\_space

Prototype: unsigned int alt\_up\_audio\_write\_fifo\_space(alt\_up\_audio\_dev

\*audio, int channel)

Include: <altera\_up\_avalon\_audio.h>
Parameters: audio - the audio device structure

channel - left or right channel enum

**Returns:** number of words available

**Description:** provides the amount of empty space in the outgoing FIFO for *channel* 

#### 4.3.13 alt\_up\_audio\_play\_r

**Prototype:** unsigned int alt\_up\_audio\_play\_r(alt\_up\_audio\_dev

\*audio, unsigned int \*buf, int len)

buf - the pointer to the data to be written. Size of buf should be no

smaller than len words.

len – the number of data in words to be written into the output FIFO

**Returns:** The total number of data written.

**Description:** Write len words of data into right output FIFO, if space available in

FIFO is above a threshold.

#### 4.3.14 alt\_up\_audio\_play\_l

**Prototype:** unsigned int alt\_up\_audio\_play\_l(alt\_up\_audio\_dev

\*audio, unsigned int \*buf, int len)

buf – the pointer to the data to be written. Size of buf should be no

smaller than len words.

len - the number of data in words to be written into the output FIFO

**Returns:** The total number of data written.

**Description:** Write len words of data into left output FIFO, if space available in FIFO

is above a threshold.

#### 4.3.15 alt\_up\_audio\_read\_fifo

**Prototype:** int alt\_up\_audio\_read\_fifo(alt\_up\_audio\_dev

\*audio, unsigned int \*buf, int len, int

channel)

buf – the pointer to the allocated memory for storing audio data. Size

of buf should be no smaller than len words.

len – the number of data in words to read from each input FIFO

channel - left or right channel selection

**Returns:** The total number of words read.

**Description:** Read *len* words of data from left input FIFO or right input FIFO, and

store data to where buf points.

#### 4.3.16 alt\_up\_audio\_write\_fifo

Prototype: int alt\_up\_audio\_write\_fifo(alt\_up\_audio\_dev

\*audio, unsigned int \*buf, int len, int

channel)

buf - the pointer to the data to be written. Size of buf should be no

smaller than len words.

len – the number of data in words to be written into each output FIFO

channel - left or right channel selector

**Returns:** The total number of data written.

**Description:** Write *len* words of data from *buf* to the left or right output FIFOs.

#### 4.3.17 alt\_up\_audio\_read\_fifo\_head

**Prototype:** unsigned int alt\_up\_audio\_read\_fifo\_head(alt\_up\_audio\_dev

\*audio, int channel)

channel - left or right channel selection

**Returns:** the word read

**Description:** Read one data word from left input FIFO or right input FIFO.

#### 4.3.18 alt\_up\_audio\_write\_fifo\_head

Prototype: void alt\_up\_audio\_write\_fifo\_head(alt\_up\_audio\_dev

\*audio, unsigned int data, int channel)

channel – left or right channel selector

**Returns:** nothing

**Description:** Write one data word to the left or right output FIFOs.

## 4.3.19 Audio core C Example using Device Drivers

```
#include "altera_up_avalon_audio.h"
int main(void)
     alt_up_audio_dev * audio_dev;
     /* used for audio record/playback */
     unsigned int 1 buf;
     unsigned int r_buf;
     // open the Audio port
     audio_dev = alt_up_audio_open_dev ("/dev/Audio");
     if ( audio_dev == NULL)
            alt_printf ("Error: could not open audio device \n");
     else
            alt_printf ("Opened audio device \n");
     /* read and echo audio data */
     \mathbf{while}(1)
           int fifospace = alt up audio read fifo avail (audio dev, ALT UP AUDIO RIGHT);
           if (fifospace > 0) // check if data is available
                 // read audio buffer
                 alt_up_audio_read_fifo (audio_dev, &(r_buf), 1, ALT_UP_AUDIO_RIGHT);
                 alt_up_audio_read_fifo (audio_dev, &(l_buf), 1, ALT_UP_AUDIO_LEFT);
                 // write audio buffer
                 alt_up_audio_write_fifo (audio_dev, &(r_buf), 1, ALT_UP_AUDIO_RIGHT);
                 alt_up_audio_write_fifo (audio_dev, &(l_buf), 1, ALT_UP_AUDIO_LEFT);
      }
}
```

Figure 3. An example of C with Device Driver Support code that uses Audio Core.

Copyright © Intel Corporation. All rights reserved. Intel, the Intel logo, Altera, Arria, Avalon, Cyclone, Enpirion, MAX, Nios, Quartus and Stratix words and logos are trademarks of Intel Corporation or its subsidiaries in the U.S. and/or other countries. Intel warrants performance of its FPGA and semiconductor products to current specifications in accordance with Intel's standard warranty, but reserves the right to make changes to any products and services at any time without notice. Intel assumes no responsibility or liability arising out of the application or use of any information, product, or service described herein except as expressly agreed to in writing by Intel. Intel customers are advised to obtain the latest version of device specifications before relying on any published information and before placing orders for products or services.

\*Other names and brands may be claimed as the property of others.