/************************************ * Rage * Against * The * Garage * Door * Opener * * Copyright (C) 2022 Paul Wieland * * GNU GENERAL PUBLIC LICENSE ************************************/ #pragma once #include "SoftwareSerial.h" // Using espsoftwareserial https://github.com/plerup/espsoftwareserial #include "callbacks.h" #include "enum.h" #include "esphome/core/component.h" #include "esphome/core/gpio.h" #include "esphome/core/log.h" #include "esphome/core/preferences.h" #include "observable.h" extern "C" { #include "secplus.h" } #include "ratgdo_state.h" namespace esphome { namespace ratgdo { class RATGDOComponent; typedef Parented RATGDOClient; static const uint8_t PACKET_LENGTH = 19; typedef uint8_t WirePacket[PACKET_LENGTH]; const float DOOR_POSITION_UNKNOWN = -1.0; const float DOOR_DELTA_UNKNOWN = -2.0; const uint16_t PAIRED_DEVICES_UNKNOWN = 0xFF; namespace data { const uint32_t LIGHT_OFF = 0; const uint32_t LIGHT_ON = 1; const uint32_t LIGHT_TOGGLE = 2; const uint32_t LIGHT_TOGGLE2 = 3; const uint32_t LOCK_OFF = 0; const uint32_t LOCK_ON = 1; const uint32_t LOCK_TOGGLE = 2; const uint32_t DOOR_CLOSE = 0; const uint32_t DOOR_OPEN = 1; const uint32_t DOOR_TOGGLE = 2; const uint32_t DOOR_STOP = 3; } ENUM(Command, uint16_t, (UNKNOWN, 0x000), (GET_STATUS, 0x080), (STATUS, 0x081), (OBST_1, 0x084), // sent when an obstruction happens? (OBST_2, 0x085), // sent when an obstruction happens? (PAIR_3, 0x0a0), (PAIR_3_RESP, 0x0a1), (LEARN, 0x181), (LOCK, 0x18c), (DOOR_ACTION, 0x280), (LIGHT, 0x281), (MOTOR_ON, 0x284), (MOTION, 0x285), (GET_PAIRED_DEVICES, 0x307), // nibble 0 for total, 1 wireless, 2 keypads, 3 wall, 4 accessories. (PAIRED_DEVICES, 0x308), // byte2 holds number of paired devices (CLEAR_PAIRED_DEVICES, 0x30D), // nibble 0 to clear remotes, 1 keypads, 2 wall, 3 accessories (offset from above) (LEARN_1, 0x391), (PING, 0x392), (PING_RESP, 0x393), (PAIR_2, 0x400), (PAIR_2_RESP, 0x401), (SET_TTC, 0x402), // ttc_in_seconds = (byte1<<8)+byte2 (CANCEL_TTC, 0x408), // ? (TTC, 0x40a), // Time to close (GET_OPENINGS, 0x48b), (OPENINGS, 0x48c), // openings = (byte1<<8)+byte2 ) inline bool operator==(const uint16_t cmd_i, const Command& cmd_e) { return cmd_i == static_cast(cmd_e); } inline bool operator==(const Command& cmd_e, const uint16_t cmd_i) { return cmd_i == static_cast(cmd_e); } struct RATGDOStore { int obstruction_low_count = 0; // count obstruction low pulses static void IRAM_ATTR HOT isr_obstruction(RATGDOStore* arg) { arg->obstruction_low_count++; } }; class RATGDOComponent : public Component { public: void setup() override; void loop() override; void dump_config() override; observable rolling_code_counter { 0 }; float start_opening { -1 }; observable opening_duration { 0 }; float start_closing { -1 }; observable closing_duration { 0 }; observable openings { 0 }; // number of times the door has been opened observable paired_total { PAIRED_DEVICES_UNKNOWN }; observable paired_remotes { PAIRED_DEVICES_UNKNOWN }; observable paired_keypads { PAIRED_DEVICES_UNKNOWN }; observable paired_wall_controls { PAIRED_DEVICES_UNKNOWN }; observable paired_accessories { PAIRED_DEVICES_UNKNOWN }; observable door_state { DoorState::UNKNOWN }; observable door_position { DOOR_POSITION_UNKNOWN }; unsigned long door_start_moving { 0 }; float door_start_position { DOOR_POSITION_UNKNOWN }; float door_move_delta { DOOR_DELTA_UNKNOWN }; observable light_state { LightState::UNKNOWN }; observable lock_state { LockState::UNKNOWN }; observable obstruction_state { ObstructionState::UNKNOWN }; observable motor_state { MotorState::UNKNOWN }; observable button_state { ButtonState::UNKNOWN }; observable motion_state { MotionState::UNKNOWN }; observable learn_state { LearnState::UNKNOWN }; OnceCallbacks door_state_received; OnceCallbacks command_sent; observable sync_failed { false }; void set_output_gdo_pin(InternalGPIOPin* pin) { this->output_gdo_pin_ = pin; } void set_input_gdo_pin(InternalGPIOPin* pin) { this->input_gdo_pin_ = pin; } void set_input_obst_pin(InternalGPIOPin* pin) { this->input_obst_pin_ = pin; } void set_client_id(uint64_t client_id) { this->client_id_ = client_id & 0xFFFFFFFF; } void gdo_state_loop(); uint16_t decode_packet(const WirePacket& packet); void obstruction_loop(); void send_command(Command command, uint32_t data = 0, bool increment = true); void send_command(Command command, uint32_t data, bool increment, std::function&& on_sent); bool transmit_packet(); void encode_packet(Command command, uint32_t data, bool increment, WirePacket& packet); void print_packet(const char* prefix, const WirePacket& packet) const; void increment_rolling_code_counter(int delta = 1); void set_rolling_code_counter(uint32_t code); // door void door_command(uint32_t data); void ensure_door_command(uint32_t data, uint32_t delay = 1500); void toggle_door(); void open_door(); void close_door(); void stop_door(); void door_move_to_position(float position); void set_door_position(float door_position) { this->door_position = door_position; } void set_opening_duration(float duration); void set_closing_duration(float duration); void schedule_door_position_sync(float update_period = 500); void door_position_update(); void cancel_position_sync_callbacks(); // light void toggle_light(); void light_on(); void light_off(); LightState get_light_state() const; // lock void toggle_lock(); void lock(); void unlock(); // Learn & Paired void activate_learn(); void inactivate_learn(); void query_paired_devices(); void query_paired_devices(PairedDevice kind); void clear_paired_devices(PairedDevice kind); // button functionality void query_status(); void query_openings(); void sync(); // children subscriptions void subscribe_rolling_code_counter(std::function&& f); void subscribe_opening_duration(std::function&& f); void subscribe_closing_duration(std::function&& f); void subscribe_openings(std::function&& f); void subscribe_paired_devices_total(std::function&& f); void subscribe_paired_remotes(std::function&& f); void subscribe_paired_keypads(std::function&& f); void subscribe_paired_wall_controls(std::function&& f); void subscribe_paired_accessories(std::function&& f); void subscribe_door_state(std::function&& f); void subscribe_light_state(std::function&& f); void subscribe_lock_state(std::function&& f); void subscribe_obstruction_state(std::function&& f); void subscribe_motor_state(std::function&& f); void subscribe_button_state(std::function&& f); void subscribe_motion_state(std::function&& f); void subscribe_sync_failed(std::function&& f); void subscribe_learn_state(std::function&& f); protected: // tx data bool transmit_pending_ { false }; uint32_t transmit_pending_start_ { 0 }; WirePacket tx_packet_; RATGDOStore isr_store_ {}; SoftwareSerial sw_serial_; bool obstruction_from_status_ { false }; bool learn_poll_status_ { true }; InternalGPIOPin* output_gdo_pin_; InternalGPIOPin* input_gdo_pin_; InternalGPIOPin* input_obst_pin_; uint64_t client_id_ { 0x539 }; }; // RATGDOComponent } // namespace ratgdo } // namespace esphome