diff --git a/ESP_Code/Counter.h b/ESP_Code/Counter.h index 58a4a0e6..172a99f7 100644 --- a/ESP_Code/Counter.h +++ b/ESP_Code/Counter.h @@ -1,53 +1,53 @@ -#ifndef _COUNTER_H_ -#define _COUNTER_H_ - -#include -#include - -template -class Counter { - -public: - Counter() { reset(); } - - void reset() { - memset(buffer, '0', max_digits); - buffer[max_digits] = '\0'; - val = 0; - len = 1; - } - - inline Counter &operator++() { - inc_string(buffer + max_digits - 1); - ++val; - return *this; - } - - inline operator unsigned int() const { return val; } - inline const char *c_str() const { return buffer + max_digits - len; } - inline size_t strlen() const { return len; } - -protected: - inline void inc_string(char *c) { - // In theory, the line below should be uncommented to avoid writing outside the buffer. In practice however, - // with max_digits set to 10 or more, we can fit all possible unsigned 32-bit integers in the buffer. - // The check is skipped to gain a small extra speed improvement. - // if (c >= buffer) return; - - if (*c < '9') { - *c += 1; - } - else { - *c = '0'; - inc_string(c - 1); - len = max(max_digits - (c - buffer) + 1, len); - } - } - -protected: - char buffer[max_digits + 1]; - unsigned int val; - size_t len; -}; - -#endif +#ifndef _COUNTER_H_ +#define _COUNTER_H_ + +#include +#include + +template +class Counter { + +public: + Counter() { reset(); } + + void reset() { + memset(buffer, '0', max_digits); + buffer[max_digits] = '\0'; + val = 0; + len = 1; + } + + inline Counter &operator++() { + inc_string(buffer + max_digits - 1); + ++val; + return *this; + } + + inline operator unsigned int() const { return val; } + inline const char *c_str() const { return buffer + max_digits - len; } + inline size_t strlen() const { return len; } + +protected: + inline void inc_string(char *c) { + // In theory, the line below should be uncommented to avoid writing outside the buffer. In practice however, + // with max_digits set to 10 or more, we can fit all possible unsigned 32-bit integers in the buffer. + // The check is skipped to gain a small extra speed improvement. + // if (c >= buffer) return; + + if (*c < '9') { + *c += 1; + } + else { + *c = '0'; + inc_string(c - 1); + len = max(max_digits - (c - buffer) + 1, len); + } + } + +protected: + char buffer[max_digits + 1]; + unsigned int val; + size_t len; +}; + +#endif diff --git a/ESP_Code/DSHA1.h b/ESP_Code/DSHA1.h index 58dc41b2..2593ecd9 100644 --- a/ESP_Code/DSHA1.h +++ b/ESP_Code/DSHA1.h @@ -1,201 +1,201 @@ -#ifndef DSHA1_H -#define DSHA1_H - -#include - -class DSHA1 { - -public: - static const size_t OUTPUT_SIZE = 20; - - DSHA1() { - initialize(s); - } - - DSHA1 &write(const unsigned char *data, size_t len) { - size_t bufsize = bytes % 64; - if (bufsize && bufsize + len >= 64) { - memcpy(buf + bufsize, data, 64 - bufsize); - bytes += 64 - bufsize; - data += 64 - bufsize; - transform(s, buf); - bufsize = 0; - } - while (len >= 64) { - transform(s, data); - bytes += 64; - data += 64; - len -= 64; - } - if (len > 0) { - memcpy(buf + bufsize, data, len); - bytes += len; - } - return *this; - } - - void finalize(unsigned char hash[OUTPUT_SIZE]) { - const unsigned char pad[64] = {0x80}; - unsigned char sizedesc[8]; - writeBE64(sizedesc, bytes << 3); - write(pad, 1 + ((119 - (bytes % 64)) % 64)); - write(sizedesc, 8); - writeBE32(hash, s[0]); - writeBE32(hash + 4, s[1]); - writeBE32(hash + 8, s[2]); - writeBE32(hash + 12, s[3]); - writeBE32(hash + 16, s[4]); - } - - DSHA1 &reset() { - bytes = 0; - initialize(s); - return *this; - } - - // Warmup the cache and get a boost in performance - DSHA1 &warmup() { - uint8_t warmup[20]; - this->write((uint8_t *)"warmupwarmupwa", 20).finalize(warmup); - return *this; - } - -private: - uint32_t s[5]; - unsigned char buf[64]; - uint64_t bytes; - - const uint32_t k1 = 0x5A827999ul; - const uint32_t k2 = 0x6ED9EBA1ul; - const uint32_t k3 = 0x8F1BBCDCul; - const uint32_t k4 = 0xCA62C1D6ul; - - uint32_t inline f1(uint32_t b, uint32_t c, uint32_t d) { return d ^ (b & (c ^ d)); } - uint32_t inline f2(uint32_t b, uint32_t c, uint32_t d) { return b ^ c ^ d; } - uint32_t inline f3(uint32_t b, uint32_t c, uint32_t d) { return (b & c) | (d & (b | c)); } - - uint32_t inline left(uint32_t x) { return (x << 1) | (x >> 31); } - - void inline Round(uint32_t a, uint32_t &b, uint32_t c, uint32_t d, uint32_t &e, - uint32_t f, uint32_t k, uint32_t w) { - e += ((a << 5) | (a >> 27)) + f + k + w; - b = (b << 30) | (b >> 2); - } - - void initialize(uint32_t s[5]) { - s[0] = 0x67452301ul; - s[1] = 0xEFCDAB89ul; - s[2] = 0x98BADCFEul; - s[3] = 0x10325476ul; - s[4] = 0xC3D2E1F0ul; - } - - void transform(uint32_t *s, const unsigned char *chunk) { - uint32_t a = s[0], b = s[1], c = s[2], d = s[3], e = s[4]; - uint32_t w0, w1, w2, w3, w4, w5, w6, w7, w8, w9, w10, w11, w12, w13, w14, w15; - - Round(a, b, c, d, e, f1(b, c, d), k1, w0 = readBE32(chunk + 0)); - Round(e, a, b, c, d, f1(a, b, c), k1, w1 = readBE32(chunk + 4)); - Round(d, e, a, b, c, f1(e, a, b), k1, w2 = readBE32(chunk + 8)); - Round(c, d, e, a, b, f1(d, e, a), k1, w3 = readBE32(chunk + 12)); - Round(b, c, d, e, a, f1(c, d, e), k1, w4 = readBE32(chunk + 16)); - Round(a, b, c, d, e, f1(b, c, d), k1, w5 = readBE32(chunk + 20)); - Round(e, a, b, c, d, f1(a, b, c), k1, w6 = readBE32(chunk + 24)); - Round(d, e, a, b, c, f1(e, a, b), k1, w7 = readBE32(chunk + 28)); - Round(c, d, e, a, b, f1(d, e, a), k1, w8 = readBE32(chunk + 32)); - Round(b, c, d, e, a, f1(c, d, e), k1, w9 = readBE32(chunk + 36)); - Round(a, b, c, d, e, f1(b, c, d), k1, w10 = readBE32(chunk + 40)); - Round(e, a, b, c, d, f1(a, b, c), k1, w11 = readBE32(chunk + 44)); - Round(d, e, a, b, c, f1(e, a, b), k1, w12 = readBE32(chunk + 48)); - Round(c, d, e, a, b, f1(d, e, a), k1, w13 = readBE32(chunk + 52)); - Round(b, c, d, e, a, f1(c, d, e), k1, w14 = readBE32(chunk + 56)); - Round(a, b, c, d, e, f1(b, c, d), k1, w15 = readBE32(chunk + 60)); - - Round(e, a, b, c, d, f1(a, b, c), k1, w0 = left(w0 ^ w13 ^ w8 ^ w2)); - Round(d, e, a, b, c, f1(e, a, b), k1, w1 = left(w1 ^ w14 ^ w9 ^ w3)); - Round(c, d, e, a, b, f1(d, e, a), k1, w2 = left(w2 ^ w15 ^ w10 ^ w4)); - Round(b, c, d, e, a, f1(c, d, e), k1, w3 = left(w3 ^ w0 ^ w11 ^ w5)); - Round(a, b, c, d, e, f2(b, c, d), k2, w4 = left(w4 ^ w1 ^ w12 ^ w6)); - Round(e, a, b, c, d, f2(a, b, c), k2, w5 = left(w5 ^ w2 ^ w13 ^ w7)); - Round(d, e, a, b, c, f2(e, a, b), k2, w6 = left(w6 ^ w3 ^ w14 ^ w8)); - Round(c, d, e, a, b, f2(d, e, a), k2, w7 = left(w7 ^ w4 ^ w15 ^ w9)); - Round(b, c, d, e, a, f2(c, d, e), k2, w8 = left(w8 ^ w5 ^ w0 ^ w10)); - Round(a, b, c, d, e, f2(b, c, d), k2, w9 = left(w9 ^ w6 ^ w1 ^ w11)); - Round(e, a, b, c, d, f2(a, b, c), k2, w10 = left(w10 ^ w7 ^ w2 ^ w12)); - Round(d, e, a, b, c, f2(e, a, b), k2, w11 = left(w11 ^ w8 ^ w3 ^ w13)); - Round(c, d, e, a, b, f2(d, e, a), k2, w12 = left(w12 ^ w9 ^ w4 ^ w14)); - Round(b, c, d, e, a, f2(c, d, e), k2, w13 = left(w13 ^ w10 ^ w5 ^ w15)); - Round(a, b, c, d, e, f2(b, c, d), k2, w14 = left(w14 ^ w11 ^ w6 ^ w0)); - Round(e, a, b, c, d, f2(a, b, c), k2, w15 = left(w15 ^ w12 ^ w7 ^ w1)); - - Round(d, e, a, b, c, f2(e, a, b), k2, w0 = left(w0 ^ w13 ^ w8 ^ w2)); - Round(c, d, e, a, b, f2(d, e, a), k2, w1 = left(w1 ^ w14 ^ w9 ^ w3)); - Round(b, c, d, e, a, f2(c, d, e), k2, w2 = left(w2 ^ w15 ^ w10 ^ w4)); - Round(a, b, c, d, e, f2(b, c, d), k2, w3 = left(w3 ^ w0 ^ w11 ^ w5)); - Round(e, a, b, c, d, f2(a, b, c), k2, w4 = left(w4 ^ w1 ^ w12 ^ w6)); - Round(d, e, a, b, c, f2(e, a, b), k2, w5 = left(w5 ^ w2 ^ w13 ^ w7)); - Round(c, d, e, a, b, f2(d, e, a), k2, w6 = left(w6 ^ w3 ^ w14 ^ w8)); - Round(b, c, d, e, a, f2(c, d, e), k2, w7 = left(w7 ^ w4 ^ w15 ^ w9)); - Round(a, b, c, d, e, f3(b, c, d), k3, w8 = left(w8 ^ w5 ^ w0 ^ w10)); - Round(e, a, b, c, d, f3(a, b, c), k3, w9 = left(w9 ^ w6 ^ w1 ^ w11)); - Round(d, e, a, b, c, f3(e, a, b), k3, w10 = left(w10 ^ w7 ^ w2 ^ w12)); - Round(c, d, e, a, b, f3(d, e, a), k3, w11 = left(w11 ^ w8 ^ w3 ^ w13)); - Round(b, c, d, e, a, f3(c, d, e), k3, w12 = left(w12 ^ w9 ^ w4 ^ w14)); - Round(a, b, c, d, e, f3(b, c, d), k3, w13 = left(w13 ^ w10 ^ w5 ^ w15)); - Round(e, a, b, c, d, f3(a, b, c), k3, w14 = left(w14 ^ w11 ^ w6 ^ w0)); - Round(d, e, a, b, c, f3(e, a, b), k3, w15 = left(w15 ^ w12 ^ w7 ^ w1)); - - Round(c, d, e, a, b, f3(d, e, a), k3, w0 = left(w0 ^ w13 ^ w8 ^ w2)); - Round(b, c, d, e, a, f3(c, d, e), k3, w1 = left(w1 ^ w14 ^ w9 ^ w3)); - Round(a, b, c, d, e, f3(b, c, d), k3, w2 = left(w2 ^ w15 ^ w10 ^ w4)); - Round(e, a, b, c, d, f3(a, b, c), k3, w3 = left(w3 ^ w0 ^ w11 ^ w5)); - Round(d, e, a, b, c, f3(e, a, b), k3, w4 = left(w4 ^ w1 ^ w12 ^ w6)); - Round(c, d, e, a, b, f3(d, e, a), k3, w5 = left(w5 ^ w2 ^ w13 ^ w7)); - Round(b, c, d, e, a, f3(c, d, e), k3, w6 = left(w6 ^ w3 ^ w14 ^ w8)); - Round(a, b, c, d, e, f3(b, c, d), k3, w7 = left(w7 ^ w4 ^ w15 ^ w9)); - Round(e, a, b, c, d, f3(a, b, c), k3, w8 = left(w8 ^ w5 ^ w0 ^ w10)); - Round(d, e, a, b, c, f3(e, a, b), k3, w9 = left(w9 ^ w6 ^ w1 ^ w11)); - Round(c, d, e, a, b, f3(d, e, a), k3, w10 = left(w10 ^ w7 ^ w2 ^ w12)); - Round(b, c, d, e, a, f3(c, d, e), k3, w11 = left(w11 ^ w8 ^ w3 ^ w13)); - Round(a, b, c, d, e, f2(b, c, d), k4, w12 = left(w12 ^ w9 ^ w4 ^ w14)); - Round(e, a, b, c, d, f2(a, b, c), k4, w13 = left(w13 ^ w10 ^ w5 ^ w15)); - Round(d, e, a, b, c, f2(e, a, b), k4, w14 = left(w14 ^ w11 ^ w6 ^ w0)); - Round(c, d, e, a, b, f2(d, e, a), k4, w15 = left(w15 ^ w12 ^ w7 ^ w1)); - - Round(b, c, d, e, a, f2(c, d, e), k4, w0 = left(w0 ^ w13 ^ w8 ^ w2)); - Round(a, b, c, d, e, f2(b, c, d), k4, w1 = left(w1 ^ w14 ^ w9 ^ w3)); - Round(e, a, b, c, d, f2(a, b, c), k4, w2 = left(w2 ^ w15 ^ w10 ^ w4)); - Round(d, e, a, b, c, f2(e, a, b), k4, w3 = left(w3 ^ w0 ^ w11 ^ w5)); - Round(c, d, e, a, b, f2(d, e, a), k4, w4 = left(w4 ^ w1 ^ w12 ^ w6)); - Round(b, c, d, e, a, f2(c, d, e), k4, w5 = left(w5 ^ w2 ^ w13 ^ w7)); - Round(a, b, c, d, e, f2(b, c, d), k4, w6 = left(w6 ^ w3 ^ w14 ^ w8)); - Round(e, a, b, c, d, f2(a, b, c), k4, w7 = left(w7 ^ w4 ^ w15 ^ w9)); - Round(d, e, a, b, c, f2(e, a, b), k4, w8 = left(w8 ^ w5 ^ w0 ^ w10)); - Round(c, d, e, a, b, f2(d, e, a), k4, w9 = left(w9 ^ w6 ^ w1 ^ w11)); - Round(b, c, d, e, a, f2(c, d, e), k4, w10 = left(w10 ^ w7 ^ w2 ^ w12)); - Round(a, b, c, d, e, f2(b, c, d), k4, w11 = left(w11 ^ w8 ^ w3 ^ w13)); - Round(e, a, b, c, d, f2(a, b, c), k4, w12 = left(w12 ^ w9 ^ w4 ^ w14)); - Round(d, e, a, b, c, f2(e, a, b), k4, left(w13 ^ w10 ^ w5 ^ w15)); - Round(c, d, e, a, b, f2(d, e, a), k4, left(w14 ^ w11 ^ w6 ^ w0)); - Round(b, c, d, e, a, f2(c, d, e), k4, left(w15 ^ w12 ^ w7 ^ w1)); - - s[0] += a; - s[1] += b; - s[2] += c; - s[3] += d; - s[4] += e; - } - - uint32_t static inline readBE32(const unsigned char *ptr) { - return __builtin_bswap32(*(uint32_t *)ptr); - } - - void static inline writeBE32(unsigned char *ptr, uint32_t x) { - *(uint32_t *)ptr = __builtin_bswap32(x); - } - - void static inline writeBE64(unsigned char *ptr, uint64_t x) { - *(uint64_t *)ptr = __builtin_bswap64(x); - } -}; -#endif +#ifndef DSHA1_H +#define DSHA1_H + +#include + +class DSHA1 { + +public: + static const size_t OUTPUT_SIZE = 20; + + DSHA1() { + initialize(s); + } + + DSHA1 &write(const unsigned char *data, size_t len) { + size_t bufsize = bytes % 64; + if (bufsize && bufsize + len >= 64) { + memcpy(buf + bufsize, data, 64 - bufsize); + bytes += 64 - bufsize; + data += 64 - bufsize; + transform(s, buf); + bufsize = 0; + } + while (len >= 64) { + transform(s, data); + bytes += 64; + data += 64; + len -= 64; + } + if (len > 0) { + memcpy(buf + bufsize, data, len); + bytes += len; + } + return *this; + } + + void finalize(unsigned char hash[OUTPUT_SIZE]) { + const unsigned char pad[64] = {0x80}; + unsigned char sizedesc[8]; + writeBE64(sizedesc, bytes << 3); + write(pad, 1 + ((119 - (bytes % 64)) % 64)); + write(sizedesc, 8); + writeBE32(hash, s[0]); + writeBE32(hash + 4, s[1]); + writeBE32(hash + 8, s[2]); + writeBE32(hash + 12, s[3]); + writeBE32(hash + 16, s[4]); + } + + DSHA1 &reset() { + bytes = 0; + initialize(s); + return *this; + } + + // Warmup the cache and get a boost in performance + DSHA1 &warmup() { + uint8_t warmup[20]; + this->write((uint8_t *)"warmupwarmupwa", 20).finalize(warmup); + return *this; + } + +private: + uint32_t s[5]; + unsigned char buf[64]; + uint64_t bytes; + + const uint32_t k1 = 0x5A827999ul; + const uint32_t k2 = 0x6ED9EBA1ul; + const uint32_t k3 = 0x8F1BBCDCul; + const uint32_t k4 = 0xCA62C1D6ul; + + uint32_t inline f1(uint32_t b, uint32_t c, uint32_t d) { return d ^ (b & (c ^ d)); } + uint32_t inline f2(uint32_t b, uint32_t c, uint32_t d) { return b ^ c ^ d; } + uint32_t inline f3(uint32_t b, uint32_t c, uint32_t d) { return (b & c) | (d & (b | c)); } + + uint32_t inline left(uint32_t x) { return (x << 1) | (x >> 31); } + + void inline Round(uint32_t a, uint32_t &b, uint32_t c, uint32_t d, uint32_t &e, + uint32_t f, uint32_t k, uint32_t w) { + e += ((a << 5) | (a >> 27)) + f + k + w; + b = (b << 30) | (b >> 2); + } + + void initialize(uint32_t s[5]) { + s[0] = 0x67452301ul; + s[1] = 0xEFCDAB89ul; + s[2] = 0x98BADCFEul; + s[3] = 0x10325476ul; + s[4] = 0xC3D2E1F0ul; + } + + void transform(uint32_t *s, const unsigned char *chunk) { + uint32_t a = s[0], b = s[1], c = s[2], d = s[3], e = s[4]; + uint32_t w0, w1, w2, w3, w4, w5, w6, w7, w8, w9, w10, w11, w12, w13, w14, w15; + + Round(a, b, c, d, e, f1(b, c, d), k1, w0 = readBE32(chunk + 0)); + Round(e, a, b, c, d, f1(a, b, c), k1, w1 = readBE32(chunk + 4)); + Round(d, e, a, b, c, f1(e, a, b), k1, w2 = readBE32(chunk + 8)); + Round(c, d, e, a, b, f1(d, e, a), k1, w3 = readBE32(chunk + 12)); + Round(b, c, d, e, a, f1(c, d, e), k1, w4 = readBE32(chunk + 16)); + Round(a, b, c, d, e, f1(b, c, d), k1, w5 = readBE32(chunk + 20)); + Round(e, a, b, c, d, f1(a, b, c), k1, w6 = readBE32(chunk + 24)); + Round(d, e, a, b, c, f1(e, a, b), k1, w7 = readBE32(chunk + 28)); + Round(c, d, e, a, b, f1(d, e, a), k1, w8 = readBE32(chunk + 32)); + Round(b, c, d, e, a, f1(c, d, e), k1, w9 = readBE32(chunk + 36)); + Round(a, b, c, d, e, f1(b, c, d), k1, w10 = readBE32(chunk + 40)); + Round(e, a, b, c, d, f1(a, b, c), k1, w11 = readBE32(chunk + 44)); + Round(d, e, a, b, c, f1(e, a, b), k1, w12 = readBE32(chunk + 48)); + Round(c, d, e, a, b, f1(d, e, a), k1, w13 = readBE32(chunk + 52)); + Round(b, c, d, e, a, f1(c, d, e), k1, w14 = readBE32(chunk + 56)); + Round(a, b, c, d, e, f1(b, c, d), k1, w15 = readBE32(chunk + 60)); + + Round(e, a, b, c, d, f1(a, b, c), k1, w0 = left(w0 ^ w13 ^ w8 ^ w2)); + Round(d, e, a, b, c, f1(e, a, b), k1, w1 = left(w1 ^ w14 ^ w9 ^ w3)); + Round(c, d, e, a, b, f1(d, e, a), k1, w2 = left(w2 ^ w15 ^ w10 ^ w4)); + Round(b, c, d, e, a, f1(c, d, e), k1, w3 = left(w3 ^ w0 ^ w11 ^ w5)); + Round(a, b, c, d, e, f2(b, c, d), k2, w4 = left(w4 ^ w1 ^ w12 ^ w6)); + Round(e, a, b, c, d, f2(a, b, c), k2, w5 = left(w5 ^ w2 ^ w13 ^ w7)); + Round(d, e, a, b, c, f2(e, a, b), k2, w6 = left(w6 ^ w3 ^ w14 ^ w8)); + Round(c, d, e, a, b, f2(d, e, a), k2, w7 = left(w7 ^ w4 ^ w15 ^ w9)); + Round(b, c, d, e, a, f2(c, d, e), k2, w8 = left(w8 ^ w5 ^ w0 ^ w10)); + Round(a, b, c, d, e, f2(b, c, d), k2, w9 = left(w9 ^ w6 ^ w1 ^ w11)); + Round(e, a, b, c, d, f2(a, b, c), k2, w10 = left(w10 ^ w7 ^ w2 ^ w12)); + Round(d, e, a, b, c, f2(e, a, b), k2, w11 = left(w11 ^ w8 ^ w3 ^ w13)); + Round(c, d, e, a, b, f2(d, e, a), k2, w12 = left(w12 ^ w9 ^ w4 ^ w14)); + Round(b, c, d, e, a, f2(c, d, e), k2, w13 = left(w13 ^ w10 ^ w5 ^ w15)); + Round(a, b, c, d, e, f2(b, c, d), k2, w14 = left(w14 ^ w11 ^ w6 ^ w0)); + Round(e, a, b, c, d, f2(a, b, c), k2, w15 = left(w15 ^ w12 ^ w7 ^ w1)); + + Round(d, e, a, b, c, f2(e, a, b), k2, w0 = left(w0 ^ w13 ^ w8 ^ w2)); + Round(c, d, e, a, b, f2(d, e, a), k2, w1 = left(w1 ^ w14 ^ w9 ^ w3)); + Round(b, c, d, e, a, f2(c, d, e), k2, w2 = left(w2 ^ w15 ^ w10 ^ w4)); + Round(a, b, c, d, e, f2(b, c, d), k2, w3 = left(w3 ^ w0 ^ w11 ^ w5)); + Round(e, a, b, c, d, f2(a, b, c), k2, w4 = left(w4 ^ w1 ^ w12 ^ w6)); + Round(d, e, a, b, c, f2(e, a, b), k2, w5 = left(w5 ^ w2 ^ w13 ^ w7)); + Round(c, d, e, a, b, f2(d, e, a), k2, w6 = left(w6 ^ w3 ^ w14 ^ w8)); + Round(b, c, d, e, a, f2(c, d, e), k2, w7 = left(w7 ^ w4 ^ w15 ^ w9)); + Round(a, b, c, d, e, f3(b, c, d), k3, w8 = left(w8 ^ w5 ^ w0 ^ w10)); + Round(e, a, b, c, d, f3(a, b, c), k3, w9 = left(w9 ^ w6 ^ w1 ^ w11)); + Round(d, e, a, b, c, f3(e, a, b), k3, w10 = left(w10 ^ w7 ^ w2 ^ w12)); + Round(c, d, e, a, b, f3(d, e, a), k3, w11 = left(w11 ^ w8 ^ w3 ^ w13)); + Round(b, c, d, e, a, f3(c, d, e), k3, w12 = left(w12 ^ w9 ^ w4 ^ w14)); + Round(a, b, c, d, e, f3(b, c, d), k3, w13 = left(w13 ^ w10 ^ w5 ^ w15)); + Round(e, a, b, c, d, f3(a, b, c), k3, w14 = left(w14 ^ w11 ^ w6 ^ w0)); + Round(d, e, a, b, c, f3(e, a, b), k3, w15 = left(w15 ^ w12 ^ w7 ^ w1)); + + Round(c, d, e, a, b, f3(d, e, a), k3, w0 = left(w0 ^ w13 ^ w8 ^ w2)); + Round(b, c, d, e, a, f3(c, d, e), k3, w1 = left(w1 ^ w14 ^ w9 ^ w3)); + Round(a, b, c, d, e, f3(b, c, d), k3, w2 = left(w2 ^ w15 ^ w10 ^ w4)); + Round(e, a, b, c, d, f3(a, b, c), k3, w3 = left(w3 ^ w0 ^ w11 ^ w5)); + Round(d, e, a, b, c, f3(e, a, b), k3, w4 = left(w4 ^ w1 ^ w12 ^ w6)); + Round(c, d, e, a, b, f3(d, e, a), k3, w5 = left(w5 ^ w2 ^ w13 ^ w7)); + Round(b, c, d, e, a, f3(c, d, e), k3, w6 = left(w6 ^ w3 ^ w14 ^ w8)); + Round(a, b, c, d, e, f3(b, c, d), k3, w7 = left(w7 ^ w4 ^ w15 ^ w9)); + Round(e, a, b, c, d, f3(a, b, c), k3, w8 = left(w8 ^ w5 ^ w0 ^ w10)); + Round(d, e, a, b, c, f3(e, a, b), k3, w9 = left(w9 ^ w6 ^ w1 ^ w11)); + Round(c, d, e, a, b, f3(d, e, a), k3, w10 = left(w10 ^ w7 ^ w2 ^ w12)); + Round(b, c, d, e, a, f3(c, d, e), k3, w11 = left(w11 ^ w8 ^ w3 ^ w13)); + Round(a, b, c, d, e, f2(b, c, d), k4, w12 = left(w12 ^ w9 ^ w4 ^ w14)); + Round(e, a, b, c, d, f2(a, b, c), k4, w13 = left(w13 ^ w10 ^ w5 ^ w15)); + Round(d, e, a, b, c, f2(e, a, b), k4, w14 = left(w14 ^ w11 ^ w6 ^ w0)); + Round(c, d, e, a, b, f2(d, e, a), k4, w15 = left(w15 ^ w12 ^ w7 ^ w1)); + + Round(b, c, d, e, a, f2(c, d, e), k4, w0 = left(w0 ^ w13 ^ w8 ^ w2)); + Round(a, b, c, d, e, f2(b, c, d), k4, w1 = left(w1 ^ w14 ^ w9 ^ w3)); + Round(e, a, b, c, d, f2(a, b, c), k4, w2 = left(w2 ^ w15 ^ w10 ^ w4)); + Round(d, e, a, b, c, f2(e, a, b), k4, w3 = left(w3 ^ w0 ^ w11 ^ w5)); + Round(c, d, e, a, b, f2(d, e, a), k4, w4 = left(w4 ^ w1 ^ w12 ^ w6)); + Round(b, c, d, e, a, f2(c, d, e), k4, w5 = left(w5 ^ w2 ^ w13 ^ w7)); + Round(a, b, c, d, e, f2(b, c, d), k4, w6 = left(w6 ^ w3 ^ w14 ^ w8)); + Round(e, a, b, c, d, f2(a, b, c), k4, w7 = left(w7 ^ w4 ^ w15 ^ w9)); + Round(d, e, a, b, c, f2(e, a, b), k4, w8 = left(w8 ^ w5 ^ w0 ^ w10)); + Round(c, d, e, a, b, f2(d, e, a), k4, w9 = left(w9 ^ w6 ^ w1 ^ w11)); + Round(b, c, d, e, a, f2(c, d, e), k4, w10 = left(w10 ^ w7 ^ w2 ^ w12)); + Round(a, b, c, d, e, f2(b, c, d), k4, w11 = left(w11 ^ w8 ^ w3 ^ w13)); + Round(e, a, b, c, d, f2(a, b, c), k4, w12 = left(w12 ^ w9 ^ w4 ^ w14)); + Round(d, e, a, b, c, f2(e, a, b), k4, left(w13 ^ w10 ^ w5 ^ w15)); + Round(c, d, e, a, b, f2(d, e, a), k4, left(w14 ^ w11 ^ w6 ^ w0)); + Round(b, c, d, e, a, f2(c, d, e), k4, left(w15 ^ w12 ^ w7 ^ w1)); + + s[0] += a; + s[1] += b; + s[2] += c; + s[3] += d; + s[4] += e; + } + + uint32_t static inline readBE32(const unsigned char *ptr) { + return __builtin_bswap32(*(uint32_t *)ptr); + } + + void static inline writeBE32(unsigned char *ptr, uint32_t x) { + *(uint32_t *)ptr = __builtin_bswap32(x); + } + + void static inline writeBE64(unsigned char *ptr, uint64_t x) { + *(uint64_t *)ptr = __builtin_bswap64(x); + } +}; +#endif diff --git a/ESP_Code/Dashboard.h b/ESP_Code/Dashboard.h index 30c88924..9281758c 100644 --- a/ESP_Code/Dashboard.h +++ b/ESP_Code/Dashboard.h @@ -1,151 +1,151 @@ -#ifndef DASHBOARD_H -#define DASHBOARD_H - -const char WEBSITE[] PROGMEM = R"=====( - - - - - - - Duino-Coin @@DEVICE@@ dashboard - - - - - -
-
-

