/**************************************************************************** * arch/avr/src/atmega/atmega_lowconsole.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include "atmega_config.h" #include #include #include #include #include #include "up_internal.h" #include "atmega.h" /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ /* USART0 Baud rate settings for normal and double speed modes */ #define AVR_NORMAL_UBRR0 \ ((((BOARD_CPU_CLOCK / 16) + (CONFIG_USART0_BAUD / 2)) / (CONFIG_USART0_BAUD)) - 1) #define AVR_DBLSPEED_UBRR0 \ ((((BOARD_CPU_CLOCK / 8) + (CONFIG_USART0_BAUD / 2)) / (CONFIG_USART0_BAUD)) - 1) /* Select normal or double speed baud settings. This is a trade-off between * the sampling rate and the accuracy of the divisor for high baud rates. * * As examples, consider: * * BOARD_CPU_CLOCK=8MHz and BAUD=115200: * AVR_NORMAL_UBRR1 = 4 (rounded), actual baud = 125,000 * AVR_DBLSPEED_UBRR1 = 9 (rounded), actual baud = 111,111 * * BOARD_CPU_CLOCK=8MHz and BAUD=9600: * AVR_NORMAL_UBRR1 = 52 (rounded), actual baud = 9615 * AVR_DBLSPEED_UBRR1 = 104 (rounded), actual baud = 9615 */ #undef USART0_DOUBLE_SPEED #ifdef CONFIG_AVR_USART0 # if BOARD_CPU_CLOCK <= 4000000 # if CONFIG_USART0_BAUD <= 9600 # define AVR_UBRR0 AVR_NORMAL_UBRR0 # else # define AVR_UBRR0 AVR_DBLSPEED_UBRR0 # define USART0_DOUBLE_SPEED 1 # endif # elif BOARD_CPU_CLOCK <= 8000000 # if CONFIG_USART0_BAUD <= 19200 # define AVR_UBRR0 AVR_NORMAL_UBRR0 # else # define AVR_UBRR0 AVR_DBLSPEED_UBRR0 # define USART0_DOUBLE_SPEED 1 # endif # elif BOARD_CPU_CLOCK <= 12000000 # if CONFIG_USART0_BAUD <= 28800 # define AVR_UBRR0 AVR_NORMAL_UBRR0 # else # define AVR_UBRR0 AVR_DBLSPEED_UBRR0 # define USART0_DOUBLE_SPEED 1 # endif # elif BOARD_CPU_CLOCK <= 16000000 # if CONFIG_USART0_BAUD <= 38400 # define AVR_UBRR0 AVR_NORMAL_UBRR0 # else # define AVR_UBRR0 AVR_DBLSPEED_UBRR0 # define USART0_DOUBLE_SPEED 1 # endif # else # if CONFIG_USART0_BAUD <= 57600 # define AVR_UBRR0 AVR_NORMAL_UBRR0 # else # define AVR_UBRR0 AVR_DBLSPEED_UBRR0 # define USART0_DOUBLE_SPEED 1 # endif # endif #endif /* USART1 Baud rate settings for normal and double speed modes */ #define AVR_NORMAL_UBRR1 \ ((((BOARD_CPU_CLOCK / 16) + (CONFIG_USART1_BAUD / 2)) / (CONFIG_USART1_BAUD)) - 1) #define AVR_DBLSPEED_UBRR1 \ ((((BOARD_CPU_CLOCK / 8) + (CONFIG_USART1_BAUD / 2)) / (CONFIG_USART1_BAUD)) - 1) /* Select normal or double speed baud settings. This is a trade-off between * the sampling rate and the accuracy of the divisor for high baud rates. * * As examples, consider: * * BOARD_CPU_CLOCK=8MHz and BAUD=115200: * AVR_NORMAL_UBRR1 = 4 (rounded), actual baud = 125,000 * AVR_DBLSPEED_UBRR1 = 9 (rounded), actual baud = 111,111 * * BOARD_CPU_CLOCK=8MHz and BAUD=9600: * AVR_NORMAL_UBRR1 = 52 (rounded), actual baud = 9615 * AVR_DBLSPEED_UBRR1 = 104 (rounded), actual baud = 9615 */ #undef USART1_DOUBLE_SPEED #ifdef CONFIG_AVR_USART1 # if BOARD_CPU_CLOCK <= 4000000 # if CONFIG_USART1_BAUD <= 9600 # define AVR_UBRR1 AVR_NORMAL_UBRR1 # else # define AVR_UBRR1 AVR_DBLSPEED_UBRR1 # define USART1_DOUBLE_SPEED 1 # endif # elif BOARD_CPU_CLOCK <= 8000000 # if CONFIG_USART1_BAUD <= 19200 # define AVR_UBRR1 AVR_NORMAL_UBRR1 # else # define AVR_UBRR1 AVR_DBLSPEED_UBRR1 # define USART1_DOUBLE_SPEED 1 # endif # elif BOARD_CPU_CLOCK <= 12000000 # if CONFIG_USART1_BAUD <= 28800 # define AVR_UBRR1 AVR_NORMAL_UBRR1 # else # define AVR_UBRR1 AVR_DBLSPEED_UBRR1 # define USART1_DOUBLE_SPEED 1 # endif # elif BOARD_CPU_CLOCK <= 16000000 # if CONFIG_USART1_BAUD <= 38400 # define AVR_UBRR1 AVR_NORMAL_UBRR1 # else # define AVR_UBRR1 AVR_DBLSPEED_UBRR1 # define USART1_DOUBLE_SPEED 1 # endif # else # if CONFIG_USART1_BAUD <= 57600 # define AVR_UBRR1 AVR_NORMAL_UBRR1 # else # define AVR_UBRR1 AVR_DBLSPEED_UBRR1 # define USART1_DOUBLE_SPEED 1 # endif # endif #endif /**************************************************************************** * Private Types ****************************************************************************/ /**************************************************************************** * Private Function Prototypes ****************************************************************************/ /**************************************************************************** * Public Data ****************************************************************************/ /**************************************************************************** * Private Data ****************************************************************************/ /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: usart0_reset and usart1_reset * * Description: * Reset USART0 or USART1. * ****************************************************************************/ #ifdef CONFIG_AVR_USART0 void usart0_reset(void) { /* Clear USART configuration */ UCSR0A = 0; UCSR0B = 0; UCSR0C = 0; /* Unconfigure pins (no action needed */ #ifdef CONFIG_ARCH_CHIP_ATMEGA1284P DDRD &= ~(1 << 1); PORTD &= ~(1 << 0); #else DDRE &= ~(1 << 1); PORTE &= ~(1 << 0); #endif /* Unconfigure BAUD divisor */ UBRR0H = 0; UBRR0L = 0; } #endif #ifdef CONFIG_AVR_USART1 void usart1_reset(void) { /* Clear USART configuration */ UCSR1A = 0; UCSR1B = 0; UCSR1C = 0; /* Unconfigure pins */ DDRD &= ~(1 << 3); PORTD &= ~(1 << 2); /* Unconfigure BAUD divisor */ UBRR1H = 0; UBRR1L = 0; } #endif /**************************************************************************** * Name: usart0_configure and usart1_configure * * Description: * Configure USART0 or USART1. * ****************************************************************************/ #ifdef CONFIG_AVR_USART0 void usart0_configure(void) { uint8_t ucsr0b; uint8_t ucsr0c; /* Select normal or double speed. */ #ifdef USART0_DOUBLE_SPEED UCSR0A = (1 << U2X0); #else UCSR0A = 0; #endif /* Select baud, parity, number of bits, stop bits, etc. */ ucsr0b = ((1 << TXEN0) | (1 << RXEN0)); ucsr0c = 0; /* Select parity */ #if CONFIG_USART0_PARITY == 1 ucsr0c |= ((1 << UPM01) | (1 << UPM00)); /* Odd parity */ #elif CONFIG_USART0_PARITY == 2 ucsr0c |= (1 << UPM00); /* Even parity */ #endif /* 1 or 2 stop bits */ #if defined(CONFIG_USART0_2STOP) && CONFIG_USART0_2STOP > 0 ucsr0c |= (1 << USBS0); /* Two stop bits */ #endif /* Word size */ #if CONFIG_USART0_BITS == 5 #elif CONFIG_USART0_BITS == 6 ucsr0c |= (1 << UCSZ00); #elif CONFIG_USART0_BITS == 7 ucsr0c |= (1 << UCSZ01); #elif CONFIG_USART0_BITS == 8 ucsr0c |= ((1 << UCSZ00) | (1 << UCSZ01)); #elif CONFIG_USART0_BITS == 9 ucsr0c |= ((1 << UCSZ0) | (1 << UCSZ01)); ucsr0b |= (1 << UCSZ02); #else # error "Unsupported word size" #endif UCSR0B = ucsr0b; UCSR0C = ucsr0c; #ifdef CONFIG_ARCH_CHIP_ATMEGA1284P /* Pin Configuration: None necessary, Port D bits 0&1 are automatically * configured: * * Port D, Bit 0: RXD0, USART0 Receive Pin. Receive Data (Data input pin * for the USART0). When the USART0 receiver is enabled this pin is * configured as an input regardless of the value of DDRD0. When the * USART0 forces this pin to be an input, a logical one in PORTD0 will * turn on the internal pull-up. * * Port D, Bit 1: TXD0, USART0 Transmit pin. */ DDRD |= (1 << 1); /* Force Port D pin 1 to be an output -- should not be necessary */ PORTD |= (1 << 0); /* Set pull-up on Port D pin 0 */ #else /* Pin Configuration: None necessary, Port E bits 0&1 are automatically * configured: * * Port E, Bit 0: RXD0, USART0 Receive Pin. Receive Data (Data input pin * for the USART0). When the USART0 receiver is enabled this pin is * configured as an input regardless of the value of DDRE0. When the * USART0 forces this pin to be an input, a logical one in PORTE0 will * turn on the internal pull-up. * * Port E, Bit 1: TXD0, USART0 Transmit pin. * * REVISIT: According to table 41, TXD0 is also automatically configured. * However, this is not explicitly stated in the text. */ DDRE |= (1 << 1); /* Force Port E pin 1 to be an output -- might not be necessary */ PORTE |= (1 << 0); /* Set pull-up on Port E pin 0 */ #endif /* Set the baud rate divisor */ UBRR0H = AVR_UBRR0 >> 8; UBRR0L = AVR_UBRR0 & 0xff; } #endif #ifdef CONFIG_AVR_USART1 void usart1_configure(void) { uint8_t ucsr1b; uint8_t ucsr1c; /* Select normal or double speed. */ #ifdef USART1_DOUBLE_SPEED UCSR1A = (1 << U2X1); #else UCSR1A = 0; #endif /* Select baud, parity, number of bits, stop bits, etc. */ ucsr1b = ((1 << TXEN1) | (1 << RXEN1)); ucsr1c = 0; /* Select parity */ #if CONFIG_USART1_PARITY == 1 ucsr1c |= ((1 << UPM11) | (1 << UPM10)); /* Odd parity */ #elif CONFIG_USART1_PARITY == 2 ucsr1c |= (1 << UPM11); /* Even parity */ #endif /* 1 or 2 stop bits */ #if defined(CONFIG_USART1_2STOP) && CONFIG_USART1_2STOP > 0 ucsr1c |= (1 << USBS1); /* Two stop bits */ #endif /* Word size */ #if CONFIG_USART1_BITS == 5 #elif CONFIG_USART1_BITS == 6 ucsr1c |= (1 << UCSZ10); #elif CONFIG_USART1_BITS == 7 ucsr1c |= (1 << UCSZ11); #elif CONFIG_USART1_BITS == 8 ucsr1c |= ((1 << UCSZ10) | (1 << UCSZ11)); #elif CONFIG_USART1_BITS == 9 ucsr1c |= ((1 << CSZ10) | (1 << UCSZ11)); ucsr1b |= (1 << UCSZ12); #else # error "Unsupported word size" #endif UCSR1B = ucsr1b; UCSR1C = ucsr1c; /* Pin Configuration: None necessary, Port D bits 2&3 are automatically * configured: * * Port D, Bit 2: RXD1, Receive Data (Data input pin for the USART1). * When the USART1 receiver is enabled this pin is configured as an * input regardless of the value of DDD2. When the USART forces this * pin to be an input, the pull-up can still be controlled by the * PORTD2 bit. * Port D, Bit 3: TXD1, Transmit Data (Data output pin for the USART1). * When the USART1 Transmitter is enabled, this pin is configured as * an output regardless of the value of DDD3. */ DDRD |= (1 << 3); /* Force Port D pin 3 to be an output -- should not be necessary */ PORTD |= (1 << 2); /* Set pull-up on port D pin 2 */ /* Set the baud rate divisor */ UBRR1H = AVR_UBRR1 >> 8; UBRR1L = AVR_UBRR1 & 0xff; } #endif /**************************************************************************** * Name: up_consoleinit * * Description: * Initialize a console for debug output. This function is called very * early in the initialization sequence to configure the serial console * uart (only). * ****************************************************************************/ void up_consoleinit(void) { #ifdef HAVE_SERIAL_CONSOLE # if defined(CONFIG_USART0_SERIAL_CONSOLE) usart0_configure(); # elif defined(CONFIG_USART1_SERIAL_CONSOLE) usart1_configure(); # endif #endif } /**************************************************************************** * Name: up_lowputc * * Description: * Output one byte on the serial console * ****************************************************************************/ void up_lowputc(char ch) { #ifdef HAVE_SERIAL_CONSOLE # if defined(CONFIG_USART0_SERIAL_CONSOLE) while ((UCSR0A & (1 << UDRE0)) == 0); UDR0 = ch; # elif defined(CONFIG_USART1_SERIAL_CONSOLE) while ((UCSR1A & (1 << UDRE1)) == 0); UDR1 = ch; # endif #endif }