Beware that this post is related to an SDK in maintenance mode
More Info: Consider nRF Connect SDK for new designs
This post is older than 2 years and might not be relevant anymore
More Info: Consider searching for newer posts

I2S change from writing in Register to driver

Hello,

I will use the I2S driver instead the direct acess to the registers.

I have implemented the registers version like this and it works / sounds normaly.

   main()
   {
   
        // Enable transmission
        NRF_I2S->CONFIG.TXEN = (I2S_CONFIG_TXEN_TXEN_ENABLE << I2S_CONFIG_TXEN_TXEN_Pos);
       // Enable MCK generator
       NRF_I2S->CONFIG.MCKEN = (I2S_CONFIG_MCKEN_MCKEN_ENABLE << I2S_CONFIG_MCKEN_MCKEN_Pos);
       // MCKFREQ
       NRF_I2S->CONFIG.MCKFREQ = I2S_CONFIG_MCKFREQ_MCKFREQ_32MDIV21  << I2S_CONFIG_MCKFREQ_MCKFREQ_Pos;
       // Ratio = 96
       NRF_I2S->CONFIG.RATIO = I2S_CONFIG_RATIO_RATIO_32X << I2S_CONFIG_RATIO_RATIO_Pos;    //64
       // Master mode, 16Bit, left aligned
       NRF_I2S->CONFIG.MODE = I2S_CONFIG_MODE_MODE_MASTER << I2S_CONFIG_MODE_MODE_Pos;
       NRF_I2S->CONFIG.SWIDTH = I2S_CONFIG_SWIDTH_SWIDTH_16BIT << I2S_CONFIG_SWIDTH_SWIDTH_Pos;
       NRF_I2S->CONFIG.ALIGN = I2S_CONFIG_ALIGN_ALIGN_LEFT << I2S_CONFIG_ALIGN_ALIGN_Pos;
       // Format = I2S
       NRF_I2S->CONFIG.FORMAT = I2S_CONFIG_FORMAT_FORMAT_I2S << I2S_CONFIG_FORMAT_FORMAT_Pos;
       // Use stereo
       //NRF_I2S->CONFIG.CHANNELS = I2S_CONFIG_CHANNELS_CHANNELS_STEREO << I2S_CONFIG_CHANNELS_CHANNELS_Pos;
       NRF_I2S->CONFIG.CHANNELS = I2S_CONFIG_CHANNELS_CHANNELS_Left << I2S_CONFIG_CHANNELS_CHANNELS_Pos;
       // Configure pins
       NRF_I2S->PSEL.MCK = (I2S_CONFIG_MCK_PIN << I2S_PSEL_MCK_PIN_Pos);
       NRF_I2S->PSEL.SCK = (I2S_CONFIG_SCK_PIN << I2S_PSEL_SCK_PIN_Pos);
       NRF_I2S->PSEL.LRCK = (I2S_CONFIG_LRCK_PIN << I2S_PSEL_LRCK_PIN_Pos);
       NRF_I2S->PSEL.SDOUT = (I2S_SDOUT_PIN << I2S_PSEL_SDOUT_PIN_Pos);
    
       NRF_I2S->ENABLE = 1;
       
        while(1)
        {
            ....
           
        
            if(NRF_I2S->EVENTS_TXPTRUPD  != 0)
            {
                 NRF_I2S->TXD.PTR = (uint32_t)OpusInstanz.pcm_bytes[bufferNr];
                 NRF_I2S->EVENTS_TXPTRUPD = 0;
                 newFrame = 1;
                 bufferNr ^= (1 << 0);
            }
              
              ....

now I have modified the code to the driver code but the output sound is slower than the original. Its like the handler is too slow.

static void i2sdata_handler(nrf_drv_i2s_buffers_t const * p_released, uint32_t status)
{
   ASSERT(p_released);
   if (!(status & NRFX_I2S_STATUS_NEXT_BUFFERS_NEEDED))
     return;

   nrf_drv_i2s_buffers_t const next_buffers = {
      .p_rx_buffer = NULL,
      .p_tx_buffer = (uint32_t *)OpusInstanz.pcm_bytes[bufferNr],
   };
   APP_ERROR_CHECK(nrf_drv_i2s_next_buffers_set(&next_buffers));

   newFrame = 1;
   bufferNr ^= (1 << 0);
}

main()
{
    ....
    

   nrf_drv_i2s_config_t config = NRF_DRV_I2S_DEFAULT_CONFIG;
   config.mck_setup = NRF_I2S_MCK_32MDIV21;
   config.ratio     = NRF_I2S_RATIO_32X;
   err = nrf_drv_i2s_init(&config, i2sdata_handler);
   APP_ERROR_CHECK(err);
   
   ...
    nrf_drv_i2s_buffers_t const initial_buffers = {
    .p_tx_buffer = (uint32_t *)OpusInstanz.pcm_bytes[bufferNr],
    .p_rx_buffer = NULL,
    };
    APP_ERROR_CHECK(nrf_drv_i2s_start(&initial_buffers, FRAME_SIZE/2, 0));
   

So I had checked the config but I can not find a mistake. I have watched the registers and they were the same in both variants.

Is there another mistake?

Parents
  • Its like the handler is too slow.

    How are you measuring this?

    Are you running other things than I2S in your application?

    Which SDK version are you using?

    What is NRFX_I2S_CONFIG_IRQ_PRIORITY set to in sdk_config.h?

    Have you tried the i2s example in the SDK?

    Best regards,

    Marjeris

  • Have you tried the i2s example in the SDK?

    I must point out that the I2S example in the SDK is of limited use for someone designing or troubleshooting a real world application. The problem is that you need an external codec chip in order to actually hear the output of the I2S module, and the Nordic NRF52 dev boards don't include one. Because of this, the SDK example can't do much except loop samples back between the TX and RX paths. This is not how most people are going to to use I2S.

    The one major difference between the "direct access to registers" and "driver" approach used here is that the former relies on polling the EVENTS_TXPTRUPD register while the latter relies on the interrupt that's supposedly generated when the EVENTS_TXPTRUPD register changes state.

    I say "supposedly" because in my own attempts to use the I2S module, it didn't seem to work that way. My expectation was that an interrupt event would be generated immediately once the I2S controller finished transferring the number of words specified in the RXTXD_MAXCNT register. I assumed that this would allow me to accurately detect when the current batch of samples had finished playing However that's not what appeared to actually happen. The contents of the EVENTS_TXPTRUPD did indeed seem to change at this point, such that polling this register produced the correct result, but the interrupt event seemed to occur at the wrong time. From what I could tell, it was triggering too early, with the result being that the output sounded garbled.

    I worked around this by using the PPI module. I tied the EVENTS_TXPTRUPD register to a PPI channel and then tied that channel to a software generated interrupt,  and then set things up so that the following would happen:

    - samples are queued up to play

    - the I2S EVENTS_TXPTRUPD interrupt occurs

    - the I2S interrupt handler acks the EVENTS_TXPTRUPD event and then enables the PPI channel

    - once the EVENTS_TXPTRUPD register actually changes state, the PPI channel triggers the software interrupt

    - the software interrupt is now triggered once the samples are actually done playing

    Below is a link to my driver code:

    https://github.com/netik/dc27_badge/blob/master/software/firmware/badge/nrf52i2s_lld.c

    https://github.com/netik/dc27_badge/blob/master/software/firmware/badge/nrf52i2s_lld.h

    I don't know if I was doing something wrong elsewhere that caused this condition to occur, but I experimented for quite some time trying to figure it out and in the end (like the poster above), I wasn't able to find where I was doing anything wrong. The manual seemed to strongly imply that the I2S module would generate an interrupt as soon as the current batch of samples was finished transferring, but that did not seem to be the case, and I nothing I did seemed to change that. My PPI workaround was good enough for my needs so I never followed up on it further. Also, since the Nordic DK boards do not have an audio codec chip on them, I wasn't sure that asking Nordic for help would do any good since there would be no easy way for them to reproduce my environment for testing. Otherwise, I would have asked:

    - How can one be sure exactly when the I2S interrupt fires? Is it supposed to trigger right when EVENTS_TXPTRUPD changes state or not? If not, when? If yes, have you guys actually tested it with a codec attached, and do you have an actual known good working example for playing audio samples that behaves correctly?

    Note that I used the ChibiOS RTOS in my application and did not use the Nordic SDK, however I don't think this had anything to do with the problem. The codec we used was a CS4344.

    -Bill

Reply
  • Have you tried the i2s example in the SDK?

    I must point out that the I2S example in the SDK is of limited use for someone designing or troubleshooting a real world application. The problem is that you need an external codec chip in order to actually hear the output of the I2S module, and the Nordic NRF52 dev boards don't include one. Because of this, the SDK example can't do much except loop samples back between the TX and RX paths. This is not how most people are going to to use I2S.

    The one major difference between the "direct access to registers" and "driver" approach used here is that the former relies on polling the EVENTS_TXPTRUPD register while the latter relies on the interrupt that's supposedly generated when the EVENTS_TXPTRUPD register changes state.

    I say "supposedly" because in my own attempts to use the I2S module, it didn't seem to work that way. My expectation was that an interrupt event would be generated immediately once the I2S controller finished transferring the number of words specified in the RXTXD_MAXCNT register. I assumed that this would allow me to accurately detect when the current batch of samples had finished playing However that's not what appeared to actually happen. The contents of the EVENTS_TXPTRUPD did indeed seem to change at this point, such that polling this register produced the correct result, but the interrupt event seemed to occur at the wrong time. From what I could tell, it was triggering too early, with the result being that the output sounded garbled.

    I worked around this by using the PPI module. I tied the EVENTS_TXPTRUPD register to a PPI channel and then tied that channel to a software generated interrupt,  and then set things up so that the following would happen:

    - samples are queued up to play

    - the I2S EVENTS_TXPTRUPD interrupt occurs

    - the I2S interrupt handler acks the EVENTS_TXPTRUPD event and then enables the PPI channel

    - once the EVENTS_TXPTRUPD register actually changes state, the PPI channel triggers the software interrupt

    - the software interrupt is now triggered once the samples are actually done playing

    Below is a link to my driver code:

    https://github.com/netik/dc27_badge/blob/master/software/firmware/badge/nrf52i2s_lld.c

    https://github.com/netik/dc27_badge/blob/master/software/firmware/badge/nrf52i2s_lld.h

    I don't know if I was doing something wrong elsewhere that caused this condition to occur, but I experimented for quite some time trying to figure it out and in the end (like the poster above), I wasn't able to find where I was doing anything wrong. The manual seemed to strongly imply that the I2S module would generate an interrupt as soon as the current batch of samples was finished transferring, but that did not seem to be the case, and I nothing I did seemed to change that. My PPI workaround was good enough for my needs so I never followed up on it further. Also, since the Nordic DK boards do not have an audio codec chip on them, I wasn't sure that asking Nordic for help would do any good since there would be no easy way for them to reproduce my environment for testing. Otherwise, I would have asked:

    - How can one be sure exactly when the I2S interrupt fires? Is it supposed to trigger right when EVENTS_TXPTRUPD changes state or not? If not, when? If yes, have you guys actually tested it with a codec attached, and do you have an actual known good working example for playing audio samples that behaves correctly?

    Note that I used the ChibiOS RTOS in my application and did not use the Nordic SDK, however I don't think this had anything to do with the problem. The codec we used was a CS4344.

    -Bill

Children
No Data
Related