forked from cyborg5/IRLib
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathIRLib.h
408 lines (368 loc) · 15.5 KB
/
IRLib.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
/* IRLib.h from IRLib – an Arduino library for infrared encoding and decoding
* Version 1.60 January 2016
* Copyright 2014-2016 by Chris Young http://cyborg5.com
* With additions by Gabriel Staples (www.ElectricRCAircraftGuy.com); see CHANGELOG.txt
*
* This library is a major rewrite of IRemote by Ken Shirriff which was covered by
* GNU LESSER GENERAL PUBLIC LICENSE which as I read it allows me to make modified versions.
* That same license applies to this modified version. See his original copyright below.
* The latest Ken Shirriff code can be found at https://github.com/shirriff/Arduino-IRremote
* My purpose was to reorganize the code to make it easier to add or remove protocols.
* As a result I have separated the act of receiving a set of raw timing codes from the act of decoding them
* by making them separate classes. That way the receiving aspect can be more black box and implementers
* of decoders and senders can just deal with the decoding of protocols. It also allows for alternative
* types of receivers independent of the decoding. This makes porting to different hardware platforms easier.
* Also added provisions to make the classes base classes that could be extended with new protocols
* which would not require recompiling of the original library nor understanding of its detailed contents.
* Some of the changes were made to reduce code size such as unnecessary use of long versus bool.
* Some changes were just my weird programming style. Also extended debugging information added.
*/
/*
* IRremote
* Version 0.1 July, 2009
* Copyright 2009 Ken Shirriff
* For details, see http://www.righto.com/2009/08/multi-protocol-infrared-remote-library.html http://www.righto.com/
*
* Interrupt code based on NECIRrcv by Joe Knapp
* http://www.arduino.cc/cgi-bin/yabb2/YaBB.pl?num=1210243556
* Also influenced by http://zovirl.com/2008/11/12/building-a-universal-remote-with-an-arduino/
*/
#ifndef IRLib_h
#define IRLib_h
#include <Arduino.h>
// The following are compile-time library options.
// If you change them, recompile the library.
// If IRLIB_TRACE is defined, some debugging information about the decode will be printed
// IRLIB_TEST must be defined for the IRtest unittests to work. It will make some
// methods virtual, which will be slightly slower, which is why it is optional.
//#define IRLIB_TRACE
// #define IRLIB_TEST
/* If not using the IRrecv class but only using IRrecvPCI or IRrecvLoop you can eliminate
* some conflicts with the duplicate definition of ISR by turning this feature off.
* Comment out the following define to eliminate the conflicts.
*/
#define USE_IRRECV
/* Similarly some other libraries have conflicts with the built in Arduino functions
* "attachInterrupt()" and "detachInterrupt()" which are used by the IRrecvPCI and
* IRfrequency classes. If you're not using either of those classes and get conflicts
* related to INT0_vect then comment out the following line to eliminate the conflicts.
*/
#define USE_ATTACH_INTERRUPTS
/* If not using either dumpResults methods of IRdecode nor IRfrequency you can
* comment out the following define to eliminate considerable program space.
*/
#define USE_DUMP
// Only used for testing; can remove virtual for shorter code
#ifdef IRLIB_TEST
#define VIRTUAL virtual
#else
#define VIRTUAL
#endif
#define RAWBUF 100 // Length of raw duration buffer (2-byte value, 0 to 65535); keep this define inside IRLib.h so the user can access it directly from their Arduino sketch
typedef uint8_t IR_types_t; //formerly was an enum
#define UNKNOWN 0
#define NEC 1
#define SONY 2
#define RC5 3
#define RC6 4
#define PANASONIC_OLD 5
#define JVC 6
#define NECX 7
//Add additional protocols here; keep HASH_CODE & LAST_PROTOCOL always at the very end
//#define ADDITIONAL (number); make additional protocol 8 and change HASH_CODE to 9
#define HASH_CODE 8
#define LAST_PROTOCOL HASH_CODE
/*
typedef enum
{
UNKNOWN = 0,
NEC,
SONY,
RC5,
RC6,
PANASONIC_OLD,
JVC,
NECX,
//Add additional protocols here; keep HASH_CODE & LAST_PROTOCOL always at the very end
HASH_CODE,
LAST_PROTOCOL = HASH_CODE
} IR_types_t;
*/
const __FlashStringHelper *Pnames(IR_types_t Type); //Returns a character string that is name of protocol.
// Base class for decoding raw results
class IRdecodeBase
{
public:
IRdecodeBase(void);
IR_types_t decode_type; // NEC, SONY, RC5, UNKNOWN etc.
unsigned long value; // Decoded value
unsigned char bits; // Number of bits in decoded value
volatile uint16_t *rawbuf; // Raw intervals in microseconds; GS: now ALWAYS points to irparams.rawbuf1; keep this variable, even though redundant with irparams.rawbuf1, for easy public access to the data
uint16_t rawlen; // Number of records in rawbuf; keep this variable, even though redundant with irparams.rawlen, for easy public access to the data
bool ignoreHeader; // Relaxed header detection allows AGC to settle
virtual void reset(void); // Initializes the decoder
virtual bool decode(void); // This base routine always returns false override with your routine
bool decodeGeneric(unsigned char Raw_Count, unsigned int Head_Mark, unsigned int Head_Space,
unsigned int Mark_One, unsigned int Mark_Zero, unsigned int Space_One, unsigned int Space_Zero);
virtual void dumpResults (void);
void useDoubleBuffer(volatile uint16_t *p_buffer); //use this to allow double-buffering; see extensive double-buffer notes in IRLibRData.h.
// void copyBuf (IRdecodeBase *source);//copies rawbuf and rawlen from one decoder to another; GS: REMOVED, NO LONGER NEEDED; double-buffers are done differently now
protected:
uint16_t offset; // Index into rawbuf used various places
};
class IRdecodeHash: public virtual IRdecodeBase
{
public:
unsigned long hash;
virtual bool decode(void);//made virtual in case you want to substitute your own hash code
protected:
int compare(unsigned int oldval, unsigned int newval);//used by decodeHash
};
class IRdecodeNEC: public virtual IRdecodeBase
{
public:
virtual bool decode(void);
};
class IRdecodeSony: public virtual IRdecodeBase
{
public:
virtual bool decode(void);
};
class IRdecodeRC: public virtual IRdecodeBase
{
public:
enum RCLevel {MARK, SPACE, ERROR};//used by decoders for RC5/RC6
// These are called by decode
RCLevel getRClevel(uint16_t *used, const unsigned int t1); //NB: datatype/size of *used must correspond to rawlen
protected:
uint8_t nbits;
uint16_t used; //NB: datatype/size of *used must correspond to rawlen
long data; //GS: why long instead of unsigned long? todo: find out if this really needs to be a signed value
};
class IRdecodeRC5: public virtual IRdecodeRC
{
public:
virtual bool decode(void);
};
class IRdecodeRC6: public virtual IRdecodeRC
{
public:
virtual bool decode(void);
};
class IRdecodePanasonic_Old: public virtual IRdecodeBase
{
public:
virtual bool decode(void);
};
class IRdecodeJVC: public virtual IRdecodeBase
{
public:
virtual bool decode(void);
};
class IRdecodeNECx: public virtual IRdecodeBase
{
public:
virtual bool decode(void);
};
// main class for decoding all supported protocols
class IRdecode:
public virtual IRdecodeNEC,
public virtual IRdecodeSony,
public virtual IRdecodeRC5,
public virtual IRdecodeRC6,
public virtual IRdecodePanasonic_Old,
public virtual IRdecodeJVC,
public virtual IRdecodeNECx
// , public virtual IRdecodeADDITIONAL //add additional protocols here
{
public:
virtual bool decode(void); // Calls each decode routine individually
};
//Base class for sending signals
class IRsendBase
{
public:
IRsendBase();
void sendGeneric(unsigned long data, unsigned char Num_Bits, unsigned int Head_Mark, unsigned int Head_Space,
unsigned int Mark_One, unsigned int Mark_Zero, unsigned int Space_One, unsigned int Space_Zero,
unsigned char kHz, bool Stop_Bits, unsigned long Max_Extent=0);
protected:
void enableIROut(unsigned char khz);
VIRTUAL void mark(unsigned int usec);
VIRTUAL void space(unsigned int usec);
unsigned long Extent;
unsigned char OnTime,OffTime,iLength;//used by bit-bang output.
};
class IRsendNEC: public virtual IRsendBase
{
public:
void send(unsigned long data);
};
class IRsendSony: public virtual IRsendBase
{
public:
void send(unsigned long data, int nbits, bool send3times=true); //by default, automatically repeat the send code 3 times; this may want to be manually set to false, however, in the event you are sending custom IR digital data streams using the Sony protocol. ~GS
};
class IRsendRaw: public virtual IRsendBase
{
public:
void send(unsigned int buf[], unsigned char len, unsigned char khz);
};
class IRsendRC5: public virtual IRsendBase
{
public:
void send(unsigned long data);
};
class IRsendRC6: public virtual IRsendBase
{
public:
void send(unsigned long data, unsigned char nbits);
};
class IRsendPanasonic_Old: public virtual IRsendBase
{
public:
void send(unsigned long data);
};
class IRsendJVC: public virtual IRsendBase
{
public:
void send(unsigned long data, bool First);
};
class IRsendNECx: public virtual IRsendBase
{
public:
void send(unsigned long data);
};
class IRsend:
public virtual IRsendNEC,
public virtual IRsendSony,
public virtual IRsendRaw,
public virtual IRsendRC5,
public virtual IRsendRC6,
public virtual IRsendPanasonic_Old,
public virtual IRsendJVC,
public virtual IRsendNECx
// , public virtual IRsendADDITIONAL //add additional protocols here
{
public:
void send(IR_types_t Type, unsigned long data, unsigned int data2, bool autoRepeatSend=true); //by default, automatically repeat the send command, if applicable: ex: for Sony, repeat the send code 3 times, per the standard; this may want to be manually set to false, however, in the event you are sending custom IR digital data streams using such protocols, in which case automatically sending each code repeatedly will corrupt the custom digital data stream being sent. ~GS
};
// Changed this to a base class so it can be extended
class IRrecvBase
{
public:
//methods (functions):
IRrecvBase(void) {};
IRrecvBase(unsigned char recvpin);
void noIROutput(void); //force IR output LED pin to Output LOW
void setBlinkLED(uint8_t pinNum=LED_BUILTIN, bool blinkActive=false);
void blink13(bool blinkActive); //same as above, except the pin is forced to be LED_BUILTIN; I left this here for backwards compatibility
bool getResults(IRdecodeBase *decoder, const unsigned int Time_per_Ticks=1);
void enableIRIn(void);
virtual void resume(void);
unsigned char getPinNum(void);
//variables:
int16_t Mark_Excess; //us; excess Mark time/lacking Space time, due to IR receiver filtering; *must* be *signed*, to allow negative values! For more info, see extensive "Notes on Mark_Excess" in IRLibMatch.h.
protected:
void init(void);
};
/* Original IRrecv class uses 50µs interrupts to sample input. While this is generally
* accurate enough for everyday purposes, it may be difficult to port to other
* hardware unless you know a lot about hardware timers and interrupts. Also
* when trying to analyze unknown protocols, the 50µs granularity may not be sufficient.
* In that case use either the IRrecvLoop or the IRrecvPCI class.
*/
#ifdef USE_IRRECV
class IRrecv: public IRrecvBase
{
public:
IRrecv(unsigned char recvpin):IRrecvBase(recvpin){};
bool getResults(IRdecodeBase *decoder);
void enableIRIn(void);
void detachInterrupt(); //disable the ISR so that it will no longer interrupt the code
void resume(void);
};
#endif
/* This receiver uses no interrupts or timers. Other interrupt driven receivers
* allow you to do other things and call getResults at your leisure to see if perhaps
* a sequence has been received. Typically you would put getResults in your loop
* and it would return false until the sequence had been received. However because this
* receiver uses no interrupts, it takes control of your program when you call getResults
* and doesn't let go until it's got something to show you. The advantage is you don't need
* interrupts which would make it easier to use and nonstandard hardware and will allow you to
* use any digital input pin. Timing of this routine is only as accurate as your "micros();"
*/
class IRrecvLoop: public IRrecvBase
{
public:
IRrecvLoop(unsigned char recvpin):IRrecvBase(recvpin){};
bool getResults(IRdecodeBase *decoder);
};
/* This receiver uses the pin change hardware interrupt to detect when your input pin
* changes state. It gives more detailed results than the 50µs interrupts of IRrecv
* and theoretically is more accurate than IRrecvLoop. However because it only detects
* pin changes, it doesn't always know when it's finished. getResults attempts to detect
* a long gap of space but sometimes the next signal gets there before getResults notices.
* This means the second set of signals can get messed up unless there is a long gap.
* This receiver is based in part on Arduino firmware for use with AnalysIR IR signal analysis
* software for Windows PCs. Many thanks to the people at http://analysir.com for their
* assistance in developing this section of code.
*/
#ifdef USE_ATTACH_INTERRUPTS
class IRrecvPCI: public IRrecvBase
{
public:
//Note this is interrupt number not pin number
IRrecvPCI(unsigned char inum);
void enableIRIn(void);
bool getResults(IRdecodeBase *decoder);
void detachInterrupt(); //disable the ISR so that it will no longer interrupt the code
void resume(); //resume data collection; see notes in IRLib.cpp
private:
unsigned char intrnum;
};
/* This class facilitates detection of frequency of an IR signal. Requires a TSMP58000
* or equivalent device connected to the hardware interrupt pin.
* Create an instance of the object passing the interrupt number.
*/
//Un-comment only one of the following three lines depending on available RAM
//#define FREQUENCY_BUFFER_TYPE unsigned char
#define FREQUENCY_BUFFER_TYPE int
//#define FREQUENCY_BUFFER_TYPE long
class IRfrequency
{
public:
//Note this is interrupt number, not pin number
IRfrequency(unsigned char inum);
void enableFreqDetect(void);
bool haveData(void); //detect if data is received
void disableFreqDetect(void);
void computeFreq(void); //computes but does not print results
void dumpResults(bool Detail); //computes and prints result
unsigned char getPinNum(void);//get value computed from interrupt number
double results; //results in kHz
unsigned char samples; //number of samples used in computation
private:
volatile unsigned FREQUENCY_BUFFER_TYPE Time_Stamp[256]; //MUST be volatile because used both in and outside ISRs
unsigned char intrnum, pin;
unsigned int i;
unsigned long sum;
};
#endif // ifdef USE_ATTACH_INTERRUPTS
//Do the actual blinking off and on
//This is not part of IRrecvBase because it may need to be inside an ISR
//and we cannot pass parameters to them.
//-pass in the pinState you'd like to set the LED to; this is usually just the inverse of the IR receiver's pin state
//--since the IR receiver is normally HIGH, and we want the LED to be normally LOW, we just use the inverse
void do_Blink(bool blinkState);
/* This routine maps interrupt numbers used by attachInterrupt() into pin numbers.
* NOTE: these interrupt numbers which are passed to “attachInterrupt()” are not
* necessarily identical to the interrupt numbers in the datasheet of the processor
* chip you are using. These interrupt numbers are a system unique to the
* “attachInterrupt()” Arduino function. It is used by both IRrecvPCI and IRfrequency.
*/
unsigned char Pin_from_Intr(unsigned char inum);
// Some useful constants
// Decoded value for NEC when a repeat code is received
#define REPEAT 0xffffffff
#endif //IRLib_h