123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154 |
- #include "usbcas.h"
- static inline void fmrx_led_on(void)
- {
-
- }
- static inline void fmrx_led_off(void)
- {
-
- }
- void fmrx_init(void)
- {
- uint8_t tccr1b;
-
- DDRD &= ~(1 << 4);
- PORTD |= (1 << 4);
-
- DDRB |= (1 << 0);
- fmrx_led_off();
-
- TCCR1B = 0;
-
- TCCR1A = 0;
-
- TIFR1 = TIMSK1 = 1 << ICIE1;
-
- TCNT1 = 0;
-
- tccr1b = (1 << ICNC1) + (0 << CS10);
- if (!(PIND & (1 << 4)))
- tccr1b |= 1 << ICES1;
- TCCR1B = tccr1b;
- }
- static uint16_t one_bit_lo, one_bit_hi, one_bit_timeout;
- static uint16_t bytes;
- static uint16_t last_edge;
- static uint16_t data;
- static uint8_t bits;
- static uint16_t bytes;
- static bool started;
- static inline void enable_timeout(uint16_t when)
- {
- OCR1A = when;
- TIMSK1 = TIFR1 = (1 << ICIE1) | (1 << OCIE1A);
- }
- static inline void disable_timeout(void)
- {
- TIFR1 = TIMSK1 = (1 << ICIE1);
- }
- void fmrx_set_speed(uint32_t baudrate)
- {
- uint8_t prescale = 1;
-
- if (baudrate < 976) {
- baudrate <<= 3;
- prescale++;
- }
- TCCR1B &= ~7;
- one_bit_lo = F_CPU / (4 * baudrate);
- one_bit_hi = 3 * one_bit_lo;
- one_bit_timeout = 16 * one_bit_lo;
- disable_timeout();
- started = false;
- bits = bytes = 0;
- last_edge = TCNT1;
- TCCR1B |= prescale;
- }
- #define CAS_SYNC_PATTERN 0x0216
- #define CAS_BLOCK_LEN (1+2+253+1+2)
- static inline ATTR_ALWAYS_INLINE void finish_bit(void)
- {
- if (!started)
- return;
- if (bytes) {
- if (--bits == 0) {
- uint8_t outbyte = data >> 8;
- fmrx_recv_byte(outbyte);
- data = 0;
- bits = 8;
- if (--bytes == 0)
- fmrx_led_off();
- }
- } else if (data == CAS_SYNC_PATTERN) {
- bits = 8;
- bytes = CAS_BLOCK_LEN;
- fmrx_led_on();
- }
- }
- ISR(TIMER1_CAPT_vect)
- {
- uint16_t edge, delta;
- TCCR1B ^= (1 << ICES1);
- edge = ICR1;
- delta = edge - last_edge;
- if (delta < one_bit_lo) {
-
- return;
- } else if (delta < one_bit_hi) {
-
- data |= (1 << 15);
- } else {
-
- last_edge = edge;
- enable_timeout(edge + one_bit_timeout);
- finish_bit();
- started = true;
- data >>= 1;
- }
- }
- ISR(TIMER1_COMPA_vect)
- {
- disable_timeout();
- finish_bit();
- started = false;
- }
|