Add an AXI-Stream Interface

In this part of the I2S tutorial for FPGAs, I would like to show how the can expand the I2S transmitter with an AXI stream interface.

For this purpose, a new top design called AXIS_I2S is created. This design should have the following interface:

Signal

Descritpion

MCLK

Master Clock

nReset

Reset input for the audio interface (active low)

ACLK

Clock input for the AXI-Stream interface

ARESETn

Reset input for the AXI-Stream interface (active low)

TDATA_RXD

Data inputs for the AXI-Stream Interface

TREADY_RXD

Ready signal for the AXI-Stream Interface

TVALID_RXD

Valid signal for the AXI-Stream Interface

SCLK

Serial clock for the I2S interface

LRCLK

Left/Right clock (WS) for the I2S interface

SD

Serial data for the I2S interface

This block design results in the following entity:

The ratio of SCLK to MCKL is defined via the parameter RATIO and the width of a data word per channel via the parameter WIDTH.

This implementation only supports 16-bit data words per channel (i.e. 32 bit for stereo). The following code must be adapted for larger bus widths.

The following components must be implemented in the design:

  • A clock prescaler to create the input clock for the I2S transmitter

  • An AXI-Stream slave interface

  • The control logic for the I2S transmitter

A process is created for the divider, which counts up a counter on the rising clock edge of MCLK and switches the signal SCLK_Int after half the period.

The next step is to implement the AXI-Stream interface. A state machine is used for this:

After a reset, the machine changes from the State_Reset state to the State_WaitForTransmitterReady state, where it waits for the ready signal from the I2S transmitter. As soon as the transmitter is ready, the TREADY_RXD signal of the AXI-Stream interface is set, whereby the master is informed that the slave is ready to receive data. The slave then changes to the State_WaitForValid state.

In this state, the slave waits for the master to set the TVALID_RXD signal to mark valid data. As soon as the signal is set, the data is written to an internal FIFO. The machine then changes to the State_WaitForTransmitterBusy state.

Now the state machine waits for the I2S transmitter to start transmitting the data and to delete the ready signal. As soon as this is done, the machine switches back to the State_WaitForTransmitterReady state and waits again until the I2S transmitter is ready.

With that, the AXI-Stream Interface would be finished in theory. Unfortunately, it gets a bit tricky at the end, since the current circuit design uses two different clock domains:

  • The clock domain for ACLK

  • The clock domain for MCLK

In general, these two clock signals cannot be generated from a clock source (e.g. via a clock divider), since the AXI interface typically runs at 100 MHz and the audio interface requires clock rates that can be neatly divided down to the sampling frequency, such as e.g. 12.288 MHz. As a result, timing errors occur during implementation due to excessive worst negative slack (WNS) and total negative slack (TNS):

Also, the risk of incorrect data due to the metastability of the flip-flops occurring with different clock domains is very high. Metastability occurs a. then when a flip-flop switches and the data changes at that very moment.

Therefore, the signals that are used by the individual clock domains must be transferred to the other clock domain in each case via corresponding circuits. Xilinx describes corresponding macros in document UG953 that can be used for this purpose.

Macro

Function

xpm_cdc_gray

This function block uses Gray code to transfer a data bus from one clock domain (src) to another clock domain (dest).

xpm_cdc_single

Converts a single signal from one clock domain (src) to another clock domain (dest).

The examples of the macros can be used directly for the VHDL code:

Finally, the I2S transmitter must be inserted and the generated signals passed on.

The AXI-Stream Interface for the I2S transmitter is now ready and ready for use. The complete code should now look like this:

The created AXI-Stream I2S transmitter can now be created as an IP core for Vivado if required, whereby I will omit the necessary steps here. Alternatively, the finished IP core can also be downloaded from the project's GitLab repository.

Last updated

Was this helpful?