LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/openfilegdb - filegdbtable_write_fields.cpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 434 509 85.3 %
Date: 2025-01-18 12:42:00 Functions: 7 7 100.0 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  OpenGIS Simple Features Reference Implementation
       4             :  * Purpose:  Implements management of FileGDB field write support
       5             :  * Author:   Even Rouault, <even dot rouault at spatialys.com>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2022, Even Rouault <even dot rouault at spatialys.com>
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #include "cpl_port.h"
      14             : 
      15             : #include "filegdbtable.h"
      16             : #include "filegdbtable_priv.h"
      17             : 
      18             : #include <algorithm>
      19             : #include <limits>
      20             : 
      21             : #include "cpl_string.h"
      22             : #include "cpl_time.h"
      23             : 
      24             : #include "ogr_core.h"
      25             : #include "ogr_api.h"
      26             : 
      27             : namespace OpenFileGDB
      28             : {
      29             : 
      30             : /************************************************************************/
      31             : /*                            CreateField()                             */
      32             : /************************************************************************/
      33             : 
      34       13196 : bool FileGDBTable::CreateField(std::unique_ptr<FileGDBField> &&psField)
      35             : {
      36       13196 :     if (!m_bUpdate)
      37           0 :         return false;
      38             : 
      39             :     // Encoded on a uint16_t
      40       13196 :     if (m_apoFields.size() == 65535)
      41             :     {
      42           0 :         CPLError(CE_Failure, CPLE_NotSupported, "Too many fields");
      43           0 :         return false;
      44             :     }
      45             : 
      46       13196 :     if (psField->GetType() == FGFT_RASTER)
      47             :     {
      48           0 :         CPLError(CE_Failure, CPLE_NotSupported, "Unhandled field type");
      49           0 :         return false;
      50             :     }
      51             : 
      52       13196 :     if (GetFieldIdx(psField->GetName()) >= 0)
      53             :     {
      54           2 :         CPLError(CE_Failure, CPLE_NotSupported, "Field %s already exists",
      55           2 :                  psField->GetName().c_str());
      56           2 :         return false;
      57             :     }
      58             : 
      59       13194 :     if (psField->GetType() == FGFT_GEOMETRY)
      60             :     {
      61         424 :         if (m_iGeomField >= 0)
      62             :         {
      63           0 :             CPLError(CE_Failure, CPLE_NotSupported,
      64             :                      "Only one geometry field supported");
      65           0 :             return false;
      66             :         }
      67         424 :         m_iGeomField = static_cast<int>(m_apoFields.size());
      68             :         m_adfSpatialIndexGridResolution =
      69             :             cpl::down_cast<const FileGDBGeomField *>(psField.get())
      70         424 :                 ->GetSpatialIndexGridResolution();
      71             :     }
      72             : 
      73       13194 :     if (psField->GetType() == FGFT_OBJECTID)
      74             :     {
      75        1654 :         if (m_iObjectIdField >= 0)
      76             :         {
      77           0 :             CPLError(CE_Failure, CPLE_NotSupported,
      78             :                      "Only one ObjectId field supported");
      79           0 :             return false;
      80             :         }
      81        1654 :         m_iObjectIdField = static_cast<int>(m_apoFields.size());
      82             :     }
      83             : 
      84       13194 :     bool bRewriteTable = false;
      85       13194 :     if (m_nTotalRecordCount != 0)
      86             :     {
      87         134 :         const bool bHasDefault = !OGR_RawField_IsNull(psField->GetDefault()) &&
      88          67 :                                  !OGR_RawField_IsUnset(psField->GetDefault());
      89          67 :         if (psField->GetType() == FGFT_GEOMETRY)
      90             :         {
      91           0 :             CPLError(CE_Failure, CPLE_NotSupported,
      92             :                      "Cannot add a geometry field to a non-empty table");
      93           0 :             return false;
      94             :         }
      95          67 :         else if (psField->GetType() == FGFT_OBJECTID)
      96             :         {
      97             :             // nothing to do but rewrite the feature definition
      98             :         }
      99          67 :         else if ((m_nCountNullableFields % 8) != 0 && psField->IsNullable())
     100             :         {
     101             :             // Adding a nullable field to a feature definition that has already
     102             :             // nullable fields, with the last bitmap byte not completely filled.
     103             :             // We just need to rewrite the feature definition, not the features.
     104             :         }
     105          45 :         else if (!psField->IsNullable() && !bHasDefault)
     106             :         {
     107           1 :             CPLError(CE_Failure, CPLE_NotSupported,
     108             :                      "Cannot add non-nullable field without default value to "
     109             :                      "a non-empty table");
     110           1 :             return false;
     111             :         }
     112             :         else
     113             :         {
     114          44 :             bRewriteTable = true;
     115             :         }
     116             :     }
     117             : 
     118       13193 :     m_nCurRow = -1;
     119       13193 :     m_bDirtyFieldDescriptors = true;
     120       13193 :     const bool bIsNullable = psField->IsNullable();
     121       13193 :     if (bIsNullable)
     122             :     {
     123        7592 :         m_nCountNullableFields++;
     124        7592 :         m_nNullableFieldsSizeInBytes =
     125        7592 :             BIT_ARRAY_SIZE_IN_BYTES(m_nCountNullableFields);
     126             :     }
     127       13193 :     psField->SetParent(this);
     128       13193 :     m_apoFields.emplace_back(std::move(psField));
     129             : 
     130       13193 :     if (bRewriteTable && !RewriteTableToAddLastAddedField())
     131             :     {
     132           6 :         if (bIsNullable)
     133             :         {
     134           0 :             m_nCountNullableFields--;
     135           0 :             m_nNullableFieldsSizeInBytes =
     136           0 :                 BIT_ARRAY_SIZE_IN_BYTES(m_nCountNullableFields);
     137             :         }
     138           6 :         m_apoFields.pop_back();
     139           6 :         m_bDirtyFieldDescriptors = true;
     140           6 :         return false;
     141             :     }
     142             : 
     143       13187 :     return true;
     144             : }
     145             : 
     146             : /************************************************************************/
     147             : /*                  RewriteTableToAddLastAddedField()                   */
     148             : /************************************************************************/
     149             : 
     150          44 : bool FileGDBTable::RewriteTableToAddLastAddedField()
     151             : {
     152          44 :     int nOldCountNullableFields = m_nCountNullableFields;
     153          44 :     if (m_apoFields.back()->IsNullable())
     154             :     {
     155           2 :         nOldCountNullableFields--;
     156             :     }
     157          44 :     const unsigned nOldNullableFieldsSizeInBytes =
     158          44 :         BIT_ARRAY_SIZE_IN_BYTES(nOldCountNullableFields);
     159          44 :     int nExtraBytes = 0;
     160          44 :     if (nOldNullableFieldsSizeInBytes != m_nNullableFieldsSizeInBytes)
     161           2 :         nExtraBytes++;
     162          88 :     std::vector<GByte> abyDefaultVal;
     163             : 
     164          44 :     const auto &psLastField = m_apoFields.back();
     165          44 :     if (!psLastField->IsNullable())
     166             :     {
     167             :         const bool bHasDefault =
     168          84 :             !OGR_RawField_IsNull(psLastField->GetDefault()) &&
     169          42 :             !OGR_RawField_IsUnset(psLastField->GetDefault());
     170          42 :         CPL_IGNORE_RET_VAL(bHasDefault);
     171          42 :         CPLAssert(bHasDefault);
     172          42 :         if (psLastField->GetType() == FGFT_STRING)
     173             :         {
     174          12 :             const std::string osDefaultVal(psLastField->GetDefault()->String);
     175          12 :             WriteVarUInt(abyDefaultVal, osDefaultVal.size());
     176             :             abyDefaultVal.insert(
     177          12 :                 abyDefaultVal.end(),
     178          12 :                 reinterpret_cast<const GByte *>(osDefaultVal.c_str()),
     179          12 :                 reinterpret_cast<const GByte *>(osDefaultVal.c_str()) +
     180          24 :                     osDefaultVal.size());
     181             :         }
     182          30 :         else if (psLastField->GetType() == FGFT_INT16)
     183             :         {
     184           6 :             WriteInt16(abyDefaultVal, static_cast<int16_t>(
     185           6 :                                           psLastField->GetDefault()->Integer));
     186             :         }
     187          24 :         else if (psLastField->GetType() == FGFT_INT32)
     188             :         {
     189           6 :             WriteInt32(abyDefaultVal, psLastField->GetDefault()->Integer);
     190             :         }
     191          18 :         else if (psLastField->GetType() == FGFT_INT64)
     192             :         {
     193           0 :             WriteInt64(abyDefaultVal, psLastField->GetDefault()->Integer64);
     194             :         }
     195          18 :         else if (psLastField->GetType() == FGFT_FLOAT32)
     196             :         {
     197           6 :             WriteFloat32(abyDefaultVal,
     198           6 :                          static_cast<float>(psLastField->GetDefault()->Real));
     199             :         }
     200          12 :         else if (psLastField->GetType() == FGFT_FLOAT64)
     201             :         {
     202           6 :             WriteFloat64(abyDefaultVal, psLastField->GetDefault()->Real);
     203             :         }
     204           6 :         else if (psLastField->GetType() == FGFT_DATETIME ||
     205           0 :                  psLastField->GetType() == FGFT_DATE)
     206             :         {
     207           6 :             WriteFloat64(abyDefaultVal, FileGDBOGRDateToDoubleDate(
     208             :                                             psLastField->GetDefault(),
     209             :                                             /* bConvertToUTC = */ true,
     210           6 :                                             psLastField->IsHighPrecision()));
     211             :         }
     212           0 :         else if (psLastField->GetType() == FGFT_TIME)
     213             :         {
     214           0 :             WriteFloat64(abyDefaultVal,
     215             :                          FileGDBOGRTimeToDoubleTime(psLastField->GetDefault()));
     216             :         }
     217           0 :         else if (psLastField->GetType() == FGFT_DATETIME_WITH_OFFSET)
     218             :         {
     219           0 :             const auto psDefault = psLastField->GetDefault();
     220           0 :             WriteFloat64(abyDefaultVal,
     221             :                          FileGDBOGRDateToDoubleDate(
     222             :                              psDefault, /* bConvertToUTC = */ false,
     223             :                              /* bIsHighPrecision= */ true));
     224           0 :             if (psDefault->Date.TZFlag > 1)
     225             :             {
     226           0 :                 WriteInt16(
     227             :                     abyDefaultVal,
     228           0 :                     static_cast<int16_t>((psDefault->Date.TZFlag - 100) * 15));
     229             :             }
     230             :             else
     231             :             {
     232           0 :                 WriteInt16(abyDefaultVal, 0);
     233             :             }
     234             :         }
     235          42 :         nExtraBytes += static_cast<int>(abyDefaultVal.size());
     236             :     }
     237          44 :     CPLAssert(nExtraBytes != 0);
     238             : 
     239          88 :     std::vector<GByte> abyBufferOffsets;
     240          44 :     abyBufferOffsets.resize(1024 * m_nTablxOffsetSize);
     241             : 
     242          88 :     WholeFileRewriter oWholeFileRewriter(*this);
     243          44 :     if (!oWholeFileRewriter.Begin())
     244           0 :         return false;
     245             : 
     246          44 :     if (CPLTestBool(CPLGetConfigOption(
     247             :             "OPENFILEGDB_SIMUL_ERROR_IN_RewriteTableToAddLastAddedField",
     248             :             "FALSE")))
     249             :     {
     250           6 :         return false;
     251             :     }
     252             : 
     253          38 :     uint32_t nRowBufferMaxSize = 0;
     254          38 :     m_nCurRow = -1;
     255             : 
     256             :     // Rewrite all features
     257          76 :     for (uint32_t iPage = 0; iPage < m_n1024BlocksPresent; ++iPage)
     258             :     {
     259          38 :         const vsi_l_offset nOffsetInTableX =
     260          38 :             16 + m_nTablxOffsetSize * static_cast<vsi_l_offset>(iPage) * 1024;
     261          38 :         VSIFSeekL(oWholeFileRewriter.m_fpOldGdbtablx, nOffsetInTableX,
     262             :                   SEEK_SET);
     263          38 :         if (VSIFReadL(abyBufferOffsets.data(), m_nTablxOffsetSize * 1024, 1,
     264          38 :                       oWholeFileRewriter.m_fpOldGdbtablx) != 1)
     265           0 :             return false;
     266             : 
     267          38 :         GByte *pabyBufferOffsets = abyBufferOffsets.data();
     268       38950 :         for (int i = 0; i < 1024; i++, pabyBufferOffsets += m_nTablxOffsetSize)
     269             :         {
     270       38912 :             const uint64_t nOffset = ReadFeatureOffset(pabyBufferOffsets);
     271       38912 :             if (nOffset != 0)
     272             :             {
     273             :                 // Read feature size
     274          77 :                 VSIFSeekL(oWholeFileRewriter.m_fpOldGdbtable, nOffset,
     275             :                           SEEK_SET);
     276          77 :                 uint32_t nFeatureSize = 0;
     277          77 :                 if (!ReadUInt32(oWholeFileRewriter.m_fpOldGdbtable,
     278             :                                 nFeatureSize))
     279           0 :                     return false;
     280             : 
     281             :                 // Read feature data
     282          77 :                 if (nFeatureSize > m_abyBuffer.size())
     283             :                 {
     284             :                     try
     285             :                     {
     286          18 :                         m_abyBuffer.resize(nFeatureSize);
     287             :                     }
     288           0 :                     catch (const std::exception &e)
     289             :                     {
     290           0 :                         CPLError(CE_Failure, CPLE_OutOfMemory, "%s", e.what());
     291           0 :                         return false;
     292             :                     }
     293             :                 }
     294          77 :                 if (VSIFReadL(m_abyBuffer.data(), nFeatureSize, 1,
     295          77 :                               oWholeFileRewriter.m_fpOldGdbtable) != 1)
     296           0 :                     return false;
     297             : 
     298             :                 // Update offset of updated feature
     299          77 :                 WriteFeatureOffset(m_nFileSize, pabyBufferOffsets);
     300             : 
     301             :                 // Write updated feature size
     302          77 :                 const uint32_t nNewFeatureSize = nFeatureSize + nExtraBytes;
     303          77 :                 if (!WriteUInt32(oWholeFileRewriter.m_fpTable, nNewFeatureSize))
     304           0 :                     return false;
     305             : 
     306             :                 // Write updated feature data
     307          77 :                 if (nOldNullableFieldsSizeInBytes != 0)
     308             :                 {
     309          77 :                     if (VSIFWriteL(m_abyBuffer.data(),
     310             :                                    nOldNullableFieldsSizeInBytes, 1,
     311          77 :                                    oWholeFileRewriter.m_fpTable) != 1)
     312           0 :                         return false;
     313             :                 }
     314          77 :                 if (nOldNullableFieldsSizeInBytes !=
     315          77 :                     m_nNullableFieldsSizeInBytes)
     316             :                 {
     317           5 :                     CPLAssert(psLastField->IsNullable());
     318           5 :                     const GByte byNewNullableFieldByte = 0xFF;
     319           5 :                     if (VSIFWriteL(&byNewNullableFieldByte, 1, 1,
     320           5 :                                    oWholeFileRewriter.m_fpTable) != 1)
     321           0 :                         return false;
     322             :                 }
     323          77 :                 if (nFeatureSize > nOldNullableFieldsSizeInBytes)
     324             :                 {
     325         231 :                     if (VSIFWriteL(m_abyBuffer.data() +
     326          77 :                                        nOldNullableFieldsSizeInBytes,
     327          77 :                                    nFeatureSize - nOldNullableFieldsSizeInBytes,
     328          77 :                                    1, oWholeFileRewriter.m_fpTable) != 1)
     329           0 :                         return false;
     330             :                 }
     331          77 :                 if (!abyDefaultVal.empty())
     332             :                 {
     333          72 :                     if (VSIFWriteL(abyDefaultVal.data(), abyDefaultVal.size(),
     334          72 :                                    1, oWholeFileRewriter.m_fpTable) != 1)
     335           0 :                         return false;
     336             :                 }
     337             : 
     338          77 :                 if (nNewFeatureSize > nRowBufferMaxSize)
     339          40 :                     nRowBufferMaxSize = nNewFeatureSize;
     340          77 :                 m_nFileSize += sizeof(uint32_t) + nNewFeatureSize;
     341             :             }
     342             :         }
     343          38 :         VSIFSeekL(oWholeFileRewriter.m_fpTableX, nOffsetInTableX, SEEK_SET);
     344          38 :         if (VSIFWriteL(abyBufferOffsets.data(), m_nTablxOffsetSize * 1024, 1,
     345          38 :                        oWholeFileRewriter.m_fpTableX) != 1)
     346           0 :             return false;
     347             :     }
     348             : 
     349          38 :     m_nRowBufferMaxSize = nRowBufferMaxSize;
     350          38 :     m_nHeaderBufferMaxSize = std::max(m_nFieldDescLength, m_nRowBufferMaxSize);
     351             : 
     352          38 :     return oWholeFileRewriter.Commit();
     353             : }
     354             : 
     355             : /************************************************************************/
     356             : /*                       WriteFieldDescriptor()                         */
     357             : /************************************************************************/
     358             : 
     359             : static void
     360       13861 : WriteFieldDescriptor(std::vector<GByte> &abyBuffer, const FileGDBField *psField,
     361             :                      bool bGeomTypeHasZ, bool bGeomTypeHasM,
     362             :                      bool bStringsAreUTF8, uint32_t &nGeomFieldBBoxOffsetOut,
     363             :                      uint32_t &nGeomFieldSpatialIndexGridResOffsetOut)
     364             : {
     365       13861 :     WriteUTF16String(abyBuffer, psField->GetName().c_str(),
     366             :                      NUMBER_OF_CHARS_ON_UINT8);
     367       13861 :     WriteUTF16String(abyBuffer, psField->GetAlias().c_str(),
     368             :                      NUMBER_OF_CHARS_ON_UINT8);
     369       13861 :     WriteUInt8(abyBuffer, static_cast<uint8_t>(psField->GetType()));
     370             : 
     371       13861 :     const auto &sDefault = *(psField->GetDefault());
     372             : 
     373       13861 :     uint8_t nFlag = 0;
     374       13861 :     if (psField->IsNullable())
     375        7872 :         nFlag = static_cast<uint8_t>(nFlag | FileGDBField::MASK_NULLABLE);
     376       13861 :     if (psField->IsRequired())
     377        2787 :         nFlag = static_cast<uint8_t>(nFlag | FileGDBField::MASK_REQUIRED);
     378       13861 :     if (psField->IsEditable())
     379       11602 :         nFlag = static_cast<uint8_t>(nFlag | FileGDBField::MASK_EDITABLE);
     380             : 
     381       13861 :     switch (psField->GetType())
     382             :     {
     383           0 :         case FGFT_UNDEFINED:
     384             :         {
     385           0 :             CPLAssert(false);
     386             :             break;
     387             :         }
     388             : 
     389         304 :         case FGFT_INT16:
     390             :         {
     391         304 :             WriteUInt8(abyBuffer, 2);  // sizeof(int16)
     392         304 :             WriteUInt8(abyBuffer, nFlag);
     393         608 :             if (!OGR_RawField_IsNull(&sDefault) &&
     394         304 :                 !OGR_RawField_IsUnset(&sDefault))
     395             :             {
     396          66 :                 WriteUInt8(abyBuffer, 2);  // sizeof(int16)
     397          66 :                 WriteInt16(abyBuffer, static_cast<int16_t>(sDefault.Integer));
     398             :             }
     399             :             else
     400             :             {
     401         238 :                 WriteUInt8(abyBuffer, 0);  // size of default value
     402             :             }
     403         304 :             break;
     404             :         }
     405             : 
     406        1316 :         case FGFT_INT32:
     407             :         {
     408        1316 :             WriteUInt8(abyBuffer, 4);  // sizeof(int32)
     409        1316 :             WriteUInt8(abyBuffer, nFlag);
     410        2632 :             if (!OGR_RawField_IsNull(&sDefault) &&
     411        1316 :                 !OGR_RawField_IsUnset(&sDefault))
     412             :             {
     413          58 :                 WriteUInt8(abyBuffer, 4);  // sizeof(int32)
     414          58 :                 WriteInt32(abyBuffer, sDefault.Integer);
     415             :             }
     416             :             else
     417             :             {
     418        1258 :                 WriteUInt8(abyBuffer, 0);  // size of default value
     419             :             }
     420        1316 :             break;
     421             :         }
     422             : 
     423          49 :         case FGFT_FLOAT32:
     424             :         {
     425          49 :             WriteUInt8(abyBuffer, 4);  // sizeof(float32)
     426          49 :             WriteUInt8(abyBuffer, nFlag);
     427          98 :             if (!OGR_RawField_IsNull(&sDefault) &&
     428          49 :                 !OGR_RawField_IsUnset(&sDefault))
     429             :             {
     430          42 :                 WriteUInt8(abyBuffer, 4);  // sizeof(float32)
     431          42 :                 WriteFloat32(abyBuffer, static_cast<float>(sDefault.Real));
     432             :             }
     433             :             else
     434             :             {
     435           7 :                 WriteUInt8(abyBuffer, 0);  // size of default value
     436             :             }
     437          49 :             break;
     438             :         }
     439             : 
     440        2352 :         case FGFT_FLOAT64:
     441             :         {
     442        2352 :             WriteUInt8(abyBuffer, 8);  // sizeof(float64)
     443        2352 :             WriteUInt8(abyBuffer, nFlag);
     444        4704 :             if (!OGR_RawField_IsNull(&sDefault) &&
     445        2352 :                 !OGR_RawField_IsUnset(&sDefault))
     446             :             {
     447          32 :                 WriteUInt8(abyBuffer, 8);  // sizeof(float64)
     448          32 :                 WriteFloat64(abyBuffer, sDefault.Real);
     449             :             }
     450             :             else
     451             :             {
     452        2320 :                 WriteUInt8(abyBuffer, 0);  // size of default value
     453             :             }
     454        2352 :             break;
     455             :         }
     456             : 
     457        3772 :         case FGFT_STRING:
     458             :         {
     459        3772 :             WriteUInt32(abyBuffer, psField->GetMaxWidth());
     460        3772 :             WriteUInt8(abyBuffer, nFlag);
     461        7544 :             if (!OGR_RawField_IsNull(&sDefault) &&
     462        3772 :                 !OGR_RawField_IsUnset(&sDefault))
     463             :             {
     464          95 :                 if (bStringsAreUTF8)
     465             :                 {
     466          94 :                     const auto nLen = strlen(sDefault.String);
     467          94 :                     WriteVarUInt(abyBuffer, nLen);
     468          94 :                     if (nLen > 0)
     469             :                     {
     470           0 :                         abyBuffer.insert(abyBuffer.end(), sDefault.String,
     471          94 :                                          sDefault.String + nLen);
     472             :                     }
     473             :                 }
     474             :                 else
     475             :                 {
     476           1 :                     WriteUTF16String(abyBuffer, sDefault.String,
     477             :                                      NUMBER_OF_BYTES_ON_VARUINT);
     478             :                 }
     479             :             }
     480             :             else
     481             :             {
     482        3677 :                 WriteUInt8(abyBuffer, 0);  // size of default value
     483             :             }
     484        3772 :             break;
     485             :         }
     486             : 
     487          61 :         case FGFT_DATETIME:
     488             :         case FGFT_DATE:
     489             :         {
     490          61 :             WriteUInt8(abyBuffer, 8);  // sizeof(float64)
     491          61 :             WriteUInt8(abyBuffer, nFlag);
     492         122 :             if (!OGR_RawField_IsNull(&sDefault) &&
     493          61 :                 !OGR_RawField_IsUnset(&sDefault))
     494             :             {
     495          20 :                 WriteUInt8(abyBuffer, 8);  // sizeof(float64)
     496          20 :                 WriteFloat64(abyBuffer,
     497             :                              FileGDBOGRDateToDoubleDate(
     498             :                                  &sDefault, /* bConvertToUTC = */ true,
     499          20 :                                  psField->IsHighPrecision()));
     500             :             }
     501             :             else
     502             :             {
     503          41 :                 WriteUInt8(abyBuffer, 0);  // size of default value
     504             :             }
     505          61 :             break;
     506             :         }
     507             : 
     508        1781 :         case FGFT_OBJECTID:
     509             :         {
     510        1781 :             WriteUInt8(abyBuffer, 4);  // sizeof(uint32) ?
     511        1781 :             WriteUInt8(abyBuffer, 2);  // magic value
     512        1781 :             break;
     513             :         }
     514             : 
     515         546 :         case FGFT_GEOMETRY:
     516             :         {
     517             :             const auto *geomField =
     518         546 :                 cpl::down_cast<const FileGDBGeomField *>(psField);
     519         546 :             WriteUInt8(abyBuffer, 0);  // unknown role
     520         546 :             WriteUInt8(abyBuffer, nFlag);
     521         546 :             WriteUTF16String(abyBuffer, geomField->GetWKT().c_str(),
     522             :                              NUMBER_OF_BYTES_ON_UINT16);
     523         546 :             WriteUInt8(
     524             :                 abyBuffer,
     525             :                 static_cast<uint8_t>(
     526             :                     1 |
     527        1092 :                     (((geomField->HasMOriginScaleTolerance() ? 1 : 0)) << 1) |
     528         546 :                     (((geomField->HasZOriginScaleTolerance() ? 1 : 0)) << 2)));
     529         546 :             WriteFloat64(abyBuffer, geomField->GetXOrigin());
     530         546 :             WriteFloat64(abyBuffer, geomField->GetYOrigin());
     531         546 :             WriteFloat64(abyBuffer, geomField->GetXYScale());
     532         546 :             if (geomField->HasMOriginScaleTolerance())
     533             :             {
     534         546 :                 WriteFloat64(abyBuffer, geomField->GetMOrigin());
     535         546 :                 WriteFloat64(abyBuffer, geomField->GetMScale());
     536             :             }
     537         546 :             if (geomField->HasZOriginScaleTolerance())
     538             :             {
     539         546 :                 WriteFloat64(abyBuffer, geomField->GetZOrigin());
     540         546 :                 WriteFloat64(abyBuffer, geomField->GetZScale());
     541             :             }
     542         546 :             WriteFloat64(abyBuffer, geomField->GetXYTolerance());
     543         546 :             if (geomField->HasMOriginScaleTolerance())
     544             :             {
     545         546 :                 WriteFloat64(abyBuffer, geomField->GetMTolerance());
     546             :             }
     547         546 :             if (geomField->HasZOriginScaleTolerance())
     548             :             {
     549         546 :                 WriteFloat64(abyBuffer, geomField->GetZTolerance());
     550             :             }
     551         546 :             nGeomFieldBBoxOffsetOut = static_cast<uint32_t>(abyBuffer.size());
     552         546 :             WriteFloat64(abyBuffer, geomField->GetXMin());
     553         546 :             WriteFloat64(abyBuffer, geomField->GetYMin());
     554         546 :             WriteFloat64(abyBuffer, geomField->GetXMax());
     555         546 :             WriteFloat64(abyBuffer, geomField->GetYMax());
     556         546 :             if (bGeomTypeHasZ)
     557             :             {
     558          49 :                 WriteFloat64(abyBuffer, geomField->GetZMin());
     559          49 :                 WriteFloat64(abyBuffer, geomField->GetZMax());
     560             :             }
     561         546 :             if (bGeomTypeHasM)
     562             :             {
     563          24 :                 WriteFloat64(abyBuffer, geomField->GetMMin());
     564          24 :                 WriteFloat64(abyBuffer, geomField->GetMMax());
     565             :             }
     566         546 :             WriteUInt8(abyBuffer, 0);  // possibly an indicator of existence of
     567             :                                        // spatial index or its type?
     568             :             const auto &adfSpatialIndexGridResolution =
     569         546 :                 geomField->GetSpatialIndexGridResolution();
     570         546 :             WriteUInt32(abyBuffer, static_cast<uint32_t>(
     571         546 :                                        adfSpatialIndexGridResolution.size()));
     572         546 :             nGeomFieldSpatialIndexGridResOffsetOut =
     573         546 :                 static_cast<uint32_t>(abyBuffer.size());
     574        1550 :             for (double dfSize : adfSpatialIndexGridResolution)
     575        1004 :                 WriteFloat64(abyBuffer, dfSize);
     576         546 :             break;
     577             :         }
     578             : 
     579         232 :         case FGFT_BINARY:
     580             :         {
     581         232 :             WriteUInt8(abyBuffer, 0);  // unknown role
     582         232 :             WriteUInt8(abyBuffer, nFlag);
     583         232 :             break;
     584             :         }
     585             : 
     586           0 :         case FGFT_RASTER:
     587             :         {
     588             :             // Not handled for now
     589           0 :             CPLAssert(false);
     590             :             break;
     591             :         }
     592             : 
     593        2523 :         case FGFT_GUID:
     594             :         case FGFT_GLOBALID:
     595             :         {
     596        2523 :             WriteUInt8(abyBuffer, 38);  // size
     597        2523 :             WriteUInt8(abyBuffer, nFlag);
     598        2523 :             break;
     599             :         }
     600             : 
     601         918 :         case FGFT_XML:
     602             :         {
     603         918 :             WriteUInt8(abyBuffer, 0);  // unknown role
     604         918 :             WriteUInt8(abyBuffer, nFlag);
     605         918 :             break;
     606             :         }
     607             : 
     608           1 :         case FGFT_INT64:
     609             :         {
     610           1 :             WriteUInt8(abyBuffer, 8);  // sizeof(int64)
     611           1 :             WriteUInt8(abyBuffer, nFlag);
     612           2 :             if (!OGR_RawField_IsNull(&sDefault) &&
     613           1 :                 !OGR_RawField_IsUnset(&sDefault))
     614             :             {
     615           1 :                 WriteUInt8(abyBuffer, 8);  // sizeof(int64)
     616           1 :                 WriteInt64(abyBuffer, sDefault.Integer64);
     617             :             }
     618             :             else
     619             :             {
     620           0 :                 WriteUInt8(abyBuffer, 0);  // size of default value
     621             :             }
     622           1 :             break;
     623             :         }
     624             : 
     625           2 :         case FGFT_TIME:
     626             :         {
     627           2 :             WriteUInt8(abyBuffer, 8);  // sizeof(float64)
     628           2 :             WriteUInt8(abyBuffer, nFlag);
     629           4 :             if (!OGR_RawField_IsNull(&sDefault) &&
     630           2 :                 !OGR_RawField_IsUnset(&sDefault))
     631             :             {
     632           2 :                 WriteUInt8(abyBuffer, 8);  // sizeof(float64)
     633           2 :                 WriteFloat64(abyBuffer, FileGDBOGRTimeToDoubleTime(&sDefault));
     634             :             }
     635             :             else
     636             :             {
     637           0 :                 WriteUInt8(abyBuffer, 0);  // size of default value
     638             :             }
     639           2 :             break;
     640             :         }
     641             : 
     642           4 :         case FGFT_DATETIME_WITH_OFFSET:
     643             :         {
     644           4 :             WriteUInt8(abyBuffer, 8 + 2);  // sizeof(float64) + sizeof(int16)
     645           4 :             WriteUInt8(abyBuffer, nFlag);
     646           8 :             if (!OGR_RawField_IsNull(&sDefault) &&
     647           4 :                 !OGR_RawField_IsUnset(&sDefault))
     648             :             {
     649           4 :                 WriteUInt8(abyBuffer,
     650             :                            8 + 2);  // sizeof(float64) + sizeof(int16)
     651           4 :                 WriteFloat64(abyBuffer,
     652             :                              FileGDBOGRDateToDoubleDate(
     653             :                                  &sDefault, /* bConvertToUTC = */ false,
     654             :                                  /* bIsHighPrecision = */ true));
     655           4 :                 if (sDefault.Date.TZFlag > 1)
     656             :                 {
     657           2 :                     WriteInt16(abyBuffer,
     658             :                                static_cast<int16_t>(
     659           2 :                                    (sDefault.Date.TZFlag - 100) * 15));
     660             :                 }
     661             :                 else
     662             :                 {
     663           2 :                     WriteInt16(abyBuffer, 0);
     664             :                 }
     665             :             }
     666             :             else
     667             :             {
     668           0 :                 WriteUInt8(abyBuffer, 0);  // size of default value
     669             :             }
     670           4 :             break;
     671             :         }
     672             :     }
     673       13861 : }
     674             : 
     675             : /************************************************************************/
     676             : /*                       WriteFieldDescriptors()                        */
     677             : /************************************************************************/
     678             : 
     679        2011 : bool FileGDBTable::WriteFieldDescriptors(VSILFILE *fpTable)
     680             : {
     681        2011 :     m_bDirtyFieldDescriptors = false;
     682             : 
     683             :     // In-memory field descriptors
     684        4022 :     std::vector<GByte> abyBuffer;
     685             : 
     686        2011 :     WriteUInt32(abyBuffer, 0);  // size of field section, excluding this field.
     687             :                                 // Will be patched later
     688        2011 :     WriteUInt32(abyBuffer, 4);  // version of the file
     689             : 
     690        2011 :     const uint32_t nLayerFlags =
     691        4022 :         static_cast<uint32_t>(m_eTableGeomType) |
     692        2011 :         ((m_bStringsAreUTF8 ? 1 : 0) << 8) |  // string encoding
     693        2011 :         (((m_eTableGeomType != FGTGT_NONE) ? 1 : 0)
     694        2011 :          << 9) |  // "high precision storage"
     695        2011 :         (static_cast<uint32_t>(m_bGeomTypeHasM) << 30U) |
     696        2011 :         (static_cast<uint32_t>(m_bGeomTypeHasZ) << 31U);
     697        2011 :     WriteUInt32(abyBuffer, nLayerFlags);
     698             : 
     699        2011 :     WriteUInt16(abyBuffer, static_cast<uint16_t>(m_apoFields.size()));
     700             : 
     701        2011 :     m_nGeomFieldBBoxSubOffset = 0;
     702       15872 :     for (const auto &poField : m_apoFields)
     703             :     {
     704       13861 :         WriteFieldDescriptor(abyBuffer, poField.get(), m_bGeomTypeHasZ,
     705       13861 :                              m_bGeomTypeHasM, m_bStringsAreUTF8,
     706       13861 :                              m_nGeomFieldBBoxSubOffset,
     707       13861 :                              m_nGeomFieldSpatialIndexGridResSubOffset);
     708             :     }
     709             : 
     710             :     // Just to imitate the behavior of the FileGDB SDK !
     711        2011 :     abyBuffer.push_back(0xDE);
     712        2011 :     abyBuffer.push_back(0xAD);
     713        2011 :     abyBuffer.push_back(0xBE);
     714        2011 :     abyBuffer.push_back(0xEF);
     715             : 
     716             :     // Patch size of field section at beginning of buffer
     717             :     const auto nFieldSectionSize =
     718        2011 :         static_cast<uint32_t>(abyBuffer.size() - sizeof(uint32_t));
     719        2011 :     WriteUInt32(abyBuffer, nFieldSectionSize, 0);
     720             : 
     721        2011 :     bool bUpdateFileSize = false;
     722        2011 :     const auto nOldFieldDescLength = m_nFieldDescLength;
     723        2011 :     if (m_nOffsetFieldDesc + m_nFieldDescLength == m_nFileSize)
     724             :     {
     725             :         // Optimization: if the field descriptor section is already at end of
     726             :         // file, we can rewrite-in-place whatever its new size
     727           0 :         VSIFSeekL(fpTable, m_nOffsetFieldDesc, SEEK_SET);
     728           0 :         bUpdateFileSize = true;
     729             :     }
     730        2011 :     else if (abyBuffer.size() > m_nFieldDescLength)
     731             :     {
     732        1991 :         if (m_nOffsetFieldDesc != 0)
     733             :         {
     734          62 :             VSIFSeekL(fpTable, m_nOffsetFieldDesc, SEEK_SET);
     735             :             // Cancel unused bytes with NUL characters
     736          62 :             std::vector<GByte> abyNul(m_nFieldDescLength + sizeof(uint32_t));
     737          62 :             CPL_IGNORE_RET_VAL(
     738          62 :                 VSIFWriteL(abyNul.data(), 1, abyNul.size(), fpTable));
     739             :         }
     740        1991 :         VSIFSeekL(fpTable, m_nFileSize, SEEK_SET);
     741        1991 :         m_bDirtyHeader = true;
     742        1991 :         m_nOffsetFieldDesc = m_nFileSize;
     743        1991 :         m_nFileSize += abyBuffer.size();
     744             :     }
     745             :     else
     746             :     {
     747          20 :         VSIFSeekL(fpTable, m_nOffsetFieldDesc, SEEK_SET);
     748             :     }
     749             : 
     750             :     // Write new field descriptor
     751        2011 :     m_nFieldDescLength = nFieldSectionSize;
     752        4022 :     if (VSIFWriteL(abyBuffer.data(), 1, abyBuffer.size(), fpTable) !=
     753        2011 :         abyBuffer.size())
     754           0 :         return false;
     755             : 
     756        2011 :     if (bUpdateFileSize)
     757             :     {
     758           0 :         m_nFileSize = VSIFTellL(fpTable);
     759           0 :         VSIFTruncateL(fpTable, m_nFileSize);
     760           0 :         m_bDirtyHeader = true;
     761             :     }
     762        2011 :     else if (nOldFieldDescLength != 0 &&
     763          82 :              m_nFieldDescLength < nOldFieldDescLength)
     764             :     {
     765             :         // Cancel unused bytes with NUL characters
     766          20 :         std::vector<GByte> abyNul(nOldFieldDescLength - m_nFieldDescLength);
     767          20 :         CPL_IGNORE_RET_VAL(
     768          20 :             VSIFWriteL(abyNul.data(), 1, abyNul.size(), fpTable));
     769             :     }
     770             : 
     771        2011 :     return true;
     772             : }
     773             : 
     774             : /************************************************************************/
     775             : /*                            DeleteField()                             */
     776             : /************************************************************************/
     777             : 
     778          14 : bool FileGDBTable::DeleteField(int iField)
     779             : {
     780          14 :     if (!m_bUpdate)
     781           0 :         return false;
     782             : 
     783          14 :     if (iField < 0 || iField >= static_cast<int>(m_apoFields.size()))
     784             :     {
     785           0 :         return false;
     786             :     }
     787             : 
     788          14 :     if (m_iGeomField == iField)
     789             :     {
     790           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     791             :                  "Geometry field deletion not supported");
     792           0 :         return false;
     793             :     }
     794             : 
     795          14 :     bool bRet = true;
     796          14 :     if (iField != m_iObjectIdField)
     797             :     {
     798          14 :         std::vector<GByte> abyBlank;
     799             : 
     800             :         // Little hack: we present the geometry field as a binary one
     801             :         // to avoid any conversion
     802          14 :         const int iGeomFieldBackup = m_iGeomField;
     803          14 :         if (m_iGeomField >= 0)
     804          12 :             m_apoFields[m_iGeomField]->m_eType = FGFT_BINARY;
     805          14 :         m_iGeomField = -1;
     806             : 
     807          40 :         for (int64_t iCurFeat = 0; iCurFeat < m_nTotalRecordCount; ++iCurFeat)
     808             :         {
     809          26 :             iCurFeat = GetAndSelectNextNonEmptyRow(iCurFeat);
     810          26 :             if (iCurFeat < 0)
     811           0 :                 break;
     812          52 :             auto asValues = GetAllFieldValues();
     813             : 
     814          26 :             if (m_nRowBlobLength > 0)
     815             :             {
     816          26 :                 if (EncodeFeature(asValues, nullptr, iField))
     817             :                 {
     818          26 :                     VSIFSeekL(m_fpTable,
     819          26 :                               VSIFTellL(m_fpTable) - sizeof(uint32_t) -
     820          26 :                                   m_nRowBlobLength,
     821             :                               SEEK_SET);
     822             : 
     823          26 :                     abyBlank.resize(m_nRowBlobLength - m_abyBuffer.size());
     824             : 
     825          26 :                     if (!WriteUInt32(m_fpTable, static_cast<uint32_t>(
     826          26 :                                                     m_abyBuffer.size())) ||
     827          26 :                         VSIFWriteL(m_abyBuffer.data(), m_abyBuffer.size(), 1,
     828          52 :                                    m_fpTable) != 1 ||
     829          26 :                         (!abyBlank.empty() &&
     830          21 :                          VSIFWriteL(abyBlank.data(), abyBlank.size(), 1,
     831             :                                     m_fpTable) != 1))
     832             :                     {
     833           0 :                         bRet = false;
     834             :                     }
     835             :                 }
     836             :                 else
     837             :                 {
     838           0 :                     bRet = false;
     839             :                 }
     840             :             }
     841             : 
     842          26 :             FreeAllFieldValues(asValues);
     843             :         }
     844             : 
     845          14 :         if (iGeomFieldBackup >= 0)
     846          12 :             m_apoFields[iGeomFieldBackup]->m_eType = FGFT_GEOMETRY;
     847          14 :         m_iGeomField = iGeomFieldBackup;
     848             :     }
     849             : 
     850             :     // Delete linked index if existing
     851          14 :     GetIndexCount();
     852          14 :     if (m_apoFields[iField]->m_poIndex)
     853             :     {
     854           4 :         for (size_t i = 0; i < m_apoIndexes.size(); ++i)
     855             :         {
     856           4 :             if (m_apoIndexes[i].get() == m_apoFields[iField]->m_poIndex)
     857             :             {
     858           1 :                 m_bDirtyGdbIndexesFile = true;
     859             : 
     860           1 :                 if (iField != m_iObjectIdField)
     861             :                 {
     862           1 :                     VSIUnlink(
     863           2 :                         CPLResetExtensionSafe(
     864             :                             m_osFilename.c_str(),
     865           2 :                             (m_apoIndexes[i]->GetIndexName() + ".atx").c_str())
     866             :                             .c_str());
     867             :                 }
     868             : 
     869           1 :                 m_apoIndexes.erase(m_apoIndexes.begin() + i);
     870           1 :                 break;
     871             :             }
     872             :         }
     873             :     }
     874             : 
     875             :     // Renumber objectId and geomField indices
     876          14 :     if (m_iObjectIdField == iField)
     877           0 :         m_iObjectIdField = -1;
     878          14 :     else if (iField < m_iObjectIdField)
     879           2 :         m_iObjectIdField--;
     880             : 
     881          14 :     if (iField < m_iGeomField)
     882           2 :         m_iGeomField--;
     883             : 
     884          14 :     if (m_apoFields[iField]->IsNullable())
     885             :     {
     886          12 :         m_nCountNullableFields--;
     887          12 :         m_nNullableFieldsSizeInBytes =
     888          12 :             BIT_ARRAY_SIZE_IN_BYTES(m_nCountNullableFields);
     889             :     }
     890             : 
     891          14 :     m_apoFields.erase(m_apoFields.begin() + iField);
     892             : 
     893          14 :     m_bDirtyFieldDescriptors = true;
     894             : 
     895          14 :     return bRet;
     896             : }
     897             : 
     898             : /************************************************************************/
     899             : /*                            AlterField()                              */
     900             : /************************************************************************/
     901             : 
     902           7 : bool FileGDBTable::AlterField(int iField, const std::string &osName,
     903             :                               const std::string &osAlias,
     904             :                               FileGDBFieldType eType, bool bNullable,
     905             :                               int nMaxWidth, const OGRField &sDefault)
     906             : {
     907           7 :     if (!m_bUpdate)
     908           0 :         return false;
     909             : 
     910           7 :     if (iField < 0 || iField >= static_cast<int>(m_apoFields.size()))
     911             :     {
     912           0 :         return false;
     913             :     }
     914             : 
     915           7 :     if (m_iGeomField == iField)
     916             :     {
     917           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     918             :                  "AlterField() not supported on geometry field");
     919           0 :         return false;
     920             :     }
     921             : 
     922           7 :     if (m_apoFields[iField]->GetType() != eType)
     923             :     {
     924           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     925             :                  "AlterField() does not support modifying the field type");
     926           0 :         return false;
     927             :     }
     928             : 
     929           7 :     if (m_apoFields[iField]->IsNullable() != bNullable)
     930             :     {
     931           0 :         CPLError(CE_Failure, CPLE_NotSupported,
     932             :                  "AlterField() does not support modifying the nullable state");
     933           0 :         return false;
     934             :     }
     935             : 
     936           7 :     const bool bRenameField = m_apoFields[iField]->GetName() != osName;
     937           7 :     if (bRenameField && GetFieldIdx(osName) >= 0)
     938             :     {
     939           1 :         CPLError(
     940             :             CE_Failure, CPLE_NotSupported,
     941             :             "AlterField() cannot rename a field to an existing field name");
     942           1 :         return false;
     943             :     }
     944             : 
     945             :     // Update linked index if existing
     946           6 :     GetIndexCount();
     947           6 :     auto poIndex = m_apoFields[iField]->m_poIndex;
     948             : 
     949          12 :     m_apoFields[iField] = std::make_unique<FileGDBField>(
     950           6 :         osName, osAlias, eType, bNullable, m_apoFields[iField]->IsRequired(),
     951          12 :         m_apoFields[iField]->IsEditable(), nMaxWidth, sDefault);
     952           6 :     m_apoFields[iField]->SetParent(this);
     953           6 :     m_apoFields[iField]->m_poIndex = poIndex;
     954           6 :     if (poIndex && bRenameField)
     955             :     {
     956           1 :         m_bDirtyGdbIndexesFile = true;
     957           1 :         if (STARTS_WITH_CI(poIndex->GetExpression().c_str(), "LOWER("))
     958           0 :             poIndex->m_osExpression = "LOWER(" + osName + ")";
     959             :         else
     960           1 :             poIndex->m_osExpression = osName;
     961             :     }
     962           6 :     m_bDirtyFieldDescriptors = true;
     963             : 
     964           6 :     return true;
     965             : }
     966             : 
     967             : /************************************************************************/
     968             : /*                          AlterGeomField()                            */
     969             : /************************************************************************/
     970             : 
     971           4 : bool FileGDBTable::AlterGeomField(const std::string &osName,
     972             :                                   const std::string &osAlias, bool bNullable,
     973             :                                   const std::string &osWKT)
     974             : {
     975           4 :     if (!m_bUpdate)
     976           0 :         return false;
     977           4 :     if (m_iGeomField < 0)
     978           0 :         return false;
     979             : 
     980             :     auto poGeomField =
     981           4 :         cpl::down_cast<FileGDBGeomField *>(m_apoFields[m_iGeomField].get());
     982           4 :     if (poGeomField->IsNullable() != bNullable)
     983             :     {
     984           0 :         CPLError(
     985             :             CE_Failure, CPLE_NotSupported,
     986             :             "AlterGeomField() does not support modifying the nullable state");
     987           0 :         return false;
     988             :     }
     989             : 
     990           4 :     const bool bRenameField = poGeomField->GetName() != osName;
     991             : 
     992           4 :     poGeomField->m_osName = osName;
     993           4 :     poGeomField->m_osAlias = osAlias;
     994           4 :     poGeomField->m_bNullable = bNullable;
     995           4 :     poGeomField->m_osWKT = osWKT;
     996           4 :     auto poIndex = poGeomField->m_poIndex;
     997           4 :     if (poIndex && bRenameField)
     998             :     {
     999           0 :         poIndex->m_osExpression = osName;
    1000           0 :         m_bDirtyGdbIndexesFile = true;
    1001             :     }
    1002           4 :     m_bDirtyFieldDescriptors = true;
    1003             : 
    1004           4 :     return true;
    1005             : }
    1006             : 
    1007             : } /* namespace OpenFileGDB */

Generated by: LCOV version 1.14