Re: I2S for generating waveforms for ledpixel devices
Posted: Sun Aug 30, 2020 10:05 am
OK I can definitely help you with quite a bit of this.
There are a couple of registers that control everything, and all of the values labelled *_NUM are a bit mask, with the corresponding *_NUM_S being the bit shift. So you'll see lots of code of the form:
unless you know enough to set the values within their range.
BCK_DIV_NUM, CLKM_DIV_NUM and BITS_MOD work together to set the I2S clock frequency and the number of significant bits in the signal. You have to find the optimal value of all three that's closest to your desired frequency. These are my own notes in my header file so I remember:
This is my I2S init routine, which should also give you an idea about some of the other values - these control whether your I2S Tx and Rx is in master or slave mode in terms of the clock, how many channels, whether the data is 16 or 24 bit, and how the values are stored within the word in the buffer.
I am only using receive in this code, but I've left in the Tx parts so I would remember:
There are a couple of registers that control everything, and all of the values labelled *_NUM are a bit mask, with the corresponding *_NUM_S being the bit shift. So you'll see lots of code of the form:
Code: Select all
setReg(main_register, specific_control_value & {control_value}_NUM << {control_value}_NUM_S);
unless you know enough to set the values within their range.
BCK_DIV_NUM, CLKM_DIV_NUM and BITS_MOD work together to set the I2S clock frequency and the number of significant bits in the signal. You have to find the optimal value of all three that's closest to your desired frequency. These are my own notes in my header file so I remember:
Code: Select all
/* These control the PLL settings for the generation of the I2S SCK or BCK from the 160MHz PLL clock (CPU speed independent)
The bit clock BCK = (Desired Sampling rate)*(Sampling Depth)*(2 channels)
CLK_I2S = 160MHz / I2S_CLKM_DIV_NUM
BCLK = CLK_I2S / I2S_BCK_DIV_NUM
WS = BCLK/ 2 / (16 + I2S_BITS_MOD)
Note that I2S_CLKM_DIV_NUM must be >5 for I2S data
I2S_CLKM_DIV_NUM - 5-63
I2S_BCK_DIV_NUM - 2-63
This needs examining in detail - not sure how the samples are transmitted or what the BITS field does to the rate or depth.
The BITS may be 24bit (bit depth), 31bit (max value in register) or 32bit (value size returned from mic) depending on how it works
For now, these values are for:
- 48kHz and 32bit size, 4 and 13 (actual rate = 48076.9)
- 48kHz and 31bit size, 6 and 9 (actual rate = 47789.7)
- 48kHz and 24bit size, 7 and 10 (actual rate = 47619.0)
- 48kHz and 17bit size, 7 and 14 (actual rate = 48019.2) <== closest possible rate to desired
*/
#define WS_I2S_BCK_DIV 4 /* Bit clock divisor 2-63 */
#define WS_I2S_CLKM_DIV 13 /* Clock Divisor 5-63 */
#define WS_I2S_BITS 15 /* Sampling depth-16. Currently needs looking into as it's 4 bits max, which gives 31 bits max?? */
/* To receive data from the I2S module, counter-intuitively we use the TXLINK part, not the RXLINK part, and vice versa.
* Note:At the transmitter side,the size of the DMAs can not be smaller than 128*4 bytes which are the
* size of the I2S FIFO.
* Note:At the receiver side,the number of the DMAs can not be smaller than 3 which is limited by the
* hardware.
*/
#define I2SDMABUFLEN 128 /* this is 32bit words for I2S, but DMA deals in bytes, so in any DMA code it's 128*4 */
#define DMABUFFERDEPTH 3
#define RX_NUM (I2SDMABUFLEN)
...
//I2S DMA buffer descriptors
static struct sdio_queue i2sBufDescRX[DMABUFFERDEPTH];
static uint32_t i2sBDRX[I2SDMABUFLEN * DMABUFFERDEPTH];
This is my I2S init routine, which should also give you an idea about some of the other values - these control whether your I2S Tx and Rx is in master or slave mode in terms of the clock, how many channels, whether the data is 16 or 24 bit, and how the values are stored within the word in the buffer.
I am only using receive in this code, but I've left in the Tx parts so I would remember:
Code: Select all
//Initialize the I2S module
//More Registor details in I2S documents.
void ICACHE_FLASH_ATTR i2s_init(void) {
//CONFIG I2S RX PIN FUNC
PIN_FUNC_SELECT(PERIPHS_IO_MUX_MTDI_U, FUNC_I2SI_DATA);
PIN_FUNC_SELECT(PERIPHS_IO_MUX_MTCK_U, FUNC_I2SI_BCK);
PIN_FUNC_SELECT(PERIPHS_IO_MUX_MTMS_U, FUNC_I2SI_WS);
//Enable a 160MHz clock to i2s subsystem
i2c_writeReg_Mask_def(i2c_bbpll, i2c_bbpll_en_audio_clock_out, 1);
//reset I2S
CLEAR_PERI_REG_MASK(I2SCONF,I2S_I2S_RESET_MASK);
SET_PERI_REG_MASK(I2SCONF,I2S_I2S_RESET_MASK);
CLEAR_PERI_REG_MASK(I2SCONF,I2S_I2S_RESET_MASK);
//Enable FIFO in i2s module
SET_PERI_REG_MASK(I2S_FIFO_CONF, I2S_I2S_DSCR_EN);
//set I2S_FIFO
//set rx,tx data size, both are "24-bit full data discontinue" here
SET_PERI_REG_MASK(I2S_FIFO_CONF, (I2S_I2S_RX_FIFO_MOD<<I2S_I2S_RX_FIFO_MOD_S)|(I2S_I2S_TX_FIFO_MOD<<I2S_I2S_TX_FIFO_MOD_S));
//set I2S_CHAN
//set rx,tx channel mode, both are "two channel" here
SET_PERI_REG_MASK(I2SCONF_CHAN, (I2S_TX_CHAN_MOD<<I2S_TX_CHAN_MOD_S)|(I2S_RX_CHAN_MOD<<I2S_RX_CHAN_MOD_S));
//set RX eof num
WRITE_PERI_REG(I2SRXEOF_NUM, RX_NUM);
//reset sample rate info
CLEAR_PERI_REG_MASK(I2SCONF, I2S_RECE_SLAVE_MOD|
(I2S_BITS_MOD<<I2S_BITS_MOD_S)|
(I2S_CLKM_DIV_NUM<<I2S_CLKM_DIV_NUM_S)|
(I2S_BCK_DIV_NUM <<I2S_BCK_DIV_NUM_S));
//use I2S clock divider to produce a 48KHz Sample Rate
//set transmit to be I2S slave & receive to be I2S master
//MSB_shift, right channel data is first, MSB is to the right
SET_PERI_REG_MASK(I2SCONF, I2S_TRANS_SLAVE_MOD|
I2S_RIGHT_FIRST|I2S_MSB_RIGHT|
I2S_RECE_MSB_SHIFT|I2S_TRANS_MSB_SHIFT|
((WS_I2S_BITS&I2S_BITS_MOD) << I2S_BITS_MOD_S)|
((WS_I2S_CLKM_DIV&I2S_CLKM_DIV_NUM) << I2S_CLKM_DIV_NUM_S)|
((WS_I2S_BCK_DIV&I2S_BCK_DIV_NUM ) << I2S_BCK_DIV_NUM_S));
//clear int
SET_PERI_REG_MASK(I2SINT_CLR, I2S_I2S_TX_REMPTY_INT_CLR|I2S_I2S_TX_WFULL_INT_CLR|I2S_I2S_RX_REMPTY_INT_CLR|
I2S_I2S_RX_WFULL_INT_CLR|I2S_I2S_PUT_DATA_INT_CLR|I2S_I2S_TAKE_DATA_INT_CLR);
CLEAR_PERI_REG_MASK(I2SINT_CLR, I2S_I2S_TX_REMPTY_INT_CLR|I2S_I2S_TX_WFULL_INT_CLR|I2S_I2S_RX_REMPTY_INT_CLR|
I2S_I2S_RX_WFULL_INT_CLR|I2S_I2S_PUT_DATA_INT_CLR|I2S_I2S_TAKE_DATA_INT_CLR);
//enable int
SET_PERI_REG_MASK(I2SINT_ENA, I2S_I2S_RX_REMPTY_INT_ENA|I2S_I2S_RX_WFULL_INT_ENA|I2S_I2S_RX_TAKE_DATA_INT_ENA);
// SET_PERI_REG_MASK(I2SINT_ENA, I2S_I2S_TX_REMPTY_INT_ENA|I2S_I2S_TX_WFULL_INT_ENA|I2S_I2S_TX_PUT_DATA_INT_ENA);
//Start transmitter and receiver
SET_PERI_REG_MASK(I2SCONF,I2S_I2S_RX_START); //I2S_I2S_TX_START|
}
void ICACHE_FLASH_ATTR i2s_stop(void) {
CLEAR_PERI_REG_MASK(I2SCONF,I2S_I2S_RX_START); //I2S_I2S_TX_START
}