Line data Source code
1 : //# Imager.cc: Implementation of Imager.h
2 : //# Copyright (C) 1997-2008
3 : //# Associated Universities, Inc. Washington DC, USA.
4 : //#
5 : //# This program is free software; you can redistribute it and/or modify it
6 : //# under the terms of the GNU General Public License as published by the Free
7 : //# Software Foundation; either version 2 of the License, or (at your option)
8 : //# any later version.
9 : //#
10 : //# This program 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 General Public License for
13 : //# more details.
14 : //#
15 : //# You should have received a copy of the GNU General Public License along
16 : //# with this program; if not, write to the Free Software Foundation, Inc.,
17 : //# 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 <casacore/casa/Exceptions/Error.h>
29 : #include <casacore/casa/iostream.h>
30 : #include <synthesis/MeasurementEquations/Imager.h>
31 : #include <synthesis/MeasurementComponents/EPJones.h>
32 : #include <synthesis/TransformMachines/VisModelData.h>
33 :
34 : #include <casacore/ms/MeasurementSets/MSHistoryHandler.h>
35 :
36 : #include <casacore/casa/Arrays/Matrix.h>
37 : #include <casacore/casa/Arrays/ArrayMath.h>
38 : #include <casacore/casa/Arrays/ArrayLogical.h>
39 :
40 : #include <casacore/casa/Logging.h>
41 : #include <casacore/casa/Logging/LogIO.h>
42 : #include <casacore/casa/Logging/LogMessage.h>
43 :
44 : #include <casacore/casa/OS/DirectoryIterator.h>
45 : #include <casacore/casa/OS/File.h>
46 : #include <casacore/casa/OS/Path.h>
47 :
48 : #include <casacore/casa/OS/HostInfo.h>
49 : #include <casacore/tables/Tables/RefRows.h>
50 : #include <casacore/tables/Tables/Table.h>
51 : #include <casacore/tables/Tables/TableUtil.h>
52 : #include <casacore/tables/Tables/SetupNewTab.h>
53 : #include <casacore/tables/TaQL/TableParse.h>
54 : #include <casacore/tables/Tables/TableRecord.h>
55 : #include <casacore/tables/Tables/TableDesc.h>
56 : #include <casacore/tables/Tables/TableLock.h>
57 : #include <casacore/tables/TaQL/ExprNode.h>
58 :
59 : #include <casacore/casa/BasicSL/String.h>
60 : #include <casacore/casa/Utilities/Assert.h>
61 : #include <casacore/casa/Utilities/Fallible.h>
62 : #include <casacore/casa/Utilities/CompositeNumber.h>
63 :
64 : #include <casacore/casa/BasicSL/Constants.h>
65 : #include <casacore/casa/Utilities/Regex.h>
66 :
67 : #include <casacore/casa/Logging/LogSink.h>
68 : #include <casacore/casa/Logging/LogMessage.h>
69 :
70 : #include <casacore/casa/Arrays/ArrayMath.h>
71 : #include <casacore/casa/Arrays/Slice.h>
72 : #include <casacore/images/Images/ImageExpr.h>
73 : #include <imageanalysis/ImageAnalysis/ImagePolarimetry.h>
74 : #include <synthesis/MeasurementEquations/ClarkCleanProgress.h>
75 : #include <casacore/lattices/LatticeMath/LatticeCleanProgress.h>
76 : #include <msvis/MSVis/MSUtil.h>
77 : #include <msvis/MSVis/VisSet.h>
78 : #include <msvis/MSVis/VisSetUtil.h>
79 : #include <msvis/MSVis/VisImagingWeight.h>
80 : /////////#include <msvis/MSVis/VisBufferAsync.h>
81 :
82 : // Disabling Imager::correct() (gmoellen 06Nov20)
83 : //#include <synthesis/MeasurementComponents/TimeVarVisJones.h>
84 :
85 : #include <casacore/measures/Measures/Stokes.h>
86 : #include <casacore/casa/Quanta/UnitMap.h>
87 : #include <casacore/casa/Quanta/UnitVal.h>
88 : #include <casacore/casa/Quanta/MVAngle.h>
89 : #include <casacore/measures/Measures/MDirection.h>
90 : #include <casacore/measures/Measures/MPosition.h>
91 : #include <casacore/casa/Quanta/MVEpoch.h>
92 : #include <casacore/casa/Quanta/MVTime.h>
93 : #include <casacore/measures/Measures/MEpoch.h>
94 : #include <casacore/measures/Measures/MeasTable.h>
95 :
96 : #include <casacore/ms/MeasurementSets/MeasurementSet.h>
97 : #include <casacore/ms/MeasurementSets/MSColumns.h>
98 : #include <casacore/ms/MSSel/MSSelection.h>
99 : #include <casacore/ms/MSSel/MSSelectionTools.h>
100 : #include <casacore/ms/MSSel/MSDataDescIndex.h>
101 : #include <casacore/ms/MeasurementSets/MSDopplerUtil.h>
102 : #include <casacore/ms/MSSel/MSSourceIndex.h>
103 : #include <casacore/ms/MSOper/MSSummary.h>
104 : #include <synthesis/MeasurementEquations/CubeSkyEquation.h>
105 : #include <synthesis/MeasurementEquations/Feather.h>
106 : #include <synthesis/MeasurementComponents/ImageSkyModel.h>
107 : #include <synthesis/MeasurementComponents/CEMemImageSkyModel.h>
108 : #include <synthesis/MeasurementComponents/MFCEMemImageSkyModel.h>
109 : #include <synthesis/MeasurementComponents/MFCleanImageSkyModel.h>
110 : #include <synthesis/MeasurementComponents/CSCleanImageSkyModel.h>
111 : #include <synthesis/MeasurementComponents/MFMSCleanImageSkyModel.h>
112 : #include <synthesis/MeasurementComponents/HogbomCleanImageSkyModel.h>
113 : #include <synthesis/MeasurementComponents/MSCleanImageSkyModel.h>
114 : #include <synthesis/MeasurementComponents/NNLSImageSkyModel.h>
115 : #include <synthesis/MeasurementComponents/WBCleanImageSkyModel.h>
116 : #include <synthesis/MeasurementComponents/GridBoth.h>
117 : #include <synthesis/TransformMachines/SetJyGridFT.h>
118 : #include <synthesis/TransformMachines/MosaicFT.h>
119 : #include <synthesis/TransformMachines/WProjectFT.h>
120 : #include <synthesis/MeasurementComponents/nPBWProjectFT.h>
121 : #include <synthesis/MeasurementComponents/PBMosaicFT.h>
122 : #include <synthesis/TransformMachines/PBMath.h>
123 : #include <synthesis/TransformMachines/SimpleComponentFTMachine.h>
124 : #include <synthesis/TransformMachines/VPSkyJones.h>
125 : #include <synthesis/TransformMachines/SynthesisError.h>
126 : #include <synthesis/TransformMachines/HetArrayConvFunc.h>
127 :
128 : #include <synthesis/DataSampling/SynDataSampling.h>
129 : #include <synthesis/DataSampling/SDDataSampling.h>
130 : #include <synthesis/DataSampling/ImageDataSampling.h>
131 :
132 : #include <casacore/lattices/LRegions/LattRegionHolder.h>
133 : #include <casacore/lattices/Lattices/TiledLineStepper.h>
134 : #include <casacore/lattices/Lattices/LatticeIterator.h>
135 : #include <casacore/lattices/LEL/LatticeExpr.h>
136 : #include <casacore/lattices/LatticeMath/LatticeFFT.h>
137 : #include <casacore/lattices/LRegions/LCEllipsoid.h>
138 : #include <casacore/lattices/LRegions/LCRegion.h>
139 : #include <casacore/lattices/LRegions/LCBox.h>
140 : #include <casacore/lattices/LRegions/LCIntersection.h>
141 : #include <casacore/lattices/LRegions/LCUnion.h>
142 : #include <casacore/lattices/LRegions/LCExtension.h>
143 :
144 : #include <casacore/images/Images/ImageRegrid.h>
145 : #include <casacore/images/Regions/ImageRegion.h>
146 : #include <casacore/images/Regions/RegionManager.h>
147 : #include <casacore/images/Regions/WCBox.h>
148 : #include <casacore/images/Regions/WCUnion.h>
149 : #include <casacore/images/Regions/WCIntersection.h>
150 : #include <synthesis/TransformMachines/PBMath.h>
151 : #include <casacore/images/Images/PagedImage.h>
152 : #include <casacore/images/Images/ImageInfo.h>
153 : #include <casacore/images/Images/SubImage.h>
154 : #include <casacore/images/Images/ImageUtilities.h>
155 : #include <casacore/coordinates/Coordinates/CoordinateSystem.h>
156 : #include <casacore/coordinates/Coordinates/DirectionCoordinate.h>
157 : #include <casacore/coordinates/Coordinates/SpectralCoordinate.h>
158 : #include <casacore/coordinates/Coordinates/StokesCoordinate.h>
159 : #include <casacore/coordinates/Coordinates/Projection.h>
160 : #include <casacore/coordinates/Coordinates/ObsInfo.h>
161 :
162 : #include <components/ComponentModels/ComponentList.h>
163 : #include <components/ComponentModels/ConstantSpectrum.h>
164 : #include <components/ComponentModels/SpectralIndex.h>
165 : #include <components/ComponentModels/TabularSpectrum.h>
166 : #include <components/ComponentModels/Flux.h>
167 : #include <components/ComponentModels/FluxStandard.h>
168 : #include <components/ComponentModels/PointShape.h>
169 : #include <components/ComponentModels/DiskShape.h>
170 :
171 : #ifdef USE_GRPC
172 : #include <synthesis/ImagerObjects/grpcInteractiveClean.h>
173 : #endif
174 :
175 : #include <casacore/casa/OS/HostInfo.h>
176 :
177 : #include <components/ComponentModels/ComponentList.h>
178 :
179 : #include <casacore/measures/Measures/UVWMachine.h>
180 :
181 : #include <casacore/casa/sstream.h>
182 :
183 : #include <sys/types.h>
184 : #include <unistd.h>
185 : #ifdef _OPENMP
186 : #include <omp.h>
187 : #endif
188 : using namespace std;
189 :
190 : using namespace casacore;
191 : namespace casa { //# NAMESPACE CASA - BEGIN
192 :
193 0 : Imager::Imager()
194 0 : : msname_p(""), vs_p(0), rvi_p(0), wvi_p(0), ft_p(0),
195 0 : cft_p(0), se_p(0),
196 0 : sm_p(0), vp_p(0), gvp_p(0), setimaged_p(false), nullSelect_p(false),
197 0 : mssFreqSel_p(), mssChanSel_p(),
198 0 : clean_panel_p(0), image_id_p(0), mask_id_p(0),
199 0 : prev_image_id_p(0), prev_mask_id_p(0), projection_p("SIN")
200 : {
201 0 : ms_p=0;
202 0 : mssel_p=0;
203 0 : lockCounter_p=0;
204 0 : numMS_p=0;
205 0 : defaults();
206 0 : };
207 :
208 :
209 0 : void Imager::defaults()
210 : {
211 :
212 0 : setimaged_p=false;
213 0 : nullSelect_p=false;
214 0 : nx_p=128; ny_p=128; facets_p=1;
215 0 : wprojPlanes_p=-1;
216 0 : mcellx_p=Quantity(1, "arcsec"); mcelly_p=Quantity(1, "arcsec");
217 0 : shiftx_p=Quantity(0.0, "arcsec"); shifty_p=Quantity(0.0, "arcsec");
218 0 : distance_p=Quantity(0.0, "m");
219 0 : stokes_p="I"; npol_p=1;
220 0 : nscales_p=5;
221 0 : ntaylor_p=1;
222 0 : reffreq_p=0.0;
223 0 : useNewMTFT_p=false;
224 0 : scaleMethod_p="nscales";
225 0 : scaleInfoValid_p=false;
226 0 : dataMode_p="none";
227 0 : imageMode_p="MFS";
228 0 : dataNchan_p=0;
229 0 : imageNchan_p=0;
230 0 : doVP_p=false;
231 0 : doDefaultVP_p = true;
232 0 : parAngleInc_p=Quantity(360.,"deg");
233 0 : skyPosThreshold_p=Quantity(180.,"deg");
234 0 : telescope_p="";
235 0 : gridfunction_p="SF";
236 0 : minWeight_p=0.;
237 0 : clipminmax_p=false;
238 0 : doMultiFields_p=false;
239 0 : doWideBand_p=false;
240 0 : multiFields_p=false;
241 : // Use half the machine memory as cache. The user can override
242 : // this via the setoptions function().
243 0 : cache_p=(HostInfo::memoryTotal(true)/8)*1024;
244 : //On 32 bit machines with more than 2G of mem this can become negative
245 : // overriding it to 2 Gb.
246 0 : if(cache_p <=0 )
247 0 : cache_p=2000000000/8;
248 0 : tile_p=16;
249 0 : ftmachine_p="ft";
250 0 : wfGridding_p=false;
251 0 : padding_p=1.2;
252 0 : sdScale_p=1.0;
253 0 : sdWeight_p=1.0;
254 0 : sdConvSupport_p=-1;
255 :
256 0 : doShift_p=false;
257 0 : spectralwindowids_p.resize(1);
258 0 : spectralwindowids_p=0;
259 0 : fieldid_p=0;
260 0 : dataspectralwindowids_p.resize(0);
261 0 : datadescids_p.resize(0);
262 0 : datafieldids_p.resize(0);
263 0 : mImageStart_p=MRadialVelocity(Quantity(0.0, "km/s"), MRadialVelocity::LSRK);
264 0 : mImageStep_p=MRadialVelocity(Quantity(0.0, "km/s"), MRadialVelocity::LSRK);
265 0 : mDataStart_p=MRadialVelocity(Quantity(0.0, "km/s"), MRadialVelocity::LSRK);
266 0 : mDataStep_p=MRadialVelocity(Quantity(0.0, "km/s"), MRadialVelocity::LSRK);
267 0 : beamValid_p=false;
268 0 : beam_p = ImageBeamSet();
269 0 : images_p.resize(0);
270 0 : masks_p.resize(0);
271 0 : fluxMasks_p.resize(0);
272 0 : residuals_p.resize(0);
273 0 : componentList_p=0;
274 :
275 0 : cyclefactor_p = 1.5;
276 0 : cyclespeedup_p = -1;
277 0 : cyclemaxpsffraction_p = 0.8;
278 0 : stoplargenegatives_p = 2;
279 0 : stoppointmode_p = -1;
280 0 : fluxscale_p.resize(0);
281 0 : scaleType_p = "NONE";
282 0 : minPB_p = 0.1;
283 0 : constPB_p = 0.4;
284 0 : redoSkyModel_p=true;
285 0 : nmodels_p=0;
286 0 : useModelCol_p=false;
287 0 : freqFrameValid_p=false;
288 0 : doTrackSource_p=false;
289 0 : freqInterpMethod_p="nearest";
290 0 : pointingDirCol_p="DIRECTION";
291 0 : logSink_p=LogSink(LogMessage::NORMAL, false);
292 0 : imwgt_p=VisImagingWeight();
293 0 : smallScaleBias_p=0.6;
294 0 : freqFrame_p=MFrequency::LSRK;
295 0 : imageTileVol_p=0;
296 0 : singlePrec_p=false;
297 0 : spwchansels_p.resize();
298 0 : flatnoise_p=true;
299 0 : freqrange_p.resize();
300 0 : numthreads_p=-1;
301 0 : avoidTempLatt_p=false;
302 0 : mssFreqSel_p.resize();
303 0 : mssChanSel_p.resize();
304 0 : projection_p=String("SIN");
305 :
306 0 : }
307 :
308 :
309 0 : Imager::Imager(MeasurementSet& theMS, Bool compress, Bool useModel)
310 0 : : msname_p(""), vs_p(0), rvi_p(0), wvi_p(0),
311 0 : ft_p(0), cft_p(0), se_p(0),
312 0 : sm_p(0), vp_p(0), gvp_p(0), setimaged_p(false), nullSelect_p(false),
313 0 : mssFreqSel_p(), mssChanSel_p(),
314 0 : clean_panel_p(0), image_id_p(0), mask_id_p(0), prev_image_id_p(0), prev_mask_id_p(0),
315 0 : projection_p("SIN")
316 :
317 : {
318 :
319 0 : mssel_p=0;
320 0 : ms_p=0;
321 0 : lockCounter_p=0;
322 0 : LogIO os(LogOrigin("Imager", "Imager(MeasurementSet &theMS)", WHERE));
323 0 : if(!open(theMS, compress, useModel)) {
324 0 : os << LogIO::SEVERE << "Open of MeasurementSet failed" << LogIO::EXCEPTION;
325 : };
326 :
327 0 : numMS_p=1;
328 0 : defaults();
329 0 : latestObsInfo_p=ObsInfo();
330 0 : }
331 :
332 :
333 :
334 0 : Imager::Imager(MeasurementSet& theMS, Bool compress)
335 0 : : msname_p(""), vs_p(0), rvi_p(0), wvi_p(0), ft_p(0), cft_p(0), se_p(0),
336 0 : sm_p(0), vp_p(0), gvp_p(0), setimaged_p(false), nullSelect_p(false),
337 0 : mssFreqSel_p(), mssChanSel_p(),
338 0 : clean_panel_p(0), image_id_p(0), mask_id_p(0),
339 0 : prev_image_id_p(0), prev_mask_id_p(0), projection_p("SIN")
340 : {
341 0 : mssel_p=0;
342 0 : ms_p=0;
343 0 : lockCounter_p=0;
344 0 : LogIO os(LogOrigin("Imager", "Imager(MeasurementSet &theMS)", WHERE));
345 0 : if(!open(theMS, compress)) {
346 0 : os << LogIO::SEVERE << "Open of MeasurementSet failed" << LogIO::EXCEPTION;
347 : };
348 :
349 0 : numMS_p=1;
350 0 : defaults();
351 :
352 0 : latestObsInfo_p=ObsInfo();
353 0 : }
354 :
355 0 : Imager::Imager(const Imager & other)
356 0 : : msname_p(""), vs_p(0), rvi_p(0), wvi_p(0),
357 0 : ft_p(0), cft_p(0), se_p(0),
358 0 : sm_p(0), vp_p(0), gvp_p(0), setimaged_p(false), nullSelect_p(false),
359 0 : clean_panel_p(0), image_id_p(0), mask_id_p(0), prev_image_id_p(0), prev_mask_id_p(0)
360 : {
361 0 : mssel_p=0;
362 0 : ms_p=0;
363 0 : operator=(other);
364 0 : }
365 :
366 0 : Imager &Imager::operator=(const Imager & other)
367 : {
368 0 : if (!ms_p.null() && this != &other) {
369 0 : *ms_p = *(other.ms_p);
370 : }
371 0 : if(this != &other){
372 : //Equating the table and ms parameters
373 0 : antab_p=other.antab_p;
374 0 : datadesctab_p=other.datadesctab_p;
375 0 : feedtab_p=other.feedtab_p;
376 0 : fieldtab_p=other.fieldtab_p;
377 0 : obstab_p=other.obstab_p;
378 0 : pointingtab_p=other.pointingtab_p;
379 0 : poltab_p=other.poltab_p;
380 0 : proctab_p=other.proctab_p;
381 0 : spwtab_p=other.spwtab_p;
382 0 : statetab_p=other.statetab_p;
383 0 : latestObsInfo_p=other.latestObsInfo_p;
384 0 : parAngleInc_p=other.parAngleInc_p;
385 0 : skyPosThreshold_p=other.skyPosThreshold_p;
386 0 : doTrackSource_p=other.doTrackSource_p;
387 0 : trackDir_p=other.trackDir_p;
388 0 : smallScaleBias_p=other.smallScaleBias_p;
389 0 : numMS_p=other.numMS_p;
390 0 : if (!mssel_p.null() && this != &other) {
391 0 : *mssel_p = *(other.mssel_p);
392 : }
393 0 : if (vs_p && this != &other) {
394 0 : *vs_p = *(other.vs_p);
395 : }
396 0 : if (wvi_p && this != &other) {
397 0 : *wvi_p = *(other.wvi_p);
398 0 : rvi_p=wvi_p;
399 : }
400 0 : else if(rvi_p && this != &other){
401 0 : *rvi_p = *(other.rvi_p);
402 0 : wvi_p=NULL;
403 : }
404 0 : if (ft_p && this != &other) {
405 0 : *ft_p = *(other.ft_p);
406 : }
407 0 : if (cft_p && this != &other) {
408 0 : *cft_p = *(other.cft_p);
409 : }
410 0 : if (se_p && this != &other) {
411 0 : *se_p = *(other.se_p);
412 : }
413 0 : if (sm_p && this != &other) {
414 0 : *sm_p = *(other.sm_p);
415 : }
416 0 : if (vp_p && this != &other) {
417 0 : *vp_p = *(other.vp_p);
418 : }
419 0 : if (gvp_p && this != &other) {
420 0 : *gvp_p = *(other.gvp_p);
421 : }
422 0 : imageTileVol_p=other.imageTileVol_p;
423 0 : flatnoise_p=other.flatnoise_p;
424 0 : mssFreqSel_p.assign(other.mssFreqSel_p);
425 0 : mssChanSel_p.assign(other.mssChanSel_p);
426 0 : projection_p = other.projection_p;
427 : }
428 0 : return *this;
429 : }
430 :
431 0 : Imager::~Imager()
432 : {
433 : try{
434 0 : destroySkyEquation();
435 0 : this->unlock(); //unlock things if they are in a locked state
436 :
437 : //if (mssel_p) {
438 : // delete mssel_p;
439 : // }
440 0 : mssel_p = 0;
441 : //if (ms_p) {
442 : // delete ms_p;
443 : //}
444 0 : ms_p = 0;
445 0 : if (vs_p) {
446 0 : delete vs_p;
447 : }
448 0 : if(rvi_p)
449 0 : delete rvi_p;
450 0 : rvi_p=wvi_p=0;
451 :
452 0 : vs_p = 0;
453 0 : if (ft_p) {
454 0 : delete ft_p;
455 : }
456 :
457 :
458 0 : ft_p = 0;
459 0 : if (cft_p) {
460 0 : delete cft_p;
461 : }
462 0 : cft_p = 0;
463 :
464 : }
465 0 : catch (AipsError x){
466 0 : String mess=x.getMesg();
467 : //This is a bug for wproject and facet together...
468 : //somebody is erasing a TempLattice before desturctor.
469 : //will keep this in place till i figure it out...its benign
470 0 : if(mess.contains("does not exist") && mess.contains("TempLattice")){
471 0 : String rootpath="/"+String(mess.after("/")).before("TempLattice");
472 0 : String pid=String(mess.after("TempLattice")).before("_");
473 0 : DirectoryIterator dir(rootpath, Regex(Regex::fromPattern("TempLattice"+pid+"*")));
474 0 : while(!dir.pastEnd()){
475 0 : Directory ledir(rootpath+"/"+dir.name());
476 0 : ledir.removeRecursive();
477 0 : dir++;
478 :
479 0 : }
480 :
481 0 : }
482 : else{
483 0 : throw(AipsError(x));
484 :
485 : }
486 :
487 0 : }
488 :
489 :
490 0 : }
491 :
492 :
493 0 : Bool Imager::open(MeasurementSet& theMs, Bool /*compress*/, Bool useModelCol)
494 : {
495 :
496 0 : LogIO os(LogOrigin("Imager", "open()", WHERE));
497 :
498 0 : if (!ms_p.null()) {
499 0 : *ms_p = theMs;
500 : } else {
501 0 : ms_p = new MeasurementSet(theMs);
502 0 : AlwaysAssert(!ms_p.null(), AipsError);
503 : }
504 :
505 :
506 : try {
507 0 : this->openSubTables();
508 0 : this->lock();
509 0 : msname_p = ms_p->tableName();
510 :
511 0 : os << "Opening MeasurementSet " << msname_p << LogIO::POST;
512 :
513 : // Check for DATA or FLOAT_DATA column
514 0 : if(!ms_p->tableDesc().isColumn("DATA") &&
515 0 : !ms_p->tableDesc().isColumn("FLOAT_DATA")) {
516 0 : ms_p->unlock();
517 : //delete ms_p;
518 0 : ms_p=0;
519 : os << LogIO::SEVERE
520 : << "Missing DATA or FLOAT_DATA column: imager cannot be run"
521 0 : << LogIO::EXCEPTION;
522 0 : return false;
523 : }
524 :
525 : // The unused return value creates a compiler warning, there shouldn't be any side-effect of this statement
526 : //(!ms_p->tableDesc().isColumn("CORRECTED_DATA")); // if no side effect then delete this statement?
527 :
528 : /*if(vs_p) {
529 : delete vs_p; vs_p=0;
530 : }
531 : */
532 0 : if(rvi_p){
533 0 : delete rvi_p;
534 0 : rvi_p=0;
535 0 : wvi_p=0;
536 : }
537 :
538 : // Now open the selected MeasurementSet to be initially the
539 : // same as the original MeasurementSet
540 :
541 0 : mssel_p=new MeasurementSet(*ms_p);
542 0 : useModelCol_p=useModelCol;
543 :
544 : // Now create the VisSet
545 0 : this->makeVisSet(*mssel_p);
546 0 : AlwaysAssert(rvi_p, AipsError);
547 :
548 : // Polarization
549 0 : MSColumns msc(*mssel_p);
550 0 : Vector<String> polType=msc.feed().polarizationType()(0);
551 0 : if (polType(0)!="X" && polType(0)!="Y" &&
552 0 : polType(0)!="R" && polType(0)!="L") {
553 0 : this->unlock();
554 : os << LogIO::SEVERE << "Warning: Unknown stokes types in feed table: "
555 0 : << polType(0) << endl
556 0 : << "Results open to question!" << LogIO::POST;
557 : }
558 0 : numMS_p=1;
559 :
560 0 : this->unlock();
561 :
562 0 : return true;
563 0 : } catch (AipsError x) {
564 0 : this->unlock();
565 0 : os << LogIO::SEVERE << "Caught Exception: "<< x.getMesg() << LogIO::EXCEPTION;
566 :
567 0 : return false;
568 0 : }
569 :
570 : return true;
571 0 : }
572 :
573 0 : Bool Imager::close()
574 : {
575 0 : if(!valid()) return false;
576 0 : if (detached()) return true;
577 0 : LogIO os(LogOrigin("imager", "close()", WHERE));
578 : os << LogIO::NORMAL // Loglevel PROGRESS
579 : << "Closing MeasurementSet and detaching from imager"
580 0 : << LogIO::POST;
581 0 : this->unlock();
582 0 : if(ft_p) delete ft_p; ft_p = 0;
583 0 : if(cft_p) delete cft_p; cft_p = 0;
584 0 : if(vs_p) delete vs_p; vs_p = 0;
585 0 : if(rvi_p) delete rvi_p;
586 0 : rvi_p=0;
587 0 : wvi_p=0;
588 : //if(mssel_p) delete mssel_p;
589 0 : mssel_p = 0;
590 : //if(ms_p) delete ms_p;
591 0 : ms_p = 0;
592 :
593 0 : if(se_p) delete se_p; se_p = 0;
594 :
595 0 : if(vp_p) delete vp_p; vp_p = 0;
596 0 : if(gvp_p) delete gvp_p; gvp_p = 0;
597 :
598 0 : destroySkyEquation();
599 :
600 0 : return true;
601 0 : }
602 :
603 0 : String Imager::name() const
604 : {
605 0 : if (detached()) {
606 0 : return "none";
607 : }
608 0 : return msname_p;
609 : }
610 :
611 :
612 :
613 :
614 0 : IPosition Imager::imageshape() const
615 : {
616 0 : return IPosition(4, nx_p, ny_p, npol_p, imageNchan_p);
617 : }
618 :
619 0 : Bool Imager::summary()
620 : {
621 0 : if(!valid()) return false;
622 0 : LogOrigin OR("imager", "Imager::summary()", WHERE);
623 :
624 0 : LogIO los(OR);
625 :
626 0 : los << "Logging summary" << LogIO::POST;
627 : try {
628 :
629 0 : this->lock();
630 0 : MSSummary mss(*ms_p);
631 0 : mss.list(los, true);
632 :
633 0 : los << endl << state() << LogIO::POST;
634 0 : this->unlock();
635 0 : return true;
636 0 : } catch (AipsError x) {
637 0 : this->unlock();
638 : los << LogIO::SEVERE << "Caught Exception: " << x.getMesg()
639 0 : << LogIO::EXCEPTION;
640 0 : return false;
641 0 : }
642 :
643 : return true;
644 0 : }
645 :
646 :
647 0 : Bool Imager::setimage(const Int nx, const Int ny,
648 : const Quantity& cellx, const Quantity& celly,
649 : const String& stokes,
650 : Bool doShift,
651 : const MDirection& phaseCenter,
652 : const Quantity& shiftx, const Quantity& shifty,
653 : const String& mode, const Int nchan,
654 : const Int start, const Int step,
655 : const MRadialVelocity& mStart, const MRadialVelocity& mStep,
656 : const Vector<Int>& spectralwindowids,
657 : const Int fieldid,
658 : const Int facets,
659 : const Quantity& distance)
660 : {
661 :
662 0 : if(!valid())
663 : {
664 0 : return false;
665 : }
666 :
667 : //Clear the sink
668 0 : logSink_p.clearLocally();
669 0 : LogIO os(LogOrigin("imager", "setimage()"), logSink_p);
670 :
671 : os << "nx=" << nx << " ny=" << ny
672 0 : << " cellx='" << cellx.getValue() << cellx.getUnit()
673 0 : << "' celly='" << celly.getValue() << celly.getUnit()
674 : << "' stokes=" << stokes << " doShift=" << doShift
675 0 : << " shiftx='" << shiftx.getValue() << shiftx.getUnit()
676 0 : << "' shifty='" << shifty.getValue() << shifty.getUnit()
677 : << "' mode=" << mode << " nchan=" << nchan
678 : << " start=" << start << " step=" << step
679 : << " spwids=" << spectralwindowids
680 : << " fieldid=" << fieldid << " facets=" << facets
681 0 : << " distance='" << distance.getValue() << distance.getUnit() <<"'";
682 0 : ostringstream clicom;
683 0 : clicom << " phaseCenter='" << phaseCenter;
684 0 : clicom << "' mStart='" << mStart << "' mStep='" << mStep << "'";
685 0 : os << String(clicom);
686 :
687 : try {
688 :
689 0 : this->lock();
690 0 : this->writeCommand(os);
691 :
692 0 : os << LogIO::NORMAL << "Defining image properties" << LogIO::POST; // Loglevel INFO
693 :
694 : /**** this check is not really needed here especially for SD imaging
695 : if(2*Int(nx/2)!=nx) {
696 : this->unlock();
697 : os << LogIO::SEVERE << "nx must be even" << LogIO::POST;
698 : return false;
699 : }
700 : if(2*Int(ny/2)!=ny) {
701 : this->unlock();
702 : os << LogIO::SEVERE << "ny must be even" << LogIO::POST;
703 : return false;
704 : }
705 :
706 : */
707 : {
708 0 : CompositeNumber cn(nx);
709 0 : if (! cn.isComposite(nx)) {
710 0 : Int nxc = (Int)cn.nextLargerEven(nx);
711 0 : Int nnxc = (Int)cn.nearestEven(nx);
712 0 : if (nxc == nnxc) {
713 : os << LogIO::WARN << "nx = " << nx << " is not composite; nx = "
714 0 : << nxc << " will be more efficient" << LogIO::POST;
715 : } else {
716 : os << LogIO::WARN << "nx = " << nx << " is not composite; nx = "
717 0 : << nxc << " or " << nnxc << " will be more efficient for FFTs" << LogIO::POST;
718 : }
719 : }
720 0 : if (! cn.isComposite(ny)) {
721 0 : Int nyc = (Int)cn.nextLargerEven(ny);
722 0 : Int nnyc = (Int)cn.nearestEven(ny);
723 0 : if (nyc == nnyc) {
724 : os << LogIO::WARN << "ny = " << ny << " is not composite; ny = "
725 0 : << nyc << " will be more efficient" << LogIO::POST;
726 : } else {
727 : os << LogIO::WARN << "ny = " << ny << " is not composite ; ny = " << nyc <<
728 0 : " or " << nnyc << " will be more efficient for FFTs" << LogIO::POST;
729 : }
730 : os << LogIO::WARN
731 : << "You may safely ignore this message for single dish imaging"
732 0 : << LogIO::POST;
733 :
734 : }
735 :
736 0 : }
737 :
738 :
739 0 : nx_p=nx;
740 0 : ny_p=ny;
741 0 : mcellx_p=cellx;
742 0 : mcelly_p=celly;
743 0 : distance_p=distance;
744 0 : stokes_p=stokes;
745 0 : imageMode_p=mode;
746 0 : imageMode_p.upcase();
747 0 : imageNchan_p=nchan;
748 0 : imageStart_p=start;
749 0 : imageStep_p=step;
750 0 : mImageStart_p=mStart;
751 0 : mImageStep_p=mStep;
752 0 : spectralwindowids_p.resize(spectralwindowids.nelements());
753 0 : spectralwindowids_p=spectralwindowids;
754 0 : fieldid_p=fieldid;
755 0 : facets_p=facets;
756 0 : redoSkyModel_p=true;
757 0 : destroySkyEquation();
758 :
759 0 : Vector<Int> whichStokes = decideNPolPlanes(false);
760 0 : if( whichStokes.nelements()==1 && whichStokes[0]==0 )
761 : {
762 0 : this->unlock();
763 0 : os << LogIO::SEVERE << "Stokes selection " << stokes_p << " is currently not supported." << LogIO::EXCEPTION;
764 0 : return false;
765 : }
766 :
767 :
768 :
769 : //THIS NEEDS TO GO
770 : ////this->setImageParam(nx_p, ny_p, npol_p, imageNchan_p);
771 0 : nchan_p = imageNchan_p; // check if this is needed.... it's the only non-redundant thing in setImageParam.
772 :
773 : // Now do the shifts
774 : // MSColumns msc(*ms_p);
775 :
776 0 : doShift_p=doShift;
777 0 : if(doShift_p) {
778 0 : phaseCenter_p=phaseCenter;
779 : }
780 : else {
781 :
782 0 : MSFieldColumns msfield(ms_p->field());
783 0 : phaseCenter_p=msfield.phaseDirMeas(fieldid_p);
784 : // phaseCenter_p=msc.field().phaseDirMeas(fieldid_p);
785 0 : }
786 :
787 : // Now add the optional shifts
788 0 : shiftx_p=shiftx;
789 0 : shifty_p=shifty;
790 0 : if(shiftx_p.get().getValue()!=0.0||shifty_p.get().getValue()!=0.0) {
791 0 : Vector<Double> vPhaseCenter(phaseCenter_p.getAngle().getValue());
792 0 : if(cos(vPhaseCenter(1))!=0.0) {
793 0 : vPhaseCenter(0)+=shiftx_p.get().getValue()/cos(vPhaseCenter(1));
794 : }
795 0 : vPhaseCenter(1)+=shifty_p.get().getValue();
796 0 : phaseCenter_p.set(MVDirection(vPhaseCenter));
797 0 : }
798 :
799 : // Now we have set the image parameters
800 0 : setimaged_p=true;
801 0 : beamValid_p=false;
802 :
803 0 : this->unlock();
804 :
805 0 : return true;
806 0 : } catch (AipsError x) {
807 :
808 0 : this->unlock();
809 :
810 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
811 0 : << LogIO::EXCEPTION;
812 0 : return false;
813 0 : }
814 :
815 : return true;
816 0 : }
817 :
818 :
819 0 : Bool Imager::defineImage(const Int nx, const Int ny,
820 : const Quantity& cellx, const Quantity& celly,
821 : const String& stokes,
822 : const MDirection& phaseCenter, const Int fieldid,
823 : const String& mode, const Int nchan,
824 : const Int start, const Int step,
825 : const MFrequency& mFreqStart,
826 : const MRadialVelocity& mStart,
827 : const Quantity& qStep,
828 : const Vector<Int>& spectralwindowids,
829 : const Int facets,
830 : const Quantity& restFreq,
831 : const MFrequency::Types& mFreqFrame,
832 : const Quantity& distance, const Bool dotrackDir,
833 : const MDirection& trackDir,
834 : const String& projection)
835 : {
836 :
837 :
838 :
839 :
840 : //Clear the sink
841 0 : logSink_p.clearLocally();
842 0 : LogIO os(LogOrigin("imager", "defineimage()"), logSink_p);
843 0 : if(cellx.getValue() == 0.0 || celly.getValue()==0.0)
844 0 : throw(AipsError("Infinite resolution not possible... please do let us know what you are drinking"));
845 0 : os << LogIO::NORMAL << "Defining image properties:"; // Loglevel INFO
846 : os << "nx=" << nx << " ny=" << ny
847 0 : << " cellx='" << cellx.getValue() << cellx.getUnit()
848 0 : << "' celly='" << celly.getValue() << celly.getUnit()
849 : << "' stokes=" << stokes
850 : << "' mode=" << mode << " nchan=" << nchan
851 : << " start=" << start << " step=" << step
852 : << " spwids=" << spectralwindowids
853 : << " fieldid=" << fieldid << " facets=" << facets
854 0 : << " frame=" << mFreqFrame
855 0 : << " distance='" << distance.getValue() << distance.getUnit() <<"'";
856 0 : os << LogIO::POST;
857 0 : ostringstream clicom;
858 0 : clicom << " phaseCenter='" ;
859 0 : if(fieldid < 0){
860 0 : MVAngle mvRA=phaseCenter.getAngle().getValue()(0);
861 0 : MVAngle mvDEC=phaseCenter.getAngle().getValue()(1);
862 0 : clicom << mvRA(0.0).string(MVAngle::TIME,8) << ", ";
863 0 : clicom << mvDEC(0.0).string(MVAngle::ANGLE_CLEAN,8) << ", ";
864 0 : }
865 : else{
866 0 : clicom << "field-" << fieldid<< " ";
867 : }
868 0 : clicom << "' mStart='" << mStart << "' qStep='" << qStep << "'";
869 0 : clicom << "' mFreqStart='" << mFreqStart;
870 0 : os << String(clicom);
871 0 : os << LogIO::POST;
872 :
873 : try {
874 :
875 0 : this->lock();
876 0 : this->writeCommand(os);
877 :
878 0 : doTrackSource_p=dotrackDir;
879 0 : trackDir_p=trackDir;
880 :
881 :
882 : /**** this check is not really needed here especially for SD imaging
883 : if(2*Int(nx/2)!=nx) {
884 : this->unlock();
885 : os << LogIO::SEVERE << "nx must be even" << LogIO::POST;
886 : return false;
887 : }
888 : if(2*Int(ny/2)!=ny) {
889 : this->unlock();
890 : os << LogIO::SEVERE << "ny must be even" << LogIO::POST;
891 : return false;
892 : }
893 :
894 : */
895 : // RVU : Disabling this Composite Number check, because image sizes are
896 : // anyway being re-calculated inside the FTMachines (accounding for padding too).
897 : if(0)
898 : {
899 : CompositeNumber cn(nx);
900 : if (! cn.isComposite(nx)) {
901 : Int nxc = (Int)cn.nextLargerEven(nx);
902 : Int nnxc = (Int)cn.nearestEven(nx);
903 : if (nxc == nnxc) {
904 : os << LogIO::POST << "nx = " << nx << " is not composite; nx = "
905 : << nxc << " will be more efficient" << LogIO::POST;
906 : } else {
907 : os << LogIO::POST << "nx = " << nx << " is not composite; nx = "
908 : << nxc << " or " << nnxc << " will be more efficient" << LogIO::POST;
909 : }
910 : }
911 : if (! cn.isComposite(ny)) {
912 : Int nyc = (Int)cn.nextLargerEven(ny);
913 : Int nnyc = (Int)cn.nearestEven(ny);
914 : if (nyc == nnyc) {
915 : os << LogIO::POST << "ny = " << ny << " is not composite; ny = "
916 : << nyc << " will be more efficient" << LogIO::POST;
917 : } else {
918 : os << LogIO::POST << "ny = " << ny << " is not composite; ny = " << nyc <<
919 : " or " << nnyc << " will be more efficient" << LogIO::POST;
920 : }
921 : os << LogIO::POST
922 : << "You may safely ignore this message for single dish imaging"
923 : << LogIO::POST;
924 :
925 : }
926 :
927 : }
928 :
929 :
930 0 : if((abs(Double(nx)*cellx.getValue("rad")) > C::pi) || (abs(Double(ny)*celly.getValue("rad")) > C::pi))
931 0 : throw(AipsError("Cannot image the extent requested for this image; more that PI ialong one or both of the axes " ));
932 :
933 :
934 :
935 0 : nx_p=nx;
936 0 : ny_p=ny;
937 0 : mcellx_p=cellx;
938 0 : mcelly_p=celly;
939 0 : distance_p=distance;
940 0 : stokes_p=stokes;
941 0 : imageMode_p=mode;
942 0 : imageMode_p.upcase();
943 0 : imageNchan_p=nchan;
944 0 : imageStart_p=start;
945 0 : imageStep_p=step;
946 0 : if(mode.contains("VEL")){
947 0 : mImageStart_p=mStart;
948 0 : mImageStep_p=MRadialVelocity(qStep);
949 : }
950 0 : if(mode.contains("FREQ")){
951 0 : mfImageStart_p=mFreqStart;
952 0 : mfImageStep_p=MFrequency(qStep);
953 : }
954 0 : restFreq_p=restFreq;
955 0 : freqFrame_p=mFreqFrame;
956 0 : spectralwindowids_p.resize(spectralwindowids.nelements());
957 0 : spectralwindowids_p=spectralwindowids;
958 0 : fieldid_p=fieldid;
959 0 : facets_p=facets;
960 0 : redoSkyModel_p=true;
961 0 : destroySkyEquation();
962 :
963 : // Now make the derived quantities
964 0 : Vector<Int> whichStokes = decideNPolPlanes(false);
965 0 : if( whichStokes.nelements()==1 && whichStokes[0]==0 )
966 : {
967 0 : this->unlock();
968 0 : os << LogIO::SEVERE << "Stokes selection " << stokes_p << " is currently not supported." << LogIO::EXCEPTION;
969 0 : return false;
970 : }
971 :
972 : // nchan we need to get rid of one of these variables
973 0 : nchan_p=imageNchan_p;
974 :
975 0 : if(fieldid < 0){
976 0 : doShift_p=true;
977 0 : phaseCenter_p=phaseCenter;
978 : }
979 : else {
980 0 : MSFieldColumns msfield(ms_p->field());
981 0 : phaseCenter_p=msfield.phaseDirMeas(fieldid_p);
982 0 : }
983 :
984 : // we only support projections SIN, CAR, TAN, and SFL
985 0 : Projection::Type const ptype = Projection::type(projection);
986 0 : if (ptype != Projection::SIN && ptype != Projection::CAR
987 0 : && ptype != Projection::TAN && ptype != Projection::SFL) {
988 0 : this->unlock();
989 0 : os << LogIO::SEVERE << "Projection " << projection << " is currently not supported." << LogIO::EXCEPTION;
990 0 : return false;
991 : } else {
992 0 : projection_p = projection;
993 : }
994 :
995 :
996 : // Now we have set the image parameters
997 0 : setimaged_p=true;
998 0 : beamValid_p=false;
999 :
1000 0 : this->unlock();
1001 :
1002 :
1003 0 : return true;
1004 0 : } catch (AipsError x) {
1005 0 : this->unlock();
1006 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
1007 0 : << LogIO::EXCEPTION;
1008 0 : return false;
1009 0 : }
1010 : return true;
1011 0 : }
1012 :
1013 :
1014 0 : Bool Imager::advise(const Bool takeAdvice, const Float amplitudeLoss,
1015 : const Quantity& fieldOfView, Quantity& cell,
1016 : Int& pixels, Int& facets, MDirection& phaseCenter)
1017 : {
1018 0 : if(!valid()) return false;
1019 :
1020 0 : LogIO os(LogOrigin("imager", "advise()", WHERE));
1021 :
1022 : try {
1023 :
1024 0 : os << "Advising image properties" << LogIO::POST;
1025 :
1026 0 : Float maxAbsUV=0.0;
1027 0 : Float maxWtAbsUV=0.0;
1028 : // To determine the number of facets, we need to fit w to
1029 : // a.u + b.v. The misfit from this (i.e. the dispersion
1030 : // will determine the error beam due to the non-coplanar
1031 : // baselines. We'll do both cases: where the position
1032 : // errors are important and where they are not. We'll use
1033 : // the latter.
1034 0 : Double sumWt = 0.0;
1035 :
1036 0 : Double sumUU=0.0;
1037 0 : Double sumUV=0.0;
1038 0 : Double sumUW=0.0;
1039 0 : Double sumVV=0.0;
1040 0 : Double sumVW=0.0;
1041 0 : Double sumWW=0.0;
1042 :
1043 0 : Double sumWtUU=0.0;
1044 0 : Double sumWtUV=0.0;
1045 0 : Double sumWtUW=0.0;
1046 0 : Double sumWtVV=0.0;
1047 0 : Double sumWtVW=0.0;
1048 0 : Double sumWtWW=0.0;
1049 :
1050 0 : Double sum = 0.0;
1051 :
1052 0 : this->lock();
1053 0 : ROVisIter& vi(*rvi_p);
1054 0 : VisBuffer vb(vi);
1055 :
1056 0 : for (vi.originChunks(); vi.moreChunks(); vi.nextChunk()) {
1057 0 : for (vi.origin();vi.more();vi++) {
1058 0 : Int nRow=vb.nRow();
1059 0 : Int nChan=vb.nChannel();
1060 0 : for (Int row=0; row<nRow; ++row) {
1061 0 : for (Int chn=0; chn<nChan; ++chn) {
1062 0 : if(!vb.flag()(chn,row)) {
1063 0 : Float f=vb.frequency()(chn)/C::c;
1064 0 : Float u=vb.uvw()(row)(0)*f;
1065 0 : Float v=vb.uvw()(row)(1)*f;
1066 0 : Float w=vb.uvw()(row)(2)*f;
1067 0 : Double wt=vb.imagingWeight()(chn,row);
1068 0 : if(wt>0.0) {
1069 0 : if(abs(u)>maxWtAbsUV) maxWtAbsUV=abs(u);
1070 0 : if(abs(v)>maxWtAbsUV) maxWtAbsUV=abs(v);
1071 0 : sumWt += wt;
1072 0 : sumWtUU += wt * u * u;
1073 0 : sumWtUV += wt * u * v;
1074 0 : sumWtUW += wt * u * w;
1075 0 : sumWtVV += wt * v * v;
1076 0 : sumWtVW += wt * v * w;
1077 0 : sumWtWW += wt * w * w;
1078 : }
1079 0 : sum += 1;
1080 0 : if(abs(u)>maxAbsUV) maxAbsUV=abs(u);
1081 0 : if(abs(v)>maxAbsUV) maxAbsUV=abs(v);
1082 0 : sumUU += u * u;
1083 0 : sumUV += u * v;
1084 0 : sumUW += u * w;
1085 0 : sumVV += v * v;
1086 0 : sumVW += v * w;
1087 0 : sumWW += w * w;
1088 : }
1089 : }
1090 : }
1091 : }
1092 : }
1093 :
1094 :
1095 0 : if(sumWt==0.0) {
1096 0 : os << LogIO::WARN << "Visibility data are not yet weighted: using unweighted values" << LogIO::POST;
1097 0 : sumWt = sum;
1098 : }
1099 : else {
1100 0 : sumUU = sumWtUU;
1101 0 : sumUV = sumWtUV;
1102 0 : sumUW = sumWtUW;
1103 0 : sumVV = sumWtVV;
1104 0 : sumVW = sumWtVW;
1105 0 : sumWW = sumWtWW;
1106 0 : maxAbsUV = maxWtAbsUV;
1107 : }
1108 :
1109 : // First find the cell size
1110 0 : if(maxAbsUV==0.0) {
1111 0 : this->unlock();
1112 0 : os << LogIO::SEVERE << "Maximum uv distance is zero" << LogIO::POST;
1113 0 : return false;
1114 : }
1115 : else {
1116 0 : cell=Quantity(0.5/maxAbsUV, "rad").get("arcsec");
1117 0 : os << "Maximum uv distance = " << maxAbsUV << " wavelengths" << endl;
1118 0 : os << "Recommended cell size < " << cell.get("arcsec").getValue()
1119 0 : << " arcsec" << LogIO::POST;
1120 : }
1121 :
1122 : // Now we can find the number of pixels for the specified field of view
1123 0 : pixels = 2*Int((fieldOfView.get("rad").getValue()/cell.get("rad").getValue())/2.0);
1124 : {
1125 0 : CompositeNumber cn(pixels);
1126 0 : pixels = (Int) (cn.nextLargerEven(pixels));
1127 0 : }
1128 0 : if(pixels < 64) pixels = 64;
1129 0 : os << "Recommended number of pixels = " << pixels << endl;
1130 :
1131 : // Rough rule for number of facets:
1132 : // For the specified facet size, the loss in amplitude
1133 : // due to the peeling of facets from the sphere should
1134 : // be equal to the amplitude error.
1135 0 : Int worstCaseFacets=1;
1136 0 : if(sumWt<=0.0||sumUU<=0.0||(sumUU+sumVV)<=0.0) {
1137 0 : this->unlock();
1138 0 : os << LogIO::SEVERE << "Sum of imaging weights is zero" << LogIO::POST;
1139 0 : return false;
1140 : }
1141 : else {
1142 0 : Double rmsUV = sqrt((sumUU + sumVV)/sumWt);
1143 0 : Double rmsW = sqrt(sumWW/sumWt);
1144 : os << "Dispersion in uv, w distance = " << rmsUV << ", "<< rmsW
1145 0 : << " wavelengths" << endl;
1146 0 : if(rmsW>0.0&&rmsUV>0.0&&litudeLoss>0.0) {
1147 0 : worstCaseFacets =
1148 0 : Int (pixels * (abs(cell.get("rad").getValue())*
1149 0 : sqrt(C::pi*rmsW/(sqrt(32.0*amplitudeLoss)))));
1150 : }
1151 : else {
1152 : os << LogIO::WARN << "Cannot calculate number of facets: using 1"
1153 0 : << LogIO::POST;
1154 0 : worstCaseFacets = 1;
1155 : }
1156 : // Solve for the parameters:
1157 0 : Double Determinant = sumUU * sumVV - square(sumUV);
1158 0 : Double rmsFittedW = rmsW;
1159 0 : if(Determinant > 0.0) {
1160 0 : Double a = ( sumVV * sumUW - sumUV * sumVW)/Determinant;
1161 0 : Double b = (-sumUV * sumUW + sumUU * sumVW)/Determinant;
1162 : os << "Best fitting plane is w = " << a << " * u + "
1163 0 : << b << " * v" << endl;
1164 : Double FittedWW =
1165 0 : sumWW + square(a) * sumUU + square(b) * sumVV +
1166 0 : + 2.0 * a * b * sumUV - 2.0 * (a * sumUW + b * sumVW);
1167 0 : rmsFittedW = sqrt(FittedWW/sumWt);
1168 : os << "Dispersion in fitted w = " << rmsFittedW
1169 0 : << " wavelengths" << endl;
1170 0 : facets = Int (pixels * (abs(cell.get("rad").getValue())*
1171 0 : sqrt(C::pi*rmsFittedW/(sqrt(32.0*amplitudeLoss)))));
1172 0 : if (facets<1) facets = 1;
1173 : }
1174 : else {
1175 0 : os << "Error in fitting plane to uvw data" << LogIO::POST;
1176 : }
1177 0 : if(worstCaseFacets<1) worstCaseFacets=1;
1178 0 : if(worstCaseFacets>1) {
1179 : os << "imager recommends that you use the wide field clean" << endl
1180 : << "For accurate positions, use " << worstCaseFacets
1181 : << " facets on each axis" << endl
1182 : << "For accurate removal of sources, you only need "
1183 0 : << facets << " facets on each axis" << LogIO::POST;
1184 : }
1185 : else {
1186 : os << "Wide field cleaning is not necessary"
1187 0 : << LogIO::POST;
1188 : }
1189 : }
1190 :
1191 0 : MSColumns msc(*mssel_p);
1192 0 : if(datafieldids_p.shape()!=0){
1193 : //If setdata has been used prior to this
1194 0 : phaseCenter=msc.field().phaseDirMeas(datafieldids_p(0));
1195 : }
1196 : else{
1197 0 : phaseCenter=msc.field().phaseDirMeas(fieldid_p);
1198 : }
1199 :
1200 :
1201 : // Now we have set the image parameters
1202 0 : if(takeAdvice) {
1203 0 : os << "Using advised image properties" << LogIO::POST;
1204 0 : mcellx_p=cell;
1205 0 : mcelly_p=cell;
1206 0 : phaseCenter_p=phaseCenter;
1207 0 : setimaged_p=true;
1208 0 : beamValid_p=false;
1209 0 : facets_p=facets;
1210 0 : nx_p=ny_p=pixels;
1211 : }
1212 :
1213 0 : this->unlock();
1214 0 : return true;
1215 0 : } catch (AipsError x) {
1216 0 : this->unlock();
1217 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
1218 0 : << LogIO::EXCEPTION;
1219 0 : return false;
1220 0 : }
1221 :
1222 : return true;
1223 0 : }
1224 :
1225 :
1226 :
1227 0 : Bool Imager::setDataPerMS(const String& msname, const String& mode,
1228 : const Vector<Int>& nchan,
1229 : const Vector<Int>& start,
1230 : const Vector<Int>& step,
1231 : const Vector<Int>& spectralwindowids,
1232 : const Vector<Int>& fieldids,
1233 : const String& msSelect, const String& timerng,
1234 : const String& fieldnames,
1235 : const Vector<Int>& antIndex,
1236 : const String& antnames,
1237 : const String& spwstring,
1238 : const String& uvdist, const String& scan,
1239 : const String& intent, const String& obs,
1240 : const Bool useModelCol, const Bool /*readonly*/)
1241 : {
1242 0 : LogIO os(LogOrigin("imager", "setdata()"), logSink_p);
1243 0 : if(msname != ""){
1244 :
1245 0 : LogIO os(LogOrigin("imager", "setdata()"), logSink_p);
1246 : os << LogIO::WARN
1247 : << "Ignoring that ms" << msname << "specified here"
1248 0 : << LogIO::POST;
1249 : os << LogIO::WARN
1250 : << "Imager was constructed with an ms "
1251 0 : << LogIO::POST;
1252 : os << LogIO::WARN
1253 : << "if multi-ms are to be used please construct imager without parameters and use setdata to specify the ms's and selection"
1254 0 : << LogIO::POST;
1255 :
1256 0 : }
1257 0 : MRadialVelocity dummy;
1258 : //Calling the old setdata
1259 0 : return setdata(mode, nchan, start, step, dummy, dummy, spectralwindowids,
1260 : fieldids, msSelect, timerng, fieldnames, antIndex,
1261 0 : antnames, spwstring, uvdist, scan, intent, obs, useModelCol);
1262 :
1263 0 : }
1264 :
1265 :
1266 0 : Bool Imager::setdata(const String& mode, const Vector<Int>& nchan,
1267 : const Vector<Int>& start, const Vector<Int>& step,
1268 : const MRadialVelocity& mStart,
1269 : const MRadialVelocity& mStep,
1270 : const Vector<Int>& spectralwindowids,
1271 : const Vector<Int>& fieldids,
1272 : const String& msSelect, const String& timerng,
1273 : const String& fieldnames, const Vector<Int>& antIndex,
1274 : const String& antnames, const String& spwstring,
1275 : const String& uvdist, const String& scan,
1276 : const String& intent,
1277 : const String& obs,
1278 : const Bool /*useModelCol*/,
1279 : const Bool be_calm)
1280 : {
1281 0 : logSink_p.clearLocally();
1282 0 : LogIO os(LogOrigin("imager", "data selection"), logSink_p);
1283 :
1284 0 : if(ms_p.null()) {
1285 : os << LogIO::SEVERE << "Program logic error: MeasurementSet pointer ms_p not yet set"
1286 0 : << LogIO::EXCEPTION;
1287 0 : nullSelect_p=true;
1288 0 : return false;
1289 : }
1290 :
1291 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1292 : << "mode=" << mode << " nchan=" << nchan
1293 0 : << " start=" << start << " step=" << step;
1294 0 : ostringstream clicom;
1295 0 : clicom << " mstart='" << mStart << "' mstep='" << mStep;
1296 0 : os << String(clicom) ;
1297 : os << "' spectralwindowids=" << spectralwindowids
1298 0 : << " fieldids=" << fieldids << " msselect=" << msSelect;
1299 :
1300 0 : nullSelect_p=false;
1301 0 : String local_spwstring(spwstring);
1302 0 : if (local_spwstring == "") local_spwstring="*";
1303 : try {
1304 :
1305 0 : this->lock();
1306 0 : this->writeCommand(os);
1307 :
1308 : os << (be_calm ? LogIO::NORMAL2 : LogIO::NORMAL)
1309 0 : << "Performing selection on MeasurementSet : " << ms_p->tableName()
1310 0 : << LogIO::POST; // Loglevel PROGRESS
1311 : //Some MSSelection
1312 0 : MSSelection thisSelection;
1313 :
1314 : // check that sorted table exists (it should), if not, make it now.
1315 : //this->makeVisSet(*ms_p);
1316 :
1317 : //MeasurementSet sorted=ms_p->keywordSet().asTable("SORTED_TABLE");
1318 : //MSSelection thisSelection (sorted, MSSelection::PARSE_NOW,timerng,antnames,
1319 : // fieldnames, local_spwstring,uvdist, msSelect,"",
1320 : // scan, obs);
1321 :
1322 0 : datafieldids_p.resize(fieldids.nelements());
1323 0 : datafieldids_p = fieldids;
1324 0 : if(datafieldids_p.nelements() > 0){
1325 0 : thisSelection.setFieldExpr(MSSelection::indexExprStr(datafieldids_p));
1326 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1327 0 : << "Selecting on field ids : " << datafieldids_p << LogIO::POST;
1328 : }
1329 0 : if(fieldnames != ""){
1330 0 : thisSelection.setFieldExpr(fieldnames);
1331 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1332 0 : << "Selecting on fields : " << fieldnames << LogIO::POST;
1333 : }
1334 :
1335 0 : dataspectralwindowids_p.resize(spectralwindowids.nelements());
1336 0 : dataspectralwindowids_p = spectralwindowids;
1337 0 : if(dataspectralwindowids_p.nelements() > 0){
1338 0 : thisSelection.setSpwExpr(MSSelection::indexExprStr(dataspectralwindowids_p));
1339 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1340 0 : << "Selecting on spectral windows" << LogIO::POST;
1341 : }
1342 0 : else if(local_spwstring != ""){
1343 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1344 : << "Selecting on spectral windows expression : " << local_spwstring
1345 0 : << LogIO::POST;
1346 0 : thisSelection.setSpwExpr(local_spwstring);
1347 : }
1348 :
1349 0 : if(antIndex.nelements() >0){
1350 0 : thisSelection.setAntennaExpr( MSSelection::indexExprStr(antIndex));
1351 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1352 0 : << "Selecting on antenna ids : " << antIndex << LogIO::POST;
1353 : }
1354 0 : if(antnames != ""){
1355 0 : Vector<String>antNames(1, antnames);
1356 : // thisSelection.setAntennaExpr(MSSelection::nameExprStr(antNames));
1357 0 : thisSelection.setAntennaExpr(antnames);
1358 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1359 0 : << "Selecting on antenna names : " << antnames << LogIO::POST;
1360 0 : }
1361 :
1362 0 : if(timerng != ""){
1363 : // Vector<String>timerange(1, timerng);
1364 0 : thisSelection.setTimeExpr(timerng);
1365 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1366 0 : << "Selecting on time range : " << timerng << LogIO::POST;
1367 : }
1368 :
1369 0 : if(uvdist != ""){
1370 0 : thisSelection.setUvDistExpr(uvdist);
1371 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1372 0 : << "Selecting on uvdist : " << uvdist << LogIO::POST;
1373 : }
1374 :
1375 0 : if(scan != ""){
1376 0 : thisSelection.setScanExpr(scan);
1377 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1378 0 : << "Selecting on scan : " << scan << LogIO::POST;
1379 : }
1380 0 : if(intent != "") {
1381 0 : thisSelection.setStateExpr(intent);
1382 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1383 0 : << "Selecting on State Expr : " << intent << LogIO::POST;
1384 : }
1385 0 : if(obs != ""){
1386 0 : thisSelection.setObservationExpr(obs);
1387 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1388 0 : << "Selecting on Observation Expr : " << obs << LogIO::POST;
1389 : }
1390 0 : if(msSelect != ""){
1391 0 : thisSelection.setTaQLExpr(msSelect);
1392 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1393 0 : << "Selecting via TaQL : " << msSelect << LogIO::POST;
1394 : }
1395 :
1396 : //***************
1397 0 : TableExprNode exprNode;
1398 : try{
1399 0 : exprNode = thisSelection.toTableExprNode(&(*ms_p));
1400 : }
1401 0 : catch(...){
1402 0 : nullSelect_p = true;
1403 0 : this->unlock();
1404 : // A "bad selection" warning message could be sent to the logger here,
1405 : // but it should be left to the calling function to do that. For
1406 : // example, this function is called by setjy, and it would be a mistake
1407 : // to print a logger message for every spw that was not observed for the
1408 : // given field.
1409 0 : return false;
1410 0 : }
1411 : //TableExprNode exprNode=thisSelection.getTEN();
1412 : //if(exprNode.isNull()){
1413 : // throw(AipsError("Selection failed...review ms and selection parameters"));
1414 : //}
1415 :
1416 : // Need to delete the ft machine as the channel flag may change
1417 0 : if(ft_p)
1418 0 : delete ft_p;
1419 0 : ft_p=0;
1420 0 : dataMode_p=mode;
1421 0 : dataNchan_p.resize();
1422 0 : dataStart_p.resize();
1423 0 : dataStep_p.resize();
1424 0 : dataNchan_p=nchan;
1425 0 : dataStart_p=start;
1426 0 : dataStep_p=step;
1427 0 : mDataStart_p=mStart;
1428 0 : mDataStep_p=mStep;
1429 : // useModelCol_p=useModelCol;
1430 :
1431 0 : if(rvi_p)
1432 0 : delete rvi_p;
1433 0 : rvi_p=0;
1434 0 : wvi_p=0;
1435 : // if(mssel_p) delete mssel_p;
1436 0 : mssel_p=0;
1437 :
1438 0 : datafieldids_p.resize();
1439 0 : datafieldids_p=thisSelection.getFieldList();
1440 0 : if(datafieldids_p.nelements()==0){
1441 0 : Int nf=ms_p->field().nrow();
1442 0 : datafieldids_p.resize(nf);
1443 0 : indgen(datafieldids_p);
1444 : }
1445 : //Now lets see what was selected as spw and match it with datadesc
1446 : //
1447 : // getSpwList could return duplicated spw ids
1448 : // when multiple channel ranges are specified.
1449 : //dataspectralwindowids_p.resize();
1450 : //dataspectralwindowids_p=thisSelection.getSpwList();
1451 : //get channel selection in spw
1452 : // TT: Added sorting option in getChanList call
1453 : // to accomodate changes related CAS-2521
1454 0 : Matrix<Int> chansels=thisSelection.getChanList(NULL, 1, true);
1455 0 : mssChanSel_p.assign(chansels);
1456 0 : mssFreqSel_p.resize();
1457 0 : mssFreqSel_p=thisSelection.getChanFreqList(NULL, true);
1458 :
1459 : //cout<<"chansels="<<chansels<<endl;
1460 : //convert the selection into flag
1461 0 : uInt nms = 1;
1462 0 : uInt nrow = chansels.nrow();
1463 0 : dataspectralwindowids_p.resize();
1464 0 : const MSSpWindowColumns spwc(ms_p->spectralWindow());
1465 0 : uInt nspw = spwc.nrow();
1466 0 : const ScalarColumn<Int> spwNchans(spwc.numChan());
1467 0 : Vector<Int> nchanvec = spwNchans.getColumn();
1468 0 : Int maxnchan = 0;
1469 0 : for (uInt i=0;i<nchanvec.nelements();i++) {
1470 0 : maxnchan=max(nchanvec[i],maxnchan);
1471 : }
1472 :
1473 0 : spwchansels_p.resize(nms,nspw,maxnchan);
1474 0 : spwchansels_p.set(0);
1475 0 : uInt nselspw=0;
1476 0 : if (nrow==0) {
1477 : //no channel selection, select all channels
1478 0 : spwchansels_p=1;
1479 0 : dataspectralwindowids_p=thisSelection.getSpwList();
1480 : }
1481 : else {
1482 0 : spwchansels_p=0; //deselect
1483 0 : Int prvspwid=-1;
1484 0 : Vector<Int> selspw;
1485 0 : for (uInt i=0;i<nrow;i++) {
1486 0 : Vector<Int> sel = chansels.row(i);
1487 0 : Int spwid = sel[0];
1488 0 : if((sel[1] >= nchanvec[spwid]) || (sel[2] >=nchanvec[spwid]))
1489 0 : throw(AipsError("Unexpected selection in spw selection of spwid "+String::toString(spwid)));
1490 0 : if (spwid != prvspwid){
1491 0 : nselspw++;
1492 0 : selspw.resize(nselspw,true);
1493 0 : selspw[nselspw-1]=spwid;
1494 : }
1495 0 : uInt minc= sel[1];
1496 0 : uInt maxc = sel[2];
1497 0 : uInt step = sel[3];
1498 : // step as the same context as in im.selectvis
1499 : // select channels
1500 0 : for (uInt k=minc;k<(maxc+1);k+=step) {
1501 0 : spwchansels_p(0,spwid,k)=1;
1502 : }
1503 0 : prvspwid=spwid;
1504 0 : }
1505 0 : dataspectralwindowids_p=selspw;
1506 0 : }
1507 :
1508 : // Map the selected spectral window ids to data description ids
1509 0 : if(dataspectralwindowids_p.nelements()==0){
1510 0 : Int nspwinms=ms_p->spectralWindow().nrow();
1511 0 : dataspectralwindowids_p.resize(nspwinms);
1512 0 : indgen(dataspectralwindowids_p);
1513 : }
1514 0 : MSDataDescIndex msDatIndex(ms_p->dataDescription());
1515 0 : datadescids_p.resize(0);
1516 0 : datadescids_p=msDatIndex.matchSpwId(dataspectralwindowids_p);
1517 :
1518 0 : if (datafieldids_p.nelements() > 1) {
1519 0 : os << LogIO::NORMAL4<< "Multiple fields specified" << LogIO::POST;
1520 0 : multiFields_p = true;
1521 : }
1522 :
1523 :
1524 0 : if(mode=="none"){
1525 : // Now channel selection from spw already stored in chansel,
1526 : // no need for this- TT
1527 : //check if we can find channel selection in the spw string
1528 : //Matrix<Int> chanselmat=thisSelection.getChanList();
1529 : //
1530 : // This not correct for multiple channel ranges TT
1531 : //if(chanselmat.nrow()==dataspectralwindowids_p.nelements()){
1532 0 : if(nselspw==dataspectralwindowids_p.nelements()){
1533 :
1534 0 : dataMode_p="channel";
1535 0 : dataStep_p.resize(dataspectralwindowids_p.nelements());
1536 0 : dataStart_p.resize(dataspectralwindowids_p.nelements());
1537 0 : dataNchan_p.resize(dataspectralwindowids_p.nelements());
1538 0 : Cube<Int> spwchansels_tmp=spwchansels_p;
1539 :
1540 0 : for (uInt k =0 ; k < dataspectralwindowids_p.nelements(); ++k){
1541 0 : uInt curspwid=dataspectralwindowids_p[k];
1542 : //dataStep_p[k]=1;
1543 0 : if (nrow > 0) {
1544 0 : dataStep_p[k]=chansels.row(k)(3);
1545 : }
1546 : else {
1547 0 : dataStep_p[k]=1;
1548 : }
1549 : //dataStart_p[k]=chanselmat.row(k)(1);
1550 0 : dataStart_p[k]=0;
1551 0 : dataNchan_p[k]=nchanvec(curspwid);
1552 : //find start
1553 0 : Bool first =true;
1554 0 : uInt nchn = 0;
1555 0 : uInt lastchan = 0;
1556 0 : for (uInt j=0 ; j < uInt(nchanvec(curspwid)); j++) {
1557 0 : if (spwchansels_p(0,curspwid,j)==1) {
1558 0 : if (first) {
1559 0 : dataStart_p[k]=j;
1560 0 : first = false;
1561 : }
1562 0 : lastchan=j;
1563 0 : nchn++;
1564 : }
1565 : }
1566 0 : dataNchan_p[k]=Int(ceil(Double(lastchan-dataStart_p[k])/Double(dataStep_p[k])))+1;
1567 : //dataNchan_p[k]=Int(ceil(Double(chanselmat.row(k)(2)-dataStart_p[k])/Double(dataStep_p[k])))+1;
1568 :
1569 : //if(dataNchan_p[k]<1)
1570 : // dataNchan_p[k]=1;
1571 :
1572 : //cout<<"modified start="<<dataStart_p[k]<<endl;
1573 : //cout<<"modified nchan="<<dataNchan_p[k]<<endl;
1574 : //
1575 : //Since msselet will be applied to the data before flags from spwchansels_p
1576 : //are applied to the data in FTMachine, shift spwchansels_p by dataStart_p
1577 : //for (uInt j=0 ; j < nchanvec(k)-dataStart_p[k]; j++){
1578 0 : for (uInt j=0 ; j < uInt(nchanvec(curspwid)); j++){
1579 0 : if ( Int(j) < nchanvec(curspwid)-dataStart_p[k]) {
1580 0 : spwchansels_tmp(0,curspwid,j) = spwchansels_p(0,curspwid,j+dataStart_p[k]);
1581 : }
1582 : else {
1583 0 : spwchansels_tmp(0,curspwid,j) = 0;
1584 : }
1585 : }
1586 : }
1587 0 : spwchansels_p = spwchansels_tmp;
1588 0 : }
1589 : }
1590 0 : if(!(exprNode.isNull())){
1591 0 : mssel_p = new MeasurementSet((*ms_p)(exprNode), &* ms_p);
1592 : }
1593 : else{
1594 : // Null take all the ms ...setdata() blank means that
1595 0 : mssel_p = new MeasurementSet(*ms_p);
1596 : }
1597 :
1598 0 : AlwaysAssert(!mssel_p.null(), AipsError);
1599 0 : if(mssel_p->nrow()==0) {
1600 : //delete mssel_p;
1601 0 : mssel_p=0;
1602 : os << (be_calm ? LogIO::NORMAL4 : LogIO::WARN)
1603 : << "Selection is empty: reverting to sorted MeasurementSet"
1604 0 : << LogIO::POST;
1605 0 : mssel_p=new MeasurementSet(*ms_p);
1606 0 : nullSelect_p=true;
1607 : }
1608 : else {
1609 0 : mssel_p->flush();
1610 0 : nullSelect_p=false;
1611 : }
1612 0 : if (nullSelect_p) {
1613 0 : if ((mssel_p->field()).nrow() > 1) {
1614 0 : os << LogIO::NORMAL4 << "Multiple fields selected" << LogIO::POST;
1615 0 : multiFields_p = true;
1616 : } else {
1617 0 : os << LogIO::NORMAL4 << "Single field selected" << LogIO::POST;
1618 0 : multiFields_p = false;
1619 : }
1620 : }
1621 :
1622 0 : uInt nvis_all = ms_p->nrow();
1623 :
1624 : // Now create the VisSet
1625 0 : this->makeVisSet(*mssel_p);
1626 0 : AlwaysAssert(rvi_p, AipsError);
1627 0 : uInt nvis_sel = mssel_p->nrow();
1628 :
1629 0 : if(nvis_sel != nvis_all) {
1630 : os << LogIO::NORMAL // Loglevel INFO
1631 : << "Selected " << nvis_sel << " out of "
1632 : << nvis_all << " rows."
1633 0 : << LogIO::POST;
1634 : }
1635 : else {
1636 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL)
1637 0 : << "Selected all " << nvis_sel << " rows" << LogIO::POST; // Loglevel INFO
1638 : }
1639 : // }
1640 :
1641 : // Tell the user how many channels have been selected.
1642 : // NOTE : This code is replicated in ImagerMultiMS.cc.
1643 0 : Vector<Int> chancounts(dataspectralwindowids_p.nelements());
1644 0 : chancounts=0;
1645 : // if( local_spwstring == "" ) os << "Selected all spws and channels" << LogIO::POST;
1646 : //else os << "Channel selection : " << local_spwstring << LogIO::POST;
1647 0 : os << (be_calm ? LogIO::NORMAL4 : LogIO::NORMAL) << "Selected:";
1648 0 : for(uInt k=0;k<dataspectralwindowids_p.nelements();k++)
1649 : {
1650 0 : for(uInt ch=0;ch<uInt(nchanvec(dataspectralwindowids_p[k]));ch++)
1651 0 : {if(spwchansels_p(0,dataspectralwindowids_p[k],ch)) chancounts[k]++; }
1652 0 : os << " [" << chancounts[k] << " chans in spw " << dataspectralwindowids_p[k] << "]";
1653 : // os << "Selected " << chancounts[k] << " channels in spw "
1654 : // << dataspectralwindowids_p[k] << LogIO::POST;
1655 : }
1656 0 : os << LogIO::POST;
1657 :
1658 : // Now we do a selection to cut down the amount of information
1659 : // passed around.
1660 :
1661 0 : this->selectDataChannel(dataspectralwindowids_p, dataMode_p,
1662 0 : dataNchan_p, dataStart_p, dataStep_p,
1663 0 : mDataStart_p, mDataStep_p);
1664 : ///Tell iterator to use on the fly imaging weights if scratch columns is
1665 : ///not in use
1666 0 : imwgt_p=VisImagingWeight("natural");
1667 0 : rvi_p->useImagingWeight(imwgt_p);
1668 :
1669 : // Guess that the beam is no longer valid
1670 0 : beamValid_p=false;
1671 0 : destroySkyEquation();
1672 0 : if(!valid()){
1673 0 : this->unlock();
1674 : os << LogIO::SEVERE << "Check your data selection or Measurement set "
1675 0 : << LogIO::EXCEPTION;
1676 0 : return false;
1677 : }
1678 0 : this->unlock();
1679 0 : return !nullSelect_p;
1680 0 : } catch (AipsError x) {
1681 0 : this->unlock();
1682 0 : throw(x);
1683 :
1684 : return false;
1685 0 : }
1686 : return !nullSelect_p;
1687 0 : }
1688 :
1689 :
1690 0 : Bool Imager::setmfcontrol(const Float cyclefactor,
1691 : const Float cyclespeedup,
1692 : const Float cyclemaxpsffraction,
1693 : const Int stoplargenegatives,
1694 : const Int stoppointmode,
1695 : const String& scaleType,
1696 : const Float minPB,
1697 : const Float constPB,
1698 : const Vector<String>& fluxscale,
1699 : const Bool flatnoise)
1700 : {
1701 0 : cyclefactor_p = cyclefactor;
1702 0 : cyclespeedup_p = cyclespeedup;
1703 0 : cyclemaxpsffraction_p = cyclemaxpsffraction;
1704 0 : stoplargenegatives_p = stoplargenegatives;
1705 0 : stoppointmode_p = stoppointmode;
1706 0 : fluxscale_p.resize( fluxscale.nelements() );
1707 0 : fluxscale_p = fluxscale;
1708 0 : scaleType_p = scaleType;
1709 0 : minPB_p = minPB;
1710 0 : flatnoise_p=flatnoise;
1711 :
1712 0 : constPB_p = constPB;
1713 0 : return true;
1714 : }
1715 :
1716 :
1717 0 : Bool Imager::setvp(const Bool dovp,
1718 : const Bool doDefaultVPs,
1719 : const String& vpTable,
1720 : const Bool doSquint,
1721 : const Quantity &parAngleInc,
1722 : const Quantity &skyPosThreshold,
1723 : String defaultTel,
1724 : const Bool verbose)
1725 : {
1726 :
1727 0 : LogIO os(LogOrigin("Imager", "setvp()", WHERE));
1728 :
1729 0 : os << LogIO::NORMAL << "Setting voltage pattern parameters" << LogIO::POST; // Loglevel PROGRESS
1730 :
1731 0 : if(!dovp && !vp_p)
1732 0 : delete vp_p;
1733 0 : vp_p=0;
1734 0 : doVP_p=dovp;
1735 0 : doDefaultVP_p = doDefaultVPs;
1736 0 : vpTableStr_p = vpTable;
1737 0 : telescope_p= defaultTel;
1738 0 : if (doSquint) {
1739 0 : squintType_p = BeamSquint::GOFIGURE;
1740 : } else {
1741 0 : squintType_p = BeamSquint::NONE;
1742 : }
1743 :
1744 0 : parAngleInc_p = parAngleInc;
1745 :
1746 0 : skyPosThreshold_p = skyPosThreshold;
1747 : os << (verbose ? LogIO::NORMAL : LogIO::NORMAL3) // Loglevel INFO
1748 : <<"Sky position tolerance is "<<skyPosThreshold_p.getValue("deg")
1749 0 : << " degrees" << LogIO::POST;
1750 :
1751 0 : if (doDefaultVP_p) {
1752 : os << (verbose ? LogIO::NORMAL : LogIO::NORMAL3) // Loglevel INFO
1753 0 : << "Using system default voltage patterns for each telescope" << LogIO::POST;
1754 : } else {
1755 : os << (verbose ? LogIO::NORMAL : LogIO::NORMAL3) // Loglevel INFO
1756 : << "Using user defined voltage patterns in Table "
1757 0 : << vpTableStr_p << LogIO::POST;
1758 : }
1759 0 : if (doSquint) {
1760 : os << (verbose ? LogIO::NORMAL : LogIO::NORMAL3) // Loglevel INFO
1761 0 : << "Beam Squint will be included in the VP model" << LogIO::POST;
1762 : os << (verbose ? LogIO::NORMAL : LogIO::NORMAL3)
1763 : << "and the Parallactic Angle increment is " // Loglevel INFO
1764 0 : << parAngleInc_p.getValue("deg") << " degrees" << LogIO::POST;
1765 : }
1766 :
1767 : // muddled with the state of SkyEquation..so redo it
1768 0 : destroySkyEquation();
1769 0 : return true;
1770 0 : }
1771 :
1772 0 : Bool Imager::setoptions(const String& ftmachine, const Long cache, const Int tile,
1773 : const String& gridfunction, const MPosition& mLocation,
1774 : const Float padding,
1775 : const Int wprojplanes,
1776 : const String& epJTableName,
1777 : const Bool applyPointingOffsets,
1778 : const Bool doPointingCorrection,
1779 : const String& cfCacheDirName,
1780 : const Float& rotPAStep,
1781 : const Float& computePAStep,
1782 : const Float& pbLimit, const String& interpMeth, const Int imageTileVol,
1783 : const Bool singprec,
1784 : const Int numthreads,
1785 : const Bool psTermOn,
1786 : const Bool aTermOn,
1787 : const Bool mTermOn,
1788 : const Bool wbawp,
1789 : const Bool conjBeams)
1790 : {
1791 0 : if(!valid())
1792 : {
1793 0 : return false;
1794 : }
1795 0 : if(!assertDefinedImageParameters())
1796 : {
1797 0 : return false;
1798 : }
1799 0 : LogIO os(LogOrigin("imager", "setoptions()", WHERE));
1800 :
1801 0 : os << LogIO::NORMAL << "Setting processing options" << LogIO::POST; // Loglevel PROGRESS
1802 :
1803 0 : ftmachine_p=downcase(ftmachine);
1804 0 : if(ftmachine_p=="gridft") {
1805 : os << LogIO::WARN
1806 : << "FT machine gridft is now called ft - please use the new name in future"
1807 0 : << LogIO::POST;
1808 0 : ftmachine_p="ft";
1809 : }
1810 :
1811 0 : if(ftmachine_p=="wfmemoryft"){
1812 0 : wfGridding_p=true;
1813 0 : ftmachine_p="ft";
1814 : }
1815 :
1816 0 : wprojPlanes_p=wprojplanes;
1817 0 : epJTableName_p = epJTableName;
1818 0 : cfCacheDirName_p = cfCacheDirName;
1819 0 : rotPAStep_p = rotPAStep;
1820 0 : computePAStep_p = computePAStep;
1821 0 : pbLimit_p = pbLimit;
1822 0 : psTermOn_p=psTermOn;
1823 0 : aTermOn_p=aTermOn;
1824 0 : mTermOn_p=mTermOn;
1825 0 : wbAWP_p=wbawp;
1826 0 : conjBeams_p=conjBeams;
1827 0 : freqInterpMethod_p=interpMeth;
1828 0 : imageTileVol_p=imageTileVol;
1829 0 : if(imageTileVol_p <= 0){
1830 0 : avoidTempLatt_p=true;
1831 0 : imageTileVol_p=-1*imageTileVol_p;
1832 : }
1833 :
1834 :
1835 0 : singlePrec_p=singprec;
1836 :
1837 0 : if(cache>0) cache_p=cache;
1838 0 : if(tile>0) tile_p=tile;
1839 0 : gridfunction_p=downcase(gridfunction);
1840 0 : mLocation_p=mLocation;
1841 0 : if(padding>=1.0) {
1842 0 : padding_p=padding;
1843 : }
1844 :
1845 : // Check if gridfunction is set to gauss or gjinc for non single dish imaging
1846 0 : if (ftmachine_p!="sd"
1847 0 : && (gridfunction_p=="gauss" || gridfunction_p=="gjinc")) {
1848 : os << LogIO::SEVERE
1849 0 : << "Grid function " << gridfunction << " is available only for single dish imaging" << LogIO::EXCEPTION;
1850 : }
1851 :
1852 : // Destroy the FTMachine
1853 0 : if(ft_p) {delete ft_p; ft_p=0;}
1854 0 : if(gvp_p) {delete gvp_p; gvp_p=0;}
1855 0 : if(cft_p) {delete cft_p; cft_p=0;}
1856 :
1857 0 : doPointing = applyPointingOffsets;
1858 0 : doPBCorr = doPointingCorrection;
1859 : ////The set below does not seemed to be remembered later in the process it
1860 : /// under some compiler version so setting a private variable to be used
1861 : /// a negative number means use all that is available
1862 0 : numthreads_p= numthreads;
1863 : #ifdef _OPENMP
1864 0 : if(numthreads > 0){
1865 0 : if(numthreads <= omp_get_max_threads()){
1866 0 : omp_set_num_threads(numthreads);
1867 : }
1868 : }
1869 : #endif
1870 0 : return true;
1871 0 : }
1872 :
1873 0 : Bool Imager::setsdoptions(const Float scale, const Float weight,
1874 : const Int convsupport, String pointCol,
1875 : const Quantity truncate,
1876 : const Quantity gwidth, const Quantity jwidth,
1877 : const Float minweight, const Bool clipminmax,
1878 : const Bool enablecache,
1879 : const String & convertfirst)
1880 : {
1881 0 : LogIO os(LogOrigin("imager", "setsdoptions()", WHERE));
1882 :
1883 0 : os << LogIO::NORMAL << "Setting single dish processing options" << LogIO::POST; // Loglevel PROGRESS
1884 :
1885 0 : sdScale_p=scale;
1886 0 : sdWeight_p=weight;
1887 :
1888 :
1889 0 : sdConvSupport_p=convsupport;
1890 0 : pointingDirCol_p=pointCol;
1891 0 : pointingDirCol_p.upcase();
1892 0 : if( (pointingDirCol_p != "DIRECTION") &&(pointingDirCol_p != "TARGET") && (pointingDirCol_p != "ENCODER") && (pointingDirCol_p != "POINTING_OFFSET") && (pointingDirCol_p != "SOURCE_OFFSET")){
1893 : os << LogIO::SEVERE
1894 0 : << "No such direction column as "<< pointingDirCol_p
1895 0 : << " in pointing table "<< LogIO::EXCEPTION;
1896 : }
1897 0 : qtruncate_p=truncate;
1898 0 : qgwidth_p=gwidth;
1899 0 : qjwidth_p=jwidth;
1900 0 : minWeight_p = minweight;
1901 0 : clipminmax_p = clipminmax;
1902 0 : enablecache_p = enablecache;
1903 0 : convertfirst_p = convertfirst;
1904 0 : convertfirst_p.upcase();
1905 :
1906 :
1907 : // Destroy the FTMachine
1908 0 : if(ft_p) {delete ft_p; ft_p=0;}
1909 0 : if(gvp_p) {delete gvp_p; gvp_p=0;}
1910 0 : if(cft_p) {delete cft_p; cft_p=0;}
1911 :
1912 0 : return true;
1913 0 : }
1914 :
1915 0 : Bool Imager::mask(const String& mask, const String& image,
1916 : const Quantity& threshold)
1917 : {
1918 : //if(!valid()) return false;
1919 0 : LogIO os(LogOrigin("imager", "mask()", WHERE));
1920 : //if(!assertDefinedImageParameters()) return false;
1921 :
1922 : try {
1923 : //this->lock();
1924 0 : if(image=="") {
1925 : //this->unlock();
1926 0 : os << LogIO::SEVERE << "Need name for template image" << LogIO::EXCEPTION;
1927 0 : return false;
1928 : }
1929 0 : String maskName(mask);
1930 0 : if(maskName=="") {
1931 0 : maskName=image+".mask";
1932 : }
1933 0 : if(!clone(image, maskName)) return false;
1934 0 : PagedImage<Float> maskImage(maskName);
1935 0 : maskImage.table().markForDelete();
1936 0 : PagedImage<Float> imageImage(image);
1937 :
1938 0 : if(threshold.check(UnitVal(1.0, "Jy"))){
1939 : os << LogIO::NORMAL // Loglevel INFO
1940 : << "Making mask image " << maskName << ", applying threshold "
1941 0 : << threshold.get("Jy").getValue() << "Jy, " << endl
1942 0 : << "to template image " << image << LogIO::POST;
1943 :
1944 0 : StokesImageUtil::MaskFrom(maskImage, imageImage, threshold);
1945 : }
1946 : else{
1947 : os << LogIO::NORMAL // Loglevel INFO
1948 : << "Making mask image " << maskName << ", applying threshold "
1949 0 : << threshold.getValue() << " " << threshold.getUnit() << endl
1950 0 : << "to template image " << image << LogIO::POST;
1951 :
1952 0 : StokesImageUtil::MaskFrom(maskImage, imageImage, threshold.getValue());
1953 : }
1954 0 : maskImage.table().unmarkForDelete();
1955 :
1956 : //this->lock();
1957 0 : return true;
1958 0 : } catch (AipsError x) {
1959 : //this->unlock();
1960 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
1961 0 : << LogIO::EXCEPTION;
1962 0 : return false;
1963 0 : }
1964 : //this->unlock();
1965 : return true;
1966 0 : }
1967 :
1968 0 : Bool Imager::boxmask(const String& mask, const Vector<Int>& blc,
1969 : const Vector<Int>& trc, const Float value)
1970 : {
1971 0 : if(!valid()) return false;
1972 :
1973 0 : LogIO os(LogOrigin("imager", "boxmask()", WHERE));
1974 :
1975 : try {
1976 :
1977 0 : if(!assertDefinedImageParameters()) return false;
1978 :
1979 0 : if(mask=="") {
1980 0 : os << LogIO::SEVERE << "Need name for mask image" << LogIO::EXCEPTION;
1981 0 : return false;
1982 : }
1983 0 : if(!Table::isWritable(mask)) {
1984 0 : make(mask);
1985 0 : this->lock();
1986 : }
1987 0 : PagedImage<Float> maskImage(mask);
1988 0 : maskImage.table().markForDelete();
1989 :
1990 :
1991 0 : IPosition iblc(blc);
1992 0 : IPosition itrc(trc);
1993 0 : IPosition iinc(iblc.nelements(), 1);
1994 0 : LCBox::verify(iblc, itrc, iinc, maskImage.shape());
1995 :
1996 : os << LogIO::DEBUG1
1997 : << "Setting '" << mask << "' blc=" << iblc
1998 0 : << " trc=" << itrc << " to " << value << LogIO::POST;
1999 :
2000 0 : StokesImageUtil::BoxMask(maskImage, iblc, itrc, value);
2001 :
2002 0 : maskImage.table().unmarkForDelete();
2003 :
2004 0 : this->unlock();
2005 0 : return true;
2006 0 : } catch (AipsError x) {
2007 0 : this->unlock();
2008 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
2009 0 : << LogIO::EXCEPTION;
2010 0 : return false;
2011 0 : }
2012 : return true;
2013 0 : }
2014 :
2015 0 : Bool Imager::regionmask(const String& maskimage, Record* imageRegRec,
2016 : Matrix<Quantity>& blctrcs, Matrix<Float>& circles,
2017 : const Float& value){
2018 :
2019 : //This function does not modify ms(s) so no need of lock
2020 0 : LogIO os(LogOrigin("imager", "regionmask()", WHERE));
2021 0 : if(!Table::isWritable(maskimage)) {
2022 0 : make(maskimage);
2023 : }
2024 0 : return Imager::regionToImageMask(maskimage, imageRegRec, blctrcs, circles, value);
2025 :
2026 0 : }
2027 :
2028 0 : Bool Imager::regionToImageMask(const String& maskimage, Record* imageRegRec, Matrix<Quantity>& blctrcs, Matrix<Float>& circles, const Float& value){
2029 0 : PagedImage<Float> maskImage(maskimage);
2030 0 : CoordinateSystem cSys=maskImage.coordinates();
2031 0 : maskImage.table().markForDelete();
2032 0 : ImageRegion *boxregions=0;
2033 0 : ImageRegion *circleregions=0;
2034 0 : RegionManager regMan;
2035 0 : regMan.setcoordsys(cSys);
2036 0 : Vector<Quantum<Double> > blc(2);
2037 0 : Vector<Quantum<Double> > trc(2);
2038 0 : if(blctrcs.nelements() !=0){
2039 0 : if(blctrcs.shape()(1) != 4)
2040 0 : throw(AipsError("Need a list of 4 elements to define a box"));
2041 0 : Int nrow=blctrcs.shape()(0);
2042 0 : Vector<Int> absRel(2, RegionType::Abs);
2043 0 : PtrBlock<const WCRegion *> lesbox(nrow);
2044 0 : for (Int k=0; k < nrow; ++k){
2045 0 : blc(0) = blctrcs(k,0);
2046 0 : blc(1) = blctrcs(k,1);
2047 0 : trc(0) = blctrcs(k,2);
2048 0 : trc(1) = blctrcs(k,3);
2049 0 : lesbox[k]= new WCBox (blc, trc, cSys, absRel);
2050 : }
2051 0 : boxregions=regMan.doUnion(lesbox);
2052 0 : for (Int k=0; k < nrow; ++k){
2053 0 : delete lesbox[k];
2054 : }
2055 0 : }
2056 0 : if((circles.nelements()) > 0){
2057 0 : if(circles.shape()(1) != 3)
2058 0 : throw(AipsError("Need a list of 3 elements to define a circle"));
2059 0 : Int nrow=circles.shape()(0);
2060 0 : Vector<Float> cent(2);
2061 0 : cent(0)=circles(0,1); cent(1)=circles(0,2);
2062 0 : Float radius=circles(0,0);
2063 0 : IPosition xyshape(2,maskImage.shape()(0),maskImage.shape()(1));
2064 0 : LCEllipsoid *circ= new LCEllipsoid(cent, radius, xyshape);
2065 : //Tell LCUnion to delete the pointers
2066 0 : LCUnion *elunion= new LCUnion(true, circ);
2067 : //now lets do the remainder
2068 0 : for (Int k=1; k < nrow; ++k){
2069 0 : cent(0)=circles(k,1); cent(1)=circles(k,2);
2070 0 : radius=circles(k,0);
2071 0 : circ= new LCEllipsoid(cent, radius, xyshape);
2072 0 : elunion=new LCUnion(true, elunion, circ);
2073 : }
2074 : //now lets extend that to the whole image
2075 0 : IPosition trc(2);
2076 0 : trc(0)=maskImage.shape()(2)-1;
2077 0 : trc(1)=maskImage.shape()(3)-1;
2078 0 : LCBox lbox(IPosition(2,0,0), trc,
2079 0 : IPosition(2,maskImage.shape()(2),maskImage.shape()(3)) );
2080 0 : LCExtension linter(*elunion, IPosition(2,2,3),lbox);
2081 0 : circleregions=new ImageRegion(linter);
2082 0 : delete elunion;
2083 0 : }
2084 :
2085 :
2086 0 : ImageRegion* recordRegion=0;
2087 0 : if(imageRegRec !=0){
2088 0 : TableRecord rec1;
2089 0 : rec1.assign(*imageRegRec);
2090 0 : recordRegion=ImageRegion::fromRecord(rec1,"");
2091 0 : }
2092 0 : ImageRegion *unionReg=0;
2093 0 : if(boxregions!=0 && recordRegion!=0){
2094 0 : unionReg=regMan.doUnion(*boxregions, *recordRegion);
2095 0 : delete boxregions; boxregions=0;
2096 0 : delete recordRegion; recordRegion=0;
2097 : }
2098 0 : else if(boxregions !=0){
2099 0 : unionReg=boxregions;
2100 : }
2101 0 : else if(recordRegion !=0){
2102 0 : unionReg=recordRegion;
2103 : }
2104 :
2105 :
2106 :
2107 :
2108 0 : if(unionReg !=0){
2109 0 : regionToMask(maskImage, *unionReg, value);
2110 0 : delete unionReg; unionReg=0;
2111 : }
2112 : //As i can't unionize LCRegions and WCRegions; do circles seperately
2113 0 : if(circleregions !=0){
2114 0 : regionToMask(maskImage, *circleregions, value);
2115 0 : delete circleregions;
2116 0 : circleregions=0;
2117 : }
2118 0 : maskImage.table().unmarkForDelete();
2119 0 : return true;
2120 :
2121 0 : }
2122 :
2123 :
2124 0 : Bool Imager::regionToMask(ImageInterface<Float>& maskImage, ImageRegion& imagreg, const Float& value){
2125 :
2126 :
2127 0 : SubImage<Float> partToMask(maskImage, imagreg, true);
2128 0 : LatticeRegion latReg=imagreg.toLatticeRegion(maskImage.coordinates(), maskImage.shape());
2129 0 : ArrayLattice<Bool> pixmask(latReg.get());
2130 0 : LatticeExpr<Float> myexpr(iif(pixmask, value, partToMask) );
2131 0 : partToMask.copyData(myexpr);
2132 :
2133 0 : return true;
2134 0 : }
2135 :
2136 :
2137 0 : Bool Imager::clipimage(const String& image, const Quantity& threshold)
2138 : {
2139 0 : if(!valid()) return false;
2140 :
2141 0 : LogIO os(LogOrigin("imager", "clipimage()", WHERE));
2142 :
2143 0 : this->lock();
2144 : try {
2145 :
2146 0 : if(!assertDefinedImageParameters()) return false;
2147 :
2148 0 : if(image=="") {
2149 0 : this->unlock();
2150 0 : os << LogIO::SEVERE << "Need name for image" << LogIO::EXCEPTION;
2151 0 : return false;
2152 : }
2153 0 : PagedImage<Float> imageImage(image);
2154 : os << LogIO::NORMAL // Loglevel PROGRESS
2155 : << "Zeroing " << image << ", for all pixels where Stokes I < threshold "
2156 0 : << threshold.get("Jy").getValue() << "Jy " << LogIO::POST;
2157 :
2158 0 : StokesImageUtil::MaskOnStokesI(imageImage, threshold);
2159 0 : this->unlock();
2160 0 : return true;
2161 0 : } catch (AipsError x) {
2162 0 : this->unlock();
2163 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
2164 0 : << LogIO::EXCEPTION;
2165 0 : return false;
2166 0 : }
2167 :
2168 : return true;
2169 0 : }
2170 :
2171 : // Add together low and high resolution images in the Fourier plane
2172 0 : Bool Imager::feather(const String& image, const String& highRes,
2173 : const String& lowRes, const String& lowPSF, const Float effDishDiam, const Bool lowPassFilterSD)
2174 : {
2175 :
2176 0 : Float effDiam=effDishDiam;
2177 0 : LoggerHolder lh (false);
2178 0 : LogIO os = lh.logio();
2179 0 : os << LogOrigin("imager", "feather()");
2180 :
2181 : try {
2182 0 : Bool noStokes=false;
2183 0 : String outLowRes=lowRes;
2184 0 : String outHighRes=highRes;
2185 : {
2186 0 : if ( ! doVP_p ) {
2187 0 : this->unlock();
2188 : os << LogIO::SEVERE <<
2189 : "Must invoke setvp() first in order to apply the primary beam"
2190 0 : << LogIO::EXCEPTION;
2191 0 : return false;
2192 : }
2193 :
2194 : os << LogIO::NORMAL // Loglevel PROGRESS
2195 0 : << "\nFeathering together high and low resolution images.\n" << LogIO::POST;
2196 :
2197 :
2198 : // Get initial images
2199 : /*{ //Drat lets deal with images that don't have stokes.
2200 : PagedImage<Float> hightemp(highRes);
2201 : PagedImage<Float> lowtemp(lowRes);
2202 : if(hightemp.shape().nelements() != lowtemp.shape().nelements()){
2203 : this->unlock();
2204 : os << LogIO::SEVERE <<
2205 : "High res. image and low res. image donot have same number of axes"
2206 : << LogIO::EXCEPTION;
2207 : return false;
2208 :
2209 : }
2210 : if ( (hightemp.coordinates().findCoordinate(Coordinate::STOKES) < 0) &&
2211 : (lowtemp.coordinates().findCoordinate(Coordinate::STOKES) < 0)){
2212 : noStokes=true;
2213 : os << LogIO::NORMAL // Loglevel PROGRESS
2214 : << "Making some temporary images as the inputs have no Stokes axis.\n"
2215 : << LogIO::POST;
2216 : std::unique_ptr<ImageInterface<Float> > outImage1;
2217 : outHighRes= highRes+"_stokes";
2218 : ImageUtilities::addDegenerateAxes (os, outImage1, hightemp, outHighRes,
2219 : false, false,
2220 : "I", false, false,
2221 : false);
2222 :
2223 : std::unique_ptr<ImageInterface<Float> > outImage2;
2224 : outLowRes= lowRes+"_stokes";
2225 : ImageUtilities::addDegenerateAxes (os, outImage2, lowtemp, outLowRes,
2226 : false, false,
2227 : "I", false, false,
2228 : false);
2229 :
2230 : }
2231 : }*/
2232 0 : PagedImage<Float> high(outHighRes);
2233 0 : PagedImage<Float> low0(outLowRes);
2234 :
2235 0 : Feather::feather(image, high, low0, sdScale_p, lowPSF, doDefaultVP_p, vpTableStr_p, effDiam, lowPassFilterSD);
2236 0 : }
2237 :
2238 0 : if(noStokes){
2239 0 : TableUtil::deleteTable(outHighRes);
2240 0 : TableUtil::deleteTable(outLowRes);
2241 : }
2242 0 : return true;
2243 0 : } catch (AipsError x) {
2244 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
2245 0 : << LogIO::EXCEPTION;
2246 0 : return false;
2247 0 : }
2248 :
2249 : return true;
2250 0 : }
2251 :
2252 :
2253 :
2254 0 : Bool Imager::linearmosaic(const String& mosaic,
2255 : const String& fluxscale,
2256 : const String& sensitivity,
2257 : const Vector<String>& images,
2258 : const Vector<Int>& fieldids)
2259 :
2260 : {
2261 0 : if(!valid()) return false;
2262 0 : LogIO os(LogOrigin("imager", "linearmosaic()", WHERE));
2263 : try{
2264 0 : if(mosaic=="") {
2265 0 : os << LogIO::SEVERE << "Need name for mosaic image" << LogIO::POST;
2266 0 : return false;
2267 : }
2268 0 : if(!Table::isWritable( mosaic )) {
2269 0 : make( mosaic );
2270 : }
2271 0 : if (images.nelements() == 0) {
2272 0 : os << LogIO::SEVERE << "Need names of images to mosaic" << LogIO::POST;
2273 0 : return false;
2274 : }
2275 0 : if (images.nelements() != fieldids.nelements()) {
2276 : os << LogIO::SEVERE << "number of fieldids doesn\'t match the"
2277 0 : << " number of images" << LogIO::POST;
2278 0 : return false;
2279 : }
2280 :
2281 0 : Double meminMB=Double(HostInfo::memoryTotal(true))/1024.0;
2282 0 : PagedImage<Float> mosaicImage( mosaic );
2283 0 : CoordinateSystem cs=mosaicImage.coordinates();
2284 0 : String err;
2285 : //for some reason subimages below fail if they are in some frames like BARY
2286 0 : if(cs.setSpectralConversion(err, "LSRK")){
2287 0 : mosaicImage.setCoordinateInfo(cs);
2288 : }
2289 0 : mosaicImage.set(0.0);
2290 0 : TempImage<Float> numerator( TiledShape(mosaicImage.shape(), mosaicImage.niceCursorShape()), mosaicImage.coordinates(), meminMB/2.0);
2291 0 : numerator.set(0.0);
2292 0 : TempImage<Float> denominator( TiledShape(mosaicImage.shape(), mosaicImage.niceCursorShape()), mosaicImage.coordinates(), meminMB/2.0);
2293 0 : denominator.set(0.0);
2294 0 : ImageRegrid<Float> regridder;
2295 :
2296 0 : MSColumns msc(*ms_p);
2297 0 : for (uInt i=0; i < images.nelements(); ++i) {
2298 0 : if(!Table::isReadable(images(i))) {
2299 : os << LogIO::SEVERE << "Image " << images(i) <<
2300 0 : " is not readable" << LogIO::POST;
2301 0 : return false;
2302 : }
2303 :
2304 0 : PagedImage<Float> smallImagedisk( images(i) );
2305 0 : cs=smallImagedisk.coordinates();
2306 : //for some reason subimages below fail if they are in some frames like BARY
2307 0 : if(!cs.setSpectralConversion(err, "LSRK")){
2308 0 : cs=smallImagedisk.coordinates();
2309 : }
2310 :
2311 0 : TempImage<Float> smallImage(smallImagedisk.shape(), cs, meminMB/8.0);
2312 0 : smallImage.copyData(smallImagedisk);
2313 0 : IPosition iblc(smallImage.shape().nelements(),0);
2314 0 : IPosition itrc(smallImage.shape());
2315 0 : itrc=itrc-Int(1);
2316 :
2317 0 : LCBox lbox(iblc, itrc, smallImage.shape());
2318 0 : ImageRegion imagreg(WCBox(lbox, cs) );
2319 : try{
2320 : // accumulate the images
2321 0 : SubImage<Float> subNum;
2322 0 : SubImage<Float> subDen;
2323 : try{
2324 0 : subNum=SubImage<Float>(numerator, imagreg, true);
2325 0 : subDen=SubImage<Float>(denominator, imagreg, true);
2326 : }
2327 0 : catch(...){
2328 : //Failed to make a subimage let us use the full image
2329 0 : subNum=SubImage<Float>(numerator, true);
2330 0 : subDen=SubImage<Float>(denominator, true);
2331 :
2332 0 : }
2333 :
2334 :
2335 :
2336 0 : TempImage<Float> fullImage(subNum.shape(), subNum.coordinates(), meminMB/8.0);
2337 :
2338 0 : os << "Processing Image " << images(i) << LogIO::POST;
2339 :
2340 0 : regridder.regrid( fullImage, Interpolate2D::LINEAR,
2341 0 : IPosition(2,0,1), smallImage );
2342 :
2343 0 : TempImage<Float> PB( subNum.shape(), subNum.coordinates(), meminMB/8.0);
2344 0 : PB.set(1.0);
2345 :
2346 0 : MDirection pointingDirection = msc.field().phaseDirMeas( fieldids(i) );
2347 :
2348 0 : Quantity pa(0.0, "deg");
2349 0 : pbguts ( PB, PB, pointingDirection, pa);
2350 :
2351 0 : fullImage.copyData( (LatticeExpr<Float>) (fullImage * PB ) );
2352 0 : subNum.copyData( (LatticeExpr<Float>) (subNum + fullImage) );
2353 0 : subDen.copyData( (LatticeExpr<Float>) (subDen + (PB*PB)) );
2354 :
2355 0 : }
2356 0 : catch (AipsError x) {
2357 : os << LogIO::WARN<< "Caught exception while processing " << images(i)
2358 : << "\n"<< x.getMesg()
2359 0 : << LogIO::POST;
2360 0 : continue;
2361 0 : }
2362 0 : catch(...){
2363 0 : os << LogIO::WARN << "Unknown error processing " << images(i) << LogIO::POST;
2364 0 : continue;
2365 0 : }
2366 0 : }
2367 :
2368 0 : LatticeExprNode LEN = max( denominator );
2369 0 : Float dMax = LEN.getFloat();
2370 :
2371 :
2372 0 : if (scaleType_p == "SAULT") {
2373 :
2374 : // truncate denominator at ggSMin1
2375 0 : denominator.copyData( (LatticeExpr<Float>)
2376 0 : (iif(denominator < (dMax * constPB_p), dMax,
2377 : denominator) ) );
2378 :
2379 0 : if (fluxscale != "") {
2380 0 : clone( mosaic, fluxscale );
2381 :
2382 0 : PagedImage<Float> fluxscaleImage( fluxscale );
2383 0 : fluxscaleImage.copyData( (LatticeExpr<Float>)
2384 0 : (iif(denominator < (dMax*minPB_p), 0.0,
2385 0 : (dMax*minPB_p)/(denominator) )) );
2386 0 : fluxscaleImage.copyData( (LatticeExpr<Float>)
2387 0 : (iif(denominator > (dMax*constPB_p), 1.0,
2388 : (fluxscaleImage) )) );
2389 0 : mosaicImage.copyData( (LatticeExpr<Float>)(iif(denominator > (dMax*minPB_p),
2390 0 : (numerator/denominator), 0)) );
2391 0 : }
2392 : } else {
2393 0 : mosaicImage.copyData( (LatticeExpr<Float>)(iif(denominator > (dMax*minPB_p),
2394 0 : (numerator/denominator), 0)) );
2395 0 : if (fluxscale != "") {
2396 0 : clone(mosaic, fluxscale );
2397 0 : PagedImage<Float> fluxscaleImage( fluxscale );
2398 0 : fluxscaleImage.copyData( (LatticeExpr<Float>)( 1.0 ) );
2399 0 : }
2400 : }
2401 0 : if (sensitivity != "") {
2402 0 : clone(mosaic, sensitivity);
2403 0 : PagedImage<Float> sensitivityImage( sensitivity );
2404 0 : sensitivityImage.copyData( (LatticeExpr<Float>)( denominator/dMax ));
2405 0 : }
2406 0 : }
2407 0 : catch (AipsError x) {
2408 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
2409 0 : << LogIO::POST;
2410 0 : return false;
2411 0 : }
2412 0 : return true;
2413 0 : }
2414 :
2415 : // Weight the MeasurementSet
2416 0 : Bool Imager::weight(const String& type, const String& crmode,
2417 : const Quantity& noise, const Double robust,
2418 : const Quantity& fieldofview,
2419 : const Int npixels, const Bool multiField)
2420 : {
2421 0 : if(!valid()) return false;
2422 0 : logSink_p.clearLocally();
2423 0 : LogIO os(LogOrigin("imager", "weight()"),logSink_p);
2424 :
2425 0 : this->lock();
2426 : try {
2427 :
2428 0 : String rmode=crmode; // can change it
2429 :
2430 :
2431 : os << LogIO::NORMAL // Loglevel INFO
2432 0 : << "Weighting MS: Imaging weights will be changed" << LogIO::POST;
2433 :
2434 0 : if (type=="natural") {
2435 : os << LogIO::NORMAL // Loglevel INFO
2436 0 : << "Natural weighting" << LogIO::POST;
2437 0 : imwgt_p=VisImagingWeight("natural");
2438 : }
2439 0 : else if(type=="superuniform"){
2440 0 : if(!assertDefinedImageParameters()) return false;
2441 : ///making usage of npixels consistent with uniform, briggs
2442 : /// don't know why this was done seperately which is kind of redundant in the code
2443 : /// one achieves superuniform with just uniform with npixels or fieldofview
2444 : /// set to non-defaults in the section below
2445 0 : Int actualNpix=npixels/2;
2446 0 : if(actualNpix <=0)
2447 0 : actualNpix=3;
2448 : os << LogIO::NORMAL // Loglevel INFO
2449 : << "SuperUniform weighting over a square cell spanning ["
2450 : << -actualNpix
2451 0 : << ", " << actualNpix << "] in the uv plane" << LogIO::POST;
2452 0 : imwgt_p=VisImagingWeight(*rvi_p, rmode, noise, robust, nx_p,
2453 0 : ny_p, mcellx_p, mcelly_p, actualNpix,
2454 0 : actualNpix, multiField);
2455 : }
2456 0 : else if ((type=="robust")||(type=="uniform")||(type=="briggs")) {
2457 0 : if(!assertDefinedImageParameters()) return false;
2458 0 : Quantity actualFieldOfView(fieldofview);
2459 0 : Int actualNPixels(npixels);
2460 0 : String wtype;
2461 0 : if(type=="briggs") {
2462 : //The user really meant to use Brigg's weighting and forgot to set norm or abs
2463 : // guessing it should be norm
2464 0 : if(rmode=="none")
2465 0 : rmode="norm";
2466 0 : wtype = "Briggs";
2467 : }
2468 : else {
2469 0 : wtype = "Uniform";
2470 : }
2471 0 : if(actualFieldOfView.get().getValue()==0.0&&actualNPixels==0) {
2472 0 : actualNPixels=nx_p;
2473 0 : actualFieldOfView=Quantity(actualNPixels*mcellx_p.get("rad").getValue(),
2474 0 : "rad");
2475 : os << LogIO::NORMAL // Loglevel INFO
2476 : << wtype
2477 : << " weighting: sidelobes will be suppressed over full image"
2478 0 : << LogIO::POST;
2479 : }
2480 0 : else if(actualFieldOfView.get().getValue()>0.0&&actualNPixels==0) {
2481 0 : actualNPixels=Int(actualFieldOfView.get("rad").getValue()/mcellx_p.get("rad").getValue());
2482 : os << LogIO::NORMAL // Loglevel INFO
2483 : << wtype
2484 : << " weighting: sidelobes will be suppressed over specified field of view: "
2485 0 : << actualFieldOfView.get("arcsec").getValue() << " arcsec" << LogIO::POST;
2486 : }
2487 0 : else if(actualFieldOfView.get().getValue()==0.0&&actualNPixels>0) {
2488 0 : actualFieldOfView=Quantity(actualNPixels*mcellx_p.get("rad").getValue(),
2489 0 : "rad");
2490 : os << LogIO::NORMAL // Loglevel INFO
2491 : << wtype
2492 : << " weighting: sidelobes will be suppressed over full image field of view: "
2493 0 : << actualFieldOfView.get("arcsec").getValue() << " arcsec" << LogIO::POST;
2494 : }
2495 : else {
2496 : os << LogIO::NORMAL // Loglevel INFO
2497 : << wtype
2498 : << " weighting: sidelobes will be suppressed over specified field of view: "
2499 0 : << actualFieldOfView.get("arcsec").getValue() << " arcsec" << LogIO::POST;
2500 : }
2501 : os << LogIO::DEBUG1
2502 : << "Weighting used " << actualNPixels << " uv pixels."
2503 0 : << LogIO::POST;
2504 0 : Quantity actualCellSize(actualFieldOfView.get("rad").getValue()/actualNPixels, "rad");
2505 :
2506 0 : imwgt_p=VisImagingWeight(*rvi_p, rmode, noise, robust,
2507 : actualNPixels, actualNPixels, actualCellSize,
2508 0 : actualCellSize, 0, 0, multiField);
2509 :
2510 0 : }
2511 0 : else if (type=="radial") {
2512 0 : os << "Radial weighting" << LogIO::POST;
2513 0 : imwgt_p=VisImagingWeight("radial");
2514 : }
2515 : else {
2516 0 : this->unlock();
2517 : os << LogIO::SEVERE << "Unknown weighting " << type
2518 0 : << LogIO::EXCEPTION;
2519 0 : return false;
2520 : }
2521 :
2522 0 : rvi_p->useImagingWeight(imwgt_p);
2523 :
2524 : // Beam is no longer valid
2525 0 : beamValid_p=false;
2526 0 : destroySkyEquation();
2527 0 : this->writeHistory(os);
2528 0 : this->unlock();
2529 0 : return true;
2530 0 : } catch (AipsError x) {
2531 0 : this->unlock();
2532 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
2533 0 : << LogIO::EXCEPTION;
2534 0 : return false;
2535 0 : }
2536 :
2537 : return true;
2538 0 : }
2539 :
2540 :
2541 0 : Bool Imager::getWeightGrid(Block<Matrix<Float> >&weightgrid, const String& type, const Vector<String>&imagenames){
2542 :
2543 0 : if(type=="imaging"){
2544 0 : weightgrid.resize(0, true, false);
2545 0 : if(imwgt_p.getType()!="uniform")
2546 0 : return false;
2547 0 : imwgt_p.getWeightDensity(weightgrid);
2548 0 : return true;
2549 : }
2550 0 : if((type=="ftweight") && (sm_p) && (Int(imagenames.nelements())== sm_p->numberOfModels())){
2551 0 : for (Int model=0; model < sm_p->numberOfModels(); ++model){
2552 0 : PagedImage<Float> wgtImage(sm_p->image(model).shape(),
2553 0 : (sm_p->image(model)).coordinates(),
2554 0 : imagenames(model));
2555 0 : se_p->getWeightImage(model, wgtImage);
2556 :
2557 :
2558 0 : }
2559 0 : return true;
2560 : }
2561 :
2562 0 : return false;
2563 : }
2564 :
2565 0 : Bool Imager::setWeightGrid(const Block<Matrix<Float> >& weightgrid, const String& type){
2566 :
2567 0 : if(type=="imaging"){
2568 0 : if(imwgt_p.getType()!="uniform")
2569 0 : return false;
2570 0 : imwgt_p.setWeightDensity(weightgrid);
2571 0 : rvi_p->useImagingWeight(imwgt_p);
2572 : }
2573 :
2574 0 : return true;
2575 : }
2576 :
2577 : // Filter the MeasurementSet
2578 0 : Bool Imager::filter(const String& type, const Quantity& bmaj,
2579 : const Quantity& bmin, const Quantity& bpa)
2580 : {
2581 0 : if(!valid()) return false;
2582 0 : logSink_p.clearLocally();
2583 0 : LogIO os(LogOrigin("imager", "filter()"),logSink_p);
2584 :
2585 0 : this->lock();
2586 : try {
2587 :
2588 : os << LogIO::NORMAL // Loglevel INFO
2589 0 : << "Imaging weights will be tapered" << LogIO::POST;
2590 0 : imwgt_p.setFilter(type, bmaj, bmin, bpa);
2591 0 : rvi_p->useImagingWeight(imwgt_p);
2592 :
2593 : // Beam is no longer valid
2594 0 : beamValid_p=false;
2595 0 : destroySkyEquation();
2596 0 : this->writeHistory(os);
2597 0 : this->unlock();
2598 0 : return true;
2599 0 : } catch (AipsError x) {
2600 0 : this->unlock();
2601 0 : throw(x);
2602 : return false;
2603 0 : }
2604 :
2605 : return true;
2606 0 : }
2607 :
2608 :
2609 : // Implement a uv range
2610 0 : Bool Imager::uvrange(const Double& uvmin, const Double& uvmax)
2611 : {
2612 0 : if(!valid()) return false;
2613 0 : logSink_p.clearLocally();
2614 0 : LogIO os(LogOrigin("imager", "uvrange()"),logSink_p);
2615 :
2616 : try {
2617 : os << LogIO::NORMAL // Loglevel INFO
2618 : << "Selecting data according to uvrange: setdata will reset this selection"
2619 0 : << LogIO::POST;
2620 :
2621 0 : Double auvmin(uvmin);
2622 0 : Double auvmax(uvmax);
2623 :
2624 0 : if(auvmax<=0.0) auvmax=1e10;
2625 0 : if(auvmax>auvmin&&(auvmin>=0.0)) {
2626 : os << LogIO::NORMAL // Loglevel INFO
2627 : << "Allowed uv range: " << auvmin << " to " << auvmax
2628 0 : << " wavelengths" << LogIO::POST;
2629 : }
2630 : else {
2631 : os << LogIO::SEVERE << "Invalid uvmin and uvmax: "
2632 : << auvmin << ", " << auvmax
2633 0 : << LogIO::EXCEPTION;
2634 0 : return false;
2635 : }
2636 0 : Vector<Double> freq;
2637 0 : ostringstream strUVmax, strUVmin, ostrInvLambda;
2638 :
2639 0 : this->lock();
2640 :
2641 0 : if(mssel_p.null()){ os << "Please setdata first before using uvrange " << LogIO::POST; return false; }
2642 :
2643 :
2644 : // use the average wavelength for the selected windows to convert
2645 : // uv-distance from lambda to meters
2646 0 : ostringstream spwsel;
2647 0 : spwsel << "select from $1 where ROWID() IN [";
2648 0 : for(uInt i=0; i < dataspectralwindowids_p.nelements(); ++i) {
2649 0 : if (i > 0) spwsel << ", ";
2650 0 : spwsel << dataspectralwindowids_p(i);
2651 : }
2652 0 : spwsel << "]";
2653 :
2654 0 : MSSpectralWindow msspw(tableCommand(spwsel.str(),
2655 0 : mssel_p->spectralWindow()).table());
2656 0 : MSSpWindowColumns spwc(msspw);
2657 :
2658 : // This averaging scheme will work even if the spectral windows are
2659 : // of different sizes. Note, however, that using an average wavelength
2660 : // may not be a good choice when the total range in frequency is
2661 : // large (e.g. mfs across double sidebands).
2662 0 : uInt nrows = msspw.nrow();
2663 0 : Double ftot = 0.0;
2664 0 : Int nchan = 0;
2665 0 : for(uInt i=0; i < nrows; ++i) {
2666 0 : nchan += (spwc.numChan())(i);
2667 0 : ftot += sum((spwc.chanFreq())(i));
2668 : }
2669 0 : Double invLambda=ftot/(nchan*C::c);
2670 :
2671 : // This is message may not be helpful as mfs is set with setimage()
2672 : // which may sometimes get called after uvrange()
2673 0 : if (nrows > 1 && imageMode_p=="MFS") {
2674 : os << LogIO::WARN
2675 : << "When using mfs over a broad range of frequencies, It is more "
2676 : << "accurate to " << endl
2677 : << "constrain uv-ranges using setdata(); try: " << endl
2678 : << " msselect='(SQUARE(UVW[1]) + SQUARE(UVW[2])) > uvmin && "
2679 : << "(SQUARE(UVW[1]) + SQUARE(UVW[2])) < uvmax'" << endl
2680 : << "where [uvmin, uvmax] is the range given in meters."
2681 0 : << LogIO::POST;
2682 : }
2683 :
2684 0 : invLambda=invLambda*invLambda;
2685 0 : auvmax=auvmax*auvmax;
2686 0 : auvmin=auvmin*auvmin;
2687 0 : strUVmax << auvmax;
2688 0 : strUVmin << auvmin;
2689 0 : ostrInvLambda << invLambda;
2690 0 : String strInvLambda=ostrInvLambda;
2691 : MeasurementSet* mssel_p2;
2692 :
2693 : // Apply the TAQL selection string, to remake the selected MS
2694 0 : String parseString="select from $1 where (SQUARE(UVW[1]) + SQUARE(UVW[2]))*" + (string) strInvLambda + " > " + strUVmin.str( ) + " && (SQUARE(UVW[1]) + SQUARE(UVW[2]))*" + (string) strInvLambda + " < " + strUVmax.str( );
2695 :
2696 0 : mssel_p2=new MeasurementSet(tableCommand(parseString,*mssel_p).table());
2697 0 : AlwaysAssert(mssel_p2, AipsError);
2698 : // Rename the selected MS as */SELECTED_UVRANGE
2699 : //mssel_p2->rename(msname_p+"/SELECTED_UVRANGE", Table::Scratch);
2700 :
2701 0 : if (mssel_p2->nrow()==0) {
2702 : os << LogIO::WARN
2703 : << "Selection string results in empty MS: "
2704 : << "reverting to sorted MeasurementSet"
2705 0 : << LogIO::POST;
2706 0 : delete mssel_p2;
2707 : } else {
2708 0 : if (!mssel_p.null()) {
2709 : os << LogIO::NORMAL // Loglevel INFO
2710 : << "By UVRANGE selection previously selected number of rows "
2711 0 : << mssel_p->nrow() << " are now reduced to "
2712 0 : << mssel_p2->nrow() << LogIO::POST;
2713 : //delete mssel_p;
2714 0 : mssel_p=mssel_p2;
2715 0 : mssel_p->flush();
2716 : }
2717 : }
2718 :
2719 :
2720 0 : this->makeVisSet(*mssel_p);
2721 0 : AlwaysAssert(rvi_p, AipsError);
2722 :
2723 : // NOW WE HAVE TO REDO THE VELOCITY INFO FOR visiter AS IN SETDATA
2724 :
2725 0 : this->selectDataChannel(dataspectralwindowids_p, dataMode_p,
2726 0 : dataNchan_p, dataStart_p, dataStep_p,
2727 0 : mDataStart_p, mDataStep_p);
2728 :
2729 0 : this->writeHistory(os);
2730 0 : this->unlock();
2731 :
2732 : // Beam is no longer valid
2733 0 : beamValid_p=false;
2734 0 : return true;
2735 0 : } catch (AipsError x) {
2736 0 : this->unlock();
2737 0 : throw(x);
2738 : return false;
2739 0 : }
2740 : return true;
2741 0 : }
2742 :
2743 : // Find the sensitivity
2744 0 : Bool Imager::sensitivity(Quantity& pointsourcesens, Double& relativesens,
2745 : Double& sumwt,
2746 : Double& effectiveBandwidth,
2747 : Double& effectiveIntegration,
2748 : Int& nBaselines,
2749 : Matrix<Int>& mssChanSel,
2750 : Vector<Vector<Int> >& nData,
2751 : Vector<Vector<Double> >& sumwtChan,
2752 : Vector<Vector<Double> >& sumwtsqChan,
2753 : Vector<Vector<Double> >& sumInverseVarianceChan)
2754 : {
2755 0 : if(!valid()) return false;
2756 0 : LogIO os(LogOrigin("imager", "sensitivity()", WHERE));
2757 :
2758 : try {
2759 :
2760 : os << LogIO::NORMAL // Loglevel INFO
2761 : << "Calculating sensitivity from imaging weights and from effective bandwidth and integration time"
2762 0 : << LogIO::POST;
2763 :
2764 0 : this->lock();
2765 0 : mssChanSel.assign(mssChanSel_p);
2766 :
2767 0 : VisSetUtil::Sensitivity(*rvi_p, mssFreqSel_p, mssChanSel, pointsourcesens, relativesens, sumwt,
2768 : effectiveBandwidth, effectiveIntegration, nBaselines,nData, sumwtChan,
2769 : sumwtsqChan, sumInverseVarianceChan);
2770 : os << LogIO::NORMAL << "RMS Point source sensitivity : " // Loglevel INFO
2771 0 : << pointsourcesens.get("Jy").getValue() << " Jy.m2/K"
2772 0 : << LogIO::POST;
2773 : os << LogIO::NORMAL // Loglevel INFO
2774 0 : << "Relative to natural weighting : " << relativesens << LogIO::POST;
2775 : os << LogIO::NORMAL // Loglevel INFO
2776 0 : << "Sum of weights : " << sumwt << LogIO::POST;
2777 0 : this->unlock();
2778 0 : return true;
2779 0 : } catch (AipsError x) {
2780 0 : this->unlock();
2781 0 : throw(x);
2782 : return false;
2783 0 : }
2784 : return true;
2785 0 : }
2786 :
2787 0 : Bool Imager::apparentSensitivity(Double& effSensitivity,
2788 : Double& relToNat) {
2789 0 : if(!valid()) return false;
2790 0 : LogIO os(LogOrigin("imager", "apparentSensitivity()", WHERE));
2791 :
2792 : try {
2793 :
2794 : os << LogIO::NORMAL // Loglevel INFO
2795 : << "Calculating apparent sensitivity from MS weights, as modified by gridding weight function"
2796 0 : << LogIO::POST;
2797 : os << LogIO::NORMAL // Loglevel INFO
2798 : << "(assuming that MS weights have correct scale and units)"
2799 0 : << LogIO::POST;
2800 :
2801 0 : this->lock();
2802 :
2803 0 : Double sumNatWt=0.0;
2804 0 : Double sumGridWt=0.0;
2805 0 : Double sumGridWt2OverNatWt=0.0;
2806 :
2807 0 : Float iNatWt(0.0),iGridWt(0.0);
2808 :
2809 0 : ROVisibilityIterator& vi(*rvi_p);
2810 0 : VisBuffer vb(vi);
2811 : //Bool doWtSp=vb.existsWeightSpectrum();
2812 0 : for (vi.originChunks();vi.moreChunks();vi.nextChunk()) {
2813 0 : for (vi.origin();vi.more();vi++) {
2814 0 : Int nRow=vb.nRow();
2815 0 : Vector<Bool> rowFlags=vb.flagRow();
2816 0 : Matrix<Bool> flag = vb.flag();
2817 :
2818 0 : Vector<Int>& a1(vb.antenna1()), a2(vb.antenna2());
2819 :
2820 0 : for (Int row=0; row<nRow; row++) {
2821 0 : if (!rowFlags(row) && a1(row)!=a2(row)) { // exclude ACs
2822 0 : iNatWt=2.0f*vb.weight()(row);
2823 0 : for (Int ich=0;ich<vb.nChannel();++ich) {
2824 0 : if(!flag(ich,row)&&(iNatWt>0.0)) {
2825 0 : iGridWt=2.0f*vb.imagingWeight()(ich,row);
2826 0 : sumNatWt+=(iNatWt);
2827 0 : sumGridWt+=(iGridWt);
2828 0 : sumGridWt2OverNatWt+=(iGridWt*iGridWt/iNatWt);
2829 : }
2830 : }
2831 : }
2832 : }
2833 0 : }
2834 : }
2835 :
2836 0 : if (sumNatWt==0.0) {
2837 0 : os << "Cannot calculate sensitivity: sum of selected natural weights is zero" << LogIO::EXCEPTION;
2838 : }
2839 0 : if (sumGridWt==0.0) {
2840 0 : os << "Cannot calculate sensitivity: sum of gridded weights is zero" << LogIO::EXCEPTION;
2841 : }
2842 :
2843 0 : effSensitivity = sqrt(sumGridWt2OverNatWt)/sumGridWt;
2844 0 : Double natSensitivity = 1.0/sqrt(sumNatWt);
2845 0 : relToNat=effSensitivity/natSensitivity;
2846 :
2847 : os << LogIO::NORMAL << "RMS Point source sensitivity : " // Loglevel INFO
2848 : << effSensitivity // << " Jy/beam" // actually, units are arbitrary
2849 0 : << LogIO::POST;
2850 : os << LogIO::NORMAL // Loglevel INFO
2851 0 : << "Relative to natural weighting : " << relToNat << LogIO::POST;
2852 :
2853 0 : this->unlock();
2854 0 : return true;
2855 0 : } catch (AipsError x) {
2856 0 : this->unlock();
2857 0 : throw(x);
2858 : return false;
2859 0 : }
2860 : return true;
2861 :
2862 0 : }
2863 :
2864 :
2865 :
2866 :
2867 :
2868 : // Calculate various sorts of image. Only one image
2869 : // can be calculated at a time. The complex Image make
2870 : // be retained if a name is given. This does not use
2871 : // the SkyEquation.
2872 0 : Bool Imager::makeimage(const String& type, const String& image,
2873 : const String& compleximage, const Bool verbose)
2874 : {
2875 0 : if(!valid())
2876 : {
2877 0 : return false;
2878 : }
2879 0 : LogIO os(LogOrigin("imager", "makeimage()", WHERE));
2880 :
2881 0 : this->lock();
2882 : try {
2883 0 : if(!assertDefinedImageParameters())
2884 : {
2885 0 : return false;
2886 : }
2887 :
2888 : os << LogIO::NORMAL // Loglevel INFO
2889 0 : << "Calculating image (without full skyequation)" << LogIO::POST;
2890 :
2891 0 : FTMachine::Type seType(FTMachine::OBSERVED);
2892 0 : Bool doSD(false);
2893 :
2894 0 : if(type=="observed") {
2895 0 : seType=FTMachine::OBSERVED;
2896 : os << LogIO::NORMAL // Loglevel INFO
2897 : << "Making dirty image from " << type << " data "
2898 0 : << LogIO::POST;
2899 : }
2900 0 : else if (type=="model") {
2901 0 : seType=FTMachine::MODEL;
2902 : os << LogIO::NORMAL // Loglevel INFO
2903 : << "Making dirty image from " << type << " data "
2904 0 : << LogIO::POST;
2905 : }
2906 0 : else if (type=="corrected") {
2907 0 : seType=FTMachine::CORRECTED;
2908 : os << LogIO::NORMAL // Loglevel INFO
2909 : << "Making dirty image from " << type << " data "
2910 0 : << LogIO::POST;
2911 : }
2912 0 : else if (type=="psf") {
2913 0 : seType=FTMachine::PSF;
2914 : os << "Making point spread function "
2915 0 : << LogIO::POST;
2916 : }
2917 0 : else if (type=="residual") {
2918 0 : seType=FTMachine::RESIDUAL;
2919 : os << LogIO::NORMAL // Loglevel INFO
2920 : << "Making dirty image from " << type << " data "
2921 0 : << LogIO::POST;
2922 : }
2923 0 : else if (type=="singledish-observed") {
2924 0 : doSD = true;
2925 0 : seType=FTMachine::OBSERVED;
2926 : os << LogIO::NORMAL // Loglevel INFO
2927 0 : << "Making single dish image from observed data" << LogIO::POST;
2928 : }
2929 0 : else if (type=="singledish") {
2930 0 : doSD = true;
2931 0 : seType=FTMachine::CORRECTED;
2932 : os << LogIO::NORMAL // Loglevel INFO
2933 0 : << "Making single dish image from corrected data" << LogIO::POST;
2934 : }
2935 0 : else if (type=="coverage") {
2936 0 : doSD = true;
2937 0 : seType=FTMachine::COVERAGE;
2938 : os << LogIO::NORMAL // Loglevel PROGRESS
2939 : << "Making single dish coverage function "
2940 0 : << LogIO::POST;
2941 : }
2942 0 : else if (type=="holography") {
2943 0 : doSD = true;
2944 0 : seType=FTMachine::CORRECTED;
2945 : os << LogIO::NORMAL // Loglevel INFO
2946 : << "Making complex holographic image from corrected data "
2947 0 : << LogIO::POST;
2948 : }
2949 0 : else if (type=="holography-observed") {
2950 0 : doSD = true;
2951 0 : seType=FTMachine::OBSERVED;
2952 : os << LogIO::NORMAL // Loglevel INFO
2953 : << "Making complex holographic image from observed data "
2954 0 : << LogIO::POST;
2955 : }
2956 0 : else if (type=="pb"){
2957 0 : if ( ! doVP_p ) {
2958 0 : if( ftmachine_p == "pbwproject" ){
2959 0 : os << LogIO::WARN << "Using pb from ft-machines" << LogIO::POST;
2960 : }
2961 : else{
2962 0 : this->unlock();
2963 : os << LogIO::SEVERE <<
2964 : "Must invoke setvp() first in order to make its image"
2965 0 : << LogIO::EXCEPTION;
2966 0 : return false;
2967 : }
2968 : }
2969 0 : CoordinateSystem coordsys;
2970 : //imagecoordinates(coordsys, verbose);
2971 0 : imagecoordinates2(coordsys, verbose);
2972 0 : if (doDefaultVP_p) {
2973 0 : if(telescope_p!=""){
2974 0 : ObsInfo myobsinfo=this->latestObsInfo();
2975 0 : myobsinfo.setTelescope(telescope_p);
2976 0 : coordsys.setObsInfo(myobsinfo);
2977 :
2978 0 : }
2979 : else{
2980 0 : telescope_p=coordsys.obsInfo().telescope();
2981 : }
2982 0 : this->unlock();
2983 0 : MSAntennaColumns ac(ms_p->antenna());
2984 0 : Double dishDiam=ac.dishDiameter()(0);
2985 0 : if(!allEQ(ac.dishDiameter().getColumn(), dishDiam))
2986 : os << LogIO::WARN
2987 : << "The MS has multiple antenna diameters ..PB could be wrong "
2988 0 : << LogIO::POST;
2989 0 : return this->makePBImage(coordsys, telescope_p, image, false, dishDiam);
2990 0 : }
2991 : else{
2992 0 : Table vpTable(vpTableStr_p);
2993 0 : this->unlock();
2994 0 : return this->makePBImage(coordsys, vpTable, image);
2995 0 : }
2996 :
2997 0 : }
2998 : else {
2999 0 : this->unlock();
3000 0 : os << LogIO::SEVERE << "Unknown image type " << type << LogIO::EXCEPTION;
3001 :
3002 0 : return false;
3003 : }
3004 :
3005 0 : if(doSD && (ftmachine_p == "ft")){
3006 : os << LogIO::SEVERE
3007 : << "To make single dish images, ftmachine in setoptions must be set to either sd or both"
3008 0 : << LogIO::EXCEPTION;
3009 : }
3010 :
3011 : // Now make the images. If we didn't specify the names then
3012 : // delete on exit.
3013 0 : String imageName(image);
3014 0 : if(image=="") {
3015 0 : imageName=Imager::imageName()+".image";
3016 : }
3017 0 : os << LogIO::NORMAL << "Image is : " << imageName << LogIO::POST; // Loglevel INFO
3018 0 : Bool keepImage=(image!="");
3019 0 : Bool keepComplexImage=(compleximage!="")||(type=="holography")||(type=="holography-observed");
3020 0 : String cImageName(compleximage);
3021 :
3022 0 : if(compleximage=="") {
3023 0 : cImageName=imageName+".compleximage";
3024 : }
3025 :
3026 0 : if(keepComplexImage) {
3027 0 : os << "Retaining complex image: " << compleximage << LogIO::POST;
3028 : }
3029 :
3030 0 : CoordinateSystem imagecoords;
3031 : //if(!imagecoordinates(imagecoords, false))
3032 0 : if(!imagecoordinates2(imagecoords, false))
3033 : {
3034 0 : return false;
3035 : }
3036 0 : make(imageName);
3037 0 : PagedImage<Float> imageImage(imageName);
3038 0 : imageImage.set(0.0);
3039 0 : imageImage.table().markForDelete();
3040 :
3041 : // Now set up the tile size, here we guess only
3042 0 : IPosition cimageShape(imageshape());
3043 0 : Int tilex=32;
3044 0 : if(imageTileVol_p >0){
3045 0 : tilex=static_cast<Int>(ceil(sqrt(imageTileVol_p/min(4,
3046 0 : cimageShape(2))/min(32,
3047 0 : cimageShape(3)))));
3048 0 : if(tilex >0){
3049 0 : if(tilex > min(Int(cimageShape(0)), Int(cimageShape(1))))
3050 0 : tilex=min(Int(cimageShape(0)), Int(cimageShape(1)));
3051 : else
3052 0 : tilex=cimageShape(0)/Int(cimageShape(0)/tilex);
3053 : }
3054 : //Not too small in x-y tile
3055 0 : if(tilex < 10)
3056 0 : tilex=10;
3057 :
3058 : }
3059 0 : IPosition tileShape(4, min(tilex, cimageShape(0)), min(tilex, cimageShape(1)),
3060 0 : min(4, cimageShape(2)), min(32, cimageShape(3)));
3061 0 : CoordinateSystem cimagecoords;
3062 : //if(!imagecoordinates(cimagecoords, false))
3063 0 : if(!imagecoordinates2(cimagecoords, false))
3064 : {
3065 0 : return false;
3066 : }
3067 :
3068 :
3069 :
3070 0 : PagedImage<Complex> cImageImage(TiledShape(cimageShape, tileShape),
3071 : cimagecoords,
3072 0 : cImageName);
3073 0 : cImageImage.set(Complex(0.0));
3074 0 : cImageImage.setMaximumCacheSize(cache_p/2);
3075 0 : cImageImage.table().markForDelete();
3076 : //
3077 : // Add the distance to the object: this is not nice. We should define the
3078 : // coordinates properly.
3079 : //
3080 0 : Record info(imageImage.miscInfo());
3081 0 : info.define("distance", distance_p.get("m").getValue());
3082 0 : cImageImage.setMiscInfo(info);
3083 :
3084 :
3085 0 : String ftmachine(ftmachine_p);
3086 0 : if (!ft_p)
3087 0 : createFTMachine();
3088 :
3089 : os << LogIO::DEBUG1 << "FTMachine is : " << ftmachine
3090 0 : << " (" << ft_p << ")" << LogIO::POST;
3091 :
3092 :
3093 : // Now make the required image
3094 0 : Matrix<Float> weight;
3095 0 : ft_p->makeImage(seType, *rvi_p, cImageImage, weight);
3096 0 : StokesImageUtil::To(imageImage, cImageImage);
3097 : //
3098 : // Dirty way to set the proper unit to SD image
3099 : //
3100 0 : String msunit("");
3101 0 : String imunit;
3102 0 : if ( ms_p->tableDesc().isColumn("DATA") ){
3103 0 : msunit = ms_p->columnUnit(MS::DATA);
3104 0 : if (msunit == String("")) {
3105 0 : ColumnDesc dataColDesc(ms_p->tableDesc().columnDesc("DATA"));
3106 0 : if (dataColDesc.keywordSet().isDefined("UNIT"))
3107 0 : msunit = dataColDesc.keywordSet().asString("UNIT");
3108 0 : }
3109 0 : } else if ( ms_p->tableDesc().isColumn("FLOAT_DATA")) {
3110 0 : msunit = ms_p->columnUnit(MS::FLOAT_DATA);
3111 0 : if (msunit == String("")) {
3112 0 : ColumnDesc dataColDesc(ms_p->tableDesc().columnDesc("FLOAT_DATA"));
3113 0 : if (dataColDesc.keywordSet().isDefined("UNIT"))
3114 0 : msunit = dataColDesc.keywordSet().asString("UNIT");
3115 0 : }
3116 : }
3117 0 : msunit.upcase();
3118 0 : if (msunit == String("K"))
3119 0 : imunit = "K";
3120 : else
3121 0 : imunit = "Jy/beam";
3122 0 : imageImage.setUnits(Unit(imunit));
3123 0 : cImageImage.setUnits(Unit(imunit));
3124 : // imageImage.setUnits(Unit("Jy/beam"));
3125 : // cImageImage.setUnits(Unit("Jy/beam"));
3126 :
3127 0 : if(keepImage) {
3128 0 : imageImage.table().unmarkForDelete();
3129 : }
3130 0 : if(keepComplexImage) {
3131 0 : cImageImage.table().unmarkForDelete();
3132 : }
3133 0 : this->unlock();
3134 0 : return true;
3135 0 : } catch (AipsError x) {
3136 0 : this->unlock();
3137 0 : throw(x);
3138 : return false;
3139 0 : }
3140 0 : catch(...){
3141 : //Unknown exception...
3142 0 : throw(AipsError("Unknown exception caught ...imager/casa may need to be exited"));
3143 0 : }
3144 : this->unlock();
3145 :
3146 : return true;
3147 0 : }
3148 :
3149 : // Restore: at least one model must be supplied
3150 0 : Bool Imager::restore(const Vector<String>& model,
3151 : const String& complist,
3152 : const Vector<String>& image,
3153 : const Vector<String>& residual)
3154 : {
3155 :
3156 0 : if(!valid()) return false;
3157 0 : LogIO os(LogOrigin("imager", "restore()", WHERE));
3158 :
3159 0 : this->lock();
3160 : try {
3161 0 : if(!assertDefinedImageParameters()) return false;
3162 :
3163 0 : if(image.nelements()>model.nelements()) {
3164 0 : this->unlock();
3165 : os << LogIO::SEVERE << "Cannot specify more output images than models"
3166 0 : << LogIO::EXCEPTION;
3167 0 : return false;
3168 : }
3169 : else {
3170 : os << LogIO::NORMAL // Loglevel PROGRESS
3171 0 : << "Restoring " << model.nelements() << " models" << LogIO::POST;
3172 : }
3173 :
3174 : ///if the skymodel is already set...no need to get rid of the psf and ftmachine state
3175 : //as long as the images match
3176 0 : if(!redoSkyModel_p){
3177 0 : Bool coordMatch=true;
3178 0 : for (Int thismodel=0;thismodel<Int(model.nelements());++thismodel) {
3179 0 : CoordinateSystem cs=(sm_p->image(thismodel)).coordinates();
3180 0 : coordMatch= coordMatch || checkCoord(cs, model(thismodel));
3181 0 : }
3182 0 : if(!coordMatch)
3183 0 : destroySkyEquation();
3184 : }
3185 :
3186 0 : if(redoSkyModel_p){
3187 0 : Vector<String> imageNames(image);
3188 0 : if(image.nelements()<model.nelements()) {
3189 0 : imageNames.resize(model.nelements());
3190 0 : for(Int i=0;i<Int(model.nelements()); ++i) {
3191 0 : imageNames(i)="";
3192 : }
3193 : }
3194 :
3195 0 : for (Int thismodel=0;thismodel<Int(model.nelements());++thismodel) {
3196 0 : if(imageNames(thismodel)=="") {
3197 0 : imageNames(thismodel)=model(thismodel)+".restored";
3198 : }
3199 0 : removeTable(imageNames(thismodel));
3200 0 : if(imageNames(thismodel)=="") {
3201 0 : this->unlock();
3202 : os << LogIO::SEVERE << "Illegal name for output image "
3203 0 : << imageNames(thismodel) << LogIO::EXCEPTION;
3204 0 : return false;
3205 : }
3206 0 : if(!clone(model(thismodel), imageNames(thismodel))) return false;
3207 : }
3208 :
3209 0 : Vector<String> residualNames(residual);
3210 0 : if(residual.nelements()<model.nelements()) {
3211 0 : residualNames.resize(model.nelements());
3212 0 : for(Int i=0;i<Int(model.nelements());++i) {
3213 0 : residualNames(i)="";
3214 : }
3215 : }
3216 :
3217 0 : for (Int thismodel=0;thismodel<Int(model.nelements()); ++thismodel) {
3218 0 : if(residualNames(thismodel)=="")
3219 0 : residualNames(thismodel)=model(thismodel)+".residual";
3220 0 : removeTable(residualNames(thismodel));
3221 0 : if(residualNames(thismodel)=="") {
3222 0 : this->unlock();
3223 : os << LogIO::SEVERE << "Illegal name for output residual "
3224 0 : << residualNames(thismodel) << LogIO::EXCEPTION;
3225 0 : return false;
3226 : }
3227 0 : if(!clone(model(thismodel), residualNames(thismodel))) return false;
3228 : }
3229 :
3230 0 : if(beamValid_p) {
3231 0 : os << LogIO::NORMAL << "Using previous beam fit" << LogIO::POST; // Loglevel INFO
3232 : }
3233 : else {
3234 : os << LogIO::NORMAL // Loglevel INFO
3235 0 : << "Calculating PSF using current parameters" << LogIO::POST;
3236 0 : String psf;
3237 0 : psf=imageNames(0)+".psf";
3238 0 : if(!clone(imageNames(0), psf)) return false;
3239 0 : Imager::makeimage("psf", psf);
3240 0 : fitpsf(psf, beam_p);
3241 0 : beamValid_p=true;
3242 0 : }
3243 :
3244 : // if (!se_p)
3245 0 : if(!createSkyEquation(model, complist)) return false;
3246 :
3247 0 : addResiduals(residualNames);
3248 0 : }
3249 0 : sm_p->solveResiduals(*se_p);
3250 0 : for (uInt k=0 ; k < residuals_p.nelements(); ++k){
3251 0 : residuals_p[k]->copyData(sm_p->getResidual(k));
3252 : }
3253 0 : restoreImages(image);
3254 :
3255 0 : this->unlock();
3256 0 : return true;
3257 0 : } catch (AipsError x) {
3258 0 : this->unlock();
3259 0 : throw(x);
3260 : return false;
3261 0 : }
3262 : this->unlock();
3263 : return true;
3264 0 : }
3265 :
3266 0 : Bool Imager::updateresidual(const Vector<String>& model,
3267 : const String& complist,
3268 : const Vector<String>& image,
3269 : const Vector<String>& residual)
3270 : {
3271 :
3272 0 : if(!valid()) return false;
3273 0 : LogIO os(LogOrigin("imager", "updateresidual()", WHERE));
3274 :
3275 0 : this->lock();
3276 : try {
3277 0 : if(!assertDefinedImageParameters()) return false;
3278 :
3279 0 : if(image.nelements()>model.nelements()) {
3280 0 : this->unlock();
3281 : os << LogIO::SEVERE << "Cannot specify more output images than models"
3282 0 : << LogIO::EXCEPTION;
3283 0 : return false;
3284 : }
3285 : else {
3286 : os << LogIO::NORMAL // Loglevel PROGRESS
3287 0 : << "updating and restoring " << model.nelements() << " models" << LogIO::POST;
3288 : }
3289 :
3290 0 : if(redoSkyModel_p)
3291 0 : throw(AipsError("use restore instead of updateresidual"));
3292 0 : if(!updateSkyModel(model, complist))
3293 0 : throw(AipsError("Could not do an updateresidual please use restore"));
3294 0 : addResiduals(residual);
3295 0 : for (Int thismodel=0;thismodel<Int(residuals_p.nelements());++thismodel) {
3296 0 : if(!residuals_p[thismodel].null())
3297 0 : sm_p->addResidual(thismodel, *residuals_p[thismodel]);
3298 : }
3299 0 : sm_p->solveResiduals(*se_p);
3300 : /*for (uInt k=0 ; k < residuals_p.nelements(); ++k){
3301 : residuals_p[k]->copyData(sm_p->getResidual(k));
3302 : }
3303 : */
3304 0 : restoreImages(image);
3305 :
3306 :
3307 0 : this->unlock();
3308 0 : return true;
3309 0 : } catch (AipsError x) {
3310 0 : this->unlock();
3311 0 : throw(x);
3312 : return false;
3313 0 : }
3314 : this->unlock();
3315 : return true;
3316 0 : }
3317 :
3318 : // Residual
3319 0 : Bool Imager::residual(const Vector<String>& model,
3320 : const String& complist,
3321 : const Vector<String>& image)
3322 : {
3323 :
3324 0 : if(!valid()) return false;
3325 0 : LogIO os(LogOrigin("imager", "residual()", WHERE));
3326 :
3327 0 : this->lock();
3328 : try {
3329 0 : if(!assertDefinedImageParameters()) return false;
3330 : os << LogIO::NORMAL // Loglevel INFO
3331 0 : << "Calculating residual image using full sky equation" << LogIO::POST;
3332 0 : Vector<String> theModels=model;
3333 :
3334 0 : Bool deleteModel=false;
3335 :
3336 0 : if(model.nelements()==1 && model[0]=="" && complist != ""
3337 0 : && image.nelements()==1){
3338 :
3339 : // A component list with no model passed...
3340 0 : theModels.resize(1);
3341 0 : theModels[0]="Imager_Scratch_model";
3342 0 : make(theModels[0]);
3343 0 : deleteModel=true;
3344 : }
3345 :
3346 0 : if(image.nelements()>theModels.nelements()) {
3347 0 : this->unlock();
3348 : os << LogIO::SEVERE << "Cannot specify more output images than models"
3349 0 : << LogIO::EXCEPTION;
3350 0 : return false;
3351 : }
3352 : else {
3353 : os << LogIO::NORMAL << "Finding residuals for " << theModels.nelements() // Loglevel INFO
3354 0 : << " models" << LogIO::POST;
3355 : }
3356 :
3357 0 : Vector<String> imageNames(image);
3358 0 : if(image.nelements()<theModels.nelements()) {
3359 0 : imageNames.resize(model.nelements());
3360 0 : for(Int i=Int(image.nelements());i<Int(theModels.nelements());++i) {
3361 0 : imageNames(i)="";
3362 : }
3363 : }
3364 :
3365 0 : for (Int thismodel=0;thismodel<Int(theModels.nelements()); ++thismodel) {
3366 0 : if(imageNames(thismodel)=="")
3367 0 : imageNames(thismodel)=model(thismodel)+".residual";
3368 0 : removeTable(imageNames(thismodel));
3369 0 : if(imageNames(thismodel)=="") {
3370 0 : this->unlock();
3371 : os << LogIO::SEVERE << "Illegal name for output image "
3372 0 : << imageNames(thismodel) << LogIO::EXCEPTION;
3373 0 : return false;
3374 : }
3375 0 : if(!clone(theModels(thismodel), imageNames(thismodel))) return false;
3376 : }
3377 0 : destroySkyEquation();
3378 0 : if(!createSkyEquation(theModels, complist)) return false;
3379 :
3380 0 : addResidualsToSkyEquation(imageNames);
3381 :
3382 0 : sm_p->solveResiduals(*se_p);
3383 0 : destroySkyEquation();
3384 0 : if(deleteModel)
3385 0 : removeTable(theModels[0]);
3386 0 : this->unlock();
3387 0 : return true;
3388 0 : } catch (AipsError x) {
3389 0 : this->unlock();
3390 0 : throw(x);
3391 : return false;
3392 0 : }
3393 : this->unlock();
3394 : return true;
3395 0 : }
3396 :
3397 : // Residual
3398 0 : Bool Imager::approximatepsf(const String& psf)
3399 : {
3400 :
3401 0 : if(!valid()) return false;
3402 0 : LogIO os(LogOrigin("imager", "approximatepsfs()", WHERE));
3403 :
3404 0 : this->lock();
3405 : try {
3406 0 : if(!assertDefinedImageParameters()) return false;
3407 : os << LogIO::NORMAL // Loglevel INFO
3408 0 : << "Calculating approximate PSFs using full sky equation" << LogIO::POST;
3409 :
3410 :
3411 0 : if(psf==""){
3412 0 : this->unlock();
3413 : os << LogIO::SEVERE << "Illegal name for output psf "
3414 0 : << psf << LogIO::EXCEPTION;
3415 0 : return false;
3416 : }
3417 0 : removeTable(psf);
3418 0 : make(psf);
3419 :
3420 0 : Vector<String>onepsf(1,psf);
3421 : // Previous SkyEquation if they exist is not useful
3422 0 : destroySkyEquation();
3423 : // if (!se_p)
3424 : // As we are not going to make any use of a useful model and to economize
3425 : // temporary image...using the psf itself as model...
3426 : // need to change this if you donot destroy the skyequation after you're done.
3427 0 : if(!createSkyEquation(onepsf)) return false;
3428 :
3429 0 : sm_p->makeApproxPSFs(*se_p);
3430 :
3431 :
3432 0 : PagedImage<Float> elpsf(psf);
3433 0 : elpsf.copyData(sm_p->PSF(0));
3434 0 : ImageBeamSet mbeam;
3435 0 : StokesImageUtil::FitGaussianPSF(elpsf, mbeam);
3436 0 : LatticeExprNode sumPSF = sum(elpsf);
3437 0 : Float volume=sumPSF.getFloat();
3438 0 : GaussianBeam elbeam=mbeam(0,0);
3439 : os << LogIO::NORMAL << "Approximate PSF " << ": size " // Loglevel INFO
3440 : << elbeam.getMajor("arcsec") << " by "
3441 : << elbeam.getMinor("arcsec") << " (arcsec) at pa "
3442 0 : << elbeam.getPA(Unit("deg")) << " (deg)" << endl
3443 0 : << "and volume = " << volume << " pixels " << LogIO::POST;
3444 :
3445 :
3446 0 : destroySkyEquation();
3447 0 : if(ft_p)
3448 0 : delete ft_p;
3449 0 : ft_p=0;
3450 :
3451 0 : this->unlock();
3452 0 : return true;
3453 0 : } catch (AipsError x) {
3454 0 : this->unlock();
3455 0 : throw(x);
3456 : return false;
3457 0 : }
3458 : this->unlock();
3459 : return true;
3460 0 : }
3461 :
3462 0 : Bool Imager::smooth(const Vector<String>& model,
3463 : const Vector<String>& image, Bool usefit,
3464 : ImageBeamSet& mbeam,
3465 : Bool normalizeVolume)
3466 : {
3467 0 : if(!valid()) return false;
3468 0 : LogIO os(LogOrigin("imager", "smooth()", WHERE));
3469 :
3470 0 : this->lock();
3471 : try {
3472 0 : if(!assertDefinedImageParameters()) return false;
3473 :
3474 0 : os << LogIO::NORMAL << "Smoothing image" << LogIO::POST; // Loglevel PROGRESS
3475 :
3476 0 : if(model.nelements()>0) {
3477 0 : for ( uInt thismodel=0;thismodel<model.nelements(); ++thismodel) {
3478 0 : if(model(thismodel)=="") {
3479 0 : this->unlock();
3480 0 : os << LogIO::SEVERE << "Need a name for model " << thismodel << LogIO::POST;
3481 0 : return false;
3482 : }
3483 : }
3484 : }
3485 :
3486 0 : if(image.nelements()>model.nelements()) {
3487 0 : this->unlock();
3488 0 : os << LogIO::SEVERE << "Cannot specify more output images than models" << LogIO::POST;
3489 0 : return false;
3490 : }
3491 :
3492 0 : if(usefit) {
3493 0 : if(beamValid_p) {
3494 0 : os << LogIO::NORMAL << "Using previous beam" << LogIO::POST; // Loglevel INFO
3495 0 : mbeam = beam_p;
3496 :
3497 : }
3498 : else {
3499 : os << LogIO::NORMAL // Loglevel INFO
3500 0 : << "Calculating PSF using current parameters" << LogIO::POST;
3501 0 : String psf;
3502 0 : psf=model(0)+".psf";
3503 0 : if(!clone(model(0), psf)) return false;
3504 0 : Imager::makeimage("psf", psf);
3505 0 : fitpsf(psf, mbeam);
3506 0 : beam_p = mbeam;
3507 0 : beamValid_p=true;
3508 0 : }
3509 : }
3510 :
3511 : // Smooth all the images
3512 0 : Vector<String> imageNames(image);
3513 0 : for (Int thismodel=0;thismodel<Int(image.nelements()); ++thismodel) {
3514 0 : if(imageNames(thismodel)=="") {
3515 0 : imageNames(thismodel)=model(thismodel)+".smoothed";
3516 : }
3517 0 : PagedImage<Float> modelImage(model(thismodel));
3518 0 : PagedImage<Float> imageImage(TiledShape(modelImage.shape(),
3519 0 : modelImage.niceCursorShape()),
3520 : modelImage.coordinates(),
3521 0 : imageNames(thismodel));
3522 0 : imageImage.table().markForDelete();
3523 0 : imageImage.copyData(modelImage);
3524 0 : StokesImageUtil::Convolve(imageImage, mbeam,
3525 : normalizeVolume);
3526 :
3527 0 : ImageInfo ii = imageImage.imageInfo();
3528 : //ii.setRestoringBeam(mbeam);
3529 0 : ii.setBeams(mbeam);
3530 0 : imageImage.setImageInfo(ii);
3531 0 : imageImage.setUnits(Unit("Jy/beam"));
3532 0 : imageImage.table().unmarkForDelete();
3533 0 : }
3534 :
3535 0 : this->unlock();
3536 0 : return true;
3537 0 : } catch (AipsError x) {
3538 0 : this->unlock();
3539 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
3540 0 : return false;
3541 0 : }
3542 : this->unlock();
3543 : return true;
3544 0 : }
3545 :
3546 : // Clean algorithm
3547 0 : Record Imager::clean(const String& algorithm,
3548 : const Int niter,
3549 : const Float gain,
3550 : const Quantity& threshold,
3551 : const Bool /*displayProgress*/,
3552 : const Vector<String>& model, const Vector<Bool>& fixed,
3553 : const String& complist,
3554 : const Vector<String>& mask,
3555 : const Vector<String>& image,
3556 : const Vector<String>& residual,
3557 : const Vector<String>& psfnames,
3558 : const Bool firstrun)
3559 : {
3560 : ////////////////////////
3561 : //Double wtime0=omp_get_wtime();
3562 : //////////////////////
3563 :
3564 :
3565 0 : Record retval;
3566 0 : Bool converged=true;
3567 0 : retval.define("converged", false);
3568 0 : retval.define("iterations", Int(0));
3569 0 : retval.define("maxresidual", Float(0.0));
3570 :
3571 :
3572 :
3573 : //ROVisibilityIterator::AsyncEnabler enabler (rvi_p);
3574 :
3575 0 : if(!valid())
3576 : {
3577 0 : return retval;
3578 : }
3579 0 : logSink_p.clearLocally();
3580 0 : LogIO os(LogOrigin("imager", "clean()"),logSink_p);
3581 :
3582 0 : this->lock();
3583 : try {
3584 0 : if(!assertDefinedImageParameters())
3585 : {
3586 0 : return retval;
3587 : }
3588 :
3589 0 : Int nmodels=model.nelements();
3590 : os << LogIO::DEBUG1
3591 0 : << "Found " << nmodels << " specified model images" << LogIO::POST;
3592 :
3593 0 : if(model.nelements()>0) {
3594 0 : for (uInt thismodel=0;thismodel<model.nelements(); ++thismodel) {
3595 0 : if(model(thismodel)=="") {
3596 0 : this->unlock();
3597 : os << LogIO::SEVERE << "Need a name for model "
3598 0 : << thismodel << LogIO::POST;
3599 :
3600 0 : return retval;
3601 : }
3602 : }
3603 : }
3604 :
3605 0 : Vector<String> modelNames=model;
3606 : // Make first image with the required shape and coordinates only if
3607 : // it doesn't exist yet. Otherwise we'll throw an exception later
3608 0 : if(modelNames(0)=="") modelNames(0)=imageName()+".clean";
3609 0 : if(!Table::isWritable(modelNames(0))) {
3610 0 : make(modelNames(0));
3611 : }
3612 : else{
3613 0 : Bool coordMatch=false;
3614 0 : CoordinateSystem coordsys;
3615 : //imagecoordinates(coordsys, firstrun);
3616 0 : imagecoordinates2(coordsys, firstrun);
3617 0 : for (uInt modelNum=0; modelNum < modelNames.nelements(); ++modelNum){
3618 0 : if(Table::isWritable(modelNames(modelNum))){
3619 0 : coordMatch= coordMatch ||
3620 0 : (this->checkCoord(coordsys, modelNames(modelNum)));
3621 :
3622 : }
3623 :
3624 : }
3625 0 : if(!coordMatch){
3626 : os << LogIO::WARN << "The model(s) image exists on disk "
3627 0 : << LogIO::POST;
3628 : os << LogIO::WARN
3629 : << "The coordinates or shape were found not to match the one "
3630 : << "defined by setimage "
3631 0 : << LogIO::POST;
3632 :
3633 : os << LogIO::WARN
3634 : << "Cleaning process is going to ignore setimage parameters and "
3635 : << "continue cleaning from from model on disk "
3636 0 : << LogIO::POST;
3637 : }
3638 0 : }
3639 0 : Vector<String> maskNames(nmodels);
3640 0 : if(Int(mask.nelements())==nmodels) {
3641 0 : maskNames=mask;
3642 : }
3643 : else {
3644 : /* For msmfs, the one input mask PER FIELD must be replicated for all
3645 : Taylor-planes PER FIELD */
3646 0 : if(algorithm=="msmfs" && (Int(mask.nelements())>=(nmodels/ntaylor_p)) ){
3647 0 : for(Int tay=0;tay<nmodels;tay++)
3648 : {
3649 0 : maskNames[tay] = mask[ tay%(nmodels/ntaylor_p) ];
3650 : }
3651 : }
3652 : else {
3653 : /* No mask */
3654 0 : maskNames="";
3655 : }
3656 : }
3657 :
3658 0 : if(sm_p){
3659 0 : if( sm_p->getAlgorithm() != "clean") destroySkyEquation();
3660 0 : if(images_p.nelements() != uInt(nmodels)){
3661 0 : destroySkyEquation();
3662 : }
3663 : else{
3664 0 : for (Int k=0; k < nmodels ; ++k){
3665 0 : if(!(images_p[k]->name().contains(modelNames[k]))) destroySkyEquation();
3666 : }
3667 : }
3668 : }
3669 :
3670 : // Always fill in the residual images
3671 0 : Vector<String> residualNames(nmodels);
3672 0 : if(Int(residual.nelements())==nmodels) {
3673 0 : residualNames=residual;
3674 : }
3675 : else {
3676 0 : residualNames="";
3677 : }
3678 0 : for (Int thismodel=0;thismodel<Int(model.nelements());++thismodel) {
3679 0 : if(residualNames[thismodel]=="")
3680 0 : residualNames(thismodel)=modelNames(thismodel)+".residual";
3681 : }
3682 0 : if(redoSkyModel_p){
3683 0 : for (Int thismodel=0;thismodel<Int(model.nelements());++thismodel) {
3684 0 : removeTable(residualNames(thismodel));
3685 0 : if(!clone(model(thismodel), residualNames(thismodel)))
3686 : {
3687 0 : return retval;
3688 : }
3689 : }
3690 : }
3691 :
3692 :
3693 : // Make an ImageSkyModel with the specified polarization representation
3694 : // (i.e. circular or linear)
3695 :
3696 0 : if( redoSkyModel_p || !sm_p){
3697 0 : if(sm_p) delete sm_p;
3698 0 : if(algorithm.substr(0,5)=="clark") {
3699 : // Support serial and parallel specializations
3700 0 : setClarkCleanImageSkyModel();
3701 0 : if(algorithm.contains("stokes"))
3702 0 : sm_p->setJointStokesClean(false);
3703 : os << LogIO::NORMAL // Loglevel INFO. Stating the algo is more for
3704 0 : << "Using Clark clean" << LogIO::POST; // the logfile than the window.
3705 : }
3706 0 : else if (algorithm=="hogbom") {
3707 0 : sm_p = new HogbomCleanImageSkyModel();
3708 : os << LogIO::NORMAL // Loglevel INFO. Stating the algo is more for
3709 0 : << "Using Hogbom clean" << LogIO::POST; // the logfile than the window.
3710 : }
3711 0 : else if (algorithm=="wfhogbom") {
3712 0 : setWFCleanImageSkyModel();
3713 0 : sm_p->setSubAlgorithm("hogbom");
3714 0 : doMultiFields_p = true;
3715 0 : doMultiFields_p = false;
3716 : os << LogIO::NORMAL // Loglevel INFO
3717 0 : << "Using wide-field algorithm with Hogbom clean" << LogIO::POST;
3718 : }
3719 0 : else if (algorithm=="multiscale") {
3720 0 : if (!scaleInfoValid_p) {
3721 0 : this->unlock();
3722 0 : os << LogIO::SEVERE << "Scales not yet set" << LogIO::POST;
3723 0 : return retval;
3724 : }
3725 0 : if (scaleMethod_p=="uservector") {
3726 0 : sm_p = new MSCleanImageSkyModel(userScaleSizes_p, stoplargenegatives_p,
3727 0 : stoppointmode_p, smallScaleBias_p);
3728 : } else {
3729 0 : sm_p = new MSCleanImageSkyModel(nscales_p, stoplargenegatives_p,
3730 0 : stoppointmode_p, smallScaleBias_p);
3731 : }
3732 0 : if(ftmachine_p=="mosaic" ||ftmachine_p=="wproject" )
3733 0 : sm_p->setSubAlgorithm("full");
3734 : os << LogIO::NORMAL // Loglevel INFO. Stating the algo is more for
3735 0 : << "Using multiscale clean" << LogIO::POST; // the logfile than the window.
3736 : }
3737 0 : else if (algorithm.substr(0,7)=="mfclark" || algorithm=="mf") {
3738 0 : sm_p = new MFCleanImageSkyModel();
3739 0 : sm_p->setSubAlgorithm("clark");
3740 0 : if(algorithm.contains("stokes"))
3741 0 : sm_p->setJointStokesClean(false);
3742 :
3743 0 : doMultiFields_p = true;
3744 0 : os << LogIO::NORMAL << "Using multifield Clark clean" << LogIO::POST; // Loglevel INFO
3745 : }
3746 0 : else if (algorithm=="csclean" || algorithm=="cs") {
3747 0 : sm_p = new CSCleanImageSkyModel();
3748 0 : doMultiFields_p = true;
3749 0 : os << LogIO::NORMAL << "Using Cotton-Schwab Clean" << LogIO::POST; // Loglevel INFO
3750 : }
3751 0 : else if (algorithm=="csfast" || algorithm=="csf") {
3752 0 : sm_p = new CSCleanImageSkyModel();
3753 0 : sm_p->setSubAlgorithm("fast");
3754 0 : doMultiFields_p = true;
3755 : os << LogIO::NORMAL // Loglevel INFO
3756 0 : << "Using Cotton-Schwab Clean (optimized)" << LogIO::POST;
3757 : }
3758 0 : else if (algorithm=="mfhogbom") {
3759 0 : sm_p = new MFCleanImageSkyModel();
3760 0 : sm_p->setSubAlgorithm("hogbom");
3761 0 : doMultiFields_p = true;
3762 0 : os << LogIO::NORMAL << "Using multifield Hogbom clean" << LogIO::POST; // Loglevel INFO
3763 : }
3764 0 : else if (algorithm=="mfmultiscale") {
3765 0 : if (!scaleInfoValid_p) {
3766 0 : this->unlock();
3767 0 : os << LogIO::SEVERE << "Scales not yet set" << LogIO::POST;
3768 0 : return retval;
3769 : }
3770 0 : if (scaleMethod_p=="uservector") {
3771 0 : sm_p = new MFMSCleanImageSkyModel(userScaleSizes_p,
3772 : stoplargenegatives_p,
3773 : stoppointmode_p,
3774 0 : smallScaleBias_p);
3775 : } else {
3776 0 : sm_p = new MFMSCleanImageSkyModel(nscales_p,
3777 : stoplargenegatives_p,
3778 : stoppointmode_p,
3779 0 : smallScaleBias_p);
3780 : }
3781 : // if(ftmachine_p=="mosaic"|| ftmachine_p=="wproject")
3782 : // For some reason this does not seem to work without full
3783 0 : sm_p->setSubAlgorithm("full");
3784 :
3785 0 : doMultiFields_p = true;
3786 : os << LogIO::NORMAL << "Using multifield multi-scale clean" // Loglevel INFO
3787 0 : << LogIO::POST;
3788 : }
3789 0 : else if (algorithm=="wfclark" || algorithm=="wf") {
3790 : // Support serial and parallel specializations
3791 0 : setWFCleanImageSkyModel();
3792 0 : sm_p->setSubAlgorithm("clark");
3793 0 : doMultiFields_p = false;
3794 : os << LogIO::NORMAL // Loglevel INFO
3795 0 : << "Using wide-field algorithm with Clark clean" << LogIO::POST;
3796 : }
3797 0 : else if (algorithm=="wfhogbom") {
3798 : // Support serial and parallel specializations
3799 0 : setWFCleanImageSkyModel();
3800 0 : sm_p->setSubAlgorithm("hogbom");
3801 0 : doMultiFields_p = false;
3802 : os << LogIO::NORMAL // Loglevel INFO
3803 0 : << "Using wide-field algorithm with Hogbom clean" << LogIO::POST;
3804 : }
3805 0 : else if (algorithm=="msmfs") {
3806 0 : doMultiFields_p = false;
3807 0 : doWideBand_p = true;
3808 :
3809 : // check for wrong ftmachine specs.
3810 0 : if ( (ftmachine_p != "ft") && (ftmachine_p != "wproject") &&
3811 0 : (ftmachine_p != "wbawp") && (ftmachine_p != "nift") &&
3812 0 : (ftmachine_p != "mosaic") && (ftmachine_p != "awproject") ) {
3813 : os << LogIO::SEVERE
3814 : << "Multi-scale Multi-frequency Clean currently works only with ft, wproject and mosaic (and wbawp,nift,awproject)"
3815 0 : << LogIO::POST;
3816 0 : return retval;
3817 : }
3818 :
3819 0 : useNewMTFT_p=false;
3820 0 : if( ftmachine_p == "awproject" ) { useNewMTFT_p=true; }
3821 :
3822 :
3823 0 : if (!scaleInfoValid_p) {
3824 0 : this->unlock();
3825 0 : os << LogIO::WARN << "Scales not yet set, using power law" << LogIO::POST;
3826 0 : sm_p = new WBCleanImageSkyModel(ntaylor_p, 1 ,reffreq_p);
3827 : }
3828 0 : if (scaleMethod_p=="uservector") {
3829 0 : sm_p = new WBCleanImageSkyModel(ntaylor_p,userScaleSizes_p,reffreq_p);
3830 : } else {
3831 0 : sm_p = new WBCleanImageSkyModel(ntaylor_p,nscales_p,reffreq_p);
3832 : }
3833 : os << LogIO::NORMAL // Loglevel INFO
3834 0 : << "Using multi frequency synthesis algorithm" << LogIO::POST;
3835 0 : ((WBCleanImageSkyModel*)sm_p)->imageNames = Vector<String>(image);
3836 : /* Check masks. Should be only one per field. Duplicate the name ntaylor_p times
3837 : Note : To store taylor-coefficients, msmfs uses the same data structure as for
3838 : multi-field imaging. In the case of multifield and msmfs, the list of
3839 : images is nested and follows a field-major ordering.
3840 : All taylor-coeffs for a single field should have the same mask (for now).
3841 : For now, since only single-field is allowed for msmfs, we have the following.*/
3842 : }
3843 : else {
3844 0 : this->unlock();
3845 : os << LogIO::SEVERE << "Unknown algorithm: " << algorithm
3846 0 : << LogIO::POST;
3847 :
3848 0 : return retval;
3849 : }
3850 :
3851 0 : AlwaysAssert(sm_p, AipsError);
3852 0 : sm_p->setAlgorithm("clean");
3853 :
3854 : // if (!se_p)
3855 0 : if(!createSkyEquation(modelNames, fixed, maskNames, complist))
3856 : {
3857 :
3858 0 : return retval;
3859 : }
3860 0 : os << LogIO::NORMAL3 << "Created Sky Equation" << LogIO::POST;
3861 0 : }
3862 : else{
3863 : //adding or modifying mask associated with skyModel
3864 0 : addMasksToSkyEquation(maskNames,fixed);
3865 : }
3866 : //No need to add residuals will let sm_p use tmpimage ones and we'll copy them in restore
3867 0 : if(!addResiduals(residualNames))
3868 0 : throw(AipsError("Problem in attaching to residual images"));
3869 : // The old plot that showed how much flux was being incorporated in each
3870 : // scale. No longer available, slated for removal.
3871 : // if (displayProgress) {
3872 : // sm_p->setDisplayProgress(true);
3873 : // sm_p->setPGPlotter( getPGPlotter() );
3874 : // }
3875 :
3876 :
3877 :
3878 0 : sm_p->setGain(gain);
3879 0 : sm_p->setNumberIterations(niter);
3880 0 : sm_p->setThreshold(threshold.get("Jy").getValue());
3881 0 : sm_p->setCycleFactor(cyclefactor_p);
3882 0 : sm_p->setCycleSpeedup(cyclespeedup_p);
3883 0 : sm_p->setCycleMaxPsfFraction(cyclemaxpsffraction_p);
3884 : {
3885 0 : ostringstream oos;
3886 0 : oos << "Clean gain = " <<gain<<", Niter = "<<niter<<", Threshold = "
3887 0 : << threshold;
3888 0 : os << LogIO::NORMAL << String(oos) << LogIO::POST; // More for the
3889 : // logfile than the
3890 : // log window.
3891 0 : }
3892 :
3893 : os << LogIO::NORMAL << (firstrun ? "Start" : "Continu")
3894 0 : << "ing deconvolution" << LogIO::POST; // Loglevel PROGRESS
3895 0 : if(se_p->solveSkyModel()) {
3896 : os << LogIO::NORMAL
3897 : << (niter == 0 ? "Image OK" : "Successfully deconvolved image")
3898 0 : << LogIO::POST; // Loglevel PROGRESS
3899 : }
3900 : else {
3901 0 : converged=false;
3902 0 : os << LogIO::NORMAL << "Threshhold not reached yet." << LogIO::POST; // Loglevel PROGRESS
3903 : }
3904 :
3905 :
3906 0 : printbeam(sm_p, os, firstrun);
3907 :
3908 0 : if(((algorithm.substr(0,5)=="clark") || algorithm=="hogbom" ||
3909 0 : algorithm=="multiscale") && (niter != 0))
3910 : //write the model visibility to ms for now
3911 0 : sm_p->solveResiduals(*se_p, true);
3912 :
3913 0 : for (uInt k=0 ; k < residuals_p.nelements(); ++k){
3914 0 : (residuals_p[k])->copyData(sm_p->getResidual(k));
3915 : }
3916 : /////////////
3917 :
3918 : //cerr << "Time taken " << omp_get_wtime()-wtime0 << endl;
3919 : ////////////
3920 0 : retval.define("maxresidual", (sm_p->threshold()));
3921 0 : retval.define("iterations", (sm_p->numberIterations()));
3922 0 : retval.define("converged", converged);
3923 0 : savePSF(psfnames);
3924 0 : redoSkyModel_p=false;
3925 0 : writeFluxScales(fluxscale_p);
3926 : // restoreImages(image); // Moved to iClean so that it happens only once.
3927 :
3928 :
3929 0 : this->unlock();
3930 :
3931 0 : return retval;
3932 0 : }
3933 0 : catch (PSFZero& x)
3934 : {
3935 : //os << LogIO::WARN << x.what() << LogIO::POST;
3936 0 : savePSF(psfnames);
3937 0 : this->unlock();
3938 0 : throw(AipsError(String("PSFZero ")+ x.getMesg() + String(" : Please check that the required data exists and is not flagged.")));
3939 : return retval;
3940 0 : }
3941 0 : catch (exception &x) {
3942 0 : this->unlock();
3943 0 : destroySkyEquation();
3944 0 : throw(AipsError(x.what()));
3945 :
3946 : return retval;
3947 0 : }
3948 :
3949 0 : catch(...){
3950 0 : this->unlock();
3951 0 : destroySkyEquation();
3952 : //Unknown exception...
3953 0 : throw(AipsError("Unknown exception caught ...imager/casa may need to be exited"));
3954 0 : }
3955 : this->unlock();
3956 :
3957 : os << LogIO::NORMAL << "Exiting Imager::clean" << LogIO::POST; // Loglevel PROGRESS
3958 : return retval;
3959 0 : }
3960 :
3961 : // Mem algorithm
3962 0 : Bool Imager::mem(const String& algorithm,
3963 : const Int niter,
3964 : const Quantity& sigma,
3965 : const Quantity& targetFlux,
3966 : const Bool constrainFlux,
3967 : const Bool displayProgress,
3968 : const Vector<String>& model,
3969 : const Vector<Bool>& fixed,
3970 : const String& complist,
3971 : const Vector<String>& prior,
3972 : const Vector<String>& mask,
3973 : const Vector<String>& image,
3974 : const Vector<String>& residual)
3975 : {
3976 0 : if(!valid())
3977 : {
3978 0 : return false;
3979 : }
3980 0 : logSink_p.clearLocally();
3981 0 : LogIO os(LogOrigin("imager", "mem()"), logSink_p);
3982 :
3983 0 : this->lock();
3984 : try {
3985 0 : if(!assertDefinedImageParameters())
3986 : {
3987 0 : return false;
3988 : }
3989 0 : os << LogIO::NORMAL << "Deconvolving images with MEM" << LogIO::POST; // Loglevel PROGRESS
3990 :
3991 0 : Int nmodels=model.nelements();
3992 : os << LogIO::NORMAL // Loglevel INFO
3993 0 : << "Found " << nmodels << " specified model images" << LogIO::POST;
3994 :
3995 0 : if(model.nelements()>0) {
3996 0 : for (uInt thismodel=0;thismodel<model.nelements();++thismodel) {
3997 0 : if(model(thismodel)=="") {
3998 0 : this->unlock();
3999 : os << LogIO::SEVERE << "Need a name for model "
4000 0 : << thismodel << LogIO::POST;
4001 0 : return false;
4002 : }
4003 : }
4004 : }
4005 :
4006 0 : Vector<String> modelNames=model;
4007 : // Make first image with the required shape and coordinates only if
4008 : // it doesn't exist yet. Otherwise we'll throw an exception later
4009 0 : if(modelNames(0)=="") modelNames(0)=imageName()+".mem";
4010 0 : if(!Table::isWritable(modelNames(0))) {
4011 0 : make(modelNames(0));
4012 : }
4013 :
4014 0 : Vector<String> maskNames(nmodels);
4015 0 : if(Int(mask.nelements())==nmodels) {
4016 0 : maskNames=mask;
4017 0 : for(Int k=0; k < nmodels; ++k){
4018 0 : if(mask(k)!=""&& !Table::isReadable(mask(k))) {
4019 : os << LogIO::WARN
4020 : << "Mask" << mask(k)
4021 : << " is unreadable; ignoring masks altogether "
4022 0 : << LogIO::POST;
4023 0 : maskNames.resize(1);
4024 0 : maskNames(0)="";
4025 : }
4026 : }
4027 : }
4028 : else {
4029 0 : maskNames.resize(1);
4030 0 : maskNames(0)="";
4031 : }
4032 :
4033 : // Always fill in the residual images
4034 0 : Vector<String> residualNames(nmodels);
4035 0 : if(Int(residual.nelements())==nmodels) {
4036 0 : residualNames=residual;
4037 : }
4038 : else {
4039 0 : residualNames="";
4040 : }
4041 0 : for (Int thismodel=0;thismodel<Int(model.nelements());++thismodel) {
4042 0 : if(residualNames(thismodel)=="") {
4043 0 : residualNames(thismodel)=modelNames(thismodel)+".residual";
4044 : }
4045 0 : removeTable(residualNames(thismodel));
4046 0 : if(!clone(model(thismodel), residualNames(thismodel)))
4047 : {
4048 0 : return false;
4049 : }
4050 : }
4051 :
4052 : // Make an ImageSkyModel with the specified polarization representation
4053 : // (i.e. circular or linear)
4054 0 : if(algorithm=="entropy") {
4055 0 : sm_p = new CEMemImageSkyModel(sigma.get("Jy").getValue(),
4056 0 : targetFlux.get("Jy").getValue(),
4057 : constrainFlux,
4058 : prior,
4059 0 : algorithm);
4060 : os << LogIO::NORMAL // Loglevel INFO
4061 0 : << "Using single-field algorithm with Maximum Entropy" << LogIO::POST;
4062 0 : if(ftmachine_p=="mosaic" ||ftmachine_p=="wproject" )
4063 0 : sm_p->setSubAlgorithm("full");
4064 : }
4065 0 : else if (algorithm=="emptiness") {
4066 0 : sm_p = new CEMemImageSkyModel(sigma.get("Jy").getValue(),
4067 0 : targetFlux.get("Jy").getValue(),
4068 : constrainFlux,
4069 : prior,
4070 0 : algorithm);
4071 : os << LogIO::NORMAL // Loglevel INFO
4072 0 : << "Using single-field algorithm with Maximum Emptiness" << LogIO::POST;
4073 0 : if(ftmachine_p=="mosaic" ||ftmachine_p=="wproject" )
4074 0 : sm_p->setSubAlgorithm("full");
4075 : }
4076 0 : else if (algorithm=="mfentropy") {
4077 0 : sm_p = new MFCEMemImageSkyModel(sigma.get("Jy").getValue(),
4078 0 : targetFlux.get("Jy").getValue(),
4079 : constrainFlux,
4080 : prior,
4081 0 : algorithm);
4082 0 : doMultiFields_p = true;
4083 0 : os << LogIO::NORMAL << "Using Maximum Entropy" << LogIO::POST; // Loglevel INFO
4084 : // if(ftmachine_p=="mosaic" ||ftmachine_p=="wproject" )
4085 0 : sm_p->setSubAlgorithm("full");
4086 0 : } else if (algorithm=="mfemptiness") {
4087 0 : sm_p = new MFCEMemImageSkyModel(sigma.get("Jy").getValue(),
4088 0 : targetFlux.get("Jy").getValue(),
4089 : constrainFlux,
4090 : prior,
4091 0 : algorithm);
4092 0 : doMultiFields_p = true;
4093 0 : os << LogIO::NORMAL << "Using Maximum Emptiness" << LogIO::POST; // Loglevel INFO
4094 : // if(ftmachine_p=="mosaic" ||ftmachine_p=="wproject" )
4095 0 : sm_p->setSubAlgorithm("full");
4096 : } else {
4097 0 : this->unlock();
4098 0 : os << LogIO::SEVERE << "Unknown algorithm: " << algorithm << LogIO::POST;
4099 0 : return false;
4100 : }
4101 0 : AlwaysAssert(sm_p, AipsError);
4102 0 : sm_p->setAlgorithm("mem");
4103 0 : if (displayProgress) {
4104 0 : sm_p->setDisplayProgress(true);
4105 : }
4106 0 : sm_p->setNumberIterations(niter);
4107 0 : sm_p->setCycleFactor(cyclefactor_p); // used by mf algs
4108 0 : sm_p->setCycleSpeedup(cyclespeedup_p); // used by mf algs
4109 0 : sm_p->setCycleMaxPsfFraction(cyclemaxpsffraction_p); // used by mf algs
4110 :
4111 : {
4112 0 : ostringstream oos;
4113 0 : oos << "MEM algorithm = " <<algorithm<<", Niter = "<<niter<<", Sigma = "
4114 0 : <<sigma << ", Target Flux = " << targetFlux;
4115 0 : os << LogIO::DEBUG1 << String(oos) << LogIO::POST;
4116 0 : }
4117 :
4118 : // if (!se_p)
4119 0 : if(!createSkyEquation(modelNames, fixed, maskNames, complist))
4120 : {
4121 0 : return false;
4122 : }
4123 0 : os << LogIO::NORMAL3 << "Created Sky Equation" << LogIO::POST;
4124 :
4125 0 : addResidualsToSkyEquation(residualNames);
4126 :
4127 0 : os << LogIO::NORMAL << "Starting deconvolution" << LogIO::POST; // Loglevel PROGRESS
4128 0 : if(se_p->solveSkyModel()) {
4129 0 : os << LogIO::NORMAL << "Successfully deconvolved image" << LogIO::POST; // Loglevel INFO
4130 : }
4131 : else {
4132 0 : os << LogIO::NORMAL << "Nominally failed deconvolution" << LogIO::POST; // Loglevel INFO
4133 : }
4134 :
4135 : // Get the PSF fit while we are here
4136 0 : if(!beamValid_p){
4137 0 : ImageBeamSet beam=sm_p->beam(0);
4138 0 : if(beam.nelements() > 0){
4139 : /*beam_p.setMajorMinor(
4140 : Quantity(abs(beam(0)), "arcsec"), Quantity(abs(beam(1)), "arcsec")
4141 : );
4142 : beam_p.setPA(Quantity(beam(2), "deg"));
4143 : */
4144 0 : beam_p=beam;
4145 0 : beamValid_p=true;
4146 : }
4147 0 : }
4148 0 : if(algorithm=="entropy" || algorithm=="emptiness" )
4149 0 : sm_p->solveResiduals(*se_p, true);
4150 0 : writeFluxScales(fluxscale_p);
4151 0 : restoreImages(image);
4152 0 : destroySkyEquation();
4153 0 : this->writeHistory(os);
4154 : try{
4155 : { // write data processing history into image logtable
4156 0 : LoggerHolder imagelog (false);
4157 0 : LogSink& sink = imagelog.sink();
4158 0 : LogOrigin lor( String("imager"), String("mem()") );
4159 0 : LogMessage msg(lor);
4160 0 : sink.postLocally(msg);
4161 0 : MSHistoryColumns msHis(ms_p->history());
4162 0 : transferHistory(imagelog, msHis);
4163 0 : for (Int thismodel=0;thismodel<Int(model.nelements());++thismodel) {
4164 : PagedImage<Float> restoredImage(image(thismodel),
4165 0 : TableLock(TableLock::UserLocking));
4166 0 : LoggerHolder& log = restoredImage.logger();
4167 0 : log.append(imagelog);
4168 0 : log.flush();
4169 0 : }
4170 0 : }
4171 : }
4172 0 : catch(exception& x){
4173 :
4174 0 : os << LogIO::WARN << "Caught exception: " << x.what()
4175 0 : << LogIO::POST;
4176 : os << LogIO::SEVERE << "This means your MS/HISTORY table may be corrupted; you may consider deleting all the rows from this table"
4177 0 : <<LogIO::POST;
4178 : //continue and wrap up this function as normal
4179 :
4180 0 : }
4181 0 : catch(...){
4182 : //Unknown exception...
4183 0 : throw(AipsError("Unknown exception caught ...imager/casa may need to be exited"));
4184 0 : }
4185 :
4186 0 : this->unlock();
4187 :
4188 0 : return true;
4189 0 : } catch (exception& x) {
4190 0 : this->unlock();
4191 0 : throw(AipsError(x.what()));
4192 :
4193 : return false;
4194 0 : }
4195 : this->unlock();
4196 : return true;
4197 :
4198 0 : }
4199 :
4200 :
4201 : // NNLS algorithm
4202 0 : Bool Imager::nnls(const String&, const Int niter, const Float tolerance,
4203 : const Vector<String>& model, const Vector<Bool>& fixed,
4204 : const String& complist,
4205 : const Vector<String>& fluxMask,
4206 : const Vector<String>& dataMask,
4207 : const Vector<String>& residual,
4208 : const Vector<String>& image)
4209 : {
4210 0 : if(!valid()) return false;
4211 0 : LogIO os(LogOrigin("imager", "nnls()", WHERE));
4212 :
4213 0 : this->lock();
4214 : try {
4215 0 : if(!assertDefinedImageParameters()) return false;
4216 :
4217 0 : os << LogIO::NORMAL << "Performing NNLS deconvolution" << LogIO::POST; // Loglevel PROGRESS
4218 :
4219 0 : if(niter<0) {
4220 0 : this->unlock();
4221 0 : os << LogIO::SEVERE << "Number of iterations must be positive" << LogIO::POST;
4222 0 : return false;
4223 : }
4224 0 : if(tolerance<0.0) {
4225 0 : this->unlock();
4226 0 : os << LogIO::SEVERE << LogIO::SEVERE << "Tolerance must be positive" << LogIO::POST;
4227 0 : return false;
4228 : }
4229 :
4230 : // Add the images to the ImageSkyModel
4231 0 : Int nmodels=model.nelements();
4232 0 : if(nmodels>1) os<< "Can only process one model" << LogIO::POST;
4233 :
4234 0 : if(model(0)=="") {
4235 0 : this->unlock();
4236 0 : os << LogIO::SEVERE << "Need a name for model " << LogIO::POST;
4237 0 : return false;
4238 : }
4239 :
4240 0 : if(!Table::isWritable(model(0))) {
4241 0 : make(model(0));
4242 0 : this->lock();
4243 : }
4244 :
4245 : // Always fill in the residual images
4246 0 : Vector<String> residualNames(nmodels);
4247 0 : if(Int(residual.nelements())==nmodels) {
4248 0 : residualNames=residual;
4249 : }
4250 : else {
4251 0 : residualNames="";
4252 : }
4253 0 : for (Int thismodel=0;thismodel<Int(model.nelements());++thismodel) {
4254 0 : if(residualNames(thismodel)=="") {
4255 0 : residualNames(thismodel)=model(thismodel)+".residual";
4256 : }
4257 0 : removeTable(residualNames(thismodel));
4258 0 : if(!clone(model(thismodel), residualNames(thismodel))) return false;
4259 : }
4260 :
4261 : // Now make the NNLS ImageSkyModel
4262 0 : sm_p= new NNLSImageSkyModel();
4263 0 : sm_p->setNumberIterations(niter);
4264 0 : sm_p->setTolerance(tolerance);
4265 0 : sm_p->setAlgorithm("nnls");
4266 : os << LogIO::DEBUG1
4267 0 : << "NNLS Niter = " << niter << ", Tolerance = " << tolerance << LogIO::POST;
4268 :
4269 : // if (!se_p)
4270 0 : if(!createSkyEquation(model, fixed, dataMask, fluxMask, complist)) return false;
4271 :
4272 0 : addResidualsToSkyEquation(residualNames);
4273 :
4274 0 : os << LogIO::NORMAL << "Starting deconvolution" << LogIO::POST; // Loglevel PROGRESS
4275 :
4276 0 : if(se_p->solveSkyModel()) {
4277 0 : os << LogIO::NORMAL << "Successfully deconvolved image" << LogIO::POST; // Loglevel INFO
4278 : }
4279 : else {
4280 0 : os << LogIO::NORMAL << "Nominally failed deconvolution" << LogIO::POST; // Loglevel INFO
4281 : }
4282 :
4283 : // Get the PSF fit while we are here
4284 0 : StokesImageUtil::FitGaussianPSF(sm_p->PSF(0), beam_p);
4285 0 : beamValid_p=true;
4286 :
4287 :
4288 : // Restore the image
4289 0 : restoreImages(image);
4290 :
4291 0 : destroySkyEquation();
4292 0 : this->unlock();
4293 0 : return true;
4294 0 : } catch (AipsError x) {
4295 0 : this->unlock();
4296 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
4297 0 : return false;
4298 0 : }
4299 : this->unlock();
4300 : return true;
4301 0 : }
4302 :
4303 : // Fourier transform the model and componentlist
4304 0 : Bool Imager::ft(const Vector<String>& model, const String& complist,
4305 : const Bool incremental, const Double phaseCenTime)
4306 : {
4307 0 : if(!valid()) return false;
4308 :
4309 0 : LogIO os(LogOrigin("imager", "ft()", WHERE));
4310 :
4311 0 : if (wvi_p==NULL)
4312 0 : os << LogIO::WARN << "Please make sure MS is writable when using Imager::ft" << LogIO::EXCEPTION;
4313 :
4314 0 : this->lock();
4315 : try {
4316 :
4317 0 : if(!redoSkyModel_p){
4318 : //let us try to update the sm_p then
4319 : //so as to keep the state and psf's etc if they have been calculated
4320 : //useful when cleaning, modify/clip model then predict, selfcal and clean again
4321 0 : if(!updateSkyModel(model, complist))
4322 0 : destroySkyEquation();
4323 : }
4324 :
4325 : os << LogIO::NORMAL // Loglevel INFO
4326 0 : << String("Fourier transforming: ") +
4327 0 : (incremental ? String("adding to "): String("replacing "))+
4328 0 : (useModelCol_p ? String("MODEL_DATA column") : String("visibility model header")) << LogIO::POST;
4329 :
4330 0 : if (redoSkyModel_p){
4331 0 : if(!createSkyEquation(model, complist)) return false;
4332 : }
4333 0 : if(incremental){
4334 0 : for (Int mod=0; mod < (sm_p->numberOfModels()); ++mod){
4335 0 : (sm_p->deltaImage(mod)).copyData(sm_p->image(mod));
4336 : }
4337 : }
4338 0 : se_p->setPhaseCenterTime(phaseCenTime);
4339 0 : se_p->predict(incremental);
4340 :
4341 : // destroySkyEquation();
4342 :
4343 0 : this->unlock();
4344 0 : return true;
4345 0 : } catch (AipsError x) {
4346 0 : this->unlock();
4347 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
4348 0 : return false;
4349 0 : }
4350 : this->unlock();
4351 : return true;
4352 0 : }
4353 :
4354 0 : Bool Imager::setjy(const Int fieldid,
4355 : const Int spectralwindowid,
4356 : const Vector<Double>& fluxDensity, const String& standard)
4357 : {
4358 : // the old interface to new interface
4359 0 : String fieldnames="";
4360 0 : String spwstring="";
4361 0 : Vector<Int>fldids(1,fieldid);
4362 0 : Vector<Int>spwids(1,spectralwindowid);
4363 0 : return setjy(fldids, spwids, fieldnames, spwstring, fluxDensity, standard);
4364 :
4365 0 : }
4366 :
4367 0 : Bool Imager::setjy(const Vector<Int>& /*fieldid*/,
4368 : const Vector<Int>& /*spectralwindowid*/,
4369 : const String& fieldnames, const String& spwstring,
4370 : const Vector<Double>& fluxDensity, const String& standard)
4371 : {
4372 0 : if(!valid()) return false;
4373 0 : logSink_p.clearLocally();
4374 0 : LogIO os(LogOrigin("imager", "setjy()"), logSink_p);
4375 0 : this->lock();
4376 :
4377 0 : String tempCL;
4378 : try {
4379 0 : Bool precompute=(fluxDensity(0) <= 0);
4380 :
4381 : // Figure out which fields/spws to treat
4382 0 : Record selrec=ms_p->msseltoindex(spwstring, fieldnames);
4383 0 : Vector<Int> fldids(selrec.asArrayInt("field"));
4384 0 : Vector<Int> spwids(selrec.asArrayInt("spw"));
4385 :
4386 0 : expand_blank_sel(spwids, ms_p->spectralWindow().nrow());
4387 0 : expand_blank_sel(fldids, ms_p->field().nrow());
4388 :
4389 : // Loop over field id. and spectral window id.
4390 0 : Vector<Double> fluxUsed(4);
4391 0 : String fluxScaleName;
4392 0 : Bool matchedScale=false;
4393 : Int spwid, fldid;
4394 0 : MSColumns msc(*ms_p);
4395 0 : ConstantSpectrum cspectrum;
4396 : // TT
4397 0 : Double meantime = msc.time()(0);
4398 0 : meantime += 0.5 * (msc.time()(msc.nrow() - 1) - meantime);
4399 0 : MEpoch mtime(msc.timeMeas()(0));
4400 0 : mtime.set(Quantity(meantime, "s"));
4401 :
4402 :
4403 0 : for (uInt kk=0; kk<fldids.nelements(); ++kk) {
4404 0 : fldid=fldids[kk];
4405 : // Extract field name and field center position
4406 0 : MDirection position=msc.field().phaseDirMeas(fldid, meantime);
4407 0 : String fieldName=msc.field().name()(fldid);
4408 :
4409 0 : for (uInt jj=0; jj< spwids.nelements(); ++jj) {
4410 0 : spwid=spwids[jj];
4411 :
4412 : // Determine spectral window center frequency
4413 0 : IPosition ipos(1,0);
4414 0 : MFrequency mfreq=msc.spectralWindow().chanFreqMeas()(spwid)(ipos);
4415 0 : Array<Double> freqArray;
4416 0 : msc.spectralWindow().chanFreq().get(spwid, freqArray, true);
4417 0 : Double medianFreq=median(freqArray);
4418 0 : mfreq.set(MVFrequency(medianFreq));
4419 :
4420 0 : fluxUsed=fluxDensity;
4421 0 : fluxScaleName="user-specified";
4422 0 : if (precompute) {
4423 : // Pre-compute flux density for standard sources if not specified
4424 : // using the specified flux scale standard or catalog.
4425 :
4426 :
4427 : FluxStandard::FluxScale fluxScaleEnum;
4428 0 : matchedScale=FluxStandard::matchStandard(standard, fluxScaleEnum,
4429 : fluxScaleName);
4430 : (void)matchedScale;
4431 0 : FluxStandard fluxStd(fluxScaleEnum);
4432 0 : Flux<Double> returnFlux, returnFluxErr;
4433 :
4434 0 : if (fluxStd.compute(fieldName, position, mfreq, mtime, returnFlux, returnFluxErr)) {
4435 : // Standard reference source identified
4436 0 : returnFlux.value(fluxUsed);
4437 : }
4438 :
4439 : // dgoscha, NCSA, 02 May, 2002
4440 : // this else condtion is to handle the case where the user
4441 : // specifies standard='SOURCE' in the setjy argument. This will
4442 : // then look into the SOURCE_MODEL column of the SOURCE subtable
4443 : // for a table-record entry that points to a component list with the
4444 : // model information in it.
4445 :
4446 :
4447 0 : else if (standard==String("SOURCE")) {
4448 : // Look in the SOURCE_MODEL column of the SOURCE subtable for
4449 : // the name of the CL which contains the model.
4450 :
4451 : // First test to make sure the SOURCE_MODEL column exists.
4452 0 : if (ms_p->source().tableDesc().isColumn("SOURCE_MODEL")) {
4453 0 : TableRecord modelRecord;
4454 0 : msc.source().sourceModel().get(0, modelRecord);
4455 :
4456 : // Get the name of the model component list from the table record
4457 : Table modelRecordTable =
4458 0 : modelRecord.asTable(modelRecord.fieldNumber(String ("model")));
4459 0 : String modelCLName = modelRecordTable.tableName();
4460 0 : modelRecord.closeTable(modelRecord.fieldNumber(String ("model")));
4461 :
4462 : // Now grab the flux from the model component list and use.
4463 0 : ComponentList modelCL = ComponentList(Path(modelCLName), true);
4464 0 : SkyComponent fluxComponent = modelCL.component(fldid);
4465 :
4466 0 : fluxUsed = 0;
4467 0 : fluxUsed = real(fluxComponent.flux().value());
4468 0 : fluxScaleName = modelCLName;
4469 0 : }
4470 : else {
4471 : os << LogIO::SEVERE << "Missing SOURCE_MODEL column."
4472 : << LogIO::SEVERE << "Using default, I=1.0"
4473 0 : << LogIO::POST;
4474 0 : fluxUsed = 0;
4475 0 : fluxUsed(0) = 1.0;
4476 : }
4477 : }
4478 :
4479 : else {
4480 : // Source not found; use Stokes I=1.0 Jy for now
4481 0 : fluxUsed=0;
4482 0 : fluxUsed(0)=1.0;
4483 0 : fluxScaleName="default";
4484 : };
4485 0 : }
4486 :
4487 : // Set the component flux density
4488 0 : Flux<Double> fluxval;
4489 0 : fluxval.setValue(fluxUsed);
4490 :
4491 : // Create a point component at the field center
4492 : // with the specified flux density
4493 0 : PointShape point(position);
4494 0 : SkyComponent skycomp(fluxval, point, cspectrum);
4495 :
4496 : // Create a component list containing this entry
4497 0 : String baseString=msname_p + "." + fieldName + ".spw" +
4498 0 : String::toString(spwid);
4499 0 : tempCL=baseString + ".tempcl";
4500 :
4501 : // Force a call to the ComponentList destructor
4502 : // using scoping rules.
4503 : {
4504 0 : ComponentList cl;
4505 0 : cl.add(skycomp);
4506 0 : cl.rename(tempCL, Table::New);
4507 0 : }
4508 :
4509 : // Select the uv-data for this field and spw. id.;
4510 : // all frequency channels selected.
4511 0 : Vector<Int> selectSpw(1), selectField(1);
4512 0 : selectSpw(0)=spwid;
4513 0 : selectField(0)=fldid;
4514 0 : String msSelectString = "";
4515 0 : Vector<Int> numDeChan(1);
4516 0 : numDeChan[0]=0;
4517 0 : Vector<Int> begin(1);
4518 0 : begin[0]=0;
4519 0 : Vector<Int> stepsize(1);
4520 0 : stepsize[0]=1;
4521 0 : setdata("channel", numDeChan, begin, stepsize, MRadialVelocity(),
4522 0 : MRadialVelocity(),
4523 0 : selectSpw, selectField, msSelectString, "", "", Vector<Int>(),
4524 : "", "", "", "", "", "",true);
4525 :
4526 0 : if (!nullSelect_p) {
4527 :
4528 : // Transform the component model table
4529 0 : Vector<String> model;
4530 0 : ft(model, tempCL, false);
4531 :
4532 : // Log flux density used for this field and spectral window
4533 0 : os.output().width(12);
4534 0 : os << fieldName << " spwid=";
4535 0 : os.output().width(3);
4536 0 : os << (spwid) << " ";
4537 0 : os.output().width(0);
4538 0 : os.output().precision(4);
4539 0 : os << LogIO::NORMAL << "[I=" << fluxUsed(0) << ", "; // Loglevel INFO
4540 0 : os << "Q=" << fluxUsed(1) << ", ";
4541 0 : os << "U=" << fluxUsed(2) << ", ";
4542 0 : os << "V=" << fluxUsed(3) << "] Jy, ";
4543 0 : os << ("(" + fluxScaleName + ")") << LogIO::POST;
4544 0 : };
4545 :
4546 : // Delete the temporary component list and image tables
4547 0 : TableUtil::deleteTable(tempCL);
4548 :
4549 0 : }
4550 0 : }
4551 0 : this->writeHistory(os);
4552 0 : this->unlock();
4553 0 : return true;
4554 :
4555 0 : } catch (AipsError x) {
4556 0 : this->unlock();
4557 0 : if(TableUtil::canDeleteTable(tempCL)) TableUtil::deleteTable(tempCL);
4558 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
4559 0 : return false;
4560 0 : }
4561 : return true;
4562 0 : }
4563 :
4564 : // This is the one used by im.setjy() (because it has a model arg).
4565 : // CURRENT SETJY CODE
4566 0 : Record Imager::setjy(const Vector<Int>& /*fieldid*/,
4567 : const Vector<Int>& /*spectralwindowid*/,
4568 : const String& fieldnames, const String& spwstring,
4569 : const String& model,
4570 : const Vector<Double>& fluxDensity,
4571 : const String& standard, const Bool chanDep,
4572 : //const Double spix, const MFrequency& reffreq,
4573 : const Vector<Double>& spix, const MFrequency& reffreq,
4574 : const Vector<Double>& pipars,const Vector<Double>& papars,
4575 : const Double& rotMeas,
4576 : const String& timerange, const String& scanstr,
4577 : const String& intentstr, const String& obsidstr,
4578 : const String& interpolation)
4579 : {
4580 : //if(!valid())
4581 : //return false;
4582 :
4583 : //Bool didAnything = false;
4584 :
4585 0 : Record retval;
4586 0 : if(!valid()) {
4587 0 : retval.define("process",false);
4588 0 : return retval;
4589 : }
4590 :
4591 0 : logSink_p.clearLocally();
4592 0 : LogIO os(LogOrigin("imager", "setjy()"), logSink_p);
4593 0 : this->lock();
4594 :
4595 : // user specified flux densities (IQUV), global to the spws and fields
4596 0 : Vector<Double> fluxdens = fluxDensity;
4597 0 : if(fluxDensity.nelements() < 4){
4598 0 : fluxdens.resize(4,true);
4599 0 : for(Int i = fluxDensity.nelements(); i < 4; ++i)
4600 0 : fluxdens[i] = 0.0;
4601 : }
4602 :
4603 0 : Vector<String> tempCLs;
4604 0 : TempImage<Float> *tmodimage(NULL);
4605 :
4606 : try{
4607 0 : Bool precompute = fluxdens[0] < 0.0;
4608 :
4609 : // Figure out which fields/spws to treat
4610 : // including intent info
4611 0 : MSSelection mssel;
4612 0 : mssel.setFieldExpr(fieldnames);
4613 0 : mssel.setSpwExpr(spwstring);
4614 0 : mssel.setStateExpr(intentstr);
4615 0 : TableExprNode exprNode = mssel.toTableExprNode(&(*ms_p));
4616 : //Vector<Int> fldids;
4617 0 : Vector<Int> fldids(mssel.getFieldList());
4618 0 : Vector<Int> selToRawSpwIds(mssel.getSpwList());
4619 : // if intent is given try to do AND with fieldIds
4620 0 : if (intentstr!="") {
4621 0 : mssel_p = new MeasurementSet((*ms_p)(exprNode), &(*ms_p));
4622 0 : MSColumns tmpmsc(*mssel_p);
4623 0 : Vector<Int> fldidv=tmpmsc.fieldId().getColumn();
4624 0 : if (fldidv.nelements()==0)
4625 0 : throw(AipsError("No field ids were selected, please check input parameters"));
4626 0 : std::set<Int> ufldids(fldidv.begin(),fldidv.end());
4627 0 : std::vector<Int> tmpv(ufldids.begin(), ufldids.end());
4628 0 : fldids.resize(tmpv.size());
4629 0 : uInt count=0;
4630 0 : for (std::vector<int>::const_iterator it=tmpv.begin();it != tmpv.end(); it++)
4631 : {
4632 0 : fldids(count) = *it;
4633 0 : count++;
4634 : }
4635 0 : }
4636 : //else {
4637 : // fldids(mssel.getFieldList());
4638 : //}
4639 : //cerr<<"fldids.nelements()="<<fldids.nelements()<<endl;
4640 : //for (uInt i = 0; i < fldids.nelements(); i++) {
4641 : // cerr<<"fldids="<<fldids(i)<<endl;
4642 : //}
4643 : //Record selrec = ms_p->msseltoindex(spwstring, fieldnames);
4644 : //Vector<Int> fldids(selrec.asArrayInt("field"));
4645 : //Vector<Int> selToRawSpwIds(selrec.asArrayInt("spw"));
4646 :
4647 0 : expand_blank_sel(selToRawSpwIds, ms_p->spectralWindow().nrow());
4648 0 : expand_blank_sel(fldids, ms_p->field().nrow());
4649 :
4650 : // Warn against multiple fields in some circumstances.
4651 0 : if (fldids.nelements() > 1 && (model != "" || !precompute)) {
4652 0 : String errmsg("setjy is applying a single ");
4653 :
4654 0 : if(model != ""){
4655 0 : errmsg += "modimage";
4656 0 : if(!precompute)
4657 0 : errmsg += " or ";
4658 : }
4659 :
4660 0 : if(!precompute)
4661 0 : errmsg += "fluxdensity";
4662 :
4663 0 : errmsg += " to multiple fields!\n";
4664 : os << LogIO::WARN
4665 : << errmsg
4666 : << "This could be a user error, but sometimes a single name will\n"
4667 : << "resolve to > 1 field index.\n"
4668 0 : << LogIO::POST;
4669 : //throw(AipsError(errmsg));
4670 0 : }
4671 :
4672 0 : os << LogIO::NORMAL;
4673 0 : if(precompute || spix[0] != 0.0)
4674 0 : os << "Using " << ((chanDep || (!precompute && spix[0] != 0.0)) ? "channel" :
4675 0 : "spw") << " dependent flux densities";
4676 : else
4677 0 : os << "The applied flux density does not depend on frequency.";
4678 0 : os << LogIO::POST;
4679 :
4680 : // Ignore user polarization if using an image.
4681 0 : if(model != "" &&
4682 0 : (fluxdens[1] != 0.0 || fluxdens[2] != 0.0 || fluxdens[3] != 0.0)){
4683 : os << LogIO::WARN
4684 : << "Using model image, so zeroing user QUV flux densities."
4685 0 : << LogIO::POST;
4686 0 : fluxdens[1] = fluxdens[2] = fluxdens[3] = 0.0;
4687 0 : writeHistory(os);
4688 : }
4689 :
4690 : // Loop over field id. and spectral window id.
4691 : //Vector<Double> fluxUsed(4);
4692 0 : String fluxScaleName("user-specified");
4693 : FluxStandard::FluxScale fluxScaleEnum;
4694 0 : if(!FluxStandard::matchStandard(standard, fluxScaleEnum, fluxScaleName))
4695 0 : throw(AipsError(standard + " is not a recognized flux density scale"));
4696 :
4697 0 : FluxStandard fluxStd(fluxScaleEnum);
4698 0 : if (fluxScaleEnum==FluxStandard::PERLEY_BUTLER_2013 ||
4699 0 : fluxScaleEnum==FluxStandard::PERLEY_BUTLER_2017 ) {
4700 0 : fluxStd.setInterpMethod(interpolation);
4701 : }
4702 :
4703 : // Setup the frequency, Flux, and ComponentList arrays.
4704 0 : uInt nspws = selToRawSpwIds.nelements();
4705 0 : Vector<Vector<Flux<Double> > > returnFluxes(nspws), returnFluxErrs(nspws);
4706 0 : Vector<Vector<MFrequency> > mfreqs(nspws);
4707 0 : Vector<Vector<Double> > fluxUsed(nspws); // fluxesUsed(nspws,4)
4708 0 : MSColumns msc(*ms_p);
4709 0 : MEpoch aveEpoch=MEpoch(msc.timeMeas()(0));
4710 : const Unit freqUnit = sjy_setup_arrs(returnFluxes, returnFluxErrs, fluxUsed, tempCLs, mfreqs,
4711 0 : msc.spectralWindow(), nspws, selToRawSpwIds,
4712 0 : chanDep);
4713 : // Processing for each field ***************************************************
4714 0 : for(Int fldInd = fldids.nelements(); fldInd--;){
4715 0 : Int fldid = fldids[fldInd];
4716 : // Extract field name and field center position
4717 0 : MDirection fieldDir = msc.field().phaseDirMeas(fldid, msc.time()(0));
4718 0 : String fieldName = msc.field().name()(fldid);
4719 0 : Bool foundSrc = false;
4720 :
4721 : //for returned flux densities
4722 :
4723 0 : Record retvalperField;
4724 : //
4725 : //fluxUsed = fluxdens;
4726 0 : fluxUsed(0) = fluxdens;
4727 : //if(precompute){
4728 : // Pre-compute flux density for standard sources if not specified
4729 : // using the specified flux scale standard or catalog.
4730 : //
4731 : // The flux densities are calculated for all spws at once to avoid
4732 : // repeatedly digging up the flux model (and possibly the ephemeris).
4733 : //
4734 : // TT: moving this outside of if(precompute) since selected ms (mssel_p)
4735 : // will be needed for other cases to clear the model using
4736 : // VisModelData::ClearModel()
4737 :
4738 0 : Vector<Int> selectField(1);
4739 0 : selectField[0] = fldid;
4740 0 : Vector<Int> numDeChan(1);
4741 0 : numDeChan[0] = 0;
4742 0 : Vector<Int> begin(1);
4743 0 : begin[0] = 0;
4744 0 : Vector<Int> stepsize(1);
4745 0 : stepsize[0] = 1;
4746 0 : String msSelectString = "";
4747 0 : setdata("none", numDeChan, begin, stepsize, MRadialVelocity(),
4748 0 : MRadialVelocity(),
4749 : selToRawSpwIds, selectField, msSelectString, timerange, "",
4750 0 : Vector<Int>(), "", "", "", scanstr, intentstr, obsidstr, true, true);
4751 0 : if(nullSelect_p){
4752 0 : os << ((timerange == "" && scanstr == ""
4753 0 : && obsidstr == "") ? LogIO::WARN : LogIO::NORMAL)
4754 : << "No data was selected for field " << fldid << "."
4755 0 : << LogIO::POST;
4756 0 : continue;
4757 : }
4758 0 : if(precompute){
4759 : // Make componentlist for each spw.
4760 : // Pre-compute flux density for standard sources if not specified
4761 : // using the specified flux scale standard or catalog.
4762 : //
4763 : // The flux densities are calculated for all spws at once to avoid
4764 : // repeatedly digging up the flux model (and possibly the ephemeris).
4765 : //
4766 0 : MSColumns msselc(*mssel_p);
4767 : //if(nullSelect_p || msselc.nrow() < 1){
4768 0 : if(!nullSelect_p and msselc.nrow() < 1){
4769 0 : os << ((timerange == "" && scanstr == ""
4770 0 : && obsidstr == "") ? LogIO::WARN : LogIO::NORMAL)
4771 : << "No data was selected for field " << fldid << "."
4772 0 : << LogIO::POST;
4773 0 : continue;
4774 : }
4775 :
4776 : // chnage to return cocantenated CL?
4777 0 : foundSrc = sjy_computeFlux(os, fluxStd, returnFluxes, returnFluxErrs, tempCLs,
4778 : fluxUsed, fluxScaleName, aveEpoch, mfreqs, model, fieldName,
4779 : msselc, fldid, fieldDir, selToRawSpwIds, standard);
4780 : (void)foundSrc;
4781 0 : }
4782 :
4783 : //*************** For loop about spw starts here ****************************
4784 0 : Vector<Double> freqscaling;
4785 0 : Vector<Double> freqsOfScale;
4786 0 : Vector<Int> rawspwids(nspws);
4787 : // make raw spw id list
4788 0 : for(uInt selspw = 0; selspw < nspws; ++selspw){
4789 0 : Int rawspwid = selToRawSpwIds[selspw];
4790 0 : rawspwids[selspw]=rawspwid;
4791 : }
4792 :
4793 : // move inside sjy_computeFlux - TT, 2014.06.13
4794 : /***
4795 : if(foundSrc){
4796 : // Log fluxes found from the standard catalog database to HISTORY table
4797 : // get I-flux density for the selected spw (returnFluxes[nspw][4])
4798 : // Read this as fluxUsed = returnFluxes[selspw][0].value().
4799 : os << "CHECK: foundSrc....."<<LogIO::POST;
4800 : returnFluxes[selspw][0].value(fluxUsed);
4801 :
4802 : // Log flux density found for this field and spectral window
4803 : os.output().width(12);
4804 : os << fieldName;
4805 : os.output().width(2);
4806 : os << " (fld ind " << fldid << ") spw ";
4807 : os << rawspwid << " ";
4808 : os.output().width(0);
4809 : os.output().precision(5);
4810 : os << LogIO::NORMAL << "[I=" << fluxUsed(0) << ", "; // Loglevel INFO
4811 : os << "Q=" << fluxUsed(1) << ", ";
4812 : os << "U=" << fluxUsed(2) << ", ";
4813 : os << "V=" << fluxUsed(3) << "] Jy, ";
4814 : os << ("(" + fluxScaleName + ")") << LogIO::POST;
4815 : writeHistory(os);
4816 : }
4817 :
4818 : // If a model image has been specified,
4819 : // rescale it according to the I f.d. determined above
4820 :
4821 : //Vector<Double> freqscaling;
4822 : //Vector<Double> freqsOfScale;
4823 :
4824 : //MEpoch mtime = msc.field().timeMeas()(fldid);
4825 :
4826 : } //spw for-loop end
4827 : ****/
4828 :
4829 0 : MEpoch mtime = msc.field().timeMeas()(fldid);
4830 :
4831 : // model image prep. changed to do all spws at once - TT, 2014.06.13
4832 0 : if(model != ""){
4833 :
4834 0 : tmodimage = sjy_prepImage(os, fluxStd, fluxUsed[0], freqsOfScale, freqscaling, model, msc.spectralWindow(),
4835 : // tmodimage = sjy_prepImage(os, fluxStd, fluxUsed, freqsOfScale, freqscaling, model, msc.spectralWindow(),
4836 : // rawspwid, chanDep, mfreqs, selspw, fieldName,
4837 : selToRawSpwIds, chanDep, mfreqs, fieldName,
4838 : fieldDir, freqUnit, fluxdens, precompute, spix,
4839 : reffreq, aveEpoch, fldid);
4840 :
4841 : }
4842 0 : else if (!precompute) {
4843 : // do it in sjy_makeComponentList()
4844 : // TODO: add polindex, polangle, rm handling
4845 : // for now ignore circular polarization
4846 : //Vector<Double> cppars(1,0.0);
4847 0 : Vector<Double> checkfluxes;
4848 0 : sjy_makeComponentList(os, tempCLs, returnFluxes, fluxUsed[0], selToRawSpwIds, mfreqs, fieldName, fieldDir,
4849 : spix, pipars, papars, rotMeas, reffreq, aveEpoch, fldid);
4850 0 : returnFluxes[0][0].value(checkfluxes);
4851 0 : }
4852 : /*** moved to sjy_makeComponentList()
4853 : // make componentlist using flux densities from the user specfied fluxdensity(per-spw)
4854 : for(uInt selspw = 0; selspw < nspws; ++selspw){
4855 : Int rawspwid = selToRawSpwIds[selspw];
4856 : rawspwids[selspw]=rawspwid;
4857 : if(model == "" && !precompute){
4858 : // **** inside spw for-loop
4859 : // fluxUsed was supplied by the user instead of FluxStandard, so
4860 : // make a component list for it now, for use in ft.
4861 :
4862 : // Set the component flux density
4863 : Flux<Double> fluxval;
4864 : Flux<Double> fluxerr;
4865 : fluxval.setValue(fluxUsed[0]);
4866 : // Create a point component at the field center
4867 : // with the specified flux density
4868 : // - obviously this does not correct for solar objects...
4869 : PointShape point(fieldDir);
4870 : SpectralIndex siModel;
4871 : if(reffreq.getValue().getValue() > 0.0){
4872 : MeasFrame mFrame(MEpoch(msc.timeMeas()(0)), mLocation_p, fieldDir);
4873 : MFrequency::Convert cvt(mfreqs[selspw][0].getRef(), MFrequency::Ref(MFrequency::castType(reffreq.getRef().getType()), mFrame));
4874 : siModel.setRefFrequency(reffreq);
4875 : siModel.setIndex(spix);
4876 : returnFluxes[selspw][0].setValue(fluxUsed[0] * siModel.sample(cvt(mfreqs[selspw][0])));
4877 : }
4878 : else{
4879 : if(spix != 0.0){ // If not the default, complain and quit.
4880 : os << LogIO::SEVERE
4881 : << "spix cannot be nonzero with reffreq = 0!"
4882 : << LogIO::POST;
4883 : //return false;
4884 : }
4885 : siModel.setRefFrequency(MFrequency(Quantity(1.0, "GHz")));
4886 : siModel.setIndex(0.0);
4887 : }
4888 :
4889 : // No worries about varying fluxes or sizes here, so any time will do.
4890 : // Moved this line up (TT 2013/05/09)
4891 : //MEpoch mtime = msc.field().timeMeas()(fldid);
4892 : tempCLs[selspw] = FluxStandard::makeComponentList(fieldName,
4893 : mfreqs[selspw][0],
4894 : mtime, fluxval, point,
4895 : siModel,
4896 : // jagonzal (CAS-4109): Specify table name to avoid clashing between different CASA engines when running vs a MMS
4897 : ms_p->tableName() +
4898 : "_setjy_spw" +
4899 : String::toString(selspw) +
4900 : "_");
4901 : }
4902 : ***/
4903 :
4904 : // clear existing model for the selected field and for all selected spws
4905 : // outside spw loop
4906 : //if (!useModelCol_p && selspw==0) {
4907 0 : if (!useModelCol_p) {
4908 0 : String tmpspwstring=spwstring;
4909 0 : if (tmpspwstring=="") tmpspwstring="*";
4910 : os << LogIO::NORMAL
4911 : << "Will clear any existing model with matching field="
4912 : << fieldName
4913 : << " and spw=" << tmpspwstring
4914 0 : << LogIO::POST;
4915 :
4916 0 : String fldidstr = String::toString(fldid);
4917 : // use field id due to possible MSSelection bug for handing field name with blanks
4918 : //VisModelData::clearModel(*mssel_p, fieldName, spwstring)
4919 0 : VisModelData::clearModel(*mssel_p, fldidstr, spwstring);
4920 0 : }
4921 : // TODO: do it for all spw at once............
4922 : //sjy_make_visibilities(tmodimage, os, rawspwid, fldid, tempCLs[selspw],
4923 : // timerange, scanstr, intentstr, obsidstr, freqsOfScale, freqscaling);
4924 :
4925 : /***
4926 : if(tmodimage)
4927 : delete tmodimage;
4928 : tmodimage = NULL;
4929 : // if (Table::canDeleteTable("temp.setjy.image")) Table::deleteTable("temp.setjy.image");
4930 :
4931 : if(tempCLs[selspw] != ""){
4932 : String errmsg;
4933 :
4934 : //didAnything = true;
4935 : // commentted out for testing of concatCLs, may need to uncommentted later!!!!!!
4936 :
4937 : if(Table::canDeleteTable(errmsg, tempCLs[selspw]))
4938 : Table::deleteTable(tempCLs[selspw]);
4939 : else
4940 : os << LogIO::WARN
4941 : << "Could not rm " << tempCLs[selspw]
4942 : << " because the " << errmsg << "."
4943 : << LogIO::POST;
4944 : }
4945 : ***/
4946 :
4947 0 : for(uInt selspw = 0; selspw < nspws; ++selspw){
4948 0 : Record subrec;
4949 : //store fluxd actually used to scale (not input fluxdensity)
4950 0 : Vector<Double> finalFluxUsed;
4951 : // Flux of first chan
4952 0 : returnFluxes[selspw][0].value(finalFluxUsed);
4953 0 : subrec.define("fluxd",finalFluxUsed);
4954 : // TODO: add fluxd error when the flux density uncertainties
4955 : // are corrrectly filled.
4956 : //
4957 : //retvalperField.defineRecord(String::toString(rawspwid),subrec);
4958 0 : retvalperField.defineRecord(String::toString(selToRawSpwIds[selspw]),subrec);
4959 0 : } // for selspw end **********************************************
4960 : //retval.defineRecord(fieldName,retvalperField);
4961 0 : retvalperField.define("fieldName",fieldName);
4962 0 : retval.defineRecord(String::toString(fldid),retvalperField);
4963 :
4964 : // cocatenate componentlists - not yet used....
4965 : //if (tempCLs[0]!="") {
4966 : // concatcl name should contains field name mjd etc...
4967 : // ostringstream oss;
4968 : // oss<< ms_p->tableName() << "_setjy_"
4969 : // << fieldName << "_" << mtime.get("d").getValue()
4970 : // << "d.cl";
4971 : // String concatcl(oss);
4972 : // sjy_concatComponentLists(os, tempCLs, concatcl);
4973 : //}
4974 :
4975 : //sjy_make_visibilities(tmodimage, os, rawspwids, fldid, concatcl,
4976 : // timerange, scanstr, intentstr, obsidstr, freqsOfScale, freqscaling);
4977 : //
4978 : //### Uncomment above once setjyFTMachine can handle multi-row componentlist #############
4979 :
4980 0 : for(uInt selspw = 0; selspw < nspws; ++selspw){
4981 0 : sjy_make_visibilities(tmodimage, os, rawspwids[selspw], fldid, tempCLs[selspw],
4982 : timerange, scanstr, intentstr, obsidstr, freqsOfScale, freqscaling);
4983 : }
4984 : // #######################################################################################
4985 : // clean-up
4986 : //
4987 0 : if(tmodimage)
4988 0 : delete tmodimage;
4989 0 : tmodimage = NULL;
4990 :
4991 0 : for(uInt selspw = 0; selspw < nspws; ++selspw){
4992 0 : if(tempCLs[selspw] != ""){
4993 0 : String errmsg;
4994 :
4995 : //didAnything = true;
4996 :
4997 0 : if(TableUtil::canDeleteTable(errmsg, tempCLs[selspw]))
4998 0 : TableUtil::deleteTable(tempCLs[selspw]);
4999 : else
5000 : os << LogIO::WARN
5001 0 : << "Could not rm " << tempCLs[selspw]
5002 : << " because the " << errmsg << "."
5003 0 : << LogIO::POST;
5004 :
5005 : //if(Table::canDeleteTable(errmsg, concatcl))
5006 : // Table::deleteTable(concatcl);
5007 0 : }
5008 : }
5009 :
5010 0 : } // End of loop over fields.
5011 : // add a format info for the returned flux densities (Record)
5012 : //retval.define("format","{field name: {spw Id: {fluxd: [I,Q,U,V] in Jy}}}");
5013 0 : retval.define("format","{field Id: {spw Id: {fluxd: [I,Q,U,V] in Jy}, 'fieldName':field name }}");
5014 :
5015 0 : if(!precompute && spix[0] != 0.0 && reffreq.getValue().getValue() > 0.0){
5016 : os << LogIO::NORMAL
5017 : << "Flux density as a function of frequency (channel 0 of each spw):\n"
5018 : << " Frequency (GHz) Flux Density (Jy, Stokes I)"
5019 0 : << LogIO::POST;
5020 0 : for(uInt selspw = 0; selspw < nspws; ++selspw)
5021 0 : os << " " << mfreqs[selspw][0].get("GHz").getValue() << " "
5022 0 : << returnFluxes[selspw][0].value(Stokes::I).getValue()
5023 0 : << LogIO::POST;
5024 : }
5025 :
5026 0 : this->writeHistory(os);
5027 0 : this->unlock();
5028 : //return true;
5029 0 : }
5030 0 : catch (AipsError x){
5031 0 : this->unlock();
5032 0 : for(Int i = tempCLs.nelements(); i--;){
5033 0 : if(tempCLs[i] != "")
5034 0 : TableUtil::deleteTable(tempCLs[i]);
5035 : }
5036 0 : if (tmodimage) delete tmodimage; tmodimage=NULL;
5037 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
5038 : //return false;
5039 0 : }
5040 : //return didAnything;
5041 0 : return retval;
5042 0 : }
5043 :
5044 0 : String Imager::make_comp(const String& objName,
5045 : const String& standard,
5046 : const MEpoch& mtime, const Vector<MFrequency>& freqv,
5047 : const String& prefix)
5048 : {
5049 0 : Bool foundSrc = false;
5050 0 : logSink_p.clearLocally();
5051 0 : LogIO os(LogOrigin("imager", "setjy()"), logSink_p);
5052 :
5053 0 : Vector<String> clistnames(1);
5054 : try{
5055 : FluxStandard::FluxScale fluxScaleEnum;
5056 0 : String fluxScaleName("user-specified");
5057 :
5058 0 : if(!FluxStandard::matchStandard(standard, fluxScaleEnum, fluxScaleName))
5059 0 : throw(AipsError(standard + " is not a recognized flux density scale"));
5060 :
5061 0 : FluxStandard fluxStd(fluxScaleEnum);
5062 :
5063 0 : Vector<Vector<Flux<Double> > > returnFluxes(1), returnFluxErrs(1);
5064 0 : Vector<Vector<MFrequency> > mfreqs(1);
5065 0 : uInt nfreqs = freqv.nelements();
5066 :
5067 0 : mfreqs[0] = freqv;
5068 0 : returnFluxes[0].resize(nfreqs);
5069 0 : returnFluxErrs[0].resize(nfreqs);
5070 :
5071 0 : MDirection objDir;
5072 :
5073 0 : if (fluxScaleEnum==FluxStandard::PERLEY_BUTLER_2013 ||
5074 0 : fluxScaleEnum==FluxStandard::PERLEY_BUTLER_2017)
5075 : {
5076 0 : fluxStd.setInterpMethod("nearest");
5077 : }
5078 0 : foundSrc = fluxStd.computeCL(objName, mfreqs, mtime, objDir,
5079 : returnFluxes, returnFluxErrs,
5080 : clistnames, prefix);
5081 0 : }
5082 0 : catch(AipsError x){
5083 0 : os << LogIO::SEVERE << "Exception Reported: " << x.getMesg() << LogIO::POST;
5084 0 : RETHROW(x);
5085 0 : }
5086 0 : return foundSrc ? clistnames[0] : "";
5087 0 : }
5088 :
5089 0 : Unit Imager::sjy_setup_arrs(Vector<Vector<Flux<Double> > >& returnFluxes,
5090 : Vector<Vector<Flux<Double> > >& returnFluxErrs,
5091 : Vector<Vector<Double> >& fluxUsed,
5092 : Vector<String>& tempCLs,
5093 : Vector<Vector<MFrequency> >& mfreqs,
5094 : const MSSpWindowColumns& spwcols, const uInt nspws,
5095 : const Vector<Int>& selToRawSpwIds, const Bool chanDep)
5096 : {
5097 : // .getUnits() is a little confusing - it seems to return a Vector which is
5098 : // a list of all the units, not the unit for each row.
5099 :
5100 :
5101 0 : const Unit freqUnit(spwcols.chanFreqQuant().getUnits()[0]);
5102 :
5103 0 : IPosition ipos(1, 0);
5104 :
5105 0 : tempCLs.resize(nspws);
5106 0 : for(uInt selspw = 0; selspw < nspws; ++selspw){
5107 0 : Int rawspwid = selToRawSpwIds[selspw];
5108 :
5109 0 : if(chanDep){
5110 0 : mfreqs[selspw] = spwcols.chanFreqMeas()(rawspwid);
5111 0 : uInt nchan = mfreqs[selspw].nelements();
5112 0 : returnFluxes[selspw].resize(nchan);
5113 0 : returnFluxErrs[selspw].resize(nchan);
5114 : }
5115 : else{
5116 0 : mfreqs[selspw].resize(1);
5117 0 : returnFluxes[selspw].resize(1);
5118 0 : returnFluxErrs[selspw].resize(1);
5119 :
5120 : // Determine spectral window center frequency
5121 0 : Double medianFreq = median(spwcols.chanFreq()(rawspwid));
5122 0 : mfreqs[selspw] = spwcols.chanFreqMeas()(rawspwid)(ipos);
5123 0 : mfreqs[selspw].set(MVFrequency(Quantum<Double>(medianFreq, freqUnit)));
5124 : }
5125 : // initialize fluxUsed to 0
5126 0 : Vector<Double> iquvF(4,0.0);
5127 0 : fluxUsed[selspw]=iquvF;
5128 0 : }
5129 :
5130 0 : return freqUnit;
5131 0 : }
5132 : // new signature should be...
5133 : /***
5134 : Bool Imager::sjy_make_visibilities(TempImage<Float> *tmodimage, LogIO& os,
5135 : const Vector<Int>& rawspwids, const Int fldid,
5136 : const String& clname, const String& timerange,
5137 : const String& scanstr, const String& intentstr, const String& obsidstr,
5138 : const Vector<Double>& freqsOfScale, const Vector<Double>& freqscaling)
5139 : ***/
5140 0 : Bool Imager::sjy_make_visibilities(TempImage<Float> *tmodimage, LogIO& os,
5141 : const Int rawspwid, const Int fldid,
5142 : const String& clname, const String& timerange,
5143 : const String& scanstr, const String& intentstr, const String& obsidstr,
5144 : const Vector<Double>& freqsOfScale, const Vector<Double>& freqscaling)
5145 : {
5146 0 : Bool made_visibilities = false;
5147 :
5148 : // Select the uv-data for this field and spw. id.;
5149 : // all frequency channels selected.
5150 0 : Vector<Int> selectSpw(1), selectField(1);
5151 : // for the new
5152 : //Vector<Int> selectSpw, selectField(1);
5153 : //selectSpw.resize(rawspwids.nelements());
5154 : //selectSpw = rawspwids;
5155 : //
5156 0 : selectSpw[0] = rawspwid;
5157 0 : selectField[0] = fldid;
5158 0 : String msSelectString = "";
5159 0 : Vector<Int> numDeChan(1);
5160 0 : numDeChan[0] = 0;
5161 0 : Vector<Int> begin(1);
5162 0 : begin[0] = 0;
5163 0 : Vector<Int> stepsize(1);
5164 0 : stepsize[0] = 1;
5165 : //TempImage<Float> tmodimage = tmodimages[0];
5166 0 : if(tmodimage || clname != "") {
5167 : //if(!tmodimages[0].null() || clname != "")
5168 0 : setdata("channel", numDeChan, begin, stepsize, MRadialVelocity(),
5169 0 : MRadialVelocity(),
5170 : selectSpw, selectField, msSelectString, timerange, "",
5171 0 : Vector<Int>(), "", "", "", scanstr, intentstr, obsidstr, true, true);
5172 : }
5173 0 : if(!nullSelect_p){
5174 : // Use ft to form visibilities
5175 0 : Vector<String> modelv;
5176 :
5177 0 : if(tmodimage){
5178 : //if(!tmodimages[0].null()){
5179 : //for (uInt imod=0; imod<tmodimages.nelements();imod++) {
5180 : //if(!tmodimages[imod].null()) {
5181 0 : if(sm_p)
5182 0 : destroySkyEquation();
5183 : // do if for each spw?
5184 0 : if(freqsOfScale.nelements() > 0){
5185 : //if(freqsOfScales[imod].nelements() > 0){
5186 0 : delete ft_p;
5187 0 : ft_p=NULL;
5188 0 : ftmachine_p="SetJyGridFT";
5189 0 : createFTMachine();
5190 0 : (static_cast<SetJyGridFT*>(ft_p))->setScale(freqsOfScale, freqscaling);
5191 : //(static_cast<SetJyGridFT*>(ft_p))->setScale(freqsOfScales[imod], freqscalings[imod]);
5192 : }
5193 0 : if(!ft_p)
5194 0 : createFTMachine();
5195 0 : sm_p = new CleanImageSkyModel();
5196 : // loop over for multiple spw model images
5197 0 : sm_p->add(*tmodimage, 1);
5198 : //sm_p->add(*(tmodimages[imod]), 1);
5199 : //} //if-tmodimages..
5200 : //} //for loop
5201 : // this has no effect (SetJyGridFT hardcoded to use linear now, Aug. 2016)
5202 : //ft_p->setFreqInterpolation("nearest");
5203 0 : setSkyEquation();
5204 0 : se_p->predict(false);
5205 0 : destroySkyEquation();
5206 :
5207 0 : made_visibilities = true;
5208 : }
5209 0 : else if(clname != ""){
5210 : //made_visibilities = ft(modelv, clname, false);
5211 0 : made_visibilities = ft(modelv, clname, false);
5212 0 : destroySkyEquation();
5213 : }
5214 : else
5215 : os << LogIO::NORMAL
5216 : << "Skipping an empty component list for spw " << rawspwid
5217 : // for new one
5218 : // << "Skipping an empty component list for spw " << rawspwids
5219 0 : << LogIO::POST;
5220 0 : }
5221 0 : return made_visibilities;
5222 0 : }
5223 :
5224 :
5225 0 : Bool Imager::sjy_concatComponentLists(LogIO& os, const Vector<String>& tempCLs, const String& outTempCL)
5226 : {
5227 0 : ComponentList concatCL;
5228 : try {
5229 0 : for (uInt icl=0; icl<tempCLs.nelements(); icl++) {
5230 0 : if (tempCLs[icl]!="") {
5231 : // expected format _setjy_spw#_...
5232 0 : String::size_type spos=tempCLs[icl].find(String("spw"));
5233 0 : String::size_type epos=tempCLs[icl].find_first_of("_",spos);
5234 0 : String clab = tempCLs[icl].substr(spos,epos-spos);
5235 0 : Path clname(tempCLs[icl]);
5236 0 : os <<LogIO::DEBUG1 << " tempCLs["<<icl<<"]="<<tempCLs[icl]<<LogIO::POST;
5237 0 : ComponentList tempcl(clname, true);
5238 0 : Vector<Int> which(1,0);
5239 0 : tempcl.setLabel(which,clab);
5240 0 : os << LogIO::DEBUG1 << "adding "<<tempCLs[icl]<<" to "<<outTempCL<<LogIO::POST;
5241 0 : concatCL.addList(tempcl);
5242 0 : }
5243 : }
5244 0 : concatCL.rename(outTempCL, Table::New);
5245 0 : return true;
5246 0 : } catch (AipsError x) {
5247 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
5248 0 : << LogIO::EXCEPTION;
5249 0 : return false;
5250 0 : }
5251 0 : }
5252 :
5253 0 : Bool Imager::sjy_computeFlux(LogIO& os, FluxStandard& fluxStd,
5254 : Vector<Vector<Flux<Double> > >& returnFluxes,
5255 : Vector<Vector<Flux<Double> > >& returnFluxErrs,
5256 : Vector<String>& tempCLs,
5257 : //Vector<Double>& fluxUsed,
5258 : Vector<Vector<Double> >& fluxUsed,
5259 : String& fluxScaleName, MEpoch& aveEpoch,
5260 : const Vector<Vector<MFrequency> >& mfreqs,
5261 : const String& model, const String& fieldName,
5262 : const MSColumns& msc, const Int fldid,
5263 : const MDirection& fieldDir, const Vector<Int>& selToRawSpwIds,
5264 : const String& standard)
5265 : {
5266 0 : Bool foundSrc = false;
5267 :
5268 0 : Double meantime = msc.time()(0);
5269 0 : meantime += 0.5 * (msc.time()(msc.nrow() - 1) - meantime);
5270 0 : MEpoch mtime(msc.timeMeas()(0));
5271 0 : mtime.set(Quantity(meantime, "s"));
5272 0 : if(model != ""){
5273 : // Just get the fluxes and their uncertainties for scaling the image.
5274 : //foundSrc = fluxStd.compute(fieldName, mfreqs, returnFluxes,
5275 : // returnFluxErrs);
5276 0 : foundSrc = fluxStd.compute(fieldName, fieldDir, mfreqs, mtime, returnFluxes,
5277 : returnFluxErrs);
5278 : }
5279 : else{
5280 : // Go ahead and get FluxStandard to make the ComponentList, since
5281 : // it knows what type of component to use.
5282 :
5283 : // This is _a_ time. It would be more accurate and safer, but
5284 : // slower, to use the weighted average of the times at which this
5285 : // source was observed, and to use the range of times in the
5286 : // estimate of the error introduced by using a single time.
5287 : //
5288 : // Obviously that would be overkill if the source does not vary.
5289 : //
5290 : /***
5291 : Double meantime = msc.time()(0);
5292 : meantime += 0.5 * (msc.time()(msc.nrow() - 1) - meantime);
5293 : MEpoch mtime(msc.timeMeas()(0));
5294 : mtime.set(Quantity(meantime, "s"));
5295 : ***/
5296 :
5297 0 : aveEpoch=mtime;
5298 :
5299 0 : foundSrc = fluxStd.computeCL(fieldName, mfreqs, mtime, fieldDir,
5300 : returnFluxes, returnFluxErrs,
5301 0 : tempCLs, ms_p->tableName()+"_setjy_");
5302 : }
5303 0 : if(!foundSrc){
5304 0 : if(standard == String("SOURCE")){
5305 : // *** THIS MODE IS NOT USED IN CURRENT SETJY ***
5306 : // dgoscha, NCSA, 02 May, 2002
5307 : // this else condtion is to handle the case where the user
5308 : // specifies standard='SOURCE' in the setjy argument. This will
5309 : // then look into the SOURCE_MODEL column of the SOURCE subtable
5310 : // for a table-record entry that points to a component list with the
5311 : // model information in it.
5312 :
5313 : // Look in the SOURCE_MODEL column of the SOURCE subtable for
5314 : // the name of the CL which contains the model.
5315 :
5316 : // First test to make sure the SOURCE_MODEL column exists.
5317 0 : if(ms_p->source().tableDesc().isColumn("SOURCE_MODEL")){
5318 0 : TableRecord modelRecord;
5319 0 : msc.source().sourceModel().get(0, modelRecord);
5320 :
5321 : // Get the name of the model component list from the table record
5322 : Table modelRecordTable =
5323 0 : modelRecord.asTable(modelRecord.fieldNumber(String ("model")));
5324 0 : String modelCLName = modelRecordTable.tableName();
5325 0 : modelRecord.closeTable(modelRecord.fieldNumber(String ("model")));
5326 :
5327 : // Now grab the flux from the model component list and use.
5328 0 : ComponentList modelCL = ComponentList(Path(modelCLName), true);
5329 0 : SkyComponent fluxComponent = modelCL.component(fldid);
5330 :
5331 : //fluxUsed = 0;
5332 0 : fluxUsed(0) = 0;
5333 : //fluxUsed = real(fluxComponent.flux().value());
5334 0 : fluxUsed(0) = real(fluxComponent.flux().value());
5335 0 : fluxScaleName = modelCLName;
5336 0 : }
5337 : else {
5338 : os << LogIO::SEVERE << "Missing SOURCE_MODEL column."
5339 : << LogIO::SEVERE << "Continuing with the default, I = 1.0 Jy"
5340 0 : << LogIO::POST;
5341 : //fluxUsed = 0;
5342 0 : fluxUsed(0) = 0;
5343 : //fluxUsed(0) = 1.0;
5344 0 : fluxUsed(0)(0) = 1.0;
5345 : }
5346 : }
5347 : else {
5348 : // Source not found; use Stokes I=1.0 Jy for now
5349 : // (The flux standard already issued a complaint like this...)
5350 : // os << LogIO::WARN
5351 : // << fieldName << " was not recognized by " << standard
5352 : // << ".\nContinuing with the default, I = 1.0 Jy"
5353 : // << LogIO::POST;
5354 : //fluxUsed = 0;
5355 0 : fluxUsed(0) = 0;
5356 : //fluxUsed(0) = 1.0;
5357 0 : fluxUsed(0)(0) = 1.0;
5358 0 : fluxScaleName = "default";
5359 : }
5360 :
5361 : // Currently, if !foundSrc, then the flux density is the same for all
5362 : // spws.
5363 : // Log the flux density found for this field.
5364 0 : os.output().width(12);
5365 0 : os << fieldName << " ";
5366 0 : os.output().width(0);
5367 0 : os.output().precision(4);
5368 : /***
5369 : os << LogIO::NORMAL << "[I=" << fluxUsed(0) << ", "; // Loglevel INFO
5370 : os << "Q=" << fluxUsed(1) << ", ";
5371 : os << "U=" << fluxUsed(2) << ", ";
5372 : os << "V=" << fluxUsed(3) << "] Jy, ";
5373 : ***/
5374 0 : os << LogIO::NORMAL << "[I=" << fluxUsed(0)(0) << ", "; // Loglevel INFO
5375 0 : os << "Q=" << fluxUsed(0)(1) << ", ";
5376 0 : os << "U=" << fluxUsed(0)(2) << ", ";
5377 0 : os << "V=" << fluxUsed(0)(3) << "] Jy @ ";
5378 0 : os << mfreqs(0)(0).getValue()<<"Hz, ";
5379 0 : os << ("(" + fluxScaleName + ")") << LogIO::POST;
5380 0 : writeHistory(os);
5381 : } // End of if(!foundSrc).
5382 : else {
5383 : // Logging/History for foundSrc=true (moved from the im.setjy method)
5384 0 : for (uInt selspw=0; selspw<selToRawSpwIds.nelements(); selspw++) {
5385 0 : returnFluxes[selspw][0].value(fluxUsed[selspw]);
5386 : // Log flux density found for this field and spectral window
5387 0 : os.output().width(12);
5388 0 : os << fieldName;
5389 0 : os.output().width(2);
5390 0 : os << " (fld ind " << fldid << ") spw ";
5391 0 : os << selToRawSpwIds[selspw] << " ";
5392 0 : os.output().width(0);
5393 0 : os.output().precision(5);
5394 0 : os << LogIO::NORMAL << "[I=" << fluxUsed(selspw)(0) << ", "; // Loglevel INFO
5395 0 : os << "Q=" << fluxUsed(selspw)(1) << ", ";
5396 0 : os << "U=" << fluxUsed(selspw)(2) << ", ";
5397 0 : os << "V=" << fluxUsed(selspw)(3) << "] Jy @ ";
5398 0 : os << mfreqs(selspw)(0).getValue()<<"Hz, ";
5399 0 : os << ("(" + fluxScaleName + ")") << LogIO::POST;
5400 0 : writeHistory(os);
5401 : }
5402 : }
5403 0 : return foundSrc;
5404 0 : }
5405 :
5406 : //make componentlist(s) with user specified flux density
5407 0 : void Imager::sjy_makeComponentList(LogIO& os, Vector<String>& tempCLs,
5408 : Vector<Vector<Flux<Double> > >& returnFluxes,
5409 : const Vector<Double>& fluxUsed,
5410 : const Vector<Int>& selToRawSpwIds,
5411 : const Vector<Vector<MFrequency> >& mfreqs,
5412 : const String& fieldName,
5413 : const MDirection& fieldDir,
5414 : const Vector<Double>& spix,
5415 : const Vector<Double>& pipars,
5416 : const Vector<Double>& papars,
5417 : const Double& rotMeas,
5418 : // circ pol parameters
5419 : //const Vector<Double>& cppars,
5420 : const MFrequency& reffreq,
5421 : const MEpoch& mtime,
5422 : const Int /*fldid*/)
5423 : {
5424 :
5425 0 : for(uInt selspw = 0; selspw < selToRawSpwIds.nelements(); ++selspw){
5426 : // fluxUsed was supplied by the user instead of FluxStandard, so
5427 : // make a component list for it now, for use in ft.
5428 :
5429 : // if spix is a float/double and q,u fluxes not set or pipars and papars not
5430 : // set => SpectralIndex
5431 : // if spix is a float/double and q,u fluxes is set but not pipars and papers
5432 : // => spectralindex
5433 : // if spix is a float/double and q,u fluxes and pipars and papars are set
5434 : // => tabular
5435 : // if spix is a vector but qu fluxes nor pipars and papers are not set
5436 : // => tabular
5437 : // if spix is a vector => tabular form
5438 : //
5439 : // Set the component flux density
5440 0 : Flux<Double> fluxval;
5441 0 : Flux<Double> fluxerr;
5442 0 : fluxval.setValue(fluxUsed);
5443 : // Create a point component at the field center
5444 : // with the specified flux density
5445 : // - obviously this does not correct for solar objects...
5446 0 : PointShape point(fieldDir);
5447 :
5448 0 : Bool useTabularFlux(false);
5449 : //check if to use tabular form or SpectralIndex model
5450 0 : if ( spix.nelements() > 1 ) {
5451 0 : useTabularFlux=true;
5452 : }
5453 : else {
5454 : //if (pipars.nelements() > 1 || papars.nelements() > 1 || rotMeas != 0.0) {
5455 0 : if (pipars.nelements() > 1 || papars.nelements() > 1 ) {
5456 0 : useTabularFlux=true;
5457 : }
5458 : }
5459 0 : SpectralIndex siModel;
5460 : //Vector<Double> iflux;
5461 : //Vector<Double> qflux;
5462 : //Vector<Double> uflux;
5463 0 : Vector<Flux<Double> > fluxvalvec;
5464 0 : Bool gotQUFlux(false);
5465 0 : Bool useFluxAsIs(false);
5466 : //
5467 0 : if(reffreq.getValue().getValue() > 0.0){
5468 : //original code uses first time of the data but shouldn't be using the same time as for
5469 : //FluxStandard::makeComponentList?
5470 : //MeasFrame mFrame(MEpoch(msc.timeMeas()(0)), mLocation_p, fieldDir);
5471 0 : MeasFrame mFrame(mtime, mLocation_p, fieldDir);
5472 0 : MFrequency::Convert cvt(mfreqs[selspw][0].getRef(), MFrequency::Ref(MFrequency::castType(reffreq.getRef().getType()), mFrame));
5473 0 : siModel.setRefFrequency(reffreq);
5474 : // if spix is not array of double,do this otherwise need to set flux densities by tabular...
5475 : //
5476 0 : Int nchn = mfreqs[selspw].nelements();
5477 0 : fluxvalvec.resize(nchn);
5478 0 : Vector<Double> iflux(nchn);
5479 0 : Vector<Double> qflux(nchn);
5480 0 : Vector<Double> uflux(nchn);
5481 0 : Vector<Double> vflux(nchn,0.0);
5482 : // circular polarization fraction
5483 0 : Double circpolFraction=0.0;
5484 0 : if ( fluxUsed[0] !=0.0 && fluxUsed[3] != 0.0 ) {
5485 0 : circpolFraction = fluxUsed[3]/fluxUsed[0];
5486 : }
5487 :
5488 0 : if (spix.nelements()==1) {
5489 : //siModel.setIndex(spix[0]);
5490 0 : Vector<Double> stokesindex(4);
5491 0 : stokesindex[0]=spix[0];
5492 0 : stokesindex[1]=0.0;
5493 0 : stokesindex[2]=rotMeas!=0.0? rotMeas: 0.0;
5494 0 : stokesindex[3]=0.0;
5495 0 : siModel.setStokesIndex(stokesindex);
5496 : // still use iflux if q,u flux=0 but polindex and polangle is set
5497 0 : for (uInt ichn = 0; ichn < uInt(nchn); ichn++) {
5498 0 : iflux[ichn] = fluxUsed[0] * siModel.sample(cvt(mfreqs[selspw][ichn]));
5499 : }
5500 0 : }
5501 : else {
5502 : // tabular case
5503 0 : sjy_calciflux(mfreqs[selspw],reffreq,fluxUsed[0],spix,iflux);
5504 : }
5505 : // linear pol
5506 0 : Vector<Double> inpipars;
5507 0 : Vector<Double> inpapars;
5508 0 : if ( pipars.nelements() > 0 || papars.nelements() > 0 ) {
5509 0 : inpipars.resize(pipars.nelements());
5510 0 : inpipars=pipars;
5511 0 : inpapars.resize(papars.nelements());
5512 0 : inpapars=papars;
5513 : }
5514 :
5515 : // Either Q or U non-zero, so use them
5516 0 : if (fluxUsed[1] != 0.0 || fluxUsed[2] != 0.0) {
5517 : // if Q U flux densities are given use that as 0th coefficient
5518 0 : Double pi0 = sqrt(fluxUsed[1] * fluxUsed[1] + fluxUsed[2] * fluxUsed[2]) / fluxUsed[0];
5519 0 : Double pa0 = 0.5 * atan2(fluxUsed[2],fluxUsed[1]);
5520 : os<<LogIO::DEBUG1<<"Polindex c0="<<pi0<<", polangle c0="<<pa0
5521 0 : <<" determined from input flux densities are used"<<LogIO::POST;
5522 0 : if ( pipars.nelements() == 0 || papars.nelements() == 0 ) {
5523 0 : inpipars.resize(1);
5524 0 : inpapars.resize(1);
5525 : }
5526 0 : inpipars[0] = pi0;
5527 0 : inpapars[0] = pa0;
5528 : }
5529 : //if (useTabularFlux) {
5530 0 : if (inpipars.nelements()!=0 && inpapars.nelements()!=0) {
5531 : // cerr<<"running sjy_calcquflux...."<<endl;
5532 : // - returns qflux and uflux
5533 0 : gotQUFlux = sjy_calcquflux(inpipars, inpapars, iflux, rotMeas, mfreqs[selspw], reffreq, qflux, uflux);
5534 : }
5535 0 : else if (fluxUsed[1] != 0.0 || fluxUsed[2] != 0.0 || fluxUsed[3] != 0.0) {
5536 0 : gotQUFlux=true;
5537 0 : useFluxAsIs=true;
5538 : }
5539 : /***
5540 : if ( !useTabularFlux ) {
5541 : Vector<Double> stokesIndex(4);
5542 : stokesIndex[0] = index[0];
5543 : //need to translate polindex ...etc to stokesIndex[1,2]
5544 : siModel.setStokesIndex(spix);
5545 : }
5546 : ***/
5547 0 : for (uInt ichn=0; ichn < iflux.nelements(); ichn++) {
5548 0 : if (!gotQUFlux) {
5549 0 : qflux[ichn] = 0.0;
5550 0 : uflux[ichn] = 0.0;
5551 : }
5552 0 : else if(useFluxAsIs) {
5553 0 : qflux[ichn] = fluxUsed[1];
5554 0 : uflux[ichn] = fluxUsed[2];
5555 0 : vflux[ichn] = fluxUsed[3];
5556 : }
5557 0 : if ( circpolFraction != 0.0) vflux[ichn] = iflux[ichn]*circpolFraction;
5558 0 : Flux<Double> iquvflux(iflux[ichn],qflux[ichn],uflux[ichn],vflux[ichn]);
5559 0 : fluxvalvec[ichn] = iquvflux;
5560 0 : }
5561 0 : returnFluxes[selspw][0]=fluxvalvec[0];
5562 0 : }
5563 : else{
5564 0 : if(spix[0] != 0.0){ // If not the default, complain and quit.
5565 : os << LogIO::SEVERE
5566 : << "spix cannot be nonzero with reffreq = 0!"
5567 0 : << LogIO::POST;
5568 : //return false;
5569 : }
5570 0 : siModel.setRefFrequency(MFrequency(Quantity(1.0, "GHz")));
5571 0 : siModel.setIndex(0.0);
5572 : }
5573 : // TODO: call tabular form method for full pol specification....
5574 : //
5575 : // No worries about varying fluxes or sizes here, so any time will do.
5576 0 : if ( useTabularFlux ) {
5577 0 : tempCLs[selspw] = FluxStandard::makeComponentList(fieldName,
5578 : mfreqs[selspw],
5579 : mtime, fluxvalvec, point,
5580 0 : ms_p->tableName() +
5581 0 : "_setjy_spw" +
5582 0 : String::toString(selspw) +
5583 0 : "_");
5584 : }
5585 : else {
5586 : //if simodel is set use this
5587 : //cerr<<"NON-Tabular makeComponentList..."<<endl;
5588 : //if (fluxval.value(1) ==0.0 && fluxval.value(2) == 0.0 && gotQUFlux) {
5589 :
5590 : // fluxval is @ thefreq, make sure proper freq is used
5591 : // Note that refreq in siModel will be overriden by thefreq
5592 0 : MFrequency thefreq = reffreq;
5593 0 : if ( gotQUFlux) {
5594 0 : fluxval=fluxvalvec[0];
5595 0 : thefreq=mfreqs[selspw][0];
5596 : }
5597 0 : tempCLs[selspw] = FluxStandard::makeComponentList(fieldName,
5598 : // mfreqs[selspw][0],
5599 : thefreq,
5600 : mtime, fluxval, point,
5601 : siModel,
5602 : // jagonzal (CAS-4109): Specify table name to avoid clashing between different CASA engines when running vs a MMS
5603 0 : ms_p->tableName() +
5604 0 : "_setjy_spw" +
5605 0 : String::toString(selspw) +
5606 0 : "_");
5607 0 : }
5608 0 : }
5609 0 : }
5610 :
5611 : // modified the input model image by regridding, scaling with a flux standard
5612 0 : TempImage<Float>* Imager::sjy_prepImage(LogIO& os, FluxStandard& fluxStd,
5613 : Vector<Double>& fluxUsed, Vector<Double>& freqsOfScale,
5614 : Vector<Double>& freqscale, const String& model,
5615 : const MSSpWindowColumns& spwcols,
5616 : //const Int rawspwid, const Bool chanDep,
5617 : const Vector<Int> rawspwids, const Bool chanDep,
5618 : const Vector<Vector<MFrequency> >& mfreqs,
5619 : //const uInt selspw, const String& fieldName,
5620 : const String& fieldName,
5621 : const MDirection& fieldDir,
5622 : const Unit& freqUnit,
5623 : const Vector<Double>& fluxdens,
5624 : const Bool precompute,
5625 : //const Double spix,
5626 : const Vector<Double>& spix,
5627 : const MFrequency& reffreq,
5628 : const MEpoch& aveEpoch,
5629 : const Int fieldId)
5630 : {
5631 0 : TempImage<Float>* tmodimage = NULL;
5632 :
5633 : Double freqMax, freqMin;
5634 0 : Vector<Vector<Int> >dummy;
5635 : // TT for MMS this may not work
5636 : //String msname=mssel_p->antenna().tableName();
5637 : //msname.erase(msname.length()-8);
5638 0 : String msname=ms_p->tableName();
5639 : //adviseChanSelex(freqMin, freqMax, 0.0, MFrequency::LSRK, dummy, dummy, dummy, msname, fieldId, true, String::toString(rawspwid));
5640 : // Get freqmin and freqmax in LSRK for the entire span of the selected spws
5641 0 : String selSpwsStr;
5642 0 : for (uInt ispw=0; ispw < rawspwids.nelements(); ispw++) {
5643 0 : if (selSpwsStr!="") selSpwsStr += ", ";
5644 0 : selSpwsStr += String::toString(rawspwids(ispw));
5645 : }
5646 0 : adviseChanSelex(freqMin, freqMax, 0.0, MFrequency::LSRK, dummy, dummy, dummy, msname, fieldId, true, selSpwsStr);
5647 :
5648 : // Find min channel width to increment to construct freqsofScale
5649 0 : Double freqWidth = 0;
5650 : //
5651 0 : for (uInt ispw = 0; ispw<rawspwids.nelements(); ispw++) {
5652 0 : Vector<Double> freqWidths = spwcols.chanWidth()(rawspwids(ispw));
5653 :
5654 0 : Double minChanWidth = min(fabs(freqWidths));
5655 : // freqWidth init....
5656 0 : if (freqWidth == 0)
5657 0 : freqWidth = minChanWidth;
5658 : else
5659 0 : freqWidth = min(freqWidth,minChanWidth);
5660 0 : }
5661 :
5662 : // ADDED for debug
5663 : //Int rawspwid = rawspwids[0];
5664 : //Vector<Double> freqArray = spwcols.chanFreq()(rawspwid);
5665 : //Int nchan=freqArray.shape()[0] ;
5666 0 : Int nchan = Int(ceil(fabs(freqMax - freqMin)/freqWidth))+1;
5667 : //cerr<<"nchan="<<nchan<<" freqMax="<<freqMax<<" freqMin="<<freqMin<<" freqWidth="<<freqWidth<<endl;
5668 : //Filling it with the LSRK values
5669 : // SetJyGridFT will trigger nearestNeighbour interpolation for nchan>=2
5670 0 : Vector<Double> freqArray(nchan);
5671 0 : if (nchan==1) {
5672 0 : freqArray[0] = freqMin;
5673 : }
5674 0 : else if (nchan==2) {
5675 0 : freqArray[0] = freqMin;
5676 0 : freqArray[1] = freqMax;
5677 : }
5678 0 : else if (nchan==3) {
5679 0 : freqArray[0] = freqMin - freqWidth;
5680 0 : freqArray[1] = freqMin;
5681 0 : freqArray[2] = freqMin + freqWidth;
5682 : }
5683 : else {
5684 0 : for (Int k =0;k < nchan; ++k){
5685 0 : freqArray[k]=freqMin+k*freqWidth;
5686 : }
5687 : }
5688 : //Vector<Double> freqInc = spwcols.chanWidth()(rawspwid);
5689 0 : Double medianFreq = median(freqArray);
5690 0 : freqsOfScale.resize();
5691 0 : freqscale.resize();
5692 :
5693 : // 2 bw channel extra
5694 : // UNCOMMENTED for debug
5695 : //freqWidth = fabs(freqMax - freqMin) + 2 * max(freqInc);
5696 0 : Matrix<Double> fluxUsedPerChan; // 4 rows nchan col ...will resize when needed
5697 :
5698 : // Set fluxUsedPerChan to the flux densities for each chan.
5699 0 : if(chanDep || (spix[0] != 0.0 && fluxdens[0] != 0.0)){
5700 0 : IPosition whichChan(1, 0);
5701 0 : Flux<Double> returnFlux;
5702 0 : Flux<Double> returnFluxErr;
5703 0 : Double reffreqInGHz = 1.0;
5704 :
5705 0 : Unit ghz("GHz");
5706 0 : if(!precompute)
5707 0 : reffreqInGHz = reffreq.get(ghz).getValue();
5708 :
5709 0 : fluxUsedPerChan.resize(4, freqArray.nelements());
5710 0 : for(uInt k = 0; k < freqArray.nelements(); ++k){
5711 0 : whichChan[0] = k;
5712 0 : if(precompute){
5713 : //fluxStd.compute(fieldName, spwcols.chanFreqMeas()(rawspwid)(whichChan),
5714 : // returnFlux, returnFluxErr);
5715 0 : fluxStd.compute(fieldName, fieldDir, MFrequency(Quantity(freqArray[k], "Hz"), MFrequency::LSRK),
5716 : aveEpoch, returnFlux, returnFluxErr);
5717 0 : returnFlux.value(fluxUsed);
5718 : }
5719 : else{
5720 : // spix: index = c0 + c1*log(f/fo) + c2*log(f/fo)^2+ ...
5721 : // = log(So) + alpha*log(f/fo) + curv1*log(f/f0)^2 ....
5722 0 : uInt order = spix.nelements();
5723 0 : Polynomial<Double> spixfunc(order);
5724 0 : Vector<Double> coeffs(order+1);
5725 0 : coeffs[0] = log10(fluxdens[0]);
5726 0 : for (uInt ispix = 1; ispix < order+1; ispix++) {
5727 0 : coeffs[ispix] = spix[ispix-1];
5728 : }
5729 0 : spixfunc.setCoefficients(coeffs);
5730 :
5731 : //Double freq = spwcols.chanFreqMeas()(rawspwids(0))(whichChan).get(ghz).getValue();
5732 : //Double specfac = pow(freq / reffreqInGHz, spix);
5733 : // TT mod-06/11/14
5734 : // freqArray may or may not be exactly match with data chan frequencies
5735 : // so probably make sense to use freqArray instead
5736 : //Double specfac = pow((freqArray[k]/1.e+09) / reffreqInGHz, spix);
5737 0 : Double specfac = pow((freqArray[k]/1.e+09) / reffreqInGHz, spixfunc(freqArray[k]));
5738 :
5739 0 : for(uInt stokes = 0; stokes < 4; ++stokes)
5740 0 : fluxUsed[stokes] = fluxdens[stokes] * specfac;
5741 0 : }
5742 0 : fluxUsedPerChan.column(k) = fluxUsed;
5743 : }
5744 0 : }
5745 0 : PagedImage<Float> modimage(model);
5746 0 : modimage.table().unmarkForDelete();
5747 0 : IPosition imshape = modimage.shape();
5748 0 : CoordinateSystem csys(modimage.coordinates());
5749 0 : Int freqAxis = CoordinateUtil::findSpectralAxis(csys);
5750 0 : Vector<Stokes::StokesTypes> whichPols;
5751 0 : Int polAxis = CoordinateUtil::findStokesAxis(whichPols, csys);
5752 0 : Int icoord = csys.findCoordinate(Coordinate::SPECTRAL);
5753 0 : SpectralCoordinate spcsys = csys.spectralCoordinate(icoord);
5754 0 : MEpoch elEpoch; MDirection elDir; MFrequency::Types elTypes; MPosition elPos;
5755 0 : spcsys.getReferenceConversion(elTypes, elEpoch, elPos, elDir);
5756 0 : spcsys.setReferenceConversion(MFrequency::LSRK, aveEpoch, elPos, elDir);
5757 0 : spcsys.setReferenceValue(Vector<Double>(1, medianFreq));
5758 0 : spcsys.setReferencePixel(Vector<Double>(1, 0.0));
5759 0 : spcsys.setWorldAxisUnits(Vector<String>(1,
5760 : //mfreqs[selspw][0].getUnit().getName()));
5761 0 : mfreqs[0][0].getUnit().getName()));
5762 : //make image freq. width wide enough for FTMachine to work correctly
5763 0 : spcsys.setIncrement(Vector<Double>(1, 2*fabs(freqMax-freqMin)));
5764 : // make a cube model if the model is a cube already
5765 0 : if(modimage.shape()(freqAxis) >1){
5766 : // model image is a cube...just regrid it then
5767 : os << LogIO::NORMAL
5768 : << "The model image is a cube, so it is being regridded but without scaling the flux density."
5769 0 : << LogIO::POST;
5770 0 : spcsys = SpectralCoordinate(
5771 : //MFrequency::castType(mfreqs[selspw][0].getRef().getType()),
5772 0 : MFrequency::castType(mfreqs[0][0].getRef().getType()),
5773 0 : freqArray, spcsys.restFrequency());
5774 0 : imshape(freqAxis)=freqArray.nelements();
5775 0 : csys.replaceCoordinate(spcsys, icoord);
5776 0 : tmodimage = new TempImage<Float>(imshape, csys);
5777 0 : sjy_regridCubeChans(tmodimage, modimage, freqAxis);
5778 : //return from here itself
5779 0 : return tmodimage;
5780 : }
5781 :
5782 0 : if(chanDep && (fluxUsedPerChan.ncolumn() > 1)){
5783 : //spcsys = SpectralCoordinate(
5784 : // MFrequency::castType(mfreqs[selspw][0].getRef().getType()),
5785 : // freqArray, spcsys.restFrequency());
5786 0 : if(freqAxis < 2 || polAxis < 2)
5787 0 : throw(AipsError("Cannot setjy with a model that has spectral or stokes axis before direction axes.\n Please reorder the axes of the image"));
5788 0 : freqscale.resize(freqArray.nelements());
5789 0 : freqsOfScale.resize(freqArray.nelements());
5790 0 : freqsOfScale=freqArray;
5791 0 : freqscale=1.0;
5792 0 : if(freqAxis == 2) {//pol and freq are swapped
5793 0 : imshape[2]=imshape[3];
5794 0 : imshape[3] = 1;
5795 0 : Vector<Int> trans(4);
5796 0 : trans[0] = 0; trans[1] = 1; trans[2] = 3; trans[3] = 2;
5797 0 : csys.transpose(trans, trans);
5798 0 : }
5799 : else{
5800 0 : imshape(freqAxis) = 1;
5801 : }
5802 : }
5803 0 : csys.replaceCoordinate(spcsys, icoord);
5804 0 : tmodimage = new TempImage<Float>(imshape, csys);
5805 0 : IPosition blcin(modimage.shape().nelements(), 0);
5806 0 : IPosition trcin=modimage.shape()-1;
5807 0 : IPosition blcout(imshape.nelements(), 0);
5808 0 : IPosition trcout=imshape-1;
5809 :
5810 0 : for (uInt ipol=0; ipol < imshape[2]; ++ipol){
5811 0 : blcin[polAxis]=ipol;
5812 0 : trcin[polAxis]=ipol;
5813 0 : blcout[2]=ipol;
5814 0 : trcout[2]=ipol;
5815 0 : Slicer slin(blcin, trcin, Slicer::endIsLast);
5816 0 : Slicer slout(blcout, trcout, Slicer::endIsLast);
5817 0 : SubImage<Float> subimout(*tmodimage, slout, true);
5818 0 : SubImage<Float> subimin(modimage, slin, false);
5819 0 : subimout.copyData(subimin);
5820 0 : }
5821 : os << LogIO::DEBUG1
5822 : << "freqUnit.getName() = " << freqUnit.getName()
5823 0 : << LogIO::POST;
5824 : os << LogIO::DEBUG1
5825 : //<< "mfreqs[selspw].get(freqUnit).getValue() = "
5826 : //<< mfreqs[selspw][0].get(freqUnit).getValue()
5827 : << "mfreqs[0][0].get(freqUnit).getValue() = "
5828 0 : << mfreqs[0][0].get(freqUnit).getValue()
5829 0 : << LogIO::POST;
5830 :
5831 : // Check direction consistency (reported in log message below)
5832 0 : String err;
5833 0 : if(!CoordinateUtil::setDirectionConversion(err, csys, fieldDir.getRefString())){
5834 : os << "LogIO::WARN "
5835 : << "Could not set direction conversion between flux image and "
5836 0 : << fieldDir.getRefString() << LogIO::POST;
5837 : }
5838 0 : Int dircoord(csys.findCoordinate(Coordinate::DIRECTION));
5839 0 : DirectionCoordinate dircsys=csys.directionCoordinate(dircoord);
5840 0 : MVDirection mvd;
5841 0 : dircsys.toWorld(mvd,dircsys.referencePixel());
5842 0 : Double sep = fieldDir.getValue().separation(mvd,"\"").getValue();
5843 :
5844 : //Apply radius limit for 3C286,3C48,3C147 and 3C138
5845 0 : sjy_setRadiusLimit(tmodimage, modimage, model, dircsys);
5846 :
5847 : // for debugging
5848 : //PagedImage<Float> checkIm(TiledShape(modimage.shape(),
5849 : // modimage.niceCursorShape()),
5850 : // modimage.coordinates(),
5851 : // "checkImage");
5852 : //checkIm.copyData((LatticeExpr<Float>)(*tmodimage));
5853 :
5854 0 : if(fluxdens[0] != 0.0){
5855 0 : Float sumI = 1.0;
5856 :
5857 : // ?: can't handle the different return types.
5858 0 : if(whichPols.nelements() > 1)
5859 : //sumI = sum(ImagePolarimetry(modimage).stokesI()).getFloat();
5860 0 : sumI = sum(ImagePolarimetry(*tmodimage).stokesI()).getFloat();
5861 : else
5862 : //sumI = sum(modimage).getFloat();
5863 0 : sumI = sum(*tmodimage).getFloat();
5864 :
5865 : //if(selspw == 0)
5866 : os << LogIO::NORMAL
5867 0 : << "Using model image " << modimage.name() // Loglevel INFO
5868 0 : << LogIO::POST;
5869 :
5870 : // scale the image
5871 0 : if(freqscale.nelements() > 0){
5872 0 : Int midchan = freqArray.nelements()/2;
5873 0 : if(modimage.shape()(freqAxis) == 1){
5874 : // IPosition blc(imshape.nelements(), 0);
5875 : //IPosition trc = imshape - 1;
5876 : os << LogIO::NORMAL
5877 : //<< "Scaling spw " << selspw << "'s model image by channel to I = "
5878 0 : << "Scaling spw(s) " << String::toString(rawspwids) << "'s model image by channel to I = "
5879 0 : << fluxUsedPerChan.row(0)(0)<<", "
5880 0 : << fluxUsedPerChan.row(0)(midchan)<<", "
5881 0 : << fluxUsedPerChan.row(0)(nchan-1)
5882 : << " Jy @("
5883 0 : << freqArray(0)<<", "
5884 0 : << freqArray(midchan)<<", "
5885 0 : << freqArray(nchan-1)
5886 : <<")Hz (LSRK) for visibility prediction (a few representative values are shown)."
5887 0 : << LogIO::POST;
5888 0 : writeHistory(os);
5889 0 : for(uInt k = 0; k < fluxUsedPerChan.ncolumn(); ++k){
5890 0 : freqscale[k] = fluxUsedPerChan.column(k)(0)/sumI;
5891 : //blc[3] = k;
5892 : //trc[3] = k;
5893 : //Slicer sl(blc, trc, Slicer::endIsLast);
5894 : //SubImage<Float> subim(*tmodimage, sl, true);
5895 : //subim.copyData((LatticeExpr<Float>)(modimage*scale));
5896 : }
5897 : // for debug
5898 : //cerr<<"freqscale="<<freqscale<<endl;
5899 : //cerr<<"freqsOfScale="<<freqsOfScale<<endl;
5900 : }
5901 : }
5902 : else{
5903 : // Scale factor
5904 0 : Float scale = fluxUsed[0] / sumI;
5905 : //for addition of sjy_setRadiusLimit
5906 : //tmodimage->copyData( (LatticeExpr<Float>)(modimage * scale) );
5907 0 : tmodimage->copyData( (LatticeExpr<Float>)(*tmodimage * scale) );
5908 : os << LogIO::NORMAL
5909 : // << "Scaling spw " << selspw << "'s model image to I = "
5910 0 : << "Scaling spw(s) " << String::toString(rawspwids) << "'s model image to I = "
5911 0 : << fluxUsed[0] // Loglevel INFO
5912 : << " Jy @ "
5913 : //<< mfreqs[0][0].getValue()
5914 0 : << freqArray(0)
5915 : << "Hz (LSRK) for visibility prediction."
5916 0 : << LogIO::POST;
5917 0 : writeHistory(os);
5918 : }
5919 : }
5920 : else{
5921 : os << LogIO::NORMAL // Loglevel INFO
5922 : << "Using the model image's original unscaled flux density for visibility prediction."
5923 0 : << LogIO::POST;
5924 0 : writeHistory(os);
5925 : // included in sjy_setRadiusLimit
5926 : //tmodimage->copyData( (LatticeExpr<Float>)(modimage) );
5927 : }
5928 :
5929 : //if(selspw == 0){
5930 : os << LogIO::NORMAL // Loglevel INFO
5931 : << "The model image's reference pixel is " << sep
5932 : << " arcsec from " << fieldName << "'s phase center."
5933 0 : << LogIO::POST;
5934 0 : writeHistory(os);
5935 : //}
5936 :
5937 0 : return tmodimage;
5938 0 : }
5939 :
5940 0 : Bool Imager::sjy_regridCubeChans(TempImage<Float>* tmodimage,
5941 : PagedImage<Float>& modimage, Int freqAxis)
5942 : {
5943 0 : if(freqAxis != 3)
5944 0 : throw(AipsError("Cannot setjy with a cube model that does not have the spectral axis as the last one.\n Please reorder the axes of the image"));
5945 0 : ImageRegrid<Float> ir;
5946 0 : IPosition axes(1, freqAxis); // regrid the spectral only
5947 0 : ir.regrid(*tmodimage, Interpolate2D::LINEAR, axes, modimage);
5948 0 : return true;
5949 0 : }
5950 :
5951 0 : Bool Imager::sjy_setRadiusLimit(TempImage<Float>* tmodimage,
5952 : PagedImage<Float>& modimage, const String& model, DirectionCoordinate& dircsys)
5953 : {
5954 0 : Path path(model);
5955 0 : String basename=path.baseName();
5956 : Float arad;
5957 : // radius limit in arcsec from AIPS
5958 0 : if (basename.find("3C286")==0) {
5959 0 : arad=3.0;
5960 : }
5961 0 : else if (basename.find("3C48")==0) {
5962 0 : arad=0.95;
5963 : }
5964 0 : else if (basename.find("3C147")==0) {
5965 0 : arad=0.85;
5966 : }
5967 0 : else if (basename.find("3C138")==0) {
5968 0 : arad=0.75;
5969 : }
5970 : else {
5971 0 : arad=0;
5972 0 : tmodimage->copyData(modimage);
5973 0 : return true;
5974 : }
5975 : try {
5976 0 : Quantity qrad(arad,"arcsec");
5977 0 : Float prad=Float(qrad.get(Unit("rad")).getValue()/abs(dircsys.increment()(0)));
5978 0 : Float radius = (prad >0.5 ? prad: 0.5);
5979 : //Add current ms name to avoid file access clash for MMS processing
5980 0 : String tempmaskname=ms_p->tableName()+"__tmp_mask_setjy_radiuslimit";
5981 0 : PagedImage<Float> maskImage(TiledShape(modimage.shape(),
5982 0 : modimage.niceCursorShape()),
5983 0 : modimage.coordinates(), tempmaskname);
5984 0 : maskImage.table().markForDelete();
5985 0 : Matrix<Float> circ(1,3);
5986 0 : Record *imrec=0;
5987 0 : Matrix<Quantity> blctrcs;
5988 0 : circ(0,0)=radius;
5989 0 : circ(0,1)=dircsys.referencePixel()(0);
5990 0 : circ(0,2)=dircsys.referencePixel()(1);
5991 0 : Imager::regionToImageMask(tempmaskname,imrec,blctrcs,circ,1.0);
5992 0 : PagedImage<Float> tmpmask(tempmaskname);
5993 0 : tmpmask.table().markForDelete();
5994 0 : tmodimage->copyData( (LatticeExpr<Float>)(tmpmask*modimage));
5995 0 : }
5996 0 : catch (...) {
5997 0 : return false;
5998 0 : }
5999 0 : return true;
6000 0 : }
6001 :
6002 0 : Bool Imager::sjy_calciflux(const Vector<MFrequency>& freqs, const MFrequency& reffreq,
6003 : const Double refflux, const Vector<Double>& vspix, Vector<Double>& iflux)
6004 : {
6005 : try {
6006 : // assume polynomical is log(S) = c0 + alpha*log(f/fo) + curv1*log(f/fo)^2+
6007 : // vspix should contains [alpha, curv1, etc..] and c0 is calculated from refflux
6008 0 : uInt porder = vspix.nelements();
6009 0 : Polynomial<Double> lf(porder);
6010 0 : Vector<Double> coeffs(porder+1);
6011 0 : coeffs[0] = log10(refflux);
6012 0 : for (uInt i = 1; i < vspix.nelements()+1; i++ ) {
6013 0 : coeffs[i] = vspix[i-1];
6014 : }
6015 0 : lf.setCoefficients(coeffs);
6016 0 : Int nf = freqs.nelements();
6017 0 : iflux.resize(nf);
6018 0 : Unit frequnit("GHz");
6019 0 : Double reffreqval = reffreq.get(frequnit).getValue();
6020 0 : for (uInt cfidx = 0; cfidx < (uInt)nf; cfidx++) {
6021 0 : iflux[cfidx] = pow(10.0,lf(log10(freqs[cfidx].get(frequnit).getValue()/reffreqval)));
6022 : }
6023 0 : }
6024 0 : catch (...) {
6025 0 : return false;
6026 0 : }
6027 0 : return true;
6028 : }
6029 :
6030 0 : Bool Imager::sjy_calcquflux(const Vector<Double>& pipars, const Vector<Double>& papars,
6031 : const Vector<Double>& iflux, const Double rotMeas,
6032 : const Vector<MFrequency>& freqs,
6033 : const MFrequency& reffreq, Vector<Double>& qflux,
6034 : Vector<Double>& uflux)
6035 : {
6036 :
6037 : try {
6038 0 : Int nf = freqs.nelements();
6039 : //polindex
6040 : // of the form, pi_o + c1*(f-fo)/fo + c2*(f-fo)/fo
6041 0 : Polynomial<Double> pipoly(pipars.nelements());
6042 0 : pipoly.setCoefficients(pipars);
6043 : //pangle
6044 0 : Polynomial<Double> papoly(papars.nelements());
6045 0 : papoly.setCoefficients(papars);
6046 0 : qflux.resize(nf);
6047 0 : uflux.resize(nf);
6048 0 : Unit ghz("Hz");
6049 0 : Double f0 = reffreq.get(ghz).getValue();
6050 :
6051 0 : for (uInt cfidx = 0; cfidx < (uInt)nf; cfidx++) {
6052 0 : Double f = freqs[cfidx].get(ghz).getValue();
6053 0 : Double ipi = pipoly((f-f0)/f0);
6054 0 : Double ipa = papoly((f-f0)/f0);
6055 0 : Double iiflux = iflux[cfidx];
6056 0 : Double qfluxval = ipi * iiflux * cos(2.0*ipa);
6057 0 : Double ufluxval = ipi * iiflux * sin(2.0*ipa);
6058 : //debug
6059 : //if (cfidx<10) cerr<<"sjy_calcquflux:: poli="<<ipi<<" pola="<<ipa<<" qflux="<<qfluxval<<" uflux="<<ufluxval<<endl;
6060 0 : if (rotMeas!=0.0 ) {
6061 0 : Double rotangle = 2*rotMeas * C::c * C::c * (f0*f0-f*f)/(f*f*f0*f0);
6062 : //if (cfidx<10) cerr<<"rotangle="<<rotangle<<endl;
6063 0 : qflux[cfidx] = qfluxval*cos(rotangle) - ufluxval*sin(rotangle);
6064 0 : uflux[cfidx] = qfluxval*sin(rotangle) + ufluxval*cos(rotangle);
6065 : }
6066 : else {
6067 0 : qflux[cfidx] = qfluxval;
6068 0 : uflux[cfidx] = ufluxval;
6069 : }
6070 : //if (cfidx<10) cerr<<"uflux/qflux["<<cfidx<<"]="<<uflux[cfidx]/qflux[cfidx]<<endl;
6071 : }
6072 0 : }
6073 0 : catch (...) {
6074 0 : return false;
6075 0 : }
6076 0 : return true;
6077 : }
6078 :
6079 :
6080 0 : Bool Imager::clone(const String& imageName, const String& newImageName)
6081 : {
6082 : //if(!valid()) return false;
6083 : // This is not needed if(!assertDefinedImageParameters()) return false;
6084 0 : LogIO os(LogOrigin("imager", "clone()", WHERE));
6085 : try {
6086 0 : PagedImage<Float> oldImage(imageName);
6087 0 : PagedImage<Float> newImage(TiledShape(oldImage.shape(),
6088 0 : oldImage.niceCursorShape()), oldImage.coordinates(),
6089 0 : newImageName);
6090 0 : newImage.set(0.0);
6091 0 : newImage.table().flush(true, true);
6092 0 : } catch (AipsError x) {
6093 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
6094 0 : return false;
6095 0 : }
6096 0 : return true;
6097 0 : }
6098 :
6099 : // Make an empty image
6100 0 : Bool Imager::make(const String& model)
6101 : {
6102 :
6103 0 : if(!valid())
6104 : {
6105 0 : return false;
6106 : }
6107 0 : LogIO os(LogOrigin("imager", "make()", WHERE));
6108 :
6109 0 : this->lock();
6110 : try {
6111 0 : if(!assertDefinedImageParameters())
6112 : {
6113 0 : return false;
6114 : }
6115 :
6116 : // Make an image with the required shape and coordinates
6117 0 : String modelName(model);
6118 0 : if(modelName=="") modelName=imageName()+".model";
6119 : os << LogIO::DEBUG1
6120 0 : << "Making empty image: " << modelName << LogIO::POST;
6121 :
6122 0 : removeTable(modelName);
6123 0 : CoordinateSystem coords;
6124 : //if(!imagecoordinates(coords, false))
6125 0 : if(!imagecoordinates2(coords, false))
6126 : {
6127 0 : this->unlock();
6128 0 : return false;
6129 : }
6130 0 : this->makeEmptyImage(coords, modelName, fieldid_p);
6131 0 : this->unlock();
6132 :
6133 0 : return true;
6134 0 : } catch (AipsError x) {
6135 0 : this->unlock();
6136 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
6137 :
6138 0 : return false;
6139 :
6140 0 : }
6141 : this->unlock();
6142 :
6143 : return true;
6144 0 : }
6145 :
6146 : // Fit the psf. If psf is blank then make the psf first.
6147 0 : Bool Imager::fitpsf(const String& psf, ImageBeamSet& mbeam) {
6148 :
6149 0 : if(!valid())
6150 : {
6151 0 : return false;
6152 : }
6153 0 : LogIO os(LogOrigin("imager", "fitpsf()", WHERE));
6154 :
6155 0 : this->lock();
6156 : try {
6157 0 : if(!assertDefinedImageParameters())
6158 : {
6159 0 : this->unlock();
6160 0 : return false;
6161 : }
6162 :
6163 0 : os << LogIO::NORMAL << "Fitting to psf" << LogIO::POST; // Loglevel PROGRESS
6164 :
6165 0 : String lpsf; lpsf=psf;
6166 0 : if(lpsf=="") {
6167 0 : lpsf=imageName()+".psf";
6168 0 : makeimage("psf", lpsf);
6169 : }
6170 :
6171 0 : if(!Table::isReadable(lpsf)) {
6172 0 : this->unlock();
6173 : os << LogIO::SEVERE << "PSF image " << lpsf << " does not exist"
6174 0 : << LogIO::POST;
6175 :
6176 0 : return false;
6177 : }
6178 :
6179 0 : PagedImage<Float> psfImage(lpsf);
6180 0 : StokesImageUtil::FitGaussianPSF(psfImage, mbeam);
6181 0 : beam_p = mbeam;
6182 0 : beamValid_p=true;
6183 :
6184 0 : GaussianBeam elbeam=beam_p(0,0);
6185 : os << LogIO::NORMAL // Loglevel INFO
6186 : << " Beam fit: " << elbeam.getMajor("arcsec") << " by "
6187 : << elbeam.getMinor("arcsec") << " (arcsec) at pa "
6188 0 : << elbeam.getPA(Unit("deg")) << " (deg) " << endl;
6189 :
6190 0 : this->unlock();
6191 :
6192 0 : return true;
6193 0 : } catch (AipsError x) {
6194 0 : this->unlock();
6195 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
6196 :
6197 0 : return false;
6198 0 : }
6199 : this->unlock();
6200 :
6201 : return true;
6202 0 : }
6203 :
6204 :
6205 0 : Bool Imager::setscales(const String& scaleMethod,
6206 : const Int inscales,
6207 : const Vector<Float>& userScaleSizes)
6208 : {
6209 0 : scaleMethod_p = scaleMethod;
6210 0 : userScaleSizes_p.resize(userScaleSizes.nelements());
6211 0 : userScaleSizes_p = userScaleSizes;
6212 0 : if (scaleMethod_p == "uservector") {
6213 0 : nscales_p = userScaleSizes.nelements();
6214 : } else {
6215 0 : nscales_p = inscales;
6216 : }
6217 : //Force the creation of a new sm_p with the new scales
6218 0 : destroySkyEquation();
6219 0 : scaleInfoValid_p = true;
6220 0 : return true;
6221 : };
6222 :
6223 0 : Bool Imager::setSmallScaleBias(const Float inbias)
6224 : {
6225 0 : smallScaleBias_p = inbias;
6226 0 : return true;
6227 : }
6228 :
6229 : // Added for wb algo.
6230 0 : Bool Imager::settaylorterms(const Int intaylor,const Double inreffreq)
6231 : {
6232 0 : ntaylor_p = intaylor;
6233 0 : reffreq_p = inreffreq;
6234 0 : return true;
6235 : };
6236 :
6237 : // Set the beam
6238 0 : Bool Imager::setbeam(const ImageBeamSet& mbeam)
6239 : {
6240 0 : if(!valid()) return false;
6241 :
6242 0 : LogIO os(LogOrigin("imager", "setbeam()", WHERE));
6243 0 : beam_p = ImageBeamSet(mbeam);
6244 0 : beamValid_p=true;
6245 :
6246 0 : return true;
6247 0 : }
6248 :
6249 : // Plot the uv plane
6250 0 : Bool Imager::plotuv(const Bool rotate)
6251 : {
6252 :
6253 0 : if(!valid()) return false;
6254 :
6255 0 : LogIO os(LogOrigin("imager", "plotuv()", WHERE));
6256 :
6257 0 : this->lock();
6258 : try {
6259 : os << LogIO::NORMAL // Loglevel PROGRESS
6260 0 : << "Plotting uv coverage for currently selected data" << LogIO::POST;
6261 :
6262 0 : ROVisIter& vi(*rvi_p);
6263 0 : VisBuffer vb(vi);
6264 :
6265 0 : uInt nVis = count_visibilities(rvi_p, true, true);
6266 :
6267 0 : if(nVis==0) {
6268 0 : this->unlock();
6269 0 : os << LogIO::SEVERE << "No unflagged visibilities" << LogIO::POST;
6270 0 : return false;
6271 : }
6272 :
6273 0 : if(rotate) {
6274 : os << LogIO::NORMAL // Loglevel INFO
6275 0 : << "UVW will be rotated to specified phase center" << LogIO::POST;
6276 : }
6277 :
6278 :
6279 0 : Vector<Float> u(nVis); u=0.0;
6280 0 : Vector<Float> v(nVis); v=0.0;
6281 0 : Vector<Float> uRotated(nVis); uRotated=0.0;
6282 0 : Vector<Float> vRotated(nVis); vRotated=0.0;
6283 0 : Float maxAbsUV=0.0;
6284 :
6285 0 : Int iVis=0;
6286 0 : for (vi.originChunks();vi.moreChunks();vi.nextChunk()) {
6287 0 : for (vi.origin();vi.more();vi++) {
6288 0 : Int nRow=vb.nRow();
6289 0 : Int nChan=vb.nChannel();
6290 0 : Vector<Double> uvwRotated(3);
6291 0 : MeasFrame mFrame((MEpoch(Quantity(vb.time()(0), "s"))), mLocation_p);
6292 0 : UVWMachine uvwMachine(phaseCenter_p, vb.phaseCenter(), mFrame);
6293 0 : for (Int row=0; row<nRow; ++row) {
6294 0 : if(rotate) {
6295 0 : for (Int dim=0;dim<3;++dim) {
6296 0 : uvwRotated(dim)=vb.uvw()(row)(dim);
6297 : }
6298 0 : uvwMachine.convertUVW(uvwRotated);
6299 : }
6300 :
6301 0 : for (Int chn=0; chn<nChan; ++chn) {
6302 0 : if(!vb.flag()(chn,row)&&vb.imagingWeight()(chn,row)>0.0) {
6303 0 : Float f=vb.frequency()(chn)/C::c;
6304 0 : u(iVis)=vb.uvw()(row)(0)*f;
6305 0 : v(iVis)=vb.uvw()(row)(1)*f;
6306 0 : if(abs(u(iVis))>maxAbsUV) maxAbsUV=abs(u(iVis));
6307 0 : if(abs(v(iVis))>maxAbsUV) maxAbsUV=abs(v(iVis));
6308 0 : if(rotate) {
6309 0 : uRotated(iVis)=uvwRotated(0)*f;
6310 0 : vRotated(iVis)=uvwRotated(1)*f;
6311 0 : if(abs(uRotated(iVis))>maxAbsUV) maxAbsUV=abs(uRotated(iVis));
6312 0 : if(abs(vRotated(iVis))>maxAbsUV) maxAbsUV=abs(vRotated(iVis));
6313 : }
6314 0 : ++iVis;
6315 : }
6316 : }
6317 : }
6318 0 : }
6319 : }
6320 :
6321 0 : if(maxAbsUV==0.0) {
6322 0 : this->unlock();
6323 0 : os << LogIO::SEVERE << "Maximum uv distance is zero" << LogIO::POST;
6324 0 : return false;
6325 : }
6326 : else {
6327 0 : Quantity cell(0.5/maxAbsUV, "rad");
6328 : os << LogIO::NORMAL // Loglevel INFO
6329 0 : << "Maximum uv distance = " << maxAbsUV << " wavelengths" << endl;
6330 : os << LogIO::NORMAL // Loglevel INFO
6331 0 : << "Recommended cell size < " << cell.get("arcsec").getValue()
6332 0 : << " arcsec" << LogIO::POST;
6333 0 : }
6334 :
6335 :
6336 0 : return false;
6337 :
6338 : this->unlock();
6339 :
6340 0 : }
6341 0 : catch (AipsError x) {
6342 0 : this->unlock();
6343 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
6344 0 : return false;
6345 :
6346 0 : }
6347 0 : catch (...) {
6348 0 : this->unlock();
6349 0 : }
6350 0 : this->unlock();
6351 :
6352 :
6353 :
6354 0 : return true;
6355 0 : }
6356 :
6357 : // Plot the visibilities
6358 0 : Bool Imager::plotvis(const String& type, const Int increment)
6359 : {
6360 :
6361 0 : if(!valid()) return false;
6362 0 : LogIO os(LogOrigin("imager", "plotvis()", WHERE));
6363 :
6364 0 : this->lock();
6365 : try {
6366 :
6367 : os << LogIO::NORMAL // Loglevel PROGRESS
6368 : << "Plotting Stokes I visibility for currently selected data"
6369 0 : << LogIO::POST;
6370 :
6371 :
6372 0 : MSColumns msc(*mssel_p);
6373 0 : Bool hasCorrected=!(msc.correctedData().isNull());
6374 0 : Bool hasModel= true; //with virtual model data service model data is always there
6375 : //why bother if it is not requested
6376 0 : if(!((type == "all") || (type=="model") || (type == "residual")))
6377 0 : hasModel=false;
6378 0 : if(!((type == "all") || (type=="corrected") || (type == "residual")))
6379 0 : hasCorrected=false;
6380 :
6381 :
6382 :
6383 0 : Bool twoPol=true;
6384 0 : Vector<String> polType=msc.feed().polarizationType()(0);
6385 0 : if (polType(0)!="X" && polType(0)!="Y" &&
6386 0 : polType(0)!="R" && polType(0)!="L") {
6387 0 : twoPol=false;
6388 : }
6389 :
6390 0 : ROVisIter& vi(*rvi_p);
6391 0 : VisBuffer vb(vi);
6392 :
6393 0 : Int nVis=0;
6394 0 : Int counter=0;
6395 0 : Float maxWeight=0;
6396 0 : for (vi.originChunks();vi.moreChunks();vi.nextChunk()) {
6397 0 : for (vi.origin();vi.more();vi++) {
6398 0 : Int nRow=vb.nRow();
6399 0 : Int nChan=vb.nChannel();
6400 0 : maxWeight=max(maxWeight, max(vb.imagingWeight()));
6401 0 : for (Int row=0; row<nRow; ++row) {
6402 0 : for (Int chn=0; chn<nChan; ++chn) {
6403 0 : if(!vb.flag()(chn,row)&&vb.imagingWeight()(chn,row)>0.0) {
6404 0 : ++counter;
6405 0 : if(counter==increment) {
6406 0 : counter=0;
6407 0 : ++nVis;
6408 : }
6409 : }
6410 : }
6411 : }
6412 : }
6413 : }
6414 :
6415 0 : if(nVis==0) {
6416 0 : os << LogIO::SEVERE << "No unflagged visibilities" << LogIO::POST;
6417 0 : if(maxWeight <=0){
6418 : os << LogIO::SEVERE << "Max of imaging-weight is " << maxWeight
6419 0 : << LogIO::POST;
6420 : os << LogIO::SEVERE << "Try setting it with the function weight"
6421 0 : << LogIO::POST;
6422 : }
6423 0 : this->unlock();
6424 0 : return false;
6425 : }
6426 :
6427 0 : if(increment>1) {
6428 : os << LogIO::NORMAL << "For increment = " << increment << ", found " << nVis // Loglevel INFO
6429 0 : << " points for plotting" << endl;
6430 : }
6431 : else {
6432 0 : os << LogIO::NORMAL << "Found " << nVis << " points for plotting" << endl; // Loglevel INFO
6433 : }
6434 0 : Vector<Float> amp(nVis); amp=0.0;
6435 0 : Vector<Float> correctedAmp(nVis); correctedAmp=0.0;
6436 0 : Vector<Float> modelAmp(nVis); modelAmp=0.0;
6437 0 : Vector<Float> residualAmp(nVis); residualAmp=0.0;
6438 0 : Vector<Float> uvDistance(nVis); uvDistance=0.0;
6439 :
6440 0 : if(!hasModel)
6441 0 : modelAmp.resize();
6442 0 : if(!hasCorrected)
6443 0 : correctedAmp.resize();
6444 0 : if(!hasCorrected || !hasModel)
6445 0 : residualAmp.resize();
6446 :
6447 :
6448 0 : Float maxuvDistance=0.0;
6449 0 : Float maxAmp=0.0;
6450 0 : Float maxCorrectedAmp=0.0;
6451 0 : Float maxModelAmp=0.0;
6452 0 : Float maxResidualAmp=0.0;
6453 0 : Int iVis=0;
6454 0 : counter=0;
6455 0 : vi.originChunks();
6456 0 : vi.origin();
6457 0 : uInt numCorrPol=vb.nCorr();
6458 0 : for (vi.originChunks();vi.moreChunks();vi.nextChunk()) {
6459 0 : for (vi.origin();vi.more();vi++) {
6460 0 : Int nRow=vb.nRow();
6461 0 : Int nChan=vb.nChannel();
6462 0 : for (Int row=0; row<nRow; ++row) {
6463 0 : for (Int chn=0; chn<nChan; ++chn) {
6464 0 : if(!vb.flag()(chn,row)&&vb.imagingWeight()(chn,row)>0.0) {
6465 0 : ++counter;
6466 0 : if(counter==increment) {
6467 0 : counter=0;
6468 0 : Float f=vb.frequency()(chn)/C::c;
6469 0 : Float u=vb.uvw()(row)(0)*f;
6470 0 : Float v=vb.uvw()(row)(1)*f;
6471 0 : uvDistance(iVis)=sqrt(square(u)+square(v));
6472 0 : if(twoPol) {
6473 0 : amp(iVis)=sqrt((square(abs(vb.visCube()(0,chn,row)))+
6474 0 : square(abs(vb.visCube()(numCorrPol,chn,row))))/2.0);
6475 0 : if(hasCorrected)
6476 0 : correctedAmp(iVis)=
6477 0 : sqrt((square(abs(vb.correctedVisCube()(0,chn,row)))+
6478 0 : square(abs(vb.correctedVisCube()(numCorrPol,chn,row))))/2.0);
6479 0 : if(hasModel)
6480 0 : modelAmp(iVis)=
6481 0 : sqrt((square(abs(vb.modelVisCube()(0,chn,row)))+
6482 0 : square(abs(vb.modelVisCube()(numCorrPol,chn,row))))/2.0);
6483 0 : if(hasCorrected && hasModel)
6484 0 : residualAmp(iVis)=
6485 0 : sqrt((square(abs(vb.modelVisCube()(0,chn,row)-
6486 0 : vb.correctedVisCube()(0,chn,row)))+
6487 0 : square(abs(vb.modelVisCube()(numCorrPol,chn,row)-
6488 0 : vb.correctedVisCube()(numCorrPol,chn,row))))/2.0);
6489 : }
6490 : else {
6491 0 : amp(iVis)=abs(vb.visCube()(0,chn,row));
6492 0 : if(hasCorrected)
6493 0 : correctedAmp(iVis)=abs(vb.correctedVisCube()(0,chn,row));
6494 0 : if(hasModel)
6495 0 : modelAmp(iVis)=abs(vb.modelVisCube()(0,chn,row));
6496 0 : if(hasCorrected && hasModel)
6497 0 : residualAmp(iVis)=
6498 0 : abs(vb.modelVisCube()(0,chn,row)-
6499 0 : vb.correctedVisCube()(0,chn,row));
6500 : }
6501 0 : if(uvDistance(iVis)>maxuvDistance) {
6502 0 : maxuvDistance=uvDistance(iVis);
6503 : }
6504 0 : if(amp(iVis)>maxAmp) {
6505 0 : maxAmp=amp(iVis);
6506 : }
6507 0 : if(hasCorrected && (correctedAmp(iVis)>maxCorrectedAmp)) {
6508 0 : maxCorrectedAmp=correctedAmp(iVis);
6509 : }
6510 0 : if(hasModel && (modelAmp(iVis)>maxModelAmp)) {
6511 0 : maxModelAmp=modelAmp(iVis);
6512 : }
6513 0 : if((hasModel&&hasCorrected) && (residualAmp(iVis)>maxResidualAmp)) {
6514 0 : maxResidualAmp=residualAmp(iVis);
6515 : }
6516 0 : ++iVis;
6517 : }
6518 : }
6519 : }
6520 : }
6521 : }
6522 : }
6523 :
6524 :
6525 :
6526 0 : if(maxuvDistance==0.0) {
6527 0 : os << LogIO::SEVERE << "Maximum uv distance is zero" << LogIO::POST;
6528 0 : this->unlock();
6529 0 : return false;
6530 : }
6531 :
6532 :
6533 :
6534 :
6535 :
6536 0 : Float Ymax(0.0);
6537 :
6538 0 : if (type.contains("corrected") && hasCorrected)
6539 0 : if(maxCorrectedAmp>Ymax) Ymax = maxCorrectedAmp;
6540 :
6541 0 : if (type.contains("model") && hasModel)
6542 0 : if(maxModelAmp>Ymax) Ymax = maxModelAmp;
6543 :
6544 0 : if (type.contains("residual") && (hasModel && hasCorrected))
6545 0 : if(maxResidualAmp>Ymax) Ymax = maxResidualAmp;
6546 :
6547 0 : if (type.contains("observed"))
6548 0 : if(maxAmp>Ymax) Ymax = maxAmp;
6549 :
6550 0 : if ((type=="all") || (type == ""))
6551 : {
6552 0 : if (maxAmp > Ymax) Ymax = maxAmp;
6553 0 : if(hasCorrected && (maxCorrectedAmp>Ymax)) Ymax = maxCorrectedAmp;
6554 0 : if(hasModel && (maxModelAmp>Ymax)) Ymax = maxModelAmp;
6555 0 : if((hasModel && hasCorrected) && maxResidualAmp>Ymax) Ymax = maxResidualAmp;
6556 : }
6557 :
6558 :
6559 0 : return false;
6560 :
6561 :
6562 : this->unlock();
6563 : return true;
6564 0 : } catch (AipsError x) {
6565 0 : this->unlock();
6566 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
6567 0 : return false;
6568 0 : }
6569 : this->unlock();
6570 :
6571 : return true;
6572 0 : }
6573 :
6574 : // Plot the weights
6575 0 : Bool Imager::plotweights(const Bool gridded, const Int increment)
6576 : {
6577 :
6578 0 : if(!valid()) return false;
6579 0 : LogIO os(LogOrigin("imager", "plotweights()", WHERE));
6580 :
6581 0 : this->lock();
6582 : try {
6583 :
6584 :
6585 : os << LogIO::NORMAL // Loglevel PROGRESS
6586 : << "Plotting imaging weights for currently selected data"
6587 0 : << LogIO::POST;
6588 :
6589 0 : ROVisIter& vi(*rvi_p);
6590 0 : VisBuffer vb(vi);
6591 :
6592 0 : if(gridded) {
6593 0 : if(!assertDefinedImageParameters()) {this->unlock(); return false;}
6594 : // First find the gridded weights
6595 : Float uscale, vscale;
6596 : Int uorigin, vorigin;
6597 0 : uscale=(nx_p*mcellx_p.get("rad").getValue())/2.0;
6598 0 : vscale=(ny_p*mcelly_p.get("rad").getValue())/2.0;
6599 0 : uorigin=nx_p/2;
6600 0 : vorigin=ny_p/2;
6601 :
6602 : // Simply declare a big matrix
6603 0 : Float maxWeight=0.0;
6604 0 : Matrix<Float> gwt(nx_p,ny_p);
6605 0 : gwt=0.0;
6606 :
6607 : Float u, v;
6608 0 : Float sumwt=0.0;
6609 0 : for (vi.originChunks();vi.moreChunks();vi.nextChunk()) {
6610 0 : for (vi.origin();vi.more();vi++) {
6611 0 : Int nRow=vb.nRow();
6612 0 : Int nChan=vb.nChannel();
6613 0 : for (Int row=0; row<nRow; ++row) {
6614 0 : for (Int chn=0; chn<nChan; ++chn) {
6615 0 : if(!vb.flag()(chn,row)&&vb.imagingWeight()(chn,row)>0.0) {
6616 0 : Float f=vb.frequency()(chn)/C::c;
6617 0 : u=vb.uvw()(row)(0)*f;
6618 0 : v=vb.uvw()(row)(1)*f;
6619 0 : Int ucell=Int(uscale*u+uorigin);
6620 0 : Int vcell=Int(vscale*v+vorigin);
6621 0 : if((ucell>0)&&(ucell<nx_p)&&(vcell>0)&&(vcell<ny_p)) {
6622 0 : gwt(ucell,vcell)+=vb.imagingWeight()(chn,row);
6623 0 : sumwt+=vb.imagingWeight()(chn,row);
6624 0 : if(vb.imagingWeight()(chn,row)>maxWeight) {
6625 0 : maxWeight=vb.imagingWeight()(chn,row);
6626 : }
6627 : }
6628 0 : ucell=Int(-uscale*u+uorigin);
6629 0 : vcell=Int(-vscale*v+vorigin);
6630 0 : if((ucell>0)&&(ucell<nx_p)&&(vcell>0)&&(vcell<ny_p)) {
6631 0 : gwt(ucell,vcell)+=vb.imagingWeight()(chn,row);
6632 : }
6633 : }
6634 : }
6635 : }
6636 : }
6637 : }
6638 :
6639 0 : if(sumwt>0.0) {
6640 0 : os << LogIO::NORMAL << "Sum of weights = " << sumwt << endl; // Loglevel INFO
6641 : }
6642 : else {
6643 0 : this->unlock();
6644 : os << LogIO::SEVERE << "Sum of weights is zero: perhaps you need to weight the data"
6645 0 : << LogIO::POST;
6646 0 : return false;
6647 : }
6648 :
6649 :
6650 : //Float umax=Float(nx_p/2)/uscale;
6651 : //Float vmax=Float(ny_p/2)/vscale;
6652 :
6653 :
6654 0 : return false;
6655 0 : }
6656 : else {
6657 :
6658 : // Now do the points plot
6659 0 : Int nVis=0;
6660 0 : Int counter=0;
6661 0 : Float maxWeight=0.0;
6662 0 : for (vi.originChunks();vi.moreChunks();vi.nextChunk()) {
6663 0 : for (vi.origin();vi.more();vi++) {
6664 0 : Int nRow=vb.nRow();
6665 0 : Int nChan=vb.nChannel();
6666 0 : for (Int row=0; row<nRow; ++row) {
6667 0 : for (Int chn=0; chn<nChan; ++chn) {
6668 0 : if(!vb.flag()(chn,row)&&vb.imagingWeight()(chn,row)>0.0) {
6669 0 : ++counter;
6670 0 : if(counter==increment) {
6671 0 : counter=0;
6672 0 : ++nVis;
6673 : }
6674 : }
6675 : }
6676 : }
6677 : }
6678 : }
6679 :
6680 0 : if(increment>1) {
6681 : os << LogIO::NORMAL // Loglevel INFO
6682 : << "For increment = " << increment << ", found " << nVis
6683 0 : << " points for plotting" << endl;
6684 : }
6685 : else {
6686 : os << LogIO::NORMAL // Loglevel INFO
6687 0 : << "Found " << nVis << " points for plotting" << endl;
6688 : }
6689 :
6690 0 : Float maxuvDistance=0.0;
6691 0 : Vector<Float> weights(nVis);
6692 0 : Vector<Float> uvDistance(nVis);
6693 0 : weights=0.0;
6694 0 : uvDistance=0.0;
6695 :
6696 0 : Int iVis=0;
6697 0 : for (vi.originChunks();vi.moreChunks();vi.nextChunk()) {
6698 0 : for (vi.origin();vi.more();vi++) {
6699 0 : Int nRow=vb.nRow();
6700 0 : Int nChan=vb.nChannel();
6701 0 : for (Int row=0; row<nRow; ++row) {
6702 0 : for (Int chn=0; chn<nChan; ++chn) {
6703 0 : if(!vb.flag()(chn,row)&&vb.imagingWeight()(chn,row)>0.0) {
6704 0 : ++counter;
6705 0 : if(counter==increment) {
6706 0 : Float f=vb.frequency()(chn)/C::c;
6707 0 : Float u=vb.uvw()(row)(0)*f;
6708 0 : Float v=vb.uvw()(row)(1)*f;
6709 0 : uvDistance(iVis)=sqrt(square(u)+square(v));
6710 0 : weights(iVis)=vb.imagingWeight()(chn,row);
6711 0 : if(vb.imagingWeight()(chn,row)>maxWeight) {
6712 0 : maxWeight=vb.imagingWeight()(chn,row);
6713 : }
6714 0 : if(uvDistance(iVis)>maxuvDistance) {
6715 0 : maxuvDistance=uvDistance(iVis);
6716 : }
6717 0 : counter=0;
6718 0 : ++iVis;
6719 : }
6720 : }
6721 : }
6722 : }
6723 : }
6724 : }
6725 :
6726 0 : if(maxuvDistance==0.0) {
6727 0 : this->unlock();
6728 0 : os << LogIO::SEVERE << "Maximum uv distance is zero" << LogIO::POST;
6729 0 : return false;
6730 : }
6731 :
6732 :
6733 0 : return false;
6734 0 : }
6735 :
6736 :
6737 :
6738 : this->unlock();
6739 : return true;
6740 0 : } catch (AipsError x) {
6741 0 : this->unlock();
6742 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
6743 0 : return false;
6744 0 : }
6745 : this->unlock();
6746 :
6747 : return true;
6748 0 : }
6749 :
6750 0 : Bool Imager::clipvis(const Quantity& threshold)
6751 : {
6752 :
6753 0 : if(!valid()) return false;
6754 :
6755 0 : LogIO os(LogOrigin("imager", "clipvis()", WHERE));
6756 :
6757 0 : this->lock();
6758 : try {
6759 :
6760 0 : Float thres=threshold.get("Jy").getValue();
6761 :
6762 : os << LogIO::NORMAL // Loglevel PROGRESS
6763 : << "Clipping visibilities where residual visibility > "
6764 0 : << thres << " Jy" << LogIO::POST;
6765 0 : if(!wvi_p){
6766 : os << LogIO::WARN
6767 : << "Cannot clip visibilities in read only mode of ms"
6768 0 : << LogIO::POST;
6769 0 : return false;
6770 : }
6771 0 : VisIter& vi(*wvi_p);
6772 0 : VisBuffer vb(vi);
6773 :
6774 :
6775 0 : vi.originChunks();
6776 0 : vi.origin();
6777 : // Making sure picking LL for [RR RL LR LL] correlations or [RR LL]
6778 0 : uInt numCorrPol=vb.modelVisCube().shape()(0) - 1 ;
6779 0 : Int nBad=0;
6780 0 : for (vi.originChunks();vi.moreChunks();vi.nextChunk()) {
6781 0 : for (vi.origin();vi.more();vi++) {
6782 0 : Int nRow=vb.nRow();
6783 0 : Int nChan=vb.nChannel();
6784 0 : for (Int row=0; row<nRow; ++row) {
6785 0 : for (Int chn=0; chn<nChan; ++chn) {
6786 0 : if(!vb.flag()(chn,row)) {
6787 : Float residualAmp=
6788 0 : sqrt((square(abs(vb.modelVisCube()(0,chn,row)-
6789 0 : vb.correctedVisCube()(0,chn,row)))+
6790 0 : square(abs(vb.modelVisCube()(numCorrPol,chn,row)-
6791 0 : vb.correctedVisCube()(numCorrPol,chn,row))))/2.0);
6792 0 : if(residualAmp>thres) {
6793 0 : vb.flag()(chn,row)=true;
6794 0 : ++nBad;
6795 : }
6796 : }
6797 : }
6798 : }
6799 0 : vi.setFlag(vb.flag());
6800 : }
6801 : }
6802 :
6803 0 : os << LogIO::NORMAL << "Flagged " << nBad << " points" << LogIO::POST; // Loglevel INFO
6804 :
6805 0 : this->unlock();
6806 0 : return true;
6807 0 : } catch (AipsError x) {
6808 0 : this->unlock();
6809 0 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
6810 0 : }
6811 0 : this->unlock();
6812 :
6813 0 : return true;
6814 0 : }
6815 :
6816 : // Plot various ids
6817 0 : Bool Imager::plotsummary()
6818 : {
6819 :
6820 0 : if(!valid()) return false;
6821 :
6822 0 : LogIO os(LogOrigin("imager", "plotsummary()", WHERE));
6823 :
6824 0 : os << LogIO::WARN << "NOT implemented " << LogIO::POST;
6825 0 : return false;
6826 :
6827 : this->lock();
6828 : try {
6829 : /*
6830 : os << "Plotting field and spectral window ids for currently selected data" << LogIO::POST;
6831 :
6832 : ROVisIter& vi(*rvi_p);
6833 : VisBuffer vb(vi);
6834 :
6835 : Int nVis=0;
6836 : for (vi.originChunks();vi.moreChunks();vi.nextChunk()) {
6837 : for (vi.origin();vi.more();vi++) {
6838 : Int nRow=vb.nRow();
6839 : for (Int row=0; row<nRow; ++row) {
6840 : ++nVis;
6841 : }
6842 : }
6843 : }
6844 :
6845 : os << "Found " << nVis << " selected records" << LogIO::POST;
6846 :
6847 : Vector<Float> fieldId(nVis);
6848 : Vector<Float> spectralWindowId(nVis);
6849 : Vector<Double> t(nVis);
6850 :
6851 : Int maxFieldId=0;
6852 : Int maxSpectralWindowId=0;
6853 : Int iVis=0;
6854 : for (vi.originChunks();vi.moreChunks();vi.nextChunk()) {
6855 : for (vi.origin();vi.more();vi++) {
6856 : Int nRow=vb.nRow();
6857 : for (Int row=0; row<nRow; ++row) {
6858 : t(iVis)=vb.time()(row);
6859 : fieldId(iVis)=vb.fieldId()+1.0;
6860 : spectralWindowId(iVis)=vb.spectralWindow()+1.003;
6861 : if(Int(fieldId(iVis))>maxFieldId) maxFieldId=Int(fieldId(iVis));
6862 : if(Int(spectralWindowId(iVis))>maxSpectralWindowId)
6863 : maxSpectralWindowId=Int(spectralWindowId(iVis));
6864 : ++iVis;
6865 : }
6866 : }
6867 : }
6868 :
6869 : Double tStart=t(0);
6870 : Vector<Float> timeFloat(nVis);
6871 : for(Int i=0;i<nVis;++i) {
6872 : timeFloat(i)=Float(t(i)-tStart);
6873 : }
6874 :
6875 : MSColumns msc(*ms_p);
6876 : PGPlotter plotter=getPGPlotter();
6877 : plotter.subp(1, 2);
6878 : plotter.page();
6879 : plotter.swin(timeFloat(0), timeFloat(nVis-1)*1.20, 0, Float(maxFieldId)*1.1);
6880 : plotter.tbox("BCSNTZHFO", 0.0, 0, "ABCNTS", 0.0, 0);
6881 : String xLabel="Time (offset from " + MVTime(tStart/86400.0).string() + ")";
6882 : plotter.lab(xLabel, "ID", "Field IDs for " +imageName());
6883 : plotter.sci(1);
6884 : for (Int fid=0;fid<maxFieldId;++fid) {
6885 : String fieldName=msc.field().name()(fid);
6886 : plotter.text(1.02*timeFloat(nVis-1), Float(fid+1), fieldName);
6887 : }
6888 : plotter.pt(timeFloat,fieldId,-1);
6889 : plotter.page();
6890 : plotter.swin(timeFloat(0), timeFloat(nVis-1)*1.20, 0,
6891 : Float(maxSpectralWindowId)*1.1);
6892 : plotter.tbox("BCSNTZHFO", 0.0, 0, "ABCNTS", 0.0, 0);
6893 : xLabel="Time (offset from " + MVTime(tStart/86400.0).string() + ")";
6894 : plotter.lab(xLabel, "ID", "Spectral Window IDs for " +imageName());
6895 : plotter.sci(1);
6896 : for(Int spwId=0;spwId<maxSpectralWindowId;++spwId) {
6897 : Vector<Double> chanFreq=msc.spectralWindow().chanFreq()(spwId);
6898 : ostringstream spwString;
6899 : spwString<<chanFreq(0)/1.0e9<<" GHz";
6900 : plotter.text(1.02*timeFloat(nVis-1), Float(spwId+1),
6901 : spwString);
6902 : }
6903 : plotter.pt(timeFloat,spectralWindowId,-1);
6904 : plotter.iden();
6905 : this->unlock();
6906 : return true;
6907 : */
6908 : } catch (AipsError x) {
6909 : this->unlock();
6910 : os << LogIO::SEVERE << "Exception: " << x.getMesg() << LogIO::POST;
6911 : return false;
6912 : }
6913 :
6914 : this->unlock();
6915 :
6916 : return true;
6917 0 : }
6918 :
6919 :
6920 0 : Bool Imager::detached() const
6921 : {
6922 0 : if (ms_p.null()) {
6923 0 : LogIO os(LogOrigin("imager", "detached()", WHERE));
6924 : os << LogIO::SEVERE <<
6925 : "imager is detached - cannot perform operation." << endl <<
6926 0 : "Call imager.open('filename') to reattach." << LogIO::POST;
6927 0 : return true;
6928 0 : }
6929 0 : return false;
6930 : }
6931 :
6932 0 : Bool Imager::makemodelfromsd(const String& sdImage, const String& modelImage,
6933 : const String& lowPSF, String& maskImage)
6934 : {
6935 :
6936 0 : if(!valid()) return false;
6937 :
6938 0 : LogIO os(LogOrigin("imager", "makemodelfromsd()", WHERE));
6939 :
6940 : try {
6941 :
6942 0 : if(!Table::isReadable(sdImage)){
6943 : os << LogIO::SEVERE << "Single Dish " << sdImage
6944 0 : << " image is not readable" << LogIO::EXCEPTION;
6945 :
6946 0 : return false;
6947 : }
6948 :
6949 : os << LogIO::NORMAL << "Creating an initial model image " << modelImage // Loglevel INFO
6950 0 : << " from single dish image " << sdImage << LogIO::POST;
6951 :
6952 0 : CoordinateSystem coordsys;
6953 : //imagecoordinates(coordsys);
6954 0 : imagecoordinates2(coordsys);
6955 0 : String modelName=modelImage;
6956 0 : this->makeEmptyImage(coordsys, modelName, fieldid_p);
6957 :
6958 0 : PagedImage<Float> model(modelImage);
6959 0 : PagedImage<Float> low0(sdImage);
6960 0 : String sdObs=low0.coordinates().obsInfo().telescope();
6961 :
6962 0 : GaussianBeam lBeam;
6963 0 : ImageInfo lowInfo=low0.imageInfo();
6964 0 : lBeam=lowInfo.restoringBeam();
6965 :
6966 0 : Float beamFactor=-1.0;
6967 :
6968 :
6969 : // regrid the single dish image
6970 : {
6971 0 : ImageRegrid<Float> ir;
6972 0 : IPosition axes(3,0,1,3); // if its a cube, regrid the spectral too
6973 0 : ir.regrid(model, Interpolate2D::LINEAR, axes, low0);
6974 0 : }
6975 :
6976 :
6977 :
6978 : // Will need to make a complex image to apply the beam
6979 0 : TempImage<Complex> ctemp(model.shape(), model.coordinates());
6980 0 : if(lowPSF=="") {
6981 : os << LogIO::NORMAL // Loglevel INFO
6982 : << "Using primary beam of single dish to determine flux scale"
6983 0 : << LogIO::POST;
6984 :
6985 0 : TempImage<Float> beamTemp(model.shape(), model.coordinates());
6986 : //Make the PB accordingly
6987 0 : if(lBeam.isNull()) {
6988 :
6989 0 : if (doDefaultVP_p) {
6990 0 : if(telescope_p!=""){
6991 0 : ObsInfo myobsinfo=this->latestObsInfo();
6992 0 : myobsinfo.setTelescope(telescope_p);
6993 0 : coordsys.setObsInfo(myobsinfo);
6994 :
6995 0 : }
6996 : else{
6997 0 : if(sdObs != ""){
6998 0 : telescope_p=sdObs;
6999 0 : ObsInfo myobsinfo=this->latestObsInfo();
7000 0 : myobsinfo.setTelescope(telescope_p);
7001 0 : coordsys.setObsInfo(myobsinfo);
7002 0 : }
7003 : else{
7004 0 : telescope_p=coordsys.obsInfo().telescope();
7005 : }
7006 : }
7007 0 : beamTemp.setCoordinateInfo(coordsys);
7008 0 : this->makePBImage(beamTemp);
7009 :
7010 : }
7011 : else{
7012 0 : Table vpTable(vpTableStr_p);
7013 0 : this->makePBImage(vpTable, beamTemp);
7014 0 : }
7015 0 : StokesImageUtil::FitGaussianPSF(beamTemp, lBeam);
7016 0 : LatticeExprNode sumImage = sum(beamTemp);
7017 0 : beamFactor=sumImage.getFloat();
7018 :
7019 0 : }
7020 :
7021 :
7022 0 : }
7023 : else {
7024 : os << LogIO::NORMAL // Loglevel INFO
7025 : << "Using specified low resolution PSF to determine sd flux scale"
7026 0 : << LogIO::POST;
7027 : // regrid the single dish psf
7028 0 : PagedImage<Float> lowpsf0(lowPSF);
7029 0 : TempImage<Float> lowpsf(model.shape(), model.coordinates());
7030 : {
7031 0 : ImageRegrid<Float> ir;
7032 0 : IPosition axes(2,0,1); //
7033 0 : ir.regrid(lowpsf, Interpolate2D::LINEAR, axes, lowpsf0);
7034 0 : }
7035 0 : LatticeExprNode sumImage = sum(lowpsf);
7036 0 : beamFactor=sumImage.getFloat();
7037 0 : if(lBeam.isNull()) {
7038 0 : os << LogIO::NORMAL << "Finding SD beam from given PSF" << LogIO::POST; // Loglevel PROGRESS
7039 0 : StokesImageUtil::FitGaussianPSF(lowpsf0, lBeam);
7040 : }
7041 0 : }
7042 :
7043 :
7044 : // This factor comes from the beam volumes
7045 0 : if(sdScale_p!=1.0)
7046 : os << LogIO::DEBUG1
7047 : << "Multiplying single dish data by user specified factor "
7048 0 : << sdScale_p << LogIO::POST;
7049 0 : Float sdScaling = sdScale_p;
7050 0 : if(! lBeam.isNull()) {
7051 0 : Int directionIndex=model.coordinates().findCoordinate(Coordinate::DIRECTION);
7052 : DirectionCoordinate
7053 0 : directionCoord=model.coordinates().directionCoordinate(directionIndex);
7054 0 : Vector<String> units(2); units.set("arcsec");
7055 0 : directionCoord.setWorldAxisUnits(units);
7056 0 : Vector<Double> incr= directionCoord.increment();
7057 0 : if(beamFactor >0.0) {
7058 0 : beamFactor=1.0/beamFactor;
7059 : }
7060 : else{
7061 : // beamFactor=
7062 : // abs(incr(0)*incr(1))/(lBeam(0).get("arcsec").getValue()*lBeam(1).get("arcsec").getValue()*1.162);
7063 : //Brute Force for now.
7064 0 : IPosition imshape(4, nx_p, ny_p, 1, 1);
7065 0 : TempImage<Float> lowpsf(imshape, coordsys);
7066 0 : lowpsf.set(0.0);
7067 0 : IPosition center(4, Int((nx_p/4)*2), Int((ny_p/4)*2),0,0);
7068 0 : lowpsf.putAt(1.0, center);
7069 0 : StokesImageUtil::Convolve(lowpsf, lBeam, false);
7070 0 : LatticeExprNode sumImage = sum(lowpsf);
7071 0 : beamFactor=1.0/sumImage.getFloat();
7072 :
7073 :
7074 0 : }
7075 : os << LogIO::NORMAL << "Beam volume factor " // Loglevel INFO
7076 0 : << beamFactor << LogIO::POST;
7077 0 : sdScaling*=beamFactor;
7078 0 : }
7079 : else {
7080 0 : os << LogIO::WARN << "Insufficient information to scale correctly" << LogIO::POST;
7081 : }
7082 :
7083 : //Convert to Jy/pixel
7084 0 : model.copyData( (LatticeExpr<Float>)((model * sdScaling)));
7085 0 : model.setUnits(Unit("Jy/pixel"));
7086 :
7087 : //make a mask image
7088 0 : this->makeEmptyImage(coordsys, maskImage, fieldid_p);
7089 0 : PagedImage<Float> mask(maskImage);
7090 0 : mask.set(1.0);
7091 0 : ArrayLattice<Bool> sdMask(model.getMask());
7092 0 : mask.copyData( LatticeExpr<Float> (mask* ntrue(sdMask)*model));
7093 0 : StokesImageUtil::MaskFrom(mask, mask, Quantity(0.0, "Jy"));
7094 0 : model.copyData( LatticeExpr<Float> (mask*model));
7095 0 : return true;
7096 0 : } catch (AipsError x) {
7097 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
7098 0 : << LogIO::POST;
7099 0 : return false;
7100 0 : }
7101 :
7102 : return true;
7103 :
7104 :
7105 0 : }
7106 :
7107 :
7108 0 : Int Imager::interactivemask(const String& image, const String& mask,
7109 : Int& niter, Int& ncycles, String& thresh, const Bool forceReload){
7110 :
7111 0 : LogIO os(LogOrigin("Imager", "interactivemask()", WHERE));
7112 0 : if(Table::isReadable(mask)) {
7113 0 : if (! Table::isWritable(mask)) {
7114 0 : os << LogIO::WARN << "Mask image is not modifiable " << LogIO::POST;
7115 0 : return false;
7116 : }
7117 : //we should regrid here if image and mask do not match
7118 : }
7119 : else{
7120 0 : clone(image, mask);
7121 : }
7122 : #ifdef USE_GRPC
7123 0 : Quantity thr;
7124 0 : if ( ! Quantity::read(thr,thresh) ) thr = Quantity(0,"Jy");
7125 0 : float thold = (float) thr.get("Jy").getValue( );
7126 0 : grpcInteractiveClean::getManager( ).setControls( niter, ncycles, thold);
7127 : #endif
7128 0 : return false;
7129 0 : }
7130 :
7131 0 : Record Imager::iClean(const String& algorithm, const Int niter, const Double gain,
7132 : const Quantity& threshold,
7133 : const Bool displayprogress,
7134 : const Vector<String>& model,
7135 : const Vector<Bool>& keepfixed, const String& complist,
7136 : const Vector<String>& mask,
7137 : const Vector<String>& image,
7138 : const Vector<String>& residual,
7139 : const Vector<String>& psfnames,
7140 : const Bool interactive, const Int npercycle,
7141 : const String& masktemplate)
7142 : {
7143 0 : Record rstat;
7144 :
7145 0 : logSink_p.clearLocally();
7146 0 : LogIO os(LogOrigin("imager", "iClean()"), logSink_p);
7147 :
7148 0 : if(!ms_p.null()) {
7149 : //try
7150 : {
7151 :
7152 0 : Vector<String> amodel(model);
7153 0 : Vector<Bool> fixed(keepfixed);
7154 0 : Vector<String> amask(mask);
7155 0 : Vector<String> aimage(image);
7156 0 : Vector<String> aresidual(residual);
7157 0 : Vector<String> apsf(psfnames);
7158 :
7159 0 : if(String(algorithm) != "msmfs") ntaylor_p=1; /* masks increment by ntaylor_p only for msmfs */
7160 0 : uInt nmods = aresidual.nelements()/ntaylor_p;
7161 :
7162 0 : if( (apsf.nelements()==1) && apsf[0]==String(""))
7163 0 : apsf.resize();
7164 0 : if(!interactive){
7165 0 : rstat = clean(String(algorithm), niter, gain,
7166 : threshold, displayprogress,
7167 0 : amodel, fixed, String(complist), amask,
7168 0 : aimage, aresidual, apsf);
7169 : }
7170 : else{
7171 0 : if((amask.nelements()==0) || (amask[0]==String(""))){
7172 0 : amask.resize(amodel.nelements());
7173 0 : for (uInt k=0; k < amask.nelements(); ++k){
7174 0 : amask[k]=amodel[k]+String(".mask");
7175 : }
7176 : }
7177 0 : Vector<Bool> nointerac(nmods);
7178 0 : nointerac.set(false);
7179 0 : if(fixed.nelements() != nmods){
7180 0 : fixed.resize(nmods);
7181 0 : fixed.set(false);
7182 : }
7183 0 : Bool forceReload=true;
7184 0 : Int nloop=0;
7185 0 : if(npercycle != 0)
7186 0 : nloop=niter/npercycle;
7187 0 : Int continter=0;
7188 0 : Int elniter=npercycle;
7189 0 : ostringstream oos;
7190 0 : threshold.print(oos);
7191 0 : String thresh=String(oos);
7192 0 : if(String(masktemplate) != String("")){
7193 0 : continter=interactivemask(masktemplate, amask[0],
7194 : elniter, nloop, thresh);
7195 : }
7196 : else {
7197 : // do a zero component clean to get started
7198 0 : rstat=clean(String(algorithm), 0, gain,
7199 : threshold, displayprogress,
7200 0 : amodel, fixed, String(complist), amask,
7201 0 : aimage, aresidual, Vector<String>(0), false);
7202 :
7203 0 : for (uInt nIm=0; nIm < nmods; nIm++){ //=ntaylor_p){
7204 0 : if(Table::isReadable(aimage[nIm]) && Table::isWritable(aresidual[nIm]) ){
7205 0 : PagedImage<Float> rest(aimage[nIm]);
7206 0 : PagedImage<Float> resi(aresidual[nIm]);
7207 0 : copyMask(resi, rest, "mask0");
7208 0 : }
7209 0 : forceReload=forceReload || (aresidual.nelements() >1);
7210 0 : continter=interactivemask(aresidual[nIm], amask[nIm],
7211 : elniter, nloop,thresh, forceReload);
7212 0 : forceReload=false;
7213 0 : if(continter>=1)
7214 0 : nointerac(nIm)=true;
7215 0 : if(continter==2)
7216 0 : fixed(nIm)=true;
7217 :
7218 : }
7219 0 : if(allEQ(nointerac, true)){
7220 0 : elniter=niter;
7221 : //make it do one more loop/clean but with all niter
7222 0 : nloop=1;
7223 : }
7224 : }
7225 0 : for (Int k=0; k < nloop; ++k){
7226 :
7227 0 : casacore::Quantity thrsh;
7228 0 : if(!casacore::Quantity::read(thrsh, thresh)){
7229 : os << LogIO::WARN << "Error interpreting threshold"
7230 0 : << LogIO::POST;
7231 0 : thrsh=casacore::Quantity(0, "Jy");
7232 0 : thresh="0.0Jy";
7233 : }
7234 0 : Vector<String> elpsf(0);
7235 : //Need to save psfs in interactive only once and lets do it the
7236 : //first time
7237 0 : if(k==0)
7238 0 : elpsf=apsf;
7239 0 : if(anyEQ(fixed, false)){
7240 0 : rstat = clean(String(algorithm), elniter, gain,
7241 : thrsh,
7242 : displayprogress,
7243 0 : amodel, fixed, String(complist),
7244 : amask,
7245 0 : aimage, aresidual, elpsf, k == 0);
7246 : //if clean converged... equivalent to stop
7247 0 : if(rstat.asBool("converged")){
7248 0 : continter=2;
7249 0 : fixed.set(true);
7250 : }
7251 0 : if(anyEQ(fixed, false) && anyEQ(nointerac,false)){
7252 0 : Int remainloop=nloop-k-1;
7253 0 : for (uInt nIm=0; nIm < nmods; nIm++){ //=ntaylor_p){
7254 0 : if(!nointerac(nIm)){
7255 0 : continter=interactivemask(aresidual[nIm], amask[nIm],
7256 :
7257 : elniter, remainloop,
7258 0 : thresh, (aresidual.nelements() >1));
7259 0 : if(continter>=1)
7260 0 : nointerac(nIm)=true;
7261 0 : if(continter==2)
7262 0 : fixed(nIm)=true;
7263 : }
7264 : }
7265 0 : k=nloop-remainloop-1;
7266 0 : if(allEQ(nointerac,true)){
7267 0 : elniter=niter-(k+1)*npercycle;
7268 : //make it do one more loop/clean but with remaining niter
7269 0 : k=nloop-2;
7270 : }
7271 : }
7272 : }
7273 0 : }
7274 : //Unset the mask in the residual
7275 : // Cause as requested in CAS-1768...
7276 0 : for (uInt nIm=0; nIm < aresidual.nelements(); ++nIm){
7277 0 : if(Table::isWritable(aresidual[nIm]) ){
7278 0 : PagedImage<Float> resi(aresidual[nIm]);
7279 0 : if(resi.hasRegion("mask0", RegionHandler::Masks)){
7280 0 : resi.setDefaultMask("");
7281 : }
7282 0 : }
7283 : }
7284 :
7285 0 : }
7286 :
7287 0 : os << "Restoring Image(s) with the clean-beam" << LogIO::POST;
7288 0 : restoreImages(aimage, (niter>=0) );
7289 0 : this->writeHistory(os);
7290 : try{
7291 : // write data processing history into image logtable
7292 0 : LoggerHolder imagelog (false);
7293 0 : LogSink& sink = imagelog.sink();
7294 0 : LogOrigin lor( String("imager"), String("clean()") );
7295 0 : LogMessage msg(lor);
7296 0 : sink.postLocally(msg);
7297 0 : MSHistoryColumns msHis(ms_p->history());
7298 0 : transferHistory(imagelog, msHis);
7299 0 : for (Int thismodel=0;thismodel<Int(aimage.nelements());++thismodel) {
7300 0 : if(Table::isWritable(aimage(thismodel))){
7301 0 : PagedImage<Float> restoredImage(aimage(thismodel),
7302 0 : TableLock(TableLock::AutoNoReadLocking));
7303 0 : LoggerHolder& log = restoredImage.logger();
7304 0 : log.append(imagelog);
7305 0 : log.flush();
7306 0 : restoredImage.table().relinquishAutoLocks(true);
7307 0 : }
7308 : }
7309 0 : }
7310 0 : catch(exception& x){
7311 :
7312 0 : this->unlock();
7313 0 : destroySkyEquation();
7314 0 : os << LogIO::WARN << "Caught exception: " << x.what()
7315 0 : << LogIO::POST;
7316 : os << LogIO::SEVERE << "This means your MS/HISTORY table may be corrupted; you may consider deleting all the rows from this table"
7317 0 : <<LogIO::POST;
7318 : //continue and wrap up this function
7319 :
7320 0 : }
7321 0 : catch(...){
7322 0 : this->unlock();
7323 0 : destroySkyEquation();
7324 0 : os << LogIO::WARN << "Caught unknown exception" << LogIO::POST;
7325 : os << LogIO::SEVERE << "The MS/HISTORY table may be corrupted; you may consider deleting all the rows from this table"
7326 0 : <<LogIO::POST;
7327 :
7328 0 : }
7329 :
7330 :
7331 0 : } //catch (AipsError x) {
7332 : //os << LogIO::SEVERE << "Exception Reported: " << x.getMesg() << LogIO::POST;
7333 : // RETHROW(x);
7334 : // }
7335 : } else {
7336 0 : os << LogIO::SEVERE << "No MeasurementSet has been assigned, please run open." << LogIO::POST;
7337 : }
7338 0 : return rstat;
7339 0 : }
7340 :
7341 0 : Bool Imager::adviseChanSelex(Double& freqStart, Double& freqEnd,
7342 : const Double& freqStep, const MFrequency::Types& freqframe,
7343 : Vector< Vector<Int> >& spw, Vector< Vector<Int> >& start,
7344 : Vector< Vector<Int> >& nchan, const String& ms, const Int field_id, const Bool getFreqRange, const String spwselection){
7345 :
7346 0 : LogIO os(LogOrigin("imager", "adviseChanSelex"));
7347 0 : if(ms==String("")){
7348 0 : if(numMS_p < 1 || !rvi_p){
7349 : os << LogIO::SEVERE << "Data selection incomplete"
7350 0 : << LogIO::POST;
7351 0 : return false;
7352 : }
7353 : }
7354 0 : spw.resize();
7355 0 : start.resize();
7356 0 : nchan.resize();
7357 : try {
7358 0 : if(!getFreqRange){
7359 0 : Block<Vector<Int> > bnchan;
7360 0 : Block<Vector<Int> > bstart;
7361 0 : Block<Vector<Int> > bspw;
7362 : Double fS, fE;
7363 0 : fS=freqStart;
7364 0 : fE=freqEnd;
7365 0 : if(freqEnd < freqStart){
7366 0 : fS=freqEnd;
7367 0 : fE=freqStart;
7368 : }
7369 :
7370 0 : if(ms==String("")){
7371 0 : rvi_p->getSpwInFreqRange(bspw, bstart, bnchan, fS, fE, fabs(freqStep), freqframe);
7372 : }
7373 : else{
7374 0 : bnchan.resize(1);
7375 0 : bstart.resize(1);
7376 0 : bspw.resize(1);
7377 0 : MeasurementSet elms(String(ms), TableLock(TableLock::AutoNoReadLocking), Table::Old);
7378 0 : MSUtil::getSpwInFreqRange(bspw[0], bstart[0], bnchan[0], elms, fS, fE, fabs(freqStep), freqframe, field_id);
7379 0 : elms.relinquishAutoLocks(true);
7380 :
7381 0 : }
7382 0 : spw=Vector<Vector<Int> >(bspw.begin( ), bspw.end( ));
7383 0 : start=Vector<Vector<Int> >(bstart.begin( ), bstart.end( ));
7384 0 : nchan=Vector<Vector<Int> >(bnchan.begin( ), bnchan.end( ));
7385 0 : }
7386 : else{
7387 0 : if(ms==String("")){
7388 0 : rvi_p->getFreqInSpwRange(freqStart, freqEnd, freqframe);
7389 : }
7390 : else{
7391 0 : MeasurementSet elms(ms, TableLock(TableLock::AutoNoReadLocking), Table::Old);
7392 0 : MSSelection thisSelection;
7393 0 : String spsel=spwselection;
7394 0 : if(spsel=="")spsel="*";
7395 0 : thisSelection.setSpwExpr(spsel);
7396 0 : TableExprNode exprNode=thisSelection.toTableExprNode(&elms);
7397 0 : Matrix<Int> chanlist=thisSelection.getChanList();
7398 0 : if(chanlist.ncolumn() <3){
7399 0 : freqStart=-1.0;
7400 0 : freqEnd=-1.0;
7401 0 : return false;
7402 : }
7403 0 : Vector<Int> elspw=chanlist.column(0);
7404 0 : Vector<Int> elstart=chanlist.column(1);
7405 0 : Vector<Int> elnchan=Vector<Int> (chanlist.column(2)-elstart)+1;
7406 0 : MSUtil::getFreqRangeInSpw(freqStart, freqEnd, elspw, elstart, elnchan, elms, freqframe, field_id);
7407 0 : }
7408 :
7409 : }
7410 :
7411 :
7412 :
7413 :
7414 0 : } catch (AipsError x) {
7415 : os << LogIO::SEVERE << "Caught exception: " << x.getMesg()
7416 0 : << LogIO::POST;
7417 0 : return false;
7418 0 : }
7419 0 : catch (...){
7420 : os << LogIO::SEVERE << "Unknown exception handled"
7421 0 : << LogIO::POST;
7422 0 : return false;
7423 :
7424 0 : }
7425 :
7426 0 : return true;
7427 :
7428 0 : }
7429 :
7430 :
7431 :
7432 : } //# NAMESPACE CASA - END
7433 :
|