first commit
This commit is contained in:
commit
a5a0434432
1126 changed files with 439481 additions and 0 deletions
4
Firmware/Source/grovepi/.gitignore
vendored
Normal file
4
Firmware/Source/grovepi/.gitignore
vendored
Normal file
|
|
@ -0,0 +1,4 @@
|
|||
.pioenvs
|
||||
.piolibdeps
|
||||
.clang_complete
|
||||
.gcc-flags.json
|
||||
67
Firmware/Source/grovepi/.travis.yml
Normal file
67
Firmware/Source/grovepi/.travis.yml
Normal file
|
|
@ -0,0 +1,67 @@
|
|||
# Continuous Integration (CI) is the practice, in software
|
||||
# engineering, of merging all developer working copies with a shared mainline
|
||||
# several times a day < http://docs.platformio.org/page/ci/index.html >
|
||||
#
|
||||
# Documentation:
|
||||
#
|
||||
# * Travis CI Embedded Builds with PlatformIO
|
||||
# < https://docs.travis-ci.com/user/integration/platformio/ >
|
||||
#
|
||||
# * PlatformIO integration with Travis CI
|
||||
# < http://docs.platformio.org/page/ci/travis.html >
|
||||
#
|
||||
# * User Guide for `platformio ci` command
|
||||
# < http://docs.platformio.org/page/userguide/cmd_ci.html >
|
||||
#
|
||||
#
|
||||
# Please choice one of the following templates (proposed below) and uncomment
|
||||
# it (remove "# " before each line) or use own configuration according to the
|
||||
# Travis CI documentation (see above).
|
||||
#
|
||||
|
||||
|
||||
#
|
||||
# Template #1: General project. Test it using existing `platformio.ini`.
|
||||
#
|
||||
|
||||
# language: python
|
||||
# python:
|
||||
# - "2.7"
|
||||
#
|
||||
# sudo: false
|
||||
# cache:
|
||||
# directories:
|
||||
# - "~/.platformio"
|
||||
#
|
||||
# install:
|
||||
# - pip install -U platformio
|
||||
# - platformio update
|
||||
#
|
||||
# script:
|
||||
# - platformio run
|
||||
|
||||
|
||||
#
|
||||
# Template #2: The project is intended to by used as a library with examples
|
||||
#
|
||||
|
||||
# language: python
|
||||
# python:
|
||||
# - "2.7"
|
||||
#
|
||||
# sudo: false
|
||||
# cache:
|
||||
# directories:
|
||||
# - "~/.platformio"
|
||||
#
|
||||
# env:
|
||||
# - PLATFORMIO_CI_SRC=path/to/test/file.c
|
||||
# - PLATFORMIO_CI_SRC=examples/file.ino
|
||||
# - PLATFORMIO_CI_SRC=path/to/test/directory
|
||||
#
|
||||
# install:
|
||||
# - pip install -U platformio
|
||||
# - platformio update
|
||||
#
|
||||
# script:
|
||||
# - platformio ci --lib="." --board=ID_1 --board=ID_2 --board=ID_N
|
||||
21
Firmware/Source/grovepi/extra_script.py
Normal file
21
Firmware/Source/grovepi/extra_script.py
Normal file
|
|
@ -0,0 +1,21 @@
|
|||
# Import('env')
|
||||
#
|
||||
# env.Replace(FUSESCMD="avrdude -c usbtiny -p m328p -U lfuse:w:0xFF:m -U hfuse:w:0xDA:m -U efuse:w:0xfd:m -B 3")
|
||||
|
||||
from os.path import join
|
||||
from SCons.Script import DefaultEnvironment
|
||||
|
||||
env = DefaultEnvironment()
|
||||
|
||||
env.Replace(
|
||||
MYUPLOADERFLAGS=[
|
||||
"-v",
|
||||
"-p", "$BOARD_MCU",
|
||||
"-c", "$UPLOAD_PROTOCOL",
|
||||
"-b", "115200",
|
||||
"-U", "lfuse:w:0xFF:m",
|
||||
"-U", "hfuse:w:0xDA:m",
|
||||
"-U", "efuse:w:0xFD:m"
|
||||
],
|
||||
UPLOADHEXCMD='"$UPLOADER" $MYUPLOADERFLAGS -U flash:w:$SOURCES:i'
|
||||
)
|
||||
41
Firmware/Source/grovepi/lib/readme.txt
Normal file
41
Firmware/Source/grovepi/lib/readme.txt
Normal file
|
|
@ -0,0 +1,41 @@
|
|||
|
||||
This directory is intended for the project specific (private) libraries.
|
||||
PlatformIO will compile them to static libraries and link to executable file.
|
||||
|
||||
The source code of each library should be placed in separate directory, like
|
||||
"lib/private_lib/[here are source files]".
|
||||
|
||||
For example, see how can be organized `Foo` and `Bar` libraries:
|
||||
|
||||
|--lib
|
||||
| |
|
||||
| |--Bar
|
||||
| | |--docs
|
||||
| | |--examples
|
||||
| | |--src
|
||||
| | |- Bar.c
|
||||
| | |- Bar.h
|
||||
| | |- library.json (optional, custom build options, etc) http://docs.platformio.org/page/librarymanager/config.html
|
||||
| |
|
||||
| |--Foo
|
||||
| | |- Foo.c
|
||||
| | |- Foo.h
|
||||
| |
|
||||
| |- readme.txt --> THIS FILE
|
||||
|
|
||||
|- platformio.ini
|
||||
|--src
|
||||
|- main.c
|
||||
|
||||
Then in `src/main.c` you should use:
|
||||
|
||||
#include <Foo.h>
|
||||
#include <Bar.h>
|
||||
|
||||
// rest H/C/CPP code
|
||||
|
||||
PlatformIO will find your libraries automatically, configure preprocessor's
|
||||
include paths and build them.
|
||||
|
||||
More information about PlatformIO Library Dependency Finder
|
||||
- http://docs.platformio.org/page/librarymanager/ldf.html
|
||||
17
Firmware/Source/grovepi/platformio.ini
Normal file
17
Firmware/Source/grovepi/platformio.ini
Normal file
|
|
@ -0,0 +1,17 @@
|
|||
; PlatformIO Project Configuration File
|
||||
;
|
||||
; Build options: build flags, source filter
|
||||
; Upload options: custom upload port, speed and extra flags
|
||||
; Library options: dependencies, extra library storages
|
||||
; Advanced options: extra scripting
|
||||
;
|
||||
; Please visit documentation for the other options and examples
|
||||
; http://docs.platformio.org/page/projectconf.html
|
||||
|
||||
[env:grovepi]
|
||||
platform = atmelavr
|
||||
framework = arduino
|
||||
board = 328p16m
|
||||
upload_protocol = usbtiny
|
||||
upload_speed = 57600
|
||||
extra_scripts = extra_script.py
|
||||
184
Firmware/Source/grovepi/src/ChainableLED.cpp
Normal file
184
Firmware/Source/grovepi/src/ChainableLED.cpp
Normal file
|
|
@ -0,0 +1,184 @@
|
|||
/*
|
||||
* Copyright (C) 2012 Paulo Marques (pjp.marques@gmail.com)
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||
* this software and associated documentation files (the "Software"), to deal in
|
||||
* the Software without restriction, including without limitation the rights to
|
||||
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
||||
* the Software, and to permit persons to whom the Software is furnished to do so,
|
||||
* subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in all
|
||||
* copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||||
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
||||
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
||||
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
*/
|
||||
|
||||
/* Information about the P9813 protocol obtained from:
|
||||
* http://www.seeedstudio.com/wiki/index.php?title=Twig_-_Chainable_RGB_LED
|
||||
*
|
||||
* HSB to RGB routine adapted from:
|
||||
* http://mjijackson.com/2008/02/rgb-to-hsl-and-rgb-to-hsv-color-model-conversion-algorithms-in-javascript
|
||||
*/
|
||||
|
||||
|
||||
// --------------------------------------------------------------------------------------
|
||||
|
||||
#include "ChainableLED.h"
|
||||
|
||||
// Forward declaration
|
||||
// float hue2rgb(float p, float q, float t);
|
||||
|
||||
// --------------------------------------------------------------------------------------
|
||||
|
||||
void ChainableLED::begin(byte clk_pin, byte data_pin, byte number_of_leds)
|
||||
{
|
||||
_clk_pin = clk_pin;
|
||||
_data_pin = data_pin;
|
||||
_num_leds = number_of_leds;
|
||||
pinMode(_clk_pin, OUTPUT);
|
||||
pinMode(_data_pin, OUTPUT);
|
||||
|
||||
_led_state = (byte*) calloc(_num_leds*3, sizeof(byte));
|
||||
|
||||
for (byte i=0; i<_num_leds; i++)
|
||||
setColorRGB(i, 0, 0, 0);
|
||||
}
|
||||
|
||||
//ChainableLED::~begin()
|
||||
//{
|
||||
// free(_led_state);
|
||||
//}
|
||||
|
||||
// --------------------------------------------------------------------------------------
|
||||
|
||||
void ChainableLED::clk(void)
|
||||
{
|
||||
digitalWrite(_clk_pin, LOW);
|
||||
delayMicroseconds(_CLK_PULSE_DELAY);
|
||||
digitalWrite(_clk_pin, HIGH);
|
||||
delayMicroseconds(_CLK_PULSE_DELAY);
|
||||
}
|
||||
|
||||
void ChainableLED::sendByte(byte b)
|
||||
{
|
||||
// Send one bit at a time, starting with the MSB
|
||||
for (byte i=0; i<8; i++)
|
||||
{
|
||||
// If MSB is 1, write one and clock it, else write 0 and clock
|
||||
if ((b & 0x80) != 0)
|
||||
digitalWrite(_data_pin, HIGH);
|
||||
else
|
||||
digitalWrite(_data_pin, LOW);
|
||||
clk();
|
||||
|
||||
// Advance to the next bit to send
|
||||
b <<= 1;
|
||||
}
|
||||
}
|
||||
|
||||
void ChainableLED::sendColor(byte red, byte green, byte blue)
|
||||
{
|
||||
// Start by sending a byte with the format "1 1 /B7 /B6 /G7 /G6 /R7 /R6"
|
||||
byte prefix = B11000000;
|
||||
if ((blue & 0x80) == 0) prefix|= B00100000;
|
||||
if ((blue & 0x40) == 0) prefix|= B00010000;
|
||||
if ((green & 0x80) == 0) prefix|= B00001000;
|
||||
if ((green & 0x40) == 0) prefix|= B00000100;
|
||||
if ((red & 0x80) == 0) prefix|= B00000010;
|
||||
if ((red & 0x40) == 0) prefix|= B00000001;
|
||||
sendByte(prefix);
|
||||
|
||||
// Now must send the 3 colors
|
||||
sendByte(blue);
|
||||
sendByte(green);
|
||||
sendByte(red);
|
||||
}
|
||||
|
||||
void ChainableLED::setColorRGB(byte led, byte red, byte green, byte blue)
|
||||
{
|
||||
// Send data frame prefix (32x "0")
|
||||
sendByte(0x00);
|
||||
sendByte(0x00);
|
||||
sendByte(0x00);
|
||||
sendByte(0x00);
|
||||
|
||||
// Send color data for each one of the leds
|
||||
for (byte i=0; i<_num_leds; i++)
|
||||
{
|
||||
if (i == led)
|
||||
{
|
||||
_led_state[i*3 + _CL_RED] = red;
|
||||
_led_state[i*3 + _CL_GREEN] = green;
|
||||
_led_state[i*3 + _CL_BLUE] = blue;
|
||||
}
|
||||
|
||||
sendColor(_led_state[i*3 + _CL_RED],
|
||||
_led_state[i*3 + _CL_GREEN],
|
||||
_led_state[i*3 + _CL_BLUE]);
|
||||
}
|
||||
|
||||
// Terminate data frame (32x "0")
|
||||
sendByte(0x00);
|
||||
sendByte(0x00);
|
||||
sendByte(0x00);
|
||||
sendByte(0x00);
|
||||
}
|
||||
|
||||
//void ChainableLED::setColorHSB(byte led, float hue, float saturation, float brightness)
|
||||
//{
|
||||
// float r, g, b;
|
||||
//
|
||||
// constrain(hue, 0.0, 1.0);
|
||||
// constrain(saturation, 0.0, 1.0);
|
||||
// constrain(brightness, 0.0, 1.0);
|
||||
//
|
||||
// if(saturation == 0.0)
|
||||
// {
|
||||
// r = g = b = brightness;
|
||||
// }
|
||||
// else
|
||||
// {
|
||||
// float q = brightness < 0.5 ?
|
||||
// brightness * (1.0 + saturation) : brightness + saturation - brightness * saturation;
|
||||
// float p = 2.0 * brightness - q;
|
||||
// r = hue2rgb(p, q, hue + 1.0/3.0);
|
||||
// g = hue2rgb(p, q, hue);
|
||||
// b = hue2rgb(p, q, hue - 1.0/3.0);
|
||||
// }
|
||||
//
|
||||
// setColorRGB(led, (byte)(255.0*r), (byte)(255.0*g), (byte)(255.0*b));
|
||||
//}
|
||||
|
||||
// --------------------------------------------------------------------------------------
|
||||
|
||||
//float hue2rgb(float p, float q, float t)
|
||||
//{
|
||||
// if (t < 0.0)
|
||||
// t += 1.0;
|
||||
// if(t > 1.0)
|
||||
// t -= 1.0;
|
||||
// if(t < 1.0/6.0)
|
||||
// return p + (q - p) * 6.0 * t;
|
||||
// if(t < 1.0/2.0)
|
||||
// return q;
|
||||
// if(t < 2.0/3.0)
|
||||
// return p + (q - p) * (2.0/3.0 - t) * 6.0;
|
||||
//
|
||||
// return p;
|
||||
//}
|
||||
|
||||
byte ChainableLED::getNumLeds()
|
||||
{
|
||||
return _num_leds;
|
||||
}
|
||||
|
||||
bool ChainableLED::ready()
|
||||
{
|
||||
return _clk_pin != 0;
|
||||
}
|
||||
68
Firmware/Source/grovepi/src/ChainableLED.h
Normal file
68
Firmware/Source/grovepi/src/ChainableLED.h
Normal file
|
|
@ -0,0 +1,68 @@
|
|||
/*
|
||||
Copyright (C) 2012 Paulo Marques (pjp.marques@gmail.com)
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||
this software and associated documentation files (the "Software"), to deal in
|
||||
the Software without restriction, including without limitation the rights to
|
||||
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
|
||||
the Software, and to permit persons to whom the Software is furnished to do so,
|
||||
subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
|
||||
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
||||
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
|
||||
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
||||
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||||
*/
|
||||
|
||||
/*
|
||||
* Library for controlling a chain of RGB LEDs based on the P9813 protocol.
|
||||
* E.g., supports the Grove Chainable RGB LED product.
|
||||
*
|
||||
* Information about the P9813 protocol obtained from:
|
||||
* http://www.seeedstudio.com/wiki/index.php?title=Twig_-_Chainable_RGB_LED
|
||||
*/
|
||||
|
||||
|
||||
|
||||
#ifndef __ChainableLED_h__
|
||||
#define __ChainableLED_h__
|
||||
|
||||
#include "Arduino.h"
|
||||
|
||||
#define _CL_RED 0
|
||||
#define _CL_GREEN 1
|
||||
#define _CL_BLUE 2
|
||||
#define _CLK_PULSE_DELAY 20
|
||||
|
||||
class ChainableLED
|
||||
{
|
||||
public:
|
||||
// ChainableLED(byte clk_pin, byte data_pin, byte number_of_leds);
|
||||
void begin(byte clk_pin, byte data_pin, byte number_of_leds);
|
||||
// ~begin();
|
||||
|
||||
void setColorRGB(byte led, byte red, byte green, byte blue);
|
||||
//void setColorHSB(byte led, float hue, float saturation, float brightness);
|
||||
|
||||
byte getNumLeds();
|
||||
bool ready();
|
||||
|
||||
private:
|
||||
byte _clk_pin;
|
||||
byte _data_pin;
|
||||
byte _num_leds;
|
||||
|
||||
byte* _led_state;
|
||||
|
||||
void clk(void);
|
||||
void sendByte(byte b);
|
||||
void sendColor(byte red, byte green, byte blue);
|
||||
};
|
||||
|
||||
#endif
|
||||
|
||||
201
Firmware/Source/grovepi/src/DHT.cpp
Normal file
201
Firmware/Source/grovepi/src/DHT.cpp
Normal file
|
|
@ -0,0 +1,201 @@
|
|||
/* DHT library
|
||||
|
||||
MIT license
|
||||
written by Adafruit Industries
|
||||
*/
|
||||
|
||||
#include "DHT.h"
|
||||
/*
|
||||
DHT::DHT(uint8_t pin, uint8_t type, uint8_t count) {
|
||||
_pin = pin;
|
||||
_type = type;
|
||||
_count = count;
|
||||
firstreading = true;
|
||||
}
|
||||
*/
|
||||
void DHT::begin(uint8_t pin, uint8_t type, uint8_t count) {
|
||||
// set up the pins!
|
||||
_pin = pin;
|
||||
_type = type;
|
||||
_count = count;
|
||||
firstreading = true;
|
||||
pinMode(_pin, INPUT);
|
||||
digitalWrite(_pin, HIGH);
|
||||
_lastreadtime = 0;
|
||||
}
|
||||
|
||||
//boolean S == Scale. True == Farenheit; False == Celcius
|
||||
float DHT::readTemperature(bool S) {
|
||||
float f;
|
||||
|
||||
if (read()) {
|
||||
switch (_type) {
|
||||
case DHT11:
|
||||
f = data[2];
|
||||
if(S)
|
||||
f = convertCtoF(f);
|
||||
|
||||
return f;
|
||||
case DHT22:
|
||||
case DHT21:
|
||||
f = data[2] & 0x7F;
|
||||
f *= 256;
|
||||
f += data[3];
|
||||
f /= 10;
|
||||
if (data[2] & 0x80)
|
||||
f *= -1;
|
||||
if(S)
|
||||
f = convertCtoF(f);
|
||||
|
||||
return f;
|
||||
}
|
||||
}
|
||||
Serial.print("Read fail");
|
||||
return NAN;
|
||||
}
|
||||
|
||||
float DHT::convertCtoF(float c) {
|
||||
return c * 9 / 5 + 32;
|
||||
}
|
||||
|
||||
float DHT::readHumidity(void) {
|
||||
float f;
|
||||
if (read()) {
|
||||
switch (_type) {
|
||||
case DHT11:
|
||||
f = data[0];
|
||||
return f;
|
||||
case DHT22:
|
||||
case DHT21:
|
||||
f = data[0];
|
||||
f *= 256;
|
||||
f += data[1];
|
||||
f /= 10;
|
||||
return f;
|
||||
}
|
||||
}
|
||||
Serial.print("Read fail");
|
||||
return NAN;
|
||||
}
|
||||
|
||||
|
||||
boolean DHT::read(void) {
|
||||
uint8_t laststate = HIGH;
|
||||
uint8_t counter = 0;
|
||||
uint8_t j = 0, i;
|
||||
unsigned long currenttime;
|
||||
|
||||
// pull the pin high and wait 250 milliseconds
|
||||
digitalWrite(_pin, HIGH);
|
||||
delay(250);
|
||||
|
||||
currenttime = millis();
|
||||
if (currenttime < _lastreadtime) {
|
||||
// ie there was a rollover
|
||||
_lastreadtime = 0;
|
||||
}
|
||||
if (!firstreading && ((currenttime - _lastreadtime) < 2000)) {
|
||||
return true; // return last correct measurement
|
||||
//delay(2000 - (currenttime - _lastreadtime));
|
||||
}
|
||||
firstreading = false;
|
||||
/*
|
||||
Serial.print("Currtime: "); Serial.print(currenttime);
|
||||
Serial.print(" Lasttime: "); Serial.print(_lastreadtime);
|
||||
*/
|
||||
_lastreadtime = millis();
|
||||
|
||||
data[0] = data[1] = data[2] = data[3] = data[4] = 0;
|
||||
|
||||
// now pull it low for ~20 milliseconds
|
||||
pinMode(_pin, OUTPUT);
|
||||
digitalWrite(_pin, LOW);
|
||||
delay(20);
|
||||
cli();
|
||||
digitalWrite(_pin, HIGH);
|
||||
delayMicroseconds(40);
|
||||
pinMode(_pin, INPUT);
|
||||
|
||||
// read in timings
|
||||
for ( i=0; i< MAXTIMINGS; i++) {
|
||||
counter = 0;
|
||||
while (digitalRead(_pin) == laststate) {
|
||||
counter++;
|
||||
delayMicroseconds(1);
|
||||
if (counter == 255) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
laststate = digitalRead(_pin);
|
||||
|
||||
if (counter == 255) break;
|
||||
|
||||
// ignore first 3 transitions
|
||||
if ((i >= 4) && (i%2 == 0)) {
|
||||
// shove each bit into the storage bytes
|
||||
data[j/8] <<= 1;
|
||||
if (counter > _count)
|
||||
data[j/8] |= 1;
|
||||
j++;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
sei();
|
||||
|
||||
/*
|
||||
Serial.println(j, DEC);
|
||||
Serial.print(data[0], HEX); Serial.print(", ");
|
||||
Serial.print(data[1], HEX); Serial.print(", ");
|
||||
Serial.print(data[2], HEX); Serial.print(", ");
|
||||
Serial.print(data[3], HEX); Serial.print(", ");
|
||||
Serial.print(data[4], HEX); Serial.print(" =? ");
|
||||
Serial.println(data[0] + data[1] + data[2] + data[3], HEX);
|
||||
*/
|
||||
|
||||
// check we read 40 bits and that the checksum matches
|
||||
if ((j >= 40) &&
|
||||
(data[4] == ((data[0] + data[1] + data[2] + data[3]) & 0xFF)) ) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
float* DHT::readTempHum(bool scale)
|
||||
{
|
||||
if(!read())
|
||||
return NULL;
|
||||
|
||||
switch(_type) {
|
||||
case DHT11:
|
||||
// temperature
|
||||
buffer[0] = data[2];
|
||||
if(scale)
|
||||
buffer[0] = convertCtoF(buffer[0]);
|
||||
|
||||
// humidity
|
||||
buffer[1] = data[0];
|
||||
break;
|
||||
case DHT22:
|
||||
case DHT21:
|
||||
// temperature
|
||||
buffer[0] = data[2] & 0x7F;
|
||||
buffer[0] *= 256;
|
||||
buffer[0] += data[3];
|
||||
buffer[0] /= 10;
|
||||
if (data[2] & 0x80)
|
||||
buffer[0] *= -1;
|
||||
if(scale)
|
||||
buffer[0] = convertCtoF(buffer[0]);
|
||||
|
||||
// humidity
|
||||
buffer[1] = data[0];
|
||||
buffer[1] *= 256;
|
||||
buffer[1] += data[1];
|
||||
buffer[1] /= 10;
|
||||
break;
|
||||
}
|
||||
|
||||
return &buffer[0];
|
||||
}
|
||||
40
Firmware/Source/grovepi/src/DHT.h
Normal file
40
Firmware/Source/grovepi/src/DHT.h
Normal file
|
|
@ -0,0 +1,40 @@
|
|||
#ifndef DHT_H
|
||||
#define DHT_H
|
||||
#if ARDUINO >= 100
|
||||
#include "Arduino.h"
|
||||
#else
|
||||
#include "WProgram.h"
|
||||
#endif
|
||||
|
||||
/* DHT library
|
||||
|
||||
MIT license
|
||||
written by Adafruit Industries
|
||||
*/
|
||||
|
||||
// how many timing transitions we need to keep track of. 2 * number bits + extra
|
||||
#define MAXTIMINGS 85
|
||||
|
||||
#define DHT11 11
|
||||
#define DHT22 22
|
||||
#define DHT21 21
|
||||
#define AM2301 21
|
||||
|
||||
class DHT {
|
||||
private:
|
||||
uint8_t data[6];
|
||||
uint8_t _pin, _type, _count;
|
||||
boolean read(void);
|
||||
unsigned long _lastreadtime;
|
||||
boolean firstreading;
|
||||
float buffer[2];
|
||||
|
||||
public:
|
||||
//DHT(uint8_t pin, uint8_t type, uint8_t count=6);
|
||||
void begin(uint8_t pin, uint8_t type, uint8_t count=6);
|
||||
float readTemperature(bool S=false);
|
||||
float convertCtoF(float);
|
||||
float readHumidity(void);
|
||||
float* readTempHum(bool scale = false);
|
||||
};
|
||||
#endif
|
||||
175
Firmware/Source/grovepi/src/Grove_LED_Bar.cpp
Normal file
175
Firmware/Source/grovepi/src/Grove_LED_Bar.cpp
Normal file
|
|
@ -0,0 +1,175 @@
|
|||
/*
|
||||
LED bar library V2.0
|
||||
2010 Copyright (c) Seeed Technology Inc. All right reserved.
|
||||
|
||||
Original Author: LG
|
||||
|
||||
Modify: Loovee, 2014-2-26
|
||||
User can choose which Io to be used.
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
This library is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public
|
||||
License along with this library; if not, write to the Free Software
|
||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
*/
|
||||
|
||||
#include <Arduino.h>
|
||||
#include "Grove_LED_Bar.h"
|
||||
|
||||
// Initialise
|
||||
//Grove_LED_Bar::Grove_LED_Bar(unsigned char pinClock, unsigned char pinData, bool greenToRed)
|
||||
void Grove_LED_Bar::begin(unsigned char pinClock, unsigned char pinData, bool greenToRed)
|
||||
{
|
||||
__pinClock = pinClock;
|
||||
__pinData = pinData;
|
||||
__greenToRed = greenToRed; // ascending or decending
|
||||
|
||||
__state = 0x00; // persist state so individual leds can be toggled
|
||||
|
||||
pinMode(__pinClock, OUTPUT);
|
||||
pinMode(__pinData, OUTPUT);
|
||||
}
|
||||
|
||||
|
||||
// Send the latch command
|
||||
void Grove_LED_Bar::latchData()
|
||||
{
|
||||
digitalWrite(__pinData, LOW);
|
||||
delayMicroseconds(10);
|
||||
|
||||
for (unsigned char i = 0; i < 4; i++)
|
||||
{
|
||||
digitalWrite(__pinData, HIGH);
|
||||
digitalWrite(__pinData, LOW);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// Send 16 bits of data
|
||||
void Grove_LED_Bar::sendData(unsigned int data)
|
||||
{
|
||||
for (unsigned char i = 0; i < 16; i++)
|
||||
{
|
||||
unsigned int state = (data & 0x8000) ? HIGH : LOW;
|
||||
digitalWrite(__pinData, state);
|
||||
|
||||
state = digitalRead(__pinClock) ? LOW : HIGH;
|
||||
digitalWrite(__pinClock, state);
|
||||
|
||||
data <<= 1;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// Change the orientation
|
||||
// Green to red, or red to green
|
||||
void Grove_LED_Bar::setGreenToRed(bool greenToRed)
|
||||
{
|
||||
__greenToRed = greenToRed;
|
||||
|
||||
setBits(__state);
|
||||
}
|
||||
|
||||
|
||||
// Set level (0-10)
|
||||
// Level 0 means all leds off
|
||||
// Level 10 means all leds on
|
||||
void Grove_LED_Bar::setLevel(unsigned char level)
|
||||
{
|
||||
level = max(0, min(10, level));
|
||||
|
||||
// Set level number of bits from the left to 1
|
||||
__state = ~(~0 << level);
|
||||
|
||||
setBits(__state);
|
||||
}
|
||||
|
||||
|
||||
// Set a single led
|
||||
// led (1-10)
|
||||
// state (0=off, 1=on)
|
||||
void Grove_LED_Bar::setLed(unsigned char led, bool state)
|
||||
{
|
||||
led = max(1, min(10, led));
|
||||
|
||||
// Zero based index 0-9 for bitwise operations
|
||||
led--;
|
||||
|
||||
// Bitwise OR or bitwise AND
|
||||
__state = state ? (__state | (0x01<<led)) : (__state & ~(0x01<<led));
|
||||
|
||||
setBits(__state);
|
||||
}
|
||||
|
||||
|
||||
// Toggle a single led
|
||||
// led (1-10)
|
||||
void Grove_LED_Bar::toggleLed(unsigned char led)
|
||||
{
|
||||
led = max(1, min(10, led));
|
||||
|
||||
// Zero based index 0-9 for bitwise operations
|
||||
led--;
|
||||
|
||||
// Bitwise XOR the leds current value
|
||||
__state ^= (0x01<<led);
|
||||
|
||||
setBits(__state);
|
||||
}
|
||||
|
||||
|
||||
// Set the current state, one bit for each led
|
||||
// 0 = 0x0 = 0b000000000000000 = all leds off
|
||||
// 5 = 0x05 = 0b000000000000101 = leds 1 and 3 on, all others off
|
||||
// 341 = 0x155 = 0b000000101010101 = leds 1,3,5,7,9 on, 2,4,6,8,10 off
|
||||
// 1023 = 0x3ff = 0b000001111111111 = all leds on
|
||||
// | |
|
||||
// 10 1
|
||||
void Grove_LED_Bar::setBits(unsigned int bits)
|
||||
{
|
||||
__state = bits & 0x3FF;
|
||||
|
||||
sendData(CMDMODE);
|
||||
|
||||
for (unsigned char i = 0; i < 10; i++)
|
||||
{
|
||||
if (__greenToRed)
|
||||
{
|
||||
// Bitwise AND the 10th bit (0x200) and left shift to cycle through all bits
|
||||
sendData((bits << i) & 0x200 ? ON : OFF);
|
||||
}
|
||||
else
|
||||
{
|
||||
// Bitwise AND the 1st bit (0x01) and right shift to cycle through all bits
|
||||
sendData((bits >> i) & 0x01 ? ON : OFF);
|
||||
}
|
||||
}
|
||||
|
||||
// Two extra empty bits for padding the command to the correct length
|
||||
sendData(0x00);
|
||||
sendData(0x00);
|
||||
|
||||
latchData();
|
||||
}
|
||||
|
||||
|
||||
// Return the current state
|
||||
unsigned int Grove_LED_Bar::getBits()
|
||||
{
|
||||
return __state;
|
||||
}
|
||||
|
||||
// Has this instance been initialised?
|
||||
bool Grove_LED_Bar::ready()
|
||||
{
|
||||
return __pinClock != 0;
|
||||
}
|
||||
59
Firmware/Source/grovepi/src/Grove_LED_Bar.h
Normal file
59
Firmware/Source/grovepi/src/Grove_LED_Bar.h
Normal file
|
|
@ -0,0 +1,59 @@
|
|||
/*
|
||||
LED bar library V2.0
|
||||
2010 Copyright (c) Seeed Technology Inc. All right reserved.
|
||||
|
||||
Original Author: LG
|
||||
|
||||
Modify: Loovee, 2014-2-26
|
||||
User can choose which Io to be used.
|
||||
|
||||
This library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
This library is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public
|
||||
License along with this library; if not, write to the Free Software
|
||||
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
*/
|
||||
|
||||
#ifndef Grove_LED_Bar_H
|
||||
#define Grove_LED_Bar_H
|
||||
|
||||
#define CMDMODE 0x00 // Work on 8-bit mode
|
||||
#define ON 0xff // 8-bit 1 data
|
||||
#define OFF 0x00 // 8-bit 0 data
|
||||
|
||||
class Grove_LED_Bar
|
||||
{
|
||||
|
||||
private:
|
||||
|
||||
unsigned int __pinClock; // Clock pin
|
||||
unsigned int __pinData; // Data pin
|
||||
bool __greenToRed; // Orientation (0 = red to green, 1 = green to red)
|
||||
unsigned int __state; // Led state
|
||||
|
||||
void sendData(unsigned int data); // Send a word to led bar
|
||||
void latchData(void); // Load data into the latch register
|
||||
|
||||
public:
|
||||
|
||||
//Grove_LED_Bar(unsigned char pinClock, unsigned char pinData, bool greenToRed); // Initialize
|
||||
void begin(unsigned char pinClock, unsigned char pinData, bool greenToRed); // Initialize
|
||||
|
||||
void setGreenToRed(bool greenToRed); // (Re)set orientation
|
||||
void setLevel(unsigned char level); // Set level
|
||||
void setLed(unsigned char led, bool state); // Set a single led
|
||||
void toggleLed(unsigned char led); // Toggle a single led
|
||||
void setBits(unsigned int bits); // Toggle leds to match given bits
|
||||
unsigned int getBits(); // Get the current state
|
||||
bool ready(); // Has this instance been initialised?
|
||||
};
|
||||
|
||||
#endif
|
||||
38
Firmware/Source/grovepi/src/IRrecvDemo.ino
Normal file
38
Firmware/Source/grovepi/src/IRrecvDemo.ino
Normal file
|
|
@ -0,0 +1,38 @@
|
|||
// /*
|
||||
// * IRremote: IRrecvDemo - demonstrates receiving IR codes with IRrecv
|
||||
// * An IR detector/demodulator must be connected to the input RECV_PIN.
|
||||
// * Version 0.1 July, 2009
|
||||
// * Copyright 2009 Ken Shirriff
|
||||
// * http://arcfn.com
|
||||
// */
|
||||
//
|
||||
// #include <IRremote.h>
|
||||
//
|
||||
// int RECV_PIN = 3;
|
||||
// int BLINK_PIN = 2;
|
||||
//
|
||||
// IRrecv irrecv(RECV_PIN, BLINK_PIN);
|
||||
//
|
||||
// decode_results results;
|
||||
//
|
||||
// void setup()
|
||||
// {
|
||||
// Serial.begin(9600);
|
||||
// // In case the interrupt driver crashes on setup, give a clue
|
||||
// // to the user what's going on.
|
||||
// Serial.println("Enabling IRin");
|
||||
// irrecv.enableIRIn(); // Start the receiver
|
||||
// Serial.println("Enabled IRin");
|
||||
// }
|
||||
//
|
||||
// void loop() {
|
||||
// if (irrecv.decode(&results)) {
|
||||
// Serial.print(results.decode_type);
|
||||
// Serial.print(" ");
|
||||
// Serial.print(results.address, HEX);
|
||||
// Serial.print(" ");
|
||||
// Serial.println(results.value, HEX);
|
||||
// irrecv.resume(); // Receive the next value
|
||||
// }
|
||||
// delay(100);
|
||||
// }
|
||||
200
Firmware/Source/grovepi/src/IRremote.cpp
Normal file
200
Firmware/Source/grovepi/src/IRremote.cpp
Normal file
|
|
@ -0,0 +1,200 @@
|
|||
//******************************************************************************
|
||||
// IRremote
|
||||
// Version 2.0.1 June, 2015
|
||||
// Copyright 2009 Ken Shirriff
|
||||
// For details, see http://arcfn.com/2009/08/multi-protocol-infrared-remote-library.html
|
||||
//
|
||||
// Modified by Paul Stoffregen <paul@pjrc.com> to support other boards and timers
|
||||
// Modified by Mitra Ardron <mitra@mitra.biz>
|
||||
// Added Sanyo and Mitsubishi controllers
|
||||
// Modified Sony to spot the repeat codes that some Sony's send
|
||||
//
|
||||
// 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/
|
||||
//
|
||||
// JVC and Panasonic protocol added by Kristian Lauszus (Thanks to zenwheel and other people at the original blog post)
|
||||
// LG added by Darryl Smith (based on the JVC protocol)
|
||||
// Whynter A/C ARC-110WD added by Francesco Meschia
|
||||
//******************************************************************************
|
||||
|
||||
// Defining IR_GLOBAL here allows us to declare the instantiation of global variables
|
||||
#define IR_GLOBAL
|
||||
# include "IRremote.h"
|
||||
# include "IRremoteInt.h"
|
||||
#undef IR_GLOBAL
|
||||
|
||||
#ifndef IR_TIMER_USE_ESP32
|
||||
#include <avr/interrupt.h>
|
||||
#endif
|
||||
|
||||
|
||||
//+=============================================================================
|
||||
// The match functions were (apparently) originally MACROs to improve code speed
|
||||
// (although this would have bloated the code) hence the names being CAPS
|
||||
// A later release implemented debug output and so they needed to be converted
|
||||
// to functions.
|
||||
// I tried to implement a dual-compile mode (DEBUG/non-DEBUG) but for some
|
||||
// reason, no matter what I did I could not get them to function as macros again.
|
||||
// I have found a *lot* of bugs in the Arduino compiler over the last few weeks,
|
||||
// and I am currently assuming that one of these bugs is my problem.
|
||||
// I may revisit this code at a later date and look at the assembler produced
|
||||
// in a hope of finding out what is going on, but for now they will remain as
|
||||
// functions even in non-DEBUG mode
|
||||
//
|
||||
int MATCH (int measured, int desired)
|
||||
{
|
||||
DBG_PRINT(F("Testing: "));
|
||||
DBG_PRINT(TICKS_LOW(desired), DEC);
|
||||
DBG_PRINT(F(" <= "));
|
||||
DBG_PRINT(measured, DEC);
|
||||
DBG_PRINT(F(" <= "));
|
||||
DBG_PRINT(TICKS_HIGH(desired), DEC);
|
||||
|
||||
bool passed = ((measured >= TICKS_LOW(desired)) && (measured <= TICKS_HIGH(desired)));
|
||||
if (passed)
|
||||
DBG_PRINTLN(F("?; passed"));
|
||||
else
|
||||
DBG_PRINTLN(F("?; FAILED"));
|
||||
return passed;
|
||||
}
|
||||
|
||||
//+========================================================
|
||||
// Due to sensor lag, when received, Marks tend to be 100us too long
|
||||
//
|
||||
int MATCH_MARK (int measured_ticks, int desired_us)
|
||||
{
|
||||
DBG_PRINT(F("Testing mark (actual vs desired): "));
|
||||
DBG_PRINT(measured_ticks * USECPERTICK, DEC);
|
||||
DBG_PRINT(F("us vs "));
|
||||
DBG_PRINT(desired_us, DEC);
|
||||
DBG_PRINT("us");
|
||||
DBG_PRINT(": ");
|
||||
DBG_PRINT(TICKS_LOW(desired_us + MARK_EXCESS) * USECPERTICK, DEC);
|
||||
DBG_PRINT(F(" <= "));
|
||||
DBG_PRINT(measured_ticks * USECPERTICK, DEC);
|
||||
DBG_PRINT(F(" <= "));
|
||||
DBG_PRINT(TICKS_HIGH(desired_us + MARK_EXCESS) * USECPERTICK, DEC);
|
||||
|
||||
bool passed = ((measured_ticks >= TICKS_LOW (desired_us + MARK_EXCESS))
|
||||
&& (measured_ticks <= TICKS_HIGH(desired_us + MARK_EXCESS)));
|
||||
if (passed)
|
||||
DBG_PRINTLN(F("?; passed"));
|
||||
else
|
||||
DBG_PRINTLN(F("?; FAILED"));
|
||||
return passed;
|
||||
}
|
||||
|
||||
//+========================================================
|
||||
// Due to sensor lag, when received, Spaces tend to be 100us too short
|
||||
//
|
||||
int MATCH_SPACE (int measured_ticks, int desired_us)
|
||||
{
|
||||
DBG_PRINT(F("Testing space (actual vs desired): "));
|
||||
DBG_PRINT(measured_ticks * USECPERTICK, DEC);
|
||||
DBG_PRINT(F("us vs "));
|
||||
DBG_PRINT(desired_us, DEC);
|
||||
DBG_PRINT("us");
|
||||
DBG_PRINT(": ");
|
||||
DBG_PRINT(TICKS_LOW(desired_us - MARK_EXCESS) * USECPERTICK, DEC);
|
||||
DBG_PRINT(F(" <= "));
|
||||
DBG_PRINT(measured_ticks * USECPERTICK, DEC);
|
||||
DBG_PRINT(F(" <= "));
|
||||
DBG_PRINT(TICKS_HIGH(desired_us - MARK_EXCESS) * USECPERTICK, DEC);
|
||||
|
||||
bool passed = ((measured_ticks >= TICKS_LOW (desired_us - MARK_EXCESS))
|
||||
&& (measured_ticks <= TICKS_HIGH(desired_us - MARK_EXCESS)));
|
||||
if (passed)
|
||||
DBG_PRINTLN(F("?; passed"));
|
||||
else
|
||||
DBG_PRINTLN(F("?; FAILED"));
|
||||
return passed;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Interrupt Service Routine - Fires every 50uS
|
||||
// TIMER2 interrupt code to collect raw data.
|
||||
// Widths of alternating SPACE, MARK are recorded in rawbuf.
|
||||
// Recorded in ticks of 50uS [microseconds, 0.000050 seconds]
|
||||
// 'rawlen' counts the number of entries recorded so far.
|
||||
// First entry is the SPACE between transmissions.
|
||||
// As soon as a the first [SPACE] entry gets long:
|
||||
// Ready is set; State switches to IDLE; Timing of SPACE continues.
|
||||
// As soon as first MARK arrives:
|
||||
// Gap width is recorded; Ready is cleared; New logging starts
|
||||
//
|
||||
#ifdef IR_TIMER_USE_ESP32
|
||||
void IRTimer()
|
||||
#else
|
||||
ISR (TIMER_INTR_NAME)
|
||||
#endif
|
||||
{
|
||||
TIMER_RESET;
|
||||
|
||||
// Read if IR Receiver -> SPACE [xmt LED off] or a MARK [xmt LED on]
|
||||
// digitalRead() is very slow. Optimisation is possible, but makes the code unportable
|
||||
uint8_t irdata = (uint8_t)digitalRead(irparams.recvpin);
|
||||
|
||||
irparams.timer++; // One more 50uS tick
|
||||
if (irparams.rawlen >= RAWBUF) irparams.rcvstate = STATE_OVERFLOW ; // Buffer overflow
|
||||
|
||||
switch(irparams.rcvstate) {
|
||||
//......................................................................
|
||||
case STATE_IDLE: // In the middle of a gap
|
||||
if (irdata == MARK) {
|
||||
if (irparams.timer < GAP_TICKS) { // Not big enough to be a gap.
|
||||
irparams.timer = 0;
|
||||
|
||||
} else {
|
||||
// Gap just ended; Record duration; Start recording transmission
|
||||
irparams.overflow = false;
|
||||
irparams.rawlen = 0;
|
||||
irparams.rawbuf[irparams.rawlen++] = irparams.timer;
|
||||
irparams.timer = 0;
|
||||
irparams.rcvstate = STATE_MARK;
|
||||
}
|
||||
}
|
||||
break;
|
||||
//......................................................................
|
||||
case STATE_MARK: // Timing Mark
|
||||
if (irdata == SPACE) { // Mark ended; Record time
|
||||
irparams.rawbuf[irparams.rawlen++] = irparams.timer;
|
||||
irparams.timer = 0;
|
||||
irparams.rcvstate = STATE_SPACE;
|
||||
}
|
||||
break;
|
||||
//......................................................................
|
||||
case STATE_SPACE: // Timing Space
|
||||
if (irdata == MARK) { // Space just ended; Record time
|
||||
irparams.rawbuf[irparams.rawlen++] = irparams.timer;
|
||||
irparams.timer = 0;
|
||||
irparams.rcvstate = STATE_MARK;
|
||||
|
||||
} else if (irparams.timer > GAP_TICKS) { // Space
|
||||
// A long Space, indicates gap between codes
|
||||
// Flag the current code as ready for processing
|
||||
// Switch to STOP
|
||||
// Don't reset timer; keep counting Space width
|
||||
irparams.rcvstate = STATE_STOP;
|
||||
}
|
||||
break;
|
||||
//......................................................................
|
||||
case STATE_STOP: // Waiting; Measuring Gap
|
||||
if (irdata == MARK) irparams.timer = 0 ; // Reset gap timer
|
||||
break;
|
||||
//......................................................................
|
||||
case STATE_OVERFLOW: // Flag up a read overflow; Stop the State Machine
|
||||
irparams.overflow = true;
|
||||
irparams.rcvstate = STATE_STOP;
|
||||
break;
|
||||
}
|
||||
|
||||
// If requested, flash LED while receiving IR data
|
||||
if (irparams.blinkflag) {
|
||||
if (irdata == MARK)
|
||||
if (irparams.blinkpin) digitalWrite(irparams.blinkpin, HIGH); // Turn user defined pin LED on
|
||||
else BLINKLED_ON() ; // if no user defined LED pin, turn default LED pin for the hardware on
|
||||
else if (irparams.blinkpin) digitalWrite(irparams.blinkpin, LOW); // Turn user defined pin LED on
|
||||
else BLINKLED_OFF() ; // if no user defined LED pin, turn default LED pin for the hardware on
|
||||
}
|
||||
}
|
||||
347
Firmware/Source/grovepi/src/IRremote.h
Normal file
347
Firmware/Source/grovepi/src/IRremote.h
Normal file
|
|
@ -0,0 +1,347 @@
|
|||
|
||||
//******************************************************************************
|
||||
// IRremote
|
||||
// Version 2.0.1 June, 2015
|
||||
// Copyright 2009 Ken Shirriff
|
||||
// For details, see http://arcfn.com/2009/08/multi-protocol-infrared-remote-library.html
|
||||
// Edited by Mitra to add new controller SANYO
|
||||
//
|
||||
// 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/
|
||||
//
|
||||
// JVC and Panasonic protocol added by Kristian Lauszus (Thanks to zenwheel and other people at the original blog post)
|
||||
// LG added by Darryl Smith (based on the JVC protocol)
|
||||
// Whynter A/C ARC-110WD added by Francesco Meschia
|
||||
//******************************************************************************
|
||||
|
||||
#ifndef IRremote_h
|
||||
#define IRremote_h
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// The ISR header contains several useful macros the user may wish to use
|
||||
//
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Supported IR protocols
|
||||
// Each protocol you include costs memory and, during decode, costs time
|
||||
// Disable (set to 0) all the protocols you do not need/want!
|
||||
//
|
||||
#define DECODE_RC5 1
|
||||
#define SEND_RC5 1
|
||||
|
||||
#define DECODE_RC6 1
|
||||
#define SEND_RC6 1
|
||||
|
||||
#define DECODE_NEC 1
|
||||
#define SEND_NEC 1
|
||||
|
||||
#define DECODE_SONY 1
|
||||
#define SEND_SONY 1
|
||||
|
||||
#define DECODE_PANASONIC 1
|
||||
#define SEND_PANASONIC 1
|
||||
|
||||
#define DECODE_JVC 1
|
||||
#define SEND_JVC 1
|
||||
|
||||
#define DECODE_SAMSUNG 1
|
||||
#define SEND_SAMSUNG 1
|
||||
|
||||
#define DECODE_WHYNTER 1
|
||||
#define SEND_WHYNTER 1
|
||||
|
||||
#define DECODE_AIWA_RC_T501 1
|
||||
#define SEND_AIWA_RC_T501 1
|
||||
|
||||
#define DECODE_LG 1
|
||||
#define SEND_LG 1
|
||||
|
||||
#define DECODE_SANYO 1
|
||||
#define SEND_SANYO 0 // NOT WRITTEN
|
||||
|
||||
#define DECODE_MITSUBISHI 1
|
||||
#define SEND_MITSUBISHI 0 // NOT WRITTEN
|
||||
|
||||
#define DECODE_DISH 0 // NOT WRITTEN
|
||||
#define SEND_DISH 1
|
||||
|
||||
#define DECODE_SHARP 0 // NOT WRITTEN
|
||||
#define SEND_SHARP 1
|
||||
|
||||
#define DECODE_DENON 1
|
||||
#define SEND_DENON 1
|
||||
|
||||
#define DECODE_PRONTO 0 // This function doe not logically make sense
|
||||
#define SEND_PRONTO 1
|
||||
|
||||
#define DECODE_LEGO_PF 0 // NOT WRITTEN
|
||||
#define SEND_LEGO_PF 1
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// When sending a Pronto code we request to send either the "once" code
|
||||
// or the "repeat" code
|
||||
// If the code requested does not exist we can request to fallback on the
|
||||
// other code (the one we did not explicitly request)
|
||||
//
|
||||
// I would suggest that "fallback" will be the standard calling method
|
||||
// The last paragraph on this page discusses the rationale of this idea:
|
||||
// http://www.remotecentral.com/features/irdisp2.htm
|
||||
//
|
||||
#define PRONTO_ONCE false
|
||||
#define PRONTO_REPEAT true
|
||||
#define PRONTO_FALLBACK true
|
||||
#define PRONTO_NOFALLBACK false
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// An enumerated list of all supported formats
|
||||
// You do NOT need to remove entries from this list when disabling protocols!
|
||||
//
|
||||
typedef
|
||||
enum {
|
||||
UNKNOWN = -1,
|
||||
UNUSED = 0,
|
||||
RC5,
|
||||
RC6,
|
||||
NEC,
|
||||
SONY,
|
||||
PANASONIC,
|
||||
JVC,
|
||||
SAMSUNG,
|
||||
WHYNTER,
|
||||
AIWA_RC_T501,
|
||||
LG,
|
||||
SANYO,
|
||||
MITSUBISHI,
|
||||
DISH,
|
||||
SHARP,
|
||||
DENON,
|
||||
PRONTO,
|
||||
LEGO_PF,
|
||||
}
|
||||
decode_type_t;
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Set DEBUG to 1 for lots of lovely debug output
|
||||
//
|
||||
#define DEBUG 0
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Debug directives
|
||||
//
|
||||
#if DEBUG
|
||||
# define DBG_PRINT(...) Serial.print(__VA_ARGS__)
|
||||
# define DBG_PRINTLN(...) Serial.println(__VA_ARGS__)
|
||||
#else
|
||||
# define DBG_PRINT(...)
|
||||
# define DBG_PRINTLN(...)
|
||||
#endif
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Mark & Space matching functions
|
||||
//
|
||||
int MATCH (int measured, int desired) ;
|
||||
int MATCH_MARK (int measured_ticks, int desired_us) ;
|
||||
int MATCH_SPACE (int measured_ticks, int desired_us) ;
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Results returned from the decoder
|
||||
//
|
||||
class decode_results
|
||||
{
|
||||
public:
|
||||
decode_type_t decode_type; // UNKNOWN, NEC, SONY, RC5, ...
|
||||
unsigned int address; // Used by Panasonic & Sharp [16-bits]
|
||||
unsigned long value; // Decoded value [max 32-bits]
|
||||
int bits; // Number of bits in decoded value
|
||||
volatile unsigned int *rawbuf; // Raw intervals in 50uS ticks
|
||||
int rawlen; // Number of records in rawbuf
|
||||
int overflow; // true iff IR raw code too long
|
||||
};
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Decoded value for NEC when a repeat code is received
|
||||
//
|
||||
#define REPEAT 0xFFFFFFFF
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Main class for receiving IR
|
||||
//
|
||||
class IRrecv
|
||||
{
|
||||
public:
|
||||
IRrecv () ;
|
||||
IRrecv (int recvpin) ;
|
||||
IRrecv (int recvpin, int blinkpin);
|
||||
|
||||
void blink13 (int blinkflag) ;
|
||||
int decode (decode_results *results) ;
|
||||
void enableIRIn ( ) ;
|
||||
bool isIdle ( ) ;
|
||||
void resume ( ) ;
|
||||
void disableIR ( ) ;
|
||||
void setRecvpin (int recvpin) ;
|
||||
|
||||
private:
|
||||
long decodeHash (decode_results *results) ;
|
||||
int compare (unsigned int oldval, unsigned int newval) ;
|
||||
|
||||
//......................................................................
|
||||
# if (DECODE_RC5 || DECODE_RC6)
|
||||
// This helper function is shared by RC5 and RC6
|
||||
int getRClevel (decode_results *results, int *offset, int *used, int t1) ;
|
||||
# endif
|
||||
# if DECODE_RC5
|
||||
bool decodeRC5 (decode_results *results) ;
|
||||
# endif
|
||||
# if DECODE_RC6
|
||||
bool decodeRC6 (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_NEC
|
||||
bool decodeNEC (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_SONY
|
||||
bool decodeSony (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_PANASONIC
|
||||
bool decodePanasonic (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_JVC
|
||||
bool decodeJVC (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_SAMSUNG
|
||||
bool decodeSAMSUNG (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_WHYNTER
|
||||
bool decodeWhynter (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_AIWA_RC_T501
|
||||
bool decodeAiwaRCT501 (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_LG
|
||||
bool decodeLG (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_SANYO
|
||||
bool decodeSanyo (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_MITSUBISHI
|
||||
bool decodeMitsubishi (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_DISH
|
||||
bool decodeDish (decode_results *results) ; // NOT WRITTEN
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_SHARP
|
||||
bool decodeSharp (decode_results *results) ; // NOT WRITTEN
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_DENON
|
||||
bool decodeDenon (decode_results *results) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if DECODE_LEGO_PF
|
||||
bool decodeLegoPowerFunctions (decode_results *results) ;
|
||||
# endif
|
||||
} ;
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Main class for sending IR
|
||||
//
|
||||
class IRsend
|
||||
{
|
||||
public:
|
||||
IRsend () { }
|
||||
|
||||
void custom_delay_usec (unsigned long uSecs);
|
||||
void enableIROut (int khz) ;
|
||||
void mark (unsigned int usec) ;
|
||||
void space (unsigned int usec) ;
|
||||
void sendRaw (const unsigned int buf[], unsigned int len, unsigned int hz) ;
|
||||
|
||||
//......................................................................
|
||||
# if SEND_RC5
|
||||
void sendRC5 (unsigned long data, int nbits) ;
|
||||
# endif
|
||||
# if SEND_RC6
|
||||
void sendRC6 (unsigned long data, int nbits) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_NEC
|
||||
void sendNEC (unsigned long data, int nbits) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_SONY
|
||||
void sendSony (unsigned long data, int nbits) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_PANASONIC
|
||||
void sendPanasonic (unsigned int address, unsigned long data) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_JVC
|
||||
// JVC does NOT repeat by sending a separate code (like NEC does).
|
||||
// The JVC protocol repeats by skipping the header.
|
||||
// To send a JVC repeat signal, send the original code value
|
||||
// and set 'repeat' to true
|
||||
void sendJVC (unsigned long data, int nbits, bool repeat) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_SAMSUNG
|
||||
void sendSAMSUNG (unsigned long data, int nbits) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_WHYNTER
|
||||
void sendWhynter (unsigned long data, int nbits) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_AIWA_RC_T501
|
||||
void sendAiwaRCT501 (int code) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_LG
|
||||
void sendLG (unsigned long data, int nbits) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_SANYO
|
||||
void sendSanyo ( ) ; // NOT WRITTEN
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_MISUBISHI
|
||||
void sendMitsubishi ( ) ; // NOT WRITTEN
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_DISH
|
||||
void sendDISH (unsigned long data, int nbits) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_SHARP
|
||||
void sendSharpRaw (unsigned long data, int nbits) ;
|
||||
void sendSharp (unsigned int address, unsigned int command) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_DENON
|
||||
void sendDenon (unsigned long data, int nbits) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_PRONTO
|
||||
void sendPronto (char* code, bool repeat, bool fallback) ;
|
||||
# endif
|
||||
//......................................................................
|
||||
# if SEND_LEGO_PF
|
||||
void sendLegoPowerFunctions (uint16_t data, bool repeat = true) ;
|
||||
# endif
|
||||
} ;
|
||||
|
||||
#endif
|
||||
113
Firmware/Source/grovepi/src/IRremoteInt.h
Normal file
113
Firmware/Source/grovepi/src/IRremoteInt.h
Normal file
|
|
@ -0,0 +1,113 @@
|
|||
//******************************************************************************
|
||||
// IRremote
|
||||
// Version 2.0.1 June, 2015
|
||||
// Copyright 2009 Ken Shirriff
|
||||
// For details, see http://arcfn.com/2009/08/multi-protocol-infrared-remote-library.html
|
||||
//
|
||||
// Modified by Paul Stoffregen <paul@pjrc.com> to support other boards and timers
|
||||
//
|
||||
// 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/
|
||||
//
|
||||
// JVC and Panasonic protocol added by Kristian Lauszus (Thanks to zenwheel and other people at the original blog post)
|
||||
// Whynter A/C ARC-110WD added by Francesco Meschia
|
||||
//******************************************************************************
|
||||
|
||||
#ifndef IRremoteint_h
|
||||
#define IRremoteint_h
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Include the right Arduino header
|
||||
//
|
||||
#if defined(ARDUINO) && (ARDUINO >= 100)
|
||||
# include <Arduino.h>
|
||||
#else
|
||||
# if !defined(IRPRONTO)
|
||||
# include <WProgram.h>
|
||||
# endif
|
||||
#endif
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// This handles definition and access to global variables
|
||||
//
|
||||
#ifdef IR_GLOBAL
|
||||
# define EXTERN
|
||||
#else
|
||||
# define EXTERN extern
|
||||
#endif
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Information for the Interrupt Service Routine
|
||||
//
|
||||
#define RAWBUF 101 // Maximum length of raw duration buffer
|
||||
|
||||
typedef
|
||||
struct {
|
||||
// The fields are ordered to reduce memory over caused by struct-padding
|
||||
uint8_t rcvstate; // State Machine state
|
||||
uint8_t recvpin; // Pin connected to IR data from detector
|
||||
uint8_t blinkpin;
|
||||
uint8_t blinkflag; // true -> enable blinking of pin on IR processing
|
||||
uint8_t rawlen; // counter of entries in rawbuf
|
||||
unsigned int timer; // State timer, counts 50uS ticks.
|
||||
unsigned int rawbuf[RAWBUF]; // raw data
|
||||
uint8_t overflow; // Raw buffer overflow occurred
|
||||
}
|
||||
irparams_t;
|
||||
|
||||
// ISR State-Machine : Receiver States
|
||||
#define STATE_IDLE 2
|
||||
#define STATE_MARK 3
|
||||
#define STATE_SPACE 4
|
||||
#define STATE_STOP 5
|
||||
#define STATE_OVERFLOW 6
|
||||
|
||||
// Allow all parts of the code access to the ISR data
|
||||
// NB. The data can be changed by the ISR at any time, even mid-function
|
||||
// Therefore we declare it as "volatile" to stop the compiler/CPU caching it
|
||||
EXTERN volatile irparams_t irparams;
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Defines for setting and clearing register bits
|
||||
//
|
||||
#ifndef cbi
|
||||
# define cbi(sfr, bit) (_SFR_BYTE(sfr) &= ~_BV(bit))
|
||||
#endif
|
||||
|
||||
#ifndef sbi
|
||||
# define sbi(sfr, bit) (_SFR_BYTE(sfr) |= _BV(bit))
|
||||
#endif
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Pulse parms are ((X*50)-100) for the Mark and ((X*50)+100) for the Space.
|
||||
// First MARK is the one after the long gap
|
||||
// Pulse parameters in uSec
|
||||
//
|
||||
|
||||
// Due to sensor lag, when received, Marks tend to be 100us too long and
|
||||
// Spaces tend to be 100us too short
|
||||
#define MARK_EXCESS 100
|
||||
|
||||
// Upper and Lower percentage tolerances in measurements
|
||||
#define TOLERANCE 25
|
||||
#define LTOL (1.0 - (TOLERANCE/100.))
|
||||
#define UTOL (1.0 + (TOLERANCE/100.))
|
||||
|
||||
// Minimum gap between IR transmissions
|
||||
#define _GAP 5000
|
||||
#define GAP_TICKS (_GAP/USECPERTICK)
|
||||
|
||||
#define TICKS_LOW(us) ((int)(((us)*LTOL/USECPERTICK)))
|
||||
#define TICKS_HIGH(us) ((int)(((us)*UTOL/USECPERTICK + 1)))
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// IR detector output is active low
|
||||
//
|
||||
#define MARK 0
|
||||
#define SPACE 1
|
||||
|
||||
// All board specific stuff has been moved to its own file, included here.
|
||||
#include "boarddefs.h"
|
||||
|
||||
#endif
|
||||
205
Firmware/Source/grovepi/src/TM1637.cpp
Normal file
205
Firmware/Source/grovepi/src/TM1637.cpp
Normal file
|
|
@ -0,0 +1,205 @@
|
|||
// Author: avishorp@gmail.com
|
||||
//
|
||||
// This library is free software; you can redistribute it and/or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
// License as published by the Free Software Foundation; either
|
||||
// version 2.1 of the License, or (at your option) any later version.
|
||||
//
|
||||
// This library is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
// Lesser General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Lesser General Public
|
||||
// License along with this library; if not, write to the Free Software
|
||||
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
|
||||
extern "C" {
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <inttypes.h>
|
||||
}
|
||||
|
||||
#include "TM1637.h"
|
||||
#include <Arduino.h>
|
||||
|
||||
#define TM1637_I2C_COMM1 0x40
|
||||
#define TM1637_I2C_COMM2 0xC0
|
||||
#define TM1637_I2C_COMM3 0x80
|
||||
|
||||
//
|
||||
// A
|
||||
// ---
|
||||
// F | | B
|
||||
// -G-
|
||||
// E | | C
|
||||
// ---
|
||||
// D
|
||||
const uint8_t digitToSegment[] = {
|
||||
//XGFEDCBA <- one bit for each segment above and X is the colon
|
||||
0b00111111, // 0
|
||||
0b00000110, // 1
|
||||
0b01011011, // 2
|
||||
0b01001111, // 3
|
||||
0b01100110, // 4
|
||||
0b01101101, // 5
|
||||
0b01111101, // 6
|
||||
0b00000111, // 7
|
||||
0b01111111, // 8
|
||||
0b01101111, // 9
|
||||
0b01110111, // A
|
||||
0b01111100, // b
|
||||
0b00111001, // C
|
||||
0b01011110, // d
|
||||
0b01111001, // E
|
||||
0b01110001 // F
|
||||
};
|
||||
|
||||
|
||||
//TM1637::TM1637(uint8_t pinClk, uint8_t pinDIO)
|
||||
void TM1637::begin(uint8_t pinClk, uint8_t pinDIO)
|
||||
{
|
||||
// Copy the pin numbers
|
||||
m_pinClk = pinClk;
|
||||
m_pinDIO = pinDIO;
|
||||
|
||||
// Set the pin direction and default value.
|
||||
// Both pins are set as inputs, allowing the pull-up resistors to pull them up
|
||||
pinMode(m_pinClk, INPUT);
|
||||
pinMode(m_pinDIO, INPUT);
|
||||
digitalWrite(m_pinClk, LOW);
|
||||
digitalWrite(m_pinDIO, LOW);
|
||||
}
|
||||
|
||||
void TM1637::setBrightness(uint8_t brightness)
|
||||
{
|
||||
m_brightness = constrain(brightness, 0, 7) + 8;
|
||||
}
|
||||
|
||||
void TM1637::setSegments(const uint8_t segments[], uint8_t length, uint8_t pos)
|
||||
{
|
||||
// Write COMM1
|
||||
start();
|
||||
writeByte(TM1637_I2C_COMM1);
|
||||
stop();
|
||||
|
||||
// Write COMM2 + first digit address
|
||||
start();
|
||||
writeByte(TM1637_I2C_COMM2 + (pos & 0x03));
|
||||
|
||||
// Write the data bytes
|
||||
for (uint8_t k = 0; k < length; k++)
|
||||
writeByte(segments[k]);
|
||||
|
||||
stop();
|
||||
|
||||
// Write COMM3 + brightness
|
||||
start();
|
||||
writeByte(TM1637_I2C_COMM3 + m_brightness);
|
||||
stop();
|
||||
}
|
||||
|
||||
void TM1637::showNumberDec(int num, bool leading_zero, uint8_t length, uint8_t pos, bool colon)
|
||||
{
|
||||
uint8_t digits[4];
|
||||
const static int divisors[] = { 1, 10, 100, 1000 };
|
||||
bool leading = true;
|
||||
|
||||
for (int8_t k = 0; k < 4; k++) {
|
||||
int divisor = divisors[4 - 1 - k];
|
||||
int d = num / divisor;
|
||||
|
||||
if (d == 0) {
|
||||
if (leading_zero || !leading || (k == 3))
|
||||
digits[k] = encodeDigit(d);
|
||||
else
|
||||
digits[k] = 0;
|
||||
}
|
||||
else {
|
||||
digits[k] = encodeDigit(d);
|
||||
num -= d * divisor;
|
||||
leading = false;
|
||||
}
|
||||
if (colon && k == 1) {
|
||||
digits[k] |= 0x80;
|
||||
}
|
||||
}
|
||||
|
||||
setSegments(digits + (4 - length), length, pos);
|
||||
}
|
||||
|
||||
void TM1637::bitDelay()
|
||||
{
|
||||
delayMicroseconds(50);
|
||||
}
|
||||
|
||||
void TM1637::start()
|
||||
{
|
||||
pinMode(m_pinDIO, OUTPUT);
|
||||
bitDelay();
|
||||
}
|
||||
|
||||
void TM1637::stop()
|
||||
{
|
||||
pinMode(m_pinDIO, OUTPUT);
|
||||
bitDelay();
|
||||
pinMode(m_pinClk, INPUT);
|
||||
bitDelay();
|
||||
pinMode(m_pinDIO, INPUT);
|
||||
bitDelay();
|
||||
}
|
||||
|
||||
bool TM1637::writeByte(uint8_t b)
|
||||
{
|
||||
uint8_t data = b;
|
||||
|
||||
// 8 Data Bits
|
||||
for (uint8_t i = 0; i < 8; i++) {
|
||||
// CLK low
|
||||
pinMode(m_pinClk, OUTPUT);
|
||||
bitDelay();
|
||||
|
||||
// Set data bit
|
||||
if (data & 0x01)
|
||||
pinMode(m_pinDIO, INPUT);
|
||||
else
|
||||
pinMode(m_pinDIO, OUTPUT);
|
||||
|
||||
bitDelay();
|
||||
|
||||
// CLK high
|
||||
pinMode(m_pinClk, INPUT);
|
||||
bitDelay();
|
||||
data = data >> 1;
|
||||
}
|
||||
|
||||
// Wait for acknowledge
|
||||
// CLK to zero
|
||||
pinMode(m_pinClk, OUTPUT);
|
||||
pinMode(m_pinDIO, INPUT);
|
||||
bitDelay();
|
||||
|
||||
// CLK to high
|
||||
pinMode(m_pinClk, INPUT);
|
||||
bitDelay();
|
||||
uint8_t ack = digitalRead(m_pinDIO);
|
||||
if (ack == 0)
|
||||
pinMode(m_pinDIO, OUTPUT);
|
||||
|
||||
bitDelay();
|
||||
pinMode(m_pinClk, OUTPUT);
|
||||
bitDelay();
|
||||
|
||||
return ack;
|
||||
}
|
||||
|
||||
uint8_t TM1637::encodeDigit(uint8_t digit)
|
||||
{
|
||||
return digitToSegment[digit & 0x0f];
|
||||
}
|
||||
|
||||
// Has this instance been initialised?
|
||||
bool TM1637::ready()
|
||||
{
|
||||
return m_pinClk != 0;
|
||||
}
|
||||
99
Firmware/Source/grovepi/src/TM1637.h
Normal file
99
Firmware/Source/grovepi/src/TM1637.h
Normal file
|
|
@ -0,0 +1,99 @@
|
|||
// Author: avishorp@gmail.com
|
||||
//
|
||||
// This library is free software; you can redistribute it and/or
|
||||
// modify it under the terms of the GNU Lesser General Public
|
||||
// License as published by the Free Software Foundation; either
|
||||
// version 2.1 of the License, or (at your option) any later version.
|
||||
//
|
||||
// This library is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
// Lesser General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Lesser General Public
|
||||
// License along with this library; if not, write to the Free Software
|
||||
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
|
||||
#ifndef __TM1637__
|
||||
#define __TM1637__
|
||||
|
||||
#include <inttypes.h>
|
||||
|
||||
#define SEG_A 0b00000001
|
||||
#define SEG_B 0b00000010
|
||||
#define SEG_C 0b00000100
|
||||
#define SEG_D 0b00001000
|
||||
#define SEG_E 0b00010000
|
||||
#define SEG_F 0b00100000
|
||||
#define SEG_G 0b01000000
|
||||
|
||||
class TM1637 {
|
||||
|
||||
public:
|
||||
//! Initialize a TM1637 object, setting the clock and data pins.
|
||||
//!
|
||||
//! @param pinClk - The number of the digital pin connected to the clock pin of the module
|
||||
//! @param pinDIO - The number of the digital pin connected to the DIO pin of the module
|
||||
//TM1637(uint8_t pinClk, uint8_t pinDIO);
|
||||
void begin(uint8_t pinClk, uint8_t pinDIO);
|
||||
|
||||
//! Sets the brightness of the display.
|
||||
//!
|
||||
//! The setting takes effect when a command is given to change the data being displayed.
|
||||
//!
|
||||
//! @param brightness A number from 0 (lowest brightness) to 7 (highest brightness)
|
||||
void setBrightness(uint8_t brightness);
|
||||
|
||||
//! Display arbitrary data on the module
|
||||
//!
|
||||
//! This function receives raw segment values as input and displays them. The segment data
|
||||
//! is given as a byte array, each byte corresponding to a single digit. Within each byte,
|
||||
//! bit 0 is segment A, bit 1 is segment B etc.
|
||||
//! The function may either set the entire display or any desirable part on its own. The first
|
||||
//! digit is given by the @ref pos argument with 0 being the leftmost digit. The @ref length
|
||||
//! argument is the number of digits to be set. Other digits are not affected.
|
||||
//!
|
||||
//! @param segments An array of size @ref length containing the raw segment values
|
||||
//! @param length The number of digits to be modified
|
||||
//! @param pos The position from which to start the modification (0 - leftmost, 3 - rightmost)
|
||||
void setSegments(const uint8_t segments[], uint8_t length = 4, uint8_t pos = 0);
|
||||
|
||||
//! Displays a decimal number
|
||||
//!
|
||||
//! Displays the given argument as a decimal number
|
||||
//!
|
||||
//! @param num The number to be shown
|
||||
//! @param leading_zero When true, leading zeros are displayed. Otherwise unnecessary digits are blank
|
||||
//! @param length The number of digits to set. The user must ensure that the number to be shown
|
||||
//! fits to the number of digits requested (for example, if two digits are to be displayed,
|
||||
//! the number must be between 0 to 99)
|
||||
//! @param pos The position least significant digit (0 - leftmost, 3 - rightmost)
|
||||
//! @param colon When true, displays two dots between the 2nd and 3rd segments
|
||||
void showNumberDec(int num, bool leading_zero = false, uint8_t length = 4, uint8_t pos = 0, bool colon = false);
|
||||
|
||||
//! Translate a single digit into 7 segment code
|
||||
//!
|
||||
//! The method accepts a number between 0 - 15 and converts it to the
|
||||
//! code required to display the number on a 7 segment display.
|
||||
//! Numbers between 10-15 are converted to hexadecimal digits (A-F)
|
||||
//!
|
||||
//! @param digit A number between 0 to 15
|
||||
//! @return A code representing the 7 segment image of the digit (LSB - segment A;
|
||||
//! bit 6 - segment G; bit 7 - always zero)
|
||||
uint8_t encodeDigit(uint8_t digit);
|
||||
|
||||
bool ready();
|
||||
|
||||
protected:
|
||||
void bitDelay();
|
||||
void start();
|
||||
void stop();
|
||||
bool writeByte(uint8_t b);
|
||||
|
||||
private:
|
||||
uint8_t m_pinClk;
|
||||
uint8_t m_pinDIO;
|
||||
uint8_t m_brightness;
|
||||
};
|
||||
|
||||
#endif // __TM1637__
|
||||
208
Firmware/Source/grovepi/src/TimerOne.cpp
Normal file
208
Firmware/Source/grovepi/src/TimerOne.cpp
Normal file
|
|
@ -0,0 +1,208 @@
|
|||
/*
|
||||
* Interrupt and PWM utilities for 16 bit Timer1 on ATmega168/328
|
||||
* Original code by Jesse Tane for http://labs.ideo.com August 2008
|
||||
* Modified March 2009 by Jérôme Despatis and Jesse Tane for ATmega328 support
|
||||
* Modified June 2009 by Michael Polli and Jesse Tane to fix a bug in setPeriod() which caused the timer to stop
|
||||
* Modified June 2011 by Lex Talionis to add a function to read the timer
|
||||
* Modified Oct 2011 by Andrew Richards to avoid certain problems:
|
||||
* - Add (long) assignments and casts to TimerOne::read() to ensure calculations involving tmp, ICR1 and TCNT1 aren't truncated
|
||||
* - Ensure 16 bit registers accesses are atomic - run with interrupts disabled when accessing
|
||||
* - Remove global enable of interrupts (sei())- could be running within an interrupt routine)
|
||||
* - Disable interrupts whilst TCTN1 == 0. Datasheet vague on this, but experiment shows that overflow interrupt
|
||||
* flag gets set whilst TCNT1 == 0, resulting in a phantom interrupt. Could just set to 1, but gets inaccurate
|
||||
* at very short durations
|
||||
* - startBottom() added to start counter at 0 and handle all interrupt enabling.
|
||||
* - start() amended to enable interrupts
|
||||
* - restart() amended to point at startBottom()
|
||||
* Modiied 7:26 PM Sunday, October 09, 2011 by Lex Talionis
|
||||
* - renamed start() to resume() to reflect it's actual role
|
||||
* - renamed startBottom() to start(). This breaks some old code that expects start to continue counting where it left off
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* See Google Code project http://code.google.com/p/arduino-timerone/ for latest
|
||||
*/
|
||||
#ifndef TIMERONE_cpp
|
||||
#define TIMERONE_cpp
|
||||
|
||||
#include "TimerOne.h"
|
||||
|
||||
TimerOne Timer1; // preinstatiate
|
||||
|
||||
ISR(TIMER1_OVF_vect) // interrupt service routine that wraps a user defined function supplied by attachInterrupt
|
||||
{
|
||||
Timer1.isrCallback();
|
||||
}
|
||||
|
||||
|
||||
void TimerOne::initialize(long microseconds)
|
||||
{
|
||||
TCCR1A = 0; // clear control register A
|
||||
TCCR1B = _BV(WGM13); // set mode 8: phase and frequency correct pwm, stop the timer
|
||||
setPeriod(microseconds);
|
||||
}
|
||||
|
||||
|
||||
void TimerOne::setPeriod(long microseconds) // AR modified for atomic access
|
||||
{
|
||||
|
||||
long cycles = (F_CPU / 10000) * microseconds; // the counter runs backwards after TOP, interrupt is at BOTTOM so divide microseconds by 2
|
||||
if(cycles < RESOLUTION) clockSelectBits = _BV(CS10); // no prescale, full xtal
|
||||
else if((cycles >>= 3) < RESOLUTION) clockSelectBits = _BV(CS11); // prescale by /8
|
||||
else if((cycles >>= 3) < RESOLUTION) clockSelectBits = _BV(CS11) | _BV(CS10); // prescale by /64
|
||||
else if((cycles >>= 2) < RESOLUTION) clockSelectBits = _BV(CS12); // prescale by /256
|
||||
else if((cycles >>= 2) < RESOLUTION) clockSelectBits = _BV(CS12) | _BV(CS10); // prescale by /1024
|
||||
else cycles = RESOLUTION - 1, clockSelectBits = _BV(CS12) | _BV(CS10); // request was out of bounds, set as maximum
|
||||
|
||||
oldSREG = SREG;
|
||||
cli(); // Disable interrupts for 16 bit register access
|
||||
ICR1 = pwmPeriod = cycles; // ICR1 is TOP in p & f correct pwm mode
|
||||
SREG = oldSREG;
|
||||
|
||||
TCCR1B &= ~(_BV(CS10) | _BV(CS11) | _BV(CS12));
|
||||
TCCR1B |= clockSelectBits; // reset clock select register, and starts the clock
|
||||
}
|
||||
|
||||
void TimerOne::setPwmDuty(char pin, int duty)
|
||||
{
|
||||
unsigned long dutyCycle = pwmPeriod;
|
||||
|
||||
dutyCycle *= duty;
|
||||
dutyCycle >>= 10;
|
||||
|
||||
oldSREG = SREG;
|
||||
cli();
|
||||
if(pin == 1 || pin == 9) OCR1A = dutyCycle;
|
||||
else if(pin == 2 || pin == 10) OCR1B = dutyCycle;
|
||||
SREG = oldSREG;
|
||||
}
|
||||
|
||||
void TimerOne::pwm(char pin, int duty, long microseconds) // expects duty cycle to be 10 bit (1024)
|
||||
{
|
||||
if(microseconds > 0) setPeriod(microseconds);
|
||||
if(pin == 1 || pin == 9) {
|
||||
DDRB |= _BV(PORTB1); // sets data direction register for pwm output pin
|
||||
TCCR1A |= _BV(COM1A1); // activates the output pin
|
||||
}
|
||||
else if(pin == 2 || pin == 10) {
|
||||
DDRB |= _BV(PORTB2);
|
||||
TCCR1A |= _BV(COM1B1);
|
||||
}
|
||||
setPwmDuty(pin, duty);
|
||||
resume(); // Lex - make sure the clock is running. We don't want to restart the count, in case we are starting the second WGM
|
||||
// and the first one is in the middle of a cycle
|
||||
}
|
||||
|
||||
void TimerOne::disablePwm(char pin)
|
||||
{
|
||||
if(pin == 1 || pin == 9) TCCR1A &= ~_BV(COM1A1); // clear the bit that enables pwm on PB1
|
||||
else if(pin == 2 || pin == 10) TCCR1A &= ~_BV(COM1B1); // clear the bit that enables pwm on PB2
|
||||
}
|
||||
|
||||
void TimerOne::attachInterrupt(void (*isr)(), long microseconds)
|
||||
{
|
||||
if(microseconds > 0) setPeriod(microseconds);
|
||||
isrCallback = isr; // register the user's callback with the real ISR
|
||||
TIMSK1 = _BV(TOIE1); // sets the timer overflow interrupt enable bit
|
||||
// AR - remove sei() - might be running with interrupts disabled (eg inside an ISR), so leave unchanged
|
||||
// sei(); // ensures that interrupts are globally enabled
|
||||
resume();
|
||||
}
|
||||
|
||||
void TimerOne::detachInterrupt()
|
||||
{
|
||||
TIMSK1 &= ~_BV(TOIE1); // clears the timer overflow interrupt enable bit
|
||||
}
|
||||
|
||||
void TimerOne::resume() // AR suggested
|
||||
{
|
||||
TCCR1B |= clockSelectBits;
|
||||
}
|
||||
|
||||
void TimerOne::restart() // Depricated - Public interface to start at zero - Lex 10/9/2011
|
||||
{
|
||||
start();
|
||||
}
|
||||
|
||||
void TimerOne::start() // AR addition, renamed by Lex to reflect it's actual role
|
||||
{
|
||||
unsigned int tcnt1;
|
||||
|
||||
TIMSK1 &= ~_BV(TOIE1); // AR added
|
||||
GTCCR |= _BV(PSRSYNC); // AR added - reset prescaler (NB: shared with all 16 bit timers);
|
||||
|
||||
oldSREG = SREG; // AR - save status register
|
||||
cli(); // AR - Disable interrupts
|
||||
TCNT1 = 0;
|
||||
SREG = oldSREG; // AR - Restore status register
|
||||
|
||||
do { // Nothing -- wait until timer moved on from zero - otherwise get a phantom interrupt
|
||||
oldSREG = SREG;
|
||||
cli();
|
||||
tcnt1 = TCNT1;
|
||||
SREG = oldSREG;
|
||||
} while (tcnt1==0);
|
||||
|
||||
// TIFR1 = 0xff; // AR - Clear interrupt flags
|
||||
// TIMSK1 = _BV(TOIE1); // sets the timer overflow interrupt enable bit
|
||||
}
|
||||
|
||||
void TimerOne::stop()
|
||||
{
|
||||
TCCR1B &= ~(_BV(CS10) | _BV(CS11) | _BV(CS12)); // clears all clock selects bits
|
||||
}
|
||||
|
||||
unsigned long TimerOne::read() //returns the value of the timer in microseconds
|
||||
{ //rember! phase and freq correct mode counts up to then down again
|
||||
unsigned long tmp; // AR amended to hold more than 65536 (could be nearly double this)
|
||||
unsigned int tcnt1; // AR added
|
||||
|
||||
oldSREG= SREG;
|
||||
cli();
|
||||
tmp=TCNT1;
|
||||
SREG = oldSREG;
|
||||
|
||||
char scale=0;
|
||||
switch (clockSelectBits)
|
||||
{
|
||||
case 1:// no prescalse
|
||||
scale=0;
|
||||
break;
|
||||
case 2:// x8 prescale
|
||||
scale=3;
|
||||
break;
|
||||
case 3:// x64
|
||||
scale=6;
|
||||
break;
|
||||
case 4:// x256
|
||||
scale=8;
|
||||
break;
|
||||
case 5:// x1024
|
||||
scale=10;
|
||||
break;
|
||||
}
|
||||
|
||||
do { // Nothing -- max delay here is ~1023 cycles. AR modified
|
||||
oldSREG = SREG;
|
||||
cli();
|
||||
tcnt1 = TCNT1;
|
||||
SREG = oldSREG;
|
||||
} while (tcnt1==tmp); //if the timer has not ticked yet
|
||||
|
||||
//if we are counting down add the top value to how far we have counted down
|
||||
tmp = ( (tcnt1>tmp) ? (tmp) : (long)(ICR1-tcnt1)+(long)ICR1 ); // AR amended to add casts and reuse previous TCNT1
|
||||
return ((tmp*1000L)/(F_CPU /1000L))<<scale;
|
||||
}
|
||||
|
||||
#endif
|
||||
70
Firmware/Source/grovepi/src/TimerOne.h
Normal file
70
Firmware/Source/grovepi/src/TimerOne.h
Normal file
|
|
@ -0,0 +1,70 @@
|
|||
/*
|
||||
* Interrupt and PWM utilities for 16 bit Timer1 on ATmega168/328
|
||||
* Original code by Jesse Tane for http://labs.ideo.com August 2008
|
||||
* Modified March 2009 by Jérôme Despatis and Jesse Tane for ATmega328 support
|
||||
* Modified June 2009 by Michael Polli and Jesse Tane to fix a bug in setPeriod() which caused the timer to stop
|
||||
* Modified June 2011 by Lex Talionis to add a function to read the timer
|
||||
* Modified Oct 2011 by Andrew Richards to avoid certain problems:
|
||||
* - Add (long) assignments and casts to TimerOne::read() to ensure calculations involving tmp, ICR1 and TCNT1 aren't truncated
|
||||
* - Ensure 16 bit registers accesses are atomic - run with interrupts disabled when accessing
|
||||
* - Remove global enable of interrupts (sei())- could be running within an interrupt routine)
|
||||
* - Disable interrupts whilst TCTN1 == 0. Datasheet vague on this, but experiment shows that overflow interrupt
|
||||
* flag gets set whilst TCNT1 == 0, resulting in a phantom interrupt. Could just set to 1, but gets inaccurate
|
||||
* at very short durations
|
||||
* - startBottom() added to start counter at 0 and handle all interrupt enabling.
|
||||
* - start() amended to enable interrupts
|
||||
* - restart() amended to point at startBottom()
|
||||
* Modiied 7:26 PM Sunday, October 09, 2011 by Lex Talionis
|
||||
* - renamed start() to resume() to reflect it's actual role
|
||||
* - renamed startBottom() to start(). This breaks some old code that expects start to continue counting where it left off
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* See Google Code project http://code.google.com/p/arduino-timerone/ for latest
|
||||
*/
|
||||
#ifndef TIMERONE_h
|
||||
#define TIMERONE_h
|
||||
|
||||
#include <avr/io.h>
|
||||
#include <avr/interrupt.h>
|
||||
|
||||
#define RESOLUTION 65536 // Timer1 is 16 bit
|
||||
|
||||
class TimerOne
|
||||
{
|
||||
public:
|
||||
|
||||
// properties
|
||||
unsigned int pwmPeriod;
|
||||
unsigned char clockSelectBits;
|
||||
char oldSREG; // To hold Status Register while ints disabled
|
||||
|
||||
// methods
|
||||
void initialize(long microseconds=1000000);
|
||||
void start();
|
||||
void stop();
|
||||
void restart();
|
||||
void resume();
|
||||
unsigned long read();
|
||||
void pwm(char pin, int duty, long microseconds=-1);
|
||||
void disablePwm(char pin);
|
||||
void attachInterrupt(void (*isr)(), long microseconds=-1);
|
||||
void detachInterrupt();
|
||||
void setPeriod(long microseconds);
|
||||
void setPwmDuty(char pin, int duty);
|
||||
void (*isrCallback)();
|
||||
};
|
||||
|
||||
extern TimerOne Timer1;
|
||||
#endif
|
||||
595
Firmware/Source/grovepi/src/boarddefs.h
Normal file
595
Firmware/Source/grovepi/src/boarddefs.h
Normal file
|
|
@ -0,0 +1,595 @@
|
|||
//******************************************************************************
|
||||
// IRremote
|
||||
// Version 2.0.1 June, 2015
|
||||
// Copyright 2009 Ken Shirriff
|
||||
// For details, see http://arcfn.com/2009/08/multi-protocol-infrared-remote-library.html
|
||||
|
||||
// This file contains all board specific information. It was previously contained within
|
||||
// IRremoteInt.h
|
||||
|
||||
// Modified by Paul Stoffregen <paul@pjrc.com> to support other boards and timers
|
||||
//
|
||||
// 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/
|
||||
//
|
||||
// JVC and Panasonic protocol added by Kristian Lauszus (Thanks to zenwheel and other people at the original blog post)
|
||||
// Whynter A/C ARC-110WD added by Francesco Meschia
|
||||
|
||||
// Sparkfun Pro Micro support by Alastair McCormack
|
||||
//******************************************************************************
|
||||
|
||||
#ifndef boarddefs_h
|
||||
#define boarddefs_h
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Defines for blinking the LED
|
||||
//
|
||||
|
||||
#if defined(CORE_LED0_PIN)
|
||||
# define BLINKLED CORE_LED0_PIN
|
||||
# define BLINKLED_ON() (digitalWrite(CORE_LED0_PIN, HIGH))
|
||||
# define BLINKLED_OFF() (digitalWrite(CORE_LED0_PIN, LOW))
|
||||
|
||||
#elif defined(__AVR_ATmega1280__) || defined(__AVR_ATmega2560__)
|
||||
# define BLINKLED 13
|
||||
# define BLINKLED_ON() (PORTB |= B10000000)
|
||||
# define BLINKLED_OFF() (PORTB &= B01111111)
|
||||
|
||||
#elif defined(__AVR_ATmega644P__) || defined(__AVR_ATmega644__)
|
||||
# define BLINKLED 0
|
||||
# define BLINKLED_ON() (PORTD |= B00000001)
|
||||
# define BLINKLED_OFF() (PORTD &= B11111110)
|
||||
|
||||
// No system LED on ESP32, disable blinking
|
||||
#elif defined(ESP32)
|
||||
# define BLINKLED 255
|
||||
# define BLINKLED_ON() 1
|
||||
# define BLINKLED_OFF() 1
|
||||
#else
|
||||
# define BLINKLED 13
|
||||
# define BLINKLED_ON() (PORTB |= B00100000)
|
||||
# define BLINKLED_OFF() (PORTB &= B11011111)
|
||||
#endif
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// CPU Frequency
|
||||
//
|
||||
#ifdef F_CPU
|
||||
# define SYSCLOCK F_CPU // main Arduino clock
|
||||
#else
|
||||
# define SYSCLOCK 16000000 // main Arduino clock
|
||||
#endif
|
||||
|
||||
// microseconds per clock interrupt tick
|
||||
#define USECPERTICK 50
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Define which timer to use
|
||||
//
|
||||
// Uncomment the timer you wish to use on your board.
|
||||
// If you are using another library which uses timer2, you have options to
|
||||
// switch IRremote to use a different timer.
|
||||
//
|
||||
|
||||
// Sparkfun Pro Micro
|
||||
#if defined(ARDUINO_AVR_PROMICRO)
|
||||
//#define IR_USE_TIMER1 // tx = pin 9
|
||||
#define IR_USE_TIMER3 // tx = pin 5
|
||||
//#define IR_USE_TIMER4_HS // tx = pin 5
|
||||
|
||||
// Arduino Mega
|
||||
#elif defined(__AVR_ATmega1280__) || defined(__AVR_ATmega2560__)
|
||||
//#define IR_USE_TIMER1 // tx = pin 11
|
||||
#define IR_USE_TIMER2 // tx = pin 9
|
||||
//#define IR_USE_TIMER3 // tx = pin 5
|
||||
//#define IR_USE_TIMER4 // tx = pin 6
|
||||
//#define IR_USE_TIMER5 // tx = pin 46
|
||||
|
||||
// Teensy 1.0
|
||||
#elif defined(__AVR_AT90USB162__)
|
||||
#define IR_USE_TIMER1 // tx = pin 17
|
||||
|
||||
// Teensy 2.0
|
||||
#elif defined(__AVR_ATmega32U4__)
|
||||
//#define IR_USE_TIMER1 // tx = pin 14
|
||||
//#define IR_USE_TIMER3 // tx = pin 9
|
||||
#define IR_USE_TIMER4_HS // tx = pin 10
|
||||
|
||||
// Teensy 3.0 / Teensy 3.1
|
||||
#elif defined(__MK20DX128__) || defined(__MK20DX256__) || defined(__MK64FX512__) || defined(__MK66FX1M0__)
|
||||
#define IR_USE_TIMER_CMT // tx = pin 5
|
||||
|
||||
// Teensy-LC
|
||||
#elif defined(__MKL26Z64__)
|
||||
#define IR_USE_TIMER_TPM1 // tx = pin 16
|
||||
|
||||
// Teensy++ 1.0 & 2.0
|
||||
#elif defined(__AVR_AT90USB646__) || defined(__AVR_AT90USB1286__)
|
||||
//#define IR_USE_TIMER1 // tx = pin 25
|
||||
#define IR_USE_TIMER2 // tx = pin 1
|
||||
//#define IR_USE_TIMER3 // tx = pin 16
|
||||
|
||||
// MightyCore - ATmega1284
|
||||
#elif defined(__AVR_ATmega1284__) || defined(__AVR_ATmega1284P__)
|
||||
//#define IR_USE_TIMER1 // tx = pin 13
|
||||
#define IR_USE_TIMER2 // tx = pin 14
|
||||
//#define IR_USE_TIMER3 // tx = pin 6
|
||||
|
||||
// MightyCore - ATmega164, ATmega324, ATmega644
|
||||
#elif defined(__AVR_ATmega644__) || defined(__AVR_ATmega644P__) \
|
||||
|| defined(__AVR_ATmega324P__) || defined(__AVR_ATmega324A__) \
|
||||
|| defined(__AVR_ATmega324PA__) || defined(__AVR_ATmega164A__) \
|
||||
|| defined(__AVR_ATmega164P__)
|
||||
//#define IR_USE_TIMER1 // tx = pin 13
|
||||
#define IR_USE_TIMER2 // tx = pin 14
|
||||
|
||||
//MegaCore - ATmega64, ATmega128
|
||||
#elif defined(__AVR_ATmega64__) || defined(__AVR_ATmega128__)
|
||||
#define IR_USE_TIMER1 // tx = pin 13
|
||||
|
||||
// MightyCore - ATmega8535, ATmega16, ATmega32
|
||||
#elif defined(__AVR_ATmega8535__) || defined(__AVR_ATmega16__) || defined(__AVR_ATmega32__)
|
||||
#define IR_USE_TIMER1 // tx = pin 13
|
||||
|
||||
// Atmega8
|
||||
#elif defined(__AVR_ATmega8__)
|
||||
#define IR_USE_TIMER1 // tx = pin 9
|
||||
|
||||
// ATtiny84
|
||||
#elif defined(__AVR_ATtiny84__)
|
||||
#define IR_USE_TIMER1 // tx = pin 6
|
||||
|
||||
//ATtiny85
|
||||
#elif defined(__AVR_ATtiny85__)
|
||||
#define IR_USE_TIMER_TINY0 // tx = pin 1
|
||||
|
||||
#elif defined(ESP32)
|
||||
#define IR_TIMER_USE_ESP32
|
||||
#else
|
||||
// Arduino Duemilanove, Diecimila, LilyPad, Mini, Fio, Nano, etc
|
||||
// ATmega48, ATmega88, ATmega168, ATmega328
|
||||
//#define IR_USE_TIMER1 // tx = pin 9
|
||||
#define IR_USE_TIMER2 // tx = pin 3
|
||||
|
||||
#endif
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Defines for Timer
|
||||
|
||||
//---------------------------------------------------------
|
||||
// Timer2 (8 bits)
|
||||
//
|
||||
#if defined(IR_USE_TIMER2)
|
||||
|
||||
#define TIMER_RESET
|
||||
#define TIMER_ENABLE_PWM (TCCR2A |= _BV(COM2B1))
|
||||
#define TIMER_DISABLE_PWM (TCCR2A &= ~(_BV(COM2B1)))
|
||||
#define TIMER_ENABLE_INTR (TIMSK2 = _BV(OCIE2A))
|
||||
#define TIMER_DISABLE_INTR (TIMSK2 = 0)
|
||||
#define TIMER_INTR_NAME TIMER2_COMPA_vect
|
||||
|
||||
#define TIMER_CONFIG_KHZ(val) ({ \
|
||||
const uint8_t pwmval = SYSCLOCK / 2000 / (val); \
|
||||
TCCR2A = _BV(WGM20); \
|
||||
TCCR2B = _BV(WGM22) | _BV(CS20); \
|
||||
OCR2A = pwmval; \
|
||||
OCR2B = pwmval / 3; \
|
||||
})
|
||||
|
||||
#define TIMER_COUNT_TOP (SYSCLOCK * USECPERTICK / 1000000)
|
||||
|
||||
//-----------------
|
||||
#if (TIMER_COUNT_TOP < 256)
|
||||
# define TIMER_CONFIG_NORMAL() ({ \
|
||||
TCCR2A = _BV(WGM21); \
|
||||
TCCR2B = _BV(CS20); \
|
||||
OCR2A = TIMER_COUNT_TOP; \
|
||||
TCNT2 = 0; \
|
||||
})
|
||||
#else
|
||||
# define TIMER_CONFIG_NORMAL() ({ \
|
||||
TCCR2A = _BV(WGM21); \
|
||||
TCCR2B = _BV(CS21); \
|
||||
OCR2A = TIMER_COUNT_TOP / 8; \
|
||||
TCNT2 = 0; \
|
||||
})
|
||||
#endif
|
||||
|
||||
//-----------------
|
||||
#if defined(CORE_OC2B_PIN)
|
||||
# define TIMER_PWM_PIN CORE_OC2B_PIN // Teensy
|
||||
#elif defined(__AVR_ATmega1280__) || defined(__AVR_ATmega2560__)
|
||||
# define TIMER_PWM_PIN 9 // Arduino Mega
|
||||
#elif defined(__AVR_ATmega1284__) || defined(__AVR_ATmega1284P__) \
|
||||
|| defined(__AVR_ATmega644__) || defined(__AVR_ATmega644P__) \
|
||||
|| defined(__AVR_ATmega324P__) || defined(__AVR_ATmega324A__) \
|
||||
|| defined(__AVR_ATmega324PA__) || defined(__AVR_ATmega164A__) \
|
||||
|| defined(__AVR_ATmega164P__)
|
||||
# define TIMER_PWM_PIN 14 // MightyCore
|
||||
#else
|
||||
# define TIMER_PWM_PIN 3 // Arduino Duemilanove, Diecimila, LilyPad, etc
|
||||
#endif // ATmega48, ATmega88, ATmega168, ATmega328
|
||||
|
||||
//---------------------------------------------------------
|
||||
// Timer1 (16 bits)
|
||||
//
|
||||
#elif defined(IR_USE_TIMER1)
|
||||
|
||||
#define TIMER_RESET
|
||||
#define TIMER_ENABLE_PWM (TCCR1A |= _BV(COM1A1))
|
||||
#define TIMER_DISABLE_PWM (TCCR1A &= ~(_BV(COM1A1)))
|
||||
|
||||
//-----------------
|
||||
#if defined(__AVR_ATmega8__) || defined(__AVR_ATmega8535__) \
|
||||
|| defined(__AVR_ATmega16__) || defined(__AVR_ATmega32__) \
|
||||
|| defined(__AVR_ATmega64__) || defined(__AVR_ATmega128__)
|
||||
# define TIMER_ENABLE_INTR (TIMSK |= _BV(OCIE1A))
|
||||
# define TIMER_DISABLE_INTR (TIMSK &= ~_BV(OCIE1A))
|
||||
#else
|
||||
# define TIMER_ENABLE_INTR (TIMSK1 = _BV(OCIE1A))
|
||||
# define TIMER_DISABLE_INTR (TIMSK1 = 0)
|
||||
#endif
|
||||
|
||||
//-----------------
|
||||
#define TIMER_INTR_NAME TIMER1_COMPA_vect
|
||||
|
||||
#define TIMER_CONFIG_KHZ(val) ({ \
|
||||
const uint16_t pwmval = SYSCLOCK / 2000 / (val); \
|
||||
TCCR1A = _BV(WGM11); \
|
||||
TCCR1B = _BV(WGM13) | _BV(CS10); \
|
||||
ICR1 = pwmval; \
|
||||
OCR1A = pwmval / 3; \
|
||||
})
|
||||
|
||||
#define TIMER_CONFIG_NORMAL() ({ \
|
||||
TCCR1A = 0; \
|
||||
TCCR1B = _BV(WGM12) | _BV(CS10); \
|
||||
OCR1A = SYSCLOCK * USECPERTICK / 1000000; \
|
||||
TCNT1 = 0; \
|
||||
})
|
||||
|
||||
//-----------------
|
||||
#if defined(CORE_OC1A_PIN)
|
||||
# define TIMER_PWM_PIN CORE_OC1A_PIN // Teensy
|
||||
#elif defined(__AVR_ATmega1280__) || defined(__AVR_ATmega2560__)
|
||||
# define TIMER_PWM_PIN 11 // Arduino Mega
|
||||
#elif defined(__AVR_ATmega1284__) || defined(__AVR_ATmega1284P__) \
|
||||
|| defined(__AVR_ATmega644__) || defined(__AVR_ATmega644P__) \
|
||||
|| defined(__AVR_ATmega324P__) || defined(__AVR_ATmega324A__) \
|
||||
|| defined(__AVR_ATmega324PA__) || defined(__AVR_ATmega164A__) \
|
||||
|| defined(__AVR_ATmega164P__) || defined(__AVR_ATmega32__) \
|
||||
|| defined(__AVR_ATmega16__) || defined(__AVR_ATmega8535__) \
|
||||
|| defined(__AVR_ATmega64__) || defined(__AVR_ATmega128__)
|
||||
# define TIMER_PWM_PIN 13 // MightyCore, MegaCore
|
||||
#elif defined(__AVR_ATtiny84__)
|
||||
# define TIMER_PWM_PIN 6
|
||||
#else
|
||||
# define TIMER_PWM_PIN 9 // Arduino Duemilanove, Diecimila, LilyPad, Sparkfun Pro Micro etc
|
||||
#endif // ATmega48, ATmega88, ATmega168, ATmega328
|
||||
|
||||
//---------------------------------------------------------
|
||||
// Timer3 (16 bits)
|
||||
//
|
||||
#elif defined(IR_USE_TIMER3)
|
||||
|
||||
#define TIMER_RESET
|
||||
#define TIMER_ENABLE_PWM (TCCR3A |= _BV(COM3A1))
|
||||
#define TIMER_DISABLE_PWM (TCCR3A &= ~(_BV(COM3A1)))
|
||||
#define TIMER_ENABLE_INTR (TIMSK3 = _BV(OCIE3A))
|
||||
#define TIMER_DISABLE_INTR (TIMSK3 = 0)
|
||||
#define TIMER_INTR_NAME TIMER3_COMPA_vect
|
||||
|
||||
#define TIMER_CONFIG_KHZ(val) ({ \
|
||||
const uint16_t pwmval = SYSCLOCK / 2000 / (val); \
|
||||
TCCR3A = _BV(WGM31); \
|
||||
TCCR3B = _BV(WGM33) | _BV(CS30); \
|
||||
ICR3 = pwmval; \
|
||||
OCR3A = pwmval / 3; \
|
||||
})
|
||||
|
||||
#define TIMER_CONFIG_NORMAL() ({ \
|
||||
TCCR3A = 0; \
|
||||
TCCR3B = _BV(WGM32) | _BV(CS30); \
|
||||
OCR3A = SYSCLOCK * USECPERTICK / 1000000; \
|
||||
TCNT3 = 0; \
|
||||
})
|
||||
|
||||
//-----------------
|
||||
#if defined(CORE_OC3A_PIN)
|
||||
# define TIMER_PWM_PIN CORE_OC3A_PIN // Teensy
|
||||
#elif defined(__AVR_ATmega1280__) || defined(__AVR_ATmega2560__) || defined(ARDUINO_AVR_PROMICRO)
|
||||
# define TIMER_PWM_PIN 5 // Arduino Mega, Sparkfun Pro Micro
|
||||
#elif defined(__AVR_ATmega1284__) || defined(__AVR_ATmega1284P__)
|
||||
# define TIMER_PWM_PIN 6 // MightyCore
|
||||
#else
|
||||
# error "Please add OC3A pin number here\n"
|
||||
#endif
|
||||
|
||||
//---------------------------------------------------------
|
||||
// Timer4 (10 bits, high speed option)
|
||||
//
|
||||
#elif defined(IR_USE_TIMER4_HS)
|
||||
|
||||
#define TIMER_RESET
|
||||
|
||||
#if defined(ARDUINO_AVR_PROMICRO) // Sparkfun Pro Micro
|
||||
#define TIMER_ENABLE_PWM (TCCR4A |= _BV(COM4A0)) // Use complimentary O̅C̅4̅A̅ output on pin 5
|
||||
#define TIMER_DISABLE_PWM (TCCR4A &= ~(_BV(COM4A0))) // (Pro Micro does not map PC7 (32/ICP3/CLK0/OC4A)
|
||||
// of ATmega32U4 )
|
||||
#else
|
||||
#define TIMER_ENABLE_PWM (TCCR4A |= _BV(COM4A1))
|
||||
#define TIMER_DISABLE_PWM (TCCR4A &= ~(_BV(COM4A1)))
|
||||
#endif
|
||||
|
||||
#define TIMER_ENABLE_INTR (TIMSK4 = _BV(TOIE4))
|
||||
#define TIMER_DISABLE_INTR (TIMSK4 = 0)
|
||||
#define TIMER_INTR_NAME TIMER4_OVF_vect
|
||||
|
||||
|
||||
#define TIMER_CONFIG_KHZ(val) ({ \
|
||||
const uint16_t pwmval = SYSCLOCK / 2000 / (val); \
|
||||
TCCR4A = (1<<PWM4A); \
|
||||
TCCR4B = _BV(CS40); \
|
||||
TCCR4C = 0; \
|
||||
TCCR4D = (1<<WGM40); \
|
||||
TCCR4E = 0; \
|
||||
TC4H = pwmval >> 8; \
|
||||
OCR4C = pwmval; \
|
||||
TC4H = (pwmval / 3) >> 8; \
|
||||
OCR4A = (pwmval / 3) & 255; \
|
||||
})
|
||||
|
||||
#define TIMER_CONFIG_NORMAL() ({ \
|
||||
TCCR4A = 0; \
|
||||
TCCR4B = _BV(CS40); \
|
||||
TCCR4C = 0; \
|
||||
TCCR4D = 0; \
|
||||
TCCR4E = 0; \
|
||||
TC4H = (SYSCLOCK * USECPERTICK / 1000000) >> 8; \
|
||||
OCR4C = (SYSCLOCK * USECPERTICK / 1000000) & 255; \
|
||||
TC4H = 0; \
|
||||
TCNT4 = 0; \
|
||||
})
|
||||
|
||||
//-----------------
|
||||
#if defined(CORE_OC4A_PIN)
|
||||
# define TIMER_PWM_PIN CORE_OC4A_PIN // Teensy
|
||||
#elif defined(ARDUINO_AVR_PROMICRO)
|
||||
# define TIMER_PWM_PIN 5 // Sparkfun Pro Micro
|
||||
#elif defined(__AVR_ATmega32U4__)
|
||||
# define TIMER_PWM_PIN 13 // Leonardo
|
||||
#else
|
||||
# error "Please add OC4A pin number here\n"
|
||||
#endif
|
||||
|
||||
//---------------------------------------------------------
|
||||
// Timer4 (16 bits)
|
||||
//
|
||||
#elif defined(IR_USE_TIMER4)
|
||||
|
||||
#define TIMER_RESET
|
||||
#define TIMER_ENABLE_PWM (TCCR4A |= _BV(COM4A1))
|
||||
#define TIMER_DISABLE_PWM (TCCR4A &= ~(_BV(COM4A1)))
|
||||
#define TIMER_ENABLE_INTR (TIMSK4 = _BV(OCIE4A))
|
||||
#define TIMER_DISABLE_INTR (TIMSK4 = 0)
|
||||
#define TIMER_INTR_NAME TIMER4_COMPA_vect
|
||||
|
||||
#define TIMER_CONFIG_KHZ(val) ({ \
|
||||
const uint16_t pwmval = SYSCLOCK / 2000 / (val); \
|
||||
TCCR4A = _BV(WGM41); \
|
||||
TCCR4B = _BV(WGM43) | _BV(CS40); \
|
||||
ICR4 = pwmval; \
|
||||
OCR4A = pwmval / 3; \
|
||||
})
|
||||
|
||||
#define TIMER_CONFIG_NORMAL() ({ \
|
||||
TCCR4A = 0; \
|
||||
TCCR4B = _BV(WGM42) | _BV(CS40); \
|
||||
OCR4A = SYSCLOCK * USECPERTICK / 1000000; \
|
||||
TCNT4 = 0; \
|
||||
})
|
||||
|
||||
//-----------------
|
||||
#if defined(CORE_OC4A_PIN)
|
||||
# define TIMER_PWM_PIN CORE_OC4A_PIN
|
||||
#elif defined(__AVR_ATmega1280__) || defined(__AVR_ATmega2560__)
|
||||
# define TIMER_PWM_PIN 6 // Arduino Mega
|
||||
#else
|
||||
# error "Please add OC4A pin number here\n"
|
||||
#endif
|
||||
|
||||
//---------------------------------------------------------
|
||||
// Timer5 (16 bits)
|
||||
//
|
||||
#elif defined(IR_USE_TIMER5)
|
||||
|
||||
#define TIMER_RESET
|
||||
#define TIMER_ENABLE_PWM (TCCR5A |= _BV(COM5A1))
|
||||
#define TIMER_DISABLE_PWM (TCCR5A &= ~(_BV(COM5A1)))
|
||||
#define TIMER_ENABLE_INTR (TIMSK5 = _BV(OCIE5A))
|
||||
#define TIMER_DISABLE_INTR (TIMSK5 = 0)
|
||||
#define TIMER_INTR_NAME TIMER5_COMPA_vect
|
||||
|
||||
#define TIMER_CONFIG_KHZ(val) ({ \
|
||||
const uint16_t pwmval = SYSCLOCK / 2000 / (val); \
|
||||
TCCR5A = _BV(WGM51); \
|
||||
TCCR5B = _BV(WGM53) | _BV(CS50); \
|
||||
ICR5 = pwmval; \
|
||||
OCR5A = pwmval / 3; \
|
||||
})
|
||||
|
||||
#define TIMER_CONFIG_NORMAL() ({ \
|
||||
TCCR5A = 0; \
|
||||
TCCR5B = _BV(WGM52) | _BV(CS50); \
|
||||
OCR5A = SYSCLOCK * USECPERTICK / 1000000; \
|
||||
TCNT5 = 0; \
|
||||
})
|
||||
|
||||
//-----------------
|
||||
#if defined(CORE_OC5A_PIN)
|
||||
# define TIMER_PWM_PIN CORE_OC5A_PIN
|
||||
#elif defined(__AVR_ATmega1280__) || defined(__AVR_ATmega2560__)
|
||||
# define TIMER_PWM_PIN 46 // Arduino Mega
|
||||
#else
|
||||
# error "Please add OC5A pin number here\n"
|
||||
#endif
|
||||
|
||||
//---------------------------------------------------------
|
||||
// Special carrier modulator timer
|
||||
//
|
||||
#elif defined(IR_USE_TIMER_CMT)
|
||||
|
||||
#define TIMER_RESET ({ \
|
||||
uint8_t tmp __attribute__((unused)) = CMT_MSC; \
|
||||
CMT_CMD2 = 30; \
|
||||
})
|
||||
|
||||
#define TIMER_ENABLE_PWM do { \
|
||||
CORE_PIN5_CONFIG = PORT_PCR_MUX(2) | PORT_PCR_DSE | PORT_PCR_SRE; \
|
||||
} while(0)
|
||||
|
||||
#define TIMER_DISABLE_PWM do { \
|
||||
CORE_PIN5_CONFIG = PORT_PCR_MUX(1) | PORT_PCR_DSE | PORT_PCR_SRE; \
|
||||
} while(0)
|
||||
|
||||
#define TIMER_ENABLE_INTR NVIC_ENABLE_IRQ(IRQ_CMT)
|
||||
#define TIMER_DISABLE_INTR NVIC_DISABLE_IRQ(IRQ_CMT)
|
||||
#define TIMER_INTR_NAME cmt_isr
|
||||
|
||||
//-----------------
|
||||
#ifdef ISR
|
||||
# undef ISR
|
||||
#endif
|
||||
#define ISR(f) void f(void)
|
||||
|
||||
//-----------------
|
||||
#define CMT_PPS_DIV ((F_BUS + 7999999) / 8000000)
|
||||
#if F_BUS < 8000000
|
||||
#error IRremote requires at least 8 MHz on Teensy 3.x
|
||||
#endif
|
||||
|
||||
//-----------------
|
||||
#define TIMER_CONFIG_KHZ(val) ({ \
|
||||
SIM_SCGC4 |= SIM_SCGC4_CMT; \
|
||||
SIM_SOPT2 |= SIM_SOPT2_PTD7PAD; \
|
||||
CMT_PPS = CMT_PPS_DIV - 1; \
|
||||
CMT_CGH1 = ((F_BUS / CMT_PPS_DIV / 3000) + ((val)/2)) / (val); \
|
||||
CMT_CGL1 = ((F_BUS / CMT_PPS_DIV / 1500) + ((val)/2)) / (val); \
|
||||
CMT_CMD1 = 0; \
|
||||
CMT_CMD2 = 30; \
|
||||
CMT_CMD3 = 0; \
|
||||
CMT_CMD4 = 0; \
|
||||
CMT_OC = 0x60; \
|
||||
CMT_MSC = 0x01; \
|
||||
})
|
||||
|
||||
#define TIMER_CONFIG_NORMAL() ({ \
|
||||
SIM_SCGC4 |= SIM_SCGC4_CMT; \
|
||||
CMT_PPS = CMT_PPS_DIV - 1; \
|
||||
CMT_CGH1 = 1; \
|
||||
CMT_CGL1 = 1; \
|
||||
CMT_CMD1 = 0; \
|
||||
CMT_CMD2 = 30; \
|
||||
CMT_CMD3 = 0; \
|
||||
CMT_CMD4 = (F_BUS / 160000 + CMT_PPS_DIV / 2) / CMT_PPS_DIV - 31; \
|
||||
CMT_OC = 0; \
|
||||
CMT_MSC = 0x03; \
|
||||
})
|
||||
|
||||
#define TIMER_PWM_PIN 5
|
||||
|
||||
// defines for TPM1 timer on Teensy-LC
|
||||
#elif defined(IR_USE_TIMER_TPM1)
|
||||
#define TIMER_RESET FTM1_SC |= FTM_SC_TOF;
|
||||
#define TIMER_ENABLE_PWM CORE_PIN16_CONFIG = PORT_PCR_MUX(3)|PORT_PCR_DSE|PORT_PCR_SRE
|
||||
#define TIMER_DISABLE_PWM CORE_PIN16_CONFIG = PORT_PCR_MUX(1)|PORT_PCR_SRE
|
||||
#define TIMER_ENABLE_INTR NVIC_ENABLE_IRQ(IRQ_FTM1)
|
||||
#define TIMER_DISABLE_INTR NVIC_DISABLE_IRQ(IRQ_FTM1)
|
||||
#define TIMER_INTR_NAME ftm1_isr
|
||||
#ifdef ISR
|
||||
#undef ISR
|
||||
#endif
|
||||
#define ISR(f) void f(void)
|
||||
#define TIMER_CONFIG_KHZ(val) ({ \
|
||||
SIM_SCGC6 |= SIM_SCGC6_TPM1; \
|
||||
FTM1_SC = 0; \
|
||||
FTM1_CNT = 0; \
|
||||
FTM1_MOD = (F_PLL/2000) / val - 1; \
|
||||
FTM1_C0V = (F_PLL/6000) / val - 1; \
|
||||
FTM1_SC = FTM_SC_CLKS(1) | FTM_SC_PS(0); \
|
||||
})
|
||||
#define TIMER_CONFIG_NORMAL() ({ \
|
||||
SIM_SCGC6 |= SIM_SCGC6_TPM1; \
|
||||
FTM1_SC = 0; \
|
||||
FTM1_CNT = 0; \
|
||||
FTM1_MOD = (F_PLL/40000) - 1; \
|
||||
FTM1_C0V = 0; \
|
||||
FTM1_SC = FTM_SC_CLKS(1) | FTM_SC_PS(0) | FTM_SC_TOF | FTM_SC_TOIE; \
|
||||
})
|
||||
#define TIMER_PWM_PIN 16
|
||||
|
||||
// defines for timer_tiny0 (8 bits)
|
||||
#elif defined(IR_USE_TIMER_TINY0)
|
||||
#define TIMER_RESET
|
||||
#define TIMER_ENABLE_PWM (TCCR0A |= _BV(COM0B1))
|
||||
#define TIMER_DISABLE_PWM (TCCR0A &= ~(_BV(COM0B1)))
|
||||
#define TIMER_ENABLE_INTR (TIMSK |= _BV(OCIE0A))
|
||||
#define TIMER_DISABLE_INTR (TIMSK &= ~(_BV(OCIE0A)))
|
||||
#define TIMER_INTR_NAME TIMER0_COMPA_vect
|
||||
#define TIMER_CONFIG_KHZ(val) ({ \
|
||||
const uint8_t pwmval = SYSCLOCK / 2000 / (val); \
|
||||
TCCR0A = _BV(WGM00); \
|
||||
TCCR0B = _BV(WGM02) | _BV(CS00); \
|
||||
OCR0A = pwmval; \
|
||||
OCR0B = pwmval / 3; \
|
||||
})
|
||||
#define TIMER_COUNT_TOP (SYSCLOCK * USECPERTICK / 1000000)
|
||||
#if (TIMER_COUNT_TOP < 256)
|
||||
#define TIMER_CONFIG_NORMAL() ({ \
|
||||
TCCR0A = _BV(WGM01); \
|
||||
TCCR0B = _BV(CS00); \
|
||||
OCR0A = TIMER_COUNT_TOP; \
|
||||
TCNT0 = 0; \
|
||||
})
|
||||
#else
|
||||
#define TIMER_CONFIG_NORMAL() ({ \
|
||||
TCCR0A = _BV(WGM01); \
|
||||
TCCR0B = _BV(CS01); \
|
||||
OCR0A = TIMER_COUNT_TOP / 8; \
|
||||
TCNT0 = 0; \
|
||||
})
|
||||
#endif
|
||||
|
||||
#define TIMER_PWM_PIN 1 /* ATtiny85 */
|
||||
|
||||
//---------------------------------------------------------
|
||||
// ESP32 (ESP8266 should likely be added here too)
|
||||
//
|
||||
|
||||
// ESP32 has it own timer API and does not use these macros, but to avoid ifdef'ing
|
||||
// them out in the common code, they are defined to no-op. This allows the code to compile
|
||||
// (which it wouldn't otherwise) but irsend will not work until ESP32 specific code is written
|
||||
// for that -- merlin
|
||||
// As a warning, sending timing specific code from an ESP32 can be challenging if you need 100%
|
||||
// reliability because the arduino code may be interrupted and cause your sent waveform to be the
|
||||
// wrong length. This is specifically an issue for neopixels which require 800Khz resolution.
|
||||
// IR may just work as is with the common code since it's lower frequency, but if not, the other
|
||||
// way to do this on ESP32 is using the RMT built in driver like in this incomplete library below
|
||||
// https://github.com/ExploreEmbedded/ESP32_RMT
|
||||
#elif defined(IR_TIMER_USE_ESP32)
|
||||
#define TIMER_RESET
|
||||
#define TIMER_ENABLE_PWM
|
||||
#define TIMER_DISABLE_PWM Serial.println("IRsend not implemented for ESP32 yet");
|
||||
#define TIMER_ENABLE_INTR
|
||||
#define TIMER_DISABLE_INTR
|
||||
#define TIMER_INTR_NAME
|
||||
|
||||
//---------------------------------------------------------
|
||||
// Unknown Timer
|
||||
//
|
||||
#else
|
||||
# error "Internal code configuration error, no known IR_USE_TIMER# defined\n"
|
||||
#endif
|
||||
|
||||
#endif // ! boarddefs_h
|
||||
862
Firmware/Source/grovepi/src/grovepi.ino
Normal file
862
Firmware/Source/grovepi/src/grovepi.ino
Normal file
|
|
@ -0,0 +1,862 @@
|
|||
#include "ChainableLED.h"
|
||||
#include "DHT.h"
|
||||
#include "Grove_LED_Bar.h"
|
||||
#include "TM1637.h"
|
||||
#include "TimerOne.h"
|
||||
#include <IRremote.h>
|
||||
#include <Wire.h>
|
||||
#include <YetAnotherPcInt.h>
|
||||
|
||||
DHT dht;
|
||||
Grove_LED_Bar ledbar[6]; // 7 instances for D2-D8, however, max 4 bars, you; can't use adjacent sockets, 4 pin display
|
||||
TM1637 fourdigit[6]; // 7 instances for D2-D8, however, max 4 displays, you; can't use adjacent sockets, 4 pin display
|
||||
ChainableLED rgbled[6]; // 7 instances for D2-D8
|
||||
|
||||
IRrecv irrecv; // object to interface with the IR receiver
|
||||
decode_results results; // results for the IR receiver
|
||||
|
||||
#define SLAVE_ADDRESS 0x04
|
||||
|
||||
// #define dust_sensor_read_cmd 10
|
||||
// #define dust_sensor_en_cmd 14
|
||||
// #define dust_sensor_dis_cmd 15
|
||||
// #define dust_sensor_int_cmd 9
|
||||
// #define dust_sensor_read_int_cmd 6
|
||||
|
||||
// #define encoder_read_cmd 11
|
||||
// #define encoder_en_cmd 16
|
||||
// #define encoder_dis_cmd 17
|
||||
|
||||
// #define flow_read_cmd 12
|
||||
// #define flow_en_cmd 18
|
||||
// #define flow_dis_cmd 13
|
||||
|
||||
#define ir_read_cmd 21
|
||||
#define ir_recv_pin_cmd 22
|
||||
#define ir_read_isdata 24
|
||||
|
||||
#define isr_set_cmd 6
|
||||
#define isr_unset_cmd 9
|
||||
#define isr_read_cmd 10
|
||||
#define isr_clear_cmd 11
|
||||
#define isr_active_cmd 12
|
||||
|
||||
#define encoder_read_cmd 13
|
||||
#define encoder_en_cmd 14
|
||||
#define encoder_dis_cmd 15
|
||||
|
||||
#define data_not_available 23
|
||||
|
||||
volatile uint8_t cmd[5];
|
||||
volatile int index = 0;
|
||||
volatile int flag = 0;
|
||||
volatile byte b[21], float_array[4], dht_b[21];
|
||||
volatile bool need_extra_loop = false;
|
||||
int pin;
|
||||
|
||||
// for interrupt stuff
|
||||
const uint8_t total_ports = 9;
|
||||
enum ISR_Type {
|
||||
COUNT_CHANGES, COUNT_LOW_DURATION
|
||||
};
|
||||
typedef struct {
|
||||
volatile unsigned long period, last_update;
|
||||
} Tracked_time;
|
||||
typedef struct {
|
||||
volatile unsigned long counted_duration, pulse_end, pulse_start;
|
||||
} Pulse_counter;
|
||||
|
||||
volatile uint8_t pins[total_ports]; // to track the available pins (passes the address to PcInt attachInterrupt method)
|
||||
volatile bool set_pcint[total_ports]; // to see which ports are being watched
|
||||
volatile uint8_t func_type[total_ports]; // type of counting function - COUNT_CHANGES or COUNT_LOW_DURATION
|
||||
Tracked_time tracked_time[total_ports]; // for tracking the time
|
||||
Pulse_counter pulse_counter[total_ports]; // for counting pulses
|
||||
volatile unsigned long change_counter[total_ports]; // for counting changes
|
||||
volatile uint32_t buffer[total_ports]; // to store the calculated values for transmission
|
||||
|
||||
// for encoders
|
||||
typedef struct {
|
||||
volatile uint8_t LOW_PIN, HIGH_PIN;
|
||||
volatile int32_t value;
|
||||
volatile uint8_t MAX_VAL = 32;
|
||||
} GroveEncoder;
|
||||
GroveEncoder ge[total_ports];
|
||||
|
||||
volatile int run_once;
|
||||
unsigned char dta[21];
|
||||
int length, i;
|
||||
int aRead = 0;
|
||||
byte accFlag = 0, clkFlag = 0;
|
||||
int8_t accv[3];
|
||||
byte rgb[] = {0, 0, 0};
|
||||
|
||||
// all user-defined functions
|
||||
void processIO();
|
||||
void flushI2C();
|
||||
void receiveData();
|
||||
void sendData();
|
||||
void detachISRPin(const uint8_t);
|
||||
void isr_buffer_filler();
|
||||
void isr_handler(uint8_t, bool);
|
||||
|
||||
void setup() {
|
||||
// Start serial
|
||||
Serial.begin(38400);
|
||||
|
||||
// Start I2C
|
||||
Wire.begin(SLAVE_ADDRESS);
|
||||
Wire.onReceive(receiveData);
|
||||
Wire.onRequest(sendData);
|
||||
|
||||
// Set port 4 to output and set it to LOW
|
||||
DDRD |= 0x10;
|
||||
PORTD &= ~0x10;
|
||||
|
||||
// initialize the pins array with the ports itself
|
||||
for(int i = 0; i < total_ports; i++) {
|
||||
pins[i] = i;
|
||||
}
|
||||
|
||||
Timer1.attachInterrupt(isr_buffer_filler, 1000); // Buffer maintenance.
|
||||
}
|
||||
|
||||
void processIO() {
|
||||
// Handles incoming data.
|
||||
// Updates sensors.
|
||||
// Sets output pins.
|
||||
|
||||
if (index == 4 && flag == 0) {
|
||||
flag = 1;
|
||||
// Digital Read
|
||||
if (cmd[0] == 1)
|
||||
{
|
||||
b[0] = cmd[0];
|
||||
b[1] = digitalRead(cmd[1]);
|
||||
}
|
||||
|
||||
// Digital Write
|
||||
else if (cmd[0] == 2)
|
||||
digitalWrite(cmd[1], cmd[2]);
|
||||
|
||||
// Analog Read
|
||||
else if (cmd[0] == 3) {
|
||||
aRead = analogRead(cmd[1]);
|
||||
b[0] = cmd[0];
|
||||
b[1] = aRead / 256;
|
||||
b[2] = aRead % 256;
|
||||
}
|
||||
|
||||
// Set up Analog Write
|
||||
else if (cmd[0] == 4)
|
||||
analogWrite(cmd[1], cmd[2]);
|
||||
|
||||
// Set up pinMode
|
||||
else if (cmd[0] == 5)
|
||||
pinMode(cmd[1], cmd[2]);
|
||||
|
||||
// Ultrasonic Read
|
||||
else if (cmd[0] == 7) {
|
||||
pin = cmd[1];
|
||||
pinMode(pin, OUTPUT);
|
||||
digitalWrite(pin, LOW);
|
||||
delayMicroseconds(2);
|
||||
digitalWrite(pin, HIGH);
|
||||
delayMicroseconds(5);
|
||||
digitalWrite(pin, LOW);
|
||||
pinMode(pin, INPUT);
|
||||
// setting timeout to 75000 microseconds
|
||||
// which is roughly the equivalent of time needed for the sound
|
||||
// to travel 10 meters from the sensor to the target
|
||||
// and backwards - where the speed of sound is 343m/s
|
||||
long dur = pulseIn(pin, HIGH, 75000);
|
||||
long RangeCm = dur / 29 / 2;
|
||||
b[0] = cmd[0];
|
||||
b[1] = RangeCm / 256;
|
||||
b[2] = RangeCm % 256;
|
||||
// Serial.println(b[1] * 256 + b[2]);
|
||||
// Serial.println(b[1]);
|
||||
// Serial.println(b[2]);
|
||||
}
|
||||
// Firmware version
|
||||
else if (cmd[0] == 8) {
|
||||
b[0] = cmd[0];
|
||||
b[1] = 1;
|
||||
b[2] = 4;
|
||||
b[3] = 0;
|
||||
}
|
||||
|
||||
// Grove temp and humidity sensor pro
|
||||
// 40- Temperature
|
||||
else if (cmd[0] == 40) {
|
||||
if (run_once) {
|
||||
if (cmd[2] == 0)
|
||||
dht.begin(cmd[1], DHT11);
|
||||
else if (cmd[2] == 1)
|
||||
dht.begin(cmd[1], DHT22);
|
||||
else if (cmd[2] == 2)
|
||||
dht.begin(cmd[1], DHT21);
|
||||
else if (cmd[2] == 3)
|
||||
dht.begin(cmd[1], AM2301);
|
||||
float *buffer;
|
||||
buffer = dht.readTempHum();
|
||||
|
||||
byte *b1 = (byte *)buffer;
|
||||
byte *b2 = (byte *)(buffer + 1);
|
||||
|
||||
dht_b[0] = cmd[0];
|
||||
for (uint8_t j = 1; j < 5; j++)
|
||||
dht_b[j] = b1[j - 1];
|
||||
for (uint8_t j = 5; j < 9; j++)
|
||||
dht_b[j] = b2[j - 5];
|
||||
run_once = 0;
|
||||
}
|
||||
}
|
||||
|
||||
// Grove LED Bar
|
||||
// http://www.seeedstudio.com/wiki/Grove_-_LED_Bar
|
||||
// pins: data,clock,vcc,gnd
|
||||
|
||||
// Commands
|
||||
// [50, pin, greenToRed, unused] initialise a LED Bar
|
||||
// [51, pin, greenToRed, unused] setGreenToRed(bool greenToRed)
|
||||
// [52, pin, level, unused] setLevel(unsigned char level)
|
||||
// [53, pin, led, state] setLed(unsigned char led, bool state)
|
||||
// [54, pin, led, unused] toggleLed(unsigned char led)
|
||||
// [55, pin, bits 1-8, bits 9-10] setBits(unsigned int bits)
|
||||
// [56, pin, unused, unused] getBits()
|
||||
|
||||
// Initialise
|
||||
// [50, pin, orientation, unused]
|
||||
else if (cmd[0] == 50) {
|
||||
// clock pin is always next to the data pin
|
||||
ledbar[cmd[1] - 2].begin(cmd[1] + 1, cmd[1],
|
||||
cmd[2]); // clock, data, orientation
|
||||
}
|
||||
|
||||
// Change the orientation
|
||||
// Green to red, or red to green
|
||||
// [51, pin, greenToRed, unused]
|
||||
else if (cmd[0] == 51 && ledbar[cmd[1] - 2].ready()) {
|
||||
ledbar[cmd[1] - 2].setGreenToRed(cmd[2]);
|
||||
}
|
||||
|
||||
// Set level (0-10)
|
||||
// Level 0 means all leds off
|
||||
// Level 10 means all leds on
|
||||
// [52, pin, level, unused]
|
||||
else if (cmd[0] == 52 && ledbar[cmd[1] - 2].ready()) {
|
||||
ledbar[cmd[1] - 2].setLevel(cmd[2]);
|
||||
}
|
||||
|
||||
// Set a single led
|
||||
// led (1-10)
|
||||
// state (0=off, 1=on)
|
||||
// [53, pin, led, state]
|
||||
else if (cmd[0] == 53 && ledbar[cmd[1] - 2].ready()) {
|
||||
ledbar[cmd[1] - 2].setLed(cmd[2], cmd[3]);
|
||||
}
|
||||
|
||||
// Toggle a single led
|
||||
// led (1-10)
|
||||
// [54, pin, led, unused]
|
||||
else if (cmd[0] == 54 && ledbar[cmd[1] - 2].ready()) {
|
||||
ledbar[cmd[1] - 2].toggleLed(cmd[2]);
|
||||
}
|
||||
|
||||
// Set the current state, one bit for each led
|
||||
// 0 = 0x0 = 0b000000000000000 = all leds off
|
||||
// 5 = 0x05 = 0b000000000000101 = leds 1 and 3 on, all others off
|
||||
// 341 = 0x155 = 0b000000101010101 = leds 1,3,5,7,9 on, 2,4,6,8,10 off
|
||||
// 1023 = 0x3ff = 0b000001111111111 = all leds on
|
||||
// | |
|
||||
// 10 1
|
||||
// [55, pin, bits 1-8, bits 9-10]
|
||||
else if (cmd[0] == 55 && ledbar[cmd[1] - 2].ready()) {
|
||||
ledbar[cmd[1] - 2].setBits(cmd[2] ^ (cmd[3] << 8));
|
||||
}
|
||||
|
||||
// Return the current state
|
||||
// [56, pin, unused, unused]
|
||||
else if (cmd[0] == 56 && ledbar[cmd[1] - 2].ready()) {
|
||||
unsigned int state = ledbar[cmd[1] - 2].getBits();
|
||||
b[0] = cmd[0];
|
||||
b[1] = state & 0xFF;
|
||||
b[2] = state >> 8;
|
||||
}
|
||||
|
||||
// end Grove LED Bar
|
||||
|
||||
// Grove 4 Digit Display (7 segment)
|
||||
// http://www.seeedstudio.com/wiki/Grove_-_4-Digit_Display
|
||||
// pins: clock,data,vcc,gnd
|
||||
|
||||
// Commands
|
||||
// [70, pin, unused, unused] initialise a 4 digit display
|
||||
// [71, pin, brightness, unused] set brightness
|
||||
// [72, pin, bits 1-8, bits 9-16] right aligned decimal value without
|
||||
// leading zeros [73, pin, bits 1-8, bits 9-16] right aligned decimal value
|
||||
// with leading zeros [74, pin, index, dec] set individual digit
|
||||
// [75, pin, index, binary] set individual segment
|
||||
// [76, pin, left, right] set left and right values with colon
|
||||
// [77, pin, analog pin, seconds] display analog read for n seconds
|
||||
// [78, pin, unused, unused] display on
|
||||
// [79, pin, unused, unused] display off
|
||||
|
||||
// initialise a 4 digit display
|
||||
// [70, pin, unused, unused]
|
||||
else if (cmd[0] == 70) {
|
||||
// clock pin is always next to the data pin
|
||||
fourdigit[cmd[1] - 2].begin(cmd[1], cmd[1] + 1); // clock, data
|
||||
}
|
||||
|
||||
// set brightness
|
||||
// [71, pin, brightness, unused]
|
||||
else if (cmd[0] == 71 && fourdigit[cmd[1] - 2].ready()) {
|
||||
fourdigit[cmd[1] - 2].setBrightness(cmd[2]); // setBrightness(brightness)
|
||||
}
|
||||
|
||||
// show right aligned decimal value without leading zeros
|
||||
// [72, pin, bits 1-8, bits 9-16]
|
||||
else if (cmd[0] == 72 && fourdigit[cmd[1] - 2].ready()) {
|
||||
fourdigit[cmd[1] - 2].showNumberDec(
|
||||
cmd[2] ^ (cmd[3] << 8), false); // showNumberDec(number, leading_zero)
|
||||
}
|
||||
|
||||
// show right aligned decimal value with leading zeros
|
||||
// [73, pin, bits 1-8, bits 9-16]
|
||||
else if (cmd[0] == 73 && fourdigit[cmd[1] - 2].ready()) {
|
||||
fourdigit[cmd[1] - 2].showNumberDec(
|
||||
cmd[2] ^ (cmd[3] << 8), true); // showNumberDec(number, leading_zero)
|
||||
}
|
||||
|
||||
// set individual digit
|
||||
// [74, pin, index, dec]
|
||||
else if (cmd[0] == 74 && fourdigit[cmd[1] - 2].ready()) {
|
||||
uint8_t data[] = {};
|
||||
data[0] =
|
||||
fourdigit[cmd[1] - 2].encodeDigit(cmd[3]); // encodeDigit(number)
|
||||
fourdigit[cmd[1] - 2].setSegments(
|
||||
data, 1, cmd[2]); // setSegments(segments[], length, position)
|
||||
}
|
||||
|
||||
// set individual segment
|
||||
// [75, pin, index, binary]
|
||||
else if (cmd[0] == 75 && fourdigit[cmd[1] - 2].ready()) {
|
||||
// 0xFF = 0b11111111 = Colon,G,F,E,D,C,B,A
|
||||
// Colon only works on 2nd segment (index 1)
|
||||
// -A-
|
||||
// F | | B
|
||||
// -G-
|
||||
// E | | C
|
||||
// -D-
|
||||
uint8_t data[] = {};
|
||||
data[0] = cmd[3]; // byte
|
||||
fourdigit[cmd[1] - 2].setSegments(
|
||||
data, 1, cmd[2]); // setSegments(segments[], length, position)
|
||||
}
|
||||
|
||||
// set left and right with colon separator
|
||||
// [76, pin, left, right]
|
||||
else if (cmd[0] == 76 && fourdigit[cmd[1] - 2].ready()) {
|
||||
uint8_t data[] = {};
|
||||
// 1st segment
|
||||
data[0] =
|
||||
fourdigit[cmd[1] - 2].encodeDigit(cmd[2] / 10); // encodeDigit(number)
|
||||
// 2nd segment
|
||||
data[1] =
|
||||
fourdigit[cmd[1] - 2].encodeDigit(cmd[2] % 10); // encodeDigit(number)
|
||||
// colon
|
||||
data[1] |= 0x80;
|
||||
// 3rd segment
|
||||
data[2] =
|
||||
fourdigit[cmd[1] - 2].encodeDigit(cmd[3] / 10); // encodeDigit(number)
|
||||
// 4th segment
|
||||
data[3] =
|
||||
fourdigit[cmd[1] - 2].encodeDigit(cmd[3] % 10); // encodeDigit(number)
|
||||
// send
|
||||
fourdigit[cmd[1] - 2].setSegments(
|
||||
data, 4, 0); // setSegments(segments[], length, position)
|
||||
}
|
||||
|
||||
// analog read
|
||||
// [77, pin, analog pin, seconds]
|
||||
else if (cmd[0] == 77 && fourdigit[cmd[1] - 2].ready()) {
|
||||
int pin = cmd[2];
|
||||
int reads = 4 * cmd[3]; // 1000/250 * cmd[3]
|
||||
|
||||
// reading analog pin 4x per second
|
||||
for (int i = 0; i < reads; i++) {
|
||||
fourdigit[cmd[1] - 2].showNumberDec(
|
||||
analogRead(pin), false); // showNumberDec(number, leading_zero)
|
||||
}
|
||||
}
|
||||
|
||||
// display on
|
||||
// [78, pin, unused, unused]
|
||||
else if (cmd[0] == 78 && fourdigit[cmd[1] - 2].ready()) {
|
||||
uint8_t data[] = {0xFF, 0xFF, 0xFF, 0xFF};
|
||||
fourdigit[cmd[1] - 2].setSegments(
|
||||
data, 4, 0); // setSegments(segments[], length, position)
|
||||
}
|
||||
|
||||
// display off
|
||||
// [79, pin, unused, unused]
|
||||
else if (cmd[0] == 79 && fourdigit[cmd[1] - 2].ready()) {
|
||||
uint8_t data[] = {0x00, 0x00, 0x00, 0x00};
|
||||
fourdigit[cmd[1] - 2].setSegments(
|
||||
data, 4, 0); // setSegments(segments[], length, position)
|
||||
}
|
||||
|
||||
// end Grove 4 Digit Display
|
||||
|
||||
// Grove Chainable RGB LED
|
||||
// http://www.seeedstudio.com/wiki/Grove_-_Chainable_RGB_LED
|
||||
// pins: ci,di,vcc,gnd and co,do,vcc,gnd
|
||||
|
||||
// Commands
|
||||
// [90, red, green, blue] store color for later use
|
||||
// [91, pin, num leds, unused] initialise a chain of leds
|
||||
// [92, pin, num leds, unused] initialise a chain of leds and set
|
||||
// all to a test color [93, pin, pattern, which led] set one or more
|
||||
// leds to the stored color by pattern [94, pin, led offset, modulo divisor]
|
||||
// set one or more leds to the stored color by modulo [95, pin, level,
|
||||
// reverse] sets leds similar to a bar graph, reversible
|
||||
|
||||
// Store RGB color for later use
|
||||
// [90, red, green, blue]
|
||||
else if (cmd[0] == 90) {
|
||||
rgb[0] = cmd[1];
|
||||
rgb[1] = cmd[2];
|
||||
rgb[2] = cmd[3];
|
||||
}
|
||||
|
||||
// Initialise a RGB LED chain
|
||||
// [91, pin, num leds, unused]
|
||||
else if (cmd[0] == 91) {
|
||||
rgbled[cmd[1] - 2].begin(cmd[1], cmd[1] + 1,
|
||||
cmd[2]); // clock, data, num leds
|
||||
}
|
||||
|
||||
// Test colors, repeating red green blue
|
||||
// color code: 0 black (off), 1 blue, 2 green, 3 cyan, 4 red, 5 magenta, 6
|
||||
// yellow, 7 white [92, pin, num leds, color code]
|
||||
else if (cmd[0] == 92) {
|
||||
rgbled[cmd[1] - 2].begin(cmd[1], cmd[1] + 1, cmd[2]);
|
||||
|
||||
// figure out which color to display, a single bit for each rgb led
|
||||
byte rr = ((cmd[3] & 4) >> 2) * 255, gg = ((cmd[3] & 2) >> 1) * 255,
|
||||
bb = ((cmd[3] & 1)) * 255;
|
||||
|
||||
// set each led to the specified color
|
||||
for (int i = 0; i < cmd[2]; i++) {
|
||||
rgbled[cmd[1] - 2].setColorRGB(i, rr, gg, bb);
|
||||
}
|
||||
}
|
||||
|
||||
// Set one or more leds to the stored color using pattern
|
||||
// pattern: 0 = this led only, 1 all leds except this led, 2 this led and
|
||||
// all leds inwards, 3 this led and all leds outwards which led: 0 = led
|
||||
// closest to the GrovePi, 1 = second led counting outwards [93, pin,
|
||||
// pattern, which led]
|
||||
else if (cmd[0] == 93) {
|
||||
if (cmd[2] == 0) {
|
||||
// set an individual led to the stored color
|
||||
rgbled[cmd[1] - 2].setColorRGB(cmd[3], rgb[0], rgb[1],
|
||||
rgb[2]); // which led, red, green, blue
|
||||
} else {
|
||||
// set all leds to stored color
|
||||
byte num_leds = rgbled[cmd[1] - 2].getNumLeds();
|
||||
|
||||
for (int i = 0; i < num_leds; i++) {
|
||||
// cmd[2] == 1: set all leds other than this one to the stored color
|
||||
// cmd[2] == 2: this led and all previous leds, inwards
|
||||
// cmd[2] == 3: this led and all next leds, outwards
|
||||
if ((cmd[2] == 1 && i != cmd[3]) || (cmd[2] == 2 && i <= cmd[3]) ||
|
||||
(cmd[2] == 3 && i >= cmd[3])) {
|
||||
rgbled[cmd[1] - 2].setColorRGB(
|
||||
i, rgb[0], rgb[1], rgb[2]); // which led, red, green, blue
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Set one or more leds to the stored color using modulo
|
||||
// led offset: 0 = led closest to the GrovePi, counting outwards
|
||||
// modulo divisor: when 1 (default) sets stored color on all leds >= offset,
|
||||
// when 2 sets every 2nd led >= offset and so on [94, pin, led offset,
|
||||
// modulo divisor]
|
||||
else if (cmd[0] == 94) {
|
||||
// modulo divisor must be >= 1
|
||||
if (cmd[3] < 1) {
|
||||
cmd[3] = 1;
|
||||
}
|
||||
|
||||
// get the chain length
|
||||
byte num_leds = rgbled[cmd[1] - 2].getNumLeds();
|
||||
|
||||
// starting at the offset, step through each led and if the result of the
|
||||
// modulo operator results in zero, set the stored color on the led
|
||||
for (int i = cmd[2]; i < num_leds; i++) {
|
||||
// use modulo to set every n led
|
||||
if ((i - cmd[2]) % cmd[3] == 0) {
|
||||
rgbled[cmd[1] - 2].setColorRGB(i, rgb[0], rgb[1],
|
||||
rgb[2]); // which led, red, green, blue
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Set level (0 to num leds), counting outwards from the GrovePi, 0 = all
|
||||
// off, 1 = first led, reversible to count inwards [95, pin, level, reverse]
|
||||
else if (cmd[0] == 95) {
|
||||
// get the chain length
|
||||
byte num_leds = rgbled[cmd[1] - 2].getNumLeds();
|
||||
|
||||
if (cmd[3] == 0) {
|
||||
// outwards
|
||||
for (int i = 0; i < num_leds; i++) {
|
||||
if (cmd[2] > i) {
|
||||
rgbled[cmd[1] - 2].setColorRGB(
|
||||
i, rgb[0], rgb[1], rgb[2]); // which led, red, green, blue
|
||||
} else {
|
||||
rgbled[cmd[1] - 2].setColorRGB(i, 0, 0,
|
||||
0); // which led, red, green, blue
|
||||
}
|
||||
}
|
||||
} else {
|
||||
// inwards
|
||||
for (int i = num_leds; i > 0; i--) {
|
||||
if ((num_leds - cmd[2]) <= i) {
|
||||
rgbled[cmd[1] - 2].setColorRGB(
|
||||
i, rgb[0], rgb[1], rgb[2]); // which led, red, green, blue
|
||||
} else {
|
||||
rgbled[cmd[1] - 2].setColorRGB(i, 0, 0,
|
||||
0); // which led, red, green, blue
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (cmd[0] == ir_recv_pin_cmd) {
|
||||
Serial.print(cmd[1]);
|
||||
irrecv.setRecvpin(cmd[1]);
|
||||
irrecv.enableIRIn();
|
||||
cmd[0] = 0;
|
||||
} else if (cmd[0] == ir_read_cmd) {
|
||||
if (irrecv.decode(&results)) {
|
||||
b[0] = cmd[0];
|
||||
b[1] = results.decode_type;
|
||||
b[2] = results.address & 0xFF;
|
||||
b[3] = results.address >> 8;
|
||||
b[4] = results.value & 0xFF;
|
||||
b[5] = (results.value >> 8) & 0xFF;
|
||||
b[6] = (results.value >> 16) & 0xFF;
|
||||
b[7] = (results.value >> 24) & 0xFF;
|
||||
|
||||
irrecv.resume(); // Receive the next value
|
||||
} else {
|
||||
b[0] = cmd[0];
|
||||
b[1] = results.decode_type;
|
||||
}
|
||||
} else if (cmd[0] == ir_read_isdata) {
|
||||
b[0] = cmd[0];
|
||||
b[1] = irrecv.decode(&results);
|
||||
}
|
||||
} else if (cmd[0] == isr_set_cmd) {
|
||||
|
||||
if (run_once == 1) {
|
||||
uint8_t pin = cmd[1] & 0x0f; // 1st 4 bits to determine pin (D2->D8)
|
||||
const uint8_t ftype = (cmd[1] >> 4) & 0x03; // take 5th & 6th bits
|
||||
const uint8_t interrupt_mode = (cmd[1] >> 6) & 0x03; // take 7th and 8th bit
|
||||
const uint16_t period = (cmd[2] << 8) + cmd[3]; // get period in ms (maximum 65535 sms)
|
||||
|
||||
// detach pin if it's already set
|
||||
if (set_pcint[pin]) {
|
||||
detachISRPin(pin);
|
||||
}
|
||||
|
||||
pinMode(pin, INPUT_PULLUP);
|
||||
set_pcint[pin] = true;
|
||||
func_type[pin] = ftype;
|
||||
tracked_time[pin] = Tracked_time({period, 0});
|
||||
if (ftype == COUNT_CHANGES) {
|
||||
PcInt::attachInterrupt<uint8_t>(pin, isr_handler, (uint8_t*)&pins[pin], interrupt_mode, true);
|
||||
} else if (ftype == COUNT_LOW_DURATION) {
|
||||
pulse_counter[pin] = Pulse_counter({0, 0, 0});
|
||||
PcInt::attachInterrupt<uint8_t>(pin, isr_handler, (uint8_t*)&pins[pin], CHANGE, false);
|
||||
}
|
||||
run_once = 0;
|
||||
}
|
||||
|
||||
} else if (cmd[0] == isr_unset_cmd) {
|
||||
|
||||
if (run_once == 1) {
|
||||
// detach pin from PCINT
|
||||
const uint8_t pin = cmd[1];
|
||||
detachISRPin(pin);
|
||||
run_once = 0;
|
||||
}
|
||||
|
||||
} else if (cmd[0] == isr_read_cmd) {
|
||||
|
||||
if (run_once == 1) {
|
||||
const uint8_t pin = cmd[1]; // from pin D2->D8
|
||||
const uint32_t val = buffer[pin];
|
||||
b[0] = cmd[0];
|
||||
b[1] = val & 0xff;
|
||||
b[2] = (val >> 8) & 0xff;
|
||||
b[3] = (val >> 16) & 0xff;
|
||||
b[4] = (val >> 24) & 0xff;
|
||||
run_once = 0;
|
||||
}
|
||||
|
||||
} else if (cmd[0] == isr_clear_cmd) {
|
||||
|
||||
if (run_once == 1) {
|
||||
// detach all pins from PCINT
|
||||
for (int idx = 0; idx < total_ports; idx++) {
|
||||
if (set_pcint[idx]) {
|
||||
detachISRPin(idx);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
} else if (cmd[0] == isr_active_cmd) {
|
||||
|
||||
if (run_once == 1) {
|
||||
const uint8_t pin = cmd[1];
|
||||
|
||||
if (pin > total_ports) {
|
||||
uint16_t ports = 0;
|
||||
for (int idx = 0; idx < total_ports; idx++) {
|
||||
ports += (set_pcint[idx] & 0x01) << idx;
|
||||
}
|
||||
|
||||
b[0] = cmd[0];
|
||||
b[1] = ports & 0xff;
|
||||
b[2] = (ports >> 8) & 0xff;
|
||||
} else {
|
||||
b[0] = cmd[0];
|
||||
b[1] = 0;
|
||||
b[2] = (set_pcint[pin] & 0x01) << pin;
|
||||
}
|
||||
run_once = 0;
|
||||
}
|
||||
} else if (cmd[0] == encoder_en_cmd) {
|
||||
|
||||
if (run_once == 1) {
|
||||
const uint8_t pin = cmd[1];
|
||||
const uint8_t steps = cmd[2];
|
||||
if (pin < total_ports - 1) {
|
||||
// detach pin if it's already set
|
||||
if (set_pcint[pin]) {
|
||||
detachISRPin(pin);
|
||||
detachISRPin(pin + 1);
|
||||
}
|
||||
ge[pin].LOW_PIN = pin;
|
||||
ge[pin].HIGH_PIN = pin + 1;
|
||||
ge[pin].value = 0;
|
||||
ge[pin].MAX_VAL = steps;
|
||||
pinMode(ge[pin].LOW_PIN, INPUT_PULLUP);
|
||||
pinMode(ge[pin].HIGH_PIN, INPUT_PULLUP);
|
||||
set_pcint[pin] = set_pcint[pin + 1] = true;
|
||||
PcInt::attachInterrupt<GroveEncoder>(ge[pin].LOW_PIN, grove_encoder_handler, &ge[pin], FALLING, false);
|
||||
PcInt::attachInterrupt<GroveEncoder>(ge[pin].HIGH_PIN, grove_encoder_handler, &ge[pin], FALLING, false);
|
||||
}
|
||||
run_once = 0;
|
||||
}
|
||||
|
||||
} else if (cmd[0] == encoder_read_cmd) {
|
||||
|
||||
if (run_once == 1) {
|
||||
const uint8_t pin = cmd[1];
|
||||
const int value = ge[pin].value;
|
||||
Serial.print(value);
|
||||
Serial.print("\n");
|
||||
|
||||
b[0] = cmd[0];
|
||||
b[1] = value & 0xff;
|
||||
b[2] = (value >> 8) & 0xff;
|
||||
b[3] = (value >> 16) & 0xff;
|
||||
b[4] = (value >> 24) & 0xff;
|
||||
|
||||
run_once = 0;
|
||||
}
|
||||
|
||||
} else if (cmd[0] == encoder_dis_cmd) {
|
||||
|
||||
if (run_once == 1) {
|
||||
const uint8_t pin = cmd[1];
|
||||
if (pin < total_ports - 1) {
|
||||
detachISRPin(pin);
|
||||
detachISRPin(pin + 1);
|
||||
}
|
||||
run_once = 0;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
void loop() {
|
||||
if (need_extra_loop == true) {
|
||||
processIO();
|
||||
need_extra_loop = false;
|
||||
} else {
|
||||
processIO();
|
||||
}
|
||||
}
|
||||
|
||||
void receiveData(int byteCount) {
|
||||
if (!need_extra_loop) {
|
||||
while (Wire.available()) {
|
||||
if (Wire.available() == 4) {
|
||||
flag = 0;
|
||||
index = 0;
|
||||
run_once = 1;
|
||||
}
|
||||
cmd[index++] = Wire.read();
|
||||
}
|
||||
need_extra_loop = true;
|
||||
} else {
|
||||
flushI2C();
|
||||
}
|
||||
}
|
||||
|
||||
void flushI2C() {
|
||||
while (Wire.available())
|
||||
Wire.read();
|
||||
}
|
||||
|
||||
// Callback for sending data
|
||||
void sendData() {
|
||||
if (need_extra_loop == false) {
|
||||
if (cmd[0] == 1)
|
||||
Wire.write((byte *)b, 2);
|
||||
if (cmd[0] == 3 || cmd[0] == 7 || cmd[0] == 56)
|
||||
Wire.write((byte *)b, 3);
|
||||
if (cmd[0] == 8 || cmd[0] == 20)
|
||||
Wire.write((byte *)b, 4);
|
||||
if (cmd[0] == 30)
|
||||
Wire.write((byte *)b, 9);
|
||||
if (cmd[0] == 40)
|
||||
Wire.write((byte *)dht_b, 9);
|
||||
if (cmd[0] == isr_read_cmd)
|
||||
Wire.write((byte *)b, 5);
|
||||
if (cmd[0] == isr_active_cmd)
|
||||
Wire.write((byte *)b, 3);
|
||||
if (cmd[0] == ir_read_cmd) {
|
||||
Wire.write((byte *)b, 8);
|
||||
b[0] = 0;
|
||||
}
|
||||
if (cmd[0] == encoder_read_cmd) {
|
||||
Wire.write((byte *)b, 5);
|
||||
}
|
||||
if (cmd[0] == ir_read_isdata)
|
||||
Wire.write((byte *)b, 2);
|
||||
}
|
||||
// otherwise just reply the Pi telling
|
||||
// there's no data available yet
|
||||
else {
|
||||
Wire.write(data_not_available);
|
||||
}
|
||||
}
|
||||
|
||||
// detaching an PCINT interrupt from a given pin
|
||||
void detachISRPin(const uint8_t pin) {
|
||||
// detach pin from PCINT
|
||||
if (set_pcint[pin]) {
|
||||
PcInt::detachInterrupt(pin);
|
||||
pinMode(pin, OUTPUT);
|
||||
set_pcint[pin] = false;
|
||||
}
|
||||
}
|
||||
|
||||
// repeatedly called ISR to update values on each interrupt-enabled pin
|
||||
void isr_buffer_filler() {
|
||||
// PORTD |= 0x10;
|
||||
|
||||
const unsigned long current = micros() / 1000;
|
||||
// iterate over all possible interrupted pins
|
||||
for (int idx = 0; idx < total_ports; idx++) {
|
||||
// if the pin has been attached to an interrupt
|
||||
if (set_pcint[idx]) {
|
||||
const unsigned long elapsed_time = current - tracked_time[idx].last_update;
|
||||
// if more time than the associated period has passed
|
||||
if (elapsed_time >= tracked_time[idx].period) {
|
||||
// save and reset depending on the selected function
|
||||
switch (func_type[idx]) {
|
||||
case COUNT_CHANGES:
|
||||
buffer[idx] = change_counter[idx];
|
||||
change_counter[idx] = 0;
|
||||
break;
|
||||
case COUNT_LOW_DURATION:
|
||||
buffer[idx] = pulse_counter[idx].counted_duration * tracked_time[idx].period / elapsed_time;
|
||||
pulse_counter[idx].counted_duration = 0;
|
||||
break;
|
||||
}
|
||||
// update time
|
||||
tracked_time[idx].last_update = current;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// PORTD &= ~0x10;
|
||||
}
|
||||
|
||||
// interrupt handler for COUNT_CHANGES & COUNT_LOW_DURATION operations
|
||||
void isr_handler(uint8_t *userdata, bool newstate) {
|
||||
|
||||
// PORTD |= 0x10;
|
||||
|
||||
// const uint8_t pin = *((uint8_t*)userdata);
|
||||
const uint8_t pin = *userdata;
|
||||
|
||||
if (func_type[pin] == COUNT_CHANGES) {
|
||||
|
||||
// count changes
|
||||
change_counter[pin] ++;
|
||||
|
||||
} else if (func_type[pin] == COUNT_LOW_DURATION) {
|
||||
|
||||
// count duration
|
||||
if (newstate == 0) {
|
||||
pulse_counter[pin].pulse_start = micros();
|
||||
} else {
|
||||
pulse_counter[pin].pulse_end = micros();
|
||||
}
|
||||
const unsigned long pulse_end = pulse_counter[pin].pulse_end;
|
||||
const unsigned long pulse_start = pulse_counter[pin].pulse_start;
|
||||
if(pulse_end > pulse_start) {
|
||||
const unsigned long duration = int((pulse_end - pulse_start) / 1000); // to get ms
|
||||
pulse_counter[pin].counted_duration += duration;
|
||||
pulse_counter[pin].pulse_end = 0;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// PORTD &= ~0x10;
|
||||
|
||||
}
|
||||
|
||||
// interrupt handler for grove encoders
|
||||
void grove_encoder_handler(GroveEncoder *ge) {
|
||||
// PORTD |= 0x10;
|
||||
|
||||
const uint8_t valA = digitalRead(ge->LOW_PIN);
|
||||
const uint8_t valB = digitalRead(ge->HIGH_PIN);
|
||||
|
||||
if (valA + valB > 0 && valA + valB < 2) {
|
||||
if (valA < valB) {
|
||||
ge->value += 1;
|
||||
} else {
|
||||
ge->value -= 1;
|
||||
}
|
||||
}
|
||||
|
||||
if (ge->value < 0)
|
||||
ge->value = 0;
|
||||
else if (ge->value > ge->MAX_VAL)
|
||||
ge->value = ge->MAX_VAL;
|
||||
|
||||
Serial.println(ge->LOW_PIN);
|
||||
Serial.println(ge->HIGH_PIN);
|
||||
|
||||
// PORTD &= ~0x10;
|
||||
}
|
||||
203
Firmware/Source/grovepi/src/grovepi_refactored.ino
Normal file
203
Firmware/Source/grovepi/src/grovepi_refactored.ino
Normal file
|
|
@ -0,0 +1,203 @@
|
|||
// #include <Wire.h>
|
||||
// #include "DHT.h"
|
||||
// #include "Grove_LED_Bar.h"
|
||||
// #include "TM1637.h"
|
||||
// #include "ChainableLED.h"
|
||||
// #include "Encoder.h"
|
||||
// #include "TimerOne.h"
|
||||
// #include "Queue.h"
|
||||
//
|
||||
// #define SLAVE_ADDRESS 0x04
|
||||
// #define MAX_PAYLOAD 10
|
||||
//
|
||||
// enum Command : uint8_t {
|
||||
// DIGITALREAD = 1,
|
||||
// DIGITALWRITE = 2,
|
||||
// ANALOGREAD = 3,
|
||||
// ANALOGWRITE = 4,
|
||||
// PINMODE = 5,
|
||||
// ULTRASONICREAD = 7,
|
||||
// VERSION = 8,
|
||||
//
|
||||
// DUSTSENSOR_READ = 10,
|
||||
// DUSTSENSOR_EN = 14,
|
||||
// DUSTSENSOR_DIS = 15,
|
||||
// ENCODER_READ = 11,
|
||||
// ENCODER_EN = 16,
|
||||
// ENCODER_DIS = 17,
|
||||
// FLOW_READ = 12,
|
||||
// FLOW_EN = 18,
|
||||
// FLOW_DIS = 13,
|
||||
// IR_READ = 21,
|
||||
// IR_SET_PIN = 22,
|
||||
//
|
||||
// ACC_XYZ = 20,
|
||||
// RTC_GETTIME = 30,
|
||||
// DHT_TEMP = 40,
|
||||
//
|
||||
// LEDBAR_INIT = 50,
|
||||
// LEDBAR_ORIENT = 51,
|
||||
// LEDBAR_LEVEL = 52,
|
||||
// LEDBAR_SETTONE = 53,
|
||||
// LEDBAR_TOGGLEONE = 54,
|
||||
// LEDBAR_SETALL = 55,
|
||||
// LEDBAR_GETALL = 56,
|
||||
//
|
||||
// F4DIGIT_INIT = 70,
|
||||
// F4DIGIT_SET_BRIGHTNESS = 71,
|
||||
// F4DIGIT_VALUE_SET = 72,
|
||||
// F4DIGIT_VALUE_SET_WZEROS = 73,
|
||||
// F4DIGIT_INDIVID_DIGITS = 74,
|
||||
// F4DIGIT_INDIVID_LEDS = 75,
|
||||
// F4DIGIT_SCORE = 76,
|
||||
// F4DIGIT_ANALOGREAD_SECONDS = 77,
|
||||
// F4DIGIT_ALLON = 78,
|
||||
// F4DIGIT_ALLOFF = 79,
|
||||
//
|
||||
// STORE_COLOR = 90,
|
||||
// CHAINRGB_INIT = 91,
|
||||
// CHAINRGB_TEST = 92,
|
||||
// CHAINRGB_SET_PATTN = 93,
|
||||
// CHAINRGB_SET_MOD = 94,
|
||||
// CHAINRGB_SET_LVL = 95
|
||||
// };
|
||||
//
|
||||
// struct Request{
|
||||
// uint8_t command_id;
|
||||
// uint8_t no_bytes;
|
||||
// uint8_t payload[MAX_PAYLOAD];
|
||||
//
|
||||
// Request& operator=(const Request &d)
|
||||
// {
|
||||
// this->command_id = d.command_id;
|
||||
// this->no_bytes = d.no_bytes;
|
||||
// for(uint8_t i = 0; i < this->no_bytes; i++)
|
||||
// this->payload[i] = d.payload[i];
|
||||
//
|
||||
// return *this;
|
||||
// }
|
||||
// };
|
||||
//
|
||||
// volatile Request request, response;
|
||||
// uint8_t main_counter = 0;
|
||||
// volatile uint8_t receive_counter = 0, send_counter = 0;
|
||||
// volatile bool busy_loop = false;
|
||||
// volatile bool new_request = false;
|
||||
// volatile bool new_response = false;
|
||||
// bool stage_response = false;
|
||||
//
|
||||
// uint8_t pin;
|
||||
// uint8_t value8;
|
||||
// uint16_t value16;
|
||||
//
|
||||
// void flushI2C();
|
||||
//
|
||||
// void setup()
|
||||
// {
|
||||
// Serial.begin(38400); // start serial for output
|
||||
//
|
||||
// Wire.begin(SLAVE_ADDRESS);
|
||||
// Wire.onReceive(receiveData);
|
||||
// Wire.onRequest(sendData);
|
||||
// }
|
||||
//
|
||||
// void loop()
|
||||
// {
|
||||
// if(new_request == true)
|
||||
// {
|
||||
// busy_loop = true;
|
||||
// new_response = false;
|
||||
//
|
||||
// Serial.print("CID=" + String(request.command_id) + "+Bytes=" + String(request.no_bytes) + "+List=");
|
||||
// main_counter = 0;
|
||||
// while(main_counter < request.no_bytes)
|
||||
// {
|
||||
// Serial.print("" + String(request.payload[main_counter]));
|
||||
// main_counter++;
|
||||
// }
|
||||
// Serial.println();
|
||||
//
|
||||
// stage_response = false;
|
||||
// response.command_id = request.command_id;
|
||||
// switch(request.command_id)
|
||||
// {
|
||||
// case DIGITALREAD:
|
||||
// response.no_bytes = 1;
|
||||
// response.payload[0] = digitalRead(request.payload[0]);
|
||||
// stage_response = true;
|
||||
// break;
|
||||
// case DIGITALWRITE:
|
||||
// pin = request.payload[0];
|
||||
// value8 = request.payload[1];
|
||||
// digitalWrite(pin, value8);
|
||||
// break;
|
||||
// case ANALOGREAD:
|
||||
// response.no_bytes = 2;
|
||||
// value16 = analogRead(request.payload[0]);
|
||||
// response.payload[0] = value16 >> 8;
|
||||
// response.payload[1] = value16 & 0xFF;
|
||||
// stage_response = true;
|
||||
// break;
|
||||
// case ANALOGWRITE:
|
||||
// pin = request.payload[0];
|
||||
// value8 = request.payload[1];
|
||||
// analogWrite(pin, value8);
|
||||
// break;
|
||||
// }
|
||||
//
|
||||
// new_request = false;
|
||||
// busy_loop = false;
|
||||
// new_response = stage_response;
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// void receiveData(int byteCount)
|
||||
// {
|
||||
// // discard any incoming data if the main loop is busy
|
||||
// if(busy_loop == true) flushI2C();
|
||||
// else
|
||||
// {
|
||||
// if(byteCount >= 2)
|
||||
// {
|
||||
// request.command_id = Wire.read();
|
||||
// request.no_bytes = Wire.read();
|
||||
// byteCount -= 2;
|
||||
//
|
||||
// if(byteCount == request.no_bytes)
|
||||
// {
|
||||
// receive_counter = 0;
|
||||
// while(receive_counter < byteCount)
|
||||
// {
|
||||
// request.payload[receive_counter] = Wire.read();
|
||||
// receive_counter++;
|
||||
// }
|
||||
// new_request = true;
|
||||
// }
|
||||
// else flushI2C();
|
||||
// }
|
||||
// else flushI2C();
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// void sendData()
|
||||
// {
|
||||
// if(new_response == true)
|
||||
// {
|
||||
// Wire.write(response.command_id);
|
||||
// Wire.write(response.no_bytes);
|
||||
// send_counter = 0;
|
||||
// while(send_counter < response.no_bytes)
|
||||
// {
|
||||
// Wire.write(response.payload[send_counter]);
|
||||
// send_counter++;
|
||||
// }
|
||||
//
|
||||
// new_response = false;
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// void flushI2C()
|
||||
// {
|
||||
// while(Wire.available())
|
||||
// Wire.read();
|
||||
// }
|
||||
513
Firmware/Source/grovepi/src/irPronto.cpp
Normal file
513
Firmware/Source/grovepi/src/irPronto.cpp
Normal file
|
|
@ -0,0 +1,513 @@
|
|||
#define TEST 0
|
||||
|
||||
#if TEST
|
||||
# define SEND_PRONTO 1
|
||||
# define PRONTO_ONCE false
|
||||
# define PRONTO_REPEAT true
|
||||
# define PRONTO_FALLBACK true
|
||||
# define PRONTO_NOFALLBACK false
|
||||
#endif
|
||||
|
||||
#if SEND_PRONTO
|
||||
|
||||
//******************************************************************************
|
||||
#if TEST
|
||||
# include <stdio.h>
|
||||
void enableIROut (int freq) { printf("\nFreq = %d KHz\n", freq); }
|
||||
void mark (int t) { printf("+%d," , t); }
|
||||
void space (int t) { printf("-%d, ", t); }
|
||||
#else
|
||||
# include "IRremote.h"
|
||||
#endif // TEST
|
||||
|
||||
//+=============================================================================
|
||||
// Check for a valid hex digit
|
||||
//
|
||||
bool ishex (char ch)
|
||||
{
|
||||
return ( ((ch >= '0') && (ch <= '9')) ||
|
||||
((ch >= 'A') && (ch <= 'F')) ||
|
||||
((ch >= 'a') && (ch <= 'f')) ) ? true : false ;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Check for a valid "blank" ... '\0' is a valid "blank"
|
||||
//
|
||||
bool isblank (char ch)
|
||||
{
|
||||
return ((ch == ' ') || (ch == '\t') || (ch == '\0')) ? true : false ;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Bypass spaces
|
||||
//
|
||||
bool byp (char** pcp)
|
||||
{
|
||||
while (isblank(**pcp)) (*pcp)++ ;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Hex-to-Byte : Decode a hex digit
|
||||
// We assume the character has already been validated
|
||||
//
|
||||
uint8_t htob (char ch)
|
||||
{
|
||||
if ((ch >= '0') && (ch <= '9')) return ch - '0' ;
|
||||
if ((ch >= 'A') && (ch <= 'F')) return ch - 'A' + 10 ;
|
||||
if ((ch >= 'a') && (ch <= 'f')) return ch - 'a' + 10 ;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Hex-to-Word : Decode a block of 4 hex digits
|
||||
// We assume the string has already been validated
|
||||
// and the pointer being passed points at the start of a block of 4 hex digits
|
||||
//
|
||||
uint16_t htow (char* cp)
|
||||
{
|
||||
return ( (htob(cp[0]) << 12) | (htob(cp[1]) << 8) |
|
||||
(htob(cp[2]) << 4) | (htob(cp[3]) ) ) ;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
//
|
||||
bool sendPronto (char* s, bool repeat, bool fallback)
|
||||
{
|
||||
int i;
|
||||
int len;
|
||||
int skip;
|
||||
char* cp;
|
||||
uint16_t freq; // Frequency in KHz
|
||||
uint8_t usec; // pronto uSec/tick
|
||||
uint8_t once;
|
||||
uint8_t rpt;
|
||||
|
||||
// Validate the string
|
||||
for (cp = s; *cp; cp += 4) {
|
||||
byp(&cp);
|
||||
if ( !ishex(cp[0]) || !ishex(cp[1]) ||
|
||||
!ishex(cp[2]) || !ishex(cp[3]) || !isblank(cp[4]) ) return false ;
|
||||
}
|
||||
|
||||
// We will use cp to traverse the string
|
||||
cp = s;
|
||||
|
||||
// Check mode = Oscillated/Learned
|
||||
byp(&cp);
|
||||
if (htow(cp) != 0000) return false;
|
||||
cp += 4;
|
||||
|
||||
// Extract & set frequency
|
||||
byp(&cp);
|
||||
freq = (int)(1000000 / (htow(cp) * 0.241246)); // Rounding errors will occur, tolerance is +/- 10%
|
||||
usec = (int)(((1.0 / freq) * 1000000) + 0.5); // Another rounding error, thank Cod for analogue electronics
|
||||
freq /= 1000; // This will introduce a(nother) rounding error which we do not want in the usec calcualtion
|
||||
cp += 4;
|
||||
|
||||
// Get length of "once" code
|
||||
byp(&cp);
|
||||
once = htow(cp);
|
||||
cp += 4;
|
||||
|
||||
// Get length of "repeat" code
|
||||
byp(&cp);
|
||||
rpt = htow(cp);
|
||||
cp += 4;
|
||||
|
||||
// Which code are we sending?
|
||||
if (fallback) { // fallback on the "other" code if "this" code is not present
|
||||
if (!repeat) { // requested 'once'
|
||||
if (once) len = once * 2, skip = 0 ; // if once exists send it
|
||||
else len = rpt * 2, skip = 0 ; // else send repeat code
|
||||
} else { // requested 'repeat'
|
||||
if (rpt) len = rpt * 2, skip = 0 ; // if rpt exists send it
|
||||
else len = once * 2, skip = 0 ; // else send once code
|
||||
}
|
||||
} else { // Send what we asked for, do not fallback if the code is empty!
|
||||
if (!repeat) len = once * 2, skip = 0 ; // 'once' starts at 0
|
||||
else len = rpt * 2, skip = once ; // 'repeat' starts where 'once' ends
|
||||
}
|
||||
|
||||
// Skip to start of code
|
||||
for (i = 0; i < skip; i++, cp += 4) byp(&cp) ;
|
||||
|
||||
// Send code
|
||||
enableIROut(freq);
|
||||
for (i = 0; i < len; i++) {
|
||||
byp(&cp);
|
||||
if (i & 1) space(htow(cp) * usec);
|
||||
else mark (htow(cp) * usec);
|
||||
cp += 4;
|
||||
}
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
#if TEST
|
||||
|
||||
int main ( )
|
||||
{
|
||||
char prontoTest[] =
|
||||
"0000 0070 0000 0032 0080 0040 0010 0010 0010 0030 " // 10
|
||||
"0010 0010 0010 0010 0010 0010 0010 0010 0010 0010 " // 20
|
||||
"0010 0010 0010 0010 0010 0010 0010 0010 0010 0010 " // 30
|
||||
"0010 0010 0010 0030 0010 0010 0010 0010 0010 0010 " // 40
|
||||
"0010 0010 0010 0010 0010 0010 0010 0010 0010 0010 " // 50
|
||||
"0010 0010 0010 0030 0010 0010 0010 0010 0010 0010 " // 60
|
||||
"0010 0010 0010 0010 0010 0010 0010 0010 0010 0010 " // 70
|
||||
"0010 0010 0010 0030 0010 0010 0010 0030 0010 0010 " // 80
|
||||
"0010 0010 0010 0030 0010 0010 0010 0010 0010 0030 " // 90
|
||||
"0010 0010 0010 0030 0010 0010 0010 0010 0010 0030 " // 100
|
||||
"0010 0030 0010 0aa6"; // 104
|
||||
|
||||
sendPronto(prontoTest, PRONTO_ONCE, PRONTO_FALLBACK); // once code
|
||||
sendPronto(prontoTest, PRONTO_REPEAT, PRONTO_FALLBACK); // repeat code
|
||||
sendPronto(prontoTest, PRONTO_ONCE, PRONTO_NOFALLBACK); // once code
|
||||
sendPronto(prontoTest, PRONTO_REPEAT, PRONTO_NOFALLBACK); // repeat code
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#endif // TEST
|
||||
|
||||
#endif // SEND_PRONTO
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
#if 0
|
||||
//******************************************************************************
|
||||
// Sources:
|
||||
// http://www.remotecentral.com/features/irdisp2.htm
|
||||
// http://www.hifi-remote.com/wiki/index.php?title=Working_With_Pronto_Hex
|
||||
//******************************************************************************
|
||||
|
||||
#include <stdint.h>
|
||||
#include <stdio.h>
|
||||
|
||||
#define IRPRONTO
|
||||
#include "IRremoteInt.h" // The Arduino IRremote library defines USECPERTICK
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Source: https://www.google.co.uk/search?q=DENON+MASTER+IR+Hex+Command+Sheet
|
||||
// -> http://assets.denon.com/documentmaster/us/denon%20master%20ir%20hex.xls
|
||||
//
|
||||
char prontoTest[] =
|
||||
"0000 0070 0000 0032 0080 0040 0010 0010 0010 0030 " // 10
|
||||
"0010 0010 0010 0010 0010 0010 0010 0010 0010 0010 " // 20
|
||||
"0010 0010 0010 0010 0010 0010 0010 0010 0010 0010 " // 30
|
||||
"0010 0010 0010 0030 0010 0010 0010 0010 0010 0010 " // 40
|
||||
"0010 0010 0010 0010 0010 0010 0010 0010 0010 0010 " // 50
|
||||
"0010 0010 0010 0030 0010 0010 0010 0010 0010 0010 " // 60
|
||||
"0010 0010 0010 0010 0010 0010 0010 0010 0010 0010 " // 70
|
||||
"0010 0010 0010 0030 0010 0010 0010 0030 0010 0010 " // 80
|
||||
"0010 0010 0010 0030 0010 0010 0010 0010 0010 0030 " // 90
|
||||
"0010 0010 0010 0030 0010 0010 0010 0010 0010 0030 " // 100
|
||||
"0010 0030 0010 0aa6"; // 104
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// This is the longest code we can support
|
||||
#define CODEMAX 200
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// This is the data we pull out of the pronto code
|
||||
typedef
|
||||
struct {
|
||||
int freq; // Carrier frequency (in Hz)
|
||||
int usec; // uSec per tick (based on freq)
|
||||
|
||||
int codeLen; // Length of code
|
||||
uint16_t code[CODEMAX]; // Code in hex
|
||||
|
||||
int onceLen; // Length of "once" transmit
|
||||
uint16_t* once; // Pointer to start within 'code'
|
||||
|
||||
int rptLen; // Length of "repeat" transmit
|
||||
uint16_t* rpt; // Pointer to start within 'code'
|
||||
}
|
||||
pronto_t;
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// From what I have seen, the only time we go over 8-bits is the 'space'
|
||||
// on the end which creates the lead-out/inter-code gap. Assuming I'm right,
|
||||
// we can code this up as a special case and otherwise halve the size of our
|
||||
// data!
|
||||
// Ignoring the first four values (the config data) and the last value
|
||||
// (the lead-out), if you find a protocol that uses values greater than 00fe
|
||||
// we are going to have to revisit this code!
|
||||
//
|
||||
//
|
||||
// So, the 0th byte will be the carrier frequency in Khz (NOT Hz)
|
||||
// " 1st " " " " length of the "once" code
|
||||
// " 2nd " " " " length of the "repeat" code
|
||||
//
|
||||
// Thereafter, odd bytes will be Mark lengths as a multiple of USECPERTICK uS
|
||||
// even " " " Space " " " " " " "
|
||||
//
|
||||
// Any occurence of "FF" in either a Mark or a Space will indicate
|
||||
// "Use the 16-bit FF value" which will also be a multiple of USECPERTICK uS
|
||||
//
|
||||
//
|
||||
// As a point of comparison, the test code (prontoTest[]) is 520 bytes
|
||||
// (yes, more than 0.5KB of our Arduino's precious 32KB) ... after conversion
|
||||
// to pronto hex that goes down to ((520/5)*2) = 208 bytes ... once converted to
|
||||
// our format we are down to ((208/2) -1 -1 +2) = 104 bytes
|
||||
//
|
||||
// In fariness this is still very memory-hungry
|
||||
// ...As a rough guide:
|
||||
// 10 codes cost 1K of memory (this will vary depending on the protocol).
|
||||
//
|
||||
// So if you're building a complex remote control, you will probably need to
|
||||
// keep the codes on an external memory device (not in the Arduino sketch) and
|
||||
// load them as you need them. Hmmm.
|
||||
//
|
||||
// This dictates that "Oscillated Pronto Codes" are probably NOT the way forward
|
||||
//
|
||||
// For example, prontoTest[] happens to be: A 48-bit IR code in Denon format
|
||||
// So we know it starts with 80/40 (Denon header)
|
||||
// and ends with 10/aa6 (Denon leadout)
|
||||
// and all (48) bits in between are either 10/10 (Denon 0)
|
||||
// or 10/30 (Denon 1)
|
||||
// So we could easily store this data in 1-byte ("Denon")
|
||||
// + 1-byte (Length=48)
|
||||
// + 6-bytes (IR code)
|
||||
// At 8-bytes per code, we can store 128 codes in 1KB or memory - that's a lot
|
||||
// better than the 2 (two) we started off with!
|
||||
//
|
||||
// And serendipitously, by reducing the amount of data, our program will run
|
||||
// a LOT faster!
|
||||
//
|
||||
// Again, I repeat, even after you have spent time converting the "Oscillated
|
||||
// Pronto Codes" in to IRremote format, it will be a LOT more memory-hungry
|
||||
// than using sendDenon() (or whichever) ...BUT these codes are easily
|
||||
// available on the internet, so we'll support them!
|
||||
//
|
||||
typedef
|
||||
struct {
|
||||
uint16_t FF;
|
||||
uint8_t code[CODEMAX];
|
||||
}
|
||||
irCode_t;
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
#define DEBUGF(...) printf(__VA_ARGS__)
|
||||
|
||||
//+=============================================================================
|
||||
// String must be block of 4 hex digits separated with blanks
|
||||
//
|
||||
bool validate (char* cp, int* len)
|
||||
{
|
||||
for (*len = 0; *cp; (*len)++, cp += 4) {
|
||||
byp(&cp);
|
||||
if ( !ishex(cp[0]) || !ishex(cp[1]) ||
|
||||
!ishex(cp[2]) || !ishex(cp[3]) || !isblank(cp[4]) ) return false ;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Hex-to-Byte : Decode a hex digit
|
||||
// We assume the character has already been validated
|
||||
//
|
||||
uint8_t htob (char ch)
|
||||
{
|
||||
if ((ch >= '0') && (ch <= '9')) return ch - '0' ;
|
||||
if ((ch >= 'A') && (ch <= 'F')) return ch - 'A' + 10 ;
|
||||
if ((ch >= 'a') && (ch <= 'f')) return ch - 'a' + 10 ;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Hex-to-Word : Decode a block of 4 hex digits
|
||||
// We assume the string has already been validated
|
||||
// and the pointer being passed points at the start of a block of 4 hex digits
|
||||
//
|
||||
uint16_t htow (char* cp)
|
||||
{
|
||||
return ( (htob(cp[0]) << 12) | (htob(cp[1]) << 8) |
|
||||
(htob(cp[2]) << 4) | (htob(cp[3]) ) ) ;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Convert the pronto string in to data
|
||||
//
|
||||
bool decode (char* s, pronto_t* p, irCode_t* ir)
|
||||
{
|
||||
int i, len;
|
||||
char* cp;
|
||||
|
||||
// Validate the Pronto string
|
||||
if (!validate(s, &p->codeLen)) {
|
||||
DEBUGF("Invalid pronto string\n");
|
||||
return false ;
|
||||
}
|
||||
DEBUGF("Found %d hex codes\n", p->codeLen);
|
||||
|
||||
// Allocate memory to store the decoded string
|
||||
//if (!(p->code = malloc(p->len))) {
|
||||
// DEBUGF("Memory allocation failed\n");
|
||||
// return false ;
|
||||
//}
|
||||
|
||||
// Check in case our code is too long
|
||||
if (p->codeLen > CODEMAX) {
|
||||
DEBUGF("Code too long, edit CODEMAX and recompile\n");
|
||||
return false ;
|
||||
}
|
||||
|
||||
// Decode the string
|
||||
cp = s;
|
||||
for (i = 0; i < p->codeLen; i++, cp += 4) {
|
||||
byp(&cp);
|
||||
p->code[i] = htow(cp);
|
||||
}
|
||||
|
||||
// Announce our findings
|
||||
DEBUGF("Input: |%s|\n", s);
|
||||
DEBUGF("Found: |");
|
||||
for (i = 0; i < p->codeLen; i++) DEBUGF("%04x ", p->code[i]) ;
|
||||
DEBUGF("|\n");
|
||||
|
||||
DEBUGF("Form [%04X] : ", p->code[0]);
|
||||
if (p->code[0] == 0x0000) DEBUGF("Oscillated (Learned)\n");
|
||||
else if (p->code[0] == 0x0100) DEBUGF("Unmodulated\n");
|
||||
else DEBUGF("Unknown\n");
|
||||
if (p->code[0] != 0x0000) return false ; // Can only handle Oscillated
|
||||
|
||||
// Calculate the carrier frequency (+/- 10%) & uSecs per pulse
|
||||
// Pronto uses a crystal which generates a timeabse of 0.241246
|
||||
p->freq = (int)(1000000 / (p->code[1] * 0.241246));
|
||||
p->usec = (int)(((1.0 / p->freq) * 1000000) + 0.5);
|
||||
ir->code[0] = p->freq / 1000;
|
||||
DEBUGF("Freq [%04X] : %d Hz (%d uS/pluse) -> %d KHz\n",
|
||||
p->code[1], p->freq, p->usec, ir->code[0]);
|
||||
|
||||
// Set the length & start pointer for the "once" code
|
||||
p->onceLen = p->code[2];
|
||||
p->once = &p->code[4];
|
||||
ir->code[1] = p->onceLen;
|
||||
DEBUGF("Once [%04X] : %d\n", p->code[2], p->onceLen);
|
||||
|
||||
// Set the length & start pointer for the "repeat" code
|
||||
p->rptLen = p->code[3];
|
||||
p->rpt = &p->code[4 + p->onceLen];
|
||||
ir->code[2] = p->rptLen;
|
||||
DEBUGF("Rpt [%04X] : %d\n", p->code[3], p->rptLen);
|
||||
|
||||
// Check everything tallies
|
||||
if (1 + 1 + 1 + 1 + (p->onceLen * 2) + (p->rptLen * 2) != p->codeLen) {
|
||||
DEBUGF("Bad code length\n");
|
||||
return false;
|
||||
}
|
||||
|
||||
// Convert the IR data to our new format
|
||||
ir->FF = p->code[p->codeLen - 1];
|
||||
|
||||
len = (p->onceLen * 2) + (p->rptLen * 2);
|
||||
DEBUGF("Encoded: |");
|
||||
for (i = 0; i < len; i++) {
|
||||
if (p->code[i+4] == ir->FF) {
|
||||
ir->code[i+3] = 0xFF;
|
||||
} else if (p->code[i+4] > 0xFE) {
|
||||
DEBUGF("\n%04X : Mark/Space overflow\n", p->code[i+4]);
|
||||
return false;
|
||||
} else {
|
||||
ir->code[i+3] = (p->code[i+4] * p->usec) / USECPERTICK;
|
||||
}
|
||||
DEBUGF("%s%d", !i ? "" : (i&1 ? "," : ", "), ir->code[i+3]);
|
||||
}
|
||||
DEBUGF("|\n");
|
||||
|
||||
ir->FF = (ir->FF * p->usec) / USECPERTICK;
|
||||
DEBUGF("FF -> %d\n", ir->FF);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
//
|
||||
void irDump (irCode_t* ir)
|
||||
{
|
||||
int i, len;
|
||||
|
||||
printf("uint8_t buttonName[%d] = {", len);
|
||||
|
||||
printf("%d,%d, ", (ir->FF >> 8), ir->FF & 0xFF);
|
||||
printf("%d,%d,%d, ", ir->code[0], ir->code[1], ir->code[2]);
|
||||
|
||||
len = (ir->code[1] * 2) + (ir->code[2] * 2);
|
||||
for (i = 0; i < len; i++) {
|
||||
printf("%s%d", !i ? "" : (i&1 ? "," : ", "), ir->code[i+3]);
|
||||
}
|
||||
|
||||
printf("};\n");
|
||||
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
//
|
||||
int main ( )
|
||||
{
|
||||
pronto_t pCode;
|
||||
irCode_t irCode;
|
||||
|
||||
decode(prontoTest, &pCode, &irCode);
|
||||
|
||||
irDump(&irCode);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
#endif //0
|
||||
258
Firmware/Source/grovepi/src/irRecv.cpp
Normal file
258
Firmware/Source/grovepi/src/irRecv.cpp
Normal file
|
|
@ -0,0 +1,258 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
#ifdef IR_TIMER_USE_ESP32
|
||||
hw_timer_t *timer;
|
||||
void IRTimer(); // defined in IRremote.cpp
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
// Decodes the received IR message
|
||||
// Returns 0 if no data ready, 1 if data ready.
|
||||
// Results of decoding are stored in results
|
||||
//
|
||||
int IRrecv::decode (decode_results *results)
|
||||
{
|
||||
results->rawbuf = irparams.rawbuf;
|
||||
results->rawlen = irparams.rawlen;
|
||||
|
||||
results->overflow = irparams.overflow;
|
||||
|
||||
if (irparams.rcvstate != STATE_STOP) return false ;
|
||||
|
||||
#if DECODE_NEC
|
||||
DBG_PRINTLN("Attempting NEC decode");
|
||||
if (decodeNEC(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_SONY
|
||||
DBG_PRINTLN("Attempting Sony decode");
|
||||
if (decodeSony(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_SANYO
|
||||
DBG_PRINTLN("Attempting Sanyo decode");
|
||||
if (decodeSanyo(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_MITSUBISHI
|
||||
DBG_PRINTLN("Attempting Mitsubishi decode");
|
||||
if (decodeMitsubishi(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_RC5
|
||||
DBG_PRINTLN("Attempting RC5 decode");
|
||||
if (decodeRC5(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_RC6
|
||||
DBG_PRINTLN("Attempting RC6 decode");
|
||||
if (decodeRC6(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_PANASONIC
|
||||
DBG_PRINTLN("Attempting Panasonic decode");
|
||||
if (decodePanasonic(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_LG
|
||||
DBG_PRINTLN("Attempting LG decode");
|
||||
if (decodeLG(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_JVC
|
||||
DBG_PRINTLN("Attempting JVC decode");
|
||||
if (decodeJVC(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_SAMSUNG
|
||||
DBG_PRINTLN("Attempting SAMSUNG decode");
|
||||
if (decodeSAMSUNG(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_WHYNTER
|
||||
DBG_PRINTLN("Attempting Whynter decode");
|
||||
if (decodeWhynter(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_AIWA_RC_T501
|
||||
DBG_PRINTLN("Attempting Aiwa RC-T501 decode");
|
||||
if (decodeAiwaRCT501(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_DENON
|
||||
DBG_PRINTLN("Attempting Denon decode");
|
||||
if (decodeDenon(results)) return true ;
|
||||
#endif
|
||||
|
||||
#if DECODE_LEGO_PF
|
||||
DBG_PRINTLN("Attempting Lego Power Functions");
|
||||
if (decodeLegoPowerFunctions(results)) return true ;
|
||||
#endif
|
||||
|
||||
// decodeHash returns a hash on any input.
|
||||
// Thus, it needs to be last in the list.
|
||||
// If you add any decodes, add them before this.
|
||||
if (decodeHash(results)) return true ;
|
||||
|
||||
// Throw away and start over
|
||||
resume();
|
||||
return false;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
IRrecv::IRrecv ()
|
||||
{
|
||||
irparams.recvpin = 2; // set the port by default to 2
|
||||
irparams.blinkflag = 0;
|
||||
}
|
||||
|
||||
IRrecv::IRrecv (int recvpin)
|
||||
{
|
||||
irparams.recvpin = recvpin;
|
||||
irparams.blinkflag = 0;
|
||||
}
|
||||
|
||||
IRrecv::IRrecv (int recvpin, int blinkpin)
|
||||
{
|
||||
irparams.recvpin = recvpin;
|
||||
irparams.blinkpin = blinkpin;
|
||||
pinMode(blinkpin, OUTPUT);
|
||||
irparams.blinkflag = 0;
|
||||
}
|
||||
|
||||
|
||||
|
||||
//+=============================================================================
|
||||
// initialization
|
||||
//
|
||||
void IRrecv::enableIRIn ( )
|
||||
{
|
||||
// Interrupt Service Routine - Fires every 50uS
|
||||
#ifdef ESP32
|
||||
// ESP32 has a proper API to setup timers, no weird chip macros needed
|
||||
// simply call the readable API versions :)
|
||||
// 3 timers, choose #1, 80 divider nanosecond precision, 1 to count up
|
||||
timer = timerBegin(1, 80, 1);
|
||||
timerAttachInterrupt(timer, &IRTimer, 1);
|
||||
// every 50ns, autoreload = true
|
||||
timerAlarmWrite(timer, 50, true);
|
||||
timerAlarmEnable(timer);
|
||||
#else
|
||||
cli();
|
||||
// Setup pulse clock timer interrupt
|
||||
// Prescale /8 (16M/8 = 0.5 microseconds per tick)
|
||||
// Therefore, the timer interval can range from 0.5 to 128 microseconds
|
||||
// Depending on the reset value (255 to 0)
|
||||
TIMER_CONFIG_NORMAL();
|
||||
|
||||
// Timer2 Overflow Interrupt Enable
|
||||
TIMER_ENABLE_INTR;
|
||||
|
||||
TIMER_RESET;
|
||||
|
||||
sei(); // enable interrupts
|
||||
#endif
|
||||
|
||||
// Initialize state machine variables
|
||||
irparams.rcvstate = STATE_IDLE;
|
||||
irparams.rawlen = 0;
|
||||
|
||||
// Set pin modes
|
||||
pinMode(irparams.recvpin, INPUT);
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Enable/disable blinking of pin 13 on IR processing
|
||||
//
|
||||
void IRrecv::blink13 (int blinkflag)
|
||||
{
|
||||
irparams.blinkflag = blinkflag;
|
||||
if (blinkflag) pinMode(BLINKLED, OUTPUT) ;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Return if receiving new IR signals
|
||||
//
|
||||
bool IRrecv::isIdle ( )
|
||||
{
|
||||
return (irparams.rcvstate == STATE_IDLE || irparams.rcvstate == STATE_STOP) ? true : false;
|
||||
}
|
||||
//+=============================================================================
|
||||
// Restart the ISR state machine
|
||||
//
|
||||
void IRrecv::resume ( )
|
||||
{
|
||||
irparams.rcvstate = STATE_IDLE;
|
||||
irparams.rawlen = 0;
|
||||
}
|
||||
//+=============================================================================
|
||||
// Shutdown the ISR
|
||||
//
|
||||
void IRrecv::disableIR ( )
|
||||
{
|
||||
#ifdef ESP32
|
||||
#else
|
||||
TIMER_DISABLE_INTR;
|
||||
#endif
|
||||
}
|
||||
//+=============================================================================
|
||||
// Set the receive pin for the IR sensor
|
||||
//
|
||||
void IRrecv::setRecvpin(int recvpin)
|
||||
{
|
||||
irparams.recvpin = recvpin;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// hashdecode - decode an arbitrary IR code.
|
||||
// Instead of decoding using a standard encoding scheme
|
||||
// (e.g. Sony, NEC, RC5), the code is hashed to a 32-bit value.
|
||||
//
|
||||
// The algorithm: look at the sequence of MARK signals, and see if each one
|
||||
// is shorter (0), the same length (1), or longer (2) than the previous.
|
||||
// Do the same with the SPACE signals. Hash the resulting sequence of 0's,
|
||||
// 1's, and 2's to a 32-bit value. This will give a unique value for each
|
||||
// different code (probably), for most code systems.
|
||||
//
|
||||
// http://arcfn.com/2010/01/using-arbitrary-remotes-with-arduino.html
|
||||
//
|
||||
// Compare two tick values, returning 0 if newval is shorter,
|
||||
// 1 if newval is equal, and 2 if newval is longer
|
||||
// Use a tolerance of 20%
|
||||
//
|
||||
int IRrecv::compare (unsigned int oldval, unsigned int newval)
|
||||
{
|
||||
if (newval < oldval * .8) return 0 ;
|
||||
else if (oldval < newval * .8) return 2 ;
|
||||
else return 1 ;
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Use FNV hash algorithm: http://isthe.com/chongo/tech/comp/fnv/#FNV-param
|
||||
// Converts the raw code values into a 32-bit hash code.
|
||||
// Hopefully this code is unique for each button.
|
||||
// This isn't a "real" decoding, just an arbitrary value.
|
||||
//
|
||||
#define FNV_PRIME_32 16777619
|
||||
#define FNV_BASIS_32 2166136261
|
||||
|
||||
long IRrecv::decodeHash (decode_results *results)
|
||||
{
|
||||
long hash = FNV_BASIS_32;
|
||||
|
||||
// Require at least 6 samples to prevent triggering on noise
|
||||
if (results->rawlen < 6) return false ;
|
||||
|
||||
for (int i = 1; (i + 2) < results->rawlen; i++) {
|
||||
int value = compare(results->rawbuf[i], results->rawbuf[i+2]);
|
||||
// Add value into the hash
|
||||
hash = (hash * FNV_PRIME_32) ^ value;
|
||||
}
|
||||
|
||||
results->value = hash;
|
||||
results->bits = 32;
|
||||
results->decode_type = UNKNOWN;
|
||||
|
||||
return true;
|
||||
}
|
||||
90
Firmware/Source/grovepi/src/irSend.cpp
Normal file
90
Firmware/Source/grovepi/src/irSend.cpp
Normal file
|
|
@ -0,0 +1,90 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//+=============================================================================
|
||||
void IRsend::sendRaw (const unsigned int buf[], unsigned int len, unsigned int hz)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(hz);
|
||||
|
||||
for (unsigned int i = 0; i < len; i++) {
|
||||
if (i & 1) space(buf[i]) ;
|
||||
else mark (buf[i]) ;
|
||||
}
|
||||
|
||||
space(0); // Always end with the LED off
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Sends an IR mark for the specified number of microseconds.
|
||||
// The mark output is modulated at the PWM frequency.
|
||||
//
|
||||
void IRsend::mark (unsigned int time)
|
||||
{
|
||||
TIMER_ENABLE_PWM; // Enable pin 3 PWM output
|
||||
if (time > 0) custom_delay_usec(time);
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Leave pin off for time (given in microseconds)
|
||||
// Sends an IR space for the specified number of microseconds.
|
||||
// A space is no output, so the PWM output is disabled.
|
||||
//
|
||||
void IRsend::space (unsigned int time)
|
||||
{
|
||||
TIMER_DISABLE_PWM; // Disable pin 3 PWM output
|
||||
if (time > 0) IRsend::custom_delay_usec(time);
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
//+=============================================================================
|
||||
// Enables IR output. The khz value controls the modulation frequency in kilohertz.
|
||||
// The IR output will be on pin 3 (OC2B).
|
||||
// This routine is designed for 36-40KHz; if you use it for other values, it's up to you
|
||||
// to make sure it gives reasonable results. (Watch out for overflow / underflow / rounding.)
|
||||
// TIMER2 is used in phase-correct PWM mode, with OCR2A controlling the frequency and OCR2B
|
||||
// controlling the duty cycle.
|
||||
// There is no prescaling, so the output frequency is 16MHz / (2 * OCR2A)
|
||||
// To turn the output on and off, we leave the PWM running, but connect and disconnect the output pin.
|
||||
// A few hours staring at the ATmega documentation and this will all make sense.
|
||||
// See my Secrets of Arduino PWM at http://arcfn.com/2009/07/secrets-of-arduino-pwm.html for details.
|
||||
//
|
||||
void IRsend::enableIROut (int khz)
|
||||
{
|
||||
// FIXME: implement ESP32 support, see IR_TIMER_USE_ESP32 in boarddefs.h
|
||||
#ifndef ESP32
|
||||
// Disable the Timer2 Interrupt (which is used for receiving IR)
|
||||
TIMER_DISABLE_INTR; //Timer2 Overflow Interrupt
|
||||
|
||||
pinMode(TIMER_PWM_PIN, OUTPUT);
|
||||
digitalWrite(TIMER_PWM_PIN, LOW); // When not sending PWM, we want it low
|
||||
|
||||
// COM2A = 00: disconnect OC2A
|
||||
// COM2B = 00: disconnect OC2B; to send signal set to 10: OC2B non-inverted
|
||||
// WGM2 = 101: phase-correct PWM with OCRA as top
|
||||
// CS2 = 000: no prescaling
|
||||
// The top value for the timer. The modulation frequency will be SYSCLOCK / 2 / OCR2A.
|
||||
TIMER_CONFIG_KHZ(khz);
|
||||
#endif
|
||||
}
|
||||
|
||||
//+=============================================================================
|
||||
// Custom delay function that circumvents Arduino's delayMicroseconds limit
|
||||
|
||||
void IRsend::custom_delay_usec(unsigned long uSecs) {
|
||||
if (uSecs > 4) {
|
||||
unsigned long start = micros();
|
||||
unsigned long endMicros = start + uSecs - 4;
|
||||
if (endMicros < start) { // Check if overflow
|
||||
while ( micros() > start ) {} // wait until overflow
|
||||
}
|
||||
while ( micros() < endMicros ) {} // normal wait
|
||||
}
|
||||
//else {
|
||||
// __asm__("nop\n\t"); // must have or compiler optimizes out
|
||||
//}
|
||||
}
|
||||
|
||||
105
Firmware/Source/grovepi/src/ir_Aiwa.cpp
Normal file
105
Firmware/Source/grovepi/src/ir_Aiwa.cpp
Normal file
|
|
@ -0,0 +1,105 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// AAA IIIII W W AAA
|
||||
// A A I W W A A
|
||||
// AAAAA I W W W AAAAA
|
||||
// A A I W W W A A
|
||||
// A A IIIII WWW A A
|
||||
//==============================================================================
|
||||
|
||||
// Based off the RC-T501 RCU
|
||||
// Lirc file http://lirc.sourceforge.net/remotes/aiwa/RC-T501
|
||||
|
||||
#define AIWA_RC_T501_HZ 38
|
||||
#define AIWA_RC_T501_BITS 15
|
||||
#define AIWA_RC_T501_PRE_BITS 26
|
||||
#define AIWA_RC_T501_POST_BITS 1
|
||||
#define AIWA_RC_T501_SUM_BITS (AIWA_RC_T501_PRE_BITS + AIWA_RC_T501_BITS + AIWA_RC_T501_POST_BITS)
|
||||
#define AIWA_RC_T501_HDR_MARK 8800
|
||||
#define AIWA_RC_T501_HDR_SPACE 4500
|
||||
#define AIWA_RC_T501_BIT_MARK 500
|
||||
#define AIWA_RC_T501_ONE_SPACE 600
|
||||
#define AIWA_RC_T501_ZERO_SPACE 1700
|
||||
|
||||
//+=============================================================================
|
||||
#if SEND_AIWA_RC_T501
|
||||
void IRsend::sendAiwaRCT501 (int code)
|
||||
{
|
||||
unsigned long pre = 0x0227EEC0; // 26-bits
|
||||
|
||||
// Set IR carrier frequency
|
||||
enableIROut(AIWA_RC_T501_HZ);
|
||||
|
||||
// Header
|
||||
mark(AIWA_RC_T501_HDR_MARK);
|
||||
space(AIWA_RC_T501_HDR_SPACE);
|
||||
|
||||
// Send "pre" data
|
||||
for (unsigned long mask = 1UL << (26 - 1); mask; mask >>= 1) {
|
||||
mark(AIWA_RC_T501_BIT_MARK);
|
||||
if (pre & mask) space(AIWA_RC_T501_ONE_SPACE) ;
|
||||
else space(AIWA_RC_T501_ZERO_SPACE) ;
|
||||
}
|
||||
|
||||
//-v- THIS CODE LOOKS LIKE IT MIGHT BE WRONG - CHECK!
|
||||
// it only send 15bits and ignores the top bit
|
||||
// then uses TOPBIT which is 0x80000000 to check the bit code
|
||||
// I suspect TOPBIT should be changed to 0x00008000
|
||||
|
||||
// Skip first code bit
|
||||
code <<= 1;
|
||||
// Send code
|
||||
for (int i = 0; i < 15; i++) {
|
||||
mark(AIWA_RC_T501_BIT_MARK);
|
||||
if (code & 0x80000000) space(AIWA_RC_T501_ONE_SPACE) ;
|
||||
else space(AIWA_RC_T501_ZERO_SPACE) ;
|
||||
code <<= 1;
|
||||
}
|
||||
|
||||
//-^- THIS CODE LOOKS LIKE IT MIGHT BE WRONG - CHECK!
|
||||
|
||||
// POST-DATA, 1 bit, 0x0
|
||||
mark(AIWA_RC_T501_BIT_MARK);
|
||||
space(AIWA_RC_T501_ZERO_SPACE);
|
||||
|
||||
mark(AIWA_RC_T501_BIT_MARK);
|
||||
space(0);
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
#if DECODE_AIWA_RC_T501
|
||||
bool IRrecv::decodeAiwaRCT501 (decode_results *results)
|
||||
{
|
||||
int data = 0;
|
||||
int offset = 1;
|
||||
|
||||
// Check SIZE
|
||||
if (irparams.rawlen < 2 * (AIWA_RC_T501_SUM_BITS) + 4) return false ;
|
||||
|
||||
// Check HDR Mark/Space
|
||||
if (!MATCH_MARK (results->rawbuf[offset++], AIWA_RC_T501_HDR_MARK )) return false ;
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], AIWA_RC_T501_HDR_SPACE)) return false ;
|
||||
|
||||
offset += 26; // skip pre-data - optional
|
||||
while(offset < irparams.rawlen - 4) {
|
||||
if (MATCH_MARK(results->rawbuf[offset], AIWA_RC_T501_BIT_MARK)) offset++ ;
|
||||
else return false ;
|
||||
|
||||
// ONE & ZERO
|
||||
if (MATCH_SPACE(results->rawbuf[offset], AIWA_RC_T501_ONE_SPACE)) data = (data << 1) | 1 ;
|
||||
else if (MATCH_SPACE(results->rawbuf[offset], AIWA_RC_T501_ZERO_SPACE)) data = (data << 1) | 0 ;
|
||||
else break ; // End of one & zero detected
|
||||
offset++;
|
||||
}
|
||||
|
||||
results->bits = (offset - 1) / 2;
|
||||
if (results->bits < 42) return false ;
|
||||
|
||||
results->value = data;
|
||||
results->decode_type = AIWA_RC_T501;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
94
Firmware/Source/grovepi/src/ir_Denon.cpp
Normal file
94
Firmware/Source/grovepi/src/ir_Denon.cpp
Normal file
|
|
@ -0,0 +1,94 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
// Reverse Engineered by looking at RAW dumps generated by IRremote
|
||||
|
||||
// I have since discovered that Denon publish all their IR codes:
|
||||
// https://www.google.co.uk/search?q=DENON+MASTER+IR+Hex+Command+Sheet
|
||||
// -> http://assets.denon.com/documentmaster/us/denon%20master%20ir%20hex.xls
|
||||
|
||||
// Having looked at the official Denon Pronto sheet and reverse engineered
|
||||
// the timing values from it, it is obvious that Denon have a range of
|
||||
// different timings and protocols ...the values here work for my AVR-3801 Amp!
|
||||
|
||||
//==============================================================================
|
||||
// DDDD EEEEE N N OOO N N
|
||||
// D D E NN N O O NN N
|
||||
// D D EEE N N N O O N N N
|
||||
// D D E N NN O O N NN
|
||||
// DDDD EEEEE N N OOO N N
|
||||
//==============================================================================
|
||||
|
||||
#define BITS 14 // The number of bits in the command
|
||||
|
||||
#define HDR_MARK 300 // The length of the Header:Mark
|
||||
#define HDR_SPACE 750 // The lenght of the Header:Space
|
||||
|
||||
#define BIT_MARK 300 // The length of a Bit:Mark
|
||||
#define ONE_SPACE 1800 // The length of a Bit:Space for 1's
|
||||
#define ZERO_SPACE 750 // The length of a Bit:Space for 0's
|
||||
|
||||
//+=============================================================================
|
||||
//
|
||||
#if SEND_DENON
|
||||
void IRsend::sendDenon (unsigned long data, int nbits)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(38);
|
||||
|
||||
// Header
|
||||
mark (HDR_MARK);
|
||||
space(HDR_SPACE);
|
||||
|
||||
// Data
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
mark (BIT_MARK);
|
||||
space(ONE_SPACE);
|
||||
} else {
|
||||
mark (BIT_MARK);
|
||||
space(ZERO_SPACE);
|
||||
}
|
||||
}
|
||||
|
||||
// Footer
|
||||
mark(BIT_MARK);
|
||||
space(0); // Always end with the LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
//
|
||||
#if DECODE_DENON
|
||||
bool IRrecv::decodeDenon (decode_results *results)
|
||||
{
|
||||
unsigned long data = 0; // Somewhere to build our code
|
||||
int offset = 1; // Skip the Gap reading
|
||||
|
||||
// Check we have the right amount of data
|
||||
if (irparams.rawlen != 1 + 2 + (2 * BITS) + 1) return false ;
|
||||
|
||||
// Check initial Mark+Space match
|
||||
if (!MATCH_MARK (results->rawbuf[offset++], HDR_MARK )) return false ;
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], HDR_SPACE)) return false ;
|
||||
|
||||
// Read the bits in
|
||||
for (int i = 0; i < BITS; i++) {
|
||||
// Each bit looks like: MARK + SPACE_1 -> 1
|
||||
// or : MARK + SPACE_0 -> 0
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], BIT_MARK)) return false ;
|
||||
|
||||
// IR data is big-endian, so we shuffle it in from the right:
|
||||
if (MATCH_SPACE(results->rawbuf[offset], ONE_SPACE)) data = (data << 1) | 1 ;
|
||||
else if (MATCH_SPACE(results->rawbuf[offset], ZERO_SPACE)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
// Success
|
||||
results->bits = BITS;
|
||||
results->value = data;
|
||||
results->decode_type = DENON;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
54
Firmware/Source/grovepi/src/ir_Dish.cpp
Normal file
54
Firmware/Source/grovepi/src/ir_Dish.cpp
Normal file
|
|
@ -0,0 +1,54 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// DDDD IIIII SSSS H H
|
||||
// D D I S H H
|
||||
// D D I SSS HHHHH
|
||||
// D D I S H H
|
||||
// DDDD IIIII SSSS H H
|
||||
//==============================================================================
|
||||
|
||||
// Sharp and DISH support by Todd Treece ( http://unionbridge.org/design/ircommand )
|
||||
//
|
||||
// The sned function needs to be repeated 4 times
|
||||
//
|
||||
// Only send the last for characters of the hex.
|
||||
// I.E. Use 0x1C10 instead of 0x0000000000001C10 as listed in the LIRC file.
|
||||
//
|
||||
// Here is the LIRC file I found that seems to match the remote codes from the
|
||||
// oscilloscope:
|
||||
// DISH NETWORK (echostar 301):
|
||||
// http://lirc.sourceforge.net/remotes/echostar/301_501_3100_5100_58xx_59xx
|
||||
|
||||
#define DISH_BITS 16
|
||||
#define DISH_HDR_MARK 400
|
||||
#define DISH_HDR_SPACE 6100
|
||||
#define DISH_BIT_MARK 400
|
||||
#define DISH_ONE_SPACE 1700
|
||||
#define DISH_ZERO_SPACE 2800
|
||||
#define DISH_RPT_SPACE 6200
|
||||
|
||||
//+=============================================================================
|
||||
#if SEND_DISH
|
||||
void IRsend::sendDISH (unsigned long data, int nbits)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(56);
|
||||
|
||||
mark(DISH_HDR_MARK);
|
||||
space(DISH_HDR_SPACE);
|
||||
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
mark(DISH_BIT_MARK);
|
||||
space(DISH_ONE_SPACE);
|
||||
} else {
|
||||
mark(DISH_BIT_MARK);
|
||||
space(DISH_ZERO_SPACE);
|
||||
}
|
||||
}
|
||||
mark(DISH_HDR_MARK); //added 26th March 2016, by AnalysIR ( https://www.AnalysIR.com )
|
||||
}
|
||||
#endif
|
||||
|
||||
101
Firmware/Source/grovepi/src/ir_JVC.cpp
Normal file
101
Firmware/Source/grovepi/src/ir_JVC.cpp
Normal file
|
|
@ -0,0 +1,101 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// JJJJJ V V CCCC
|
||||
// J V V C
|
||||
// J V V C
|
||||
// J J V V C
|
||||
// J V CCCC
|
||||
//==============================================================================
|
||||
|
||||
#define JVC_BITS 16
|
||||
#define JVC_HDR_MARK 8000
|
||||
#define JVC_HDR_SPACE 4000
|
||||
#define JVC_BIT_MARK 600
|
||||
#define JVC_ONE_SPACE 1600
|
||||
#define JVC_ZERO_SPACE 550
|
||||
#define JVC_RPT_LENGTH 60000
|
||||
|
||||
//+=============================================================================
|
||||
// JVC does NOT repeat by sending a separate code (like NEC does).
|
||||
// The JVC protocol repeats by skipping the header.
|
||||
// To send a JVC repeat signal, send the original code value
|
||||
// and set 'repeat' to true
|
||||
//
|
||||
#if SEND_JVC
|
||||
void IRsend::sendJVC (unsigned long data, int nbits, bool repeat)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(38);
|
||||
|
||||
// Only send the Header if this is NOT a repeat command
|
||||
if (!repeat){
|
||||
mark(JVC_HDR_MARK);
|
||||
space(JVC_HDR_SPACE);
|
||||
}
|
||||
|
||||
// Data
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
mark(JVC_BIT_MARK);
|
||||
space(JVC_ONE_SPACE);
|
||||
} else {
|
||||
mark(JVC_BIT_MARK);
|
||||
space(JVC_ZERO_SPACE);
|
||||
}
|
||||
}
|
||||
|
||||
// Footer
|
||||
mark(JVC_BIT_MARK);
|
||||
space(0); // Always end with the LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
#if DECODE_JVC
|
||||
bool IRrecv::decodeJVC (decode_results *results)
|
||||
{
|
||||
long data = 0;
|
||||
int offset = 1; // Skip first space
|
||||
|
||||
// Check for repeat
|
||||
if ( (irparams.rawlen - 1 == 33)
|
||||
&& MATCH_MARK(results->rawbuf[offset], JVC_BIT_MARK)
|
||||
&& MATCH_MARK(results->rawbuf[irparams.rawlen-1], JVC_BIT_MARK)
|
||||
) {
|
||||
results->bits = 0;
|
||||
results->value = REPEAT;
|
||||
results->decode_type = JVC;
|
||||
return true;
|
||||
}
|
||||
|
||||
// Initial mark
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], JVC_HDR_MARK)) return false ;
|
||||
|
||||
if (irparams.rawlen < (2 * JVC_BITS) + 1 ) return false ;
|
||||
|
||||
// Initial space
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], JVC_HDR_SPACE)) return false ;
|
||||
|
||||
for (int i = 0; i < JVC_BITS; i++) {
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], JVC_BIT_MARK)) return false ;
|
||||
|
||||
if (MATCH_SPACE(results->rawbuf[offset], JVC_ONE_SPACE)) data = (data << 1) | 1 ;
|
||||
else if (MATCH_SPACE(results->rawbuf[offset], JVC_ZERO_SPACE)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
// Stop bit
|
||||
if (!MATCH_MARK(results->rawbuf[offset], JVC_BIT_MARK)) return false ;
|
||||
|
||||
// Success
|
||||
results->bits = JVC_BITS;
|
||||
results->value = data;
|
||||
results->decode_type = JVC;
|
||||
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
|
||||
80
Firmware/Source/grovepi/src/ir_LG.cpp
Normal file
80
Firmware/Source/grovepi/src/ir_LG.cpp
Normal file
|
|
@ -0,0 +1,80 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// L GGGG
|
||||
// L G
|
||||
// L G GG
|
||||
// L G G
|
||||
// LLLLL GGG
|
||||
//==============================================================================
|
||||
|
||||
#define LG_BITS 28
|
||||
|
||||
#define LG_HDR_MARK 8000
|
||||
#define LG_HDR_SPACE 4000
|
||||
#define LG_BIT_MARK 600
|
||||
#define LG_ONE_SPACE 1600
|
||||
#define LG_ZERO_SPACE 550
|
||||
#define LG_RPT_LENGTH 60000
|
||||
|
||||
//+=============================================================================
|
||||
#if DECODE_LG
|
||||
bool IRrecv::decodeLG (decode_results *results)
|
||||
{
|
||||
long data = 0;
|
||||
int offset = 1; // Skip first space
|
||||
|
||||
// Check we have the right amount of data
|
||||
if (irparams.rawlen < (2 * LG_BITS) + 1 ) return false ;
|
||||
|
||||
// Initial mark/space
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], LG_HDR_MARK)) return false ;
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], LG_HDR_SPACE)) return false ;
|
||||
|
||||
for (int i = 0; i < LG_BITS; i++) {
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], LG_BIT_MARK)) return false ;
|
||||
|
||||
if (MATCH_SPACE(results->rawbuf[offset], LG_ONE_SPACE)) data = (data << 1) | 1 ;
|
||||
else if (MATCH_SPACE(results->rawbuf[offset], LG_ZERO_SPACE)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
// Stop bit
|
||||
if (!MATCH_MARK(results->rawbuf[offset], LG_BIT_MARK)) return false ;
|
||||
|
||||
// Success
|
||||
results->bits = LG_BITS;
|
||||
results->value = data;
|
||||
results->decode_type = LG;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
#if SEND_LG
|
||||
void IRsend::sendLG (unsigned long data, int nbits)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(38);
|
||||
|
||||
// Header
|
||||
mark(LG_HDR_MARK);
|
||||
space(LG_HDR_SPACE);
|
||||
mark(LG_BIT_MARK);
|
||||
|
||||
// Data
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
space(LG_ONE_SPACE);
|
||||
mark(LG_BIT_MARK);
|
||||
} else {
|
||||
space(LG_ZERO_SPACE);
|
||||
mark(LG_BIT_MARK);
|
||||
}
|
||||
}
|
||||
space(0); // Always end with the LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
46
Firmware/Source/grovepi/src/ir_Lego_PF.cpp
Normal file
46
Firmware/Source/grovepi/src/ir_Lego_PF.cpp
Normal file
|
|
@ -0,0 +1,46 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
#include "ir_Lego_PF_BitStreamEncoder.h"
|
||||
|
||||
//==============================================================================
|
||||
// L EEEEEE EEEE OOOO
|
||||
// L E E O O
|
||||
// L EEEE E EEE O O
|
||||
// L E E E O O LEGO Power Functions
|
||||
// LLLLLL EEEEEE EEEE OOOO Copyright (c) 2016 Philipp Henkel
|
||||
//==============================================================================
|
||||
|
||||
// Supported Devices
|
||||
// LEGO® Power Functions IR Receiver 8884
|
||||
|
||||
//+=============================================================================
|
||||
//
|
||||
#if SEND_LEGO_PF
|
||||
|
||||
#if DEBUG
|
||||
namespace {
|
||||
void logFunctionParameters(uint16_t data, bool repeat) {
|
||||
DBG_PRINT("sendLegoPowerFunctions(data=");
|
||||
DBG_PRINT(data);
|
||||
DBG_PRINT(", repeat=");
|
||||
DBG_PRINTLN(repeat?"true)" : "false)");
|
||||
}
|
||||
} // anonymous namespace
|
||||
#endif // DEBUG
|
||||
|
||||
void IRsend::sendLegoPowerFunctions(uint16_t data, bool repeat)
|
||||
{
|
||||
#if DEBUG
|
||||
::logFunctionParameters(data, repeat);
|
||||
#endif // DEBUG
|
||||
|
||||
enableIROut(38);
|
||||
static LegoPfBitStreamEncoder bitStreamEncoder;
|
||||
bitStreamEncoder.reset(data, repeat);
|
||||
do {
|
||||
mark(bitStreamEncoder.getMarkDuration());
|
||||
space(bitStreamEncoder.getPauseDuration());
|
||||
} while (bitStreamEncoder.next());
|
||||
}
|
||||
|
||||
#endif // SEND_LEGO_PF
|
||||
115
Firmware/Source/grovepi/src/ir_Lego_PF_BitStreamEncoder.h
Normal file
115
Firmware/Source/grovepi/src/ir_Lego_PF_BitStreamEncoder.h
Normal file
|
|
@ -0,0 +1,115 @@
|
|||
|
||||
//==============================================================================
|
||||
// L EEEEEE EEEE OOOO
|
||||
// L E E O O
|
||||
// L EEEE E EEE O O
|
||||
// L E E E O O LEGO Power Functions
|
||||
// LLLLLL EEEEEE EEEE OOOO Copyright (c) 2016, 2017 Philipp Henkel
|
||||
//==============================================================================
|
||||
|
||||
//+=============================================================================
|
||||
//
|
||||
|
||||
class LegoPfBitStreamEncoder {
|
||||
private:
|
||||
uint16_t data;
|
||||
bool repeatMessage;
|
||||
uint8_t messageBitIdx;
|
||||
uint8_t repeatCount;
|
||||
uint16_t messageLength;
|
||||
|
||||
public:
|
||||
// HIGH data bit = IR mark + high pause
|
||||
// LOW data bit = IR mark + low pause
|
||||
static const uint16_t LOW_BIT_DURATION = 421;
|
||||
static const uint16_t HIGH_BIT_DURATION = 711;
|
||||
static const uint16_t START_BIT_DURATION = 1184;
|
||||
static const uint16_t STOP_BIT_DURATION = 1184;
|
||||
static const uint8_t IR_MARK_DURATION = 158;
|
||||
static const uint16_t HIGH_PAUSE_DURATION = HIGH_BIT_DURATION - IR_MARK_DURATION;
|
||||
static const uint16_t LOW_PAUSE_DURATION = LOW_BIT_DURATION - IR_MARK_DURATION;
|
||||
static const uint16_t START_PAUSE_DURATION = START_BIT_DURATION - IR_MARK_DURATION;
|
||||
static const uint16_t STOP_PAUSE_DURATION = STOP_BIT_DURATION - IR_MARK_DURATION;
|
||||
static const uint8_t MESSAGE_BITS = 18;
|
||||
static const uint16_t MAX_MESSAGE_LENGTH = 16000;
|
||||
|
||||
void reset(uint16_t data, bool repeatMessage) {
|
||||
this->data = data;
|
||||
this->repeatMessage = repeatMessage;
|
||||
messageBitIdx = 0;
|
||||
repeatCount = 0;
|
||||
messageLength = getMessageLength();
|
||||
}
|
||||
|
||||
int getChannelId() const { return 1 + ((data >> 12) & 0x3); }
|
||||
|
||||
uint16_t getMessageLength() const {
|
||||
// Sum up all marks
|
||||
uint16_t length = MESSAGE_BITS * IR_MARK_DURATION;
|
||||
|
||||
// Sum up all pauses
|
||||
length += START_PAUSE_DURATION;
|
||||
for (unsigned long mask = 1UL << 15; mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
length += HIGH_PAUSE_DURATION;
|
||||
} else {
|
||||
length += LOW_PAUSE_DURATION;
|
||||
}
|
||||
}
|
||||
length += STOP_PAUSE_DURATION;
|
||||
return length;
|
||||
}
|
||||
|
||||
boolean next() {
|
||||
messageBitIdx++;
|
||||
if (messageBitIdx >= MESSAGE_BITS) {
|
||||
repeatCount++;
|
||||
messageBitIdx = 0;
|
||||
}
|
||||
if (repeatCount >= 1 && !repeatMessage) {
|
||||
return false;
|
||||
} else if (repeatCount >= 5) {
|
||||
return false;
|
||||
} else {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
uint8_t getMarkDuration() const { return IR_MARK_DURATION; }
|
||||
|
||||
uint32_t getPauseDuration() const {
|
||||
if (messageBitIdx == 0)
|
||||
return START_PAUSE_DURATION;
|
||||
else if (messageBitIdx < MESSAGE_BITS - 1) {
|
||||
return getDataBitPause();
|
||||
} else {
|
||||
return getStopPause();
|
||||
}
|
||||
}
|
||||
|
||||
private:
|
||||
uint16_t getDataBitPause() const {
|
||||
const int pos = MESSAGE_BITS - 2 - messageBitIdx;
|
||||
const bool isHigh = data & (1 << pos);
|
||||
return isHigh ? HIGH_PAUSE_DURATION : LOW_PAUSE_DURATION;
|
||||
}
|
||||
|
||||
uint32_t getStopPause() const {
|
||||
if (repeatMessage) {
|
||||
return getRepeatStopPause();
|
||||
} else {
|
||||
return STOP_PAUSE_DURATION;
|
||||
}
|
||||
}
|
||||
|
||||
uint32_t getRepeatStopPause() const {
|
||||
if (repeatCount == 0 || repeatCount == 1) {
|
||||
return STOP_PAUSE_DURATION + (uint32_t)5 * MAX_MESSAGE_LENGTH - messageLength;
|
||||
} else if (repeatCount == 2 || repeatCount == 3) {
|
||||
return STOP_PAUSE_DURATION
|
||||
+ (uint32_t)(6 + 2 * getChannelId()) * MAX_MESSAGE_LENGTH - messageLength;
|
||||
} else {
|
||||
return STOP_PAUSE_DURATION;
|
||||
}
|
||||
}
|
||||
};
|
||||
85
Firmware/Source/grovepi/src/ir_Mitsubishi.cpp
Normal file
85
Firmware/Source/grovepi/src/ir_Mitsubishi.cpp
Normal file
|
|
@ -0,0 +1,85 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// MMMMM IIIII TTTTT SSSS U U BBBB IIIII SSSS H H IIIII
|
||||
// M M M I T S U U B B I S H H I
|
||||
// M M M I T SSS U U BBBB I SSS HHHHH I
|
||||
// M M I T S U U B B I S H H I
|
||||
// M M IIIII T SSSS UUU BBBBB IIIII SSSS H H IIIII
|
||||
//==============================================================================
|
||||
|
||||
// Looks like Sony except for timings, 48 chars of data and time/space different
|
||||
|
||||
#define MITSUBISHI_BITS 16
|
||||
|
||||
// Mitsubishi RM 75501
|
||||
// 14200 7 41 7 42 7 42 7 17 7 17 7 18 7 41 7 18 7 17 7 17 7 18 7 41 8 17 7 17 7 18 7 17 7
|
||||
// #define MITSUBISHI_HDR_MARK 250 // seen range 3500
|
||||
#define MITSUBISHI_HDR_SPACE 350 // 7*50+100
|
||||
#define MITSUBISHI_ONE_MARK 1950 // 41*50-100
|
||||
#define MITSUBISHI_ZERO_MARK 750 // 17*50-100
|
||||
// #define MITSUBISHI_DOUBLE_SPACE_USECS 800 // usually ssee 713 - not using ticks as get number wrapround
|
||||
// #define MITSUBISHI_RPT_LENGTH 45000
|
||||
|
||||
//+=============================================================================
|
||||
#if DECODE_MITSUBISHI
|
||||
bool IRrecv::decodeMitsubishi (decode_results *results)
|
||||
{
|
||||
// Serial.print("?!? decoding Mitsubishi:");Serial.print(irparams.rawlen); Serial.print(" want "); Serial.println( 2 * MITSUBISHI_BITS + 2);
|
||||
long data = 0;
|
||||
if (irparams.rawlen < 2 * MITSUBISHI_BITS + 2) return false ;
|
||||
int offset = 0; // Skip first space
|
||||
// Initial space
|
||||
|
||||
#if 0
|
||||
// Put this back in for debugging - note can't use #DEBUG as if Debug on we don't see the repeat cos of the delay
|
||||
Serial.print("IR Gap: ");
|
||||
Serial.println( results->rawbuf[offset]);
|
||||
Serial.println( "test against:");
|
||||
Serial.println(results->rawbuf[offset]);
|
||||
#endif
|
||||
|
||||
#if 0
|
||||
// Not seeing double keys from Mitsubishi
|
||||
if (results->rawbuf[offset] < MITSUBISHI_DOUBLE_SPACE_USECS) {
|
||||
// Serial.print("IR Gap found: ");
|
||||
results->bits = 0;
|
||||
results->value = REPEAT;
|
||||
results->decode_type = MITSUBISHI;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
|
||||
offset++;
|
||||
|
||||
// Typical
|
||||
// 14200 7 41 7 42 7 42 7 17 7 17 7 18 7 41 7 18 7 17 7 17 7 18 7 41 8 17 7 17 7 18 7 17 7
|
||||
|
||||
// Initial Space
|
||||
if (!MATCH_MARK(results->rawbuf[offset], MITSUBISHI_HDR_SPACE)) return false ;
|
||||
offset++;
|
||||
|
||||
while (offset + 1 < irparams.rawlen) {
|
||||
if (MATCH_MARK(results->rawbuf[offset], MITSUBISHI_ONE_MARK)) data = (data << 1) | 1 ;
|
||||
else if (MATCH_MARK(results->rawbuf[offset], MITSUBISHI_ZERO_MARK)) data <<= 1 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
|
||||
if (!MATCH_SPACE(results->rawbuf[offset], MITSUBISHI_HDR_SPACE)) break ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
// Success
|
||||
results->bits = (offset - 1) / 2;
|
||||
if (results->bits < MITSUBISHI_BITS) {
|
||||
results->bits = 0;
|
||||
return false;
|
||||
}
|
||||
|
||||
results->value = data;
|
||||
results->decode_type = MITSUBISHI;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
|
||||
98
Firmware/Source/grovepi/src/ir_NEC.cpp
Normal file
98
Firmware/Source/grovepi/src/ir_NEC.cpp
Normal file
|
|
@ -0,0 +1,98 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// N N EEEEE CCCC
|
||||
// NN N E C
|
||||
// N N N EEE C
|
||||
// N NN E C
|
||||
// N N EEEEE CCCC
|
||||
//==============================================================================
|
||||
|
||||
#define NEC_BITS 32
|
||||
#define NEC_HDR_MARK 9000
|
||||
#define NEC_HDR_SPACE 4500
|
||||
#define NEC_BIT_MARK 560
|
||||
#define NEC_ONE_SPACE 1690
|
||||
#define NEC_ZERO_SPACE 560
|
||||
#define NEC_RPT_SPACE 2250
|
||||
|
||||
//+=============================================================================
|
||||
#if SEND_NEC
|
||||
void IRsend::sendNEC (unsigned long data, int nbits)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(38);
|
||||
|
||||
// Header
|
||||
mark(NEC_HDR_MARK);
|
||||
space(NEC_HDR_SPACE);
|
||||
|
||||
// Data
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
mark(NEC_BIT_MARK);
|
||||
space(NEC_ONE_SPACE);
|
||||
} else {
|
||||
mark(NEC_BIT_MARK);
|
||||
space(NEC_ZERO_SPACE);
|
||||
}
|
||||
}
|
||||
|
||||
// Footer
|
||||
mark(NEC_BIT_MARK);
|
||||
space(0); // Always end with the LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
// NECs have a repeat only 4 items long
|
||||
//
|
||||
#if DECODE_NEC
|
||||
bool IRrecv::decodeNEC (decode_results *results)
|
||||
{
|
||||
long data = 0; // We decode in to here; Start with nothing
|
||||
int offset = 1; // Index in to results; Skip first entry!?
|
||||
|
||||
// Check header "mark"
|
||||
if (!MATCH_MARK(results->rawbuf[offset], NEC_HDR_MARK)) return false ;
|
||||
offset++;
|
||||
|
||||
// Check for repeat
|
||||
if ( (irparams.rawlen == 4)
|
||||
&& MATCH_SPACE(results->rawbuf[offset ], NEC_RPT_SPACE)
|
||||
&& MATCH_MARK (results->rawbuf[offset+1], NEC_BIT_MARK )
|
||||
) {
|
||||
results->bits = 0;
|
||||
results->value = REPEAT;
|
||||
results->decode_type = NEC;
|
||||
return true;
|
||||
}
|
||||
|
||||
// Check we have enough data
|
||||
if (irparams.rawlen < (2 * NEC_BITS) + 4) return false ;
|
||||
|
||||
// Check header "space"
|
||||
if (!MATCH_SPACE(results->rawbuf[offset], NEC_HDR_SPACE)) return false ;
|
||||
offset++;
|
||||
|
||||
// Build the data
|
||||
for (int i = 0; i < NEC_BITS; i++) {
|
||||
// Check data "mark"
|
||||
if (!MATCH_MARK(results->rawbuf[offset], NEC_BIT_MARK)) return false ;
|
||||
offset++;
|
||||
// Suppend this bit
|
||||
if (MATCH_SPACE(results->rawbuf[offset], NEC_ONE_SPACE )) data = (data << 1) | 1 ;
|
||||
else if (MATCH_SPACE(results->rawbuf[offset], NEC_ZERO_SPACE)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
// Success
|
||||
results->bits = NEC_BITS;
|
||||
results->value = data;
|
||||
results->decode_type = NEC;
|
||||
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
78
Firmware/Source/grovepi/src/ir_Panasonic.cpp
Normal file
78
Firmware/Source/grovepi/src/ir_Panasonic.cpp
Normal file
|
|
@ -0,0 +1,78 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// PPPP AAA N N AAA SSSS OOO N N IIIII CCCC
|
||||
// P P A A NN N A A S O O NN N I C
|
||||
// PPPP AAAAA N N N AAAAA SSS O O N N N I C
|
||||
// P A A N NN A A S O O N NN I C
|
||||
// P A A N N A A SSSS OOO N N IIIII CCCC
|
||||
//==============================================================================
|
||||
|
||||
#define PANASONIC_BITS 48
|
||||
#define PANASONIC_HDR_MARK 3502
|
||||
#define PANASONIC_HDR_SPACE 1750
|
||||
#define PANASONIC_BIT_MARK 502
|
||||
#define PANASONIC_ONE_SPACE 1244
|
||||
#define PANASONIC_ZERO_SPACE 400
|
||||
|
||||
//+=============================================================================
|
||||
#if SEND_PANASONIC
|
||||
void IRsend::sendPanasonic (unsigned int address, unsigned long data)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(35);
|
||||
|
||||
// Header
|
||||
mark(PANASONIC_HDR_MARK);
|
||||
space(PANASONIC_HDR_SPACE);
|
||||
|
||||
// Address
|
||||
for (unsigned long mask = 1UL << (16 - 1); mask; mask >>= 1) {
|
||||
mark(PANASONIC_BIT_MARK);
|
||||
if (address & mask) space(PANASONIC_ONE_SPACE) ;
|
||||
else space(PANASONIC_ZERO_SPACE) ;
|
||||
}
|
||||
|
||||
// Data
|
||||
for (unsigned long mask = 1UL << (32 - 1); mask; mask >>= 1) {
|
||||
mark(PANASONIC_BIT_MARK);
|
||||
if (data & mask) space(PANASONIC_ONE_SPACE) ;
|
||||
else space(PANASONIC_ZERO_SPACE) ;
|
||||
}
|
||||
|
||||
// Footer
|
||||
mark(PANASONIC_BIT_MARK);
|
||||
space(0); // Always end with the LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
#if DECODE_PANASONIC
|
||||
bool IRrecv::decodePanasonic (decode_results *results)
|
||||
{
|
||||
unsigned long long data = 0;
|
||||
int offset = 1;
|
||||
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], PANASONIC_HDR_MARK )) return false ;
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], PANASONIC_HDR_SPACE)) return false ;
|
||||
|
||||
// decode address
|
||||
for (int i = 0; i < PANASONIC_BITS; i++) {
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], PANASONIC_BIT_MARK)) return false ;
|
||||
|
||||
if (MATCH_SPACE(results->rawbuf[offset],PANASONIC_ONE_SPACE )) data = (data << 1) | 1 ;
|
||||
else if (MATCH_SPACE(results->rawbuf[offset],PANASONIC_ZERO_SPACE)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
results->value = (unsigned long)data;
|
||||
results->address = (unsigned int)(data >> 32);
|
||||
results->decode_type = PANASONIC;
|
||||
results->bits = PANASONIC_BITS;
|
||||
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
|
||||
207
Firmware/Source/grovepi/src/ir_RC5_RC6.cpp
Normal file
207
Firmware/Source/grovepi/src/ir_RC5_RC6.cpp
Normal file
|
|
@ -0,0 +1,207 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//+=============================================================================
|
||||
// Gets one undecoded level at a time from the raw buffer.
|
||||
// The RC5/6 decoding is easier if the data is broken into time intervals.
|
||||
// E.g. if the buffer has MARK for 2 time intervals and SPACE for 1,
|
||||
// successive calls to getRClevel will return MARK, MARK, SPACE.
|
||||
// offset and used are updated to keep track of the current position.
|
||||
// t1 is the time interval for a single bit in microseconds.
|
||||
// Returns -1 for error (measured time interval is not a multiple of t1).
|
||||
//
|
||||
#if (DECODE_RC5 || DECODE_RC6)
|
||||
int IRrecv::getRClevel (decode_results *results, int *offset, int *used, int t1)
|
||||
{
|
||||
int width;
|
||||
int val;
|
||||
int correction;
|
||||
int avail;
|
||||
|
||||
if (*offset >= results->rawlen) return SPACE ; // After end of recorded buffer, assume SPACE.
|
||||
width = results->rawbuf[*offset];
|
||||
val = ((*offset) % 2) ? MARK : SPACE;
|
||||
correction = (val == MARK) ? MARK_EXCESS : - MARK_EXCESS;
|
||||
|
||||
if (MATCH(width, ( t1) + correction)) avail = 1 ;
|
||||
else if (MATCH(width, (2*t1) + correction)) avail = 2 ;
|
||||
else if (MATCH(width, (3*t1) + correction)) avail = 3 ;
|
||||
else return -1 ;
|
||||
|
||||
(*used)++;
|
||||
if (*used >= avail) {
|
||||
*used = 0;
|
||||
(*offset)++;
|
||||
}
|
||||
|
||||
DBG_PRINTLN( (val == MARK) ? "MARK" : "SPACE" );
|
||||
|
||||
return val;
|
||||
}
|
||||
#endif
|
||||
|
||||
//==============================================================================
|
||||
// RRRR CCCC 55555
|
||||
// R R C 5
|
||||
// RRRR C 5555
|
||||
// R R C 5
|
||||
// R R CCCC 5555
|
||||
//
|
||||
// NB: First bit must be a one (start bit)
|
||||
//
|
||||
#define MIN_RC5_SAMPLES 11
|
||||
#define RC5_T1 889
|
||||
#define RC5_RPT_LENGTH 46000
|
||||
|
||||
//+=============================================================================
|
||||
#if SEND_RC5
|
||||
void IRsend::sendRC5 (unsigned long data, int nbits)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(36);
|
||||
|
||||
// Start
|
||||
mark(RC5_T1);
|
||||
space(RC5_T1);
|
||||
mark(RC5_T1);
|
||||
|
||||
// Data
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
space(RC5_T1); // 1 is space, then mark
|
||||
mark(RC5_T1);
|
||||
} else {
|
||||
mark(RC5_T1);
|
||||
space(RC5_T1);
|
||||
}
|
||||
}
|
||||
|
||||
space(0); // Always end with the LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
#if DECODE_RC5
|
||||
bool IRrecv::decodeRC5 (decode_results *results)
|
||||
{
|
||||
int nbits;
|
||||
long data = 0;
|
||||
int used = 0;
|
||||
int offset = 1; // Skip gap space
|
||||
|
||||
if (irparams.rawlen < MIN_RC5_SAMPLES + 2) return false ;
|
||||
|
||||
// Get start bits
|
||||
if (getRClevel(results, &offset, &used, RC5_T1) != MARK) return false ;
|
||||
if (getRClevel(results, &offset, &used, RC5_T1) != SPACE) return false ;
|
||||
if (getRClevel(results, &offset, &used, RC5_T1) != MARK) return false ;
|
||||
|
||||
for (nbits = 0; offset < irparams.rawlen; nbits++) {
|
||||
int levelA = getRClevel(results, &offset, &used, RC5_T1);
|
||||
int levelB = getRClevel(results, &offset, &used, RC5_T1);
|
||||
|
||||
if ((levelA == SPACE) && (levelB == MARK )) data = (data << 1) | 1 ;
|
||||
else if ((levelA == MARK ) && (levelB == SPACE)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
}
|
||||
|
||||
// Success
|
||||
results->bits = nbits;
|
||||
results->value = data;
|
||||
results->decode_type = RC5;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
// RRRR CCCC 6666
|
||||
// R R C 6
|
||||
// RRRR C 6666
|
||||
// R R C 6 6
|
||||
// R R CCCC 666
|
||||
//
|
||||
// NB : Caller needs to take care of flipping the toggle bit
|
||||
//
|
||||
#define MIN_RC6_SAMPLES 1
|
||||
#define RC6_HDR_MARK 2666
|
||||
#define RC6_HDR_SPACE 889
|
||||
#define RC6_T1 444
|
||||
#define RC6_RPT_LENGTH 46000
|
||||
|
||||
#if SEND_RC6
|
||||
void IRsend::sendRC6 (unsigned long data, int nbits)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(36);
|
||||
|
||||
// Header
|
||||
mark(RC6_HDR_MARK);
|
||||
space(RC6_HDR_SPACE);
|
||||
|
||||
// Start bit
|
||||
mark(RC6_T1);
|
||||
space(RC6_T1);
|
||||
|
||||
// Data
|
||||
for (unsigned long i = 1, mask = 1UL << (nbits - 1); mask; i++, mask >>= 1) {
|
||||
// The fourth bit we send is a "double width trailer bit"
|
||||
int t = (i == 4) ? (RC6_T1 * 2) : (RC6_T1) ;
|
||||
if (data & mask) {
|
||||
mark(t);
|
||||
space(t);
|
||||
} else {
|
||||
space(t);
|
||||
mark(t);
|
||||
}
|
||||
}
|
||||
|
||||
space(0); // Always end with the LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
#if DECODE_RC6
|
||||
bool IRrecv::decodeRC6 (decode_results *results)
|
||||
{
|
||||
int nbits;
|
||||
long data = 0;
|
||||
int used = 0;
|
||||
int offset = 1; // Skip first space
|
||||
|
||||
if (results->rawlen < MIN_RC6_SAMPLES) return false ;
|
||||
|
||||
// Initial mark
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], RC6_HDR_MARK)) return false ;
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], RC6_HDR_SPACE)) return false ;
|
||||
|
||||
// Get start bit (1)
|
||||
if (getRClevel(results, &offset, &used, RC6_T1) != MARK) return false ;
|
||||
if (getRClevel(results, &offset, &used, RC6_T1) != SPACE) return false ;
|
||||
|
||||
for (nbits = 0; offset < results->rawlen; nbits++) {
|
||||
int levelA, levelB; // Next two levels
|
||||
|
||||
levelA = getRClevel(results, &offset, &used, RC6_T1);
|
||||
if (nbits == 3) {
|
||||
// T bit is double wide; make sure second half matches
|
||||
if (levelA != getRClevel(results, &offset, &used, RC6_T1)) return false;
|
||||
}
|
||||
|
||||
levelB = getRClevel(results, &offset, &used, RC6_T1);
|
||||
if (nbits == 3) {
|
||||
// T bit is double wide; make sure second half matches
|
||||
if (levelB != getRClevel(results, &offset, &used, RC6_T1)) return false;
|
||||
}
|
||||
|
||||
if ((levelA == MARK ) && (levelB == SPACE)) data = (data << 1) | 1 ; // inverted compared to RC5
|
||||
else if ((levelA == SPACE) && (levelB == MARK )) data = (data << 1) | 0 ; // ...
|
||||
else return false ; // Error
|
||||
}
|
||||
|
||||
// Success
|
||||
results->bits = nbits;
|
||||
results->value = data;
|
||||
results->decode_type = RC6;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
92
Firmware/Source/grovepi/src/ir_Samsung.cpp
Normal file
92
Firmware/Source/grovepi/src/ir_Samsung.cpp
Normal file
|
|
@ -0,0 +1,92 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// SSSS AAA MMM SSSS U U N N GGGG
|
||||
// S A A M M M S U U NN N G
|
||||
// SSS AAAAA M M M SSS U U N N N G GG
|
||||
// S A A M M S U U N NN G G
|
||||
// SSSS A A M M SSSS UUU N N GGG
|
||||
//==============================================================================
|
||||
|
||||
#define SAMSUNG_BITS 32
|
||||
#define SAMSUNG_HDR_MARK 5000
|
||||
#define SAMSUNG_HDR_SPACE 5000
|
||||
#define SAMSUNG_BIT_MARK 560
|
||||
#define SAMSUNG_ONE_SPACE 1600
|
||||
#define SAMSUNG_ZERO_SPACE 560
|
||||
#define SAMSUNG_RPT_SPACE 2250
|
||||
|
||||
//+=============================================================================
|
||||
#if SEND_SAMSUNG
|
||||
void IRsend::sendSAMSUNG (unsigned long data, int nbits)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(38);
|
||||
|
||||
// Header
|
||||
mark(SAMSUNG_HDR_MARK);
|
||||
space(SAMSUNG_HDR_SPACE);
|
||||
|
||||
// Data
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
mark(SAMSUNG_BIT_MARK);
|
||||
space(SAMSUNG_ONE_SPACE);
|
||||
} else {
|
||||
mark(SAMSUNG_BIT_MARK);
|
||||
space(SAMSUNG_ZERO_SPACE);
|
||||
}
|
||||
}
|
||||
|
||||
// Footer
|
||||
mark(SAMSUNG_BIT_MARK);
|
||||
space(0); // Always end with the LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
// SAMSUNGs have a repeat only 4 items long
|
||||
//
|
||||
#if DECODE_SAMSUNG
|
||||
bool IRrecv::decodeSAMSUNG (decode_results *results)
|
||||
{
|
||||
long data = 0;
|
||||
int offset = 1; // Skip first space
|
||||
|
||||
// Initial mark
|
||||
if (!MATCH_MARK(results->rawbuf[offset], SAMSUNG_HDR_MARK)) return false ;
|
||||
offset++;
|
||||
|
||||
// Check for repeat
|
||||
if ( (irparams.rawlen == 4)
|
||||
&& MATCH_SPACE(results->rawbuf[offset], SAMSUNG_RPT_SPACE)
|
||||
&& MATCH_MARK(results->rawbuf[offset+1], SAMSUNG_BIT_MARK)
|
||||
) {
|
||||
results->bits = 0;
|
||||
results->value = REPEAT;
|
||||
results->decode_type = SAMSUNG;
|
||||
return true;
|
||||
}
|
||||
if (irparams.rawlen < (2 * SAMSUNG_BITS) + 4) return false ;
|
||||
|
||||
// Initial space
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], SAMSUNG_HDR_SPACE)) return false ;
|
||||
|
||||
for (int i = 0; i < SAMSUNG_BITS; i++) {
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], SAMSUNG_BIT_MARK)) return false ;
|
||||
|
||||
if (MATCH_SPACE(results->rawbuf[offset], SAMSUNG_ONE_SPACE)) data = (data << 1) | 1 ;
|
||||
else if (MATCH_SPACE(results->rawbuf[offset], SAMSUNG_ZERO_SPACE)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
// Success
|
||||
results->bits = SAMSUNG_BITS;
|
||||
results->value = data;
|
||||
results->decode_type = SAMSUNG;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
|
||||
76
Firmware/Source/grovepi/src/ir_Sanyo.cpp
Normal file
76
Firmware/Source/grovepi/src/ir_Sanyo.cpp
Normal file
|
|
@ -0,0 +1,76 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// SSSS AAA N N Y Y OOO
|
||||
// S A A NN N Y Y O O
|
||||
// SSS AAAAA N N N Y O O
|
||||
// S A A N NN Y O O
|
||||
// SSSS A A N N Y OOO
|
||||
//==============================================================================
|
||||
|
||||
// I think this is a Sanyo decoder: Serial = SA 8650B
|
||||
// Looks like Sony except for timings, 48 chars of data and time/space different
|
||||
|
||||
#define SANYO_BITS 12
|
||||
#define SANYO_HDR_MARK 3500 // seen range 3500
|
||||
#define SANYO_HDR_SPACE 950 // seen 950
|
||||
#define SANYO_ONE_MARK 2400 // seen 2400
|
||||
#define SANYO_ZERO_MARK 700 // seen 700
|
||||
#define SANYO_DOUBLE_SPACE_USECS 800 // usually ssee 713 - not using ticks as get number wrapround
|
||||
#define SANYO_RPT_LENGTH 45000
|
||||
|
||||
//+=============================================================================
|
||||
#if DECODE_SANYO
|
||||
bool IRrecv::decodeSanyo (decode_results *results)
|
||||
{
|
||||
long data = 0;
|
||||
int offset = 0; // Skip first space <-- CHECK THIS!
|
||||
|
||||
if (irparams.rawlen < (2 * SANYO_BITS) + 2) return false ;
|
||||
|
||||
#if 0
|
||||
// Put this back in for debugging - note can't use #DEBUG as if Debug on we don't see the repeat cos of the delay
|
||||
Serial.print("IR Gap: ");
|
||||
Serial.println( results->rawbuf[offset]);
|
||||
Serial.println( "test against:");
|
||||
Serial.println(results->rawbuf[offset]);
|
||||
#endif
|
||||
|
||||
// Initial space
|
||||
if (results->rawbuf[offset] < SANYO_DOUBLE_SPACE_USECS) {
|
||||
//Serial.print("IR Gap found: ");
|
||||
results->bits = 0;
|
||||
results->value = REPEAT;
|
||||
results->decode_type = SANYO;
|
||||
return true;
|
||||
}
|
||||
offset++;
|
||||
|
||||
// Initial mark
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], SANYO_HDR_MARK)) return false ;
|
||||
|
||||
// Skip Second Mark
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], SANYO_HDR_MARK)) return false ;
|
||||
|
||||
while (offset + 1 < irparams.rawlen) {
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], SANYO_HDR_SPACE)) break ;
|
||||
|
||||
if (MATCH_MARK(results->rawbuf[offset], SANYO_ONE_MARK)) data = (data << 1) | 1 ;
|
||||
else if (MATCH_MARK(results->rawbuf[offset], SANYO_ZERO_MARK)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
// Success
|
||||
results->bits = (offset - 1) / 2;
|
||||
if (results->bits < 12) {
|
||||
results->bits = 0;
|
||||
return false;
|
||||
}
|
||||
|
||||
results->value = data;
|
||||
results->decode_type = SANYO;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
71
Firmware/Source/grovepi/src/ir_Sharp.cpp
Normal file
71
Firmware/Source/grovepi/src/ir_Sharp.cpp
Normal file
|
|
@ -0,0 +1,71 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// SSSS H H AAA RRRR PPPP
|
||||
// S H H A A R R P P
|
||||
// SSS HHHHH AAAAA RRRR PPPP
|
||||
// S H H A A R R P
|
||||
// SSSS H H A A R R P
|
||||
//==============================================================================
|
||||
|
||||
// Sharp and DISH support by Todd Treece: http://unionbridge.org/design/ircommand
|
||||
//
|
||||
// The send function has the necessary repeat built in because of the need to
|
||||
// invert the signal.
|
||||
//
|
||||
// Sharp protocol documentation:
|
||||
// http://www.sbprojects.com/knowledge/ir/sharp.htm
|
||||
//
|
||||
// Here is the LIRC file I found that seems to match the remote codes from the
|
||||
// oscilloscope:
|
||||
// Sharp LCD TV:
|
||||
// http://lirc.sourceforge.net/remotes/sharp/GA538WJSA
|
||||
|
||||
#define SHARP_BITS 15
|
||||
#define SHARP_BIT_MARK 245
|
||||
#define SHARP_ONE_SPACE 1805
|
||||
#define SHARP_ZERO_SPACE 795
|
||||
#define SHARP_GAP 600000
|
||||
#define SHARP_RPT_SPACE 3000
|
||||
|
||||
#define SHARP_TOGGLE_MASK 0x3FF
|
||||
|
||||
//+=============================================================================
|
||||
#if SEND_SHARP
|
||||
void IRsend::sendSharpRaw (unsigned long data, int nbits)
|
||||
{
|
||||
enableIROut(38);
|
||||
|
||||
// Sending codes in bursts of 3 (normal, inverted, normal) makes transmission
|
||||
// much more reliable. That's the exact behaviour of CD-S6470 remote control.
|
||||
for (int n = 0; n < 3; n++) {
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
mark(SHARP_BIT_MARK);
|
||||
space(SHARP_ONE_SPACE);
|
||||
} else {
|
||||
mark(SHARP_BIT_MARK);
|
||||
space(SHARP_ZERO_SPACE);
|
||||
}
|
||||
}
|
||||
|
||||
mark(SHARP_BIT_MARK);
|
||||
space(SHARP_ZERO_SPACE);
|
||||
delay(40);
|
||||
|
||||
data = data ^ SHARP_TOGGLE_MASK;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
// Sharp send compatible with data obtained through decodeSharp()
|
||||
// ^^^^^^^^^^^^^ FUNCTION MISSING!
|
||||
//
|
||||
#if SEND_SHARP
|
||||
void IRsend::sendSharp (unsigned int address, unsigned int command)
|
||||
{
|
||||
sendSharpRaw((address << 10) | (command << 2) | 2, SHARP_BITS);
|
||||
}
|
||||
#endif
|
||||
95
Firmware/Source/grovepi/src/ir_Sony.cpp
Normal file
95
Firmware/Source/grovepi/src/ir_Sony.cpp
Normal file
|
|
@ -0,0 +1,95 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// SSSS OOO N N Y Y
|
||||
// S O O NN N Y Y
|
||||
// SSS O O N N N Y
|
||||
// S O O N NN Y
|
||||
// SSSS OOO N N Y
|
||||
//==============================================================================
|
||||
|
||||
#define SONY_BITS 12
|
||||
#define SONY_HDR_MARK 2400
|
||||
#define SONY_HDR_SPACE 600
|
||||
#define SONY_ONE_MARK 1200
|
||||
#define SONY_ZERO_MARK 600
|
||||
#define SONY_RPT_LENGTH 45000
|
||||
#define SONY_DOUBLE_SPACE_USECS 500 // usually ssee 713 - not using ticks as get number wrapround
|
||||
|
||||
//+=============================================================================
|
||||
#if SEND_SONY
|
||||
void IRsend::sendSony (unsigned long data, int nbits)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(40);
|
||||
|
||||
// Header
|
||||
mark(SONY_HDR_MARK);
|
||||
space(SONY_HDR_SPACE);
|
||||
|
||||
// Data
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
mark(SONY_ONE_MARK);
|
||||
space(SONY_HDR_SPACE);
|
||||
} else {
|
||||
mark(SONY_ZERO_MARK);
|
||||
space(SONY_HDR_SPACE);
|
||||
}
|
||||
}
|
||||
|
||||
// We will have ended with LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
#if DECODE_SONY
|
||||
bool IRrecv::decodeSony (decode_results *results)
|
||||
{
|
||||
long data = 0;
|
||||
int offset = 0; // Dont skip first space, check its size
|
||||
|
||||
if (irparams.rawlen < (2 * SONY_BITS) + 2) return false ;
|
||||
|
||||
// Some Sony's deliver repeats fast after first
|
||||
// unfortunately can't spot difference from of repeat from two fast clicks
|
||||
if (results->rawbuf[offset] < SONY_DOUBLE_SPACE_USECS) {
|
||||
// Serial.print("IR Gap found: ");
|
||||
results->bits = 0;
|
||||
results->value = REPEAT;
|
||||
|
||||
# ifdef DECODE_SANYO
|
||||
results->decode_type = SANYO;
|
||||
# else
|
||||
results->decode_type = UNKNOWN;
|
||||
# endif
|
||||
|
||||
return true;
|
||||
}
|
||||
offset++;
|
||||
|
||||
// Initial mark
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], SONY_HDR_MARK)) return false ;
|
||||
|
||||
while (offset + 1 < irparams.rawlen) {
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], SONY_HDR_SPACE)) break ;
|
||||
|
||||
if (MATCH_MARK(results->rawbuf[offset], SONY_ONE_MARK)) data = (data << 1) | 1 ;
|
||||
else if (MATCH_MARK(results->rawbuf[offset], SONY_ZERO_MARK)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
// Success
|
||||
results->bits = (offset - 1) / 2;
|
||||
if (results->bits < 12) {
|
||||
results->bits = 0;
|
||||
return false;
|
||||
}
|
||||
results->value = data;
|
||||
results->decode_type = SONY;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
|
||||
179
Firmware/Source/grovepi/src/ir_Template.cpp
Normal file
179
Firmware/Source/grovepi/src/ir_Template.cpp
Normal file
|
|
@ -0,0 +1,179 @@
|
|||
/*
|
||||
Assuming the protocol we are adding is for the (imaginary) manufacturer: Shuzu
|
||||
|
||||
Our fantasy protocol is a standard protocol, so we can use this standard
|
||||
template without too much work. Some protocols are quite unique and will require
|
||||
considerably more work in this file! It is way beyond the scope of this text to
|
||||
explain how to reverse engineer "unusual" IR protocols. But, unless you own an
|
||||
oscilloscope, the starting point is probably to use the rawDump.ino sketch and
|
||||
try to spot the pattern!
|
||||
|
||||
Before you start, make sure the IR library is working OK:
|
||||
# Open up the Arduino IDE
|
||||
# Load up the rawDump.ino example sketch
|
||||
# Run it
|
||||
|
||||
Now we can start to add our new protocol...
|
||||
|
||||
1. Copy this file to : ir_Shuzu.cpp
|
||||
|
||||
2. Replace all occurrences of "Shuzu" with the name of your protocol.
|
||||
|
||||
3. Tweak the #defines to suit your protocol.
|
||||
|
||||
4. If you're lucky, tweaking the #defines will make the default send() function
|
||||
work.
|
||||
|
||||
5. Again, if you're lucky, tweaking the #defines will have made the default
|
||||
decode() function work.
|
||||
|
||||
You have written the code to support your new protocol!
|
||||
|
||||
Now you must do a few things to add it to the IRremote system:
|
||||
|
||||
1. Open IRremote.h and make the following changes:
|
||||
REMEMEBER to change occurences of "SHUZU" with the name of your protocol
|
||||
|
||||
A. At the top, in the section "Supported Protocols", add:
|
||||
#define DECODE_SHUZU 1
|
||||
#define SEND_SHUZU 1
|
||||
|
||||
B. In the section "enumerated list of all supported formats", add:
|
||||
SHUZU,
|
||||
to the end of the list (notice there is a comma after the protocol name)
|
||||
|
||||
C. Further down in "Main class for receiving IR", add:
|
||||
//......................................................................
|
||||
#if DECODE_SHUZU
|
||||
bool decodeShuzu (decode_results *results) ;
|
||||
#endif
|
||||
|
||||
D. Further down in "Main class for sending IR", add:
|
||||
//......................................................................
|
||||
#if SEND_SHUZU
|
||||
void sendShuzu (unsigned long data, int nbits) ;
|
||||
#endif
|
||||
|
||||
E. Save your changes and close the file
|
||||
|
||||
2. Now open irRecv.cpp and make the following change:
|
||||
|
||||
A. In the function IRrecv::decode(), add:
|
||||
#ifdef DECODE_NEC
|
||||
DBG_PRINTLN("Attempting Shuzu decode");
|
||||
if (decodeShuzu(results)) return true ;
|
||||
#endif
|
||||
|
||||
B. Save your changes and close the file
|
||||
|
||||
You will probably want to add your new protocol to the example sketch
|
||||
|
||||
3. Open MyDocuments\Arduino\libraries\IRremote\examples\IRrecvDumpV2.ino
|
||||
|
||||
A. In the encoding() function, add:
|
||||
case SHUZU: Serial.print("SHUZU"); break ;
|
||||
|
||||
Now open the Arduino IDE, load up the rawDump.ino sketch, and run it.
|
||||
Hopefully it will compile and upload.
|
||||
If it doesn't, you've done something wrong. Check your work.
|
||||
If you can't get it to work - seek help from somewhere.
|
||||
|
||||
If you get this far, I will assume you have successfully added your new protocol
|
||||
There is one last thing to do.
|
||||
|
||||
1. Delete this giant instructional comment.
|
||||
|
||||
2. Send a copy of your work to us so we can include it in the library and
|
||||
others may benefit from your hard work and maybe even write a song about how
|
||||
great you are for helping them! :)
|
||||
|
||||
Regards,
|
||||
BlueChip
|
||||
*/
|
||||
|
||||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
//
|
||||
//
|
||||
// S H U Z U
|
||||
//
|
||||
//
|
||||
//==============================================================================
|
||||
|
||||
#define BITS 32 // The number of bits in the command
|
||||
|
||||
#define HDR_MARK 1000 // The length of the Header:Mark
|
||||
#define HDR_SPACE 2000 // The lenght of the Header:Space
|
||||
|
||||
#define BIT_MARK 3000 // The length of a Bit:Mark
|
||||
#define ONE_SPACE 4000 // The length of a Bit:Space for 1's
|
||||
#define ZERO_SPACE 5000 // The length of a Bit:Space for 0's
|
||||
|
||||
#define OTHER 1234 // Other things you may need to define
|
||||
|
||||
//+=============================================================================
|
||||
//
|
||||
#if SEND_SHUZU
|
||||
void IRsend::sendShuzu (unsigned long data, int nbits)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(38);
|
||||
|
||||
// Header
|
||||
mark (HDR_MARK);
|
||||
space(HDR_SPACE);
|
||||
|
||||
// Data
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
mark (BIT_MARK);
|
||||
space(ONE_SPACE);
|
||||
} else {
|
||||
mark (BIT_MARK);
|
||||
space(ZERO_SPACE);
|
||||
}
|
||||
}
|
||||
|
||||
// Footer
|
||||
mark(BIT_MARK);
|
||||
space(0); // Always end with the LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
//
|
||||
#if DECODE_SHUZU
|
||||
bool IRrecv::decodeShuzu (decode_results *results)
|
||||
{
|
||||
unsigned long data = 0; // Somewhere to build our code
|
||||
int offset = 1; // Skip the Gap reading
|
||||
|
||||
// Check we have the right amount of data
|
||||
if (irparams.rawlen != 1 + 2 + (2 * BITS) + 1) return false ;
|
||||
|
||||
// Check initial Mark+Space match
|
||||
if (!MATCH_MARK (results->rawbuf[offset++], HDR_MARK )) return false ;
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], HDR_SPACE)) return false ;
|
||||
|
||||
// Read the bits in
|
||||
for (int i = 0; i < SHUZU_BITS; i++) {
|
||||
// Each bit looks like: MARK + SPACE_1 -> 1
|
||||
// or : MARK + SPACE_0 -> 0
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], BIT_MARK)) return false ;
|
||||
|
||||
// IR data is big-endian, so we shuffle it in from the right:
|
||||
if (MATCH_SPACE(results->rawbuf[offset], ONE_SPACE)) data = (data << 1) | 1 ;
|
||||
else if (MATCH_SPACE(results->rawbuf[offset], ZERO_SPACE)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
// Success
|
||||
results->bits = BITS;
|
||||
results->value = data;
|
||||
results->decode_type = SHUZU;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
91
Firmware/Source/grovepi/src/ir_Whynter.cpp
Normal file
91
Firmware/Source/grovepi/src/ir_Whynter.cpp
Normal file
|
|
@ -0,0 +1,91 @@
|
|||
#include "IRremote.h"
|
||||
#include "IRremoteInt.h"
|
||||
|
||||
//==============================================================================
|
||||
// W W H H Y Y N N TTTTT EEEEE RRRRR
|
||||
// W W H H Y Y NN N T E R R
|
||||
// W W W HHHHH Y N N N T EEE RRRR
|
||||
// W W W H H Y N NN T E R R
|
||||
// WWW H H Y N N T EEEEE R R
|
||||
//==============================================================================
|
||||
|
||||
#define WHYNTER_BITS 32
|
||||
#define WHYNTER_HDR_MARK 2850
|
||||
#define WHYNTER_HDR_SPACE 2850
|
||||
#define WHYNTER_BIT_MARK 750
|
||||
#define WHYNTER_ONE_MARK 750
|
||||
#define WHYNTER_ONE_SPACE 2150
|
||||
#define WHYNTER_ZERO_MARK 750
|
||||
#define WHYNTER_ZERO_SPACE 750
|
||||
|
||||
//+=============================================================================
|
||||
#if SEND_WHYNTER
|
||||
void IRsend::sendWhynter (unsigned long data, int nbits)
|
||||
{
|
||||
// Set IR carrier frequency
|
||||
enableIROut(38);
|
||||
|
||||
// Start
|
||||
mark(WHYNTER_ZERO_MARK);
|
||||
space(WHYNTER_ZERO_SPACE);
|
||||
|
||||
// Header
|
||||
mark(WHYNTER_HDR_MARK);
|
||||
space(WHYNTER_HDR_SPACE);
|
||||
|
||||
// Data
|
||||
for (unsigned long mask = 1UL << (nbits - 1); mask; mask >>= 1) {
|
||||
if (data & mask) {
|
||||
mark(WHYNTER_ONE_MARK);
|
||||
space(WHYNTER_ONE_SPACE);
|
||||
} else {
|
||||
mark(WHYNTER_ZERO_MARK);
|
||||
space(WHYNTER_ZERO_SPACE);
|
||||
}
|
||||
}
|
||||
|
||||
// Footer
|
||||
mark(WHYNTER_ZERO_MARK);
|
||||
space(WHYNTER_ZERO_SPACE); // Always end with the LED off
|
||||
}
|
||||
#endif
|
||||
|
||||
//+=============================================================================
|
||||
#if DECODE_WHYNTER
|
||||
bool IRrecv::decodeWhynter (decode_results *results)
|
||||
{
|
||||
long data = 0;
|
||||
int offset = 1; // skip initial space
|
||||
|
||||
// Check we have the right amount of data
|
||||
if (irparams.rawlen < (2 * WHYNTER_BITS) + 6) return false ;
|
||||
|
||||
// Sequence begins with a bit mark and a zero space
|
||||
if (!MATCH_MARK (results->rawbuf[offset++], WHYNTER_BIT_MARK )) return false ;
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], WHYNTER_ZERO_SPACE)) return false ;
|
||||
|
||||
// header mark and space
|
||||
if (!MATCH_MARK (results->rawbuf[offset++], WHYNTER_HDR_MARK )) return false ;
|
||||
if (!MATCH_SPACE(results->rawbuf[offset++], WHYNTER_HDR_SPACE)) return false ;
|
||||
|
||||
// data bits
|
||||
for (int i = 0; i < WHYNTER_BITS; i++) {
|
||||
if (!MATCH_MARK(results->rawbuf[offset++], WHYNTER_BIT_MARK)) return false ;
|
||||
|
||||
if (MATCH_SPACE(results->rawbuf[offset], WHYNTER_ONE_SPACE )) data = (data << 1) | 1 ;
|
||||
else if (MATCH_SPACE(results->rawbuf[offset], WHYNTER_ZERO_SPACE)) data = (data << 1) | 0 ;
|
||||
else return false ;
|
||||
offset++;
|
||||
}
|
||||
|
||||
// trailing mark
|
||||
if (!MATCH_MARK(results->rawbuf[offset], WHYNTER_BIT_MARK)) return false ;
|
||||
|
||||
// Success
|
||||
results->bits = WHYNTER_BITS;
|
||||
results->value = data;
|
||||
results->decode_type = WHYNTER;
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
|
||||
Loading…
Add table
Add a link
Reference in a new issue