mirror of
https://github.com/DarkFlippers/unleashed-firmware
synced 2024-12-29 22:23:06 +00:00
aa2ecbe80f
* infrared: add Kaseikyo IR protocol Add Kaseikyo IR protocol support. This protocol is also called the Panasonic protocol and is used by a number of manufacturers including Denon. The protocol includes a vendor field and a number of fields that are vendor specific. To support the format of address+command used by flipper the vendor+genre1+genre2+id fields are encoded into the address while the data is used for the command. There are older versions of the protocol that used a reverse bit order that are not supported. Protocol information: - https://github.com/Arduino-IRremote/Arduino-IRremote/blob/master/src/ir_Kaseikyo.hpp - http://www.hifi-remote.com/johnsfine/DecodeIR.html#Kaseikyo - https://www.denon.com/-/media/files/documentmaster/denonna/avr-x3700h_avc-x3700h_ir_code_v01_04062020.doc * Format and add unit test to Kaseikyo IR codec. Co-authored-by: Georgii Surkov <37121527+gsurkov@users.noreply.github.com>
320 lines
16 KiB
C
320 lines
16 KiB
C
#pragma once
|
|
|
|
#include <stddef.h>
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
#include "infrared.h"
|
|
#include "common/infrared_common_i.h"
|
|
|
|
/***************************************************************************************************
|
|
* NEC protocol description
|
|
* https://radioparty.ru/manuals/encyclopedia/213-ircontrol?start=1
|
|
****************************************************************************************************
|
|
* Preamble Preamble Pulse Distance/Width Pause Preamble Preamble Stop
|
|
* mark space Modulation up to period repeat repeat bit
|
|
* mark space
|
|
*
|
|
* 9000 4500 32 bit + stop bit ...110000 9000 2250
|
|
* __________ _ _ _ _ _ _ _ _ _ _ _ _ _ ___________ _
|
|
* ____ __________ _ _ _ __ __ __ _ _ __ __ _ _ ________________ ____________ ___
|
|
*
|
|
***************************************************************************************************/
|
|
|
|
#define INFRARED_NEC_PREAMBLE_MARK 9000
|
|
#define INFRARED_NEC_PREAMBLE_SPACE 4500
|
|
#define INFRARED_NEC_BIT1_MARK 560
|
|
#define INFRARED_NEC_BIT1_SPACE 1690
|
|
#define INFRARED_NEC_BIT0_MARK 560
|
|
#define INFRARED_NEC_BIT0_SPACE 560
|
|
#define INFRARED_NEC_REPEAT_PERIOD 110000
|
|
#define INFRARED_NEC_SILENCE INFRARED_NEC_REPEAT_PERIOD
|
|
#define INFRARED_NEC_MIN_SPLIT_TIME INFRARED_NEC_REPEAT_PAUSE_MIN
|
|
#define INFRARED_NEC_REPEAT_PAUSE_MIN 4000
|
|
#define INFRARED_NEC_REPEAT_PAUSE_MAX 150000
|
|
#define INFRARED_NEC_REPEAT_MARK 9000
|
|
#define INFRARED_NEC_REPEAT_SPACE 2250
|
|
#define INFRARED_NEC_PREAMBLE_TOLERANCE 200 // us
|
|
#define INFRARED_NEC_BIT_TOLERANCE 120 // us
|
|
|
|
void* infrared_decoder_nec_alloc(void);
|
|
void infrared_decoder_nec_reset(void* decoder);
|
|
void infrared_decoder_nec_free(void* decoder);
|
|
InfraredMessage* infrared_decoder_nec_check_ready(void* decoder);
|
|
InfraredMessage* infrared_decoder_nec_decode(void* decoder, bool level, uint32_t duration);
|
|
void* infrared_encoder_nec_alloc(void);
|
|
InfraredStatus infrared_encoder_nec_encode(void* encoder_ptr, uint32_t* duration, bool* level);
|
|
void infrared_encoder_nec_reset(void* encoder_ptr, const InfraredMessage* message);
|
|
void infrared_encoder_nec_free(void* encoder_ptr);
|
|
bool infrared_decoder_nec_interpret(InfraredCommonDecoder* decoder);
|
|
InfraredStatus infrared_decoder_nec_decode_repeat(InfraredCommonDecoder* decoder);
|
|
InfraredStatus infrared_encoder_nec_encode_repeat(
|
|
InfraredCommonEncoder* encoder,
|
|
uint32_t* duration,
|
|
bool* level);
|
|
const InfraredProtocolSpecification* infrared_nec_get_spec(InfraredProtocol protocol);
|
|
|
|
extern const InfraredCommonProtocolSpec protocol_nec;
|
|
|
|
/***************************************************************************************************
|
|
* SAMSUNG32 protocol description
|
|
* https://www.mikrocontroller.net/articles/IRMP_-_english#SAMSUNG
|
|
****************************************************************************************************
|
|
* Preamble Preamble Pulse Distance/Width Pause Preamble Preamble Bit1 Stop
|
|
* mark space Modulation repeat repeat bit
|
|
* mark space
|
|
*
|
|
* 4500 4500 32 bit + stop bit 40000/100000 4500 4500
|
|
* __________ _ _ _ _ _ _ _ _ _ _ _ ___________ _ _
|
|
* _ __________ __ _ __ __ __ _ _ __ __ _ ________________ ____________ ____ ___
|
|
*
|
|
***************************************************************************************************/
|
|
|
|
#define INFRARED_SAMSUNG_PREAMBLE_MARK 4500
|
|
#define INFRARED_SAMSUNG_PREAMBLE_SPACE 4500
|
|
#define INFRARED_SAMSUNG_BIT1_MARK 550
|
|
#define INFRARED_SAMSUNG_BIT1_SPACE 1650
|
|
#define INFRARED_SAMSUNG_BIT0_MARK 550
|
|
#define INFRARED_SAMSUNG_BIT0_SPACE 550
|
|
#define INFRARED_SAMSUNG_REPEAT_PAUSE_MIN 30000
|
|
#define INFRARED_SAMSUNG_REPEAT_PAUSE1 46000
|
|
#define INFRARED_SAMSUNG_REPEAT_PAUSE2 97000
|
|
/* Samsung silence have to be greater than REPEAT MAX
|
|
* otherwise there can be problems during unit tests parsing
|
|
* of some data. Real tolerances we don't know, but in real life
|
|
* silence time should be greater than max repeat time. This is
|
|
* because of similar preambule timings for repeat and first messages. */
|
|
#define INFRARED_SAMSUNG_MIN_SPLIT_TIME 5000
|
|
#define INFRARED_SAMSUNG_SILENCE 145000
|
|
#define INFRARED_SAMSUNG_REPEAT_PAUSE_MAX 140000
|
|
#define INFRARED_SAMSUNG_REPEAT_MARK 4500
|
|
#define INFRARED_SAMSUNG_REPEAT_SPACE 4500
|
|
#define INFRARED_SAMSUNG_PREAMBLE_TOLERANCE 200 // us
|
|
#define INFRARED_SAMSUNG_BIT_TOLERANCE 120 // us
|
|
|
|
void* infrared_decoder_samsung32_alloc(void);
|
|
void infrared_decoder_samsung32_reset(void* decoder);
|
|
void infrared_decoder_samsung32_free(void* decoder);
|
|
InfraredMessage* infrared_decoder_samsung32_check_ready(void* ctx);
|
|
InfraredMessage* infrared_decoder_samsung32_decode(void* decoder, bool level, uint32_t duration);
|
|
InfraredStatus
|
|
infrared_encoder_samsung32_encode(void* encoder_ptr, uint32_t* duration, bool* level);
|
|
void infrared_encoder_samsung32_reset(void* encoder_ptr, const InfraredMessage* message);
|
|
void* infrared_encoder_samsung32_alloc(void);
|
|
void infrared_encoder_samsung32_free(void* encoder_ptr);
|
|
bool infrared_decoder_samsung32_interpret(InfraredCommonDecoder* decoder);
|
|
InfraredStatus infrared_decoder_samsung32_decode_repeat(InfraredCommonDecoder* decoder);
|
|
InfraredStatus infrared_encoder_samsung32_encode_repeat(
|
|
InfraredCommonEncoder* encoder,
|
|
uint32_t* duration,
|
|
bool* level);
|
|
const InfraredProtocolSpecification* infrared_samsung32_get_spec(InfraredProtocol protocol);
|
|
|
|
extern const InfraredCommonProtocolSpec protocol_samsung32;
|
|
|
|
/***************************************************************************************************
|
|
* RC6 protocol description
|
|
* https://www.mikrocontroller.net/articles/IRMP_-_english#RC6_.2B_RC6A
|
|
****************************************************************************************************
|
|
* Preamble Manchester/biphase Silence
|
|
* mark/space Modulation
|
|
*
|
|
* 2666 889 444/888 - bit (x2 for toggle bit) 2666
|
|
*
|
|
* ________ __ __ __ __ ____ __ __ __ __ __ __ __ __
|
|
* _ _________ ____ __ __ ____ __ __ __ __ __ __ __ __ _______________
|
|
* | 1 | 0 | 0 | 0 | 0 | ... | ... | |
|
|
* s m2 m1 m0 T address (MSB) command (MSB)
|
|
*
|
|
* s - start bit (always 1)
|
|
* m0-2 - mode (000 for RC6)
|
|
* T - toggle bit, twice longer
|
|
* address - 8 bit
|
|
* command - 8 bit
|
|
***************************************************************************************************/
|
|
|
|
#define INFRARED_RC6_CARRIER_FREQUENCY 36000
|
|
#define INFRARED_RC6_DUTY_CYCLE 0.33
|
|
|
|
#define INFRARED_RC6_PREAMBLE_MARK 2666
|
|
#define INFRARED_RC6_PREAMBLE_SPACE 889
|
|
#define INFRARED_RC6_BIT 444 // half of time-quant for 1 bit
|
|
#define INFRARED_RC6_PREAMBLE_TOLERANCE 200 // us
|
|
#define INFRARED_RC6_BIT_TOLERANCE 120 // us
|
|
/* protocol allows 2700 silence, but it is hard to send 1 message without repeat */
|
|
#define INFRARED_RC6_SILENCE (2700 * 10)
|
|
#define INFRARED_RC6_MIN_SPLIT_TIME 2700
|
|
|
|
void* infrared_decoder_rc6_alloc(void);
|
|
void infrared_decoder_rc6_reset(void* decoder);
|
|
void infrared_decoder_rc6_free(void* decoder);
|
|
InfraredMessage* infrared_decoder_rc6_check_ready(void* ctx);
|
|
InfraredMessage* infrared_decoder_rc6_decode(void* decoder, bool level, uint32_t duration);
|
|
void* infrared_encoder_rc6_alloc(void);
|
|
void infrared_encoder_rc6_reset(void* encoder_ptr, const InfraredMessage* message);
|
|
void infrared_encoder_rc6_free(void* decoder);
|
|
InfraredStatus infrared_encoder_rc6_encode(void* encoder_ptr, uint32_t* duration, bool* polarity);
|
|
bool infrared_decoder_rc6_interpret(InfraredCommonDecoder* decoder);
|
|
InfraredStatus infrared_decoder_rc6_decode_manchester(
|
|
InfraredCommonDecoder* decoder,
|
|
bool level,
|
|
uint32_t timing);
|
|
InfraredStatus infrared_encoder_rc6_encode_manchester(
|
|
InfraredCommonEncoder* encoder_ptr,
|
|
uint32_t* duration,
|
|
bool* polarity);
|
|
const InfraredProtocolSpecification* infrared_rc6_get_spec(InfraredProtocol protocol);
|
|
|
|
extern const InfraredCommonProtocolSpec protocol_rc6;
|
|
|
|
/***************************************************************************************************
|
|
* RC5 protocol description
|
|
* https://www.mikrocontroller.net/articles/IRMP_-_english#RC5_.2B_RC5X
|
|
****************************************************************************************************
|
|
* Manchester/biphase
|
|
* Modulation
|
|
*
|
|
* 888/1776 - bit (x2 for toggle bit)
|
|
*
|
|
* __ ____ __ __ __ __ __ __ __ __
|
|
* __ __ ____ __ __ __ __ __ __ __ _
|
|
* | 1 | 1 | 0 | ... | ... |
|
|
* s si T address (MSB) command (MSB)
|
|
*
|
|
* Note: manchester starts from space timing, so it have to be handled properly
|
|
* s - start bit (always 1)
|
|
* si - RC5: start bit (always 1), RC5X - 7-th bit of address (in our case always 0)
|
|
* T - toggle bit, change it's value every button press
|
|
* address - 5 bit
|
|
* command - 6/7 bit
|
|
***************************************************************************************************/
|
|
|
|
#define INFRARED_RC5_CARRIER_FREQUENCY 36000
|
|
#define INFRARED_RC5_DUTY_CYCLE 0.33
|
|
|
|
#define INFRARED_RC5_PREAMBLE_MARK 0
|
|
#define INFRARED_RC5_PREAMBLE_SPACE 0
|
|
#define INFRARED_RC5_BIT 888 // half of time-quant for 1 bit
|
|
#define INFRARED_RC5_PREAMBLE_TOLERANCE 200 // us
|
|
#define INFRARED_RC5_BIT_TOLERANCE 120 // us
|
|
/* protocol allows 2700 silence, but it is hard to send 1 message without repeat */
|
|
#define INFRARED_RC5_SILENCE (2700 * 10)
|
|
#define INFRARED_RC5_MIN_SPLIT_TIME 2700
|
|
|
|
void* infrared_decoder_rc5_alloc(void);
|
|
void infrared_decoder_rc5_reset(void* decoder);
|
|
void infrared_decoder_rc5_free(void* decoder);
|
|
InfraredMessage* infrared_decoder_rc5_check_ready(void* ctx);
|
|
InfraredMessage* infrared_decoder_rc5_decode(void* decoder, bool level, uint32_t duration);
|
|
void* infrared_encoder_rc5_alloc(void);
|
|
void infrared_encoder_rc5_reset(void* encoder_ptr, const InfraredMessage* message);
|
|
void infrared_encoder_rc5_free(void* decoder);
|
|
InfraredStatus infrared_encoder_rc5_encode(void* encoder_ptr, uint32_t* duration, bool* polarity);
|
|
bool infrared_decoder_rc5_interpret(InfraredCommonDecoder* decoder);
|
|
const InfraredProtocolSpecification* infrared_rc5_get_spec(InfraredProtocol protocol);
|
|
|
|
extern const InfraredCommonProtocolSpec protocol_rc5;
|
|
|
|
/***************************************************************************************************
|
|
* Sony SIRC protocol description
|
|
* https://www.sbprojects.net/knowledge/ir/sirc.php
|
|
* http://picprojects.org.uk/
|
|
****************************************************************************************************
|
|
* Preamble Preamble Pulse Width Modulation Pause Entirely repeat
|
|
* mark space up to period message..
|
|
*
|
|
* 2400 600 12/15/20 bits (600,1200) ...45000 2400 600
|
|
* __________ _ _ _ _ _ _ _ _ _ _ _ _ _ __________ _ _
|
|
* ____ __________ _ _ _ __ __ __ _ _ __ __ _ _ ____________________ __________ _
|
|
* | command | address |
|
|
* SIRC | 7b LSB | 5b LSB |
|
|
* SIRC15 | 7b LSB | 8b LSB |
|
|
* SIRC20 | 7b LSB | 13b LSB |
|
|
*
|
|
* No way to determine either next message is repeat or not,
|
|
* so recognize only fact message received. Sony remotes always send at least 3 messages.
|
|
* Assume 8 last extended bits for SIRC20 are address bits.
|
|
***************************************************************************************************/
|
|
|
|
#define INFRARED_SIRC_CARRIER_FREQUENCY 40000
|
|
#define INFRARED_SIRC_DUTY_CYCLE 0.33
|
|
#define INFRARED_SIRC_PREAMBLE_MARK 2400
|
|
#define INFRARED_SIRC_PREAMBLE_SPACE 600
|
|
#define INFRARED_SIRC_BIT1_MARK 1200
|
|
#define INFRARED_SIRC_BIT1_SPACE 600
|
|
#define INFRARED_SIRC_BIT0_MARK 600
|
|
#define INFRARED_SIRC_BIT0_SPACE 600
|
|
#define INFRARED_SIRC_PREAMBLE_TOLERANCE 200 // us
|
|
#define INFRARED_SIRC_BIT_TOLERANCE 120 // us
|
|
#define INFRARED_SIRC_SILENCE 10000
|
|
#define INFRARED_SIRC_MIN_SPLIT_TIME (INFRARED_SIRC_SILENCE - 1000)
|
|
#define INFRARED_SIRC_REPEAT_PERIOD 45000
|
|
|
|
void* infrared_decoder_sirc_alloc(void);
|
|
void infrared_decoder_sirc_reset(void* decoder);
|
|
InfraredMessage* infrared_decoder_sirc_check_ready(void* decoder);
|
|
uint32_t infrared_decoder_sirc_get_timeout(void* decoder);
|
|
void infrared_decoder_sirc_free(void* decoder);
|
|
InfraredMessage* infrared_decoder_sirc_decode(void* decoder, bool level, uint32_t duration);
|
|
void* infrared_encoder_sirc_alloc(void);
|
|
void infrared_encoder_sirc_reset(void* encoder_ptr, const InfraredMessage* message);
|
|
void infrared_encoder_sirc_free(void* decoder);
|
|
InfraredStatus infrared_encoder_sirc_encode(void* encoder_ptr, uint32_t* duration, bool* polarity);
|
|
bool infrared_decoder_sirc_interpret(InfraredCommonDecoder* decoder);
|
|
const InfraredProtocolSpecification* infrared_sirc_get_spec(InfraredProtocol protocol);
|
|
InfraredStatus infrared_encoder_sirc_encode_repeat(
|
|
InfraredCommonEncoder* encoder,
|
|
uint32_t* duration,
|
|
bool* level);
|
|
|
|
extern const InfraredCommonProtocolSpec protocol_sirc;
|
|
|
|
/***************************************************************************************************
|
|
* Kaseikyo protocol description
|
|
* https://github.com/Arduino-IRremote/Arduino-IRremote/blob/master/src/ir_Kaseikyo.hpp
|
|
****************************************************************************************************
|
|
* Preamble Preamble Pulse Distance/Width Pause Preamble Preamble
|
|
* mark space Modulation up to period repeat repeat
|
|
* mark space
|
|
*
|
|
* 3360 1665 48 bit ...130000 3456 1728
|
|
* __________ _ _ _ _ _ _ _ _ _ _ _ _ _ ___________
|
|
* ____ __________ _ _ _ __ __ __ _ _ __ __ _ _ ________________ ___________
|
|
*
|
|
***************************************************************************************************/
|
|
|
|
#define INFRARED_KASEIKYO_UNIT 432
|
|
#define INFRARED_KASEIKYO_PREAMBLE_MARK (8 * INFRARED_KASEIKYO_UNIT)
|
|
#define INFRARED_KASEIKYO_PREAMBLE_SPACE (4 * INFRARED_KASEIKYO_UNIT)
|
|
#define INFRARED_KASEIKYO_BIT1_MARK INFRARED_KASEIKYO_UNIT
|
|
#define INFRARED_KASEIKYO_BIT1_SPACE (3 * INFRARED_KASEIKYO_UNIT)
|
|
#define INFRARED_KASEIKYO_BIT0_MARK INFRARED_KASEIKYO_UNIT
|
|
#define INFRARED_KASEIKYO_BIT0_SPACE INFRARED_KASEIKYO_UNIT
|
|
#define INFRARED_KASEIKYO_REPEAT_PERIOD 130000
|
|
#define INFRARED_KASEIKYO_SILENCE INFRARED_KASEIKYO_REPEAT_PERIOD
|
|
#define INFRARED_KASEIKYO_MIN_SPLIT_TIME INFRARED_KASEIKYO_REPEAT_PAUSE_MIN
|
|
#define INFRARED_KASEIKYO_REPEAT_PAUSE_MIN 4000
|
|
#define INFRARED_KASEIKYO_REPEAT_PAUSE_MAX 150000
|
|
#define INFRARED_KASEIKYO_REPEAT_MARK INFRARED_KASEIKYO_PREAMBLE_MARK
|
|
#define INFRARED_KASEIKYO_REPEAT_SPACE (INFRARED_KASEIKYO_REPEAT_PERIOD - 56000)
|
|
#define INFRARED_KASEIKYO_PREAMBLE_TOLERANCE 200 // us
|
|
#define INFRARED_KASEIKYO_BIT_TOLERANCE 120 // us
|
|
|
|
void* infrared_decoder_kaseikyo_alloc(void);
|
|
void infrared_decoder_kaseikyo_reset(void* decoder);
|
|
void infrared_decoder_kaseikyo_free(void* decoder);
|
|
InfraredMessage* infrared_decoder_kaseikyo_check_ready(void* decoder);
|
|
InfraredMessage* infrared_decoder_kaseikyo_decode(void* decoder, bool level, uint32_t duration);
|
|
void* infrared_encoder_kaseikyo_alloc(void);
|
|
InfraredStatus
|
|
infrared_encoder_kaseikyo_encode(void* encoder_ptr, uint32_t* duration, bool* level);
|
|
void infrared_encoder_kaseikyo_reset(void* encoder_ptr, const InfraredMessage* message);
|
|
void infrared_encoder_kaseikyo_free(void* encoder_ptr);
|
|
bool infrared_decoder_kaseikyo_interpret(InfraredCommonDecoder* decoder);
|
|
InfraredStatus infrared_decoder_kaseikyo_decode_repeat(InfraredCommonDecoder* decoder);
|
|
InfraredStatus infrared_encoder_kaseikyo_encode_repeat(
|
|
InfraredCommonEncoder* encoder,
|
|
uint32_t* duration,
|
|
bool* level);
|
|
const InfraredProtocolSpecification* infrared_kaseikyo_get_spec(InfraredProtocol protocol);
|
|
|
|
extern const InfraredCommonProtocolSpec protocol_kaseikyo;
|