mirror of
https://gitlab.uni-freiburg.de/opensourcevdi/spice
synced 2025-12-26 14:41:25 +00:00
Signed-off-by: Francois Gouget <fgouget@codeweavers.com> Acked-by: Frediano Ziglio <fziglio@redhat.com>
1791 lines
66 KiB
C
1791 lines
66 KiB
C
/* -*- Mode: C; c-basic-offset: 4; indent-tabs-mode: nil -*- */
|
|
/*
|
|
Copyright (C) 2015 Jeremy White
|
|
Copyright (C) 2015-2016 Francois Gouget
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
This library 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
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include <config.h>
|
|
|
|
#include <inttypes.h>
|
|
#include <pthread.h>
|
|
|
|
#include <gst/gst.h>
|
|
#include <gst/app/gstappsrc.h>
|
|
#include <gst/app/gstappsink.h>
|
|
#include <gst/video/video.h>
|
|
#include <orc/orcprogram.h>
|
|
|
|
#include "red-common.h"
|
|
#include "video-encoder.h"
|
|
#include "utils.h"
|
|
|
|
|
|
#define SPICE_GST_DEFAULT_FPS 30
|
|
|
|
#ifndef HAVE_GSTREAMER_0_10
|
|
# define DO_ZERO_COPY
|
|
#endif
|
|
|
|
|
|
typedef struct {
|
|
SpiceBitmapFmt spice_format;
|
|
uint32_t bpp;
|
|
#ifndef HAVE_GSTREAMER_0_10
|
|
char format[8];
|
|
GstVideoFormat gst_format;
|
|
#else
|
|
uint32_t depth;
|
|
uint32_t endianness;
|
|
uint32_t blue_mask;
|
|
uint32_t green_mask;
|
|
uint32_t red_mask;
|
|
#endif
|
|
} SpiceFormatForGStreamer;
|
|
|
|
#ifndef HAVE_GSTREAMER_0_10
|
|
#define FMT_DESC(spice_format, bpp, format, gst_format, depth, endianness, \
|
|
blue_mask, green_mask, red_mask) \
|
|
{ spice_format, bpp, format, gst_format }
|
|
#else
|
|
#define FMT_DESC(spice_format, bpp, format, gst_format, depth, endianness, \
|
|
blue_mask, green_mask, red_mask) \
|
|
{ spice_format, bpp, depth, endianness, blue_mask, green_mask, red_mask }
|
|
#endif
|
|
|
|
typedef struct SpiceGstVideoBuffer {
|
|
VideoBuffer base;
|
|
GstBuffer *gst_buffer;
|
|
#ifndef HAVE_GSTREAMER_0_10
|
|
GstMapInfo map;
|
|
#endif
|
|
} SpiceGstVideoBuffer;
|
|
|
|
typedef struct {
|
|
uint32_t mm_time;
|
|
uint32_t size;
|
|
uint64_t duration;
|
|
} SpiceGstFrameInformation;
|
|
|
|
typedef enum SpiceGstBitRateStatus {
|
|
SPICE_GST_BITRATE_DECREASING,
|
|
SPICE_GST_BITRATE_INCREASING,
|
|
SPICE_GST_BITRATE_STABLE,
|
|
} SpiceGstBitRateStatus;
|
|
|
|
typedef struct SpiceGstEncoder {
|
|
VideoEncoder base;
|
|
|
|
/* Callbacks to adjust the refcount of the bitmap being encoded. */
|
|
bitmap_ref_t bitmap_ref;
|
|
bitmap_unref_t bitmap_unref;
|
|
|
|
#ifdef DO_ZERO_COPY
|
|
GAsyncQueue *unused_bitmap_opaques;
|
|
#endif
|
|
|
|
/* Rate control callbacks */
|
|
VideoEncoderRateControlCbs cbs;
|
|
|
|
/* Spice's initial bit rate estimation in bits per second. */
|
|
uint64_t starting_bit_rate;
|
|
|
|
/* ---------- Video characteristics ---------- */
|
|
|
|
uint32_t width;
|
|
uint32_t height;
|
|
const SpiceFormatForGStreamer *format;
|
|
SpiceBitmapFmt spice_format;
|
|
|
|
/* Number of consecutive frame encoding errors. */
|
|
uint32_t errors;
|
|
|
|
/* ---------- GStreamer pipeline ---------- */
|
|
|
|
/* Pointers to the GStreamer pipeline elements. If pipeline is NULL the
|
|
* other pointers are invalid.
|
|
*/
|
|
GstElement *pipeline;
|
|
GstAppSink *appsink;
|
|
GstAppSrc *appsrc;
|
|
GstCaps *src_caps;
|
|
GstElement *gstenc;
|
|
GParamSpec *gstenc_bitrate_param;
|
|
|
|
/* True if the encoder's bitrate can be modified while playing. */
|
|
gboolean gstenc_bitrate_is_dynamic;
|
|
|
|
/* Pipeline parameters to modify before the next frame. */
|
|
# define SPICE_GST_VIDEO_PIPELINE_STATE 0x1
|
|
# define SPICE_GST_VIDEO_PIPELINE_BITRATE 0x2
|
|
# define SPICE_GST_VIDEO_PIPELINE_CAPS 0x4
|
|
uint32_t set_pipeline;
|
|
|
|
/* Output buffer */
|
|
pthread_mutex_t outbuf_mutex;
|
|
pthread_cond_t outbuf_cond;
|
|
VideoBuffer *outbuf;
|
|
|
|
/* The video bit rate. */
|
|
uint64_t video_bit_rate;
|
|
|
|
/* Don't bother changing the GStreamer bit rate if close enough. */
|
|
# define SPICE_GST_VIDEO_BITRATE_MARGIN 0.05
|
|
|
|
|
|
/* ---------- Encoded frame statistics ---------- */
|
|
|
|
/* Should be >= than FRAME_STATISTICS_COUNT. This is also used to
|
|
* annotate the client report debug traces with bit rate information.
|
|
*/
|
|
# define SPICE_GST_HISTORY_SIZE 60
|
|
|
|
/* A circular buffer containing the past encoded frames information. */
|
|
SpiceGstFrameInformation history[SPICE_GST_HISTORY_SIZE];
|
|
|
|
/* The indices of the oldest and newest frames in the history buffer. */
|
|
uint32_t history_first;
|
|
uint32_t history_last;
|
|
|
|
/* How many frames to take into account when computing the effective
|
|
* bit rate, average frame size, etc. This should be large enough so the
|
|
* I and P frames average out, and short enough for it to reflect the
|
|
* current situation.
|
|
*/
|
|
# define SPICE_GST_FRAME_STATISTICS_COUNT 21
|
|
|
|
/* The index of the oldest frame taken into account for the statistics. */
|
|
uint32_t stat_first;
|
|
|
|
/* Used to compute the average frame encoding time. */
|
|
uint64_t stat_duration_sum;
|
|
|
|
/* Used to compute the average frame size. */
|
|
uint64_t stat_size_sum;
|
|
|
|
/* Tracks the maximum frame size. */
|
|
uint32_t stat_size_max;
|
|
|
|
|
|
/* ---------- Encoder bit rate control ----------
|
|
*
|
|
* GStreamer encoders don't follow the specified video_bit_rate very
|
|
* closely. These fields are used to ensure we don't exceed the desired
|
|
* stream bit rate, regardless of the GStreamer encoder's output.
|
|
*/
|
|
|
|
/* The bit rate target for the outgoing network stream. (bits per second) */
|
|
uint64_t bit_rate;
|
|
|
|
/* The minimum bit rate / bit rate increment. */
|
|
# define SPICE_GST_MIN_BITRATE (128 * 1024)
|
|
|
|
/* The default bit rate. */
|
|
# define SPICE_GST_DEFAULT_BITRATE (8 * 1024 * 1024)
|
|
|
|
/* The bit rate control is performed using a virtual buffer to allow
|
|
* short term variations: bursts are allowed until the virtual buffer is
|
|
* full. Then frames are dropped to limit the bit rate. VBUFFER_SIZE
|
|
* defines the size of the virtual buffer in milliseconds worth of data
|
|
* while vbuffer_size holds the limit in bytes for the current bit rate.
|
|
*/
|
|
# define SPICE_GST_VBUFFER_SIZE 300
|
|
|
|
int32_t vbuffer_size;
|
|
int32_t vbuffer_free;
|
|
|
|
/* When dropping frames, this is set to the minimum mm_time of the next
|
|
* frame to encode. Otherwise set to zero.
|
|
*/
|
|
uint32_t next_frame_mm_time;
|
|
|
|
/* Defines the minimum allowed fps. */
|
|
# define SPICE_GST_MAX_PERIOD (NSEC_PER_SEC / 3)
|
|
|
|
/* How big of a margin to take to cover for latency jitter. */
|
|
# define SPICE_GST_LATENCY_MARGIN 0.1
|
|
|
|
|
|
/* ---------- Network bit rate control ----------
|
|
*
|
|
* State information for figuring out the optimal bit rate for the
|
|
* current network conditions.
|
|
*/
|
|
|
|
/* The mm_time of the last bit rate change. */
|
|
uint32_t last_change;
|
|
|
|
/* How much to reduce the bit rate in case of network congestion. */
|
|
# define SPICE_GST_BITRATE_CUT 2
|
|
# define SPICE_GST_BITRATE_REDUCE (4.0 / 3.0)
|
|
|
|
/* Never increase the bit rate by more than this amount (bits per second). */
|
|
# define SPICE_GST_BITRATE_MAX_STEP (1024 * 1024)
|
|
|
|
/* The maximum bit rate that one can maybe use without causing network
|
|
* congestion.
|
|
*/
|
|
uint64_t max_bit_rate;
|
|
|
|
/* The last bit rate that let us recover from network congestion. */
|
|
uint64_t min_bit_rate;
|
|
|
|
/* Defines when the spread between max_bit_rate and min_bit_rate has been
|
|
* narrowed down enough. Note that this value should be large enough for
|
|
* min_bit_rate to allow recovery from network congestion in a reasonable
|
|
* time frame, and to absorb transient traffic spikes (potentially from
|
|
* other sources).
|
|
* This is also used as a multiplier for the video_bit_rate so it does
|
|
* not have to be changed too often.
|
|
*/
|
|
# define SPICE_GST_BITRATE_MARGIN SPICE_GST_BITRATE_REDUCE
|
|
|
|
/* Whether the bit rate was last decreased, increased or kept stable. */
|
|
SpiceGstBitRateStatus status;
|
|
|
|
/* The network bit rate control uses an AIMD scheme (Additive Increase,
|
|
* Multiplicative Decrease). The increment step depends on the spread
|
|
* between the minimum and maximum bit rates.
|
|
*/
|
|
uint64_t bit_rate_step;
|
|
|
|
/* How often to increase the bit rate. */
|
|
uint32_t increase_interval;
|
|
|
|
# define SPICE_GST_BITRATE_UP_INTERVAL (MSEC_PER_SEC * 2)
|
|
# define SPICE_GST_BITRATE_UP_CLIENT_STABLE (MSEC_PER_SEC * 60 * 2)
|
|
# define SPICE_GST_BITRATE_UP_SERVER_STABLE (MSEC_PER_SEC * 3600 * 4)
|
|
# define SPICE_GST_BITRATE_UP_RESET_MAX (MSEC_PER_SEC * 30)
|
|
|
|
|
|
/* ---------- Client feedback ---------- */
|
|
|
|
/* TRUE if gst_encoder_client_stream_report() is being called. */
|
|
gboolean has_client_reports;
|
|
|
|
/* The margin is the amount of time between the reception of a piece of
|
|
* media data by the client and the time when it should be displayed.
|
|
* Increasing the bit rate increases the transmission time and thus
|
|
* reduces the margin.
|
|
*/
|
|
int32_t last_video_margin;
|
|
int32_t max_video_margin;
|
|
uint32_t max_audio_margin;
|
|
|
|
# define SPICE_GST_VIDEO_MARGIN_GOOD 0.75
|
|
# define SPICE_GST_VIDEO_MARGIN_AVERAGE 0.5
|
|
# define SPICE_GST_VIDEO_MARGIN_BAD 0.3
|
|
|
|
# define SPICE_GST_VIDEO_DELTA_BAD 0.2
|
|
# define SPICE_GST_VIDEO_DELTA_AVERAGE 0.15
|
|
|
|
# define SPICE_GST_AUDIO_MARGIN_BAD 0.5
|
|
# define SPICE_GST_AUDIO_VIDEO_RATIO 1.25
|
|
|
|
|
|
/* ---------- Server feedback ---------- */
|
|
|
|
/* How many frames were dropped by the server since the last encoded frame. */
|
|
uint32_t server_drops;
|
|
} SpiceGstEncoder;
|
|
|
|
|
|
/* ---------- The SpiceGstVideoBuffer implementation ---------- */
|
|
|
|
static void spice_gst_video_buffer_free(VideoBuffer *video_buffer)
|
|
{
|
|
SpiceGstVideoBuffer *buffer = (SpiceGstVideoBuffer*)video_buffer;
|
|
if (buffer->gst_buffer) {
|
|
#ifndef HAVE_GSTREAMER_0_10
|
|
gst_buffer_unmap(buffer->gst_buffer, &buffer->map);
|
|
#endif
|
|
gst_buffer_unref(buffer->gst_buffer);
|
|
}
|
|
g_free(buffer);
|
|
}
|
|
|
|
static SpiceGstVideoBuffer* create_gst_video_buffer(void)
|
|
{
|
|
SpiceGstVideoBuffer *buffer = g_new0(SpiceGstVideoBuffer, 1);
|
|
buffer->base.free = spice_gst_video_buffer_free;
|
|
return buffer;
|
|
}
|
|
|
|
|
|
/* ---------- Miscellaneous SpiceGstEncoder helpers ---------- */
|
|
|
|
static inline double get_mbps(uint64_t bit_rate)
|
|
{
|
|
return (double)bit_rate / 1024 / 1024;
|
|
}
|
|
|
|
/* Returns the source frame rate which may change at any time so don't store
|
|
* the result.
|
|
*/
|
|
static uint32_t get_source_fps(const SpiceGstEncoder *encoder)
|
|
{
|
|
return encoder->cbs.get_source_fps ?
|
|
encoder->cbs.get_source_fps(encoder->cbs.opaque) : SPICE_GST_DEFAULT_FPS;
|
|
}
|
|
|
|
static uint32_t get_network_latency(const SpiceGstEncoder *encoder)
|
|
{
|
|
/* Assume that the network latency is symmetric */
|
|
return encoder->cbs.get_roundtrip_ms ?
|
|
encoder->cbs.get_roundtrip_ms(encoder->cbs.opaque) / 2 : 0;
|
|
}
|
|
|
|
static void set_pipeline_changes(SpiceGstEncoder *encoder, uint32_t flags)
|
|
{
|
|
encoder->set_pipeline |= flags;
|
|
}
|
|
|
|
static void free_pipeline(SpiceGstEncoder *encoder)
|
|
{
|
|
if (encoder->src_caps) {
|
|
gst_caps_unref(encoder->src_caps);
|
|
encoder->src_caps = NULL;
|
|
}
|
|
if (encoder->pipeline) {
|
|
gst_element_set_state(encoder->pipeline, GST_STATE_NULL);
|
|
gst_object_unref(encoder->appsrc);
|
|
gst_object_unref(encoder->gstenc);
|
|
gst_object_unref(encoder->appsink);
|
|
gst_object_unref(encoder->pipeline);
|
|
encoder->pipeline = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
/* ---------- Encoded frame statistics ---------- */
|
|
|
|
static inline uint32_t get_last_frame_mm_time(const SpiceGstEncoder *encoder)
|
|
{
|
|
return encoder->history[encoder->history_last].mm_time;
|
|
}
|
|
|
|
/* Returns the current bit rate based on the last
|
|
* SPICE_GST_FRAME_STATISTICS_COUNT frames.
|
|
*/
|
|
static uint64_t get_effective_bit_rate(const SpiceGstEncoder *encoder)
|
|
{
|
|
uint32_t next_mm_time = encoder->next_frame_mm_time ?
|
|
encoder->next_frame_mm_time :
|
|
get_last_frame_mm_time(encoder) +
|
|
MSEC_PER_SEC / get_source_fps(encoder);
|
|
uint32_t elapsed = next_mm_time - encoder->history[encoder->stat_first].mm_time;
|
|
return elapsed ? encoder->stat_size_sum * 8 * MSEC_PER_SEC / elapsed : 0;
|
|
}
|
|
|
|
static uint64_t get_average_encoding_time(const SpiceGstEncoder *encoder)
|
|
{
|
|
uint32_t count = encoder->history_last +
|
|
(encoder->history_last < encoder->stat_first ? SPICE_GST_HISTORY_SIZE : 0) -
|
|
encoder->stat_first + 1;
|
|
return encoder->stat_duration_sum / count;
|
|
}
|
|
|
|
static uint32_t get_average_frame_size(const SpiceGstEncoder *encoder)
|
|
{
|
|
uint32_t count = encoder->history_last +
|
|
(encoder->history_last < encoder->stat_first ? SPICE_GST_HISTORY_SIZE : 0) -
|
|
encoder->stat_first + 1;
|
|
return encoder->stat_size_sum / count;
|
|
}
|
|
|
|
/* Look for the largest frame and store it in stat_size_max to reduce how
|
|
* often we have to scan the history for the largest frame.
|
|
* Then all we need to keep things consistent is to:
|
|
* - Update stat_size_max when adding a larger frame to the history.
|
|
* - Reset stat_size_max to zero when the largest frame falls out of
|
|
* the history.
|
|
*/
|
|
static uint32_t get_maximum_frame_size(SpiceGstEncoder *encoder)
|
|
{
|
|
if (encoder->stat_size_max == 0) {
|
|
uint32_t index = encoder->history_last;
|
|
while (1) {
|
|
encoder->stat_size_max = MAX(encoder->stat_size_max,
|
|
encoder->history[index].size);
|
|
if (index == encoder->stat_first) {
|
|
break;
|
|
}
|
|
index = (index ? index : SPICE_GST_HISTORY_SIZE) - 1;
|
|
}
|
|
}
|
|
return encoder->stat_size_max;
|
|
}
|
|
|
|
/* Returns the bit rate of the specified period. from and to must be the
|
|
* mm time of the first and last frame to consider.
|
|
*/
|
|
static uint64_t get_period_bit_rate(const SpiceGstEncoder *encoder,
|
|
uint32_t from, uint32_t to)
|
|
{
|
|
uint32_t sum = 0;
|
|
uint32_t last_mm_time = 0;
|
|
uint32_t index = encoder->history_last;
|
|
while (1) {
|
|
if (encoder->history[index].mm_time == to) {
|
|
if (last_mm_time == 0) {
|
|
/* We don't know how much time elapsed between the period's
|
|
* last frame and the next so we cannot include it.
|
|
*/
|
|
sum = 1;
|
|
last_mm_time = to;
|
|
} else {
|
|
sum = encoder->history[index].size + 1;
|
|
}
|
|
|
|
} else if (encoder->history[index].mm_time == from) {
|
|
sum += encoder->history[index].size;
|
|
return (sum - 1) * 8 * MSEC_PER_SEC / (last_mm_time - from);
|
|
|
|
} else if (sum > 0) {
|
|
sum += encoder->history[index].size;
|
|
|
|
} else {
|
|
last_mm_time = encoder->history[index].mm_time;
|
|
}
|
|
|
|
if (index == encoder->history_first) {
|
|
/* This period is outside the recorded history */
|
|
spice_debug("period (%u-%u) outside known history (%u-%u)",
|
|
from, to,
|
|
encoder->history[encoder->history_first].mm_time,
|
|
encoder->history[encoder->history_last].mm_time);
|
|
return 0;
|
|
}
|
|
index = (index ? index : SPICE_GST_HISTORY_SIZE) - 1;
|
|
}
|
|
|
|
}
|
|
|
|
static void add_frame(SpiceGstEncoder *encoder, uint32_t frame_mm_time,
|
|
uint64_t duration, uint32_t size)
|
|
{
|
|
/* Update the statistics */
|
|
uint32_t count = encoder->history_last +
|
|
(encoder->history_last < encoder->stat_first ? SPICE_GST_HISTORY_SIZE : 0) -
|
|
encoder->stat_first + 1;
|
|
if (count == SPICE_GST_FRAME_STATISTICS_COUNT) {
|
|
encoder->stat_duration_sum -= encoder->history[encoder->stat_first].duration;
|
|
encoder->stat_size_sum -= encoder->history[encoder->stat_first].size;
|
|
if (encoder->stat_size_max == encoder->history[encoder->stat_first].size) {
|
|
encoder->stat_size_max = 0;
|
|
}
|
|
encoder->stat_first = (encoder->stat_first + 1) % SPICE_GST_HISTORY_SIZE;
|
|
}
|
|
encoder->stat_duration_sum += duration;
|
|
encoder->stat_size_sum += size;
|
|
if (encoder->stat_size_max > 0 && size > encoder->stat_size_max) {
|
|
encoder->stat_size_max = size;
|
|
}
|
|
|
|
/* Update the frame history */
|
|
encoder->history_last = (encoder->history_last + 1) % SPICE_GST_HISTORY_SIZE;
|
|
if (encoder->history_last == encoder->history_first) {
|
|
encoder->history_first = (encoder->history_first + 1) % SPICE_GST_HISTORY_SIZE;
|
|
}
|
|
encoder->history[encoder->history_last].mm_time = frame_mm_time;
|
|
encoder->history[encoder->history_last].duration = duration;
|
|
encoder->history[encoder->history_last].size = size;
|
|
}
|
|
|
|
|
|
/* ---------- Encoder bit rate control ---------- */
|
|
|
|
static void set_gstenc_bitrate(SpiceGstEncoder *encoder);
|
|
|
|
static void set_video_bit_rate(SpiceGstEncoder *encoder, uint64_t bit_rate)
|
|
{
|
|
if (encoder->video_bit_rate != bit_rate &&
|
|
encoder->gstenc_bitrate_is_dynamic) {
|
|
encoder->video_bit_rate = bit_rate;
|
|
set_gstenc_bitrate(encoder);
|
|
|
|
} else if (abs(bit_rate - encoder->video_bit_rate) > encoder->video_bit_rate * SPICE_GST_VIDEO_BITRATE_MARGIN) {
|
|
encoder->video_bit_rate = bit_rate;
|
|
set_pipeline_changes(encoder, SPICE_GST_VIDEO_PIPELINE_BITRATE);
|
|
}
|
|
}
|
|
|
|
static uint32_t get_min_playback_delay(SpiceGstEncoder *encoder)
|
|
{
|
|
/* Make sure the delay is large enough to send a large frame (typically
|
|
* an I frame) and an average frame. This also takes into account the
|
|
* frames dropped by the encoder bit rate control.
|
|
*/
|
|
uint32_t size = get_maximum_frame_size(encoder) + get_average_frame_size(encoder);
|
|
uint32_t send_time = ((uint64_t)MSEC_PER_SEC * 8) * size / encoder->bit_rate;
|
|
|
|
/* Also factor in the network latency with a margin for jitter. */
|
|
uint32_t net_latency = get_network_latency(encoder) * (1.0 + SPICE_GST_LATENCY_MARGIN);
|
|
|
|
return send_time + net_latency + get_average_encoding_time(encoder) / NSEC_PER_MILLISEC;
|
|
}
|
|
|
|
static void update_client_playback_delay(SpiceGstEncoder *encoder)
|
|
{
|
|
if (encoder->cbs.update_client_playback_delay) {
|
|
uint32_t min_delay = get_min_playback_delay(encoder);
|
|
encoder->cbs.update_client_playback_delay(encoder->cbs.opaque, min_delay);
|
|
}
|
|
}
|
|
|
|
static void update_next_frame_mm_time(SpiceGstEncoder *encoder)
|
|
{
|
|
uint64_t period_ns = NSEC_PER_SEC / get_source_fps(encoder);
|
|
uint64_t min_delay_ns = get_average_encoding_time(encoder);
|
|
if (min_delay_ns > period_ns) {
|
|
spice_warning("your system seems to be too slow to encode this %dx%d@%d video in real time",
|
|
encoder->width, encoder->height, get_source_fps(encoder));
|
|
}
|
|
|
|
min_delay_ns = MIN(min_delay_ns, SPICE_GST_MAX_PERIOD);
|
|
if (encoder->vbuffer_free >= 0) {
|
|
encoder->next_frame_mm_time = get_last_frame_mm_time(encoder) +
|
|
min_delay_ns / NSEC_PER_MILLISEC;
|
|
return;
|
|
}
|
|
|
|
/* Figure out how many frames to drop to not exceed the current bit rate.
|
|
* Use nanoseconds to avoid precision loss.
|
|
*/
|
|
uint64_t delay_ns = -encoder->vbuffer_free * 8 * NSEC_PER_SEC / encoder->bit_rate;
|
|
uint32_t drops = (delay_ns + period_ns - 1) / period_ns; /* round up */
|
|
spice_debug("drops=%u vbuffer %d/%d", drops, encoder->vbuffer_free,
|
|
encoder->vbuffer_size);
|
|
|
|
delay_ns = drops * period_ns + period_ns / 2;
|
|
if (delay_ns > SPICE_GST_MAX_PERIOD) {
|
|
/* Reduce the video bit rate so we don't have to drop so many frames. */
|
|
if (encoder->video_bit_rate > encoder->bit_rate * SPICE_GST_BITRATE_MARGIN) {
|
|
set_video_bit_rate(encoder, encoder->bit_rate * SPICE_GST_BITRATE_MARGIN);
|
|
} else {
|
|
set_video_bit_rate(encoder, encoder->bit_rate);
|
|
}
|
|
delay_ns = SPICE_GST_MAX_PERIOD;
|
|
}
|
|
encoder->next_frame_mm_time = get_last_frame_mm_time(encoder) +
|
|
MAX(delay_ns, min_delay_ns) / NSEC_PER_MILLISEC;
|
|
|
|
/* Drops mean a higher delay between encoded frames so update the
|
|
* playback delay.
|
|
*/
|
|
update_client_playback_delay(encoder);
|
|
}
|
|
|
|
|
|
/* ---------- Network bit rate control ---------- */
|
|
|
|
/* The maximum bit rate we will use for the current video.
|
|
*
|
|
* This is based on a 10x compression ratio which should be more than enough
|
|
* for even MJPEG to provide good quality.
|
|
*/
|
|
static uint64_t get_bit_rate_cap(const SpiceGstEncoder *encoder)
|
|
{
|
|
uint32_t raw_frame_bits = encoder->width * encoder->height * encoder->format->bpp;
|
|
return raw_frame_bits * get_source_fps(encoder) / 10;
|
|
}
|
|
|
|
static void set_bit_rate(SpiceGstEncoder *encoder, uint64_t bit_rate)
|
|
{
|
|
if (bit_rate == 0) {
|
|
/* Use the default value */
|
|
bit_rate = SPICE_GST_DEFAULT_BITRATE;
|
|
}
|
|
if (bit_rate == encoder->bit_rate) {
|
|
return;
|
|
}
|
|
if (bit_rate < SPICE_GST_MIN_BITRATE) {
|
|
/* Don't let the bit rate go too low... */
|
|
bit_rate = SPICE_GST_MIN_BITRATE;
|
|
} else if (bit_rate > encoder->bit_rate) {
|
|
/* or too high */
|
|
bit_rate = MIN(bit_rate, get_bit_rate_cap(encoder));
|
|
}
|
|
|
|
if (bit_rate < encoder->min_bit_rate) {
|
|
encoder->min_bit_rate = bit_rate;
|
|
encoder->bit_rate_step = 0;
|
|
} else if (encoder->status == SPICE_GST_BITRATE_DECREASING &&
|
|
bit_rate > encoder->bit_rate) {
|
|
encoder->min_bit_rate = encoder->bit_rate;
|
|
encoder->bit_rate_step = 0;
|
|
} else if (encoder->status != SPICE_GST_BITRATE_DECREASING &&
|
|
bit_rate < encoder->bit_rate) {
|
|
encoder->max_bit_rate = encoder->bit_rate - SPICE_GST_MIN_BITRATE;
|
|
encoder->bit_rate_step = 0;
|
|
}
|
|
encoder->increase_interval = SPICE_GST_BITRATE_UP_INTERVAL;
|
|
|
|
if (encoder->bit_rate_step == 0) {
|
|
encoder->bit_rate_step = MAX(SPICE_GST_MIN_BITRATE,
|
|
MIN(SPICE_GST_BITRATE_MAX_STEP,
|
|
(encoder->max_bit_rate - encoder->min_bit_rate) / 10));
|
|
encoder->status = (bit_rate < encoder->bit_rate) ? SPICE_GST_BITRATE_DECREASING : SPICE_GST_BITRATE_INCREASING;
|
|
if (encoder->max_bit_rate / SPICE_GST_BITRATE_MARGIN < encoder->min_bit_rate) {
|
|
/* We have sufficiently narrowed down the optimal bit rate range.
|
|
* Settle on the lower end to keep a safety margin and stop
|
|
* rocking the boat.
|
|
*/
|
|
bit_rate = encoder->min_bit_rate;
|
|
encoder->status = SPICE_GST_BITRATE_STABLE;
|
|
encoder->increase_interval = encoder->has_client_reports ? SPICE_GST_BITRATE_UP_CLIENT_STABLE : SPICE_GST_BITRATE_UP_SERVER_STABLE;
|
|
set_video_bit_rate(encoder, bit_rate);
|
|
}
|
|
}
|
|
spice_debug("%u set_bit_rate(%.3fMbps) eff %.3f %.3f-%.3f %d",
|
|
get_last_frame_mm_time(encoder) - encoder->last_change,
|
|
get_mbps(bit_rate), get_mbps(get_effective_bit_rate(encoder)),
|
|
get_mbps(encoder->min_bit_rate),
|
|
get_mbps(encoder->max_bit_rate), encoder->status);
|
|
|
|
encoder->last_change = get_last_frame_mm_time(encoder);
|
|
encoder->bit_rate = bit_rate;
|
|
/* Adjust the vbuffer size without ever increasing vbuffer_free to avoid
|
|
* sudden bit rate increases.
|
|
*/
|
|
int32_t new_size = bit_rate * SPICE_GST_VBUFFER_SIZE / MSEC_PER_SEC / 8;
|
|
if (new_size < encoder->vbuffer_size && encoder->vbuffer_free > 0) {
|
|
encoder->vbuffer_free = MAX(0, encoder->vbuffer_free + new_size - encoder->vbuffer_size);
|
|
}
|
|
encoder->vbuffer_size = new_size;
|
|
update_next_frame_mm_time(encoder);
|
|
|
|
/* Frames preceeding the bit rate change are not relevant to the current
|
|
* situation anymore.
|
|
*/
|
|
encoder->stat_first = encoder->history_last;
|
|
encoder->stat_duration_sum = encoder->history[encoder->history_last].duration;
|
|
encoder->stat_size_sum = encoder->stat_size_max = encoder->history[encoder->history_last].size;
|
|
|
|
if (bit_rate > encoder->video_bit_rate) {
|
|
set_video_bit_rate(encoder, bit_rate * SPICE_GST_BITRATE_MARGIN);
|
|
}
|
|
}
|
|
|
|
static void increase_bit_rate(SpiceGstEncoder *encoder)
|
|
{
|
|
if (get_effective_bit_rate(encoder) < encoder->bit_rate) {
|
|
/* The GStreamer encoder currently uses less bandwidth than allowed.
|
|
* So increasing the limit again makes no sense.
|
|
*/
|
|
return;
|
|
}
|
|
|
|
if (encoder->bit_rate == encoder->max_bit_rate &&
|
|
get_last_frame_mm_time(encoder) - encoder->last_change > SPICE_GST_BITRATE_UP_RESET_MAX) {
|
|
/* The maximum bit rate seems to be sustainable so it was probably
|
|
* set too low. Probe for the maximum bit rate again.
|
|
*/
|
|
encoder->max_bit_rate = get_bit_rate_cap(encoder);
|
|
encoder->status = SPICE_GST_BITRATE_INCREASING;
|
|
}
|
|
|
|
uint64_t new_bit_rate = MIN(encoder->bit_rate + encoder->bit_rate_step,
|
|
encoder->max_bit_rate);
|
|
spice_debug("increase bit rate to %.3fMbps %.3f-%.3fMbps %d",
|
|
get_mbps(new_bit_rate), get_mbps(encoder->min_bit_rate),
|
|
get_mbps(encoder->max_bit_rate), encoder->status);
|
|
set_bit_rate(encoder, new_bit_rate);
|
|
}
|
|
|
|
|
|
/* ---------- Server feedback ---------- */
|
|
|
|
/* A helper for gst_encoder_encode_frame()
|
|
*
|
|
* Checks how many frames got dropped since the last encoded frame and
|
|
* adjusts the bit rate accordingly.
|
|
*/
|
|
static inline gboolean handle_server_drops(SpiceGstEncoder *encoder,
|
|
uint32_t frame_mm_time)
|
|
{
|
|
if (encoder->server_drops == 0) {
|
|
return FALSE;
|
|
}
|
|
|
|
spice_debug("server report: got %u drops in %ums after %ums",
|
|
encoder->server_drops,
|
|
frame_mm_time - get_last_frame_mm_time(encoder),
|
|
frame_mm_time - encoder->last_change);
|
|
|
|
/* The server dropped a frame so clearly the buffer is full. */
|
|
encoder->vbuffer_free = MIN(encoder->vbuffer_free, 0);
|
|
/* Add a 0 byte frame so the time spent dropping frames is not counted as
|
|
* time during which the buffer was refilling. This implies dropping this
|
|
* frame.
|
|
*/
|
|
add_frame(encoder, frame_mm_time, 0, 0);
|
|
|
|
if (encoder->server_drops >= get_source_fps(encoder)) {
|
|
spice_debug("cut the bit rate");
|
|
uint64_t bit_rate = (encoder->bit_rate == encoder->min_bit_rate) ?
|
|
encoder->bit_rate / SPICE_GST_BITRATE_CUT :
|
|
MAX(encoder->min_bit_rate, encoder->bit_rate / SPICE_GST_BITRATE_CUT);
|
|
set_bit_rate(encoder, bit_rate);
|
|
|
|
} else {
|
|
spice_debug("reduce the bit rate");
|
|
uint64_t bit_rate = (encoder->bit_rate == encoder->min_bit_rate) ?
|
|
encoder->bit_rate / SPICE_GST_BITRATE_REDUCE :
|
|
MAX(encoder->min_bit_rate, encoder->bit_rate / SPICE_GST_BITRATE_REDUCE);
|
|
set_bit_rate(encoder, bit_rate);
|
|
}
|
|
encoder->server_drops = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
/* A helper for gst_encoder_encode_frame() */
|
|
static inline void server_increase_bit_rate(SpiceGstEncoder *encoder,
|
|
uint32_t frame_mm_time)
|
|
{
|
|
/* Let gst_encoder_client_stream_report() deal with bit rate increases if
|
|
* we receive client reports.
|
|
*/
|
|
if (!encoder->has_client_reports && encoder->server_drops == 0 &&
|
|
frame_mm_time - encoder->last_change >= encoder->increase_interval) {
|
|
increase_bit_rate(encoder);
|
|
}
|
|
}
|
|
|
|
|
|
/* ---------- GStreamer pipeline ---------- */
|
|
|
|
/* See GStreamer's part-mediatype-video-raw.txt and
|
|
* section-types-definitions.html documents.
|
|
*/
|
|
static const SpiceFormatForGStreamer format_map[] = {
|
|
/* First item is invalid.
|
|
* It's located first so the loop catch invalid values.
|
|
*/
|
|
FMT_DESC(SPICE_BITMAP_FMT_INVALID, 0, "", GST_VIDEO_FORMAT_UNKNOWN, 0, 0, 0, 0, 0),
|
|
FMT_DESC(SPICE_BITMAP_FMT_RGBA, 32, "BGRA", GST_VIDEO_FORMAT_BGRA, 24, 4321, 0xff000000, 0xff0000, 0xff00),
|
|
FMT_DESC(SPICE_BITMAP_FMT_16BIT, 16, "RGB15", GST_VIDEO_FORMAT_RGB15, 15, 4321, 0x001f, 0x03E0, 0x7C00),
|
|
/* TODO: Test the other formats under GStreamer 0.10*/
|
|
FMT_DESC(SPICE_BITMAP_FMT_32BIT, 32, "BGRx", GST_VIDEO_FORMAT_BGRx, 24, 4321, 0xff000000, 0xff0000, 0xff00),
|
|
FMT_DESC(SPICE_BITMAP_FMT_24BIT, 24, "BGR", GST_VIDEO_FORMAT_BGR, 24, 4321, 0xff0000, 0xff00, 0xff),
|
|
};
|
|
#define GSTREAMER_FORMAT_INVALID (&format_map[0])
|
|
|
|
/* A helper for spice_gst_encoder_encode_frame() */
|
|
static const SpiceFormatForGStreamer *map_format(SpiceBitmapFmt format)
|
|
{
|
|
int i;
|
|
for (i = 0; i < G_N_ELEMENTS(format_map); i++) {
|
|
if (format_map[i].spice_format == format) {
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
if (i > 2) {
|
|
spice_warning("The %d format has not been tested yet", format);
|
|
}
|
|
#endif
|
|
return &format_map[i];
|
|
}
|
|
}
|
|
|
|
return GSTREAMER_FORMAT_INVALID;
|
|
}
|
|
|
|
static void set_appsrc_caps(SpiceGstEncoder *encoder)
|
|
{
|
|
if (encoder->src_caps) {
|
|
gst_caps_unref(encoder->src_caps);
|
|
}
|
|
encoder->src_caps = gst_caps_new_simple(
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
"video/x-raw-rgb",
|
|
"bpp", G_TYPE_INT, encoder->format->bpp,
|
|
"depth", G_TYPE_INT, encoder->format->depth,
|
|
"endianness", G_TYPE_INT, encoder->format->endianness,
|
|
"red_mask", G_TYPE_INT, encoder->format->red_mask,
|
|
"green_mask", G_TYPE_INT, encoder->format->green_mask,
|
|
"blue_mask", G_TYPE_INT, encoder->format->blue_mask,
|
|
#else
|
|
"video/x-raw",
|
|
"format", G_TYPE_STRING, encoder->format->format,
|
|
#endif
|
|
"width", G_TYPE_INT, encoder->width,
|
|
"height", G_TYPE_INT, encoder->height,
|
|
"framerate", GST_TYPE_FRACTION, get_source_fps(encoder), 1,
|
|
NULL);
|
|
gst_app_src_set_caps(encoder->appsrc, encoder->src_caps);
|
|
}
|
|
|
|
static GstBusSyncReply handle_pipeline_message(GstBus *bus, GstMessage *msg, gpointer video_encoder)
|
|
{
|
|
SpiceGstEncoder *encoder = video_encoder;
|
|
|
|
if (GST_MESSAGE_TYPE(msg) == GST_MESSAGE_ERROR) {
|
|
GError *err = NULL;
|
|
gchar *debug_info = NULL;
|
|
gst_message_parse_error(msg, &err, &debug_info);
|
|
spice_warning("GStreamer error from element %s: %s",
|
|
GST_OBJECT_NAME(msg->src), err->message);
|
|
if (debug_info) {
|
|
spice_debug("debug details: %s", debug_info);
|
|
g_free(debug_info);
|
|
}
|
|
g_clear_error(&err);
|
|
|
|
/* Unblock the main thread */
|
|
pthread_mutex_lock(&encoder->outbuf_mutex);
|
|
encoder->outbuf = (VideoBuffer*)create_gst_video_buffer();
|
|
pthread_cond_signal(&encoder->outbuf_cond);
|
|
pthread_mutex_unlock(&encoder->outbuf_mutex);
|
|
}
|
|
return GST_BUS_PASS;
|
|
}
|
|
|
|
static GstFlowReturn new_sample(GstAppSink *gstappsink, gpointer video_encoder)
|
|
{
|
|
SpiceGstEncoder *encoder = (SpiceGstEncoder*)video_encoder;
|
|
SpiceGstVideoBuffer *outbuf = create_gst_video_buffer();
|
|
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
outbuf->gst_buffer = gst_app_sink_pull_buffer(encoder->appsink);
|
|
if (outbuf->gst_buffer) {
|
|
outbuf->base.data = GST_BUFFER_DATA(outbuf->gst_buffer);
|
|
outbuf->base.size = GST_BUFFER_SIZE(outbuf->gst_buffer);
|
|
}
|
|
#else
|
|
GstSample *sample = gst_app_sink_pull_sample(encoder->appsink);
|
|
if (sample) {
|
|
outbuf->gst_buffer = gst_sample_get_buffer(sample);
|
|
gst_buffer_ref(outbuf->gst_buffer);
|
|
gst_sample_unref(sample);
|
|
if (gst_buffer_map(outbuf->gst_buffer, &outbuf->map, GST_MAP_READ)) {
|
|
outbuf->base.data = outbuf->map.data;
|
|
outbuf->base.size = gst_buffer_get_size(outbuf->gst_buffer);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* Notify the main thread that the output buffer is ready */
|
|
pthread_mutex_lock(&encoder->outbuf_mutex);
|
|
encoder->outbuf = (VideoBuffer*)outbuf;
|
|
pthread_cond_signal(&encoder->outbuf_cond);
|
|
pthread_mutex_unlock(&encoder->outbuf_mutex);
|
|
|
|
return GST_FLOW_OK;
|
|
}
|
|
|
|
static const gchar* get_gst_codec_name(const SpiceGstEncoder *encoder)
|
|
{
|
|
switch (encoder->base.codec_type)
|
|
{
|
|
case SPICE_VIDEO_CODEC_TYPE_MJPEG:
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
return "ffenc_mjpeg";
|
|
#else
|
|
return "avenc_mjpeg";
|
|
#endif
|
|
case SPICE_VIDEO_CODEC_TYPE_VP8:
|
|
return "vp8enc";
|
|
case SPICE_VIDEO_CODEC_TYPE_H264:
|
|
return "x264enc";
|
|
case SPICE_VIDEO_CODEC_TYPE_VP9:
|
|
return "vp9enc";
|
|
default:
|
|
/* gstreamer_encoder_new() should have rejected this codec type */
|
|
spice_warning("unsupported codec type %d", encoder->base.codec_type);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
static gboolean create_pipeline(SpiceGstEncoder *encoder)
|
|
{
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
const gchar *converter = "ffmpegcolorspace";
|
|
#else
|
|
const gchar *converter = "videoconvert";
|
|
#endif
|
|
const gchar* gstenc_name = get_gst_codec_name(encoder);
|
|
if (!gstenc_name) {
|
|
return FALSE;
|
|
}
|
|
gchar* gstenc_opts;
|
|
switch (encoder->base.codec_type)
|
|
{
|
|
case SPICE_VIDEO_CODEC_TYPE_MJPEG:
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
gstenc_opts = g_strdup("");
|
|
#else
|
|
/* Set max-threads to ensure zero-frame latency */
|
|
gstenc_opts = g_strdup("max-threads=1");
|
|
#endif
|
|
break;
|
|
case SPICE_VIDEO_CODEC_TYPE_VP9:
|
|
case SPICE_VIDEO_CODEC_TYPE_VP8: {
|
|
/* See http://www.webmproject.org/docs/encoder-parameters/
|
|
* - Set mode/end-usage to get a constant bitrate to help with
|
|
* streaming.
|
|
* - min-quantizer ensures the bitrate does not get needlessly high.
|
|
* - resize-allowed would be useful for low bitrate situations but
|
|
* the decoder does not return a frame of the expected size so
|
|
* avoid it.
|
|
* - error-resilient minimises artifacts in case the client drops a
|
|
* frame.
|
|
* - Set lag-in-frames, deadline and cpu-used to match
|
|
* "Profile Realtime". max-latency/lag-in-frames ensures zero-frame
|
|
* latency, deadline turns on realtime behavior, cpu-used targets a
|
|
* 75% CPU usage while speed simply prioritizes encoding speed.
|
|
* - deadline is supposed to be set in microseconds but in practice
|
|
* it behaves like a boolean.
|
|
*/
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
gstenc_opts = g_strdup_printf("mode=cbr min-quantizer=10 error-resilient=true max-latency=0 speed=7");
|
|
#else
|
|
gstenc_opts = g_strdup_printf("end-usage=cbr min-quantizer=10 error-resilient=default lag-in-frames=0 deadline=1 cpu-used=4");
|
|
#endif
|
|
break;
|
|
}
|
|
case SPICE_VIDEO_CODEC_TYPE_H264:
|
|
/* - Set tune and sliced-threads to ensure a zero-frame latency
|
|
* - qp-min ensures the bitrate does not get needlessly high.
|
|
* - qp-max ensures the compression does not go so high that the video
|
|
* is unrecognizable. When that threshold is reached it is better to
|
|
* drop frames to lower the bit rate further.
|
|
* - Set speed-preset to get realtime speed.
|
|
* - Set intra-refresh to get more uniform compressed frame sizes,
|
|
* thus helping with streaming.
|
|
*/
|
|
gstenc_opts = g_strdup("byte-stream=true aud=true qp-min=15 qp-max=35 tune=4 sliced-threads=true speed-preset=ultrafast intra-refresh=true");
|
|
break;
|
|
default:
|
|
/* gstreamer_encoder_new() should have rejected this codec type */
|
|
spice_warning("unsupported codec type %d", encoder->base.codec_type);
|
|
return FALSE;
|
|
}
|
|
|
|
GError *err = NULL;
|
|
gchar *desc = g_strdup_printf("appsrc is-live=true format=time do-timestamp=true name=src !"
|
|
" %s ! %s name=encoder %s ! appsink name=sink",
|
|
converter, gstenc_name, gstenc_opts);
|
|
spice_debug("GStreamer pipeline: %s", desc);
|
|
encoder->pipeline = gst_parse_launch_full(desc, NULL, GST_PARSE_FLAG_FATAL_ERRORS, &err);
|
|
g_free(gstenc_opts);
|
|
g_free(desc);
|
|
if (!encoder->pipeline || err) {
|
|
spice_warning("GStreamer error: %s", err->message);
|
|
g_clear_error(&err);
|
|
if (encoder->pipeline) {
|
|
gst_object_unref(encoder->pipeline);
|
|
encoder->pipeline = NULL;
|
|
}
|
|
return FALSE;
|
|
}
|
|
encoder->appsrc = GST_APP_SRC(gst_bin_get_by_name(GST_BIN(encoder->pipeline), "src"));
|
|
encoder->gstenc = gst_bin_get_by_name(GST_BIN(encoder->pipeline), "encoder");
|
|
encoder->appsink = GST_APP_SINK(gst_bin_get_by_name(GST_BIN(encoder->pipeline), "sink"));
|
|
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
GstAppSinkCallbacks appsink_cbs = {NULL, NULL, &new_sample, NULL, {NULL}};
|
|
#else
|
|
GstAppSinkCallbacks appsink_cbs = {NULL, NULL, &new_sample, {NULL}};
|
|
#endif
|
|
gst_app_sink_set_callbacks(encoder->appsink, &appsink_cbs, encoder, NULL);
|
|
|
|
/* Hook into the bus so we can handle errors */
|
|
GstBus *bus = gst_element_get_bus(encoder->pipeline);
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
gst_bus_set_sync_handler(bus, handle_pipeline_message, encoder);
|
|
#else
|
|
gst_bus_set_sync_handler(bus, handle_pipeline_message, encoder, NULL);
|
|
#endif
|
|
gst_object_unref(bus);
|
|
|
|
if (encoder->base.codec_type == SPICE_VIDEO_CODEC_TYPE_MJPEG) {
|
|
/* See https://bugzilla.gnome.org/show_bug.cgi?id=753257 */
|
|
spice_debug("removing the pipeline clock");
|
|
gst_pipeline_use_clock(GST_PIPELINE(encoder->pipeline), NULL);
|
|
}
|
|
|
|
/* Figure out which parameter controls the GStreamer encoder's bitrate */
|
|
GObjectClass *class = G_OBJECT_GET_CLASS(encoder->gstenc);
|
|
encoder->gstenc_bitrate_param = g_object_class_find_property(class, "bitrate");
|
|
if (encoder->gstenc_bitrate_param == NULL) {
|
|
encoder->gstenc_bitrate_param = g_object_class_find_property(class, "target-bitrate");
|
|
}
|
|
if (encoder->gstenc_bitrate_param) {
|
|
encoder->gstenc_bitrate_is_dynamic = (encoder->gstenc_bitrate_param->flags & GST_PARAM_MUTABLE_PLAYING);
|
|
} else {
|
|
spice_warning("GStreamer error: could not find the %s bitrate parameter", gstenc_name);
|
|
}
|
|
|
|
set_pipeline_changes(encoder, SPICE_GST_VIDEO_PIPELINE_STATE |
|
|
SPICE_GST_VIDEO_PIPELINE_BITRATE |
|
|
SPICE_GST_VIDEO_PIPELINE_CAPS);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* A helper for configure_pipeline() */
|
|
static void set_gstenc_bitrate(SpiceGstEncoder *encoder)
|
|
{
|
|
GParamSpec *param = encoder->gstenc_bitrate_param;
|
|
if (!param) {
|
|
return;
|
|
}
|
|
|
|
uint64_t gst_bit_rate = encoder->video_bit_rate;
|
|
if (strstr(g_param_spec_get_blurb(param), "kbit")) {
|
|
gst_bit_rate = gst_bit_rate / 1024;
|
|
}
|
|
|
|
GObject * gobject = G_OBJECT(encoder->gstenc);
|
|
const gchar *prop = g_param_spec_get_name(param);
|
|
switch (param->value_type) {
|
|
case G_TYPE_INT: {
|
|
GParamSpecInt *range = G_PARAM_SPEC_INT(param);
|
|
gst_bit_rate = MAX(range->minimum, MIN(range->maximum, gst_bit_rate));
|
|
g_object_set(gobject, prop, (gint)gst_bit_rate, NULL);
|
|
break;
|
|
}
|
|
case G_TYPE_UINT: {
|
|
GParamSpecUInt *range = G_PARAM_SPEC_UINT(param);
|
|
gst_bit_rate = MAX(range->minimum, MIN(range->maximum, gst_bit_rate));
|
|
g_object_set(gobject, prop, (guint)gst_bit_rate, NULL);
|
|
break;
|
|
}
|
|
case G_TYPE_LONG: {
|
|
GParamSpecLong *range = G_PARAM_SPEC_LONG(param);
|
|
gst_bit_rate = MAX(range->minimum, MIN(range->maximum, gst_bit_rate));
|
|
g_object_set(gobject, prop, (glong)gst_bit_rate, NULL);
|
|
break;
|
|
}
|
|
case G_TYPE_ULONG: {
|
|
GParamSpecULong *range = G_PARAM_SPEC_ULONG(param);
|
|
gst_bit_rate = MAX(range->minimum, MIN(range->maximum, gst_bit_rate));
|
|
g_object_set(gobject, prop, (gulong)gst_bit_rate, NULL);
|
|
break;
|
|
}
|
|
case G_TYPE_INT64: {
|
|
GParamSpecInt64 *range = G_PARAM_SPEC_INT64(param);
|
|
gst_bit_rate = MAX(range->minimum, MIN(range->maximum, gst_bit_rate));
|
|
g_object_set(gobject, prop, (gint64)gst_bit_rate, NULL);
|
|
break;
|
|
}
|
|
case G_TYPE_UINT64: {
|
|
GParamSpecUInt64 *range = G_PARAM_SPEC_UINT64(param);
|
|
gst_bit_rate = MAX(range->minimum, MIN(range->maximum, gst_bit_rate));
|
|
g_object_set(gobject, prop, (guint64)gst_bit_rate, NULL);
|
|
break;
|
|
}
|
|
default:
|
|
spice_warning("the %s property has an unsupported type %" G_GSIZE_FORMAT,
|
|
prop, param->value_type);
|
|
}
|
|
spice_debug("setting the GStreamer %s to %"G_GUINT64_FORMAT, prop, gst_bit_rate);
|
|
}
|
|
|
|
/* A helper for spice_gst_encoder_encode_frame() */
|
|
static gboolean configure_pipeline(SpiceGstEncoder *encoder)
|
|
{
|
|
if (!encoder->pipeline && !create_pipeline(encoder)) {
|
|
return FALSE;
|
|
}
|
|
if (!encoder->set_pipeline) {
|
|
return TRUE;
|
|
}
|
|
|
|
/* If the pipeline state does not need to be changed it's because it is
|
|
* already in the PLAYING state. So first set it to the NULL state so it
|
|
* can be (re)configured.
|
|
*/
|
|
if (!(encoder->set_pipeline & SPICE_GST_VIDEO_PIPELINE_STATE) &&
|
|
gst_element_set_state(encoder->pipeline, GST_STATE_NULL) == GST_STATE_CHANGE_FAILURE) {
|
|
spice_debug("GStreamer error: could not stop the pipeline");
|
|
free_pipeline(encoder);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Configure the encoder bitrate */
|
|
if (encoder->set_pipeline & SPICE_GST_VIDEO_PIPELINE_BITRATE) {
|
|
set_gstenc_bitrate(encoder);
|
|
}
|
|
|
|
/* Set the source caps */
|
|
if (encoder->set_pipeline & SPICE_GST_VIDEO_PIPELINE_CAPS) {
|
|
set_appsrc_caps(encoder);
|
|
}
|
|
|
|
/* Start playing */
|
|
if (gst_element_set_state(encoder->pipeline, GST_STATE_PLAYING) == GST_STATE_CHANGE_FAILURE) {
|
|
spice_warning("GStreamer error: unable to set the pipeline to the playing state");
|
|
free_pipeline(encoder);
|
|
return FALSE;
|
|
}
|
|
|
|
encoder->set_pipeline = 0;
|
|
return TRUE;
|
|
}
|
|
|
|
/* A helper for the *_copy() functions */
|
|
static int is_chunk_stride_aligned(const SpiceBitmap *bitmap, uint32_t index)
|
|
{
|
|
SpiceChunks *chunks = bitmap->data;
|
|
if (chunks->chunk[index].len % bitmap->stride != 0) {
|
|
/* A line straddles two chunks. This is not supported */
|
|
spice_warning("chunk %d/%d contains an incomplete line, cannot copy",
|
|
index, chunks->num_chunks);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
#ifdef DO_ZERO_COPY
|
|
typedef struct {
|
|
gint refs;
|
|
SpiceGstEncoder *encoder;
|
|
gpointer opaque;
|
|
} BitmapWrapper;
|
|
|
|
static void clear_zero_copy_queue(SpiceGstEncoder *encoder, gboolean unref_queue)
|
|
{
|
|
gpointer bitmap_opaque;
|
|
while ((bitmap_opaque = g_async_queue_try_pop(encoder->unused_bitmap_opaques))) {
|
|
encoder->bitmap_unref(bitmap_opaque);
|
|
}
|
|
if (unref_queue) {
|
|
g_async_queue_unref(encoder->unused_bitmap_opaques);
|
|
}
|
|
}
|
|
|
|
static BitmapWrapper *bitmap_wrapper_new(SpiceGstEncoder *encoder, gpointer bitmap_opaque)
|
|
{
|
|
BitmapWrapper *wrapper = g_new(BitmapWrapper, 1);
|
|
wrapper->refs = 1;
|
|
wrapper->encoder = encoder;
|
|
wrapper->opaque = bitmap_opaque;
|
|
encoder->bitmap_ref(bitmap_opaque);
|
|
return wrapper;
|
|
}
|
|
|
|
static void bitmap_wrapper_unref(gpointer data)
|
|
{
|
|
BitmapWrapper *wrapper = data;
|
|
if (g_atomic_int_dec_and_test(&wrapper->refs)) {
|
|
g_async_queue_push(wrapper->encoder->unused_bitmap_opaques, wrapper->opaque);
|
|
g_free(wrapper);
|
|
}
|
|
}
|
|
|
|
|
|
/* A helper for push_raw_frame() */
|
|
static inline int zero_copy(SpiceGstEncoder *encoder,
|
|
const SpiceBitmap *bitmap, gpointer bitmap_opaque,
|
|
GstBuffer *buffer, uint32_t *chunk_index,
|
|
uint32_t *chunk_offset, uint32_t *len)
|
|
{
|
|
const SpiceChunks *chunks = bitmap->data;
|
|
while (*chunk_index < chunks->num_chunks &&
|
|
*chunk_offset >= chunks->chunk[*chunk_index].len) {
|
|
if (!is_chunk_stride_aligned(bitmap, *chunk_index)) {
|
|
return FALSE;
|
|
}
|
|
*chunk_offset -= chunks->chunk[*chunk_index].len;
|
|
(*chunk_index)++;
|
|
}
|
|
|
|
int max_block_count = gst_buffer_get_max_memory();
|
|
if (chunks->num_chunks - *chunk_index > max_block_count) {
|
|
/* There are more chunks than we can fit memory objects in a
|
|
* buffer. This will cause the buffer to merge memory objects to
|
|
* fit the extra chunks, which means doing wasteful memory copies.
|
|
* So use the zero-copy approach for the first max_mem-1 chunks, and
|
|
* let push_raw_frame() deal with the rest.
|
|
*/
|
|
max_block_count = *chunk_index + max_block_count - 1;
|
|
} else {
|
|
max_block_count = chunks->num_chunks;
|
|
}
|
|
|
|
BitmapWrapper *wrapper = NULL;
|
|
while (*len && *chunk_index < max_block_count) {
|
|
if (!is_chunk_stride_aligned(bitmap, *chunk_index)) {
|
|
return FALSE;
|
|
}
|
|
if (wrapper) {
|
|
g_atomic_int_inc(&wrapper->refs);
|
|
} else {
|
|
wrapper = bitmap_wrapper_new(encoder, bitmap_opaque);
|
|
}
|
|
uint32_t thislen = MIN(chunks->chunk[*chunk_index].len - *chunk_offset, *len);
|
|
GstMemory *mem = gst_memory_new_wrapped(GST_MEMORY_FLAG_READONLY,
|
|
chunks->chunk[*chunk_index].data,
|
|
chunks->chunk[*chunk_index].len,
|
|
*chunk_offset, thislen,
|
|
wrapper, bitmap_wrapper_unref);
|
|
gst_buffer_append_memory(buffer, mem);
|
|
*len -= thislen;
|
|
*chunk_offset = 0;
|
|
(*chunk_index)++;
|
|
}
|
|
return TRUE;
|
|
}
|
|
#else
|
|
static void clear_zero_copy_queue(SpiceGstEncoder *encoder, gboolean unref_queue)
|
|
{
|
|
/* Nothing to do */
|
|
}
|
|
|
|
#endif
|
|
|
|
/* A helper for push_raw_frame() */
|
|
static inline int line_copy(SpiceGstEncoder *encoder, const SpiceBitmap *bitmap,
|
|
uint32_t chunk_offset, uint32_t stream_stride,
|
|
uint32_t height, uint8_t *buffer)
|
|
{
|
|
uint8_t *dst = buffer;
|
|
SpiceChunks *chunks = bitmap->data;
|
|
uint32_t chunk_index = 0;
|
|
for (int l = 0; l < height; l++) {
|
|
/* We may have to move forward by more than one chunk the first
|
|
* time around. This also protects us against 0-byte chunks.
|
|
*/
|
|
while (chunk_offset >= chunks->chunk[chunk_index].len) {
|
|
if (!is_chunk_stride_aligned(bitmap, chunk_index)) {
|
|
return FALSE;
|
|
}
|
|
chunk_offset -= chunks->chunk[chunk_index].len;
|
|
chunk_index++;
|
|
}
|
|
|
|
/* Copy the line */
|
|
uint8_t *src = chunks->chunk[chunk_index].data + chunk_offset;
|
|
memcpy(dst, src, MIN(stream_stride, bitmap->stride));
|
|
dst += stream_stride;
|
|
chunk_offset += bitmap->stride;
|
|
}
|
|
spice_return_val_if_fail(dst - buffer == stream_stride * height, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
/* A helper for push_raw_frame() */
|
|
static inline int chunk_copy(SpiceGstEncoder *encoder, const SpiceBitmap *bitmap,
|
|
uint32_t chunk_index, uint32_t chunk_offset,
|
|
uint32_t len, uint8_t *dst)
|
|
{
|
|
SpiceChunks *chunks = bitmap->data;
|
|
/* Skip chunks until we find the start of the frame */
|
|
while (chunk_index < chunks->num_chunks &&
|
|
chunk_offset >= chunks->chunk[chunk_index].len) {
|
|
if (!is_chunk_stride_aligned(bitmap, chunk_index)) {
|
|
return FALSE;
|
|
}
|
|
chunk_offset -= chunks->chunk[chunk_index].len;
|
|
chunk_index++;
|
|
}
|
|
|
|
/* We can copy the frame chunk by chunk */
|
|
while (len && chunk_index < chunks->num_chunks) {
|
|
if (!is_chunk_stride_aligned(bitmap, chunk_index)) {
|
|
return FALSE;
|
|
}
|
|
uint8_t *src = chunks->chunk[chunk_index].data + chunk_offset;
|
|
uint32_t thislen = MIN(chunks->chunk[chunk_index].len - chunk_offset, len);
|
|
memcpy(dst, src, thislen);
|
|
dst += thislen;
|
|
len -= thislen;
|
|
chunk_offset = 0;
|
|
chunk_index++;
|
|
}
|
|
spice_return_val_if_fail(len == 0, FALSE);
|
|
return TRUE;
|
|
}
|
|
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
/* Dummy structure to avoid too many #ifdef in the main codepaths */
|
|
typedef struct {
|
|
gpointer memory;
|
|
} GstMapInfo;
|
|
#endif
|
|
|
|
/* A helper for push_raw_frame()
|
|
* Note: In case of error the buffer is unref-ed.
|
|
*/
|
|
static uint8_t *allocate_and_map_memory(gsize size, GstMapInfo *map, GstBuffer *buffer)
|
|
{
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
buffer->malloc_data = g_malloc(size);
|
|
GST_BUFFER_DATA(buffer) = buffer->malloc_data;
|
|
GST_BUFFER_SIZE(buffer) = size;
|
|
|
|
return GST_BUFFER_DATA(buffer);
|
|
#else
|
|
GstMemory *mem = gst_allocator_alloc(NULL, size, NULL);
|
|
if (!mem) {
|
|
gst_buffer_unref(buffer);
|
|
return NULL;
|
|
}
|
|
if (!gst_memory_map(mem, map, GST_MAP_WRITE)) {
|
|
gst_memory_unref(mem);
|
|
gst_buffer_unref(buffer);
|
|
return NULL;
|
|
}
|
|
return map->data;
|
|
#endif
|
|
}
|
|
|
|
static void unmap_and_release_memory(GstMapInfo *map, GstBuffer *buffer)
|
|
{
|
|
#ifndef HAVE_GSTREAMER_0_10
|
|
gst_memory_unmap(map->memory, map);
|
|
gst_memory_unref(map->memory);
|
|
#endif
|
|
gst_buffer_unref(buffer);
|
|
}
|
|
|
|
/* A helper for spice_gst_encoder_encode_frame() */
|
|
static int push_raw_frame(SpiceGstEncoder *encoder,
|
|
const SpiceBitmap *bitmap,
|
|
const SpiceRect *src, int top_down,
|
|
gpointer bitmap_opaque)
|
|
{
|
|
uint32_t height = src->bottom - src->top;
|
|
// GStreamer require the stream to be 4 bytes aligned
|
|
uint32_t stream_stride = GST_ROUND_UP_4((src->right - src->left) * encoder->format->bpp / 8);
|
|
uint32_t len = stream_stride * height;
|
|
GstBuffer *buffer = gst_buffer_new();
|
|
/* TODO Use GST_MAP_INFO_INIT once GStreamer 1.4.5 is no longer relevant */
|
|
GstMapInfo map = { .memory = NULL };
|
|
|
|
/* Note that we should not reorder the lines, even if top_down is false.
|
|
* It just changes the number of lines to skip at the start of the bitmap.
|
|
*/
|
|
uint32_t skip_lines = top_down ? src->top : bitmap->y - (src->bottom - 0);
|
|
uint32_t chunk_offset = bitmap->stride * skip_lines;
|
|
|
|
if (stream_stride != bitmap->stride) {
|
|
/* We have to do a line-by-line copy because for each we have to
|
|
* leave out pixels on the left or right.
|
|
*/
|
|
uint8_t *dst = allocate_and_map_memory(len, &map, buffer);
|
|
if (!dst) {
|
|
return VIDEO_ENCODER_FRAME_UNSUPPORTED;
|
|
}
|
|
|
|
chunk_offset += src->left * encoder->format->bpp / 8;
|
|
if (!line_copy(encoder, bitmap, chunk_offset, stream_stride, height, dst)) {
|
|
unmap_and_release_memory(&map, buffer);
|
|
return VIDEO_ENCODER_FRAME_UNSUPPORTED;
|
|
}
|
|
} else {
|
|
/* We can copy the bitmap chunk by chunk */
|
|
uint32_t chunk_index = 0;
|
|
#ifdef DO_ZERO_COPY
|
|
if (!zero_copy(encoder, bitmap, bitmap_opaque, buffer, &chunk_index,
|
|
&chunk_offset, &len)) {
|
|
gst_buffer_unref(buffer);
|
|
return VIDEO_ENCODER_FRAME_UNSUPPORTED;
|
|
}
|
|
/* len now contains the remaining number of bytes to copy.
|
|
* However we must avoid any write to the GstBuffer object as it
|
|
* would cause a copy of the read-only memory objects we just added.
|
|
* Fortunately we can append extra writable memory objects instead.
|
|
*/
|
|
#endif
|
|
|
|
if (len) {
|
|
uint8_t *dst = allocate_and_map_memory(len, &map, buffer);
|
|
if (!dst) {
|
|
return VIDEO_ENCODER_FRAME_UNSUPPORTED;
|
|
}
|
|
if (!chunk_copy(encoder, bitmap, chunk_index, chunk_offset,
|
|
len, dst)) {
|
|
unmap_and_release_memory(&map, buffer);
|
|
return VIDEO_ENCODER_FRAME_UNSUPPORTED;
|
|
}
|
|
}
|
|
}
|
|
#ifdef HAVE_GSTREAMER_0_10
|
|
gst_buffer_set_caps(buffer, encoder->src_caps);
|
|
#else
|
|
if (map.memory) {
|
|
gst_memory_unmap(map.memory, &map);
|
|
gst_buffer_append_memory(buffer, map.memory);
|
|
}
|
|
#endif
|
|
|
|
GstFlowReturn ret = gst_app_src_push_buffer(encoder->appsrc, buffer);
|
|
if (ret != GST_FLOW_OK) {
|
|
spice_warning("GStreamer error: unable to push source buffer (%d)", ret);
|
|
return VIDEO_ENCODER_FRAME_UNSUPPORTED;
|
|
}
|
|
|
|
return VIDEO_ENCODER_FRAME_ENCODE_DONE;
|
|
}
|
|
|
|
/* A helper for spice_gst_encoder_encode_frame() */
|
|
static int pull_compressed_buffer(SpiceGstEncoder *encoder,
|
|
VideoBuffer **outbuf)
|
|
{
|
|
pthread_mutex_lock(&encoder->outbuf_mutex);
|
|
while (!encoder->outbuf) {
|
|
pthread_cond_wait(&encoder->outbuf_cond, &encoder->outbuf_mutex);
|
|
}
|
|
*outbuf = encoder->outbuf;
|
|
encoder->outbuf = NULL;
|
|
pthread_mutex_unlock(&encoder->outbuf_mutex);
|
|
|
|
if ((*outbuf)->data) {
|
|
return VIDEO_ENCODER_FRAME_ENCODE_DONE;
|
|
}
|
|
|
|
spice_debug("failed to pull the compressed buffer");
|
|
(*outbuf)->free(*outbuf);
|
|
*outbuf = NULL;
|
|
return VIDEO_ENCODER_FRAME_UNSUPPORTED;
|
|
}
|
|
|
|
|
|
/* ---------- VideoEncoder's public API ---------- */
|
|
|
|
static void spice_gst_encoder_destroy(VideoEncoder *video_encoder)
|
|
{
|
|
SpiceGstEncoder *encoder = (SpiceGstEncoder*)video_encoder;
|
|
|
|
free_pipeline(encoder);
|
|
pthread_mutex_destroy(&encoder->outbuf_mutex);
|
|
pthread_cond_destroy(&encoder->outbuf_cond);
|
|
|
|
/* Unref any lingering bitmap opaque structures from past frames */
|
|
clear_zero_copy_queue(encoder, TRUE);
|
|
|
|
g_free(encoder);
|
|
}
|
|
|
|
static int spice_gst_encoder_encode_frame(VideoEncoder *video_encoder,
|
|
uint32_t frame_mm_time,
|
|
const SpiceBitmap *bitmap,
|
|
const SpiceRect *src, int top_down,
|
|
gpointer bitmap_opaque,
|
|
VideoBuffer **outbuf)
|
|
{
|
|
SpiceGstEncoder *encoder = (SpiceGstEncoder*)video_encoder;
|
|
g_return_val_if_fail(outbuf != NULL, VIDEO_ENCODER_FRAME_UNSUPPORTED);
|
|
*outbuf = NULL;
|
|
|
|
/* Unref the last frame's bitmap_opaque structures if any */
|
|
clear_zero_copy_queue(encoder, FALSE);
|
|
|
|
uint32_t width = src->right - src->left;
|
|
uint32_t height = src->bottom - src->top;
|
|
if (width != encoder->width || height != encoder->height ||
|
|
encoder->spice_format != bitmap->format) {
|
|
spice_debug("video format change: width %d -> %d, height %d -> %d, format %d -> %d",
|
|
encoder->width, width, encoder->height, height,
|
|
encoder->spice_format, bitmap->format);
|
|
encoder->format = map_format(bitmap->format);
|
|
if (encoder->format == GSTREAMER_FORMAT_INVALID) {
|
|
spice_warning("unable to map format type %d", bitmap->format);
|
|
encoder->errors = 4;
|
|
return VIDEO_ENCODER_FRAME_UNSUPPORTED;
|
|
}
|
|
encoder->spice_format = bitmap->format;
|
|
encoder->width = width;
|
|
encoder->height = height;
|
|
if (encoder->bit_rate == 0) {
|
|
encoder->history[0].mm_time = frame_mm_time;
|
|
encoder->max_bit_rate = get_bit_rate_cap(encoder);
|
|
encoder->min_bit_rate = SPICE_GST_MIN_BITRATE;
|
|
encoder->status = SPICE_GST_BITRATE_DECREASING;
|
|
set_bit_rate(encoder, encoder->starting_bit_rate);
|
|
encoder->vbuffer_free = 0; /* Slow start */
|
|
} else if (encoder->pipeline) {
|
|
set_pipeline_changes(encoder, SPICE_GST_VIDEO_PIPELINE_CAPS);
|
|
}
|
|
encoder->errors = 0;
|
|
} else if (encoder->errors >= 3) {
|
|
/* The pipeline keeps failing to handle the frames we send it, which
|
|
* is usually because they are too small (mouse pointer-sized).
|
|
* So give up until something changes.
|
|
*/
|
|
if (encoder->errors == 3) {
|
|
spice_debug("%s cannot compress %dx%d:%dbpp frames",
|
|
get_gst_codec_name(encoder), encoder->width,
|
|
encoder->height, encoder->format->bpp);
|
|
encoder->errors++;
|
|
}
|
|
return VIDEO_ENCODER_FRAME_UNSUPPORTED;
|
|
}
|
|
|
|
if (handle_server_drops(encoder, frame_mm_time) ||
|
|
frame_mm_time < encoder->next_frame_mm_time) {
|
|
/* Drop the frame to limit the outgoing bit rate. */
|
|
return VIDEO_ENCODER_FRAME_DROP;
|
|
}
|
|
|
|
if (!configure_pipeline(encoder)) {
|
|
encoder->errors++;
|
|
return VIDEO_ENCODER_FRAME_UNSUPPORTED;
|
|
}
|
|
|
|
uint64_t start = spice_get_monotonic_time_ns();
|
|
int rc = push_raw_frame(encoder, bitmap, src, top_down, bitmap_opaque);
|
|
if (rc == VIDEO_ENCODER_FRAME_ENCODE_DONE) {
|
|
rc = pull_compressed_buffer(encoder, outbuf);
|
|
if (rc != VIDEO_ENCODER_FRAME_ENCODE_DONE) {
|
|
/* The input buffer will be stuck in the pipeline, preventing
|
|
* later ones from being processed. Furthermore something went
|
|
* wrong with this pipeline, so it may be safer to rebuild it
|
|
* from scratch.
|
|
*/
|
|
free_pipeline(encoder);
|
|
encoder->errors++;
|
|
}
|
|
}
|
|
|
|
/* Unref the last frame's bitmap_opaque structures if any */
|
|
clear_zero_copy_queue(encoder, FALSE);
|
|
|
|
if (rc != VIDEO_ENCODER_FRAME_ENCODE_DONE) {
|
|
return rc;
|
|
}
|
|
uint32_t last_mm_time = get_last_frame_mm_time(encoder);
|
|
add_frame(encoder, frame_mm_time, spice_get_monotonic_time_ns() - start,
|
|
(*outbuf)->size);
|
|
|
|
int32_t refill = encoder->bit_rate * (frame_mm_time - last_mm_time) / MSEC_PER_SEC / 8;
|
|
encoder->vbuffer_free = MIN(encoder->vbuffer_free + refill,
|
|
encoder->vbuffer_size) - (*outbuf)->size;
|
|
|
|
server_increase_bit_rate(encoder, frame_mm_time);
|
|
update_next_frame_mm_time(encoder);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static void spice_gst_encoder_client_stream_report(VideoEncoder *video_encoder,
|
|
uint32_t num_frames,
|
|
uint32_t num_drops,
|
|
uint32_t start_frame_mm_time,
|
|
uint32_t end_frame_mm_time,
|
|
int32_t video_margin,
|
|
uint32_t audio_margin)
|
|
{
|
|
SpiceGstEncoder *encoder = (SpiceGstEncoder*)video_encoder;
|
|
encoder->has_client_reports = TRUE;
|
|
|
|
encoder->max_video_margin = MAX(encoder->max_video_margin, video_margin);
|
|
encoder->max_audio_margin = MAX(encoder->max_audio_margin, audio_margin);
|
|
int32_t margin_delta = video_margin - encoder->last_video_margin;
|
|
encoder->last_video_margin = video_margin;
|
|
|
|
uint64_t period_bit_rate = get_period_bit_rate(encoder, start_frame_mm_time, end_frame_mm_time);
|
|
spice_debug("client report: %u/%u drops in %ums margins video %3d/%3d audio %3u/%3u bw %.3f/%.3fMbps%s",
|
|
num_drops, num_frames, end_frame_mm_time - start_frame_mm_time,
|
|
video_margin, encoder->max_video_margin,
|
|
audio_margin, encoder->max_audio_margin,
|
|
get_mbps(period_bit_rate),
|
|
get_mbps(get_effective_bit_rate(encoder)),
|
|
start_frame_mm_time < encoder->last_change ? " obsolete" : "");
|
|
if (encoder->status == SPICE_GST_BITRATE_DECREASING &&
|
|
start_frame_mm_time < encoder->last_change) {
|
|
/* Some of this data predates the last bit rate reduction
|
|
* so it is obsolete.
|
|
*/
|
|
return;
|
|
}
|
|
|
|
/* We normally arrange for even the largest frames to arrive a bit over
|
|
* one period before they should be displayed.
|
|
*/
|
|
uint32_t min_margin = MSEC_PER_SEC / get_source_fps(encoder) +
|
|
get_network_latency(encoder) * SPICE_GST_LATENCY_MARGIN;
|
|
|
|
/* A low video margin indicates that the bit rate is too high. */
|
|
uint32_t score;
|
|
if (num_drops) {
|
|
score = 4;
|
|
} else if (margin_delta >= 0) {
|
|
/* The situation was bad but seems to be improving */
|
|
score = 0;
|
|
} else if (video_margin < min_margin * SPICE_GST_VIDEO_MARGIN_BAD ||
|
|
video_margin < encoder->max_video_margin * SPICE_GST_VIDEO_MARGIN_BAD) {
|
|
score = 3;
|
|
} else if (video_margin < min_margin ||
|
|
video_margin < encoder->max_video_margin * SPICE_GST_VIDEO_MARGIN_AVERAGE) {
|
|
score = 2;
|
|
} else if (video_margin < encoder->max_video_margin * SPICE_GST_VIDEO_MARGIN_GOOD) {
|
|
score = 1;
|
|
} else {
|
|
score = 0;
|
|
}
|
|
/* A fast dropping video margin is a compounding factor. */
|
|
if (margin_delta < -abs(encoder->max_video_margin) * SPICE_GST_VIDEO_DELTA_BAD) {
|
|
score += 2;
|
|
} else if (margin_delta < -abs(encoder->max_video_margin) * SPICE_GST_VIDEO_DELTA_AVERAGE) {
|
|
score += 1;
|
|
}
|
|
|
|
if (score > 3) {
|
|
spice_debug("score %u, cut the bit rate", score);
|
|
uint64_t bit_rate = (encoder->bit_rate == encoder->min_bit_rate) ?
|
|
encoder->bit_rate / SPICE_GST_BITRATE_CUT :
|
|
MAX(encoder->min_bit_rate, encoder->bit_rate / SPICE_GST_BITRATE_CUT);
|
|
set_bit_rate(encoder, bit_rate);
|
|
|
|
} else if (score == 3) {
|
|
spice_debug("score %u, reduce the bit rate", score);
|
|
uint64_t bit_rate = (encoder->bit_rate == encoder->min_bit_rate) ?
|
|
encoder->bit_rate / SPICE_GST_BITRATE_REDUCE :
|
|
MAX(encoder->min_bit_rate, encoder->bit_rate / SPICE_GST_BITRATE_REDUCE);
|
|
set_bit_rate(encoder, bit_rate);
|
|
|
|
} else if (score == 2) {
|
|
spice_debug("score %u, decrement the bit rate", score);
|
|
set_bit_rate(encoder, encoder->bit_rate - encoder->bit_rate_step);
|
|
|
|
} else if (audio_margin < encoder->max_audio_margin * SPICE_GST_AUDIO_MARGIN_BAD &&
|
|
audio_margin * SPICE_GST_AUDIO_VIDEO_RATIO < video_margin) {
|
|
/* The audio margin has decreased a lot while the video_margin
|
|
* remained higher. It may be that the video stream is starving the
|
|
* audio one of bandwidth. So reduce the bit rate.
|
|
*/
|
|
spice_debug("free some bandwidth for the audio stream");
|
|
set_bit_rate(encoder, encoder->bit_rate - encoder->bit_rate_step);
|
|
|
|
} else if (score == 1 && period_bit_rate <= encoder->bit_rate &&
|
|
encoder->status == SPICE_GST_BITRATE_INCREASING) {
|
|
/* We only increase the bit rate when score == 0 so things got worse
|
|
* since the last increase, and not because of a transient bit rate
|
|
* peak.
|
|
*/
|
|
spice_debug("degraded margin, decrement bit rate %.3f <= %.3fMbps",
|
|
get_mbps(period_bit_rate), get_mbps(encoder->bit_rate));
|
|
set_bit_rate(encoder, encoder->bit_rate - encoder->bit_rate_step);
|
|
|
|
} else if (score == 0 &&
|
|
get_last_frame_mm_time(encoder) - encoder->last_change >= encoder->increase_interval) {
|
|
/* The video margin is consistently high so increase the bit rate. */
|
|
increase_bit_rate(encoder);
|
|
}
|
|
}
|
|
|
|
static void spice_gst_encoder_notify_server_frame_drop(VideoEncoder *video_encoder)
|
|
{
|
|
SpiceGstEncoder *encoder = (SpiceGstEncoder*)video_encoder;
|
|
if (encoder->server_drops == 0) {
|
|
spice_debug("server report: getting frame drops...");
|
|
}
|
|
encoder->server_drops++;
|
|
}
|
|
|
|
static uint64_t spice_gst_encoder_get_bit_rate(VideoEncoder *video_encoder)
|
|
{
|
|
SpiceGstEncoder *encoder = (SpiceGstEncoder*)video_encoder;
|
|
return get_effective_bit_rate(encoder);
|
|
}
|
|
|
|
static void spice_gst_encoder_get_stats(VideoEncoder *video_encoder,
|
|
VideoEncoderStats *stats)
|
|
{
|
|
SpiceGstEncoder *encoder = (SpiceGstEncoder*)video_encoder;
|
|
uint64_t raw_bit_rate = encoder->width * encoder->height * encoder->format->bpp * get_source_fps(encoder);
|
|
|
|
spice_return_if_fail(stats != NULL);
|
|
stats->starting_bit_rate = encoder->starting_bit_rate;
|
|
stats->cur_bit_rate = get_effective_bit_rate(encoder);
|
|
|
|
/* Use the compression level as a proxy for the quality */
|
|
stats->avg_quality = stats->cur_bit_rate ? 100.0 - raw_bit_rate / stats->cur_bit_rate : 0;
|
|
if (stats->avg_quality < 0) {
|
|
stats->avg_quality = 0;
|
|
}
|
|
}
|
|
|
|
/* Check if ORC library can work.
|
|
* ORC library is used quite extensively by GStreamer
|
|
* to generate code dynamically. If ORC cannot work, GStreamer
|
|
* will abort(3) the entire process.
|
|
*/
|
|
static bool orc_check(void)
|
|
{
|
|
static bool orc_checked = false;
|
|
static bool orc_dynamic_code_ok = false;
|
|
|
|
if (SPICE_UNLIKELY(!orc_checked)) {
|
|
OrcCode *code = orc_code_new();
|
|
if (code) {
|
|
/* allocating 0 byte for code makes the function not crash
|
|
* but it does all the initializations and checks */
|
|
orc_code_allocate_codemem(code, 0);
|
|
orc_dynamic_code_ok = code->code != NULL;
|
|
orc_code_free(code);
|
|
}
|
|
orc_checked = true;
|
|
}
|
|
return orc_dynamic_code_ok;
|
|
}
|
|
|
|
VideoEncoder *gstreamer_encoder_new(SpiceVideoCodecType codec_type,
|
|
uint64_t starting_bit_rate,
|
|
VideoEncoderRateControlCbs *cbs,
|
|
bitmap_ref_t bitmap_ref,
|
|
bitmap_unref_t bitmap_unref)
|
|
{
|
|
SPICE_VERIFY(SPICE_GST_FRAME_STATISTICS_COUNT <= SPICE_GST_HISTORY_SIZE);
|
|
spice_return_val_if_fail(codec_type == SPICE_VIDEO_CODEC_TYPE_MJPEG ||
|
|
codec_type == SPICE_VIDEO_CODEC_TYPE_VP8 ||
|
|
codec_type == SPICE_VIDEO_CODEC_TYPE_VP9 ||
|
|
codec_type == SPICE_VIDEO_CODEC_TYPE_H264, NULL);
|
|
|
|
GError *err = NULL;
|
|
if (!gst_init_check(NULL, NULL, &err)) {
|
|
spice_warning("GStreamer error: %s", err->message);
|
|
g_clear_error(&err);
|
|
return NULL;
|
|
}
|
|
|
|
// avoid aborting the process
|
|
if (!orc_check()) {
|
|
return NULL;
|
|
}
|
|
|
|
SpiceGstEncoder *encoder = g_new0(SpiceGstEncoder, 1);
|
|
encoder->base.destroy = spice_gst_encoder_destroy;
|
|
encoder->base.encode_frame = spice_gst_encoder_encode_frame;
|
|
encoder->base.client_stream_report = spice_gst_encoder_client_stream_report;
|
|
encoder->base.notify_server_frame_drop = spice_gst_encoder_notify_server_frame_drop;
|
|
encoder->base.get_bit_rate = spice_gst_encoder_get_bit_rate;
|
|
encoder->base.get_stats = spice_gst_encoder_get_stats;
|
|
encoder->base.codec_type = codec_type;
|
|
#ifdef DO_ZERO_COPY
|
|
encoder->unused_bitmap_opaques = g_async_queue_new();
|
|
#endif
|
|
|
|
encoder->starting_bit_rate = starting_bit_rate;
|
|
encoder->cbs = *cbs;
|
|
encoder->bitmap_ref = bitmap_ref;
|
|
encoder->bitmap_unref = bitmap_unref;
|
|
encoder->format = GSTREAMER_FORMAT_INVALID;
|
|
pthread_mutex_init(&encoder->outbuf_mutex, NULL);
|
|
pthread_cond_init(&encoder->outbuf_cond, NULL);
|
|
|
|
/* All the other fields are initialized to zero by g_new0(). */
|
|
|
|
if (!create_pipeline(encoder)) {
|
|
/* Some GStreamer dependency is probably missing */
|
|
pthread_cond_destroy(&encoder->outbuf_cond);
|
|
pthread_mutex_destroy(&encoder->outbuf_mutex);
|
|
g_free(encoder);
|
|
encoder = NULL;
|
|
}
|
|
return (VideoEncoder*)encoder;
|
|
}
|