LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/openfilegdb - ogropenfilegdbdatasource.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 958 1062 90.2 %
Date: 2025-01-18 12:42:00 Functions: 45 45 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  OpenGIS Simple Features Reference Implementation
       4             :  * Purpose:  Implements Open FileGDB OGR driver.
       5             :  * Author:   Even Rouault, <even dot rouault at spatialys.com>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2014, Even Rouault <even dot rouault at spatialys.com>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #include "cpl_port.h"
      14             : #include "ogr_openfilegdb.h"
      15             : 
      16             : #include <stddef.h>
      17             : #include <stdio.h>
      18             : #include <string.h>
      19             : #include <map>
      20             : #include <memory>
      21             : #include <string>
      22             : #include <utility>
      23             : #include <vector>
      24             : 
      25             : #include "cpl_conv.h"
      26             : #include "cpl_error.h"
      27             : #include "cpl_string.h"
      28             : #include "cpl_vsi.h"
      29             : #include "filegdbtable.h"
      30             : #include "gdal.h"
      31             : #include "ogr_core.h"
      32             : #include "ogr_feature.h"
      33             : #include "ogr_geometry.h"
      34             : #include "ogr_mem.h"
      35             : #include "ogrsf_frmts.h"
      36             : #include "ogr_swq.h"
      37             : 
      38             : #include "filegdb_fielddomain.h"
      39             : #include "filegdb_relationship.h"
      40             : 
      41             : /***********************************************************************/
      42             : /*                      OGROpenFileGDBGroup                            */
      43             : /***********************************************************************/
      44             : 
      45             : class OGROpenFileGDBGroup final : public GDALGroup
      46             : {
      47             :   protected:
      48             :     friend class OGROpenFileGDBDataSource;
      49             :     std::vector<std::shared_ptr<GDALGroup>> m_apoSubGroups{};
      50             :     std::vector<OGRLayer *> m_apoLayers{};
      51             :     std::string m_osDefinition{};
      52             : 
      53             :   public:
      54         569 :     OGROpenFileGDBGroup(const std::string &osParentName, const char *pszName)
      55         569 :         : GDALGroup(osParentName, pszName)
      56             :     {
      57         569 :     }
      58             : 
      59          20 :     void SetDefinition(const std::string &osDefinition)
      60             :     {
      61          20 :         m_osDefinition = osDefinition;
      62          20 :     }
      63             : 
      64          23 :     const std::string &GetDefinition() const
      65             :     {
      66          23 :         return m_osDefinition;
      67             :     }
      68             : 
      69             :     std::vector<std::string>
      70             :     GetGroupNames(CSLConstList papszOptions) const override;
      71             :     std::shared_ptr<GDALGroup>
      72             :     OpenGroup(const std::string &osName,
      73             :               CSLConstList papszOptions) const override;
      74             : 
      75             :     std::vector<std::string>
      76             :     GetVectorLayerNames(CSLConstList papszOptions) const override;
      77             :     OGRLayer *OpenVectorLayer(const std::string &osName,
      78             :                               CSLConstList papszOptions) const override;
      79             : };
      80             : 
      81             : /************************************************************************/
      82             : /*                      OGROpenFileGDBDataSource()                      */
      83             : /************************************************************************/
      84         847 : OGROpenFileGDBDataSource::OGROpenFileGDBDataSource()
      85         847 :     : m_papszFiles(nullptr), bLastSQLUsedOptimizedImplementation(false)
      86             : {
      87         847 : }
      88             : 
      89             : /************************************************************************/
      90             : /*                     ~OGROpenFileGDBDataSource()                      */
      91             : /************************************************************************/
      92        1694 : OGROpenFileGDBDataSource::~OGROpenFileGDBDataSource()
      93             : 
      94             : {
      95         847 :     OGROpenFileGDBDataSource::Close();
      96        1694 : }
      97             : 
      98             : /************************************************************************/
      99             : /*                              Close()                                 */
     100             : /************************************************************************/
     101             : 
     102        1565 : CPLErr OGROpenFileGDBDataSource::Close()
     103             : {
     104        1565 :     CPLErr eErr = CE_None;
     105        1565 :     if (nOpenFlags != OPEN_FLAGS_CLOSED)
     106             :     {
     107         847 :         if (m_bInTransaction)
     108           2 :             OGROpenFileGDBDataSource::RollbackTransaction();
     109             : 
     110         847 :         if (OGROpenFileGDBDataSource::FlushCache(true) != CE_None)
     111           0 :             eErr = CE_Failure;
     112             : 
     113         847 :         m_apoLayers.clear();
     114         847 :         m_apoHiddenLayers.clear();
     115         847 :         CSLDestroy(m_papszFiles);
     116             : 
     117         847 :         if (GDALDataset::Close() != CE_None)
     118           0 :             eErr = CE_Failure;
     119             :     }
     120        1565 :     return eErr;
     121             : }
     122             : 
     123             : /************************************************************************/
     124             : /*                             FileExists()                             */
     125             : /************************************************************************/
     126             : 
     127       17526 : int OGROpenFileGDBDataSource::FileExists(const char *pszFilename)
     128             : {
     129       17526 :     if (m_papszFiles)
     130       17517 :         return CSLFindString(m_papszFiles, CPLGetFilename(pszFilename)) >= 0;
     131             : 
     132             :     VSIStatBufL sStat;
     133           9 :     CPLString osFilename(pszFilename);
     134           9 :     return VSIStatExL(osFilename, &sStat, VSI_STAT_EXISTS_FLAG) == 0;
     135             : }
     136             : 
     137             : /************************************************************************/
     138             : /*                                Open()                                */
     139             : /************************************************************************/
     140             : 
     141         615 : bool OGROpenFileGDBDataSource::Open(const GDALOpenInfo *poOpenInfo,
     142             :                                     bool &bRetryFileGDBOut)
     143             : {
     144         615 :     bRetryFileGDBOut = false;
     145             : 
     146         615 :     if (poOpenInfo->nOpenFlags == (GDAL_OF_RASTER | GDAL_OF_UPDATE))
     147             :     {
     148           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     149             :                  "Update mode of rasters is not supported");
     150           0 :         return false;
     151             :     }
     152             : 
     153         615 :     m_osDirName = poOpenInfo->pszFilename;
     154             : 
     155        1230 :     std::string osRasterLayerName;
     156         615 :     if (STARTS_WITH(poOpenInfo->pszFilename, "OpenFileGDB:"))
     157             :     {
     158             :         const CPLStringList aosTokens(CSLTokenizeString2(
     159          29 :             poOpenInfo->pszFilename, ":", CSLT_HONOURSTRINGS));
     160          29 :         if (aosTokens.size() == 4 && strlen(aosTokens[1]) == 1)
     161             :         {
     162           0 :             m_osDirName = aosTokens[1];
     163           0 :             m_osDirName += ':';
     164           0 :             m_osDirName += aosTokens[2];
     165           0 :             osRasterLayerName = aosTokens[3];
     166             :         }
     167          29 :         else if (aosTokens.size() == 3)
     168             :         {
     169          29 :             m_osDirName = aosTokens[1];
     170          29 :             osRasterLayerName = aosTokens[2];
     171             :         }
     172             :         else
     173             :         {
     174           0 :             CPLError(CE_Failure, CPLE_AppDefined, "Invalid connection string");
     175           0 :             return false;
     176             :         }
     177             :     }
     178             : 
     179        1230 :     FileGDBTable oTable;
     180             : 
     181             :     // Whether to open directly a given .gdbtable file (mostly for debugging
     182             :     // purposes)
     183         615 :     int nInterestTable = 0;
     184         615 :     unsigned int unInterestTable = 0;
     185         615 :     const char *pszFilenameWithoutPath = CPLGetFilename(m_osDirName.c_str());
     186        1343 :     if (poOpenInfo->nHeaderBytes > 0 &&
     187         113 :         strlen(pszFilenameWithoutPath) == strlen("a00000000.gdbtable") &&
     188          57 :         pszFilenameWithoutPath[0] == 'a' &&
     189         784 :         EQUAL(pszFilenameWithoutPath + strlen("a00000000"), ".gdbtable") &&
     190          56 :         sscanf(pszFilenameWithoutPath, "a%08x.gdbtable", &unInterestTable) == 1)
     191             :     {
     192          56 :         nInterestTable = static_cast<int>(unInterestTable);
     193          56 :         m_osDirName = CPLGetPathSafe(m_osDirName);
     194             :     }
     195             : 
     196         705 :     if (EQUAL(CPLGetExtensionSafe(m_osDirName).c_str(), "zip") &&
     197          90 :         !STARTS_WITH(m_osDirName, "/vsizip/"))
     198             :     {
     199          84 :         m_osDirName = "/vsizip/" + m_osDirName;
     200             :     }
     201         531 :     else if (EQUAL(CPLGetExtensionSafe(m_osDirName).c_str(), "tar") &&
     202           0 :              !STARTS_WITH(m_osDirName, "/vsitar/"))
     203             :     {
     204           0 :         m_osDirName = "/vsitar/" + m_osDirName;
     205             :     }
     206             : 
     207        1137 :     if (STARTS_WITH(m_osDirName, "/vsizip/") ||
     208         522 :         STARTS_WITH(m_osDirName, "/vsitar/"))
     209             :     {
     210             :         /* Look for one subdirectory ending with .gdb extension */
     211          94 :         char **papszDir = VSIReadDir(m_osDirName);
     212          94 :         int iCandidate = -1;
     213         621 :         for (int i = 0; papszDir && papszDir[i] != nullptr; i++)
     214             :         {
     215             :             VSIStatBufL sStat;
     216         527 :             if (EQUAL(CPLGetExtensionSafe(papszDir[i]).c_str(), "gdb") &&
     217          84 :                 VSIStatL(CPLSPrintf("%s/%s", m_osDirName.c_str(), papszDir[i]),
     218         611 :                          &sStat) == 0 &&
     219          84 :                 VSI_ISDIR(sStat.st_mode))
     220             :             {
     221          84 :                 if (iCandidate < 0)
     222          84 :                     iCandidate = i;
     223             :                 else
     224             :                 {
     225           0 :                     iCandidate = -1;
     226           0 :                     break;
     227             :                 }
     228             :             }
     229             :         }
     230          94 :         if (iCandidate >= 0)
     231             :         {
     232          84 :             m_osDirName += "/";
     233          84 :             m_osDirName += papszDir[iCandidate];
     234             :         }
     235          94 :         CSLDestroy(papszDir);
     236             :     }
     237             : 
     238             :     const std::string osTransactionBackupDirname = CPLFormFilenameSafe(
     239        1230 :         m_osDirName.c_str(), ".ogrtransaction_backup", nullptr);
     240             :     VSIStatBufL sStat;
     241         615 :     if (VSIStatL(osTransactionBackupDirname.c_str(), &sStat) == 0)
     242             :     {
     243           2 :         if (poOpenInfo->eAccess == GA_Update)
     244             :         {
     245           1 :             CPLError(
     246             :                 CE_Failure, CPLE_AppDefined,
     247             :                 "A previous backup directory %s already exists, which means "
     248             :                 "that a previous transaction was not cleanly committed or "
     249             :                 "rolled back.\n"
     250             :                 "Either manually restore the previous state from that "
     251             :                 "directory or remove it, before opening in update mode.",
     252             :                 osTransactionBackupDirname.c_str());
     253           1 :             return false;
     254             :         }
     255             :         else
     256             :         {
     257           1 :             CPLError(
     258             :                 CE_Warning, CPLE_AppDefined,
     259             :                 "A previous backup directory %s already exists, which means "
     260             :                 "that a previous transaction was not cleanly committed or "
     261             :                 "rolled back.\n"
     262             :                 "You may want to restore the previous state from that "
     263             :                 "directory or remove it.",
     264             :                 osTransactionBackupDirname.c_str());
     265             :         }
     266             :     }
     267             : 
     268         614 :     m_papszFiles = VSIReadDir(m_osDirName);
     269             : 
     270             :     /* Explore catalog table */
     271             :     m_osGDBSystemCatalogFilename =
     272         614 :         CPLFormFilenameSafe(m_osDirName, "a00000001", "gdbtable");
     273        1218 :     if (!FileExists(m_osGDBSystemCatalogFilename.c_str()) ||
     274         604 :         !oTable.Open(m_osGDBSystemCatalogFilename.c_str(),
     275         604 :                      poOpenInfo->eAccess == GA_Update))
     276             :     {
     277          36 :         if (nInterestTable > 0 && FileExists(poOpenInfo->pszFilename))
     278             :         {
     279           1 :             const char *pszLyrName = CPLSPrintf("a%08x", nInterestTable);
     280             :             auto poLayer = std::make_unique<OGROpenFileGDBLayer>(
     281           1 :                 this, poOpenInfo->pszFilename, pszLyrName, "", "",
     282           2 :                 eAccess == GA_Update);
     283             :             const std::string osTablX =
     284           2 :                 CPLResetExtensionSafe(poOpenInfo->pszFilename, "gdbtablx");
     285           1 :             if ((!FileExists(osTablX.c_str()) &&
     286           0 :                  poLayer->GetLayerDefn()->GetFieldCount() == 0 &&
     287           2 :                  poLayer->GetFeatureCount() == 0) ||
     288           1 :                 !poLayer->IsValidLayerDefn())
     289             :             {
     290           0 :                 return false;
     291             :             }
     292           1 :             m_apoLayers.push_back(std::move(poLayer));
     293           1 :             return true;
     294             :         }
     295          35 :         return false;
     296             :     }
     297             : 
     298         578 :     const int idxName = oTable.GetFieldIdx("Name");
     299         578 :     const int idxFileformat = oTable.GetFieldIdx("FileFormat");
     300        2306 :     if (!(idxName >= 0 && idxFileformat >= 0 &&
     301         576 :           oTable.GetField(idxName)->GetType() == FGFT_STRING &&
     302        1152 :           (oTable.GetField(idxFileformat)->GetType() == FGFT_INT16 ||
     303         576 :            oTable.GetField(idxFileformat)->GetType() == FGFT_INT32)))
     304             :     {
     305           2 :         return false;
     306             :     }
     307             : 
     308         576 :     int iGDBItems = -1;          /* V10 */
     309         576 :     int iGDBFeatureClasses = -1; /* V9.X */
     310         576 :     int iGDBObjectClasses = -1;  /* V9.X */
     311             : 
     312        1152 :     std::vector<std::string> aosTableNames;
     313             :     try
     314             :     {
     315       20633 :         for (int i = 0; i < oTable.GetTotalRecordCount(); i++)
     316             :         {
     317       20057 :             if (!oTable.SelectRow(i))
     318             :             {
     319        5003 :                 if (oTable.HasGotError())
     320           0 :                     break;
     321        5003 :                 aosTableNames.push_back("");
     322        5003 :                 continue;
     323             :             }
     324             : 
     325       15054 :             const OGRField *psField = oTable.GetFieldValue(idxName);
     326       15054 :             if (psField != nullptr)
     327             :             {
     328       14964 :                 aosTableNames.push_back(psField->String);
     329       14964 :                 if (strcmp(psField->String, "GDB_SpatialRefs") == 0)
     330             :                 {
     331             :                     // Normally a00000003
     332        1148 :                     m_osGDBSpatialRefsFilename = CPLFormFilenameSafe(
     333             :                         m_osDirName, CPLSPrintf("a%08x.gdbtable", i + 1),
     334         574 :                         nullptr);
     335             :                 }
     336       14390 :                 else if (strcmp(psField->String, "GDB_Items") == 0)
     337             :                 {
     338             :                     // Normally a00000004 but it has been seen in some datasets
     339             :                     // to not be a00000004
     340        1098 :                     m_osGDBItemsFilename = CPLFormFilenameSafe(
     341             :                         m_osDirName, CPLSPrintf("a%08x.gdbtable", i + 1),
     342         549 :                         nullptr);
     343         549 :                     iGDBItems = i;
     344             :                 }
     345       13841 :                 else if (strcmp(psField->String, "GDB_ItemRelationships") == 0)
     346             :                 {
     347             :                     // Normally a00000006 but can be a00000005 sometimes (e.g
     348             :                     // autotest/ogr/data/filegdb/Domains.gdb)
     349        1102 :                     m_osGDBItemRelationshipsFilename = CPLFormFilenameSafe(
     350             :                         m_osDirName, CPLSPrintf("a%08x.gdbtable", i + 1),
     351         551 :                         nullptr);
     352             :                 }
     353       13290 :                 else if (strcmp(psField->String, "GDB_FeatureClasses") == 0)
     354             :                 {
     355             :                     // FileGDB v9
     356          23 :                     iGDBFeatureClasses = i;
     357             :                 }
     358       13267 :                 else if (strcmp(psField->String, "GDB_ObjectClasses") == 0)
     359             :                 {
     360             :                     // FileGDB v9
     361          23 :                     iGDBObjectClasses = i;
     362             :                 }
     363       14964 :                 m_osMapNameToIdx[psField->String] = 1 + i;
     364             :             }
     365             :             else
     366             :             {
     367          90 :                 aosTableNames.push_back("");
     368             :             }
     369             :         }
     370             :     }
     371           0 :     catch (const std::exception &)
     372             :     {
     373           0 :         return false;
     374             :     }
     375             : 
     376         576 :     oTable.Close();
     377             : 
     378        1152 :     std::set<int> oSetIgnoredRasterLayerTableNum;
     379         576 :     if (iGDBItems >= 0)
     380             :     {
     381         549 :         eAccess = poOpenInfo->eAccess;
     382             : 
     383         549 :         const bool bRet = OpenFileGDBv10(
     384             :             iGDBItems, nInterestTable, poOpenInfo, osRasterLayerName,
     385             :             oSetIgnoredRasterLayerTableNum, bRetryFileGDBOut);
     386         549 :         if (!bRet)
     387           2 :             return false;
     388             :     }
     389          27 :     else if (iGDBFeatureClasses >= 0 && iGDBObjectClasses >= 0)
     390             :     {
     391          23 :         if (poOpenInfo->eAccess == GA_Update)
     392             :         {
     393           0 :             CPLError(CE_Failure, CPLE_NotSupported,
     394             :                      "Edition of existing FileGDB v9 datasets not supported");
     395           0 :             return false;
     396             :         }
     397             : 
     398          23 :         int bRet = OpenFileGDBv9(iGDBFeatureClasses, iGDBObjectClasses,
     399             :                                  nInterestTable, poOpenInfo, osRasterLayerName,
     400             :                                  oSetIgnoredRasterLayerTableNum);
     401          23 :         if (!bRet)
     402          23 :             return false;
     403             :     }
     404             :     else
     405             :     {
     406           4 :         CPLError(CE_Failure, CPLE_AppDefined,
     407             :                  "No GDB_Items nor GDB_FeatureClasses table");
     408           4 :         return false;
     409             :     }
     410             : 
     411         570 :     if (m_apoLayers.empty() && nInterestTable > 0)
     412             :     {
     413          54 :         if (FileExists(poOpenInfo->pszFilename))
     414             :         {
     415          54 :             const char *pszLyrName = nullptr;
     416         108 :             if (nInterestTable <= static_cast<int>(aosTableNames.size()) &&
     417          54 :                 !aosTableNames[nInterestTable - 1].empty())
     418          54 :                 pszLyrName = aosTableNames[nInterestTable - 1].c_str();
     419             :             else
     420           0 :                 pszLyrName = CPLSPrintf("a%08x", nInterestTable);
     421          54 :             m_apoLayers.push_back(std::make_unique<OGROpenFileGDBLayer>(
     422          54 :                 this, poOpenInfo->pszFilename, pszLyrName, "", "",
     423         108 :                 eAccess == GA_Update));
     424             :         }
     425             :         else
     426             :         {
     427           0 :             return false;
     428             :         }
     429             :     }
     430             : 
     431         570 :     if (nInterestTable == 0)
     432             :     {
     433         515 :         const bool bListAllTables = CPLTestBool(CSLFetchNameValueDef(
     434         515 :             poOpenInfo->papszOpenOptions, "LIST_ALL_TABLES", "NO"));
     435             : 
     436             :         // add additional tables which are not present in the
     437             :         // GDB_Items/GDB_FeatureClasses/GDB_ObjectClasses tables
     438       11154 :         for (const auto &oIter : m_osMapNameToIdx)
     439             :         {
     440             :             // test if layer is already added
     441       10639 :             if (GDALDataset::GetLayerByName(oIter.first.c_str()))
     442           0 :                 continue;
     443             : 
     444       10639 :             if (bListAllTables || !IsPrivateLayerName(oIter.first))
     445             :             {
     446        5908 :                 const int idx = oIter.second;
     447        5908 :                 const CPLString osFilename(CPLFormFilenameSafe(
     448       11816 :                     m_osDirName, CPLSPrintf("a%08x", idx), "gdbtable"));
     449       11576 :                 if (!cpl::contains(oSetIgnoredRasterLayerTableNum, idx) &&
     450        5668 :                     FileExists(osFilename))
     451             :                 {
     452             :                     m_apoLayers.emplace_back(
     453        5287 :                         std::make_unique<OGROpenFileGDBLayer>(
     454        5287 :                             this, osFilename, oIter.first.c_str(), "", "",
     455       10574 :                             eAccess == GA_Update));
     456        5287 :                     if (m_poRootGroup)
     457             :                     {
     458        5263 :                         cpl::down_cast<OGROpenFileGDBGroup *>(
     459             :                             m_poRootGroup.get())
     460             :                             ->m_apoLayers.emplace_back(
     461        5263 :                                 m_apoLayers.back().get());
     462             :                     }
     463             :                 }
     464             :             }
     465             :         }
     466             :     }
     467             : 
     468         570 :     if (!osRasterLayerName.empty())
     469             :     {
     470          29 :         m_aosSubdatasets.Clear();
     471          29 :         SetDescription(poOpenInfo->pszFilename);
     472          29 :         return nBands != 0;
     473             :     }
     474             : 
     475             :     // If opening in raster-only mode, return false if there are no raster
     476             :     // layers
     477         541 :     if ((poOpenInfo->nOpenFlags & GDAL_OF_RASTER) != 0 &&
     478          51 :         (poOpenInfo->nOpenFlags & GDAL_OF_VECTOR) == 0)
     479             :     {
     480          24 :         if (m_aosSubdatasets.empty())
     481             :         {
     482          16 :             return false;
     483             :         }
     484           8 :         else if (m_aosSubdatasets.size() == 2)
     485             :         {
     486             :             // If there is a single raster dataset, open it right away.
     487           7 :             return true;
     488             :         }
     489             :     }
     490             :     // If opening in vector-only mode, return false if there are no vector
     491             :     // layers and opened in read-only mode
     492         517 :     else if ((poOpenInfo->nOpenFlags & GDAL_OF_RASTER) == 0 &&
     493         490 :              (poOpenInfo->nOpenFlags & GDAL_OF_VECTOR) != 0)
     494             :     {
     495         487 :         if (m_apoLayers.empty() && poOpenInfo->eAccess == GA_ReadOnly)
     496             :         {
     497           2 :             return false;
     498             :         }
     499             :     }
     500             : 
     501         516 :     return true;
     502             : }
     503             : 
     504             : /***********************************************************************/
     505             : /*                             AddLayer()                              */
     506             : /***********************************************************************/
     507             : 
     508        3785 : OGRLayer *OGROpenFileGDBDataSource::AddLayer(
     509             :     const CPLString &osName, int nInterestTable, int &nCandidateLayers,
     510             :     int &nLayersCDF, const CPLString &osDefinition,
     511             :     const CPLString &osDocumentation, OGRwkbGeometryType eGeomType,
     512             :     const std::string &osParentDefinition)
     513             : {
     514        3785 :     const auto oIter = m_osMapNameToIdx.find(osName);
     515        3785 :     int idx = 0;
     516        3785 :     if (oIter != m_osMapNameToIdx.end())
     517        3771 :         idx = oIter->second;
     518        3785 :     if (idx > 0 && (nInterestTable <= 0 || nInterestTable == idx))
     519             :     {
     520        3685 :         m_osMapNameToIdx.erase(oIter);
     521             : 
     522        7370 :         const CPLString osFilename = CPLFormFilenameSafe(
     523        7370 :             m_osDirName, CPLOPrintf("a%08x", idx).c_str(), "gdbtable");
     524        3685 :         if (FileExists(osFilename))
     525             :         {
     526        3671 :             nCandidateLayers++;
     527             : 
     528        3671 :             if (m_papszFiles != nullptr)
     529             :             {
     530             :                 const CPLString osSDC =
     531        3671 :                     CPLResetExtensionSafe(osFilename, "gdbtable.sdc");
     532             :                 const CPLString osCDF =
     533        3671 :                     CPLResetExtensionSafe(osFilename, "gdbtable.cdf");
     534        3671 :                 if (FileExists(osCDF))
     535             :                 {
     536           3 :                     nLayersCDF++;
     537             : 
     538           3 :                     if (GDALGetDriverByName("FileGDB"))
     539             :                     {
     540           2 :                         CPLDebug(
     541             :                             "OpenFileGDB",
     542             :                             "%s layer has a %s file using Compressed Data "
     543             :                             "Format (CDF) that is unhandled by the OpenFileGDB "
     544             :                             "driver, but could be handled by the FileGDB "
     545             :                             "driver (which will be attempted).",
     546             :                             osName.c_str(), osCDF.c_str());
     547             :                     }
     548             :                     else
     549             :                     {
     550             :                         // Look into hidden drivers, that is drivers that have
     551             :                         // been built as plugins, but are not currently available
     552             :                         // on the system.
     553           2 :                         std::string osFileGDBDriverMsg;
     554           1 :                         auto poDM = GetGDALDriverManager();
     555             :                         const int nAllDrivercount =
     556           1 :                             poDM->GetDriverCount(/* bIncludeHidden = */ true);
     557         240 :                         for (int i = 0; i < nAllDrivercount; ++i)
     558             :                         {
     559             :                             auto poMissingPluginDriver =
     560         239 :                                 poDM->GetDriver(i, /* bIncludeHidden = */ true);
     561         239 :                             if (EQUAL(poMissingPluginDriver->GetDescription(),
     562             :                                       "FileGDB"))
     563             :                             {
     564           0 :                                 osFileGDBDriverMsg += " However the plugin ";
     565             :                                 osFileGDBDriverMsg +=
     566             :                                     poMissingPluginDriver->GetMetadataItem(
     567           0 :                                         "MISSING_PLUGIN_FILENAME");
     568             :                                 osFileGDBDriverMsg +=
     569             :                                     " is not available in your "
     570           0 :                                     "installation.";
     571           0 :                                 if (const char *pszInstallationMsg =
     572           0 :                                         poMissingPluginDriver->GetMetadataItem(
     573           0 :                                             GDAL_DMD_PLUGIN_INSTALLATION_MESSAGE))
     574             :                                 {
     575           0 :                                     osFileGDBDriverMsg += " ";
     576           0 :                                     osFileGDBDriverMsg += pszInstallationMsg;
     577             :                                 }
     578           0 :                                 break;
     579             :                             }
     580             :                         }
     581           1 :                         CPLError(CE_Warning, CPLE_AppDefined,
     582             :                                  "%s layer has a %s file using Compressed Data "
     583             :                                  "Format (CDF) that is unhandled by the "
     584             :                                  "OpenFileGDB driver, but could be handled by "
     585             :                                  "the FileGDB driver.%s",
     586             :                                  osName.c_str(), osCDF.c_str(),
     587             :                                  osFileGDBDriverMsg.c_str());
     588             :                     }
     589           3 :                     return nullptr;
     590             :                 }
     591        3668 :                 else if (FileExists(osSDC))
     592             :                 {
     593             :                     // SDC is not handled by the FileGDB driver
     594           0 :                     CPLError(CE_Warning, CPLE_AppDefined,
     595             :                              "%s layer has a %s file using Smart Data "
     596             :                              "Compression (SDC) that is unhandled.",
     597             :                              osName.c_str(), osSDC.c_str());
     598           0 :                     return nullptr;
     599             :                 }
     600             :             }
     601             : 
     602        3668 :             m_apoLayers.emplace_back(std::make_unique<OGROpenFileGDBLayer>(
     603             :                 this, osFilename, osName, osDefinition, osDocumentation,
     604        3668 :                 eAccess == GA_Update, eGeomType, osParentDefinition));
     605        3668 :             return m_apoLayers.back().get();
     606             :         }
     607             :     }
     608         114 :     return nullptr;
     609             : }
     610             : 
     611          13 : std::vector<std::string> OGROpenFileGDBGroup::GetGroupNames(CSLConstList) const
     612             : {
     613          13 :     std::vector<std::string> ret;
     614          20 :     for (const auto &poSubGroup : m_apoSubGroups)
     615           7 :         ret.emplace_back(poSubGroup->GetName());
     616          13 :     return ret;
     617             : }
     618             : 
     619             : std::shared_ptr<GDALGroup>
     620           6 : OGROpenFileGDBGroup::OpenGroup(const std::string &osName, CSLConstList) const
     621             : {
     622          10 :     for (const auto &poSubGroup : m_apoSubGroups)
     623             :     {
     624           9 :         if (poSubGroup->GetName() == osName)
     625           5 :             return poSubGroup;
     626             :     }
     627           1 :     return nullptr;
     628             : }
     629             : 
     630             : std::vector<std::string>
     631          13 : OGROpenFileGDBGroup::GetVectorLayerNames(CSLConstList) const
     632             : {
     633          13 :     std::vector<std::string> ret;
     634          53 :     for (const auto &poLayer : m_apoLayers)
     635          40 :         ret.emplace_back(poLayer->GetName());
     636          13 :     return ret;
     637             : }
     638             : 
     639          24 : OGRLayer *OGROpenFileGDBGroup::OpenVectorLayer(const std::string &osName,
     640             :                                                CSLConstList) const
     641             : {
     642         106 :     for (const auto &poLayer : m_apoLayers)
     643             :     {
     644         105 :         if (poLayer->GetName() == osName)
     645          23 :             return poLayer;
     646             :     }
     647           1 :     return nullptr;
     648             : }
     649             : 
     650             : /***********************************************************************/
     651             : /*                         OpenFileGDBv10()                            */
     652             : /***********************************************************************/
     653             : 
     654         549 : bool OGROpenFileGDBDataSource::OpenFileGDBv10(
     655             :     int iGDBItems, int nInterestTable, const GDALOpenInfo *poOpenInfo,
     656             :     const std::string &osRasterLayerName,
     657             :     std::set<int> &oSetIgnoredRasterLayerTableNum, bool &bRetryFileGDBOut)
     658             : {
     659             : 
     660         549 :     CPLDebug("OpenFileGDB", "FileGDB v10 or later");
     661             : 
     662        1098 :     FileGDBTable oTable;
     663             : 
     664         549 :     const CPLString osFilename(CPLFormFilenameSafe(
     665        1098 :         m_osDirName, CPLSPrintf("a%08x.gdbtable", iGDBItems + 1), nullptr));
     666             : 
     667             :     // Normally we don't need to update in update mode the GDB_Items table,
     668             :     // but this may help repairing it, if we have corrupted it with past
     669             :     // GDAL versions.
     670         549 :     const bool bOpenInUpdateMode =
     671         549 :         (poOpenInfo->nOpenFlags & GDAL_OF_UPDATE) != 0;
     672         549 :     if (!oTable.Open(osFilename, bOpenInUpdateMode))
     673           0 :         return false;
     674             : 
     675         549 :     const int iUUID = oTable.GetFieldIdx("UUID");
     676         549 :     const int iType = oTable.GetFieldIdx("Type");
     677         549 :     const int iName = oTable.GetFieldIdx("Name");
     678         549 :     const int iPath = oTable.GetFieldIdx("Path");
     679         549 :     const int iDefinition = oTable.GetFieldIdx("Definition");
     680         549 :     const int iDocumentation = oTable.GetFieldIdx("Documentation");
     681         549 :     if (iUUID < 0 || iType < 0 || iName < 0 || iPath < 0 || iDefinition < 0 ||
     682         549 :         iDocumentation < 0 ||
     683        1098 :         oTable.GetField(iUUID)->GetType() != FGFT_GLOBALID ||
     684        1098 :         oTable.GetField(iType)->GetType() != FGFT_GUID ||
     685        1098 :         oTable.GetField(iName)->GetType() != FGFT_STRING ||
     686        1098 :         oTable.GetField(iPath)->GetType() != FGFT_STRING ||
     687        1647 :         oTable.GetField(iDefinition)->GetType() != FGFT_XML ||
     688         549 :         oTable.GetField(iDocumentation)->GetType() != FGFT_XML)
     689             :     {
     690           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     691             :                  "Wrong structure for GDB_Items table");
     692           0 :         return false;
     693             :     }
     694             : 
     695        1098 :     auto poRootGroup = std::make_shared<OGROpenFileGDBGroup>(std::string(), "");
     696         549 :     m_poRootGroup = poRootGroup;
     697             :     std::map<std::string, std::shared_ptr<OGROpenFileGDBGroup>>
     698        1098 :         oMapPathToFeatureDataset;
     699             : 
     700             :     // First pass to collect FeatureDatasets
     701        9890 :     for (int i = 0; i < oTable.GetTotalRecordCount(); i++)
     702             :     {
     703        9341 :         if (!oTable.SelectRow(i))
     704             :         {
     705        2137 :             if (oTable.HasGotError())
     706           0 :                 break;
     707        2137 :             continue;
     708             :         }
     709             : 
     710       14408 :         CPLString osType;
     711        7204 :         const OGRField *psTypeField = oTable.GetFieldValue(iType);
     712        7204 :         if (psTypeField != nullptr)
     713             :         {
     714        7204 :             const char *pszType = psTypeField->String;
     715        7204 :             if (pszType)
     716        7204 :                 osType = pszType;
     717             :         }
     718             : 
     719       14408 :         std::string osPath;
     720        7204 :         const OGRField *psField = oTable.GetFieldValue(iPath);
     721        7204 :         if (psField != nullptr)
     722             :         {
     723        7204 :             const char *pszPath = psField->String;
     724        7204 :             if (pszPath)
     725        7204 :                 osPath = pszPath;
     726             :         }
     727             : 
     728        7204 :         if (osPath == "\\")
     729             :         {
     730         549 :             psField = oTable.GetFieldValue(iUUID);
     731         549 :             if (psField != nullptr)
     732             :             {
     733         549 :                 const char *pszUUID = psField->String;
     734         549 :                 if (pszUUID)
     735         549 :                     m_osRootGUID = pszUUID;
     736             :             }
     737             :         }
     738             : 
     739        7204 :         psField = oTable.GetFieldValue(iDefinition);
     740        7204 :         if (psField != nullptr &&
     741        6655 :             strstr(psField->String, "DEFeatureDataset") != nullptr)
     742             :         {
     743          40 :             const std::string osDefinition(psField->String);
     744             : 
     745          40 :             std::string osName;
     746          20 :             psField = oTable.GetFieldValue(iName);
     747          20 :             if (psField != nullptr)
     748             :             {
     749          20 :                 const char *pszName = psField->String;
     750          20 :                 if (pszName)
     751          20 :                     osName = pszName;
     752             :             }
     753             : 
     754          20 :             if (!osName.empty() && !osPath.empty())
     755             :             {
     756             :                 auto poSubGroup = std::make_shared<OGROpenFileGDBGroup>(
     757          40 :                     poRootGroup->GetName(), osName.c_str());
     758          20 :                 poSubGroup->SetDefinition(osDefinition);
     759          20 :                 oMapPathToFeatureDataset[osPath] = poSubGroup;
     760          20 :                 poRootGroup->m_apoSubGroups.emplace_back(poSubGroup);
     761          20 :             }
     762             :         }
     763       13819 :         else if (psField != nullptr &&
     764        6635 :                  osType.tolower().compare(pszRelationshipTypeUUID) == 0)
     765             :         {
     766             :             // relationship item
     767         720 :             auto poRelationship = ParseXMLRelationshipDef(psField->String);
     768         240 :             if (poRelationship)
     769             :             {
     770         240 :                 const auto &relationshipName = poRelationship->GetName();
     771         480 :                 m_osMapRelationships.insert(std::pair(
     772         720 :                     std::string(relationshipName), std::move(poRelationship)));
     773             :             }
     774             :         }
     775             :     }
     776             : 
     777             :     // Now collect layers
     778         549 :     int nCandidateLayers = 0;
     779         549 :     int nLayersCDF = 0;
     780         549 :     bool bRet = true;
     781        9890 :     for (int i = 0; i < oTable.GetTotalRecordCount(); i++)
     782             :     {
     783        9341 :         if (!oTable.SelectRow(i))
     784             :         {
     785        2137 :             if (oTable.HasGotError())
     786           0 :                 break;
     787        2137 :             continue;
     788             :         }
     789             : 
     790        7204 :         const OGRField *psField = oTable.GetFieldValue(iDefinition);
     791        7204 :         if (psField && (poOpenInfo->nOpenFlags & GDAL_OF_VECTOR) != 0 &&
     792        5339 :             (strstr(psField->String, "DEFeatureClassInfo") != nullptr ||
     793        2488 :              strstr(psField->String, "DETableInfo") != nullptr))
     794             :         {
     795        6812 :             CPLString osDefinition(psField->String);
     796             : 
     797        3406 :             psField = oTable.GetFieldValue(iDocumentation);
     798             :             CPLString osDocumentation(psField != nullptr ? psField->String
     799        6812 :                                                          : "");
     800             : 
     801        3406 :             std::shared_ptr<OGROpenFileGDBGroup> poParent;
     802        3406 :             psField = oTable.GetFieldValue(iPath);
     803        3406 :             if (psField != nullptr)
     804             :             {
     805        3406 :                 const char *pszPath = psField->String;
     806        3406 :                 if (pszPath)
     807             :                 {
     808        6812 :                     std::string osPath(pszPath);
     809        3406 :                     const auto nPos = osPath.rfind('\\');
     810        3406 :                     if (nPos != 0 && nPos != std::string::npos)
     811             :                     {
     812          46 :                         std::string osPathParent = osPath.substr(0, nPos);
     813             :                         const auto oIter =
     814          23 :                             oMapPathToFeatureDataset.find(osPathParent);
     815          23 :                         if (oIter == oMapPathToFeatureDataset.end())
     816             :                         {
     817           0 :                             CPLError(CE_Warning, CPLE_AppDefined,
     818             :                                      "Cannot find feature dataset of "
     819             :                                      "path %s referenced by table %s",
     820             :                                      osPathParent.c_str(), pszPath);
     821             :                         }
     822             :                         else
     823             :                         {
     824          23 :                             poParent = oIter->second;
     825             :                         }
     826             :                     }
     827             :                 }
     828             :             }
     829             : 
     830        3406 :             psField = oTable.GetFieldValue(iName);
     831        3406 :             if (psField != nullptr)
     832             :             {
     833        3406 :                 OGRLayer *poLayer = AddLayer(
     834        3406 :                     psField->String, nInterestTable, nCandidateLayers,
     835             :                     nLayersCDF, osDefinition, osDocumentation, wkbUnknown,
     836        6812 :                     poParent ? poParent->GetDefinition() : std::string());
     837             : 
     838        3406 :                 if (poLayer)
     839             :                 {
     840        3289 :                     if (poParent)
     841             :                     {
     842             :                         // Add the layer to the group of the corresponding
     843             :                         // feature dataset
     844          19 :                         poParent->m_apoLayers.emplace_back(poLayer);
     845             :                     }
     846             :                     else
     847             :                     {
     848        3270 :                         poRootGroup->m_apoLayers.emplace_back(poLayer);
     849             :                     }
     850             :                 }
     851        3406 :             }
     852             :         }
     853        3798 :         else if (psField && (poOpenInfo->nOpenFlags & GDAL_OF_VECTOR) != 0 &&
     854        1933 :                  (strstr(psField->String, "GPCodedValueDomain2") != nullptr ||
     855         857 :                   strstr(psField->String, "GPRangeDomain2") != nullptr))
     856             :         {
     857        3285 :             auto poDomain = ParseXMLFieldDomainDef(psField->String);
     858        1095 :             if (poDomain)
     859             :             {
     860        1095 :                 const auto &domainName = poDomain->GetName();
     861             :                 m_oMapFieldDomains.insert(
     862        1095 :                     std::pair(std::string(domainName), std::move(poDomain)));
     863        1095 :             }
     864             :         }
     865        2703 :         else if (psField && (poOpenInfo->nOpenFlags & GDAL_OF_RASTER) != 0 &&
     866        1444 :                  strstr(psField->String, "DERasterDataset"))
     867             :         {
     868        2048 :             const std::string osDefinition(psField->String);
     869             : 
     870        1024 :             psField = oTable.GetFieldValue(iName);
     871        1024 :             if (psField)
     872             :             {
     873        2048 :                 const std::string osLayerName(psField->String);
     874             : 
     875        1024 :                 psField = oTable.GetFieldValue(iDocumentation);
     876             :                 const std::string osDocumentation(psField ? psField->String
     877        2048 :                                                           : "");
     878             : 
     879        1024 :                 if (!osRasterLayerName.empty())
     880             :                 {
     881         930 :                     if (osRasterLayerName == osLayerName)
     882             :                     {
     883          27 :                         bRet = OpenRaster(poOpenInfo, osLayerName, osDefinition,
     884             :                                           osDocumentation);
     885             :                     }
     886             :                 }
     887             :                 else
     888             :                 {
     889          94 :                     const int iSubDSNum = 1 + m_aosSubdatasets.size() / 2;
     890             :                     m_aosSubdatasets.SetNameValue(
     891             :                         CPLSPrintf("SUBDATASET_%d_NAME", iSubDSNum),
     892             :                         CPLSPrintf("OpenFileGDB:\"%s\":%s",
     893          94 :                                    poOpenInfo->pszFilename,
     894          94 :                                    osLayerName.c_str()));
     895             : 
     896          94 :                     std::string osDesc(osLayerName);
     897          94 :                     if (!osDocumentation.empty())
     898             :                     {
     899             :                         CPLXMLTreeCloser psTree(
     900         186 :                             CPLParseXMLString(osDocumentation.c_str()));
     901          93 :                         if (psTree)
     902             :                         {
     903          93 :                             const auto psRastInfo = CPLGetXMLNode(
     904             :                                 psTree.get(), "=metadata.spdoinfo.rastinfo");
     905          93 :                             if (psRastInfo)
     906             :                             {
     907          93 :                                 const char *pszRowCount = CPLGetXMLValue(
     908             :                                     psRastInfo, "rowcount", nullptr);
     909          93 :                                 const char *pszColCount = CPLGetXMLValue(
     910             :                                     psRastInfo, "colcount", nullptr);
     911          93 :                                 const char *pszRastBand = CPLGetXMLValue(
     912             :                                     psRastInfo, "rastband", nullptr);
     913          93 :                                 const char *pszRastBPP = CPLGetXMLValue(
     914             :                                     psRastInfo, "rastbpp", nullptr);
     915          93 :                                 if (pszRowCount && pszColCount && pszRastBand &&
     916             :                                     pszRastBPP)
     917             :                                 {
     918             :                                     osDesc += CPLSPrintf(
     919             :                                         " [%sx%sx%s], %s bits", pszColCount,
     920          93 :                                         pszRowCount, pszRastBand, pszRastBPP);
     921             :                                 }
     922             :                             }
     923             :                         }
     924             :                     }
     925             : 
     926             :                     m_aosSubdatasets.SetNameValue(
     927             :                         CPLSPrintf("SUBDATASET_%d_DESC", iSubDSNum),
     928          94 :                         ("Raster " + osDesc).c_str());
     929             :                 }
     930        1024 :             }
     931             :         }
     932        1679 :         else if (psField && (poOpenInfo->nOpenFlags & GDAL_OF_RASTER) == 0 &&
     933         710 :                  strstr(psField->String, "DERasterDataset"))
     934             :         {
     935          50 :             psField = oTable.GetFieldValue(iName);
     936          50 :             if (psField)
     937             :             {
     938         100 :                 const std::string osLayerName(psField->String);
     939          50 :                 auto oIter = m_osMapNameToIdx.find(osLayerName);
     940          50 :                 if (oIter != m_osMapNameToIdx.end())
     941             :                 {
     942          47 :                     oSetIgnoredRasterLayerTableNum.insert(oIter->second);
     943             : 
     944         188 :                     for (const char *pszPrefix :
     945         235 :                          {"fras_ras_", "fras_aux_", "fras_bnd_", "fras_blk_"})
     946             :                     {
     947             :                         oIter = m_osMapNameToIdx.find(
     948         188 :                             std::string(pszPrefix).append(osLayerName).c_str());
     949         188 :                         if (oIter != m_osMapNameToIdx.end())
     950             :                         {
     951             :                             oSetIgnoredRasterLayerTableNum.insert(
     952         188 :                                 oIter->second);
     953             :                         }
     954             :                     }
     955             :                 }
     956             :             }
     957             :         }
     958             :     }
     959             : 
     960             :     // If there's at least one .cdf layer and the FileGDB driver is present,
     961             :     // retry with it.
     962         549 :     if (nLayersCDF > 0 && GDALGetDriverByName("FileGDB") != nullptr)
     963             :     {
     964           2 :         bRetryFileGDBOut = true;
     965           2 :         return false;
     966             :     }
     967             : 
     968         547 :     return bRet;
     969             : }
     970             : 
     971             : /***********************************************************************/
     972             : /*                         OpenFileGDBv9()                             */
     973             : /***********************************************************************/
     974             : 
     975          23 : int OGROpenFileGDBDataSource::OpenFileGDBv9(
     976             :     int iGDBFeatureClasses, int iGDBObjectClasses, int nInterestTable,
     977             :     const GDALOpenInfo *poOpenInfo, const std::string &osRasterLayerName,
     978             :     std::set<int> &oSetIgnoredRasterLayerTableNum)
     979             : {
     980          46 :     auto poTable = std::make_unique<FileGDBTable>();
     981             : 
     982          23 :     CPLDebug("OpenFileGDB", "FileGDB v9");
     983             : 
     984             :     /* Fetch names of layers */
     985          23 :     CPLString osFilename(CPLFormFilenameSafe(
     986          46 :         m_osDirName, CPLSPrintf("a%08x", iGDBObjectClasses + 1), "gdbtable"));
     987          23 :     if (!poTable->Open(osFilename, false))
     988           0 :         return FALSE;
     989             : 
     990          23 :     int iName = poTable->GetFieldIdx("Name");
     991          23 :     int iCLSID = poTable->GetFieldIdx("CLSID");
     992          23 :     if (iName < 0 || poTable->GetField(iName)->GetType() != FGFT_STRING ||
     993          46 :         iCLSID < 0 || poTable->GetField(iCLSID)->GetType() != FGFT_STRING)
     994             :     {
     995           0 :         CPLError(CE_Failure, CPLE_AppDefined,
     996             :                  "Wrong structure for GDB_ObjectClasses table");
     997           0 :         return FALSE;
     998             :     }
     999             : 
    1000          46 :     std::vector<std::string> aosName;
    1001          23 :     int nCandidateLayers = 0, nLayersCDF = 0;
    1002         433 :     for (int i = 0; i < poTable->GetTotalRecordCount(); i++)
    1003             :     {
    1004         410 :         if (!poTable->SelectRow(i))
    1005             :         {
    1006          27 :             if (poTable->HasGotError())
    1007           0 :                 break;
    1008          27 :             aosName.push_back("");
    1009          27 :             continue;
    1010             :         }
    1011             : 
    1012         383 :         const OGRField *psField = poTable->GetFieldValue(iName);
    1013         383 :         if (psField != nullptr)
    1014             :         {
    1015         766 :             std::string osName(psField->String);
    1016         383 :             psField = poTable->GetFieldValue(iCLSID);
    1017         383 :             if (psField != nullptr)
    1018             :             {
    1019             :                 /* Is it a non-spatial table ? */
    1020         383 :                 if (strcmp(psField->String,
    1021             :                            "{7A566981-C114-11D2-8A28-006097AFF44E}") == 0)
    1022             :                 {
    1023          51 :                     aosName.push_back("");
    1024          51 :                     AddLayer(osName, nInterestTable, nCandidateLayers,
    1025         102 :                              nLayersCDF, "", "", wkbNone, std::string());
    1026             :                 }
    1027             :                 else
    1028             :                 {
    1029             :                     /* We should perhaps also check that the CLSID is the one of
    1030             :                      * a spatial table */
    1031         332 :                     aosName.push_back(osName);
    1032             :                 }
    1033             :             }
    1034             :         }
    1035             :     }
    1036          23 :     poTable->Close();
    1037             : 
    1038          23 :     poTable = std::make_unique<FileGDBTable>();
    1039             : 
    1040             :     /* Find tables that are spatial layers */
    1041          23 :     osFilename = CPLFormFilenameSafe(
    1042          23 :         m_osDirName, CPLSPrintf("a%08x", iGDBFeatureClasses + 1), "gdbtable");
    1043          23 :     if (!poTable->Open(osFilename, false))
    1044           0 :         return FALSE;
    1045             : 
    1046          23 :     const int iObjectClassID = poTable->GetFieldIdx("ObjectClassID");
    1047          23 :     const int iFeatureType = poTable->GetFieldIdx("FeatureType");
    1048          23 :     const int iGeometryType = poTable->GetFieldIdx("GeometryType");
    1049          46 :     if (iObjectClassID < 0 || iGeometryType < 0 || iFeatureType < 0 ||
    1050          46 :         poTable->GetField(iObjectClassID)->GetType() != FGFT_INT32 ||
    1051          69 :         poTable->GetField(iFeatureType)->GetType() != FGFT_INT32 ||
    1052          23 :         poTable->GetField(iGeometryType)->GetType() != FGFT_INT32)
    1053             :     {
    1054           0 :         CPLError(CE_Failure, CPLE_AppDefined,
    1055             :                  "Wrong structure for GDB_FeatureClasses table");
    1056           0 :         return FALSE;
    1057             :     }
    1058             : 
    1059          23 :     bool bRet = true;
    1060         379 :     for (int i = 0; i < poTable->GetTotalRecordCount(); i++)
    1061             :     {
    1062         356 :         if (!poTable->SelectRow(i))
    1063             :         {
    1064          24 :             if (poTable->HasGotError())
    1065           0 :                 break;
    1066          24 :             continue;
    1067             :         }
    1068             : 
    1069         332 :         const OGRField *psField = poTable->GetFieldValue(iGeometryType);
    1070         332 :         if (psField == nullptr)
    1071           0 :             continue;
    1072         332 :         const int nGeomType = psField->Integer;
    1073         332 :         OGRwkbGeometryType eGeomType = wkbUnknown;
    1074         332 :         switch (nGeomType)
    1075             :         {
    1076           0 :             case FGTGT_NONE: /* doesn't make sense ! */
    1077           0 :                 break;
    1078          64 :             case FGTGT_POINT:
    1079          64 :                 eGeomType = wkbPoint;
    1080          64 :                 break;
    1081          52 :             case FGTGT_MULTIPOINT:
    1082          52 :                 eGeomType = wkbMultiPoint;
    1083          52 :                 break;
    1084          76 :             case FGTGT_LINE:
    1085          76 :                 eGeomType = wkbMultiLineString;
    1086          76 :                 break;
    1087         124 :             case FGTGT_POLYGON:
    1088         124 :                 eGeomType = wkbMultiPolygon;
    1089         124 :                 break;
    1090          16 :             case FGTGT_MULTIPATCH:
    1091          16 :                 eGeomType = wkbUnknown;
    1092          16 :                 break;
    1093             :         }
    1094             : 
    1095         332 :         psField = poTable->GetFieldValue(iObjectClassID);
    1096         332 :         if (psField == nullptr)
    1097           0 :             continue;
    1098             : 
    1099         332 :         const int idx = psField->Integer;
    1100         664 :         if (idx > 0 && idx <= static_cast<int>(aosName.size()) &&
    1101         332 :             !aosName[idx - 1].empty())
    1102             :         {
    1103         664 :             const std::string osName(aosName[idx - 1]);
    1104             : 
    1105         332 :             psField = poTable->GetFieldValue(iFeatureType);
    1106         332 :             const bool bIsRaster = psField && psField->Integer == 14;
    1107             : 
    1108         332 :             if (bIsRaster && (poOpenInfo->nOpenFlags & GDAL_OF_RASTER) != 0)
    1109             :             {
    1110           3 :                 if (!osRasterLayerName.empty())
    1111             :                 {
    1112           2 :                     if (osRasterLayerName == osName)
    1113             :                     {
    1114           2 :                         bRet = OpenRaster(poOpenInfo, osName, "", "");
    1115             :                     }
    1116             :                 }
    1117             :                 else
    1118             :                 {
    1119           1 :                     const int iSubDSNum = 1 + m_aosSubdatasets.size() / 2;
    1120             :                     m_aosSubdatasets.SetNameValue(
    1121             :                         CPLSPrintf("SUBDATASET_%d_NAME", iSubDSNum),
    1122             :                         CPLSPrintf("OpenFileGDB:\"%s\":%s",
    1123           1 :                                    poOpenInfo->pszFilename, osName.c_str()));
    1124             :                     m_aosSubdatasets.SetNameValue(
    1125             :                         CPLSPrintf("SUBDATASET_%d_DESC", iSubDSNum),
    1126           1 :                         ("Raster " + osName).c_str());
    1127           3 :                 }
    1128             :             }
    1129         329 :             else if (bIsRaster)
    1130             :             {
    1131           1 :                 auto oIter = m_osMapNameToIdx.find(osName);
    1132           1 :                 if (oIter != m_osMapNameToIdx.end())
    1133             :                 {
    1134           1 :                     oSetIgnoredRasterLayerTableNum.insert(oIter->second);
    1135             : 
    1136           4 :                     for (const char *pszPrefix :
    1137           5 :                          {"fras_ras_", "fras_aux_", "fras_bnd_", "fras_blk_"})
    1138             :                     {
    1139             :                         oIter = m_osMapNameToIdx.find(
    1140           4 :                             std::string(pszPrefix).append(osName).c_str());
    1141           4 :                         if (oIter != m_osMapNameToIdx.end())
    1142             :                         {
    1143             :                             oSetIgnoredRasterLayerTableNum.insert(
    1144           4 :                                 oIter->second);
    1145             :                         }
    1146             :                     }
    1147             :                 }
    1148             :             }
    1149             :             else
    1150             :             {
    1151         328 :                 AddLayer(osName, nInterestTable, nCandidateLayers, nLayersCDF,
    1152         656 :                          "", "", eGeomType, std::string());
    1153             :             }
    1154             :         }
    1155             :     }
    1156             : 
    1157          23 :     if (m_apoLayers.empty() && nCandidateLayers > 0 &&
    1158           0 :         nCandidateLayers == nLayersCDF)
    1159           0 :         return FALSE;
    1160             : 
    1161          23 :     return bRet;
    1162             : }
    1163             : 
    1164             : /***********************************************************************/
    1165             : /*                         TestCapability()                            */
    1166             : /***********************************************************************/
    1167             : 
    1168         693 : int OGROpenFileGDBDataSource::TestCapability(const char *pszCap)
    1169             : {
    1170         693 :     if (EQUAL(pszCap, ODsCCreateLayer) || EQUAL(pszCap, ODsCDeleteLayer) ||
    1171         610 :         EQUAL(pszCap, ODsCAddFieldDomain) ||
    1172         606 :         EQUAL(pszCap, ODsCDeleteFieldDomain) ||
    1173         605 :         EQUAL(pszCap, ODsCUpdateFieldDomain) ||
    1174         604 :         EQUAL(pszCap, GDsCAddRelationship) ||
    1175         604 :         EQUAL(pszCap, GDsCDeleteRelationship) ||
    1176         604 :         EQUAL(pszCap, GDsCUpdateRelationship) ||
    1177         604 :         EQUAL(pszCap, ODsCEmulatedTransactions))
    1178             :     {
    1179          92 :         return eAccess == GA_Update;
    1180             :     }
    1181             : 
    1182         601 :     else if (EQUAL(pszCap, ODsCMeasuredGeometries))
    1183         181 :         return TRUE;
    1184         420 :     else if (EQUAL(pszCap, ODsCZGeometries))
    1185         181 :         return TRUE;
    1186         239 :     else if (EQUAL(pszCap, ODsCCurveGeometries))
    1187         198 :         return TRUE;
    1188             : 
    1189          41 :     return FALSE;
    1190             : }
    1191             : 
    1192             : /***********************************************************************/
    1193             : /*                            GetLayer()                               */
    1194             : /***********************************************************************/
    1195             : 
    1196     1195130 : OGRLayer *OGROpenFileGDBDataSource::GetLayer(int iIndex)
    1197             : {
    1198     1195130 :     if (iIndex < 0 || iIndex >= static_cast<int>(m_apoLayers.size()))
    1199          16 :         return nullptr;
    1200     1195110 :     return m_apoLayers[iIndex].get();
    1201             : }
    1202             : 
    1203             : /***********************************************************************/
    1204             : /*                      BuildLayerFromName()                           */
    1205             : /***********************************************************************/
    1206             : 
    1207             : std::unique_ptr<OGROpenFileGDBLayer>
    1208         803 : OGROpenFileGDBDataSource::BuildLayerFromName(const char *pszName)
    1209             : {
    1210             : 
    1211         803 :     const auto oIter = m_osMapNameToIdx.find(pszName);
    1212         803 :     if (oIter != m_osMapNameToIdx.end())
    1213             :     {
    1214         164 :         const int idx = oIter->second;
    1215         164 :         const CPLString osFilename(CPLFormFilenameSafe(
    1216         164 :             m_osDirName, CPLSPrintf("a%08x", idx), "gdbtable"));
    1217         164 :         if (FileExists(osFilename))
    1218             :         {
    1219             :             return std::make_unique<OGROpenFileGDBLayer>(
    1220         163 :                 this, osFilename, pszName, "", "", eAccess == GA_Update);
    1221             :         }
    1222             :     }
    1223         640 :     return nullptr;
    1224             : }
    1225             : 
    1226             : /***********************************************************************/
    1227             : /*                          GetLayerByName()                           */
    1228             : /***********************************************************************/
    1229             : 
    1230             : OGROpenFileGDBLayer *
    1231        3329 : OGROpenFileGDBDataSource::GetLayerByName(const char *pszName)
    1232             : {
    1233       34397 :     for (auto &poLayer : m_apoLayers)
    1234             :     {
    1235       33654 :         if (EQUAL(poLayer->GetName(), pszName))
    1236        2586 :             return poLayer.get();
    1237             :     }
    1238             : 
    1239        4797 :     for (auto &poLayer : m_apoHiddenLayers)
    1240             :     {
    1241        4055 :         if (EQUAL(poLayer->GetName(), pszName))
    1242           1 :             return poLayer.get();
    1243             :     }
    1244             : 
    1245        1484 :     auto poLayer = BuildLayerFromName(pszName);
    1246         742 :     if (poLayer)
    1247             :     {
    1248         103 :         m_apoHiddenLayers.emplace_back(std::move(poLayer));
    1249         103 :         return m_apoHiddenLayers.back().get();
    1250             :     }
    1251         639 :     return nullptr;
    1252             : }
    1253             : 
    1254             : /***********************************************************************/
    1255             : /*                          IsPrivateLayerName()                       */
    1256             : /***********************************************************************/
    1257             : 
    1258       10687 : bool OGROpenFileGDBDataSource::IsPrivateLayerName(const CPLString &osName)
    1259             : {
    1260       10687 :     const CPLString osLCTableName(CPLString(osName).tolower());
    1261             : 
    1262             :     // tables beginning with "GDB_" are private tables
    1263             :     // Also consider "VAT_" tables as private ones.
    1264       27311 :     return osLCTableName.size() >= 4 && (osLCTableName.substr(0, 4) == "gdb_" ||
    1265       27311 :                                          osLCTableName.substr(0, 4) == "vat_");
    1266             : }
    1267             : 
    1268             : /***********************************************************************/
    1269             : /*                          IsLayerPrivate()                           */
    1270             : /***********************************************************************/
    1271             : 
    1272         103 : bool OGROpenFileGDBDataSource::IsLayerPrivate(int iLayer) const
    1273             : {
    1274         103 :     if (iLayer < 0 || iLayer >= static_cast<int>(m_apoLayers.size()))
    1275           0 :         return false;
    1276             : 
    1277         103 :     const std::string osName(m_apoLayers[iLayer]->GetName());
    1278         103 :     return IsPrivateLayerName(osName);
    1279             : }
    1280             : 
    1281             : /***********************************************************************/
    1282             : /*                           GetMetadata()                             */
    1283             : /***********************************************************************/
    1284             : 
    1285          27 : char **OGROpenFileGDBDataSource::GetMetadata(const char *pszDomain)
    1286             : {
    1287          27 :     if (pszDomain && EQUAL(pszDomain, "SUBDATASETS"))
    1288           7 :         return m_aosSubdatasets.List();
    1289          20 :     return GDALDataset::GetMetadata(pszDomain);
    1290             : }
    1291             : 
    1292             : /************************************************************************/
    1293             : /*                 OGROpenFileGDBSingleFeatureLayer()                   */
    1294             : /************************************************************************/
    1295             : 
    1296         239 : OGROpenFileGDBSingleFeatureLayer::OGROpenFileGDBSingleFeatureLayer(
    1297         239 :     const char *pszLayerName, const char *pszValIn)
    1298         239 :     : pszVal(pszValIn ? CPLStrdup(pszValIn) : nullptr),
    1299         478 :       poFeatureDefn(new OGRFeatureDefn(pszLayerName)), iNextShapeId(0)
    1300             : {
    1301         239 :     SetDescription(poFeatureDefn->GetName());
    1302         239 :     poFeatureDefn->Reference();
    1303         478 :     OGRFieldDefn oField("FIELD_1", OFTString);
    1304         239 :     poFeatureDefn->AddFieldDefn(&oField);
    1305         239 : }
    1306             : 
    1307             : /************************************************************************/
    1308             : /*                 ~OGROpenFileGDBSingleFeatureLayer()                  */
    1309             : /************************************************************************/
    1310             : 
    1311         478 : OGROpenFileGDBSingleFeatureLayer::~OGROpenFileGDBSingleFeatureLayer()
    1312             : {
    1313         239 :     if (poFeatureDefn != nullptr)
    1314         239 :         poFeatureDefn->Release();
    1315         239 :     CPLFree(pszVal);
    1316         478 : }
    1317             : 
    1318             : /************************************************************************/
    1319             : /*                           GetNextFeature()                           */
    1320             : /************************************************************************/
    1321             : 
    1322         258 : OGRFeature *OGROpenFileGDBSingleFeatureLayer::GetNextFeature()
    1323             : {
    1324         258 :     if (iNextShapeId != 0)
    1325          19 :         return nullptr;
    1326             : 
    1327         239 :     OGRFeature *poFeature = new OGRFeature(poFeatureDefn);
    1328         239 :     if (pszVal)
    1329         239 :         poFeature->SetField(0, pszVal);
    1330         239 :     poFeature->SetFID(iNextShapeId++);
    1331         239 :     return poFeature;
    1332             : }
    1333             : 
    1334             : /***********************************************************************/
    1335             : /*                     OGROpenFileGDBSimpleSQLLayer                    */
    1336             : /***********************************************************************/
    1337             : 
    1338             : class OGROpenFileGDBSimpleSQLLayer final : public OGRLayer
    1339             : {
    1340             :     OGRLayer *poBaseLayer;
    1341             :     FileGDBIterator *poIter;
    1342             :     OGRFeatureDefn *poFeatureDefn;
    1343             :     GIntBig m_nOffset;
    1344             :     GIntBig m_nLimit;
    1345             :     GIntBig m_nSkipped = 0;
    1346             :     GIntBig m_nIterated = 0;
    1347             : 
    1348             :     CPL_DISALLOW_COPY_ASSIGN(OGROpenFileGDBSimpleSQLLayer)
    1349             : 
    1350             :   public:
    1351             :     OGROpenFileGDBSimpleSQLLayer(OGRLayer *poBaseLayer, FileGDBIterator *poIter,
    1352             :                                  int nColumns, const swq_col_def *pasColDefs,
    1353             :                                  GIntBig nOffset, GIntBig nLimit);
    1354             :     virtual ~OGROpenFileGDBSimpleSQLLayer();
    1355             : 
    1356             :     virtual void ResetReading() override;
    1357             :     virtual OGRFeature *GetNextFeature() override;
    1358             :     virtual OGRFeature *GetFeature(GIntBig nFeatureId) override;
    1359             : 
    1360         158 :     virtual OGRFeatureDefn *GetLayerDefn() override
    1361             :     {
    1362         158 :         return poFeatureDefn;
    1363             :     }
    1364             : 
    1365             :     virtual int TestCapability(const char *) override;
    1366             : 
    1367          60 :     virtual const char *GetFIDColumn() override
    1368             :     {
    1369          60 :         return poBaseLayer->GetFIDColumn();
    1370             :     }
    1371             : 
    1372          12 :     virtual OGRErr GetExtent(OGREnvelope *psExtent, int bForce) override
    1373             :     {
    1374          12 :         return poBaseLayer->GetExtent(psExtent, bForce);
    1375             :     }
    1376             : 
    1377          12 :     virtual OGRErr GetExtent(int iGeomField, OGREnvelope *psExtent,
    1378             :                              int bForce) override
    1379             :     {
    1380          12 :         return OGRLayer::GetExtent(iGeomField, psExtent, bForce);
    1381             :     }
    1382             : 
    1383             :     virtual GIntBig GetFeatureCount(int bForce) override;
    1384             : };
    1385             : 
    1386             : /***********************************************************************/
    1387             : /*                    OGROpenFileGDBSimpleSQLLayer()                   */
    1388             : /***********************************************************************/
    1389             : 
    1390          16 : OGROpenFileGDBSimpleSQLLayer::OGROpenFileGDBSimpleSQLLayer(
    1391             :     OGRLayer *poBaseLayerIn, FileGDBIterator *poIterIn, int nColumns,
    1392          16 :     const swq_col_def *pasColDefs, GIntBig nOffset, GIntBig nLimit)
    1393             :     : poBaseLayer(poBaseLayerIn), poIter(poIterIn), poFeatureDefn(nullptr),
    1394          16 :       m_nOffset(nOffset), m_nLimit(nLimit)
    1395             : {
    1396          16 :     if (nColumns == 1 && strcmp(pasColDefs[0].field_name, "*") == 0)
    1397             :     {
    1398          14 :         poFeatureDefn = poBaseLayer->GetLayerDefn();
    1399          14 :         poFeatureDefn->Reference();
    1400             :     }
    1401             :     else
    1402             :     {
    1403           2 :         poFeatureDefn = new OGRFeatureDefn(poBaseLayer->GetName());
    1404           2 :         poFeatureDefn->SetGeomType(poBaseLayer->GetGeomType());
    1405           2 :         poFeatureDefn->Reference();
    1406           2 :         if (poBaseLayer->GetGeomType() != wkbNone)
    1407             :         {
    1408           4 :             poFeatureDefn->GetGeomFieldDefn(0)->SetName(
    1409           2 :                 poBaseLayer->GetGeometryColumn());
    1410           2 :             poFeatureDefn->GetGeomFieldDefn(0)->SetSpatialRef(
    1411           2 :                 poBaseLayer->GetSpatialRef());
    1412             :         }
    1413           6 :         for (int i = 0; i < nColumns; i++)
    1414             :         {
    1415           4 :             if (strcmp(pasColDefs[i].field_name, "*") == 0)
    1416             :             {
    1417           0 :                 for (int j = 0;
    1418           0 :                      j < poBaseLayer->GetLayerDefn()->GetFieldCount(); j++)
    1419           0 :                     poFeatureDefn->AddFieldDefn(
    1420           0 :                         poBaseLayer->GetLayerDefn()->GetFieldDefn(j));
    1421             :             }
    1422             :             else
    1423             :             {
    1424             :                 OGRFieldDefn *poFieldDefn =
    1425           8 :                     poBaseLayer->GetLayerDefn()->GetFieldDefn(
    1426           8 :                         poBaseLayer->GetLayerDefn()->GetFieldIndex(
    1427           4 :                             pasColDefs[i].field_name));
    1428           4 :                 CPLAssert(poFieldDefn != nullptr); /* already checked before */
    1429           4 :                 poFeatureDefn->AddFieldDefn(poFieldDefn);
    1430             :             }
    1431             :         }
    1432             :     }
    1433          16 :     SetDescription(poFeatureDefn->GetName());
    1434          16 :     OGROpenFileGDBSimpleSQLLayer::ResetReading();
    1435          16 : }
    1436             : 
    1437             : /***********************************************************************/
    1438             : /*                   ~OGROpenFileGDBSimpleSQLLayer()                   */
    1439             : /***********************************************************************/
    1440             : 
    1441          32 : OGROpenFileGDBSimpleSQLLayer::~OGROpenFileGDBSimpleSQLLayer()
    1442             : {
    1443          16 :     if (poFeatureDefn)
    1444             :     {
    1445          16 :         poFeatureDefn->Release();
    1446             :     }
    1447          16 :     delete poIter;
    1448          32 : }
    1449             : 
    1450             : /***********************************************************************/
    1451             : /*                          ResetReading()                             */
    1452             : /***********************************************************************/
    1453             : 
    1454         204 : void OGROpenFileGDBSimpleSQLLayer::ResetReading()
    1455             : {
    1456         204 :     poIter->Reset();
    1457         204 :     m_nSkipped = 0;
    1458         204 :     m_nIterated = 0;
    1459         204 : }
    1460             : 
    1461             : /***********************************************************************/
    1462             : /*                          GetFeature()                               */
    1463             : /***********************************************************************/
    1464             : 
    1465         682 : OGRFeature *OGROpenFileGDBSimpleSQLLayer::GetFeature(GIntBig nFeatureId)
    1466             : {
    1467         682 :     OGRFeature *poSrcFeature = poBaseLayer->GetFeature(nFeatureId);
    1468         682 :     if (poSrcFeature == nullptr)
    1469           9 :         return nullptr;
    1470             : 
    1471         673 :     if (poFeatureDefn == poBaseLayer->GetLayerDefn())
    1472         528 :         return poSrcFeature;
    1473             :     else
    1474             :     {
    1475         145 :         OGRFeature *poFeature = new OGRFeature(poFeatureDefn);
    1476         145 :         poFeature->SetFrom(poSrcFeature);
    1477         145 :         poFeature->SetFID(poSrcFeature->GetFID());
    1478         145 :         delete poSrcFeature;
    1479         145 :         return poFeature;
    1480             :     }
    1481             : }
    1482             : 
    1483             : /***********************************************************************/
    1484             : /*                         GetNextFeature()                            */
    1485             : /***********************************************************************/
    1486             : 
    1487         735 : OGRFeature *OGROpenFileGDBSimpleSQLLayer::GetNextFeature()
    1488             : {
    1489             :     while (true)
    1490             :     {
    1491         735 :         if (m_nLimit >= 0 && m_nIterated == m_nLimit)
    1492           1 :             return nullptr;
    1493             : 
    1494         734 :         const int64_t nRow = poIter->GetNextRowSortedByValue();
    1495         734 :         if (nRow < 0)
    1496          73 :             return nullptr;
    1497         661 :         OGRFeature *poFeature = GetFeature(nRow + 1);
    1498         661 :         if (poFeature == nullptr)
    1499           0 :             return nullptr;
    1500         661 :         if (m_nOffset >= 0 && m_nSkipped < m_nOffset)
    1501             :         {
    1502         343 :             delete poFeature;
    1503         343 :             m_nSkipped++;
    1504         343 :             continue;
    1505             :         }
    1506         318 :         m_nIterated++;
    1507             : 
    1508         746 :         if ((m_poFilterGeom == nullptr ||
    1509         603 :              FilterGeometry(poFeature->GetGeometryRef())) &&
    1510         285 :             (m_poAttrQuery == nullptr || m_poAttrQuery->Evaluate(poFeature)))
    1511             :         {
    1512         235 :             return poFeature;
    1513             :         }
    1514             : 
    1515          83 :         delete poFeature;
    1516         426 :     }
    1517             : }
    1518             : 
    1519             : /***********************************************************************/
    1520             : /*                         GetFeatureCount()                           */
    1521             : /***********************************************************************/
    1522             : 
    1523          51 : GIntBig OGROpenFileGDBSimpleSQLLayer::GetFeatureCount(int bForce)
    1524             : {
    1525             : 
    1526             :     /* No filter */
    1527          51 :     if (m_poFilterGeom == nullptr && m_poAttrQuery == nullptr)
    1528             :     {
    1529          33 :         GIntBig nRowCount = poIter->GetRowCount();
    1530          33 :         if (m_nOffset > 0)
    1531             :         {
    1532           3 :             if (m_nOffset <= nRowCount)
    1533           2 :                 nRowCount -= m_nOffset;
    1534             :             else
    1535           1 :                 nRowCount = 0;
    1536             :         }
    1537          33 :         if (m_nLimit >= 0 && nRowCount > m_nLimit)
    1538           5 :             nRowCount = m_nLimit;
    1539          33 :         return nRowCount;
    1540             :     }
    1541             : 
    1542          18 :     return OGRLayer::GetFeatureCount(bForce);
    1543             : }
    1544             : 
    1545             : /***********************************************************************/
    1546             : /*                         TestCapability()                            */
    1547             : /***********************************************************************/
    1548             : 
    1549          96 : int OGROpenFileGDBSimpleSQLLayer::TestCapability(const char *pszCap)
    1550             : {
    1551             : 
    1552          96 :     if (EQUAL(pszCap, OLCFastFeatureCount))
    1553             :     {
    1554           0 :         return m_poFilterGeom == nullptr && m_poAttrQuery == nullptr;
    1555             :     }
    1556          96 :     else if (EQUAL(pszCap, OLCFastGetExtent))
    1557             :     {
    1558           6 :         return TRUE;
    1559             :     }
    1560          90 :     else if (EQUAL(pszCap, OLCRandomRead))
    1561             :     {
    1562           0 :         return TRUE;
    1563             :     }
    1564          90 :     else if (EQUAL(pszCap, OLCStringsAsUTF8))
    1565             :     {
    1566          27 :         return TRUE; /* ? */
    1567             :     }
    1568             : 
    1569          63 :     return FALSE;
    1570             : }
    1571             : 
    1572             : /***********************************************************************/
    1573             : /*                            ExecuteSQL()                             */
    1574             : /***********************************************************************/
    1575             : 
    1576         793 : OGRLayer *OGROpenFileGDBDataSource::ExecuteSQL(const char *pszSQLCommand,
    1577             :                                                OGRGeometry *poSpatialFilter,
    1578             :                                                const char *pszDialect)
    1579             : {
    1580             : 
    1581             :     /* -------------------------------------------------------------------- */
    1582             :     /*      Special case GetLayerDefinition                                 */
    1583             :     /* -------------------------------------------------------------------- */
    1584         793 :     if (STARTS_WITH_CI(pszSQLCommand, "GetLayerDefinition "))
    1585             :     {
    1586          32 :         const char *pszLayerName =
    1587             :             pszSQLCommand + strlen("GetLayerDefinition ");
    1588          32 :         auto poLayer = GetLayerByName(pszLayerName);
    1589          32 :         if (poLayer)
    1590             :         {
    1591             :             OGRLayer *poRet = new OGROpenFileGDBSingleFeatureLayer(
    1592          31 :                 "LayerDefinition", poLayer->GetXMLDefinition().c_str());
    1593          31 :             return poRet;
    1594             :         }
    1595           1 :         CPLError(CE_Failure, CPLE_AppDefined, "Invalid layer name: %s",
    1596             :                  pszLayerName);
    1597           1 :         return nullptr;
    1598             :     }
    1599             : 
    1600             :     /* -------------------------------------------------------------------- */
    1601             :     /*      Special case GetLayerMetadata                                   */
    1602             :     /* -------------------------------------------------------------------- */
    1603         761 :     if (STARTS_WITH_CI(pszSQLCommand, "GetLayerMetadata "))
    1604             :     {
    1605          21 :         const char *pszLayerName = pszSQLCommand + strlen("GetLayerMetadata ");
    1606          21 :         auto poLayer = GetLayerByName(pszLayerName);
    1607          21 :         if (poLayer)
    1608             :         {
    1609             :             OGRLayer *poRet = new OGROpenFileGDBSingleFeatureLayer(
    1610          20 :                 "LayerMetadata", poLayer->GetXMLDocumentation().c_str());
    1611          20 :             return poRet;
    1612             :         }
    1613           1 :         CPLError(CE_Failure, CPLE_AppDefined, "Invalid layer name: %s",
    1614             :                  pszLayerName);
    1615           1 :         return nullptr;
    1616             :     }
    1617             : 
    1618             :     /* -------------------------------------------------------------------- */
    1619             :     /*      Special case GetLayerAttrIndexUse (only for debugging purposes) */
    1620             :     /* -------------------------------------------------------------------- */
    1621         740 :     if (STARTS_WITH_CI(pszSQLCommand, "GetLayerAttrIndexUse "))
    1622             :     {
    1623             :         auto poLayer =
    1624         141 :             GetLayerByName(pszSQLCommand + strlen("GetLayerAttrIndexUse "));
    1625         141 :         if (poLayer)
    1626             :         {
    1627             :             OGRLayer *poRet = new OGROpenFileGDBSingleFeatureLayer(
    1628             :                 "LayerAttrIndexUse",
    1629         141 :                 CPLSPrintf("%d", poLayer->GetAttrIndexUse()));
    1630         141 :             return poRet;
    1631             :         }
    1632             : 
    1633           0 :         return nullptr;
    1634             :     }
    1635             : 
    1636             :     /* -------------------------------------------------------------------- */
    1637             :     /*      Special case GetLayerSpatialIndexState (only for debugging purposes)
    1638             :      */
    1639             :     /* -------------------------------------------------------------------- */
    1640         599 :     if (STARTS_WITH_CI(pszSQLCommand, "GetLayerSpatialIndexState "))
    1641             :     {
    1642          12 :         auto poLayer = GetLayerByName(pszSQLCommand +
    1643             :                                       strlen("GetLayerSpatialIndexState "));
    1644          12 :         if (poLayer)
    1645             :         {
    1646             :             OGRLayer *poRet = new OGROpenFileGDBSingleFeatureLayer(
    1647             :                 "LayerSpatialIndexState",
    1648          12 :                 CPLSPrintf("%d", poLayer->GetSpatialIndexState()));
    1649          12 :             return poRet;
    1650             :         }
    1651             : 
    1652           0 :         return nullptr;
    1653             :     }
    1654             : 
    1655             :     /* -------------------------------------------------------------------- */
    1656             :     /*      Special case GetLastSQLUsedOptimizedImplementation (only for
    1657             :      * debugging purposes) */
    1658             :     /* -------------------------------------------------------------------- */
    1659         587 :     if (EQUAL(pszSQLCommand, "GetLastSQLUsedOptimizedImplementation"))
    1660             :     {
    1661             :         OGRLayer *poRet = new OGROpenFileGDBSingleFeatureLayer(
    1662             :             "GetLastSQLUsedOptimizedImplementation",
    1663          46 :             CPLSPrintf("%d",
    1664          23 :                        static_cast<int>(bLastSQLUsedOptimizedImplementation)));
    1665          23 :         return poRet;
    1666             :     }
    1667             : 
    1668             :     /* -------------------------------------------------------------------- */
    1669             :     /*      Special case for "CREATE SPATIAL INDEX ON "                     */
    1670             :     /* -------------------------------------------------------------------- */
    1671         564 :     if (STARTS_WITH_CI(pszSQLCommand, "CREATE SPATIAL INDEX ON "))
    1672             :     {
    1673           0 :         const char *pszLayerName =
    1674             :             pszSQLCommand + strlen("CREATE SPATIAL INDEX ON ");
    1675           0 :         auto poLayer = GetLayerByName(pszLayerName);
    1676           0 :         if (poLayer)
    1677             :         {
    1678           0 :             poLayer->CreateSpatialIndex();
    1679             :         }
    1680           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Invalid layer name: %s",
    1681             :                  pszLayerName);
    1682           0 :         return nullptr;
    1683             :     }
    1684             : 
    1685             :     /* -------------------------------------------------------------------- */
    1686             :     /*      Special case for "CREATE INDEX idx_name ON table_name(col_name) */
    1687             :     /* -------------------------------------------------------------------- */
    1688         564 :     if (STARTS_WITH_CI(pszSQLCommand, "CREATE INDEX "))
    1689             :     {
    1690          42 :         CPLString osSQL(pszSQLCommand);
    1691          21 :         const auto nPosON = osSQL.ifind(" ON ");
    1692          21 :         if (nPosON != std::string::npos)
    1693             :         {
    1694             :             const std::string osIdxName = osSQL.substr(
    1695          21 :                 strlen("CREATE INDEX "), nPosON - strlen("CREATE INDEX "));
    1696          21 :             const std::string afterOn = osSQL.substr(nPosON + strlen(" ON "));
    1697          21 :             const auto nOpenParPos = afterOn.find('(');
    1698          21 :             if (nOpenParPos != std::string::npos && afterOn.back() == ')')
    1699             :             {
    1700          42 :                 const std::string osLayerName = afterOn.substr(0, nOpenParPos);
    1701             :                 const std::string osExpression = afterOn.substr(
    1702          42 :                     nOpenParPos + 1, afterOn.size() - nOpenParPos - 2);
    1703          21 :                 if (osExpression.find(',') != std::string::npos)
    1704             :                 {
    1705           1 :                     CPLError(
    1706             :                         CE_Failure, CPLE_NotSupported,
    1707             :                         "Creation of multiple-column indices is not supported");
    1708           1 :                     return nullptr;
    1709             :                 }
    1710          20 :                 auto poLayer = GetLayerByName(osLayerName.c_str());
    1711          20 :                 if (poLayer)
    1712             :                 {
    1713          19 :                     poLayer->CreateIndex(osIdxName, osExpression);
    1714          19 :                     return nullptr;
    1715             :                 }
    1716           1 :                 CPLError(CE_Failure, CPLE_AppDefined, "Invalid layer name: %s",
    1717             :                          osLayerName.c_str());
    1718           1 :                 return nullptr;
    1719             :             }
    1720             :         }
    1721           0 :         CPLError(CE_Failure, CPLE_AppDefined,
    1722             :                  "Bad syntax. Expected CREATE INDEX idx_name ON "
    1723             :                  "table_name(col_name)");
    1724           0 :         return nullptr;
    1725             :     }
    1726             : 
    1727             :     /* -------------------------------------------------------------------- */
    1728             :     /*      Special case for "RECOMPUTE EXTENT ON "                         */
    1729             :     /* -------------------------------------------------------------------- */
    1730         543 :     if (STARTS_WITH_CI(pszSQLCommand, "RECOMPUTE EXTENT ON "))
    1731             :     {
    1732           3 :         const char *pszLayerName =
    1733             :             pszSQLCommand + strlen("RECOMPUTE EXTENT ON ");
    1734           3 :         auto poLayer = GetLayerByName(pszLayerName);
    1735           3 :         if (poLayer)
    1736             :         {
    1737           2 :             poLayer->RecomputeExtent();
    1738             :         }
    1739             :         else
    1740             :         {
    1741           1 :             CPLError(CE_Failure, CPLE_AppDefined, "Invalid layer name: %s",
    1742             :                      pszLayerName);
    1743             :         }
    1744           3 :         return nullptr;
    1745             :     }
    1746             : 
    1747             :     /* -------------------------------------------------------------------- */
    1748             :     /*      Special case for "DELLAYER:"                                    */
    1749             :     /* -------------------------------------------------------------------- */
    1750         540 :     if (STARTS_WITH_CI(pszSQLCommand, "DELLAYER:"))
    1751             :     {
    1752           3 :         const char *pszLayerName = pszSQLCommand + strlen("DELLAYER:");
    1753           5 :         for (int i = 0; i < static_cast<int>(m_apoLayers.size()); ++i)
    1754             :         {
    1755           3 :             if (strcmp(pszLayerName, m_apoLayers[i]->GetName()) == 0)
    1756             :             {
    1757           1 :                 DeleteLayer(i);
    1758           1 :                 return nullptr;
    1759             :             }
    1760             :         }
    1761           2 :         CPLError(CE_Failure, CPLE_AppDefined, "Invalid layer name: %s",
    1762             :                  pszLayerName);
    1763           2 :         return nullptr;
    1764             :     }
    1765             : 
    1766             :     /* -------------------------------------------------------------------- */
    1767             :     /*      Special case for "CHECK_FREELIST_CONSISTENCY:"                  */
    1768             :     /* -------------------------------------------------------------------- */
    1769         537 :     if (STARTS_WITH_CI(pszSQLCommand, "CHECK_FREELIST_CONSISTENCY:"))
    1770             :     {
    1771           8 :         const char *pszLayerName =
    1772             :             pszSQLCommand + strlen("CHECK_FREELIST_CONSISTENCY:");
    1773           8 :         auto poLayer = GetLayerByName(pszLayerName);
    1774           8 :         if (poLayer)
    1775             :         {
    1776             :             return new OGROpenFileGDBSingleFeatureLayer(
    1777             :                 "result",
    1778          16 :                 CPLSPrintf("%d", static_cast<int>(
    1779           8 :                                      poLayer->CheckFreeListConsistency())));
    1780             :         }
    1781           0 :         CPLError(CE_Failure, CPLE_AppDefined, "Invalid layer name: %s",
    1782             :                  pszLayerName);
    1783           0 :         return nullptr;
    1784             :     }
    1785             : 
    1786             :     /* -------------------------------------------------------------------- */
    1787             :     /*      Special case for "REPACK"                                       */
    1788             :     /* -------------------------------------------------------------------- */
    1789         529 :     if (EQUAL(pszSQLCommand, "REPACK"))
    1790             :     {
    1791           3 :         bool bSuccess = true;
    1792           6 :         for (auto &poLayer : m_apoLayers)
    1793             :         {
    1794           3 :             if (!poLayer->Repack())
    1795           0 :                 bSuccess = false;
    1796             :         }
    1797             :         return new OGROpenFileGDBSingleFeatureLayer(
    1798           3 :             "result", bSuccess ? "true" : "false");
    1799             :     }
    1800         526 :     else if (STARTS_WITH(pszSQLCommand, "REPACK "))
    1801             :     {
    1802           2 :         const char *pszLayerName = pszSQLCommand + strlen("REPACK ");
    1803           2 :         auto poLayer = GetLayerByName(pszLayerName);
    1804           2 :         if (poLayer)
    1805             :         {
    1806           1 :             const bool bSuccess = poLayer->Repack();
    1807             :             return new OGROpenFileGDBSingleFeatureLayer(
    1808           1 :                 "result", bSuccess ? "true" : "false");
    1809             :         }
    1810           1 :         CPLError(CE_Failure, CPLE_AppDefined, "Invalid layer name: %s",
    1811             :                  pszLayerName);
    1812           1 :         return nullptr;
    1813             :     }
    1814             : 
    1815         524 :     bLastSQLUsedOptimizedImplementation = false;
    1816             : 
    1817             :     /* -------------------------------------------------------------------- */
    1818             :     /*      Special cases for SQL optimizations                             */
    1819             :     /* -------------------------------------------------------------------- */
    1820         516 :     if (STARTS_WITH_CI(pszSQLCommand, "SELECT ") &&
    1821          39 :         (pszDialect == nullptr || EQUAL(pszDialect, "") ||
    1822        1040 :          EQUAL(pszDialect, "OGRSQL")) &&
    1823         516 :         CPLTestBool(CPLGetConfigOption("OPENFILEGDB_USE_INDEX", "YES")))
    1824             :     {
    1825         516 :         swq_select oSelect;
    1826         516 :         if (oSelect.preparse(pszSQLCommand) != CE_None)
    1827           1 :             return nullptr;
    1828             : 
    1829             :         /* --------------------------------------------------------------------
    1830             :          */
    1831             :         /*      MIN/MAX/SUM/AVG/COUNT optimization */
    1832             :         /* --------------------------------------------------------------------
    1833             :          */
    1834         515 :         if (oSelect.join_count == 0 && oSelect.poOtherSelect == nullptr &&
    1835         515 :             oSelect.table_count == 1 && oSelect.order_specs == 0 &&
    1836         488 :             oSelect.query_mode != SWQM_DISTINCT_LIST &&
    1837         488 :             oSelect.where_expr == nullptr)
    1838             :         {
    1839             :             OGROpenFileGDBLayer *poLayer =
    1840         618 :                 reinterpret_cast<OGROpenFileGDBLayer *>(
    1841         309 :                     GetLayerByName(oSelect.table_defs[0].table_name));
    1842         309 :             if (poLayer)
    1843             :             {
    1844         308 :                 OGRMemLayer *poMemLayer = nullptr;
    1845             : 
    1846         308 :                 int i = 0;  // Used after for.
    1847         366 :                 for (; i < oSelect.result_columns(); i++)
    1848             :                 {
    1849         330 :                     swq_col_func col_func = oSelect.column_defs[i].col_func;
    1850         509 :                     if (!(col_func == SWQCF_MIN || col_func == SWQCF_MAX ||
    1851         185 :                           col_func == SWQCF_COUNT || col_func == SWQCF_AVG ||
    1852             :                           col_func == SWQCF_SUM))
    1853         272 :                         break;
    1854             : 
    1855         152 :                     if (oSelect.column_defs[i].field_name == nullptr)
    1856           0 :                         break;
    1857         152 :                     if (oSelect.column_defs[i].distinct_flag)
    1858           0 :                         break;
    1859         152 :                     if (oSelect.column_defs[i].target_type != SWQ_OTHER)
    1860           0 :                         break;
    1861             : 
    1862         152 :                     int idx = poLayer->GetLayerDefn()->GetFieldIndex(
    1863         152 :                         oSelect.column_defs[i].field_name);
    1864         152 :                     if (idx < 0)
    1865          89 :                         break;
    1866             : 
    1867             :                     OGRFieldDefn *poFieldDefn =
    1868          63 :                         poLayer->GetLayerDefn()->GetFieldDefn(idx);
    1869             : 
    1870          64 :                     if (col_func == SWQCF_SUM &&
    1871           1 :                         poFieldDefn->GetType() == OFTDateTime)
    1872           0 :                         break;
    1873             : 
    1874          63 :                     int eOutOGRType = -1;
    1875          63 :                     const OGRField *psField = nullptr;
    1876             :                     OGRField sField;
    1877          63 :                     if (col_func == SWQCF_MIN || col_func == SWQCF_MAX)
    1878             :                     {
    1879          55 :                         psField = poLayer->GetMinMaxValue(
    1880             :                             poFieldDefn, col_func == SWQCF_MIN, eOutOGRType);
    1881          55 :                         if (eOutOGRType < 0)
    1882           5 :                             break;
    1883             :                     }
    1884             :                     else
    1885             :                     {
    1886           8 :                         double dfMin = 0.0;
    1887           8 :                         double dfMax = 0.0;
    1888           8 :                         int nCount = 0;
    1889           8 :                         double dfSum = 0.0;
    1890             : 
    1891           8 :                         if (!poLayer->GetMinMaxSumCount(poFieldDefn, dfMin,
    1892             :                                                         dfMax, dfSum, nCount))
    1893           0 :                             break;
    1894           8 :                         psField = &sField;
    1895           8 :                         if (col_func == SWQCF_AVG)
    1896             :                         {
    1897           6 :                             if (nCount == 0)
    1898             :                             {
    1899           1 :                                 eOutOGRType = OFTReal;
    1900           1 :                                 psField = nullptr;
    1901             :                             }
    1902             :                             else
    1903             :                             {
    1904           5 :                                 if (poFieldDefn->GetType() == OFTDateTime)
    1905             :                                 {
    1906           1 :                                     eOutOGRType = OFTDateTime;
    1907           1 :                                     FileGDBDoubleDateToOGRDate(dfSum / nCount,
    1908             :                                                                false, &sField);
    1909             :                                 }
    1910             :                                 else
    1911             :                                 {
    1912           4 :                                     eOutOGRType = OFTReal;
    1913           4 :                                     sField.Real = dfSum / nCount;
    1914             :                                 }
    1915             :                             }
    1916             :                         }
    1917           2 :                         else if (col_func == SWQCF_COUNT)
    1918             :                         {
    1919           1 :                             sField.Integer = nCount;
    1920           1 :                             eOutOGRType = OFTInteger;
    1921             :                         }
    1922             :                         else
    1923             :                         {
    1924           1 :                             sField.Real = dfSum;
    1925           1 :                             eOutOGRType = OFTReal;
    1926             :                         }
    1927             :                     }
    1928             : 
    1929          58 :                     if (poMemLayer == nullptr)
    1930             :                     {
    1931          36 :                         poMemLayer =
    1932          36 :                             new OGRMemLayer("SELECT", nullptr, wkbNone);
    1933             :                         OGRFeature *poFeature =
    1934          36 :                             new OGRFeature(poMemLayer->GetLayerDefn());
    1935          36 :                         CPL_IGNORE_RET_VAL(
    1936          36 :                             poMemLayer->CreateFeature(poFeature));
    1937          36 :                         delete poFeature;
    1938             :                     }
    1939             : 
    1940             :                     const char *pszMinMaxFieldName =
    1941         102 :                         CPLSPrintf("%s_%s",
    1942             :                                    (col_func == SWQCF_MIN)   ? "MIN"
    1943          52 :                                    : (col_func == SWQCF_MAX) ? "MAX"
    1944          10 :                                    : (col_func == SWQCF_AVG) ? "AVG"
    1945           2 :                                    : (col_func == SWQCF_SUM) ? "SUM"
    1946             :                                                              : "COUNT",
    1947          58 :                                    oSelect.column_defs[i].field_name);
    1948             :                     OGRFieldDefn oFieldDefn(
    1949             :                         pszMinMaxFieldName,
    1950         116 :                         static_cast<OGRFieldType>(eOutOGRType));
    1951          58 :                     poMemLayer->CreateField(&oFieldDefn);
    1952          58 :                     if (psField != nullptr)
    1953             :                     {
    1954          55 :                         OGRFeature *poFeature = poMemLayer->GetFeature(0);
    1955          55 :                         poFeature->SetField(oFieldDefn.GetNameRef(), psField);
    1956          55 :                         CPL_IGNORE_RET_VAL(poMemLayer->SetFeature(poFeature));
    1957          55 :                         delete poFeature;
    1958             :                     }
    1959             :                 }
    1960         308 :                 if (i != oSelect.result_columns())
    1961             :                 {
    1962         272 :                     delete poMemLayer;
    1963             :                 }
    1964             :                 else
    1965             :                 {
    1966          36 :                     CPLDebug(
    1967             :                         "OpenFileGDB",
    1968             :                         "Using optimized MIN/MAX/SUM/AVG/COUNT implementation");
    1969          36 :                     bLastSQLUsedOptimizedImplementation = true;
    1970          36 :                     return poMemLayer;
    1971             :                 }
    1972             :             }
    1973             :         }
    1974             : 
    1975             :         /* --------------------------------------------------------------------
    1976             :          */
    1977             :         /*      ORDER BY optimization */
    1978             :         /* --------------------------------------------------------------------
    1979             :          */
    1980         479 :         if (oSelect.join_count == 0 && oSelect.poOtherSelect == nullptr &&
    1981         479 :             oSelect.table_count == 1 && oSelect.order_specs == 1 &&
    1982          26 :             oSelect.query_mode != SWQM_DISTINCT_LIST)
    1983             :         {
    1984             :             OGROpenFileGDBLayer *poLayer =
    1985          50 :                 reinterpret_cast<OGROpenFileGDBLayer *>(
    1986          25 :                     GetLayerByName(oSelect.table_defs[0].table_name));
    1987          50 :             if (poLayer != nullptr &&
    1988          25 :                 poLayer->HasIndexForField(oSelect.order_defs[0].field_name))
    1989             :             {
    1990          23 :                 OGRErr eErr = OGRERR_NONE;
    1991          23 :                 if (oSelect.where_expr != nullptr)
    1992             :                 {
    1993             :                     /* The where must be a simple comparison on the column */
    1994             :                     /* that is used for ordering */
    1995          12 :                     if (oSelect.where_expr->eNodeType == SNT_OPERATION &&
    1996           4 :                         OGROpenFileGDBIsComparisonOp(
    1997           4 :                             oSelect.where_expr->nOperation) &&
    1998           3 :                         oSelect.where_expr->nOperation != SWQ_NE &&
    1999           3 :                         oSelect.where_expr->nSubExprCount == 2 &&
    2000           3 :                         (oSelect.where_expr->papoSubExpr[0]->eNodeType ==
    2001           0 :                              SNT_COLUMN ||
    2002           0 :                          oSelect.where_expr->papoSubExpr[0]->eNodeType ==
    2003           3 :                              SNT_CONSTANT) &&
    2004           3 :                         oSelect.where_expr->papoSubExpr[0]->field_type ==
    2005           3 :                             SWQ_STRING &&
    2006           3 :                         EQUAL(oSelect.where_expr->papoSubExpr[0]->string_value,
    2007           8 :                               oSelect.order_defs[0].field_name) &&
    2008           2 :                         oSelect.where_expr->papoSubExpr[1]->eNodeType ==
    2009             :                             SNT_CONSTANT)
    2010             :                     {
    2011             :                         /* ok */
    2012             :                     }
    2013             :                     else
    2014           2 :                         eErr = OGRERR_FAILURE;
    2015             :                 }
    2016          23 :                 if (eErr == OGRERR_NONE)
    2017             :                 {
    2018          21 :                     int i = 0;  // Used after for.
    2019          40 :                     for (; i < oSelect.result_columns(); i++)
    2020             :                     {
    2021          23 :                         if (oSelect.column_defs[i].col_func != SWQCF_NONE)
    2022           1 :                             break;
    2023          22 :                         if (oSelect.column_defs[i].field_name == nullptr)
    2024           0 :                             break;
    2025          22 :                         if (oSelect.column_defs[i].distinct_flag)
    2026           0 :                             break;
    2027          22 :                         if (oSelect.column_defs[i].target_type != SWQ_OTHER)
    2028           1 :                             break;
    2029          21 :                         if (strcmp(oSelect.column_defs[i].field_name, "*") !=
    2030          27 :                                 0 &&
    2031          12 :                             poLayer->GetLayerDefn()->GetFieldIndex(
    2032           6 :                                 oSelect.column_defs[i].field_name) < 0)
    2033           2 :                             break;
    2034             :                     }
    2035          21 :                     if (i != oSelect.result_columns())
    2036           4 :                         eErr = OGRERR_FAILURE;
    2037             :                 }
    2038          23 :                 if (eErr == OGRERR_NONE)
    2039             :                 {
    2040          17 :                     int op = -1;
    2041          17 :                     swq_expr_node *poValue = nullptr;
    2042          17 :                     if (oSelect.where_expr != nullptr)
    2043             :                     {
    2044           2 :                         op = oSelect.where_expr->nOperation;
    2045           2 :                         poValue = oSelect.where_expr->papoSubExpr[1];
    2046             :                     }
    2047             : 
    2048          34 :                     FileGDBIterator *poIter = poLayer->BuildIndex(
    2049          17 :                         oSelect.order_defs[0].field_name,
    2050          17 :                         oSelect.order_defs[0].ascending_flag, op, poValue);
    2051             : 
    2052             :                     /* Check that they are no NULL values */
    2053          32 :                     if (oSelect.where_expr == nullptr && poIter != nullptr &&
    2054          15 :                         poIter->GetRowCount() !=
    2055          15 :                             poLayer->GetFeatureCount(FALSE))
    2056             :                     {
    2057           1 :                         delete poIter;
    2058           1 :                         poIter = nullptr;
    2059             :                     }
    2060             : 
    2061          17 :                     if (poIter != nullptr)
    2062             :                     {
    2063          16 :                         CPLDebug("OpenFileGDB",
    2064             :                                  "Using OGROpenFileGDBSimpleSQLLayer");
    2065          16 :                         bLastSQLUsedOptimizedImplementation = true;
    2066             :                         return new OGROpenFileGDBSimpleSQLLayer(
    2067          16 :                             poLayer, poIter, oSelect.result_columns(),
    2068          16 :                             oSelect.column_defs.data(), oSelect.offset,
    2069          16 :                             oSelect.limit);
    2070             :                     }
    2071             :                 }
    2072             :             }
    2073             :         }
    2074             :     }
    2075             : 
    2076         471 :     return GDALDataset::ExecuteSQL(pszSQLCommand, poSpatialFilter, pszDialect);
    2077             : }
    2078             : 
    2079             : /***********************************************************************/
    2080             : /*                           ReleaseResultSet()                        */
    2081             : /***********************************************************************/
    2082             : 
    2083         743 : void OGROpenFileGDBDataSource::ReleaseResultSet(OGRLayer *poResultsSet)
    2084             : {
    2085         743 :     delete poResultsSet;
    2086         743 : }
    2087             : 
    2088             : /***********************************************************************/
    2089             : /*                           GetFileList()                             */
    2090             : /***********************************************************************/
    2091             : 
    2092           8 : char **OGROpenFileGDBDataSource::GetFileList()
    2093             : {
    2094           8 :     int nInterestTable = -1;
    2095           8 :     const char *pszFilenameWithoutPath = CPLGetFilename(m_osDirName.c_str());
    2096          16 :     CPLString osFilenameRadix;
    2097           8 :     unsigned int unInterestTable = 0;
    2098          16 :     if (strlen(pszFilenameWithoutPath) == strlen("a00000000.gdbtable") &&
    2099           8 :         pszFilenameWithoutPath[0] == 'a' &&
    2100           0 :         sscanf(pszFilenameWithoutPath, "a%08x.gdbtable", &unInterestTable) == 1)
    2101             :     {
    2102           0 :         nInterestTable = static_cast<int>(unInterestTable);
    2103           0 :         osFilenameRadix = CPLSPrintf("a%08x.", nInterestTable);
    2104             :     }
    2105             : 
    2106           8 :     char **papszFiles = VSIReadDir(m_osDirName);
    2107          16 :     CPLStringList osStringList;
    2108           8 :     char **papszIter = papszFiles;
    2109        1148 :     for (; papszIter != nullptr && *papszIter != nullptr; papszIter++)
    2110             :     {
    2111        1140 :         if (strcmp(*papszIter, ".") == 0 || strcmp(*papszIter, "..") == 0)
    2112           4 :             continue;
    2113        1136 :         if (osFilenameRadix.empty() ||
    2114           0 :             strncmp(*papszIter, osFilenameRadix, osFilenameRadix.size()) == 0)
    2115             :         {
    2116        1136 :             osStringList.push_back(
    2117        2272 :                 CPLFormFilenameSafe(m_osDirName, *papszIter, nullptr));
    2118             :         }
    2119             :     }
    2120           8 :     CSLDestroy(papszFiles);
    2121          16 :     return osStringList.StealList();
    2122             : }
    2123             : 
    2124             : /************************************************************************/
    2125             : /*                           BuildSRS()                                 */
    2126             : /************************************************************************/
    2127             : 
    2128             : OGRSpatialReference *
    2129        2796 : OGROpenFileGDBDataSource::BuildSRS(const CPLXMLNode *psInfo)
    2130             : {
    2131             :     const char *pszWKT =
    2132        2796 :         CPLGetXMLValue(psInfo, "SpatialReference.WKT", nullptr);
    2133             :     const int nWKID =
    2134        2796 :         atoi(CPLGetXMLValue(psInfo, "SpatialReference.WKID", "0"));
    2135             :     // The concept of LatestWKID is explained in
    2136             :     // https://support.esri.com/en/technical-article/000013950
    2137             :     int nLatestWKID =
    2138        2796 :         atoi(CPLGetXMLValue(psInfo, "SpatialReference.LatestWKID", "0"));
    2139             : 
    2140        2796 :     std::unique_ptr<OGRSpatialReference, OGRSpatialReferenceReleaser> poSRS;
    2141        2796 :     if (nWKID > 0 || nLatestWKID > 0)
    2142             :     {
    2143             :         const auto ImportFromCode =
    2144        2386 :             [](OGRSpatialReference &oSRS, int nLatestCode, int nCode)
    2145             :         {
    2146        2386 :             bool bSuccess = false;
    2147        2386 :             CPLErrorStateBackuper oQuietError(CPLQuietErrorHandler);
    2148             : 
    2149             :             // Try first with nLatestWKID as there is a higher chance it is a
    2150             :             // EPSG code and not an ESRI one.
    2151        2386 :             if (nLatestCode > 0)
    2152             :             {
    2153         124 :                 if (nLatestCode > 32767)
    2154             :                 {
    2155           1 :                     if (oSRS.SetFromUserInput(
    2156           1 :                             CPLSPrintf("ESRI:%d", nLatestCode)) == OGRERR_NONE)
    2157             :                     {
    2158           0 :                         bSuccess = true;
    2159             :                     }
    2160             :                 }
    2161         123 :                 else if (oSRS.importFromEPSG(nLatestCode) == OGRERR_NONE)
    2162             :                 {
    2163         123 :                     bSuccess = true;
    2164             :                 }
    2165         124 :                 if (!bSuccess)
    2166             :                 {
    2167           1 :                     CPLDebug("OpenFileGDB", "Cannot import SRID %d",
    2168             :                              nLatestCode);
    2169             :                 }
    2170             :             }
    2171        2386 :             if (!bSuccess && nCode > 0)
    2172             :             {
    2173        2263 :                 if (nCode > 32767)
    2174             :                 {
    2175           1 :                     if (oSRS.SetFromUserInput(CPLSPrintf("ESRI:%d", nCode)) ==
    2176             :                         OGRERR_NONE)
    2177             :                     {
    2178           0 :                         bSuccess = true;
    2179             :                     }
    2180             :                 }
    2181        2262 :                 else if (oSRS.importFromEPSG(nCode) == OGRERR_NONE)
    2182             :                 {
    2183        2262 :                     bSuccess = true;
    2184             :                 }
    2185        2263 :                 if (!bSuccess)
    2186             :                 {
    2187           1 :                     CPLDebug("OpenFileGDB", "Cannot import SRID %d", nCode);
    2188             :                 }
    2189             :             }
    2190             : 
    2191        4772 :             return bSuccess;
    2192             :         };
    2193             : 
    2194             :         poSRS =
    2195        4770 :             std::unique_ptr<OGRSpatialReference, OGRSpatialReferenceReleaser>(
    2196        4770 :                 new OGRSpatialReference());
    2197        2385 :         poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
    2198        2385 :         if (!ImportFromCode(*poSRS.get(), nLatestWKID, nWKID))
    2199             :         {
    2200           0 :             poSRS.reset();
    2201             :         }
    2202             :         else
    2203             :         {
    2204        2385 :             const int nLatestVCSWKID = atoi(
    2205             :                 CPLGetXMLValue(psInfo, "SpatialReference.LatestVCSWKID", "0"));
    2206             :             const int nVCSWKID =
    2207        2385 :                 atoi(CPLGetXMLValue(psInfo, "SpatialReference.VCSWKID", "0"));
    2208        2385 :             if (nVCSWKID > 0 || nLatestVCSWKID > 0)
    2209             :             {
    2210             :                 auto poVertSRS = std::unique_ptr<OGRSpatialReference,
    2211             :                                                  OGRSpatialReferenceReleaser>(
    2212           2 :                     new OGRSpatialReference());
    2213           1 :                 if (ImportFromCode(*poVertSRS.get(), nLatestVCSWKID, nVCSWKID))
    2214             :                 {
    2215             :                     auto poCompoundSRS =
    2216             :                         std::unique_ptr<OGRSpatialReference,
    2217             :                                         OGRSpatialReferenceReleaser>(
    2218           0 :                             new OGRSpatialReference());
    2219           0 :                     if (poCompoundSRS->SetCompoundCS(
    2220           0 :                             std::string(poSRS->GetName())
    2221           0 :                                 .append(" + ")
    2222           0 :                                 .append(poVertSRS->GetName())
    2223             :                                 .c_str(),
    2224           0 :                             poSRS.get(), poVertSRS.get()) == OGRERR_NONE)
    2225             :                     {
    2226           0 :                         poCompoundSRS->SetAxisMappingStrategy(
    2227             :                             OAMS_TRADITIONAL_GIS_ORDER);
    2228           0 :                         poSRS = std::move(poCompoundSRS);
    2229             :                     }
    2230             :                 }
    2231           2 :                 if (!poSRS->IsCompound() &&
    2232           1 :                     !(pszWKT != nullptr && pszWKT[0] != '{'))
    2233             :                 {
    2234           0 :                     poSRS.reset();
    2235             :                 }
    2236             :             }
    2237             :         }
    2238             :     }
    2239        7566 :     if (pszWKT != nullptr && pszWKT[0] != '{' &&
    2240        2386 :         (poSRS == nullptr ||
    2241        2384 :          (strstr(pszWKT, "VERTCS") && !poSRS->IsCompound())))
    2242             :     {
    2243           3 :         poSRS.reset(BuildSRS(pszWKT));
    2244             :     }
    2245        5592 :     return poSRS.release();
    2246             : }
    2247             : 
    2248             : /************************************************************************/
    2249             : /*                           BuildSRS()                                 */
    2250             : /************************************************************************/
    2251             : 
    2252         386 : OGRSpatialReference *OGROpenFileGDBDataSource::BuildSRS(const char *pszWKT)
    2253             : {
    2254         386 :     std::shared_ptr<OGRSpatialReference> poSharedObj;
    2255         386 :     m_oCacheWKTToSRS.tryGet(pszWKT, poSharedObj);
    2256         386 :     if (poSharedObj)
    2257         199 :         return poSharedObj->Clone();
    2258             : 
    2259         187 :     OGRSpatialReference *poSRS = new OGRSpatialReference();
    2260         187 :     poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
    2261         187 :     if (poSRS->importFromWkt(pszWKT) != OGRERR_NONE)
    2262             :     {
    2263           0 :         delete poSRS;
    2264           0 :         poSRS = nullptr;
    2265             :     }
    2266         187 :     if (poSRS != nullptr)
    2267             :     {
    2268         187 :         if (CPLTestBool(CPLGetConfigOption("USE_OSR_FIND_MATCHES", "YES")))
    2269             :         {
    2270         187 :             auto poSRSMatch = poSRS->FindBestMatch(100);
    2271         187 :             if (poSRSMatch)
    2272             :             {
    2273         186 :                 poSRS->Release();
    2274         186 :                 poSRS = poSRSMatch;
    2275         186 :                 poSRS->SetAxisMappingStrategy(OAMS_TRADITIONAL_GIS_ORDER);
    2276             :             }
    2277             :             m_oCacheWKTToSRS.insert(
    2278         187 :                 pszWKT, std::shared_ptr<OGRSpatialReference>(poSRS->Clone()));
    2279             :         }
    2280             :         else
    2281             :         {
    2282           0 :             poSRS->AutoIdentifyEPSG();
    2283             :         }
    2284             :     }
    2285         187 :     return poSRS;
    2286             : }

Generated by: LCOV version 1.14