This thread has been locked.

If you have a related question, please click the "Ask a related question" button in the top right corner. The newly created question will be automatically linked to this question.

MSP432P401R: UART A3: Detecting Dropped Bytes (UCOE) when sending and receiving simultaneously

Part Number: MSP432P401R

When sending and receiving simultaneously on the UART using interrupts, I am detecting UCOE receive buffer overruns at an unacceptable rate (several per minute). 

No other interrupts are enabled, this is the only task being performed by this device. The baud rate is 115.2Kbps on UART3. All bytes that do come through are correct.

See code below. My development environment is IAR ARM 7.80.3. Any help would be greatly appreciated, thanks -David

//////INIT CODE BEGIN//////////////////
const eUSCI_UART_Config uartA3Config =
{
        EUSCI_A_UART_CLOCKSOURCE_SMCLK,         
        13,                                      // BRDIV = 13
        0,                                       // UCxBRF = 0
        37,                                       // UCxBRS = 0
        EUSCI_A_UART_NO_PARITY,                  // No Parity
        EUSCI_A_UART_LSB_FIRST,                  // LSB First
        EUSCI_A_UART_ONE_STOP_BIT,               // One stop bit
        EUSCI_A_UART_MODE,                      // UART mode
        EUSCI_A_UART_OVERSAMPLING_BAUDRATE_GENERATION  // Oversampling
};
    P9DIR |=  BIT7;
    MAP_GPIO_setAsPeripheralModuleFunctionOutputPin( GPIO_PORT_P9,
                                                    GPIO_PIN7,
                                                   GPIO_PRIMARY_MODULE_FUNCTION);
    P9DIR &=~BIT6;
    MAP_GPIO_setAsPeripheralModuleFunctionInputPin( GPIO_PORT_P9,
                                                   GPIO_PIN6,
                                                   GPIO_PRIMARY_MODULE_FUNCTION);
    MAP_UART_initModule(EUSCI_A3_MODULE, &uartA3Config);
    MAP_UART_enableModule(EUSCI_A3_MODULE);
////// INIT CODE END//////////////////

//////INTERRUPT HANDLER BEGIN //////////

char test_buf[256] = {0};
char buf_beg = 0;
char buf_end = 0;
unsigned short test_crc = 0;

int overflowCounter = 0;
int rxErrCounter = 0;
unsigned int count = 0;
// UART A2: Fast Steering Mirror Comm
void UARTa3_isr(void)
  {   
  if(UCA3IFG & UCRXIFG)
    {    
    if ((UCA3STATW & UCOE) != 0)
      {
      overflowCounter++;
      }
    
    if ((UCA3STATW & UCRXERR) != 0)
      {
      rxErrCounter++;
      }
    
    inbufS15[ibufS15] = UCA3RXBUF;
    ibufS15++;   
    if(ibufS15 >= INBUFS15LGTH) 
      {
      ibufS15 = 0;
      s15_charcount+=512;
      }         
    }
  
  if(UCA3IFG & UCTXIFG)
    {          
    if (buf_beg < buf_end)
      {      
      UCA3TXBUF = test_buf[buf_beg++];
      }
    else
      {      
      UCA3IE &= ~UCTXIE;
      UCA3IFG &= ~UCTXIFG;
      buf_beg = 0;
      buf_end = 0;
      }
    }
}
//////INTERRUPT HANDLER END //////////

//////MAIN LOOP BEGIN //////////
    while (1)
      {      
      if ((buf_beg < buf_end) && !(UCA3IE & UCTXIE))
        {
        UCA3IE |= UCTXIE;
        UCA3TXBUF = test_buf[buf_beg++];
        }
      
      ProcessInputBytes(); 
      
      if (buf_beg == 0 && buf_end == 0)
        {
        memset(test_buf, 0x00, sizeof(test_buf));
        
        test_buf[buf_end++] = count;
        test_buf[buf_end++] = count >> 8;
        test_buf[buf_end++] = count >> 16;
        test_buf[buf_end++] = count >> 24;
        count++;
        }
      }
    
//////MAIN LOOP END /////////////

  • Additional note: If I disable transmits from the device and only receive data, I never miss a byte and UCOE is never set. Only when both are happening simultaneously do I get UCOE errors.
  • > UCA3IFG &= ~UCTXIFG;
    I recommend that you not do this. Turning off the TXIE will suffice to keep things quiet.

    Doing this invites a Read/Modify/Write race that can lose nascent status (like RXIFG). This isn't a theoretical concern -- maybe 10 people here have tripped over it in the past year.
  • Bruce, I removed the offending line and, as a result, also had to remove the following line in the main loop:
    > UCA3TXBUF = test_buf[buf_beg++];

    This appears to have fixed it! I have no more dropped bytesin either direction. If possible, could you explain to me a little more about what the read/modify/write race condition is and why it was causing me to occasionally fail to service the receive interrupt?
  • The Cortex-M4 (Thumb2) doesn't have an atomic bit-set instruction, so the software must load, modify, then store the IFG register in 3 steps. If an RXIFG (or TXIFG, which we've also seen) arrives in the middle, it gets lost. You never saw it, though as far as the USCI is concerned you forgot (UCOE).

    Bit-banding doesn't help in this case, since it also does a R/M/W on the target word. It does it inside the CPU, which saves you from (other) software, but it's still two discrete bus transactions, so the USCI can change state in between.

    As you might guess, the hazard is greatest at high bit rates, but it's always there (as you've seen).
  • Thanks for the reply, that makes total sense. My background is in MSP430 development, does the MSP430 have the same vulnerability? In other words, is it safe to do UCA3IFG &= ~UCTXIFG/UCRXIFG on an MSP430, or would that cause similar problems to the MSP432?
  • For the MSP430, I don't know the answer. BIC/BIS are described in RMW terms, though there may be something unwritten that guarantees per-bit atomicity. Also: The compiler seems to always generate these for (&=,|=), but I've never seen a guarantee of that either.

    I wasn't sure I wanted to find out (by accident), so I've made a practice of not (ever) manipulating the USCI RXIFG/TXIFG directly, for either 430 or 432.

**Attention** This is a public forum