2020-11-14 04:04:59 +01:00
|
|
|
/* SPDX-License-Identifier: BSD-3-Clause */
|
|
|
|
/* See the end of this file for copyright, licensing, and warranty information. */
|
|
|
|
|
|
|
|
#include <ardix/serial.h>
|
|
|
|
#include <ardix/types.h>
|
|
|
|
#include <ardix/ringbuf.h>
|
|
|
|
|
2020-11-19 01:44:51 +01:00
|
|
|
#include <arch/at91sam3x8e/hardware.h>
|
|
|
|
#include <arch/at91sam3x8e/interrupt.h>
|
2020-11-14 04:04:59 +01:00
|
|
|
#include <arch/serial.h>
|
|
|
|
|
2020-11-26 19:20:06 +01:00
|
|
|
#include <stddef.h>
|
|
|
|
|
|
|
|
struct serial_interface arch_serial_default_interface = {
|
|
|
|
.tx = NULL,
|
|
|
|
.rx = NULL,
|
|
|
|
.id = 0,
|
|
|
|
.baud = 0,
|
|
|
|
};
|
|
|
|
struct serial_interface *serial_default_interface = &arch_serial_default_interface;
|
|
|
|
|
2020-11-14 04:04:59 +01:00
|
|
|
int arch_serial_init(struct serial_interface *interface)
|
|
|
|
{
|
2020-11-19 01:44:51 +01:00
|
|
|
if (interface->baud <= 0 || interface->id != 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* enable peripheral clock for UART (which has peripheral id 8) */
|
2020-11-26 19:20:06 +01:00
|
|
|
REG_PMC_PCER0 |= REG_PMC_PCER0_PID(8);
|
2020-11-19 01:44:51 +01:00
|
|
|
|
2020-11-26 19:20:06 +01:00
|
|
|
/* ensure the PIO controller is turned off on the serial pins */
|
|
|
|
REG_PIO_PDR(PIOA) = (1 << 8) | (1 << 9);
|
|
|
|
|
|
|
|
/* turn off PDC channel (we are manually writing byte-by-byte here) */
|
|
|
|
REG_UART_PDC_PTCR = REG_UART_PDC_PTCR_RXTDIS_MASK | REG_UART_PDC_PTCR_TXTDIS_MASK;
|
2020-11-19 01:44:51 +01:00
|
|
|
|
|
|
|
/* reset & disable rx and tx */
|
|
|
|
REG_UART_CR = REG_UART_CR_RXDIS_MASK | REG_UART_CR_RSTRX_MASK
|
|
|
|
| REG_UART_CR_TXDIS_MASK | REG_UART_CR_RSTTX_MASK;
|
|
|
|
|
|
|
|
/* no parity, normal mode */
|
|
|
|
REG_UART_MR = REG_UART_MR_PAR_NO | REG_UART_MR_CHMODE_NORMAL;
|
|
|
|
|
|
|
|
/* From Atmel Datasheet: baud rate = MCK / (REG_UART_BRGR * 16) */
|
2020-11-26 19:20:06 +01:00
|
|
|
REG_UART_BRGR = (uint16_t)(( sys_core_clock / (uint32_t)interface->baud ) >> 4);
|
2020-11-19 01:44:51 +01:00
|
|
|
|
|
|
|
/* choose the events we want an interrupt on */
|
|
|
|
REG_UART_IDR = 0xFFFFFFFF; /* make sure all interrupts are disabled first */
|
2020-11-26 19:20:06 +01:00
|
|
|
/* TXRDY is not selected because the output buffer is initially empty anyway */
|
|
|
|
REG_UART_IER = REG_UART_IER_RXRDY_MASK | REG_UART_IER_OVRE_MASK | REG_UART_IER_FRAME_MASK | REG_UART_IER_TXRDY_MASK;
|
2020-11-19 01:44:51 +01:00
|
|
|
|
|
|
|
arch_irq_enable(IRQNO_UART);
|
|
|
|
|
2020-11-26 19:20:06 +01:00
|
|
|
/* enable receiver and transmitter */
|
2020-11-19 01:44:51 +01:00
|
|
|
REG_UART_CR = REG_UART_CR_RXEN_MASK | REG_UART_CR_TXEN_MASK;
|
|
|
|
|
|
|
|
return 0;
|
2020-11-14 04:04:59 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void arch_serial_exit(struct serial_interface *interface)
|
|
|
|
{
|
2020-11-19 01:44:51 +01:00
|
|
|
if (interface->id != 0)
|
|
|
|
return;
|
|
|
|
|
2020-11-26 19:20:06 +01:00
|
|
|
/* disable receiver and transmitter */
|
2020-11-19 01:44:51 +01:00
|
|
|
REG_UART_CR = REG_UART_CR_RXDIS_MASK | REG_UART_CR_TXDIS_MASK;
|
|
|
|
|
|
|
|
arch_irq_disable(IRQNO_UART);
|
|
|
|
|
2020-11-26 19:20:06 +01:00
|
|
|
/* disable peripheral clock for UART (PID is taken from Atmel Datasheet, Section 9.1 */
|
2020-11-19 01:44:51 +01:00
|
|
|
REG_PMC_PCDR0 = REG_PMC_PCDR0_PID(8);
|
2020-11-14 04:04:59 +01:00
|
|
|
|
2020-11-26 19:20:06 +01:00
|
|
|
interface->id = -1;
|
2020-11-14 04:04:59 +01:00
|
|
|
}
|
|
|
|
|
2020-11-26 19:20:06 +01:00
|
|
|
void arch_serial_notify(struct serial_interface *interface)
|
2020-11-14 04:04:59 +01:00
|
|
|
{
|
2020-11-26 19:20:06 +01:00
|
|
|
/* unmask the TXRDY interrupt */
|
|
|
|
REG_UART_IER = REG_UART_IER_TXRDY_MASK;
|
2020-11-14 04:04:59 +01:00
|
|
|
}
|
|
|
|
|
2020-11-26 19:20:06 +01:00
|
|
|
void irq_uart(void)
|
2020-11-14 04:04:59 +01:00
|
|
|
{
|
2020-11-26 19:20:06 +01:00
|
|
|
uint8_t tmp;
|
|
|
|
size_t len;
|
|
|
|
uint32_t state = REG_UART_SR;
|
|
|
|
|
|
|
|
/* RX has received a byte, store it into the ring buffer */
|
|
|
|
if (state & REG_UART_SR_RXRDY_MASK) {
|
|
|
|
tmp = REG_UART_RHR;
|
|
|
|
ringbuf_write(serial_default_interface->rx, &tmp, sizeof(tmp));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TX is ready to transmit the next byte */
|
|
|
|
if (state & REG_UART_SR_TXRDY_MASK) {
|
|
|
|
len = ringbuf_read(&tmp, serial_default_interface->tx, sizeof(tmp));
|
|
|
|
|
|
|
|
if (len) {
|
|
|
|
/* there is data in the queue, so write it to TX holding register */
|
|
|
|
REG_UART_THR = tmp;
|
|
|
|
} else {
|
|
|
|
/* TX queue is empty, mask the TXRDY event */
|
|
|
|
REG_UART_IDR = REG_UART_IDR_TXRDY_MASK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check for error conditions */
|
|
|
|
if ((state & REG_UART_SR_OVRE_MASK) || (state & REG_UART_SR_FRAME_MASK)) {
|
|
|
|
/* TODO: write some proper error handling routines ffs */
|
|
|
|
REG_UART_CR = REG_UART_CR_RSTSTA_MASK;
|
|
|
|
}
|
2020-11-14 04:04:59 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2020 Felix Kopp <sandtler@sandtler.club>
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without modification, are permitted
|
|
|
|
* provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright notice, this list of
|
|
|
|
* conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice, this list of
|
|
|
|
* conditions and the following disclaimer in the documentation and/or other materials
|
|
|
|
* provided with the distribution.
|
|
|
|
* 3. Neither the name of the copyright holder nor the names of its contributors may be
|
|
|
|
* used to endorse or promote products derived from this software without specific prior
|
|
|
|
* written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
|
|
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY
|
|
|
|
* WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|