mirror of
https://github.com/esphome/esphome.git
synced 2025-09-12 16:22:22 +01:00
Merge branch 'dev' into multi_device
This commit is contained in:
@@ -1,2 +1,4 @@
|
|||||||
[run]
|
[run]
|
||||||
omit = esphome/components/*
|
omit =
|
||||||
|
esphome/components/*
|
||||||
|
tests/integration/*
|
||||||
|
4
.github/actions/build-image/action.yaml
vendored
4
.github/actions/build-image/action.yaml
vendored
@@ -47,7 +47,7 @@ runs:
|
|||||||
|
|
||||||
- name: Build and push to ghcr by digest
|
- name: Build and push to ghcr by digest
|
||||||
id: build-ghcr
|
id: build-ghcr
|
||||||
uses: docker/build-push-action@v6.17.0
|
uses: docker/build-push-action@v6.18.0
|
||||||
env:
|
env:
|
||||||
DOCKER_BUILD_SUMMARY: false
|
DOCKER_BUILD_SUMMARY: false
|
||||||
DOCKER_BUILD_RECORD_UPLOAD: false
|
DOCKER_BUILD_RECORD_UPLOAD: false
|
||||||
@@ -73,7 +73,7 @@ runs:
|
|||||||
|
|
||||||
- name: Build and push to dockerhub by digest
|
- name: Build and push to dockerhub by digest
|
||||||
id: build-dockerhub
|
id: build-dockerhub
|
||||||
uses: docker/build-push-action@v6.17.0
|
uses: docker/build-push-action@v6.18.0
|
||||||
env:
|
env:
|
||||||
DOCKER_BUILD_SUMMARY: false
|
DOCKER_BUILD_SUMMARY: false
|
||||||
DOCKER_BUILD_RECORD_UPLOAD: false
|
DOCKER_BUILD_RECORD_UPLOAD: false
|
||||||
|
2
.github/workflows/ci-api-proto.yml
vendored
2
.github/workflows/ci-api-proto.yml
vendored
@@ -21,7 +21,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Set up Python
|
- name: Set up Python
|
||||||
uses: actions/setup-python@v5.6.0
|
uses: actions/setup-python@v5.6.0
|
||||||
with:
|
with:
|
||||||
|
2
.github/workflows/ci-docker.yml
vendored
2
.github/workflows/ci-docker.yml
vendored
@@ -43,7 +43,7 @@ jobs:
|
|||||||
- "docker"
|
- "docker"
|
||||||
# - "lint"
|
# - "lint"
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4.1.7
|
- uses: actions/checkout@v4.2.2
|
||||||
- name: Set up Python
|
- name: Set up Python
|
||||||
uses: actions/setup-python@v5.6.0
|
uses: actions/setup-python@v5.6.0
|
||||||
with:
|
with:
|
||||||
|
30
.github/workflows/ci.yml
vendored
30
.github/workflows/ci.yml
vendored
@@ -36,7 +36,7 @@ jobs:
|
|||||||
cache-key: ${{ steps.cache-key.outputs.key }}
|
cache-key: ${{ steps.cache-key.outputs.key }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Generate cache-key
|
- name: Generate cache-key
|
||||||
id: cache-key
|
id: cache-key
|
||||||
run: echo key="${{ hashFiles('requirements.txt', 'requirements_test.txt') }}" >> $GITHUB_OUTPUT
|
run: echo key="${{ hashFiles('requirements.txt', 'requirements_test.txt') }}" >> $GITHUB_OUTPUT
|
||||||
@@ -68,7 +68,7 @@ jobs:
|
|||||||
- common
|
- common
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Restore Python
|
- name: Restore Python
|
||||||
uses: ./.github/actions/restore-python
|
uses: ./.github/actions/restore-python
|
||||||
with:
|
with:
|
||||||
@@ -89,7 +89,7 @@ jobs:
|
|||||||
- common
|
- common
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Restore Python
|
- name: Restore Python
|
||||||
uses: ./.github/actions/restore-python
|
uses: ./.github/actions/restore-python
|
||||||
with:
|
with:
|
||||||
@@ -110,7 +110,7 @@ jobs:
|
|||||||
- common
|
- common
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Restore Python
|
- name: Restore Python
|
||||||
uses: ./.github/actions/restore-python
|
uses: ./.github/actions/restore-python
|
||||||
with:
|
with:
|
||||||
@@ -131,7 +131,7 @@ jobs:
|
|||||||
- common
|
- common
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Restore Python
|
- name: Restore Python
|
||||||
uses: ./.github/actions/restore-python
|
uses: ./.github/actions/restore-python
|
||||||
with:
|
with:
|
||||||
@@ -152,7 +152,7 @@ jobs:
|
|||||||
- common
|
- common
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Restore Python
|
- name: Restore Python
|
||||||
uses: ./.github/actions/restore-python
|
uses: ./.github/actions/restore-python
|
||||||
with:
|
with:
|
||||||
@@ -202,7 +202,7 @@ jobs:
|
|||||||
- common
|
- common
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Restore Python
|
- name: Restore Python
|
||||||
uses: ./.github/actions/restore-python
|
uses: ./.github/actions/restore-python
|
||||||
with:
|
with:
|
||||||
@@ -214,12 +214,12 @@ jobs:
|
|||||||
if: matrix.os == 'windows-latest'
|
if: matrix.os == 'windows-latest'
|
||||||
run: |
|
run: |
|
||||||
./venv/Scripts/activate
|
./venv/Scripts/activate
|
||||||
pytest -vv --cov-report=xml --tb=native tests
|
pytest -vv --cov-report=xml --tb=native -n auto tests
|
||||||
- name: Run pytest
|
- name: Run pytest
|
||||||
if: matrix.os == 'ubuntu-latest' || matrix.os == 'macOS-latest'
|
if: matrix.os == 'ubuntu-latest' || matrix.os == 'macOS-latest'
|
||||||
run: |
|
run: |
|
||||||
. venv/bin/activate
|
. venv/bin/activate
|
||||||
pytest -vv --cov-report=xml --tb=native tests
|
pytest -vv --cov-report=xml --tb=native -n auto tests
|
||||||
- name: Upload coverage to Codecov
|
- name: Upload coverage to Codecov
|
||||||
uses: codecov/codecov-action@v5.4.3
|
uses: codecov/codecov-action@v5.4.3
|
||||||
with:
|
with:
|
||||||
@@ -232,7 +232,7 @@ jobs:
|
|||||||
- common
|
- common
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Restore Python
|
- name: Restore Python
|
||||||
uses: ./.github/actions/restore-python
|
uses: ./.github/actions/restore-python
|
||||||
with:
|
with:
|
||||||
@@ -300,7 +300,7 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Restore Python
|
- name: Restore Python
|
||||||
uses: ./.github/actions/restore-python
|
uses: ./.github/actions/restore-python
|
||||||
with:
|
with:
|
||||||
@@ -356,7 +356,7 @@ jobs:
|
|||||||
count: ${{ steps.list-components.outputs.count }}
|
count: ${{ steps.list-components.outputs.count }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
with:
|
with:
|
||||||
# Fetch enough history so `git merge-base refs/remotes/origin/dev HEAD` works.
|
# Fetch enough history so `git merge-base refs/remotes/origin/dev HEAD` works.
|
||||||
fetch-depth: 500
|
fetch-depth: 500
|
||||||
@@ -406,7 +406,7 @@ jobs:
|
|||||||
sudo apt-get install libsdl2-dev
|
sudo apt-get install libsdl2-dev
|
||||||
|
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Restore Python
|
- name: Restore Python
|
||||||
uses: ./.github/actions/restore-python
|
uses: ./.github/actions/restore-python
|
||||||
with:
|
with:
|
||||||
@@ -432,7 +432,7 @@ jobs:
|
|||||||
matrix: ${{ steps.split.outputs.components }}
|
matrix: ${{ steps.split.outputs.components }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Split components into 20 groups
|
- name: Split components into 20 groups
|
||||||
id: split
|
id: split
|
||||||
run: |
|
run: |
|
||||||
@@ -462,7 +462,7 @@ jobs:
|
|||||||
sudo apt-get install libsdl2-dev
|
sudo apt-get install libsdl2-dev
|
||||||
|
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Restore Python
|
- name: Restore Python
|
||||||
uses: ./.github/actions/restore-python
|
uses: ./.github/actions/restore-python
|
||||||
with:
|
with:
|
||||||
|
8
.github/workflows/release.yml
vendored
8
.github/workflows/release.yml
vendored
@@ -20,7 +20,7 @@ jobs:
|
|||||||
branch_build: ${{ steps.tag.outputs.branch_build }}
|
branch_build: ${{ steps.tag.outputs.branch_build }}
|
||||||
deploy_env: ${{ steps.tag.outputs.deploy_env }}
|
deploy_env: ${{ steps.tag.outputs.deploy_env }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4.1.7
|
- uses: actions/checkout@v4.2.2
|
||||||
- name: Get tag
|
- name: Get tag
|
||||||
id: tag
|
id: tag
|
||||||
# yamllint disable rule:line-length
|
# yamllint disable rule:line-length
|
||||||
@@ -60,7 +60,7 @@ jobs:
|
|||||||
contents: read
|
contents: read
|
||||||
id-token: write
|
id-token: write
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4.1.7
|
- uses: actions/checkout@v4.2.2
|
||||||
- name: Set up Python
|
- name: Set up Python
|
||||||
uses: actions/setup-python@v5.6.0
|
uses: actions/setup-python@v5.6.0
|
||||||
with:
|
with:
|
||||||
@@ -92,7 +92,7 @@ jobs:
|
|||||||
os: "ubuntu-24.04-arm"
|
os: "ubuntu-24.04-arm"
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4.1.7
|
- uses: actions/checkout@v4.2.2
|
||||||
- name: Set up Python
|
- name: Set up Python
|
||||||
uses: actions/setup-python@v5.6.0
|
uses: actions/setup-python@v5.6.0
|
||||||
with:
|
with:
|
||||||
@@ -168,7 +168,7 @@ jobs:
|
|||||||
- ghcr
|
- ghcr
|
||||||
- dockerhub
|
- dockerhub
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4.1.7
|
- uses: actions/checkout@v4.2.2
|
||||||
|
|
||||||
- name: Download digests
|
- name: Download digests
|
||||||
uses: actions/download-artifact@v4.3.0
|
uses: actions/download-artifact@v4.3.0
|
||||||
|
2
.github/workflows/yaml-lint.yml
vendored
2
.github/workflows/yaml-lint.yml
vendored
@@ -18,7 +18,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out code from GitHub
|
- name: Check out code from GitHub
|
||||||
uses: actions/checkout@v4.1.7
|
uses: actions/checkout@v4.2.2
|
||||||
- name: Run yamllint
|
- name: Run yamllint
|
||||||
uses: frenck/action-yamllint@v1.5.0
|
uses: frenck/action-yamllint@v1.5.0
|
||||||
with:
|
with:
|
||||||
|
@@ -28,7 +28,7 @@ repos:
|
|||||||
- --branch=release
|
- --branch=release
|
||||||
- --branch=beta
|
- --branch=beta
|
||||||
- repo: https://github.com/asottile/pyupgrade
|
- repo: https://github.com/asottile/pyupgrade
|
||||||
rev: v3.19.1
|
rev: v3.20.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: pyupgrade
|
- id: pyupgrade
|
||||||
args: [--py310-plus]
|
args: [--py310-plus]
|
||||||
|
@@ -139,6 +139,7 @@ esphome/components/es7210/* @kahrendt
|
|||||||
esphome/components/es7243e/* @kbx81
|
esphome/components/es7243e/* @kbx81
|
||||||
esphome/components/es8156/* @kbx81
|
esphome/components/es8156/* @kbx81
|
||||||
esphome/components/es8311/* @kahrendt @kroimon
|
esphome/components/es8311/* @kahrendt @kroimon
|
||||||
|
esphome/components/es8388/* @P4uLT
|
||||||
esphome/components/esp32/* @esphome/core
|
esphome/components/esp32/* @esphome/core
|
||||||
esphome/components/esp32_ble/* @Rapsssito @jesserockz
|
esphome/components/esp32_ble/* @Rapsssito @jesserockz
|
||||||
esphome/components/esp32_ble_client/* @jesserockz
|
esphome/components/esp32_ble_client/* @jesserockz
|
||||||
|
@@ -593,15 +593,20 @@ def command_update_all(args):
|
|||||||
middle_text = f" {middle_text} "
|
middle_text = f" {middle_text} "
|
||||||
width = len(click.unstyle(middle_text))
|
width = len(click.unstyle(middle_text))
|
||||||
half_line = "=" * ((twidth - width) // 2)
|
half_line = "=" * ((twidth - width) // 2)
|
||||||
click.echo(f"{half_line}{middle_text}{half_line}")
|
safe_print(f"{half_line}{middle_text}{half_line}")
|
||||||
|
|
||||||
for f in files:
|
for f in files:
|
||||||
print(f"Updating {color(AnsiFore.CYAN, f)}")
|
safe_print(f"Updating {color(AnsiFore.CYAN, f)}")
|
||||||
print("-" * twidth)
|
safe_print("-" * twidth)
|
||||||
print()
|
safe_print()
|
||||||
|
if CORE.dashboard:
|
||||||
rc = run_external_process(
|
rc = run_external_process(
|
||||||
"esphome", "--dashboard", "run", f, "--no-logs", "--device", "OTA"
|
"esphome", "--dashboard", "run", f, "--no-logs", "--device", "OTA"
|
||||||
)
|
)
|
||||||
|
else:
|
||||||
|
rc = run_external_process(
|
||||||
|
"esphome", "run", f, "--no-logs", "--device", "OTA"
|
||||||
|
)
|
||||||
if rc == 0:
|
if rc == 0:
|
||||||
print_bar(f"[{color(AnsiFore.BOLD_GREEN, 'SUCCESS')}] {f}")
|
print_bar(f"[{color(AnsiFore.BOLD_GREEN, 'SUCCESS')}] {f}")
|
||||||
success[f] = True
|
success[f] = True
|
||||||
@@ -609,17 +614,17 @@ def command_update_all(args):
|
|||||||
print_bar(f"[{color(AnsiFore.BOLD_RED, 'ERROR')}] {f}")
|
print_bar(f"[{color(AnsiFore.BOLD_RED, 'ERROR')}] {f}")
|
||||||
success[f] = False
|
success[f] = False
|
||||||
|
|
||||||
print()
|
safe_print()
|
||||||
print()
|
safe_print()
|
||||||
print()
|
safe_print()
|
||||||
|
|
||||||
print_bar(f"[{color(AnsiFore.BOLD_WHITE, 'SUMMARY')}]")
|
print_bar(f"[{color(AnsiFore.BOLD_WHITE, 'SUMMARY')}]")
|
||||||
failed = 0
|
failed = 0
|
||||||
for f in files:
|
for f in files:
|
||||||
if success[f]:
|
if success[f]:
|
||||||
print(f" - {f}: {color(AnsiFore.GREEN, 'SUCCESS')}")
|
safe_print(f" - {f}: {color(AnsiFore.GREEN, 'SUCCESS')}")
|
||||||
else:
|
else:
|
||||||
print(f" - {f}: {color(AnsiFore.BOLD_RED, 'FAILED')}")
|
safe_print(f" - {f}: {color(AnsiFore.BOLD_RED, 'FAILED')}")
|
||||||
failed += 1
|
failed += 1
|
||||||
return failed
|
return failed
|
||||||
|
|
||||||
|
@@ -7,7 +7,7 @@ namespace a4988 {
|
|||||||
static const char *const TAG = "a4988.stepper";
|
static const char *const TAG = "a4988.stepper";
|
||||||
|
|
||||||
void A4988::setup() {
|
void A4988::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up A4988...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
if (this->sleep_pin_ != nullptr) {
|
if (this->sleep_pin_ != nullptr) {
|
||||||
this->sleep_pin_->setup();
|
this->sleep_pin_->setup();
|
||||||
this->sleep_pin_->digital_write(false);
|
this->sleep_pin_->digital_write(false);
|
||||||
|
@@ -7,7 +7,7 @@ namespace absolute_humidity {
|
|||||||
static const char *const TAG = "absolute_humidity.sensor";
|
static const char *const TAG = "absolute_humidity.sensor";
|
||||||
|
|
||||||
void AbsoluteHumidityComponent::setup() {
|
void AbsoluteHumidityComponent::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up absolute humidity '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'", this->get_name().c_str());
|
||||||
|
|
||||||
ESP_LOGD(TAG, " Added callback for temperature '%s'", this->temperature_sensor_->get_name().c_str());
|
ESP_LOGD(TAG, " Added callback for temperature '%s'", this->temperature_sensor_->get_name().c_str());
|
||||||
this->temperature_sensor_->add_on_state_callback([this](float state) { this->temperature_callback_(state); });
|
this->temperature_sensor_->add_on_state_callback([this](float state) { this->temperature_callback_(state); });
|
||||||
|
@@ -22,7 +22,7 @@ static const int ADC_MAX = (1 << SOC_ADC_RTC_MAX_BITWIDTH) - 1;
|
|||||||
static const int ADC_HALF = (1 << SOC_ADC_RTC_MAX_BITWIDTH) >> 1;
|
static const int ADC_HALF = (1 << SOC_ADC_RTC_MAX_BITWIDTH) >> 1;
|
||||||
|
|
||||||
void ADCSensor::setup() {
|
void ADCSensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'...", this->get_name().c_str());
|
||||||
|
|
||||||
if (this->channel1_ != ADC1_CHANNEL_MAX) {
|
if (this->channel1_ != ADC1_CHANNEL_MAX) {
|
||||||
adc1_config_width(ADC_WIDTH_MAX_SOC_BITS);
|
adc1_config_width(ADC_WIDTH_MAX_SOC_BITS);
|
||||||
|
@@ -17,7 +17,7 @@ namespace adc {
|
|||||||
static const char *const TAG = "adc.esp8266";
|
static const char *const TAG = "adc.esp8266";
|
||||||
|
|
||||||
void ADCSensor::setup() {
|
void ADCSensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'...", this->get_name().c_str());
|
||||||
#ifndef USE_ADC_SENSOR_VCC
|
#ifndef USE_ADC_SENSOR_VCC
|
||||||
this->pin_->setup();
|
this->pin_->setup();
|
||||||
#endif
|
#endif
|
||||||
|
@@ -9,7 +9,7 @@ namespace adc {
|
|||||||
static const char *const TAG = "adc.libretiny";
|
static const char *const TAG = "adc.libretiny";
|
||||||
|
|
||||||
void ADCSensor::setup() {
|
void ADCSensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'...", this->get_name().c_str());
|
||||||
#ifndef USE_ADC_SENSOR_VCC
|
#ifndef USE_ADC_SENSOR_VCC
|
||||||
this->pin_->setup();
|
this->pin_->setup();
|
||||||
#endif // !USE_ADC_SENSOR_VCC
|
#endif // !USE_ADC_SENSOR_VCC
|
||||||
|
@@ -14,7 +14,7 @@ namespace adc {
|
|||||||
static const char *const TAG = "adc.rp2040";
|
static const char *const TAG = "adc.rp2040";
|
||||||
|
|
||||||
void ADCSensor::setup() {
|
void ADCSensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADC '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'...", this->get_name().c_str());
|
||||||
static bool initialized = false;
|
static bool initialized = false;
|
||||||
if (!initialized) {
|
if (!initialized) {
|
||||||
adc_init();
|
adc_init();
|
||||||
|
@@ -9,7 +9,7 @@ static const char *const TAG = "adc128s102";
|
|||||||
float ADC128S102::get_setup_priority() const { return setup_priority::HARDWARE; }
|
float ADC128S102::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
|
|
||||||
void ADC128S102::setup() {
|
void ADC128S102::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up adc128s102");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->spi_setup();
|
this->spi_setup();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -10,15 +10,13 @@ static const uint8_t ADS1115_REGISTER_CONVERSION = 0x00;
|
|||||||
static const uint8_t ADS1115_REGISTER_CONFIG = 0x01;
|
static const uint8_t ADS1115_REGISTER_CONFIG = 0x01;
|
||||||
|
|
||||||
void ADS1115Component::setup() {
|
void ADS1115Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADS1115...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint16_t value;
|
uint16_t value;
|
||||||
if (!this->read_byte_16(ADS1115_REGISTER_CONVERSION, &value)) {
|
if (!this->read_byte_16(ADS1115_REGISTER_CONVERSION, &value)) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGCONFIG(TAG, "Configuring ADS1115...");
|
|
||||||
|
|
||||||
uint16_t config = 0;
|
uint16_t config = 0;
|
||||||
// Clear single-shot bit
|
// Clear single-shot bit
|
||||||
// 0b0xxxxxxxxxxxxxxx
|
// 0b0xxxxxxxxxxxxxxx
|
||||||
@@ -68,10 +66,10 @@ void ADS1115Component::setup() {
|
|||||||
this->prev_config_ = config;
|
this->prev_config_ = config;
|
||||||
}
|
}
|
||||||
void ADS1115Component::dump_config() {
|
void ADS1115Component::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ADS1115...");
|
ESP_LOGCONFIG(TAG, "ADS1115:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with ADS1115 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
float ADS1115Component::request_measurement(ADS1115Multiplexer multiplexer, ADS1115Gain gain,
|
float ADS1115Component::request_measurement(ADS1115Multiplexer multiplexer, ADS1115Gain gain,
|
||||||
|
@@ -8,7 +8,7 @@ static const char *const TAG = "ads1118";
|
|||||||
static const uint8_t ADS1118_DATA_RATE_860_SPS = 0b111;
|
static const uint8_t ADS1118_DATA_RATE_860_SPS = 0b111;
|
||||||
|
|
||||||
void ADS1118::setup() {
|
void ADS1118::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ads1118");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->spi_setup();
|
this->spi_setup();
|
||||||
|
|
||||||
this->config_ = 0;
|
this->config_ = 0;
|
||||||
|
@@ -23,7 +23,7 @@ static const uint16_t ZP_CURRENT = 0x0000;
|
|||||||
static const uint16_t ZP_DEFAULT = 0xFFFF;
|
static const uint16_t ZP_DEFAULT = 0xFFFF;
|
||||||
|
|
||||||
void AGS10Component::setup() {
|
void AGS10Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ags10...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
auto version = this->read_version_();
|
auto version = this->read_version_();
|
||||||
if (version) {
|
if (version) {
|
||||||
@@ -65,7 +65,7 @@ void AGS10Component::dump_config() {
|
|||||||
case NONE:
|
case NONE:
|
||||||
break;
|
break;
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication with AGS10 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case CRC_CHECK_FAILED:
|
case CRC_CHECK_FAILED:
|
||||||
ESP_LOGE(TAG, "The crc check failed");
|
ESP_LOGE(TAG, "The crc check failed");
|
||||||
|
@@ -15,6 +15,7 @@
|
|||||||
#include "aht10.h"
|
#include "aht10.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
#include "esphome/core/hal.h"
|
#include "esphome/core/hal.h"
|
||||||
|
#include "esphome/core/helpers.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace aht10 {
|
namespace aht10 {
|
||||||
@@ -34,57 +35,59 @@ static const uint8_t AHT10_INIT_ATTEMPTS = 10;
|
|||||||
|
|
||||||
static const uint8_t AHT10_STATUS_BUSY = 0x80;
|
static const uint8_t AHT10_STATUS_BUSY = 0x80;
|
||||||
|
|
||||||
|
static const float AHT10_DIVISOR = 1048576.0f; // 2^20, used for temperature and humidity calculations
|
||||||
|
|
||||||
void AHT10Component::setup() {
|
void AHT10Component::setup() {
|
||||||
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
if (this->write(AHT10_SOFTRESET_CMD, sizeof(AHT10_SOFTRESET_CMD)) != i2c::ERROR_OK) {
|
if (this->write(AHT10_SOFTRESET_CMD, sizeof(AHT10_SOFTRESET_CMD)) != i2c::ERROR_OK) {
|
||||||
ESP_LOGE(TAG, "Reset AHT10 failed!");
|
ESP_LOGE(TAG, "Reset failed");
|
||||||
}
|
}
|
||||||
delay(AHT10_SOFTRESET_DELAY);
|
delay(AHT10_SOFTRESET_DELAY);
|
||||||
|
|
||||||
i2c::ErrorCode error_code = i2c::ERROR_INVALID_ARGUMENT;
|
i2c::ErrorCode error_code = i2c::ERROR_INVALID_ARGUMENT;
|
||||||
switch (this->variant_) {
|
switch (this->variant_) {
|
||||||
case AHT10Variant::AHT20:
|
case AHT10Variant::AHT20:
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AHT20");
|
|
||||||
error_code = this->write(AHT20_INITIALIZE_CMD, sizeof(AHT20_INITIALIZE_CMD));
|
error_code = this->write(AHT20_INITIALIZE_CMD, sizeof(AHT20_INITIALIZE_CMD));
|
||||||
break;
|
break;
|
||||||
case AHT10Variant::AHT10:
|
case AHT10Variant::AHT10:
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AHT10");
|
|
||||||
error_code = this->write(AHT10_INITIALIZE_CMD, sizeof(AHT10_INITIALIZE_CMD));
|
error_code = this->write(AHT10_INITIALIZE_CMD, sizeof(AHT10_INITIALIZE_CMD));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (error_code != i2c::ERROR_OK) {
|
if (error_code != i2c::ERROR_OK) {
|
||||||
ESP_LOGE(TAG, "Communication with AHT10 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
uint8_t cal_attempts = 0;
|
||||||
uint8_t data = AHT10_STATUS_BUSY;
|
uint8_t data = AHT10_STATUS_BUSY;
|
||||||
int cal_attempts = 0;
|
|
||||||
while (data & AHT10_STATUS_BUSY) {
|
while (data & AHT10_STATUS_BUSY) {
|
||||||
delay(AHT10_DEFAULT_DELAY);
|
delay(AHT10_DEFAULT_DELAY);
|
||||||
if (this->read(&data, 1) != i2c::ERROR_OK) {
|
if (this->read(&data, 1) != i2c::ERROR_OK) {
|
||||||
ESP_LOGE(TAG, "Communication with AHT10 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
++cal_attempts;
|
++cal_attempts;
|
||||||
if (cal_attempts > AHT10_INIT_ATTEMPTS) {
|
if (cal_attempts > AHT10_INIT_ATTEMPTS) {
|
||||||
ESP_LOGE(TAG, "AHT10 initialization timed out!");
|
ESP_LOGE(TAG, "Initialization timed out");
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if ((data & 0x68) != 0x08) { // Bit[6:5] = 0b00, NORMAL mode and Bit[3] = 0b1, CALIBRATED
|
if ((data & 0x68) != 0x08) { // Bit[6:5] = 0b00, NORMAL mode and Bit[3] = 0b1, CALIBRATED
|
||||||
ESP_LOGE(TAG, "AHT10 initialization failed!");
|
ESP_LOGE(TAG, "Initialization failed");
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGV(TAG, "AHT10 initialization");
|
ESP_LOGV(TAG, "Initialization complete");
|
||||||
}
|
}
|
||||||
|
|
||||||
void AHT10Component::restart_read_() {
|
void AHT10Component::restart_read_() {
|
||||||
if (this->read_count_ == AHT10_ATTEMPTS) {
|
if (this->read_count_ == AHT10_ATTEMPTS) {
|
||||||
this->read_count_ = 0;
|
this->read_count_ = 0;
|
||||||
this->status_set_error("Measurements reading timed-out!");
|
this->status_set_error("Reading timed out");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this->read_count_++;
|
this->read_count_++;
|
||||||
@@ -97,24 +100,24 @@ void AHT10Component::read_data_() {
|
|||||||
ESP_LOGD(TAG, "Read attempt %d at %ums", this->read_count_, (unsigned) (millis() - this->start_time_));
|
ESP_LOGD(TAG, "Read attempt %d at %ums", this->read_count_, (unsigned) (millis() - this->start_time_));
|
||||||
}
|
}
|
||||||
if (this->read(data, 6) != i2c::ERROR_OK) {
|
if (this->read(data, 6) != i2c::ERROR_OK) {
|
||||||
this->status_set_warning("AHT10 read failed, retrying soon");
|
this->status_set_warning("Read failed, will retry");
|
||||||
this->restart_read_();
|
this->restart_read_();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((data[0] & 0x80) == 0x80) { // Bit[7] = 0b1, device is busy
|
if ((data[0] & 0x80) == 0x80) { // Bit[7] = 0b1, device is busy
|
||||||
ESP_LOGD(TAG, "AHT10 is busy, waiting...");
|
ESP_LOGD(TAG, "Device busy, will retry");
|
||||||
this->restart_read_();
|
this->restart_read_();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (data[1] == 0x0 && data[2] == 0x0 && (data[3] >> 4) == 0x0) {
|
if (data[1] == 0x0 && data[2] == 0x0 && (data[3] >> 4) == 0x0) {
|
||||||
// Unrealistic humidity (0x0)
|
// Invalid humidity (0x0)
|
||||||
if (this->humidity_sensor_ == nullptr) {
|
if (this->humidity_sensor_ == nullptr) {
|
||||||
ESP_LOGV(TAG, "ATH10 Unrealistic humidity (0x0), but humidity is not required");
|
ESP_LOGV(TAG, "Invalid humidity (reading not required)");
|
||||||
} else {
|
} else {
|
||||||
ESP_LOGD(TAG, "ATH10 Unrealistic humidity (0x0), retrying...");
|
ESP_LOGD(TAG, "Invalid humidity, retrying...");
|
||||||
if (this->write(AHT10_MEASURE_CMD, sizeof(AHT10_MEASURE_CMD)) != i2c::ERROR_OK) {
|
if (this->write(AHT10_MEASURE_CMD, sizeof(AHT10_MEASURE_CMD)) != i2c::ERROR_OK) {
|
||||||
this->status_set_warning("Communication with AHT10 failed!");
|
this->status_set_warning(ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
this->restart_read_();
|
this->restart_read_();
|
||||||
return;
|
return;
|
||||||
@@ -123,22 +126,17 @@ void AHT10Component::read_data_() {
|
|||||||
if (this->read_count_ > 1) {
|
if (this->read_count_ > 1) {
|
||||||
ESP_LOGD(TAG, "Success at %ums", (unsigned) (millis() - this->start_time_));
|
ESP_LOGD(TAG, "Success at %ums", (unsigned) (millis() - this->start_time_));
|
||||||
}
|
}
|
||||||
uint32_t raw_temperature = ((data[3] & 0x0F) << 16) | (data[4] << 8) | data[5];
|
uint32_t raw_temperature = encode_uint24(data[3] & 0xF, data[4], data[5]);
|
||||||
uint32_t raw_humidity = ((data[1] << 16) | (data[2] << 8) | data[3]) >> 4;
|
uint32_t raw_humidity = encode_uint24(data[1], data[2], data[3]) >> 4;
|
||||||
|
|
||||||
if (this->temperature_sensor_ != nullptr) {
|
if (this->temperature_sensor_ != nullptr) {
|
||||||
float temperature = ((200.0f * (float) raw_temperature) / 1048576.0f) - 50.0f;
|
float temperature = ((200.0f * static_cast<float>(raw_temperature)) / AHT10_DIVISOR) - 50.0f;
|
||||||
this->temperature_sensor_->publish_state(temperature);
|
this->temperature_sensor_->publish_state(temperature);
|
||||||
}
|
}
|
||||||
if (this->humidity_sensor_ != nullptr) {
|
if (this->humidity_sensor_ != nullptr) {
|
||||||
float humidity;
|
float humidity = raw_humidity == 0 ? NAN : static_cast<float>(raw_humidity) * 100.0f / AHT10_DIVISOR;
|
||||||
if (raw_humidity == 0) { // unrealistic value
|
|
||||||
humidity = NAN;
|
|
||||||
} else {
|
|
||||||
humidity = (float) raw_humidity * 100.0f / 1048576.0f;
|
|
||||||
}
|
|
||||||
if (std::isnan(humidity)) {
|
if (std::isnan(humidity)) {
|
||||||
ESP_LOGW(TAG, "Invalid humidity! Sensor reported 0%% Hum");
|
ESP_LOGW(TAG, "Invalid humidity reading (0%%), ");
|
||||||
}
|
}
|
||||||
this->humidity_sensor_->publish_state(humidity);
|
this->humidity_sensor_->publish_state(humidity);
|
||||||
}
|
}
|
||||||
@@ -150,7 +148,7 @@ void AHT10Component::update() {
|
|||||||
return;
|
return;
|
||||||
this->start_time_ = millis();
|
this->start_time_ = millis();
|
||||||
if (this->write(AHT10_MEASURE_CMD, sizeof(AHT10_MEASURE_CMD)) != i2c::ERROR_OK) {
|
if (this->write(AHT10_MEASURE_CMD, sizeof(AHT10_MEASURE_CMD)) != i2c::ERROR_OK) {
|
||||||
this->status_set_warning("Communication with AHT10 failed!");
|
this->status_set_warning(ESP_LOG_MSG_COMM_FAIL);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this->restart_read_();
|
this->restart_read_();
|
||||||
@@ -162,7 +160,7 @@ void AHT10Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "AHT10:");
|
ESP_LOGCONFIG(TAG, "AHT10:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AHT10 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
|
@@ -17,7 +17,7 @@ static const char *const TAG = "aic3204";
|
|||||||
}
|
}
|
||||||
|
|
||||||
void AIC3204::setup() {
|
void AIC3204::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AIC3204...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
// Set register page to 0
|
// Set register page to 0
|
||||||
ERROR_CHECK(this->write_byte(AIC3204_PAGE_CTRL, 0x00), "Set page 0 failed");
|
ERROR_CHECK(this->write_byte(AIC3204_PAGE_CTRL, 0x00), "Set page 0 failed");
|
||||||
@@ -113,7 +113,7 @@ void AIC3204::dump_config() {
|
|||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
|
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AIC3204 failed");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -90,7 +90,7 @@ bool AM2315C::convert_(uint8_t *data, float &humidity, float &temperature) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void AM2315C::setup() {
|
void AM2315C::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AM2315C...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
// get status
|
// get status
|
||||||
uint8_t status = 0;
|
uint8_t status = 0;
|
||||||
@@ -188,7 +188,7 @@ void AM2315C::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "AM2315C:");
|
ESP_LOGCONFIG(TAG, "AM2315C:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AM2315C failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
|
@@ -34,7 +34,7 @@ void AM2320Component::update() {
|
|||||||
this->status_clear_warning();
|
this->status_clear_warning();
|
||||||
}
|
}
|
||||||
void AM2320Component::setup() {
|
void AM2320Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AM2320...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t data[8];
|
uint8_t data[8];
|
||||||
data[0] = 0;
|
data[0] = 0;
|
||||||
data[1] = 4;
|
data[1] = 4;
|
||||||
@@ -47,7 +47,7 @@ void AM2320Component::dump_config() {
|
|||||||
ESP_LOGD(TAG, "AM2320:");
|
ESP_LOGD(TAG, "AM2320:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AM2320 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
|
@@ -54,7 +54,7 @@ enum { // APDS9306 registers
|
|||||||
}
|
}
|
||||||
|
|
||||||
void APDS9306::setup() {
|
void APDS9306::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up APDS9306...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
uint8_t id;
|
uint8_t id;
|
||||||
if (!this->read_byte(APDS9306_PART_ID, &id)) { // Part ID register
|
if (!this->read_byte(APDS9306_PART_ID, &id)) { // Part ID register
|
||||||
@@ -97,7 +97,7 @@ void APDS9306::dump_config() {
|
|||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
switch (this->error_code_) {
|
switch (this->error_code_) {
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication with APDS9306 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case WRONG_ID:
|
case WRONG_ID:
|
||||||
ESP_LOGE(TAG, "APDS9306 has invalid id!");
|
ESP_LOGE(TAG, "APDS9306 has invalid id!");
|
||||||
|
@@ -15,7 +15,7 @@ static const char *const TAG = "apds9960";
|
|||||||
#define APDS9960_WRITE_BYTE(reg, value) APDS9960_ERROR_CHECK(this->write_byte(reg, value));
|
#define APDS9960_WRITE_BYTE(reg, value) APDS9960_ERROR_CHECK(this->write_byte(reg, value));
|
||||||
|
|
||||||
void APDS9960::setup() {
|
void APDS9960::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up APDS9960...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t id;
|
uint8_t id;
|
||||||
if (!this->read_byte(0x92, &id)) { // ID register
|
if (!this->read_byte(0x92, &id)) { // ID register
|
||||||
this->error_code_ = COMMUNICATION_FAILED;
|
this->error_code_ = COMMUNICATION_FAILED;
|
||||||
@@ -141,7 +141,7 @@ void APDS9960::dump_config() {
|
|||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
switch (this->error_code_) {
|
switch (this->error_code_) {
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication with APDS9960 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case WRONG_ID:
|
case WRONG_ID:
|
||||||
ESP_LOGE(TAG, "APDS9960 has invalid id!");
|
ESP_LOGE(TAG, "APDS9960 has invalid id!");
|
||||||
|
@@ -4,11 +4,11 @@
|
|||||||
#include <cinttypes>
|
#include <cinttypes>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
#include "esphome/components/network/util.h"
|
#include "esphome/components/network/util.h"
|
||||||
|
#include "esphome/core/application.h"
|
||||||
#include "esphome/core/entity_base.h"
|
#include "esphome/core/entity_base.h"
|
||||||
#include "esphome/core/hal.h"
|
#include "esphome/core/hal.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
#include "esphome/core/version.h"
|
#include "esphome/core/version.h"
|
||||||
#include "esphome/core/application.h"
|
|
||||||
|
|
||||||
#ifdef USE_DEEP_SLEEP
|
#ifdef USE_DEEP_SLEEP
|
||||||
#include "esphome/components/deep_sleep/deep_sleep_component.h"
|
#include "esphome/components/deep_sleep/deep_sleep_component.h"
|
||||||
@@ -135,6 +135,9 @@ void APIConnection::loop() {
|
|||||||
api_error_to_str(err), errno);
|
api_error_to_str(err), errno);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Check if socket has data ready before attempting to read
|
||||||
|
if (this->helper_->is_socket_ready()) {
|
||||||
ReadPacketBuffer buffer;
|
ReadPacketBuffer buffer;
|
||||||
err = this->helper_->read_packet(&buffer);
|
err = this->helper_->read_packet(&buffer);
|
||||||
if (err == APIError::WOULD_BLOCK) {
|
if (err == APIError::WOULD_BLOCK) {
|
||||||
@@ -153,10 +156,15 @@ void APIConnection::loop() {
|
|||||||
} else {
|
} else {
|
||||||
this->last_traffic_ = App.get_loop_component_start_time();
|
this->last_traffic_ = App.get_loop_component_start_time();
|
||||||
// read a packet
|
// read a packet
|
||||||
|
if (buffer.data_len > 0) {
|
||||||
this->read_message(buffer.data_len, buffer.type, &buffer.container[buffer.data_offset]);
|
this->read_message(buffer.data_len, buffer.type, &buffer.container[buffer.data_offset]);
|
||||||
|
} else {
|
||||||
|
this->read_message(0, buffer.type, nullptr);
|
||||||
|
}
|
||||||
if (this->remove_)
|
if (this->remove_)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!this->deferred_message_queue_.empty() && this->helper_->can_write_without_blocking()) {
|
if (!this->deferred_message_queue_.empty() && this->helper_->can_write_without_blocking()) {
|
||||||
this->deferred_message_queue_.process_queue();
|
this->deferred_message_queue_.process_queue();
|
||||||
|
@@ -7,20 +7,13 @@
|
|||||||
#include "proto.h"
|
#include "proto.h"
|
||||||
#include "api_pb2_size.h"
|
#include "api_pb2_size.h"
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
#include <cinttypes>
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace api {
|
namespace api {
|
||||||
|
|
||||||
static const char *const TAG = "api.socket";
|
static const char *const TAG = "api.socket";
|
||||||
|
|
||||||
/// Is the given return value (from write syscalls) a wouldblock error?
|
|
||||||
bool is_would_block(ssize_t ret) {
|
|
||||||
if (ret == -1) {
|
|
||||||
return errno == EWOULDBLOCK || errno == EAGAIN;
|
|
||||||
}
|
|
||||||
return ret == 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const char *api_error_to_str(APIError err) {
|
const char *api_error_to_str(APIError err) {
|
||||||
// not using switch to ensure compiler doesn't try to build a big table out of it
|
// not using switch to ensure compiler doesn't try to build a big table out of it
|
||||||
if (err == APIError::OK) {
|
if (err == APIError::OK) {
|
||||||
@@ -73,92 +66,154 @@ const char *api_error_to_str(APIError err) {
|
|||||||
return "UNKNOWN";
|
return "UNKNOWN";
|
||||||
}
|
}
|
||||||
|
|
||||||
// Common implementation for writing raw data to socket
|
// Helper method to buffer data from IOVs
|
||||||
template<typename StateEnum>
|
void APIFrameHelper::buffer_data_from_iov_(const struct iovec *iov, int iovcnt, uint16_t total_write_len) {
|
||||||
APIError APIFrameHelper::write_raw_(const struct iovec *iov, int iovcnt, socket::Socket *socket,
|
SendBuffer buffer;
|
||||||
std::vector<uint8_t> &tx_buf, const std::string &info, StateEnum &state,
|
buffer.data.reserve(total_write_len);
|
||||||
StateEnum failed_state) {
|
for (int i = 0; i < iovcnt; i++) {
|
||||||
// This method writes data to socket or buffers it
|
const uint8_t *data = reinterpret_cast<uint8_t *>(iov[i].iov_base);
|
||||||
|
buffer.data.insert(buffer.data.end(), data, data + iov[i].iov_len);
|
||||||
|
}
|
||||||
|
this->tx_buf_.push_back(std::move(buffer));
|
||||||
|
}
|
||||||
|
|
||||||
|
// This method writes data to socket or buffers it
|
||||||
|
APIError APIFrameHelper::write_raw_(const struct iovec *iov, int iovcnt) {
|
||||||
// Returns APIError::OK if successful (or would block, but data has been buffered)
|
// Returns APIError::OK if successful (or would block, but data has been buffered)
|
||||||
// Returns APIError::SOCKET_WRITE_FAILED if socket write failed, and sets state to failed_state
|
// Returns APIError::SOCKET_WRITE_FAILED if socket write failed, and sets state to FAILED
|
||||||
|
|
||||||
if (iovcnt == 0)
|
if (iovcnt == 0)
|
||||||
return APIError::OK; // Nothing to do, success
|
return APIError::OK; // Nothing to do, success
|
||||||
|
|
||||||
size_t total_write_len = 0;
|
uint16_t total_write_len = 0;
|
||||||
for (int i = 0; i < iovcnt; i++) {
|
for (int i = 0; i < iovcnt; i++) {
|
||||||
#ifdef HELPER_LOG_PACKETS
|
#ifdef HELPER_LOG_PACKETS
|
||||||
ESP_LOGVV(TAG, "Sending raw: %s",
|
ESP_LOGVV(TAG, "Sending raw: %s",
|
||||||
format_hex_pretty(reinterpret_cast<uint8_t *>(iov[i].iov_base), iov[i].iov_len).c_str());
|
format_hex_pretty(reinterpret_cast<uint8_t *>(iov[i].iov_base), iov[i].iov_len).c_str());
|
||||||
#endif
|
#endif
|
||||||
total_write_len += iov[i].iov_len;
|
total_write_len += static_cast<uint16_t>(iov[i].iov_len);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!tx_buf.empty()) {
|
// Try to send any existing buffered data first if there is any
|
||||||
// try to empty tx_buf first
|
if (!this->tx_buf_.empty()) {
|
||||||
while (!tx_buf.empty()) {
|
APIError send_result = try_send_tx_buf_();
|
||||||
ssize_t sent = socket->write(tx_buf.data(), tx_buf.size());
|
// If real error occurred (not just WOULD_BLOCK), return it
|
||||||
if (is_would_block(sent)) {
|
if (send_result != APIError::OK && send_result != APIError::WOULD_BLOCK) {
|
||||||
break;
|
return send_result;
|
||||||
} else if (sent == -1) {
|
|
||||||
ESP_LOGVV(TAG, "%s: Socket write failed with errno %d", info.c_str(), errno);
|
|
||||||
state = failed_state;
|
|
||||||
return APIError::SOCKET_WRITE_FAILED; // Socket write failed
|
|
||||||
}
|
|
||||||
// TODO: inefficient if multiple packets in txbuf
|
|
||||||
// replace with deque of buffers
|
|
||||||
tx_buf.erase(tx_buf.begin(), tx_buf.begin() + sent);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!tx_buf.empty()) {
|
// If there is still data in the buffer, we can't send, buffer
|
||||||
// tx buf not empty, can't write now because then stream would be inconsistent
|
// the new data and return
|
||||||
// Reserve space upfront to avoid multiple reallocations
|
if (!this->tx_buf_.empty()) {
|
||||||
tx_buf.reserve(tx_buf.size() + total_write_len);
|
this->buffer_data_from_iov_(iov, iovcnt, total_write_len);
|
||||||
for (int i = 0; i < iovcnt; i++) {
|
|
||||||
tx_buf.insert(tx_buf.end(), reinterpret_cast<uint8_t *>(iov[i].iov_base),
|
|
||||||
reinterpret_cast<uint8_t *>(iov[i].iov_base) + iov[i].iov_len);
|
|
||||||
}
|
|
||||||
return APIError::OK; // Success, data buffered
|
return APIError::OK; // Success, data buffered
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t sent = socket->writev(iov, iovcnt);
|
|
||||||
if (is_would_block(sent)) {
|
|
||||||
// operation would block, add buffer to tx_buf
|
|
||||||
// Reserve space upfront to avoid multiple reallocations
|
|
||||||
tx_buf.reserve(tx_buf.size() + total_write_len);
|
|
||||||
for (int i = 0; i < iovcnt; i++) {
|
|
||||||
tx_buf.insert(tx_buf.end(), reinterpret_cast<uint8_t *>(iov[i].iov_base),
|
|
||||||
reinterpret_cast<uint8_t *>(iov[i].iov_base) + iov[i].iov_len);
|
|
||||||
}
|
}
|
||||||
return APIError::OK; // Success, data buffered
|
|
||||||
} else if (sent == -1) {
|
|
||||||
// an error occurred
|
|
||||||
ESP_LOGVV(TAG, "%s: Socket write failed with errno %d", info.c_str(), errno);
|
|
||||||
state = failed_state;
|
|
||||||
return APIError::SOCKET_WRITE_FAILED; // Socket write failed
|
|
||||||
} else if ((size_t) sent != total_write_len) {
|
|
||||||
// partially sent, add end to tx_buf
|
|
||||||
size_t remaining = total_write_len - sent;
|
|
||||||
// Reserve space upfront to avoid multiple reallocations
|
|
||||||
tx_buf.reserve(tx_buf.size() + remaining);
|
|
||||||
|
|
||||||
size_t to_consume = sent;
|
// Try to send directly if no buffered data
|
||||||
|
ssize_t sent = this->socket_->writev(iov, iovcnt);
|
||||||
|
|
||||||
|
if (sent == -1) {
|
||||||
|
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
||||||
|
// Socket would block, buffer the data
|
||||||
|
this->buffer_data_from_iov_(iov, iovcnt, total_write_len);
|
||||||
|
return APIError::OK; // Success, data buffered
|
||||||
|
}
|
||||||
|
// Socket error
|
||||||
|
ESP_LOGVV(TAG, "%s: Socket write failed with errno %d", this->info_.c_str(), errno);
|
||||||
|
this->state_ = State::FAILED;
|
||||||
|
return APIError::SOCKET_WRITE_FAILED; // Socket write failed
|
||||||
|
} else if (static_cast<uint16_t>(sent) < total_write_len) {
|
||||||
|
// Partially sent, buffer the remaining data
|
||||||
|
SendBuffer buffer;
|
||||||
|
uint16_t to_consume = static_cast<uint16_t>(sent);
|
||||||
|
uint16_t remaining = total_write_len - static_cast<uint16_t>(sent);
|
||||||
|
|
||||||
|
buffer.data.reserve(remaining);
|
||||||
|
|
||||||
for (int i = 0; i < iovcnt; i++) {
|
for (int i = 0; i < iovcnt; i++) {
|
||||||
if (to_consume >= iov[i].iov_len) {
|
if (to_consume >= iov[i].iov_len) {
|
||||||
to_consume -= iov[i].iov_len;
|
// This segment was fully sent
|
||||||
|
to_consume -= static_cast<uint16_t>(iov[i].iov_len);
|
||||||
} else {
|
} else {
|
||||||
tx_buf.insert(tx_buf.end(), reinterpret_cast<uint8_t *>(iov[i].iov_base) + to_consume,
|
// This segment was partially sent or not sent at all
|
||||||
reinterpret_cast<uint8_t *>(iov[i].iov_base) + iov[i].iov_len);
|
const uint8_t *data = reinterpret_cast<uint8_t *>(iov[i].iov_base) + to_consume;
|
||||||
|
uint16_t len = static_cast<uint16_t>(iov[i].iov_len) - to_consume;
|
||||||
|
buffer.data.insert(buffer.data.end(), data, data + len);
|
||||||
to_consume = 0;
|
to_consume = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return APIError::OK; // Success, data buffered
|
|
||||||
|
this->tx_buf_.push_back(std::move(buffer));
|
||||||
}
|
}
|
||||||
return APIError::OK; // Success, all data sent
|
|
||||||
|
return APIError::OK; // Success, all data sent or buffered
|
||||||
}
|
}
|
||||||
|
|
||||||
#define HELPER_LOG(msg, ...) ESP_LOGVV(TAG, "%s: " msg, info_.c_str(), ##__VA_ARGS__)
|
// Common implementation for trying to send buffered data
|
||||||
|
// IMPORTANT: Caller MUST ensure tx_buf_ is not empty before calling this method
|
||||||
|
APIError APIFrameHelper::try_send_tx_buf_() {
|
||||||
|
// Try to send from tx_buf - we assume it's not empty as it's the caller's responsibility to check
|
||||||
|
bool tx_buf_empty = false;
|
||||||
|
while (!tx_buf_empty) {
|
||||||
|
// Get the first buffer in the queue
|
||||||
|
SendBuffer &front_buffer = this->tx_buf_.front();
|
||||||
|
|
||||||
|
// Try to send the remaining data in this buffer
|
||||||
|
ssize_t sent = this->socket_->write(front_buffer.current_data(), front_buffer.remaining());
|
||||||
|
|
||||||
|
if (sent == -1) {
|
||||||
|
if (errno != EWOULDBLOCK && errno != EAGAIN) {
|
||||||
|
// Real socket error (not just would block)
|
||||||
|
ESP_LOGVV(TAG, "%s: Socket write failed with errno %d", this->info_.c_str(), errno);
|
||||||
|
this->state_ = State::FAILED;
|
||||||
|
return APIError::SOCKET_WRITE_FAILED; // Socket write failed
|
||||||
|
}
|
||||||
|
// Socket would block, we'll try again later
|
||||||
|
return APIError::WOULD_BLOCK;
|
||||||
|
} else if (sent == 0) {
|
||||||
|
// Nothing sent but not an error
|
||||||
|
return APIError::WOULD_BLOCK;
|
||||||
|
} else if (static_cast<uint16_t>(sent) < front_buffer.remaining()) {
|
||||||
|
// Partially sent, update offset
|
||||||
|
// Cast to ensure no overflow issues with uint16_t
|
||||||
|
front_buffer.offset += static_cast<uint16_t>(sent);
|
||||||
|
return APIError::WOULD_BLOCK; // Stop processing more buffers if we couldn't send a complete buffer
|
||||||
|
} else {
|
||||||
|
// Buffer completely sent, remove it from the queue
|
||||||
|
this->tx_buf_.pop_front();
|
||||||
|
// Update empty status for the loop condition
|
||||||
|
tx_buf_empty = this->tx_buf_.empty();
|
||||||
|
// Continue loop to try sending the next buffer
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return APIError::OK; // All buffers sent successfully
|
||||||
|
}
|
||||||
|
|
||||||
|
APIError APIFrameHelper::init_common_() {
|
||||||
|
if (state_ != State::INITIALIZE || this->socket_ == nullptr) {
|
||||||
|
ESP_LOGVV(TAG, "%s: Bad state for init %d", this->info_.c_str(), (int) state_);
|
||||||
|
return APIError::BAD_STATE;
|
||||||
|
}
|
||||||
|
int err = this->socket_->setblocking(false);
|
||||||
|
if (err != 0) {
|
||||||
|
state_ = State::FAILED;
|
||||||
|
ESP_LOGVV(TAG, "%s: Setting nonblocking failed with errno %d", this->info_.c_str(), errno);
|
||||||
|
return APIError::TCP_NONBLOCKING_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
int enable = 1;
|
||||||
|
err = this->socket_->setsockopt(IPPROTO_TCP, TCP_NODELAY, &enable, sizeof(int));
|
||||||
|
if (err != 0) {
|
||||||
|
state_ = State::FAILED;
|
||||||
|
ESP_LOGVV(TAG, "%s: Setting nodelay failed with errno %d", this->info_.c_str(), errno);
|
||||||
|
return APIError::TCP_NODELAY_FAILED;
|
||||||
|
}
|
||||||
|
return APIError::OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define HELPER_LOG(msg, ...) ESP_LOGVV(TAG, "%s: " msg, this->info_.c_str(), ##__VA_ARGS__)
|
||||||
// uncomment to log raw packets
|
// uncomment to log raw packets
|
||||||
//#define HELPER_LOG_PACKETS
|
//#define HELPER_LOG_PACKETS
|
||||||
|
|
||||||
@@ -206,23 +261,9 @@ std::string noise_err_to_str(int err) {
|
|||||||
|
|
||||||
/// Initialize the frame helper, returns OK if successful.
|
/// Initialize the frame helper, returns OK if successful.
|
||||||
APIError APINoiseFrameHelper::init() {
|
APIError APINoiseFrameHelper::init() {
|
||||||
if (state_ != State::INITIALIZE || socket_ == nullptr) {
|
APIError err = init_common_();
|
||||||
HELPER_LOG("Bad state for init %d", (int) state_);
|
if (err != APIError::OK) {
|
||||||
return APIError::BAD_STATE;
|
return err;
|
||||||
}
|
|
||||||
int err = socket_->setblocking(false);
|
|
||||||
if (err != 0) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Setting nonblocking failed with errno %d", errno);
|
|
||||||
return APIError::TCP_NONBLOCKING_FAILED;
|
|
||||||
}
|
|
||||||
|
|
||||||
int enable = 1;
|
|
||||||
err = socket_->setsockopt(IPPROTO_TCP, TCP_NODELAY, &enable, sizeof(int));
|
|
||||||
if (err != 0) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Setting nodelay failed with errno %d", errno);
|
|
||||||
return APIError::TCP_NODELAY_FAILED;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// init prologue
|
// init prologue
|
||||||
@@ -234,17 +275,16 @@ APIError APINoiseFrameHelper::init() {
|
|||||||
/// Run through handshake messages (if in that phase)
|
/// Run through handshake messages (if in that phase)
|
||||||
APIError APINoiseFrameHelper::loop() {
|
APIError APINoiseFrameHelper::loop() {
|
||||||
APIError err = state_action_();
|
APIError err = state_action_();
|
||||||
if (err == APIError::WOULD_BLOCK)
|
if (err != APIError::OK && err != APIError::WOULD_BLOCK) {
|
||||||
return APIError::OK;
|
|
||||||
if (err != APIError::OK)
|
|
||||||
return err;
|
return err;
|
||||||
if (!tx_buf_.empty()) {
|
}
|
||||||
|
if (!this->tx_buf_.empty()) {
|
||||||
err = try_send_tx_buf_();
|
err = try_send_tx_buf_();
|
||||||
if (err != APIError::OK) {
|
if (err != APIError::OK && err != APIError::WOULD_BLOCK) {
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return APIError::OK;
|
return APIError::OK; // Convert WOULD_BLOCK to OK to avoid connection termination
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Read a packet into the rx_buf_. If successful, stores frame data in the frame parameter
|
/** Read a packet into the rx_buf_. If successful, stores frame data in the frame parameter
|
||||||
@@ -270,8 +310,8 @@ APIError APINoiseFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
// read header
|
// read header
|
||||||
if (rx_header_buf_len_ < 3) {
|
if (rx_header_buf_len_ < 3) {
|
||||||
// no header information yet
|
// no header information yet
|
||||||
size_t to_read = 3 - rx_header_buf_len_;
|
uint8_t to_read = 3 - rx_header_buf_len_;
|
||||||
ssize_t received = socket_->read(&rx_header_buf_[rx_header_buf_len_], to_read);
|
ssize_t received = this->socket_->read(&rx_header_buf_[rx_header_buf_len_], to_read);
|
||||||
if (received == -1) {
|
if (received == -1) {
|
||||||
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
||||||
return APIError::WOULD_BLOCK;
|
return APIError::WOULD_BLOCK;
|
||||||
@@ -284,8 +324,8 @@ APIError APINoiseFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
HELPER_LOG("Connection closed");
|
HELPER_LOG("Connection closed");
|
||||||
return APIError::CONNECTION_CLOSED;
|
return APIError::CONNECTION_CLOSED;
|
||||||
}
|
}
|
||||||
rx_header_buf_len_ += received;
|
rx_header_buf_len_ += static_cast<uint8_t>(received);
|
||||||
if ((size_t) received != to_read) {
|
if (static_cast<uint8_t>(received) != to_read) {
|
||||||
// not a full read
|
// not a full read
|
||||||
return APIError::WOULD_BLOCK;
|
return APIError::WOULD_BLOCK;
|
||||||
}
|
}
|
||||||
@@ -317,8 +357,8 @@ APIError APINoiseFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
|
|
||||||
if (rx_buf_len_ < msg_size) {
|
if (rx_buf_len_ < msg_size) {
|
||||||
// more data to read
|
// more data to read
|
||||||
size_t to_read = msg_size - rx_buf_len_;
|
uint16_t to_read = msg_size - rx_buf_len_;
|
||||||
ssize_t received = socket_->read(&rx_buf_[rx_buf_len_], to_read);
|
ssize_t received = this->socket_->read(&rx_buf_[rx_buf_len_], to_read);
|
||||||
if (received == -1) {
|
if (received == -1) {
|
||||||
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
||||||
return APIError::WOULD_BLOCK;
|
return APIError::WOULD_BLOCK;
|
||||||
@@ -331,8 +371,8 @@ APIError APINoiseFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
HELPER_LOG("Connection closed");
|
HELPER_LOG("Connection closed");
|
||||||
return APIError::CONNECTION_CLOSED;
|
return APIError::CONNECTION_CLOSED;
|
||||||
}
|
}
|
||||||
rx_buf_len_ += received;
|
rx_buf_len_ += static_cast<uint16_t>(received);
|
||||||
if ((size_t) received != to_read) {
|
if (static_cast<uint16_t>(received) != to_read) {
|
||||||
// not all read
|
// not all read
|
||||||
return APIError::WOULD_BLOCK;
|
return APIError::WOULD_BLOCK;
|
||||||
}
|
}
|
||||||
@@ -381,6 +421,8 @@ APIError APINoiseFrameHelper::state_action_() {
|
|||||||
if (aerr != APIError::OK)
|
if (aerr != APIError::OK)
|
||||||
return aerr;
|
return aerr;
|
||||||
// ignore contents, may be used in future for flags
|
// ignore contents, may be used in future for flags
|
||||||
|
// Reserve space for: existing prologue + 2 size bytes + frame data
|
||||||
|
prologue_.reserve(prologue_.size() + 2 + frame.msg.size());
|
||||||
prologue_.push_back((uint8_t) (frame.msg.size() >> 8));
|
prologue_.push_back((uint8_t) (frame.msg.size() >> 8));
|
||||||
prologue_.push_back((uint8_t) frame.msg.size());
|
prologue_.push_back((uint8_t) frame.msg.size());
|
||||||
prologue_.insert(prologue_.end(), frame.msg.begin(), frame.msg.end());
|
prologue_.insert(prologue_.end(), frame.msg.begin(), frame.msg.end());
|
||||||
@@ -389,16 +431,20 @@ APIError APINoiseFrameHelper::state_action_() {
|
|||||||
}
|
}
|
||||||
if (state_ == State::SERVER_HELLO) {
|
if (state_ == State::SERVER_HELLO) {
|
||||||
// send server hello
|
// send server hello
|
||||||
|
const std::string &name = App.get_name();
|
||||||
|
const std::string &mac = get_mac_address();
|
||||||
|
|
||||||
std::vector<uint8_t> msg;
|
std::vector<uint8_t> msg;
|
||||||
|
// Reserve space for: 1 byte proto + name + null + mac + null
|
||||||
|
msg.reserve(1 + name.size() + 1 + mac.size() + 1);
|
||||||
|
|
||||||
// chosen proto
|
// chosen proto
|
||||||
msg.push_back(0x01);
|
msg.push_back(0x01);
|
||||||
|
|
||||||
// node name, terminated by null byte
|
// node name, terminated by null byte
|
||||||
const std::string &name = App.get_name();
|
|
||||||
const uint8_t *name_ptr = reinterpret_cast<const uint8_t *>(name.c_str());
|
const uint8_t *name_ptr = reinterpret_cast<const uint8_t *>(name.c_str());
|
||||||
msg.insert(msg.end(), name_ptr, name_ptr + name.size() + 1);
|
msg.insert(msg.end(), name_ptr, name_ptr + name.size() + 1);
|
||||||
// node mac, terminated by null byte
|
// node mac, terminated by null byte
|
||||||
const std::string &mac = get_mac_address();
|
|
||||||
const uint8_t *mac_ptr = reinterpret_cast<const uint8_t *>(mac.c_str());
|
const uint8_t *mac_ptr = reinterpret_cast<const uint8_t *>(mac.c_str());
|
||||||
msg.insert(msg.end(), mac_ptr, mac_ptr + mac.size() + 1);
|
msg.insert(msg.end(), mac_ptr, mac_ptr + mac.size() + 1);
|
||||||
|
|
||||||
@@ -505,7 +551,6 @@ void APINoiseFrameHelper::send_explicit_handshake_reject_(const std::string &rea
|
|||||||
write_frame_(data.data(), data.size());
|
write_frame_(data.data(), data.size());
|
||||||
state_ = orig_state;
|
state_ = orig_state;
|
||||||
}
|
}
|
||||||
|
|
||||||
APIError APINoiseFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
APIError APINoiseFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
||||||
int err;
|
int err;
|
||||||
APIError aerr;
|
APIError aerr;
|
||||||
@@ -533,7 +578,7 @@ APIError APINoiseFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
|||||||
return APIError::CIPHERSTATE_DECRYPT_FAILED;
|
return APIError::CIPHERSTATE_DECRYPT_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t msg_size = mbuf.size;
|
uint16_t msg_size = mbuf.size;
|
||||||
uint8_t *msg_data = frame.msg.data();
|
uint8_t *msg_data = frame.msg.data();
|
||||||
if (msg_size < 4) {
|
if (msg_size < 4) {
|
||||||
state_ = State::FAILED;
|
state_ = State::FAILED;
|
||||||
@@ -559,7 +604,6 @@ APIError APINoiseFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
|||||||
buffer->type = type;
|
buffer->type = type;
|
||||||
return APIError::OK;
|
return APIError::OK;
|
||||||
}
|
}
|
||||||
bool APINoiseFrameHelper::can_write_without_blocking() { return state_ == State::DATA && tx_buf_.empty(); }
|
|
||||||
APIError APINoiseFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) {
|
APIError APINoiseFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) {
|
||||||
int err;
|
int err;
|
||||||
APIError aerr;
|
APIError aerr;
|
||||||
@@ -574,9 +618,9 @@ APIError APINoiseFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuf
|
|||||||
|
|
||||||
std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
|
std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
|
||||||
// Message data starts after padding
|
// Message data starts after padding
|
||||||
size_t payload_len = raw_buffer->size() - frame_header_padding_;
|
uint16_t payload_len = raw_buffer->size() - frame_header_padding_;
|
||||||
size_t padding = 0;
|
uint16_t padding = 0;
|
||||||
size_t msg_len = 4 + payload_len + padding;
|
uint16_t msg_len = 4 + payload_len + padding;
|
||||||
|
|
||||||
// We need to resize to include MAC space, but we already reserved it in create_buffer
|
// We need to resize to include MAC space, but we already reserved it in create_buffer
|
||||||
raw_buffer->resize(raw_buffer->size() + frame_footer_size_);
|
raw_buffer->resize(raw_buffer->size() + frame_footer_size_);
|
||||||
@@ -609,7 +653,7 @@ APIError APINoiseFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuf
|
|||||||
return APIError::CIPHERSTATE_ENCRYPT_FAILED;
|
return APIError::CIPHERSTATE_ENCRYPT_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t total_len = 3 + mbuf.size;
|
uint16_t total_len = 3 + mbuf.size;
|
||||||
buf_start[1] = (uint8_t) (mbuf.size >> 8);
|
buf_start[1] = (uint8_t) (mbuf.size >> 8);
|
||||||
buf_start[2] = (uint8_t) mbuf.size;
|
buf_start[2] = (uint8_t) mbuf.size;
|
||||||
|
|
||||||
@@ -620,29 +664,9 @@ APIError APINoiseFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuf
|
|||||||
iov.iov_len = total_len;
|
iov.iov_len = total_len;
|
||||||
|
|
||||||
// write raw to not have two packets sent if NAGLE disabled
|
// write raw to not have two packets sent if NAGLE disabled
|
||||||
return write_raw_(&iov, 1);
|
return this->write_raw_(&iov, 1);
|
||||||
}
|
}
|
||||||
APIError APINoiseFrameHelper::try_send_tx_buf_() {
|
APIError APINoiseFrameHelper::write_frame_(const uint8_t *data, uint16_t len) {
|
||||||
// try send from tx_buf
|
|
||||||
while (state_ != State::CLOSED && !tx_buf_.empty()) {
|
|
||||||
ssize_t sent = socket_->write(tx_buf_.data(), tx_buf_.size());
|
|
||||||
if (sent == -1) {
|
|
||||||
if (errno == EWOULDBLOCK || errno == EAGAIN)
|
|
||||||
break;
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Socket write failed with errno %d", errno);
|
|
||||||
return APIError::SOCKET_WRITE_FAILED;
|
|
||||||
} else if (sent == 0) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
// TODO: inefficient if multiple packets in txbuf
|
|
||||||
// replace with deque of buffers
|
|
||||||
tx_buf_.erase(tx_buf_.begin(), tx_buf_.begin() + sent);
|
|
||||||
}
|
|
||||||
|
|
||||||
return APIError::OK;
|
|
||||||
}
|
|
||||||
APIError APINoiseFrameHelper::write_frame_(const uint8_t *data, size_t len) {
|
|
||||||
uint8_t header[3];
|
uint8_t header[3];
|
||||||
header[0] = 0x01; // indicator
|
header[0] = 0x01; // indicator
|
||||||
header[1] = (uint8_t) (len >> 8);
|
header[1] = (uint8_t) (len >> 8);
|
||||||
@@ -652,12 +676,12 @@ APIError APINoiseFrameHelper::write_frame_(const uint8_t *data, size_t len) {
|
|||||||
iov[0].iov_base = header;
|
iov[0].iov_base = header;
|
||||||
iov[0].iov_len = 3;
|
iov[0].iov_len = 3;
|
||||||
if (len == 0) {
|
if (len == 0) {
|
||||||
return write_raw_(iov, 1);
|
return this->write_raw_(iov, 1);
|
||||||
}
|
}
|
||||||
iov[1].iov_base = const_cast<uint8_t *>(data);
|
iov[1].iov_base = const_cast<uint8_t *>(data);
|
||||||
iov[1].iov_len = len;
|
iov[1].iov_len = len;
|
||||||
|
|
||||||
return write_raw_(iov, 2);
|
return this->write_raw_(iov, 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Initiate the data structures for the handshake.
|
/** Initiate the data structures for the handshake.
|
||||||
@@ -752,22 +776,6 @@ APINoiseFrameHelper::~APINoiseFrameHelper() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
APIError APINoiseFrameHelper::close() {
|
|
||||||
state_ = State::CLOSED;
|
|
||||||
int err = socket_->close();
|
|
||||||
if (err == -1)
|
|
||||||
return APIError::CLOSE_FAILED;
|
|
||||||
return APIError::OK;
|
|
||||||
}
|
|
||||||
APIError APINoiseFrameHelper::shutdown(int how) {
|
|
||||||
int err = socket_->shutdown(how);
|
|
||||||
if (err == -1)
|
|
||||||
return APIError::SHUTDOWN_FAILED;
|
|
||||||
if (how == SHUT_RDWR) {
|
|
||||||
state_ = State::CLOSED;
|
|
||||||
}
|
|
||||||
return APIError::OK;
|
|
||||||
}
|
|
||||||
extern "C" {
|
extern "C" {
|
||||||
// declare how noise generates random bytes (here with a good HWRNG based on the RF system)
|
// declare how noise generates random bytes (here with a good HWRNG based on the RF system)
|
||||||
void noise_rand_bytes(void *output, size_t len) {
|
void noise_rand_bytes(void *output, size_t len) {
|
||||||
@@ -778,32 +786,15 @@ void noise_rand_bytes(void *output, size_t len) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Explicit template instantiation for Noise
|
|
||||||
template APIError APIFrameHelper::write_raw_<APINoiseFrameHelper::State>(
|
|
||||||
const struct iovec *iov, int iovcnt, socket::Socket *socket, std::vector<uint8_t> &tx_buf_, const std::string &info,
|
|
||||||
APINoiseFrameHelper::State &state, APINoiseFrameHelper::State failed_state);
|
|
||||||
#endif // USE_API_NOISE
|
#endif // USE_API_NOISE
|
||||||
|
|
||||||
#ifdef USE_API_PLAINTEXT
|
#ifdef USE_API_PLAINTEXT
|
||||||
|
|
||||||
/// Initialize the frame helper, returns OK if successful.
|
/// Initialize the frame helper, returns OK if successful.
|
||||||
APIError APIPlaintextFrameHelper::init() {
|
APIError APIPlaintextFrameHelper::init() {
|
||||||
if (state_ != State::INITIALIZE || socket_ == nullptr) {
|
APIError err = init_common_();
|
||||||
HELPER_LOG("Bad state for init %d", (int) state_);
|
if (err != APIError::OK) {
|
||||||
return APIError::BAD_STATE;
|
return err;
|
||||||
}
|
|
||||||
int err = socket_->setblocking(false);
|
|
||||||
if (err != 0) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Setting nonblocking failed with errno %d", errno);
|
|
||||||
return APIError::TCP_NONBLOCKING_FAILED;
|
|
||||||
}
|
|
||||||
int enable = 1;
|
|
||||||
err = socket_->setsockopt(IPPROTO_TCP, TCP_NODELAY, &enable, sizeof(int));
|
|
||||||
if (err != 0) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Setting nodelay failed with errno %d", errno);
|
|
||||||
return APIError::TCP_NODELAY_FAILED;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
state_ = State::DATA;
|
state_ = State::DATA;
|
||||||
@@ -814,14 +805,13 @@ APIError APIPlaintextFrameHelper::loop() {
|
|||||||
if (state_ != State::DATA) {
|
if (state_ != State::DATA) {
|
||||||
return APIError::BAD_STATE;
|
return APIError::BAD_STATE;
|
||||||
}
|
}
|
||||||
// try send pending TX data
|
if (!this->tx_buf_.empty()) {
|
||||||
if (!tx_buf_.empty()) {
|
|
||||||
APIError err = try_send_tx_buf_();
|
APIError err = try_send_tx_buf_();
|
||||||
if (err != APIError::OK) {
|
if (err != APIError::OK && err != APIError::WOULD_BLOCK) {
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return APIError::OK;
|
return APIError::OK; // Convert WOULD_BLOCK to OK to avoid connection termination
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Read a packet into the rx_buf_. If successful, stores frame data in the frame parameter
|
/** Read a packet into the rx_buf_. If successful, stores frame data in the frame parameter
|
||||||
@@ -841,12 +831,15 @@ APIError APIPlaintextFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
|
|
||||||
// read header
|
// read header
|
||||||
while (!rx_header_parsed_) {
|
while (!rx_header_parsed_) {
|
||||||
uint8_t data;
|
// Now that we know when the socket is ready, we can read up to 3 bytes
|
||||||
// Reading one byte at a time is fastest in practice for ESP32 when
|
// into the rx_header_buf_ before we have to switch back to reading
|
||||||
// there is no data on the wire (which is the common case).
|
// one byte at a time to ensure we don't read past the message and
|
||||||
// This results in faster failure detection compared to
|
// into the next one.
|
||||||
// attempting to read multiple bytes at once.
|
|
||||||
ssize_t received = socket_->read(&data, 1);
|
// Read directly into rx_header_buf_ at the current position
|
||||||
|
// Try to get to at least 3 bytes total (indicator + 2 varint bytes), then read one byte at a time
|
||||||
|
ssize_t received =
|
||||||
|
this->socket_->read(&rx_header_buf_[rx_header_buf_pos_], rx_header_buf_pos_ < 3 ? 3 - rx_header_buf_pos_ : 1);
|
||||||
if (received == -1) {
|
if (received == -1) {
|
||||||
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
||||||
return APIError::WOULD_BLOCK;
|
return APIError::WOULD_BLOCK;
|
||||||
@@ -860,64 +853,74 @@ APIError APIPlaintextFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
return APIError::CONNECTION_CLOSED;
|
return APIError::CONNECTION_CLOSED;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Successfully read a byte
|
// If this was the first read, validate the indicator byte
|
||||||
|
if (rx_header_buf_pos_ == 0 && received > 0) {
|
||||||
// Process byte according to current buffer position
|
if (rx_header_buf_[0] != 0x00) {
|
||||||
if (rx_header_buf_pos_ == 0) { // Case 1: First byte (indicator byte)
|
|
||||||
if (data != 0x00) {
|
|
||||||
state_ = State::FAILED;
|
state_ = State::FAILED;
|
||||||
HELPER_LOG("Bad indicator byte %u", data);
|
HELPER_LOG("Bad indicator byte %u", rx_header_buf_[0]);
|
||||||
return APIError::BAD_INDICATOR;
|
return APIError::BAD_INDICATOR;
|
||||||
}
|
}
|
||||||
// We don't store the indicator byte, just increment position
|
|
||||||
rx_header_buf_pos_ = 1; // Set to 1 directly
|
|
||||||
continue; // Need more bytes before we can parse
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check buffer overflow before storing
|
rx_header_buf_pos_ += received;
|
||||||
if (rx_header_buf_pos_ == 5) { // Case 2: Buffer would overflow (5 bytes is max allowed)
|
|
||||||
|
// Check for buffer overflow
|
||||||
|
if (rx_header_buf_pos_ >= sizeof(rx_header_buf_)) {
|
||||||
state_ = State::FAILED;
|
state_ = State::FAILED;
|
||||||
HELPER_LOG("Header buffer overflow");
|
HELPER_LOG("Header buffer overflow");
|
||||||
return APIError::BAD_DATA_PACKET;
|
return APIError::BAD_DATA_PACKET;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Store byte in buffer (adjust index to account for skipped indicator byte)
|
// Need at least 3 bytes total (indicator + 2 varint bytes) before trying to parse
|
||||||
rx_header_buf_[rx_header_buf_pos_ - 1] = data;
|
if (rx_header_buf_pos_ < 3) {
|
||||||
|
continue;
|
||||||
// Increment position after storing
|
|
||||||
rx_header_buf_pos_++;
|
|
||||||
|
|
||||||
// Case 3: If we only have one varint byte, we need more
|
|
||||||
if (rx_header_buf_pos_ == 2) { // Have read indicator + 1 byte
|
|
||||||
continue; // Need more bytes before we can parse
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// At this point, we have at least 3 bytes total:
|
// At this point, we have at least 3 bytes total:
|
||||||
// - Validated indicator byte (0x00) but not stored
|
// - Validated indicator byte (0x00) stored at position 0
|
||||||
// - At least 2 bytes in the buffer for the varints
|
// - At least 2 bytes in the buffer for the varints
|
||||||
// Buffer layout:
|
// Buffer layout:
|
||||||
// First 1-3 bytes: Message size varint (variable length)
|
// [0]: indicator byte (0x00)
|
||||||
// - 2 bytes would only allow up to 16383, which is less than noise's 65535
|
// [1-3]: Message size varint (variable length)
|
||||||
|
// - 2 bytes would only allow up to 16383, which is less than noise's UINT16_MAX (65535)
|
||||||
// - 3 bytes allows up to 2097151, ensuring we support at least as much as noise
|
// - 3 bytes allows up to 2097151, ensuring we support at least as much as noise
|
||||||
// Remaining 1-2 bytes: Message type varint (variable length)
|
// [2-5]: Message type varint (variable length)
|
||||||
// We now attempt to parse both varints. If either is incomplete,
|
// We now attempt to parse both varints. If either is incomplete,
|
||||||
// we'll continue reading more bytes.
|
// we'll continue reading more bytes.
|
||||||
|
|
||||||
|
// Skip indicator byte at position 0
|
||||||
|
uint8_t varint_pos = 1;
|
||||||
uint32_t consumed = 0;
|
uint32_t consumed = 0;
|
||||||
auto msg_size_varint = ProtoVarInt::parse(&rx_header_buf_[0], rx_header_buf_pos_ - 1, &consumed);
|
|
||||||
|
auto msg_size_varint = ProtoVarInt::parse(&rx_header_buf_[varint_pos], rx_header_buf_pos_ - varint_pos, &consumed);
|
||||||
if (!msg_size_varint.has_value()) {
|
if (!msg_size_varint.has_value()) {
|
||||||
// not enough data there yet
|
// not enough data there yet
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
rx_header_parsed_len_ = msg_size_varint->as_uint32();
|
if (msg_size_varint->as_uint32() > std::numeric_limits<uint16_t>::max()) {
|
||||||
|
state_ = State::FAILED;
|
||||||
|
HELPER_LOG("Bad packet: message size %" PRIu32 " exceeds maximum %u", msg_size_varint->as_uint32(),
|
||||||
|
std::numeric_limits<uint16_t>::max());
|
||||||
|
return APIError::BAD_DATA_PACKET;
|
||||||
|
}
|
||||||
|
rx_header_parsed_len_ = msg_size_varint->as_uint16();
|
||||||
|
|
||||||
auto msg_type_varint = ProtoVarInt::parse(&rx_header_buf_[consumed], rx_header_buf_pos_ - 1 - consumed, &consumed);
|
// Move to next varint position
|
||||||
|
varint_pos += consumed;
|
||||||
|
|
||||||
|
auto msg_type_varint = ProtoVarInt::parse(&rx_header_buf_[varint_pos], rx_header_buf_pos_ - varint_pos, &consumed);
|
||||||
if (!msg_type_varint.has_value()) {
|
if (!msg_type_varint.has_value()) {
|
||||||
// not enough data there yet
|
// not enough data there yet
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
rx_header_parsed_type_ = msg_type_varint->as_uint32();
|
if (msg_type_varint->as_uint32() > std::numeric_limits<uint16_t>::max()) {
|
||||||
|
state_ = State::FAILED;
|
||||||
|
HELPER_LOG("Bad packet: message type %" PRIu32 " exceeds maximum %u", msg_type_varint->as_uint32(),
|
||||||
|
std::numeric_limits<uint16_t>::max());
|
||||||
|
return APIError::BAD_DATA_PACKET;
|
||||||
|
}
|
||||||
|
rx_header_parsed_type_ = msg_type_varint->as_uint16();
|
||||||
rx_header_parsed_ = true;
|
rx_header_parsed_ = true;
|
||||||
}
|
}
|
||||||
// header reading done
|
// header reading done
|
||||||
@@ -929,8 +932,8 @@ APIError APIPlaintextFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
|
|
||||||
if (rx_buf_len_ < rx_header_parsed_len_) {
|
if (rx_buf_len_ < rx_header_parsed_len_) {
|
||||||
// more data to read
|
// more data to read
|
||||||
size_t to_read = rx_header_parsed_len_ - rx_buf_len_;
|
uint16_t to_read = rx_header_parsed_len_ - rx_buf_len_;
|
||||||
ssize_t received = socket_->read(&rx_buf_[rx_buf_len_], to_read);
|
ssize_t received = this->socket_->read(&rx_buf_[rx_buf_len_], to_read);
|
||||||
if (received == -1) {
|
if (received == -1) {
|
||||||
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
||||||
return APIError::WOULD_BLOCK;
|
return APIError::WOULD_BLOCK;
|
||||||
@@ -943,8 +946,8 @@ APIError APIPlaintextFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
HELPER_LOG("Connection closed");
|
HELPER_LOG("Connection closed");
|
||||||
return APIError::CONNECTION_CLOSED;
|
return APIError::CONNECTION_CLOSED;
|
||||||
}
|
}
|
||||||
rx_buf_len_ += received;
|
rx_buf_len_ += static_cast<uint16_t>(received);
|
||||||
if ((size_t) received != to_read) {
|
if (static_cast<uint16_t>(received) != to_read) {
|
||||||
// not all read
|
// not all read
|
||||||
return APIError::WOULD_BLOCK;
|
return APIError::WOULD_BLOCK;
|
||||||
}
|
}
|
||||||
@@ -962,7 +965,6 @@ APIError APIPlaintextFrameHelper::try_read_frame_(ParsedFrame *frame) {
|
|||||||
rx_header_parsed_ = false;
|
rx_header_parsed_ = false;
|
||||||
return APIError::OK;
|
return APIError::OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
APIError APIPlaintextFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
APIError APIPlaintextFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
||||||
APIError aerr;
|
APIError aerr;
|
||||||
|
|
||||||
@@ -990,7 +992,7 @@ APIError APIPlaintextFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
|||||||
"Bad indicator byte";
|
"Bad indicator byte";
|
||||||
iov[0].iov_base = (void *) msg;
|
iov[0].iov_base = (void *) msg;
|
||||||
iov[0].iov_len = 19;
|
iov[0].iov_len = 19;
|
||||||
write_raw_(iov, 1);
|
this->write_raw_(iov, 1);
|
||||||
}
|
}
|
||||||
return aerr;
|
return aerr;
|
||||||
}
|
}
|
||||||
@@ -1001,7 +1003,6 @@ APIError APIPlaintextFrameHelper::read_packet(ReadPacketBuffer *buffer) {
|
|||||||
buffer->type = rx_header_parsed_type_;
|
buffer->type = rx_header_parsed_type_;
|
||||||
return APIError::OK;
|
return APIError::OK;
|
||||||
}
|
}
|
||||||
bool APIPlaintextFrameHelper::can_write_without_blocking() { return state_ == State::DATA && tx_buf_.empty(); }
|
|
||||||
APIError APIPlaintextFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) {
|
APIError APIPlaintextFrameHelper::write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) {
|
||||||
if (state_ != State::DATA) {
|
if (state_ != State::DATA) {
|
||||||
return APIError::BAD_STATE;
|
return APIError::BAD_STATE;
|
||||||
@@ -1009,12 +1010,12 @@ APIError APIPlaintextFrameHelper::write_protobuf_packet(uint16_t type, ProtoWrit
|
|||||||
|
|
||||||
std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
|
std::vector<uint8_t> *raw_buffer = buffer.get_buffer();
|
||||||
// Message data starts after padding (frame_header_padding_ = 6)
|
// Message data starts after padding (frame_header_padding_ = 6)
|
||||||
size_t payload_len = raw_buffer->size() - frame_header_padding_;
|
uint16_t payload_len = static_cast<uint16_t>(raw_buffer->size() - frame_header_padding_);
|
||||||
|
|
||||||
// Calculate varint sizes for header components
|
// Calculate varint sizes for header components
|
||||||
size_t size_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(payload_len));
|
uint8_t size_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(payload_len));
|
||||||
size_t type_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(type));
|
uint8_t type_varint_len = api::ProtoSize::varint(static_cast<uint32_t>(type));
|
||||||
size_t total_header_len = 1 + size_varint_len + type_varint_len;
|
uint8_t total_header_len = 1 + size_varint_len + type_varint_len;
|
||||||
|
|
||||||
if (total_header_len > frame_header_padding_) {
|
if (total_header_len > frame_header_padding_) {
|
||||||
// Header is too large to fit in the padding
|
// Header is too large to fit in the padding
|
||||||
@@ -1044,7 +1045,7 @@ APIError APIPlaintextFrameHelper::write_protobuf_packet(uint16_t type, ProtoWrit
|
|||||||
// [4-5] - Message type varint (2 bytes, for types 128-32767)
|
// [4-5] - Message type varint (2 bytes, for types 128-32767)
|
||||||
// [6...] - Actual payload data
|
// [6...] - Actual payload data
|
||||||
uint8_t *buf_start = raw_buffer->data();
|
uint8_t *buf_start = raw_buffer->data();
|
||||||
size_t header_offset = frame_header_padding_ - total_header_len;
|
uint8_t header_offset = frame_header_padding_ - total_header_len;
|
||||||
|
|
||||||
// Write the plaintext header
|
// Write the plaintext header
|
||||||
buf_start[header_offset] = 0x00; // indicator
|
buf_start[header_offset] = 0x00; // indicator
|
||||||
@@ -1063,46 +1064,7 @@ APIError APIPlaintextFrameHelper::write_protobuf_packet(uint16_t type, ProtoWrit
|
|||||||
|
|
||||||
return write_raw_(&iov, 1);
|
return write_raw_(&iov, 1);
|
||||||
}
|
}
|
||||||
APIError APIPlaintextFrameHelper::try_send_tx_buf_() {
|
|
||||||
// try send from tx_buf
|
|
||||||
while (state_ != State::CLOSED && !tx_buf_.empty()) {
|
|
||||||
ssize_t sent = socket_->write(tx_buf_.data(), tx_buf_.size());
|
|
||||||
if (is_would_block(sent)) {
|
|
||||||
break;
|
|
||||||
} else if (sent == -1) {
|
|
||||||
state_ = State::FAILED;
|
|
||||||
HELPER_LOG("Socket write failed with errno %d", errno);
|
|
||||||
return APIError::SOCKET_WRITE_FAILED;
|
|
||||||
}
|
|
||||||
// TODO: inefficient if multiple packets in txbuf
|
|
||||||
// replace with deque of buffers
|
|
||||||
tx_buf_.erase(tx_buf_.begin(), tx_buf_.begin() + sent);
|
|
||||||
}
|
|
||||||
|
|
||||||
return APIError::OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
APIError APIPlaintextFrameHelper::close() {
|
|
||||||
state_ = State::CLOSED;
|
|
||||||
int err = socket_->close();
|
|
||||||
if (err == -1)
|
|
||||||
return APIError::CLOSE_FAILED;
|
|
||||||
return APIError::OK;
|
|
||||||
}
|
|
||||||
APIError APIPlaintextFrameHelper::shutdown(int how) {
|
|
||||||
int err = socket_->shutdown(how);
|
|
||||||
if (err == -1)
|
|
||||||
return APIError::SHUTDOWN_FAILED;
|
|
||||||
if (how == SHUT_RDWR) {
|
|
||||||
state_ = State::CLOSED;
|
|
||||||
}
|
|
||||||
return APIError::OK;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Explicit template instantiation for Plaintext
|
|
||||||
template APIError APIFrameHelper::write_raw_<APIPlaintextFrameHelper::State>(
|
|
||||||
const struct iovec *iov, int iovcnt, socket::Socket *socket, std::vector<uint8_t> &tx_buf_, const std::string &info,
|
|
||||||
APIPlaintextFrameHelper::State &state, APIPlaintextFrameHelper::State failed_state);
|
|
||||||
#endif // USE_API_PLAINTEXT
|
#endif // USE_API_PLAINTEXT
|
||||||
|
|
||||||
} // namespace api
|
} // namespace api
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <deque>
|
#include <deque>
|
||||||
|
#include <limits>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
@@ -12,6 +13,7 @@
|
|||||||
|
|
||||||
#include "api_noise_context.h"
|
#include "api_noise_context.h"
|
||||||
#include "esphome/components/socket/socket.h"
|
#include "esphome/components/socket/socket.h"
|
||||||
|
#include "esphome/core/application.h"
|
||||||
|
|
||||||
namespace esphome {
|
namespace esphome {
|
||||||
namespace api {
|
namespace api {
|
||||||
@@ -21,15 +23,8 @@ class ProtoWriteBuffer;
|
|||||||
struct ReadPacketBuffer {
|
struct ReadPacketBuffer {
|
||||||
std::vector<uint8_t> container;
|
std::vector<uint8_t> container;
|
||||||
uint16_t type;
|
uint16_t type;
|
||||||
size_t data_offset;
|
uint16_t data_offset;
|
||||||
size_t data_len;
|
uint16_t data_len;
|
||||||
};
|
|
||||||
|
|
||||||
struct PacketBuffer {
|
|
||||||
const std::vector<uint8_t> container;
|
|
||||||
uint16_t type;
|
|
||||||
uint8_t data_offset;
|
|
||||||
uint8_t data_len;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
enum class APIError : int {
|
enum class APIError : int {
|
||||||
@@ -62,38 +57,119 @@ const char *api_error_to_str(APIError err);
|
|||||||
|
|
||||||
class APIFrameHelper {
|
class APIFrameHelper {
|
||||||
public:
|
public:
|
||||||
|
APIFrameHelper() = default;
|
||||||
|
explicit APIFrameHelper(std::unique_ptr<socket::Socket> socket) : socket_owned_(std::move(socket)) {
|
||||||
|
socket_ = socket_owned_.get();
|
||||||
|
}
|
||||||
virtual ~APIFrameHelper() = default;
|
virtual ~APIFrameHelper() = default;
|
||||||
virtual APIError init() = 0;
|
virtual APIError init() = 0;
|
||||||
virtual APIError loop() = 0;
|
virtual APIError loop() = 0;
|
||||||
virtual APIError read_packet(ReadPacketBuffer *buffer) = 0;
|
virtual APIError read_packet(ReadPacketBuffer *buffer) = 0;
|
||||||
virtual bool can_write_without_blocking() = 0;
|
bool can_write_without_blocking() { return state_ == State::DATA && tx_buf_.empty(); }
|
||||||
virtual APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) = 0;
|
std::string getpeername() { return socket_->getpeername(); }
|
||||||
virtual std::string getpeername() = 0;
|
int getpeername(struct sockaddr *addr, socklen_t *addrlen) { return socket_->getpeername(addr, addrlen); }
|
||||||
virtual int getpeername(struct sockaddr *addr, socklen_t *addrlen) = 0;
|
APIError close() {
|
||||||
virtual APIError close() = 0;
|
state_ = State::CLOSED;
|
||||||
virtual APIError shutdown(int how) = 0;
|
int err = this->socket_->close();
|
||||||
|
if (err == -1)
|
||||||
|
return APIError::CLOSE_FAILED;
|
||||||
|
return APIError::OK;
|
||||||
|
}
|
||||||
|
APIError shutdown(int how) {
|
||||||
|
int err = this->socket_->shutdown(how);
|
||||||
|
if (err == -1)
|
||||||
|
return APIError::SHUTDOWN_FAILED;
|
||||||
|
if (how == SHUT_RDWR) {
|
||||||
|
state_ = State::CLOSED;
|
||||||
|
}
|
||||||
|
return APIError::OK;
|
||||||
|
}
|
||||||
// Give this helper a name for logging
|
// Give this helper a name for logging
|
||||||
virtual void set_log_info(std::string info) = 0;
|
void set_log_info(std::string info) { info_ = std::move(info); }
|
||||||
|
virtual APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) = 0;
|
||||||
// Get the frame header padding required by this protocol
|
// Get the frame header padding required by this protocol
|
||||||
virtual uint8_t frame_header_padding() = 0;
|
virtual uint8_t frame_header_padding() = 0;
|
||||||
// Get the frame footer size required by this protocol
|
// Get the frame footer size required by this protocol
|
||||||
virtual uint8_t frame_footer_size() = 0;
|
virtual uint8_t frame_footer_size() = 0;
|
||||||
|
// Check if socket has data ready to read
|
||||||
|
bool is_socket_ready() const { return socket_ != nullptr && socket_->ready(); }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
// Struct for holding parsed frame data
|
||||||
|
struct ParsedFrame {
|
||||||
|
std::vector<uint8_t> msg;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Buffer containing data to be sent
|
||||||
|
struct SendBuffer {
|
||||||
|
std::vector<uint8_t> data;
|
||||||
|
uint16_t offset{0}; // Current offset within the buffer (uint16_t to reduce memory usage)
|
||||||
|
|
||||||
|
// Using uint16_t reduces memory usage since ESPHome API messages are limited to UINT16_MAX (65535) bytes
|
||||||
|
uint16_t remaining() const { return static_cast<uint16_t>(data.size()) - offset; }
|
||||||
|
const uint8_t *current_data() const { return data.data() + offset; }
|
||||||
|
};
|
||||||
|
|
||||||
|
// Queue of data buffers to be sent
|
||||||
|
std::deque<SendBuffer> tx_buf_;
|
||||||
|
|
||||||
|
// Common state enum for all frame helpers
|
||||||
|
// Note: Not all states are used by all implementations
|
||||||
|
// - INITIALIZE: Used by both Noise and Plaintext
|
||||||
|
// - CLIENT_HELLO, SERVER_HELLO, HANDSHAKE: Only used by Noise protocol
|
||||||
|
// - DATA: Used by both Noise and Plaintext
|
||||||
|
// - CLOSED: Used by both Noise and Plaintext
|
||||||
|
// - FAILED: Used by both Noise and Plaintext
|
||||||
|
// - EXPLICIT_REJECT: Only used by Noise protocol
|
||||||
|
enum class State {
|
||||||
|
INITIALIZE = 1,
|
||||||
|
CLIENT_HELLO = 2, // Noise only
|
||||||
|
SERVER_HELLO = 3, // Noise only
|
||||||
|
HANDSHAKE = 4, // Noise only
|
||||||
|
DATA = 5,
|
||||||
|
CLOSED = 6,
|
||||||
|
FAILED = 7,
|
||||||
|
EXPLICIT_REJECT = 8, // Noise only
|
||||||
|
};
|
||||||
|
|
||||||
|
// Current state of the frame helper
|
||||||
|
State state_{State::INITIALIZE};
|
||||||
|
|
||||||
|
// Helper name for logging
|
||||||
|
std::string info_;
|
||||||
|
|
||||||
|
// Socket for communication
|
||||||
|
socket::Socket *socket_{nullptr};
|
||||||
|
std::unique_ptr<socket::Socket> socket_owned_;
|
||||||
|
|
||||||
// Common implementation for writing raw data to socket
|
// Common implementation for writing raw data to socket
|
||||||
|
APIError write_raw_(const struct iovec *iov, int iovcnt);
|
||||||
|
|
||||||
|
// Try to send data from the tx buffer
|
||||||
|
APIError try_send_tx_buf_();
|
||||||
|
|
||||||
|
// Helper method to buffer data from IOVs
|
||||||
|
void buffer_data_from_iov_(const struct iovec *iov, int iovcnt, uint16_t total_write_len);
|
||||||
template<typename StateEnum>
|
template<typename StateEnum>
|
||||||
APIError write_raw_(const struct iovec *iov, int iovcnt, socket::Socket *socket, std::vector<uint8_t> &tx_buf,
|
APIError write_raw_(const struct iovec *iov, int iovcnt, socket::Socket *socket, std::vector<uint8_t> &tx_buf,
|
||||||
const std::string &info, StateEnum &state, StateEnum failed_state);
|
const std::string &info, StateEnum &state, StateEnum failed_state);
|
||||||
|
|
||||||
uint8_t frame_header_padding_{0};
|
uint8_t frame_header_padding_{0};
|
||||||
uint8_t frame_footer_size_{0};
|
uint8_t frame_footer_size_{0};
|
||||||
|
|
||||||
|
// Receive buffer for reading frame data
|
||||||
|
std::vector<uint8_t> rx_buf_;
|
||||||
|
uint16_t rx_buf_len_ = 0;
|
||||||
|
|
||||||
|
// Common initialization for both plaintext and noise protocols
|
||||||
|
APIError init_common_();
|
||||||
};
|
};
|
||||||
|
|
||||||
#ifdef USE_API_NOISE
|
#ifdef USE_API_NOISE
|
||||||
class APINoiseFrameHelper : public APIFrameHelper {
|
class APINoiseFrameHelper : public APIFrameHelper {
|
||||||
public:
|
public:
|
||||||
APINoiseFrameHelper(std::unique_ptr<socket::Socket> socket, std::shared_ptr<APINoiseContext> ctx)
|
APINoiseFrameHelper(std::unique_ptr<socket::Socket> socket, std::shared_ptr<APINoiseContext> ctx)
|
||||||
: socket_(std::move(socket)), ctx_(std::move(ctx)) {
|
: APIFrameHelper(std::move(socket)), ctx_(std::move(ctx)) {
|
||||||
// Noise header structure:
|
// Noise header structure:
|
||||||
// Pos 0: indicator (0x01)
|
// Pos 0: indicator (0x01)
|
||||||
// Pos 1-2: encrypted payload size (16-bit big-endian)
|
// Pos 1-2: encrypted payload size (16-bit big-endian)
|
||||||
@@ -105,49 +181,25 @@ class APINoiseFrameHelper : public APIFrameHelper {
|
|||||||
APIError init() override;
|
APIError init() override;
|
||||||
APIError loop() override;
|
APIError loop() override;
|
||||||
APIError read_packet(ReadPacketBuffer *buffer) override;
|
APIError read_packet(ReadPacketBuffer *buffer) override;
|
||||||
bool can_write_without_blocking() override;
|
|
||||||
APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) override;
|
APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) override;
|
||||||
std::string getpeername() override { return this->socket_->getpeername(); }
|
|
||||||
int getpeername(struct sockaddr *addr, socklen_t *addrlen) override {
|
|
||||||
return this->socket_->getpeername(addr, addrlen);
|
|
||||||
}
|
|
||||||
APIError close() override;
|
|
||||||
APIError shutdown(int how) override;
|
|
||||||
// Give this helper a name for logging
|
|
||||||
void set_log_info(std::string info) override { info_ = std::move(info); }
|
|
||||||
// Get the frame header padding required by this protocol
|
// Get the frame header padding required by this protocol
|
||||||
uint8_t frame_header_padding() override { return frame_header_padding_; }
|
uint8_t frame_header_padding() override { return frame_header_padding_; }
|
||||||
// Get the frame footer size required by this protocol
|
// Get the frame footer size required by this protocol
|
||||||
uint8_t frame_footer_size() override { return frame_footer_size_; }
|
uint8_t frame_footer_size() override { return frame_footer_size_; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
struct ParsedFrame {
|
|
||||||
std::vector<uint8_t> msg;
|
|
||||||
};
|
|
||||||
|
|
||||||
APIError state_action_();
|
APIError state_action_();
|
||||||
APIError try_read_frame_(ParsedFrame *frame);
|
APIError try_read_frame_(ParsedFrame *frame);
|
||||||
APIError try_send_tx_buf_();
|
APIError write_frame_(const uint8_t *data, uint16_t len);
|
||||||
APIError write_frame_(const uint8_t *data, size_t len);
|
|
||||||
inline APIError write_raw_(const struct iovec *iov, int iovcnt) {
|
|
||||||
return APIFrameHelper::write_raw_(iov, iovcnt, socket_.get(), tx_buf_, info_, state_, State::FAILED);
|
|
||||||
}
|
|
||||||
APIError init_handshake_();
|
APIError init_handshake_();
|
||||||
APIError check_handshake_finished_();
|
APIError check_handshake_finished_();
|
||||||
void send_explicit_handshake_reject_(const std::string &reason);
|
void send_explicit_handshake_reject_(const std::string &reason);
|
||||||
|
|
||||||
std::unique_ptr<socket::Socket> socket_;
|
|
||||||
|
|
||||||
std::string info_;
|
|
||||||
// Fixed-size header buffer for noise protocol:
|
// Fixed-size header buffer for noise protocol:
|
||||||
// 1 byte for indicator + 2 bytes for message size (16-bit value, not varint)
|
// 1 byte for indicator + 2 bytes for message size (16-bit value, not varint)
|
||||||
// Note: Maximum message size is 65535, with a limit of 128 bytes during handshake phase
|
// Note: Maximum message size is UINT16_MAX (65535), with a limit of 128 bytes during handshake phase
|
||||||
uint8_t rx_header_buf_[3];
|
uint8_t rx_header_buf_[3];
|
||||||
size_t rx_header_buf_len_ = 0;
|
uint8_t rx_header_buf_len_ = 0;
|
||||||
std::vector<uint8_t> rx_buf_;
|
|
||||||
size_t rx_buf_len_ = 0;
|
|
||||||
|
|
||||||
std::vector<uint8_t> tx_buf_;
|
|
||||||
std::vector<uint8_t> prologue_;
|
std::vector<uint8_t> prologue_;
|
||||||
|
|
||||||
std::shared_ptr<APINoiseContext> ctx_;
|
std::shared_ptr<APINoiseContext> ctx_;
|
||||||
@@ -155,24 +207,13 @@ class APINoiseFrameHelper : public APIFrameHelper {
|
|||||||
NoiseCipherState *send_cipher_{nullptr};
|
NoiseCipherState *send_cipher_{nullptr};
|
||||||
NoiseCipherState *recv_cipher_{nullptr};
|
NoiseCipherState *recv_cipher_{nullptr};
|
||||||
NoiseProtocolId nid_;
|
NoiseProtocolId nid_;
|
||||||
|
|
||||||
enum class State {
|
|
||||||
INITIALIZE = 1,
|
|
||||||
CLIENT_HELLO = 2,
|
|
||||||
SERVER_HELLO = 3,
|
|
||||||
HANDSHAKE = 4,
|
|
||||||
DATA = 5,
|
|
||||||
CLOSED = 6,
|
|
||||||
FAILED = 7,
|
|
||||||
EXPLICIT_REJECT = 8,
|
|
||||||
} state_ = State::INITIALIZE;
|
|
||||||
};
|
};
|
||||||
#endif // USE_API_NOISE
|
#endif // USE_API_NOISE
|
||||||
|
|
||||||
#ifdef USE_API_PLAINTEXT
|
#ifdef USE_API_PLAINTEXT
|
||||||
class APIPlaintextFrameHelper : public APIFrameHelper {
|
class APIPlaintextFrameHelper : public APIFrameHelper {
|
||||||
public:
|
public:
|
||||||
APIPlaintextFrameHelper(std::unique_ptr<socket::Socket> socket) : socket_(std::move(socket)) {
|
APIPlaintextFrameHelper(std::unique_ptr<socket::Socket> socket) : APIFrameHelper(std::move(socket)) {
|
||||||
// Plaintext header structure (worst case):
|
// Plaintext header structure (worst case):
|
||||||
// Pos 0: indicator (0x00)
|
// Pos 0: indicator (0x00)
|
||||||
// Pos 1-3: payload size varint (up to 3 bytes)
|
// Pos 1-3: payload size varint (up to 3 bytes)
|
||||||
@@ -184,60 +225,26 @@ class APIPlaintextFrameHelper : public APIFrameHelper {
|
|||||||
APIError init() override;
|
APIError init() override;
|
||||||
APIError loop() override;
|
APIError loop() override;
|
||||||
APIError read_packet(ReadPacketBuffer *buffer) override;
|
APIError read_packet(ReadPacketBuffer *buffer) override;
|
||||||
bool can_write_without_blocking() override;
|
|
||||||
APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) override;
|
APIError write_protobuf_packet(uint16_t type, ProtoWriteBuffer buffer) override;
|
||||||
std::string getpeername() override { return this->socket_->getpeername(); }
|
|
||||||
int getpeername(struct sockaddr *addr, socklen_t *addrlen) override {
|
|
||||||
return this->socket_->getpeername(addr, addrlen);
|
|
||||||
}
|
|
||||||
APIError close() override;
|
|
||||||
APIError shutdown(int how) override;
|
|
||||||
// Give this helper a name for logging
|
|
||||||
void set_log_info(std::string info) override { info_ = std::move(info); }
|
|
||||||
// Get the frame header padding required by this protocol
|
|
||||||
uint8_t frame_header_padding() override { return frame_header_padding_; }
|
uint8_t frame_header_padding() override { return frame_header_padding_; }
|
||||||
// Get the frame footer size required by this protocol
|
// Get the frame footer size required by this protocol
|
||||||
uint8_t frame_footer_size() override { return frame_footer_size_; }
|
uint8_t frame_footer_size() override { return frame_footer_size_; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
struct ParsedFrame {
|
|
||||||
std::vector<uint8_t> msg;
|
|
||||||
};
|
|
||||||
|
|
||||||
APIError try_read_frame_(ParsedFrame *frame);
|
APIError try_read_frame_(ParsedFrame *frame);
|
||||||
APIError try_send_tx_buf_();
|
|
||||||
inline APIError write_raw_(const struct iovec *iov, int iovcnt) {
|
|
||||||
return APIFrameHelper::write_raw_(iov, iovcnt, socket_.get(), tx_buf_, info_, state_, State::FAILED);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::unique_ptr<socket::Socket> socket_;
|
|
||||||
|
|
||||||
std::string info_;
|
|
||||||
// Fixed-size header buffer for plaintext protocol:
|
// Fixed-size header buffer for plaintext protocol:
|
||||||
// We only need space for the two varints since we validate the indicator byte separately.
|
// We now store the indicator byte + the two varints.
|
||||||
// To match noise protocol's maximum message size (65535), we need:
|
// To match noise protocol's maximum message size (UINT16_MAX = 65535), we need:
|
||||||
// 3 bytes for message size varint (supports up to 2097151) + 2 bytes for message type varint
|
// 1 byte for indicator + 3 bytes for message size varint (supports up to 2097151) + 2 bytes for message type varint
|
||||||
//
|
//
|
||||||
// While varints could theoretically be up to 10 bytes each for 64-bit values,
|
// While varints could theoretically be up to 10 bytes each for 64-bit values,
|
||||||
// attempting to process messages with headers that large would likely crash the
|
// attempting to process messages with headers that large would likely crash the
|
||||||
// ESP32 due to memory constraints.
|
// ESP32 due to memory constraints.
|
||||||
uint8_t rx_header_buf_[5]; // 5 bytes for varints (3 for size + 2 for type)
|
uint8_t rx_header_buf_[6]; // 1 byte indicator + 5 bytes for varints (3 for size + 2 for type)
|
||||||
uint8_t rx_header_buf_pos_ = 0;
|
uint8_t rx_header_buf_pos_ = 0;
|
||||||
bool rx_header_parsed_ = false;
|
bool rx_header_parsed_ = false;
|
||||||
uint32_t rx_header_parsed_type_ = 0;
|
uint16_t rx_header_parsed_type_ = 0;
|
||||||
uint32_t rx_header_parsed_len_ = 0;
|
uint16_t rx_header_parsed_len_ = 0;
|
||||||
|
|
||||||
std::vector<uint8_t> rx_buf_;
|
|
||||||
size_t rx_buf_len_ = 0;
|
|
||||||
|
|
||||||
std::vector<uint8_t> tx_buf_;
|
|
||||||
|
|
||||||
enum class State {
|
|
||||||
INITIALIZE = 1,
|
|
||||||
DATA = 2,
|
|
||||||
CLOSED = 3,
|
|
||||||
FAILED = 4,
|
|
||||||
} state_ = State::INITIALIZE;
|
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@@ -27,7 +27,7 @@ APIServer *global_api_server = nullptr; // NOLINT(cppcoreguidelines-avoid-non-c
|
|||||||
APIServer::APIServer() { global_api_server = this; }
|
APIServer::APIServer() { global_api_server = this; }
|
||||||
|
|
||||||
void APIServer::setup() {
|
void APIServer::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up Home Assistant API server...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->setup_controller();
|
this->setup_controller();
|
||||||
|
|
||||||
#ifdef USE_API_NOISE
|
#ifdef USE_API_NOISE
|
||||||
@@ -43,7 +43,7 @@ void APIServer::setup() {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
this->socket_ = socket::socket_ip(SOCK_STREAM, 0);
|
this->socket_ = socket::socket_ip_loop_monitored(SOCK_STREAM, 0); // monitored for incoming connections
|
||||||
if (this->socket_ == nullptr) {
|
if (this->socket_ == nullptr) {
|
||||||
ESP_LOGW(TAG, "Could not create socket");
|
ESP_LOGW(TAG, "Could not create socket");
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -112,11 +112,12 @@ void APIServer::setup() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void APIServer::loop() {
|
void APIServer::loop() {
|
||||||
// Accept new clients
|
// Accept new clients only if the socket has incoming connections
|
||||||
|
if (this->socket_->ready()) {
|
||||||
while (true) {
|
while (true) {
|
||||||
struct sockaddr_storage source_addr;
|
struct sockaddr_storage source_addr;
|
||||||
socklen_t addr_len = sizeof(source_addr);
|
socklen_t addr_len = sizeof(source_addr);
|
||||||
auto sock = this->socket_->accept((struct sockaddr *) &source_addr, &addr_len);
|
auto sock = this->socket_->accept_loop_monitored((struct sockaddr *) &source_addr, &addr_len);
|
||||||
if (!sock)
|
if (!sock)
|
||||||
break;
|
break;
|
||||||
ESP_LOGD(TAG, "Accepted %s", sock->getpeername().c_str());
|
ESP_LOGD(TAG, "Accepted %s", sock->getpeername().c_str());
|
||||||
@@ -125,6 +126,7 @@ void APIServer::loop() {
|
|||||||
this->clients_.emplace_back(conn);
|
this->clients_.emplace_back(conn);
|
||||||
conn->start();
|
conn->start();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Process clients and remove disconnected ones in a single pass
|
// Process clients and remove disconnected ones in a single pass
|
||||||
if (!this->clients_.empty()) {
|
if (!this->clients_.empty()) {
|
||||||
@@ -155,7 +157,7 @@ void APIServer::loop() {
|
|||||||
const uint32_t now = millis();
|
const uint32_t now = millis();
|
||||||
if (!this->is_connected()) {
|
if (!this->is_connected()) {
|
||||||
if (now - this->last_connected_ > this->reboot_timeout_) {
|
if (now - this->last_connected_ > this->reboot_timeout_) {
|
||||||
ESP_LOGE(TAG, "No client connected to API. Rebooting...");
|
ESP_LOGE(TAG, "No client connected; rebooting");
|
||||||
App.reboot();
|
App.reboot();
|
||||||
}
|
}
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
|
@@ -55,6 +55,7 @@ class ProtoVarInt {
|
|||||||
return {}; // Incomplete or invalid varint
|
return {}; // Incomplete or invalid varint
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint16_t as_uint16() const { return this->value_; }
|
||||||
uint32_t as_uint32() const { return this->value_; }
|
uint32_t as_uint32() const { return this->value_; }
|
||||||
uint64_t as_uint64() const { return this->value_; }
|
uint64_t as_uint64() const { return this->value_; }
|
||||||
bool as_bool() const { return this->value_; }
|
bool as_bool() const { return this->value_; }
|
||||||
|
@@ -7,7 +7,7 @@ namespace as3935 {
|
|||||||
static const char *const TAG = "as3935";
|
static const char *const TAG = "as3935";
|
||||||
|
|
||||||
void AS3935Component::setup() {
|
void AS3935Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AS3935...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
this->irq_pin_->setup();
|
this->irq_pin_->setup();
|
||||||
LOG_PIN(" IRQ Pin: ", this->irq_pin_);
|
LOG_PIN(" IRQ Pin: ", this->irq_pin_);
|
||||||
|
@@ -23,7 +23,7 @@ static const uint8_t REGISTER_AGC = 0x1A; // 8 bytes / R
|
|||||||
static const uint8_t REGISTER_MAGNITUDE = 0x1B; // 16 bytes / R
|
static const uint8_t REGISTER_MAGNITUDE = 0x1B; // 16 bytes / R
|
||||||
|
|
||||||
void AS5600Component::setup() {
|
void AS5600Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AS5600...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
if (!this->read_byte(REGISTER_STATUS).has_value()) {
|
if (!this->read_byte(REGISTER_STATUS).has_value()) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -91,7 +91,7 @@ void AS5600Component::dump_config() {
|
|||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
|
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AS5600 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -8,7 +8,7 @@ namespace as7341 {
|
|||||||
static const char *const TAG = "as7341";
|
static const char *const TAG = "as7341";
|
||||||
|
|
||||||
void AS7341Component::setup() {
|
void AS7341Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AS7341...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
|
|
||||||
// Verify device ID
|
// Verify device ID
|
||||||
@@ -38,7 +38,7 @@ void AS7341Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "AS7341:");
|
ESP_LOGCONFIG(TAG, "AS7341:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with AS7341 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
ESP_LOGCONFIG(TAG, " Gain: %u", get_gain());
|
ESP_LOGCONFIG(TAG, " Gain: %u", get_gain());
|
||||||
|
@@ -71,7 +71,7 @@ bool AT581XComponent::i2c_read_reg(uint8_t addr, uint8_t &data) {
|
|||||||
return this->read_register(addr, &data, 1) == esphome::i2c::NO_ERROR;
|
return this->read_register(addr, &data, 1) == esphome::i2c::NO_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
void AT581XComponent::setup() { ESP_LOGCONFIG(TAG, "Setting up AT581X..."); }
|
void AT581XComponent::setup() { ESP_LOGCONFIG(TAG, "Running setup"); }
|
||||||
void AT581XComponent::dump_config() { LOG_I2C_DEVICE(this); }
|
void AT581XComponent::dump_config() { LOG_I2C_DEVICE(this); }
|
||||||
#define ARRAY_SIZE(X) (sizeof(X) / sizeof((X)[0]))
|
#define ARRAY_SIZE(X) (sizeof(X) / sizeof((X)[0]))
|
||||||
bool AT581XComponent::i2c_write_config() {
|
bool AT581XComponent::i2c_write_config() {
|
||||||
|
@@ -41,7 +41,7 @@ void ATM90E26Component::update() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ATM90E26Component::setup() {
|
void ATM90E26Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ATM90E26 Component...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->spi_setup();
|
this->spi_setup();
|
||||||
|
|
||||||
uint16_t mmode = 0x422; // default values for everything but L/N line current gains
|
uint16_t mmode = 0x422; // default values for everything but L/N line current gains
|
||||||
@@ -135,7 +135,7 @@ void ATM90E26Component::dump_config() {
|
|||||||
ESP_LOGCONFIG("", "ATM90E26:");
|
ESP_LOGCONFIG("", "ATM90E26:");
|
||||||
LOG_PIN(" CS Pin: ", this->cs_);
|
LOG_PIN(" CS Pin: ", this->cs_);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with ATM90E26 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
LOG_SENSOR(" ", "Voltage A", this->voltage_sensor_);
|
LOG_SENSOR(" ", "Voltage A", this->voltage_sensor_);
|
||||||
|
@@ -108,7 +108,7 @@ void ATM90E32Component::update() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ATM90E32Component::setup() {
|
void ATM90E32Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ATM90E32 Component...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->spi_setup();
|
this->spi_setup();
|
||||||
|
|
||||||
uint16_t mmode0 = 0x87; // 3P4W 50Hz
|
uint16_t mmode0 = 0x87; // 3P4W 50Hz
|
||||||
@@ -217,7 +217,7 @@ void ATM90E32Component::dump_config() {
|
|||||||
ESP_LOGCONFIG("", "ATM90E32:");
|
ESP_LOGCONFIG("", "ATM90E32:");
|
||||||
LOG_PIN(" CS Pin: ", this->cs_);
|
LOG_PIN(" CS Pin: ", this->cs_);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with ATM90E32 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
LOG_SENSOR(" ", "Voltage A", this->phase_[PHASEA].voltage_sensor_);
|
LOG_SENSOR(" ", "Voltage A", this->phase_[PHASEA].voltage_sensor_);
|
||||||
|
@@ -17,7 +17,7 @@ constexpr static const uint8_t AXS_READ_TOUCHPAD[11] = {0xb5, 0xab, 0xa5, 0x5a,
|
|||||||
}
|
}
|
||||||
|
|
||||||
void AXS15231Touchscreen::setup() {
|
void AXS15231Touchscreen::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up AXS15231 Touchscreen...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
if (this->reset_pin_ != nullptr) {
|
if (this->reset_pin_ != nullptr) {
|
||||||
this->reset_pin_->setup();
|
this->reset_pin_->setup();
|
||||||
this->reset_pin_->digital_write(false);
|
this->reset_pin_->digital_write(false);
|
||||||
|
@@ -119,7 +119,7 @@ void spi_dma_tx_finish_callback(unsigned int param) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BekenSPILEDStripLightOutput::setup() {
|
void BekenSPILEDStripLightOutput::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up Beken SPI LED Strip...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
size_t buffer_size = this->get_buffer_size_();
|
size_t buffer_size = this->get_buffer_size_();
|
||||||
size_t dma_buffer_size = (buffer_size * 8) + (2 * 64);
|
size_t dma_buffer_size = (buffer_size * 8) + (2 * 64);
|
||||||
|
@@ -38,7 +38,7 @@ MTreg:
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
void BH1750Sensor::setup() {
|
void BH1750Sensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BH1750 '%s'...", this->name_.c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'", this->name_.c_str());
|
||||||
uint8_t turn_on = BH1750_COMMAND_POWER_ON;
|
uint8_t turn_on = BH1750_COMMAND_POWER_ON;
|
||||||
if (this->write(&turn_on, 1) != i2c::ERROR_OK) {
|
if (this->write(&turn_on, 1) != i2c::ERROR_OK) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -118,7 +118,7 @@ void BH1750Sensor::dump_config() {
|
|||||||
LOG_SENSOR("", "BH1750", this);
|
LOG_SENSOR("", "BH1750", this);
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with BH1750 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL_FOR, this->get_name().c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
|
@@ -9,6 +9,7 @@ from esphome.const import (
|
|||||||
CONF_ID,
|
CONF_ID,
|
||||||
CONF_LINE_FREQUENCY,
|
CONF_LINE_FREQUENCY,
|
||||||
CONF_POWER,
|
CONF_POWER,
|
||||||
|
CONF_RESET,
|
||||||
CONF_VOLTAGE,
|
CONF_VOLTAGE,
|
||||||
DEVICE_CLASS_CURRENT,
|
DEVICE_CLASS_CURRENT,
|
||||||
DEVICE_CLASS_ENERGY,
|
DEVICE_CLASS_ENERGY,
|
||||||
@@ -27,7 +28,6 @@ from esphome.const import (
|
|||||||
CONF_CURRENT_REFERENCE = "current_reference"
|
CONF_CURRENT_REFERENCE = "current_reference"
|
||||||
CONF_ENERGY_REFERENCE = "energy_reference"
|
CONF_ENERGY_REFERENCE = "energy_reference"
|
||||||
CONF_POWER_REFERENCE = "power_reference"
|
CONF_POWER_REFERENCE = "power_reference"
|
||||||
CONF_RESET = "reset"
|
|
||||||
CONF_VOLTAGE_REFERENCE = "voltage_reference"
|
CONF_VOLTAGE_REFERENCE = "voltage_reference"
|
||||||
|
|
||||||
DEPENDENCIES = ["uart"]
|
DEPENDENCIES = ["uart"]
|
||||||
|
@@ -88,7 +88,7 @@ const char *oversampling_to_str(BME280Oversampling oversampling) { // NOLINT
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BME280Component::setup() {
|
void BME280Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BME280...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t chip_id = 0;
|
uint8_t chip_id = 0;
|
||||||
|
|
||||||
// Mark as not failed before initializing. Some devices will turn off sensors to save on batteries
|
// Mark as not failed before initializing. Some devices will turn off sensors to save on batteries
|
||||||
@@ -182,7 +182,7 @@ void BME280Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "BME280:");
|
ESP_LOGCONFIG(TAG, "BME280:");
|
||||||
switch (this->error_code_) {
|
switch (this->error_code_) {
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication with BME280 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case WRONG_CHIP_ID:
|
case WRONG_CHIP_ID:
|
||||||
ESP_LOGE(TAG, "BME280 has wrong chip ID! Is it a BME280?");
|
ESP_LOGE(TAG, "BME280 has wrong chip ID! Is it a BME280?");
|
||||||
|
@@ -71,7 +71,7 @@ static const char *iir_filter_to_str(BME680IIRFilter filter) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BME680Component::setup() {
|
void BME680Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BME680...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t chip_id;
|
uint8_t chip_id;
|
||||||
if (!this->read_byte(BME680_REGISTER_CHIPID, &chip_id) || chip_id != 0x61) {
|
if (!this->read_byte(BME680_REGISTER_CHIPID, &chip_id) || chip_id != 0x61) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -215,7 +215,7 @@ void BME680Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "BME680:");
|
ESP_LOGCONFIG(TAG, "BME680:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with BME680 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
ESP_LOGCONFIG(TAG, " IIR Filter: %s", iir_filter_to_str(this->iir_filter_));
|
ESP_LOGCONFIG(TAG, " IIR Filter: %s", iir_filter_to_str(this->iir_filter_));
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
@@ -307,7 +307,7 @@ void BME680Component::read_data_() {
|
|||||||
this->humidity_sensor_->publish_state(NAN);
|
this->humidity_sensor_->publish_state(NAN);
|
||||||
if (this->gas_resistance_sensor_ != nullptr)
|
if (this->gas_resistance_sensor_ != nullptr)
|
||||||
this->gas_resistance_sensor_->publish_state(NAN);
|
this->gas_resistance_sensor_->publish_state(NAN);
|
||||||
ESP_LOGW(TAG, "Communication with BME680 failed!");
|
ESP_LOGW(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@@ -15,7 +15,7 @@ std::vector<BME680BSECComponent *>
|
|||||||
uint8_t BME680BSECComponent::work_buffer_[BSEC_MAX_WORKBUFFER_SIZE] = {0};
|
uint8_t BME680BSECComponent::work_buffer_[BSEC_MAX_WORKBUFFER_SIZE] = {0};
|
||||||
|
|
||||||
void BME680BSECComponent::setup() {
|
void BME680BSECComponent::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BME680(%s) via BSEC...", this->device_id_.c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'", this->device_id_.c_str());
|
||||||
|
|
||||||
uint8_t new_idx = BME680BSECComponent::instances.size();
|
uint8_t new_idx = BME680BSECComponent::instances.size();
|
||||||
BME680BSECComponent::instances.push_back(this);
|
BME680BSECComponent::instances.push_back(this);
|
||||||
|
@@ -16,7 +16,7 @@ CODEOWNERS = ["@neffs", "@kbx81"]
|
|||||||
|
|
||||||
DOMAIN = "bme68x_bsec2"
|
DOMAIN = "bme68x_bsec2"
|
||||||
|
|
||||||
BSEC2_LIBRARY_VERSION = "v1.8.2610"
|
BSEC2_LIBRARY_VERSION = "1.10.2610"
|
||||||
|
|
||||||
CONF_ALGORITHM_OUTPUT = "algorithm_output"
|
CONF_ALGORITHM_OUTPUT = "algorithm_output"
|
||||||
CONF_BME68X_BSEC2_ID = "bme68x_bsec2_id"
|
CONF_BME68X_BSEC2_ID = "bme68x_bsec2_id"
|
||||||
@@ -145,7 +145,6 @@ CONFIG_SCHEMA_BASE = (
|
|||||||
): cv.positive_time_period_minutes,
|
): cv.positive_time_period_minutes,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
.add_extra(cv.only_with_arduino)
|
|
||||||
.add_extra(validate_bme68x)
|
.add_extra(validate_bme68x)
|
||||||
.add_extra(download_bme68x_blob)
|
.add_extra(download_bme68x_blob)
|
||||||
)
|
)
|
||||||
@@ -179,11 +178,13 @@ async def to_code_base(config):
|
|||||||
bsec2_arr = cg.progmem_array(config[CONF_RAW_DATA_ID], rhs)
|
bsec2_arr = cg.progmem_array(config[CONF_RAW_DATA_ID], rhs)
|
||||||
cg.add(var.set_bsec2_configuration(bsec2_arr, len(rhs)))
|
cg.add(var.set_bsec2_configuration(bsec2_arr, len(rhs)))
|
||||||
|
|
||||||
# Although this component does not use SPI, the BSEC2 library requires the SPI library
|
# Although this component does not use SPI, the BSEC2 Arduino library requires the SPI library
|
||||||
|
if core.CORE.using_arduino:
|
||||||
cg.add_library("SPI", None)
|
cg.add_library("SPI", None)
|
||||||
cg.add_library(
|
cg.add_library(
|
||||||
"BME68x Sensor library",
|
"BME68x Sensor library",
|
||||||
"1.1.40407",
|
"1.3.40408",
|
||||||
|
"https://github.com/boschsensortec/Bosch-BME68x-Library",
|
||||||
)
|
)
|
||||||
cg.add_library(
|
cg.add_library(
|
||||||
"BSEC2 Software Library",
|
"BSEC2 Software Library",
|
||||||
|
@@ -1,4 +1,5 @@
|
|||||||
#include "esphome/core/defines.h"
|
#include "esphome/core/defines.h"
|
||||||
|
#include "esphome/core/hal.h"
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
@@ -20,7 +21,7 @@ static const char *const TAG = "bme68x_bsec2.sensor";
|
|||||||
static const std::string IAQ_ACCURACY_STATES[4] = {"Stabilizing", "Uncertain", "Calibrating", "Calibrated"};
|
static const std::string IAQ_ACCURACY_STATES[4] = {"Stabilizing", "Uncertain", "Calibrating", "Calibrated"};
|
||||||
|
|
||||||
void BME68xBSEC2Component::setup() {
|
void BME68xBSEC2Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BME68X via BSEC2...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
this->bsec_status_ = bsec_init_m(&this->bsec_instance_);
|
this->bsec_status_ = bsec_init_m(&this->bsec_instance_);
|
||||||
if (this->bsec_status_ != BSEC_OK) {
|
if (this->bsec_status_ != BSEC_OK) {
|
||||||
|
@@ -1,4 +1,5 @@
|
|||||||
#include "esphome/core/defines.h"
|
#include "esphome/core/defines.h"
|
||||||
|
#include "esphome/core/hal.h"
|
||||||
#include "esphome/core/helpers.h"
|
#include "esphome/core/helpers.h"
|
||||||
#include "esphome/core/log.h"
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
|
@@ -119,7 +119,7 @@ const float GRAVITY_EARTH = 9.80665f;
|
|||||||
void BMI160Component::internal_setup_(int stage) {
|
void BMI160Component::internal_setup_(int stage) {
|
||||||
switch (stage) {
|
switch (stage) {
|
||||||
case 0:
|
case 0:
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BMI160...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t chipid;
|
uint8_t chipid;
|
||||||
if (!this->read_byte(BMI160_REGISTER_CHIPID, &chipid) || (chipid != 0b11010001)) {
|
if (!this->read_byte(BMI160_REGISTER_CHIPID, &chipid) || (chipid != 0b11010001)) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -189,7 +189,7 @@ void BMI160Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "BMI160:");
|
ESP_LOGCONFIG(TAG, "BMI160:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with BMI160 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
LOG_SENSOR(" ", "Acceleration X", this->accel_x_sensor_);
|
LOG_SENSOR(" ", "Acceleration X", this->accel_x_sensor_);
|
||||||
|
@@ -20,7 +20,7 @@ void BMP085Component::update() {
|
|||||||
this->set_timeout("temperature", 5, [this]() { this->read_temperature_(); });
|
this->set_timeout("temperature", 5, [this]() { this->read_temperature_(); });
|
||||||
}
|
}
|
||||||
void BMP085Component::setup() {
|
void BMP085Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BMP085...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t data[22];
|
uint8_t data[22];
|
||||||
if (!this->read_bytes(BMP085_REGISTER_AC1_H, data, 22)) {
|
if (!this->read_bytes(BMP085_REGISTER_AC1_H, data, 22)) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
|
@@ -57,7 +57,7 @@ static const char *iir_filter_to_str(BMP280IIRFilter filter) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void BMP280Component::setup() {
|
void BMP280Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BMP280...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t chip_id = 0;
|
uint8_t chip_id = 0;
|
||||||
|
|
||||||
// Read the chip id twice, to work around a bug where the first read is 0.
|
// Read the chip id twice, to work around a bug where the first read is 0.
|
||||||
@@ -132,7 +132,7 @@ void BMP280Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "BMP280:");
|
ESP_LOGCONFIG(TAG, "BMP280:");
|
||||||
switch (this->error_code_) {
|
switch (this->error_code_) {
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication with BMP280 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case WRONG_CHIP_ID:
|
case WRONG_CHIP_ID:
|
||||||
ESP_LOGE(TAG, "BMP280 has wrong chip ID! Is it a BME280?");
|
ESP_LOGE(TAG, "BMP280 has wrong chip ID! Is it a BME280?");
|
||||||
|
@@ -70,7 +70,7 @@ static const LogString *iir_filter_to_str(IIRFilter filter) {
|
|||||||
|
|
||||||
void BMP3XXComponent::setup() {
|
void BMP3XXComponent::setup() {
|
||||||
this->error_code_ = NONE;
|
this->error_code_ = NONE;
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BMP3XX...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
// Call the Device base class "initialise" function
|
// Call the Device base class "initialise" function
|
||||||
if (!reset()) {
|
if (!reset()) {
|
||||||
ESP_LOGE(TAG, "Failed to reset BMP3XX...");
|
ESP_LOGE(TAG, "Failed to reset BMP3XX...");
|
||||||
@@ -154,7 +154,7 @@ void BMP3XXComponent::dump_config() {
|
|||||||
case NONE:
|
case NONE:
|
||||||
break;
|
break;
|
||||||
case ERROR_COMMUNICATION_FAILED:
|
case ERROR_COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication with BMP3XX failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case ERROR_WRONG_CHIP_ID:
|
case ERROR_WRONG_CHIP_ID:
|
||||||
ESP_LOGE(
|
ESP_LOGE(
|
||||||
|
@@ -122,7 +122,7 @@ void BMP581Component::setup() {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
this->error_code_ = NONE;
|
this->error_code_ = NONE;
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BMP581...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
////////////////////
|
////////////////////
|
||||||
// 1) Soft reboot //
|
// 1) Soft reboot //
|
||||||
|
@@ -15,7 +15,7 @@ static const uint8_t BP1658CJ_ADDR_START_5CH = 0x30;
|
|||||||
static const uint8_t BP1658CJ_DELAY = 2;
|
static const uint8_t BP1658CJ_DELAY = 2;
|
||||||
|
|
||||||
void BP1658CJ::setup() {
|
void BP1658CJ::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BP1658CJ Output Component...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->data_pin_->setup();
|
this->data_pin_->setup();
|
||||||
this->data_pin_->digital_write(false);
|
this->data_pin_->digital_write(false);
|
||||||
this->clock_pin_->setup();
|
this->clock_pin_->setup();
|
||||||
|
@@ -20,7 +20,7 @@ static const uint8_t BP5758D_ALL_DATA_CHANNEL_ENABLEMENT = 0b00011111;
|
|||||||
static const uint8_t BP5758D_DELAY = 2;
|
static const uint8_t BP5758D_DELAY = 2;
|
||||||
|
|
||||||
void BP5758D::setup() {
|
void BP5758D::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up BP5758D Output Component...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->data_pin_->setup();
|
this->data_pin_->setup();
|
||||||
this->data_pin_->digital_write(false);
|
this->data_pin_->digital_write(false);
|
||||||
delayMicroseconds(BP5758D_DELAY);
|
delayMicroseconds(BP5758D_DELAY);
|
||||||
|
@@ -7,7 +7,7 @@ namespace canbus {
|
|||||||
static const char *const TAG = "canbus";
|
static const char *const TAG = "canbus";
|
||||||
|
|
||||||
void Canbus::setup() {
|
void Canbus::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up Canbus...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
if (!this->setup_internal()) {
|
if (!this->setup_internal()) {
|
||||||
ESP_LOGE(TAG, "setup error!");
|
ESP_LOGE(TAG, "setup error!");
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
|
@@ -8,7 +8,7 @@ namespace cap1188 {
|
|||||||
static const char *const TAG = "cap1188";
|
static const char *const TAG = "cap1188";
|
||||||
|
|
||||||
void CAP1188Component::setup() {
|
void CAP1188Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up CAP1188...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
// Reset device using the reset pin
|
// Reset device using the reset pin
|
||||||
if (this->reset_pin_ != nullptr) {
|
if (this->reset_pin_ != nullptr) {
|
||||||
|
@@ -29,7 +29,7 @@ void CaptivePortal::handle_config(AsyncWebServerRequest *request) {
|
|||||||
void CaptivePortal::handle_wifisave(AsyncWebServerRequest *request) {
|
void CaptivePortal::handle_wifisave(AsyncWebServerRequest *request) {
|
||||||
std::string ssid = request->arg("ssid").c_str();
|
std::string ssid = request->arg("ssid").c_str();
|
||||||
std::string psk = request->arg("psk").c_str();
|
std::string psk = request->arg("psk").c_str();
|
||||||
ESP_LOGI(TAG, "Captive Portal Requested WiFi Settings Change:");
|
ESP_LOGI(TAG, "Requested WiFi Settings Change:");
|
||||||
ESP_LOGI(TAG, " SSID='%s'", ssid.c_str());
|
ESP_LOGI(TAG, " SSID='%s'", ssid.c_str());
|
||||||
ESP_LOGI(TAG, " Password=" LOG_SECRET("'%s'"), psk.c_str());
|
ESP_LOGI(TAG, " Password=" LOG_SECRET("'%s'"), psk.c_str());
|
||||||
wifi::global_wifi_component->save_wifi_sta(ssid, psk);
|
wifi::global_wifi_component->save_wifi_sta(ssid, psk);
|
||||||
|
@@ -163,7 +163,7 @@ void CCS811Component::dump_config() {
|
|||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
switch (this->error_code_) {
|
switch (this->error_code_) {
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGW(TAG, "Communication failed! Is the sensor connected?");
|
ESP_LOGW(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case INVALID_ID:
|
case INVALID_ID:
|
||||||
ESP_LOGW(TAG, "Sensor reported an invalid ID. Is this a CCS811?");
|
ESP_LOGW(TAG, "Sensor reported an invalid ID. Is this a CCS811?");
|
||||||
|
@@ -10,7 +10,7 @@ static const char *const TAG = "cd74hc4067";
|
|||||||
float CD74HC4067Component::get_setup_priority() const { return setup_priority::DATA; }
|
float CD74HC4067Component::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void CD74HC4067Component::setup() {
|
void CD74HC4067Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up CD74HC4067...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
this->pin_s0_->setup();
|
this->pin_s0_->setup();
|
||||||
this->pin_s1_->setup();
|
this->pin_s1_->setup();
|
||||||
|
@@ -14,7 +14,7 @@ static const uint8_t CH422G_REG_OUT_UPPER = 0x23; // write reg for output bit
|
|||||||
static const char *const TAG = "ch422g";
|
static const char *const TAG = "ch422g";
|
||||||
|
|
||||||
void CH422GComponent::setup() {
|
void CH422GComponent::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up CH422G...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
// set outputs before mode
|
// set outputs before mode
|
||||||
this->write_outputs_();
|
this->write_outputs_();
|
||||||
// Set mode and check for errors
|
// Set mode and check for errors
|
||||||
@@ -37,7 +37,7 @@ void CH422GComponent::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "CH422G:");
|
ESP_LOGCONFIG(TAG, "CH422G:");
|
||||||
LOG_I2C_DEVICE(this)
|
LOG_I2C_DEVICE(this)
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with CH422G failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -4,7 +4,7 @@ namespace esphome {
|
|||||||
namespace chsc6x {
|
namespace chsc6x {
|
||||||
|
|
||||||
void CHSC6XTouchscreen::setup() {
|
void CHSC6XTouchscreen::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up CHSC6X Touchscreen...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
if (this->interrupt_pin_ != nullptr) {
|
if (this->interrupt_pin_ != nullptr) {
|
||||||
this->interrupt_pin_->setup();
|
this->interrupt_pin_->setup();
|
||||||
this->attach_interrupt_(this->interrupt_pin_, gpio::INTERRUPT_FALLING_EDGE);
|
this->attach_interrupt_(this->interrupt_pin_, gpio::INTERRUPT_FALLING_EDGE);
|
||||||
|
@@ -20,10 +20,10 @@ uint8_t cm1106_checksum(const uint8_t *response, size_t len) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void CM1106Component::setup() {
|
void CM1106Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up CM1106...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t response[8] = {0};
|
uint8_t response[8] = {0};
|
||||||
if (!this->cm1106_write_command_(C_M1106_CMD_GET_CO2, sizeof(C_M1106_CMD_GET_CO2), response, sizeof(response))) {
|
if (!this->cm1106_write_command_(C_M1106_CMD_GET_CO2, sizeof(C_M1106_CMD_GET_CO2), response, sizeof(response))) {
|
||||||
ESP_LOGE(TAG, "Communication with CM1106 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -104,7 +104,7 @@ void CM1106Component::dump_config() {
|
|||||||
LOG_SENSOR(" ", "CO2", this->co2_sensor_);
|
LOG_SENSOR(" ", "CO2", this->co2_sensor_);
|
||||||
this->check_uart_settings(9600);
|
this->check_uart_settings(9600);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with CM1106 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -52,7 +52,7 @@ bool CS5460AComponent::softreset_() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void CS5460AComponent::setup() {
|
void CS5460AComponent::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up CS5460A...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
float current_full_scale = (pga_gain_ == CS5460A_PGA_GAIN_10X) ? 0.25 : 0.10;
|
float current_full_scale = (pga_gain_ == CS5460A_PGA_GAIN_10X) ? 0.25 : 0.10;
|
||||||
float voltage_full_scale = 0.25;
|
float voltage_full_scale = 0.25;
|
||||||
|
@@ -42,7 +42,7 @@ static const uint8_t CSE7761_CMD_ENABLE_WRITE = 0xE5; // Enable write operation
|
|||||||
enum CSE7761 { RMS_IAC, RMS_IBC, RMS_UC, POWER_PAC, POWER_PBC, POWER_SC, ENERGY_AC, ENERGY_BC };
|
enum CSE7761 { RMS_IAC, RMS_IBC, RMS_UC, POWER_PAC, POWER_PBC, POWER_SC, ENERGY_AC, ENERGY_BC };
|
||||||
|
|
||||||
void CSE7761Component::setup() {
|
void CSE7761Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up CSE7761...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->write_(CSE7761_SPECIAL_COMMAND, CSE7761_CMD_RESET);
|
this->write_(CSE7761_SPECIAL_COMMAND, CSE7761_CMD_RESET);
|
||||||
uint16_t syscon = this->read_(0x00, 2); // Default 0x0A04
|
uint16_t syscon = this->read_(0x00, 2); // Default 0x0A04
|
||||||
if ((0x0A04 == syscon) && this->chip_init_()) {
|
if ((0x0A04 == syscon) && this->chip_init_()) {
|
||||||
@@ -57,7 +57,7 @@ void CSE7761Component::setup() {
|
|||||||
void CSE7761Component::dump_config() {
|
void CSE7761Component::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "CSE7761:");
|
ESP_LOGCONFIG(TAG, "CSE7761:");
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with CSE7761 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
this->check_uart_settings(38400, 1, uart::UART_CONFIG_PARITY_EVEN, 8);
|
this->check_uart_settings(38400, 1, uart::UART_CONFIG_PARITY_EVEN, 8);
|
||||||
|
@@ -6,7 +6,7 @@ namespace cst226 {
|
|||||||
static const char *const TAG = "cst226.touchscreen";
|
static const char *const TAG = "cst226.touchscreen";
|
||||||
|
|
||||||
void CST226Touchscreen::setup() {
|
void CST226Touchscreen::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up CST226 Touchscreen...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
if (this->reset_pin_ != nullptr) {
|
if (this->reset_pin_ != nullptr) {
|
||||||
this->reset_pin_->setup();
|
this->reset_pin_->setup();
|
||||||
this->reset_pin_->digital_write(true);
|
this->reset_pin_->digital_write(true);
|
||||||
|
@@ -38,7 +38,7 @@ void CST816Touchscreen::continue_setup_() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void CST816Touchscreen::setup() {
|
void CST816Touchscreen::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up CST816 Touchscreen...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
if (this->reset_pin_ != nullptr) {
|
if (this->reset_pin_ != nullptr) {
|
||||||
this->reset_pin_->setup();
|
this->reset_pin_->setup();
|
||||||
this->reset_pin_->digital_write(true);
|
this->reset_pin_->digital_write(true);
|
||||||
|
@@ -20,7 +20,7 @@ static const uint8_t DAC7678_REG_INTERNAL_REF_0 = 0x80;
|
|||||||
static const uint8_t DAC7678_REG_INTERNAL_REF_1 = 0x90;
|
static const uint8_t DAC7678_REG_INTERNAL_REF_1 = 0x90;
|
||||||
|
|
||||||
void DAC7678Output::setup() {
|
void DAC7678Output::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up DAC7678OutputComponent...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
ESP_LOGV(TAG, "Resetting device...");
|
ESP_LOGV(TAG, "Resetting device...");
|
||||||
|
|
||||||
|
@@ -70,7 +70,7 @@ bool DallasTemperatureSensor::read_scratch_pad_() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void DallasTemperatureSensor::setup() {
|
void DallasTemperatureSensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "setting up Dallas temperature sensor...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
if (!this->check_address_())
|
if (!this->check_address_())
|
||||||
return;
|
return;
|
||||||
if (!this->read_scratch_pad_())
|
if (!this->read_scratch_pad_())
|
||||||
@@ -80,7 +80,7 @@ void DallasTemperatureSensor::setup() {
|
|||||||
|
|
||||||
if ((this->address_ & 0xff) == DALLAS_MODEL_DS18S20) {
|
if ((this->address_ & 0xff) == DALLAS_MODEL_DS18S20) {
|
||||||
// DS18S20 doesn't support resolution.
|
// DS18S20 doesn't support resolution.
|
||||||
ESP_LOGW(TAG, "DS18S20 doesn't support setting resolution.");
|
ESP_LOGW(TAG, "DS18S20 doesn't support setting resolution");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -125,7 +125,6 @@ bool DallasTemperatureSensor::check_scratch_pad_() {
|
|||||||
crc8(this->scratch_pad_, 8));
|
crc8(this->scratch_pad_, 8));
|
||||||
#endif
|
#endif
|
||||||
if (!chksum_validity) {
|
if (!chksum_validity) {
|
||||||
ESP_LOGW(TAG, "'%s' - Scratch pad checksum invalid!", this->get_name().c_str());
|
|
||||||
this->status_set_warning("scratch pad checksum invalid");
|
this->status_set_warning("scratch pad checksum invalid");
|
||||||
ESP_LOGD(TAG, "Scratch pad: %02X.%02X.%02X.%02X.%02X.%02X.%02X.%02X.%02X (%02X)", this->scratch_pad_[0],
|
ESP_LOGD(TAG, "Scratch pad: %02X.%02X.%02X.%02X.%02X.%02X.%02X.%02X.%02X (%02X)", this->scratch_pad_[0],
|
||||||
this->scratch_pad_[1], this->scratch_pad_[2], this->scratch_pad_[3], this->scratch_pad_[4],
|
this->scratch_pad_[1], this->scratch_pad_[2], this->scratch_pad_[3], this->scratch_pad_[4],
|
||||||
|
@@ -10,20 +10,20 @@ static const char *const TAG = "deep_sleep";
|
|||||||
bool global_has_deep_sleep = false; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
|
bool global_has_deep_sleep = false; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
|
||||||
|
|
||||||
void DeepSleepComponent::setup() {
|
void DeepSleepComponent::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up Deep Sleep...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
global_has_deep_sleep = true;
|
global_has_deep_sleep = true;
|
||||||
|
|
||||||
const optional<uint32_t> run_duration = get_run_duration_();
|
const optional<uint32_t> run_duration = get_run_duration_();
|
||||||
if (run_duration.has_value()) {
|
if (run_duration.has_value()) {
|
||||||
ESP_LOGI(TAG, "Scheduling Deep Sleep to start in %" PRIu32 " ms", *run_duration);
|
ESP_LOGI(TAG, "Scheduling in %" PRIu32 " ms", *run_duration);
|
||||||
this->set_timeout(*run_duration, [this]() { this->begin_sleep(); });
|
this->set_timeout(*run_duration, [this]() { this->begin_sleep(); });
|
||||||
} else {
|
} else {
|
||||||
ESP_LOGD(TAG, "Not scheduling Deep Sleep, as no run duration is configured.");
|
ESP_LOGD(TAG, "Not scheduling; no run duration configured");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void DeepSleepComponent::dump_config() {
|
void DeepSleepComponent::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up Deep Sleep...");
|
ESP_LOGCONFIG(TAG, "Deep sleep:");
|
||||||
if (this->sleep_duration_.has_value()) {
|
if (this->sleep_duration_.has_value()) {
|
||||||
uint32_t duration = *this->sleep_duration_ / 1000;
|
uint32_t duration = *this->sleep_duration_ / 1000;
|
||||||
ESP_LOGCONFIG(TAG, " Sleep Duration: %" PRIu32 " ms", duration);
|
ESP_LOGCONFIG(TAG, " Sleep Duration: %" PRIu32 " ms", duration);
|
||||||
@@ -57,7 +57,7 @@ void DeepSleepComponent::begin_sleep(bool manual) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGI(TAG, "Beginning Deep Sleep");
|
ESP_LOGI(TAG, "Beginning sleep");
|
||||||
if (this->sleep_duration_.has_value()) {
|
if (this->sleep_duration_.has_value()) {
|
||||||
ESP_LOGI(TAG, "Sleeping for %" PRId64 "us", *this->sleep_duration_);
|
ESP_LOGI(TAG, "Sleeping for %" PRId64 "us", *this->sleep_duration_);
|
||||||
}
|
}
|
||||||
|
@@ -59,7 +59,7 @@ bool DeepSleepComponent::prepare_to_sleep_() {
|
|||||||
// Defer deep sleep until inactive
|
// Defer deep sleep until inactive
|
||||||
if (!this->next_enter_deep_sleep_) {
|
if (!this->next_enter_deep_sleep_) {
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
ESP_LOGW(TAG, "Waiting wakeup pin state change to enter deep sleep...");
|
ESP_LOGW(TAG, "Waiting for wakeup pin state change");
|
||||||
}
|
}
|
||||||
this->next_enter_deep_sleep_ = true;
|
this->next_enter_deep_sleep_ = true;
|
||||||
return false;
|
return false;
|
||||||
|
@@ -1,14 +1,22 @@
|
|||||||
import esphome.codegen as cg
|
import esphome.codegen as cg
|
||||||
from esphome.components import (
|
from esphome.components import (
|
||||||
|
alarm_control_panel,
|
||||||
binary_sensor,
|
binary_sensor,
|
||||||
|
button,
|
||||||
climate,
|
climate,
|
||||||
cover,
|
cover,
|
||||||
|
datetime,
|
||||||
|
event,
|
||||||
fan,
|
fan,
|
||||||
light,
|
light,
|
||||||
|
lock,
|
||||||
number,
|
number,
|
||||||
|
select,
|
||||||
sensor,
|
sensor,
|
||||||
switch,
|
switch,
|
||||||
|
text,
|
||||||
text_sensor,
|
text_sensor,
|
||||||
|
valve,
|
||||||
)
|
)
|
||||||
import esphome.config_validation as cv
|
import esphome.config_validation as cv
|
||||||
from esphome.const import (
|
from esphome.const import (
|
||||||
@@ -20,7 +28,9 @@ from esphome.const import (
|
|||||||
CONF_INVERTED,
|
CONF_INVERTED,
|
||||||
CONF_MAX_VALUE,
|
CONF_MAX_VALUE,
|
||||||
CONF_MIN_VALUE,
|
CONF_MIN_VALUE,
|
||||||
|
CONF_MODE,
|
||||||
CONF_NAME,
|
CONF_NAME,
|
||||||
|
CONF_OPTIONS,
|
||||||
CONF_OUTPUT_ID,
|
CONF_OUTPUT_ID,
|
||||||
CONF_SENSORS,
|
CONF_SENSORS,
|
||||||
CONF_STATE_CLASS,
|
CONF_STATE_CLASS,
|
||||||
@@ -31,9 +41,11 @@ from esphome.const import (
|
|||||||
CONF_UNIT_OF_MEASUREMENT,
|
CONF_UNIT_OF_MEASUREMENT,
|
||||||
DEVICE_CLASS_ENERGY,
|
DEVICE_CLASS_ENERGY,
|
||||||
DEVICE_CLASS_HUMIDITY,
|
DEVICE_CLASS_HUMIDITY,
|
||||||
|
DEVICE_CLASS_IDENTIFY,
|
||||||
DEVICE_CLASS_MOISTURE,
|
DEVICE_CLASS_MOISTURE,
|
||||||
DEVICE_CLASS_MOTION,
|
DEVICE_CLASS_MOTION,
|
||||||
DEVICE_CLASS_TEMPERATURE,
|
DEVICE_CLASS_TEMPERATURE,
|
||||||
|
DEVICE_CLASS_UPDATE,
|
||||||
ICON_BLUETOOTH,
|
ICON_BLUETOOTH,
|
||||||
ICON_BLUR,
|
ICON_BLUR,
|
||||||
ICON_THERMOMETER,
|
ICON_THERMOMETER,
|
||||||
@@ -45,38 +57,68 @@ from esphome.const import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
AUTO_LOAD = [
|
AUTO_LOAD = [
|
||||||
|
"alarm_control_panel",
|
||||||
"binary_sensor",
|
"binary_sensor",
|
||||||
|
"button",
|
||||||
"climate",
|
"climate",
|
||||||
"cover",
|
"cover",
|
||||||
|
"datetime",
|
||||||
|
"event",
|
||||||
"fan",
|
"fan",
|
||||||
"light",
|
"light",
|
||||||
|
"lock",
|
||||||
"number",
|
"number",
|
||||||
|
"select",
|
||||||
"sensor",
|
"sensor",
|
||||||
"switch",
|
"switch",
|
||||||
|
"text",
|
||||||
"text_sensor",
|
"text_sensor",
|
||||||
|
"valve",
|
||||||
]
|
]
|
||||||
|
|
||||||
demo_ns = cg.esphome_ns.namespace("demo")
|
demo_ns = cg.esphome_ns.namespace("demo")
|
||||||
|
DemoAlarmControlPanel = demo_ns.class_(
|
||||||
|
"DemoAlarmControlPanel", alarm_control_panel.AlarmControlPanel, cg.Component
|
||||||
|
)
|
||||||
|
DemoAlarmControlPanelType = demo_ns.enum("DemoAlarmControlPanelType", is_class=True)
|
||||||
DemoBinarySensor = demo_ns.class_(
|
DemoBinarySensor = demo_ns.class_(
|
||||||
"DemoBinarySensor", binary_sensor.BinarySensor, cg.PollingComponent
|
"DemoBinarySensor", binary_sensor.BinarySensor, cg.PollingComponent
|
||||||
)
|
)
|
||||||
|
DemoButton = demo_ns.class_("DemoButton", button.Button)
|
||||||
DemoClimate = demo_ns.class_("DemoClimate", climate.Climate, cg.Component)
|
DemoClimate = demo_ns.class_("DemoClimate", climate.Climate, cg.Component)
|
||||||
DemoClimateType = demo_ns.enum("DemoClimateType", is_class=True)
|
DemoClimateType = demo_ns.enum("DemoClimateType", is_class=True)
|
||||||
DemoCover = demo_ns.class_("DemoCover", cover.Cover, cg.Component)
|
DemoCover = demo_ns.class_("DemoCover", cover.Cover, cg.Component)
|
||||||
DemoCoverType = demo_ns.enum("DemoCoverType", is_class=True)
|
DemoCoverType = demo_ns.enum("DemoCoverType", is_class=True)
|
||||||
|
DemoDate = demo_ns.class_("DemoDate", datetime.DateEntity, cg.Component)
|
||||||
|
DemoDateTime = demo_ns.class_("DemoDateTime", datetime.DateTimeEntity, cg.Component)
|
||||||
|
DemoTime = demo_ns.class_("DemoTime", datetime.TimeEntity, cg.Component)
|
||||||
|
DemoEvent = demo_ns.class_("DemoEvent", event.Event, cg.Component)
|
||||||
DemoFan = demo_ns.class_("DemoFan", fan.Fan, cg.Component)
|
DemoFan = demo_ns.class_("DemoFan", fan.Fan, cg.Component)
|
||||||
DemoFanType = demo_ns.enum("DemoFanType", is_class=True)
|
DemoFanType = demo_ns.enum("DemoFanType", is_class=True)
|
||||||
DemoLight = demo_ns.class_("DemoLight", light.LightOutput, cg.Component)
|
DemoLight = demo_ns.class_("DemoLight", light.LightOutput, cg.Component)
|
||||||
DemoLightType = demo_ns.enum("DemoLightType", is_class=True)
|
DemoLightType = demo_ns.enum("DemoLightType", is_class=True)
|
||||||
|
DemoLock = demo_ns.class_("DemoLock", lock.Lock, cg.Component)
|
||||||
|
DemoLockType = demo_ns.enum("DemoLockType", is_class=True)
|
||||||
DemoNumber = demo_ns.class_("DemoNumber", number.Number, cg.Component)
|
DemoNumber = demo_ns.class_("DemoNumber", number.Number, cg.Component)
|
||||||
DemoNumberType = demo_ns.enum("DemoNumberType", is_class=True)
|
DemoNumberType = demo_ns.enum("DemoNumberType", is_class=True)
|
||||||
|
DemoSelect = demo_ns.class_("DemoSelect", select.Select, cg.Component)
|
||||||
|
DemoSelectType = demo_ns.enum("DemoSelectType", is_class=True)
|
||||||
DemoSensor = demo_ns.class_("DemoSensor", sensor.Sensor, cg.PollingComponent)
|
DemoSensor = demo_ns.class_("DemoSensor", sensor.Sensor, cg.PollingComponent)
|
||||||
DemoSwitch = demo_ns.class_("DemoSwitch", switch.Switch, cg.Component)
|
DemoSwitch = demo_ns.class_("DemoSwitch", switch.Switch, cg.Component)
|
||||||
|
DemoText = demo_ns.class_("DemoText", text.Text, cg.Component)
|
||||||
|
DemoTextType = demo_ns.enum("DemoTextType", is_class=True)
|
||||||
DemoTextSensor = demo_ns.class_(
|
DemoTextSensor = demo_ns.class_(
|
||||||
"DemoTextSensor", text_sensor.TextSensor, cg.PollingComponent
|
"DemoTextSensor", text_sensor.TextSensor, cg.PollingComponent
|
||||||
)
|
)
|
||||||
|
DemoValve = demo_ns.class_("DemoValve", valve.Valve)
|
||||||
|
DemoValveType = demo_ns.enum("DemoValveType", is_class=True)
|
||||||
|
|
||||||
|
|
||||||
|
ALARM_CONTROL_PANEL_TYPES = {
|
||||||
|
1: DemoAlarmControlPanelType.TYPE_1,
|
||||||
|
2: DemoAlarmControlPanelType.TYPE_2,
|
||||||
|
3: DemoAlarmControlPanelType.TYPE_3,
|
||||||
|
}
|
||||||
CLIMATE_TYPES = {
|
CLIMATE_TYPES = {
|
||||||
1: DemoClimateType.TYPE_1,
|
1: DemoClimateType.TYPE_1,
|
||||||
2: DemoClimateType.TYPE_2,
|
2: DemoClimateType.TYPE_2,
|
||||||
@@ -103,21 +145,67 @@ LIGHT_TYPES = {
|
|||||||
6: DemoLightType.TYPE_6,
|
6: DemoLightType.TYPE_6,
|
||||||
7: DemoLightType.TYPE_7,
|
7: DemoLightType.TYPE_7,
|
||||||
}
|
}
|
||||||
|
LOCK_TYPES = {
|
||||||
|
1: DemoLockType.TYPE_1,
|
||||||
|
2: DemoLockType.TYPE_2,
|
||||||
|
}
|
||||||
NUMBER_TYPES = {
|
NUMBER_TYPES = {
|
||||||
1: DemoNumberType.TYPE_1,
|
1: DemoNumberType.TYPE_1,
|
||||||
2: DemoNumberType.TYPE_2,
|
2: DemoNumberType.TYPE_2,
|
||||||
3: DemoNumberType.TYPE_3,
|
3: DemoNumberType.TYPE_3,
|
||||||
}
|
}
|
||||||
|
TEXT_TYPES = {
|
||||||
|
1: DemoTextType.TYPE_1,
|
||||||
|
2: DemoTextType.TYPE_2,
|
||||||
|
}
|
||||||
|
VALVE_TYPES = {
|
||||||
|
1: DemoValveType.TYPE_1,
|
||||||
|
2: DemoValveType.TYPE_2,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
CONF_ALARM_CONTROL_PANELS = "alarm_control_panels"
|
||||||
|
CONF_BUTTONS = "buttons"
|
||||||
CONF_CLIMATES = "climates"
|
CONF_CLIMATES = "climates"
|
||||||
CONF_COVERS = "covers"
|
CONF_COVERS = "covers"
|
||||||
|
CONF_DATETIMES = "datetimes"
|
||||||
CONF_FANS = "fans"
|
CONF_FANS = "fans"
|
||||||
CONF_LIGHTS = "lights"
|
CONF_LIGHTS = "lights"
|
||||||
|
CONF_LOCKS = "locks"
|
||||||
CONF_NUMBERS = "numbers"
|
CONF_NUMBERS = "numbers"
|
||||||
|
CONF_SELECTS = "selects"
|
||||||
|
CONF_TEXTS = "texts"
|
||||||
|
CONF_VALVES = "valves"
|
||||||
|
|
||||||
CONFIG_SCHEMA = cv.Schema(
|
CONFIG_SCHEMA = cv.Schema(
|
||||||
{
|
{
|
||||||
|
cv.Optional(
|
||||||
|
CONF_ALARM_CONTROL_PANELS,
|
||||||
|
default=[
|
||||||
|
{
|
||||||
|
CONF_NAME: "Demo Alarm Control Panel",
|
||||||
|
CONF_TYPE: 1,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
CONF_NAME: "Demo Alarm Control Panel Code",
|
||||||
|
CONF_TYPE: 2,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
CONF_NAME: "Demo Alarm Control Panel Code to Arm",
|
||||||
|
CONF_TYPE: 3,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
): [
|
||||||
|
alarm_control_panel.alarm_control_panel_schema(
|
||||||
|
DemoAlarmControlPanel
|
||||||
|
).extend(
|
||||||
|
{
|
||||||
|
cv.Required(CONF_TYPE): cv.enum(
|
||||||
|
ALARM_CONTROL_PANEL_TYPES, int=True
|
||||||
|
),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
],
|
||||||
cv.Optional(
|
cv.Optional(
|
||||||
CONF_BINARY_SENSORS,
|
CONF_BINARY_SENSORS,
|
||||||
default=[
|
default=[
|
||||||
@@ -135,6 +223,21 @@ CONFIG_SCHEMA = cv.Schema(
|
|||||||
cv.polling_component_schema("60s")
|
cv.polling_component_schema("60s")
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
|
cv.Optional(
|
||||||
|
CONF_BUTTONS,
|
||||||
|
default=[
|
||||||
|
{
|
||||||
|
CONF_NAME: "Demo Update Button",
|
||||||
|
CONF_DEVICE_CLASS: DEVICE_CLASS_UPDATE,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
CONF_NAME: "Demo Button Identify",
|
||||||
|
CONF_DEVICE_CLASS: DEVICE_CLASS_IDENTIFY,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
): [
|
||||||
|
button.button_schema(DemoButton),
|
||||||
|
],
|
||||||
cv.Optional(
|
cv.Optional(
|
||||||
CONF_CLIMATES,
|
CONF_CLIMATES,
|
||||||
default=[
|
default=[
|
||||||
@@ -191,6 +294,20 @@ CONFIG_SCHEMA = cv.Schema(
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
|
cv.Optional(
|
||||||
|
CONF_DATETIMES,
|
||||||
|
default=[
|
||||||
|
{CONF_NAME: "Demo DateTime", CONF_TYPE: "DATETIME"},
|
||||||
|
{CONF_NAME: "Demo Date", CONF_TYPE: "DATE"},
|
||||||
|
{CONF_NAME: "Demo Time", CONF_TYPE: "TIME"},
|
||||||
|
],
|
||||||
|
): [
|
||||||
|
cv.Any(
|
||||||
|
datetime.date_schema(DemoDate),
|
||||||
|
datetime.datetime_schema(DemoDateTime),
|
||||||
|
datetime.time_schema(DemoTime),
|
||||||
|
)
|
||||||
|
],
|
||||||
cv.Optional(
|
cv.Optional(
|
||||||
CONF_FANS,
|
CONF_FANS,
|
||||||
default=[
|
default=[
|
||||||
@@ -262,6 +379,19 @@ CONFIG_SCHEMA = cv.Schema(
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
|
cv.Optional(
|
||||||
|
CONF_LOCKS,
|
||||||
|
default=[
|
||||||
|
{CONF_NAME: "Demo Lock", CONF_TYPE: 1},
|
||||||
|
{CONF_NAME: "Demo Lock and Open", CONF_TYPE: 2},
|
||||||
|
],
|
||||||
|
): [
|
||||||
|
lock.lock_schema(DemoLock).extend(
|
||||||
|
{
|
||||||
|
cv.Required(CONF_TYPE): cv.enum(LOCK_TYPES, int=True),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
],
|
||||||
cv.Optional(
|
cv.Optional(
|
||||||
CONF_NUMBERS,
|
CONF_NUMBERS,
|
||||||
default=[
|
default=[
|
||||||
@@ -299,6 +429,25 @@ CONFIG_SCHEMA = cv.Schema(
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
|
cv.Optional(
|
||||||
|
CONF_SELECTS,
|
||||||
|
default=[
|
||||||
|
{
|
||||||
|
CONF_NAME: "Demo Select 1",
|
||||||
|
CONF_OPTIONS: ["Option 1", "Option 2", "Option 3"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
CONF_NAME: "Demo Select 2",
|
||||||
|
CONF_OPTIONS: ["Option A", "Option B", "Option C"],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
): [
|
||||||
|
select.select_schema(DemoSelect).extend(
|
||||||
|
{
|
||||||
|
cv.Required(CONF_OPTIONS): cv.ensure_list(cv.string_strict),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
],
|
||||||
cv.Optional(
|
cv.Optional(
|
||||||
CONF_SENSORS,
|
CONF_SENSORS,
|
||||||
default=[
|
default=[
|
||||||
@@ -355,6 +504,19 @@ CONFIG_SCHEMA = cv.Schema(
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
): [switch.switch_schema(DemoSwitch).extend(cv.COMPONENT_SCHEMA)],
|
): [switch.switch_schema(DemoSwitch).extend(cv.COMPONENT_SCHEMA)],
|
||||||
|
cv.Optional(
|
||||||
|
CONF_TEXTS,
|
||||||
|
default=[
|
||||||
|
{CONF_NAME: "Demo Text 1", CONF_MODE: "TEXT", CONF_TYPE: 1},
|
||||||
|
{CONF_NAME: "Demo Text 2", CONF_MODE: "PASSWORD", CONF_TYPE: 2},
|
||||||
|
],
|
||||||
|
): [
|
||||||
|
text.text_schema(DemoText).extend(
|
||||||
|
{
|
||||||
|
cv.Required(CONF_TYPE): cv.enum(TEXT_TYPES, int=True),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
],
|
||||||
cv.Optional(
|
cv.Optional(
|
||||||
CONF_TEXT_SENSORS,
|
CONF_TEXT_SENSORS,
|
||||||
default=[
|
default=[
|
||||||
@@ -371,15 +533,36 @@ CONFIG_SCHEMA = cv.Schema(
|
|||||||
cv.polling_component_schema("60s")
|
cv.polling_component_schema("60s")
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
|
cv.Optional(
|
||||||
|
CONF_VALVES,
|
||||||
|
default=[
|
||||||
|
{CONF_NAME: "Demo Valve 1", CONF_TYPE: 1},
|
||||||
|
{CONF_NAME: "Demo Valve 2", CONF_TYPE: 2},
|
||||||
|
],
|
||||||
|
): [
|
||||||
|
valve.valve_schema(DemoValve).extend(
|
||||||
|
{
|
||||||
|
cv.Required(CONF_TYPE): cv.enum(VALVE_TYPES, int=True),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
],
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
async def to_code(config):
|
async def to_code(config):
|
||||||
|
for conf in config[CONF_ALARM_CONTROL_PANELS]:
|
||||||
|
var = await alarm_control_panel.new_alarm_control_panel(conf)
|
||||||
|
cg.add(var.set_type(conf[CONF_TYPE]))
|
||||||
|
await cg.register_component(var, conf)
|
||||||
|
|
||||||
for conf in config[CONF_BINARY_SENSORS]:
|
for conf in config[CONF_BINARY_SENSORS]:
|
||||||
var = await binary_sensor.new_binary_sensor(conf)
|
var = await binary_sensor.new_binary_sensor(conf)
|
||||||
await cg.register_component(var, conf)
|
await cg.register_component(var, conf)
|
||||||
|
|
||||||
|
for conf in config[CONF_BUTTONS]:
|
||||||
|
await button.new_button(conf)
|
||||||
|
|
||||||
for conf in config[CONF_CLIMATES]:
|
for conf in config[CONF_CLIMATES]:
|
||||||
var = await climate.new_climate(conf)
|
var = await climate.new_climate(conf)
|
||||||
await cg.register_component(var, conf)
|
await cg.register_component(var, conf)
|
||||||
@@ -390,6 +573,10 @@ async def to_code(config):
|
|||||||
await cg.register_component(var, conf)
|
await cg.register_component(var, conf)
|
||||||
cg.add(var.set_type(conf[CONF_TYPE]))
|
cg.add(var.set_type(conf[CONF_TYPE]))
|
||||||
|
|
||||||
|
for conf in config[CONF_DATETIMES]:
|
||||||
|
var = await datetime.new_datetime(conf)
|
||||||
|
await cg.register_component(var, conf)
|
||||||
|
|
||||||
for conf in config[CONF_FANS]:
|
for conf in config[CONF_FANS]:
|
||||||
var = await fan.new_fan(conf)
|
var = await fan.new_fan(conf)
|
||||||
await cg.register_component(var, conf)
|
await cg.register_component(var, conf)
|
||||||
@@ -400,6 +587,11 @@ async def to_code(config):
|
|||||||
await cg.register_component(var, conf)
|
await cg.register_component(var, conf)
|
||||||
cg.add(var.set_type(conf[CONF_TYPE]))
|
cg.add(var.set_type(conf[CONF_TYPE]))
|
||||||
|
|
||||||
|
for conf in config[CONF_LOCKS]:
|
||||||
|
var = await lock.new_lock(conf)
|
||||||
|
if conf[CONF_TYPE] == 2:
|
||||||
|
cg.add(var.traits.set_supports_open(True))
|
||||||
|
|
||||||
for conf in config[CONF_NUMBERS]:
|
for conf in config[CONF_NUMBERS]:
|
||||||
var = await number.new_number(
|
var = await number.new_number(
|
||||||
conf,
|
conf,
|
||||||
@@ -410,6 +602,10 @@ async def to_code(config):
|
|||||||
await cg.register_component(var, conf)
|
await cg.register_component(var, conf)
|
||||||
cg.add(var.set_type(conf[CONF_TYPE]))
|
cg.add(var.set_type(conf[CONF_TYPE]))
|
||||||
|
|
||||||
|
for conf in config[CONF_SELECTS]:
|
||||||
|
var = await select.new_select(conf, options=conf[CONF_OPTIONS])
|
||||||
|
await cg.register_component(var, conf)
|
||||||
|
|
||||||
for conf in config[CONF_SENSORS]:
|
for conf in config[CONF_SENSORS]:
|
||||||
var = await sensor.new_sensor(conf)
|
var = await sensor.new_sensor(conf)
|
||||||
await cg.register_component(var, conf)
|
await cg.register_component(var, conf)
|
||||||
@@ -418,6 +614,16 @@ async def to_code(config):
|
|||||||
var = await switch.new_switch(conf)
|
var = await switch.new_switch(conf)
|
||||||
await cg.register_component(var, conf)
|
await cg.register_component(var, conf)
|
||||||
|
|
||||||
|
for conf in config[CONF_TEXTS]:
|
||||||
|
var = await text.new_text(conf)
|
||||||
|
await cg.register_component(var, conf)
|
||||||
|
if conf[CONF_TYPE] == 2:
|
||||||
|
cg.add(var.traits.set_mode(text.TextMode.TEXT_MODE_PASSWORD))
|
||||||
|
|
||||||
for conf in config[CONF_TEXT_SENSORS]:
|
for conf in config[CONF_TEXT_SENSORS]:
|
||||||
var = await text_sensor.new_text_sensor(conf)
|
var = await text_sensor.new_text_sensor(conf)
|
||||||
await cg.register_component(var, conf)
|
await cg.register_component(var, conf)
|
||||||
|
|
||||||
|
for conf in config[CONF_VALVES]:
|
||||||
|
var = await valve.new_valve(conf)
|
||||||
|
cg.add(var.set_type(conf[CONF_TYPE]))
|
||||||
|
65
esphome/components/demo/demo_alarm_control_panel.h
Normal file
65
esphome/components/demo/demo_alarm_control_panel.h
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/components/alarm_control_panel/alarm_control_panel.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace demo {
|
||||||
|
|
||||||
|
using namespace alarm_control_panel;
|
||||||
|
|
||||||
|
enum class DemoAlarmControlPanelType {
|
||||||
|
TYPE_1,
|
||||||
|
TYPE_2,
|
||||||
|
TYPE_3,
|
||||||
|
};
|
||||||
|
|
||||||
|
class DemoAlarmControlPanel : public AlarmControlPanel, public Component {
|
||||||
|
public:
|
||||||
|
void setup() override {}
|
||||||
|
|
||||||
|
uint32_t get_supported_features() const override { return ACP_FEAT_ARM_AWAY | ACP_FEAT_TRIGGER; }
|
||||||
|
|
||||||
|
bool get_requires_code() const override { return this->type_ != DemoAlarmControlPanelType::TYPE_1; }
|
||||||
|
|
||||||
|
bool get_requires_code_to_arm() const override { return this->type_ == DemoAlarmControlPanelType::TYPE_3; }
|
||||||
|
|
||||||
|
void set_type(DemoAlarmControlPanelType type) { this->type_ = type; }
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void control(const AlarmControlPanelCall &call) override {
|
||||||
|
auto state = call.get_state().value_or(ACP_STATE_DISARMED);
|
||||||
|
switch (state) {
|
||||||
|
case ACP_STATE_ARMED_AWAY:
|
||||||
|
if (this->get_requires_code_to_arm() && call.get_code().has_value()) {
|
||||||
|
if (call.get_code().value() != "1234") {
|
||||||
|
this->status_momentary_error("Invalid code", 5000);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this->publish_state(ACP_STATE_ARMED_AWAY);
|
||||||
|
break;
|
||||||
|
case ACP_STATE_DISARMED:
|
||||||
|
if (this->get_requires_code() && call.get_code().has_value()) {
|
||||||
|
if (call.get_code().value() != "1234") {
|
||||||
|
this->status_momentary_error("Invalid code", 5000);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this->publish_state(ACP_STATE_DISARMED);
|
||||||
|
return;
|
||||||
|
case ACP_STATE_TRIGGERED:
|
||||||
|
this->publish_state(ACP_STATE_TRIGGERED);
|
||||||
|
return;
|
||||||
|
case ACP_STATE_PENDING:
|
||||||
|
this->publish_state(ACP_STATE_PENDING);
|
||||||
|
return;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
DemoAlarmControlPanelType type_{};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace demo
|
||||||
|
} // namespace esphome
|
15
esphome/components/demo/demo_button.h
Normal file
15
esphome/components/demo/demo_button.h
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/components/button/button.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace demo {
|
||||||
|
|
||||||
|
class DemoButton : public button::Button {
|
||||||
|
protected:
|
||||||
|
void press_action() override {}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace demo
|
||||||
|
} // namespace esphome
|
34
esphome/components/demo/demo_date.h
Normal file
34
esphome/components/demo/demo_date.h
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/core/defines.h"
|
||||||
|
|
||||||
|
#ifdef USE_DATETIME_DATE
|
||||||
|
|
||||||
|
#include "esphome/components/datetime/date_entity.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace demo {
|
||||||
|
|
||||||
|
class DemoDate : public datetime::DateEntity, public Component {
|
||||||
|
public:
|
||||||
|
void setup() override {
|
||||||
|
this->year_ = 2038;
|
||||||
|
this->month_ = 01;
|
||||||
|
this->day_ = 19;
|
||||||
|
this->publish_state();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void control(const datetime::DateCall &call) override {
|
||||||
|
this->year_ = call.get_year().value_or(this->year_);
|
||||||
|
this->month_ = call.get_month().value_or(this->month_);
|
||||||
|
this->day_ = call.get_day().value_or(this->day_);
|
||||||
|
this->publish_state();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace demo
|
||||||
|
} // namespace esphome
|
||||||
|
|
||||||
|
#endif
|
40
esphome/components/demo/demo_datetime.h
Normal file
40
esphome/components/demo/demo_datetime.h
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/core/defines.h"
|
||||||
|
|
||||||
|
#ifdef USE_DATETIME_DATETIME
|
||||||
|
|
||||||
|
#include "esphome/components/datetime/datetime_entity.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace demo {
|
||||||
|
|
||||||
|
class DemoDateTime : public datetime::DateTimeEntity, public Component {
|
||||||
|
public:
|
||||||
|
void setup() override {
|
||||||
|
this->year_ = 2038;
|
||||||
|
this->month_ = 01;
|
||||||
|
this->day_ = 19;
|
||||||
|
this->hour_ = 3;
|
||||||
|
this->minute_ = 14;
|
||||||
|
this->second_ = 8;
|
||||||
|
this->publish_state();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void control(const datetime::DateTimeCall &call) override {
|
||||||
|
this->year_ = call.get_year().value_or(this->year_);
|
||||||
|
this->month_ = call.get_month().value_or(this->month_);
|
||||||
|
this->day_ = call.get_day().value_or(this->day_);
|
||||||
|
this->hour_ = call.get_hour().value_or(this->hour_);
|
||||||
|
this->minute_ = call.get_minute().value_or(this->minute_);
|
||||||
|
this->second_ = call.get_second().value_or(this->second_);
|
||||||
|
this->publish_state();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace demo
|
||||||
|
} // namespace esphome
|
||||||
|
|
||||||
|
#endif
|
17
esphome/components/demo/demo_lock.h
Normal file
17
esphome/components/demo/demo_lock.h
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/components/lock/lock.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace demo {
|
||||||
|
|
||||||
|
class DemoLock : public lock::Lock {
|
||||||
|
protected:
|
||||||
|
void control(const lock::LockCall &call) override {
|
||||||
|
auto state = *call.get_state();
|
||||||
|
this->publish_state(state);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace demo
|
||||||
|
} // namespace esphome
|
15
esphome/components/demo/demo_select.h
Normal file
15
esphome/components/demo/demo_select.h
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/components/select/select.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace demo {
|
||||||
|
|
||||||
|
class DemoSelect : public select::Select, public Component {
|
||||||
|
protected:
|
||||||
|
void control(const std::string &value) override { this->publish_state(value); }
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace demo
|
||||||
|
} // namespace esphome
|
18
esphome/components/demo/demo_text.h
Normal file
18
esphome/components/demo/demo_text.h
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/components/text/text.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace demo {
|
||||||
|
|
||||||
|
class DemoText : public text::Text, public Component {
|
||||||
|
public:
|
||||||
|
void setup() override { this->publish_state("I am a text entity"); }
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void control(const std::string &value) override { this->publish_state(value); }
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace demo
|
||||||
|
} // namespace esphome
|
34
esphome/components/demo/demo_time.h
Normal file
34
esphome/components/demo/demo_time.h
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/core/defines.h"
|
||||||
|
|
||||||
|
#ifdef USE_DATETIME_TIME
|
||||||
|
|
||||||
|
#include "esphome/components/datetime/time_entity.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace demo {
|
||||||
|
|
||||||
|
class DemoTime : public datetime::TimeEntity, public Component {
|
||||||
|
public:
|
||||||
|
void setup() override {
|
||||||
|
this->hour_ = 3;
|
||||||
|
this->minute_ = 14;
|
||||||
|
this->second_ = 8;
|
||||||
|
this->publish_state();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void control(const datetime::TimeCall &call) override {
|
||||||
|
this->hour_ = call.get_hour().value_or(this->hour_);
|
||||||
|
this->minute_ = call.get_minute().value_or(this->minute_);
|
||||||
|
this->second_ = call.get_second().value_or(this->second_);
|
||||||
|
this->publish_state();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace demo
|
||||||
|
} // namespace esphome
|
||||||
|
|
||||||
|
#endif
|
54
esphome/components/demo/demo_valve.h
Normal file
54
esphome/components/demo/demo_valve.h
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "esphome/components/valve/valve.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace demo {
|
||||||
|
|
||||||
|
enum class DemoValveType {
|
||||||
|
TYPE_1,
|
||||||
|
TYPE_2,
|
||||||
|
};
|
||||||
|
|
||||||
|
class DemoValve : public valve::Valve {
|
||||||
|
public:
|
||||||
|
valve::ValveTraits get_traits() override {
|
||||||
|
valve::ValveTraits traits;
|
||||||
|
if (this->type_ == DemoValveType::TYPE_2) {
|
||||||
|
traits.set_supports_position(true);
|
||||||
|
traits.set_supports_toggle(true);
|
||||||
|
traits.set_supports_stop(true);
|
||||||
|
}
|
||||||
|
return traits;
|
||||||
|
}
|
||||||
|
|
||||||
|
void set_type(DemoValveType type) { this->type_ = type; }
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void control(const valve::ValveCall &call) override {
|
||||||
|
if (call.get_position().has_value()) {
|
||||||
|
this->position = *call.get_position();
|
||||||
|
this->publish_state();
|
||||||
|
return;
|
||||||
|
} else if (call.get_toggle().has_value()) {
|
||||||
|
if (call.get_toggle().value()) {
|
||||||
|
if (this->position == valve::VALVE_OPEN) {
|
||||||
|
this->position = valve::VALVE_CLOSED;
|
||||||
|
this->publish_state();
|
||||||
|
} else {
|
||||||
|
this->position = valve::VALVE_OPEN;
|
||||||
|
this->publish_state();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
} else if (call.get_stop()) {
|
||||||
|
this->current_operation = valve::VALVE_OPERATION_IDLE;
|
||||||
|
this->publish_state(); // Keep the current position
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
DemoValveType type_{};
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace demo
|
||||||
|
} // namespace esphome
|
@@ -8,25 +8,19 @@ namespace dht {
|
|||||||
static const char *const TAG = "dht";
|
static const char *const TAG = "dht";
|
||||||
|
|
||||||
void DHT::setup() {
|
void DHT::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up DHT...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->pin_->digital_write(true);
|
this->pin_->digital_write(true);
|
||||||
this->pin_->setup();
|
this->pin_->setup();
|
||||||
this->pin_->digital_write(true);
|
this->pin_->digital_write(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void DHT::dump_config() {
|
void DHT::dump_config() {
|
||||||
ESP_LOGCONFIG(TAG, "DHT:");
|
ESP_LOGCONFIG(TAG, "DHT:");
|
||||||
LOG_PIN(" Pin: ", this->pin_);
|
LOG_PIN(" Pin: ", this->pin_);
|
||||||
if (this->is_auto_detect_) {
|
ESP_LOGCONFIG(TAG, " %sModel: %s", this->is_auto_detect_ ? "Auto-detected " : "",
|
||||||
ESP_LOGCONFIG(TAG, " Auto-detected model: %s", this->model_ == DHT_MODEL_DHT11 ? "DHT11" : "DHT22");
|
this->model_ == DHT_MODEL_DHT11 ? "DHT11" : "DHT22 or equivalent");
|
||||||
} else if (this->model_ == DHT_MODEL_DHT11) {
|
ESP_LOGCONFIG(TAG, " Internal pull-up: %s", ONOFF(this->pin_->get_flags() & gpio::FLAG_PULLUP));
|
||||||
ESP_LOGCONFIG(TAG, " Model: DHT11");
|
|
||||||
} else {
|
|
||||||
ESP_LOGCONFIG(TAG, " Model: DHT22 (or equivalent)");
|
|
||||||
}
|
|
||||||
ESP_LOGCONFIG(TAG, " Internal Pull-up: %s", ONOFF(this->pin_->get_flags() & gpio::FLAG_PULLUP));
|
|
||||||
|
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
|
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
}
|
}
|
||||||
@@ -46,7 +40,7 @@ void DHT::update() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (success) {
|
if (success) {
|
||||||
ESP_LOGD(TAG, "Got Temperature=%.1f°C Humidity=%.1f%%", temperature, humidity);
|
ESP_LOGD(TAG, "Temperature %.1f°C Humidity %.1f%%", temperature, humidity);
|
||||||
|
|
||||||
if (this->temperature_sensor_ != nullptr)
|
if (this->temperature_sensor_ != nullptr)
|
||||||
this->temperature_sensor_->publish_state(temperature);
|
this->temperature_sensor_->publish_state(temperature);
|
||||||
@@ -54,11 +48,8 @@ void DHT::update() {
|
|||||||
this->humidity_sensor_->publish_state(humidity);
|
this->humidity_sensor_->publish_state(humidity);
|
||||||
this->status_clear_warning();
|
this->status_clear_warning();
|
||||||
} else {
|
} else {
|
||||||
const char *str = "";
|
ESP_LOGW(TAG, "Invalid readings! Check pin number and pull-up resistor%s.",
|
||||||
if (this->is_auto_detect_) {
|
this->is_auto_detect_ ? " and try manually specifying the model" : "");
|
||||||
str = " and consider manually specifying the DHT model using the model option";
|
|
||||||
}
|
|
||||||
ESP_LOGW(TAG, "Invalid readings! Please check your wiring (pull-up resistor, pin number)%s.", str);
|
|
||||||
if (this->temperature_sensor_ != nullptr)
|
if (this->temperature_sensor_ != nullptr)
|
||||||
this->temperature_sensor_->publish_state(NAN);
|
this->temperature_sensor_->publish_state(NAN);
|
||||||
if (this->humidity_sensor_ != nullptr)
|
if (this->humidity_sensor_ != nullptr)
|
||||||
@@ -68,10 +59,12 @@ void DHT::update() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
float DHT::get_setup_priority() const { return setup_priority::DATA; }
|
float DHT::get_setup_priority() const { return setup_priority::DATA; }
|
||||||
|
|
||||||
void DHT::set_dht_model(DHTModel model) {
|
void DHT::set_dht_model(DHTModel model) {
|
||||||
this->model_ = model;
|
this->model_ = model;
|
||||||
this->is_auto_detect_ = model == DHT_MODEL_AUTO_DETECT;
|
this->is_auto_detect_ = model == DHT_MODEL_AUTO_DETECT;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool HOT IRAM_ATTR DHT::read_sensor_(float *temperature, float *humidity, bool report_errors) {
|
bool HOT IRAM_ATTR DHT::read_sensor_(float *temperature, float *humidity, bool report_errors) {
|
||||||
*humidity = NAN;
|
*humidity = NAN;
|
||||||
*temperature = NAN;
|
*temperature = NAN;
|
||||||
@@ -121,9 +114,9 @@ bool HOT IRAM_ATTR DHT::read_sensor_(float *temperature, float *humidity, bool r
|
|||||||
while (!this->pin_->digital_read()) {
|
while (!this->pin_->digital_read()) {
|
||||||
if (micros() - start_time > 90) {
|
if (micros() - start_time > 90) {
|
||||||
if (i < 0) {
|
if (i < 0) {
|
||||||
error_code = 1;
|
error_code = 1; // line didn't clear
|
||||||
} else {
|
} else {
|
||||||
error_code = 2;
|
error_code = 2; // rising edge for bit i timeout
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -139,9 +132,9 @@ bool HOT IRAM_ATTR DHT::read_sensor_(float *temperature, float *humidity, bool r
|
|||||||
end_time = micros();
|
end_time = micros();
|
||||||
if (end_time - start_time > 90) {
|
if (end_time - start_time > 90) {
|
||||||
if (i < 0) {
|
if (i < 0) {
|
||||||
error_code = 3;
|
error_code = 3; // requesting data failed
|
||||||
} else {
|
} else {
|
||||||
error_code = 4;
|
error_code = 4; // falling edge for bit i timeout
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -166,22 +159,9 @@ bool HOT IRAM_ATTR DHT::read_sensor_(float *temperature, float *humidity, bool r
|
|||||||
if (!report_errors && error_code != 0)
|
if (!report_errors && error_code != 0)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
switch (error_code) {
|
if (error_code) {
|
||||||
case 1:
|
ESP_LOGW(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
ESP_LOGW(TAG, "Waiting for DHT communication to clear failed!");
|
|
||||||
return false;
|
return false;
|
||||||
case 2:
|
|
||||||
ESP_LOGW(TAG, "Rising edge for bit %d failed!", i);
|
|
||||||
return false;
|
|
||||||
case 3:
|
|
||||||
ESP_LOGW(TAG, "Requesting data from DHT failed!");
|
|
||||||
return false;
|
|
||||||
case 4:
|
|
||||||
ESP_LOGW(TAG, "Falling edge for bit %d failed!", i);
|
|
||||||
return false;
|
|
||||||
case 0:
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGVV(TAG,
|
ESP_LOGVV(TAG,
|
||||||
@@ -206,15 +186,15 @@ bool HOT IRAM_ATTR DHT::read_sensor_(float *temperature, float *humidity, bool r
|
|||||||
if (checksum_a == data[4]) {
|
if (checksum_a == data[4]) {
|
||||||
// Data format: 8bit integral RH data + 8bit decimal RH data + 8bit integral T data + 8bit decimal T data + 8bit
|
// Data format: 8bit integral RH data + 8bit decimal RH data + 8bit integral T data + 8bit decimal T data + 8bit
|
||||||
// check sum - some models always have 0 in the decimal part
|
// check sum - some models always have 0 in the decimal part
|
||||||
const uint16_t raw_temperature = uint16_t(data[2]) * 10 + (data[3] & 0x7F);
|
const uint16_t raw_temperature = static_cast<uint16_t>(data[2]) * 10 + (data[3] & 0x7F);
|
||||||
*temperature = raw_temperature / 10.0f;
|
*temperature = static_cast<float>(raw_temperature) / 10.0f;
|
||||||
if ((data[3] & 0x80) != 0) {
|
if ((data[3] & 0x80) != 0) {
|
||||||
// negative
|
// negative
|
||||||
*temperature *= -1;
|
*temperature *= -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
const uint16_t raw_humidity = uint16_t(data[0]) * 10 + data[1];
|
const uint16_t raw_humidity = static_cast<uint16_t>(data[0]) * 10 + data[1];
|
||||||
*humidity = raw_humidity / 10.0f;
|
*humidity = static_cast<float>(raw_humidity) / 10.0f;
|
||||||
} else {
|
} else {
|
||||||
// For compatibility with DHT11 models which might only use 2 bytes checksums, only use the data from these two
|
// For compatibility with DHT11 models which might only use 2 bytes checksums, only use the data from these two
|
||||||
// bytes
|
// bytes
|
||||||
@@ -222,8 +202,8 @@ bool HOT IRAM_ATTR DHT::read_sensor_(float *temperature, float *humidity, bool r
|
|||||||
*humidity = data[0];
|
*humidity = data[0];
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
uint16_t raw_humidity = (uint16_t(data[0] & 0xFF) << 8) | (data[1] & 0xFF);
|
uint16_t raw_humidity = encode_uint16(data[0], data[1]);
|
||||||
uint16_t raw_temperature = (uint16_t(data[2] & 0xFF) << 8) | (data[3] & 0xFF);
|
uint16_t raw_temperature = encode_uint16(data[2], data[3]);
|
||||||
|
|
||||||
if (raw_temperature & 0x8000) {
|
if (raw_temperature & 0x8000) {
|
||||||
if (!(raw_temperature & 0x4000))
|
if (!(raw_temperature & 0x4000))
|
||||||
@@ -234,24 +214,23 @@ bool HOT IRAM_ATTR DHT::read_sensor_(float *temperature, float *humidity, bool r
|
|||||||
|
|
||||||
if (raw_temperature == 1 && raw_humidity == 10) {
|
if (raw_temperature == 1 && raw_humidity == 10) {
|
||||||
if (report_errors) {
|
if (report_errors) {
|
||||||
ESP_LOGW(TAG, "Invalid temperature+humidity! Sensor reported 1°C and 1%% Hum");
|
ESP_LOGW(TAG, "Invalid data");
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
*humidity = raw_humidity * 0.1f;
|
*humidity = static_cast<float>(raw_humidity) * 0.1f;
|
||||||
if (*humidity > 100)
|
if (*humidity > 100.0f)
|
||||||
*humidity = NAN;
|
*humidity = NAN;
|
||||||
*temperature = int16_t(raw_temperature) * 0.1f;
|
*temperature = static_cast<int16_t>(raw_temperature) * 0.1f;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (*temperature == 0.0f && (*humidity == 1.0f || *humidity == 2.0f)) {
|
if (*temperature == 0.0f && (*humidity == 1.0f || *humidity == 2.0f)) {
|
||||||
if (report_errors) {
|
if (report_errors) {
|
||||||
ESP_LOGW(TAG, "DHT reports invalid data. Is the update interval too high or the sensor damaged?");
|
ESP_LOGW(TAG, "Invalid data");
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -34,7 +34,7 @@ void DHT12Component::update() {
|
|||||||
this->status_clear_warning();
|
this->status_clear_warning();
|
||||||
}
|
}
|
||||||
void DHT12Component::setup() {
|
void DHT12Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up DHT12...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t data[5];
|
uint8_t data[5];
|
||||||
if (!this->read_data_(data)) {
|
if (!this->read_data_(data)) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -45,7 +45,7 @@ void DHT12Component::dump_config() {
|
|||||||
ESP_LOGD(TAG, "DHT12:");
|
ESP_LOGD(TAG, "DHT12:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with DHT12 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
|
||||||
|
@@ -12,7 +12,7 @@ void DPS310Component::setup() {
|
|||||||
auto timer = DPS310_INIT_TIMEOUT;
|
auto timer = DPS310_INIT_TIMEOUT;
|
||||||
uint8_t reg = 0;
|
uint8_t reg = 0;
|
||||||
|
|
||||||
ESP_LOGCONFIG(TAG, "Setting up DPS310...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
// first, reset the sensor
|
// first, reset the sensor
|
||||||
if (!this->write_byte(DPS310_REG_RESET, DPS310_CMD_RESET)) {
|
if (!this->write_byte(DPS310_REG_RESET, DPS310_CMD_RESET)) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
@@ -91,7 +91,7 @@ void DPS310Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, " Revision ID: %u", (this->prod_rev_id_ >> 4) & 0x0F);
|
ESP_LOGCONFIG(TAG, " Revision ID: %u", (this->prod_rev_id_ >> 4) & 0x0F);
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with DPS310 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
LOG_UPDATE_INTERVAL(this);
|
LOG_UPDATE_INTERVAL(this);
|
||||||
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
|
||||||
|
@@ -10,7 +10,7 @@ namespace ds1307 {
|
|||||||
static const char *const TAG = "ds1307";
|
static const char *const TAG = "ds1307";
|
||||||
|
|
||||||
void DS1307Component::setup() {
|
void DS1307Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up DS1307...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
if (!this->read_rtc_()) {
|
if (!this->read_rtc_()) {
|
||||||
this->mark_failed();
|
this->mark_failed();
|
||||||
}
|
}
|
||||||
@@ -22,7 +22,7 @@ void DS1307Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "DS1307:");
|
ESP_LOGCONFIG(TAG, "DS1307:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with DS1307 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
ESP_LOGCONFIG(TAG, " Timezone: '%s'", this->timezone_.c_str());
|
ESP_LOGCONFIG(TAG, " Timezone: '%s'", this->timezone_.c_str());
|
||||||
}
|
}
|
||||||
|
@@ -8,7 +8,7 @@ namespace duty_cycle {
|
|||||||
static const char *const TAG = "duty_cycle";
|
static const char *const TAG = "duty_cycle";
|
||||||
|
|
||||||
void DutyCycleSensor::setup() {
|
void DutyCycleSensor::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up Duty Cycle Sensor '%s'...", this->get_name().c_str());
|
ESP_LOGCONFIG(TAG, "Running setup for '%s'", this->get_name().c_str());
|
||||||
this->pin_->setup();
|
this->pin_->setup();
|
||||||
this->store_.pin = this->pin_->to_isr();
|
this->store_.pin = this->pin_->to_isr();
|
||||||
this->store_.last_level = this->pin_->digital_read();
|
this->store_.last_level = this->pin_->digital_read();
|
||||||
|
@@ -16,7 +16,7 @@ static const uint16_t PRESSURE_ADDRESS = 0x04B0;
|
|||||||
|
|
||||||
void EE895Component::setup() {
|
void EE895Component::setup() {
|
||||||
uint16_t crc16_check = 0;
|
uint16_t crc16_check = 0;
|
||||||
ESP_LOGCONFIG(TAG, "Setting up EE895...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
write_command_(SERIAL_NUMBER, 8);
|
write_command_(SERIAL_NUMBER, 8);
|
||||||
uint8_t serial_number[20];
|
uint8_t serial_number[20];
|
||||||
this->read(serial_number, 20);
|
this->read(serial_number, 20);
|
||||||
@@ -35,7 +35,7 @@ void EE895Component::dump_config() {
|
|||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
switch (this->error_code_) {
|
switch (this->error_code_) {
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication with EE895 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case CRC_CHECK_FAILED:
|
case CRC_CHECK_FAILED:
|
||||||
ESP_LOGE(TAG, "The crc check failed");
|
ESP_LOGE(TAG, "The crc check failed");
|
||||||
|
@@ -16,7 +16,7 @@ static const uint8_t GET_Y_RES[4] = {0x53, 0x63, 0x00, 0x00};
|
|||||||
static const uint8_t GET_POWER_STATE_CMD[4] = {0x53, 0x50, 0x00, 0x01};
|
static const uint8_t GET_POWER_STATE_CMD[4] = {0x53, 0x50, 0x00, 0x01};
|
||||||
|
|
||||||
void EKTF2232Touchscreen::setup() {
|
void EKTF2232Touchscreen::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up EKT2232 Touchscreen...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
this->interrupt_pin_->pin_mode(gpio::FLAG_INPUT | gpio::FLAG_PULLUP);
|
this->interrupt_pin_->pin_mode(gpio::FLAG_INPUT | gpio::FLAG_PULLUP);
|
||||||
this->interrupt_pin_->setup();
|
this->interrupt_pin_->setup();
|
||||||
|
|
||||||
|
@@ -57,7 +57,7 @@ static const uint8_t EMC2101_POLARITY_BIT = 1 << 4;
|
|||||||
float Emc2101Component::get_setup_priority() const { return setup_priority::HARDWARE; }
|
float Emc2101Component::get_setup_priority() const { return setup_priority::HARDWARE; }
|
||||||
|
|
||||||
void Emc2101Component::setup() {
|
void Emc2101Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up Emc2101 sensor...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
// make sure we're talking to the right chip
|
// make sure we're talking to the right chip
|
||||||
uint8_t chip_id = reg(EMC2101_REGISTER_WHOAMI).get();
|
uint8_t chip_id = reg(EMC2101_REGISTER_WHOAMI).get();
|
||||||
@@ -94,7 +94,7 @@ void Emc2101Component::dump_config() {
|
|||||||
ESP_LOGCONFIG(TAG, "Emc2101 component:");
|
ESP_LOGCONFIG(TAG, "Emc2101 component:");
|
||||||
LOG_I2C_DEVICE(this);
|
LOG_I2C_DEVICE(this);
|
||||||
if (this->is_failed()) {
|
if (this->is_failed()) {
|
||||||
ESP_LOGE(TAG, "Communication with EMC2101 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
}
|
}
|
||||||
ESP_LOGCONFIG(TAG, " Mode: %s", this->dac_mode_ ? "DAC" : "PWM");
|
ESP_LOGCONFIG(TAG, " Mode: %s", this->dac_mode_ ? "DAC" : "PWM");
|
||||||
if (this->dac_mode_) {
|
if (this->dac_mode_) {
|
||||||
@@ -110,7 +110,7 @@ void Emc2101Component::set_duty_cycle(float value) {
|
|||||||
uint8_t duty_cycle = remap(value, 0.0f, 1.0f, (uint8_t) 0, this->max_output_value_);
|
uint8_t duty_cycle = remap(value, 0.0f, 1.0f, (uint8_t) 0, this->max_output_value_);
|
||||||
ESP_LOGD(TAG, "Setting duty fan setting to %02X", duty_cycle);
|
ESP_LOGD(TAG, "Setting duty fan setting to %02X", duty_cycle);
|
||||||
if (!this->write_byte(EMC2101_REGISTER_FAN_SETTING, duty_cycle)) {
|
if (!this->write_byte(EMC2101_REGISTER_FAN_SETTING, duty_cycle)) {
|
||||||
ESP_LOGE(TAG, "Communication with EMC2101 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -119,7 +119,7 @@ void Emc2101Component::set_duty_cycle(float value) {
|
|||||||
float Emc2101Component::get_duty_cycle() {
|
float Emc2101Component::get_duty_cycle() {
|
||||||
uint8_t duty_cycle;
|
uint8_t duty_cycle;
|
||||||
if (!this->read_byte(EMC2101_REGISTER_FAN_SETTING, &duty_cycle)) {
|
if (!this->read_byte(EMC2101_REGISTER_FAN_SETTING, &duty_cycle)) {
|
||||||
ESP_LOGE(TAG, "Communication with EMC2101 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
return NAN;
|
return NAN;
|
||||||
}
|
}
|
||||||
@@ -129,7 +129,7 @@ float Emc2101Component::get_duty_cycle() {
|
|||||||
float Emc2101Component::get_internal_temperature() {
|
float Emc2101Component::get_internal_temperature() {
|
||||||
uint8_t temperature;
|
uint8_t temperature;
|
||||||
if (!this->read_byte(EMC2101_REGISTER_INTERNAL_TEMP, &temperature)) {
|
if (!this->read_byte(EMC2101_REGISTER_INTERNAL_TEMP, &temperature)) {
|
||||||
ESP_LOGE(TAG, "Communication with EMC2101 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
return NAN;
|
return NAN;
|
||||||
}
|
}
|
||||||
@@ -141,7 +141,7 @@ float Emc2101Component::get_external_temperature() {
|
|||||||
uint8_t lsb, msb;
|
uint8_t lsb, msb;
|
||||||
if (!this->read_byte(EMC2101_REGISTER_EXTERNAL_TEMP_MSB, &msb) ||
|
if (!this->read_byte(EMC2101_REGISTER_EXTERNAL_TEMP_MSB, &msb) ||
|
||||||
!this->read_byte(EMC2101_REGISTER_EXTERNAL_TEMP_LSB, &lsb)) {
|
!this->read_byte(EMC2101_REGISTER_EXTERNAL_TEMP_LSB, &lsb)) {
|
||||||
ESP_LOGE(TAG, "Communication with EMC2101 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
return NAN;
|
return NAN;
|
||||||
}
|
}
|
||||||
@@ -155,7 +155,7 @@ float Emc2101Component::get_speed() {
|
|||||||
// Read **LSB** first to match 'Data Read Interlock' behavior from 6.1 of datasheet
|
// Read **LSB** first to match 'Data Read Interlock' behavior from 6.1 of datasheet
|
||||||
uint8_t lsb, msb;
|
uint8_t lsb, msb;
|
||||||
if (!this->read_byte(EMC2101_REGISTER_TACH_LSB, &lsb) || !this->read_byte(EMC2101_REGISTER_TACH_MSB, &msb)) {
|
if (!this->read_byte(EMC2101_REGISTER_TACH_LSB, &lsb) || !this->read_byte(EMC2101_REGISTER_TACH_MSB, &msb)) {
|
||||||
ESP_LOGE(TAG, "Communication with EMC2101 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
return NAN;
|
return NAN;
|
||||||
}
|
}
|
||||||
|
@@ -49,7 +49,7 @@ static const uint8_t ENS160_DATA_STATUS_NEWGPR = 0x01;
|
|||||||
static const uint8_t ENS160_DATA_AQI = 0x07;
|
static const uint8_t ENS160_DATA_AQI = 0x07;
|
||||||
|
|
||||||
void ENS160Component::setup() {
|
void ENS160Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ENS160...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
// check part_id
|
// check part_id
|
||||||
uint16_t part_id;
|
uint16_t part_id;
|
||||||
@@ -279,7 +279,7 @@ void ENS160Component::dump_config() {
|
|||||||
|
|
||||||
switch (this->error_code_) {
|
switch (this->error_code_) {
|
||||||
case COMMUNICATION_FAILED:
|
case COMMUNICATION_FAILED:
|
||||||
ESP_LOGE(TAG, "Communication failed! Is the sensor connected?");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
break;
|
break;
|
||||||
case READ_FAILED:
|
case READ_FAILED:
|
||||||
ESP_LOGE(TAG, "Error reading from register");
|
ESP_LOGE(TAG, "Error reading from register");
|
||||||
|
@@ -87,7 +87,7 @@ static uint32_t crc7(uint32_t value) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ENS210Component::setup() {
|
void ENS210Component::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ENS210...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
uint8_t data[2];
|
uint8_t data[2];
|
||||||
uint16_t part_id = 0;
|
uint16_t part_id = 0;
|
||||||
// Reset
|
// Reset
|
||||||
@@ -163,7 +163,7 @@ void ENS210Component::update() {
|
|||||||
|
|
||||||
// Read T_VAL and H_VAL
|
// Read T_VAL and H_VAL
|
||||||
if (!this->read_bytes(ENS210_REGISTER_T_VAL, data, 6)) {
|
if (!this->read_bytes(ENS210_REGISTER_T_VAL, data, 6)) {
|
||||||
ESP_LOGE(TAG, "Communication with ENS210 failed!");
|
ESP_LOGE(TAG, ESP_LOG_MSG_COMM_FAIL);
|
||||||
this->status_set_warning();
|
this->status_set_warning();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@@ -36,7 +36,7 @@ void ES7210::dump_config() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ES7210::setup() {
|
void ES7210::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ES7210...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
// Software reset
|
// Software reset
|
||||||
ES7210_ERROR_FAILED(this->write_byte(ES7210_RESET_REG00, 0xff));
|
ES7210_ERROR_FAILED(this->write_byte(ES7210_RESET_REG00, 0xff));
|
||||||
|
@@ -34,7 +34,7 @@ void ES7243E::dump_config() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ES7243E::setup() {
|
void ES7243E::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ES7243E...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
ES7243E_ERROR_FAILED(this->write_byte(ES7243E_CLOCK_MGR_REG01, 0x3A));
|
ES7243E_ERROR_FAILED(this->write_byte(ES7243E_CLOCK_MGR_REG01, 0x3A));
|
||||||
ES7243E_ERROR_FAILED(this->write_byte(ES7243E_RESET_REG00, 0x80));
|
ES7243E_ERROR_FAILED(this->write_byte(ES7243E_RESET_REG00, 0x80));
|
||||||
|
@@ -17,7 +17,7 @@ static const char *const TAG = "es8156";
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ES8156::setup() {
|
void ES8156::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ES8156...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
ES8156_ERROR_FAILED(this->write_byte(ES8156_REG02_SCLK_MODE, 0x04));
|
ES8156_ERROR_FAILED(this->write_byte(ES8156_REG02_SCLK_MODE, 0x04));
|
||||||
ES8156_ERROR_FAILED(this->write_byte(ES8156_REG20_ANALOG_SYS1, 0x2A));
|
ES8156_ERROR_FAILED(this->write_byte(ES8156_REG20_ANALOG_SYS1, 0x2A));
|
||||||
|
@@ -22,7 +22,7 @@ static const char *const TAG = "es8311";
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ES8311::setup() {
|
void ES8311::setup() {
|
||||||
ESP_LOGCONFIG(TAG, "Setting up ES8311...");
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
// Reset
|
// Reset
|
||||||
ES8311_ERROR_FAILED(this->write_byte(ES8311_REG00_RESET, 0x1F));
|
ES8311_ERROR_FAILED(this->write_byte(ES8311_REG00_RESET, 0x1F));
|
||||||
|
0
esphome/components/es8388/__init__.py
Normal file
0
esphome/components/es8388/__init__.py
Normal file
26
esphome/components/es8388/audio_dac.py
Normal file
26
esphome/components/es8388/audio_dac.py
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
import esphome.codegen as cg
|
||||||
|
from esphome.components import i2c
|
||||||
|
from esphome.components.audio_dac import AudioDac
|
||||||
|
import esphome.config_validation as cv
|
||||||
|
from esphome.const import CONF_ID
|
||||||
|
|
||||||
|
CODEOWNERS = ["@P4uLT"]
|
||||||
|
CONF_ES8388_ID = "es8388_id"
|
||||||
|
|
||||||
|
es8388_ns = cg.esphome_ns.namespace("es8388")
|
||||||
|
|
||||||
|
ES8388 = es8388_ns.class_("ES8388", AudioDac, cg.Component, i2c.I2CDevice)
|
||||||
|
|
||||||
|
DEPENDENCIES = ["i2c"]
|
||||||
|
|
||||||
|
CONFIG_SCHEMA = (
|
||||||
|
cv.Schema({cv.GenerateID(): cv.declare_id(ES8388)})
|
||||||
|
.extend(cv.COMPONENT_SCHEMA)
|
||||||
|
.extend(i2c.i2c_device_schema(0x10))
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def to_code(config):
|
||||||
|
var = cg.new_Pvariable(config[CONF_ID])
|
||||||
|
await cg.register_component(var, config)
|
||||||
|
await i2c.register_i2c_device(var, config)
|
289
esphome/components/es8388/es8388.cpp
Normal file
289
esphome/components/es8388/es8388.cpp
Normal file
@@ -0,0 +1,289 @@
|
|||||||
|
#include "es8388.h"
|
||||||
|
|
||||||
|
#include <cinttypes>
|
||||||
|
#include "esphome/core/hal.h"
|
||||||
|
#include "esphome/core/log.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace es8388 {
|
||||||
|
|
||||||
|
static const char *const TAG = "es8388";
|
||||||
|
|
||||||
|
// Mark the component as failed; use only in setup
|
||||||
|
#define ES8388_ERROR_FAILED(func) \
|
||||||
|
if (!(func)) { \
|
||||||
|
this->mark_failed(); \
|
||||||
|
return; \
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return false; use outside of setup
|
||||||
|
#define ES8388_ERROR_CHECK(func) \
|
||||||
|
if (!(func)) { \
|
||||||
|
return false; \
|
||||||
|
}
|
||||||
|
|
||||||
|
void ES8388::setup() {
|
||||||
|
ESP_LOGCONFIG(TAG, "Running setup");
|
||||||
|
|
||||||
|
// mute DAC
|
||||||
|
this->set_mute_state_(true);
|
||||||
|
|
||||||
|
// I2S worker mode
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_MASTERMODE, 0x00));
|
||||||
|
|
||||||
|
/* Chip Control and Power Management */
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_CONTROL2, 0x50));
|
||||||
|
// normal all and power up all
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_CHIPPOWER, 0x00));
|
||||||
|
|
||||||
|
// vmidsel/500k
|
||||||
|
// EnRef=0,Play&Record Mode,(0x17-both of mic&play)
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_CONTROL1, 0x12));
|
||||||
|
|
||||||
|
// i2s 16 bits
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_DACCONTROL1, 0x18));
|
||||||
|
// sample freq 256
|
||||||
|
// DACFsMode,SINGLE SPEED; DACFsRatio,256
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_DACCONTROL2, 0x02));
|
||||||
|
// 0x00 audio on LIN1&RIN1, 0x09 LIN2&RIN2
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_DACCONTROL16, 0x00));
|
||||||
|
// only left DAC to left mixer enable 0db
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_DACCONTROL17, 0x90));
|
||||||
|
// only right DAC to right mixer enable 0db
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_DACCONTROL20, 0x90));
|
||||||
|
// set internal ADC and DAC use the same LRCK clock, ADC LRCK as internal LRCK
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_DACCONTROL21, 0x80));
|
||||||
|
// vroi=0 - 1.5k VREF to analog output resistance (default)
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_DACCONTROL23, 0x00));
|
||||||
|
|
||||||
|
// power down adc and line in
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCPOWER, 0xFF));
|
||||||
|
|
||||||
|
//@nightdav
|
||||||
|
ES8388_ERROR_FAILED(
|
||||||
|
this->write_byte(ES8388_ADCCONTROL1, 0x00)); // +21dB : recommended value for ALC & voice recording
|
||||||
|
|
||||||
|
// set to Mono Right
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL3, 0x02));
|
||||||
|
|
||||||
|
// I2S 16 Bits length and I2S serial audio data format
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL4, 0x0d));
|
||||||
|
// ADCFsMode,singel SPEED,RATIO=256
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL5, 0x02));
|
||||||
|
|
||||||
|
// ADC Volume
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL8, 0x00));
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL9, 0x00));
|
||||||
|
|
||||||
|
//@nightDav
|
||||||
|
// ALC Config (as recommended by ES8388 user guide for voice recording)
|
||||||
|
|
||||||
|
// Reg 0x12 = 0xe2 (ALC enable, PGA Max. Gain=23.5dB, Min. Gain=0dB)
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL10, 0xe2));
|
||||||
|
|
||||||
|
// Reg 0x13 = 0xa0 (ALC Target=-1.5dB, ALC Hold time =0 mS)
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL11, 0xa0));
|
||||||
|
// Reg 0x14 = 0x12(Decay time =820uS , Attack time = 416 uS)
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL12, 0x12));
|
||||||
|
|
||||||
|
// Reg 0x15 = 0x06(ALC mode)
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL13, 0x06));
|
||||||
|
|
||||||
|
// Reg 0x16 = 0xc3(nose gate = -40.5dB, NGG = 0x01(mute ADC))
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL14, 0xc3));
|
||||||
|
|
||||||
|
// Power on ADC
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_DACCONTROL21, 0x80));
|
||||||
|
|
||||||
|
// Start state machine
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_CHIPPOWER, 0xF0));
|
||||||
|
delay(1);
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_CHIPPOWER, 0x00));
|
||||||
|
|
||||||
|
// DAC volume max
|
||||||
|
// Set initial volume
|
||||||
|
// this->set_volume(0.75); // 0.75 = 0xBF = 0dB
|
||||||
|
|
||||||
|
this->set_mute_state_(false);
|
||||||
|
|
||||||
|
// unmute ADC with fade in
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCCONTROL7, 0x60));
|
||||||
|
// unmute DAC with fade in
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_DACCONTROL3, 0x20));
|
||||||
|
|
||||||
|
// Power on ADC, Enable LIN&RIN, Power off MICBIAS, set int1lp to low power mode
|
||||||
|
ES8388_ERROR_FAILED(this->write_byte(ES8388_ADCPOWER, 0x09));
|
||||||
|
|
||||||
|
#ifdef USE_SELECT
|
||||||
|
if (this->dac_output_select_ != nullptr) {
|
||||||
|
auto dac_power = this->get_dac_power();
|
||||||
|
if (dac_power.has_value()) {
|
||||||
|
auto dac_power_str = this->dac_output_select_->at(dac_power.value());
|
||||||
|
if (dac_power_str.has_value()) {
|
||||||
|
this->dac_output_select_->publish_state(dac_power_str.value());
|
||||||
|
} else {
|
||||||
|
ESP_LOGW(TAG, "Unknown DAC output power value: %d", dac_power.value());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (this->adc_input_mic_select_ != nullptr) {
|
||||||
|
auto mic_input = this->get_mic_input();
|
||||||
|
if (mic_input.has_value()) {
|
||||||
|
auto mic_input_str = this->adc_input_mic_select_->at(mic_input.value());
|
||||||
|
if (mic_input_str.has_value()) {
|
||||||
|
this->adc_input_mic_select_->publish_state(mic_input_str.value());
|
||||||
|
} else {
|
||||||
|
ESP_LOGW(TAG, "Unknown ADC input mic value: %d", mic_input.value());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void ES8388::dump_config() {
|
||||||
|
ESP_LOGCONFIG(TAG, "ES8388 Audio Codec:");
|
||||||
|
LOG_I2C_DEVICE(this);
|
||||||
|
#ifdef USE_SELECT
|
||||||
|
LOG_SELECT(" ", "DacOutputSelect", this->dac_output_select_);
|
||||||
|
LOG_SELECT(" ", "ADCInputMicSelect", this->adc_input_mic_select_);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (this->is_failed()) {
|
||||||
|
ESP_LOGCONFIG(TAG, " Failed to initialize");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ES8388::set_volume(float volume) {
|
||||||
|
volume = clamp(volume, 0.0f, 1.0f);
|
||||||
|
uint8_t value = remap<uint8_t, float>(volume, 0.0f, 1.0f, -96, 0);
|
||||||
|
ESP_LOGD(TAG, "Setting ES8388_DACCONTROL4 / ES8388_DACCONTROL5 to 0x%02X (volume: %f)", value, volume);
|
||||||
|
ES8388_ERROR_CHECK(this->write_byte(ES8388_DACCONTROL4, value));
|
||||||
|
ES8388_ERROR_CHECK(this->write_byte(ES8388_DACCONTROL5, value));
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
float ES8388::volume() {
|
||||||
|
uint8_t value;
|
||||||
|
ES8388_ERROR_CHECK(this->read_byte(ES8388_DACCONTROL4, &value));
|
||||||
|
return remap<float, uint8_t>(value, -96, 0, 0.0f, 1.0f);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ES8388::set_mute_state_(bool mute_state) {
|
||||||
|
uint8_t value = 0;
|
||||||
|
|
||||||
|
this->is_muted_ = mute_state;
|
||||||
|
|
||||||
|
ES8388_ERROR_CHECK(this->read_byte(ES8388_DACCONTROL3, &value));
|
||||||
|
ESP_LOGV(TAG, "Read ES8388_DACCONTROL3: 0x%02X", value);
|
||||||
|
|
||||||
|
if (mute_state) {
|
||||||
|
value = 0x3C;
|
||||||
|
}
|
||||||
|
|
||||||
|
ESP_LOGV(TAG, "Setting ES8388_DACCONTROL3 to 0x%02X (muted: %s)", value, YESNO(mute_state));
|
||||||
|
return this->write_byte(ES8388_DACCONTROL3, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set dac power output
|
||||||
|
bool ES8388::set_dac_output(DacOutputLine line) {
|
||||||
|
uint8_t reg_out1 = 0;
|
||||||
|
uint8_t reg_out2 = 0;
|
||||||
|
uint8_t dac_power = 0;
|
||||||
|
|
||||||
|
// 0x00: -30dB , 0x1E: 0dB
|
||||||
|
switch (line) {
|
||||||
|
case DAC_OUTPUT_LINE1:
|
||||||
|
reg_out1 = 0x1E;
|
||||||
|
dac_power = ES8388_DAC_OUTPUT_LOUT1_ROUT1;
|
||||||
|
break;
|
||||||
|
case DAC_OUTPUT_LINE2:
|
||||||
|
reg_out2 = 0x1E;
|
||||||
|
dac_power = ES8388_DAC_OUTPUT_LOUT2_ROUT2;
|
||||||
|
break;
|
||||||
|
case DAC_OUTPUT_BOTH:
|
||||||
|
reg_out1 = 0x1E;
|
||||||
|
reg_out2 = 0x1E;
|
||||||
|
dac_power = ES8388_DAC_OUTPUT_BOTH;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ESP_LOGE(TAG, "Unknown DAC output line: %d", line);
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
ESP_LOGV(TAG, "Setting ES8388_DACPOWER to 0x%02X", dac_power);
|
||||||
|
ESP_LOGV(TAG, "Setting ES8388_DACCONTROL24 / ES8388_DACCONTROL25 to 0x%02X", reg_out1);
|
||||||
|
ESP_LOGV(TAG, "Setting ES8388_DACCONTROL26 / ES8388_DACCONTROL27 to 0x%02X", reg_out2);
|
||||||
|
|
||||||
|
ES8388_ERROR_CHECK(this->write_byte(ES8388_DACCONTROL24, reg_out1)); // LOUT1VOL
|
||||||
|
ES8388_ERROR_CHECK(this->write_byte(ES8388_DACCONTROL25, reg_out1)); // ROUT1VOL
|
||||||
|
ES8388_ERROR_CHECK(this->write_byte(ES8388_DACCONTROL26, reg_out2)); // LOUT2VOL
|
||||||
|
ES8388_ERROR_CHECK(this->write_byte(ES8388_DACCONTROL27, reg_out2)); // ROUT1VOL
|
||||||
|
|
||||||
|
return this->write_byte(ES8388_DACPOWER, dac_power);
|
||||||
|
}
|
||||||
|
|
||||||
|
optional<DacOutputLine> ES8388::get_dac_power() {
|
||||||
|
uint8_t dac_power;
|
||||||
|
if (!this->read_byte(ES8388_DACPOWER, &dac_power)) {
|
||||||
|
this->status_momentary_warning("Failed to read ES8388_DACPOWER");
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
switch (dac_power) {
|
||||||
|
case ES8388_DAC_OUTPUT_LOUT1_ROUT1:
|
||||||
|
return DAC_OUTPUT_LINE1;
|
||||||
|
case ES8388_DAC_OUTPUT_LOUT2_ROUT2:
|
||||||
|
return DAC_OUTPUT_LINE2;
|
||||||
|
case ES8388_DAC_OUTPUT_BOTH:
|
||||||
|
return DAC_OUTPUT_BOTH;
|
||||||
|
default:
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set ADC input MIC
|
||||||
|
bool ES8388::set_adc_input_mic(AdcInputMicLine line) {
|
||||||
|
uint8_t mic_input = 0;
|
||||||
|
|
||||||
|
switch (line) {
|
||||||
|
case ADC_INPUT_MIC_LINE1:
|
||||||
|
mic_input = ES8388_ADC_INPUT_LINPUT1_RINPUT1;
|
||||||
|
break;
|
||||||
|
case ADC_INPUT_MIC_LINE2:
|
||||||
|
mic_input = ES8388_ADC_INPUT_LINPUT2_RINPUT2;
|
||||||
|
break;
|
||||||
|
case ADC_INPUT_MIC_DIFFERENCE:
|
||||||
|
mic_input = ES8388_ADC_INPUT_DIFFERENCE;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ESP_LOGE(TAG, "Unknown ADC input mic line: %d", line);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
ESP_LOGV(TAG, "Setting ES8388_ADCCONTROL2 to 0x%02X", mic_input);
|
||||||
|
ES8388_ERROR_CHECK(this->write_byte(ES8388_ADCCONTROL2, mic_input));
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
optional<AdcInputMicLine> ES8388::get_mic_input() {
|
||||||
|
uint8_t mic_input;
|
||||||
|
if (!this->read_byte(ES8388_ADCCONTROL2, &mic_input)) {
|
||||||
|
this->status_momentary_warning("Failed to read ES8388_ADCCONTROL2");
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
switch (mic_input) {
|
||||||
|
case ES8388_ADC_INPUT_LINPUT1_RINPUT1:
|
||||||
|
return ADC_INPUT_MIC_LINE1;
|
||||||
|
case ES8388_ADC_INPUT_LINPUT2_RINPUT2:
|
||||||
|
return ADC_INPUT_MIC_LINE2;
|
||||||
|
case ES8388_ADC_INPUT_DIFFERENCE:
|
||||||
|
return ADC_INPUT_MIC_DIFFERENCE;
|
||||||
|
default:
|
||||||
|
return {};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace es8388
|
||||||
|
} // namespace esphome
|
81
esphome/components/es8388/es8388.h
Normal file
81
esphome/components/es8388/es8388.h
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <map>
|
||||||
|
|
||||||
|
#include "esphome/components/audio_dac/audio_dac.h"
|
||||||
|
#include "esphome/components/i2c/i2c.h"
|
||||||
|
#include "esphome/core/component.h"
|
||||||
|
#ifdef USE_SELECT
|
||||||
|
#include "esphome/components/select/select.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "es8388_const.h"
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace es8388 {
|
||||||
|
|
||||||
|
enum DacOutputLine : uint8_t {
|
||||||
|
DAC_OUTPUT_LINE1,
|
||||||
|
DAC_OUTPUT_LINE2,
|
||||||
|
DAC_OUTPUT_BOTH,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum AdcInputMicLine : uint8_t {
|
||||||
|
ADC_INPUT_MIC_LINE1,
|
||||||
|
ADC_INPUT_MIC_LINE2,
|
||||||
|
ADC_INPUT_MIC_DIFFERENCE,
|
||||||
|
};
|
||||||
|
|
||||||
|
class ES8388 : public audio_dac::AudioDac, public Component, public i2c::I2CDevice {
|
||||||
|
#ifdef USE_SELECT
|
||||||
|
SUB_SELECT(dac_output)
|
||||||
|
SUB_SELECT(adc_input_mic)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
public:
|
||||||
|
/////////////////////////
|
||||||
|
// Component overrides //
|
||||||
|
/////////////////////////
|
||||||
|
|
||||||
|
void setup() override;
|
||||||
|
float get_setup_priority() const override { return setup_priority::DATA; }
|
||||||
|
void dump_config() override;
|
||||||
|
|
||||||
|
////////////////////////
|
||||||
|
// AudioDac overrides //
|
||||||
|
////////////////////////
|
||||||
|
|
||||||
|
/// @brief Writes the volume out to the DAC
|
||||||
|
/// @param volume floating point between 0.0 and 1.0
|
||||||
|
/// @return True if successful and false otherwise
|
||||||
|
bool set_volume(float volume) override;
|
||||||
|
|
||||||
|
/// @brief Gets the current volume out from the DAC
|
||||||
|
/// @return floating point between 0.0 and 1.0
|
||||||
|
float volume() override;
|
||||||
|
|
||||||
|
/// @brief Disables mute for audio out
|
||||||
|
/// @return True if successful and false otherwise
|
||||||
|
bool set_mute_off() override { return this->set_mute_state_(false); }
|
||||||
|
|
||||||
|
/// @brief Enables mute for audio out
|
||||||
|
/// @return True if successful and false otherwise
|
||||||
|
bool set_mute_on() override { return this->set_mute_state_(true); }
|
||||||
|
|
||||||
|
bool is_muted() override { return this->is_muted_; }
|
||||||
|
|
||||||
|
optional<DacOutputLine> get_dac_power();
|
||||||
|
optional<AdcInputMicLine> get_mic_input();
|
||||||
|
|
||||||
|
bool set_dac_output(DacOutputLine line);
|
||||||
|
bool set_adc_input_mic(AdcInputMicLine line);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
/// @brief Mutes or unmutes the DAC audio out
|
||||||
|
/// @param mute_state True to mute, false to unmute
|
||||||
|
/// @return True if successful and false otherwise
|
||||||
|
bool set_mute_state_(bool mute_state);
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace es8388
|
||||||
|
} // namespace esphome
|
83
esphome/components/es8388/es8388_const.h
Normal file
83
esphome/components/es8388/es8388_const.h
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
#pragma once
|
||||||
|
#include <cstdint>
|
||||||
|
|
||||||
|
namespace esphome {
|
||||||
|
namespace es8388 {
|
||||||
|
|
||||||
|
/* ES8388 register */
|
||||||
|
static const uint8_t ES8388_CONTROL1 = 0x00;
|
||||||
|
static const uint8_t ES8388_CONTROL2 = 0x01;
|
||||||
|
|
||||||
|
static const uint8_t ES8388_CHIPPOWER = 0x02;
|
||||||
|
|
||||||
|
static const uint8_t ES8388_ADCPOWER = 0x03;
|
||||||
|
static const uint8_t ES8388_DACPOWER = 0x04;
|
||||||
|
|
||||||
|
static const uint8_t ES8388_CHIPLOPOW1 = 0x05;
|
||||||
|
static const uint8_t ES8388_CHIPLOPOW2 = 0x06;
|
||||||
|
|
||||||
|
static const uint8_t ES8388_ANAVOLMANAG = 0x07;
|
||||||
|
|
||||||
|
static const uint8_t ES8388_MASTERMODE = 0x08;
|
||||||
|
|
||||||
|
/* ADC */
|
||||||
|
static const uint8_t ES8388_ADCCONTROL1 = 0x09;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL2 = 0x0a;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL3 = 0x0b;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL4 = 0x0c;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL5 = 0x0d;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL6 = 0x0e;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL7 = 0x0f;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL8 = 0x10;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL9 = 0x11;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL10 = 0x12;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL11 = 0x13;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL12 = 0x14;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL13 = 0x15;
|
||||||
|
static const uint8_t ES8388_ADCCONTROL14 = 0x16;
|
||||||
|
/* DAC */
|
||||||
|
static const uint8_t ES8388_DACCONTROL1 = 0x17;
|
||||||
|
static const uint8_t ES8388_DACCONTROL2 = 0x18;
|
||||||
|
static const uint8_t ES8388_DACCONTROL3 = 0x19;
|
||||||
|
static const uint8_t ES8388_DACCONTROL4 = 0x1a;
|
||||||
|
static const uint8_t ES8388_DACCONTROL5 = 0x1b;
|
||||||
|
static const uint8_t ES8388_DACCONTROL6 = 0x1c;
|
||||||
|
static const uint8_t ES8388_DACCONTROL7 = 0x1d;
|
||||||
|
static const uint8_t ES8388_DACCONTROL8 = 0x1e;
|
||||||
|
static const uint8_t ES8388_DACCONTROL9 = 0x1f;
|
||||||
|
static const uint8_t ES8388_DACCONTROL10 = 0x20;
|
||||||
|
static const uint8_t ES8388_DACCONTROL11 = 0x21;
|
||||||
|
static const uint8_t ES8388_DACCONTROL12 = 0x22;
|
||||||
|
static const uint8_t ES8388_DACCONTROL13 = 0x23;
|
||||||
|
static const uint8_t ES8388_DACCONTROL14 = 0x24;
|
||||||
|
static const uint8_t ES8388_DACCONTROL15 = 0x25;
|
||||||
|
static const uint8_t ES8388_DACCONTROL16 = 0x26;
|
||||||
|
static const uint8_t ES8388_DACCONTROL17 = 0x27;
|
||||||
|
static const uint8_t ES8388_DACCONTROL18 = 0x28;
|
||||||
|
static const uint8_t ES8388_DACCONTROL19 = 0x29;
|
||||||
|
static const uint8_t ES8388_DACCONTROL20 = 0x2a;
|
||||||
|
static const uint8_t ES8388_DACCONTROL21 = 0x2b;
|
||||||
|
static const uint8_t ES8388_DACCONTROL22 = 0x2c;
|
||||||
|
static const uint8_t ES8388_DACCONTROL23 = 0x2d;
|
||||||
|
static const uint8_t ES8388_DACCONTROL24 = 0x2e;
|
||||||
|
static const uint8_t ES8388_DACCONTROL25 = 0x2f;
|
||||||
|
static const uint8_t ES8388_DACCONTROL26 = 0x30;
|
||||||
|
static const uint8_t ES8388_DACCONTROL27 = 0x31;
|
||||||
|
static const uint8_t ES8388_DACCONTROL28 = 0x32;
|
||||||
|
static const uint8_t ES8388_DACCONTROL29 = 0x33;
|
||||||
|
static const uint8_t ES8388_DACCONTROL30 = 0x34;
|
||||||
|
|
||||||
|
static const uint8_t ES8388_DAC_OUTPUT_NONE = 0xC0; // ALL DAC DOWN
|
||||||
|
|
||||||
|
static const uint8_t ES8388_DAC_OUTPUT_LOUT1_ROUT1 = 0x30;
|
||||||
|
static const uint8_t ES8388_DAC_OUTPUT_LOUT2_ROUT2 = 0x0C;
|
||||||
|
static const uint8_t ES8388_DAC_OUTPUT_BOTH = 0x3C;
|
||||||
|
|
||||||
|
static const uint8_t ES8388_ADC_INPUT_LINPUT1_RINPUT1 = 0x00;
|
||||||
|
static const uint8_t ES8388_ADC_INPUT_MIC1 = 0x05;
|
||||||
|
static const uint8_t ES8388_ADC_INPUT_MIC2 = 0x06;
|
||||||
|
static const uint8_t ES8388_ADC_INPUT_LINPUT2_RINPUT2 = 0x50;
|
||||||
|
static const uint8_t ES8388_ADC_INPUT_DIFFERENCE = 0xf0;
|
||||||
|
|
||||||
|
} // namespace es8388
|
||||||
|
} // namespace esphome
|
47
esphome/components/es8388/select/__init__.py
Normal file
47
esphome/components/es8388/select/__init__.py
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
import esphome.codegen as cg
|
||||||
|
from esphome.components import select
|
||||||
|
import esphome.config_validation as cv
|
||||||
|
from esphome.const import ENTITY_CATEGORY_CONFIG, ICON_CHIP # noqa: F401
|
||||||
|
|
||||||
|
from ..audio_dac import CONF_ES8388_ID, ES8388, es8388_ns
|
||||||
|
|
||||||
|
CONF_DAC_OUTPUT = "dac_output"
|
||||||
|
CONF_ADC_INPUT_MIC = "adc_input_mic"
|
||||||
|
|
||||||
|
DacOutputSelect = es8388_ns.class_("DacOutputSelect", select.Select)
|
||||||
|
ADCInputMicSelect = es8388_ns.class_("ADCInputMicSelect", select.Select)
|
||||||
|
|
||||||
|
CONFIG_SCHEMA = cv.All(
|
||||||
|
{
|
||||||
|
cv.GenerateID(CONF_ES8388_ID): cv.use_id(ES8388),
|
||||||
|
cv.Optional(CONF_DAC_OUTPUT): select.select_schema(
|
||||||
|
DacOutputSelect,
|
||||||
|
entity_category=ENTITY_CATEGORY_CONFIG,
|
||||||
|
icon=ICON_CHIP,
|
||||||
|
),
|
||||||
|
cv.Optional(CONF_ADC_INPUT_MIC): select.select_schema(
|
||||||
|
ADCInputMicSelect,
|
||||||
|
entity_category=ENTITY_CATEGORY_CONFIG,
|
||||||
|
icon=ICON_CHIP,
|
||||||
|
),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def to_code(config):
|
||||||
|
parent = await cg.get_variable(config[CONF_ES8388_ID])
|
||||||
|
if dac_output_config := config.get(CONF_DAC_OUTPUT):
|
||||||
|
s = await select.new_select(
|
||||||
|
dac_output_config,
|
||||||
|
options=["LINE1", "LINE2", "BOTH"],
|
||||||
|
)
|
||||||
|
await cg.register_parented(s, parent)
|
||||||
|
cg.add(parent.set_dac_output_select(s))
|
||||||
|
|
||||||
|
if adc_input_mic_config := config.get(CONF_ADC_INPUT_MIC):
|
||||||
|
s = await select.new_select(
|
||||||
|
adc_input_mic_config,
|
||||||
|
options=["LINE1", "LINE2", "DIFFERENCE"],
|
||||||
|
)
|
||||||
|
await cg.register_parented(s, parent)
|
||||||
|
cg.add(parent.set_adc_input_mic_select(s))
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user