使用usart正确格式化重新定义的stdio的Printf

时间:2016-06-21 01:08:26

标签: c atmega atmel usart

我正在使用带有Atmel Studio的AVR引导程序的Atmega328p微控制器与Arduino Uno合作。

我已经设置了USART通信,它成功地可以发送我的计算机上终端读取的数据,但使用printf时格式不正确。

    /*
 * USART_HelloWorld.c
 *
 * Created: 6/20/2016 4:17:16 PM
 * Author : Kevin
 */ 

#define F_CPU 16000000
#define BAUD 9600
#define MYUBRR F_CPU/16/BAUD-1
#include <avr/io.h>
#include <stdio.h>

static int USART_Transmit(char, FILE*);
static FILE mystdout = FDEV_SETUP_STREAM(USART_Transmit, NULL, _FDEV_SETUP_WRITE);

USART_Init(unsigned int ubrr){
    UCSR0B |= (1<<TXEN0)|(RXEN0);//enables transmitting and receiving
    UCSR0C |= 3<UCSZ00; //8 bit data, 1 stop bit
    UBRR0H |= (unsigned char)(ubrr>>8);  //sets baud rate
    UBRR0L |= (unsigned char)ubrr;//sets baud rate

}
static int USART_Transmit(char data, FILE *stream){
    if(data=='\n') 
        USART_Transmit('\r',stream);
    while(!((UCSR0A)&(1<<UDRE0)));
    UDR0 = data;
    return 0;
}
int main(void)
{
    USART_Init(MYUBRR);
    stdout = &mystdout;
    printf("Server name= 0x%X\nMy name is %s\nThis number is %d!\n", 0xDEADBEEF, "Kevin", 5);
    while (1);
}

在制作时我引用了 this

我应该输出如下:

Server name=0xDEADBEEF
My name is Kevin
This number is 5!

但我明白了:

Server name= 0xBEEF
My name is 
This number is 270!

任何建议

1 个答案:

答案 0 :(得分:1)

正如Joachim Pileborg和Barmar指出的那样,我需要将%X更改为%lx并将十六进制数更改为long。完成此操作后,输出大部分都是正确的,但输出的某些位将被省略。

事实证明这是因为'\ n'的所有传输都被转换为'\ r',如源示例所示。事实证明,这不是我想要的。我删除后:

if(data=='\n') 
        USART_Transmit('\r',stream);

从Usart_Transmit函数输出格式正确。

谢谢大家的帮助。