2011-05-30 11:22:19 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2011 The WebRTC project authors. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
|
* tree. An additional intellectual property rights grant can be found
|
|
|
|
* in the file PATENTS. All contributing project authors may
|
|
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "content_metrics_processing.h"
|
|
|
|
#include "tick_time.h"
|
|
|
|
#include "module_common_types.h"
|
|
|
|
#include "video_coding_defines.h"
|
|
|
|
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
namespace webrtc {
|
|
|
|
|
|
|
|
//////////////////////////////////
|
|
|
|
/// VCMContentMetricsProcessing //
|
|
|
|
//////////////////////////////////
|
|
|
|
|
|
|
|
VCMContentMetricsProcessing::VCMContentMetricsProcessing():
|
|
|
|
_frameRate(0),
|
|
|
|
_recAvgFactor(1 / 150.0f), // matched to 30fps
|
2011-06-24 18:08:33 +00:00
|
|
|
_frameCntRecursiveAvg(0),
|
|
|
|
_frameCntUniformAvg(0),
|
|
|
|
_avgMotionLevel(0.0f),
|
|
|
|
_avgSpatialLevel(0.0f)
|
2011-05-30 11:22:19 +00:00
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg = new VideoContentMetrics();
|
|
|
|
_uniformAvg = new VideoContentMetrics();
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VCMContentMetricsProcessing::~VCMContentMetricsProcessing()
|
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
delete _recursiveAvg;
|
|
|
|
delete _uniformAvg;
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
WebRtc_Word32
|
|
|
|
VCMContentMetricsProcessing::Reset()
|
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg->Reset();
|
|
|
|
_uniformAvg->Reset();
|
2011-05-30 11:22:19 +00:00
|
|
|
_frameRate = 0;
|
2011-06-24 18:08:33 +00:00
|
|
|
_frameCntRecursiveAvg = 0;
|
|
|
|
_frameCntUniformAvg = 0;
|
|
|
|
_avgMotionLevel = 0.0f;
|
|
|
|
_avgSpatialLevel = 0.0f;
|
2011-05-30 11:22:19 +00:00
|
|
|
return VCM_OK;
|
|
|
|
}
|
2011-06-24 18:08:33 +00:00
|
|
|
|
2011-05-30 11:22:19 +00:00
|
|
|
void
|
|
|
|
VCMContentMetricsProcessing::UpdateFrameRate(WebRtc_UWord32 frameRate)
|
|
|
|
{
|
|
|
|
_frameRate = frameRate;
|
2011-06-24 18:08:33 +00:00
|
|
|
// Update factor for recursive averaging.
|
|
|
|
_recAvgFactor = (float) 1000.0f / ((float)(_frameRate * kQmMinIntervalMs));
|
2011-05-30 11:22:19 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
VideoContentMetrics*
|
|
|
|
VCMContentMetricsProcessing::LongTermAvgData()
|
2011-05-30 11:22:19 +00:00
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
if (_frameCntRecursiveAvg == 0)
|
2011-05-30 11:22:19 +00:00
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
return NULL;
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
2011-06-24 18:08:33 +00:00
|
|
|
return _recursiveAvg;
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VideoContentMetrics*
|
2011-06-24 18:08:33 +00:00
|
|
|
VCMContentMetricsProcessing::ShortTermAvgData()
|
2011-05-30 11:22:19 +00:00
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
if (_frameCntUniformAvg == 0)
|
2011-05-30 11:22:19 +00:00
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
// Two metrics are used: motion and spatial level.
|
|
|
|
_uniformAvg->motionMagnitudeNZ = _avgMotionLevel /
|
|
|
|
(float)(_frameCntUniformAvg);
|
|
|
|
_uniformAvg->spatialPredErr = _avgSpatialLevel /
|
|
|
|
(float)(_frameCntUniformAvg);
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
return _uniformAvg;
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
void
|
|
|
|
VCMContentMetricsProcessing::ResetShortTermAvgData()
|
2011-05-30 11:22:19 +00:00
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
// Reset
|
|
|
|
_avgMotionLevel = 0.0f;
|
|
|
|
_avgSpatialLevel = 0.0f;
|
|
|
|
_frameCntUniformAvg = 0;
|
|
|
|
}
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
WebRtc_Word32
|
|
|
|
VCMContentMetricsProcessing::UpdateContentData(const VideoContentMetrics *contentMetrics)
|
|
|
|
{
|
|
|
|
if (contentMetrics == NULL)
|
2011-05-30 11:22:19 +00:00
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
return VCM_OK;
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
2011-06-24 18:08:33 +00:00
|
|
|
return ProcessContent(contentMetrics);
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
}
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
WebRtc_UWord32
|
|
|
|
VCMContentMetricsProcessing::ProcessContent(const VideoContentMetrics *contentMetrics)
|
|
|
|
{
|
|
|
|
// Update the recursive averaged metrics
|
|
|
|
// average is over longer window of time: over QmMinIntervalMs ms.
|
|
|
|
UpdateRecursiveAvg(contentMetrics);
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
// Update the uniform averaged metrics:
|
|
|
|
// average is over shorter window of time: based on ~RTCP reports.
|
|
|
|
UpdateUniformAvg(contentMetrics);
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
return VCM_OK;
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
void
|
|
|
|
VCMContentMetricsProcessing::UpdateUniformAvg(const VideoContentMetrics *contentMetrics)
|
2011-05-30 11:22:19 +00:00
|
|
|
{
|
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
// Update frame counter
|
|
|
|
_frameCntUniformAvg += 1;
|
|
|
|
|
|
|
|
// Update averaged metrics: motion and spatial level are used.
|
|
|
|
_avgMotionLevel += contentMetrics->motionMagnitudeNZ;
|
|
|
|
_avgSpatialLevel += contentMetrics->spatialPredErr;
|
2011-05-30 11:22:19 +00:00
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
void
|
2011-06-24 18:08:33 +00:00
|
|
|
VCMContentMetricsProcessing::UpdateRecursiveAvg(const VideoContentMetrics *contentMetrics)
|
2011-05-30 11:22:19 +00:00
|
|
|
{
|
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
// Threshold for size of zero motion cluster:
|
|
|
|
// Use for updating 3 motion vector derived metrics:
|
|
|
|
// motion magnitude, cluster distortion, and horizontalness.
|
2011-05-30 11:22:19 +00:00
|
|
|
float nonZeroMvThr = 0.1f;
|
|
|
|
|
|
|
|
float tmpRecAvgFactor = _recAvgFactor;
|
2011-06-24 18:08:33 +00:00
|
|
|
|
|
|
|
// Take value as is for first frame (no motion search in frame zero).
|
|
|
|
if (_frameCntRecursiveAvg < 1)
|
2011-05-30 11:22:19 +00:00
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
tmpRecAvgFactor = 1;
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg->motionPredErr = (1 - tmpRecAvgFactor) *
|
|
|
|
_recursiveAvg->motionPredErr +
|
|
|
|
tmpRecAvgFactor * contentMetrics->motionPredErr;
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg->sizeZeroMotion = (1 - tmpRecAvgFactor) *
|
|
|
|
_recursiveAvg->sizeZeroMotion +
|
|
|
|
tmpRecAvgFactor * contentMetrics->sizeZeroMotion;
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg->spatialPredErr = (1 - tmpRecAvgFactor) *
|
|
|
|
_recursiveAvg->spatialPredErr +
|
|
|
|
tmpRecAvgFactor * contentMetrics->spatialPredErr;
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg->spatialPredErrH = (1 - tmpRecAvgFactor) *
|
|
|
|
_recursiveAvg->spatialPredErrH +
|
|
|
|
tmpRecAvgFactor * contentMetrics->spatialPredErrH;
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg->spatialPredErrV = (1 - tmpRecAvgFactor) *
|
|
|
|
_recursiveAvg->spatialPredErrV +
|
|
|
|
tmpRecAvgFactor * contentMetrics->spatialPredErrV;
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
// motionMag metric is derived from NFD (normalized frame difference).
|
2011-05-30 11:22:19 +00:00
|
|
|
if (kNfdMetric == 1)
|
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg->motionMagnitudeNZ = (1 - tmpRecAvgFactor) *
|
|
|
|
_recursiveAvg->motionMagnitudeNZ +
|
|
|
|
tmpRecAvgFactor * contentMetrics->motionMagnitudeNZ;
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (contentMetrics->sizeZeroMotion > nonZeroMvThr)
|
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg->motionClusterDistortion = (1 - tmpRecAvgFactor) *
|
|
|
|
_recursiveAvg->motionClusterDistortion +
|
|
|
|
tmpRecAvgFactor *contentMetrics->motionClusterDistortion;
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg->motionHorizontalness = (1 - _recAvgFactor) *
|
|
|
|
_recursiveAvg->motionHorizontalness +
|
|
|
|
tmpRecAvgFactor * contentMetrics->motionHorizontalness;
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
// motionMag metric is derived from motion vectors.
|
2011-05-30 11:22:19 +00:00
|
|
|
if (kNfdMetric == 0)
|
|
|
|
{
|
2011-06-24 18:08:33 +00:00
|
|
|
_recursiveAvg->motionMagnitudeNZ = (1 - tmpRecAvgFactor) *
|
|
|
|
_recursiveAvg->motionMagnitudeNZ +
|
|
|
|
tmpRecAvgFactor * contentMetrics->motionMagnitudeNZ;
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
// Update native values:
|
|
|
|
// TODO (marpan): we don't need to update this every frame.
|
|
|
|
_recursiveAvg->nativeHeight = contentMetrics->nativeHeight;
|
|
|
|
_recursiveAvg->nativeWidth = contentMetrics->nativeWidth;
|
|
|
|
_recursiveAvg->nativeFrameRate = contentMetrics->nativeFrameRate;
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
_frameCntRecursiveAvg++;
|
2011-05-30 11:22:19 +00:00
|
|
|
|
2011-06-24 18:08:33 +00:00
|
|
|
return;
|
2011-05-30 11:22:19 +00:00
|
|
|
}
|
2011-06-24 18:08:33 +00:00
|
|
|
} //end of namespace
|