* App Lfrfid: init * HAL-resources: add external gpios * HAL-pwm: fix frequency calculation * App LFRFID: generic manchester decoder * App LFRFID: em-marine decoder * App iButton: fix dwt timing acquire * App LFRFID: rfid reader * App LFRFID: temporary read keys on read scene * App LFRFID: remove atomic bool init. * App LFRFID: add *.c to build * App LFRFID: unstable HID decoder * App LFRFID: HID-26 reading * HAL OS: disable sleep * App LFRFID: HID-26 reader: remove debug * App LFRFID: static data decoder-analyzer * App LFRFID: very raw Indala decoder * App LFRFID: multiprotocol reader * App LFRFID: more reliable HID decoder * App LFRFID: syntax fix * App LFRFID: simple read scene * Gui: force redraw on screen stream connect * HAL-OS: allow sleep * App LFRFID: notify api, tune view, tune scene * App LFRFID: simple rfid emulator * App LFRFID: more scenes, more reliable EM decoder. * App LFRFID: format fix * App LFRFID: warning fix * Api-hal-resources: add rfid pins, rename external pins * App LFRFID: remove unused emulator * App LFRFID: use new gpio hal api * App accessor: use new ext gpio name * App LFRFID: remove unused emulator * App LFRFID: remove debug gpio * Api-hal-resources: alternate functions init * Api-hal-rfid: new api * Api-hal-ibutton: new api * Api-hal: new headers * App LFRFID: use new api in reader subroutines * App LFRFID: use new api in emulator subroutines * App LFRFID: remove old app * App LFRFID, App iButton: fix memleak * Api-hal-rfid: comments * App LFRFID: pulse joiner helper, it combines pulses of different polarity into one pulse suitable for a timer * App LFRFID: pulse joiner, now can accept only ne pulse * App LFRFID: pulse joiner, fixes * App LFRFID: EM encoder and emulation * App LFRFID: format fixes * App LFRFID: emmarine encoder cleanup * App LFRFID: HID Encoder blank * App LFRFID: Indala Encoder blank
		
			
				
	
	
		
			186 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			186 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
#include "decoder-hid26.h"
 | 
						|
#include <api-hal.h>
 | 
						|
 | 
						|
constexpr uint32_t clocks_in_us = 64;
 | 
						|
 | 
						|
constexpr uint32_t jitter_time_us = 20;
 | 
						|
constexpr uint32_t min_time_us = 64;
 | 
						|
constexpr uint32_t max_time_us = 80;
 | 
						|
 | 
						|
constexpr uint32_t min_time = (min_time_us - jitter_time_us) * clocks_in_us;
 | 
						|
constexpr uint32_t mid_time = ((max_time_us - min_time_us) / 2 + min_time_us) * clocks_in_us;
 | 
						|
constexpr uint32_t max_time = (max_time_us + jitter_time_us) * clocks_in_us;
 | 
						|
 | 
						|
