mirror of
https://github.com/jomjol/AI-on-the-edge-device.git
synced 2025-12-08 12:36:52 +03:00
Compare commits
546 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f542d842cf | ||
|
|
5bbc2f3da5 | ||
|
|
2831478e02 | ||
|
|
1e0cdfaba1 | ||
|
|
bfebcd5d15 | ||
|
|
cf96d49bd0 | ||
|
|
94a53b38b8 | ||
|
|
00ac2130c2 | ||
|
|
cd1165e547 | ||
|
|
c587ca3224 | ||
|
|
76f45a5927 | ||
|
|
063c4827d0 | ||
|
|
c6b8823417 | ||
|
|
bb5e693077 | ||
|
|
424df641cc | ||
|
|
8ddbda16bf | ||
|
|
8f5bf209d9 | ||
|
|
ec639d4236 | ||
|
|
5304981733 | ||
|
|
7a9955477a | ||
|
|
04fe25e875 | ||
|
|
02dcd584bf | ||
|
|
68a262ef22 | ||
|
|
f2b6b4f819 | ||
|
|
6b672ff9a5 | ||
|
|
26770d877e | ||
|
|
e60c12b25d | ||
|
|
b10336b59c | ||
|
|
dadb004e85 | ||
|
|
e2f3e3d05b | ||
|
|
8c5a6528d9 | ||
|
|
69024adac7 | ||
|
|
9f1b7c9ef7 | ||
|
|
15fd7a6a33 | ||
|
|
9c3fbb4aff | ||
|
|
eb4c9276d5 | ||
|
|
842229ea98 | ||
|
|
b83717aece | ||
|
|
b01e42b893 | ||
|
|
902f1bc2a8 | ||
|
|
5c51990b40 | ||
|
|
55cbbf02cb | ||
|
|
e37817e3e2 | ||
|
|
a78ca53d60 | ||
|
|
20a0d8530a | ||
|
|
af1e3257be | ||
|
|
93c21d30a1 | ||
|
|
7111a7fdcd | ||
|
|
bd541ede60 | ||
|
|
52aab2f8a6 | ||
|
|
a1aee5b346 | ||
|
|
71f31dc841 | ||
|
|
f6a363a871 | ||
|
|
7de18753d9 | ||
|
|
fea0c1b859 | ||
|
|
603fcfef33 | ||
|
|
be3312e912 | ||
|
|
ff657ebb5c | ||
|
|
42d4916cb8 | ||
|
|
a73cd97629 | ||
|
|
636d816727 | ||
|
|
c6f6341c87 | ||
|
|
ad886898a8 | ||
|
|
f16cf1406c | ||
|
|
5dfd9a7db0 | ||
|
|
89dc941d40 | ||
|
|
3f3c845e9f | ||
|
|
1e60d839b7 | ||
|
|
6e474441f6 | ||
|
|
bf2a2f553f | ||
|
|
21ab5cb203 | ||
|
|
1af1796ee0 | ||
|
|
095cf984c4 | ||
|
|
b72d809e59 | ||
|
|
c6a593ba0d | ||
|
|
c41a0a476d | ||
|
|
c68463359b | ||
|
|
a1c2145e77 | ||
|
|
2986c6122d | ||
|
|
a348a51f14 | ||
|
|
6da894142e | ||
|
|
4846a52d45 | ||
|
|
4d74d0c522 | ||
|
|
53e818186a | ||
|
|
26ca15e18a | ||
|
|
f4dccc1d52 | ||
|
|
a07c2cf46d | ||
|
|
2f39d9bdd4 | ||
|
|
7836323fbc | ||
|
|
708fd68cf5 | ||
|
|
1b5ff2ef1a | ||
|
|
1701aaed42 | ||
|
|
c07e5a740a | ||
|
|
3868cf98f6 | ||
|
|
179005f4ce | ||
|
|
50ada0a5a8 | ||
|
|
be7146c886 | ||
|
|
8f89a396f8 | ||
|
|
9e84d28ee9 | ||
|
|
3f4aaf303f | ||
|
|
08baf1824c | ||
|
|
2c7740ec16 | ||
|
|
3ee32d8cf0 | ||
|
|
25ae0465d8 | ||
|
|
14e9d6a9cc | ||
|
|
94c2de1c2a | ||
|
|
541e9841c2 | ||
|
|
fc3c70c63a | ||
|
|
bfde8316ce | ||
|
|
fe0fec7fac | ||
|
|
56abefe3df | ||
|
|
43f15fcba3 | ||
|
|
dadf18f5c9 | ||
|
|
c448ece680 | ||
|
|
07d6eca456 | ||
|
|
4083d35b61 | ||
|
|
2319f8d302 | ||
|
|
224e4380a8 | ||
|
|
14d0c88a28 | ||
|
|
3c9b2c46c8 | ||
|
|
0752694600 | ||
|
|
cf0ed268dd | ||
|
|
f1979a142e | ||
|
|
2a4f0f4a2d | ||
|
|
45269bf45b | ||
|
|
57db60fee4 | ||
|
|
e1c49f39f0 | ||
|
|
7bebae3776 | ||
|
|
0a4560ea95 | ||
|
|
b44db21714 | ||
|
|
34796ed091 | ||
|
|
a46dfd1c23 | ||
|
|
32eb583036 | ||
|
|
6ee83b8413 | ||
|
|
f034232f36 | ||
|
|
b80e43dfe9 | ||
|
|
40c7c253ea | ||
|
|
d11b312a96 | ||
|
|
2c1e531ed2 | ||
|
|
c352e539bc | ||
|
|
a0333d906f | ||
|
|
ec00e943da | ||
|
|
6175471a00 | ||
|
|
948e76876f | ||
|
|
f4edb206d8 | ||
|
|
a9db769595 | ||
|
|
f0497eb924 | ||
|
|
e53af6de0d | ||
|
|
3b7fe93d02 | ||
|
|
2f1d7e577c | ||
|
|
2a7247abfe | ||
|
|
37ba85717f | ||
|
|
97656114b6 | ||
|
|
6cf1d5ad98 | ||
|
|
fed729bcee | ||
|
|
89c36374b3 | ||
|
|
bb69929247 | ||
|
|
e1ea09c501 | ||
|
|
39a827258d | ||
|
|
1d8c6fa257 | ||
|
|
a9aadbdb06 | ||
|
|
a5927f98d2 | ||
|
|
c708e2374a | ||
|
|
4c3dcd8c29 | ||
|
|
be93567956 | ||
|
|
d4406f47ea | ||
|
|
52efedcfa0 | ||
|
|
28d93253f0 | ||
|
|
c7fdc46df2 | ||
|
|
74491e9bde | ||
|
|
a092142c65 | ||
|
|
f6a3dc5851 | ||
|
|
d027adf006 | ||
|
|
bd5be5c5ec | ||
|
|
ca01f5a38f | ||
|
|
3b3a3ebcc9 | ||
|
|
7d62cf67fe | ||
|
|
f6bdd48bca | ||
|
|
5496573369 | ||
|
|
4522ba087f | ||
|
|
d370ba5fe6 | ||
|
|
f39dacc1c5 | ||
|
|
aad1a0e78d | ||
|
|
276efef783 | ||
|
|
79476a8458 | ||
|
|
2b7da5b44e | ||
|
|
ebcec97d1d | ||
|
|
85375b6505 | ||
|
|
085c47b651 | ||
|
|
20a04b888f | ||
|
|
098b1bd025 | ||
|
|
d1c815ce69 | ||
|
|
67c3020d7d | ||
|
|
61bca4ebb8 | ||
|
|
3219202c53 | ||
|
|
cd29690b96 | ||
|
|
3a34564ee2 | ||
|
|
174743ae7f | ||
|
|
e5eca6a53f | ||
|
|
d8e37dce48 | ||
|
|
822753bb4f | ||
|
|
7225792b4b | ||
|
|
b4f6b1a4fb | ||
|
|
21ec58daa0 | ||
|
|
2c69e90fac | ||
|
|
5c57522b71 | ||
|
|
f8eb4db171 | ||
|
|
c9a3df4eec | ||
|
|
acf669900f | ||
|
|
cb3f082218 | ||
|
|
20980b2bcd | ||
|
|
773d21a875 | ||
|
|
3a9c9ac2a4 | ||
|
|
c7e340d3a5 | ||
|
|
0441753a33 | ||
|
|
fb6cb44728 | ||
|
|
ae69942dd9 | ||
|
|
65437727f7 | ||
|
|
ccefe57795 | ||
|
|
e3ff049720 | ||
|
|
39e84baf8b | ||
|
|
eb7d078a1a | ||
|
|
b6c6805a08 | ||
|
|
d567a5d7f2 | ||
|
|
6922970185 | ||
|
|
9a4b51d6de | ||
|
|
2546ab81ff | ||
|
|
7b7544079f | ||
|
|
17fe87b349 | ||
|
|
f99dc8fdfc | ||
|
|
cce992754c | ||
|
|
eefccf6e11 | ||
|
|
64bb4f0ff6 | ||
|
|
f534741205 | ||
|
|
ee38bc7dc6 | ||
|
|
132834c3fd | ||
|
|
d3d9c64f3b | ||
|
|
aa2a4edf7e | ||
|
|
8012b7f43e | ||
|
|
009ab4c896 | ||
|
|
c54ca18e4e | ||
|
|
beb09593eb | ||
|
|
1a76ae121c | ||
|
|
1300242d4a | ||
|
|
79543df23b | ||
|
|
4049d752ba | ||
|
|
dc90972659 | ||
|
|
32282ecfe2 | ||
|
|
ae6a94544b | ||
|
|
70b031eacc | ||
|
|
c3fadf5c2a | ||
|
|
7e5f6bf4a5 | ||
|
|
88b531ae8b | ||
|
|
8481cc4b26 | ||
|
|
ecaed38c1d | ||
|
|
d6a1838d47 | ||
|
|
5194c466be | ||
|
|
043de9265a | ||
|
|
08a350172d | ||
|
|
7e806df64d | ||
|
|
dccfb5e91e | ||
|
|
fbe4609bb9 | ||
|
|
3e85cfb456 | ||
|
|
5dff4ca8cf | ||
|
|
0d0b0187f4 | ||
|
|
4cf9ea6c45 | ||
|
|
1b76e0f449 | ||
|
|
d4a0ad20ff | ||
|
|
df72445e79 | ||
|
|
456cb93809 | ||
|
|
d968a7adc6 | ||
|
|
9a52b8b2f3 | ||
|
|
8caa852bbf | ||
|
|
4faca4c46c | ||
|
|
238fc5fae3 | ||
|
|
c32ca5a23c | ||
|
|
827d9d1700 | ||
|
|
0ea4b3b3ce | ||
|
|
3e26c6c743 | ||
|
|
fa5c99b3cb | ||
|
|
6feae4e239 | ||
|
|
ef64be3888 | ||
|
|
cc89d625f2 | ||
|
|
08ba754b88 | ||
|
|
6b38e44d7f | ||
|
|
141aea7fa7 | ||
|
|
bcd07761b6 | ||
|
|
fe4d861e15 | ||
|
|
71322c9fbe | ||
|
|
f8b4881a50 | ||
|
|
438d5696e4 | ||
|
|
0d391c8780 | ||
|
|
59de6319a1 | ||
|
|
3805687219 | ||
|
|
c6a789dc45 | ||
|
|
246f9cfc31 | ||
|
|
b2d8c60bb1 | ||
|
|
1d573cd18a | ||
|
|
b2e5cdd8a3 | ||
|
|
f7fde7c430 | ||
|
|
2c19080a66 | ||
|
|
35663c5fd4 | ||
|
|
5b449d5c45 | ||
|
|
3a5f3496d5 | ||
|
|
00434d01c3 | ||
|
|
3e0bb81e32 | ||
|
|
4f57f9eafd | ||
|
|
f24ec581e6 | ||
|
|
0d78bb78ea | ||
|
|
47aea007b3 | ||
|
|
284b3f428e | ||
|
|
92d45c7971 | ||
|
|
ff1d9d3b4f | ||
|
|
2b57dd0853 | ||
|
|
dd8f5eea22 | ||
|
|
60c5305378 | ||
|
|
8b1c65a38a | ||
|
|
4f5933c4f2 | ||
|
|
06c9bfb0de | ||
|
|
69f1a99b55 | ||
|
|
a8fb88a35d | ||
|
|
d418d22155 | ||
|
|
53e8cf49f9 | ||
|
|
ac4f823cbf | ||
|
|
444dc0fa39 | ||
|
|
5053a31245 | ||
|
|
44cf8933d4 | ||
|
|
35de56be04 | ||
|
|
80a6fc1dc3 | ||
|
|
d2c47fcde2 | ||
|
|
7b3a493587 | ||
|
|
217f543578 | ||
|
|
797fc5e764 | ||
|
|
7a4e82a44e | ||
|
|
019069cd16 | ||
|
|
4de1152cf3 | ||
|
|
0e0fb459dc | ||
|
|
8410df6144 | ||
|
|
4990858101 | ||
|
|
fc1f8ee242 | ||
|
|
252c399a76 | ||
|
|
eb7f2b3705 | ||
|
|
2ed6fb0f0d | ||
|
|
b5213b01af | ||
|
|
473e458b85 | ||
|
|
4f3f3d9af2 | ||
|
|
b5a4cfed96 | ||
|
|
6fca4d8d95 | ||
|
|
a11e19fb0c | ||
|
|
09fa94c95f | ||
|
|
f01b4dbd13 | ||
|
|
dbf1770016 | ||
|
|
006f3aa063 | ||
|
|
24c46c38b4 | ||
|
|
9ced147d9c | ||
|
|
0808895bd6 | ||
|
|
d2dec9fa59 | ||
|
|
7e7bc3dd68 | ||
|
|
5b09cd0d59 | ||
|
|
74d4f20858 | ||
|
|
e790a14caa | ||
|
|
f023a6b739 | ||
|
|
1e463188ea | ||
|
|
9991196961 | ||
|
|
ecece0f7fc | ||
|
|
1b3b7595c1 | ||
|
|
6d10f712d1 | ||
|
|
b84a2db050 | ||
|
|
374462a7d8 | ||
|
|
4facd7be05 | ||
|
|
3baf5865ad | ||
|
|
02138c44ac | ||
|
|
1094c8a0a8 | ||
|
|
75b15b8e9d | ||
|
|
36c12b400b | ||
|
|
35d90cd0ee | ||
|
|
999a8d9374 | ||
|
|
8a4269c6a0 | ||
|
|
8f5579cca5 | ||
|
|
e58b3a2cf8 | ||
|
|
222ee0921c | ||
|
|
e2bfcd26c9 | ||
|
|
18917b2d82 | ||
|
|
f0dea3abcb | ||
|
|
bd2d8b4a15 | ||
|
|
80e3f50a5b | ||
|
|
f6ca32d69f | ||
|
|
49e919c481 | ||
|
|
de768c4f44 | ||
|
|
002fc033aa | ||
|
|
f4af8de699 | ||
|
|
e4d6707a0b | ||
|
|
9f03e68690 | ||
|
|
b1df7df580 | ||
|
|
ebc9be4a28 | ||
|
|
5d0fc73c13 | ||
|
|
40a1aa0430 | ||
|
|
d7a733512f | ||
|
|
dc9f1aad27 | ||
|
|
cd3e641bcc | ||
|
|
ad72ffa37c | ||
|
|
2d45a0ed26 | ||
|
|
e8065ef414 | ||
|
|
33893eb566 | ||
|
|
3fbff0ad33 | ||
|
|
28cabea7f2 | ||
|
|
5d06130a88 | ||
|
|
3497a86084 | ||
|
|
84707fb27a | ||
|
|
0b81af7cb8 | ||
|
|
61efe1a6ef | ||
|
|
09ecd722cc | ||
|
|
5615fd8137 | ||
|
|
7ebf68411f | ||
|
|
34835dca84 | ||
|
|
697ff4c4b6 | ||
|
|
0f7c685933 | ||
|
|
3ace5aeff1 | ||
|
|
d47ed060b0 | ||
|
|
e7dbebffa1 | ||
|
|
def13d46af | ||
|
|
f82a6bf513 | ||
|
|
05deecee00 | ||
|
|
8339b6788f | ||
|
|
4e5b084932 | ||
|
|
62fcefee78 | ||
|
|
14128ca3a7 | ||
|
|
431551fb45 | ||
|
|
ff52d785cd | ||
|
|
f5a7c082d6 | ||
|
|
f3e90c6293 | ||
|
|
cbd14a267f | ||
|
|
ba7d6b3621 | ||
|
|
d4a492463b | ||
|
|
46589288e7 | ||
|
|
81a57d32a8 | ||
|
|
17994494c9 | ||
|
|
c581d64a26 | ||
|
|
70c2b88f8a | ||
|
|
13d01a6c96 | ||
|
|
331e3533cc | ||
|
|
5a00bdd7f6 | ||
|
|
db7ffb30c4 | ||
|
|
9a07f271ff | ||
|
|
6a45ab7693 | ||
|
|
9fc876853b | ||
|
|
c30881f73c | ||
|
|
e75e720869 | ||
|
|
43aae5a3cd | ||
|
|
54cd7e511a | ||
|
|
c5a82e839f | ||
|
|
9f97a2b223 | ||
|
|
8d06de5792 | ||
|
|
1326585a33 | ||
|
|
d7a507ca05 | ||
|
|
b24b7d5ce2 | ||
|
|
fc719da0ae | ||
|
|
9b1a83c8b4 | ||
|
|
6defcf8d4c | ||
|
|
c2a55e7c86 | ||
|
|
a636ce3679 | ||
|
|
a20fec1094 | ||
|
|
929796c87f | ||
|
|
e40ceb54ce | ||
|
|
eebdd7c6eb | ||
|
|
2a7f3b33a3 | ||
|
|
262d83ee6f | ||
|
|
de92c29245 | ||
|
|
19158c998f | ||
|
|
17ffd28c05 | ||
|
|
9ca02e0a7f | ||
|
|
7488d7bf23 | ||
|
|
e7bfba4b01 | ||
|
|
63ac38a52d | ||
|
|
e2cf8337d4 | ||
|
|
e995d6c498 | ||
|
|
0e3a50d0c1 | ||
|
|
df12deae00 | ||
|
|
b6bfeea936 | ||
|
|
f79e03faa2 | ||
|
|
de1dcc4d06 | ||
|
|
33bfef0af4 | ||
|
|
727b871fc5 | ||
|
|
03c84a1ff3 | ||
|
|
db36fe2522 | ||
|
|
9ffaf6e3f8 | ||
|
|
fa09680711 | ||
|
|
e2b66aa73a | ||
|
|
c4b990ada0 | ||
|
|
267782d083 | ||
|
|
e4a6fd33fe | ||
|
|
5db20d3687 | ||
|
|
58185a0569 | ||
|
|
b5e0d6ee66 | ||
|
|
d93c5daf14 | ||
|
|
6ff83445ac | ||
|
|
d944e8676b | ||
|
|
a8d7b29507 | ||
|
|
ab0fc72257 | ||
|
|
933215c116 | ||
|
|
e81a7eebe8 | ||
|
|
7d33c3ee5f | ||
|
|
863856ca5d | ||
|
|
eefc41d6ff | ||
|
|
d1807a1b3d | ||
|
|
dfc45772b7 | ||
|
|
4dd41c486f | ||
|
|
ff81fcbd7f | ||
|
|
5e5d2e2f72 | ||
|
|
53cee961f4 | ||
|
|
c81f901a19 | ||
|
|
512d7f95b4 | ||
|
|
6642f6f995 | ||
|
|
431df73e52 | ||
|
|
1f5d4de5f3 | ||
|
|
ce00192684 | ||
|
|
28f3ad0242 | ||
|
|
2dfd55e1c3 | ||
|
|
f84f20b2e8 | ||
|
|
806adcb4d0 | ||
|
|
4dc4752823 | ||
|
|
827423023c | ||
|
|
6ca7897fa0 | ||
|
|
39b8b5b07c | ||
|
|
5b98acaa32 | ||
|
|
d3d241c7b9 | ||
|
|
8c7e86fea4 | ||
|
|
98d85b2c6c | ||
|
|
a7dc37761b | ||
|
|
2dd2d03f6c | ||
|
|
18e96d62a6 | ||
|
|
a1a77ae5d9 | ||
|
|
c5b20f3680 | ||
|
|
add6cf5c33 | ||
|
|
493bd4df2f | ||
|
|
53ff190860 | ||
|
|
9a9aa68a65 | ||
|
|
a92cb69067 | ||
|
|
90fa44380c | ||
|
|
7a9f61a8d8 | ||
|
|
a8f8189543 | ||
|
|
2c4bda9e66 | ||
|
|
7b2a80a13d | ||
|
|
95d312b920 | ||
|
|
886cd4ffa5 | ||
|
|
b0de37b762 |
@@ -4,6 +4,31 @@
|
|||||||
# Due to the way it works, you have to add each response twice, once for the issue, once for the discussions!
|
# Due to the way it works, you have to add each response twice, once for the issue, once for the discussions!
|
||||||
|
|
||||||
labels:
|
labels:
|
||||||
|
#######################################################################
|
||||||
|
# Bot Response: Documentation
|
||||||
|
#######################################################################
|
||||||
|
- name: bot-reply Documentation
|
||||||
|
labeled:
|
||||||
|
issue:
|
||||||
|
body: |
|
||||||
|
Please have a look on our documentation: https://jomjol.github.io/AI-on-the-edge-device-docs
|
||||||
|
discussion:
|
||||||
|
body: |
|
||||||
|
Please have a look on our documentation: https://jomjol.github.io/AI-on-the-edge-device-docs
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
# Bot Response: ROI setup
|
||||||
|
#######################################################################
|
||||||
|
- name: bot-reply ROI Setup
|
||||||
|
labeled:
|
||||||
|
issue:
|
||||||
|
body: |
|
||||||
|
Make sure to setup your ROIs properly. Have a look on our documentation: https://jomjol.github.io/AI-on-the-edge-device-docs/ROI-Configuration/#how-to-setup-the-digit-rois-perfectly
|
||||||
|
discussion:
|
||||||
|
body: |
|
||||||
|
Make sure to setup your ROIs properly. Have a look on our documentation: https://jomjol.github.io/AI-on-the-edge-device-docs/ROI-Configuration/#how-to-setup-the-digit-rois-perfectly
|
||||||
|
|
||||||
|
|
||||||
#######################################################################
|
#######################################################################
|
||||||
# Bot Response: Logfile
|
# Bot Response: Logfile
|
||||||
#######################################################################
|
#######################################################################
|
||||||
@@ -90,9 +115,9 @@
|
|||||||
labeled:
|
labeled:
|
||||||
issue:
|
issue:
|
||||||
body: |
|
body: |
|
||||||
See [Digital Digits](https://jomjol.github.io/neural-network-digital-counter-readout) resp. [Analogue Pointers](https://jomjol.github.io/neural-network-analog-needle-readout) for an overview of all trained data.
|
See [Digits](https://jomjol.github.io/neural-network-digital-counter-readout) resp. [Analogue Pointers](https://jomjol.github.io/neural-network-analog-needle-readout) for an overview of all trained data.
|
||||||
If your type is not contained it can be added to our training material, see [here](https://jomjol.github.io/AI-on-the-edge-device-docs/collect-new-images/).
|
If your type is not contained it can be added to our training material, see [here](https://jomjol.github.io/AI-on-the-edge-device-docs/collect-new-images/).
|
||||||
discussion:
|
discussion:
|
||||||
body: |
|
body: |
|
||||||
See [Digital Digits](https://jomjol.github.io/neural-network-digital-counter-readout) resp. [Analogue Pointers](https://jomjol.github.io/neural-network-analog-needle-readout) for an overview of all trained data.
|
See [Digits](https://jomjol.github.io/neural-network-digital-counter-readout) resp. [Analogue Pointers](https://jomjol.github.io/neural-network-analog-needle-readout) for an overview of all trained data.
|
||||||
If your type is not contained it can be added to our training material, see [here](https://jomjol.github.io/AI-on-the-edge-device-docs/collect-new-images/).
|
If your type is not contained it can be added to our training material, see [here](https://jomjol.github.io/AI-on-the-edge-device-docs/collect-new-images/).
|
||||||
137
.github/workflows/build.yaml
vendored
137
.github/workflows/build.yaml
vendored
@@ -15,7 +15,7 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
concurrent_skipping: same_content_newer
|
concurrent_skipping: same_content_newer
|
||||||
|
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
submodules: recursive
|
submodules: recursive
|
||||||
|
|
||||||
@@ -25,39 +25,40 @@ jobs:
|
|||||||
echo "sha_short=$(git rev-parse --short HEAD)" >> $GITHUB_OUTPUT
|
echo "sha_short=$(git rev-parse --short HEAD)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Update PIP cache on every commit
|
- name: Update PIP cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: ~/.cache/pip
|
path: ~/.cache/pip
|
||||||
key: pip-${{ github.run_id }}
|
key: pip-${{ github.run_id }}
|
||||||
restore-keys: pip # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
restore-keys: pip # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
||||||
|
|
||||||
- name: Update PlatformIO cache on every commit
|
- name: Update PlatformIO cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: ~/.platformio
|
path: ~/.platformio
|
||||||
key: platformio-${{ github.run_id }}
|
key: platformio-${{ github.run_id }}
|
||||||
restore-keys: platformio # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
restore-keys: platformio # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
||||||
|
|
||||||
- name: Update Build cache on every commit
|
- name: Update Build cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: ./code/.pio/
|
path: ./code/.pio/
|
||||||
key: build-${{ github.run_id }}
|
key: build-${{ github.run_id }}
|
||||||
restore-keys: build # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
restore-keys: build # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
||||||
|
|
||||||
- name: Update generated-files cache on every commit
|
- name: Update generated-files cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: |
|
path: |
|
||||||
./code/.pio/build/esp32cam/firmware.bin
|
./code/.pio/build/esp32cam/firmware.bin
|
||||||
./code/.pio/build/esp32cam/partitions.bin
|
./code/.pio/build/esp32cam/partitions.bin
|
||||||
./code/.pio/build/esp32cam/bootloader.bin
|
./code/.pio/build/esp32cam/bootloader.bin
|
||||||
./html/*
|
./html/*
|
||||||
|
./demo/*
|
||||||
key: generated-files-${{ github.run_id }}
|
key: generated-files-${{ github.run_id }}
|
||||||
restore-keys: generated-files # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
restore-keys: generated-files # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
||||||
|
|
||||||
- name: Set up Python
|
- name: Set up Python
|
||||||
uses: actions/setup-python@v4
|
uses: actions/setup-python@v5
|
||||||
with:
|
with:
|
||||||
python-version: '3.10'
|
python-version: '3.10'
|
||||||
|
|
||||||
@@ -70,14 +71,38 @@ jobs:
|
|||||||
#run: echo "Testing... ${{ github.ref_name }}, ${{ steps.vars.outputs.sha_short }}" > ./sd-card/html/version.txt; mkdir -p ./code/.pio/build/esp32cam/; cd ./code/.pio/build/esp32cam/; echo "${{ steps.vars.outputs.sha_short }}" > firmware.bin; cp firmware.bin partitions.bin; cp firmware.bin bootloader.bin # Testing
|
#run: echo "Testing... ${{ github.ref_name }}, ${{ steps.vars.outputs.sha_short }}" > ./sd-card/html/version.txt; mkdir -p ./code/.pio/build/esp32cam/; cd ./code/.pio/build/esp32cam/; echo "${{ steps.vars.outputs.sha_short }}" > firmware.bin; cp firmware.bin partitions.bin; cp firmware.bin bootloader.bin # Testing
|
||||||
run: cd code; platformio run --environment esp32cam
|
run: cd code; platformio run --environment esp32cam
|
||||||
|
|
||||||
- name: Prepare Web UI (copy data from repo and update hashes in all files)
|
- name: Prepare Web UI (generate tooltip pages and update hashes in all files)
|
||||||
run: |
|
run: |
|
||||||
rm -rf ./html
|
rm -rf ./html
|
||||||
mkdir html
|
mkdir html
|
||||||
cp ./sd-card/html/* ./html/
|
cp -r ./sd-card/html/* ./html/
|
||||||
cd html; find . -type f -exec sed -i 's/$COMMIT_HASH/${{ steps.vars.outputs.sha_short }}/g' {} \;
|
|
||||||
|
|
||||||
|
python -m pip install markdown
|
||||||
|
mkdir html/param-tooltips
|
||||||
|
cd tools/parameter-tooltip-generator
|
||||||
|
python generate-param-doc-tooltips.py
|
||||||
|
cd ../..
|
||||||
|
|
||||||
|
cp -r ./sd-card/html/* ./html/
|
||||||
|
rm -f ./html/edit_config_template.html # Remove the config page template, it is no longer needed
|
||||||
|
|
||||||
|
echo "Replacing variables..."
|
||||||
|
cd html
|
||||||
|
find . -type f -exec sed -i 's/$COMMIT_HASH/${{ steps.vars.outputs.sha_short }}/g' {} \;
|
||||||
|
echo "compressing all html files..."
|
||||||
|
find . -name "*.html" -type f -exec gzip {} \;
|
||||||
|
find . -name "*.css" -type f -exec gzip {} \;
|
||||||
|
find . -name "*.js" -type f -exec gzip {} \;
|
||||||
|
find . -name "*.jpg" -type f -exec gzip {} \;
|
||||||
|
find . -name "*.png" -type f -exec gzip {} \;
|
||||||
|
find . -name "*.svg" -type f -exec gzip {} \;
|
||||||
|
find . -name "*.map" -type f -exec gzip {} \;
|
||||||
|
|
||||||
|
- name: Prepare Demo mode files
|
||||||
|
run: |
|
||||||
|
rm -rf ./demo
|
||||||
|
mkdir demo
|
||||||
|
cp -r ./sd-card/demo/* ./demo/
|
||||||
|
|
||||||
#########################################################################################
|
#########################################################################################
|
||||||
## Pack for Update
|
## Pack for Update
|
||||||
@@ -86,27 +111,29 @@ jobs:
|
|||||||
# New OTA concept
|
# New OTA concept
|
||||||
# update__version.zip file with following content:
|
# update__version.zip file with following content:
|
||||||
# - /firmware.bin
|
# - /firmware.bin
|
||||||
# - (optional) /html/*
|
# - (optional) /html/* (inkl. subfolders)
|
||||||
# - (optional) /config/*.tfl
|
# - (optional) /config/*.tfl
|
||||||
|
# - (optional) /demo/*
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: build
|
needs: build
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Update generated-files cache on every commit
|
- name: Update generated-files cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: |
|
path: |
|
||||||
./code/.pio/build/esp32cam/firmware.bin
|
./code/.pio/build/esp32cam/firmware.bin
|
||||||
./code/.pio/build/esp32cam/partitions.bin
|
./code/.pio/build/esp32cam/partitions.bin
|
||||||
./code/.pio/build/esp32cam/bootloader.bin
|
./code/.pio/build/esp32cam/bootloader.bin
|
||||||
./html/*
|
./html/*
|
||||||
|
./demo/*
|
||||||
key: generated-files-${{ github.run_id }}
|
key: generated-files-${{ github.run_id }}
|
||||||
restore-keys: generated-files # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
restore-keys: generated-files # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
||||||
|
|
||||||
- name: Update update cache on every commit
|
- name: Update update cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: update
|
path: update
|
||||||
key: update-${{ github.run_id }}
|
key: update-${{ github.run_id }}
|
||||||
@@ -127,6 +154,9 @@ jobs:
|
|||||||
- name: Add Web UI to update
|
- name: Add Web UI to update
|
||||||
run: cp -r ./html ./update/
|
run: cp -r ./html ./update/
|
||||||
|
|
||||||
|
- name: Add Demo mode files to update
|
||||||
|
run: cp -r ./demo ./update/
|
||||||
|
|
||||||
- name: Add CNN to update
|
- name: Add CNN to update
|
||||||
run: |
|
run: |
|
||||||
rm -rf ./update/config/
|
rm -rf ./update/config/
|
||||||
@@ -135,13 +165,12 @@ jobs:
|
|||||||
cp ./sd-card/config/*.tflite ./update/config/ 2>/dev/null || true
|
cp ./sd-card/config/*.tflite ./update/config/ 2>/dev/null || true
|
||||||
|
|
||||||
- name: Upload update as update.zip artifact (Firmware + Web UI + CNN)
|
- name: Upload update as update.zip artifact (Firmware + Web UI + CNN)
|
||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: "AI-on-the-edge-device__update__${{ steps.vars.outputs.branch }}_(${{ steps.vars.outputs.sha_short }})"
|
name: "AI-on-the-edge-device__update__${{ steps.vars.outputs.branch }}_(${{ steps.vars.outputs.sha_short }})"
|
||||||
path: ./update/*
|
path: ./update/*
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#########################################################################################
|
#########################################################################################
|
||||||
## Pack for Remote Setup
|
## Pack for Remote Setup
|
||||||
#########################################################################################
|
#########################################################################################
|
||||||
@@ -149,27 +178,29 @@ jobs:
|
|||||||
# New Remote Setup concept
|
# New Remote Setup concept
|
||||||
# remote_setup__version.zip file with following content:
|
# remote_setup__version.zip file with following content:
|
||||||
# - /firmware.bin
|
# - /firmware.bin
|
||||||
# - /html/*
|
# - /html/* (inkl. subfolders)
|
||||||
|
# - /demo/*
|
||||||
# - /config/*
|
# - /config/*
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: build
|
needs: build
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Update generated-files cache on every commit
|
- name: Update generated-files cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: |
|
path: |
|
||||||
./code/.pio/build/esp32cam/firmware.bin
|
./code/.pio/build/esp32cam/firmware.bin
|
||||||
./code/.pio/build/esp32cam/partitions.bin
|
./code/.pio/build/esp32cam/partitions.bin
|
||||||
./code/.pio/build/esp32cam/bootloader.bin
|
./code/.pio/build/esp32cam/bootloader.bin
|
||||||
./html/*
|
./html/*
|
||||||
|
./demo/*
|
||||||
key: generated-files-${{ github.run_id }}
|
key: generated-files-${{ github.run_id }}
|
||||||
restore-keys: generated-files # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
restore-keys: generated-files # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
||||||
|
|
||||||
- name: Update remote_setup cache on every commit
|
- name: Update remote_setup cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: remote_setup
|
path: remote_setup
|
||||||
key: remote_setup-${{ github.run_id }}
|
key: remote_setup-${{ github.run_id }}
|
||||||
@@ -189,6 +220,9 @@ jobs:
|
|||||||
- name: Add Web UI to remote_setup
|
- name: Add Web UI to remote_setup
|
||||||
run: cp -r ./html ./remote_setup/
|
run: cp -r ./html ./remote_setup/
|
||||||
|
|
||||||
|
- name: Add Demo mode files to update
|
||||||
|
run: cp -r ./demo ./update/
|
||||||
|
|
||||||
- name: Add whole config folder to remote_setup
|
- name: Add whole config folder to remote_setup
|
||||||
run: |
|
run: |
|
||||||
rm -rf ./remote_setup/config/
|
rm -rf ./remote_setup/config/
|
||||||
@@ -196,7 +230,7 @@ jobs:
|
|||||||
cp ./sd-card/config/* ./remote_setup/config/ 2>/dev/null || true
|
cp ./sd-card/config/* ./remote_setup/config/ 2>/dev/null || true
|
||||||
|
|
||||||
- name: Upload remote_setup as remote_setup.zip artifact (Firmware + Web UI + Config)
|
- name: Upload remote_setup as remote_setup.zip artifact (Firmware + Web UI + Config)
|
||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: "AI-on-the-edge-device__remote-setup__${{ steps.vars.outputs.branch }}_(${{ steps.vars.outputs.sha_short }})"
|
name: "AI-on-the-edge-device__remote-setup__${{ steps.vars.outputs.branch }}_(${{ steps.vars.outputs.sha_short }})"
|
||||||
path: ./remote_setup/*
|
path: ./remote_setup/*
|
||||||
@@ -211,21 +245,22 @@ jobs:
|
|||||||
needs: build
|
needs: build
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Update generated-files cache on every commit
|
- name: Update generated-files cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: |
|
path: |
|
||||||
./code/.pio/build/esp32cam/firmware.bin
|
./code/.pio/build/esp32cam/firmware.bin
|
||||||
./code/.pio/build/esp32cam/partitions.bin
|
./code/.pio/build/esp32cam/partitions.bin
|
||||||
./code/.pio/build/esp32cam/bootloader.bin
|
./code/.pio/build/esp32cam/bootloader.bin
|
||||||
./html/*
|
./html/*
|
||||||
|
./demo/*
|
||||||
key: generated-files-${{ github.run_id }}
|
key: generated-files-${{ github.run_id }}
|
||||||
restore-keys: generated-files # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
restore-keys: generated-files # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
||||||
|
|
||||||
- name: Update manual_setup cache on every commit
|
- name: Update manual_setup cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: manual_setup
|
path: manual_setup
|
||||||
key: manual_setup-${{ github.run_id }}
|
key: manual_setup-${{ github.run_id }}
|
||||||
@@ -249,12 +284,14 @@ jobs:
|
|||||||
cp -f "./code/.pio/build/esp32cam/bootloader.bin" "manual_setup/bootloader.bin"
|
cp -f "./code/.pio/build/esp32cam/bootloader.bin" "manual_setup/bootloader.bin"
|
||||||
cp -f "./code/.pio/build/esp32cam/partitions.bin" "manual_setup/partitions.bin"
|
cp -f "./code/.pio/build/esp32cam/partitions.bin" "manual_setup/partitions.bin"
|
||||||
rm -rf ./sd-card/html
|
rm -rf ./sd-card/html
|
||||||
|
rm -rf ./sd-card/demo
|
||||||
cp -r ./html ./sd-card/ # Overwrite the Web UI with the preprocessed files
|
cp -r ./html ./sd-card/ # Overwrite the Web UI with the preprocessed files
|
||||||
cd sd-card; zip -r ../manual_setup/sd-card.zip *; cd ..
|
cp -r ./demo ./sd-card/
|
||||||
|
cd sd-card; rm -rf html/param-tooltips; zip -r ../manual_setup/sd-card.zip *; cd ..
|
||||||
cd ./manual_setup
|
cd ./manual_setup
|
||||||
|
|
||||||
- name: Upload manual_setup.zip artifact (Firmware + Bootloader + Partitions + Web UI)
|
- name: Upload manual_setup.zip artifact (Firmware + Bootloader + Partitions + Web UI)
|
||||||
uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: "AI-on-the-edge-device__manual-setup__${{ steps.vars.outputs.branch }}_(${{ steps.vars.outputs.sha_short }})"
|
name: "AI-on-the-edge-device__manual-setup__${{ steps.vars.outputs.branch }}_(${{ steps.vars.outputs.sha_short }})"
|
||||||
path: ./manual_setup
|
path: ./manual_setup
|
||||||
@@ -263,7 +300,7 @@ jobs:
|
|||||||
#########################################################################################
|
#########################################################################################
|
||||||
## Prepare and create release
|
## Prepare and create release
|
||||||
#########################################################################################
|
#########################################################################################
|
||||||
release:
|
prepare-release:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs: [pack-for-update, pack-for-manual_setup, pack-for-remote_setup]
|
needs: [pack-for-update, pack-for-manual_setup, pack-for-remote_setup]
|
||||||
if: startsWith(github.ref, 'refs/tags/')
|
if: startsWith(github.ref, 'refs/tags/')
|
||||||
@@ -275,24 +312,24 @@ jobs:
|
|||||||
id-token: write
|
id-token: write
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Update update cache on every commit
|
- name: Update update cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: update
|
path: update
|
||||||
key: update-${{ github.run_id }}
|
key: update-${{ github.run_id }}
|
||||||
restore-keys: update # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
restore-keys: update # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
||||||
|
|
||||||
- name: Update remote_setup cache on every commit
|
- name: Update remote_setup cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: remote_setup
|
path: remote_setup
|
||||||
key: remote_setup-${{ github.run_id }}
|
key: remote_setup-${{ github.run_id }}
|
||||||
restore-keys: remote_setup # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
restore-keys: remote_setup # This matches above key as it is only used as a prefix. it the restores the nearest cache, see https://github.com/restore-keys:/blob/main/tips-and-workarounds.md#update-a-cache
|
||||||
|
|
||||||
- name: Update manual_setup cache on every commit
|
- name: Update manual_setup cache on every commit
|
||||||
uses: actions/cache@v3.2.3
|
uses: actions/cache@v4
|
||||||
with:
|
with:
|
||||||
path: manual_setup
|
path: manual_setup
|
||||||
key: manual_setup-${{ github.run_id }}
|
key: manual_setup-${{ github.run_id }}
|
||||||
@@ -323,13 +360,11 @@ jobs:
|
|||||||
|
|
||||||
# extract the version used in next step
|
# extract the version used in next step
|
||||||
- id: get_version
|
- id: get_version
|
||||||
if: startsWith(github.ref, 'refs/tags/')
|
uses: drewg13/get-version-action@98dda2a47a257e202c2e6c2ed2e6072ec23f448e
|
||||||
uses: Simply007/get-version-action@v2
|
|
||||||
|
|
||||||
# # the changelog [unreleased] will now be changed to the release version
|
# # the changelog [unreleased] will now be changed to the release version
|
||||||
# - name: Update changelog
|
# - name: Update changelog
|
||||||
# uses: thomaseizinger/keep-a-changelog-new-release@v1
|
# uses: thomaseizinger/keep-a-changelog-new-release@v1
|
||||||
# if: startsWith(github.ref, 'refs/tags/')
|
|
||||||
# with:
|
# with:
|
||||||
# changelogPath: Changelog.md
|
# changelogPath: Changelog.md
|
||||||
# version: ${{ steps.get_version.outputs.version-without-v }}
|
# version: ${{ steps.get_version.outputs.version-without-v }}
|
||||||
@@ -337,7 +372,6 @@ jobs:
|
|||||||
# # the release notes will be extracted from changelog
|
# # the release notes will be extracted from changelog
|
||||||
# - name: Extract release notes
|
# - name: Extract release notes
|
||||||
# id: extract-release-notes
|
# id: extract-release-notes
|
||||||
# if: startsWith(github.ref, 'refs/tags/')
|
|
||||||
# uses: ffurrer2/extract-release-notes@v1
|
# uses: ffurrer2/extract-release-notes@v1
|
||||||
# with:
|
# with:
|
||||||
# changelog_file: Changelog.md
|
# changelog_file: Changelog.md
|
||||||
@@ -345,12 +379,11 @@ jobs:
|
|||||||
# Releases should only be created on master by tagging the last commit.
|
# Releases should only be created on master by tagging the last commit.
|
||||||
# all artifacts in firmware folder pushed to the release
|
# all artifacts in firmware folder pushed to the release
|
||||||
- name: Release
|
- name: Release
|
||||||
uses: softprops/action-gh-release@v1
|
uses: softprops/action-gh-release@v2.0.8
|
||||||
# Note:
|
# Note:
|
||||||
# If you get the error "Resource not accessible by integration",
|
# If you get the error "Resource not accessible by integration",
|
||||||
# The access rights are not sufficient, see
|
# The access rights are not sufficient, see
|
||||||
# https://github.com/softprops/action-gh-release/issues/232#issuecomment-1131379440
|
# https://github.com/softprops/action-gh-release/issues/232#issuecomment-1131379440
|
||||||
if: startsWith(github.ref, 'refs/tags/')
|
|
||||||
with:
|
with:
|
||||||
name: ${{ steps.get_version.outputs.version-without-v }}
|
name: ${{ steps.get_version.outputs.version-without-v }}
|
||||||
body: ${{ steps.extract-release-notes.outputs.release_notes }}
|
body: ${{ steps.extract-release-notes.outputs.release_notes }}
|
||||||
@@ -359,7 +392,6 @@ jobs:
|
|||||||
|
|
||||||
# # Commit&Push Changelog to master branch. Must be manually merged back to rolling
|
# # Commit&Push Changelog to master branch. Must be manually merged back to rolling
|
||||||
# - name: Commit changes and push changes
|
# - name: Commit changes and push changes
|
||||||
# if: startsWith(github.ref, 'refs/tags/')
|
|
||||||
# run: |
|
# run: |
|
||||||
# git config user.name github-actions
|
# git config user.name github-actions
|
||||||
# git config user.email github-actions@github.com
|
# git config user.email github-actions@github.com
|
||||||
@@ -371,9 +403,10 @@ jobs:
|
|||||||
#########################################################################################
|
#########################################################################################
|
||||||
## Update the Web Installer on a release
|
## Update the Web Installer on a release
|
||||||
#########################################################################################
|
#########################################################################################
|
||||||
# This is the same as in the update-webinstaller.yml
|
# Make sure to also update update-webinstaller.yml!
|
||||||
update-web-installer:
|
update-web-installer:
|
||||||
needs: [release]
|
if: github.event_name == 'release' && github.event.action == 'published' # Only run on release but not on prerelease
|
||||||
|
needs: [prepare-release]
|
||||||
environment:
|
environment:
|
||||||
name: github-pages
|
name: github-pages
|
||||||
url: ${{ steps.deployment.outputs.page_url }}
|
url: ${{ steps.deployment.outputs.page_url }}
|
||||||
@@ -387,32 +420,36 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Get version of last release
|
- name: Get version of last release
|
||||||
id: last_release
|
id: last_release
|
||||||
uses: mindojo/get-latest-release@0b8ef1434d7468d6bffcc8263baff5c777f72321
|
uses: joutvhu/get-release@v1
|
||||||
with:
|
with:
|
||||||
myToken: ${{ github.token }}
|
latest: true
|
||||||
exclude_types: "draft|prerelease"
|
prerelease: false
|
||||||
view_top: 1
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
- name: Add binary to Web Installer and update manifest
|
- name: Add binary to Web Installer and update manifest
|
||||||
run: |
|
run: |
|
||||||
|
echo "Updating Web installer to use firmware from ${{ steps.last_release.outputs.tag_name }}..."
|
||||||
rm -f docs/binary/firmware.bin
|
rm -f docs/binary/firmware.bin
|
||||||
wget https://github.com/jomjol/AI-on-the-edge-device/releases/download/${{ steps.last_release.outputs.tag_name }}/AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
wget ${{ github.server_url }}/${{ github.repository }}/releases/download/${{ steps.last_release.outputs.tag_name }}/AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
||||||
unzip AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
unzip AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
||||||
cp -f firmware.bin docs/binary/firmware.bin
|
cp -f firmware.bin docs/binary/firmware.bin
|
||||||
cp -f docs/manifest_template.json docs/manifest.json
|
echo "Updating index and manifest file..."
|
||||||
sed -i 's/VERSION/${{ steps.last_release.outputs.tag_name }}/g' docs/manifest.json
|
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' docs/index.html
|
||||||
|
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' docs/manifest.json
|
||||||
|
|
||||||
- name: Setup Pages
|
- name: Setup Pages
|
||||||
uses: actions/configure-pages@v2
|
uses: actions/configure-pages@v4
|
||||||
|
|
||||||
- name: Upload artifact
|
- name: Upload artifact
|
||||||
uses: actions/upload-pages-artifact@v1
|
uses: actions/upload-pages-artifact@v2
|
||||||
with:
|
with:
|
||||||
path: 'docs'
|
path: 'docs'
|
||||||
|
|
||||||
- name: Deploy to GitHub Pages
|
- name: Deploy to GitHub Pages
|
||||||
id: deployment
|
id: deployment
|
||||||
uses: actions/deploy-pages@v1
|
uses: actions/deploy-pages@v3 # Note: v4 does not work!
|
||||||
|
|||||||
21
.github/workflows/manual-update-contributors-list.yaml
vendored
Normal file
21
.github/workflows/manual-update-contributors-list.yaml
vendored
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
# This updates the Contributors list in the README.md
|
||||||
|
# it only gets run on:
|
||||||
|
# - Manually triggered
|
||||||
|
|
||||||
|
name: Manually update contributors list
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch: # Run on manual trigger
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
manually-update-contributors-list:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
name: A job to automatically update the contributors list in the README.md
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
pull-requests: write
|
||||||
|
steps:
|
||||||
|
- name: Contribute List
|
||||||
|
uses: akhilmhdh/contributors-readme-action@v2.3.10
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
# This updates the Web Installer with the files from the docs folder and the binary of the latest release
|
# This updates the Web Installer with the files from the docs folder and the binary of the latest release
|
||||||
# it only gets run on:
|
# it only gets run on:
|
||||||
# - Changes to the docs folder in the `rolling` branch
|
# - Manually triggered
|
||||||
# - On a release
|
# Make sure to also update the lower part of build.yml!
|
||||||
|
|
||||||
name: Manual Web Installer Update
|
name: Manual Web Installer Update
|
||||||
|
|
||||||
@@ -32,30 +32,32 @@ jobs:
|
|||||||
|
|
||||||
- name: Get version of last release
|
- name: Get version of last release
|
||||||
id: last_release
|
id: last_release
|
||||||
uses: mindojo/get-latest-release@0b8ef1434d7468d6bffcc8263baff5c777f72321
|
uses: joutvhu/get-release@v1
|
||||||
with:
|
with:
|
||||||
myToken: ${{ github.token }}
|
latest: true
|
||||||
exclude_types: "draft|prerelease"
|
prerelease: false
|
||||||
view_top: 1
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
- name: Add binary to Web Installer and update manifest
|
- name: Add binary to Web Installer and update manifest
|
||||||
run: |
|
run: |
|
||||||
|
echo "Updating Web installer to use firmware from ${{ steps.last_release.outputs.tag_name }}..."
|
||||||
rm -f docs/binary/firmware.bin
|
rm -f docs/binary/firmware.bin
|
||||||
wget https://github.com/jomjol/AI-on-the-edge-device/releases/download/${{ steps.last_release.outputs.tag_name }}/AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
wget https://github.com/jomjol/AI-on-the-edge-device/releases/download/${{ steps.last_release.outputs.tag_name }}/AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
||||||
unzip AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
unzip AI-on-the-edge-device__update__${{ steps.last_release.outputs.tag_name }}.zip
|
||||||
cp -f firmware.bin docs/binary/firmware.bin
|
cp -f firmware.bin docs/binary/firmware.bin
|
||||||
cp -f docs/manifest_template.json docs/manifest.json
|
echo "Updating index and manifest file..."
|
||||||
sed -i 's/VERSION/${{ steps.last_release.outputs.tag_name }}/g' docs/manifest.json
|
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' docs/index.html
|
||||||
|
sed -i 's/$VERSION/${{ steps.last_release.outputs.tag_name }}/g' docs/manifest.json
|
||||||
|
|
||||||
- name: Setup Pages
|
- name: Setup Pages
|
||||||
uses: actions/configure-pages@v2
|
uses: actions/configure-pages@v5
|
||||||
|
|
||||||
- name: Upload artifact
|
- name: Upload artifact
|
||||||
uses: actions/upload-pages-artifact@v1
|
uses: actions/upload-pages-artifact@v3
|
||||||
with:
|
with:
|
||||||
path: 'docs'
|
path: 'docs'
|
||||||
|
|
||||||
- name: Deploy to GitHub Pages
|
- name: Deploy to GitHub Pages
|
||||||
id: deployment
|
id: deployment
|
||||||
uses: actions/deploy-pages@v1
|
uses: actions/deploy-pages@v4
|
||||||
|
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
# Reply Bot
|
# Reply Bot
|
||||||
# It uses the configuration in .github/label-commenter-config.yml
|
# It uses the configuration in .github/label-commenter-config.yaml
|
||||||
# See https://github.com/peaceiris/actions-label-commenter
|
# See https://github.com/peaceiris/actions-label-commenter
|
||||||
|
|
||||||
name: Reply-Bot
|
name: Reply-Bot
|
||||||
@@ -20,12 +20,11 @@ jobs:
|
|||||||
comment:
|
comment:
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-20.04
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
|
|
||||||
####################################################################
|
####################################################################
|
||||||
## Remove labels again (issues only)
|
## Remove labels again (issues only)
|
||||||
## Make sure to also add the reply message to .github/label-commenter-config.yml!
|
## Make sure to also add the reply message to .github/label-commenter-config.yaml!
|
||||||
## This currently seems no longer to work due to changes on the actions-cool/issues-helper!
|
## This currently seems no longer to work due to changes on the actions-cool/issues-helper!
|
||||||
####################################################################
|
####################################################################
|
||||||
# - name: Remove 'Logfile' label again (issues only)
|
# - name: Remove 'Logfile' label again (issues only)
|
||||||
@@ -76,4 +75,5 @@ jobs:
|
|||||||
- name: Write Response
|
- name: Write Response
|
||||||
uses: peaceiris/actions-label-commenter@c2d00660c86f2b9ed0fb35b372c451558eba85b3
|
uses: peaceiris/actions-label-commenter@c2d00660c86f2b9ed0fb35b372c451558eba85b3
|
||||||
with:
|
with:
|
||||||
repo-token: "${{ secrets.GITHUB_TOKEN }}"
|
github_token: "${{ secrets.GITHUB_TOKEN }}"
|
||||||
|
config_file: .github/label-commenter-config.yaml
|
||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -23,3 +23,5 @@ CTestTestfile.cmake
|
|||||||
_deps
|
_deps
|
||||||
code/edgeAI.code-workspace
|
code/edgeAI.code-workspace
|
||||||
.DS_Store
|
.DS_Store
|
||||||
|
tools/parameter-tooltip-generator/html
|
||||||
|
tools/parameter-tooltip-generator/AI-on-the-edge-device-docs
|
||||||
|
|||||||
12
.gitmodules
vendored
12
.gitmodules
vendored
@@ -4,6 +4,12 @@
|
|||||||
[submodule "code/components/esp-nn"]
|
[submodule "code/components/esp-nn"]
|
||||||
path = code/components/esp-nn
|
path = code/components/esp-nn
|
||||||
url = https://github.com/espressif/esp-nn.git
|
url = https://github.com/espressif/esp-nn.git
|
||||||
[submodule "code/components/tflite-micro-esp-examples"]
|
[submodule "code/components/esp-tflite-micro"]
|
||||||
path = code/components/tflite-micro-esp-examples
|
path = code/components/esp-tflite-micro
|
||||||
url = https://github.com/espressif/tflite-micro-esp-examples.git
|
url = https://github.com/espressif/esp-tflite-micro.git
|
||||||
|
[submodule "code/components/stb"]
|
||||||
|
path = code/components/stb
|
||||||
|
url = https://github.com/nothings/stb.git
|
||||||
|
[submodule "code/components/esp-protocols"]
|
||||||
|
path = code/components/esp-protocols
|
||||||
|
url = https://github.com/espressif/esp-protocols.git
|
||||||
|
|||||||
614
Changelog.md
614
Changelog.md
File diff suppressed because it is too large
Load Diff
@@ -2,25 +2,54 @@
|
|||||||
|
|
||||||
**There are a lot of ideas for further improvements, but only limited capacity on side of the developer.** Therefore I have created this page as a collection of ideas.
|
**There are a lot of ideas for further improvements, but only limited capacity on side of the developer.** Therefore I have created this page as a collection of ideas.
|
||||||
|
|
||||||
1. Who ever has a new idea can put it here, so it that it is not forgotten.
|
1. Whoever has a new idea can put it here, so that it is not forgotten.
|
||||||
|
|
||||||
2. Who ever has time, capacity and passion to support, can take any of the ideas and implement them.
|
2. Whoever has the time, capacity and passion to support the project can take any of the ideas and implement them. I will provide support and help wherever I can!
|
||||||
I will support and help where ever I can!
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
____
|
____
|
||||||
|
|
||||||
|
|
||||||
|
#### #40 Trigger with cron like exact time slot
|
||||||
|
|
||||||
|
* https://github.com/jomjol/AI-on-the-edge-device/issues/2470
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#### #39 upnp implementation to auto detect the device
|
||||||
|
|
||||||
|
* https://github.com/jomjol/AI-on-the-edge-device/issues/2481
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#### #38 Energy Saving
|
||||||
|
|
||||||
|
* Deep sleep between recognition
|
||||||
|
* https://github.com/jomjol/AI-on-the-edge-device/issues/2486
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#### #37 Auto init SD card
|
||||||
|
|
||||||
|
* Fully implement the SD card handling (including formatting) into the firmware
|
||||||
|
* https://github.com/jomjol/AI-on-the-edge-device/issues/2488Demo
|
||||||
|
|
||||||
|
#### #36 Run demo without camera
|
||||||
|
|
||||||
|
Demo mode requires a working camera (if not, one receives a 'Cam bad' error). Would be nice to demo or play around on other ESP32 boards (or on ESP32-CAM boards when you broke the camera cable...).
|
||||||
|
|
||||||
#### #35 Use the same model, but provide the image from a Smartphone Camera
|
#### #35 Use the same model, but provide the image from a Smartphone Camera
|
||||||
as reading the Electricity or Water meter every few minutues only delivers apparent accuracy (DE: "Scheingenauigkeit") you could just as well take a picture with your Smartphone evey so often (e.g. once a week when you are in the Basement anyway), then with some "semi clever" tricks pass this image to the model developed here, and the values then on to who ever needs them e.g. via MQTT.
|
as reading the Electricity or Water meter every few minutes only delivers apparent accuracy (DE: "Scheingenauigkeit") you could just as well take a picture with your Smartphone every so often (e.g. once a week when you are in the Basement anyway), then with some "semi clever" tricks pass this image to the model developed here, and the values than on to whoever needs them e.g. via MQTT.
|
||||||
IMO: It is not needed to have that many readings (datapoints) as our behaviour (Use of electricity or water) doesn't vary that much, say, over a weeks time. The interpolation between weekly readings will give sufficient information on the power and/or water usage.
|
IMO: It is not needed to have that many readings (datapoints) as our behaviour (Use of electricity or water) doesn't vary that much, say, over a weeks time. The interpolation between weekly readings will give sufficient information on the power and/or water usage.
|
||||||
|
|
||||||
|
|
||||||
#### #34 implement state and Roi for water leak detection
|
#### #34 implement state and Roi for water leak detection
|
||||||
for example see Roi on the next picture..
|
for example see Roi in the next picture..
|
||||||

|

|
||||||
in case of position change between the measurments set this state to true, if there is no change set it back to false.
|
in case of position change between the measurements set this state to true, if there is no change set it back to false.
|
||||||
In a defined time window this movement can lead into an alarm state / water leak..
|
In a defined time window this movement can lead into an alarm state / water leak..
|
||||||
haveing this state in the mqtt broker can trigger functions like closing the ater pipe walve and so on...
|
having this state in the mqtt broker can trigger functions like closing the water pipe valve and so on...
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -36,7 +65,7 @@ haveing this state in the mqtt broker can trigger functions like closing the ate
|
|||||||
|
|
||||||
#### #31 Implement InfluxDB v2.x interface
|
#### #31 Implement InfluxDB v2.x interface
|
||||||
|
|
||||||
* Currently only InfluxDB v1.x is supportet, extend to v2.x
|
* Currently only InfluxDB v1.x is supported, extend to v2.x
|
||||||
* Remark: interface has changed
|
* Remark: interface has changed
|
||||||
* see [#1160](https://github.com/jomjol/AI-on-the-edge-device/issues/1160)
|
* see [#1160](https://github.com/jomjol/AI-on-the-edge-device/issues/1160)
|
||||||
|
|
||||||
@@ -48,12 +77,12 @@ haveing this state in the mqtt broker can trigger functions like closing the ate
|
|||||||
|
|
||||||
#### ~~#29 Add favicon and use the hostname for the website~~- implemented v11.3.1
|
#### ~~#29 Add favicon and use the hostname for the website~~- implemented v11.3.1
|
||||||
|
|
||||||
~~* https://github.com/jomjol/AI-on-the-edge-device/issues/927~~
|
* ~~https://github.com/jomjol/AI-on-the-edge-device/issues/927~~
|
||||||
|
|
||||||
#### #28 Improved error handling for ROIs
|
#### #28 Improved error handling for ROIs
|
||||||
|
|
||||||
* In case a ROI is out of the image, there is no error message, but a non sense image is used
|
* In case a ROI is out of the image, there is no error message, but a non sense image is used
|
||||||
* Implement a error message for wrong configuratioin of ROI
|
* Implement a error message for wrong configuration of ROI
|
||||||
|
|
||||||
#### #27 Use Homie Spec for Mqtt binding
|
#### #27 Use Homie Spec for Mqtt binding
|
||||||
|
|
||||||
@@ -86,7 +115,7 @@ haveing this state in the mqtt broker can trigger functions like closing the ate
|
|||||||
|
|
||||||
#### ~~#22 Direct hint to the different neural network files in the other repositories~~- implemented >v11.3.1
|
#### ~~#22 Direct hint to the different neural network files in the other repositories~~- implemented >v11.3.1
|
||||||
|
|
||||||
~~* https://github.com/jomjol/AI-on-the-edge-device/issues/644~~
|
* ~~https://github.com/jomjol/AI-on-the-edge-device/issues/644~~
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
68
Licence.md
Normal file
68
Licence.md
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
# **Dual Use License for AI-on-the-Edge Device**
|
||||||
|
|
||||||
|
Version: 1.0
|
||||||
|
Date: 2025-01-05 (5th January 2025)
|
||||||
|
|
||||||
|
## **Preamble**
|
||||||
|
|
||||||
|
This license allows individuals to use, modify, and share AI-on-the-Edge freely for private, non-commercial purposes. Any commercial use requires a separate licensing agreement with the rights holder.
|
||||||
|
|
||||||
|
------
|
||||||
|
|
||||||
|
## **1. Grant of License**
|
||||||
|
|
||||||
|
### 1.1 **Private Use**
|
||||||
|
|
||||||
|
The licensor grants the licensee a free, non-exclusive, worldwide license to use, modify, and distribute the software for private, non-commercial purposes.
|
||||||
|
|
||||||
|
### 1.2 **Commercial Use**
|
||||||
|
|
||||||
|
The use of the software or any derivative works in any commercial context (including, but not limited to, selling, renting, providing as a service, or integrating into commercial products) is prohibited without a separate commercial license.
|
||||||
|
|
||||||
|
------
|
||||||
|
|
||||||
|
## **2. Obligation to Private Derivatives**
|
||||||
|
|
||||||
|
In modified private versions of the software, the unchanged license as well as the reference to the original source and authors must always be stated (https://github.com/jomjol/AI-on-the-edge-device).
|
||||||
|
|
||||||
|
Modified versions of the software must be clearly marked as such and must not imply they are provided by the original licensor.
|
||||||
|
|
||||||
|
------
|
||||||
|
|
||||||
|
## **3. Commercial Licensing**
|
||||||
|
|
||||||
|
Companies, organizations, or individuals wishing to use the software for commercial purposes must obtain a separate commercial license. Please contact mueller.josef(@)gmail.com for further details.
|
||||||
|
|
||||||
|
------
|
||||||
|
|
||||||
|
## **4. Terms of Cooperation**
|
||||||
|
|
||||||
|
By contributing to the AI-on-the-Edge software, this license is considered accepted. This applies to, but is not limited to, code, error corrections, extensions, artwork, documentation, and new features. Any contribution, including libraries and sources, must comply with the terms of this license.
|
||||||
|
|
||||||
|
The contributor agrees that the added code and functionality may also be used in commercial versions without compensation to the contributor.
|
||||||
|
|
||||||
|
------
|
||||||
|
|
||||||
|
## **5. Disclaimer of Liability**
|
||||||
|
|
||||||
|
### 5.1 **General Disclaimer**
|
||||||
|
|
||||||
|
The software is provided "as is", without any express or implied warranties. The licensor is not liable for any damages resulting from the use of the software.
|
||||||
|
|
||||||
|
### 5.2 **No Usage in Safety or Security Environments**
|
||||||
|
|
||||||
|
The image processing uses neural networks, among other algorithms, whose results can produce incorrect or unexpected outcomes due to their functionality and the underlying training data. Therefore, this system must not be used or offered for safety-relevant systems or systems with high reliability requirements.
|
||||||
|
|
||||||
|
------
|
||||||
|
|
||||||
|
## **6. General Provisions**
|
||||||
|
|
||||||
|
### 6.1 **Severability Clause**
|
||||||
|
|
||||||
|
If any provision of this license is deemed invalid, the remaining provisions shall remain in full force and effect.
|
||||||
|
|
||||||
|
------
|
||||||
|
|
||||||
|
## **Acceptance**
|
||||||
|
|
||||||
|
By using this software, the licensee agrees to the terms of this license.
|
||||||
733
README.md
733
README.md
@@ -1,102 +1,681 @@
|
|||||||
# Welcome to the AI-on-the-edge-device
|
<h1 align="center">AI on the Edge Device: Digitizing Your non-digital meters with an ESP32-CAM</h1>
|
||||||
<img src="images/icon/watermeter.svg" width="100px">
|
<br>
|
||||||
|
<br>
|
||||||
|
|
||||||
Artificial intelligence based systems have been established in our every days live. Just think of speech or image recognition. Most of the systems relay on either powerful processors or a direct connection to the cloud for doing the calculations up there. With the increasing power of modern processors the AI systems are coming closer to the end user - which is usually called **edge computing**.
|
## Table of Contents
|
||||||
Here this edge computing is brought into a practical oriented example, where a AI network is implemented on a ESP32 device so: **AI on the edge**.
|
- [Key Features 🚀](#key-features-)
|
||||||
|
- [Workflow 🔧](#workflow-)
|
||||||
|
- [Impressions 📷](#impressions-)
|
||||||
|
- [AI-on-the-edge-device on a Water Meter 💧](#ai-on-the-edge-device-on-a-water-meter-)
|
||||||
|
- [Web Interface (Water Meter) 💻](#web-interface-water-meter-)
|
||||||
|
- [AI-on-the-edge-device on an Electrical Power Meter ⚡](#ai-on-the-edge-device-on-an-electrical-power-meter-)
|
||||||
|
- [Setup 🛠️](#setup-%EF%B8%8F)
|
||||||
|
- [Download 🔽](#download-)
|
||||||
|
- [Flashing the ESP32 💾](#flashing-the-esp32-)
|
||||||
|
- [Flashing the SD Card 💾](#flashing-the-sd-card-)
|
||||||
|
- [Casing 🛠️](#casing-%EF%B8%8F)
|
||||||
|
- [Donate ☕](#donate-)
|
||||||
|
- [Support 💬](#support-)
|
||||||
|
- [Changes and History 📜](#changes-and-history-)
|
||||||
|
- [Build It Yourself 🔨](#build-it-yourself-)
|
||||||
|
- [Tools 🛠️](#tools-%EF%B8%8F)
|
||||||
|
- [Additional Ideas 💡](#additional-ideas-)
|
||||||
|
- [Our Contributors ❤️](#our-contributors-%EF%B8%8F)
|
||||||
|
|
||||||
This projects allows you to digitalize your **analoge** water, gas, power and other meters using cheap and easily available hardware.
|
<p align="center">
|
||||||
|
<a href="#top">
|
||||||
All you need is an [ESP32 board with a supported camera](https://jomjol.github.io/AI-on-the-edge-device-docs/Hardware-Compatibility/) and a bit of a practical hand.
|
<img src="https://img.shields.io/badge/Back%20to%20Top-000000?style=for-the-badge&logo=github&logoColor=white" alt="Back to Top">
|
||||||
|
</a>
|
||||||
<img src="images/esp32-cam.png" width="200px">
|
</p>
|
||||||
|
|
||||||
## Key features
|
|
||||||
- Tensorflow Lite (TFlite) integration - including easy to use wrapper
|
|
||||||
- Inline Image processing (feature detection, alignment, ROI extraction)
|
|
||||||
- **Small** and **cheap** device (3x4.5x2 cm³, < 10 EUR)
|
|
||||||
- camera and illumination integrated
|
|
||||||
- Web surface to administrate and control
|
|
||||||
- OTA-Interface to update directly through the web interface
|
|
||||||
- Full integration into Homeassistant
|
|
||||||
- Support for Influx DB 1
|
|
||||||
- MQTT
|
|
||||||
- REST API
|
|
||||||
|
|
||||||
## Workflow
|
|
||||||
The device takes a photo of your meter at a defined interval. It then extracts the Regions of Interest (ROI's) out of it and runs them through an artificial inteligence. As a result, you get the digitalized value of your meter.
|
|
||||||
|
|
||||||
There are several options what to do with that value. Either send it to a MQTT broker, write it to an InfluxDb or simply provide it throug a REST API.
|
|
||||||
|
|
||||||
<img src="https://raw.githubusercontent.com/jomjol/AI-on-the-edge-device/master/images/idea.jpg" width="600">
|
|
||||||
|
|
||||||
## Impressions
|
|
||||||
### AI-on-the-edge-device on a Water Meter
|
|
||||||
<img src="https://raw.githubusercontent.com/jomjol/AI-on-the-edge-device/master/images/watermeter_all.jpg" width="200"><img src="https://raw.githubusercontent.com/jomjol/AI-on-the-edge-device/master/images/main.jpg" width="200"><img src="https://raw.githubusercontent.com/jomjol/AI-on-the-edge-device/master/images/size.png" width="200">
|
|
||||||
|
|
||||||
### Web Interface (Water Meter)
|
|
||||||
<img src="https://raw.githubusercontent.com/jomjol/AI-on-the-edge-device/master/images/watermeter.jpg" width="600">
|
|
||||||
|
|
||||||
### AI-on-the-edge-device on a Electrical Power Meter
|
|
||||||
<img src="https://raw.githubusercontent.com/jomjol/AI-on-the-edge-device/master/images/powermeter.jpg" width="600">
|
|
||||||
|
|
||||||
|
|
||||||
## Setup
|
|
||||||
There is a growing [documentation](https://jomjol.github.io/AI-on-the-edge-device-docs/) which provides you with a lot of information.
|
|
||||||
Head there to get a start, set it up and configure it.
|
|
||||||
|
|
||||||
There are also a articles in the German Heise magazine "make:" about the setup and the technical background (behind a paywall) : [DIY - Setup](https://www.heise.de/select/make/2021/2/2103513300897420296)
|
|
||||||
|
|
||||||
For further background information, head to [Neural Networks](https://www.heise.de/select/make/2021/6/2126410443385102621), [Training Neural Networks](https://www.heise.de/select/make/2022/1/2134114065999161585) and [Programming on the ESP32](https://www.heise.de/select/make/2022/2/2204010051597422030)
|
[](https://github.com/jomjol/AI-on-the-edge-device/tree/main/code)
|
||||||
|
[](https://jomjol.github.io/AI-on-the-edge-device-docs/)
|
||||||
|
[](https://GitHub.com/jomjol/AI-on-the-edge-device/releases/)
|
||||||
|
[](https://GitHub.com/jomjol/AI-on-the-edge-device/releases/)
|
||||||
|
[](https://GitHub.com/jomjol/AI-on-the-edge-device/network/)
|
||||||
|
[](https://GitHub.com/jomjol/AI-on-the-edge-device/stargazers/)
|
||||||
|
|
||||||
### Download
|
<p align="center" id="top">
|
||||||
The latest available version is available on the [Releases page](https://github.com/jomjol/AI-on-the-edge-device/releases).
|
<img src="images/icon/watermeter.svg" width="150px">
|
||||||
|
</p>
|
||||||
|
|
||||||
### Flashing of the ESP32
|
Artificial intelligence is everywhere, from speech to image recognition. While most AI systems rely on powerful processors or cloud computing, **edge computing** brings AI closer to the end user by utilizing the capabilities of modern processors.
|
||||||
Initially you will have to flash the ESP32 through an USB connection. Later an update is possible directly over the Air (OTA).
|
This project demonstrates edge computing using the **ESP32**, a low-cost, AI-capable device, to digitize your analog meters—whether water, gas, or electricity. With affordable hardware and simple instructions, you can turn any standard meter into a smart device.
|
||||||
|
|
||||||
|
Let's explore how to make **AI on the Edge** a reality! 🌟
|
||||||
|
|
||||||
|
All you need is an [ESP32 board with a supported camera](https://jomjol.github.io/AI-on-the-edge-device-docs/Hardware-Compatibility/) and some practical skills. 🛠️
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Key Features 🚀
|
||||||
|
- 🔗 **Tensorflow Lite (TFLite) integration** – including an easy-to-use wrapper.
|
||||||
|
- 📸 **Inline image processing** (feature detection, alignment, ROI extraction).
|
||||||
|
- 💡 **Small** and **affordable** device (3 x 4.5 x 2 cm³, less than 10 EUR).
|
||||||
|
- 📷 Integrated camera and illumination.
|
||||||
|
- 🌐 Web interface for administration and control.
|
||||||
|
- 🔄 OTA interface for updating directly via the web interface.
|
||||||
|
- 🏠 Full integration with Home Assistant.
|
||||||
|
- 📊 Support for **Influx DB 1** and **2**.
|
||||||
|
- 📡 **MQTT protocol** support.
|
||||||
|
- 📥 **REST API** available for data access.
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Workflow 🔧
|
||||||
|
The device captures a photo of your meter at set intervals. It then extracts the Regions of Interest (ROIs) from the image and runs them through artificial intelligence. As a result, you get the digitized value of your meter.
|
||||||
|
|
||||||
|
There are several options for what to do with that value:
|
||||||
|
- 📤 Send it to a **MQTT broker**.
|
||||||
|
- 📝 Write it to an **InfluxDb**.
|
||||||
|
- 🔗 Provide access via a **REST API**.
|
||||||
|
|
||||||
|
<p align="center">
|
||||||
|
<img src="images/idea.jpg" width="600">
|
||||||
|
</p>
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Impressions 📷
|
||||||
|
|
||||||
|
+ ### AI-on-the-edge-device on a Water Meter 💧
|
||||||
|
<p align="center">
|
||||||
|
<img src="images/watermeter_all.jpg" width="200"><img
|
||||||
|
src="images/main.jpg" width="200"><img
|
||||||
|
src="images/size.png" width="200">
|
||||||
|
</p>
|
||||||
|
|
||||||
|
+ ### Web Interface (Water Meter) 💻
|
||||||
|
<p align="center">
|
||||||
|
<img src="images/watermeter.jpg" width="600">
|
||||||
|
</p>
|
||||||
|
|
||||||
|
+ ### AI-on-the-edge-device on an Electrical Power Meter ⚡
|
||||||
|
<p align="center">
|
||||||
|
<img src="images/powermeter.jpg" width="600">
|
||||||
|
</p>
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Setup 🛠️
|
||||||
|
There is growing [documentation](https://jomjol.github.io/AI-on-the-edge-device-docs/) which provides you with a lot of information. Head there to get started, set it up, and configure it.
|
||||||
|
|
||||||
|
There are also articles in the German Heise magazine "make:" about the setup and technical background (behind a paywall): [DIY - Setup](https://www.heise.de/select/make/2021/2/2103513300897420296) 📰
|
||||||
|
|
||||||
|
A lot of people have created useful YouTube videos that might help you get started:
|
||||||
|
- 🎥 [youtube.com/watch?v=HKBofb1cnNc](https://www.youtube.com/watch?v=HKBofb1cnNc)
|
||||||
|
- 🎥 [youtube.com/watch?v=yyf0ORNLCk4](https://www.youtube.com/watch?v=yyf0ORNLCk4)
|
||||||
|
- 🎥 [youtube.com/watch?v=XxmTubGek6M](https://www.youtube.com/watch?v=XxmTubGek6M)
|
||||||
|
- 🎥 [youtube.com/watch?v=mDIJEyElkAU](https://www.youtube.com/watch?v=mDIJEyElkAU)
|
||||||
|
- 🎥 [youtube.com/watch?v=SssiPkyKVVs](https://www.youtube.com/watch?v=SssiPkyKVVs)
|
||||||
|
- 🎥 [youtube.com/watch?v=MAHE_QyHZFQ](https://www.youtube.com/watch?v=MAHE_QyHZFQ)
|
||||||
|
- 🎥 [youtube.com/watch?v=Uap_6bwtILQ](https://www.youtube.com/watch?v=Uap_6bwtILQ)
|
||||||
|
|
||||||
|
For further background information, head to:
|
||||||
|
- [Neural Networks](https://www.heise.de/select/make/2021/6/2126410443385102621)
|
||||||
|
- [Training Neural Networks](https://www.heise.de/select/make/2022/1/2134114065999161585)
|
||||||
|
- [Programming on the ESP32](https://www.heise.de/select/make/2022/2/2204010051597422030)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Download 🔽
|
||||||
|
The latest available version can be found on the [Releases page](https://github.com/jomjol/AI-on-the-edge-device/releases).
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Flashing the ESP32 💾
|
||||||
|
Initially, you will have to flash the ESP32 via a USB connection. Later updates are possible directly over the air (OTA using Wi-Fi).
|
||||||
|
|
||||||
There are different ways to flash your ESP32:
|
There are different ways to flash your ESP32:
|
||||||
- [Web Installer and Console](https://jomjol.github.io/AI-on-the-edge-device/index.html) (Webbrowser based tool to flash the ESP32 and extract the Log over USB)
|
- The preferred way is the [Web Installer and Console](https://jomjol.github.io/AI-on-the-edge-device/index.html), a browser-based tool to flash the ESP32 and extract the log over USB:
|
||||||
|

|
||||||
- Flash Tool from Espressif
|
- Flash Tool from Espressif
|
||||||
- ESPtool (Command Line Tool)
|
- ESPtool (command-line tool)
|
||||||
|
|
||||||
See the [Docu](https://jomjol.github.io/AI-on-the-edge-device-docs/Installation/) for more information.
|
See the [documentation](https://jomjol.github.io/AI-on-the-edge-device-docs/Installation/) for more information.
|
||||||
|
|
||||||
### Flashing the SD-Card
|
---
|
||||||
The SD-Card must be flashed separately, see the [Docu](https://jomjol.github.io/AI-on-the-edge-device-docs/Installation/) for details.
|
|
||||||
|
|
||||||
## Casing
|
<br>
|
||||||
|
|
||||||
A 3d-printable housing can be found here:
|
## Flashing the SD Card 💾
|
||||||
- https://www.thingiverse.com/thing:4573481 (Water Meter)
|
The SD card can be set up automatically after the firmware is installed. See the [documentation](https://jomjol.github.io/AI-on-the-edge-device-docs/Installation/#remote-setup-using-the-built-in-access-point) for details. For this to work, the SD card must be FAT formatted (which is the default on a new SD card).
|
||||||
- https://www.thingiverse.com/thing:5028229 (Power Meter)
|
|
||||||
- https://www.thingiverse.com/thing:5224101 (Gas Meter)
|
|
||||||
- https://www.thingiverse.com/thing:4571627 (ESP32-Cam housing only)
|
|
||||||
|
|
||||||
## Build it yourself
|
Alternatively, the SD card can still be set up manually. See the [documentation](https://jomjol.github.io/AI-on-the-edge-device-docs/Installation/#3-sd-card) for details.
|
||||||
See [Build Instructions](code/README.md).
|
|
||||||
|
|
||||||
## Donate
|
---
|
||||||
If you would like to support the developer with a cup of coffee you can do that via [Paypal](https://www.paypal.com/donate?hosted_button_id=8TRSVYNYKDSWL).
|
|
||||||
|
|
||||||
<form action="https://www.paypal.com/donate" method="post" target="_top">
|
<br>
|
||||||
<input type="hidden" name="hosted_button_id" value="8TRSVYNYKDSWL" />
|
|
||||||
<input type="image" src="https://www.paypalobjects.com/en_US/DK/i/btn/btn_donateCC_LG.gif" border="0" name="submit" title="PayPal - The safer, easier way to pay online!" alt="Donate with PayPal button" />
|
|
||||||
<img alt="" border="0" src="https://www.paypal.com/en_DE/i/scr/pixel.gif" width="1" height="1" />
|
|
||||||
</form>
|
|
||||||
If you have any technical topics, you can create an [Issue](https://github.com/jomjol/AI-on-the-edge-device/issues).
|
|
||||||
|
|
||||||
In other cases you can contact the developer via email: <img src="https://raw.githubusercontent.com/jomjol/AI-on-the-edge-device/master/images/mail.jpg" height="25">
|
## Casing 🛠️
|
||||||
|
Various 3D-printable housings can be found here:
|
||||||
|
- 💧 [Water Meter](https://www.thingiverse.com/thing:4573481)
|
||||||
|
- ⚡ [Power Meter](https://www.thingiverse.com/thing:5028229)
|
||||||
|
- 🔥 [Gas Meter](https://www.thingiverse.com/thing:5224101)
|
||||||
|
- 📷 [ESP32-cam housing only](https://www.thingiverse.com/thing:4571627)
|
||||||
|
|
||||||
## Changes and History
|
---
|
||||||
See [Changelog](Changelog.md)
|
|
||||||
|
|
||||||
## Tools
|
<br>
|
||||||
|
|
||||||
* Logfile downloader and combiner (Thx to [reserve85](https://github.com/reserve85))
|
## Donate ☕
|
||||||
* Files see ['/tools/logfile-tool'](tbd), How-to see [Docu](https://jomjol.github.io/AI-on-the-edge-device-docs/outdated--Gasmeter-Log-Downloader/)
|
If you'd like to support the developer with a cup of coffee, you can do so via [PayPal](https://www.paypal.com/donate?hosted_button_id=8TRSVYNYKDSWL).
|
||||||
|
|
||||||
## Additional Ideas
|
<p align="center">
|
||||||
There are some ideas and feature requests which are not followed currently - mainly due to capacity reasons on side of the developer. They are collected here: [FeatureRequest.md](FeatureRequest.md)
|
<a href="https://www.paypal.com/donate?hosted_button_id=8TRSVYNYKDSWL"><img border="0" src="images/paypal.png" width="200px" target="_blank"></a>
|
||||||
|
</p>
|
||||||
|
|
||||||
------
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Support 💬
|
||||||
|
If you have any technical problems, please search the [discussions](https://github.com/jomjol/AI-on-the-edge-device/discussions). In case you find a bug or have a feature request, please open an [issue](https://github.com/jomjol/AI-on-the-edge-device/issues).
|
||||||
|
|
||||||
|
For any other issues, you can contact the developer via email:
|
||||||
|
<p align="center">
|
||||||
|
<img src="images/mail.jpg" height="25">
|
||||||
|
</p>
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Changes and History 📜
|
||||||
|
See the [Changelog](Changelog.md) for detailed information.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Build It Yourself 🔨
|
||||||
|
See the [Build Instructions](code/README.md) for step-by-step guidance.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Tools 🛠️
|
||||||
|
* Logfile downloader and combiner (Thanks to [reserve85](https://github.com/reserve85))
|
||||||
|
* It can be found at ['/tools/logfile-tool'](https://github.com/jomjol/AI-on-the-edge-device/tree/main/tools/logfile-tool).
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Additional Ideas 💡
|
||||||
|
There are some ideas and feature requests which are not currently being pursued—mainly due to capacity constraints on the part of the developers. These features are collected in the [issues](https://github.com/jomjol/AI-on-the-edge-device/issues) and in [FeatureRequest.md](FeatureRequest.md).
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<br>
|
||||||
|
|
||||||
|
## Our Contributors ❤️
|
||||||
|
<!-- Do not manually edit this section! It should get updated using the Github action "Manually update contributors list" -->
|
||||||
|
<!-- readme: contributors -start -->
|
||||||
|
<table>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/jomjol">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/30766535?v=4" width="100;" alt="jomjol"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>jomjol</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/caco3">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/1783586?v=4" width="100;" alt="caco3"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>CaCO3</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/haverland">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/412645?v=4" width="100;" alt="haverland"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Frank Haverland</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/SybexX">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/587201?v=4" width="100;" alt="SybexX"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>michael</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/Slider0007">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/115730895?v=4" width="100;" alt="Slider0007"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Slider0007</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/nliaudat">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/6782613?v=4" width="100;" alt="nliaudat"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Nicolas Liaudat</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/Zwer2k">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/10438794?v=4" width="100;" alt="Zwer2k"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Zwer2k</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/phlupp">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/6304863?v=4" width="100;" alt="phlupp"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>phlupp</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/jasaw">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/721280?v=4" width="100;" alt="jasaw"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>jasaw</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/dockSquadron">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/11964767?v=4" width="100;" alt="dockSquadron"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>dockSquadron</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/rdmueller">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/1856308?v=4" width="100;" alt="rdmueller"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Ralf D. Müller</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/cristianmitran">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/36613624?v=4" width="100;" alt="cristianmitran"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>cristianmitran</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/michaeljoos72">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/20517474?v=4" width="100;" alt="michaeljoos72"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>michaeljoos72</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/henrythasler">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/6277203?v=4" width="100;" alt="henrythasler"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Henry Thasler</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/amantyagiprojects">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/174239452?v=4" width="100;" alt="amantyagiprojects"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Naman Tyagi</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/pixeldoc2000">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/376715?v=4" width="100;" alt="pixeldoc2000"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>pixel::doc</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/mad2xlc">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/37449746?v=4" width="100;" alt="mad2xlc"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Stefan</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/jochenchrist">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/2930448?v=4" width="100;" alt="jochenchrist"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>jochenchrist</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/parhedberg">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/13777521?v=4" width="100;" alt="parhedberg"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>parhedberg</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/fsck-block">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/58307481?v=4" width="100;" alt="fsck-block"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>fsck-block</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/slovdahl">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/1417619?v=4" width="100;" alt="slovdahl"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Sebastian Lövdahl</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/RaHehl">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/7577984?v=4" width="100;" alt="RaHehl"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Raphael Hehl</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/LordGuilly">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/13271835?v=4" width="100;" alt="LordGuilly"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>LordGuilly</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/muggenhor">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/484066?v=4" width="100;" alt="muggenhor"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Giel van Schijndel</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/bilalmirza74">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/84387676?v=4" width="100;" alt="bilalmirza74"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Bilal Mirza</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/AngryApostrophe">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/89547888?v=4" width="100;" alt="AngryApostrophe"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>AngryApostrophe</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/ralf1307">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/46164027?v=4" width="100;" alt="ralf1307"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Ralf Rachinger</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/Ranjana761">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/129291313?v=4" width="100;" alt="Ranjana761"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Ranjana761</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/SURYANSH-RAI">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/79277130?v=4" width="100;" alt="SURYANSH-RAI"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>SURYANSH RAI</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/SkylightXD">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/16561545?v=4" width="100;" alt="SkylightXD"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>SkylightXD</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/ottk3">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/5236802?v=4" width="100;" alt="ottk3"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Sven Rojek</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/Turbo87">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/141300?v=4" width="100;" alt="Turbo87"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Tobias Bieniek</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/tkopczuk">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/101632?v=4" width="100;" alt="tkopczuk"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Tomek Kopczuk</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/yonz2">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/13886257?v=4" width="100;" alt="yonz2"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Yonz</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/Yveaux">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/7716005?v=4" width="100;" alt="Yveaux"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Yveaux</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/flooxo">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/93255373?v=4" width="100;" alt="flooxo"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>flox_x</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/gneluka">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/32097881?v=4" width="100;" alt="gneluka"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>gneluka</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/kalwados">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/11840444?v=4" width="100;" alt="kalwados"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>kalwados</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/kub3let">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/95883234?v=4" width="100;" alt="kub3let"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>kub3let</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/pfeifferch">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/73090220?v=4" width="100;" alt="pfeifferch"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>pfeifferch</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/rstephan">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/8532364?v=4" width="100;" alt="rstephan"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>rstephan</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/smartboart">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/38385805?v=4" width="100;" alt="smartboart"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>smartboart</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/wetneb">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/309908?v=4" width="100;" alt="wetneb"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Antonin Delpeuch</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/adarazs">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/6269603?v=4" width="100;" alt="adarazs"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Attila Darazs</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/austindrenski">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/21338699?v=4" width="100;" alt="austindrenski"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Austin Drenski</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/PLCHome">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/29116097?v=4" width="100;" alt="PLCHome"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>PLCHome</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/CFenner">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/9592452?v=4" width="100;" alt="CFenner"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Christopher Fenner</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/dkneisz">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/43378003?v=4" width="100;" alt="dkneisz"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Dave</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/FarukhS52">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/129654632?v=4" width="100;" alt="FarukhS52"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Farookh Zaheer Siddiqui</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/hex7c0">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/4419146?v=4" width="100;" alt="hex7c0"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Francesco Carnielli</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/040medien">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/115072?v=4" width="100;" alt="040medien"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Frederik Kemner</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/eltociear">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/22633385?v=4" width="100;" alt="eltociear"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Ikko Eltociear Ashimine</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/queeek">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/9533371?v=4" width="100;" alt="queeek"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Ina</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/joergrosenkranz">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/310438?v=4" width="100;" alt="joergrosenkranz"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Joerg Rosenkranz</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/Innovatorcloudy">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/183274513?v=4" width="100;" alt="Innovatorcloudy"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>KrishCode</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/myxor">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/1397377?v=4" width="100;" alt="myxor"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Marco H</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/rainman110">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/3213107?v=4" width="100;" alt="rainman110"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Martin Siggel</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/mkelley88">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/5567324?v=4" width="100;" alt="mkelley88"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Matthew T. Kelley</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/toolsfactory">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/7744975?v=4" width="100;" alt="toolsfactory"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Michael Geissler</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td align="center">
|
||||||
|
<a href="https://github.com/ppisljar">
|
||||||
|
<img src="https://avatars.githubusercontent.com/u/13629809?v=4" width="100;" alt="ppisljar"/>
|
||||||
|
<br />
|
||||||
|
<sub><b>Peter Pisljar</b></sub>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<tbody>
|
||||||
|
</table>
|
||||||
|
<!-- readme: contributors -end -->
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
<div align="center">
|
||||||
|
<a href="#top">
|
||||||
|
<img src="https://img.shields.io/badge/Back%20to%20Top-000000?style=for-the-badge&logo=github&logoColor=white" alt="Back to Top">
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
|||||||
1
code/.gitignore
vendored
1
code/.gitignore
vendored
@@ -1,4 +1,5 @@
|
|||||||
.pio
|
.pio
|
||||||
|
.idea
|
||||||
.vscode/.browse.c_cpp.db*
|
.vscode/.browse.c_cpp.db*
|
||||||
.vscode/c_cpp_properties.json
|
.vscode/c_cpp_properties.json
|
||||||
.vscode/launch.json
|
.vscode/launch.json
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
cmake_minimum_required(VERSION 3.16.0)
|
cmake_minimum_required(VERSION 3.16.0)
|
||||||
|
|
||||||
list(APPEND EXTRA_COMPONENT_DIRS $ENV{IDF_PATH}/examples/common_components/protocol_examples_common components/tflite-micro-esp-examples/components/tflite-lib)
|
list(APPEND EXTRA_COMPONENT_DIRS $ENV{IDF_PATH}/examples/common_components/protocol_examples_common components/esp-tflite-micro components/esp-protocols/components/mdns)
|
||||||
|
|
||||||
ADD_CUSTOM_COMMAND(
|
ADD_CUSTOM_COMMAND(
|
||||||
OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/version.cpp
|
OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/version.cpp
|
||||||
|
|||||||
@@ -8,6 +8,15 @@ git checkout rolling
|
|||||||
git submodule update --init
|
git submodule update --init
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Update Submodules
|
||||||
|
```
|
||||||
|
cd /components/submodule-name (e.g. tflite-micro-example)
|
||||||
|
git checkout VERSION (e.g. HASH of latest tflite-micro-example build)
|
||||||
|
cd ../../ (auf Ebene von code)
|
||||||
|
git submodule update --init
|
||||||
|
```
|
||||||
|
Evt. muss man vorher noch einige Verzeichnisse in compenents von Hand löschen, da sie beim checkout nicht gelöscht wurden (vor update -- init)
|
||||||
|
|
||||||
## Build and Flash within terminal
|
## Build and Flash within terminal
|
||||||
See further down to build it within an IDE.
|
See further down to build it within an IDE.
|
||||||
### Compile
|
### Compile
|
||||||
@@ -60,3 +69,6 @@ pio device monitor -p /dev/ttyUSB0
|
|||||||
- `pio run --target erase` to erase the flash
|
- `pio run --target erase` to erase the flash
|
||||||
- `pio run --target upload` this will upload the `bootloader.bin, partitions.bin,firmware.bin` from the `code/.pio/build/esp32cam/` folder.
|
- `pio run --target upload` this will upload the `bootloader.bin, partitions.bin,firmware.bin` from the `code/.pio/build/esp32cam/` folder.
|
||||||
- `pio device monitor` to observe the logs via uart
|
- `pio device monitor` to observe the logs via uart
|
||||||
|
|
||||||
|
# Update Parameters
|
||||||
|
If you create or rename a parameter, make sure to update its documentation in `../param-docs/parameter-pages`! Check the `../param-docs/README.md` for more information.
|
||||||
|
|||||||
Submodule code/components/esp-nn updated: 6b3ef8e226...9195e969a7
1
code/components/esp-protocols
Submodule
1
code/components/esp-protocols
Submodule
Submodule code/components/esp-protocols added at 9b74256b51
1
code/components/esp-tflite-micro
Submodule
1
code/components/esp-tflite-micro
Submodule
Submodule code/components/esp-tflite-micro added at 07c014eb65
Submodule code/components/esp32-camera updated: 5c8349f4cf...dba8da9898
@@ -1,27 +1,53 @@
|
|||||||
|
/********************************************************************************
|
||||||
|
* https://github.com/RoboticsBrno/SmartLeds
|
||||||
|
*
|
||||||
|
* MIT License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2017 RoboticsBrno (RobotikaBrno)
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*******************************************************************************/
|
||||||
|
|
||||||
#include "Color.h"
|
#include "Color.h"
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <cmath>
|
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
|
#include <cmath>
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
// Int -> fixed point
|
// Int -> fixed point
|
||||||
int up( int x ) { return x * 255; }
|
int up(int x) { return x * 255; }
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
int iRgbSqrt( int num ) {
|
int iRgbSqrt(int num) {
|
||||||
// https://en.wikipedia.org/wiki/Methods_of_computing_square_roots#Binary_numeral_system_.28base_2.29
|
// https://en.wikipedia.org/wiki/Methods_of_computing_square_roots#Binary_numeral_system_.28base_2.29
|
||||||
assert( "sqrt input should be non-negative" && num >= 0 );
|
assert("sqrt input should be non-negative" && num >= 0);
|
||||||
assert( "sqrt input should no exceed 16 bits" && num <= 0xFFFF );
|
assert("sqrt input should no exceed 16 bits" && num <= 0xFFFF);
|
||||||
int res = 0;
|
int res = 0;
|
||||||
int bit = 1 << 16;
|
int bit = 1 << 16;
|
||||||
while ( bit > num )
|
while (bit > num)
|
||||||
bit >>= 2;
|
bit >>= 2;
|
||||||
while ( bit != 0 ) {
|
while (bit != 0) {
|
||||||
if ( num >= res + bit ) {
|
if (num >= res + bit) {
|
||||||
num -= res + bit;
|
num -= res + bit;
|
||||||
res = ( res >> 1 ) + bit;
|
res = (res >> 1) + bit;
|
||||||
} else
|
} else
|
||||||
res >>= 1;
|
res >>= 1;
|
||||||
bit >>= 2;
|
bit >>= 2;
|
||||||
@@ -29,104 +55,133 @@ int iRgbSqrt( int num ) {
|
|||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
Rgb::Rgb( Hsv y ) {
|
Rgb::Rgb(const Hsv& y) {
|
||||||
// https://stackoverflow.com/questions/24152553/hsv-to-rgb-and-back-without-floating-point-math-in-python
|
// https://stackoverflow.com/questions/24152553/hsv-to-rgb-and-back-without-floating-point-math-in-python
|
||||||
// greyscale
|
// greyscale
|
||||||
if( y.s == 0 ) {
|
if (y.s == 0) {
|
||||||
r = g = b = y.v;
|
r = g = b = y.v;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const int region = y.h / 43;
|
const int region = y.h / 43;
|
||||||
const int remainder = ( y.h - ( region * 43 ) ) * 6;
|
const int remainder = (y.h - (region * 43)) * 6;
|
||||||
|
|
||||||
const int p = ( y.v * ( 255 - y.s ) ) >> 8;
|
const int p = (y.v * (255 - y.s)) >> 8;
|
||||||
const int q = ( y.v * ( 255 - ( ( y.s * remainder ) >> 8 ) ) ) >> 8;
|
const int q = (y.v * (255 - ((y.s * remainder) >> 8))) >> 8;
|
||||||
const int t = ( y.v * ( 255 - ( ( y.s * (255 -remainder ) ) >> 8 ) ) ) >> 8;
|
const int t = (y.v * (255 - ((y.s * (255 - remainder)) >> 8))) >> 8;
|
||||||
|
|
||||||
switch( region ) {
|
switch (region) {
|
||||||
case 0: r = y.v; g = t; b = p; break;
|
case 0:
|
||||||
case 1: r = q; g = y.v; b = p; break;
|
r = y.v;
|
||||||
case 2: r = p; g = y.v; b = t; break;
|
g = t;
|
||||||
case 3: r = p; g = q; b = y.v; break;
|
b = p;
|
||||||
case 4: r = t; g = p; b = y.v; break;
|
break;
|
||||||
case 5: r = y.v; g = p; b = q; break;
|
case 1:
|
||||||
default: __builtin_trap();
|
r = q;
|
||||||
|
g = y.v;
|
||||||
|
b = p;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
r = p;
|
||||||
|
g = y.v;
|
||||||
|
b = t;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
r = p;
|
||||||
|
g = q;
|
||||||
|
b = y.v;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
r = t;
|
||||||
|
g = p;
|
||||||
|
b = y.v;
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
r = y.v;
|
||||||
|
g = p;
|
||||||
|
b = q;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
__builtin_trap();
|
||||||
}
|
}
|
||||||
|
|
||||||
a = y.a;
|
a = y.a;
|
||||||
}
|
}
|
||||||
|
|
||||||
Rgb& Rgb::operator=( Hsv hsv ) {
|
Rgb& Rgb::operator=(const Hsv& hsv) {
|
||||||
Rgb r{ hsv };
|
Rgb r { hsv };
|
||||||
swap( r );
|
swap(r);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
Rgb Rgb::operator+( Rgb in ) const {
|
Rgb Rgb::operator+(const Rgb& in) const {
|
||||||
auto copy = *this;
|
auto copy = *this;
|
||||||
copy += in;
|
copy += in;
|
||||||
return copy;
|
return copy;
|
||||||
}
|
}
|
||||||
|
|
||||||
Rgb& Rgb::operator+=( Rgb in ) {
|
Rgb& Rgb::operator+=(const Rgb& in) {
|
||||||
unsigned int red = r + in.r;
|
unsigned int red = r + in.r;
|
||||||
r = ( red < 255 ) ? red : 255;
|
r = (red < 255) ? red : 255;
|
||||||
unsigned int green = g + in.g;
|
unsigned int green = g + in.g;
|
||||||
g = ( green < 255 ) ? green : 255;
|
g = (green < 255) ? green : 255;
|
||||||
unsigned int blue = b + in.b;
|
unsigned int blue = b + in.b;
|
||||||
b = ( blue < 255 ) ? blue : 255;
|
b = (blue < 255) ? blue : 255;
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
Rgb& Rgb::blend( Rgb in ) {
|
Rgb Rgb::operator-(const Rgb& in) const {
|
||||||
unsigned int inAlpha = in.a * ( 255 - a );
|
auto copy = *this;
|
||||||
|
copy -= in;
|
||||||
|
return copy;
|
||||||
|
}
|
||||||
|
|
||||||
|
Rgb& Rgb::operator-=(const Rgb& in) {
|
||||||
|
r = (in.r > r) ? 0 : r - in.r;
|
||||||
|
g = (in.g > g) ? 0 : g - in.g;
|
||||||
|
b = (in.b > b) ? 0 : b - in.b;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
Rgb& Rgb::blend(const Rgb& in) {
|
||||||
|
unsigned int inAlpha = in.a * (255 - a);
|
||||||
unsigned int alpha = a + inAlpha;
|
unsigned int alpha = a + inAlpha;
|
||||||
r = iRgbSqrt( ( ( r * r * a ) + ( in.r * in.r * inAlpha ) ) / alpha );
|
r = iRgbSqrt(((r * r * a) + (in.r * in.r * inAlpha)) / alpha);
|
||||||
g = iRgbSqrt( ( ( g * g * a ) + ( in.g * in.g * inAlpha ) ) / alpha );
|
g = iRgbSqrt(((g * g * a) + (in.g * in.g * inAlpha)) / alpha);
|
||||||
b = iRgbSqrt( ( ( b * b * a ) + ( in.b * in.b * inAlpha ) ) / alpha );
|
b = iRgbSqrt(((b * b * a) + (in.b * in.b * inAlpha)) / alpha);
|
||||||
a = alpha;
|
a = alpha;
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t IRAM_ATTR Rgb::getGrb( int idx ) {
|
Hsv::Hsv(const Rgb& r) {
|
||||||
switch ( idx ) {
|
int min = std::min(r.r, std::min(r.g, r.b));
|
||||||
case 0: return g;
|
int max = std::max(r.r, std::max(r.g, r.b));
|
||||||
case 1: return r;
|
|
||||||
case 2: return b;
|
|
||||||
}
|
|
||||||
__builtin_unreachable();
|
|
||||||
}
|
|
||||||
|
|
||||||
Hsv::Hsv( Rgb r ) {
|
|
||||||
int min = std::min( r.r, std::min( r.g, r.b ) );
|
|
||||||
int max = std::max( r.r, std::max( r.g, r.b ) );
|
|
||||||
int chroma = max - min;
|
int chroma = max - min;
|
||||||
|
|
||||||
v = max;
|
v = max;
|
||||||
if ( chroma == 0 ) {
|
if (chroma == 0) {
|
||||||
h = s = 0;
|
h = s = 0;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
s = up( chroma ) / max;
|
s = up(chroma) / max;
|
||||||
int hh;
|
int hh;
|
||||||
if ( max == r.r )
|
if (max == r.r)
|
||||||
hh = ( up( int( r.g ) - int( r.b ) ) ) / chroma / 6;
|
hh = (up(int(r.g) - int(r.b))) / chroma / 6;
|
||||||
else if ( max == r.g )
|
else if (max == r.g)
|
||||||
hh = 255 / 3 + ( up( int( r.b ) - int( r.r ) ) ) / chroma / 6;
|
hh = 255 / 3 + (up(int(r.b) - int(r.r))) / chroma / 6;
|
||||||
else
|
else
|
||||||
hh = 2 * 255 / 3 + ( up( int( r.r ) - int( r.g ) ) ) / chroma / 6;
|
hh = 2 * 255 / 3 + (up(int(r.r) - int(r.g))) / chroma / 6;
|
||||||
|
|
||||||
if ( hh < 0 )
|
if (hh < 0)
|
||||||
hh += 255;
|
hh += 255;
|
||||||
h = hh;
|
h = hh;
|
||||||
|
|
||||||
a = r.a;
|
a = r.a;
|
||||||
}
|
}
|
||||||
|
|
||||||
Hsv& Hsv::operator=( Rgb rgb ) {
|
Hsv& Hsv::operator=(const Rgb& rgb) {
|
||||||
Hsv h{ rgb };
|
Hsv h { rgb };
|
||||||
swap( h );
|
swap(h);
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,51 +1,90 @@
|
|||||||
|
/********************************************************************************
|
||||||
|
* https://github.com/RoboticsBrno/SmartLeds
|
||||||
|
*
|
||||||
|
* MIT License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2017 RoboticsBrno (RobotikaBrno)
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*******************************************************************************/
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#ifndef COLOR_H
|
|
||||||
#define COLOR_H
|
|
||||||
|
|
||||||
#include <cstdint>
|
|
||||||
#include "esp_attr.h"
|
#include "esp_attr.h"
|
||||||
|
#include <cstdint>
|
||||||
union Hsv;
|
union Hsv;
|
||||||
|
|
||||||
union Rgb {
|
union Rgb {
|
||||||
struct __attribute__ ((packed)) {
|
struct __attribute__((packed)) {
|
||||||
uint8_t r, g, b, a;
|
uint8_t g, r, b, a;
|
||||||
};
|
};
|
||||||
uint32_t value;
|
uint32_t value;
|
||||||
|
|
||||||
Rgb( uint8_t r = 0, uint8_t g = 0, uint8_t b = 0, uint8_t a = 255 ) : r( r ), g( g ), b( b ), a( a ) {}
|
Rgb(uint8_t r = 0, uint8_t g = 0, uint8_t b = 0, uint8_t a = 255)
|
||||||
Rgb( Hsv c );
|
: g(g)
|
||||||
Rgb& operator=( Rgb rgb ) { swap( rgb ); return *this; }
|
, r(r)
|
||||||
Rgb& operator=( Hsv hsv );
|
, b(b)
|
||||||
Rgb operator+( Rgb in ) const;
|
, a(a) {}
|
||||||
Rgb& operator+=( Rgb in );
|
Rgb(const Hsv& c);
|
||||||
bool operator==( Rgb in ) const { return in.value == value; }
|
Rgb(const Rgb&) = default;
|
||||||
Rgb& blend( Rgb in );
|
Rgb& operator=(const Rgb& rgb) {
|
||||||
void swap( Rgb& o ) { value = o.value; }
|
swap(rgb);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
Rgb& operator=(const Hsv& hsv);
|
||||||
|
Rgb operator+(const Rgb& in) const;
|
||||||
|
Rgb& operator+=(const Rgb& in);
|
||||||
|
Rgb operator-(const Rgb& in) const;
|
||||||
|
Rgb& operator-=(const Rgb& in);
|
||||||
|
bool operator==(const Rgb& in) const { return in.value == value; }
|
||||||
|
Rgb& blend(const Rgb& in);
|
||||||
|
void swap(const Rgb& o) { value = o.value; }
|
||||||
void linearize() {
|
void linearize() {
|
||||||
r = channelGamma(r);
|
r = channelGamma(r);
|
||||||
g = channelGamma(g);
|
g = channelGamma(g);
|
||||||
b = channelGamma(b);
|
b = channelGamma(b);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t IRAM_ATTR getGrb( int idx );
|
inline uint8_t IRAM_ATTR getGrb(int idx) {
|
||||||
|
switch (idx) {
|
||||||
void stretchChannels( uint8_t maxR, uint8_t maxG, uint8_t maxB ) {
|
case 0:
|
||||||
r = stretch( r, maxR );
|
return g;
|
||||||
g = stretch( g, maxG );
|
case 1:
|
||||||
b = stretch( b, maxB );
|
return r;
|
||||||
|
case 2:
|
||||||
|
return b;
|
||||||
|
}
|
||||||
|
__builtin_unreachable();
|
||||||
}
|
}
|
||||||
|
|
||||||
void stretchChannelsEvenly( uint8_t max ) {
|
void stretchChannels(uint8_t maxR, uint8_t maxG, uint8_t maxB) {
|
||||||
stretchChannels( max, max, max );
|
r = stretch(r, maxR);
|
||||||
|
g = stretch(g, maxG);
|
||||||
|
b = stretch(b, maxB);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void stretchChannelsEvenly(uint8_t max) { stretchChannels(max, max, max); }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
uint8_t stretch( int value, uint8_t max ) {
|
uint8_t stretch(int value, uint8_t max) { return (value * max) >> 8; }
|
||||||
return ( value * max ) >> 8;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t channelGamma( int channel ) {
|
uint8_t channelGamma(int channel) {
|
||||||
/* The optimal gamma correction is x^2.8. However, this is expensive to
|
/* The optimal gamma correction is x^2.8. However, this is expensive to
|
||||||
* compute. Therefore, we use x^3 for gamma correction. Also, we add a
|
* compute. Therefore, we use x^3 for gamma correction. Also, we add a
|
||||||
* bias as the WS2812 LEDs do not turn on for values less than 4. */
|
* bias as the WS2812 LEDs do not turn on for values less than 4. */
|
||||||
@@ -53,22 +92,27 @@ private:
|
|||||||
return channel;
|
return channel;
|
||||||
channel = channel * channel * channel * 251;
|
channel = channel * channel * channel * 251;
|
||||||
channel >>= 24;
|
channel >>= 24;
|
||||||
return static_cast< uint8_t >( 4 + channel );
|
return static_cast<uint8_t>(4 + channel);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
union Hsv {
|
union Hsv {
|
||||||
struct __attribute__ ((packed)) {
|
struct __attribute__((packed)) {
|
||||||
uint8_t h, s, v, a;
|
uint8_t h, s, v, a;
|
||||||
};
|
};
|
||||||
uint32_t value;
|
uint32_t value;
|
||||||
|
|
||||||
Hsv( uint8_t h, uint8_t s = 0, uint8_t v = 0, uint8_t a = 255 ) : h( h ), s( s ), v( v ), a( a ) {}
|
Hsv(uint8_t h, uint8_t s = 0, uint8_t v = 0, uint8_t a = 255)
|
||||||
Hsv( Rgb r );
|
: h(h)
|
||||||
Hsv& operator=( Hsv h ) { swap( h ); return *this; }
|
, s(s)
|
||||||
Hsv& operator=( Rgb rgb );
|
, v(v)
|
||||||
bool operator==( Hsv in ) const { return in.value == value; }
|
, a(a) {}
|
||||||
void swap( Hsv& o ) { value = o.value; }
|
Hsv(const Rgb& r);
|
||||||
|
Hsv& operator=(const Hsv& h) {
|
||||||
|
swap(h);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
Hsv& operator=(const Rgb& rgb);
|
||||||
|
bool operator==(const Hsv& in) const { return in.value == value; }
|
||||||
|
void swap(const Hsv& o) { value = o.value; }
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif //COLOR_H
|
|
||||||
|
|||||||
60
code/components/jomjol_controlGPIO/RmtDriver.h
Normal file
60
code/components/jomjol_controlGPIO/RmtDriver.h
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
/********************************************************************************
|
||||||
|
* https://github.com/RoboticsBrno/SmartLeds
|
||||||
|
*
|
||||||
|
* MIT License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2017 RoboticsBrno (RobotikaBrno)
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*******************************************************************************/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <esp_system.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
#if defined(ESP_IDF_VERSION)
|
||||||
|
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 0, 0)
|
||||||
|
#define SMARTLEDS_NEW_RMT_DRIVER 1
|
||||||
|
#else
|
||||||
|
#define SMARTLEDS_NEW_RMT_DRIVER 0
|
||||||
|
#endif
|
||||||
|
#else
|
||||||
|
#define SMARTLEDS_NEW_RMT_DRIVER 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
|
struct TimingParams {
|
||||||
|
uint32_t T0H;
|
||||||
|
uint32_t T1H;
|
||||||
|
uint32_t T0L;
|
||||||
|
uint32_t T1L;
|
||||||
|
uint32_t TRS;
|
||||||
|
};
|
||||||
|
|
||||||
|
using LedType = TimingParams;
|
||||||
|
|
||||||
|
} // namespace detail
|
||||||
|
|
||||||
|
#if SMARTLEDS_NEW_RMT_DRIVER
|
||||||
|
#include "RmtDriver5.h"
|
||||||
|
#else
|
||||||
|
#include "RmtDriver4.h"
|
||||||
|
#endif
|
||||||
143
code/components/jomjol_controlGPIO/RmtDriver4.cpp
Normal file
143
code/components/jomjol_controlGPIO/RmtDriver4.cpp
Normal file
@@ -0,0 +1,143 @@
|
|||||||
|
/********************************************************************************
|
||||||
|
* https://github.com/RoboticsBrno/SmartLeds
|
||||||
|
*
|
||||||
|
* MIT License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2017 RoboticsBrno (RobotikaBrno)
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*******************************************************************************/
|
||||||
|
|
||||||
|
#include "RmtDriver4.h"
|
||||||
|
|
||||||
|
#if !SMARTLEDS_NEW_RMT_DRIVER
|
||||||
|
#include "SmartLeds.h"
|
||||||
|
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
|
// 8 still seems to work, but timings become marginal
|
||||||
|
static const int DIVIDER = 4;
|
||||||
|
// minimum time of a single RMT duration based on clock ns
|
||||||
|
static const double RMT_DURATION_NS = 12.5;
|
||||||
|
|
||||||
|
RmtDriver::RmtDriver(const LedType& timing, int count, int pin, int channel_num, SemaphoreHandle_t finishedFlag)
|
||||||
|
: _timing(timing)
|
||||||
|
, _count(count)
|
||||||
|
, _pin((gpio_num_t)pin)
|
||||||
|
, _finishedFlag(finishedFlag)
|
||||||
|
, _channel((rmt_channel_t)channel_num) {
|
||||||
|
_bitToRmt[0].level0 = 1;
|
||||||
|
_bitToRmt[0].level1 = 0;
|
||||||
|
_bitToRmt[0].duration0 = _timing.T0H / (RMT_DURATION_NS * DIVIDER);
|
||||||
|
_bitToRmt[0].duration1 = _timing.T0L / (RMT_DURATION_NS * DIVIDER);
|
||||||
|
|
||||||
|
_bitToRmt[1].level0 = 1;
|
||||||
|
_bitToRmt[1].level1 = 0;
|
||||||
|
_bitToRmt[1].duration0 = _timing.T1H / (RMT_DURATION_NS * DIVIDER);
|
||||||
|
_bitToRmt[1].duration1 = _timing.T1L / (RMT_DURATION_NS * DIVIDER);
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t RmtDriver::init() {
|
||||||
|
rmt_config_t config = RMT_DEFAULT_CONFIG_TX(_pin, _channel);
|
||||||
|
config.rmt_mode = RMT_MODE_TX;
|
||||||
|
config.clk_div = DIVIDER;
|
||||||
|
config.mem_block_num = 1;
|
||||||
|
|
||||||
|
return rmt_config(&config);
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t RmtDriver::registerIsr(bool isFirstRegisteredChannel) {
|
||||||
|
auto err = rmt_driver_install(_channel, 0,
|
||||||
|
#if defined(CONFIG_RMT_ISR_IRAM_SAFE)
|
||||||
|
ESP_INTR_FLAG_IRAM
|
||||||
|
#else
|
||||||
|
0
|
||||||
|
#endif
|
||||||
|
);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isFirstRegisteredChannel) {
|
||||||
|
rmt_register_tx_end_callback(txEndCallback, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
err = rmt_translator_init(_channel, translateSample);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
return rmt_translator_set_context(_channel, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t RmtDriver::unregisterIsr() { return rmt_driver_uninstall(_channel); }
|
||||||
|
|
||||||
|
void IRAM_ATTR RmtDriver::txEndCallback(rmt_channel_t channel, void* arg) {
|
||||||
|
xSemaphoreGiveFromISR(SmartLed::ledForChannel(channel)->_finishedFlag, nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void IRAM_ATTR RmtDriver::translateSample(const void* src, rmt_item32_t* dest, size_t src_size,
|
||||||
|
size_t wanted_rmt_items_num, size_t* out_consumed_src_bytes, size_t* out_used_rmt_items) {
|
||||||
|
RmtDriver* self;
|
||||||
|
ESP_ERROR_CHECK(rmt_translator_get_context(out_used_rmt_items, (void**)&self));
|
||||||
|
|
||||||
|
const auto& _bitToRmt = self->_bitToRmt;
|
||||||
|
const auto src_offset = self->_translatorSourceOffset;
|
||||||
|
|
||||||
|
auto* src_components = (const uint8_t*)src;
|
||||||
|
size_t consumed_src_bytes = 0;
|
||||||
|
size_t used_rmt_items = 0;
|
||||||
|
|
||||||
|
while (consumed_src_bytes < src_size && used_rmt_items + 7 < wanted_rmt_items_num) {
|
||||||
|
uint8_t val = *src_components;
|
||||||
|
|
||||||
|
// each bit, from highest to lowest
|
||||||
|
for (uint8_t j = 0; j != 8; j++, val <<= 1) {
|
||||||
|
dest->val = _bitToRmt[val >> 7].val;
|
||||||
|
++dest;
|
||||||
|
}
|
||||||
|
|
||||||
|
used_rmt_items += 8;
|
||||||
|
++src_components;
|
||||||
|
++consumed_src_bytes;
|
||||||
|
|
||||||
|
// skip alpha byte
|
||||||
|
if (((src_offset + consumed_src_bytes) % 4) == 3) {
|
||||||
|
++src_components;
|
||||||
|
++consumed_src_bytes;
|
||||||
|
|
||||||
|
// TRST delay after last pixel in strip
|
||||||
|
if (consumed_src_bytes == src_size) {
|
||||||
|
(dest - 1)->duration1 = self->_timing.TRS / (detail::RMT_DURATION_NS * detail::DIVIDER);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
self->_translatorSourceOffset = src_offset + consumed_src_bytes;
|
||||||
|
*out_consumed_src_bytes = consumed_src_bytes;
|
||||||
|
*out_used_rmt_items = used_rmt_items;
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t RmtDriver::transmit(const Rgb* buffer) {
|
||||||
|
static_assert(sizeof(Rgb) == 4); // The translator code above assumes RGB is 4 bytes
|
||||||
|
|
||||||
|
_translatorSourceOffset = 0;
|
||||||
|
return rmt_write_sample(_channel, (const uint8_t*)buffer, _count * 4, false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
#endif // !SMARTLEDS_NEW_RMT_DRIVER
|
||||||
68
code/components/jomjol_controlGPIO/RmtDriver4.h
Normal file
68
code/components/jomjol_controlGPIO/RmtDriver4.h
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
/********************************************************************************
|
||||||
|
* https://github.com/RoboticsBrno/SmartLeds
|
||||||
|
*
|
||||||
|
* MIT License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2017 RoboticsBrno (RobotikaBrno)
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*******************************************************************************/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "RmtDriver.h"
|
||||||
|
|
||||||
|
#if !SMARTLEDS_NEW_RMT_DRIVER
|
||||||
|
#include "Color.h"
|
||||||
|
#include <driver/rmt.h>
|
||||||
|
#include <freertos/FreeRTOS.h>
|
||||||
|
#include <freertos/semphr.h>
|
||||||
|
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
|
constexpr const int CHANNEL_COUNT = RMT_CHANNEL_MAX;
|
||||||
|
|
||||||
|
class RmtDriver {
|
||||||
|
public:
|
||||||
|
RmtDriver(const LedType& timing, int count, int pin, int channel_num, SemaphoreHandle_t finishedFlag);
|
||||||
|
RmtDriver(const RmtDriver&) = delete;
|
||||||
|
|
||||||
|
esp_err_t init();
|
||||||
|
esp_err_t registerIsr(bool isFirstRegisteredChannel);
|
||||||
|
esp_err_t unregisterIsr();
|
||||||
|
esp_err_t transmit(const Rgb* buffer);
|
||||||
|
|
||||||
|
private:
|
||||||
|
static void IRAM_ATTR txEndCallback(rmt_channel_t channel, void* arg);
|
||||||
|
|
||||||
|
static void IRAM_ATTR translateSample(const void* src, rmt_item32_t* dest, size_t src_size,
|
||||||
|
size_t wanted_rmt_items_num, size_t* out_consumed_src_bytes, size_t* out_used_rmt_items);
|
||||||
|
|
||||||
|
const LedType& _timing;
|
||||||
|
int _count;
|
||||||
|
gpio_num_t _pin;
|
||||||
|
SemaphoreHandle_t _finishedFlag;
|
||||||
|
|
||||||
|
rmt_channel_t _channel;
|
||||||
|
rmt_item32_t _bitToRmt[2];
|
||||||
|
size_t _translatorSourceOffset;
|
||||||
|
};
|
||||||
|
|
||||||
|
};
|
||||||
|
#endif // !SMARTLEDS_NEW_RMT_DRIVER
|
||||||
202
code/components/jomjol_controlGPIO/RmtDriver5.cpp
Normal file
202
code/components/jomjol_controlGPIO/RmtDriver5.cpp
Normal file
@@ -0,0 +1,202 @@
|
|||||||
|
/********************************************************************************
|
||||||
|
* https://github.com/RoboticsBrno/SmartLeds
|
||||||
|
*
|
||||||
|
* MIT License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2017 RoboticsBrno (RobotikaBrno)
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*******************************************************************************/
|
||||||
|
|
||||||
|
#include "RmtDriver5.h"
|
||||||
|
|
||||||
|
#if SMARTLEDS_NEW_RMT_DRIVER
|
||||||
|
#include <cstddef>
|
||||||
|
|
||||||
|
#include "SmartLeds.h"
|
||||||
|
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
|
static constexpr const uint32_t RMT_RESOLUTION_HZ = 20 * 1000 * 1000; // 20 MHz
|
||||||
|
static constexpr const uint32_t RMT_NS_PER_TICK = 1000000000LLU / RMT_RESOLUTION_HZ;
|
||||||
|
|
||||||
|
static RmtEncoderWrapper* IRAM_ATTR encSelf(rmt_encoder_t* encoder) {
|
||||||
|
return (RmtEncoderWrapper*)(((intptr_t)encoder) - offsetof(RmtEncoderWrapper, base));
|
||||||
|
}
|
||||||
|
|
||||||
|
static size_t IRAM_ATTR encEncode(rmt_encoder_t* encoder, rmt_channel_handle_t tx_channel, const void* primary_data,
|
||||||
|
size_t data_size, rmt_encode_state_t* ret_state) {
|
||||||
|
auto* self = encSelf(encoder);
|
||||||
|
|
||||||
|
// Delay after last pixel
|
||||||
|
if ((self->last_state & RMT_ENCODING_COMPLETE) && self->frame_idx == data_size) {
|
||||||
|
*ret_state = (rmt_encode_state_t)0;
|
||||||
|
return self->copy_encoder->encode(
|
||||||
|
self->copy_encoder, tx_channel, (const void*)&self->reset_code, sizeof(self->reset_code), ret_state);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (self->last_state & RMT_ENCODING_COMPLETE) {
|
||||||
|
Rgb* pixel = ((Rgb*)primary_data) + self->frame_idx;
|
||||||
|
self->buffer_len = sizeof(self->buffer);
|
||||||
|
for (size_t i = 0; i < sizeof(self->buffer); ++i) {
|
||||||
|
self->buffer[i] = pixel->getGrb(self->component_idx);
|
||||||
|
if (++self->component_idx == 3) {
|
||||||
|
self->component_idx = 0;
|
||||||
|
if (++self->frame_idx == data_size) {
|
||||||
|
self->buffer_len = i + 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
++pixel;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
self->last_state = (rmt_encode_state_t)0;
|
||||||
|
auto encoded_symbols = self->bytes_encoder->encode(
|
||||||
|
self->bytes_encoder, tx_channel, (const void*)&self->buffer, self->buffer_len, &self->last_state);
|
||||||
|
if (self->last_state & RMT_ENCODING_MEM_FULL) {
|
||||||
|
*ret_state = RMT_ENCODING_MEM_FULL;
|
||||||
|
} else {
|
||||||
|
*ret_state = (rmt_encode_state_t)0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return encoded_symbols;
|
||||||
|
}
|
||||||
|
|
||||||
|
static esp_err_t encReset(rmt_encoder_t* encoder) {
|
||||||
|
auto* self = encSelf(encoder);
|
||||||
|
rmt_encoder_reset(self->bytes_encoder);
|
||||||
|
rmt_encoder_reset(self->copy_encoder);
|
||||||
|
self->last_state = RMT_ENCODING_COMPLETE;
|
||||||
|
self->frame_idx = 0;
|
||||||
|
self->component_idx = 0;
|
||||||
|
return ESP_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static esp_err_t encDelete(rmt_encoder_t* encoder) {
|
||||||
|
auto* self = encSelf(encoder);
|
||||||
|
rmt_del_encoder(self->bytes_encoder);
|
||||||
|
rmt_del_encoder(self->copy_encoder);
|
||||||
|
return ESP_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
RmtDriver::RmtDriver(const LedType& timing, int count, int pin, int channel_num, SemaphoreHandle_t finishedFlag)
|
||||||
|
: _timing(timing)
|
||||||
|
, _count(count)
|
||||||
|
, _pin(pin)
|
||||||
|
, _finishedFlag(finishedFlag)
|
||||||
|
, _channel(nullptr)
|
||||||
|
, _encoder {} {}
|
||||||
|
|
||||||
|
esp_err_t RmtDriver::init() {
|
||||||
|
_encoder.base.encode = encEncode;
|
||||||
|
_encoder.base.reset = encReset;
|
||||||
|
_encoder.base.del = encDelete;
|
||||||
|
|
||||||
|
_encoder.reset_code.duration0 = _timing.TRS / RMT_NS_PER_TICK;
|
||||||
|
|
||||||
|
rmt_bytes_encoder_config_t bytes_cfg = {
|
||||||
|
.bit0 = {
|
||||||
|
.duration0 = uint16_t(_timing.T0H / RMT_NS_PER_TICK),
|
||||||
|
.level0 = 1,
|
||||||
|
.duration1 = uint16_t(_timing.T0L / RMT_NS_PER_TICK),
|
||||||
|
.level1 = 0,
|
||||||
|
},
|
||||||
|
.bit1 = {
|
||||||
|
.duration0 = uint16_t(_timing.T1H / RMT_NS_PER_TICK),
|
||||||
|
.level0 = 1,
|
||||||
|
.duration1 = uint16_t(_timing.T1L / RMT_NS_PER_TICK),
|
||||||
|
.level1 = 0,
|
||||||
|
},
|
||||||
|
.flags = {
|
||||||
|
.msb_first = 1,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
auto err = rmt_new_bytes_encoder(&bytes_cfg, &_encoder.bytes_encoder);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
rmt_copy_encoder_config_t copy_cfg = {};
|
||||||
|
err = rmt_new_copy_encoder(©_cfg, &_encoder.copy_encoder);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
// The config must be in registerIsr, because rmt_new_tx_channel
|
||||||
|
// registers the ISR
|
||||||
|
return ESP_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t RmtDriver::registerIsr(bool isFirstRegisteredChannel) {
|
||||||
|
rmt_tx_channel_config_t conf = {
|
||||||
|
.gpio_num = (gpio_num_t)_pin,
|
||||||
|
.clk_src = RMT_CLK_SRC_DEFAULT, //.clk_src = RMT_CLK_SRC_APB,
|
||||||
|
.resolution_hz = RMT_RESOLUTION_HZ,
|
||||||
|
.mem_block_symbols = SOC_RMT_MEM_WORDS_PER_CHANNEL,
|
||||||
|
.trans_queue_depth = 1,
|
||||||
|
.flags = {},
|
||||||
|
};
|
||||||
|
|
||||||
|
auto err = rmt_new_tx_channel(&conf, &_channel);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
rmt_tx_event_callbacks_t callbacks_cfg = {};
|
||||||
|
callbacks_cfg.on_trans_done = txDoneCallback;
|
||||||
|
|
||||||
|
err = rmt_tx_register_event_callbacks(_channel, &callbacks_cfg, this);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rmt_enable(_channel);
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t RmtDriver::unregisterIsr() {
|
||||||
|
auto err = rmt_del_encoder(&_encoder.base);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
err = rmt_disable(_channel);
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
return rmt_del_channel(_channel);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IRAM_ATTR RmtDriver::txDoneCallback(
|
||||||
|
rmt_channel_handle_t tx_chan, const rmt_tx_done_event_data_t* edata, void* user_ctx) {
|
||||||
|
auto* self = (RmtDriver*)user_ctx;
|
||||||
|
auto taskWoken = pdTRUE;
|
||||||
|
xSemaphoreGiveFromISR(self->_finishedFlag, &taskWoken);
|
||||||
|
return taskWoken == pdTRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t RmtDriver::transmit(const Rgb* buffer) {
|
||||||
|
rmt_encoder_reset(&_encoder.base);
|
||||||
|
rmt_transmit_config_t cfg = {};
|
||||||
|
return rmt_transmit(_channel, &_encoder.base, buffer, _count, &cfg);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
#endif // !SMARTLEDS_NEW_RMT_DRIVER
|
||||||
91
code/components/jomjol_controlGPIO/RmtDriver5.h
Normal file
91
code/components/jomjol_controlGPIO/RmtDriver5.h
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
/********************************************************************************
|
||||||
|
* https://github.com/RoboticsBrno/SmartLeds
|
||||||
|
*
|
||||||
|
* MIT License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2017 RoboticsBrno (RobotikaBrno)
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*******************************************************************************/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "RmtDriver.h"
|
||||||
|
|
||||||
|
#if SMARTLEDS_NEW_RMT_DRIVER
|
||||||
|
#include <driver/rmt_tx.h>
|
||||||
|
#include <freertos/FreeRTOS.h>
|
||||||
|
#include <freertos/semphr.h>
|
||||||
|
#include <type_traits>
|
||||||
|
|
||||||
|
#include "Color.h"
|
||||||
|
|
||||||
|
#if !defined(CONFIG_RMT_ISR_IRAM_SAFE) && !defined(SMARTLEDS_DISABLE_IRAM_WARNING)
|
||||||
|
#warning "Please enable CONFIG_RMT_ISR_IRAM_SAFE IDF option." \
|
||||||
|
"without it, the IDF driver is not able to supply data fast enough."
|
||||||
|
#endif
|
||||||
|
|
||||||
|
namespace detail {
|
||||||
|
|
||||||
|
constexpr const int CHANNEL_COUNT = SOC_RMT_GROUPS * SOC_RMT_CHANNELS_PER_GROUP;
|
||||||
|
|
||||||
|
class RmtDriver;
|
||||||
|
|
||||||
|
// This is ridiculous
|
||||||
|
struct RmtEncoderWrapper {
|
||||||
|
struct rmt_encoder_t base;
|
||||||
|
struct rmt_encoder_t* bytes_encoder;
|
||||||
|
struct rmt_encoder_t* copy_encoder;
|
||||||
|
RmtDriver* driver;
|
||||||
|
rmt_symbol_word_t reset_code;
|
||||||
|
|
||||||
|
uint8_t buffer[SOC_RMT_MEM_WORDS_PER_CHANNEL / 8];
|
||||||
|
rmt_encode_state_t last_state;
|
||||||
|
size_t frame_idx;
|
||||||
|
uint8_t component_idx;
|
||||||
|
uint8_t buffer_len;
|
||||||
|
};
|
||||||
|
|
||||||
|
static_assert(std::is_standard_layout<RmtEncoderWrapper>::value == true);
|
||||||
|
|
||||||
|
class RmtDriver {
|
||||||
|
public:
|
||||||
|
RmtDriver(const LedType& timing, int count, int pin, int channel_num, SemaphoreHandle_t finishedFlag);
|
||||||
|
RmtDriver(const RmtDriver&) = delete;
|
||||||
|
|
||||||
|
esp_err_t init();
|
||||||
|
esp_err_t registerIsr(bool isFirstRegisteredChannel);
|
||||||
|
esp_err_t unregisterIsr();
|
||||||
|
esp_err_t transmit(const Rgb* buffer);
|
||||||
|
|
||||||
|
private:
|
||||||
|
static bool IRAM_ATTR txDoneCallback(
|
||||||
|
rmt_channel_handle_t tx_chan, const rmt_tx_done_event_data_t* edata, void* user_ctx);
|
||||||
|
|
||||||
|
const LedType& _timing;
|
||||||
|
int _count;
|
||||||
|
int _pin;
|
||||||
|
SemaphoreHandle_t _finishedFlag;
|
||||||
|
|
||||||
|
rmt_channel_handle_t _channel;
|
||||||
|
RmtEncoderWrapper _encoder;
|
||||||
|
};
|
||||||
|
|
||||||
|
};
|
||||||
|
#endif // !SMARTLEDS_NEW_RMT_DRIVER
|
||||||
@@ -1,63 +1,35 @@
|
|||||||
|
/********************************************************************************
|
||||||
|
* https://github.com/RoboticsBrno/SmartLeds
|
||||||
|
*
|
||||||
|
* MIT License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2017 RoboticsBrno (RobotikaBrno)
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*******************************************************************************/
|
||||||
|
|
||||||
#include "SmartLeds.h"
|
#include "SmartLeds.h"
|
||||||
|
|
||||||
IsrCore SmartLed::_interruptCore = CoreCurrent;
|
IsrCore SmartLed::_interruptCore = CoreCurrent;
|
||||||
intr_handle_t SmartLed::_interruptHandle = NULL;
|
|
||||||
|
|
||||||
SmartLed*& IRAM_ATTR SmartLed::ledForChannel( int channel ) {
|
SmartLed*& IRAM_ATTR SmartLed::ledForChannel(int channel) {
|
||||||
static SmartLed* table[8] = { nullptr };
|
static SmartLed* table[detail::CHANNEL_COUNT] = {};
|
||||||
assert( channel < 8 );
|
assert(channel < detail::CHANNEL_COUNT);
|
||||||
return table[ channel ];
|
return table[channel];
|
||||||
}
|
|
||||||
|
|
||||||
void IRAM_ATTR SmartLed::interruptHandler(void*) {
|
|
||||||
for (int channel = 0; channel != 8; channel++) {
|
|
||||||
auto self = ledForChannel( channel );
|
|
||||||
|
|
||||||
if ( RMT.int_st.val & (1 << (24 + channel ) ) ) { // tx_thr_event
|
|
||||||
if ( self )
|
|
||||||
self->copyRmtHalfBlock();
|
|
||||||
RMT.int_clr.val |= 1 << ( 24 + channel );
|
|
||||||
} else if ( RMT.int_st.val & ( 1 << (3 * channel ) ) ) { // tx_end
|
|
||||||
if ( self )
|
|
||||||
xSemaphoreGiveFromISR( self->_finishedFlag, nullptr );
|
|
||||||
RMT.int_clr.val |= 1 << ( 3 * channel );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void IRAM_ATTR SmartLed::copyRmtHalfBlock() {
|
|
||||||
int offset = detail::MAX_PULSES * _halfIdx;
|
|
||||||
_halfIdx = !_halfIdx;
|
|
||||||
int len = 3 - _componentPosition + 3 * ( _count - 1 );
|
|
||||||
len = std::min( len, detail::MAX_PULSES / 8 );
|
|
||||||
|
|
||||||
if ( !len ) {
|
|
||||||
for ( int i = 0; i < detail::MAX_PULSES; i++) {
|
|
||||||
RMTMEM.chan[ _channel].data32[i + offset ].val = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
int i;
|
|
||||||
for ( i = 0; i != len && _pixelPosition != _count; i++ ) {
|
|
||||||
uint8_t val = _buffer[ _pixelPosition ].getGrb( _componentPosition );
|
|
||||||
for ( int j = 0; j != 8; j++, val <<= 1 ) {
|
|
||||||
int bit = val >> 7;
|
|
||||||
int idx = i * 8 + offset + j;
|
|
||||||
RMTMEM.chan[ _channel ].data32[ idx ].val = _bitToRmt[ bit & 0x01 ].value;
|
|
||||||
}
|
|
||||||
if ( _pixelPosition == _count - 1 && _componentPosition == 2 ) {
|
|
||||||
RMTMEM.chan[ _channel ].data32[ i * 8 + offset + 7 ].duration1 =
|
|
||||||
_timing.TRS / ( detail::RMT_DURATION_NS * detail::DIVIDER );
|
|
||||||
}
|
|
||||||
|
|
||||||
_componentPosition++;
|
|
||||||
if ( _componentPosition == 3 ) {
|
|
||||||
_componentPosition = 0;
|
|
||||||
_pixelPosition++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for ( i *= 8; i != detail::MAX_PULSES; i++ ) {
|
|
||||||
RMTMEM.chan[ _channel ].data32[ i + offset ].val = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,30 @@
|
|||||||
#pragma once
|
/********************************************************************************
|
||||||
|
* https://github.com/RoboticsBrno/SmartLeds
|
||||||
|
*
|
||||||
|
* MIT License
|
||||||
|
*
|
||||||
|
* Copyright (c) 2017 RoboticsBrno (RobotikaBrno)
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in all
|
||||||
|
* copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
* SOFTWARE.
|
||||||
|
*******************************************************************************/
|
||||||
|
|
||||||
#ifndef SMARTLEDS_H
|
#pragma once
|
||||||
#define SMARTLEDS_H
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* A C++ driver for the WS2812 LEDs using the RMT peripheral on the ESP32.
|
* A C++ driver for the WS2812 LEDs using the RMT peripheral on the ESP32.
|
||||||
@@ -31,270 +54,196 @@
|
|||||||
* THE SOFTWARE.
|
* THE SOFTWARE.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <memory>
|
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
#if defined ( ARDUINO )
|
#include <driver/gpio.h>
|
||||||
extern "C" { // ...someone forgot to put in the includes...
|
#include <driver/spi_master.h>
|
||||||
#include "esp32-hal.h"
|
#include <esp_intr_alloc.h>
|
||||||
#include "esp_intr_alloc.h"
|
#include <esp_ipc.h>
|
||||||
#include "esp_ipc.h"
|
#include <freertos/FreeRTOS.h>
|
||||||
#include "driver/gpio.h"
|
#include <freertos/semphr.h>
|
||||||
#include "driver/periph_ctrl.h"
|
|
||||||
#include "freertos/semphr.h"
|
|
||||||
#include "soc/rmt_struct.h"
|
|
||||||
#include <driver/spi_master.h>
|
|
||||||
#include "esp_idf_version.h"
|
|
||||||
#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL( 4, 0, 0 )
|
|
||||||
#include "soc/dport_reg.h"
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
#elif defined ( ESP_PLATFORM )
|
|
||||||
extern "C" { // ...someone forgot to put in the includes...
|
|
||||||
#include <esp_intr_alloc.h>
|
|
||||||
#include <esp_ipc.h>
|
|
||||||
#include <driver/gpio.h>
|
|
||||||
#include <freertos/FreeRTOS.h>
|
|
||||||
#include <freertos/semphr.h>
|
|
||||||
#include <soc/dport_reg.h>
|
|
||||||
#include <soc/gpio_sig_map.h>
|
|
||||||
#include <soc/rmt_struct.h>
|
|
||||||
#include <driver/spi_master.h>
|
|
||||||
}
|
|
||||||
#include <stdio.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "Color.h"
|
#include "Color.h"
|
||||||
|
|
||||||
namespace detail {
|
#include "RmtDriver.h"
|
||||||
|
|
||||||
struct TimingParams {
|
|
||||||
uint32_t T0H;
|
|
||||||
uint32_t T1H;
|
|
||||||
uint32_t T0L;
|
|
||||||
uint32_t T1L;
|
|
||||||
uint32_t TRS;
|
|
||||||
};
|
|
||||||
|
|
||||||
union RmtPulsePair {
|
|
||||||
struct {
|
|
||||||
int duration0:15;
|
|
||||||
int level0:1;
|
|
||||||
int duration1:15;
|
|
||||||
int level1:1;
|
|
||||||
};
|
|
||||||
uint32_t value;
|
|
||||||
};
|
|
||||||
|
|
||||||
static const int DIVIDER = 4; // 8 still seems to work, but timings become marginal
|
|
||||||
static const int MAX_PULSES = 32; // A channel has a 64 "pulse" buffer - we use half per pass
|
|
||||||
static const double RMT_DURATION_NS = 12.5; // minimum time of a single RMT duration based on clock ns
|
|
||||||
|
|
||||||
} // namespace detail
|
|
||||||
|
|
||||||
using LedType = detail::TimingParams;
|
using LedType = detail::TimingParams;
|
||||||
|
|
||||||
|
// Times are in nanoseconds,
|
||||||
|
// The RMT driver runs at 20MHz, so minimal representable time is 50 nanoseconds
|
||||||
static const LedType LED_WS2812 = { 350, 700, 800, 600, 50000 };
|
static const LedType LED_WS2812 = { 350, 700, 800, 600, 50000 };
|
||||||
static const LedType LED_WS2812B = { 400, 850, 850, 400, 50100 };
|
// longer reset time because https://blog.adafruit.com/2017/05/03/psa-the-ws2812b-rgb-led-has-been-revised-will-require-code-tweak/
|
||||||
|
static const LedType LED_WS2812B = { 400, 800, 850, 450, 300000 }; // universal
|
||||||
|
static const LedType LED_WS2812B_NEWVARIANT = { 200, 750, 750, 200, 300000 };
|
||||||
|
static const LedType LED_WS2812B_OLDVARIANT = { 400, 800, 850, 450, 50000 };
|
||||||
|
// This is timing from datasheet, but does not seem to actually work - try LED_WS2812B
|
||||||
|
static const LedType LED_WS2812C = { 250, 550, 550, 250, 280000 };
|
||||||
static const LedType LED_SK6812 = { 300, 600, 900, 600, 80000 };
|
static const LedType LED_SK6812 = { 300, 600, 900, 600, 80000 };
|
||||||
static const LedType LED_WS2813 = { 350, 800, 350, 350, 300000 };
|
static const LedType LED_WS2813 = { 350, 800, 350, 350, 300000 };
|
||||||
|
|
||||||
|
// Single buffer == can't touch the Rgbs between show() and wait()
|
||||||
enum BufferType { SingleBuffer = 0, DoubleBuffer };
|
enum BufferType { SingleBuffer = 0, DoubleBuffer };
|
||||||
|
|
||||||
enum IsrCore { CoreFirst = 0, CoreSecond = 1, CoreCurrent = 2};
|
enum IsrCore { CoreFirst = 0, CoreSecond = 1, CoreCurrent = 2 };
|
||||||
|
|
||||||
class SmartLed {
|
class SmartLed {
|
||||||
public:
|
public:
|
||||||
|
friend class detail::RmtDriver;
|
||||||
|
|
||||||
// The RMT interrupt must not run on the same core as WiFi interrupts, otherwise SmartLeds
|
// The RMT interrupt must not run on the same core as WiFi interrupts, otherwise SmartLeds
|
||||||
// can't fill the RMT buffer fast enough, resulting in rendering artifacts.
|
// can't fill the RMT buffer fast enough, resulting in rendering artifacts.
|
||||||
// Usually, that means you have to set isrCore == CoreSecond.
|
// Usually, that means you have to set isrCore == CoreSecond.
|
||||||
//
|
//
|
||||||
// If you use anything other than CoreCurrent, the FreeRTOS scheduler MUST be already running,
|
// If you use anything other than CoreCurrent, the FreeRTOS scheduler MUST be already running,
|
||||||
// so you can't use it if you define SmartLed as global variable.
|
// so you can't use it if you define SmartLed as global variable.
|
||||||
SmartLed( const LedType& type, int count, int pin, int channel = 0, BufferType doubleBuffer = SingleBuffer, IsrCore isrCore = CoreCurrent)
|
//
|
||||||
: _timing( type ),
|
// Does nothing on chips that only have one core.
|
||||||
_channel( channel ),
|
SmartLed(const LedType& type, int count, int pin, int channel = 0, BufferType doubleBuffer = DoubleBuffer,
|
||||||
_count( count ),
|
IsrCore isrCore = CoreCurrent)
|
||||||
_firstBuffer( new Rgb[ count ] ),
|
: _finishedFlag(xSemaphoreCreateBinary())
|
||||||
_secondBuffer( doubleBuffer ? new Rgb[ count ] : nullptr ),
|
, _driver(type, count, pin, channel, _finishedFlag)
|
||||||
_finishedFlag( xSemaphoreCreateBinary() )
|
, _channel(channel)
|
||||||
{
|
, _count(count)
|
||||||
assert( channel >= 0 && channel < 8 );
|
, _firstBuffer(new Rgb[count])
|
||||||
assert( ledForChannel( channel ) == nullptr );
|
, _secondBuffer(doubleBuffer ? new Rgb[count] : nullptr) {
|
||||||
|
assert(channel >= 0 && channel < detail::CHANNEL_COUNT);
|
||||||
|
assert(ledForChannel(channel) == nullptr);
|
||||||
|
|
||||||
xSemaphoreGive( _finishedFlag );
|
xSemaphoreGive(_finishedFlag);
|
||||||
|
|
||||||
DPORT_SET_PERI_REG_MASK( DPORT_PERIP_CLK_EN_REG, DPORT_RMT_CLK_EN );
|
_driver.init();
|
||||||
DPORT_CLEAR_PERI_REG_MASK( DPORT_PERIP_RST_EN_REG, DPORT_RMT_RST );
|
|
||||||
|
|
||||||
PIN_FUNC_SELECT( GPIO_PIN_MUX_REG[ pin ], 2 );
|
#if !defined(SOC_CPU_CORES_NUM) || SOC_CPU_CORES_NUM > 1
|
||||||
gpio_set_direction( static_cast< gpio_num_t >( pin ), GPIO_MODE_OUTPUT );
|
if (!anyAlive() && isrCore != CoreCurrent) {
|
||||||
gpio_matrix_out( static_cast< gpio_num_t >( pin ), RMT_SIG_OUT0_IDX + _channel, 0, 0 );
|
|
||||||
initChannel( _channel );
|
|
||||||
|
|
||||||
RMT.tx_lim_ch[ _channel ].limit = detail::MAX_PULSES;
|
|
||||||
RMT.int_ena.val |= 1 << ( 24 + _channel );
|
|
||||||
RMT.int_ena.val |= 1 << ( 3 * _channel );
|
|
||||||
|
|
||||||
_bitToRmt[ 0 ].level0 = 1;
|
|
||||||
_bitToRmt[ 0 ].level1 = 0;
|
|
||||||
_bitToRmt[ 0 ].duration0 = _timing.T0H / ( detail::RMT_DURATION_NS * detail::DIVIDER );
|
|
||||||
_bitToRmt[ 0 ].duration1 = _timing.T0L / ( detail::RMT_DURATION_NS * detail::DIVIDER );
|
|
||||||
|
|
||||||
_bitToRmt[ 1 ].level0 = 1;
|
|
||||||
_bitToRmt[ 1 ].level1 = 0;
|
|
||||||
_bitToRmt[ 1 ].duration0 = _timing.T1H / ( detail::RMT_DURATION_NS * detail::DIVIDER );
|
|
||||||
_bitToRmt[ 1 ].duration1 = _timing.T1L / ( detail::RMT_DURATION_NS * detail::DIVIDER );
|
|
||||||
|
|
||||||
if ( !anyAlive() ) {
|
|
||||||
_interruptCore = isrCore;
|
_interruptCore = isrCore;
|
||||||
if(isrCore != CoreCurrent) {
|
ESP_ERROR_CHECK(esp_ipc_call_blocking(isrCore, registerInterrupt, (void*)this));
|
||||||
ESP_ERROR_CHECK(esp_ipc_call_blocking(isrCore, registerInterrupt, NULL));
|
} else
|
||||||
} else {
|
#endif
|
||||||
registerInterrupt(NULL);
|
{
|
||||||
}
|
registerInterrupt((void*)this);
|
||||||
}
|
}
|
||||||
|
|
||||||
ledForChannel( channel ) = this;
|
ledForChannel(channel) = this;
|
||||||
}
|
}
|
||||||
|
|
||||||
~SmartLed() {
|
~SmartLed() {
|
||||||
ledForChannel( _channel ) = nullptr;
|
ledForChannel(_channel) = nullptr;
|
||||||
if ( !anyAlive() ) {
|
#if !defined(SOC_CPU_CORES_NUM) || SOC_CPU_CORES_NUM > 1
|
||||||
if(_interruptCore != CoreCurrent) {
|
if (!anyAlive() && _interruptCore != CoreCurrent) {
|
||||||
ESP_ERROR_CHECK(esp_ipc_call_blocking(_interruptCore, unregisterInterrupt, NULL));
|
ESP_ERROR_CHECK(esp_ipc_call_blocking(_interruptCore, unregisterInterrupt, (void*)this));
|
||||||
} else {
|
} else
|
||||||
unregisterInterrupt(NULL);
|
#endif
|
||||||
|
{
|
||||||
|
unregisterInterrupt((void*)this);
|
||||||
}
|
}
|
||||||
}
|
vSemaphoreDelete(_finishedFlag);
|
||||||
vSemaphoreDelete( _finishedFlag );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Rgb& operator[]( int idx ) {
|
Rgb& operator[](int idx) { return _firstBuffer[idx]; }
|
||||||
return _firstBuffer[ idx ];
|
|
||||||
}
|
|
||||||
|
|
||||||
const Rgb& operator[]( int idx ) const {
|
const Rgb& operator[](int idx) const { return _firstBuffer[idx]; }
|
||||||
return _firstBuffer[ idx ];
|
|
||||||
}
|
|
||||||
|
|
||||||
void show() {
|
esp_err_t show() {
|
||||||
_buffer = _firstBuffer.get();
|
esp_err_t err = startTransmission();
|
||||||
startTransmission();
|
|
||||||
swapBuffers();
|
swapBuffers();
|
||||||
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool wait( TickType_t timeout = portMAX_DELAY ) {
|
bool wait(TickType_t timeout = portMAX_DELAY) {
|
||||||
if( xSemaphoreTake( _finishedFlag, timeout ) == pdTRUE ) {
|
if (xSemaphoreTake(_finishedFlag, timeout) == pdTRUE) {
|
||||||
xSemaphoreGive( _finishedFlag );
|
xSemaphoreGive(_finishedFlag);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
int size() const {
|
int size() const { return _count; }
|
||||||
return _count;
|
int channel() const { return _channel; }
|
||||||
}
|
|
||||||
|
|
||||||
Rgb *begin() { return _firstBuffer.get(); }
|
Rgb* begin() { return _firstBuffer.get(); }
|
||||||
const Rgb *begin() const { return _firstBuffer.get(); }
|
const Rgb* begin() const { return _firstBuffer.get(); }
|
||||||
const Rgb *cbegin() const { return _firstBuffer.get(); }
|
const Rgb* cbegin() const { return _firstBuffer.get(); }
|
||||||
|
|
||||||
Rgb *end() { return _firstBuffer.get() + _count; }
|
Rgb* end() { return _firstBuffer.get() + _count; }
|
||||||
const Rgb *end() const { return _firstBuffer.get() + _count; }
|
const Rgb* end() const { return _firstBuffer.get() + _count; }
|
||||||
const Rgb *cend() const { return _firstBuffer.get() + _count; }
|
const Rgb* cend() const { return _firstBuffer.get() + _count; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static intr_handle_t _interruptHandle;
|
|
||||||
static IsrCore _interruptCore;
|
static IsrCore _interruptCore;
|
||||||
|
|
||||||
static void initChannel( int channel ) {
|
static void registerInterrupt(void* selfVoid) {
|
||||||
RMT.apb_conf.fifo_mask = 1; //enable memory access, instead of FIFO mode.
|
auto* self = (SmartLed*)selfVoid;
|
||||||
RMT.apb_conf.mem_tx_wrap_en = 1; //wrap around when hitting end of buffer
|
ESP_ERROR_CHECK(self->_driver.registerIsr(!anyAlive()));
|
||||||
RMT.conf_ch[ channel ].conf0.div_cnt = detail::DIVIDER;
|
|
||||||
RMT.conf_ch[ channel ].conf0.mem_size = 1;
|
|
||||||
RMT.conf_ch[ channel ].conf0.carrier_en = 0;
|
|
||||||
RMT.conf_ch[ channel ].conf0.carrier_out_lv = 1;
|
|
||||||
RMT.conf_ch[ channel ].conf0.mem_pd = 0;
|
|
||||||
|
|
||||||
RMT.conf_ch[ channel ].conf1.rx_en = 0;
|
|
||||||
RMT.conf_ch[ channel ].conf1.mem_owner = 0;
|
|
||||||
RMT.conf_ch[ channel ].conf1.tx_conti_mode = 0; //loop back mode.
|
|
||||||
RMT.conf_ch[ channel ].conf1.ref_always_on = 1; // use apb clock: 80M
|
|
||||||
RMT.conf_ch[ channel ].conf1.idle_out_en = 1;
|
|
||||||
RMT.conf_ch[ channel ].conf1.idle_out_lv = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void registerInterrupt(void *) {
|
static void unregisterInterrupt(void* selfVoid) {
|
||||||
ESP_ERROR_CHECK(esp_intr_alloc( ETS_RMT_INTR_SOURCE, 0, interruptHandler, nullptr, &_interruptHandle));
|
auto* self = (SmartLed*)selfVoid;
|
||||||
|
ESP_ERROR_CHECK(self->_driver.unregisterIsr());
|
||||||
}
|
}
|
||||||
|
|
||||||
static void unregisterInterrupt(void*) {
|
static SmartLed*& IRAM_ATTR ledForChannel(int channel);
|
||||||
esp_intr_free( _interruptHandle );
|
|
||||||
}
|
|
||||||
|
|
||||||
static SmartLed*& IRAM_ATTR ledForChannel( int channel );
|
|
||||||
static void IRAM_ATTR interruptHandler( void* );
|
|
||||||
|
|
||||||
void IRAM_ATTR copyRmtHalfBlock();
|
|
||||||
|
|
||||||
void swapBuffers() {
|
|
||||||
if ( _secondBuffer )
|
|
||||||
_firstBuffer.swap( _secondBuffer );
|
|
||||||
}
|
|
||||||
|
|
||||||
void startTransmission() {
|
|
||||||
// Invalid use of the library
|
|
||||||
if( xSemaphoreTake( _finishedFlag, 0 ) != pdTRUE )
|
|
||||||
abort();
|
|
||||||
|
|
||||||
_pixelPosition = _componentPosition = _halfIdx = 0;
|
|
||||||
copyRmtHalfBlock();
|
|
||||||
if ( _pixelPosition < _count )
|
|
||||||
copyRmtHalfBlock();
|
|
||||||
|
|
||||||
RMT.conf_ch[ _channel ].conf1.mem_rd_rst = 1;
|
|
||||||
RMT.conf_ch[ _channel ].conf1.tx_start = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool anyAlive() {
|
static bool anyAlive() {
|
||||||
for ( int i = 0; i != 8; i++ )
|
for (int i = 0; i != detail::CHANNEL_COUNT; i++)
|
||||||
if ( ledForChannel( i ) != nullptr ) return true;
|
if (ledForChannel(i) != nullptr)
|
||||||
|
return true;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const LedType& _timing;
|
void swapBuffers() {
|
||||||
|
if (_secondBuffer)
|
||||||
|
_firstBuffer.swap(_secondBuffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
esp_err_t startTransmission() {
|
||||||
|
// Invalid use of the library, you must wait() fir previous frame to get processed first
|
||||||
|
if (xSemaphoreTake(_finishedFlag, 0) != pdTRUE)
|
||||||
|
abort();
|
||||||
|
|
||||||
|
auto err = _driver.transmit(_firstBuffer.get());
|
||||||
|
if (err != ESP_OK) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ESP_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
SemaphoreHandle_t _finishedFlag;
|
||||||
|
detail::RmtDriver _driver;
|
||||||
int _channel;
|
int _channel;
|
||||||
detail::RmtPulsePair _bitToRmt[ 2 ];
|
|
||||||
int _count;
|
int _count;
|
||||||
std::unique_ptr< Rgb[] > _firstBuffer;
|
std::unique_ptr<Rgb[]> _firstBuffer;
|
||||||
std::unique_ptr< Rgb[] > _secondBuffer;
|
std::unique_ptr<Rgb[]> _secondBuffer;
|
||||||
Rgb *_buffer;
|
|
||||||
|
|
||||||
xSemaphoreHandle _finishedFlag;
|
|
||||||
|
|
||||||
int _pixelPosition;
|
|
||||||
int _componentPosition;
|
|
||||||
int _halfIdx;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#if defined(CONFIG_IDF_TARGET_ESP32S3) || defined(CONFIG_IDF_TARGET_ESP32C3)
|
||||||
|
#define _SMARTLEDS_SPI_HOST SPI2_HOST
|
||||||
|
#define _SMARTLEDS_SPI_DMA_CHAN SPI_DMA_CH_AUTO
|
||||||
|
#else
|
||||||
|
#define _SMARTLEDS_SPI_HOST HSPI_HOST
|
||||||
|
#define _SMARTLEDS_SPI_DMA_CHAN 1
|
||||||
|
#endif
|
||||||
|
|
||||||
class Apa102 {
|
class Apa102 {
|
||||||
public:
|
public:
|
||||||
struct ApaRgb {
|
struct ApaRgb {
|
||||||
ApaRgb( uint8_t r = 0, uint8_t g = 0, uint32_t b = 0, uint32_t v = 0xFF )
|
ApaRgb(uint8_t r = 0, uint8_t g = 0, uint32_t b = 0, uint32_t v = 0xFF)
|
||||||
: v( 0xE0 | v ), b( b ), g( g ), r( r )
|
: v(0xE0 | v)
|
||||||
{}
|
, b(b)
|
||||||
|
, g(g)
|
||||||
|
, r(r) {}
|
||||||
|
|
||||||
ApaRgb& operator=( const Rgb& o ) {
|
ApaRgb& operator=(const Rgb& o) {
|
||||||
r = o.r;
|
r = o.r;
|
||||||
g = o.g;
|
g = o.g;
|
||||||
b = o.b;
|
b = o.b;
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
ApaRgb& operator=( const Hsv& o ) {
|
ApaRgb& operator=(const Hsv& o) {
|
||||||
*this = Rgb{ o };
|
*this = Rgb { o };
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -304,14 +253,14 @@ public:
|
|||||||
static const int FINAL_FRAME_SIZE = 4;
|
static const int FINAL_FRAME_SIZE = 4;
|
||||||
static const int TRANS_COUNT = 2 + 8;
|
static const int TRANS_COUNT = 2 + 8;
|
||||||
|
|
||||||
Apa102( int count, int clkpin, int datapin, BufferType doubleBuffer = SingleBuffer )
|
Apa102(int count, int clkpin, int datapin, BufferType doubleBuffer = SingleBuffer, int clock_speed_hz = 1000000)
|
||||||
: _count( count ),
|
: _count(count)
|
||||||
_firstBuffer( new ApaRgb[ count ] ),
|
, _firstBuffer(new ApaRgb[count])
|
||||||
_secondBuffer( doubleBuffer ? new ApaRgb[ count ] : nullptr ),
|
, _secondBuffer(doubleBuffer ? new ApaRgb[count] : nullptr)
|
||||||
_initFrame( 0 )
|
, _transCount(0)
|
||||||
{
|
, _initFrame(0) {
|
||||||
spi_bus_config_t buscfg;
|
spi_bus_config_t buscfg;
|
||||||
memset( &buscfg, 0, sizeof( buscfg ) );
|
memset(&buscfg, 0, sizeof(buscfg));
|
||||||
buscfg.mosi_io_num = datapin;
|
buscfg.mosi_io_num = datapin;
|
||||||
buscfg.miso_io_num = -1;
|
buscfg.miso_io_num = -1;
|
||||||
buscfg.sclk_io_num = clkpin;
|
buscfg.sclk_io_num = clkpin;
|
||||||
@@ -320,33 +269,29 @@ public:
|
|||||||
buscfg.max_transfer_sz = 65535;
|
buscfg.max_transfer_sz = 65535;
|
||||||
|
|
||||||
spi_device_interface_config_t devcfg;
|
spi_device_interface_config_t devcfg;
|
||||||
memset( &devcfg, 0, sizeof( devcfg ) );
|
memset(&devcfg, 0, sizeof(devcfg));
|
||||||
devcfg.clock_speed_hz = 1000000;
|
devcfg.clock_speed_hz = clock_speed_hz;
|
||||||
devcfg.mode = 0;
|
devcfg.mode = 0;
|
||||||
devcfg.spics_io_num = -1;
|
devcfg.spics_io_num = -1;
|
||||||
devcfg.queue_size = TRANS_COUNT;
|
devcfg.queue_size = TRANS_COUNT;
|
||||||
devcfg.pre_cb = nullptr;
|
devcfg.pre_cb = nullptr;
|
||||||
|
|
||||||
auto ret = spi_bus_initialize( HSPI_HOST, &buscfg, 1 );
|
auto ret = spi_bus_initialize(_SMARTLEDS_SPI_HOST, &buscfg, _SMARTLEDS_SPI_DMA_CHAN);
|
||||||
assert( ret == ESP_OK );
|
assert(ret == ESP_OK);
|
||||||
|
|
||||||
ret = spi_bus_add_device( HSPI_HOST, &devcfg, &_spi );
|
ret = spi_bus_add_device(_SMARTLEDS_SPI_HOST, &devcfg, &_spi);
|
||||||
assert( ret == ESP_OK );
|
assert(ret == ESP_OK);
|
||||||
|
|
||||||
std::fill_n( _finalFrame, FINAL_FRAME_SIZE, 0xFFFFFFFF );
|
std::fill_n(_finalFrame, FINAL_FRAME_SIZE, 0xFFFFFFFF);
|
||||||
}
|
}
|
||||||
|
|
||||||
~Apa102() {
|
~Apa102() {
|
||||||
// ToDo
|
// ToDo
|
||||||
}
|
}
|
||||||
|
|
||||||
ApaRgb& operator[]( int idx ) {
|
ApaRgb& operator[](int idx) { return _firstBuffer[idx]; }
|
||||||
return _firstBuffer[ idx ];
|
|
||||||
}
|
|
||||||
|
|
||||||
const ApaRgb& operator[]( int idx ) const {
|
const ApaRgb& operator[](int idx) const { return _firstBuffer[idx]; }
|
||||||
return _firstBuffer[ idx ];
|
|
||||||
}
|
|
||||||
|
|
||||||
void show() {
|
void show() {
|
||||||
_buffer = _firstBuffer.get();
|
_buffer = _firstBuffer.get();
|
||||||
@@ -355,93 +300,95 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
void wait() {
|
void wait() {
|
||||||
for ( int i = 0; i != _transCount; i++ ) {
|
for (int i = 0; i != _transCount; i++) {
|
||||||
spi_transaction_t *t;
|
spi_transaction_t* t;
|
||||||
spi_device_get_trans_result( _spi, &t, portMAX_DELAY );
|
spi_device_get_trans_result(_spi, &t, portMAX_DELAY);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void swapBuffers() {
|
void swapBuffers() {
|
||||||
if ( _secondBuffer )
|
if (_secondBuffer)
|
||||||
_firstBuffer.swap( _secondBuffer );
|
_firstBuffer.swap(_secondBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
void startTransmission() {
|
void startTransmission() {
|
||||||
for ( int i = 0; i != TRANS_COUNT; i++ ) {
|
for (int i = 0; i != TRANS_COUNT; i++) {
|
||||||
_transactions[ i ].cmd = 0;
|
_transactions[i].cmd = 0;
|
||||||
_transactions[ i ].addr = 0;
|
_transactions[i].addr = 0;
|
||||||
_transactions[ i ].flags = 0;
|
_transactions[i].flags = 0;
|
||||||
_transactions[ i ].rxlength = 0;
|
_transactions[i].rxlength = 0;
|
||||||
_transactions[ i ].rx_buffer = nullptr;
|
_transactions[i].rx_buffer = nullptr;
|
||||||
}
|
}
|
||||||
// Init frame
|
// Init frame
|
||||||
_transactions[ 0 ].length = 32;
|
_transactions[0].length = 32;
|
||||||
_transactions[ 0 ].tx_buffer = &_initFrame;
|
_transactions[0].tx_buffer = &_initFrame;
|
||||||
spi_device_queue_trans( _spi, _transactions + 0, portMAX_DELAY );
|
spi_device_queue_trans(_spi, _transactions + 0, portMAX_DELAY);
|
||||||
// Data
|
// Data
|
||||||
_transactions[ 1 ].length = 32 * _count;
|
_transactions[1].length = 32 * _count;
|
||||||
_transactions[ 1 ].tx_buffer = _buffer;
|
_transactions[1].tx_buffer = _buffer;
|
||||||
spi_device_queue_trans( _spi, _transactions + 1, portMAX_DELAY );
|
spi_device_queue_trans(_spi, _transactions + 1, portMAX_DELAY);
|
||||||
_transCount = 2;
|
_transCount = 2;
|
||||||
// End frame
|
// End frame
|
||||||
for ( int i = 0; i != 1 + _count / 32 / FINAL_FRAME_SIZE; i++ ) {
|
for (int i = 0; i != 1 + _count / 32 / FINAL_FRAME_SIZE; i++) {
|
||||||
_transactions[ 2 + i ].length = 32 * FINAL_FRAME_SIZE;
|
_transactions[2 + i].length = 32 * FINAL_FRAME_SIZE;
|
||||||
_transactions[ 2 + i ].tx_buffer = _finalFrame;
|
_transactions[2 + i].tx_buffer = _finalFrame;
|
||||||
spi_device_queue_trans( _spi, _transactions + 2 + i, portMAX_DELAY );
|
spi_device_queue_trans(_spi, _transactions + 2 + i, portMAX_DELAY);
|
||||||
_transCount++;
|
_transCount++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
spi_device_handle_t _spi;
|
spi_device_handle_t _spi;
|
||||||
int _count;
|
int _count;
|
||||||
std::unique_ptr< ApaRgb[] > _firstBuffer, _secondBuffer;
|
std::unique_ptr<ApaRgb[]> _firstBuffer, _secondBuffer;
|
||||||
ApaRgb *_buffer;
|
ApaRgb* _buffer;
|
||||||
|
|
||||||
spi_transaction_t _transactions[ TRANS_COUNT ];
|
spi_transaction_t _transactions[TRANS_COUNT];
|
||||||
int _transCount;
|
int _transCount;
|
||||||
|
|
||||||
uint32_t _initFrame;
|
uint32_t _initFrame;
|
||||||
uint32_t _finalFrame[ FINAL_FRAME_SIZE ];
|
uint32_t _finalFrame[FINAL_FRAME_SIZE];
|
||||||
};
|
};
|
||||||
|
|
||||||
class LDP8806 {
|
class LDP8806 {
|
||||||
public:
|
public:
|
||||||
struct LDP8806_GRB {
|
struct LDP8806_GRB {
|
||||||
|
|
||||||
LDP8806_GRB( uint8_t g_7bit = 0, uint8_t r_7bit = 0, uint32_t b_7bit = 0 )
|
LDP8806_GRB(uint8_t g_7bit = 0, uint8_t r_7bit = 0, uint32_t b_7bit = 0)
|
||||||
: g( g_7bit ), r( r_7bit ), b( b_7bit )
|
: g(g_7bit)
|
||||||
{
|
, r(r_7bit)
|
||||||
}
|
, b(b_7bit) {}
|
||||||
|
|
||||||
LDP8806_GRB& operator=( const Rgb& o ) {
|
LDP8806_GRB& operator=(const Rgb& o) {
|
||||||
//Convert 8->7bit colour
|
//Convert 8->7bit colour
|
||||||
r = ( o.r * 127 / 256 ) | 0x80;
|
r = (o.r * 127 / 256) | 0x80;
|
||||||
g = ( o.g * 127 / 256 ) | 0x80;
|
g = (o.g * 127 / 256) | 0x80;
|
||||||
b = ( o.b * 127 / 256 ) | 0x80;
|
b = (o.b * 127 / 256) | 0x80;
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
LDP8806_GRB& operator=( const Hsv& o ) {
|
LDP8806_GRB& operator=(const Hsv& o) {
|
||||||
*this = Rgb{ o };
|
*this = Rgb { o };
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t g, r, b;
|
uint8_t g, r, b;
|
||||||
};
|
};
|
||||||
|
|
||||||
static const int LED_FRAME_SIZE_BYTES = sizeof( LDP8806_GRB );
|
static const int LED_FRAME_SIZE_BYTES = sizeof(LDP8806_GRB);
|
||||||
static const int LATCH_FRAME_SIZE_BYTES = 3;
|
static const int LATCH_FRAME_SIZE_BYTES = 3;
|
||||||
static const int TRANS_COUNT_MAX = 20;//Arbitrary, supports up to 600 LED
|
static const int TRANS_COUNT_MAX = 20; //Arbitrary, supports up to 600 LED
|
||||||
|
|
||||||
LDP8806( int count, int clkpin, int datapin, BufferType doubleBuffer = SingleBuffer, uint32_t clock_speed_hz = 2000000 )
|
LDP8806(
|
||||||
: _count( count ),
|
int count, int clkpin, int datapin, BufferType doubleBuffer = SingleBuffer, uint32_t clock_speed_hz = 2000000)
|
||||||
_firstBuffer( new LDP8806_GRB[ count ] ),
|
: _count(count)
|
||||||
_secondBuffer( doubleBuffer ? new LDP8806_GRB[ count ] : nullptr ),
|
, _firstBuffer(new LDP8806_GRB[count])
|
||||||
|
, _secondBuffer(doubleBuffer ? new LDP8806_GRB[count] : nullptr)
|
||||||
|
,
|
||||||
// one 'latch'/start-of-data mark frame for every 32 leds
|
// one 'latch'/start-of-data mark frame for every 32 leds
|
||||||
_latchFrames( ( count + 31 ) / 32 )
|
_latchFrames((count + 31) / 32) {
|
||||||
{
|
|
||||||
spi_bus_config_t buscfg;
|
spi_bus_config_t buscfg;
|
||||||
memset( &buscfg, 0, sizeof( buscfg ) );
|
memset(&buscfg, 0, sizeof(buscfg));
|
||||||
buscfg.mosi_io_num = datapin;
|
buscfg.mosi_io_num = datapin;
|
||||||
buscfg.miso_io_num = -1;
|
buscfg.miso_io_num = -1;
|
||||||
buscfg.sclk_io_num = clkpin;
|
buscfg.sclk_io_num = clkpin;
|
||||||
@@ -450,33 +397,29 @@ public:
|
|||||||
buscfg.max_transfer_sz = 65535;
|
buscfg.max_transfer_sz = 65535;
|
||||||
|
|
||||||
spi_device_interface_config_t devcfg;
|
spi_device_interface_config_t devcfg;
|
||||||
memset( &devcfg, 0, sizeof( devcfg ) );
|
memset(&devcfg, 0, sizeof(devcfg));
|
||||||
devcfg.clock_speed_hz = clock_speed_hz;
|
devcfg.clock_speed_hz = clock_speed_hz;
|
||||||
devcfg.mode = 0;
|
devcfg.mode = 0;
|
||||||
devcfg.spics_io_num = -1;
|
devcfg.spics_io_num = -1;
|
||||||
devcfg.queue_size = TRANS_COUNT_MAX;
|
devcfg.queue_size = TRANS_COUNT_MAX;
|
||||||
devcfg.pre_cb = nullptr;
|
devcfg.pre_cb = nullptr;
|
||||||
|
|
||||||
auto ret = spi_bus_initialize( HSPI_HOST, &buscfg, 1 );
|
auto ret = spi_bus_initialize(_SMARTLEDS_SPI_HOST, &buscfg, _SMARTLEDS_SPI_DMA_CHAN);
|
||||||
assert( ret == ESP_OK );
|
assert(ret == ESP_OK);
|
||||||
|
|
||||||
ret = spi_bus_add_device( HSPI_HOST, &devcfg, &_spi );
|
ret = spi_bus_add_device(_SMARTLEDS_SPI_HOST, &devcfg, &_spi);
|
||||||
assert( ret == ESP_OK );
|
assert(ret == ESP_OK);
|
||||||
|
|
||||||
std::fill_n( _latchBuffer, LATCH_FRAME_SIZE_BYTES, 0x0 );
|
std::fill_n(_latchBuffer, LATCH_FRAME_SIZE_BYTES, 0x0);
|
||||||
}
|
}
|
||||||
|
|
||||||
~LDP8806() {
|
~LDP8806() {
|
||||||
// noop
|
// noop
|
||||||
}
|
}
|
||||||
|
|
||||||
LDP8806_GRB& operator[]( int idx ) {
|
LDP8806_GRB& operator[](int idx) { return _firstBuffer[idx]; }
|
||||||
return _firstBuffer[ idx ];
|
|
||||||
}
|
|
||||||
|
|
||||||
const LDP8806_GRB& operator[]( int idx ) const {
|
const LDP8806_GRB& operator[](int idx) const { return _firstBuffer[idx]; }
|
||||||
return _firstBuffer[ idx ];
|
|
||||||
}
|
|
||||||
|
|
||||||
void show() {
|
void show() {
|
||||||
_buffer = _firstBuffer.get();
|
_buffer = _firstBuffer.get();
|
||||||
@@ -485,51 +428,50 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
void wait() {
|
void wait() {
|
||||||
while ( _transCount-- ) {
|
while (_transCount--) {
|
||||||
spi_transaction_t *t;
|
spi_transaction_t* t;
|
||||||
spi_device_get_trans_result( _spi, &t, portMAX_DELAY );
|
spi_device_get_trans_result(_spi, &t, portMAX_DELAY);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void swapBuffers() {
|
void swapBuffers() {
|
||||||
if ( _secondBuffer )
|
if (_secondBuffer)
|
||||||
_firstBuffer.swap( _secondBuffer );
|
_firstBuffer.swap(_secondBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
void startTransmission() {
|
void startTransmission() {
|
||||||
_transCount = 0;
|
_transCount = 0;
|
||||||
for ( int i = 0; i != TRANS_COUNT_MAX; i++ ) {
|
for (int i = 0; i != TRANS_COUNT_MAX; i++) {
|
||||||
_transactions[ i ].cmd = 0;
|
_transactions[i].cmd = 0;
|
||||||
_transactions[ i ].addr = 0;
|
_transactions[i].addr = 0;
|
||||||
_transactions[ i ].flags = 0;
|
_transactions[i].flags = 0;
|
||||||
_transactions[ i ].rxlength = 0;
|
_transactions[i].rxlength = 0;
|
||||||
_transactions[ i ].rx_buffer = nullptr;
|
_transactions[i].rx_buffer = nullptr;
|
||||||
}
|
}
|
||||||
// LED Data
|
// LED Data
|
||||||
_transactions[ 0 ].length = ( LED_FRAME_SIZE_BYTES * 8 ) * _count;
|
_transactions[0].length = (LED_FRAME_SIZE_BYTES * 8) * _count;
|
||||||
_transactions[ 0 ].tx_buffer = _buffer;
|
_transactions[0].tx_buffer = _buffer;
|
||||||
spi_device_queue_trans( _spi, _transactions + _transCount, portMAX_DELAY );
|
spi_device_queue_trans(_spi, _transactions + _transCount, portMAX_DELAY);
|
||||||
_transCount++;
|
_transCount++;
|
||||||
|
|
||||||
// 'latch'/start-of-data marker frames
|
// 'latch'/start-of-data marker frames
|
||||||
for ( int i = 0; i < _latchFrames; i++ ) {
|
for (int i = 0; i < _latchFrames; i++) {
|
||||||
_transactions[ _transCount ].length = ( LATCH_FRAME_SIZE_BYTES * 8 );
|
_transactions[_transCount].length = (LATCH_FRAME_SIZE_BYTES * 8);
|
||||||
_transactions[ _transCount ].tx_buffer = _latchBuffer;
|
_transactions[_transCount].tx_buffer = _latchBuffer;
|
||||||
spi_device_queue_trans( _spi, _transactions + _transCount, portMAX_DELAY );
|
spi_device_queue_trans(_spi, _transactions + _transCount, portMAX_DELAY);
|
||||||
_transCount++;
|
_transCount++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
spi_device_handle_t _spi;
|
spi_device_handle_t _spi;
|
||||||
int _count;
|
int _count;
|
||||||
std::unique_ptr< LDP8806_GRB[] > _firstBuffer, _secondBuffer;
|
std::unique_ptr<LDP8806_GRB[]> _firstBuffer, _secondBuffer;
|
||||||
LDP8806_GRB *_buffer;
|
LDP8806_GRB* _buffer;
|
||||||
|
|
||||||
spi_transaction_t _transactions[ TRANS_COUNT_MAX ];
|
spi_transaction_t _transactions[TRANS_COUNT_MAX];
|
||||||
int _transCount;
|
int _transCount;
|
||||||
|
|
||||||
int _latchFrames;
|
int _latchFrames;
|
||||||
uint8_t _latchBuffer[ LATCH_FRAME_SIZE_BYTES ];
|
uint8_t _latchBuffer[LATCH_FRAME_SIZE_BYTES];
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif //SMARTLEDS_H
|
|
||||||
|
|||||||
@@ -6,9 +6,6 @@
|
|||||||
#include "freertos/FreeRTOS.h"
|
#include "freertos/FreeRTOS.h"
|
||||||
#include "freertos/task.h"
|
#include "freertos/task.h"
|
||||||
#include "esp_system.h"
|
#include "esp_system.h"
|
||||||
#include "esp_event.h"
|
|
||||||
|
|
||||||
#include "server_tflite.h"
|
|
||||||
|
|
||||||
#include "esp_log.h"
|
#include "esp_log.h"
|
||||||
|
|
||||||
@@ -22,10 +19,13 @@
|
|||||||
#include "ClassLogFile.h"
|
#include "ClassLogFile.h"
|
||||||
#include "configFile.h"
|
#include "configFile.h"
|
||||||
#include "Helper.h"
|
#include "Helper.h"
|
||||||
|
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
#include "interface_mqtt.h"
|
#include "interface_mqtt.h"
|
||||||
|
#include "server_mqtt.h"
|
||||||
#endif //ENABLE_MQTT
|
#endif //ENABLE_MQTT
|
||||||
|
|
||||||
|
#include "basic_auth.h"
|
||||||
|
|
||||||
static const char *TAG = "GPIO";
|
static const char *TAG = "GPIO";
|
||||||
QueueHandle_t gpio_queue_handle = NULL;
|
QueueHandle_t gpio_queue_handle = NULL;
|
||||||
@@ -82,10 +82,10 @@ static void gpioHandlerTask(void *arg) {
|
|||||||
|
|
||||||
void GpioPin::gpioInterrupt(int value) {
|
void GpioPin::gpioInterrupt(int value) {
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
if (_mqttTopic != "") {
|
if (_mqttTopic.compare("") != 0) {
|
||||||
ESP_LOGD(TAG, "gpioInterrupt %s %d", _mqttTopic.c_str(), value);
|
ESP_LOGD(TAG, "gpioInterrupt %s %d", _mqttTopic.c_str(), value);
|
||||||
|
|
||||||
MQTTPublish(_mqttTopic, value ? "true" : "false");
|
MQTTPublish(_mqttTopic, value ? "true" : "false", 1);
|
||||||
}
|
}
|
||||||
#endif //ENABLE_MQTT
|
#endif //ENABLE_MQTT
|
||||||
currentState = value;
|
currentState = value;
|
||||||
@@ -115,7 +115,7 @@ void GpioPin::init()
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
if ((_mqttTopic != "") && ((_mode == GPIO_PIN_MODE_OUTPUT) || (_mode == GPIO_PIN_MODE_OUTPUT_PWM) || (_mode == GPIO_PIN_MODE_BUILT_IN_FLASH_LED))) {
|
if ((_mqttTopic.compare("") != 0) && ((_mode == GPIO_PIN_MODE_OUTPUT) || (_mode == GPIO_PIN_MODE_OUTPUT_PWM) || (_mode == GPIO_PIN_MODE_BUILT_IN_FLASH_LED))) {
|
||||||
std::function<bool(std::string, char*, int)> f = std::bind(&GpioPin::handleMQTT, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3);
|
std::function<bool(std::string, char*, int)> f = std::bind(&GpioPin::handleMQTT, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3);
|
||||||
MQTTregisterSubscribeFunction(_mqttTopic, f);
|
MQTTregisterSubscribeFunction(_mqttTopic, f);
|
||||||
}
|
}
|
||||||
@@ -141,8 +141,8 @@ void GpioPin::setValue(bool value, gpio_set_source setSource, std::string* error
|
|||||||
gpio_set_level(_gpio, value);
|
gpio_set_level(_gpio, value);
|
||||||
|
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
if ((_mqttTopic != "") && (setSource != GPIO_SET_SOURCE_MQTT)) {
|
if ((_mqttTopic.compare("") != 0) && (setSource != GPIO_SET_SOURCE_MQTT)) {
|
||||||
MQTTPublish(_mqttTopic, value ? "true" : "false");
|
MQTTPublish(_mqttTopic, value ? "true" : "false", 1);
|
||||||
}
|
}
|
||||||
#endif //ENABLE_MQTT
|
#endif //ENABLE_MQTT
|
||||||
}
|
}
|
||||||
@@ -153,7 +153,8 @@ void GpioPin::publishState() {
|
|||||||
if (newState != currentState) {
|
if (newState != currentState) {
|
||||||
ESP_LOGD(TAG,"publish state of GPIO %d new state %d", _gpio, newState);
|
ESP_LOGD(TAG,"publish state of GPIO %d new state %d", _gpio, newState);
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
MQTTPublish(_mqttTopic, newState ? "true" : "false");
|
if (_mqttTopic.compare("") != 0)
|
||||||
|
MQTTPublish(_mqttTopic, newState ? "true" : "false", 1);
|
||||||
#endif //ENABLE_MQTT
|
#endif //ENABLE_MQTT
|
||||||
currentState = newState;
|
currentState = newState;
|
||||||
}
|
}
|
||||||
@@ -330,7 +331,7 @@ bool GpioHandler::readConfig()
|
|||||||
|
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
// std::string mainTopicMQTT = "";
|
// std::string mainTopicMQTT = "";
|
||||||
std::string mainTopicMQTT = GetMQTTMainTopic();
|
std::string mainTopicMQTT = mqttServer_getMainTopic();
|
||||||
if (mainTopicMQTT.length() > 0)
|
if (mainTopicMQTT.length() > 0)
|
||||||
{
|
{
|
||||||
mainTopicMQTT = mainTopicMQTT + "/GPIO";
|
mainTopicMQTT = mainTopicMQTT + "/GPIO";
|
||||||
@@ -359,9 +360,9 @@ bool GpioHandler::readConfig()
|
|||||||
gpio_int_type_t intType = resolveIntType(toLower(splitted[2]));
|
gpio_int_type_t intType = resolveIntType(toLower(splitted[2]));
|
||||||
uint16_t dutyResolution = (uint8_t)atoi(splitted[3].c_str());
|
uint16_t dutyResolution = (uint8_t)atoi(splitted[3].c_str());
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
bool mqttEnabled = toLower(splitted[4]) == "true";
|
bool mqttEnabled = (toLower(splitted[4]) == "true");
|
||||||
#endif // ENABLE_MQTT
|
#endif // ENABLE_MQTT
|
||||||
bool httpEnabled = toLower(splitted[5]) == "true";
|
bool httpEnabled = (toLower(splitted[5]) == "true");
|
||||||
char gpioName[100];
|
char gpioName[100];
|
||||||
if (splitted.size() >= 7) {
|
if (splitted.size() >= 7) {
|
||||||
strcpy(gpioName, trim(splitted[6]).c_str());
|
strcpy(gpioName, trim(splitted[6]).c_str());
|
||||||
@@ -458,7 +459,7 @@ void GpioHandler::registerGpioUri()
|
|||||||
httpd_uri_t camuri = { };
|
httpd_uri_t camuri = { };
|
||||||
camuri.method = HTTP_GET;
|
camuri.method = HTTP_GET;
|
||||||
camuri.uri = "/GPIO";
|
camuri.uri = "/GPIO";
|
||||||
camuri.handler = callHandleHttpRequest;
|
camuri.handler = APPLY_BASIC_AUTH_FILTER(callHandleHttpRequest);
|
||||||
camuri.user_ctx = (void*)this;
|
camuri.user_ctx = (void*)this;
|
||||||
httpd_register_uri_handler(_httpServer, &camuri);
|
httpd_register_uri_handler(_httpServer, &camuri);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,6 +4,6 @@ list(APPEND EXTRA_COMPONENT_DIRS $ENV{IDF_PATH}/examples/common_components/proto
|
|||||||
|
|
||||||
idf_component_register(SRCS ${app_sources}
|
idf_component_register(SRCS ${app_sources}
|
||||||
INCLUDE_DIRS "."
|
INCLUDE_DIRS "."
|
||||||
REQUIRES esp32-camera esp_http_server jomjol_logfile jomjol_image_proc nvs_flash jomjol_fileserver_ota jomjol_controlGPIO)
|
REQUIRES esp_timer esp32-camera esp_http_server jomjol_logfile jomjol_image_proc nvs_flash jomjol_fileserver_ota jomjol_controlGPIO)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -15,48 +15,103 @@
|
|||||||
#include "CImageBasis.h"
|
#include "CImageBasis.h"
|
||||||
#include "../../include/defines.h"
|
#include "../../include/defines.h"
|
||||||
|
|
||||||
class CCamera {
|
typedef struct
|
||||||
protected:
|
{
|
||||||
int ActualQuality;
|
uint16_t CamSensor_id;
|
||||||
framesize_t ActualResolution;
|
|
||||||
int brightness, contrast, saturation;
|
|
||||||
bool isFixedExposure;
|
|
||||||
int waitbeforepicture_org;
|
|
||||||
int led_intensity = 4095;
|
|
||||||
|
|
||||||
|
framesize_t ImageFrameSize = FRAMESIZE_VGA; // 0 - 10
|
||||||
|
gainceiling_t ImageGainceiling; // Image gain (GAINCEILING_x2, x4, x8, x16, x32, x64 or x128)
|
||||||
|
|
||||||
|
int ImageQuality; // 0 - 63
|
||||||
|
int ImageBrightness; // (-2 to 2) - set brightness
|
||||||
|
int ImageContrast; //-2 - 2
|
||||||
|
int ImageSaturation; //-2 - 2
|
||||||
|
int ImageSharpness; //-2 - 2
|
||||||
|
bool ImageAutoSharpness;
|
||||||
|
int ImageSpecialEffect; // 0 - 6
|
||||||
|
int ImageWbMode; // 0 to 4 - if awb_gain enabled (0 - Auto, 1 - Sunny, 2 - Cloudy, 3 - Office, 4 - Home)
|
||||||
|
int ImageAwb; // white balance enable (0 or 1)
|
||||||
|
int ImageAwbGain; // Auto White Balance enable (0 or 1)
|
||||||
|
int ImageAec; // auto exposure off (1 or 0)
|
||||||
|
int ImageAec2; // automatic exposure sensor (0 or 1)
|
||||||
|
int ImageAeLevel; // auto exposure levels (-2 to 2)
|
||||||
|
int ImageAecValue; // set exposure manually (0-1200)
|
||||||
|
int ImageAgc; // auto gain off (1 or 0)
|
||||||
|
int ImageAgcGain; // set gain manually (0 - 30)
|
||||||
|
int ImageBpc; // black pixel correction
|
||||||
|
int ImageWpc; // white pixel correction
|
||||||
|
int ImageRawGma; // (1 or 0)
|
||||||
|
int ImageLenc; // lens correction (1 or 0)
|
||||||
|
int ImageHmirror; // (0 or 1) flip horizontally
|
||||||
|
int ImageVflip; // Invert image (0 or 1)
|
||||||
|
int ImageDcw; // downsize enable (1 or 0)
|
||||||
|
|
||||||
|
int ImageDenoiseLevel; // The OV2640 does not support it, OV3660 and OV5640 (0 to 8)
|
||||||
|
|
||||||
|
int ImageWidth;
|
||||||
|
int ImageHeight;
|
||||||
|
|
||||||
|
int ImageLedIntensity;
|
||||||
|
|
||||||
|
bool ImageZoomEnabled;
|
||||||
|
int ImageZoomOffsetX;
|
||||||
|
int ImageZoomOffsetY;
|
||||||
|
int ImageZoomSize;
|
||||||
|
|
||||||
|
int WaitBeforePicture;
|
||||||
|
bool isImageSize;
|
||||||
|
|
||||||
|
bool CameraInitSuccessful;
|
||||||
|
bool changedCameraSettings;
|
||||||
|
bool DemoMode;
|
||||||
|
bool SaveAllFiles;
|
||||||
|
} camera_controll_config_temp_t;
|
||||||
|
|
||||||
|
extern camera_controll_config_temp_t CCstatus;
|
||||||
|
|
||||||
|
class CCamera
|
||||||
|
{
|
||||||
|
protected:
|
||||||
void ledc_init(void);
|
void ledc_init(void);
|
||||||
bool CameraInitSuccessful = false;
|
|
||||||
bool demoMode = false;
|
|
||||||
|
|
||||||
bool loadNextDemoImage(camera_fb_t *fb);
|
bool loadNextDemoImage(camera_fb_t *fb);
|
||||||
long GetFileSize(std::string filename);
|
long GetFileSize(std::string filename);
|
||||||
|
void SetCamWindow(sensor_t *s, int frameSizeX, int frameSizeY, int xOffset, int yOffset, int xTotal, int yTotal, int xOutput, int yOutput, int imageVflip);
|
||||||
|
void SetImageWidthHeightFromResolution(framesize_t resol);
|
||||||
|
void SanitizeZoomParams(int imageSize, int frameSizeX, int frameSizeY, int &imageWidth, int &imageHeight, int &zoomOffsetX, int &zoomOffsetY);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
int image_height, image_width;
|
int LedIntensity = 4096;
|
||||||
|
|
||||||
CCamera();
|
CCamera(void);
|
||||||
esp_err_t InitCam();
|
esp_err_t InitCam(void);
|
||||||
|
|
||||||
void LightOnOff(bool status);
|
void LightOnOff(bool status);
|
||||||
void LEDOnOff(bool status);
|
void LEDOnOff(bool status);
|
||||||
|
|
||||||
|
esp_err_t setSensorDatenFromCCstatus(void);
|
||||||
|
esp_err_t getSensorDatenToCCstatus(void);
|
||||||
|
|
||||||
|
int SetCamGainceiling(sensor_t *s, gainceiling_t gainceilingLevel);
|
||||||
|
void SetCamSharpness(bool autoSharpnessEnabled, int sharpnessLevel);
|
||||||
|
void SetCamSpecialEffect(sensor_t *s, int specialEffect);
|
||||||
|
void SetCamContrastBrightness(sensor_t *s, int _contrast, int _brightness);
|
||||||
|
|
||||||
esp_err_t CaptureToHTTP(httpd_req_t *req, int delay = 0);
|
esp_err_t CaptureToHTTP(httpd_req_t *req, int delay = 0);
|
||||||
void SetQualitySize(int qual, framesize_t resol);
|
esp_err_t CaptureToStream(httpd_req_t *req, bool FlashlightOn);
|
||||||
bool SetBrightnessContrastSaturation(int _brightness, int _contrast, int _saturation);
|
|
||||||
void GetCameraParameter(httpd_req_t *req, int &qual, framesize_t &resol);
|
void SetQualityZoomSize(int qual, framesize_t resol, bool zoomEnabled, int zoomOffsetX, int zoomOffsetY, int imageSize, int imageVflip);
|
||||||
void SetLEDIntensity(float _intrel);
|
void SetZoomSize(bool zoomEnabled, int zoomOffsetX, int zoomOffsetY, int imageSize, int imageVflip);
|
||||||
|
|
||||||
|
int SetLEDIntensity(int _intrel);
|
||||||
bool testCamera(void);
|
bool testCamera(void);
|
||||||
void EnableAutoExposure(int flash_duration);
|
bool getCameraInitSuccessful(void);
|
||||||
bool getCameraInitSuccessful();
|
|
||||||
void useDemoMode(void);
|
void useDemoMode(void);
|
||||||
|
|
||||||
|
framesize_t TextToFramesize(const char *text);
|
||||||
framesize_t TextToFramesize(const char * text);
|
|
||||||
|
|
||||||
esp_err_t CaptureToFile(std::string nm, int delay = 0);
|
esp_err_t CaptureToFile(std::string nm, int delay = 0);
|
||||||
esp_err_t CaptureToBasisImage(CImageBasis *_Image, int delay = 0);
|
esp_err_t CaptureToBasisImage(CImageBasis *_Image, int delay = 0);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
extern CCamera Camera;
|
extern CCamera Camera;
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
@@ -0,0 +1,88 @@
|
|||||||
|
// Workaround - bug in cam library - enable bits are set without using bitwise OR logic -> only latest enable setting is used
|
||||||
|
// Reference: https://esp32.com/viewtopic.php?f=19&t=14376#p93178
|
||||||
|
/* The memory structure is as follows for
|
||||||
|
byte_0 = enable_bits
|
||||||
|
byte_0->bit0 = enable saturation and hue --> OK
|
||||||
|
byte_0->bit1 = enable saturation --> OK
|
||||||
|
byte_0->bit2 = enable brightness and contrast --> OK
|
||||||
|
byte_0->bit3 = enable green -> blue spitial effect (Antique and blunish and greenish and reddish and b&w) enable
|
||||||
|
byte_0->bit4 = anable gray -> red spitial effect (Antique and blunish and greenish and reddish and b&w) enable
|
||||||
|
byte_0->bit5 = remove (UV) in YUV color system
|
||||||
|
byte_0->bit6 = enable negative
|
||||||
|
byte_0->bit7 = remove (Y) in YUV color system
|
||||||
|
byte_1 = saturation1 0-255 --> ?
|
||||||
|
byte_2 = hue 0-255 --> OK
|
||||||
|
byte_3 = saturation2 0-255 --> OK
|
||||||
|
byte_4 = reenter saturation2 in documents --> ?
|
||||||
|
byte_5 = spital effect green -> blue 0-255 --> ?
|
||||||
|
byte_6 = spital effect gray -> red 0-255 --> ?
|
||||||
|
byte_7 = contrast lower byte 0-255 --> OK
|
||||||
|
byte_8 = contrast higher byte 0-255 --> OK
|
||||||
|
byte_9 = brightness 0-255 --> OK
|
||||||
|
byte_10 = if byte_10==4 contrast effective --> ?
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include "esp_camera.h"
|
||||||
|
#include "ov2640_contrast_brightness.h"
|
||||||
|
|
||||||
|
static const uint8_t brightness_regs[6][5] = {
|
||||||
|
{0x7C, 0x7D, 0x7C, 0x7D, 0x7D },
|
||||||
|
{0x00, 0x04, 0x09, 0x00, 0x00 }, /* -2 */
|
||||||
|
{0x00, 0x04, 0x09, 0x10, 0x00 }, /* -1 */
|
||||||
|
{0x00, 0x04, 0x09, 0x20, 0x00 }, /* 0 */
|
||||||
|
{0x00, 0x04, 0x09, 0x30, 0x00 }, /* +1 */
|
||||||
|
{0x00, 0x04, 0x09, 0x40, 0x00 }, /* +2 */
|
||||||
|
};
|
||||||
|
|
||||||
|
static const uint8_t contrast_regs[6][7] = {
|
||||||
|
{0x7C, 0x7D, 0x7C, 0x7D, 0x7D, 0x7D, 0x7D },
|
||||||
|
{0x00, 0x04, 0x07, 0x20, 0x18, 0x34, 0x06 }, /* -2 */
|
||||||
|
{0x00, 0x04, 0x07, 0x20, 0x1c, 0x2a, 0x06 }, /* -1 */
|
||||||
|
{0x00, 0x04, 0x07, 0x20, 0x20, 0x20, 0x06 }, /* 0 */
|
||||||
|
{0x00, 0x04, 0x07, 0x20, 0x24, 0x16, 0x06 }, /* +1 */
|
||||||
|
{0x00, 0x04, 0x07, 0x20, 0x28, 0x0c, 0x06 }, /* +2 */
|
||||||
|
};
|
||||||
|
|
||||||
|
int ov2640_set_contrast_brightness(sensor_t *sensor, int _contrast, int _brightness)
|
||||||
|
{
|
||||||
|
int ret=0;
|
||||||
|
|
||||||
|
_contrast += 3;
|
||||||
|
if (_contrast <= 0) {
|
||||||
|
_contrast = 3;
|
||||||
|
}
|
||||||
|
else if (_contrast > 5)
|
||||||
|
{
|
||||||
|
_contrast = 5;
|
||||||
|
}
|
||||||
|
sensor->status.contrast = _contrast-3;
|
||||||
|
|
||||||
|
_brightness += 3;
|
||||||
|
if (_brightness <= 0) {
|
||||||
|
_brightness = 3;
|
||||||
|
}
|
||||||
|
else if (_brightness > 5)
|
||||||
|
{
|
||||||
|
_brightness = 5;
|
||||||
|
}
|
||||||
|
int brightness = brightness_regs[_brightness][3];
|
||||||
|
sensor->status.brightness = _brightness-3;
|
||||||
|
|
||||||
|
// sensor->set_reg(sensor, int reg, int mask, int value)
|
||||||
|
sensor->set_reg(sensor, 0xFF, 0x01, 0x00); // Select DSP bank
|
||||||
|
|
||||||
|
for (int i=0; i<7; i++)
|
||||||
|
{
|
||||||
|
if (i == 5)
|
||||||
|
{
|
||||||
|
sensor->set_reg(sensor, contrast_regs[0][i], 0xFF, (brightness | contrast_regs[_contrast][i]));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sensor->set_reg(sensor, contrast_regs[0][i], 0xFF, contrast_regs[_contrast][i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifndef OV2640_CONTRAST_BRIGHTNESS_H
|
||||||
|
#define OV2640_CONTRAST_BRIGHTNESS_H
|
||||||
|
|
||||||
|
#include "esp_camera.h"
|
||||||
|
|
||||||
|
int ov2640_set_contrast_brightness(sensor_t *sensor, int _contrast, int _brightness);
|
||||||
|
|
||||||
|
#endif
|
||||||
151
code/components/jomjol_controlcamera/ov2640_sharpness.cpp
Normal file
151
code/components/jomjol_controlcamera/ov2640_sharpness.cpp
Normal file
@@ -0,0 +1,151 @@
|
|||||||
|
#include <stdint.h>
|
||||||
|
#include "esp_camera.h"
|
||||||
|
#include "ov2640_sharpness.h"
|
||||||
|
|
||||||
|
const static uint8_t OV2640_SHARPNESS_AUTO[]=
|
||||||
|
{
|
||||||
|
//reg, val, mask
|
||||||
|
0xFF, 0x00, 0xFF,
|
||||||
|
0x92, 0x01, 0xFF,
|
||||||
|
0x93, 0x20, 0x20,
|
||||||
|
0x00, 0x00, 0x00
|
||||||
|
};
|
||||||
|
|
||||||
|
const static uint8_t OV2640_SHARPNESS_MANUAL[]=
|
||||||
|
{
|
||||||
|
//reg, val, mask
|
||||||
|
0xFF, 0x00, 0xFF,
|
||||||
|
0x92, 0x01, 0xFF,
|
||||||
|
0x93, 0x00, 0x20,
|
||||||
|
0x00, 0x00, 0x00
|
||||||
|
};
|
||||||
|
|
||||||
|
const static uint8_t OV2640_SHARPNESS_LEVEL0[]=
|
||||||
|
{
|
||||||
|
//reg, val, mask
|
||||||
|
0xFF, 0x00, 0xFF,
|
||||||
|
0x92, 0x01, 0xFF,
|
||||||
|
0x93, 0xC0, 0x1F,
|
||||||
|
0x00, 0x00, 0x00
|
||||||
|
};
|
||||||
|
const static uint8_t OV2640_SHARPNESS_LEVEL1[]=
|
||||||
|
{
|
||||||
|
//reg, val, mask
|
||||||
|
0xFF, 0x00, 0xFF,
|
||||||
|
0x92, 0x01, 0xFF,
|
||||||
|
0x93, 0xC1, 0x1F,
|
||||||
|
0x00, 0x00, 0x00
|
||||||
|
};
|
||||||
|
const static uint8_t OV2640_SHARPNESS_LEVEL2[]=
|
||||||
|
{
|
||||||
|
//reg, val, mask
|
||||||
|
0xFF, 0x00, 0xFF,
|
||||||
|
0x92, 0x01, 0xFF,
|
||||||
|
0x93, 0xC2, 0x1F,
|
||||||
|
0x00, 0x00, 0x00
|
||||||
|
};
|
||||||
|
const static uint8_t OV2640_SHARPNESS_LEVEL3[]=
|
||||||
|
{
|
||||||
|
//reg, val, mask
|
||||||
|
0xFF, 0x00, 0xFF,
|
||||||
|
0x92, 0x01, 0xFF,
|
||||||
|
0x93, 0xC4, 0x1F,
|
||||||
|
0x00, 0x00, 0x00
|
||||||
|
};
|
||||||
|
const static uint8_t OV2640_SHARPNESS_LEVEL4[]=
|
||||||
|
{
|
||||||
|
//reg, val, mask
|
||||||
|
0xFF, 0x00, 0xFF,
|
||||||
|
0x92, 0x01, 0xFF,
|
||||||
|
0x93, 0xC8, 0x1F,
|
||||||
|
0x00, 0x00, 0x00
|
||||||
|
};
|
||||||
|
const static uint8_t OV2640_SHARPNESS_LEVEL5[]=
|
||||||
|
{
|
||||||
|
//reg, val, mask
|
||||||
|
0xFF, 0x00, 0xFF,
|
||||||
|
0x92, 0x01, 0xFF,
|
||||||
|
0x93, 0xD0, 0x1F,
|
||||||
|
0x00, 0x00, 0x00
|
||||||
|
};
|
||||||
|
const static uint8_t OV2640_SHARPNESS_LEVEL6[]=
|
||||||
|
{
|
||||||
|
//reg, val, mask
|
||||||
|
0xFF, 0x00, 0xFF,
|
||||||
|
0x92, 0x01, 0xFF,
|
||||||
|
0x93, 0xDF, 0x1F,
|
||||||
|
0x00, 0x00, 0x00
|
||||||
|
};
|
||||||
|
|
||||||
|
const static uint8_t *OV2640_SETTING_SHARPNESS[]=
|
||||||
|
{
|
||||||
|
OV2640_SHARPNESS_LEVEL0, // -3 sharpness
|
||||||
|
OV2640_SHARPNESS_LEVEL1,
|
||||||
|
OV2640_SHARPNESS_LEVEL2,
|
||||||
|
OV2640_SHARPNESS_LEVEL3,
|
||||||
|
OV2640_SHARPNESS_LEVEL4,
|
||||||
|
OV2640_SHARPNESS_LEVEL5,
|
||||||
|
OV2640_SHARPNESS_LEVEL6 // +3 sharpness
|
||||||
|
};
|
||||||
|
|
||||||
|
#define OV2640_MAXLEVEL_SHARPNESS 6
|
||||||
|
|
||||||
|
static int table_mask_write(sensor_t *sensor, const uint8_t* ptab)
|
||||||
|
{
|
||||||
|
uint8_t address;
|
||||||
|
uint8_t value;
|
||||||
|
uint8_t orgval;
|
||||||
|
uint8_t mask;
|
||||||
|
const uint8_t *pdata = ptab;
|
||||||
|
|
||||||
|
if (pdata == NULL)
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (1)
|
||||||
|
{
|
||||||
|
address = *pdata++;
|
||||||
|
value = *pdata++;
|
||||||
|
mask = *pdata++;
|
||||||
|
|
||||||
|
if ((address == 0) && (value == 0) && (mask == 0))
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
sensor->set_reg(sensor, address, mask, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ov2640_enable_auto_sharpness(sensor_t *sensor)
|
||||||
|
{
|
||||||
|
table_mask_write(sensor, OV2640_SHARPNESS_AUTO);
|
||||||
|
sensor->status.sharpness = 0;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ov2640_set_sharpness(sensor_t *sensor, int sharpness)
|
||||||
|
{
|
||||||
|
int sharpness_temp = 0;
|
||||||
|
|
||||||
|
if (sharpness < -3)
|
||||||
|
{
|
||||||
|
sharpness_temp = -3;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sharpness > OV2640_MAXLEVEL_SHARPNESS - 3)
|
||||||
|
{
|
||||||
|
sharpness_temp = OV2640_MAXLEVEL_SHARPNESS - 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
table_mask_write(sensor, OV2640_SHARPNESS_MANUAL);
|
||||||
|
table_mask_write(sensor, OV2640_SETTING_SHARPNESS[sharpness_temp + 3]);
|
||||||
|
|
||||||
|
sensor->status.sharpness = sharpness;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
11
code/components/jomjol_controlcamera/ov2640_sharpness.h
Normal file
11
code/components/jomjol_controlcamera/ov2640_sharpness.h
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifndef OV2640_SHARPNESS_H
|
||||||
|
#define OV2640_SHARPNESS_H
|
||||||
|
|
||||||
|
#include "esp_camera.h"
|
||||||
|
|
||||||
|
int ov2640_enable_auto_sharpness(sensor_t *sensor);
|
||||||
|
int ov2640_set_sharpness(sensor_t *sensor, int sharpness); // -3 to +3, -4 for auto-sharpness
|
||||||
|
|
||||||
|
#endif
|
||||||
@@ -0,0 +1,66 @@
|
|||||||
|
// Workaround - bug in cam library - enable bits are set without using bitwise OR logic -> only latest enable setting is used
|
||||||
|
// Reference: https://esp32.com/viewtopic.php?f=19&t=14376#p93178
|
||||||
|
/* The memory structure is as follows for
|
||||||
|
byte_0 = enable_bits
|
||||||
|
byte_0->bit0 = enable saturation and hue --> OK
|
||||||
|
byte_0->bit1 = enable saturation --> OK
|
||||||
|
byte_0->bit2 = enable brightness and contrast --> OK
|
||||||
|
byte_0->bit3 = enable green -> blue spitial effect (Antique and blunish and greenish and reddish and b&w) enable
|
||||||
|
byte_0->bit4 = anable gray -> red spitial effect (Antique and blunish and greenish and reddish and b&w) enable
|
||||||
|
byte_0->bit5 = remove (UV) in YUV color system
|
||||||
|
byte_0->bit6 = enable negative
|
||||||
|
byte_0->bit7 = remove (Y) in YUV color system
|
||||||
|
byte_1 = saturation1 0-255 --> ?
|
||||||
|
byte_2 = hue 0-255 --> OK
|
||||||
|
byte_3 = saturation2 0-255 --> OK
|
||||||
|
byte_4 = reenter saturation2 in documents --> ?
|
||||||
|
byte_5 = spital effect green -> blue 0-255 --> ?
|
||||||
|
byte_6 = spital effect gray -> red 0-255 --> ?
|
||||||
|
byte_7 = contrast lower byte 0-255 --> OK
|
||||||
|
byte_8 = contrast higher byte 0-255 --> OK
|
||||||
|
byte_9 = brightness 0-255 --> OK
|
||||||
|
byte_10 = if byte_10==4 contrast effective --> ?
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include "esp_camera.h"
|
||||||
|
#include "ov2640_specialEffect.h"
|
||||||
|
|
||||||
|
static const uint8_t special_effects_regs[8][5] = {
|
||||||
|
{0x7C, 0x7D, 0x7C, 0x7D, 0x7D},
|
||||||
|
{0x00, 0X00, 0x05, 0X80, 0X80}, /* no effect */
|
||||||
|
{0x00, 0X40, 0x05, 0X80, 0X80}, /* negative */
|
||||||
|
{0x00, 0X18, 0x05, 0X80, 0X80}, /* black and white */
|
||||||
|
{0x00, 0X18, 0x05, 0X40, 0XC0}, /* reddish */
|
||||||
|
{0x00, 0X18, 0x05, 0X40, 0X40}, /* greenish */
|
||||||
|
{0x00, 0X18, 0x05, 0XA0, 0X40}, /* blue */
|
||||||
|
{0x00, 0X18, 0x05, 0X40, 0XA6}, /* retro */
|
||||||
|
};
|
||||||
|
|
||||||
|
int ov2640_set_special_effect(sensor_t *sensor, int effect)
|
||||||
|
{
|
||||||
|
int ret = 0;
|
||||||
|
effect++;
|
||||||
|
|
||||||
|
if (effect <= 0 || effect > 7)
|
||||||
|
{
|
||||||
|
effect = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
sensor->status.special_effect = effect - 1;
|
||||||
|
|
||||||
|
int registerValue = 0x06; // enable saturation, contrast, brightness
|
||||||
|
registerValue |= special_effects_regs[effect][1];
|
||||||
|
|
||||||
|
// sensor->set_reg(sensor, int reg, int mask, int value)
|
||||||
|
sensor->set_reg(sensor, 0xFF, 0x01, 0x00); // Select DSP bank
|
||||||
|
sensor->set_reg(sensor, special_effects_regs[0][0], 0xFF, 0x00);
|
||||||
|
sensor->set_reg(sensor, special_effects_regs[0][1], 0x5E, registerValue);
|
||||||
|
|
||||||
|
for (int i = 2; i < 5; i++)
|
||||||
|
{
|
||||||
|
sensor->set_reg(sensor, special_effects_regs[0][i], 0xFF, special_effects_regs[effect][i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
10
code/components/jomjol_controlcamera/ov2640_specialEffect.h
Normal file
10
code/components/jomjol_controlcamera/ov2640_specialEffect.h
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifndef OV2640_SPECIALEFFECT_H
|
||||||
|
#define OV2640_SPECIALEFFECT_H
|
||||||
|
|
||||||
|
#include "esp_camera.h"
|
||||||
|
|
||||||
|
int ov2640_set_special_effect(sensor_t *sensor, int effect);
|
||||||
|
|
||||||
|
#endif
|
||||||
@@ -5,45 +5,50 @@
|
|||||||
|
|
||||||
#include "esp_camera.h"
|
#include "esp_camera.h"
|
||||||
#include "ClassControllCamera.h"
|
#include "ClassControllCamera.h"
|
||||||
|
#include "MainFlowControl.h"
|
||||||
|
|
||||||
#include "ClassLogFile.h"
|
#include "ClassLogFile.h"
|
||||||
#include "esp_log.h"
|
#include "esp_log.h"
|
||||||
|
|
||||||
|
#include "basic_auth.h"
|
||||||
|
|
||||||
#include "../../include/defines.h"
|
#include "../../include/defines.h"
|
||||||
|
|
||||||
static const char *TAG = "server_cam";
|
static const char *TAG = "server_cam";
|
||||||
|
|
||||||
|
void PowerResetCamera()
|
||||||
void PowerResetCamera(){
|
{
|
||||||
|
#if CAM_PIN_PWDN == GPIO_NUM_NC // Use reset only if pin is available
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "No power down pin availbale to reset camera");
|
||||||
|
#else
|
||||||
ESP_LOGD(TAG, "Resetting camera by power down line");
|
ESP_LOGD(TAG, "Resetting camera by power down line");
|
||||||
gpio_config_t conf;
|
gpio_config_t conf;
|
||||||
conf.intr_type = GPIO_INTR_DISABLE;
|
conf.intr_type = GPIO_INTR_DISABLE;
|
||||||
conf.pin_bit_mask = 1LL << GPIO_NUM_32;
|
conf.pin_bit_mask = 1LL << CAM_PIN_PWDN;
|
||||||
conf.mode = GPIO_MODE_OUTPUT;
|
conf.mode = GPIO_MODE_OUTPUT;
|
||||||
conf.pull_down_en = GPIO_PULLDOWN_DISABLE;
|
conf.pull_down_en = GPIO_PULLDOWN_DISABLE;
|
||||||
conf.pull_up_en = GPIO_PULLUP_DISABLE;
|
conf.pull_up_en = GPIO_PULLUP_DISABLE;
|
||||||
gpio_config(&conf);
|
gpio_config(&conf);
|
||||||
|
|
||||||
// carefull, logic is inverted compared to reset pin
|
// carefull, logic is inverted compared to reset pin
|
||||||
gpio_set_level(GPIO_NUM_32, 1);
|
gpio_set_level(CAM_PIN_PWDN, 1);
|
||||||
vTaskDelay(1000 / portTICK_PERIOD_MS);
|
vTaskDelay(1000 / portTICK_PERIOD_MS);
|
||||||
gpio_set_level(GPIO_NUM_32, 0);
|
gpio_set_level(CAM_PIN_PWDN, 0);
|
||||||
vTaskDelay(1000 / portTICK_PERIOD_MS);
|
vTaskDelay(1000 / portTICK_PERIOD_MS);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
esp_err_t handler_lightOn(httpd_req_t *req)
|
esp_err_t handler_lightOn(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("handler_lightOn - Start");
|
LogFile.WriteHeapInfo("handler_lightOn - Start");
|
||||||
ESP_LOGD(TAG, "handler_lightOn uri: %s", req->uri);
|
ESP_LOGD(TAG, "handler_lightOn uri: %s", req->uri);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (Camera.getCameraInitSuccessful())
|
if (Camera.getCameraInitSuccessful())
|
||||||
{
|
{
|
||||||
Camera.LightOnOff(true);
|
Camera.LightOnOff(true);
|
||||||
const char* resp_str = (const char*) req->user_ctx;
|
const char *resp_str = (const char *)req->user_ctx;
|
||||||
httpd_resp_send(req, resp_str, strlen(resp_str));
|
httpd_resp_send(req, resp_str, strlen(resp_str));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -52,25 +57,24 @@ esp_err_t handler_lightOn(httpd_req_t *req)
|
|||||||
return ESP_ERR_NOT_FOUND;
|
return ESP_ERR_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("handler_lightOn - Done");
|
LogFile.WriteHeapInfo("handler_lightOn - Done");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
esp_err_t handler_lightOff(httpd_req_t *req)
|
esp_err_t handler_lightOff(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("handler_lightOff - Start");
|
LogFile.WriteHeapInfo("handler_lightOff - Start");
|
||||||
ESP_LOGD(TAG, "handler_lightOff uri: %s", req->uri);
|
ESP_LOGD(TAG, "handler_lightOff uri: %s", req->uri);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (Camera.getCameraInitSuccessful())
|
if (Camera.getCameraInitSuccessful())
|
||||||
{
|
{
|
||||||
Camera.LightOnOff(false);
|
Camera.LightOnOff(false);
|
||||||
const char* resp_str = (const char*) req->user_ctx;
|
const char *resp_str = (const char *)req->user_ctx;
|
||||||
httpd_resp_send(req, resp_str, strlen(resp_str));
|
httpd_resp_send(req, resp_str, strlen(resp_str));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -79,39 +83,40 @@ esp_err_t handler_lightOff(httpd_req_t *req)
|
|||||||
return ESP_ERR_NOT_FOUND;
|
return ESP_ERR_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("handler_lightOff - Done");
|
LogFile.WriteHeapInfo("handler_lightOff - Done");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
esp_err_t handler_capture(httpd_req_t *req)
|
esp_err_t handler_capture(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("handler_capture - Start");
|
LogFile.WriteHeapInfo("handler_capture - Start");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (Camera.getCameraInitSuccessful())
|
if (Camera.getCameraInitSuccessful())
|
||||||
{
|
{
|
||||||
int quality;
|
// If the camera settings were changed by creating a new reference image, they must be reset
|
||||||
framesize_t res;
|
if (CFstatus.changedCameraSettings)
|
||||||
|
{
|
||||||
|
Camera.setSensorDatenFromCCstatus(); // CCstatus >>> Kamera
|
||||||
|
Camera.SetQualityZoomSize(CCstatus.ImageQuality, CCstatus.ImageFrameSize, CCstatus.ImageZoomEnabled, CCstatus.ImageZoomOffsetX, CCstatus.ImageZoomOffsetY, CCstatus.ImageZoomSize, CCstatus.ImageVflip);
|
||||||
|
Camera.LedIntensity = CCstatus.ImageLedIntensity;
|
||||||
|
CFstatus.changedCameraSettings = false;
|
||||||
|
}
|
||||||
|
|
||||||
Camera.GetCameraParameter(req, quality, res);
|
#ifdef DEBUG_DETAIL_ON
|
||||||
|
ESP_LOGD(TAG, "Size: %d, Quality: %d", CCstatus.ImageFrameSize, CCstatus.ImageQuality);
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#endif
|
||||||
ESP_LOGD(TAG, "Size: %d, Quality: %d", res, quality);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
Camera.SetQualitySize(quality, res);
|
|
||||||
|
|
||||||
esp_err_t result;
|
esp_err_t result;
|
||||||
result = Camera.CaptureToHTTP(req);
|
result = Camera.CaptureToHTTP(req);
|
||||||
|
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("handler_capture - Done");
|
LogFile.WriteHeapInfo("handler_capture - Done");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -122,56 +127,61 @@ esp_err_t handler_capture(httpd_req_t *req)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
esp_err_t handler_capture_with_light(httpd_req_t *req)
|
esp_err_t handler_capture_with_light(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("handler_capture_with_light - Start");
|
LogFile.WriteHeapInfo("handler_capture_with_light - Start");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (Camera.getCameraInitSuccessful())
|
if (Camera.getCameraInitSuccessful())
|
||||||
{
|
{
|
||||||
char _query[100];
|
char _query[100];
|
||||||
char _delay[10];
|
char _delay[10];
|
||||||
|
|
||||||
int quality;
|
|
||||||
framesize_t res;
|
|
||||||
int delay = 2500;
|
int delay = 2500;
|
||||||
|
|
||||||
if (httpd_req_get_url_query_str(req, _query, 100) == ESP_OK)
|
if (httpd_req_get_url_query_str(req, _query, 100) == ESP_OK)
|
||||||
{
|
{
|
||||||
ESP_LOGD(TAG, "Query: %s", _query);
|
ESP_LOGD(TAG, "Query: %s", _query);
|
||||||
|
|
||||||
if (httpd_query_key_value(_query, "delay", _delay, 10) == ESP_OK)
|
if (httpd_query_key_value(_query, "delay", _delay, 10) == ESP_OK)
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
ESP_LOGD(TAG, "Delay: %s", _delay);
|
ESP_LOGD(TAG, "Delay: %s", _delay);
|
||||||
#endif
|
#endif
|
||||||
delay = atoi(_delay);
|
delay = atoi(_delay);
|
||||||
|
|
||||||
if (delay < 0)
|
if (delay < 0)
|
||||||
|
{
|
||||||
delay = 0;
|
delay = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Camera.GetCameraParameter(req, quality, res);
|
// If the camera settings were changed by creating a new reference image, they must be reset
|
||||||
|
if (CFstatus.changedCameraSettings)
|
||||||
|
{
|
||||||
|
Camera.setSensorDatenFromCCstatus(); // CCstatus >>> Kamera
|
||||||
|
Camera.SetQualityZoomSize(CCstatus.ImageQuality, CCstatus.ImageFrameSize, CCstatus.ImageZoomEnabled, CCstatus.ImageZoomOffsetX, CCstatus.ImageZoomOffsetY, CCstatus.ImageZoomSize, CCstatus.ImageVflip);
|
||||||
|
Camera.LedIntensity = CCstatus.ImageLedIntensity;
|
||||||
|
CFstatus.changedCameraSettings = false;
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
ESP_LOGD(TAG, "Size: %d, Quality: %d", res, quality);
|
ESP_LOGD(TAG, "Size: %d, Quality: %d", CCstatus.ImageFrameSize, CCstatus.ImageQuality);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
Camera.SetQualitySize(quality, res);
|
|
||||||
Camera.LightOnOff(true);
|
Camera.LightOnOff(true);
|
||||||
const TickType_t xDelay = delay / portTICK_PERIOD_MS;
|
const TickType_t xDelay = delay / portTICK_PERIOD_MS;
|
||||||
vTaskDelay( xDelay );
|
vTaskDelay(xDelay);
|
||||||
|
|
||||||
esp_err_t result;
|
esp_err_t result;
|
||||||
result = Camera.CaptureToHTTP(req);
|
result = Camera.CaptureToHTTP(req);
|
||||||
|
|
||||||
Camera.LightOnOff(false);
|
Camera.LightOnOff(false);
|
||||||
|
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("handler_capture_with_light - Done");
|
LogFile.WriteHeapInfo("handler_capture_with_light - Done");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -182,12 +192,11 @@ esp_err_t handler_capture_with_light(httpd_req_t *req)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
esp_err_t handler_capture_save_to_file(httpd_req_t *req)
|
esp_err_t handler_capture_save_to_file(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("handler_capture_save_to_file - Start");
|
LogFile.WriteHeapInfo("handler_capture_save_to_file - Start");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (Camera.getCameraInitSuccessful())
|
if (Camera.getCameraInitSuccessful())
|
||||||
{
|
{
|
||||||
@@ -197,52 +206,62 @@ esp_err_t handler_capture_save_to_file(httpd_req_t *req)
|
|||||||
char filename[100];
|
char filename[100];
|
||||||
std::string fn = "/sdcard/";
|
std::string fn = "/sdcard/";
|
||||||
|
|
||||||
|
|
||||||
int quality;
|
|
||||||
framesize_t res;
|
|
||||||
|
|
||||||
if (httpd_req_get_url_query_str(req, _query, 100) == ESP_OK)
|
if (httpd_req_get_url_query_str(req, _query, 100) == ESP_OK)
|
||||||
{
|
{
|
||||||
ESP_LOGD(TAG, "Query: %s", _query);
|
ESP_LOGD(TAG, "Query: %s", _query);
|
||||||
|
|
||||||
if (httpd_query_key_value(_query, "filename", filename, 100) == ESP_OK)
|
if (httpd_query_key_value(_query, "filename", filename, 100) == ESP_OK)
|
||||||
{
|
{
|
||||||
fn.append(filename);
|
fn.append(filename);
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
ESP_LOGD(TAG, "Filename: %s", fn.c_str());
|
ESP_LOGD(TAG, "Filename: %s", fn.c_str());
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
fn.append("noname.jpg");
|
fn.append("noname.jpg");
|
||||||
|
}
|
||||||
|
|
||||||
if (httpd_query_key_value(_query, "delay", _delay, 10) == ESP_OK)
|
if (httpd_query_key_value(_query, "delay", _delay, 10) == ESP_OK)
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
ESP_LOGD(TAG, "Delay: %s", _delay);
|
ESP_LOGD(TAG, "Delay: %s", _delay);
|
||||||
#endif
|
#endif
|
||||||
delay = atoi(_delay);
|
delay = atoi(_delay);
|
||||||
|
|
||||||
if (delay < 0)
|
if (delay < 0)
|
||||||
|
{
|
||||||
delay = 0;
|
delay = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
fn.append("noname.jpg");
|
fn.append("noname.jpg");
|
||||||
|
}
|
||||||
|
|
||||||
Camera.GetCameraParameter(req, quality, res);
|
// If the camera settings were changed by creating a new reference image, they must be reset
|
||||||
#ifdef DEBUG_DETAIL_ON
|
if (CFstatus.changedCameraSettings)
|
||||||
ESP_LOGD(TAG, "Size: %d, Quality: %d", res, quality);
|
{
|
||||||
#endif
|
Camera.setSensorDatenFromCCstatus(); // CCstatus >>> Kamera
|
||||||
Camera.SetQualitySize(quality, res);
|
Camera.SetQualityZoomSize(CCstatus.ImageQuality, CCstatus.ImageFrameSize, CCstatus.ImageZoomEnabled, CCstatus.ImageZoomOffsetX, CCstatus.ImageZoomOffsetY, CCstatus.ImageZoomSize, CCstatus.ImageVflip);
|
||||||
|
Camera.LedIntensity = CCstatus.ImageLedIntensity;
|
||||||
|
CFstatus.changedCameraSettings = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef DEBUG_DETAIL_ON
|
||||||
|
ESP_LOGD(TAG, "Size: %d, Quality: %d", CCstatus.ImageFrameSize, CCstatus.ImageQuality);
|
||||||
|
#endif
|
||||||
|
|
||||||
esp_err_t result;
|
esp_err_t result;
|
||||||
result = Camera.CaptureToFile(fn, delay);
|
result = Camera.CaptureToFile(fn, delay);
|
||||||
|
|
||||||
const char* resp_str = (const char*) fn.c_str();
|
const char *resp_str = (const char *)fn.c_str();
|
||||||
httpd_resp_send(req, resp_str, strlen(resp_str));
|
httpd_resp_send(req, resp_str, strlen(resp_str));
|
||||||
|
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("handler_capture_save_to_file - Done");
|
LogFile.WriteHeapInfo("handler_capture_save_to_file - Done");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
@@ -253,38 +272,37 @@ esp_err_t handler_capture_save_to_file(httpd_req_t *req)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void register_server_camera_uri(httpd_handle_t server)
|
void register_server_camera_uri(httpd_handle_t server)
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
ESP_LOGI(TAG, "server_part_camera - Registering URI handlers");
|
ESP_LOGI(TAG, "server_part_camera - Registering URI handlers");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
httpd_uri_t camuri = { };
|
httpd_uri_t camuri = {};
|
||||||
camuri.method = HTTP_GET;
|
camuri.method = HTTP_GET;
|
||||||
|
|
||||||
camuri.uri = "/lighton";
|
camuri.uri = "/lighton";
|
||||||
camuri.handler = handler_lightOn;
|
camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_lightOn);
|
||||||
camuri.user_ctx = (void*) "Light On";
|
camuri.user_ctx = (void *)"Light On";
|
||||||
httpd_register_uri_handler(server, &camuri);
|
httpd_register_uri_handler(server, &camuri);
|
||||||
|
|
||||||
camuri.uri = "/lightoff";
|
camuri.uri = "/lightoff";
|
||||||
camuri.handler = handler_lightOff;
|
camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_lightOff);
|
||||||
camuri.user_ctx = (void*) "Light Off";
|
camuri.user_ctx = (void *)"Light Off";
|
||||||
httpd_register_uri_handler(server, &camuri);
|
httpd_register_uri_handler(server, &camuri);
|
||||||
|
|
||||||
camuri.uri = "/capture";
|
camuri.uri = "/capture";
|
||||||
camuri.handler = handler_capture;
|
camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_capture);
|
||||||
camuri.user_ctx = NULL;
|
camuri.user_ctx = NULL;
|
||||||
httpd_register_uri_handler(server, &camuri);
|
httpd_register_uri_handler(server, &camuri);
|
||||||
|
|
||||||
camuri.uri = "/capture_with_flashlight";
|
camuri.uri = "/capture_with_flashlight";
|
||||||
camuri.handler = handler_capture_with_light;
|
camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_capture_with_light);
|
||||||
camuri.user_ctx = NULL;
|
camuri.user_ctx = NULL;
|
||||||
httpd_register_uri_handler(server, &camuri);
|
httpd_register_uri_handler(server, &camuri);
|
||||||
|
|
||||||
camuri.uri = "/save";
|
camuri.uri = "/save";
|
||||||
camuri.handler = handler_capture_save_to_file;
|
camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_capture_save_to_file);
|
||||||
camuri.user_ctx = NULL;
|
camuri.user_ctx = NULL;
|
||||||
httpd_register_uri_handler(server, &camuri);
|
httpd_register_uri_handler(server, &camuri);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,7 +10,6 @@
|
|||||||
//#include "ClassControllCamera.h"
|
//#include "ClassControllCamera.h"
|
||||||
|
|
||||||
void register_server_camera_uri(httpd_handle_t server);
|
void register_server_camera_uri(httpd_handle_t server);
|
||||||
|
|
||||||
void PowerResetCamera();
|
void PowerResetCamera();
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
FILE(GLOB_RECURSE app_sources ${CMAKE_CURRENT_SOURCE_DIR}/*.*)
|
FILE(GLOB_RECURSE app_sources ${CMAKE_CURRENT_SOURCE_DIR}/*.*)
|
||||||
|
|
||||||
idf_component_register(SRCS ${app_sources}
|
idf_component_register(SRCS ${app_sources}
|
||||||
INCLUDE_DIRS "." "../../include"
|
INCLUDE_DIRS "." "../../include" "miniz"
|
||||||
REQUIRES tflite-lib esp_http_server app_update esp_http_client nvs_flash jomjol_tfliteclass jomjol_flowcontroll spiffs jomjol_helper jomjol_controlGPIO miniz)
|
REQUIRES vfs esp_http_server app_update esp_http_client nvs_flash jomjol_tfliteclass jomjol_flowcontroll spiffs jomjol_helper jomjol_controlGPIO)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
226
code/components/jomjol_fileserver_ota/md5.cpp
Normal file
226
code/components/jomjol_fileserver_ota/md5.cpp
Normal file
@@ -0,0 +1,226 @@
|
|||||||
|
/* Src: https://github.com/Zunawe/md5-c, commit: f3529b6
|
||||||
|
* License: Unlicense */
|
||||||
|
/*
|
||||||
|
* Derived from the RSA Data Security, Inc. MD5 Message-Digest Algorithm
|
||||||
|
* and modified slightly to be functionally identical but condensed into control structures.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "md5.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Constants defined by the MD5 algorithm
|
||||||
|
*/
|
||||||
|
#define A 0x67452301
|
||||||
|
#define B 0xefcdab89
|
||||||
|
#define C 0x98badcfe
|
||||||
|
#define D 0x10325476
|
||||||
|
|
||||||
|
static uint32_t S[] = {7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22,
|
||||||
|
5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20,
|
||||||
|
4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23,
|
||||||
|
6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21};
|
||||||
|
|
||||||
|
static uint32_t K[] = {0xd76aa478, 0xe8c7b756, 0x242070db, 0xc1bdceee,
|
||||||
|
0xf57c0faf, 0x4787c62a, 0xa8304613, 0xfd469501,
|
||||||
|
0x698098d8, 0x8b44f7af, 0xffff5bb1, 0x895cd7be,
|
||||||
|
0x6b901122, 0xfd987193, 0xa679438e, 0x49b40821,
|
||||||
|
0xf61e2562, 0xc040b340, 0x265e5a51, 0xe9b6c7aa,
|
||||||
|
0xd62f105d, 0x02441453, 0xd8a1e681, 0xe7d3fbc8,
|
||||||
|
0x21e1cde6, 0xc33707d6, 0xf4d50d87, 0x455a14ed,
|
||||||
|
0xa9e3e905, 0xfcefa3f8, 0x676f02d9, 0x8d2a4c8a,
|
||||||
|
0xfffa3942, 0x8771f681, 0x6d9d6122, 0xfde5380c,
|
||||||
|
0xa4beea44, 0x4bdecfa9, 0xf6bb4b60, 0xbebfbc70,
|
||||||
|
0x289b7ec6, 0xeaa127fa, 0xd4ef3085, 0x04881d05,
|
||||||
|
0xd9d4d039, 0xe6db99e5, 0x1fa27cf8, 0xc4ac5665,
|
||||||
|
0xf4292244, 0x432aff97, 0xab9423a7, 0xfc93a039,
|
||||||
|
0x655b59c3, 0x8f0ccc92, 0xffeff47d, 0x85845dd1,
|
||||||
|
0x6fa87e4f, 0xfe2ce6e0, 0xa3014314, 0x4e0811a1,
|
||||||
|
0xf7537e82, 0xbd3af235, 0x2ad7d2bb, 0xeb86d391};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Padding used to make the size (in bits) of the input congruent to 448 mod 512
|
||||||
|
*/
|
||||||
|
static uint8_t PADDING[] = {0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
||||||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Bit-manipulation functions defined by the MD5 algorithm
|
||||||
|
*/
|
||||||
|
#define F(X, Y, Z) ((X & Y) | (~X & Z))
|
||||||
|
#define G(X, Y, Z) ((X & Z) | (Y & ~Z))
|
||||||
|
#define H(X, Y, Z) (X ^ Y ^ Z)
|
||||||
|
#define I(X, Y, Z) (Y ^ (X | ~Z))
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Rotates a 32-bit word left by n bits
|
||||||
|
*/
|
||||||
|
uint32_t rotateLeft(uint32_t x, uint32_t n){
|
||||||
|
return (x << n) | (x >> (32 - n));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Initialize a context
|
||||||
|
*/
|
||||||
|
void md5Init(MD5Context *ctx){
|
||||||
|
ctx->size = (uint64_t)0;
|
||||||
|
|
||||||
|
ctx->buffer[0] = (uint32_t)A;
|
||||||
|
ctx->buffer[1] = (uint32_t)B;
|
||||||
|
ctx->buffer[2] = (uint32_t)C;
|
||||||
|
ctx->buffer[3] = (uint32_t)D;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Add some amount of input to the context
|
||||||
|
*
|
||||||
|
* If the input fills out a block of 512 bits, apply the algorithm (md5Step)
|
||||||
|
* and save the result in the buffer. Also updates the overall size.
|
||||||
|
*/
|
||||||
|
void md5Update(MD5Context *ctx, uint8_t *input_buffer, size_t input_len){
|
||||||
|
uint32_t input[16];
|
||||||
|
unsigned int offset = ctx->size % 64;
|
||||||
|
ctx->size += (uint64_t)input_len;
|
||||||
|
|
||||||
|
// Copy each byte in input_buffer into the next space in our context input
|
||||||
|
for(unsigned int i = 0; i < input_len; ++i){
|
||||||
|
ctx->input[offset++] = (uint8_t)*(input_buffer + i);
|
||||||
|
|
||||||
|
// If we've filled our context input, copy it into our local array input
|
||||||
|
// then reset the offset to 0 and fill in a new buffer.
|
||||||
|
// Every time we fill out a chunk, we run it through the algorithm
|
||||||
|
// to enable some back and forth between cpu and i/o
|
||||||
|
if(offset % 64 == 0){
|
||||||
|
for(unsigned int j = 0; j < 16; ++j){
|
||||||
|
// Convert to little-endian
|
||||||
|
// The local variable `input` our 512-bit chunk separated into 32-bit words
|
||||||
|
// we can use in calculations
|
||||||
|
input[j] = (uint32_t)(ctx->input[(j * 4) + 3]) << 24 |
|
||||||
|
(uint32_t)(ctx->input[(j * 4) + 2]) << 16 |
|
||||||
|
(uint32_t)(ctx->input[(j * 4) + 1]) << 8 |
|
||||||
|
(uint32_t)(ctx->input[(j * 4)]);
|
||||||
|
}
|
||||||
|
md5Step(ctx->buffer, input);
|
||||||
|
offset = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Pad the current input to get to 448 bytes, append the size in bits to the very end,
|
||||||
|
* and save the result of the final iteration into digest.
|
||||||
|
*/
|
||||||
|
void md5Finalize(MD5Context *ctx){
|
||||||
|
uint32_t input[16];
|
||||||
|
unsigned int offset = ctx->size % 64;
|
||||||
|
unsigned int padding_length = offset < 56 ? 56 - offset : (56 + 64) - offset;
|
||||||
|
|
||||||
|
// Fill in the padding and undo the changes to size that resulted from the update
|
||||||
|
md5Update(ctx, PADDING, padding_length);
|
||||||
|
ctx->size -= (uint64_t)padding_length;
|
||||||
|
|
||||||
|
// Do a final update (internal to this function)
|
||||||
|
// Last two 32-bit words are the two halves of the size (converted from bytes to bits)
|
||||||
|
for(unsigned int j = 0; j < 14; ++j){
|
||||||
|
input[j] = (uint32_t)(ctx->input[(j * 4) + 3]) << 24 |
|
||||||
|
(uint32_t)(ctx->input[(j * 4) + 2]) << 16 |
|
||||||
|
(uint32_t)(ctx->input[(j * 4) + 1]) << 8 |
|
||||||
|
(uint32_t)(ctx->input[(j * 4)]);
|
||||||
|
}
|
||||||
|
input[14] = (uint32_t)(ctx->size * 8);
|
||||||
|
input[15] = (uint32_t)((ctx->size * 8) >> 32);
|
||||||
|
|
||||||
|
md5Step(ctx->buffer, input);
|
||||||
|
|
||||||
|
// Move the result into digest (convert from little-endian)
|
||||||
|
for(unsigned int i = 0; i < 4; ++i){
|
||||||
|
ctx->digest[(i * 4) + 0] = (uint8_t)((ctx->buffer[i] & 0x000000FF));
|
||||||
|
ctx->digest[(i * 4) + 1] = (uint8_t)((ctx->buffer[i] & 0x0000FF00) >> 8);
|
||||||
|
ctx->digest[(i * 4) + 2] = (uint8_t)((ctx->buffer[i] & 0x00FF0000) >> 16);
|
||||||
|
ctx->digest[(i * 4) + 3] = (uint8_t)((ctx->buffer[i] & 0xFF000000) >> 24);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Step on 512 bits of input with the main MD5 algorithm.
|
||||||
|
*/
|
||||||
|
void md5Step(uint32_t *buffer, uint32_t *input){
|
||||||
|
uint32_t AA = buffer[0];
|
||||||
|
uint32_t BB = buffer[1];
|
||||||
|
uint32_t CC = buffer[2];
|
||||||
|
uint32_t DD = buffer[3];
|
||||||
|
|
||||||
|
uint32_t E;
|
||||||
|
|
||||||
|
unsigned int j;
|
||||||
|
|
||||||
|
for(unsigned int i = 0; i < 64; ++i){
|
||||||
|
switch(i / 16){
|
||||||
|
case 0:
|
||||||
|
E = F(BB, CC, DD);
|
||||||
|
j = i;
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
E = G(BB, CC, DD);
|
||||||
|
j = ((i * 5) + 1) % 16;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
E = H(BB, CC, DD);
|
||||||
|
j = ((i * 3) + 5) % 16;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
E = I(BB, CC, DD);
|
||||||
|
j = (i * 7) % 16;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t temp = DD;
|
||||||
|
DD = CC;
|
||||||
|
CC = BB;
|
||||||
|
BB = BB + rotateLeft(AA + E + K[i] + input[j], S[i]);
|
||||||
|
AA = temp;
|
||||||
|
}
|
||||||
|
|
||||||
|
buffer[0] += AA;
|
||||||
|
buffer[1] += BB;
|
||||||
|
buffer[2] += CC;
|
||||||
|
buffer[3] += DD;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Functions that run the algorithm on the provided input and put the digest into result.
|
||||||
|
* result should be able to store 16 bytes.
|
||||||
|
*/
|
||||||
|
void md5String(char *input, uint8_t *result){
|
||||||
|
MD5Context ctx;
|
||||||
|
md5Init(&ctx);
|
||||||
|
md5Update(&ctx, (uint8_t *)input, strlen(input));
|
||||||
|
md5Finalize(&ctx);
|
||||||
|
|
||||||
|
memcpy(result, ctx.digest, 16);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void md5File(FILE *file, uint8_t *result){
|
||||||
|
void *input_buffer = malloc(1024);
|
||||||
|
size_t input_size = 0;
|
||||||
|
|
||||||
|
MD5Context ctx;
|
||||||
|
md5Init(&ctx);
|
||||||
|
|
||||||
|
while((input_size = fread(input_buffer, 1, 1024, file)) > 0){
|
||||||
|
md5Update(&ctx, (uint8_t *)input_buffer, input_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
md5Finalize(&ctx);
|
||||||
|
|
||||||
|
free(input_buffer);
|
||||||
|
|
||||||
|
memcpy(result, ctx.digest, 16);
|
||||||
|
}
|
||||||
28
code/components/jomjol_fileserver_ota/md5.h
Normal file
28
code/components/jomjol_fileserver_ota/md5.h
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
/* Src: https://github.com/Zunawe/md5-c, commit: f3529b6
|
||||||
|
* License: Unlicense */
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifndef MD5_H
|
||||||
|
#define MD5_H
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
typedef struct{
|
||||||
|
uint64_t size; // Size of input in bytes
|
||||||
|
uint32_t buffer[4]; // Current accumulation of hash
|
||||||
|
uint8_t input[64]; // Input to be used in the next step
|
||||||
|
uint8_t digest[16]; // Result of algorithm
|
||||||
|
}MD5Context;
|
||||||
|
|
||||||
|
void md5Init(MD5Context *ctx);
|
||||||
|
void md5Update(MD5Context *ctx, uint8_t *input, size_t input_len);
|
||||||
|
void md5Finalize(MD5Context *ctx);
|
||||||
|
void md5Step(uint32_t *buffer, uint32_t *input);
|
||||||
|
|
||||||
|
void md5String(char *input, uint8_t *result);
|
||||||
|
void md5File(FILE *file, uint8_t *result);
|
||||||
|
|
||||||
|
#endif // MD5_H
|
||||||
@@ -153,7 +153,7 @@
|
|||||||
/*#define MINIZ_NO_MALLOC */
|
/*#define MINIZ_NO_MALLOC */
|
||||||
|
|
||||||
#ifdef MINIZ_NO_INFLATE_APIS
|
#ifdef MINIZ_NO_INFLATE_APIS
|
||||||
#define MINIZ_NO_ARCHIVE_APIS
|
//#define MINIZ_NO_ARCHIVE_APIS
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef MINIZ_NO_DEFLATE_APIS
|
#ifdef MINIZ_NO_DEFLATE_APIS
|
||||||
@@ -4,5 +4,5 @@ It should be possible to include the repo directly as a submodule, how ever it c
|
|||||||
- https://github.com/richgel999/miniz/issues/145
|
- https://github.com/richgel999/miniz/issues/145
|
||||||
- https://github.com/espressif/esptool/pull/500#issuecomment-574879468
|
- https://github.com/espressif/esptool/pull/500#issuecomment-574879468
|
||||||
|
|
||||||
For sumplicity we therefore use the release files as suggested in the readme.
|
For simplicity we therefore use the release files as suggested in the readme.
|
||||||
Additionally we added the CMakeLists.txt and this readme.
|
Additionally we added the CMakeLists.txt and this readme.
|
||||||
@@ -6,11 +6,8 @@
|
|||||||
software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
|
software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
|
||||||
CONDITIONS OF ANY KIND, either express or implied.
|
CONDITIONS OF ANY KIND, either express or implied.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
#include "server_file.h"
|
#include "server_file.h"
|
||||||
|
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
@@ -36,9 +33,10 @@ extern "C" {
|
|||||||
#include "../../include/defines.h"
|
#include "../../include/defines.h"
|
||||||
#include "ClassLogFile.h"
|
#include "ClassLogFile.h"
|
||||||
|
|
||||||
#include "server_tflite.h"
|
#include "MainFlowControl.h"
|
||||||
|
|
||||||
#include "server_help.h"
|
#include "server_help.h"
|
||||||
|
#include "md5.h"
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
#include "interface_mqtt.h"
|
#include "interface_mqtt.h"
|
||||||
#endif //ENABLE_MQTT
|
#endif //ENABLE_MQTT
|
||||||
@@ -46,7 +44,7 @@ extern "C" {
|
|||||||
|
|
||||||
#include "Helper.h"
|
#include "Helper.h"
|
||||||
#include "miniz.h"
|
#include "miniz.h"
|
||||||
|
#include "basic_auth.h"
|
||||||
|
|
||||||
static const char *TAG = "OTA FILE";
|
static const char *TAG = "OTA FILE";
|
||||||
|
|
||||||
@@ -58,23 +56,20 @@ struct file_server_data {
|
|||||||
char scratch[SERVER_FILER_SCRATCH_BUFSIZE];
|
char scratch[SERVER_FILER_SCRATCH_BUFSIZE];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <dirent.h>
|
#include <dirent.h>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
string SUFFIX_ZW = "_0xge";
|
string SUFFIX_ZW = "_tmp";
|
||||||
|
|
||||||
|
|
||||||
static esp_err_t send_logfile(httpd_req_t *req, bool send_full_file);
|
static esp_err_t send_logfile(httpd_req_t *req, bool send_full_file);
|
||||||
static esp_err_t send_datafile(httpd_req_t *req, bool send_full_file);
|
static esp_err_t send_datafile(httpd_req_t *req, bool send_full_file);
|
||||||
|
|
||||||
|
|
||||||
esp_err_t get_numbers_file_handler(httpd_req_t *req)
|
esp_err_t get_numbers_file_handler(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
std::string ret = tfliteflow.getNumbersName();
|
std::string ret = flowctrl.getNumbersName();
|
||||||
|
|
||||||
// ESP_LOGI(TAG, "Result get_numbers_file_handler: %s", ret.c_str());
|
// ESP_LOGI(TAG, "Result get_numbers_file_handler: %s", ret.c_str());
|
||||||
|
|
||||||
@@ -87,7 +82,6 @@ esp_err_t get_numbers_file_handler(httpd_req_t *req)
|
|||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
esp_err_t get_data_file_handler(httpd_req_t *req)
|
esp_err_t get_data_file_handler(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
struct dirent *entry;
|
struct dirent *entry;
|
||||||
@@ -131,7 +125,6 @@ esp_err_t get_data_file_handler(httpd_req_t *req)
|
|||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
esp_err_t get_tflite_file_handler(httpd_req_t *req)
|
esp_err_t get_tflite_file_handler(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
struct dirent *entry;
|
struct dirent *entry;
|
||||||
@@ -175,12 +168,11 @@ esp_err_t get_tflite_file_handler(httpd_req_t *req)
|
|||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Send HTTP response with a run-time generated html consisting of
|
/* Send HTTP response with a run-time generated html consisting of
|
||||||
* a list of all files and folders under the requested path.
|
* a list of all files and folders under the requested path.
|
||||||
* In case of SPIFFS this returns empty list when path is any
|
* In case of SPIFFS this returns empty list when path is any
|
||||||
* string other than '/', since SPIFFS doesn't support directories */
|
* string other than '/', since SPIFFS doesn't support directories */
|
||||||
static esp_err_t http_resp_dir_html(httpd_req_t *req, const char *dirpath, const char* uripath, bool readonly)
|
static esp_err_t http_resp_dir_html(httpd_req_t *req, const char *dirpath, const char *uripath, bool readonly)
|
||||||
{
|
{
|
||||||
char entrypath[FILE_PATH_MAX];
|
char entrypath[FILE_PATH_MAX];
|
||||||
char entrysize[16];
|
char entrysize[16];
|
||||||
@@ -192,82 +184,85 @@ static esp_err_t http_resp_dir_html(httpd_req_t *req, const char *dirpath, const
|
|||||||
char dirpath_corrected[FILE_PATH_MAX];
|
char dirpath_corrected[FILE_PATH_MAX];
|
||||||
strcpy(dirpath_corrected, dirpath);
|
strcpy(dirpath_corrected, dirpath);
|
||||||
|
|
||||||
file_server_data * server_data = (file_server_data *) req->user_ctx;
|
file_server_data *server_data = (file_server_data *)req->user_ctx;
|
||||||
if ((strlen(dirpath_corrected)-1) > strlen(server_data->base_path)) // if dirpath is not mountpoint, the last "\" needs to be removed
|
|
||||||
dirpath_corrected[strlen(dirpath_corrected)-1] = '\0';
|
|
||||||
|
|
||||||
DIR *dir = opendir(dirpath_corrected);
|
if ((strlen(dirpath_corrected) - 1) > strlen(server_data->base_path)) {
|
||||||
|
// if dirpath is not mountpoint, the last "\" needs to be removed
|
||||||
|
dirpath_corrected[strlen(dirpath_corrected) - 1] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
DIR *pdir = opendir(dirpath_corrected);
|
||||||
|
|
||||||
const size_t dirpath_len = strlen(dirpath);
|
const size_t dirpath_len = strlen(dirpath);
|
||||||
ESP_LOGD(TAG, "Dirpath: <%s>, Pathlength: %d", dirpath, dirpath_len);
|
ESP_LOGD(TAG, "Dirpath: <%s>, Pathlength: %d", dirpath, dirpath_len);
|
||||||
|
|
||||||
/* Retrieve the base path of file storage to construct the full path */
|
// Retrieve the base path of file storage to construct the full path
|
||||||
strlcpy(entrypath, dirpath, sizeof(entrypath));
|
strlcpy(entrypath, dirpath, sizeof(entrypath));
|
||||||
ESP_LOGD(TAG, "entrypath: <%s>", entrypath);
|
ESP_LOGD(TAG, "entrypath: <%s>", entrypath);
|
||||||
|
|
||||||
if (!dir) {
|
if (!pdir) {
|
||||||
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to stat dir: " + std::string(dirpath) + "!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to stat dir: " + std::string(dirpath) + "!");
|
||||||
/* Respond with 404 Not Found */
|
// Respond with 404 Not Found
|
||||||
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
|
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
|
|
||||||
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
||||||
|
|
||||||
/* Send HTML file header */
|
// Send HTML file header
|
||||||
httpd_resp_sendstr_chunk(req, "<!DOCTYPE html><html><body>");
|
httpd_resp_sendstr_chunk(req, "<!DOCTYPE html><html lang=\"en\" xml:lang=\"en\"><head>");
|
||||||
|
httpd_resp_sendstr_chunk(req, "<link href=\"/file_server.css\" rel=\"stylesheet\">");
|
||||||
|
httpd_resp_sendstr_chunk(req, "<link href=\"/firework.css\" rel=\"stylesheet\">");
|
||||||
|
httpd_resp_sendstr_chunk(req, "<script type=\"text/javascript\" src=\"/jquery-3.6.0.min.js\"></script>");
|
||||||
|
httpd_resp_sendstr_chunk(req, "<script type=\"text/javascript\" src=\"/firework.js\"></script></head>");
|
||||||
|
|
||||||
/////////////////////////////////////////////////
|
httpd_resp_sendstr_chunk(req, "<body>");
|
||||||
if (!readonly) {
|
|
||||||
FILE *fd = fopen("/sdcard/html/file_server.html", "r");
|
httpd_resp_sendstr_chunk(req, "<table class=\"fixed\" border=\"0\" width=100% style=\"font-family: arial\">");
|
||||||
char *chunk = ((struct file_server_data *)req->user_ctx)->scratch;
|
httpd_resp_sendstr_chunk(req, "<tr><td style=\"vertical-align: top;width: 300px;\"><h2>Fileserver</h2></td>"
|
||||||
size_t chunksize;
|
"<td rowspan=\"2\"><table border=\"0\" style=\"width:100%\"><tr><td style=\"width:80px\">"
|
||||||
do {
|
"<label for=\"newfile\">Source</label></td><td colspan=\"2\">"
|
||||||
chunksize = fread(chunk, 1, SERVER_FILER_SCRATCH_BUFSIZE, fd);
|
"<input id=\"newfile\" type=\"file\" onchange=\"setpath()\" style=\"width:100%;\"></td></tr>"
|
||||||
// ESP_LOGD(TAG, "Chunksize %d", chunksize);
|
"<tr><td><label for=\"filepath\">Destination</label></td><td>"
|
||||||
if (chunksize > 0){
|
"<input id=\"filepath\" type=\"text\" style=\"width:94%;\"></td><td>"
|
||||||
if (httpd_resp_send_chunk(req, chunk, chunksize) != ESP_OK) {
|
"<button id=\"upload\" type=\"button\" class=\"button\" onclick=\"upload()\">Upload</button></td></tr>"
|
||||||
fclose(fd);
|
"</table></td></tr><tr></tr><tr><td colspan=\"2\">"
|
||||||
ESP_LOGE(TAG, "File sending failed!");
|
"<button style=\"font-size:16px; padding: 5px 10px\" id=\"dirup\" type=\"button\" onclick=\"dirup()\""
|
||||||
return ESP_FAIL;
|
"disabled>🡹 Directory up</button><span style=\"padding-left:15px\" id=\"currentpath\">"
|
||||||
}
|
"</span></td></tr>");
|
||||||
}
|
httpd_resp_sendstr_chunk(req, "</table>");
|
||||||
} while (chunksize != 0);
|
|
||||||
fclose(fd);
|
httpd_resp_sendstr_chunk(req, "<script type=\"text/javascript\" src=\"/file_server.js\"></script>");
|
||||||
// ESP_LOGI(TAG, "File sending complete");
|
httpd_resp_sendstr_chunk(req, "<script type=\"text/javascript\">initFileServer();</script>");
|
||||||
}
|
|
||||||
///////////////////////////////
|
|
||||||
|
|
||||||
std::string _zw = std::string(dirpath);
|
std::string _zw = std::string(dirpath);
|
||||||
_zw = _zw.substr(8, _zw.length() - 8);
|
_zw = _zw.substr(8, _zw.length() - 8);
|
||||||
_zw = "/delete/" + _zw + "?task=deldircontent";
|
_zw = "/delete/" + _zw + "?task=deldircontent";
|
||||||
|
|
||||||
|
// Send file-list table definition and column labels
|
||||||
/* Send file-list table definition and column labels */
|
httpd_resp_sendstr_chunk(req, "<table id=\"files_table\">"
|
||||||
httpd_resp_sendstr_chunk(req,
|
"<col width=\"800px\"><col width=\"300px\"><col width=\"300px\"><col width=\"100px\">"
|
||||||
"<table id=\"files_table\">"
|
|
||||||
"<col width=\"800px\" /><col width=\"300px\" /><col width=\"300px\" /><col width=\"100px\" />"
|
|
||||||
"<thead><tr><th>Name</th><th>Type</th><th>Size</th>");
|
"<thead><tr><th>Name</th><th>Type</th><th>Size</th>");
|
||||||
|
|
||||||
if (!readonly) {
|
if (!readonly) {
|
||||||
httpd_resp_sendstr_chunk(req, "<th>"
|
httpd_resp_sendstr_chunk(req, "<th><form method=\"post\" action=\"");
|
||||||
"<form method=\"post\" action=\"");
|
|
||||||
httpd_resp_sendstr_chunk(req, _zw.c_str());
|
httpd_resp_sendstr_chunk(req, _zw.c_str());
|
||||||
httpd_resp_sendstr_chunk(req,
|
httpd_resp_sendstr_chunk(req, "\"><button type=\"submit\">DELETE ALL!</button></form></th></tr>");
|
||||||
"\"><button type=\"submit\">DELETE ALL!</button></form>"
|
|
||||||
"</th></tr>");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
httpd_resp_sendstr_chunk(req, "</thead><tbody>\n");
|
httpd_resp_sendstr_chunk(req, "</thead><tbody>\n");
|
||||||
|
|
||||||
/* Iterate over all files / folders and fetch their names and sizes */
|
// Iterate over all files / folders and fetch their names and sizes
|
||||||
while ((entry = readdir(dir)) != NULL) {
|
while ((entry = readdir(pdir)) != NULL) {
|
||||||
if (strcmp("wlan.ini", entry->d_name) != 0 ) // wlan.ini soll nicht angezeigt werden!
|
// wlan.ini soll nicht angezeigt werden!
|
||||||
{
|
if (strcmp("wlan.ini", entry->d_name) != 0) {
|
||||||
entrytype = (entry->d_type == DT_DIR ? "directory" : "file");
|
entrytype = (entry->d_type == DT_DIR ? "directory" : "file");
|
||||||
|
|
||||||
strlcpy(entrypath + dirpath_len, entry->d_name, sizeof(entrypath) - dirpath_len);
|
strlcpy(entrypath + dirpath_len, entry->d_name, sizeof(entrypath) - dirpath_len);
|
||||||
ESP_LOGD(TAG, "Entrypath: %s", entrypath);
|
ESP_LOGD(TAG, "Entrypath: %s", entrypath);
|
||||||
|
|
||||||
if (stat(entrypath, &entry_stat) == -1) {
|
if (stat(entrypath, &entry_stat) == -1) {
|
||||||
ESP_LOGE(TAG, "Failed to stat %s: %s", entrytype, entry->d_name);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to stat " + std::string(entrytype) + ": " + std::string(entry->d_name));
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -283,22 +278,25 @@ static esp_err_t http_resp_dir_html(httpd_req_t *req, const char *dirpath, const
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGI(TAG, "Found %s: %s (%s bytes)", entrytype, entry->d_name, entrysize);
|
ESP_LOGD(TAG, "Found %s: %s (%s bytes)", entrytype, entry->d_name, entrysize);
|
||||||
|
|
||||||
/* Send chunk of HTML file containing table entries with file name and size */
|
// Send chunk of HTML file containing table entries with file name and size
|
||||||
httpd_resp_sendstr_chunk(req, "<tr><td><a href=\"");
|
httpd_resp_sendstr_chunk(req, "<tr><td><a href=\"");
|
||||||
httpd_resp_sendstr_chunk(req, "/fileserver");
|
httpd_resp_sendstr_chunk(req, "/fileserver");
|
||||||
httpd_resp_sendstr_chunk(req, uripath);
|
httpd_resp_sendstr_chunk(req, uripath);
|
||||||
httpd_resp_sendstr_chunk(req, entry->d_name);
|
httpd_resp_sendstr_chunk(req, entry->d_name);
|
||||||
|
|
||||||
if (entry->d_type == DT_DIR) {
|
if (entry->d_type == DT_DIR) {
|
||||||
httpd_resp_sendstr_chunk(req, "/");
|
httpd_resp_sendstr_chunk(req, "/");
|
||||||
}
|
}
|
||||||
|
|
||||||
httpd_resp_sendstr_chunk(req, "\">");
|
httpd_resp_sendstr_chunk(req, "\">");
|
||||||
httpd_resp_sendstr_chunk(req, entry->d_name);
|
httpd_resp_sendstr_chunk(req, entry->d_name);
|
||||||
httpd_resp_sendstr_chunk(req, "</a></td><td>");
|
httpd_resp_sendstr_chunk(req, "</a></td><td>");
|
||||||
httpd_resp_sendstr_chunk(req, entrytype);
|
httpd_resp_sendstr_chunk(req, entrytype);
|
||||||
httpd_resp_sendstr_chunk(req, "</td><td>");
|
httpd_resp_sendstr_chunk(req, "</td><td>");
|
||||||
httpd_resp_sendstr_chunk(req, entrysize);
|
httpd_resp_sendstr_chunk(req, entrysize);
|
||||||
|
|
||||||
if (!readonly) {
|
if (!readonly) {
|
||||||
httpd_resp_sendstr_chunk(req, "</td><td>");
|
httpd_resp_sendstr_chunk(req, "</td><td>");
|
||||||
httpd_resp_sendstr_chunk(req, "<form method=\"post\" action=\"/delete");
|
httpd_resp_sendstr_chunk(req, "<form method=\"post\" action=\"/delete");
|
||||||
@@ -306,31 +304,28 @@ static esp_err_t http_resp_dir_html(httpd_req_t *req, const char *dirpath, const
|
|||||||
httpd_resp_sendstr_chunk(req, entry->d_name);
|
httpd_resp_sendstr_chunk(req, entry->d_name);
|
||||||
httpd_resp_sendstr_chunk(req, "\"><button type=\"submit\">Delete</button></form>");
|
httpd_resp_sendstr_chunk(req, "\"><button type=\"submit\">Delete</button></form>");
|
||||||
}
|
}
|
||||||
|
|
||||||
httpd_resp_sendstr_chunk(req, "</td></tr>\n");
|
httpd_resp_sendstr_chunk(req, "</td></tr>\n");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
closedir(dir);
|
|
||||||
|
|
||||||
/* Finish the file list table */
|
closedir(pdir);
|
||||||
|
|
||||||
|
// Finish the file list table
|
||||||
httpd_resp_sendstr_chunk(req, "</tbody></table>");
|
httpd_resp_sendstr_chunk(req, "</tbody></table>");
|
||||||
|
|
||||||
/* Send remaining chunk of HTML file to complete it */
|
// Send remaining chunk of HTML file to complete it
|
||||||
httpd_resp_sendstr_chunk(req, "</body></html>");
|
httpd_resp_sendstr_chunk(req, "</body></html>");
|
||||||
|
|
||||||
/* Send empty chunk to signal HTTP response completion */
|
// Send empty chunk to signal HTTP response completion
|
||||||
httpd_resp_sendstr_chunk(req, NULL);
|
httpd_resp_sendstr_chunk(req, NULL);
|
||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
/*
|
|
||||||
#define IS_FILE_EXT(filename, ext) \
|
|
||||||
(strcasecmp(&filename[strlen(filename) - sizeof(ext) + 1], ext) == 0)
|
|
||||||
*/
|
|
||||||
|
|
||||||
static esp_err_t logfileact_get_full_handler(httpd_req_t *req) {
|
static esp_err_t logfileact_get_full_handler(httpd_req_t *req) {
|
||||||
return send_logfile(req, true);
|
return send_logfile(req, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static esp_err_t logfileact_get_last_part_handler(httpd_req_t *req) {
|
static esp_err_t logfileact_get_last_part_handler(httpd_req_t *req) {
|
||||||
return send_logfile(req, false);
|
return send_logfile(req, false);
|
||||||
}
|
}
|
||||||
@@ -339,7 +334,6 @@ static esp_err_t datafileact_get_full_handler(httpd_req_t *req) {
|
|||||||
return send_datafile(req, true);
|
return send_datafile(req, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static esp_err_t datafileact_get_last_part_handler(httpd_req_t *req) {
|
static esp_err_t datafileact_get_last_part_handler(httpd_req_t *req) {
|
||||||
return send_datafile(req, false);
|
return send_datafile(req, false);
|
||||||
}
|
}
|
||||||
@@ -357,7 +351,7 @@ static esp_err_t send_datafile(httpd_req_t *req, bool send_full_file)
|
|||||||
|
|
||||||
fd = fopen(currentfilename.c_str(), "r");
|
fd = fopen(currentfilename.c_str(), "r");
|
||||||
if (!fd) {
|
if (!fd) {
|
||||||
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to read file: " + std::string(currentfilename) +"!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to read file: " + currentfilename + "!");
|
||||||
/* Respond with 404 Error */
|
/* Respond with 404 Error */
|
||||||
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
|
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
@@ -373,7 +367,7 @@ static esp_err_t send_datafile(httpd_req_t *req, bool send_full_file)
|
|||||||
|
|
||||||
/* Adapted from https://www.geeksforgeeks.org/implement-your-own-tail-read-last-n-lines-of-a-huge-file/ */
|
/* Adapted from https://www.geeksforgeeks.org/implement-your-own-tail-read-last-n-lines-of-a-huge-file/ */
|
||||||
if (fseek(fd, 0, SEEK_END)) {
|
if (fseek(fd, 0, SEEK_END)) {
|
||||||
ESP_LOGE(TAG, "Failed to get to end of file!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to get to end of file!");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -381,7 +375,7 @@ static esp_err_t send_datafile(httpd_req_t *req, bool send_full_file)
|
|||||||
ESP_LOGI(TAG, "File contains %ld bytes", pos);
|
ESP_LOGI(TAG, "File contains %ld bytes", pos);
|
||||||
|
|
||||||
if (fseek(fd, pos - std::min((long)LOGFILE_LAST_PART_BYTES, pos), SEEK_SET)) { // Go LOGFILE_LAST_PART_BYTES bytes back from EOF
|
if (fseek(fd, pos - std::min((long)LOGFILE_LAST_PART_BYTES, pos), SEEK_SET)) { // Go LOGFILE_LAST_PART_BYTES bytes back from EOF
|
||||||
ESP_LOGE(TAG, "Failed to go back %ld bytes within the file!", std::min((long)LOGFILE_LAST_PART_BYTES, pos));
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to go back " + to_string(std::min((long)LOGFILE_LAST_PART_BYTES, pos)) + " bytes within the file!");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -404,7 +398,7 @@ static esp_err_t send_datafile(httpd_req_t *req, bool send_full_file)
|
|||||||
/* Send the buffer contents as HTTP response chunk */
|
/* Send the buffer contents as HTTP response chunk */
|
||||||
if (httpd_resp_send_chunk(req, chunk, chunksize) != ESP_OK) {
|
if (httpd_resp_send_chunk(req, chunk, chunksize) != ESP_OK) {
|
||||||
fclose(fd);
|
fclose(fd);
|
||||||
ESP_LOGE(TAG, "File sending failed!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "File sending failed!");
|
||||||
/* Abort sending file */
|
/* Abort sending file */
|
||||||
httpd_resp_sendstr_chunk(req, NULL);
|
httpd_resp_sendstr_chunk(req, NULL);
|
||||||
/* Respond with 500 Internal Server Error */
|
/* Respond with 500 Internal Server Error */
|
||||||
@@ -424,7 +418,6 @@ static esp_err_t send_datafile(httpd_req_t *req, bool send_full_file)
|
|||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static esp_err_t send_logfile(httpd_req_t *req, bool send_full_file)
|
static esp_err_t send_logfile(httpd_req_t *req, bool send_full_file)
|
||||||
{
|
{
|
||||||
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "log_get_last_part_handler");
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "log_get_last_part_handler");
|
||||||
@@ -443,7 +436,7 @@ static esp_err_t send_logfile(httpd_req_t *req, bool send_full_file)
|
|||||||
|
|
||||||
fd = fopen(currentfilename.c_str(), "r");
|
fd = fopen(currentfilename.c_str(), "r");
|
||||||
if (!fd) {
|
if (!fd) {
|
||||||
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to read file: " + std::string(currentfilename.c_str()) +"!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to read file: " + currentfilename + "!");
|
||||||
/* Respond with 404 Error */
|
/* Respond with 404 Error */
|
||||||
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
|
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
@@ -459,7 +452,7 @@ static esp_err_t send_logfile(httpd_req_t *req, bool send_full_file)
|
|||||||
|
|
||||||
/* Adapted from https://www.geeksforgeeks.org/implement-your-own-tail-read-last-n-lines-of-a-huge-file/ */
|
/* Adapted from https://www.geeksforgeeks.org/implement-your-own-tail-read-last-n-lines-of-a-huge-file/ */
|
||||||
if (fseek(fd, 0, SEEK_END)) {
|
if (fseek(fd, 0, SEEK_END)) {
|
||||||
ESP_LOGE(TAG, "Failed to get to end of file!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to get to end of file!");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -467,7 +460,7 @@ static esp_err_t send_logfile(httpd_req_t *req, bool send_full_file)
|
|||||||
ESP_LOGI(TAG, "File contains %ld bytes", pos);
|
ESP_LOGI(TAG, "File contains %ld bytes", pos);
|
||||||
|
|
||||||
if (fseek(fd, pos - std::min((long)LOGFILE_LAST_PART_BYTES, pos), SEEK_SET)) { // Go LOGFILE_LAST_PART_BYTES bytes back from EOF
|
if (fseek(fd, pos - std::min((long)LOGFILE_LAST_PART_BYTES, pos), SEEK_SET)) { // Go LOGFILE_LAST_PART_BYTES bytes back from EOF
|
||||||
ESP_LOGE(TAG, "Failed to go back %ld bytes within the file!", std::min((long)LOGFILE_LAST_PART_BYTES, pos));
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to go back " + to_string(std::min((long)LOGFILE_LAST_PART_BYTES, pos)) + " bytes within the file!");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -490,7 +483,7 @@ static esp_err_t send_logfile(httpd_req_t *req, bool send_full_file)
|
|||||||
/* Send the buffer contents as HTTP response chunk */
|
/* Send the buffer contents as HTTP response chunk */
|
||||||
if (httpd_resp_send_chunk(req, chunk, chunksize) != ESP_OK) {
|
if (httpd_resp_send_chunk(req, chunk, chunksize) != ESP_OK) {
|
||||||
fclose(fd);
|
fclose(fd);
|
||||||
ESP_LOGE(TAG, "File sending failed!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "File sending failed!");
|
||||||
/* Abort sending file */
|
/* Abort sending file */
|
||||||
httpd_resp_sendstr_chunk(req, NULL);
|
httpd_resp_sendstr_chunk(req, NULL);
|
||||||
/* Respond with 500 Internal Server Error */
|
/* Respond with 500 Internal Server Error */
|
||||||
@@ -510,7 +503,6 @@ static esp_err_t send_logfile(httpd_req_t *req, bool send_full_file)
|
|||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Handler to download a file kept on the server */
|
/* Handler to download a file kept on the server */
|
||||||
static esp_err_t download_get_handler(httpd_req_t *req)
|
static esp_err_t download_get_handler(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
@@ -528,9 +520,8 @@ static esp_err_t download_get_handler(httpd_req_t *req)
|
|||||||
// filename = get_path_from_uri(filepath, ((struct file_server_data *)req->user_ctx)->base_path,
|
// filename = get_path_from_uri(filepath, ((struct file_server_data *)req->user_ctx)->base_path,
|
||||||
// req->uri, sizeof(filepath));
|
// req->uri, sizeof(filepath));
|
||||||
|
|
||||||
|
|
||||||
if (!filename) {
|
if (!filename) {
|
||||||
ESP_LOGE(TAG, "Filename is too long");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Filename is too long");
|
||||||
/* Respond with 414 Error */
|
/* Respond with 414 Error */
|
||||||
httpd_resp_send_err(req, HTTPD_414_URI_TOO_LONG, "Filename too long");
|
httpd_resp_send_err(req, HTTPD_414_URI_TOO_LONG, "Filename too long");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
@@ -548,7 +539,7 @@ static esp_err_t download_get_handler(httpd_req_t *req)
|
|||||||
/* Get value of expected key from query string */
|
/* Get value of expected key from query string */
|
||||||
if (httpd_query_key_value(buf, "readonly", param, sizeof(param)) == ESP_OK) {
|
if (httpd_query_key_value(buf, "readonly", param, sizeof(param)) == ESP_OK) {
|
||||||
ESP_LOGI(TAG, "Found URL query parameter => readonly=%s", param);
|
ESP_LOGI(TAG, "Found URL query parameter => readonly=%s", param);
|
||||||
readonly = param && strcmp(param,"true")==0;
|
readonly = (strcmp(param,"true") == 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -571,7 +562,7 @@ static esp_err_t download_get_handler(httpd_req_t *req)
|
|||||||
|
|
||||||
fd = fopen(filepath, "r");
|
fd = fopen(filepath, "r");
|
||||||
if (!fd) {
|
if (!fd) {
|
||||||
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to read file: " + std::string(filepath) +"!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to read file: " + std::string(filepath) + "!");
|
||||||
/* Respond with 404 Error */
|
/* Respond with 404 Error */
|
||||||
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
|
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
@@ -589,10 +580,12 @@ static esp_err_t download_get_handler(httpd_req_t *req)
|
|||||||
/* Read file in chunks into the scratch buffer */
|
/* Read file in chunks into the scratch buffer */
|
||||||
chunksize = fread(chunk, 1, SERVER_FILER_SCRATCH_BUFSIZE, fd);
|
chunksize = fread(chunk, 1, SERVER_FILER_SCRATCH_BUFSIZE, fd);
|
||||||
|
|
||||||
/* Send the buffer contents as HTTP response chunk */
|
/* Send buffer contents as HTTP chunk. If empty this functions as a
|
||||||
|
* last-chunk message, signaling end-of-response, to the HTTP client.
|
||||||
|
* See RFC 2616, section 3.6.1 for details on Chunked Transfer Encoding. */
|
||||||
if (httpd_resp_send_chunk(req, chunk, chunksize) != ESP_OK) {
|
if (httpd_resp_send_chunk(req, chunk, chunksize) != ESP_OK) {
|
||||||
fclose(fd);
|
fclose(fd);
|
||||||
ESP_LOGE(TAG, "File sending failed!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "File sending failed!");
|
||||||
/* Abort sending file */
|
/* Abort sending file */
|
||||||
httpd_resp_sendstr_chunk(req, NULL);
|
httpd_resp_sendstr_chunk(req, NULL);
|
||||||
/* Respond with 500 Internal Server Error */
|
/* Respond with 500 Internal Server Error */
|
||||||
@@ -607,8 +600,6 @@ static esp_err_t download_get_handler(httpd_req_t *req)
|
|||||||
fclose(fd);
|
fclose(fd);
|
||||||
ESP_LOGD(TAG, "File successfully sent");
|
ESP_LOGD(TAG, "File successfully sent");
|
||||||
|
|
||||||
/* Respond with an empty chunk to signal HTTP response completion */
|
|
||||||
httpd_resp_send_chunk(req, NULL, 0);
|
|
||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -620,6 +611,8 @@ static esp_err_t upload_post_handler(httpd_req_t *req)
|
|||||||
FILE *fd = NULL;
|
FILE *fd = NULL;
|
||||||
struct stat file_stat;
|
struct stat file_stat;
|
||||||
|
|
||||||
|
ESP_LOGI(TAG, "uri: %s", req->uri);
|
||||||
|
|
||||||
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
||||||
|
|
||||||
/* Skip leading "/upload" from URI to get filename */
|
/* Skip leading "/upload" from URI to get filename */
|
||||||
@@ -634,14 +627,14 @@ static esp_err_t upload_post_handler(httpd_req_t *req)
|
|||||||
|
|
||||||
/* Filename cannot have a trailing '/' */
|
/* Filename cannot have a trailing '/' */
|
||||||
if (filename[strlen(filename) - 1] == '/') {
|
if (filename[strlen(filename) - 1] == '/') {
|
||||||
ESP_LOGE(TAG, "Invalid filename: %s", filename);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Invalid filename: " + string(filename));
|
||||||
/* Respond with 400 Bad Request */
|
/* Respond with 400 Bad Request */
|
||||||
httpd_resp_send_err(req, HTTPD_400_BAD_REQUEST, "Invalid filename");
|
httpd_resp_send_err(req, HTTPD_400_BAD_REQUEST, "Invalid filename");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stat(filepath, &file_stat) == 0) {
|
if (stat(filepath, &file_stat) == 0) {
|
||||||
ESP_LOGE(TAG, "File already exists: %s", filepath);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "File already exists: " + string(filepath));
|
||||||
/* Respond with 400 Bad Request */
|
/* Respond with 400 Bad Request */
|
||||||
httpd_resp_send_err(req, HTTPD_400_BAD_REQUEST, "File already exists");
|
httpd_resp_send_err(req, HTTPD_400_BAD_REQUEST, "File already exists");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
@@ -649,7 +642,7 @@ static esp_err_t upload_post_handler(httpd_req_t *req)
|
|||||||
|
|
||||||
/* File cannot be larger than a limit */
|
/* File cannot be larger than a limit */
|
||||||
if (req->content_len > MAX_FILE_SIZE) {
|
if (req->content_len > MAX_FILE_SIZE) {
|
||||||
ESP_LOGE(TAG, "File too large: %d bytes", req->content_len);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "File too large: " + to_string(req->content_len) + " bytes");
|
||||||
/* Respond with 400 Bad Request */
|
/* Respond with 400 Bad Request */
|
||||||
httpd_resp_send_err(req, HTTPD_400_BAD_REQUEST,
|
httpd_resp_send_err(req, HTTPD_400_BAD_REQUEST,
|
||||||
"File size must be less than "
|
"File size must be less than "
|
||||||
@@ -661,7 +654,7 @@ static esp_err_t upload_post_handler(httpd_req_t *req)
|
|||||||
|
|
||||||
fd = fopen(filepath, "w");
|
fd = fopen(filepath, "w");
|
||||||
if (!fd) {
|
if (!fd) {
|
||||||
ESP_LOGE(TAG, "Failed to create file: %s", filepath);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to create file: " + string(filepath));
|
||||||
/* Respond with 500 Internal Server Error */
|
/* Respond with 500 Internal Server Error */
|
||||||
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to create file");
|
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to create file");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
@@ -692,7 +685,7 @@ static esp_err_t upload_post_handler(httpd_req_t *req)
|
|||||||
fclose(fd);
|
fclose(fd);
|
||||||
unlink(filepath);
|
unlink(filepath);
|
||||||
|
|
||||||
ESP_LOGE(TAG, "File reception failed!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "File reception failed!");
|
||||||
/* Respond with 500 Internal Server Error */
|
/* Respond with 500 Internal Server Error */
|
||||||
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to receive file");
|
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to receive file");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
@@ -705,7 +698,7 @@ static esp_err_t upload_post_handler(httpd_req_t *req)
|
|||||||
fclose(fd);
|
fclose(fd);
|
||||||
unlink(filepath);
|
unlink(filepath);
|
||||||
|
|
||||||
ESP_LOGE(TAG, "File write failed!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "File write failed!");
|
||||||
/* Respond with 500 Internal Server Error */
|
/* Respond with 500 Internal Server Error */
|
||||||
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to write file to storage");
|
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to write file to storage");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
@@ -718,8 +711,41 @@ static esp_err_t upload_post_handler(httpd_req_t *req)
|
|||||||
|
|
||||||
/* Close file upon upload completion */
|
/* Close file upon upload completion */
|
||||||
fclose(fd);
|
fclose(fd);
|
||||||
ESP_LOGI(TAG, "File reception complete");
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "File saved: " + string(filename));
|
||||||
|
ESP_LOGI(TAG, "File reception completed");
|
||||||
|
|
||||||
|
string s = req->uri;
|
||||||
|
if (isInString(s, "?md5")) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Calculate and return MD5 sum...");
|
||||||
|
|
||||||
|
fd = fopen(filepath, "r");
|
||||||
|
if (!fd) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to open file for reading: " + string(filepath));
|
||||||
|
/* Respond with 500 Internal Server Error */
|
||||||
|
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to open file for reading");
|
||||||
|
return ESP_FAIL;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t result[16];
|
||||||
|
string md5hex = "";
|
||||||
|
string response = "{\"md5\":";
|
||||||
|
char hex[3];
|
||||||
|
|
||||||
|
md5File(fd, result);
|
||||||
|
fclose(fd);
|
||||||
|
|
||||||
|
for (int i = 0; i < sizeof(result); i++) {
|
||||||
|
snprintf(hex, sizeof(hex), "%02x", result[i]);
|
||||||
|
md5hex.append(hex);
|
||||||
|
}
|
||||||
|
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "MD5 of " + string(filepath) + ": " + md5hex);
|
||||||
|
response.append("\"" + md5hex + "\"");
|
||||||
|
response.append("}");
|
||||||
|
|
||||||
|
httpd_resp_sendstr(req, response.c_str());
|
||||||
|
}
|
||||||
|
else { // Return file server page
|
||||||
std::string directory = std::string(filepath);
|
std::string directory = std::string(filepath);
|
||||||
size_t zw = directory.find("/");
|
size_t zw = directory.find("/");
|
||||||
size_t found = zw;
|
size_t found = zw;
|
||||||
@@ -734,24 +760,30 @@ static esp_err_t upload_post_handler(httpd_req_t *req)
|
|||||||
ESP_LOGD(TAG, "Directory: %s, start_fn: %d, found: %d", directory.c_str(), start_fn, found);
|
ESP_LOGD(TAG, "Directory: %s, start_fn: %d, found: %d", directory.c_str(), start_fn, found);
|
||||||
directory = directory.substr(start_fn, found - start_fn + 1);
|
directory = directory.substr(start_fn, found - start_fn + 1);
|
||||||
directory = "/fileserver" + directory;
|
directory = "/fileserver" + directory;
|
||||||
// ESP_LOGD(TAG, "Directory danach 2: %s", directory.c_str());
|
// ESP_LOGD(TAG, "Directory danach 2: %s", directory.c_str());
|
||||||
|
|
||||||
/* Redirect onto root to see the updated file list */
|
/* Redirect onto root to see the updated file list */
|
||||||
httpd_resp_set_status(req, "303 See Other");
|
if (strcmp(filename, "/config/config.ini") == 0 ||
|
||||||
httpd_resp_set_hdr(req, "Location", directory.c_str());
|
strcmp(filename, "/config/ref0.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/config/ref0_org.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/config/ref1.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/config/ref1_org.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/config/reference.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/img_tmp/ref0.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/img_tmp/ref0_org.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/img_tmp/ref1.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/img_tmp/ref1_org.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/img_tmp/reference.jpg") == 0 )
|
||||||
|
{
|
||||||
|
httpd_resp_set_status(req, HTTPD_200); // Avoid reloading of folder content
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
httpd_resp_set_status(req, "303 See Other"); // Reload folder content after upload
|
||||||
|
}
|
||||||
|
|
||||||
/* Redirect onto root to see the updated file list */
|
|
||||||
httpd_resp_set_status(req, "303 See Other");
|
|
||||||
httpd_resp_set_hdr(req, "Location", directory.c_str());
|
httpd_resp_set_hdr(req, "Location", directory.c_str());
|
||||||
httpd_resp_sendstr(req, "File uploaded successfully");
|
httpd_resp_sendstr(req, "File uploaded successfully");
|
||||||
|
|
||||||
/*
|
|
||||||
if (strcmp(filepath, CONFIG_FILE) == 0) {
|
|
||||||
ESP_LOGD(TAG, "New config found. Reload handler.");
|
|
||||||
gpio_handler_deinit();
|
|
||||||
MQTTdestroy();
|
|
||||||
}
|
}
|
||||||
*/
|
|
||||||
|
|
||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
@@ -763,7 +795,6 @@ static esp_err_t delete_post_handler(httpd_req_t *req)
|
|||||||
char filepath[FILE_PATH_MAX];
|
char filepath[FILE_PATH_MAX];
|
||||||
struct stat file_stat;
|
struct stat file_stat;
|
||||||
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////
|
||||||
char _query[200];
|
char _query[200];
|
||||||
char _valuechar[30];
|
char _valuechar[30];
|
||||||
@@ -780,7 +811,7 @@ static esp_err_t delete_post_handler(httpd_req_t *req)
|
|||||||
|
|
||||||
if (httpd_query_key_value(_query, "task", _valuechar, 30) == ESP_OK)
|
if (httpd_query_key_value(_query, "task", _valuechar, 30) == ESP_OK)
|
||||||
{
|
{
|
||||||
ESP_LOGD(TAG, "task is found: %s", _valuechar);
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "task is found: " + string(_valuechar));
|
||||||
_task = std::string(_valuechar);
|
_task = std::string(_valuechar);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -826,28 +857,27 @@ static esp_err_t delete_post_handler(httpd_req_t *req)
|
|||||||
|
|
||||||
/* Filename cannot have a trailing '/' */
|
/* Filename cannot have a trailing '/' */
|
||||||
if (filename[strlen(filename) - 1] == '/') {
|
if (filename[strlen(filename) - 1] == '/') {
|
||||||
ESP_LOGE(TAG, "Invalid filename: %s", filename);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Invalid filename: " + string(filename));
|
||||||
/* Respond with 400 Bad Request */
|
/* Respond with 400 Bad Request */
|
||||||
httpd_resp_send_err(req, HTTPD_400_BAD_REQUEST, "Invalid filename");
|
httpd_resp_send_err(req, HTTPD_400_BAD_REQUEST, "Invalid filename");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strcmp(filename, "wlan.ini") == 0) {
|
if (strcmp(filename, "wlan.ini") == 0) {
|
||||||
ESP_LOGE(TAG, "Trying to delete protected file : %s", filename);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to delete protected file : " + string(filename));
|
||||||
httpd_resp_send_err(req, HTTPD_403_FORBIDDEN, "Not allowed to delete wlan.ini");
|
httpd_resp_send_err(req, HTTPD_403_FORBIDDEN, "Not allowed to delete wlan.ini");
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stat(filepath, &file_stat) == -1) {
|
if (stat(filepath, &file_stat) == -1) { // File does not exist
|
||||||
ESP_LOGE(TAG, "File does not exist: %s", filename);
|
/* This is ok, we would delete it anyway */
|
||||||
/* Respond with 400 Bad Request */
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "File does not exist: " + string(filename));
|
||||||
httpd_resp_send_err(req, HTTPD_400_BAD_REQUEST, "File does not exist");
|
|
||||||
return ESP_FAIL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGI(TAG, "Deleting file: %s", filename);
|
|
||||||
/* Delete file */
|
/* Delete file */
|
||||||
unlink(filepath);
|
unlink(filepath);
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "File deleted: " + string(filename));
|
||||||
|
ESP_LOGI(TAG, "File deletion completed");
|
||||||
|
|
||||||
directory = std::string(filepath);
|
directory = std::string(filepath);
|
||||||
size_t zw = directory.find("/");
|
size_t zw = directory.find("/");
|
||||||
@@ -864,22 +894,34 @@ static esp_err_t delete_post_handler(httpd_req_t *req)
|
|||||||
directory = directory.substr(start_fn, found - start_fn + 1);
|
directory = directory.substr(start_fn, found - start_fn + 1);
|
||||||
directory = "/fileserver" + directory;
|
directory = "/fileserver" + directory;
|
||||||
ESP_LOGD(TAG, "Directory danach 4: %s", directory.c_str());
|
ESP_LOGD(TAG, "Directory danach 4: %s", directory.c_str());
|
||||||
}
|
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
/* Redirect onto root to see the updated file list */
|
/* Redirect onto root to see the updated file list */
|
||||||
httpd_resp_set_status(req, "303 See Other");
|
if (strcmp(filename, "/config/config.ini") == 0 ||
|
||||||
|
strcmp(filename, "/config/ref0.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/config/ref0_org.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/config/ref1.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/config/ref1_org.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/config/reference.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/img_tmp/ref0.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/img_tmp/ref0_org.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/img_tmp/ref1.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/img_tmp/ref1_org.jpg") == 0 ||
|
||||||
|
strcmp(filename, "/img_tmp/reference.jpg") == 0 )
|
||||||
|
{
|
||||||
|
httpd_resp_set_status(req, HTTPD_200); // Avoid reloading of folder content
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
httpd_resp_set_status(req, "303 See Other"); // Reload folder content after upload
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
httpd_resp_set_hdr(req, "Location", directory.c_str());
|
httpd_resp_set_hdr(req, "Location", directory.c_str());
|
||||||
httpd_resp_sendstr(req, "File successfully deleted");
|
httpd_resp_sendstr(req, "File successfully deleted");
|
||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void delete_all_in_directory(std::string _directory)
|
void delete_all_in_directory(std::string _directory)
|
||||||
{
|
{
|
||||||
struct dirent *entry;
|
struct dirent *entry;
|
||||||
@@ -887,7 +929,7 @@ void delete_all_in_directory(std::string _directory)
|
|||||||
std::string filename;
|
std::string filename;
|
||||||
|
|
||||||
if (!dir) {
|
if (!dir) {
|
||||||
ESP_LOGE(TAG, "Failed to stat dir: %s", _directory.c_str());
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to stat dir: " + _directory);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -896,7 +938,7 @@ void delete_all_in_directory(std::string _directory)
|
|||||||
if (!(entry->d_type == DT_DIR)){
|
if (!(entry->d_type == DT_DIR)){
|
||||||
if (strcmp("wlan.ini", entry->d_name) != 0){ // auf wlan.ini soll nicht zugegriffen werden !!!
|
if (strcmp("wlan.ini", entry->d_name) != 0){ // auf wlan.ini soll nicht zugegriffen werden !!!
|
||||||
filename = _directory + "/" + std::string(entry->d_name);
|
filename = _directory + "/" + std::string(entry->d_name);
|
||||||
ESP_LOGI(TAG, "Deleting file: %s", filename.c_str());
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Deleting file: " + filename);
|
||||||
/* Delete file */
|
/* Delete file */
|
||||||
unlink(filename.c_str());
|
unlink(filename.c_str());
|
||||||
}
|
}
|
||||||
@@ -905,7 +947,7 @@ void delete_all_in_directory(std::string _directory)
|
|||||||
closedir(dir);
|
closedir(dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string unzip_new(std::string _in_zip_file, std::string _target_zip, std::string _target_bin, std::string _main, bool _initial_setup)
|
std::string unzip_new(std::string _in_zip_file, std::string _html_tmp, std::string _html_final, std::string _target_bin, std::string _main, bool _initial_setup)
|
||||||
{
|
{
|
||||||
int i, sort_iter;
|
int i, sort_iter;
|
||||||
mz_bool status;
|
mz_bool status;
|
||||||
@@ -930,7 +972,7 @@ std::string unzip_new(std::string _in_zip_file, std::string _target_zip, std::st
|
|||||||
|
|
||||||
// Get and print information about each file in the archive.
|
// Get and print information about each file in the archive.
|
||||||
int numberoffiles = (int)mz_zip_reader_get_num_files(&zip_archive);
|
int numberoffiles = (int)mz_zip_reader_get_num_files(&zip_archive);
|
||||||
ESP_LOGI(TAG, "Numbers of files to be extracted: %d", numberoffiles);
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Files to be extracted: " + to_string(numberoffiles));
|
||||||
|
|
||||||
sort_iter = 0;
|
sort_iter = 0;
|
||||||
{
|
{
|
||||||
@@ -953,7 +995,7 @@ std::string unzip_new(std::string _in_zip_file, std::string _target_zip, std::st
|
|||||||
p = mz_zip_reader_extract_file_to_heap(&zip_archive, archive_filename, &uncomp_size, 0);
|
p = mz_zip_reader_extract_file_to_heap(&zip_archive, archive_filename, &uncomp_size, 0);
|
||||||
if (!p)
|
if (!p)
|
||||||
{
|
{
|
||||||
ESP_LOGE(TAG, "mz_zip_reader_extract_file_to_heap() failed on file %s", archive_filename);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "mz_zip_reader_extract_file_to_heap() failed on file " + string(archive_filename));
|
||||||
mz_zip_reader_end(&zip_archive);
|
mz_zip_reader_end(&zip_archive);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@@ -987,14 +1029,19 @@ std::string unzip_new(std::string _in_zip_file, std::string _target_zip, std::st
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
zw = _target_zip + zw;
|
zw = _html_tmp + zw;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// files in the html folder shall be redirected to the temporary html folder
|
||||||
|
if (zw.find(_html_final) == 0) {
|
||||||
|
FindReplace(zw, _html_final, _html_tmp);
|
||||||
|
}
|
||||||
|
|
||||||
string filename_zw = zw + SUFFIX_ZW;
|
string filename_zw = zw + SUFFIX_ZW;
|
||||||
|
|
||||||
ESP_LOGI(TAG, "Filename to extract: %s, Zwischenfilename: %s", zw.c_str(), filename_zw.c_str());
|
ESP_LOGI(TAG, "File to extract: %s, Temp. Filename: %s", zw.c_str(), filename_zw.c_str());
|
||||||
|
|
||||||
std::string folder = filename_zw.substr(0, filename_zw.find_last_of('/'));
|
std::string folder = filename_zw.substr(0, filename_zw.find_last_of('/'));
|
||||||
MakeDir(folder);
|
MakeDir(folder);
|
||||||
@@ -1015,21 +1062,22 @@ std::string unzip_new(std::string _in_zip_file, std::string _target_zip, std::st
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
isokay = false;
|
isokay = false;
|
||||||
ESP_LOGD(TAG, "ERROR in writting extracted file (function fwrite) extracted file \"%s\", size %u", archive_filename, (uint)uncomp_size);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "ERROR in writting extracted file (function fwrite) extracted file \"" +
|
||||||
|
string(archive_filename) + "\", size " + to_string(uncomp_size));
|
||||||
}
|
}
|
||||||
|
|
||||||
DeleteFile(zw);
|
DeleteFile(zw);
|
||||||
if (!isokay)
|
if (!isokay)
|
||||||
ESP_LOGE(TAG, "ERROR in fwrite \"%s\", size %u", archive_filename, (uint)uncomp_size);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "ERROR in fwrite \"" + string(archive_filename) + "\", size " + to_string(uncomp_size));
|
||||||
isokay = isokay && RenameFile(filename_zw, zw);
|
isokay = isokay && RenameFile(filename_zw, zw);
|
||||||
if (!isokay)
|
if (!isokay)
|
||||||
ESP_LOGE(TAG, "ERROR in Rename \"%s\" to \"%s\"", filename_zw.c_str(), zw.c_str());
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "ERROR in Rename \"" + filename_zw + "\" to \"" + zw);
|
||||||
|
|
||||||
if (isokay)
|
if (isokay)
|
||||||
ESP_LOGI(TAG, "Successfully extracted file \"%s\", size %u", archive_filename, (uint)uncomp_size);
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Successfully extracted file \"" + string(archive_filename) + "\", size " + to_string(uncomp_size));
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ESP_LOGE(TAG, "ERROR in extracting file \"%s\", size %u", archive_filename, (uint)uncomp_size);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "ERROR in extracting file \"" + string(archive_filename) + "\", size " + to_string(uncomp_size));
|
||||||
ret = "ERROR";
|
ret = "ERROR";
|
||||||
}
|
}
|
||||||
mz_free(p);
|
mz_free(p);
|
||||||
@@ -1063,7 +1111,7 @@ void unzip(std::string _in_zip_file, std::string _target_directory){
|
|||||||
status = mz_zip_reader_init_file(&zip_archive, _in_zip_file.c_str(), 0);
|
status = mz_zip_reader_init_file(&zip_archive, _in_zip_file.c_str(), 0);
|
||||||
if (!status)
|
if (!status)
|
||||||
{
|
{
|
||||||
ESP_LOGD(TAG, "mz_zip_reader_init_file() failed!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "mz_zip_reader_init_file() failed!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1075,7 +1123,7 @@ void unzip(std::string _in_zip_file, std::string _target_directory){
|
|||||||
status = mz_zip_reader_init_file(&zip_archive, _in_zip_file.c_str(), sort_iter ? MZ_ZIP_FLAG_DO_NOT_SORT_CENTRAL_DIRECTORY : 0);
|
status = mz_zip_reader_init_file(&zip_archive, _in_zip_file.c_str(), sort_iter ? MZ_ZIP_FLAG_DO_NOT_SORT_CENTRAL_DIRECTORY : 0);
|
||||||
if (!status)
|
if (!status)
|
||||||
{
|
{
|
||||||
ESP_LOGD(TAG, "mz_zip_reader_init_file() failed!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "mz_zip_reader_init_file() failed!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1089,7 +1137,7 @@ void unzip(std::string _in_zip_file, std::string _target_directory){
|
|||||||
p = mz_zip_reader_extract_file_to_heap(&zip_archive, archive_filename, &uncomp_size, 0);
|
p = mz_zip_reader_extract_file_to_heap(&zip_archive, archive_filename, &uncomp_size, 0);
|
||||||
if (!p)
|
if (!p)
|
||||||
{
|
{
|
||||||
ESP_LOGD(TAG, "mz_zip_reader_extract_file_to_heap() failed!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "mz_zip_reader_extract_file_to_heap() failed!");
|
||||||
mz_zip_reader_end(&zip_archive);
|
mz_zip_reader_end(&zip_archive);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -1097,7 +1145,7 @@ void unzip(std::string _in_zip_file, std::string _target_directory){
|
|||||||
// Save to File.
|
// Save to File.
|
||||||
zw = std::string(archive_filename);
|
zw = std::string(archive_filename);
|
||||||
zw = _target_directory + zw;
|
zw = _target_directory + zw;
|
||||||
ESP_LOGD(TAG, "Filename to extract: %s", zw.c_str());
|
ESP_LOGD(TAG, "File to extract: %s", zw.c_str());
|
||||||
FILE* fpTargetFile = fopen(zw.c_str(), "wb");
|
FILE* fpTargetFile = fopen(zw.c_str(), "wb");
|
||||||
fwrite(p, 1, (uint)uncomp_size, fpTargetFile);
|
fwrite(p, 1, (uint)uncomp_size, fpTargetFile);
|
||||||
fclose(fpTargetFile);
|
fclose(fpTargetFile);
|
||||||
@@ -1116,8 +1164,6 @@ void unzip(std::string _in_zip_file, std::string _target_directory){
|
|||||||
ESP_LOGD(TAG, "Success.");
|
ESP_LOGD(TAG, "Success.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void register_server_file_uri(httpd_handle_t server, const char *base_path)
|
void register_server_file_uri(httpd_handle_t server, const char *base_path)
|
||||||
{
|
{
|
||||||
static struct file_server_data *server_data = NULL;
|
static struct file_server_data *server_data = NULL;
|
||||||
@@ -1125,26 +1171,24 @@ void register_server_file_uri(httpd_handle_t server, const char *base_path)
|
|||||||
/* Validate file storage base path */
|
/* Validate file storage base path */
|
||||||
if (!base_path) {
|
if (!base_path) {
|
||||||
// if (!base_path || strcmp(base_path, "/spiffs") != 0) {
|
// if (!base_path || strcmp(base_path, "/spiffs") != 0) {
|
||||||
ESP_LOGE(TAG, "File server base_path not set");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "File server base_path not set");
|
||||||
// return ESP_ERR_INVALID_ARG;
|
// return ESP_ERR_INVALID_ARG;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (server_data) {
|
if (server_data) {
|
||||||
ESP_LOGE(TAG, "File server already started");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "File server already started");
|
||||||
// return ESP_ERR_INVALID_STATE;
|
// return ESP_ERR_INVALID_STATE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Allocate memory for server data */
|
/* Allocate memory for server data */
|
||||||
server_data = (file_server_data *) calloc(1, sizeof(struct file_server_data));
|
server_data = (file_server_data *) calloc(1, sizeof(struct file_server_data));
|
||||||
if (!server_data) {
|
if (!server_data) {
|
||||||
ESP_LOGE(TAG, "Failed to allocate memory for server data");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to allocate memory for server data");
|
||||||
// return ESP_ERR_NO_MEM;
|
// return ESP_ERR_NO_MEM;
|
||||||
}
|
}
|
||||||
strlcpy(server_data->base_path, base_path,
|
strlcpy(server_data->base_path, base_path,
|
||||||
sizeof(server_data->base_path));
|
sizeof(server_data->base_path));
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* URI handler for getting uploaded files */
|
/* URI handler for getting uploaded files */
|
||||||
// char zw[sizeof(serverprefix)+1];
|
// char zw[sizeof(serverprefix)+1];
|
||||||
// strcpy(zw, serverprefix);
|
// strcpy(zw, serverprefix);
|
||||||
@@ -1154,25 +1198,23 @@ void register_server_file_uri(httpd_handle_t server, const char *base_path)
|
|||||||
httpd_uri_t file_download = {
|
httpd_uri_t file_download = {
|
||||||
.uri = "/fileserver*", // Match all URIs of type /path/to/file
|
.uri = "/fileserver*", // Match all URIs of type /path/to/file
|
||||||
.method = HTTP_GET,
|
.method = HTTP_GET,
|
||||||
.handler = download_get_handler,
|
.handler = APPLY_BASIC_AUTH_FILTER(download_get_handler),
|
||||||
.user_ctx = server_data // Pass server data as context
|
.user_ctx = server_data // Pass server data as context
|
||||||
};
|
};
|
||||||
httpd_register_uri_handler(server, &file_download);
|
httpd_register_uri_handler(server, &file_download);
|
||||||
|
|
||||||
|
|
||||||
httpd_uri_t file_datafileact = {
|
httpd_uri_t file_datafileact = {
|
||||||
.uri = "/datafileact", // Match all URIs of type /path/to/file
|
.uri = "/datafileact", // Match all URIs of type /path/to/file
|
||||||
.method = HTTP_GET,
|
.method = HTTP_GET,
|
||||||
.handler = datafileact_get_full_handler,
|
.handler = APPLY_BASIC_AUTH_FILTER(datafileact_get_full_handler),
|
||||||
.user_ctx = server_data // Pass server data as context
|
.user_ctx = server_data // Pass server data as context
|
||||||
};
|
};
|
||||||
httpd_register_uri_handler(server, &file_datafileact);
|
httpd_register_uri_handler(server, &file_datafileact);
|
||||||
|
|
||||||
|
|
||||||
httpd_uri_t file_datafile_last_part_handle = {
|
httpd_uri_t file_datafile_last_part_handle = {
|
||||||
.uri = "/data", // Match all URIs of type /path/to/file
|
.uri = "/data", // Match all URIs of type /path/to/file
|
||||||
.method = HTTP_GET,
|
.method = HTTP_GET,
|
||||||
.handler = datafileact_get_last_part_handler,
|
.handler = APPLY_BASIC_AUTH_FILTER(datafileact_get_last_part_handler),
|
||||||
.user_ctx = server_data // Pass server data as context
|
.user_ctx = server_data // Pass server data as context
|
||||||
};
|
};
|
||||||
httpd_register_uri_handler(server, &file_datafile_last_part_handle);
|
httpd_register_uri_handler(server, &file_datafile_last_part_handle);
|
||||||
@@ -1180,26 +1222,24 @@ void register_server_file_uri(httpd_handle_t server, const char *base_path)
|
|||||||
httpd_uri_t file_logfileact = {
|
httpd_uri_t file_logfileact = {
|
||||||
.uri = "/logfileact", // Match all URIs of type /path/to/file
|
.uri = "/logfileact", // Match all URIs of type /path/to/file
|
||||||
.method = HTTP_GET,
|
.method = HTTP_GET,
|
||||||
.handler = logfileact_get_full_handler,
|
.handler = APPLY_BASIC_AUTH_FILTER(logfileact_get_full_handler),
|
||||||
.user_ctx = server_data // Pass server data as context
|
.user_ctx = server_data // Pass server data as context
|
||||||
};
|
};
|
||||||
httpd_register_uri_handler(server, &file_logfileact);
|
httpd_register_uri_handler(server, &file_logfileact);
|
||||||
|
|
||||||
|
|
||||||
httpd_uri_t file_logfile_last_part_handle = {
|
httpd_uri_t file_logfile_last_part_handle = {
|
||||||
.uri = "/log", // Match all URIs of type /path/to/file
|
.uri = "/log", // Match all URIs of type /path/to/file
|
||||||
.method = HTTP_GET,
|
.method = HTTP_GET,
|
||||||
.handler = logfileact_get_last_part_handler,
|
.handler = APPLY_BASIC_AUTH_FILTER(logfileact_get_last_part_handler),
|
||||||
.user_ctx = server_data // Pass server data as context
|
.user_ctx = server_data // Pass server data as context
|
||||||
};
|
};
|
||||||
httpd_register_uri_handler(server, &file_logfile_last_part_handle);
|
httpd_register_uri_handler(server, &file_logfile_last_part_handle);
|
||||||
|
|
||||||
|
|
||||||
/* URI handler for uploading files to server */
|
/* URI handler for uploading files to server */
|
||||||
httpd_uri_t file_upload = {
|
httpd_uri_t file_upload = {
|
||||||
.uri = "/upload/*", // Match all URIs of type /upload/path/to/file
|
.uri = "/upload/*", // Match all URIs of type /upload/path/to/file
|
||||||
.method = HTTP_POST,
|
.method = HTTP_POST,
|
||||||
.handler = upload_post_handler,
|
.handler = APPLY_BASIC_AUTH_FILTER(upload_post_handler),
|
||||||
.user_ctx = server_data // Pass server data as context
|
.user_ctx = server_data // Pass server data as context
|
||||||
};
|
};
|
||||||
httpd_register_uri_handler(server, &file_upload);
|
httpd_register_uri_handler(server, &file_upload);
|
||||||
@@ -1208,9 +1248,8 @@ void register_server_file_uri(httpd_handle_t server, const char *base_path)
|
|||||||
httpd_uri_t file_delete = {
|
httpd_uri_t file_delete = {
|
||||||
.uri = "/delete/*", // Match all URIs of type /delete/path/to/file
|
.uri = "/delete/*", // Match all URIs of type /delete/path/to/file
|
||||||
.method = HTTP_POST,
|
.method = HTTP_POST,
|
||||||
.handler = delete_post_handler,
|
.handler = APPLY_BASIC_AUTH_FILTER(delete_post_handler),
|
||||||
.user_ctx = server_data // Pass server data as context
|
.user_ctx = server_data // Pass server data as context
|
||||||
};
|
};
|
||||||
httpd_register_uri_handler(server, &file_delete);
|
httpd_register_uri_handler(server, &file_delete);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
void register_server_file_uri(httpd_handle_t server, const char *base_path);
|
void register_server_file_uri(httpd_handle_t server, const char *base_path);
|
||||||
|
|
||||||
void unzip(std::string _in_zip_file, std::string _target_directory);
|
void unzip(std::string _in_zip_file, std::string _target_directory);
|
||||||
std::string unzip_new(std::string _in_zip_file, std::string _target_zip, std::string _target_bin, std::string _main = "/sdcard/", bool _initial_setup = false);
|
std::string unzip_new(std::string _in_zip_file, std::string _html_tmp, std::string _html_final, std::string _target_bin, std::string _main = "/sdcard/", bool _initial_setup = false);
|
||||||
|
|
||||||
|
|
||||||
void delete_all_in_directory(std::string _directory);
|
void delete_all_in_directory(std::string _directory);
|
||||||
|
|||||||
@@ -16,59 +16,86 @@ extern "C" {
|
|||||||
|
|
||||||
#include "esp_err.h"
|
#include "esp_err.h"
|
||||||
#include "esp_log.h"
|
#include "esp_log.h"
|
||||||
|
|
||||||
#include "Helper.h"
|
#include "Helper.h"
|
||||||
|
|
||||||
#include "esp_http_server.h"
|
#include "esp_http_server.h"
|
||||||
|
|
||||||
#include "../../include/defines.h"
|
#include "../../include/defines.h"
|
||||||
|
|
||||||
|
|
||||||
static const char *TAG = "SERVER HELP";
|
static const char *TAG = "SERVER HELP";
|
||||||
|
|
||||||
char scratch[SERVER_HELPER_SCRATCH_BUFSIZE];
|
char scratch[SERVER_HELPER_SCRATCH_BUFSIZE];
|
||||||
|
|
||||||
|
bool endsWith(std::string const &str, std::string const &suffix)
|
||||||
bool endsWith(std::string const &str, std::string const &suffix) {
|
{
|
||||||
if (str.length() < suffix.length()) {
|
if (str.length() < suffix.length()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return str.compare(str.length() - suffix.length(), suffix.length(), suffix) == 0;
|
return str.compare(str.length() - suffix.length(), suffix.length(), suffix) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
esp_err_t send_file(httpd_req_t *req, std::string filename)
|
esp_err_t send_file(httpd_req_t *req, std::string filename)
|
||||||
{
|
{
|
||||||
|
std::string _filename_old = filename;
|
||||||
|
struct stat file_stat;
|
||||||
|
bool _gz_file_exists = false;
|
||||||
|
|
||||||
|
ESP_LOGD(TAG, "old filename: %s", filename.c_str());
|
||||||
|
std::string _filename_temp = std::string(filename) + ".gz";
|
||||||
|
|
||||||
|
// Checks whether the file is available as .gz
|
||||||
|
if (stat(_filename_temp.c_str(), &file_stat) == 0) {
|
||||||
|
filename = _filename_temp;
|
||||||
|
|
||||||
|
ESP_LOGD(TAG, "new filename: %s", filename.c_str());
|
||||||
|
_gz_file_exists = true;
|
||||||
|
}
|
||||||
|
|
||||||
FILE *fd = fopen(filename.c_str(), "r");
|
FILE *fd = fopen(filename.c_str(), "r");
|
||||||
if (!fd) {
|
if (!fd) {
|
||||||
ESP_LOGE(TAG, "Failed to read file: %s", filename.c_str());
|
ESP_LOGE(TAG, "Failed to read file: %s", filename.c_str());
|
||||||
|
|
||||||
/* Respond with 404 Error */
|
/* Respond with 404 Error */
|
||||||
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
|
httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
|
|
||||||
ESP_LOGD(TAG, "Sending file: %s ...", filename.c_str());
|
ESP_LOGD(TAG, "Sending file: %s ...", filename.c_str());
|
||||||
// httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
|
|
||||||
|
|
||||||
/* For all files with the following file extention tell
|
/* For all files with the following file extention tell the webbrowser to cache them for 12h */
|
||||||
the webbrowser to cache them for 24h */
|
|
||||||
if (endsWith(filename, ".html") ||
|
if (endsWith(filename, ".html") ||
|
||||||
endsWith(filename, ".htm") ||
|
endsWith(filename, ".htm") ||
|
||||||
|
endsWith(filename, ".xml") ||
|
||||||
endsWith(filename, ".css") ||
|
endsWith(filename, ".css") ||
|
||||||
endsWith(filename, ".js") ||
|
endsWith(filename, ".js") ||
|
||||||
endsWith(filename, ".map") ||
|
endsWith(filename, ".map") ||
|
||||||
endsWith(filename, ".jpg") ||
|
endsWith(filename, ".jpg") ||
|
||||||
endsWith(filename, ".jpeg") ||
|
endsWith(filename, ".jpeg") ||
|
||||||
endsWith(filename, ".ico") ||
|
endsWith(filename, ".ico") ||
|
||||||
endsWith(filename, ".png")) {
|
endsWith(filename, ".png") ||
|
||||||
httpd_resp_set_hdr(req, "Cache-Control", "max-age=86400");
|
endsWith(filename, ".gif") ||
|
||||||
}
|
// endsWith(filename, ".zip") ||
|
||||||
|
endsWith(filename, ".gz")) {
|
||||||
|
if (filename == "/sdcard/html/setup.html") {
|
||||||
|
httpd_resp_set_hdr(req, "Clear-Site-Data", "\"*\"");
|
||||||
set_content_type_from_file(req, filename.c_str());
|
set_content_type_from_file(req, filename.c_str());
|
||||||
|
}
|
||||||
|
else if (_gz_file_exists) {
|
||||||
|
httpd_resp_set_hdr(req, "Cache-Control", "max-age=43200");
|
||||||
|
httpd_resp_set_hdr(req, "Content-Encoding", "gzip");
|
||||||
|
set_content_type_from_file(req, _filename_old.c_str());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
httpd_resp_set_hdr(req, "Cache-Control", "max-age=43200");
|
||||||
|
set_content_type_from_file(req, filename.c_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
set_content_type_from_file(req, filename.c_str());
|
||||||
|
}
|
||||||
|
|
||||||
/* Retrieve the pointer to scratch buffer for temporary storage */
|
/* Retrieve the pointer to scratch buffer for temporary storage */
|
||||||
char *chunk = scratch;
|
char *chunk = scratch;
|
||||||
size_t chunksize;
|
size_t chunksize;
|
||||||
|
|
||||||
do {
|
do {
|
||||||
/* Read file in chunks into the scratch buffer */
|
/* Read file in chunks into the scratch buffer */
|
||||||
chunksize = fread(chunk, 1, SERVER_HELPER_SCRATCH_BUFSIZE, fd);
|
chunksize = fread(chunk, 1, SERVER_HELPER_SCRATCH_BUFSIZE, fd);
|
||||||
@@ -77,10 +104,13 @@ esp_err_t send_file(httpd_req_t *req, std::string filename)
|
|||||||
if (httpd_resp_send_chunk(req, chunk, chunksize) != ESP_OK) {
|
if (httpd_resp_send_chunk(req, chunk, chunksize) != ESP_OK) {
|
||||||
fclose(fd);
|
fclose(fd);
|
||||||
ESP_LOGE(TAG, "File sending failed!");
|
ESP_LOGE(TAG, "File sending failed!");
|
||||||
|
|
||||||
/* Abort sending file */
|
/* Abort sending file */
|
||||||
httpd_resp_sendstr_chunk(req, NULL);
|
httpd_resp_sendstr_chunk(req, NULL);
|
||||||
|
|
||||||
/* Respond with 500 Internal Server Error */
|
/* Respond with 500 Internal Server Error */
|
||||||
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to send file");
|
httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to send file");
|
||||||
|
|
||||||
return ESP_FAIL;
|
return ESP_FAIL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -90,12 +120,10 @@ esp_err_t send_file(httpd_req_t *req, std::string filename)
|
|||||||
/* Close file after sending complete */
|
/* Close file after sending complete */
|
||||||
fclose(fd);
|
fclose(fd);
|
||||||
ESP_LOGD(TAG, "File sending complete");
|
ESP_LOGD(TAG, "File sending complete");
|
||||||
|
|
||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* Copies the full path into destination buffer and returns
|
/* Copies the full path into destination buffer and returns
|
||||||
* pointer to path (skipping the preceding base path) */
|
* pointer to path (skipping the preceding base path) */
|
||||||
const char* get_path_from_uri(char *dest, const char *base_path, const char *uri, size_t destsize)
|
const char* get_path_from_uri(char *dest, const char *base_path, const char *uri, size_t destsize)
|
||||||
@@ -125,25 +153,49 @@ const char* get_path_from_uri(char *dest, const char *base_path, const char *uri
|
|||||||
return dest + base_pathlen;
|
return dest + base_pathlen;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Set HTTP response content type according to file extension */
|
/* Set HTTP response content type according to file extension */
|
||||||
esp_err_t set_content_type_from_file(httpd_req_t *req, const char *filename)
|
esp_err_t set_content_type_from_file(httpd_req_t *req, const char *filename)
|
||||||
{
|
{
|
||||||
if (IS_FILE_EXT(filename, ".pdf")) {
|
if (IS_FILE_EXT(filename, ".pdf")) {
|
||||||
return httpd_resp_set_type(req, "application/pdf");
|
return httpd_resp_set_type(req, "application/x-pdf");
|
||||||
} else if (IS_FILE_EXT(filename, ".html")) {
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".htm")) {
|
||||||
return httpd_resp_set_type(req, "text/html");
|
return httpd_resp_set_type(req, "text/html");
|
||||||
} else if (IS_FILE_EXT(filename, ".jpeg")) {
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".html")) {
|
||||||
|
return httpd_resp_set_type(req, "text/html");
|
||||||
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".jpeg")) {
|
||||||
return httpd_resp_set_type(req, "image/jpeg");
|
return httpd_resp_set_type(req, "image/jpeg");
|
||||||
} else if (IS_FILE_EXT(filename, ".jpg")) {
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".jpg")) {
|
||||||
return httpd_resp_set_type(req, "image/jpeg");
|
return httpd_resp_set_type(req, "image/jpeg");
|
||||||
} else if (IS_FILE_EXT(filename, ".ico")) {
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".gif")) {
|
||||||
|
return httpd_resp_set_type(req, "image/gif");
|
||||||
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".png")) {
|
||||||
|
return httpd_resp_set_type(req, "image/png");
|
||||||
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".ico")) {
|
||||||
return httpd_resp_set_type(req, "image/x-icon");
|
return httpd_resp_set_type(req, "image/x-icon");
|
||||||
} else if (IS_FILE_EXT(filename, ".js")) {
|
}
|
||||||
return httpd_resp_set_type(req, "text/javascript");
|
else if (IS_FILE_EXT(filename, ".js")) {
|
||||||
} else if (IS_FILE_EXT(filename, ".css")) {
|
return httpd_resp_set_type(req, "application/javascript");
|
||||||
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".css")) {
|
||||||
return httpd_resp_set_type(req, "text/css");
|
return httpd_resp_set_type(req, "text/css");
|
||||||
}
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".xml")) {
|
||||||
|
return httpd_resp_set_type(req, "text/xml");
|
||||||
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".zip")) {
|
||||||
|
return httpd_resp_set_type(req, "application/x-zip");
|
||||||
|
}
|
||||||
|
else if (IS_FILE_EXT(filename, ".gz")) {
|
||||||
|
return httpd_resp_set_type(req, "application/x-gzip");
|
||||||
|
}
|
||||||
|
|
||||||
/* This is a limited set only */
|
/* This is a limited set only */
|
||||||
/* For any other type always set as plain text */
|
/* For any other type always set as plain text */
|
||||||
return httpd_resp_set_type(req, "text/plain");
|
return httpd_resp_set_type(req, "text/plain");
|
||||||
|
|||||||
@@ -3,7 +3,10 @@
|
|||||||
#include <string>
|
#include <string>
|
||||||
#include "string.h"
|
#include "string.h"
|
||||||
|
|
||||||
#include <esp_int_wdt.h>
|
/* TODO Rethink the usage of the int watchdog. It is no longer to be used, see
|
||||||
|
https://docs.espressif.com/projects/esp-idf/en/latest/esp32/migration-guides/release-5.x/5.0/system.html?highlight=esp_int_wdt */
|
||||||
|
#include "esp_private/esp_int_wdt.h"
|
||||||
|
|
||||||
#include <esp_task_wdt.h>
|
#include <esp_task_wdt.h>
|
||||||
|
|
||||||
|
|
||||||
@@ -11,14 +14,13 @@
|
|||||||
#include "freertos/FreeRTOS.h"
|
#include "freertos/FreeRTOS.h"
|
||||||
#include "freertos/task.h"
|
#include "freertos/task.h"
|
||||||
#include "esp_system.h"
|
#include "esp_system.h"
|
||||||
#include "esp_event.h"
|
|
||||||
#include "esp_event.h"
|
|
||||||
#include "esp_log.h"
|
#include "esp_log.h"
|
||||||
#include <esp_ota_ops.h>
|
#include <esp_ota_ops.h>
|
||||||
#include "esp_http_client.h"
|
#include "esp_http_client.h"
|
||||||
#include "esp_flash_partitions.h"
|
#include "esp_flash_partitions.h"
|
||||||
#include "esp_partition.h"
|
#include "esp_partition.h"
|
||||||
#include <nvs.h>
|
#include <nvs.h>
|
||||||
|
#include "esp_app_format.h"
|
||||||
#include "nvs_flash.h"
|
#include "nvs_flash.h"
|
||||||
#include "driver/gpio.h"
|
#include "driver/gpio.h"
|
||||||
// #include "protocol_examples_common.h"
|
// #include "protocol_examples_common.h"
|
||||||
@@ -26,7 +28,7 @@
|
|||||||
|
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
|
|
||||||
#include "server_tflite.h"
|
#include "MainFlowControl.h"
|
||||||
#include "server_file.h"
|
#include "server_file.h"
|
||||||
#include "server_GPIO.h"
|
#include "server_GPIO.h"
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
@@ -39,6 +41,8 @@
|
|||||||
#include "ClassLogFile.h"
|
#include "ClassLogFile.h"
|
||||||
|
|
||||||
#include "Helper.h"
|
#include "Helper.h"
|
||||||
|
#include "statusled.h"
|
||||||
|
#include "basic_auth.h"
|
||||||
#include "../../include/defines.h"
|
#include "../../include/defines.h"
|
||||||
|
|
||||||
/*an ota data write buffer ready to write to the flash*/
|
/*an ota data write buffer ready to write to the flash*/
|
||||||
@@ -56,9 +60,9 @@ bool initial_setup = false;
|
|||||||
static void infinite_loop(void)
|
static void infinite_loop(void)
|
||||||
{
|
{
|
||||||
int i = 0;
|
int i = 0;
|
||||||
ESP_LOGI(TAG, "When a new firmware is available on the server, press the reset button to download it");
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "When a new firmware is available on the server, press the reset button to download it");
|
||||||
while(1) {
|
while(1) {
|
||||||
ESP_LOGI(TAG, "Waiting for a new firmware... %d", ++i);
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Waiting for a new firmware... (" + to_string(++i) + ")");
|
||||||
vTaskDelay(1000 / portTICK_PERIOD_MS);
|
vTaskDelay(1000 / portTICK_PERIOD_MS);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -66,34 +70,51 @@ static void infinite_loop(void)
|
|||||||
|
|
||||||
void task_do_Update_ZIP(void *pvParameter)
|
void task_do_Update_ZIP(void *pvParameter)
|
||||||
{
|
{
|
||||||
|
StatusLED(AP_OR_OTA, 1, true); // Signaling an OTA update
|
||||||
|
|
||||||
std::string filetype = toUpper(getFileType(_file_name_update));
|
std::string filetype = toUpper(getFileType(_file_name_update));
|
||||||
|
|
||||||
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "File: " + _file_name_update + " Filetype: " + filetype);
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "File: " + _file_name_update + " Filetype: " + filetype);
|
||||||
|
|
||||||
|
|
||||||
if (filetype == "ZIP")
|
if (filetype == "ZIP")
|
||||||
{
|
{
|
||||||
std::string in, out, outbin, zw, retfirmware;
|
std::string in, outHtml, outHtmlTmp, outHtmlOld, outbin, zw, retfirmware;
|
||||||
|
|
||||||
out = "/sdcard/html";
|
outHtml = "/sdcard/html";
|
||||||
|
outHtmlTmp = "/sdcard/html_tmp";
|
||||||
|
outHtmlOld = "/sdcard/html_old";
|
||||||
outbin = "/sdcard/firmware";
|
outbin = "/sdcard/firmware";
|
||||||
|
|
||||||
retfirmware = unzip_new(_file_name_update, out+"/", outbin+"/", "/sdcard/", initial_setup);
|
/* Remove the old and tmp html folder in case they still exist */
|
||||||
|
removeFolder(outHtmlTmp.c_str(), TAG);
|
||||||
|
removeFolder(outHtmlOld.c_str(), TAG);
|
||||||
|
|
||||||
|
/* Extract the ZIP file. The content of the html folder gets extracted to the temporar folder html-temp. */
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Extracting ZIP file " + _file_name_update + "...");
|
||||||
|
retfirmware = unzip_new(_file_name_update, outHtmlTmp+"/", outHtml+"/", outbin+"/", "/sdcard/", initial_setup);
|
||||||
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Files unzipped.");
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Files unzipped.");
|
||||||
|
|
||||||
|
/* ZIP file got extracted, replace the old html folder with the new one */
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Renaming folder " + outHtml + " to " + outHtmlOld + "...");
|
||||||
|
RenameFolder(outHtml, outHtmlOld);
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Renaming folder " + outHtmlTmp + " to " + outHtml + "...");
|
||||||
|
RenameFolder(outHtmlTmp, outHtml);
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Deleting folder " + outHtmlOld + "...");
|
||||||
|
removeFolder(outHtmlOld.c_str(), TAG);
|
||||||
|
|
||||||
if (retfirmware.length() > 0)
|
if (retfirmware.length() > 0)
|
||||||
{
|
{
|
||||||
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Found firmware.bin");
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Found firmware.bin");
|
||||||
ota_update_task(retfirmware);
|
ota_update_task(retfirmware);
|
||||||
}
|
}
|
||||||
|
|
||||||
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Trigger reboot due to firmware update.");
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Trigger reboot due to firmware update");
|
||||||
doRebootOTA();
|
doRebootOTA();
|
||||||
} else if (filetype == "BIN")
|
} else if (filetype == "BIN")
|
||||||
{
|
{
|
||||||
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Do firmware update - file: " + _file_name_update);
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Do firmware update - file: " + _file_name_update);
|
||||||
ota_update_task(_file_name_update);
|
ota_update_task(_file_name_update);
|
||||||
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Trigger reboot due to firmware update.");
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Trigger reboot due to firmware update");
|
||||||
doRebootOTA();
|
doRebootOTA();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -108,7 +129,7 @@ void CheckUpdate()
|
|||||||
FILE *pfile;
|
FILE *pfile;
|
||||||
if ((pfile = fopen("/sdcard/update.txt", "r")) == NULL)
|
if ((pfile = fopen("/sdcard/update.txt", "r")) == NULL)
|
||||||
{
|
{
|
||||||
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "No update triggered.");
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "No pending update");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -120,13 +141,13 @@ void CheckUpdate()
|
|||||||
std::string _szw = std::string(zw);
|
std::string _szw = std::string(zw);
|
||||||
if (_szw == "init")
|
if (_szw == "init")
|
||||||
{
|
{
|
||||||
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Inital Setup triggered.");
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Inital Setup triggered");
|
||||||
initial_setup = true; }
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fclose(pfile);
|
fclose(pfile);
|
||||||
DeleteFile("/sdcard/update.txt"); // Prevent Boot Loop!!!
|
DeleteFile("/sdcard/update.txt"); // Prevent Boot Loop!!!
|
||||||
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Update during boot triggered - Update File: " + _file_name_update);
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Start update process (" + _file_name_update + ")");
|
||||||
|
|
||||||
|
|
||||||
xTaskCreate(&task_do_Update_ZIP, "task_do_Update_ZIP", configMINIMAL_STACK_SIZE * 35, NULL, tskIDLE_PRIORITY+1, NULL);
|
xTaskCreate(&task_do_Update_ZIP, "task_do_Update_ZIP", configMINIMAL_STACK_SIZE * 35, NULL, tskIDLE_PRIORITY+1, NULL);
|
||||||
@@ -149,17 +170,17 @@ static bool ota_update_task(std::string fn)
|
|||||||
const esp_partition_t *running = esp_ota_get_running_partition();
|
const esp_partition_t *running = esp_ota_get_running_partition();
|
||||||
|
|
||||||
if (configured != running) {
|
if (configured != running) {
|
||||||
ESP_LOGW(TAG, "Configured OTA boot partition at offset 0x%08x, but running from offset 0x%08x",
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Configured OTA boot partition at offset " + to_string(configured->address) +
|
||||||
configured->address, running->address);
|
", but running from offset " + to_string(running->address));
|
||||||
ESP_LOGW(TAG, "(This can happen if either the OTA boot data or preferred boot image become somehow corrupted.)");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "(This can happen if either the OTA boot data or preferred boot image become somehow corrupted.)");
|
||||||
}
|
}
|
||||||
ESP_LOGI(TAG, "Running partition type %d subtype %d (offset 0x%08x)",
|
ESP_LOGI(TAG, "Running partition type %d subtype %d (offset 0x%08x)",
|
||||||
running->type, running->subtype, running->address);
|
running->type, running->subtype, (unsigned int)running->address);
|
||||||
|
|
||||||
|
|
||||||
update_partition = esp_ota_get_next_update_partition(NULL);
|
update_partition = esp_ota_get_next_update_partition(NULL);
|
||||||
ESP_LOGI(TAG, "Writing to partition subtype %d at offset 0x%x",
|
ESP_LOGI(TAG, "Writing to partition subtype %d at offset 0x%x",
|
||||||
update_partition->subtype, update_partition->address);
|
update_partition->subtype, (unsigned int)update_partition->address);
|
||||||
// assert(update_partition != NULL);
|
// assert(update_partition != NULL);
|
||||||
|
|
||||||
int binary_file_length = 0;
|
int binary_file_length = 0;
|
||||||
@@ -179,7 +200,7 @@ static bool ota_update_task(std::string fn)
|
|||||||
|
|
||||||
while (data_read > 0) {
|
while (data_read > 0) {
|
||||||
if (data_read < 0) {
|
if (data_read < 0) {
|
||||||
ESP_LOGE(TAG, "Error: SSL data read error");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Error: SSL data read error");
|
||||||
return false;
|
return false;
|
||||||
} else if (data_read > 0) {
|
} else if (data_read > 0) {
|
||||||
if (image_header_was_checked == false) {
|
if (image_header_was_checked == false) {
|
||||||
@@ -203,16 +224,17 @@ static bool ota_update_task(std::string fn)
|
|||||||
// check current version with last invalid partition
|
// check current version with last invalid partition
|
||||||
if (last_invalid_app != NULL) {
|
if (last_invalid_app != NULL) {
|
||||||
if (memcmp(invalid_app_info.version, new_app_info.version, sizeof(new_app_info.version)) == 0) {
|
if (memcmp(invalid_app_info.version, new_app_info.version, sizeof(new_app_info.version)) == 0) {
|
||||||
ESP_LOGW(TAG, "New version is the same as invalid version.");
|
LogFile.WriteToFile(ESP_LOG_WARN, TAG, "New version is the same as invalid version");
|
||||||
ESP_LOGW(TAG, "Previously, there was an attempt to launch the firmware with %s version, but it failed.", invalid_app_info.version);
|
LogFile.WriteToFile(ESP_LOG_WARN, TAG, "Previously, there was an attempt to launch the firmware with " +
|
||||||
ESP_LOGW(TAG, "The firmware has been rolled back to the previous version.");
|
string(invalid_app_info.version) + " version, but it failed");
|
||||||
|
LogFile.WriteToFile(ESP_LOG_WARN, TAG, "The firmware has been rolled back to the previous version");
|
||||||
infinite_loop();
|
infinite_loop();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
if (memcmp(new_app_info.version, running_app_info.version, sizeof(new_app_info.version)) == 0) {
|
if (memcmp(new_app_info.version, running_app_info.version, sizeof(new_app_info.version)) == 0) {
|
||||||
ESP_LOGW(TAG, "Current running version is the same as a new. We will not continue the update.");
|
LogFile.WriteToFile(ESP_LOG_WARN, TAG, "Current running version is the same as a new. We will not continue the update");
|
||||||
infinite_loop();
|
infinite_loop();
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
@@ -220,12 +242,12 @@ static bool ota_update_task(std::string fn)
|
|||||||
|
|
||||||
err = esp_ota_begin(update_partition, OTA_SIZE_UNKNOWN, &update_handle);
|
err = esp_ota_begin(update_partition, OTA_SIZE_UNKNOWN, &update_handle);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
ESP_LOGE(TAG, "esp_ota_begin failed (%s)", esp_err_to_name(err));
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "esp_ota_begin failed (" + string(esp_err_to_name(err)) + ")");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
ESP_LOGI(TAG, "esp_ota_begin succeeded");
|
ESP_LOGI(TAG, "esp_ota_begin succeeded");
|
||||||
} else {
|
} else {
|
||||||
ESP_LOGE(TAG, "received package is not fit len");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "received package is not fit len");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -241,7 +263,7 @@ static bool ota_update_task(std::string fn)
|
|||||||
// * `errno` to check for underlying transport connectivity closure if any
|
// * `errno` to check for underlying transport connectivity closure if any
|
||||||
//
|
//
|
||||||
if (errno == ECONNRESET || errno == ENOTCONN) {
|
if (errno == ECONNRESET || errno == ENOTCONN) {
|
||||||
ESP_LOGE(TAG, "Connection closed, errno = %d", errno);
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Connection closed, errno = " + to_string(errno));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -254,15 +276,15 @@ static bool ota_update_task(std::string fn)
|
|||||||
err = esp_ota_end(update_handle);
|
err = esp_ota_end(update_handle);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
if (err == ESP_ERR_OTA_VALIDATE_FAILED) {
|
if (err == ESP_ERR_OTA_VALIDATE_FAILED) {
|
||||||
ESP_LOGE(TAG, "Image validation failed, image is corrupted");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Image validation failed, image is corrupted");
|
||||||
}
|
}
|
||||||
ESP_LOGE(TAG, "esp_ota_end failed (%s)!", esp_err_to_name(err));
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "esp_ota_end failed (" + string(esp_err_to_name(err)) + ")!");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = esp_ota_set_boot_partition(update_partition);
|
err = esp_ota_set_boot_partition(update_partition);
|
||||||
if (err != ESP_OK) {
|
if (err != ESP_OK) {
|
||||||
ESP_LOGE(TAG, "esp_ota_set_boot_partition failed (%s)!", esp_err_to_name(err));
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "esp_ota_set_boot_partition failed (" + string(esp_err_to_name(err)) + ")!");
|
||||||
|
|
||||||
}
|
}
|
||||||
// ESP_LOGI(TAG, "Prepare to restart system!");
|
// ESP_LOGI(TAG, "Prepare to restart system!");
|
||||||
@@ -329,7 +351,7 @@ void CheckOTAUpdate(void)
|
|||||||
ESP_LOGI(TAG, "Diagnostics completed successfully! Continuing execution...");
|
ESP_LOGI(TAG, "Diagnostics completed successfully! Continuing execution...");
|
||||||
esp_ota_mark_app_valid_cancel_rollback();
|
esp_ota_mark_app_valid_cancel_rollback();
|
||||||
} else {
|
} else {
|
||||||
ESP_LOGE(TAG, "Diagnostics failed! Start rollback to the previous version...");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Diagnostics failed! Start rollback to the previous version...");
|
||||||
esp_ota_mark_app_invalid_rollback_and_reboot();
|
esp_ota_mark_app_invalid_rollback_and_reboot();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -353,7 +375,7 @@ void CheckOTAUpdate(void)
|
|||||||
ESP_LOGI(TAG, "Diagnostics completed successfully! Continuing execution...");
|
ESP_LOGI(TAG, "Diagnostics completed successfully! Continuing execution...");
|
||||||
esp_ota_mark_app_valid_cancel_rollback();
|
esp_ota_mark_app_valid_cancel_rollback();
|
||||||
} else {
|
} else {
|
||||||
ESP_LOGE(TAG, "Diagnostics failed! Start rollback to the previous version...");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Diagnostics failed! Start rollback to the previous version...");
|
||||||
esp_ota_mark_app_invalid_rollback_and_reboot();
|
esp_ota_mark_app_invalid_rollback_and_reboot();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -427,7 +449,6 @@ esp_err_t handler_ota_update(httpd_req_t *req)
|
|||||||
return ESP_OK;
|
return ESP_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if ((filetype == "TFLITE") || (filetype == "TFL"))
|
if ((filetype == "TFLITE") || (filetype == "TFL"))
|
||||||
{
|
{
|
||||||
std::string out = "/sdcard/config/" + getFileFullFileName(fn);
|
std::string out = "/sdcard/config/" + getFileFullFileName(fn);
|
||||||
@@ -445,7 +466,7 @@ esp_err_t handler_ota_update(httpd_req_t *req)
|
|||||||
if ((filetype == "ZIP") || (filetype == "BIN"))
|
if ((filetype == "ZIP") || (filetype == "BIN"))
|
||||||
{
|
{
|
||||||
FILE *pfile;
|
FILE *pfile;
|
||||||
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Update for reboot.");
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Update for reboot");
|
||||||
pfile = fopen("/sdcard/update.txt", "w");
|
pfile = fopen("/sdcard/update.txt", "w");
|
||||||
fwrite(fn.c_str(), fn.length(), 1, pfile);
|
fwrite(fn.c_str(), fn.length(), 1, pfile);
|
||||||
fclose(pfile);
|
fclose(pfile);
|
||||||
@@ -463,7 +484,7 @@ esp_err_t handler_ota_update(httpd_req_t *req)
|
|||||||
{
|
{
|
||||||
const char* resp_str;
|
const char* resp_str;
|
||||||
|
|
||||||
KillTFliteTasks();
|
DeleteMainFlowTask();
|
||||||
gpio_handler_deinit();
|
gpio_handler_deinit();
|
||||||
if (ota_update_task(fn))
|
if (ota_update_task(fn))
|
||||||
{
|
{
|
||||||
@@ -522,7 +543,7 @@ esp_err_t handler_ota_update(httpd_req_t *req)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
ESP_LOGD(TAG, "File does not exist: %s", fn.c_str());
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "File does not exist: " + fn);
|
||||||
}
|
}
|
||||||
/* Respond with an empty chunk to signal HTTP response completion */
|
/* Respond with an empty chunk to signal HTTP response completion */
|
||||||
std::string zw = "file deleted\n";
|
std::string zw = "file deleted\n";
|
||||||
@@ -537,12 +558,12 @@ esp_err_t handler_ota_update(httpd_req_t *req)
|
|||||||
httpd_resp_send(req, zw.c_str(), strlen(zw.c_str()));
|
httpd_resp_send(req, zw.c_str(), strlen(zw.c_str()));
|
||||||
httpd_resp_send_chunk(req, NULL, 0);
|
httpd_resp_send_chunk(req, NULL, 0);
|
||||||
|
|
||||||
ESP_LOGE(TAG, "ota without parameter - should not be the case!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "ota without parameter - should not be the case!");
|
||||||
|
|
||||||
/*
|
/*
|
||||||
const char* resp_str;
|
const char* resp_str;
|
||||||
|
|
||||||
KillTFliteTasks();
|
DeleteMainFlowTask();
|
||||||
gpio_handler_deinit();
|
gpio_handler_deinit();
|
||||||
if (ota_update_task(fn))
|
if (ota_update_task(fn))
|
||||||
{
|
{
|
||||||
@@ -566,13 +587,19 @@ esp_err_t handler_ota_update(httpd_req_t *req)
|
|||||||
|
|
||||||
void hard_restart()
|
void hard_restart()
|
||||||
{
|
{
|
||||||
esp_task_wdt_init(1,true);
|
esp_task_wdt_config_t twdt_config = {
|
||||||
|
.timeout_ms = 1,
|
||||||
|
.idle_core_mask = (1 << portNUM_PROCESSORS) - 1, // Bitmask of all cores
|
||||||
|
.trigger_panic = true,
|
||||||
|
};
|
||||||
|
ESP_ERROR_CHECK(esp_task_wdt_init(&twdt_config));
|
||||||
|
|
||||||
esp_task_wdt_add(NULL);
|
esp_task_wdt_add(NULL);
|
||||||
while(true);
|
while(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void task_reboot(void *KillAutoFlow)
|
void task_reboot(void *DeleteMainFlow)
|
||||||
{
|
{
|
||||||
// write a reboot, to identify a reboot by purpouse
|
// write a reboot, to identify a reboot by purpouse
|
||||||
FILE* pfile = fopen("/sdcard/reboot.txt", "w");
|
FILE* pfile = fopen("/sdcard/reboot.txt", "w");
|
||||||
@@ -582,10 +609,13 @@ void task_reboot(void *KillAutoFlow)
|
|||||||
|
|
||||||
vTaskDelay(3000 / portTICK_PERIOD_MS);
|
vTaskDelay(3000 / portTICK_PERIOD_MS);
|
||||||
|
|
||||||
if ((bool)KillAutoFlow) {
|
if ((bool)DeleteMainFlow) {
|
||||||
KillTFliteTasks(); // Kill autoflow task if executed in extra task, if not don't kill parent task
|
DeleteMainFlowTask(); // Kill autoflow task if executed in extra task, if not don't kill parent task
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Camera.LightOnOff(false);
|
||||||
|
StatusLEDOff();
|
||||||
|
|
||||||
/* Stop service tasks */
|
/* Stop service tasks */
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
MQTTdestroy_client(true);
|
MQTTdestroy_client(true);
|
||||||
@@ -600,20 +630,20 @@ void task_reboot(void *KillAutoFlow)
|
|||||||
vTaskDelay(5000 / portTICK_PERIOD_MS);
|
vTaskDelay(5000 / portTICK_PERIOD_MS);
|
||||||
hard_restart(); // Reset type: System reset (Triggered by watchdog), if esp_restart stalls (WDT needs to be activated)
|
hard_restart(); // Reset type: System reset (Triggered by watchdog), if esp_restart stalls (WDT needs to be activated)
|
||||||
|
|
||||||
ESP_LOGE(TAG, "Reboot failed!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Reboot failed!");
|
||||||
vTaskDelete(NULL); //Delete this task if it comes to this point
|
vTaskDelete(NULL); //Delete this task if it comes to this point
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void doReboot()
|
void doReboot()
|
||||||
{
|
{
|
||||||
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Reboot triggered by Software (5s).");
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Reboot triggered by Software (5s)");
|
||||||
LogFile.WriteToFile(ESP_LOG_WARN, TAG, "Reboot in 5sec");
|
LogFile.WriteToFile(ESP_LOG_WARN, TAG, "Reboot in 5sec");
|
||||||
|
|
||||||
BaseType_t xReturned = xTaskCreate(&task_reboot, "task_reboot", configMINIMAL_STACK_SIZE * 3, (void*) true, 10, NULL);
|
BaseType_t xReturned = xTaskCreate(&task_reboot, "task_reboot", configMINIMAL_STACK_SIZE * 4, (void*) true, 10, NULL);
|
||||||
if( xReturned != pdPASS )
|
if( xReturned != pdPASS )
|
||||||
{
|
{
|
||||||
ESP_LOGE(TAG, "task_reboot not created -> force reboot without killing flow");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "task_reboot not created -> force reboot without killing flow");
|
||||||
task_reboot((void*) false);
|
task_reboot((void*) false);
|
||||||
}
|
}
|
||||||
vTaskDelay(10000 / portTICK_PERIOD_MS); // Prevent serving web client fetch response until system is shuting down
|
vTaskDelay(10000 / portTICK_PERIOD_MS); // Prevent serving web client fetch response until system is shuting down
|
||||||
@@ -624,6 +654,8 @@ void doRebootOTA()
|
|||||||
{
|
{
|
||||||
LogFile.WriteToFile(ESP_LOG_WARN, TAG, "Reboot in 5sec");
|
LogFile.WriteToFile(ESP_LOG_WARN, TAG, "Reboot in 5sec");
|
||||||
|
|
||||||
|
Camera.LightOnOff(false);
|
||||||
|
StatusLEDOff();
|
||||||
esp_camera_deinit();
|
esp_camera_deinit();
|
||||||
|
|
||||||
vTaskDelay(5000 / portTICK_PERIOD_MS);
|
vTaskDelay(5000 / portTICK_PERIOD_MS);
|
||||||
@@ -641,7 +673,7 @@ esp_err_t handler_reboot(httpd_req_t *req)
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "handler_reboot");
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "handler_reboot");
|
||||||
ESP_LOGI(TAG, "!!! System will restart within 5 sec!!!");
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "!!! System will restart within 5 sec!!!");
|
||||||
|
|
||||||
std::string response =
|
std::string response =
|
||||||
"<html><head><script>"
|
"<html><head><script>"
|
||||||
@@ -673,13 +705,13 @@ void register_server_ota_sdcard_uri(httpd_handle_t server)
|
|||||||
httpd_uri_t camuri = { };
|
httpd_uri_t camuri = { };
|
||||||
camuri.method = HTTP_GET;
|
camuri.method = HTTP_GET;
|
||||||
camuri.uri = "/ota";
|
camuri.uri = "/ota";
|
||||||
camuri.handler = handler_ota_update;
|
camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_ota_update);
|
||||||
camuri.user_ctx = (void*) "Do OTA";
|
camuri.user_ctx = (void*) "Do OTA";
|
||||||
httpd_register_uri_handler(server, &camuri);
|
httpd_register_uri_handler(server, &camuri);
|
||||||
|
|
||||||
camuri.method = HTTP_GET;
|
camuri.method = HTTP_GET;
|
||||||
camuri.uri = "/reboot";
|
camuri.uri = "/reboot";
|
||||||
camuri.handler = handler_reboot;
|
camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_reboot);
|
||||||
camuri.user_ctx = (void*) "Reboot";
|
camuri.user_ctx = (void*) "Reboot";
|
||||||
httpd_register_uri_handler(server, &camuri);
|
httpd_register_uri_handler(server, &camuri);
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,6 @@ FILE(GLOB_RECURSE app_sources ${CMAKE_CURRENT_SOURCE_DIR}/*.*)
|
|||||||
|
|
||||||
idf_component_register(SRCS ${app_sources}
|
idf_component_register(SRCS ${app_sources}
|
||||||
INCLUDE_DIRS "."
|
INCLUDE_DIRS "."
|
||||||
REQUIRES jomjol_tfliteclass jomjol_helper jomjol_controlcamera jomjol_mqtt jomjol_influxdb jomjol_fileserver_ota jomjol_image_proc jomjol_wlan)
|
REQUIRES esp_timer esp_wifi jomjol_tfliteclass jomjol_helper jomjol_controlcamera jomjol_mqtt jomjol_influxdb jomjol_webhook jomjol_fileserver_ota jomjol_image_proc jomjol_wlan openmetrics)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -67,11 +67,6 @@ string ClassFlow::getHTMLSingleStep(string host){
|
|||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
string ClassFlow::getReadout()
|
|
||||||
{
|
|
||||||
return string();
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string ClassFlow::GetParameterName(std::string _input)
|
std::string ClassFlow::GetParameterName(std::string _input)
|
||||||
{
|
{
|
||||||
string _param;
|
string _param;
|
||||||
|
|||||||
@@ -46,7 +46,6 @@ public:
|
|||||||
virtual bool ReadParameter(FILE* pfile, string &aktparamgraph);
|
virtual bool ReadParameter(FILE* pfile, string &aktparamgraph);
|
||||||
virtual bool doFlow(string time);
|
virtual bool doFlow(string time);
|
||||||
virtual string getHTMLSingleStep(string host);
|
virtual string getHTMLSingleStep(string host);
|
||||||
virtual string getReadout();
|
|
||||||
virtual string name(){return "ClassFlow";};
|
virtual string name(){return "ClassFlow";};
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,26 +1,23 @@
|
|||||||
#include "ClassFlowAlignment.h"
|
#include "ClassFlowAlignment.h"
|
||||||
#include "ClassFlowTakeImage.h"
|
#include "ClassFlowTakeImage.h"
|
||||||
#include "ClassFlow.h"
|
#include "ClassFlow.h"
|
||||||
#include "server_tflite.h"
|
#include "MainFlowControl.h"
|
||||||
|
|
||||||
#include "CRotateImage.h"
|
#include "CRotateImage.h"
|
||||||
#include "esp_log.h"
|
#include "esp_log.h"
|
||||||
|
|
||||||
|
|
||||||
#include "ClassLogFile.h"
|
#include "ClassLogFile.h"
|
||||||
|
#include "psram.h"
|
||||||
#include "../../include/defines.h"
|
#include "../../include/defines.h"
|
||||||
|
|
||||||
|
|
||||||
static const char *TAG = "ALIGN";
|
static const char *TAG = "ALIGN";
|
||||||
|
|
||||||
// #define DEBUG_DETAIL_ON
|
// #define DEBUG_DETAIL_ON
|
||||||
|
|
||||||
|
|
||||||
void ClassFlowAlignment::SetInitialParameter(void)
|
void ClassFlowAlignment::SetInitialParameter(void)
|
||||||
{
|
{
|
||||||
initalrotate = 0;
|
initialrotate = 0;
|
||||||
anz_ref = 0;
|
anz_ref = 0;
|
||||||
initialmirror = false;
|
|
||||||
use_antialiasing = false;
|
use_antialiasing = false;
|
||||||
initialflip = false;
|
initialflip = false;
|
||||||
SaveAllFiles = false;
|
SaveAllFiles = false;
|
||||||
@@ -30,133 +27,136 @@ void ClassFlowAlignment::SetInitialParameter(void)
|
|||||||
AlignAndCutImage = NULL;
|
AlignAndCutImage = NULL;
|
||||||
ImageBasis = NULL;
|
ImageBasis = NULL;
|
||||||
ImageTMP = NULL;
|
ImageTMP = NULL;
|
||||||
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
||||||
AlgROI = (ImageData*)heap_caps_malloc(sizeof(ImageData), MALLOC_CAP_8BIT | MALLOC_CAP_SPIRAM);
|
AlgROI = (ImageData *)malloc_psram_heap(std::string(TAG) + "->AlgROI", sizeof(ImageData), MALLOC_CAP_8BIT | MALLOC_CAP_SPIRAM);
|
||||||
#endif
|
#endif
|
||||||
previousElement = NULL;
|
previousElement = NULL;
|
||||||
disabled = false;
|
disabled = false;
|
||||||
SAD_criteria = 0.05;
|
SAD_criteria = 0.05;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ClassFlowAlignment::ClassFlowAlignment(std::vector<ClassFlow *> *lfc)
|
||||||
ClassFlowAlignment::ClassFlowAlignment(std::vector<ClassFlow*>* lfc)
|
|
||||||
{
|
{
|
||||||
SetInitialParameter();
|
SetInitialParameter();
|
||||||
ListFlowControll = lfc;
|
ListFlowControll = lfc;
|
||||||
|
|
||||||
for (int i = 0; i < ListFlowControll->size(); ++i)
|
for (int i = 0; i < ListFlowControll->size(); ++i) {
|
||||||
{
|
if (((*ListFlowControll)[i])->name().compare("ClassFlowTakeImage") == 0) {
|
||||||
if (((*ListFlowControll)[i])->name().compare("ClassFlowTakeImage") == 0)
|
ImageBasis = ((ClassFlowTakeImage *)(*ListFlowControll)[i])->rawImage;
|
||||||
{
|
|
||||||
ImageBasis = ((ClassFlowTakeImage*) (*ListFlowControll)[i])->rawImage;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!ImageBasis) // the function take pictures does not exist --> must be created first ONLY FOR TEST PURPOSES
|
// the function take pictures does not exist --> must be created first ONLY FOR TEST PURPOSES
|
||||||
{
|
if (!ImageBasis) {
|
||||||
ESP_LOGD(TAG, "CImageBasis had to be created");
|
ESP_LOGD(TAG, "CImageBasis had to be created");
|
||||||
ImageBasis = new CImageBasis(namerawimage);
|
ImageBasis = new CImageBasis("ImageBasis", namerawimage);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool ClassFlowAlignment::ReadParameter(FILE *pfile, string &aktparamgraph)
|
||||||
bool ClassFlowAlignment::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|
||||||
{
|
{
|
||||||
std::vector<string> splitted;
|
std::vector<string> splitted;
|
||||||
int suchex = 40;
|
int suchex = 40;
|
||||||
int suchey = 40;
|
int suchey = 40;
|
||||||
int alg_algo = 0; //default=0; 1 =HIGHACCURACY; 2= FAST; 3= OFF //add disable aligment algo |01.2023
|
int alg_algo = 0; // default=0; 1 =HIGHACCURACY; 2= FAST; 3= OFF //add disable aligment algo |01.2023
|
||||||
|
|
||||||
|
|
||||||
aktparamgraph = trim(aktparamgraph);
|
aktparamgraph = trim(aktparamgraph);
|
||||||
|
|
||||||
if (aktparamgraph.size() == 0)
|
if (aktparamgraph.size() == 0)
|
||||||
if (!this->GetNextParagraph(pfile, aktparamgraph))
|
{
|
||||||
|
if (!this->GetNextParagraph(pfile, aktparamgraph)) {
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (aktparamgraph.compare("[Alignment]") != 0) //Paragraph does not fit Alignment
|
if (aktparamgraph.compare("[Alignment]") != 0)
|
||||||
|
{
|
||||||
|
// Paragraph does not fit Alignment
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
while (this->getNextLine(pfile, &aktparamgraph) && !this->isNewParagraph(aktparamgraph))
|
while (this->getNextLine(pfile, &aktparamgraph) && !this->isNewParagraph(aktparamgraph))
|
||||||
{
|
{
|
||||||
splitted = ZerlegeZeile(aktparamgraph);
|
splitted = ZerlegeZeile(aktparamgraph);
|
||||||
if ((toUpper(splitted[0]) == "FLIPIMAGESIZE") && (splitted.size() > 1))
|
|
||||||
{
|
if ((toUpper(splitted[0]) == "FLIPIMAGESIZE") && (splitted.size() > 1)) {
|
||||||
if (toUpper(splitted[1]) == "TRUE")
|
initialflip = alphanumericToBoolean(splitted[1]);
|
||||||
initialflip = true;
|
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "INITIALMIRROR") && (splitted.size() > 1))
|
else if (((toUpper(splitted[0]) == "initialrotate") || (toUpper(splitted[0]) == "INITIALROTATE")) && (splitted.size() > 1)) {
|
||||||
{
|
if (isStringNumeric(splitted[1])) {
|
||||||
if (toUpper(splitted[1]) == "TRUE")
|
this->initialrotate = std::stod(splitted[1]);
|
||||||
initialmirror = true;
|
|
||||||
}
|
}
|
||||||
if (((toUpper(splitted[0]) == "INITALROTATE") || (toUpper(splitted[0]) == "INITIALROTATE")) && (splitted.size() > 1))
|
|
||||||
{
|
|
||||||
this->initalrotate = std::stod(splitted[1]);
|
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "SEARCHFIELDX") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "SEARCHFIELDX") && (splitted.size() > 1)) {
|
||||||
{
|
if (isStringNumeric(splitted[1])) {
|
||||||
suchex = std::stod(splitted[1]);
|
suchex = std::stod(splitted[1]);
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "SEARCHFIELDY") && (splitted.size() > 1))
|
}
|
||||||
{
|
else if ((toUpper(splitted[0]) == "SEARCHFIELDY") && (splitted.size() > 1)) {
|
||||||
|
if (isStringNumeric(splitted[1])) {
|
||||||
suchey = std::stod(splitted[1]);
|
suchey = std::stod(splitted[1]);
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "ANTIALIASING") && (splitted.size() > 1))
|
|
||||||
{
|
|
||||||
if (toUpper(splitted[1]) == "TRUE")
|
|
||||||
use_antialiasing = true;
|
|
||||||
}
|
}
|
||||||
if ((splitted.size() == 3) && (anz_ref < 2))
|
else if ((toUpper(splitted[0]) == "ANTIALIASING") && (splitted.size() > 1)) {
|
||||||
|
use_antialiasing = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
else if ((splitted.size() == 3) && (anz_ref < 2)) {
|
||||||
|
if ((isStringNumeric(splitted[1])) && (isStringNumeric(splitted[2])))
|
||||||
{
|
{
|
||||||
References[anz_ref].image_file = FormatFileName("/sdcard" + splitted[0]);
|
References[anz_ref].image_file = FormatFileName("/sdcard" + splitted[0]);
|
||||||
References[anz_ref].target_x = std::stod(splitted[1]);
|
References[anz_ref].target_x = std::stod(splitted[1]);
|
||||||
References[anz_ref].target_y = std::stod(splitted[2]);
|
References[anz_ref].target_y = std::stod(splitted[2]);
|
||||||
anz_ref++;
|
anz_ref++;
|
||||||
}
|
}
|
||||||
|
else
|
||||||
if ((toUpper(splitted[0]) == "SAVEALLFILES") && (splitted.size() > 1))
|
|
||||||
{
|
{
|
||||||
if (toUpper(splitted[1]) == "TRUE")
|
References[anz_ref].image_file = FormatFileName("/sdcard" + splitted[0]);
|
||||||
SaveAllFiles = true;
|
References[anz_ref].target_x = 10;
|
||||||
|
References[anz_ref].target_y = 10;
|
||||||
|
anz_ref++;
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "ALIGNMENTALGO") && (splitted.size() > 1))
|
}
|
||||||
{
|
|
||||||
#ifdef DEBUG_DETAIL_ON
|
else if ((toUpper(splitted[0]) == "SAVEALLFILES") && (splitted.size() > 1)) {
|
||||||
|
SaveAllFiles = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
else if ((toUpper(splitted[0]) == "ALIGNMENTALGO") && (splitted.size() > 1)) {
|
||||||
|
#ifdef DEBUG_DETAIL_ON
|
||||||
std::string zw2 = "Alignment mode selected: " + splitted[1];
|
std::string zw2 = "Alignment mode selected: " + splitted[1];
|
||||||
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, zw2);
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, zw2);
|
||||||
#endif
|
#endif
|
||||||
if (toUpper(splitted[1]) == "HIGHACCURACY")
|
if (toUpper(splitted[1]) == "HIGHACCURACY") {
|
||||||
alg_algo = 1;
|
alg_algo = 1;
|
||||||
if (toUpper(splitted[1]) == "FAST")
|
}
|
||||||
|
if (toUpper(splitted[1]) == "FAST") {
|
||||||
alg_algo = 2;
|
alg_algo = 2;
|
||||||
if (toUpper(splitted[1]) == "OFF") //no align algo if set to 3 = off => no draw ref //add disable aligment algo |01.2023
|
}
|
||||||
|
if (toUpper(splitted[1]) == "OFF") {
|
||||||
|
// no align algo if set to 3 = off => no draw ref //add disable aligment algo |01.2023
|
||||||
alg_algo = 3;
|
alg_algo = 3;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for (int i = 0; i < anz_ref; ++i)
|
for (int i = 0; i < anz_ref; ++i) {
|
||||||
{
|
|
||||||
References[i].search_x = suchex;
|
References[i].search_x = suchex;
|
||||||
References[i].search_y = suchey;
|
References[i].search_y = suchey;
|
||||||
References[i].fastalg_SAD_criteria = SAD_criteria;
|
References[i].fastalg_SAD_criteria = SAD_criteria;
|
||||||
References[i].alignment_algo = alg_algo;
|
References[i].alignment_algo = alg_algo;
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
std::string zw2 = "Alignment mode written: " + std::to_string(alg_algo);
|
std::string zw2 = "Alignment mode written: " + std::to_string(alg_algo);
|
||||||
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, zw2);
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, zw2);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
//no align algo if set to 3 = off => no draw ref //add disable aligment algo |01.2023
|
// no align algo if set to 3 = off => no draw ref //add disable aligment algo |01.2023
|
||||||
if(References[0].alignment_algo != 3){
|
if (References[0].alignment_algo != 3) {
|
||||||
LoadReferenceAlignmentValues();
|
return LoadReferenceAlignmentValues();
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
string ClassFlowAlignment::getHTMLSingleStep(string host)
|
string ClassFlowAlignment::getHTMLSingleStep(string host)
|
||||||
{
|
{
|
||||||
string result;
|
string result;
|
||||||
@@ -167,49 +167,46 @@ string ClassFlowAlignment::getHTMLSingleStep(string host)
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool ClassFlowAlignment::doFlow(string time)
|
bool ClassFlowAlignment::doFlow(string time)
|
||||||
{
|
{
|
||||||
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
||||||
if (!AlgROI) // AlgROI needs to be allocated before ImageTMP to avoid heap fragmentation
|
// AlgROI needs to be allocated before ImageTMP to avoid heap fragmentation
|
||||||
{
|
if (!AlgROI) {
|
||||||
AlgROI = (ImageData*)heap_caps_realloc(AlgROI, sizeof(ImageData), MALLOC_CAP_8BIT | MALLOC_CAP_SPIRAM);
|
AlgROI = (ImageData *)heap_caps_realloc(AlgROI, sizeof(ImageData), MALLOC_CAP_8BIT | MALLOC_CAP_SPIRAM);
|
||||||
if (!AlgROI)
|
|
||||||
{
|
if (!AlgROI) {
|
||||||
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Can't allocate AlgROI");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Can't allocate AlgROI");
|
||||||
LogFile.WriteHeapInfo("ClassFlowAlignment-doFlow");
|
LogFile.WriteHeapInfo("ClassFlowAlignment-doFlow");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (AlgROI)
|
if (AlgROI) {
|
||||||
{
|
ImageBasis->writeToMemoryAsJPG((ImageData *)AlgROI, 90);
|
||||||
ImageBasis->writeToMemoryAsJPG((ImageData*)AlgROI, 90);
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (!ImageTMP)
|
if (!ImageTMP) {
|
||||||
{
|
ImageTMP = new CImageBasis("tmpImage", ImageBasis); // Make sure the name does not get change, it is relevant for the PSRAM allocation!
|
||||||
ImageTMP = new CImageBasis(ImageBasis);
|
|
||||||
if (!ImageTMP)
|
if (!ImageTMP) {
|
||||||
{
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Can't allocate tmpImage -> Exec this round aborted!");
|
||||||
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Can't allocate ImageTMP -> Exec this round aborted!");
|
|
||||||
LogFile.WriteHeapInfo("ClassFlowAlignment-doFlow");
|
LogFile.WriteHeapInfo("ClassFlowAlignment-doFlow");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
delete AlignAndCutImage;
|
delete AlignAndCutImage;
|
||||||
AlignAndCutImage = new CAlignAndCutImage(ImageBasis, ImageTMP);
|
AlignAndCutImage = new CAlignAndCutImage("AlignAndCutImage", ImageBasis, ImageTMP);
|
||||||
if (!AlignAndCutImage)
|
|
||||||
{
|
if (!AlignAndCutImage) {
|
||||||
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Can't allocate AlignAndCutImage -> Exec this round aborted!");
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Can't allocate AlignAndCutImage -> Exec this round aborted!");
|
||||||
LogFile.WriteHeapInfo("ClassFlowAlignment-doFlow");
|
LogFile.WriteHeapInfo("ClassFlowAlignment-doFlow");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
CRotateImage rt(AlignAndCutImage, ImageTMP, initialflip);
|
CRotateImage rt("rawImage", AlignAndCutImage, ImageTMP, initialflip);
|
||||||
if (initialflip)
|
|
||||||
{
|
if (initialflip) {
|
||||||
int _zw = ImageBasis->height;
|
int _zw = ImageBasis->height;
|
||||||
ImageBasis->height = ImageBasis->width;
|
ImageBasis->height = ImageBasis->width;
|
||||||
ImageBasis->width = _zw;
|
ImageBasis->width = _zw;
|
||||||
@@ -219,50 +216,40 @@ bool ClassFlowAlignment::doFlow(string time)
|
|||||||
ImageTMP->height = _zw;
|
ImageTMP->height = _zw;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (initialmirror)
|
if ((initialrotate != 0) || initialflip) {
|
||||||
{
|
if (use_antialiasing) {
|
||||||
ESP_LOGD(TAG, "do mirror");
|
rt.RotateAntiAliasing(initialrotate);
|
||||||
rt.Mirror();
|
}
|
||||||
|
else {
|
||||||
if (SaveAllFiles)
|
rt.Rotate(initialrotate);
|
||||||
AlignAndCutImage->SaveToFile(FormatFileName("/sdcard/img_tmp/mirror.jpg"));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((initalrotate != 0) || initialflip)
|
if (SaveAllFiles) {
|
||||||
{
|
|
||||||
if (use_antialiasing)
|
|
||||||
rt.RotateAntiAliasing(initalrotate);
|
|
||||||
else
|
|
||||||
rt.Rotate(initalrotate);
|
|
||||||
|
|
||||||
if (SaveAllFiles)
|
|
||||||
AlignAndCutImage->SaveToFile(FormatFileName("/sdcard/img_tmp/rot.jpg"));
|
AlignAndCutImage->SaveToFile(FormatFileName("/sdcard/img_tmp/rot.jpg"));
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// no align algo if set to 3 = off //add disable aligment algo |01.2023
|
||||||
//no align algo if set to 3 = off //add disable aligment algo |01.2023
|
if (References[0].alignment_algo != 3) {
|
||||||
if(References[0].alignment_algo != 3){
|
if (!AlignAndCutImage->Align(&References[0], &References[1])) {
|
||||||
if (!AlignAndCutImage->Align(&References[0], &References[1]))
|
|
||||||
{
|
|
||||||
SaveReferenceAlignmentValues();
|
SaveReferenceAlignmentValues();
|
||||||
}
|
}
|
||||||
}// no align
|
} // no align
|
||||||
|
|
||||||
|
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
||||||
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
|
||||||
if (AlgROI) {
|
if (AlgROI) {
|
||||||
//no align algo if set to 3 = off => no draw ref //add disable aligment algo |01.2023
|
// no align algo if set to 3 = off => no draw ref //add disable aligment algo |01.2023
|
||||||
if(References[0].alignment_algo != 3){
|
if (References[0].alignment_algo != 3) {
|
||||||
DrawRef(ImageTMP);
|
DrawRef(ImageTMP);
|
||||||
}
|
}
|
||||||
tfliteflow.DigitalDrawROI(ImageTMP);
|
|
||||||
tfliteflow.AnalogDrawROI(ImageTMP);
|
|
||||||
ImageTMP->writeToMemoryAsJPG((ImageData*)AlgROI, 90);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (SaveAllFiles)
|
flowctrl.DigitDrawROI(ImageTMP);
|
||||||
{
|
flowctrl.AnalogDrawROI(ImageTMP);
|
||||||
|
ImageTMP->writeToMemoryAsJPG((ImageData *)AlgROI, 90);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (SaveAllFiles) {
|
||||||
AlignAndCutImage->SaveToFile(FormatFileName("/sdcard/img_tmp/alg.jpg"));
|
AlignAndCutImage->SaveToFile(FormatFileName("/sdcard/img_tmp/alg.jpg"));
|
||||||
ImageTMP->SaveToFile(FormatFileName("/sdcard/img_tmp/alg_roi.jpg"));
|
ImageTMP->SaveToFile(FormatFileName("/sdcard/img_tmp/alg_roi.jpg"));
|
||||||
}
|
}
|
||||||
@@ -271,26 +258,24 @@ bool ClassFlowAlignment::doFlow(string time)
|
|||||||
delete ImageTMP;
|
delete ImageTMP;
|
||||||
ImageTMP = NULL;
|
ImageTMP = NULL;
|
||||||
|
|
||||||
//no align algo if set to 3 = off => no draw ref //add disable aligment algo |01.2023
|
// no align algo if set to 3 = off => no draw ref //add disable aligment algo |01.2023
|
||||||
if(References[0].alignment_algo != 3){
|
if (References[0].alignment_algo != 3) {
|
||||||
LoadReferenceAlignmentValues();
|
return LoadReferenceAlignmentValues();
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void ClassFlowAlignment::SaveReferenceAlignmentValues()
|
void ClassFlowAlignment::SaveReferenceAlignmentValues()
|
||||||
{
|
{
|
||||||
FILE* pFile;
|
FILE *pFile;
|
||||||
std::string zwtime, zwvalue;
|
std::string zwtime, zwvalue;
|
||||||
|
|
||||||
pFile = fopen(FileStoreRefAlignment.c_str(), "w");
|
pFile = fopen(FileStoreRefAlignment.c_str(), "w");
|
||||||
|
|
||||||
if (strlen(zwtime.c_str()) == 0)
|
if (strlen(zwtime.c_str()) == 0) {
|
||||||
{
|
|
||||||
time_t rawtime;
|
time_t rawtime;
|
||||||
struct tm* timeinfo;
|
struct tm *timeinfo;
|
||||||
char buffer[80];
|
char buffer[80];
|
||||||
|
|
||||||
time(&rawtime);
|
time(&rawtime);
|
||||||
@@ -304,40 +289,40 @@ void ClassFlowAlignment::SaveReferenceAlignmentValues()
|
|||||||
fputs("\n", pFile);
|
fputs("\n", pFile);
|
||||||
|
|
||||||
zwvalue = std::to_string(References[0].fastalg_x) + "\t" + std::to_string(References[0].fastalg_y);
|
zwvalue = std::to_string(References[0].fastalg_x) + "\t" + std::to_string(References[0].fastalg_y);
|
||||||
zwvalue = zwvalue + "\t" +std::to_string(References[0].fastalg_SAD)+ "\t" +std::to_string(References[0].fastalg_min);
|
zwvalue = zwvalue + "\t" + std::to_string(References[0].fastalg_SAD) + "\t" + std::to_string(References[0].fastalg_min);
|
||||||
zwvalue = zwvalue + "\t" +std::to_string(References[0].fastalg_max)+ "\t" +std::to_string(References[0].fastalg_avg);
|
zwvalue = zwvalue + "\t" + std::to_string(References[0].fastalg_max) + "\t" + std::to_string(References[0].fastalg_avg);
|
||||||
fputs(zwvalue.c_str(), pFile);
|
fputs(zwvalue.c_str(), pFile);
|
||||||
fputs("\n", pFile);
|
fputs("\n", pFile);
|
||||||
|
|
||||||
zwvalue = std::to_string(References[1].fastalg_x) + "\t" + std::to_string(References[1].fastalg_y);
|
zwvalue = std::to_string(References[1].fastalg_x) + "\t" + std::to_string(References[1].fastalg_y);
|
||||||
zwvalue = zwvalue + "\t" +std::to_string(References[1].fastalg_SAD)+ "\t" +std::to_string(References[1].fastalg_min);
|
zwvalue = zwvalue + "\t" + std::to_string(References[1].fastalg_SAD) + "\t" + std::to_string(References[1].fastalg_min);
|
||||||
zwvalue = zwvalue + "\t" +std::to_string(References[1].fastalg_max)+ "\t" +std::to_string(References[1].fastalg_avg);
|
zwvalue = zwvalue + "\t" + std::to_string(References[1].fastalg_max) + "\t" + std::to_string(References[1].fastalg_avg);
|
||||||
fputs(zwvalue.c_str(), pFile);
|
fputs(zwvalue.c_str(), pFile);
|
||||||
fputs("\n", pFile);
|
fputs("\n", pFile);
|
||||||
|
|
||||||
fclose(pFile);
|
fclose(pFile);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool ClassFlowAlignment::LoadReferenceAlignmentValues(void)
|
bool ClassFlowAlignment::LoadReferenceAlignmentValues(void)
|
||||||
{
|
{
|
||||||
FILE* pFile;
|
FILE *pFile;
|
||||||
char zw[1024];
|
char zw[1024];
|
||||||
string zwvalue;
|
string zwvalue;
|
||||||
std::vector<string> splitted;
|
std::vector<string> splitted;
|
||||||
|
|
||||||
|
|
||||||
pFile = fopen(FileStoreRefAlignment.c_str(), "r");
|
pFile = fopen(FileStoreRefAlignment.c_str(), "r");
|
||||||
if (pFile == NULL)
|
|
||||||
|
if (pFile == NULL) {
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
fgets(zw, 1024, pFile);
|
fgets(zw, 1024, pFile);
|
||||||
ESP_LOGD(TAG, "%s", zw);
|
ESP_LOGD(TAG, "%s", zw);
|
||||||
|
|
||||||
fgets(zw, 1024, pFile);
|
fgets(zw, 1024, pFile);
|
||||||
splitted = ZerlegeZeile(std::string(zw), " \t");
|
splitted = ZerlegeZeile(std::string(zw), " \t");
|
||||||
if (splitted.size() < 6)
|
|
||||||
{
|
if (splitted.size() < 6) {
|
||||||
fclose(pFile);
|
fclose(pFile);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -351,8 +336,8 @@ bool ClassFlowAlignment::LoadReferenceAlignmentValues(void)
|
|||||||
|
|
||||||
fgets(zw, 1024, pFile);
|
fgets(zw, 1024, pFile);
|
||||||
splitted = ZerlegeZeile(std::string(zw));
|
splitted = ZerlegeZeile(std::string(zw));
|
||||||
if (splitted.size() < 6)
|
|
||||||
{
|
if (splitted.size() < 6) {
|
||||||
fclose(pFile);
|
fclose(pFile);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -366,7 +351,6 @@ bool ClassFlowAlignment::LoadReferenceAlignmentValues(void)
|
|||||||
|
|
||||||
fclose(pFile);
|
fclose(pFile);
|
||||||
|
|
||||||
|
|
||||||
/*#ifdef DEBUG_DETAIL_ON
|
/*#ifdef DEBUG_DETAIL_ON
|
||||||
std::string _zw = "\tLoadReferences[0]\tx,y:\t" + std::to_string(References[0].fastalg_x) + "\t" + std::to_string(References[0].fastalg_x);
|
std::string _zw = "\tLoadReferences[0]\tx,y:\t" + std::to_string(References[0].fastalg_x) + "\t" + std::to_string(References[0].fastalg_x);
|
||||||
_zw = _zw + "\tSAD, min, max, avg:\t" + std::to_string(References[0].fastalg_SAD) + "\t" + std::to_string(References[0].fastalg_min);
|
_zw = _zw + "\tSAD, min, max, avg:\t" + std::to_string(References[0].fastalg_SAD) + "\t" + std::to_string(References[0].fastalg_min);
|
||||||
@@ -381,11 +365,9 @@ bool ClassFlowAlignment::LoadReferenceAlignmentValues(void)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void ClassFlowAlignment::DrawRef(CImageBasis *_zw)
|
void ClassFlowAlignment::DrawRef(CImageBasis *_zw)
|
||||||
{
|
{
|
||||||
if (_zw->ImageOkay())
|
if (_zw->ImageOkay()) {
|
||||||
{
|
|
||||||
_zw->drawRect(References[0].target_x, References[0].target_y, References[0].width, References[0].height, 255, 0, 0, 2);
|
_zw->drawRect(References[0].target_x, References[0].target_y, References[0].width, References[0].height, 255, 0, 0, 2);
|
||||||
_zw->drawRect(References[1].target_x, References[1].target_y, References[1].width, References[1].height, 255, 0, 0, 2);
|
_zw->drawRect(References[1].target_x, References[1].target_y, References[1].width, References[1].height, 255, 0, 0, 2);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,12 +12,10 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
class ClassFlowAlignment :
|
class ClassFlowAlignment : public ClassFlow
|
||||||
public ClassFlow
|
|
||||||
{
|
{
|
||||||
protected:
|
protected:
|
||||||
float initalrotate;
|
float initialrotate;
|
||||||
bool initialmirror;
|
|
||||||
bool initialflip;
|
bool initialflip;
|
||||||
bool use_antialiasing;
|
bool use_antialiasing;
|
||||||
RefInfo References[2];
|
RefInfo References[2];
|
||||||
@@ -34,21 +32,20 @@ protected:
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
CImageBasis *ImageBasis, *ImageTMP;
|
CImageBasis *ImageBasis, *ImageTMP;
|
||||||
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
||||||
ImageData *AlgROI;
|
ImageData *AlgROI;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
ClassFlowAlignment(std::vector<ClassFlow*>* lfc);
|
ClassFlowAlignment(std::vector<ClassFlow *> *lfc);
|
||||||
|
|
||||||
CAlignAndCutImage* GetAlignAndCutImage(){return AlignAndCutImage;};
|
CAlignAndCutImage *GetAlignAndCutImage() { return AlignAndCutImage; };
|
||||||
|
|
||||||
void DrawRef(CImageBasis *_zw);
|
void DrawRef(CImageBasis *_zw);
|
||||||
|
|
||||||
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
bool ReadParameter(FILE *pfile, string &aktparamgraph);
|
||||||
bool doFlow(string time);
|
bool doFlow(string time);
|
||||||
string getHTMLSingleStep(string host);
|
string getHTMLSingleStep(string host);
|
||||||
string name(){return "ClassFlowAlignment";};
|
string name() { return "ClassFlowAlignment"; };
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#endif // CLASSFLOWALIGNMENT_H
|
||||||
#endif //CLASSFLOWALIGNMENT_H
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -11,10 +11,10 @@ enum t_CNNType {
|
|||||||
AutoDetect,
|
AutoDetect,
|
||||||
Analogue,
|
Analogue,
|
||||||
Analogue100,
|
Analogue100,
|
||||||
Digital,
|
Digit,
|
||||||
DigitalHyprid10,
|
DigitHyprid10,
|
||||||
DoubleHyprid10,
|
DoubleHyprid10,
|
||||||
Digital100,
|
Digit100,
|
||||||
None
|
None
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -26,15 +26,6 @@ protected:
|
|||||||
std::vector<general*> GENERAL;
|
std::vector<general*> GENERAL;
|
||||||
float CNNGoodThreshold;
|
float CNNGoodThreshold;
|
||||||
|
|
||||||
//moved to define.h
|
|
||||||
//float Analog_error = 3.0;
|
|
||||||
//float AnalogToDigtalFehler = 0.8;
|
|
||||||
//float Digital_Uncertainty = 0.2;
|
|
||||||
//int DigitalBand = 3;
|
|
||||||
//float Digital_Transition_Range_Predecessor = 2;
|
|
||||||
//float Digital_Transition_Area_Predecessor = 0.7; // 9.3 - 0.7
|
|
||||||
//float Digital_Transition_Area_Forward = 9.7; // Pre-run zero crossing only happens from approx. 9.7 onwards
|
|
||||||
|
|
||||||
string cnnmodelfile;
|
string cnnmodelfile;
|
||||||
int modelxsize, modelysize, modelchannel;
|
int modelxsize, modelysize, modelchannel;
|
||||||
bool isLogImageSelect;
|
bool isLogImageSelect;
|
||||||
@@ -44,8 +35,8 @@ protected:
|
|||||||
bool SaveAllFiles;
|
bool SaveAllFiles;
|
||||||
|
|
||||||
int PointerEvalAnalogNew(float zahl, int numeral_preceder);
|
int PointerEvalAnalogNew(float zahl, int numeral_preceder);
|
||||||
int PointerEvalAnalogToDigitNew(float zahl, float numeral_preceder, int eval_predecessors, float analogDigitalTransitionStart);
|
int PointerEvalAnalogToDigitNew(float zahl, float numeral_preceder, int eval_predecessors, float AnalogToDigitTransitionStart);
|
||||||
int PointerEvalHybridNew(float zahl, float number_of_predecessors, int eval_predecessors, bool Analog_Predecessors = false, float analogDigitalTransitionStart=9.2);
|
int PointerEvalHybridNew(float zahl, float number_of_predecessors, int eval_predecessors, bool Analog_Predecessors = false, float AnalogToDigitTransitionStart=9.2);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -61,7 +52,7 @@ public:
|
|||||||
bool doFlow(string time);
|
bool doFlow(string time);
|
||||||
|
|
||||||
string getHTMLSingleStep(string host);
|
string getHTMLSingleStep(string host);
|
||||||
string getReadout(int _analog, bool _extendedResolution = false, int prev = -1, float _before_narrow_Analog = -1, float analogDigitalTransitionStart=9.2);
|
string getReadout(int _analog, bool _extendedResolution = false, int prev = -1, float _before_narrow_Analog = -1, float AnalogToDigitTransitionStart=9.2);
|
||||||
|
|
||||||
string getReadoutRawString(int _analog);
|
string getReadoutRawString(int _analog);
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -17,8 +17,10 @@
|
|||||||
#include "ClassFlowInfluxDB.h"
|
#include "ClassFlowInfluxDB.h"
|
||||||
#include "ClassFlowInfluxDBv2.h"
|
#include "ClassFlowInfluxDBv2.h"
|
||||||
#endif //ENABLE_INFLUXDB
|
#endif //ENABLE_INFLUXDB
|
||||||
|
#ifdef ENABLE_WEBHOOK
|
||||||
|
#include "ClassFlowWebhook.h"
|
||||||
|
#endif //ENABLE_WEBHOOK
|
||||||
#include "ClassFlowCNNGeneral.h"
|
#include "ClassFlowCNNGeneral.h"
|
||||||
#include "ClassFlowWriteList.h"
|
|
||||||
|
|
||||||
class ClassFlowControll :
|
class ClassFlowControll :
|
||||||
public ClassFlow
|
public ClassFlow
|
||||||
@@ -35,32 +37,31 @@ protected:
|
|||||||
|
|
||||||
bool AutoStart;
|
bool AutoStart;
|
||||||
float AutoInterval;
|
float AutoInterval;
|
||||||
bool SetupModeActive;
|
|
||||||
void SetInitialParameter(void);
|
void SetInitialParameter(void);
|
||||||
|
std::string aktstatusWithTime;
|
||||||
std::string aktstatus;
|
std::string aktstatus;
|
||||||
int aktRunNr;
|
int aktRunNr;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
bool SetupModeActive;
|
||||||
|
|
||||||
void InitFlow(std::string config);
|
void InitFlow(std::string config);
|
||||||
bool doFlow(string time);
|
bool doFlow(string time);
|
||||||
void doFlowTakeImageOnly(string time);
|
void doFlowTakeImageOnly(string time);
|
||||||
bool getStatusSetupModus(){return SetupModeActive;};
|
bool getStatusSetupModus(){return SetupModeActive;};
|
||||||
string getReadout(bool _rawvalue, bool _noerror);
|
string getReadout(bool _rawvalue, bool _noerror, int _number);
|
||||||
string getReadoutAll(int _type);
|
string getReadoutAll(int _type);
|
||||||
string UpdatePrevalue(std::string _newvalue, std::string _numbers, bool _extern);
|
bool UpdatePrevalue(std::string _newvalue, std::string _numbers, bool _extern);
|
||||||
string GetPrevalue(std::string _number = "");
|
string GetPrevalue(std::string _number = "");
|
||||||
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
||||||
string getJSON();
|
string getJSON();
|
||||||
|
const std::vector<NumberPost*> &getNumbers();
|
||||||
string getNumbersName();
|
string getNumbersName();
|
||||||
|
|
||||||
string TranslateAktstatus(std::string _input);
|
string TranslateAktstatus(std::string _input);
|
||||||
|
|
||||||
#ifdef ENABLE_MQTT
|
|
||||||
string GetMQTTMainTopic();
|
|
||||||
#endif //ENABLE_MQTT
|
|
||||||
|
|
||||||
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
||||||
void DigitalDrawROI(CImageBasis *_zw);
|
void DigitDrawROI(CImageBasis *_zw);
|
||||||
void AnalogDrawROI(CImageBasis *_zw);
|
void AnalogDrawROI(CImageBasis *_zw);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -69,15 +70,17 @@ public:
|
|||||||
|
|
||||||
std::string doSingleStep(std::string _stepname, std::string _host);
|
std::string doSingleStep(std::string _stepname, std::string _host);
|
||||||
|
|
||||||
bool isAutoStart(long &_interval);
|
bool getIsAutoStart();
|
||||||
|
void setAutoStartInterval(long &_interval);
|
||||||
|
|
||||||
|
std::string* getActStatusWithTime();
|
||||||
std::string* getActStatus();
|
std::string* getActStatus();
|
||||||
void setActStatus(std::string _aktstatus);
|
void setActStatus(std::string _aktstatus);
|
||||||
|
|
||||||
std::vector<HTMLInfo*> GetAllDigital();
|
std::vector<HTMLInfo*> GetAllDigit();
|
||||||
std::vector<HTMLInfo*> GetAllAnalog();
|
std::vector<HTMLInfo*> GetAllAnalog();
|
||||||
|
|
||||||
t_CNNType GetTypeDigital();
|
t_CNNType GetTypeDigit();
|
||||||
t_CNNType GetTypeAnalog();
|
t_CNNType GetTypeAnalog();
|
||||||
|
|
||||||
#ifdef ENABLE_MQTT
|
#ifdef ENABLE_MQTT
|
||||||
|
|||||||
@@ -5,6 +5,10 @@
|
|||||||
|
|
||||||
#include "ClassFlowImage.h"
|
#include "ClassFlowImage.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Properties of one ROI
|
||||||
|
* FIXME: naming of members could use some refactoring to comply with common C++ coding style guidelines
|
||||||
|
*/
|
||||||
struct roi {
|
struct roi {
|
||||||
int posx, posy, deltax, deltay;
|
int posx, posy, deltax, deltay;
|
||||||
float result_float;
|
float result_float;
|
||||||
@@ -14,51 +18,68 @@ struct roi {
|
|||||||
CImageBasis *image, *image_org;
|
CImageBasis *image, *image_org;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* FIXME: Why is this additional layer needed?
|
||||||
|
*/
|
||||||
struct general {
|
struct general {
|
||||||
string name;
|
string name;
|
||||||
std::vector<roi*> ROI;
|
std::vector<roi*> ROI;
|
||||||
};
|
};
|
||||||
|
|
||||||
enum t_RateType {
|
enum t_RateType {
|
||||||
AbsoluteChange,
|
AbsoluteChange, // ignores the time difference; only the value difference is used comparison with NumberPost.maxRate
|
||||||
RateChange
|
RateChange // time difference is considered and a normalized rate is used for comparison with NumberPost.maxRate
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Holds all properties and settings of a sequence. A sequence is a set of digit and/or analog ROIs that are combined to
|
||||||
|
* provide one meter reading (value).
|
||||||
|
* FIXME: can be renamed to `Sequence`
|
||||||
|
*/
|
||||||
struct NumberPost {
|
struct NumberPost {
|
||||||
float MaxRateValue;
|
float MaxRateValue; // maxRate; upper bound for the difference between two consecutive readings; affected by maxRateType;
|
||||||
bool useMaxRateValue;
|
bool useMaxRateValue; // consistencyChecksEnabled; enables consistency checks; uses maxRate and maxRateType
|
||||||
t_RateType RateType;
|
t_RateType MaxRateType; // maxRateType; affects how the value of maxRate is used for comparing the current and previous value
|
||||||
bool ErrorMessage;
|
bool ErrorMessage; // FIXME: not used; can be removed
|
||||||
bool PreValueOkay;
|
int ChangeRateThreshold; // threshold parameter for negative rate detection
|
||||||
bool AllowNegativeRates;
|
bool PreValueOkay; // previousValueValid; indicates that the reading of the previous round has no errors
|
||||||
bool checkDigitIncreaseConsistency;
|
bool AllowNegativeRates; // allowNegativeRate; defines if the consistency checks allow negative rates between consecutive meter readings.
|
||||||
time_t lastvalue;
|
bool IgnoreLeadingNaN;
|
||||||
string timeStamp;
|
bool checkDigitIncreaseConsistency; // extendedConsistencyCheck; performs an additional consistency check to avoid wrong readings
|
||||||
double FlowRateAct; // m3 / min
|
time_t timeStampLastValue; // Timestamp for the last read value; is used for the log
|
||||||
double PreValue; // last value that was read out well
|
time_t timeStampLastPreValue; // Timestamp for the last PreValue set; is used for useMaxRateValue
|
||||||
double Value; // last value read out, incl. corrections
|
time_t timeStampTimeUTC; // FIXME: not used; can be removed.
|
||||||
string ReturnRateValue; // return value rate
|
string timeStamp; // localTimeStr; timestamp of last valid reading formatted as local time
|
||||||
string ReturnChangeAbsolute; // return value rate
|
double FlowRateAct; // currentRate; ΔValue/min; since usage is not limited to water meters, the physical unit is not known.
|
||||||
string ReturnRawValue; // Raw value (with N & leading 0)
|
double PreValue; // lastValidValue; most recent value that could be read w/o any errors
|
||||||
string ReturnValue; // corrected return value, if necessary with error message
|
double Value; // value; most recent readout; may include corrections
|
||||||
string ReturnPreValue; // corrected return value without error message
|
string ReturnRateValue; // currentRateStr; current normalized rate; ΔValue/min
|
||||||
string ErrorMessageText; // Error message for consistency check
|
string ReturnChangeAbsolute; // currentChangeStr; absolute difference between current and previous measurement
|
||||||
int AnzahlAnalog;
|
string ReturnRawValue; // rawValueStr; Raw value (with N & leading 0)
|
||||||
int AnzahlDigital;
|
string ReturnValue; // valueStr; corrected return value, if necessary with error message
|
||||||
int DecimalShift;
|
string ReturnPreValue; // lastValidValueStr; corrected return value without error message
|
||||||
int DecimalShiftInitial;
|
string ErrorMessageText; // errorMessage; Error message for consistency checks
|
||||||
float AnalogDigitalTransitionStart; // When is the digit > x.1, i.e. when does it start to tilt?
|
int AnzahlAnalog; // numAnalogRoi; number of analog ROIs used in this sequence
|
||||||
int Nachkomma;
|
int AnzahlDigit; // numDigitRoi; number of digit ROIs used in this sequence
|
||||||
string Fieldname; // Fieldname in InfluxDB2
|
int DecimalShift; // decimalShift; each increment shifts the decimal separator by one digit; value=value*10^decimalShift; pos. value shifts to the right
|
||||||
|
int DecimalShiftInitial; // decimalShiftInitial; same as decimalShift but is a const to reset decimalShift after calculations
|
||||||
|
float AnalogToDigitTransitionStart; // AnalogToDigitTransitionStartValue; FIXME: need a better description; When is the digit > x.1, i.e. when does it start to tilt?
|
||||||
|
int Nachkomma; // decimalPlaces; usually defined by the number of analog ROIs; affected by DecimalShift
|
||||||
|
|
||||||
bool isExtendedResolution;
|
string DomoticzIdx; // Domoticz counter Idx
|
||||||
|
|
||||||
general *digit_roi;
|
string FieldV1; // influxdbFieldName_v1; Name of the Field in InfluxDBv1
|
||||||
general *analog_roi;
|
string MeasurementV1; // influxdbMeasurementName_v1; Name of the Measurement in InfluxDBv1
|
||||||
|
|
||||||
string name;
|
string FieldV2; // influxdbFieldName_v2; Name of the Field in InfluxDBv2
|
||||||
|
string MeasurementV2; // influxdbMeasurementName_v2; Name of the Measurement in InfluxDBv2
|
||||||
|
|
||||||
|
bool isExtendedResolution; // extendResolution; Adds the decimal place of the least significant analog ROI to the value
|
||||||
|
|
||||||
|
general *digit_roi; // digitRoi; set of digit ROIs for the sequence
|
||||||
|
general *analog_roi; // analogRoi; set of analog ROIs for the sequence
|
||||||
|
|
||||||
|
string name; // name; Designation for the sequence
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -17,7 +17,8 @@ extern "C" {
|
|||||||
#include "esp_log.h"
|
#include "esp_log.h"
|
||||||
#include "../../include/defines.h"
|
#include "../../include/defines.h"
|
||||||
|
|
||||||
static const char* TAG = "IMG";
|
static const char* TAG = "FLOWIMAGE";
|
||||||
|
|
||||||
|
|
||||||
ClassFlowImage::ClassFlowImage(const char* logTag)
|
ClassFlowImage::ClassFlowImage(const char* logTag)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -15,13 +15,12 @@
|
|||||||
|
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
|
|
||||||
static const char* TAG = "class_flow_influxDb";
|
static const char* TAG = "INFLUXDB";
|
||||||
|
|
||||||
void ClassFlowInfluxDB::SetInitialParameter(void)
|
void ClassFlowInfluxDB::SetInitialParameter(void)
|
||||||
{
|
{
|
||||||
uri = "";
|
uri = "";
|
||||||
database = "";
|
database = "";
|
||||||
measurement = "";
|
|
||||||
|
|
||||||
OldValue = "";
|
OldValue = "";
|
||||||
flowpostprocessing = NULL;
|
flowpostprocessing = NULL;
|
||||||
@@ -86,33 +85,44 @@ bool ClassFlowInfluxDB::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|||||||
{
|
{
|
||||||
ESP_LOGD(TAG, "while loop reading line: %s", aktparamgraph.c_str());
|
ESP_LOGD(TAG, "while loop reading line: %s", aktparamgraph.c_str());
|
||||||
splitted = ZerlegeZeile(aktparamgraph);
|
splitted = ZerlegeZeile(aktparamgraph);
|
||||||
if ((toUpper(splitted[0]) == "USER") && (splitted.size() > 1))
|
std::string _param = GetParameterName(splitted[0]);
|
||||||
|
|
||||||
|
if ((toUpper(_param) == "USER") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
this->user = splitted[1];
|
this->user = splitted[1];
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "PASSWORD") && (splitted.size() > 1))
|
if ((toUpper(_param) == "PASSWORD") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
this->password = splitted[1];
|
this->password = splitted[1];
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "URI") && (splitted.size() > 1))
|
if ((toUpper(_param) == "URI") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
this->uri = splitted[1];
|
this->uri = splitted[1];
|
||||||
}
|
}
|
||||||
if (((toUpper(splitted[0]) == "MEASUREMENT")) && (splitted.size() > 1))
|
if (((toUpper(_param) == "DATABASE")) && (splitted.size() > 1))
|
||||||
{
|
|
||||||
this->measurement = splitted[1];
|
|
||||||
}
|
|
||||||
if (((toUpper(splitted[0]) == "DATABASE")) && (splitted.size() > 1))
|
|
||||||
{
|
{
|
||||||
this->database = splitted[1];
|
this->database = splitted[1];
|
||||||
}
|
}
|
||||||
|
if (((toUpper(_param) == "MEASUREMENT")) && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
handleMeasurement(splitted[0], splitted[1]);
|
||||||
|
}
|
||||||
|
if (((toUpper(_param) == "FIELD")) && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
handleFieldname(splitted[0], splitted[1]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((uri.length() > 0) && (database.length() > 0) && (measurement.length() > 0))
|
if ((uri.length() > 0) && (database.length() > 0))
|
||||||
{
|
{
|
||||||
// ESP_LOGD(TAG, "Init InfluxDB with uri: %s, measurement: %s, user: %s, password: %s", uri.c_str(), measurement.c_str(), user.c_str(), password.c_str());
|
// ESP_LOGD(TAG, "Init InfluxDB with uri: %s, measurement: %s, user: %s, password: %s", uri.c_str(), measurement.c_str(), user.c_str(), password.c_str());
|
||||||
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Init InfluxDB with uri: " + uri + ", measurement: " + measurement + ", user: " + user + ", password: " + password);
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Init InfluxDB with uri: " + uri + ", user: " + user + ", password: " + password);
|
||||||
InfluxDBInit(uri, database, measurement, user, password);
|
|
||||||
|
/////////////////////// NEW //////////////////////////
|
||||||
|
// InfluxDBInit(uri, database, user, password);
|
||||||
|
influxDB.InfluxDBInitV1(uri, database, user, password);
|
||||||
|
/////////////////////// NEW //////////////////////////
|
||||||
|
|
||||||
InfluxDBenable = true;
|
InfluxDBenable = true;
|
||||||
} else {
|
} else {
|
||||||
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "InfluxDB init skipped as we are missing some parameters");
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "InfluxDB init skipped as we are missing some parameters");
|
||||||
@@ -121,23 +131,18 @@ bool ClassFlowInfluxDB::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
string ClassFlowInfluxDB::GetInfluxDBMeasurement()
|
|
||||||
{
|
|
||||||
return measurement;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool ClassFlowInfluxDB::doFlow(string zwtime)
|
bool ClassFlowInfluxDB::doFlow(string zwtime)
|
||||||
{
|
{
|
||||||
if (!InfluxDBenable)
|
if (!InfluxDBenable)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
std::string result;
|
std::string result;
|
||||||
|
std::string measurement;
|
||||||
std::string resulterror = "";
|
std::string resulterror = "";
|
||||||
std::string resultraw = "";
|
std::string resultraw = "";
|
||||||
std::string resultrate = "";
|
std::string resultrate = "";
|
||||||
std::string resulttimestamp = "";
|
std::string resulttimestamp = "";
|
||||||
|
long int timeutc;
|
||||||
string zw = "";
|
string zw = "";
|
||||||
string namenumber = "";
|
string namenumber = "";
|
||||||
|
|
||||||
@@ -147,20 +152,34 @@ bool ClassFlowInfluxDB::doFlow(string zwtime)
|
|||||||
|
|
||||||
for (int i = 0; i < (*NUMBERS).size(); ++i)
|
for (int i = 0; i < (*NUMBERS).size(); ++i)
|
||||||
{
|
{
|
||||||
|
measurement = (*NUMBERS)[i]->MeasurementV1;
|
||||||
result = (*NUMBERS)[i]->ReturnValue;
|
result = (*NUMBERS)[i]->ReturnValue;
|
||||||
resultraw = (*NUMBERS)[i]->ReturnRawValue;
|
resultraw = (*NUMBERS)[i]->ReturnRawValue;
|
||||||
resulterror = (*NUMBERS)[i]->ErrorMessageText;
|
resulterror = (*NUMBERS)[i]->ErrorMessageText;
|
||||||
resultrate = (*NUMBERS)[i]->ReturnRateValue;
|
resultrate = (*NUMBERS)[i]->ReturnRateValue;
|
||||||
resulttimestamp = (*NUMBERS)[i]->timeStamp;
|
resulttimestamp = (*NUMBERS)[i]->timeStamp;
|
||||||
|
timeutc = (*NUMBERS)[i]->timeStampTimeUTC;
|
||||||
|
|
||||||
|
if ((*NUMBERS)[i]->FieldV1.length() > 0)
|
||||||
|
{
|
||||||
|
namenumber = (*NUMBERS)[i]->FieldV1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
namenumber = (*NUMBERS)[i]->name;
|
namenumber = (*NUMBERS)[i]->name;
|
||||||
if (namenumber == "default")
|
if (namenumber == "default")
|
||||||
namenumber = "value";
|
namenumber = "value";
|
||||||
else
|
else
|
||||||
namenumber = namenumber + "/value";
|
namenumber = namenumber + "/value";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result.length() > 0)
|
||||||
|
//////////////////////// NEW //////////////////////////
|
||||||
|
// InfluxDBPublish(measurement, namenumber, result, timeutc);
|
||||||
|
influxDB.InfluxDBPublish(measurement, namenumber, result, timeutc);
|
||||||
|
//////////////////////// NEW //////////////////////////
|
||||||
|
|
||||||
|
|
||||||
if (result.length() > 0 && resulttimestamp.length() > 0)
|
|
||||||
InfluxDBPublish(namenumber, result, resulttimestamp);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -169,4 +188,50 @@ bool ClassFlowInfluxDB::doFlow(string zwtime)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ClassFlowInfluxDB::handleMeasurement(string _decsep, string _value)
|
||||||
|
{
|
||||||
|
string _digit, _decpos;
|
||||||
|
int _pospunkt = _decsep.find_first_of(".");
|
||||||
|
// ESP_LOGD(TAG, "Name: %s, Pospunkt: %d", _decsep.c_str(), _pospunkt);
|
||||||
|
if (_pospunkt > -1)
|
||||||
|
_digit = _decsep.substr(0, _pospunkt);
|
||||||
|
else
|
||||||
|
_digit = "default";
|
||||||
|
for (int j = 0; j < flowpostprocessing->NUMBERS.size(); ++j)
|
||||||
|
{
|
||||||
|
if (_digit == "default") // Set to default first (if nothing else is set)
|
||||||
|
{
|
||||||
|
flowpostprocessing->NUMBERS[j]->MeasurementV1 = _value;
|
||||||
|
}
|
||||||
|
if (flowpostprocessing->NUMBERS[j]->name == _digit)
|
||||||
|
{
|
||||||
|
flowpostprocessing->NUMBERS[j]->MeasurementV1 = _value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void ClassFlowInfluxDB::handleFieldname(string _decsep, string _value)
|
||||||
|
{
|
||||||
|
string _digit, _decpos;
|
||||||
|
int _pospunkt = _decsep.find_first_of(".");
|
||||||
|
// ESP_LOGD(TAG, "Name: %s, Pospunkt: %d", _decsep.c_str(), _pospunkt);
|
||||||
|
if (_pospunkt > -1)
|
||||||
|
_digit = _decsep.substr(0, _pospunkt);
|
||||||
|
else
|
||||||
|
_digit = "default";
|
||||||
|
for (int j = 0; j < flowpostprocessing->NUMBERS.size(); ++j)
|
||||||
|
{
|
||||||
|
if (_digit == "default") // Set to default first (if nothing else is set)
|
||||||
|
{
|
||||||
|
flowpostprocessing->NUMBERS[j]->FieldV1 = _value;
|
||||||
|
}
|
||||||
|
if (flowpostprocessing->NUMBERS[j]->name == _digit)
|
||||||
|
{
|
||||||
|
flowpostprocessing->NUMBERS[j]->FieldV1 = _value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#endif //ENABLE_INFLUXDB
|
#endif //ENABLE_INFLUXDB
|
||||||
@@ -8,6 +8,7 @@
|
|||||||
#include "ClassFlow.h"
|
#include "ClassFlow.h"
|
||||||
|
|
||||||
#include "ClassFlowPostProcessing.h"
|
#include "ClassFlowPostProcessing.h"
|
||||||
|
#include "interface_influxdb.h"
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
@@ -21,14 +22,21 @@ protected:
|
|||||||
std::string user, password;
|
std::string user, password;
|
||||||
bool InfluxDBenable;
|
bool InfluxDBenable;
|
||||||
|
|
||||||
|
InfluxDB influxDB;
|
||||||
|
|
||||||
void SetInitialParameter(void);
|
void SetInitialParameter(void);
|
||||||
|
|
||||||
|
void handleFieldname(string _decsep, string _value);
|
||||||
|
void handleMeasurement(string _decsep, string _value);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
ClassFlowInfluxDB();
|
ClassFlowInfluxDB();
|
||||||
ClassFlowInfluxDB(std::vector<ClassFlow*>* lfc);
|
ClassFlowInfluxDB(std::vector<ClassFlow*>* lfc);
|
||||||
ClassFlowInfluxDB(std::vector<ClassFlow*>* lfc, ClassFlow *_prev);
|
ClassFlowInfluxDB(std::vector<ClassFlow*>* lfc, ClassFlow *_prev);
|
||||||
|
|
||||||
string GetInfluxDBMeasurement();
|
// string GetInfluxDBMeasurement();
|
||||||
|
|
||||||
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
||||||
bool doFlow(string time);
|
bool doFlow(string time);
|
||||||
|
|||||||
@@ -15,13 +15,12 @@
|
|||||||
|
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
|
|
||||||
static const char* TAG = "class_flow_influxDbv2";
|
static const char* TAG = "INFLUXDBV2";
|
||||||
|
|
||||||
void ClassFlowInfluxDBv2::SetInitialParameter(void)
|
void ClassFlowInfluxDBv2::SetInitialParameter(void)
|
||||||
{
|
{
|
||||||
uri = "";
|
uri = "";
|
||||||
database = "";
|
bucket = "";
|
||||||
measurement = "";
|
|
||||||
dborg = "";
|
dborg = "";
|
||||||
dbtoken = "";
|
dbtoken = "";
|
||||||
// dbfield = "";
|
// dbfield = "";
|
||||||
@@ -102,30 +101,36 @@ bool ClassFlowInfluxDBv2::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|||||||
{
|
{
|
||||||
this->uri = splitted[1];
|
this->uri = splitted[1];
|
||||||
}
|
}
|
||||||
if (((toUpper(_param) == "MEASUREMENT")) && (splitted.size() > 1))
|
if (((toUpper(_param) == "FIELD")) && (splitted.size() > 1))
|
||||||
{
|
|
||||||
this->measurement = splitted[1];
|
|
||||||
}
|
|
||||||
if (((toUpper(_param) == "FIELDNAME")) && (splitted.size() > 1))
|
|
||||||
{
|
{
|
||||||
handleFieldname(splitted[0], splitted[1]);
|
handleFieldname(splitted[0], splitted[1]);
|
||||||
}
|
}
|
||||||
if (((toUpper(splitted[0]) == "DATABASE")) && (splitted.size() > 1))
|
if (((toUpper(_param) == "MEASUREMENT")) && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
this->database = splitted[1];
|
handleMeasurement(splitted[0], splitted[1]);
|
||||||
|
}
|
||||||
|
if (((toUpper(splitted[0]) == "BUCKET")) && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
this->bucket = splitted[1];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
printf("uri: %s\n", uri.c_str());
|
printf("uri: %s\n", uri.c_str());
|
||||||
printf("measurement: %s\n", measurement.c_str());
|
|
||||||
printf("org: %s\n", dborg.c_str());
|
printf("org: %s\n", dborg.c_str());
|
||||||
printf("token: %s\n", dbtoken.c_str());
|
printf("token: %s\n", dbtoken.c_str());
|
||||||
|
|
||||||
if ((uri.length() > 0) && (database.length() > 0) && (measurement.length() > 0) && (dbtoken.length() > 0) && (dborg.length() > 0))
|
if ((uri.length() > 0) && (bucket.length() > 0) && (dbtoken.length() > 0) && (dborg.length() > 0))
|
||||||
{
|
{
|
||||||
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Init InfluxDB with uri: " + uri + ", measurement: " + measurement + ", org: " + dborg + ", token: *****");
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Init InfluxDB with uri: " + uri + ", org: " + dborg + ", token: *****");
|
||||||
// printf("vor V2 Init\n");
|
// printf("vor V2 Init\n");
|
||||||
InfluxDB_V2_Init(uri, database, measurement, dborg, dbtoken);
|
|
||||||
|
|
||||||
|
////////////////////////////////////////// NEW ////////////////////////////////////////////
|
||||||
|
// InfluxDB_V2_Init(uri, bucket, dborg, dbtoken);
|
||||||
|
// InfluxDB_V2_Init(uri, bucket, dborg, dbtoken);
|
||||||
|
influxdb.InfluxDBInitV2(uri, bucket, dborg, dbtoken);
|
||||||
|
////////////////////////////////////////// NEW ////////////////////////////////////////////
|
||||||
|
|
||||||
// printf("nach V2 Init\n");
|
// printf("nach V2 Init\n");
|
||||||
InfluxDBenable = true;
|
InfluxDBenable = true;
|
||||||
} else {
|
} else {
|
||||||
@@ -135,11 +140,12 @@ bool ClassFlowInfluxDBv2::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
string ClassFlowInfluxDBv2::GetInfluxDBMeasurement()
|
string ClassFlowInfluxDBv2::GetInfluxDBMeasurement()
|
||||||
{
|
{
|
||||||
return measurement;
|
return measurement;
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
void ClassFlowInfluxDBv2::handleFieldname(string _decsep, string _value)
|
void ClassFlowInfluxDBv2::handleFieldname(string _decsep, string _value)
|
||||||
{
|
{
|
||||||
@@ -154,15 +160,36 @@ void ClassFlowInfluxDBv2::handleFieldname(string _decsep, string _value)
|
|||||||
{
|
{
|
||||||
if (_digit == "default") // Set to default first (if nothing else is set)
|
if (_digit == "default") // Set to default first (if nothing else is set)
|
||||||
{
|
{
|
||||||
flowpostprocessing->NUMBERS[j]->Fieldname = _value;
|
flowpostprocessing->NUMBERS[j]->FieldV2 = _value;
|
||||||
}
|
}
|
||||||
if (flowpostprocessing->NUMBERS[j]->name == _digit)
|
if (flowpostprocessing->NUMBERS[j]->name == _digit)
|
||||||
{
|
{
|
||||||
flowpostprocessing->NUMBERS[j]->Fieldname = _value;
|
flowpostprocessing->NUMBERS[j]->FieldV2 = _value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ClassFlowInfluxDBv2::handleMeasurement(string _decsep, string _value)
|
||||||
|
{
|
||||||
|
string _digit, _decpos;
|
||||||
|
int _pospunkt = _decsep.find_first_of(".");
|
||||||
|
// ESP_LOGD(TAG, "Name: %s, Pospunkt: %d", _decsep.c_str(), _pospunkt);
|
||||||
|
if (_pospunkt > -1)
|
||||||
|
_digit = _decsep.substr(0, _pospunkt);
|
||||||
|
else
|
||||||
|
_digit = "default";
|
||||||
|
for (int j = 0; j < flowpostprocessing->NUMBERS.size(); ++j)
|
||||||
|
{
|
||||||
|
if (_digit == "default") // Set to default first (if nothing else is set)
|
||||||
|
{
|
||||||
|
flowpostprocessing->NUMBERS[j]->MeasurementV2 = _value;
|
||||||
|
}
|
||||||
|
if (flowpostprocessing->NUMBERS[j]->name == _digit)
|
||||||
|
{
|
||||||
|
flowpostprocessing->NUMBERS[j]->MeasurementV2 = _value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
bool ClassFlowInfluxDBv2::doFlow(string zwtime)
|
bool ClassFlowInfluxDBv2::doFlow(string zwtime)
|
||||||
@@ -170,29 +197,35 @@ bool ClassFlowInfluxDBv2::doFlow(string zwtime)
|
|||||||
if (!InfluxDBenable)
|
if (!InfluxDBenable)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
std::string measurement;
|
||||||
std::string result;
|
std::string result;
|
||||||
std::string resulterror = "";
|
std::string resulterror = "";
|
||||||
std::string resultraw = "";
|
std::string resultraw = "";
|
||||||
std::string resultrate = "";
|
std::string resultrate = "";
|
||||||
std::string resulttimestamp = "";
|
std::string resulttimestamp = "";
|
||||||
|
long int resulttimeutc = 0;
|
||||||
string zw = "";
|
string zw = "";
|
||||||
string namenumber = "";
|
string namenumber = "";
|
||||||
|
|
||||||
|
|
||||||
if (flowpostprocessing)
|
if (flowpostprocessing)
|
||||||
{
|
{
|
||||||
std::vector<NumberPost*>* NUMBERS = flowpostprocessing->GetNumbers();
|
std::vector<NumberPost*>* NUMBERS = flowpostprocessing->GetNumbers();
|
||||||
|
|
||||||
for (int i = 0; i < (*NUMBERS).size(); ++i)
|
for (int i = 0; i < (*NUMBERS).size(); ++i)
|
||||||
{
|
{
|
||||||
|
measurement = (*NUMBERS)[i]->MeasurementV2;
|
||||||
result = (*NUMBERS)[i]->ReturnValue;
|
result = (*NUMBERS)[i]->ReturnValue;
|
||||||
resultraw = (*NUMBERS)[i]->ReturnRawValue;
|
resultraw = (*NUMBERS)[i]->ReturnRawValue;
|
||||||
resulterror = (*NUMBERS)[i]->ErrorMessageText;
|
resulterror = (*NUMBERS)[i]->ErrorMessageText;
|
||||||
resultrate = (*NUMBERS)[i]->ReturnRateValue;
|
resultrate = (*NUMBERS)[i]->ReturnRateValue;
|
||||||
resulttimestamp = (*NUMBERS)[i]->timeStamp;
|
resulttimestamp = (*NUMBERS)[i]->timeStamp;
|
||||||
|
resulttimeutc = (*NUMBERS)[i]->timeStampTimeUTC;
|
||||||
|
|
||||||
if ((*NUMBERS)[i]->Fieldname.length() > 0)
|
|
||||||
|
if ((*NUMBERS)[i]->FieldV2.length() > 0)
|
||||||
{
|
{
|
||||||
namenumber = (*NUMBERS)[i]->Fieldname;
|
namenumber = (*NUMBERS)[i]->FieldV2;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -206,8 +239,8 @@ bool ClassFlowInfluxDBv2::doFlow(string zwtime)
|
|||||||
printf("vor sende Influx_DB_V2 - namenumber. %s, result: %s, timestampt: %s", namenumber.c_str(), result.c_str(), resulttimestamp.c_str());
|
printf("vor sende Influx_DB_V2 - namenumber. %s, result: %s, timestampt: %s", namenumber.c_str(), result.c_str(), resulttimestamp.c_str());
|
||||||
|
|
||||||
if (result.length() > 0)
|
if (result.length() > 0)
|
||||||
InfluxDB_V2_Publish(namenumber, result, resulttimestamp);
|
influxdb.InfluxDBPublish(measurement, namenumber, result, resulttimeutc);
|
||||||
// InfluxDB_V2_Publish(namenumber, result, resulttimestamp);
|
// InfluxDB_V2_Publish(measurement, namenumber, result, resulttimeutc);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -9,28 +9,34 @@
|
|||||||
|
|
||||||
#include "ClassFlowPostProcessing.h"
|
#include "ClassFlowPostProcessing.h"
|
||||||
|
|
||||||
|
#include "interface_influxdb.h"
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
class ClassFlowInfluxDBv2 :
|
class ClassFlowInfluxDBv2 :
|
||||||
public ClassFlow
|
public ClassFlow
|
||||||
{
|
{
|
||||||
protected:
|
protected:
|
||||||
std::string uri, database, measurement;
|
std::string uri, bucket;
|
||||||
std::string dborg, dbtoken, dbfield;
|
std::string dborg, dbtoken, dbfield;
|
||||||
std::string OldValue;
|
std::string OldValue;
|
||||||
ClassFlowPostProcessing* flowpostprocessing;
|
ClassFlowPostProcessing* flowpostprocessing;
|
||||||
bool InfluxDBenable;
|
bool InfluxDBenable;
|
||||||
|
|
||||||
|
InfluxDB influxdb;
|
||||||
|
|
||||||
void SetInitialParameter(void);
|
void SetInitialParameter(void);
|
||||||
|
|
||||||
void handleFieldname(string _decsep, string _value);
|
void handleFieldname(string _decsep, string _value);
|
||||||
|
void handleMeasurement(string _decsep, string _value);
|
||||||
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
ClassFlowInfluxDBv2();
|
ClassFlowInfluxDBv2();
|
||||||
ClassFlowInfluxDBv2(std::vector<ClassFlow*>* lfc);
|
ClassFlowInfluxDBv2(std::vector<ClassFlow*>* lfc);
|
||||||
ClassFlowInfluxDBv2(std::vector<ClassFlow*>* lfc, ClassFlow *_prev);
|
ClassFlowInfluxDBv2(std::vector<ClassFlow*>* lfc, ClassFlow *_prev);
|
||||||
|
|
||||||
string GetInfluxDBMeasurement();
|
// string GetInfluxDBMeasurement();
|
||||||
|
|
||||||
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
||||||
bool doFlow(string time);
|
bool doFlow(string time);
|
||||||
|
|||||||
@@ -5,6 +5,7 @@
|
|||||||
#include "ClassFlowMQTT.h"
|
#include "ClassFlowMQTT.h"
|
||||||
#include "Helper.h"
|
#include "Helper.h"
|
||||||
#include "connect_wlan.h"
|
#include "connect_wlan.h"
|
||||||
|
#include "read_wlanini.h"
|
||||||
#include "ClassLogFile.h"
|
#include "ClassLogFile.h"
|
||||||
|
|
||||||
#include "time_sntp.h"
|
#include "time_sntp.h"
|
||||||
@@ -31,12 +32,16 @@ void ClassFlowMQTT::SetInitialParameter(void)
|
|||||||
topicError = "";
|
topicError = "";
|
||||||
topicRate = "";
|
topicRate = "";
|
||||||
topicTimeStamp = "";
|
topicTimeStamp = "";
|
||||||
maintopic = hostname;
|
maintopic = wlan_config.hostname;
|
||||||
|
|
||||||
topicUptime = "";
|
topicUptime = "";
|
||||||
topicFreeMem = "";
|
topicFreeMem = "";
|
||||||
|
|
||||||
clientname = "AIOTED-" + getMac();
|
caCertFilename = "";
|
||||||
|
clientCertFilename = "";
|
||||||
|
clientKeyFilename = "";
|
||||||
|
validateServerCert = true;
|
||||||
|
clientname = wlan_config.hostname;
|
||||||
|
|
||||||
OldValue = "";
|
OldValue = "";
|
||||||
flowpostprocessing = NULL;
|
flowpostprocessing = NULL;
|
||||||
@@ -47,6 +52,7 @@ void ClassFlowMQTT::SetInitialParameter(void)
|
|||||||
ListFlowControll = NULL;
|
ListFlowControll = NULL;
|
||||||
disabled = false;
|
disabled = false;
|
||||||
keepAlive = 25*60;
|
keepAlive = 25*60;
|
||||||
|
domoticzintopic = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
ClassFlowMQTT::ClassFlowMQTT()
|
ClassFlowMQTT::ClassFlowMQTT()
|
||||||
@@ -101,31 +107,46 @@ bool ClassFlowMQTT::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|||||||
while (this->getNextLine(pfile, &aktparamgraph) && !this->isNewParagraph(aktparamgraph))
|
while (this->getNextLine(pfile, &aktparamgraph) && !this->isNewParagraph(aktparamgraph))
|
||||||
{
|
{
|
||||||
splitted = ZerlegeZeile(aktparamgraph);
|
splitted = ZerlegeZeile(aktparamgraph);
|
||||||
if ((toUpper(splitted[0]) == "USER") && (splitted.size() > 1))
|
std::string _param = GetParameterName(splitted[0]);
|
||||||
|
if ((toUpper(_param) == "CACERT") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
this->caCertFilename = "/sdcard" + splitted[1];
|
||||||
|
}
|
||||||
|
if ((toUpper(_param) == "VALIDATESERVERCERT") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
validateServerCert = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
if ((toUpper(_param) == "CLIENTCERT") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
this->clientCertFilename = "/sdcard" + splitted[1];
|
||||||
|
}
|
||||||
|
if ((toUpper(_param) == "CLIENTKEY") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
this->clientKeyFilename = "/sdcard" + splitted[1];
|
||||||
|
}
|
||||||
|
if ((toUpper(_param) == "USER") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
this->user = splitted[1];
|
this->user = splitted[1];
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "PASSWORD") && (splitted.size() > 1))
|
if ((toUpper(_param) == "PASSWORD") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
this->password = splitted[1];
|
this->password = splitted[1];
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "URI") && (splitted.size() > 1))
|
if ((toUpper(_param) == "URI") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
this->uri = splitted[1];
|
this->uri = splitted[1];
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "RETAINMESSAGES") && (splitted.size() > 1))
|
if ((toUpper(_param) == "RETAINMESSAGES") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
if (toUpper(splitted[1]) == "TRUE") {
|
SetRetainFlag = alphanumericToBoolean(splitted[1]);
|
||||||
SetRetainFlag = true;
|
|
||||||
setMqtt_Server_Retain(SetRetainFlag);
|
setMqtt_Server_Retain(SetRetainFlag);
|
||||||
}
|
}
|
||||||
}
|
if ((toUpper(_param) == "HOMEASSISTANTDISCOVERY") && (splitted.size() > 1))
|
||||||
if ((toUpper(splitted[0]) == "HOMEASSISTANTDISCOVERY") && (splitted.size() > 1))
|
|
||||||
{
|
{
|
||||||
if (toUpper(splitted[1]) == "TRUE")
|
if (toUpper(splitted[1]) == "TRUE")
|
||||||
SetHomeassistantDiscoveryEnabled(true);
|
SetHomeassistantDiscoveryEnabled(true);
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "METERTYPE") && (splitted.size() > 1)) {
|
if ((toUpper(_param) == "METERTYPE") && (splitted.size() > 1)) {
|
||||||
/* Use meter type for the device class
|
/* Use meter type for the device class
|
||||||
Make sure it is a listed one on https://developers.home-assistant.io/docs/core/entity/sensor/#available-device-classes */
|
Make sure it is a listed one on https://developers.home-assistant.io/docs/core/entity/sensor/#available-device-classes */
|
||||||
if (toUpper(splitted[1]) == "WATER_M3") {
|
if (toUpper(splitted[1]) == "WATER_M3") {
|
||||||
@@ -135,7 +156,7 @@ bool ClassFlowMQTT::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|||||||
mqttServer_setMeterType("water", "L", "h", "L/h");
|
mqttServer_setMeterType("water", "L", "h", "L/h");
|
||||||
}
|
}
|
||||||
else if (toUpper(splitted[1]) == "WATER_FT3") {
|
else if (toUpper(splitted[1]) == "WATER_FT3") {
|
||||||
mqttServer_setMeterType("water", "ft³", "m", "ft³/m"); // Minutes
|
mqttServer_setMeterType("water", "ft³", "min", "ft³/min"); // min = Minutes
|
||||||
}
|
}
|
||||||
else if (toUpper(splitted[1]) == "WATER_GAL") {
|
else if (toUpper(splitted[1]) == "WATER_GAL") {
|
||||||
mqttServer_setMeterType("water", "gal", "h", "gal/h");
|
mqttServer_setMeterType("water", "gal", "h", "gal/h");
|
||||||
@@ -144,7 +165,7 @@ bool ClassFlowMQTT::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|||||||
mqttServer_setMeterType("gas", "m³", "h", "m³/h");
|
mqttServer_setMeterType("gas", "m³", "h", "m³/h");
|
||||||
}
|
}
|
||||||
else if (toUpper(splitted[1]) == "GAS_FT3") {
|
else if (toUpper(splitted[1]) == "GAS_FT3") {
|
||||||
mqttServer_setMeterType("gas", "ft³", "m", "ft³/m"); // Minutes
|
mqttServer_setMeterType("gas", "ft³", "min", "ft³/min"); // min = Minutes
|
||||||
}
|
}
|
||||||
else if (toUpper(splitted[1]) == "ENERGY_WH") {
|
else if (toUpper(splitted[1]) == "ENERGY_WH") {
|
||||||
mqttServer_setMeterType("energy", "Wh", "h", "W");
|
mqttServer_setMeterType("energy", "Wh", "h", "W");
|
||||||
@@ -155,18 +176,40 @@ bool ClassFlowMQTT::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|||||||
else if (toUpper(splitted[1]) == "ENERGY_MWH") {
|
else if (toUpper(splitted[1]) == "ENERGY_MWH") {
|
||||||
mqttServer_setMeterType("energy", "MWh", "h", "MW");
|
mqttServer_setMeterType("energy", "MWh", "h", "MW");
|
||||||
}
|
}
|
||||||
|
else if (toUpper(splitted[1]) == "ENERGY_GJ") {
|
||||||
|
mqttServer_setMeterType("energy", "GJ", "h", "GJ/h");
|
||||||
|
}
|
||||||
|
else if (toUpper(splitted[1]) == "TEMPERATURE_C") {
|
||||||
|
mqttServer_setMeterType("temperature", "°C", "min", "°C/min"); // min = Minutes
|
||||||
|
}
|
||||||
|
else if (toUpper(splitted[1]) == "TEMPERATURE_F") {
|
||||||
|
mqttServer_setMeterType("temperature", "°F", "min", "°F/min"); // min = Minutes
|
||||||
|
}
|
||||||
|
else if (toUpper(splitted[1]) == "TEMPERATURE_K") {
|
||||||
|
mqttServer_setMeterType("temperature", "K", "min", "K/m"); // min = Minutes
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "CLIENTID") && (splitted.size() > 1))
|
if ((toUpper(_param) == "CLIENTID") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
this->clientname = splitted[1];
|
this->clientname = splitted[1];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (((toUpper(splitted[0]) == "TOPIC") || (toUpper(splitted[0]) == "MAINTOPIC")) && (splitted.size() > 1))
|
if (((toUpper(_param) == "TOPIC") || (toUpper(splitted[0]) == "MAINTOPIC")) && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
maintopic = splitted[1];
|
maintopic = splitted[1];
|
||||||
mqttServer_setMainTopic(maintopic);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (((toUpper(_param) == "DOMOTICZTOPICIN")) && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
this->domoticzintopic = splitted[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (((toUpper(_param) == "DOMOTICZIDX")) && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
handleIdx(splitted[0], splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Note:
|
/* Note:
|
||||||
@@ -174,16 +217,13 @@ bool ClassFlowMQTT::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|||||||
* How ever we need the interval parameter from the ClassFlowControll, but that only gets started later.
|
* How ever we need the interval parameter from the ClassFlowControll, but that only gets started later.
|
||||||
* To work around this, we delay the start and trigger it from ClassFlowControll::ReadParameter() */
|
* To work around this, we delay the start and trigger it from ClassFlowControll::ReadParameter() */
|
||||||
|
|
||||||
|
mqttServer_setMainTopic(maintopic);
|
||||||
|
mqttServer_setDmoticzInTopic(domoticzintopic);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
string ClassFlowMQTT::GetMQTTMainTopic()
|
|
||||||
{
|
|
||||||
return maintopic;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool ClassFlowMQTT::Start(float AutoInterval)
|
bool ClassFlowMQTT::Start(float AutoInterval)
|
||||||
{
|
{
|
||||||
roundInterval = AutoInterval; // Minutes
|
roundInterval = AutoInterval; // Minutes
|
||||||
@@ -196,8 +236,9 @@ bool ClassFlowMQTT::Start(float AutoInterval)
|
|||||||
|
|
||||||
mqttServer_setParameter(flowpostprocessing->GetNumbers(), keepAlive, roundInterval);
|
mqttServer_setParameter(flowpostprocessing->GetNumbers(), keepAlive, roundInterval);
|
||||||
|
|
||||||
bool MQTTConfigCheck = MQTT_Configure(uri, clientname, user, password, maintopic, LWT_TOPIC, LWT_CONNECTED,
|
bool MQTTConfigCheck = MQTT_Configure(uri, clientname, user, password, maintopic, domoticzintopic, LWT_TOPIC, LWT_CONNECTED,
|
||||||
LWT_DISCONNECTED, keepAlive, SetRetainFlag, (void *)&GotConnected);
|
LWT_DISCONNECTED, caCertFilename, validateServerCert, clientCertFilename, clientKeyFilename,
|
||||||
|
keepAlive, SetRetainFlag, (void *)&GotConnected);
|
||||||
|
|
||||||
if (!MQTTConfigCheck) {
|
if (!MQTTConfigCheck) {
|
||||||
return false;
|
return false;
|
||||||
@@ -209,6 +250,7 @@ bool ClassFlowMQTT::Start(float AutoInterval)
|
|||||||
|
|
||||||
bool ClassFlowMQTT::doFlow(string zwtime)
|
bool ClassFlowMQTT::doFlow(string zwtime)
|
||||||
{
|
{
|
||||||
|
bool success;
|
||||||
std::string result;
|
std::string result;
|
||||||
std::string resulterror = "";
|
std::string resulterror = "";
|
||||||
std::string resultraw = "";
|
std::string resultraw = "";
|
||||||
@@ -219,8 +261,14 @@ bool ClassFlowMQTT::doFlow(string zwtime)
|
|||||||
std::string resultchangabs = "";
|
std::string resultchangabs = "";
|
||||||
string zw = "";
|
string zw = "";
|
||||||
string namenumber = "";
|
string namenumber = "";
|
||||||
|
string domoticzpayload = "";
|
||||||
|
string DomoticzIdx = "";
|
||||||
|
int qos = 1;
|
||||||
|
|
||||||
publishSystemData();
|
/* Send the the Homeassistant Discovery and the Static Topics in case they where scheduled */
|
||||||
|
sendDiscovery_and_static_Topics();
|
||||||
|
|
||||||
|
success = publishSystemData(qos);
|
||||||
|
|
||||||
if (flowpostprocessing && getMQTTisConnected())
|
if (flowpostprocessing && getMQTTisConnected())
|
||||||
{
|
{
|
||||||
@@ -238,21 +286,25 @@ bool ClassFlowMQTT::doFlow(string zwtime)
|
|||||||
resultchangabs = (*NUMBERS)[i]->ReturnChangeAbsolute; // Units per round
|
resultchangabs = (*NUMBERS)[i]->ReturnChangeAbsolute; // Units per round
|
||||||
resulttimestamp = (*NUMBERS)[i]->timeStamp;
|
resulttimestamp = (*NUMBERS)[i]->timeStamp;
|
||||||
|
|
||||||
|
DomoticzIdx = (*NUMBERS)[i]->DomoticzIdx;
|
||||||
|
domoticzpayload = "{\"command\":\"udevice\",\"idx\":" + DomoticzIdx + ",\"svalue\":\""+ result + "\"}";
|
||||||
|
|
||||||
namenumber = (*NUMBERS)[i]->name;
|
namenumber = (*NUMBERS)[i]->name;
|
||||||
if (namenumber == "default")
|
if (namenumber == "default")
|
||||||
namenumber = maintopic + "/";
|
namenumber = maintopic + "/";
|
||||||
else
|
else
|
||||||
namenumber = maintopic + "/" + namenumber + "/";
|
namenumber = maintopic + "/" + namenumber + "/";
|
||||||
|
|
||||||
|
if ((domoticzintopic.length() > 0) && (result.length() > 0))
|
||||||
|
success |= MQTTPublish(domoticzintopic, domoticzpayload, qos, SetRetainFlag);
|
||||||
|
|
||||||
if (result.length() > 0)
|
if (result.length() > 0)
|
||||||
MQTTPublish(namenumber + "value", result, SetRetainFlag);
|
success |= MQTTPublish(namenumber + "value", result, qos, SetRetainFlag);
|
||||||
|
|
||||||
if (resulterror.length() > 0)
|
if (resulterror.length() > 0)
|
||||||
MQTTPublish(namenumber + "error", resulterror, SetRetainFlag);
|
success |= MQTTPublish(namenumber + "error", resulterror, qos, SetRetainFlag);
|
||||||
|
|
||||||
if (resultrate.length() > 0) {
|
if (resultrate.length() > 0) {
|
||||||
MQTTPublish(namenumber + "rate", resultrate, SetRetainFlag);
|
success |= MQTTPublish(namenumber + "rate", resultrate, qos, SetRetainFlag);
|
||||||
|
|
||||||
std::string resultRatePerTimeUnit;
|
std::string resultRatePerTimeUnit;
|
||||||
if (getTimeUnit() == "h") { // Need conversion to be per hour
|
if (getTimeUnit() == "h") { // Need conversion to be per hour
|
||||||
@@ -261,22 +313,22 @@ bool ClassFlowMQTT::doFlow(string zwtime)
|
|||||||
else { // Keep per minute
|
else { // Keep per minute
|
||||||
resultRatePerTimeUnit = resultrate;
|
resultRatePerTimeUnit = resultrate;
|
||||||
}
|
}
|
||||||
MQTTPublish(namenumber + "rate_per_time_unit", resultRatePerTimeUnit, SetRetainFlag);
|
success |= MQTTPublish(namenumber + "rate_per_time_unit", resultRatePerTimeUnit, qos, SetRetainFlag);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (resultchangabs.length() > 0) {
|
if (resultchangabs.length() > 0) {
|
||||||
MQTTPublish(namenumber + "changeabsolut", resultchangabs, SetRetainFlag); // Legacy API
|
success |= MQTTPublish(namenumber + "changeabsolut", resultchangabs, qos, SetRetainFlag); // Legacy API
|
||||||
MQTTPublish(namenumber + "rate_per_digitalization_round", resultchangabs, SetRetainFlag);
|
success |= MQTTPublish(namenumber + "rate_per_digitization_round", resultchangabs, qos, SetRetainFlag);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (resultraw.length() > 0)
|
if (resultraw.length() > 0)
|
||||||
MQTTPublish(namenumber + "raw", resultraw, SetRetainFlag);
|
success |= MQTTPublish(namenumber + "raw", resultraw, qos, SetRetainFlag);
|
||||||
|
|
||||||
if (resulttimestamp.length() > 0)
|
if (resulttimestamp.length() > 0)
|
||||||
MQTTPublish(namenumber + "timestamp", resulttimestamp, SetRetainFlag);
|
success |= MQTTPublish(namenumber + "timestamp", resulttimestamp, qos, SetRetainFlag);
|
||||||
|
|
||||||
std::string json = flowpostprocessing->getJsonFromNumber(i, "\n");
|
std::string json = flowpostprocessing->getJsonFromNumber(i, "\n");
|
||||||
MQTTPublish(namenumber + "json", json, SetRetainFlag);
|
success |= MQTTPublish(namenumber + "json", json, qos, SetRetainFlag);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -294,13 +346,37 @@ bool ClassFlowMQTT::doFlow(string zwtime)
|
|||||||
// result = result + "\t" + zw;
|
// result = result + "\t" + zw;
|
||||||
// }
|
// }
|
||||||
// }
|
// }
|
||||||
// MQTTPublish(topic, result, SetRetainFlag);
|
// success |= MQTTPublish(topic, result, qos, SetRetainFlag);
|
||||||
// }
|
// }
|
||||||
|
|
||||||
OldValue = result;
|
OldValue = result;
|
||||||
|
|
||||||
|
if (!success) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_WARN, TAG, "One or more MQTT topics failed to be published!");
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
void ClassFlowMQTT::handleIdx(string _decsep, string _value)
|
||||||
|
{
|
||||||
|
string _digit, _decpos;
|
||||||
|
int _pospunkt = _decsep.find_first_of(".");
|
||||||
|
// ESP_LOGD(TAG, "Name: %s, Pospunkt: %d", _decsep.c_str(), _pospunkt);
|
||||||
|
if (_pospunkt > -1)
|
||||||
|
_digit = _decsep.substr(0, _pospunkt);
|
||||||
|
else
|
||||||
|
_digit = "default";
|
||||||
|
for (int j = 0; j < flowpostprocessing->NUMBERS.size(); ++j)
|
||||||
|
{
|
||||||
|
if (_digit == "default") // Set to default first (if nothing else is set)
|
||||||
|
{
|
||||||
|
flowpostprocessing->NUMBERS[j]->DomoticzIdx = _value;
|
||||||
|
}
|
||||||
|
if (flowpostprocessing->NUMBERS[j]->name == _digit)
|
||||||
|
{
|
||||||
|
flowpostprocessing->NUMBERS[j]->DomoticzIdx = _value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#endif //ENABLE_MQTT
|
#endif //ENABLE_MQTT
|
||||||
@@ -19,19 +19,20 @@ protected:
|
|||||||
std::string OldValue;
|
std::string OldValue;
|
||||||
ClassFlowPostProcessing* flowpostprocessing;
|
ClassFlowPostProcessing* flowpostprocessing;
|
||||||
std::string user, password;
|
std::string user, password;
|
||||||
|
std::string caCertFilename, clientCertFilename, clientKeyFilename;
|
||||||
|
bool validateServerCert;
|
||||||
bool SetRetainFlag;
|
bool SetRetainFlag;
|
||||||
int keepAlive; // Seconds
|
int keepAlive; // Seconds
|
||||||
float roundInterval; // Minutes
|
float roundInterval; // Minutes
|
||||||
|
std::string maintopic, domoticzintopic;
|
||||||
std::string maintopic;
|
|
||||||
void SetInitialParameter(void);
|
void SetInitialParameter(void);
|
||||||
|
void handleIdx(string _decsep, string _value);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
ClassFlowMQTT();
|
ClassFlowMQTT();
|
||||||
ClassFlowMQTT(std::vector<ClassFlow*>* lfc);
|
ClassFlowMQTT(std::vector<ClassFlow*>* lfc);
|
||||||
ClassFlowMQTT(std::vector<ClassFlow*>* lfc, ClassFlow *_prev);
|
ClassFlowMQTT(std::vector<ClassFlow*>* lfc, ClassFlow *_prev);
|
||||||
|
|
||||||
string GetMQTTMainTopic();
|
|
||||||
bool Start(float AutoInterval);
|
bool Start(float AutoInterval);
|
||||||
|
|
||||||
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -10,7 +10,6 @@
|
|||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
|
|
||||||
class ClassFlowPostProcessing :
|
class ClassFlowPostProcessing :
|
||||||
public ClassFlow
|
public ClassFlow
|
||||||
{
|
{
|
||||||
@@ -19,13 +18,10 @@ protected:
|
|||||||
|
|
||||||
int PreValueAgeStartup;
|
int PreValueAgeStartup;
|
||||||
bool ErrorMessage;
|
bool ErrorMessage;
|
||||||
bool IgnoreLeadingNaN; // SPECIAL CASE for User Gustl ???
|
|
||||||
|
|
||||||
|
|
||||||
ClassFlowCNNGeneral* flowAnalog;
|
ClassFlowCNNGeneral* flowAnalog;
|
||||||
ClassFlowCNNGeneral* flowDigit;
|
ClassFlowCNNGeneral* flowDigit;
|
||||||
|
|
||||||
|
|
||||||
string FilePreValue;
|
string FilePreValue;
|
||||||
|
|
||||||
ClassFlowTakeImage *flowTakeImage;
|
ClassFlowTakeImage *flowTakeImage;
|
||||||
@@ -37,25 +33,23 @@ protected:
|
|||||||
float checkDigitConsistency(double input, int _decilamshift, bool _isanalog, double _preValue);
|
float checkDigitConsistency(double input, int _decilamshift, bool _isanalog, double _preValue);
|
||||||
|
|
||||||
void InitNUMBERS();
|
void InitNUMBERS();
|
||||||
|
|
||||||
void handleDecimalSeparator(string _decsep, string _value);
|
void handleDecimalSeparator(string _decsep, string _value);
|
||||||
void handleMaxRateValue(string _decsep, string _value);
|
void handleMaxRateValue(string _decsep, string _value);
|
||||||
void handleDecimalExtendedResolution(string _decsep, string _value);
|
void handleDecimalExtendedResolution(string _decsep, string _value);
|
||||||
void handleMaxRateType(string _decsep, string _value);
|
void handleMaxRateType(string _decsep, string _value);
|
||||||
void handleAnalogDigitalTransitionStart(string _decsep, string _value);
|
void handleAnalogToDigitTransitionStart(string _decsep, string _value);
|
||||||
void handleAllowNegativeRate(string _decsep, string _value);
|
void handleAllowNegativeRate(string _decsep, string _value);
|
||||||
|
void handleIgnoreLeadingNaN(string _decsep, string _value);
|
||||||
std::string GetStringReadouts(general);
|
void handleChangeRateThreshold(string _decsep, string _value);
|
||||||
|
void handlecheckDigitIncreaseConsistency(std::string _decsep, std::string _value);
|
||||||
|
|
||||||
void WriteDataLog(int _index);
|
void WriteDataLog(int _index);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
bool PreValueUse;
|
bool PreValueUse;
|
||||||
std::vector<NumberPost*> NUMBERS;
|
std::vector<NumberPost*> NUMBERS;
|
||||||
|
|
||||||
|
|
||||||
ClassFlowPostProcessing(std::vector<ClassFlow*>* lfc, ClassFlowCNNGeneral *_analog, ClassFlowCNNGeneral *_digit);
|
ClassFlowPostProcessing(std::vector<ClassFlow*>* lfc, ClassFlowCNNGeneral *_analog, ClassFlowCNNGeneral *_digit);
|
||||||
virtual ~ClassFlowPostProcessing(){};
|
virtual ~ClassFlowPostProcessing(){};
|
||||||
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
||||||
@@ -68,7 +62,7 @@ public:
|
|||||||
void SavePreValue();
|
void SavePreValue();
|
||||||
string getJsonFromNumber(int i, std::string _lineend);
|
string getJsonFromNumber(int i, std::string _lineend);
|
||||||
string GetPreValue(std::string _number = "");
|
string GetPreValue(std::string _number = "");
|
||||||
void SetPreValue(double zw, string _numbers, bool _extern = false);
|
bool SetPreValue(double zw, string _numbers, bool _extern = false);
|
||||||
|
|
||||||
std::string GetJSON(std::string _lineend = "\n");
|
std::string GetJSON(std::string _lineend = "\n");
|
||||||
std::string getNumbersName();
|
std::string getNumbersName();
|
||||||
@@ -80,5 +74,4 @@ public:
|
|||||||
string name(){return "ClassFlowPostProcessing";};
|
string name(){return "ClassFlowPostProcessing";};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
#endif //CLASSFFLOWPOSTPROCESSING_H
|
#endif //CLASSFFLOWPOSTPROCESSING_H
|
||||||
|
|||||||
@@ -1,23 +1,30 @@
|
|||||||
|
#include <iostream>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
#include <regex>
|
||||||
|
|
||||||
#include "ClassFlowTakeImage.h"
|
#include "ClassFlowTakeImage.h"
|
||||||
#include "Helper.h"
|
#include "Helper.h"
|
||||||
#include "ClassLogFile.h"
|
#include "ClassLogFile.h"
|
||||||
|
|
||||||
#include "CImageBasis.h"
|
#include "CImageBasis.h"
|
||||||
#include "ClassControllCamera.h"
|
#include "ClassControllCamera.h"
|
||||||
|
#include "MainFlowControl.h"
|
||||||
|
|
||||||
#include "esp_wifi.h"
|
#include "esp_wifi.h"
|
||||||
#include "esp_log.h"
|
#include "esp_log.h"
|
||||||
#include "../../include/defines.h"
|
#include "../../include/defines.h"
|
||||||
|
#include "psram.h"
|
||||||
|
|
||||||
#include <time.h>
|
#include <time.h>
|
||||||
|
|
||||||
// #define DEBUG_DETAIL_ON
|
// #define DEBUG_DETAIL_ON
|
||||||
|
|
||||||
// #define WIFITURNOFF
|
// #define WIFITURNOFF
|
||||||
|
|
||||||
static const char* TAG = "flow_make_image";
|
static const char *TAG = "TAKEIMAGE";
|
||||||
|
|
||||||
esp_err_t ClassFlowTakeImage::camera_capture(){
|
esp_err_t ClassFlowTakeImage::camera_capture(void)
|
||||||
|
{
|
||||||
string nm = namerawimage;
|
string nm = namerawimage;
|
||||||
Camera.CaptureToFile(nm);
|
Camera.CaptureToFile(nm);
|
||||||
time(&TimeImageTaken);
|
time(&TimeImageTaken);
|
||||||
@@ -29,149 +36,499 @@ esp_err_t ClassFlowTakeImage::camera_capture(){
|
|||||||
void ClassFlowTakeImage::takePictureWithFlash(int flash_duration)
|
void ClassFlowTakeImage::takePictureWithFlash(int flash_duration)
|
||||||
{
|
{
|
||||||
// in case the image is flipped, it must be reset here //
|
// in case the image is flipped, it must be reset here //
|
||||||
rawImage->width = image_width;
|
rawImage->width = CCstatus.ImageWidth;
|
||||||
rawImage->height = image_height;
|
rawImage->height = CCstatus.ImageHeight;
|
||||||
/////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
ESP_LOGD(TAG, "flash_duration: %d", flash_duration);
|
ESP_LOGD(TAG, "flash_duration: %d", flash_duration);
|
||||||
|
|
||||||
Camera.CaptureToBasisImage(rawImage, flash_duration);
|
Camera.CaptureToBasisImage(rawImage, flash_duration);
|
||||||
|
|
||||||
time(&TimeImageTaken);
|
time(&TimeImageTaken);
|
||||||
localtime(&TimeImageTaken);
|
localtime(&TimeImageTaken);
|
||||||
|
|
||||||
if (SaveAllFiles) rawImage->SaveToFile(namerawimage);
|
if (CCstatus.SaveAllFiles)
|
||||||
|
{
|
||||||
|
rawImage->SaveToFile(namerawimage);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void ClassFlowTakeImage::SetInitialParameter(void)
|
void ClassFlowTakeImage::SetInitialParameter(void)
|
||||||
{
|
{
|
||||||
waitbeforepicture = 5;
|
|
||||||
isImageSize = false;
|
|
||||||
ImageQuality = -1;
|
|
||||||
TimeImageTaken = 0;
|
TimeImageTaken = 0;
|
||||||
ImageQuality = 5;
|
|
||||||
rawImage = NULL;
|
rawImage = NULL;
|
||||||
ImageSize = FRAMESIZE_VGA;
|
|
||||||
SaveAllFiles = false;
|
|
||||||
disabled = false;
|
disabled = false;
|
||||||
FixedExposure = false;
|
|
||||||
namerawimage = "/sdcard/img_tmp/raw.jpg";
|
namerawimage = "/sdcard/img_tmp/raw.jpg";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// auslesen der Kameraeinstellungen aus der config.ini
|
||||||
ClassFlowTakeImage::ClassFlowTakeImage(std::vector<ClassFlow*>* lfc) : ClassFlowImage(lfc, TAG)
|
// wird beim Start aufgerufen
|
||||||
|
bool ClassFlowTakeImage::ReadParameter(FILE *pfile, string &aktparamgraph)
|
||||||
{
|
{
|
||||||
imagesLocation = "/log/source";
|
Camera.getSensorDatenToCCstatus(); // Kamera >>> CCstatus
|
||||||
imagesRetention = 5;
|
|
||||||
SetInitialParameter();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool ClassFlowTakeImage::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|
||||||
{
|
|
||||||
std::vector<string> splitted;
|
std::vector<string> splitted;
|
||||||
|
|
||||||
aktparamgraph = trim(aktparamgraph);
|
aktparamgraph = trim(aktparamgraph);
|
||||||
int _brightness = -100;
|
|
||||||
int _contrast = -100;
|
|
||||||
int _saturation = -100;
|
|
||||||
|
|
||||||
if (aktparamgraph.size() == 0)
|
if (aktparamgraph.size() == 0)
|
||||||
|
{
|
||||||
if (!this->GetNextParagraph(pfile, aktparamgraph))
|
if (!this->GetNextParagraph(pfile, aktparamgraph))
|
||||||
|
{
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (aktparamgraph.compare("[TakeImage]") != 0) // Paragraph does not fit TakeImage
|
if (aktparamgraph.compare("[TakeImage]") != 0)
|
||||||
|
{
|
||||||
|
// Paragraph does not fit TakeImage
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
while (this->getNextLine(pfile, &aktparamgraph) && !this->isNewParagraph(aktparamgraph))
|
while (this->getNextLine(pfile, &aktparamgraph) && !this->isNewParagraph(aktparamgraph))
|
||||||
{
|
{
|
||||||
splitted = ZerlegeZeile(aktparamgraph);
|
splitted = ZerlegeZeile(aktparamgraph);
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "RAWIMAGESLOCATION") && (splitted.size() > 1))
|
if ((toUpper(splitted[0]) == "RAWIMAGESLOCATION") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
imagesLocation = "/sdcard" + splitted[1];
|
imagesLocation = "/sdcard" + splitted[1];
|
||||||
isLogImage = true;
|
isLogImage = true;
|
||||||
}
|
}
|
||||||
if ((toUpper(splitted[0]) == "IMAGEQUALITY") && (splitted.size() > 1))
|
|
||||||
ImageQuality = std::stod(splitted[1]);
|
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "IMAGESIZE") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "RAWIMAGESRETENTION") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
ImageSize = Camera.TextToFramesize(splitted[1].c_str());
|
if (isStringNumeric(splitted[1]))
|
||||||
isImageSize = true;
|
{
|
||||||
|
this->imagesRetention = std::stod(splitted[1]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "SAVEALLFILES") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "SAVEALLFILES") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
if (toUpper(splitted[1]) == "TRUE")
|
CCstatus.SaveAllFiles = alphanumericToBoolean(splitted[1]);
|
||||||
SaveAllFiles = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "WAITBEFORETAKINGPICTURE") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "WAITBEFORETAKINGPICTURE") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
waitbeforepicture = stoi(splitted[1]);
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _WaitBeforePicture = std::stoi(splitted[1]);
|
||||||
|
if (_WaitBeforePicture != 0)
|
||||||
|
{
|
||||||
|
CCstatus.WaitBeforePicture = _WaitBeforePicture;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CCstatus.WaitBeforePicture = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "RAWIMAGESRETENTION") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "CAMGAINCEILING") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
this->imagesRetention = std::stoi(splitted[1]);
|
std::string _ImageGainceiling = toUpper(splitted[1]);
|
||||||
|
|
||||||
|
if (isStringNumeric(_ImageGainceiling))
|
||||||
|
{
|
||||||
|
int _ImageGainceiling_ = std::stoi(_ImageGainceiling);
|
||||||
|
switch (_ImageGainceiling_)
|
||||||
|
{
|
||||||
|
case 1:
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_4X;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_8X;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_16X;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_32X;
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_64X;
|
||||||
|
break;
|
||||||
|
case 6:
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_128X;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_2X;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (_ImageGainceiling == "X4")
|
||||||
|
{
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_4X;
|
||||||
|
}
|
||||||
|
else if (_ImageGainceiling == "X8")
|
||||||
|
{
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_8X;
|
||||||
|
}
|
||||||
|
else if (_ImageGainceiling == "X16")
|
||||||
|
{
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_16X;
|
||||||
|
}
|
||||||
|
else if (_ImageGainceiling == "X32")
|
||||||
|
{
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_32X;
|
||||||
|
}
|
||||||
|
else if (_ImageGainceiling == "X64")
|
||||||
|
{
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_64X;
|
||||||
|
}
|
||||||
|
else if (_ImageGainceiling == "X128")
|
||||||
|
{
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_128X;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CCstatus.ImageGainceiling = GAINCEILING_2X;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "BRIGHTNESS") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "CAMQUALITY") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
_brightness = stoi(splitted[1]);
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _ImageQuality = std::stoi(splitted[1]);
|
||||||
|
CCstatus.ImageQuality = clipInt(_ImageQuality, 63, 6);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "CONTRAST") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "CAMBRIGHTNESS") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
_contrast = stoi(splitted[1]);
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _ImageBrightness = std::stoi(splitted[1]);
|
||||||
|
CCstatus.ImageBrightness = clipInt(_ImageBrightness, 2, -2);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "SATURATION") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "CAMCONTRAST") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
_saturation = stoi(splitted[1]);
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _ImageContrast = std::stoi(splitted[1]);
|
||||||
|
CCstatus.ImageContrast = clipInt(_ImageContrast, 2, -2);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "FIXEDEXPOSURE") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "CAMSATURATION") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
if (toUpper(splitted[1]) == "TRUE")
|
if (isStringNumeric(splitted[1]))
|
||||||
FixedExposure = true;
|
{
|
||||||
|
int _ImageSaturation = std::stoi(splitted[1]);
|
||||||
|
CCstatus.ImageSaturation = clipInt(_ImageSaturation, 2, -2);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "LEDINTENSITY") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "CAMSHARPNESS") && (splitted.size() > 1))
|
||||||
{
|
{
|
||||||
float ledintensity = stof(splitted[1]);
|
if (isStringNumeric(splitted[1]))
|
||||||
ledintensity = min((float) 100, ledintensity);
|
{
|
||||||
ledintensity = max((float) 0, ledintensity);
|
int _ImageSharpness = std::stoi(splitted[1]);
|
||||||
Camera.SetLEDIntensity(ledintensity);
|
if (CCstatus.CamSensor_id == OV2640_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageSharpness = clipInt(_ImageSharpness, 2, -2);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CCstatus.ImageSharpness = clipInt(_ImageSharpness, 3, -3);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((toUpper(splitted[0]) == "DEMO") && (splitted.size() > 1))
|
else if ((toUpper(splitted[0]) == "CAMAUTOSHARPNESS") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageAutoSharpness = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMSPECIALEFFECT") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
std::string _ImageSpecialEffect = toUpper(splitted[1]);
|
||||||
|
|
||||||
|
if (isStringNumeric(_ImageSpecialEffect))
|
||||||
|
{
|
||||||
|
int _ImageSpecialEffect_ = std::stoi(_ImageSpecialEffect);
|
||||||
|
CCstatus.ImageSpecialEffect = clipInt(_ImageSpecialEffect_, 6, 0);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (_ImageSpecialEffect == "NEGATIVE")
|
||||||
|
{
|
||||||
|
CCstatus.ImageSpecialEffect = 1;
|
||||||
|
}
|
||||||
|
else if (_ImageSpecialEffect == "GRAYSCALE")
|
||||||
|
{
|
||||||
|
CCstatus.ImageSpecialEffect = 2;
|
||||||
|
}
|
||||||
|
else if (_ImageSpecialEffect == "RED")
|
||||||
|
{
|
||||||
|
CCstatus.ImageSpecialEffect = 3;
|
||||||
|
}
|
||||||
|
else if (_ImageSpecialEffect == "GREEN")
|
||||||
|
{
|
||||||
|
CCstatus.ImageSpecialEffect = 4;
|
||||||
|
}
|
||||||
|
else if (_ImageSpecialEffect == "BLUE")
|
||||||
|
{
|
||||||
|
CCstatus.ImageSpecialEffect = 5;
|
||||||
|
}
|
||||||
|
else if (_ImageSpecialEffect == "RETRO")
|
||||||
|
{
|
||||||
|
CCstatus.ImageSpecialEffect = 6;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CCstatus.ImageSpecialEffect = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMWBMODE") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
std::string _ImageWbMode = toUpper(splitted[1]);
|
||||||
|
|
||||||
|
if (isStringNumeric(_ImageWbMode))
|
||||||
|
{
|
||||||
|
int _ImageWbMode_ = std::stoi(_ImageWbMode);
|
||||||
|
CCstatus.ImageWbMode = clipInt(_ImageWbMode_, 4, 0);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (_ImageWbMode == "SUNNY")
|
||||||
|
{
|
||||||
|
CCstatus.ImageWbMode = 1;
|
||||||
|
}
|
||||||
|
else if (_ImageWbMode == "CLOUDY")
|
||||||
|
{
|
||||||
|
CCstatus.ImageWbMode = 2;
|
||||||
|
}
|
||||||
|
else if (_ImageWbMode == "OFFICE")
|
||||||
|
{
|
||||||
|
CCstatus.ImageWbMode = 3;
|
||||||
|
}
|
||||||
|
else if (_ImageWbMode == "HOME")
|
||||||
|
{
|
||||||
|
CCstatus.ImageWbMode = 4;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CCstatus.ImageWbMode = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMAWB") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageAwb = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMAWBGAIN") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageAwbGain = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMAEC") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageAec = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMAEC2") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageAec2 = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMAELEVEL") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _ImageAeLevel = std::stoi(splitted[1]);
|
||||||
|
if (CCstatus.CamSensor_id == OV2640_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageAeLevel = clipInt(_ImageAeLevel, 2, -2);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CCstatus.ImageAeLevel = clipInt(_ImageAeLevel, 5, -5);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMAECVALUE") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _ImageAecValue = std::stoi(splitted[1]);
|
||||||
|
CCstatus.ImageAecValue = clipInt(_ImageAecValue, 1200, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMAGC") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageAgc = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMAGCGAIN") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _ImageAgcGain = std::stoi(splitted[1]);
|
||||||
|
CCstatus.ImageAgcGain = clipInt(_ImageAgcGain, 30, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMBPC") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageBpc = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMWPC") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageWpc = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMRAWGMA") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageRawGma = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMLENC") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageLenc = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMHMIRROR") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageHmirror = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMVFLIP") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageVflip = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMDCW") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageDcw = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMDENOISE") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _ImageDenoiseLevel = std::stoi(splitted[1]);
|
||||||
|
if (CCstatus.CamSensor_id == OV2640_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageDenoiseLevel = 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CCstatus.ImageDenoiseLevel = clipInt(_ImageDenoiseLevel, 8, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMZOOM") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.ImageZoomEnabled = alphanumericToBoolean(splitted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMZOOMOFFSETX") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _ImageZoomOffsetX = std::stoi(splitted[1]);
|
||||||
|
if (CCstatus.CamSensor_id == OV2640_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageZoomOffsetX = clipInt(_ImageZoomOffsetX, 480, -480);
|
||||||
|
}
|
||||||
|
else if (CCstatus.CamSensor_id == OV3660_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageZoomOffsetX = clipInt(_ImageZoomOffsetX, 704, -704);
|
||||||
|
}
|
||||||
|
else if (CCstatus.CamSensor_id == OV5640_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageZoomOffsetX = clipInt(_ImageZoomOffsetX, 960, -960);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMZOOMOFFSETY") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _ImageZoomOffsetY = std::stoi(splitted[1]);
|
||||||
|
if (CCstatus.CamSensor_id == OV2640_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageZoomOffsetY = clipInt(_ImageZoomOffsetY, 360, -360);
|
||||||
|
}
|
||||||
|
else if (CCstatus.CamSensor_id == OV3660_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageZoomOffsetY = clipInt(_ImageZoomOffsetY, 528, -528);
|
||||||
|
}
|
||||||
|
else if (CCstatus.CamSensor_id == OV5640_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageZoomOffsetY = clipInt(_ImageZoomOffsetY, 720, -720);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "CAMZOOMSIZE") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int _ImageZoomSize = std::stoi(splitted[1]);
|
||||||
|
if (CCstatus.CamSensor_id == OV2640_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageZoomSize = clipInt(_ImageZoomSize, 29, 0);
|
||||||
|
}
|
||||||
|
else if (CCstatus.CamSensor_id == OV3660_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageZoomSize = clipInt(_ImageZoomSize, 43, 0);
|
||||||
|
}
|
||||||
|
else if (CCstatus.CamSensor_id == OV5640_PID)
|
||||||
|
{
|
||||||
|
CCstatus.ImageZoomSize = clipInt(_ImageZoomSize, 59, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "LEDINTENSITY") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
if (isStringNumeric(splitted[1]))
|
||||||
|
{
|
||||||
|
int ledintensity = std::stoi(splitted[1]);
|
||||||
|
CCstatus.ImageLedIntensity = Camera.SetLEDIntensity(ledintensity);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if ((toUpper(splitted[0]) == "DEMO") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
CCstatus.DemoMode = alphanumericToBoolean(splitted[1]);
|
||||||
|
if (CCstatus.DemoMode == true)
|
||||||
{
|
{
|
||||||
if (toUpper(splitted[1]) == "TRUE")
|
|
||||||
Camera.useDemoMode();
|
Camera.useDemoMode();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Camera.SetBrightnessContrastSaturation(_brightness, _contrast, _saturation);
|
|
||||||
Camera.SetQualitySize(ImageQuality, ImageSize);
|
|
||||||
|
|
||||||
image_width = Camera.image_width;
|
|
||||||
image_height = Camera.image_height;
|
|
||||||
rawImage = new CImageBasis();
|
|
||||||
rawImage->CreateEmptyImage(image_width, image_height, 3);
|
|
||||||
|
|
||||||
waitbeforepicture_store = waitbeforepicture;
|
|
||||||
if (FixedExposure && (waitbeforepicture > 0))
|
|
||||||
{
|
|
||||||
// ESP_LOGD(TAG, "Fixed Exposure enabled!");
|
|
||||||
int flash_duration = (int) (waitbeforepicture * 1000);
|
|
||||||
Camera.EnableAutoExposure(flash_duration);
|
|
||||||
waitbeforepicture = 0.2;
|
|
||||||
// flash_duration = (int) (waitbeforepicture * 1000);
|
|
||||||
// takePictureWithFlash(flash_duration);
|
|
||||||
// rawImage->SaveToFile("/sdcard/init2.jpg");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Camera.setSensorDatenFromCCstatus(); // CCstatus >>> Kamera
|
||||||
|
Camera.SetQualityZoomSize(CCstatus.ImageQuality, CCstatus.ImageFrameSize, CCstatus.ImageZoomEnabled, CCstatus.ImageZoomOffsetX, CCstatus.ImageZoomOffsetY, CCstatus.ImageZoomSize, CCstatus.ImageVflip);
|
||||||
|
|
||||||
|
rawImage = new CImageBasis("rawImage");
|
||||||
|
rawImage->CreateEmptyImage(CCstatus.ImageWidth, CCstatus.ImageHeight, 3);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ClassFlowTakeImage::ClassFlowTakeImage(std::vector<ClassFlow *> *lfc) : ClassFlowImage(lfc, TAG)
|
||||||
|
{
|
||||||
|
imagesLocation = "/log/source";
|
||||||
|
imagesRetention = 5;
|
||||||
|
SetInitialParameter();
|
||||||
|
}
|
||||||
|
|
||||||
string ClassFlowTakeImage::getHTMLSingleStep(string host)
|
string ClassFlowTakeImage::getHTMLSingleStep(string host)
|
||||||
{
|
{
|
||||||
@@ -180,60 +537,69 @@ string ClassFlowTakeImage::getHTMLSingleStep(string host)
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// wird bei jeder Auswertrunde aufgerufen
|
||||||
bool ClassFlowTakeImage::doFlow(string zwtime)
|
bool ClassFlowTakeImage::doFlow(string zwtime)
|
||||||
{
|
{
|
||||||
|
psram_init_shared_memory_for_take_image_step();
|
||||||
|
|
||||||
string logPath = CreateLogFolder(zwtime);
|
string logPath = CreateLogFolder(zwtime);
|
||||||
|
|
||||||
int flash_duration = (int) (waitbeforepicture * 1000);
|
int flash_duration = (int)(CCstatus.WaitBeforePicture * 1000);
|
||||||
|
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("ClassFlowTakeImage::doFlow - Before takePictureWithFlash");
|
LogFile.WriteHeapInfo("ClassFlowTakeImage::doFlow - Before takePictureWithFlash");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef WIFITURNOFF
|
||||||
#ifdef WIFITURNOFF
|
|
||||||
esp_wifi_stop(); // to save power usage and
|
esp_wifi_stop(); // to save power usage and
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
// wenn die Kameraeinstellungen durch Erstellen eines neuen Referenzbildes verändert wurden, müssen sie neu gesetzt werden
|
||||||
|
if (CFstatus.changedCameraSettings)
|
||||||
|
{
|
||||||
|
Camera.setSensorDatenFromCCstatus(); // CCstatus >>> Kamera
|
||||||
|
Camera.SetQualityZoomSize(CCstatus.ImageQuality, CCstatus.ImageFrameSize, CCstatus.ImageZoomEnabled, CCstatus.ImageZoomOffsetX, CCstatus.ImageZoomOffsetY, CCstatus.ImageZoomSize, CCstatus.ImageVflip);
|
||||||
|
Camera.LedIntensity = CCstatus.ImageLedIntensity;
|
||||||
|
CFstatus.changedCameraSettings = false;
|
||||||
|
}
|
||||||
|
|
||||||
takePictureWithFlash(flash_duration);
|
takePictureWithFlash(flash_duration);
|
||||||
|
|
||||||
#ifdef WIFITURNOFF
|
#ifdef WIFITURNOFF
|
||||||
esp_wifi_start();
|
esp_wifi_start();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef DEBUG_DETAIL_ON
|
||||||
#ifdef DEBUG_DETAIL_ON
|
|
||||||
LogFile.WriteHeapInfo("ClassFlowTakeImage::doFlow - After takePictureWithFlash");
|
LogFile.WriteHeapInfo("ClassFlowTakeImage::doFlow - After takePictureWithFlash");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
LogImage(logPath, "raw", NULL, NULL, zwtime, rawImage);
|
LogImage(logPath, "raw", NULL, NULL, zwtime, rawImage);
|
||||||
|
|
||||||
RemoveOldLogs();
|
RemoveOldLogs();
|
||||||
|
|
||||||
#ifdef DEBUG_DETAIL_ON
|
#ifdef DEBUG_DETAIL_ON
|
||||||
LogFile.WriteHeapInfo("ClassFlowTakeImage::doFlow - After RemoveOldLogs");
|
LogFile.WriteHeapInfo("ClassFlowTakeImage::doFlow - After RemoveOldLogs");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
psram_deinit_shared_memory_for_take_image_step();
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
esp_err_t ClassFlowTakeImage::SendRawJPG(httpd_req_t *req)
|
esp_err_t ClassFlowTakeImage::SendRawJPG(httpd_req_t *req)
|
||||||
{
|
{
|
||||||
int flash_duration = (int) (waitbeforepicture * 1000);
|
int flash_duration = (int)(CCstatus.WaitBeforePicture * 1000);
|
||||||
time(&TimeImageTaken);
|
time(&TimeImageTaken);
|
||||||
localtime(&TimeImageTaken);
|
localtime(&TimeImageTaken);
|
||||||
|
|
||||||
return Camera.CaptureToHTTP(req, flash_duration);
|
return Camera.CaptureToHTTP(req, flash_duration);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ImageData *ClassFlowTakeImage::SendRawImage(void)
|
||||||
ImageData* ClassFlowTakeImage::SendRawImage()
|
|
||||||
{
|
{
|
||||||
CImageBasis *zw = new CImageBasis(rawImage);
|
CImageBasis *zw = new CImageBasis("SendRawImage", rawImage);
|
||||||
ImageData *id;
|
ImageData *id;
|
||||||
int flash_duration = (int) (waitbeforepicture * 1000);
|
int flash_duration = (int)(CCstatus.WaitBeforePicture * 1000);
|
||||||
Camera.CaptureToBasisImage(zw, flash_duration);
|
Camera.CaptureToBasisImage(zw, flash_duration);
|
||||||
time(&TimeImageTaken);
|
time(&TimeImageTaken);
|
||||||
localtime(&TimeImageTaken);
|
localtime(&TimeImageTaken);
|
||||||
@@ -243,7 +609,7 @@ ImageData* ClassFlowTakeImage::SendRawImage()
|
|||||||
return id;
|
return id;
|
||||||
}
|
}
|
||||||
|
|
||||||
time_t ClassFlowTakeImage::getTimeImageTaken()
|
time_t ClassFlowTakeImage::getTimeImageTaken(void)
|
||||||
{
|
{
|
||||||
return TimeImageTaken;
|
return TimeImageTaken;
|
||||||
}
|
}
|
||||||
@@ -252,4 +618,3 @@ ClassFlowTakeImage::~ClassFlowTakeImage(void)
|
|||||||
{
|
{
|
||||||
delete rawImage;
|
delete rawImage;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -9,47 +9,32 @@
|
|||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
class ClassFlowTakeImage :
|
class ClassFlowTakeImage : public ClassFlowImage
|
||||||
public ClassFlowImage
|
|
||||||
{
|
{
|
||||||
protected:
|
protected:
|
||||||
float waitbeforepicture;
|
|
||||||
float waitbeforepicture_store;
|
|
||||||
framesize_t ImageSize;
|
|
||||||
bool isImageSize;
|
|
||||||
int ImageQuality;
|
|
||||||
time_t TimeImageTaken;
|
time_t TimeImageTaken;
|
||||||
string namerawimage;
|
string namerawimage;
|
||||||
int image_height, image_width;
|
|
||||||
bool SaveAllFiles;
|
|
||||||
bool FixedExposure;
|
|
||||||
|
|
||||||
|
esp_err_t camera_capture(void);
|
||||||
|
|
||||||
void CopyFile(string input, string output);
|
|
||||||
|
|
||||||
esp_err_t camera_capture();
|
|
||||||
void takePictureWithFlash(int flash_duration);
|
void takePictureWithFlash(int flash_duration);
|
||||||
|
|
||||||
|
|
||||||
void SetInitialParameter(void);
|
void SetInitialParameter(void);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
CImageBasis *rawImage;
|
CImageBasis *rawImage;
|
||||||
|
|
||||||
ClassFlowTakeImage(std::vector<ClassFlow*>* lfc);
|
ClassFlowTakeImage(std::vector<ClassFlow *> *lfc);
|
||||||
|
|
||||||
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
bool ReadParameter(FILE *pfile, string &aktparamgraph);
|
||||||
bool doFlow(string time);
|
bool doFlow(string time);
|
||||||
string getHTMLSingleStep(string host);
|
string getHTMLSingleStep(string host);
|
||||||
time_t getTimeImageTaken();
|
time_t getTimeImageTaken(void);
|
||||||
string name(){return "ClassFlowTakeImage";};
|
string name() { return "ClassFlowTakeImage"; };
|
||||||
|
|
||||||
ImageData* SendRawImage();
|
ImageData *SendRawImage(void);
|
||||||
esp_err_t SendRawJPG(httpd_req_t *req);
|
esp_err_t SendRawJPG(httpd_req_t *req);
|
||||||
|
|
||||||
~ClassFlowTakeImage(void);
|
~ClassFlowTakeImage(void);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#endif // CLASSFFLOWTAKEIMAGE_H
|
||||||
#endif //CLASSFFLOWTAKEIMAGE_H
|
|
||||||
171
code/components/jomjol_flowcontroll/ClassFlowWebhook.cpp
Normal file
171
code/components/jomjol_flowcontroll/ClassFlowWebhook.cpp
Normal file
@@ -0,0 +1,171 @@
|
|||||||
|
#ifdef ENABLE_WEBHOOK
|
||||||
|
#include <sstream>
|
||||||
|
#include "ClassFlowWebhook.h"
|
||||||
|
#include "Helper.h"
|
||||||
|
#include "connect_wlan.h"
|
||||||
|
|
||||||
|
#include "time_sntp.h"
|
||||||
|
#include "interface_webhook.h"
|
||||||
|
|
||||||
|
#include "ClassFlowPostProcessing.h"
|
||||||
|
#include "ClassFlowAlignment.h"
|
||||||
|
#include "esp_log.h"
|
||||||
|
#include "../../include/defines.h"
|
||||||
|
|
||||||
|
#include "ClassLogFile.h"
|
||||||
|
|
||||||
|
#include <time.h>
|
||||||
|
|
||||||
|
static const char* TAG = "WEBHOOK";
|
||||||
|
|
||||||
|
void ClassFlowWebhook::SetInitialParameter(void)
|
||||||
|
{
|
||||||
|
uri = "";
|
||||||
|
flowpostprocessing = NULL;
|
||||||
|
flowAlignment = NULL;
|
||||||
|
previousElement = NULL;
|
||||||
|
ListFlowControll = NULL;
|
||||||
|
disabled = false;
|
||||||
|
WebhookEnable = false;
|
||||||
|
WebhookUploadImg = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
ClassFlowWebhook::ClassFlowWebhook()
|
||||||
|
{
|
||||||
|
SetInitialParameter();
|
||||||
|
}
|
||||||
|
|
||||||
|
ClassFlowWebhook::ClassFlowWebhook(std::vector<ClassFlow*>* lfc)
|
||||||
|
{
|
||||||
|
SetInitialParameter();
|
||||||
|
|
||||||
|
ListFlowControll = lfc;
|
||||||
|
for (int i = 0; i < ListFlowControll->size(); ++i)
|
||||||
|
{
|
||||||
|
if (((*ListFlowControll)[i])->name().compare("ClassFlowPostProcessing") == 0)
|
||||||
|
{
|
||||||
|
flowpostprocessing = (ClassFlowPostProcessing*) (*ListFlowControll)[i];
|
||||||
|
}
|
||||||
|
if (((*ListFlowControll)[i])->name().compare("ClassFlowAlignment") == 0)
|
||||||
|
{
|
||||||
|
flowAlignment = (ClassFlowAlignment*) (*ListFlowControll)[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ClassFlowWebhook::ClassFlowWebhook(std::vector<ClassFlow*>* lfc, ClassFlow *_prev)
|
||||||
|
{
|
||||||
|
SetInitialParameter();
|
||||||
|
|
||||||
|
previousElement = _prev;
|
||||||
|
ListFlowControll = lfc;
|
||||||
|
|
||||||
|
for (int i = 0; i < ListFlowControll->size(); ++i)
|
||||||
|
{
|
||||||
|
if (((*ListFlowControll)[i])->name().compare("ClassFlowPostProcessing") == 0)
|
||||||
|
{
|
||||||
|
flowpostprocessing = (ClassFlowPostProcessing*) (*ListFlowControll)[i];
|
||||||
|
}
|
||||||
|
if (((*ListFlowControll)[i])->name().compare("ClassFlowAlignment") == 0)
|
||||||
|
{
|
||||||
|
flowAlignment = (ClassFlowAlignment*) (*ListFlowControll)[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool ClassFlowWebhook::ReadParameter(FILE* pfile, string& aktparamgraph)
|
||||||
|
{
|
||||||
|
std::vector<string> splitted;
|
||||||
|
|
||||||
|
aktparamgraph = trim(aktparamgraph);
|
||||||
|
printf("akt param: %s\n", aktparamgraph.c_str());
|
||||||
|
|
||||||
|
if (aktparamgraph.size() == 0)
|
||||||
|
if (!this->GetNextParagraph(pfile, aktparamgraph))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (toUpper(aktparamgraph).compare("[WEBHOOK]") != 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
while (this->getNextLine(pfile, &aktparamgraph) && !this->isNewParagraph(aktparamgraph))
|
||||||
|
{
|
||||||
|
ESP_LOGD(TAG, "while loop reading line: %s", aktparamgraph.c_str());
|
||||||
|
splitted = ZerlegeZeile(aktparamgraph);
|
||||||
|
std::string _param = GetParameterName(splitted[0]);
|
||||||
|
|
||||||
|
if ((toUpper(_param) == "URI") && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
this->uri = splitted[1];
|
||||||
|
}
|
||||||
|
if (((toUpper(_param) == "APIKEY")) && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
this->apikey = splitted[1];
|
||||||
|
}
|
||||||
|
if (((toUpper(_param) == "UPLOADIMG")) && (splitted.size() > 1))
|
||||||
|
{
|
||||||
|
if (toUpper(splitted[1]) == "1")
|
||||||
|
{
|
||||||
|
this->WebhookUploadImg = 1;
|
||||||
|
} else if (toUpper(splitted[1]) == "2")
|
||||||
|
{
|
||||||
|
this->WebhookUploadImg = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
WebhookInit(uri,apikey);
|
||||||
|
WebhookEnable = true;
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Webhook Enabled for Uri " + uri);
|
||||||
|
|
||||||
|
printf("uri: %s\n", uri.c_str());
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void ClassFlowWebhook::handleMeasurement(string _decsep, string _value)
|
||||||
|
{
|
||||||
|
string _digit, _decpos;
|
||||||
|
int _pospunkt = _decsep.find_first_of(".");
|
||||||
|
// ESP_LOGD(TAG, "Name: %s, Pospunkt: %d", _decsep.c_str(), _pospunkt);
|
||||||
|
if (_pospunkt > -1)
|
||||||
|
_digit = _decsep.substr(0, _pospunkt);
|
||||||
|
else
|
||||||
|
_digit = "default";
|
||||||
|
for (int j = 0; j < flowpostprocessing->NUMBERS.size(); ++j)
|
||||||
|
{
|
||||||
|
if (_digit == "default") // Set to default first (if nothing else is set)
|
||||||
|
{
|
||||||
|
flowpostprocessing->NUMBERS[j]->MeasurementV2 = _value;
|
||||||
|
}
|
||||||
|
if (flowpostprocessing->NUMBERS[j]->name == _digit)
|
||||||
|
{
|
||||||
|
flowpostprocessing->NUMBERS[j]->MeasurementV2 = _value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool ClassFlowWebhook::doFlow(string zwtime)
|
||||||
|
{
|
||||||
|
if (!WebhookEnable)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
if (flowpostprocessing)
|
||||||
|
{
|
||||||
|
printf("vor sende WebHook");
|
||||||
|
bool numbersWithError = WebhookPublish(flowpostprocessing->GetNumbers());
|
||||||
|
|
||||||
|
#ifdef ALGROI_LOAD_FROM_MEM_AS_JPG
|
||||||
|
if ((WebhookUploadImg == 1 || (WebhookUploadImg != 0 && numbersWithError)) && flowAlignment && flowAlignment->AlgROI) {
|
||||||
|
WebhookUploadPic(flowAlignment->AlgROI);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
#endif //ENABLE_WEBHOOK
|
||||||
43
code/components/jomjol_flowcontroll/ClassFlowWebhook.h
Normal file
43
code/components/jomjol_flowcontroll/ClassFlowWebhook.h
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
#ifdef ENABLE_WEBHOOK
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifndef CLASSFWEBHOOK_H
|
||||||
|
#define CLASSFWEBHOOK_H
|
||||||
|
|
||||||
|
#include "ClassFlow.h"
|
||||||
|
|
||||||
|
#include "ClassFlowPostProcessing.h"
|
||||||
|
#include "ClassFlowAlignment.h"
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
class ClassFlowWebhook :
|
||||||
|
public ClassFlow
|
||||||
|
{
|
||||||
|
protected:
|
||||||
|
std::string uri, apikey;
|
||||||
|
ClassFlowPostProcessing* flowpostprocessing;
|
||||||
|
ClassFlowAlignment* flowAlignment;
|
||||||
|
|
||||||
|
bool WebhookEnable;
|
||||||
|
int WebhookUploadImg;
|
||||||
|
|
||||||
|
void SetInitialParameter(void);
|
||||||
|
|
||||||
|
void handleFieldname(string _decsep, string _value);
|
||||||
|
void handleMeasurement(string _decsep, string _value);
|
||||||
|
|
||||||
|
|
||||||
|
public:
|
||||||
|
ClassFlowWebhook();
|
||||||
|
ClassFlowWebhook(std::vector<ClassFlow*>* lfc);
|
||||||
|
ClassFlowWebhook(std::vector<ClassFlow*>* lfc, ClassFlow *_prev);
|
||||||
|
|
||||||
|
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
||||||
|
bool doFlow(string time);
|
||||||
|
string name(){return "ClassFlowWebhook";};
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif //CLASSFWEBHOOK_H
|
||||||
|
#endif //ENABLE_WEBHOOK
|
||||||
@@ -1,91 +0,0 @@
|
|||||||
#include <sstream>
|
|
||||||
#include "ClassFlowWriteList.h"
|
|
||||||
#include "Helper.h"
|
|
||||||
|
|
||||||
#include "time_sntp.h"
|
|
||||||
#include "../../include/defines.h"
|
|
||||||
|
|
||||||
#include <time.h>
|
|
||||||
|
|
||||||
void ClassFlowWriteList::SetInitialParameter(void)
|
|
||||||
{
|
|
||||||
flowpostprocessing = NULL;
|
|
||||||
previousElement = NULL;
|
|
||||||
ListFlowControll = NULL;
|
|
||||||
disabled = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassFlowWriteList::ClassFlowWriteList()
|
|
||||||
{
|
|
||||||
SetInitialParameter();
|
|
||||||
}
|
|
||||||
|
|
||||||
ClassFlowWriteList::ClassFlowWriteList(std::vector<ClassFlow*>* lfc)
|
|
||||||
{
|
|
||||||
SetInitialParameter();
|
|
||||||
|
|
||||||
ListFlowControll = lfc;
|
|
||||||
for (int i = 0; i < ListFlowControll->size(); ++i)
|
|
||||||
{
|
|
||||||
if (((*ListFlowControll)[i])->name().compare("ClassFlowPostProcessing") == 0)
|
|
||||||
{
|
|
||||||
flowpostprocessing = (ClassFlowPostProcessing*) (*ListFlowControll)[i];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool ClassFlowWriteList::ReadParameter(FILE* pfile, string& aktparamgraph)
|
|
||||||
{
|
|
||||||
std::vector<string> splitted;
|
|
||||||
|
|
||||||
aktparamgraph = trim(aktparamgraph);
|
|
||||||
|
|
||||||
if (aktparamgraph.size() == 0)
|
|
||||||
if (!this->GetNextParagraph(pfile, aktparamgraph))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (toUpper(aktparamgraph).compare("[MQTT]") != 0)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
while (this->getNextLine(pfile, &aktparamgraph) && !this->isNewParagraph(aktparamgraph))
|
|
||||||
{
|
|
||||||
splitted = ZerlegeZeile(aktparamgraph);
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
bool ClassFlowWriteList::doFlow(string zwtime)
|
|
||||||
{
|
|
||||||
std::string line = "";
|
|
||||||
|
|
||||||
std::string result;
|
|
||||||
std::string resulterror = "";
|
|
||||||
std::string resultraw = "";
|
|
||||||
std::string resultrate = "";
|
|
||||||
std::string resulttimestamp = "";
|
|
||||||
string zw = "";
|
|
||||||
string namenumber = "";
|
|
||||||
|
|
||||||
if (flowpostprocessing)
|
|
||||||
{
|
|
||||||
std::vector<NumberPost*>* NUMBERS = flowpostprocessing->GetNumbers();
|
|
||||||
|
|
||||||
for (int i = 0; i < (*NUMBERS).size(); ++i)
|
|
||||||
{
|
|
||||||
result = (*NUMBERS)[i]->ReturnValue;
|
|
||||||
resultraw = (*NUMBERS)[i]->ReturnRawValue;
|
|
||||||
resulterror = (*NUMBERS)[i]->ErrorMessageText;
|
|
||||||
resultrate = (*NUMBERS)[i]->ReturnRateValue;
|
|
||||||
resulttimestamp = (*NUMBERS)[i]->timeStamp;
|
|
||||||
|
|
||||||
line = line + resulttimestamp + "\t" + resultraw + "\t" + result + "\t" + resultraw + "\t" + resultrate + "\t" + resulttimestamp + "\t";
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#ifndef CLASSFFLOWPWRITELIST_H
|
|
||||||
#define CLASSFFLOWPWRITELIST_H
|
|
||||||
|
|
||||||
#include "ClassFlow.h"
|
|
||||||
#include "ClassFlowPostProcessing.h"
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
class ClassFlowWriteList :
|
|
||||||
public ClassFlow
|
|
||||||
{
|
|
||||||
protected:
|
|
||||||
ClassFlowPostProcessing* flowpostprocessing;
|
|
||||||
void SetInitialParameter(void);
|
|
||||||
|
|
||||||
public:
|
|
||||||
ClassFlowWriteList();
|
|
||||||
ClassFlowWriteList(std::vector<ClassFlow*>* lfc);
|
|
||||||
|
|
||||||
bool ReadParameter(FILE* pfile, string& aktparamgraph);
|
|
||||||
bool doFlow(string time);
|
|
||||||
string name(){return "ClassFlowWriteList";};
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif //CLASSFFLOWPWRITELIST_H
|
|
||||||
1892
code/components/jomjol_flowcontroll/MainFlowControl.cpp
Normal file
1892
code/components/jomjol_flowcontroll/MainFlowControl.cpp
Normal file
File diff suppressed because it is too large
Load Diff
91
code/components/jomjol_flowcontroll/MainFlowControl.h
Normal file
91
code/components/jomjol_flowcontroll/MainFlowControl.h
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifndef MAINFLOWCONTROL_H
|
||||||
|
#define MAINFLOWCONTROL_H
|
||||||
|
|
||||||
|
#include <esp_log.h>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include <esp_http_server.h>
|
||||||
|
#include "CImageBasis.h"
|
||||||
|
#include "ClassFlowControll.h"
|
||||||
|
#include "openmetrics.h"
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
uint16_t CamSensor_id;
|
||||||
|
|
||||||
|
framesize_t ImageFrameSize = FRAMESIZE_VGA; // 0 - 10
|
||||||
|
gainceiling_t ImageGainceiling; // Image gain (GAINCEILING_x2, x4, x8, x16, x32, x64 or x128)
|
||||||
|
|
||||||
|
int ImageQuality; // 0 - 63
|
||||||
|
int ImageBrightness; // (-2 to 2) - set brightness
|
||||||
|
int ImageContrast; //-2 - 2
|
||||||
|
int ImageSaturation; //-2 - 2
|
||||||
|
int ImageSharpness; //-2 - 2
|
||||||
|
bool ImageAutoSharpness;
|
||||||
|
int ImageSpecialEffect; // 0 - 6
|
||||||
|
int ImageWbMode; // 0 to 4 - if awb_gain enabled (0 - Auto, 1 - Sunny, 2 - Cloudy, 3 - Office, 4 - Home)
|
||||||
|
int ImageAwb; // white balance enable (0 or 1)
|
||||||
|
int ImageAwbGain; // Auto White Balance enable (0 or 1)
|
||||||
|
int ImageAec; // auto exposure off (1 or 0)
|
||||||
|
int ImageAec2; // automatic exposure sensor (0 or 1)
|
||||||
|
int ImageAeLevel; // auto exposure levels (-2 to 2)
|
||||||
|
int ImageAecValue; // set exposure manually (0-1200)
|
||||||
|
int ImageAgc; // auto gain off (1 or 0)
|
||||||
|
int ImageAgcGain; // set gain manually (0 - 30)
|
||||||
|
int ImageBpc; // black pixel correction
|
||||||
|
int ImageWpc; // white pixel correction
|
||||||
|
int ImageRawGma; // (1 or 0)
|
||||||
|
int ImageLenc; // lens correction (1 or 0)
|
||||||
|
int ImageHmirror; // (0 or 1) flip horizontally
|
||||||
|
int ImageVflip; // Invert image (0 or 1)
|
||||||
|
int ImageDcw; // downsize enable (1 or 0)
|
||||||
|
|
||||||
|
int ImageDenoiseLevel; // The OV2640 does not support it, OV3660 and OV5640 (0 to 8)
|
||||||
|
|
||||||
|
int ImageWidth;
|
||||||
|
int ImageHeight;
|
||||||
|
|
||||||
|
int ImageLedIntensity;
|
||||||
|
|
||||||
|
bool ImageZoomEnabled;
|
||||||
|
int ImageZoomOffsetX;
|
||||||
|
int ImageZoomOffsetY;
|
||||||
|
int ImageZoomSize;
|
||||||
|
|
||||||
|
int WaitBeforePicture;
|
||||||
|
bool isImageSize;
|
||||||
|
|
||||||
|
bool CameraInitSuccessful;
|
||||||
|
bool changedCameraSettings;
|
||||||
|
bool DemoMode;
|
||||||
|
bool SaveAllFiles;
|
||||||
|
} camera_flow_config_temp_t;
|
||||||
|
|
||||||
|
extern camera_flow_config_temp_t CFstatus;
|
||||||
|
extern ClassFlowControll flowctrl;
|
||||||
|
|
||||||
|
esp_err_t setCCstatusToCFstatus(void); // CCstatus >>> CFstatus
|
||||||
|
esp_err_t setCFstatusToCCstatus(void); // CFstatus >>> CCstatus
|
||||||
|
esp_err_t setCFstatusToCam(void); // CFstatus >>> Kamera
|
||||||
|
|
||||||
|
void register_server_main_flow_task_uri(httpd_handle_t server);
|
||||||
|
|
||||||
|
void CheckIsPlannedReboot(void);
|
||||||
|
bool getIsPlannedReboot(void);
|
||||||
|
|
||||||
|
void InitializeFlowTask(void);
|
||||||
|
void DeleteMainFlowTask(void);
|
||||||
|
bool isSetupModusActive(void);
|
||||||
|
|
||||||
|
int getCountFlowRounds(void);
|
||||||
|
|
||||||
|
#ifdef ENABLE_MQTT
|
||||||
|
esp_err_t MQTTCtrlFlowStart(std::string _topic);
|
||||||
|
#endif // ENABLE_MQTT
|
||||||
|
|
||||||
|
esp_err_t GetRawJPG(httpd_req_t *req);
|
||||||
|
esp_err_t GetJPG(std::string _filename, httpd_req_t *req);
|
||||||
|
|
||||||
|
#endif // MAINFLOWCONTROL_H
|
||||||
@@ -2,6 +2,6 @@ FILE(GLOB_RECURSE app_sources ${CMAKE_CURRENT_SOURCE_DIR}/*.*)
|
|||||||
|
|
||||||
idf_component_register(SRCS ${app_sources}
|
idf_component_register(SRCS ${app_sources}
|
||||||
INCLUDE_DIRS "."
|
INCLUDE_DIRS "."
|
||||||
REQUIRES tflite-lib jomjol_logfile fatfs sdmmc)
|
REQUIRES esp_timer esp-tflite-micro jomjol_logfile fatfs sdmmc vfs)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -6,6 +6,7 @@
|
|||||||
#include <string>
|
#include <string>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "sdmmc_cmd.h"
|
#include "sdmmc_cmd.h"
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
@@ -15,11 +16,12 @@ std::size_t file_size(const std::string& file_name);
|
|||||||
void FindReplace(std::string& line, std::string& oldString, std::string& newString);
|
void FindReplace(std::string& line, std::string& oldString, std::string& newString);
|
||||||
|
|
||||||
bool CopyFile(string input, string output);
|
bool CopyFile(string input, string output);
|
||||||
bool DeleteFile(string fn);
|
bool DeleteFile(string filename);
|
||||||
bool RenameFile(string from, string to);
|
bool RenameFile(string from, string to);
|
||||||
|
bool RenameFolder(string from, string to);
|
||||||
bool MakeDir(std::string _what);
|
bool MakeDir(std::string _what);
|
||||||
bool FileExists(string filename);
|
bool FileExists(string filename);
|
||||||
|
bool FolderExists(string foldername);
|
||||||
|
|
||||||
string RundeOutput(double _in, int _anzNachkomma);
|
string RundeOutput(double _in, int _anzNachkomma);
|
||||||
|
|
||||||
@@ -32,7 +34,6 @@ string getFileType(string filename);
|
|||||||
string getFileFullFileName(string filename);
|
string getFileFullFileName(string filename);
|
||||||
string getDirectory(string filename);
|
string getDirectory(string filename);
|
||||||
|
|
||||||
|
|
||||||
int mkdir_r(const char *dir, const mode_t mode);
|
int mkdir_r(const char *dir, const mode_t mode);
|
||||||
int removeFolder(const char* folderPath, const char* logTag);
|
int removeFolder(const char* folderPath, const char* logTag);
|
||||||
|
|
||||||
@@ -41,6 +42,7 @@ string toUpper(string in);
|
|||||||
|
|
||||||
float temperatureRead();
|
float temperatureRead();
|
||||||
|
|
||||||
|
std::string intToHexString(int _valueInt);
|
||||||
time_t addDays(time_t startTime, int days);
|
time_t addDays(time_t startTime, int days);
|
||||||
|
|
||||||
void memCopyGen(uint8_t* _source, uint8_t* _target, int _size);
|
void memCopyGen(uint8_t* _source, uint8_t* _target, int _size);
|
||||||
@@ -67,7 +69,6 @@ string getSDCardSectorSize();
|
|||||||
|
|
||||||
string getMac(void);
|
string getMac(void);
|
||||||
|
|
||||||
|
|
||||||
/* Error bit fields
|
/* Error bit fields
|
||||||
One bit per error
|
One bit per error
|
||||||
Make sure it matches https://jomjol.github.io/AI-on-the-edge-device-docs/Error-Codes */
|
Make sure it matches https://jomjol.github.io/AI-on-the-edge-device-docs/Error-Codes */
|
||||||
@@ -76,6 +77,8 @@ enum SystemStatusFlag_t { // One bit per error
|
|||||||
SYSTEM_STATUS_PSRAM_BAD = 1 << 0, // 1, Critical Error
|
SYSTEM_STATUS_PSRAM_BAD = 1 << 0, // 1, Critical Error
|
||||||
SYSTEM_STATUS_HEAP_TOO_SMALL = 1 << 1, // 2, Critical Error
|
SYSTEM_STATUS_HEAP_TOO_SMALL = 1 << 1, // 2, Critical Error
|
||||||
SYSTEM_STATUS_CAM_BAD = 1 << 2, // 4, Critical Error
|
SYSTEM_STATUS_CAM_BAD = 1 << 2, // 4, Critical Error
|
||||||
|
SYSTEM_STATUS_SDCARD_CHECK_BAD = 1 << 3, // 8, Critical Error
|
||||||
|
SYSTEM_STATUS_FOLDER_CHECK_BAD = 1 << 4, // 16, Critical Error
|
||||||
|
|
||||||
// Second Byte
|
// Second Byte
|
||||||
SYSTEM_STATUS_CAM_FB_BAD = 1 << (0+8), // 8, Flow still might work
|
SYSTEM_STATUS_CAM_FB_BAD = 1 << (0+8), // 8, Flow still might work
|
||||||
@@ -95,4 +98,18 @@ const char* get404(void);
|
|||||||
|
|
||||||
std::string UrlDecode(const std::string& value);
|
std::string UrlDecode(const std::string& value);
|
||||||
|
|
||||||
|
void replaceAll(std::string& s, const std::string& toReplace, const std::string& replaceWith);
|
||||||
|
bool replaceString(std::string& s, std::string const& toReplace, std::string const& replaceWith);
|
||||||
|
bool replaceString(std::string& s, std::string const& toReplace, std::string const& replaceWith, bool logIt);
|
||||||
|
bool isInString(std::string& s, std::string const& toFind);
|
||||||
|
|
||||||
|
bool isStringNumeric(std::string &input);
|
||||||
|
bool isStringAlphabetic(std::string &input);
|
||||||
|
bool isStringAlphanumeric(std::string &input);
|
||||||
|
bool alphanumericToBoolean(std::string &input);
|
||||||
|
|
||||||
|
int clipInt(int input, int high, int low);
|
||||||
|
bool numericStrToBool(std::string input);
|
||||||
|
bool stringToBoolean(std::string input);
|
||||||
|
|
||||||
#endif //HELPER_H
|
#endif //HELPER_H
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
#ifdef DEBUG_ENABLE_SYSINFO
|
#ifdef DEBUG_ENABLE_SYSINFO
|
||||||
|
|
||||||
#include "esp_sys.h"
|
#include "esp_sys.h"
|
||||||
|
#include "esp_chip_info.h"
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
@@ -121,7 +122,7 @@ std::string get_device_info()
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifdef USE_HIMEM_IF_AVAILABLE
|
#ifdef USE_HIMEM_IF_AVAILABLE
|
||||||
sprintf(aMsgBuf,"spiram size %u\n", esp_spiram_get_size());
|
sprintf(aMsgBuf,"spiram size %u\n", esp_psram_get_size());
|
||||||
espInfoResultStr += std::string(aMsgBuf);
|
espInfoResultStr += std::string(aMsgBuf);
|
||||||
sprintf(aMsgBuf,"himem free %u\n", esp_himem_get_free_size());
|
sprintf(aMsgBuf,"himem free %u\n", esp_himem_get_free_size());
|
||||||
espInfoResultStr += std::string(aMsgBuf);
|
espInfoResultStr += std::string(aMsgBuf);
|
||||||
|
|||||||
@@ -16,9 +16,9 @@
|
|||||||
#include <esp_spi_flash.h>
|
#include <esp_spi_flash.h>
|
||||||
#include <esp_heap_caps.h>
|
#include <esp_heap_caps.h>
|
||||||
|
|
||||||
// for esp_spiram_get_size
|
// for esp_psram_get_size
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#include <esp32/spiram.h>
|
#include "esp_psram.h"
|
||||||
#ifdef USE_HIMEM_IF_AVAILABLE
|
#ifdef USE_HIMEM_IF_AVAILABLE
|
||||||
#include <esp32/himem.h>
|
#include <esp32/himem.h>
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
211
code/components/jomjol_helper/psram.cpp
Normal file
211
code/components/jomjol_helper/psram.cpp
Normal file
@@ -0,0 +1,211 @@
|
|||||||
|
#include "ClassLogFile.h"
|
||||||
|
#include "../../include/defines.h"
|
||||||
|
#include "psram.h"
|
||||||
|
|
||||||
|
static const char* TAG = "PSRAM";
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
|
||||||
|
void *shared_region = NULL;
|
||||||
|
uint32_t allocatedBytesForSTBI = 0;
|
||||||
|
std::string sharedMemoryInUseFor = "";
|
||||||
|
|
||||||
|
|
||||||
|
/** Reserve a large block in the PSRAM which will be shared between the different steps.
|
||||||
|
* Each step uses it differently but only wiuthin itself. */
|
||||||
|
bool reserve_psram_shared_region(void) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Allocating shared PSRAM region (" +
|
||||||
|
std::to_string(TENSOR_ARENA_SIZE + MAX_MODEL_SIZE) + " bytes)...");
|
||||||
|
shared_region = malloc_psram_heap("Shared PSRAM region", TENSOR_ARENA_SIZE + MAX_MODEL_SIZE,
|
||||||
|
MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT);
|
||||||
|
|
||||||
|
if (shared_region == NULL) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to allocating shared PSRAM region!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************
|
||||||
|
* Memory used in Take Image (STBI)
|
||||||
|
*******************************************************************/
|
||||||
|
bool psram_init_shared_memory_for_take_image_step(void) {
|
||||||
|
if (sharedMemoryInUseFor != "") {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Shared memory in PSRAM already in use for " + sharedMemoryInUseFor + "!");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Init shared memory for step 'Take Image' (STBI buffers)");
|
||||||
|
allocatedBytesForSTBI = 0;
|
||||||
|
sharedMemoryInUseFor = "TakeImage";
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void psram_deinit_shared_memory_for_take_image_step(void) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Deinit shared memory for step 'Take Image' (STBI buffers)");
|
||||||
|
allocatedBytesForSTBI = 0;
|
||||||
|
sharedMemoryInUseFor = "";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void *psram_reserve_shared_stbi_memory(size_t size) {
|
||||||
|
/* Only large buffers should be placed in the shared PSRAM
|
||||||
|
* If we also place all smaller STBI buffers here, we get artefacts for some reasons. */
|
||||||
|
if (size >= 100000) {
|
||||||
|
if ((allocatedBytesForSTBI + size) > TENSOR_ARENA_SIZE + MAX_MODEL_SIZE) { // Check if it still fits in the shared region
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Shared memory in PSRAM too small (STBI) to fit additional " +
|
||||||
|
std::to_string(size) + " bytes! Available: " + std::to_string(TENSOR_ARENA_SIZE + MAX_MODEL_SIZE - allocatedBytesForSTBI) + " bytes!");
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Allocating memory (" + std::to_string(size) + " bytes) for STBI (use shared memory in PSRAM)...");
|
||||||
|
allocatedBytesForSTBI += size;
|
||||||
|
return (uint8_t *)shared_region + allocatedBytesForSTBI - size;
|
||||||
|
}
|
||||||
|
else { // Normal PSRAM
|
||||||
|
return malloc_psram_heap("STBI", size, MALLOC_CAP_SPIRAM);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void *psram_reallocate_shared_stbi_memory(void *ptr, size_t newsize) {
|
||||||
|
char buf[20];
|
||||||
|
sprintf(buf, "%p", ptr);
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "STBI requested realloc for " + std::string(buf) + " but this is currently unsupported!");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void psram_free_shared_stbi_memory(void *p) {
|
||||||
|
if ((p >= shared_region) && (p <= ((uint8_t *)shared_region + allocatedBytesForSTBI))) { // was allocated inside the shared memory
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Part of shared memory used for STBI (PSRAM, part of shared memory) is free again");
|
||||||
|
}
|
||||||
|
else { // Normal PSRAM
|
||||||
|
free_psram_heap("STBI", p);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************
|
||||||
|
* Memory used in Aligning Step
|
||||||
|
* During this step we only use the shared part of the PSRAM
|
||||||
|
* for the tmpImage.
|
||||||
|
*******************************************************************/
|
||||||
|
void *psram_reserve_shared_tmp_image_memory(void) {
|
||||||
|
if (sharedMemoryInUseFor != "") {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Shared memory in PSRAM already in use for " + sharedMemoryInUseFor + "!");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Allocating tmpImage (" + std::to_string(IMAGE_SIZE) + " bytes, use shared memory in PSRAM)...");
|
||||||
|
sharedMemoryInUseFor = "Aligning";
|
||||||
|
return shared_region; // Use 1th part of the shared memory for the tmpImage (only user)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void psram_free_shared_temp_image_memory(void) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Shared memory used for tmpImage (PSRAM, part of shared memory) is free again");
|
||||||
|
sharedMemoryInUseFor = "";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************
|
||||||
|
* Memory used in Digitization Steps
|
||||||
|
* During this step we only use the shared part of the PSRAM for the
|
||||||
|
* Tensor Arena and one of the Models.
|
||||||
|
* The shared memory is large enough for the largest model and the
|
||||||
|
* Tensor Arena. Therefore we do not need to monitor the usage.
|
||||||
|
*******************************************************************/
|
||||||
|
void *psram_get_shared_tensor_arena_memory(void) {
|
||||||
|
if ((sharedMemoryInUseFor == "") || (sharedMemoryInUseFor == "Digitization_Model")) {
|
||||||
|
sharedMemoryInUseFor = "Digitization_Tensor";
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Allocating Tensor Arena (" + std::to_string(TENSOR_ARENA_SIZE) + " bytes, use shared memory in PSRAM)...");
|
||||||
|
return shared_region; // Use 1th part of the shared memory for Tensor
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Shared memory in PSRAM already in use for " + sharedMemoryInUseFor + "!");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void *psram_get_shared_model_memory(void) {
|
||||||
|
if ((sharedMemoryInUseFor == "") || (sharedMemoryInUseFor == "Digitization_Tensor")) {
|
||||||
|
sharedMemoryInUseFor = "Digitization_Model";
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Allocating Model memory (" + std::to_string(MAX_MODEL_SIZE) + " bytes, use shared memory in PSRAM)...");
|
||||||
|
return (uint8_t *)shared_region + TENSOR_ARENA_SIZE; // Use 2nd part of the shared memory (after Tensor Arena) for the model
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Shared memory in PSRAM already in use for " + sharedMemoryInUseFor + "!");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void psram_free_shared_tensor_arena_and_model_memory(void) {
|
||||||
|
sharedMemoryInUseFor = "";
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Shared memory used for Tensor Arena and model (PSRAM, part of shared memory) is free again");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*******************************************************************
|
||||||
|
* General
|
||||||
|
*******************************************************************/
|
||||||
|
void *malloc_psram_heap(std::string name, size_t size, uint32_t caps) {
|
||||||
|
void *ptr;
|
||||||
|
|
||||||
|
ptr = heap_caps_malloc(size, caps);
|
||||||
|
if (ptr != NULL) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Allocated " + to_string(size) + " bytes in PSRAM for '" + name + "'");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to allocate " + to_string(size) + " bytes in PSRAM for '" + name + "'!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void *realloc_psram_heap(std::string name, void *ptr, size_t size, uint32_t caps) {
|
||||||
|
ptr = heap_caps_realloc(ptr, size, caps);
|
||||||
|
if (ptr != NULL) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Reallocated " + to_string(size) + " bytes in PSRAM for '" + name + "'");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to reallocate " + to_string(size) + " bytes in PSRAM for '" + name + "'!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void *calloc_psram_heap(std::string name, size_t n, size_t size, uint32_t caps) {
|
||||||
|
void *ptr;
|
||||||
|
|
||||||
|
ptr = heap_caps_calloc(n, size, caps);
|
||||||
|
if (ptr != NULL) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Allocated " + to_string(size) + " bytes in PSRAM for '" + name + "'");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Failed to allocate " + to_string(size) + " bytes in PSRAM for '" + name + "'!");
|
||||||
|
}
|
||||||
|
|
||||||
|
return ptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void free_psram_heap(std::string name, void *ptr) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "Freeing memory in PSRAM used for '" + name + "'...");
|
||||||
|
heap_caps_free(ptr);
|
||||||
|
}
|
||||||
35
code/components/jomjol_helper/psram.h
Normal file
35
code/components/jomjol_helper/psram.h
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
#pragma once
|
||||||
|
#ifndef PSRAM_h
|
||||||
|
#define PSRAM_h
|
||||||
|
|
||||||
|
#include "esp_heap_caps.h"
|
||||||
|
|
||||||
|
|
||||||
|
bool reserve_psram_shared_region(void);
|
||||||
|
|
||||||
|
|
||||||
|
/* Memory used in Take Image Step */
|
||||||
|
bool psram_init_shared_memory_for_take_image_step(void);
|
||||||
|
void psram_deinit_shared_memory_for_take_image_step(void);
|
||||||
|
void *psram_reserve_shared_stbi_memory(size_t size);
|
||||||
|
void *psram_reallocate_shared_stbi_memory(void *ptr, size_t newsize);
|
||||||
|
void psram_free_shared_stbi_memory(void *p);
|
||||||
|
|
||||||
|
|
||||||
|
/* Memory used in Aligning Step */
|
||||||
|
void *psram_reserve_shared_tmp_image_memory(void);
|
||||||
|
void psram_free_shared_temp_image_memory(void);
|
||||||
|
|
||||||
|
/* Memory used in Digitization Steps */
|
||||||
|
void *psram_get_shared_tensor_arena_memory(void);
|
||||||
|
void *psram_get_shared_model_memory(void);
|
||||||
|
void psram_free_shared_tensor_arena_and_model_memory(void);
|
||||||
|
|
||||||
|
/* General */
|
||||||
|
void *malloc_psram_heap(std::string name, size_t size, uint32_t caps);
|
||||||
|
void *realloc_psram_heap(std::string name, void *ptr, size_t size, uint32_t caps);
|
||||||
|
void *calloc_psram_heap(std::string name, size_t n, size_t size, uint32_t caps);
|
||||||
|
|
||||||
|
void free_psram_heap(std::string name, void *ptr);
|
||||||
|
|
||||||
|
#endif // PSRAM_h
|
||||||
161
code/components/jomjol_helper/sdcard_check.cpp
Normal file
161
code/components/jomjol_helper/sdcard_check.cpp
Normal file
@@ -0,0 +1,161 @@
|
|||||||
|
#include "sdcard_check.h"
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <inttypes.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
|
||||||
|
#include "esp_rom_crc.h"
|
||||||
|
#include "ClassLogFile.h"
|
||||||
|
|
||||||
|
static const char *TAG = "SDCARD";
|
||||||
|
|
||||||
|
int SDCardCheckRW(void)
|
||||||
|
{
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Basic R/W check started...");
|
||||||
|
FILE* pFile = NULL;
|
||||||
|
int iCRCMessage = 0;
|
||||||
|
|
||||||
|
pFile = fopen("/sdcard/sdcheck.txt","w");
|
||||||
|
if (pFile == NULL) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Basic R/W check: (E1) No able to open file to write");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
std::string sMessage = "This message is used for a SD-Card basic check!";
|
||||||
|
iCRCMessage = esp_rom_crc16_le(0, (uint8_t*)sMessage.c_str(), sMessage.length());
|
||||||
|
if (fwrite(sMessage.c_str(), sMessage.length(), 1, pFile) == 0 ) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Basic R/W check: (E2) Not able to write file");
|
||||||
|
fclose(pFile);
|
||||||
|
unlink("/sdcard/sdcheck.txt");
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
fclose(pFile);
|
||||||
|
}
|
||||||
|
|
||||||
|
pFile = fopen("/sdcard/sdcheck.txt","r");
|
||||||
|
if (pFile == NULL) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Basic R/W check: (E3) Not able to open file to read back");
|
||||||
|
unlink("/sdcard/sdcheck.txt");
|
||||||
|
return -3;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
char cReadBuf[50];
|
||||||
|
if (fgets(cReadBuf, sizeof(cReadBuf), pFile) == 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Basic R/W check: (E4) Not able to read file back");
|
||||||
|
fclose(pFile);
|
||||||
|
unlink("/sdcard/sdcheck.txt");
|
||||||
|
return -4;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (esp_rom_crc16_le(0, (uint8_t*)cReadBuf, strlen(cReadBuf)) != iCRCMessage) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Basic R/W check: (E5) Read back, but wrong CRC");
|
||||||
|
fclose(pFile);
|
||||||
|
unlink("/sdcard/sdcheck.txt");
|
||||||
|
return -5;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fclose(pFile);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (unlink("/sdcard/sdcheck.txt") != 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Basic R/W check: (E6) Unable to delete the file");
|
||||||
|
return -6;
|
||||||
|
}
|
||||||
|
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Basic R/W check successful");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SDCardCheckFolderFilePresence()
|
||||||
|
{
|
||||||
|
struct stat sb;
|
||||||
|
bool bRetval = true;
|
||||||
|
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Folder/file presence check started...");
|
||||||
|
/* check if folder exists: config */
|
||||||
|
if (stat("/sdcard/config", &sb) != 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: Folder /config not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if folder exists: html */
|
||||||
|
if (stat("/sdcard/html", &sb) != 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: Folder /html not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if folder exists: firmware */
|
||||||
|
if (stat("/sdcard/firmware", &sb) != 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: Folder /firmware not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if folder exists: img_tmp */
|
||||||
|
if (stat("/sdcard/img_tmp", &sb) != 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: Folder /img_tmp not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if folder exists: log */
|
||||||
|
if (stat("/sdcard/log", &sb) != 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: Folder /log not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if folder exists: demo */
|
||||||
|
if (stat("/sdcard/demo", &sb) != 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: Folder /demo not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if file exists: wlan.ini */
|
||||||
|
if (stat("/sdcard/wlan.ini", &sb) != 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: File /wlan.ini not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if file exists: config.ini */
|
||||||
|
if (stat("/sdcard/config/config.ini", &sb) != 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: File /config/config.ini not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if file exists: index.html */
|
||||||
|
if ((stat("/sdcard/html/index.html", &sb) != 0) && (stat("/sdcard/html/index.html.gz", &sb) != 0)) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: File /html/index.html not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if file exists: ota.html */
|
||||||
|
if ((stat("/sdcard/html/ota_page.html", &sb) != 0) && (stat("/sdcard/html/ota_page.html.gz", &sb) != 0)) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: File /html/ota.html not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if file exists: log.html */
|
||||||
|
if ((stat("/sdcard/html/log.html", &sb) != 0) && (stat("/sdcard/html/log.html.gz", &sb) != 0)) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: File /html/log.html not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if file exists: common.js */
|
||||||
|
if ((stat("/sdcard/html/common.js", &sb) != 0) && (stat("/sdcard/html/common.js.gz", &sb) != 0)) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: File /html/common.js not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* check if file exists: version.txt */
|
||||||
|
if (stat("/sdcard/html/version.txt", &sb) != 0) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_ERROR, TAG, "Folder/file check: File /html/version.txt not found");
|
||||||
|
bRetval = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bRetval) {
|
||||||
|
LogFile.WriteToFile(ESP_LOG_INFO, TAG, "Folder/file presence check successful");
|
||||||
|
}
|
||||||
|
|
||||||
|
return bRetval;
|
||||||
|
}
|
||||||
11
code/components/jomjol_helper/sdcard_check.h
Normal file
11
code/components/jomjol_helper/sdcard_check.h
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifndef COMPONENTS_HELPER_SDCARD_CHECK_H
|
||||||
|
#define COMPONENTS_HELPER_SDCARD_CHECK_H
|
||||||
|
|
||||||
|
#include "../../include/defines.h"
|
||||||
|
|
||||||
|
int SDCardCheckRW(void);
|
||||||
|
bool SDCardCheckFolderFilePresence(void);
|
||||||
|
|
||||||
|
#endif /* COMPONENTS_HELPER_SDCARD_CHECK_H */
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user