forked from s60sc/ESP32-CAM_MJPEG2SD
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathmotionDetect.cpp
273 lines (244 loc) · 9.64 KB
/
motionDetect.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
/*
Detect movement in sequential images using background subtraction.
Very small bitmaps are used both to provide image smoothing to reduce spurious motion changes
and to enable rapid processing
The amount of change between images will depend on the frame rate.
A faster frame rate will need a higher sensitivity
When frame size is changed the OV2640 outputs a few glitched frames whilst it
makes the transition. These could be interpreted as spurious motion.
s60sc 2020
*/
#include "appGlobals.h"
using namespace std;
#define RGB888_BYTES 3 // number of bytes per pixel
// motion recording parameters
int detectMotionFrames = 5; // min sequence of changed frames to confirm motion
int detectNightFrames = 10; // frames of sequential darkness to avoid spurious day / night switching
// define region of interest, ie exclude top and bottom of image from movement detection if required
// divide image into detectNumBands horizontal bands, define start and end bands of interest, 1 = top
int detectNumBands = 10;
int detectStartBand = 3;
int detectEndBand = 8; // inclusive
int detectChangeThreshold = 15; // min difference in pixel comparison to indicate a change
uint8_t lightLevel; // Current ambient light level
uint8_t nightSwitch = 20; // initial white level % for night/day switching
float motionVal = 8.0; // initial motion sensitivity setting
static uint8_t* jpgImg = NULL;
static size_t jpgImgSize = 0;
/**********************************************************************************/
static bool jpg2rgb(const uint8_t *src, size_t src_len, uint8_t ** out, jpg_scale_t scale);
bool isNight(uint8_t nightSwitch) {
// check if night time for suspending recording
// or for switching on lamp if enabled
static bool nightTime = false;
static uint16_t nightCnt = 0;
if (nightTime) {
if (lightLevel > nightSwitch) {
// light image
nightCnt--;
// signal day time after given sequence of light frames
if (nightCnt == 0) {
nightTime = false;
LOG_INF("Day time");
}
}
} else {
if (lightLevel < nightSwitch) {
// dark image
nightCnt++;
// signal night time after given sequence of dark frames
if (nightCnt > detectNightFrames) {
nightTime = true;
LOG_INF("Night time");
}
}
}
return nightTime;
}
bool checkMotion(camera_fb_t* fb, bool motionStatus) {
// check difference between current and previous image (subtract background)
// convert image from JPEG to downscaled RGB888 bitmap to 8 bit grayscale
uint32_t dTime = millis();
uint32_t lux = 0;
static uint32_t motionCnt = 0;
uint8_t* rgb_buf = NULL;
uint8_t* jpg_buf = NULL;
size_t jpg_len = 0;
// calculate parameters for sample size
uint8_t scaling = frameData[fsizePtr].scaleFactor;
uint16_t reducer = frameData[fsizePtr].sampleRate;
uint8_t downsize = pow(2, scaling) * reducer;
int sampleWidth = frameData[fsizePtr].frameWidth / downsize;
int sampleHeight = frameData[fsizePtr].frameHeight / downsize;
int num_pixels = sampleWidth * sampleHeight;
if (!jpg2rgb((uint8_t*)fb->buf, fb->len, &rgb_buf, (jpg_scale_t)scaling)) {
LOG_ERR("motionDetect: jpg2rgb() failed");
free(rgb_buf);
rgb_buf = NULL;
return motionStatus;
}
/*
if (reducer > 1)
// further reduce size of bitmap
for (int r=0; r<sampleHeight; r++)
for (int c=0; c<sampleWidth; c++)
rgb_buf[c+(r*sampleWidth)] = rgb_buf[(c+(r*sampleWidth))*reducer];
*/
LOG_DBG("JPEG to greyscale conversion %u bytes in %lums", num_pixels, millis() - dTime);
dTime = millis();
// allocate buffer space on heap
int maxSize = 32*1024; // max size downscaled UXGA 30k
static uint8_t* changeMap = (uint8_t*)ps_malloc(maxSize);
static uint8_t* prev_buf = (uint8_t*)ps_malloc(maxSize);
static uint8_t* _jpgImg = (uint8_t*)ps_malloc(maxSize);
jpgImg = _jpgImg;
// compare each pixel in current frame with previous frame
int changeCount = 0;
// set horizontal region of interest in image
uint16_t startPixel = num_pixels*(detectStartBand-1)/detectNumBands;
uint16_t endPixel = num_pixels*(detectEndBand)/detectNumBands;
int moveThreshold = (endPixel-startPixel) * (11-motionVal)/100; // number of changed pixels that constitute a movement
for (int i=0; i<num_pixels; i++) {
if (abs((int)rgb_buf[i] - (int)prev_buf[i]) > detectChangeThreshold) {
if (i > startPixel && i < endPixel) changeCount++; // number of changed pixels
if (dbgMotion) changeMap[i] = 192; // populate changeMap image with changed pixels in gray
} else if (dbgMotion) changeMap[i] = 255; // set white
lux += rgb_buf[i]; // for calculating light level
}
lightLevel = (lux*100)/(num_pixels*255); // light value as a %
nightTime = isNight(nightSwitch);
memcpy(prev_buf, rgb_buf, num_pixels); // save image for next comparison
// esp32-cam issue #126
if (rgb_buf == NULL) LOG_ERR("Memory leak, heap now: %u, pSRAM now: %u", ESP.getFreeHeap(), ESP.getFreePsram());
free(rgb_buf);
rgb_buf = NULL;
LOG_DBG("Detected %u changes, threshold %u, light level %u, in %lums", changeCount, moveThreshold, lightLevel, millis() - dTime);
dTime = millis();
if (changeCount > moveThreshold) {
LOG_DBG("### Change detected");
motionCnt++; // number of consecutive changes
// need minimum sequence of changes to signal valid movement
if (!motionStatus && motionCnt >= detectMotionFrames) {
LOG_DBG("***** Motion - START");
motionStatus = true; // motion started
if (mqtt_active) {
sprintf(jsonBuff, "{\"MOTION\":\"ON\",\"TIME\":\"%s\"}",esp_log_system_timestamp());
mqttPublish(jsonBuff);
}
#ifdef INCLUDE_WEBSOCKET_SERVER
socketSendToServer("MotionStart");
#endif
}
if (dbgMotion)
// to highlight movement detected in changeMap image, set all gray in region of interest to black
for (int i=0; i<num_pixels; i++)
if (i > startPixel && i < endPixel && changeMap[i] < 255) changeMap[i] = 0;
} else {
// insufficient change
if (motionStatus) {
LOG_DBG("***** Motion - STOP after %u frames", motionCnt);
motionCnt = 0;
motionStatus = false; // motion stopped
if (mqtt_active) {
sprintf(jsonBuff, "{\"MOTION\":\"OFF\",\"TIME\":\"%s\"}", esp_log_system_timestamp());
mqttPublish(jsonBuff);
}
#ifdef INCLUDE_WEBSOCKET_SERVER
socketSendToServer("MotionEnd");
#endif
}
}
if (motionStatus) LOG_DBG("*** Motion - ongoing %u frames", motionCnt);
if (dbgMotion) {
// build jpeg of changeMap for debug streaming
dTime = millis();
if (!fmt2jpg(changeMap, num_pixels, sampleWidth, sampleHeight, PIXFORMAT_GRAYSCALE, 80, &jpg_buf, &jpg_len))
LOG_ERR("motionDetect: fmt2jpg() failed");
// prevent streaming from accessing jpeg while it is being updated
xSemaphoreTake(motionMutex, portMAX_DELAY);
memcpy(jpgImg, jpg_buf, jpg_len);
jpgImgSize = jpg_len;
xSemaphoreGive(motionMutex);
free(jpg_buf);
jpg_buf = NULL;
LOG_DBG("Created changeMap JPEG %d bytes in %lums", jpg_len, millis() - dTime);
}
if (dbgVerbose) checkMemory();
// motionStatus indicates whether motion previously ongoing or not
return nightTime ? false : motionStatus;
}
bool fetchMoveMap(uint8_t **out, size_t *out_len) {
// return change map jpeg for streaming
if (useMotion){
*out = jpgImg;
*out_len = jpgImgSize;
static size_t lastImgLen = 0;
if (lastImgLen != jpgImgSize) {
// image changed
lastImgLen = jpgImgSize;
return true;
} else return false;
}else{
// dummy if motionDetect.cpp not used
*out_len = 0;
return false;
}
}
/************* copied and modified from esp32-camera/to_bmp.c to access jpg_scale_t *****************/
typedef struct {
uint16_t width;
uint16_t height;
uint16_t data_offset;
const uint8_t *input;
uint8_t *output;
} rgb_jpg_decoder;
static bool _rgb_write(void * arg, uint16_t x, uint16_t y, uint16_t w, uint16_t h, uint8_t *data) {
// mpjpeg2sd: mofified to generate 8 bit grayscale
rgb_jpg_decoder * jpeg = (rgb_jpg_decoder *)arg;
if (!data){
if (x == 0 && y == 0) {
// write start
jpeg->width = w;
jpeg->height = h;
// if output is null, this is BMP
if (!jpeg->output) {
jpeg->output = (uint8_t*)ps_malloc((w*h)+jpeg->data_offset);
if (!jpeg->output) return false;
}
}
return true;
}
size_t jw = jpeg->width*RGB888_BYTES;
size_t t = y * jw;
size_t b = t + (h * jw);
size_t l = x * RGB888_BYTES;
uint8_t *out = jpeg->output+jpeg->data_offset;
uint8_t *o = out;
size_t iy, ix;
w *= RGB888_BYTES;
for (iy=t; iy<b; iy+=jw) {
o = out+(iy+l)/RGB888_BYTES;
for (ix=0; ix<w; ix+=RGB888_BYTES) {
uint16_t grayscale = (data[ix+2]+data[ix+1]+data[ix])/RGB888_BYTES;
o[ix/RGB888_BYTES] = (uint8_t)grayscale;
}
data+=w;
}
return true;
}
static uint32_t _jpg_read(void * arg, size_t index, uint8_t *buf, size_t len) {
rgb_jpg_decoder * jpeg = (rgb_jpg_decoder *)arg;
if (buf) memcpy(buf, jpeg->input + index, len);
return len;
}
static bool jpg2rgb(const uint8_t *src, size_t src_len, uint8_t **out, jpg_scale_t scale) {
rgb_jpg_decoder jpeg;
jpeg.width = 0;
jpeg.height = 0;
jpeg.input = src;
jpeg.output = NULL;
jpeg.data_offset = 0;
esp_err_t res = esp_jpg_decode(src_len, scale, _jpg_read, _rgb_write, (void*)&jpeg);
*out = jpeg.output;
return (res == ESP_OK) ? true : false;
}