Skip to content

IRMP-org/IRMP

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

IRMP - Infrared Multi Protocol Decoder + Encoder

Available as Arduino library "IRMP"

Version 3.5.2 - work in progress

License: GPL v3 Installation instructions Commits since latest Build Status Hit Counter

50 IR protocols supported and low memory footprint

  • Only 39 protocols can be enabled for receive at the same time, because some of them are quite similar and conflicts with each other.
  • 39 protocols are available for send.
  • FLASH usage in bytes: 1500 for one protocol, 4300 for 15 main and 8000 for all 39 protocols.
  • RAM usage in bytes: 52, 73 and 100.
Nano running AllProtocol example YouTube Video Instructable
Nano running AllProtocol example YouTube Video Instructable

List of protocols

  • Sony SIRCS, NEC + APPLE + ONKYO, Samsung + Samsg32, Kaseikyo
  • JVC, NEC16, NEC42, Matsushita, DENON, Sharp, RC5, RC6 & RC6A, IR60 (SDA2008) Grundig, Siemens Gigaset, Nokia
  • BOSE, Kathrein, NUBERT, FAN (ventilator), SPEAKER (~NUBERT), Bang & Olufsen, RECS80 (SAA3004), RECS80EXT (SAA3008), Thomson, NIKON camera, Netbox keyboard, ORTEK (Hama), Telefunken 1560, FDC3402 keyboard, RC Car, iRobot Roomba, RUWIDO, T-Home, A1 TV BOX, LEGO Power RC, RCMM 12,24, or 32, LG Air Condition, Samsung48, Merlin, Pentax, S100, ACP24, TECHNICS, PANASONIC Beamer, Mitsubishi Aircond, VINCENT, SAMSUNG AH, GREE CLIMATE, RCII T+A, RADIO e.g. TEVION, METZ
  • NEC, Kaseiko, Denon, RC6, Samsung + Samsg32 were successfully tested in interrupt mode, but there are many protocols which in principle cannot be decoded in this mode.

Features

  • You may use every pin for input or output.
  • Interrupt mode for major protocols.
  • Callback after successful receive of a command.
  • Support for inverted feedback LED (for send and receive feedback).
  • Support for inverted IR output for LED connected to VCC.
  • Unmodulated IR signal output enables direct replacment of an IR receiver circuit.
  • Compatible with Arduino tone library.
  • Send can also also wait for trailing space/gap.

Restrictions

  • Send IR frequency is fixed at 38 kHz.

Minimal version

For applications only requiring NEC protocol, there is a receiver which has very small codesize of 500 bytes and does NOT require any timer. See the MinimalReceiver and IRDispatcherDemo example how to use it. Mapping of pins to interrupts can be found here.

Schematic for Arduino UNO

The VS1838B is used as receiver for all examples and tests. This module has a 120 µs on/low and a 100 µs off/high delay between received signal and output. So it shortens the mark and extends the space by 20 µs.

IR-Receiver connection Serial LCD connection
Fritzing schematic for Arduino UNO Fritzing schematic for Arduino UNO + LCD

Supported Arduino architectures / CPU's / boards

Digispark boards are tested with the recommended ATTinyCore using New Style pin mapping for the pro board.

Architecture CPU Board
avr ATmega16, ATmega328P, ATmega32U4, ATtinyX5, ATtinyX7 Uno, Nano, Leonardo, Sparkfun Pro Micro, Digispark etc.
megaavr ATmega4809 Uno WiFi Rev 2, Nano Every
samd SAMD21G18A Zero, MKR*, etc. but not DUE, which is sam architecture
esp8266 % all
esp32 % all
stm32 STM32F1xx BluePill
STM32F1 STM32F1xx BluePill
apollo3 Ambiq Apollo3 Sparkfun Apollo3 + Artemis
mbed nRF528x Nano 33 BLE
Teensiduino all - but limited support >= Teensy 3

Quick comparison of 4 Arduino IR receiving libraries

This is a short comparison and may not be complete or correct

