2019-03-26 10:35:41 +00:00
|
|
|
|
#include "usart.h"
|
|
|
|
|
|
|
|
|
|
void USART::init()
|
|
|
|
|
{
|
2019-03-27 10:45:30 +00:00
|
|
|
|
UCSR0A = _BV(U2X0);
|
|
|
|
|
|
2019-04-02 08:59:37 +00:00
|
|
|
|
UCSR0B = _BV(RXEN0) | _BV(TXEN0) | _BV(RXCIE0);
|
2019-03-26 10:35:41 +00:00
|
|
|
|
|
|
|
|
|
// Einstellen des Datenformats: 8 Datenbits, 1 Stoppbit
|
|
|
|
|
UCSR0C = _BV(UCSZ00) |_BV(UCSZ01);// (1<<URSEL0)|(1<<UCSZ10)|(1<<UCSZ00);
|
|
|
|
|
|
|
|
|
|
// setze Baudrate
|
2019-03-27 10:45:30 +00:00
|
|
|
|
UBRR0H = (((F_CPU / (8UL * BAUDRATE))-1) >> 8) & 0xFF;
|
|
|
|
|
UBRR0L = ((F_CPU / (8UL * BAUDRATE))-1) & 0xFF;
|
2019-03-26 10:35:41 +00:00
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-02 08:59:37 +00:00
|
|
|
|
void USART::clearInputBuffer()
|
2019-03-29 12:35:18 +00:00
|
|
|
|
{
|
|
|
|
|
uint8_t dummy;
|
2019-04-01 06:37:00 +00:00
|
|
|
|
do
|
|
|
|
|
{
|
2019-03-29 12:35:18 +00:00
|
|
|
|
dummy = UDR0;
|
2019-04-01 06:37:00 +00:00
|
|
|
|
_delay_us((1000000 / BAUDRATE) * 16); // Warte <20>bertragungszeit von 16 Bit ab
|
|
|
|
|
}
|
|
|
|
|
while (UCSR0A & (1<<RXC0));
|
2019-03-29 12:35:18 +00:00
|
|
|
|
|
|
|
|
|
if(dummy) // taeusche dummy Verwendung vor
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-26 10:35:41 +00:00
|
|
|
|
void USART::writeByte(uint8_t b)
|
|
|
|
|
{
|
2019-03-27 10:45:30 +00:00
|
|
|
|
while (!(UCSR0A & (1<<UDRE0)));
|
2019-03-26 10:35:41 +00:00
|
|
|
|
UDR0 = b;
|
2019-03-28 12:32:24 +00:00
|
|
|
|
|
2019-03-29 12:35:18 +00:00
|
|
|
|
//while(!(UCSR0A & _BV(TXC0)));
|
2019-03-26 10:35:41 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-03-26 14:02:58 +00:00
|
|
|
|
|
|
|
|
|
void USART::writeInt(uint16_t v)
|
|
|
|
|
{
|
2019-03-28 12:32:24 +00:00
|
|
|
|
|
|
|
|
|
|
2019-03-27 10:45:30 +00:00
|
|
|
|
while (!(UCSR0A & (1<<UDRE0)));
|
2019-03-26 14:02:58 +00:00
|
|
|
|
UDR0 = v & 0xFF;
|
2019-03-27 10:45:30 +00:00
|
|
|
|
|
2019-03-26 14:02:58 +00:00
|
|
|
|
v >>= 8;
|
2019-03-27 10:45:30 +00:00
|
|
|
|
while (!(UCSR0A & (1<<UDRE0)));
|
2019-03-26 14:02:58 +00:00
|
|
|
|
UDR0 = v & 0xFF;
|
2019-03-28 12:32:24 +00:00
|
|
|
|
|
2019-03-29 12:35:18 +00:00
|
|
|
|
//while(!(UCSR0A & _BV(TXC0)));
|
2019-03-26 14:02:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-03-27 14:48:36 +00:00
|
|
|
|
void USART::writeStr(const char* str, uint8_t len)
|
|
|
|
|
{
|
|
|
|
|
writeByte(len);
|
|
|
|
|
while(len--)
|
|
|
|
|
writeByte(*str++);
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-28 14:22:17 +00:00
|
|
|
|
uint8_t USART::writeBlock(uint8_t* ptr, uint8_t len)
|
|
|
|
|
{
|
|
|
|
|
writeByte(len);
|
|
|
|
|
|
|
|
|
|
uint8_t crc = 0;
|
|
|
|
|
while(len--)
|
|
|
|
|
{
|
|
|
|
|
writeByte(*ptr);
|
|
|
|
|
crc ^= *ptr++;
|
|
|
|
|
for (uint8_t i = 0; i < 8; i++)
|
|
|
|
|
{
|
|
|
|
|
if (crc & 1)
|
|
|
|
|
crc ^= CRC7_POLY;
|
|
|
|
|
crc >>= 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
writeByte(crc);
|
|
|
|
|
|
|
|
|
|
return readByte();
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-26 10:35:41 +00:00
|
|
|
|
uint8_t USART::readByte()
|
|
|
|
|
{
|
|
|
|
|
while (!(UCSR0A & (1<<RXC0)));
|
|
|
|
|
return UDR0;
|
|
|
|
|
}
|
2019-03-26 14:02:58 +00:00
|
|
|
|
|
|
|
|
|
uint16_t USART::readInt()
|
|
|
|
|
{
|
|
|
|
|
uint16_t v = readByte();
|
|
|
|
|
v |= readByte() << 8;
|
|
|
|
|
return v;
|
|
|
|
|
}
|
2019-03-29 12:35:18 +00:00
|
|
|
|
|
2019-04-02 08:59:37 +00:00
|
|
|
|
void USART::nextByte(uint8_t byte)
|
|
|
|
|
{
|
|
|
|
|
switch(seq)
|
|
|
|
|
{
|
|
|
|
|
case IDLE:
|
|
|
|
|
{
|
|
|
|
|
if(byte > MAX_BLOCK_SIZE)
|
|
|
|
|
{
|
|
|
|
|
clearInputBuffer();
|
|
|
|
|
writeByte(MSG_FAIL);
|
|
|
|
|
seq = BlockSequence::IDLE;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
block_buffer[0] = byte;
|
|
|
|
|
crc = 0;
|
|
|
|
|
block_pos = 0;
|
|
|
|
|
seq = BlockSequence::LEN;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case LEN:
|
|
|
|
|
{
|
|
|
|
|
block_buffer[block_pos] = byte;
|
|
|
|
|
seq = BlockSequence::DATA;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case DATA:
|
|
|
|
|
{
|
|
|
|
|
block_buffer[block_pos] = byte;
|
|
|
|
|
crc ^= byte;
|
|
|
|
|
for (uint8_t i = 0; i < 8; i++)
|
|
|
|
|
{
|
|
|
|
|
if (crc & 1)
|
|
|
|
|
crc ^= CRC7_POLY;
|
|
|
|
|
crc >>= 1;
|
|
|
|
|
}
|
|
|
|
|
if(block_pos == block_buffer[0])
|
|
|
|
|
seq = BlockSequence::CRC;
|
|
|
|
|
else if(block_pos >= block_buffer[0])
|
|
|
|
|
{
|
|
|
|
|
clearInputBuffer();
|
|
|
|
|
writeByte(MSG_FAIL);
|
|
|
|
|
seq = BlockSequence::IDLE;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case CRC:
|
|
|
|
|
{
|
|
|
|
|
block_buffer[block_pos] = byte;
|
|
|
|
|
crc ^= byte;
|
|
|
|
|
for (uint8_t i = 0; i < 8; i++)
|
|
|
|
|
{
|
|
|
|
|
if (crc & 1)
|
|
|
|
|
crc ^= CRC7_POLY;
|
|
|
|
|
crc >>= 1;
|
|
|
|
|
}
|
|
|
|
|
seq = BlockSequence::END;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case END:
|
|
|
|
|
{
|
|
|
|
|
clearInputBuffer();
|
|
|
|
|
writeByte(crc == 0 ? MSG_OK : MSG_FAIL);
|
|
|
|
|
seq = BlockSequence::IDLE;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
block_pos++;
|
|
|
|
|
}
|
|
|
|
|
|
2019-03-29 12:35:18 +00:00
|
|
|
|
void USART::readBlock(uint8_t* ptr, uint8_t offset)
|
|
|
|
|
{
|
|
|
|
|
ptr += offset;
|
2019-04-01 14:08:32 +00:00
|
|
|
|
uint8_t crc = 0x7F;
|
2019-04-01 06:37:00 +00:00
|
|
|
|
do
|
2019-03-29 12:35:18 +00:00
|
|
|
|
{
|
2019-04-01 06:37:00 +00:00
|
|
|
|
uint8_t len = readByte();
|
2019-04-01 14:08:32 +00:00
|
|
|
|
|
|
|
|
|
if(len == 0x80) // out of sync, war bereits stoppbyte
|
|
|
|
|
{
|
|
|
|
|
writeByte(MSG_FAIL);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
else if(len > MAX_BLOCK_SIZE)
|
2019-04-01 06:37:00 +00:00
|
|
|
|
len = 0;
|
2019-03-29 12:35:18 +00:00
|
|
|
|
|
2019-04-01 14:08:32 +00:00
|
|
|
|
crc = 0;
|
|
|
|
|
|
2019-04-01 06:37:00 +00:00
|
|
|
|
for(uint8_t k = 0; k <= len; k++) // len + 1 Durchg<68>nge (+ crc)
|
2019-03-29 12:35:18 +00:00
|
|
|
|
{
|
2019-04-01 06:37:00 +00:00
|
|
|
|
uint8_t next = readByte();
|
|
|
|
|
|
|
|
|
|
crc ^= next;
|
|
|
|
|
for (uint8_t i = 0; i < 8; i++)
|
|
|
|
|
{
|
|
|
|
|
if (crc & 1)
|
|
|
|
|
crc ^= CRC7_POLY;
|
|
|
|
|
crc >>= 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(k < len)
|
|
|
|
|
ptr[k] = next;
|
2019-03-29 12:35:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-01 06:37:00 +00:00
|
|
|
|
|
2019-04-02 08:59:37 +00:00
|
|
|
|
clearInputBuffer(); // leere Eingangspuffer
|
2019-04-01 06:37:00 +00:00
|
|
|
|
|
|
|
|
|
writeByte(crc == 0 ? MSG_OK : MSG_FAIL);
|
2019-03-29 12:35:18 +00:00
|
|
|
|
}
|
2019-04-01 06:37:00 +00:00
|
|
|
|
while(crc != 0);
|
2019-03-29 12:35:18 +00:00
|
|
|
|
}
|