mirror of
https://github.com/pikvm/ustreamer.git
synced 2026-02-27 04:06:30 +00:00
190 lines
5.9 KiB
C
190 lines
5.9 KiB
C
/*****************************************************************************
|
|
# #
|
|
# uStreamer - Lightweight and fast MJPG-HTTP streamer. #
|
|
# #
|
|
# Copyright (C) 2018-2021 Maxim Devaev <mdevaev@gmail.com> #
|
|
# #
|
|
# This program is free software: you can redistribute it and/or modify #
|
|
# it under the terms of the GNU General Public License as published by #
|
|
# the Free Software Foundation, either version 3 of the License, or #
|
|
# (at your option) any later version. #
|
|
# #
|
|
# This program is distributed in the hope that it will be useful, #
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of #
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #
|
|
# GNU General Public License for more details. #
|
|
# #
|
|
# You should have received a copy of the GNU General Public License #
|
|
# along with this program. If not, see <https://www.gnu.org/licenses/>. #
|
|
# #
|
|
*****************************************************************************/
|
|
|
|
|
|
#include "memsink.h"
|
|
|
|
|
|
memsink_s *memsink_init(
|
|
const char *name, const char *obj, bool server,
|
|
mode_t mode, bool rm, unsigned client_ttl, unsigned timeout) {
|
|
|
|
memsink_s *sink;
|
|
A_CALLOC(sink, 1);
|
|
sink->name = name;
|
|
sink->obj = obj;
|
|
sink->server = server;
|
|
sink->rm = rm;
|
|
sink->client_ttl = client_ttl;
|
|
sink->timeout = timeout;
|
|
sink->fd = -1;
|
|
sink->mem = MAP_FAILED;
|
|
|
|
LOG_INFO("Using %s-sink: %s", name, obj);
|
|
|
|
mode_t mask = umask(0);
|
|
sink->fd = shm_open(sink->obj, (server ? O_RDWR | O_CREAT : O_RDWR), mode);
|
|
umask(mask);
|
|
if (sink->fd == -1) {
|
|
umask(mask);
|
|
LOG_PERROR("%s-sink: Can't open shared memory", name);
|
|
goto error;
|
|
}
|
|
|
|
if (sink->server && ftruncate(sink->fd, sizeof(memsink_shared_s)) < 0) {
|
|
LOG_PERROR("%s-sink: Can't truncate shared memory", name);
|
|
goto error;
|
|
}
|
|
|
|
if ((sink->mem = memsink_shared_map(sink->fd)) == NULL) {
|
|
LOG_PERROR("%s-sink: Can't mmap shared memory", name);
|
|
goto error;
|
|
}
|
|
|
|
return sink;
|
|
|
|
error:
|
|
memsink_destroy(sink);
|
|
return NULL;
|
|
}
|
|
|
|
void memsink_destroy(memsink_s *sink) {
|
|
if (sink->mem != MAP_FAILED) {
|
|
if (memsink_shared_unmap(sink->mem) < 0) {
|
|
LOG_PERROR("%s-sink: Can't unmap shared memory", sink->name);
|
|
}
|
|
}
|
|
if (sink->fd >= 0) {
|
|
if (close(sink->fd) < 0) {
|
|
LOG_PERROR("%s-sink: Can't close shared memory fd", sink->name);
|
|
}
|
|
if (sink->rm && shm_unlink(sink->obj) < 0) {
|
|
if (errno != ENOENT) {
|
|
LOG_PERROR("%s-sink: Can't remove shared memory", sink->name);
|
|
}
|
|
}
|
|
}
|
|
free(sink);
|
|
}
|
|
|
|
bool memsink_server_check(memsink_s *sink, const frame_s *frame) {
|
|
// Возвращает true, если есть клиенты ИЛИ изменились метаданные
|
|
|
|
assert(sink->server);
|
|
|
|
if (flock(sink->fd, LOCK_EX | LOCK_NB) < 0) {
|
|
if (errno == EWOULDBLOCK) {
|
|
sink->has_clients = true;
|
|
return true;
|
|
}
|
|
LOG_PERROR("%s-sink: Can't lock memory", sink->name);
|
|
return false;
|
|
}
|
|
|
|
sink->has_clients = (sink->mem->last_client_ts + sink->client_ttl > get_now_monotonic());
|
|
|
|
bool retval = (sink->has_clients || !FRAME_COMPARE_META_USED_NOTS(sink->mem, frame));
|
|
|
|
if (flock(sink->fd, LOCK_UN) < 0) {
|
|
LOG_PERROR("%s-sink: Can't unlock memory", sink->name);
|
|
return false;
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
int memsink_server_put(memsink_s *sink, const frame_s *frame) {
|
|
assert(sink->server);
|
|
|
|
const long double now = get_now_monotonic();
|
|
|
|
if (frame->used > MEMSINK_MAX_DATA) {
|
|
LOG_ERROR("%s-sink: Can't put frame: is too big (%zu > %zu)",
|
|
sink->name, frame->used, MEMSINK_MAX_DATA);
|
|
return 0; // -2
|
|
}
|
|
|
|
if (flock_timedwait_monotonic(sink->fd, 1) == 0) {
|
|
LOG_VERBOSE("%s-sink: >>>>> Exposing new frame ...", sink->name);
|
|
|
|
sink->last_id = get_now_id();
|
|
sink->mem->id = sink->last_id;
|
|
|
|
memcpy(sink->mem->data, frame->data, frame->used);
|
|
sink->mem->used = frame->used;
|
|
FRAME_COPY_META(frame, sink->mem);
|
|
|
|
sink->mem->magic = MEMSINK_MAGIC;
|
|
sink->mem->version = MEMSINK_VERSION;
|
|
sink->has_clients = (sink->mem->last_client_ts + sink->client_ttl > get_now_monotonic());
|
|
|
|
if (flock(sink->fd, LOCK_UN) < 0) {
|
|
LOG_PERROR("%s-sink: Can't unlock memory", sink->name);
|
|
return -1;
|
|
}
|
|
LOG_VERBOSE("%s-sink: Exposed new frame; full exposition time = %.3Lf",
|
|
sink->name, get_now_monotonic() - now);
|
|
|
|
} else if (errno == EWOULDBLOCK) {
|
|
LOG_VERBOSE("%s-sink: ===== Shared memory is busy now; frame skipped", sink->name);
|
|
|
|
} else {
|
|
LOG_PERROR("%s-sink: Can't lock memory", sink->name);
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int memsink_client_get(memsink_s *sink, frame_s *frame) { // cppcheck-suppress unusedFunction
|
|
assert(!sink->server); // Client only
|
|
|
|
if (flock_timedwait_monotonic(sink->fd, sink->timeout) < 0) {
|
|
if (errno == EWOULDBLOCK) {
|
|
return -2;
|
|
}
|
|
LOG_PERROR("%s-sink: Can't lock memory", sink->name);
|
|
return -1;
|
|
}
|
|
|
|
int retval = -2; // Not updated
|
|
if (sink->mem->magic == MEMSINK_MAGIC) {
|
|
if (sink->mem->version != MEMSINK_VERSION) {
|
|
LOG_ERROR("%s-sink: Protocol version mismatch: sink=%u, required=%u",
|
|
sink->name, sink->mem->version, MEMSINK_VERSION);
|
|
retval = -1;
|
|
goto done;
|
|
}
|
|
if (sink->mem->id != sink->last_id) { // When updated
|
|
sink->last_id = sink->mem->id;
|
|
frame_set_data(frame, sink->mem->data, sink->mem->used);
|
|
FRAME_COPY_META(sink->mem, frame);
|
|
retval = 0;
|
|
}
|
|
sink->mem->last_client_ts = get_now_monotonic();
|
|
}
|
|
|
|
done:
|
|
if (flock(sink->fd, LOCK_UN) < 0) {
|
|
LOG_PERROR("%s-sink: Can't unlock memory", sink->name);
|
|
return -1;
|
|
}
|
|
return retval;
|
|
}
|