mirror of https://github.com/lumapu/ahoy.git
committed by
GitHub
94 changed files with 6009 additions and 2938 deletions
@ -0,0 +1 @@ |
|||||
|
patches/GxEPD2_HAL.patch eol=lf |
@ -0,0 +1,56 @@ |
|||||
|
# Generate factory firmware (ESP32) |
||||
|
|
||||
|
If the firmware should already contain predefined settings this guide will help you to compile these into a single binary file. |
||||
|
|
||||
|
## Generate default settings |
||||
|
|
||||
|
First install on the requested platform the standard firmware and configure everything to your needs. Once you did all changes store them and export them to a `json` file. |
||||
|
|
||||
|
## Further prepare default settings |
||||
|
|
||||
|
First create a directory `data` inside the following project path: `src/`. |
||||
|
|
||||
|
As the export removes all your passwords you need to add them again to the `json` file. Open the `json` file with a text editor and search for all the `"pwd":""` sections. Between the second bunch of quotation marks you have to place the password. |
||||
|
|
||||
|
*Note: It's recommended to keep all information in one line to save space on the ESP littlefs partition* |
||||
|
|
||||
|
Next rename your export file to `settings.json` and move it to the new created directory. It should be look similar to this: |
||||
|
|
||||
|
``` |
||||
|
ahoy |
||||
|
|-- src |
||||
|
|-- data |
||||
|
|-- settings.json |
||||
|
|-- config |
||||
|
|-- network |
||||
|
... |
||||
|
``` |
||||
|
|
||||
|
## build firmware |
||||
|
|
||||
|
Choose your prefered environment and build firmware as usual. Once the process is finished you should find along with the standard `firmware.bin` an additional file called `firmware.factory.bin`. Both files are located here: `src/.pio/build/[ENVIRONMENT]/` |
||||
|
|
||||
|
## Upload to device |
||||
|
|
||||
|
Navigate to the firmware output directory `src/.pio/build/[ENVIRONMENT]/` and open a terminal or vice versa. |
||||
|
|
||||
|
Python: |
||||
|
`esptool.py -b 921600 write_flash --flash_mode dio --flash_size detect 0x0 firmware.factory.bin` |
||||
|
|
||||
|
Windows: |
||||
|
`esptool.exe -b 921600 write_flash --flash_mode dio --flash_size detect 0x0 firmware.factory.bin` |
||||
|
|
||||
|
The upload should be finished within one minute. |
||||
|
|
||||
|
## Testing |
||||
|
|
||||
|
Reboot your ESP an check if all your settings are present. |
||||
|
|
||||
|
## Get updated with 'Mainline' |
||||
|
|
||||
|
From time to time a new version of AhoyDTU will be published. To get the changes into your already prepared factory binary generation environment you have to do only a few steps: |
||||
|
|
||||
|
1. pull new changes from remote: `git pull` |
||||
|
2. check if the `data` folder is still there and contains the `settings.json` |
||||
|
3. build and upload |
||||
|
4. enjoy |
@ -1,26 +1,26 @@ |
|||||
diff --git a/src/AsyncWebSocket.cpp b/src/AsyncWebSocket.cpp
|
diff --git a/src/AsyncWebSocket.cpp b/src/AsyncWebSocket.cpp
|
||||
index 12be5f8..cffeed7 100644
|
index 6e88da9..09359c3 100644
|
||||
--- a/src/AsyncWebSocket.cpp
|
--- a/src/AsyncWebSocket.cpp
|
||||
+++ b/src/AsyncWebSocket.cpp
|
+++ b/src/AsyncWebSocket.cpp
|
||||
@@ -737,7 +737,7 @@ void AsyncWebSocketClient::binary(const __FlashStringHelper *data, size_t len)
|
@@ -827,7 +827,7 @@ void AsyncWebSocketClient::binary(AsyncWebSocketMessageBuffer * buffer)
|
||||
IPAddress AsyncWebSocketClient::remoteIP() const |
|
||||
{ |
|
||||
if (!_client) |
|
||||
- return IPAddress(0U);
|
|
||||
+ return IPAddress();
|
|
||||
|
|
||||
|
IPAddress AsyncWebSocketClient::remoteIP() { |
||||
|
if(!_client) { |
||||
|
- return IPAddress((uint32_t)0);
|
||||
|
+ return IPAddress();
|
||||
|
} |
||||
return _client->remoteIP(); |
return _client->remoteIP(); |
||||
} |
} |
||||
diff --git a/src/WebResponses.cpp b/src/WebResponses.cpp
|
diff --git a/src/WebResponses.cpp b/src/WebResponses.cpp
|
||||
index 22a549f..e0b36b3 100644
|
index a22e991..babef18 100644
|
||||
--- a/src/WebResponses.cpp
|
--- a/src/WebResponses.cpp
|
||||
+++ b/src/WebResponses.cpp
|
+++ b/src/WebResponses.cpp
|
||||
@@ -318,7 +318,7 @@ size_t AsyncAbstractResponse::_ack(AsyncWebServerRequest *request, size_t len, u
|
@@ -317,7 +317,7 @@ size_t AsyncAbstractResponse::_ack(AsyncWebServerRequest *request, size_t len, u
|
||||
free(buf); |
free(buf); |
||||
return 0; |
return 0; |
||||
} |
} |
||||
- outLen = sprintf_P((char*)buf+headLen, PSTR("%x"), readLen) + headLen;
|
- outLen = sprintf((char*)buf+headLen, "%x", readLen) + headLen;
|
||||
+ outLen = sprintf_P((char*)buf+headLen, PSTR("%04x"), readLen) + headLen;
|
+ outLen = sprintf((char*)buf+headLen, "%04x", readLen) + headLen;
|
||||
while(outLen < headLen + 4) buf[outLen++] = ' '; |
while(outLen < headLen + 4) buf[outLen++] = ' '; |
||||
buf[outLen++] = '\r'; |
buf[outLen++] = '\r'; |
||||
buf[outLen++] = '\n'; |
buf[outLen++] = '\n'; |
||||
|
@ -0,0 +1,392 @@ |
|||||
|
diff --git a/src/GxEPD2_EPD.cpp b/src/GxEPD2_EPD.cpp
|
||||
|
index 8df8bef..e9dfb19 100644
|
||||
|
--- a/src/GxEPD2_EPD.cpp
|
||||
|
+++ b/src/GxEPD2_EPD.cpp
|
||||
|
@@ -17,11 +17,10 @@
|
||||
|
#include <avr/pgmspace.h> |
||||
|
#endif |
||||
|
|
||||
|
-GxEPD2_EPD::GxEPD2_EPD(int16_t cs, int16_t dc, int16_t rst, int16_t busy, int16_t busy_level, uint32_t busy_timeout,
|
||||
|
+GxEPD2_EPD::GxEPD2_EPD(GxEPD2_HalInterface *hal, int16_t busy_level, uint32_t busy_timeout,
|
||||
|
uint16_t w, uint16_t h, GxEPD2::Panel p, bool c, bool pu, bool fpu) : |
||||
|
WIDTH(w), HEIGHT(h), panel(p), hasColor(c), hasPartialUpdate(pu), hasFastPartialUpdate(fpu), |
||||
|
- _cs(cs), _dc(dc), _rst(rst), _busy(busy), _busy_level(busy_level), _busy_timeout(busy_timeout), _diag_enabled(false),
|
||||
|
- _pSPIx(&SPI), _spi_settings(4000000, MSBFIRST, SPI_MODE0)
|
||||
|
+ _hal(hal), _busy_level(busy_level), _busy_timeout(busy_timeout), _diag_enabled(false)
|
||||
|
{ |
||||
|
_initial_write = true; |
||||
|
_initial_refresh = true; |
||||
|
@@ -54,44 +53,10 @@ void GxEPD2_EPD::init(uint32_t serial_diag_bitrate, bool initial, uint16_t reset
|
||||
|
Serial.begin(serial_diag_bitrate); |
||||
|
_diag_enabled = true; |
||||
|
} |
||||
|
- if (_cs >= 0)
|
||||
|
- {
|
||||
|
- digitalWrite(_cs, HIGH); // preset (less glitch for any analyzer)
|
||||
|
- pinMode(_cs, OUTPUT);
|
||||
|
- digitalWrite(_cs, HIGH); // set (needed e.g. for RP2040)
|
||||
|
- }
|
||||
|
- if (_dc >= 0)
|
||||
|
- {
|
||||
|
- digitalWrite(_dc, HIGH); // preset (less glitch for any analyzer)
|
||||
|
- pinMode(_dc, OUTPUT);
|
||||
|
- digitalWrite(_dc, HIGH); // set (needed e.g. for RP2040)
|
||||
|
- }
|
||||
|
- _reset();
|
||||
|
- if (_busy >= 0)
|
||||
|
- {
|
||||
|
- pinMode(_busy, INPUT);
|
||||
|
- }
|
||||
|
- _pSPIx->begin();
|
||||
|
- if (_busy == MISO) // may be overridden
|
||||
|
- {
|
||||
|
- pinMode(_busy, INPUT);
|
||||
|
- }
|
||||
|
- if (_dc == MISO) // may be overridden, TTGO T5 V2.66
|
||||
|
- {
|
||||
|
- pinMode(_dc, OUTPUT);
|
||||
|
- }
|
||||
|
- if (_cs == MISO) // may be overridden
|
||||
|
- {
|
||||
|
- pinMode(_cs, INPUT);
|
||||
|
- }
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::end() |
||||
|
{ |
||||
|
- _pSPIx->end();
|
||||
|
- if (_cs >= 0) pinMode(_cs, INPUT);
|
||||
|
- if (_dc >= 0) pinMode(_dc, INPUT);
|
||||
|
- if (_rst >= 0) pinMode(_rst, INPUT);
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::setBusyCallback(void (*busyCallback)(const void*), const void* busy_callback_parameter) |
||||
|
@@ -100,34 +65,27 @@ void GxEPD2_EPD::setBusyCallback(void (*busyCallback)(const void*), const void*
|
||||
|
_busy_callback_parameter = busy_callback_parameter; |
||||
|
} |
||||
|
|
||||
|
-void GxEPD2_EPD::selectSPI(SPIClass& spi, SPISettings spi_settings)
|
||||
|
-{
|
||||
|
- _pSPIx = &spi;
|
||||
|
- _spi_settings = spi_settings;
|
||||
|
-}
|
||||
|
-
|
||||
|
void GxEPD2_EPD::_reset() |
||||
|
{ |
||||
|
- if (_rst >= 0)
|
||||
|
{ |
||||
|
if (_pulldown_rst_mode) |
||||
|
{ |
||||
|
- digitalWrite(_rst, LOW);
|
||||
|
- pinMode(_rst, OUTPUT);
|
||||
|
- digitalWrite(_rst, LOW);
|
||||
|
+ _hal->rst(LOW);
|
||||
|
+ _hal->rstMode(OUTPUT);
|
||||
|
+ _hal->rst(LOW);
|
||||
|
delay(_reset_duration); |
||||
|
- pinMode(_rst, INPUT_PULLUP);
|
||||
|
+ _hal->rstMode(INPUT_PULLUP);
|
||||
|
delay(_reset_duration > 10 ? _reset_duration : 10); |
||||
|
} |
||||
|
else |
||||
|
{ |
||||
|
- digitalWrite(_rst, HIGH); // NEEDED for Waveshare "clever" reset circuit, power controller before reset pulse, preset (less glitch for any analyzer)
|
||||
|
- pinMode(_rst, OUTPUT);
|
||||
|
- digitalWrite(_rst, HIGH); // NEEDED for Waveshare "clever" reset circuit, power controller before reset pulse, set (needed e.g. for RP2040)
|
||||
|
+ _hal->rst(HIGH); // NEEDED for Waveshare "clever" reset circuit, power controller before reset pulse, preset (less glitch for any analyzer)
|
||||
|
+ _hal->rstMode(OUTPUT);
|
||||
|
+ _hal->rst(HIGH); // NEEDED for Waveshare "clever" reset circuit, power controller before reset pulse, set (needed e.g. for RP2040)
|
||||
|
delay(10); // NEEDED for Waveshare "clever" reset circuit, at least delay(2); |
||||
|
- digitalWrite(_rst, LOW);
|
||||
|
+ _hal->rst(LOW);
|
||||
|
delay(_reset_duration); |
||||
|
- digitalWrite(_rst, HIGH);
|
||||
|
+ _hal->rst(HIGH);
|
||||
|
delay(_reset_duration > 10 ? _reset_duration : 10); |
||||
|
} |
||||
|
_hibernating = false; |
||||
|
@@ -136,16 +94,15 @@ void GxEPD2_EPD::_reset()
|
||||
|
|
||||
|
void GxEPD2_EPD::_waitWhileBusy(const char* comment, uint16_t busy_time) |
||||
|
{ |
||||
|
- if (_busy >= 0)
|
||||
|
{ |
||||
|
delay(1); // add some margin to become active |
||||
|
unsigned long start = micros(); |
||||
|
while (1) |
||||
|
{ |
||||
|
- if (digitalRead(_busy) != _busy_level) break;
|
||||
|
+ if (_hal->getBusy() != _busy_level) break;
|
||||
|
if (_busy_callback) _busy_callback(_busy_callback_parameter); |
||||
|
else delay(1); |
||||
|
- if (digitalRead(_busy) != _busy_level) break;
|
||||
|
+ if (_hal->getBusy() != _busy_level) break;
|
||||
|
if (micros() - start > _busy_timeout) |
||||
|
{ |
||||
|
Serial.println("Busy Timeout!"); |
||||
|
@@ -169,120 +126,59 @@ void GxEPD2_EPD::_waitWhileBusy(const char* comment, uint16_t busy_time)
|
||||
|
} |
||||
|
(void) start; |
||||
|
} |
||||
|
- else delay(busy_time);
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::_writeCommand(uint8_t c) |
||||
|
{ |
||||
|
- _pSPIx->beginTransaction(_spi_settings);
|
||||
|
- if (_dc >= 0) digitalWrite(_dc, LOW);
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, LOW);
|
||||
|
- _pSPIx->transfer(c);
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, HIGH);
|
||||
|
- if (_dc >= 0) digitalWrite(_dc, HIGH);
|
||||
|
- _pSPIx->endTransaction();
|
||||
|
+ _hal->writeCmd(c);
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::_writeData(uint8_t d) |
||||
|
{ |
||||
|
- _pSPIx->beginTransaction(_spi_settings);
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, LOW);
|
||||
|
- _pSPIx->transfer(d);
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, HIGH);
|
||||
|
- _pSPIx->endTransaction();
|
||||
|
+ _hal->write(d);
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::_writeData(const uint8_t* data, uint16_t n) |
||||
|
{ |
||||
|
- _pSPIx->beginTransaction(_spi_settings);
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, LOW);
|
||||
|
- for (uint16_t i = 0; i < n; i++)
|
||||
|
- {
|
||||
|
- _pSPIx->transfer(*data++);
|
||||
|
- }
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, HIGH);
|
||||
|
- _pSPIx->endTransaction();
|
||||
|
+ _hal->write(data, n);
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::_writeDataPGM(const uint8_t* data, uint16_t n, int16_t fill_with_zeroes) |
||||
|
{ |
||||
|
- _pSPIx->beginTransaction(_spi_settings);
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, LOW);
|
||||
|
- for (uint16_t i = 0; i < n; i++)
|
||||
|
- {
|
||||
|
- _pSPIx->transfer(pgm_read_byte(&*data++));
|
||||
|
- }
|
||||
|
- while (fill_with_zeroes > 0)
|
||||
|
- {
|
||||
|
- _pSPIx->transfer(0x00);
|
||||
|
- fill_with_zeroes--;
|
||||
|
- }
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, HIGH);
|
||||
|
- _pSPIx->endTransaction();
|
||||
|
+ _hal->write(data, n, fill_with_zeroes);
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::_writeDataPGM_sCS(const uint8_t* data, uint16_t n, int16_t fill_with_zeroes) |
||||
|
{ |
||||
|
- _pSPIx->beginTransaction(_spi_settings);
|
||||
|
- for (uint8_t i = 0; i < n; i++)
|
||||
|
- {
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, LOW);
|
||||
|
- _pSPIx->transfer(pgm_read_byte(&*data++));
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, HIGH);
|
||||
|
- }
|
||||
|
- while (fill_with_zeroes > 0)
|
||||
|
- {
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, LOW);
|
||||
|
- _pSPIx->transfer(0x00);
|
||||
|
- fill_with_zeroes--;
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, HIGH);
|
||||
|
+ _hal->write(data, n);
|
||||
|
+ if (fill_with_zeroes > 0) {
|
||||
|
+ uint8_t buf[fill_with_zeroes];
|
||||
|
+ memset(buf, 0, fill_with_zeroes);
|
||||
|
+ _hal->write(buf, fill_with_zeroes);
|
||||
|
} |
||||
|
- _pSPIx->endTransaction();
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::_writeCommandData(const uint8_t* pCommandData, uint8_t datalen) |
||||
|
{ |
||||
|
- _pSPIx->beginTransaction(_spi_settings);
|
||||
|
- if (_dc >= 0) digitalWrite(_dc, LOW);
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, LOW);
|
||||
|
- _pSPIx->transfer(*pCommandData++);
|
||||
|
- if (_dc >= 0) digitalWrite(_dc, HIGH);
|
||||
|
- for (uint8_t i = 0; i < datalen - 1; i++) // sub the command
|
||||
|
- {
|
||||
|
- _pSPIx->transfer(*pCommandData++);
|
||||
|
- }
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, HIGH);
|
||||
|
- _pSPIx->endTransaction();
|
||||
|
+ _hal->writeCmd(pCommandData, datalen, false);
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::_writeCommandDataPGM(const uint8_t* pCommandData, uint8_t datalen) |
||||
|
{ |
||||
|
- _pSPIx->beginTransaction(_spi_settings);
|
||||
|
- if (_dc >= 0) digitalWrite(_dc, LOW);
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, LOW);
|
||||
|
- _pSPIx->transfer(pgm_read_byte(&*pCommandData++));
|
||||
|
- if (_dc >= 0) digitalWrite(_dc, HIGH);
|
||||
|
- for (uint8_t i = 0; i < datalen - 1; i++) // sub the command
|
||||
|
- {
|
||||
|
- _pSPIx->transfer(pgm_read_byte(&*pCommandData++));
|
||||
|
- }
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, HIGH);
|
||||
|
- _pSPIx->endTransaction();
|
||||
|
+ _hal->writeCmd(pCommandData, datalen, true);
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::_startTransfer() |
||||
|
{ |
||||
|
- _pSPIx->beginTransaction(_spi_settings);
|
||||
|
- if (_cs >= 0) digitalWrite(_cs, LOW);
|
||||
|
+ _hal->startTransfer();
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::_transfer(uint8_t value) |
||||
|
{ |
||||
|
- _pSPIx->transfer(value);
|
||||
|
+ _hal->transfer(value);
|
||||
|
} |
||||
|
|
||||
|
void GxEPD2_EPD::_endTransfer() |
||||
|
{ |
||||
|
- if (_cs >= 0) digitalWrite(_cs, HIGH);
|
||||
|
- _pSPIx->endTransaction();
|
||||
|
+ _hal->endTransfer();
|
||||
|
} |
||||
|
diff --git a/src/GxEPD2_EPD.h b/src/GxEPD2_EPD.h
|
||||
|
index 34c1145..1e8ea64 100644
|
||||
|
--- a/src/GxEPD2_EPD.h
|
||||
|
+++ b/src/GxEPD2_EPD.h
|
||||
|
@@ -13,9 +13,9 @@
|
||||
|
#define _GxEPD2_EPD_H_ |
||||
|
|
||||
|
#include <Arduino.h> |
||||
|
-#include <SPI.h>
|
||||
|
|
||||
|
#include <GxEPD2.h> |
||||
|
+#include <GxEPD2_Hal.h>
|
||||
|
|
||||
|
#pragma GCC diagnostic ignored "-Wunused-parameter" |
||||
|
//#pragma GCC diagnostic ignored "-Wsign-compare" |
||||
|
@@ -31,7 +31,7 @@ class GxEPD2_EPD
|
||||
|
const bool hasPartialUpdate; |
||||
|
const bool hasFastPartialUpdate; |
||||
|
// constructor |
||||
|
- GxEPD2_EPD(int16_t cs, int16_t dc, int16_t rst, int16_t busy, int16_t busy_level, uint32_t busy_timeout,
|
||||
|
+ GxEPD2_EPD(GxEPD2_HalInterface *hal, int16_t busy_level, uint32_t busy_timeout,
|
||||
|
uint16_t w, uint16_t h, GxEPD2::Panel p, bool c, bool pu, bool fpu); |
||||
|
virtual void init(uint32_t serial_diag_bitrate = 0); // serial_diag_bitrate = 0 : disabled |
||||
|
virtual void init(uint32_t serial_diag_bitrate, bool initial, uint16_t reset_duration = 10, bool pulldown_rst_mode = false); |
||||
|
@@ -97,7 +97,6 @@ class GxEPD2_EPD
|
||||
|
{ |
||||
|
return (a > b ? a : b); |
||||
|
}; |
||||
|
- void selectSPI(SPIClass& spi, SPISettings spi_settings);
|
||||
|
protected: |
||||
|
void _reset(); |
||||
|
void _waitWhileBusy(const char* comment = 0, uint16_t busy_time = 5000); |
||||
|
@@ -112,16 +111,15 @@ class GxEPD2_EPD
|
||||
|
void _transfer(uint8_t value); |
||||
|
void _endTransfer(); |
||||
|
protected: |
||||
|
- int16_t _cs, _dc, _rst, _busy, _busy_level;
|
||||
|
+ GxEPD2_HalInterface *_hal;
|
||||
|
+ int16_t _busy_level;
|
||||
|
uint32_t _busy_timeout; |
||||
|
bool _diag_enabled, _pulldown_rst_mode; |
||||
|
- SPIClass* _pSPIx;
|
||||
|
- SPISettings _spi_settings;
|
||||
|
bool _initial_write, _initial_refresh; |
||||
|
bool _power_is_on, _using_partial_mode, _hibernating; |
||||
|
bool _init_display_done; |
||||
|
uint16_t _reset_duration; |
||||
|
- void (*_busy_callback)(const void*);
|
||||
|
+ void (*_busy_callback)(const void*);
|
||||
|
const void* _busy_callback_parameter; |
||||
|
}; |
||||
|
|
||||
|
diff --git a/src/GxEPD2_Hal.h b/src/GxEPD2_Hal.h
|
||||
|
new file mode 100644 |
||||
|
index 0000000..13424b6
|
||||
|
--- /dev/null
|
||||
|
+++ b/src/GxEPD2_Hal.h
|
||||
|
@@ -0,0 +1,19 @@
|
||||
|
+#pragma once
|
||||
|
+
|
||||
|
+class GxEPD2_HalInterface {
|
||||
|
+ public:
|
||||
|
+ virtual void rstMode(uint8_t mode) = 0;
|
||||
|
+ virtual void rst(bool level) = 0;
|
||||
|
+ virtual int getBusy(void) = 0;
|
||||
|
+ virtual bool isRst(void) = 0;
|
||||
|
+
|
||||
|
+ virtual void write(uint8_t buf) = 0;
|
||||
|
+ virtual void write(const uint8_t *buf, uint16_t n) = 0;
|
||||
|
+ virtual void write(const uint8_t *buf, uint16_t n, int16_t fill_with_zeroes) = 0;
|
||||
|
+ virtual void writeCmd(const uint8_t val) = 0;
|
||||
|
+ virtual void writeCmd(const uint8_t* pCommandData, uint8_t datalen, bool isPGM) = 0;
|
||||
|
+
|
||||
|
+ virtual void startTransfer(void) = 0;
|
||||
|
+ virtual void endTransfer(void) = 0;
|
||||
|
+ virtual void transfer(const uint8_t val) = 0;
|
||||
|
+};
|
||||
|
diff --git a/src/epd/GxEPD2_150_BN.cpp b/src/epd/GxEPD2_150_BN.cpp
|
||||
|
index bfb3ddf..dba3d78 100644
|
||||
|
--- a/src/epd/GxEPD2_150_BN.cpp
|
||||
|
+++ b/src/epd/GxEPD2_150_BN.cpp
|
||||
|
@@ -14,8 +14,8 @@
|
||||
|
|
||||
|
#include "GxEPD2_150_BN.h" |
||||
|
|
||||
|
-GxEPD2_150_BN::GxEPD2_150_BN(int16_t cs, int16_t dc, int16_t rst, int16_t busy) :
|
||||
|
- GxEPD2_EPD(cs, dc, rst, busy, HIGH, 10000000, WIDTH, HEIGHT, panel, hasColor, hasPartialUpdate, hasFastPartialUpdate)
|
||||
|
+GxEPD2_150_BN::GxEPD2_150_BN(GxEPD2_HalInterface *hal) :
|
||||
|
+ GxEPD2_EPD(hal, HIGH, 10000000, WIDTH, HEIGHT, panel, hasColor, hasPartialUpdate, hasFastPartialUpdate)
|
||||
|
{ |
||||
|
} |
||||
|
|
||||
|
@@ -269,7 +269,7 @@ void GxEPD2_150_BN::refresh(int16_t x, int16_t y, int16_t w, int16_t h)
|
||||
|
int16_t y1 = y < 0 ? 0 : y; // limit |
||||
|
w1 = x1 + w1 < int16_t(WIDTH) ? w1 : int16_t(WIDTH) - x1; // limit |
||||
|
h1 = y1 + h1 < int16_t(HEIGHT) ? h1 : int16_t(HEIGHT) - y1; // limit |
||||
|
- if ((w1 <= 0) || (h1 <= 0)) return;
|
||||
|
+ if ((w1 <= 0) || (h1 <= 0)) return;
|
||||
|
// make x1, w1 multiple of 8 |
||||
|
w1 += x1 % 8; |
||||
|
if (w1 % 8 > 0) w1 += 8 - w1 % 8; |
||||
|
@@ -287,7 +287,7 @@ void GxEPD2_150_BN::powerOff()
|
||||
|
void GxEPD2_150_BN::hibernate() |
||||
|
{ |
||||
|
_PowerOff(); |
||||
|
- if (_rst >= 0)
|
||||
|
+ if (_hal->isRst())
|
||||
|
{ |
||||
|
_writeCommand(0x10); // deep sleep mode |
||||
|
_writeData(0x1); // enter deep sleep |
||||
|
diff --git a/src/epd/GxEPD2_150_BN.h b/src/epd/GxEPD2_150_BN.h
|
||||
|
index bc46a45..954b9c4 100644
|
||||
|
--- a/src/epd/GxEPD2_150_BN.h
|
||||
|
+++ b/src/epd/GxEPD2_150_BN.h
|
||||
|
@@ -16,6 +16,7 @@
|
||||
|
#define _GxEPD2_150_BN_H_ |
||||
|
|
||||
|
#include "../GxEPD2_EPD.h" |
||||
|
+#include "../GxEPD2_Hal.h"
|
||||
|
|
||||
|
class GxEPD2_150_BN : public GxEPD2_EPD |
||||
|
{ |
||||
|
@@ -33,7 +34,7 @@ class GxEPD2_150_BN : public GxEPD2_EPD
|
||||
|
static const uint16_t full_refresh_time = 4000; // ms, e.g. 3825000us |
||||
|
static const uint16_t partial_refresh_time = 800; // ms, e.g. 736000us |
||||
|
// constructor |
||||
|
- GxEPD2_150_BN(int16_t cs, int16_t dc, int16_t rst, int16_t busy);
|
||||
|
+ GxEPD2_150_BN(GxEPD2_HalInterface *hal);
|
||||
|
// methods (virtual) |
||||
|
// Support for Bitmaps (Sprites) to Controller Buffer and to Screen |
||||
|
void clearScreen(uint8_t value = 0xFF); // init controller memory and screen (default white) |
@ -1,362 +0,0 @@ |
|||||
diff --git a/src/GxEPD2_EPD.cpp b/src/GxEPD2_EPD.cpp
|
|
||||
index 8df8bef..91d7f49 100644
|
|
||||
--- a/src/GxEPD2_EPD.cpp
|
|
||||
+++ b/src/GxEPD2_EPD.cpp
|
|
||||
@@ -19,9 +19,9 @@
|
|
||||
|
|
||||
GxEPD2_EPD::GxEPD2_EPD(int16_t cs, int16_t dc, int16_t rst, int16_t busy, int16_t busy_level, uint32_t busy_timeout, |
|
||||
uint16_t w, uint16_t h, GxEPD2::Panel p, bool c, bool pu, bool fpu) : |
|
||||
- WIDTH(w), HEIGHT(h), panel(p), hasColor(c), hasPartialUpdate(pu), hasFastPartialUpdate(fpu),
|
|
||||
+ WIDTH(w), HEIGHT(h), panel(p), hasColor(c), hasPartialUpdate(pu), hasFastPartialUpdate(fpu), _sck(-1), _mosi(-1),
|
|
||||
_cs(cs), _dc(dc), _rst(rst), _busy(busy), _busy_level(busy_level), _busy_timeout(busy_timeout), _diag_enabled(false), |
|
||||
- _pSPIx(&SPI), _spi_settings(4000000, MSBFIRST, SPI_MODE0)
|
|
||||
+ _spi_settings(4000000, MSBFIRST, SPI_MODE0)
|
|
||||
{ |
|
||||
_initial_write = true; |
|
||||
_initial_refresh = true; |
|
||||
@@ -71,27 +71,30 @@ void GxEPD2_EPD::init(uint32_t serial_diag_bitrate, bool initial, uint16_t reset
|
|
||||
{ |
|
||||
pinMode(_busy, INPUT); |
|
||||
} |
|
||||
- _pSPIx->begin();
|
|
||||
- if (_busy == MISO) // may be overridden
|
|
||||
- {
|
|
||||
- pinMode(_busy, INPUT);
|
|
||||
- }
|
|
||||
- if (_dc == MISO) // may be overridden, TTGO T5 V2.66
|
|
||||
- {
|
|
||||
- pinMode(_dc, OUTPUT);
|
|
||||
- }
|
|
||||
- if (_cs == MISO) // may be overridden
|
|
||||
+ if (_sck < 0) SPI.begin();
|
|
||||
+}
|
|
||||
+
|
|
||||
+void GxEPD2_EPD::init(int16_t sck, int16_t mosi, uint32_t serial_diag_bitrate, bool initial, uint16_t reset_duration, bool pulldown_rst_mode)
|
|
||||
+{
|
|
||||
+ if ((sck >= 0) && (mosi >= 0))
|
|
||||
{ |
|
||||
- pinMode(_cs, INPUT);
|
|
||||
- }
|
|
||||
+ _sck = sck;
|
|
||||
+ _mosi = mosi;
|
|
||||
+ digitalWrite(_sck, LOW);
|
|
||||
+ digitalWrite(_mosi, LOW);
|
|
||||
+ pinMode(_sck, OUTPUT);
|
|
||||
+ pinMode(_mosi, OUTPUT);
|
|
||||
+ } else _sck = -1;
|
|
||||
+ init(serial_diag_bitrate, initial, reset_duration, pulldown_rst_mode);
|
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::end() |
|
||||
{ |
|
||||
- _pSPIx->end();
|
|
||||
if (_cs >= 0) pinMode(_cs, INPUT); |
|
||||
if (_dc >= 0) pinMode(_dc, INPUT); |
|
||||
if (_rst >= 0) pinMode(_rst, INPUT); |
|
||||
+ if (_sck >= 0) pinMode(_sck, INPUT);
|
|
||||
+ if (_mosi >= 0) pinMode(_mosi, INPUT);
|
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::setBusyCallback(void (*busyCallback)(const void*), const void* busy_callback_parameter) |
|
||||
@@ -100,12 +103,6 @@ void GxEPD2_EPD::setBusyCallback(void (*busyCallback)(const void*), const void*
|
|
||||
_busy_callback_parameter = busy_callback_parameter; |
|
||||
} |
|
||||
|
|
||||
-void GxEPD2_EPD::selectSPI(SPIClass& spi, SPISettings spi_settings)
|
|
||||
-{
|
|
||||
- _pSPIx = &spi;
|
|
||||
- _spi_settings = spi_settings;
|
|
||||
-}
|
|
||||
-
|
|
||||
void GxEPD2_EPD::_reset() |
|
||||
{ |
|
||||
if (_rst >= 0) |
|
||||
@@ -174,115 +171,201 @@ void GxEPD2_EPD::_waitWhileBusy(const char* comment, uint16_t busy_time)
|
|
||||
|
|
||||
void GxEPD2_EPD::_writeCommand(uint8_t c) |
|
||||
{ |
|
||||
- _pSPIx->beginTransaction(_spi_settings);
|
|
||||
+ _beginTransaction(_spi_settings);
|
|
||||
if (_dc >= 0) digitalWrite(_dc, LOW); |
|
||||
if (_cs >= 0) digitalWrite(_cs, LOW); |
|
||||
- _pSPIx->transfer(c);
|
|
||||
+ _spi_write(c);
|
|
||||
if (_cs >= 0) digitalWrite(_cs, HIGH); |
|
||||
if (_dc >= 0) digitalWrite(_dc, HIGH); |
|
||||
- _pSPIx->endTransaction();
|
|
||||
+ _endTransaction();
|
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::_writeData(uint8_t d) |
|
||||
{ |
|
||||
- _pSPIx->beginTransaction(_spi_settings);
|
|
||||
+ _beginTransaction(_spi_settings);
|
|
||||
if (_cs >= 0) digitalWrite(_cs, LOW); |
|
||||
- _pSPIx->transfer(d);
|
|
||||
+ _spi_write(d);
|
|
||||
if (_cs >= 0) digitalWrite(_cs, HIGH); |
|
||||
- _pSPIx->endTransaction();
|
|
||||
+ _endTransaction();
|
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::_writeData(const uint8_t* data, uint16_t n) |
|
||||
{ |
|
||||
- _pSPIx->beginTransaction(_spi_settings);
|
|
||||
+ _beginTransaction(_spi_settings);
|
|
||||
if (_cs >= 0) digitalWrite(_cs, LOW); |
|
||||
- for (uint16_t i = 0; i < n; i++)
|
|
||||
+ for (uint8_t i = 0; i < n; i++)
|
|
||||
{ |
|
||||
- _pSPIx->transfer(*data++);
|
|
||||
+ _spi_write(*data++);
|
|
||||
} |
|
||||
if (_cs >= 0) digitalWrite(_cs, HIGH); |
|
||||
- _pSPIx->endTransaction();
|
|
||||
+ _endTransaction();
|
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::_writeDataPGM(const uint8_t* data, uint16_t n, int16_t fill_with_zeroes) |
|
||||
{ |
|
||||
- _pSPIx->beginTransaction(_spi_settings);
|
|
||||
+ _beginTransaction(_spi_settings);
|
|
||||
if (_cs >= 0) digitalWrite(_cs, LOW); |
|
||||
- for (uint16_t i = 0; i < n; i++)
|
|
||||
+ for (uint8_t i = 0; i < n; i++)
|
|
||||
{ |
|
||||
- _pSPIx->transfer(pgm_read_byte(&*data++));
|
|
||||
+ _spi_write(pgm_read_byte(&*data++));
|
|
||||
} |
|
||||
while (fill_with_zeroes > 0) |
|
||||
{ |
|
||||
- _pSPIx->transfer(0x00);
|
|
||||
+ _spi_write(0x00);
|
|
||||
fill_with_zeroes--; |
|
||||
} |
|
||||
if (_cs >= 0) digitalWrite(_cs, HIGH); |
|
||||
- _pSPIx->endTransaction();
|
|
||||
+ _endTransaction();
|
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::_writeDataPGM_sCS(const uint8_t* data, uint16_t n, int16_t fill_with_zeroes) |
|
||||
{ |
|
||||
- _pSPIx->beginTransaction(_spi_settings);
|
|
||||
+ _beginTransaction(_spi_settings);
|
|
||||
for (uint8_t i = 0; i < n; i++) |
|
||||
{ |
|
||||
if (_cs >= 0) digitalWrite(_cs, LOW); |
|
||||
- _pSPIx->transfer(pgm_read_byte(&*data++));
|
|
||||
+ _spi_write(pgm_read_byte(&*data++));
|
|
||||
if (_cs >= 0) digitalWrite(_cs, HIGH); |
|
||||
} |
|
||||
while (fill_with_zeroes > 0) |
|
||||
{ |
|
||||
if (_cs >= 0) digitalWrite(_cs, LOW); |
|
||||
- _pSPIx->transfer(0x00);
|
|
||||
+ _spi_write(0x00);
|
|
||||
fill_with_zeroes--; |
|
||||
if (_cs >= 0) digitalWrite(_cs, HIGH); |
|
||||
} |
|
||||
- _pSPIx->endTransaction();
|
|
||||
+ _endTransaction();
|
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::_writeCommandData(const uint8_t* pCommandData, uint8_t datalen) |
|
||||
{ |
|
||||
- _pSPIx->beginTransaction(_spi_settings);
|
|
||||
+ _beginTransaction(_spi_settings);
|
|
||||
if (_dc >= 0) digitalWrite(_dc, LOW); |
|
||||
if (_cs >= 0) digitalWrite(_cs, LOW); |
|
||||
- _pSPIx->transfer(*pCommandData++);
|
|
||||
+ _spi_write(*pCommandData++);
|
|
||||
if (_dc >= 0) digitalWrite(_dc, HIGH); |
|
||||
for (uint8_t i = 0; i < datalen - 1; i++) // sub the command |
|
||||
{ |
|
||||
- _pSPIx->transfer(*pCommandData++);
|
|
||||
+ _spi_write(*pCommandData++);
|
|
||||
} |
|
||||
if (_cs >= 0) digitalWrite(_cs, HIGH); |
|
||||
- _pSPIx->endTransaction();
|
|
||||
+ _endTransaction();
|
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::_writeCommandDataPGM(const uint8_t* pCommandData, uint8_t datalen) |
|
||||
{ |
|
||||
- _pSPIx->beginTransaction(_spi_settings);
|
|
||||
+ _beginTransaction(_spi_settings);
|
|
||||
if (_dc >= 0) digitalWrite(_dc, LOW); |
|
||||
if (_cs >= 0) digitalWrite(_cs, LOW); |
|
||||
- _pSPIx->transfer(pgm_read_byte(&*pCommandData++));
|
|
||||
+ _spi_write(pgm_read_byte(&*pCommandData++));
|
|
||||
if (_dc >= 0) digitalWrite(_dc, HIGH); |
|
||||
for (uint8_t i = 0; i < datalen - 1; i++) // sub the command |
|
||||
{ |
|
||||
- _pSPIx->transfer(pgm_read_byte(&*pCommandData++));
|
|
||||
+ _spi_write(pgm_read_byte(&*pCommandData++));
|
|
||||
} |
|
||||
if (_cs >= 0) digitalWrite(_cs, HIGH); |
|
||||
- _pSPIx->endTransaction();
|
|
||||
+ _endTransaction();
|
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::_startTransfer() |
|
||||
{ |
|
||||
- _pSPIx->beginTransaction(_spi_settings);
|
|
||||
+ _beginTransaction(_spi_settings);
|
|
||||
if (_cs >= 0) digitalWrite(_cs, LOW); |
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::_transfer(uint8_t value) |
|
||||
{ |
|
||||
- _pSPIx->transfer(value);
|
|
||||
+ _spi_write(value);
|
|
||||
} |
|
||||
|
|
||||
void GxEPD2_EPD::_endTransfer() |
|
||||
{ |
|
||||
if (_cs >= 0) digitalWrite(_cs, HIGH); |
|
||||
- _pSPIx->endTransaction();
|
|
||||
+ _endTransaction();
|
|
||||
+}
|
|
||||
+
|
|
||||
+void GxEPD2_EPD::_beginTransaction(const SPISettings& settings)
|
|
||||
+{
|
|
||||
+ if (_sck < 0) SPI.beginTransaction(settings);
|
|
||||
+}
|
|
||||
+
|
|
||||
+void GxEPD2_EPD::_spi_write(uint8_t data)
|
|
||||
+{
|
|
||||
+ if (_sck < 0) SPI.transfer(data);
|
|
||||
+ else
|
|
||||
+ {
|
|
||||
+#if defined (ESP8266)
|
|
||||
+ yield();
|
|
||||
+#endif
|
|
||||
+ for (int i = 0; i < 8; i++)
|
|
||||
+ {
|
|
||||
+ digitalWrite(_mosi, (data & 0x80) ? HIGH : LOW);
|
|
||||
+ data <<= 1;
|
|
||||
+ digitalWrite(_sck, HIGH);
|
|
||||
+ digitalWrite(_sck, LOW);
|
|
||||
+ }
|
|
||||
+ }
|
|
||||
+}
|
|
||||
+
|
|
||||
+void GxEPD2_EPD::_endTransaction()
|
|
||||
+{
|
|
||||
+ if (_sck < 0) SPI.endTransaction();
|
|
||||
+}
|
|
||||
+
|
|
||||
+uint8_t GxEPD2_EPD::_readData()
|
|
||||
+{
|
|
||||
+ uint8_t data = 0;
|
|
||||
+ _beginTransaction(_spi_settings);
|
|
||||
+ if (_cs >= 0) digitalWrite(_cs, LOW);
|
|
||||
+ if (_sck < 0)
|
|
||||
+ {
|
|
||||
+ data = SPI.transfer(0);
|
|
||||
+ }
|
|
||||
+ else
|
|
||||
+ {
|
|
||||
+ pinMode(_mosi, INPUT);
|
|
||||
+ for (int i = 0; i < 8; i++)
|
|
||||
+ {
|
|
||||
+ data <<= 1;
|
|
||||
+ digitalWrite(_sck, HIGH);
|
|
||||
+ data |= digitalRead(_mosi);
|
|
||||
+ digitalWrite(_sck, LOW);
|
|
||||
+ }
|
|
||||
+ pinMode(_mosi, OUTPUT);
|
|
||||
+ }
|
|
||||
+ if (_cs >= 0) digitalWrite(_cs, HIGH);
|
|
||||
+ _endTransaction();
|
|
||||
+ return data;
|
|
||||
+}
|
|
||||
+
|
|
||||
+void GxEPD2_EPD::_readData(uint8_t* data, uint16_t n)
|
|
||||
+{
|
|
||||
+ _beginTransaction(_spi_settings);
|
|
||||
+ if (_cs >= 0) digitalWrite(_cs, LOW);
|
|
||||
+ if (_sck < 0)
|
|
||||
+ {
|
|
||||
+ for (uint8_t i = 0; i < n; i++)
|
|
||||
+ {
|
|
||||
+ *data++ = SPI.transfer(0);
|
|
||||
+ }
|
|
||||
+ }
|
|
||||
+ else
|
|
||||
+ {
|
|
||||
+ pinMode(_mosi, INPUT);
|
|
||||
+ for (uint8_t i = 0; i < n; i++)
|
|
||||
+ {
|
|
||||
+ *data = 0;
|
|
||||
+ for (int i = 0; i < 8; i++)
|
|
||||
+ {
|
|
||||
+ *data <<= 1;
|
|
||||
+ digitalWrite(_sck, HIGH);
|
|
||||
+ *data |= digitalRead(_mosi);
|
|
||||
+ digitalWrite(_sck, LOW);
|
|
||||
+ }
|
|
||||
+ data++;
|
|
||||
+ }
|
|
||||
+ pinMode(_mosi, OUTPUT);
|
|
||||
+ }
|
|
||||
+ if (_cs >= 0) digitalWrite(_cs, HIGH);
|
|
||||
+ _endTransaction();
|
|
||||
} |
|
||||
diff --git a/src/GxEPD2_EPD.h b/src/GxEPD2_EPD.h
|
|
||||
index 34c1145..c480b7d 100644
|
|
||||
--- a/src/GxEPD2_EPD.h
|
|
||||
+++ b/src/GxEPD2_EPD.h
|
|
||||
@@ -8,6 +8,10 @@
|
|
||||
// Version: see library.properties |
|
||||
// |
|
||||
// Library: https://github.com/ZinggJM/GxEPD2 |
|
||||
+// To use SW SPI with GxEPD2:
|
|
||||
+// add the special call to the added init method BEFORE the normal init method:
|
|
||||
+// display.epd2.init(SW_SCK, SW_MOSI, 115200, true, 20, false); // define or replace SW_SCK, SW_MOSI
|
|
||||
+// display.init(115200); // needed to init upper level
|
|
||||
|
|
||||
#ifndef _GxEPD2_EPD_H_ |
|
||||
#define _GxEPD2_EPD_H_ |
|
||||
@@ -35,6 +39,7 @@ class GxEPD2_EPD
|
|
||||
uint16_t w, uint16_t h, GxEPD2::Panel p, bool c, bool pu, bool fpu); |
|
||||
virtual void init(uint32_t serial_diag_bitrate = 0); // serial_diag_bitrate = 0 : disabled |
|
||||
virtual void init(uint32_t serial_diag_bitrate, bool initial, uint16_t reset_duration = 10, bool pulldown_rst_mode = false); |
|
||||
+ virtual void init(int16_t sck, int16_t mosi, uint32_t serial_diag_bitrate, bool initial, uint16_t reset_duration = 20, bool pulldown_rst_mode = false);
|
|
||||
virtual void end(); // release SPI and control pins |
|
||||
// Support for Bitmaps (Sprites) to Controller Buffer and to Screen |
|
||||
virtual void clearScreen(uint8_t value) = 0; // init controller memory and screen (default white) |
|
||||
@@ -97,7 +102,6 @@ class GxEPD2_EPD
|
|
||||
{ |
|
||||
return (a > b ? a : b); |
|
||||
}; |
|
||||
- void selectSPI(SPIClass& spi, SPISettings spi_settings);
|
|
||||
protected: |
|
||||
void _reset(); |
|
||||
void _waitWhileBusy(const char* comment = 0, uint16_t busy_time = 5000); |
|
||||
@@ -111,17 +115,22 @@ class GxEPD2_EPD
|
|
||||
void _startTransfer(); |
|
||||
void _transfer(uint8_t value); |
|
||||
void _endTransfer(); |
|
||||
+ void _beginTransaction(const SPISettings& settings);
|
|
||||
+ void _spi_write(uint8_t data);
|
|
||||
+ void _endTransaction();
|
|
||||
+ public:
|
|
||||
+ uint8_t _readData();
|
|
||||
+ void _readData(uint8_t* data, uint16_t n);
|
|
||||
protected: |
|
||||
- int16_t _cs, _dc, _rst, _busy, _busy_level;
|
|
||||
+ int16_t _cs, _dc, _rst, _busy, _busy_level, _sck, _mosi;;
|
|
||||
uint32_t _busy_timeout; |
|
||||
bool _diag_enabled, _pulldown_rst_mode; |
|
||||
- SPIClass* _pSPIx;
|
|
||||
SPISettings _spi_settings; |
|
||||
bool _initial_write, _initial_refresh; |
|
||||
bool _power_is_on, _using_partial_mode, _hibernating; |
|
||||
bool _init_display_done; |
|
||||
uint16_t _reset_duration; |
|
||||
- void (*_busy_callback)(const void*);
|
|
||||
+ void (*_busy_callback)(const void*);
|
|
||||
const void* _busy_callback_parameter; |
|
||||
}; |
|
||||
|
|
@ -0,0 +1,79 @@ |
|||||
|
import os |
||||
|
import subprocess |
||||
|
import shutil |
||||
|
from SCons.Script import DefaultEnvironment |
||||
|
Import("env") |
||||
|
|
||||
|
|
||||
|
def build_littlefs(): |
||||
|
if os.path.isfile('data/settings.json') == False: |
||||
|
return # nothing to do |
||||
|
|
||||
|
result = subprocess.run(["pio", "run", "--target", "buildfs", "--environment", env['PIOENV']]) |
||||
|
if result.returncode != 0: |
||||
|
print("Error building LittleFS:") |
||||
|
exit(1) |
||||
|
else: |
||||
|
print("LittleFS build successful") |
||||
|
|
||||
|
def merge_bins(): |
||||
|
if os.path.isfile('data/settings.json') == False: |
||||
|
return # nothing to do |
||||
|
|
||||
|
BOOTLOADER_OFFSET = 0x0000 |
||||
|
PARTITIONS_OFFSET = 0x8000 |
||||
|
FIRMWARE_OFFSET = 0x10000 |
||||
|
|
||||
|
if env['PIOENV'][:13] == "esp32-wroom32": |
||||
|
BOOTLOADER_OFFSET = 0x1000 |
||||
|
|
||||
|
flash_size = int(env.BoardConfig().get("upload.maximum_size", "1310720")) # 0x140000 |
||||
|
app0_offset = 0x10000 |
||||
|
if env['PIOENV'][:7] == "esp8266": |
||||
|
app0_offset = 0 |
||||
|
elif env['PIOENV'][:7] == "esp8285": |
||||
|
app0_offset = 0 |
||||
|
|
||||
|
littlefs_offset = 0x290000 |
||||
|
if flash_size == 0x330000: |
||||
|
littlefs_offset = 0x670000 |
||||
|
elif flash_size == 0x640000: |
||||
|
littlefs_offset = 0xc90000 |
||||
|
|
||||
|
# save current wd |
||||
|
start = os.getcwd() |
||||
|
os.chdir('.pio/build/' + env['PIOENV'] + '/') |
||||
|
|
||||
|
with open("bootloader.bin", "rb") as bootloader_file: |
||||
|
bootloader_data = bootloader_file.read() |
||||
|
|
||||
|
with open("partitions.bin", "rb") as partitions_file: |
||||
|
partitions_data = partitions_file.read() |
||||
|
|
||||
|
with open("firmware.bin", "rb") as firmware_file: |
||||
|
firmware_data = firmware_file.read() |
||||
|
|
||||
|
with open("littlefs.bin", "rb") as littlefs_file: |
||||
|
littlefs_data = littlefs_file.read() |
||||
|
|
||||
|
with open("firmware.factory.bin", "wb") as merged_file: |
||||
|
merged_file.write(b'\xFF' * BOOTLOADER_OFFSET) |
||||
|
merged_file.write(bootloader_data) |
||||
|
|
||||
|
merged_file.write(b'\xFF' * (PARTITIONS_OFFSET - (BOOTLOADER_OFFSET + len(bootloader_data)))) |
||||
|
merged_file.write(partitions_data) |
||||
|
|
||||
|
merged_file.write(b'\xFF' * (FIRMWARE_OFFSET - (PARTITIONS_OFFSET + len(partitions_data)))) |
||||
|
merged_file.write(firmware_data) |
||||
|
|
||||
|
merged_file.write(b'\xFF' * (littlefs_offset - (FIRMWARE_OFFSET + len(firmware_data)))) |
||||
|
merged_file.write(littlefs_data) |
||||
|
|
||||
|
os.chdir(start) |
||||
|
|
||||
|
def main(target, source, env): |
||||
|
build_littlefs() |
||||
|
merge_bins() |
||||
|
|
||||
|
# ensure that script is called once firmeware was compiled |
||||
|
env.AddPostAction("$BUILD_DIR/${PROGNAME}.bin", main) |
@ -0,0 +1,40 @@ |
|||||
|
import re |
||||
|
import os |
||||
|
import queue |
||||
|
|
||||
|
def error(msg): |
||||
|
print("ERROR: " + msg) |
||||
|
exit() |
||||
|
|
||||
|
def check(inp, lst, pattern): |
||||
|
q = queue.LifoQueue() |
||||
|
out = [] |
||||
|
keep = True |
||||
|
for line in inp: |
||||
|
x = re.findall(pattern, line) |
||||
|
if len(x) > 0: |
||||
|
if line.find("ENDIF_") != -1: |
||||
|
if not q.empty(): |
||||
|
e = q.get() |
||||
|
if e[0] == x[0]: |
||||
|
keep = e[1] |
||||
|
elif line.find("IF_") != -1: |
||||
|
q.put((x[0], keep)) |
||||
|
if keep is True: |
||||
|
keep = x[0] in lst |
||||
|
elif line.find("E") != -1: |
||||
|
if q.empty(): |
||||
|
error("(ELSE) missing open statement!") |
||||
|
e = q.get() |
||||
|
q.put(e) |
||||
|
if e[1] is True: |
||||
|
keep = not keep |
||||
|
else: |
||||
|
if keep is True: |
||||
|
out.append(line) |
||||
|
return out |
||||
|
|
||||
|
def conv(inp, lst): |
||||
|
#print(lst) |
||||
|
out = check(inp, lst, r'\/\*(?:IF_|ELS|ENDIF_)([A-Z0-9\-_]+)?\*\/') |
||||
|
return check(out, lst, r'\<\!\-\-(?:IF_|ELS|ENDIF_)([A-Z0-9\-_]+)?\-\-\>') |
@ -1,41 +1,54 @@ |
|||||
Changelog v0.8.83 |
Changelog v0.8.140 |
||||
|
|
||||
* added German translations for all variants |
* added HMS-400-1T support (serial number 1125...) |
||||
* added reading grid profile |
* added further ESP8266 versions (-all, -minimal) because of small ressources on ESP8266 |
||||
* added decimal place for active power control (APC aka power limit) |
* added some Gridprofiles |
||||
* added information about working IRQ for NRF24 and CMT2300A to `/system` |
* added support for characters in serial number of inverter (A-F) |
||||
* added loss rate to `/visualization` in the statistics window and MqTT |
* added default coordinates on fresh install, needed for history graph on display and WebUI |
||||
* added optional output to display whether it's night time or not. Can be reused as output to control battery system or mapped to a LED |
* added option to reset values on communication start (sunrise) |
||||
* added timestamp for `max ac power` as tooltip |
* added max inverter temperature to WebUI |
||||
* added wizard for initial WiFi connection |
* added yield day to history graph |
||||
* added history graph (still under development) |
* added script and [instructions](../manual/factory_firmware.md) how to generate factory firmware which includes predefined settings |
||||
* added simulator (must be activated before compile, standard: off) |
* added button for downloading coredump (ESP32 variants only) to `/system`. Once a crash happens the reason can be checked afterwards (even after a reboot) |
||||
* added minimal version (without: MqTT, Display, History), WebUI is not changed! (not compiled automatically) |
* added support of HERF inverters, serial number is converted in Javascript |
||||
* added info about installed binary to `/update` |
* added device name to HTML title |
||||
* added protection to prevent update to wrong firmware (environment check) |
* added feature to restart Ahoy using MqTT |
||||
* added optional custom link to the menu |
* added feature to publish MqTT messages as JSON as well (new setting) |
||||
* added support for other regions (USA, Indonesia) |
* add timestamp to JSON output |
||||
* added warning for WiFi channel 12-14 (ESP8266 only) |
* improved communication to inverter |
||||
* added `max_power` to MqTT total values |
* improved translation to German |
||||
* added API-Token authentification for external scripts |
* improved HTML pages, reduced in size by only including relevant contents depending by chip type |
||||
* improved MqTT by marking sent data and improved `last_success` resends |
* improved history graph in WebUI |
||||
* improved communication for HM and MI inverters |
* improved network routines |
||||
* improved reading live data from inverter |
* improved Wizard |
||||
* improved sending active power control command faster |
* improved WebUI by disabling upload and import buttons when no file is selected |
||||
* improved `/settings`: pinout has an own subgroup |
* improved queue, only add new object once they not exist in queue |
||||
* improved export by saving settings before they are exported (to have everything in JSON) |
* improved MqTT `OnMessage` (threadsafe) |
||||
* improved code quality (cppcheck) |
* improved read of alarms, prevent duplicates, update alarm time if there is an update |
||||
* seperated sunrise and sunset offset to two fields |
* improved alarms are now sorted in ascending direction |
||||
* fix MqTT night communication |
* improved by prevent add inverter multiple times |
||||
* fix missing favicon to html header |
* improved sending active power controll commands |
||||
* fix build on Windows of `opendtufusion` environments (git: trailing whitespaces) |
* improved refresh routine of ePaper, full refresh each 12h |
||||
* fix generation of DTU-ID |
* redesigned WebUI on `/system` |
||||
* fix: protect commands from popup in `/live` if password is set |
* changed MqTT retained flags |
||||
* fix: prevent sending commands to inverter which isn't active |
* change MqTT return value of power limit acknowledge from `boolean` to `float`. The value returned is the same as it was set to confirm reception (not the read back value) |
||||
* combined firmware and hardware version to JSON topics (MqTT) |
* converted ePaper and Ethernet to hal-SPI |
||||
* updated Prometheus with latest changes |
* combined Ethernet and WiFi variants - Ethernet is now always included, but needs to be enabled if needed |
||||
* upgraded most libraries to newer versions |
* changed: Ethernet variants (W5500) now support WiFi as fall back / configuration |
||||
* beautified typography, added spaces between value and unit for `/visualization` |
* switch AsyncWebserver library |
||||
* removed add to total (MqTT) inverter setting |
* fixed autodiscovery for homeassistant |
||||
|
* fix reset values functionality |
||||
|
* fix read back of active power control value, now it has one decimal place |
||||
|
* fix NTP issues |
||||
|
* fixed MqTT discovery field `ALARM_MES_ID` |
||||
|
* fix close button color of modal windows in dark mode |
||||
|
* fixed calculation of max AC power |
||||
|
* fixed reset values at midnight if WiFi isn't available |
||||
|
* fixed HMT-1800-4T number of inputs |
||||
|
* fix crash if invalid serial number was set -> inverter will be disabled automatically |
||||
|
* fixed ESP8266, ESP32 static IP |
||||
|
* fixed ethernet MAC address read back |
||||
|
* update several libraries to more recent versions |
||||
|
* removed `yield efficiency` because the inverter already calculates correct |
||||
|
|
||||
full version log: [Development Log](https://github.com/lumapu/ahoy/blob/development03/src/CHANGES.md) |
full version log: [Development Log](https://github.com/lumapu/ahoy/blob/development03/src/CHANGES.md) |
||||
|
@ -1,261 +0,0 @@ |
|||||
//-----------------------------------------------------------------------------
|
|
||||
// 2023 Ahoy, https://www.mikrocontroller.net/topic/525778
|
|
||||
// Creative Commons - http://creativecommons.org/licenses/by-nc-sa/3.0/de/
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
|
|
||||
#if defined(ETHERNET) |
|
||||
|
|
||||
#if defined(ESP32) && defined(F) |
|
||||
#undef F |
|
||||
#define F(sl) (sl) |
|
||||
#endif |
|
||||
#include "ahoyeth.h" |
|
||||
#include <ESPmDNS.h> |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
ahoyeth::ahoyeth() |
|
||||
{ |
|
||||
// WiFi.onEvent(ESP32_W5500_event);
|
|
||||
} |
|
||||
|
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoyeth::setup(settings_t *config, uint32_t *utcTimestamp, OnNetworkCB onNetworkCB, OnTimeCB onTimeCB) { |
|
||||
mConfig = config; |
|
||||
mUtcTimestamp = utcTimestamp; |
|
||||
mOnNetworkCB = onNetworkCB; |
|
||||
mOnTimeCB = onTimeCB; |
|
||||
|
|
||||
Serial.flush(); |
|
||||
WiFi.onEvent([this](WiFiEvent_t event, arduino_event_info_t info) -> void { this->onEthernetEvent(event, info); }); |
|
||||
|
|
||||
Serial.flush(); |
|
||||
#if defined(CONFIG_IDF_TARGET_ESP32S3) |
|
||||
mEthSpi.begin(DEF_ETH_MISO_PIN, DEF_ETH_MOSI_PIN, DEF_ETH_SCK_PIN, DEF_ETH_CS_PIN, DEF_ETH_IRQ_PIN, DEF_ETH_RST_PIN); |
|
||||
#else |
|
||||
ETH.begin(DEF_ETH_MISO_PIN, DEF_ETH_MOSI_PIN, DEF_ETH_SCK_PIN, DEF_ETH_CS_PIN, DEF_ETH_IRQ_PIN, ETH_SPI_CLOCK_MHZ, ETH_SPI_HOST); |
|
||||
#endif |
|
||||
|
|
||||
if(mConfig->sys.ip.ip[0] != 0) { |
|
||||
IPAddress ip(mConfig->sys.ip.ip); |
|
||||
IPAddress mask(mConfig->sys.ip.mask); |
|
||||
IPAddress dns1(mConfig->sys.ip.dns1); |
|
||||
IPAddress dns2(mConfig->sys.ip.dns2); |
|
||||
IPAddress gateway(mConfig->sys.ip.gateway); |
|
||||
if(!ETH.config(ip, gateway, mask, dns1, dns2)) |
|
||||
DPRINTLN(DBG_ERROR, F("failed to set static IP!")); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
bool ahoyeth::updateNtpTime(void) { |
|
||||
DPRINTLN(DBG_DEBUG, F(__FUNCTION__)); Serial.flush(); |
|
||||
Serial.printf("ETH.linkUp()=%s\n", ETH.linkUp() ? "up" : "down"); |
|
||||
Serial.print("ETH.localIP()="); |
|
||||
Serial.println(ETH.localIP()); |
|
||||
Serial.printf("Go on? %s\n", (!ETH.localIP()) ? "No..." : "Yes..."); |
|
||||
if (!ETH.localIP()) |
|
||||
return false; |
|
||||
|
|
||||
DPRINTLN(DBG_DEBUG, F("updateNtpTime: checking udp \"connection\"...")); Serial.flush(); |
|
||||
if (!mUdp.connected()) { |
|
||||
DPRINTLN(DBG_DEBUG, F("updateNtpTime: About to (re)connect...")); Serial.flush(); |
|
||||
IPAddress timeServer; |
|
||||
if (!WiFi.hostByName(mConfig->ntp.addr, timeServer)) |
|
||||
return false; |
|
||||
|
|
||||
if (!mUdp.connect(timeServer, mConfig->ntp.port)) |
|
||||
return false; |
|
||||
|
|
||||
DPRINTLN(DBG_DEBUG, F("updateNtpTime: Connected...")); Serial.flush(); |
|
||||
mUdp.onPacket([this](AsyncUDPPacket packet) { |
|
||||
DPRINTLN(DBG_DEBUG, F("updateNtpTime: about to handle ntp packet...")); Serial.flush(); |
|
||||
this->handleNTPPacket(packet); |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
DPRINTLN(DBG_DEBUG, F("updateNtpTime: prepare packet...")); Serial.flush(); |
|
||||
|
|
||||
// set all bytes in the buffer to 0
|
|
||||
memset(mUdpPacketBuffer, 0, NTP_PACKET_SIZE); |
|
||||
// Initialize values needed to form NTP request
|
|
||||
// (see URL above for details on the packets)
|
|
||||
|
|
||||
mUdpPacketBuffer[0] = 0b11100011; // LI, Version, Mode
|
|
||||
mUdpPacketBuffer[1] = 0; // Stratum, or type of clock
|
|
||||
mUdpPacketBuffer[2] = 6; // Polling Interval
|
|
||||
mUdpPacketBuffer[3] = 0xEC; // Peer Clock Precision
|
|
||||
|
|
||||
// 8 bytes of zero for Root Delay & Root Dispersion
|
|
||||
mUdpPacketBuffer[12] = 49; |
|
||||
mUdpPacketBuffer[13] = 0x4E; |
|
||||
mUdpPacketBuffer[14] = 49; |
|
||||
mUdpPacketBuffer[15] = 52; |
|
||||
|
|
||||
//Send unicast
|
|
||||
DPRINTLN(DBG_DEBUG, F("updateNtpTime: send packet...")); Serial.flush(); |
|
||||
mUdp.write(mUdpPacketBuffer, sizeof(mUdpPacketBuffer)); |
|
||||
|
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoyeth::handleNTPPacket(AsyncUDPPacket packet) { |
|
||||
char buf[80]; |
|
||||
|
|
||||
memcpy(buf, packet.data(), sizeof(buf)); |
|
||||
|
|
||||
unsigned long highWord = word(buf[40], buf[41]); |
|
||||
unsigned long lowWord = word(buf[42], buf[43]); |
|
||||
|
|
||||
// combine the four bytes (two words) into a long integer
|
|
||||
// this is NTP time (seconds since Jan 1 1900):
|
|
||||
unsigned long secsSince1900 = highWord << 16 | lowWord; |
|
||||
|
|
||||
*mUtcTimestamp = secsSince1900 - 2208988800UL; // UTC time
|
|
||||
DPRINTLN(DBG_INFO, "[NTP]: " + ah::getDateTimeStr(*mUtcTimestamp) + " UTC"); |
|
||||
mOnTimeCB(true); |
|
||||
} |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoyeth::welcome(String ip, String mode) { |
|
||||
DBGPRINTLN(F("\n\n--------------------------------")); |
|
||||
DBGPRINTLN(F("Welcome to AHOY!")); |
|
||||
DBGPRINT(F("\npoint your browser to http://")); |
|
||||
DBGPRINT(ip); |
|
||||
DBGPRINTLN(mode); |
|
||||
DBGPRINTLN(F("to configure your device")); |
|
||||
DBGPRINTLN(F("--------------------------------\n")); |
|
||||
} |
|
||||
|
|
||||
void ahoyeth::onEthernetEvent(WiFiEvent_t event, arduino_event_info_t info) { |
|
||||
AWS_LOG(F("[ETH]: Got event...")); |
|
||||
switch (event) { |
|
||||
#if ( ( defined(ESP_ARDUINO_VERSION_MAJOR) && (ESP_ARDUINO_VERSION_MAJOR >= 2) ) && ( ARDUINO_ESP32_GIT_VER != 0x46d5afb1 ) ) |
|
||||
// For breaking core v2.0.0
|
|
||||
// Why so strange to define a breaking enum arduino_event_id_t in WiFiGeneric.h
|
|
||||
// compared to the old system_event_id_t, now in tools/sdk/esp32/include/esp_event/include/esp_event_legacy.h
|
|
||||
// You can preserve the old enum order and just adding new items to do no harm
|
|
||||
case ARDUINO_EVENT_ETH_START: |
|
||||
AWS_LOG(F("\nETH Started")); |
|
||||
//set eth hostname here
|
|
||||
if(String(mConfig->sys.deviceName) != "") |
|
||||
ETH.setHostname(mConfig->sys.deviceName); |
|
||||
else |
|
||||
ETH.setHostname("ESP32_W5500"); |
|
||||
break; |
|
||||
|
|
||||
case ARDUINO_EVENT_ETH_CONNECTED: |
|
||||
AWS_LOG(F("ETH Connected")); |
|
||||
break; |
|
||||
|
|
||||
case ARDUINO_EVENT_ETH_GOT_IP: |
|
||||
if (!ESP32_W5500_eth_connected) { |
|
||||
#if defined (CONFIG_IDF_TARGET_ESP32S3) |
|
||||
AWS_LOG3(F("ETH MAC: "), mEthSpi.macAddress(), F(", IPv4: "), ETH.localIP()); |
|
||||
#else |
|
||||
AWS_LOG3(F("ETH MAC: "), ETH.macAddress(), F(", IPv4: "), ETH.localIP()); |
|
||||
#endif |
|
||||
|
|
||||
if (ETH.fullDuplex()) { |
|
||||
AWS_LOG0(F("FULL_DUPLEX, ")); |
|
||||
} else { |
|
||||
AWS_LOG0(F("HALF_DUPLEX, ")); |
|
||||
} |
|
||||
|
|
||||
AWS_LOG1(ETH.linkSpeed(), F("Mbps")); |
|
||||
|
|
||||
ESP32_W5500_eth_connected = true; |
|
||||
mOnNetworkCB(true); |
|
||||
} |
|
||||
if (!MDNS.begin(mConfig->sys.deviceName)) { |
|
||||
DPRINTLN(DBG_ERROR, F("Error setting up MDNS responder!")); |
|
||||
} else { |
|
||||
DBGPRINT(F("[WiFi] mDNS established: ")); |
|
||||
DBGPRINT(mConfig->sys.deviceName); |
|
||||
DBGPRINTLN(F(".local")); |
|
||||
} |
|
||||
break; |
|
||||
|
|
||||
case ARDUINO_EVENT_ETH_DISCONNECTED: |
|
||||
AWS_LOG("ETH Disconnected"); |
|
||||
ESP32_W5500_eth_connected = false; |
|
||||
mUdp.close(); |
|
||||
mOnNetworkCB(false); |
|
||||
break; |
|
||||
|
|
||||
case ARDUINO_EVENT_ETH_STOP: |
|
||||
AWS_LOG("\nETH Stopped"); |
|
||||
ESP32_W5500_eth_connected = false; |
|
||||
mUdp.close(); |
|
||||
mOnNetworkCB(false); |
|
||||
break; |
|
||||
|
|
||||
#else |
|
||||
|
|
||||
// For old core v1.0.6-
|
|
||||
// Core v2.0.0 defines a stupid enum arduino_event_id_t, breaking any code for ESP32_W5500 written for previous core
|
|
||||
// Why so strange to define a breaking enum arduino_event_id_t in WiFiGeneric.h
|
|
||||
// compared to the old system_event_id_t, now in tools/sdk/esp32/include/esp_event/include/esp_event_legacy.h
|
|
||||
// You can preserve the old enum order and just adding new items to do no harm
|
|
||||
case SYSTEM_EVENT_ETH_START: |
|
||||
AWS_LOG(F("\nETH Started")); |
|
||||
//set eth hostname here
|
|
||||
if(String(mConfig->sys.deviceName) != "") |
|
||||
ETH.setHostname(mConfig->sys.deviceName); |
|
||||
else |
|
||||
ETH.setHostname("ESP32_W5500"); |
|
||||
break; |
|
||||
|
|
||||
case SYSTEM_EVENT_ETH_CONNECTED: |
|
||||
AWS_LOG(F("ETH Connected")); |
|
||||
break; |
|
||||
|
|
||||
case SYSTEM_EVENT_ETH_GOT_IP: |
|
||||
if (!ESP32_W5500_eth_connected) { |
|
||||
AWS_LOG3(F("ETH MAC: "), ETH.macAddress(), F(", IPv4: "), ETH.localIP()); |
|
||||
|
|
||||
if (ETH.fullDuplex()) { |
|
||||
AWS_LOG0(F("FULL_DUPLEX, ")); |
|
||||
} else { |
|
||||
AWS_LOG0(F("HALF_DUPLEX, ")); |
|
||||
} |
|
||||
|
|
||||
AWS_LOG1(ETH.linkSpeed(), F("Mbps")); |
|
||||
|
|
||||
ESP32_W5500_eth_connected = true; |
|
||||
mOnNetworkCB(true); |
|
||||
} |
|
||||
if (!MDNS.begin(mConfig->sys.deviceName)) { |
|
||||
DPRINTLN(DBG_ERROR, F("Error setting up MDNS responder!")); |
|
||||
} else { |
|
||||
DBGPRINT(F("[WiFi] mDNS established: ")); |
|
||||
DBGPRINT(mConfig->sys.deviceName); |
|
||||
DBGPRINTLN(F(".local")); |
|
||||
} |
|
||||
break; |
|
||||
|
|
||||
case SYSTEM_EVENT_ETH_DISCONNECTED: |
|
||||
AWS_LOG("ETH Disconnected"); |
|
||||
ESP32_W5500_eth_connected = false; |
|
||||
mUdp.close(); |
|
||||
mOnNetworkCB(false); |
|
||||
break; |
|
||||
|
|
||||
case SYSTEM_EVENT_ETH_STOP: |
|
||||
AWS_LOG("\nETH Stopped"); |
|
||||
ESP32_W5500_eth_connected = false; |
|
||||
mUdp.close(); |
|
||||
mOnNetworkCB(false); |
|
||||
break; |
|
||||
#endif |
|
||||
|
|
||||
default: |
|
||||
|
|
||||
break; |
|
||||
} |
|
||||
|
|
||||
} |
|
||||
|
|
||||
#endif /* defined(ETHERNET) */ |
|
@ -1,64 +0,0 @@ |
|||||
//-----------------------------------------------------------------------------
|
|
||||
// 2024 Ahoy, https://github.com/lumpapu/ahoy
|
|
||||
// Creative Commons - http://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
|
|
||||
#if defined(ETHERNET) |
|
||||
#ifndef __AHOYETH_H__ |
|
||||
#define __AHOYETH_H__ |
|
||||
|
|
||||
#include <functional> |
|
||||
|
|
||||
#include <Arduino.h> |
|
||||
#include <AsyncUDP.h> |
|
||||
#include <DNSServer.h> |
|
||||
|
|
||||
#include "ethSpi.h" |
|
||||
#include "../utils/dbg.h" |
|
||||
#include "../config/config.h" |
|
||||
#include "../config/settings.h" |
|
||||
|
|
||||
#include "AsyncWebServer_ESP32_W5500.h" |
|
||||
|
|
||||
|
|
||||
class app; |
|
||||
|
|
||||
#define NTP_PACKET_SIZE 48 |
|
||||
|
|
||||
class ahoyeth { |
|
||||
public: /* types */ |
|
||||
typedef std::function<void(bool)> OnNetworkCB; |
|
||||
typedef std::function<void(bool)> OnTimeCB; |
|
||||
|
|
||||
public: |
|
||||
ahoyeth(); |
|
||||
|
|
||||
void setup(settings_t *config, uint32_t *utcTimestamp, OnNetworkCB onNetworkCB, OnTimeCB onTimeCB); |
|
||||
bool updateNtpTime(void); |
|
||||
|
|
||||
private: |
|
||||
void setupEthernet(); |
|
||||
|
|
||||
void handleNTPPacket(AsyncUDPPacket packet); |
|
||||
|
|
||||
void welcome(String ip, String mode); |
|
||||
|
|
||||
void onEthernetEvent(WiFiEvent_t event, arduino_event_info_t info); |
|
||||
|
|
||||
private: |
|
||||
#if defined(CONFIG_IDF_TARGET_ESP32S3) |
|
||||
EthSpi mEthSpi; |
|
||||
#endif |
|
||||
settings_t *mConfig = nullptr; |
|
||||
|
|
||||
uint32_t *mUtcTimestamp; |
|
||||
AsyncUDP mUdp; // for time server
|
|
||||
byte mUdpPacketBuffer[NTP_PACKET_SIZE]; // buffer to hold incoming and outgoing packets
|
|
||||
|
|
||||
OnNetworkCB mOnNetworkCB; |
|
||||
OnTimeCB mOnTimeCB; |
|
||||
|
|
||||
}; |
|
||||
|
|
||||
#endif /*__AHOYETH_H__*/ |
|
||||
#endif /* defined(ETHERNET) */ |
|
@ -0,0 +1,151 @@ |
|||||
|
//-----------------------------------------------------------------------------
|
||||
|
// 2024 Ahoy, https://ahoydtu.de
|
||||
|
// Creative Commons - https://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
||||
|
//-----------------------------------------------------------------------------
|
||||
|
|
||||
|
#ifndef __AHOY_ETHERNET_H__ |
||||
|
#define __AHOY_ETHERNET_H__ |
||||
|
|
||||
|
#if defined(ETHERNET) |
||||
|
#include <functional> |
||||
|
#include <AsyncUDP.h> |
||||
|
#include <ETH.h> |
||||
|
#include "AhoyEthernetSpi.h" |
||||
|
#include "AhoyNetwork.h" |
||||
|
#include "AhoyWifiEsp32.h" |
||||
|
|
||||
|
class AhoyEthernet : public AhoyWifi { |
||||
|
private: |
||||
|
enum class Mode { |
||||
|
WIRED, |
||||
|
WIRELESS |
||||
|
}; |
||||
|
|
||||
|
public: |
||||
|
AhoyEthernet() |
||||
|
: mMode (Mode::WIRELESS) {} |
||||
|
|
||||
|
virtual void begin() override { |
||||
|
mMode = Mode::WIRELESS; |
||||
|
mAp.enable(); |
||||
|
AhoyWifi::begin(); |
||||
|
|
||||
|
if(!mConfig->sys.eth.enabled) |
||||
|
return; |
||||
|
|
||||
|
mEthSpi.begin(mConfig->sys.eth.pinMiso, mConfig->sys.eth.pinMosi, mConfig->sys.eth.pinSclk, mConfig->sys.eth.pinCs, mConfig->sys.eth.pinIrq, mConfig->sys.eth.pinRst); |
||||
|
ETH.setHostname(mConfig->sys.deviceName); |
||||
|
} |
||||
|
|
||||
|
virtual String getIp(void) override { |
||||
|
if(Mode::WIRELESS == mMode) |
||||
|
return AhoyWifi::getIp(); |
||||
|
else |
||||
|
return ETH.localIP().toString(); |
||||
|
} |
||||
|
|
||||
|
virtual String getMac(void) override { |
||||
|
if(Mode::WIRELESS == mMode) |
||||
|
return AhoyWifi::getMac(); |
||||
|
else |
||||
|
return mEthSpi.macAddress(); |
||||
|
} |
||||
|
|
||||
|
virtual bool isWiredConnection() override { |
||||
|
return (Mode::WIRED == mMode); |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
virtual void OnEvent(WiFiEvent_t event) override { |
||||
|
switch(event) { |
||||
|
case ARDUINO_EVENT_ETH_CONNECTED: |
||||
|
mMode = Mode::WIRED; // needed for static IP
|
||||
|
[[fallthrough]]; |
||||
|
case SYSTEM_EVENT_STA_CONNECTED: |
||||
|
mWifiConnecting = false; |
||||
|
if(NetworkState::CONNECTED != mStatus) { |
||||
|
if(ARDUINO_EVENT_ETH_CONNECTED == event) |
||||
|
WiFi.disconnect(); |
||||
|
|
||||
|
mStatus = NetworkState::CONNECTED; |
||||
|
DPRINTLN(DBG_INFO, F("Network connected")); |
||||
|
setStaticIp(); |
||||
|
} |
||||
|
break; |
||||
|
|
||||
|
case SYSTEM_EVENT_STA_GOT_IP: |
||||
|
mStatus = NetworkState::GOT_IP; |
||||
|
if(mAp.isEnabled()) |
||||
|
mAp.disable(); |
||||
|
|
||||
|
mMode = Mode::WIRELESS; |
||||
|
if(!mConnected) { |
||||
|
mConnected = true; |
||||
|
ah::welcome(WiFi.localIP().toString(), F("Station WiFi")); |
||||
|
MDNS.begin(mConfig->sys.deviceName); |
||||
|
mOnNetworkCB(true); |
||||
|
} |
||||
|
break; |
||||
|
|
||||
|
case ARDUINO_EVENT_ETH_GOT_IP: |
||||
|
mStatus = NetworkState::GOT_IP; |
||||
|
mMode = Mode::WIRED; |
||||
|
if(!mConnected) { |
||||
|
mAp.disable(); |
||||
|
mConnected = true; |
||||
|
ah::welcome(ETH.localIP().toString(), F("Station Ethernet")); |
||||
|
MDNS.begin(mConfig->sys.deviceName); |
||||
|
mOnNetworkCB(true); |
||||
|
WiFi.disconnect(); |
||||
|
} |
||||
|
break; |
||||
|
|
||||
|
case ARDUINO_EVENT_ETH_STOP: |
||||
|
[[fallthrough]]; |
||||
|
case ARDUINO_EVENT_ETH_DISCONNECTED: |
||||
|
mStatus = NetworkState::DISCONNECTED; |
||||
|
if(mConnected) { |
||||
|
mMode = Mode::WIRELESS; |
||||
|
mConnected = false; |
||||
|
mOnNetworkCB(false); |
||||
|
MDNS.end(); |
||||
|
AhoyWifi::begin(); |
||||
|
} |
||||
|
break; |
||||
|
|
||||
|
case ARDUINO_EVENT_WIFI_STA_LOST_IP: |
||||
|
[[fallthrough]]; |
||||
|
case ARDUINO_EVENT_WIFI_STA_STOP: |
||||
|
[[fallthrough]]; |
||||
|
case SYSTEM_EVENT_STA_DISCONNECTED: |
||||
|
mStatus = NetworkState::DISCONNECTED; |
||||
|
if(mConnected && (Mode::WIRELESS == mMode)) { |
||||
|
mConnected = false; |
||||
|
mOnNetworkCB(false); |
||||
|
MDNS.end(); |
||||
|
AhoyWifi::begin(); |
||||
|
} |
||||
|
break; |
||||
|
|
||||
|
default: |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
void setStaticIp() override { |
||||
|
setupIp([this](IPAddress ip, IPAddress gateway, IPAddress mask, IPAddress dns1, IPAddress dns2) -> bool { |
||||
|
if(Mode::WIRELESS == mMode) |
||||
|
return WiFi.config(ip, gateway, mask, dns1, dns2); |
||||
|
else |
||||
|
return ETH.config(ip, gateway, mask, dns1, dns2); |
||||
|
}); |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
AhoyEthernetSpi mEthSpi; |
||||
|
Mode mMode; |
||||
|
|
||||
|
}; |
||||
|
|
||||
|
#endif /*ETHERNET*/ |
||||
|
#endif /*__AHOY_ETHERNET_H__*/ |
@ -0,0 +1,254 @@ |
|||||
|
//-----------------------------------------------------------------------------
|
||||
|
// 2024 Ahoy, https://ahoydtu.de
|
||||
|
// Creative Commons - https://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
||||
|
//-----------------------------------------------------------------------------
|
||||
|
|
||||
|
#ifndef __AHOY_NETWORK_H__ |
||||
|
#define __AHOY_NETWORK_H__ |
||||
|
|
||||
|
#include "AhoyNetworkHelper.h" |
||||
|
#include "../config/settings.h" |
||||
|
#include "../utils/helper.h" |
||||
|
#include "AhoyWifiAp.h" |
||||
|
#include "AsyncJson.h" |
||||
|
|
||||
|
#define NTP_PACKET_SIZE 48 |
||||
|
|
||||
|
class AhoyNetwork { |
||||
|
public: |
||||
|
typedef std::function<void(bool)> OnNetworkCB; |
||||
|
typedef std::function<void(bool)> OnTimeCB; |
||||
|
|
||||
|
public: |
||||
|
void setup(settings_t *config, uint32_t *utcTimestamp, OnNetworkCB onNetworkCB, OnTimeCB onTimeCB) { |
||||
|
mConfig = config; |
||||
|
mUtcTimestamp = utcTimestamp; |
||||
|
mOnNetworkCB = onNetworkCB; |
||||
|
mOnTimeCB = onTimeCB; |
||||
|
|
||||
|
if('\0' == mConfig->sys.deviceName[0]) |
||||
|
snprintf(mConfig->sys.deviceName, DEVNAME_LEN, "%s", DEF_DEVICE_NAME); |
||||
|
|
||||
|
mAp.setup(&mConfig->sys); |
||||
|
|
||||
|
#if defined(ESP32) |
||||
|
WiFi.onEvent([this](WiFiEvent_t event, arduino_event_info_t info) -> void { |
||||
|
OnEvent(event); |
||||
|
}); |
||||
|
#else |
||||
|
wifiConnectHandler = WiFi.onStationModeConnected( |
||||
|
[this](const WiFiEventStationModeConnected& event) -> void { |
||||
|
OnEvent((WiFiEvent_t)SYSTEM_EVENT_STA_CONNECTED); |
||||
|
}); |
||||
|
wifiGotIPHandler = WiFi.onStationModeGotIP( |
||||
|
[this](const WiFiEventStationModeGotIP& event) -> void { |
||||
|
OnEvent((WiFiEvent_t)SYSTEM_EVENT_STA_GOT_IP); |
||||
|
}); |
||||
|
wifiDisconnectHandler = WiFi.onStationModeDisconnected( |
||||
|
[this](const WiFiEventStationModeDisconnected& event) -> void { |
||||
|
OnEvent((WiFiEvent_t)SYSTEM_EVENT_STA_DISCONNECTED); |
||||
|
}); |
||||
|
#endif |
||||
|
} |
||||
|
|
||||
|
bool isConnected() const { |
||||
|
return (mStatus == NetworkState::CONNECTED); |
||||
|
} |
||||
|
|
||||
|
bool updateNtpTime(void) { |
||||
|
if(NetworkState::GOT_IP != mStatus) |
||||
|
return false; |
||||
|
|
||||
|
if (!mUdp.connected()) { |
||||
|
IPAddress timeServer; |
||||
|
if (!WiFi.hostByName(mConfig->ntp.addr, timeServer)) |
||||
|
return false; |
||||
|
if (!mUdp.connect(timeServer, mConfig->ntp.port)) |
||||
|
return false; |
||||
|
} |
||||
|
|
||||
|
mUdp.onPacket([this](AsyncUDPPacket packet) { |
||||
|
this->handleNTPPacket(packet); |
||||
|
}); |
||||
|
sendNTPpacket(); |
||||
|
|
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
public: |
||||
|
virtual void begin() = 0; |
||||
|
virtual void tickNetworkLoop() = 0; |
||||
|
virtual String getIp(void) = 0; |
||||
|
virtual String getMac(void) = 0; |
||||
|
|
||||
|
virtual bool getWasInCh12to14() { |
||||
|
return false; |
||||
|
} |
||||
|
|
||||
|
virtual bool isWiredConnection() { |
||||
|
return false; |
||||
|
} |
||||
|
|
||||
|
bool isApActive() { |
||||
|
return mAp.isEnabled(); |
||||
|
} |
||||
|
|
||||
|
bool getAvailNetworks(JsonObject obj, IApp *app) { |
||||
|
if(!mScanActive) { |
||||
|
app->addOnce([this]() {scan();}, 1, "scan"); |
||||
|
return false; |
||||
|
} |
||||
|
|
||||
|
int n = WiFi.scanComplete(); |
||||
|
if (WIFI_SCAN_RUNNING == n) |
||||
|
return false; |
||||
|
|
||||
|
if(n > 0) { |
||||
|
JsonArray nets = obj.createNestedArray(F("networks")); |
||||
|
int sort[n]; |
||||
|
sortRSSI(&sort[0], n); |
||||
|
for (int i = 0; i < n; ++i) { |
||||
|
nets[i][F("ssid")] = WiFi.SSID(sort[i]); |
||||
|
nets[i][F("rssi")] = WiFi.RSSI(sort[i]); |
||||
|
} |
||||
|
} |
||||
|
mScanActive = false; |
||||
|
WiFi.scanDelete(); |
||||
|
|
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
void scan(void) { |
||||
|
mScanActive = true; |
||||
|
if(mWifiConnecting) { |
||||
|
mWifiConnecting = false; |
||||
|
WiFi.disconnect(); |
||||
|
} |
||||
|
WiFi.scanNetworks(true, true); |
||||
|
} |
||||
|
|
||||
|
protected: |
||||
|
virtual void setStaticIp() = 0; |
||||
|
|
||||
|
void setupIp(std::function<bool(IPAddress ip, IPAddress gateway, IPAddress mask, IPAddress dns1, IPAddress dns2)> cb) { |
||||
|
if(mConfig->sys.ip.ip[0] != 0) { |
||||
|
IPAddress ip(mConfig->sys.ip.ip); |
||||
|
IPAddress mask(mConfig->sys.ip.mask); |
||||
|
IPAddress dns1(mConfig->sys.ip.dns1); |
||||
|
IPAddress dns2(mConfig->sys.ip.dns2); |
||||
|
IPAddress gateway(mConfig->sys.ip.gateway); |
||||
|
if(cb(ip, gateway, mask, dns1, dns2)) |
||||
|
DPRINTLN(DBG_ERROR, F("failed to set static IP!")); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
virtual void OnEvent(WiFiEvent_t event) { |
||||
|
switch(event) { |
||||
|
case SYSTEM_EVENT_STA_CONNECTED: |
||||
|
[[fallthrough]]; |
||||
|
case ARDUINO_EVENT_ETH_CONNECTED: |
||||
|
if(NetworkState::CONNECTED != mStatus) { |
||||
|
mStatus = NetworkState::CONNECTED; |
||||
|
DPRINTLN(DBG_INFO, F("Network connected")); |
||||
|
} |
||||
|
break; |
||||
|
|
||||
|
case SYSTEM_EVENT_STA_GOT_IP: |
||||
|
[[fallthrough]]; |
||||
|
case ARDUINO_EVENT_ETH_GOT_IP: |
||||
|
mStatus = NetworkState::GOT_IP; |
||||
|
break; |
||||
|
|
||||
|
case ARDUINO_EVENT_WIFI_STA_LOST_IP: |
||||
|
[[fallthrough]]; |
||||
|
case ARDUINO_EVENT_WIFI_STA_STOP: |
||||
|
[[fallthrough]]; |
||||
|
case SYSTEM_EVENT_STA_DISCONNECTED: |
||||
|
[[fallthrough]]; |
||||
|
case ARDUINO_EVENT_ETH_STOP: |
||||
|
[[fallthrough]]; |
||||
|
case ARDUINO_EVENT_ETH_DISCONNECTED: |
||||
|
mStatus = NetworkState::DISCONNECTED; |
||||
|
break; |
||||
|
|
||||
|
default: |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
void sortRSSI(int *sort, int n) { |
||||
|
for (int i = 0; i < n; i++) |
||||
|
sort[i] = i; |
||||
|
for (int i = 0; i < n; i++) |
||||
|
for (int j = i + 1; j < n; j++) |
||||
|
if (WiFi.RSSI(sort[j]) > WiFi.RSSI(sort[i])) |
||||
|
std::swap(sort[i], sort[j]); |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
void sendNTPpacket(void) { |
||||
|
uint8_t buf[NTP_PACKET_SIZE]; |
||||
|
memset(buf, 0, NTP_PACKET_SIZE); |
||||
|
|
||||
|
buf[0] = 0b11100011; // LI, Version, Mode
|
||||
|
buf[1] = 0; // Stratum
|
||||
|
buf[2] = 6; // Max Interval between messages in seconds
|
||||
|
buf[3] = 0xEC; // Clock Precision
|
||||
|
// bytes 4 - 11 are for Root Delay and Dispersion and were set to 0 by memset
|
||||
|
buf[12] = 49; // four-byte reference ID identifying
|
||||
|
buf[13] = 0x4E; |
||||
|
buf[14] = 49; |
||||
|
buf[15] = 52; |
||||
|
|
||||
|
mUdp.write(buf, NTP_PACKET_SIZE); |
||||
|
} |
||||
|
|
||||
|
void handleNTPPacket(AsyncUDPPacket packet) { |
||||
|
char buf[80]; |
||||
|
|
||||
|
memcpy(buf, packet.data(), sizeof(buf)); |
||||
|
|
||||
|
unsigned long highWord = word(buf[40], buf[41]); |
||||
|
unsigned long lowWord = word(buf[42], buf[43]); |
||||
|
|
||||
|
// combine the four bytes (two words) into a long integer
|
||||
|
// this is NTP time (seconds since Jan 1 1900):
|
||||
|
unsigned long secsSince1900 = highWord << 16 | lowWord; |
||||
|
|
||||
|
*mUtcTimestamp = secsSince1900 - 2208988800UL; // UTC time
|
||||
|
DPRINTLN(DBG_INFO, "[NTP]: " + ah::getDateTimeStr(*mUtcTimestamp) + " UTC"); |
||||
|
mOnTimeCB(true); |
||||
|
mUdp.close(); |
||||
|
} |
||||
|
|
||||
|
protected: |
||||
|
enum class NetworkState : uint8_t { |
||||
|
DISCONNECTED, |
||||
|
CONNECTED, |
||||
|
GOT_IP, |
||||
|
SCAN_READY, // ESP8266
|
||||
|
CONNECTING // ESP8266
|
||||
|
}; |
||||
|
|
||||
|
protected: |
||||
|
settings_t *mConfig = nullptr; |
||||
|
uint32_t *mUtcTimestamp = nullptr; |
||||
|
bool mConnected = false; |
||||
|
bool mScanActive = false; |
||||
|
bool mWifiConnecting = false; |
||||
|
|
||||
|
OnNetworkCB mOnNetworkCB; |
||||
|
OnTimeCB mOnTimeCB; |
||||
|
|
||||
|
NetworkState mStatus = NetworkState::DISCONNECTED; |
||||
|
|
||||
|
AhoyWifiAp mAp; |
||||
|
DNSServer mDns; |
||||
|
|
||||
|
AsyncUDP mUdp; // for time server
|
||||
|
#if defined(ESP8266) |
||||
|
WiFiEventHandler wifiConnectHandler, wifiDisconnectHandler, wifiGotIPHandler; |
||||
|
#endif |
||||
|
}; |
||||
|
|
||||
|
#endif /*__AHOY_NETWORK_H__*/ |
@ -0,0 +1,20 @@ |
|||||
|
//-----------------------------------------------------------------------------
|
||||
|
// 2024 Ahoy, https://ahoydtu.de
|
||||
|
// Creative Commons - https://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
||||
|
//-----------------------------------------------------------------------------
|
||||
|
|
||||
|
#include "AhoyNetworkHelper.h" |
||||
|
|
||||
|
namespace ah { |
||||
|
void welcome(String ip, String info) { |
||||
|
DBGPRINTLN(F("\n\n-------------------")); |
||||
|
DBGPRINTLN(F("Welcome to AHOY!")); |
||||
|
DBGPRINT(F("\npoint your browser to http://")); |
||||
|
DBGPRINT(ip); |
||||
|
DBGPRINT(" ("); |
||||
|
DBGPRINT(info); |
||||
|
DBGPRINTLN(")"); |
||||
|
DBGPRINTLN(F("to configure your device")); |
||||
|
DBGPRINTLN(F("-------------------\n")); |
||||
|
} |
||||
|
} |
@ -0,0 +1,39 @@ |
|||||
|
//-----------------------------------------------------------------------------
|
||||
|
// 2024 Ahoy, https://ahoydtu.de
|
||||
|
// Creative Commons - https://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
||||
|
//-----------------------------------------------------------------------------
|
||||
|
|
||||
|
#ifndef __AHOY_NETWORK_HELPER_H__ |
||||
|
#define __AHOY_NETWORK_HELPER_H__ |
||||
|
|
||||
|
#include "../utils/dbg.h" |
||||
|
#include <Arduino.h> |
||||
|
#if defined(ESP32) |
||||
|
#include "ESPAsyncWebServer.h" |
||||
|
#include <WiFiType.h> |
||||
|
#include <ESPmDNS.h> |
||||
|
#else |
||||
|
#include <ESP8266WiFi.h> |
||||
|
#include <ESP8266mDNS.h> |
||||
|
//#include <WiFiUdp.h>
|
||||
|
#include "ESPAsyncUDP.h" |
||||
|
|
||||
|
enum { |
||||
|
SYSTEM_EVENT_STA_CONNECTED = 1, |
||||
|
ARDUINO_EVENT_ETH_CONNECTED, |
||||
|
SYSTEM_EVENT_STA_GOT_IP, |
||||
|
ARDUINO_EVENT_ETH_GOT_IP, |
||||
|
ARDUINO_EVENT_WIFI_STA_LOST_IP, |
||||
|
ARDUINO_EVENT_WIFI_STA_STOP, |
||||
|
SYSTEM_EVENT_STA_DISCONNECTED, |
||||
|
ARDUINO_EVENT_ETH_STOP, |
||||
|
ARDUINO_EVENT_ETH_DISCONNECTED |
||||
|
}; |
||||
|
#endif |
||||
|
#include <DNSServer.h> |
||||
|
|
||||
|
namespace ah { |
||||
|
void welcome(String ip, String info); |
||||
|
} |
||||
|
|
||||
|
#endif /*__AHOY_NETWORK_HELPER_H__*/ |
@ -0,0 +1,76 @@ |
|||||
|
//-----------------------------------------------------------------------------
|
||||
|
// 2024 Ahoy, https://ahoydtu.de
|
||||
|
// Creative Commons - https://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
||||
|
//-----------------------------------------------------------------------------
|
||||
|
|
||||
|
#ifndef __AHOY_WIFI_AP_H__ |
||||
|
#define __AHOY_WIFI_AP_H__ |
||||
|
|
||||
|
#include "../utils/dbg.h" |
||||
|
#include <Arduino.h> |
||||
|
#include "../config/settings.h" |
||||
|
#include "AhoyNetworkHelper.h" |
||||
|
|
||||
|
class AhoyWifiAp { |
||||
|
public: |
||||
|
AhoyWifiAp() : mIp(192, 168, 4, 1) {} |
||||
|
|
||||
|
void setup(cfgSys_t *cfg) { |
||||
|
mCfg = cfg; |
||||
|
} |
||||
|
|
||||
|
void tickLoop() { |
||||
|
if(mEnabled) |
||||
|
mDns.processNextRequest(); |
||||
|
|
||||
|
if (WiFi.softAPgetStationNum() != mLast) { |
||||
|
mLast = WiFi.softAPgetStationNum(); |
||||
|
if(mLast > 0) |
||||
|
DBGPRINTLN(F("AP client connected")); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
void enable() { |
||||
|
if(mEnabled) |
||||
|
return; |
||||
|
|
||||
|
ah::welcome(mIp.toString(), String(F("Password: ") + String(mCfg->apPwd))); |
||||
|
|
||||
|
WiFi.mode(WIFI_AP_STA); |
||||
|
WiFi.softAPConfig(mIp, mIp, IPAddress(255, 255, 255, 0)); |
||||
|
WiFi.softAP(WIFI_AP_SSID, mCfg->apPwd); |
||||
|
|
||||
|
mDns.setErrorReplyCode(DNSReplyCode::NoError); |
||||
|
mDns.start(53, "*", mIp); |
||||
|
|
||||
|
mEnabled = true; |
||||
|
tickLoop(); |
||||
|
} |
||||
|
|
||||
|
void disable() { |
||||
|
if(!mEnabled) |
||||
|
return; |
||||
|
|
||||
|
if(WiFi.softAPgetStationNum() > 0) |
||||
|
return; |
||||
|
|
||||
|
mDns.stop(); |
||||
|
WiFi.softAPdisconnect(); |
||||
|
WiFi.mode(WIFI_STA); |
||||
|
|
||||
|
mEnabled = false; |
||||
|
} |
||||
|
|
||||
|
bool isEnabled() const { |
||||
|
return mEnabled; |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
cfgSys_t *mCfg = nullptr; |
||||
|
DNSServer mDns; |
||||
|
IPAddress mIp; |
||||
|
bool mEnabled = false; |
||||
|
uint8_t mLast = 0; |
||||
|
}; |
||||
|
|
||||
|
#endif /*__AHOY_WIFI_AP_H__*/ |
@ -0,0 +1,103 @@ |
|||||
|
//-----------------------------------------------------------------------------
|
||||
|
// 2024 Ahoy, https://ahoydtu.de
|
||||
|
// Creative Commons - https://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
||||
|
//-----------------------------------------------------------------------------
|
||||
|
|
||||
|
#ifndef __AHOY_WIFI_ESP32_H__ |
||||
|
#define __AHOY_WIFI_ESP32_H__ |
||||
|
|
||||
|
#if defined(ESP32) |
||||
|
#include <functional> |
||||
|
#include <AsyncUDP.h> |
||||
|
#include "AhoyNetwork.h" |
||||
|
#include "ESPAsyncWebServer.h" |
||||
|
|
||||
|
class AhoyWifi : public AhoyNetwork { |
||||
|
public: |
||||
|
virtual void begin() override { |
||||
|
mAp.enable(); |
||||
|
|
||||
|
if(strlen(mConfig->sys.stationSsid) == 0) |
||||
|
return; // no station wifi defined
|
||||
|
|
||||
|
|
||||
|
WiFi.disconnect(); // clean up
|
||||
|
WiFi.setHostname(mConfig->sys.deviceName); |
||||
|
#if !defined(AP_ONLY) |
||||
|
WiFi.setScanMethod(WIFI_ALL_CHANNEL_SCAN); |
||||
|
WiFi.setSortMethod(WIFI_CONNECT_AP_BY_SIGNAL); |
||||
|
setStaticIp(); |
||||
|
WiFi.begin(mConfig->sys.stationSsid, mConfig->sys.stationPwd, WIFI_ALL_CHANNEL_SCAN); |
||||
|
mWifiConnecting = true; |
||||
|
|
||||
|
DBGPRINT(F("connect to network '")); |
||||
|
DBGPRINT(mConfig->sys.stationSsid); |
||||
|
DBGPRINTLN(F("'")); |
||||
|
#endif |
||||
|
} |
||||
|
|
||||
|
void tickNetworkLoop() override { |
||||
|
if(mAp.isEnabled()) |
||||
|
mAp.tickLoop(); |
||||
|
} |
||||
|
|
||||
|
virtual String getIp(void) override { |
||||
|
return WiFi.localIP().toString(); |
||||
|
} |
||||
|
|
||||
|
virtual String getMac(void) override { |
||||
|
return WiFi.macAddress(); |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
virtual void OnEvent(WiFiEvent_t event) override { |
||||
|
switch(event) { |
||||
|
case SYSTEM_EVENT_STA_CONNECTED: |
||||
|
if(NetworkState::CONNECTED != mStatus) { |
||||
|
mStatus = NetworkState::CONNECTED; |
||||
|
mWifiConnecting = false; |
||||
|
DPRINTLN(DBG_INFO, F("Network connected")); |
||||
|
} |
||||
|
break; |
||||
|
|
||||
|
case SYSTEM_EVENT_STA_GOT_IP: |
||||
|
mStatus = NetworkState::GOT_IP; |
||||
|
if(mAp.isEnabled()) |
||||
|
mAp.disable(); |
||||
|
|
||||
|
if(!mConnected) { |
||||
|
mConnected = true; |
||||
|
ah::welcome(WiFi.localIP().toString(), F("Station")); |
||||
|
MDNS.begin(mConfig->sys.deviceName); |
||||
|
mOnNetworkCB(true); |
||||
|
} |
||||
|
break; |
||||
|
|
||||
|
case ARDUINO_EVENT_WIFI_STA_LOST_IP: |
||||
|
[[fallthrough]]; |
||||
|
case ARDUINO_EVENT_WIFI_STA_STOP: |
||||
|
[[fallthrough]]; |
||||
|
case SYSTEM_EVENT_STA_DISCONNECTED: |
||||
|
mStatus = NetworkState::DISCONNECTED; |
||||
|
if(mConnected) { |
||||
|
mConnected = false; |
||||
|
mOnNetworkCB(false); |
||||
|
MDNS.end(); |
||||
|
begin(); |
||||
|
} |
||||
|
break; |
||||
|
|
||||
|
default: |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
virtual void setStaticIp() override { |
||||
|
setupIp([this](IPAddress ip, IPAddress gateway, IPAddress mask, IPAddress dns1, IPAddress dns2) -> bool { |
||||
|
return WiFi.config(ip, gateway, mask, dns1, dns2); |
||||
|
}); |
||||
|
} |
||||
|
}; |
||||
|
|
||||
|
#endif /*ESP32*/ |
||||
|
#endif /*__AHOY_WIFI_ESP32_H__*/ |
@ -0,0 +1,170 @@ |
|||||
|
//-----------------------------------------------------------------------------
|
||||
|
// 2024 Ahoy, https://ahoydtu.de
|
||||
|
// Creative Commons - https://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
||||
|
//-----------------------------------------------------------------------------
|
||||
|
|
||||
|
#ifndef __AHOY_WIFI_ESP8266_H__ |
||||
|
#define __AHOY_WIFI_ESP8266_H__ |
||||
|
|
||||
|
#if defined(ESP8266) |
||||
|
#include <functional> |
||||
|
#include <list> |
||||
|
#include <WiFiUdp.h> |
||||
|
#include "AhoyNetwork.h" |
||||
|
#include "ESPAsyncWebServer.h" |
||||
|
|
||||
|
class AhoyWifi : public AhoyNetwork { |
||||
|
public: |
||||
|
void begin() override { |
||||
|
mAp.enable(); |
||||
|
|
||||
|
WiFi.setHostname(mConfig->sys.deviceName); |
||||
|
mBSSIDList.clear(); |
||||
|
} |
||||
|
|
||||
|
void tickNetworkLoop() override { |
||||
|
if(mAp.isEnabled()) |
||||
|
mAp.tickLoop(); |
||||
|
|
||||
|
mCnt++; |
||||
|
|
||||
|
switch(mStatus) { |
||||
|
case NetworkState::DISCONNECTED: |
||||
|
if(mConnected) { |
||||
|
mConnected = false; |
||||
|
mWifiConnecting = false; |
||||
|
mOnNetworkCB(false); |
||||
|
mAp.enable(); |
||||
|
MDNS.end(); |
||||
|
} |
||||
|
|
||||
|
if (WiFi.softAPgetStationNum() > 0) { |
||||
|
DBGPRINTLN(F("AP client connected")); |
||||
|
} |
||||
|
#if !defined(AP_ONLY) |
||||
|
else if (!mScanActive) { |
||||
|
DBGPRINT(F("scanning APs with SSID ")); |
||||
|
DBGPRINTLN(String(mConfig->sys.stationSsid)); |
||||
|
mScanCnt = 0; |
||||
|
mCnt = 0; |
||||
|
mScanActive = true; |
||||
|
WiFi.scanNetworks(true, true, 0U, ([this]() { |
||||
|
if (mConfig->sys.isHidden) |
||||
|
return (uint8_t*)NULL; |
||||
|
return (uint8_t*)(mConfig->sys.stationSsid); |
||||
|
})()); |
||||
|
} else if(getBSSIDs()) { |
||||
|
mStatus = NetworkState::SCAN_READY; |
||||
|
DBGPRINT(F("connect to network '")); Serial.flush(); |
||||
|
DBGPRINTLN(mConfig->sys.stationSsid); |
||||
|
} |
||||
|
#endif |
||||
|
break; |
||||
|
|
||||
|
case NetworkState::SCAN_READY: |
||||
|
mStatus = NetworkState::CONNECTING; |
||||
|
DBGPRINT(F("try to connect to BSSID:")); |
||||
|
uint8_t bssid[6]; |
||||
|
for (int j = 0; j < 6; j++) { |
||||
|
bssid[j] = mBSSIDList.front(); |
||||
|
mBSSIDList.pop_front(); |
||||
|
DBGPRINT(" " + String(bssid[j], HEX)); |
||||
|
} |
||||
|
DBGPRINTLN(""); |
||||
|
setStaticIp(); |
||||
|
WiFi.begin(mConfig->sys.stationSsid, mConfig->sys.stationPwd, 0, &bssid[0]); |
||||
|
mWifiConnecting = true; |
||||
|
break; |
||||
|
|
||||
|
case NetworkState::CONNECTING: |
||||
|
if (isTimeout(TIMEOUT)) { |
||||
|
WiFi.disconnect(); |
||||
|
mWifiConnecting = false; |
||||
|
mStatus = mBSSIDList.empty() ? NetworkState::DISCONNECTED : NetworkState::SCAN_READY; |
||||
|
} |
||||
|
break; |
||||
|
|
||||
|
case NetworkState::CONNECTED: |
||||
|
break; |
||||
|
|
||||
|
case NetworkState::GOT_IP: |
||||
|
if(!mConnected) { |
||||
|
mAp.disable(); |
||||
|
mConnected = true; |
||||
|
ah::welcome(WiFi.localIP().toString(), F("Station")); |
||||
|
MDNS.begin(mConfig->sys.deviceName); |
||||
|
MDNSResponder::hMDNSService hRes = MDNS.addService(NULL, "http", "tcp", 80); |
||||
|
MDNS.addServiceTxt(hRes, "path", "/"); |
||||
|
MDNS.announce(); |
||||
|
mOnNetworkCB(true); |
||||
|
} |
||||
|
|
||||
|
MDNS.update(); |
||||
|
|
||||
|
if(WiFi.channel() > 11) |
||||
|
mWasInCh12to14 = true; |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
String getIp(void) override { |
||||
|
return WiFi.localIP().toString(); |
||||
|
} |
||||
|
|
||||
|
String getMac(void) override { |
||||
|
return WiFi.macAddress(); |
||||
|
} |
||||
|
|
||||
|
bool getWasInCh12to14() override { |
||||
|
return mWasInCh12to14; |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
void setStaticIp() override { |
||||
|
setupIp([this](IPAddress ip, IPAddress gateway, IPAddress mask, IPAddress dns1, IPAddress dns2) -> bool { |
||||
|
return WiFi.config(ip, gateway, mask, dns1, dns2); |
||||
|
}); |
||||
|
} |
||||
|
|
||||
|
bool getBSSIDs() { |
||||
|
bool result = false; |
||||
|
int n = WiFi.scanComplete(); |
||||
|
if (n < 0) { |
||||
|
if (++mScanCnt < 20) |
||||
|
return false; |
||||
|
} |
||||
|
if(n > 0) { |
||||
|
mBSSIDList.clear(); |
||||
|
int sort[n]; |
||||
|
sortRSSI(&sort[0], n); |
||||
|
for (int i = 0; i < n; i++) { |
||||
|
DBGPRINT("BSSID " + String(i) + ":"); |
||||
|
uint8_t *bssid = WiFi.BSSID(sort[i]); |
||||
|
for (int j = 0; j < 6; j++){ |
||||
|
DBGPRINT(" " + String(bssid[j], HEX)); |
||||
|
mBSSIDList.push_back(bssid[j]); |
||||
|
} |
||||
|
DBGPRINTLN(""); |
||||
|
} |
||||
|
result = true; |
||||
|
} |
||||
|
mScanActive = false; |
||||
|
WiFi.scanDelete(); |
||||
|
return result; |
||||
|
} |
||||
|
|
||||
|
bool isTimeout(uint8_t timeout) { |
||||
|
return ((mCnt % timeout) == 0); |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
uint8_t mCnt = 0; |
||||
|
uint8_t mScanCnt = 0; |
||||
|
std::list<uint8_t> mBSSIDList; |
||||
|
bool mWasInCh12to14 = false; |
||||
|
static constexpr uint8_t TIMEOUT = 20; |
||||
|
static constexpr uint8_t SCAN_TIMEOUT = 10; |
||||
|
}; |
||||
|
|
||||
|
#endif /*ESP8266*/ |
||||
|
#endif /*__AHOY_WIFI_ESP8266_H__*/ |
@ -0,0 +1,304 @@ |
|||||
|
//-----------------------------------------------------------------------------
|
||||
|
// 2024 Ahoy, https://github.com/lumpapu/ahoy
|
||||
|
// Creative Commons - http://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
||||
|
//-----------------------------------------------------------------------------
|
||||
|
|
||||
|
#ifndef __EPD_HAL_H__ |
||||
|
#define __EPD_HAL_H__ |
||||
|
|
||||
|
#pragma once |
||||
|
#include "../../utils/spiPatcher.h" |
||||
|
#include <esp_rom_gpio.h> |
||||
|
#include <GxEPD2_BW.h> |
||||
|
|
||||
|
|
||||
|
#define EPD_DEFAULT_SPI_SPEED 4000000 // 4 MHz
|
||||
|
|
||||
|
class epdHal: public GxEPD2_HalInterface, public SpiPatcherHandle { |
||||
|
public: |
||||
|
epdHal() {} |
||||
|
|
||||
|
void patch() override { |
||||
|
esp_rom_gpio_connect_out_signal(mPinMosi, spi_periph_signal[mHostDevice].spid_out, false, false); |
||||
|
esp_rom_gpio_connect_in_signal(mPinBusy, spi_periph_signal[mHostDevice].spiq_in, false); |
||||
|
esp_rom_gpio_connect_out_signal(mPinClk, spi_periph_signal[mHostDevice].spiclk_out, false, false); |
||||
|
} |
||||
|
|
||||
|
void unpatch() override { |
||||
|
esp_rom_gpio_connect_out_signal(mPinMosi, SIG_GPIO_OUT_IDX, false, false); |
||||
|
esp_rom_gpio_connect_in_signal(mPinBusy, GPIO_MATRIX_CONST_ZERO_INPUT, false); |
||||
|
esp_rom_gpio_connect_out_signal(mPinClk, SIG_GPIO_OUT_IDX, false, false); |
||||
|
} |
||||
|
|
||||
|
void init(int8_t mosi, int8_t dc, int8_t sclk, int8_t cs, int8_t rst, int8_t busy, int32_t speed = EPD_DEFAULT_SPI_SPEED) { |
||||
|
mPinMosi = static_cast<gpio_num_t>(mosi); |
||||
|
mPinDc = static_cast<gpio_num_t>(dc); |
||||
|
mPinClk = static_cast<gpio_num_t>(sclk); |
||||
|
mPinCs = static_cast<gpio_num_t>(cs); |
||||
|
mPinRst = static_cast<gpio_num_t>(rst); |
||||
|
mPinBusy = static_cast<gpio_num_t>(busy); |
||||
|
mSpiSpeed = speed; |
||||
|
|
||||
|
#if defined(CONFIG_IDF_TARGET_ESP32S3) |
||||
|
mHostDevice = SPI3_HOST; |
||||
|
#else |
||||
|
mHostDevice = (14 == sclk) ? SPI2_HOST : SPI_HOST_OTHER; |
||||
|
#endif |
||||
|
|
||||
|
mSpiPatcher = SpiPatcher::getInstance(mHostDevice); |
||||
|
|
||||
|
gpio_reset_pin(mPinMosi); |
||||
|
gpio_set_direction(mPinMosi, GPIO_MODE_OUTPUT); |
||||
|
gpio_set_level(mPinMosi, 1); |
||||
|
|
||||
|
gpio_reset_pin(mPinClk); |
||||
|
gpio_set_direction(mPinClk, GPIO_MODE_OUTPUT); |
||||
|
gpio_set_level(mPinClk, 0); |
||||
|
|
||||
|
gpio_reset_pin(mPinCs); |
||||
|
spi_device_interface_config_t devcfg = { |
||||
|
.command_bits = 0, |
||||
|
.address_bits = 0, |
||||
|
.dummy_bits = 0, |
||||
|
.mode = 0, |
||||
|
.duty_cycle_pos = 0, |
||||
|
.cs_ena_pretrans = 0, |
||||
|
.cs_ena_posttrans = 0, |
||||
|
.clock_speed_hz = mSpiSpeed, |
||||
|
.input_delay_ns = 0, |
||||
|
.spics_io_num = mPinCs, |
||||
|
.flags = 0, |
||||
|
.queue_size = 1, |
||||
|
.pre_cb = nullptr, |
||||
|
.post_cb = nullptr |
||||
|
}; |
||||
|
mSpiPatcher->addDevice(mHostDevice, &devcfg, &spi); |
||||
|
|
||||
|
if(GPIO_NUM_NC != mPinRst) { |
||||
|
gpio_reset_pin(mPinRst); |
||||
|
gpio_set_direction(mPinRst, GPIO_MODE_OUTPUT); |
||||
|
gpio_set_level(mPinRst, HIGH); |
||||
|
} |
||||
|
|
||||
|
gpio_reset_pin(mPinDc); |
||||
|
gpio_set_direction(mPinDc, GPIO_MODE_OUTPUT); |
||||
|
gpio_set_level(mPinDc, HIGH); |
||||
|
|
||||
|
//gpio_reset_pin(mPinBusy);
|
||||
|
//gpio_set_direction(mPinBusy, GPIO_MODE_INPUT);
|
||||
|
} |
||||
|
|
||||
|
void rstMode(uint8_t mode) override { |
||||
|
if(GPIO_NUM_NC != mPinRst) |
||||
|
gpio_set_direction(mPinRst, static_cast<gpio_mode_t>(mode)); |
||||
|
} |
||||
|
|
||||
|
void rst(bool level) override { |
||||
|
if(GPIO_NUM_NC != mPinRst) |
||||
|
gpio_set_level(mPinRst, level); |
||||
|
} |
||||
|
|
||||
|
int getBusy(void) override { |
||||
|
return gpio_get_level(mPinBusy); |
||||
|
} |
||||
|
|
||||
|
bool isRst(void) override { |
||||
|
return (GPIO_NUM_NC != mPinRst); |
||||
|
} |
||||
|
|
||||
|
void write(uint8_t buf) override { |
||||
|
uint8_t data[1]; |
||||
|
data[0] = buf; |
||||
|
request_spi(); |
||||
|
|
||||
|
size_t spiLen = static_cast<size_t>(1u) << 3; |
||||
|
spi_transaction_t t = { |
||||
|
.flags = 0, |
||||
|
.cmd = 0, |
||||
|
.addr = 0, |
||||
|
.length = spiLen, |
||||
|
.rxlength = spiLen, |
||||
|
.user = NULL, |
||||
|
.tx_buffer = data, |
||||
|
.rx_buffer = data |
||||
|
}; |
||||
|
ESP_ERROR_CHECK(spi_device_polling_transmit(spi, &t)); |
||||
|
|
||||
|
release_spi(); |
||||
|
} |
||||
|
|
||||
|
void write(const uint8_t *buf, uint16_t n) override { |
||||
|
uint8_t data[n]; |
||||
|
std::copy(&buf[0], &buf[n], &data[0]); |
||||
|
|
||||
|
request_spi(); |
||||
|
|
||||
|
size_t spiLen = static_cast<size_t>(n) << 3; |
||||
|
spi_transaction_t t = { |
||||
|
.flags = 0, |
||||
|
.cmd = 0, |
||||
|
.addr = 0, |
||||
|
.length = spiLen, |
||||
|
.rxlength = spiLen, |
||||
|
.user = NULL, |
||||
|
.tx_buffer = data, |
||||
|
.rx_buffer = data |
||||
|
}; |
||||
|
ESP_ERROR_CHECK(spi_device_polling_transmit(spi, &t)); |
||||
|
|
||||
|
release_spi(); |
||||
|
} |
||||
|
|
||||
|
void write(const uint8_t *buf, uint16_t n, int16_t fill_with_zeroes) override { |
||||
|
uint8_t data[n + fill_with_zeroes]; |
||||
|
memset(data, 0, (n + fill_with_zeroes)); |
||||
|
for (uint16_t i = 0; i < n; i++) { |
||||
|
data[i] = pgm_read_byte(&*buf++); |
||||
|
} |
||||
|
|
||||
|
request_spi(); |
||||
|
spi_transaction_t t = { |
||||
|
.flags = SPI_TRANS_CS_KEEP_ACTIVE, |
||||
|
.cmd = 0, |
||||
|
.addr = 0, |
||||
|
.length = 1u, |
||||
|
.rxlength = 1u, |
||||
|
.user = NULL, |
||||
|
.tx_buffer = data, |
||||
|
.rx_buffer = data |
||||
|
}; |
||||
|
|
||||
|
size_t offs = 0; |
||||
|
spi_device_acquire_bus(spi, portMAX_DELAY); |
||||
|
while(offs < (n + fill_with_zeroes)) { |
||||
|
t.length = (64u << 3); |
||||
|
t.rxlength = t.length; |
||||
|
t.tx_buffer = &data[offs]; |
||||
|
offs += 64; |
||||
|
ESP_ERROR_CHECK(spi_device_polling_transmit(spi, &t)); |
||||
|
} |
||||
|
spi_device_release_bus(spi); |
||||
|
|
||||
|
release_spi(); |
||||
|
} |
||||
|
|
||||
|
void writeCmd(const uint8_t val) override { |
||||
|
uint8_t data[1]; |
||||
|
data[0] = val; |
||||
|
|
||||
|
request_spi(); |
||||
|
gpio_set_level(mPinDc, LOW); |
||||
|
|
||||
|
size_t spiLen = static_cast<size_t>(1u) << 3; |
||||
|
spi_transaction_t t = { |
||||
|
.flags = 0, |
||||
|
.cmd = 0, |
||||
|
.addr = 0, |
||||
|
.length = spiLen, |
||||
|
.rxlength = spiLen, |
||||
|
.user = NULL, |
||||
|
.tx_buffer = data, |
||||
|
.rx_buffer = data |
||||
|
}; |
||||
|
ESP_ERROR_CHECK(spi_device_polling_transmit(spi, &t)); |
||||
|
gpio_set_level(mPinDc, HIGH); |
||||
|
|
||||
|
release_spi(); |
||||
|
} |
||||
|
|
||||
|
void writeCmd(const uint8_t *buf, uint8_t n, bool isPGM) override { |
||||
|
uint8_t data[n-1]; |
||||
|
data[0] = (isPGM) ? pgm_read_byte(&*buf++) : buf[0]; |
||||
|
|
||||
|
request_spi(); |
||||
|
gpio_set_level(mPinDc, LOW); |
||||
|
spi_device_acquire_bus(spi, portMAX_DELAY); |
||||
|
|
||||
|
size_t spiLen = static_cast<size_t>(1u) << 3; |
||||
|
spi_transaction_t t = { |
||||
|
.flags = SPI_TRANS_CS_KEEP_ACTIVE, |
||||
|
.cmd = 0, |
||||
|
.addr = 0, |
||||
|
.length = spiLen, |
||||
|
.rxlength = spiLen, |
||||
|
.user = NULL, |
||||
|
.tx_buffer = data, |
||||
|
.rx_buffer = data |
||||
|
}; |
||||
|
ESP_ERROR_CHECK(spi_device_polling_transmit(spi, &t)); |
||||
|
gpio_set_level(mPinDc, HIGH); |
||||
|
|
||||
|
if(isPGM) { |
||||
|
for (uint16_t i = 0; i < n; i++) { |
||||
|
data[i] = pgm_read_byte(&*buf++); |
||||
|
} |
||||
|
} else |
||||
|
std::copy(&buf[1], &buf[n], &data[0]); |
||||
|
|
||||
|
spiLen = static_cast<size_t>(n-1) << 3; |
||||
|
spi_transaction_t t1 = { |
||||
|
.flags = SPI_TRANS_CS_KEEP_ACTIVE, |
||||
|
.cmd = 0, |
||||
|
.addr = 0, |
||||
|
.length = spiLen, |
||||
|
.rxlength = spiLen, |
||||
|
.user = NULL, |
||||
|
.tx_buffer = data, |
||||
|
.rx_buffer = data |
||||
|
}; |
||||
|
ESP_ERROR_CHECK(spi_device_polling_transmit(spi, &t1)); |
||||
|
spi_device_release_bus(spi); |
||||
|
|
||||
|
release_spi(); |
||||
|
} |
||||
|
|
||||
|
void startTransfer(void) override { |
||||
|
request_spi(); |
||||
|
} |
||||
|
|
||||
|
void endTransfer(void) override { |
||||
|
release_spi(); |
||||
|
} |
||||
|
|
||||
|
void transfer(const uint8_t val) override { |
||||
|
uint8_t data[1]; |
||||
|
data[0] = val; |
||||
|
|
||||
|
size_t spiLen = static_cast<size_t>(1u) << 3; |
||||
|
spi_transaction_t t = { |
||||
|
.flags = 0, |
||||
|
.cmd = 0, |
||||
|
.addr = 0, |
||||
|
.length = spiLen, |
||||
|
.rxlength = spiLen, |
||||
|
.user = NULL, |
||||
|
.tx_buffer = data, |
||||
|
.rx_buffer = data |
||||
|
}; |
||||
|
ESP_ERROR_CHECK(spi_device_polling_transmit(spi, &t)); |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
inline void request_spi() { |
||||
|
mSpiPatcher->request(this); |
||||
|
} |
||||
|
|
||||
|
inline void release_spi() { |
||||
|
mSpiPatcher->release(); |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
gpio_num_t mPinMosi = GPIO_NUM_NC; |
||||
|
gpio_num_t mPinDc = GPIO_NUM_NC; |
||||
|
gpio_num_t mPinClk = GPIO_NUM_NC; |
||||
|
gpio_num_t mPinCs = GPIO_NUM_NC; |
||||
|
gpio_num_t mPinRst = GPIO_NUM_NC; |
||||
|
gpio_num_t mPinBusy = GPIO_NUM_NC; |
||||
|
int32_t mSpiSpeed = EPD_DEFAULT_SPI_SPEED; |
||||
|
|
||||
|
spi_host_device_t mHostDevice; |
||||
|
spi_device_handle_t spi; |
||||
|
SpiPatcher *mSpiPatcher; |
||||
|
}; |
||||
|
|
||||
|
#endif /*__EPD_HAL_H__*/ |
@ -0,0 +1,67 @@ |
|||||
|
//-----------------------------------------------------------------------------
|
||||
|
// 2024 Ahoy, https://github.com/lumpapu/ahoy
|
||||
|
// Creative Commons - http://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
||||
|
//-----------------------------------------------------------------------------
|
||||
|
|
||||
|
#ifndef __MAX_VALUE__ |
||||
|
#define __MAX_VALUE__ |
||||
|
#pragma once |
||||
|
|
||||
|
#include <array> |
||||
|
#include <utility> |
||||
|
#include "../hm/hmDefines.h" |
||||
|
|
||||
|
template<class T=float> |
||||
|
class MaxPower { |
||||
|
public: |
||||
|
MaxPower() { |
||||
|
mTs = nullptr; |
||||
|
mMaxDiff = 60; |
||||
|
reset(); |
||||
|
} |
||||
|
|
||||
|
void setup(uint32_t *ts, uint16_t interval) { |
||||
|
mTs = ts; |
||||
|
mMaxDiff = interval * 4; |
||||
|
} |
||||
|
|
||||
|
void reset(void) { |
||||
|
mValues.fill(std::make_pair(0, 0.0)); |
||||
|
mLast = 0.0; |
||||
|
} |
||||
|
|
||||
|
void payloadEvent(uint8_t cmd, Inverter<> *iv) { |
||||
|
if(RealTimeRunData_Debug != cmd) |
||||
|
return; |
||||
|
|
||||
|
if(nullptr == iv) |
||||
|
return; |
||||
|
|
||||
|
if(iv->id >= MAX_NUM_INVERTERS) |
||||
|
return; |
||||
|
|
||||
|
record_t<> *rec = iv->getRecordStruct(RealTimeRunData_Debug); |
||||
|
mValues[iv->id] = std::make_pair(*mTs, iv->getChannelFieldValue(CH0, FLD_PAC, rec)); |
||||
|
} |
||||
|
|
||||
|
T getTotalMaxPower(void) { |
||||
|
T val = 0; |
||||
|
for(uint8_t i = 0; i < MAX_NUM_INVERTERS; i ++) { |
||||
|
if((mValues[i].first + mMaxDiff) >= *mTs) |
||||
|
val += mValues[i].second; |
||||
|
else if(mValues[i].first > 0) |
||||
|
break; // old data
|
||||
|
} |
||||
|
if(val > mLast) |
||||
|
mLast = val; |
||||
|
return mLast; |
||||
|
} |
||||
|
|
||||
|
private: |
||||
|
uint32_t *mTs; |
||||
|
uint32_t mMaxDiff; |
||||
|
float mLast; |
||||
|
std::array<std::pair<uint32_t, T>, MAX_NUM_INVERTERS> mValues; |
||||
|
}; |
||||
|
|
||||
|
#endif |
@ -0,0 +1,44 @@ |
|||||
|
//-----------------------------------------------------------------------------
|
||||
|
// 2024 Ahoy, https://ahoydtu.de
|
||||
|
// Creative Commons - http://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
||||
|
//-----------------------------------------------------------------------------
|
||||
|
|
||||
|
#ifndef __PLUGIN_LANG_H__ |
||||
|
#define __PLUGIN_LANG_H__ |
||||
|
|
||||
|
#ifdef LANG_DE |
||||
|
#define STR_MONTHNAME_3_CHAR_LIST "ErrJanFebMrzAprMaiJunJulAugSepOktNovDez" |
||||
|
#define STR_DAYNAME_3_CHAR_LIST "ErrSonMonDieMitDonFreSam" |
||||
|
#define STR_OFFLINE "aus" |
||||
|
#define STR_ONLINE "aktiv" |
||||
|
#define STR_NO_INVERTER "kein inverter" |
||||
|
#define STR_NO_WIFI "WLAN nicht verbunden" |
||||
|
#define STR_VERSION "Version" |
||||
|
#define STR_ACTIVE_INVERTERS "aktive WR" |
||||
|
#define STR_TODAY "heute" |
||||
|
#define STR_TOTAL "Gesamt" |
||||
|
#elif LANG_FR |
||||
|
#define STR_MONTHNAME_3_CHAR_LIST "ErrJanFevMarAvrMaiJunJulAouSepOctNovDec" |
||||
|
#define STR_DAYNAME_3_CHAR_LIST "ErrDimLunMarMerJeuVenSam" |
||||
|
#define STR_OFFLINE "eteint" |
||||
|
#define STR_ONLINE "online" |
||||
|
#define STR_NO_INVERTER "pas d'onduleur" |
||||
|
#define STR_NO_WIFI "WiFi not connected" |
||||
|
#define STR_VERSION "Version" |
||||
|
#define STR_ACTIVE_INVERTERS "active Inv" |
||||
|
#define STR_TODAY "today" |
||||
|
#define STR_TOTAL "total" |
||||
|
#else |
||||
|
#define STR_MONTHNAME_3_CHAR_LIST "ErrJanFebMarAprMayJunJulAugSepOctNovDec" |
||||
|
#define STR_DAYNAME_3_CHAR_LIST "ErrSunMonTueWedThuFriSat" |
||||
|
#define STR_OFFLINE "offline" |
||||
|
#define STR_ONLINE "online" |
||||
|
#define STR_NO_INVERTER "no inverter" |
||||
|
#define STR_NO_WIFI "WiFi not connected" |
||||
|
#define STR_VERSION "Version" |
||||
|
#define STR_ACTIVE_INVERTERS "active Inv" |
||||
|
#define STR_TODAY "today" |
||||
|
#define STR_TOTAL "total" |
||||
|
#endif |
||||
|
|
||||
|
#endif /*__PLUGIN_LANG_H__*/ |
@ -1,488 +0,0 @@ |
|||||
//-----------------------------------------------------------------------------
|
|
||||
// 2024 Ahoy, https://ahoydtu.de
|
|
||||
// Creative Commons - https://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
|
|
||||
#if !defined(ETHERNET) |
|
||||
#if defined(ESP32) && defined(F) |
|
||||
#undef F |
|
||||
#define F(sl) (sl) |
|
||||
#endif |
|
||||
#include "ahoywifi.h" |
|
||||
|
|
||||
#if defined(ESP32) |
|
||||
#include <ESPmDNS.h> |
|
||||
#else |
|
||||
#include <ESP8266mDNS.h> |
|
||||
#endif |
|
||||
|
|
||||
// NTP CONFIG
|
|
||||
#define NTP_PACKET_SIZE 48 |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
ahoywifi::ahoywifi() : mApIp(192, 168, 4, 1) {} |
|
||||
|
|
||||
|
|
||||
/**
|
|
||||
* TODO: ESP32 has native strongest AP support! |
|
||||
* WiFi.setScanMethod(WIFI_ALL_CHANNEL_SCAN); |
|
||||
WiFi.setSortMethod(WIFI_CONNECT_AP_BY_SIGNAL); |
|
||||
*/ |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoywifi::setup(settings_t *config, uint32_t *utcTimestamp, appWifiCb cb) { |
|
||||
mConfig = config; |
|
||||
mUtcTimestamp = utcTimestamp; |
|
||||
mAppWifiCb = cb; |
|
||||
|
|
||||
mGotDisconnect = false; |
|
||||
mStaConn = DISCONNECTED; |
|
||||
mCnt = 0; |
|
||||
mScanActive = false; |
|
||||
mScanCnt = 0; |
|
||||
mStopApAllowed = true; |
|
||||
|
|
||||
#if defined(ESP8266) |
|
||||
wifiConnectHandler = WiFi.onStationModeConnected(std::bind(&ahoywifi::onConnect, this, std::placeholders::_1)); |
|
||||
wifiGotIPHandler = WiFi.onStationModeGotIP(std::bind(&ahoywifi::onGotIP, this, std::placeholders::_1)); |
|
||||
wifiDisconnectHandler = WiFi.onStationModeDisconnected(std::bind(&ahoywifi::onDisconnect, this, std::placeholders::_1)); |
|
||||
#else |
|
||||
WiFi.onEvent(std::bind(&ahoywifi::onWiFiEvent, this, std::placeholders::_1)); |
|
||||
#endif |
|
||||
|
|
||||
setupWifi(true); |
|
||||
} |
|
||||
|
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoywifi::setupWifi(bool startAP = false) { |
|
||||
#if !defined(FB_WIFI_OVERRIDDEN) |
|
||||
if(startAP) { |
|
||||
setupAp(); |
|
||||
delay(1000); |
|
||||
} |
|
||||
#endif |
|
||||
#if !defined(AP_ONLY) |
|
||||
#if defined(FB_WIFI_OVERRIDDEN) |
|
||||
snprintf(mConfig->sys.stationSsid, SSID_LEN, "%s", FB_WIFI_SSID); |
|
||||
snprintf(mConfig->sys.stationPwd, PWD_LEN, "%s", FB_WIFI_PWD); |
|
||||
setupStation(); |
|
||||
#else |
|
||||
if(mConfig->valid) { |
|
||||
if(strncmp(mConfig->sys.stationSsid, FB_WIFI_SSID, 14) != 0) |
|
||||
setupStation(); |
|
||||
} |
|
||||
#endif |
|
||||
#endif |
|
||||
} |
|
||||
|
|
||||
|
|
||||
void ahoywifi::tickWifiLoop() { |
|
||||
static const uint8_t TIMEOUT = 20; |
|
||||
static const uint8_t SCAN_TIMEOUT = 10; |
|
||||
#if !defined(AP_ONLY) |
|
||||
|
|
||||
mCnt++; |
|
||||
|
|
||||
switch (mStaConn) { |
|
||||
case IN_STA_MODE: |
|
||||
// Nothing to do
|
|
||||
if (mGotDisconnect) { |
|
||||
mStaConn = RESET; |
|
||||
} |
|
||||
#if !defined(ESP32) |
|
||||
MDNS.update(); |
|
||||
if(WiFi.channel() > 11) |
|
||||
mWasInCh12to14 = true; |
|
||||
#endif |
|
||||
return; |
|
||||
case IN_AP_MODE: |
|
||||
if ((WiFi.softAPgetStationNum() == 0) || (!mStopApAllowed)) { |
|
||||
mCnt = 0; |
|
||||
mDns.stop(); |
|
||||
WiFi.mode(WIFI_AP_STA); |
|
||||
mStaConn = DISCONNECTED; |
|
||||
} else { |
|
||||
mDns.processNextRequest(); |
|
||||
return; |
|
||||
} |
|
||||
break; |
|
||||
case DISCONNECTED: |
|
||||
if ((WiFi.softAPgetStationNum() > 0) && (mStopApAllowed)) { |
|
||||
mStaConn = IN_AP_MODE; |
|
||||
// first time switch to AP Mode
|
|
||||
if (mScanActive) { |
|
||||
WiFi.scanDelete(); |
|
||||
mScanActive = false; |
|
||||
} |
|
||||
DBGPRINTLN(F("AP client connected")); |
|
||||
welcome(mApIp.toString(), ""); |
|
||||
WiFi.mode(WIFI_AP); |
|
||||
mDns.start(53, "*", mApIp); |
|
||||
mAppWifiCb(true); |
|
||||
mDns.processNextRequest(); |
|
||||
return; |
|
||||
} else if (!mScanActive) { |
|
||||
DBGPRINT(F("scanning APs with SSID ")); |
|
||||
DBGPRINTLN(String(mConfig->sys.stationSsid)); |
|
||||
mScanCnt = 0; |
|
||||
mCnt = 0; |
|
||||
mScanActive = true; |
|
||||
#if defined(ESP8266) |
|
||||
WiFi.scanNetworks(true, true, 0U, ([this]() { |
|
||||
if (mConfig->sys.isHidden) |
|
||||
return (uint8_t*)NULL; |
|
||||
return (uint8_t*)(mConfig->sys.stationSsid); |
|
||||
})()); |
|
||||
#else |
|
||||
WiFi.scanNetworks(true, true, false, 300U, 0U, ([this]() { |
|
||||
if (mConfig->sys.isHidden) |
|
||||
return (char*)NULL; |
|
||||
return (mConfig->sys.stationSsid); |
|
||||
})()); |
|
||||
#endif |
|
||||
return; |
|
||||
} else if(getBSSIDs()) { |
|
||||
// Scan ready
|
|
||||
mStaConn = SCAN_READY; |
|
||||
} else { |
|
||||
// In case of a timeout, what do we do?
|
|
||||
// For now we start scanning again as the original code did.
|
|
||||
// Would be better to into PA mode
|
|
||||
|
|
||||
if (isTimeout(SCAN_TIMEOUT)) { |
|
||||
WiFi.scanDelete(); |
|
||||
mScanActive = false; |
|
||||
} |
|
||||
} |
|
||||
break; |
|
||||
case SCAN_READY: |
|
||||
mStaConn = CONNECTING; |
|
||||
mCnt = 0; |
|
||||
DBGPRINT(F("try to connect to AP with BSSID:")); |
|
||||
uint8_t bssid[6]; |
|
||||
for (int j = 0; j < 6; j++) { |
|
||||
bssid[j] = mBSSIDList.front(); |
|
||||
mBSSIDList.pop_front(); |
|
||||
DBGPRINT(" " + String(bssid[j], HEX)); |
|
||||
} |
|
||||
DBGPRINTLN(""); |
|
||||
mGotDisconnect = false; |
|
||||
WiFi.begin(mConfig->sys.stationSsid, mConfig->sys.stationPwd, 0, &bssid[0]); |
|
||||
|
|
||||
break; |
|
||||
case CONNECTING: |
|
||||
if (isTimeout(TIMEOUT)) { |
|
||||
WiFi.disconnect(); |
|
||||
mStaConn = mBSSIDList.empty() ? DISCONNECTED : SCAN_READY; |
|
||||
} |
|
||||
break; |
|
||||
case CONNECTED: |
|
||||
// Connection but no IP yet
|
|
||||
if (isTimeout(TIMEOUT) || mGotDisconnect) { |
|
||||
mStaConn = RESET; |
|
||||
} |
|
||||
break; |
|
||||
case GOT_IP: |
|
||||
welcome(WiFi.localIP().toString(), F(" (Station)")); |
|
||||
if(mStopApAllowed) { |
|
||||
WiFi.softAPdisconnect(); |
|
||||
WiFi.mode(WIFI_STA); |
|
||||
DBGPRINTLN(F("[WiFi] AP disabled")); |
|
||||
delay(100); |
|
||||
} |
|
||||
mAppWifiCb(true); |
|
||||
mGotDisconnect = false; |
|
||||
mStaConn = IN_STA_MODE; |
|
||||
|
|
||||
if (!MDNS.begin(mConfig->sys.deviceName)) { |
|
||||
DPRINTLN(DBG_ERROR, F("Error setting up MDNS responder!")); |
|
||||
} else { |
|
||||
DBGPRINT(F("[WiFi] mDNS established: ")); |
|
||||
DBGPRINT(mConfig->sys.deviceName); |
|
||||
DBGPRINTLN(F(".local")); |
|
||||
} |
|
||||
|
|
||||
break; |
|
||||
case RESET: |
|
||||
mGotDisconnect = false; |
|
||||
mStaConn = DISCONNECTED; |
|
||||
mCnt = 5; // try to reconnect in 5 sec
|
|
||||
setupWifi(); // reconnect with AP / Station setup
|
|
||||
mAppWifiCb(false); |
|
||||
DPRINTLN(DBG_INFO, "[WiFi] Connection Lost"); |
|
||||
break; |
|
||||
default: |
|
||||
DBGPRINTLN(F("Unhandled status")); |
|
||||
break; |
|
||||
} |
|
||||
|
|
||||
#endif |
|
||||
} |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoywifi::setupAp(void) { |
|
||||
DPRINTLN(DBG_VERBOSE, F("wifi::setupAp")); |
|
||||
|
|
||||
DBGPRINTLN(F("\n---------\nAhoyDTU Info:")); |
|
||||
DBGPRINT(F("Version: ")); |
|
||||
DBGPRINT(String(VERSION_MAJOR)); |
|
||||
DBGPRINT(F(".")); |
|
||||
DBGPRINT(String(VERSION_MINOR)); |
|
||||
DBGPRINT(F(".")); |
|
||||
DBGPRINTLN(String(VERSION_PATCH)); |
|
||||
DBGPRINT(F("Github Hash: ")); |
|
||||
DBGPRINTLN(String(AUTO_GIT_HASH)); |
|
||||
|
|
||||
DBGPRINT(F("\n---------\nAP MODE\nSSID: ")); |
|
||||
DBGPRINTLN(WIFI_AP_SSID); |
|
||||
DBGPRINT(F("PWD: ")); |
|
||||
DBGPRINTLN(mConfig->sys.apPwd); |
|
||||
DBGPRINT(F("IP Address: http://")); |
|
||||
DBGPRINTLN(mApIp.toString()); |
|
||||
DBGPRINTLN(F("---------\n")); |
|
||||
|
|
||||
if(String(mConfig->sys.deviceName) != "") |
|
||||
WiFi.hostname(mConfig->sys.deviceName); |
|
||||
|
|
||||
WiFi.mode(WIFI_AP_STA); |
|
||||
WiFi.softAPConfig(mApIp, mApIp, IPAddress(255, 255, 255, 0)); |
|
||||
WiFi.softAP(WIFI_AP_SSID, mConfig->sys.apPwd); |
|
||||
} |
|
||||
|
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoywifi::setupStation(void) { |
|
||||
DPRINTLN(DBG_VERBOSE, F("wifi::setupStation")); |
|
||||
if(mConfig->sys.ip.ip[0] != 0) { |
|
||||
IPAddress ip(mConfig->sys.ip.ip); |
|
||||
IPAddress mask(mConfig->sys.ip.mask); |
|
||||
IPAddress dns1(mConfig->sys.ip.dns1); |
|
||||
IPAddress dns2(mConfig->sys.ip.dns2); |
|
||||
IPAddress gateway(mConfig->sys.ip.gateway); |
|
||||
if(!WiFi.config(ip, gateway, mask, dns1, dns2)) |
|
||||
DPRINTLN(DBG_ERROR, F("failed to set static IP!")); |
|
||||
} |
|
||||
mBSSIDList.clear(); |
|
||||
if(String(mConfig->sys.deviceName) != "") |
|
||||
WiFi.hostname(mConfig->sys.deviceName); |
|
||||
WiFi.mode(WIFI_AP_STA); |
|
||||
|
|
||||
DBGPRINT(F("connect to network '")); |
|
||||
DBGPRINT(mConfig->sys.stationSsid); |
|
||||
DBGPRINTLN(F("' ...")); |
|
||||
} |
|
||||
|
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
bool ahoywifi::getNtpTime(void) { |
|
||||
if(IN_STA_MODE != mStaConn) |
|
||||
return false; |
|
||||
|
|
||||
IPAddress timeServer; |
|
||||
uint8_t buf[NTP_PACKET_SIZE]; |
|
||||
uint8_t retry = 0; |
|
||||
|
|
||||
if (WiFi.hostByName(mConfig->ntp.addr, timeServer) != 1) |
|
||||
return false; |
|
||||
|
|
||||
mUdp.begin(mConfig->ntp.port); |
|
||||
sendNTPpacket(timeServer); |
|
||||
|
|
||||
while(retry++ < 5) { |
|
||||
int wait = 150; |
|
||||
while(--wait) { |
|
||||
if(NTP_PACKET_SIZE <= mUdp.parsePacket()) { |
|
||||
uint64_t secsSince1900; |
|
||||
mUdp.read(buf, NTP_PACKET_SIZE); |
|
||||
secsSince1900 = ((uint64_t)buf[40] << 24); |
|
||||
secsSince1900 |= (buf[41] << 16); |
|
||||
secsSince1900 |= (buf[42] << 8); |
|
||||
secsSince1900 |= (buf[43] ); |
|
||||
|
|
||||
*mUtcTimestamp = secsSince1900 - 2208988800UL; // UTC time
|
|
||||
DPRINTLN(DBG_INFO, "[NTP]: " + ah::getDateTimeStr(*mUtcTimestamp) + " UTC"); |
|
||||
return true; |
|
||||
} else |
|
||||
delay(10); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
DPRINTLN(DBG_INFO, F("[NTP]: getNtpTime failed")); |
|
||||
return false; |
|
||||
} |
|
||||
|
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoywifi::sendNTPpacket(IPAddress& address) { |
|
||||
//DPRINTLN(DBG_VERBOSE, F("wifi::sendNTPpacket"));
|
|
||||
uint8_t buf[NTP_PACKET_SIZE] = {0}; |
|
||||
|
|
||||
buf[0] = B11100011; // LI, Version, Mode
|
|
||||
buf[1] = 0; // Stratum
|
|
||||
buf[2] = 6; // Max Interval between messages in seconds
|
|
||||
buf[3] = 0xEC; // Clock Precision
|
|
||||
// bytes 4 - 11 are for Root Delay and Dispersion and were set to 0 by memset
|
|
||||
buf[12] = 49; // four-byte reference ID identifying
|
|
||||
buf[13] = 0x4E; |
|
||||
buf[14] = 49; |
|
||||
buf[15] = 52; |
|
||||
|
|
||||
mUdp.beginPacket(address, 123); // NTP request, port 123
|
|
||||
mUdp.write(buf, NTP_PACKET_SIZE); |
|
||||
mUdp.endPacket(); |
|
||||
} |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoywifi::sortRSSI(int *sort, int n) { |
|
||||
for (int i = 0; i < n; i++) |
|
||||
sort[i] = i; |
|
||||
for (int i = 0; i < n; i++) |
|
||||
for (int j = i + 1; j < n; j++) |
|
||||
if (WiFi.RSSI(sort[j]) > WiFi.RSSI(sort[i])) |
|
||||
std::swap(sort[i], sort[j]); |
|
||||
} |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoywifi::scanAvailNetworks(void) { |
|
||||
if(!mScanActive) { |
|
||||
mScanActive = true; |
|
||||
if(WIFI_AP == WiFi.getMode()) |
|
||||
WiFi.mode(WIFI_AP_STA); |
|
||||
WiFi.scanNetworks(true); |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
bool ahoywifi::getAvailNetworks(JsonObject obj) { |
|
||||
JsonArray nets = obj.createNestedArray("networks"); |
|
||||
|
|
||||
int n = WiFi.scanComplete(); |
|
||||
if (n < 0) |
|
||||
return false; |
|
||||
if(n > 0) { |
|
||||
int sort[n]; |
|
||||
sortRSSI(&sort[0], n); |
|
||||
for (int i = 0; i < n; ++i) { |
|
||||
nets[i]["ssid"] = WiFi.SSID(sort[i]); |
|
||||
nets[i]["rssi"] = WiFi.RSSI(sort[i]); |
|
||||
} |
|
||||
} |
|
||||
mScanActive = false; |
|
||||
WiFi.scanDelete(); |
|
||||
if(mStaConn == IN_AP_MODE) |
|
||||
WiFi.mode(WIFI_AP); |
|
||||
|
|
||||
return true; |
|
||||
} |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
bool ahoywifi::getBSSIDs() { |
|
||||
bool result = false; |
|
||||
int n = WiFi.scanComplete(); |
|
||||
if (n < 0) { |
|
||||
if (++mScanCnt < 20) |
|
||||
return false; |
|
||||
} |
|
||||
if(n > 0) { |
|
||||
mBSSIDList.clear(); |
|
||||
int sort[n]; |
|
||||
sortRSSI(&sort[0], n); |
|
||||
for (int i = 0; i < n; i++) { |
|
||||
DBGPRINT("BSSID " + String(i) + ":"); |
|
||||
uint8_t *bssid = WiFi.BSSID(sort[i]); |
|
||||
for (int j = 0; j < 6; j++){ |
|
||||
DBGPRINT(" " + String(bssid[j], HEX)); |
|
||||
mBSSIDList.push_back(bssid[j]); |
|
||||
} |
|
||||
DBGPRINTLN(""); |
|
||||
} |
|
||||
result = true; |
|
||||
} |
|
||||
mScanActive = false; |
|
||||
WiFi.scanDelete(); |
|
||||
return result; |
|
||||
} |
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoywifi::connectionEvent(WiFiStatus_t status) { |
|
||||
DPRINTLN(DBG_INFO, "connectionEvent"); |
|
||||
|
|
||||
switch(status) { |
|
||||
case CONNECTED: |
|
||||
if(mStaConn != CONNECTED) { |
|
||||
mStaConn = CONNECTED; |
|
||||
mGotDisconnect = false; |
|
||||
DBGPRINTLN(F("\n[WiFi] Connected")); |
|
||||
} |
|
||||
break; |
|
||||
|
|
||||
case GOT_IP: |
|
||||
mStaConn = GOT_IP; |
|
||||
break; |
|
||||
|
|
||||
case DISCONNECTED: |
|
||||
mGotDisconnect = true; |
|
||||
break; |
|
||||
|
|
||||
default: |
|
||||
break; |
|
||||
} |
|
||||
} |
|
||||
|
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
#if defined(ESP8266) |
|
||||
//-------------------------------------------------------------------------
|
|
||||
void ahoywifi::onConnect(const WiFiEventStationModeConnected& event) { |
|
||||
connectionEvent(CONNECTED); |
|
||||
} |
|
||||
|
|
||||
//-------------------------------------------------------------------------
|
|
||||
void ahoywifi::onGotIP(const WiFiEventStationModeGotIP& event) { |
|
||||
connectionEvent(GOT_IP); |
|
||||
} |
|
||||
|
|
||||
//-------------------------------------------------------------------------
|
|
||||
void ahoywifi::onDisconnect(const WiFiEventStationModeDisconnected& event) { |
|
||||
connectionEvent(DISCONNECTED); |
|
||||
} |
|
||||
|
|
||||
#else |
|
||||
//-------------------------------------------------------------------------
|
|
||||
void ahoywifi::onWiFiEvent(WiFiEvent_t event) { |
|
||||
DBGPRINT(F("Wifi event: ")); |
|
||||
DBGPRINTLN(String(event)); |
|
||||
|
|
||||
switch(event) { |
|
||||
case SYSTEM_EVENT_STA_CONNECTED: |
|
||||
connectionEvent(CONNECTED); |
|
||||
break; |
|
||||
|
|
||||
case SYSTEM_EVENT_STA_GOT_IP: |
|
||||
connectionEvent(GOT_IP); |
|
||||
break; |
|
||||
|
|
||||
case SYSTEM_EVENT_STA_DISCONNECTED: |
|
||||
connectionEvent(DISCONNECTED); |
|
||||
break; |
|
||||
|
|
||||
default: |
|
||||
break; |
|
||||
} |
|
||||
} |
|
||||
#endif |
|
||||
|
|
||||
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
void ahoywifi::welcome(String ip, String mode) { |
|
||||
DBGPRINTLN(F("\n\n--------------------------------")); |
|
||||
DBGPRINTLN(F("Welcome to AHOY!")); |
|
||||
DBGPRINT(F("\npoint your browser to http://")); |
|
||||
DBGPRINT(ip); |
|
||||
DBGPRINTLN(mode); |
|
||||
DBGPRINTLN(F("to configure your device")); |
|
||||
DBGPRINTLN(F("--------------------------------\n")); |
|
||||
} |
|
||||
|
|
||||
#endif /* !defined(ETHERNET) */ |
|
@ -1,97 +0,0 @@ |
|||||
//------------------------------------//-----------------------------------------------------------------------------
|
|
||||
// 2024 Ahoy, https://github.com/lumpapu/ahoy
|
|
||||
// Creative Commons - http://creativecommons.org/licenses/by-nc-sa/4.0/deed
|
|
||||
//-----------------------------------------------------------------------------
|
|
||||
|
|
||||
#if !defined(ETHERNET) |
|
||||
#ifndef __AHOYWIFI_H__ |
|
||||
#define __AHOYWIFI_H__ |
|
||||
|
|
||||
#include "../utils/dbg.h" |
|
||||
#include <Arduino.h> |
|
||||
#include <WiFiUdp.h> |
|
||||
#include <DNSServer.h> |
|
||||
#include "ESPAsyncWebServer.h" |
|
||||
|
|
||||
#include "../config/settings.h" |
|
||||
|
|
||||
class app; |
|
||||
|
|
||||
class ahoywifi { |
|
||||
public: |
|
||||
typedef std::function<void(bool)> appWifiCb; |
|
||||
|
|
||||
ahoywifi(); |
|
||||
|
|
||||
|
|
||||
void setup(settings_t *config, uint32_t *utcTimestamp, appWifiCb cb); |
|
||||
void tickWifiLoop(void); |
|
||||
bool getNtpTime(void); |
|
||||
void scanAvailNetworks(void); |
|
||||
bool getAvailNetworks(JsonObject obj); |
|
||||
void setStopApAllowedMode(bool allowed) { |
|
||||
mStopApAllowed = allowed; |
|
||||
} |
|
||||
String getStationIp(void) { |
|
||||
return WiFi.localIP().toString(); |
|
||||
} |
|
||||
void setupStation(void); |
|
||||
|
|
||||
bool getWasInCh12to14() const { |
|
||||
return mWasInCh12to14; |
|
||||
} |
|
||||
|
|
||||
private: |
|
||||
typedef enum WiFiStatus { |
|
||||
DISCONNECTED = 0, |
|
||||
SCAN_READY, |
|
||||
CONNECTING, |
|
||||
CONNECTED, |
|
||||
IN_AP_MODE, |
|
||||
GOT_IP, |
|
||||
IN_STA_MODE, |
|
||||
RESET |
|
||||
} WiFiStatus_t; |
|
||||
|
|
||||
void setupWifi(bool startAP); |
|
||||
void setupAp(void); |
|
||||
void sendNTPpacket(IPAddress& address); |
|
||||
void sortRSSI(int *sort, int n); |
|
||||
bool getBSSIDs(void); |
|
||||
void connectionEvent(WiFiStatus_t status); |
|
||||
bool isTimeout(uint8_t timeout) { return (mCnt % timeout) == 0; } |
|
||||
|
|
||||
#if defined(ESP8266) |
|
||||
void onConnect(const WiFiEventStationModeConnected& event); |
|
||||
void onGotIP(const WiFiEventStationModeGotIP& event); |
|
||||
void onDisconnect(const WiFiEventStationModeDisconnected& event); |
|
||||
#else |
|
||||
void onWiFiEvent(WiFiEvent_t event); |
|
||||
#endif |
|
||||
void welcome(String ip, String mode); |
|
||||
|
|
||||
|
|
||||
settings_t *mConfig = nullptr; |
|
||||
appWifiCb mAppWifiCb; |
|
||||
|
|
||||
DNSServer mDns; |
|
||||
IPAddress mApIp; |
|
||||
WiFiUDP mUdp; // for time server
|
|
||||
#if defined(ESP8266) |
|
||||
WiFiEventHandler wifiConnectHandler, wifiDisconnectHandler, wifiGotIPHandler; |
|
||||
#endif |
|
||||
|
|
||||
WiFiStatus_t mStaConn = DISCONNECTED; |
|
||||
uint8_t mCnt = 0; |
|
||||
uint32_t *mUtcTimestamp = nullptr; |
|
||||
|
|
||||
uint8_t mScanCnt = 0; |
|
||||
bool mScanActive = false; |
|
||||
bool mGotDisconnect = false; |
|
||||
std::list<uint8_t> mBSSIDList; |
|
||||
bool mStopApAllowed = false; |
|
||||
bool mWasInCh12to14 = false; |
|
||||
}; |
|
||||
|
|
||||
#endif /*__AHOYWIFI_H__*/ |
|
||||
#endif /* !defined(ETHERNET) */ |
|
@ -0,0 +1,466 @@ |
|||||
|
[ |
||||
|
{ |
||||
|
"id": "67bced2c4e728783", |
||||
|
"type": "mqtt in", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "", |
||||
|
"topic": "hoymiles/+", |
||||
|
"qos": "0", |
||||
|
"datatype": "auto-detect", |
||||
|
"broker": "319864a4e0fd913f", |
||||
|
"nl": false, |
||||
|
"rap": true, |
||||
|
"rh": 0, |
||||
|
"inputs": 0, |
||||
|
"x": 80, |
||||
|
"y": 2100, |
||||
|
"wires": [ |
||||
|
[ |
||||
|
"a55632ad0dff0b69" |
||||
|
] |
||||
|
] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "a7f0d307d7cf77e2", |
||||
|
"type": "mqtt in", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "", |
||||
|
"topic": "hoymiles/X/#", |
||||
|
"qos": "0", |
||||
|
"datatype": "auto-detect", |
||||
|
"broker": "319864a4e0fd913f", |
||||
|
"nl": false, |
||||
|
"rap": true, |
||||
|
"rh": 0, |
||||
|
"inputs": 0, |
||||
|
"x": 90, |
||||
|
"y": 2260, |
||||
|
"wires": [ |
||||
|
[ |
||||
|
"7e17e5a3f4df3011", |
||||
|
"1a8cca488d53394a" |
||||
|
] |
||||
|
] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "7e17e5a3f4df3011", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "Inverter X", |
||||
|
"active": false, |
||||
|
"tosidebar": true, |
||||
|
"console": false, |
||||
|
"tostatus": false, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "", |
||||
|
"statusType": "auto", |
||||
|
"x": 340, |
||||
|
"y": 2260, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "fb7357db50501627", |
||||
|
"type": "change", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "Tags setzen", |
||||
|
"rules": [ |
||||
|
{ |
||||
|
"t": "set", |
||||
|
"p": "payload", |
||||
|
"pt": "msg", |
||||
|
"to": "(\t $a := $split(topic, '/');\t [\t payload,\t {\t \"device\":$a[0],\t \"name\":$a[1],\t \"channel\":$a[2]\t }\t ]\t)\t", |
||||
|
"tot": "jsonata" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "delete", |
||||
|
"p": "topic", |
||||
|
"pt": "msg" |
||||
|
} |
||||
|
], |
||||
|
"action": "", |
||||
|
"property": "", |
||||
|
"from": "", |
||||
|
"to": "", |
||||
|
"reg": false, |
||||
|
"x": 610, |
||||
|
"y": 2360, |
||||
|
"wires": [ |
||||
|
[ |
||||
|
"91a4607dfda84b67" |
||||
|
] |
||||
|
] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "670eb9fbb5c31b2c", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "InfluxDB", |
||||
|
"active": false, |
||||
|
"tosidebar": true, |
||||
|
"console": false, |
||||
|
"tostatus": false, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "", |
||||
|
"statusType": "auto", |
||||
|
"x": 940, |
||||
|
"y": 2360, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "1a8cca488d53394a", |
||||
|
"type": "switch", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "", |
||||
|
"property": "$split(topic, '/')[2]", |
||||
|
"propertyType": "jsonata", |
||||
|
"rules": [ |
||||
|
{ |
||||
|
"t": "eq", |
||||
|
"v": "available", |
||||
|
"vt": "str" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "eq", |
||||
|
"v": "last_success", |
||||
|
"vt": "str" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "regex", |
||||
|
"v": "(ch[0-6])\\b", |
||||
|
"vt": "str", |
||||
|
"case": false |
||||
|
}, |
||||
|
{ |
||||
|
"t": "eq", |
||||
|
"v": "radio_stat", |
||||
|
"vt": "str" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "eq", |
||||
|
"v": "firmware", |
||||
|
"vt": "str" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "eq", |
||||
|
"v": "hardware", |
||||
|
"vt": "str" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "eq", |
||||
|
"v": "alarm", |
||||
|
"vt": "str" |
||||
|
} |
||||
|
], |
||||
|
"checkall": "true", |
||||
|
"repair": false, |
||||
|
"outputs": 7, |
||||
|
"x": 330, |
||||
|
"y": 2380, |
||||
|
"wires": [ |
||||
|
[ |
||||
|
"845aeb93e39092c5" |
||||
|
], |
||||
|
[ |
||||
|
"241a8e70e9fde93c" |
||||
|
], |
||||
|
[ |
||||
|
"fb7357db50501627" |
||||
|
], |
||||
|
[ |
||||
|
"9d38f021308664c1" |
||||
|
], |
||||
|
[ |
||||
|
"a508355f0cc87966" |
||||
|
], |
||||
|
[ |
||||
|
"d2c9aa1a8978aca6" |
||||
|
], |
||||
|
[ |
||||
|
"b27032beb597d5a7" |
||||
|
] |
||||
|
] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "845aeb93e39092c5", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "available", |
||||
|
"active": true, |
||||
|
"tosidebar": false, |
||||
|
"console": false, |
||||
|
"tostatus": true, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "payload", |
||||
|
"statusType": "auto", |
||||
|
"x": 600, |
||||
|
"y": 2240, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "241a8e70e9fde93c", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "last_success", |
||||
|
"active": true, |
||||
|
"tosidebar": false, |
||||
|
"console": false, |
||||
|
"tostatus": true, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "payload", |
||||
|
"statusType": "auto", |
||||
|
"x": 610, |
||||
|
"y": 2300, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "9d38f021308664c1", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "radio_stat", |
||||
|
"active": false, |
||||
|
"tosidebar": true, |
||||
|
"console": false, |
||||
|
"tostatus": false, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "", |
||||
|
"statusType": "auto", |
||||
|
"x": 600, |
||||
|
"y": 2400, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "a508355f0cc87966", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "firmware", |
||||
|
"active": false, |
||||
|
"tosidebar": true, |
||||
|
"console": false, |
||||
|
"tostatus": false, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "", |
||||
|
"statusType": "auto", |
||||
|
"x": 600, |
||||
|
"y": 2440, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "d2c9aa1a8978aca6", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "hardware", |
||||
|
"active": false, |
||||
|
"tosidebar": true, |
||||
|
"console": false, |
||||
|
"tostatus": false, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "", |
||||
|
"statusType": "auto", |
||||
|
"x": 600, |
||||
|
"y": 2480, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "b27032beb597d5a7", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "alarm", |
||||
|
"active": false, |
||||
|
"tosidebar": true, |
||||
|
"console": false, |
||||
|
"tostatus": false, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "", |
||||
|
"statusType": "auto", |
||||
|
"x": 590, |
||||
|
"y": 2520, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "d814738cf55ad663", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "total", |
||||
|
"active": false, |
||||
|
"tosidebar": true, |
||||
|
"console": false, |
||||
|
"tostatus": false, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "", |
||||
|
"statusType": "auto", |
||||
|
"x": 590, |
||||
|
"y": 2160, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "a55632ad0dff0b69", |
||||
|
"type": "switch", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "", |
||||
|
"property": "$split(topic, '/')[1]", |
||||
|
"propertyType": "jsonata", |
||||
|
"rules": [ |
||||
|
{ |
||||
|
"t": "eq", |
||||
|
"v": "uptime", |
||||
|
"vt": "str" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "eq", |
||||
|
"v": "wifi_rssi", |
||||
|
"vt": "str" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "eq", |
||||
|
"v": "status", |
||||
|
"vt": "str" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "eq", |
||||
|
"v": "total", |
||||
|
"vt": "str" |
||||
|
} |
||||
|
], |
||||
|
"checkall": "true", |
||||
|
"repair": false, |
||||
|
"outputs": 4, |
||||
|
"x": 330, |
||||
|
"y": 2100, |
||||
|
"wires": [ |
||||
|
[ |
||||
|
"1fbb0674d2576ee7" |
||||
|
], |
||||
|
[ |
||||
|
"e6be1c98ac55f511" |
||||
|
], |
||||
|
[ |
||||
|
"f9c2d3b30e34fdda" |
||||
|
], |
||||
|
[ |
||||
|
"d814738cf55ad663" |
||||
|
] |
||||
|
] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "f9c2d3b30e34fdda", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "status", |
||||
|
"active": false, |
||||
|
"tosidebar": false, |
||||
|
"console": false, |
||||
|
"tostatus": true, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "payload", |
||||
|
"statusType": "auto", |
||||
|
"x": 590, |
||||
|
"y": 2100, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "e6be1c98ac55f511", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "wifi_rssi", |
||||
|
"active": false, |
||||
|
"tosidebar": false, |
||||
|
"console": false, |
||||
|
"tostatus": true, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "payload", |
||||
|
"statusType": "auto", |
||||
|
"x": 600, |
||||
|
"y": 2040, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "1fbb0674d2576ee7", |
||||
|
"type": "debug", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "uptime", |
||||
|
"active": false, |
||||
|
"tosidebar": false, |
||||
|
"console": false, |
||||
|
"tostatus": true, |
||||
|
"complete": "payload", |
||||
|
"targetType": "msg", |
||||
|
"statusVal": "payload", |
||||
|
"statusType": "auto", |
||||
|
"x": 590, |
||||
|
"y": 1980, |
||||
|
"wires": [] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "91a4607dfda84b67", |
||||
|
"type": "change", |
||||
|
"z": "5de5756d190f9086", |
||||
|
"name": "Lösche", |
||||
|
"rules": [ |
||||
|
{ |
||||
|
"t": "delete", |
||||
|
"p": "payload[0].YieldDay", |
||||
|
"pt": "msg" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "delete", |
||||
|
"p": "payload[0].MaxPower", |
||||
|
"pt": "msg" |
||||
|
}, |
||||
|
{ |
||||
|
"t": "delete", |
||||
|
"p": "payload[0].ALARM_MES_ID", |
||||
|
"pt": "msg" |
||||
|
} |
||||
|
], |
||||
|
"action": "", |
||||
|
"property": "", |
||||
|
"from": "", |
||||
|
"to": "", |
||||
|
"reg": false, |
||||
|
"x": 780, |
||||
|
"y": 2360, |
||||
|
"wires": [ |
||||
|
[ |
||||
|
"670eb9fbb5c31b2c" |
||||
|
] |
||||
|
] |
||||
|
}, |
||||
|
{ |
||||
|
"id": "319864a4e0fd913f", |
||||
|
"type": "mqtt-broker", |
||||
|
"name": "broker", |
||||
|
"broker": "localhost", |
||||
|
"port": "1883", |
||||
|
"clientid": "", |
||||
|
"autoConnect": true, |
||||
|
"usetls": false, |
||||
|
"protocolVersion": "4", |
||||
|
"keepalive": "60", |
||||
|
"cleansession": true, |
||||
|
"birthTopic": "", |
||||
|
"birthQos": "0", |
||||
|
"birthPayload": "", |
||||
|
"birthMsg": {}, |
||||
|
"closeTopic": "", |
||||
|
"closeQos": "0", |
||||
|
"closePayload": "", |
||||
|
"closeMsg": {}, |
||||
|
"willTopic": "", |
||||
|
"willQos": "0", |
||||
|
"willPayload": "", |
||||
|
"willMsg": {}, |
||||
|
"userProps": "", |
||||
|
"sessionExpiry": "" |
||||
|
} |
||||
|
] |
Loading…
Reference in new issue