Go Down

Topic: Problem with Serial on Arduino Pro 328 3.3V 8MHz (Read 4257 times) previous topic - next topic

EmilyJane

Jun 20, 2009, 04:28 pm Last Edit: Jun 20, 2009, 04:42 pm by EmilyJane Reason: 1
Using Arduino 16 on OS X 10.5.7

When this sketch is run on an  Arduino Pro [glow]328[/glow] 3.3V 8MHz with the IDE Serial Monitor set to 9600 baud :

Code: [Select]
void setup()  {
 Serial.begin(9600);
}

void loop()  {
 Serial.println("hello world");
 delay(1000);
}


the resulting output is this:

Code: [Select]
?Æ?åÅDó?Æ?åÅDó?Æ?åÅDó?Æ?åÅDó?Æ?åÅD

If you set the Serial Monitor to 19200 baud, this is the output:

Code: [Select]
hello world
hello world
hello world
hello world
hello world


If I compile and run the same sketch on an Arduino Duemilanove (328 16MHz) I get the correct output at the programmed baud rate.

Also, on the Arduino Pro [glow]168[/glow] 3.3V 8MHz the sketch runs correctly.

I have double checked the Board settings in Arduino 16.

Has anyone found the solution to this problem?

kg4wsv

Which board do you have selected in the boards menu?  Sounds like you may have selected a 16MHz arduino, and have 16MHz code on 8MHz hardware.

IIRC, you want the lilypad selection for most 8MHz arduinos.

-j

EmilyJane

Quote
I have double checked the Board settings in Arduino 16


Arduino 16 has settings for both the 8 MHz Pros. It is no longer necessary to pretend you have a Lilypad.

Besides, if that were the case, wouldn't the serial port run half as fast, not twice as fast?

mellis

Maybe you actually got a 5V / 16 MHz Arduino Pro?

From what EmilyJane described, the processor is actually running at twice the speed they are expecting.  Thus, as mellis has suggested, you probably have a 16MHz board instead of 8MHz.

(you're asking the board to communicate at 9600bps, but it's actually xmitting at 19200)

b

EmilyJane

Ah, those suggestions actually make sense. Unfortunately, the crystal/resonator is clearly marked with an 8, just like the 168 8 MHz Pro that works correctly, so unless it is oscillating at twice the marked frequency, that is not the problem.

I'll print out the contents of the baud rate registers and see if what the chip is being told to do makes sense based on what it is actually doing.

EmilyJane

Okay, I have found the problem. USART0 register UBRR0 is being programmed to 51 which is correct for an 8 MHz clock but register U2X0 is being incorrectly programmed to a 1 when it should be a 0.

