LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/gpkg - ogr_geopackage.h (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 117 129 90.7 %
Date: 2025-01-18 12:42:00 Functions: 48 52 92.3 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  GeoPackage Translator
       4             :  * Purpose:  Definition of classes for OGR GeoPackage driver.
       5             :  * Author:   Paul Ramsey, pramsey@boundlessgeo.com
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2013, Paul Ramsey <pramsey@boundlessgeo.com>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #ifndef OGR_GEOPACKAGE_H_INCLUDED
      14             : #define OGR_GEOPACKAGE_H_INCLUDED
      15             : 
      16             : #include "ogrsf_frmts.h"
      17             : #include "ogrsqlitebase.h"
      18             : #include "gpkgmbtilescommon.h"
      19             : #include "ogrsqliteutility.h"
      20             : #include "cpl_threadsafe_queue.hpp"
      21             : #include "ograrrowarrayhelper.h"
      22             : #include "ogr_p.h"
      23             : #include "ogr_wkb.h"
      24             : 
      25             : #include <condition_variable>
      26             : #include <limits>
      27             : #include <mutex>
      28             : #include <queue>
      29             : #include <vector>
      30             : #include <set>
      31             : #include <thread>
      32             : #include <cctype>
      33             : 
      34             : #define UNKNOWN_SRID -2
      35             : #define DEFAULT_SRID 0
      36             : 
      37             : #define ENABLE_GPKG_OGR_CONTENTS
      38             : 
      39             : #if defined(DEBUG) || defined(FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION) ||     \
      40             :     defined(ALLOW_FORMAT_DUMPS)
      41             : // Enable accepting a SQL dump (starting with a "-- SQL GPKG" line) as a valid
      42             : // file. This makes fuzzer life easier
      43             : #define ENABLE_SQL_GPKG_FORMAT
      44             : #endif
      45             : 
      46             : typedef enum
      47             : {
      48             :     GPKG_ATTRIBUTES,
      49             :     NOT_REGISTERED,
      50             : } GPKGASpatialVariant;
      51             : 
      52             : // Requirement 2
      53             : static const GUInt32 GP10_APPLICATION_ID = 0x47503130U;
      54             : static const GUInt32 GP11_APPLICATION_ID = 0x47503131U;
      55             : static const GUInt32 GPKG_APPLICATION_ID = 0x47504B47U;
      56             : static const GUInt32 GPKG_1_2_VERSION = 10200U;
      57             : static const GUInt32 GPKG_1_3_VERSION = 10300U;
      58             : static const GUInt32 GPKG_1_4_VERSION = 10400U;
      59             : 
      60             : static const size_t knApplicationIdPos = 68;
      61             : static const size_t knUserVersionPos = 60;
      62             : 
      63             : struct GPKGExtensionDesc
      64             : {
      65             :     CPLString osExtensionName{};
      66             :     CPLString osDefinition{};
      67             :     CPLString osScope{};
      68             : };
      69             : 
      70             : struct GPKGContentsDesc
      71             : {
      72             :     CPLString osDataType{};
      73             :     CPLString osIdentifier{};
      74             :     CPLString osDescription{};
      75             :     CPLString osMinX{};
      76             :     CPLString osMinY{};
      77             :     CPLString osMaxX{};
      78             :     CPLString osMaxY{};
      79             : };
      80             : 
      81             : class OGRGeoPackageLayer;
      82             : 
      83             : struct OGRGPKGTableLayerFillArrowArray
      84             : {
      85             :     std::unique_ptr<OGRArrowArrayHelper> psHelper{};
      86             :     int nCountRows = 0;
      87             :     bool bErrorOccurred = false;
      88             :     bool bMemoryLimitReached = false;
      89             :     bool bDateTimeAsString = false;
      90             :     std::string osErrorMsg{};
      91             :     OGRFeatureDefn *poFeatureDefn = nullptr;
      92             :     OGRGeoPackageLayer *poLayer = nullptr;
      93             : 
      94             :     struct tm brokenDown
      95             :     {
      96             :     };
      97             : 
      98             :     sqlite3 *hDB = nullptr;
      99             :     int nMaxBatchSize = 0;
     100             :     bool bAsynchronousMode = false;
     101             :     std::mutex oMutex{};
     102             :     std::condition_variable oCV{};
     103             :     bool bIsFinished = false;
     104             :     GIntBig nCurFID = 0;
     105             :     uint32_t nMemLimit = 0;
     106             :     // For spatial filtering
     107             :     const OGRLayer *poLayerForFilterGeom = nullptr;
     108             : };
     109             : 
     110             : void OGR_GPKG_Intersects_Spatial_Filter(sqlite3_context *pContext, int argc,
     111             :                                         sqlite3_value **argv);
     112             : 
     113             : /************************************************************************/
     114             : /*                          GDALGeoPackageDataset                       */
     115             : /************************************************************************/
     116             : 
     117             : class OGRGeoPackageTableLayer;
     118             : 
     119             : class GDALGeoPackageDataset final : public OGRSQLiteBaseDataSource,
     120             :                                     public GDALGPKGMBTilesLikePseudoDataset
     121             : {
     122             :     friend class GDALGeoPackageRasterBand;
     123             :     friend class OGRGeoPackageLayer;
     124             :     friend class OGRGeoPackageTableLayer;
     125             :     friend void OGRGeoPackageTransform(sqlite3_context *pContext, int argc,
     126             :                                        sqlite3_value **argv);
     127             : 
     128             :     std::string m_osFilenameInZip{};
     129             :     void *m_pSQLFunctionData = nullptr;
     130             :     GUInt32 m_nApplicationId = GPKG_APPLICATION_ID;
     131             :     GUInt32 m_nUserVersion = GPKG_1_2_VERSION;
     132             :     OGRGeoPackageTableLayer **m_papoLayers = nullptr;
     133             :     int m_nLayers = 0;
     134             :     void CheckUnknownExtensions(bool bCheckRasterTable = false);
     135             : #ifdef ENABLE_GPKG_OGR_CONTENTS
     136             :     bool m_bHasGPKGOGRContents = false;
     137             : #endif
     138             :     bool m_bHasGPKGGeometryColumns = false;
     139             :     bool m_bHasDefinition12_063 = false;
     140             :     bool m_bHasEpochColumn =
     141             :         false;  // whether gpkg_spatial_ref_sys has a epoch column
     142             :     bool m_bNonSpatialTablesNonRegisteredInGpkgContentsFound = false;
     143             :     mutable int m_nHasMetadataTables = -1;  // -1 = unknown, 0 = false, 1 = true
     144             :     int m_nCreateMetadataTables = -1;  // -1 = on demand, 0 = false, 1 = true
     145             : 
     146             :     // Set by CreateTileGriddedTable() and used by FinalizeRasterRegistration()
     147             :     std::string m_osSQLInsertIntoGpkg2dGriddedCoverageAncillary{};
     148             : 
     149             :     CPLString m_osIdentifier{};
     150             :     bool m_bIdentifierAsCO = false;
     151             :     CPLString m_osDescription{};
     152             :     bool m_bDescriptionAsCO = false;
     153             :     bool m_bGridCellEncodingAsCO = false;
     154             :     bool m_bHasReadMetadataFromStorage = false;
     155             :     bool m_bMetadataDirty = false;
     156             :     CPLStringList m_aosSubDatasets{};
     157             :     OGRSpatialReference m_oSRS{};
     158             :     bool m_bRecordInsertedInGPKGContent = false;
     159             :     bool m_bGeoTransformValid = false;
     160             :     double m_adfGeoTransform[6];
     161             :     int m_nSRID = -1;  // Unknown Cartesain
     162             :     double m_dfTMSMinX = 0.0;
     163             :     double m_dfTMSMaxY = 0.0;
     164             :     int m_nBandCountFromMetadata = 0;
     165             :     std::unique_ptr<GDALColorTable> m_poCTFromMetadata{};
     166             :     std::string m_osTFFromMetadata{};
     167             :     std::string m_osNodataValueFromMetadata{};
     168             : 
     169             :     // Used by OGRGeoPackageTransform
     170             :     int m_nLastCachedCTSrcSRId = -1;
     171             :     int m_nLastCachedCTDstSRId = -1;
     172             :     std::unique_ptr<OGRCoordinateTransformation> m_poLastCachedCT{};
     173             :     OGRWKBTransformCache m_oWKBTransformCache{};
     174             :     std::vector<GByte> m_abyWKBTransformCache{};
     175             : 
     176             :     int m_nOverviewCount = 0;
     177             :     GDALGeoPackageDataset **m_papoOverviewDS = nullptr;
     178             :     bool m_bZoomOther = false;
     179             : 
     180             :     bool m_bInFlushCache = false;
     181             : 
     182             :     bool m_bDateTimeWithTZ = true;
     183             : 
     184             :     bool m_bRemoveOGREmptyTable = false;
     185             : 
     186             :     CPLString m_osTilingScheme = "CUSTOM";
     187             : 
     188             :     // To optimize reading table constraints
     189             :     int m_nReadTableDefCount = 0;
     190             :     std::vector<SQLSqliteMasterContent> m_aoSqliteMasterContent{};
     191             : 
     192         769 :     void IncrementReadTableDefCounter()
     193             :     {
     194         769 :         m_nReadTableDefCount++;
     195         769 :     }
     196             : 
     197         761 :     int GetReadTableDefCounter() const
     198             :     {
     199         761 :         return m_nReadTableDefCount;
     200             :     }
     201             : 
     202             :     const std::vector<SQLSqliteMasterContent> &GetSqliteMasterContent();
     203             : 
     204             :     bool ComputeTileAndPixelShifts();
     205             :     bool AllocCachedTiles();
     206             :     bool InitRaster(GDALGeoPackageDataset *poParentDS, const char *pszTableName,
     207             :                     double dfMinX, double dfMinY, double dfMaxX, double dfMaxY,
     208             :                     const char *pszContentsMinX, const char *pszContentsMinY,
     209             :                     const char *pszContentsMaxX, const char *pszContentsMaxY,
     210             :                     char **papszOpenOptions, const SQLResult &oResult,
     211             :                     int nIdxInResult);
     212             :     bool InitRaster(GDALGeoPackageDataset *poParentDS, const char *pszTableName,
     213             :                     int nZoomLevel, int nBandCount, double dfTMSMinX,
     214             :                     double dfTMSMaxY, double dfPixelXSize, double dfPixelYSize,
     215             :                     int nTileWidth, int nTileHeight, int nTileMatrixWidth,
     216             :                     int nTileMatrixHeight, double dfGDALMinX, double dfGDALMinY,
     217             :                     double dfGDALMaxX, double dfGDALMaxY);
     218             : 
     219             :     bool OpenRaster(const char *pszTableName, const char *pszIdentifier,
     220             :                     const char *pszDescription, int nSRSId, double dfMinX,
     221             :                     double dfMinY, double dfMaxX, double dfMaxY,
     222             :                     const char *pszContentsMinX, const char *pszContentsMinY,
     223             :                     const char *pszContentsMaxX, const char *pszContentsMaxY,
     224             :                     bool bIsTiles, char **papszOptions);
     225             :     CPLErr FinalizeRasterRegistration();
     226             : 
     227             :     bool RegisterWebPExtension();
     228             :     bool RegisterZoomOtherExtension();
     229             :     void ParseCompressionOptions(char **papszOptions);
     230             : 
     231             :     bool HasMetadataTables() const;
     232             :     bool CreateMetadataTables();
     233             :     const char *CheckMetadataDomain(const char *pszDomain);
     234             :     void
     235             :     WriteMetadata(CPLXMLNode *psXMLNode, /* will be destroyed by the method */
     236             :                   const char *pszTableName);
     237             :     void FlushMetadata();
     238             : 
     239             :     int FindLayerIndex(const char *pszLayerName);
     240             : 
     241             :     bool HasGriddedCoverageAncillaryTable();
     242             :     bool CreateTileGriddedTable(char **papszOptions);
     243             : 
     244             :     void RemoveOGREmptyTable();
     245             : 
     246             :     std::map<CPLString, CPLString> m_oMapNameToType{};
     247             :     const std::map<CPLString, CPLString> &GetNameTypeMapFromSQliteMaster();
     248             :     void RemoveTableFromSQLiteMasterCache(const char *pszTableName);
     249             : 
     250             :     bool m_bMapTableToExtensionsBuilt = false;
     251             :     std::map<CPLString, std::vector<GPKGExtensionDesc>>
     252             :         m_oMapTableToExtensions{};
     253             :     const std::map<CPLString, std::vector<GPKGExtensionDesc>> &
     254             :     GetUnknownExtensionsTableSpecific();
     255             : 
     256             :     bool m_bMapTableToContentsBuilt = false;
     257             :     std::map<CPLString, GPKGContentsDesc> m_oMapTableToContents{};
     258             :     const std::map<CPLString, GPKGContentsDesc> &GetContents();
     259             : 
     260             :     std::map<int, OGRSpatialReference *> m_oMapSrsIdToSrs{};
     261             : 
     262             :     OGRErr DeleteLayerCommon(const char *pszLayerName);
     263             :     OGRErr DeleteRasterLayer(const char *pszLayerName);
     264             :     bool DeleteVectorOrRasterLayer(const char *pszLayerName);
     265             :     bool RenameVectorOrRasterLayer(const char *pszLayerName,
     266             :                                    const char *pszNewName);
     267             :     bool RenameRasterLayer(const char *pszLayerName,
     268             :                            const char *pszNewLayerName);
     269             : 
     270             :     bool ConvertGpkgSpatialRefSysToExtensionWkt2(bool bForceEpoch);
     271             :     void DetectSpatialRefSysColumns();
     272             : 
     273             :     std::map<int, bool> m_oSetGPKGLayerWarnings{};
     274             : 
     275             :     void FixupWrongRTreeTrigger();
     276             :     void FixupWrongMedataReferenceColumnNameUpdate();
     277             :     void ClearCachedRelationships();
     278             :     void LoadRelationships() const override;
     279             :     void LoadRelationshipsUsingRelatedTablesExtension() const;
     280             :     static std::string
     281             :     GenerateNameForRelationship(const char *pszBaseTableName,
     282             :                                 const char *pszRelatedTableName,
     283             :                                 const char *pszType);
     284             :     bool ValidateRelationship(const GDALRelationship *poRelationship,
     285             :                               std::string &failureReason);
     286             : 
     287             :     // Used by GDALGeoPackageDataset::GetRasterLayerDataset()
     288             :     std::map<std::string, std::unique_ptr<GDALDataset>> m_oCachedRasterDS{};
     289             : 
     290             :     bool CloseDB();
     291             :     CPLErr Close() override;
     292             : 
     293             :     CPL_DISALLOW_COPY_ASSIGN(GDALGeoPackageDataset)
     294             : 
     295             :   public:
     296             :     GDALGeoPackageDataset();
     297             :     virtual ~GDALGeoPackageDataset();
     298             : 
     299             :     char **GetFileList(void) override;
     300             : 
     301             :     virtual char **GetMetadata(const char *pszDomain = "") override;
     302             :     virtual const char *GetMetadataItem(const char *pszName,
     303             :                                         const char *pszDomain = "") override;
     304             :     virtual char **GetMetadataDomainList() override;
     305             :     virtual CPLErr SetMetadata(char **papszMetadata,
     306             :                                const char *pszDomain = "") override;
     307             :     virtual CPLErr SetMetadataItem(const char *pszName, const char *pszValue,
     308             :                                    const char *pszDomain = "") override;
     309             : 
     310             :     const OGRSpatialReference *GetSpatialRef() const override;
     311             :     CPLErr SetSpatialRef(const OGRSpatialReference *poSRS) override;
     312             : 
     313             :     virtual CPLErr GetGeoTransform(double *padfGeoTransform) override;
     314             :     virtual CPLErr SetGeoTransform(double *padfGeoTransform) override;
     315             : 
     316             :     virtual CPLErr FlushCache(bool bAtClosing) override;
     317             :     virtual CPLErr IBuildOverviews(const char *, int, const int *, int,
     318             :                                    const int *, GDALProgressFunc, void *,
     319             :                                    CSLConstList papszOptions) override;
     320             : 
     321       19902 :     virtual int GetLayerCount() override
     322             :     {
     323       19902 :         return m_nLayers;
     324             :     }
     325             : 
     326             :     int Open(GDALOpenInfo *poOpenInfo, const std::string &osFilenameInZip);
     327             :     int Create(const char *pszFilename, int nXSize, int nYSize, int nBands,
     328             :                GDALDataType eDT, char **papszOptions);
     329             :     OGRLayer *GetLayer(int iLayer) override;
     330             :     OGRErr DeleteLayer(int iLayer) override;
     331             :     OGRLayer *ICreateLayer(const char *pszName,
     332             :                            const OGRGeomFieldDefn *poGeomFieldDefn,
     333             :                            CSLConstList papszOptions) override;
     334             :     int TestCapability(const char *) override;
     335             : 
     336             :     std::vector<std::string>
     337             :     GetFieldDomainNames(CSLConstList papszOptions = nullptr) const override;
     338             :     const OGRFieldDomain *
     339             :     GetFieldDomain(const std::string &name) const override;
     340             :     bool AddFieldDomain(std::unique_ptr<OGRFieldDomain> &&domain,
     341             :                         std::string &failureReason) override;
     342             : 
     343             :     bool AddRelationship(std::unique_ptr<GDALRelationship> &&relationship,
     344             :                          std::string &failureReason) override;
     345             : 
     346             :     bool DeleteRelationship(const std::string &name,
     347             :                             std::string &failureReason) override;
     348             : 
     349             :     bool UpdateRelationship(std::unique_ptr<GDALRelationship> &&relationship,
     350             :                             std::string &failureReason) override;
     351             : 
     352             :     virtual std::pair<OGRLayer *, IOGRSQLiteGetSpatialWhere *>
     353             :     GetLayerWithGetSpatialWhereByName(const char *pszName) override;
     354             : 
     355             :     virtual OGRLayer *ExecuteSQL(const char *pszSQLCommand,
     356             :                                  OGRGeometry *poSpatialFilter,
     357             :                                  const char *pszDialect) override;
     358             :     virtual void ReleaseResultSet(OGRLayer *poLayer) override;
     359             : 
     360             :     virtual OGRErr CommitTransaction() override;
     361             :     virtual OGRErr RollbackTransaction() override;
     362             : 
     363        4839 :     inline bool IsInTransaction() const
     364             :     {
     365        4839 :         return nSoftTransactionLevel > 0;
     366             :     }
     367             : 
     368             :     static std::string LaunderName(const std::string &osStr);
     369             : 
     370             :     // At least 100000 to avoid conflicting with EPSG codes
     371             :     static constexpr int FIRST_CUSTOM_SRSID = 100000;
     372             : 
     373             :     int GetSrsId(const OGRSpatialReference *poSRS);
     374             :     const char *GetSrsName(const OGRSpatialReference &oSRS);
     375             :     std::unique_ptr<OGRSpatialReference, OGRSpatialReferenceReleaser>
     376             :     GetSpatialRef(int iSrsId, bool bFallbackToEPSG = false,
     377             :                   bool bEmitErrorIfNotFound = true);
     378             :     OGRErr CreateExtensionsTableIfNecessary();
     379             :     bool HasExtensionsTable();
     380             : 
     381         395 :     void SetMetadataDirty()
     382             :     {
     383         395 :         m_bMetadataDirty = true;
     384         395 :     }
     385             : 
     386             :     bool HasDataColumnsTable() const;
     387             :     bool HasDataColumnConstraintsTable() const;
     388             :     bool HasDataColumnConstraintsTableGPKG_1_0() const;
     389             :     bool CreateColumnsTableAndColumnConstraintsTablesIfNecessary();
     390             :     bool HasGpkgextRelationsTable() const;
     391             :     bool CreateRelationsTableIfNecessary();
     392             :     bool HasQGISLayerStyles() const;
     393             : 
     394          52 :     bool HasNonSpatialTablesNonRegisteredInGpkgContents() const
     395             :     {
     396          52 :         return m_bNonSpatialTablesNonRegisteredInGpkgContentsFound;
     397             :     }
     398             : 
     399             :     const char *GetGeometryTypeString(OGRwkbGeometryType eType);
     400             : 
     401             :     void ResetReadingAllLayers();
     402             :     OGRErr UpdateGpkgContentsLastChange(const char *pszTableName);
     403             : 
     404             :     static GDALDataset *CreateCopy(const char *pszFilename,
     405             :                                    GDALDataset *poSrcDS, int bStrict,
     406             :                                    char **papszOptions,
     407             :                                    GDALProgressFunc pfnProgress,
     408             :                                    void *pProgressData);
     409             : 
     410             :     static std::string GetCurrentDateEscapedSQL();
     411             : 
     412             :     GDALDataset *GetRasterLayerDataset(const char *pszLayerName);
     413             : 
     414             :   protected:
     415             :     virtual CPLErr IRasterIO(GDALRWFlag, int, int, int, int, void *, int, int,
     416             :                              GDALDataType, int, BANDMAP_TYPE, GSpacing,
     417             :                              GSpacing, GSpacing,
     418             :                              GDALRasterIOExtraArg *psExtraArg) override;
     419             : 
     420             :     // Coming from GDALGPKGMBTilesLikePseudoDataset
     421             : 
     422             :     virtual CPLErr IFlushCacheWithErrCode(bool bAtClosing) override;
     423             : 
     424       26020 :     virtual int IGetRasterCount() override
     425             :     {
     426       26020 :         return nBands;
     427             :     }
     428             : 
     429       54525 :     virtual GDALRasterBand *IGetRasterBand(int nBand) override
     430             :     {
     431       54525 :         return GetRasterBand(nBand);
     432             :     }
     433             : 
     434        9979 :     virtual sqlite3 *IGetDB() override
     435             :     {
     436        9979 :         return GetDB();
     437             :     }
     438             : 
     439       15147 :     virtual bool IGetUpdate() override
     440             :     {
     441       15147 :         return GetUpdate();
     442             :     }
     443             : 
     444             :     virtual bool ICanIWriteBlock() override;
     445             : 
     446         159 :     virtual OGRErr IStartTransaction() override
     447             :     {
     448         159 :         return SoftStartTransaction();
     449             :     }
     450             : 
     451         159 :     virtual OGRErr ICommitTransaction() override
     452             :     {
     453         159 :         return SoftCommitTransaction();
     454             :     }
     455             : 
     456          26 :     virtual const char *IGetFilename() override
     457             :     {
     458          26 :         return m_pszFilename;
     459             :     }
     460             : 
     461        9767 :     virtual int GetRowFromIntoTopConvention(int nRow) override
     462             :     {
     463        9767 :         return nRow;
     464             :     }
     465             : 
     466             :   private:
     467             :     OGRErr SetApplicationAndUserVersionId();
     468             :     bool ReOpenDB();
     469             :     bool OpenOrCreateDB(int flags);
     470             :     void InstallSQLFunctions();
     471             :     bool HasGDALAspatialExtension();
     472             :     std::string CreateRasterTriggersSQL(const std::string &osTableName);
     473             : };
     474             : 
     475             : /************************************************************************/
     476             : /*                   GPKGTemporaryForeignKeyCheckDisabler               */
     477             : /************************************************************************/
     478             : 
     479             : //! Instance of that class temporarily disable foreign key checks
     480             : class GPKGTemporaryForeignKeyCheckDisabler
     481             : {
     482             :   public:
     483         103 :     explicit GPKGTemporaryForeignKeyCheckDisabler(GDALGeoPackageDataset *poDS)
     484         103 :         : m_poDS(poDS), m_nPragmaForeignKeysOldValue(SQLGetInteger(
     485         103 :                             m_poDS->GetDB(), "PRAGMA foreign_keys", nullptr))
     486             :     {
     487         103 :         if (m_nPragmaForeignKeysOldValue)
     488             :         {
     489           0 :             CPL_IGNORE_RET_VAL(
     490           0 :                 SQLCommand(m_poDS->GetDB(), "PRAGMA foreign_keys = 0"));
     491             :         }
     492         103 :     }
     493             : 
     494         103 :     ~GPKGTemporaryForeignKeyCheckDisabler()
     495         103 :     {
     496         103 :         if (m_nPragmaForeignKeysOldValue)
     497             :         {
     498           0 :             CPL_IGNORE_RET_VAL(
     499           0 :                 SQLCommand(m_poDS->GetDB(), "PRAGMA foreign_keys = 1"));
     500             :         }
     501         103 :     }
     502             : 
     503             :   private:
     504             :     CPL_DISALLOW_COPY_ASSIGN(GPKGTemporaryForeignKeyCheckDisabler)
     505             : 
     506             :     GDALGeoPackageDataset *m_poDS = nullptr;
     507             :     int m_nPragmaForeignKeysOldValue = 0;
     508             : };
     509             : 
     510             : /************************************************************************/
     511             : /*                        GDALGeoPackageRasterBand                      */
     512             : /************************************************************************/
     513             : 
     514             : class GDALGeoPackageRasterBand final : public GDALGPKGMBTilesLikeRasterBand
     515             : {
     516             :     // Whether STATISTICS_MINIMUM and/or STATISTICS_MAXIMUM have been computed
     517             :     // from the min, max columns of the gpkg_2d_gridded_tile_ancillary table
     518             :     // (only for non-Byte data)
     519             :     bool m_bMinMaxComputedFromTileAncillary = false;
     520             :     double m_dfStatsMinFromTileAncillary =
     521             :         std::numeric_limits<double>::quiet_NaN();
     522             :     double m_dfStatsMaxFromTileAncillary =
     523             :         std::numeric_limits<double>::quiet_NaN();
     524             :     CPLStringList m_aosMD{};
     525             : 
     526             :     // Whether gpkg_metadata has been read to set initial metadata
     527             :     bool m_bHasReadMetadataFromStorage = false;
     528             : 
     529             :     // Whether STATISTICS_* have been set in this "session"
     530             :     bool m_bStatsMetadataSetInThisSession = false;
     531             : 
     532             :     bool m_bAddImplicitStatistics = true;
     533             : 
     534             :     void LoadBandMetadata();
     535             : 
     536             :   public:
     537             :     GDALGeoPackageRasterBand(GDALGeoPackageDataset *poDS, int nTileWidth,
     538             :                              int nTileHeight);
     539             : 
     540             :     virtual int GetOverviewCount() override;
     541             :     virtual GDALRasterBand *GetOverview(int nIdx) override;
     542             : 
     543             :     virtual CPLErr SetNoDataValue(double dfNoDataValue) override;
     544             : 
     545             :     virtual char **GetMetadata(const char *pszDomain = "") override;
     546             :     virtual const char *GetMetadataItem(const char *pszName,
     547             :                                         const char *pszDomain = "") override;
     548             :     virtual CPLErr SetMetadata(char **papszMetadata,
     549             :                                const char *pszDomain = "") override;
     550             :     virtual CPLErr SetMetadataItem(const char *pszName, const char *pszValue,
     551             :                                    const char *pszDomain = "") override;
     552             : 
     553         472 :     void AddImplicitStatistics(bool b)
     554             :     {
     555         472 :         m_bAddImplicitStatistics = b;
     556         472 :     }
     557             : 
     558         341 :     inline bool HaveStatsMetadataBeenSetInThisSession() const
     559             :     {
     560         341 :         return m_bStatsMetadataSetInThisSession;
     561             :     }
     562             : 
     563             :     void InvalidateStatistics();
     564             : };
     565             : 
     566             : /************************************************************************/
     567             : /*                           OGRGeoPackageLayer                         */
     568             : /************************************************************************/
     569             : 
     570             : class OGRGeoPackageLayer CPL_NON_FINAL : public OGRLayer,
     571             :                                          public IOGRSQLiteGetSpatialWhere
     572             : {
     573             :   protected:
     574             :     friend void OGR_GPKG_FillArrowArray_Step(sqlite3_context *pContext,
     575             :                                              int /*argc*/,
     576             :                                              sqlite3_value **argv);
     577             : 
     578             :     GDALGeoPackageDataset *m_poDS = nullptr;
     579             : 
     580             :     OGRFeatureDefn *m_poFeatureDefn = nullptr;
     581             :     GIntBig m_iNextShapeId = 0;
     582             : 
     583             :     sqlite3_stmt *m_poQueryStatement = nullptr;
     584             :     bool m_bDoStep = true;
     585             :     bool m_bEOF = false;
     586             : 
     587             :     char *m_pszFidColumn = nullptr;
     588             : 
     589             :     int m_iFIDCol = -1;
     590             :     int m_iGeomCol = -1;
     591             :     std::vector<int> m_anFieldOrdinals{};
     592             : 
     593             :     //! Whether to call OGRGeometry::SetPrecision() when reading back geometries from the database
     594             :     bool m_bUndoDiscardCoordLSBOnReading = false;
     595             : 
     596             :     void ClearStatement();
     597             :     virtual OGRErr ResetStatement() = 0;
     598             : 
     599             :     void BuildFeatureDefn(const char *pszLayerName, sqlite3_stmt *hStmt);
     600             : 
     601             :     OGRFeature *TranslateFeature(sqlite3_stmt *hStmt);
     602             :     bool ParseDateField(const char *pszTxt, OGRField *psField,
     603             :                         const OGRFieldDefn *poFieldDefn, GIntBig nFID);
     604             :     bool ParseDateField(sqlite3_stmt *hStmt, int iRawField, int nSqlite3ColType,
     605             :                         OGRField *psField, const OGRFieldDefn *poFieldDefn,
     606             :                         GIntBig nFID);
     607             :     bool ParseDateTimeField(const char *pszTxt, OGRField *psField,
     608             :                             const OGRFieldDefn *poFieldDefn, GIntBig nFID);
     609             :     bool ParseDateTimeField(sqlite3_stmt *hStmt, int iRawField,
     610             :                             int nSqlite3ColType, OGRField *psField,
     611             :                             const OGRFieldDefn *poFieldDefn, GIntBig nFID);
     612             : 
     613         159 :     GDALDataset *GetDataset() override
     614             :     {
     615         159 :         return m_poDS;
     616             :     }
     617             : 
     618             :     virtual bool GetArrowStream(struct ArrowArrayStream *out_stream,
     619             :                                 CSLConstList papszOptions = nullptr) override;
     620             :     virtual int GetNextArrowArray(struct ArrowArrayStream *,
     621             :                                   struct ArrowArray *out_array) override;
     622             : 
     623             :     CPL_DISALLOW_COPY_ASSIGN(OGRGeoPackageLayer)
     624             : 
     625             :   public:
     626             :     explicit OGRGeoPackageLayer(GDALGeoPackageDataset *poDS);
     627             :     virtual ~OGRGeoPackageLayer();
     628             :     /************************************************************************/
     629             :     /* OGR API methods */
     630             : 
     631             :     OGRFeature *GetNextFeature() override;
     632             :     const char *GetFIDColumn() override;
     633             :     void ResetReading() override;
     634             :     int TestCapability(const char *) override;
     635             : 
     636         523 :     OGRFeatureDefn *GetLayerDefn() override
     637             :     {
     638         523 :         return m_poFeatureDefn;
     639             :     }
     640             : 
     641             :     OGRErr SetIgnoredFields(CSLConstList papszFields) override;
     642             : 
     643           0 :     virtual bool HasFastSpatialFilter(int /*iGeomCol*/) override
     644             :     {
     645           0 :         return false;
     646             :     }
     647             : 
     648           0 :     virtual CPLString GetSpatialWhere(int /*iGeomCol*/,
     649             :                                       OGRGeometry * /*poFilterGeom*/) override
     650             :     {
     651           0 :         return "";
     652             :     }
     653             : };
     654             : 
     655             : /************************************************************************/
     656             : /*                        OGRGeoPackageTableLayer                       */
     657             : /************************************************************************/
     658             : 
     659             : struct OGRGPKGTableLayerFillArrowArray;
     660             : struct sqlite_rtree_bl;
     661             : 
     662             : class OGRGeoPackageTableLayer final : public OGRGeoPackageLayer
     663             : {
     664             :     char *m_pszTableName = nullptr;
     665             :     bool m_bIsTable = true;  // sensible init for creation mode
     666             :     bool m_bIsSpatial = false;
     667             :     bool m_bIsInGpkgContents = false;
     668             :     bool m_bFeatureDefnCompleted = false;
     669             :     int m_iSrs = 0;
     670             :     int m_nZFlag = 0;
     671             :     int m_nMFlag = 0;
     672             :     OGRGeomCoordinateBinaryPrecision m_sBinaryPrecision{};
     673             :     OGREnvelope *m_poExtent = nullptr;
     674             : #ifdef ENABLE_GPKG_OGR_CONTENTS
     675             :     GIntBig m_nTotalFeatureCount = -1;
     676             :     bool m_bOGRFeatureCountTriggersEnabled = false;
     677             :     bool m_bAddOGRFeatureCountTriggers = false;
     678             :     bool m_bFeatureCountTriggersDeletedInTransaction = false;
     679             : #endif
     680             :     CPLString m_soColumns{};
     681             :     std::vector<bool> m_abGeneratedColumns{};  // .size() ==
     682             :         // m_poFeatureDefn->GetFieldDefnCount()
     683             :     CPLString m_soFilter{};
     684             :     CPLString osQuery{};
     685             :     CPLString m_osRTreeName{};
     686             :     CPLString m_osFIDForRTree{};
     687             :     bool m_bExtentChanged = false;
     688             :     bool m_bContentChanged = false;
     689             :     sqlite3_stmt *m_poUpdateStatement = nullptr;
     690             :     std::string m_osUpdateStatementSQL{};
     691             :     bool m_bInsertStatementWithFID = false;
     692             :     bool m_bInsertStatementWithUpsert = false;
     693             :     std::string m_osInsertStatementUpsertUniqueColumnName{};
     694             :     sqlite3_stmt *m_poInsertStatement = nullptr;
     695             :     sqlite3_stmt *m_poGetFeatureStatement = nullptr;
     696             :     bool m_bDeferredSpatialIndexCreation = false;
     697             :     // m_bHasSpatialIndex cannot be bool.  -1 is unset.
     698             :     int m_bHasSpatialIndex = -1;
     699             :     bool m_bDropRTreeTable = false;
     700             :     bool m_abHasGeometryExtension[wkbTriangle + 1];
     701             :     bool m_bPreservePrecision = true;
     702             :     bool m_bTruncateFields = false;
     703             :     bool m_bDeferredCreation = false;
     704             :     bool m_bTableCreatedInTransaction = false;
     705             :     bool m_bLaunder = false;
     706             :     int m_iFIDAsRegularColumnIndex = -1;
     707             :     std::string m_osInsertionBuffer{};  // used by FeatureBindParameters to
     708             :                                         // store datetime values
     709             : 
     710             :     CPLString m_osIdentifierLCO{};
     711             :     CPLString m_osDescriptionLCO{};
     712             :     bool m_bHasReadMetadataFromStorage = false;
     713             :     bool m_bHasTriedDetectingFID64 = false;
     714             :     GPKGASpatialVariant m_eASpatialVariant = GPKG_ATTRIBUTES;
     715             :     std::set<OGRwkbGeometryType> m_eSetBadGeomTypeWarned{};
     716             : 
     717             :     int m_nIsCompatOfOptimizedGetNextArrowArray = -1;
     718             :     bool m_bGetNextArrowArrayCalledSinceResetReading = false;
     719             : 
     720             :     int m_nCountInsertInTransactionThreshold = -1;
     721             :     GIntBig m_nCountInsertInTransaction = 0;
     722             :     std::vector<CPLString> m_aoRTreeTriggersSQL{};
     723             :     bool m_bUpdate1TriggerDisabled = false;
     724             :     bool m_bHasUpdate6And7Triggers = false;
     725             :     std::string m_osUpdate1Trigger{};
     726             : 
     727             :     typedef struct
     728             :     {
     729             :         GIntBig nId;
     730             :         float fMinX;
     731             :         float fMinY;
     732             :         float fMaxX;
     733             :         float fMaxY;
     734             :     } GPKGRTreeEntry;
     735             : 
     736             :     std::vector<GPKGRTreeEntry> m_aoRTreeEntries{};
     737             : 
     738             :     // Variables used for background RTree building
     739             :     std::string m_osAsyncDBName{};
     740             :     std::string m_osAsyncDBAttachName{};
     741             :     sqlite3 *m_hAsyncDBHandle = nullptr;
     742             :     sqlite_rtree_bl *m_hRTree = nullptr;
     743             :     cpl::ThreadSafeQueue<std::vector<GPKGRTreeEntry>> m_oQueueRTreeEntries{};
     744             :     bool m_bAllowedRTreeThread = false;
     745             :     bool m_bThreadRTreeStarted = false;
     746             :     bool m_bErrorDuringRTreeThread = false;
     747             :     size_t m_nRTreeBatchSize =
     748             :         10 * 1000;  // maximum size of a std::vector<GPKGRTreeEntry> item in
     749             :                     // m_oQueueRTreeEntries
     750             :     size_t m_nRTreeBatchesBeforeStart =
     751             :         10;  // number of items in m_oQueueRTreeEntries before starting the
     752             :              // thread
     753             :     std::thread m_oThreadRTree{};
     754             : 
     755             :     OGRISO8601Format m_sDateTimeFormat = {OGRISO8601Precision::AUTO};
     756             : 
     757             :     void StartAsyncRTree();
     758             :     void CancelAsyncRTree();
     759             :     void RemoveAsyncRTreeTempDB();
     760             :     void AsyncRTreeThreadFunction();
     761             : 
     762             :     OGRErr ResetStatementInternal(GIntBig nStartIndex);
     763             :     virtual OGRErr ResetStatement() override;
     764             : 
     765             :     void BuildWhere();
     766             :     OGRErr RegisterGeometryColumn();
     767             : 
     768             :     CPLString
     769             :     GetColumnsOfCreateTable(const std::vector<OGRFieldDefn *> &apoFields);
     770             :     CPLString
     771             :     BuildSelectFieldList(const std::vector<OGRFieldDefn *> &apoFields);
     772             :     OGRErr RecreateTable(const CPLString &osColumnsForCreate,
     773             :                          const CPLString &osFieldListForSelect);
     774             : #ifdef ENABLE_GPKG_OGR_CONTENTS
     775             :     void CreateFeatureCountTriggers(const char *pszTableName = nullptr);
     776             :     void DisableFeatureCountTriggers(bool bNullifyFeatureCount = true);
     777             : #endif
     778             : 
     779             :     void CheckGeometryType(const OGRFeature *poFeature);
     780             : 
     781             :     OGRErr ReadTableDefinition();
     782             :     void InitView();
     783             : 
     784             :     bool DoSpecialProcessingForColumnCreation(const OGRFieldDefn *poField);
     785             : 
     786             :     bool StartDeferredSpatialIndexUpdate();
     787             :     bool FlushPendingSpatialIndexUpdate();
     788             :     void WorkaroundUpdate1TriggerIssue();
     789             :     void RevertWorkaroundUpdate1TriggerIssue();
     790             : 
     791             :     OGRErr RenameFieldInAuxiliaryTables(const char *pszOldName,
     792             :                                         const char *pszNewName);
     793             : 
     794             :     OGRErr CreateOrUpsertFeature(OGRFeature *poFeature, bool bUpsert);
     795             : 
     796             :     GIntBig GetTotalFeatureCount();
     797             : 
     798             :     CPL_DISALLOW_COPY_ASSIGN(OGRGeoPackageTableLayer)
     799             : 
     800             :     // Used when m_nIsCompatOfOptimizedGetNextArrowArray == TRUE
     801             :     struct ArrowArrayPrefetchTask
     802             :     {
     803             :         std::thread m_oThread{};
     804             :         std::condition_variable m_oCV{};
     805             :         std::mutex m_oMutex{};
     806             :         bool m_bArrayReady = false;
     807             :         bool m_bFetchRows = false;
     808             :         bool m_bStop = false;
     809             :         bool m_bMemoryLimitReached = false;
     810             :         std::string m_osErrorMsg{};
     811             :         std::unique_ptr<GDALGeoPackageDataset> m_poDS{};
     812             :         OGRGeoPackageTableLayer *m_poLayer{};
     813             :         GIntBig m_iStartShapeId = 0;
     814             :         std::unique_ptr<struct ArrowArray> m_psArrowArray = nullptr;
     815             :     };
     816             : 
     817             :     std::queue<std::unique_ptr<ArrowArrayPrefetchTask>>
     818             :         m_oQueueArrowArrayPrefetchTasks{};
     819             : 
     820             :     // Used when m_nIsCompatOfOptimizedGetNextArrowArray == FALSE
     821             :     std::thread m_oThreadNextArrowArray{};
     822             :     std::unique_ptr<OGRGPKGTableLayerFillArrowArray> m_poFillArrowArray{};
     823             :     std::unique_ptr<GDALGeoPackageDataset> m_poOtherDS{};
     824             : 
     825             :     virtual int GetNextArrowArray(struct ArrowArrayStream *,
     826             :                                   struct ArrowArray *out_array) override;
     827             :     int GetNextArrowArrayInternal(struct ArrowArray *out_array,
     828             :                                   std::string &osErrorMsg,
     829             :                                   bool &bMemoryLimitReached);
     830             :     int GetNextArrowArrayAsynchronous(struct ArrowArrayStream *stream,
     831             :                                       struct ArrowArray *out_array);
     832             :     void GetNextArrowArrayAsynchronousWorker();
     833             :     void CancelAsyncNextArrowArray();
     834             : 
     835             :   protected:
     836             :     friend void OGR_GPKG_Intersects_Spatial_Filter(sqlite3_context *pContext,
     837             :                                                    int /*argc*/,
     838             :                                                    sqlite3_value **argv);
     839             : 
     840             :   public:
     841             :     OGRGeoPackageTableLayer(GDALGeoPackageDataset *poDS,
     842             :                             const char *pszTableName);
     843             :     virtual ~OGRGeoPackageTableLayer();
     844             : 
     845             :     /************************************************************************/
     846             :     /* OGR API methods */
     847             : 
     848       11184 :     const char *GetName() override
     849             :     {
     850       11184 :         return GetDescription();
     851             :     }
     852             : 
     853             :     const char *GetFIDColumn() override;
     854             :     OGRwkbGeometryType GetGeomType() override;
     855             :     const char *GetGeometryColumn() override;
     856             :     OGRFeatureDefn *GetLayerDefn() override;
     857             :     int TestCapability(const char *) override;
     858             :     OGRErr CreateField(const OGRFieldDefn *poField,
     859             :                        int bApproxOK = TRUE) override;
     860             :     OGRErr CreateGeomField(const OGRGeomFieldDefn *poGeomFieldIn,
     861             :                            int bApproxOK = TRUE) override;
     862             :     virtual OGRErr DeleteField(int iFieldToDelete) override;
     863             :     virtual OGRErr AlterFieldDefn(int iFieldToAlter,
     864             :                                   OGRFieldDefn *poNewFieldDefn,
     865             :                                   int nFlagsIn) override;
     866             :     virtual OGRErr
     867             :     AlterGeomFieldDefn(int iGeomFieldToAlter,
     868             :                        const OGRGeomFieldDefn *poNewGeomFieldDefn,
     869             :                        int nFlagsIn) override;
     870             :     virtual OGRErr ReorderFields(int *panMap) override;
     871             :     void ResetReading() override;
     872             :     OGRErr SetNextByIndex(GIntBig nIndex) override;
     873             :     OGRErr ICreateFeature(OGRFeature *poFeature) override;
     874             :     OGRErr ISetFeature(OGRFeature *poFeature) override;
     875             :     OGRErr IUpsertFeature(OGRFeature *poFeature) override;
     876             :     OGRErr IUpdateFeature(OGRFeature *poFeature, int nUpdatedFieldsCount,
     877             :                           const int *panUpdatedFieldsIdx,
     878             :                           int nUpdatedGeomFieldsCount,
     879             :                           const int *panUpdatedGeomFieldsIdx,
     880             :                           bool bUpdateStyleString) override;
     881             :     OGRErr DeleteFeature(GIntBig nFID) override;
     882             :     virtual void SetSpatialFilter(OGRGeometry *) override;
     883             : 
     884          21 :     virtual void SetSpatialFilter(int iGeomField, OGRGeometry *poGeom) override
     885             :     {
     886          21 :         OGRGeoPackageLayer::SetSpatialFilter(iGeomField, poGeom);
     887          21 :     }
     888             : 
     889             :     OGRErr SetAttributeFilter(const char *pszQuery) override;
     890             :     OGRErr SyncToDisk() override;
     891             :     OGRFeature *GetNextFeature() override;
     892             :     OGRFeature *GetFeature(GIntBig nFID) override;
     893             :     OGRErr StartTransaction() override;
     894             :     OGRErr CommitTransaction() override;
     895             :     OGRErr RollbackTransaction() override;
     896             :     GIntBig GetFeatureCount(int) override;
     897             :     OGRErr GetExtent(OGREnvelope *psExtent, int bForce = TRUE) override;
     898             : 
     899          75 :     virtual OGRErr GetExtent(int iGeomField, OGREnvelope *psExtent,
     900             :                              int bForce) override
     901             :     {
     902          75 :         return OGRGeoPackageLayer::GetExtent(iGeomField, psExtent, bForce);
     903             :     }
     904             : 
     905             :     virtual OGRErr GetExtent3D(int iGeomField, OGREnvelope3D *psExtent3D,
     906             :                                int bForce) override;
     907             :     OGRGeometryTypeCounter *GetGeometryTypes(int iGeomField, int nFlagsGGT,
     908             :                                              int &nEntryCountOut,
     909             :                                              GDALProgressFunc pfnProgress,
     910             :                                              void *pProgressData) override;
     911             : 
     912             :     void RecomputeExtent();
     913             : 
     914             :     void SetOpeningParameters(const char *pszTableName,
     915             :                               const char *pszObjectType, bool bIsInGpkgContents,
     916             :                               bool bIsSpatial, const char *pszGeomColName,
     917             :                               const char *pszGeomType, bool bHasZ, bool bHasM);
     918             :     void SetCreationParameters(
     919             :         OGRwkbGeometryType eGType, const char *pszGeomColumnName,
     920             :         int bGeomNullable, const OGRSpatialReference *poSRS,
     921             :         const char *pszSRID, const OGRGeomCoordinatePrecision &oCoordPrec,
     922             :         bool bDiscardCoordLSB, bool bUndoDiscardCoordLSBOnReading,
     923             :         const char *pszFIDColumnName, const char *pszIdentifier,
     924             :         const char *pszDescription);
     925             :     void SetDeferredSpatialIndexCreation(bool bFlag);
     926             : 
     927          52 :     void SetASpatialVariant(GPKGASpatialVariant eASpatialVariant)
     928             :     {
     929          52 :         m_eASpatialVariant = eASpatialVariant;
     930          52 :         if (eASpatialVariant == GPKG_ATTRIBUTES)
     931          40 :             m_bIsInGpkgContents = true;
     932          52 :     }
     933             : 
     934         669 :     void SetDateTimePrecision(OGRISO8601Precision ePrecision)
     935             :     {
     936         669 :         m_sDateTimeFormat.ePrecision = ePrecision;
     937         669 :     }
     938             : 
     939             :     void CreateSpatialIndexIfNecessary();
     940             :     void FinishOrDisableThreadedRTree();
     941             :     bool FlushInMemoryRTree(sqlite3 *hRTreeDB, const char *pszRTreeName);
     942             :     bool CreateSpatialIndex(const char *pszTableName = nullptr);
     943             :     bool DropSpatialIndex(bool bCalledFromSQLFunction = false);
     944             :     CPLString ReturnSQLCreateSpatialIndexTriggers(const char *pszTableName,
     945             :                                                   const char *pszGeomColName);
     946             :     CPLString ReturnSQLDropSpatialIndexTriggers();
     947             : 
     948             :     virtual char **GetMetadata(const char *pszDomain = "") override;
     949             :     virtual const char *GetMetadataItem(const char *pszName,
     950             :                                         const char *pszDomain = "") override;
     951             :     virtual char **GetMetadataDomainList() override;
     952             : 
     953             :     virtual CPLErr SetMetadata(char **papszMetadata,
     954             :                                const char *pszDomain = "") override;
     955             :     virtual CPLErr SetMetadataItem(const char *pszName, const char *pszValue,
     956             :                                    const char *pszDomain = "") override;
     957             : 
     958             :     virtual OGRErr Rename(const char *pszDstTableName) override;
     959             : 
     960             :     virtual bool HasFastSpatialFilter(int iGeomCol) override;
     961             :     virtual CPLString GetSpatialWhere(int iGeomCol,
     962             :                                       OGRGeometry *poFilterGeom) override;
     963             : 
     964             :     OGRErr Truncate();
     965             : 
     966             :     bool HasSpatialIndex();
     967             : 
     968         681 :     void SetPrecisionFlag(int bFlag)
     969             :     {
     970         681 :         m_bPreservePrecision = CPL_TO_BOOL(bFlag);
     971         681 :     }
     972             : 
     973         681 :     void SetTruncateFieldsFlag(int bFlag)
     974             :     {
     975         681 :         m_bTruncateFields = CPL_TO_BOOL(bFlag);
     976         681 :     }
     977             : 
     978         681 :     void SetLaunder(bool bFlag)
     979             :     {
     980         681 :         m_bLaunder = bFlag;
     981         681 :     }
     982             : 
     983             :     OGRErr RunDeferredCreationIfNecessary();
     984             :     bool RunDeferredDropRTreeTableIfNecessary();
     985             :     bool DoJobAtTransactionCommit();
     986             :     bool DoJobAtTransactionRollback();
     987             :     bool RunDeferredSpatialIndexUpdate();
     988             : 
     989             : #ifdef ENABLE_GPKG_OGR_CONTENTS
     990          35 :     bool GetAddOGRFeatureCountTriggers() const
     991             :     {
     992          35 :         return m_bAddOGRFeatureCountTriggers;
     993             :     }
     994             : 
     995          63 :     void SetAddOGRFeatureCountTriggers(bool b)
     996             :     {
     997          63 :         m_bAddOGRFeatureCountTriggers = b;
     998          63 :     }
     999             : 
    1000          35 :     bool GetOGRFeatureCountTriggersDeletedInTransaction() const
    1001             :     {
    1002          35 :         return m_bFeatureCountTriggersDeletedInTransaction;
    1003             :     }
    1004             : 
    1005           7 :     void SetOGRFeatureCountTriggersEnabled(bool b)
    1006             :     {
    1007           7 :         m_bOGRFeatureCountTriggersEnabled = b;
    1008           7 :     }
    1009             : 
    1010             :     void DisableFeatureCount();
    1011             : #endif
    1012             : 
    1013             :     bool CreateGeometryExtensionIfNecessary(OGRwkbGeometryType eGType);
    1014             : 
    1015             :     /************************************************************************/
    1016             :     /* GPKG methods */
    1017             : 
    1018             :   private:
    1019             :     bool CheckUpdatableTable(const char *pszOperation);
    1020             :     OGRErr UpdateExtent(const OGREnvelope *poExtent);
    1021             :     OGRErr SaveExtent();
    1022             :     OGRErr SaveTimestamp();
    1023             :     OGRErr BuildColumns();
    1024             :     bool IsGeomFieldSet(OGRFeature *poFeature);
    1025             :     std::string FeatureGenerateUpdateSQL(const OGRFeature *poFeature) const;
    1026             :     std::string FeatureGenerateUpdateSQL(
    1027             :         const OGRFeature *poFeature, int nUpdatedFieldsCount,
    1028             :         const int *panUpdatedFieldsIdx, int nUpdatedGeomFieldsCount,
    1029             :         const int *panUpdatedGeomFieldsIdx) const;
    1030             :     CPLString
    1031             :     FeatureGenerateInsertSQL(OGRFeature *poFeature, bool bAddFID,
    1032             :                              bool bBindUnsetFields, bool bUpsert,
    1033             :                              const std::string &osUpsertUniqueColumnName);
    1034             :     OGRErr FeatureBindUpdateParameters(OGRFeature *poFeature,
    1035             :                                        sqlite3_stmt *poStmt);
    1036             :     OGRErr FeatureBindInsertParameters(OGRFeature *poFeature,
    1037             :                                        sqlite3_stmt *poStmt, bool bAddFID,
    1038             :                                        bool bBindUnsetFields);
    1039             :     OGRErr FeatureBindParameters(OGRFeature *poFeature, sqlite3_stmt *poStmt,
    1040             :                                  int *pnColCount, bool bAddFID,
    1041             :                                  bool bBindUnsetFields, int nUpdatedFieldsCount,
    1042             :                                  const int *panUpdatedFieldsIdx,
    1043             :                                  int nUpdatedGeomFieldsCount,
    1044             :                                  const int *panUpdatedGeomFieldsIdx);
    1045             : 
    1046             :     void UpdateContentsToNullExtent();
    1047             : 
    1048             :     void CheckUnknownExtensions();
    1049             :     bool CreateGeometryExtensionIfNecessary(const OGRGeometry *poGeom);
    1050             : };
    1051             : 
    1052             : /************************************************************************/
    1053             : /*                         OGRGeoPackageSelectLayer                     */
    1054             : /************************************************************************/
    1055             : 
    1056             : class OGRGeoPackageSelectLayer final : public OGRGeoPackageLayer,
    1057             :                                        public IOGRSQLiteSelectLayer
    1058             : {
    1059             :     CPL_DISALLOW_COPY_ASSIGN(OGRGeoPackageSelectLayer)
    1060             : 
    1061             :     OGRSQLiteSelectLayerCommonBehaviour *poBehavior = nullptr;
    1062             : 
    1063             :     virtual OGRErr ResetStatement() override;
    1064             : 
    1065             :   public:
    1066             :     OGRGeoPackageSelectLayer(GDALGeoPackageDataset *, const CPLString &osSQL,
    1067             :                              sqlite3_stmt *,
    1068             :                              bool bUseStatementForGetNextFeature,
    1069             :                              bool bEmptyLayer);
    1070             :     virtual ~OGRGeoPackageSelectLayer();
    1071             : 
    1072             :     virtual void ResetReading() override;
    1073             : 
    1074             :     virtual OGRFeature *GetNextFeature() override;
    1075             :     virtual GIntBig GetFeatureCount(int) override;
    1076             : 
    1077          22 :     virtual void SetSpatialFilter(OGRGeometry *poGeom) override
    1078             :     {
    1079          22 :         SetSpatialFilter(0, poGeom);
    1080          22 :     }
    1081             : 
    1082             :     virtual void SetSpatialFilter(int iGeomField, OGRGeometry *) override;
    1083             :     virtual OGRErr SetAttributeFilter(const char *) override;
    1084             : 
    1085             :     virtual int TestCapability(const char *) override;
    1086             : 
    1087           1 :     virtual OGRErr GetExtent(OGREnvelope *psExtent, int bForce = TRUE) override
    1088             :     {
    1089           1 :         return GetExtent(0, psExtent, bForce);
    1090             :     }
    1091             : 
    1092             :     virtual OGRErr GetExtent(int iGeomField, OGREnvelope *psExtent,
    1093             :                              int bForce = TRUE) override;
    1094             : 
    1095         523 :     virtual OGRFeatureDefn *GetLayerDefn() override
    1096             :     {
    1097         523 :         return OGRGeoPackageLayer::GetLayerDefn();
    1098             :     }
    1099             : 
    1100          95 :     virtual char *&GetAttrQueryString() override
    1101             :     {
    1102          95 :         return m_pszAttrQueryString;
    1103             :     }
    1104             : 
    1105         375 :     virtual OGRFeatureQuery *&GetFeatureQuery() override
    1106             :     {
    1107         375 :         return m_poAttrQuery;
    1108             :     }
    1109             : 
    1110         250 :     virtual OGRGeometry *&GetFilterGeom() override
    1111             :     {
    1112         250 :         return m_poFilterGeom;
    1113             :     }
    1114             : 
    1115          65 :     virtual int &GetIGeomFieldFilter() override
    1116             :     {
    1117          65 :         return m_iGeomFieldFilter;
    1118             :     }
    1119             : 
    1120         112 :     virtual OGRSpatialReference *GetSpatialRef() override
    1121             :     {
    1122         112 :         return OGRGeoPackageLayer::GetSpatialRef();
    1123             :     }
    1124             : 
    1125          45 :     virtual int InstallFilter(OGRGeometry *poGeomIn) override
    1126             :     {
    1127          45 :         return OGRGeoPackageLayer::InstallFilter(poGeomIn);
    1128             :     }
    1129             : 
    1130         177 :     virtual int HasReadFeature() override
    1131             :     {
    1132         177 :         return m_iNextShapeId > 0;
    1133             :     }
    1134             : 
    1135          88 :     virtual void BaseResetReading() override
    1136             :     {
    1137          88 :         OGRGeoPackageLayer::ResetReading();
    1138          88 :     }
    1139             : 
    1140        1946 :     virtual OGRFeature *BaseGetNextFeature() override
    1141             :     {
    1142        1946 :         return OGRGeoPackageLayer::GetNextFeature();
    1143             :     }
    1144             : 
    1145           0 :     virtual OGRErr BaseSetAttributeFilter(const char *pszQuery) override
    1146             :     {
    1147           0 :         return OGRGeoPackageLayer::SetAttributeFilter(pszQuery);
    1148             :     }
    1149             : 
    1150           1 :     virtual GIntBig BaseGetFeatureCount(int bForce) override
    1151             :     {
    1152           1 :         return OGRGeoPackageLayer::GetFeatureCount(bForce);
    1153             :     }
    1154             : 
    1155          24 :     virtual int BaseTestCapability(const char *pszCap) override
    1156             :     {
    1157          24 :         return OGRGeoPackageLayer::TestCapability(pszCap);
    1158             :     }
    1159             : 
    1160           2 :     virtual OGRErr BaseGetExtent(OGREnvelope *psExtent, int bForce) override
    1161             :     {
    1162           2 :         return OGRGeoPackageLayer::GetExtent(psExtent, bForce);
    1163             :     }
    1164             : 
    1165           0 :     virtual OGRErr BaseGetExtent(int iGeomField, OGREnvelope *psExtent,
    1166             :                                  int bForce) override
    1167             :     {
    1168           0 :         return OGRGeoPackageLayer::GetExtent(iGeomField, psExtent, bForce);
    1169             :     }
    1170             : 
    1171             :     bool
    1172          47 :     ValidateGeometryFieldIndexForSetSpatialFilter(int iGeomField,
    1173             :                                                   const OGRGeometry *poGeomIn,
    1174             :                                                   bool bIsSelectLayer) override
    1175             :     {
    1176             :         return OGRGeoPackageLayer::
    1177          47 :             ValidateGeometryFieldIndexForSetSpatialFilter(iGeomField, poGeomIn,
    1178          47 :                                                           bIsSelectLayer);
    1179             :     }
    1180             : };
    1181             : 
    1182             : #endif /* OGR_GEOPACKAGE_H_INCLUDED */

Generated by: LCOV version 1.14