mirror of
https://github.com/esphome/esphome.git
synced 2025-01-18 12:05:41 +00:00
Add new Toshiba AC unit protocol (#1987)
This commit is contained in:
parent
90c0d3e12f
commit
922f7167f5
@ -136,6 +136,7 @@ esphome/components/tm1637/* @glmnet
|
||||
esphome/components/tmp102/* @timsavage
|
||||
esphome/components/tmp117/* @Azimath
|
||||
esphome/components/tof10120/* @wstrzalka
|
||||
esphome/components/toshiba/* @kbx81
|
||||
esphome/components/tuya/binary_sensor/* @jesserockz
|
||||
esphome/components/tuya/climate/* @jesserockz
|
||||
esphome/components/tuya/sensor/* @jesserockz
|
||||
|
@ -1,16 +1,25 @@
|
||||
import esphome.codegen as cg
|
||||
import esphome.config_validation as cv
|
||||
from esphome.components import climate_ir
|
||||
from esphome.const import CONF_ID
|
||||
from esphome.const import CONF_ID, CONF_MODEL
|
||||
|
||||
AUTO_LOAD = ["climate_ir"]
|
||||
CODEOWNERS = ["@kbx81"]
|
||||
|
||||
toshiba_ns = cg.esphome_ns.namespace("toshiba")
|
||||
ToshibaClimate = toshiba_ns.class_("ToshibaClimate", climate_ir.ClimateIR)
|
||||
|
||||
Model = toshiba_ns.enum("Model")
|
||||
MODELS = {
|
||||
"GENERIC": Model.MODEL_GENERIC,
|
||||
"RAC-PT1411HWRU-C": Model.MODEL_RAC_PT1411HWRU_C,
|
||||
"RAC-PT1411HWRU-F": Model.MODEL_RAC_PT1411HWRU_F,
|
||||
}
|
||||
|
||||
CONFIG_SCHEMA = climate_ir.CLIMATE_IR_WITH_RECEIVER_SCHEMA.extend(
|
||||
{
|
||||
cv.GenerateID(): cv.declare_id(ToshibaClimate),
|
||||
cv.Optional(CONF_MODEL, default="generic"): cv.enum(MODELS, upper=True),
|
||||
}
|
||||
)
|
||||
|
||||
@ -18,3 +27,4 @@ CONFIG_SCHEMA = climate_ir.CLIMATE_IR_WITH_RECEIVER_SCHEMA.extend(
|
||||
async def to_code(config):
|
||||
var = cg.new_Pvariable(config[CONF_ID])
|
||||
await climate_ir.register_climate_ir(var, config)
|
||||
cg.add(var.set_model(config[CONF_MODEL]))
|
||||
|
@ -3,13 +3,23 @@
|
||||
namespace esphome {
|
||||
namespace toshiba {
|
||||
|
||||
struct RacPt1411hwruFanSpeed {
|
||||
uint8_t code1;
|
||||
uint8_t code2;
|
||||
};
|
||||
|
||||
static const char *const TAG = "toshiba.climate";
|
||||
// Timings for IR bits/data
|
||||
const uint16_t TOSHIBA_HEADER_MARK = 4380;
|
||||
const uint16_t TOSHIBA_HEADER_SPACE = 4370;
|
||||
const uint16_t TOSHIBA_GAP_SPACE = 5480;
|
||||
const uint16_t TOSHIBA_PACKET_SPACE = 10500;
|
||||
const uint16_t TOSHIBA_BIT_MARK = 540;
|
||||
const uint16_t TOSHIBA_ZERO_SPACE = 540;
|
||||
const uint16_t TOSHIBA_ONE_SPACE = 1620;
|
||||
|
||||
const uint16_t TOSHIBA_CARRIER_FREQUENCY = 38000;
|
||||
const uint8_t TOSHIBA_HEADER_LENGTH = 4;
|
||||
// Generic Toshiba commands/flags
|
||||
const uint8_t TOSHIBA_COMMAND_DEFAULT = 0x01;
|
||||
const uint8_t TOSHIBA_COMMAND_TIMER = 0x02;
|
||||
const uint8_t TOSHIBA_COMMAND_POWER = 0x08;
|
||||
@ -36,36 +46,122 @@ const uint8_t TOSHIBA_POWER_ECO = 0x03;
|
||||
const uint8_t TOSHIBA_MOTION_SWING = 0x04;
|
||||
const uint8_t TOSHIBA_MOTION_FIX = 0x00;
|
||||
|
||||
static const char *const TAG = "toshiba.climate";
|
||||
// RAC-PT1411HWRU temperature code flag bits
|
||||
const uint8_t RAC_PT1411HWRU_FLAG_FAH = 0x01;
|
||||
const uint8_t RAC_PT1411HWRU_FLAG_FRAC = 0x20;
|
||||
const uint8_t RAC_PT1411HWRU_FLAG_NEG = 0x10;
|
||||
// RAC-PT1411HWRU temperature short code flags mask
|
||||
const uint8_t RAC_PT1411HWRU_FLAG_MASK = 0x0F;
|
||||
// RAC-PT1411HWRU Headers, Footers and such
|
||||
const uint8_t RAC_PT1411HWRU_MESSAGE_HEADER0 = 0xB2;
|
||||
const uint8_t RAC_PT1411HWRU_MESSAGE_HEADER1 = 0xD5;
|
||||
const uint8_t RAC_PT1411HWRU_MESSAGE_LENGTH = 6;
|
||||
// RAC-PT1411HWRU "Comfort Sense" feature bits
|
||||
const uint8_t RAC_PT1411HWRU_CS_ENABLED = 0x40;
|
||||
const uint8_t RAC_PT1411HWRU_CS_DATA = 0x80;
|
||||
const uint8_t RAC_PT1411HWRU_CS_HEADER = 0xBA;
|
||||
const uint8_t RAC_PT1411HWRU_CS_FOOTER_AUTO = 0x7A;
|
||||
const uint8_t RAC_PT1411HWRU_CS_FOOTER_COOL = 0x72;
|
||||
const uint8_t RAC_PT1411HWRU_CS_FOOTER_HEAT = 0x7E;
|
||||
// RAC-PT1411HWRU Swing
|
||||
const uint8_t RAC_PT1411HWRU_SWING_HEADER = 0xB9;
|
||||
const std::vector<uint8_t> RAC_PT1411HWRU_SWING_VERTICAL{0xB9, 0x46, 0xF5, 0x0A, 0x04, 0xFB};
|
||||
const std::vector<uint8_t> RAC_PT1411HWRU_SWING_OFF{0xB9, 0x46, 0xF5, 0x0A, 0x05, 0xFA};
|
||||
// RAC-PT1411HWRU Fan speeds
|
||||
const uint8_t RAC_PT1411HWRU_FAN_OFF = 0x7B;
|
||||
constexpr RacPt1411hwruFanSpeed RAC_PT1411HWRU_FAN_AUTO{0xBF, 0x66};
|
||||
constexpr RacPt1411hwruFanSpeed RAC_PT1411HWRU_FAN_LOW{0x9F, 0x28};
|
||||
constexpr RacPt1411hwruFanSpeed RAC_PT1411HWRU_FAN_MED{0x5F, 0x3C};
|
||||
constexpr RacPt1411hwruFanSpeed RAC_PT1411HWRU_FAN_HIGH{0x3F, 0x64};
|
||||
// RAC-PT1411HWRU Fan speed for Auto and Dry climate modes
|
||||
const RacPt1411hwruFanSpeed RAC_PT1411HWRU_NO_FAN{0x1F, 0x65};
|
||||
// RAC-PT1411HWRU Modes
|
||||
const uint8_t RAC_PT1411HWRU_MODE_AUTO = 0x08;
|
||||
const uint8_t RAC_PT1411HWRU_MODE_COOL = 0x00;
|
||||
const uint8_t RAC_PT1411HWRU_MODE_DRY = 0x04;
|
||||
const uint8_t RAC_PT1411HWRU_MODE_FAN = 0x04;
|
||||
const uint8_t RAC_PT1411HWRU_MODE_HEAT = 0x0C;
|
||||
const uint8_t RAC_PT1411HWRU_MODE_OFF = 0x00;
|
||||
// RAC-PT1411HWRU Fan-only "temperature"/system off
|
||||
const uint8_t RAC_PT1411HWRU_TEMPERATURE_FAN_ONLY = 0x0E;
|
||||
// RAC-PT1411HWRU temperature codes are not sequential; they instead follow a modified Gray code.
|
||||
// Hence these look-up tables. In addition, the upper nibble is used here for additional
|
||||
// "negative" and "fractional value" flags as required for some temperatures.
|
||||
// RAC-PT1411HWRU °C Temperatures (short codes)
|
||||
const std::vector<uint8_t> RAC_PT1411HWRU_TEMPERATURE_C{0x10, 0x00, 0x01, 0x03, 0x02, 0x06, 0x07, 0x05,
|
||||
0x04, 0x0C, 0x0D, 0x09, 0x08, 0x0A, 0x0B};
|
||||
// RAC-PT1411HWRU °F Temperatures (short codes)
|
||||
const std::vector<uint8_t> RAC_PT1411HWRU_TEMPERATURE_F{0x10, 0x30, 0x00, 0x20, 0x01, 0x21, 0x03, 0x23, 0x02,
|
||||
0x22, 0x06, 0x26, 0x07, 0x05, 0x25, 0x04, 0x24, 0x0C,
|
||||
0x2C, 0x0D, 0x2D, 0x09, 0x08, 0x28, 0x0A, 0x2A, 0x0B};
|
||||
|
||||
void ToshibaClimate::setup() {
|
||||
if (this->sensor_) {
|
||||
this->sensor_->add_on_state_callback([this](float state) {
|
||||
this->current_temperature = state;
|
||||
this->transmit_rac_pt1411hwru_temp_();
|
||||
// current temperature changed, publish state
|
||||
this->publish_state();
|
||||
});
|
||||
this->current_temperature = this->sensor_->state;
|
||||
} else
|
||||
this->current_temperature = NAN;
|
||||
// restore set points
|
||||
auto restore = this->restore_state_();
|
||||
if (restore.has_value()) {
|
||||
restore->apply(this);
|
||||
} else {
|
||||
// restore from defaults
|
||||
this->mode = climate::CLIMATE_MODE_OFF;
|
||||
// initialize target temperature to some value so that it's not NAN
|
||||
this->target_temperature =
|
||||
roundf(clamp<float>(this->current_temperature, this->minimum_temperature_, this->maximum_temperature_));
|
||||
this->fan_mode = climate::CLIMATE_FAN_AUTO;
|
||||
this->swing_mode = climate::CLIMATE_SWING_OFF;
|
||||
}
|
||||
// Set supported modes & temperatures based on model
|
||||
this->minimum_temperature_ = this->temperature_min_();
|
||||
this->maximum_temperature_ = this->temperature_max_();
|
||||
this->supports_dry_ = this->toshiba_supports_dry_();
|
||||
this->supports_fan_only_ = this->toshiba_supports_fan_only_();
|
||||
this->fan_modes_ = this->toshiba_fan_modes_();
|
||||
this->swing_modes_ = this->toshiba_swing_modes_();
|
||||
// Never send nan to HA
|
||||
if (isnan(this->target_temperature))
|
||||
this->target_temperature = 24;
|
||||
}
|
||||
|
||||
void ToshibaClimate::transmit_state() {
|
||||
if (this->model_ == MODEL_RAC_PT1411HWRU_C || this->model_ == MODEL_RAC_PT1411HWRU_F) {
|
||||
transmit_rac_pt1411hwru_();
|
||||
} else {
|
||||
transmit_generic_();
|
||||
}
|
||||
}
|
||||
|
||||
void ToshibaClimate::transmit_generic_() {
|
||||
uint8_t message[16] = {0};
|
||||
uint8_t message_length = 9;
|
||||
|
||||
/* Header */
|
||||
// Header
|
||||
message[0] = 0xf2;
|
||||
message[1] = 0x0d;
|
||||
|
||||
/* Message length */
|
||||
// Message length
|
||||
message[2] = message_length - 6;
|
||||
|
||||
/* First checksum */
|
||||
// First checksum
|
||||
message[3] = message[0] ^ message[1] ^ message[2];
|
||||
|
||||
/* Command */
|
||||
// Command
|
||||
message[4] = TOSHIBA_COMMAND_DEFAULT;
|
||||
|
||||
/* Temperature */
|
||||
uint8_t temperature = static_cast<uint8_t>(this->target_temperature);
|
||||
if (temperature < 17) {
|
||||
temperature = 17;
|
||||
}
|
||||
if (temperature > 30) {
|
||||
temperature = 30;
|
||||
}
|
||||
message[5] = (temperature - 17) << 4;
|
||||
// Temperature
|
||||
uint8_t temperature = static_cast<uint8_t>(
|
||||
clamp<float>(this->target_temperature, TOSHIBA_GENERIC_TEMP_C_MIN, TOSHIBA_GENERIC_TEMP_C_MAX));
|
||||
message[5] = (temperature - static_cast<uint8_t>(TOSHIBA_GENERIC_TEMP_C_MIN)) << 4;
|
||||
|
||||
/* Mode and fan */
|
||||
// Mode and fan
|
||||
uint8_t mode;
|
||||
switch (this->mode) {
|
||||
case climate::CLIMATE_MODE_OFF:
|
||||
@ -87,28 +183,504 @@ void ToshibaClimate::transmit_state() {
|
||||
|
||||
message[6] |= mode | TOSHIBA_FAN_SPEED_AUTO;
|
||||
|
||||
/* Zero */
|
||||
// Zero
|
||||
message[7] = 0x00;
|
||||
|
||||
/* If timers bit in the command is set, two extra bytes are added here */
|
||||
// If timers bit in the command is set, two extra bytes are added here
|
||||
|
||||
/* If power bit is set in the command, one extra byte is added here */
|
||||
// If power bit is set in the command, one extra byte is added here
|
||||
|
||||
/* The last byte is the xor of all bytes from [4] */
|
||||
// The last byte is the xor of all bytes from [4]
|
||||
for (uint8_t i = 4; i < 8; i++) {
|
||||
message[8] ^= message[i];
|
||||
}
|
||||
|
||||
/* Transmit */
|
||||
// Transmit
|
||||
auto transmit = this->transmitter_->transmit();
|
||||
auto data = transmit.get_data();
|
||||
data->set_carrier_frequency(38000);
|
||||
|
||||
for (uint8_t copy = 0; copy < 2; copy++) {
|
||||
data->mark(TOSHIBA_HEADER_MARK);
|
||||
data->space(TOSHIBA_HEADER_SPACE);
|
||||
encode_(data, message, message_length, 1);
|
||||
|
||||
for (uint8_t byte = 0; byte < message_length; byte++) {
|
||||
transmit.perform();
|
||||
}
|
||||
|
||||
void ToshibaClimate::transmit_rac_pt1411hwru_() {
|
||||
uint8_t code = 0, index = 0, message[RAC_PT1411HWRU_MESSAGE_LENGTH * 2] = {0};
|
||||
float temperature =
|
||||
clamp<float>(this->target_temperature, TOSHIBA_RAC_PT1411HWRU_TEMP_C_MIN, TOSHIBA_RAC_PT1411HWRU_TEMP_C_MAX);
|
||||
float temp_adjd = temperature - TOSHIBA_RAC_PT1411HWRU_TEMP_C_MIN;
|
||||
auto transmit = this->transmitter_->transmit();
|
||||
auto data = transmit.get_data();
|
||||
|
||||
// Byte 0: Header upper (0xB2)
|
||||
message[0] = RAC_PT1411HWRU_MESSAGE_HEADER0;
|
||||
// Byte 1: Header lower (0x4D)
|
||||
message[1] = ~message[0];
|
||||
// Byte 2u: Fan speed
|
||||
// Byte 2l: 1111 (on) or 1011 (off)
|
||||
if (this->mode == climate::CLIMATE_MODE_OFF) {
|
||||
message[2] = RAC_PT1411HWRU_FAN_OFF;
|
||||
} else if ((this->mode == climate::CLIMATE_MODE_HEAT_COOL) || (this->mode == climate::CLIMATE_MODE_DRY)) {
|
||||
message[2] = RAC_PT1411HWRU_NO_FAN.code1;
|
||||
message[7] = RAC_PT1411HWRU_NO_FAN.code2;
|
||||
} else {
|
||||
switch (this->fan_mode.value()) {
|
||||
case climate::CLIMATE_FAN_LOW:
|
||||
message[2] = RAC_PT1411HWRU_FAN_LOW.code1;
|
||||
message[7] = RAC_PT1411HWRU_FAN_LOW.code2;
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_FAN_MEDIUM:
|
||||
message[2] = RAC_PT1411HWRU_FAN_MED.code1;
|
||||
message[7] = RAC_PT1411HWRU_FAN_MED.code2;
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_FAN_HIGH:
|
||||
message[2] = RAC_PT1411HWRU_FAN_HIGH.code1;
|
||||
message[7] = RAC_PT1411HWRU_FAN_HIGH.code2;
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_FAN_AUTO:
|
||||
default:
|
||||
message[2] = RAC_PT1411HWRU_FAN_AUTO.code1;
|
||||
message[7] = RAC_PT1411HWRU_FAN_AUTO.code2;
|
||||
}
|
||||
}
|
||||
// Byte 3u: ~Fan speed
|
||||
// Byte 3l: 0000 (on) or 0100 (off)
|
||||
message[3] = ~message[2];
|
||||
// Byte 4u: Temp
|
||||
if (this->model_ == MODEL_RAC_PT1411HWRU_F) {
|
||||
temperature = (temperature * 1.8) + 32;
|
||||
temp_adjd = temperature - TOSHIBA_RAC_PT1411HWRU_TEMP_F_MIN;
|
||||
}
|
||||
|
||||
index = static_cast<uint8_t>(roundf(temp_adjd));
|
||||
|
||||
if (this->model_ == MODEL_RAC_PT1411HWRU_F) {
|
||||
code = RAC_PT1411HWRU_TEMPERATURE_F[index];
|
||||
message[9] |= RAC_PT1411HWRU_FLAG_FAH;
|
||||
} else {
|
||||
code = RAC_PT1411HWRU_TEMPERATURE_C[index];
|
||||
}
|
||||
if ((this->mode == climate::CLIMATE_MODE_FAN_ONLY) || (this->mode == climate::CLIMATE_MODE_OFF)) {
|
||||
code = RAC_PT1411HWRU_TEMPERATURE_FAN_ONLY;
|
||||
}
|
||||
|
||||
if (code & RAC_PT1411HWRU_FLAG_FRAC) {
|
||||
message[8] |= RAC_PT1411HWRU_FLAG_FRAC;
|
||||
}
|
||||
if (code & RAC_PT1411HWRU_FLAG_NEG) {
|
||||
message[9] |= RAC_PT1411HWRU_FLAG_NEG;
|
||||
}
|
||||
message[4] = (code & RAC_PT1411HWRU_FLAG_MASK) << 4;
|
||||
// Byte 4l: Mode
|
||||
switch (this->mode) {
|
||||
case climate::CLIMATE_MODE_OFF:
|
||||
// zerooooo
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_MODE_HEAT:
|
||||
message[4] |= RAC_PT1411HWRU_MODE_HEAT;
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_MODE_COOL:
|
||||
message[4] |= RAC_PT1411HWRU_MODE_COOL;
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_MODE_DRY:
|
||||
message[4] |= RAC_PT1411HWRU_MODE_DRY;
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_MODE_FAN_ONLY:
|
||||
message[4] |= RAC_PT1411HWRU_MODE_FAN;
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_MODE_HEAT_COOL:
|
||||
default:
|
||||
message[4] |= RAC_PT1411HWRU_MODE_AUTO;
|
||||
}
|
||||
|
||||
// Byte 5u: ~Temp
|
||||
// Byte 5l: ~Mode
|
||||
message[5] = ~message[4];
|
||||
|
||||
if (this->mode != climate::CLIMATE_MODE_OFF) {
|
||||
// Byte 6: Header (0xD5)
|
||||
message[6] = RAC_PT1411HWRU_MESSAGE_HEADER1;
|
||||
// Byte 7: Fan speed part 2 (done above)
|
||||
// Byte 8: 0x20 for °F frac, else 0 (done above)
|
||||
// Byte 9: 0x10=NEG, 0x01=°F (done above)
|
||||
// Byte 10: 0
|
||||
// Byte 11: Checksum (bytes 6 through 10)
|
||||
for (index = 6; index <= 10; index++) {
|
||||
message[11] += message[index];
|
||||
}
|
||||
}
|
||||
ESP_LOGV(TAG, "*** Generated codes: 0x%.2X%.2X%.2X%.2X%.2X%.2X 0x%.2X%.2X%.2X%.2X%.2X%.2X", message[0], message[1],
|
||||
message[2], message[3], message[4], message[5], message[6], message[7], message[8], message[9], message[10],
|
||||
message[11]);
|
||||
|
||||
// load first block of IR code and repeat it once
|
||||
encode_(data, &message[0], RAC_PT1411HWRU_MESSAGE_LENGTH, 1);
|
||||
// load second block of IR code, if present
|
||||
if (message[6] != 0) {
|
||||
encode_(data, &message[6], RAC_PT1411HWRU_MESSAGE_LENGTH, 0);
|
||||
}
|
||||
|
||||
transmit.perform();
|
||||
|
||||
// Swing Mode
|
||||
data->reset();
|
||||
data->space(TOSHIBA_PACKET_SPACE);
|
||||
switch (this->swing_mode) {
|
||||
case climate::CLIMATE_SWING_VERTICAL:
|
||||
encode_(data, &RAC_PT1411HWRU_SWING_VERTICAL[0], RAC_PT1411HWRU_MESSAGE_LENGTH, 1);
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_SWING_OFF:
|
||||
default:
|
||||
encode_(data, &RAC_PT1411HWRU_SWING_OFF[0], RAC_PT1411HWRU_MESSAGE_LENGTH, 1);
|
||||
}
|
||||
|
||||
data->space(TOSHIBA_PACKET_SPACE);
|
||||
transmit.perform();
|
||||
|
||||
if (this->sensor_) {
|
||||
transmit_rac_pt1411hwru_temp_(true, false);
|
||||
}
|
||||
}
|
||||
|
||||
void ToshibaClimate::transmit_rac_pt1411hwru_temp_(const bool cs_state, const bool cs_send_update) {
|
||||
if ((this->mode == climate::CLIMATE_MODE_HEAT) || (this->mode == climate::CLIMATE_MODE_COOL) ||
|
||||
(this->mode == climate::CLIMATE_MODE_HEAT_COOL)) {
|
||||
uint8_t message[RAC_PT1411HWRU_MESSAGE_LENGTH] = {0};
|
||||
float temperature = clamp<float>(this->current_temperature, 0.0, TOSHIBA_RAC_PT1411HWRU_TEMP_C_MAX + 1);
|
||||
auto transmit = this->transmitter_->transmit();
|
||||
auto data = transmit.get_data();
|
||||
// "Comfort Sense" feature notes
|
||||
// IR Code: 0xBA45 xxXX yyYY
|
||||
// xx: Temperature in °C
|
||||
// Bit 6: feature state (on/off)
|
||||
// Bit 7: message contains temperature data for feature (bit 6 must also be set)
|
||||
// XX: Bitwise complement of xx
|
||||
// yy: Mode: Auto=0x7A, Cool=0x72, Heat=0x7E
|
||||
// YY: Bitwise complement of yy
|
||||
//
|
||||
// Byte 0: Header upper (0xBA)
|
||||
message[0] = RAC_PT1411HWRU_CS_HEADER;
|
||||
// Byte 1: Header lower (0x45)
|
||||
message[1] = ~message[0];
|
||||
// Byte 2: Temperature in °C
|
||||
message[2] = static_cast<uint8_t>(roundf(temperature));
|
||||
if (cs_send_update) {
|
||||
message[2] |= RAC_PT1411HWRU_CS_ENABLED | RAC_PT1411HWRU_CS_DATA;
|
||||
} else if (cs_state) {
|
||||
message[2] |= RAC_PT1411HWRU_CS_ENABLED;
|
||||
}
|
||||
// Byte 3: Bitwise complement of byte 2
|
||||
message[3] = ~message[2];
|
||||
// Byte 4: Footer upper
|
||||
switch (this->mode) {
|
||||
case climate::CLIMATE_MODE_HEAT:
|
||||
message[4] = RAC_PT1411HWRU_CS_FOOTER_HEAT;
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_MODE_COOL:
|
||||
message[4] = RAC_PT1411HWRU_CS_FOOTER_COOL;
|
||||
break;
|
||||
|
||||
case climate::CLIMATE_MODE_HEAT_COOL:
|
||||
message[4] = RAC_PT1411HWRU_CS_FOOTER_AUTO;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
// Byte 5: Footer lower/bitwise complement of byte 4
|
||||
message[5] = ~message[4];
|
||||
|
||||
ESP_LOGV(TAG, "*** Generated code: 0x%.2X%.2X%.2X%.2X%.2X%.2X", message[0], message[1], message[2], message[3],
|
||||
message[4], message[5]);
|
||||
// load IR code and repeat it once
|
||||
encode_(data, message, RAC_PT1411HWRU_MESSAGE_LENGTH, 1);
|
||||
|
||||
transmit.perform();
|
||||
}
|
||||
}
|
||||
|
||||
uint8_t ToshibaClimate::is_valid_rac_pt1411hwru_header_(const uint8_t *message) {
|
||||
const std::vector<uint8_t> header{RAC_PT1411HWRU_MESSAGE_HEADER0, RAC_PT1411HWRU_CS_HEADER,
|
||||
RAC_PT1411HWRU_SWING_HEADER};
|
||||
|
||||
for (auto i : header) {
|
||||
if ((message[0] == i) && (message[1] == static_cast<uint8_t>(~i)))
|
||||
return i;
|
||||
}
|
||||
if (message[0] == RAC_PT1411HWRU_MESSAGE_HEADER1)
|
||||
return RAC_PT1411HWRU_MESSAGE_HEADER1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
bool ToshibaClimate::compare_rac_pt1411hwru_packets_(const uint8_t *message1, const uint8_t *message2) {
|
||||
for (uint8_t i = 0; i < RAC_PT1411HWRU_MESSAGE_LENGTH; i++) {
|
||||
if (message1[i] != message2[i])
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
bool ToshibaClimate::is_valid_rac_pt1411hwru_message_(const uint8_t *message) {
|
||||
uint8_t checksum = 0;
|
||||
|
||||
switch (is_valid_rac_pt1411hwru_header_(message)) {
|
||||
case RAC_PT1411HWRU_MESSAGE_HEADER0:
|
||||
case RAC_PT1411HWRU_CS_HEADER:
|
||||
case RAC_PT1411HWRU_SWING_HEADER:
|
||||
if (is_valid_rac_pt1411hwru_header_(message) && (message[2] == static_cast<uint8_t>(~message[3])) &&
|
||||
(message[4] == static_cast<uint8_t>(~message[5]))) {
|
||||
return true;
|
||||
}
|
||||
break;
|
||||
|
||||
case RAC_PT1411HWRU_MESSAGE_HEADER1:
|
||||
for (uint8_t i = 0; i < RAC_PT1411HWRU_MESSAGE_LENGTH - 1; i++) {
|
||||
checksum += message[i];
|
||||
}
|
||||
if (checksum == message[RAC_PT1411HWRU_MESSAGE_LENGTH - 1]) {
|
||||
return true;
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
bool ToshibaClimate::on_receive(remote_base::RemoteReceiveData data) {
|
||||
uint8_t message[18] = {0};
|
||||
uint8_t message_length = TOSHIBA_HEADER_LENGTH, temperature_code = 0;
|
||||
|
||||
// Validate header
|
||||
if (!data.expect_item(TOSHIBA_HEADER_MARK, TOSHIBA_HEADER_SPACE)) {
|
||||
return false;
|
||||
}
|
||||
// Read incoming bits into buffer
|
||||
if (!decode_(&data, message, message_length)) {
|
||||
return false;
|
||||
}
|
||||
// Determine incoming message protocol version and/or length
|
||||
if (is_valid_rac_pt1411hwru_header_(message)) {
|
||||
// We already received four bytes
|
||||
message_length = RAC_PT1411HWRU_MESSAGE_LENGTH - 4;
|
||||
} else if ((message[0] ^ message[1] ^ message[2]) != message[3]) {
|
||||
// Return false if first checksum was not valid
|
||||
return false;
|
||||
} else {
|
||||
// First checksum was valid so continue receiving the remaining bits
|
||||
message_length = message[2] + 2;
|
||||
}
|
||||
// Decode the remaining bytes
|
||||
if (!decode_(&data, &message[4], message_length)) {
|
||||
return false;
|
||||
}
|
||||
// If this is a RAC-PT1411HWRU message, we expect the first packet a second time and also possibly a third packet
|
||||
if (is_valid_rac_pt1411hwru_header_(message)) {
|
||||
// There is always a space between packets
|
||||
if (!data.expect_item(TOSHIBA_BIT_MARK, TOSHIBA_GAP_SPACE)) {
|
||||
return false;
|
||||
}
|
||||
// Validate header 2
|
||||
if (!data.expect_item(TOSHIBA_HEADER_MARK, TOSHIBA_HEADER_SPACE)) {
|
||||
return false;
|
||||
}
|
||||
if (!decode_(&data, &message[6], RAC_PT1411HWRU_MESSAGE_LENGTH)) {
|
||||
return false;
|
||||
}
|
||||
// If this is a RAC-PT1411HWRU message, there may also be a third packet.
|
||||
// We do not fail the receive if we don't get this; it isn't always present
|
||||
if (data.expect_item(TOSHIBA_BIT_MARK, TOSHIBA_GAP_SPACE)) {
|
||||
// Validate header 3
|
||||
data.expect_item(TOSHIBA_HEADER_MARK, TOSHIBA_HEADER_SPACE);
|
||||
if (decode_(&data, &message[12], RAC_PT1411HWRU_MESSAGE_LENGTH)) {
|
||||
if (!is_valid_rac_pt1411hwru_message_(&message[12])) {
|
||||
// If a third packet was received but the checksum is not valid, fail
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!compare_rac_pt1411hwru_packets_(&message[0], &message[6])) {
|
||||
// If the first two packets don't match each other, fail
|
||||
return false;
|
||||
}
|
||||
if (!is_valid_rac_pt1411hwru_message_(&message[0])) {
|
||||
// If the first packet isn't valid, fail
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
// Header has been verified, now determine protocol version and set the climate component properties
|
||||
switch (is_valid_rac_pt1411hwru_header_(message)) {
|
||||
// Power, temperature, mode, fan speed
|
||||
case RAC_PT1411HWRU_MESSAGE_HEADER0:
|
||||
// Get the mode
|
||||
switch (message[4] & 0x0F) {
|
||||
case RAC_PT1411HWRU_MODE_AUTO:
|
||||
this->mode = climate::CLIMATE_MODE_HEAT_COOL;
|
||||
break;
|
||||
|
||||
// case RAC_PT1411HWRU_MODE_OFF:
|
||||
case RAC_PT1411HWRU_MODE_COOL:
|
||||
if (((message[4] >> 4) == RAC_PT1411HWRU_TEMPERATURE_FAN_ONLY) && (message[2] == RAC_PT1411HWRU_FAN_OFF)) {
|
||||
this->mode = climate::CLIMATE_MODE_OFF;
|
||||
} else {
|
||||
this->mode = climate::CLIMATE_MODE_COOL;
|
||||
}
|
||||
break;
|
||||
|
||||
// case RAC_PT1411HWRU_MODE_DRY:
|
||||
case RAC_PT1411HWRU_MODE_FAN:
|
||||
if ((message[4] >> 4) == RAC_PT1411HWRU_TEMPERATURE_FAN_ONLY)
|
||||
this->mode = climate::CLIMATE_MODE_FAN_ONLY;
|
||||
else
|
||||
this->mode = climate::CLIMATE_MODE_DRY;
|
||||
break;
|
||||
|
||||
case RAC_PT1411HWRU_MODE_HEAT:
|
||||
this->mode = climate::CLIMATE_MODE_HEAT;
|
||||
break;
|
||||
|
||||
default:
|
||||
this->mode = climate::CLIMATE_MODE_OFF;
|
||||
break;
|
||||
}
|
||||
// Get the fan speed/mode
|
||||
switch (message[2]) {
|
||||
case RAC_PT1411HWRU_FAN_LOW.code1:
|
||||
this->fan_mode = climate::CLIMATE_FAN_LOW;
|
||||
break;
|
||||
|
||||
case RAC_PT1411HWRU_FAN_MED.code1:
|
||||
this->fan_mode = climate::CLIMATE_FAN_MEDIUM;
|
||||
break;
|
||||
|
||||
case RAC_PT1411HWRU_FAN_HIGH.code1:
|
||||
this->fan_mode = climate::CLIMATE_FAN_HIGH;
|
||||
break;
|
||||
|
||||
case RAC_PT1411HWRU_FAN_AUTO.code1:
|
||||
default:
|
||||
this->fan_mode = climate::CLIMATE_FAN_AUTO;
|
||||
break;
|
||||
}
|
||||
// Get the target temperature
|
||||
if (is_valid_rac_pt1411hwru_message_(&message[12])) {
|
||||
temperature_code =
|
||||
(message[4] >> 4) | (message[14] & RAC_PT1411HWRU_FLAG_FRAC) | (message[15] & RAC_PT1411HWRU_FLAG_NEG);
|
||||
if (message[15] & RAC_PT1411HWRU_FLAG_FAH) {
|
||||
for (uint8_t i = 0; i < RAC_PT1411HWRU_TEMPERATURE_F.size(); i++) {
|
||||
if (RAC_PT1411HWRU_TEMPERATURE_F[i] == temperature_code) {
|
||||
this->target_temperature = static_cast<float>((i + TOSHIBA_RAC_PT1411HWRU_TEMP_F_MIN - 32) * 5) / 9;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
for (uint8_t i = 0; i < RAC_PT1411HWRU_TEMPERATURE_C.size(); i++) {
|
||||
if (RAC_PT1411HWRU_TEMPERATURE_C[i] == temperature_code) {
|
||||
this->target_temperature = i + TOSHIBA_RAC_PT1411HWRU_TEMP_C_MIN;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
break;
|
||||
// "Comfort Sense" temperature packet
|
||||
case RAC_PT1411HWRU_CS_HEADER:
|
||||
// "Comfort Sense" feature notes
|
||||
// IR Code: 0xBA45 xxXX yyYY
|
||||
// xx: Temperature in °C
|
||||
// Bit 6: feature state (on/off)
|
||||
// Bit 7: message contains temperature data for feature (bit 6 must also be set)
|
||||
// XX: Bitwise complement of xx
|
||||
// yy: Mode: Auto: 7A
|
||||
// Cool: 72
|
||||
// Heat: 7E
|
||||
// YY: Bitwise complement of yy
|
||||
if ((message[2] & RAC_PT1411HWRU_CS_ENABLED) && (message[2] & RAC_PT1411HWRU_CS_DATA)) {
|
||||
// Setting current_temperature this way allows the unit's remote to provide the temperature to HA
|
||||
this->current_temperature = message[2] & ~(RAC_PT1411HWRU_CS_ENABLED | RAC_PT1411HWRU_CS_DATA);
|
||||
}
|
||||
break;
|
||||
// Swing mode
|
||||
case RAC_PT1411HWRU_SWING_HEADER:
|
||||
if (message[4] == RAC_PT1411HWRU_SWING_VERTICAL[4]) {
|
||||
this->swing_mode = climate::CLIMATE_SWING_VERTICAL;
|
||||
} else {
|
||||
this->swing_mode = climate::CLIMATE_SWING_OFF;
|
||||
}
|
||||
break;
|
||||
// Generic (old) Toshiba packet
|
||||
default:
|
||||
uint8_t checksum = 0;
|
||||
// Add back the length of the header (we pruned it above)
|
||||
message_length += TOSHIBA_HEADER_LENGTH;
|
||||
// Validate the second checksum before trusting any more of the message
|
||||
for (uint8_t i = TOSHIBA_HEADER_LENGTH; i < message_length - 1; i++) {
|
||||
checksum ^= message[i];
|
||||
}
|
||||
// Did our computed checksum and the provided checksum match?
|
||||
if (checksum != message[message_length - 1]) {
|
||||
return false;
|
||||
}
|
||||
// Check if this is a short swing/fix message
|
||||
if (message[4] & TOSHIBA_COMMAND_MOTION) {
|
||||
// Not supported yet
|
||||
return false;
|
||||
}
|
||||
|
||||
// Get the mode
|
||||
switch (message[6] & 0x0F) {
|
||||
case TOSHIBA_MODE_OFF:
|
||||
this->mode = climate::CLIMATE_MODE_OFF;
|
||||
break;
|
||||
|
||||
case TOSHIBA_MODE_COOL:
|
||||
this->mode = climate::CLIMATE_MODE_COOL;
|
||||
break;
|
||||
|
||||
case TOSHIBA_MODE_DRY:
|
||||
this->mode = climate::CLIMATE_MODE_DRY;
|
||||
break;
|
||||
|
||||
case TOSHIBA_MODE_FAN_ONLY:
|
||||
this->mode = climate::CLIMATE_MODE_FAN_ONLY;
|
||||
break;
|
||||
|
||||
case TOSHIBA_MODE_HEAT:
|
||||
this->mode = climate::CLIMATE_MODE_HEAT;
|
||||
break;
|
||||
|
||||
case TOSHIBA_MODE_AUTO:
|
||||
default:
|
||||
this->mode = climate::CLIMATE_MODE_HEAT_COOL;
|
||||
}
|
||||
|
||||
// Get the target temperature
|
||||
this->target_temperature = (message[5] >> 4) + TOSHIBA_GENERIC_TEMP_C_MIN;
|
||||
}
|
||||
|
||||
this->publish_state();
|
||||
return true;
|
||||
}
|
||||
|
||||
void ToshibaClimate::encode_(remote_base::RemoteTransmitData *data, const uint8_t *message, const uint8_t nbytes,
|
||||
const uint8_t repeat) {
|
||||
data->set_carrier_frequency(TOSHIBA_CARRIER_FREQUENCY);
|
||||
|
||||
for (uint8_t copy = 0; copy <= repeat; copy++) {
|
||||
data->item(TOSHIBA_HEADER_MARK, TOSHIBA_HEADER_SPACE);
|
||||
|
||||
for (uint8_t byte = 0; byte < nbytes; byte++) {
|
||||
for (uint8_t bit = 0; bit < 8; bit++) {
|
||||
data->mark(TOSHIBA_BIT_MARK);
|
||||
if (message[byte] & (1 << (7 - bit))) {
|
||||
@ -118,87 +690,24 @@ void ToshibaClimate::transmit_state() {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
data->mark(TOSHIBA_BIT_MARK);
|
||||
data->space(TOSHIBA_GAP_SPACE);
|
||||
data->item(TOSHIBA_BIT_MARK, TOSHIBA_GAP_SPACE);
|
||||
}
|
||||
|
||||
transmit.perform();
|
||||
}
|
||||
|
||||
bool ToshibaClimate::on_receive(remote_base::RemoteReceiveData data) {
|
||||
uint8_t message[16] = {0};
|
||||
uint8_t message_length = 4;
|
||||
|
||||
/* Validate header */
|
||||
if (!data.expect_item(TOSHIBA_HEADER_MARK, TOSHIBA_HEADER_SPACE)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
/* Decode bytes */
|
||||
for (uint8_t byte = 0; byte < message_length; byte++) {
|
||||
bool ToshibaClimate::decode_(remote_base::RemoteReceiveData *data, uint8_t *message, const uint8_t nbytes) {
|
||||
for (uint8_t byte = 0; byte < nbytes; byte++) {
|
||||
for (uint8_t bit = 0; bit < 8; bit++) {
|
||||
if (data.expect_item(TOSHIBA_BIT_MARK, TOSHIBA_ONE_SPACE)) {
|
||||
if (data->expect_item(TOSHIBA_BIT_MARK, TOSHIBA_ONE_SPACE)) {
|
||||
message[byte] |= 1 << (7 - bit);
|
||||
} else if (data.expect_item(TOSHIBA_BIT_MARK, TOSHIBA_ZERO_SPACE)) {
|
||||
/* Bit is already clear */
|
||||
} else if (data->expect_item(TOSHIBA_BIT_MARK, TOSHIBA_ZERO_SPACE)) {
|
||||
message[byte] &= static_cast<uint8_t>(~(1 << (7 - bit)));
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/* Update length */
|
||||
if (byte == 3) {
|
||||
/* Validate the first checksum before trusting the length field */
|
||||
if ((message[0] ^ message[1] ^ message[2]) != message[3]) {
|
||||
return false;
|
||||
}
|
||||
message_length = message[2] + 6;
|
||||
}
|
||||
}
|
||||
|
||||
/* Validate the second checksum before trusting any more of the message */
|
||||
uint8_t checksum = 0;
|
||||
for (uint8_t i = 4; i < message_length - 1; i++) {
|
||||
checksum ^= message[i];
|
||||
}
|
||||
|
||||
if (checksum != message[message_length - 1]) {
|
||||
return false;
|
||||
}
|
||||
|
||||
/* Check if this is a short swing/fix message */
|
||||
if (message[4] & TOSHIBA_COMMAND_MOTION) {
|
||||
/* Not supported yet */
|
||||
return false;
|
||||
}
|
||||
|
||||
/* Get the mode. */
|
||||
switch (message[6] & 0x0f) {
|
||||
case TOSHIBA_MODE_OFF:
|
||||
this->mode = climate::CLIMATE_MODE_OFF;
|
||||
break;
|
||||
|
||||
case TOSHIBA_MODE_HEAT:
|
||||
this->mode = climate::CLIMATE_MODE_HEAT;
|
||||
break;
|
||||
|
||||
case TOSHIBA_MODE_COOL:
|
||||
this->mode = climate::CLIMATE_MODE_COOL;
|
||||
break;
|
||||
|
||||
case TOSHIBA_MODE_AUTO:
|
||||
default:
|
||||
/* Note: Dry and Fan-only modes are reported as Auto, as they are not supported yet */
|
||||
this->mode = climate::CLIMATE_MODE_HEAT_COOL;
|
||||
}
|
||||
|
||||
/* Get the target temperature */
|
||||
this->target_temperature = (message[5] >> 4) + 17;
|
||||
|
||||
this->publish_state();
|
||||
return true;
|
||||
}
|
||||
|
||||
} /* namespace toshiba */
|
||||
} /* namespace esphome */
|
||||
} // namespace toshiba
|
||||
} // namespace esphome
|
||||
|
@ -5,17 +5,69 @@
|
||||
namespace esphome {
|
||||
namespace toshiba {
|
||||
|
||||
const float TOSHIBA_TEMP_MIN = 17.0;
|
||||
const float TOSHIBA_TEMP_MAX = 30.0;
|
||||
// Simple enum to represent models.
|
||||
enum Model {
|
||||
MODEL_GENERIC = 0, // Temperature range is from 17 to 30
|
||||
MODEL_RAC_PT1411HWRU_C = 1, // Temperature range is from 16 to 30
|
||||
MODEL_RAC_PT1411HWRU_F = 2, // Temperature range is from 16 to 30
|
||||
};
|
||||
|
||||
// Supported temperature ranges
|
||||
const float TOSHIBA_GENERIC_TEMP_C_MIN = 17.0;
|
||||
const float TOSHIBA_GENERIC_TEMP_C_MAX = 30.0;
|
||||
const float TOSHIBA_RAC_PT1411HWRU_TEMP_C_MIN = 16.0;
|
||||
const float TOSHIBA_RAC_PT1411HWRU_TEMP_C_MAX = 30.0;
|
||||
const float TOSHIBA_RAC_PT1411HWRU_TEMP_F_MIN = 60.0;
|
||||
const float TOSHIBA_RAC_PT1411HWRU_TEMP_F_MAX = 86.0;
|
||||
|
||||
class ToshibaClimate : public climate_ir::ClimateIR {
|
||||
public:
|
||||
ToshibaClimate() : climate_ir::ClimateIR(TOSHIBA_TEMP_MIN, TOSHIBA_TEMP_MAX, 1.0f) {}
|
||||
ToshibaClimate() : climate_ir::ClimateIR(TOSHIBA_GENERIC_TEMP_C_MIN, TOSHIBA_GENERIC_TEMP_C_MAX, 1.0f) {}
|
||||
|
||||
void setup() override;
|
||||
void set_model(Model model) { this->model_ = model; }
|
||||
|
||||
protected:
|
||||
void transmit_state() override;
|
||||
void transmit_generic_();
|
||||
void transmit_rac_pt1411hwru_();
|
||||
void transmit_rac_pt1411hwru_temp_(bool cs_state = true, bool cs_send_update = true);
|
||||
// Returns the header if valid, else returns zero
|
||||
uint8_t is_valid_rac_pt1411hwru_header_(const uint8_t *message);
|
||||
// Returns true if message is a valid RAC-PT1411HWRU IR message, regardless if first or second packet
|
||||
bool is_valid_rac_pt1411hwru_message_(const uint8_t *message);
|
||||
// Returns true if message1 and message 2 are the same
|
||||
bool compare_rac_pt1411hwru_packets_(const uint8_t *message1, const uint8_t *message2);
|
||||
bool on_receive(remote_base::RemoteReceiveData data) override;
|
||||
|
||||
float temperature_min_() {
|
||||
return (this->model_ == MODEL_GENERIC) ? TOSHIBA_GENERIC_TEMP_C_MIN : TOSHIBA_RAC_PT1411HWRU_TEMP_C_MIN;
|
||||
}
|
||||
float temperature_max_() {
|
||||
return (this->model_ == MODEL_GENERIC) ? TOSHIBA_GENERIC_TEMP_C_MAX : TOSHIBA_RAC_PT1411HWRU_TEMP_C_MAX;
|
||||
}
|
||||
bool toshiba_supports_dry_() {
|
||||
return ((this->model_ == MODEL_RAC_PT1411HWRU_C) || (this->model_ == MODEL_RAC_PT1411HWRU_F));
|
||||
}
|
||||
bool toshiba_supports_fan_only_() {
|
||||
return ((this->model_ == MODEL_RAC_PT1411HWRU_C) || (this->model_ == MODEL_RAC_PT1411HWRU_F));
|
||||
}
|
||||
std::set<climate::ClimateFanMode> toshiba_fan_modes_() {
|
||||
return (this->model_ == MODEL_GENERIC)
|
||||
? std::set<climate::ClimateFanMode>{}
|
||||
: std::set<climate::ClimateFanMode>{climate::CLIMATE_FAN_AUTO, climate::CLIMATE_FAN_LOW,
|
||||
climate::CLIMATE_FAN_MEDIUM, climate::CLIMATE_FAN_HIGH};
|
||||
}
|
||||
std::set<climate::ClimateSwingMode> toshiba_swing_modes_() {
|
||||
return (this->model_ == MODEL_GENERIC)
|
||||
? std::set<climate::ClimateSwingMode>{}
|
||||
: std::set<climate::ClimateSwingMode>{climate::CLIMATE_SWING_OFF, climate::CLIMATE_SWING_VERTICAL};
|
||||
}
|
||||
void encode_(remote_base::RemoteTransmitData *data, const uint8_t *message, uint8_t nbytes, uint8_t repeat);
|
||||
bool decode_(remote_base::RemoteReceiveData *data, uint8_t *message, uint8_t nbytes);
|
||||
|
||||
Model model_;
|
||||
};
|
||||
|
||||
} /* namespace toshiba */
|
||||
} /* namespace esphome */
|
||||
} // namespace toshiba
|
||||
} // namespace esphome
|
||||
|
Loading…
Reference in New Issue
Block a user