Wow, that was a puzzler!  >:(

EmilyJane

#7
Jun 22, 2009, 12:34 am Last Edit: Jun 22, 2009, 12:39 am by EmilyJane Reason: 1
In iom328p.h, U2X0 is incorrectly #defined to be a 1 when it should be a 0, unless the designers meant to use Asynchronous Double Speed mode, in which case UBRR0 is being programmed incorrectly.

EmilyJane

mellis, what happened to wiring_serial.c? It doesn't seem to be included in Arduino 16.

I'm trying to find where the baud rate registers get loaded.

BenF

"In iom328p.h, U2X0 is incorrectly #defined to be a 1 when it should be a 0"

I think U2X0 defines the bit mask (bit 1 of register ucsra)  and not the value of the bit as you seem to suggest.

The baud rate registers gets loaded when you call Serial.begin and you will find the source in .."\hardware\cores\arduino\HardwareSerial.cpp".

ucsra is never written to by HardwareSerial (it is only used as a status register to check if tx data register is empty). Atmel states in the 328 datasheet that bit U2X0 of ucsra defaults to 0. I don't have have a 3.3V version to check, but my suspicion is that U2X0 will be set by the bootloader (if the value is indeed "1") and retains its value when your sketch starts.

If above is the case, setting of the baud rate divisors will indeed be wrong as you suggest.

EmilyJane

#10
Jun 24, 2009, 10:29 am Last Edit: Jun 24, 2009, 11:11 am by EmilyJane Reason: 1
Yes, I discovered that the #define was a bit offset and not a value in om328p.h. It turns out that at some point the value of U2X0 is written to "1", as you can verify by printing the value of UCSR0A.

This is only the case for the 328 version of the 3.3V Pro. In the 168 version of the 3.3V Pro, it is initialized to "0" as it should be.

Does anyone know where that initialization occurs? Where is the source for the bootloader?


Okay, this may be the problem:
Code: [Select]
atmega328_pro8: CFLAGS += '-DMAX_TIME_COUNT=F_CPU>>4' '-DNUM_LED_FLASHES=1' -DBAUD_RATE=57600 -DDOUBLE_SPEED


from the bootloader makefile. It is the only processor/clock speed combo that selects double speed.

From the bootloader source:

Code: [Select]
#ifdef DOUBLE_SPEED
     UCSR0A = (1<<U2X0); //Double speed mode USART0
     UBRR0L = (uint8_t)(F_CPU/(BAUD_RATE*8L)-1);
     UBRR0H = (F_CPU/(BAUD_RATE*8L)-1) >> 8;



BenF

Mystery solved then!

As long as HardwareSerial never initializes UCSR0A I would argue that the fix should be in HardwareSerial. E.g. test for U2X0 in UCSR0A and set baudrate divisors accordingly (this would be in HardwareSerial::begin).

EmilyJane

I agree. Especially since that would obviate having to re-flash no telling how many 328 bootloaders.

I gather that the U2X bit was being used as a quick cheat for compiling the bootloader (hey, if it all works at 16MHz, just set the U2X bit on the 8MHz parts, and we don't have to make any other #define's in the code).  I'll look more into the bootloader later.

Here is the fixed HardwareSerial.h and HardwareSerial.cpp:

HardwareSerial.h
Code: [Select]

/*
 HardwareSerial.h - Hardware serial library for Wiring
 Copyright (c) 2006 Nicholas Zambetti.  All right reserved.

 This library is free software; you can redistribute it and/or
 modify it under the terms of the GNU Lesser General Public
 License as published by the Free Software Foundation; either
 version 2.1 of the License, or (at your option) any later version.

 This library is distributed in the hope that it will be useful,
 but WITHOUT ANY WARRANTY; without even the implied warranty of
 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 Lesser General Public License for more details.

 You should have received a copy of the GNU Lesser General Public
 License along with this library; if not, write to the Free Software
 Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
*/

#ifndef HardwareSerial_h
#define HardwareSerial_h

#include <inttypes.h>

#include "Print.h"

struct ring_buffer;

class HardwareSerial : public Print
{
 private:
   ring_buffer *_rx_buffer;
   volatile uint8_t *_ubrrh;
   volatile uint8_t *_ubrrl;
   volatile uint8_t *_ucsra;
   volatile uint8_t *_ucsrb;
   volatile uint8_t *_udr;
   uint8_t _rxen;
   uint8_t _txen;
   uint8_t _rxcie;
   uint8_t _udre;
   uint8_t _u2x;
 public:
   HardwareSerial(ring_buffer *rx_buffer,
     volatile uint8_t *ubrrh, volatile uint8_t *ubrrl,
     volatile uint8_t *ucsra, volatile uint8_t *ucsrb,
     volatile uint8_t *udr,
     uint8_t rxen, uint8_t txen, uint8_t rxcie, uint8_t udre, uint8_t u2x);
   void begin(long);
   uint8_t available(void);
   int read(void);
   void flush(void);
   virtual void write(uint8_t);
   using Print::write; // pull in write(str) and write(buf, size) from Print
};

extern HardwareSerial Serial;

#if defined(__AVR_ATmega1280__)
extern HardwareSerial Serial1;
extern HardwareSerial Serial2;
extern HardwareSerial Serial3;
#endif

#endif


and HardwareSerial.cpp

Code: [Select]

/*
 HardwareSerial.cpp - Hardware serial library for Wiring
 Copyright (c) 2006 Nicholas Zambetti.  All right reserved.

 This library is free software; you can redistribute it and/or
 modify it under the terms of the GNU Lesser General Public
 License as published by the Free Software Foundation; either
 version 2.1 of the License, or (at your option) any later version.

 This library is distributed in the hope that it will be useful,
 but WITHOUT ANY WARRANTY; without even the implied warranty of
 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 Lesser General Public License for more details.

 You should have received a copy of the GNU Lesser General Public
 License along with this library; if not, write to the Free Software
 Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 
 Modified 23 November 2006 by David A. Mellis
*/

#include <stdio.h>
#include <string.h>
#include <inttypes.h>
#include "wiring.h"
#include "wiring_private.h"

#include "HardwareSerial.h"

// Define constants and variables for buffering incoming serial data.  We're
// using a ring buffer (I think), in which rx_buffer_head is the index of the
// location to which to write the next incoming character and rx_buffer_tail
// is the index of the location from which to read.
#define RX_BUFFER_SIZE 128

struct ring_buffer {
 unsigned char buffer[RX_BUFFER_SIZE];
 int head;
 int tail;
};

ring_buffer rx_buffer = { { 0 }, 0, 0 };

#if defined(__AVR_ATmega1280__)
ring_buffer rx_buffer1 = { { 0 }, 0, 0 };
ring_buffer rx_buffer2 = { { 0 }, 0, 0 };
ring_buffer rx_buffer3 = { { 0 }, 0, 0 };
#endif

inline void store_char(unsigned char c, ring_buffer *rx_buffer)
{
     int i = (rx_buffer->head + 1) % RX_BUFFER_SIZE;

     // if we should be storing the received character into the location
     // just before the tail (meaning that the head would advance to the
     // current location of the tail), we're about to overflow the buffer
     // and so we don't write the character or advance the head.
     if (i != rx_buffer->tail) {
           rx_buffer->buffer[rx_buffer->head] = c;
           rx_buffer->head = i;
     }
}

#if defined(__AVR_ATmega1280__)

SIGNAL(SIG_USART0_RECV)
{
     unsigned char c = UDR0;
 store_char(c, &rx_buffer);
}

SIGNAL(SIG_USART1_RECV)
{
     unsigned char c = UDR1;
 store_char(c, &rx_buffer1);
}

SIGNAL(SIG_USART2_RECV)
{
     unsigned char c = UDR2;
 store_char(c, &rx_buffer2);
}

SIGNAL(SIG_USART3_RECV)
{
     unsigned char c = UDR3;
 store_char(c, &rx_buffer3);
}

#else

#if defined(__AVR_ATmega8__)
SIGNAL(SIG_UART_RECV)
#else
SIGNAL(USART_RX_vect)
#endif
{
#if defined(__AVR_ATmega8__)
     unsigned char c = UDR;
#else
     unsigned char c = UDR0;
#endif
 store_char(c, &rx_buffer);
}

#endif

// Constructors ////////////////////////////////////////////////////////////////

HardwareSerial::HardwareSerial(ring_buffer *rx_buffer,
 volatile uint8_t *ubrrh, volatile uint8_t *ubrrl,
 volatile uint8_t *ucsra, volatile uint8_t *ucsrb,
 volatile uint8_t *udr,
 uint8_t rxen, uint8_t txen, uint8_t rxcie, uint8_t udre, uint8_t u2x)
{
 _rx_buffer = rx_buffer;
 _ubrrh = ubrrh;
 _ubrrl = ubrrl;
 _ucsra = ucsra;
 _ucsrb = ucsrb;
 _udr = udr;
 _rxen = rxen;
 _txen = txen;
 _rxcie = rxcie;
 _udre = udre;
 _u2x = u2x;
}

// Public Methods //////////////////////////////////////////////////////////////

void HardwareSerial::begin(long speed)
{
     *_ubrrh = ((F_CPU / 16 + speed / 2) / speed - 1) >> 8;
     *_ubrrl = ((F_CPU / 16 + speed / 2) / speed - 1);
     cbi(*_ucsra, _u2x);
 sbi(*_ucsrb, _rxen);
 sbi(*_ucsrb, _txen);
 sbi(*_ucsrb, _rxcie);
}

uint8_t HardwareSerial::available(void)
{
     return (RX_BUFFER_SIZE + _rx_buffer->head - _rx_buffer->tail) % RX_BUFFER_SIZE;
}

int HardwareSerial::read(void)
{
     // if the head isn't ahead of the tail, we don't have any characters
     if (_rx_buffer->head == _rx_buffer->tail) {
           return -1;
     } else {
           unsigned char c = _rx_buffer->buffer[_rx_buffer->tail];
           _rx_buffer->tail = (_rx_buffer->tail + 1) % RX_BUFFER_SIZE;
           return c;
     }
}

void HardwareSerial::flush()
{
     // don't reverse this or there may be problems if the RX interrupt
     // occurs after reading the value of rx_buffer_head but before writing
     // the value to rx_buffer_tail; the previous value of rx_buffer_head
     // may be written to rx_buffer_tail, making it appear as if the buffer
     // don't reverse this or there may be problems if the RX interrupt
     // occurs after reading the value of rx_buffer_head but before writing
     // the value to rx_buffer_tail; the previous value of rx_buffer_head
     // may be written to rx_buffer_tail, making it appear as if the buffer
     // were full, not empty.
     _rx_buffer->head = _rx_buffer->tail;
}

void HardwareSerial::write(uint8_t c)
{
     while (!((*_ucsra) & (1 << _udre)))
           ;

     *_udr = c;
}

// Preinstantiate Objects //////////////////////////////////////////////////////

#if defined(__AVR_ATmega8__)
HardwareSerial Serial(&rx_buffer, &UBRRH, &UBRRL, &UCSRA, &UCSRB, &UDR, RXEN, TXEN, RXCIE, UDRE, U2X);
#else
HardwareSerial Serial(&rx_buffer, &UBRR0H, &UBRR0L, &UCSR0A, &UCSR0B, &UDR0, RXEN0, TXEN0, RXCIE0, UDRE0, U2X0);
#endif

#if defined(__AVR_ATmega1280__)
HardwareSerial Serial1(&rx_buffer1, &UBRR1H, &UBRR1L, &UCSR1A, &UCSR1B, &UDR1, RXEN1, TXEN1, RXCIE1, UDRE1, U2X1);
HardwareSerial Serial2(&rx_buffer2, &UBRR2H, &UBRR2L, &UCSR2A, &UCSR2B, &UDR2, RXEN2, TXEN2, RXCIE2, UDRE2, U2X2);
HardwareSerial Serial3(&rx_buffer3, &UBRR3H, &UBRR3L, &UCSR3A, &UCSR3B, &UDR3, RXEN3, TXEN3, RXCIE3, UDRE3, U2X3);
#endif


Compiled and tested on atmega328P @ 16MHz.



b

EmilyJane


Go Up