* 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>
		
			
				
	
	
		
			321 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			321 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;
 |