LCOV - code coverage report
Current view: top level - synthesis/MeasurementEquations - IncCEMemModel.cc (source / functions) Hit Total Coverage
Test: casacpp_coverage.info Lines: 0 433 0.0 %
Date: 2024-12-11 20:54:31 Functions: 0 24 0.0 %

          Line data    Source code
       1             : //# IncCEMemModel.cc:  this implements IncCEMemModel
       2             : //# Copyright (C) 1996,1997,1998,1999,2000,2001,2003
       3             : //# Associated Universities, Inc. Washington DC, USA.
       4             : //#
       5             : //# This library is free software; you can redistribute it and/or modify it
       6             : //# under the terms of the GNU Library General Public License as published by
       7             : //# the Free Software Foundation; either version 2 of the License, or (at your
       8             : //# option) any later version.
       9             : //#
      10             : //# This library is distributed in the hope that it will be useful, but WITHOUT
      11             : //# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
      12             : //# FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Library General Public
      13             : //# License for more details.
      14             : //#
      15             : //# You should have received a copy of the GNU Library General Public License
      16             : //# along with this library; if not, write to the Free Software Foundation,
      17             : //# Inc., 675 Massachusetts Ave, Cambridge, MA 02139, USA.
      18             : //#
      19             : //# Correspondence concerning AIPS++ should be addressed as follows:
      20             : //#        Internet email: casa-feedback@nrao.edu.
      21             : //#        Postal address: AIPS++ Project Office
      22             : //#                        National Radio Astronomy Observatory
      23             : //#                        520 Edgemont Road
      24             : //#                        Charlottesville, VA 22903-2475 USA
      25             : //#
      26             : //# $Id$
      27             : 
      28             : #include <synthesis/MeasurementEquations/IncCEMemModel.h>
      29             : #include <synthesis/MeasurementEquations/CEMemProgress.h>
      30             : #include <synthesis/MeasurementEquations/LatConvEquation.h>
      31             : #include <casacore/casa/Arrays/IPosition.h>
      32             : #include <casacore/lattices/LEL/LatticeExpr.h>
      33             : #include <casacore/lattices/LEL/LatticeExprNode.h>
      34             : #include <casacore/lattices/Lattices/TiledLineStepper.h>
      35             : #include <casacore/lattices/Lattices/TempLattice.h>
      36             : #include <casacore/lattices/Lattices/LatticeIterator.h>
      37             : #include <casacore/casa/Arrays/Slice.h>
      38             : #include <casacore/casa/Arrays/Matrix.h>
      39             : #include <casacore/casa/Arrays/Vector.h>
      40             : #include <casacore/casa/Arrays/Array.h>
      41             : #include <casacore/casa/Arrays/ArrayError.h>
      42             : #include <casacore/casa/Arrays/ArrayMath.h>
      43             : #include <casacore/casa/Arrays/VectorIter.h>
      44             : #include <casacore/casa/Logging/LogOrigin.h>
      45             : #include <casacore/casa/BasicMath/Math.h>
      46             : #include <casacore/casa/Exceptions/Error.h>
      47             : #include <casacore/casa/Utilities/Assert.h>
      48             : #include <casacore/casa/System/PGPlotter.h>
      49             : #include <casacore/coordinates/Coordinates.h>
      50             : #include <iostream> 
      51             : #include <sstream>
      52             : 
      53             : 
      54             : using namespace casacore;
      55             : namespace casa { //# NAMESPACE CASA - BEGIN
      56             : 
      57             : //----------------------------------------------------------------------
      58           0 : IncCEMemModel::IncCEMemModel(IncEntropy &ent, 
      59             :                              Lattice<Float> & model,
      60             :                              Lattice<Float> & deltaModel,
      61             :                              uInt nIntegrations,
      62             :                              Float sigma,
      63             :                              Float targetFlux,
      64             :                              Bool useFluxConstraint,
      65             :                              Bool initializeModel,
      66           0 :                              Bool doImageplane) :
      67           0 :   itsEntropy_ptr(&ent),
      68           0 :   itsResidualEquation_ptr(0),
      69           0 :   itsModel_ptr(&model),
      70           0 :   itsDeltaModel_ptr(&deltaModel),
      71           0 :   itsPrior_ptr(0),
      72           0 :   itsMask_ptr(0),
      73           0 :   itsStep_ptr(0),
      74           0 :   itsResidual_ptr(0),
      75           0 :   itsInitializeModel(initializeModel),
      76           0 :   itsNumberIterations(nIntegrations),
      77           0 :   itsSigma(sigma),
      78           0 :   itsTargetFlux(targetFlux),
      79           0 :   itsQ(10.0),
      80           0 :   itsUseFluxConstraint(useFluxConstraint),
      81           0 :   itsDoImagePlane(doImageplane),
      82           0 :   itsThreshold0(0.0),
      83           0 :   itsThresholdSpeedup(0.0),
      84           0 :   itsAlpha(0.0),
      85           0 :   itsBeta(0.0),
      86           0 :   itsFlux(0.0),
      87           0 :   itsModelFlux(0.0),
      88           0 :   itsDeltaFlux(0.0),
      89           0 :   itsFirstIteration(0),
      90           0 :   itsChoose(false),
      91           0 :   itsLog(LogOrigin("IncCEMemModel", 
      92             :                    "IncCEMemModel(const Lattice<Float> & model)")),
      93           0 :   itsProgressPtr(0)
      94             : {
      95           0 :   initStuff();
      96           0 : };
      97             : 
      98             : //----------------------------------------------------------------------
      99           0 : IncCEMemModel::IncCEMemModel(IncEntropy &ent, 
     100             :                              Lattice<Float> & model,
     101             :                              Lattice<Float> & deltaModel,
     102             :                              Lattice<Float> & prior,
     103             :                              uInt nIntegrations,
     104             :                              Float sigma,
     105             :                              Float targetFlux,
     106             :                              Bool useFluxConstraint,
     107             :                              Bool initializeModel,
     108           0 :                              Bool doImageplane) :
     109           0 :   itsEntropy_ptr(&ent),
     110           0 :   itsResidualEquation_ptr(0),
     111           0 :   itsModel_ptr(&model),
     112           0 :   itsDeltaModel_ptr(&deltaModel),
     113           0 :   itsPrior_ptr(&prior),
     114           0 :   itsMask_ptr(0),
     115           0 :   itsStep_ptr(0),
     116           0 :   itsResidual_ptr(0),
     117           0 :   itsInitializeModel(initializeModel),
     118           0 :   itsNumberIterations(nIntegrations),
     119           0 :   itsSigma(sigma),
     120           0 :   itsTargetFlux(targetFlux),
     121           0 :   itsQ(10.0),
     122           0 :   itsUseFluxConstraint(useFluxConstraint),
     123           0 :   itsDoImagePlane(doImageplane),
     124           0 :   itsThreshold0(0.0),
     125           0 :   itsThresholdSpeedup(0.0),
     126           0 :   itsAlpha(0.0),
     127           0 :   itsBeta(0.0),
     128           0 :   itsFlux(0.0),
     129           0 :   itsModelFlux(0.0),
     130           0 :   itsDeltaFlux(0.0),
     131           0 :   itsFirstIteration(0),
     132           0 :   itsChoose(false),
     133           0 :   itsLog(LogOrigin("IncCEMemModel", 
     134             :                    "IncCEMemModel(const Lattice<Float> & model)")),
     135           0 :    itsProgressPtr(0)
     136             : {
     137           0 :   initStuff();
     138           0 : };
     139             : 
     140             : //----------------------------------------------------------------------
     141           0 : IncCEMemModel::IncCEMemModel(IncEntropy &ent, 
     142             :                        Lattice<Float> & model,
     143             :                        Lattice<Float> & deltaModel,
     144             :                        Lattice<Float> & prior,
     145             :                        Lattice<Float> & mask,
     146             :                        uInt nIntegrations,
     147             :                        Float sigma,
     148             :                        Float targetFlux,
     149             :                        Bool useFluxConstraint,
     150             :                        Bool initializeModel,
     151           0 :                        Bool doImageplane)
     152             :   :
     153           0 :   itsEntropy_ptr(&ent),
     154           0 :   itsResidualEquation_ptr(0),
     155           0 :   itsModel_ptr(&model),
     156           0 :   itsDeltaModel_ptr(&deltaModel),
     157           0 :   itsPrior_ptr(&prior),
     158           0 :   itsMask_ptr(&mask),
     159           0 :   itsStep_ptr(0),
     160           0 :   itsResidual_ptr(0),
     161           0 :   itsInitializeModel(initializeModel),
     162           0 :   itsNumberIterations(nIntegrations),
     163           0 :   itsSigma(sigma),
     164           0 :   itsTargetFlux(targetFlux),
     165           0 :   itsQ(10.0),
     166           0 :   itsUseFluxConstraint(useFluxConstraint),
     167           0 :   itsDoImagePlane(doImageplane),
     168           0 :   itsThreshold0(0.0),
     169           0 :   itsThresholdSpeedup(0.0),
     170           0 :   itsAlpha(0.0),
     171           0 :   itsBeta(0.0),
     172           0 :   itsFlux(0.0),
     173           0 :   itsModelFlux(0.0),
     174           0 :   itsDeltaFlux(0.0),
     175           0 :   itsFirstIteration(0),
     176           0 :   itsChoose(false), 
     177           0 :   itsLog(LogOrigin("IncCEMemModel", 
     178             :                    "IncCEMemModel(const Lattice<Float> & model)")),
     179           0 :   itsProgressPtr(0)
     180             : 
     181             : {
     182           0 :   initStuff();
     183           0 : };
     184             : 
     185             : //----------------------------------------------------------------------
     186           0 : IncCEMemModel::IncCEMemModel(IncEntropy &ent, 
     187             :                        Lattice<Float> & model,
     188             :                        Lattice<Float> & deltaModel,
     189             :                        uInt nIntegrations,
     190             :                        Lattice<Float> & mask,
     191             :                        Float sigma,
     192             :                        Float targetFlux,
     193             :                        Bool useFluxConstraint,
     194             :                        Bool initializeModel,
     195           0 :                        Bool doImageplane)
     196             :   :
     197           0 :   itsEntropy_ptr(&ent),
     198           0 :   itsResidualEquation_ptr(0),
     199           0 :   itsModel_ptr(&model),
     200           0 :   itsDeltaModel_ptr(&deltaModel),
     201           0 :   itsPrior_ptr(0),
     202           0 :   itsMask_ptr(&mask),
     203           0 :   itsStep_ptr(0),
     204           0 :   itsResidual_ptr(0),
     205           0 :   itsInitializeModel(initializeModel),
     206           0 :   itsNumberIterations(nIntegrations),
     207           0 :   itsSigma(sigma),
     208           0 :   itsTargetFlux(targetFlux),
     209           0 :   itsQ(10.0),
     210           0 :   itsUseFluxConstraint(useFluxConstraint),
     211           0 :   itsDoImagePlane(doImageplane),
     212           0 :   itsThreshold0(0.0),
     213           0 :   itsThresholdSpeedup(0.0),
     214           0 :   itsAlpha(0.0),
     215           0 :   itsBeta(0.0),
     216           0 :   itsFlux(0.0),
     217           0 :   itsModelFlux(0.0),
     218           0 :   itsDeltaFlux(0.0),
     219           0 :   itsFirstIteration(0),
     220           0 :   itsChoose(false), 
     221           0 :   itsLog(LogOrigin("IncCEMemModel", 
     222             :                    "IncCEMemModel(const Lattice<Float> & model)")),
     223           0 :   itsProgressPtr(0)
     224             : 
     225             : {
     226           0 :   initStuff();
     227           0 : };
     228             : 
     229             : 
     230             : //----------------------------------------------------------------------
     231           0 : IncCEMemModel::~IncCEMemModel()
     232             : {
     233           0 :   delete itsStep_ptr; itsStep_ptr=0;
     234           0 :   delete itsResidual_ptr; itsResidual_ptr=0;
     235           0 : };
     236             : 
     237             : 
     238             : 
     239             : //----------------------------------------------------------------------
     240           0 : void IncCEMemModel::setPrior(Lattice<Float> & prior)
     241             : {
     242           0 :   checkImages(itsModel_ptr, &prior); 
     243           0 :   itsPrior_ptr = &prior; 
     244           0 :   initStuff();
     245           0 : }
     246             : 
     247             : //----------------------------------------------------------------------
     248           0 : Float IncCEMemModel::getThreshold()
     249             : {
     250           0 :   if (itsThresholdSpeedup<= 0.0) {
     251           0 :     return itsThreshold0;
     252             :   } else {
     253           0 :     Float factor = pow(2.0, (Float)(itsIteration - itsFirstIteration)
     254           0 :                        / ((Float)(itsThresholdSpeedup)) );
     255           0 :     return (itsThreshold0 * factor);
     256             :   }
     257             : };
     258             : 
     259             : 
     260             : //----------------------------------------------------------------------
     261           0 : Bool IncCEMemModel::initStuff()
     262             : {
     263           0 :   checkImage(itsModel_ptr);
     264           0 :   checkImage(itsDeltaModel_ptr);
     265           0 :   checkImages(itsModel_ptr, itsDeltaModel_ptr);
     266             : 
     267             :   // initialize various parameters
     268             : 
     269           0 :   if (itsMask_ptr) {
     270           0 :     LatticeExprNode LEN = sum(*itsMask_ptr);
     271           0 :     itsNumberPixels = LEN.getFloat();
     272           0 :   } else {
     273           0 :     itsNumberPixels = itsModel_ptr->nelements();
     274             :   }
     275             : 
     276           0 :   formFlux();
     277             : 
     278           0 :   String entType;
     279           0 :   itsEntropy_ptr->entropyName(entType);
     280             : 
     281           0 :   Float clipVal = 1.0e-7;
     282           0 :   if (itsPrior_ptr != 0) {
     283           0 :     itsDefaultLevel = 0.0;
     284           0 :     if (entType == "ENTROPY") {
     285           0 :       Lattice<Float> &prior = *itsPrior_ptr;
     286           0 :       prior.copyData(  (LatticeExpr<Float>) (iif(prior < clipVal, clipVal, prior)) );     
     287             :     }
     288             :   } else {
     289           0 :     itsDefaultLevel = abs(itsTargetFlux) / itsNumberPixels;
     290             :   }
     291             : 
     292           0 :   Lattice<Float> &model = *itsModel_ptr;
     293           0 :   Lattice<Float> &delta = *itsDeltaModel_ptr;
     294           0 :   if (itsInitializeModel) {    
     295           0 :     if (itsFirstIteration == 0) {
     296           0 :       delta.set(itsDefaultLevel);
     297           0 :       if (entType == "ENTROPY") {
     298           0 :         delta.copyData( (LatticeExpr<Float>) (iif( (model+delta)<clipVal, (clipVal-model), delta)) );
     299             :       }
     300             :     } else {
     301           0 :       delta.set(0.0);
     302             :     }
     303             :   } else {
     304           0 :      if (entType == "ENTROPY") {
     305             :        // use the model passed in, but clip it to avoid negativity
     306           0 :        delta.copyData( (LatticeExpr<Float>) (iif((model+delta)<clipVal, (clipVal-model), delta)) );     
     307             :      }
     308             :   }
     309             : 
     310           0 :   applyMask( model );  // zero out masked pixels in model
     311             :    
     312           0 :   itsTolerance = 0.05;
     313           0 :   itsGain = 0.3;
     314           0 :   itsMaxNormGrad = 100.0;
     315             : 
     316           0 :   itsDoInit = true;
     317           0 :   Bool isOK = true;
     318             : 
     319             :   //Create temporary images
     320             : 
     321           0 :   if(itsStep_ptr) delete itsStep_ptr; itsStep_ptr=0;
     322           0 :   itsStep_ptr = new TempLattice<Float> (itsModel_ptr->shape(), 2);
     323           0 :   if(itsResidual_ptr) delete itsResidual_ptr; itsResidual_ptr=0;
     324           0 :   itsResidual_ptr = new TempLattice<Float> (itsModel_ptr->shape(), 2);
     325             : 
     326           0 :   if (itsStep_ptr &&  itsResidual_ptr) {
     327           0 :     itsStep_ptr->set(0.0);
     328           0 :     itsResidual_ptr->set(0.0);
     329             :   } else {
     330           0 :     isOK = false;
     331             :   }
     332             : 
     333             :   // We have been given an Entropy object, now we have to
     334             :   // tell it about US
     335           0 :   itsEntropy_ptr->setMemModel( *this );
     336             : 
     337           0 :   return isOK;
     338           0 : };
     339             : 
     340             : //----------------------------------------------------------------------
     341           0 : void IncCEMemModel::initializeAlphaBeta()
     342             : {
     343             :   // Note: in practice, this routine seems to never get called;
     344             :   // included for consistency with SDE MEM routine
     345           0 :   if (! itsUseFluxConstraint) {
     346           0 :     itsAlpha = max(0.0,  itsGDG(H,C)/ itsGDG(C,C) );
     347           0 :     itsBeta = 0.0;
     348             :   } else {
     349           0 :     Double det = itsGDG(C,C)*itsGDG(F,F) - itsGDG(C,F)*itsGDG(C,F);
     350           0 :     itsAlpha = (itsGDG(F,F)*itsGDG(H,C) - itsGDG(C,F)*itsGDG(H,F) )/det;
     351           0 :     itsBeta =  (itsGDG(C,C)*itsGDG(H,F) - itsGDG(C,F)*itsGDG(H,C) )/det;
     352             :   }
     353           0 : };
     354             : 
     355             : //----------------------------------------------------------------------
     356           0 : void IncCEMemModel::updateAlphaBeta()
     357             : {
     358             : 
     359             :   // code stolen from SDE's mem.f
     360           0 :   Double a = itsGDG(C,J)/itsGDG(C,C);
     361           0 :   Double b = square(a) - (itsGDG(J, J) - itsGain*itsLength)/ itsGDG(C, C);
     362             : 
     363             :   Double dAMax;
     364             :   Double dAMin;
     365             :   Double dBMax;
     366             :   Double dBMin;
     367             : 
     368           0 :   if (b > 0.0) {
     369           0 :     b = sqrt(b);
     370           0 :     dAMax = a + b;
     371           0 :     dAMin = a - b;
     372             :   } else {
     373           0 :     dAMax = 0.0;
     374           0 :     dAMin = 0.0;
     375             :   }
     376             : 
     377             :   Double dChisq;
     378             :   Double dAlpha;
     379             :   Double dBeta;
     380             :   Double dFlux;
     381             : 
     382           0 :   if ( ! itsUseFluxConstraint ) {
     383           0 :     dChisq =  itsChisq -  itsTargetChisq + itsGDG (C, J);
     384           0 :     dAlpha = dChisq/itsGDG(C, C);
     385             : 
     386           0 :     dAlpha = max(dAMin, min(dAMax, dAlpha));
     387           0 :     itsAlpha = max(0.0, itsAlpha + dAlpha);
     388             :   } else {
     389           0 :     a = itsGDG(F,J)/itsGDG(F,F);
     390           0 :     b = square(a) - (itsGDG(J, J) - itsGain*itsLength)/itsGDG(F,F);
     391           0 :     if ( b > 0.0) {
     392           0 :       b = sqrt((double)b);
     393           0 :       dBMax = a + b;
     394           0 :       dBMin = a - b;
     395             :     } else {
     396           0 :       dBMax = 0.0;
     397           0 :       dBMin = 0.0;
     398             :     }
     399             :     
     400           0 :     dChisq = itsChisq - itsTargetChisq + itsGDG(C, J);
     401           0 :     dFlux = itsFlux - itsTargetFlux +  itsGDG(F, J);
     402           0 :     Double det = itsGDG(C, C)*itsGDG(F, F) - square(itsGDG(F, C));
     403           0 :     dAlpha = (itsGDG(F, F)*dChisq - itsGDG(C, F)*dFlux)/det;
     404           0 :     dBeta =  (itsGDG(C, C)*dFlux  - itsGDG(C, F)*dChisq)/det;
     405             :     
     406           0 :     dAlpha = max(dAMin, min(dAMax, dAlpha));
     407           0 :     itsAlpha = max(0.0, itsAlpha + dAlpha);
     408           0 :     dBeta    = max(dBMin, min(dBMax, dBeta));
     409           0 :     itsBeta  = itsBeta + dBeta;
     410             :   }
     411             : 
     412           0 : };
     413             : 
     414             : 
     415             : 
     416             : //----------------------------------------------------------------------
     417           0 : void IncCEMemModel::changeAlphaBeta()
     418             : {
     419           0 :   formGDG();
     420           0 :   itsLength = itsGDG(H,H) + square(itsAlpha) * itsGDG(C,C) 
     421           0 :     + square(itsBeta) * itsGDG(F,F);
     422           0 :   if (itsAlpha == 0.0 && itsBeta == 0.0) {
     423           0 :     itsLength = itsGDG(F,F);
     424             :   }
     425           0 :   itsNormGrad = itsGDG(J,J) / itsLength;
     426           0 :   if (itsAlpha == 0.0) 
     427           0 :     itsNormGrad = 0.0;
     428           0 :   if (itsNormGrad < itsGain) {
     429           0 :     updateAlphaBeta();
     430             :   } else {
     431           0 :     initializeAlphaBeta();
     432             :   }
     433           0 : };
     434             : 
     435             : 
     436             : 
     437             : //----------------------------------------------------------------------
     438           0 : Bool IncCEMemModel::checkImage(const Lattice<Float> * /*im*/)
     439             : {
     440             :   // I guess we don't have anything to do
     441           0 :   return true;
     442             : };
     443             : 
     444             : 
     445             : 
     446             : //----------------------------------------------------------------------
     447           0 : Bool IncCEMemModel::checkImages(const Lattice<Float> *one, const Lattice<Float> *other)
     448             : {
     449           0 :   Bool isOK = true;
     450             : 
     451           0 :   for (uInt i = 0; i < one->ndim(); i++) {
     452           0 :     AlwaysAssert(one->shape()(i) == other->shape()(i), AipsError);
     453             :   }
     454           0 :   return isOK;
     455             : };
     456             : 
     457             : 
     458             : //----------------------------------------------------------------------
     459           0 : Bool IncCEMemModel::ok() 
     460             : {
     461           0 :   Bool isOK = true;
     462           0 :   if (!itsModel_ptr) {
     463           0 :     isOK = false;
     464           0 :   } else if (!itsDeltaModel_ptr) {
     465           0 :     isOK = false;
     466             :   } else {
     467           0 :     isOK = (checkImage(itsModel_ptr) && checkImage(itsDeltaModel_ptr));
     468           0 :     checkImages(itsModel_ptr, itsDeltaModel_ptr);
     469           0 :     if (itsPrior_ptr) {
     470           0 :       checkImages(itsModel_ptr, itsPrior_ptr);
     471             :     }
     472           0 :     if (itsMask_ptr) {
     473           0 :       checkImages(itsModel_ptr, itsMask_ptr);
     474             :     }
     475           0 :     if (itsStep_ptr) {
     476           0 :       checkImages(itsModel_ptr, itsStep_ptr);
     477             :     }
     478           0 :     if (itsResidual_ptr) {
     479           0 :       checkImages(itsModel_ptr, itsResidual_ptr);
     480             :     }
     481           0 :     if (! itsEntropy_ptr) {
     482           0 :       isOK = false;
     483             :     }
     484             : 
     485             :     // Also need to check state variables in the future!
     486             :   }
     487           0 :   return isOK;
     488             : };
     489             : 
     490           0 : void IncCEMemModel::setMask(Lattice<Float> & mask)
     491             : { 
     492           0 :   checkImages(itsModel_ptr, &mask); 
     493           0 :   itsMask_ptr = &mask;
     494           0 :   initStuff();
     495           0 : }
     496             : 
     497             : 
     498           0 : void IncCEMemModel::state()
     499             : {
     500           0 :    if (itsPrior_ptr == 0) {
     501           0 :     itsLog << "Using blank prior image" << LogIO::POST;
     502             :   } else {
     503           0 :     itsLog << "Using prior image "  << LogIO::POST;
     504             :   }
     505           0 :   if (itsMask_ptr != 0) {
     506           0 :     itsLog << "Using mask to restrict emission" << LogIO::POST;
     507             :   }
     508           0 : }
     509             : 
     510             : //----------------------------------------------------------------------
     511           0 : Bool IncCEMemModel::solve(ResidualEquation<Lattice<Float> >  & eqn) 
     512             : {
     513             : 
     514           0 :   itsResidualEquation_ptr = &eqn;
     515           0 :   Bool converged = false;
     516           0 :   Bool endNow = false;
     517           0 :   state();
     518             : 
     519           0 :   itsEntropy_ptr->infoBanner();
     520             : 
     521           0 :   for (itsIteration = itsFirstIteration;
     522           0 :        ( (itsIteration < itsNumberIterations) && !endNow);
     523           0 :        itsIteration++) {
     524           0 :     if (itsDoImagePlane) {
     525           0 :       itsTargetChisq = square(itsSigma) * itsNumberPixels;
     526             :     } else {
     527           0 :       itsTargetChisq = square(itsSigma) * itsNumberPixels / itsQ;
     528             :     }
     529             : 
     530           0 :     oneIteration();
     531             : 
     532           0 :     itsFit = sqrt((double)( itsChisq / itsTargetChisq) );
     533           0 :     itsAFit = max(itsFit, 1.0f) * 
     534           0 :       sqrt((double)( itsTargetChisq / itsNumberPixels) );
     535           0 :     itsEntropy_ptr->infoPerIteration(itsIteration);
     536             : 
     537           0 :     converged = itsEntropy_ptr->testConvergence();
     538           0 :     if (! converged && getThreshold() > 0.0) {
     539           0 :       converged = testConvergenceThreshold();
     540             :     }
     541             : 
     542           0 :     if (itsNormGrad > itsMaxNormGrad) {
     543           0 :       endNow = true;
     544           0 :       itsLog << " Excessive gradient: stopping now" << LogIO::EXCEPTION;
     545             :     }
     546             : 
     547           0 :     if (converged) {
     548           0 :       endNow = true;;
     549           0 :       itsLog << "Converged at iteration " << itsIteration+1 << LogIO::POST;
     550             :     }
     551             :   }
     552           0 :   if (!converged) {
     553           0 :     itsLog << "MEM failed to converge after " <<  itsNumberIterations+1 
     554           0 :            << " iterations" << LogIO::POST;
     555             :   }    
     556             : 
     557           0 :   formFlux();
     558           0 :   return converged;
     559             : };
     560             : 
     561           0 : Bool IncCEMemModel::testConvergenceThreshold()
     562             : { 
     563           0 :   Bool less = false; 
     564           0 :   if (getThreshold() > 0.0) {
     565           0 :    if (itsCurrentPeakResidual < getThreshold() ) {
     566           0 :      less = true;
     567             :    }
     568             :   }
     569           0 :   return less;
     570             : };
     571             : 
     572           0 : Bool IncCEMemModel::applyMask( Lattice<Float> & lat ) {
     573           0 :   if (itsMask_ptr) {
     574           0 :     LatticeExpr<Float> exp = ( (*itsMask_ptr) * (lat) );
     575           0 :     lat.copyData( exp );
     576           0 :     return true;
     577           0 :   } else {
     578           0 :     return false;
     579             :   }
     580             : };
     581             : 
     582             : 
     583           0 : void  IncCEMemModel::oneIteration()
     584             : {
     585           0 :   ok();
     586             : 
     587           0 :   if (itsDoInit) {
     588           0 :     itsDoInit = false;
     589             :     // passing *this reverts to the LinearModel from which we are derived
     590           0 :     if (itsMask_ptr) {
     591           0 :       itsResidualEquation_ptr->residual( *itsResidual_ptr, 
     592           0 :                                          itsChisq,
     593           0 :                                          *itsMask_ptr,
     594             :                                          *this);
     595             :     } else {
     596           0 :       itsResidualEquation_ptr->residual( *itsResidual_ptr, 
     597           0 :                                          itsChisq,
     598             :                                          *this);
     599             :     }
     600           0 :     if (!itsDoImagePlane) itsChisq /= itsQ;  
     601             : 
     602           0 :     if (itsAlpha == 0.0 )
     603           0 :       changeAlphaBeta();
     604             :   }
     605             : 
     606           0 :   calculateStep();
     607           0 :   relaxMin();
     608             :   
     609             :   // limit the step to less than the gain
     610           0 :   Float scale = 1.0;
     611           0 :   Float scalem = 1.0;
     612           0 :   if (itsNormGrad > 0.0)
     613           0 :     scalem = itsGain/itsNormGrad;
     614           0 :   scale = min(1.0f, scalem);
     615             : 
     616           0 :   takeStep(1.0, scale);
     617             : 
     618             :   // passing *this reverts to the LinearModel from which we are derived
     619           0 :   if (itsMask_ptr) {
     620           0 :     itsResidualEquation_ptr->residual( *itsResidual_ptr, 
     621           0 :                                        itsChisq,
     622           0 :                                        *itsMask_ptr,
     623             :                                        *this);
     624             :   } else {
     625           0 :       itsResidualEquation_ptr->residual( *itsResidual_ptr, 
     626           0 :                                          itsChisq,
     627             :                                          *this);
     628             :   }
     629           0 :   if (!itsDoImagePlane) itsChisq /= itsQ;  
     630             : 
     631           0 :   if (itsThreshold0 > 0.0) {
     632           0 :     LatticeExprNode LEN = max(*itsResidual_ptr);
     633           0 :     Float rmax = LEN.getFloat();
     634           0 :     LEN = min(*itsResidual_ptr);
     635           0 :     Float rmin = LEN.getFloat();
     636           0 :     itsCurrentPeakResidual =  max (abs(rmax), abs(rmin));
     637           0 :   }
     638             : 
     639             :   // form Gradient dot Step
     640           0 :   formGDS();
     641             :   
     642             : 
     643             :   // determine optimal step
     644           0 :   Double eps = 1.0;
     645           0 :   if (itsGradDotStep0 != itsGradDotStep1) 
     646           0 :     eps = itsGradDotStep0/(itsGradDotStep0-itsGradDotStep1);
     647           0 :   if (scale != 0.0f) eps = min(eps, (Double)(scalem/scale) );
     648           0 :   if (eps <= 0.0) {
     649           0 :     eps = 1.0;
     650           0 :     itsLog << LogIO::WARN << "Step is zero" << LogIO::POST;
     651             :   }
     652             :   
     653             :   // Step to optimum point
     654             :   
     655           0 :   if (abs(eps-1.0) > itsGain) {
     656           0 :     takeStep(1.0, scale*(eps-1.0));
     657             : 
     658             :     // passing *this reverts to the LinearModel from which we are derived
     659           0 :     if (itsMask_ptr) {
     660           0 :       itsResidualEquation_ptr->residual( *itsResidual_ptr, 
     661           0 :                                          itsChisq,
     662           0 :                                          *itsMask_ptr,
     663             :                                          *this);                                       
     664             :     } else {
     665           0 :       itsResidualEquation_ptr->residual( *itsResidual_ptr, 
     666           0 :                                          itsChisq,
     667             :                                          *this);                                       
     668             :     }
     669             : 
     670           0 :     if (!itsDoImagePlane) itsChisq /= itsQ;  
     671             : 
     672           0 :     if (itsThreshold0 > 0.0) {
     673           0 :       LatticeExprNode LEN = max(*itsResidual_ptr);
     674           0 :       Float rmax = LEN.getFloat();
     675           0 :       LEN = min(*itsResidual_ptr);
     676           0 :       Float rmin = LEN.getFloat();
     677           0 :       itsCurrentPeakResidual =  max (abs(rmax), abs(rmin));
     678           0 :     }    
     679             :   }
     680             : 
     681           0 :   formEntropy();
     682           0 :   formFlux();
     683             : 
     684             :   // readjust beam volume
     685           0 :   itsQ = itsQ*(1.0/max(0.5, min(2.0,eps))+1.0)/2.0;
     686             : 
     687           0 :   changeAlphaBeta();
     688           0 : };
     689             : 
     690             : 
     691             : 
     692             : //----------------------------------------------------------------------
     693           0 : void IncCEMemModel::calculateStep()
     694             : {
     695             :   
     696           0 :   formGDGStep();
     697             : 
     698           0 :   formFlux();
     699           0 :   itsGradDotStep0 = itsGDG(J,J);
     700           0 :   itsLength = itsGDG(H,H) + square(itsAlpha) * itsGDG(C,C) 
     701           0 :     + square(itsBeta) * itsGDG(F,F);
     702           0 :   if (itsLength <= 0.0) {
     703           0 :     itsLength = itsGDG(F,F);
     704             :   }
     705           0 :   itsNormGrad = itsGDG(J,J) / itsLength;
     706           0 : };
     707             : 
     708             : 
     709             : //----------------------------------------------------------------------
     710           0 : void IncCEMemModel::takeStep(Float wt1, Float wt2)
     711             : {
     712           0 :   Lattice<Float> &model = *itsModel_ptr;
     713           0 :   Lattice<Float> &delta = *itsDeltaModel_ptr;
     714           0 :   Lattice<Float> &step  = *itsStep_ptr;
     715           0 :   LatticeExprNode  node;
     716           0 :   String entType;
     717           0 :   itsEntropy_ptr->entropyName(entType);
     718           0 :   if (entType == "ENTROPY") {
     719             : 
     720           0 :     itsRequiredModelMin = max (itsRequiredModelMin, 1.0e-7);
     721             :     
     722           0 :     node = iif( (model + wt1*delta  + wt2*step) > itsRequiredModelMin,
     723           0 :                 (wt1*delta  + wt2*step),
     724           0 :                 itsRequiredModelMin - model);
     725           0 :     delta.copyData( (LatticeExpr<Float>)node);
     726           0 :   } else if (entType == "EMPTINESS") {
     727             : 
     728           0 :     delta.copyData( (LatticeExpr<Float>)( wt1*delta  + wt2*step ) );
     729             : 
     730             :   } else {
     731           0 :     cout << "UNKNOWN ENTROPY TYPE" << endl;
     732           0 :     AlwaysAssert( ((entType == "ENTROPY") || (entType == "EMPTINESS")),
     733             :                   AipsError);
     734             :   }
     735             : 
     736           0 :   applyMask( delta );
     737           0 : };
     738             : 
     739             : 
     740             : 
     741             : //----------------------------------------------------------------------
     742           0 : Float IncCEMemModel::formFlux()
     743             : {
     744           0 :   itsFlux = 0.0;
     745           0 :   itsModelFlux = 0.0;
     746           0 :   itsDeltaFlux = 0.0;
     747           0 :   itsModelMin =  1e+20;
     748           0 :   itsModelMax = -1e+20;
     749             : 
     750           0 :   Lattice<Float> &model = *itsModel_ptr;
     751           0 :   Lattice<Float> &delta = *itsDeltaModel_ptr;
     752           0 :   TiledLineStepper mtls(model.shape(), model.niceCursorShape(), 0);
     753           0 :   RO_LatticeIterator<Float> mod( model, mtls); 
     754           0 :   RO_LatticeIterator<Float> del( delta, mtls); 
     755           0 :   Float pixsum = 0.0;
     756             : 
     757           0 :   for (mod.reset(), del.reset(); !mod.atEnd(); mod++, del++) {
     758           0 :     for (uInt i=0;i<mod.vectorCursor().nelements();i++) {
     759           0 :       itsModelFlux +=  mod.vectorCursor()(i);
     760           0 :       itsDeltaFlux +=  del.vectorCursor()(i);
     761           0 :       pixsum = mod.vectorCursor()(i) + del.vectorCursor()(i);      
     762           0 :       if (pixsum > itsModelMax)
     763           0 :         itsModelMax = pixsum;
     764           0 :       if (pixsum < itsModelMin)
     765           0 :         itsModelMin = pixsum;
     766             :     }
     767             :   }
     768           0 :   itsFlux = itsDeltaFlux + itsModelFlux;
     769             : 
     770           0 :   return itsFlux;
     771           0 : };
     772             : 
     773             : 
     774             : 
     775             : } //# NAMESPACE CASA - END
     776             : 

Generated by: LCOV version 1.16