2017-05-05 54 views
0

下面是一个代码运行在一个ATMEGA328P。它应该每秒发送一次“abcdef”到我的电脑。但是,它每秒只给我发“ab”。这里有什么问题?AVR USART仅发送2字符

#include <avr/io.h> 
#include <util/delay.h> 

void USART_transmit(unsigned char data); 
void print(const unsigned char *buffer, size_t n); 
void print(const char* str); 

int main(void) { 
    // Serial.begin(115200) 
    UCSR0B |= (1<<TXEN0); 
    UBRR0L = 8; 

    while(1){ 
     // Serial.write(i) 
     print("abcdef"); 

     _delay_ms(1000); 
    } 
} 

void USART_transmit(const uint8_t data) { 
    /* wait for empty transmit buffer */ 
    while (!UCSR0A & (1<<UDRE0)); 
    UDR0 = data; 
} 

void print(const uint8_t *buffer, size_t n) { 
    while(n--){ 
     USART_transmit(*buffer++); //** 
    } 
} 

void print(const char *str) { 
    if(strlen(str) != 0) print((const uint8_t *) str, strlen(str)); 
} 

的代码导致:

ababababababababababab ...

USART_transmit(*buffer++);USART_transmit(n + 48);更改(+48转换为字符)导致:

5454545454545454545454545454 ...

所以我猜这个循环不应该被阻止?

回答

2

“数据寄存器空”检查是错误的。

while (!UCSR0A & (1<<UDRE0)); 

应该

while (!(UCSR0A & (1 << UDRE0))); 

在你的情况,检查没有阻止,直到缓冲区为空。我认为一个字节在USART输出缓冲区中被缓冲,一个字节在UDR中处于待处理状态。然后丢弃每个额外的字节,这就是为什么你只看到“ab”。