/****************************************************************************** * arch/avr/src/at91uc3/at91uc3_lowconsole.c * * Copyright (C) 2010 Gregory Nutt. All rights reserved. * Author: Gregory Nutt * * 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 NuttX 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 OWNER 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. * ******************************************************************************/ /****************************************************************************** * Included Files ******************************************************************************/ #include #include "at91uc3_config.h" #include #include #include #include #include "up_arch.h" #include "up_internal.h" #include "at91uc3_internal.h" #include "at91uc3_usart.h" #include "at91uc3_pinmux.h" /****************************************************************************** * Private Definitions ******************************************************************************/ /* Select USART parameters for the selected console */ #if defined(CONFIG_USART0_SERIAL_CONSOLE) # define AVR32_CONSOLE_BASE AVR32_USART0_BASE # define AVR32_CONSOLE_BAUD CONFIG_USART0_BAUD # define AVR32_CONSOLE_BITS CONFIG_USART0_BITS # define AVR32_CONSOLE_PARITY CONFIG_USART0_PARITY # define AVR32_CONSOLE_2STOP CONFIG_USART0_2STOP #elif defined(CONFIG_USART1_SERIAL_CONSOLE) # define AVR32_CONSOLE_BASE AVR32_USART1_BASE # define AVR32_CONSOLE_BAUD CONFIG_USART1_BAUD # define AVR32_CONSOLE_BITS CONFIG_USART1_BITS # define AVR32_CONSOLE_PARITY CONFIG_USART1_PARITY # define AVR32_CONSOLE_2STOP CONFIG_USART1_2STOP #elif defined(CONFIG_USART2_SERIAL_CONSOLE) # define AVR32_CONSOLE_BASE AVR32_USART2_BASE # define AVR32_CONSOLE_BAUD CONFIG_USART2_BAUD # define AVR32_CONSOLE_BITS CONFIG_USART2_BITS # define AVR32_CONSOLE_PARITY CONFIG_USART2_PARITY # define AVR32_CONSOLE_2STOP CONFIG_USART2_2STOP #else # error "No CONFIG_USARTn_SERIAL_CONSOLE Setting" #endif /****************************************************************************** * Private Types ******************************************************************************/ /****************************************************************************** * Private Function Prototypes ******************************************************************************/ /****************************************************************************** * Global Variables ******************************************************************************/ /****************************************************************************** * Private Variables ******************************************************************************/ /****************************************************************************** * Private Functions ******************************************************************************/ /****************************************************************************** * Name: usart_putreg * * Description: * Write a value to a USART register * ******************************************************************************/ #ifdef HAVE_RS232_DEVICE static inline void usart_putreg(uintptr_t usart_base, unsigned int offset, uint32_t value) { putreg32(value, usart_base + offset); } #endif /****************************************************************************** * Name: usart_getreg * * Description: * Get a value from a USART register * ******************************************************************************/ #ifdef HAVE_RS232_DEVICE static inline uint32_t usart_getreg(uintptr_t usart_base, unsigned int offset) { return getreg32(usart_base + offset); } #endif /****************************************************************************** * Name: usart_setbaudrate * * Description: * Configure the UART baud rate. * ******************************************************************************/ #ifdef HAVE_RS232_DEVICE static void usart_setbaudrate(uintptr_t usart_base, uint32_t baudrate) { uint32_t cd; uint32_t mr; /* Select 16x or 8x oversampling mode or go to synchronous mode */ mr = usart_getreg(usart_base, AVR32_USART_MR_OFFSET); if (baudrate < AVR32_PBA_CLOCK / 16) { /* Select 16x oversampling mode and clear the SYNC mode bit */ mr &= ~(USART_MR_OVER|USART_MR_SYNC); /* Calculate the clock divider assuming 16x oversampling */ cd = (AVR32_PBA_CLOCK + (baudrate << 3)) / (baudrate << 4); } else if (baudrate < AVR32_PBA_CLOCK / 8) { /* Select 8x oversampling mode and clear the SYNC mode bit */ mr &= ~USART_MR_SYNC; mr |= USART_MR_OVER; /* Calculate the clock divider assuming 16x oversampling */ cd = (AVR32_PBA_CLOCK + (baudrate << 2)) / (baudrate << 3); } else { /* Set the SYNC mode bit */ mr |= USART_MR_SYNC; /* Use the undivided PBA clock */ cd = AVR32_PBA_CLOCK / baudrate; } DEBUGASSERT(cd > 0 && cd < 65536); usart_putreg(usart_base, AVR32_USART_MR_OFFSET, mr); usart_putreg(usart_base, AVR32_USART_BRGR_OFFSET, cd); } #endif /****************************************************************************** * Public Functions ******************************************************************************/ /****************************************************************************** * Name: usart_reset * * Description: * Reset a USART. * ******************************************************************************/ #ifdef HAVE_RS232_DEVICE void usart_reset(uintptr_t usart_base) { irqstate_t flags; /* Disable all USART interrupts */ flags = irqsave(); usart_putreg(usart_base, AVR32_USART_IDR_OFFSET, 0xffffffff); irqrestore(flags); /* Reset mode and other registers */ usart_putreg(usart_base, AVR32_USART_MR_OFFSET, 0); /* Reset mode register */ usart_putreg(usart_base, AVR32_USART_RTOR_OFFSET, 0); /* Reset receiver timeout register */ usart_putreg(usart_base, AVR32_USART_TTGR_OFFSET, 0); /* Reset transmitter timeguard reg */ /* Disable RX and TX, put USART in reset, disable handshaking signals */ usart_putreg(usart_base, AVR32_USART_CR_OFFSET, USART_CR_RSTRX|USART_CR_RSTTX|USART_CR_RSTSTA|USART_CR_RSTIT| USART_CR_RSTNACK|USART_CR_DTRDIS|USART_CR_RTSDIS); } #endif /****************************************************************************** * Name: usart_configure * * Description: * Configure a USART as a RS-232 UART. * ******************************************************************************/ #ifdef HAVE_RS232_DEVICE void usart_configure(uintptr_t usart_base, uint32_t baud, unsigned int parity, unsigned int nbits, bool stop2) { uint32_t regval; /* Reset the USART and disable RX and TX */ usart_reset(usart_base); /* Set the baud rate generation register */ usart_setbaudrate(usart_base, baud); /* Configure STOP bits */ regval = USART_MR_MODE_NORMAL|USART_MR_CHMODE_NORMAL; /* Normal RS-232 mode */ regval |= stop2 ? USART_MR_NBSTOP_2 : USART_MR_NBSTOP_1; /* Configure parity */ switch (parity) { case 0: default: regval |= USART_MR_PAR_NONE; break; case 1: regval |= USART_MR_PAR_ODD; break; case 2: regval |= USART_MR_PAR_EVEN; break; } /* Configure the number of bits per word */ DEBUGASSERT(nbits >= 5 && nbits <= 9); if (nbits == 9) { regval |= USART_MR_MODE9; } else { regval |= USART_MR_CHRL_BITS(nbits); } usart_putreg(usart_base, AVR32_USART_MR_OFFSET, regval); /* Enable RX and TX */ regval = usart_getreg(usart_base, AVR32_USART_CR_OFFSET); regval |= (USART_CR_RXEN|USART_CR_TXEN); usart_putreg(usart_base, AVR32_USART_CR_OFFSET, regval); } #endif /****************************************************************************** * Name: up_consoleinit * * Description: * Initialize a console for debug output. This function is called very * early in the intialization sequence to configure the serial console uart * (only). * ******************************************************************************/ #ifndef CONFIG_USE_EARLYSERIALINIT void up_consoleinit(void) { /* Enable clocking for the enabled USART modules. Hmmm... It looks like the * default state for CLK_USART0, USART1 CLK_USART1, and USART2 CLK_USART2 is * "enabled" in the PM's PBAMASK register. */ /* Setup GPIO pins for each configured USART/UART */ #ifdef CONFIG_AVR32_USART0_RS232 /* PINMUX_USART0_RXD and PINMUX_USART0_TXD must be defined in board.h. It * must define them be be one of {PINMUX_USART0_RXD_1, PINMUX_USART0_RXD_2} * and {PINMUX_USART_0TXD_1, PINMUX_USART0_TXD_2}, respectively. */ at91uc3_configgpio(PINMUX_USART0_RXD); at91uc3_configgpio(PINMUX_USART0_TXD); #endif #ifdef CONFIG_AVR32_USART1_RS232 /* PINMUX_USART1_RXD and PINMUX_USART1_TXD must be defined in board.h. It * must define them be be one of {PINMUX_USART1_RXD_1, PINMUX_USART1_RXD_2, * PINMUX_USART1_RXD_3} and {PINMUX_USART1_TXD_1, PINMUX_USART1_TXD_2, * PINMUX_USART1_TXD_3}, respectively. */ at91uc3_configgpio(PINMUX_USART1_RXD); at91uc3_configgpio(PINMUX_USART1_TXD); #endif #ifdef CONFIG_AVR32_USART2_RS232 /* PINMUX_USART2_RXD and PINMUX_USART2_TXD must be defined in board.h. It * must define them be be one of {PINMUX_USART2_RXD_1, PINMUX_USART2_RXD_2} * and {PINMUX_USART2_TXD_1, PINMUX_USART2_TXD_2}, respectively. */ at91uc3_configgpio(PINMUX_USART2_RXD); at91uc3_configgpio(PINMUX_USART2_TXD); #endif /* Then configure the console here (if it is not going to be configured * by up_earlyserialinit()). */ #if defined(HAVE_SERIAL_CONSOLE) && !defined(CONFIG_USE_EARLYSERIALINIT) usart_configure(AVR32_CONSOLE_BASE, AVR32_CONSOLE_BAUD, AVR32_CONSOLE_PARITY, AVR32_CONSOLE_BITS, (bool)AVR32_CONSOLE_2STOP); #endif } #endif /****************************************************************************** * Name: up_lowputc * * Description: * Output one byte on the serial console * ******************************************************************************/ void up_lowputc(char ch) { #ifdef HAVE_SERIAL_CONSOLE /* Wait until the TX to become ready */ while (usart_getreg(AVR32_CONSOLE_BASE, USART_CSR_TXRDY) == 0); /* Then send the character */ usart_putreg(AVR32_CONSOLE_BASE, AVR32_USART_THR_OFFSET, (uint32_t)ch); #endif }