Line data Source code
1 : //# CTTimeInterp1.cc: Implementation of CTTimeInterp1.h
2 : //# Copyright (C) 2012
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 adressed 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 :
28 : #include <synthesis/CalTables/CTTimeInterp1.h>
29 : #include <synthesis/CalTables/CTMainColumns.h>
30 : #include <synthesis/CalTables/RIorAParray.h>
31 : #include <casacore/casa/aips.h>
32 : #include <casacore/casa/BasicSL/Constants.h>
33 : #include <casacore/casa/Arrays/Array.h>
34 : #include <casacore/casa/IO/ArrayIO.h>
35 : #include <casacore/casa/Arrays/ArrayMath.h>
36 : #include <casacore/casa/Logging/LogMessage.h>
37 : #include <casacore/casa/Logging/LogSink.h>
38 : #include <casacore/scimath/Functionals/Interpolate1D.h>
39 : #include <casacore/scimath/Functionals/ScalarSampledFunctional.h>
40 : #include <casacore/scimath/Functionals/ArraySampledFunctional.h>
41 :
42 : #define CTTIMEINTERPVERB1 false
43 :
44 : using namespace casacore;
45 : namespace casa { //# NAMESPACE CASA - BEGIN
46 :
47 : // Null ctor
48 : //CTTimeInterp1::CTTimeInterp1() {};
49 :
50 : // From NewCalTable
51 0 : CTTimeInterp1::CTTimeInterp1(NewCalTable& ct,
52 : const String& timetype,
53 : Array<Float>& result,
54 0 : Array<Bool>& rflag) :
55 0 : ct_(ct),
56 0 : mcols_p(NULL),
57 0 : timeType_(timetype),
58 0 : currTime_(-999.0),
59 0 : currIdx_(-1),
60 0 : lastWasExact_(false),
61 0 : timeRef_(0.0),
62 0 : timelist_(),
63 0 : domain_(2,0.0),
64 0 : flaglist_(),
65 0 : tInterpolator_p(NULL),
66 0 : cfreq_(-1.0),
67 0 : cycles_(),
68 0 : result_(),
69 0 : rflag_()
70 : {
71 :
72 : if (CTTIMEINTERPVERB1) cout << "CTTimeInterp1::ctor()" << endl;
73 :
74 : // Access to main columns (move to NewCalTable?)
75 0 : mcols_p = new ROCTMainColumns(ct_);
76 :
77 : // Flags
78 0 : mcols_p->flag().getColumn(flaglist_);
79 :
80 : // Record referenced timelist_
81 : // TBD: handle flagged times
82 0 : Vector<Double> dtime;
83 0 : mcols_p->time().getColumn(dtime);
84 0 : domain_(0)=min(dtime);
85 0 : domain_(1)=max(dtime);
86 :
87 : // NB: time is rendered as Vector<Float> for processing
88 : // because Interpolate1D doesn't work if it is V<Double>...
89 : //timeRef_=86400.0*floor(dtime(0)/86400.0);
90 : // 2016Aug22 (gmoellen): use ~current time (not prior midnight)
91 : // as timeRef_ to ~minimize loss of precision for faster sampling
92 0 : timeRef_=floor(dtime(0)-1.0f);
93 0 : dtime-=timeRef_; // Relative to reference time
94 0 : timelist_.resize(dtime.shape());
95 0 : convertArray(timelist_,dtime); // store referenced times as Floats
96 :
97 : // Create the _time_ interpolator
98 : // TBD: f(par) (because flags may be different for each par...)
99 0 : tInterpolator_p = new Interpolate1D<Float,Array<Float> >();
100 0 : setInterpType(timeType_);
101 :
102 : // Get fiducial frequency
103 :
104 : // Get cycles, if nec.
105 0 : if (timetype.contains("PD")) {
106 0 : Int spw=mcols_p->spwId()(0);
107 0 : MSSpWindowColumns spwcol(ct_.spectralWindow());
108 0 : Int nChan=spwcol.numChan()(spw);
109 0 : cfreq_=Vector<Double>(spwcol.chanFreq()(spw))(nChan/2);
110 : //cout << "cfreq_ = " << cfreq_ << endl;
111 0 : mcols_p->cycles(cycles_);
112 : //cout << "ant = " << mcols_p->antenna1()(0) << ": cycles_ = " << cycles_ << endl;
113 0 : }
114 :
115 : // Reference supplied arrays for results
116 : // Elsewhere, must always use strict (non-shape-changing) assignment for these!
117 : // TBD: verify shapes..
118 0 : result_.reference(result);
119 0 : rflag_.reference(rflag);
120 :
121 0 : }
122 :
123 : // Destructor
124 0 : CTTimeInterp1::~CTTimeInterp1() {
125 0 : if (tInterpolator_p)
126 0 : delete tInterpolator_p;
127 0 : if (mcols_p)
128 0 : delete mcols_p;
129 0 : };
130 :
131 0 : Bool CTTimeInterp1::interpolate(Double newtime) {
132 :
133 : if (CTTIMEINTERPVERB1) {cout.precision(12);cout << "CTTimeInterp1::interpolate("<<newtime<<"):" << endl;}
134 :
135 : // Don't work unnecessarily
136 0 : if (newtime==currTime_)
137 0 : return false; // no change
138 :
139 : if (CTTIMEINTERPVERB1) cout << " newtime is new..." << endl;
140 :
141 : // A new time is specified, so some work may be required
142 :
143 : // Convert supplied time value to Float (referenced to timeRef_)
144 0 : Float fnewtime(newtime-timeRef_);
145 :
146 : // Establish registration in time
147 0 : Bool exact(false);
148 0 : Int newIdx(currIdx_);
149 0 : Bool newReg=findTimeRegistration(newIdx,exact,fnewtime);
150 :
151 : if (CTTIMEINTERPVERB1)
152 : cout <<boolalpha<< " newReg="<<newReg<< " newIdx="<<newIdx<< " exact="<<exact
153 : << " lastWasExact_=" << lastWasExact_ << endl;
154 :
155 : // Update interpolator coeffs if new registr. and not nearest
156 0 : if (newReg || (!exact && lastWasExact_)) {
157 : // Only bother if not 'nearest' nor exact...
158 0 : if (!timeType().contains("nearest") && !exact) {
159 0 : if (timeType().contains("linear")) {
160 0 : ScalarSampledFunctional<Float> xf(timelist_(Slice(newIdx,2)));
161 0 : Vector<uInt> rows(2); indgen(rows); rows+=uInt(newIdx);
162 0 : Array<Float> ya(mcols_p->fparamArray("",rows));
163 0 : ArraySampledFunctional<Array<Float> > yf(ya);
164 0 : tInterpolator_p->setData(xf,yf,true);
165 0 : } else if (timeType().contains("cubic")) { // Added for CAS-10787 (16/2/2018 WK)
166 0 : Int newIdxCubic(newIdx-1);
167 0 : if (newIdxCubic < 0) {
168 0 : newIdxCubic = 0;
169 0 : } else if (newIdxCubic > (Int)timelist_.nelements()-4) {
170 0 : newIdxCubic = timelist_.nelements()-4;
171 : }
172 : //cout << "{newIdxCubic = " << newIdxCubic << " / " << timelist_.nelements() << "}" << flush;
173 0 : ScalarSampledFunctional<Float> xf(timelist_(Slice(newIdxCubic,4)));
174 0 : Vector<uInt> rows(4); indgen(rows); rows+=uInt(newIdxCubic);
175 0 : Array<Float> ya(mcols_p->fparamArray("",rows));
176 0 : ArraySampledFunctional<Array<Float> > yf(ya);
177 0 : tInterpolator_p->setData(xf,yf,true);
178 0 : }
179 : }
180 0 : }
181 : else
182 : // Escape if registration unchanged and 'nearest' or exact
183 0 : if (timeType().contains("nearest") || exact) return false; // no change
184 :
185 : // Now calculate the interpolation result
186 :
187 0 : if (timeType().contains("nearest") || exact) {
188 : if (CTTIMEINTERPVERB1) cout << " nearest or exact" << endl;
189 0 : Cube<Float> r(mcols_p->fparamArray("",Vector<uInt>(1,newIdx)));
190 0 : result_=r.xyPlane(0);
191 0 : rflag_=flaglist_.xyPlane(newIdx);
192 0 : }
193 : else {
194 : if (CTTIMEINTERPVERB1) cout << " non-trivial non-nearest" << endl;
195 : // Delegate to the interpolator
196 : // The next line is needed to restore the given interpolation type, which has been overwritten to linear in setData() above - CAS-10787 (16/2/2018 WK)
197 0 : setInterpType(timeType());
198 0 : result_=(*tInterpolator_p)(fnewtime);
199 0 : rflag_=(flaglist_.xyPlane(newIdx) || flaglist_.xyPlane(newIdx+1));
200 : }
201 :
202 : // Now remember for next round
203 0 : currTime_=newtime;
204 0 : currIdx_=newIdx;
205 0 : lastWasExact_=exact;
206 :
207 0 : return true;
208 :
209 : }
210 :
211 0 : Bool CTTimeInterp1::interpolate(Double newtime, Double freq) {
212 :
213 0 : Bool newcal=this->interpolate(newtime);
214 :
215 0 : if (newcal && timeType().contains("PD"))
216 0 : applyPhaseDelay(freq);
217 :
218 0 : return newcal;
219 :
220 : }
221 :
222 0 : void CTTimeInterp1::state(Bool verbose) {
223 :
224 0 : cout << endl << "-state---------" << endl;
225 0 : cout.precision(12);
226 0 : cout << " timeType_= " << timeType_ << endl;
227 0 : cout << " ntime = " << timelist_.nelements() << endl;
228 0 : cout << " currTime_= " << currTime_ << " (" << Float(currTime_-timeRef_) << ")" << endl;
229 0 : cout << " currIdx_ = " << currIdx_ << endl;
230 0 : cout << " timeRef_ = " << timeRef_ << endl;
231 0 : cout << " domain_ = " << domain_ << endl;
232 0 : if (verbose) {
233 0 : cout << " result_ = " << result_ << endl;
234 0 : cout << boolalpha;
235 0 : cout << " rflag_ = " << rflag_ << endl;
236 : }
237 0 : cout << "---------------" << endl;
238 0 : }
239 :
240 0 : void CTTimeInterp1::setInterpType(String strtype) {
241 0 : timeType_=strtype;
242 0 : currTime_=-999.0; // ensure force refreshed calculation
243 0 : if (strtype.contains("nearest")) {
244 0 : tInterpolator_p->setMethod(Interpolate1D<Float,Array<Float> >::nearestNeighbour);
245 0 : return;
246 : }
247 0 : if (strtype.contains("linear")) {
248 0 : tInterpolator_p->setMethod(Interpolate1D<Float,Array<Float> >::linear);
249 0 : return;
250 : }
251 0 : if (strtype.contains("cubic")) {
252 0 : tInterpolator_p->setMethod(Interpolate1D<Float,Array<Float> >::cubic);
253 0 : return;
254 : }
255 0 : if (strtype.contains("spline")) {
256 0 : tInterpolator_p->setMethod(Interpolate1D<Float,Array<Float> >::spline);
257 0 : return;
258 : }
259 0 : throw(AipsError("Unknown interp type: '"+strtype+"'!!"));
260 : }
261 :
262 0 : Bool CTTimeInterp1::findTimeRegistration(Int& idx,Bool& exact,Float newtime) {
263 :
264 : if (CTTIMEINTERPVERB1) cout << " CTTimeInterp1::findTimeRegistration()" << endl;
265 :
266 0 : Int ntime=timelist_.nelements();
267 :
268 : // If only one time in timelist, that's it, and its exact (behave as nearest)
269 0 : if (ntime==1) {
270 0 : idx=0;
271 0 : exact=true;
272 : }
273 : else {
274 : // More than one element in timelist, find the right one:
275 :
276 : // Behave as nearest outside absolute range of available calibration
277 : // to avoid wild extrapolation, else search for the correct soln slot
278 0 : if (newtime<timelist_(0)) {
279 : // Before first timestamp, use first slot as nearest one
280 0 : idx=0;
281 0 : exact=true;
282 : }
283 0 : else if (newtime>timelist_(ntime-1)) {
284 : // After last timestamp, use last slot as nearest one
285 0 : idx=ntime-1;
286 0 : exact=true;
287 : }
288 : else
289 : // Find index in timelist where time would be:
290 : // TBD: can we use last result to help this?
291 0 : idx=binarySearch(exact,timelist_,newtime,ntime,0);
292 :
293 : // If not (yet) an exact match...
294 0 : if ( !exact ) {
295 :
296 : // identify this time via index just prior
297 0 : if (idx>0) idx--;
298 :
299 : // If nearest, fine-tune slot to actual nearest:
300 0 : if ( timeType().contains("nearest") ) {
301 : // exact=true; // Nearest behaves like exact match
302 0 : if (idx!=(ntime-1) &&
303 0 : (timelist_(idx+1)-newtime) < (newtime-timelist_(idx)) )
304 0 : idx++;
305 : } else {
306 : // linear modes require one later slot
307 0 : if (idx==ntime-1) idx--;
308 : }
309 : }
310 :
311 : }
312 : // Return if new
313 0 : return (idx!=currIdx_);
314 :
315 : }
316 :
317 0 : void CTTimeInterp1::applyPhaseDelay(Double freq) {
318 :
319 0 : if (freq>0.0) {
320 0 : IPosition blc(2,1,0),trc(result_.shape()-1),stp(2,2,1);
321 0 : Matrix<Float> rph(result_(blc,trc,stp));
322 0 : if (cfreq_>0.0) {
323 0 : rph+=cycles_.xyPlane(currIdx_);
324 0 : rph*=Float(freq/cfreq_);
325 : }
326 0 : }
327 : else
328 0 : throw(AipsError("CTTimeInterp1::applyPhaseDelay: invalid frequency."));
329 :
330 0 : }
331 :
332 : } //# NAMESPACE CASA - END
|