HardwareSerial::_tx_udr_empty_irq()

Abstract

The HardwareSerial::_tx_udr_empty_irq() copies data from Arduino serial communication transmit buffer to ATmega328P data transmit register.

This function is called from HardwareSerial::flush(), HardwareSerial::write() and USART_UDRE_vect.

Source Code

The HardwareSerial::_tx_udr_empty_irq() is defined in hardware/arduino/avr/cores/arduino/HardwareSerial.cpp as below.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
void HardwareSerial::_tx_udr_empty_irq(void)
{
  // If interrupts are enabled, there must be more data in the output
  // buffer. Send the next byte
  unsigned char c = _tx_buffer[_tx_buffer_tail];
  _tx_buffer_tail = (_tx_buffer_tail + 1) % SERIAL_TX_BUFFER_SIZE;

  *_udr = c;

  // clear the TXC bit -- "can be cleared by writing a one to its bit
  // location". This makes sure flush() won't return until the bytes
  // actually got written. Other r/w bits are preserved, and zeroes
  // written to the rest.

  *_ucsra = ((*_ucsra) & ((1 << U2X0) | (1 << MPCM0))) | (1 << TXC0);

  if (_tx_buffer_head == _tx_buffer_tail) {
    // Buffer empty, so disable interrupts
    cbi(*_ucsrb, UDRIE0);
  }
}

No inputs, no outputs.

1
2
void HardwareSerial::_tx_udr_empty_irq(void)
{

Get the oldest data(_tx_buffer_tail-th data) from transmit buffer(_tx_buffer), substitutes to c.

3
4
5
6
7
8
  // If interrupts are enabled, there must be more data in the output
  // buffer. Send the next byte
  unsigned char c = _tx_buffer[_tx_buffer_tail];
  _tx_buffer_tail = (_tx_buffer_tail + 1) % SERIAL_TX_BUFFER_SIZE;
 
  *_udr = c;

Because we got a data out of the buffer, increment _tx_buffer_tail. Note _tx_buffer is a ring buffer, correct the value by calculating division reminder.

Put the c to _udr(UDRn), which is the addres of transmit buffer.

10
11
12
13
14
15
  // clear the TXC bit -- "can be cleared by writing a one to its bit
  // location". This makes sure flush() won't return until the bytes
  // actually got written. Other r/w bits are preserved, and zeroes
  // written to the rest.
 
  *_ucsra = ((*_ucsra) & ((1 << U2X0) | (1 << MPCM0))) | (1 << TXC0);

Set TXC0 bit of _ucsra(UCSR0A to clear TXC0 bit. This means unsent data exists.

15
16
17
18
19
  if (_tx_buffer_head == _tx_buffer_tail) {
    // Buffer empty, so disable interrupts
    cbi(*_ucsrb, UDRIE0);
  }
}

If the _tx_buffer_head and _tx_buffer_tail is same, it means the transmit buffer of Arduino is empty, set UDRIE0 bit of _ucsrb(UCSR0B register to 0, using cbi(), to disable interrupts.

Version

Arduino AVR Boards 1.8.6

Last Update

August 25, 2019

inserted by FC2 system