uart/uart.hpp

390 lines
8.0 KiB
C++
Raw Permalink Normal View History

#pragma once
2022-05-29 16:15:11 +02:00
#include <limits>
#include <cstdint>
2019-08-03 15:00:41 +02:00
2019-07-28 12:15:19 +02:00
#include "config.hpp"
#include "software.hpp"
#include "hardware0.hpp"
#include "hardware1.hpp"
#include "../flash/flash.hpp"
2022-05-29 16:15:11 +02:00
#include "../util/util.hpp"
namespace uart {
2019-08-03 15:00:41 +02:00
namespace detail {
2022-05-29 16:15:11 +02:00
template <typename T, T Limit, std::size_t Base>
static constexpr std::size_t cntDigits()
2019-08-03 15:00:41 +02:00
{
2020-04-05 03:23:12 +02:00
T num = Limit;
2022-05-29 16:15:11 +02:00
std::size_t cnt = 0;
2019-08-03 15:00:41 +02:00
do {
num /= Base;
2019-08-03 15:00:41 +02:00
++cnt;
} while (num > 0);
return cnt;
}
2022-05-29 16:15:11 +02:00
template <typename T, std::size_t Base>
static constexpr std::size_t maxNumDigits()
2019-08-03 15:00:41 +02:00
{
2022-05-29 16:15:11 +02:00
constexpr T MinVal = std::numeric_limits<T>::min();
constexpr T MaxVal = std::numeric_limits<T>::max();
constexpr T MinDigits = cntDigits<T, MinVal, Base>();
constexpr T MaxDigits = cntDigits<T, MaxVal, Base>();
2019-08-03 15:00:41 +02:00
return (MinDigits < MaxDigits) ? MaxDigits : MinDigits;
2019-08-03 15:00:41 +02:00
}
} // namespace detail
2019-07-28 12:15:19 +02:00
template <class Driver>
class Uart {
public:
using data_t = typename Driver::data_t;
// Constructing a uart object does not initialize the driver to allow different specializations with the same
// back-end to exists at the same time
// Note that init must be called every time when switching specializations with the same back-end
Uart() = default;
// Moving and copying uart objects is not supported
Uart(const Uart &) = delete;
Uart(Uart &&) = delete;
Uart &operator=(const Uart &) = delete;
Uart &operator=(Uart &&) = delete;
// Before using the uart init must be called
static void init()
{
Driver::init();
}
static void txByte(const data_t &byte)
{
2019-07-28 12:15:19 +02:00
Driver::txByte(byte);
}
static bool rxByte(data_t &byte)
{
return Driver::rxByte(byte);
}
static bool peek(data_t &byte)
{
return Driver::peek(byte);
}
static bool peek()
{
2019-07-28 12:15:19 +02:00
return Driver::peek();
}
2019-08-03 17:52:28 +02:00
static void flushTx()
{
Driver::flushTx();
}
static void txString(const char *str)
{
2019-07-28 12:15:19 +02:00
static_assert(Driver::DATA_BITS == DataBits::EIGHT, "Strings are only supported with 8 data bits");
while (char ch = *str++)
txByte(ch);
}
static void txString(const ::detail::FlashString *str)
{
2019-07-28 12:15:19 +02:00
static_assert(Driver::DATA_BITS == DataBits::EIGHT, "Strings are only supported with 8 data bits");
const char *strIt = reinterpret_cast<const char *>(str);
while (char ch = pgm_read_byte(strIt++))
txByte(ch);
}
2022-05-29 16:15:11 +02:00
template <typename T, std::size_t Base = 10, std::size_t Padding = 0, char PadChar = '0', bool LowerCase = true>
static void txNumber(const T &val)
2019-08-03 15:00:41 +02:00
{
2022-05-29 16:15:11 +02:00
static_assert(std::is_integral_v<T>, "Only supported on integral types");
static_assert(Base >= 2, "Numbers with base less than 2 make no sense");
static_assert(Base <= 16, "Numbers with base higher than 16 are not supported");
static_assert(Padding <= detail::maxNumDigits<T, Base>(), "Cannot pad more than maximum length of number");
2019-08-03 15:00:41 +02:00
constexpr char AlphaChar = (LowerCase) ? 'a' : 'A';
2022-05-29 16:15:11 +02:00
constexpr std::size_t NumDigits = detail::maxNumDigits<T, Base>();
2019-08-03 15:00:41 +02:00
T digits = val;
if (digits < 0) {
digits = -digits;
txByte('-');
}
data_t buffer[NumDigits];
data_t *bufEnd = buffer + NumDigits - 1;
2020-04-05 03:28:45 +02:00
2019-08-03 15:00:41 +02:00
do {
2020-04-05 03:30:22 +02:00
const data_t lastDigit = digits % Base;
*bufEnd-- = (lastDigit < 10) ? ('0' + lastDigit) : (AlphaChar + lastDigit - 10);
2019-08-03 15:00:41 +02:00
digits /= Base;
} while (digits > 0);
if (Padding > 0) {
2022-05-29 16:15:11 +02:00
std::size_t strLen = buffer + NumDigits - (bufEnd + 1);
if (Padding > strLen) {
2022-05-29 16:15:11 +02:00
for (std::size_t i = Padding; i > strLen && bufEnd >= buffer; --i) {
*bufEnd-- = PadChar;
}
}
}
for (data_t *buf = bufEnd + 1; buf < buffer + NumDigits; ++buf)
2019-08-03 15:00:41 +02:00
txByte(*buf);
}
//////////////////////////////////////////////////////////////////////////
// Output stream overloads
Uart &operator<<(const char *str)
{
txString(str);
return *this;
}
Uart &operator<<(const ::detail::FlashString *str)
{
txString(str);
return *this;
}
Uart &operator<<(const char &val)
{
txByte(val);
return *this;
}
Uart &operator<<(const signed char &val)
{
txNumber(val);
return *this;
}
Uart &operator<<(const unsigned char &val)
{
txNumber(val);
return *this;
}
Uart &operator<<(const short &val)
{
txNumber(val);
return *this;
}
2020-04-06 21:44:28 +02:00
Uart &operator<<(const unsigned short &val)
{
txNumber(val);
return *this;
}
2019-08-03 15:00:41 +02:00
Uart &operator<<(const int &val)
{
2019-08-03 15:00:41 +02:00
txNumber(val);
return *this;
}
Uart &operator<<(const unsigned int &val)
{
txNumber(val);
return *this;
}
Uart &operator<<(const long &val)
{
txNumber(val);
return *this;
}
2020-04-06 21:44:28 +02:00
Uart &operator<<(const unsigned long &val)
{
txNumber(val);
return *this;
}
2020-04-06 21:44:28 +02:00
Uart &operator<<(const long long &val)
{
txNumber(val);
return *this;
}
2020-04-06 21:44:28 +02:00
Uart &operator<<(const unsigned long long &val)
{
txNumber(val);
return *this;
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator<<(float) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not supported by hardware");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator<<(double) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not supported by hardware");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator<<(long double) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not supported by hardware");
}
Uart &operator<<(const bool &val)
{
txString(val ? F("true") : F("false"));
return *this;
}
Uart &operator<<(const void *val)
{
txString(F("0x"));
2022-05-29 16:15:11 +02:00
txNumber<std::uint16_t, 16, 4, '0', false>(reinterpret_cast<std::uint16_t>(val));
return *this;
}
//////////////////////////////////////////////////////////////////////////
// Input stream overloads
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(char &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(unsigned char &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(short &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(unsigned short &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(int &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(unsigned int &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(long &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(unsigned long &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(long long &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(unsigned long long &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(float &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not supported by hardware");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(double &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not supported by hardware");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(long double &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not supported by hardware");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(bool &) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
2019-07-28 10:35:11 +02:00
template <typename... Ts>
2019-08-03 15:00:41 +02:00
Uart &operator>>(const void *&) const
{
2020-05-16 17:42:52 +02:00
static_assert(util::always_false_v<Ts...>, "Not implemented");
}
private:
friend void ::USART0_RX_vect();
friend void ::USART0_UDRE_vect();
#ifdef HAS_UART1
friend void ::USART1_RX_vect();
friend void ::USART1_UDRE_vect();
#endif
2022-05-29 16:15:11 +02:00
[[gnu::always_inline]] static void rxIntHandler()
{
Driver::rxIntHandler();
}
2022-05-29 16:15:11 +02:00
[[gnu::always_inline]] static void dataRegEmptyIntHandler()
{
Driver::dataRegEmptyIntHandler();
}
};
2019-07-28 17:58:23 +02:00
template <typename cfg = Config<>>
2019-08-05 20:05:59 +02:00
using Uart0 = Uart<Hardware0<cfg, Driven::INTERRUPT, Mode::ASYNCHRONOUS>>;
2019-07-28 17:58:23 +02:00
#ifdef HAS_UART1
template <typename cfg = Config<>>
2019-08-05 20:05:59 +02:00
using Uart1 = Uart<Hardware1<cfg, Driven::INTERRUPT, Mode::ASYNCHRONOUS>>;
#endif
} // namespace uart
#undef HAS_UART1