39namespace remote_base {
41static const char *
const TAG =
"remote.pronto";
47 if (data1.size() != data2.size() || data1.empty()) {
53 for (
size_t i = 0; i < data1.size() - 1; ++i) {
54 int diff = data2[i] - data1[i];
56 if (rhs.
delta == -1 && diff > 9)
62 return total_diff <= (rhs.
delta == -1 ? data1.size() * 3 : rhs.
delta);
66static constexpr uint16_t MICROSECONDS_T_MAX = 0xFFFFU;
67static constexpr uint16_t LEARNED_TOKEN = 0x0000U;
68static constexpr uint16_t LEARNED_NON_MODULATED_TOKEN = 0x0100U;
69static constexpr uint16_t BITS_IN_HEXADECIMAL = 4U;
70static constexpr uint16_t DIGITS_IN_PRONTO_NUMBER = 4U;
71static constexpr uint16_t NUMBERS_IN_PREAMBLE = 4U;
72static constexpr uint16_t HEX_MASK = 0xFU;
73static constexpr uint32_t REFERENCE_FREQUENCY = 4145146UL;
74static constexpr uint16_t FALLBACK_FREQUENCY = 64767U;
75static constexpr uint32_t MICROSECONDS_IN_SECONDS = 1000000UL;
76static constexpr uint16_t PRONTO_DEFAULT_GAP = 45000;
77static constexpr uint16_t MARK_EXCESS_MICROS = 20;
78static constexpr size_t PRONTO_LOG_CHUNK_SIZE = 230;
80static uint16_t to_frequency_k_hz(uint16_t code) {
84 return ((REFERENCE_FREQUENCY / code) + 500) / 1000;
90void ProntoProtocol::send_pronto_(RemoteTransmitData *dst,
const std::vector<uint16_t> &data) {
94 uint16_t timebase = (MICROSECONDS_IN_SECONDS * data[1] + REFERENCE_FREQUENCY / 2) / REFERENCE_FREQUENCY;
98 khz = to_frequency_k_hz(data[1]);
100 case LEARNED_NON_MODULATED_TOKEN:
106 ESP_LOGD(TAG,
"Send Pronto: frequency=%dkHz", khz);
107 dst->set_carrier_frequency(khz * 1000);
109 uint16_t intros = 2 * data[2];
110 uint16_t repeats = 2 * data[3];
111 ESP_LOGD(TAG,
"Send Pronto: intros=%d, repeats=%d", intros, repeats);
112 if (NUMBERS_IN_PREAMBLE + intros + repeats != data.size()) {
113 ESP_LOGE(TAG,
"Inconsistent data, not sending");
121 dst->reserve(intros + repeats);
123 for (uint16_t i = 0; i < intros + repeats; i += 2) {
124 uint32_t duration0 = ((
uint32_t) data[i + 0 + NUMBERS_IN_PREAMBLE]) * timebase;
125 duration0 = duration0 < MICROSECONDS_T_MAX ? duration0 : MICROSECONDS_T_MAX;
127 uint32_t duration1 = ((
uint32_t) data[i + 1 + NUMBERS_IN_PREAMBLE]) * timebase;
128 duration1 = duration1 < MICROSECONDS_T_MAX ? duration1 : MICROSECONDS_T_MAX;
130 dst->item(duration0, duration1);
135 size_t len = str.length() / (DIGITS_IN_PRONTO_NUMBER + 1) + 1;
136 std::vector<uint16_t> data;
137 const char *p = str.c_str();
140 for (
size_t i = 0; i <
len; i++) {
141 uint16_t
x = strtol(p, endptr, 16);
142 if (
x == 0 && i >= NUMBERS_IN_PREAMBLE) {
153void ProntoProtocol::send_pronto_(RemoteTransmitData *dst,
const std::string &str) {
155 send_pronto_(dst, data);
160uint16_t ProntoProtocol::effective_frequency_(uint16_t
frequency) {
164uint16_t ProntoProtocol::to_timebase_(uint16_t
frequency) {
165 return MICROSECONDS_IN_SECONDS / effective_frequency_(
frequency);
168uint16_t ProntoProtocol::to_frequency_code_(uint16_t
frequency) {
169 return REFERENCE_FREQUENCY / effective_frequency_(
frequency);
172std::string ProntoProtocol::dump_digit_(uint8_t
x) {
173 return std::string(1, (
char) (
x <= 9 ? (
'0' +
x) : (
'A' + (
x - 10))));
176std::string ProntoProtocol::dump_number_(uint16_t number,
bool end ) {
179 for (uint8_t i = 0; i < DIGITS_IN_PRONTO_NUMBER; ++i) {
180 uint8_t shifts = BITS_IN_HEXADECIMAL * (DIGITS_IN_PRONTO_NUMBER - 1 - i);
181 num += dump_digit_((number >> shifts) & HEX_MASK);
190std::string ProntoProtocol::dump_duration_(
uint32_t duration, uint16_t timebase,
bool end ) {
191 return dump_number_((
duration + timebase / 2) / timebase,
end);
194std::string ProntoProtocol::compensate_and_dump_sequence_(
const RawTimings &data, uint16_t timebase) {
197 for (int32_t t_length : data) {
201 t_duration = t_length - MARK_EXCESS_MICROS;
203 t_duration = -t_length + MARK_EXCESS_MICROS;
205 out += dump_duration_(t_duration, timebase);
216 std::string prontodata;
218 prontodata += dump_number_(
frequency > 0 ? LEARNED_TOKEN : LEARNED_NON_MODULATED_TOKEN);
219 prontodata += dump_number_(to_frequency_code_(
frequency));
220 prontodata += dump_number_((data.size() + 1) / 2);
221 prontodata += dump_number_(0);
222 uint16_t timebase = to_timebase_(
frequency);
223 prontodata += compensate_and_dump_sequence_(data, timebase);
225 out.
data = prontodata;
232 ESP_LOGI(TAG,
"Received Pronto: data=");
234 const char *ptr = data.
data.c_str();
235 size_t remaining = data.
data.size();
239 size_t chunk_size = remaining < PRONTO_LOG_CHUNK_SIZE ? remaining : PRONTO_LOG_CHUNK_SIZE;
240 ESP_LOGI(TAG,
"%.*s", (
int) chunk_size, ptr);
242 remaining -= chunk_size;
243 }
while (remaining > 0);
void encode(RemoteTransmitData *dst, const ProntoData &data) override
optional< ProntoData > decode(RemoteReceiveData src) override
void dump(const ProntoData &data) override
const RawTimings & get_raw_data() const
std::vector< uint16_t > encode_pronto(const std::string &str)
std::vector< int32_t > RawTimings
Providing packet encoding functions for exchanging data with a remote host.
bool operator==(const ProntoData &rhs) const