mirror of
https://github.com/pikvm/ustreamer.git
synced 2026-02-27 12:16:31 +00:00
Compare commits
5 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f11d390b22 | ||
|
|
f1e50b6f9b | ||
|
|
fdf3340a7d | ||
|
|
02513be220 | ||
|
|
d29ce42f08 |
@@ -1,7 +1,7 @@
|
|||||||
[bumpversion]
|
[bumpversion]
|
||||||
commit = True
|
commit = True
|
||||||
tag = True
|
tag = True
|
||||||
current_version = 5.10
|
current_version = 5.13
|
||||||
parse = (?P<major>\d+)\.(?P<minor>\d+)
|
parse = (?P<major>\d+)\.(?P<minor>\d+)
|
||||||
serialize =
|
serialize =
|
||||||
{major}.{minor}
|
{major}.{minor}
|
||||||
|
|||||||
@@ -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.10" "January 2021"
|
.TH USTREAMER-DUMP 1 "version 5.13" "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.10" "November 2020"
|
.TH USTREAMER 1 "version 5.13" "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.10
|
pkgver=5.13
|
||||||
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.10
|
PKG_VERSION:=5.13
|
||||||
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.10",
|
version="5.13",
|
||||||
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 10
|
#define VERSION_MINOR 13
|
||||||
|
|
||||||
#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)
|
||||||
|
|||||||
@@ -25,9 +25,9 @@
|
|||||||
|
|
||||||
static m2m_encoder_s *_m2m_encoder_init(
|
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, unsigned bitrate, unsigned gop, unsigned quality, bool allow_dma);
|
||||||
|
|
||||||
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,
|
||||||
@@ -46,26 +46,11 @@ static int _m2m_encoder_compress_raw(m2m_encoder_s *enc, const frame_s *src, fra
|
|||||||
|
|
||||||
|
|
||||||
m2m_encoder_s *m2m_h264_encoder_init(const char *name, const char *path, unsigned bitrate, unsigned gop) {
|
m2m_encoder_s *m2m_h264_encoder_init(const char *name, const char *path, unsigned bitrate, unsigned gop) {
|
||||||
# define OPTION(_key, _value) {#_key, V4L2_CID_MPEG_VIDEO_##_key, _value}
|
|
||||||
|
|
||||||
m2m_option_s options[] = {
|
|
||||||
OPTION(BITRATE, bitrate * 1000),
|
|
||||||
// OPTION(BITRATE_PEAK, bitrate * 1000),
|
|
||||||
OPTION(H264_I_PERIOD, gop),
|
|
||||||
OPTION(H264_PROFILE, V4L2_MPEG_VIDEO_H264_PROFILE_CONSTRAINED_BASELINE),
|
|
||||||
OPTION(H264_LEVEL, V4L2_MPEG_VIDEO_H264_LEVEL_4_0),
|
|
||||||
OPTION(REPEAT_SEQ_HEADER, 1),
|
|
||||||
OPTION(H264_MIN_QP, 16),
|
|
||||||
OPTION(H264_MAX_QP, 32),
|
|
||||||
{NULL, 0, 0},
|
|
||||||
};
|
|
||||||
|
|
||||||
# undef OPTION
|
|
||||||
|
|
||||||
// FIXME: 30 or 0? https://github.com/6by9/yavta/blob/master/yavta.c#L2100
|
// FIXME: 30 or 0? https://github.com/6by9/yavta/blob/master/yavta.c#L2100
|
||||||
// По логике вещей правильно 0, но почему-то на низких разрешениях типа 640x480
|
// По логике вещей правильно 0, но почему-то на низких разрешениях типа 640x480
|
||||||
// енкодер через несколько секунд перестает производить корректные фреймы.
|
// енкодер через несколько секунд перестает производить корректные фреймы.
|
||||||
return _m2m_encoder_init(name, path, V4L2_PIX_FMT_H264, 30, true, options);
|
bitrate *= 1000; // From Kbps
|
||||||
|
return _m2m_encoder_init(name, path, V4L2_PIX_FMT_H264, 30, bitrate, gop, 0, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
m2m_encoder_s *m2m_mjpeg_encoder_init(const char *name, const char *path, unsigned quality) {
|
m2m_encoder_s *m2m_mjpeg_encoder_init(const char *name, const char *path, unsigned quality) {
|
||||||
@@ -76,30 +61,18 @@ m2m_encoder_s *m2m_mjpeg_encoder_init(const char *name, const char *path, unsign
|
|||||||
bitrate = step * round(bitrate / step);
|
bitrate = step * round(bitrate / step);
|
||||||
bitrate *= 1000; // From Kbps
|
bitrate *= 1000; // From Kbps
|
||||||
assert(bitrate > 0);
|
assert(bitrate > 0);
|
||||||
|
|
||||||
m2m_option_s options[] = {
|
|
||||||
{"BITRATE", V4L2_CID_MPEG_VIDEO_BITRATE, bitrate},
|
|
||||||
{NULL, 0, 0},
|
|
||||||
};
|
|
||||||
|
|
||||||
// FIXME: То же самое про 30 or 0, но еще даже не проверено на низких разрешениях
|
// FIXME: То же самое про 30 or 0, но еще даже не проверено на низких разрешениях
|
||||||
return _m2m_encoder_init(name, path, V4L2_PIX_FMT_MJPEG, 30, true, options);
|
return _m2m_encoder_init(name, path, V4L2_PIX_FMT_MJPEG, 30, bitrate, 0, 0, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
m2m_encoder_s *m2m_jpeg_encoder_init(const char *name, const char *path, unsigned quality) {
|
m2m_encoder_s *m2m_jpeg_encoder_init(const char *name, const char *path, unsigned quality) {
|
||||||
m2m_option_s options[] = {
|
|
||||||
{"QUALITY", V4L2_CID_JPEG_COMPRESSION_QUALITY, quality},
|
|
||||||
{NULL, 0, 0},
|
|
||||||
};
|
|
||||||
|
|
||||||
// FIXME: DMA не работает
|
// FIXME: DMA не работает
|
||||||
return _m2m_encoder_init(name, path, V4L2_PIX_FMT_JPEG, 30, false, options);
|
return _m2m_encoder_init(name, path, V4L2_PIX_FMT_JPEG, 30, 0, 0, quality, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
void m2m_encoder_destroy(m2m_encoder_s *enc) {
|
void m2m_encoder_destroy(m2m_encoder_s *enc) {
|
||||||
E_LOG_INFO("Destroying encoder ...");
|
E_LOG_INFO("Destroying encoder ...");
|
||||||
_m2m_encoder_cleanup(enc);
|
_m2m_encoder_cleanup(enc);
|
||||||
free(enc->options);
|
|
||||||
free(enc->path);
|
free(enc->path);
|
||||||
free(enc->name);
|
free(enc->name);
|
||||||
free(enc);
|
free(enc);
|
||||||
@@ -117,9 +90,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));
|
||||||
@@ -141,7 +115,7 @@ int m2m_encoder_compress(m2m_encoder_s *enc, const frame_s *src, frame_s *dest,
|
|||||||
|
|
||||||
static m2m_encoder_s *_m2m_encoder_init(
|
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, unsigned bitrate, unsigned gop, unsigned quality, bool allow_dma) {
|
||||||
|
|
||||||
LOG_INFO("%s: Initializing encoder ...", name);
|
LOG_INFO("%s: Initializing encoder ...", name);
|
||||||
|
|
||||||
@@ -160,15 +134,11 @@ static m2m_encoder_s *_m2m_encoder_init(
|
|||||||
}
|
}
|
||||||
enc->output_format = output_format;
|
enc->output_format = output_format;
|
||||||
enc->fps = fps;
|
enc->fps = fps;
|
||||||
|
enc->bitrate = bitrate;
|
||||||
|
enc->gop = gop;
|
||||||
|
enc->quality = quality;
|
||||||
enc->allow_dma = allow_dma;
|
enc->allow_dma = allow_dma;
|
||||||
enc->run = run;
|
enc->run = run;
|
||||||
|
|
||||||
unsigned count = 0;
|
|
||||||
for (; options[count].name != NULL; ++count);
|
|
||||||
++count;
|
|
||||||
A_CALLOC(enc->options, count);
|
|
||||||
memcpy(enc->options, options, sizeof(m2m_option_s) * count);
|
|
||||||
|
|
||||||
return enc;
|
return enc;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -179,7 +149,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);
|
||||||
@@ -198,15 +168,34 @@ static int _m2m_encoder_prepare(m2m_encoder_s *enc, const frame_s *frame) {
|
|||||||
}
|
}
|
||||||
E_LOG_DEBUG("Encoder device fd=%d opened", RUN(fd));
|
E_LOG_DEBUG("Encoder device fd=%d opened", RUN(fd));
|
||||||
|
|
||||||
for (m2m_option_s *option = enc->options; option->name != NULL; ++option) {
|
# define SET_OPTION(_cid, _value) { \
|
||||||
struct v4l2_control ctl = {0};
|
struct v4l2_control _ctl = {0}; \
|
||||||
ctl.id = option->id;
|
_ctl.id = _cid; \
|
||||||
ctl.value = option->value;
|
_ctl.value = _value; \
|
||||||
|
E_LOG_DEBUG("Configuring option " #_cid " ..."); \
|
||||||
|
E_XIOCTL(VIDIOC_S_CTRL, &_ctl, "Can't set option " #_cid); \
|
||||||
|
}
|
||||||
|
|
||||||
E_LOG_DEBUG("Configuring option %s ...", option->name);
|
if (enc->output_format == V4L2_PIX_FMT_H264) {
|
||||||
E_XIOCTL(VIDIOC_S_CTRL, &ctl, "Can't set option %s", option->name);
|
SET_OPTION(V4L2_CID_MPEG_VIDEO_BITRATE, enc->bitrate);
|
||||||
|
SET_OPTION(V4L2_CID_MPEG_VIDEO_H264_I_PERIOD, enc->gop);
|
||||||
|
SET_OPTION(V4L2_CID_MPEG_VIDEO_H264_PROFILE, V4L2_MPEG_VIDEO_H264_PROFILE_CONSTRAINED_BASELINE);
|
||||||
|
if (RUN(width) * RUN(height) <= 1920 * 1080) { // https://forums.raspberrypi.com/viewtopic.php?t=291447#p1762296
|
||||||
|
SET_OPTION(V4L2_CID_MPEG_VIDEO_H264_LEVEL, V4L2_MPEG_VIDEO_H264_LEVEL_4_0);
|
||||||
|
} else {
|
||||||
|
SET_OPTION(V4L2_CID_MPEG_VIDEO_H264_LEVEL, V4L2_MPEG_VIDEO_H264_LEVEL_5_1);
|
||||||
|
}
|
||||||
|
SET_OPTION(V4L2_CID_MPEG_VIDEO_REPEAT_SEQ_HEADER, 1);
|
||||||
|
SET_OPTION(V4L2_CID_MPEG_VIDEO_H264_MIN_QP, 16);
|
||||||
|
SET_OPTION(V4L2_CID_MPEG_VIDEO_H264_MAX_QP, 32);
|
||||||
|
} else if (enc->output_format == V4L2_PIX_FMT_MJPEG) {
|
||||||
|
SET_OPTION(V4L2_CID_MPEG_VIDEO_BITRATE, enc->bitrate);
|
||||||
|
} else if (enc->output_format == V4L2_PIX_FMT_JPEG) {
|
||||||
|
SET_OPTION(V4L2_CID_JPEG_COMPRESSION_QUALITY, enc->quality);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# undef SET_OPTION
|
||||||
|
|
||||||
{
|
{
|
||||||
struct v4l2_format fmt = {0};
|
struct v4l2_format fmt = {0};
|
||||||
fmt.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
|
fmt.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
|
||||||
@@ -273,12 +262,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(
|
||||||
|
|||||||
@@ -47,12 +47,6 @@ typedef struct {
|
|||||||
size_t allocated;
|
size_t allocated;
|
||||||
} m2m_buffer_s;
|
} m2m_buffer_s;
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
char *name;
|
|
||||||
uint32_t id;
|
|
||||||
int32_t value;
|
|
||||||
} m2m_option_s;
|
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
int fd;
|
int fd;
|
||||||
m2m_buffer_s *input_bufs;
|
m2m_buffer_s *input_bufs;
|
||||||
@@ -75,8 +69,10 @@ typedef struct {
|
|||||||
char *path;
|
char *path;
|
||||||
unsigned output_format;
|
unsigned output_format;
|
||||||
unsigned fps;
|
unsigned fps;
|
||||||
|
unsigned bitrate;
|
||||||
|
unsigned gop;
|
||||||
|
unsigned quality;
|
||||||
bool allow_dma;
|
bool allow_dma;
|
||||||
m2m_option_s *options;
|
|
||||||
|
|
||||||
m2m_encoder_runtime_s *run;
|
m2m_encoder_runtime_s *run;
|
||||||
} m2m_encoder_s;
|
} m2m_encoder_s;
|
||||||
|
|||||||
Reference in New Issue
Block a user