LCOV - code coverage report
Current view: top level - synthesis/MeasurementComponents - VisCalSolver.cc (source / functions) Hit Total Coverage
Test: casacpp_coverage.info Lines: 24 596 4.0 %
Date: 2024-11-06 17:42:47 Functions: 2 20 10.0 %

          Line data    Source code
       1             : //# VisCalSolver.cc: Implementation of generic visibility solving
       2             : //# Copyright (C) 1996,1997,1998,1999,2000,2001,2002,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             : 
      27             : #include <synthesis/MeasurementComponents/VisCalSolver.h>
      28             : 
      29             : #include <msvis/MSVis/VisBuffer.h>
      30             : 
      31             : #include <casacore/casa/Arrays/ArrayMath.h>
      32             : #include <casacore/casa/Arrays/MaskArrMath.h>
      33             : #include <casacore/casa/Arrays/ArrayLogical.h>
      34             : #include <casacore/casa/Arrays/ArrayIter.h>
      35             : //#include <scimath/Mathematics/MatrixMathLA.h>
      36             : #include <casacore/casa/BasicSL/String.h>
      37             : #include <casacore/casa/BasicMath/Math.h>
      38             : #include <casacore/casa/Utilities/Assert.h>
      39             : #include <casacore/casa/Exceptions/Error.h>
      40             : #include <casacore/casa/OS/Memory.h>
      41             : #include <casacore/casa/OS/Path.h>
      42             : 
      43             : #include <sstream>
      44             : 
      45             : #include <casacore/casa/Logging/LogMessage.h>
      46             : #include <casacore/casa/Logging/LogSink.h>
      47             : 
      48             : #define VCS_PRTLEV 0
      49             : 
      50             : using namespace casacore;
      51             : namespace casa { //# NAMESPACE CASA - BEGIN
      52             : 
      53             : 
      54             : // **********************************************************
      55             : //  UVMod Implementations
      56             : //
      57             : 
      58           2 : VisCalSolver::VisCalSolver() :
      59           2 :   svb_(NULL),
      60           2 :   vbga_(NULL),
      61           2 :   ve_(NULL),
      62           2 :   svc_(NULL),
      63           2 :   nTotalPar_(0),
      64           2 :   R_(),dR_(),Rflg_(),
      65           2 :   maxIter_(50),
      66           2 :   chiSq_(0.0),
      67           2 :   chiSqV_(4,0.0),
      68           2 :   lastChiSq_(0.0),dChiSq_(0.0),
      69           2 :   sumWt_(0.0),nWt_(0),
      70           2 :   cvrgcount_(0),
      71           2 :   par_(), parOK_(), parErr_(), srcPar_(), lastCalPar_(), lastSrcPar_(),
      72           2 :   dpar_(), dcalpar_(), dsrcpar_(),
      73           2 :   grad_(),hess_(),
      74           2 :   lambda_(2.0),
      75           2 :   optstep_(true),
      76           2 :   prtlev_(VCS_PRTLEV)
      77             : {
      78           2 :   if (prtlev()>0) cout << "VCS::VCS()" << endl;
      79           2 : }
      80             : 
      81           2 : VisCalSolver::~VisCalSolver() 
      82             : {  
      83           2 :   if (prtlev()>0) cout << "VCS::~VCS()" << endl;
      84           2 : }
      85             : 
      86           0 : Bool VisCalSolver::solve(VisEquation& ve, SolvableVisCal& svc, VisBuffer& svb) {
      87             : 
      88           0 :   if (prtlev()>1) cout << "VCS::solve(,,)" << endl;
      89             : 
      90             :   /*
      91             :   LogSink logsink;
      92             :   {
      93             :     LogMessage message(LogOrigin("VisCalSolver", "solve"));
      94             :     ostringstream o; o<<"Beginning solve...";
      95             :     message.message(o);
      96             :     logsink.post(message);
      97             :   }
      98             :   */
      99             :   // Pointers to local ve,svc,svb
     100           0 :   ve_=&ve;
     101           0 :   svc_=&svc;
     102           0 :   svb_=&svb;
     103           0 :   vbga_=NULL;
     104             : 
     105             :   // Verify that VisEq has the correct svc:
     106             :   // TBD?
     107             : 
     108             :   // Initialize everything 
     109           0 :   initSolve();
     110             : 
     111           0 :   Vector<Float> steplist(maxIter_+2,0.0);
     112           0 :   Vector<Float> rsteplist(maxIter_+2,0.0);
     113             : 
     114             :   //  cout << "svb.modelVisCube() = " << phase(svb.modelVisCube())*180.0/C::pi << endl;
     115             :   //  cout << "svb.modelVisCube() = " << amplitude(svb.modelVisCube()) << endl;
     116             : 
     117             :   // Verify VisBuffer validity for solving
     118             :   //   (this sets parOK() on per-antenna basis (for focusChan)
     119             :   //    based on data weights and baseline participation)
     120           0 :   Bool oktosolve = svc_->verifyForSolve(*svb_);
     121             : 
     122           0 :   if (oktosolve) {
     123             :     
     124           0 :     if (prtlev()>1) cout << "First guess:" << endl
     125           0 :                          << "amp = " << amplitude(par()) << endl
     126           0 :                          << "pha = " << phase(par()) 
     127           0 :                          << endl;
     128             : 
     129             :     // Iterate solution
     130           0 :     Int iter(0);
     131           0 :     Bool done(false);
     132           0 :     while (!done) {
     133             :       
     134           0 :       if (prtlev()>2) cout << " Beginning iteration " << iter 
     135           0 :                            << "---------------------------------" << endl;
     136             :       
     137             :       // Differentiate the VB and get current Chi2
     138           0 :       differentiate();
     139           0 :       chiSquare();
     140           0 :       if (chiSq()==0.0) {
     141             :         //      cout << "CHI2 IS SPURIOUSLY ZERO!*************************************" << endl;
     142             :         //      cout << "R() = " << R() << endl;
     143             :         //      cout << "wtmat = " << svb.weightMat() << endl;
     144             :         //      cout << "flag = " << svb.flag() << endl;
     145             :         //      cout << "sum(wtmat) = " << sum(wtmat) << endl;
     146           0 :         return false;
     147             :       }
     148             : 
     149           0 :       dChiSq() = chiSq()-lastChiSq();
     150             : 
     151             :       //      cout << "chi2 = " << chiSq() << " " << dChiSq() << " " << dChiSq()/chiSq() << endl;
     152             :       
     153             :       // Continuue if we haven't converged
     154           0 :       if (!converged()) {
     155             :         
     156           0 :         if (dChiSq()<=0.0) {
     157             :           // last step was good...
     158           0 :           lastChiSq()=chiSq();
     159             :           
     160             :           // so accumulate new grad/hess...
     161           0 :           accGradHess();
     162             :           
     163             :           //...and adjust lambda downward
     164             :           //    lambda()/=2.0;
     165             :           //    lambda()=0.8;
     166           0 :           lambda()=1.0;
     167             :         }
     168             :         else {
     169             :           //      cout << "reverting..." << chiSq() << " " << dChiSq() << " (" << iter << ")" << endl;
     170             :           // last step was bad, revert to previous 
     171           0 :           revert();
     172             :           //...with a larger lambda
     173             :           //    lambda()*=4.0;
     174           0 :           lambda()=1.0;
     175             :         }
     176             :         
     177             :         // Solve for the parameter step
     178           0 :         solveGradHess();
     179             :         
     180             :         // Remember curr pars
     181           0 :         lastCalPar()=par();
     182           0 :         if (svc_->solvePol())
     183           0 :           lastSrcPar()=srcPar();
     184             :         
     185             :         // Refine the step size by exploring chi2 in the
     186             :         //  gradient direction
     187           0 :         if (optstep_) //  && cvrgcount_>=3)
     188           0 :           optStepSize();
     189             :         
     190             :         // Update current parameters (saves a copy of them)
     191           0 :         updatePar();
     192             : 
     193             :         //      cout << "srcPar() = " << srcPar() << endl;
     194             : 
     195           0 :         steplist(iter)=max(amplitude(dCalPar()));
     196           0 :         rsteplist(iter)=max(amplitude(dCalPar())/amplitude(par()));
     197             : 
     198             :       }
     199             :       else {
     200             :         // Convergence means we're done!
     201           0 :         done=true;
     202             : 
     203           0 :         if (prtlev()>0) {
     204           0 :           cout << "Iterations =" << iter << endl;
     205           0 :           cout << "par()=" << par() << endl;
     206           0 :           cout << "srcPar()=" << srcPar() << endl;
     207             :         }
     208             : 
     209             : 
     210             :         /*
     211             :         cout << " good pars=" << ntrue(parOK())
     212             :              << " iterations=" << iter << endl
     213             :              << " steps=" << steplist(IPosition(1,0),IPosition(1,iter)) 
     214             :              << endl
     215             :              << " rsteps=" << rsteplist(IPosition(1,0),IPosition(1,iter)) 
     216             :              << endl;
     217             :         */   
     218             : 
     219             :         // Get parameter errors:
     220           0 :         accGradHess();
     221           0 :         getErrors();
     222             : 
     223             :         // Return, signaling success if at least 1 good solution
     224           0 :         return (ntrue(parOK())>0);
     225             :         
     226             :       }
     227             :       
     228             :       // Escape iteration loop via iteration limit
     229           0 :       if (iter==maxIter()) {
     230           0 :         cout << "Reached iteration limit: " << iter << " iterations.  " << endl;
     231             :         //      cout << " good pars = " << ntrue(parOK())
     232             :         //           << "  steps = " << steplist
     233             :         //           << endl;
     234           0 :         done=true;
     235             :       }
     236             :       
     237             :       // Advance iteration counter
     238           0 :       iter++;
     239             :     }
     240             :     
     241             :   }
     242             :   else {
     243           0 :     cout << " Insufficient unflagged antennas to proceed with this solve." << endl;
     244             :   }
     245             : 
     246           0 :   return false;
     247             :     
     248           0 : }
     249             : 
     250             : // New VisBuffGroupAcc version
     251           0 : Bool VisCalSolver::solve(VisEquation& ve, SolvableVisCal& svc, VisBuffGroupAcc& vbga) {
     252             : 
     253           0 :   if (prtlev()>1) cout << "VCS::solve(,,VBGA)" << endl;
     254             : 
     255             :   /*
     256             :   LogSink logsink;
     257             :   {
     258             :     LogMessage message(LogOrigin("VisCalSolver", "solve"));
     259             :     ostringstream o; o<<"Beginning solve...";
     260             :     message.message(o);
     261             :     logsink.post(message);
     262             :   }
     263             :   */
     264             :   // Pointers to local ve,svc,svb
     265           0 :   ve_=&ve;
     266           0 :   svc_=&svc;
     267           0 :   svb_=NULL;
     268           0 :   vbga_=&vbga;
     269             : 
     270             :   // Verify that VisEq has the correct svc:
     271             :   // TBD?
     272             : 
     273             :   // Initialize everything 
     274           0 :   initSolve();
     275             : 
     276           0 :   Vector<Float> steplist(maxIter_+2,0.0);
     277           0 :   Vector<Float> rsteplist(maxIter_+2,0.0);
     278             : 
     279             :   // Verify Data's validity for solve w.r.t. baselines available
     280             :   //   (this sets parOK() on per-antenna basis (for focusChan)
     281             :   //    based on data weights and baseline participation)
     282           0 :   Bool oktosolve = svc_->verifyConstraints(*vbga_);
     283             : 
     284           0 :   if (oktosolve) {
     285             :     
     286           0 :     if (prtlev()>1) cout << "First guess:" << endl
     287           0 :                          << "amp = " << amplitude(par()) << endl
     288           0 :                          << "pha = " << phase(par()) 
     289           0 :                          << endl;
     290             : 
     291             :     // Iterate solution
     292           0 :     Int iter(0);
     293           0 :     Bool done(false);
     294           0 :     while (!done) {
     295             :       
     296           0 :       if (prtlev()>2) cout << " Beginning iteration " << iter 
     297           0 :                            << "---------------------------------" << endl;
     298             :       
     299             :       // Differentiate the VB and get current Chi2
     300           0 :       differentiate2();
     301           0 :       chiSquare2();
     302           0 :       if (chiSq()==0.0) {
     303             :         //      cout << "CHI2 IS SPURIOUSLY ZERO!*************************************" << endl;
     304             :         //      cout << "R() = " << R() << endl;
     305             :         //      cout << "wtmat = " << svb.weightMat() << endl;
     306             :         //      cout << "flag = " << svb.flag() << endl;
     307             :         //      cout << "sum(wtmat) = " << sum(wtmat) << endl;
     308           0 :         return false;
     309             :       }
     310             : 
     311           0 :       dChiSq() = chiSq()-lastChiSq();
     312             : 
     313             :       //      cout << "chi2 = " << chiSq() << " " << dChiSq() << " " << dChiSq()/chiSq() << endl;
     314             :       
     315             :       // Continuue if we haven't converged
     316           0 :       if (!converged()) {
     317             :         
     318           0 :         if (dChiSq()<=0.0) {
     319             :           // last step was good...
     320           0 :           lastChiSq()=chiSq();
     321             :           
     322             :           // so accumulate new grad/hess...
     323           0 :           accGradHess2();
     324             :           
     325             :           //...and adjust lambda downward
     326             :           //    lambda()/=2.0;
     327             :           //    lambda()=0.8;
     328           0 :           lambda()=1.0;
     329             :         }
     330             :         else {
     331             :           //      cout << "reverting..." << chiSq() << " " << dChiSq() << " (" << iter << ")" << endl;
     332             :           // last step was bad, revert to previous 
     333           0 :           revert();
     334             :           //...with a larger lambda
     335             :           //    lambda()*=4.0;
     336           0 :           lambda()=1.0;
     337             :         }
     338             :         
     339             :         // Solve for the parameter step
     340           0 :         solveGradHess();
     341             :         
     342             :         // Remember curr pars
     343           0 :         lastCalPar()=par();
     344           0 :         if (svc_->solvePol())
     345           0 :           lastSrcPar()=srcPar();
     346             :         
     347             :         // Refine the step size by exploring chi2 in the
     348             :         //  gradient direction
     349           0 :         if (optstep_) //  && cvrgcount_>=3)
     350           0 :           optStepSize2();
     351             :         
     352             :         // Update current parameters (saves a copy of them)
     353           0 :         updatePar();
     354             : 
     355             :         //      cout << "srcPar() = " << srcPar() << endl;
     356             : 
     357           0 :         steplist(iter)=max(amplitude(dCalPar()));
     358           0 :         rsteplist(iter)=max(amplitude(dCalPar())/amplitude(par()));
     359             : 
     360             :       }
     361             :       else {
     362             :         // Convergence means we're done!
     363           0 :         done=true;
     364             : 
     365           0 :         if (prtlev()>0) {
     366           0 :           cout << "Iterations =" << iter << endl;
     367           0 :           cout << "par()=" << par() << endl;
     368           0 :           cout << "srcPar()=" << srcPar() << endl;
     369             :         }
     370             : 
     371             : 
     372             :         /*
     373             :         cout << " good pars=" << ntrue(parOK())
     374             :              << " iterations=" << iter << endl
     375             :              << " steps=" << steplist(IPosition(1,0),IPosition(1,iter)) 
     376             :              << endl
     377             :              << " rsteps=" << rsteplist(IPosition(1,0),IPosition(1,iter)) 
     378             :              << endl;
     379             :         */   
     380             : 
     381             :         // Get parameter errors:
     382           0 :         accGradHess2();
     383           0 :         getErrors();
     384             : 
     385             :         // Return, signaling success if at least 1 good solution
     386           0 :         return (ntrue(parOK())>0);
     387             :         
     388             :       }
     389             :       
     390             :       // Escape iteration loop via iteration limit
     391           0 :       if (iter==maxIter()) {
     392           0 :         cout << "Reached iteration limit: " << iter << " iterations.  " << endl;
     393             :         //      cout << " good pars = " << ntrue(parOK())
     394             :         //           << "  steps = " << steplist
     395             :         //           << endl;
     396           0 :         done=true;
     397             :       }
     398             :       
     399             :       // Advance iteration counter
     400           0 :       iter++;
     401             :     }
     402             :     
     403             :   }
     404             :   else {
     405           0 :     cout << " Insufficient unflagged antennas to proceed with this solve." << endl;
     406             :   }
     407             : 
     408           0 :   return false;
     409             :     
     410           0 : }
     411             :   
     412           0 : void VisCalSolver::initSolve() {
     413             :     
     414           0 :   if (prtlev()>2) cout << " VCS::initSolve()" << endl;
     415             : 
     416             :   // Get total number of cal parameters from svc info
     417           0 :   nCalPar()=svc().nTotalPar();
     418             : 
     419             :   // solvePol() tells us how many source pol params we are solving for
     420           0 :   nSrcPar()=svc().solvePol();
     421             : 
     422             :   // the total number of parameters
     423           0 :   nTotalPar()=nCalPar()+nSrcPar();
     424             : 
     425           0 :   if (prtlev()>2)
     426           0 :     cout << "  Total parameters in solve: " << nTotalPar() << endl;
     427             : 
     428             :   // Chi2 and weights
     429           0 :   chiSq()=0.0;
     430           0 :   lastChiSq()=DBL_MAX;
     431           0 :   dChiSq()=0.0;
     432             :     
     433           0 :   sumWt()=0.0;
     434           0 :   nWt()=0;
     435             : 
     436             :   // Link up svc's internal pars with local reference
     437             :   //   (only if shape is correct)
     438             : 
     439           0 :   if (svc().solveCPar().nelements()==uInt(nCalPar())) {
     440           0 :     par().reference(svc().solveCPar().reform(IPosition(1,nCalPar())));
     441           0 :     parOK().reference(svc().solveParOK().reform(IPosition(1,nCalPar())));
     442           0 :     parErr().reference(svc().solveParErr().reform(IPosition(1,nCalPar())));
     443           0 :     if (svc().solvePol()) 
     444           0 :       srcPar().reference(svc().srcPolPar());
     445             :   }
     446             :   else
     447           0 :     throw(AipsError("Solver and SVC cannot synchronize parameters."));
     448             : 
     449             :   // Pars
     450             : 
     451           0 :   dpar().resize(nTotalPar());
     452           0 :   dpar()=0.0;
     453             : 
     454           0 :   lastCalPar().resize(nCalPar());
     455           0 :   dCalPar().reference(dpar()(IPosition(1,0),IPosition(1,nCalPar()-1)));
     456             : 
     457           0 :   if (svc().solvePol()) {
     458           0 :     lastSrcPar().resize(nSrcPar());
     459           0 :     dSrcPar().reference(dpar()(IPosition(1,nCalPar()),IPosition(1,nTotalPar()-1)));
     460             :   }
     461             :   else {
     462           0 :     lastSrcPar().resize();
     463           0 :     dSrcPar().resize();
     464             :   }
     465             : 
     466             :   // Gradient and Hessian
     467           0 :   grad().resize(nTotalPar());
     468           0 :   grad()=0.0;
     469             : 
     470           0 :   hess().resize(nTotalPar());
     471           0 :   hess()=0.0;
     472             : 
     473             :   // Levenberg-Marquardt factor
     474           0 :   lambda()=2.0;
     475             : 
     476             :   // Convergence anticipation
     477           0 :   cvrgcount_=0;
     478             : 
     479           0 : }
     480             : 
     481           0 : void VisCalSolver::residualate() {
     482             : 
     483           0 :   if (prtlev()>2) cout << "  VCS::residualate()" << endl;
     484             : 
     485             :   // Delegate to VisEquation
     486             :   //  ve().residuals(svb(),R(),Rflag());
     487             : 
     488             :   // For now, just use ve.diffResid, until we have
     489             :   //  implemented focuschan-aware trial corrupt in SVC
     490             :   //  (this will hurt performance a bit)
     491           0 :   ve().diffResiduals(svb(),R(),dR(),Rflg());
     492             : 
     493           0 : }
     494             : 
     495           0 : void VisCalSolver::residualate2() {
     496             : 
     497           0 :   if (prtlev()>2) cout << "  VCS::residualate()" << endl;
     498             : 
     499             :   // Delegate to VisEquation
     500             :   //  ve().residuals(svb(),R(),Rflag());
     501             : 
     502             :   // For now, just use ve.diffResid, until we have
     503             :   //  implemented focuschan-aware trial corrupt in SVC
     504             :   //  (this will hurt performance a bit)
     505             :   
     506           0 :   for (Int ibuf=0;ibuf<vbga().nBuf();++ibuf) 
     507           0 :     ve().diffResiduals(vbga()(ibuf));
     508           0 : }
     509             : 
     510           0 : void VisCalSolver::differentiate() {
     511             : 
     512           0 :   if (prtlev()>2) cout << "  VCS::differentiate()" << endl;
     513             : 
     514             :   // Delegate to VisEquation
     515           0 :   ve().diffResiduals(svb(),R(),dR(),Rflg());
     516             : 
     517             : 
     518           0 :   if (svc().solvePol()) {
     519             : 
     520             :     // Differentiate w.r.t source
     521           0 :     svc().diffSrc(svb(),dSrc());
     522             :   }
     523             : 
     524           0 :   if (prtlev()>6) {  // R, dR
     525           0 :     cout << "   R= " << R() << endl;
     526           0 :     cout << "   dR=" << dR() << endl;
     527             :   }
     528             : 
     529           0 : }
     530             : 
     531           0 : void VisCalSolver::differentiate2() {
     532             : 
     533           0 :   if (prtlev()>2) cout << "  VCS::differentiate(VBGA version)" << endl;
     534             : 
     535             :   // Delegate to VisEquation
     536           0 :   for (Int ibuf=0;ibuf<vbga().nBuf();++ibuf)
     537           0 :     ve().diffResiduals(vbga()(ibuf));
     538             : 
     539           0 :   if (svc().solvePol()) {
     540             :     //    throw(AipsError("solvePol not yet sorted w.r.t. VBGA."));
     541             :     // Differentiate w.r.t source
     542           0 :     svc().diffSrc(vbga()(0),dSrc());
     543             :   }
     544             : 
     545           0 : }
     546             : 
     547           0 : void VisCalSolver::chiSquare() {
     548             : 
     549           0 :   if (prtlev()>2) cout << "   VCS::chiSquare()" << endl;
     550             : 
     551             :   // NB: Assumes R() is up-to-date
     552             : 
     553             :   //  TBD: Review correctness of summing weights 
     554             :   //     inside the channel loop?
     555             : 
     556             :   // TBD: per-ant/bln chiSq?
     557             : 
     558             :   //  cout << "VCS::chiSquare: svc().focusChan() = " << svc().focusChan() << endl;
     559             : 
     560           0 :   chiSq()=0.0;
     561           0 :   chiSqV()=0.0;
     562           0 :   sumWt()=0.0;
     563           0 :   nWt()=0;
     564             : 
     565             :   // Shapes for iteration
     566           0 :   IPosition shR(R().shape());
     567           0 :   Int nCorr=shR(0);
     568           0 :   Int nChan=shR(1);
     569           0 :   Int nRow=shR(2);
     570             : 
     571           0 :   ArrayIterator<Bool>  flag(svb().flag(),1);        // fl(chan) by (row)
     572           0 :   ArrayIterator<Float> wtMat(svb().weightMat(),1);  // wt(corr) by (row)
     573           0 :   ArrayIterator<Complex> Rit(R(),1);                // R(corr)  by (chan,row)
     574             : 
     575           0 :   Bool*  flR = svb().flagRow().data();
     576             :   Bool*  fl;
     577             :   Float* wt;
     578             :   Complex *Rp;
     579             : 
     580           0 :   for (Int irow=0;irow<nRow;++irow) { 
     581           0 :     if (!*flR) {
     582             :       // This row's wt(corr), flag(chan)
     583           0 :       wt = wtMat.array().data(); 
     584             :       //      wt[1]=wt[2]=0.0;
     585           0 :       fl = flag.array().data() + svc().focusChan();
     586             :       // Register R for this row, 0th channel
     587           0 :       for (Int ich=0;ich<nChan;++ich) {
     588           0 :         if (!*fl) { 
     589             : 
     590           0 :           Rp = Rit.array().data();
     591           0 :           for (Int icorr=0;icorr<nCorr;++icorr) {
     592             : 
     593             :    /*
     594             :             if (svb().weightMat()(icorr,irow)>0.0) {
     595             :               cout << irow << " " << icorr << " "
     596             :                    << svb().weightMat()(icorr,irow) << " "
     597             :                    << *wt << " "
     598             :                    << R()(icorr,0,irow) << " "
     599             :                    << *Rp << " "
     600             :                    << &R()(icorr,0,irow) << " "
     601             :                    << Rp << " "
     602             :                    << Rp-&R()(icorr,0,irow) << "    "
     603             :                    << &R()(0,0,irow)-&R()(3,0,irow) << " "
     604             :                    << Rit.array().shape()
     605             :                    << endl;
     606             :             }
     607             :    */
     608           0 :             chiSq()+=Double( (*wt)*real((*Rp)*conj(*Rp)) );
     609             :             //      chiSq()+=Double( real((*Rp)*conj(*Rp)) );
     610           0 :             chiSqV()(icorr)+=Double( (*wt)*real((*Rp)*conj(*Rp)) );
     611           0 :             sumWt()+=Double(*wt);   // for each channel?!
     612             : 
     613           0 :             if (*wt>0.0) nWt()++;
     614             :             
     615             :             //Advance to next corr
     616           0 :             ++wt;
     617           0 :             ++Rp;
     618             :           }
     619             :           // Use same wt(corr) vectors for each channel
     620           0 :           wt-=nCorr;
     621             :         }
     622             :         // Advance to next channel
     623           0 :         Rit.next();
     624           0 :         ++fl;
     625             :       }
     626             :     }
     627             :     else 
     628             :       // Advance over flagged row!
     629           0 :       for (Int ich=0;ich<nChan;++ich) Rit.next();
     630             :     
     631             :     // Advance to next row
     632           0 :     ++flR;
     633           0 :     flag.next();
     634           0 :     wtMat.next();
     635             :   }
     636             : 
     637             : 
     638           0 : }
     639             : 
     640           0 : void VisCalSolver::chiSquare2() {
     641             : 
     642           0 :   if (prtlev()>2) cout << "   VCS::chiSquare(CVB version)" << endl;
     643             : 
     644             :   // NB: Assumes R() is up-to-date
     645             : 
     646             :   //  TBD: Review correctness of summing weights 
     647             :   //     inside the channel loop?
     648             : 
     649             :   // TBD: per-ant/bln chiSq?
     650             : 
     651             :   //  cout << "VCS::chiSquare: svc().focusChan() = " << svc().focusChan() << endl;
     652             : 
     653           0 :   chiSq()=0.0;
     654           0 :   chiSqV()=0.0;
     655           0 :   sumWt()=0.0;
     656           0 :   nWt()=0;
     657             : 
     658             :   // Loop over CVBs
     659           0 :   for (Int ibuf=0;ibuf<vbga().nBuf();++ibuf) {
     660             : 
     661             :     // Focus on one CVB at a time
     662           0 :     CalVisBuffer& cvb(vbga()(ibuf));
     663             : 
     664           0 :     R().reference(cvb.residuals());
     665             : 
     666             :     // Shapes for iteration
     667           0 :     IPosition shR(R().shape());
     668           0 :     Int nCorr=shR(0);
     669           0 :     Int nChan=shR(1);
     670           0 :     Int nRow=shR(2);
     671             :     
     672           0 :     ArrayIterator<Bool>  flag(cvb.residFlag(),1);        // fl(chan) by (row)
     673           0 :     ArrayIterator<Float> wtMat(cvb.weightMat(),1);  // wt(corr) by (row)
     674           0 :     ArrayIterator<Complex> Rit(R(),1);                // R(corr)  by (chan,row)
     675             :     
     676           0 :     Bool*  flR = cvb.flagRow().data();
     677             :     Bool*  fl;
     678             :     Float* wt;
     679             :     Complex *Rp;
     680             :     
     681           0 :     for (Int irow=0;irow<nRow;++irow) { 
     682           0 :       if (!*flR) {
     683             :         // This row's wt(corr), flag(chan)
     684           0 :         wt = wtMat.array().data(); 
     685           0 :         fl = flag.array().data();
     686             :         // Register R for this row, 0th channel
     687           0 :         for (Int ich=0;ich<nChan;++ich) {
     688           0 :           if (!*fl) { 
     689             :             
     690           0 :             Rp = Rit.array().data();
     691           0 :             for (Int icorr=0;icorr<nCorr;++icorr) {
     692             :               
     693           0 :               chiSq()+=Double( (*wt)*real((*Rp)*conj(*Rp)) );
     694           0 :               chiSqV()(icorr)+=Double( (*wt)*real((*Rp)*conj(*Rp)) );
     695           0 :               sumWt()+=Double(*wt);   // for each channel?!
     696             :               
     697           0 :               if (*wt>0.0) nWt()++;
     698             :               
     699             :               //Advance to next corr
     700           0 :               ++wt;
     701           0 :               ++Rp;
     702             :             }
     703             :             // Use same wt(corr) vectors for each channel
     704           0 :             wt-=nCorr;
     705             :           }
     706             :           // Advance to next channel
     707           0 :           Rit.next();
     708           0 :           ++fl;
     709             :         }
     710             :       }
     711             :       else 
     712             :         // Advance over flagged row!
     713           0 :         for (Int ich=0;ich<nChan;++ich) Rit.next();
     714             :       
     715             :       // Advance to next row
     716           0 :       ++flR;
     717           0 :       flag.next();
     718           0 :       wtMat.next();
     719             :     }
     720           0 :   } // ibuf
     721           0 : }
     722             : 
     723             : 
     724           0 : Bool VisCalSolver::converged() {
     725             : 
     726           0 :   if (prtlev()>2) cout << "    VCS::converged()" << endl;
     727             : 
     728             :   // Change in chi2
     729           0 :   dChiSq() = chiSq()-lastChiSq();
     730           0 :   Float fChiSq(dChiSq()/chiSq());
     731             : 
     732             :   //  if (prtlev()>0)
     733             :   //    cout << "chiSq: " << chiSq() << " " << chiSqV() << " " << lambda() << endl;
     734             : 
     735             :   // Consider convergence if chi2 decreases...
     736             :   //  if (dChiSq()<=0.0) {
     737           0 :   if (fChiSq<=0.001) {
     738             : 
     739             :     // ...and the change is small:
     740           0 :     if (abs(dChiSq()) < 0.1*chiSq()) {
     741           0 :       ++cvrgcount_;
     742             : 
     743             :       //      if (cvrgcount_==2) lambda()=2.0;
     744             : 
     745             :       // Four such steps we believe we have converged!
     746             :       //      if (cvrgcount_>3)
     747             :       //if (cvrgcount_>10)
     748           0 :       if (cvrgcount_>5)
     749           0 :         return true;
     750             :     }
     751             : 
     752           0 :     if (prtlev()>0)
     753           0 :       cout << "     Good: chiSq=" << chiSq()
     754           0 :            << " dChiSq=" << dChiSq()
     755           0 :            << " fChiSq=" << dChiSq()/chiSq()
     756           0 :            << " cvrgcnt=" << cvrgcount_
     757           0 :            << " lambda=" << lambda()
     758           0 :            << endl;
     759             :      
     760             :   }
     761             :   else {
     762             :     // (chi2 increased)
     763             : 
     764             :     // If a large increase, don't anticipate yet
     765           0 :     if (abs(dChiSq()) > 0.1*chiSq())
     766           0 :       cvrgcount_=0;
     767             :     else {
     768             :       // anticipate a little less if upward change is small
     769             :       //  TBD:  is this right?
     770           0 :       --cvrgcount_;
     771           0 :       cvrgcount_=max(cvrgcount_,0);  // never less than zero
     772             :     }
     773             : 
     774           0 :     if (prtlev()>0)
     775           0 :       cout << "     Bad:  chiSq=" << chiSq()
     776           0 :            << " dChiSq=" << dChiSq()
     777           0 :            << " fChiSq=" << dChiSq()/chiSq()
     778           0 :            << " cvrgcnt=" << cvrgcount_
     779           0 :            << " lambda=" << lambda()
     780           0 :            << endl;
     781             : 
     782             : 
     783             :   }
     784             : 
     785             :   // Not yet converged
     786           0 :   return false;
     787             : 
     788             : }
     789             : 
     790             : 
     791           0 : void VisCalSolver::accGradHess() {
     792             : 
     793           0 :   if (prtlev()>2) cout << "     VCS::accGradHess()" << endl;
     794             : 
     795           0 :   grad()=0.0;
     796           0 :   hess()=0.0;
     797             : 
     798           0 :   IPosition dRip(dR().shape());
     799             : 
     800           0 :   Int nRow(dRip(3));
     801           0 :   Int nChan(dRip(2));
     802           0 :   Int nPar(dRip(1));   // pars per antenna
     803           0 :   Int nCorr(dRip(0));
     804             : 
     805           0 :   ArrayIterator<Bool>  flag(svb().flag(),1);        // fl(chan) by (row)
     806           0 :   ArrayIterator<Float> wtMat(svb().weightMat(),1);  // wt(corr) by (row)
     807             : 
     808           0 :   ArrayIterator<Complex> Rit(R(),1);       // R(corr)       by (chan,row)
     809             :   
     810           0 :   Array<Complex> dR0, dR1;
     811             :   {
     812           0 :     ArrayIterator<Complex> dRit(dR(),4);  // dR(corr,par,chan,row) by (ant)
     813           0 :     dR0.reference(dRit.array());
     814           0 :     dRit.next();
     815           0 :     dR1.reference(dRit.array());
     816           0 :   }
     817           0 :   ArrayIterator<Complex> dR0it(dR0,2);   // dR0(corr,par) by (chan,row)
     818           0 :   ArrayIterator<Complex> dR1it(dR1,2);   // dR1(corr,par) by (chan,row)
     819             : 
     820           0 :   Bool*  flR = svb().flagRow().data();
     821             :   Bool*  fl;
     822             :   Float* wt;
     823           0 :   Int*   a1 = svb().antenna1().data();
     824           0 :   Int*   a2 = svb().antenna2().data();
     825             :   Complex *Rp;
     826             :   Complex *dR0p;
     827             :   Complex *dR1p;
     828             :   DComplex *G1;
     829             :   DComplex *G2;
     830             :   Double *H1;
     831             :   Double *H2;
     832             : 
     833             :   //  cout << "flag.array().data() = " << flag.array().data() << endl;
     834             :   //  cout << "VCS::accGradHess:  svc().focusChan()   = " << svc().focusChan() << endl;
     835             : 
     836           0 :   for (Int irow=0;irow<nRow;++irow) { 
     837           0 :     if (!*flR) {
     838             :       // Register grad, hess for ants in this baseline
     839           0 :       Int p1((*a1)*nPar);
     840           0 :       Int p2((*a2)*nPar);
     841           0 :       G1 = grad().data() + p1;
     842           0 :       G2 = grad().data() + p2;
     843           0 :       H1 = hess().data() + p1;
     844           0 :       H2 = hess().data() + p2;
     845             :       // This row's wt(corr), flag(chan)
     846           0 :       wt = wtMat.array().data(); 
     847           0 :       fl = flag.array().data() + svc().focusChan();
     848           0 :       for (Int ich=0;ich<nChan;++ich) {
     849           0 :         if (!*fl) { 
     850             : 
     851             :           // Do source bits, if necessary
     852           0 :           if (svc().solvePol()) {
     853           0 :             for (Int icorr=0;icorr<nCorr;++icorr) {
     854           0 :               Float swt=Vector<Float>(wtMat.array())(icorr);
     855             : 
     856           0 :               for (Int ispar=0;ispar<nSrcPar();++ispar) {
     857           0 :                 grad()(nCalPar()+ispar) += DComplex((swt)*2.0*real(dSrc()(IPosition(4,icorr,ich,irow,ispar))*
     858           0 :                                                                    conj(R()(icorr,ich,irow))));
     859           0 :                 hess()(nCalPar()+ispar) += Double((swt)*2.0*real(dSrc()(IPosition(4,icorr,ich,irow,ispar))*
     860           0 :                                                                  conj(dSrc()(IPosition(4,icorr,ich,irow,ispar)))));
     861             :               }
     862             :             }
     863             :           }
     864             : 
     865             :           // Register R,dR for this channel, row
     866           0 :           Rp = Rit.array().data();
     867           0 :           dR0p = dR0it.array().data();
     868           0 :           dR1p = dR1it.array().data();
     869             : 
     870             :    /*
     871             :           cout << "irow=" << irow << endl;
     872             :           cout << "R=" << Rit.array() << endl;
     873             :           cout << "dR0=" << dR0it.array()
     874             :                << "dR1=" << dR1it.array()
     875             :                << endl;
     876             :    */
     877             : 
     878           0 :           for (int ip=0;ip<nPar;++ip) {
     879             : 
     880           0 :             for (Int icorr=0;icorr<nCorr;++icorr) {
     881             : 
     882           0 :               (*G1) += DComplex( (*wt)*((*Rp)  *conj(*dR0p)) );
     883           0 :               (*G2) += DComplex( (*wt)*((*dR1p)*conj(*Rp)) );
     884           0 :               (*H1) +=   Double( (*wt)*real((*dR0p)*conj(*dR0p)) );
     885           0 :               (*H2) +=   Double( (*wt)*real((*dR1p)*conj(*dR1p)) );
     886             :      /*
     887             :               (*G1) += DComplex( (*Rp)  *conj(*dR0p) );
     888             :               (*G2) += DComplex( (*dR1p)*conj(*Rp) );
     889             :               (*H1) +=   Double( real((*dR0p)*conj(*dR0p)) );
     890             :               (*H2) +=   Double( real((*dR1p)*conj(*dR1p)) );
     891             :      */
     892             :     
     893             : 
     894             :               //Advance to next corr
     895           0 :               ++wt;
     896           0 :               ++Rp;
     897           0 :               ++dR0p;++dR1p;
     898             :             }
     899             :             // Advance to next par
     900           0 :             ++G1; ++G2;
     901           0 :             ++H1; ++H2;
     902             :             // Use same Rp(corr), wt(corr) vectors for each parameter
     903           0 :             Rp-=nCorr;
     904           0 :             wt-=nCorr;
     905             :           }
     906             :           // Accumulate to same grad(par) & hess(par) for each channel
     907           0 :           G1-=nPar; G2-=nPar;
     908           0 :           H1-=nPar; H2-=nPar;
     909             :         }
     910             :         // Advance to next channel
     911           0 :         ++fl;
     912           0 :         Rit.next();
     913           0 :         dR0it.next();
     914           0 :         dR1it.next();
     915             :       }
     916             :     } // !*flgR
     917             :     else {
     918             :       // Advance over flagged row
     919           0 :       for (Int ich=0;ich<nChan;++ich) {
     920           0 :         Rit.next();
     921           0 :         dR0it.next();
     922           0 :         dR1it.next();
     923             :       }
     924             :     }
     925             :     // Advance to next row
     926           0 :     ++flR;
     927           0 :     ++a1;++a2;
     928           0 :     flag.next();
     929           0 :     wtMat.next();
     930             :   }
     931             : 
     932           0 :   if (prtlev()>4) {  // grad, hess
     933           0 :     cout << "      grad= " << grad() << endl;
     934           0 :     cout << "      hess= " << hess() << endl;
     935             :   }
     936             : 
     937             : 
     938             : /*  
     939             :   // This is the slower way with array indexing
     940             :   for (Int irow=0;irow<nRow;++irow,++flR,++a1,++a2) {
     941             :     if (!*flR) {
     942             :       par1=(*a1)*nPar;
     943             :       par2=(*a2)*nPar;
     944             :       for (Int ich=0;ich<nChan;++ich,++fl) {
     945             :         if (!*fl) {
     946             :           for (int ip=0;ip<nPar;++ip) {
     947             :             for (Int icorr=0;icorr<nCorr;++icorr) {
     948             :               grad(par1+ip) += ( Double(wt(icorr,irow))*
     949             :                                  Double( R()(icorr,ich,irow  )*
     950             :                                          conj(dR()(icorr,ip,ich,irow,0)) ) );
     951             :               grad(par2+ip) += ( Double(wt(icorr,irow))*
     952             :                                  Double( dR()(icorr,ip,ich,irow,1)*
     953             :                                          conj(R()(icorr,ich,irow  )) ) );
     954             :               
     955             :               hess(par1+ip) += ( Double(wt(icorr,irow))*
     956             :                                  Double( dR()(icorr,ip,ich,irow,0)*
     957             :                                          conj(dR()(icorr,ip,ich,irow,0)) ) );
     958             :               hess(par2+ip) += ( Double(wt(icorr,irow))*
     959             :                                  Double( dR()(icorr,ip,ich,irow,1)*
     960             :                                          conj(dR()(icorr,ip,ich,irow,1)) ) );
     961             :             }
     962             :           }
     963             :         }
     964             :       }
     965             :     }
     966             :     else {
     967             :       // Advance over flagged row
     968             :       fl+=nChan;
     969             :     }
     970             : 
     971             :   }
     972             : */
     973             : 
     974             : 
     975           0 : }
     976             : 
     977             : 
     978           0 : void VisCalSolver::accGradHess2() {
     979             : 
     980           0 :   if (prtlev()>2) cout << "     VCS::accGradHess(CVB version)" << endl;
     981             : 
     982           0 :   grad()=0.0;
     983           0 :   hess()=0.0;
     984             : 
     985           0 :   for (Int ibuf=0;ibuf<vbga().nBuf();++ibuf) {
     986             : 
     987           0 :     CalVisBuffer& cvb(vbga()(ibuf));
     988             : 
     989           0 :     R().reference(cvb.residuals());
     990           0 :     dR().reference(cvb.diffResiduals());
     991             : 
     992             :     
     993           0 :     IPosition dRip(dR().shape());
     994             :     
     995           0 :     Int nRow(dRip(3));
     996           0 :     Int nChan(dRip(2));
     997           0 :     Int nPar(dRip(1));   // pars per antenna
     998           0 :     Int nCorr(dRip(0));
     999             :     
    1000           0 :     ArrayIterator<Bool>  flag(cvb.residFlag(),1);        // fl(chan) by (row)
    1001           0 :     ArrayIterator<Float> wtMat(cvb.weightMat(),1);  // wt(corr) by (row)
    1002             :     
    1003           0 :     ArrayIterator<Complex> Rit(R(),1);       // R(corr)       by (chan,row)
    1004             :     
    1005           0 :     Array<Complex> dR0, dR1;
    1006             :     {
    1007           0 :       ArrayIterator<Complex> dRit(dR(),4);  // dR(corr,par,chan,row) by (ant)
    1008           0 :       dR0.reference(dRit.array());
    1009           0 :       dRit.next();
    1010           0 :       dR1.reference(dRit.array());
    1011           0 :     }
    1012           0 :     ArrayIterator<Complex> dR0it(dR0,2);   // dR0(corr,par) by (chan,row)
    1013           0 :     ArrayIterator<Complex> dR1it(dR1,2);   // dR1(corr,par) by (chan,row)
    1014             :     
    1015           0 :     Bool*  flR = cvb.flagRow().data();
    1016             :     Bool*  fl;
    1017             :     Float* wt;
    1018           0 :     Int*   a1 = cvb.antenna1().data();
    1019           0 :     Int*   a2 = cvb.antenna2().data();
    1020             :     Complex *Rp;
    1021             :     Complex *dR0p;
    1022             :     Complex *dR1p;
    1023             :     DComplex *G1;
    1024             :     DComplex *G2;
    1025             :     Double *H1;
    1026             :     Double *H2;
    1027             : 
    1028           0 :     for (Int irow=0;irow<nRow;++irow) { 
    1029           0 :       if (!*flR) {
    1030             :         // Register grad, hess for ants in this baseline
    1031           0 :         Int p1((*a1)*nPar);
    1032           0 :         Int p2((*a2)*nPar);
    1033           0 :         G1 = grad().data() + p1;
    1034           0 :         G2 = grad().data() + p2;
    1035           0 :         H1 = hess().data() + p1;
    1036           0 :         H2 = hess().data() + p2;
    1037             :         // This row's wt(corr), flag(chan)
    1038           0 :         wt = wtMat.array().data(); 
    1039           0 :         fl = flag.array().data();
    1040           0 :         for (Int ich=0;ich<nChan;++ich) {
    1041           0 :           if (!*fl) { 
    1042             :             
    1043             :             // Do source bits, if necessary
    1044           0 :             if (svc().solvePol()) {
    1045           0 :               for (Int icorr=0;icorr<nCorr;++icorr) {
    1046           0 :                 Float swt=Vector<Float>(wtMat.array())(icorr);
    1047             :                 
    1048           0 :                 for (Int ispar=0;ispar<nSrcPar();++ispar) {
    1049           0 :                   grad()(nCalPar()+ispar) += DComplex((swt)*2.0*real(dSrc()(IPosition(4,icorr,ich,irow,ispar))*
    1050           0 :                                                                      conj(R()(icorr,ich,irow))));
    1051           0 :                   hess()(nCalPar()+ispar) += Double((swt)*2.0*real(dSrc()(IPosition(4,icorr,ich,irow,ispar))*
    1052           0 :                                                                    conj(dSrc()(IPosition(4,icorr,ich,irow,ispar)))));
    1053             :                 }
    1054             :               }
    1055             :             }
    1056             :             
    1057             :             // Register R,dR for this channel, row
    1058           0 :             Rp = Rit.array().data();
    1059           0 :             dR0p = dR0it.array().data();
    1060           0 :             dR1p = dR1it.array().data();
    1061             :             
    1062           0 :             for (int ip=0;ip<nPar;++ip) {
    1063             :               
    1064           0 :               for (Int icorr=0;icorr<nCorr;++icorr) {
    1065             :                 
    1066           0 :                 (*G1) += DComplex( (*wt)*((*Rp)  *conj(*dR0p)) );
    1067           0 :                 (*G2) += DComplex( (*wt)*((*dR1p)*conj(*Rp)) );
    1068           0 :                 (*H1) +=   Double( (*wt)*real((*dR0p)*conj(*dR0p)) );
    1069           0 :                 (*H2) +=   Double( (*wt)*real((*dR1p)*conj(*dR1p)) );
    1070             :                 
    1071             :                 //Advance to next corr
    1072           0 :                 ++wt;
    1073           0 :                 ++Rp;
    1074           0 :                 ++dR0p;++dR1p;
    1075             :               }
    1076             :               // Advance to next par
    1077           0 :               ++G1; ++G2;
    1078           0 :               ++H1; ++H2;
    1079             :               // Use same Rp(corr), wt(corr) vectors for each parameter
    1080           0 :               Rp-=nCorr;
    1081           0 :               wt-=nCorr;
    1082             :             }
    1083             :             // Accumulate to same grad(par) & hess(par) for each channel
    1084           0 :             G1-=nPar; G2-=nPar;
    1085           0 :             H1-=nPar; H2-=nPar;
    1086             :           }
    1087             :           // Advance to next channel
    1088           0 :           ++fl;
    1089           0 :           Rit.next();
    1090           0 :           dR0it.next();
    1091           0 :           dR1it.next();
    1092             :         }
    1093             :       } // !*flgR
    1094             :       else {
    1095             :         // Advance over flagged row
    1096           0 :         for (Int ich=0;ich<nChan;++ich) {
    1097           0 :           Rit.next();
    1098           0 :           dR0it.next();
    1099           0 :           dR1it.next();
    1100             :         }
    1101             :       }
    1102             :       // Advance to next row
    1103           0 :       ++flR;
    1104           0 :       ++a1;++a2;
    1105           0 :       flag.next();
    1106           0 :       wtMat.next();
    1107             :     }
    1108             :     
    1109           0 :     if (prtlev()>4) {  // grad, hess
    1110           0 :       cout << "      grad= " << grad() << endl;
    1111           0 :       cout << "      hess= " << hess() << endl;
    1112             :     }    
    1113             : 
    1114           0 :   } // ibuf
    1115           0 : }
    1116             : 
    1117           0 : void VisCalSolver::revert() {
    1118             : 
    1119           0 :   if (prtlev()>2) cout << "     VCS::revert()" << endl;
    1120             : 
    1121             :   // Recall the last decent parameter set
    1122             :   //  TBD: the OK flag?
    1123           0 :   par()=lastCalPar();
    1124           0 :   if (svc().solvePol())
    1125           0 :     srcPar()=lastSrcPar();
    1126           0 : }
    1127             : 
    1128           0 : void VisCalSolver::solveGradHess() {
    1129             : 
    1130           0 :   if (prtlev()>2) cout << "      VCS::solveGradHess()" << endl;
    1131             : 
    1132             :   // TBD: explicit option to avoid lmfact?
    1133             :   // TBD: pointer (or MaskedArray?) optimization?
    1134             : 
    1135           0 :   Double lmfact(1.0+lambda());
    1136             : 
    1137           0 :   lmfact=2.0;
    1138             : 
    1139           0 :   dpar()=Complex(0.0);
    1140           0 :   for (Int ipar=0; ipar<nCalPar(); ipar++) {
    1141           0 :     if ( parOK()(ipar) && hess()(ipar)!=0.0) {
    1142             :       // good hess for this par:
    1143           0 :       dpar()(ipar) = grad()(ipar)/hess()(ipar);
    1144           0 :       dpar()(ipar)/=lmfact;
    1145             :     }
    1146             :     else {
    1147           0 :       dpar()(ipar)=0.0; 
    1148           0 :       parOK()(ipar)=false;
    1149             :     }
    1150             :   }
    1151             :   
    1152             :   // Turn off source pol update for the moment
    1153           0 :   if (svc().solvePol() && false) 
    1154           0 :     for (Int ipar=nCalPar(); ipar<nTotalPar(); ipar++) {
    1155           0 :       if ( hess()(ipar)!=0.0) {
    1156             :         // good hess for this par:
    1157           0 :         dpar()(ipar) = grad()(ipar)/hess()(ipar);
    1158           0 :         dpar()(ipar)/=lmfact;
    1159             :       }
    1160             :       else {
    1161           0 :         dpar()(ipar)=0.0; 
    1162             :       }
    1163             :     }
    1164             : 
    1165             :   // Negate (so updatePar() can _add_)
    1166           0 :   dpar()*=Complex(-1.0f);
    1167             : 
    1168           0 : }
    1169             : 
    1170           0 : void VisCalSolver::updatePar() {
    1171             : 
    1172           0 :   if (prtlev()>2) cout << "       VCS::updatePar()" << endl;
    1173             : 
    1174             :   //  if (prtlev()>4) cout << "        old =" << par() << endl;
    1175             : 
    1176             :   //  if (prtlev()>4) cout << "        dpar=" << dpar() << endl;
    1177             : 
    1178             : 
    1179             :   //  cout << "dCalPar() = " << dCalPar() << endl;
    1180             :   //  cout << "dSrcPar() = " << dSrcPar() << endl;
    1181             : 
    1182             :   // Tell svc to update the par 
    1183             :   //   (permits svc() to condition the current solutions)
    1184           0 :   svc().updatePar(dCalPar(),dSrcPar());
    1185             : 
    1186           0 :   if (prtlev()>4) 
    1187           0 :     cout << "        new =" << endl
    1188           0 :          << "amp = " << amplitude(par()) << endl
    1189           0 :          << "pha = " << phase(par()) << endl;
    1190             : 
    1191           0 : }
    1192             : 
    1193             : 
    1194           0 : void VisCalSolver::optStepSize() {
    1195             : 
    1196           0 :   Vector<Double> x2(3,-999.0);
    1197           0 :   Float step(1.0);
    1198             :   
    1199             :   // Starting point is curr chiSq
    1200           0 :   x2(0)=chiSq();
    1201             : 
    1202             :   // take nominal step
    1203           0 :   par()+=dCalPar();  
    1204           0 :   if (svc().solvePol()) srcPar()+=dSrcPar();  
    1205           0 :   residualate();
    1206           0 :   chiSquare();
    1207           0 :   x2(1)=chiSq();
    1208             : 
    1209             :   // If nominal step is an improvement...
    1210           0 :   if (x2(1)<x2(0)) {
    1211             : 
    1212             :     // ...double step size until x2 starts increasing
    1213           0 :     par()=dCalPar(); par()*=Complex(2.0*step); par()+=lastCalPar();
    1214           0 :     if (svc().solvePol()) { srcPar()=dSrcPar(); srcPar()*=Complex(2.0*step); srcPar()+=lastSrcPar(); };
    1215           0 :     residualate();
    1216           0 :     chiSquare();
    1217           0 :     x2(2)=chiSq();
    1218             :     //    cout <<   "  down:    " << step << " " << x2-x2(0) << LogicalArray(x2>=x2(0)) <<endl;
    1219           0 :     while (x2(2)<x2(1)) {    //  && step<4.0) {
    1220           0 :       step*=2.0;
    1221           0 :       par()=dCalPar(); par()*=Complex(2.0*step); par()+=lastCalPar();
    1222           0 :       if (svc().solvePol()) { srcPar()=dSrcPar(); srcPar()*=Complex(2.0*step); srcPar()+=lastSrcPar(); };
    1223           0 :       residualate();
    1224           0 :       chiSquare();
    1225           0 :       x2(1)=x2(2);
    1226           0 :       x2(2)=chiSq();
    1227             :       //      cout << "  stretch: " << step << " " << x2-x2(0) << LogicalArray(x2>=x2(0)) <<endl;
    1228             : 
    1229             :     }
    1230             :   }
    1231             :   // else nominal step too big, so...
    1232             :   else {
    1233             : 
    1234             :     // ... contract by halves until we bracket a minimum
    1235           0 :     step*=0.5;
    1236           0 :     par()=dCalPar(); par()*=Complex(step); par()+=lastCalPar();
    1237           0 :     if (svc().solvePol()) {srcPar()=dSrcPar(); srcPar()*=Complex(step); srcPar()+=lastSrcPar();};
    1238           0 :     residualate();
    1239           0 :     chiSquare();
    1240           0 :     x2(2)=x2(1);
    1241           0 :     x2(1)=chiSq();
    1242             :     //    cout <<   "  up:       " << step << " " << x2-x2(0) << LogicalArray(x2>=x2(0)) <<endl;
    1243           0 :     while (x2(1)>x2(0)) { //  && step>0.125) {
    1244           0 :       step*=0.5;
    1245           0 :       par()=dCalPar(); par()*=Complex(step); par()+=lastCalPar();
    1246           0 :       if (svc().solvePol()) {srcPar()=dSrcPar(); srcPar()*=Complex(step); srcPar()+=lastSrcPar();};
    1247           0 :       residualate();
    1248           0 :       chiSquare();
    1249           0 :       x2(2)=x2(1);
    1250           0 :       x2(1)=chiSq();
    1251             :       //      cout << "  contract: " << step << " " << x2-x2(0) << LogicalArray(x2>=x2(0)) <<endl;
    1252             :     }
    1253             : 
    1254             :   }
    1255             : 
    1256             :   // At this point   x2(0) > x2(1) < x2(2), so 
    1257             :   //   calculate (quadratic) step optimization factor
    1258           0 :   Double optfactor(0.0);
    1259           0 :   Double optn(x2(2)-x2(1));
    1260           0 :   Double optd(x2(0)-2*x2(1)+x2(2));
    1261             :               
    1262           0 :   if (abs(optd)>0.0)
    1263           0 :     optfactor=Double(step)*(1.5-optn/optd);
    1264             :   
    1265             :   /*  
    1266             :     cout << "Optimization: " 
    1267             :        << step << " " 
    1268             :        << optfactor << " "
    1269             :        << x2 << " "
    1270             :        << "(" << min(amplitude(lastPar())) << ") "
    1271             :        << max(amplitude(dpar())/amplitude(lastPar()))*180.0/C::pi << " ";
    1272             :   */
    1273             : 
    1274           0 :   par()=lastCalPar();
    1275           0 :   srcPar()=lastSrcPar();
    1276             :   
    1277             :   // Adjust step by the optfactor
    1278           0 :   if (optfactor>0.0)
    1279           0 :     dpar()*=Complex(optfactor);
    1280             : 
    1281             :   /*
    1282             :   cout << max(amplitude(dpar())/amplitude(lastPar()))*180.0/C::pi
    1283             :        << endl;
    1284             :   */
    1285           0 : }
    1286             : 
    1287           0 : void VisCalSolver::optStepSize2() {
    1288             : 
    1289           0 :   Vector<Double> x2(3,-999.0);
    1290           0 :   Float step(1.0);
    1291             :   
    1292             :   // Starting point is curr chiSq
    1293           0 :   x2(0)=chiSq();
    1294             : 
    1295             :   // take nominal step
    1296           0 :   par()+=dCalPar();  
    1297           0 :   if (svc().solvePol()) srcPar()+=dSrcPar();  
    1298           0 :   residualate2();
    1299           0 :   chiSquare2();
    1300           0 :   x2(1)=chiSq();
    1301             : 
    1302             :   // If nominal step is an improvement...
    1303           0 :   if (x2(1)<x2(0)) {
    1304             : 
    1305             :     // ...double step size until x2 starts increasing
    1306           0 :     par()=dCalPar(); par()*=Complex(2.0*step); par()+=lastCalPar();
    1307           0 :     if (svc().solvePol()) { srcPar()=dSrcPar(); srcPar()*=Complex(2.0*step); srcPar()+=lastSrcPar(); };
    1308           0 :     residualate2();
    1309           0 :     chiSquare2();
    1310           0 :     x2(2)=chiSq();
    1311             :     //    cout <<   "  down:    " << step << " " << x2-x2(0) << LogicalArray(x2>=x2(0)) <<endl;
    1312           0 :     while (x2(2)<x2(1)) {    //  && step<4.0) {
    1313           0 :       step*=2.0;
    1314           0 :       par()=dCalPar(); par()*=Complex(2.0*step); par()+=lastCalPar();
    1315           0 :       if (svc().solvePol()) { srcPar()=dSrcPar(); srcPar()*=Complex(2.0*step); srcPar()+=lastSrcPar(); };
    1316           0 :       residualate2();
    1317           0 :       chiSquare2();
    1318           0 :       x2(1)=x2(2);
    1319           0 :       x2(2)=chiSq();
    1320             :       //      cout << "  stretch: " << step << " " << x2-x2(0) << LogicalArray(x2>=x2(0)) <<endl;
    1321             : 
    1322             :     }
    1323             :   }
    1324             :   // else nominal step too big, so...
    1325             :   else {
    1326             : 
    1327             :     // ... contract by halves until we bracket a minimum
    1328           0 :     step*=0.5;
    1329           0 :     par()=dCalPar(); par()*=Complex(step); par()+=lastCalPar();
    1330           0 :     if (svc().solvePol()) {srcPar()=dSrcPar(); srcPar()*=Complex(step); srcPar()+=lastSrcPar();};
    1331           0 :     residualate2();
    1332           0 :     chiSquare2();
    1333           0 :     x2(2)=x2(1);
    1334           0 :     x2(1)=chiSq();
    1335             :     //    cout <<   "  up:       " << step << " " << x2-x2(0) << LogicalArray(x2>=x2(0)) <<endl;
    1336           0 :     while (x2(1)>x2(0)) { //  && step>0.125) {
    1337           0 :       step*=0.5;
    1338           0 :       par()=dCalPar(); par()*=Complex(step); par()+=lastCalPar();
    1339           0 :       if (svc().solvePol()) {srcPar()=dSrcPar(); srcPar()*=Complex(step); srcPar()+=lastSrcPar();};
    1340           0 :       residualate2();
    1341           0 :       chiSquare2();
    1342           0 :       x2(2)=x2(1);
    1343           0 :       x2(1)=chiSq();
    1344             :       //      cout << "  contract: " << step << " " << x2-x2(0) << LogicalArray(x2>=x2(0)) <<endl;
    1345             :     }
    1346             : 
    1347             :   }
    1348             : 
    1349             :   // At this point   x2(0) > x2(1) < x2(2), so 
    1350             :   //   calculate (quadratic) step optimization factor
    1351           0 :   Double optfactor(0.0);
    1352           0 :   Double optn(x2(2)-x2(1));
    1353           0 :   Double optd(x2(0)-2*x2(1)+x2(2));
    1354             :               
    1355           0 :   if (abs(optd)>0.0)
    1356           0 :     optfactor=Double(step)*(1.5-optn/optd);
    1357             :   
    1358             :   /*  
    1359             :     cout << "Optimization: " 
    1360             :        << step << " " 
    1361             :        << optfactor << " "
    1362             :        << x2 << " "
    1363             :        << "(" << min(amplitude(lastPar())) << ") "
    1364             :        << max(amplitude(dpar())/amplitude(lastPar()))*180.0/C::pi << " ";
    1365             :   */
    1366             : 
    1367           0 :   par()=lastCalPar();
    1368           0 :   srcPar()=lastSrcPar();
    1369             :   
    1370             :   // Adjust step by the optfactor
    1371           0 :   if (optfactor>0.0)
    1372           0 :     dpar()*=Complex(optfactor);
    1373             : 
    1374             :   /*
    1375             :   cout << max(amplitude(dpar())/amplitude(lastPar()))*180.0/C::pi
    1376             :        << endl;
    1377             :   */
    1378           0 : }
    1379             : 
    1380           0 : void VisCalSolver::getErrors() {
    1381             : 
    1382             :   // Number of *REAL* dof
    1383             :   //  Int nDOF=2*(nWt()-ntrue(parOK()));  // !!!! this is zero for 3 antennas!
    1384           0 :   Int nDOF=max(2*(nWt()-ntrue(parOK())), 1u);
    1385             : 
    1386           0 :   Double k2=chiSq()/Double(nDOF);
    1387             : 
    1388           0 :   parErr()=0.0;
    1389             :   //  Vector<Double> snr(nTotalPar(),0.0);
    1390           0 :   for (Int i=0;i<nCalPar();++i) 
    1391           0 :     if (hess()(i)>0.0) {
    1392           0 :       parErr()(i)=1.0/sqrt(hess()(i)/k2/2.0);   // 2 is from def of Hess!
    1393             :       //      snr(i)=Double(abs(par()(i)))/errs(i);
    1394             :     }
    1395             : 
    1396             : 
    1397             :   if (false) {
    1398             : 
    1399             :     cout << "ChiSq  = " << chiSq() << endl;
    1400             :     cout << "ChiSqV = " << chiSqV() << endl;
    1401             :     cout << "sumWt  = " << sumWt() << endl;
    1402             :     cout << "nWt    = " << nWt()
    1403             :          << "; nTotalPar() = " << nTotalPar() 
    1404             :          << "; nParOK = " << ntrue(parOK())
    1405             :          << "; nDOF = " << nDOF 
    1406             :          << endl;
    1407             :     
    1408             :     cout << "rChiSq = " << k2 << endl;
    1409             :     cout << "max(dpar) = " << max(amplitude(dpar())) << endl;
    1410             :     cout << "Amplitudes = " << amplitude(par()) << endl;
    1411             :     cout << "Errors = " << mean(parErr()(parOK())) << endl;
    1412             :     
    1413             :   }
    1414           0 : }
    1415             : 
    1416             : 
    1417             : } //# NAMESPACE CASA - END
    1418             : 

Generated by: LCOV version 1.16