Skip to content

Commit bb341c6

Browse files
committed
Modifications to make serial transmit interrupt work more reliably. Also, added the availableForWrite function.
1 parent 4eb05c3 commit bb341c6

File tree

3 files changed

+30
-15
lines changed

3 files changed

+30
-15
lines changed

hardware/arduino/sam/cores/arduino/USARTClass.cpp

Lines changed: 23 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@
2323

2424
// Constructors ////////////////////////////////////////////////////////////////
2525

26-
USARTClass::USARTClass( Usart* pUsart, IRQn_Type dwIrq, uint32_t dwId, RingBuffer* pRx_buffer, volatile RingBuffer* pTx_buffer )
26+
USARTClass::USARTClass( Usart* pUsart, IRQn_Type dwIrq, uint32_t dwId, RingBuffer* pRx_buffer, RingBuffer* pTx_buffer )
2727
{
2828
_rx_buffer = pRx_buffer;
2929
_tx_buffer = pTx_buffer;
@@ -33,7 +33,6 @@ USARTClass::USARTClass( Usart* pUsart, IRQn_Type dwIrq, uint32_t dwId, RingBuffe
3333
_dwId=dwId ;
3434
}
3535

36-
3736
// Public Methods //////////////////////////////////////////////////////////////
3837

3938
void USARTClass::begin( const uint32_t dwBaudRate )
@@ -66,6 +65,10 @@ void USARTClass::begin( const uint32_t dwBaudRate, const uint32_t config )
6665
// Enable UART interrupt in NVIC
6766
NVIC_EnableIRQ( _dwIrq ) ;
6867

68+
//make sure both ring buffers are initialized back to empty.
69+
_rx_buffer->_iHead = _rx_buffer->_iTail = 0;
70+
_tx_buffer->_iHead = _tx_buffer->_iTail = 0;
71+
6972
// Enable receiver and transmitter
7073
_pUsart->US_CR = US_CR_RXEN | US_CR_TXEN ;
7174
}
@@ -91,6 +94,14 @@ int USARTClass::available( void )
9194
return (uint32_t)(SERIAL_BUFFER_SIZE + _rx_buffer->_iHead - _rx_buffer->_iTail) % SERIAL_BUFFER_SIZE ;
9295
}
9396

97+
int USARTClass::availableForWrite(void)
98+
{
99+
int head = _tx_buffer->_iHead;
100+
int tail = _tx_buffer->_iTail;
101+
if (head >= tail) return SERIAL_BUFFER_SIZE - 1 - head + tail;
102+
return tail - head - 1;
103+
}
104+
94105
int USARTClass::peek( void )
95106
{
96107
if ( _rx_buffer->_iHead == _rx_buffer->_iTail )
@@ -142,17 +153,20 @@ void USARTClass::IrqHandler( void )
142153
uint32_t status = _pUsart->US_CSR;
143154

144155
// Did we receive data ?
145-
if ((status & US_CSR_RXRDY) == US_CSR_RXRDY)
146-
_rx_buffer->store_char( _pUsart->US_RHR ) ;
147-
156+
if ((status & US_CSR_RXRDY) == US_CSR_RXRDY)
157+
{
158+
_rx_buffer->store_char(_pUsart->US_RHR);
159+
}
148160
//Do we need to keep sending data?
149161
if ((status & US_CSR_TXRDY) == US_CSR_TXRDY)
150162
{
151-
_pUsart->US_THR = _tx_buffer->_aucBuffer[_tx_buffer->_iTail];
152-
_tx_buffer->_iTail = (unsigned int)(_tx_buffer->_iTail + 1) % SERIAL_BUFFER_SIZE;
153-
if (_tx_buffer->_iTail == _tx_buffer->_iHead) //if this is true we have no more data to transmit
163+
if (_tx_buffer->_iTail != _tx_buffer->_iHead) { //just in case
164+
_pUsart->US_THR = _tx_buffer->_aucBuffer[_tx_buffer->_iTail];
165+
_tx_buffer->_iTail = (unsigned int)(_tx_buffer->_iTail + 1) % SERIAL_BUFFER_SIZE;
166+
}
167+
else
154168
{
155-
_pUsart->US_IDR = US_IDR_TXRDY; //mask off transmit interrupt so we don't get it anymore
169+
_pUsart->US_IDR = US_IDR_TXRDY; //mask off transmit interrupt so we don't get it anymore
156170
}
157171
}
158172

hardware/arduino/sam/cores/arduino/USARTClass.h

Lines changed: 4 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -59,21 +59,22 @@
5959
class USARTClass : public HardwareSerial
6060
{
6161
protected:
62-
RingBuffer *_rx_buffer ;
63-
volatile RingBuffer *_tx_buffer;
62+
RingBuffer *_rx_buffer;
63+
RingBuffer *_tx_buffer;
6464

6565
protected:
6666
Usart* _pUsart ;
6767
IRQn_Type _dwIrq ;
6868
uint32_t _dwId ;
6969

7070
public:
71-
USARTClass( Usart* pUsart, IRQn_Type dwIrq, uint32_t dwId, RingBuffer* pRx_buffer, volatile RingBuffer* pTx_buffer ) ;
71+
USARTClass( Usart* pUsart, IRQn_Type dwIrq, uint32_t dwId, RingBuffer* pRx_buffer, RingBuffer* pTx_buffer ) ;
7272

7373
void begin( const uint32_t dwBaudRate ) ;
7474
void begin( const uint32_t dwBaudRate , const uint32_t config ) ;
7575
void end( void ) ;
7676
int available( void ) ;
77+
int availableForWrite(void);
7778
int peek( void ) ;
7879
int read( void ) ;
7980
void flush( void ) ;

hardware/arduino/sam/variants/arduino_due_x/variant.cpp

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -317,9 +317,9 @@ void UART_Handler(void)
317317
RingBuffer rx_buffer2;
318318
RingBuffer rx_buffer3;
319319
RingBuffer rx_buffer4;
320-
volatile RingBuffer tx_buffer2;
321-
volatile RingBuffer tx_buffer3;
322-
volatile RingBuffer tx_buffer4;
320+
RingBuffer tx_buffer2;
321+
RingBuffer tx_buffer3;
322+
RingBuffer tx_buffer4;
323323

324324
USARTClass Serial1(USART0, USART0_IRQn, ID_USART0, &rx_buffer2, &tx_buffer2);
325325
void serialEvent1() __attribute__((weak));

0 commit comments

Comments
 (0)