Skip to content

Commit

Permalink
[uart] Refactor SAM UART driver
Browse files Browse the repository at this point in the history
  • Loading branch information
salkinium committed Feb 20, 2021
1 parent 5d9c2a7 commit 0e423af
Show file tree
Hide file tree
Showing 6 changed files with 56 additions and 72 deletions.
12 changes: 1 addition & 11 deletions src/modm/platform/uart/sam/module.lb
Original file line number Diff line number Diff line change
Expand Up @@ -30,9 +30,6 @@ class Instance(Module):
device = env[":target"].identifier
global props
props["id"] = self.instance
props["driver"] = self.driver
props["features"] = self.driver["feature"] if "feature" in self.driver else []
props["uart_name"] = 'Uart'
props["sercom_name"] = self.driver["name"].capitalize()

env.substitutions = props
Expand All @@ -52,7 +49,7 @@ def init(module):

def prepare(module, options):
device = options[":target"]
if not (device.has_driver("sercom:*")):
if not (device.has_driver("sercom:sam")):
return False

module.depends(
Expand All @@ -67,11 +64,6 @@ def prepare(module, options):

global props
drivers = options[":target"].get_all_drivers("sercom")
props["extended_driver"] = ("extended" in drivers[0]["type"])
props["over8_sampling"] = ("feature" in drivers[0]) and ("over8" in drivers[0]["feature"])
props["tcbgt"] = ("feature" in drivers[0]) and ("tcbgt" in drivers[0]["feature"])
props["instances"] = []

for driver in drivers:
for instance in driver["instance"]:
module.add_submodule(Instance(driver, instance))
Expand All @@ -80,8 +72,6 @@ def prepare(module, options):
return True

def build(env):
device = env[":target"]

global props
env.substitutions = props
env.outbasepath = "modm/src/modm/platform/uart"
Expand Down
29 changes: 17 additions & 12 deletions src/modm/platform/uart/sam/uart.cpp.in
Original file line number Diff line number Diff line change
Expand Up @@ -16,36 +16,39 @@
*/
// ----------------------------------------------------------------------------

%% set name = uart_name ~ id
%% set hal = uart_name ~ "Hal" ~ id
%% set name = "Uart" ~ id
%% set hal = "UartHal" ~ id

#include "../device.hpp"
#include "uart_hal_{{ id }}.hpp"
#include "uart_{{ id }}.hpp"

namespace modm::platform
{

void
modm::platform::{{ name }}::writeBlocking(uint8_t data)
{{ name }}::writeBlocking(uint8_t data)
{
while(!{{ hal }}::isTransmitRegisterEmpty());
{{ hal }}::write(data);
}

void
modm::platform::{{ name }}::writeBlocking(const uint8_t *data, std::size_t length)
{{ name }}::writeBlocking(const uint8_t *data, std::size_t length)
{
while (length-- != 0) {
writeBlocking(*data++);
}
}

void
modm::platform::{{ name }}::flushWriteBuffer()
{{ name }}::flushWriteBuffer()
{
return;
}

bool
modm::platform::{{ name }}::write(uint8_t data)
{{ name }}::write(uint8_t data)
{
if({{ hal }}::isTransmitRegisterEmpty()) {
{{ hal }}::write(data);
Expand All @@ -56,7 +59,7 @@ modm::platform::{{ name }}::write(uint8_t data)
}

std::size_t
modm::platform::{{ name }}::write(const uint8_t *data, std::size_t length)
{{ name }}::write(const uint8_t *data, std::size_t length)
{
uint32_t i = 0;
for (; i < length; ++i)
Expand All @@ -69,19 +72,19 @@ modm::platform::{{ name }}::write(const uint8_t *data, std::size_t length)
}

bool
modm::platform::{{ name }}::isWriteFinished()
{{ name }}::isWriteFinished()
{
return {{ hal }}::isTransmitRegisterEmpty();
}

std::size_t
modm::platform::{{ name }}::discardTransmitBuffer()
{{ name }}::discardTransmitBuffer()
{
return 0;
}

bool
modm::platform::{{ name }}::read(uint8_t &data)
{{ name }}::read(uint8_t &data)
{
if({{ hal }}::isReceiveRegisterNotEmpty()) {
{{ hal }}::read(data);
Expand All @@ -92,7 +95,7 @@ modm::platform::{{ name }}::read(uint8_t &data)
}

std::size_t
modm::platform::{{ name }}::read(uint8_t *data, std::size_t length)
{{ name }}::read(uint8_t *data, std::size_t length)
{
(void)length; // avoid compiler warning
if(read(*data)) {
Expand All @@ -103,7 +106,9 @@ modm::platform::{{ name }}::read(uint8_t *data, std::size_t length)
}

std::size_t
modm::platform::{{ name }}::discardReceiveBuffer()
{{ name }}::discardReceiveBuffer()
{
return 0;
}

} // namespace modm::platform
18 changes: 6 additions & 12 deletions src/modm/platform/uart/sam/uart.hpp.in
Original file line number Diff line number Diff line change
Expand Up @@ -15,9 +15,8 @@
*/
// ----------------------------------------------------------------------------

%% set hal = uart_name ~ "Hal" ~ id
%% set name = uart_name ~ id
%% set sercom = sercom_name ~ id
%% set hal = "UartHal" ~ id
%% set name = "Uart" ~ id

#pragma once

Expand All @@ -26,14 +25,11 @@
#include "uart_hal_{{ id }}.hpp"
#include <modm/platform/gpio/pin.hpp>

namespace modm
{

namespace platform
namespace modm::platform
{

/**
* Universal asynchronous receiver transmitter ({{ uart_name | upper ~ id }})
* Universal asynchronous receiver transmitter ({{ "Uart" | upper ~ id }})
*
* @author Kevin Laeufer
* @author Niklas Hauser
Expand Down Expand Up @@ -72,7 +68,7 @@ public:
static void modm_always_inline
initialize(Parity parity = Parity::Disabled)
{
{{ hal }}::initialize<SystemClock, baudrate>(parity);
{{ hal }}::initialize<SystemClock, baudrate, tolerance>(parity);
{{ hal }}::setTransmitterEnable(true);
{{ hal }}::setReceiverEnable(true);
}
Expand Down Expand Up @@ -108,6 +104,4 @@ public:
discardReceiveBuffer();
};

} // namespace platform

} // namespace modm
} // namespace modm::platform
15 changes: 5 additions & 10 deletions src/modm/platform/uart/sam/uart_base.hpp.in
Original file line number Diff line number Diff line change
Expand Up @@ -21,29 +21,24 @@
#include <modm/math/units.hpp>


namespace modm
{
namespace platform
namespace modm::platform
{
/**
* Base class for the UART classes
*
* Provides some common enum that do not depend on the specific UART.
*
* @author Kevin Laeufer
* @ingroup modm_platform_uart
*/
class UartBase
{
public:
enum class Parity : uint32_t
{
Even = 0,
Odd = 1,
Disabled = 2,
Even = 0,
Odd = 1,
Disabled = 2,
};
};

} // namespace platform

} // namespace modm
} // namespace modm::platform
17 changes: 6 additions & 11 deletions src/modm/platform/uart/sam/uart_hal.hpp.in
Original file line number Diff line number Diff line change
Expand Up @@ -19,14 +19,11 @@
#include "uart_base.hpp"
#include "modm/platform/clock/gclk.hpp"

namespace modm
{

namespace platform
namespace modm::platform
{

/**
* Universal asynchronous receiver transmitter ({{ uart_name ~ "Hal" ~ id }})
* Universal asynchronous receiver transmitter (UartHal{{ id }})
*
* Not available on the low- and medium density devices.
*
Expand All @@ -36,7 +33,7 @@ namespace platform
* @author Kevin Laeufer
* @ingroup modm_platform_uart
*/
class {{ uart_name ~ "Hal" ~ id }} : public UartBase
class UartHal{{ id }} : public UartBase
{
private:
/**
Expand Down Expand Up @@ -68,9 +65,9 @@ public:
* Enables clocks, the UART peripheral (but neither TX nor RX)
* Sets baudrate and parity.
*/
template< class SystemClock, baudrate_t baudrate >
template< class SystemClock, baudrate_t baudrate, percent_t tolerance=pct(1) >
static void
initialize( Parity parity = Parity::Disabled);
initialize(Parity parity = Parity::Disabled);

/**
* \brief Write a single byte to the transmit register
Expand Down Expand Up @@ -117,8 +114,6 @@ public:
isTransmitRegisterEmpty();
};

} // namespace platform

} // namespace modm
} // namespace modm::platform

#include "uart_hal_{{ id }}_impl.hpp"
37 changes: 21 additions & 16 deletions src/modm/platform/uart/sam/uart_hal_impl.hpp.in
Original file line number Diff line number Diff line change
Expand Up @@ -15,13 +15,16 @@
*/
// ----------------------------------------------------------------------------

%% set name = uart_name ~ "Hal" ~ id
%% set name = "UartHal" ~ id
%% set sercom = sercom_name ~ id
%% set peripheral = sercom | upper

namespace modm::platform
{

// ----------------------------------------------------------------------------
void
modm::platform::{{ name }}::setParity(const Parity parity)
{{ name }}::setParity(const Parity parity)
{
if (parity != Parity::Disabled) {
{{ peripheral }}->USART.CTRLA.bit.FORM = 0x1; // Enable parity.
Expand All @@ -32,29 +35,29 @@ modm::platform::{{ name }}::setParity(const Parity parity)
}

void
modm::platform::{{ name }}::enable()
{{ name }}::enable()
{
{{ peripheral }}->USART.CTRLA.bit.ENABLE = true; // Uart Enable
while({{ peripheral }}->USART.SYNCBUSY.bit.ENABLE); // Wait for sync.
}

void
modm::platform::{{ name }}::disable()
{{ name }}::disable()
{
{{ peripheral }}->USART.CTRLA.bit.ENABLE = false; // Uart Disable
while({{ peripheral }}->USART.SYNCBUSY.bit.ENABLE); // Wait for sync.
}

void
modm::platform::{{ name }}::reset()
{{ name }}::reset()
{
{{ peripheral }}->USART.CTRLA.bit.SWRST = true; // Uart Reset
while({{ peripheral }}->USART.SYNCBUSY.bit.SWRST); // Wait for sync.
}

template<class SystemClock, modm::baudrate_t baudrate>
void modm_always_inline
modm::platform::{{ name }}::initialize(Parity parity)
template<class SystemClock, modm::baudrate_t baudrate, modm::percent_t tolerance>
void
{{ name }}::initialize(Parity parity)
{
// Enable peripheral clock in power manager.
PM->APBCMASK.bit.{{ peripheral }}_ = true;
Expand Down Expand Up @@ -93,49 +96,51 @@ modm::platform::{{ name }}::initialize(Parity parity)


void
modm::platform::{{ name }}::write(uint8_t data)
{{ name }}::write(uint8_t data)
{
{{ peripheral }}->USART.DATA.reg = data;
}

void
modm::platform::{{ name }}::read(uint8_t &data)
{{ name }}::read(uint8_t &data)
{
data = {{ peripheral }}->USART.DATA.reg;
}

void
modm::platform::{{ name }}::setTransmitterEnable(const bool enable)
{{ name }}::setTransmitterEnable(const bool enable)
{
{{ peripheral }}->USART.CTRLB.bit.TXEN = enable;
}

void
modm::platform::{{ name }}::setReceiverEnable(bool enable)
{{ name }}::setReceiverEnable(bool enable)
{
{{ peripheral }}->USART.CTRLB.bit.RXEN = enable;
}

void
modm::platform::{{ name }}::enableOperation()
{{ name }}::enableOperation()
{
enable();
}

void
modm::platform::{{ name }}::disableOperation()
{{ name }}::disableOperation()
{
disable();
}

bool
modm::platform::{{ name }}::isReceiveRegisterNotEmpty()
{{ name }}::isReceiveRegisterNotEmpty()
{
return {{ peripheral }}->USART.INTFLAG.bit.RXC;
}

bool
modm::platform::{{ name }}::isTransmitRegisterEmpty()
{{ name }}::isTransmitRegisterEmpty()
{
return {{ peripheral }}->USART.INTFLAG.bit.DRE;
}

} // namespace modm::platform

0 comments on commit 0e423af

Please sign in to comment.