bool DecoderHID26::read(uint8_t* data, uint8_t data_size) {
 | 
						|
    bool result = false;
 | 
						|
    furi_assert(data_size >= 3);
 | 
						|
 | 
						|
    if(ready) {
 | 
						|
        result = true;
 | 
						|
        data[0] = facility;
 | 
						|
        data[1] = (uint8_t)(number >> 8);
 | 
						|
        data[2] = (uint8_t)number;
 | 
						|
 | 
						|
        //printf("HID %02X %02X %02X\r\n", facility, (uint8_t)(number >> 8), (uint8_t)number);
 | 
						|
        ready = false;
 | 
						|
    }
 | 
						|
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
void DecoderHID26::process_front(bool polarity, uint32_t time) {
 | 
						|
    if(ready) return;
 | 
						|
 | 
						|
    if(polarity == true) {
 | 
						|
        last_pulse_time = time;
 | 
						|
    } else {
 | 
						|
        last_pulse_time += time;
 | 
						|
 | 
						|
        if(last_pulse_time > min_time && last_pulse_time < max_time) {
 | 
						|
            bool pulse;
 | 
						|
 | 
						|
            if(last_pulse_time < mid_time) {
 | 
						|
                // 6 pulses
 | 
						|
                pulse = false;
 | 
						|
            } else {
 | 
						|
                // 5 pulses
 | 
						|
                pulse = true;
 | 
						|
            }
 | 
						|
 | 
						|
            if(last_pulse == pulse) {
 | 
						|
                pulse_count++;
 | 
						|
 | 
						|
                if(pulse) {
 | 
						|
                    if(pulse_count > 4) {
 | 
						|
                        pulse_count = 0;
 | 
						|
                        store_data(1);
 | 
						|
                    }
 | 
						|
                } else {
 | 
						|
                    if(pulse_count > 5) {
 | 
						|
                        pulse_count = 0;
 | 
						|
                        store_data(0);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            } else {
 | 
						|
                if(last_pulse) {
 | 
						|
                    if(pulse_count > 2) {
 | 
						|
                        store_data(1);
 | 
						|
                    }
 | 
						|
                } else {
 | 
						|
                    if(pulse_count > 3) {
 | 
						|
                        store_data(0);
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                pulse_count = 0;
 | 
						|
                last_pulse = pulse;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
DecoderHID26::DecoderHID26() {
 | 
						|
    reset_state();
 | 
						|
}
 | 
						|
 | 
						|
void DecoderHID26::store_data(bool data) {
 | 
						|
    stored_data[0] = (stored_data[0] << 1) | ((stored_data[1] >> 31) & 1);
 | 
						|
    stored_data[1] = (stored_data[1] << 1) | ((stored_data[2] >> 31) & 1);
 | 
						|
    stored_data[2] = (stored_data[2] << 1) | data;
 | 
						|
    validate_stored_data();
 | 
						|
}
 | 
						|
 | 
						|
void DecoderHID26::validate_stored_data() {
 | 
						|
    // packet preamble
 | 
						|
    // raw data
 | 
						|
    if(*(reinterpret_cast<uint8_t*>(stored_data) + 3) != 0x1D) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    // encoded company/oem
 | 
						|
    // coded with 01 = 0, 10 = 1 transitions
 | 
						|
    // stored in word 0
 | 
						|
    if((*stored_data >> 10 & 0x3FFF) != 0x1556) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    // encoded format/length
 | 
						|
    // coded with 01 = 0, 10 = 1 transitions
 | 
						|
    // stored in word 0 and word 1
 | 
						|
    if((((*stored_data & 0x3FF) << 12) | ((*(stored_data + 1) >> 20) & 0xFFF)) != 0x155556) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    // data decoding
 | 
						|
    uint32_t result = 0;
 | 
						|
 | 
						|
    // decode from word 1
 | 
						|
    // coded with 01 = 0, 10 = 1 transitions
 | 
						|
    for(int8_t i = 9; i >= 0; i--) {
 | 
						|
        switch((*(stored_data + 1) >> (2 * i)) & 0b11) {
 | 
						|
        case 0b01:
 | 
						|
            result = (result << 1) | 0;
 | 
						|
            break;
 | 
						|
        case 0b10:
 | 
						|
            result = (result << 1) | 1;
 | 
						|
            break;
 | 
						|
        default:
 | 
						|
            return;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    // decode from word 2
 | 
						|
    // coded with 01 = 0, 10 = 1 transitions
 | 
						|
    for(int8_t i = 15; i >= 0; i--) {
 | 
						|
        switch((*(stored_data + 2) >> (2 * i)) & 0b11) {
 | 
						|
        case 0b01:
 | 
						|
            result = (result << 1) | 0;
 | 
						|
            break;
 | 
						|
        case 0b10:
 | 
						|
            result = (result << 1) | 1;
 | 
						|
            break;
 | 
						|
        default:
 | 
						|
            return;
 | 
						|
            break;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    // store decoded data
 | 
						|
    facility = result >> 17;
 | 
						|
    number = result >> 1;
 | 
						|
 | 
						|
    // trailing parity (odd) test
 | 
						|
    uint8_t parity_sum = 0;
 | 
						|
    for(int8_t i = 0; i < 13; i++) {
 | 
						|
        if(((result >> i) & 1) == 1) {
 | 
						|
            parity_sum++;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if((parity_sum % 2) != 1) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    // leading parity (even) test
 | 
						|
    parity_sum = 0;
 | 
						|
    for(int8_t i = 13; i < 26; i++) {
 | 
						|
        if(((result >> i) & 1) == 1) {
 | 
						|
            parity_sum++;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if((parity_sum % 2) == 1) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    ready = true;
 | 
						|
}
 | 
						|
 | 
						|
void DecoderHID26::reset_state() {
 | 
						|
    last_pulse = false;
 | 
						|
    pulse_count = 0;
 | 
						|
    ready = false;
 | 
						|
    last_pulse_time = 0;
 | 
						|
}
 |