LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/sqlite - ogrsqlitevfs.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 136 177 76.8 %
Date: 2025-01-18 12:42:00 Functions: 19 26 73.1 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  OpenGIS Simple Features Reference Implementation
       4             :  * Purpose:  Implements SQLite VFS
       5             :  * Author:   Even Rouault, <even dot rouault at spatialys.com>
       6             : 
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2011-2012, Even Rouault <even dot rouault at spatialys.com>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #include "cpl_port.h"
      14             : #include "ogr_sqlite.h"
      15             : 
      16             : #include <cstdio>
      17             : #include <cstdlib>
      18             : #include <cstring>
      19             : 
      20             : #include "cpl_conv.h"
      21             : #include "cpl_error.h"
      22             : #include "cpl_string.h"
      23             : #include "cpl_vsi.h"
      24             : #include "ogrsqlitevfs.h"
      25             : 
      26             : #ifdef DEBUG_IO
      27             : #define DEBUG_ONLY
      28             : #else
      29             : #define DEBUG_ONLY CPL_UNUSED
      30             : #endif
      31             : 
      32             : // #define DEBUG_IO 1
      33             : 
      34             : typedef struct
      35             : {
      36             :     char szVFSName[64];
      37             :     sqlite3_vfs *pDefaultVFS;
      38             :     pfnNotifyFileOpenedType pfn;
      39             :     void *pfnUserData;
      40             : } OGRSQLiteVFSAppDataStruct;
      41             : 
      42             : #define GET_UNDERLYING_VFS(pVFS)                                               \
      43             :     static_cast<OGRSQLiteVFSAppDataStruct *>(pVFS->pAppData)->pDefaultVFS
      44             : 
      45             : typedef struct
      46             : {
      47             :     const struct sqlite3_io_methods *pMethods;
      48             :     VSILFILE *fp;
      49             :     int bDeleteOnClose;
      50             :     char *pszFilename;
      51             : } OGRSQLiteFileStruct;
      52             : 
      53       21112 : static int OGRSQLiteIOClose(sqlite3_file *pFile)
      54             : {
      55       21112 :     OGRSQLiteFileStruct *pMyFile =
      56             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
      57             : #ifdef DEBUG_IO
      58             :     CPLDebug("SQLITE", "OGRSQLiteIOClose(%p (%s))", pMyFile->fp,
      59             :              pMyFile->pszFilename);
      60             : #endif
      61       21112 :     VSIFCloseL(pMyFile->fp);
      62       21112 :     if (pMyFile->bDeleteOnClose)
      63           0 :         VSIUnlink(pMyFile->pszFilename);
      64       21112 :     CPLFree(pMyFile->pszFilename);
      65       21112 :     return SQLITE_OK;
      66             : }
      67             : 
      68      231841 : static int OGRSQLiteIORead(sqlite3_file *pFile, void *pBuffer, int iAmt,
      69             :                            sqlite3_int64 iOfst)
      70             : {
      71      231841 :     OGRSQLiteFileStruct *pMyFile =
      72             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
      73      231841 :     VSIFSeekL(pMyFile->fp, static_cast<vsi_l_offset>(iOfst), SEEK_SET);
      74      231841 :     int nRead = static_cast<int>(VSIFReadL(pBuffer, 1, iAmt, pMyFile->fp));
      75             : #ifdef DEBUG_IO
      76             :     CPLDebug("SQLITE", "OGRSQLiteIORead(%p, %d, %d) = %d", pMyFile->fp, iAmt,
      77             :              static_cast<int>(iOfst), nRead);
      78             : #endif
      79      231841 :     if (nRead < iAmt)
      80             :     {
      81        4662 :         memset(static_cast<char *>(pBuffer) + nRead, 0, iAmt - nRead);
      82        4662 :         return SQLITE_IOERR_SHORT_READ;
      83             :     }
      84      227179 :     return SQLITE_OK;
      85             : }
      86             : 
      87      420471 : static int OGRSQLiteIOWrite(sqlite3_file *pFile, const void *pBuffer, int iAmt,
      88             :                             sqlite3_int64 iOfst)
      89             : {
      90      420471 :     OGRSQLiteFileStruct *pMyFile =
      91             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
      92      420471 :     VSIFSeekL(pMyFile->fp, static_cast<vsi_l_offset>(iOfst), SEEK_SET);
      93      420471 :     int nWritten = static_cast<int>(VSIFWriteL(pBuffer, 1, iAmt, pMyFile->fp));
      94             : #ifdef DEBUG_IO
      95             :     CPLDebug("SQLITE", "OGRSQLiteIOWrite(%p, %d, %d) = %d", pMyFile->fp, iAmt,
      96             :              static_cast<int>(iOfst), nWritten);
      97             : #endif
      98      420471 :     if (nWritten < iAmt)
      99             :     {
     100          24 :         return SQLITE_IOERR_WRITE;
     101             :     }
     102      420447 :     return SQLITE_OK;
     103             : }
     104             : 
     105          23 : static int OGRSQLiteIOTruncate(sqlite3_file *pFile, sqlite3_int64 size)
     106             : {
     107          23 :     OGRSQLiteFileStruct *pMyFile =
     108             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
     109             : #ifdef DEBUG_IO
     110             :     CPLDebug("SQLITE", "OGRSQLiteIOTruncate(%p, " CPL_FRMT_GIB ")", pMyFile->fp,
     111             :              size);
     112             : #endif
     113          23 :     int nRet = VSIFTruncateL(pMyFile->fp, size);
     114          23 :     return (nRet == 0) ? SQLITE_OK : SQLITE_IOERR_TRUNCATE;
     115             : }
     116             : 
     117        6680 : static int OGRSQLiteIOSync(DEBUG_ONLY sqlite3_file *pFile, DEBUG_ONLY int flags)
     118             : {
     119             : #ifdef DEBUG_IO
     120             :     OGRSQLiteFileStruct *pMyFile =
     121             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
     122             :     CPLDebug("SQLITE", "OGRSQLiteIOSync(%p, %d)", pMyFile->fp, flags);
     123             : #endif
     124        6680 :     return SQLITE_OK;
     125             : }
     126             : 
     127       93217 : static int OGRSQLiteIOFileSize(sqlite3_file *pFile, sqlite3_int64 *pSize)
     128             : {
     129       93217 :     OGRSQLiteFileStruct *pMyFile =
     130             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
     131       93217 :     vsi_l_offset nCurOffset = VSIFTellL(pMyFile->fp);
     132       93217 :     VSIFSeekL(pMyFile->fp, 0, SEEK_END);
     133       93217 :     *pSize = VSIFTellL(pMyFile->fp);
     134       93217 :     VSIFSeekL(pMyFile->fp, nCurOffset, SEEK_SET);
     135             : #ifdef DEBUG_IO
     136             :     CPLDebug("SQLITE", "OGRSQLiteIOFileSize(%p) = " CPL_FRMT_GIB, pMyFile->fp,
     137             :              *pSize);
     138             : #endif
     139       93217 :     return SQLITE_OK;
     140             : }
     141             : 
     142      150453 : static int OGRSQLiteIOLock(DEBUG_ONLY sqlite3_file *pFile, DEBUG_ONLY int flags)
     143             : {
     144             : #ifdef DEBUG_IO
     145             :     OGRSQLiteFileStruct *pMyFile =
     146             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
     147             :     CPLDebug("SQLITE", "OGRSQLiteIOLock(%p)", pMyFile->fp);
     148             : #endif
     149      150453 :     return SQLITE_OK;
     150             : }
     151             : 
     152      124569 : static int OGRSQLiteIOUnlock(DEBUG_ONLY sqlite3_file *pFile,
     153             :                              DEBUG_ONLY int flags)
     154             : {
     155             : #ifdef DEBUG_IO
     156             :     OGRSQLiteFileStruct *pMyFile =
     157             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
     158             :     CPLDebug("SQLITE", "OGRSQLiteIOUnlock(%p)", pMyFile->fp);
     159             : #endif
     160      124569 :     return SQLITE_OK;
     161             : }
     162             : 
     163           6 : static int OGRSQLiteIOCheckReservedLock(DEBUG_ONLY sqlite3_file *pFile,
     164             :                                         DEBUG_ONLY int *pResOut)
     165             : {
     166             : #ifdef DEBUG_IO
     167             :     OGRSQLiteFileStruct *pMyFile =
     168             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
     169             :     CPLDebug("SQLITE", "OGRSQLiteIOCheckReservedLock(%p)", pMyFile->fp);
     170             : #endif
     171           6 :     *pResOut = 0;
     172           6 :     return SQLITE_OK;
     173             : }
     174             : 
     175      106273 : static int OGRSQLiteIOFileControl(DEBUG_ONLY sqlite3_file *pFile,
     176             :                                   DEBUG_ONLY int op, DEBUG_ONLY void *pArg)
     177             : {
     178             : #ifdef DEBUG_IO
     179             :     OGRSQLiteFileStruct *pMyFile =
     180             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
     181             :     CPLDebug("SQLITE", "OGRSQLiteIOFileControl(%p, %d)", pMyFile->fp, op);
     182             : #endif
     183      106273 :     return SQLITE_NOTFOUND;
     184             : }
     185             : 
     186        4840 : static int OGRSQLiteIOSectorSize(DEBUG_ONLY sqlite3_file *pFile)
     187             : {
     188             : #ifdef DEBUG_IO
     189             :     OGRSQLiteFileStruct *pMyFile =
     190             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
     191             :     CPLDebug("SQLITE", "OGRSQLiteIOSectorSize(%p)", pMyFile->fp);
     192             : #endif
     193        4840 :     return 0;
     194             : }
     195             : 
     196      107999 : static int OGRSQLiteIODeviceCharacteristics(DEBUG_ONLY sqlite3_file *pFile)
     197             : {
     198             : #ifdef DEBUG_IO
     199             :     OGRSQLiteFileStruct *pMyFile =
     200             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
     201             :     CPLDebug("SQLITE", "OGRSQLiteIODeviceCharacteristics(%p)", pMyFile->fp);
     202             : #endif
     203      107999 :     return 0;
     204             : }
     205             : 
     206             : static const sqlite3_io_methods OGRSQLiteIOMethods = {
     207             :     1,
     208             :     OGRSQLiteIOClose,
     209             :     OGRSQLiteIORead,
     210             :     OGRSQLiteIOWrite,
     211             :     OGRSQLiteIOTruncate,
     212             :     OGRSQLiteIOSync,
     213             :     OGRSQLiteIOFileSize,
     214             :     OGRSQLiteIOLock,
     215             :     OGRSQLiteIOUnlock,
     216             :     OGRSQLiteIOCheckReservedLock,
     217             :     OGRSQLiteIOFileControl,
     218             :     OGRSQLiteIOSectorSize,
     219             :     OGRSQLiteIODeviceCharacteristics,
     220             :     nullptr,  // xShmMap
     221             :     nullptr,  // xShmLock
     222             :     nullptr,  // xShmBarrier
     223             :     nullptr,  // xShmUnmap
     224             :     nullptr,  // xFetch
     225             :     nullptr,  // xUnfetch
     226             : };
     227             : 
     228       21117 : static int OGRSQLiteVFSOpen(sqlite3_vfs *pVFS, const char *zNameIn,
     229             :                             sqlite3_file *pFile, int flags, int *pOutFlags)
     230             : {
     231             : #ifdef DEBUG_IO
     232             :     CPLDebug("SQLITE", "OGRSQLiteVFSOpen(%s, %d)", zNameIn ? zNameIn : "(null)",
     233             :              flags);
     234             : #endif
     235             : 
     236       21117 :     OGRSQLiteVFSAppDataStruct *pAppData =
     237             :         static_cast<OGRSQLiteVFSAppDataStruct *>(pVFS->pAppData);
     238             : 
     239             :     const std::string osName(
     240       42234 :         zNameIn ? zNameIn : VSIMemGenerateHiddenFilename("sqlitevfs"));
     241             : 
     242       21117 :     OGRSQLiteFileStruct *pMyFile =
     243             :         reinterpret_cast<OGRSQLiteFileStruct *>(pFile);
     244       21117 :     pMyFile->pMethods = nullptr;
     245       21117 :     pMyFile->bDeleteOnClose = FALSE;
     246       21117 :     pMyFile->pszFilename = nullptr;
     247       21117 :     if (flags & SQLITE_OPEN_READONLY)
     248         866 :         pMyFile->fp = VSIFOpenL(osName.c_str(), "rb");
     249       20251 :     else if (flags & SQLITE_OPEN_CREATE)
     250             :     {
     251             :         VSIStatBufL sStatBufL;
     252       19467 :         if (VSIStatExL(osName.c_str(), &sStatBufL, VSI_STAT_EXISTS_FLAG) == 0)
     253          56 :             pMyFile->fp = VSIFOpenL(osName.c_str(), "rb+");
     254             :         else
     255       19411 :             pMyFile->fp = VSIFOpenL(osName.c_str(), "wb+");
     256             :     }
     257         784 :     else if (flags & SQLITE_OPEN_READWRITE)
     258         784 :         pMyFile->fp = VSIFOpenL(osName.c_str(), "rb+");
     259             :     else
     260           0 :         pMyFile->fp = nullptr;
     261             : 
     262       21117 :     if (pMyFile->fp == nullptr)
     263           4 :         return SQLITE_CANTOPEN;
     264             : 
     265             : #ifdef DEBUG_IO
     266             :     CPLDebug("SQLITE", "OGRSQLiteVFSOpen() = %p", pMyFile->fp);
     267             : #endif
     268             : 
     269       21113 :     pfnNotifyFileOpenedType pfn = pAppData->pfn;
     270       21113 :     if (pfn)
     271             :     {
     272       20686 :         pfn(pAppData->pfnUserData, osName.c_str(), pMyFile->fp);
     273             :     }
     274             : 
     275       21113 :     pMyFile->pMethods = &OGRSQLiteIOMethods;
     276       21113 :     pMyFile->bDeleteOnClose = (flags & SQLITE_OPEN_DELETEONCLOSE);
     277       21113 :     pMyFile->pszFilename = CPLStrdup(osName.c_str());
     278             : 
     279       21113 :     if (pOutFlags != nullptr)
     280        2838 :         *pOutFlags = flags;
     281             : 
     282       21113 :     return SQLITE_OK;
     283             : }
     284             : 
     285       18275 : static int OGRSQLiteVFSDelete(DEBUG_ONLY sqlite3_vfs *pVFS, const char *zName,
     286             :                               int DEBUG_ONLY syncDir)
     287             : {
     288             : #ifdef DEBUG_IO
     289             :     CPLDebug("SQLITE", "OGRSQLiteVFSDelete(%s)", zName);
     290             : #endif
     291       18275 :     VSIUnlink(zName);
     292       18275 :     return SQLITE_OK;
     293             : }
     294             : 
     295      186072 : static int OGRSQLiteVFSAccess(DEBUG_ONLY sqlite3_vfs *pVFS, const char *zName,
     296             :                               int flags, int *pResOut)
     297             : {
     298             : #ifdef DEBUG_IO
     299             :     CPLDebug("SQLITE", "OGRSQLiteVFSAccess(%s, %d)", zName, flags);
     300             : #endif
     301             :     VSIStatBufL sStatBufL;
     302             :     int nRet;  // TODO(schwehr): Cleanup nRet and pResOut.  bools?
     303      186072 :     if (flags == SQLITE_ACCESS_EXISTS)
     304             :     {
     305             :         /* Do not try to check the presence of a journal or a wal on /vsicurl !
     306             :          */
     307      186072 :         if ((STARTS_WITH(zName, "/vsicurl/") ||
     308      185904 :              STARTS_WITH(zName, "/vsitar/") ||
     309      185904 :              STARTS_WITH(zName, "/vsizip/")) &&
     310         300 :             ((strlen(zName) > strlen("-journal") &&
     311         300 :               strcmp(zName + strlen(zName) - strlen("-journal"), "-journal") ==
     312         150 :                   0) ||
     313         150 :              (strlen(zName) > strlen("-wal") &&
     314         150 :               strcmp(zName + strlen(zName) - strlen("-wal"), "-wal") == 0)))
     315             :         {
     316         300 :             nRet = -1;
     317             :         }
     318             :         else
     319             :         {
     320      185772 :             nRet = VSIStatExL(zName, &sStatBufL, VSI_STAT_EXISTS_FLAG);
     321             :         }
     322             :     }
     323           0 :     else if (flags == SQLITE_ACCESS_READ)
     324             :     {
     325           0 :         VSILFILE *fp = VSIFOpenL(zName, "rb");
     326           0 :         nRet = fp ? 0 : -1;
     327           0 :         if (fp)
     328           0 :             VSIFCloseL(fp);
     329             :     }
     330           0 :     else if (flags == SQLITE_ACCESS_READWRITE)
     331             :     {
     332           0 :         VSILFILE *fp = VSIFOpenL(zName, "rb+");
     333           0 :         nRet = fp ? 0 : -1;
     334           0 :         if (fp)
     335           0 :             VSIFCloseL(fp);
     336             :     }
     337             :     else
     338             :     {
     339           0 :         nRet = -1;
     340             :     }
     341      186072 :     *pResOut = (nRet == 0);
     342      186072 :     return SQLITE_OK;
     343             : }
     344             : 
     345        2830 : static int OGRSQLiteVFSFullPathname(sqlite3_vfs *pVFS, const char *zName,
     346             :                                     int nOut, char *zOut)
     347             : {
     348        2830 :     sqlite3_vfs *pUnderlyingVFS = GET_UNDERLYING_VFS(pVFS);
     349             : #ifdef DEBUG_IO
     350             :     CPLDebug("SQLITE", "OGRSQLiteVFSFullPathname(%s)", zName);
     351             : #endif
     352        2830 :     if (zName[0] == '/')
     353             :     {
     354        2828 :         if (static_cast<int>(strlen(zName)) >= nOut)
     355             :         {
     356             :             // The +8 comes from the fact that sqlite3 does this check as
     357             :             // it needs to be able to append .journal to the filename
     358           0 :             CPLError(CE_Failure, CPLE_AppDefined,
     359             :                      "Maximum pathname length reserved for SQLite3 VFS "
     360             :                      "isn't large enough. Try raising "
     361             :                      "OGR_SQLITE_VFS_MAXPATHNAME to at least %d",
     362           0 :                      static_cast<int>(strlen(zName)) + 8);
     363           0 :             return SQLITE_CANTOPEN;
     364             :         }
     365        2828 :         strncpy(zOut, zName, nOut);
     366        2828 :         zOut[nOut - 1] = '\0';
     367        2828 :         return SQLITE_OK;
     368             :     }
     369           2 :     return pUnderlyingVFS->xFullPathname(pUnderlyingVFS, zName, nOut, zOut);
     370             : }
     371             : 
     372           0 : static void *OGRSQLiteVFSDlOpen(sqlite3_vfs *pVFS, const char *zFilename)
     373             : {
     374           0 :     sqlite3_vfs *pUnderlyingVFS = GET_UNDERLYING_VFS(pVFS);
     375             :     // CPLDebug("SQLITE", "OGRSQLiteVFSDlOpen(%s)", zFilename);
     376           0 :     return pUnderlyingVFS->xDlOpen(pUnderlyingVFS, zFilename);
     377             : }
     378             : 
     379           0 : static void OGRSQLiteVFSDlError(sqlite3_vfs *pVFS, int nByte, char *zErrMsg)
     380             : {
     381           0 :     sqlite3_vfs *pUnderlyingVFS = GET_UNDERLYING_VFS(pVFS);
     382             :     // CPLDebug("SQLITE", "OGRSQLiteVFSDlError()");
     383           0 :     pUnderlyingVFS->xDlError(pUnderlyingVFS, nByte, zErrMsg);
     384           0 : }
     385             : 
     386           0 : static void (*OGRSQLiteVFSDlSym(sqlite3_vfs *pVFS, void *pHandle,
     387             :                                 const char *zSymbol))(void)
     388             : {
     389           0 :     sqlite3_vfs *pUnderlyingVFS = GET_UNDERLYING_VFS(pVFS);
     390             :     // CPLDebug("SQLITE", "OGRSQLiteVFSDlSym(%s)", zSymbol);
     391           0 :     return pUnderlyingVFS->xDlSym(pUnderlyingVFS, pHandle, zSymbol);
     392             : }
     393             : 
     394           0 : static void OGRSQLiteVFSDlClose(sqlite3_vfs *pVFS, void *pHandle)
     395             : {
     396           0 :     sqlite3_vfs *pUnderlyingVFS = GET_UNDERLYING_VFS(pVFS);
     397             :     // CPLDebug("SQLITE", "OGRSQLiteVFSDlClose(%p)", pHandle);
     398           0 :     pUnderlyingVFS->xDlClose(pUnderlyingVFS, pHandle);
     399           0 : }
     400             : 
     401           0 : static int OGRSQLiteVFSRandomness(sqlite3_vfs *pVFS, int nByte, char *zOut)
     402             : {
     403           0 :     sqlite3_vfs *pUnderlyingVFS = GET_UNDERLYING_VFS(pVFS);
     404             :     // CPLDebug("SQLITE", "OGRSQLiteVFSRandomness()");
     405           0 :     return pUnderlyingVFS->xRandomness(pUnderlyingVFS, nByte, zOut);
     406             : }
     407             : 
     408           0 : static int OGRSQLiteVFSSleep(sqlite3_vfs *pVFS, int microseconds)
     409             : {
     410           0 :     sqlite3_vfs *pUnderlyingVFS = GET_UNDERLYING_VFS(pVFS);
     411             :     // CPLDebug("SQLITE", "OGRSQLiteVFSSleep()");
     412           0 :     return pUnderlyingVFS->xSleep(pUnderlyingVFS, microseconds);
     413             : }
     414             : 
     415             : // Derived for sqlite3.c implementation of unixCurrentTime64 and
     416             : // winCurrentTime64
     417             : #ifdef _WIN32
     418             : #include <windows.h>
     419             : 
     420             : static int OGRSQLiteVFSCurrentTimeInt64(sqlite3_vfs * /*pVFS*/,
     421             :                                         sqlite3_int64 *piNow)
     422             : {
     423             :     FILETIME ft;
     424             :     constexpr sqlite3_int64 winFiletimeEpoch =
     425             :         23058135 * static_cast<sqlite3_int64>(8640000);
     426             :     constexpr sqlite3_int64 max32BitValue =
     427             :         static_cast<sqlite3_int64>(2000000000) +
     428             :         static_cast<sqlite3_int64>(2000000000) +
     429             :         static_cast<sqlite3_int64>(294967296);
     430             : 
     431             : #if defined(_WIN32_WCE)
     432             :     SYSTEMTIME time;
     433             :     GetSystemTime(&time);
     434             :     /* if SystemTimeToFileTime() fails, it returns zero. */
     435             :     if (!SystemTimeToFileTime(&time, &ft))
     436             :     {
     437             :         return SQLITE_ERROR;
     438             :     }
     439             : #else
     440             :     GetSystemTimeAsFileTime(&ft);
     441             : #endif
     442             :     *piNow = winFiletimeEpoch +
     443             :              ((static_cast<sqlite3_int64>(ft.dwHighDateTime) * max32BitValue) +
     444             :               static_cast<sqlite3_int64>(ft.dwLowDateTime)) /
     445             :                  static_cast<sqlite3_int64>(10000);
     446             :     return SQLITE_OK;
     447             : }
     448             : #else
     449             : #include <sys/time.h>
     450             : 
     451        4195 : static int OGRSQLiteVFSCurrentTimeInt64(sqlite3_vfs * /*pVFS*/,
     452             :                                         sqlite3_int64 *piNow)
     453             : {
     454             :     struct timeval sNow;
     455        4195 :     constexpr sqlite3_int64 unixEpoch =
     456             :         24405875 * static_cast<sqlite3_int64>(8640000);
     457        4195 :     (void)gettimeofday(&sNow, nullptr); /* Cannot fail given valid arguments */
     458        4195 :     *piNow = unixEpoch + 1000 * static_cast<sqlite3_int64>(sNow.tv_sec) +
     459        4195 :              sNow.tv_usec / 1000;
     460             : 
     461        4195 :     return SQLITE_OK;
     462             : }
     463             : #endif
     464             : 
     465           0 : static int OGRSQLiteVFSCurrentTime(sqlite3_vfs * /*pVFS*/, double *p1)
     466             : {
     467           0 :     sqlite3_int64 i = 0;
     468           0 :     int rc = OGRSQLiteVFSCurrentTimeInt64(nullptr, &i);
     469           0 :     *p1 = i / 86400000.0;
     470           0 :     return rc;
     471             : }
     472             : 
     473          28 : static int OGRSQLiteVFSGetLastError(sqlite3_vfs *pVFS, int p1, char *p2)
     474             : {
     475          28 :     sqlite3_vfs *pUnderlyingVFS = GET_UNDERLYING_VFS(pVFS);
     476             :     // CPLDebug("SQLITE", "OGRSQLiteVFSGetLastError()");
     477          28 :     return pUnderlyingVFS->xGetLastError(pUnderlyingVFS, p1, p2);
     478             : }
     479             : 
     480        2705 : sqlite3_vfs *OGRSQLiteCreateVFS(pfnNotifyFileOpenedType pfn, void *pfnUserData)
     481             : {
     482        2705 :     sqlite3_vfs *pDefaultVFS = sqlite3_vfs_find(nullptr);
     483             :     sqlite3_vfs *pMyVFS =
     484        2705 :         static_cast<sqlite3_vfs *>(CPLCalloc(1, sizeof(sqlite3_vfs)));
     485             : 
     486             :     OGRSQLiteVFSAppDataStruct *pVFSAppData =
     487             :         static_cast<OGRSQLiteVFSAppDataStruct *>(
     488        2705 :             CPLCalloc(1, sizeof(OGRSQLiteVFSAppDataStruct)));
     489             :     char szPtr[32];
     490        2705 :     snprintf(szPtr, sizeof(szPtr), "%p", pVFSAppData);
     491        2705 :     snprintf(pVFSAppData->szVFSName, sizeof(pVFSAppData->szVFSName),
     492             :              "OGRSQLITEVFS_%s", szPtr);
     493        2705 :     pVFSAppData->pDefaultVFS = pDefaultVFS;
     494        2705 :     pVFSAppData->pfn = pfn;
     495        2705 :     pVFSAppData->pfnUserData = pfnUserData;
     496             : 
     497        2705 :     pMyVFS->iVersion = 2;
     498        2705 :     pMyVFS->szOsFile = sizeof(OGRSQLiteFileStruct);
     499             :     // must be large enough to hold potentially very long names like
     500             :     // /vsicurl/.... with AWS S3 security tokens
     501        2705 :     pMyVFS->mxPathname =
     502        2705 :         atoi(CPLGetConfigOption("OGR_SQLITE_VFS_MAXPATHNAME", "2048"));
     503        2705 :     pMyVFS->zName = pVFSAppData->szVFSName;
     504        2705 :     pMyVFS->pAppData = pVFSAppData;
     505        2705 :     pMyVFS->xOpen = OGRSQLiteVFSOpen;
     506        2705 :     pMyVFS->xDelete = OGRSQLiteVFSDelete;
     507        2705 :     pMyVFS->xAccess = OGRSQLiteVFSAccess;
     508        2705 :     pMyVFS->xFullPathname = OGRSQLiteVFSFullPathname;
     509        2705 :     pMyVFS->xDlOpen = OGRSQLiteVFSDlOpen;
     510        2705 :     pMyVFS->xDlError = OGRSQLiteVFSDlError;
     511        2705 :     pMyVFS->xDlSym = OGRSQLiteVFSDlSym;
     512        2705 :     pMyVFS->xDlClose = OGRSQLiteVFSDlClose;
     513        2705 :     pMyVFS->xRandomness = OGRSQLiteVFSRandomness;
     514        2705 :     pMyVFS->xSleep = OGRSQLiteVFSSleep;
     515        2705 :     pMyVFS->xCurrentTime = OGRSQLiteVFSCurrentTime;
     516        2705 :     pMyVFS->xGetLastError = OGRSQLiteVFSGetLastError;
     517        2705 :     pMyVFS->xCurrentTimeInt64 = OGRSQLiteVFSCurrentTimeInt64;
     518             : 
     519        2705 :     return pMyVFS;
     520             : }

Generated by: LCOV version 1.14