|
@@ -6,8 +6,38 @@
|
|
|
|
|
|
static uint16_t one_bit_lo, one_bit_hi; /* Range for an edge being a "1" */
|
|
|
|
|
|
+static inline void fmrx_led_on(void)
|
|
|
+{
|
|
|
+ PORTB &= ~(1 << 0);
|
|
|
+}
|
|
|
+
|
|
|
+static inline void fmrx_led_off(void)
|
|
|
+{
|
|
|
+ PORTB |= (1 << 0);
|
|
|
+}
|
|
|
+
|
|
|
+static void say(const char *str)
|
|
|
+{
|
|
|
+#if 0
|
|
|
+ char c;
|
|
|
+
|
|
|
+ while ((c = *str++))
|
|
|
+ fmrx_recv_byte(c);
|
|
|
+#endif
|
|
|
+}
|
|
|
+
|
|
|
void fmrx_init(void)
|
|
|
{
|
|
|
+ uint8_t tccr1b;
|
|
|
+
|
|
|
+ /* ICP1/PD4 is input, with pullup */
|
|
|
+ DDRD &= ~(1 << 4);
|
|
|
+ PORTD |= (1 << 4);
|
|
|
+
|
|
|
+ /* PB0 is RXLED */
|
|
|
+ DDRB |= (1 << 0);
|
|
|
+ fmrx_led_off();
|
|
|
+
|
|
|
/* Stop counter */
|
|
|
TCCR1B = 0;
|
|
|
|
|
@@ -20,25 +50,36 @@ void fmrx_init(void)
|
|
|
/* Clear timer counter */
|
|
|
TCNT1 = 0;
|
|
|
|
|
|
- /* Input capture noise canceler on, no prescaler, normal mode;
|
|
|
+ /*
|
|
|
+ * Input capture noise canceler on, no prescaler, normal mode;
|
|
|
* start timer
|
|
|
*/
|
|
|
- TCCR1B = (1 << ICNC1) + (1 << CS10);
|
|
|
+ tccr1b = (1 << ICNC1) + (1 << CS10);
|
|
|
+ if (!(PIND & (1 << 4)))
|
|
|
+ tccr1b |= 1 << ICES1;
|
|
|
+ TCCR1B = tccr1b;
|
|
|
}
|
|
|
|
|
|
static uint16_t bytes;
|
|
|
|
|
|
void fmrx_set_speed(uint32_t baudrate)
|
|
|
{
|
|
|
+ char msg[256];
|
|
|
+
|
|
|
one_bit_lo = F_CPU / (4 * baudrate);
|
|
|
one_bit_hi = 3 * one_bit_lo;
|
|
|
|
|
|
bytes = 0;
|
|
|
+ sprintf(msg, "\r\nbaudrate = %lu\r\n"
|
|
|
+ "one_bit_lo = %u\r\n"
|
|
|
+ "one_bit_hi = %u\r\n",
|
|
|
+ baudrate, one_bit_lo, one_bit_hi);
|
|
|
+ say(msg);
|
|
|
}
|
|
|
|
|
|
/* We "hunt" for a block until we find this bit pattern; same as ABC800 */
|
|
|
-#define CAS_SYNC_PATTERN 0x0216
|
|
|
-#define CAS_BLOCK_LEN (1+1+2+253+1+2) /* STX+type+num+data+ETX+checksum */
|
|
|
+#define CAS_SYNC_PATTERN 0x0216 /* SYNC + STX */
|
|
|
+#define CAS_BLOCK_LEN (1+2+253+1+2) /* type+num+data+ETX+checksum */
|
|
|
|
|
|
/* Interrupt routine for edge capture */
|
|
|
ISR(TIMER1_CAPT_vect)
|
|
@@ -46,6 +87,7 @@ ISR(TIMER1_CAPT_vect)
|
|
|
static uint16_t last_edge;
|
|
|
static uint16_t data;
|
|
|
static uint8_t bits;
|
|
|
+ char tbuf[16];
|
|
|
|
|
|
uint16_t edge, delta;
|
|
|
|
|
@@ -53,27 +95,42 @@ ISR(TIMER1_CAPT_vect)
|
|
|
edge = ICR1;
|
|
|
|
|
|
delta = edge - last_edge;
|
|
|
- last_edge = edge;
|
|
|
|
|
|
if (delta < one_bit_lo) {
|
|
|
/* Early edge, no action */
|
|
|
+ say("E");
|
|
|
return;
|
|
|
} else if (delta < one_bit_hi) {
|
|
|
+ say("D");
|
|
|
/* A data bit edge */
|
|
|
- data |= 1;
|
|
|
+ data |= (1 << 15);
|
|
|
} else {
|
|
|
+ say("C");
|
|
|
/* Clock edge */
|
|
|
+ last_edge = edge;
|
|
|
if (bytes) {
|
|
|
if (--bits == 0) {
|
|
|
- fmrx_recv_byte(data);
|
|
|
+ uint8_t outbyte = data >> 8;
|
|
|
+ sprintf(tbuf, "<%02x>", outbyte);
|
|
|
+ say(tbuf);
|
|
|
+ fmrx_recv_byte(outbyte);
|
|
|
data = 0; /* To avoid false sync */
|
|
|
bits = 8;
|
|
|
- bytes--;
|
|
|
+ if (--bytes == 0)
|
|
|
+ fmrx_led_off();
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ sprintf(tbuf, "[%04x]", data);
|
|
|
+ say(tbuf);
|
|
|
+
|
|
|
+ if (data == CAS_SYNC_PATTERN) {
|
|
|
+ bits = 8;
|
|
|
+ bytes = CAS_BLOCK_LEN;
|
|
|
+ fmrx_led_on();
|
|
|
}
|
|
|
- } else if (data == CAS_SYNC_PATTERN) {
|
|
|
- bits = 8;
|
|
|
- bytes = CAS_BLOCK_LEN;
|
|
|
}
|
|
|
- data <<= 1;
|
|
|
+ data >>= 1;
|
|
|
}
|
|
|
+ sprintf(tbuf, "%u ", delta);
|
|
|
+ say(tbuf);
|
|
|
}
|