-
-
Notifications
You must be signed in to change notification settings - Fork 55
/
http_jpeg.c
101 lines (85 loc) · 2.97 KB
/
http_jpeg.c
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
#include <stdio.h>
#include <stdlib.h>
#include "output.h"
#include "util/http/http.h"
#include "util/opts/log.h"
#include "device/buffer.h"
#include "device/buffer_lock.h"
#define SNAPSHOT_TIMEOUT_MS 3000
#define SNAPSHOT_DEFAULT_DELAY_PARAM 300
#define PART_BOUNDARY "123456789000000000000987654321"
#define CONTENT_TYPE "image/jpeg"
#define CONTENT_LENGTH "Content-Length"
static const char *const STREAM_HEADER = "HTTP/1.0 200 OK\r\n"
"Access-Control-Allow-Origin: *\r\n"
"Connection: close\r\n"
"Content-Type: multipart/x-mixed-replace;boundary=" PART_BOUNDARY "\r\n"
"\r\n"
"--" PART_BOUNDARY "\r\n";
static const char *const STREAM_PART = "Content-Type: " CONTENT_TYPE "\r\n" CONTENT_LENGTH ": %u\r\n\r\n";
static const char *const STREAM_BOUNDARY = "\r\n"
"--" PART_BOUNDARY "\r\n";
typedef struct
{
FILE *stream;
uint64_t start_time_us;
} http_snapshot_t;
int http_snapshot_buf_part(buffer_lock_t *buf_lock, buffer_t *buf, int frame, http_snapshot_t *snapshot)
{
// Ignore frames that are captured
if (buf->captured_time_us < snapshot->start_time_us) {
return 0;
}
fprintf(snapshot->stream, "HTTP/1.1 200 OK\r\n");
fprintf(snapshot->stream, "Content-Type: image/jpeg\r\n");
fprintf(snapshot->stream, "Content-Length: %zu\r\n", buf->used);
fprintf(snapshot->stream, "\r\n");
fwrite(buf->start, buf->used, 1, snapshot->stream);
return 1;
}
void http_snapshot(http_worker_t *worker, FILE *stream)
{
int max_delay_value = SNAPSHOT_DEFAULT_DELAY_PARAM;
// passing the max_delay=0 will ensure that frame is capture at this exact moment
char *max_delay = http_get_param(worker, "max_delay");
if (max_delay) {
max_delay_value = atoi(max_delay);
free(max_delay);
}
http_snapshot_t snapshot = {
.stream = stream,
.start_time_us = get_monotonic_time_us(NULL, NULL) - max_delay_value * 1000
};
int n = buffer_lock_write_loop(&snapshot_lock, 1, SNAPSHOT_TIMEOUT_MS,
(buffer_write_fn)http_snapshot_buf_part, &snapshot);
if (n <= 0) {
http_500(stream, NULL);
fprintf(stream, "No snapshot captured yet.\r\n");
}
}
int http_stream_buf_part(buffer_lock_t *buf_lock, buffer_t *buf, int frame, FILE *stream)
{
if (!frame && !fputs(STREAM_HEADER, stream)) {
return -1;
}
if (!fprintf(stream, STREAM_PART, buf->used)) {
return -1;
}
if (!fwrite(buf->start, buf->used, 1, stream)) {
return -1;
}
if (!fputs(STREAM_BOUNDARY, stream)) {
return -1;
}
return 1;
}
void http_stream(http_worker_t *worker, FILE *stream)
{
int n = buffer_lock_write_loop(&stream_lock, 0, 0, (buffer_write_fn)http_stream_buf_part, stream);
if (n == 0) {
http_500(stream, NULL);
fprintf(stream, "No frames.\n");
} else if (n < 0) {
fprintf(stream, "Interrupted. Received %d frames", -n);
}
}