214 lines
5 KiB
C++
214 lines
5 KiB
C++
/*
|
|
Copyright (c) 2015 Arduino LLC. All right reserved.
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
See the GNU Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with this library; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include "Uart.h"
|
|
#include "Arduino.h"
|
|
#include "wiring_private.h"
|
|
|
|
#define NO_RTS_PIN 255
|
|
#define NO_CTS_PIN 255
|
|
#define RTS_RX_THRESHOLD 10
|
|
|
|
Uart::Uart(SERCOM *_s, uint8_t _pinRX, uint8_t _pinTX, SercomRXPad _padRX, SercomUartTXPad _padTX) :
|
|
Uart(_s, _pinRX, _pinTX, _padRX, _padTX, NO_RTS_PIN, NO_CTS_PIN)
|
|
{
|
|
}
|
|
|
|
Uart::Uart(SERCOM *_s, uint8_t _pinRX, uint8_t _pinTX, SercomRXPad _padRX, SercomUartTXPad _padTX, uint8_t _pinRTS, uint8_t _pinCTS)
|
|
{
|
|
sercom = _s;
|
|
uc_pinRX = _pinRX;
|
|
uc_pinTX = _pinTX;
|
|
uc_padRX = _padRX ;
|
|
uc_padTX = _padTX;
|
|
uc_pinRTS = _pinRTS;
|
|
uc_pinCTS = _pinCTS;
|
|
}
|
|
|
|
void Uart::begin(unsigned long baudrate)
|
|
{
|
|
begin(baudrate, SERIAL_8N1);
|
|
}
|
|
|
|
void Uart::begin(unsigned long baudrate, uint16_t config)
|
|
{
|
|
pinPeripheral(uc_pinRX, g_APinDescription[uc_pinRX].ulPinType);
|
|
pinPeripheral(uc_pinTX, g_APinDescription[uc_pinTX].ulPinType);
|
|
|
|
if (uc_padTX == UART_TX_RTS_CTS_PAD_0_2_3) {
|
|
if (uc_pinCTS != NO_CTS_PIN) {
|
|
pinPeripheral(uc_pinCTS, g_APinDescription[uc_pinCTS].ulPinType);
|
|
}
|
|
}
|
|
|
|
if (uc_pinRTS != NO_RTS_PIN) {
|
|
pinMode(uc_pinRTS, OUTPUT);
|
|
|
|
EPortType rtsPort = g_APinDescription[uc_pinRTS].ulPort;
|
|
pul_outsetRTS = &PORT->Group[rtsPort].OUTSET.reg;
|
|
pul_outclrRTS = &PORT->Group[rtsPort].OUTCLR.reg;
|
|
ul_pinMaskRTS = (1ul << g_APinDescription[uc_pinRTS].ulPin);
|
|
|
|
*pul_outclrRTS = ul_pinMaskRTS;
|
|
}
|
|
|
|
sercom->initUART(UART_INT_CLOCK, SAMPLE_RATE_x16, baudrate);
|
|
sercom->initFrame(extractCharSize(config), LSB_FIRST, extractParity(config), extractNbStopBit(config));
|
|
sercom->initPads(uc_padTX, uc_padRX);
|
|
|
|
sercom->enableUART();
|
|
}
|
|
|
|
void Uart::end()
|
|
{
|
|
sercom->resetUART();
|
|
rxBuffer.clear();
|
|
txBuffer.clear();
|
|
}
|
|
|
|
void Uart::flush()
|
|
{
|
|
while(txBuffer.available()); // wait until TX buffer is empty
|
|
|
|
sercom->flushUART();
|
|
}
|
|
|
|
void Uart::IrqHandler()
|
|
{
|
|
if (sercom->availableDataUART()) {
|
|
rxBuffer.store_char(sercom->readDataUART());
|
|
|
|
if (uc_pinRTS != NO_RTS_PIN) {
|
|
// RX buffer space is below the threshold, de-assert RTS
|
|
if (rxBuffer.availableForStore() < RTS_RX_THRESHOLD) {
|
|
*pul_outsetRTS = ul_pinMaskRTS;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (sercom->isDataRegisterEmptyUART()) {
|
|
if (txBuffer.available()) {
|
|
uint8_t data = txBuffer.read_char();
|
|
|
|
sercom->writeDataUART(data);
|
|
} else {
|
|
sercom->disableDataRegisterEmptyInterruptUART();
|
|
}
|
|
}
|
|
|
|
if (sercom->isUARTError()) {
|
|
sercom->acknowledgeUARTError();
|
|
// TODO: if (sercom->isBufferOverflowErrorUART()) ....
|
|
// TODO: if (sercom->isFrameErrorUART()) ....
|
|
// TODO: if (sercom->isParityErrorUART()) ....
|
|
sercom->clearStatusUART();
|
|
}
|
|
}
|
|
|
|
int Uart::available()
|
|
{
|
|
return rxBuffer.available();
|
|
}
|
|
|
|
int Uart::availableForWrite()
|
|
{
|
|
return txBuffer.availableForStore();
|
|
}
|
|
|
|
int Uart::peek()
|
|
{
|
|
return rxBuffer.peek();
|
|
}
|
|
|
|
int Uart::read()
|
|
{
|
|
int c = rxBuffer.read_char();
|
|
|
|
if (uc_pinRTS != NO_RTS_PIN) {
|
|
// if there is enough space in the RX buffer, assert RTS
|
|
if (rxBuffer.availableForStore() > RTS_RX_THRESHOLD) {
|
|
*pul_outclrRTS = ul_pinMaskRTS;
|
|
}
|
|
}
|
|
|
|
return c;
|
|
}
|
|
|
|
size_t Uart::write(const uint8_t data)
|
|
{
|
|
if (sercom->isDataRegisterEmptyUART() && txBuffer.available() == 0) {
|
|
sercom->writeDataUART(data);
|
|
} else {
|
|
while(txBuffer.isFull()); // spin lock until a spot opens up in the buffer
|
|
|
|
txBuffer.store_char(data);
|
|
|
|
sercom->enableDataRegisterEmptyInterruptUART();
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
SercomNumberStopBit Uart::extractNbStopBit(uint16_t config)
|
|
{
|
|
switch(config & HARDSER_STOP_BIT_MASK)
|
|
{
|
|
case HARDSER_STOP_BIT_1:
|
|
default:
|
|
return SERCOM_STOP_BIT_1;
|
|
|
|
case HARDSER_STOP_BIT_2:
|
|
return SERCOM_STOP_BITS_2;
|
|
}
|
|
}
|
|
|
|
SercomUartCharSize Uart::extractCharSize(uint16_t config)
|
|
{
|
|
switch(config & HARDSER_DATA_MASK)
|
|
{
|
|
case HARDSER_DATA_5:
|
|
return UART_CHAR_SIZE_5_BITS;
|
|
|
|
case HARDSER_DATA_6:
|
|
return UART_CHAR_SIZE_6_BITS;
|
|
|
|
case HARDSER_DATA_7:
|
|
return UART_CHAR_SIZE_7_BITS;
|
|
|
|
case HARDSER_DATA_8:
|
|
default:
|
|
return UART_CHAR_SIZE_8_BITS;
|
|
|
|
}
|
|
}
|
|
|
|
SercomParityMode Uart::extractParity(uint16_t config)
|
|
{
|
|
switch(config & HARDSER_PARITY_MASK)
|
|
{
|
|
case HARDSER_PARITY_NONE:
|
|
default:
|
|
return SERCOM_NO_PARITY;
|
|
|
|
case HARDSER_PARITY_EVEN:
|
|
return SERCOM_EVEN_PARITY;
|
|
|
|
case HARDSER_PARITY_ODD:
|
|
return SERCOM_ODD_PARITY;
|
|
}
|
|
}
|