Tasmota/lib/lib_basic/IRremoteESP8266-2.7.15/src/ir_Voltas.cpp
2021-02-13 14:51:52 +01:00

517 lines
18 KiB
C++

// Copyright 2020 David Conran (crankyoldgit)
// Copyright 2020 manj9501
/// @file
/// @brief Support for Voltas A/C protocol
/// @see https://github.com/crankyoldgit/IRremoteESP8266/issues/1238
#include "ir_Voltas.h"
#include <algorithm>
#include <cstring>
#include "IRrecv.h"
#include "IRsend.h"
#include "IRtext.h"
#include "IRutils.h"
using irutils::addBoolToString;
using irutils::addModelToString;
using irutils::addModeToString;
using irutils::addFanToString;
using irutils::addLabeledString;
using irutils::addTempToString;
using irutils::minsToString;
// Constants
const uint16_t kVoltasBitMark = 1026; ///< uSeconds.
const uint16_t kVoltasOneSpace = 2553; ///< uSeconds.
const uint16_t kVoltasZeroSpace = 554; ///< uSeconds.
const uint16_t kVoltasFreq = 38000; ///< Hz.
#if SEND_VOLTAS
/// Send a Voltas formatted message.
/// Status: STABLE / Working on real device.
/// @param[in] data An array of bytes containing the IR command.
/// It is assumed to be in MSB order for this code.
/// e.g.
/// @code
/// uint8_t data[kVoltasStateLength] = {0x33, 0x28, 0x88, 0x1A, 0x3B, 0x3B,
/// 0x3B, 0x11, 0x00, 0x40};
/// @endcode
/// @param[in] nbytes Nr. of bytes of data in the array. (>=kVoltasStateLength)
/// @param[in] repeat Nr. of times the message is to be repeated.
void IRsend::sendVoltas(const uint8_t data[], const uint16_t nbytes,
const uint16_t repeat) {
sendGeneric(0, 0,
kVoltasBitMark, kVoltasOneSpace,
kVoltasBitMark, kVoltasZeroSpace,
kVoltasBitMark, kDefaultMessageGap,
data, nbytes,
kVoltasFreq, true, repeat, kDutyDefault);
}
#endif // SEND_VOLTAS
#if DECODE_VOLTAS
/// Decode the supplied Voltas message.
/// Status: STABLE / Working on real device.
/// @param[in,out] results Ptr to the data to decode & where to store the decode
/// @param[in] offset The starting index to use when attempting to decode the
/// raw data. Typically/Defaults to kStartOffset.
/// @param[in] nbits The number of data bits to expect.
/// @param[in] strict Flag indicating if we should perform strict matching.
/// @return A boolean. True if it can decode it, false if it can't.
bool IRrecv::decodeVoltas(decode_results *results, uint16_t offset,
const uint16_t nbits, const bool strict) {
if (strict && nbits != kVoltasBits) return false;
// Data + Footer
if (!matchGeneric(results->rawbuf + offset, results->state,
results->rawlen - offset, nbits,
0, 0, // No header
kVoltasBitMark, kVoltasOneSpace,
kVoltasBitMark, kVoltasZeroSpace,
kVoltasBitMark, kDefaultMessageGap, true)) return false;
// Compliance
if (strict && !IRVoltas::validChecksum(results->state, nbits / 8))
return false;
// Success
results->decode_type = decode_type_t::VOLTAS;
results->bits = nbits;
return true;
}
#endif // DECODE_VOLTAS
/// Class constructor
/// @param[in] pin GPIO to be used when sending.
/// @param[in] inverted Is the output signal to be inverted?
/// @param[in] use_modulation Is frequency modulation to be used?
IRVoltas::IRVoltas(const uint16_t pin, const bool inverted,
const bool use_modulation)
: _irsend(pin, inverted, use_modulation) {
stateReset();
}
// Reset the internal state to a fixed known good state.
void IRVoltas::stateReset() {
// This resets to a known-good state.
// ref: https://github.com/crankyoldgit/IRremoteESP8266/issues/1238#issuecomment-674699746
const uint8_t kReset[kVoltasStateLength] = {
0x33, 0x28, 0x00, 0x17, 0x3B, 0x3B, 0x3B, 0x11, 0x00, 0xCB};
setRaw(kReset);
}
/// Set up hardware to be able to send a message.
void IRVoltas::begin() { _irsend.begin(); }
#if SEND_VOLTAS
/// Send the current internal state as an IR message.
/// @param[in] repeat Nr. of times the message will be repeated.
void IRVoltas::send(const uint16_t repeat) {
_irsend.sendVoltas(getRaw(), kVoltasStateLength, repeat);
}
#endif // SEND_VOLTAS
/// Get the model information currently known.
/// @param[in] raw Work out the model info from the current raw state.
/// @return The known model number.
voltas_ac_remote_model_t IRVoltas::getModel(const bool raw) const {
if (raw) {
switch (_.SwingHChange) {
case kVoltasSwingHNoChange:
return voltas_ac_remote_model_t::kVoltas122LZF;
default:
return voltas_ac_remote_model_t::kVoltasUnknown;
}
} else {
return _model;
}
}
/// Set the current model for the remote.
/// @param[in] model The model number.
void IRVoltas::setModel(const voltas_ac_remote_model_t model) {
switch (model) {
case voltas_ac_remote_model_t::kVoltas122LZF:
_model = model;
setSwingHChange(false);
break;
default: _model = voltas_ac_remote_model_t::kVoltasUnknown;
}
}
/// Get a PTR to the internal state/code for this protocol.
/// @return PTR to a code for this protocol based on the current internal state.
uint8_t* IRVoltas::getRaw(void) {
checksum(); // Ensure correct settings before sending.
return _.raw;
}
/// Set the internal state from a valid code for this protocol.
/// @param[in] new_code A valid code for this protocol.
void IRVoltas::setRaw(const uint8_t new_code[]) {
std::memcpy(_.raw, new_code, kVoltasStateLength);
setModel(getModel(true));
}
/// Calculate and set the checksum values for the internal state.
void IRVoltas::checksum(void) {
_.Checksum = calcChecksum(_.raw);
}
/// Verify the checksum is valid for a given state.
/// @param[in] state The array to verify the checksum of.
/// @param[in] length The length of the state array.
/// @return true, if the state has a valid checksum. Otherwise, false.
bool IRVoltas::validChecksum(const uint8_t state[], const uint16_t length) {
if (length) return state[length - 1] == calcChecksum(state, length);
return true;
}
/// Calculate the checksum is valid for a given state.
/// @param[in] state The array to calculate the checksum of.
/// @param[in] length The length of the state array.
/// @return The valid checksum value for the state.
uint8_t IRVoltas::calcChecksum(const uint8_t state[], const uint16_t length) {
uint8_t result = 0;
if (length)
result = sumBytes(state, length - 1);
return ~result;
}
/// Change the power setting to On.
void IRVoltas::on() { setPower(true); }
/// Change the power setting to Off.
void IRVoltas::off() { setPower(false); }
/// Change the power setting.
/// @param[in] on true, the setting is on. false, the setting is off.
void IRVoltas::setPower(const bool on) { _.Power = on; }
/// Get the value of the current power setting.
/// @return true, the setting is on. false, the setting is off.
bool IRVoltas::getPower(void) const { return _.Power; }
/// Set the operating mode of the A/C.
/// @param[in] mode The desired operating mode.
/// @note If we get an unexpected mode, default to AUTO.
void IRVoltas::setMode(const uint8_t mode) {
_.Mode = mode;
switch (mode) {
case kVoltasFan:
setFan(getFan()); // Force the fan speed to a correct one fo the mode.
break;
case kVoltasDry:
setFan(kVoltasFanLow);
setTemp(kVoltasDryTemp);
break;
case kVoltasHeat:
case kVoltasCool:
break;
default:
setMode(kVoltasCool);
return;
}
// Reset some settings if needed.
setEcono(getEcono());
setTurbo(getTurbo());
setSleep(getSleep());
}
/// Get the operating mode setting of the A/C.
/// @return The current operating mode setting.
uint8_t IRVoltas::getMode(void) { return _.Mode; }
/// Convert a stdAc::opmode_t enum into its native mode.
/// @param[in] mode The enum to be converted.
/// @return The native equivalent of the enum.
uint8_t IRVoltas::convertMode(const stdAc::opmode_t mode) {
switch (mode) {
case stdAc::opmode_t::kHeat: return kVoltasHeat;
case stdAc::opmode_t::kDry: return kVoltasDry;
case stdAc::opmode_t::kFan: return kVoltasFan;
default: return kVoltasCool;
}
}
/// Convert a native mode into its stdAc equivalent.
/// @param[in] mode The native setting to be converted.
/// @return The stdAc equivalent of the native setting.
stdAc::opmode_t IRVoltas::toCommonMode(const uint8_t mode) {
switch (mode) {
case kVoltasHeat: return stdAc::opmode_t::kHeat;
case kVoltasDry: return stdAc::opmode_t::kDry;
case kVoltasFan: return stdAc::opmode_t::kFan;
default: return stdAc::opmode_t::kCool;
}
}
/// Set the temperature.
/// @param[in] temp The temperature in degrees celsius.
void IRVoltas::setTemp(const uint8_t temp) {
uint8_t new_temp = std::max(kVoltasMinTemp, temp);
new_temp = std::min(kVoltasMaxTemp, new_temp);
_.Temp = new_temp - kVoltasMinTemp;
}
/// Get the current temperature setting.
/// @return The current setting for temp. in degrees celsius.
uint8_t IRVoltas::getTemp(void) { return _.Temp + kVoltasMinTemp; }
/// Set the speed of the fan.
/// @param[in] fan The desired setting.
void IRVoltas::setFan(const uint8_t fan) {
switch (fan) {
case kVoltasFanAuto:
if (_.Mode == kVoltasFan) { // Auto speed is not available in fan mode.
setFan(kVoltasFanHigh);
return;
}
// FALL-THRU
case kVoltasFanLow:
case kVoltasFanMed:
case kVoltasFanHigh:
_.FanSpeed = fan;
break;
default:
setFan(kVoltasFanAuto);
}
}
/// Get the current fan speed setting.
/// @return The current fan speed/mode.
uint8_t IRVoltas::getFan(void) { return _.FanSpeed; }
/// Convert a stdAc::fanspeed_t enum into it's native speed.
/// @param[in] speed The enum to be converted.
/// @return The native equivalent of the enum.
uint8_t IRVoltas::convertFan(const stdAc::fanspeed_t speed) {
switch (speed) {
case stdAc::fanspeed_t::kMin:
case stdAc::fanspeed_t::kLow: return kVoltasFanLow;
case stdAc::fanspeed_t::kMedium: return kVoltasFanMed;
case stdAc::fanspeed_t::kHigh:
case stdAc::fanspeed_t::kMax: return kVoltasFanHigh;
default: return kVoltasFanAuto;
}
}
/// Convert a native fan speed into its stdAc equivalent.
/// @param[in] spd The native setting to be converted.
/// @return The stdAc equivalent of the native setting.
stdAc::fanspeed_t IRVoltas::toCommonFanSpeed(const uint8_t spd) {
switch (spd) {
case kVoltasFanHigh: return stdAc::fanspeed_t::kMax;
case kVoltasFanMed: return stdAc::fanspeed_t::kMedium;
case kVoltasFanLow: return stdAc::fanspeed_t::kMin;
default: return stdAc::fanspeed_t::kAuto;
}
}
/// Set the Vertical Swing setting of the A/C.
/// @param[in] on true, the setting is on. false, the setting is off.
void IRVoltas::setSwingV(const bool on) { _.SwingV = on ? 0b111 : 0b000; }
/// Get the Vertical Swing setting of the A/C.
/// @return true, the setting is on. false, the setting is off.
bool IRVoltas::getSwingV(void) const { return _.SwingV == 0b111; }
/// Set the Horizontal Swing setting of the A/C.
/// @param[in] on true, the setting is on. false, the setting is off.
void IRVoltas::setSwingH(const bool on) {
switch (_model) {
case voltas_ac_remote_model_t::kVoltas122LZF:
break; // unsupported on these models.
default:
_.SwingH = on;
setSwingHChange(true);
}
}
/// Get the Horizontal Swing setting of the A/C.
/// @return true, the setting is on. false, the setting is off.
bool IRVoltas::getSwingH(void) const {
switch (_model) {
case voltas_ac_remote_model_t::kVoltas122LZF:
return false; // unsupported on these models.
default:
return _.SwingH;
}
}
/// Set the bits for changing the Horizontal Swing setting of the A/C.
/// @param[in] on true, the change bits are set.
/// false, the "no change" bits are set.
void IRVoltas::setSwingHChange(const bool on) {
_.SwingHChange = on ? kVoltasSwingHChange : kVoltasSwingHNoChange;
if (!on) _.SwingH = true; // "No Change" also sets SwingH to 1.
}
/// Are the Horizontal Swing change bits set in the message?
/// @return true, the correct bits are set. false, the correct bits are not set.
bool IRVoltas::getSwingHChange(void) const {
return _.SwingHChange == kVoltasSwingHChange;
}
/// Change the Wifi setting.
/// @param[in] on true, the setting is on. false, the setting is off.
void IRVoltas::setWifi(const bool on) { _.Wifi = on; }
/// Get the value of the current Wifi setting.
/// @return true, the setting is on. false, the setting is off.
bool IRVoltas::getWifi(void) const { return _.Wifi; }
/// Change the Turbo setting.
/// @param[in] on true, the setting is on. false, the setting is off.
/// @note The Turbo setting is only available in Cool mode.
void IRVoltas::setTurbo(const bool on) {
if (on && _.Mode == kVoltasCool)
_.Turbo = true;
else
_.Turbo = false;
}
/// Get the value of the current Turbo setting.
/// @return true, the setting is on. false, the setting is off.
bool IRVoltas::getTurbo(void) const { return _.Turbo; }
/// Change the Economy setting.
/// @param[in] on true, the setting is on. false, the setting is off.
/// @note The Economy setting is only available in Cool mode.
void IRVoltas::setEcono(const bool on) {
if (on && _.Mode == kVoltasCool)
_.Econo = true;
else
_.Econo = false;
}
/// Get the value of the current Econo setting.
/// @return true, the setting is on. false, the setting is off.
bool IRVoltas::getEcono(void) const { return _.Econo; }
/// Change the Light setting.
/// @param[in] on true, the setting is on. false, the setting is off.
void IRVoltas::setLight(const bool on) { _.Light = on; }
/// Get the value of the current Light setting.
/// @return true, the setting is on. false, the setting is off.
bool IRVoltas::getLight(void) const { return _.Light; }
/// Change the Sleep setting.
/// @param[in] on true, the setting is on. false, the setting is off.
/// @note The Sleep setting is only available in Cool mode.
void IRVoltas::setSleep(const bool on) {
if (on && _.Mode == kVoltasCool)
_.Sleep = true;
else
_.Sleep = false;
}
/// Get the value of the current Sleep setting.
/// @return true, the setting is on. false, the setting is off.
bool IRVoltas::getSleep(void) const { return _.Sleep; }
/// Get the value of the On Timer time.
/// @return Number of minutes before the timer activates.
uint16_t IRVoltas::getOnTime(void) const {
return std::min((unsigned)(12 * _.OnTimer12Hr + _.OnTimerHrs - 1), 23U) * 60 +
_.OnTimerMins;
}
/// Set the value of the On Timer time.
/// @param[in] nr_of_mins Number of minutes before the timer activates.
/// 0 disables the timer. Max is 23 hrs & 59 mins (1439 mins)
void IRVoltas::setOnTime(const uint16_t nr_of_mins) {
// Cap the total number of mins.
uint16_t mins = std::min(nr_of_mins, (uint16_t)(23 * 60 + 59));
uint16_t hrs = (mins / 60) + 1;
_.OnTimerMins = mins % 60;
_.OnTimer12Hr = hrs / 12;
_.OnTimerHrs = hrs % 12;
_.OnTimerEnable = (mins > 0); // Is the timer is to be enabled?
}
/// Get the value of the On Timer time.
/// @return Number of minutes before the timer activates.
uint16_t IRVoltas::getOffTime(void) const {
return std::min((unsigned)(12 * _.OffTimer12Hr + _.OffTimerHrs - 1), 23U) *
60 + _.OffTimerMins;
}
/// Set the value of the Off Timer time.
/// @param[in] nr_of_mins Number of minutes before the timer activates.
/// 0 disables the timer. Max is 23 hrs & 59 mins (1439 mins)
void IRVoltas::setOffTime(const uint16_t nr_of_mins) {
// Cap the total number of mins.
uint16_t mins = std::min(nr_of_mins, (uint16_t)(23 * 60 + 59));
uint16_t hrs = (mins / 60) + 1;
_.OffTimerMins = mins % 60;
_.OffTimer12Hr = hrs / 12;
_.OffTimerHrs = hrs % 12;
_.OffTimerEnable = (mins > 0); // Is the timer is to be enabled?
}
/// Convert the current internal state into its stdAc::state_t equivalent.
/// @param[in] prev Ptr to the previous state if available.
/// @return The stdAc equivalent of the native settings.
stdAc::state_t IRVoltas::toCommon(const stdAc::state_t *prev) {
stdAc::state_t result;
// Start with the previous state if given it.
if (prev != NULL) {
result = *prev;
} else {
// Set defaults for non-zero values that are not implicitly set for when
// there is no previous state.
result.swingh = stdAc::swingh_t::kOff;
}
result.model = getModel();
result.protocol = decode_type_t::VOLTAS;
result.power = _.Power;
result.mode = toCommonMode(_.Mode);
result.celsius = true;
result.degrees = getTemp();
result.fanspeed = toCommonFanSpeed(_.FanSpeed);
result.swingv = getSwingV() ? stdAc::swingv_t::kAuto : stdAc::swingv_t::kOff;
if (getSwingHChange())
result.swingh = _.SwingH ? stdAc::swingh_t::kAuto : stdAc::swingh_t::kOff;
result.turbo = _.Turbo;
result.econo = _.Econo;
result.light = _.Light;
result.sleep = _.Sleep ? 0 : -1;
// Not supported.
result.quiet = false;
result.filter = false;
result.clean = false;
result.beep = false;
result.clock = -1;
return result;
}
/// Convert the current internal state into a human readable string.
/// @return A human readable string.
String IRVoltas::toString() {
String result = "";
result.reserve(200); // Reserve some heap for the string to reduce fragging.
result += addModelToString(decode_type_t::VOLTAS, getModel(), false);
result += addBoolToString(_.Power, kPowerStr);
result += addModeToString(_.Mode, 255, kVoltasCool, kVoltasHeat,
kVoltasDry, kVoltasFan);
result += addTempToString(getTemp());
result += addFanToString(_.FanSpeed, kVoltasFanHigh, kVoltasFanLow,
kVoltasFanAuto, kVoltasFanAuto, kVoltasFanMed);
result += addBoolToString(getSwingV(), kSwingVStr);
if (getSwingHChange())
result += addBoolToString(_.SwingH, kSwingHStr);
else
result += addLabeledString(kNAStr, kSwingHStr);
result += addBoolToString(_.Turbo, kTurboStr);
result += addBoolToString(_.Econo, kEconoStr);
result += addBoolToString(_.Wifi, kWifiStr);
result += addBoolToString(_.Light, kLightStr);
result += addBoolToString(_.Sleep, kSleepStr);
result += addLabeledString(_.OnTimerEnable ? minsToString(getOnTime())
: kOffStr, kOnTimerStr);
result += addLabeledString(_.OffTimerEnable ? minsToString(getOffTime())
: kOffStr, kOffTimerStr);
return result;
}