I created this comparison matrix for myself in order to choose a small IR lib for my project and to have a quick overview, when to choose which library.
It is dated from 03.02.2021. If you have complains about the data or request for extensions, please send a PM or open a discussion.

Subject IRMP IRLremote IRLib2
mostly unmaintained
IRremote Minimal NEC
Number of protocols 50 Nec + Panasonic + Hash * 12 + Hash * 17 + Hash * NEC
3.Party libs needed % PinChangeInterrupt if not pin 2 or 3 % % %
Timing method receive Timer2 or interrupt for pin 2 or 3 Interrupt Timer2 or interrupt for pin 2 or 3 Timer2 or interrupt for NEC Interrupt
Timing method send PWM and timing with Timer2 interrupts Timer2 interrupts Timer2 and blocking wait PWM with Timer2 and blocking wait with delayMicroseconds() %
Send pins All All All ? Timer dependent %
Decode method OnTheFly OnTheFly RAM RAM OnTheFly
Encode method OnTheFly OnTheFly OnTheFly OnTheFly or RAM %
Callback suppport x % % % x
Repeat handling Receive + Send (partially) % ? Receive + Send x
LED feedback x % x x x
FLASH usage (simple NEC example with 5 prints) 1820
(4300 for 15 main / 8000 for all 40 protocols)
(+200 for callback)
(+80 for interrupt at pin 2+3)
1270
(1400 for pin 2+3)
4830 1770 900
RAM usage 52
(73 / 100 for 15 (main) / 40 protocols)
62 334 227 19
Supported platforms avr, megaAVR, attiny, Digispark (Pro), esp8266, ESP32, STM32, SAMD 21, Apollo3
(plus arm and pic for non Arduino IDE)
avr, esp8266 avr, SAMD 21, SAMD 51 avr, attiny, esp8266, esp32, SAM, SAMD All platforms with attachInterrupt()
Last library update 2/2021 4/2018 9/2019 10/2021 10/2021
Remarks Decodes 40 protocols concurrently.
39 Protocols to send.
Only one protocol at a time. Consists of 5 libraries. Project containing bugs - 45 issues, no reaction for at least one year. Decoding and sending are easy to extend.
Supports Pronto codes.
Work in progress.
Requires no timer.

* The Hash protocol gives you a hash as code, which may be sufficient to distinguish your keys on the remote, but may not work with some protocols like Mitsubishi

Pin usage

You may use every pin for input or output, just define it like #define IRMP_INPUT_PIN 2 and #define IRSND_OUTPUT_PIN 3. The PWM of the output pin is generated by software bit banging.

Dynamic pins numbers

If you want to use pin numbers for input, output and LED feedback specified at runtime, you must define IRMP_IRSND_ALLOW_DYNAMIC_PINS. See ReceiveAndSendDynamicPins example.
The irmp_init and irsnd_init function then allows up to 3 parameters uint_fast8_t aIrmpInputPin/aIrsndOutputPin, uint_fast8_t aIrmpFeedbackLedPin, bool aIrmpLedFeedbackPinIsActiveLow.
Be aware, only one pin and enable flag for receive and send feedback LED is supported.

Receiving and sending simultaneously

Receiving and sending is possible with this library, but since we use only 1 timer, receiving is inhibited while sending the IR signal.
Sending the IR signal starts with saving current timer configuration, setting the timer to the send configuration / frequency, sending the signal (and waiting for the gap after the signal) and then automatically reset the timer to its previous (receiving) configuration.

API

IRMP

// Init functions
void irmp_init (void);
void irmp_init(uint_fast8_t aIrmpInputPin);
void irmp_init(uint_fast8_t aIrmpInputPin, uint_fast8_t aIrmpFeedbackLedPin);
void irmp_init(uint_fast8_t aIrmpInputPin, uint_fast8_t aIrmpFeedbackLedPin, bool aIrmpLedFeedbackPinIsActiveLow);
void irmp_register_complete_callback_function(void (*aCompleteCallbackFunction)(void));

// Info function
bool irmp_IsBusy();
void irmp_print_active_protocols(Print *aSerial);

