voidHardwareSerial::_tx_udr_empty_irq(void){// If interrupts are enabled, there must be more data in the output
// buffer. Send the next byte
unsignedcharc=_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
voidHardwareSerial::_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
unsignedcharc=_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.