mirror of
https://github.com/pikvm/ustreamer.git
synced 2026-02-27 12:16:31 +00:00
Compare commits
9 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
fdf3340a7d | ||
|
|
02513be220 | ||
|
|
d29ce42f08 | ||
|
|
aa6fc7fe04 | ||
|
|
c91341a375 | ||
|
|
3de7e26a36 | ||
|
|
63cc66e8a7 | ||
|
|
92a090dec3 | ||
|
|
8b0ef8a271 |
@@ -1,7 +1,7 @@
|
|||||||
[bumpversion]
|
[bumpversion]
|
||||||
commit = True
|
commit = True
|
||||||
tag = True
|
tag = True
|
||||||
current_version = 5.7
|
current_version = 5.12
|
||||||
parse = (?P<major>\d+)\.(?P<minor>\d+)
|
parse = (?P<major>\d+)\.(?P<minor>\d+)
|
||||||
serialize =
|
serialize =
|
||||||
{major}.{minor}
|
{major}.{minor}
|
||||||
|
|||||||
@@ -191,10 +191,10 @@ static void *_pcm_thread(void *v_audio) {
|
|||||||
while (!atomic_load(&audio->stop)) {
|
while (!atomic_load(&audio->stop)) {
|
||||||
int frames = snd_pcm_readi(audio->pcm, in, audio->pcm_frames);
|
int frames = snd_pcm_readi(audio->pcm, in, audio->pcm_frames);
|
||||||
if (frames < 0) {
|
if (frames < 0) {
|
||||||
JLOG_PERROR_ALSA(frames, "audio", "Can't capture PCM frames; breaking audio ...");
|
JLOG_PERROR_ALSA(frames, "audio", "Fatal: Can't capture PCM frames");
|
||||||
break;
|
break;
|
||||||
} else if (frames < (int)audio->pcm_frames) {
|
} else if (frames < (int)audio->pcm_frames) {
|
||||||
JLOG_ERROR("audio", "Too few PCM frames captured; breaking audio ...");
|
JLOG_ERROR("audio", "Fatal: Too few PCM frames captured");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -238,7 +238,7 @@ static void *_encoder_thread(void *v_audio) {
|
|||||||
int size = opus_encode(audio->enc, in_ptr, HZ_TO_FRAMES(ENCODER_INPUT_HZ), out->data, ARRAY_LEN(out->data));
|
int size = opus_encode(audio->enc, in_ptr, HZ_TO_FRAMES(ENCODER_INPUT_HZ), out->data, ARRAY_LEN(out->data));
|
||||||
free(in);
|
free(in);
|
||||||
if (size < 0) {
|
if (size < 0) {
|
||||||
JLOG_PERROR_OPUS(size, "audio", "Can't encode PCM frame to OPUS; breaking audio ...");
|
JLOG_PERROR_OPUS(size, "audio", "Fatal: Can't encode PCM frame to OPUS");
|
||||||
free(out);
|
free(out);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -116,14 +116,15 @@ typedef struct _client_sx {
|
|||||||
} _client_s;
|
} _client_s;
|
||||||
|
|
||||||
|
|
||||||
static char *_g_memsink_obj = NULL;
|
static char *_g_video_sink_name = NULL;
|
||||||
const long double _g_wait_timeout = 1;
|
const long double _g_sink_wait_timeout = 1;
|
||||||
const long double _g_lock_timeout = 1;
|
const long double _g_sink_lock_timeout = 1;
|
||||||
const useconds_t _g_lock_polling = 1000;
|
const useconds_t _g_sink_lock_polling = 1000;
|
||||||
const useconds_t _g_watchers_polling = 100000;
|
|
||||||
|
|
||||||
static char *_g_audio_dev = NULL;
|
static char *_g_audio_dev_name = NULL;
|
||||||
static char *_g_tc358743_dev = NULL;
|
static char *_g_tc358743_dev_path = NULL;
|
||||||
|
|
||||||
|
const useconds_t _g_watchers_polling = 100000;
|
||||||
|
|
||||||
static _client_s *_g_clients = NULL;
|
static _client_s *_g_clients = NULL;
|
||||||
static janus_callbacks *_g_gw = NULL;
|
static janus_callbacks *_g_gw = NULL;
|
||||||
@@ -149,10 +150,10 @@ static atomic_bool _g_has_watchers = false;
|
|||||||
|
|
||||||
|
|
||||||
static int _wait_frame(int fd, memsink_shared_s* mem, uint64_t last_id) {
|
static int _wait_frame(int fd, memsink_shared_s* mem, uint64_t last_id) {
|
||||||
long double deadline_ts = get_now_monotonic() + _g_wait_timeout;
|
long double deadline_ts = get_now_monotonic() + _g_sink_wait_timeout;
|
||||||
long double now;
|
long double now;
|
||||||
do {
|
do {
|
||||||
int result = flock_timedwait_monotonic(fd, _g_lock_timeout);
|
int result = flock_timedwait_monotonic(fd, _g_sink_lock_timeout);
|
||||||
now = get_now_monotonic();
|
now = get_now_monotonic();
|
||||||
if (result < 0 && errno != EWOULDBLOCK) {
|
if (result < 0 && errno != EWOULDBLOCK) {
|
||||||
JLOG_PERROR("video", "Can't lock memsink");
|
JLOG_PERROR("video", "Can't lock memsink");
|
||||||
@@ -166,7 +167,7 @@ static int _wait_frame(int fd, memsink_shared_s* mem, uint64_t last_id) {
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
usleep(_g_lock_polling);
|
usleep(_g_sink_lock_polling);
|
||||||
} while (now < deadline_ts);
|
} while (now < deadline_ts);
|
||||||
return -2;
|
return -2;
|
||||||
}
|
}
|
||||||
@@ -201,6 +202,11 @@ static void _relay_rtp_clients(const rtp_s *rtp) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define IF_NOT_REPORTED(...) { \
|
||||||
|
unsigned _error_code = __LINE__; \
|
||||||
|
if (error_reported != _error_code) { __VA_ARGS__; error_reported = _error_code; } \
|
||||||
|
}
|
||||||
|
|
||||||
static void *_clients_video_thread(UNUSED void *arg) {
|
static void *_clients_video_thread(UNUSED void *arg) {
|
||||||
A_THREAD_RENAME("us_v_clients");
|
A_THREAD_RENAME("us_v_clients");
|
||||||
atomic_store(&_g_video_tid_created, true);
|
atomic_store(&_g_video_tid_created, true);
|
||||||
@@ -211,13 +217,9 @@ static void *_clients_video_thread(UNUSED void *arg) {
|
|||||||
|
|
||||||
unsigned error_reported = 0;
|
unsigned error_reported = 0;
|
||||||
|
|
||||||
# define IF_NOT_REPORTED(_error, ...) { \
|
|
||||||
if (error_reported != _error) { __VA_ARGS__; error_reported = _error; } \
|
|
||||||
}
|
|
||||||
|
|
||||||
while (!STOP) {
|
while (!STOP) {
|
||||||
if (!HAS_WATCHERS) {
|
if (!HAS_WATCHERS) {
|
||||||
IF_NOT_REPORTED(1, { JLOG_INFO("video", "No active watchers, memsink disconnected"); });
|
IF_NOT_REPORTED({ JLOG_INFO("video", "No active watchers, memsink disconnected"); });
|
||||||
usleep(_g_watchers_polling);
|
usleep(_g_watchers_polling);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@@ -225,13 +227,13 @@ static void *_clients_video_thread(UNUSED void *arg) {
|
|||||||
int fd = -1;
|
int fd = -1;
|
||||||
memsink_shared_s *mem = NULL;
|
memsink_shared_s *mem = NULL;
|
||||||
|
|
||||||
if ((fd = shm_open(_g_memsink_obj, O_RDWR, 0)) <= 0) {
|
if ((fd = shm_open(_g_video_sink_name, O_RDWR, 0)) <= 0) {
|
||||||
IF_NOT_REPORTED(2, { JLOG_PERROR("video", "Can't open memsink"); });
|
IF_NOT_REPORTED({ JLOG_PERROR("video", "Can't open memsink"); });
|
||||||
goto close_memsink;
|
goto close_memsink;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((mem = memsink_shared_map(fd)) == NULL) {
|
if ((mem = memsink_shared_map(fd)) == NULL) {
|
||||||
IF_NOT_REPORTED(3, { JLOG_PERROR("video", "Can't map memsink"); });
|
IF_NOT_REPORTED({ JLOG_PERROR("video", "Can't map memsink"); });
|
||||||
goto close_memsink;
|
goto close_memsink;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -265,8 +267,6 @@ static void *_clients_video_thread(UNUSED void *arg) {
|
|||||||
sleep(1); // error_delay
|
sleep(1); // error_delay
|
||||||
}
|
}
|
||||||
|
|
||||||
# undef IF_NOT_REPORTED
|
|
||||||
|
|
||||||
frame_destroy(frame);
|
frame_destroy(frame);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
@@ -274,8 +274,10 @@ static void *_clients_video_thread(UNUSED void *arg) {
|
|||||||
static void *_clients_audio_thread(UNUSED void *arg) {
|
static void *_clients_audio_thread(UNUSED void *arg) {
|
||||||
A_THREAD_RENAME("us_a_clients");
|
A_THREAD_RENAME("us_a_clients");
|
||||||
atomic_store(&_g_audio_tid_created, true);
|
atomic_store(&_g_audio_tid_created, true);
|
||||||
assert(_g_audio_dev);
|
assert(_g_audio_dev_name);
|
||||||
assert(_g_tc358743_dev);
|
assert(_g_tc358743_dev_path);
|
||||||
|
|
||||||
|
unsigned error_reported = 0;
|
||||||
|
|
||||||
while (!STOP) {
|
while (!STOP) {
|
||||||
if (!HAS_WATCHERS) {
|
if (!HAS_WATCHERS) {
|
||||||
@@ -286,17 +288,23 @@ static void *_clients_audio_thread(UNUSED void *arg) {
|
|||||||
tc358743_info_s info = {0};
|
tc358743_info_s info = {0};
|
||||||
audio_s *audio = NULL;
|
audio_s *audio = NULL;
|
||||||
|
|
||||||
if (
|
if (tc358743_read_info(_g_tc358743_dev_path, &info) < 0) {
|
||||||
tc358743_read_info(_g_tc358743_dev, &info) < 0
|
goto close_audio;
|
||||||
|| !info.has_audio
|
}
|
||||||
|| (audio = audio_init(_g_audio_dev, info.audio_hz)) == NULL
|
if (!info.has_audio) {
|
||||||
) {
|
IF_NOT_REPORTED({ JLOG_INFO("audio", "No audio presented from the host"); });
|
||||||
|
goto close_audio;
|
||||||
|
}
|
||||||
|
IF_NOT_REPORTED({ JLOG_INFO("audio", "Detected host audio"); });
|
||||||
|
if ((audio = audio_init(_g_audio_dev_name, info.audio_hz)) == NULL) {
|
||||||
goto close_audio;
|
goto close_audio;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
error_reported = 0;
|
||||||
|
|
||||||
while (!STOP && HAS_WATCHERS) {
|
while (!STOP && HAS_WATCHERS) {
|
||||||
if (
|
if (
|
||||||
tc358743_read_info(_g_tc358743_dev, &info) < 0
|
tc358743_read_info(_g_tc358743_dev_path, &info) < 0
|
||||||
|| !info.has_audio
|
|| !info.has_audio
|
||||||
|| audio->pcm_hz != info.audio_hz
|
|| audio->pcm_hz != info.audio_hz
|
||||||
) {
|
) {
|
||||||
@@ -325,6 +333,8 @@ static void *_clients_audio_thread(UNUSED void *arg) {
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#undef IF_NOT_REPORTED
|
||||||
|
|
||||||
static char *_get_config_value(janus_config *config, const char *section, const char *option) {
|
static char *_get_config_value(janus_config *config, const char *section, const char *option) {
|
||||||
janus_config_category *section_obj = janus_config_get_create(config, NULL, janus_config_type_category, section);
|
janus_config_category *section_obj = janus_config_get_create(config, NULL, janus_config_type_category, section);
|
||||||
janus_config_item *option_obj = janus_config_get(config, section_obj, janus_config_type_item, option);
|
janus_config_item *option_obj = janus_config_get(config, section_obj, janus_config_type_item, option);
|
||||||
@@ -335,6 +345,8 @@ static char *_get_config_value(janus_config *config, const char *section, const
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int _read_config(const char *config_dir_path) {
|
static int _read_config(const char *config_dir_path) {
|
||||||
|
int retval = 0;
|
||||||
|
|
||||||
char *config_file_path;
|
char *config_file_path;
|
||||||
janus_config *config = NULL;
|
janus_config *config = NULL;
|
||||||
|
|
||||||
@@ -348,19 +360,21 @@ static int _read_config(const char *config_dir_path) {
|
|||||||
}
|
}
|
||||||
janus_config_print(config);
|
janus_config_print(config);
|
||||||
|
|
||||||
if ((_g_memsink_obj = _get_config_value(config, "memsink", "object")) == NULL) {
|
if (
|
||||||
JLOG_ERROR("main", "Missing config value: memsink.object");
|
(_g_video_sink_name = _get_config_value(config, "memsink", "object")) == NULL
|
||||||
|
&& (_g_video_sink_name = _get_config_value(config, "video", "sink")) == NULL
|
||||||
|
) {
|
||||||
|
JLOG_ERROR("main", "Missing config value: video.sink (ex. memsink.object)");
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
if ((_g_audio_dev = _get_config_value(config, "audio", "device")) != NULL) {
|
if ((_g_audio_dev_name = _get_config_value(config, "audio", "device")) != NULL) {
|
||||||
JLOG_INFO("main", "Enabled the experimental AUDIO feature");
|
JLOG_INFO("main", "Enabled the experimental AUDIO feature");
|
||||||
if ((_g_tc358743_dev = _get_config_value(config, "audio", "tc358743")) == NULL) {
|
if ((_g_tc358743_dev_path = _get_config_value(config, "audio", "tc358743")) == NULL) {
|
||||||
JLOG_INFO("main", "Missing config value: audio.tc358743");
|
JLOG_INFO("main", "Missing config value: audio.tc358743");
|
||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int retval = 0;
|
|
||||||
goto ok;
|
goto ok;
|
||||||
error:
|
error:
|
||||||
retval = -1;
|
retval = -1;
|
||||||
@@ -389,7 +403,7 @@ static int _plugin_init(janus_callbacks *gw, const char *config_dir_path) {
|
|||||||
}
|
}
|
||||||
_g_gw = gw;
|
_g_gw = gw;
|
||||||
_g_rtpv = rtpv_init(_relay_rtp_clients);
|
_g_rtpv = rtpv_init(_relay_rtp_clients);
|
||||||
if (_g_audio_dev) {
|
if (_g_audio_dev_name) {
|
||||||
_g_rtpa = rtpa_init(_relay_rtp_clients);
|
_g_rtpa = rtpa_init(_relay_rtp_clients);
|
||||||
A_THREAD_CREATE(&_g_audio_tid, _clients_audio_thread, NULL);
|
A_THREAD_CREATE(&_g_audio_tid, _clients_audio_thread, NULL);
|
||||||
}
|
}
|
||||||
@@ -417,9 +431,9 @@ static void _plugin_destroy(void) {
|
|||||||
DEL(rtpa_destroy, _g_rtpa);
|
DEL(rtpa_destroy, _g_rtpa);
|
||||||
DEL(rtpv_destroy, _g_rtpv);
|
DEL(rtpv_destroy, _g_rtpv);
|
||||||
_g_gw = NULL;
|
_g_gw = NULL;
|
||||||
DEL(free, _g_tc358743_dev);
|
DEL(free, _g_tc358743_dev_path);
|
||||||
DEL(free, _g_audio_dev);
|
DEL(free, _g_audio_dev_name);
|
||||||
DEL(free, _g_memsink_obj);
|
DEL(free, _g_video_sink_name);
|
||||||
# undef DEL
|
# undef DEL
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
.\" Manpage for ustreamer-dump.
|
.\" Manpage for ustreamer-dump.
|
||||||
.\" Open an issue or pull request to https://github.com/pikvm/ustreamer to correct errors or typos
|
.\" Open an issue or pull request to https://github.com/pikvm/ustreamer to correct errors or typos
|
||||||
.TH USTREAMER-DUMP 1 "version 5.7" "January 2021"
|
.TH USTREAMER-DUMP 1 "version 5.12" "January 2021"
|
||||||
|
|
||||||
.SH NAME
|
.SH NAME
|
||||||
ustreamer-dump \- Dump uStreamer's memory sink to file
|
ustreamer-dump \- Dump uStreamer's memory sink to file
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
.\" Manpage for ustreamer.
|
.\" Manpage for ustreamer.
|
||||||
.\" Open an issue or pull request to https://github.com/pikvm/ustreamer to correct errors or typos
|
.\" Open an issue or pull request to https://github.com/pikvm/ustreamer to correct errors or typos
|
||||||
.TH USTREAMER 1 "version 5.7" "November 2020"
|
.TH USTREAMER 1 "version 5.12" "November 2020"
|
||||||
|
|
||||||
.SH NAME
|
.SH NAME
|
||||||
ustreamer \- stream MJPEG video from any V4L2 device to the network
|
ustreamer \- stream MJPEG video from any V4L2 device to the network
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
|
|
||||||
pkgname=ustreamer
|
pkgname=ustreamer
|
||||||
pkgver=5.7
|
pkgver=5.12
|
||||||
pkgrel=1
|
pkgrel=1
|
||||||
pkgdesc="Lightweight and fast MJPEG-HTTP streamer"
|
pkgdesc="Lightweight and fast MJPEG-HTTP streamer"
|
||||||
url="https://github.com/pikvm/ustreamer"
|
url="https://github.com/pikvm/ustreamer"
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
include $(TOPDIR)/rules.mk
|
include $(TOPDIR)/rules.mk
|
||||||
|
|
||||||
PKG_NAME:=ustreamer
|
PKG_NAME:=ustreamer
|
||||||
PKG_VERSION:=5.7
|
PKG_VERSION:=5.12
|
||||||
PKG_RELEASE:=1
|
PKG_RELEASE:=1
|
||||||
PKG_MAINTAINER:=Maxim Devaev <mdevaev@gmail.com>
|
PKG_MAINTAINER:=Maxim Devaev <mdevaev@gmail.com>
|
||||||
|
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ def _find_sources(suffix: str) -> List[str]:
|
|||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
setup(
|
setup(
|
||||||
name="ustreamer",
|
name="ustreamer",
|
||||||
version="5.7",
|
version="5.12",
|
||||||
description="uStreamer tools",
|
description="uStreamer tools",
|
||||||
author="Maxim Devaev",
|
author="Maxim Devaev",
|
||||||
author_email="mdevaev@gmail.com",
|
author_email="mdevaev@gmail.com",
|
||||||
|
|||||||
@@ -23,7 +23,7 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#define VERSION_MAJOR 5
|
#define VERSION_MAJOR 5
|
||||||
#define VERSION_MINOR 7
|
#define VERSION_MINOR 12
|
||||||
|
|
||||||
#define MAKE_VERSION2(_major, _minor) #_major "." #_minor
|
#define MAKE_VERSION2(_major, _minor) #_major "." #_minor
|
||||||
#define MAKE_VERSION1(_major, _minor) MAKE_VERSION2(_major, _minor)
|
#define MAKE_VERSION1(_major, _minor) MAKE_VERSION2(_major, _minor)
|
||||||
|
|||||||
@@ -224,8 +224,9 @@ static bool _worker_run_job(worker_s *wr) {
|
|||||||
} else if (ER(type) == ENCODER_TYPE_NOOP) {
|
} else if (ER(type) == ENCODER_TYPE_NOOP) {
|
||||||
LOG_VERBOSE("Compressing JPEG using NOOP (do nothing): worker=%s, buffer=%u",
|
LOG_VERBOSE("Compressing JPEG using NOOP (do nothing): worker=%s, buffer=%u",
|
||||||
wr->name, job->hw->buf.index);
|
wr->name, job->hw->buf.index);
|
||||||
|
frame_encoding_begin(src, dest, V4L2_PIX_FMT_JPEG);
|
||||||
usleep(5000); // Просто чтобы работала логика desired_fps
|
usleep(5000); // Просто чтобы работала логика desired_fps
|
||||||
dest->encode_end_ts = get_now_monotonic();
|
dest->encode_end_ts = get_now_monotonic(); // frame_encoding_end()
|
||||||
}
|
}
|
||||||
|
|
||||||
LOG_VERBOSE("Compressed new JPEG: size=%zu, time=%0.3Lf, worker=%s, buffer=%u",
|
LOG_VERBOSE("Compressed new JPEG: size=%zu, time=%0.3Lf, worker=%s, buffer=%u",
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ static m2m_encoder_s *_m2m_encoder_init(
|
|||||||
const char *name, const char *path, unsigned output_format,
|
const char *name, const char *path, unsigned output_format,
|
||||||
unsigned fps, bool allow_dma, m2m_option_s *options);
|
unsigned fps, bool allow_dma, m2m_option_s *options);
|
||||||
|
|
||||||
static int _m2m_encoder_prepare(m2m_encoder_s *enc, const frame_s *frame);
|
static void _m2m_encoder_prepare(m2m_encoder_s *enc, const frame_s *frame);
|
||||||
|
|
||||||
static int _m2m_encoder_init_buffers(
|
static int _m2m_encoder_init_buffers(
|
||||||
m2m_encoder_s *enc, const char *name, enum v4l2_buf_type type,
|
m2m_encoder_s *enc, const char *name, enum v4l2_buf_type type,
|
||||||
@@ -117,9 +117,10 @@ int m2m_encoder_compress(m2m_encoder_s *enc, const frame_s *src, frame_s *dest,
|
|||||||
|| RUN(stride) != src->stride
|
|| RUN(stride) != src->stride
|
||||||
|| RUN(dma) != (enc->allow_dma && src->dma_fd >= 0)
|
|| RUN(dma) != (enc->allow_dma && src->dma_fd >= 0)
|
||||||
) {
|
) {
|
||||||
if (_m2m_encoder_prepare(enc, src) < 0) {
|
_m2m_encoder_prepare(enc, src);
|
||||||
return -1;
|
}
|
||||||
}
|
if (!RUN(ready)) { // Already prepared but failed
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
force_key = (enc->output_format == V4L2_PIX_FMT_H264 && (force_key || RUN(last_online) != src->online));
|
force_key = (enc->output_format == V4L2_PIX_FMT_H264 && (force_key || RUN(last_online) != src->online));
|
||||||
@@ -179,7 +180,7 @@ static m2m_encoder_s *_m2m_encoder_init(
|
|||||||
} \
|
} \
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _m2m_encoder_prepare(m2m_encoder_s *enc, const frame_s *frame) {
|
static void _m2m_encoder_prepare(m2m_encoder_s *enc, const frame_s *frame) {
|
||||||
bool dma = (enc->allow_dma && frame->dma_fd >= 0);
|
bool dma = (enc->allow_dma && frame->dma_fd >= 0);
|
||||||
|
|
||||||
E_LOG_INFO("Configuring encoder: DMA=%d ...", dma);
|
E_LOG_INFO("Configuring encoder: DMA=%d ...", dma);
|
||||||
@@ -273,12 +274,11 @@ static int _m2m_encoder_prepare(m2m_encoder_s *enc, const frame_s *frame) {
|
|||||||
|
|
||||||
RUN(ready) = true;
|
RUN(ready) = true;
|
||||||
E_LOG_DEBUG("Encoder state: *** READY ***");
|
E_LOG_DEBUG("Encoder state: *** READY ***");
|
||||||
return 0;
|
return;
|
||||||
|
|
||||||
error:
|
error:
|
||||||
_m2m_encoder_cleanup(enc);
|
_m2m_encoder_cleanup(enc);
|
||||||
E_LOG_ERROR("Encoder destroyed due an error (prepare)");
|
E_LOG_ERROR("Encoder destroyed due an error (prepare)");
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _m2m_encoder_init_buffers(
|
static int _m2m_encoder_init_buffers(
|
||||||
|
|||||||
Reference in New Issue
Block a user