// Main check for result function used in loop()
bool irmp_get_data (IRMP_DATA *)

// Result print functions
void irmp_result_print(Print *aSerial, IRMP_DATA * aIRMPDataPtr);
void irmp_result_print(IRMP_DATA *aIRMPDataPtr);

IRSND

// Init functions
void irsnd_init (void);
// 3 additional init functions if IRMP_IRSND_ALLOW_DYNAMIC_PINS is defined
void irsnd_init(uint_fast8_t aIrsndOutputPin);
void irsnd_init(uint_fast8_t aIrsndOutputPin, uint_fast8_t aIrmpFeedbackLedPin);
void irsnd_init(uint_fast8_t aIrsndOutputPin, uint_fast8_t aIrmpFeedbackLedPin, bool aIrmpLedFeedbackPinIsActiveLow);

// Send function - sends frame AND trailing space
bool irsnd_send_data (IRMP_DATA *, uint8_t);

// Info functions
bool irsnd_is_busy (void);

void irsnd_stop (void);

IRMP and IRSND

// LED feedback function
void irmp_irsnd_LEDFeedback(bool aEnableBlinkLed);

// Timer management functions for 
void disableIRTimerInterrupt(void);
void enableIRTimerInterrupt(void);
void storeIRTimer(void);
void restoreIRTimer(void);

Examples

In order to fit the examples to the 8K flash of ATtiny85 and ATtiny88, the Arduino library ATtinySerialOut is required for this CPU's.

SimpleReceiver + SimpleSender

This examples are a good starting point.
SimpleReceiver can be tested online with WOKWI. Click on the receiver while simulation is running to specify individual IR codes.

MinimalReceiver + SmallReceiver

If code size matters, look at these examples.
MinimalReceiver can be tested online with WOKWI. Click on the receiver while simulation is running to specify individual IR codes.

ReceiverTimingAnalysis

This example analyzes the signal delivered by your IR receiver module. Values can be used to determine the stability of the received signal as well as a hint for determining the protocol.
It also computes the MARK_EXCESS_MICROS value, which is the extension of the mark (pulse) duration introduced by the IR receiver module.
It can be tested online with WOKWI. Click on the receiver while simulation is running to specify individual IR codes.

Compile options / macros for this library

