diff --git a/boards/arduino-mkr-common/include/periph_conf.h b/boards/arduino-mkr-common/include/periph_conf.h index 6fc69bec0d3068096eb6b0c1c3086ef8c7f95396..a538729ad6f87bc0321da3df78b6ba4781444f5e 100644 --- a/boards/arduino-mkr-common/include/periph_conf.h +++ b/boards/arduino-mkr-common/include/periph_conf.h @@ -104,12 +104,14 @@ extern "C" { */ static const uart_conf_t uart_config[] = { { - .dev = &SERCOM5->USART, - .rx_pin = GPIO_PIN(PB,23), /* ARDUINO_PIN_13, RX Pin */ - .tx_pin = GPIO_PIN(PB,22), /* ARDUINO_PIN_14, TX Pin */ - .mux = GPIO_MUX_D, - .rx_pad = UART_PAD_RX_3, - .tx_pad = UART_PAD_TX_2 + .dev = &SERCOM5->USART, + .rx_pin = GPIO_PIN(PB,23), /* ARDUINO_PIN_13, RX Pin */ + .tx_pin = GPIO_PIN(PB,22), /* ARDUINO_PIN_14, TX Pin */ + .mux = GPIO_MUX_D, + .rx_pad = UART_PAD_RX_3, + .tx_pad = UART_PAD_TX_2, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 } }; diff --git a/boards/arduino-zero/include/periph_conf.h b/boards/arduino-zero/include/periph_conf.h index 773ada0dd267478dc38de989dfde141829a4cdc0..a34cd81e83652a6521e177569485e9707d66df4c 100644 --- a/boards/arduino-zero/include/periph_conf.h +++ b/boards/arduino-zero/include/periph_conf.h @@ -107,20 +107,24 @@ extern "C" { */ static const uart_conf_t uart_config[] = { { - .dev = &SERCOM5->USART, - .rx_pin = GPIO_PIN(PB,23), - .tx_pin = GPIO_PIN(PB,22), - .mux = GPIO_MUX_D, - .rx_pad = UART_PAD_RX_3, - .tx_pad = UART_PAD_TX_2 + .dev = &SERCOM5->USART, + .rx_pin = GPIO_PIN(PB,23), + .tx_pin = GPIO_PIN(PB,22), + .mux = GPIO_MUX_D, + .rx_pad = UART_PAD_RX_3, + .tx_pad = UART_PAD_TX_2, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 }, { - .dev = &SERCOM0->USART, - .rx_pin = GPIO_PIN(PA,11), - .tx_pin = GPIO_PIN(PA,10), - .mux = GPIO_MUX_C, - .rx_pad = UART_PAD_RX_3, - .tx_pad = UART_PAD_TX_2 + .dev = &SERCOM0->USART, + .rx_pin = GPIO_PIN(PA,11), + .tx_pin = GPIO_PIN(PA,10), + .mux = GPIO_MUX_C, + .rx_pad = UART_PAD_RX_3, + .tx_pad = UART_PAD_TX_2, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 } }; diff --git a/boards/samd21-xpro/include/periph_conf.h b/boards/samd21-xpro/include/periph_conf.h index 33a080b404eceefb61b5cce40353340d02539c41..02e138ae643d2b3da89cc538fdc7a88a3893c4bc 100644 --- a/boards/samd21-xpro/include/periph_conf.h +++ b/boards/samd21-xpro/include/periph_conf.h @@ -112,12 +112,14 @@ extern "C" { */ static const uart_conf_t uart_config[] = { { /* Virtual COM Port */ - .dev = &SERCOM3->USART, - .rx_pin = GPIO_PIN(PA,23), - .tx_pin = GPIO_PIN(PA,22), - .mux = GPIO_MUX_C, - .rx_pad = UART_PAD_RX_1, - .tx_pad = UART_PAD_TX_0 + .dev = &SERCOM3->USART, + .rx_pin = GPIO_PIN(PA,23), + .tx_pin = GPIO_PIN(PA,22), + .mux = GPIO_MUX_C, + .rx_pad = UART_PAD_RX_1, + .tx_pad = UART_PAD_TX_0, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 }, { /* EXT1 */ .dev = &SERCOM4->USART, @@ -125,7 +127,9 @@ static const uart_conf_t uart_config[] = { .tx_pin = GPIO_PIN(PB,8), .mux = GPIO_MUX_D, .rx_pad = UART_PAD_RX_1, - .tx_pad = UART_PAD_TX_0 + .tx_pad = UART_PAD_TX_0, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 }, { /* EXT2/3 */ .dev = &SERCOM4->USART, @@ -133,7 +137,9 @@ static const uart_conf_t uart_config[] = { .tx_pin = GPIO_PIN(PB,10), .mux = GPIO_MUX_D, .rx_pad = UART_PAD_RX_3, - .tx_pad = UART_PAD_TX_2 + .tx_pad = UART_PAD_TX_2, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 } }; diff --git a/boards/saml21-xpro/include/periph_conf.h b/boards/saml21-xpro/include/periph_conf.h index 2a06a983989f5dfe187ecd88830048bc3b0f24f6..d3b77a23b0969c6de77ca75c8ed297ded122f45d 100644 --- a/boards/saml21-xpro/include/periph_conf.h +++ b/boards/saml21-xpro/include/periph_conf.h @@ -52,22 +52,34 @@ extern "C" { * @name UART configuration * @{ */ -#define UART_NUMOF (1U) -#define UART_0_EN 1 -#define UART_IRQ_PRIO 1 +static const uart_conf_t uart_config[] = { + { /* Virtual COM Port */ + .dev = &SERCOM3->USART, + .rx_pin = GPIO_PIN(PA,23), + .tx_pin = GPIO_PIN(PA,22), + .mux = GPIO_MUX_C, + .rx_pad = UART_PAD_RX_1, + .tx_pad = UART_PAD_TX_0, + .runstdby = 0, + .gclk_src = GCLK_PCHCTRL_GEN_GCLK0 + }, + { /* EXT1 header */ + .dev = &SERCOM4->USART, + .rx_pin = GPIO_PIN(PB, 9), + .tx_pin = GPIO_PIN(PB, 8), + .mux = GPIO_MUX_D, + .rx_pad = UART_PAD_RX_1, + .tx_pad = UART_PAD_TX_0, + .runstdby = 0, + .gclk_src = GCLK_PCHCTRL_GEN_GCLK0 + } +}; -/* UART 0 device configuration */ -#define UART_0_DEV SERCOM3->USART -#define UART_0_IRQ SERCOM3_IRQn +/* interrupt function name mapping */ #define UART_0_ISR isr_sercom3 -#define UART_0_REF_F (16000000UL) -#define UART_0_RUNSTDBY 1 - -/* UART 0 pin configuration */ -#define UART_0_PORT (PORT->Group[0]) -#define UART_0_TX_PIN (22) -#define UART_0_RX_PIN (23) -#define UART_0_PINS (((PORT_PA22 | PORT_PA23) >> 16) | PORT_WRCONFIG_HWSEL) +#define UART_1_ISR isr_sercom4 + +#define UART_NUMOF (sizeof(uart_config) / sizeof(uart_config[0])) /** @} */ /** diff --git a/boards/samr21-xpro/include/periph_conf.h b/boards/samr21-xpro/include/periph_conf.h index aae2a47786c0c75468ebce393a89ef1fe30e1083..18881a5b8508b4fa74b6a8bdac91616370b5dfd3 100644 --- a/boards/samr21-xpro/include/periph_conf.h +++ b/boards/samr21-xpro/include/periph_conf.h @@ -117,7 +117,9 @@ static const uart_conf_t uart_config[] = { .tx_pin = GPIO_PIN(PA,4), .mux = GPIO_MUX_D, .rx_pad = UART_PAD_RX_1, - .tx_pad = UART_PAD_TX_0 + .tx_pad = UART_PAD_TX_0, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 }, { .dev = &SERCOM5->USART, @@ -125,7 +127,9 @@ static const uart_conf_t uart_config[] = { .tx_pin = GPIO_PIN(PA,22), .mux = GPIO_MUX_D, .rx_pad = UART_PAD_RX_1, - .tx_pad = UART_PAD_TX_0 + .tx_pad = UART_PAD_TX_0, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 } }; diff --git a/boards/sodaq-autonomo/include/periph_conf.h b/boards/sodaq-autonomo/include/periph_conf.h index c4a272ef5271e308d01211af9d4c47200235e075..e8e77dbcaf421e8ed4de6e5dd3044603239b614a 100644 --- a/boards/sodaq-autonomo/include/periph_conf.h +++ b/boards/sodaq-autonomo/include/periph_conf.h @@ -103,36 +103,44 @@ extern "C" { */ static const uart_conf_t uart_config[] = { { - .dev = &SERCOM0->USART, - .rx_pin = GPIO_PIN(PA,9), - .tx_pin = GPIO_PIN(PA,10), - .mux = GPIO_MUX_C, - .rx_pad = UART_PAD_RX_1, - .tx_pad = UART_PAD_TX_2, + .dev = &SERCOM0->USART, + .rx_pin = GPIO_PIN(PA,9), + .tx_pin = GPIO_PIN(PA,10), + .mux = GPIO_MUX_C, + .rx_pad = UART_PAD_RX_1, + .tx_pad = UART_PAD_TX_2, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 }, { - .dev = &SERCOM5->USART, - .rx_pin = GPIO_PIN(PB,31), - .tx_pin = GPIO_PIN(PB,30), - .mux = GPIO_MUX_D, - .rx_pad = UART_PAD_RX_1, - .tx_pad = UART_PAD_TX_0_RTS_2_CTS_3, + .dev = &SERCOM5->USART, + .rx_pin = GPIO_PIN(PB,31), + .tx_pin = GPIO_PIN(PB,30), + .mux = GPIO_MUX_D, + .rx_pad = UART_PAD_RX_1, + .tx_pad = UART_PAD_TX_0_RTS_2_CTS_3, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 }, { - .dev = &SERCOM4->USART, - .rx_pin = GPIO_PIN(PB,13), - .tx_pin = GPIO_PIN(PA,14), - .mux = GPIO_MUX_C, - .rx_pad = UART_PAD_RX_1, - .tx_pad = UART_PAD_TX_2, + .dev = &SERCOM4->USART, + .rx_pin = GPIO_PIN(PB,13), + .tx_pin = GPIO_PIN(PA,14), + .mux = GPIO_MUX_C, + .rx_pad = UART_PAD_RX_1, + .tx_pad = UART_PAD_TX_2, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 }, { - .dev = &SERCOM1->USART, - .rx_pin = GPIO_PIN(PA,17), - .tx_pin = GPIO_PIN(PA,18), - .mux = GPIO_MUX_C, - .rx_pad = UART_PAD_RX_1, - .tx_pad = UART_PAD_TX_2, + .dev = &SERCOM1->USART, + .rx_pin = GPIO_PIN(PA,17), + .tx_pin = GPIO_PIN(PA,18), + .mux = GPIO_MUX_C, + .rx_pad = UART_PAD_RX_1, + .tx_pad = UART_PAD_TX_2, + .runstdby = 0, + .gclk_src = GCLK_CLKCTRL_GEN_GCLK0 } }; diff --git a/cpu/sam0_common/include/periph_cpu_common.h b/cpu/sam0_common/include/periph_cpu_common.h index fbf6c6adad67f449d24bde394e959aff997015e0..81e4f22a3e2f4b5a43fc2d3601c4d1c09e8c887d 100644 --- a/cpu/sam0_common/include/periph_cpu_common.h +++ b/cpu/sam0_common/include/periph_cpu_common.h @@ -117,6 +117,20 @@ typedef enum { * and CTS on pad 3 */ } uart_txpad_t; +/** + * @brief UART device configuration + */ +typedef struct { + SercomUsart *dev; /**< pointer to the used UART device */ + gpio_t rx_pin; /**< pin used for RX */ + gpio_t tx_pin; /**< pin used for TX */ + gpio_mux_t mux; /**< alternative function for pins */ + uart_rxpad_t rx_pad; /**< pad selection for RX line */ + uart_txpad_t tx_pad; /**< pad selection for TX line */ + uint8_t runstdby; /**< allow SERCOM to run in standby mode */ + uint8_t gclk_src; /**< GCLK source which supplys SERCOM */ +} uart_conf_t; + /** * @brief Available values for SERCOM SPI MISO pad selection */ diff --git a/cpu/sam0_common/periph/uart.c b/cpu/sam0_common/periph/uart.c new file mode 100644 index 0000000000000000000000000000000000000000..f564af957fbeac02ecf97021217501120fed589d --- /dev/null +++ b/cpu/sam0_common/periph/uart.c @@ -0,0 +1,306 @@ +/* + * Copyright (C) 2015 Freie Universität Berlin + * 2015 FreshTemp, LLC. + * + * This file is subject to the terms and conditions of the GNU Lesser + * General Public License v2.1. See the file LICENSE in the top level + * directory for more details. + */ + +/** + * @ingroup driver_periph + * @{ + * + * @file + * @brief Low-level UART driver implementation + * + * @author Thomas Eichinger <thomas.eichinger@fu-berlin.de> + * @author Troels Hoffmeyer <troels.d.hoffmeyer@gmail.com> + * @author Hauke Petersen <hauke.petersen@fu-berlin.de> + * @author Dylan Laduranty <dylanladuranty@gmail.com> + * + * @} + */ + +#include "cpu.h" + +#include "periph/uart.h" +#include "periph/gpio.h" + +#define ENABLE_DEBUG (0) +#include "debug.h" + +/** + * @brief Allocate memory to store the callback functions + */ +static uart_isr_ctx_t uart_ctx[UART_NUMOF]; + +/** + * @brief Get the pointer to the base register of the given UART device + * + * @param[in] dev UART device identifier + * + * @return base register address + */ +static inline SercomUsart *_uart(uart_t dev) +{ + return uart_config[dev].dev; +} + +#ifdef CPU_FAM_SAML21 + +static uint64_t _long_division(uint64_t n, uint64_t d); + +static uint8_t sercom_gclk_id[] = + { + SERCOM0_GCLK_ID_CORE, + SERCOM1_GCLK_ID_CORE, + SERCOM2_GCLK_ID_CORE, + SERCOM3_GCLK_ID_CORE, + SERCOM4_GCLK_ID_CORE, + SERCOM5_GCLK_ID_CORE + }; + + +#endif + +static int init_base(uart_t uart, uint32_t baudrate); + +int uart_init(uart_t uart, uint32_t baudrate, uart_rx_cb_t rx_cb, void *arg) +{ + /* initialize basic functionality */ + int res = init_base(uart, baudrate); + if (res != UART_OK) { + return res; + } + + /* register callbacks */ + uart_ctx[uart].rx_cb = rx_cb; + uart_ctx[uart].arg = arg; + /* configure interrupts and enable RX interrupt */ + NVIC_EnableIRQ(SERCOM0_IRQn + sercom_id(_uart(uart))); + _uart(uart)->INTENSET.reg |= SERCOM_USART_INTENSET_RXC; + + return UART_OK; +} + +static int init_base(uart_t uart, uint32_t baudrate) +{ + + if ((unsigned int)uart >= UART_NUMOF) { + return UART_NODEV; + } + + /* configure pins */ + gpio_init(uart_config[uart].rx_pin, GPIO_IN); + gpio_init_mux(uart_config[uart].rx_pin, uart_config[uart].mux); + gpio_init(uart_config[uart].tx_pin, GPIO_OUT); + gpio_init_mux(uart_config[uart].tx_pin, uart_config[uart].mux); + +#ifdef CPU_FAM_SAMD21 + /* calculate baudrate */ + uint32_t baud = ((((uint32_t)CLOCK_CORECLOCK * 10) / baudrate) / 16); + /* enable sync and async clocks */ + uart_poweron(uart); + /* reset the UART device */ + _uart(uart)->CTRLA.reg = SERCOM_USART_CTRLA_SWRST; + while (_uart(uart)->SYNCBUSY.reg & SERCOM_USART_SYNCBUSY_SWRST) {} + /* set asynchronous mode w/o parity, LSB first, TX and RX pad as specified + * by the board in the periph_conf.h, x16 sampling and use internal clock */ + _uart(uart)->CTRLA.reg = (SERCOM_USART_CTRLA_DORD | + SERCOM_USART_CTRLA_SAMPR(0x1) | + SERCOM_USART_CTRLA_TXPO(uart_config[uart].tx_pad) | + SERCOM_USART_CTRLA_RXPO(uart_config[uart].rx_pad) | + SERCOM_USART_CTRLA_MODE_USART_INT_CLK | + (uart_config[uart].runstdby ? + SERCOM_USART_CTRLA_RUNSTDBY : 0)); + + /* set baudrate */ + _uart(uart)->BAUD.FRAC.FP = (baud % 10); + _uart(uart)->BAUD.FRAC.BAUD = (baud / 10); + /* enable receiver and transmitter, use 1 stop bit */ + _uart(uart)->CTRLB.reg = (SERCOM_USART_CTRLB_RXEN | SERCOM_USART_CTRLB_TXEN); + while (_uart(uart)->SYNCBUSY.reg & SERCOM_USART_SYNCBUSY_CTRLB) {} +#elif CPU_FAM_SAML21 + /* Calculate the BAUD value */ + uint64_t temp1 = ((16 * ((uint64_t)baudrate)) << 32); + uint64_t ratio = _long_division(temp1 , CLOCK_CORECLOCK); + uint64_t scale = ((uint64_t)1 << 32) - ratio; + uint64_t baud_calculated = (65536 * scale) >> 32; + + _uart(uart)->CTRLA.bit.ENABLE = 0; /* Disable to write, need to sync tho */ + while(_uart(uart)->SYNCBUSY.bit.ENABLE) {} + + /* set to LSB, asynchronous mode without parity, PAD0 Tx, PAD1 Rx, + * 16x over-sampling, internal clk */ + _uart(uart)->CTRLA.reg = SERCOM_USART_CTRLA_DORD \ + | SERCOM_USART_CTRLA_FORM(0x0) \ + | SERCOM_USART_CTRLA_SAMPA(0x0) \ + | SERCOM_USART_CTRLA_TXPO(uart_config[uart].tx_pad) \ + | SERCOM_USART_CTRLA_RXPO(uart_config[uart].rx_pad) \ + | SERCOM_USART_CTRLA_SAMPR(0x0) \ + | SERCOM_USART_CTRLA_MODE(0x1) \ + | (uart_config[uart].runstdby ? + SERCOM_USART_CTRLA_RUNSTDBY : 0); + + /* Set baud rate */ + _uart(uart)->BAUD.bit.BAUD = baud_calculated; + + /* enable receiver and transmitter, one stop bit*/ + _uart(uart)->CTRLB.reg = (SERCOM_USART_CTRLB_RXEN | SERCOM_USART_CTRLB_TXEN); + while(_uart(uart)->SYNCBUSY.bit.CTRLB) {} + uart_poweron(uart); +#endif + return UART_OK; +} + +void uart_write(uart_t uart, const uint8_t *data, size_t len) +{ + for (size_t i = 0; i < len; i++) { + while (!(_uart(uart)->INTFLAG.reg & SERCOM_USART_INTFLAG_DRE)) {} + _uart(uart)->DATA.reg = data[i]; + while (_uart(uart)->INTFLAG.reg & SERCOM_USART_INTFLAG_TXC) {} + } +} + +void uart_poweron(uart_t uart) +{ +#ifdef CPU_FAM_SAMD21 + PM->APBCMASK.reg |= (PM_APBCMASK_SERCOM0 << sercom_id(_uart(uart))); + GCLK->CLKCTRL.reg = (GCLK_CLKCTRL_CLKEN | + GCLK_CLKCTRL_GEN(uart_config[uart].gclk_src) | + (SERCOM0_GCLK_ID_CORE + sercom_id(_uart(uart))) << + GCLK_CLKCTRL_ID_Pos); + while (GCLK->STATUS.reg & GCLK_STATUS_SYNCBUSY) {} +#elif CPU_FAM_SAML21 + /* Enable the peripheral channel */ + GCLK->PCHCTRL[sercom_gclk_id[sercom_id(_uart(uart))]].reg |= + GCLK_PCHCTRL_CHEN | GCLK_PCHCTRL_GEN(uart_config[uart].gclk_src); + + while (!(GCLK->PCHCTRL[sercom_gclk_id[sercom_id(_uart(uart))]].reg & + GCLK_PCHCTRL_CHEN)) {} + if(sercom_gclk_id[sercom_id(_uart(uart))] < 5) { + MCLK->APBCMASK.reg |= MCLK_APBCMASK_SERCOM0 << sercom_id(_uart(uart)); + } + else { + MCLK->APBDMASK.reg |= MCLK_APBDMASK_SERCOM5; + } + while (_uart(uart)->SYNCBUSY.reg) {} +#endif + /* finally, enable the device */ + _uart(uart)->CTRLA.reg |= SERCOM_USART_CTRLA_ENABLE; +} + +void uart_poweroff(uart_t uart) +{ +#ifdef CPU_FAM_SAMD21 + PM->APBCMASK.reg &= ~(PM_APBCMASK_SERCOM0 << sercom_id(_uart(uart))); + GCLK->CLKCTRL.reg = ((SERCOM0_GCLK_ID_CORE + sercom_id(_uart(uart))) << + GCLK_CLKCTRL_ID_Pos); + while (GCLK->STATUS.reg & GCLK_STATUS_SYNCBUSY) {} +#elif CPU_FAM_SAML21 + /* Enable the peripheral channel */ + GCLK->PCHCTRL[sercom_gclk_id[sercom_id(_uart(uart))]].reg &= ~GCLK_PCHCTRL_CHEN; + + if(sercom_gclk_id[sercom_id(_uart(uart))] < 5) { + MCLK->APBCMASK.reg &= ~(MCLK_APBCMASK_SERCOM0 << sercom_id(_uart(uart))); + } + else { + MCLK->APBDMASK.reg &= ~MCLK_APBDMASK_SERCOM5; + } + while (_uart(uart)->SYNCBUSY.reg) {} + +#endif +} + +static inline void irq_handler(uint8_t uartnum) +{ +#ifdef CPU_FAM_SAMD21 + if (_uart(uartnum)->INTFLAG.reg & SERCOM_USART_INTFLAG_RXC) { + /* interrupt flag is cleared by reading the data register */ + uart_ctx[uartnum].rx_cb(uart_ctx[uartnum].arg, + (uint8_t)(_uart(uartnum)->DATA.reg)); + } + else if (_uart(uartnum)->INTFLAG.reg & SERCOM_USART_INTFLAG_ERROR) { + /* clear error flag */ + _uart(uartnum)->INTFLAG.reg = SERCOM_USART_INTFLAG_ERROR; + } +#elif CPU_FAM_SAML21 + if (_uart(uartnum)->INTFLAG.bit.RXC) { + /* cleared by reading DATA regiser */ + uint8_t data = (uint8_t)_uart(uartnum)->DATA.reg; + uart_ctx[uartnum].rx_cb(uart_ctx[uartnum].arg, data); + } + else if (_uart(uartnum)->INTFLAG.bit.ERROR) { + /* clear error flag */ + _uart(uartnum)->INTFLAG.reg |= SERCOM_USART_INTFLAG_ERROR; + } +#endif + cortexm_isr_end(); +} + +#ifdef UART_0_ISR +void UART_0_ISR(void) +{ + irq_handler(0); +} +#endif + +#ifdef UART_1_ISR +void UART_1_ISR(void) +{ + irq_handler(1); +} +#endif + +#ifdef UART_2_ISR +void UART_2_ISR(void) +{ + irq_handler(2); +} +#endif + +#ifdef UART_3_ISR +void UART_3_ISR(void) +{ + irq_handler(3); +} +#endif + +#ifdef UART_4_ISR +void UART_4_ISR(void) +{ + irq_handler(4); +} +#endif + +#ifdef UART_5_ISR +void UART_5_ISR(void) +{ + irq_handler(5); +} +#endif + +#ifdef CPU_FAM_SAML21 +static uint64_t _long_division(uint64_t n, uint64_t d) +{ + int32_t i; + uint64_t q = 0, r = 0, bit_shift; + for (i = 63; i >= 0; i--) { + bit_shift = (uint64_t)1 << i; + + r = r << 1; + + if (n & bit_shift) { + r |= 0x01; + } + + if (r >= d) { + r = r - d; + q |= bit_shift; + } + } + + return q; +} +#endif diff --git a/cpu/samd21/include/periph_cpu.h b/cpu/samd21/include/periph_cpu.h index 459e41f254abec4358082f13f630d4370790c931..94ab5cdd7935e98776c588738f68c566d8e91e0d 100644 --- a/cpu/samd21/include/periph_cpu.h +++ b/cpu/samd21/include/periph_cpu.h @@ -88,18 +88,6 @@ typedef struct { pwm_conf_chan_t chan[3]; /**< channel configuration */ } pwm_conf_t; -/** - * @brief UART device configuration - */ -typedef struct { - SercomUsart *dev; /**< pointer to the used UART device */ - gpio_t rx_pin; /**< pin used for RX */ - gpio_t tx_pin; /**< pin used for TX */ - gpio_mux_t mux; /**< alternative function for pins */ - uart_rxpad_t rx_pad; /**< pad selection for RX line */ - uart_txpad_t tx_pad; /**< pad selection for TX line */ -} uart_conf_t; - /** * @brief Return the numeric id of a SERCOM device derived from its address * diff --git a/cpu/samd21/periph/uart.c b/cpu/samd21/periph/uart.c deleted file mode 100644 index 24b8ef53da40a8e353fcf303cf05fce3bffe8e56..0000000000000000000000000000000000000000 --- a/cpu/samd21/periph/uart.c +++ /dev/null @@ -1,189 +0,0 @@ -/* - * Copyright (C) 2015 Freie Universität Berlin - * - * This file is subject to the terms and conditions of the GNU Lesser - * General Public License v2.1. See the file LICENSE in the top level - * directory for more details. - */ - -/** - * @ingroup cpu_samd21 - * @ingroup drivers_periph_uart - * @{ - * - * @file - * @brief Low-level UART driver implementation - * - * @author Thomas Eichinger <thomas.eichinger@fu-berlin.de> - * @author Troels Hoffmeyer <troels.d.hoffmeyer@gmail.com> - * @author Hauke Petersen <hauke.petersen@fu-berlin.de> - * - * @} - */ - -#include "cpu.h" - -#include "periph/uart.h" -#include "periph/gpio.h" - -#ifdef UART_NUMOF -/** - * @brief Allocate memory to store the callback functions - */ -static uart_isr_ctx_t uart_ctx[UART_NUMOF]; - -/** - * @brief Get the pointer to the base register of the given UART device - * - * @param[in] dev UART device identifier - * - * @return base register address - */ -static inline SercomUsart *_uart(uart_t dev) -{ - return uart_config[dev].dev; -} - -static int init_base(uart_t uart, uint32_t baudrate); - -int uart_init(uart_t uart, uint32_t baudrate, uart_rx_cb_t rx_cb, void *arg) -{ - /* initialize basic functionality */ - int res = init_base(uart, baudrate); - if (res != UART_OK) { - return res; - } - - /* register callbacks */ - uart_ctx[uart].rx_cb = rx_cb; - uart_ctx[uart].arg = arg; - /* configure interrupts and enable RX interrupt */ - _uart(uart)->INTENSET.reg = SERCOM_USART_INTENSET_RXC; - NVIC_EnableIRQ(SERCOM0_IRQn + sercom_id(_uart(uart))); - return UART_OK; -} - -static int init_base(uart_t uart, uint32_t baudrate) -{ - uint32_t baud; - SercomUsart *dev; - - if ((unsigned int)uart >= UART_NUMOF) { - return UART_NODEV; - } - - /* get the devices base register */ - dev = _uart(uart); - /* calculate baudrate */ - baud = ((((uint32_t)CLOCK_CORECLOCK * 10) / baudrate) / 16); - /* enable sync and async clocks */ - uart_poweron(uart); - /* configure pins */ - gpio_init(uart_config[uart].rx_pin, GPIO_IN); - gpio_init_mux(uart_config[uart].rx_pin, uart_config[uart].mux); - gpio_init(uart_config[uart].tx_pin, GPIO_OUT); - gpio_init_mux(uart_config[uart].tx_pin, uart_config[uart].mux); - /* reset the UART device */ - dev->CTRLA.reg = SERCOM_USART_CTRLA_SWRST; - while (dev->SYNCBUSY.reg & SERCOM_USART_SYNCBUSY_SWRST) {} - /* set asynchronous mode w/o parity, LSB first, TX and RX pad as specified - * by the board in the periph_conf.h, x16 sampling and use internal clock */ - dev->CTRLA.reg = (SERCOM_USART_CTRLA_DORD | - SERCOM_USART_CTRLA_SAMPR(0x1) | - SERCOM_USART_CTRLA_TXPO(uart_config[uart].tx_pad) | - SERCOM_USART_CTRLA_RXPO(uart_config[uart].rx_pad) | - SERCOM_USART_CTRLA_MODE_USART_INT_CLK); - /* set baudrate */ - dev->BAUD.FRAC.FP = (baud % 10); - dev->BAUD.FRAC.BAUD = (baud / 10); - /* enable receiver and transmitter, use 1 stop bit */ - dev->CTRLB.reg = (SERCOM_USART_CTRLB_RXEN | SERCOM_USART_CTRLB_TXEN); - while (dev->SYNCBUSY.reg & SERCOM_USART_SYNCBUSY_CTRLB) {} - /* finally, enable the device */ - dev->CTRLA.reg |= SERCOM_USART_CTRLA_ENABLE; - return UART_OK; -} - -void uart_write(uart_t uart, const uint8_t *data, size_t len) -{ - for (size_t i = 0; i < len; i++) { - while (!(_uart(uart)->INTFLAG.reg & SERCOM_USART_INTFLAG_DRE)) {} - _uart(uart)->DATA.reg = data[i]; - } -} - -void uart_poweron(uart_t uart) -{ - PM->APBCMASK.reg |= (PM_APBCMASK_SERCOM0 << sercom_id(_uart(uart))); - GCLK->CLKCTRL.reg = (GCLK_CLKCTRL_CLKEN | - GCLK_CLKCTRL_GEN_GCLK0 | - (SERCOM0_GCLK_ID_CORE + sercom_id(_uart(uart))) << - GCLK_CLKCTRL_ID_Pos); - while (GCLK->STATUS.reg & GCLK_STATUS_SYNCBUSY) {} -} - -void uart_poweroff(uart_t uart) -{ - PM->APBCMASK.reg &= ~(PM_APBCMASK_SERCOM0 << sercom_id(_uart(uart))); - GCLK->CLKCTRL.reg = ((SERCOM0_GCLK_ID_CORE + sercom_id(_uart(uart))) << - GCLK_CLKCTRL_ID_Pos); - while (GCLK->STATUS.reg & GCLK_STATUS_SYNCBUSY) {} -} - -static inline void irq_handler(int dev) -{ - SercomUsart *uart = _uart(dev); - - if (uart->INTFLAG.reg & SERCOM_USART_INTFLAG_RXC) { - /* interrupt flag is cleared by reading the data register */ - uart_ctx[dev].rx_cb(uart_ctx[dev].arg, (uint8_t)(uart->DATA.reg)); - } - else if (uart->INTFLAG.reg & SERCOM_USART_INTFLAG_ERROR) { - /* clear error flag */ - uart->INTFLAG.reg = SERCOM_USART_INTFLAG_ERROR; - } - cortexm_isr_end(); -} - -#ifdef UART_0_ISR -void UART_0_ISR(void) -{ - irq_handler(0); -} -#endif - -#ifdef UART_1_ISR -void UART_1_ISR(void) -{ - irq_handler(1); -} -#endif - -#ifdef UART_2_ISR -void UART_2_ISR(void) -{ - irq_handler(2); -} -#endif - -#ifdef UART_3_ISR -void UART_3_ISR(void) -{ - irq_handler(3); -} -#endif - -#ifdef UART_4_ISR -void UART_4_ISR(void) -{ - irq_handler(4); -} -#endif - -#ifdef UART_5_ISR -void UART_5_ISR(void) -{ - irq_handler(5); -} -#endif -#endif diff --git a/cpu/saml21/periph/uart.c b/cpu/saml21/periph/uart.c deleted file mode 100644 index 2fd462411d4cf6de2f20c7401ca6199caf141746..0000000000000000000000000000000000000000 --- a/cpu/saml21/periph/uart.c +++ /dev/null @@ -1,187 +0,0 @@ -/* - * Copyright (C) 2014 Freie Universität Berlin - * 2015 Kaspar Schleiser <kaspar@schleiser.de> - * 2015 FreshTemp, LLC. - * - * This file is subject to the terms and conditions of the GNU Lesser - * General Public License v2.1. See the file LICENSE in the top level - * directory for more details. - */ - -/** - * @ingroup cpu_saml21 - * @ingroup drivers_periph_uart - * @{ - * - * @file uart.c - * @brief Low-level UART driver implementation - * - * @author Thomas Eichinger <thomas.eichinger@fu-berlin.de> - * @author Troels Hoffmeyer <troels.d.hoffmeyer@gmail.com> - * - * @} - */ - -#include "cpu.h" -#include "periph/uart.h" - -/** - * @brief Allocate memory to store the callback functions. - */ -static uart_isr_ctx_t uart_config[UART_NUMOF]; - -static uint64_t _long_division(uint64_t n, uint64_t d); - -static int init_base(uart_t uart, uint32_t baudrate); - -int uart_init(uart_t uart, uint32_t baudrate, uart_rx_cb_t rx_cb, void *arg) -{ - /* initialize basic functionality */ - int res = init_base(uart, baudrate); - if (res != UART_OK) { - return UART_NODEV; - } - - /* register callbacks */ - uart_config[uart].rx_cb = rx_cb; - uart_config[uart].arg = arg; - - /* configure interrupts and enable RX interrupt */ - switch (uart) { - case UART_0: - NVIC_SetPriority(UART_0_IRQ, UART_IRQ_PRIO); - NVIC_EnableIRQ(UART_0_IRQ); - UART_0_DEV.INTENSET.bit.RXC = 1; - break; - } - return UART_OK; -} - -static int init_base(uart_t uart, uint32_t baudrate) -{ - /* Calculate the BAUD value */ - uint64_t temp1 = ((16 * ((uint64_t)baudrate)) << 32); - uint64_t ratio = _long_division(temp1 , UART_0_REF_F); - uint64_t scale = ((uint64_t)1 << 32) - ratio; - uint64_t baud_calculated = (65536 * scale) >> 32; - - switch (uart) { -#if UART_0_EN - case UART_0: - /* Enable the peripheral channel */ - GCLK->PCHCTRL[SERCOM3_GCLK_ID_CORE].reg |= GCLK_PCHCTRL_CHEN | GCLK_PCHCTRL_GEN_GCLK0; - - while (!(GCLK->PCHCTRL[SERCOM3_GCLK_ID_CORE].reg & GCLK_PCHCTRL_CHEN)) { - /* Wait for clock synchronization */ - } - - MCLK->APBCMASK.reg |= MCLK_APBCMASK_SERCOM3; - /* configure PINS to input/output*/ - UART_0_PORT.DIRSET.reg = (1 << UART_0_TX_PIN); /* tx's direction is output */ - UART_0_PORT.PINCFG[UART_0_RX_PIN % 32].bit.INEN = true; /* buffer rx pin's value */ - - /* enable PMUX for pins and set to config C. */ - UART_0_PORT.WRCONFIG.reg = PORT_WRCONFIG_WRPINCFG \ - | PORT_WRCONFIG_WRPMUX \ - | PORT_WRCONFIG_PMUX(0x2) \ - | PORT_WRCONFIG_PMUXEN \ - | UART_0_PINS; - - UART_0_DEV.CTRLA.bit.ENABLE = 0; /* Disable to write, need to sync tho */ - while(UART_0_DEV.SYNCBUSY.bit.ENABLE) {} - - /* set to LSB, asynchronous mode without parity, PAD0 Tx, PAD1 Rx, - * 16x over-sampling, internal clk */ - UART_0_DEV.CTRLA.reg = SERCOM_USART_CTRLA_DORD \ - | SERCOM_USART_CTRLA_FORM(0x0) \ - | SERCOM_USART_CTRLA_SAMPA(0x0) \ - | SERCOM_USART_CTRLA_TXPO(0x0) \ - | SERCOM_USART_CTRLA_RXPO(0x1) \ - | SERCOM_USART_CTRLA_SAMPR(0x0) \ - | SERCOM_USART_CTRLA_MODE(0x1) \ - | (UART_0_RUNSTDBY ? SERCOM_USART_CTRLA_RUNSTDBY : 0); - - /* Set baud rate */ - UART_0_DEV.BAUD.bit.BAUD = baud_calculated; - - /* enable receiver and transmitter, one stop bit*/ - UART_0_DEV.CTRLB.reg = (SERCOM_USART_CTRLB_RXEN | SERCOM_USART_CTRLB_TXEN); - while(UART_0_DEV.SYNCBUSY.bit.CTRLB) {} - - break; -#endif - default: - (void)baud_calculated; - return UART_NODEV; - } - - uart_poweron(uart); - return UART_OK; -} - -void uart_write(uart_t uart, const uint8_t *data, size_t len) -{ - if (uart == UART_0) { - for (size_t i = 0; i < len; i++) { - while (UART_0_DEV.INTFLAG.bit.DRE == 0) {} - while(UART_0_DEV.SYNCBUSY.bit.ENABLE) {} - UART_0_DEV.DATA.reg = data[i]; - while (UART_0_DEV.INTFLAG.reg & SERCOM_USART_INTFLAG_TXC) {} - } - } -} - -static inline void irq_handler(uint8_t uartnum, SercomUsart *dev) -{ - if (dev->INTFLAG.bit.RXC) { - /* cleared by reading DATA regiser */ - uint8_t data = (uint8_t)dev->DATA.reg; - uart_config[uartnum].rx_cb(uart_config[uartnum].arg, data); - } - else if (dev->INTFLAG.bit.ERROR) { - /* clear error flag */ - dev->INTFLAG.reg |= SERCOM_USART_INTFLAG_ERROR; - } - cortexm_isr_end(); -} - -void uart_poweron(uart_t uart) -{ - while (UART_0_DEV.SYNCBUSY.reg) {} - UART_0_DEV.CTRLA.reg |= SERCOM_USART_CTRLA_ENABLE; -} - -void uart_poweroff(uart_t uart) -{ - while (UART_0_DEV.SYNCBUSY.reg) {} - UART_0_DEV.CTRLA.reg &= ~SERCOM_USART_CTRLA_ENABLE; -} - -#if UART_0_EN -void UART_0_ISR(void) -{ - irq_handler(UART_0, &UART_0_DEV); -} -#endif - -static uint64_t _long_division(uint64_t n, uint64_t d) -{ - int32_t i; - uint64_t q = 0, r = 0, bit_shift; - for (i = 63; i >= 0; i--) { - bit_shift = (uint64_t)1 << i; - - r = r << 1; - - if (n & bit_shift) { - r |= 0x01; - } - - if (r >= d) { - r = r - d; - q |= bit_shift; - } - } - - return q; -}