mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-12 02:15:31 +00:00
1685 lines
50 KiB
C
1685 lines
50 KiB
C
/* GStreamer
|
|
* Copyright (C) <2009> Руслан Ижбулатов <lrn1986 _at_ gmail _dot_ com>
|
|
*
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
* Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
/**
|
|
* SECTION:element-ssim
|
|
*
|
|
* The ssim calculates SSIM (Structural SIMilarity) index for two or more
|
|
* streams, for each frame.
|
|
* First stream is the original, other streams are modified (compressed) ones.
|
|
* ssim will calculate SSIM index of each frame of each modified stream, using
|
|
* original stream as a reference.
|
|
*
|
|
* The ssim accepts only YUV planar top-first data and calculates only Y-SSIM.
|
|
* All streams must have the same width, height and colorspace.
|
|
* Output streams are greyscale video streams, where bright pixels indicate
|
|
* high SSIM values, dark pixels - low SSIM values.
|
|
* The ssim also calculates mean SSIM index for each frame and emits is as a
|
|
* message.
|
|
* ssim is intended to be used with videomeasure_collector element to catch the
|
|
* events (such as mean SSIM index values) and save them into a file.
|
|
*
|
|
* <refsect2>
|
|
* <title>Example launch line</title>
|
|
* |[
|
|
* gst-launch ssim name=ssim ssim.src0 ! videoconvert ! glimagesink filesrc
|
|
* location=orig.avi ! decodebin2 ! ssim.original filesrc location=compr.avi !
|
|
* decodebin2 ! ssim.modified0
|
|
* ]| This pipeline produces a video stream that consists of SSIM frames.
|
|
* </refsect2>
|
|
*/
|
|
/* Element-Checklist-Version: 5 */
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "gstvideomeasure.h"
|
|
#include "gstvideomeasure_ssim.h"
|
|
#include <gst/audio/audio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <math.h>
|
|
|
|
#define GST_CAT_DEFAULT gst_ssim_debug
|
|
GST_DEBUG_CATEGORY_STATIC (GST_CAT_DEFAULT);
|
|
|
|
/* elementfactory information */
|
|
|
|
#define SINK_CAPS \
|
|
"video/x-raw-yuv, " \
|
|
"format = (fourcc) { I420, YV12, Y41B, Y42B } "
|
|
|
|
|
|
#define SRC_CAPS \
|
|
"video/x-raw-gray, " \
|
|
"width = (int) [ 1, MAX ], " \
|
|
"height = (int) [ 1, MAX ], " \
|
|
"framerate = (fraction) [ 0/1, MAX ], " \
|
|
"bpp = (int) 8, " \
|
|
"depth = (int) 8 "
|
|
|
|
static GstStaticPadTemplate gst_ssim_src_template =
|
|
GST_STATIC_PAD_TEMPLATE ("src_%u",
|
|
GST_PAD_SRC,
|
|
GST_PAD_SOMETIMES,
|
|
GST_STATIC_CAPS (SRC_CAPS)
|
|
);
|
|
|
|
static GstStaticPadTemplate gst_ssim_sink_original_template =
|
|
GST_STATIC_PAD_TEMPLATE ("original",
|
|
GST_PAD_SINK,
|
|
GST_PAD_REQUEST,
|
|
GST_STATIC_CAPS (SINK_CAPS)
|
|
);
|
|
|
|
static GstStaticPadTemplate gst_ssim_sink_modified_template =
|
|
GST_STATIC_PAD_TEMPLATE ("modified_%u",
|
|
GST_PAD_SINK,
|
|
GST_PAD_REQUEST,
|
|
GST_STATIC_CAPS (SINK_CAPS)
|
|
);
|
|
|
|
static void gst_ssim_class_init (GstSSimClass * klass);
|
|
static void gst_ssim_init (GstSSim * ssim);
|
|
static void gst_ssim_finalize (GObject * object);
|
|
|
|
static gboolean gst_ssim_setcaps (GstPad * pad, GstCaps * caps);
|
|
static gboolean gst_ssim_query (GstPad * pad, GstQuery * query);
|
|
static gboolean gst_ssim_src_event (GstPad * pad, GstEvent * event);
|
|
static gboolean gst_ssim_sink_event (GstPad * pad, GstEvent * event);
|
|
|
|
static GstPad *gst_ssim_request_new_pad (GstElement * element,
|
|
GstPadTemplate * temp, const gchar * unused);
|
|
static void gst_ssim_release_pad (GstElement * element, GstPad * pad);
|
|
|
|
static GstStateChangeReturn gst_ssim_change_state (GstElement * element,
|
|
GstStateChange transition);
|
|
|
|
static GstFlowReturn gst_ssim_collected (GstCollectPads * pads,
|
|
gpointer user_data);
|
|
|
|
static GstElementClass *parent_class = NULL;
|
|
|
|
GType
|
|
gst_ssim_get_type (void)
|
|
{
|
|
static GType ssim_type = 0;
|
|
|
|
if (G_UNLIKELY (ssim_type == 0)) {
|
|
static const GTypeInfo ssim_info = {
|
|
sizeof (GstSSimClass), NULL, NULL,
|
|
(GClassInitFunc) gst_ssim_class_init, NULL, NULL,
|
|
sizeof (GstSSim), 0,
|
|
(GInstanceInitFunc) gst_ssim_init,
|
|
};
|
|
|
|
ssim_type = g_type_register_static (GST_TYPE_ELEMENT, "GstSSim",
|
|
&ssim_info, 0);
|
|
GST_DEBUG_CATEGORY_INIT (GST_CAT_DEFAULT, "ssim", 0, "SSIM calculator");
|
|
}
|
|
return ssim_type;
|
|
}
|
|
|
|
static void
|
|
gst_ssim_post_message (GstSSim * ssim, GstBuffer * buffer, gfloat mssim,
|
|
gfloat lowest, gfloat highest)
|
|
{
|
|
GstMessage *m;
|
|
guint64 offset;
|
|
|
|
offset = GST_BUFFER_OFFSET (buffer);
|
|
|
|
m = gst_message_new_element (GST_OBJECT_CAST (ssim),
|
|
gst_structure_new ("SSIM",
|
|
"offset", G_TYPE_UINT64, offset,
|
|
"timestamp", GST_TYPE_CLOCK_TIME, GST_BUFFER_TIMESTAMP (buffer),
|
|
"mean", G_TYPE_FLOAT, mssim,
|
|
"lowest", G_TYPE_FLOAT, lowest,
|
|
"highest", G_TYPE_FLOAT, highest, NULL));
|
|
|
|
GST_DEBUG_OBJECT (GST_OBJECT (ssim), "Frame %" G_GINT64_FORMAT
|
|
" @ %" GST_TIME_FORMAT " mean SSIM is %f, l-h is %f-%f", offset,
|
|
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (buffer)), mssim, lowest, highest);
|
|
|
|
gst_element_post_message (GST_ELEMENT_CAST (ssim), m);
|
|
}
|
|
|
|
static GstCaps *
|
|
gst_ssim_src_getcaps (GstPad * pad)
|
|
{
|
|
GstCaps *result;
|
|
gchar *capstr;
|
|
|
|
result = gst_caps_copy (gst_pad_get_pad_template_caps (pad));
|
|
capstr = gst_caps_to_string (result);
|
|
GST_DEBUG ("getsrccaps - return static caps: %s", capstr);
|
|
g_free (capstr);
|
|
return result;
|
|
}
|
|
|
|
static GstCaps *
|
|
gst_ssim_sink_getcaps (GstPad * pad)
|
|
{
|
|
GstCaps *result = NULL;
|
|
GstSSim *ssim;
|
|
gchar *capstr;
|
|
|
|
ssim = GST_SSIM (GST_PAD_PARENT (pad));
|
|
|
|
GST_OBJECT_LOCK (ssim);
|
|
|
|
result = gst_pad_get_fixed_caps_func (pad);
|
|
capstr = gst_caps_to_string (result);
|
|
GST_DEBUG ("getsinkcaps - return caps: %s", capstr);
|
|
g_free (capstr);
|
|
|
|
GST_OBJECT_UNLOCK (ssim);
|
|
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
calculate_mu (GstSSim * ssim, gfloat * outmu, guint8 * buf)
|
|
{
|
|
gint oy, ox, iy, ix;
|
|
|
|
for (oy = 0; oy < ssim->height; oy++) {
|
|
for (ox = 0; ox < ssim->width; ox++) {
|
|
gfloat mu = 0;
|
|
gfloat elsumm;
|
|
gint weight_y_base, weight_x_base;
|
|
gint weight_offset;
|
|
gint pixel_offset;
|
|
gint winstart_y;
|
|
gint wghstart_y;
|
|
gint winend_y;
|
|
gint winstart_x;
|
|
gint wghstart_x;
|
|
gint winend_x;
|
|
gfloat weight;
|
|
gint source_offset;
|
|
|
|
source_offset = oy * ssim->width + ox;
|
|
|
|
winstart_x = ssim->windows[source_offset].x_window_start;
|
|
wghstart_x = ssim->windows[source_offset].x_weight_start;
|
|
winend_x = ssim->windows[source_offset].x_window_end;
|
|
winstart_y = ssim->windows[source_offset].y_window_start;
|
|
wghstart_y = ssim->windows[source_offset].y_weight_start;
|
|
winend_y = ssim->windows[source_offset].y_window_end;
|
|
elsumm = ssim->windows[source_offset].element_summ;
|
|
|
|
switch (ssim->windowtype) {
|
|
case 0:
|
|
for (iy = winstart_y; iy <= winend_y; iy++) {
|
|
pixel_offset = iy * ssim->width;
|
|
for (ix = winstart_x; ix <= winend_x; ix++)
|
|
mu += buf[pixel_offset + ix];
|
|
}
|
|
mu = mu / elsumm;
|
|
break;
|
|
case 1:
|
|
|
|
weight_y_base = wghstart_y - winstart_y;
|
|
weight_x_base = wghstart_x - winstart_x;
|
|
|
|
for (iy = winstart_y; iy <= winend_y; iy++) {
|
|
pixel_offset = iy * ssim->width;
|
|
weight_offset = (weight_y_base + iy) * ssim->windowsize +
|
|
weight_x_base;
|
|
for (ix = winstart_x; ix <= winend_x; ix++) {
|
|
weight = ssim->weights[weight_offset + ix];
|
|
mu += weight * buf[pixel_offset + ix];
|
|
}
|
|
}
|
|
mu = mu / elsumm;
|
|
break;
|
|
}
|
|
outmu[oy * ssim->width + ox] = mu;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
static void
|
|
calcssim_without_mu (GstSSim * ssim, guint8 * org, gfloat * orgmu, guint8 * mod,
|
|
guint8 * out, gfloat * mean, gfloat * lowest, gfloat * highest)
|
|
{
|
|
gint oy, ox, iy, ix;
|
|
gfloat cumulative_ssim = 0;
|
|
*lowest = G_MAXFLOAT;
|
|
*highest = -G_MAXFLOAT;
|
|
|
|
for (oy = 0; oy < ssim->height; oy++) {
|
|
for (ox = 0; ox < ssim->width; ox++) {
|
|
gfloat mu_o = 128, mu_m = 128;
|
|
gdouble sigma_o = 0, sigma_m = 0, sigma_om = 0;
|
|
gfloat tmp1 = 0, tmp2 = 0;
|
|
gfloat elsumm = 0;
|
|
gint weight_y_base, weight_x_base;
|
|
gint weight_offset;
|
|
gint pixel_offset;
|
|
gint winstart_y;
|
|
gint wghstart_y;
|
|
gint winend_y;
|
|
gint winstart_x;
|
|
gint wghstart_x;
|
|
gint winend_x;
|
|
gfloat weight;
|
|
gint source_offset;
|
|
|
|
source_offset = oy * ssim->width + ox;
|
|
|
|
winstart_x = ssim->windows[source_offset].x_window_start;
|
|
wghstart_x = ssim->windows[source_offset].x_weight_start;
|
|
winend_x = ssim->windows[source_offset].x_window_end;
|
|
winstart_y = ssim->windows[source_offset].y_window_start;
|
|
wghstart_y = ssim->windows[source_offset].y_weight_start;
|
|
winend_y = ssim->windows[source_offset].y_window_end;
|
|
elsumm = ssim->windows[source_offset].element_summ;
|
|
|
|
weight_y_base = wghstart_y - winstart_y;
|
|
weight_x_base = wghstart_x - winstart_x;
|
|
switch (ssim->windowtype) {
|
|
case 0:
|
|
for (iy = winstart_y; iy <= winend_y; iy++) {
|
|
guint8 *org_with_offset, *mod_with_offset;
|
|
pixel_offset = iy * ssim->width;
|
|
org_with_offset = &org[pixel_offset];
|
|
mod_with_offset = &mod[pixel_offset];
|
|
for (ix = winstart_x; ix <= winend_x; ix++) {
|
|
tmp1 = org_with_offset[ix] - mu_o;
|
|
sigma_o += tmp1 * tmp1;
|
|
tmp2 = mod_with_offset[ix] - mu_m;
|
|
sigma_m += tmp2 * tmp2;
|
|
sigma_om += tmp1 * tmp2;
|
|
}
|
|
}
|
|
break;
|
|
case 1:
|
|
|
|
weight_y_base = wghstart_y - winstart_y;
|
|
weight_x_base = wghstart_x - winstart_x;
|
|
|
|
for (iy = winstart_y; iy <= winend_y; iy++) {
|
|
guint8 *org_with_offset, *mod_with_offset;
|
|
gfloat *weights_with_offset;
|
|
gfloat wt1, wt2;
|
|
pixel_offset = iy * ssim->width;
|
|
weight_offset = (weight_y_base + iy) * ssim->windowsize +
|
|
weight_x_base;
|
|
org_with_offset = &org[pixel_offset];
|
|
mod_with_offset = &mod[pixel_offset];
|
|
weights_with_offset = &ssim->weights[weight_offset];
|
|
for (ix = winstart_x; ix <= winend_x; ix++) {
|
|
weight = weights_with_offset[ix];
|
|
tmp1 = org_with_offset[ix] - mu_o;
|
|
tmp2 = mod_with_offset[ix] - mu_m;
|
|
wt1 = weight * tmp1;
|
|
wt2 = weight * tmp2;
|
|
sigma_o += wt1 * tmp1;
|
|
sigma_m += wt2 * tmp2;
|
|
sigma_om += wt1 * tmp2;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
sigma_o = sqrt (sigma_o / elsumm);
|
|
sigma_m = sqrt (sigma_m / elsumm);
|
|
sigma_om = sigma_om / elsumm;
|
|
tmp1 = (2 * mu_o * mu_m + ssim->const1) * (2 * sigma_om + ssim->const2) /
|
|
((mu_o * mu_o + mu_m * mu_m + ssim->const1) *
|
|
(sigma_o * sigma_o + sigma_m * sigma_m + ssim->const2));
|
|
|
|
/* SSIM can go negative, that's why it is
|
|
127 + index * 128 instead of index * 255 */
|
|
out[oy * ssim->width + ox] = 127 + tmp1 * 128;
|
|
*lowest = MIN (*lowest, tmp1);
|
|
*highest = MAX (*highest, tmp1);
|
|
cumulative_ssim += tmp1;
|
|
}
|
|
}
|
|
*mean = cumulative_ssim / (ssim->width * ssim->height);
|
|
}
|
|
|
|
static void
|
|
calcssim_canonical (GstSSim * ssim, guint8 * org, gfloat * orgmu, guint8 * mod,
|
|
guint8 * out, gfloat * mean, gfloat * lowest, gfloat * highest)
|
|
{
|
|
gint oy, ox, iy, ix;
|
|
gfloat cumulative_ssim = 0;
|
|
*lowest = G_MAXFLOAT;
|
|
*highest = -G_MAXFLOAT;
|
|
|
|
for (oy = 0; oy < ssim->height; oy++) {
|
|
for (ox = 0; ox < ssim->width; ox++) {
|
|
gfloat mu_o = 0, mu_m = 0;
|
|
gdouble sigma_o = 0, sigma_m = 0, sigma_om = 0;
|
|
gfloat tmp1, tmp2;
|
|
gfloat elsumm = 0;
|
|
gint weight_y_base, weight_x_base;
|
|
gint weight_offset;
|
|
gint pixel_offset;
|
|
gint winstart_y;
|
|
gint wghstart_y;
|
|
gint winend_y;
|
|
gint winstart_x;
|
|
gint wghstart_x;
|
|
gint winend_x;
|
|
gfloat weight;
|
|
gint source_offset;
|
|
|
|
source_offset = oy * ssim->width + ox;
|
|
|
|
winstart_x = ssim->windows[source_offset].x_window_start;
|
|
wghstart_x = ssim->windows[source_offset].x_weight_start;
|
|
winend_x = ssim->windows[source_offset].x_window_end;
|
|
winstart_y = ssim->windows[source_offset].y_window_start;
|
|
wghstart_y = ssim->windows[source_offset].y_weight_start;
|
|
winend_y = ssim->windows[source_offset].y_window_end;
|
|
elsumm = ssim->windows[source_offset].element_summ;
|
|
|
|
switch (ssim->windowtype) {
|
|
case 0:
|
|
for (iy = winstart_y; iy <= winend_y; iy++) {
|
|
pixel_offset = iy * ssim->width;
|
|
for (ix = winstart_x; ix <= winend_x; ix++) {
|
|
mu_m += mod[pixel_offset + ix];
|
|
}
|
|
}
|
|
mu_m = mu_m / elsumm;
|
|
mu_o = orgmu[oy * ssim->width + ox];
|
|
for (iy = winstart_y; iy <= winend_y; iy++) {
|
|
pixel_offset = iy * ssim->width;
|
|
for (ix = winstart_x; ix <= winend_x; ix++) {
|
|
tmp1 = org[pixel_offset + ix] - mu_o;
|
|
tmp2 = mod[pixel_offset + ix] - mu_m;
|
|
sigma_o += tmp1 * tmp1;
|
|
sigma_m += tmp2 * tmp2;
|
|
sigma_om += tmp1 * tmp2;
|
|
}
|
|
}
|
|
break;
|
|
case 1:
|
|
|
|
weight_y_base = wghstart_y - winstart_y;
|
|
weight_x_base = wghstart_x - winstart_x;
|
|
|
|
for (iy = winstart_y; iy <= winend_y; iy++) {
|
|
pixel_offset = iy * ssim->width;
|
|
weight_offset = (weight_y_base + iy) * ssim->windowsize +
|
|
weight_x_base;
|
|
for (ix = winstart_x; ix <= winend_x; ix++) {
|
|
weight = ssim->weights[weight_offset + ix];
|
|
mu_o += weight * org[pixel_offset + ix];
|
|
mu_m += weight * mod[pixel_offset + ix];
|
|
}
|
|
}
|
|
mu_m = mu_m / elsumm;
|
|
mu_o = orgmu[oy * ssim->width + ox];
|
|
for (iy = winstart_y; iy <= winend_y; iy++) {
|
|
gfloat *weights_with_offset;
|
|
guint8 *org_with_offset, *mod_with_offset;
|
|
gfloat wt1, wt2;
|
|
pixel_offset = iy * ssim->width;
|
|
weight_offset = (weight_y_base + iy) * ssim->windowsize +
|
|
weight_x_base;
|
|
weights_with_offset = &ssim->weights[weight_offset];
|
|
org_with_offset = &org[pixel_offset];
|
|
mod_with_offset = &mod[pixel_offset];
|
|
for (ix = winstart_x; ix <= winend_x; ix++) {
|
|
weight = weights_with_offset[ix];
|
|
tmp1 = org_with_offset[ix] - mu_o;
|
|
tmp2 = mod_with_offset[ix] - mu_m;
|
|
wt1 = weight * tmp1;
|
|
wt2 = weight * tmp2;
|
|
sigma_o += wt1 * tmp1;
|
|
sigma_m += wt2 * tmp2;
|
|
sigma_om += wt1 * tmp2;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
sigma_o = sqrt (sigma_o / elsumm);
|
|
sigma_m = sqrt (sigma_m / elsumm);
|
|
sigma_om = sigma_om / elsumm;
|
|
tmp1 = (2 * mu_o * mu_m + ssim->const1) * (2 * sigma_om + ssim->const2) /
|
|
((mu_o * mu_o + mu_m * mu_m + ssim->const1) *
|
|
(sigma_o * sigma_o + sigma_m * sigma_m + ssim->const2));
|
|
|
|
/* SSIM can go negative, that's why it is
|
|
127 + index * 128 instead of index * 255 */
|
|
out[oy * ssim->width + ox] = 127 + tmp1 * 128;
|
|
*lowest = MIN (*lowest, tmp1);
|
|
*highest = MAX (*highest, tmp1);
|
|
cumulative_ssim += tmp1;
|
|
}
|
|
}
|
|
*mean = cumulative_ssim / (ssim->width * ssim->height);
|
|
}
|
|
|
|
|
|
/* the first caps we receive on any of the sinkpads will define the caps for all
|
|
* the other sinkpads because we can only measure streams with the same caps.
|
|
*/
|
|
static gboolean
|
|
gst_ssim_setcaps (GstPad * pad, GstCaps * caps)
|
|
{
|
|
GstSSim *ssim;
|
|
GList *pads;
|
|
const char *media_type;
|
|
GstStructure *capsstr;
|
|
gint width, height, fps_n, fps_d;
|
|
guint32 fourcc;
|
|
|
|
ssim = GST_SSIM (GST_PAD_PARENT (pad));
|
|
|
|
GST_DEBUG_OBJECT (ssim, "setting caps on pad %p,%s to %" GST_PTR_FORMAT, pad,
|
|
GST_PAD_NAME (pad), caps);
|
|
|
|
capsstr = gst_caps_get_structure (caps, 0);
|
|
gst_structure_get_int (capsstr, "width", &width);
|
|
gst_structure_get_int (capsstr, "height", &height);
|
|
gst_structure_get_fraction (capsstr, "framerate", &fps_n, &fps_d);
|
|
gst_structure_get_fourcc (capsstr, "format", &fourcc);
|
|
|
|
GST_OBJECT_LOCK (ssim);
|
|
|
|
/* Sink caps are stored only once. At the moment it doesn't feel
|
|
* right to measure streams with variable caps.
|
|
*/
|
|
if (G_UNLIKELY (!ssim->sinkcaps)) {
|
|
GstStructure *newstr;
|
|
GValue list = { 0, }
|
|
, fourcc = {
|
|
0,};
|
|
|
|
g_value_init (&list, GST_TYPE_LIST);
|
|
g_value_init (&fourcc, GST_TYPE_FOURCC);
|
|
|
|
gst_value_set_fourcc (&fourcc, GST_MAKE_FOURCC ('I', '4', '2', '0'));
|
|
gst_value_list_append_value (&list, &fourcc);
|
|
gst_value_set_fourcc (&fourcc, GST_MAKE_FOURCC ('Y', 'V', '1', '2'));
|
|
gst_value_list_append_value (&list, &fourcc);
|
|
gst_value_set_fourcc (&fourcc, GST_MAKE_FOURCC ('Y', '4', '1', 'B'));
|
|
gst_value_list_append_value (&list, &fourcc);
|
|
gst_value_set_fourcc (&fourcc, GST_MAKE_FOURCC ('Y', '4', '2', 'B'));
|
|
gst_value_list_append_value (&list, &fourcc);
|
|
|
|
newstr = gst_structure_new ("video/x-raw-yuv", NULL);
|
|
gst_structure_set (newstr, "width", G_TYPE_INT, width, NULL);
|
|
gst_structure_set (newstr, "height", G_TYPE_INT, height, NULL);
|
|
gst_structure_set_value (newstr, "format", &list);
|
|
|
|
ssim->sinkcaps = gst_caps_new_full (newstr, NULL);
|
|
|
|
g_value_unset (&list);
|
|
g_value_unset (&fourcc);
|
|
}
|
|
|
|
if (G_UNLIKELY (!ssim->srccaps)) {
|
|
GstStructure *newstr;
|
|
|
|
newstr = gst_structure_new ("video/x-raw-gray", NULL);
|
|
gst_structure_set (newstr, "width", G_TYPE_INT, width, NULL);
|
|
gst_structure_set (newstr, "height", G_TYPE_INT, height, NULL);
|
|
gst_structure_set (newstr, "framerate", GST_TYPE_FRACTION, fps_n, fps_d,
|
|
NULL);
|
|
/* Calculates SSIM only for Y channel, hence the output is monochrome.
|
|
* TODO: an option (a mask?) to calculate SSIM for more than one channel,
|
|
* will probably output RGB, one metric per channel...that would
|
|
* look kinda funny :)
|
|
*/
|
|
gst_structure_set (newstr, "bpp", G_TYPE_INT, 8, "depth", G_TYPE_INT, 8,
|
|
NULL);
|
|
|
|
ssim->srccaps = gst_caps_new_full (newstr, NULL);
|
|
}
|
|
|
|
pads = GST_ELEMENT (ssim)->pads;
|
|
while (pads) {
|
|
GstPadDirection direction;
|
|
GstPad *otherpad = GST_PAD (pads->data);
|
|
direction = gst_pad_get_direction (otherpad);
|
|
|
|
GST_DEBUG_OBJECT (ssim, "checking caps on pad %p", otherpad);
|
|
if (direction == GST_PAD_SINK) {
|
|
gchar *capstr;
|
|
capstr = gst_caps_to_string (GST_PAD_CAPS (otherpad));
|
|
GST_DEBUG_OBJECT (ssim, "old caps on pad %p,%s were %s", otherpad,
|
|
GST_PAD_NAME (otherpad), capstr);
|
|
g_free (capstr);
|
|
gst_caps_replace (&GST_PAD_CAPS (otherpad), ssim->sinkcaps);
|
|
capstr = gst_caps_to_string (ssim->sinkcaps);
|
|
GST_DEBUG_OBJECT (ssim, "new caps on pad %p,%s are %s", otherpad,
|
|
GST_PAD_NAME (otherpad), capstr);
|
|
g_free (capstr);
|
|
} else if (direction == GST_PAD_SRC) {
|
|
gst_caps_replace (&GST_PAD_CAPS (otherpad), ssim->srccaps);
|
|
}
|
|
pads = g_list_next (pads);
|
|
}
|
|
|
|
/* parse caps now */
|
|
media_type = gst_structure_get_name (capsstr);
|
|
GST_DEBUG_OBJECT (ssim, "media type is %s", media_type);
|
|
if (strcmp (media_type, "video/x-raw-yuv") == 0) {
|
|
ssim->width = width;
|
|
ssim->height = height;
|
|
ssim->frame_rate = fps_n;
|
|
ssim->frame_rate_base = fps_d;
|
|
|
|
GST_INFO_OBJECT (ssim, "parse_caps sets ssim to yuv format "
|
|
"%d, %dx%d, %d/%d fps", fourcc, ssim->width, ssim->height,
|
|
ssim->frame_rate, ssim->frame_rate_base);
|
|
|
|
/* Only planar formats are supported.
|
|
* TODO: implement support for interleaved formats
|
|
* Only YUV formats are supported. There's no sense in calculating the
|
|
* index for R, G or B channels separately.
|
|
*/
|
|
switch (fourcc) {
|
|
case GST_MAKE_FOURCC ('I', '4', '2', '0'):
|
|
case GST_MAKE_FOURCC ('Y', 'V', '1', '2'):
|
|
case GST_MAKE_FOURCC ('Y', '4', '1', 'B'):
|
|
case GST_MAKE_FOURCC ('Y', '4', '2', 'B'):
|
|
break;
|
|
default:
|
|
goto not_supported;
|
|
}
|
|
|
|
} else {
|
|
goto not_supported;
|
|
}
|
|
|
|
GST_OBJECT_UNLOCK (ssim);
|
|
return TRUE;
|
|
/* ERRORS */
|
|
not_supported:
|
|
{
|
|
GST_OBJECT_UNLOCK (ssim);
|
|
GST_DEBUG_OBJECT (ssim, "unsupported format set as caps");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
gst_ssim_query_latency (GstSSim * ssim, GstQuery * query)
|
|
{
|
|
GstClockTime min, max;
|
|
gboolean live;
|
|
gboolean res;
|
|
GstIterator *it;
|
|
gboolean done;
|
|
|
|
res = TRUE;
|
|
done = FALSE;
|
|
|
|
live = FALSE;
|
|
min = 0;
|
|
max = GST_CLOCK_TIME_NONE;
|
|
|
|
/* Take maximum of all latency values */
|
|
it = gst_element_iterate_sink_pads (GST_ELEMENT_CAST (ssim));
|
|
while (!done) {
|
|
GstIteratorResult ires;
|
|
|
|
gpointer item;
|
|
|
|
ires = gst_iterator_next (it, &item);
|
|
switch (ires) {
|
|
case GST_ITERATOR_DONE:
|
|
done = TRUE;
|
|
break;
|
|
case GST_ITERATOR_OK:
|
|
{
|
|
GstPad *pad = GST_PAD_CAST (item);
|
|
GstQuery *peerquery;
|
|
GstClockTime min_cur, max_cur;
|
|
gboolean live_cur;
|
|
|
|
peerquery = gst_query_new_latency ();
|
|
|
|
/* Ask peer for latency */
|
|
res &= gst_pad_peer_query (pad, peerquery);
|
|
|
|
/* take max from all valid return values */
|
|
if (res) {
|
|
gst_query_parse_latency (peerquery, &live_cur, &min_cur, &max_cur);
|
|
|
|
if (live_cur) {
|
|
if (min_cur > min)
|
|
min = min_cur;
|
|
|
|
if (max == GST_CLOCK_TIME_NONE)
|
|
max = max_cur;
|
|
else if (max_cur < max)
|
|
max = max_cur;
|
|
|
|
live = TRUE;
|
|
}
|
|
}
|
|
|
|
gst_query_unref (peerquery);
|
|
gst_object_unref (pad);
|
|
break;
|
|
}
|
|
case GST_ITERATOR_RESYNC:
|
|
live = FALSE;
|
|
min = 0;
|
|
max = GST_CLOCK_TIME_NONE;
|
|
res = TRUE;
|
|
gst_iterator_resync (it);
|
|
break;
|
|
default:
|
|
res = FALSE;
|
|
done = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
gst_iterator_free (it);
|
|
|
|
if (res) {
|
|
/* store the results */
|
|
GST_DEBUG_OBJECT (ssim, "Calculated total latency: live %s, min %"
|
|
GST_TIME_FORMAT ", max %" GST_TIME_FORMAT,
|
|
(live ? "yes" : "no"), GST_TIME_ARGS (min), GST_TIME_ARGS (max));
|
|
gst_query_set_latency (query, live, min, max);
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
static gboolean
|
|
gst_ssim_query_duration (GstSSim * ssim, GstQuery * query)
|
|
{
|
|
gint64 max, min;
|
|
gboolean res;
|
|
GstFormat format;
|
|
GstIterator *it;
|
|
gboolean done;
|
|
|
|
/* parse format */
|
|
gst_query_parse_duration (query, &format, NULL);
|
|
|
|
max = -1;
|
|
min = G_MAXINT64;
|
|
res = TRUE;
|
|
done = FALSE;
|
|
|
|
it = gst_element_iterate_sink_pads (GST_ELEMENT_CAST (ssim));
|
|
while (!done) {
|
|
GstIteratorResult ires;
|
|
|
|
gpointer item;
|
|
|
|
ires = gst_iterator_next (it, &item);
|
|
switch (ires) {
|
|
case GST_ITERATOR_DONE:
|
|
done = TRUE;
|
|
break;
|
|
case GST_ITERATOR_OK:
|
|
{
|
|
GstPad *pad = GST_PAD_CAST (item);
|
|
|
|
gint64 duration;
|
|
|
|
/* ask sink peer for duration */
|
|
res &= gst_pad_query_peer_duration (pad, &format, &duration);
|
|
/* take min&max from all valid return values */
|
|
if (res) {
|
|
/* valid unknown length, stop searching */
|
|
if (duration == -1) {
|
|
max = duration;
|
|
done = TRUE;
|
|
}
|
|
/* else see if bigger than current max */
|
|
else {
|
|
if (duration > max)
|
|
max = duration;
|
|
if (duration < min)
|
|
min = duration;
|
|
}
|
|
}
|
|
gst_object_unref (pad);
|
|
break;
|
|
}
|
|
case GST_ITERATOR_RESYNC:
|
|
max = -1;
|
|
min = G_MAXINT64;
|
|
res = TRUE;
|
|
gst_iterator_resync (it);
|
|
break;
|
|
default:
|
|
res = FALSE;
|
|
done = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
gst_iterator_free (it);
|
|
|
|
if (res) {
|
|
/* and store the max */
|
|
GST_DEBUG_OBJECT (ssim, "Total duration in format %s: %"
|
|
GST_TIME_FORMAT, gst_format_get_name (format), GST_TIME_ARGS (min));
|
|
gst_query_set_duration (query, format, min);
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
gst_ssim_query (GstPad * pad, GstQuery * query)
|
|
{
|
|
GstSSim *ssim = GST_SSIM (gst_pad_get_parent (pad));
|
|
gboolean res = FALSE;
|
|
|
|
switch (GST_QUERY_TYPE (query)) {
|
|
case GST_QUERY_POSITION:
|
|
{
|
|
GstFormat format;
|
|
|
|
gst_query_parse_position (query, &format, NULL);
|
|
|
|
switch (format) {
|
|
case GST_FORMAT_TIME:
|
|
/* FIXME, bring to stream time, might be tricky */
|
|
gst_query_set_position (query, format, ssim->timestamp);
|
|
res = TRUE;
|
|
break;
|
|
case GST_FORMAT_DEFAULT:
|
|
gst_query_set_position (query, format, ssim->offset);
|
|
res = TRUE;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
case GST_QUERY_DURATION:
|
|
res = gst_ssim_query_duration (ssim, query);
|
|
break;
|
|
case GST_QUERY_LATENCY:
|
|
res = gst_ssim_query_latency (ssim, query);
|
|
break;
|
|
default:
|
|
/* FIXME, needs a custom query handler because we have multiple
|
|
* sinkpads
|
|
*/
|
|
res = gst_pad_query_default (pad, query);
|
|
break;
|
|
}
|
|
|
|
gst_object_unref (ssim);
|
|
return res;
|
|
}
|
|
|
|
static gboolean
|
|
forward_event_func (GstPad * pad, GValue * ret, GstEvent * event)
|
|
{
|
|
gst_event_ref (event);
|
|
GST_LOG_OBJECT (pad, "About to send event %s", GST_EVENT_TYPE_NAME (event));
|
|
if (!gst_pad_push_event (pad, event)) {
|
|
g_value_set_boolean (ret, FALSE);
|
|
GST_LOG_OBJECT (pad, "Sending event %p (%s) failed.",
|
|
event, GST_EVENT_TYPE_NAME (event));
|
|
} else {
|
|
GST_LOG_OBJECT (pad, "Sent event %p (%s).",
|
|
event, GST_EVENT_TYPE_NAME (event));
|
|
}
|
|
gst_object_unref (pad);
|
|
return TRUE;
|
|
}
|
|
|
|
/* forwards the event to all sinkpads, takes ownership of the
|
|
* event
|
|
*
|
|
* Returns: TRUE if the event could be forwarded on all
|
|
* sinkpads.
|
|
*/
|
|
static gboolean
|
|
forward_event (GstSSim * ssim, GstEvent * event)
|
|
{
|
|
GstIterator *it;
|
|
GValue vret = { 0 };
|
|
|
|
GST_LOG_OBJECT (ssim, "Forwarding event %p (%s)", event,
|
|
GST_EVENT_TYPE_NAME (event));
|
|
|
|
g_value_init (&vret, G_TYPE_BOOLEAN);
|
|
g_value_set_boolean (&vret, TRUE);
|
|
it = gst_element_iterate_sink_pads (GST_ELEMENT_CAST (ssim));
|
|
gst_iterator_fold (it, (GstIteratorFoldFunction) forward_event_func, &vret,
|
|
event);
|
|
gst_iterator_free (it);
|
|
gst_event_unref (event);
|
|
|
|
return g_value_get_boolean (&vret);
|
|
}
|
|
|
|
static gboolean
|
|
gst_ssim_src_event (GstPad * pad, GstEvent * event)
|
|
{
|
|
GstSSim *ssim;
|
|
gboolean result;
|
|
|
|
ssim = GST_SSIM (gst_pad_get_parent (pad));
|
|
|
|
switch (GST_EVENT_TYPE (event)) {
|
|
case GST_EVENT_QOS:
|
|
/* QoS might be tricky */
|
|
result = FALSE;
|
|
break;
|
|
case GST_EVENT_SEEK:
|
|
{
|
|
GstSeekFlags flags;
|
|
GstSeekType curtype;
|
|
gint64 cur;
|
|
|
|
/* parse the seek parameters */
|
|
gst_event_parse_seek (event, &ssim->segment_rate, NULL, &flags, &curtype,
|
|
&cur, NULL, NULL);
|
|
|
|
/* check if we are flushing */
|
|
if (flags & GST_SEEK_FLAG_FLUSH) {
|
|
/* make sure we accept nothing anymore and return WRONG_STATE */
|
|
gst_collect_pads_set_flushing (ssim->collect, TRUE);
|
|
|
|
/* flushing seek, start flush downstream, the flush will be done
|
|
* when all pads received a FLUSH_STOP. */
|
|
gst_pad_push_event (pad, gst_event_new_flush_start ());
|
|
}
|
|
/* now wait for the collected to be finished and mark a new
|
|
* segment */
|
|
GST_OBJECT_LOCK (ssim->collect);
|
|
if (curtype == GST_SEEK_TYPE_SET)
|
|
ssim->segment_position = cur;
|
|
else
|
|
ssim->segment_position = 0;
|
|
{
|
|
GstSSimOutputContext *c;
|
|
gint i = 0;
|
|
for (i = 0; i < ssim->src->len; i++) {
|
|
c = (GstSSimOutputContext *) g_ptr_array_index (ssim->src, i);
|
|
c->segment_pending = TRUE;
|
|
}
|
|
}
|
|
GST_OBJECT_UNLOCK (ssim->collect);
|
|
|
|
result = forward_event (ssim, event);
|
|
break;
|
|
}
|
|
case GST_EVENT_NAVIGATION:
|
|
/* navigation is rather pointless. */
|
|
result = FALSE;
|
|
break;
|
|
default:
|
|
/* just forward the rest for now */
|
|
result = forward_event (ssim, event);
|
|
break;
|
|
}
|
|
gst_object_unref (ssim);
|
|
|
|
return result;
|
|
}
|
|
|
|
static gboolean
|
|
gst_ssim_sink_event (GstPad * pad, GstEvent * event)
|
|
{
|
|
GstSSim *ssim;
|
|
gboolean ret;
|
|
|
|
ssim = GST_SSIM (gst_pad_get_parent (pad));
|
|
|
|
GST_DEBUG ("Got %s event on pad %s:%s", GST_EVENT_TYPE_NAME (event),
|
|
GST_DEBUG_PAD_NAME (pad));
|
|
|
|
switch (GST_EVENT_TYPE (event)) {
|
|
case GST_EVENT_NEWSEGMENT:
|
|
{
|
|
gboolean update;
|
|
gdouble rate;
|
|
gdouble applied_rate;
|
|
GstFormat format;
|
|
gint64 start;
|
|
gint64 stop;
|
|
gint64 position;
|
|
gst_event_parse_new_segment_full (event, &update, &rate, &applied_rate,
|
|
&format, &start, &stop, &position);
|
|
GST_DEBUG ("NEWSEGMENTEVENT: update(%d), rate(%f), app_rate(%f), "
|
|
"format(%d), start(%" GST_TIME_FORMAT ") stop(%" GST_TIME_FORMAT ") "
|
|
"position(%" GST_TIME_FORMAT ")", update, rate, applied_rate, format,
|
|
GST_TIME_ARGS (start), GST_TIME_ARGS (stop),
|
|
GST_TIME_ARGS (position));
|
|
break;
|
|
}
|
|
case GST_EVENT_FLUSH_STOP:
|
|
/* mark a pending new segment. This event is synchronized
|
|
* with the streaming thread so we can safely update the
|
|
* variable without races. It's somewhat weird because we
|
|
* assume the collectpads forwarded the FLUSH_STOP past us
|
|
* and downstream (using our source pad, the bastard!).
|
|
*/
|
|
{
|
|
GstSSimOutputContext *c;
|
|
gint i = 0;
|
|
for (i = 0; i < ssim->src->len; i++) {
|
|
c = (GstSSimOutputContext *) g_ptr_array_index (ssim->src, i);
|
|
c->segment_pending = TRUE;
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
/* now GstCollectPads can take care of the rest, e.g. EOS */
|
|
GST_DEBUG ("Dispatching %s event on pad %s:%s", GST_EVENT_TYPE_NAME (event),
|
|
GST_DEBUG_PAD_NAME (pad));
|
|
ret = ssim->collect_event (pad, event);
|
|
GST_DEBUG ("Event %s on pad %s:%s is dispatched", GST_EVENT_TYPE_NAME (event),
|
|
GST_DEBUG_PAD_NAME (pad));
|
|
gst_object_unref (ssim);
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
gst_ssim_set_property (GObject * object, guint prop_id,
|
|
const GValue * value, GParamSpec * pspec)
|
|
{
|
|
GstSSim *ssim;
|
|
|
|
ssim = GST_SSIM (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_SSIM_TYPE:
|
|
ssim->ssimtype = g_value_get_int (value);
|
|
break;
|
|
case PROP_WINDOW_TYPE:
|
|
ssim->windowtype = g_value_get_int (value);
|
|
g_free (ssim->windows);
|
|
ssim->windows = NULL;
|
|
break;
|
|
case PROP_WINDOW_SIZE:
|
|
ssim->windowsize = g_value_get_int (value);
|
|
g_free (ssim->windows);
|
|
ssim->windows = NULL;
|
|
break;
|
|
case PROP_GAUSS_SIGMA:
|
|
ssim->sigma = g_value_get_float (value);
|
|
g_free (ssim->windows);
|
|
ssim->windows = NULL;
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_ssim_get_property (GObject * object, guint prop_id, GValue * value,
|
|
GParamSpec * pspec)
|
|
{
|
|
GstSSim *ssim;
|
|
|
|
ssim = GST_SSIM (object);
|
|
|
|
switch (prop_id) {
|
|
case PROP_SSIM_TYPE:
|
|
g_value_set_int (value, ssim->ssimtype);
|
|
break;
|
|
case PROP_WINDOW_TYPE:
|
|
g_value_set_int (value, ssim->windowtype);
|
|
break;
|
|
case PROP_WINDOW_SIZE:
|
|
g_value_set_int (value, ssim->windowsize);
|
|
break;
|
|
case PROP_GAUSS_SIGMA:
|
|
g_value_set_float (value, ssim->sigma);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
gst_ssim_class_init (GstSSimClass * klass)
|
|
{
|
|
GObjectClass *gobject_class = (GObjectClass *) klass;
|
|
GstElementClass *gstelement_class = (GstElementClass *) klass;
|
|
|
|
gobject_class->set_property = gst_ssim_set_property;
|
|
gobject_class->get_property = gst_ssim_get_property;
|
|
gobject_class->finalize = GST_DEBUG_FUNCPTR (gst_ssim_finalize);
|
|
|
|
g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_SSIM_TYPE,
|
|
g_param_spec_int ("ssim-type", "SSIM type",
|
|
"Type of the SSIM metric. 0 - canonical. 1 - with fixed mu "
|
|
"(almost the same results, but roughly 20% faster)",
|
|
0, 1, 0, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_WINDOW_TYPE,
|
|
g_param_spec_int ("window-type", "Window type",
|
|
"Type of the weighting in the window. "
|
|
"0 - no weighting. 1 - Gaussian weighting (controlled by \"sigma\")",
|
|
0, 1, 1, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_WINDOW_SIZE,
|
|
g_param_spec_int ("window-size", "Window size",
|
|
"Size of a window.", 1, 22, 11,
|
|
G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
g_object_class_install_property (G_OBJECT_CLASS (klass), PROP_GAUSS_SIGMA,
|
|
g_param_spec_float ("gauss-sigma", "Deviation (for Gauss function)",
|
|
"Used to calculate Gussian weights "
|
|
"(only when using Gaussian window).",
|
|
G_MINFLOAT, 10, 1.5, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
|
|
|
|
gst_element_class_add_pad_template (gstelement_class,
|
|
gst_static_pad_template_get (&gst_ssim_src_template));
|
|
gst_element_class_add_pad_template (gstelement_class,
|
|
gst_static_pad_template_get (&gst_ssim_sink_original_template));
|
|
gst_element_class_add_pad_template (gstelement_class,
|
|
gst_static_pad_template_get (&gst_ssim_sink_modified_template));
|
|
gst_element_class_set_static_metadata (gstelement_class, "SSim",
|
|
"Filter/Analyzer/Video",
|
|
"Calculate Y-SSIM for n+2 YUV video streams",
|
|
"Руслан Ижбулатов <lrn1986 _at_ gmail _dot_ com>");
|
|
|
|
parent_class = g_type_class_peek_parent (klass);
|
|
|
|
gstelement_class->request_new_pad =
|
|
GST_DEBUG_FUNCPTR (gst_ssim_request_new_pad);
|
|
gstelement_class->release_pad = GST_DEBUG_FUNCPTR (gst_ssim_release_pad);
|
|
|
|
gstelement_class->change_state = GST_DEBUG_FUNCPTR (gst_ssim_change_state);
|
|
}
|
|
|
|
static GstPad *
|
|
gst_ssim_request_new_pad (GstElement * element, GstPadTemplate * templ,
|
|
const gchar * padname)
|
|
{
|
|
gchar *name;
|
|
GstSSim *ssim;
|
|
GstPad *newpad;
|
|
GstPad *newsrc;
|
|
gint padcount;
|
|
GstPadTemplate *template;
|
|
guint num = -1;
|
|
|
|
if (templ->direction != GST_PAD_SINK)
|
|
goto not_sink;
|
|
|
|
ssim = GST_SSIM (element);
|
|
|
|
padcount = ssim->padcount;
|
|
|
|
GST_DEBUG_OBJECT (ssim, "number of pads = %d", padcount);
|
|
|
|
if (padname)
|
|
GST_DEBUG_OBJECT (ssim, "reqested pad %s", padname);
|
|
else
|
|
goto unnamed_pad;
|
|
|
|
if (strcmp (padname, "original") == 0) {
|
|
newpad = gst_pad_new_from_template (templ, "original");
|
|
GST_DEBUG_OBJECT (ssim, "request new sink pad original");
|
|
ssim->orig = newpad;
|
|
} else if (strncmp (padname, "modified_", 9) == 0) {
|
|
const gchar *numstr = &padname[9];
|
|
num = strtoul (numstr, NULL, 10);
|
|
if (errno == EINVAL || errno == ERANGE)
|
|
goto bad_name;
|
|
newpad = gst_pad_new_from_template (templ, padname);
|
|
GST_DEBUG_OBJECT (ssim, "request new sink pad %s", padname);
|
|
} else
|
|
goto bad_name;
|
|
|
|
gst_pad_set_getcaps_function (newpad,
|
|
GST_DEBUG_FUNCPTR (gst_ssim_sink_getcaps));
|
|
gst_pad_set_setcaps_function (newpad, GST_DEBUG_FUNCPTR (gst_ssim_setcaps));
|
|
gst_collect_pads_add_pad (ssim->collect, newpad, sizeof (GstCollectData),
|
|
NULL, TRUE);
|
|
|
|
/* FIXME: hacked way to override/extend the event function of
|
|
* GstCollectPads; because it sets its own event function giving the
|
|
* element no access to events
|
|
*/
|
|
GST_DEBUG_OBJECT (ssim, "Current collect_event is %p, changing to %p",
|
|
ssim->collect_event, GST_PAD_EVENTFUNC (newpad));
|
|
ssim->collect_event = (GstPadEventFunction) GST_PAD_EVENTFUNC (newpad);
|
|
gst_pad_set_event_function (newpad, GST_DEBUG_FUNCPTR (gst_ssim_sink_event));
|
|
|
|
GST_DEBUG_OBJECT (ssim, "Adding a pad...");
|
|
/* takes ownership of the pad */
|
|
if (!gst_element_add_pad (GST_ELEMENT (ssim), newpad))
|
|
goto could_not_add_sink;
|
|
else
|
|
/* increment pad counter */
|
|
#if GLIB_CHECK_VERSION(2,29,5)
|
|
padcount = g_atomic_int_add (&ssim->padcount, 1);
|
|
#else
|
|
padcount = g_atomic_int_exchange_and_add (&ssim->padcount, 1);
|
|
#endif
|
|
|
|
if (num != -1) {
|
|
GstSSimOutputContext *c;
|
|
|
|
template = gst_static_pad_template_get (&gst_ssim_src_template);
|
|
name = g_strdup_printf ("src_%u", num);
|
|
newsrc = gst_pad_new_from_template (template, name);
|
|
GST_DEBUG_OBJECT (ssim, "creating src pad %s", name);
|
|
g_free (name);
|
|
gst_object_unref (template);
|
|
|
|
gst_pad_set_getcaps_function (newsrc,
|
|
GST_DEBUG_FUNCPTR (gst_ssim_src_getcaps));
|
|
gst_pad_set_query_function (newsrc, GST_DEBUG_FUNCPTR (gst_ssim_query));
|
|
gst_pad_set_event_function (newsrc, GST_DEBUG_FUNCPTR (gst_ssim_src_event));
|
|
|
|
if (!gst_element_add_pad (GST_ELEMENT (ssim), newsrc))
|
|
goto could_not_add_src;
|
|
|
|
c = g_new (GstSSimOutputContext, 1);
|
|
c->pad = newsrc;
|
|
g_object_set_data (G_OBJECT (newpad), "ssim-match-output-context", c);
|
|
g_ptr_array_add (ssim->src, (gpointer) c);
|
|
}
|
|
|
|
return newpad;
|
|
|
|
/* errors */
|
|
bad_name:
|
|
{
|
|
g_warning ("gstssim: request new pad with bad name %s (must be "
|
|
"'modified')\n", padname);
|
|
return NULL;
|
|
}
|
|
unnamed_pad:
|
|
{
|
|
g_warning ("gstssim: request new pad without a name (must be "
|
|
"'modified')\n");
|
|
return NULL;
|
|
}
|
|
not_sink:
|
|
{
|
|
g_warning ("gstssim: request new pad that is not a SINK pad\n");
|
|
return NULL;
|
|
}
|
|
could_not_add_src:
|
|
{
|
|
GST_DEBUG_OBJECT (ssim, "could not add src pad");
|
|
gst_object_unref (newsrc);
|
|
}
|
|
could_not_add_sink:
|
|
{
|
|
GST_DEBUG_OBJECT (ssim, "could not add sink pad");
|
|
gst_collect_pads_remove_pad (ssim->collect, newpad);
|
|
gst_object_unref (newpad);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
static void
|
|
gst_ssim_release_pad (GstElement * element, GstPad * pad)
|
|
{
|
|
GstSSim *ssim;
|
|
|
|
ssim = GST_SSIM (element);
|
|
|
|
GST_DEBUG_OBJECT (ssim, "release pad %s:%s", GST_DEBUG_PAD_NAME (pad));
|
|
|
|
gst_collect_pads_remove_pad (ssim->collect, pad);
|
|
gst_element_remove_pad (element, pad);
|
|
}
|
|
|
|
|
|
static void
|
|
gst_ssim_init (GstSSim * ssim)
|
|
{
|
|
ssim->windowsize = 11;
|
|
ssim->windowtype = 1;
|
|
ssim->windows = NULL;
|
|
ssim->sigma = 1.5;
|
|
ssim->ssimtype = 0;
|
|
ssim->src = g_ptr_array_new ();
|
|
ssim->padcount = 0;
|
|
ssim->collect_event = NULL;
|
|
ssim->sinkcaps = NULL;
|
|
|
|
/* keep track of the sinkpads requested */
|
|
ssim->collect = gst_collect_pads_new ();
|
|
gst_collect_pads_set_function (ssim->collect,
|
|
GST_DEBUG_FUNCPTR (gst_ssim_collected), ssim);
|
|
}
|
|
|
|
static void
|
|
gst_ssim_finalize (GObject * object)
|
|
{
|
|
GstSSim *ssim = GST_SSIM (object);
|
|
|
|
gst_object_unref (ssim->collect);
|
|
ssim->collect = NULL;
|
|
|
|
g_free (ssim->windows);
|
|
ssim->windows = NULL;
|
|
|
|
g_free (ssim->weights);
|
|
ssim->weights = NULL;
|
|
|
|
if (ssim->sinkcaps)
|
|
gst_caps_unref (ssim->sinkcaps);
|
|
if (ssim->srccaps)
|
|
gst_caps_unref (ssim->srccaps);
|
|
|
|
g_ptr_array_free (ssim->src, TRUE);
|
|
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
typedef gfloat (*GstSSimWeightFunc) (GstSSim * ssim, gint y, gint x);
|
|
|
|
static gfloat
|
|
gst_ssim_weight_func_none (GstSSim * ssim, gint y, gint x)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static gfloat
|
|
gst_ssim_weight_func_gauss (GstSSim * ssim, gint y, gint x)
|
|
{
|
|
gfloat coord = sqrt (x * x + y * y);
|
|
return exp (-1 * (coord * coord) / (2 * ssim->sigma * ssim->sigma)) /
|
|
(ssim->sigma * sqrt (2 * G_PI));
|
|
}
|
|
|
|
static gboolean
|
|
gst_ssim_regenerate_windows (GstSSim * ssim)
|
|
{
|
|
gint windowiseven;
|
|
gint y, x, y2, x2;
|
|
GstSSimWeightFunc func;
|
|
gfloat normal_summ = 0;
|
|
gint normal_count = 0;
|
|
|
|
g_free (ssim->weights);
|
|
|
|
ssim->weights = g_new (gfloat, ssim->windowsize * ssim->windowsize);
|
|
|
|
windowiseven = ((gint) ssim->windowsize / 2) * 2 == ssim->windowsize ? 1 : 0;
|
|
|
|
g_free (ssim->windows);
|
|
|
|
ssim->windows = g_new (GstSSimWindowCache, ssim->height * ssim->width);
|
|
|
|
switch (ssim->windowtype) {
|
|
case 0:
|
|
func = gst_ssim_weight_func_none;
|
|
break;
|
|
case 1:
|
|
func = gst_ssim_weight_func_gauss;
|
|
break;
|
|
default:
|
|
GST_WARNING_OBJECT (ssim, "unknown window type - %d. Defaulting to %d",
|
|
ssim->windowtype, 1);
|
|
ssim->windowtype = 1;
|
|
func = gst_ssim_weight_func_gauss;
|
|
}
|
|
|
|
for (y = 0; y < ssim->windowsize; y++) {
|
|
gint yoffset = y * ssim->windowsize;
|
|
for (x = 0; x < ssim->windowsize; x++) {
|
|
ssim->weights[yoffset + x] = func (ssim, x - ssim->windowsize / 2 +
|
|
windowiseven, y - ssim->windowsize / 2 + windowiseven);
|
|
normal_summ += ssim->weights[yoffset + x];
|
|
normal_count++;
|
|
}
|
|
}
|
|
|
|
for (y = 0; y < ssim->height; y++) {
|
|
for (x = 0; x < ssim->width; x++) {
|
|
GstSSimWindowCache win;
|
|
gint element_count = 0;
|
|
|
|
win.x_window_start = x - ssim->windowsize / 2 + windowiseven;
|
|
win.x_weight_start = 0;
|
|
if (win.x_window_start < 0) {
|
|
win.x_weight_start = -win.x_window_start;
|
|
win.x_window_start = 0;
|
|
}
|
|
|
|
win.x_window_end = x + ssim->windowsize / 2;
|
|
if (win.x_window_end >= ssim->width)
|
|
win.x_window_end = ssim->width - 1;
|
|
|
|
win.y_window_start = y - ssim->windowsize / 2 + windowiseven;
|
|
win.y_weight_start = 0;
|
|
if (win.y_window_start < 0) {
|
|
win.y_weight_start = -win.y_window_start;
|
|
win.y_window_start = 0;
|
|
}
|
|
|
|
win.y_window_end = y + ssim->windowsize / 2;
|
|
if (win.y_window_end >= ssim->height)
|
|
win.y_window_end = ssim->height - 1;
|
|
|
|
win.element_summ = 0;
|
|
element_count = (win.y_window_end - win.y_window_start + 1) *
|
|
(win.x_window_end - win.x_window_start + 1);
|
|
if (element_count == normal_count)
|
|
win.element_summ = normal_summ;
|
|
else {
|
|
for (y2 = win.y_weight_start; y2 < ssim->windowsize; y2++) {
|
|
for (x2 = win.x_weight_start; x2 < ssim->windowsize; x2++) {
|
|
win.element_summ += ssim->weights[y2 * ssim->windowsize + x2];
|
|
}
|
|
}
|
|
}
|
|
ssim->windows[(y * ssim->width + x)] = win;
|
|
}
|
|
}
|
|
|
|
/* FIXME: while 0.01 and 0.03 are pretty much static, the 255 implies that
|
|
* we're working with 8-bit-per-color-component format, which may not be true
|
|
*/
|
|
ssim->const1 = 0.01 * 255 * 0.01 * 255;
|
|
ssim->const2 = 0.03 * 255 * 0.03 * 255;
|
|
return TRUE;
|
|
}
|
|
|
|
static GstFlowReturn
|
|
gst_ssim_collected (GstCollectPads * pads, gpointer user_data)
|
|
{
|
|
GstSSim *ssim;
|
|
GSList *collected;
|
|
GstFlowReturn ret = GST_FLOW_OK;
|
|
GstBuffer *orgbuf = NULL;
|
|
gfloat *orgmu = NULL;
|
|
GstBuffer *outbuf = NULL;
|
|
gpointer outdata = NULL;
|
|
guint outsize = 0;
|
|
gfloat mssim = 0, lowest = 1, highest = -1;
|
|
gboolean ready = TRUE;
|
|
gint padnumber = 0;
|
|
|
|
ssim = GST_SSIM (user_data);
|
|
|
|
if (G_UNLIKELY (ssim->windows == NULL)) {
|
|
GST_DEBUG_OBJECT (ssim, "Regenerating windows");
|
|
gst_ssim_regenerate_windows (ssim);
|
|
}
|
|
|
|
switch (ssim->ssimtype) {
|
|
case 0:
|
|
ssim->func = (GstSSimFunction) calcssim_canonical;
|
|
break;
|
|
case 1:
|
|
ssim->func = (GstSSimFunction) calcssim_without_mu;
|
|
break;
|
|
default:
|
|
return GST_FLOW_ERROR;
|
|
}
|
|
|
|
for (collected = pads->data; collected; collected = g_slist_next (collected)) {
|
|
GstCollectData *collect_data;
|
|
GstBuffer *inbuf;
|
|
|
|
collect_data = (GstCollectData *) collected->data;
|
|
|
|
inbuf = gst_collect_pads_peek (pads, collect_data);
|
|
|
|
if (inbuf == NULL) {
|
|
GST_LOG_OBJECT (ssim, "channel %p: no bytes available", collect_data);
|
|
ready = FALSE;
|
|
} else
|
|
gst_buffer_unref (inbuf);
|
|
}
|
|
|
|
/* if _collected() was called, all pads should have data, but if
|
|
* one of them doesn't, it means that it is EOS and we can't go any further
|
|
*
|
|
* FIXME, shouldn't we do something about pads that DO have data?
|
|
* Flush them or something?
|
|
*/
|
|
if (G_UNLIKELY (!ready))
|
|
goto eos;
|
|
|
|
/* Mu is just a blur, we can calculate it once */
|
|
if (ssim->ssimtype == 0) {
|
|
orgmu = g_new (gfloat, ssim->width * ssim->height);
|
|
|
|
for (collected = pads->data; collected;
|
|
collected = g_slist_next (collected)) {
|
|
GstCollectData *collect_data;
|
|
|
|
collect_data = (GstCollectData *) collected->data;
|
|
|
|
if (collect_data->pad == ssim->orig) {
|
|
orgbuf = gst_collect_pads_pop (pads, collect_data);
|
|
|
|
GST_DEBUG_OBJECT (ssim, "Original stream - flags(0x%x), timestamp(%"
|
|
GST_TIME_FORMAT "), duration(%" GST_TIME_FORMAT ")",
|
|
GST_BUFFER_FLAGS (orgbuf),
|
|
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (orgbuf)),
|
|
GST_TIME_ARGS (GST_BUFFER_DURATION (orgbuf)));
|
|
calculate_mu (ssim, orgmu, GST_BUFFER_DATA (orgbuf));
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
GST_LOG_OBJECT (ssim, "starting to cycle through streams");
|
|
|
|
for (collected = pads->data; collected; collected = g_slist_next (collected)) {
|
|
GstCollectData *collect_data;
|
|
GstBuffer *inbuf;
|
|
guint8 *indata;
|
|
|
|
collect_data = (GstCollectData *) collected->data;
|
|
|
|
if (collect_data->pad != ssim->orig) {
|
|
inbuf = gst_collect_pads_pop (pads, collect_data);
|
|
|
|
indata = GST_BUFFER_DATA (inbuf);
|
|
|
|
GST_DEBUG_OBJECT (ssim, "Modified stream - flags(0x%x), timestamp(%"
|
|
GST_TIME_FORMAT "), duration(%" GST_TIME_FORMAT ")",
|
|
GST_BUFFER_FLAGS (inbuf),
|
|
GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (inbuf)),
|
|
GST_TIME_ARGS (GST_BUFFER_DURATION (inbuf)));
|
|
|
|
if (!GST_BUFFER_FLAG_IS_SET (inbuf, GST_BUFFER_FLAG_GAP)) {
|
|
GstSSimOutputContext *c;
|
|
GstEvent *measured;
|
|
guint64 offset;
|
|
GValue vmean = { 0 }
|
|
, vlowest = {
|
|
0}
|
|
, vhighest = {
|
|
0};
|
|
|
|
c = (GstSSimOutputContext *)
|
|
g_object_get_data (G_OBJECT (collect_data->pad),
|
|
"ssim-match-output-context");
|
|
|
|
GST_DEBUG_OBJECT (ssim, "Output context is %" GST_PTR_FORMAT
|
|
", pad will be %" GST_PTR_FORMAT, c, c->pad);
|
|
|
|
outsize = GST_ROUND_UP_4 (ssim->width) * ssim->height;
|
|
GST_LOG_OBJECT (ssim, "channel %p: making output buffer of %d bytes",
|
|
collect_data, outsize);
|
|
|
|
/* first buffer, alloc outsize.
|
|
* FIXME: we can easily subbuffer and _make_writable.
|
|
* FIXME: only create empty buffer for first non-gap buffer, so that we
|
|
* only use ssim function when really calculating
|
|
*/
|
|
outbuf = gst_buffer_new_and_alloc (GST_ROUND_UP_4 (ssim->width) *
|
|
ssim->height);
|
|
outdata = GST_BUFFER_DATA (outbuf);
|
|
gst_buffer_set_caps (outbuf, gst_pad_get_fixed_caps_func (c->pad));
|
|
|
|
/* Videos should match, so the output video has the same characteristics
|
|
* as the input video
|
|
*/
|
|
/* set timestamps on the output buffer */
|
|
gst_buffer_copy_metadata (outbuf, inbuf, (GstBufferCopyFlags)
|
|
GST_BUFFER_COPY_FLAGS | GST_BUFFER_COPY_TIMESTAMPS);
|
|
|
|
g_value_init (&vmean, G_TYPE_FLOAT);
|
|
g_value_init (&vlowest, G_TYPE_FLOAT);
|
|
g_value_init (&vhighest, G_TYPE_FLOAT);
|
|
|
|
GST_LOG_OBJECT (ssim, "channel %p: calculating SSIM", collect_data);
|
|
|
|
ssim->func (ssim, GST_BUFFER_DATA (orgbuf), orgmu, indata, outdata,
|
|
&mssim, &lowest, &highest);
|
|
|
|
GST_DEBUG_OBJECT (GST_OBJECT (ssim), "MSSIM is %f, l-h is %f - %f",
|
|
mssim, lowest, highest);
|
|
|
|
gst_ssim_post_message (ssim, outbuf, mssim, lowest, highest);
|
|
|
|
g_value_set_float (&vmean, mssim);
|
|
g_value_set_float (&vlowest, lowest);
|
|
g_value_set_float (&vhighest, highest);
|
|
offset = GST_BUFFER_OFFSET (inbuf);
|
|
|
|
/* our timestamping is very simple, just an ever incrementing
|
|
* counter, the new segment time will take care of their respective
|
|
* stream time.
|
|
*/
|
|
if (c->segment_pending) {
|
|
GstEvent *event;
|
|
|
|
/* FIXME, use rate/applied_rate as set on all sinkpads.
|
|
* - currently we just set rate as received from last seek-event
|
|
* We could potentially figure out the duration as well using
|
|
* the current segment positions and the stated stop positions.
|
|
* Also we just start from stream time 0 which is rather
|
|
* weird. For non-synchronized mixing, the time should be
|
|
* the min of the stream times of all received segments,
|
|
* rationale being that the duration is at least going to
|
|
* be as long as the earliest stream we start mixing. This
|
|
* would also be correct for synchronized mixing but then
|
|
* the later streams would be delayed until the stream times`
|
|
* match.
|
|
*/
|
|
event = gst_event_new_new_segment_full (FALSE, ssim->segment_rate,
|
|
1.0, GST_FORMAT_TIME, ssim->timestamp, -1,
|
|
ssim->segment_position);
|
|
|
|
gst_pad_push_event (c->pad, event);
|
|
c->segment_pending = FALSE;
|
|
}
|
|
|
|
measured = gst_event_new_measured (offset,
|
|
GST_BUFFER_TIMESTAMP (inbuf), "SSIM", &vmean, &vlowest, &vhighest);
|
|
gst_pad_push_event (c->pad, measured);
|
|
|
|
/* send it out */
|
|
GST_DEBUG_OBJECT (ssim, "pushing outbuf, timestamp %" GST_TIME_FORMAT
|
|
", size %d", GST_TIME_ARGS (GST_BUFFER_TIMESTAMP (outbuf)),
|
|
GST_BUFFER_SIZE (outbuf));
|
|
ret &= gst_pad_push (c->pad, outbuf);
|
|
|
|
} else {
|
|
GST_LOG_OBJECT (ssim, "channel %p: skipping", collect_data);
|
|
}
|
|
gst_buffer_unref (inbuf);
|
|
padnumber++;
|
|
}
|
|
}
|
|
gst_buffer_unref (orgbuf);
|
|
|
|
if (ssim->ssimtype == 0)
|
|
g_free (orgmu);
|
|
|
|
ssim->segment_position = 0;
|
|
|
|
return ret;
|
|
|
|
/* ERRORS */
|
|
eos:
|
|
{
|
|
gint i;
|
|
GST_DEBUG_OBJECT (ssim, "no data available, must be EOS");
|
|
for (i = 0; i < ssim->src->len; i++) {
|
|
GstSSimOutputContext *c =
|
|
(GstSSimOutputContext *) g_ptr_array_index (ssim->src, i);
|
|
gst_pad_push_event (c->pad, gst_event_new_eos ());
|
|
}
|
|
|
|
return GST_FLOW_UNEXPECTED;
|
|
}
|
|
}
|
|
|
|
static GstStateChangeReturn
|
|
gst_ssim_change_state (GstElement * element, GstStateChange transition)
|
|
{
|
|
GstSSim *ssim;
|
|
GstStateChangeReturn ret;
|
|
|
|
ssim = GST_SSIM (element);
|
|
|
|
switch (transition) {
|
|
case GST_STATE_CHANGE_NULL_TO_READY:
|
|
break;
|
|
case GST_STATE_CHANGE_READY_TO_PAUSED:
|
|
ssim->timestamp = 0;
|
|
ssim->offset = 0;
|
|
{
|
|
GstSSimOutputContext *c;
|
|
gint i = 0;
|
|
for (i = 0; i < ssim->src->len; i++) {
|
|
c = (GstSSimOutputContext *) g_ptr_array_index (ssim->src, i);
|
|
c->segment_pending = TRUE;
|
|
}
|
|
}
|
|
ssim->segment_position = 0;
|
|
ssim->segment_rate = 1.0;
|
|
gst_segment_init (&ssim->segment, GST_FORMAT_UNDEFINED);
|
|
gst_collect_pads_start (ssim->collect);
|
|
break;
|
|
case GST_STATE_CHANGE_PAUSED_TO_PLAYING:
|
|
break;
|
|
case GST_STATE_CHANGE_PAUSED_TO_READY:
|
|
/* need to unblock the collectpads before calling the
|
|
* parent change_state so that streaming can finish
|
|
*/
|
|
gst_collect_pads_stop (ssim->collect);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
ret = GST_ELEMENT_CLASS (parent_class)->change_state (element, transition);
|
|
|
|
switch (transition) {
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return ret;
|
|
}
|