Use SAR metadata in SarRadiometricCalibrationToImageFilter
Rationale
Fix #2159 (closed)
Copyright
The copyright owner is CNES and has signed the ORFEO ToolBox Contributor License Agreement.
Check before merging:
- All discussions are resolved
- At least 2
👍 votes from core developers, no👎 vote. - The feature branch is (reasonably) up-to-date with the base branch
- Dashboard is green
- Copyright owner has signed the ORFEO ToolBox Contributor License Agreement
- Optionally, run
git diff develop... -U0 --no-color | clang-format-diff.py -p1 -i
on latest changes and commit
Merge request reports
Activity
changed milestone to %8.0.0
added refactoring label
added 2 commits
added 1 commit
- 52fac90e - ENH: Separate Calib data from SARParam. Add CheckFetch methods.
added 1 commit
- efac1e8b - BUG: Deal with CalScale in different bands. Add SARCalib in ToJSON
added 1 commit
- 3bb30cc1 - BUG: Add the geom param to LoadRadiometricCalibrationData
added 1 commit
- f38ec549 - FEAT: Add compatibility for geom files that have missing manifest_data keys
17 * See the License for the specific language governing permissions and 18 * limitations under the License. 19 */ 20 21 #ifndef otbSarSensorModel_h 22 #define otbSarSensorModel_h 23 24 #include "otbImageMetadata.h" 25 26 27 #include "itkPoint.h" 28 29 namespace otb 30 { 31 32 class SarSensorModel 296 296 297 297 } 298 298 299 std::vector<Orbit> CosmoImageMetadataInterface::getOrbits(const MetadataSupplierInterface & mds, const std::string & reference_UTC) 299 std::vector<Orbit> CosmoImageMetadataInterface::getOrbits(const std::string & reference_UTC) 347 347 return orbitVector ; 348 348 } 349 349 350 void CosmoImageMetadataInterface::ParseGdal(const MetadataSupplierInterface & mds) 350 void CosmoImageMetadataInterface::ParseGdal(ImageMetadata & imd) 351 351 { 352 352 // Check acquisition mode and product type 353 Fetch(MDStr::Mode, mds, "Acquisition_Mode"); 354 if((m_Imd[MDStr::Mode] != "HIMAGE") && 355 (m_Imd[MDStr::Mode] != "SPOTLIGHT") && 356 (m_Imd[MDStr::Mode] != "ENHANCED SPOTLIGHT")) 353 Fetch(MDStr::Mode, imd, "Acquisition_Mode"); 354 if((imd[MDStr::Mode] != "HIMAGE") && {"HIMAGE", "SPOTLIGHT"...}
is iterable. I've bugged on the complex expression of the test that spawns on 3 lines with increased indentation everytime.I haven't used the right algorithm given the test, but this is the idea. https://quick-bench.com/q/3oJTX53iQqKc1r6YM8TIhx9T-AM
Then the test could become
if (otb::not_in(m_Imd[MDStr::Mode], {"HIMAGE", "SPOTLIGHT", "ENHANCED SPOTLIGHT"}) // for instance
BTW, I see it's duplicated elsewhere as well. It'd be best to factorize the duplications of these 3 cases.
230 229 231 230 virtual bool CanRead() const {return false;} 232 231 233 virtual void Parse(const MetadataSupplierInterface &) 234 { 232 void SupplierCheck() const 233 { 234 if (m_MetadataSupplierInterface == nullptr) 235 throw std::runtime_error("In ImageMetadataInterface, the MetadataSupplier needs to be set before calling the Parse function."); 236 } 237 238 virtual void Parse(ImageMetadata &) 96 96 double GetCenterIncidenceAngle() const override; 97 97 98 98 /*get lookup data for calculating backscatter */ 99 void CreateCalibrationLookupData(const short type) override; 99 const LookupDataPointerType CreateCalibrationLookupData(const short type) const override; Why not use thing like
std::put_time
https://en.cppreference.com/w/cpp/io/manip/put_time?
added 15 commits
- db1fe22b - REFAC: Created a new SARSensorModel class SarSensorModel as a replacement of SarSensorModelAdapter
- d2284d21 - REFAC: use itk::Point instead of individual coordinates in Orbit parameters
- fd671c4a - BUG: update expat version in Superbuild (security issue)
- 9cfd44d1 - REFAC: implement WorldToLineSample (adapted from ossim plugins)
- 2adf8cba - TEST: validate SAR inverse transform using product GCPs
- ba383d3b - REFAC: add support for cosmo-skymed products in SarSensorModel
- 5b625e3d - FIX: Update Baseline for ioTuMetaDataKeyTest
- 193b4f06 - BUG: Fix key paths in Sentinel1IMI
- f22b8204 - BUG: Update baseline with SARCalib
- fc0f84b3 - ENH: Add GetNumberOf function to MetadataSupplierInterface
- 3367d6e5 - WIP: Parse Calibration metadata for TerrasarX
- 4a3140d3 - BUG: correct date formatting when time components have only one digit
- f6b41eb0 - Merge branch '2160-SarSensorModel' into 2159
- fc1b5242 - ENH: Add GetJulianDay and GetModifiedJulian functions to MetaData::Time
- 5217bf98 - BUG: Parse Calibration metadata for TerrasarX
Toggle commit listmarked this merge request as draft from 3367d6e5
added 20 commits
-
5217bf98...ebb01245 - 19 commits from branch
develop
- 77583d2a - Merge branch 'develop' into 2159
-
5217bf98...ebb01245 - 19 commits from branch
added 1 commit
- 86261af7 - ENH: Implement CreateCalibrationLookupData for Radarsat2ImageMetadataInterface
added 1 commit
- 28e07a0c - BUG: Deal with SARCalib metadata stored in bands
383 389 line = (timeSinceStart/m_SarParam.azimuthTimeInterval) + currentBurst->startLine; 384 390 } 385 391 392 void SarSensorModel::SlantRangeToGroundRange(const double & slantRange, const TimeType & azimuthTime, double & groundRange) const 393 { 394 ApplyCoordinateConversion(slantRange, azimuthTime, m_SarParam.slantRangeToGroundRangeRecords , groundRange); 395 } 396 397 void SarSensorModel::ApplyCoordinateConversion(const double & in, 398 const TimeType& azimuthTime, 399 const std::vector<CoordinateConversionRecord> & records, 400 double & out) const 401 { All these loops can be replaced with something more concise (and I guess the other as well). In SRGR case I already did the work in an unpublished branch
Note that we don't need to copy the coef vector (
srgrRecord = records.back();
x 2), but as I left this optimization in comments I guess the difference wasn't perceptible -- I my benchs most of the time was lost in the use of MAT library (too many allocations)void ossimSarSensorModel::applyCoordinateConversion(const double & in, const TimeType& azimuthTime, const std::vector<CoordinateConversionRecordType> & records, double & out) const { assert(!records.empty()&&"The records vector is empty."); // std::clog << "conv coord(" << in << ", az="<<azimuthTime<<")\n"; // First, we need to find the correct pair of records for interpolation auto it = records.begin(); CoordinateConversionRecordType srgrRecord; auto nextRecord = std::find_if(std::begin(records), std::end(records), [azimuthTime](auto const& record) { return azimuthTime < record.azimuthTime; } ); // Note: I did try with upper_bound instead of find_if (as the elements are sorted, but there aren't enough elements to make a perceptible difference assert(nextRecord == records.end() || nextRecord->azimuthTime > azimuthTime); if (nextRecord == records.end()) { srgrRecord = records.back(); // ### don't copy the coefs vector // auto const& ref = records.back(); // srgrRecord.azimuthTime = ref.azimuthTime; // srgrRecord.rg0 = ref.rg0 ; } else if (nextRecord == records.begin()) { assert(azimuthTime < nextRecord->azimuthTime); srgrRecord = records.front(); // ### don't copy the coefs vector // auto const& ref = records.front(); // srgrRecord.azimuthTime = ref.azimuthTime; // srgrRecord.rg0 = ref.rg0 ; } else { auto previousRecord = nextRecord - 1; assert(previousRecord->azimuthTime <= azimuthTime); assert(azimuthTime < nextRecord->azimuthTime); assert(nextRecord != records.end()); assert(!previousRecord->coefs.empty()&&"previousRecord coefficients vector is empty."); assert(!nextRecord->coefs.empty()&&"nextRecord coefficients vector is empty."); // If azimuth time is between 2 records, interpolate const double interp = DurationType(azimuthTime - previousRecord->azimuthTime) / (nextRecord->azimuthTime - previousRecord->azimuthTime) ; // std::clog << "interp: " << interp << " =" // << " (" << azimuthTime << " - " << previousRecord->azimuthTime << " (="<< (azimuthTime - previousRecord->azimuthTime)<< ") )" // << "/(" << nextRecord->azimuthTime << " - " << previousRecord->azimuthTime << " (="<< (nextRecord->azimuthTime - previousRecord->azimuthTime)<< ") )" // << "\n"; srgrRecord.rg0 = (1.0-interp) * previousRecord->rg0 + interp*nextRecord->rg0; srgrRecord.coefs.clear(); auto pIt = previousRecord->coefs.begin(); auto nIt = nextRecord->coefs.begin(); srgrRecord.coefs.reserve(std::min(previousRecord->coefs.size(), nextRecord->coefs.size())); for(;pIt != previousRecord->coefs.end() && nIt != nextRecord->coefs.end();++pIt,++nIt) { srgrRecord.coefs.push_back(interp*(*nIt)+(1.0-interp)*(*pIt)); } assert(!srgrRecord.coefs.empty()&&"Slant range to ground range interpolated coefficients vector is empty."); } // Now that we have the interpolated coefs, compute ground range // from slant range const double sr_minus_sr0 = in-srgrRecord.rg0; assert(!srgrRecord.coefs.empty()&&"Slant range to ground range coefficients vector is empty."); double rout = 0.0; // avoid aliasing in loop for(auto cIt = srgrRecord.coefs.rbegin();cIt!=srgrRecord.coefs.rend();++cIt) { rout = *cIt + sr_minus_sr0*rout; } out = rout; }
Eigen3 has been integrated into ITK release 5.0.0. Quoting the ITK release notes:
Eigen, a C++ library for linear algebra, was integrated in ITK and gradual transition from VXL to Eigen for Linear algebra is planned
It looks like a long term goal of ITK is to replace uses of VXL by Eigen (e.g. use vectors from Eigen in itk::VariableLengthVector). See for example https://discourse.itk.org/t/add-internal-third-party-module-eigen3/1429
Not that in current release (5.2.1), vnl objects are still used as internal type of ITK vector and matrix objects.
added 8 commits
- 7235f699 - REFAC: add support for TerraSAR-X products in SarSensorModel
- 009b2b90 - TEST: add a test for SarSensorModel using a TerraSAR-X product
- d4f3ba79 - ENH: read noise range LUT in S1 products generated by IPF version > 2.90
- ae97a181 - ENH: add support for GRD and multi-burst products
- 23517628 - COMP: merge 2159 into 2160-SarSensorModel
- f8eeec26 - BUG: Read noise metadata for TerrasarX
- 5ec85f33 - WIP: Read band-dependant metadata
- ab26cbd3 - Merge branch '2160-SarSensorModel' into 2159
Toggle commit list509 double minDistance = std::numeric_limits<double>::max(); 510 511 for (const auto & elem : gcpParam.GCPs) 512 { 513 auto currentDistance = squaredDistance(imPt, elem); 514 if (currentDistance < minDistance) 515 { 516 minDistance = currentDistance; 517 closest = &elem; 518 } 519 } 520 521 return *closest; 522 } 523 524 void SarSensorModel::projToSurface(const GCP & gcp, const Point2DType& imPt, double heightAboveEllipsoid, Point3DType & ecefPoint) const 134 134 135 135 } 136 136 137 void SarSensorModel::LineSampleHeightToWorld(const Point2DType& imPt, 138 const double & heightAboveEllipsoid, 139 Point3DType& worldPt) const 140 { 141 assert(m_Imd.Has(MDGeom::GCP)); 142 143 const auto gcp = findClosestGCP(imPt, m_Imd.GetGCPParam()); added 4 commits
Toggle commit listadded 1 commit
- ba18f577 - BUG: Read terrasar's polarization later prevents surcharging polarization for other sensors
added 1 commit
- 8b32a001 - BUG: Correctly read sentinel1's instrument metadata
added 8 commits
- 0965c370 - ENH: implements LineSampleHeightToWorld in SarSensorModel
- 9cd225f3 - Merge remote-tracking branch 'origin/2159' into 2160-SarSensorModel
- cfb3932d - PERF: copy input doubles instead of using const references
- 791a115f - ENH: various optimizations (code review)
- fb6e996b - ENH: add test for the forward sensor model
- 08a2be01 - ENH: add a test for Sentinel1 SLC geom product
- 7d467b7d - Merge remote-tracking branch 'origin/2160-SarSensorModel' into 2159
- b631f032 - REFAC: use mono-band (scalar or complex) products in SarSensorModel
Toggle commit listadded 4 commits
Toggle commit listadded 1 commit
- dff9aad5 - BUG: handle old geoms when reading burst informations
added 1 commit
- 08e6d26f - BUG: throw a MissingMetadataException for invalidd Cosmo product types
added 1 commit
- 4420544c - BUG: Correctly construct GeomMetadataSupplier