|
@@ -5,28 +5,51 @@
|
|
|
|
|
|
#include "usbcas.h"
|
|
|
|
|
|
-/*
|
|
|
- * FM nybble patters; these should be transmitted LSB first
|
|
|
- * and XOR'd with the MSB of the pattern byte last transmitted.
|
|
|
- */
|
|
|
-static const uint8_t fm_nyb[16] = {
|
|
|
- 0x33, 0xcd, 0xcb, 0x35, 0xd3, 0x2d, 0x2b, 0xd5,
|
|
|
- 0xb3, 0x4d, 0x4b, 0xb5, 0x53, 0xad, 0xab, 0x55
|
|
|
-};
|
|
|
+#define BITS_PER_PATTERN_BYTE 2
|
|
|
+#define PATTERN_BIT_MASK ((1 << BITS_PER_PATTERN_BYTE) - 1)
|
|
|
+
|
|
|
+#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));
|
|
|
+}
|
|
|
|
|
|
/*
|
|
|
- * Send the least significant nybble of this data item and return
|
|
|
- * the next nybble, if any. Check with fmtx_send_ready() before
|
|
|
- * calling this function.
|
|
|
+ * Sends another chunk of output.
|
|
|
*/
|
|
|
-uint8_t fmtx_send_nybble(uint8_t d)
|
|
|
+uint8_t fmtx_send_more(void)
|
|
|
{
|
|
|
- uint8_t nyb = fm_nyb[d & 15] ^ parity;
|
|
|
+ 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);
|
|
|
- UDR1 = nyb;
|
|
|
- parity = -((int8_t)nyb < 0);
|
|
|
- return d >> 4;
|
|
|
+ UDR1 = pattern;
|
|
|
+ data = d >> 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();
|
|
|
}
|
|
|
|
|
|
/*
|
|
@@ -42,8 +65,22 @@ void fmtx_drain(void)
|
|
|
* Initialize the USART hardware and set transmission baudrate.
|
|
|
* This is mostly the same as SerialSPI_Init().
|
|
|
*/
|
|
|
+#define FMTX_UBRRVAL(b) (SERIAL_SPI_UBBRVAL(b * 8/BITS_PER_PATTERN_BYTE))
|
|
|
+#define UBRR_MAX 4095
|
|
|
+#define FMTX_ABC80_UBRR FMTX_UBRRVAL(CAS_BAUDRATE_ABC80)
|
|
|
+
|
|
|
+#if FMTX_ABC80_UBRR > UBRR_MAX
|
|
|
+# error "UBRR overflows for standard ABC80 baud rate"
|
|
|
+#endif
|
|
|
+
|
|
|
void fmtx_init_speed(uint32_t baudrate)
|
|
|
{
|
|
|
+ uint32_t ubrrval;
|
|
|
+
|
|
|
+ ubrrval = FMTX_UBRRVAL(baudrate);
|
|
|
+ if (ubrrval > UBRR_MAX)
|
|
|
+ ubrrval = FMTX_ABC80_UBRR;
|
|
|
+
|
|
|
DDRD |= 0x28; /* PD3 = outdata, PD5 = XCK/TXLED */
|
|
|
PORTD = (PORTD & ~0x28) | (parity & 0x20) | 0x08;
|
|
|
|
|
@@ -63,9 +100,8 @@ void fmtx_init_speed(uint32_t baudrate)
|
|
|
/* Enable transmitter, but not receiver */
|
|
|
UCSR1B = (1 << TXEN1);
|
|
|
|
|
|
- /* Configure baud rate (UBBRVAL is a typo in LUFA) */
|
|
|
- /* We need two SPI ticks per symbol, hence double... */
|
|
|
- UBRR1 = SERIAL_SPI_UBBRVAL(baudrate << 1);
|
|
|
+ /* Configure baud rate */
|
|
|
+ UBRR1 = ubrrval;
|
|
|
|
|
|
/* Output the current idle state of the line */
|
|
|
UDR1 = parity;
|