LCOV - code coverage report
Current view: top level - frmts/jaxapalsar - jaxapalsardataset.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 19 246 7.7 %
Date: 2025-01-18 12:42:00 Functions: 2 15 13.3 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  PALSAR JAXA imagery reader
       4             :  * Purpose:  Support for PALSAR L1.1/1.5 imagery and appropriate metadata from
       5             :  *           JAXA and JAXA-supported ground stations (ASF, ESA, etc.). This
       6             :  *           driver does not support ERSDAC products.
       7             :  * Author:   Philippe Vachon <philippe@cowpig.ca>
       8             :  *
       9             :  ******************************************************************************
      10             :  * Copyright (c) 2007, Philippe P. Vachon <philippe@cowpig.ca>
      11             :  * Copyright (c) 2008-2011, Even Rouault <even dot rouault at spatialys.com>
      12             :  *
      13             :  * SPDX-License-Identifier: MIT
      14             :  ****************************************************************************/
      15             : 
      16             : #include "gdal_frmts.h"
      17             : #include "gdal_pam.h"
      18             : 
      19             : #if defined(_WIN32)
      20             : #define SEP_STRING "\\"
      21             : #else
      22             : #define SEP_STRING "/"
      23             : #endif
      24             : 
      25             : /* read binary fields */
      26             : #ifdef CPL_LSB
      27             : #define READ_WORD(f, x)                                                        \
      28             :     do                                                                         \
      29             :     {                                                                          \
      30             :         VSIFReadL(&(x), 4, 1, (f));                                            \
      31             :         (x) = CPL_SWAP32((x));                                                 \
      32             :     } while (false);
      33             : #define READ_SHORT(f, x)                                                       \
      34             :     do                                                                         \
      35             :     {                                                                          \
      36             :         VSIFReadL(&(x), 2, 1, (f));                                            \
      37             :         (x) = CPL_SWAP16((x));                                                 \
      38             :     } while (false);
      39             : #else
      40             : #define READ_WORD(f, x)                                                        \
      41             :     do                                                                         \
      42             :     {                                                                          \
      43             :         VSIFReadL(&(x), 4, 1, (f));                                            \
      44             :     } while (false);
      45             : #define READ_SHORT(f, x)                                                       \
      46             :     do                                                                         \
      47             :     {                                                                          \
      48             :         VSIFReadL(&(x), 2, 1, (f));                                            \
      49             :     } while (false);
      50             : #endif /* def CPL_LSB */
      51             : #define READ_BYTE(f, x)                                                        \
      52             :     do                                                                         \
      53             :     {                                                                          \
      54             :         VSIFReadL(&(x), 1, 1, (f));                                            \
      55             :     } while (false);
      56             : 
      57             : /* read floating point value stored as ASCII */
      58             : #define READ_CHAR_FLOAT(n, l, f)                                               \
      59             :     do                                                                         \
      60             :     {                                                                          \
      61             :         char psBuf[(l + 1)];                                                   \
      62             :         psBuf[(l)] = '\0';                                                     \
      63             :         VSIFReadL(&psBuf, (l), 1, (f));                                        \
      64             :         (n) = CPLAtof(psBuf);                                                  \
      65             :     } while (false);
      66             : 
      67             : /* read numbers stored as ASCII */
      68             : #define READ_CHAR_VAL(x, n, f)                                                 \
      69             :     do                                                                         \
      70             :     {                                                                          \
      71             :         char psBuf[(n + 1)];                                                   \
      72             :         psBuf[(n)] = '\0';                                                     \
      73             :         VSIFReadL(&psBuf, (n), 1, (f));                                        \
      74             :         (x) = atoi(psBuf);                                                     \
      75             :     } while (false);
      76             : 
      77             : /* read string fields
      78             :  * note: string must be size of field to be extracted + 1
      79             :  */
      80             : #define READ_STRING(s, n, f)                                                   \
      81             :     do                                                                         \
      82             :     {                                                                          \
      83             :         VSIFReadL(&(s), 1, (n), (f));                                          \
      84             :         (s)[(n)] = '\0';                                                       \
      85             :     } while (false);
      86             : 
      87             : /*************************************************************************/
      88             : /* a few key offsets in the volume directory file */
      89             : #define VOL_DESC_RECORD_LENGTH 360
      90             : #define FILE_PTR_RECORD_LENGTH 360
      91             : #define NUM_RECORDS_OFFSET 160
      92             : 
      93             : /* a few key offsets and values within the File Pointer record */
      94             : #define REF_FILE_CLASS_CODE_OFFSET 66
      95             : #define REF_FILE_CLASS_CODE_LENGTH 4
      96             : #define FILE_NAME_OFFSET 310
      97             : 
      98             : /* some image option descriptor records */
      99             : #define BITS_PER_SAMPLE_OFFSET 216
     100             : #define BITS_PER_SAMPLE_LENGTH 4
     101             : #define SAMPLES_PER_GROUP_OFFSET 220
     102             : #define SAMPLES_PER_GROUP_LENGTH 4
     103             : #define NUMBER_LINES_OFFSET 236
     104             : #define NUMBER_LINES_LENGTH 8
     105             : #define SAR_DATA_RECORD_LENGTH_OFFSET 186
     106             : #define SAR_DATA_RECORD_LENGTH_LENGTH 6
     107             : 
     108             : #define IMAGE_OPT_DESC_LENGTH 720
     109             : 
     110             : #define SIG_DAT_REC_OFFSET 412
     111             : #define PROC_DAT_REC_OFFSET 192
     112             : 
     113             : /* metadata to be extracted from the leader file */
     114             : #define LEADER_FILE_DESCRIPTOR_LENGTH 720
     115             : #define DATA_SET_SUMMARY_LENGTH 4096
     116             : 
     117             : /* relative to end of leader file descriptor */
     118             : #define EFFECTIVE_LOOKS_AZIMUTH_OFFSET 1174 /* floating point text */
     119             : #define EFFECTIVE_LOOKS_AZIMUTH_LENGTH 16
     120             : 
     121             : /* relative to leader file descriptor + dataset summary length */
     122             : #define PIXEL_SPACING_OFFSET 92
     123             : #define LINE_SPACING_OFFSET 108
     124             : #define ALPHANUMERIC_PROJECTION_NAME_OFFSET 412
     125             : #define TOP_LEFT_LAT_OFFSET 1072
     126             : #define TOP_LEFT_LON_OFFSET 1088
     127             : #define TOP_RIGHT_LAT_OFFSET 1104
     128             : #define TOP_RIGHT_LON_OFFSET 1120
     129             : #define BOTTOM_RIGHT_LAT_OFFSET 1136
     130             : #define BOTTOM_RIGHT_LON_OFFSET 1152
     131             : #define BOTTOM_LEFT_LAT_OFFSET 1168
     132             : #define BOTTOM_LEFT_LON_OFFSET 1184
     133             : 
     134             : namespace gdal::PSALSARJaxa
     135             : {
     136             : /* a few useful enums */
     137             : enum eFileType
     138             : {
     139             :     level_11 = 0,
     140             :     level_15,
     141             :     level_10,
     142             :     level_unknown = 999,
     143             : };
     144             : 
     145             : enum ePolarization
     146             : {
     147             :     hh = 0,
     148             :     hv,
     149             :     vh,
     150             :     vv
     151             : };
     152             : }  // namespace gdal::PSALSARJaxa
     153             : 
     154             : using namespace gdal::PSALSARJaxa;
     155             : 
     156             : /************************************************************************/
     157             : /* ==================================================================== */
     158             : /*                        PALSARJaxaDataset                             */
     159             : /* ==================================================================== */
     160             : /************************************************************************/
     161             : 
     162             : class PALSARJaxaRasterBand;
     163             : 
     164             : class PALSARJaxaDataset final : public GDALPamDataset
     165             : {
     166             :     friend class PALSARJaxaRasterBand;
     167             : 
     168             :   private:
     169             :     GDAL_GCP *pasGCPList;
     170             :     int nGCPCount;
     171             :     eFileType nFileType;
     172             : 
     173             :   public:
     174             :     PALSARJaxaDataset();
     175             :     ~PALSARJaxaDataset();
     176             : 
     177             :     int GetGCPCount() override;
     178             :     const GDAL_GCP *GetGCPs() override;
     179             : 
     180             :     static GDALDataset *Open(GDALOpenInfo *poOpenInfo);
     181             :     static int Identify(GDALOpenInfo *poOpenInfo);
     182             :     static void ReadMetadata(PALSARJaxaDataset *poDS, VSILFILE *fp);
     183             : };
     184             : 
     185           0 : PALSARJaxaDataset::PALSARJaxaDataset()
     186           0 :     : pasGCPList(nullptr), nGCPCount(0), nFileType(level_unknown)
     187             : {
     188           0 : }
     189             : 
     190           0 : PALSARJaxaDataset::~PALSARJaxaDataset()
     191             : {
     192           0 :     if (nGCPCount > 0)
     193             :     {
     194           0 :         GDALDeinitGCPs(nGCPCount, pasGCPList);
     195           0 :         CPLFree(pasGCPList);
     196             :     }
     197           0 : }
     198             : 
     199             : /************************************************************************/
     200             : /* ==================================================================== */
     201             : /*                        PALSARJaxaRasterBand                          */
     202             : /* ==================================================================== */
     203             : /************************************************************************/
     204             : 
     205             : class PALSARJaxaRasterBand final : public GDALRasterBand
     206             : {
     207             :     VSILFILE *fp;
     208             :     ePolarization nPolarization;
     209             :     eFileType nFileType;
     210             :     int nBitsPerSample;
     211             :     int nSamplesPerGroup;
     212             :     int nRecordSize;
     213             : 
     214             :   public:
     215             :     PALSARJaxaRasterBand(PALSARJaxaDataset *poDS, int nBand, VSILFILE *fp);
     216             :     ~PALSARJaxaRasterBand();
     217             : 
     218             :     CPLErr IReadBlock(int nBlockXOff, int nBlockYOff, void *pImage) override;
     219             : };
     220             : 
     221             : /************************************************************************/
     222             : /*                         PALSARJaxaRasterBand()                       */
     223             : /************************************************************************/
     224             : 
     225           0 : PALSARJaxaRasterBand::PALSARJaxaRasterBand(PALSARJaxaDataset *poDSIn,
     226           0 :                                            int nBandIn, VSILFILE *fpIn)
     227             :     : fp(fpIn), nPolarization(hh), nBitsPerSample(0), nSamplesPerGroup(0),
     228           0 :       nRecordSize(0)
     229             : {
     230           0 :     poDS = poDSIn;
     231           0 :     nBand = nBandIn;
     232             : 
     233             :     /* Read image options record to determine the type of data */
     234           0 :     VSIFSeekL(fp, BITS_PER_SAMPLE_OFFSET, SEEK_SET);
     235           0 :     READ_CHAR_VAL(nBitsPerSample, BITS_PER_SAMPLE_LENGTH, fp);
     236           0 :     READ_CHAR_VAL(nSamplesPerGroup, SAMPLES_PER_GROUP_LENGTH, fp);
     237             : 
     238           0 :     if (nBitsPerSample == 32 && nSamplesPerGroup == 2)
     239             :     {
     240           0 :         eDataType = GDT_CFloat32;
     241           0 :         nFileType = level_11;
     242             :     }
     243           0 :     else if (nBitsPerSample == 8 && nSamplesPerGroup == 2)
     244             :     {
     245           0 :         eDataType = GDT_CInt16; /* shuold be 2 x signed byte */
     246           0 :         nFileType = level_10;
     247             :     }
     248             :     else
     249             :     {
     250           0 :         eDataType = GDT_UInt16;
     251           0 :         nFileType = level_15;
     252             :     }
     253             : 
     254           0 :     poDSIn->nFileType = nFileType;
     255             : 
     256             :     /* Read number of range/azimuth lines */
     257           0 :     VSIFSeekL(fp, NUMBER_LINES_OFFSET, SEEK_SET);
     258           0 :     READ_CHAR_VAL(nRasterYSize, NUMBER_LINES_LENGTH, fp);
     259           0 :     VSIFSeekL(fp, SAR_DATA_RECORD_LENGTH_OFFSET, SEEK_SET);
     260           0 :     READ_CHAR_VAL(nRecordSize, SAR_DATA_RECORD_LENGTH_LENGTH, fp);
     261           0 :     const int nDenom = ((nBitsPerSample / 8) * nSamplesPerGroup);
     262           0 :     if (nDenom != 0)
     263           0 :         nRasterXSize =
     264           0 :             (nRecordSize - (nFileType != level_15 ? SIG_DAT_REC_OFFSET
     265           0 :                                                   : PROC_DAT_REC_OFFSET)) /
     266             :             nDenom;
     267             : 
     268           0 :     poDSIn->nRasterXSize = nRasterXSize;
     269           0 :     poDSIn->nRasterYSize = nRasterYSize;
     270             : 
     271             :     /* Polarization */
     272           0 :     switch (nBand)
     273             :     {
     274           0 :         case 0:
     275           0 :             nPolarization = hh;
     276           0 :             SetMetadataItem("POLARIMETRIC_INTERP", "HH");
     277           0 :             break;
     278           0 :         case 1:
     279           0 :             nPolarization = hv;
     280           0 :             SetMetadataItem("POLARIMETRIC_INTERP", "HV");
     281           0 :             break;
     282           0 :         case 2:
     283           0 :             nPolarization = vh;
     284           0 :             SetMetadataItem("POLARIMETRIC_INTERP", "VH");
     285           0 :             break;
     286           0 :         case 3:
     287           0 :             nPolarization = vv;
     288           0 :             SetMetadataItem("POLARIMETRIC_INTERP", "VV");
     289           0 :             break;
     290             :             // TODO: What about the default?
     291             :     }
     292             : 
     293             :     /* size of block we can read */
     294           0 :     nBlockXSize = nRasterXSize;
     295           0 :     nBlockYSize = 1;
     296             : 
     297             :     /* set the file pointer to the first SAR data record */
     298           0 :     VSIFSeekL(fp, IMAGE_OPT_DESC_LENGTH, SEEK_SET);
     299           0 : }
     300             : 
     301             : /************************************************************************/
     302             : /*                        ~PALSARJaxaRasterBand()                       */
     303             : /************************************************************************/
     304             : 
     305           0 : PALSARJaxaRasterBand::~PALSARJaxaRasterBand()
     306             : {
     307           0 :     if (fp)
     308           0 :         VSIFCloseL(fp);
     309           0 : }
     310             : 
     311             : /************************************************************************/
     312             : /*                             IReadBlock()                             */
     313             : /************************************************************************/
     314             : 
     315           0 : CPLErr PALSARJaxaRasterBand::IReadBlock(CPL_UNUSED int nBlockXOff,
     316             :                                         int nBlockYOff, void *pImage)
     317             : {
     318           0 :     int nNumBytes = 0;
     319           0 :     if (nFileType == level_11)
     320             :     {
     321           0 :         nNumBytes = 8;
     322             :     }
     323             :     else
     324             :     {
     325           0 :         nNumBytes = 2;
     326             :     }
     327             : 
     328           0 :     int nOffset =
     329           0 :         IMAGE_OPT_DESC_LENGTH + ((nBlockYOff - 1) * nRecordSize) +
     330           0 :         (nFileType == level_11 ? SIG_DAT_REC_OFFSET : PROC_DAT_REC_OFFSET);
     331             : 
     332           0 :     VSIFSeekL(fp, nOffset, SEEK_SET);
     333           0 :     VSIFReadL(pImage, nNumBytes, nRasterXSize, fp);
     334             : 
     335             : #ifdef CPL_LSB
     336           0 :     if (nFileType == level_11)
     337           0 :         GDALSwapWords(pImage, 4, nBlockXSize * 2, 4);
     338             :     else
     339           0 :         GDALSwapWords(pImage, 2, nBlockXSize, 2);
     340             : #endif
     341             : 
     342           0 :     return CE_None;
     343             : }
     344             : 
     345             : /************************************************************************/
     346             : /* ==================================================================== */
     347             : /*                      PALSARJaxaDataset                               */
     348             : /* ==================================================================== */
     349             : /************************************************************************/
     350             : 
     351             : /************************************************************************/
     352             : /*                          ReadMetadata()                              */
     353             : /************************************************************************/
     354             : 
     355           0 : int PALSARJaxaDataset::GetGCPCount()
     356             : {
     357           0 :     return nGCPCount;
     358             : }
     359             : 
     360             : /************************************************************************/
     361             : /*                             GetGCPs()                                */
     362             : /************************************************************************/
     363             : 
     364           0 : const GDAL_GCP *PALSARJaxaDataset::GetGCPs()
     365             : {
     366           0 :     return pasGCPList;
     367             : }
     368             : 
     369             : /************************************************************************/
     370             : /*                            ReadMetadata()                            */
     371             : /************************************************************************/
     372             : 
     373           0 : void PALSARJaxaDataset::ReadMetadata(PALSARJaxaDataset *poDS, VSILFILE *fp)
     374             : {
     375             :     /* seek to the end of the leader file descriptor */
     376           0 :     VSIFSeekL(fp, LEADER_FILE_DESCRIPTOR_LENGTH, SEEK_SET);
     377           0 :     if (poDS->nFileType == level_10)
     378             :     {
     379           0 :         poDS->SetMetadataItem("PRODUCT_LEVEL", "1.0");
     380           0 :         poDS->SetMetadataItem("AZIMUTH_LOOKS", "1.0");
     381             :     }
     382           0 :     else if (poDS->nFileType == level_11)
     383             :     {
     384           0 :         poDS->SetMetadataItem("PRODUCT_LEVEL", "1.1");
     385           0 :         poDS->SetMetadataItem("AZIMUTH_LOOKS", "1.0");
     386             :     }
     387             :     else
     388             :     {
     389           0 :         poDS->SetMetadataItem("PRODUCT_LEVEL", "1.5");
     390             :         /* extract equivalent number of looks */
     391           0 :         VSIFSeekL(
     392             :             fp, LEADER_FILE_DESCRIPTOR_LENGTH + EFFECTIVE_LOOKS_AZIMUTH_OFFSET,
     393             :             SEEK_SET);
     394             :         char szENL[17];
     395             :         double dfENL;
     396           0 :         READ_CHAR_FLOAT(dfENL, 16, fp);
     397           0 :         snprintf(szENL, sizeof(szENL), "%-16.1f", dfENL);
     398           0 :         poDS->SetMetadataItem("AZIMUTH_LOOKS", szENL);
     399             : 
     400             :         /* extract pixel spacings */
     401           0 :         VSIFSeekL(fp,
     402             :                   LEADER_FILE_DESCRIPTOR_LENGTH + DATA_SET_SUMMARY_LENGTH +
     403             :                       PIXEL_SPACING_OFFSET,
     404             :                   SEEK_SET);
     405             :         double dfPixelSpacing;
     406             :         double dfLineSpacing;
     407             :         char szPixelSpacing[33];
     408             :         char szLineSpacing[33];
     409           0 :         READ_CHAR_FLOAT(dfPixelSpacing, 16, fp);
     410           0 :         READ_CHAR_FLOAT(dfLineSpacing, 16, fp);
     411           0 :         snprintf(szPixelSpacing, sizeof(szPixelSpacing), "%-32.1f",
     412             :                  dfPixelSpacing);
     413           0 :         snprintf(szLineSpacing, sizeof(szLineSpacing), "%-32.1f",
     414             :                  dfLineSpacing);
     415           0 :         poDS->SetMetadataItem("PIXEL_SPACING", szPixelSpacing);
     416           0 :         poDS->SetMetadataItem("LINE_SPACING", szPixelSpacing);
     417             : 
     418             :         /* Alphanumeric projection name */
     419           0 :         VSIFSeekL(fp,
     420             :                   LEADER_FILE_DESCRIPTOR_LENGTH + DATA_SET_SUMMARY_LENGTH +
     421             :                       ALPHANUMERIC_PROJECTION_NAME_OFFSET,
     422             :                   SEEK_SET);
     423             :         char szProjName[33];
     424           0 :         READ_STRING(szProjName, 32, fp);
     425           0 :         poDS->SetMetadataItem("PROJECTION_NAME", szProjName);
     426             : 
     427             :         /* Extract corner GCPs */
     428           0 :         poDS->nGCPCount = 4;
     429           0 :         poDS->pasGCPList =
     430           0 :             (GDAL_GCP *)CPLCalloc(sizeof(GDAL_GCP), poDS->nGCPCount);
     431           0 :         GDALInitGCPs(poDS->nGCPCount, poDS->pasGCPList);
     432             : 
     433             :         /* setup the GCPs */
     434             :         int i;
     435           0 :         for (i = 0; i < poDS->nGCPCount; i++)
     436             :         {
     437             :             char szID[30];
     438           0 :             snprintf(szID, sizeof(szID), "%d", i + 1);
     439           0 :             CPLFree(poDS->pasGCPList[i].pszId);
     440           0 :             poDS->pasGCPList[i].pszId = CPLStrdup(szID);
     441           0 :             poDS->pasGCPList[i].dfGCPZ = 0.0;
     442             :         }
     443             : 
     444           0 :         double dfTemp = 0.0;
     445             :         /* seek to start of GCPs */
     446           0 :         VSIFSeekL(fp,
     447             :                   LEADER_FILE_DESCRIPTOR_LENGTH + DATA_SET_SUMMARY_LENGTH +
     448             :                       TOP_LEFT_LAT_OFFSET,
     449             :                   SEEK_SET);
     450             : 
     451             :         /* top-left GCP */
     452           0 :         READ_CHAR_FLOAT(dfTemp, 16, fp);
     453           0 :         poDS->pasGCPList[0].dfGCPY = dfTemp;
     454           0 :         READ_CHAR_FLOAT(dfTemp, 16, fp);
     455           0 :         poDS->pasGCPList[0].dfGCPX = dfTemp;
     456           0 :         poDS->pasGCPList[0].dfGCPLine = 0.5;
     457           0 :         poDS->pasGCPList[0].dfGCPPixel = 0.5;
     458             : 
     459             :         /* top right GCP */
     460           0 :         READ_CHAR_FLOAT(dfTemp, 16, fp);
     461           0 :         poDS->pasGCPList[1].dfGCPY = dfTemp;
     462           0 :         READ_CHAR_FLOAT(dfTemp, 16, fp);
     463           0 :         poDS->pasGCPList[1].dfGCPX = dfTemp;
     464           0 :         poDS->pasGCPList[1].dfGCPLine = 0.5;
     465           0 :         poDS->pasGCPList[1].dfGCPPixel = poDS->nRasterYSize - 0.5;
     466             : 
     467             :         /* bottom right GCP */
     468           0 :         READ_CHAR_FLOAT(dfTemp, 16, fp);
     469           0 :         poDS->pasGCPList[2].dfGCPY = dfTemp;
     470           0 :         READ_CHAR_FLOAT(dfTemp, 16, fp);
     471           0 :         poDS->pasGCPList[2].dfGCPX = dfTemp;
     472           0 :         poDS->pasGCPList[2].dfGCPLine = poDS->nRasterYSize - 0.5;
     473           0 :         poDS->pasGCPList[2].dfGCPPixel = poDS->nRasterYSize - 0.5;
     474             : 
     475             :         /* bottom left GCP */
     476           0 :         READ_CHAR_FLOAT(dfTemp, 16, fp);
     477           0 :         poDS->pasGCPList[3].dfGCPY = dfTemp;
     478           0 :         READ_CHAR_FLOAT(dfTemp, 16, fp);
     479           0 :         poDS->pasGCPList[3].dfGCPX = dfTemp;
     480           0 :         poDS->pasGCPList[3].dfGCPLine = poDS->nRasterYSize - 0.5;
     481           0 :         poDS->pasGCPList[3].dfGCPPixel = 0.5;
     482             :     }
     483             : 
     484             :     /* some generic metadata items */
     485           0 :     poDS->SetMetadataItem("SENSOR_BAND", "L"); /* PALSAR is L-band */
     486           0 :     poDS->SetMetadataItem("RANGE_LOOKS", "1.0");
     487             : 
     488             :     /* Check if this is a PolSAR dataset */
     489           0 :     if (poDS->GetRasterCount() == 4)
     490             :     {
     491             :         /* PALSAR data is only available from JAXA in Scattering Matrix form */
     492           0 :         poDS->SetMetadataItem("MATRIX_REPRESENTATION", "SCATTERING");
     493             :     }
     494           0 : }
     495             : 
     496             : /************************************************************************/
     497             : /*                              Identify()                              */
     498             : /************************************************************************/
     499             : 
     500           0 : static void ReadWord(VSILFILE *fp, int *pVal)
     501             : {
     502           0 :     READ_WORD(fp, *pVal);
     503           0 : }
     504             : 
     505           0 : static void ReadByte(VSILFILE *fp, int *pVal)
     506             : {
     507           0 :     READ_BYTE(fp, *pVal);
     508           0 : }
     509             : 
     510       57572 : int PALSARJaxaDataset::Identify(GDALOpenInfo *poOpenInfo)
     511             : {
     512       57572 :     if (poOpenInfo->nHeaderBytes < 360 || poOpenInfo->fpL == nullptr)
     513       51327 :         return 0;
     514             : 
     515             :     /* First, check that this is a PALSAR image indeed */
     516        6245 :     if (!STARTS_WITH_CI((char *)(poOpenInfo->pabyHeader + 60), "AL") ||
     517        6245 :         !STARTS_WITH_CI(
     518             :             CPLGetBasenameSafe(poOpenInfo->pszFilename).substr(4).c_str(),
     519             :             "ALPSR"))
     520             :     {
     521        6245 :         return 0;
     522             :     }
     523             : 
     524             :     /* Check that this is a volume directory file */
     525           0 :     int nRecordSeq = 0;
     526           0 :     int nRecordSubtype = 0;
     527           0 :     int nRecordType = 0;
     528           0 :     int nSecondSubtype = 0;
     529           0 :     int nThirdSubtype = 0;
     530           0 :     int nLengthRecord = 0;
     531             : 
     532           0 :     VSIFSeekL(poOpenInfo->fpL, 0, SEEK_SET);
     533             : 
     534           0 :     ReadWord(poOpenInfo->fpL, &nRecordSeq);
     535           0 :     ReadByte(poOpenInfo->fpL, &nRecordSubtype);
     536           0 :     ReadByte(poOpenInfo->fpL, &nRecordType);
     537           0 :     ReadByte(poOpenInfo->fpL, &nSecondSubtype);
     538           0 :     ReadByte(poOpenInfo->fpL, &nThirdSubtype);
     539           0 :     ReadWord(poOpenInfo->fpL, &nLengthRecord);
     540             : 
     541           0 :     VSIFSeekL(poOpenInfo->fpL, 0, SEEK_SET);
     542             : 
     543             :     /* Check that we have the right record */
     544           0 :     if (nRecordSeq == 1 && nRecordSubtype == 192 && nRecordType == 192 &&
     545           0 :         nSecondSubtype == 18 && nThirdSubtype == 18 && nLengthRecord == 360)
     546             :     {
     547           0 :         return 1;
     548             :     }
     549             : 
     550           0 :     return 0;
     551             : }
     552             : 
     553             : /************************************************************************/
     554             : /*                                Open()                                */
     555             : /************************************************************************/
     556           0 : GDALDataset *PALSARJaxaDataset::Open(GDALOpenInfo *poOpenInfo)
     557             : {
     558             :     /* Check that this actually is a JAXA PALSAR product */
     559           0 :     if (!PALSARJaxaDataset::Identify(poOpenInfo))
     560           0 :         return nullptr;
     561             : 
     562             :     /* -------------------------------------------------------------------- */
     563             :     /*      Confirm the requested access is supported.                      */
     564             :     /* -------------------------------------------------------------------- */
     565           0 :     if (poOpenInfo->eAccess == GA_Update)
     566             :     {
     567           0 :         CPLError(
     568             :             CE_Failure, CPLE_NotSupported,
     569             :             "The JAXAPALSAR driver does not support update access to existing"
     570             :             " datasets.\n");
     571           0 :         return nullptr;
     572             :     }
     573             : 
     574           0 :     PALSARJaxaDataset *poDS = new PALSARJaxaDataset();
     575             : 
     576             :     /* Get the suffix of the filename, we'll need this */
     577             :     char *pszSuffix =
     578           0 :         VSIStrdup((char *)(CPLGetFilename(poOpenInfo->pszFilename) + 3));
     579             : 
     580             :     /* Try to read each of the polarizations */
     581             :     const size_t nImgFileLen =
     582           0 :         CPLGetDirnameSafe(poOpenInfo->pszFilename).size() + strlen(pszSuffix) +
     583           0 :         8;
     584           0 :     char *pszImgFile = (char *)CPLMalloc(nImgFileLen);
     585             : 
     586           0 :     int nBandNum = 1;
     587             : 
     588             :     /* HH */
     589           0 :     snprintf(pszImgFile, nImgFileLen, "%s%sIMG-HH%s",
     590           0 :              CPLGetDirnameSafe(poOpenInfo->pszFilename).c_str(), SEP_STRING,
     591             :              pszSuffix);
     592           0 :     VSILFILE *fpHH = VSIFOpenL(pszImgFile, "rb");
     593           0 :     if (fpHH != nullptr)
     594             :     {
     595           0 :         poDS->SetBand(nBandNum, new PALSARJaxaRasterBand(poDS, 0, fpHH));
     596           0 :         nBandNum++;
     597             :     }
     598             : 
     599             :     /* HV */
     600           0 :     snprintf(pszImgFile, nImgFileLen, "%s%sIMG-HV%s",
     601           0 :              CPLGetDirnameSafe(poOpenInfo->pszFilename).c_str(), SEP_STRING,
     602             :              pszSuffix);
     603           0 :     VSILFILE *fpHV = VSIFOpenL(pszImgFile, "rb");
     604           0 :     if (fpHV != nullptr)
     605             :     {
     606           0 :         poDS->SetBand(nBandNum, new PALSARJaxaRasterBand(poDS, 1, fpHV));
     607           0 :         nBandNum++;
     608             :     }
     609             : 
     610             :     /* VH */
     611           0 :     snprintf(pszImgFile, nImgFileLen, "%s%sIMG-VH%s",
     612           0 :              CPLGetDirnameSafe(poOpenInfo->pszFilename).c_str(), SEP_STRING,
     613             :              pszSuffix);
     614           0 :     VSILFILE *fpVH = VSIFOpenL(pszImgFile, "rb");
     615           0 :     if (fpVH != nullptr)
     616             :     {
     617           0 :         poDS->SetBand(nBandNum, new PALSARJaxaRasterBand(poDS, 2, fpVH));
     618           0 :         nBandNum++;
     619             :     }
     620             : 
     621             :     /* VV */
     622           0 :     snprintf(pszImgFile, nImgFileLen, "%s%sIMG-VV%s",
     623           0 :              CPLGetDirnameSafe(poOpenInfo->pszFilename).c_str(), SEP_STRING,
     624             :              pszSuffix);
     625           0 :     VSILFILE *fpVV = VSIFOpenL(pszImgFile, "rb");
     626           0 :     if (fpVV != nullptr)
     627             :     {
     628           0 :         poDS->SetBand(nBandNum, new PALSARJaxaRasterBand(poDS, 3, fpVV));
     629             :         /* nBandNum++; */
     630             :     }
     631             : 
     632           0 :     VSIFree(pszImgFile);
     633             : 
     634             :     /* did we get at least one band? */
     635           0 :     if (fpVV == nullptr && fpVH == nullptr && fpHV == nullptr &&
     636             :         fpHH == nullptr)
     637             :     {
     638           0 :         CPLError(
     639             :             CE_Failure, CPLE_AppDefined,
     640             :             "Unable to find any image data. Aborting opening as PALSAR image.");
     641           0 :         delete poDS;
     642           0 :         VSIFree(pszSuffix);
     643           0 :         return nullptr;
     644             :     }
     645             : 
     646             :     /* Level 1.0 products are not supported */
     647           0 :     if (poDS->nFileType == level_10)
     648             :     {
     649           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     650             :                  "ALOS PALSAR Level 1.0 products are not supported. Aborting "
     651             :                  "opening as PALSAR image.");
     652           0 :         delete poDS;
     653           0 :         VSIFree(pszSuffix);
     654           0 :         return nullptr;
     655             :     }
     656             : 
     657             :     /* read metadata from Leader file. */
     658             :     const size_t nLeaderFilenameLen =
     659           0 :         strlen(CPLGetDirnameSafe(poOpenInfo->pszFilename).c_str()) +
     660           0 :         strlen(pszSuffix) + 5;
     661           0 :     char *pszLeaderFilename = (char *)CPLMalloc(nLeaderFilenameLen);
     662           0 :     snprintf(pszLeaderFilename, nLeaderFilenameLen, "%s%sLED%s",
     663           0 :              CPLGetDirnameSafe(poOpenInfo->pszFilename).c_str(), SEP_STRING,
     664             :              pszSuffix);
     665             : 
     666           0 :     VSILFILE *fpLeader = VSIFOpenL(pszLeaderFilename, "rb");
     667             :     /* check if the leader is actually present */
     668           0 :     if (fpLeader != nullptr)
     669             :     {
     670           0 :         ReadMetadata(poDS, fpLeader);
     671           0 :         VSIFCloseL(fpLeader);
     672             :     }
     673             : 
     674           0 :     VSIFree(pszLeaderFilename);
     675             : 
     676           0 :     VSIFree(pszSuffix);
     677             : 
     678             :     /* -------------------------------------------------------------------- */
     679             :     /*      Initialize any PAM information.                                 */
     680             :     /* -------------------------------------------------------------------- */
     681           0 :     poDS->SetDescription(poOpenInfo->pszFilename);
     682           0 :     poDS->TryLoadXML();
     683             : 
     684             :     /* -------------------------------------------------------------------- */
     685             :     /*      Check for overviews.                                            */
     686             :     /* -------------------------------------------------------------------- */
     687           0 :     poDS->oOvManager.Initialize(poDS, poOpenInfo->pszFilename);
     688             : 
     689           0 :     return poDS;
     690             : }
     691             : 
     692             : /************************************************************************/
     693             : /*                      GDALRegister_PALSARJaxa()                       */
     694             : /************************************************************************/
     695             : 
     696        1682 : void GDALRegister_PALSARJaxa()
     697             : 
     698             : {
     699        1682 :     if (GDALGetDriverByName("JAXAPALSAR") != nullptr)
     700         301 :         return;
     701             : 
     702        1381 :     GDALDriver *poDriver = new GDALDriver();
     703             : 
     704        1381 :     poDriver->SetDescription("JAXAPALSAR");
     705        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
     706        1381 :     poDriver->SetMetadataItem(GDAL_DMD_LONGNAME,
     707        1381 :                               "JAXA PALSAR Product Reader (Level 1.1/1.5)");
     708        1381 :     poDriver->SetMetadataItem(GDAL_DMD_HELPTOPIC, "drivers/raster/palsar.html");
     709             : 
     710        1381 :     poDriver->pfnOpen = PALSARJaxaDataset::Open;
     711        1381 :     poDriver->pfnIdentify = PALSARJaxaDataset::Identify;
     712        1381 :     poDriver->SetMetadataItem(GDAL_DCAP_VIRTUALIO, "YES");
     713             : 
     714        1381 :     GetGDALDriverManager()->RegisterDriver(poDriver);
     715             : }

Generated by: LCOV version 1.14