/* Copyright (c) 2017 Heiko Krupp and Theo Arends. All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: - Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. - Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifdef USE_IR_REMOTE /*********************************************************************************************\ * IR Remote send using IRremoteESP8266 library \*********************************************************************************************/ // * Add support for Toshiba and Mitsubishi HVAC IR control (#257) // #define USE_IR_HVAC // Support for HVAC system using IR (+2k code) #ifndef USE_IR_HVAC #include #else #include // Currently firmware.elf section `.text' will not fit in region `iram1_0_seg' // HVAC TOSHIBA_ #define HVAC_TOSHIBA_HDR_MARK 4400 #define HVAC_TOSHIBA_HDR_SPACE 4300 #define HVAC_TOSHIBA_BIT_MARK 543 #define HVAC_TOSHIBA_ONE_SPACE 1623 #define HVAC_MISTUBISHI_ZERO_SPACE 472 #define HVAC_TOSHIBA_RPT_MARK 440 #define HVAC_TOSHIBA_RPT_SPACE 7048 // Above original iremote limit #define HVAC_TOSHIBA_DATALEN 9 IRMitsubishiAC *mitsubir = NULL; #endif IRsend *irsend = NULL; void ir_send_init(void) { irsend = new IRsend(pin[GPIO_IRSEND]); // an IR led is at GPIO_IRSEND irsend->begin(); #ifdef USE_IR_HVAC mitsubir = new IRMitsubishiAC(pin[GPIO_IRSEND]); #endif //USE_IR_HVAC } /*********************************************************************************************\ * Commands \*********************************************************************************************/ /* * ArduinoJSON entry used to calculate jsonBuf: JSON_OBJECT_SIZE(3) + 40 = 96 IRsend: { "protocol": "SAMSUNG", "bits": 32, "data": 551502015 } IRhvac: { "Vendor": "", "Power": <0|1>, "Mode": "", "FanSpeed": "<1|2|3|4|5|Auto|Silence>", "Temp": <17..30> } */ boolean ir_send_command(char *type, uint16_t index, char *dataBufUc, uint16_t data_len, int16_t payload, char *svalue, uint16_t ssvalue) { boolean serviced = true; boolean error = false; const char *protocol; uint8_t bits = 0; uint32_t data = 0; const char *HVAC_Mode; const char *HVAC_FanMode; const char *HVAC_Vendor; int HVAC_Temp = 21; boolean HVAC_Power = true; // char log[LOGSZ]; if (!strcmp(type,"IRSEND")) { if (data_len) { StaticJsonBuffer<128> jsonBuf; JsonObject &ir_json = jsonBuf.parseObject(dataBufUc); if (!ir_json.success()) { snprintf_P(svalue, ssvalue, PSTR("{\"IRSend\":\"Invalid JSON\"}")); // JSON decode failed } else { snprintf_P(svalue, ssvalue, PSTR("{\"IRSend\":\"Done\"}")); protocol = ir_json["PROTOCOL"]; bits = ir_json["BITS"]; data = ir_json["DATA"]; if (protocol && bits && data) { if (!strcmp(protocol,"NEC")) irsend->sendNEC(data, bits); else if (!strcmp(protocol,"SONY")) irsend->sendSony(data, bits); else if (!strcmp(protocol,"RC5")) irsend->sendRC5(data, bits); else if (!strcmp(protocol,"RC6")) irsend->sendRC6(data, bits); else if (!strcmp(protocol,"DISH")) irsend->sendDISH(data, bits); else if (!strcmp(protocol,"JVC")) irsend->sendJVC(data, bits, 1); else if (!strcmp(protocol,"SAMSUNG")) irsend->sendSAMSUNG(data, bits); else { snprintf_P(svalue, ssvalue, PSTR("{\"IRSend\":\"Protocol not supported\"}")); } } else error = true; } } else error = true; if (error) snprintf_P(svalue, ssvalue, PSTR("{\"IRSend\":\"No protocol, bits or data\"}")); } #ifdef USE_IR_HVAC else if (!strcmp(type,"IRHVAC")) { if (data_len) { StaticJsonBuffer<164> jsonBufer; JsonObject &root = jsonBufer.parseObject(dataBufUc); if (!root.success()) { snprintf_P(svalue, ssvalue, PSTR("{\"IRHVAC\":\"Invalid JSON\"}")); // JSON decode failed } else { snprintf_P(svalue, ssvalue, PSTR("{\"IRHVAC\":\"Done\"}")); HVAC_Vendor = root["VENDOR"]; HVAC_Power = root["POWER"]; HVAC_Mode = root["MODE"]; HVAC_FanMode = root["FANSPEED"]; HVAC_Temp = root["TEMP"]; // snprintf_P(log, sizeof(log), PSTR("IRHVAC: Received Vendor %s, Power %d, Mode %s, FanSpeed %s, Temp %d"), // HVAC_Vendor, HVAC_Power, HVAC_Mode, HVAC_FanMode, HVAC_Temp); // addLog(LOG_LEVEL_DEBUG, log); if (HVAC_Vendor == NULL || !strcmp(HVAC_Vendor,"TOSHIBA")) { error = ir_hvac_toshiba(HVAC_Mode, HVAC_FanMode, HVAC_Power, HVAC_Temp); } else if (!strcmp(HVAC_Vendor,"MITSUBISHI")) { error = ir_hvac_mitsubishi(HVAC_Mode, HVAC_FanMode, HVAC_Power, HVAC_Temp); } else error = true; } } else error = true; if (error) snprintf_P(svalue, ssvalue, PSTR("{\"IRHVAC\":\"Wrong parameters value for Vendor, Mode and/or FanSpeed\"}")); } #endif // USE_IR_HVAC else { serviced = false; // Unknown command } return serviced; } #ifdef USE_IR_HVAC boolean ir_hvac_toshiba(const char *HVAC_Mode, const char *HVAC_FanMode, boolean HVAC_Power, int HVAC_Temp) { unsigned int rawdata[2 + 2*8*HVAC_TOSHIBA_DATALEN + 2]; byte data[HVAC_TOSHIBA_DATALEN] = { 0xF2, 0x0D, 0x03, 0xFC, 0x01, 0x00, 0x00, 0x00, 0x00 }; boolean error = false; if (HVAC_Mode == NULL || !strcmp(HVAC_Mode,"HOT")) { //default HVAC_HOT data[6] = (byte) B00000011; } else if (HVAC_Mode && !strcmp(HVAC_Mode,"COLD")) { data[6] = (byte) B00000001; } else if (HVAC_Mode && !strcmp(HVAC_Mode,"DRY")) { data[6] = (byte) B00000010; } else if (HVAC_Mode && !strcmp(HVAC_Mode,"AUTO")) { data[6] = (byte) B00000000; } else error = true; if (!HVAC_Power) data[6] = (byte) 0x07; // Turn OFF HVAC if (HVAC_FanMode && !strcmp(HVAC_FanMode,"1")) { data[6] = data[6] | (byte) B01000000; } else if (HVAC_FanMode && !strcmp(HVAC_FanMode,"2")) { data[6] = data[6] | (byte) B01100000; } else if (HVAC_FanMode && !strcmp(HVAC_FanMode,"3")) { data[6] = data[6] | (byte) B10000000; } else if (HVAC_FanMode && !strcmp(HVAC_FanMode,"4")) { data[6] = data[6] | (byte) B10100000; } else if (HVAC_FanMode && !strcmp(HVAC_FanMode,"5")) { data[6] = data[6] | (byte) B11000000; } else if (HVAC_FanMode == NULL || !strcmp(HVAC_FanMode,"AUTO")) { // default FAN_SPEED_AUTO data[6] = data[6] | (byte) B00000000; } else if (HVAC_FanMode && !strcmp(HVAC_FanMode,"SILENT")) { data[6] = data[6] | (byte) B00000000; } else error = true; byte Temp; if (HVAC_Temp > 30) { Temp = 30; } else if (HVAC_Temp < 17) { Temp = 17; } else Temp = HVAC_Temp; data[5] = (byte) Temp - 17 << 4; data[HVAC_TOSHIBA_DATALEN-1] = 0; for (int x = 0; x < HVAC_TOSHIBA_DATALEN - 1; x++) { data[HVAC_TOSHIBA_DATALEN-1] = (byte) data[x] ^ data[HVAC_TOSHIBA_DATALEN -1]; // CRC is a simple bits addition } int i = 0; byte mask = 1; //header rawdata[i++] = HVAC_TOSHIBA_HDR_MARK; rawdata[i++] = HVAC_TOSHIBA_HDR_SPACE; //data for (int b = 0; b < HVAC_TOSHIBA_DATALEN; b++) { for (mask = B10000000; mask > 0; mask >>= 1) { //iterate through bit mask if (data[b] & mask) { // Bit ONE rawdata[i++] = HVAC_TOSHIBA_BIT_MARK; rawdata[i++] = HVAC_TOSHIBA_ONE_SPACE; } else { // Bit ZERO rawdata[i++] = HVAC_TOSHIBA_BIT_MARK; rawdata[i++] = HVAC_MISTUBISHI_ZERO_SPACE; } } } //trailer rawdata[i++] = HVAC_TOSHIBA_RPT_MARK; rawdata[i++] = HVAC_TOSHIBA_RPT_SPACE; noInterrupts(); irsend->sendRaw(rawdata,i,38); irsend->sendRaw(rawdata,i,38); interrupts(); return error; } boolean ir_hvac_mitsubishi(const char *HVAC_Mode,const char *HVAC_FanMode, boolean HVAC_Power, int HVAC_Temp) { boolean error = false; char log[LOGSZ]; mitsubir->stateReset(); if (HVAC_Mode == NULL || !strcmp(HVAC_Mode,"HOT")) { // default HVAC_HOT mitsubir->setMode(MITSUBISHI_AC_HEAT); } else if (HVAC_Mode && !strcmp(HVAC_Mode,"COLD")) { mitsubir->setMode(MITSUBISHI_AC_COOL); } else if (HVAC_Mode && !strcmp(HVAC_Mode,"DRY")) { mitsubir->setMode(MITSUBISHI_AC_DRY); } else if (HVAC_Mode && !strcmp(HVAC_Mode,"AUTO")) { mitsubir->setMode(MITSUBISHI_AC_AUTO); } else error = true; mitsubir->setPower(~HVAC_Power); if (HVAC_FanMode && !strcmp(HVAC_FanMode,"1")) { mitsubir->setFan(1); } else if (HVAC_FanMode && !strcmp(HVAC_FanMode,"2")) { mitsubir->setFan(2); } else if (HVAC_FanMode && !strcmp(HVAC_FanMode,"3")) { mitsubir->setFan(3); } else if (HVAC_FanMode && !strcmp(HVAC_FanMode,"4")) { mitsubir->setFan(4); } else if (HVAC_FanMode && !strcmp(HVAC_FanMode,"5")) { mitsubir->setFan(5); } else if (HVAC_FanMode == NULL || !strcmp(HVAC_FanMode,"AUTO")) { // default FAN_SPEED_AUTO mitsubir->setFan(MITSUBISHI_AC_FAN_AUTO); } else if (HVAC_FanMode && !strcmp(HVAC_FanMode,"SILENT")) { mitsubir->setFan(MITSUBISHI_AC_FAN_SILENT); } else error = true; mitsubir->setTemp(HVAC_Temp); mitsubir->setVane(MITSUBISHI_AC_VANE_AUTO); mitsubir->send(); snprintf_P(log, sizeof(log), PSTR("IRHVAC: Sent to Mitsubishi. Power %d, Mode %d, FanSpeed %d, Temp %d, VaneMode %d"), mitsubir->getPower(), mitsubir->getMode(), mitsubir->getFan(), mitsubir->getTemp(), mitsubir->getVane()); addLog(LOG_LEVEL_DEBUG, log); return error; } #endif // USE_IR_HVAC #endif // USE_IR_REMOTE