|
@@ -11,45 +11,36 @@
|
|
|
#include "fmpat.h"
|
|
|
|
|
|
static uint8_t parity = 0;
|
|
|
-static uint8_t data;
|
|
|
-static uint8_t left;
|
|
|
-
|
|
|
-static inline bool fmtx_send_ready(void) ATTR_ALWAYS_INLINE;
|
|
|
-static inline bool fmtx_send_ready(void)
|
|
|
-{
|
|
|
- return !!(UCSR1A & (1 << UDRE1));
|
|
|
-}
|
|
|
|
|
|
/*
|
|
|
- * Sends another chunk of output.
|
|
|
+ * Send stuff if we have anything to send
|
|
|
*/
|
|
|
-uint8_t fmtx_send_more(void)
|
|
|
+ISR(USART1_UDRE_vect)
|
|
|
{
|
|
|
- uint8_t l = left;
|
|
|
-
|
|
|
- if (!fmtx_send_ready())
|
|
|
- return l; /* Not ready yet */
|
|
|
-
|
|
|
- uint8_t d = data;
|
|
|
- uint8_t pattern = fm_pat[d & PATTERN_BIT_MASK] ^ parity;
|
|
|
-
|
|
|
- UCSR1A = (1 << TXC1);
|
|
|
+ static uint8_t data, left;
|
|
|
+ uint8_t pattern;
|
|
|
+
|
|
|
+ if (!left) {
|
|
|
+ /* Fetch a new byte if there is one */
|
|
|
+ int nd = fmtx_next_byte();
|
|
|
+ if (nd < 0) {
|
|
|
+ /*
|
|
|
+ * Nothing to send. Disable interrupt.
|
|
|
+ */
|
|
|
+ UCSR1B &= ~(1 << UDRIE1);
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ data = nd;
|
|
|
+ left = 8;
|
|
|
+ }
|
|
|
+
|
|
|
+ pattern = fm_pat[data & PATTERN_BIT_MASK] ^ parity;
|
|
|
+
|
|
|
+ UCSR1A = (1 << TXC1); /* Make TXC1 bit useful */
|
|
|
UDR1 = pattern;
|
|
|
data >>= BITS_PER_PATTERN_BYTE;
|
|
|
left -= BITS_PER_PATTERN_BYTE;
|
|
|
parity = -((int8_t)pattern < 0);
|
|
|
- return left;
|
|
|
-}
|
|
|
-
|
|
|
-/*
|
|
|
- * Begin sending a new byte. Use when fmtx_send_*() has returned zero.
|
|
|
- */
|
|
|
-
|
|
|
-uint8_t fmtx_send_byte(uint8_t d)
|
|
|
-{
|
|
|
- data = d;
|
|
|
- left = 8;
|
|
|
- return fmtx_send_more();
|
|
|
}
|
|
|
|
|
|
/*
|
|
@@ -57,7 +48,10 @@ uint8_t fmtx_send_byte(uint8_t d)
|
|
|
*/
|
|
|
void fmtx_drain(void)
|
|
|
{
|
|
|
- while (!(UCSR1A & (1 << TXC1)))
|
|
|
+ const uint8_t ucsr1a_mask = (1 << UDRE1) + (1 << TXC1);
|
|
|
+
|
|
|
+ /* Wait until UDRE1 and TXC1 are both set */
|
|
|
+ while ((UCSR1A & ucsr1a_mask) != ucsr1a_mask)
|
|
|
do_usb_task();
|
|
|
}
|
|
|
|
|
@@ -84,16 +78,12 @@ uint32_t fmtx_real_baudrate(uint32_t baudrate)
|
|
|
|
|
|
void fmtx_init_speed(uint32_t baudrate)
|
|
|
{
|
|
|
- uint32_t ubrrval;
|
|
|
-
|
|
|
- ubrrval = FMTX_UBRRVAL(baudrate);
|
|
|
+ UCSR1B = 0; /* Disable transmitter and ISR */
|
|
|
+ UCSR1A = (1 << TXC1); /* Clear TXC bit */
|
|
|
|
|
|
DDRD |= 0x28; /* PD3 = outdata, PD5 = XCK/TXLED */
|
|
|
PORTD = (PORTD & ~0x28) | 0x20 | (parity & 0x08);
|
|
|
|
|
|
- UCSR1B = 0; /* Disable USART */
|
|
|
- UCSR1A = 0; /* Clear error bits */
|
|
|
-
|
|
|
/*
|
|
|
* SPI master mode, mode 1, LSB first
|
|
|
* UCPHA doesn't matter, but have UCPOL=1 to that the clock
|
|
@@ -104,12 +94,20 @@ void fmtx_init_speed(uint32_t baudrate)
|
|
|
/* UBRR must be zero when the transmitter is enabled */
|
|
|
UBRR1 = 0;
|
|
|
|
|
|
- /* Enable transmitter, but not receiver */
|
|
|
+ /* Enable transmitter, but not receiver and not the ISR (yet) */
|
|
|
UCSR1B = (1 << TXEN1);
|
|
|
|
|
|
/* Configure baud rate */
|
|
|
- UBRR1 = ubrrval;
|
|
|
+ UBRR1 = FMTX_UBRRVAL(baudrate);
|
|
|
|
|
|
- /* Output the current idle state of the line */
|
|
|
+ /*
|
|
|
+ * Output the current idle state of the line. This will also
|
|
|
+ * cause the TXC bit to get set afterwards.
|
|
|
+ */
|
|
|
UDR1 = parity;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Enable transmitter ISR (which may immediately disable itself...)
|
|
|
+ */
|
|
|
+ fmtx_enable();
|
|
|
}
|