quality_monitor.cpp
上传用户:xjjlds
上传日期:2015-12-05
资源大小:22823k
文件大小:9k
源码类别:

多媒体编程

开发平台:

Visual C++

  1. /* ***** BEGIN LICENSE BLOCK *****
  2. *
  3. * $Id: quality_monitor.cpp,v 1.3 2005/01/30 05:11:41 gabest Exp $ $Name:  $
  4. *
  5. * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  6. *
  7. * The contents of this file are subject to the Mozilla Public License
  8. * Version 1.1 (the "License"); you may not use this file except in compliance
  9. * with the License. You may obtain a copy of the License at
  10. * http://www.mozilla.org/MPL/
  11. *
  12. * Software distributed under the License is distributed on an "AS IS" basis,
  13. * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for
  14. * the specific language governing rights and limitations under the License.
  15. *
  16. * The Original Code is BBC Research and Development code.
  17. *
  18. * The Initial Developer of the Original Code is the British Broadcasting
  19. * Corporation.
  20. * Portions created by the Initial Developer are Copyright (C) 2004.
  21. * All Rights Reserved.
  22. *
  23. * Contributor(s): Thomas Davies (Original Author)
  24. *
  25. * Alternatively, the contents of this file may be used under the terms of
  26. * the GNU General Public License Version 2 (the "GPL"), or the GNU Lesser
  27. * Public License Version 2.1 (the "LGPL"), in which case the provisions of
  28. * the GPL or the LGPL are applicable instead of those above. If you wish to
  29. * allow use of your version of this file only under the terms of the either
  30. * the GPL or LGPL and not to allow others to use your version of this file
  31. * under the MPL, indicate your decision by deleting the provisions above
  32. * and replace them with the notice and other provisions required by the GPL
  33. * or LGPL. If you do not delete the provisions above, a recipient may use
  34. * your version of this file under the terms of any one of the MPL, the GPL
  35. * or the LGPL.
  36. * ***** END LICENSE BLOCK ***** */
  37. #include <libdirac_encoder/quality_monitor.h>
  38. #include <libdirac_common/wavelet_utils.h>
  39. using namespace dirac;
  40. using std::log10;
  41. QualityMonitor::QualityMonitor(EncoderParams& encp, 
  42.                                const SeqParams& sparams)
  43. :
  44.     m_encparams(encp),
  45.     m_cformat( sparams.CFormat() ),
  46.     m_true_xl( sparams.Xl() ),
  47.     m_true_yl( sparams.Yl() ),
  48.     m_target_quality(3),
  49.     m_last_quality(3),
  50.     m_slope(3),
  51.     m_offset(3),
  52.     m_last_lambda(3)
  53. {
  54.     ResetAll();
  55. }
  56. void QualityMonitor::ResetAll()
  57. {
  58.     // set target qualities
  59.   m_target_quality[I_frame] = 0.28 * m_encparams.Qf()* m_encparams.Qf() + 20.0 ;
  60. m_target_quality[L1_frame] = m_target_quality[I_frame] - 1.0;
  61. m_target_quality[L2_frame] = m_target_quality[I_frame] - 2.0;
  62.     // assume we hit those targets last time
  63.     m_last_quality = m_target_quality;
  64.     // set defaults for the model
  65.      m_slope[I_frame] = -4.0;
  66.      m_slope[L1_frame] = -4.0;
  67.      m_slope[L2_frame] = -4.0;
  68.      m_offset[I_frame] = 38.5,
  69.      m_offset[L1_frame] = 43.3;
  70.      m_offset[L2_frame] = 43.3;
  71.     for (size_t fsort=0; fsort<3; ++fsort)
  72.     {
  73.         m_last_lambda[fsort] = std::pow( 10.0, (m_target_quality[fsort] - m_offset[fsort])/m_slope[fsort] );
  74.     }// fsort
  75.     // set a default ratio for the motion estimation lambda
  76.     // Exact value TBD - will incorporate stuff about blocks and so on
  77.     // Also need to think about how this can be adapted for sequences with more or less motion 
  78.     m_me_ratio = 0.1;
  79.   // set up the Lagrangian parameters
  80.     for (size_t fsort=0; fsort<3; ++fsort)
  81.     {
  82.         m_encparams.SetLambda( FrameSort(fsort), m_last_lambda[fsort] );
  83.     }// fsort
  84.     m_encparams.SetL1MELambda( std::sqrt(m_encparams.L1Lambda())*m_me_ratio );
  85.     m_encparams.SetL2MELambda( std::sqrt(m_encparams.L2Lambda())*m_me_ratio );
  86. }
  87. bool QualityMonitor::UpdateModel(const Frame& ld_frame, const Frame& orig_frame , const int count)
  88. {
  89.     // The return value - true if we need to recode, false otherwise
  90.     bool recode = false;
  91. const FrameSort& fsort = ld_frame.GetFparams().FSort();
  92. double target_quality;
  93. // Parameters relating to the last frame we measured
  94. double last_lambda;
  95. double last_quality;
  96. // Parameters relating to the current frame
  97. double current_lambda;
  98. double current_quality;
  99.     // Set up local parameters for the particular frame type
  100.     current_lambda = m_encparams.Lambda(fsort);
  101.     last_lambda = m_last_lambda[fsort];
  102.     last_quality = m_last_quality[fsort];
  103.     target_quality = m_target_quality[fsort];
  104.     // Get the quality of the current frame
  105. current_quality = QualityVal( ld_frame.Ydata() , orig_frame.Ydata() , 0.0 , fsort );
  106.     // Copy current data into memory for last frame data
  107.     m_last_lambda[fsort] = m_encparams.Lambda(fsort);
  108.     m_last_quality[fsort] = current_quality;
  109. // Ok, so we've got an actual quality value to use. We know the lambda used before and the resulting
  110. // quality then allows us to estimate the slope of the curve of quality versus log of lambda
  111. if ( std::abs(current_quality - last_quality)> 0.2 && 
  112.          std::abs(log10(current_lambda) - log10(last_lambda)) > 0.1 ) 
  113. {// if we can adjust model accurately, do so
  114.         double slope, offset;
  115.         // Calculate the slope of WPSNR versus log(lambda) from prior measurements
  116.       slope = (current_quality - last_quality)/( log10(current_lambda) - log10(last_lambda) );
  117.  
  118.         //Restrict so that the value isn't too extreme
  119.         slope = std::min( std::max( -10.0 , slope ), -0.1);
  120.    // Calculate the resulting offset
  121. offset = current_quality - ( log10(current_lambda) * slope );
  122.         if ( count != 1 )
  123.         {
  124.             // Update the default values using a simple recursive filter ...
  125.             m_slope[fsort] = (3.0*m_slope[fsort] + slope)/4.0;
  126.             m_offset[fsort] = (3.0*m_offset[fsort] + offset)/4.0;            
  127.         }
  128.         else
  129.         {
  130.             // .. unless we're recoding a frame for the first time            
  131.             m_slope[fsort] = (m_slope[fsort] + slope)/2.0;
  132.             m_offset[fsort] = (m_offset[fsort] + offset)/2.0;
  133.         }
  134.         m_slope[fsort] = std::min( std::max( -10.0 , m_slope[fsort] ), -1.5);
  135.     }
  136.     // If we need to adjust the lambdas, do so
  137. if ( std::abs(current_quality - target_quality)> 0.2 )
  138. {
  139.         // Update the lambdas as appropriate
  140.         float quality_diff = m_target_quality[fsort] - current_quality;
  141.         CalcNewLambdas(fsort , std::min( m_slope[fsort] , -1.0 ), quality_diff );
  142.     }
  143.     // if we have a large difference in quality, recode)
  144.     if ( std::abs( current_quality - target_quality )>1.5 )
  145.         recode = true;
  146.     return recode;
  147. }
  148. void QualityMonitor::CalcNewLambdas(const FrameSort fsort, const double slope, const double quality_diff )
  149. {
  150.      const double clipped_quality_ratio = std::min( 2.0 , std::max( quality_diff/slope , -2.0 ) );
  151.      if ( m_encparams.Lambda(fsort) > 100001.0 && clipped_quality_ratio > 0.0 )
  152.          m_encparams.SetLambda(fsort, 100000.0);
  153.      else
  154.          m_encparams.SetLambda(fsort, m_encparams.Lambda(fsort) *
  155.                              std::pow( (double)10.0, clipped_quality_ratio ) );
  156.      if (fsort == L1_frame)
  157.   m_encparams.SetL1MELambda( std::sqrt(m_encparams.L1Lambda()) * m_me_ratio );
  158.      else if (fsort == L2_frame)
  159.   m_encparams.SetL2MELambda( std::sqrt(m_encparams.L2Lambda()) * m_me_ratio );
  160. }
  161. double QualityMonitor::QualityVal(const PicArray& coded_data, const PicArray& orig_data , double cpd , const FrameSort fsort)
  162. {
  163.     // The number of regions to look at in assessing quality
  164.     int xregions( 4 );
  165.     int yregions( 3 );
  166.     if ( fsort == I_frame )
  167.     {
  168.         xregions = 1;
  169.         yregions = 1;
  170.     }
  171.     TwoDArray<long double> diff_array( yregions , xregions);
  172. long double diff;
  173.     OneDArray<int> xstart( diff_array.LengthX() );
  174.     OneDArray<int> xend( diff_array.LengthX() );
  175.     OneDArray<int> ystart( diff_array.LengthY() );
  176.     OneDArray<int> yend( diff_array.LengthX() );
  177.     for ( int i=0 ; i<xstart.Length() ; ++i)
  178.     { 
  179.         xstart[i] =( i * m_true_xl )/xstart.Length();
  180.         xend[i] = ( (i+1) * m_true_xl )/xstart.Length();
  181.     }
  182.     for ( int i=0 ; i<ystart.Length() ; ++i)
  183.     { 
  184.         ystart[i] =( i * m_true_yl )/ystart.Length();
  185.         yend[i] = ( (i+1) * m_true_yl )/ystart.Length();
  186.     }
  187.     for ( int q=0 ; q<diff_array.LengthY() ; ++q )
  188.     { 
  189.         for ( int p=0 ; p<diff_array.LengthX() ; ++p )
  190.         { 
  191.             diff_array[q][p] = 0.0;
  192.             for (int j=ystart[q]; j<yend[q]; ++j)
  193.             {
  194.                 for (int i=xstart[p]; i<xend[p]; ++i)
  195.                 {
  196.                     diff = static_cast<long double> ( coded_data[j][i] - orig_data[j][i] );
  197.  
  198.                     diff *= diff;
  199.                     diff *= diff;
  200.                     diff_array[q][p] += diff;
  201.                 }//i
  202.             }//j
  203.             diff_array[q][p] /= ( xend[p]-xstart[p] ) * ( yend[q]-ystart[q] );
  204.             diff_array[q][p] = std::sqrt( diff_array[q][p] );
  205.             // now compensate for the fact that we've got two extra bits
  206.             diff_array[q][p] /= 16.0;
  207.         }// p
  208.     }// q
  209.      
  210.     // return the worst area
  211.     long double worst_diff = diff_array[0][0];
  212.     for ( int q=0 ; q<diff_array.LengthY() ; ++q )
  213.     { 
  214.         for ( int p=0 ; p<diff_array.LengthX() ; ++p )
  215.         { 
  216.             if ( diff_array[q][p] > worst_diff )          
  217.                 worst_diff = diff_array[q][p];
  218.         }// p
  219.     }// q
  220. return static_cast<double> ( 10.0 * std::log10( 255.0*255.0 / worst_diff ) );
  221. }