To customize the library to different requirements, there are some compile options / macros available, which must be set before including the library e.g. with #include <irmp.hpp>.
Modify it by setting the value to 1 or 0. Or define the macro with the -D compiler option for global compile (the latter is not possible with the Arduino IDE, so consider using Sloeber.

Macro Enable value Description
IRMP_INPUT_PIN defined The pin number which gets compiled in, if not using IRMP_IRSND_ALLOW_DYNAMIC_PINS.
IRMP_FEEDBACK_LED_PIN defined The pin number for the feedback led which gets compiled in, if not using IRMP_IRSND_ALLOW_DYNAMIC_PINS.
If not defined, LED_BUILTIN ist taken.
FEEDBACK_LED_IS_ACTIVE_LOW defined Required on some boards (like my like my BluePill and my ESP8266 board), where the feedback LED is active low.
IRMP_IRSND_ALLOW_DYNAMIC_PINS defined Allows to specify pin number at irmp_init() -see above. This requires additional program space.
IRMP_PROTOCOL_NAMES 1 Enable protocol number mapping to protocol strings - needs some program memory.
IRMP_USE_COMPLETE_CALLBACK 1 Use Callback if complete data was received. Requires call to irmp_register_complete_callback_function().
IRMP_ENABLE_PIN_CHANGE_INTERRUPT defined Use Arduino attachInterrupt() and do no polling with timer ISR. This restricts the available input pins and protocols. The results are equivalent to results acquired with a sampling rate of 15625 Hz (chosen to avoid time consuming divisions). For AVR boards an own interrupt handler for INT0 or INT1 is used instead of Arduino attachInterrupt().
IRMP_ENABLE_RELEASE_DETECTION 1 If user releases a key on the remote control, last protocol/address/command will be returned with flag IRMP_FLAG_RELEASE set.
IRMP_HIGH_ACTIVE 1 Set to 1 if you use a RF receiver, which has an active HIGH output signal.
IRMP_32_BIT 1 This enables MERLIN protocol, but decreases performance for AVR.
F_INTERRUPTS The IRMP sampling frequency. Default value is 15000.
USE_ONE_TIMER_FOR_IRMP_AND_IRSND defined Must be defined if you use receiving and sending simultaneously, since both use the same timer resource.
IRSND_USE_CALLBACK 1 Calls a function if send output signal changes to active (sending the IR signal).
IR_OUTPUT_IS_ACTIVE_LOW defined Output LOW for active IR signal. Use it if IR LED is connected between VCC and output pin.
IRSND_GENERATE_NO_SEND_RF defined Do not generate the carrier frequency (of 38 kHz), just simulate an active low receiver signal.
IRSND_IR_FREQUENCY The modulation frequency for sent signal. Default value is 38000. The send signal is generated by bit banging, so the internal interrupt frequency is IRSND_IR_FREQUENCY * 2. The send control function is called at a rate of IRSND_IR_FREQUENCY / 2, resulting in a send packet signal resolution of 2 on/off periods.
IR_INPUT_PIN defined The pin number for TinyIRReceiver IR input which gets compiled in.
IR_FEEDBACK_LED_PIN defined The pin number for TinyIRReceiver feedback LED which gets compiled in.
DO_NOT_USE_FEEDBACK_LED defined Enable it to disable the feedback LED function.
- - -
IRMP_MEASURE_TIMING + IR_TIMING_TEST_PIN defined For development only. The test pin is switched high at the very beginning and low at the end of the ISR.

Modifying compile options with Arduino IDE

First, use Sketch > Show Sketch Folder (Ctrl+K).
If you did not yet stored the example as your own sketch, then you are instantly in the right library folder.
Otherwise you have to navigate to the parallel libraries folder and select the library you want to access.
In both cases the library files itself are located in the src directory.

Modifying compile options with Sloeber IDE

If you are using Sloeber as your IDE, you can easily define global symbols with Properties > Arduino > CompileOptions.
Sloeber settings

The IRMP receive library works by polling the input pin at a rate of 10 to 20 kHz. Default is 15 kHz.
Some protocols (NEC, Kaseiko, Denon, RC6, Samsung + Samsg32) can be received without timer usage, just by using interrupts from the input pin by defining IRMP_ENABLE_PIN_CHANGE_INTERRUPT. There are many protocols which in principle cannot be decoded in this mode. See Interrupt example.
In interrupt mode, the micros() function is used as timebase.

The IRMP send library works by bit banging the output pin at a frequency of 38 kHz. This avoids blocking waits and allows to choose an arbitrary pin, you are not restricted to PWM generating pins like pin 3 or 11. The interrupts for send pin bit banging require 50% CPU time on a 16 MHz AVR.
If both receiving and sending is required, the timer is set up for receiving and reconfigured for the duration of sending data, thus preventing receiving in polling mode while sending data.
The tone library (using timer 2) is still available. You can use it alternating with IR receive and send, see ReceiveAndSend example.

  • For AVR timer 2 (Tone timer) is used for receiving and sending. For variants, which have no timer 2 like ATtiny85 or ATtiny167, timer 1 (or timer 0 for digispark core) is used.
  • For SAMD TC3 is used.
  • For Apollo3 Timer 3 segment B is used.
  • For ESP8266 and ESP32 timer1 is used.
  • For STM32 (BluePill) timer 3 (Servo timer) channel 1 is used as default.

Tips and tricks

  • To port the library to another device, you merely have to extend IRTimer.hpp.
  • The minimal CPU clock required for receiving is 8MHz.
  • To save power, you can use the interrupt mode or polling mode with no-sending detection and power down sleep. This is not available for ATtiny85 running with the High Speed PLL clock (as on Digispark boards) because of the high startup time from sleep of 4 to 5 ms for this clock. You have to use the ISP to rewrite the CKSEL fuses and to load the program.
  • The best way to increase the IR power is to use 2 or 3 IR diodes in series. One diode requires 1.1 to 1.5 volt so you can supply 3 diodes with a 5 volt output.To keep the current, you must reduce the resistor by (5 - 1.3) / (5 - 2.6) = 1.5 e.g. from 150 ohm to 100 ohm for 25 mA and 2 diodes with 1.3 volt and a 5 volt supply. For 3 diodes it requires factor 2.5 e.g. from 150 ohm to 60 ohm.
  • A lot of recent IR diodes can be powered with max. 200 mA at 50% duty cycle, but for this you will require an external driver / transistor / (mos)fet.
  • In order to fit the examples to the 8K flash of ATtiny85 and ATtiny88, the Arduino library ATtinySerialOut is required for this CPU's.

AllProtocol example

Serial LCD output Arduino Serial Monitor output
LCD start Serial Monitor

Sample Protocols

NEC NEC42 RC5 KASEIKYO
DENON GRUNDIG IR60 MATSUSHITA
NUBERT ONKYO RECS80 RUWIDO
SAMSUNG SIEMENS TELEFUNKEN TELEFUNKEN

Documentation at mikrocontroller.net

English

http://www.mikrocontroller.net/articles/IRMP_-_english
http://www.mikrocontroller.net/articles/IRSND_-_english

German

http://www.mikrocontroller.net/articles/IRMP
http://www.mikrocontroller.net/articles/IRSND

Revision History

Version 3.5.1

  • Fixed ESP8266 F_CPU error introduced with 3.4.1.

Version 3.5.0

  • Renamed *.c.h and *.cpp.h to .hpp. You must change: #include <irmp.c.h> to: #include <irmp.hpp>!
  • Fix Timer1 initialization for ATtinyX7 parts for ATTinyCore.
  • Modifying digitalWriteFast.h to be compatible with ATTinyCore Digispark Pro default pin mapping.
  • Renamed initPCIInterrupt() to enablePCIInterrupt() and added disablePCIInterrupt().
  • Changed return value for irsnd_send_data() to be false on error conditions.
  • Fixed ICACHE_RAM_ATTR error introduced with 3.4.1.

Version 3.4.1

  • Changed default pins for ATmega328 platform from 3,4,5 to 2,3,4.
  • Adapted to TinyCore 0.0.7.
  • Renamed macro IRMP_TIMING_TEST_PIN to IR_TIMING_TEST_PIN.
  • Changed pins in PinDefinitionsAndMore.h.
  • Never send a trailing space for Arduino.
  • ATTiny88 support.

Version 3.4.0

  • Added ATtiny3217 / TinyCore support.
  • Added Melinera protocol and single repeat for NEC from upstream.

Version 3.3.5

  • Added TinyIRReceiver and updated IRDispatcherDemo examples.
  • Fixed "enabling OUTPUT for dynamic pin" bug.
  • Improved Apollo3 and MegaCore support.

Version 3.3.4

  • Removed convenience function irmp_tone(), since it statically allocates the tone interrupt vector.

Version 3.3.3

  • Added ATmega8 support.
  • Added IRSND_GENERATE_NO_SEND_RF compile switch.
  • Added function irsnd_data_print().
  • New SendAllProtocols example.
  • New DispatcherDemo example.
  • Added IRMP_FEEDBACK_LED_PIN compile switch.
  • Removed IRMP16 protocol from the all list.
  • Added missing Leonardo support.

Version 3.3.2

  • Added missing Medion entry in irmp_protocol_names.
  • Added function irmp_print_protocol_name().
  • Added Teensyduino support.
  • Fixed macro redefinitions in IRSND.

Version 3.3.1

  • Fix for function bool irmp_IsBusy() if IRMP_ENABLE_PIN_CHANGE_INTERRUPT is defined.

Version 3.3.0

  • Added function bool irmp_IsBusy().

Version 3.2.3

  • Fixed warning for missing USE_ONE_TIMER_FOR_IRMP_AND_IRSND macro.

Version 3.2.2

  • Removed blocking wait for ATmega32U4 Serial in examples.
  • Restored missing line reset interrupt flags found by user yumkam.
  • Fixed bug for sending only on no AVR platforms.

Version 3.2.1

  • Fixed bug in feedback LED handling for dynamic pins for send and receive.
  • Fixed wrong timer selection for STM32F1xx / ARDUINO_ARCH_STM32.

Version 3.2.0

  • MBED support for Arduino Nano 33 BLE.
  • Added ARDUINO_ARCH_STM32 definition.
  • Fixed ESP8266 wrong memcpy_p definition introduced in 3.0.0.

Version 3.1.2

  • Fixed interrupt mode bug introduced by merging upstream code for version 3.1.0.
  • Fixed ESP8266 wrong memcpy_p definition introduced in 3.0.0. - Fix was not complete :-(. Use 3.2.0.

Version 3.1.1

  • Added MinimalReceiver example.

Version 3.1.0

  • Added RF_MEDION protocol
  • IRAM attribute etc.
  • Fixed bug irmp_init used instead of irsnd_init
  • New function irmp_print_active_protocols().
  • Use timer3 for ESP32.
  • Fix missing check for LED pin == 0.
  • Merged upstream v3.2.2 + IRMP_ENABLE_RELEASE_DETECTION.
  • Dynamic pin for feedback LED added.

Version 3.0.0

  • Support of RF (433MHz) remotes. 2 protocols Generic 24 bit format and X10 format added.
  • MegaAVR (ATmega4809) support.
  • Added IRMP_IRSND_ALLOW_DYNAMIC_PINS and extended irmp_init() and irsnd_init()to allow input, output and LED feedback pin selection at runtime.
  • Support more protocols simultaneously on 32 bit CPUs.
  • Use same pin and enable flag for receive and send feedback LED.
  • New function irmp_print_active_protocols().

Version 2.2.1

  • Improved pin layout.
  • Fixed bug with stm32duino 1.9.
  • Version number.
  • Blink13 -> LEDFeedback.

Version 2.2.0

  • Supported Apollo3 platform.
  • Fixed DigisparkPro bug.

Version 2.1.0

  • Supported SAMD platform.
  • IRSND enabled for non AVR platforms.

Version 2.0.0

  • Added IR send fuctionality (IRSND).
  • Use TIMER2_COMPB_vect to be compatible with tone library.
  • No longer required to call initPCIInterrupt() manually if IRMP_ENABLE_PIN_CHANGE_INTERRUPT is set.
  • Separated code for timer to IRTimer.hpp.
  • Separated code for Pin change interrupt to irmpPinChangeInterrupt.hpp.
  • Fixed wrong pin numbers for BluePill.

Version 1.2.2

  • Fixed bugs introduced in 1.2.1.

Version 1.2.1

  • Bug for AVR architecture fixed.
  • ATtiny85 + ATtiny167 support for ATTinyCore and Digistump core.
  • Support for "Generic STM32F1 series" from "STM32 Boards (selected from submenu)" of Arduino Board manager.

Version 1.2.0 - This version contains a bug for the AVR architecture

  • Added STM32 M3 (BluePill) support.

Version 1.1.0

  • Added functions irmp_disable_timer_interrupt() and irmp_enable_timer_interrupt().
  • Added function irmp_result_print(Print *aSerial).
  • Improved examples.

Version 1.0.1

  • Added ESP8266 + ESP32 support.

CI

Since Travis CI is unreliable and slow (5 times slower), the library examples are now tested with GitHub Actions for the following boards:

  • arduino:avr:uno
  • arduino:avr:mega
  • arduino:samd:arduino_zero_edbg
  • digistump:avr:digispark-tiny16
  • digistump:avr:digispark-pro (ATtiny167)
  • ATTinyCore:avr:attinyx5:chip=85,clock=8internal
  • esp8266:esp8266:huzzah:eesz=4M3M,xtal=80
  • esp32:esp32:featheresp32:FlashFreq=80
  • STM32:stm32:GenF1:pnum=BLUEPILL_F103C8

If you find this library useful, please give it a star.