- - @@DEVICE@@ (@@ID@@) -

-

- Self-hosted, lightweight, official dashboard for your Duino-Coin miner -

-
-
-
-
-
-
-

- Mining statistics -

-
-
-
- @@HASHRATE@@ kH/s -
-
- Hashrate -
-
-
-
- @@DIFF@@ -
-
- Difficulty -
-
-
-
- @@SHARES@@ -
-
- Shares -
-
-
-
- @@NODE@@ -
-
- Node -
-
-
-
-
-
-
-

- Device information -

-
-
-
- @@DEVICE@@ -
-
- Device type -
-
-
-
- @@ID@@ -
-
- Device ID -
-
-
-
- @@MEMORY@@ -
-
- Free memory -
-
-
-
- @@VERSION@@ -
-
- Miner version -
-
-
-
- @@SENSOR@@ -
-
- Sensor reading(s) -
-
-
-
-
-
-
-
- -
-
-
- - -)====="; - -#endif +#ifndef DASHBOARD_H +#define DASHBOARD_H + +const char WEBSITE[] PROGMEM = R"=====( + + + + + + + Duino-Coin @@DEVICE@@ dashboard + + + + + +
+
+

+ + @@DEVICE@@ (@@ID@@) +

+

+ Self-hosted, lightweight, official dashboard for your Duino-Coin miner +

+
+
+
+
+
+
+

+ Mining statistics +

+
+
+
+ @@HASHRATE@@ kH/s +
+
+ Hashrate +
+
+
+
+ @@DIFF@@ +
+
+ Difficulty +
+
+
+
+ @@SHARES@@ +
+
+ Shares +
+
+
+
+ @@NODE@@ +
+
+ Node +
+
+
+
+
+
+
+

+ Device information +

+
+
+
+ @@DEVICE@@ +
+
+ Device type +
+
+
+
+ @@ID@@ +
+
+ Device ID +
+
+
+
+ @@MEMORY@@ +
+
+ Free memory +
+
+
+
+ @@VERSION@@ +
+
+ Miner version +
+
+
+
+ @@SENSOR@@ +
+
+ Sensor reading(s) +
+
+
+
+
+
+
+
+ +
+
+
+ + +)====="; + +#endif diff --git a/ESP_Code/DisplayHal.h b/ESP_Code/DisplayHal.h index dfde2490..99b7f228 100644 --- a/ESP_Code/DisplayHal.h +++ b/ESP_Code/DisplayHal.h @@ -1,296 +1,1050 @@ -// Abstraction layer for handling various types of screens -// See Settings.h for enabling the screen of your choice -#ifndef DISPLAY_HAL_H -#define DISPLAY_HAL_H - -// Abstraction layer: custom fonts, images, etc. -#if defined(DISPLAY_SSD1306) - static const unsigned char image_check_contour_bits[] U8X8_PROGMEM = {0x00,0x04,0x00,0x0a,0x04,0x11,0x8a,0x08,0x51,0x04,0x22,0x02,0x04,0x01,0x88,0x00,0x50,0x00,0x20,0x00}; - static const unsigned char image_network_1_bar_bits[] U8X8_PROGMEM = {0x00,0x70,0x00,0x50,0x00,0x50,0x00,0x50,0x00,0x57,0x00,0x55,0x00,0x55,0x00,0x55,0x70,0x55,0x50,0x55,0x50,0x55,0x50,0x55,0x57,0x55,0x57,0x55,0x77,0x77,0x00,0x00}; - static const unsigned char image_network_2_bars_bits[] U8X8_PROGMEM = {0x00,0x70,0x00,0x50,0x00,0x50,0x00,0x50,0x00,0x57,0x00,0x55,0x00,0x55,0x00,0x55,0x70,0x55,0x70,0x55,0x70,0x55,0x70,0x55,0x77,0x55,0x77,0x55,0x77,0x77,0x00,0x00}; - static const unsigned char image_network_3_bars_bits[] U8X8_PROGMEM = {0x00,0x70,0x00,0x50,0x00,0x50,0x00,0x50,0x00,0x57,0x00,0x57,0x00,0x57,0x00,0x57,0x70,0x57,0x70,0x57,0x70,0x57,0x70,0x57,0x77,0x57,0x77,0x57,0x77,0x77,0x00,0x00}; - static const unsigned char image_network_4_bars_bits[] U8X8_PROGMEM = {0x00,0x70,0x00,0x70,0x00,0x70,0x00,0x70,0x00,0x77,0x00,0x77,0x00,0x77,0x00,0x77,0x70,0x77,0x70,0x77,0x70,0x77,0x70,0x77,0x77,0x77,0x77,0x77,0x77,0x77,0x00,0x00}; - static const unsigned char image_duco_logo_bits[] U8X8_PROGMEM = {0x20,0x00,0x20,0x00,0x20,0x00,0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7f,0x00,0xff,0x00,0xc0,0x01,0x9f,0x01,0x20,0x01,0x20,0x01,0x20,0x01,0x9f,0x01,0xc0,0x01,0xff,0x00,0x7f,0x00}; - static const unsigned char image_duco_logo_big_bits[] U8X8_PROGMEM = {0x00,0x00,0x00,0x00,0x00,0x00,0xfc,0xff,0xff,0x01,0x00,0x00,0xfc,0xff,0xff,0x0f,0x00,0x00,0xfc,0xff,0xff,0x3f,0x00,0x00,0xfc,0xff,0xff,0x7f,0x00,0x00,0xfc,0xff,0xff,0xff,0x00,0x00,0xfc,0xff,0xff,0xff,0x01,0x00,0xfc,0xff,0xff,0xff,0x03,0x00,0xf8,0xff,0xff,0xff,0x07,0x00,0x00,0x00,0x00,0xfe,0x0f,0x00,0xfc,0xff,0x03,0xf8,0x0f,0x00,0xfc,0xff,0x0f,0xf0,0x1f,0x00,0xfc,0xff,0x1f,0xe0,0x1f,0x00,0xfc,0xff,0x3f,0xe0,0x3f,0x00,0xfc,0xff,0x7f,0xc0,0x3f,0x00,0xfc,0xff,0xff,0xc0,0x7f,0x00,0xf8,0xff,0xff,0x80,0x7f,0x00,0x00,0x00,0xfe,0x80,0x7f,0x00,0x00,0x00,0xfc,0x81,0x7f,0x00,0x00,0x00,0xfc,0x01,0x7f,0x00,0x00,0x00,0xfc,0x01,0xff,0x00,0x00,0x00,0xfc,0x01,0xff,0x00,0x00,0x00,0xfc,0x01,0xff,0x00,0x00,0x00,0xfc,0x01,0xff,0x00,0x00,0x00,0xfc,0x01,0xff,0x00,0x00,0x00,0xfc,0x81,0x7f,0x00,0x00,0x00,0xfe,0x81,0x7f,0x00,0x00,0x80,0xff,0x80,0x7f,0x00,0xfc,0xff,0xff,0xc0,0x7f,0x00,0xfc,0xff,0xff,0xc0,0x7f,0x00,0xfc,0xff,0x7f,0xe0,0x3f,0x00,0xfc,0xff,0x3f,0xf0,0x3f,0x00,0xfc,0xff,0x1f,0xf8,0x1f,0x00,0xfc,0xff,0x0f,0xfc,0x1f,0x00,0xf8,0xff,0x01,0xfe,0x0f,0x00,0x00,0x00,0xc0,0xff,0x0f,0x00,0xfc,0xff,0xff,0xff,0x07,0x00,0xfc,0xff,0xff,0xff,0x03,0x00,0xfc,0xff,0xff,0xff,0x01,0x00,0xfc,0xff,0xff,0xff,0x00,0x00,0xfc,0xff,0xff,0x7f,0x00,0x00,0xfc,0xff,0xff,0x1f,0x00,0x00,0xfc,0xff,0xff,0x07,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00}; -#endif - -#if defined(DISPLAY_16X2) - static byte duco_logo[] = {0x1E, 0x01, 0x1D, 0x05, 0x1D, 0x01, 0x1E, 0x00}; - static byte check_mark[] = {0x00, 0x00, 0x00, 0x01, 0x02,0x14, 0x08, 0x00}; - static byte kh[] = {0x08, 0x0A, 0x0C, 0x0A, 0x00, 0x0A, 0x0E, 0x0A}; - static byte msec[] = {0x0A, 0x15, 0x11, 0x06, 0x08, 0x04, 0x02, 0x0C}; -#endif - - #if defined(DISPLAY_SSD1306) - void drawStrMultiline(const char *msg, int xloc, int yloc) { - //https://github.com/olikraus/u8g2/discussions/1479 - int dspwidth = u8g2.getDisplayWidth(); - int strwidth = 0; - char glyph[2]; glyph[1] = 0; - - for (const char *ptr = msg, *lastblank = NULL; *ptr; ++ptr) { - while (xloc == 0 && *msg == ' ') - if (ptr == msg++) ++ptr; - - glyph[0] = *ptr; - strwidth += u8g2.getStrWidth(glyph); - if (*ptr == ' ') lastblank = ptr; - else ++strwidth; - - if (xloc + strwidth > dspwidth) { - int starting_xloc = xloc; - while (msg < (lastblank ? lastblank : ptr)) { - glyph[0] = *msg++; - xloc += u8g2.drawStr(xloc, yloc, glyph); - } - - strwidth -= xloc - starting_xloc; - yloc += u8g2.getMaxCharHeight(); - xloc = 0; lastblank = NULL; - } - } - while (*msg) { - glyph[0] = *msg++; - xloc += u8g2.drawStr(xloc, yloc, glyph); - } - } -#endif - -#if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - void screen_setup() { - // Ran during setup() - // Abstraction layer: screen initialization - - #if defined(DISPLAY_SSD1306) - u8g2.begin(); - u8g2.clearBuffer(); - u8g2.setFontMode(1); - u8g2.setBitmapMode(1); - u8g2.sendBuffer(); - #endif - - #if defined(DISPLAY_16X2) - lcd.begin(16, 2); - lcd.createChar(0, duco_logo); - lcd.createChar(1, check_mark); - lcd.createChar(2, kh); - lcd.createChar(3, msec); - lcd.home(); - lcd.clear(); - #endif - } - - - void display_boot() { - // Abstraction layer: compilation time, features, etc. - - #if defined(DISPLAY_16X2) - lcd.clear(); - #if defined(ESP8266) - lcd.print("ESP8266 "); - #elif defined(CONFIG_FREERTOS_UNICORE) - lcd.print("ESP32S2 "); - #else - lcd.print("ESP32 "); - #endif - #if defined(ESP8266) - lcd.print(String(ESP.getCpuFreqMHz()).c_str()); - #else - lcd.print(String(getCpuFrequencyMhz()).c_str()); - #endif - lcd.print(" MHz"); - - lcd.setCursor(0, 1); - lcd.print(__DATE__); - #endif - - #if defined(DISPLAY_SSD1306) - u8g2.clearBuffer(); - - u8g2.setFont(u8g2_font_profont15_tr); - u8g2.setCursor(2, 13); - #if defined(ESP8266) - u8g2.print("ESP8266 "); - #elif defined(CONFIG_FREERTOS_UNICORE) - u8g2.print("ESP32S2/C3 "); - #else - u8g2.print("ESP32 "); - #endif - - #if defined(ESP8266) - u8g2.print(String(ESP.getCpuFreqMHz()).c_str()); - #else - u8g2.print(String(getCpuFrequencyMhz()).c_str()); - #endif - u8g2.print(" MHz"); - - u8g2.setFont(u8g2_font_profont10_tr); - u8g2.drawLine(1, 27, 126, 27); - u8g2.setCursor(2, 24); - u8g2.print("Compiled "); - u8g2.print(__DATE__); - - - u8g2.drawStr(2, 37, "Features:"); - u8g2.setCursor(2, 46); - String features_str = "OTA "; - #if defined(USE_LAN) - features_str += "LAN "; - #endif - #if defined(LED_BLINKING) - features_str += "Blink "; - #endif - #if defined(SERIAL_PRINTING) - features_str += "Serial "; - #endif - #if defined(WEB_DASHBOARD) - features_str += "Webserver "; - #endif - #if defined(DISPLAY_16X2) - features_str += "LCD16X2 "; - #endif - #if defined(DISPLAY_SSD1306) - features_str += "SSD1306 "; - #endif - #if defined(USE_INTERNAL_SENSOR) - features_str += "Int. sensor "; - #endif - #if defined(USE_DS18B20) - features_str += "DS18B20 "; - #endif - #if defined(USE_DHT) - features_str += "DHT "; - #endif - #if defined(USE_HSU07M) - features_str += "HSU07M "; - #endif - drawStrMultiline(features_str.c_str(), 2, 46); - u8g2.sendBuffer(); - #endif - } - - void display_info(String message) { - // Abstraction layer: info screens (setups) - - #if defined(DISPLAY_SSD1306) - u8g2.clearBuffer(); - u8g2.drawXBMP(-1, 3, 41, 45, image_duco_logo_big_bits); - u8g2.setFont(u8g2_font_t0_16b_tr); - #if defined(ESP8266) - u8g2.drawStr(42, 27, "ESP8266"); - #elif defined(CONFIG_FREERTOS_UNICORE) - u8g2.drawStr(42, 27, "ESP32S2/C3"); - #else - u8g2.drawStr(42, 27, "ESP32"); - #endif - u8g2.setFont(u8g2_font_t0_13b_tr); - u8g2.drawStr(41, 14, "Duino-Coin"); - u8g2.setFont(u8g2_font_6x10_tr); - u8g2.drawStr(98, 36, "MINER"); - u8g2.setFont(u8g2_font_6x13_tr); - u8g2.drawStr(1, 60, message.c_str()); - u8g2.setFont(u8g2_font_5x8_tr); - u8g2.drawStr(42, 46, "www.duinocoin.com"); - u8g2.setFont(u8g2_font_4x6_tr); - u8g2.drawStr(116, 14, String(SOFTWARE_VERSION).c_str()); - u8g2.sendBuffer(); - #endif - - #if defined(DISPLAY_16X2) - lcd.clear(); - lcd.setCursor(0, 0); - lcd.write(0); - lcd.print(" Duino-Coin "); - lcd.print(SOFTWARE_VERSION); - lcd.setCursor(0, 1); - lcd.print(message); - #endif - } - - - void display_mining_results(String hashrate, String accepted_shares, String total_shares, String uptime, String node, - String difficulty, String sharerate, String ping, String accept_rate) { - // Ran after each found share - // Abstraction layer: displaying mining results - Serial.println("Displaying mining results"); - - #if defined(DISPLAY_SSD1306) - u8g2.clearBuffer(); - u8g2.setFont(u8g2_font_profont10_tr); - u8g2.drawStr(67, 26, "kH"); - if (hashrate.toFloat() < 100.0) { - u8g2.setFont(u8g2_font_profont29_tr); - u8g2.drawStr(2, 36, hashrate.c_str()); - } else { - u8g2.setFont(u8g2_font_profont22_tr); - u8g2.drawStr(3, 35, hashrate.c_str()); - } - - u8g2.setFont(u8g2_font_haxrcorp4089_tr); - u8g2.drawStr(52, 12, node.c_str()); - - u8g2.setFont(u8g2_font_t0_11_tr); - u8g2.drawStr(17, 47, (accepted_shares + "/" + total_shares).c_str()); - u8g2.setFont(u8g2_font_5x7_tr); - u8g2.drawStr(88, 47, ("(" + accept_rate + "%)").c_str()); - - u8g2.setFont(u8g2_font_profont12_tr); - u8g2.drawStr(20, 12, (ping + "ms").c_str()); - u8g2.drawStr(69, 36, "s"); - - u8g2.setFont(u8g2_font_6x13_tr); - u8g2.drawStr(125-u8g2.getStrWidth(uptime.c_str()), 61, uptime.c_str()); - - u8g2.drawStr(85, 38, sharerate.c_str()); - u8g2.drawStr(85, 27, difficulty.c_str()); - u8g2.drawLine(67, 28, 75, 28); - - u8g2.drawXBMP(2, 38, 13, 10, image_check_contour_bits); - - if (WiFi.RSSI() > -40) { - u8g2.drawXBMP(1, 0, 15, 16, image_network_4_bars_bits); - } else if (WiFi.RSSI() > -60) { - u8g2.drawXBMP(1, 0, 15, 16, image_network_3_bars_bits); - } else if (WiFi.RSSI() > -75) { - u8g2.drawXBMP(1, 0, 15, 16, image_network_2_bars_bits); - } else { - u8g2.drawXBMP(1, 0, 15, 16, image_network_1_bar_bits); - } - - u8g2.setFont(u8g2_font_4x6_tr); - u8g2.drawStr(14, 61, String(WiFi.localIP().toString()).c_str()); - u8g2.drawStr(14, 55, ("Duino-Coin " + String(SOFTWARE_VERSION)).c_str()); - u8g2.drawXBMP(2, 11, 9, 50, image_duco_logo_bits); - u8g2.drawStr(111, 27, "diff"); - u8g2.drawStr(107, 38, "shr/s"); - - u8g2.sendBuffer(); - #endif - - #if defined(DISPLAY_16X2) - lcd.clear(); - lcd.setCursor(0,0); - lcd.print(hashrate); - lcd.setCursor(4,0); - lcd.write(2); // kh - - lcd.setCursor(7, 0); - lcd.print(difficulty); - lcd.print(" diff"); - - lcd.setCursor(0, 1); - lcd.write(1); // checkmark - lcd.print(accepted_shares); - - lcd.setCursor(7, 1); - lcd.print(ping); - lcd.write(3); // ms - - lcd.setCursor(12, 1); - lcd.print(sharerate); - lcd.print("s"); - #endif - } -#endif - -#endif +// Abstraction layer for handling various types of screens +// See Settings.h for enabling the screen of your choice +#ifndef DISPLAY_HAL_H +#define DISPLAY_HAL_H + +// Abstraction layer: custom fonts, images, etc. +#if defined(DISPLAY_SSD1306) + static const unsigned char image_check_contour_bits[] U8X8_PROGMEM = {0x00,0x04,0x00,0x0a,0x04,0x11,0x8a,0x08,0x51,0x04,0x22,0x02,0x04,0x01,0x88,0x00,0x50,0x00,0x20,0x00}; + static const unsigned char image_network_1_bar_bits[] U8X8_PROGMEM = {0x00,0x70,0x00,0x50,0x00,0x50,0x00,0x50,0x00,0x57,0x00,0x55,0x00,0x55,0x00,0x55,0x70,0x55,0x50,0x55,0x50,0x55,0x50,0x55,0x57,0x55,0x57,0x55,0x77,0x77,0x00,0x00}; + static const unsigned char image_network_2_bars_bits[] U8X8_PROGMEM = {0x00,0x70,0x00,0x50,0x00,0x50,0x00,0x50,0x00,0x57,0x00,0x55,0x00,0x55,0x00,0x55,0x70,0x55,0x70,0x55,0x70,0x55,0x70,0x55,0x77,0x55,0x77,0x55,0x77,0x77,0x00,0x00}; + static const unsigned char image_network_3_bars_bits[] U8X8_PROGMEM = {0x00,0x70,0x00,0x50,0x00,0x50,0x00,0x50,0x00,0x57,0x00,0x57,0x00,0x57,0x00,0x57,0x70,0x57,0x70,0x57,0x70,0x57,0x70,0x57,0x77,0x57,0x77,0x57,0x77,0x77,0x00,0x00}; + static const unsigned char image_network_4_bars_bits[] U8X8_PROGMEM = {0x00,0x70,0x00,0x70,0x00,0x70,0x00,0x70,0x00,0x77,0x00,0x77,0x00,0x77,0x00,0x77,0x70,0x77,0x70,0x77,0x70,0x77,0x70,0x77,0x77,0x77,0x77,0x77,0x77,0x77,0x00,0x00}; + static const unsigned char image_duco_logo_bits[] U8X8_PROGMEM = {0x20,0x00,0x20,0x00,0x20,0x00,0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7f,0x00,0xff,0x00,0xc0,0x01,0x9f,0x01,0x20,0x01,0x20,0x01,0x20,0x01,0x9f,0x01,0xc0,0x01,0xff,0x00,0x7f,0x00}; + static const unsigned char image_duco_logo_big_bits[] U8X8_PROGMEM = {0x00,0x00,0x00,0x00,0x00,0x00,0xfc,0xff,0xff,0x01,0x00,0x00,0xfc,0xff,0xff,0x0f,0x00,0x00,0xfc,0xff,0xff,0x3f,0x00,0x00,0xfc,0xff,0xff,0x7f,0x00,0x00,0xfc,0xff,0xff,0xff,0x00,0x00,0xfc,0xff,0xff,0xff,0x01,0x00,0xfc,0xff,0xff,0xff,0x03,0x00,0xf8,0xff,0xff,0xff,0x07,0x00,0x00,0x00,0x00,0xfe,0x0f,0x00,0xfc,0xff,0x03,0xf8,0x0f,0x00,0xfc,0xff,0x0f,0xf0,0x1f,0x00,0xfc,0xff,0x1f,0xe0,0x1f,0x00,0xfc,0xff,0x3f,0xe0,0x3f,0x00,0xfc,0xff,0x7f,0xc0,0x3f,0x00,0xfc,0xff,0xff,0xc0,0x7f,0x00,0xf8,0xff,0xff,0x80,0x7f,0x00,0x00,0x00,0xfe,0x80,0x7f,0x00,0x00,0x00,0xfc,0x81,0x7f,0x00,0x00,0x00,0xfc,0x01,0x7f,0x00,0x00,0x00,0xfc,0x01,0xff,0x00,0x00,0x00,0xfc,0x01,0xff,0x00,0x00,0x00,0xfc,0x01,0xff,0x00,0x00,0x00,0xfc,0x01,0xff,0x00,0x00,0x00,0xfc,0x01,0xff,0x00,0x00,0x00,0xfc,0x81,0x7f,0x00,0x00,0x00,0xfe,0x81,0x7f,0x00,0x00,0x80,0xff,0x80,0x7f,0x00,0xfc,0xff,0xff,0xc0,0x7f,0x00,0xfc,0xff,0xff,0xc0,0x7f,0x00,0xfc,0xff,0x7f,0xe0,0x3f,0x00,0xfc,0xff,0x3f,0xf0,0x3f,0x00,0xfc,0xff,0x1f,0xf8,0x1f,0x00,0xfc,0xff,0x0f,0xfc,0x1f,0x00,0xf8,0xff,0x01,0xfe,0x0f,0x00,0x00,0x00,0xc0,0xff,0x0f,0x00,0xfc,0xff,0xff,0xff,0x07,0x00,0xfc,0xff,0xff,0xff,0x03,0x00,0xfc,0xff,0xff,0xff,0x01,0x00,0xfc,0xff,0xff,0xff,0x00,0x00,0xfc,0xff,0xff,0x7f,0x00,0x00,0xfc,0xff,0xff,0x1f,0x00,0x00,0xfc,0xff,0xff,0x07,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00}; +#endif + +#if defined(DISPLAY_114) + static const unsigned char PROGMEM image_duco_square_bits[] = {0xaa,0xaa,0xa0,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x03,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x7f,0xff,0xf0,0x00,0x00,0x00,0x00,0x1f,0xff,0xf8,0x00,0x00,0x00,0x00,0x07,0xff,0xf8,0x00,0x00,0x00,0x00,0x03,0xff,0xf8,0x00,0x00,0x00,0x00,0x03,0xff,0xfc,0xff,0xff,0x80,0x00,0x01,0xff,0xfc,0xff,0xff,0xf0,0x00,0x00,0xff,0xfc,0xff,0xff,0xfe,0x00,0x00,0xff,0xfe,0xff,0xff,0xff,0x00,0x00,0xff,0xfe,0xff,0xff,0xff,0x80,0x00,0x7f,0xfe,0xff,0xff,0xff,0xc0,0x00,0x7f,0xfe,0xff,0xff,0xff,0xe0,0x00,0x7f,0xfe,0xff,0xff,0xff,0xe0,0x00,0x7f,0xfe,0x00,0x0f,0xff,0xf0,0x00,0x3f,0xff,0x00,0x00,0xff,0xf0,0x00,0x3f,0xff,0x00,0x00,0x7f,0xf0,0x00,0x3f,0xff,0x00,0x00,0x3f,0xf8,0x00,0x3f,0xff,0x00,0x00,0x1f,0xf8,0x00,0x3f,0xff,0x00,0x00,0x1f,0xf8,0x00,0x3f,0xff,0x00,0x00,0x1f,0xf8,0x00,0x3f,0xff,0x00,0x00,0x1f,0xf8,0x00,0x3f,0xff,0x00,0x00,0x1f,0xf8,0x00,0x3f,0xff,0x00,0x00,0x1f,0xf8,0x00,0x3f,0xff,0x00,0x00,0x1f,0xf8,0x00,0x3f,0xff,0x00,0x00,0x3f,0xf8,0x00,0x3f,0xff,0x00,0x00,0x7f,0xf0,0x00,0x3f,0xff,0x00,0x00,0xff,0xf0,0x00,0x3f,0xff,0x00,0x07,0xff,0xf0,0x00,0x3f,0xff,0xff,0xff,0xff,0xf0,0x00,0x3f,0xfe,0xff,0xff,0xff,0xe0,0x00,0x7f,0xfe,0xff,0xff,0xff,0xc0,0x00,0x7f,0xfe,0xff,0xff,0xff,0xc0,0x00,0x7f,0xfe,0xff,0xff,0xff,0x80,0x00,0x7f,0xfe,0xff,0xff,0xfe,0x00,0x00,0xff,0xfe,0xff,0xff,0xf8,0x00,0x00,0xff,0xfc,0xff,0xff,0xc0,0x00,0x01,0xff,0xfc,0xaa,0xa0,0x00,0x00,0x01,0xff,0xfc,0x00,0x00,0x00,0x00,0x03,0xff,0xfc,0x00,0x00,0x00,0x00,0x07,0xff,0xf8,0x00,0x00,0x00,0x00,0x0f,0xff,0xf8,0x00,0x00,0x00,0x00,0x3f,0xff,0xf0,0x00,0x00,0x00,0x00,0xff,0xff,0xf0,0x00,0x00,0x00,0x1f,0xff,0xff,0xe0,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0xff,0xff,0xfa,0x80,0x00,0x00,0x00}; + static const unsigned char PROGMEM image_duco_logo_bits[] = {0xff,0xf8,0x00,0xff,0xfe,0x00,0xff,0xff,0x80,0x00,0x07,0xc0,0x00,0x03,0xe0,0x00,0x01,0xf0,0xff,0xf1,0xf0,0x00,0x18,0xf8,0x00,0x0c,0x78,0x00,0x04,0x78,0x00,0x04,0x78,0x00,0x04,0x78,0x00,0x04,0x78,0x00,0x04,0x78,0x00,0x04,0x78,0x00,0x0c,0x78,0x00,0x18,0xf0,0xff,0xf0,0xf0,0x00,0x01,0xe0,0x00,0x01,0xc0,0x00,0x07,0xc0,0xff,0xff,0x80,0xff,0xff,0x00,0xff,0xf8,0x00}; + static const unsigned char PROGMEM image_check_contour_bits[] = {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x20,0x00,0x50,0x20,0x88,0x51,0x10,0x8a,0x20,0x44,0x40,0x20,0x80,0x11,0x00,0x0a,0x00,0x04,0x00,0x00,0x00,0x00,0x00,0x00,0x00}; + static const unsigned char PROGMEM image_ButtonRight_bits[] = {0x80,0xc0,0xe0,0xf0,0xe0,0xc0,0x80}; + static const unsigned char PROGMEM image_clock_bits[] = {0x07,0xc0,0x18,0x30,0x29,0x28,0x41,0x04,0x61,0x0c,0x81,0x02,0x81,0x02,0xe1,0x0e,0x80,0x82,0x80,0x42,0x60,0x2c,0x40,0x04,0x29,0x28,0x19,0x30,0x07,0xc0,0x00,0x00}; + static const unsigned char PROGMEM image_paint_0_bits[] = {0x7f,0xff,0xff,0xf0,0x00,0x00,0x7f,0xff,0xff,0xfc,0x00,0x00,0x7f,0xff,0xff,0xff,0x00,0x00,0x7f,0xff,0xff,0xff,0x80,0x00,0x7f,0xff,0xff,0xff,0xe0,0x00,0x7f,0xff,0xff,0xff,0xf0,0x00,0x7f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x07,0xfc,0x00,0x00,0x00,0x00,0x03,0xfe,0x00,0x00,0x00,0x00,0x01,0xfe,0x00,0x00,0x00,0x00,0x00,0xff,0x00,0x7f,0xff,0xff,0xc0,0x7f,0x00,0x7f,0xff,0xff,0xf0,0x3f,0x80,0x00,0x00,0x00,0x38,0x3f,0x80,0x00,0x00,0x00,0x1c,0x1f,0x80,0x00,0x00,0x00,0x0e,0x1f,0x80,0x00,0x00,0x00,0x07,0x1f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x03,0x0f,0x80,0x00,0x00,0x00,0x07,0x0f,0x80,0x00,0x00,0x00,0x0e,0x1f,0x80,0x00,0x00,0x00,0x1c,0x1f,0x80,0x00,0x00,0x00,0x38,0x3f,0x80,0xff,0xff,0xff,0xf0,0x7f,0x80,0xff,0xff,0xff,0xc0,0x7f,0x00,0x00,0x00,0x00,0x00,0xff,0x00,0x00,0x00,0x00,0x03,0xfe,0x00,0x00,0x00,0x00,0x0f,0xfe,0x00,0x00,0x00,0x00,0x3f,0xfc,0x00,0xff,0xff,0xff,0xff,0xf8,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0xff,0xff,0xff,0xff,0xe0,0x00,0xff,0xff,0xff,0xff,0x80,0x00,0xff,0xff,0xff,0xfe,0x00,0x00,0xff,0xff,0xff,0xf0,0x00,0x00,0xff,0xff,0xff,0xc0,0x00,0x00}; + static const unsigned char PROGMEM image_network_1_bits[] = {0x00,0x0e,0x00,0x0a,0x00,0x0a,0x00,0x0a,0x00,0xea,0x00,0xaa,0x00,0xaa,0x00,0xaa,0x0e,0xaa,0x0a,0xaa,0x0a,0xaa,0x0a,0xaa,0xea,0xaa,0xaa,0xaa,0xee,0xee,0x00,0x00}; + static const unsigned char PROGMEM image_network_1_bar_bits[] = {0x00,0x0e,0x00,0x0a,0x00,0x0a,0x00,0x0a,0x00,0xea,0x00,0xaa,0x00,0xaa,0x00,0xaa,0x0e,0xaa,0x0a,0xaa,0x0a,0xaa,0x0a,0xaa,0xea,0xaa,0xea,0xaa,0xee,0xee,0x00,0x00}; + static const unsigned char PROGMEM image_network_2_bars_bits[] = {0x00,0x0e,0x00,0x0a,0x00,0x0a,0x00,0x0a,0x00,0xea,0x00,0xaa,0x00,0xaa,0x00,0xaa,0x0e,0xaa,0x0e,0xaa,0x0e,0xaa,0x0e,0xaa,0xee,0xaa,0xee,0xaa,0xee,0xee,0x00,0x00}; + static const unsigned char PROGMEM image_network_4_bars_bits[] = {0x00,0x0e,0x00,0x0e,0x00,0x0e,0x00,0x0e,0x00,0xee,0x00,0xee,0x00,0xee,0x00,0xee,0x0e,0xee,0x0e,0xee,0x0e,0xee,0x0e,0xee,0xee,0xee,0xee,0xee,0xee,0xee,0x00,0x00}; + static const unsigned char PROGMEM image_network_3_bars_bits[] = {0x00,0x0e,0x00,0x0a,0x00,0x0a,0x00,0x0a,0x00,0xea,0x00,0xea,0x00,0xea,0x00,0xea,0x0e,0xea,0x0e,0xea,0x0e,0xea,0x0e,0xea,0xee,0xea,0xee,0xea,0xee,0xee,0x00,0x00}; +#endif + +#if defined(DISPLAY_2432S08) + static const unsigned char PROGMEM image_duco_2_bits[] = {0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0x80,0x00,0x00,0x00,0x00,0xff,0xff,0x80,0x01,0xff,0xc0,0x00,0x7f,0x00,0x00,0xff,0xff,0xf0,0x00,0xff,0xe0,0x00,0x0f,0xe0,0x00,0xff,0xff,0xfc,0x00,0x7f,0xe0,0x00,0x03,0xf8,0x00,0xff,0xff,0xfe,0x00,0x3f,0xf0,0x00,0x00,0x7e,0x00,0xff,0xff,0xff,0x00,0x3f,0xf0,0x00,0x00,0x1f,0x00,0xff,0xff,0xff,0x80,0x1f,0xf8,0x00,0x00,0x0f,0x80,0xff,0xff,0xff,0xc0,0x0f,0xf8,0x00,0x00,0x0f,0x80,0xff,0xff,0xff,0xc0,0x0f,0xf8,0x00,0x00,0x1f,0xc0,0x00,0x00,0x7f,0xe0,0x0f,0xfc,0x00,0x00,0x3f,0xc0,0x00,0x00,0x3f,0xe0,0x07,0xfc,0x00,0x00,0x79,0xe0,0x00,0x00,0x1f,0xe0,0x07,0xfc,0x00,0x00,0xf0,0xe0,0x00,0x00,0x0f,0xf0,0x07,0xfc,0x00,0x01,0xe0,0xe0,0x00,0x00,0x0f,0xf0,0x07,0xfc,0x00,0x03,0xc0,0x60,0x00,0x00,0x0f,0xf0,0x03,0xfc,0x00,0x07,0x80,0x70,0x00,0x00,0x07,0xf0,0x03,0xfc,0x00,0x0f,0x00,0x30,0x00,0x00,0x07,0xf0,0x03,0xfc,0x00,0x1e,0x00,0x30,0x00,0x00,0x07,0xf0,0x03,0xfc,0x00,0x3c,0x00,0x30,0x00,0x00,0x0f,0xf0,0x03,0xfc,0x00,0x78,0x00,0x10,0x00,0x00,0x0f,0xf0,0x07,0xfc,0x00,0xf0,0x00,0x10,0x00,0x00,0x0f,0xf0,0x07,0xfc,0x01,0xe0,0x00,0x10,0x00,0x00,0x1f,0xf0,0x07,0xfc,0x03,0xc0,0x00,0x00,0x00,0x00,0x1f,0xe0,0x07,0xfc,0x07,0x80,0x00,0x00,0x00,0x00,0x3f,0xe0,0x07,0xfc,0x0f,0x00,0x00,0x00,0xff,0xff,0xff,0xc0,0x0f,0xf8,0x1e,0x00,0x00,0x00,0xff,0xff,0xff,0xc0,0x0f,0xf8,0x3c,0x00,0x00,0x00,0xff,0xff,0xff,0x80,0x1f,0xf8,0x38,0x00,0x00,0x00,0xff,0xff,0xff,0x80,0x1f,0xf0,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0x00,0x3f,0xf0,0x00,0x00,0x00,0x00,0xff,0xff,0xfc,0x00,0x7f,0xf0,0x00,0x00,0x00,0x00,0xff,0xff,0xf8,0x00,0xff,0xe0,0x00,0x00,0x00,0x00,0xff,0xff,0xe0,0x01,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xc7,0x0e,0x1f,0xc0,0xfc,0x00,0x00,0x00,0x0f,0xff,0x87,0x0e,0x3f,0xe1,0xfe,0x00,0x00,0x00,0xff,0xff,0x07,0x0e,0x7f,0xe3,0xcf,0xff,0xff,0xff,0xff,0xfe,0x07,0x0e,0x70,0x03,0x87,0xff,0xff,0xff,0xff,0xfc,0x07,0x0e,0x60,0x03,0x03,0xff,0xff,0xff,0xff,0xf8,0x07,0x0e,0x60,0x03,0x03,0xff,0xff,0xff,0xff,0xf0,0x07,0x0e,0x60,0x03,0x03,0xff,0xff,0xff,0xff,0xe0,0x07,0x0e,0x70,0x03,0x87,0xff,0xff,0xff,0xff,0x80,0x07,0x9e,0x7f,0xe3,0xcf,0xff,0xff,0xff,0xfe,0x00,0x03,0xfc,0x3f,0xe1,0xfe,0xff,0xff,0xff,0xf0,0x00,0x01,0xf8,0x1f,0xc0,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x70,0x70,0xff,0xe1,0x80,0x10,0x7f,0xf8,0x1f,0xfc,0xf8,0xf9,0xff,0xf3,0xc0,0x38,0xff,0xfc,0x3f,0xfe,0xfd,0xf8,0xff,0xe3,0xe0,0x38,0xff,0xf8,0x7f,0xff,0xef,0xb8,0x0e,0x03,0xf0,0x38,0xe0,0x00,0x78,0x07,0xe7,0x38,0x0e,0x03,0xb0,0x38,0xe0,0x00,0x70,0x03,0xe2,0x38,0x0e,0x03,0xb8,0x38,0xe0,0x00,0x70,0x03,0xe0,0x38,0x0e,0x03,0x9c,0x38,0xe0,0x00,0x78,0x07,0xe0,0x38,0x0e,0x03,0x9c,0x38,0xff,0x00,0x7f,0xff,0xe0,0x38,0x0e,0x03,0x8e,0x38,0xff,0x80,0x7f,0xfe,0xe0,0x38,0x0e,0x03,0x87,0x38,0xff,0x00,0x7f,0xfc,0xe0,0x38,0x0e,0x03,0x87,0x38,0xe0,0x00,0x70,0xe0,0xe0,0x38,0x0e,0x03,0x83,0xb8,0xe0,0x00,0x70,0x70,0xe0,0x38,0x0e,0x03,0x81,0xf8,0xe0,0x00,0x70,0x38,0xe0,0x38,0x0e,0x03,0x80,0xf8,0xe0,0x00,0x70,0x1c,0xe0,0x38,0xff,0xe3,0x80,0xf8,0xff,0xf8,0x70,0x0e,0xe0,0x39,0xff,0xf3,0x80,0x78,0xff,0xfc,0x70,0x07,0x40,0x10,0xff,0xe1,0x00,0x30,0x7f,0xf8,0x20,0x03}; + static const unsigned char PROGMEM image_check_big_bits[] = {0x00,0x00,0xf0,0x00,0x00,0xf0,0x00,0x03,0xf0,0x00,0x03,0xf0,0x00,0x03,0xc0,0x00,0x0f,0xc0,0x00,0x0f,0x00,0xf0,0x3f,0x00,0xf0,0x3f,0x00,0xfc,0xfc,0x00,0xfc,0xfc,0x00,0x3f,0xf0,0x00,0x3f,0xf0,0x00,0x0f,0xc0,0x00,0x0f,0xc0,0x00,0x03,0x00,0x00,0x03,0x00,0x00}; + static const unsigned char PROGMEM image_clock_bits[] = {0x07,0xc0,0x18,0x30,0x29,0x28,0x41,0x04,0x61,0x0c,0x81,0x02,0x81,0x02,0xe1,0x0e,0x80,0x82,0x80,0x42,0x60,0x2c,0x40,0x04,0x29,0x28,0x19,0x30,0x07,0xc0,0x00,0x00}; + static const unsigned char PROGMEM image_hand_thumbs_up_bits[] = {0x00,0x10,0x00,0x28,0x00,0x28,0x00,0x48,0x00,0x50,0x00,0x90,0x01,0x3e,0xfa,0x03,0x8c,0x05,0x88,0x03,0x88,0x05,0x88,0x03,0x88,0x05,0xa8,0x02,0x8e,0x06,0xf9,0xfc}; + static const unsigned char PROGMEM image_music_radio_broadcast_bits[] = {0x07,0xc0,0x18,0x30,0x27,0xc8,0x48,0x24,0x93,0x92,0xa4,0x4a,0xa9,0x2a,0xa3,0x8a,0x06,0xc0,0x03,0x80,0x01,0x00,0x03,0x80,0x02,0x80,0x06,0xc0,0x04,0x40,0x00,0x00}; +// static const unsigned char PROGMEM image_duco_bits[] = {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x03,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x0f,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7f,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1f,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xc0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xf0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xfc,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3f,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xe0,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00}; + static const unsigned char PROGMEM image_network_1_bits[] = {0x00,0x0e,0x00,0x0a,0x00,0x0a,0x00,0x0a,0x00,0xea,0x00,0xaa,0x00,0xaa,0x00,0xaa,0x0e,0xaa,0x0a,0xaa,0x0a,0xaa,0x0a,0xaa,0xea,0xaa,0xaa,0xaa,0xee,0xee,0x00,0x00}; + static const unsigned char PROGMEM image_network_1_bar_bits[] = {0x00,0x0e,0x00,0x0a,0x00,0x0a,0x00,0x0a,0x00,0xea,0x00,0xaa,0x00,0xaa,0x00,0xaa,0x0e,0xaa,0x0a,0xaa,0x0a,0xaa,0x0a,0xaa,0xea,0xaa,0xea,0xaa,0xee,0xee,0x00,0x00}; + static const unsigned char PROGMEM image_network_2_bars_bits[] = {0x00,0x0e,0x00,0x0a,0x00,0x0a,0x00,0x0a,0x00,0xea,0x00,0xaa,0x00,0xaa,0x00,0xaa,0x0e,0xaa,0x0e,0xaa,0x0e,0xaa,0x0e,0xaa,0xee,0xaa,0xee,0xaa,0xee,0xee,0x00,0x00}; + static const unsigned char PROGMEM image_network_4_bars_bits[] = {0x00,0x0e,0x00,0x0e,0x00,0x0e,0x00,0x0e,0x00,0xee,0x00,0xee,0x00,0xee,0x00,0xee,0x0e,0xee,0x0e,0xee,0x0e,0xee,0x0e,0xee,0xee,0xee,0xee,0xee,0xee,0xee,0x00,0x00}; + static const unsigned char PROGMEM image_network_3_bars_bits[] = {0x00,0x0e,0x00,0x0a,0x00,0x0a,0x00,0x0a,0x00,0xea,0x00,0xea,0x00,0xea,0x00,0xea,0x0e,0xea,0x0e,0xea,0x0e,0xea,0x0e,0xea,0xee,0xea,0xee,0xea,0xee,0xee,0x00,0x00}; +#endif + +#if defined(DISPLAY_7735) + static const unsigned char PROGMEM image_duco_bits[] = {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x07,0xff,0xfc,0x00,0x00,0x07,0xff,0xff,0x80,0x00,0x07,0xff,0xff,0xc0,0x00,0x07,0xff,0xff,0xe0,0x00,0x07,0xff,0xff,0xf0,0x00,0x00,0x00,0x03,0xf8,0x00,0x07,0xfe,0x01,0xfc,0x00,0x07,0xff,0xc0,0xfc,0x00,0x07,0xff,0xe0,0x7e,0x00,0x07,0xff,0xf0,0x3e,0x00,0x07,0xff,0xf8,0x3f,0x00,0x00,0x00,0xf8,0x1f,0x00,0x00,0x00,0x78,0x1f,0x00,0x00,0x00,0x7c,0x1f,0x00,0x00,0x00,0x7c,0x1f,0x00,0x00,0x00,0x7c,0x1f,0x00,0x00,0x00,0x7c,0x1f,0x00,0x00,0x00,0x78,0x1f,0x00,0x00,0x00,0xf8,0x1f,0x00,0x00,0x01,0xf8,0x1f,0x00,0x07,0xff,0xf0,0x3f,0x00,0x07,0xff,0xf0,0x3e,0x00,0x07,0xff,0xe0,0x7e,0x00,0x07,0xff,0x80,0xfc,0x00,0x00,0x00,0x01,0xfc,0x00,0x00,0x00,0x07,0xf8,0x00,0x07,0xff,0xff,0xf0,0x00,0x07,0xff,0xff,0xe0,0x00,0x07,0xff,0xff,0xc0,0x00,0x07,0xff,0xff,0x00,0x00,0x07,0xff,0xf8,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00}; + static const unsigned char PROGMEM image_check_contour_bits[] = {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x20,0x00,0x50,0x20,0x88,0x51,0x10,0x8a,0x20,0x44,0x40,0x20,0x80,0x11,0x00,0x0a,0x00,0x04,0x00,0x00,0x00,0x00,0x00,0x00,0x00}; + static const unsigned char PROGMEM image_ButtonRightSmall_bits[] = {0x80,0xc0,0xe0,0xc0,0x80}; + static const unsigned char PROGMEM image_clock_bits[] = {0x07,0xc0,0x18,0x30,0x29,0x28,0x41,0x04,0x61,0x0c,0x81,0x02,0x81,0x02,0xe1,0x0e,0x80,0x82,0x80,0x42,0x60,0x2c,0x40,0x04,0x29,0x28,0x19,0x30,0x07,0xc0,0x00,0x00}; + static const unsigned char PROGMEM image_duco_logo_bits[] = {0xff,0xf8,0x00,0xff,0xfe,0x00,0xff,0xff,0x80,0x00,0x07,0xc0,0x00,0x03,0xe0,0x00,0x01,0xf0,0xff,0xf1,0xf0,0x00,0x18,0xf8,0x00,0x0c,0x78,0x00,0x04,0x78,0x00,0x04,0x78,0x00,0x04,0x78,0x00,0x04,0x78,0x00,0x04,0x78,0x00,0x04,0x78,0x00,0x0c,0x78,0x00,0x18,0xf0,0xff,0xf0,0xf0,0x00,0x01,0xe0,0x00,0x01,0xc0,0x00,0x07,0xc0,0xff,0xff,0x80,0xff,0xff,0x00,0xff,0xf8,0x00}; +#endif + +#if defined(DISPLAY_16X2) + static byte duco_logo[] = {0x1E, 0x01, 0x1D, 0x05, 0x1D, 0x01, 0x1E, 0x00}; + static byte check_mark[] = {0x00, 0x00, 0x00, 0x01, 0x02,0x14, 0x08, 0x00}; + static byte kh[] = {0x08, 0x0A, 0x0C, 0x0A, 0x00, 0x0A, 0x0E, 0x0A}; + static byte msec[] = {0x0A, 0x15, 0x11, 0x06, 0x08, 0x04, 0x02, 0x0C}; +#endif + + #if defined(DISPLAY_SSD1306) + void drawStrMultiline(const char *msg, int xloc, int yloc) { + //https://github.com/olikraus/u8g2/discussions/1479 + int dspwidth = u8g2.getDisplayWidth(); + int strwidth = 0; + char glyph[2]; glyph[1] = 0; + + for (const char *ptr = msg, *lastblank = NULL; *ptr; ++ptr) { + while (xloc == 0 && *msg == ' ') + if (ptr == msg++) ++ptr; + + glyph[0] = *ptr; + strwidth += u8g2.getStrWidth(glyph); + if (*ptr == ' ') lastblank = ptr; + else ++strwidth; + + if (xloc + strwidth > dspwidth) { + int starting_xloc = xloc; + while (msg < (lastblank ? lastblank : ptr)) { + glyph[0] = *msg++; + xloc += u8g2.drawStr(xloc, yloc, glyph); + } + + strwidth -= xloc - starting_xloc; + yloc += u8g2.getMaxCharHeight(); + xloc = 0; lastblank = NULL; + } + } + while (*msg) { + glyph[0] = *msg++; + xloc += u8g2.drawStr(xloc, yloc, glyph); + } + } +#endif + +#if defined(DISPLAY_114) + void drawStrMultiline(const char *msg, int xloc, int yloc) { + //https://github.com/olikraus/u8g2/discussions/1479 + int dspwidth = 240; + int strwidth = 0; + char glyph[2]; glyph[1] = 0; + + for (const char *ptr = msg, *lastblank = NULL; *ptr; ++ptr) { + while (xloc == 0 && *msg == ' ') + if (ptr == msg++) ++ptr; + + glyph[0] = *ptr; + strwidth = 10; + if (*ptr == ' ') lastblank = ptr; + else ++strwidth; + + if (xloc + strwidth > dspwidth) { + int starting_xloc = xloc; + while (msg < (lastblank ? lastblank : ptr)) { + glyph[0] = *msg++; + xloc += tft.drawString(glyph,xloc, yloc ); + } + + strwidth -= xloc - starting_xloc; + yloc += 10; + xloc = 0; lastblank = NULL; + } + } + while (*msg) { + glyph[0] = *msg++; + xloc += tft.drawString(glyph,xloc, yloc ); + } + } +#endif + +#if defined(DISPLAY_2432S08) + void drawStrMultiline(const char *msg, int xloc, int yloc) { + //https://github.com/olikraus/u8g2/discussions/1479 + int dspwidth = 320; + int strwidth = 0; + char glyph[2]; glyph[1] = 0; + + for (const char *ptr = msg, *lastblank = NULL; *ptr; ++ptr) { + while (xloc == 0 && *msg == ' ') + if (ptr == msg++) ++ptr; + + glyph[0] = *ptr; + strwidth = 10; + if (*ptr == ' ') lastblank = ptr; + else ++strwidth; + + if (xloc + strwidth > dspwidth) { + int starting_xloc = xloc; + while (msg < (lastblank ? lastblank : ptr)) { + glyph[0] = *msg++; + xloc += tft.drawString(glyph,xloc, yloc ); + } + + strwidth -= xloc - starting_xloc; + yloc += 10; + xloc = 0; lastblank = NULL; + } + } + while (*msg) { + glyph[0] = *msg++; + xloc += tft.drawString(glyph,xloc, yloc ); + } + } +#endif + +#if defined(DISPLAY_7735) + void drawStrMultiline(const char *msg, int xloc, int yloc) { + //https://github.com/olikraus/u8g2/discussions/1479 + int dspwidth = 160; + int strwidth = 0; + char glyph[2]; glyph[1] = 0; + + for (const char *ptr = msg, *lastblank = NULL; *ptr; ++ptr) { + while (xloc == 0 && *msg == ' ') + if (ptr == msg++) ++ptr; + + glyph[0] = *ptr; + strwidth = 10; + if (*ptr == ' ') lastblank = ptr; + else ++strwidth; + + if (xloc + strwidth > dspwidth) { + int starting_xloc = xloc; + while (msg < (lastblank ? lastblank : ptr)) { + glyph[0] = *msg++; + xloc += tft.drawString(glyph,xloc, yloc ); + } + + strwidth -= xloc - starting_xloc; + yloc += 10; + xloc = 0; lastblank = NULL; + } + } + while (*msg) { + glyph[0] = *msg++; + xloc += tft.drawString(glyph,xloc, yloc ); + } + } +#endif + +#if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2)|| defined(DISPLAY_114)|| defined(DISPLAY_7735)|| defined(DISPLAY_2432S08) + void screen_setup() { + // Ran during setup() + // Abstraction layer: screen initialization + + #if defined(DISPLAY_SSD1306) + u8g2.begin(); + u8g2.clearBuffer(); + u8g2.setFontMode(1); + u8g2.setBitmapMode(1); + u8g2.sendBuffer(); + #endif + + #if defined(DISPLAY_16X2) + lcd.begin(16, 2); + lcd.createChar(0, duco_logo); + lcd.createChar(1, check_mark); + lcd.createChar(2, kh); + lcd.createChar(3, msec); + lcd.home(); + lcd.clear(); + #endif + + #if defined(DISPLAY_114) + tft.init(); + tft.begin(); + tft.setRotation(1); + tft.fillScreen(TFT_BLACK); + + #endif + + #if defined(DISPLAY_2432S08) + tft.init(); + tft.begin(); + tft.setRotation(1); + tft.fillScreen(TFT_BLACK); + + #endif + + #if defined(DISPLAY_7735) + tft.init(); + tft.begin(); + tft.setRotation(1); + tft.fillScreen(TFT_BLACK); + + #endif + } + + + void display_boot() { + // Abstraction layer: compilation time, features, etc. + + #if defined(DISPLAY_16X2) + lcd.clear(); + #if defined(ESP8266) + lcd.print("ESP8266 "); + #elif defined(CONFIG_FREERTOS_UNICORE) + lcd.print("ESP32S2 "); + #else + lcd.print("ESP32 "); + #endif + #if defined(ESP8266) + lcd.print(String(ESP.getCpuFreqMHz()).c_str()); + #else + lcd.print(String(getCpuFrequencyMhz()).c_str()); + #endif + lcd.print(" MHz"); + + lcd.setCursor(0, 1); + lcd.print(__DATE__); + #endif + + #if defined(DISPLAY_SSD1306) + u8g2.clearBuffer(); + + u8g2.setFont(u8g2_font_profont15_tr); + u8g2.setCursor(2, 13); + #if defined(ESP8266) + u8g2.print("ESP8266 "); + #elif defined(CONFIG_FREERTOS_UNICORE) + u8g2.print("ESP32S2/C3 "); + #else + u8g2.print("ESP32 "); + #endif + + #if defined(ESP8266) + u8g2.print(String(ESP.getCpuFreqMHz()).c_str()); + #else + u8g2.print(String(getCpuFrequencyMhz()).c_str()); + #endif + u8g2.print(" MHz"); + + u8g2.setFont(u8g2_font_profont10_tr); + u8g2.drawLine(1, 27, 126, 27); + u8g2.setCursor(2, 24); + u8g2.print("Compiled "); + u8g2.print(__DATE__); + + + u8g2.drawStr(2, 37, "Features:"); + u8g2.setCursor(2, 46); + String features_str = "OTA "; + #if defined(USE_LAN) + features_str += "LAN "; + #endif + #if defined(LED_BLINKING) + features_str += "Blink "; + #endif + #if defined(SERIAL_PRINTING) + features_str += "Serial "; + #endif + #if defined(WEB_DASHBOARD) + features_str += "Webserver "; + #endif + #if defined(DISPLAY_16X2) + features_str += "LCD16X2 "; + #endif + #if defined(DISPLAY_SSD1306) + features_str += "SSD1306 "; + #endif + #if defined(USE_INTERNAL_SENSOR) + features_str += "Int. sensor "; + #endif + #if defined(USE_DS18B20) + features_str += "DS18B20 "; + #endif + #if defined(USE_DHT) + features_str += "DHT "; + #endif + #if defined(USE_HSU07M) + features_str += "HSU07M "; + #endif + drawStrMultiline(features_str.c_str(), 2, 46); + u8g2.sendBuffer(); + #endif + + #if defined(DISPLAY_114) + tft.fillScreen(TFT_BLACK); + tft.setTextColor(TFT_WHITE,TFT_BLACK); + // tft.drawBitmap(0, 3, image_duco_logo_bits, 21, 24, 0xFC00); + tft.drawBitmap(0, 4, image_paint_0_bits, 41, 51, 0xFC00); + tft.setTextSize(2); + tft.setCursor(55, 15); + #if defined(ESP8266) + tft.print("ESP8266 "); + #elif defined(CONFIG_FREERTOS_UNICORE) + tft.print("ESP32S2/C3 "); + #else + tft.print("ESP32 "); + #endif + + #if defined(ESP8266) + tft.print(String(ESP.getCpuFreqMHz()).c_str()); + #else + tft.print(String(getCpuFrequencyMhz()).c_str()); + #endif + tft.print(" MHz"); + tft.setTextSize(1); + tft.setCursor(2, 70); + tft.print("Compiled "); + tft.print(__DATE__); + + delay(2500); + + tft.drawString("Features:",2, 85,1); + String features_str = "OTA "; + #if defined(USE_LAN) + features_str += "LAN "; + #endif + #if defined(LED_BLINKING) + features_str += "Blink "; + #endif + #if defined(SERIAL_PRINTING) + features_str += "Serial "; + #endif + #if defined(WEB_DASHBOARD) + features_str += "Webserver "; + #endif + #if defined(DISPLAY_16X2) + features_str += "LCD16X2 "; + #endif + #if defined(DISPLAY_SSD1306) + features_str += "SSD1306 "; + #endif + #if defined(DISPLAY_114) + features_str += "TT-GO "; + #endif + #if defined(USE_INTERNAL_SENSOR) + features_str += "Int. sensor "; + #endif + #if defined(USE_DS18B20) + features_str += "DS18B20 "; + #endif + #if defined(USE_DHT) + features_str += "DHT "; + #endif + #if defined(USE_HSU07M) + features_str += "HSU07M "; + #endif + drawStrMultiline(features_str.c_str(), 2, 102); + delay(2500); + #endif + + #if defined(DISPLAY_2432S08) + tft.fillScreen(TFT_BLACK); + tft.drawBitmap(110, 80, image_duco_2_bits, 80, 74, 0xFC00); + delay(2500); + tft.fillScreen(TFT_BLACK); + tft.drawRect(0, 0, 319, 239, 0xFFE0); + tft.setTextColor(TFT_WHITE,TFT_BLACK); + tft.drawBitmap(17, 20, image_duco_2_bits, 80, 74, 0xFC00); + tft.setTextSize(2); + tft.setCursor(55, 110); + #if defined(ESP8266) + tft.print("ESP8266 "); + #elif defined(CONFIG_FREERTOS_UNICORE) + tft.print("ESP32S2/C3 "); + #else + tft.print("ESP32 "); + #endif + + #if defined(ESP8266) + tft.print(String(ESP.getCpuFreqMHz()).c_str()); + #else + tft.print(String(getCpuFrequencyMhz()).c_str()); + #endif + tft.print(" MHz"); + tft.setTextSize(1); + tft.setCursor(6, 140); + tft.print("Compiled "); + tft.print(__DATE__); + + delay(2500); + + tft.drawString("Features:",6, 160,1); + String features_str = "OTA "; + #if defined(USE_LAN) + features_str += "LAN "; + #endif + #if defined(LED_BLINKING) + features_str += "Blink "; + #endif + #if defined(SERIAL_PRINTING) + features_str += "Serial "; + #endif + #if defined(WEB_DASHBOARD) + features_str += "Webserver "; + #endif + #if defined(DISPLAY_16X2) + features_str += "LCD16X2 "; + #endif + #if defined(DISPLAY_SSD1306) + features_str += "SSD1306 "; + #endif + #if defined(DISPLAY_114) + features_str += "TT-GO "; + #endif + #if defined(DISPLAY_2432S08) + features_str += "2432S028 "; + #endif + #if defined(USE_INTERNAL_SENSOR) + features_str += "Int. sensor "; + #endif + #if defined(USE_DS18B20) + features_str += "DS18B20 "; + #endif + #if defined(USE_DHT) + features_str += "DHT "; + #endif + #if defined(USE_HSU07M) + features_str += "HSU07M "; + #endif + drawStrMultiline(features_str.c_str(), 6, 180); + delay(2500); + #endif + + #if defined(DISPLAY_7735) + tft.fillScreen(TFT_BLACK); + tft.setTextColor(TFT_WHITE,TFT_BLACK); + tft.drawBitmap(0, 3, image_duco_logo_bits, 21, 24, 0xFC00); + tft.setTextSize(2); + tft.setCursor(26, 12); + #if defined(ESP8266) + tft.print("ESP8266 "); + #elif defined(CONFIG_FREERTOS_UNICORE) + tft.print("ESP32S2/C3 "); + #else + tft.print("ESP32 "); + #endif + tft.setTextSize(1); + #if defined(ESP8266) + tft.print(String(ESP.getCpuFreqMHz()).c_str()); + #else + tft.print(String(getCpuFrequencyMhz()).c_str()); + #endif + tft.print(" MHz"); + tft.setTextSize(1); + tft.setCursor(2, 70); + tft.print("Compiled "); + tft.print(__DATE__); + + delay(2500); + + tft.drawString("Features:",2, 85,1); + String features_str = "OTA "; + #if defined(USE_LAN) + features_str += "LAN "; + #endif + #if defined(LED_BLINKING) + features_str += "Blink "; + #endif + #if defined(SERIAL_PRINTING) + features_str += "Serial "; + #endif + #if defined(WEB_DASHBOARD) + features_str += "Webserver "; + #endif + #if defined(DISPLAY_16X2) + features_str += "LCD16X2 "; + #endif + #if defined(DISPLAY_SSD1306) + features_str += "SSD1306 "; + #endif + #if defined(DISPLAY_114) + features_str += "TT-GO "; + #endif + #if defined(DISPLAY_7735) + features_str += "7735 "; + #endif + #if defined(USE_INTERNAL_SENSOR) + features_str += "Int. sensor "; + #endif + #if defined(USE_DS18B20) + features_str += "DS18B20 "; + #endif + #if defined(USE_DHT) + features_str += "DHT "; + #endif + #if defined(USE_HSU07M) + features_str += "HSU07M "; + #endif + drawStrMultiline(features_str.c_str(), 2, 102); + delay(2500); + #endif + } + + void display_info(String message) { + // Abstraction layer: info screens (setups) + + #if defined(DISPLAY_SSD1306) + u8g2.clearBuffer(); + u8g2.drawXBMP(-1, 3, 41, 45, image_duco_logo_big_bits); + u8g2.setFont(u8g2_font_t0_16b_tr); + #if defined(ESP8266) + u8g2.drawStr(42, 27, "ESP8266 "); + #elif defined(CONFIG_FREERTOS_UNICORE) + u8g2.drawStr(42, 27, "ESP32S2/C3 "); + #else + u8g2.drawStr(42, 27, "ESP32 "); + #endif + u8g2.setFont(u8g2_font_t0_13b_tr); + u8g2.drawStr(41, 14, "Duino-Coin"); + u8g2.setFont(u8g2_font_6x10_tr); + u8g2.drawStr(98, 36, "MINER"); + u8g2.setFont(u8g2_font_6x13_tr); + u8g2.drawStr(1, 60, message.c_str()); + u8g2.setFont(u8g2_font_5x8_tr); + u8g2.drawStr(42, 46, "www.duinocoin.com"); + u8g2.setFont(u8g2_font_4x6_tr); + u8g2.drawStr(116, 14, String(SOFTWARE_VERSION).c_str()); + u8g2.sendBuffer(); + #endif + + #if defined(DISPLAY_16X2) + lcd.clear(); + lcd.setCursor(0, 0); + lcd.write(0); + lcd.print(" Duino-Coin "); + lcd.print(SOFTWARE_VERSION); + lcd.setCursor(0, 1); + lcd.print(message); + #endif + + #if defined(DISPLAY_114) + tft.fillScreen(TFT_BLACK); + tft.setTextColor(0xFFFF); + tft.setTextSize(2); + #if defined(ESP8266) + tft.drawString("ESP8266",54,24); + #elif defined(CONFIG_FREERTOS_UNICORE) + tft.drawString("ESP32S2/C3",54,24); + #else + tft.drawString("ESP32",54,24); + #endif + tft.setTextColor(0x4E04); + tft.drawString(String(SOFTWARE_VERSION).c_str(),180, 27); + tft.setTextColor(0xFFFF); + tft.drawString("Duino-Coin",14,69 ); + tft.drawString("MINER",156,69 ); + tft.setTextSize(1); + tft.drawString(message.c_str(),3,110 ); + tft.drawString("www.duinocoin.com",120,95 ); + tft.drawBitmap(0, 4, image_paint_0_bits, 41, 51, 0xFC00); + tft.drawString("Rev", 189, 8); + delay(2500); + #endif + + #if defined(DISPLAY_2432S08) + tft.fillScreen(TFT_BLACK); + tft.setTextColor(0xFFFF); + tft.setTextSize(2); + tft.drawRect(0, 0, 319, 239, 0xFFE0); + tft.drawBitmap(17, 20, image_duco_2_bits, 80, 74, 0xFC00); + #if defined(ESP8266) + tft.drawString("ESP8266",10,110); + #elif defined(CONFIG_FREERTOS_UNICORE) + tft.drawString("ESP32S2/C3",10,110); + #else + tft.drawString("ESP32",10,110); + #endif + tft.drawString("Duino-Coin",10,140 ); + tft.drawString("MINER",130,140 ); + tft.setTextColor(0x4E04); + tft.drawString(String(SOFTWARE_VERSION).c_str(),250, 140); + tft.setTextColor(0xFFFF); + tft.setTextSize(1); + tft.drawString("Rev", 228, 145); + tft.drawString(message.c_str(),10,160 ); + tft.drawString("www.duinocoin.com",160,220 ); + + delay(2500); + #endif + + #if defined(DISPLAY_7735) + tft.fillScreen(TFT_BLACK); + tft.setTextColor(0xFFFF); + tft.setTextSize(2); + #if defined(ESP8266) + tft.drawString("ESP8266",25,12); + #elif defined(CONFIG_FREERTOS_UNICORE) + tft.drawString("ESP32S2/C3",25,12); + #else + tft.drawString("ESP32",25,12); + #endif + tft.drawBitmap(0, 3, image_duco_logo_bits, 21, 24, 0xFC00); + tft.setTextSize(1); + tft.setTextColor(0x4E04); + tft.drawString("Rev", 115, 8); + tft.drawString(String(SOFTWARE_VERSION).c_str(),115, 20); + tft.setTextColor(0xFFFF); + tft.drawString("Duino-Coin",4,60 ); + tft.drawString("MINER",75,60 ); + tft.setTextSize(1); + tft.drawString(message.c_str(),3,110 ); + tft.drawString("www.duinocoin.com",40,87 ); + delay(2500); + #endif + } + + void display_background() { + tft.fillScreen(TFT_BLACK); + tft.drawRect(0, 0, 319, 239, 0xFFE0); + tft.drawBitmap(17, 14, image_duco_2_bits, 80, 74, 0xFC00); + tft.drawLine(1, 98, 318, 98, 0xFFE0); + tft.drawLine(116, 61, 116, 0, 0xFFE0); + tft.drawRect(116, 60, 203, 39, 0xFFE0); + tft.fillRect(160, 166, 154, 68, 0xFFE0); + tft.setTextColor(0x7E0); + tft.setFreeFont(); + tft.setTextSize(1); + tft.drawString("Balance", 121, 64); + tft.drawString("Miners", 278, 64); + tft.drawString("Bal update every 5 m", 195,100); + tft.setTextColor(0x0); + tft.setTextSize(1); + tft.setFreeFont(&FreeSansBold9pt7b); + tft.drawString("kH/s", 272, 217); + tft.drawBitmap(8, 173, image_check_big_bits, 20, 17, 0x7E0); + tft.drawBitmap(121, 9, image_clock_bits, 15, 16, 0x41F); + tft.drawLine(160, 166, 1, 166, 0xFFE0); + tft.drawBitmap(11, 193, image_hand_thumbs_up_bits, 16, 16, 0x7E0); + tft.drawBitmap(270, 9, image_music_radio_broadcast_bits, 15, 16, 0x41F); + tft.setTextColor(0x7E0); + tft.setTextSize(1); + tft.setFreeFont(); + tft.drawString("Diff:", 7, 219); + tft.drawString("shares/s", 108, 219); + tft.drawString("shares", 121, 177); + tft.drawString("accepted", 107, 199); + tft.setTextColor(0x408); + tft.drawString("Node:", 121, 46); + tft.drawString("Ping:", 246, 47); + tft.setTextColor(0x41F); + tft.drawString("IP:", 207, 32); + tft.setTextColor(0x400); + tft.drawString(String(SOFTWARE_VERSION).c_str(),79,15 ); + tft.setTextColor(0xAD55); + tft.drawString("Miner Uptime", 147, 8); + tft.drawString("ESP32 ", 121, 32); + tft.drawString(String(getCpuFrequencyMhz()).c_str(),157,32); + tft.drawString("MHz",180,32); + } + + void display_balance(String result_balance_balance,String total_miner, String result_balance_username){ + + #if defined(DISPLAY_2432S08) + + tft.setFreeFont(); + tft.setTextSize(1); + tft.setTextColor(0x410); + if (first_display == true) { tft.drawString(result_balance_username,168,64); + first_display = false; + } + // tft.fillRect(168, 64, 108, 9, 0x0); + // tft.drawString((result_balance_username),168,64); + tft.setTextColor(0xC618); + tft.setTextSize(2); + tft.fillRect(120, 76, 156, 16, 0x0); + tft.drawString(result_balance_balance + " D", 120, 76); + tft.fillRect(278, 76, 35, 16, 0x0); + tft.setTextColor(0x4410); + tft.drawString(total_miner, 278, 76); + #else {} + #endif + } + + void display_time(String(mytime)){ + + #if defined(DISPLAY_2432S08) + + tft.setFreeFont(&FreeSans18pt7b); + tft.setTextColor(0xFFFF); + tft.setTextSize(2); + tft.fillRect(19, 104, 170, 55, 0x0); + tft.drawString(mytime, 19, 104); + tft.setFreeFont(); + tft.setTextSize(1); + tft.fillRect(215, 120, 100, 40, 0x0); + tft.drawString(myday, 215,120); + tft.drawString(mydate, 215, 135); + #else {} + #endif + } + + void display_mining_results(String hashrate, String accepted_shares, String total_shares, String uptime, String node, + String difficulty, String sharerate, String ping, String accept_rate) { + // Ran after each found share + // Abstraction layer: displaying mining results + Serial.println("Displaying mining results"); + + #if defined(DISPLAY_SSD1306) + u8g2.clearBuffer(); + u8g2.setFont(u8g2_font_profont10_tr); + u8g2.drawStr(67, 26, "kH"); + if (hashrate.toFloat() < 100.0) { + u8g2.setFont(u8g2_font_profont29_tr); + u8g2.drawStr(2, 36, hashrate.c_str()); + } else { + u8g2.setFont(u8g2_font_profont22_tr); + u8g2.drawStr(3, 35, hashrate.c_str()); + } + + u8g2.setFont(u8g2_font_haxrcorp4089_tr); + u8g2.drawStr(52, 12, node.c_str()); + + u8g2.setFont(u8g2_font_t0_11_tr); + u8g2.drawStr(17, 47, (accepted_shares + "/" + total_shares).c_str()); + u8g2.setFont(u8g2_font_5x7_tr); + u8g2.drawStr(88, 47, ("(" + accept_rate + "%)").c_str()); + + u8g2.setFont(u8g2_font_profont12_tr); + u8g2.drawStr(20, 12, (ping + "ms").c_str()); + u8g2.drawStr(69, 36, "s"); + + u8g2.setFont(u8g2_font_6x13_tr); + u8g2.drawStr(125-u8g2.getStrWidth(uptime.c_str()), 61, uptime.c_str()); + + u8g2.drawStr(85, 38, sharerate.c_str()); + u8g2.drawStr(85, 27, difficulty.c_str()); + u8g2.drawLine(67, 28, 75, 28); + + u8g2.drawXBMP(2, 38, 13, 10, image_check_contour_bits); + + if (WiFi.RSSI() > -40) { + u8g2.drawXBMP(1, 0, 15, 16, image_network_4_bars_bits); + } else if (WiFi.RSSI() > -60) { + u8g2.drawXBMP(1, 0, 15, 16, image_network_3_bars_bits); + } else if (WiFi.RSSI() > -75) { + u8g2.drawXBMP(1, 0, 15, 16, image_network_2_bars_bits); + } else { + u8g2.drawXBMP(1, 0, 15, 16, image_network_1_bar_bits); + } + + u8g2.setFont(u8g2_font_4x6_tr); + u8g2.drawStr(14, 61, String(WiFi.localIP().toString()).c_str()); + u8g2.drawStr(14, 55, ("Duino-Coin " + String(SOFTWARE_VERSION)).c_str()); + u8g2.drawXBMP(2, 11, 9, 50, image_duco_logo_bits); + u8g2.drawStr(111, 27, "diff"); + u8g2.drawStr(107, 38, "shr/s"); + + u8g2.sendBuffer(); + #endif + + #if defined(DISPLAY_16X2) + lcd.clear(); + lcd.setCursor(0,0); + lcd.print(hashrate); + lcd.setCursor(4,0); + lcd.write(2); // kh + + lcd.setCursor(7, 0); + lcd.print(difficulty); + lcd.print(" diff"); + + lcd.setCursor(0, 1); + lcd.write(1); // checkmark + lcd.print(accepted_shares); + + lcd.setCursor(7, 1); + lcd.print(ping); + lcd.write(3); // ms + + lcd.setCursor(12, 1); + lcd.print(sharerate); + lcd.print("s"); + #endif + + #if defined(DISPLAY_114) + tft.fillScreen(TFT_BLACK); + tft.drawBitmap(2, 2, image_duco_square_bits, 56, 69, 0xFC00); + tft.setFreeFont(); + tft.setTextSize(3); + tft.setTextColor(TFT_WHITE,TFT_BLACK); + if (hashrate.toFloat() < 100.0) { + tft.drawString(hashrate.c_str(), 120, 10 ); + } else { + tft.drawString(hashrate.c_str(), 120, 10 ); + } + tft.drawLine(115, 40, 238, 40, 0x418); + tft.setTextSize(1); + tft.drawString("kH/s", 211, 24); + tft.setTextColor(0x400); + tft.drawString("Rev", 66, 8); + tft.setTextSize(2); + tft.drawString(String(SOFTWARE_VERSION).c_str(),66,22 ); + tft.drawLine(0, 75, 240, 75, 0x41F); + tft.setTextColor(0xFC00); + tft.drawString("uinocoin MINER", 61, 57); + tft.drawBitmap(1, 76, image_check_contour_bits, 13, 16, 0x7E8); + tft.setTextColor(0x410); + tft.setTextSize(1); + tft.drawString((accepted_shares + "/" + total_shares).c_str(),18,81 ); + tft.drawLine(240, 41, 114, 41, 0x15); + tft.setTextColor(0xAD55); + tft.drawString("IP:", 130, 44); + tft.setTextColor(0x8430); + tft.drawString(String(WiFi.localIP().toString()).c_str(),149, 44); + tft.drawBitmap(6, 92, image_ButtonRight_bits, 4, 7, 0x7E0); + tft.setTextColor(0x410); + tft.drawString("acc. sh", 69, 92); + tft.drawString(("(" + accept_rate + "%)").c_str(),18, 92 ); + tft.drawLine(119, 76, 119, 134, 0x41F); + tft.drawLine(237, 39, 116, 39, 0x57FF); + tft.drawLine(0, 105, 240, 105, 0x41F); + tft.setTextColor(0xA800); + tft.drawString("diff:", 125, 81); + tft.setTextColor(0xFFE0); + tft.drawString("sh/s:", 125, 92); + tft.setTextColor(0xAD55); + tft.drawString(sharerate.c_str(),164, 92 ); + tft.drawString(difficulty.c_str(),162, 81 ); + tft.drawString("Wifi", 199, 81); + tft.drawString("Ping", 123, 123); + tft.setTextColor(0x8410); + tft.drawString("Miner Uptime", 28, 110); + tft.drawString(uptime.c_str(),35, 122 ); + tft.drawBitmap(5.5, 112, image_clock_bits, 15, 16, 0x8410); + tft.drawLine(115, 40, 115, 1, 0x418); + tft.setTextColor(0xAD55); + tft.drawString("ESP32", 69, 44); + tft.drawLine(116, 39, 116, 2, 0x7FF); + tft.drawString("Node", 123, 111); + tft.drawLine(114, 41, 114, 0, 0x15); + tft.setTextColor(0x4666); + tft.drawString(node.c_str(), 149, 111); + tft.drawString((ping + "ms").c_str(),156, 123 ); + tft.drawLine(117, 2, 237, 2, 0x87FF); + tft.drawLine(115, 1, 238, 1, 0x418); + tft.drawLine(237, 2, 237, 39, 0x87FF); + tft.drawLine(238, 2, 238, 39, 0x418); + tft.drawLine(115, 0, 239, 0, 0x15); + tft.drawLine(239, 1, 239, 40, 0x15); + + if (WiFi.RSSI() > -40) { + tft.drawBitmap(218, 87, image_network_4_bars_bits, 15, 16, 0xFFFF); + } else if (WiFi.RSSI() > -60) { + tft.drawBitmap(218, 86, image_network_3_bars_bits, 15, 16, 0xFFFF); + } else if (WiFi.RSSI() > -75) { + tft.drawBitmap(218, 86, image_network_2_bars_bits, 15, 16, 0xFFFF); + } else { + tft.drawBitmap(218, 86, image_network_1_bar_bits, 15, 16, 0xFFFF); + } + + + #endif + + #if defined(DISPLAY_2432S08) + + tft.setFreeFont(); + tft.setTextColor(0x0); + tft.setTextSize(4); + tft.fillRect(173, 183, 116, 28, 0xFFE0); + if (hashrate.toFloat() < 100.0) { + tft.drawString(hashrate.c_str(), 188, 183 ); + } else { + tft.drawString(hashrate.c_str(), 173, 183 ); + } + tft.setTextSize(1); + tft.setTextColor(0x41F); + tft.setFreeFont(); + tft.fillRect(39, 177, 78, 7, 0x0); + tft.drawString((accepted_shares + "/" + total_shares).c_str(),39,177 ); + tft.fillRect(39, 199, 65, 7, 0x0); + tft.drawString(("(" + accept_rate + "%)").c_str(),39, 199 ); + tft.setTextColor(0xF800); + tft.fillRect(78, 219, 25, 7, 0x0); + tft.drawString(sharerate.c_str(),78, 219 ); + tft.fillRect(40, 219, 27, 7, 0x0); + tft.drawString(difficulty.c_str(),40, 219 ); + tft.setTextColor(0xAD55); + tft.fillRect(228, 32, 70, 7, 0x0); + tft.drawString(String(WiFi.localIP().toString()).c_str(),228, 32); + tft.fillRect(145, 18, 85, 7, 0x0); + tft.drawString(uptime.c_str(),160, 18); + tft.drawString(node.c_str(), 155, 46); + tft.fillRect(279, 47, 37, 7, 0x0); + tft.drawString((ping + "ms").c_str(),279, 47 ); + + + + tft.fillRect(290, 8, 15, 16, 0x0); + if (WiFi.RSSI() > -40) { + tft.drawBitmap(290, 8, image_network_4_bars_bits, 15, 16, 0xFFFF); + } else if (WiFi.RSSI() > -60) { + tft.drawBitmap(290, 8, image_network_3_bars_bits, 15, 16, 0xFFFF); + } else if (WiFi.RSSI() > -75) { + tft.drawBitmap(290, 8, image_network_2_bars_bits, 15, 16, 0xFFFF); + } else { + tft.drawBitmap(290, 8, image_network_1_bar_bits, 15, 16, 0xFFFF); + } + + #endif + + #if defined(DISPLAY_7735) + tft.fillScreen(TFT_BLACK); + tft.drawBitmap(-2, 6, image_duco_bits, 35, 35, 0xFC00); + tft.setTextColor(0xFC00); + tft.setTextSize(1); + tft.setFreeFont(); + tft.drawString("uinocoin MINER", 30, 32); + tft.setTextColor(0x540); + tft.drawString("Rev", 118, 32); + + tft.setTextColor(0xFC00); + tft.setFreeFont(); + tft.drawString("duinocoin.com", 73, 78); + tft.setFreeFont(&FreeSansBold12pt7b); + // tft.setTextSize(3); + tft.fillRect(66, 4, 89, 21, 0xFD60); + tft.setTextColor(0x0); + if (hashrate.toFloat() < 100.0) { + tft.drawString(hashrate.c_str(), 67, 5 ); + } else { + tft.drawString(hashrate.c_str(), 67, 5 ); + } + + tft.setFreeFont(); + tft.setTextSize(1); + //tft.setTextColor(0x0); + tft.drawString("kH/s", 129, 16); + tft.setTextColor(0x400); + tft.drawString(String(SOFTWARE_VERSION).c_str(),138,32 ); + tft.drawRect(64, 3, 92, 24, 0xB208); + tft.setTextColor(0xA815); + tft.drawString("ESP 32", 3, 45); + tft.drawRect(63, 2, 94, 26, 0xD2EB); + tft.drawLine(0, 58, 0, 126, 0xFFEA); + tft.drawRect(0, 0, 160, 58, 0xFFEA); + tft.drawLine(0, 0, 0, 0, 0xFFFF); + tft.drawLine(159, 126, 159, 58, 0xFFEA); + tft.drawRect(62, 1, 96, 28, 0xFAAA); + tft.setTextColor(0xAD55); + tft.drawString("Ip:", 52, 45); + tft.drawString(String(WiFi.localIP().toString()).c_str(),72, 45); + + tft.drawBitmap(3, 59, image_check_contour_bits, 13, 16, 0x540); + tft.setTextColor(0x52BF); + tft.drawString((accepted_shares + "/" + total_shares).c_str(),17,64 ); + + tft.drawBitmap(99, 65, image_ButtonRightSmall_bits, 3, 5, 0x540); + tft.drawString(("(" + accept_rate + "%)").c_str(),107, 64 ); + + tft.drawLine(0, 75, 170, 75, 0xFFEA); + tft.setTextColor(0xAD55); + tft.drawString("wifi", 35, 20); + tft.setTextColor(0xA800); + tft.drawString("diff.", 3, 79); + tft.setTextColor(0x57EA); + tft.drawString("sh/s:", 3, 90); + tft.setTextColor(0xAD55); + tft.drawString(difficulty.c_str(),35, 79 ); + + tft.drawString(sharerate.c_str(),38, 90 ); + + tft.drawBitmap(71, 92, image_clock_bits, 15, 16, 0x540); + tft.drawString(uptime.c_str(),92, 96 ); + + tft.drawLine(-2, 126, 169, 126, 0xFFEA); + tft.setTextColor(0x555); + tft.drawString("Ping:", 3, 103); + tft.setTextColor(0xAD55); + tft.drawString((ping + "ms").c_str(),35, 103 ); + + tft.setTextColor(0x211D); + tft.drawString("Node:", 3, 115); + tft.setTextColor(0xAD55); + tft.drawString(node.c_str(), 35, 115); + + + if (WiFi.RSSI() > -40) { + tft.drawRect(51, 5, 2, 11, 0xFFFF); + tft.drawRect(47, 7, 2, 9, 0xFFFF); + tft.drawRect(43, 10, 2, 6, 0xFFFF); + tft.drawRect(39, 13, 2, 3, 0xFFFF); + } else if (WiFi.RSSI() > -60) { + tft.drawRect(47, 7, 2, 9, 0xFFFF); + tft.drawRect(43, 10, 2, 6, 0xFFFF); + tft.drawRect(39, 13, 2, 3, 0xFFFF); + } else if (WiFi.RSSI() > -75) { + tft.drawRect(43, 10, 2, 6, 0xFFFF); + tft.drawRect(39, 13, 2, 3, 0xFFFF); + } else { + tft.drawRect(39, 13, 2, 3, 0xFFFF); + } + + + #endif + } +#endif + +#endif diff --git a/ESP_Code/ESP_Code.ino b/ESP_Code/ESP_Code.ino index e0fa6171..638caa5e 100644 --- a/ESP_Code/ESP_Code.ino +++ b/ESP_Code/ESP_Code.ino @@ -1,817 +1,995 @@ -/* - ____ __ __ ____ _ _ _____ ___ _____ ____ _ _ - ( _ \( )( )(_ _)( \( )( _ )___ / __)( _ )(_ _)( \( ) - )(_) ))(__)( _)(_ ) ( )(_)((___)( (__ )(_)( _)(_ ) ( - (____/(______)(____)(_)\_)(_____) \___)(_____)(____)(_)\_) - Official code for all ESP8266/32 boards version 4.3 - Main .ino file - - The Duino-Coin Team & Community 2019-2024 © MIT Licensed - https://duinocoin.com - https://github.com/revoxhere/duino-coin - - If you don't know where to start, visit official website and navigate to - the Getting Started page. Have fun mining! - - To edit the variables (username, WiFi settings, etc.) use the Settings.h tab! -*/ - -/* If optimizations cause problems, change them to -O0 (the default) */ -#pragma GCC optimize("-Ofast") - -/* If during compilation the line below causes a - "fatal error: arduinoJson.h: No such file or directory" - message to occur; it means that you do NOT have the - ArduinoJSON library installed. To install it, - go to the below link and follow the instructions: - https://github.com/revoxhere/duino-coin/issues/832 */ -#include - -#if defined(ESP8266) - #include - #include - #include - #include -#else - #include - #include - #include - #include - #include -#endif - -#include -#include -#include -#include -#include -#include -#include - -#include "MiningJob.h" -#include "Settings.h" - -#ifdef USE_LAN - #include -#endif - -#if defined(WEB_DASHBOARD) - #include "Dashboard.h" -#endif - -#if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - #include "DisplayHal.h" -#endif - -#if !defined(ESP8266) && defined(DISABLE_BROWNOUT) - #include "soc/soc.h" - #include "soc/rtc_cntl_reg.h" -#endif - -// Auto adjust physical core count -// (ESP32-S2/C3 have 1 core, ESP32 has 2 cores, ESP8266 has 1 core) -#if defined(ESP8266) - #define CORE 1 - typedef ESP8266WebServer WebServer; -#elif defined(CONFIG_FREERTOS_UNICORE) - #define CORE 1 -#else - #define CORE 2 - // Install TridentTD_EasyFreeRTOS32 if you get an error - #include - - void Task1Code( void * parameter ); - void Task2Code( void * parameter ); - TaskHandle_t Task1; - TaskHandle_t Task2; -#endif - -#if defined(WEB_DASHBOARD) - WebServer server(80); -#endif - -#if defined(CAPTIVE_PORTAL) - #include // This needs to be first, or it all crashes and burns... - #include - #include - char duco_username[40]; - char duco_password[40]; - char duco_rigid[24]; - WiFiManager wifiManager; - Preferences preferences; - WiFiManagerParameter custom_duco_username("duco_usr", "Duino-Coin username", duco_username, 40); - WiFiManagerParameter custom_duco_password("duco_pwd", "Duino-Coin mining key (if enabled in the wallet)", duco_password, 40); - WiFiManagerParameter custom_duco_rigid("duco_rig", "Custom miner identifier (optional)", duco_rigid, 24); - - void saveConfigCallback() { - preferences.begin("duino_config", false); - preferences.putString("duco_username", custom_duco_username.getValue()); - preferences.putString("duco_password", custom_duco_password.getValue()); - preferences.putString("duco_rigid", custom_duco_rigid.getValue()); - preferences.end(); - RestartESP("Settings saved"); - } - - void reset_settings() { - server.send(200, "text/html", "Settings have been erased. Please redo the configuration by connecting to the WiFi network that will be created"); - delay(500); - wifiManager.resetSettings(); - RestartESP("Manual settings reset"); - } - - void saveParamCallback(){ - Serial.println("[CALLBACK] saveParamCallback fired"); - Serial.println("PARAM customfieldid = " + getParam("customfieldid")); - } - - String getParam(String name){ - //read parameter from server, for customhmtl input - String value; - if(wifiManager.server->hasArg(name)) { - value = wifiManager.server->arg(name); - } - return value; - } -#endif - -void RestartESP(String msg) { - #if defined(SERIAL_PRINTING) - Serial.println(msg); - Serial.println("Restarting ESP..."); - #endif - - #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - display_info("Restarting ESP..."); - #endif - - #if defined(ESP8266) - ESP.reset(); - #else - ESP.restart(); - abort(); - #endif -} - -#if defined(BLUSHYBOX) - Ticker blinker; - bool lastLedState = false; - void changeState() { - analogWrite(LED_BUILTIN, lastLedState ? 255 : 0); - lastLedState = !lastLedState; - } -#endif - -#if defined(ESP8266) - // WDT Loop - // See lwdtcb() and lwdtFeed() below - Ticker lwdTimer; - - unsigned long lwdCurrentMillis = 0; - unsigned long lwdTimeOutMillis = LWD_TIMEOUT; - - void ICACHE_RAM_ATTR lwdtcb(void) { - if ((millis() - lwdCurrentMillis > LWD_TIMEOUT) || (lwdTimeOutMillis - lwdCurrentMillis != LWD_TIMEOUT)) - RestartESP("Loop WDT Failed!"); - } - - void lwdtFeed(void) { - lwdCurrentMillis = millis(); - lwdTimeOutMillis = lwdCurrentMillis + LWD_TIMEOUT; - } -#else - void lwdtFeed(void) { - Serial.println("lwdtFeed()"); - } -#endif - -namespace { - MiningConfig *configuration = new MiningConfig( - DUCO_USER, - RIG_IDENTIFIER, - MINER_KEY - ); - - #if defined(ESP32) && CORE == 2 - EasyMutex mutexClientData, mutexConnectToServer; - #endif - - #ifdef USE_LAN - static bool eth_connected = false; - #endif - - void UpdateHostPort(String input) { - // Thanks @ricaun for the code - DynamicJsonDocument doc(256); - deserializeJson(doc, input); - const char *name = doc["name"]; - - configuration->host = doc["ip"].as().c_str(); - configuration->port = doc["port"].as(); - node_id = String(name); - - #if defined(SERIAL_PRINTING) - Serial.println("Poolpicker selected the best mining node: " + node_id); - #endif - - #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - display_info(node_id); - #endif - } - - void VerifyWifi() { - #ifdef USE_LAN - while ((!eth_connected) || (ETH.localIP() == IPAddress(0, 0, 0, 0))) { - #if defined(SERIAL_PRINTING) - Serial.println("Ethernet connection lost. Reconnect..." ); - #endif - SetupWifi(); - } - #else - while (WiFi.status() != WL_CONNECTED - || WiFi.localIP() == IPAddress(0, 0, 0, 0) - || WiFi.localIP() == IPAddress(192, 168, 4, 2) - || WiFi.localIP() == IPAddress(192, 168, 4, 3)) { - #if defined(SERIAL_PRINTING) - Serial.println("WiFi reconnecting..."); - #endif - WiFi.disconnect(); - delay(500); - WiFi.reconnect(); - delay(500); - } - #endif - } - - String httpGetString(String URL) { - String payload = ""; - - WiFiClientSecure client; - HTTPClient https; - client.setInsecure(); - - https.begin(client, URL); - https.addHeader("Accept", "*/*"); - - int httpCode = https.GET(); - #if defined(SERIAL_PRINTING) - Serial.printf("HTTP Response code: %d\n", httpCode); - #endif - - if (httpCode == HTTP_CODE_OK || httpCode == HTTP_CODE_MOVED_PERMANENTLY) { - payload = https.getString(); - } else { - #if defined(SERIAL_PRINTING) - Serial.printf("Error fetching node from poolpicker: %s\n", https.errorToString(httpCode).c_str()); - VerifyWifi(); - #endif - #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - display_info(http.errorToString(httpCode)); - #endif - } - https.end(); - return payload; - } - - void SelectNode() { - String input = ""; - int waitTime = 1; - int poolIndex = 0; - - while (input == "") { - #if defined(SERIAL_PRINTING) - Serial.println("Fetching mining node from the poolpicker in " + String(waitTime) + "s"); - #endif - delay(waitTime * 1000); - - input = httpGetString("https://server.duinocoin.com/getPool"); - - // Increase wait time till a maximum of 32 seconds - // (addresses: Limit connection requests on failure in ESP boards #1041) - waitTime *= 2; - if (waitTime > 32) - RestartESP("Node fetch unavailable"); - } - - UpdateHostPort(input); - } - - #ifdef USE_LAN - void WiFiEvent(WiFiEvent_t event) { - switch (event) { - case ARDUINO_EVENT_ETH_START: - #if defined(SERIAL_PRINTING) - Serial.println("ETH Started"); - #endif - // The hostname must be set after the interface is started, but needs - // to be set before DHCP, so set it from the event handler thread. - ETH.setHostname("esp32-ethernet"); - break; - case ARDUINO_EVENT_ETH_CONNECTED: - #if defined(SERIAL_PRINTING) - Serial.println("ETH Connected"); - #endif - break; - case ARDUINO_EVENT_ETH_GOT_IP: - #if defined(SERIAL_PRINTING) - Serial.println("ETH Got IP"); - #endif - eth_connected = true; - break; - case ARDUINO_EVENT_ETH_DISCONNECTED: - #if defined(SERIAL_PRINTING) - Serial.println("ETH Disconnected"); - #endif - eth_connected = false; - break; - case ARDUINO_EVENT_ETH_STOP: - #if defined(SERIAL_PRINTING) - Serial.println("ETH Stopped"); - #endif - eth_connected = false; - break; - default: - break; - } - } - #endif - - void SetupWifi() { - #ifdef USE_LAN - #if defined(SERIAL_PRINTING) - Serial.println("Connecting to Ethernet..."); - #endif - WiFi.onEvent(WiFiEvent); // Will call WiFiEvent() from another thread. - ETH.begin(); - - while (!eth_connected) { - delay(500); - #if defined(SERIAL_PRINTING) - Serial.print("."); - #endif - } - - #if defined(SERIAL_PRINTING) - Serial.println("\n\nSuccessfully connected to Ethernet"); - Serial.println("Local IP address: " + ETH.localIP().toString()); - Serial.println("Rig name: " + String(RIG_IDENTIFIER)); - Serial.println(); - #endif - - #else - #if defined(SERIAL_PRINTING) - Serial.println("Connecting to: " + String(SSID)); - #endif - - WiFi.begin(SSID, PASSWORD); - while(WiFi.status() != WL_CONNECTED) { - Serial.print("."); - delay(100); - } - VerifyWifi(); - - #if !defined(ESP8266) - WiFi.config(WiFi.localIP(), WiFi.gatewayIP(), WiFi.subnetMask(), DNS_SERVER); - #endif - - #if defined(SERIAL_PRINTING) - Serial.println("\n\nSuccessfully connected to WiFi"); - Serial.println("Rig name: " + String(RIG_IDENTIFIER)); - Serial.println("Local IP address: " + WiFi.localIP().toString()); - Serial.println("Gateway: " + WiFi.gatewayIP().toString()); - Serial.println("DNS: " + WiFi.dnsIP().toString()); - Serial.println(); - #endif - - #endif - - #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - display_info("Waiting for node..."); - #endif - SelectNode(); - } - - void SetupOTA() { - // Prepare OTA handler - ArduinoOTA.onStart([]() - { - #if defined(SERIAL_PRINTING) - Serial.println("Start"); - #endif - }); - ArduinoOTA.onEnd([]() - { - #if defined(SERIAL_PRINTING) - Serial.println("\nEnd"); - #endif - }); - ArduinoOTA.onProgress([](unsigned int progress, unsigned int total) - { - #if defined(SERIAL_PRINTING) - Serial.printf("Progress: %u%%\r", (progress / (total / 100))); - #endif - }); - ArduinoOTA.onError([](ota_error_t error) - { - Serial.printf("Error[%u]: ", error); - #if defined(SERIAL_PRINTING) - if (error == OTA_AUTH_ERROR) Serial.println("Auth Failed"); - else if (error == OTA_BEGIN_ERROR) Serial.println("Begin Failed"); - else if (error == OTA_CONNECT_ERROR) Serial.println("Connect Failed"); - else if (error == OTA_RECEIVE_ERROR) Serial.println("Receive Failed"); - else if (error == OTA_END_ERROR) Serial.println("End Failed"); - #endif - }); - - ArduinoOTA.setHostname(RIG_IDENTIFIER); // Give port a name - ArduinoOTA.begin(); - } - - #if defined(WEB_DASHBOARD) - void dashboard() { - #if defined(SERIAL_PRINTING) - Serial.println("Handling HTTP client"); - #endif - String s = WEBSITE; - #ifdef USE_LAN - s.replace("@@IP_ADDR@@", ETH.localIP().toString()); - #else - s.replace("@@IP_ADDR@@", WiFi.localIP().toString()); - #endif - - s.replace("@@HASHRATE@@", String((hashrate+hashrate_core_two) / 1000)); - s.replace("@@DIFF@@", String(difficulty / 100)); - s.replace("@@SHARES@@", String(share_count)); - s.replace("@@NODE@@", String(node_id)); - - #if defined(ESP8266) - s.replace("@@DEVICE@@", "ESP8266"); - #elif defined(CONFIG_FREERTOS_UNICORE) - s.replace("@@DEVICE@@", "ESP32-S2/C3"); - #else - s.replace("@@DEVICE@@", "ESP32"); - #endif - - s.replace("@@ID@@", String(RIG_IDENTIFIER)); - s.replace("@@MEMORY@@", String(ESP.getFreeHeap())); - s.replace("@@VERSION@@", String(SOFTWARE_VERSION)); - - #if defined(CAPTIVE_PORTAL) - s.replace("@@RESET_SETTINGS@@", "• Reset settings"); - #else - s.replace("@@RESET_SETTINGS@@", ""); - #endif - - #if defined(USE_DS18B20) - sensors.requestTemperatures(); - float temp = sensors.getTempCByIndex(0); - s.replace("@@SENSOR@@", "DS18B20: " + String(temp) + "*C"); - #elif defined(USE_DHT) - float temp = dht.readTemperature(); - float hum = dht.readHumidity(); - s.replace("@@SENSOR@@", "DHT11/22: " + String(temp) + "*C, " + String(hum) + "rh%"); - #elif defined(USE_HSU07M) - float temp = read_hsu07m(); - s.replace("@@SENSOR@@", "HSU07M: " + String(temp) + "*C"); - #elif defined(USE_INTERNAL_SENSOR) - float temp = 0; - temp_sensor_read_celsius(&temp); - s.replace("@@SENSOR@@", "CPU: " + String(temp) + "*C"); - #else - s.replace("@@SENSOR@@", "None"); - #endif - - server.send(200, "text/html", s); - } - #endif - -} // End of namespace - -MiningJob *job[CORE]; - -#if CORE == 2 - EasyFreeRTOS32 task1, task2; -#endif - -void task1_func(void *) { - #if defined(ESP32) && CORE == 2 - VOID SETUP() { } - - VOID LOOP() { - job[0]->mine(); - - #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - float hashrate_float = (hashrate+hashrate_core_two) / 1000.0; - float accept_rate = (accepted_share_count / 0.01 / share_count); - - long millisecs = millis(); - int uptime_secs = int((millisecs / 1000) % 60); - int uptime_mins = int((millisecs / (1000 * 60)) % 60); - int uptime_hours = int((millisecs / (1000 * 60 * 60)) % 24); - String uptime = String(uptime_hours) + "h" + String(uptime_mins) + "m" + String(uptime_secs) + "s"; - - float sharerate = share_count / (millisecs / 1000.0); - - display_mining_results(String(hashrate_float, 1), String(accepted_share_count), String(share_count), String(uptime), - String(node_id), String(difficulty / 100), String(sharerate, 1), - String(ping), String(accept_rate, 1)); - #endif - } - #endif -} - -void task2_func(void *) { - #if defined(ESP32) && CORE == 2 - VOID SETUP() { - job[1] = new MiningJob(1, configuration); - } - - VOID LOOP() { - job[1]->mine(); - - #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - float hashrate_float = (hashrate+hashrate_core_two) / 1000.0; - float accept_rate = (accepted_share_count / 0.01 / share_count); - - long millisecs = millis(); - int uptime_secs = int((millisecs / 1000) % 60); - int uptime_mins = int((millisecs / (1000 * 60)) % 60); - int uptime_hours = int((millisecs / (1000 * 60 * 60)) % 24); - String uptime = String(uptime_hours) + "h" + String(uptime_mins) + "m" + String(uptime_secs) + "s"; - - float sharerate = share_count / (millisecs / 1000.0); - - display_mining_results(String(hashrate_float, 1), String(accepted_share_count), String(share_count), String(uptime), - String(node_id), String(difficulty / 100), String(sharerate, 1), - String(ping), String(accept_rate, 1)); - #endif - } - #endif -} - -void setup() { - #if !defined(ESP8266) && defined(DISABLE_BROWNOUT) - WRITE_PERI_REG(RTC_CNTL_BROWN_OUT_REG, 0); - #endif - - #if defined(SERIAL_PRINTING) - Serial.begin(SERIAL_BAUDRATE); - Serial.println("\n\nDuino-Coin " + String(configuration->MINER_VER)); - #endif - pinMode(LED_BUILTIN, OUTPUT); - - #if defined(BLUSHYBOX) - analogWrite(LED_BUILTIN, 255); - for (int i = 255; i > 0; i--) { - analogWrite(LED_BUILTIN, i); - delay(1); - } - pinMode(GAUGE_PIN, OUTPUT); - - // Gauge up and down effect on startup - for (int i = GAUGE_MIN; i < GAUGE_MAX; i++) { - analogWrite(GAUGE_PIN, i); - delay(10); - } - for (int i = GAUGE_MAX; i > GAUGE_MIN; i--) { - analogWrite(GAUGE_PIN, i); - delay(10); - } - #endif - - #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - screen_setup(); - display_boot(); - delay(2500); - #endif - - assert(CORE == 1 || CORE == 2); - WALLET_ID = String(random(0, 2811)); // Needed for miner grouping in the wallet - job[0] = new MiningJob(0, configuration); - - #if defined(USE_DHT) - #if defined(SERIAL_PRINTING) - Serial.println("Initializing DHT sensor (Duino IoT)"); - #endif - dht.begin(); - #if defined(SERIAL_PRINTING) - Serial.println("Test reading: " + String(dht.readHumidity()) + "% humidity"); - Serial.println("Test reading: temperature " + String(dht.readTemperature()) + "°C"); - #endif - #endif - - #if defined(USE_DS18B20) - #if defined(SERIAL_PRINTING) - Serial.println("Initializing DS18B20 sensor (Duino IoT)"); - #endif - sensors.begin(); - sensors.requestTemperatures(); - #if defined(SERIAL_PRINTING) - Serial.println("Test reading: " + String(sensors.getTempCByIndex(0)) + "°C"); - #endif - #endif - - #if defined(USE_HSU07M) - #if defined(SERIAL_PRINTING) - Serial.println("Initializing HSU07M sensor (Duino IoT)"); - Serial.println("Test reading: " + String(read_hsu07m()) + "°C"); - #endif - #endif - - #if defined(USE_INTERNAL_SENSOR) - #if defined(SERIAL_PRINTING) - Serial.println("Initializing internal ESP32 temperature sensor (Duino IoT)"); - #endif - temp_sensor_config_t temp_sensor = TSENS_CONFIG_DEFAULT(); - temp_sensor.dac_offset = TSENS_DAC_L2; - temp_sensor_set_config(temp_sensor); - temp_sensor_start(); - float result = 0; - temp_sensor_read_celsius(&result); - #if defined(SERIAL_PRINTING) - Serial.println("Test reading: " + String(result) + "°C"); - #endif - #endif - - WiFi.mode(WIFI_STA); // Setup ESP in client mode - //WiFi.disconnect(true); - #if defined(ESP8266) - WiFi.setSleepMode(WIFI_NONE_SLEEP); - #else - WiFi.setSleep(false); - #endif - - #if defined(CAPTIVE_PORTAL) - preferences.begin("duino_config", false); - strcpy(duco_username, preferences.getString("duco_username", "username").c_str()); - strcpy(duco_password, preferences.getString("duco_password", "None").c_str()); - strcpy(duco_rigid, preferences.getString("duco_rigid", "None").c_str()); - preferences.end(); - configuration->DUCO_USER = duco_username; - configuration->RIG_IDENTIFIER = duco_rigid; - configuration->MINER_KEY = duco_password; - - String captivePortalHTML = R"( - Duino BlushyBox - - )"; - - wifiManager.setCustomHeadElement(captivePortalHTML.c_str()); - - wifiManager.setSaveConfigCallback(saveConfigCallback); - wifiManager.addParameter(&custom_duco_username); - wifiManager.addParameter(&custom_duco_password); - wifiManager.addParameter(&custom_duco_rigid); - - #if defined(BLUSHYBOX) - blinker.attach_ms(200, changeState); - #endif - wifiManager.autoConnect("Duino-Coin"); - delay(1000); - VerifyWifi(); - #if defined(BLUSHYBOX) - blinker.detach(); - #endif - - #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - display_info("Waiting for node..."); - #endif - #if defined(BLUSHYBOX) - blinker.attach_ms(500, changeState); - #endif - SelectNode(); - #if defined(BLUSHYBOX) - blinker.detach(); - #endif - #else - #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - display_info("Waiting for WiFi..."); - #endif - SetupWifi(); - #endif - SetupOTA(); - - #if defined(WEB_DASHBOARD) - if (!MDNS.begin(RIG_IDENTIFIER)) { - #if defined(SERIAL_PRINTING) - Serial.println("mDNS unavailable"); - #endif - } - MDNS.addService("http", "tcp", 80); - #if defined(SERIAL_PRINTING) - #ifdef USE_LAN - Serial.println("Configured mDNS for dashboard on http://" + String(RIG_IDENTIFIER) - + ".local (or http://" + ETH.localIP().toString() + ")"); - #else - Serial.println("Configured mDNS for dashboard on http://" + String(RIG_IDENTIFIER) - + ".local (or http://" + WiFi.localIP().toString() + ")"); - #endif - #endif - - server.on("/", dashboard); - #if defined(CAPTIVE_PORTAL) - server.on("/reset", reset_settings); - #endif - server.begin(); - #endif - - #if defined(ESP8266) - // Start the WDT watchdog - lwdtFeed(); - lwdTimer.attach_ms(LWD_TIMEOUT, lwdtcb); - #endif - - #if defined(ESP8266) - // Fastest clock mode for 8266s - system_update_cpu_freq(160); - os_update_cpu_frequency(160); - // Feed the watchdog - lwdtFeed(); - #else - // Fastest clock mode for 32s - setCpuFrequencyMhz(240); - #endif - - job[0]->blink(BLINK_SETUP_COMPLETE); - - #if defined(ESP32) && CORE == 2 - mutexClientData = xSemaphoreCreateMutex(); - mutexConnectToServer = xSemaphoreCreateMutex(); - - xTaskCreatePinnedToCore(system_events_func, "system_events_func", 10000, NULL, 1, NULL, 0); - xTaskCreatePinnedToCore(task1_func, "task1_func", 10000, NULL, 1, &Task1, 0); - xTaskCreatePinnedToCore(task2_func, "task2_func", 10000, NULL, 1, &Task2, 1); - #endif -} - -void system_events_func(void* parameter) { - while (true) { - delay(10); - #if defined(WEB_DASHBOARD) - server.handleClient(); - #endif - ArduinoOTA.handle(); - } -} - -void single_core_loop() { - job[0]->mine(); - - lwdtFeed(); - - #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) - float hashrate_float = (hashrate+hashrate_core_two) / 1000.0; - float accept_rate = (accepted_share_count / 0.01 / share_count); - - long millisecs = millis(); - int uptime_secs = int((millisecs / 1000) % 60); - int uptime_mins = int((millisecs / (1000 * 60)) % 60); - int uptime_hours = int((millisecs / (1000 * 60 * 60)) % 24); - String uptime = String(uptime_hours) + "h" + String(uptime_mins) + "m" + String(uptime_secs) + "s"; - - float sharerate = share_count / (millisecs / 1000.0); - - display_mining_results(String(hashrate_float, 1), String(accepted_share_count), String(share_count), String(uptime), - String(node_id), String(difficulty / 100), String(sharerate, 1), - String(ping), String(accept_rate, 1)); - #endif - - VerifyWifi(); - ArduinoOTA.handle(); - #if defined(WEB_DASHBOARD) - server.handleClient(); - #endif -} - -void loop() { - #if defined(ESP8266) || defined(CONFIG_FREERTOS_UNICORE) - single_core_loop(); - #endif - delay(10); -} +/* + ____ __ __ ____ _ _ _____ ___ _____ ____ _ _ + ( _ \( )( )(_ _)( \( )( _ )___ / __)( _ )(_ _)( \( ) + )(_) ))(__)( _)(_ ) ( )(_)((___)( (__ )(_)( _)(_ ) ( + (____/(______)(____)(_)\_)(_____) \___)(_____)(____)(_)\_) + Official code for all ESP8266/32 boards version 4.3 + Main .ino file + + The Duino-Coin Team & Community 2019-2025 © MIT Licensed + https://duinocoin.com + https://github.com/revoxhere/duino-coin + + If you don't know where to start, visit official website and navigate to + the Getting Started page. Have fun mining! + + To edit the variables (username, WiFi settings, etc.) use the Settings.h tab! +*/ + +/* If optimizations cause problems, change them to -O0 (the default) */ +#pragma GCC optimize("-Ofast") + +/* If during compilation the line below causes a + "fatal error: arduinoJson.h: No such file or directory" + message to occur; it means that you do NOT have the + ArduinoJSON library installed. To install it, + go to the below link and follow the instructions: + https://github.com/revoxhere/duino-coin/issues/832 */ +#include + +#if defined(ESP8266) + #include + #include + #include + #include +#else + #include + #include + #include + #include + #include +#endif + +#include +#include +#include +#include +#include +#include +#include + + +//-----------------------------used for the balance display interval-------------------// +boolean runEvery(unsigned long interval) { + static unsigned long previousMillis = 0; + unsigned long currentMillis = millis(); + if (currentMillis - previousMillis >= interval) { + previousMillis = currentMillis; + return true; + } + return false; +} +//-------------------------------------------------------------------------------------// + +void handleSystemEvents(void) { + #if defined(ESP32) && CORE == 2 + esp_task_wdt_reset(); + #endif + delay(10); // Required vTaskDelay by ESP-IDF + yield(); + ArduinoOTA.handle(); + } + +#include "MiningJob.h" +#include "Settings.h" + +#ifdef USE_LAN + #include +#endif + +#if defined(WEB_DASHBOARD) + #include "Dashboard.h" +#endif + +#if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2)|| defined(DISPLAY_114)|| defined(DISPLAY_7735)|| defined(DISPLAY_2432S08) + #include "DisplayHal.h" +#endif + +#if !defined(ESP8266) && defined(DISABLE_BROWNOUT) + #include "soc/soc.h" + #include "soc/rtc_cntl_reg.h" +#endif + +// Auto adjust physical core count +// (ESP32-S2/C3 have 1 core, ESP32 has 2 cores, ESP8266 has 1 core) +#if defined(ESP8266) + #define CORE 1 + typedef ESP8266WebServer WebServer; +#elif defined(CONFIG_FREERTOS_UNICORE) + #define CORE 1 +#else + #define CORE 2 + // Install TridentTD_EasyFreeRTOS32 if you get an error + #include + + void Task1Code( void * parameter ); + void Task2Code( void * parameter ); + TaskHandle_t Task1; + TaskHandle_t Task2; +#endif + +#if defined(WEB_DASHBOARD) + WebServer server(80); +#endif + +#if defined(CAPTIVE_PORTAL) + #include // This needs to be first, or it all crashes and burns... + #include + #include + char duco_username[40]; + char duco_password[40]; + char duco_rigid[24]; + WiFiManager wifiManager; + Preferences preferences; + WiFiManagerParameter custom_duco_username("duco_usr", "Duino-Coin username", duco_username, 40); + WiFiManagerParameter custom_duco_password("duco_pwd", "Duino-Coin mining key (if enabled in the wallet)", duco_password, 40); + WiFiManagerParameter custom_duco_rigid("duco_rig", "Custom miner identifier (optional)", duco_rigid, 24); + + + void saveConfigCallback() { + preferences.begin("duino_config", false); + preferences.putString("duco_username", custom_duco_username.getValue()); + preferences.putString("duco_password", custom_duco_password.getValue()); + preferences.putString("duco_rigid", custom_duco_rigid.getValue()); + preferences.end(); + RestartESP("Settings saved"); + } + + + + void reset_settings() { + server.send(200, "text/html", "Settings have been erased. Please redo the configuration by connecting to the WiFi network that will be created"); + delay(500); + wifiManager.resetSettings(); + RestartESP("Manual settings reset"); + } + + void saveParamCallback(){ + Serial.println("[CALLBACK] saveParamCallback fired"); + Serial.println("PARAM customfieldid = " + getParam("customfieldid")); + } + + String getParam(String name){ + //read parameter from server, for customhmtl input + String value; + if(wifiManager.server->hasArg(name)) { + value = wifiManager.server->arg(name); + } + return value; + } +#endif + +void RestartESP(String msg) { + #if defined(SERIAL_PRINTING) + Serial.println(msg); + Serial.println("Restarting ESP..."); + #endif + + #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) || defined(DISPLAY_114)|| defined(DISPLAY_7735)|| defined(DISPLAY_2432S08) + display_info("Restarting ESP..."); + #if defined(DISPLAY_2432S08) + display_background(); + #endif + #endif + + #if defined(ESP8266) + ESP.reset(); + #else + ESP.restart(); + abort(); + #endif +} + +#if defined(BLUSHYBOX) + Ticker blinker; + bool lastLedState = false; + void changeState() { + analogWrite(LED_BUILTIN, lastLedState ? 255 : 0); + lastLedState = !lastLedState; + } +#endif + +#if defined(ESP8266) + // WDT Loop + // See lwdtcb() and lwdtFeed() below + Ticker lwdTimer; + + unsigned long lwdCurrentMillis = 0; + unsigned long lwdTimeOutMillis = LWD_TIMEOUT; + + void ICACHE_RAM_ATTR lwdtcb(void) { + if ((millis() - lwdCurrentMillis > LWD_TIMEOUT) || (lwdTimeOutMillis - lwdCurrentMillis != LWD_TIMEOUT)) + RestartESP("Loop WDT Failed!"); + } + + void lwdtFeed(void) { + lwdCurrentMillis = millis(); + lwdTimeOutMillis = lwdCurrentMillis + LWD_TIMEOUT; + } +#else + void lwdtFeed(void) { + Serial.println("lwdtFeed()"); + } +#endif + +namespace { + MiningConfig *configuration = new MiningConfig( + DUCO_USER, + RIG_IDENTIFIER, + MINER_KEY + ); + + #if defined(ESP32) && CORE == 2 + EasyMutex mutexClientData, mutexConnectToServer; + #endif + + #ifdef USE_LAN + static bool eth_connected = false; + #endif + + + + void UpdateHostPort(String input) { + // Thanks @ricaun for the code + DynamicJsonDocument doc(256); + deserializeJson(doc, input); + const char *name = doc["name"]; + + configuration->host = doc["ip"].as().c_str(); + configuration->port = doc["port"].as(); + node_id = String(name); + + #if defined(SERIAL_PRINTING) + Serial.println("Poolpicker selected the best mining node: " + node_id); + #endif + + #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) || defined(DISPLAY_114)|| defined(DISPLAY_7735)||defined(DISPLAY_2432S08) + display_info(node_id); + #endif + } + + void VerifyWifi() { + #ifdef USE_LAN + while ((!eth_connected) || (ETH.localIP() == IPAddress(0, 0, 0, 0))) { + #if defined(SERIAL_PRINTING) + Serial.println("Ethernet connection lost. Reconnect..." ); + #endif + SetupWifi(); + } + #else + while (WiFi.status() != WL_CONNECTED + || WiFi.localIP() == IPAddress(0, 0, 0, 0) + || WiFi.localIP() == IPAddress(192, 168, 4, 2) + || WiFi.localIP() == IPAddress(192, 168, 4, 3)) { + #if defined(SERIAL_PRINTING) + Serial.println("WiFi reconnecting..."); + #endif + WiFi.disconnect(); + delay(500); + WiFi.reconnect(); + delay(500); + } + #endif + } + + String httpGetString(String URL) { + String payload = ""; + + WiFiClientSecure client; + HTTPClient https; + client.setInsecure(); + + https.begin(client, URL); + https.addHeader("Accept", "*/*"); + + int httpCode = https.GET(); + #if defined(SERIAL_PRINTING) + Serial.printf("HTTP Response code: %d\n", httpCode); + #endif + + if (httpCode == HTTP_CODE_OK || httpCode == HTTP_CODE_MOVED_PERMANENTLY) { + payload = https.getString(); + } else { + #if defined(SERIAL_PRINTING) + Serial.printf("Error fetching node from poolpicker: %s\n", https.errorToString(httpCode).c_str()); + VerifyWifi(); + #endif + #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2)|| defined(DISPLAY_114)|| defined(DISPLAY_7735)|| defined(DISPLAY_2432S08) + display_info(https.errorToString(httpCode)); + #if defined(DISPLAY_2432S08) + display_background(); + #endif + #endif + } + https.end(); + return payload; + } + + void SelectNode() { + String input = ""; + int waitTime = 1; + int poolIndex = 0; + + while (input == "") { + #if defined(SERIAL_PRINTING) + Serial.println("Fetching mining node from the poolpicker in " + String(waitTime) + "s"); + #endif + delay(waitTime * 1000); + + input = httpGetString("https://server.duinocoin.com/getPool"); + + // Increase wait time till a maximum of 32 seconds + // (addresses: Limit connection requests on failure in ESP boards #1041) + waitTime *= 2; + if (waitTime > 32) + RestartESP("Node fetch unavailable"); + } + + UpdateHostPort(input); + } + + + + #ifdef USE_LAN + void WiFiEvent(WiFiEvent_t event) { + switch (event) { + case ARDUINO_EVENT_ETH_START: + #if defined(SERIAL_PRINTING) + Serial.println("ETH Started"); + #endif + // The hostname must be set after the interface is started, but needs + // to be set before DHCP, so set it from the event handler thread. + ETH.setHostname("esp32-ethernet"); + break; + case ARDUINO_EVENT_ETH_CONNECTED: + #if defined(SERIAL_PRINTING) + Serial.println("ETH Connected"); + #endif + break; + case ARDUINO_EVENT_ETH_GOT_IP: + #if defined(SERIAL_PRINTING) + Serial.println("ETH Got IP"); + #endif + eth_connected = true; + break; + case ARDUINO_EVENT_ETH_DISCONNECTED: + #if defined(SERIAL_PRINTING) + Serial.println("ETH Disconnected"); + #endif + eth_connected = false; + break; + case ARDUINO_EVENT_ETH_STOP: + #if defined(SERIAL_PRINTING) + Serial.println("ETH Stopped"); + #endif + eth_connected = false; + break; + default: + break; + } + } + #endif + + void SetupWifi() { + #ifdef USE_LAN + #if defined(SERIAL_PRINTING) + Serial.println("Connecting to Ethernet..."); + #endif + WiFi.onEvent(WiFiEvent); // Will call WiFiEvent() from another thread. + ETH.begin(); + + while (!eth_connected) { + delay(500); + #if defined(SERIAL_PRINTING) + Serial.print("."); + #endif + } + + #if defined(SERIAL_PRINTING) + Serial.println("\n\nSuccessfully connected to Ethernet"); + Serial.println("Local IP address: " + ETH.localIP().toString()); + Serial.println("Rig name: " + String(RIG_IDENTIFIER)); + Serial.println(); + #endif + + #else + #if defined(SERIAL_PRINTING) + Serial.println("Connecting to: " + String(SSID)); + #endif + + WiFi.begin(SSID, PASSWORD); + while(WiFi.status() != WL_CONNECTED) { + Serial.print("."); + delay(100); + } + VerifyWifi(); + + #if !defined(ESP8266) + WiFi.config(WiFi.localIP(), WiFi.gatewayIP(), WiFi.subnetMask(), DNS_SERVER); + #endif + + #if defined(SERIAL_PRINTING) + Serial.println("\n\nSuccessfully connected to WiFi"); + Serial.println("Rig name: " + String(RIG_IDENTIFIER)); + Serial.println("Local IP address: " + WiFi.localIP().toString()); + Serial.println("Gateway: " + WiFi.gatewayIP().toString()); + Serial.println("DNS: " + WiFi.dnsIP().toString()); + Serial.println(); + #endif + + #endif + + #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) || defined(DISPLAY_114)|| defined(DISPLAY_7735)|| defined(DISPLAY_2432S08) + display_info("Waiting for node..."); + #endif + SelectNode(); + + } + + + void SetupOTA() { + // Prepare OTA handler + ArduinoOTA.onStart([]() + { + #if defined(SERIAL_PRINTING) + Serial.println("Start"); + #endif + }); + ArduinoOTA.onEnd([]() + { + #if defined(SERIAL_PRINTING) + Serial.println("\nEnd"); + #endif + }); + ArduinoOTA.onProgress([](unsigned int progress, unsigned int total) + { + #if defined(SERIAL_PRINTING) + Serial.printf("Progress: %u%%\r", (progress / (total / 100))); + #endif + }); + ArduinoOTA.onError([](ota_error_t error) + { + Serial.printf("Error[%u]: ", error); + #if defined(SERIAL_PRINTING) + if (error == OTA_AUTH_ERROR) Serial.println("Auth Failed"); + else if (error == OTA_BEGIN_ERROR) Serial.println("Begin Failed"); + else if (error == OTA_CONNECT_ERROR) Serial.println("Connect Failed"); + else if (error == OTA_RECEIVE_ERROR) Serial.println("Receive Failed"); + else if (error == OTA_END_ERROR) Serial.println("End Failed"); + #endif + }); + + ArduinoOTA.setHostname(RIG_IDENTIFIER); // Give port a name + ArduinoOTA.begin(); + } + + #if defined(WEB_DASHBOARD) + void dashboard() { + #if defined(SERIAL_PRINTING) + Serial.println("Handling HTTP client"); + #endif + String s = WEBSITE; + #ifdef USE_LAN + s.replace("@@IP_ADDR@@", ETH.localIP().toString()); + #else + s.replace("@@IP_ADDR@@", WiFi.localIP().toString()); + #endif + + s.replace("@@HASHRATE@@", String((hashrate+hashrate_core_two) / 1000)); + s.replace("@@DIFF@@", String(difficulty / 100)); + s.replace("@@SHARES@@", String(share_count)); + s.replace("@@NODE@@", String(node_id)); + + #if defined(ESP8266) + s.replace("@@DEVICE@@", "ESP8266"); + #elif defined(CONFIG_FREERTOS_UNICORE) + s.replace("@@DEVICE@@", "ESP32-S2/C3"); + #else + s.replace("@@DEVICE@@", "ESP32"); + #endif + + s.replace("@@ID@@", String(RIG_IDENTIFIER)); + s.replace("@@MEMORY@@", String(ESP.getFreeHeap())); + s.replace("@@VERSION@@", String(SOFTWARE_VERSION)); + + #if defined(CAPTIVE_PORTAL) + s.replace("@@RESET_SETTINGS@@", "• Reset settings"); + #else + s.replace("@@RESET_SETTINGS@@", ""); + #endif + + #if defined(USE_DS18B20) + sensors.requestTemperatures(); + float temp = sensors.getTempCByIndex(0); + s.replace("@@SENSOR@@", "DS18B20: " + String(temp) + "*C"); + #elif defined(USE_DHT) + float temp = dht.readTemperature(); + float hum = dht.readHumidity(); + s.replace("@@SENSOR@@", "DHT11/22: " + String(temp) + "*C, " + String(hum) + "rh%"); + #elif defined(USE_HSU07M) + float temp = read_hsu07m(); + s.replace("@@SENSOR@@", "HSU07M: " + String(temp) + "*C"); + #elif defined(USE_INTERNAL_SENSOR) + float temp = 0; + temp_sensor_read_celsius(&temp); + s.replace("@@SENSOR@@", "CPU: " + String(temp) + "*C"); + #else + s.replace("@@SENSOR@@", "None"); + #endif + + server.send(200, "text/html", s); + } + #endif + +} // End of namespace + + +MiningJob *job[CORE]; + +#if CORE == 2 + EasyFreeRTOS32 task1, task2; +#endif + +void task1_func(void *) { + #if defined(ESP32) && CORE == 2 + VOID SETUP() { } + + VOID LOOP() { + job[0]->mine(); + + #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) || defined(DISPLAY_114)|| defined(DISPLAY_7735)|| defined(DISPLAY_2432S08) + float hashrate_float = (hashrate+hashrate_core_two) / 1000.0; + float accept_rate = (accepted_share_count / 0.01 / share_count); + + long millisecs = millis(); + int uptime_secs = int((millisecs / 1000) % 60); + int uptime_mins = int((millisecs / (1000 * 60)) % 60); + int uptime_hours = int((millisecs / (1000 * 60 * 60)) % 24); + String uptime = String(uptime_hours) + "h" + String(uptime_mins) + "m" + String(uptime_secs) + "s"; + + float sharerate = share_count / (millisecs / 1000.0); + + printTime(); + + while(displayLock) { + handleSystemEvents(); + } + displayLock = true; + + display_mining_results(String(hashrate_float, 1), String(accepted_share_count), String(share_count), String(uptime), + String(node_id), String(difficulty / 100), String(sharerate, 2), + String(ping), String(accept_rate, 1)); + + display_balance(String(result_balance_balance),String(total_miner),String(result_balance_username)); + + display_time(String(mytime)); + + displayLock = false; + #endif + } + #endif +} + +void task2_func(void *) { + #if defined(ESP32) && CORE == 2 + VOID SETUP() { + job[1] = new MiningJob(1, configuration); + } + + VOID LOOP() { + job[1]->mine(); + + #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) || defined(DISPLAY_114)|| defined(DISPLAY_7735)|| defined(DISPLAY_2432S08) + + float hashrate_float = (hashrate+hashrate_core_two) / 1000.0; + float accept_rate = (accepted_share_count / 0.01 / share_count); + + long millisecs = millis(); + int uptime_secs = int((millisecs / 1000) % 60); + int uptime_mins = int((millisecs / (1000 * 60)) % 60); + int uptime_hours = int((millisecs / (1000 * 60 * 60)) % 24); + String uptime = String(uptime_hours) + "h" + String(uptime_mins) + "m" + String(uptime_secs) + "s"; + + float sharerate = share_count / (millisecs / 1000.0); + + printTime(); + + while(displayLock) { + handleSystemEvents(); + } + displayLock = true; + + display_mining_results(String(hashrate_float, 1), String(accepted_share_count), String(share_count), String(uptime), + String(node_id), String(difficulty / 100), String(sharerate, 2), + String(ping), String(accept_rate, 1)); + + display_balance(String(result_balance_balance),String(total_miner),String(result_balance_username)); + + display_time(String(mytime)); + + displayLock = false; + #endif + } + #endif +} + +//------------------------- Settings and code for time display on ESP32-2432S028 CYD------------------------- + +void notify(struct timeval* t) { + #if defined(SERIAL_PRINTING) + Serial.println("SNTP synchronized"); + #endif +} + +void initSNTP() { + sntp_set_sync_interval(4 * 60 * 60 * 1000UL); // 1 hour + sntp_set_time_sync_notification_cb(notify); + esp_sntp_setoperatingmode(ESP_SNTP_OPMODE_POLL); + esp_sntp_setservername(0, "pool.ntp.org"); + esp_sntp_init(); + setTimezone(); +} + +void setTimezone() { + setenv("TZ", "CET-1CEST,M3.5.0,M10.5.0/3", 1); // Paris TZ, get your TZ at https://github.com/nayarsystems/posix_tz_db/blob/master/zones.csv + tzset(); +} + +void printTime() { + struct tm timeinfo; + getLocalTime(&timeinfo); + #if defined(SERIAL_PRINTING) + Serial.println(&timeinfo, "%A, %B %d %Y %H:%M:%S"); + #endif + strftime (mytime, sizeof (mytime), "%H:%M", &timeinfo); + strftime (mydate, sizeof (mydate), "%d/%m/%Y", &timeinfo); //date in European format + strftime (myday, sizeof (myday), "%A", &timeinfo); +} + +//--------------------------------------------------------------------------------------------------------- + + + +void setup() { + #if !defined(ESP8266) && defined(DISABLE_BROWNOUT) + WRITE_PERI_REG(RTC_CNTL_BROWN_OUT_REG, 0); + #endif + + #if defined(SERIAL_PRINTING) + Serial.begin(SERIAL_BAUDRATE); + Serial.println("\n\nDuino-Coin " + String(configuration->MINER_VER)); + #endif + pinMode(LED_BUILTIN, OUTPUT); + + #if defined(BLUSHYBOX) + analogWrite(LED_BUILTIN, 255); + for (int i = 255; i > 0; i--) { + analogWrite(LED_BUILTIN, i); + delay(1); + } + pinMode(GAUGE_PIN, OUTPUT); + + // Gauge up and down effect on startup + for (int i = GAUGE_MIN; i < GAUGE_MAX; i++) { + analogWrite(GAUGE_PIN, i); + delay(10); + } + for (int i = GAUGE_MAX; i > GAUGE_MIN; i--) { + analogWrite(GAUGE_PIN, i); + delay(10); + } + #endif + + #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2)|| defined(DISPLAY_114)|| defined(DISPLAY_7735)|| defined(DISPLAY_2432S08) + screen_setup(); + display_boot(); + + #endif + + assert(CORE == 1 || CORE == 2); + WALLET_ID = String(random(0, 2811)); // Needed for miner grouping in the wallet + job[0] = new MiningJob(0, configuration); + + #if defined(USE_DHT) + #if defined(SERIAL_PRINTING) + Serial.println("Initializing DHT sensor (Duino IoT)"); + #endif + dht.begin(); + #if defined(SERIAL_PRINTING) + Serial.println("Test reading: " + String(dht.readHumidity()) + "% humidity"); + Serial.println("Test reading: temperature " + String(dht.readTemperature()) + "°C"); + #endif + #endif + + #if defined(USE_DS18B20) + #if defined(SERIAL_PRINTING) + Serial.println("Initializing DS18B20 sensor (Duino IoT)"); + #endif + sensors.begin(); + sensors.requestTemperatures(); + #if defined(SERIAL_PRINTING) + Serial.println("Test reading: " + String(sensors.getTempCByIndex(0)) + "°C"); + #endif + #endif + + #if defined(USE_HSU07M) + #if defined(SERIAL_PRINTING) + Serial.println("Initializing HSU07M sensor (Duino IoT)"); + Serial.println("Test reading: " + String(read_hsu07m()) + "°C"); + #endif + #endif + + #if defined(USE_INTERNAL_SENSOR) + #if defined(SERIAL_PRINTING) + Serial.println("Initializing internal ESP32 temperature sensor (Duino IoT)"); + #endif + temp_sensor_config_t temp_sensor = TSENS_CONFIG_DEFAULT(); + temp_sensor.dac_offset = TSENS_DAC_L2; + temp_sensor_set_config(temp_sensor); + temp_sensor_start(); + float result = 0; + temp_sensor_read_celsius(&result); + #if defined(SERIAL_PRINTING) + Serial.println("Test reading: " + String(result) + "°C"); + #endif + #endif + + WiFi.mode(WIFI_STA); // Setup ESP in client mode + //WiFi.disconnect(true); + #if defined(ESP8266) + WiFi.setSleepMode(WIFI_NONE_SLEEP); + #else + WiFi.setSleep(false); + #endif + + #if defined(CAPTIVE_PORTAL) + preferences.begin("duino_config", false); + strcpy(duco_username, preferences.getString("duco_username", "username").c_str()); + strcpy(duco_password, preferences.getString("duco_password", "None").c_str()); + strcpy(duco_rigid, preferences.getString("duco_rigid", "None").c_str()); + preferences.end(); + configuration->DUCO_USER = duco_username; + configuration->RIG_IDENTIFIER = duco_rigid; + configuration->MINER_KEY = duco_password; + + String captivePortalHTML = R"( + Duino BlushyBox + + )"; + + wifiManager.setCustomHeadElement(captivePortalHTML.c_str()); + + wifiManager.setSaveConfigCallback(saveConfigCallback); + wifiManager.addParameter(&custom_duco_username); + wifiManager.addParameter(&custom_duco_password); + wifiManager.addParameter(&custom_duco_rigid); + + #if defined(BLUSHYBOX) + blinker.attach_ms(200, changeState); + #endif + wifiManager.autoConnect("Duino-Coin"); + delay(1000); + VerifyWifi(); + #if defined(BLUSHYBOX) + blinker.detach(); + #endif + + #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2)|| defined(DISPLAY_114)|| defined(DISPLAY_7735)|| defined(DISPLAY_2432S08) + display_info("Waiting for node..."); + #endif + #if defined(BLUSHYBOX) + blinker.attach_ms(500, changeState); + #endif + SelectNode(); + #if defined(BLUSHYBOX) + blinker.detach(); + #endif + #else + #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2)|| defined(DISPLAY_114)|| defined(DISPLAY_7735)||defined(DISPLAY_2432S08) + display_info("Waiting for WiFi..."); + #endif + SetupWifi(); + #endif + SetupOTA(); + + #if defined(DISPLAY_2432S08) + preferences.begin("duino_config", false); + strcpy(duco_username, preferences.getString("duco_username", "username").c_str()); + preferences.end(); + #if defined(SERIAL_PRINTING) + Serial.println("---- Welcome ---- 2438S028 user ! ----"); + Serial.println(duco_username); + Serial.println("--------------------------------------"); + #endif + #endif + + #if defined(WEB_DASHBOARD) + if (!MDNS.begin(RIG_IDENTIFIER)) { + #if defined(SERIAL_PRINTING) + Serial.println("mDNS unavailable"); + #endif + } + MDNS.addService("http", "tcp", 80); + #if defined(SERIAL_PRINTING) + #ifdef USE_LAN + Serial.println("Configured mDNS for dashboard on http://" + String(RIG_IDENTIFIER) + + ".local (or http://" + ETH.localIP().toString() + ")"); + #else + Serial.println("Configured mDNS for dashboard on http://" + String(RIG_IDENTIFIER) + + ".local (or http://" + WiFi.localIP().toString() + ")"); + #endif + #endif + + server.on("/", dashboard); + #if defined(CAPTIVE_PORTAL) + server.on("/reset", reset_settings); + #endif + server.begin(); + #endif + + #if defined(ESP8266) + // Start the WDT watchdog + lwdtFeed(); + lwdTimer.attach_ms(LWD_TIMEOUT, lwdtcb); + #endif + + #if defined(ESP8266) + // Fastest clock mode for 8266s + system_update_cpu_freq(160); + os_update_cpu_frequency(160); + // Feed the watchdog + lwdtFeed(); + #else + // Fastest clock mode for 32s + setCpuFrequencyMhz(240); + #endif + + job[0]->blink(BLINK_SETUP_COMPLETE); + + #if defined(ESP32) && CORE == 2 + mutexClientData = xSemaphoreCreateMutex(); + mutexConnectToServer = xSemaphoreCreateMutex(); + + xTaskCreatePinnedToCore(system_events_func, "system_events_func", 10000, NULL, 1, NULL, 0); + xTaskCreatePinnedToCore(task1_func, "task1_func", 10000, NULL, 1, &Task1, 0); + xTaskCreatePinnedToCore(task2_func, "task2_func", 10000, NULL, 1, &Task2, 1); + #endif + + #if defined(DISPLAY_2432S08) + display_background(); + #endif + +//----------------------------------SNTP connect------------------------------------------------// + + initSNTP(); + +//----------------------------------------------------------------------------------------------// + +} + +void system_events_func(void* parameter) { + while (true) { + delay(10); + #if defined(WEB_DASHBOARD) + server.handleClient(); + #endif + ArduinoOTA.handle(); + } +} + +void single_core_loop() { + job[0]->mine(); + + lwdtFeed(); + + #if defined(DISPLAY_SSD1306) || defined(DISPLAY_16X2) || defined(DISPLAY_114)|| defined(DISPLAY_7735) // || defined(DISPLAY_2432S08) + float hashrate_float = (hashrate+hashrate_core_two) / 1000.0; + float accept_rate = (accepted_share_count / 0.01 / share_count); + + long millisecs = millis(); + int uptime_secs = int((millisecs / 1000) % 60); + int uptime_mins = int((millisecs / (1000 * 60)) % 60); + int uptime_hours = int((millisecs / (1000 * 60 * 60)) % 24); + String uptime = String(uptime_hours) + "h" + String(uptime_mins) + "m" + String(uptime_secs) + "s"; + + float sharerate = share_count / (millisecs / 1000.0); + + display_mining_results(String(hashrate_float, 1), String(accepted_share_count), String(share_count), String(uptime), + String(node_id), String(difficulty / 100), String(sharerate, 2), + String(ping), String(accept_rate, 1)); + #endif + + VerifyWifi(); + ArduinoOTA.handle(); + #if defined(WEB_DASHBOARD) + server.handleClient(); + #endif +} + +void loop() { + #if defined(ESP8266) || defined(CONFIG_FREERTOS_UNICORE) + single_core_loop(); + #endif + delay(10); + +// displaying user balance and number of miners, update every 300 s (can be changed in Settings.h) + +#if defined(DISPLAY_2432S08) + + if ((runEvery(run_in_ms)) || (first_start == true)) + { + + String ducoReportJsonUrl = ("https://server.duinocoin.com/v2/users/" + String(duco_username)); + + #if defined(SERIAL_PRINTING) + Serial.println("retrieving balance from the API..."); + Serial.println(String(ducoReportJsonUrl)); + Serial.println(); + #endif + + first_start = false; + String input2 = httpGetString(ducoReportJsonUrl); + DynamicJsonDocument doc (30000); + DeserializationError error = deserializeJson(doc, input2); + + if (error) { + Serial.print("deserializeJson() failed: "); + Serial.println(error.c_str()); + return; + } + + JsonObject result = doc["result"]; + + JsonObject result_balance = result["balance"]; + + total_miner = 0; + + for (JsonObject result_miner : result["miners"].as()) { + result_balance_balance = result_balance["balance"]; + result_balance_username = result_balance["username"].as(); // Assign as String + total_miner++; + + +// for (JsonObject result_miner : result["miners"].as()) { +// result_balance_balance = result_balance["balance"]; +// result_balance_username = result_balance["username"]; +// total_miner++; + } + #if defined(SERIAL_PRINTING) + Serial.println(); + Serial.println("Username : " + String(result_balance_username)); + Serial.println("Balance : " + String(result_balance_balance) + " DUCO"); + Serial.println("Total miners : " + String(total_miner)); + #endif + } + +#endif + +} diff --git a/ESP_Code/ESP_Code.ino.jczn_2432s028r.bin b/ESP_Code/ESP_Code.ino.jczn_2432s028r.bin new file mode 100644 index 00000000..fbf6404e Binary files /dev/null and b/ESP_Code/ESP_Code.ino.jczn_2432s028r.bin differ diff --git a/ESP_Code/MiningJob.h b/ESP_Code/MiningJob.h index bb200914..7394f383 100644 --- a/ESP_Code/MiningJob.h +++ b/ESP_Code/MiningJob.h @@ -1,423 +1,425 @@ -#pragma GCC optimize("-Ofast") - -#ifndef MINING_JOB_H -#define MINING_JOB_H - -#include -#include -#include -#include -#include - -#include "DSHA1.h" -#include "Counter.h" -#include "Settings.h" - -// https://github.com/esp8266/Arduino/blob/master/cores/esp8266/TypeConversion.cpp -const char base36Chars[36] PROGMEM = { - '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', - 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z' -}; - -const uint8_t base36CharValues[75] PROGMEM{ - 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0, 0, 0, 0, 0, 0, 0, // 0 to 9 - 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 0, 0, 0, 0, 0, 0, // Upper case letters - 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35 // Lower case letters -}; - -#define SPC_TOKEN ' ' -#define END_TOKEN '\n' -#define SEP_TOKEN ',' -#define IOT_TOKEN '@' - -struct MiningConfig { - String host = ""; - int port = 0; - String DUCO_USER = ""; - String RIG_IDENTIFIER = ""; - String MINER_KEY = ""; - String MINER_VER = SOFTWARE_VERSION; - #if defined(ESP8266) - // "High-band" 8266 diff - String START_DIFF = "ESP8266H"; - #elif defined(CONFIG_FREERTOS_UNICORE) - // Single core 32 diff - String START_DIFF = "ESP32S"; - #else - // Normal 32 diff - String START_DIFF = "ESP32"; - #endif - - MiningConfig(String DUCO_USER, String RIG_IDENTIFIER, String MINER_KEY) - : DUCO_USER(DUCO_USER), RIG_IDENTIFIER(RIG_IDENTIFIER), MINER_KEY(MINER_KEY) {} -}; - -class MiningJob { - -public: - MiningConfig *config; - int core = 0; - - MiningJob(int core, MiningConfig *config) { - this->core = core; - this->config = config; - this->client_buffer = ""; - dsha1 = new DSHA1(); - dsha1->warmup(); - generateRigIdentifier(); - } - - void blink(uint8_t count, uint8_t pin = LED_BUILTIN) { - #if defined(LED_BLINKING) - uint8_t state = HIGH; - - for (int x = 0; x < (count << 1); ++x) { - digitalWrite(pin, state ^= HIGH); - delay(50); - } - #else - digitalWrite(LED_BUILTIN, HIGH); - #endif - } - - bool max_micros_elapsed(unsigned long current, unsigned long max_elapsed) { - static unsigned long _start = 0; - - if ((current - _start) > max_elapsed) { - _start = current; - return true; - } - return false; - } - - void handleSystemEvents(void) { - #if defined(ESP32) && CORE == 2 - esp_task_wdt_reset(); - #endif - delay(10); // Required vTaskDelay by ESP-IDF - yield(); - ArduinoOTA.handle(); - } - - void mine() { - connectToNode(); - askForJob(); - - dsha1->reset().write((const unsigned char *)getLastBlockHash().c_str(), getLastBlockHash().length()); - - int start_time = micros(); - max_micros_elapsed(start_time, 0); - #if defined(LED_BLINKING) - #if defined(BLUSHYBOX) - for (int i = 0; i < 72; i++) { - analogWrite(LED_BUILTIN, i); - delay(1); - } - #else - digitalWrite(LED_BUILTIN, LOW); - #endif - #endif - for (Counter<10> counter; counter < difficulty; ++counter) { - DSHA1 ctx = *dsha1; - ctx.write((const unsigned char *)counter.c_str(), counter.strlen()).finalize(hashArray); - - #ifndef CONFIG_FREERTOS_UNICORE - #if defined(ESP32) - #define SYSTEM_TIMEOUT 100000 // 10ms for esp32 looks like the lowest value without false watchdog triggers - #else - #define SYSTEM_TIMEOUT 500000 // 50ms for 8266 for same reason as above - #endif - if (max_micros_elapsed(micros(), SYSTEM_TIMEOUT)) { - handleSystemEvents(); - } - #endif - - if (memcmp(getExpectedHash(), hashArray, 20) == 0) { - unsigned long elapsed_time = micros() - start_time; - float elapsed_time_s = elapsed_time * .000001f; - share_count++; - - #if defined(LED_BLINKING) - #if defined(BLUSHYBOX) - for (int i = 72; i > 0; i--) { - analogWrite(LED_BUILTIN, i); - delay(1); - } - #else - digitalWrite(LED_BUILTIN, HIGH); - #endif - #endif - - if (String(core) == "0") { - hashrate = counter / elapsed_time_s; - submit(counter, hashrate, elapsed_time_s); - } else { - hashrate_core_two = counter / elapsed_time_s; - submit(counter, hashrate_core_two, elapsed_time_s); - } - - #if defined(BLUSHYBOX) - gauge_set(hashrate + hashrate_core_two); - #endif - - break; - } - } - } - -private: - String client_buffer; - uint8_t hashArray[20]; - String last_block_hash; - String expected_hash_str; - uint8_t expected_hash[20]; - DSHA1 *dsha1; - WiFiClient client; - String chipID = ""; - - #if defined(ESP8266) - #if defined(BLUSHYBOX) - String MINER_BANNER = "Official BlushyBox Miner (ESP8266)"; - #else - String MINER_BANNER = "Official ESP8266 Miner"; - #endif - #elif defined(CONFIG_FREERTOS_UNICORE) - String MINER_BANNER = "Official ESP32-S2 Miner"; - #else - #if defined(BLUSHYBOX) - String MINER_BANNER = "Official BlushyBox Miner (ESP32)"; - #else - String MINER_BANNER = "Official ESP32 Miner"; - #endif - #endif - - uint8_t *hexStringToUint8Array(const String &hexString, uint8_t *uint8Array, const uint32_t arrayLength) { - assert(hexString.length() >= arrayLength * 2); - const char *hexChars = hexString.c_str(); - for (uint32_t i = 0; i < arrayLength; ++i) { - uint8Array[i] = (pgm_read_byte(base36CharValues + hexChars[i * 2] - '0') << 4) + pgm_read_byte(base36CharValues + hexChars[i * 2 + 1] - '0'); - } - return uint8Array; - } - - void generateRigIdentifier() { - String AutoRigName = ""; - - #if defined(ESP8266) - chipID = String(ESP.getChipId(), HEX); - - if (strcmp(config->RIG_IDENTIFIER.c_str(), "Auto") != 0) - return; - - AutoRigName = "ESP8266-" + chipID; - AutoRigName.toUpperCase(); - config->RIG_IDENTIFIER = AutoRigName.c_str(); - #else - uint64_t chip_id = ESP.getEfuseMac(); - uint16_t chip = (uint16_t)(chip_id >> 32); // Prepare to print a 64 bit value into a char array - char fullChip[23]; - snprintf(fullChip, 23, "%04X%08X", chip, - (uint32_t)chip_id); // Store the (actually) 48 bit chip_id into a char array - - chipID = String(fullChip); - - if (strcmp(config->RIG_IDENTIFIER.c_str(), "Auto") != 0) - return; - // Autogenerate ID if required - AutoRigName = "ESP32-" + String(fullChip); - AutoRigName.toUpperCase(); - config->RIG_IDENTIFIER = AutoRigName.c_str(); - #endif - #if defined(SERIAL_PRINTING) - Serial.println("Core [" + String(core) + "] - Rig identifier: " - + config->RIG_IDENTIFIER); - #endif - } - - void connectToNode() { - if (client.connected()) return; - - unsigned int stopWatch = millis(); - #if defined(SERIAL_PRINTING) - Serial.println("Core [" + String(core) + "] - Connecting to a Duino-Coin node..."); - #endif - while (!client.connect(config->host.c_str(), config->port)) { - if (max_micros_elapsed(micros(), 100000)) { - handleSystemEvents(); - } - if (millis()-stopWatch>100000) ESP.restart(); - } - - waitForClientData(); - #if defined(SERIAL_PRINTING) - Serial.println("Core [" + String(core) + "] - Connected. Node reported version: " - + client_buffer); - #endif - - blink(BLINK_CLIENT_CONNECT); - - /* client.print("MOTD" + END_TOKEN); - waitForClientData(); - #if defined(SERIAL_PRINTING) - Serial.println("Core [" + String(core) + "] - MOTD: " - + client_buffer); - #endif */ - } - - void waitForClientData() { - client_buffer = ""; - - while (client.connected()) { - if (client.available()) { - client_buffer = client.readStringUntil(END_TOKEN); - if (client_buffer.length() == 1 && client_buffer[0] == END_TOKEN) - client_buffer = "???\n"; // NOTE: Should never happen - break; - } - } - } - - void submit(unsigned long counter, float hashrate, float elapsed_time_s) { - client.print(String(counter) + - SEP_TOKEN + String(hashrate) + - SEP_TOKEN + MINER_BANNER + - SPC_TOKEN + config->MINER_VER + - SEP_TOKEN + config->RIG_IDENTIFIER + - SEP_TOKEN + "DUCOID" + String(chipID) + - SEP_TOKEN + String(WALLET_ID) + - END_TOKEN); - - unsigned long ping_start = millis(); - waitForClientData(); - ping = millis() - ping_start; - - if (client_buffer == "GOOD") { - accepted_share_count++; - } - - #if defined(SERIAL_PRINTING) - Serial.println("Core [" + String(core) + "] - " + - client_buffer + - " share #" + String(share_count) + - " (" + String(counter) + ")" + - " hashrate: " + String(hashrate / 1000, 2) + " kH/s (" + - String(elapsed_time_s) + "s)\n"); - #endif - } - - bool parse() { - // Create a non-constant copy of the input string - char *job_str_copy = strdup(client_buffer.c_str()); - - if (job_str_copy) { - String tokens[3]; - char *token = strtok(job_str_copy, ","); - for (int i = 0; token != NULL && i < 3; i++) { - tokens[i] = token; - token = strtok(NULL, ","); - } - - last_block_hash = tokens[0]; - expected_hash_str = tokens[1]; - hexStringToUint8Array(expected_hash_str, expected_hash, 20); - difficulty = tokens[2].toInt() * 100 + 1; - - // Free the memory allocated by strdup - free(job_str_copy); - - return true; - } - else { - // Handle memory allocation failure - return false; - } - } - - void askForJob() { - Serial.println("Core [" + String(core) + "] - Asking for a new job for user: " - + String(config->DUCO_USER)); - - #if defined(USE_DS18B20) - sensors.requestTemperatures(); - float temp = sensors.getTempCByIndex(0); - #if defined(SERIAL_PRINTING) - Serial.println("DS18B20 reading: " + String(temp) + "°C"); - #endif - - client.print("JOB," + - String(config->DUCO_USER) + - SEP_TOKEN + config->START_DIFF + - SEP_TOKEN + String(config->MINER_KEY) + - SEP_TOKEN + "Temp:" + String(temp) + "*C" + - END_TOKEN); - #elif defined(USE_DHT) - float temp = dht.readTemperature(); - float hum = dht.readHumidity(); - #if defined(SERIAL_PRINTING) - Serial.println("DHT reading: " + String(temp) + "°C"); - Serial.println("DHT reading: " + String(hum) + "%"); - #endif - - client.print("JOB," + - String(config->DUCO_USER) + - SEP_TOKEN + config->START_DIFF + - SEP_TOKEN + String(config->MINER_KEY) + - SEP_TOKEN + "Temp:" + String(temp) + "*C" + - IOT_TOKEN + "Hum:" + String(hum) + "%" + - END_TOKEN); - #elif defined(USE_HSU07M) - float temp = read_hsu07m(); - #if defined(SERIAL_PRINTING) - Serial.println("HSU reading: " + String(temp) + "°C"); - #endif - - client.print("JOB," + - String(config->DUCO_USER) + - SEP_TOKEN + config->START_DIFF + - SEP_TOKEN + String(config->MINER_KEY) + - SEP_TOKEN + "Temp:" + String(temp) + "*C" + - END_TOKEN); - #elif defined(USE_INTERNAL_SENSOR) - float temp = 0; - temp_sensor_read_celsius(&temp); - #if defined(SERIAL_PRINTING) - Serial.println("Internal temp sensor reading: " + String(temp) + "°C"); - #endif - - client.print("JOB," + - String(config->DUCO_USER) + - SEP_TOKEN + config->START_DIFF + - SEP_TOKEN + String(config->MINER_KEY) + - SEP_TOKEN + "CPU Temp:" + String(temp) + "*C" + - END_TOKEN); - #else - client.print("JOB," + - String(config->DUCO_USER) + - SEP_TOKEN + config->START_DIFF + - SEP_TOKEN + String(config->MINER_KEY) + - END_TOKEN); - #endif - - waitForClientData(); - #if defined(SERIAL_PRINTING) - Serial.println("Core [" + String(core) + "] - Received job with size of " - + String(client_buffer.length()) - + " bytes " + client_buffer); - #endif - - parse(); - #if defined(SERIAL_PRINTING) - Serial.println("Core [" + String(core) + "] - Parsed job: " - + getLastBlockHash() + " " - + getExpectedHashStr() + " " - + String(getDifficulty())); - #endif - } - - const String &getLastBlockHash() const { return last_block_hash; } - const String &getExpectedHashStr() const { return expected_hash_str; } - const uint8_t *getExpectedHash() const { return expected_hash; } - unsigned int getDifficulty() const { return difficulty; } -}; - -#endif +#pragma GCC optimize("-Ofast") + +#ifndef MINING_JOB_H +#define MINING_JOB_H + +#include +#include +#include +#include +#include + +#include "DSHA1.h" +#include "Counter.h" +#include "Settings.h" + +// https://github.com/esp8266/Arduino/blob/master/cores/esp8266/TypeConversion.cpp +const char base36Chars[36] PROGMEM = { + '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', + 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z' +}; + +const uint8_t base36CharValues[75] PROGMEM{ + 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 0, 0, 0, 0, 0, 0, 0, // 0 to 9 + 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 0, 0, 0, 0, 0, 0, // Upper case letters + 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35 // Lower case letters +}; + +#define SPC_TOKEN ' ' +#define END_TOKEN '\n' +#define SEP_TOKEN ',' +#define IOT_TOKEN '@' + +struct MiningConfig { + String host = ""; + int port = 0; + String DUCO_USER = ""; + String RIG_IDENTIFIER = ""; + String MINER_KEY = ""; + String MINER_VER = SOFTWARE_VERSION; + #if defined(ESP8266) + // "High-band" 8266 diff + String START_DIFF = "ESP8266H"; + #elif defined(CONFIG_FREERTOS_UNICORE) + // Single core 32 diff + String START_DIFF = "ESP32S"; + #else + // Normal 32 diff + String START_DIFF = "ESP32"; + #endif + + MiningConfig(String DUCO_USER, String RIG_IDENTIFIER, String MINER_KEY) + : DUCO_USER(DUCO_USER), RIG_IDENTIFIER(RIG_IDENTIFIER), MINER_KEY(MINER_KEY) {} +}; + +class MiningJob { + +public: + MiningConfig *config; + int core = 0; + + MiningJob(int core, MiningConfig *config) { + this->core = core; + this->config = config; + this->client_buffer = ""; + dsha1 = new DSHA1(); + dsha1->warmup(); + generateRigIdentifier(); + } + + void blink(uint8_t count, uint8_t pin = LED_BUILTIN) { + #if defined(LED_BLINKING) + uint8_t state = HIGH; + + for (int x = 0; x < (count << 1); ++x) { + digitalWrite(pin, state ^= HIGH); + delay(50); + } + #else + digitalWrite(LED_BUILTIN, HIGH); + #endif + } + + bool max_micros_elapsed(unsigned long current, unsigned long max_elapsed) { + static unsigned long _start = 0; + + if ((current - _start) > max_elapsed) { + _start = current; + return true; + } + return false; + } + + /* void handleSystemEvents(void) { + #if defined(ESP32) && CORE == 2 + esp_task_wdt_reset(); + #endif + delay(10); // Required vTaskDelay by ESP-IDF + yield(); + ArduinoOTA.handle(); + }*/ + + void mine() { + connectToNode(); + askForJob(); + + dsha1->reset().write((const unsigned char *)getLastBlockHash().c_str(), getLastBlockHash().length()); + + int start_time = micros(); + max_micros_elapsed(start_time, 0); + #if defined(LED_BLINKING) + #if defined(BLUSHYBOX) + for (int i = 0; i < 72; i++) { + analogWrite(LED_BUILTIN, i); + delay(1); + } + #else + digitalWrite(LED_BUILTIN, LOW); + #endif + #endif + for (Counter<10> counter; counter < difficulty; ++counter) { + DSHA1 ctx = *dsha1; + ctx.write((const unsigned char *)counter.c_str(), counter.strlen()).finalize(hashArray); + + #ifndef CONFIG_FREERTOS_UNICORE + #if defined(ESP32) + #define SYSTEM_TIMEOUT 100000 // 10ms for esp32 looks like the lowest value without false watchdog triggers + #else + #define SYSTEM_TIMEOUT 500000 // 50ms for 8266 for same reason as above + #endif + if (max_micros_elapsed(micros(), SYSTEM_TIMEOUT)) { + handleSystemEvents(); + } + #endif + + if (memcmp(getExpectedHash(), hashArray, 20) == 0) { + unsigned long elapsed_time = micros() - start_time; + float elapsed_time_s = elapsed_time * .000001f; + share_count++; + + #if defined(LED_BLINKING) + #if defined(BLUSHYBOX) + for (int i = 72; i > 0; i--) { + analogWrite(LED_BUILTIN, i); + delay(1); + } + #else + digitalWrite(LED_BUILTIN, HIGH); + #endif + #endif + + if (String(core) == "0") { + hashrate = counter / elapsed_time_s; + submit(counter, hashrate, elapsed_time_s); + } else { + hashrate_core_two = counter / elapsed_time_s; + submit(counter, hashrate_core_two, elapsed_time_s); + } + + #if defined(BLUSHYBOX) + gauge_set(hashrate + hashrate_core_two); + #endif + + break; + } + } + } + +private: + String client_buffer; + uint8_t hashArray[20]; + String last_block_hash; + String expected_hash_str; + uint8_t expected_hash[20]; + DSHA1 *dsha1; + WiFiClient client; + String chipID = ""; + + #if defined(ESP8266) + #if defined(BLUSHYBOX) + String MINER_BANNER = "Official BlushyBox Miner (ESP8266)"; + #else + String MINER_BANNER = "Official ESP8266 Miner"; + #endif + #elif defined(CONFIG_FREERTOS_UNICORE) + String MINER_BANNER = "Official ESP32-S2 Miner"; + #else + #if defined(BLUSHYBOX) + String MINER_BANNER = "Official BlushyBox Miner (ESP32)"; + #else + String MINER_BANNER = "Official ESP32 Miner"; + #endif + #endif + + uint8_t *hexStringToUint8Array(const String &hexString, uint8_t *uint8Array, const uint32_t arrayLength) { + assert(hexString.length() >= arrayLength * 2); + const char *hexChars = hexString.c_str(); + for (uint32_t i = 0; i < arrayLength; ++i) { + uint8Array[i] = (pgm_read_byte(base36CharValues + hexChars[i * 2] - '0') << 4) + pgm_read_byte(base36CharValues + hexChars[i * 2 + 1] - '0'); + } + return uint8Array; + } + + void generateRigIdentifier() { + String AutoRigName = ""; + + #if defined(ESP8266) + chipID = String(ESP.getChipId(), HEX); + + if (strcmp(config->RIG_IDENTIFIER.c_str(), "Auto") != 0) + return; + + AutoRigName = "ESP8266-" + chipID; + AutoRigName.toUpperCase(); + config->RIG_IDENTIFIER = AutoRigName.c_str(); + #else + uint64_t chip_id = ESP.getEfuseMac(); + uint16_t chip = (uint16_t)(chip_id >> 32); // Prepare to print a 64 bit value into a char array + char fullChip[23]; + snprintf(fullChip, 23, "%04X%08X", chip, + (uint32_t)chip_id); // Store the (actually) 48 bit chip_id into a char array + + chipID = String(fullChip); + + if (strcmp(config->RIG_IDENTIFIER.c_str(), "Auto") != 0) + return; + // Autogenerate ID if required + AutoRigName = "ESP32-" + String(fullChip); + AutoRigName.toUpperCase(); + config->RIG_IDENTIFIER = AutoRigName.c_str(); + #endif + #if defined(SERIAL_PRINTING) + Serial.println("Core [" + String(core) + "] - Rig identifier: " + + config->RIG_IDENTIFIER); + #endif + } + + void connectToNode() { + if (client.connected()) return; + + unsigned int stopWatch = millis(); + #if defined(SERIAL_PRINTING) + Serial.println("Core [" + String(core) + "] - Connecting to a Duino-Coin node..."); + #endif + while (!client.connect(config->host.c_str(), config->port)) { + if (max_micros_elapsed(micros(), 100000)) { + handleSystemEvents(); + } + if (millis()-stopWatch>100000) ESP.restart(); + } + + waitForClientData(); + #if defined(SERIAL_PRINTING) + Serial.println("Core [" + String(core) + "] - Connected. Node reported version: " + + client_buffer); + #endif + + blink(BLINK_CLIENT_CONNECT); + + /* client.print("MOTD" + END_TOKEN); + waitForClientData(); + #if defined(SERIAL_PRINTING) + Serial.println("Core [" + String(core) + "] - MOTD: " + + client_buffer); + #endif */ + } + + void waitForClientData() { + client_buffer = ""; + + while (client.connected()) { + if (client.available()) { + client_buffer = client.readStringUntil(END_TOKEN); + if (client_buffer.length() == 1 && client_buffer[0] == END_TOKEN) + client_buffer = "???\n"; // NOTE: Should never happen + break; + } + } + } + + void submit(unsigned long counter, float hashrate, float elapsed_time_s) { + client.print(String(counter) + + SEP_TOKEN + String(hashrate) + + SEP_TOKEN + MINER_BANNER + + SPC_TOKEN + config->MINER_VER + + SEP_TOKEN + config->RIG_IDENTIFIER + + SEP_TOKEN + "DUCOID" + String(chipID) + + SEP_TOKEN + String(WALLET_ID) + + END_TOKEN); + + unsigned long ping_start = millis(); + waitForClientData(); + ping = millis() - ping_start; + + if (client_buffer == "GOOD") { + accepted_share_count++; + } + + #if defined(SERIAL_PRINTING) + Serial.println("Core [" + String(core) + "] - " + + client_buffer + + " share #" + String(share_count) + + " (" + String(counter) + ")" + + " hashrate: " + String(hashrate / 1000, 2) + " kH/s (" + + String(elapsed_time_s) + "s)\n"); + #endif + } + + bool parse() { + // Create a non-constant copy of the input string + char *job_str_copy = strdup(client_buffer.c_str()); + + if (job_str_copy) { + String tokens[3]; + char *token = strtok(job_str_copy, ","); + for (int i = 0; token != NULL && i < 3; i++) { + tokens[i] = token; + token = strtok(NULL, ","); + } + + last_block_hash = tokens[0]; + expected_hash_str = tokens[1]; + hexStringToUint8Array(expected_hash_str, expected_hash, 20); + difficulty = tokens[2].toInt() * 100 + 1; + + // Free the memory allocated by strdup + free(job_str_copy); + + return true; + } + else { + // Handle memory allocation failure + return false; + } + } + + + + void askForJob() { + Serial.println("Core [" + String(core) + "] - Asking for a new job for user: " + + String(config->DUCO_USER)); + + #if defined(USE_DS18B20) + sensors.requestTemperatures(); + float temp = sensors.getTempCByIndex(0); + #if defined(SERIAL_PRINTING) + Serial.println("DS18B20 reading: " + String(temp) + "°C"); + #endif + + client.print("JOB," + + String(config->DUCO_USER) + + SEP_TOKEN + config->START_DIFF + + SEP_TOKEN + String(config->MINER_KEY) + + SEP_TOKEN + "Temp:" + String(temp) + "*C" + + END_TOKEN); + #elif defined(USE_DHT) + float temp = dht.readTemperature(); + float hum = dht.readHumidity(); + #if defined(SERIAL_PRINTING) + Serial.println("DHT reading: " + String(temp) + "°C"); + Serial.println("DHT reading: " + String(hum) + "%"); + #endif + + client.print("JOB," + + String(config->DUCO_USER) + + SEP_TOKEN + config->START_DIFF + + SEP_TOKEN + String(config->MINER_KEY) + + SEP_TOKEN + "Temp:" + String(temp) + "*C" + + IOT_TOKEN + "Hum:" + String(hum) + "%" + + END_TOKEN); + #elif defined(USE_HSU07M) + float temp = read_hsu07m(); + #if defined(SERIAL_PRINTING) + Serial.println("HSU reading: " + String(temp) + "°C"); + #endif + + client.print("JOB," + + String(config->DUCO_USER) + + SEP_TOKEN + config->START_DIFF + + SEP_TOKEN + String(config->MINER_KEY) + + SEP_TOKEN + "Temp:" + String(temp) + "*C" + + END_TOKEN); + #elif defined(USE_INTERNAL_SENSOR) + float temp = 0; + temp_sensor_read_celsius(&temp); + #if defined(SERIAL_PRINTING) + Serial.println("Internal temp sensor reading: " + String(temp) + "°C"); + #endif + + client.print("JOB," + + String(config->DUCO_USER) + + SEP_TOKEN + config->START_DIFF + + SEP_TOKEN + String(config->MINER_KEY) + + SEP_TOKEN + "CPU Temp:" + String(temp) + "*C" + + END_TOKEN); + #else + client.print("JOB," + + String(config->DUCO_USER) + + SEP_TOKEN + config->START_DIFF + + SEP_TOKEN + String(config->MINER_KEY) + + END_TOKEN); + #endif + + waitForClientData(); + #if defined(SERIAL_PRINTING) + Serial.println("Core [" + String(core) + "] - Received job with size of " + + String(client_buffer.length()) + + " bytes " + client_buffer); + #endif + + parse(); + #if defined(SERIAL_PRINTING) + Serial.println("Core [" + String(core) + "] - Parsed job: " + + getLastBlockHash() + " " + + getExpectedHashStr() + " " + + String(getDifficulty())); + #endif + } + + const String &getLastBlockHash() const { return last_block_hash; } + const String &getExpectedHashStr() const { return expected_hash_str; } + const uint8_t *getExpectedHash() const { return expected_hash; } + unsigned int getDifficulty() const { return difficulty; } +}; + +#endif diff --git a/ESP_Code/Settings.h b/ESP_Code/Settings.h index 70f39a4c..9791c78b 100644 --- a/ESP_Code/Settings.h +++ b/ESP_Code/Settings.h @@ -1,225 +1,284 @@ -// Settings.h -#ifndef SETTINGS_H -#define SETTINGS_H - -// ---------------------- General settings ---------------------- // -// Change the part in brackets to your Duino-Coin username -extern char *DUCO_USER = "my_cool_username"; -// Change the part in brackets to your mining key (if you have set it in the wallet) -extern char *MINER_KEY = "mySecretPass"; -// Change the part in brackets if you want to set a custom miner name -// Use Auto to autogenerate, None for no custom identifier -extern char *RIG_IDENTIFIER = "None"; -// Change the part in brackets to your WiFi name -extern const char SSID[] = "SSID"; -// Change the part in brackets to your WiFi password -extern const char PASSWORD[] = "PASSW0RD"; -// -------------------------------------------------------------- // - -// -------------------- Advanced options ------------------------ // -// Uncomment if you want to host the dashboard page (available on ESPs IP address and mDNS) -// #define WEB_DASHBOARD - -// Comment out the line below if you wish to disable LED blinking -#define LED_BLINKING - -// Uncomment if you want to use LAN8720. WLAN-credentials will be ignored using LAN -// Select correct Board in ArduinoIDE!!! Really! -// #define USE_LAN - -// Comment out the line below if you wish to disable Serial printing -#define SERIAL_PRINTING - -// Edit the line below if you wish to change the serial speed (low values may reduce performance but are less prone to interference) -#define SERIAL_BAUDRATE 500000 - -// ESP8266 WDT loop watchdog. Do not edit this value, but if you must - do not set it too low or it will falsely trigger during mining! -#define LWD_TIMEOUT 30000 - -// Uncomment to disable ESP32 brownout detector if you're suffering from faulty insufficient power detection -// #define DISABLE_BROWNOUT - -// Uncomment to enable WiFiManager captive portal in AP mode -// The board will create its own network you connect to and change the settings -// #define CAPTIVE_PORTAL -// -------------------------------------------------------------- // - -// ------------------------ Displays ---------------------------- // - -// Uncomment to enable a SSD1306 OLED screen on the I2C bus to display mining info in real time -// Default connections (can be overriden by using a different u8g2 initializer, see line 140): -// GND - GND -// VCC - 5V or 3.3V depending on display -// SCL - GPIO22 (ESP32) or GPIO5 (D2 on ESP8266) or GPIO35 (ESP32-S2) -// SDA - GPIO21 (ESP32) or GPIO4 (D1 on ESP8266) or GPIO33 (ESP32-S2) -// #define DISPLAY_SSD1306 - -// Uncomment to enable a 16x2 LCD screen on a direct bus to display mining info in real time -// See line 150 for connections and initializer -// #define DISPLAY_16X2 - -// Uncomment if your device is a Duino BlushyBox device -// #define BLUSHYBOX -// -------------------------------------------------------------- // - -// ---------------------- IoT examples -------------------------- // -// https://github.com/revoxhere/duino-coin/wiki/Duino's-take-on-the-Internet-of-Things - -// Uncomment the line below if you wish to use the internal temperature sensor (Duino IoT example) -// Only ESP32-S2, -S3, -H2, -C2, -C3, -C6 and some old models have one! -// More info: https://www.espboards.dev/blog/esp32-inbuilt-temperature-sensor/ -// NOTE: Mining performance will decrease by about 20 kH/s! -// #define USE_INTERNAL_SENSOR - -// Uncomment the line below if you wish to use a DS18B20 temperature sensor (Duino IoT example) -// NOTE: Mining performance should stay the same -// #define USE_DS18B20 - -// Uncomment the line below if you wish to use a DHT11/22 temperature and humidity sensor (Duino IoT example) -// NOTE: Mining performance should stay the same -// #define USE_DHT - -// Uncomment the line below if you wish to use a HSU07M sensor (Duino IoT Example) -// NOTE: Untested as of right now -// #define USE_HSU07M -// -------------------------------------------------------------- // - -// ---------------- Variables and definitions ------------------- // -// You generally do not need to edit stuff below this line -// unless you're know what you're doing. - -#if defined(ESP8266) - // ESP8266 - #define LED_BUILTIN 2 -#elif defined(CONFIG_FREERTOS_UNICORE) - #if defined(CONFIG_IDF_TARGET_ESP32C3) - // ESP32-C3 - #define LED_BUILTIN 8 - #else - // ESP32-S2 - #define LED_BUILTIN 15 - #endif -#else - // ESP32 - #ifndef LED_BUILTIN - #define LED_BUILTIN 2 - #endif - #if defined(BLUSHYBOX) - #define LED_BUILTIN 4 - #endif -#endif - -#define BLINK_SETUP_COMPLETE 2 -#define BLINK_CLIENT_CONNECT 5 - -#define SOFTWARE_VERSION "4.3" -extern unsigned int hashrate = 0; -extern unsigned int hashrate_core_two = 0; -extern unsigned int difficulty = 0; -extern unsigned long share_count = 0; -extern unsigned long accepted_share_count = 0; -extern String node_id = ""; -extern String WALLET_ID = ""; -extern unsigned int ping = 0; - -#if defined(USE_INTERNAL_SENSOR) - #include "driver/temp_sensor.h" -#endif - -#if defined(USE_DS18B20) - // Install OneWire and DallasTemperature libraries if you get an error - #include - #include - // Change 12 to the pin you've connected your sensor to - const int DSPIN = 12; - - OneWire oneWire(DSPIN); - DallasTemperature extern sensors(&oneWire); -#endif - -#if defined(USE_DHT) - // Install "DHT sensor library" if you get an error - #include - // Change 12 to the pin you've connected your sensor to - #define DHTPIN 12 - // Set DHT11 or DHT22 type accordingly - #define DHTTYPE DHT11 - - DHT extern dht(DHTPIN, DHTTYPE); -#endif - -#if defined(DISPLAY_SSD1306) - // Install "u8g2" if you get an error - #include - #include - // Display definition from the U8G2 library. Edit if you use a different display - // For software I2C, use ..._F_SW_I2C and define the pins in the initializer - U8G2_SSD1306_128X64_NONAME_F_HW_I2C u8g2(U8G2_R0, U8X8_PIN_NONE); -#endif - -#if defined(DISPLAY_16X2) - #include "Wire.h" - // Install "Adafruit_LiquidCrystal" if you get an error - #include "Adafruit_LiquidCrystal.h" - - // initialize the library with the numbers of the interface pins - // RS E D4 D5 D6 D7 - Adafruit_LiquidCrystal lcd(1, 2, 3, 4, 5, 6); -#endif - -#if defined(USE_HSU07M) - #include "Wire.h" - #define HSU07M_ADDRESS 0x4B // Change this if your sensor has a different address - - float read_hsu07m() { - Wire.beginTransmission(HSU07M_ADDRESS); - Wire.write(0x00); - Wire.endTransmission(); - delay(100); - Wire.requestFrom(HSU07M_ADDRESS, 2); - if(Wire.available() >= 2) { - byte tempMSB = Wire.read(); - byte tempLSB = Wire.read(); - int tempRaw = (tempMSB << 8) | tempLSB; - float tempC = (tempRaw / 16.0) - 40.0; - return tempC; - } - return -1.0; - } -#endif - -#if defined(BLUSHYBOX) - #define GAUGE_PIN 5 - #define GAUGE_MAX 190 - #define GAUGE_MIN 0 - #if defined(ESP8266) - #define GAUGE_MAX_HR 80000 - #else - #define GAUGE_MAX_HR 200000 - #endif - extern float hashrate_old = 0.0; - - void gauge_set(float hashrate) { - float old = hashrate_old; - float new_val = hashrate; - - if (hashrate_old == 0) { - float delta = (new_val - old) / 50; - for (int x=0; x < 50; x++) { - analogWrite(5, map(old + x*delta, 0, GAUGE_MAX_HR, GAUGE_MIN, GAUGE_MAX) + random(0, 10)); - delay(20); - } - } else { - float delta = (new_val - old) / 10; - for (int x=0; x < 10; x++) { - analogWrite(5, map(old + x*delta, 0, GAUGE_MAX_HR, GAUGE_MIN, GAUGE_MAX) + random(0, 10)); - delay(10); - } - } - hashrate_old = hashrate; - } -#endif - -IPAddress DNS_SERVER(1, 1, 1, 1); // Cloudflare DNS server - -#endif // End of SETTINGS_H +// Settings.h +#ifndef SETTINGS_H +#define SETTINGS_H + +extern bool displayLock = false; + +// ---------------------- General settings ---------------------- // +// Change the part in brackets to your Duino-Coin username +extern char *DUCO_USER = "your username"; +// Change the part in brackets to your mining key (if you have set it in the wallet) +extern char *MINER_KEY = "your mining key"; +// Change the part in brackets if you want to set a custom miner name +// Use Auto to autogenerate, None for no custom identifier +extern char *RIG_IDENTIFIER = "rig id"; +// Change the part in brackets to your WiFi name +extern const char SSID[] = "your_wifi_ssid"; +// Change the part in brackets to your WiFi password +extern const char PASSWORD[] = "your_wifi_psswd"; +// -------------------------------------------------------------- // + +//-------------------------Balance display settings--------------------------// +extern bool first_display = true; +extern bool first_start = true; // to activate balance fetching on boot without waiting for the delay +extern String ducoReportJsonUrl = ""; // url init +extern const int run_in_ms = 300000; // 5 minutes between each balance update +extern double result_balance_balance = 0; +extern String result_balance_username = "username"; +extern unsigned int total_miner = 0; +//---------------------------------------------------------------------------// + +//---------------------------Time Display------------------------------------// +extern char mytime[6] =""; +extern char mydate[12] = ""; +extern char myday[10] = ""; +// -------------------- Advanced options ------------------------ // +// Uncomment if you want to host the dashboard page (available on ESPs IP address and mDNS) + #define WEB_DASHBOARD + +// Comment out the line below if you wish to disable LED blinking +#define LED_BLINKING + +// Uncomment if you want to use LAN8720. WLAN-credentials will be ignored using LAN +// Select correct Board in ArduinoIDE!!! Really! +// #define USE_LAN + +// Comment out the line below if you wish to disable Serial printing +#define SERIAL_PRINTING + +// Edit the line below if you wish to change the serial speed (low values may reduce performance but are less prone to interference) +#define SERIAL_BAUDRATE 500000 + +// ESP8266 WDT loop watchdog. Do not edit this value, but if you must - do not set it too low or it will falsely trigger during mining! +#define LWD_TIMEOUT 30000 + +// Uncomment to disable ESP32 brownout detector if you're suffering from faulty insufficient power detection +// #define DISABLE_BROWNOUT + +// Uncomment to enable WiFiManager captive portal in AP mode +// The board will create its own network you connect to and change the settings + #define CAPTIVE_PORTAL +// -------------------------------------------------------------- // + +// ------------------------ Displays ---------------------------- // + +// Uncomment to enable a SSD1306 OLED screen on the I2C bus to display mining info in real time +// Default connections (can be overriden by using a different u8g2 initializer, see line 140): +// GND - GND +// VCC - 5V or 3.3V depending on display +// SCL - GPIO22 (ESP32) or GPIO5 (D2 on ESP8266) or GPIO35 (ESP32-S2) +// SDA - GPIO21 (ESP32) or GPIO4 (D1 on ESP8266) or GPIO33 (ESP32-S2) +// #define DISPLAY_SSD1306 + +//uncomment to enable Lilygo tt display 1.14 module +// #define DISPLAY_114 + +// uncomment for tft7735 display +// #define DISPLAY_7735 + +//uncomment to enable ESP32 2432S08 module + #define DISPLAY_2432S08 + +// Uncomment to enable a 16x2 LCD screen on a direct bus to display mining info in real time +// See line 150 for connections and initializer +// #define DISPLAY_16X2 + +// Uncomment if your device is a Duino BlushyBox device +// #define BLUSHYBOX +// -------------------------------------------------------------- // + +// ---------------------- IoT examples -------------------------- // +// https://github.com/revoxhere/duino-coin/wiki/Duino's-take-on-the-Internet-of-Things + +// Uncomment the line below if you wish to use the internal temperature sensor (Duino IoT example) +// Only ESP32-S2, -S3, -H2, -C2, -C3, -C6 and some old models have one! +// More info: https://www.espboards.dev/blog/esp32-inbuilt-temperature-sensor/ +// NOTE: Mining performance will decrease by about 20 kH/s! +// #define USE_INTERNAL_SENSOR + +// Uncomment the line below if you wish to use a DS18B20 temperature sensor (Duino IoT example) +// NOTE: Mining performance should stay the same +// #define USE_DS18B20 + +// Uncomment the line below if you wish to use a DHT11/22 temperature and humidity sensor (Duino IoT example) +// NOTE: Mining performance should stay the same +// #define USE_DHT + +// Uncomment the line below if you wish to use a HSU07M sensor (Duino IoT Example) +// NOTE: Untested as of right now +// #define USE_HSU07M +// -------------------------------------------------------------- // + +// ---------------- Variables and definitions ------------------- // +// You generally do not need to edit stuff below this line +// unless you're know what you're doing. + +#if defined(ESP8266) + // ESP8266 + #define LED_BUILTIN 2 +#elif defined(CONFIG_FREERTOS_UNICORE) + #if defined(CONFIG_IDF_TARGET_ESP32C3) + // ESP32-C3 + #define LED_BUILTIN 8 + #else + // ESP32-S2 + #define LED_BUILTIN 15 + #endif +#else + // ESP32 + #ifndef LED_BUILTIN + #define LED_BUILTIN 2 + #endif + #if defined(BLUSHYBOX) + #define LED_BUILTIN 4 + #endif + #if defined(DISPLAY_2432S08) + #define LED_BUILTIN 4 + #endif +#endif + +#define BLINK_SETUP_COMPLETE 2 +#define BLINK_CLIENT_CONNECT 5 + +#define SOFTWARE_VERSION "4.3" + +extern unsigned int hashrate = 0; +extern unsigned int hashrate_core_two = 0; +extern unsigned int difficulty = 0; +extern unsigned long share_count = 0; +extern unsigned long accepted_share_count = 0; +extern String node_id = ""; +extern String WALLET_ID = ""; +extern unsigned int ping = 0; + +#if defined(USE_INTERNAL_SENSOR) + #include "driver/temp_sensor.h" +#endif + +#if defined(USE_DS18B20) + // Install OneWire and DallasTemperature libraries if you get an error + #include + #include + // Change 12 to the pin you've connected your sensor to + const int DSPIN = 12; + + OneWire oneWire(DSPIN); + DallasTemperature extern sensors(&oneWire); +#endif + +#if defined(USE_DHT) + // Install "DHT sensor library" if you get an error + #include + // Change 12 to the pin you've connected your sensor to + #define DHTPIN 12 + // Set DHT11 or DHT22 type accordingly + #define DHTTYPE DHT11 + + DHT extern dht(DHTPIN, DHTTYPE); +#endif + +#if defined(DISPLAY_SSD1306) + // Install "u8g2" if you get an error + #include + #include + // Display definition from the U8G2 library. Edit if you use a different display + // For software I2C, use ..._F_SW_I2C and define the pins in the initializer + U8G2_SSD1306_128X64_NONAME_F_HW_I2C u8g2(U8G2_R0, U8X8_PIN_NONE); +#endif + +#if defined(DISPLAY_16X2) + #include "Wire.h" + // Install "Adafruit_LiquidCrystal" if you get an error + #include "Adafruit_LiquidCrystal.h" + + // initialize the library with the numbers of the interface pins + // RS E D4 D5 D6 D7 + Adafruit_LiquidCrystal lcd(1, 2, 3, 4, 5, 6); +#endif + +#if defined(DISPLAY_114) + // Install "lilygo lib TFT_eSPI.h" if you get an error + #include + #include + #include + // Display definition from the tft_eSPI library. Edit if you use a different display + TFT_eSPI tft = TFT_eSPI(); // Invoke library, pins defined in User_Setup.h +#endif + +#if defined(DISPLAY_2432S08) + // Install "lilygo lib TFT_eSPI.h" if you get an error + #include "esp_sntp.h" + #include + #include + #include + + + // Display definition from the tft_eSPI library. Edit if you use a different display + TFT_eSPI tft = TFT_eSPI(); // Invoke library, pins defined in User_Setup.h +#endif + +#if defined(DISPLAY_7735) + // Install "lilygo lib TFT_eSPI.h" if you get an error + #include + #include + #include + // Display definition from the tft_eSPI library. Edit if you use a different display + TFT_eSPI tft = TFT_eSPI(); // Invoke library, pins defined in User_Setup.h +#endif + +#if defined(USE_HSU07M) + #include "Wire.h" + #define HSU07M_ADDRESS 0x4B // Change this if your sensor has a different address + + float read_hsu07m() { + Wire.beginTransmission(HSU07M_ADDRESS); + Wire.write(0x00); + Wire.endTransmission(); + delay(100); + Wire.requestFrom(HSU07M_ADDRESS, 2); + if(Wire.available() >= 2) { + byte tempMSB = Wire.read(); + byte tempLSB = Wire.read(); + int tempRaw = (tempMSB << 8) | tempLSB; + float tempC = (tempRaw / 16.0) - 40.0; + return tempC; + } + return -1.0; + } +#endif + +#if defined(BLUSHYBOX) + #define GAUGE_PIN 5 + #define GAUGE_MAX 190 + #define GAUGE_MIN 0 + #if defined(ESP8266) + #define GAUGE_MAX_HR 80000 + #else + #define GAUGE_MAX_HR 200000 + #endif + extern float hashrate_old = 0.0; + + void gauge_set(float hashrate) { + float old = hashrate_old; + float new_val = hashrate; + + if (hashrate_old == 0) { + float delta = (new_val - old) / 50; + for (int x=0; x < 50; x++) { + analogWrite(5, map(old + x*delta, 0, GAUGE_MAX_HR, GAUGE_MIN, GAUGE_MAX) + random(0, 10)); + delay(20); + } + } else { + float delta = (new_val - old) / 10; + for (int x=0; x < 10; x++) { + analogWrite(5, map(old + x*delta, 0, GAUGE_MAX_HR, GAUGE_MIN, GAUGE_MAX) + random(0, 10)); + delay(10); + } + } + hashrate_old = hashrate; + } +#endif + +IPAddress DNS_SERVER(1, 1, 1, 1); // Cloudflare DNS server + +#endif // End of SETTINGS_H diff --git a/ESP_Code/User_Setup.h b/ESP_Code/User_Setup.h new file mode 100644 index 00000000..d0a16e68 --- /dev/null +++ b/ESP_Code/User_Setup.h @@ -0,0 +1,424 @@ +// USER DEFINED SETTINGS +// Set driver type, fonts to be loaded, pins used and SPI control method etc. +// +// See the User_Setup_Select.h file if you wish to be able to define multiple +// setups and then easily select which setup file is used by the compiler. +// +// If this file is edited correctly then all the library example sketches should +// run without the need to make any more changes for a particular hardware setup! +// Note that some sketches are designed for a particular TFT pixel width/height + +// User defined information reported by "Read_User_Setup" test & diagnostics example +#define USER_SETUP_INFO "User_Setup" + +// Define to disable all #warnings in library (can be put in User_Setup_Select.h) +//#define DISABLE_ALL_LIBRARY_WARNINGS + +// ################################################################################## +// +// Section 1. Call up the right driver file and any options for it +// +// ################################################################################## + +// Define STM32 to invoke optimised processor support (only for STM32) +//#define STM32 + +// Defining the STM32 board allows the library to optimise the performance +// for UNO compatible "MCUfriend" style shields +//#define NUCLEO_64_TFT +//#define NUCLEO_144_TFT + +// STM32 8-bit parallel only: +// If STN32 Port A or B pins 0-7 are used for 8-bit parallel data bus bits 0-7 +// then this will improve rendering performance by a factor of ~8x +//#define STM_PORTA_DATA_BUS +//#define STM_PORTB_DATA_BUS + +// Tell the library to use parallel mode (otherwise SPI is assumed) +//#define TFT_PARALLEL_8_BIT +//#defined TFT_PARALLEL_16_BIT // **** 16-bit parallel ONLY for RP2040 processor **** + +// Display type - only define if RPi display +//#define RPI_DISPLAY_TYPE // 20MHz maximum SPI + +// Only define one driver, the other ones must be commented out +// select ST7789_DRIVER for tt-go 1.14 display, st7735_DRIVER for TFT 7735 display, ILI9341_2 for ESP32-2432S028 + +//#define ILI9341_DRIVER // Generic driver for common displays +#define ILI9341_2_DRIVER // Alternative ILI9341 driver, see https://github.com/Bodmer/TFT_eSPI/issues/1172 // FOR CYD ESP32-2432S028 +//#define ST7735_DRIVER // Define additional parameters below for this display +//#define ILI9163_DRIVER // Define additional parameters below for this display +//#define S6D02A1_DRIVER +//#define RPI_ILI9486_DRIVER // 20MHz maximum SPI +//#define HX8357D_DRIVER +//#define ILI9481_DRIVER +//#define ILI9486_DRIVER +//#define ILI9488_DRIVER // WARNING: Do not connect ILI9488 display SDO to MISO if other devices share the SPI bus (TFT SDO does NOT tristate when CS is high) +//#define ST7789_DRIVER // Full configuration option, define additional parameters below for this display +//#define ST7789_2_DRIVER // Minimal configuration option, define additional parameters below for this display +//#define R61581_DRIVER +//#define RM68140_DRIVER +//#define ST7796_DRIVER +//#define SSD1351_DRIVER +//#define SSD1963_480_DRIVER +//#define SSD1963_800_DRIVER +//#define SSD1963_800ALT_DRIVER +//#define ILI9225_DRIVER +//#define GC9A01_DRIVER + +// Some displays support SPI reads via the MISO pin, other displays have a single +// bi-directional SDA pin and the library will try to read this via the MOSI line. +// To use the SDA line for reading data from the TFT uncomment the following line: + +// #define TFT_SDA_READ // This option is for ESP32 ONLY, tested with ST7789 and GC9A01 display only + +// For ST7735, ST7789 and ILI9341 ONLY, define the colour order IF the blue and red are swapped on your display +// Try ONE option at a time to find the correct colour order for your display + +// #define TFT_RGB_ORDER TFT_RGB // Colour order Red-Green-Blue +// #define TFT_RGB_ORDER TFT_BGR // Colour order Blue-Green-Red + +// For M5Stack ESP32 module with integrated ILI9341 display ONLY, remove // in line below + +// #define M5STACK + +// For ST7789, ST7735, ILI9163 and GC9A01 ONLY, define the pixel width and height in portrait orientation +// for TT-GO 1.14 by Lilygo, set TFT_WIDTH to 240 and TFT_HEIGHT to 135 +// FOR ST7735 DISPLAY, SET TFT_WIDTH to 128 and TFT_HEIGHT to 160 +// #define TFT_WIDTH 80 +// #define TFT_WIDTH 128 // st7735 +// #define TFT_WIDTH 135 // Lilygo TT-go 1.14 +// #define TFT_WIDTH 172 // ST7789 172 x 320 +// #define TFT_WIDTH 170 // ST7789 170 x 320 + #define TFT_WIDTH 240 // ST7789 240 x 240 and 240 x 320 & ESP32-2432S028 +// #define TFT_HEIGHT 160 +// #define TFT_HEIGHT 135 +// #define TFT_HEIGHT 160 // st7735 + #define TFT_HEIGHT 320 // ST7789 240 x 320 & ESP32-2432S028 +// #define TFT_HEIGHT 240 // GC9A01 240 x 240 and Lilygo TT-GO 1.14 + +// For ST7735 ONLY, define the type of display, originally this was based on the +// colour of the tab on the screen protector film but this is not always true, so try +// out the different options below if the screen does not display graphics correctly, +// e.g. colours wrong, mirror images, or stray pixels at the edges. +// Comment out ALL BUT ONE of these options for a ST7735 display driver, save this +// this User_Setup file, then rebuild and upload the sketch to the board again: + +// #define ST7735_INITB +// #define ST7735_GREENTAB +// #define ST7735_GREENTAB2 +// #define ST7735_GREENTAB3 +// #define ST7735_GREENTAB128 // For 128 x 128 display +// #define ST7735_GREENTAB160x80 // For 160 x 80 display (BGR, inverted, 26 offset) +// #define ST7735_ROBOTLCD // For some RobotLCD Arduino shields (128x160, BGR, https://docs.arduino.cc/retired/getting-started-guides/TFT) +// #define ST7735_REDTAB +// #define ST7735_BLACKTAB +// #define ST7735_REDTAB160x80 // For 160 x 80 display with 24 pixel offset + +// If colours are inverted (white shows as black) then uncomment one of the next +// 2 lines try both options, one of the options should correct the inversion. + +// #define TFT_INVERSION_ON +// #define TFT_INVERSION_OFF + + +// ################################################################################## +// +// Section 2. Define the pins that are used to interface with the display here +// +// ################################################################################## + +// If a backlight control signal is available then define the TFT_BL pin in Section 2 +// below. The backlight will be turned ON when tft.begin() is called, but the library +// needs to know if the LEDs are ON with the pin HIGH or LOW. If the LEDs are to be +// driven with a PWM signal or turned OFF/ON then this must be handled by the user +// sketch. e.g. with digitalWrite(TFT_BL, LOW); + +// #define TFT_BL 32 // LED back-light control pin +// #define TFT_BACKLIGHT_ON HIGH // Level to turn ON back-light (HIGH or LOW) + + + +// We must use hardware SPI, a minimum of 3 GPIO pins is needed. +// Typical setup for ESP8266 NodeMCU ESP-12 is : +// +// Display SDO/MISO to NodeMCU pin D6 (or leave disconnected if not reading TFT) +// Display LED to NodeMCU pin VIN (or 5V, see below) +// Display SCK to NodeMCU pin D5 +// Display SDI/MOSI to NodeMCU pin D7 +// Display DC (RS/AO)to NodeMCU pin D3 +// Display RESET to NodeMCU pin D4 (or RST, see below) +// Display CS to NodeMCU pin D8 (or GND, see below) +// Display GND to NodeMCU pin GND (0V) +// Display VCC to NodeMCU 5V or 3.3V +// +// The TFT RESET pin can be connected to the NodeMCU RST pin or 3.3V to free up a control pin +// +// The DC (Data Command) pin may be labelled AO or RS (Register Select) +// +// With some displays such as the ILI9341 the TFT CS pin can be connected to GND if no more +// SPI devices (e.g. an SD Card) are connected, in this case comment out the #define TFT_CS +// line below so it is NOT defined. Other displays such at the ST7735 require the TFT CS pin +// to be toggled during setup, so in these cases the TFT_CS line must be defined and connected. +// +// The NodeMCU D0 pin can be used for RST +// +// +// Note: only some versions of the NodeMCU provide the USB 5V on the VIN pin +// If 5V is not available at a pin you can use 3.3V but backlight brightness +// will be lower. + + +// ###### EDIT THE PIN NUMBERS IN THE LINES FOLLOWING TO SUIT YOUR ESP8266 SETUP ###### + +// For NodeMCU - use pin numbers in the form PIN_Dx where Dx is the NodeMCU pin designation +//#define TFT_MISO PIN_D6 // Automatically assigned with ESP8266 if not defined +//#define TFT_MOSI PIN_D7 // Automatically assigned with ESP8266 if not defined +//#define TFT_SCLK PIN_D5 // Automatically assigned with ESP8266 if not defined + +//#define TFT_CS PIN_D8 // Chip select control pin D8 +//#define TFT_DC PIN_D3 // Data Command control pin +//#define TFT_RST PIN_D4 // Reset pin (could connect to NodeMCU RST, see next line) +//#define TFT_RST -1 // Set TFT_RST to -1 if the display RESET is connected to NodeMCU RST or 3.3V + + +//#define TFT_BL PIN_D1 // LED back-light (only for ST7789 with backlight control pin) + +//#define TOUCH_CS PIN_D2 // Chip select pin (T_CS) of touch screen + +//#define TFT_WR PIN_D2 // Write strobe for modified Raspberry Pi TFT only + + +// ###### FOR ESP8266 OVERLAP MODE EDIT THE PIN NUMBERS IN THE FOLLOWING LINES ###### + +// Overlap mode shares the ESP8266 FLASH SPI bus with the TFT so has a performance impact +// but saves pins for other functions. It is best not to connect MISO as some displays +// do not tristate that line when chip select is high! +// Note: Only one SPI device can share the FLASH SPI lines, so a SPI touch controller +// cannot be connected as well to the same SPI signals. +// On NodeMCU 1.0 SD0=MISO, SD1=MOSI, CLK=SCLK to connect to TFT in overlap mode +// On NodeMCU V3 S0 =MISO, S1 =MOSI, S2 =SCLK +// In ESP8266 overlap mode the following must be defined + +//#define TFT_SPI_OVERLAP + +// In ESP8266 overlap mode the TFT chip select MUST connect to pin D3 +//#define TFT_CS PIN_D3 +//#define TFT_DC PIN_D5 // Data Command control pin +//#define TFT_RST PIN_D4 // Reset pin (could connect to NodeMCU RST, see next line) +//#define TFT_RST -1 // Set TFT_RST to -1 if the display RESET is connected to NodeMCU RST or 3.3V + + +// ###### EDIT THE PIN NUMBERS IN THE LINES FOLLOWING TO SUIT YOUR ESP32 SETUP ###### + +// For ESP32 Dev board (7735 tft) +// The hardware SPI can be mapped to any pins +// ONLY ONE DISPLAY TYPE SHOULD BE UNCOMMENTED + +// FOR TFT 7735 Display uncomment the following lines + +//#define TFT_MOSI 23 +//#define TFT_SCLK 18 +//#define TFT_CS 15 // Chip select control pin +//#define TFT_DC 16 // Data Command control pin +//#define TFT_RST 4 // Reset pin (could connect to RST pin) + +// FOR Lilygo TT-GO 1.14 uncomment the following lines + +////#define TFT_MOSI 19 +//#define TFT_SCLK 36 +//#define TFT_CS 37 // Chip select control pin +//#define TFT_DC 35 // Data Command control pin +//#define TFT_BL 38 // LED back-light +//#define TFT_RST 33 // Reset pin (could connect to RST pin) + +// For ESP32 Dev board (only tested with ILI9341 display) Tested and working with ESP32-2432S028 aka CYD +// The hardware SPI can be mapped to any pins + +#define TFT_BL 21 // LED back-light control pin +#define TFT_BACKLIGHT_ON HIGH // Level to turn ON back-light (HIGH or LOW) + +#define TFT_MISO 12 +#define TFT_MOSI 13 +#define TFT_SCLK 14 +#define TFT_CS 15 // Chip select control pin +#define TFT_DC 2 // Data Command control pin +// #define TFT_RST 2 // Reset pin (could connect to RST pin) +#define TFT_RST -1 // Set TFT_RST to -1 if display RESET is connected to ESP32 board RST + +//#define TOUCH_MISO 39 +//#define TOUCH_MOSI 32 +//#define TOUCH_SCLK 25 +#define TOUCH_CS 33 +//#define TOUCH_IRQ 36 + +// For ESP32 Dev board (only tested with GC9A01 display) +// The hardware SPI can be mapped to any pins + +//#define TFT_MOSI 15 // In some display driver board, it might be written as "SDA" and so on. +//#define TFT_SCLK 14 +//#define TFT_CS 5 // Chip select control pin +//#define TFT_DC 27 // Data Command control pin +//#define TFT_RST 33 // Reset pin (could connect to Arduino RESET pin) +//#define TFT_BL 22 // LED back-light + +//#define TOUCH_CS 21 // Chip select pin (T_CS) of touch screen + +//#define TFT_WR 22 // Write strobe for modified Raspberry Pi TFT only + +// For the M5Stack module use these #define lines +//#define TFT_MISO 19 +//#define TFT_MOSI 23 +//#define TFT_SCLK 18 +//#define TFT_CS 14 // Chip select control pin +//#define TFT_DC 27 // Data Command control pin +//#define TFT_RST 33 // Reset pin (could connect to Arduino RESET pin) +//#define TFT_BL 32 // LED back-light (required for M5Stack) + +// ###### EDIT THE PINs BELOW TO SUIT YOUR ESP32 PARALLEL TFT SETUP ###### + +// The library supports 8-bit parallel TFTs with the ESP32, the pin +// selection below is compatible with ESP32 boards in UNO format. +// Wemos D32 boards need to be modified, see diagram in Tools folder. +// Only ILI9481 and ILI9341 based displays have been tested! + +// Parallel bus is only supported for the STM32 and ESP32 +// Example below is for ESP32 Parallel interface with UNO displays + +// Tell the library to use 8-bit parallel mode (otherwise SPI is assumed) +//#define TFT_PARALLEL_8_BIT + +// The ESP32 and TFT the pins used for testing are: +//#define TFT_CS 33 // Chip select control pin (library pulls permanently low +//#define TFT_DC 15 // Data Command control pin - must use a pin in the range 0-31 +//#define TFT_RST 32 // Reset pin, toggles on startup + +//#define TFT_WR 4 // Write strobe control pin - must use a pin in the range 0-31 +//#define TFT_RD 2 // Read strobe control pin + +//#define TFT_D0 12 // Must use pins in the range 0-31 for the data bus +//#define TFT_D1 13 // so a single register write sets/clears all bits. +//#define TFT_D2 26 // Pins can be randomly assigned, this does not affect +//#define TFT_D3 25 // TFT screen update performance. +//#define TFT_D4 17 +//#define TFT_D5 16 +//#define TFT_D6 27 +//#define TFT_D7 14 + +// ###### EDIT THE PINs BELOW TO SUIT YOUR STM32 SPI TFT SETUP ###### + +// The TFT can be connected to SPI port 1 or 2 +//#define TFT_SPI_PORT 1 // SPI port 1 maximum clock rate is 55MHz +//#define TFT_MOSI PA7 +//#define TFT_MISO PA6 +//#define TFT_SCLK PA5 + +//#define TFT_SPI_PORT 2 // SPI port 2 maximum clock rate is 27MHz +//#define TFT_MOSI PB15 +//#define TFT_MISO PB14 +//#define TFT_SCLK PB13 + +// Can use Ardiuno pin references, arbitrary allocation, TFT_eSPI controls chip select +//#define TFT_CS D5 // Chip select control pin to TFT CS +//#define TFT_DC D6 // Data Command control pin to TFT DC (may be labelled RS = Register Select) +//#define TFT_RST D7 // Reset pin to TFT RST (or RESET) +// OR alternatively, we can use STM32 port reference names PXnn +//#define TFT_CS PE11 // Nucleo-F767ZI equivalent of D5 +//#define TFT_DC PE9 // Nucleo-F767ZI equivalent of D6 +//#define TFT_RST PF13 // Nucleo-F767ZI equivalent of D7 + +//#define TFT_RST -1 // Set TFT_RST to -1 if the display RESET is connected to processor reset + // Use an Arduino pin for initial testing as connecting to processor reset + // may not work (pulse too short at power up?) + +// ################################################################################## +// +// Section 3. Define the fonts that are to be used here +// +// ################################################################################## + +// Comment out the #defines below with // to stop that font being loaded +// The ESP8366 and ESP32 have plenty of memory so commenting out fonts is not +// normally necessary. If all fonts are loaded the extra FLASH space required is +// about 17Kbytes. To save FLASH space only enable the fonts you need! + +#define LOAD_GLCD // Font 1. Original Adafruit 8 pixel font needs ~1820 bytes in FLASH +#define LOAD_FONT2 // Font 2. Small 16 pixel high font, needs ~3534 bytes in FLASH, 96 characters +#define LOAD_FONT4 // Font 4. Medium 26 pixel high font, needs ~5848 bytes in FLASH, 96 characters +#define LOAD_FONT6 // Font 6. Large 48 pixel font, needs ~2666 bytes in FLASH, only characters 1234567890:-.apm +#define LOAD_FONT7 // Font 7. 7 segment 48 pixel font, needs ~2438 bytes in FLASH, only characters 1234567890:-. +#define LOAD_FONT8 // Font 8. Large 75 pixel font needs ~3256 bytes in FLASH, only characters 1234567890:-. +//#define LOAD_FONT8N // Font 8. Alternative to Font 8 above, slightly narrower, so 3 digits fit a 160 pixel TFT +#define LOAD_GFXFF // FreeFonts. Include access to the 48 Adafruit_GFX free fonts FF1 to FF48 and custom fonts + +// Comment out the #define below to stop the SPIFFS filing system and smooth font code being loaded +// this will save ~20kbytes of FLASH +#define SMOOTH_FONT + + +// ################################################################################## +// +// Section 4. Other options +// +// ################################################################################## + +// For RP2040 processor and SPI displays, uncomment the following line to use the PIO interface. +//#define RP2040_PIO_SPI // Leave commented out to use standard RP2040 SPI port interface + +// For RP2040 processor and 8 or 16-bit parallel displays: +// The parallel interface write cycle period is derived from a division of the CPU clock +// speed so scales with the processor clock. This means that the divider ratio may need +// to be increased when overclocking. It may also need to be adjusted dependant on the +// display controller type (ILI94341, HX8357C etc.). If RP2040_PIO_CLK_DIV is not defined +// the library will set default values which may not suit your display. +// The display controller data sheet will specify the minimum write cycle period. The +// controllers often work reliably for shorter periods, however if the period is too short +// the display may not initialise or graphics will become corrupted. +// PIO write cycle frequency = (CPU clock/(4 * RP2040_PIO_CLK_DIV)) +//#define RP2040_PIO_CLK_DIV 1 // 32ns write cycle at 125MHz CPU clock +//#define RP2040_PIO_CLK_DIV 2 // 64ns write cycle at 125MHz CPU clock +//#define RP2040_PIO_CLK_DIV 3 // 96ns write cycle at 125MHz CPU clock + +// For the RP2040 processor define the SPI port channel used (default 0 if undefined) +//#define TFT_SPI_PORT 1 // Set to 0 if SPI0 pins are used, or 1 if spi1 pins used + +// For the STM32 processor define the SPI port channel used (default 1 if undefined) +//#define TFT_SPI_PORT 2 // Set to 1 for SPI port 1, or 2 for SPI port 2 + +// Define the SPI clock frequency, this affects the graphics rendering speed. Too +// fast and the TFT driver will not keep up and display corruption appears. +// With an ILI9341 display 40MHz works OK, 80MHz sometimes fails +// With a ST7735 display more than 27MHz may not work (spurious pixels and lines) +// With an ILI9163 display 27 MHz works OK. + +// #define SPI_FREQUENCY 1000000 +// #define SPI_FREQUENCY 5000000 +// #define SPI_FREQUENCY 10000000 +// #define SPI_FREQUENCY 20000000 +#define SPI_FREQUENCY 27000000 +// #define SPI_FREQUENCY 40000000 +// #define SPI_FREQUENCY 55000000 // STM32 SPI1 only (SPI2 maximum is 27MHz) +// #define SPI_FREQUENCY 80000000 + +// Optional reduced SPI frequency for reading TFT +#define SPI_READ_FREQUENCY 20000000 + +// The XPT2046 requires a lower SPI clock rate of 2.5MHz so we define that here: +#define SPI_TOUCH_FREQUENCY 2500000 + +// The ESP32 has 2 free SPI ports i.e. VSPI and HSPI, the VSPI is the default. +// If the VSPI port is in use and pins are not accessible (e.g. TTGO T-Beam) +// then uncomment the following line: +//#define USE_HSPI_PORT + +// Comment out the following #define if "SPI Transactions" do not need to be +// supported. When commented out the code size will be smaller and sketches will +// run slightly faster, so leave it commented out unless you need it! + +// Transaction support is needed to work with SD library but not needed with TFT_SdFat +// Transaction support is required if other SPI devices are connected. + +// Transactions are automatically enabled by the library for an ESP32 (to use HAL mutex) +// so changing it here has no effect + +// #define SUPPORT_TRANSACTIONS