esphome 2024.12.4__py3-none-any.whl → 2025.2.0b2__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (358) hide show
  1. esphome/__main__.py +16 -3
  2. esphome/components/adc/__init__.py +17 -11
  3. esphome/components/adc/adc_sensor.h +17 -0
  4. esphome/components/adc/adc_sensor_common.cpp +55 -0
  5. esphome/components/adc/adc_sensor_esp32.cpp +8 -5
  6. esphome/components/adc/adc_sensor_esp8266.cpp +10 -6
  7. esphome/components/adc/adc_sensor_libretiny.cpp +11 -6
  8. esphome/components/adc/adc_sensor_rp2040.cpp +13 -10
  9. esphome/components/adc/sensor.py +9 -3
  10. esphome/components/ads1115/ads1115.cpp +56 -7
  11. esphome/components/ads1115/ads1115.h +13 -1
  12. esphome/components/ads1115/sensor/__init__.py +16 -0
  13. esphome/components/ads1115/sensor/ads1115_sensor.cpp +2 -1
  14. esphome/components/ads1115/sensor/ads1115_sensor.h +2 -0
  15. esphome/components/animation/__init__.py +23 -261
  16. esphome/components/animation/animation.cpp +2 -2
  17. esphome/components/animation/animation.h +2 -1
  18. esphome/components/api/api_pb2.cpp +14 -0
  19. esphome/components/api/api_pb2.h +1 -0
  20. esphome/components/audio/__init__.py +112 -0
  21. esphome/components/audio/audio.cpp +67 -0
  22. esphome/components/audio/audio.h +125 -7
  23. esphome/components/audio/audio_decoder.cpp +361 -0
  24. esphome/components/audio/audio_decoder.h +135 -0
  25. esphome/components/audio/audio_reader.cpp +308 -0
  26. esphome/components/audio/audio_reader.h +85 -0
  27. esphome/components/audio/audio_resampler.cpp +159 -0
  28. esphome/components/audio/audio_resampler.h +101 -0
  29. esphome/components/audio/audio_transfer_buffer.cpp +165 -0
  30. esphome/components/audio/audio_transfer_buffer.h +139 -0
  31. esphome/components/audio_adc/__init__.py +41 -0
  32. esphome/components/audio_adc/audio_adc.h +17 -0
  33. esphome/components/audio_adc/automation.h +23 -0
  34. esphome/components/bk72xx/__init__.py +1 -0
  35. esphome/components/ble_client/ble_client.cpp +1 -2
  36. esphome/components/ble_client/sensor/__init__.py +1 -1
  37. esphome/components/ble_client/text_sensor/__init__.py +1 -1
  38. esphome/components/bluetooth_proxy/bluetooth_connection.cpp +5 -0
  39. esphome/components/bluetooth_proxy/bluetooth_connection.h +1 -0
  40. esphome/components/bluetooth_proxy/bluetooth_proxy.cpp +5 -0
  41. esphome/components/ch422g/ch422g.h +2 -0
  42. esphome/components/climate/__init__.py +1 -1
  43. esphome/components/climate_ir/climate_ir.cpp +2 -1
  44. esphome/components/coolix/coolix.cpp +2 -1
  45. esphome/components/cse7766/cse7766.cpp +8 -16
  46. esphome/components/custom/__init__.py +0 -3
  47. esphome/components/custom/binary_sensor/__init__.py +2 -28
  48. esphome/components/custom/climate/__init__.py +2 -27
  49. esphome/components/custom/cover/__init__.py +2 -27
  50. esphome/components/custom/light/__init__.py +2 -27
  51. esphome/components/custom/output/__init__.py +2 -58
  52. esphome/components/custom/sensor/__init__.py +2 -24
  53. esphome/components/custom/switch/__init__.py +2 -24
  54. esphome/components/custom/text_sensor/__init__.py +2 -29
  55. esphome/components/custom_component/__init__.py +3 -27
  56. esphome/components/daly_bms/daly_bms.cpp +6 -0
  57. esphome/components/daly_bms/daly_bms.h +2 -0
  58. esphome/components/daly_bms/sensor.py +6 -0
  59. esphome/components/debug/debug_component.cpp +4 -0
  60. esphome/components/debug/debug_component.h +14 -0
  61. esphome/components/debug/debug_esp32.cpp +154 -74
  62. esphome/components/dfplayer/dfplayer.cpp +15 -2
  63. esphome/components/dfrobot_sen0395/dfrobot_sen0395.cpp +2 -1
  64. esphome/components/dht/dht.cpp +2 -1
  65. esphome/components/display/__init__.py +18 -5
  66. esphome/components/display/display.cpp +2 -1
  67. esphome/components/display/rect.cpp +2 -1
  68. esphome/components/es7210/__init__.py +0 -0
  69. esphome/components/es7210/audio_adc.py +51 -0
  70. esphome/components/es7210/es7210.cpp +228 -0
  71. esphome/components/es7210/es7210.h +62 -0
  72. esphome/components/es7210/es7210_const.h +129 -0
  73. esphome/components/es7243e/__init__.py +0 -0
  74. esphome/components/es7243e/audio_adc.py +34 -0
  75. esphome/components/es7243e/es7243e.cpp +125 -0
  76. esphome/components/es7243e/es7243e.h +37 -0
  77. esphome/components/es7243e/es7243e_const.h +54 -0
  78. esphome/components/es8156/__init__.py +0 -0
  79. esphome/components/es8156/audio_dac.py +27 -0
  80. esphome/components/es8156/es8156.cpp +87 -0
  81. esphome/components/es8156/es8156.h +51 -0
  82. esphome/components/es8156/es8156_const.h +68 -0
  83. esphome/components/es8311/audio_dac.py +1 -2
  84. esphome/components/esp32/__init__.py +1 -0
  85. esphome/components/esp32/core.cpp +5 -1
  86. esphome/components/esp32/gpio.h +2 -0
  87. esphome/components/esp32_ble/__init__.py +39 -0
  88. esphome/components/esp32_ble/queue.h +4 -4
  89. esphome/components/esp32_ble_client/ble_client_base.cpp +46 -0
  90. esphome/components/esp32_ble_client/ble_client_base.h +2 -0
  91. esphome/components/esp32_ble_server/__init__.py +582 -12
  92. esphome/components/esp32_ble_server/ble_characteristic.cpp +48 -60
  93. esphome/components/esp32_ble_server/ble_characteristic.h +24 -17
  94. esphome/components/esp32_ble_server/ble_descriptor.cpp +21 -9
  95. esphome/components/esp32_ble_server/ble_descriptor.h +17 -6
  96. esphome/components/esp32_ble_server/ble_server.cpp +62 -67
  97. esphome/components/esp32_ble_server/ble_server.h +28 -32
  98. esphome/components/esp32_ble_server/ble_server_automations.cpp +77 -0
  99. esphome/components/esp32_ble_server/ble_server_automations.h +115 -0
  100. esphome/components/esp32_ble_server/ble_service.cpp +17 -15
  101. esphome/components/esp32_ble_server/ble_service.h +10 -14
  102. esphome/components/esp32_ble_tracker/__init__.py +6 -39
  103. esphome/components/esp32_ble_tracker/esp32_ble_tracker.cpp +33 -10
  104. esphome/components/esp32_ble_tracker/esp32_ble_tracker.h +8 -4
  105. esphome/components/esp32_improv/__init__.py +2 -8
  106. esphome/components/esp32_improv/esp32_improv_component.cpp +21 -20
  107. esphome/components/esp32_improv/esp32_improv_component.h +3 -4
  108. esphome/components/esp32_rmt/__init__.py +28 -3
  109. esphome/components/esp32_rmt_led_strip/led_strip.cpp +73 -6
  110. esphome/components/esp32_rmt_led_strip/led_strip.h +21 -3
  111. esphome/components/esp32_rmt_led_strip/light.py +72 -7
  112. esphome/components/esp32_touch/esp32_touch.cpp +5 -0
  113. esphome/components/esp8266/__init__.py +1 -0
  114. esphome/components/esp8266/gpio.h +1 -0
  115. esphome/components/ethernet/__init__.py +10 -10
  116. esphome/components/event/event.cpp +4 -2
  117. esphome/components/event/event.h +2 -0
  118. esphome/components/event_emitter/__init__.py +5 -0
  119. esphome/components/event_emitter/event_emitter.cpp +14 -0
  120. esphome/components/event_emitter/event_emitter.h +63 -0
  121. esphome/components/gcja5/gcja5.cpp +2 -1
  122. esphome/components/graph/graph.cpp +4 -9
  123. esphome/components/haier/haier_base.cpp +2 -1
  124. esphome/components/haier/hon_climate.cpp +2 -1
  125. esphome/components/heatpumpir/heatpumpir.cpp +2 -1
  126. esphome/components/host/__init__.py +1 -0
  127. esphome/components/host/gpio.h +1 -0
  128. esphome/components/http_request/http_request.h +2 -2
  129. esphome/components/http_request/http_request_arduino.cpp +1 -1
  130. esphome/components/http_request/http_request_idf.cpp +1 -1
  131. esphome/components/i2c/i2c_bus_esp_idf.cpp +4 -0
  132. esphome/components/i2s_audio/microphone/i2s_audio_microphone.cpp +7 -5
  133. esphome/components/i2s_audio/speaker/__init__.py +53 -6
  134. esphome/components/i2s_audio/speaker/i2s_audio_speaker.cpp +92 -46
  135. esphome/components/i2s_audio/speaker/i2s_audio_speaker.h +8 -0
  136. esphome/components/ili9xxx/display.py +29 -11
  137. esphome/components/ili9xxx/ili9xxx_display.cpp +2 -5
  138. esphome/components/ili9xxx/ili9xxx_display.h +2 -1
  139. esphome/components/image/__init__.py +443 -255
  140. esphome/components/image/image.cpp +115 -61
  141. esphome/components/image/image.h +15 -24
  142. esphome/components/json/json_util.cpp +8 -34
  143. esphome/components/libretiny/__init__.py +1 -0
  144. esphome/components/libretiny/gpio_arduino.h +1 -0
  145. esphome/components/light/light_color_values.h +1 -1
  146. esphome/components/logger/__init__.py +43 -7
  147. esphome/components/logger/logger.cpp +16 -11
  148. esphome/components/logger/logger.h +11 -7
  149. esphome/components/logger/select/__init__.py +29 -0
  150. esphome/components/logger/select/logger_level_select.cpp +27 -0
  151. esphome/components/logger/select/logger_level_select.h +15 -0
  152. esphome/components/lvgl/__init__.py +96 -73
  153. esphome/components/lvgl/automation.py +39 -7
  154. esphome/components/lvgl/defines.py +8 -2
  155. esphome/components/lvgl/lvgl_esphome.cpp +8 -15
  156. esphome/components/lvgl/lvgl_esphome.h +20 -5
  157. esphome/components/lvgl/schemas.py +25 -14
  158. esphome/components/lvgl/trigger.py +27 -3
  159. esphome/components/lvgl/widgets/dropdown.py +1 -1
  160. esphome/components/lvgl/widgets/keyboard.py +8 -1
  161. esphome/components/lvgl/widgets/meter.py +2 -1
  162. esphome/components/lvgl/widgets/msgbox.py +1 -1
  163. esphome/components/lvgl/widgets/obj.py +1 -12
  164. esphome/components/lvgl/widgets/page.py +37 -2
  165. esphome/components/lvgl/widgets/tabview.py +1 -1
  166. esphome/components/max6956/max6956.h +2 -0
  167. esphome/components/mcp23016/mcp23016.h +2 -0
  168. esphome/components/mcp23xxx_base/mcp23xxx_base.h +2 -0
  169. esphome/components/mdns/__init__.py +1 -1
  170. esphome/components/media_player/__init__.py +37 -8
  171. esphome/components/media_player/automation.h +11 -2
  172. esphome/components/media_player/media_player.cpp +8 -0
  173. esphome/components/media_player/media_player.h +8 -4
  174. esphome/components/micronova/switch/micronova_switch.cpp +4 -2
  175. esphome/components/midea/ac_automations.h +3 -1
  176. esphome/components/midea/air_conditioner.cpp +7 -5
  177. esphome/components/midea/air_conditioner.h +1 -1
  178. esphome/components/midea/climate.py +4 -2
  179. esphome/components/midea/ir_transmitter.h +36 -5
  180. esphome/components/mixer/__init__.py +0 -0
  181. esphome/components/mixer/speaker/__init__.py +172 -0
  182. esphome/components/mixer/speaker/automation.h +19 -0
  183. esphome/components/mixer/speaker/mixer_speaker.cpp +624 -0
  184. esphome/components/mixer/speaker/mixer_speaker.h +207 -0
  185. esphome/components/modbus_controller/text_sensor/modbus_textsensor.cpp +7 -13
  186. esphome/components/mpr121/mpr121.h +2 -0
  187. esphome/components/mqtt/__init__.py +1 -1
  188. esphome/components/mqtt/mqtt_client.cpp +7 -1
  189. esphome/components/mqtt/mqtt_client.h +1 -1
  190. esphome/components/mqtt/mqtt_climate.cpp +2 -2
  191. esphome/components/network/ip_address.h +2 -0
  192. esphome/components/nextion/automation.h +17 -0
  193. esphome/components/nextion/display.py +42 -17
  194. esphome/components/nextion/nextion.cpp +4 -10
  195. esphome/components/nextion/nextion.h +89 -82
  196. esphome/components/nextion/nextion_commands.cpp +10 -10
  197. esphome/components/ntc/sensor.py +2 -4
  198. esphome/components/online_image/__init__.py +98 -46
  199. esphome/components/online_image/bmp_image.cpp +101 -0
  200. esphome/components/online_image/bmp_image.h +40 -0
  201. esphome/components/online_image/image_decoder.cpp +28 -2
  202. esphome/components/online_image/image_decoder.h +24 -15
  203. esphome/components/online_image/jpeg_image.cpp +90 -0
  204. esphome/components/online_image/jpeg_image.h +34 -0
  205. esphome/components/online_image/online_image.cpp +112 -53
  206. esphome/components/online_image/online_image.h +24 -7
  207. esphome/components/online_image/png_image.cpp +7 -3
  208. esphome/components/online_image/png_image.h +2 -1
  209. esphome/components/opentherm/__init__.py +73 -7
  210. esphome/components/opentherm/automation.h +25 -0
  211. esphome/components/opentherm/const.py +1 -0
  212. esphome/components/opentherm/generate.py +39 -6
  213. esphome/components/opentherm/hub.cpp +117 -79
  214. esphome/components/opentherm/hub.h +31 -15
  215. esphome/components/opentherm/opentherm.cpp +47 -23
  216. esphome/components/opentherm/opentherm.h +27 -6
  217. esphome/components/opentherm/opentherm_macros.h +11 -0
  218. esphome/components/opentherm/schema.py +78 -1
  219. esphome/components/opentherm/validate.py +7 -2
  220. esphome/components/pca6416a/pca6416a.h +2 -0
  221. esphome/components/pca9554/pca9554.h +2 -0
  222. esphome/components/pcf8574/pcf8574.h +2 -0
  223. esphome/components/preferences/__init__.py +2 -4
  224. esphome/components/preferences/syncer.h +10 -3
  225. esphome/components/prometheus/prometheus_handler.cpp +313 -0
  226. esphome/components/prometheus/prometheus_handler.h +48 -7
  227. esphome/components/psram/psram.cpp +8 -1
  228. esphome/components/pulse_counter/pulse_counter_sensor.cpp +14 -9
  229. esphome/components/pulse_counter/pulse_counter_sensor.h +4 -4
  230. esphome/components/pulse_meter/pulse_meter_sensor.cpp +2 -0
  231. esphome/components/qspi_dbi/__init__.py +3 -0
  232. esphome/components/qspi_dbi/display.py +74 -47
  233. esphome/components/qspi_dbi/models.py +245 -2
  234. esphome/components/qspi_dbi/qspi_dbi.cpp +9 -16
  235. esphome/components/qspi_dbi/qspi_dbi.h +2 -2
  236. esphome/components/remote_base/__init__.py +77 -25
  237. esphome/components/remote_base/remote_base.cpp +1 -1
  238. esphome/components/remote_base/remote_base.h +20 -2
  239. esphome/components/remote_base/toto_protocol.cpp +100 -0
  240. esphome/components/remote_base/toto_protocol.h +45 -0
  241. esphome/components/remote_receiver/__init__.py +55 -10
  242. esphome/components/remote_receiver/remote_receiver.h +36 -3
  243. esphome/components/remote_receiver/remote_receiver_esp32.cpp +145 -6
  244. esphome/components/remote_transmitter/__init__.py +62 -4
  245. esphome/components/remote_transmitter/remote_transmitter.h +21 -2
  246. esphome/components/remote_transmitter/remote_transmitter_esp32.cpp +140 -4
  247. esphome/components/remote_transmitter/remote_transmitter_esp8266.cpp +3 -3
  248. esphome/components/remote_transmitter/remote_transmitter_libretiny.cpp +3 -3
  249. esphome/components/resampler/__init__.py +0 -0
  250. esphome/components/resampler/speaker/__init__.py +103 -0
  251. esphome/components/resampler/speaker/resampler_speaker.cpp +318 -0
  252. esphome/components/resampler/speaker/resampler_speaker.h +107 -0
  253. esphome/components/resistance/resistance_sensor.h +2 -3
  254. esphome/components/resistance/sensor.py +2 -9
  255. esphome/components/rotary_encoder/rotary_encoder.cpp +8 -4
  256. esphome/components/rp2040/__init__.py +1 -0
  257. esphome/components/rp2040/gpio.h +1 -0
  258. esphome/components/rtl87xx/__init__.py +2 -0
  259. esphome/components/sdl/binary_sensor.py +270 -0
  260. esphome/components/sdl/sdl_esphome.cpp +16 -0
  261. esphome/components/sdl/sdl_esphome.h +9 -0
  262. esphome/components/seeed_mr60bha2/binary_sensor.py +25 -0
  263. esphome/components/seeed_mr60bha2/seeed_mr60bha2.cpp +26 -2
  264. esphome/components/seeed_mr60bha2/seeed_mr60bha2.h +9 -20
  265. esphome/components/seeed_mr60bha2/sensor.py +9 -1
  266. esphome/components/sn74hc165/sn74hc165.h +3 -0
  267. esphome/components/sn74hc595/sn74hc595.h +3 -0
  268. esphome/components/speaker/__init__.py +5 -4
  269. esphome/components/speaker/media_player/__init__.py +458 -0
  270. esphome/components/speaker/media_player/audio_pipeline.cpp +568 -0
  271. esphome/components/speaker/media_player/audio_pipeline.h +159 -0
  272. esphome/components/speaker/media_player/automation.h +26 -0
  273. esphome/components/speaker/media_player/speaker_media_player.cpp +577 -0
  274. esphome/components/speaker/media_player/speaker_media_player.h +160 -0
  275. esphome/components/speaker/speaker.h +20 -0
  276. esphome/components/spi/__init__.py +1 -5
  277. esphome/components/spi/spi.cpp +7 -1
  278. esphome/components/spi/spi.h +21 -2
  279. esphome/components/spi_led_strip/light.py +3 -5
  280. esphome/components/spi_led_strip/spi_led_strip.cpp +67 -0
  281. esphome/components/spi_led_strip/spi_led_strip.h +8 -60
  282. esphome/components/sprinkler/sprinkler.cpp +3 -1
  283. esphome/components/sx1509/sx1509_gpio_pin.h +2 -0
  284. esphome/components/tca9555/tca9555.h +2 -0
  285. esphome/components/toshiba/toshiba.cpp +2 -1
  286. esphome/components/tuya/light/tuya_light.cpp +4 -2
  287. esphome/components/uart/uart_component_esp32_arduino.cpp +2 -2
  288. esphome/components/uart/uart_component_esp_idf.cpp +2 -2
  289. esphome/components/udp/__init__.py +8 -2
  290. esphome/components/udp/udp_component.cpp +25 -56
  291. esphome/components/udp/udp_component.h +3 -0
  292. esphome/components/uponor_smatrix/sensor/__init__.py +14 -4
  293. esphome/components/uponor_smatrix/sensor/uponor_smatrix_sensor.cpp +5 -0
  294. esphome/components/uponor_smatrix/sensor/uponor_smatrix_sensor.h +1 -0
  295. esphome/components/uptime/text_sensor/__init__.py +19 -0
  296. esphome/components/uptime/text_sensor/uptime_text_sensor.cpp +63 -0
  297. esphome/components/uptime/text_sensor/uptime_text_sensor.h +25 -0
  298. esphome/components/voice_assistant/voice_assistant.cpp +24 -14
  299. esphome/components/voice_assistant/voice_assistant.h +8 -0
  300. esphome/components/waveshare_epaper/display.py +22 -1
  301. esphome/components/waveshare_epaper/waveshare_213v3.cpp +9 -3
  302. esphome/components/waveshare_epaper/waveshare_epaper.cpp +1155 -44
  303. esphome/components/waveshare_epaper/waveshare_epaper.h +208 -7
  304. esphome/components/web_server/web_server.cpp +28 -6
  305. esphome/components/weikai/weikai.h +2 -0
  306. esphome/components/wifi/__init__.py +6 -6
  307. esphome/components/wifi/wifi_component.cpp +1 -1
  308. esphome/components/wifi/wifi_component_esp32_arduino.cpp +30 -1
  309. esphome/components/wireguard/__init__.py +2 -2
  310. esphome/components/xl9535/xl9535.h +2 -0
  311. esphome/components/xxtea/__init__.py +3 -0
  312. esphome/components/xxtea/xxtea.cpp +46 -0
  313. esphome/components/xxtea/xxtea.h +26 -0
  314. esphome/components/yashima/yashima.cpp +2 -1
  315. esphome/config.py +9 -5
  316. esphome/config_validation.py +55 -17
  317. esphome/const.py +7 -10
  318. esphome/core/__init__.py +6 -13
  319. esphome/core/base_automation.h +1 -0
  320. esphome/core/config.py +59 -72
  321. esphome/core/defines.h +9 -1
  322. esphome/core/gpio.h +7 -0
  323. esphome/core/helpers.cpp +19 -15
  324. esphome/core/helpers.h +57 -8
  325. esphome/core/log.h +9 -7
  326. esphome/cpp_generator.py +2 -2
  327. esphome/espota2.py +3 -2
  328. esphome/loader.py +12 -4
  329. esphome/log.py +5 -7
  330. esphome/yaml_util.py +2 -2
  331. {esphome-2024.12.4.dist-info → esphome-2025.2.0b2.dist-info}/METADATA +12 -7
  332. {esphome-2024.12.4.dist-info → esphome-2025.2.0b2.dist-info}/RECORD +341 -292
  333. esphome/components/custom/binary_sensor/custom_binary_sensor.cpp +0 -16
  334. esphome/components/custom/binary_sensor/custom_binary_sensor.h +0 -26
  335. esphome/components/custom/climate/custom_climate.h +0 -22
  336. esphome/components/custom/cover/custom_cover.h +0 -21
  337. esphome/components/custom/light/custom_light_output.h +0 -24
  338. esphome/components/custom/output/custom_output.h +0 -37
  339. esphome/components/custom/sensor/custom_sensor.cpp +0 -16
  340. esphome/components/custom/sensor/custom_sensor.h +0 -24
  341. esphome/components/custom/switch/custom_switch.cpp +0 -16
  342. esphome/components/custom/switch/custom_switch.h +0 -24
  343. esphome/components/custom/text_sensor/custom_text_sensor.cpp +0 -16
  344. esphome/components/custom/text_sensor/custom_text_sensor.h +0 -26
  345. esphome/components/custom_component/custom_component.h +0 -28
  346. esphome/components/esp32_ble_server/ble_2901.cpp +0 -18
  347. esphome/components/esp32_ble_server/ble_2901.h +0 -19
  348. esphome/components/resistance_sampler/__init__.py +0 -6
  349. esphome/components/resistance_sampler/resistance_sampler.h +0 -10
  350. esphome/components/uptime/{sensor.py → sensor/__init__.py} +3 -3
  351. /esphome/components/uptime/{uptime_seconds_sensor.cpp → sensor/uptime_seconds_sensor.cpp} +0 -0
  352. /esphome/components/uptime/{uptime_seconds_sensor.h → sensor/uptime_seconds_sensor.h} +0 -0
  353. /esphome/components/uptime/{uptime_timestamp_sensor.cpp → sensor/uptime_timestamp_sensor.cpp} +0 -0
  354. /esphome/components/uptime/{uptime_timestamp_sensor.h → sensor/uptime_timestamp_sensor.h} +0 -0
  355. {esphome-2024.12.4.dist-info → esphome-2025.2.0b2.dist-info}/LICENSE +0 -0
  356. {esphome-2024.12.4.dist-info → esphome-2025.2.0b2.dist-info}/WHEEL +0 -0
  357. {esphome-2024.12.4.dist-info → esphome-2025.2.0b2.dist-info}/entry_points.txt +0 -0
  358. {esphome-2024.12.4.dist-info → esphome-2025.2.0b2.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,100 @@
1
+ #include "toto_protocol.h"
2
+ #include "esphome/core/log.h"
3
+
4
+ namespace esphome {
5
+ namespace remote_base {
6
+
7
+ static const char *const TAG = "remote.toto";
8
+
9
+ static const uint32_t PREAMBLE_HIGH_US = 6200;
10
+ static const uint32_t PREAMBLE_LOW_US = 2800;
11
+ static const uint32_t BIT_HIGH_US = 550;
12
+ static const uint32_t BIT_ONE_LOW_US = 1700;
13
+ static const uint32_t BIT_ZERO_LOW_US = 550;
14
+ static const uint32_t TOTO_HEADER = 0x2008;
15
+
16
+ void TotoProtocol::encode(RemoteTransmitData *dst, const TotoData &data) {
17
+ uint32_t payload = 0;
18
+
19
+ payload = data.rc_code_1 << 20;
20
+ payload |= data.rc_code_2 << 16;
21
+ payload |= data.command << 8;
22
+ payload |= ((payload & 0xFF0000) >> 16) ^ ((payload & 0x00FF00) >> 8);
23
+
24
+ dst->reserve(80);
25
+ dst->set_carrier_frequency(38000);
26
+ dst->item(PREAMBLE_HIGH_US, PREAMBLE_LOW_US);
27
+
28
+ for (uint32_t mask = 1UL << 14; mask; mask >>= 1) {
29
+ if (TOTO_HEADER & mask) {
30
+ dst->item(BIT_HIGH_US, BIT_ONE_LOW_US);
31
+ } else {
32
+ dst->item(BIT_HIGH_US, BIT_ZERO_LOW_US);
33
+ }
34
+ }
35
+
36
+ for (uint32_t mask = 1UL << 23; mask; mask >>= 1) {
37
+ if (payload & mask) {
38
+ dst->item(BIT_HIGH_US, BIT_ONE_LOW_US);
39
+ } else {
40
+ dst->item(BIT_HIGH_US, BIT_ZERO_LOW_US);
41
+ }
42
+ }
43
+
44
+ dst->mark(BIT_HIGH_US);
45
+ }
46
+ optional<TotoData> TotoProtocol::decode(RemoteReceiveData src) {
47
+ uint16_t header = 0;
48
+ uint32_t payload = 0;
49
+
50
+ TotoData data{
51
+ .rc_code_1 = 0,
52
+ .rc_code_2 = 0,
53
+ .command = 0,
54
+ };
55
+
56
+ if (!src.expect_item(PREAMBLE_HIGH_US, PREAMBLE_LOW_US)) {
57
+ return {};
58
+ }
59
+
60
+ for (uint32_t mask = 1UL << 14; mask; mask >>= 1) {
61
+ if (src.expect_item(BIT_HIGH_US, BIT_ONE_LOW_US)) {
62
+ header |= mask;
63
+ } else if (src.expect_item(BIT_HIGH_US, BIT_ZERO_LOW_US)) {
64
+ header &= ~mask;
65
+ } else {
66
+ return {};
67
+ }
68
+ }
69
+
70
+ if (header != TOTO_HEADER) {
71
+ return {};
72
+ }
73
+
74
+ for (uint32_t mask = 1UL << 23; mask; mask >>= 1) {
75
+ if (src.expect_item(BIT_HIGH_US, BIT_ONE_LOW_US)) {
76
+ payload |= mask;
77
+ } else if (src.expect_item(BIT_HIGH_US, BIT_ZERO_LOW_US)) {
78
+ payload &= ~mask;
79
+ } else {
80
+ return {};
81
+ }
82
+ }
83
+
84
+ if ((((payload & 0xFF0000) >> 16) ^ ((payload & 0x00FF00) >> 8)) != (payload & 0x0000FF)) {
85
+ return {};
86
+ }
87
+
88
+ data.rc_code_1 = (payload & 0xF00000) >> 20;
89
+ data.rc_code_2 = (payload & 0x0F0000) >> 16;
90
+ data.command = (payload & 0x00FF00) >> 8;
91
+
92
+ return data;
93
+ }
94
+ void TotoProtocol::dump(const TotoData &data) {
95
+ ESP_LOGI(TAG, "Received Toto data: rc_code_1=0x%01X, rc_code_2=0x%01X, command=0x%02X", data.rc_code_1,
96
+ data.rc_code_2, data.command);
97
+ }
98
+
99
+ } // namespace remote_base
100
+ } // namespace esphome
@@ -0,0 +1,45 @@
1
+ #pragma once
2
+
3
+ #include "remote_base.h"
4
+
5
+ namespace esphome {
6
+ namespace remote_base {
7
+
8
+ struct TotoData {
9
+ uint8_t rc_code_1 : 4;
10
+ uint8_t rc_code_2 : 4;
11
+ uint8_t command;
12
+
13
+ bool operator==(const TotoData &rhs) const {
14
+ return (rc_code_1 == rhs.rc_code_1) && (rc_code_2 == rhs.rc_code_2) && (command == rhs.command);
15
+ }
16
+ };
17
+
18
+ class TotoProtocol : public RemoteProtocol<TotoData> {
19
+ public:
20
+ void encode(RemoteTransmitData *dst, const TotoData &data) override;
21
+ optional<TotoData> decode(RemoteReceiveData src) override;
22
+ void dump(const TotoData &data) override;
23
+ };
24
+
25
+ DECLARE_REMOTE_PROTOCOL(Toto)
26
+
27
+ template<typename... Ts> class TotoAction : public RemoteTransmitterActionBase<Ts...> {
28
+ public:
29
+ TEMPLATABLE_VALUE(uint8_t, rc_code_1)
30
+ TEMPLATABLE_VALUE(uint8_t, rc_code_2)
31
+ TEMPLATABLE_VALUE(uint8_t, command)
32
+
33
+ void encode(RemoteTransmitData *dst, Ts... x) override {
34
+ TotoData data{};
35
+ data.rc_code_1 = this->rc_code_1_.value(x...);
36
+ data.rc_code_2 = this->rc_code_2_.value(x...);
37
+ data.command = this->command_.value(x...);
38
+ this->set_send_times(this->send_times_.value_or(x..., 3));
39
+ this->set_send_wait(this->send_wait_.value_or(x..., 32000));
40
+ TotoProtocol().encode(dst, data);
41
+ }
42
+ };
43
+
44
+ } // namespace remote_base
45
+ } // namespace esphome
@@ -5,6 +5,7 @@ import esphome.config_validation as cv
5
5
  from esphome.const import (
6
6
  CONF_BUFFER_SIZE,
7
7
  CONF_CLOCK_DIVIDER,
8
+ CONF_CLOCK_RESOLUTION,
8
9
  CONF_DUMP,
9
10
  CONF_FILTER,
10
11
  CONF_ID,
@@ -12,12 +13,17 @@ from esphome.const import (
12
13
  CONF_MEMORY_BLOCKS,
13
14
  CONF_PIN,
14
15
  CONF_RMT_CHANNEL,
16
+ CONF_RMT_SYMBOLS,
15
17
  CONF_TOLERANCE,
16
18
  CONF_TYPE,
19
+ CONF_USE_DMA,
17
20
  CONF_VALUE,
18
21
  )
19
22
  from esphome.core import CORE, TimePeriod
20
23
 
24
+ CONF_FILTER_SYMBOLS = "filter_symbols"
25
+ CONF_RECEIVE_SYMBOLS = "receive_symbols"
26
+
21
27
  AUTO_LOAD = ["remote_base"]
22
28
  remote_receiver_ns = cg.esphome_ns.namespace("remote_receiver")
23
29
  remote_base_ns = cg.esphome_ns.namespace("remote_base")
@@ -97,15 +103,43 @@ CONFIG_SCHEMA = remote_base.validate_triggers(
97
103
  cv.positive_time_period_microseconds,
98
104
  cv.Range(max=TimePeriod(microseconds=4294967295)),
99
105
  ),
100
- cv.SplitDefault(CONF_CLOCK_DIVIDER, esp32=80): cv.All(
101
- cv.only_on_esp32, cv.Range(min=1, max=255)
106
+ cv.SplitDefault(CONF_CLOCK_DIVIDER, esp32_arduino=80): cv.All(
107
+ cv.only_on_esp32,
108
+ cv.only_with_arduino,
109
+ cv.int_range(min=1, max=255),
110
+ ),
111
+ cv.Optional(CONF_CLOCK_RESOLUTION): cv.All(
112
+ cv.only_on_esp32,
113
+ cv.only_with_esp_idf,
114
+ esp32_rmt.validate_clock_resolution(),
102
115
  ),
103
116
  cv.Optional(CONF_IDLE, default="10ms"): cv.All(
104
117
  cv.positive_time_period_microseconds,
105
118
  cv.Range(max=TimePeriod(microseconds=4294967295)),
106
119
  ),
107
- cv.Optional(CONF_MEMORY_BLOCKS, default=3): cv.Range(min=1, max=8),
108
- cv.Optional(CONF_RMT_CHANNEL): esp32_rmt.validate_rmt_channel(tx=False),
120
+ cv.SplitDefault(CONF_MEMORY_BLOCKS, esp32_arduino=3): cv.All(
121
+ cv.only_with_arduino, cv.int_range(min=1, max=8)
122
+ ),
123
+ cv.Optional(CONF_RMT_CHANNEL): cv.All(
124
+ cv.only_with_arduino, esp32_rmt.validate_rmt_channel(tx=False)
125
+ ),
126
+ cv.SplitDefault(
127
+ CONF_RMT_SYMBOLS,
128
+ esp32_idf=192,
129
+ esp32_s2_idf=192,
130
+ esp32_s3_idf=192,
131
+ esp32_c3_idf=96,
132
+ esp32_c6_idf=96,
133
+ esp32_h2_idf=96,
134
+ ): cv.All(cv.only_with_esp_idf, cv.int_range(min=2)),
135
+ cv.Optional(CONF_FILTER_SYMBOLS): cv.All(
136
+ cv.only_with_esp_idf, cv.int_range(min=0)
137
+ ),
138
+ cv.SplitDefault(
139
+ CONF_RECEIVE_SYMBOLS,
140
+ esp32_idf=192,
141
+ ): cv.All(cv.only_with_esp_idf, cv.int_range(min=2)),
142
+ cv.Optional(CONF_USE_DMA): cv.All(cv.only_with_esp_idf, cv.boolean),
109
143
  }
110
144
  ).extend(cv.COMPONENT_SCHEMA)
111
145
  )
