This post is older than 2 years and might not be relevant anymore
More Info: Consider searching for newer posts

Use SPI Master within NUS Data Handler

Hi,

I can use the SPI master functionality within nrf_drv_spi.c/.h just fine (i.e. calls from main) until I connect to my nRF52840 and utilize NUS. If I try to issue a nrf_drv_spi_transfer() call from within my NUS data handler (i.e. when handing incoming data over NUS) to read 2 registers from a SPI connected slave (I am the master), I am not getting the expected SPI event handler with event NRF_DRV_SPI_EVENT_DONE when a SPI transfer completes. In the code below, I am calling bmi160_spiReadRegs() from within my NUS data handler. This function is blocking because m_xfer_done is never set to TRUE as a result of spi_event_handler() being called.

void spi_event_handler(nrf_drv_spi_evt_t const * p_event,
                       void *                    p_context)
{
  // The only type passed to this handler is "NRF_DRV_SPI_EVENT_DONE"
  m_xfer_done = true;  
}

void spi_init()
{
  ret_code_t err_code;

  if(m_initialized) { return; } // Nothing to do.

  nrf_drv_spi_config_t spi_config = NRF_DRV_SPI_DEFAULT_CONFIG;
  spi_config.ss_pin   = ACC_SPI_CSN;          // Driver control.
  spi_config.miso_pin = ACC_SPI_MISO;
  spi_config.mosi_pin = ACC_SPI_MOSI;
  spi_config.sck_pin  = ACC_SPI_SCK;
  spi_config.mode     = NRF_DRV_SPI_MODE_0;   // Prev BMI160 exp.
  spi_config.frequency = NRF_DRV_SPI_FREQ_8M; // Max Nordic SPI Speed - chip max is 10M
  spi_config.bit_order = NRF_DRV_SPI_BIT_ORDER_MSB_FIRST; // From prev egs.
  err_code = nrf_drv_spi_init(&spi, &spi_config, spi_event_handler, NULL);
  APP_ERROR_CHECK(err_code);

  m_initialized = true;
}

///////////////////////////////////////////////////////////////////////////
// BMI160.c slave functions for read/write via SPI. Fcn signature is set.
///////////////////////////////////////////////////////////////////////////

int8_t bmi160_spiReadRegs(uint8_t dev_addr, uint8_t reg_addr, uint8_t *p_data, short unsigned int data_len)
{
  ret_code_t err_code;
  uint8_t* p_newRxBuf;

  // Watch for this to change to true in handler.
  m_xfer_done = false;

  // Cannot continue if not initialized.
  if(!m_initialized) { return 1; }

  // Add read bit to register. 
  reg_addr = reg_addr | SPI_READ_MASK;

  // Create a rx buffer that has an extra byte for the response (first byte back is ignored as per egs).
  p_newRxBuf = malloc(data_len + 1);
  memset(p_newRxBuf, 0, data_len+1);

  // Read directly into p_data. nrf_drv_spi_transfer() should hold down cs to ensable values from all regs.
  // as specified by data_len.
  err_code = nrf_drv_spi_transfer(&spi, &reg_addr, 1, p_newRxBuf, data_len+1);
  APP_ERROR_CHECK(err_code);

  // Wait for a result.
  while(!m_xfer_done) { nrf_delay_ms(1); }

  // Copy Rx data into p_data
  memcpy(p_data, p_newRxBuf+1, data_len);

  free(p_newRxBuf);
  
  return 0;
}

If I issue the same SPI read on main using nrf_drv_spi_transfer(), the SPI event handler is called with event NRF_DRV_SPI_EVENT_DONE as expected. But, if I issue the nrf_drv_spi_transfer() from my NUS data handler, my SPI event handler is never called (at all).

I see other references to similar issues on DevZone. For these issues, NUS or SPI priority changes are suggested. I have tried this, but to no avail. 

Specifically, I have set NRFX_SPI_DEFAULT_CONFIG_IRQ_PRIORITY to 2 to match NUS, but to no avail. See below.

 

// <o> NRFX_SPI_DEFAULT_CONFIG_IRQ_PRIORITY  - Interrupt priority
 
// <0=> 0 (highest) 
// <1=> 1 
// <2=> 2 
// <3=> 3 
// <4=> 4 
// <5=> 5 
// <6=> 6 default
// <7=> 7 

#ifndef NRFX_SPI_DEFAULT_CONFIG_IRQ_PRIORITY
#define NRFX_SPI_DEFAULT_CONFIG_IRQ_PRIORITY 2
#endif
 

I have also tried changing the priority of NUS to match SPI. I set it to 6 as below.

// <o> BLE_NUS_BLE_OBSERVER_PRIO  
// <i> Priority with which BLE events are dispatched to the UART Service.
// Default 2

#ifndef BLE_NUS_BLE_OBSERVER_PRIO
#define BLE_NUS_BLE_OBSERVER_PRIO 6
#endif

Neither of these addressed the issue. If I call nrf_drv_spi_transfer() within my NUS data handler, the SPI event handler is never called with event NRF_DRV_SPI_EVENT_DONE.

This confuses me as the NUS UART example calls UART reads/writes within the NUS data handler and the UART priority is the same as SPI (both 6).

Perhaps this is not priority related?

Help, clues, etc are appreciated.

Thanks,

Mark J

Related