@@ -114,13 +148,24 @@ CONFIG_SCHEMA = remote_base.validate_triggers(
114
148
  async def to_code(config):
115
149
  pin = await cg.gpio_pin_expression(config[CONF_PIN])
116
150
  if CORE.is_esp32:
117
- if (rmt_channel := config.get(CONF_RMT_CHANNEL, None)) is not None:
118
- var = cg.new_Pvariable(
119
- config[CONF_ID], pin, rmt_channel, config[CONF_MEMORY_BLOCKS]
120
- )
151
+ if esp32_rmt.use_new_rmt_driver():
152
+ var = cg.new_Pvariable(config[CONF_ID], pin)
153
+ cg.add(var.set_rmt_symbols(config[CONF_RMT_SYMBOLS]))
154
+ cg.add(var.set_receive_symbols(config[CONF_RECEIVE_SYMBOLS]))
155
+ if CONF_USE_DMA in config:
156
+ cg.add(var.set_with_dma(config[CONF_USE_DMA]))
157
+ if CONF_CLOCK_RESOLUTION in config:
158
+ cg.add(var.set_clock_resolution(config[CONF_CLOCK_RESOLUTION]))
159
+ if CONF_FILTER_SYMBOLS in config:
160
+ cg.add(var.set_filter_symbols(config[CONF_FILTER_SYMBOLS]))
121
161
  else:
122
- var = cg.new_Pvariable(config[CONF_ID], pin, config[CONF_MEMORY_BLOCKS])
123
- cg.add(var.set_clock_divider(config[CONF_CLOCK_DIVIDER]))
162
+ if (rmt_channel := config.get(CONF_RMT_CHANNEL, None)) is not None:
163
+ var = cg.new_Pvariable(
164
+ config[CONF_ID], pin, rmt_channel, config[CONF_MEMORY_BLOCKS]
165
+ )
166
+ else:
167
+ var = cg.new_Pvariable(config[CONF_ID], pin, config[CONF_MEMORY_BLOCKS])
168
+ cg.add(var.set_clock_divider(config[CONF_CLOCK_DIVIDER]))
124
169
  else:
125
170
  var = cg.new_Pvariable(config[CONF_ID], pin)
126
171
 
@@ -5,6 +5,10 @@
5
5
 
6
6
  #include <cinttypes>
7
7
 
8
+ #if defined(USE_ESP32) && ESP_IDF_VERSION_MAJOR >= 5
9
+ #include <driver/rmt_rx.h>
10
+ #endif
11
+
8
12
  namespace esphome {
9
13
  namespace remote_receiver {
10
14
 
@@ -25,6 +29,21 @@ struct RemoteReceiverComponentStore {
25
29
  uint32_t filter_us{10};
26
30
  ISRInternalGPIOPin pin;
27
31
  };
32
+ #elif defined(USE_ESP32) && ESP_IDF_VERSION_MAJOR >= 5
33
+ struct RemoteReceiverComponentStore {
34
+ /// Stores RMT symbols and rx done event data
35
+ volatile uint8_t *buffer{nullptr};
36
+ /// The position last written to
37
+ volatile uint32_t buffer_write{0};
38
+ /// The position last read from
39
+ volatile uint32_t buffer_read{0};
40
+ bool overflow{false};
41
+ uint32_t buffer_size{1000};
42
+ uint32_t receive_size{0};
43
+ uint32_t filter_symbols{0};
44
+ esp_err_t error{ESP_OK};
45
+ rmt_receive_config_t config;
46
+ };
28
47
  #endif
29
48
 
30
49
  class RemoteReceiverComponent : public remote_base::RemoteReceiverBase,
@@ -33,9 +52,10 @@ class RemoteReceiverComponent : public remote_base::RemoteReceiverBase,
33
52
  ,
34
53
  public remote_base::RemoteRMTChannel
35
54
  #endif
55
+
36
56
  {
37
57
  public:
38
- #ifdef USE_ESP32
58
+ #if defined(USE_ESP32) && ESP_IDF_VERSION_MAJOR < 5
39
59
  RemoteReceiverComponent(InternalGPIOPin *pin, uint8_t mem_block_num = 1)
40
60
  : RemoteReceiverBase(pin), remote_base::RemoteRMTChannel(mem_block_num) {}
41
61
 
@@ -49,19 +69,32 @@ class RemoteReceiverComponent : public remote_base::RemoteReceiverBase,
49
69
  void loop() override;
50
70
  float get_setup_priority() const override { return setup_priority::DATA; }
51
71
 
72
+ #if defined(USE_ESP32) && ESP_IDF_VERSION_MAJOR >= 5
73
+ void set_filter_symbols(uint32_t filter_symbols) { this->filter_symbols_ = filter_symbols; }
74
+ void set_receive_symbols(uint32_t receive_symbols) { this->receive_symbols_ = receive_symbols; }
75
+ void set_with_dma(bool with_dma) { this->with_dma_ = with_dma; }
76
+ #endif
52
77
  void set_buffer_size(uint32_t buffer_size) { this->buffer_size_ = buffer_size; }
53
78
  void set_filter_us(uint32_t filter_us) { this->filter_us_ = filter_us; }
54
79
  void set_idle_us(uint32_t idle_us) { this->idle_us_ = idle_us; }
55
80
 
56
81
  protected:
57
82
  #ifdef USE_ESP32
58
- void decode_rmt_(rmt_item32_t *item, size_t len);
83
+ #if ESP_IDF_VERSION_MAJOR >= 5
84
+ void decode_rmt_(rmt_symbol_word_t *item, size_t item_count);
85
+ rmt_channel_handle_t channel_{NULL};
86
+ uint32_t filter_symbols_{0};
87
+ uint32_t receive_symbols_{0};
88
+ bool with_dma_{false};
89
+ #else
90
+ void decode_rmt_(rmt_item32_t *item, size_t item_count);
59
91
  RingbufHandle_t ringbuf_;
92
+ #endif
60
93
  esp_err_t error_code_{ESP_OK};
61
94
  std::string error_string_{""};
62
95
  #endif
63
96
 
64
- #if defined(USE_ESP8266) || defined(USE_LIBRETINY)
97
+ #if defined(USE_ESP8266) || defined(USE_LIBRETINY) || (defined(USE_ESP32) && ESP_IDF_VERSION_MAJOR >= 5)
65
98
  RemoteReceiverComponentStore store_;
66
99
  HighFrequencyLoopRequester high_freq_;
67
100
  #endif
@@ -2,15 +2,110 @@
2
2
  #include "esphome/core/log.h"
3
3
 
4
4
  #ifdef USE_ESP32
5
- #include <driver/rmt.h>
5
+ #include <driver/gpio.h>
6
6
 
7
7
  namespace esphome {
8
8
  namespace remote_receiver {
9
9
 
10
10
  static const char *const TAG = "remote_receiver.esp32";
11
+ #ifdef USE_ESP32_VARIANT_ESP32H2
12
+ static const uint32_t RMT_CLK_FREQ = 32000000;
13
+ #else
14
+ static const uint32_t RMT_CLK_FREQ = 80000000;
15
+ #endif
16
+
17
+ #if ESP_IDF_VERSION_MAJOR >= 5
18
+ static bool IRAM_ATTR HOT rmt_callback(rmt_channel_handle_t channel, const rmt_rx_done_event_data_t *event, void *arg) {
19
+ RemoteReceiverComponentStore *store = (RemoteReceiverComponentStore *) arg;
20
+ rmt_rx_done_event_data_t *event_buffer = (rmt_rx_done_event_data_t *) (store->buffer + store->buffer_write);
21
+ uint32_t event_size = sizeof(rmt_rx_done_event_data_t);
22
+ uint32_t next_write = store->buffer_write + event_size + event->num_symbols * sizeof(rmt_symbol_word_t);
23
+ if (next_write + event_size + store->receive_size > store->buffer_size) {
24
+ next_write = 0;
25
+ }
26
+ if (store->buffer_read - next_write < event_size + store->receive_size) {
27
+ next_write = store->buffer_write;
28
+ store->overflow = true;
29
+ }
30
+ if (event->num_symbols <= store->filter_symbols) {
31
+ next_write = store->buffer_write;
32
+ }
33
+ store->error =
34
+ rmt_receive(channel, (uint8_t *) store->buffer + next_write + event_size, store->receive_size, &store->config);
35
+ event_buffer->num_symbols = event->num_symbols;
36
+ event_buffer->received_symbols = event->received_symbols;
37
+ store->buffer_write = next_write;
38
+ return false;
39
+ }
40
+ #endif
11
41
 
12
42
  void RemoteReceiverComponent::setup() {
13
43
  ESP_LOGCONFIG(TAG, "Setting up Remote Receiver...");
44
+ #if ESP_IDF_VERSION_MAJOR >= 5
45
+ rmt_rx_channel_config_t channel;
46
+ memset(&channel, 0, sizeof(channel));
47
+ channel.clk_src = RMT_CLK_SRC_DEFAULT;
48
+ channel.resolution_hz = this->clock_resolution_;
49
+ channel.mem_block_symbols = rmt_symbols_;
50
+ channel.gpio_num = gpio_num_t(this->pin_->get_pin());
51
+ channel.intr_priority = 0;
52
+ channel.flags.invert_in = 0;
53
+ channel.flags.with_dma = this->with_dma_;
54
+ channel.flags.io_loop_back = 0;
55
+ esp_err_t error = rmt_new_rx_channel(&channel, &this->channel_);
56
+ if (error != ESP_OK) {
57
+ this->error_code_ = error;
58
+ if (error == ESP_ERR_NOT_FOUND) {
59
+ this->error_string_ = "out of RMT symbol memory";
60
+ } else {
61
+ this->error_string_ = "in rmt_new_rx_channel";
62
+ }
63
+ this->mark_failed();
64
+ return;
65
+ }
66
+ if (this->pin_->get_flags() & gpio::FLAG_PULLUP) {
67
+ gpio_pullup_en(gpio_num_t(this->pin_->get_pin()));
68
+ } else {
69
+ gpio_pullup_dis(gpio_num_t(this->pin_->get_pin()));
70
+ }
71
+ error = rmt_enable(this->channel_);
72
+ if (error != ESP_OK) {
73
+ this->error_code_ = error;
74
+ this->error_string_ = "in rmt_enable";
75
+ this->mark_failed();
76
+ return;
77
+ }
78
+
79
+ rmt_rx_event_callbacks_t callbacks;
80
+ memset(&callbacks, 0, sizeof(callbacks));
81
+ callbacks.on_recv_done = rmt_callback;
82
+ error = rmt_rx_register_event_callbacks(this->channel_, &callbacks, &this->store_);
83
+ if (error != ESP_OK) {
84
+ this->error_code_ = error;
85
+ this->error_string_ = "in rmt_rx_register_event_callbacks";
86
+ this->mark_failed();
87
+ return;
88
+ }
89
+
90
+ uint32_t event_size = sizeof(rmt_rx_done_event_data_t);
91
+ uint32_t max_filter_ns = 255u * 1000 / (RMT_CLK_FREQ / 1000000);
92
+ uint32_t max_idle_ns = 65535u * 1000;
93
+ memset(&this->store_.config, 0, sizeof(this->store_.config));
94
+ this->store_.config.signal_range_min_ns = std::min(this->filter_us_ * 1000, max_filter_ns);
95
+ this->store_.config.signal_range_max_ns = std::min(this->idle_us_ * 1000, max_idle_ns);
96
+ this->store_.filter_symbols = this->filter_symbols_;
97
+ this->store_.receive_size = this->receive_symbols_ * sizeof(rmt_symbol_word_t);
98
+ this->store_.buffer_size = std::max((event_size + this->store_.receive_size) * 2, this->buffer_size_);
99
+ this->store_.buffer = new uint8_t[this->buffer_size_];
100
+ error = rmt_receive(this->channel_, (uint8_t *) this->store_.buffer + event_size, this->store_.receive_size,
101
+ &this->store_.config);
102
+ if (error != ESP_OK) {
103
+ this->error_code_ = error;
104
+ this->error_string_ = "in rmt_receive";
105
+ this->mark_failed();
106
+ return;
107
+ }
108
+ #else
14
109
  this->pin_->setup();
15
110
  rmt_config_t rmt{};
16
111
  this->config_rmt(rmt);
@@ -59,10 +154,18 @@ void RemoteReceiverComponent::setup() {
59
154
  this->mark_failed();
60
155
  return;
61
156
  }
157
+ #endif
62
158
  }
159
+
63
160
  void RemoteReceiverComponent::dump_config() {
64
161
  ESP_LOGCONFIG(TAG, "Remote Receiver:");
65
162
  LOG_PIN(" Pin: ", this->pin_);
163
+ #if ESP_IDF_VERSION_MAJOR >= 5
164
+ ESP_LOGCONFIG(TAG, " Clock resolution: %" PRIu32 " hz", this->clock_resolution_);
165
+ ESP_LOGCONFIG(TAG, " RMT symbols: %" PRIu32, this->rmt_symbols_);
166
+ ESP_LOGCONFIG(TAG, " Filter symbols: %" PRIu32, this->filter_symbols_);
167
+ ESP_LOGCONFIG(TAG, " Receive symbols: %" PRIu32, this->receive_symbols_);
168
+ #else
66
169
  if (this->pin_->digital_read()) {
67
170
  ESP_LOGW(TAG, "Remote Receiver Signal starts with a HIGH value. Usually this means you have to "
68
171
  "invert the signal using 'inverted: True' in the pin schema!");
@@ -70,6 +173,7 @@ void RemoteReceiverComponent::dump_config() {
70
173
  ESP_LOGCONFIG(TAG, " Channel: %d", this->channel_);
71
174
  ESP_LOGCONFIG(TAG, " RMT memory blocks: %d", this->mem_block_num_);
72
175
  ESP_LOGCONFIG(TAG, " Clock divider: %u", this->clock_divider_);
176
+ #endif
73
177
  ESP_LOGCONFIG(TAG, " Tolerance: %" PRIu32 "%s", this->tolerance_,
74
178
  (this->tolerance_mode_ == remote_base::TOLERANCE_MODE_TIME) ? " us" : "%");
75
179
  ESP_LOGCONFIG(TAG, " Filter out pulses shorter than: %" PRIu32 " us", this->filter_us_);
@@ -81,10 +185,38 @@ void RemoteReceiverComponent::dump_config() {
81
185
  }
82
186
 
83
187
  void RemoteReceiverComponent::loop() {
188
+ #if ESP_IDF_VERSION_MAJOR >= 5
189
+ if (this->store_.error != ESP_OK) {
190
+ ESP_LOGE(TAG, "Receive error");
191
+ this->error_code_ = this->store_.error;
192
+ this->error_string_ = "in rmt_callback";
193
+ this->mark_failed();
194
+ }
195
+ if (this->store_.overflow) {
196
+ ESP_LOGW(TAG, "Buffer overflow");
197
+ this->store_.overflow = false;
198
+ }
199
+ uint32_t buffer_write = this->store_.buffer_write;
200
+ while (this->store_.buffer_read != buffer_write) {
201
+ rmt_rx_done_event_data_t *event = (rmt_rx_done_event_data_t *) (this->store_.buffer + this->store_.buffer_read);
202
+ uint32_t event_size = sizeof(rmt_rx_done_event_data_t);
203
+ uint32_t next_read = this->store_.buffer_read + event_size + event->num_symbols * sizeof(rmt_symbol_word_t);
204
+ if (next_read + event_size + this->store_.receive_size > this->store_.buffer_size) {
205
+ next_read = 0;
206
+ }
207
+ this->decode_rmt_(event->received_symbols, event->num_symbols);
208
+ this->store_.buffer_read = next_read;
209
+
210
+ if (!this->temp_.empty()) {
211
+ this->temp_.push_back(-this->idle_us_);
212
+ this->call_listeners_dumpers_();
213
+ }
214
+ }
215
+ #else
84
216
  size_t len = 0;
85
217
  auto *item = (rmt_item32_t *) xRingbufferReceive(this->ringbuf_, &len, 0);
86
218
  if (item != nullptr) {
87
- this->decode_rmt_(item, len);
219
+ this->decode_rmt_(item, len / sizeof(rmt_item32_t));
88
220
  vRingbufferReturnItem(this->ringbuf_, item);
89
221
 
90
222
  if (this->temp_.empty())
@@ -93,13 +225,18 @@ void RemoteReceiverComponent::loop() {
93
225
  this->temp_.push_back(-this->idle_us_);
94
226
  this->call_listeners_dumpers_();
95
227
  }
228
+ #endif
96
229
  }
97
- void RemoteReceiverComponent::decode_rmt_(rmt_item32_t *item, size_t len) {
230
+
231
+ #if ESP_IDF_VERSION_MAJOR >= 5
232
+ void RemoteReceiverComponent::decode_rmt_(rmt_symbol_word_t *item, size_t item_count) {
233
+ #else
234
+ void RemoteReceiverComponent::decode_rmt_(rmt_item32_t *item, size_t item_count) {
235
+ #endif
98
236
  bool prev_level = false;
99
237
  uint32_t prev_length = 0;
100
238
  this->temp_.clear();
101
239
  int32_t multiplier = this->pin_->is_inverted() ? -1 : 1;
102
- size_t item_count = len / sizeof(rmt_item32_t);
103
240
  uint32_t filter_ticks = this->from_microseconds_(this->filter_us_);
104
241
 
105
242
  ESP_LOGVV(TAG, "START:");
@@ -124,7 +261,8 @@ void RemoteReceiverComponent::decode_rmt_(rmt_item32_t *item, size_t len) {
124
261
  this->temp_.reserve(item_count * 2); // each RMT item has 2 pulses
125
262
  for (size_t i = 0; i < item_count; i++) {
126
263
  if (item[i].duration0 == 0u) {
127
- // Do nothing
264
+ // EOF, sometimes garbage follows, break early
265
+ break;
128
266
  } else if ((bool(item[i].level0) == prev_level) || (item[i].duration0 < filter_ticks)) {
129
267
  prev_length += item[i].duration0;
130
268
  } else {
@@ -140,7 +278,8 @@ void RemoteReceiverComponent::decode_rmt_(rmt_item32_t *item, size_t len) {
140
278
  }
141
279
 
142
280
  if (item[i].duration1 == 0u) {
143
- // Do nothing
281
+ // EOF, sometimes garbage follows, break early
282
+ break;
144
283
  } else if ((bool(item[i].level1) == prev_level) || (item[i].duration1 < filter_ticks)) {
145
284
  prev_length += item[i].duration1;
146
285
  } else {
@@ -2,10 +2,24 @@ from esphome import automation, pins
2
2
  import esphome.codegen as cg
3
3
  from esphome.components import esp32_rmt, remote_base
4
4
  import esphome.config_validation as cv
5
- from esphome.const import CONF_CARRIER_DUTY_PERCENT, CONF_ID, CONF_PIN, CONF_RMT_CHANNEL
5
+ from esphome.const import (
6
+ CONF_CARRIER_DUTY_PERCENT,
7
+ CONF_CLOCK_DIVIDER,
8
+ CONF_CLOCK_RESOLUTION,
9
+ CONF_ID,
10
+ CONF_INVERTED,
11
+ CONF_MODE,
12
+ CONF_OPEN_DRAIN,
13
+ CONF_PIN,
14
+ CONF_RMT_CHANNEL,
15
+ CONF_RMT_SYMBOLS,
16
+ CONF_USE_DMA,
17
+ )
18
+ from esphome.core import CORE
6
19
 
7
20
  AUTO_LOAD = ["remote_base"]
8
21
 
22
+ CONF_EOT_LEVEL = "eot_level"
9
23
  CONF_ON_TRANSMIT = "on_transmit"
10
24
  CONF_ON_COMPLETE = "on_complete"
11
25
 
@@ -22,7 +36,28 @@ CONFIG_SCHEMA = cv.Schema(
22
36
  cv.Required(CONF_CARRIER_DUTY_PERCENT): cv.All(
23
37
  cv.percentage_int, cv.Range(min=1, max=100)
24
38
  ),
25
- cv.Optional(CONF_RMT_CHANNEL): esp32_rmt.validate_rmt_channel(tx=True),
39
+ cv.Optional(CONF_CLOCK_RESOLUTION): cv.All(
40
+ cv.only_on_esp32,
41
+ cv.only_with_esp_idf,
42
+ esp32_rmt.validate_clock_resolution(),
43
+ ),
44
+ cv.Optional(CONF_CLOCK_DIVIDER): cv.All(
45
+ cv.only_on_esp32, cv.only_with_arduino, cv.int_range(min=1, max=255)
46
+ ),
47
+ cv.Optional(CONF_EOT_LEVEL): cv.All(cv.only_with_esp_idf, cv.boolean),
48
+ cv.Optional(CONF_USE_DMA): cv.All(cv.only_with_esp_idf, cv.boolean),
49
+ cv.SplitDefault(
50
+ CONF_RMT_SYMBOLS,
51
+ esp32_idf=64,
52
+ esp32_s2_idf=64,
53
+ esp32_s3_idf=48,
54
+ esp32_c3_idf=48,
55
+ esp32_c6_idf=48,
56
+ esp32_h2_idf=48,
57
+ ): cv.All(cv.only_with_esp_idf, cv.int_range(min=2)),
58
+ cv.Optional(CONF_RMT_CHANNEL): cv.All(
59
+ cv.only_with_arduino, esp32_rmt.validate_rmt_channel(tx=True)
60
+ ),
26
61
  cv.Optional(CONF_ON_TRANSMIT): automation.validate_automation(single=True),
27
62
  cv.Optional(CONF_ON_COMPLETE): automation.validate_automation(single=True),
28
63
  }
@@ -31,8 +66,31 @@ CONFIG_SCHEMA = cv.Schema(
31
66
 
32
67
  async def to_code(config):
33
68
  pin = await cg.gpio_pin_expression(config[CONF_PIN])
34
- if (rmt_channel := config.get(CONF_RMT_CHANNEL, None)) is not None:
35
- var = cg.new_Pvariable(config[CONF_ID], pin, rmt_channel)
69
+ if CORE.is_esp32:
70
+ if esp32_rmt.use_new_rmt_driver():
71
+ var = cg.new_Pvariable(config[CONF_ID], pin)
72
+ cg.add(var.set_rmt_symbols(config[CONF_RMT_SYMBOLS]))
73
+ if CONF_CLOCK_RESOLUTION in config:
74
+ cg.add(var.set_clock_resolution(config[CONF_CLOCK_RESOLUTION]))
75
+ if CONF_USE_DMA in config:
76
+ cg.add(var.set_with_dma(config[CONF_USE_DMA]))
77
+ if CONF_EOT_LEVEL in config:
78
+ cg.add(var.set_eot_level(config[CONF_EOT_LEVEL]))
79
+ else:
80
+ cg.add(
81
+ var.set_eot_level(
82
+ config[CONF_PIN][CONF_MODE][CONF_OPEN_DRAIN]
83
+ or config[CONF_PIN][CONF_INVERTED]
84
+ )
85
+ )
86
+ else:
87
+ if (rmt_channel := config.get(CONF_RMT_CHANNEL, None)) is not None:
88
+ var = cg.new_Pvariable(config[CONF_ID], pin, rmt_channel)
89
+ else:
90
+ var = cg.new_Pvariable(config[CONF_ID], pin)
91
+ if CONF_CLOCK_DIVIDER in config:
92
+ cg.add(var.set_clock_divider(config[CONF_CLOCK_DIVIDER]))
93
+
36
94
  else:
37
95
  var = cg.new_Pvariable(config[CONF_ID], pin)
38
96
  await cg.register_component(var, config)
@@ -5,6 +5,10 @@
5
5
 
6
6
  #include <vector>
7
7
 
8
+ #if defined(USE_ESP32) && ESP_IDF_VERSION_MAJOR >= 5
9
+ #include <driver/rmt_tx.h>
10
+ #endif
11
+
8
12
  namespace esphome {
9
13
  namespace remote_transmitter {
10
14
 
@@ -16,7 +20,7 @@ class RemoteTransmitterComponent : public remote_base::RemoteTransmitterBase,
16
20
  #endif
17
21
  {
18
22
  public:
19
- #ifdef USE_ESP32
23
+ #if defined(USE_ESP32) && ESP_IDF_VERSION_MAJOR < 5
20
24
  RemoteTransmitterComponent(InternalGPIOPin *pin, uint8_t mem_block_num = 1)
21
25
  : remote_base::RemoteTransmitterBase(pin), remote_base::RemoteRMTChannel(mem_block_num) {}
22
26
 
@@ -29,10 +33,17 @@ class RemoteTransmitterComponent : public remote_base::RemoteTransmitterBase,
29
33
 
30
34
  void dump_config() override;
31
35
 
32
- float get_setup_priority() const override { return setup_priority::DATA; }
36
+ // transmitter setup must run after receiver setup to allow the same GPIO to be used by both
37
+ float get_setup_priority() const override { return setup_priority::DATA - 1; }
33
38
 
34
39
  void set_carrier_duty_percent(uint8_t carrier_duty_percent) { this->carrier_duty_percent_ = carrier_duty_percent; }
35
40
 
41
+ #if defined(USE_ESP32) && ESP_IDF_VERSION_MAJOR >= 5
42
+ void set_with_dma(bool with_dma) { this->with_dma_ = with_dma; }
43
+ void set_eot_level(bool eot_level) { this->eot_level_ = eot_level; }
44
+ void digital_write(bool value);
45
+ #endif
46
+
36
47
  Trigger<> *get_transmit_trigger() const { return this->transmit_trigger_; };
37
48
  Trigger<> *get_complete_trigger() const { return this->complete_trigger_; };
38
49
 
@@ -54,7 +65,15 @@ class RemoteTransmitterComponent : public remote_base::RemoteTransmitterBase,
54
65
 
55
66
  uint32_t current_carrier_frequency_{38000};
56
67
  bool initialized_{false};
68
+ #if ESP_IDF_VERSION_MAJOR >= 5
69
+ std::vector<rmt_symbol_word_t> rmt_temp_;
70
+ bool with_dma_{false};
71
+ bool eot_level_{false};
72
+ rmt_channel_handle_t channel_{NULL};
73
+ rmt_encoder_handle_t encoder_{NULL};
74
+ #else
57
75
  std::vector<rmt_item32_t> rmt_temp_;
76
+ #endif
58
77
  esp_err_t error_code_{ESP_OK};
59
78
  std::string error_string_{""};
60
79
  bool inverted_{false};