LCOV - code coverage report
Current view: top level - ogr/ogrsf_frmts/arrow_common - ograrrowlayer.hpp (source / functions) Hit Total Coverage
Test: gdal_filtered.info Lines: 2765 3145 87.9 %
Date: 2025-01-18 12:42:00 Functions: 114 121 94.2 %

          Line data    Source code
       1             : /******************************************************************************
       2             :  *
       3             :  * Project:  Arrow generic code
       4             :  * Purpose:  Arrow generic code
       5             :  * Author:   Even Rouault, <even.rouault at spatialys.com>
       6             :  *
       7             :  ******************************************************************************
       8             :  * Copyright (c) 2022, Planet Labs
       9             :  *
      10             :  * SPDX-License-Identifier: MIT
      11             :  ****************************************************************************/
      12             : 
      13             : #ifndef OGARROWLAYER_HPP_INCLUDED
      14             : #define OGARROWLAYER_HPP_INCLUDED
      15             : 
      16             : #include "ogr_arrow.h"
      17             : 
      18             : #include "cpl_float.h"
      19             : #include "cpl_json.h"
      20             : #include "cpl_time.h"
      21             : #include "ogrlayerarrow.h"
      22             : #include "ogr_p.h"
      23             : #include "ogr_swq.h"
      24             : #include "ogr_wkb.h"
      25             : 
      26             : #include <algorithm>
      27             : #include <cinttypes>
      28             : #include <limits>
      29             : #include <string_view>
      30             : 
      31             : #define SWQ_ISNOTNULL (-SWQ_ISNULL)
      32             : 
      33             : /************************************************************************/
      34             : /*                         OGRArrowLayer()                              */
      35             : /************************************************************************/
      36             : 
      37        1665 : inline OGRArrowLayer::OGRArrowLayer(OGRArrowDataset *poDS,
      38        1665 :                                     const char *pszLayerName)
      39        1665 :     : m_poArrowDS(poDS), m_poMemoryPool(poDS->GetMemoryPool())
      40             : {
      41        1665 :     m_poFeatureDefn = new OGRFeatureDefn(pszLayerName);
      42        1665 :     m_poFeatureDefn->SetGeomType(wkbNone);
      43        1665 :     m_poFeatureDefn->Reference();
      44        1665 :     SetDescription(pszLayerName);
      45        1665 : }
      46             : 
      47             : /************************************************************************/
      48             : /*                        ~OGRFeatherLayer()                            */
      49             : /************************************************************************/
      50             : 
      51        1665 : inline OGRArrowLayer::~OGRArrowLayer()
      52             : {
      53        1665 :     if (m_sCachedSchema.release)
      54          59 :         m_sCachedSchema.release(&m_sCachedSchema);
      55             : 
      56        1665 :     CPLDebug("ARROW", "Memory pool: bytes_allocated = %" PRId64,
      57        1665 :              m_poMemoryPool->bytes_allocated());
      58        1665 :     CPLDebug("ARROW", "Memory pool: max_memory = %" PRId64,
      59        1665 :              m_poMemoryPool->max_memory());
      60        1665 :     m_poFeatureDefn->Release();
      61        1665 : }
      62             : 
      63             : /************************************************************************/
      64             : /*                         LoadGDALSchema()                             */
      65             : /************************************************************************/
      66             : 
      67             : inline std::map<std::string, std::unique_ptr<OGRFieldDefn>>
      68        1665 : OGRArrowLayer::LoadGDALSchema(const arrow::KeyValueMetadata *kv_metadata)
      69             : {
      70             :     std::map<std::string, std::unique_ptr<OGRFieldDefn>>
      71        1665 :         oMapFieldNameToGDALSchemaFieldDefn;
      72        1702 :     if (kv_metadata && kv_metadata->Contains("gdal:schema") &&
      73          37 :         CPLTestBool(CPLGetConfigOption(
      74        1702 :             ("OGR_" + GetDriverUCName() + "_READ_GDAL_SCHEMA").c_str(), "YES")))
      75             :     {
      76          74 :         auto gdalSchema = kv_metadata->Get("gdal:schema");
      77          37 :         if (gdalSchema.ok())
      78             :         {
      79          37 :             CPLDebug(GetDriverUCName().c_str(), "gdal:schema = %s",
      80             :                      gdalSchema->c_str());
      81          74 :             CPLJSONDocument oDoc;
      82          37 :             if (oDoc.LoadMemory(*gdalSchema))
      83             :             {
      84          74 :                 auto oRoot = oDoc.GetRoot();
      85             : 
      86          37 :                 m_osFIDColumn = oRoot.GetString("fid");
      87             : 
      88         111 :                 auto oColumns = oRoot.GetObj("columns");
      89          37 :                 if (oColumns.IsValid())
      90             :                 {
      91         414 :                     for (const auto &oColumn : oColumns.GetChildren())
      92             :                     {
      93         754 :                         const auto osName = oColumn.GetName();
      94        1131 :                         const auto osType = oColumn.GetString("type");
      95        1131 :                         const auto osSubType = oColumn.GetString("subtype");
      96             :                         auto poFieldDefn = std::make_unique<OGRFieldDefn>(
      97         754 :                             osName.c_str(), OFTString);
      98        2267 :                         for (int iType = 0;
      99        2267 :                              iType <= static_cast<int>(OFTMaxType); iType++)
     100             :                         {
     101        2267 :                             if (EQUAL(osType.c_str(),
     102             :                                       OGRFieldDefn::GetFieldTypeName(
     103             :                                           static_cast<OGRFieldType>(iType))))
     104             :                             {
     105         377 :                                 poFieldDefn->SetType(
     106             :                                     static_cast<OGRFieldType>(iType));
     107         377 :                                 break;
     108             :                             }
     109             :                         }
     110         377 :                         if (!osSubType.empty())
     111             :                         {
     112         500 :                             for (int iSubType = 0;
     113         500 :                                  iSubType <= static_cast<int>(OFSTMaxSubType);
     114             :                                  iSubType++)
     115             :                             {
     116         500 :                                 if (EQUAL(osSubType.c_str(),
     117             :                                           OGRFieldDefn::GetFieldSubTypeName(
     118             :                                               static_cast<OGRFieldSubType>(
     119             :                                                   iSubType))))
     120             :                                 {
     121         112 :                                     poFieldDefn->SetSubType(
     122             :                                         static_cast<OGRFieldSubType>(iSubType));
     123         112 :                                     break;
     124             :                                 }
     125             :                             }
     126             :                         }
     127         377 :                         poFieldDefn->SetWidth(oColumn.GetInteger("width"));
     128         377 :                         poFieldDefn->SetPrecision(
     129             :                             oColumn.GetInteger("precision"));
     130             : 
     131             :                         const auto osAlternativeName =
     132        1131 :                             oColumn.GetString("alternative_name");
     133         377 :                         if (!osAlternativeName.empty())
     134           2 :                             poFieldDefn->SetAlternativeName(
     135             :                                 osAlternativeName.c_str());
     136             : 
     137        1131 :                         const auto osComment = oColumn.GetString("comment");
     138         377 :                         if (!osComment.empty())
     139           3 :                             poFieldDefn->SetComment(osComment);
     140             : 
     141         377 :                         oMapFieldNameToGDALSchemaFieldDefn[osName] =
     142         754 :                             std::move(poFieldDefn);
     143             :                     }
     144             :                 }
     145             :             }
     146             :         }
     147             :     }
     148        1665 :     return oMapFieldNameToGDALSchemaFieldDefn;
     149             : }
     150             : 
     151             : /************************************************************************/
     152             : /*                        LoadGDALMetadata()                            */
     153             : /************************************************************************/
     154             : 
     155             : inline void
     156        1120 : OGRArrowLayer::LoadGDALMetadata(const arrow::KeyValueMetadata *kv_metadata)
     157             : {
     158        1120 :     if (kv_metadata && kv_metadata->Contains("gdal:metadata"))
     159             :     {
     160          14 :         auto gdalMetadata = kv_metadata->Get("gdal:metadata");
     161           7 :         if (gdalMetadata.ok())
     162             :         {
     163          14 :             CPLJSONDocument oDoc;
     164           7 :             if (oDoc.LoadMemory(*gdalMetadata))
     165             :             {
     166          14 :                 auto oRoot = oDoc.GetRoot();
     167          16 :                 for (const auto &oDomain : oRoot.GetChildren())
     168             :                 {
     169          10 :                     if (STARTS_WITH(oDomain.GetName().c_str(), "json:") &&
     170           1 :                         oDomain.GetType() == CPLJSONObject::Type::Object)
     171             :                     {
     172           1 :                         char **papszMD = nullptr;
     173           1 :                         papszMD = CSLAddString(
     174             :                             papszMD,
     175           2 :                             oDomain.Format(CPLJSONObject::PrettyFormat::Plain)
     176             :                                 .c_str());
     177           1 :                         SetMetadata(papszMD, oDomain.GetName().c_str());
     178           1 :                         CSLDestroy(papszMD);
     179             :                     }
     180           9 :                     else if (STARTS_WITH(oDomain.GetName().c_str(), "xml:") &&
     181           1 :                              oDomain.GetType() == CPLJSONObject::Type::String)
     182             :                     {
     183           1 :                         char **papszMD = nullptr;
     184             :                         papszMD =
     185           1 :                             CSLAddString(papszMD, oDomain.ToString().c_str());
     186           1 :                         SetMetadata(papszMD, oDomain.GetName().c_str());
     187           1 :                         CSLDestroy(papszMD);
     188             :                     }
     189             :                     else
     190             :                     {
     191          14 :                         for (const auto &oItem : oDomain.GetChildren())
     192             :                         {
     193           7 :                             if (oItem.GetType() == CPLJSONObject::Type::String)
     194             :                             {
     195          14 :                                 SetMetadataItem(oItem.GetName().c_str(),
     196          14 :                                                 oItem.ToString().c_str(),
     197          14 :                                                 oDomain.GetName().c_str());
     198             :                             }
     199             :                         }
     200             :                     }
     201             :                 }
     202             :             }
     203             :         }
     204             :     }
     205        1120 : }
     206             : 
     207             : /************************************************************************/
     208             : /*                        IsIntegerArrowType()                          */
     209             : /************************************************************************/
     210             : 
     211        8950 : inline bool OGRArrowLayer::IsIntegerArrowType(arrow::Type::type typeId)
     212             : {
     213        8549 :     return typeId == arrow::Type::INT8 || typeId == arrow::Type::UINT8 ||
     214        7747 :            typeId == arrow::Type::INT16 || typeId == arrow::Type::UINT16 ||
     215        6538 :            typeId == arrow::Type::INT32 || typeId == arrow::Type::UINT32 ||
     216       17499 :            typeId == arrow::Type::INT64 || typeId == arrow::Type::UINT64;
     217             : }
     218             : 
     219             : /************************************************************************/
     220             : /*                         IsHandledListOrMapType()                     */
     221             : /************************************************************************/
     222             : 
     223        8944 : inline bool OGRArrowLayer::IsHandledListOrMapType(
     224             :     const std::shared_ptr<arrow::DataType> &valueType)
     225             : {
     226        8944 :     const auto itemTypeId = valueType->id();
     227        8543 :     return itemTypeId == arrow::Type::BOOL || IsIntegerArrowType(itemTypeId) ||
     228        5289 :            itemTypeId == arrow::Type::HALF_FLOAT ||
     229        4888 :            itemTypeId == arrow::Type::FLOAT ||
     230        4487 :            itemTypeId == arrow::Type::DOUBLE ||
     231             : #if ARROW_VERSION_MAJOR >= 18
     232        4487 :            itemTypeId == arrow::Type::DECIMAL32 ||
     233        4487 :            itemTypeId == arrow::Type::DECIMAL64 ||
     234             : #endif
     235        3713 :            itemTypeId == arrow::Type::DECIMAL128 ||
     236        3689 :            itemTypeId == arrow::Type::DECIMAL256 ||
     237        1713 :            itemTypeId == arrow::Type::STRING ||
     238        1689 :            itemTypeId == arrow::Type::LARGE_STRING ||
     239             : #if ARROW_VERSION_MAJOR >= 15
     240        1685 :            itemTypeId == arrow::Type::STRING_VIEW ||
     241             : #endif
     242        1261 :            itemTypeId == arrow::Type::STRUCT ||
     243          26 :            (itemTypeId == arrow::Type::MAP &&
     244          13 :             IsHandledMapType(
     245       27692 :                 std::static_pointer_cast<arrow::MapType>(valueType))) ||
     246          48 :            ((itemTypeId == arrow::Type::LIST ||
     247          24 :              itemTypeId == arrow::Type::LARGE_LIST ||
     248        2496 :              itemTypeId == arrow::Type::FIXED_SIZE_LIST) &&
     249        1248 :             IsHandledListType(
     250       19136 :                 std::static_pointer_cast<arrow::BaseListType>(valueType)));
     251             : }
     252             : 
     253             : /************************************************************************/
     254             : /*                         IsHandledListType()                          */
     255             : /************************************************************************/
     256             : 
     257        1715 : inline bool OGRArrowLayer::IsHandledListType(
     258             :     const std::shared_ptr<arrow::BaseListType> &listType)
     259             : {
     260        1715 :     return IsHandledListOrMapType(listType->value_type());
     261             : }
     262             : 
     263             : /************************************************************************/
     264             : /*                          IsHandledMapType()                          */
     265             : /************************************************************************/
     266             : 
     267             : inline bool
     268        7229 : OGRArrowLayer::IsHandledMapType(const std::shared_ptr<arrow::MapType> &mapType)
     269             : {
     270        7229 :     const auto typeId = mapType->key_type()->id();
     271             :     return (typeId == arrow::Type::STRING
     272             : #if ARROW_VERSION_MAJOR >= 15
     273           2 :             || typeId == arrow::Type::STRING_VIEW
     274             : #endif
     275       14458 :             ) &&
     276       14458 :            IsHandledListOrMapType(mapType->item_type());
     277             : }
     278             : 
     279             : /************************************************************************/
     280             : /*                        MapArrowTypeToOGR()                           */
     281             : /************************************************************************/
     282             : 
     283       35855 : inline bool OGRArrowLayer::MapArrowTypeToOGR(
     284             :     const std::shared_ptr<arrow::DataType> &typeIn,
     285             :     const std::shared_ptr<arrow::Field> &field, OGRFieldDefn &oField,
     286             :     OGRFieldType &eType, OGRFieldSubType &eSubType,
     287             :     const std::vector<int> &path,
     288             :     const std::map<std::string, std::unique_ptr<OGRFieldDefn>>
     289             :         &oMapFieldNameToGDALSchemaFieldDefn)
     290             : {
     291       35855 :     bool bTypeOK = false;
     292             : 
     293       71710 :     std::string osExtensionName;
     294       35855 :     std::shared_ptr<arrow::DataType> type(typeIn);
     295       35855 :     if (type->id() == arrow::Type::EXTENSION)
     296             :     {
     297         167 :         auto extensionType = cpl::down_cast<arrow::ExtensionType *>(type.get());
     298         167 :         osExtensionName = extensionType->extension_name();
     299         167 :         type = extensionType->storage_type();
     300             :     }
     301       35693 :     else if (const auto &field_kv_metadata = field->metadata())
     302             :     {
     303          10 :         auto extension_name = field_kv_metadata->Get(ARROW_EXTENSION_NAME_KEY);
     304           5 :         if (extension_name.ok())
     305             :         {
     306           3 :             osExtensionName = *extension_name;
     307             :         }
     308             :     }
     309             : 
     310       36025 :     if (!osExtensionName.empty() &&
     311         170 :         osExtensionName != EXTENSION_NAME_ARROW_JSON)
     312             :     {
     313           6 :         CPLDebug(GetDriverUCName().c_str(),
     314             :                  "Dealing with field %s of extension type %s as %s",
     315           3 :                  field->name().c_str(), osExtensionName.c_str(),
     316           6 :                  type->ToString().c_str());
     317             :     }
     318             : 
     319       35855 :     switch (type->id())
     320             :     {
     321         382 :         case arrow::Type::NA:
     322         382 :             break;
     323             : 
     324         413 :         case arrow::Type::BOOL:
     325         413 :             bTypeOK = true;
     326         413 :             eType = OFTInteger;
     327         413 :             eSubType = OFSTBoolean;
     328         413 :             break;
     329        1194 :         case arrow::Type::UINT8:
     330             :         case arrow::Type::INT8:
     331             :         case arrow::Type::UINT16:
     332        1194 :             bTypeOK = true;
     333        1194 :             eType = OFTInteger;
     334        1194 :             break;
     335         406 :         case arrow::Type::INT16:
     336         406 :             bTypeOK = true;
     337         406 :             eType = OFTInteger;
     338         406 :             eSubType = OFSTInt16;
     339         406 :             break;
     340          24 :         case arrow::Type::UINT32:
     341          24 :             bTypeOK = true;
     342          24 :             eType = OFTInteger64;
     343          24 :             break;
     344        1398 :         case arrow::Type::INT32:
     345        1398 :             bTypeOK = true;
     346        1398 :             eType = OFTInteger;
     347        1398 :             break;
     348         398 :         case arrow::Type::UINT64:
     349         398 :             bTypeOK = true;
     350         398 :             eType = OFTReal;  // potential loss
     351         398 :             break;
     352        1670 :         case arrow::Type::INT64:
     353        1670 :             bTypeOK = true;
     354        1670 :             eType = OFTInteger64;
     355        1670 :             break;
     356         519 :         case arrow::Type::HALF_FLOAT:  // should use OFSTFloat16 if we had it
     357             :         case arrow::Type::FLOAT:
     358         519 :             bTypeOK = true;
     359         519 :             eType = OFTReal;
     360         519 :             eSubType = OFSTFloat32;
     361         519 :             break;
     362         880 :         case arrow::Type::DOUBLE:
     363         880 :             bTypeOK = true;
     364         880 :             eType = OFTReal;
     365         880 :             break;
     366        2474 :         case arrow::Type::STRING:
     367             :         case arrow::Type::LARGE_STRING:
     368             : #if ARROW_VERSION_MAJOR >= 15
     369             :         case arrow::Type::STRING_VIEW:
     370             : #endif
     371        2474 :             bTypeOK = true;
     372        2474 :             eType = OFTString;
     373        2474 :             if (osExtensionName == EXTENSION_NAME_ARROW_JSON)
     374         167 :                 eSubType = OFSTJSON;
     375        2474 :             break;
     376         816 :         case arrow::Type::BINARY:
     377             :         case arrow::Type::LARGE_BINARY:
     378             : #if ARROW_VERSION_MAJOR >= 15
     379             :         case arrow::Type::BINARY_VIEW:
     380             : #endif
     381         816 :             bTypeOK = true;
     382         816 :             eType = OFTBinary;
     383         816 :             break;
     384         406 :         case arrow::Type::FIXED_SIZE_BINARY:
     385         406 :             bTypeOK = true;
     386         406 :             eType = OFTBinary;
     387         406 :             oField.SetWidth(
     388         812 :                 std::static_pointer_cast<arrow::FixedSizeBinaryType>(type)
     389         406 :                     ->byte_width());
     390         406 :             break;
     391             : 
     392         857 :         case arrow::Type::DATE32:
     393             :         case arrow::Type::DATE64:
     394         857 :             bTypeOK = true;
     395         857 :             eType = OFTDate;
     396         857 :             break;
     397             : 
     398        2482 :         case arrow::Type::TIMESTAMP:
     399             :         {
     400        2482 :             bTypeOK = true;
     401             :             const auto timestampType =
     402        2482 :                 static_cast<arrow::TimestampType *>(type.get());
     403        2482 :             eType = OFTDateTime;
     404        2482 :             const auto &osTZ = timestampType->timezone();
     405        2482 :             int nTZFlag = OGRTimezoneToTZFlag(osTZ.c_str(), false);
     406        2482 :             if (nTZFlag == OGR_TZFLAG_UNKNOWN && !osTZ.empty())
     407             :             {
     408           0 :                 CPLDebug(GetDriverUCName().c_str(),
     409             :                          "Field %s has unrecognized timezone %s. "
     410             :                          "UTC datetime will be used instead.",
     411           0 :                          field->name().c_str(), osTZ.c_str());
     412           0 :                 nTZFlag = OGR_TZFLAG_UTC;
     413             :             }
     414        2482 :             oField.SetTZFlag(nTZFlag);
     415        2482 :             break;
     416             :         }
     417             : 
     418         812 :         case arrow::Type::TIME32:
     419         812 :             bTypeOK = true;
     420         812 :             eType = OFTTime;
     421         812 :             break;
     422             : 
     423         796 :         case arrow::Type::TIME64:
     424         796 :             bTypeOK = true;
     425         796 :             eType = OFTInteger64;  // our OFTTime doesn't have micro or
     426             :                                    // nanosecond accuracy
     427         796 :             break;
     428             : 
     429             : #if ARROW_VERSION_MAJOR >= 18
     430         822 :         case arrow::Type::DECIMAL32:
     431             :         case arrow::Type::DECIMAL64:
     432             : #endif
     433             :         case arrow::Type::DECIMAL128:
     434             :         case arrow::Type::DECIMAL256:
     435             :         {
     436         822 :             bTypeOK = true;
     437             :             const auto decimalType =
     438        1644 :                 std::static_pointer_cast<arrow::DecimalType>(type);
     439         822 :             eType = OFTReal;
     440         822 :             oField.SetWidth(decimalType->precision());
     441         822 :             oField.SetPrecision(decimalType->scale());
     442         822 :             break;
     443             :         }
     444             : 
     445       11890 :         case arrow::Type::LIST:
     446             :         case arrow::Type::FIXED_SIZE_LIST:
     447             :         {
     448       11890 :             bTypeOK = true;
     449       23780 :             auto listType = std::static_pointer_cast<arrow::BaseListType>(type);
     450       11890 :             switch (listType->value_type()->id())
     451             :             {
     452         812 :                 case arrow::Type::BOOL:
     453         812 :                     eType = OFTIntegerList;
     454         812 :                     eSubType = OFSTBoolean;
     455         812 :                     break;
     456        4060 :                 case arrow::Type::UINT8:
     457             :                 case arrow::Type::INT8:
     458             :                 case arrow::Type::UINT16:
     459             :                 case arrow::Type::INT16:
     460             :                 case arrow::Type::INT32:
     461        4060 :                     eType = OFTIntegerList;
     462        4060 :                     break;
     463          46 :                 case arrow::Type::UINT32:
     464          46 :                     eType = OFTInteger64List;
     465          46 :                     break;
     466         796 :                 case arrow::Type::UINT64:
     467         796 :                     eType = OFTRealList;  // potential loss
     468         796 :                     break;
     469        1985 :                 case arrow::Type::INT64:
     470        1985 :                     eType = OFTInteger64List;
     471        1985 :                     break;
     472         841 :                 case arrow::Type::HALF_FLOAT:  // should use OFSTFloat16 if we
     473             :                                                // had it
     474             :                 case arrow::Type::FLOAT:
     475         841 :                     eType = OFTRealList;
     476         841 :                     eSubType = OFSTFloat32;
     477         841 :                     break;
     478        1643 :                 case arrow::Type::DOUBLE:
     479             : #if ARROW_VERSION_MAJOR >= 18
     480             :                 case arrow::Type::DECIMAL32:
     481             :                 case arrow::Type::DECIMAL64:
     482             : #endif
     483             :                 case arrow::Type::DECIMAL128:
     484             :                 case arrow::Type::DECIMAL256:
     485        1643 :                     eType = OFTRealList;
     486        1643 :                     break;
     487        1240 :                 case arrow::Type::STRING:
     488             :                 case arrow::Type::LARGE_STRING:
     489             : #if ARROW_VERSION_MAJOR >= 15
     490             :                 case arrow::Type::STRING_VIEW:
     491             : #endif
     492        1240 :                     eType = OFTStringList;
     493        1240 :                     break;
     494         467 :                 default:
     495             :                 {
     496         467 :                     if (IsHandledListType(listType))
     497             :                     {
     498         467 :                         eType = OFTString;
     499         467 :                         eSubType = OFSTJSON;
     500             :                     }
     501             :                     else
     502             :                     {
     503           0 :                         bTypeOK = false;
     504           0 :                         CPLError(CE_Warning, CPLE_AppDefined,
     505             :                                  "Field %s of unhandled type %s ignored",
     506           0 :                                  field->name().c_str(),
     507           0 :                                  type->ToString().c_str());
     508             :                     }
     509         467 :                     break;
     510             :                 }
     511             :             }
     512       11890 :             break;
     513             :         }
     514             : 
     515        7216 :         case arrow::Type::MAP:
     516             :         {
     517        7216 :             bTypeOK = true;
     518       14432 :             auto mapType = std::static_pointer_cast<arrow::MapType>(type);
     519        7216 :             if (IsHandledMapType(mapType))
     520             :             {
     521        7216 :                 eType = OFTString;
     522        7216 :                 eSubType = OFSTJSON;
     523             :             }
     524             :             else
     525             :             {
     526           0 :                 bTypeOK = false;
     527           0 :                 CPLError(CE_Warning, CPLE_AppDefined,
     528             :                          "Field %s of unhandled type %s ignored",
     529           0 :                          field->name().c_str(), type->ToString().c_str());
     530             :             }
     531        7216 :             break;
     532             :         }
     533             : 
     534           0 :         case arrow::Type::STRUCT:
     535             :             // should be handled by specialized code
     536           0 :             CPLAssert(false);
     537             :             break;
     538             : 
     539             :             // unhandled types
     540             : 
     541           0 :         case arrow::Type::INTERVAL_MONTHS:
     542             :         case arrow::Type::INTERVAL_DAY_TIME:
     543             :         case arrow::Type::SPARSE_UNION:
     544             :         case arrow::Type::DENSE_UNION:
     545             :         case arrow::Type::DICTIONARY:
     546             :         case arrow::Type::EXTENSION:
     547             :         case arrow::Type::DURATION:
     548             :         case arrow::Type::LARGE_LIST:
     549             :         case arrow::Type::INTERVAL_MONTH_DAY_NANO:
     550             : #if ARROW_VERSION_MAJOR >= 12
     551             :         case arrow::Type::RUN_END_ENCODED:
     552             : #endif
     553             : #if ARROW_VERSION_MAJOR >= 15
     554             :         case arrow::Type::LIST_VIEW:
     555             :         case arrow::Type::LARGE_LIST_VIEW:
     556             : #endif
     557             :         case arrow::Type::MAX_ID:
     558             :         {
     559           0 :             CPLError(CE_Warning, CPLE_AppDefined,
     560             :                      "Field %s of unhandled type %s ignored",
     561           0 :                      field->name().c_str(), type->ToString().c_str());
     562           0 :             break;
     563             :         }
     564             :     }
     565             : 
     566       35855 :     if (bTypeOK)
     567             :     {
     568             :         const auto oIter =
     569       35473 :             oMapFieldNameToGDALSchemaFieldDefn.find(field->name());
     570       35473 :         oField.SetType(eType);
     571       35473 :         if (oIter != oMapFieldNameToGDALSchemaFieldDefn.end())
     572             :         {
     573         377 :             const auto &poGDALFieldDefn = oIter->second;
     574         377 :             if (poGDALFieldDefn->GetType() == eType)
     575             :             {
     576         377 :                 if (eSubType == OFSTNone)
     577             :                 {
     578         265 :                     eSubType = poGDALFieldDefn->GetSubType();
     579             :                 }
     580         112 :                 else if (eSubType != poGDALFieldDefn->GetSubType())
     581             :                 {
     582           0 :                     CPLDebug(
     583           0 :                         GetDriverUCName().c_str(),
     584             :                         "Field subtype inferred from Parquet/Arrow schema is "
     585             :                         "%s, "
     586             :                         "whereas the one in gdal:schema is %s. "
     587             :                         "Using the former one.",
     588             :                         OGR_GetFieldSubTypeName(eSubType),
     589             :                         OGR_GetFieldSubTypeName(poGDALFieldDefn->GetSubType()));
     590             :                 }
     591             :             }
     592             :             else
     593             :             {
     594           0 :                 CPLDebug(GetDriverUCName().c_str(),
     595             :                          "Field type inferred from Parquet/Arrow schema is %s, "
     596             :                          "whereas the one in gdal:schema is %s. "
     597             :                          "Using the former one.",
     598             :                          OGR_GetFieldTypeName(eType),
     599             :                          OGR_GetFieldTypeName(poGDALFieldDefn->GetType()));
     600             :             }
     601         377 :             if (poGDALFieldDefn->GetWidth() > 0)
     602          36 :                 oField.SetWidth(poGDALFieldDefn->GetWidth());
     603         377 :             if (poGDALFieldDefn->GetPrecision() > 0)
     604          16 :                 oField.SetPrecision(poGDALFieldDefn->GetPrecision());
     605         377 :             if (poGDALFieldDefn->GetAlternativeNameRef()[0])
     606           2 :                 oField.SetAlternativeName(
     607             :                     poGDALFieldDefn->GetAlternativeNameRef());
     608         377 :             if (!poGDALFieldDefn->GetComment().empty())
     609           3 :                 oField.SetComment(poGDALFieldDefn->GetComment());
     610             :         }
     611       35473 :         oField.SetSubType(eSubType);
     612       35473 :         oField.SetNullable(field->nullable());
     613       35473 :         m_poFeatureDefn->AddFieldDefn(&oField);
     614       35473 :         m_anMapFieldIndexToArrowColumn.push_back(path);
     615             :     }
     616             : 
     617       71710 :     return bTypeOK;
     618             : }
     619             : 
     620             : /************************************************************************/
     621             : /*                         CreateFieldFromSchema()                      */
     622             : /************************************************************************/
     623             : 
     624       10289 : inline void OGRArrowLayer::CreateFieldFromSchema(
     625             :     const std::shared_ptr<arrow::Field> &field, const std::vector<int> &path,
     626             :     const std::map<std::string, std::unique_ptr<OGRFieldDefn>>
     627             :         &oMapFieldNameToGDALSchemaFieldDefn)
     628             : {
     629       20578 :     OGRFieldDefn oField(field->name().c_str(), OFTString);
     630       10289 :     OGRFieldType eType = OFTString;
     631       10289 :     OGRFieldSubType eSubType = OFSTNone;
     632       10289 :     bool bTypeOK = true;
     633             : 
     634       20578 :     auto type = field->type();
     635       10289 :     if (type->id() == arrow::Type::DICTIONARY && path.size() == 1)
     636             :     {
     637             :         const auto dictionaryType =
     638         218 :             std::static_pointer_cast<arrow::DictionaryType>(field->type());
     639         218 :         const auto indexType = dictionaryType->index_type();
     640         218 :         if (dictionaryType->value_type()->id() == arrow::Type::STRING &&
     641         109 :             IsIntegerArrowType(indexType->id()))
     642             :         {
     643         218 :             std::string osDomainName(field->name() + "Domain");
     644         109 :             m_poArrowDS->RegisterDomainName(osDomainName,
     645         109 :                                             m_poFeatureDefn->GetFieldCount());
     646         109 :             oField.SetDomainName(osDomainName);
     647         109 :             type = indexType;
     648             :         }
     649             :         else
     650             :         {
     651           0 :             bTypeOK = false;
     652             :         }
     653             :     }
     654             : 
     655       10289 :     if (type->id() == arrow::Type::STRUCT)
     656             :     {
     657         420 :         const auto subfields = field->Flatten();
     658         420 :         auto newpath = path;
     659         210 :         newpath.push_back(0);
     660         941 :         for (int j = 0; j < static_cast<int>(subfields.size()); j++)
     661             :         {
     662         731 :             const auto &subfield = subfields[j];
     663         731 :             newpath.back() = j;
     664         731 :             CreateFieldFromSchema(subfield, newpath,
     665             :                                   oMapFieldNameToGDALSchemaFieldDefn);
     666             :         }
     667             :     }
     668       10079 :     else if (bTypeOK)
     669             :     {
     670       10079 :         MapArrowTypeToOGR(type, field, oField, eType, eSubType, path,
     671             :                           oMapFieldNameToGDALSchemaFieldDefn);
     672             :     }
     673       10289 : }
     674             : 
     675             : /************************************************************************/
     676             : /*                       BuildDomainFromBatch()                         */
     677             : /************************************************************************/
     678             : 
     679          35 : inline std::unique_ptr<OGRFieldDomain> OGRArrowLayer::BuildDomainFromBatch(
     680             :     const std::string &osDomainName,
     681             :     const std::shared_ptr<arrow::RecordBatch> &poBatch, int iCol) const
     682             : {
     683          70 :     const auto array = poBatch->column(iCol);
     684          70 :     auto castArray = std::static_pointer_cast<arrow::DictionaryArray>(array);
     685          70 :     auto dict = castArray->dictionary();
     686          35 :     CPLAssert(dict->type_id() == arrow::Type::STRING);
     687          35 :     OGRFieldType eType = OFTInteger;
     688          35 :     const auto indexTypeId = castArray->dict_type()->index_type()->id();
     689          35 :     if (indexTypeId == arrow::Type::UINT32 ||
     690          35 :         indexTypeId == arrow::Type::UINT64 || indexTypeId == arrow::Type::INT64)
     691           0 :         eType = OFTInteger64;
     692          70 :     auto values = std::static_pointer_cast<arrow::StringArray>(dict);
     693          70 :     std::vector<OGRCodedValue> asValues;
     694          35 :     if (values->length() > INT_MAX)
     695             :     {
     696           0 :         CPLError(CE_Failure, CPLE_OutOfMemory,
     697             :                  "BuildDomainFromBatch(): too many values");
     698           0 :         return nullptr;
     699             :     }
     700             :     try
     701             :     {
     702          35 :         asValues.reserve(static_cast<size_t>(values->length()));
     703             :     }
     704           0 :     catch (const std::bad_alloc &)
     705             :     {
     706           0 :         CPLError(CE_Failure, CPLE_OutOfMemory,
     707             :                  "BuildDomainFromBatch(): out of memory");
     708           0 :         return nullptr;
     709             :     }
     710         140 :     for (int i = 0; i < static_cast<int>(values->length()); ++i)
     711             :     {
     712         105 :         if (!values->IsNull(i))
     713             :         {
     714             :             OGRCodedValue val;
     715         105 :             val.pszCode = CPLStrdup(CPLSPrintf("%d", i));
     716         105 :             val.pszValue = CPLStrdup(values->GetString(i).c_str());
     717         105 :             asValues.emplace_back(val);
     718             :         }
     719             :     }
     720          70 :     return std::make_unique<OGRCodedFieldDomain>(
     721         105 :         osDomainName, std::string(), eType, OFSTNone, std::move(asValues));
     722             : }
     723             : 
     724             : /************************************************************************/
     725             : /*               ComputeGeometryColumnTypeProcessBatch()                */
     726             : /************************************************************************/
     727             : 
     728         535 : inline OGRwkbGeometryType OGRArrowLayer::ComputeGeometryColumnTypeProcessBatch(
     729             :     const std::shared_ptr<arrow::RecordBatch> &poBatch, int iGeomCol,
     730             :     int iBatchCol, OGRwkbGeometryType eGeomType) const
     731             : {
     732        1070 :     const auto array = poBatch->column(iBatchCol);
     733             :     const auto castBinaryArray =
     734         535 :         (m_aeGeomEncoding[iGeomCol] == OGRArrowGeomEncoding::WKB)
     735             :             ? std::dynamic_pointer_cast<arrow::BinaryArray>(array)
     736        1070 :             : nullptr;
     737             :     const auto castLargeBinaryArray =
     738         535 :         (m_aeGeomEncoding[iGeomCol] == OGRArrowGeomEncoding::WKB)
     739             :             ? std::dynamic_pointer_cast<arrow::LargeBinaryArray>(array)
     740        1070 :             : nullptr;
     741             :     const auto castStringArray =
     742         535 :         (m_aeGeomEncoding[iGeomCol] == OGRArrowGeomEncoding::WKT)
     743             :             ? std::dynamic_pointer_cast<arrow::StringArray>(array)
     744        1070 :             : nullptr;
     745             :     const auto castLargeStringArray =
     746         535 :         (m_aeGeomEncoding[iGeomCol] == OGRArrowGeomEncoding::WKT)
     747             :             ? std::dynamic_pointer_cast<arrow::LargeStringArray>(array)
     748        1070 :             : nullptr;
     749        3208 :     for (int64_t i = 0; i < poBatch->num_rows(); i++)
     750             :     {
     751        2675 :         if (!array->IsNull(i))
     752             :         {
     753        2152 :             OGRwkbGeometryType eThisGeomType = wkbNone;
     754        4192 :             if (m_aeGeomEncoding[iGeomCol] == OGRArrowGeomEncoding::WKB &&
     755        2040 :                 castBinaryArray)
     756             :             {
     757        2040 :                 arrow::BinaryArray::offset_type out_length = 0;
     758        2040 :                 const uint8_t *data = castBinaryArray->GetValue(i, &out_length);
     759        2040 :                 if (out_length >= 5)
     760             :                 {
     761        2040 :                     OGRReadWKBGeometryType(data, wkbVariantIso, &eThisGeomType);
     762             :                 }
     763             :             }
     764         112 :             else if (m_aeGeomEncoding[iGeomCol] == OGRArrowGeomEncoding::WKB &&
     765           0 :                      castLargeBinaryArray)
     766             :             {
     767           0 :                 arrow::LargeBinaryArray::offset_type out_length = 0;
     768             :                 const uint8_t *data =
     769           0 :                     castLargeBinaryArray->GetValue(i, &out_length);
     770           0 :                 if (out_length >= 5)
     771             :                 {
     772           0 :                     OGRReadWKBGeometryType(data, wkbVariantIso, &eThisGeomType);
     773             :                 }
     774             :             }
     775         224 :             else if (m_aeGeomEncoding[iGeomCol] == OGRArrowGeomEncoding::WKT &&
     776         112 :                      castStringArray)
     777             :             {
     778         224 :                 const auto osWKT = castStringArray->GetString(i);
     779         112 :                 if (!osWKT.empty())
     780             :                 {
     781         112 :                     OGRReadWKTGeometryType(osWKT.c_str(), &eThisGeomType);
     782             :                 }
     783             :             }
     784           0 :             else if (m_aeGeomEncoding[iGeomCol] == OGRArrowGeomEncoding::WKT &&
     785           0 :                      castLargeStringArray)
     786             :             {
     787           0 :                 const auto osWKT = castLargeStringArray->GetString(i);
     788           0 :                 if (!osWKT.empty())
     789             :                 {
     790           0 :                     OGRReadWKTGeometryType(osWKT.c_str(), &eThisGeomType);
     791             :                 }
     792             :             }
     793             : 
     794        2152 :             if (eThisGeomType != wkbNone)
     795             :             {
     796        2152 :                 if (eGeomType == wkbNone)
     797         531 :                     eGeomType = eThisGeomType;
     798        1621 :                 else if (wkbFlatten(eThisGeomType) == wkbFlatten(eGeomType))
     799             :                     ;
     800           6 :                 else if (wkbFlatten(eThisGeomType) == wkbMultiLineString &&
     801           0 :                          wkbFlatten(eGeomType) == wkbLineString)
     802             :                 {
     803           0 :                     eGeomType = OGR_GT_SetModifier(
     804             :                         wkbMultiLineString,
     805           0 :                         OGR_GT_HasZ(eThisGeomType) || OGR_GT_HasZ(eGeomType),
     806           0 :                         OGR_GT_HasM(eThisGeomType) || OGR_GT_HasM(eGeomType));
     807             :                 }
     808           8 :                 else if (wkbFlatten(eThisGeomType) == wkbLineString &&
     809           2 :                          wkbFlatten(eGeomType) == wkbMultiLineString)
     810             :                     ;
     811           6 :                 else if (wkbFlatten(eThisGeomType) == wkbMultiPolygon &&
     812           0 :                          wkbFlatten(eGeomType) == wkbPolygon)
     813             :                 {
     814           0 :                     eGeomType = OGR_GT_SetModifier(
     815             :                         wkbMultiPolygon,
     816           0 :                         OGR_GT_HasZ(eThisGeomType) || OGR_GT_HasZ(eGeomType),
     817           0 :                         OGR_GT_HasM(eThisGeomType) || OGR_GT_HasM(eGeomType));
     818             :                 }
     819          10 :                 else if (wkbFlatten(eThisGeomType) == wkbPolygon &&
     820           4 :                          wkbFlatten(eGeomType) == wkbMultiPolygon)
     821             :                     ;
     822             :                 else
     823           2 :                     return wkbUnknown;
     824             : 
     825        2150 :                 eGeomType = OGR_GT_SetModifier(
     826             :                     eGeomType,
     827        2150 :                     OGR_GT_HasZ(eThisGeomType) || OGR_GT_HasZ(eGeomType),
     828        2150 :                     OGR_GT_HasM(eThisGeomType) || OGR_GT_HasM(eGeomType));
     829             :             }
     830             :         }
     831             :     }
     832         533 :     return eGeomType;
     833             : }
     834             : 
     835             : /************************************************************************/
     836             : /*                           IsPointType()                              */
     837             : /************************************************************************/
     838             : 
     839         828 : static bool IsPointType(const std::shared_ptr<arrow::DataType> &type,
     840             :                         bool &bHasZOut, bool &bHasMOut)
     841             : {
     842         828 :     if (type->id() != arrow::Type::FIXED_SIZE_LIST)
     843         498 :         return false;
     844         660 :     auto poListType = std::static_pointer_cast<arrow::FixedSizeListType>(type);
     845         330 :     const int nOutDimensionality = poListType->list_size();
     846         660 :     const std::string osValueFieldName(poListType->value_field()->name());
     847         330 :     if (nOutDimensionality == 2)
     848             :     {
     849         114 :         bHasZOut = false;
     850         114 :         bHasMOut = false;
     851             :     }
     852         216 :     else if (nOutDimensionality == 3)
     853             :     {
     854         156 :         if (osValueFieldName == "xym")
     855             :         {
     856          60 :             bHasZOut = false;
     857          60 :             bHasMOut = true;
     858             :         }
     859             :         else /* if (osValueFieldName == "xyz" || osValueFieldName == "element") */
     860             :         {
     861          96 :             bHasMOut = false;
     862          96 :             bHasZOut = true;
     863             :         }
     864             :     }
     865          60 :     else if (nOutDimensionality == 4)
     866             :     {
     867          60 :         bHasMOut = true;
     868          60 :         bHasZOut = true;
     869             :     }
     870             :     else
     871             :     {
     872           0 :         return false;
     873             :     }
     874         330 :     return poListType->value_type()->id() == arrow::Type::DOUBLE;
     875             : }
     876             : 
     877             : /************************************************************************/
     878             : /*                         IsListOfPointType()                          */
     879             : /************************************************************************/
     880             : 
     881        1298 : static bool IsListOfPointType(const std::shared_ptr<arrow::DataType> &type,
     882             :                               int nDepth, bool &bHasZOut, bool &bHasMOut)
     883             : {
     884        1298 :     if (type->id() != arrow::Type::LIST)
     885           0 :         return false;
     886        1298 :     auto poListType = std::static_pointer_cast<arrow::ListType>(type);
     887             :     return nDepth == 1
     888        1298 :                ? IsPointType(poListType->value_type(), bHasZOut, bHasMOut)
     889         607 :                : IsListOfPointType(poListType->value_type(), nDepth - 1,
     890        1298 :                                    bHasZOut, bHasMOut);
     891             : }
     892             : 
     893             : /************************************************************************/
     894             : /*                         IsPointStructType()                          */
     895             : /************************************************************************/
     896             : 
     897         498 : static bool IsPointStructType(const std::shared_ptr<arrow::DataType> &type,
     898             :                               bool &bHasZOut, bool &bHasMOut)
     899             : {
     900         498 :     if (type->id() != arrow::Type::STRUCT)
     901           0 :         return false;
     902         996 :     auto poStructType = std::static_pointer_cast<arrow::StructType>(type);
     903         498 :     const int nNumFields = poStructType->num_fields();
     904         498 :     if (nNumFields < 2 || nNumFields > 4)
     905           0 :         return false;
     906         498 :     bHasZOut = false;
     907         498 :     bHasMOut = false;
     908         996 :     const auto poFieldX = poStructType->field(0);
     909         996 :     if (poFieldX->name() != "x" ||
     910         498 :         poFieldX->type()->id() != arrow::Type::DOUBLE)
     911           0 :         return false;
     912         996 :     const auto poFieldY = poStructType->field(1);
     913         996 :     if (poFieldY->name() != "y" ||
     914         498 :         poFieldY->type()->id() != arrow::Type::DOUBLE)
     915           0 :         return false;
     916         498 :     if (nNumFields == 2)
     917         288 :         return true;
     918         420 :     const auto poField2 = poStructType->field(2);
     919         210 :     if (poField2->type()->id() != arrow::Type::DOUBLE)
     920           0 :         return false;
     921         210 :     if (poField2->name() == "z")
     922             :     {
     923         204 :         bHasZOut = true;
     924         204 :         if (nNumFields == 4)
     925             :         {
     926           6 :             const auto poField3 = poStructType->field(3);
     927          12 :             if (poField3->name() != "m" ||
     928           6 :                 poField3->type()->id() != arrow::Type::DOUBLE)
     929           0 :                 return false;
     930           6 :             bHasMOut = true;
     931             :         }
     932             :     }
     933           6 :     else if (poField2->name() == "m")
     934             :     {
     935           6 :         bHasMOut = true;
     936             :     }
     937             :     else
     938             :     {
     939           0 :         return false;
     940             :     }
     941         210 :     return true;
     942             : }
     943             : 
     944             : /************************************************************************/
     945             : /*                    IsListOfPointStructType()                         */
     946             : /************************************************************************/
     947             : 
     948             : static bool
     949         790 : IsListOfPointStructType(const std::shared_ptr<arrow::DataType> &type,
     950             :                         int nDepth, bool &bHasZOut, bool &bHasMOut)
     951             : {
     952         790 :     if (type->id() != arrow::Type::LIST)
     953           0 :         return false;
     954         790 :     auto poListType = std::static_pointer_cast<arrow::ListType>(type);
     955             :     return nDepth == 1
     956         790 :                ? IsPointStructType(poListType->value_type(), bHasZOut, bHasMOut)
     957         376 :                : IsListOfPointStructType(poListType->value_type(), nDepth - 1,
     958         790 :                                          bHasZOut, bHasMOut);
     959             : }
     960             : 
     961             : /************************************************************************/
     962             : /*                        IsValidGeometryEncoding()                     */
     963             : /************************************************************************/
     964             : 
     965        1628 : inline bool OGRArrowLayer::IsValidGeometryEncoding(
     966             :     const std::shared_ptr<arrow::Field> &field, const std::string &osEncoding,
     967             :     bool bWarnIfUnknownEncoding, OGRwkbGeometryType &eGeomTypeOut,
     968             :     OGRArrowGeomEncoding &eOGRArrowGeomEncodingOut)
     969             : {
     970        1628 :     const auto &fieldName = field->name();
     971        3256 :     std::shared_ptr<arrow::DataType> fieldType = field->type();
     972        1628 :     auto fieldTypeId = fieldType->id();
     973             : 
     974        1628 :     if (fieldTypeId == arrow::Type::EXTENSION)
     975             :     {
     976             :         auto extensionType =
     977           0 :             cpl::down_cast<arrow::ExtensionType *>(fieldType.get());
     978           0 :         fieldType = extensionType->storage_type();
     979           0 :         fieldTypeId = fieldType->id();
     980             :     }
     981             : 
     982        1628 :     eGeomTypeOut = wkbUnknown;
     983             : 
     984        3201 :     if (osEncoding == "WKT" ||  // As used in Parquet geo metadata
     985        1573 :         osEncoding ==
     986        3201 :             "ogc.wkt" ||  // As used in ARROW:extension:name field metadata
     987        1573 :         osEncoding ==
     988             :             "geoarrow.wkt"  // As used in ARROW:extension:name field metadata
     989             :     )
     990             :     {
     991          55 :         if (fieldTypeId != arrow::Type::LARGE_STRING &&
     992             :             fieldTypeId != arrow::Type::STRING)
     993             :         {
     994           0 :             CPLError(CE_Warning, CPLE_AppDefined,
     995             :                      "Geometry column %s has a non String type: %s. "
     996             :                      "Handling it as a regular field",
     997           0 :                      fieldName.c_str(), fieldType->ToString().c_str());
     998           0 :             return false;
     999             :         }
    1000          55 :         eOGRArrowGeomEncodingOut = OGRArrowGeomEncoding::WKT;
    1001          55 :         return true;
    1002             :     }
    1003             : 
    1004        2404 :     if (osEncoding == "WKB" ||  // As used in Parquet geo metadata
    1005         831 :         osEncoding ==
    1006        2404 :             "ogc.wkb" ||  // As used in ARROW:extension:name field metadata
    1007         831 :         osEncoding ==
    1008             :             "geoarrow.wkb"  // As used in ARROW:extension:name field metadata
    1009             :     )
    1010             :     {
    1011         743 :         if (fieldTypeId != arrow::Type::LARGE_BINARY &&
    1012             :             fieldTypeId != arrow::Type::BINARY)
    1013             :         {
    1014           0 :             CPLError(CE_Warning, CPLE_AppDefined,
    1015             :                      "Geometry column %s has a non Binary type: %s. "
    1016             :                      "Handling it as a regular field",
    1017           0 :                      fieldName.c_str(), fieldType->ToString().c_str());
    1018           0 :             return false;
    1019             :         }
    1020         743 :         eOGRArrowGeomEncodingOut = OGRArrowGeomEncoding::WKB;
    1021         743 :         return true;
    1022             :     }
    1023             : 
    1024         830 :     bool bHasZ = false;
    1025         830 :     bool bHasM = false;
    1026         830 :     if (osEncoding == "geoarrow.point" || osEncoding == "point")
    1027             :     {
    1028         137 :         if (IsPointType(fieldType, bHasZ, bHasM))
    1029             :         {
    1030          53 :             eOGRArrowGeomEncodingOut = OGRArrowGeomEncoding::GEOARROW_FSL_POINT;
    1031             :         }
    1032          84 :         else if (IsPointStructType(fieldType, bHasZ, bHasM))
    1033             :         {
    1034          84 :             eOGRArrowGeomEncodingOut =
    1035             :                 OGRArrowGeomEncoding::GEOARROW_STRUCT_POINT;
    1036             :         }
    1037             :         else
    1038             :         {
    1039           0 :             CPLError(CE_Warning, CPLE_AppDefined,
    1040             :                      "Geometry column %s has a type != fixed_size_list<xy: "
    1041             :                      "double>[2]> and != struct<x: double, y: double>: %s. "
    1042             :                      "Handling it as a regular field",
    1043           0 :                      fieldName.c_str(), fieldType->name().c_str());
    1044           0 :             return false;
    1045             :         }
    1046         137 :         eGeomTypeOut = OGR_GT_SetModifier(wkbPoint, static_cast<int>(bHasZ),
    1047             :                                           static_cast<int>(bHasM));
    1048         137 :         return true;
    1049             :     }
    1050             : 
    1051         693 :     else if (osEncoding == "geoarrow.linestring" || osEncoding == "linestring")
    1052             :     {
    1053         122 :         if (IsListOfPointType(fieldType, 1, bHasZ, bHasM))
    1054             :         {
    1055          52 :             eOGRArrowGeomEncodingOut =
    1056             :                 OGRArrowGeomEncoding::GEOARROW_FSL_LINESTRING;
    1057             :         }
    1058          70 :         else if (IsListOfPointStructType(fieldType, 1, bHasZ, bHasM))
    1059             :         {
    1060          70 :             eOGRArrowGeomEncodingOut =
    1061             :                 OGRArrowGeomEncoding::GEOARROW_STRUCT_LINESTRING;
    1062             :         }
    1063             :         else
    1064             :         {
    1065           0 :             CPLError(CE_Warning, CPLE_AppDefined,
    1066             :                      "Geometry column %s has a type != fixed_size_list<xy: "
    1067             :                      "double>[2]> and != list<element: struct<x: double, y: "
    1068             :                      "double>>: %s. "
    1069             :                      "Handling it as a regular field",
    1070           0 :                      fieldName.c_str(), fieldType->ToString().c_str());
    1071           0 :             return false;
    1072             :         }
    1073         122 :         eGeomTypeOut = OGR_GT_SetModifier(
    1074             :             wkbLineString, static_cast<int>(bHasZ), static_cast<int>(bHasM));
    1075         122 :         return true;
    1076             :     }
    1077             : 
    1078         571 :     else if (osEncoding == "geoarrow.polygon" || osEncoding == "polygon")
    1079             :     {
    1080         165 :         if (IsListOfPointType(fieldType, 2, bHasZ, bHasM))
    1081             :         {
    1082          63 :             eOGRArrowGeomEncodingOut =
    1083             :                 OGRArrowGeomEncoding::GEOARROW_FSL_POLYGON;
    1084             :         }
    1085         102 :         else if (IsListOfPointStructType(fieldType, 2, bHasZ, bHasM))
    1086             :         {
    1087         102 :             eOGRArrowGeomEncodingOut =
    1088             :                 OGRArrowGeomEncoding::GEOARROW_STRUCT_POLYGON;
    1089             :         }
    1090             :         else
    1091             :         {
    1092           0 :             CPLError(CE_Warning, CPLE_AppDefined,
    1093             :                      "Geometry column %s has a type != list<vertices: "
    1094             :                      "fixed_size_list<xy: double>[2]>> and != list<element: "
    1095             :                      "list<element: struct<x: double, y: double>>>: %s. "
    1096             :                      "Handling it as a regular field",
    1097           0 :                      fieldName.c_str(), fieldType->ToString().c_str());
    1098           0 :             return false;
    1099             :         }
    1100         165 :         eGeomTypeOut = OGR_GT_SetModifier(wkbPolygon, static_cast<int>(bHasZ),
    1101             :                                           static_cast<int>(bHasM));
    1102         165 :         return true;
    1103             :     }
    1104             : 
    1105         406 :     else if (osEncoding == "geoarrow.multipoint" || osEncoding == "multipoint")
    1106             :     {
    1107         122 :         if (IsListOfPointType(fieldType, 1, bHasZ, bHasM))
    1108             :         {
    1109          52 :             eOGRArrowGeomEncodingOut =
    1110             :                 OGRArrowGeomEncoding::GEOARROW_FSL_MULTIPOINT;
    1111             :         }
    1112          70 :         else if (IsListOfPointStructType(fieldType, 1, bHasZ, bHasM))
    1113             :         {
    1114          70 :             eOGRArrowGeomEncodingOut =
    1115             :                 OGRArrowGeomEncoding::GEOARROW_STRUCT_MULTIPOINT;
    1116             :         }
    1117             :         else
    1118             :         {
    1119           0 :             CPLError(CE_Warning, CPLE_AppDefined,
    1120             :                      "Geometry column %s has a type != fixed_size_list<xy: "
    1121             :                      "double>[2]> and != list<element: struct<x: double, y: "
    1122             :                      "double>>: %s. "
    1123             :                      "Handling it as a regular field",
    1124           0 :                      fieldName.c_str(), fieldType->ToString().c_str());
    1125           0 :             return false;
    1126             :         }
    1127         122 :         eGeomTypeOut = OGR_GT_SetModifier(
    1128             :             wkbMultiPoint, static_cast<int>(bHasZ), static_cast<int>(bHasM));
    1129         122 :         return true;
    1130             :     }
    1131             : 
    1132         516 :     else if (osEncoding == "geoarrow.multilinestring" ||
    1133         232 :              osEncoding == "multilinestring")
    1134             :     {
    1135         122 :         if (IsListOfPointType(fieldType, 2, bHasZ, bHasM))
    1136             :         {
    1137          52 :             eOGRArrowGeomEncodingOut =
    1138             :                 OGRArrowGeomEncoding::GEOARROW_FSL_MULTILINESTRING;
    1139             :         }
    1140          70 :         else if (IsListOfPointStructType(fieldType, 2, bHasZ, bHasM))
    1141             :         {
    1142          70 :             eOGRArrowGeomEncodingOut =
    1143             :                 OGRArrowGeomEncoding::GEOARROW_STRUCT_MULTILINESTRING;
    1144             :         }
    1145             :         else
    1146             :         {
    1147           0 :             CPLError(CE_Warning, CPLE_AppDefined,
    1148             :                      "Geometry column %s has a type != list<vertices: "
    1149             :                      "fixed_size_list<xy: double>[2]>> and != list<element: "
    1150             :                      "list<element: struct<x: double, y: double>>>: %s. "
    1151             :                      "Handling it as a regular field",
    1152           0 :                      fieldName.c_str(), fieldType->ToString().c_str());
    1153           0 :             return false;
    1154             :         }
    1155         122 :         eGeomTypeOut =
    1156         122 :             OGR_GT_SetModifier(wkbMultiLineString, static_cast<int>(bHasZ),
    1157             :                                static_cast<int>(bHasM));
    1158         122 :         return true;
    1159             :     }
    1160             : 
    1161         266 :     else if (osEncoding == "geoarrow.multipolygon" ||
    1162         104 :              osEncoding == "multipolygon")
    1163             :     {
    1164         160 :         if (IsListOfPointType(fieldType, 3, bHasZ, bHasM))
    1165             :         {
    1166          58 :             eOGRArrowGeomEncodingOut =
    1167             :                 OGRArrowGeomEncoding::GEOARROW_FSL_MULTIPOLYGON;
    1168             :         }
    1169         102 :         else if (IsListOfPointStructType(fieldType, 3, bHasZ, bHasM))
    1170             :         {
    1171         102 :             eOGRArrowGeomEncodingOut =
    1172             :                 OGRArrowGeomEncoding::GEOARROW_STRUCT_MULTIPOLYGON;
    1173             :         }
    1174             :         else
    1175             :         {
    1176           0 :             CPLError(
    1177             :                 CE_Warning, CPLE_AppDefined,
    1178             :                 "Geometry column %s has a type != list<polygons: list<rings: "
    1179             :                 "list<vertices: fixed_size_list<xy: double>[2]>>> and != "
    1180             :                 "list<element: list<element: list<element: struct<x: double, "
    1181             :                 "y: double>>>>: %s. "
    1182             :                 "Handling it as a regular field",
    1183           0 :                 fieldName.c_str(), fieldType->ToString().c_str());
    1184           0 :             return false;
    1185             :         }
    1186         160 :         eGeomTypeOut = OGR_GT_SetModifier(
    1187             :             wkbMultiPolygon, static_cast<int>(bHasZ), static_cast<int>(bHasM));
    1188         160 :         return true;
    1189             :     }
    1190             : 
    1191           2 :     if (bWarnIfUnknownEncoding)
    1192             :     {
    1193           0 :         CPLError(CE_Warning, CPLE_AppDefined,
    1194             :                  "Geometry column %s uses a unhandled encoding: %s. "
    1195             :                  "Handling it as a regular field",
    1196             :                  fieldName.c_str(), osEncoding.c_str());
    1197             :     }
    1198           2 :     return false;
    1199             : }
    1200             : 
    1201             : /************************************************************************/
    1202             : /*                    GetGeometryTypeFromString()                       */
    1203             : /************************************************************************/
    1204             : 
    1205             : inline OGRwkbGeometryType
    1206         263 : OGRArrowLayer::GetGeometryTypeFromString(const std::string &osType)
    1207             : {
    1208         263 :     OGRwkbGeometryType eGeomType = wkbUnknown;
    1209         263 :     OGRReadWKTGeometryType(osType.c_str(), &eGeomType);
    1210         263 :     if (eGeomType == wkbUnknown && !osType.empty())
    1211             :     {
    1212           0 :         CPLDebug("ARROW", "Unknown geometry type: %s", osType.c_str());
    1213             :     }
    1214         263 :     return eGeomType;
    1215             : }
    1216             : 
    1217             : static CPLJSONObject GetObjectAsJSON(const arrow::Array *array,
    1218             :                                      const size_t nIdx);
    1219             : 
    1220             : /************************************************************************/
    1221             : /*                               AddToArray()                           */
    1222             : /************************************************************************/
    1223             : 
    1224        6342 : static void AddToArray(CPLJSONArray &oArray, const arrow::Array *array,
    1225             :                        const size_t nIdx)
    1226             : {
    1227        6342 :     switch (array->type()->id())
    1228             :     {
    1229           9 :         case arrow::Type::BOOL:
    1230             :         {
    1231           9 :             oArray.Add(
    1232           9 :                 static_cast<const arrow::BooleanArray *>(array)->Value(nIdx));
    1233           9 :             break;
    1234             :         }
    1235           9 :         case arrow::Type::UINT8:
    1236             :         {
    1237           9 :             oArray.Add(
    1238           9 :                 static_cast<const arrow::UInt8Array *>(array)->Value(nIdx));
    1239           9 :             break;
    1240             :         }
    1241           9 :         case arrow::Type::INT8:
    1242             :         {
    1243           9 :             oArray.Add(
    1244           9 :                 static_cast<const arrow::Int8Array *>(array)->Value(nIdx));
    1245           9 :             break;
    1246             :         }
    1247           9 :         case arrow::Type::UINT16:
    1248             :         {
    1249           9 :             oArray.Add(
    1250           9 :                 static_cast<const arrow::UInt16Array *>(array)->Value(nIdx));
    1251           9 :             break;
    1252             :         }
    1253           9 :         case arrow::Type::INT16:
    1254             :         {
    1255           9 :             oArray.Add(
    1256           9 :                 static_cast<const arrow::Int16Array *>(array)->Value(nIdx));
    1257           9 :             break;
    1258             :         }
    1259           9 :         case arrow::Type::INT32:
    1260             :         {
    1261           9 :             oArray.Add(
    1262             :                 static_cast<const arrow::Int32Array *>(array)->Value(nIdx));
    1263           9 :             break;
    1264             :         }
    1265           9 :         case arrow::Type::UINT32:
    1266             :         {
    1267           9 :             oArray.Add(static_cast<GInt64>(
    1268           9 :                 static_cast<const arrow::UInt32Array *>(array)->Value(nIdx)));
    1269           9 :             break;
    1270             :         }
    1271           9 :         case arrow::Type::INT64:
    1272             :         {
    1273           9 :             oArray.Add(static_cast<GInt64>(
    1274           9 :                 static_cast<const arrow::Int64Array *>(array)->Value(nIdx)));
    1275           9 :             break;
    1276             :         }
    1277           9 :         case arrow::Type::UINT64:
    1278             :         {
    1279           9 :             oArray.Add(static_cast<uint64_t>(
    1280           9 :                 static_cast<const arrow::UInt64Array *>(array)->Value(nIdx)));
    1281           9 :             break;
    1282             :         }
    1283         225 :         case arrow::Type::HALF_FLOAT:
    1284             :         {
    1285             :             const uint16_t nFloat16 =
    1286         225 :                 static_cast<const arrow::HalfFloatArray *>(array)->Value(nIdx);
    1287         225 :             uint32_t nFloat32 = CPLHalfToFloat(nFloat16);
    1288             :             float f;
    1289         225 :             memcpy(&f, &nFloat32, sizeof(nFloat32));
    1290         225 :             oArray.Add(f);
    1291         225 :             break;
    1292             :         }
    1293           9 :         case arrow::Type::FLOAT:
    1294             :         {
    1295           9 :             oArray.Add(
    1296           9 :                 static_cast<const arrow::FloatArray *>(array)->Value(nIdx));
    1297           9 :             break;
    1298             :         }
    1299           9 :         case arrow::Type::DOUBLE:
    1300             :         {
    1301           9 :             oArray.Add(
    1302             :                 static_cast<const arrow::DoubleArray *>(array)->Value(nIdx));
    1303           9 :             break;
    1304             :         }
    1305             : 
    1306             : #if ARROW_VERSION_MAJOR >= 18
    1307           0 :         case arrow::Type::DECIMAL32:
    1308             :         {
    1309           0 :             oArray.Add(CPLAtof(static_cast<const arrow::Decimal32Array *>(array)
    1310           0 :                                    ->FormatValue(nIdx)
    1311             :                                    .c_str()));
    1312           0 :             break;
    1313             :         }
    1314           0 :         case arrow::Type::DECIMAL64:
    1315             :         {
    1316           0 :             oArray.Add(CPLAtof(static_cast<const arrow::Decimal64Array *>(array)
    1317           0 :                                    ->FormatValue(nIdx)
    1318             :                                    .c_str()));
    1319           0 :             break;
    1320             :         }
    1321             : #endif
    1322           4 :         case arrow::Type::DECIMAL128:
    1323             :         {
    1324           4 :             oArray.Add(
    1325             :                 CPLAtof(static_cast<const arrow::Decimal128Array *>(array)
    1326           8 :                             ->FormatValue(nIdx)
    1327             :                             .c_str()));
    1328           4 :             break;
    1329             :         }
    1330           4 :         case arrow::Type::DECIMAL256:
    1331             :         {
    1332           4 :             oArray.Add(
    1333             :                 CPLAtof(static_cast<const arrow::Decimal256Array *>(array)
    1334           8 :                             ->FormatValue(nIdx)
    1335             :                             .c_str()));
    1336           4 :             break;
    1337             :         }
    1338        2182 :         case arrow::Type::STRING:
    1339             :         {
    1340        2182 :             oArray.Add(
    1341        4364 :                 static_cast<const arrow::StringArray *>(array)->GetString(
    1342             :                     nIdx));
    1343        2182 :             break;
    1344             :         }
    1345           4 :         case arrow::Type::LARGE_STRING:
    1346             :         {
    1347           4 :             oArray.Add(
    1348           8 :                 static_cast<const arrow::LargeStringArray *>(array)->GetString(
    1349             :                     nIdx));
    1350           4 :             break;
    1351             :         }
    1352             : #if ARROW_VERSION_MAJOR >= 15
    1353           3 :         case arrow::Type::STRING_VIEW:
    1354             :         {
    1355           3 :             oArray.Add(
    1356           6 :                 static_cast<const arrow::StringViewArray *>(array)->GetString(
    1357             :                     nIdx));
    1358           3 :             break;
    1359             :         }
    1360             : #endif
    1361        3821 :         case arrow::Type::LIST:
    1362             :         case arrow::Type::LARGE_LIST:
    1363             :         case arrow::Type::FIXED_SIZE_LIST:
    1364             :         case arrow::Type::MAP:
    1365             :         case arrow::Type::STRUCT:
    1366             :         {
    1367        3821 :             oArray.Add(GetObjectAsJSON(array, nIdx));
    1368        3821 :             break;
    1369             :         }
    1370             : 
    1371           0 :         default:
    1372             :         {
    1373           0 :             CPLDebug("ARROW", "AddToArray(): unexpected data type %s",
    1374           0 :                      array->type()->ToString().c_str());
    1375           0 :             break;
    1376             :         }
    1377             :     }
    1378        6342 : }
    1379             : 
    1380             : /************************************************************************/
    1381             : /*                         GetListAsJSON()                              */
    1382             : /************************************************************************/
    1383             : 
    1384             : template <class ArrowType>
    1385        4922 : static CPLJSONArray GetListAsJSON(const ArrowType *array,
    1386             :                                   const size_t nIdxInArray)
    1387             : {
    1388        9844 :     const auto values = std::static_pointer_cast<ArrowType>(array->values());
    1389        4922 :     const auto nIdxStart = array->value_offset(nIdxInArray);
    1390        4922 :     const auto nCount = array->value_length(nIdxInArray);
    1391        4922 :     CPLJSONArray oArray;
    1392       13592 :     for (auto k = decltype(nCount){0}; k < nCount; k++)
    1393             :     {
    1394        8670 :         if (values->IsNull(nIdxStart + k))
    1395        2328 :             oArray.AddNull();
    1396             :         else
    1397        6342 :             AddToArray(oArray, values.get(),
    1398        6342 :                        static_cast<size_t>(nIdxStart + k));
    1399             :     }
    1400        9844 :     return oArray;
    1401             : }
    1402             : 
    1403             : /************************************************************************/
    1404             : /*                              AddToDict()                             */
    1405             : /************************************************************************/
    1406             : 
    1407       20386 : static void AddToDict(CPLJSONObject &oDict, const std::string &osKey,
    1408             :                       const arrow::Array *array, const size_t nIdx)
    1409             : {
    1410       20386 :     switch (array->type()->id())
    1411             :     {
    1412         718 :         case arrow::Type::BOOL:
    1413             :         {
    1414         718 :             oDict.Add(
    1415             :                 osKey,
    1416         718 :                 static_cast<const arrow::BooleanArray *>(array)->Value(nIdx));
    1417         718 :             break;
    1418             :         }
    1419         725 :         case arrow::Type::UINT8:
    1420             :         {
    1421         725 :             oDict.Add(
    1422             :                 osKey,
    1423         725 :                 static_cast<const arrow::UInt8Array *>(array)->Value(nIdx));
    1424         725 :             break;
    1425             :         }
    1426         725 :         case arrow::Type::INT8:
    1427             :         {
    1428         725 :             oDict.Add(
    1429             :                 osKey,
    1430         725 :                 static_cast<const arrow::Int8Array *>(array)->Value(nIdx));
    1431         725 :             break;
    1432             :         }
    1433         725 :         case arrow::Type::UINT16:
    1434             :         {
    1435         725 :             oDict.Add(
    1436             :                 osKey,
    1437         725 :                 static_cast<const arrow::UInt16Array *>(array)->Value(nIdx));
    1438         725 :             break;
    1439             :         }
    1440         725 :         case arrow::Type::INT16:
    1441             :         {
    1442         725 :             oDict.Add(
    1443             :                 osKey,
    1444         725 :                 static_cast<const arrow::Int16Array *>(array)->Value(nIdx));
    1445         725 :             break;
    1446             :         }
    1447         725 :         case arrow::Type::INT32:
    1448             :         {
    1449         725 :             oDict.Add(
    1450             :                 osKey,
    1451             :                 static_cast<const arrow::Int32Array *>(array)->Value(nIdx));
    1452         725 :             break;
    1453             :         }
    1454         108 :         case arrow::Type::UINT32:
    1455             :         {
    1456         108 :             oDict.Add(osKey,
    1457             :                       static_cast<GInt64>(
    1458         108 :                           static_cast<const arrow::UInt32Array *>(array)->Value(
    1459             :                               nIdx)));
    1460         108 :             break;
    1461             :         }
    1462        4864 :         case arrow::Type::INT64:
    1463             :         {
    1464        4864 :             oDict.Add(osKey,
    1465             :                       static_cast<GInt64>(
    1466        4864 :                           static_cast<const arrow::Int64Array *>(array)->Value(
    1467             :                               nIdx)));
    1468        4864 :             break;
    1469             :         }
    1470         725 :         case arrow::Type::UINT64:
    1471             :         {
    1472         725 :             oDict.Add(osKey,
    1473             :                       static_cast<uint64_t>(
    1474         725 :                           static_cast<const arrow::UInt64Array *>(array)->Value(
    1475             :                               nIdx)));
    1476         725 :             break;
    1477             :         }
    1478         105 :         case arrow::Type::HALF_FLOAT:
    1479             :         {
    1480             :             const uint16_t nFloat16 =
    1481         105 :                 static_cast<const arrow::HalfFloatArray *>(array)->Value(nIdx);
    1482         105 :             uint32_t nFloat32 = CPLHalfToFloat(nFloat16);
    1483             :             float f;
    1484         105 :             memcpy(&f, &nFloat32, sizeof(nFloat32));
    1485         105 :             oDict.Add(osKey, f);
    1486         105 :             break;
    1487             :         }
    1488         725 :         case arrow::Type::FLOAT:
    1489             :         {
    1490         725 :             oDict.Add(
    1491             :                 osKey,
    1492         725 :                 static_cast<const arrow::FloatArray *>(array)->Value(nIdx));
    1493         725 :             break;
    1494             :         }
    1495        3939 :         case arrow::Type::DOUBLE:
    1496             :         {
    1497        3939 :             oDict.Add(
    1498             :                 osKey,
    1499             :                 static_cast<const arrow::DoubleArray *>(array)->Value(nIdx));
    1500        3939 :             break;
    1501             :         }
    1502             : 
    1503             : #if ARROW_VERSION_MAJOR >= 18
    1504           0 :         case arrow::Type::DECIMAL32:
    1505             :         {
    1506           0 :             oDict.Add(osKey,
    1507             :                       CPLAtof(static_cast<const arrow::Decimal32Array *>(array)
    1508           0 :                                   ->FormatValue(nIdx)
    1509             :                                   .c_str()));
    1510           0 :             break;
    1511             :         }
    1512           0 :         case arrow::Type::DECIMAL64:
    1513             :         {
    1514           0 :             oDict.Add(osKey,
    1515             :                       CPLAtof(static_cast<const arrow::Decimal64Array *>(array)
    1516           0 :                                   ->FormatValue(nIdx)
    1517             :                                   .c_str()));
    1518           0 :             break;
    1519             :         }
    1520             : #endif
    1521        1339 :         case arrow::Type::DECIMAL128:
    1522             :         {
    1523        1339 :             oDict.Add(osKey,
    1524             :                       CPLAtof(static_cast<const arrow::Decimal128Array *>(array)
    1525        2678 :                                   ->FormatValue(nIdx)
    1526             :                                   .c_str()));
    1527        1339 :             break;
    1528             :         }
    1529         105 :         case arrow::Type::DECIMAL256:
    1530             :         {
    1531         105 :             oDict.Add(osKey,
    1532             :                       CPLAtof(static_cast<const arrow::Decimal256Array *>(array)
    1533         210 :                                   ->FormatValue(nIdx)
    1534             :                                   .c_str()));
    1535         105 :             break;
    1536             :         }
    1537        1397 :         case arrow::Type::STRING:
    1538             :         {
    1539        1397 :             oDict.Add(osKey,
    1540        2794 :                       static_cast<const arrow::StringArray *>(array)->GetString(
    1541             :                           nIdx));
    1542        1397 :             break;
    1543             :         }
    1544         105 :         case arrow::Type::LARGE_STRING:
    1545             :         {
    1546         105 :             oDict.Add(osKey, static_cast<const arrow::LargeStringArray *>(array)
    1547         210 :                                  ->GetString(nIdx));
    1548         105 :             break;
    1549             :         }
    1550             : #if ARROW_VERSION_MAJOR >= 15
    1551           1 :         case arrow::Type::STRING_VIEW:
    1552             :         {
    1553           1 :             oDict.Add(osKey, static_cast<const arrow::StringViewArray *>(array)
    1554           2 :                                  ->GetString(nIdx));
    1555           1 :             break;
    1556             :         }
    1557             : #endif
    1558        2630 :         case arrow::Type::LIST:
    1559             :         case arrow::Type::LARGE_LIST:
    1560             :         case arrow::Type::FIXED_SIZE_LIST:
    1561             :         case arrow::Type::MAP:
    1562             :         case arrow::Type::STRUCT:
    1563             :         {
    1564        2630 :             oDict.Add(osKey, GetObjectAsJSON(array, nIdx));
    1565        2630 :             break;
    1566             :         }
    1567             : 
    1568           0 :         default:
    1569             :         {
    1570           0 :             CPLDebug("ARROW", "AddToDict(): unexpected data type %s",
    1571           0 :                      array->type()->ToString().c_str());
    1572           0 :             break;
    1573             :         }
    1574             :     }
    1575       20386 : }
    1576             : 
    1577             : /************************************************************************/
    1578             : /*                         GetMapAsJSON()                               */
    1579             : /************************************************************************/
    1580             : 
    1581             : template <class KeyArrayType>
    1582       24926 : static CPLJSONObject GetMapAsJSON(const arrow::Array *array,
    1583             :                                   const size_t nIdxInArray)
    1584             : {
    1585       24926 :     const auto mapArray = static_cast<const arrow::MapArray *>(array);
    1586       49852 :     const auto keys = std::static_pointer_cast<KeyArrayType>(mapArray->keys());
    1587       49852 :     const auto values = mapArray->items();
    1588       24926 :     const auto nIdxStart = mapArray->value_offset(nIdxInArray);
    1589       24926 :     const int nCount = mapArray->value_length(nIdxInArray);
    1590       24926 :     CPLJSONObject oRoot;
    1591       45342 :     for (int k = 0; k < nCount; k++)
    1592             :     {
    1593       20416 :         if (!keys->IsNull(nIdxStart + k))
    1594             :         {
    1595       40832 :             const auto osKey = keys->GetString(nIdxStart + k);
    1596       20416 :             if (!values->IsNull(nIdxStart + k))
    1597       13597 :                 AddToDict(oRoot, osKey, values.get(), nIdxStart + k);
    1598             :             else
    1599        6819 :                 oRoot.AddNull(osKey);
    1600             :         }
    1601             :     }
    1602       49852 :     return oRoot;
    1603             : }
    1604             : 
    1605       24926 : static CPLJSONObject GetMapAsJSON(const arrow::Array *array,
    1606             :                                   const size_t nIdxInArray)
    1607             : {
    1608       24926 :     const auto mapArray = static_cast<const arrow::MapArray *>(array);
    1609       24926 :     const auto eKeyType = mapArray->keys()->type()->id();
    1610       24926 :     if (eKeyType == arrow::Type::STRING)
    1611       24924 :         return GetMapAsJSON<arrow::StringArray>(array, nIdxInArray);
    1612             : #if ARROW_VERSION_MAJOR >= 15
    1613           2 :     else if (eKeyType == arrow::Type::STRING_VIEW)
    1614           2 :         return GetMapAsJSON<arrow::StringViewArray>(array, nIdxInArray);
    1615             : #endif
    1616             :     else
    1617             :     {
    1618           0 :         CPLAssert(false);
    1619             :         return CPLJSONObject();
    1620             :     }
    1621             : }
    1622             : 
    1623             : /************************************************************************/
    1624             : /*                        GetStructureAsJSON()                          */
    1625             : /************************************************************************/
    1626             : 
    1627        3541 : static CPLJSONObject GetStructureAsJSON(const arrow::Array *array,
    1628             :                                         const size_t nIdxInArray)
    1629             : {
    1630        3541 :     CPLJSONObject oRoot;
    1631        3541 :     const auto structArray = static_cast<const arrow::StructArray *>(array);
    1632        7082 :     const auto structArrayType = structArray->type();
    1633       13559 :     for (int i = 0; i < structArrayType->num_fields(); ++i)
    1634             :     {
    1635       20036 :         const auto field = structArray->field(i);
    1636       10018 :         if (!field->IsNull(nIdxInArray))
    1637             :         {
    1638        6789 :             AddToDict(oRoot, structArrayType->field(i)->name(), field.get(),
    1639             :                       nIdxInArray);
    1640             :         }
    1641             :         else
    1642        3229 :             oRoot.AddNull(structArrayType->field(i)->name());
    1643             :     }
    1644             : 
    1645        7082 :     return oRoot;
    1646             : }
    1647             : 
    1648             : /************************************************************************/
    1649             : /*                        GetObjectAsJSON()                             */
    1650             : /************************************************************************/
    1651             : 
    1652        6451 : static CPLJSONObject GetObjectAsJSON(const arrow::Array *array,
    1653             :                                      const size_t nIdxInArray)
    1654             : {
    1655        6451 :     switch (array->type()->id())
    1656             :     {
    1657          26 :         case arrow::Type::MAP:
    1658          26 :             return GetMapAsJSON(array, nIdxInArray);
    1659        2601 :         case arrow::Type::LIST:
    1660        5202 :             return GetListAsJSON(static_cast<const arrow::ListArray *>(array),
    1661        2601 :                                  nIdxInArray);
    1662         108 :         case arrow::Type::LARGE_LIST:
    1663         216 :             return GetListAsJSON(
    1664         108 :                 static_cast<const arrow::LargeListArray *>(array), nIdxInArray);
    1665         175 :         case arrow::Type::FIXED_SIZE_LIST:
    1666         350 :             return GetListAsJSON(
    1667             :                 static_cast<const arrow::FixedSizeListArray *>(array),
    1668         175 :                 nIdxInArray);
    1669        3541 :         case arrow::Type::STRUCT:
    1670        3541 :             return GetStructureAsJSON(array, nIdxInArray);
    1671           0 :         default:
    1672             :         {
    1673           0 :             CPLError(CE_Failure, CPLE_AppDefined,
    1674             :                      "GetObjectAsJSON(): unhandled value format: %s",
    1675           0 :                      array->type()->ToString().c_str());
    1676           0 :             return CPLJSONObject();
    1677             :         }
    1678             :     }
    1679             : }
    1680             : 
    1681             : template <class OGRType, class ArrowType, class ArrayType>
    1682       32323 : static void ReadList(OGRFeature *poFeature, int i, int64_t nIdxInArray,
    1683             :                      const ArrayType *array)
    1684             : {
    1685       64646 :     const auto values = std::static_pointer_cast<ArrowType>(array->values());
    1686       32323 :     const auto nIdxStart = array->value_offset(nIdxInArray);
    1687       32323 :     const int nCount = array->value_length(nIdxInArray);
    1688       64646 :     std::vector<OGRType> aValues;
    1689       32323 :     aValues.reserve(nCount);
    1690       96781 :     for (int k = 0; k < nCount; k++)
    1691             :     {
    1692       64458 :         aValues.push_back(static_cast<OGRType>(values->Value(nIdxStart + k)));
    1693             :     }
    1694       32323 :     poFeature->SetField(i, nCount, aValues.data());
    1695       32323 : }
    1696             : 
    1697             : template <class ArrowType, class ArrayType>
    1698        7203 : static void ReadListDouble(OGRFeature *poFeature, int i, int64_t nIdxInArray,
    1699             :                            const ArrayType *array)
    1700             : {
    1701       14406 :     const auto values = std::static_pointer_cast<ArrowType>(array->values());
    1702        7203 :     const auto rawValues = values->raw_values();
    1703        7203 :     const auto nIdxStart = array->value_offset(nIdxInArray);
    1704        7203 :     const int nCount = array->value_length(nIdxInArray);
    1705       14406 :     std::vector<double> aValues;
    1706        7203 :     aValues.reserve(nCount);
    1707       21099 :     for (int k = 0; k < nCount; k++)
    1708             :     {
    1709       13896 :         if (values->IsNull(nIdxStart + k))
    1710        2948 :             aValues.push_back(std::numeric_limits<double>::quiet_NaN());
    1711             :         else
    1712       10948 :             aValues.push_back(rawValues[nIdxStart + k]);
    1713             :     }
    1714        7203 :     poFeature->SetField(i, nCount, aValues.data());
    1715        7203 : }
    1716             : 
    1717             : template <class ArrayType>
    1718       49675 : static void ReadList(OGRFeature *poFeature, int i, int64_t nIdxInArray,
    1719             :                      const ArrayType *array, arrow::Type::type valueTypeId)
    1720             : {
    1721       49675 :     switch (valueTypeId)
    1722             :     {
    1723        3405 :         case arrow::Type::BOOL:
    1724             :         {
    1725        3405 :             ReadList<int, arrow::BooleanArray>(poFeature, i, nIdxInArray,
    1726             :                                                array);
    1727        3405 :             break;
    1728             :         }
    1729        3235 :         case arrow::Type::UINT8:
    1730             :         {
    1731        3235 :             ReadList<int, arrow::UInt8Array>(poFeature, i, nIdxInArray, array);
    1732        3235 :             break;
    1733             :         }
    1734        3235 :         case arrow::Type::INT8:
    1735             :         {
    1736        3235 :             ReadList<int, arrow::Int8Array>(poFeature, i, nIdxInArray, array);
    1737        3235 :             break;
    1738             :         }
    1739        3235 :         case arrow::Type::UINT16:
    1740             :         {
    1741        3235 :             ReadList<int, arrow::UInt16Array>(poFeature, i, nIdxInArray, array);
    1742        3235 :             break;
    1743             :         }
    1744        3235 :         case arrow::Type::INT16:
    1745             :         {
    1746        3235 :             ReadList<int, arrow::Int16Array>(poFeature, i, nIdxInArray, array);
    1747        3235 :             break;
    1748             :         }
    1749        4085 :         case arrow::Type::INT32:
    1750             :         {
    1751        4085 :             ReadList<int, arrow::Int32Array>(poFeature, i, nIdxInArray, array);
    1752        4085 :             break;
    1753             :         }
    1754         429 :         case arrow::Type::UINT32:
    1755             :         {
    1756         429 :             ReadList<GIntBig, arrow::UInt32Array>(poFeature, i, nIdxInArray,
    1757             :                                                   array);
    1758         429 :             break;
    1759             :         }
    1760        8229 :         case arrow::Type::INT64:
    1761             :         {
    1762        8229 :             ReadList<GIntBig, arrow::Int64Array>(poFeature, i, nIdxInArray,
    1763             :                                                  array);
    1764        8229 :             break;
    1765             :         }
    1766        3235 :         case arrow::Type::UINT64:
    1767             :         {
    1768        3235 :             ReadList<double, arrow::UInt64Array>(poFeature, i, nIdxInArray,
    1769             :                                                  array);
    1770        3235 :             break;
    1771             :         }
    1772         194 :         case arrow::Type::HALF_FLOAT:
    1773             :         {
    1774         388 :             const auto values = std::static_pointer_cast<arrow::HalfFloatArray>(
    1775             :                 array->values());
    1776         194 :             const auto nIdxStart = array->value_offset(nIdxInArray);
    1777         194 :             const int nCount = array->value_length(nIdxInArray);
    1778         388 :             std::vector<double> aValues;
    1779         194 :             aValues.reserve(nCount);
    1780         547 :             for (int k = 0; k < nCount; k++)
    1781             :             {
    1782         353 :                 if (values->IsNull(nIdxStart + k))
    1783         128 :                     aValues.push_back(std::numeric_limits<double>::quiet_NaN());
    1784             :                 else
    1785             :                 {
    1786         225 :                     const uint16_t nFloat16 = values->Value(nIdxStart + k);
    1787         225 :                     uint32_t nFloat32 = CPLHalfToFloat(nFloat16);
    1788             :                     float f;
    1789         225 :                     memcpy(&f, &nFloat32, sizeof(nFloat32));
    1790         225 :                     aValues.push_back(f);
    1791             :                 }
    1792             :             }
    1793         194 :             poFeature->SetField(i, nCount, aValues.data());
    1794         194 :             break;
    1795             :         }
    1796        3466 :         case arrow::Type::FLOAT:
    1797             :         {
    1798        3466 :             ReadListDouble<arrow::FloatArray>(poFeature, i, nIdxInArray, array);
    1799        3466 :             break;
    1800             :         }
    1801        3737 :         case arrow::Type::DOUBLE:
    1802             :         {
    1803        3737 :             ReadListDouble<arrow::DoubleArray>(poFeature, i, nIdxInArray,
    1804             :                                                array);
    1805        3737 :             break;
    1806             :         }
    1807             : 
    1808             : #if ARROW_VERSION_MAJOR >= 18
    1809           0 :         case arrow::Type::DECIMAL32:
    1810             :         {
    1811           0 :             const auto values = std::static_pointer_cast<arrow::Decimal32Array>(
    1812             :                 array->values());
    1813           0 :             const auto nIdxStart = array->value_offset(nIdxInArray);
    1814           0 :             const int nCount = array->value_length(nIdxInArray);
    1815           0 :             std::vector<double> aValues;
    1816           0 :             aValues.reserve(nCount);
    1817           0 :             for (int k = 0; k < nCount; k++)
    1818             :             {
    1819           0 :                 if (values->IsNull(nIdxStart + k))
    1820           0 :                     aValues.push_back(std::numeric_limits<double>::quiet_NaN());
    1821             :                 else
    1822           0 :                     aValues.push_back(
    1823           0 :                         CPLAtof(values->FormatValue(nIdxStart + k).c_str()));
    1824             :             }
    1825           0 :             poFeature->SetField(i, nCount, aValues.data());
    1826           0 :             break;
    1827             :         }
    1828             : 
    1829           0 :         case arrow::Type::DECIMAL64:
    1830             :         {
    1831           0 :             const auto values = std::static_pointer_cast<arrow::Decimal64Array>(
    1832             :                 array->values());
    1833           0 :             const auto nIdxStart = array->value_offset(nIdxInArray);
    1834           0 :             const int nCount = array->value_length(nIdxInArray);
    1835           0 :             std::vector<double> aValues;
    1836           0 :             aValues.reserve(nCount);
    1837           0 :             for (int k = 0; k < nCount; k++)
    1838             :             {
    1839           0 :                 if (values->IsNull(nIdxStart + k))
    1840           0 :                     aValues.push_back(std::numeric_limits<double>::quiet_NaN());
    1841             :                 else
    1842           0 :                     aValues.push_back(
    1843           0 :                         CPLAtof(values->FormatValue(nIdxStart + k).c_str()));
    1844             :             }
    1845           0 :             poFeature->SetField(i, nCount, aValues.data());
    1846           0 :             break;
    1847             :         }
    1848             : #endif
    1849             : 
    1850        1474 :         case arrow::Type::DECIMAL128:
    1851             :         {
    1852        2948 :             const auto values =
    1853             :                 std::static_pointer_cast<arrow::Decimal128Array>(
    1854             :                     array->values());
    1855        1474 :             const auto nIdxStart = array->value_offset(nIdxInArray);
    1856        1474 :             const int nCount = array->value_length(nIdxInArray);
    1857        2948 :             std::vector<double> aValues;
    1858        1474 :             aValues.reserve(nCount);
    1859        2948 :             for (int k = 0; k < nCount; k++)
    1860             :             {
    1861        1474 :                 if (values->IsNull(nIdxStart + k))
    1862         344 :                     aValues.push_back(std::numeric_limits<double>::quiet_NaN());
    1863             :                 else
    1864        1130 :                     aValues.push_back(
    1865        1130 :                         CPLAtof(values->FormatValue(nIdxStart + k).c_str()));
    1866             :             }
    1867        1474 :             poFeature->SetField(i, nCount, aValues.data());
    1868        1474 :             break;
    1869             :         }
    1870             : 
    1871        1474 :         case arrow::Type::DECIMAL256:
    1872             :         {
    1873        2948 :             const auto values =
    1874             :                 std::static_pointer_cast<arrow::Decimal256Array>(
    1875             :                     array->values());
    1876        1474 :             const auto nIdxStart = array->value_offset(nIdxInArray);
    1877        1474 :             const int nCount = array->value_length(nIdxInArray);
    1878        2948 :             std::vector<double> aValues;
    1879        1474 :             aValues.reserve(nCount);
    1880        2948 :             for (int k = 0; k < nCount; k++)
    1881             :             {
    1882        1474 :                 if (values->IsNull(nIdxStart + k))
    1883         344 :                     aValues.push_back(std::numeric_limits<double>::quiet_NaN());
    1884             :                 else
    1885        1130 :                     aValues.push_back(
    1886        1130 :                         CPLAtof(values->FormatValue(nIdxStart + k).c_str()));
    1887             :             }
    1888        1474 :             poFeature->SetField(i, nCount, aValues.data());
    1889        1474 :             break;
    1890             :         }
    1891             : 
    1892        3493 :         case arrow::Type::STRING:
    1893             :         {
    1894        6986 :             const auto values =
    1895             :                 std::static_pointer_cast<arrow::StringArray>(array->values());
    1896        3493 :             const auto nIdxStart = array->value_offset(nIdxInArray);
    1897        3493 :             const int nCount = array->value_length(nIdxInArray);
    1898        6986 :             CPLStringList aosList;
    1899        9104 :             for (int k = 0; k < nCount; k++)
    1900             :             {
    1901        5611 :                 if (values->IsNull(nIdxStart + k))
    1902         352 :                     aosList.AddString(
    1903             :                         "");  // we cannot have null strings in a list
    1904             :                 else
    1905        5259 :                     aosList.AddString(values->GetString(nIdxStart + k).c_str());
    1906             :             }
    1907        3493 :             poFeature->SetField(i, aosList.List());
    1908        3493 :             break;
    1909             :         }
    1910        1474 :         case arrow::Type::LARGE_STRING:
    1911             :         {
    1912        2948 :             const auto values =
    1913             :                 std::static_pointer_cast<arrow::LargeStringArray>(
    1914             :                     array->values());
    1915        1474 :             const auto nIdxStart = array->value_offset(nIdxInArray);
    1916        1474 :             const auto nCount = array->value_length(nIdxInArray);
    1917        2948 :             CPLStringList aosList;
    1918        3215 :             for (auto k = decltype(nCount){0}; k < nCount; k++)
    1919             :             {
    1920        1741 :                 if (values->IsNull(nIdxStart + k))
    1921         352 :                     aosList.AddString(
    1922             :                         "");  // we cannot have null strings in a list
    1923             :                 else
    1924        1389 :                     aosList.AddString(values->GetString(nIdxStart + k).c_str());
    1925             :             }
    1926        1474 :             poFeature->SetField(i, aosList.List());
    1927        1474 :             break;
    1928             :         }
    1929             : #if ARROW_VERSION_MAJOR >= 15
    1930           2 :         case arrow::Type::STRING_VIEW:
    1931             :         {
    1932           4 :             const auto values =
    1933             :                 std::static_pointer_cast<arrow::StringViewArray>(
    1934             :                     array->values());
    1935           2 :             const auto nIdxStart = array->value_offset(nIdxInArray);
    1936           2 :             const int nCount = array->value_length(nIdxInArray);
    1937           4 :             CPLStringList aosList;
    1938           6 :             for (int k = 0; k < nCount; k++)
    1939             :             {
    1940           4 :                 if (values->IsNull(nIdxStart + k))
    1941           1 :                     aosList.AddString(
    1942             :                         "");  // we cannot have null strings in a list
    1943             :                 else
    1944           3 :                     aosList.AddString(values->GetString(nIdxStart + k).c_str());
    1945             :             }
    1946           2 :             poFeature->SetField(i, aosList.List());
    1947           2 :             break;
    1948             :         }
    1949             : #endif
    1950        2038 :         case arrow::Type::LIST:
    1951             :         case arrow::Type::LARGE_LIST:
    1952             :         case arrow::Type::FIXED_SIZE_LIST:
    1953             :         case arrow::Type::MAP:
    1954             :         case arrow::Type::STRUCT:
    1955             :         {
    1956        2038 :             poFeature->SetField(
    1957             :                 i, GetListAsJSON(array, static_cast<size_t>(nIdxInArray))
    1958             :                        .Format(CPLJSONObject::PrettyFormat::Plain)
    1959             :                        .c_str());
    1960        2038 :             break;
    1961             :         }
    1962             : 
    1963           0 :         default:
    1964             :         {
    1965           0 :             CPLDebug("ARROW", "ReadList(): unexpected data type %s",
    1966           0 :                      array->values()->type()->ToString().c_str());
    1967           0 :             break;
    1968             :         }
    1969             :     }
    1970       49675 : }
    1971             : 
    1972             : /************************************************************************/
    1973             : /*                         SetPointsOfLine()                            */
    1974             : /************************************************************************/
    1975             : 
    1976             : template <bool bHasZ, bool bHasM, int nDim>
    1977         948 : void SetPointsOfLine(OGRLineString *poLS, const arrow::DoubleArray *pointValues,
    1978             :                      size_t pointOffset, int numPoints)
    1979             : {
    1980             :     if (!bHasZ && !bHasM)
    1981             :     {
    1982             :         static_assert(sizeof(OGRRawPoint) == 2 * sizeof(double),
    1983             :                       "sizeof(OGRRawPoint) == 2 * sizeof(double)");
    1984        1216 :         poLS->setPoints(numPoints,
    1985             :                         reinterpret_cast<const OGRRawPoint *>(
    1986         608 :                             pointValues->raw_values() + pointOffset));
    1987         608 :         return;
    1988             :     }
    1989             : 
    1990         340 :     poLS->setNumPoints(numPoints, FALSE);
    1991        1634 :     for (int k = 0; k < numPoints; k++)
    1992             :     {
    1993             :         if constexpr (bHasZ)
    1994             :         {
    1995             :             if constexpr (bHasM)
    1996             :             {
    1997         330 :                 poLS->setPoint(k, pointValues->Value(pointOffset + nDim * k),
    1998         330 :                                pointValues->Value(pointOffset + nDim * k + 1),
    1999         330 :                                pointValues->Value(pointOffset + nDim * k + 2),
    2000         330 :                                pointValues->Value(pointOffset + nDim * k + 3));
    2001             :             }
    2002             :             else
    2003             :             {
    2004         634 :                 poLS->setPoint(k, pointValues->Value(pointOffset + nDim * k),
    2005         634 :                                pointValues->Value(pointOffset + nDim * k + 1),
    2006         634 :                                pointValues->Value(pointOffset + nDim * k + 2));
    2007             :             }
    2008             :         }
    2009             :         else /* if( bHasM ) */
    2010             :         {
    2011         330 :             poLS->setPointM(k, pointValues->Value(pointOffset + nDim * k),
    2012         330 :                             pointValues->Value(pointOffset + nDim * k + 1),
    2013         330 :                             pointValues->Value(pointOffset + nDim * k + 2));
    2014             :         }
    2015             :     }
    2016             : }
    2017             : 
    2018             : typedef void (*SetPointsOfLineType)(OGRLineString *, const arrow::DoubleArray *,
    2019             :                                     size_t, int);
    2020             : 
    2021         824 : static SetPointsOfLineType GetSetPointsOfLine(bool bHasZ, bool bHasM)
    2022             : {
    2023         824 :     if (bHasZ && bHasM)
    2024          66 :         return SetPointsOfLine<true, true, 4>;
    2025         758 :     if (bHasZ)
    2026         168 :         return SetPointsOfLine<true, false, 3>;
    2027         590 :     if (bHasM)
    2028          66 :         return SetPointsOfLine<false, true, 3>;
    2029         524 :     return SetPointsOfLine<false, false, 2>;
    2030             : }
    2031             : 
    2032             : /************************************************************************/
    2033             : /*                        SetPointsOfLineStruct()                       */
    2034             : /************************************************************************/
    2035             : 
    2036             : template <bool bHasZ, bool bHasM, int nDim>
    2037        1428 : void SetPointsOfLineStruct(OGRLineString *poLS,
    2038             :                            const arrow::StructArray *structArray,
    2039             :                            size_t pointOffset, int numPoints)
    2040             : {
    2041        1428 :     CPLAssert(structArray->num_fields() == nDim);
    2042        1428 :     const auto &fields = structArray->fields();
    2043        1428 :     const auto &fieldX = fields[0];
    2044        1428 :     CPLAssert(fieldX->type_id() == arrow::Type::DOUBLE);
    2045        1428 :     const auto fieldXDouble = static_cast<arrow::DoubleArray *>(fieldX.get());
    2046        1428 :     const auto &fieldY = fields[1];
    2047        1428 :     CPLAssert(fieldY->type_id() == arrow::Type::DOUBLE);
    2048        1428 :     const auto fieldYDouble = static_cast<arrow::DoubleArray *>(fieldY.get());
    2049        1428 :     const arrow::DoubleArray *fieldZDouble = nullptr;
    2050        1428 :     const arrow::DoubleArray *fieldMDouble = nullptr;
    2051        1428 :     int iField = 2;
    2052             :     if constexpr (bHasZ)
    2053             :     {
    2054         466 :         const auto &field = fields[iField];
    2055         466 :         ++iField;
    2056         466 :         CPLAssert(field->type_id() == arrow::Type::DOUBLE);
    2057         466 :         fieldZDouble = static_cast<arrow::DoubleArray *>(field.get());
    2058             :     }
    2059             :     if constexpr (bHasM)
    2060             :     {
    2061          26 :         const auto &field = fields[iField];
    2062          26 :         CPLAssert(field->type_id() == arrow::Type::DOUBLE);
    2063          26 :         fieldMDouble = static_cast<arrow::DoubleArray *>(field.get());
    2064             :     }
    2065             : 
    2066        1428 :     poLS->setNumPoints(numPoints, FALSE);
    2067        6544 :     for (int k = 0; k < numPoints; k++)
    2068             :     {
    2069             :         if constexpr (bHasZ)
    2070             :         {
    2071             :             if constexpr (bHasM)
    2072             :             {
    2073          55 :                 poLS->setPoint(k, fieldXDouble->Value(pointOffset + k),
    2074          55 :                                fieldYDouble->Value(pointOffset + k),
    2075          55 :                                fieldZDouble->Value(pointOffset + k),
    2076          55 :                                fieldMDouble->Value(pointOffset + k));
    2077             :             }
    2078             :             else
    2079             :             {
    2080        1335 :                 poLS->setPoint(k, fieldXDouble->Value(pointOffset + k),
    2081        1335 :                                fieldYDouble->Value(pointOffset + k),
    2082        1335 :                                fieldZDouble->Value(pointOffset + k));
    2083             :             }
    2084             :         }
    2085             :         else if constexpr (bHasM)
    2086             :         {
    2087          55 :             poLS->setPointM(k, fieldXDouble->Value(pointOffset + k),
    2088          55 :                             fieldYDouble->Value(pointOffset + k),
    2089          55 :                             fieldMDouble->Value(pointOffset + k));
    2090             :         }
    2091             :         else
    2092             :         {
    2093        3671 :             poLS->setPoint(k, fieldXDouble->Value(pointOffset + k),
    2094        3671 :                            fieldYDouble->Value(pointOffset + k));
    2095             :         }
    2096             :     }
    2097        1428 : }
    2098             : 
    2099             : typedef void (*SetPointsOfLineStructType)(OGRLineString *,
    2100             :                                           const arrow::StructArray *, size_t,
    2101             :                                           int);
    2102             : 
    2103        1100 : static SetPointsOfLineStructType GetSetPointsOfLineStruct(bool bHasZ,
    2104             :                                                           bool bHasM)
    2105             : {
    2106        1100 :     if (bHasZ && bHasM)
    2107          11 :         return SetPointsOfLineStruct<true, true, 4>;
    2108        1089 :     if (bHasZ)
    2109         419 :         return SetPointsOfLineStruct<true, false, 3>;
    2110         670 :     if (bHasM)
    2111          11 :         return SetPointsOfLineStruct<false, true, 3>;
    2112         659 :     return SetPointsOfLineStruct<false, false, 2>;
    2113             : }
    2114             : 
    2115             : /************************************************************************/
    2116             : /*                            TimestampToOGR()                          */
    2117             : /************************************************************************/
    2118             : 
    2119             : inline void
    2120       11122 : OGRArrowLayer::TimestampToOGR(int64_t timestamp,
    2121             :                               const arrow::TimestampType *timestampType,
    2122             :                               int nTZFlag, OGRField *psField)
    2123             : {
    2124       11122 :     const auto unit = timestampType->unit();
    2125       11122 :     double floatingPart = 0;
    2126       11122 :     if (unit == arrow::TimeUnit::MILLI)
    2127             :     {
    2128        7363 :         floatingPart = (timestamp % 1000) / 1e3;
    2129        7363 :         timestamp /= 1000;
    2130             :     }
    2131        3759 :     else if (unit == arrow::TimeUnit::MICRO)
    2132             :     {
    2133        3289 :         floatingPart = (timestamp % (1000 * 1000)) / 1e6;
    2134        3289 :         timestamp /= 1000 * 1000;
    2135             :     }
    2136         470 :     else if (unit == arrow::TimeUnit::NANO)
    2137             :     {
    2138         235 :         floatingPart = (timestamp % (1000 * 1000 * 1000)) / 1e9;
    2139         235 :         timestamp /= 1000 * 1000 * 1000;
    2140             :     }
    2141       11122 :     if (nTZFlag > OGR_TZFLAG_MIXED_TZ)
    2142             :     {
    2143        5563 :         const int TZOffset = (nTZFlag - OGR_TZFLAG_UTC) * 15;
    2144        5563 :         timestamp += TZOffset * 60;
    2145             :     }
    2146             :     struct tm dt;
    2147       11122 :     CPLUnixTimeToYMDHMS(timestamp, &dt);
    2148       11122 :     psField->Date.Year = static_cast<GInt16>(dt.tm_year + 1900);
    2149       11122 :     psField->Date.Month = static_cast<GByte>(dt.tm_mon + 1);
    2150       11122 :     psField->Date.Day = static_cast<GByte>(dt.tm_mday);
    2151       11122 :     psField->Date.Hour = static_cast<GByte>(dt.tm_hour);
    2152       11122 :     psField->Date.Minute = static_cast<GByte>(dt.tm_min);
    2153       11122 :     psField->Date.TZFlag = static_cast<GByte>(nTZFlag);
    2154       11122 :     psField->Date.Second = static_cast<float>(dt.tm_sec + floatingPart);
    2155       11122 : }
    2156             : 
    2157             : /************************************************************************/
    2158             : /*                         GetStorageArray()                            */
    2159             : /************************************************************************/
    2160             : 
    2161      186031 : static const arrow::Array *GetStorageArray(const arrow::Array *array)
    2162             : {
    2163      186031 :     if (array->type_id() == arrow::Type::EXTENSION)
    2164             :     {
    2165             :         auto extensionArray =
    2166        1896 :             cpl::down_cast<const arrow::ExtensionArray *>(array);
    2167        1896 :         array = extensionArray->storage().get();
    2168             :     }
    2169      186031 :     return array;
    2170             : }
    2171             : 
    2172             : /************************************************************************/
    2173             : /*                            ReadFeature()                             */
    2174             : /************************************************************************/
    2175             : 
    2176        9819 : inline OGRFeature *OGRArrowLayer::ReadFeature(
    2177             :     int64_t nIdxInBatch,
    2178             :     const std::vector<std::shared_ptr<arrow::Array>> &poColumnArrays) const
    2179             : {
    2180        9819 :     OGRFeature *poFeature = new OGRFeature(m_poFeatureDefn);
    2181             : 
    2182        9819 :     if (m_iFIDArrowColumn >= 0)
    2183             :     {
    2184        3396 :         const int iCol =
    2185        3396 :             m_bIgnoredFields ? m_nRequestedFIDColumn : m_iFIDArrowColumn;
    2186        3396 :         const arrow::Array *array = poColumnArrays[iCol].get();
    2187        3396 :         if (!array->IsNull(nIdxInBatch))
    2188             :         {
    2189        3396 :             if (array->type_id() == arrow::Type::INT64)
    2190             :             {
    2191        3396 :                 const auto castArray =
    2192             :                     static_cast<const arrow::Int64Array *>(array);
    2193        3396 :                 poFeature->SetFID(
    2194        3396 :                     static_cast<GIntBig>(castArray->Value(nIdxInBatch)));
    2195             :             }
    2196           0 :             else if (array->type_id() == arrow::Type::INT32)
    2197             :             {
    2198           0 :                 const auto castArray =
    2199             :                     static_cast<const arrow::Int32Array *>(array);
    2200           0 :                 poFeature->SetFID(castArray->Value(nIdxInBatch));
    2201             :             }
    2202             :         }
    2203             :     }
    2204             : 
    2205        9819 :     const int nFieldCount = m_poFeatureDefn->GetFieldCount();
    2206      175113 :     for (int i = 0; i < nFieldCount; ++i)
    2207             :     {
    2208             :         int iCol;
    2209      165294 :         if (m_bIgnoredFields)
    2210             :         {
    2211       16634 :             iCol = m_anMapFieldIndexToArrayIndex[i];
    2212       16634 :             if (iCol < 0)
    2213        5040 :                 continue;
    2214             :         }
    2215             :         else
    2216             :         {
    2217      148660 :             iCol = m_anMapFieldIndexToArrowColumn[i][0];
    2218             :         }
    2219             : 
    2220      160254 :         const arrow::Array *array = GetStorageArray(poColumnArrays[iCol].get());
    2221      160254 :         if (array->IsNull(nIdxInBatch))
    2222             :         {
    2223       16636 :             poFeature->SetFieldNull(i);
    2224       16636 :             continue;
    2225             :         }
    2226             : 
    2227      143618 :         int j = 1;
    2228      143618 :         bool bSkipToNextField = false;
    2229      157738 :         while (array->type_id() == arrow::Type::STRUCT)
    2230             :         {
    2231       14120 :             const auto castArray =
    2232             :                 static_cast<const arrow::StructArray *>(array);
    2233       14120 :             const auto &subArrays = castArray->fields();
    2234       14120 :             CPLAssert(
    2235             :                 j < static_cast<int>(m_anMapFieldIndexToArrowColumn[i].size()));
    2236       14120 :             const int iArrowSubcol = m_anMapFieldIndexToArrowColumn[i][j];
    2237       14120 :             j++;
    2238       14120 :             CPLAssert(iArrowSubcol < static_cast<int>(subArrays.size()));
    2239       14120 :             array = GetStorageArray(subArrays[iArrowSubcol].get());
    2240       14120 :             if (array->IsNull(nIdxInBatch))
    2241             :             {
    2242           0 :                 poFeature->SetFieldNull(i);
    2243           0 :                 bSkipToNextField = true;
    2244           0 :                 break;
    2245             :             }
    2246             :         }
    2247      143618 :         if (bSkipToNextField)
    2248           0 :             continue;
    2249             : 
    2250      143618 :         if (array->type_id() == arrow::Type::DICTIONARY)
    2251             :         {
    2252        1495 :             const auto castArray =
    2253             :                 static_cast<const arrow::DictionaryArray *>(array);
    2254             :             m_poReadFeatureTmpArray =
    2255        1495 :                 castArray->indices();  // does not return a const reference
    2256        1495 :             array = GetStorageArray(m_poReadFeatureTmpArray.get());
    2257        1495 :             if (array->IsNull(nIdxInBatch))
    2258             :             {
    2259           0 :                 poFeature->SetFieldNull(i);
    2260           0 :                 continue;
    2261             :             }
    2262             :         }
    2263             : 
    2264      143618 :         switch (array->type_id())
    2265             :         {
    2266           0 :             case arrow::Type::NA:
    2267           0 :                 break;
    2268             : 
    2269        1543 :             case arrow::Type::BOOL:
    2270             :             {
    2271        1543 :                 const auto castArray =
    2272             :                     static_cast<const arrow::BooleanArray *>(array);
    2273        1543 :                 poFeature->SetFieldSameTypeUnsafe(
    2274        1543 :                     i, castArray->Value(nIdxInBatch));
    2275        1543 :                 break;
    2276             :             }
    2277        1478 :             case arrow::Type::UINT8:
    2278             :             {
    2279        1478 :                 const auto castArray =
    2280             :                     static_cast<const arrow::UInt8Array *>(array);
    2281        1478 :                 poFeature->SetFieldSameTypeUnsafe(
    2282        1478 :                     i, castArray->Value(nIdxInBatch));
    2283        1478 :                 break;
    2284             :             }
    2285        1474 :             case arrow::Type::INT8:
    2286             :             {
    2287        1474 :                 const auto castArray =
    2288             :                     static_cast<const arrow::Int8Array *>(array);
    2289        1474 :                 poFeature->SetFieldSameTypeUnsafe(
    2290        1474 :                     i, castArray->Value(nIdxInBatch));
    2291        1474 :                 break;
    2292             :             }
    2293        1474 :             case arrow::Type::UINT16:
    2294             :             {
    2295        1474 :                 const auto castArray =
    2296             :                     static_cast<const arrow::UInt16Array *>(array);
    2297        1474 :                 poFeature->SetFieldSameTypeUnsafe(
    2298        1474 :                     i, castArray->Value(nIdxInBatch));
    2299        1474 :                 break;
    2300             :             }
    2301        1552 :             case arrow::Type::INT16:
    2302             :             {
    2303        1552 :                 const auto castArray =
    2304             :                     static_cast<const arrow::Int16Array *>(array);
    2305        1552 :                 poFeature->SetFieldSameTypeUnsafe(
    2306        1552 :                     i, castArray->Value(nIdxInBatch));
    2307        1552 :                 break;
    2308             :             }
    2309         194 :             case arrow::Type::UINT32:
    2310             :             {
    2311         194 :                 const auto castArray =
    2312             :                     static_cast<const arrow::UInt32Array *>(array);
    2313         194 :                 poFeature->SetFieldSameTypeUnsafe(
    2314         194 :                     i, static_cast<GIntBig>(castArray->Value(nIdxInBatch)));
    2315         194 :                 break;
    2316             :             }
    2317        8405 :             case arrow::Type::INT32:
    2318             :             {
    2319        8405 :                 const auto castArray =
    2320             :                     static_cast<const arrow::Int32Array *>(array);
    2321        8405 :                 poFeature->SetFieldSameTypeUnsafe(
    2322             :                     i, castArray->Value(nIdxInBatch));
    2323        8405 :                 break;
    2324             :             }
    2325        1474 :             case arrow::Type::UINT64:
    2326             :             {
    2327        1474 :                 const auto castArray =
    2328             :                     static_cast<const arrow::UInt64Array *>(array);
    2329        1474 :                 poFeature->SetFieldSameTypeUnsafe(
    2330        1474 :                     i, static_cast<double>(castArray->Value(nIdxInBatch)));
    2331        1474 :                 break;
    2332             :             }
    2333        7455 :             case arrow::Type::INT64:
    2334             :             {
    2335        7455 :                 const auto castArray =
    2336             :                     static_cast<const arrow::Int64Array *>(array);
    2337        7455 :                 poFeature->SetFieldSameTypeUnsafe(
    2338        7455 :                     i, static_cast<GIntBig>(castArray->Value(nIdxInBatch)));
    2339        7455 :                 break;
    2340             :             }
    2341         194 :             case arrow::Type::HALF_FLOAT:
    2342             :             {
    2343         194 :                 const auto castArray =
    2344             :                     static_cast<const arrow::HalfFloatArray *>(array);
    2345         194 :                 const uint16_t nFloat16 = castArray->Value(nIdxInBatch);
    2346         194 :                 uint32_t nFloat32 = CPLHalfToFloat(nFloat16);
    2347             :                 float f;
    2348         194 :                 memcpy(&f, &nFloat32, sizeof(nFloat32));
    2349         194 :                 poFeature->SetFieldSameTypeUnsafe(i, f);
    2350         194 :                 break;
    2351             :             }
    2352        1633 :             case arrow::Type::FLOAT:
    2353             :             {
    2354        1633 :                 const auto castArray =
    2355             :                     static_cast<const arrow::FloatArray *>(array);
    2356        1633 :                 poFeature->SetFieldSameTypeUnsafe(
    2357        1633 :                     i, castArray->Value(nIdxInBatch));
    2358        1633 :                 break;
    2359             :             }
    2360        3593 :             case arrow::Type::DOUBLE:
    2361             :             {
    2362        3593 :                 const auto castArray =
    2363             :                     static_cast<const arrow::DoubleArray *>(array);
    2364        3593 :                 poFeature->SetFieldSameTypeUnsafe(
    2365             :                     i, castArray->Value(nIdxInBatch));
    2366        3593 :                 break;
    2367             :             }
    2368        7413 :             case arrow::Type::STRING:
    2369             :             {
    2370        7413 :                 const auto castArray =
    2371             :                     static_cast<const arrow::StringArray *>(array);
    2372        7413 :                 int out_length = 0;
    2373             :                 const uint8_t *data =
    2374        7413 :                     castArray->GetValue(nIdxInBatch, &out_length);
    2375             :                 char *pszString =
    2376        7413 :                     static_cast<char *>(CPLMalloc(out_length + 1));
    2377        7413 :                 memcpy(pszString, data, out_length);
    2378        7413 :                 pszString[out_length] = 0;
    2379        7413 :                 poFeature->SetFieldSameTypeUnsafe(i, pszString);
    2380        7413 :                 break;
    2381             :             }
    2382        1945 :             case arrow::Type::BINARY:
    2383             :             {
    2384        1945 :                 const auto castArray =
    2385             :                     static_cast<const arrow::BinaryArray *>(array);
    2386        1945 :                 int out_length = 0;
    2387             :                 const uint8_t *data =
    2388        1945 :                     castArray->GetValue(nIdxInBatch, &out_length);
    2389        1945 :                 poFeature->SetField(i, out_length, data);
    2390        1945 :                 break;
    2391             :             }
    2392             : #if ARROW_VERSION_MAJOR >= 15
    2393           3 :             case arrow::Type::BINARY_VIEW:
    2394             :             {
    2395           3 :                 const auto castArray =
    2396             :                     static_cast<const arrow::BinaryViewArray *>(array);
    2397           3 :                 const auto view = castArray->GetView(nIdxInBatch);
    2398           3 :                 poFeature->SetField(i, static_cast<int>(view.size()),
    2399           3 :                                     view.data());
    2400           3 :                 break;
    2401             :             }
    2402             : #endif
    2403             : #if ARROW_VERSION_MAJOR >= 15
    2404           3 :             case arrow::Type::STRING_VIEW:
    2405             :             {
    2406           3 :                 const auto castArray =
    2407             :                     static_cast<const arrow::StringViewArray *>(array);
    2408           3 :                 const auto strView = castArray->GetView(nIdxInBatch);
    2409             :                 char *pszString =
    2410           3 :                     static_cast<char *>(CPLMalloc(strView.length() + 1));
    2411           3 :                 memcpy(pszString, strView.data(), strView.length());
    2412           3 :                 pszString[strView.length()] = 0;
    2413           3 :                 poFeature->SetFieldSameTypeUnsafe(i, pszString);
    2414           3 :                 break;
    2415             :             }
    2416             : #endif
    2417        1853 :             case arrow::Type::FIXED_SIZE_BINARY:
    2418             :             {
    2419        1853 :                 const auto castArray =
    2420             :                     static_cast<const arrow::FixedSizeBinaryArray *>(array);
    2421        1853 :                 const uint8_t *data = castArray->GetValue(nIdxInBatch);
    2422        1853 :                 poFeature->SetField(i, castArray->byte_width(), data);
    2423        1853 :                 break;
    2424             :             }
    2425        3471 :             case arrow::Type::DATE32:
    2426             :             {
    2427             :                 // number of days since Epoch
    2428        3471 :                 const auto castArray =
    2429             :                     static_cast<const arrow::Date32Array *>(array);
    2430             :                 int64_t timestamp =
    2431        3471 :                     static_cast<int64_t>(castArray->Value(nIdxInBatch)) * 3600 *
    2432        3471 :                     24;
    2433             :                 struct tm dt;
    2434        3471 :                 CPLUnixTimeToYMDHMS(timestamp, &dt);
    2435        3471 :                 poFeature->SetField(i, dt.tm_year + 1900, dt.tm_mon + 1,
    2436             :                                     dt.tm_mday, 0, 0, 0);
    2437        3471 :                 break;
    2438             :             }
    2439         235 :             case arrow::Type::DATE64:
    2440             :             {
    2441             :                 // number of milliseconds since Epoch
    2442         235 :                 const auto castArray =
    2443             :                     static_cast<const arrow::Date64Array *>(array);
    2444             :                 int64_t timestamp =
    2445         235 :                     static_cast<int64_t>(castArray->Value(nIdxInBatch)) / 1000;
    2446             :                 struct tm dt;
    2447         235 :                 CPLUnixTimeToYMDHMS(timestamp, &dt);
    2448         235 :                 poFeature->SetField(i, dt.tm_year + 1900, dt.tm_mon + 1,
    2449             :                                     dt.tm_mday, 0, 0, 0);
    2450         235 :                 break;
    2451             :             }
    2452       11120 :             case arrow::Type::TIMESTAMP:
    2453             :             {
    2454             :                 const auto timestampType = static_cast<arrow::TimestampType *>(
    2455       11120 :                     array->data()->type.get());
    2456       11120 :                 const auto castArray =
    2457             :                     static_cast<const arrow::Int64Array *>(array);
    2458       11120 :                 const int64_t timestamp = castArray->Value(nIdxInBatch);
    2459             :                 OGRField sField;
    2460       11120 :                 sField.Set.nMarker1 = OGRUnsetMarker;
    2461       11120 :                 sField.Set.nMarker2 = OGRUnsetMarker;
    2462       11120 :                 sField.Set.nMarker3 = OGRUnsetMarker;
    2463       11120 :                 TimestampToOGR(timestamp, timestampType,
    2464       11120 :                                m_poFeatureDefn->GetFieldDefn(i)->GetTZFlag(),
    2465             :                                &sField);
    2466       11120 :                 poFeature->SetField(i, &sField);
    2467       11120 :                 break;
    2468             :             }
    2469        3335 :             case arrow::Type::TIME32:
    2470             :             {
    2471             :                 const auto timestampType =
    2472        3335 :                     static_cast<arrow::Time32Type *>(array->data()->type.get());
    2473        3335 :                 const auto castArray =
    2474             :                     static_cast<const arrow::Int32Array *>(array);
    2475        3335 :                 const auto unit = timestampType->unit();
    2476        3335 :                 int value = castArray->Value(nIdxInBatch);
    2477        3335 :                 double floatingPart = 0;
    2478        3335 :                 if (unit == arrow::TimeUnit::MILLI)
    2479             :                 {
    2480        3139 :                     floatingPart = (value % 1000) / 1e3;
    2481        3139 :                     value /= 1000;
    2482             :                 }
    2483        3335 :                 const int nHour = value / 3600;
    2484        3335 :                 const int nMinute = (value / 60) % 60;
    2485        3335 :                 const int nSecond = value % 60;
    2486        3335 :                 poFeature->SetField(i, 0, 0, 0, nHour, nMinute,
    2487        3335 :                                     static_cast<float>(nSecond + floatingPart));
    2488        3335 :                 break;
    2489             :             }
    2490        3039 :             case arrow::Type::TIME64:
    2491             :             {
    2492        3039 :                 const auto castArray =
    2493             :                     static_cast<const arrow::Time64Array *>(array);
    2494        3039 :                 poFeature->SetField(
    2495        3039 :                     i, static_cast<GIntBig>(castArray->Value(nIdxInBatch)));
    2496        3039 :                 break;
    2497             :             }
    2498             : 
    2499             : #if ARROW_VERSION_MAJOR >= 18
    2500           0 :             case arrow::Type::DECIMAL32:
    2501             :             {
    2502           0 :                 const auto castArray =
    2503             :                     static_cast<const arrow::Decimal32Array *>(array);
    2504           0 :                 poFeature->SetField(
    2505           0 :                     i, CPLAtof(castArray->FormatValue(nIdxInBatch).c_str()));
    2506           0 :                 break;
    2507             :             }
    2508             : 
    2509           0 :             case arrow::Type::DECIMAL64:
    2510             :             {
    2511           0 :                 const auto castArray =
    2512             :                     static_cast<const arrow::Decimal64Array *>(array);
    2513           0 :                 poFeature->SetField(
    2514           0 :                     i, CPLAtof(castArray->FormatValue(nIdxInBatch).c_str()));
    2515           0 :                 break;
    2516             :             }
    2517             : #endif
    2518             : 
    2519        1553 :             case arrow::Type::DECIMAL128:
    2520             :             {
    2521        1553 :                 const auto castArray =
    2522             :                     static_cast<const arrow::Decimal128Array *>(array);
    2523        1553 :                 poFeature->SetField(
    2524        3106 :                     i, CPLAtof(castArray->FormatValue(nIdxInBatch).c_str()));
    2525        1553 :                 break;
    2526             :             }
    2527             : 
    2528        1369 :             case arrow::Type::DECIMAL256:
    2529             :             {
    2530        1369 :                 const auto castArray =
    2531             :                     static_cast<const arrow::Decimal256Array *>(array);
    2532        1369 :                 poFeature->SetField(
    2533        2738 :                     i, CPLAtof(castArray->FormatValue(nIdxInBatch).c_str()));
    2534        1369 :                 break;
    2535             :             }
    2536             : 
    2537       28543 :             case arrow::Type::LIST:
    2538             :             {
    2539       28543 :                 const auto castArray =
    2540             :                     static_cast<const arrow::ListArray *>(array);
    2541             :                 const auto listType = static_cast<const arrow::ListType *>(
    2542       28543 :                     array->data()->type.get());
    2543       28543 :                 ReadList(poFeature, i, nIdxInBatch, castArray,
    2544       28543 :                          listType->value_field()->type()->id());
    2545       28543 :                 break;
    2546             :             }
    2547             : 
    2548       21132 :             case arrow::Type::FIXED_SIZE_LIST:
    2549             :             {
    2550       21132 :                 const auto castArray =
    2551             :                     static_cast<const arrow::FixedSizeListArray *>(array);
    2552             :                 const auto listType =
    2553             :                     static_cast<const arrow::FixedSizeListType *>(
    2554       21132 :                         array->data()->type.get());
    2555       21132 :                 ReadList(poFeature, i, nIdxInBatch, castArray,
    2556       21132 :                          listType->value_field()->type()->id());
    2557       21132 :                 break;
    2558             :             }
    2559             : 
    2560        1474 :             case arrow::Type::LARGE_STRING:
    2561             :             {
    2562        1474 :                 const auto castArray =
    2563             :                     static_cast<const arrow::LargeStringArray *>(array);
    2564        1474 :                 poFeature->SetField(i,
    2565        2948 :                                     castArray->GetString(nIdxInBatch).c_str());
    2566        1474 :                 break;
    2567             :             }
    2568        1761 :             case arrow::Type::LARGE_BINARY:
    2569             :             {
    2570        1761 :                 const auto castArray =
    2571             :                     static_cast<const arrow::LargeBinaryArray *>(array);
    2572        1761 :                 arrow::LargeBinaryArray::offset_type out_length = 0;
    2573             :                 const uint8_t *data =
    2574        1761 :                     castArray->GetValue(nIdxInBatch, &out_length);
    2575        1761 :                 if (out_length >= 0 && out_length <= INT_MAX - 1)
    2576             :                 {
    2577             :                     // coverity[overflow_sink]
    2578        1761 :                     poFeature->SetField(i, static_cast<int>(out_length), data);
    2579             :                 }
    2580             :                 else
    2581             :                 {
    2582             :                     // this is probably the most likely code path if people use
    2583             :                     // LargeBinary...
    2584           0 :                     CPLError(CE_Warning, CPLE_AppDefined,
    2585             :                              "Too large binary: " CPL_FRMT_GUIB " bytes",
    2586             :                              static_cast<GUIntBig>(out_length));
    2587             :                 }
    2588        1761 :                 break;
    2589             :             }
    2590             : 
    2591       24900 :             case arrow::Type::MAP:
    2592             :             {
    2593       24900 :                 const auto castArray =
    2594             :                     static_cast<const arrow::MapArray *>(array);
    2595       24900 :                 poFeature->SetField(
    2596       24900 :                     i, GetMapAsJSON(castArray, static_cast<size_t>(nIdxInBatch))
    2597       49800 :                            .Format(CPLJSONObject::PrettyFormat::Plain)
    2598             :                            .c_str());
    2599       24900 :                 break;
    2600             :             }
    2601             : 
    2602             :             // unhandled types
    2603           0 :             case arrow::Type::STRUCT:  // should not happen
    2604             :             case arrow::Type::INTERVAL_MONTHS:
    2605             :             case arrow::Type::INTERVAL_DAY_TIME:
    2606             :             case arrow::Type::SPARSE_UNION:
    2607             :             case arrow::Type::DENSE_UNION:
    2608             :             case arrow::Type::DICTIONARY:
    2609             :             case arrow::Type::EXTENSION:
    2610             :             case arrow::Type::DURATION:
    2611             :             case arrow::Type::LARGE_LIST:
    2612             :             case arrow::Type::INTERVAL_MONTH_DAY_NANO:
    2613             : #if ARROW_VERSION_MAJOR >= 12
    2614             :             case arrow::Type::RUN_END_ENCODED:
    2615             : #endif
    2616             : #if ARROW_VERSION_MAJOR >= 15
    2617             :             case arrow::Type::LIST_VIEW:
    2618             :             case arrow::Type::LARGE_LIST_VIEW:
    2619             : #endif
    2620             :             case arrow::Type::MAX_ID:
    2621             :             {
    2622             :                 // Shouldn't happen normally as we should have discarded those
    2623             :                 // fields when creating OGR field definitions
    2624           0 :                 CPLError(CE_Warning, CPLE_AppDefined,
    2625             :                          "Cannot read content for field %s",
    2626           0 :                          m_poFeatureDefn->GetFieldDefn(i)->GetNameRef());
    2627           0 :                 break;
    2628             :             }
    2629             :         }
    2630             :     }
    2631             : 
    2632        9819 :     const int nGeomFieldCount = m_poFeatureDefn->GetGeomFieldCount();
    2633       19340 :     for (int i = 0; i < nGeomFieldCount; ++i)
    2634             :     {
    2635             :         int iCol;
    2636        9521 :         if (m_bIgnoredFields)
    2637             :         {
    2638        1119 :             iCol = m_anMapGeomFieldIndexToArrayIndex[i];
    2639        1119 :             if (iCol < 0)
    2640          52 :                 continue;
    2641             :         }
    2642             :         else
    2643             :         {
    2644        8402 :             iCol = m_anMapGeomFieldIndexToArrowColumn[i];
    2645             :         }
    2646             : 
    2647        9469 :         const auto array = GetStorageArray(poColumnArrays[iCol].get());
    2648        9469 :         auto poGeometry = ReadGeometry(i, array, nIdxInBatch);
    2649        9469 :         if (poGeometry)
    2650             :         {
    2651        6917 :             const auto poGeomFieldDefn = m_poFeatureDefn->GetGeomFieldDefn(i);
    2652        7317 :             if (wkbFlatten(poGeometry->getGeometryType()) == wkbLineString &&
    2653         400 :                 wkbFlatten(poGeomFieldDefn->GetType()) == wkbMultiLineString)
    2654             :             {
    2655             :                 poGeometry =
    2656           3 :                     OGRGeometryFactory::forceToMultiLineString(poGeometry);
    2657             :             }
    2658        7790 :             else if (wkbFlatten(poGeometry->getGeometryType()) == wkbPolygon &&
    2659         876 :                      wkbFlatten(poGeomFieldDefn->GetType()) == wkbMultiPolygon)
    2660             :             {
    2661             :                 poGeometry =
    2662          33 :                     OGRGeometryFactory::forceToMultiPolygon(poGeometry);
    2663             :             }
    2664        6917 :             if (OGR_GT_HasZ(poGeomFieldDefn->GetType()) && !poGeometry->Is3D())
    2665             :             {
    2666          63 :                 poGeometry->set3D(true);
    2667             :             }
    2668        6917 :             poFeature->SetGeomFieldDirectly(i, poGeometry);
    2669             :         }
    2670             :     }
    2671             : 
    2672        9819 :     return poFeature;
    2673             : }
    2674             : 
    2675             : /************************************************************************/
    2676             : /*                           ReadGeometry()                             */
    2677             : /************************************************************************/
    2678             : 
    2679        9689 : inline OGRGeometry *OGRArrowLayer::ReadGeometry(int iGeomField,
    2680             :                                                 const arrow::Array *array,
    2681             :                                                 int64_t nIdxInBatch) const
    2682             : {
    2683        9689 :     if (array->IsNull(nIdxInBatch))
    2684             :     {
    2685        2601 :         return nullptr;
    2686             :     }
    2687        7088 :     OGRGeometry *poGeometry = nullptr;
    2688        7088 :     const auto poGeomFieldDefn = m_poFeatureDefn->GetGeomFieldDefn(iGeomField);
    2689        7088 :     const auto eGeomType = poGeomFieldDefn->GetType();
    2690        7088 :     const bool bHasZ = CPL_TO_BOOL(OGR_GT_HasZ(eGeomType));
    2691        7088 :     const bool bHasM = CPL_TO_BOOL(OGR_GT_HasM(eGeomType));
    2692        7088 :     const int nDim = 2 + (bHasZ ? 1 : 0) + (bHasM ? 1 : 0);
    2693             : 
    2694             :     const auto CreatePoint =
    2695         798 :         [bHasZ, bHasM](const arrow::DoubleArray *pointValues, int pointOffset)
    2696             :     {
    2697         399 :         if (bHasZ)
    2698             :         {
    2699         199 :             if (bHasM)
    2700             :             {
    2701          66 :                 return new OGRPoint(pointValues->Value(pointOffset),
    2702          66 :                                     pointValues->Value(pointOffset + 1),
    2703          66 :                                     pointValues->Value(pointOffset + 2),
    2704          66 :                                     pointValues->Value(pointOffset + 3));
    2705             :             }
    2706             :             else
    2707             :             {
    2708         133 :                 return new OGRPoint(pointValues->Value(pointOffset),
    2709         133 :                                     pointValues->Value(pointOffset + 1),
    2710         133 :                                     pointValues->Value(pointOffset + 2));
    2711             :             }
    2712             :         }
    2713         200 :         else if (bHasM)
    2714             :         {
    2715          66 :             return OGRPoint::createXYM(pointValues->Value(pointOffset),
    2716          66 :                                        pointValues->Value(pointOffset + 1),
    2717         132 :                                        pointValues->Value(pointOffset + 2));
    2718             :         }
    2719             :         else
    2720             :         {
    2721         134 :             return new OGRPoint(pointValues->Value(pointOffset),
    2722         134 :                                 pointValues->Value(pointOffset + 1));
    2723             :         }
    2724        7088 :     };
    2725             : 
    2726             :     const auto CreateStructPoint =
    2727         667 :         [nDim, bHasZ, bHasM](const arrow::StructArray *structArray,
    2728        2001 :                              int64_t pointOffset)
    2729             :     {
    2730         667 :         CPL_IGNORE_RET_VAL(nDim);
    2731         667 :         CPLAssert(structArray->num_fields() == nDim);
    2732         667 :         const auto &fieldX = structArray->field(0);
    2733         667 :         CPLAssert(fieldX->type_id() == arrow::Type::DOUBLE);
    2734             :         const auto fieldXDouble =
    2735         667 :             static_cast<arrow::DoubleArray *>(fieldX.get());
    2736         667 :         const auto &fieldY = structArray->field(1);
    2737         667 :         CPLAssert(fieldY->type_id() == arrow::Type::DOUBLE);
    2738             :         const auto fieldYDouble =
    2739         667 :             static_cast<arrow::DoubleArray *>(fieldY.get());
    2740         667 :         if (bHasZ)
    2741             :         {
    2742         278 :             const auto &fieldZ = structArray->field(2);
    2743         278 :             CPLAssert(fieldZ->type_id() == arrow::Type::DOUBLE);
    2744             :             const auto fieldZDouble =
    2745         278 :                 static_cast<arrow::DoubleArray *>(fieldZ.get());
    2746         278 :             if (bHasM)
    2747             :             {
    2748          11 :                 const auto &fieldM = structArray->field(3);
    2749          11 :                 CPLAssert(fieldM->type_id() == arrow::Type::DOUBLE);
    2750             :                 const auto fieldMDouble =
    2751          11 :                     static_cast<arrow::DoubleArray *>(fieldM.get());
    2752          11 :                 return new OGRPoint(fieldXDouble->Value(pointOffset),
    2753          11 :                                     fieldYDouble->Value(pointOffset),
    2754          11 :                                     fieldZDouble->Value(pointOffset),
    2755          11 :                                     fieldMDouble->Value(pointOffset));
    2756             :             }
    2757             :             else
    2758             :             {
    2759         267 :                 return new OGRPoint(fieldXDouble->Value(pointOffset),
    2760         267 :                                     fieldYDouble->Value(pointOffset),
    2761         267 :                                     fieldZDouble->Value(pointOffset));
    2762             :             }
    2763             :         }
    2764         389 :         else if (bHasM)
    2765             :         {
    2766          11 :             const auto &fieldM = structArray->field(2);
    2767          11 :             CPLAssert(fieldM->type_id() == arrow::Type::DOUBLE);
    2768             :             const auto fieldMDouble =
    2769          11 :                 static_cast<arrow::DoubleArray *>(fieldM.get());
    2770          11 :             return OGRPoint::createXYM(fieldXDouble->Value(pointOffset),
    2771             :                                        fieldYDouble->Value(pointOffset),
    2772          11 :                                        fieldMDouble->Value(pointOffset));
    2773             :         }
    2774             :         else
    2775             :         {
    2776         378 :             return new OGRPoint(fieldXDouble->Value(pointOffset),
    2777         378 :                                 fieldYDouble->Value(pointOffset));
    2778             :         }
    2779        7088 :     };
    2780             : 
    2781             :     // Arrow 14 since https://github.com/apache/arrow/commit/95a8bfb319b2729c8f6daa069433caba3b4ddddd
    2782             :     // returns reference to shared pointers, so we can safely take the raw pointer
    2783             :     // and cast it.
    2784             :     // Earlier versions returned a non-reference shared pointer, so formally it
    2785             :     // is safer to use static_pointer_cast (although in practice given that
    2786             :     // "values" is a member variable), the Arrow >= 14 path might work...
    2787             : #if ARROW_VERSION_MAJOR >= 14
    2788             : #define GET_PTR_FROM_VALUES(var, type, values)                                 \
    2789             :     const auto var = static_cast<const type *>((values).get())
    2790             : #else
    2791             : #define GET_PTR_FROM_VALUES(var, type, values)                                 \
    2792             :     const auto var##tmp = std::static_pointer_cast<type>(values);              \
    2793             :     const auto var = var##tmp.get()
    2794             : #endif
    2795             : 
    2796        7088 :     switch (m_aeGeomEncoding[iGeomField])
    2797             :     {
    2798        4160 :         case OGRArrowGeomEncoding::WKB:
    2799             :         {
    2800        4160 :             int out_length = 0;
    2801             :             const uint8_t *data;
    2802        4160 :             if (array->type_id() == arrow::Type::BINARY)
    2803             :             {
    2804        4149 :                 const auto castArray =
    2805             :                     static_cast<const arrow::BinaryArray *>(array);
    2806        4149 :                 data = castArray->GetValue(nIdxInBatch, &out_length);
    2807             :             }
    2808             :             else
    2809             :             {
    2810          11 :                 CPLAssert(array->type_id() == arrow::Type::LARGE_BINARY);
    2811          11 :                 const auto castArray =
    2812             :                     static_cast<const arrow::LargeBinaryArray *>(array);
    2813          11 :                 int64_t out_length64 = 0;
    2814          11 :                 data = castArray->GetValue(nIdxInBatch, &out_length64);
    2815          11 :                 if (out_length64 > INT_MAX)
    2816             :                 {
    2817           0 :                     CPLError(CE_Failure, CPLE_AppDefined, "Too large geometry");
    2818           0 :                     return nullptr;
    2819             :                 }
    2820          11 :                 out_length = static_cast<int>(out_length64);
    2821             :             }
    2822        4160 :             if (OGRGeometryFactory::createFromWkb(
    2823        4160 :                     data, poGeomFieldDefn->GetSpatialRef(), &poGeometry,
    2824        4160 :                     out_length) == OGRERR_NONE)
    2825             :             {
    2826             : #ifdef DEBUG_ReadWKBBoundingBox
    2827             :                 OGREnvelope sEnvelopeFromWKB;
    2828             :                 bool bRet =
    2829             :                     OGRWKBGetBoundingBox(data, out_length, sEnvelopeFromWKB);
    2830             :                 CPLAssert(bRet);
    2831             :                 OGREnvelope sEnvelopeFromGeom;
    2832             :                 poGeometry->getEnvelope(&sEnvelopeFromGeom);
    2833             :                 CPLAssert(sEnvelopeFromWKB == sEnvelopeFromGeom);
    2834             : #endif
    2835             :             }
    2836        4160 :             break;
    2837             :         }
    2838             : 
    2839         120 :         case OGRArrowGeomEncoding::WKT:
    2840             :         {
    2841         120 :             if (array->type_id() == arrow::Type::STRING)
    2842             :             {
    2843         119 :                 const auto castArray =
    2844             :                     static_cast<const arrow::StringArray *>(array);
    2845         238 :                 const auto osWKT = castArray->GetString(nIdxInBatch);
    2846         119 :                 OGRGeometryFactory::createFromWkt(
    2847         119 :                     osWKT.c_str(), poGeomFieldDefn->GetSpatialRef(),
    2848             :                     &poGeometry);
    2849             :             }
    2850             :             else
    2851             :             {
    2852           1 :                 CPLAssert(array->type_id() == arrow::Type::LARGE_STRING);
    2853           1 :                 const auto castArray =
    2854             :                     static_cast<const arrow::LargeStringArray *>(array);
    2855           2 :                 const auto osWKT = castArray->GetString(nIdxInBatch);
    2856           1 :                 OGRGeometryFactory::createFromWkt(
    2857           1 :                     osWKT.c_str(), poGeomFieldDefn->GetSpatialRef(),
    2858             :                     &poGeometry);
    2859             :             }
    2860         120 :             break;
    2861             :         }
    2862             : 
    2863           0 :         case OGRArrowGeomEncoding::GEOARROW_FSL_GENERIC:
    2864             :         case OGRArrowGeomEncoding::GEOARROW_STRUCT_GENERIC:
    2865             :         {
    2866           0 :             CPLAssert(false);
    2867             :             break;
    2868             :         }
    2869             : 
    2870         113 :         case OGRArrowGeomEncoding::GEOARROW_FSL_POINT:
    2871             :         {
    2872         113 :             CPLAssert(array->type_id() == arrow::Type::FIXED_SIZE_LIST);
    2873         113 :             const auto listArray =
    2874             :                 static_cast<const arrow::FixedSizeListArray *>(array);
    2875         113 :             CPLAssert(listArray->values()->type_id() == arrow::Type::DOUBLE);
    2876         113 :             GET_PTR_FROM_VALUES(pointValues, arrow::DoubleArray,
    2877             :                                 listArray->values());
    2878         113 :             if (!pointValues->IsNull(nDim * nIdxInBatch))
    2879             :             {
    2880         113 :                 poGeometry = CreatePoint(pointValues,
    2881             :                                          static_cast<int>(nDim * nIdxInBatch));
    2882         113 :                 poGeometry->assignSpatialReference(
    2883         113 :                     poGeomFieldDefn->GetSpatialRef());
    2884             :             }
    2885         113 :             break;
    2886             :         }
    2887             : 
    2888         100 :         case OGRArrowGeomEncoding::GEOARROW_FSL_LINESTRING:
    2889             :         {
    2890         100 :             CPLAssert(array->type_id() == arrow::Type::LIST);
    2891         100 :             const auto listArray = static_cast<const arrow::ListArray *>(array);
    2892         100 :             CPLAssert(listArray->values()->type_id() ==
    2893             :                       arrow::Type::FIXED_SIZE_LIST);
    2894         100 :             GET_PTR_FROM_VALUES(listOfPointsValues, arrow::FixedSizeListArray,
    2895             :                                 listArray->values());
    2896         100 :             CPLAssert(listOfPointsValues->values()->type_id() ==
    2897             :                       arrow::Type::DOUBLE);
    2898         100 :             GET_PTR_FROM_VALUES(pointValues, arrow::DoubleArray,
    2899             :                                 listOfPointsValues->values());
    2900         100 :             const auto nPoints = listArray->value_length(nIdxInBatch);
    2901             :             const auto nPointOffset =
    2902         100 :                 listArray->value_offset(nIdxInBatch) * nDim;
    2903         100 :             auto poLineString = new OGRLineString();
    2904         100 :             poGeometry = poLineString;
    2905         100 :             poGeometry->assignSpatialReference(
    2906         100 :                 poGeomFieldDefn->GetSpatialRef());
    2907         100 :             if (nPoints)
    2908             :             {
    2909          64 :                 GetSetPointsOfLine(bHasZ, bHasM)(poLineString, pointValues,
    2910             :                                                  nPointOffset, nPoints);
    2911             :             }
    2912             :             else
    2913             :             {
    2914          36 :                 poGeometry->set3D(bHasZ);
    2915          36 :                 poGeometry->setMeasured(bHasM);
    2916             :             }
    2917         100 :             break;
    2918             :         }
    2919             : 
    2920         456 :         case OGRArrowGeomEncoding::GEOARROW_FSL_POLYGON:
    2921             :         {
    2922         456 :             CPLAssert(array->type_id() == arrow::Type::LIST);
    2923         456 :             const auto listOfRingsArray =
    2924             :                 static_cast<const arrow::ListArray *>(array);
    2925         456 :             CPLAssert(listOfRingsArray->values()->type_id() ==
    2926             :                       arrow::Type::LIST);
    2927         456 :             GET_PTR_FROM_VALUES(listOfRingsValues, arrow::ListArray,
    2928             :                                 listOfRingsArray->values());
    2929         456 :             CPLAssert(listOfRingsValues->values()->type_id() ==
    2930             :                       arrow::Type::FIXED_SIZE_LIST);
    2931         456 :             GET_PTR_FROM_VALUES(listOfPointsValues, arrow::FixedSizeListArray,
    2932             :                                 listOfRingsValues->values());
    2933         456 :             CPLAssert(listOfPointsValues->values()->type_id() ==
    2934             :                       arrow::Type::DOUBLE);
    2935         456 :             GET_PTR_FROM_VALUES(pointValues, arrow::DoubleArray,
    2936             :                                 listOfPointsValues->values());
    2937         456 :             const auto setPointsFun = GetSetPointsOfLine(bHasZ, bHasM);
    2938         456 :             const auto nRings = listOfRingsArray->value_length(nIdxInBatch);
    2939             :             const auto nRingOffset =
    2940         456 :                 listOfRingsArray->value_offset(nIdxInBatch);
    2941         456 :             auto poPoly = new OGRPolygon();
    2942         456 :             poGeometry = poPoly;
    2943         456 :             poGeometry->assignSpatialReference(
    2944         456 :                 poGeomFieldDefn->GetSpatialRef());
    2945         930 :             for (auto k = decltype(nRings){0}; k < nRings; k++)
    2946             :             {
    2947             :                 const auto nPoints =
    2948         474 :                     listOfRingsValues->value_length(nRingOffset + k);
    2949             :                 const auto nPointOffset =
    2950         474 :                     listOfRingsValues->value_offset(nRingOffset + k) * nDim;
    2951         474 :                 auto poRing = new OGRLinearRing();
    2952         474 :                 if (nPoints)
    2953             :                 {
    2954         474 :                     setPointsFun(poRing, pointValues, nPointOffset, nPoints);
    2955             :                 }
    2956         474 :                 poPoly->addRingDirectly(poRing);
    2957             :             }
    2958         456 :             if (poGeometry->IsEmpty())
    2959             :             {
    2960         119 :                 poGeometry->set3D(bHasZ);
    2961         119 :                 poGeometry->setMeasured(bHasM);
    2962             :             }
    2963         456 :             break;
    2964             :         }
    2965             : 
    2966         148 :         case OGRArrowGeomEncoding::GEOARROW_FSL_MULTIPOINT:
    2967             :         {
    2968         148 :             CPLAssert(array->type_id() == arrow::Type::LIST);
    2969         148 :             const auto listArray = static_cast<const arrow::ListArray *>(array);
    2970         148 :             CPLAssert(listArray->values()->type_id() ==
    2971             :                       arrow::Type::FIXED_SIZE_LIST);
    2972         148 :             GET_PTR_FROM_VALUES(listOfPointsValues, arrow::FixedSizeListArray,
    2973             :                                 listArray->values());
    2974         148 :             CPLAssert(listOfPointsValues->values()->type_id() ==
    2975             :                       arrow::Type::DOUBLE);
    2976         148 :             GET_PTR_FROM_VALUES(pointValues, arrow::DoubleArray,
    2977             :                                 listOfPointsValues->values());
    2978         148 :             const auto nPoints = listArray->value_length(nIdxInBatch);
    2979             :             const auto nPointOffset =
    2980         148 :                 listArray->value_offset(nIdxInBatch) * nDim;
    2981         148 :             auto poMultiPoint = new OGRMultiPoint();
    2982         148 :             poGeometry = poMultiPoint;
    2983         148 :             poGeometry->assignSpatialReference(
    2984         148 :                 poGeomFieldDefn->GetSpatialRef());
    2985         434 :             for (auto k = decltype(nPoints){0}; k < nPoints; k++)
    2986             :             {
    2987         572 :                 poMultiPoint->addGeometryDirectly(
    2988         286 :                     CreatePoint(pointValues, nPointOffset + k * nDim));
    2989             :             }
    2990         148 :             if (poGeometry->IsEmpty())
    2991             :             {
    2992          34 :                 poGeometry->set3D(bHasZ);
    2993          34 :                 poGeometry->setMeasured(bHasM);
    2994             :             }
    2995         148 :             break;
    2996             :         }
    2997             : 
    2998         144 :         case OGRArrowGeomEncoding::GEOARROW_FSL_MULTILINESTRING:
    2999             :         {
    3000         144 :             CPLAssert(array->type_id() == arrow::Type::LIST);
    3001         144 :             const auto listOfStringsArray =
    3002             :                 static_cast<const arrow::ListArray *>(array);
    3003         144 :             CPLAssert(listOfStringsArray->values()->type_id() ==
    3004             :                       arrow::Type::LIST);
    3005         144 :             GET_PTR_FROM_VALUES(listOfStringsValues, arrow::ListArray,
    3006             :                                 listOfStringsArray->values());
    3007         144 :             CPLAssert(listOfStringsValues->values()->type_id() ==
    3008             :                       arrow::Type::FIXED_SIZE_LIST);
    3009         144 :             GET_PTR_FROM_VALUES(listOfPointsValues, arrow::FixedSizeListArray,
    3010             :                                 listOfStringsValues->values());
    3011         144 :             CPLAssert(listOfPointsValues->values()->type_id() ==
    3012             :                       arrow::Type::DOUBLE);
    3013         144 :             GET_PTR_FROM_VALUES(pointValues, arrow::DoubleArray,
    3014             :                                 listOfPointsValues->values());
    3015         144 :             const auto setPointsFun = GetSetPointsOfLine(bHasZ, bHasM);
    3016         144 :             const auto nStrings = listOfStringsArray->value_length(nIdxInBatch);
    3017             :             const auto nRingOffset =
    3018         144 :                 listOfStringsArray->value_offset(nIdxInBatch);
    3019         144 :             auto poMLS = new OGRMultiLineString();
    3020         144 :             poGeometry = poMLS;
    3021         144 :             poGeometry->assignSpatialReference(
    3022         144 :                 poGeomFieldDefn->GetSpatialRef());
    3023         316 :             for (auto k = decltype(nStrings){0}; k < nStrings; k++)
    3024             :             {
    3025             :                 const auto nPoints =
    3026         172 :                     listOfStringsValues->value_length(nRingOffset + k);
    3027             :                 const auto nPointOffset =
    3028         172 :                     listOfStringsValues->value_offset(nRingOffset + k) * nDim;
    3029         172 :                 auto poLS = new OGRLineString();
    3030         172 :                 if (nPoints)
    3031             :                 {
    3032         172 :                     setPointsFun(poLS, pointValues, nPointOffset, nPoints);
    3033             :                 }
    3034         172 :                 poMLS->addGeometryDirectly(poLS);
    3035             :             }
    3036         144 :             if (poGeometry->IsEmpty())
    3037             :             {
    3038          36 :                 poGeometry->set3D(bHasZ);
    3039          36 :                 poGeometry->setMeasured(bHasM);
    3040             :             }
    3041         144 :             break;
    3042             :         }
    3043             : 
    3044         160 :         case OGRArrowGeomEncoding::GEOARROW_FSL_MULTIPOLYGON:
    3045             :         {
    3046         160 :             CPLAssert(array->type_id() == arrow::Type::LIST);
    3047         160 :             const auto listOfPartsArray =
    3048             :                 static_cast<const arrow::ListArray *>(array);
    3049         160 :             CPLAssert(listOfPartsArray->values()->type_id() ==
    3050             :                       arrow::Type::LIST);
    3051         160 :             GET_PTR_FROM_VALUES(listOfPartsValues, arrow::ListArray,
    3052             :                                 listOfPartsArray->values());
    3053         160 :             CPLAssert(listOfPartsValues->values()->type_id() ==
    3054             :                       arrow::Type::LIST);
    3055         160 :             GET_PTR_FROM_VALUES(listOfRingsValues, arrow::ListArray,
    3056             :                                 listOfPartsValues->values());
    3057         160 :             CPLAssert(listOfRingsValues->values()->type_id() ==
    3058             :                       arrow::Type::FIXED_SIZE_LIST);
    3059         160 :             GET_PTR_FROM_VALUES(listOfPointsValues, arrow::FixedSizeListArray,
    3060             :                                 listOfRingsValues->values());
    3061         160 :             CPLAssert(listOfPointsValues->values()->type_id() ==
    3062             :                       arrow::Type::DOUBLE);
    3063         160 :             GET_PTR_FROM_VALUES(pointValues, arrow::DoubleArray,
    3064             :                                 listOfPointsValues->values());
    3065         160 :             auto poMP = new OGRMultiPolygon();
    3066         160 :             poGeometry = poMP;
    3067         160 :             poGeometry->assignSpatialReference(
    3068         160 :                 poGeomFieldDefn->GetSpatialRef());
    3069         160 :             const auto setPointsFun = GetSetPointsOfLine(bHasZ, bHasM);
    3070         160 :             const auto nParts = listOfPartsArray->value_length(nIdxInBatch);
    3071             :             const auto nPartOffset =
    3072         160 :                 listOfPartsArray->value_offset(nIdxInBatch);
    3073         356 :             for (auto j = decltype(nParts){0}; j < nParts; j++)
    3074             :             {
    3075             :                 const auto nRings =
    3076         196 :                     listOfPartsValues->value_length(nPartOffset + j);
    3077             :                 const auto nRingOffset =
    3078         196 :                     listOfPartsValues->value_offset(nPartOffset + j);
    3079         196 :                 auto poPoly = new OGRPolygon();
    3080         434 :                 for (auto k = decltype(nRings){0}; k < nRings; k++)
    3081             :                 {
    3082             :                     const auto nPoints =
    3083         238 :                         listOfRingsValues->value_length(nRingOffset + k);
    3084             :                     const auto nPointOffset =
    3085         238 :                         listOfRingsValues->value_offset(nRingOffset + k) * nDim;
    3086         238 :                     auto poRing = new OGRLinearRing();
    3087         238 :                     if (nPoints)
    3088             :                     {
    3089         238 :                         setPointsFun(poRing, pointValues, nPointOffset,
    3090             :                                      nPoints);
    3091             :                     }
    3092         238 :                     poPoly->addRingDirectly(poRing);
    3093             :                 }
    3094         196 :                 poMP->addGeometryDirectly(poPoly);
    3095             :             }
    3096         160 :             if (poGeometry->IsEmpty())
    3097             :             {
    3098          36 :                 poGeometry->set3D(bHasZ);
    3099          36 :                 poGeometry->setMeasured(bHasM);
    3100             :             }
    3101         160 :             break;
    3102             :         }
    3103             : 
    3104         311 :         case OGRArrowGeomEncoding::GEOARROW_STRUCT_POINT:
    3105             :         {
    3106         311 :             CPLAssert(array->type_id() == arrow::Type::STRUCT);
    3107         311 :             const auto structArray =
    3108             :                 static_cast<const arrow::StructArray *>(array);
    3109         311 :             if (!structArray->IsNull(nIdxInBatch))
    3110             :             {
    3111         311 :                 poGeometry = CreateStructPoint(structArray, nIdxInBatch);
    3112         311 :                 poGeometry->assignSpatialReference(
    3113         311 :                     poGeomFieldDefn->GetSpatialRef());
    3114             :             }
    3115         311 :             break;
    3116             :         }
    3117             : 
    3118         200 :         case OGRArrowGeomEncoding::GEOARROW_STRUCT_LINESTRING:
    3119             :         {
    3120         200 :             CPLAssert(array->type_id() == arrow::Type::LIST);
    3121         200 :             const auto listArray = static_cast<const arrow::ListArray *>(array);
    3122         200 :             CPLAssert(listArray->values()->type_id() == arrow::Type::STRUCT);
    3123         200 :             GET_PTR_FROM_VALUES(pointValues, arrow::StructArray,
    3124             :                                 listArray->values());
    3125         200 :             const auto nPoints = listArray->value_length(nIdxInBatch);
    3126         200 :             const auto nPointOffset = listArray->value_offset(nIdxInBatch);
    3127         200 :             auto poLineString = new OGRLineString();
    3128         200 :             poGeometry = poLineString;
    3129         200 :             poGeometry->assignSpatialReference(
    3130         200 :                 poGeomFieldDefn->GetSpatialRef());
    3131         200 :             if (nPoints)
    3132             :             {
    3133         164 :                 GetSetPointsOfLineStruct(bHasZ, bHasM)(
    3134             :                     poLineString, pointValues, nPointOffset, nPoints);
    3135             :             }
    3136             :             else
    3137             :             {
    3138          36 :                 poGeometry->set3D(bHasZ);
    3139          36 :                 poGeometry->setMeasured(bHasM);
    3140             :             }
    3141         200 :             break;
    3142             :         }
    3143             : 
    3144         316 :         case OGRArrowGeomEncoding::GEOARROW_STRUCT_POLYGON:
    3145             :         {
    3146         316 :             CPLAssert(array->type_id() == arrow::Type::LIST);
    3147         316 :             const auto listOfRingsArray =
    3148             :                 static_cast<const arrow::ListArray *>(array);
    3149         316 :             CPLAssert(listOfRingsArray->values()->type_id() ==
    3150             :                       arrow::Type::LIST);
    3151         316 :             GET_PTR_FROM_VALUES(listOfRingsValues, arrow::ListArray,
    3152             :                                 listOfRingsArray->values());
    3153         316 :             CPLAssert(listOfRingsValues->values()->type_id() ==
    3154             :                       arrow::Type::STRUCT);
    3155         316 :             GET_PTR_FROM_VALUES(pointValues, arrow::StructArray,
    3156             :                                 listOfRingsValues->values());
    3157         316 :             const auto setPointsFun = GetSetPointsOfLineStruct(bHasZ, bHasM);
    3158         316 :             const auto nRings = listOfRingsArray->value_length(nIdxInBatch);
    3159             :             const auto nRingOffset =
    3160         316 :                 listOfRingsArray->value_offset(nIdxInBatch);
    3161         316 :             auto poPoly = new OGRPolygon();
    3162         316 :             poGeometry = poPoly;
    3163         316 :             poGeometry->assignSpatialReference(
    3164         316 :                 poGeomFieldDefn->GetSpatialRef());
    3165         648 :             for (auto k = decltype(nRings){0}; k < nRings; k++)
    3166             :             {
    3167             :                 const auto nPoints =
    3168         332 :                     listOfRingsValues->value_length(nRingOffset + k);
    3169             :                 const auto nPointOffset =
    3170         332 :                     listOfRingsValues->value_offset(nRingOffset + k);
    3171         332 :                 auto poRing = new OGRLinearRing();
    3172         332 :                 if (nPoints)
    3173             :                 {
    3174         332 :                     setPointsFun(poRing, pointValues, nPointOffset, nPoints);
    3175             :                 }
    3176         332 :                 poPoly->addRingDirectly(poRing);
    3177             :             }
    3178         316 :             if (poGeometry->IsEmpty())
    3179             :             {
    3180          52 :                 poGeometry->set3D(bHasZ);
    3181          52 :                 poGeometry->setMeasured(bHasM);
    3182             :             }
    3183         316 :             break;
    3184             :         }
    3185             : 
    3186         240 :         case OGRArrowGeomEncoding::GEOARROW_STRUCT_MULTIPOINT:
    3187             :         {
    3188         240 :             CPLAssert(array->type_id() == arrow::Type::LIST);
    3189         240 :             const auto listArray = static_cast<const arrow::ListArray *>(array);
    3190         240 :             CPLAssert(listArray->values()->type_id() == arrow::Type::STRUCT);
    3191         240 :             GET_PTR_FROM_VALUES(pointValues, arrow::StructArray,
    3192             :                                 listArray->values());
    3193         240 :             const auto nPoints = listArray->value_length(nIdxInBatch);
    3194         240 :             const auto nPointOffset = listArray->value_offset(nIdxInBatch);
    3195         240 :             auto poMultiPoint = new OGRMultiPoint();
    3196         240 :             poGeometry = poMultiPoint;
    3197         240 :             poGeometry->assignSpatialReference(
    3198         240 :                 poGeomFieldDefn->GetSpatialRef());
    3199         596 :             for (auto k = decltype(nPoints){0}; k < nPoints; k++)
    3200             :             {
    3201         712 :                 poMultiPoint->addGeometryDirectly(
    3202         356 :                     CreateStructPoint(pointValues, nPointOffset + k));
    3203             :             }
    3204         240 :             if (poGeometry->IsEmpty())
    3205             :             {
    3206          36 :                 poGeometry->set3D(bHasZ);
    3207          36 :                 poGeometry->setMeasured(bHasM);
    3208             :             }
    3209         240 :             break;
    3210             :         }
    3211             : 
    3212         284 :         case OGRArrowGeomEncoding::GEOARROW_STRUCT_MULTILINESTRING:
    3213             :         {
    3214         284 :             CPLAssert(array->type_id() == arrow::Type::LIST);
    3215         284 :             const auto listOfStringsArray =
    3216             :                 static_cast<const arrow::ListArray *>(array);
    3217         284 :             CPLAssert(listOfStringsArray->values()->type_id() ==
    3218             :                       arrow::Type::LIST);
    3219         284 :             GET_PTR_FROM_VALUES(listOfStringsValues, arrow::ListArray,
    3220             :                                 listOfStringsArray->values());
    3221         284 :             CPLAssert(listOfStringsValues->values()->type_id() ==
    3222             :                       arrow::Type::STRUCT);
    3223         284 :             GET_PTR_FROM_VALUES(pointValues, arrow::StructArray,
    3224             :                                 listOfStringsValues->values());
    3225         284 :             const auto setPointsFun = GetSetPointsOfLineStruct(bHasZ, bHasM);
    3226         284 :             const auto nStrings = listOfStringsArray->value_length(nIdxInBatch);
    3227             :             const auto nRingOffset =
    3228         284 :                 listOfStringsArray->value_offset(nIdxInBatch);
    3229         284 :             auto poMLS = new OGRMultiLineString();
    3230         284 :             poGeometry = poMLS;
    3231         284 :             poGeometry->assignSpatialReference(
    3232         284 :                 poGeomFieldDefn->GetSpatialRef());
    3233         696 :             for (auto k = decltype(nStrings){0}; k < nStrings; k++)
    3234             :             {
    3235             :                 const auto nPoints =
    3236         412 :                     listOfStringsValues->value_length(nRingOffset + k);
    3237             :                 const auto nPointOffset =
    3238         412 :                     listOfStringsValues->value_offset(nRingOffset + k);
    3239         412 :                 auto poLS = new OGRLineString();
    3240         412 :                 if (nPoints)
    3241             :                 {
    3242         412 :                     setPointsFun(poLS, pointValues, nPointOffset, nPoints);
    3243             :                 }
    3244         412 :                 poMLS->addGeometryDirectly(poLS);
    3245             :             }
    3246         284 :             if (poGeometry->IsEmpty())
    3247             :             {
    3248          36 :                 poGeometry->set3D(bHasZ);
    3249          36 :                 poGeometry->setMeasured(bHasM);
    3250             :             }
    3251         284 :             break;
    3252             :         }
    3253             : 
    3254         336 :         case OGRArrowGeomEncoding::GEOARROW_STRUCT_MULTIPOLYGON:
    3255             :         {
    3256         336 :             CPLAssert(array->type_id() == arrow::Type::LIST);
    3257         336 :             const auto listOfPartsArray =
    3258             :                 static_cast<const arrow::ListArray *>(array);
    3259         336 :             CPLAssert(listOfPartsArray->values()->type_id() ==
    3260             :                       arrow::Type::LIST);
    3261         336 :             GET_PTR_FROM_VALUES(listOfPartsValues, arrow::ListArray,
    3262             :                                 listOfPartsArray->values());
    3263         336 :             CPLAssert(listOfPartsValues->values()->type_id() ==
    3264             :                       arrow::Type::LIST);
    3265         336 :             GET_PTR_FROM_VALUES(listOfRingsValues, arrow::ListArray,
    3266             :                                 listOfPartsValues->values());
    3267         336 :             CPLAssert(listOfRingsValues->values()->type_id() ==
    3268             :                       arrow::Type::STRUCT);
    3269         336 :             GET_PTR_FROM_VALUES(pointValues, arrow::StructArray,
    3270             :                                 listOfRingsValues->values());
    3271         336 :             auto poMP = new OGRMultiPolygon();
    3272         336 :             poGeometry = poMP;
    3273         336 :             poGeometry->assignSpatialReference(
    3274         336 :                 poGeomFieldDefn->GetSpatialRef());
    3275         336 :             const auto setPointsFun = GetSetPointsOfLineStruct(bHasZ, bHasM);
    3276         336 :             const auto nParts = listOfPartsArray->value_length(nIdxInBatch);
    3277             :             const auto nPartOffset =
    3278         336 :                 listOfPartsArray->value_offset(nIdxInBatch);
    3279         756 :             for (auto j = decltype(nParts){0}; j < nParts; j++)
    3280             :             {
    3281             :                 const auto nRings =
    3282         420 :                     listOfPartsValues->value_length(nPartOffset + j);
    3283             :                 const auto nRingOffset =
    3284         420 :                     listOfPartsValues->value_offset(nPartOffset + j);
    3285         420 :                 auto poPoly = new OGRPolygon();
    3286         940 :                 for (auto k = decltype(nRings){0}; k < nRings; k++)
    3287             :                 {
    3288             :                     const auto nPoints =
    3289         520 :                         listOfRingsValues->value_length(nRingOffset + k);
    3290             :                     const auto nPointOffset =
    3291         520 :                         listOfRingsValues->value_offset(nRingOffset + k);
    3292         520 :                     auto poRing = new OGRLinearRing();
    3293         520 :                     if (nPoints)
    3294             :                     {
    3295         520 :                         setPointsFun(poRing, pointValues, nPointOffset,
    3296             :                                      nPoints);
    3297             :                     }
    3298         520 :                     poPoly->addRingDirectly(poRing);
    3299             :                 }
    3300         420 :                 poMP->addGeometryDirectly(poPoly);
    3301             :             }
    3302         336 :             if (poGeometry->IsEmpty())
    3303             :             {
    3304          52 :                 poGeometry->set3D(bHasZ);
    3305          52 :                 poGeometry->setMeasured(bHasM);
    3306             :             }
    3307         336 :             break;
    3308             :         }
    3309             :     }
    3310        7088 :     return poGeometry;
    3311             : }
    3312             : 
    3313             : /************************************************************************/
    3314             : /*                           ResetReading()                             */
    3315             : /************************************************************************/
    3316             : 
    3317        7121 : inline void OGRArrowLayer::ResetReading()
    3318             : {
    3319        7121 :     m_bEOF = false;
    3320        7121 :     m_nFeatureIdx = 0;
    3321        7121 :     m_nIdxInBatch = 0;
    3322        7121 :     m_poReadFeatureTmpArray.reset();
    3323        7121 :     if (m_iRecordBatch != 0)
    3324             :     {
    3325        6307 :         m_iRecordBatch = -1;
    3326        6307 :         m_poBatch.reset();
    3327        6307 :         m_poBatchColumns.clear();
    3328             :     }
    3329        7121 : }
    3330             : 
    3331             : /***********************************************************************/
    3332             : /*                        GetColumnSubNode()                           */
    3333             : /***********************************************************************/
    3334             : 
    3335             : /* static*/
    3336             : inline const swq_expr_node *
    3337         284 : OGRArrowLayer::GetColumnSubNode(const swq_expr_node *poNode)
    3338             : {
    3339         284 :     if (poNode->eNodeType == SNT_OPERATION && poNode->nSubExprCount == 2)
    3340             :     {
    3341         284 :         if (poNode->papoSubExpr[0]->eNodeType == SNT_COLUMN)
    3342         202 :             return poNode->papoSubExpr[0];
    3343          82 :         if (poNode->papoSubExpr[1]->eNodeType == SNT_COLUMN)
    3344          12 :             return poNode->papoSubExpr[1];
    3345             :     }
    3346          70 :     return nullptr;
    3347             : }
    3348             : 
    3349             : /***********************************************************************/
    3350             : /*                        GetConstantSubNode()                         */
    3351             : /***********************************************************************/
    3352             : 
    3353             : /* static */
    3354             : inline const swq_expr_node *
    3355         284 : OGRArrowLayer::GetConstantSubNode(const swq_expr_node *poNode)
    3356             : {
    3357         284 :     if (poNode->eNodeType == SNT_OPERATION && poNode->nSubExprCount == 2)
    3358             :     {
    3359         284 :         if (poNode->papoSubExpr[1]->eNodeType == SNT_CONSTANT)
    3360         270 :             return poNode->papoSubExpr[1];
    3361          14 :         if (poNode->papoSubExpr[0]->eNodeType == SNT_CONSTANT)
    3362          12 :             return poNode->papoSubExpr[0];
    3363             :     }
    3364           2 :     return nullptr;
    3365             : }
    3366             : 
    3367             : /***********************************************************************/
    3368             : /*                           IsComparisonOp()                          */
    3369             : /***********************************************************************/
    3370             : 
    3371             : /* static*/
    3372         428 : inline bool OGRArrowLayer::IsComparisonOp(int op)
    3373             : {
    3374         166 :     return (op == SWQ_EQ || op == SWQ_NE || op == SWQ_LT || op == SWQ_LE ||
    3375         594 :             op == SWQ_GT || op == SWQ_GE);
    3376             : }
    3377             : 
    3378             : /***********************************************************************/
    3379             : /*                     FillTargetValueFromSrcExpr()                    */
    3380             : /***********************************************************************/
    3381             : 
    3382         212 : static bool FillTargetValueFromSrcExpr(const OGRFieldDefn *poFieldDefn,
    3383             :                                        OGRArrowLayer::Constraint *psConstraint,
    3384             :                                        const swq_expr_node *poSrcValue)
    3385             : {
    3386         212 :     switch (poFieldDefn->GetType())
    3387             :     {
    3388         111 :         case OFTInteger:
    3389         111 :             psConstraint->eType = OGRArrowLayer::Constraint::Type::Integer;
    3390         111 :             if (poSrcValue->field_type == SWQ_FLOAT)
    3391           2 :                 psConstraint->sValue.Integer =
    3392           2 :                     static_cast<int>(poSrcValue->float_value);
    3393             :             else
    3394         109 :                 psConstraint->sValue.Integer =
    3395         109 :                     static_cast<int>(poSrcValue->int_value);
    3396             :             psConstraint->osValue =
    3397         111 :                 std::to_string(psConstraint->sValue.Integer);
    3398         111 :             break;
    3399             : 
    3400          29 :         case OFTInteger64:
    3401          29 :             psConstraint->eType = OGRArrowLayer::Constraint::Type::Integer64;
    3402          29 :             if (poSrcValue->field_type == SWQ_FLOAT)
    3403           0 :                 psConstraint->sValue.Integer64 =
    3404           0 :                     static_cast<GIntBig>(poSrcValue->float_value);
    3405             :             else
    3406          29 :                 psConstraint->sValue.Integer64 = poSrcValue->int_value;
    3407             :             psConstraint->osValue =
    3408          29 :                 std::to_string(psConstraint->sValue.Integer64);
    3409          29 :             break;
    3410             : 
    3411          25 :         case OFTReal:
    3412          25 :             psConstraint->eType = OGRArrowLayer::Constraint::Type::Real;
    3413          25 :             psConstraint->sValue.Real = poSrcValue->float_value;
    3414          25 :             psConstraint->osValue = std::to_string(psConstraint->sValue.Real);
    3415          25 :             break;
    3416             : 
    3417          22 :         case OFTString:
    3418          22 :             psConstraint->eType = OGRArrowLayer::Constraint::Type::String;
    3419          22 :             psConstraint->sValue.String = poSrcValue->string_value;
    3420          22 :             psConstraint->osValue = psConstraint->sValue.String;
    3421          22 :             break;
    3422             : #ifdef not_yet_handled
    3423             :         case OFTDate:
    3424             :         case OFTTime:
    3425             :         case OFTDateTime:
    3426             :             if (poSrcValue->field_type == SWQ_TIMESTAMP ||
    3427             :                 poSrcValue->field_type == SWQ_DATE ||
    3428             :                 poSrcValue->field_type == SWQ_TIME)
    3429             :             {
    3430             :                 int nYear = 0, nMonth = 0, nDay = 0, nHour = 0, nMin = 0,
    3431             :                     nSec = 0;
    3432             :                 if (sscanf(poSrcValue->string_value,
    3433             :                            "%04d/%02d/%02d %02d:%02d:%02d", &nYear, &nMonth,
    3434             :                            &nDay, &nHour, &nMin, &nSec) == 6 ||
    3435             :                     sscanf(poSrcValue->string_value, "%04d/%02d/%02d", &nYear,
    3436             :                            &nMonth, &nDay) == 3 ||
    3437             :                     sscanf(poSrcValue->string_value, "%02d:%02d:%02d", &nHour,
    3438             :                            &nMin, &nSec) == 3)
    3439             :                 {
    3440             :                     psConstraint->eType =
    3441             :                         OGRArrowLayer::Constraint::Type::DateTime;
    3442             :                     psConstraint->sValue.Date.Year = (GInt16)nYear;
    3443             :                     psConstraint->sValue.Date.Month = (GByte)nMonth;
    3444             :                     psConstraint->sValue.Date.Day = (GByte)nDay;
    3445             :                     psConstraint->sValue.Date.Hour = (GByte)nHour;
    3446             :                     psConstraint->sValue.Date.Minute = (GByte)nMin;
    3447             :                     psConstraint->sValue.Date.Second = (GByte)nSec;
    3448             :                     psConstraint->sValue.Date.TZFlag = 0;
    3449             :                     psConstraint->sValue.Date.Reserved = 0;
    3450             :                 }
    3451             :                 else
    3452             :                     return false;
    3453             :             }
    3454             :             else
    3455             :                 return false;
    3456             :             break;
    3457             : #endif
    3458          25 :         default:
    3459          25 :             return false;
    3460             :     }
    3461         187 :     return true;
    3462             : }
    3463             : 
    3464             : /***********************************************************************/
    3465             : /*                  ComputeConstraintsArrayIdx()                       */
    3466             : /***********************************************************************/
    3467             : 
    3468         530 : inline void OGRArrowLayer::ComputeConstraintsArrayIdx()
    3469             : {
    3470         731 :     for (auto &constraint : m_asAttributeFilterConstraints)
    3471             :     {
    3472         202 :         if (m_bIgnoredFields)
    3473             :         {
    3474          25 :             if (constraint.iField == m_poFeatureDefn->GetFieldCount() + SPF_FID)
    3475             :             {
    3476           1 :                 constraint.iArrayIdx = m_nRequestedFIDColumn;
    3477           1 :                 if (constraint.iArrayIdx < 0 && m_osFIDColumn.empty())
    3478           1 :                     return;
    3479             :             }
    3480             :             else
    3481             :             {
    3482          24 :                 constraint.iArrayIdx =
    3483          24 :                     m_anMapFieldIndexToArrayIndex[constraint.iField];
    3484             :             }
    3485          24 :             if (constraint.iArrayIdx < 0)
    3486             :             {
    3487           1 :                 CPLError(CE_Failure, CPLE_AppDefined,
    3488             :                          "Constraint on field %s cannot be applied due to "
    3489             :                          "it being ignored",
    3490           1 :                          constraint.iField ==
    3491           1 :                                  m_poFeatureDefn->GetFieldCount() + SPF_FID
    3492           0 :                              ? m_osFIDColumn.c_str()
    3493           1 :                              : m_poFeatureDefn->GetFieldDefn(constraint.iField)
    3494           1 :                                    ->GetNameRef());
    3495             :             }
    3496             :         }
    3497             :         else
    3498             :         {
    3499         177 :             if (constraint.iField == m_poFeatureDefn->GetFieldCount() + SPF_FID)
    3500             :             {
    3501           8 :                 constraint.iArrayIdx = m_iFIDArrowColumn;
    3502           8 :                 if (constraint.iArrayIdx < 0 && !m_osFIDColumn.empty())
    3503             :                 {
    3504           0 :                     CPLDebug(GetDriverUCName().c_str(),
    3505             :                              "Constraint on field %s cannot be applied",
    3506             :                              m_osFIDColumn.c_str());
    3507             :                 }
    3508             :             }
    3509             :             else
    3510             :             {
    3511         169 :                 constraint.iArrayIdx =
    3512         169 :                     m_anMapFieldIndexToArrowColumn[constraint.iField][0];
    3513             :             }
    3514             :         }
    3515             :     }
    3516             : }
    3517             : 
    3518             : /***********************************************************************/
    3519             : /*                     ExploreExprNode()                               */
    3520             : /***********************************************************************/
    3521             : 
    3522         347 : inline void OGRArrowLayer::ExploreExprNode(const swq_expr_node *poNode)
    3523             : {
    3524         199 :     const auto AddConstraint = [this](Constraint &constraint)
    3525         199 :     { m_asAttributeFilterConstraints.emplace_back(constraint); };
    3526             : 
    3527         347 :     if (poNode->eNodeType == SNT_OPERATION && poNode->nOperation == SWQ_AND &&
    3528          18 :         poNode->nSubExprCount == 2)
    3529             :     {
    3530          18 :         ExploreExprNode(poNode->papoSubExpr[0]);
    3531          18 :         ExploreExprNode(poNode->papoSubExpr[1]);
    3532             :     }
    3533             : 
    3534         329 :     else if (poNode->eNodeType == SNT_OPERATION &&
    3535         658 :              IsComparisonOp(poNode->nOperation) && poNode->nSubExprCount == 2)
    3536             :     {
    3537         284 :         const swq_expr_node *poColumn = GetColumnSubNode(poNode);
    3538         284 :         const swq_expr_node *poValue = GetConstantSubNode(poNode);
    3539         496 :         if (poColumn != nullptr && poValue != nullptr &&
    3540         212 :             (poColumn->field_index < m_poFeatureDefn->GetFieldCount() ||
    3541           8 :              poColumn->field_index ==
    3542           8 :                  m_poFeatureDefn->GetFieldCount() + SPF_FID))
    3543             :         {
    3544             :             const OGRFieldDefn oDummyFIDFieldDefn(m_osFIDColumn.c_str(),
    3545         424 :                                                   OFTInteger64);
    3546             :             const OGRFieldDefn *poFieldDefn =
    3547         212 :                 (poColumn->field_index ==
    3548         212 :                  m_poFeatureDefn->GetFieldCount() + SPF_FID)
    3549         212 :                     ? &oDummyFIDFieldDefn
    3550         204 :                     : m_poFeatureDefn->GetFieldDefn(poColumn->field_index);
    3551             : 
    3552         424 :             Constraint constraint;
    3553         212 :             constraint.iField = poColumn->field_index;
    3554         212 :             constraint.nOperation = poNode->nOperation;
    3555             : 
    3556         212 :             if (FillTargetValueFromSrcExpr(poFieldDefn, &constraint, poValue))
    3557             :             {
    3558         187 :                 if (poColumn == poNode->papoSubExpr[0])
    3559             :                 {
    3560             :                     // nothing to do
    3561             :                 }
    3562             :                 else
    3563             :                 {
    3564             :                     /* If "constant op column", then we must reverse */
    3565             :                     /* the operator for LE, LT, GE, GT */
    3566          12 :                     switch (poNode->nOperation)
    3567             :                     {
    3568           2 :                         case SWQ_LE:
    3569           2 :                             constraint.nOperation = SWQ_GE;
    3570           2 :                             break;
    3571           2 :                         case SWQ_LT:
    3572           2 :                             constraint.nOperation = SWQ_GT;
    3573           2 :                             break;
    3574           2 :                         case SWQ_NE: /* do nothing */;
    3575           2 :                             break;
    3576           2 :                         case SWQ_EQ: /* do nothing */;
    3577           2 :                             break;
    3578           2 :                         case SWQ_GE:
    3579           2 :                             constraint.nOperation = SWQ_LE;
    3580           2 :                             break;
    3581           2 :                         case SWQ_GT:
    3582           2 :                             constraint.nOperation = SWQ_LT;
    3583           2 :                             break;
    3584           0 :                         default:
    3585           0 :                             CPLAssert(false);
    3586             :                             break;
    3587             :                     }
    3588             :                 }
    3589             : 
    3590         187 :                 AddConstraint(constraint);
    3591             :             }
    3592             :         }
    3593             :     }
    3594             : 
    3595          45 :     else if (poNode->eNodeType == SNT_OPERATION &&
    3596          45 :              poNode->nOperation == SWQ_ISNULL && poNode->nSubExprCount == 1)
    3597             :     {
    3598           4 :         const swq_expr_node *poColumn = poNode->papoSubExpr[0];
    3599           8 :         if (poColumn->eNodeType == SNT_COLUMN &&
    3600           4 :             poColumn->field_index < m_poFeatureDefn->GetFieldCount())
    3601             :         {
    3602           8 :             Constraint constraint;
    3603           4 :             constraint.iField = poColumn->field_index;
    3604           4 :             constraint.nOperation = poNode->nOperation;
    3605           4 :             AddConstraint(constraint);
    3606           4 :         }
    3607             :     }
    3608             : 
    3609          41 :     else if (poNode->eNodeType == SNT_OPERATION &&
    3610          41 :              poNode->nOperation == SWQ_NOT && poNode->nSubExprCount == 1 &&
    3611          20 :              poNode->papoSubExpr[0]->eNodeType == SNT_OPERATION &&
    3612          20 :              poNode->papoSubExpr[0]->nOperation == SWQ_ISNULL &&
    3613          16 :              poNode->papoSubExpr[0]->nSubExprCount == 1)
    3614             :     {
    3615          16 :         const swq_expr_node *poColumn = poNode->papoSubExpr[0]->papoSubExpr[0];
    3616          32 :         if (poColumn->eNodeType == SNT_COLUMN &&
    3617          16 :             poColumn->field_index < m_poFeatureDefn->GetFieldCount())
    3618             :         {
    3619          16 :             Constraint constraint;
    3620           8 :             constraint.iField = poColumn->field_index;
    3621           8 :             constraint.nOperation = SWQ_ISNOTNULL;
    3622           8 :             AddConstraint(constraint);
    3623             :         }
    3624             :     }
    3625         347 : }
    3626             : 
    3627             : /***********************************************************************/
    3628             : /*                         SetAttributeFilter()                        */
    3629             : /***********************************************************************/
    3630             : 
    3631         748 : inline OGRErr OGRArrowLayer::SetAttributeFilter(const char *pszFilter)
    3632             : {
    3633         748 :     m_asAttributeFilterConstraints.clear();
    3634             : 
    3635             :     // When changing filters, we need to invalidate cached batches, as
    3636             :     // PostFilterArrowArray() has potentially modified array contents
    3637         748 :     if (m_poAttrQuery)
    3638         110 :         InvalidateCachedBatches();
    3639             : 
    3640         748 :     OGRErr eErr = OGRLayer::SetAttributeFilter(pszFilter);
    3641         748 :     if (eErr != OGRERR_NONE)
    3642           0 :         return eErr;
    3643             : 
    3644         748 :     if (m_poAttrQuery != nullptr)
    3645             :     {
    3646         461 :         if (m_nUseOptimizedAttributeFilter < 0)
    3647             :         {
    3648         364 :             m_nUseOptimizedAttributeFilter = CPLTestBool(CPLGetConfigOption(
    3649         728 :                 ("OGR_" + GetDriverUCName() + "_OPTIMIZED_ATTRIBUTE_FILTER")
    3650             :                     .c_str(),
    3651             :                 "YES"));
    3652             :         }
    3653         461 :         if (m_nUseOptimizedAttributeFilter)
    3654             :         {
    3655             :             swq_expr_node *poNode =
    3656         311 :                 static_cast<swq_expr_node *>(m_poAttrQuery->GetSWQExpr());
    3657         311 :             poNode->ReplaceBetweenByGEAndLERecurse();
    3658         311 :             ExploreExprNode(poNode);
    3659         311 :             ComputeConstraintsArrayIdx();
    3660             :         }
    3661             :     }
    3662             : 
    3663         748 :     return OGRERR_NONE;
    3664             : }
    3665             : 
    3666             : /************************************************************************/
    3667             : /*                        ConstraintEvaluator()                         */
    3668             : /************************************************************************/
    3669             : 
    3670             : namespace
    3671             : {
    3672             : template <class T, class U> struct CompareGeneric
    3673             : {
    3674         289 :     static inline bool get(int op, const T &val1, const U &val2)
    3675             :     {
    3676         289 :         switch (op)
    3677             :         {
    3678           8 :             case SWQ_LE:
    3679           8 :                 return val1 <= val2;
    3680           8 :             case SWQ_LT:
    3681           8 :                 return val1 < val2;
    3682          90 :             case SWQ_NE:
    3683          90 :                 return val1 != val2;
    3684         163 :             case SWQ_EQ:
    3685         163 :                 return val1 == val2;
    3686          12 :             case SWQ_GE:
    3687          12 :                 return val1 >= val2;
    3688           8 :             case SWQ_GT:
    3689           8 :                 return val1 > val2;
    3690             :                 break;
    3691           0 :             default:
    3692           0 :                 CPLAssert(false);
    3693             :         }
    3694             :         return true;
    3695             :     }
    3696             : };
    3697             : 
    3698             : template <class T, class U> struct Compare
    3699             : {
    3700             : };
    3701             : 
    3702             : template <class T> struct Compare<T, T> : public CompareGeneric<T, T>
    3703             : {
    3704             : };
    3705             : 
    3706             : template <> struct Compare<int, GIntBig> : public CompareGeneric<int, GIntBig>
    3707             : {
    3708             : };
    3709             : 
    3710             : template <> struct Compare<double, GIntBig>
    3711             : {
    3712           0 :     static inline bool get(int op, double val1, GIntBig val2)
    3713             :     {
    3714           0 :         return CompareGeneric<double, double>::get(op, val1,
    3715           0 :                                                    static_cast<double>(val2));
    3716             :     }
    3717             : };
    3718             : 
    3719             : template <> struct Compare<GIntBig, int> : public CompareGeneric<GIntBig, int>
    3720             : {
    3721             : };
    3722             : 
    3723             : template <> struct Compare<double, int> : public CompareGeneric<double, int>
    3724             : {
    3725             : };
    3726             : 
    3727             : template <class T>
    3728         289 : static bool ConstraintEvaluator(const OGRArrowLayer::Constraint &constraint,
    3729             :                                 const T value)
    3730             : {
    3731         289 :     bool b = false;
    3732         289 :     switch (constraint.eType)
    3733             :     {
    3734         204 :         case OGRArrowLayer::Constraint::Type::Integer:
    3735         408 :             b = Compare<T, int>::get(constraint.nOperation, value,
    3736         204 :                                      constraint.sValue.Integer);
    3737         204 :             break;
    3738          43 :         case OGRArrowLayer::Constraint::Type::Integer64:
    3739          86 :             b = Compare<T, GIntBig>::get(constraint.nOperation, value,
    3740          43 :                                          constraint.sValue.Integer64);
    3741          43 :             break;
    3742          42 :         case OGRArrowLayer::Constraint::Type::Real:
    3743          84 :             b = Compare<double, double>::get(constraint.nOperation,
    3744           0 :                                              static_cast<double>(value),
    3745          42 :                                              constraint.sValue.Real);
    3746          42 :             break;
    3747           0 :         case OGRArrowLayer::Constraint::Type::String:
    3748           0 :             b = Compare<std::string, std::string>::get(constraint.nOperation,
    3749             :                                                        std::to_string(value),
    3750           0 :                                                        constraint.osValue);
    3751           0 :             break;
    3752             :     }
    3753         289 :     return b;
    3754             : }
    3755             : 
    3756          45 : inline bool CompareStr(int op, const std::string_view &val1,
    3757             :                        const std::string &val2)
    3758             : {
    3759          45 :     if (op == SWQ_EQ)
    3760             :     {
    3761          14 :         return val1 == val2;
    3762             :     }
    3763          31 :     const int cmpRes = val2.compare(val1);
    3764          31 :     switch (op)
    3765             :     {
    3766           4 :         case SWQ_LE:
    3767           4 :             return cmpRes >= 0;
    3768           4 :         case SWQ_LT:
    3769           4 :             return cmpRes > 0;
    3770          23 :         case SWQ_NE:
    3771          23 :             return cmpRes != 0;
    3772             :         // case SWQ_EQ: return cmpRes == 0;
    3773           0 :         case SWQ_GE:
    3774           0 :             return cmpRes <= 0;
    3775           0 :         case SWQ_GT:
    3776           0 :             return cmpRes < 0;
    3777             :             break;
    3778           0 :         default:
    3779           0 :             CPLAssert(false);
    3780             :     }
    3781             :     return true;
    3782             : }
    3783             : 
    3784          45 : inline bool ConstraintEvaluator(const OGRArrowLayer::Constraint &constraint,
    3785             :                                 const std::string_view &value)
    3786             : {
    3787          45 :     return CompareStr(constraint.nOperation, value, constraint.osValue);
    3788             : }
    3789             : 
    3790             : }  // namespace
    3791             : 
    3792             : /************************************************************************/
    3793             : /*                 SkipToNextFeatureDueToAttributeFilter()              */
    3794             : /************************************************************************/
    3795             : 
    3796         441 : inline bool OGRArrowLayer::SkipToNextFeatureDueToAttributeFilter() const
    3797             : {
    3798         679 :     for (const auto &constraint : m_asAttributeFilterConstraints)
    3799             :     {
    3800         441 :         if (constraint.iArrayIdx < 0)
    3801             :         {
    3802          42 :             if (constraint.iField ==
    3803          34 :                     m_poFeatureDefn->GetFieldCount() + SPF_FID &&
    3804          16 :                 m_osFIDColumn.empty())
    3805             :             {
    3806          16 :                 if (!ConstraintEvaluator(constraint,
    3807          16 :                                          static_cast<GIntBig>(m_nFeatureIdx)))
    3808             :                 {
    3809         203 :                     return true;
    3810             :                 }
    3811           4 :                 continue;
    3812             :             }
    3813             :             else
    3814             :             {
    3815             :                 // can happen if ignoring a field that is needed by the
    3816             :                 // attribute filter. ComputeConstraintsArrayIdx() will have
    3817             :                 // warned about that
    3818           2 :                 continue;
    3819             :             }
    3820             :         }
    3821             : 
    3822             :         const arrow::Array *array =
    3823         423 :             m_poBatchColumns[constraint.iArrayIdx].get();
    3824             : 
    3825         423 :         const bool bIsNull = array->IsNull(m_nIdxInBatch);
    3826         423 :         if (constraint.nOperation == SWQ_ISNULL)
    3827             :         {
    3828           7 :             if (bIsNull)
    3829             :             {
    3830           3 :                 continue;
    3831             :             }
    3832           4 :             return true;
    3833             :         }
    3834         416 :         else if (constraint.nOperation == SWQ_ISNOTNULL)
    3835             :         {
    3836          22 :             if (!bIsNull)
    3837             :             {
    3838          18 :                 continue;
    3839             :             }
    3840           4 :             return true;
    3841             :         }
    3842         394 :         else if (bIsNull)
    3843             :         {
    3844          65 :             return true;
    3845             :         }
    3846             : 
    3847         329 :         switch (array->type_id())
    3848             :         {
    3849           0 :             case arrow::Type::NA:
    3850           0 :                 break;
    3851             : 
    3852          92 :             case arrow::Type::BOOL:
    3853             :             {
    3854          92 :                 const auto castArray =
    3855             :                     static_cast<const arrow::BooleanArray *>(array);
    3856          92 :                 if (!ConstraintEvaluator(
    3857             :                         constraint,
    3858          92 :                         static_cast<int>(castArray->Value(m_nIdxInBatch))))
    3859             :                 {
    3860          42 :                     return true;
    3861             :                 }
    3862          50 :                 break;
    3863             :             }
    3864           7 :             case arrow::Type::UINT8:
    3865             :             {
    3866           7 :                 const auto castArray =
    3867             :                     static_cast<const arrow::UInt8Array *>(array);
    3868           7 :                 if (!ConstraintEvaluator(
    3869             :                         constraint,
    3870           7 :                         static_cast<int>(castArray->Value(m_nIdxInBatch))))
    3871             :                 {
    3872           3 :                     return true;
    3873             :                 }
    3874           4 :                 break;
    3875             :             }
    3876          30 :             case arrow::Type::INT8:
    3877             :             {
    3878          30 :                 const auto castArray =
    3879             :                     static_cast<const arrow::Int8Array *>(array);
    3880          30 :                 if (!ConstraintEvaluator(
    3881             :                         constraint,
    3882          30 :                         static_cast<int>(castArray->Value(m_nIdxInBatch))))
    3883             :                 {
    3884           2 :                     return true;
    3885             :                 }
    3886          28 :                 break;
    3887             :             }
    3888           4 :             case arrow::Type::UINT16:
    3889             :             {
    3890           4 :                 const auto castArray =
    3891             :                     static_cast<const arrow::UInt16Array *>(array);
    3892           4 :                 if (!ConstraintEvaluator(
    3893             :                         constraint,
    3894           4 :                         static_cast<int>(castArray->Value(m_nIdxInBatch))))
    3895             :                 {
    3896           2 :                     return true;
    3897             :                 }
    3898           2 :                 break;
    3899             :             }
    3900           0 :             case arrow::Type::INT16:
    3901             :             {
    3902           0 :                 const auto castArray =
    3903             :                     static_cast<const arrow::Int16Array *>(array);
    3904           0 :                 if (!ConstraintEvaluator(
    3905             :                         constraint,
    3906           0 :                         static_cast<int>(castArray->Value(m_nIdxInBatch))))
    3907             :                 {
    3908           0 :                     return true;
    3909             :                 }
    3910           0 :                 break;
    3911             :             }
    3912           0 :             case arrow::Type::UINT32:
    3913             :             {
    3914           0 :                 const auto castArray =
    3915             :                     static_cast<const arrow::UInt32Array *>(array);
    3916           0 :                 if (!ConstraintEvaluator(
    3917             :                         constraint,
    3918           0 :                         static_cast<GIntBig>(castArray->Value(m_nIdxInBatch))))
    3919             :                 {
    3920           0 :                     return true;
    3921             :                 }
    3922           0 :                 break;
    3923             :             }
    3924          71 :             case arrow::Type::INT32:
    3925             :             {
    3926          71 :                 const auto castArray =
    3927             :                     static_cast<const arrow::Int32Array *>(array);
    3928          71 :                 if (!ConstraintEvaluator(constraint,
    3929          71 :                                          castArray->Value(m_nIdxInBatch)))
    3930             :                 {
    3931          28 :                     return true;
    3932             :                 }
    3933          43 :                 break;
    3934             :             }
    3935           8 :             case arrow::Type::UINT64:
    3936             :             {
    3937           8 :                 const auto castArray =
    3938             :                     static_cast<const arrow::UInt64Array *>(array);
    3939           8 :                 if (!ConstraintEvaluator(
    3940             :                         constraint,
    3941           8 :                         static_cast<double>(castArray->Value(m_nIdxInBatch))))
    3942             :                 {
    3943           6 :                     return true;
    3944             :                 }
    3945           2 :                 break;
    3946             :             }
    3947          27 :             case arrow::Type::INT64:
    3948             :             {
    3949          27 :                 const auto castArray =
    3950             :                     static_cast<const arrow::Int64Array *>(array);
    3951          27 :                 if (!ConstraintEvaluator(
    3952             :                         constraint,
    3953          27 :                         static_cast<GIntBig>(castArray->Value(m_nIdxInBatch))))
    3954             :                 {
    3955          10 :                     return true;
    3956             :                 }
    3957          17 :                 break;
    3958             :             }
    3959           0 :             case arrow::Type::HALF_FLOAT:
    3960             :             {
    3961           0 :                 const auto castArray =
    3962             :                     static_cast<const arrow::HalfFloatArray *>(array);
    3963           0 :                 const uint16_t nFloat16 = castArray->Value(m_nIdxInBatch);
    3964           0 :                 uint32_t nFloat32 = CPLHalfToFloat(nFloat16);
    3965             :                 float f;
    3966           0 :                 memcpy(&f, &nFloat32, sizeof(nFloat32));
    3967           0 :                 if (!ConstraintEvaluator(constraint, static_cast<double>(f)))
    3968             :                 {
    3969           0 :                     return true;
    3970             :                 }
    3971           0 :                 break;
    3972             :             }
    3973           4 :             case arrow::Type::FLOAT:
    3974             :             {
    3975           4 :                 const auto castArray =
    3976             :                     static_cast<const arrow::FloatArray *>(array);
    3977           4 :                 if (!ConstraintEvaluator(
    3978             :                         constraint,
    3979           4 :                         static_cast<double>(castArray->Value(m_nIdxInBatch))))
    3980             :                 {
    3981           2 :                     return true;
    3982             :                 }
    3983           2 :                 break;
    3984             :             }
    3985          22 :             case arrow::Type::DOUBLE:
    3986             :             {
    3987          22 :                 const auto castArray =
    3988             :                     static_cast<const arrow::DoubleArray *>(array);
    3989          22 :                 if (!ConstraintEvaluator(constraint,
    3990          22 :                                          castArray->Value(m_nIdxInBatch)))
    3991             :                 {
    3992           6 :                     return true;
    3993             :                 }
    3994          16 :                 break;
    3995             :             }
    3996          45 :             case arrow::Type::STRING:
    3997             :             {
    3998          45 :                 const auto castArray =
    3999             :                     static_cast<const arrow::StringArray *>(array);
    4000          45 :                 int out_length = 0;
    4001             :                 const uint8_t *data =
    4002          45 :                     castArray->GetValue(m_nIdxInBatch, &out_length);
    4003          45 :                 if (!ConstraintEvaluator(
    4004             :                         constraint,
    4005          90 :                         std::string_view(reinterpret_cast<const char *>(data),
    4006             :                                          out_length)))
    4007             :                 {
    4008          13 :                     return true;
    4009             :                 }
    4010          32 :                 break;
    4011             :             }
    4012             : 
    4013             : #if ARROW_VERSION_MAJOR >= 18
    4014           0 :             case arrow::Type::DECIMAL32:
    4015             :             {
    4016           0 :                 const auto castArray =
    4017             :                     static_cast<const arrow::Decimal32Array *>(array);
    4018           0 :                 if (!ConstraintEvaluator(
    4019             :                         constraint,
    4020           0 :                         CPLAtof(castArray->FormatValue(m_nIdxInBatch).c_str())))
    4021             :                 {
    4022           0 :                     return true;
    4023             :                 }
    4024           0 :                 break;
    4025             :             }
    4026             : 
    4027           0 :             case arrow::Type::DECIMAL64:
    4028             :             {
    4029           0 :                 const auto castArray =
    4030             :                     static_cast<const arrow::Decimal64Array *>(array);
    4031           0 :                 if (!ConstraintEvaluator(
    4032             :                         constraint,
    4033           0 :                         CPLAtof(castArray->FormatValue(m_nIdxInBatch).c_str())))
    4034             :                 {
    4035           0 :                     return true;
    4036             :                 }
    4037           0 :                 break;
    4038             :             }
    4039             : #endif
    4040             : 
    4041           4 :             case arrow::Type::DECIMAL128:
    4042             :             {
    4043           4 :                 const auto castArray =
    4044             :                     static_cast<const arrow::Decimal128Array *>(array);
    4045           4 :                 if (!ConstraintEvaluator(
    4046             :                         constraint,
    4047           8 :                         CPLAtof(castArray->FormatValue(m_nIdxInBatch).c_str())))
    4048             :                 {
    4049           2 :                     return true;
    4050             :                 }
    4051           2 :                 break;
    4052             :             }
    4053             : 
    4054           4 :             case arrow::Type::DECIMAL256:
    4055             :             {
    4056           4 :                 const auto castArray =
    4057             :                     static_cast<const arrow::Decimal256Array *>(array);
    4058           4 :                 if (!ConstraintEvaluator(
    4059             :                         constraint,
    4060           8 :                         CPLAtof(castArray->FormatValue(m_nIdxInBatch).c_str())))
    4061             :                 {
    4062           2 :                     return true;
    4063             :                 }
    4064           2 :                 break;
    4065             :             }
    4066             : 
    4067          11 :             default:
    4068          11 :                 break;
    4069             :         }
    4070             :     }
    4071         238 :     return false;
    4072             : }
    4073             : 
    4074             : /************************************************************************/
    4075             : /*                           SetBatch()                                 */
    4076             : /************************************************************************/
    4077             : 
    4078             : inline void
    4079        3668 : OGRArrowLayer::SetBatch(const std::shared_ptr<arrow::RecordBatch> &poBatch)
    4080             : {
    4081        3668 :     m_poBatch = poBatch;
    4082        3668 :     m_poBatchColumns.clear();
    4083        3668 :     m_poArrayWKB = nullptr;
    4084        3668 :     m_poArrayWKBLarge = nullptr;
    4085        3668 :     m_poArrayBBOX = nullptr;
    4086        3668 :     m_poArrayXMinDouble = nullptr;
    4087        3668 :     m_poArrayYMinDouble = nullptr;
    4088        3668 :     m_poArrayXMaxDouble = nullptr;
    4089        3668 :     m_poArrayYMaxDouble = nullptr;
    4090        3668 :     m_poArrayXMinFloat = nullptr;
    4091        3668 :     m_poArrayYMinFloat = nullptr;
    4092        3668 :     m_poArrayXMaxFloat = nullptr;
    4093        3668 :     m_poArrayYMaxFloat = nullptr;
    4094             : 
    4095        3668 :     if (m_poBatch)
    4096             :     {
    4097        2639 :         m_poBatchColumns = m_poBatch->columns();
    4098        2639 :         SanityCheckOfSetBatch();
    4099             :     }
    4100             : 
    4101        3668 :     if (m_poBatch && m_poFilterGeom && !m_bBaseArrowIgnoreSpatialFilterRect)
    4102             :     {
    4103             :         int iCol;
    4104         563 :         if (m_bIgnoredFields)
    4105             :         {
    4106         148 :             iCol = m_anMapGeomFieldIndexToArrayIndex[m_iGeomFieldFilter];
    4107             :         }
    4108             :         else
    4109             :         {
    4110         415 :             iCol = m_anMapGeomFieldIndexToArrowColumn[m_iGeomFieldFilter];
    4111             :         }
    4112        1126 :         if (iCol >= 0 &&
    4113         563 :             m_aeGeomEncoding[m_iGeomFieldFilter] == OGRArrowGeomEncoding::WKB)
    4114             :         {
    4115             :             const arrow::Array *poArrayWKB =
    4116         110 :                 GetStorageArray(m_poBatchColumns[iCol].get());
    4117         110 :             if (poArrayWKB->type_id() == arrow::Type::BINARY)
    4118         110 :                 m_poArrayWKB =
    4119             :                     static_cast<const arrow::BinaryArray *>(poArrayWKB);
    4120             :             else
    4121             :             {
    4122           0 :                 CPLAssert(poArrayWKB->type_id() == arrow::Type::LARGE_BINARY);
    4123           0 :                 m_poArrayWKBLarge =
    4124             :                     static_cast<const arrow::LargeBinaryArray *>(poArrayWKB);
    4125             :             }
    4126             :         }
    4127             : 
    4128        1126 :         if (iCol >= 0 &&
    4129         563 :             CPLTestBool(CPLGetConfigOption(
    4130        1126 :                 ("OGR_" + GetDriverUCName() + "_USE_BBOX").c_str(), "YES")))
    4131             :         {
    4132             :             const auto oIter =
    4133         563 :                 m_oMapGeomFieldIndexToGeomColBBOX.find(m_iGeomFieldFilter);
    4134         563 :             if (oIter != m_oMapGeomFieldIndexToGeomColBBOX.end())
    4135             :             {
    4136         260 :                 const int idx = m_bIgnoredFields ? oIter->second.iArrayIdx
    4137         102 :                                                  : oIter->second.iArrowCol;
    4138         158 :                 if (idx >= 0)
    4139             :                 {
    4140         130 :                     CPLAssert(static_cast<size_t>(idx) <
    4141             :                               m_poBatchColumns.size());
    4142         130 :                     m_poArrayBBOX = m_poBatchColumns[idx].get();
    4143         130 :                     CPLAssert(m_poArrayBBOX->type_id() == arrow::Type::STRUCT);
    4144         130 :                     const auto castArray =
    4145             :                         static_cast<const arrow::StructArray *>(m_poArrayBBOX);
    4146         130 :                     const auto &subArrays = castArray->fields();
    4147         130 :                     CPLAssert(
    4148             :                         static_cast<size_t>(oIter->second.iArrowSubfieldXMin) <
    4149             :                         subArrays.size());
    4150             :                     const auto xminArray =
    4151         130 :                         subArrays[oIter->second.iArrowSubfieldXMin].get();
    4152         130 :                     CPLAssert(
    4153             :                         static_cast<size_t>(oIter->second.iArrowSubfieldYMin) <
    4154             :                         subArrays.size());
    4155             :                     const auto yminArray =
    4156         130 :                         subArrays[oIter->second.iArrowSubfieldYMin].get();
    4157         130 :                     CPLAssert(
    4158             :                         static_cast<size_t>(oIter->second.iArrowSubfieldXMax) <
    4159             :                         subArrays.size());
    4160             :                     const auto xmaxArray =
    4161         130 :                         subArrays[oIter->second.iArrowSubfieldXMax].get();
    4162         130 :                     CPLAssert(
    4163             :                         static_cast<size_t>(oIter->second.iArrowSubfieldYMax) <
    4164             :                         subArrays.size());
    4165             :                     const auto ymaxArray =
    4166         130 :                         subArrays[oIter->second.iArrowSubfieldYMax].get();
    4167         130 :                     if (oIter->second.bIsFloat)
    4168             :                     {
    4169         129 :                         CPLAssert(xminArray->type_id() == arrow::Type::FLOAT);
    4170         129 :                         m_poArrayXMinFloat =
    4171             :                             static_cast<const arrow::FloatArray *>(xminArray);
    4172         129 :                         CPLAssert(yminArray->type_id() == arrow::Type::FLOAT);
    4173         129 :                         m_poArrayYMinFloat =
    4174             :                             static_cast<const arrow::FloatArray *>(yminArray);
    4175         129 :                         CPLAssert(xmaxArray->type_id() == arrow::Type::FLOAT);
    4176         129 :                         m_poArrayXMaxFloat =
    4177             :                             static_cast<const arrow::FloatArray *>(xmaxArray);
    4178         129 :                         CPLAssert(ymaxArray->type_id() == arrow::Type::FLOAT);
    4179         129 :                         m_poArrayYMaxFloat =
    4180             :                             static_cast<const arrow::FloatArray *>(ymaxArray);
    4181             :                     }
    4182             :                     else
    4183             :                     {
    4184           1 :                         CPLAssert(xminArray->type_id() == arrow::Type::DOUBLE);
    4185           1 :                         m_poArrayXMinDouble =
    4186             :                             static_cast<const arrow::DoubleArray *>(xminArray);
    4187           1 :                         CPLAssert(yminArray->type_id() == arrow::Type::DOUBLE);
    4188           1 :                         m_poArrayYMinDouble =
    4189             :                             static_cast<const arrow::DoubleArray *>(yminArray);
    4190           1 :                         CPLAssert(xmaxArray->type_id() == arrow::Type::DOUBLE);
    4191           1 :                         m_poArrayXMaxDouble =
    4192             :                             static_cast<const arrow::DoubleArray *>(xmaxArray);
    4193           1 :                         CPLAssert(ymaxArray->type_id() == arrow::Type::DOUBLE);
    4194           1 :                         m_poArrayYMaxDouble =
    4195             :                             static_cast<const arrow::DoubleArray *>(ymaxArray);
    4196             :                     }
    4197             :                 }
    4198             :             }
    4199             :         }
    4200             :     }
    4201        3668 : }
    4202             : 
    4203             : /************************************************************************/
    4204             : /*                      SanityCheckOfSetBatch()                         */
    4205             : /************************************************************************/
    4206             : 
    4207        2639 : inline void OGRArrowLayer::SanityCheckOfSetBatch() const
    4208             : {
    4209             : #ifdef DEBUG
    4210        2639 :     CPLAssert(m_poBatch);
    4211             : 
    4212        2639 :     const auto &poColumns = m_poBatch->columns();
    4213             : 
    4214             :     // Sanity checks
    4215        2639 :     CPLAssert(m_poBatch->num_columns() == (m_bIgnoredFields
    4216             :                                                ? m_nExpectedBatchColumns
    4217             :                                                : m_poSchema->num_fields()));
    4218        2639 :     const auto &fields = m_poSchema->fields();
    4219             : 
    4220       94829 :     for (int i = 0; i < m_poFeatureDefn->GetFieldCount(); ++i)
    4221             :     {
    4222             :         int iCol;
    4223       92190 :         if (m_bIgnoredFields)
    4224             :         {
    4225       15274 :             iCol = m_anMapFieldIndexToArrayIndex[i];
    4226       15274 :             if (iCol < 0)
    4227        3917 :                 continue;
    4228             :         }
    4229             :         else
    4230             :         {
    4231       76916 :             iCol = m_anMapFieldIndexToArrowColumn[i][0];
    4232             :         }
    4233       88273 :         CPL_IGNORE_RET_VAL(iCol);  // to make cppcheck happy
    4234             : 
    4235       88273 :         CPLAssert(iCol < static_cast<int>(poColumns.size()));
    4236       88273 :         CPLAssert(fields[m_anMapFieldIndexToArrowColumn[i][0]]->type()->id() ==
    4237             :                   poColumns[iCol]->type_id());
    4238             :     }
    4239             : 
    4240        5173 :     for (int i = 0; i < m_poFeatureDefn->GetGeomFieldCount(); ++i)
    4241             :     {
    4242             :         int iCol;
    4243        2534 :         if (m_bIgnoredFields)
    4244             :         {
    4245         513 :             iCol = m_anMapGeomFieldIndexToArrayIndex[i];
    4246         513 :             if (iCol < 0)
    4247          30 :                 continue;
    4248             :         }
    4249             :         else
    4250             :         {
    4251        2021 :             iCol = m_anMapGeomFieldIndexToArrowColumn[i];
    4252             :         }
    4253        2504 :         CPL_IGNORE_RET_VAL(iCol);  // to make cppcheck happy
    4254             : 
    4255        2504 :         CPLAssert(iCol < static_cast<int>(poColumns.size()));
    4256        2504 :         CPLAssert(fields[m_anMapGeomFieldIndexToArrowColumn[i]]->type()->id() ==
    4257             :                   poColumns[iCol]->type_id());
    4258             :     }
    4259             : #else
    4260             :     CPL_IGNORE_RET_VAL(m_nExpectedBatchColumns);
    4261             : #endif
    4262        2639 : }
    4263             : 
    4264             : /************************************************************************/
    4265             : /*                        GetNextRawFeature()                           */
    4266             : /************************************************************************/
    4267             : 
    4268       11791 : inline OGRFeature *OGRArrowLayer::GetNextRawFeature()
    4269             : {
    4270       11791 :     if (m_bEOF || !m_bSpatialFilterIntersectsLayerExtent)
    4271         391 :         return nullptr;
    4272             : 
    4273       11400 :     if (m_poBatch == nullptr || m_nIdxInBatch == m_poBatch->num_rows())
    4274             :     {
    4275        3440 :         m_bEOF = !ReadNextBatch();
    4276        3440 :         if (m_bEOF)
    4277        1388 :             return nullptr;
    4278             :     }
    4279             : 
    4280             :     // Evaluate spatial filter by computing the bounding box of each geometry
    4281             :     // but without creating a OGRGeometry
    4282       10012 :     if (m_poFilterGeom && !m_bBaseArrowIgnoreSpatialFilterRect)
    4283             :     {
    4284             :         int iCol;
    4285         973 :         if (m_bIgnoredFields)
    4286             :         {
    4287         206 :             iCol = m_anMapGeomFieldIndexToArrayIndex[m_iGeomFieldFilter];
    4288             :         }
    4289             :         else
    4290             :         {
    4291         767 :             iCol = m_anMapGeomFieldIndexToArrowColumn[m_iGeomFieldFilter];
    4292             :         }
    4293             : 
    4294         973 :         if (iCol >= 0 && (m_poArrayXMinFloat || m_poArrayXMinDouble))
    4295             :         {
    4296         269 :             OGREnvelope sEnvelopeSkipToNextFeatureDueToBBOX;
    4297             :             const auto IntersectsBBOX =
    4298        2101 :                 [this, &sEnvelopeSkipToNextFeatureDueToBBOX]()
    4299             :             {
    4300         599 :                 if (m_poArrayXMinFloat &&
    4301         299 :                     !m_poArrayXMinFloat->IsNull(m_nIdxInBatch))
    4302             :                 {
    4303         299 :                     sEnvelopeSkipToNextFeatureDueToBBOX.MinX =
    4304         299 :                         m_poArrayXMinFloat->Value(m_nIdxInBatch);
    4305         299 :                     sEnvelopeSkipToNextFeatureDueToBBOX.MinY =
    4306         299 :                         m_poArrayYMinFloat->Value(m_nIdxInBatch);
    4307         299 :                     sEnvelopeSkipToNextFeatureDueToBBOX.MaxX =
    4308         299 :                         m_poArrayXMaxFloat->Value(m_nIdxInBatch);
    4309         299 :                     sEnvelopeSkipToNextFeatureDueToBBOX.MaxY =
    4310         299 :                         m_poArrayYMaxFloat->Value(m_nIdxInBatch);
    4311         299 :                     if (m_sFilterEnvelope.Intersects(
    4312         299 :                             sEnvelopeSkipToNextFeatureDueToBBOX))
    4313             :                     {
    4314         261 :                         return true;
    4315             :                     }
    4316             :                 }
    4317           2 :                 else if (m_poArrayXMinDouble &&
    4318           1 :                          !m_poArrayXMinDouble->IsNull(m_nIdxInBatch))
    4319             :                 {
    4320           1 :                     sEnvelopeSkipToNextFeatureDueToBBOX.MinX =
    4321           1 :                         m_poArrayXMinDouble->Value(m_nIdxInBatch);
    4322           1 :                     sEnvelopeSkipToNextFeatureDueToBBOX.MinY =
    4323           1 :                         m_poArrayYMinDouble->Value(m_nIdxInBatch);
    4324           1 :                     sEnvelopeSkipToNextFeatureDueToBBOX.MaxX =
    4325           1 :                         m_poArrayXMaxDouble->Value(m_nIdxInBatch);
    4326           1 :                     sEnvelopeSkipToNextFeatureDueToBBOX.MaxY =
    4327           1 :                         m_poArrayYMaxDouble->Value(m_nIdxInBatch);
    4328           1 :                     if (m_sFilterEnvelope.Intersects(
    4329           1 :                             sEnvelopeSkipToNextFeatureDueToBBOX))
    4330             :                     {
    4331           1 :                         return true;
    4332             :                     }
    4333             :                 }
    4334          38 :                 return false;
    4335         269 :             };
    4336             : 
    4337             :             while (true)
    4338             :             {
    4339         790 :                 if (!m_poArrayBBOX->IsNull(m_nIdxInBatch) && IntersectsBBOX() &&
    4340         262 :                     (m_asAttributeFilterConstraints.empty() ||
    4341           2 :                      !SkipToNextFeatureDueToAttributeFilter()))
    4342             :                 {
    4343         261 :                     break;
    4344             :                 }
    4345             : 
    4346         265 :                 IncrFeatureIdx();
    4347         265 :                 m_nIdxInBatch++;
    4348         265 :                 if (m_nIdxInBatch == m_poBatch->num_rows())
    4349             :                 {
    4350          17 :                     m_bEOF = !ReadNextBatch();
    4351          17 :                     if (m_bEOF)
    4352           8 :                         return nullptr;
    4353             :                 }
    4354         261 :             }
    4355             :         }
    4356         704 :         else if (iCol >= 0 && m_aeGeomEncoding[m_iGeomFieldFilter] ==
    4357             :                                   OGRArrowGeomEncoding::WKB)
    4358             :         {
    4359         131 :             CPLAssert(m_poArrayWKB || m_poArrayWKBLarge);
    4360         131 :             OGREnvelope sEnvelope;
    4361             : 
    4362             :             while (true)
    4363             :             {
    4364         220 :                 bool bMatchBBOX = false;
    4365         417 :                 if ((m_poArrayWKB && m_poArrayWKB->IsNull(m_nIdxInBatch)) ||
    4366         197 :                     (m_poArrayWKBLarge &&
    4367           0 :                      m_poArrayWKBLarge->IsNull(m_nIdxInBatch)))
    4368             :                 {
    4369             :                     // nothing to do
    4370             :                 }
    4371             :                 else
    4372             :                 {
    4373         197 :                     if (m_poArrayWKB)
    4374             :                     {
    4375         197 :                         int out_length = 0;
    4376             :                         const uint8_t *data =
    4377         197 :                             m_poArrayWKB->GetValue(m_nIdxInBatch, &out_length);
    4378         394 :                         if (OGRWKBGetBoundingBox(data, out_length, sEnvelope) &&
    4379         197 :                             m_sFilterEnvelope.Intersects(sEnvelope))
    4380             :                         {
    4381         115 :                             bMatchBBOX = true;
    4382             :                         }
    4383             :                     }
    4384             :                     else
    4385             :                     {
    4386           0 :                         CPLAssert(m_poArrayWKBLarge);
    4387           0 :                         int64_t out_length64 = 0;
    4388           0 :                         const uint8_t *data = m_poArrayWKBLarge->GetValue(
    4389             :                             m_nIdxInBatch, &out_length64);
    4390           0 :                         if (out_length64 < INT_MAX &&
    4391           0 :                             OGRWKBGetBoundingBox(data,
    4392             :                                                  static_cast<int>(out_length64),
    4393           0 :                                                  sEnvelope) &&
    4394           0 :                             m_sFilterEnvelope.Intersects(sEnvelope))
    4395             :                         {
    4396           0 :                             bMatchBBOX = true;
    4397             :                         }
    4398             :                     }
    4399             :                 }
    4400         221 :                 if (bMatchBBOX && (m_asAttributeFilterConstraints.empty() ||
    4401           1 :                                    !SkipToNextFeatureDueToAttributeFilter()))
    4402             :                 {
    4403         115 :                     break;
    4404             :                 }
    4405             : 
    4406         105 :                 IncrFeatureIdx();
    4407         105 :                 m_nIdxInBatch++;
    4408         105 :                 if (m_nIdxInBatch == m_poBatch->num_rows())
    4409             :                 {
    4410          25 :                     m_bEOF = !ReadNextBatch();
    4411          25 :                     if (m_bEOF)
    4412          16 :                         return nullptr;
    4413             :                 }
    4414          89 :             }
    4415             :         }
    4416        1146 :         else if (iCol >= 0 &&
    4417         573 :                  m_aeGeomEncoding[m_iGeomFieldFilter] ==
    4418             :                      OGRArrowGeomEncoding::GEOARROW_FSL_MULTIPOLYGON)
    4419             :         {
    4420             :             const auto poGeomFieldDefn =
    4421          10 :                 m_poFeatureDefn->GetGeomFieldDefn(m_iGeomFieldFilter);
    4422          10 :             const auto eGeomType = poGeomFieldDefn->GetType();
    4423          10 :             const bool bHasZ = CPL_TO_BOOL(OGR_GT_HasZ(eGeomType));
    4424          10 :             const bool bHasM = CPL_TO_BOOL(OGR_GT_HasM(eGeomType));
    4425          10 :             const int nDim = 2 + (bHasZ ? 1 : 0) + (bHasM ? 1 : 0);
    4426             : 
    4427             :             bool bReturnFeature;
    4428           0 :             do
    4429             :             {
    4430          10 :                 bReturnFeature = false;
    4431          10 :                 auto array = GetStorageArray(m_poBatchColumns[iCol].get());
    4432          10 :                 CPLAssert(array->type_id() == arrow::Type::LIST);
    4433          10 :                 auto listOfPartsArray =
    4434             :                     static_cast<const arrow::ListArray *>(array);
    4435          10 :                 CPLAssert(listOfPartsArray->values()->type_id() ==
    4436             :                           arrow::Type::LIST);
    4437             :                 auto listOfPartsValues =
    4438             :                     std::static_pointer_cast<arrow::ListArray>(
    4439          10 :                         listOfPartsArray->values());
    4440          10 :                 CPLAssert(listOfPartsValues->values()->type_id() ==
    4441             :                           arrow::Type::LIST);
    4442             :                 auto listOfRingsValues =
    4443             :                     std::static_pointer_cast<arrow::ListArray>(
    4444          10 :                         listOfPartsValues->values());
    4445          10 :                 CPLAssert(listOfRingsValues->values()->type_id() ==
    4446             :                           arrow::Type::FIXED_SIZE_LIST);
    4447             :                 auto listOfPointsValues =
    4448             :                     std::static_pointer_cast<arrow::FixedSizeListArray>(
    4449          10 :                         listOfRingsValues->values());
    4450          10 :                 CPLAssert(listOfPointsValues->values()->type_id() ==
    4451             :                           arrow::Type::DOUBLE);
    4452             :                 auto pointValues = std::static_pointer_cast<arrow::DoubleArray>(
    4453          10 :                     listOfPointsValues->values());
    4454             : 
    4455             :                 while (true)
    4456             :                 {
    4457          18 :                     bool bMatchBBOX = false;
    4458          18 :                     if (!listOfPartsArray->IsNull(m_nIdxInBatch))
    4459             :                     {
    4460          14 :                         OGREnvelope sEnvelope;
    4461             :                         const auto nParts =
    4462          14 :                             listOfPartsArray->value_length(m_nIdxInBatch);
    4463             :                         const auto nPartOffset =
    4464          14 :                             listOfPartsArray->value_offset(m_nIdxInBatch);
    4465          28 :                         for (auto j = decltype(nParts){0}; j < nParts; j++)
    4466             :                         {
    4467          14 :                             const auto nRings = listOfPartsValues->value_length(
    4468          14 :                                 nPartOffset + j);
    4469             :                             const auto nRingOffset =
    4470          14 :                                 listOfPartsValues->value_offset(nPartOffset +
    4471             :                                                                 j);
    4472          14 :                             if (nRings >= 1)
    4473             :                             {
    4474             :                                 const auto nPoints =
    4475          14 :                                     listOfRingsValues->value_length(
    4476             :                                         nRingOffset);
    4477             :                                 const auto nPointOffset =
    4478          14 :                                     listOfRingsValues->value_offset(
    4479             :                                         nRingOffset) *
    4480          14 :                                     nDim;
    4481             :                                 const double *padfRawValue =
    4482          14 :                                     pointValues->raw_values() + nPointOffset;
    4483          73 :                                 for (auto l = decltype(nPoints){0}; l < nPoints;
    4484             :                                      ++l)
    4485             :                                 {
    4486          59 :                                     sEnvelope.Merge(padfRawValue[nDim * l],
    4487          59 :                                                     padfRawValue[nDim * l + 1]);
    4488             :                                 }
    4489             :                                 // for bounding box, only the first ring matters
    4490             :                             }
    4491             :                         }
    4492             : 
    4493          24 :                         if (nParts != 0 &&
    4494          10 :                             m_sFilterEnvelope.Intersects(sEnvelope))
    4495             :                         {
    4496          10 :                             bMatchBBOX = true;
    4497             :                         }
    4498             :                     }
    4499          28 :                     if (bMatchBBOX &&
    4500          10 :                         (m_asAttributeFilterConstraints.empty() ||
    4501           0 :                          !SkipToNextFeatureDueToAttributeFilter()))
    4502             :                     {
    4503          10 :                         bReturnFeature = true;
    4504          10 :                         break;
    4505             :                     }
    4506             : 
    4507           8 :                     IncrFeatureIdx();
    4508           8 :                     m_nIdxInBatch++;
    4509           8 :                     if (m_nIdxInBatch == m_poBatch->num_rows())
    4510             :                     {
    4511           0 :                         m_bEOF = !ReadNextBatch();
    4512           0 :                         if (m_bEOF)
    4513           0 :                             return nullptr;
    4514           0 :                         break;
    4515             :                     }
    4516           8 :                 }
    4517          10 :             } while (!bReturnFeature);
    4518             :         }
    4519         563 :         else if (iCol >= 0 && m_aeGeomEncoding[m_iGeomFieldFilter] ==
    4520             :                                   OGRArrowGeomEncoding::GEOARROW_STRUCT_POINT)
    4521             :         {
    4522             :             bool bReturnFeature;
    4523          10 :             do
    4524             :             {
    4525          44 :                 bReturnFeature = false;
    4526          44 :                 auto array = GetStorageArray(m_poBatchColumns[iCol].get());
    4527          44 :                 CPLAssert(array->type_id() == arrow::Type::STRUCT);
    4528          44 :                 auto pointValues =
    4529             :                     static_cast<const arrow::StructArray *>(array);
    4530          44 :                 const auto &fields = pointValues->fields();
    4531          44 :                 const auto &fieldX = fields[0];
    4532          44 :                 CPLAssert(fieldX->type_id() == arrow::Type::DOUBLE);
    4533             :                 const auto fieldXDouble =
    4534          44 :                     static_cast<arrow::DoubleArray *>(fieldX.get());
    4535          44 :                 const auto &fieldY = fields[1];
    4536          44 :                 CPLAssert(fieldY->type_id() == arrow::Type::DOUBLE);
    4537             :                 const auto fieldYDouble =
    4538          44 :                     static_cast<arrow::DoubleArray *>(fieldY.get());
    4539             : 
    4540             :                 while (true)
    4541             :                 {
    4542          98 :                     bool bMatchBBOX = false;
    4543          98 :                     if (!array->IsNull(m_nIdxInBatch))
    4544             :                     {
    4545          76 :                         const double dfX = fieldXDouble->Value(m_nIdxInBatch);
    4546          76 :                         const double dfY = fieldYDouble->Value(m_nIdxInBatch);
    4547          76 :                         if (dfX >= m_sFilterEnvelope.MinX &&
    4548          46 :                             dfY >= m_sFilterEnvelope.MinY &&
    4549          46 :                             dfX <= m_sFilterEnvelope.MaxX &&
    4550          30 :                             dfY <= m_sFilterEnvelope.MaxY)
    4551             :                         {
    4552          30 :                             bMatchBBOX = true;
    4553             :                         }
    4554             :                     }
    4555         128 :                     if (bMatchBBOX &&
    4556          30 :                         (m_asAttributeFilterConstraints.empty() ||
    4557           0 :                          !SkipToNextFeatureDueToAttributeFilter()))
    4558             :                     {
    4559          30 :                         bReturnFeature = true;
    4560          30 :                         break;
    4561             :                     }
    4562             : 
    4563          68 :                     IncrFeatureIdx();
    4564          68 :                     m_nIdxInBatch++;
    4565          68 :                     if (m_nIdxInBatch == m_poBatch->num_rows())
    4566             :                     {
    4567          14 :                         m_bEOF = !ReadNextBatch();
    4568          14 :                         if (m_bEOF)
    4569           4 :                             return nullptr;
    4570          10 :                         break;
    4571             :                     }
    4572          54 :                 }
    4573          40 :             } while (!bReturnFeature);
    4574             :         }
    4575        1058 :         else if (iCol >= 0 &&
    4576         529 :                  m_aeGeomEncoding[m_iGeomFieldFilter] ==
    4577             :                      OGRArrowGeomEncoding::GEOARROW_STRUCT_LINESTRING)
    4578             :         {
    4579             :             bool bReturnFeature;
    4580           0 :             do
    4581             :             {
    4582          64 :                 bReturnFeature = false;
    4583          64 :                 auto array = GetStorageArray(m_poBatchColumns[iCol].get());
    4584          64 :                 CPLAssert(array->type_id() == arrow::Type::LIST);
    4585          64 :                 const auto listArray =
    4586             :                     static_cast<const arrow::ListArray *>(array);
    4587          64 :                 CPLAssert(listArray->values()->type_id() ==
    4588             :                           arrow::Type::STRUCT);
    4589             :                 auto pointValues = std::static_pointer_cast<arrow::StructArray>(
    4590          64 :                     listArray->values());
    4591          64 :                 const auto &fields = pointValues->fields();
    4592          64 :                 const auto &fieldX = fields[0];
    4593          64 :                 CPLAssert(fieldX->type_id() == arrow::Type::DOUBLE);
    4594             :                 const auto fieldXDouble =
    4595          64 :                     static_cast<arrow::DoubleArray *>(fieldX.get());
    4596          64 :                 const auto &fieldY = fields[1];
    4597          64 :                 CPLAssert(fieldY->type_id() == arrow::Type::DOUBLE);
    4598             :                 const auto fieldYDouble =
    4599          64 :                     static_cast<arrow::DoubleArray *>(fieldY.get());
    4600             : 
    4601             :                 while (true)
    4602             :                 {
    4603         160 :                     bool bMatchBBOX = false;
    4604         160 :                     if (!listArray->IsNull(m_nIdxInBatch))
    4605             :                     {
    4606         120 :                         OGREnvelope sEnvelope;
    4607             :                         const auto nPoints =
    4608         120 :                             listArray->value_length(m_nIdxInBatch);
    4609             :                         const auto nPointOffset =
    4610         120 :                             listArray->value_offset(m_nIdxInBatch);
    4611         120 :                         if (nPoints > 0)
    4612             :                         {
    4613             :                             const double *padfRawXValue =
    4614          80 :                                 fieldXDouble->raw_values() + nPointOffset;
    4615             :                             const double *padfRawYValue =
    4616          80 :                                 fieldYDouble->raw_values() + nPointOffset;
    4617         240 :                             for (auto l = decltype(nPoints){0}; l < nPoints;
    4618             :                                  ++l)
    4619             :                             {
    4620         160 :                                 sEnvelope.Merge(padfRawXValue[l],
    4621         160 :                                                 padfRawYValue[l]);
    4622             :                             }
    4623          80 :                             if (m_sFilterEnvelope.Intersects(sEnvelope))
    4624             :                             {
    4625          48 :                                 bMatchBBOX = true;
    4626             :                             }
    4627             :                         }
    4628             :                     }
    4629         208 :                     if (bMatchBBOX &&
    4630          48 :                         (m_asAttributeFilterConstraints.empty() ||
    4631           0 :                          !SkipToNextFeatureDueToAttributeFilter()))
    4632             :                     {
    4633          48 :                         bReturnFeature = true;
    4634          48 :                         break;
    4635             :                     }
    4636             : 
    4637         112 :                     IncrFeatureIdx();
    4638         112 :                     m_nIdxInBatch++;
    4639         112 :                     if (m_nIdxInBatch == m_poBatch->num_rows())
    4640             :                     {
    4641          16 :                         m_bEOF = !ReadNextBatch();
    4642          16 :                         if (m_bEOF)
    4643          16 :                             return nullptr;
    4644           0 :                         break;
    4645             :                     }
    4646          96 :                 }
    4647          48 :             } while (!bReturnFeature);
    4648             :         }
    4649         465 :         else if (iCol >= 0 && m_aeGeomEncoding[m_iGeomFieldFilter] ==
    4650             :                                   OGRArrowGeomEncoding::GEOARROW_STRUCT_POLYGON)
    4651             :         {
    4652             :             bool bReturnFeature;
    4653           0 :             do
    4654             :             {
    4655          96 :                 bReturnFeature = false;
    4656          96 :                 auto array = GetStorageArray(m_poBatchColumns[iCol].get());
    4657          96 :                 CPLAssert(array->type_id() == arrow::Type::LIST);
    4658          96 :                 const auto listOfRingsArray =
    4659             :                     static_cast<const arrow::ListArray *>(array);
    4660          96 :                 CPLAssert(listOfRingsArray->values()->type_id() ==
    4661             :                           arrow::Type::LIST);
    4662             :                 const auto listOfRingsValues =
    4663             :                     std::static_pointer_cast<arrow::ListArray>(
    4664          96 :                         listOfRingsArray->values());
    4665          96 :                 CPLAssert(listOfRingsValues->values()->type_id() ==
    4666             :                           arrow::Type::STRUCT);
    4667             :                 auto pointValues = std::static_pointer_cast<arrow::StructArray>(
    4668          96 :                     listOfRingsValues->values());
    4669          96 :                 const auto &fields = pointValues->fields();
    4670          96 :                 const auto &fieldX = fields[0];
    4671          96 :                 CPLAssert(fieldX->type_id() == arrow::Type::DOUBLE);
    4672             :                 const auto fieldXDouble =
    4673          96 :                     static_cast<arrow::DoubleArray *>(fieldX.get());
    4674          96 :                 const auto &fieldY = fields[1];
    4675          96 :                 CPLAssert(fieldY->type_id() == arrow::Type::DOUBLE);
    4676             :                 const auto fieldYDouble =
    4677          96 :                     static_cast<arrow::DoubleArray *>(fieldY.get());
    4678             : 
    4679             :                 while (true)
    4680             :                 {
    4681         240 :                     bool bMatchBBOX = false;
    4682         240 :                     if (!listOfRingsArray->IsNull(m_nIdxInBatch))
    4683             :                     {
    4684         184 :                         OGREnvelope sEnvelope;
    4685             :                         const auto nRings =
    4686         184 :                             listOfRingsArray->value_length(m_nIdxInBatch);
    4687             :                         const auto nRingOffset =
    4688         184 :                             listOfRingsArray->value_offset(m_nIdxInBatch);
    4689         184 :                         if (nRings >= 1)
    4690             :                         {
    4691             :                             const auto nPoints =
    4692         128 :                                 listOfRingsValues->value_length(nRingOffset);
    4693             :                             const auto nPointOffset =
    4694         128 :                                 listOfRingsValues->value_offset(nRingOffset);
    4695             :                             const double *padfRawXValue =
    4696         128 :                                 fieldXDouble->raw_values() + nPointOffset;
    4697             :                             const double *padfRawYValue =
    4698         128 :                                 fieldYDouble->raw_values() + nPointOffset;
    4699         688 :                             for (auto l = decltype(nPoints){0}; l < nPoints;
    4700             :                                  ++l)
    4701             :                             {
    4702         560 :                                 sEnvelope.Merge(padfRawXValue[l],
    4703         560 :                                                 padfRawYValue[l]);
    4704             :                             }
    4705             :                             // for bounding box, only the first ring matters
    4706             : 
    4707         128 :                             if (m_sFilterEnvelope.Intersects(sEnvelope))
    4708             :                             {
    4709          72 :                                 bMatchBBOX = true;
    4710             :                             }
    4711             :                         }
    4712             :                     }
    4713         312 :                     if (bMatchBBOX &&
    4714          72 :                         (m_asAttributeFilterConstraints.empty() ||
    4715           0 :                          !SkipToNextFeatureDueToAttributeFilter()))
    4716             :                     {
    4717          72 :                         bReturnFeature = true;
    4718          72 :                         break;
    4719             :                     }
    4720             : 
    4721         168 :                     IncrFeatureIdx();
    4722         168 :                     m_nIdxInBatch++;
    4723         168 :                     if (m_nIdxInBatch == m_poBatch->num_rows())
    4724             :                     {
    4725          24 :                         m_bEOF = !ReadNextBatch();
    4726          24 :                         if (m_bEOF)
    4727          24 :                             return nullptr;
    4728           0 :                         break;
    4729             :                     }
    4730         144 :                 }
    4731          72 :             } while (!bReturnFeature);
    4732             :         }
    4733         738 :         else if (iCol >= 0 &&
    4734         369 :                  m_aeGeomEncoding[m_iGeomFieldFilter] ==
    4735             :                      OGRArrowGeomEncoding::GEOARROW_STRUCT_MULTIPOINT)
    4736             :         {
    4737             :             bool bReturnFeature;
    4738           0 :             do
    4739             :             {
    4740          64 :                 bReturnFeature = false;
    4741          64 :                 auto array = GetStorageArray(m_poBatchColumns[iCol].get());
    4742          64 :                 CPLAssert(array->type_id() == arrow::Type::LIST);
    4743          64 :                 const auto listArray =
    4744             :                     static_cast<const arrow::ListArray *>(array);
    4745          64 :                 CPLAssert(listArray->values()->type_id() ==
    4746             :                           arrow::Type::STRUCT);
    4747             :                 auto pointValues = std::static_pointer_cast<arrow::StructArray>(
    4748          64 :                     listArray->values());
    4749          64 :                 const auto &fields = pointValues->fields();
    4750          64 :                 const auto &fieldX = fields[0];
    4751          64 :                 CPLAssert(fieldX->type_id() == arrow::Type::DOUBLE);
    4752             :                 const auto fieldXDouble =
    4753          64 :                     static_cast<arrow::DoubleArray *>(fieldX.get());
    4754          64 :                 const auto &fieldY = fields[1];
    4755          64 :                 CPLAssert(fieldY->type_id() == arrow::Type::DOUBLE);
    4756             :                 const auto fieldYDouble =
    4757          64 :                     static_cast<arrow::DoubleArray *>(fieldY.get());
    4758             : 
    4759             :                 while (true)
    4760             :                 {
    4761         160 :                     bool bMatchBBOX = false;
    4762         160 :                     if (!listArray->IsNull(m_nIdxInBatch))
    4763             :                     {
    4764             :                         const auto nPoints =
    4765         128 :                             listArray->value_length(m_nIdxInBatch);
    4766             :                         const auto nPointOffset =
    4767         128 :                             listArray->value_offset(m_nIdxInBatch);
    4768         128 :                         if (nPoints > 0)
    4769             :                         {
    4770             :                             const double *padfRawXValue =
    4771          96 :                                 fieldXDouble->raw_values() + nPointOffset;
    4772             :                             const double *padfRawYValue =
    4773          96 :                                 fieldYDouble->raw_values() + nPointOffset;
    4774         176 :                             for (auto l = decltype(nPoints){0}; l < nPoints;
    4775             :                                  ++l)
    4776             :                             {
    4777         128 :                                 if (padfRawXValue[l] >=
    4778         128 :                                         m_sFilterEnvelope.MinX &&
    4779         108 :                                     padfRawYValue[l] >=
    4780         108 :                                         m_sFilterEnvelope.MinY &&
    4781          88 :                                     padfRawXValue[l] <=
    4782          88 :                                         m_sFilterEnvelope.MaxX &&
    4783          68 :                                     padfRawYValue[l] <= m_sFilterEnvelope.MaxY)
    4784             :                                 {
    4785          48 :                                     bMatchBBOX = true;
    4786          48 :                                     break;
    4787             :                                 }
    4788             :                             }
    4789             :                         }
    4790             :                     }
    4791         208 :                     if (bMatchBBOX &&
    4792          48 :                         (m_asAttributeFilterConstraints.empty() ||
    4793           0 :                          !SkipToNextFeatureDueToAttributeFilter()))
    4794             :                     {
    4795          48 :                         bReturnFeature = true;
    4796          48 :                         break;
    4797             :                     }
    4798             : 
    4799         112 :                     IncrFeatureIdx();
    4800         112 :                     m_nIdxInBatch++;
    4801         112 :                     if (m_nIdxInBatch == m_poBatch->num_rows())
    4802             :                     {
    4803          16 :                         m_bEOF = !ReadNextBatch();
    4804          16 :                         if (m_bEOF)
    4805          16 :                             return nullptr;
    4806           0 :                         break;
    4807             :                     }
    4808          96 :                 }
    4809          48 :             } while (!bReturnFeature);
    4810             :         }
    4811         610 :         else if (iCol >= 0 &&
    4812         305 :                  m_aeGeomEncoding[m_iGeomFieldFilter] ==
    4813             :                      OGRArrowGeomEncoding::GEOARROW_STRUCT_MULTILINESTRING)
    4814             :         {
    4815             :             bool bReturnFeature;
    4816           0 :             do
    4817             :             {
    4818          88 :                 bReturnFeature = false;
    4819          88 :                 auto array = GetStorageArray(m_poBatchColumns[iCol].get());
    4820          88 :                 CPLAssert(array->type_id() == arrow::Type::LIST);
    4821          88 :                 auto listOfPartsArray =
    4822             :                     static_cast<const arrow::ListArray *>(array);
    4823          88 :                 CPLAssert(listOfPartsArray->values()->type_id() ==
    4824             :                           arrow::Type::LIST);
    4825             :                 auto listOfPartsValues =
    4826             :                     std::static_pointer_cast<arrow::ListArray>(
    4827          88 :                         listOfPartsArray->values());
    4828          88 :                 CPLAssert(listOfPartsValues->values()->type_id() ==
    4829             :                           arrow::Type::STRUCT);
    4830             :                 auto pointValues = std::static_pointer_cast<arrow::StructArray>(
    4831          88 :                     listOfPartsValues->values());
    4832          88 :                 const auto &fields = pointValues->fields();
    4833          88 :                 const auto &fieldX = fields[0];
    4834          88 :                 CPLAssert(fieldX->type_id() == arrow::Type::DOUBLE);
    4835             :                 const auto fieldXDouble =
    4836          88 :                     static_cast<arrow::DoubleArray *>(fieldX.get());
    4837          88 :                 const auto &fieldY = fields[1];
    4838          88 :                 CPLAssert(fieldY->type_id() == arrow::Type::DOUBLE);
    4839             :                 const auto fieldYDouble =
    4840          88 :                     static_cast<arrow::DoubleArray *>(fieldY.get());
    4841             : 
    4842             :                 while (true)
    4843             :                 {
    4844         200 :                     bool bMatchBBOX = false;
    4845         200 :                     if (!listOfPartsArray->IsNull(m_nIdxInBatch))
    4846             :                     {
    4847             :                         const auto nParts =
    4848         160 :                             listOfPartsArray->value_length(m_nIdxInBatch);
    4849             :                         const auto nPartOffset =
    4850         160 :                             listOfPartsArray->value_offset(m_nIdxInBatch);
    4851         312 :                         for (auto j = decltype(nParts){0};
    4852         312 :                              j < nParts && !bMatchBBOX; j++)
    4853             :                         {
    4854         152 :                             OGREnvelope sEnvelope;
    4855             :                             const auto nPoints =
    4856         152 :                                 listOfPartsValues->value_length(nPartOffset +
    4857             :                                                                 j);
    4858             :                             const auto nPointOffset =
    4859         152 :                                 listOfPartsValues->value_offset(nPartOffset +
    4860             :                                                                 j);
    4861             :                             const double *padfRawXValue =
    4862         152 :                                 fieldXDouble->raw_values() + nPointOffset;
    4863             :                             const double *padfRawYValue =
    4864         152 :                                 fieldYDouble->raw_values() + nPointOffset;
    4865         488 :                             for (auto l = decltype(nPoints){0}; l < nPoints;
    4866             :                                  ++l)
    4867             :                             {
    4868         336 :                                 sEnvelope.Merge(padfRawXValue[l],
    4869         336 :                                                 padfRawYValue[l]);
    4870             :                             }
    4871             : 
    4872         152 :                             if (m_sFilterEnvelope.Intersects(sEnvelope))
    4873             :                             {
    4874          72 :                                 bMatchBBOX = true;
    4875             :                             }
    4876             :                         }
    4877             :                     }
    4878         272 :                     if (bMatchBBOX &&
    4879          72 :                         (m_asAttributeFilterConstraints.empty() ||
    4880           0 :                          !SkipToNextFeatureDueToAttributeFilter()))
    4881             :                     {
    4882          72 :                         bReturnFeature = true;
    4883          72 :                         break;
    4884             :                     }
    4885             : 
    4886         128 :                     IncrFeatureIdx();
    4887         128 :                     m_nIdxInBatch++;
    4888         128 :                     if (m_nIdxInBatch == m_poBatch->num_rows())
    4889             :                     {
    4890          16 :                         m_bEOF = !ReadNextBatch();
    4891          16 :                         if (m_bEOF)
    4892          16 :                             return nullptr;
    4893           0 :                         break;
    4894             :                     }
    4895         112 :                 }
    4896          72 :             } while (!bReturnFeature);
    4897             :         }
    4898         434 :         else if (iCol >= 0 &&
    4899         217 :                  m_aeGeomEncoding[m_iGeomFieldFilter] ==
    4900             :                      OGRArrowGeomEncoding::GEOARROW_STRUCT_MULTIPOLYGON)
    4901             :         {
    4902             :             bool bReturnFeature;
    4903           0 :             do
    4904             :             {
    4905          96 :                 bReturnFeature = false;
    4906          96 :                 auto array = GetStorageArray(m_poBatchColumns[iCol].get());
    4907          96 :                 CPLAssert(array->type_id() == arrow::Type::LIST);
    4908          96 :                 auto listOfPartsArray =
    4909             :                     static_cast<const arrow::ListArray *>(array);
    4910          96 :                 CPLAssert(listOfPartsArray->values()->type_id() ==
    4911             :                           arrow::Type::LIST);
    4912             :                 auto listOfPartsValues =
    4913             :                     std::static_pointer_cast<arrow::ListArray>(
    4914          96 :                         listOfPartsArray->values());
    4915          96 :                 CPLAssert(listOfPartsValues->values()->type_id() ==
    4916             :                           arrow::Type::LIST);
    4917             :                 auto listOfRingsValues =
    4918             :                     std::static_pointer_cast<arrow::ListArray>(
    4919          96 :                         listOfPartsValues->values());
    4920          96 :                 CPLAssert(listOfRingsValues->values()->type_id() ==
    4921             :                           arrow::Type::STRUCT);
    4922             :                 auto pointValues = std::static_pointer_cast<arrow::StructArray>(
    4923          96 :                     listOfRingsValues->values());
    4924          96 :                 const auto &fields = pointValues->fields();
    4925          96 :                 const auto &fieldX = fields[0];
    4926          96 :                 CPLAssert(fieldX->type_id() == arrow::Type::DOUBLE);
    4927             :                 const auto fieldXDouble =
    4928          96 :                     static_cast<arrow::DoubleArray *>(fieldX.get());
    4929          96 :                 const auto &fieldY = fields[1];
    4930          96 :                 CPLAssert(fieldY->type_id() == arrow::Type::DOUBLE);
    4931             :                 const auto fieldYDouble =
    4932          96 :                     static_cast<arrow::DoubleArray *>(fieldY.get());
    4933             : 
    4934             :                 while (true)
    4935             :                 {
    4936         240 :                     bool bMatchBBOX = false;
    4937         240 :                     if (!listOfPartsArray->IsNull(m_nIdxInBatch))
    4938             :                     {
    4939             :                         const auto nParts =
    4940         188 :                             listOfPartsArray->value_length(m_nIdxInBatch);
    4941             :                         const auto nPartOffset =
    4942         188 :                             listOfPartsArray->value_offset(m_nIdxInBatch);
    4943         356 :                         for (auto j = decltype(nParts){0};
    4944         356 :                              j < nParts && !bMatchBBOX; j++)
    4945             :                         {
    4946         168 :                             OGREnvelope sEnvelope;
    4947         168 :                             const auto nRings = listOfPartsValues->value_length(
    4948         168 :                                 nPartOffset + j);
    4949             :                             const auto nRingOffset =
    4950         168 :                                 listOfPartsValues->value_offset(nPartOffset +
    4951             :                                                                 j);
    4952         168 :                             if (nRings >= 1)
    4953             :                             {
    4954             :                                 const auto nPoints =
    4955         168 :                                     listOfRingsValues->value_length(
    4956             :                                         nRingOffset);
    4957             :                                 const auto nPointOffset =
    4958         168 :                                     listOfRingsValues->value_offset(
    4959             :                                         nRingOffset);
    4960             :                                 const double *padfRawXValue =
    4961         168 :                                     fieldXDouble->raw_values() + nPointOffset;
    4962             :                                 const double *padfRawYValue =
    4963         168 :                                     fieldYDouble->raw_values() + nPointOffset;
    4964         888 :                                 for (auto l = decltype(nPoints){0}; l < nPoints;
    4965             :                                      ++l)
    4966             :                                 {
    4967         720 :                                     sEnvelope.Merge(padfRawXValue[l],
    4968         720 :                                                     padfRawYValue[l]);
    4969             :                                 }
    4970             : 
    4971         168 :                                 if (m_sFilterEnvelope.Intersects(sEnvelope))
    4972             :                                 {
    4973          72 :                                     bMatchBBOX = true;
    4974             :                                 }
    4975             :                                 // for bounding box, only the first ring matters
    4976             :                             }
    4977             :                         }
    4978             :                     }
    4979         312 :                     if (bMatchBBOX &&
    4980          72 :                         (m_asAttributeFilterConstraints.empty() ||
    4981           0 :                          !SkipToNextFeatureDueToAttributeFilter()))
    4982             :                     {
    4983          72 :                         bReturnFeature = true;
    4984          72 :                         break;
    4985             :                     }
    4986             : 
    4987         168 :                     IncrFeatureIdx();
    4988         168 :                     m_nIdxInBatch++;
    4989         168 :                     if (m_nIdxInBatch == m_poBatch->num_rows())
    4990             :                     {
    4991          24 :                         m_bEOF = !ReadNextBatch();
    4992          24 :                         if (m_bEOF)
    4993          24 :                             return nullptr;
    4994           0 :                         break;
    4995             :                     }
    4996         144 :                 }
    4997          72 :             } while (!bReturnFeature);
    4998             :         }
    4999         121 :         else if (iCol >= 0)
    5000             :         {
    5001         121 :             auto array = GetStorageArray(m_poBatchColumns[iCol].get());
    5002             :             while (true)
    5003             :             {
    5004         220 :                 bool bMatchBBOX = false;
    5005             : 
    5006             :                 auto poGeometry = std::unique_ptr<OGRGeometry>(
    5007         220 :                     ReadGeometry(m_iGeomFieldFilter, array, m_nIdxInBatch));
    5008         220 :                 if (poGeometry && !poGeometry->IsEmpty())
    5009             :                 {
    5010         115 :                     OGREnvelope sEnvelope;
    5011         115 :                     poGeometry->getEnvelope(&sEnvelope);
    5012         115 :                     if (m_sFilterEnvelope.Intersects(sEnvelope))
    5013             :                     {
    5014          88 :                         bMatchBBOX = true;
    5015             :                     }
    5016             :                 }
    5017         220 :                 if (bMatchBBOX && (m_asAttributeFilterConstraints.empty() ||
    5018           0 :                                    !SkipToNextFeatureDueToAttributeFilter()))
    5019             :                 {
    5020          88 :                     break;
    5021             :                 }
    5022             : 
    5023         132 :                 IncrFeatureIdx();
    5024         132 :                 m_nIdxInBatch++;
    5025         132 :                 if (m_nIdxInBatch == m_poBatch->num_rows())
    5026             :                 {
    5027          33 :                     m_bEOF = !ReadNextBatch();
    5028          33 :                     if (m_bEOF)
    5029          33 :                         return nullptr;
    5030           0 :                     array = GetStorageArray(m_poBatchColumns[iCol].get());
    5031             :                 }
    5032          99 :             }
    5033         816 :         }
    5034             :     }
    5035             : 
    5036        9039 :     else if (!m_asAttributeFilterConstraints.empty())
    5037             :     {
    5038             :         while (true)
    5039             :         {
    5040         438 :             if (!SkipToNextFeatureDueToAttributeFilter())
    5041             :             {
    5042         236 :                 break;
    5043             :             }
    5044             : 
    5045         202 :             IncrFeatureIdx();
    5046         202 :             m_nIdxInBatch++;
    5047         202 :             if (m_nIdxInBatch == m_poBatch->num_rows())
    5048             :             {
    5049          90 :                 m_bEOF = !ReadNextBatch();
    5050          90 :                 if (m_bEOF)
    5051          52 :                     return nullptr;
    5052             :             }
    5053             :         }
    5054             :     }
    5055             : 
    5056        9803 :     auto poFeature = ReadFeature(m_nIdxInBatch, m_poBatchColumns);
    5057             : 
    5058        9803 :     if (m_iFIDArrowColumn < 0)
    5059        6409 :         poFeature->SetFID(m_nFeatureIdx);
    5060             : 
    5061        9803 :     IncrFeatureIdx();
    5062        9803 :     m_nIdxInBatch++;
    5063             : 
    5064        9803 :     return poFeature;
    5065             : }
    5066             : 
    5067             : /************************************************************************/
    5068             : /*                            GetExtent()                               */
    5069             : /************************************************************************/
    5070             : 
    5071          11 : inline OGRErr OGRArrowLayer::GetExtent(OGREnvelope *psExtent, int bForce)
    5072             : {
    5073          11 :     return GetExtent(0, psExtent, bForce);
    5074             : }
    5075             : 
    5076             : /************************************************************************/
    5077             : /*                       GetExtentFromMetadata()                        */
    5078             : /************************************************************************/
    5079             : 
    5080             : inline OGRErr
    5081         756 : OGRArrowLayer::GetExtentFromMetadata(const CPLJSONObject &oJSONDef,
    5082             :                                      OGREnvelope3D *psExtent)
    5083             : {
    5084        2268 :     const auto oBBox = oJSONDef.GetArray("bbox");
    5085         756 :     if (oBBox.IsValid() && oBBox.Size() == 4)
    5086             :     {
    5087         451 :         psExtent->MinX = oBBox[0].ToDouble();
    5088         451 :         psExtent->MinY = oBBox[1].ToDouble();
    5089         451 :         psExtent->MinZ = std::numeric_limits<double>::infinity();
    5090         451 :         psExtent->MaxX = oBBox[2].ToDouble();
    5091         451 :         psExtent->MaxY = oBBox[3].ToDouble();
    5092         451 :         psExtent->MaxZ = -std::numeric_limits<double>::infinity();
    5093         451 :         if (psExtent->MinX <= psExtent->MaxX)
    5094         451 :             return OGRERR_NONE;
    5095             :     }
    5096         305 :     else if (oBBox.IsValid() && oBBox.Size() == 6)
    5097             :     {
    5098         236 :         psExtent->MinX = oBBox[0].ToDouble();
    5099         236 :         psExtent->MinY = oBBox[1].ToDouble();
    5100         236 :         psExtent->MinZ = oBBox[2].ToDouble();
    5101         236 :         psExtent->MaxX = oBBox[3].ToDouble();
    5102         236 :         psExtent->MaxY = oBBox[4].ToDouble();
    5103         236 :         psExtent->MaxZ = oBBox[5].ToDouble();
    5104         236 :         if (psExtent->MinX <= psExtent->MaxX)
    5105         236 :             return OGRERR_NONE;
    5106             :     }
    5107          69 :     return OGRERR_FAILURE;
    5108             : }
    5109             : 
    5110             : /************************************************************************/
    5111             : /*                        SetSpatialFilter()                            */
    5112             : /************************************************************************/
    5113             : 
    5114        1346 : inline void OGRArrowLayer::SetSpatialFilter(int iGeomField,
    5115             :                                             OGRGeometry *poGeomIn)
    5116             : 
    5117             : {
    5118        1346 :     if (!ValidateGeometryFieldIndexForSetSpatialFilter(iGeomField, poGeomIn))
    5119          18 :         return;
    5120             : 
    5121             :     // When changing filters, we need to invalidate cached batches, as
    5122             :     // PostFilterArrowArray() has potentially modified array contents
    5123        1328 :     if (m_poFilterGeom)
    5124         777 :         InvalidateCachedBatches();
    5125             : 
    5126        1328 :     m_bSpatialFilterIntersectsLayerExtent = true;
    5127        1328 :     if (iGeomField < GetLayerDefn()->GetGeomFieldCount())
    5128             :     {
    5129        1306 :         m_iGeomFieldFilter = iGeomField;
    5130        1306 :         if (InstallFilter(poGeomIn))
    5131        1105 :             ResetReading();
    5132        1306 :         if (m_poFilterGeom != nullptr)
    5133             :         {
    5134        1041 :             OGREnvelope sLayerExtent;
    5135        1041 :             if (FastGetExtent(iGeomField, &sLayerExtent))
    5136             :             {
    5137         619 :                 m_bSpatialFilterIntersectsLayerExtent =
    5138         619 :                     m_sFilterEnvelope.Intersects(sLayerExtent);
    5139             :             }
    5140             :         }
    5141             :     }
    5142             : 
    5143        1328 :     SetBatch(m_poBatch);
    5144             : }
    5145             : 
    5146             : /************************************************************************/
    5147             : /*                         FastGetExtent()                              */
    5148             : /************************************************************************/
    5149             : 
    5150         723 : inline bool OGRArrowLayer::FastGetExtent(int iGeomField,
    5151             :                                          OGREnvelope *psExtent) const
    5152             : {
    5153             :     {
    5154         723 :         const auto oIter = m_oMapExtents.find(iGeomField);
    5155         723 :         if (oIter != m_oMapExtents.end())
    5156             :         {
    5157           5 :             *psExtent = oIter->second;
    5158           5 :             return true;
    5159             :         }
    5160             :     }
    5161             : 
    5162             :     const char *pszGeomFieldName =
    5163         718 :         m_poFeatureDefn->GetGeomFieldDefn(iGeomField)->GetNameRef();
    5164         718 :     const auto oIter = m_oMapGeometryColumns.find(pszGeomFieldName);
    5165        1434 :     if (oIter != m_oMapGeometryColumns.end() &&
    5166         716 :         CPLTestBool(CPLGetConfigOption(
    5167        1434 :             ("OGR_" + GetDriverUCName() + "_USE_BBOX").c_str(), "YES")))
    5168             :     {
    5169         710 :         const auto &oJSONDef = oIter->second;
    5170         710 :         OGREnvelope3D sEnvelope3D;
    5171         710 :         if (GetExtentFromMetadata(oJSONDef, &sEnvelope3D) == OGRERR_NONE)
    5172             :         {
    5173         651 :             *psExtent = sEnvelope3D;
    5174         651 :             return true;
    5175             :         }
    5176             :     }
    5177          67 :     return false;
    5178             : }
    5179             : 
    5180             : /************************************************************************/
    5181             : /*                            GetExtent()                               */
    5182             : /************************************************************************/
    5183             : 
    5184         106 : inline OGRErr OGRArrowLayer::GetExtent(int iGeomField, OGREnvelope *psExtent,
    5185             :                                        int bForce)
    5186             : {
    5187         106 :     if (iGeomField < 0 || iGeomField >= m_poFeatureDefn->GetGeomFieldCount())
    5188             :     {
    5189          44 :         if (iGeomField != 0)
    5190             :         {
    5191          44 :             CPLError(CE_Failure, CPLE_AppDefined,
    5192             :                      "Invalid geometry field index : %d", iGeomField);
    5193             :         }
    5194          44 :         return OGRERR_FAILURE;
    5195             :     }
    5196             : 
    5197          62 :     if (FastGetExtent(iGeomField, psExtent))
    5198             :     {
    5199          41 :         return OGRERR_NONE;
    5200             :     }
    5201             : 
    5202          21 :     if (!bForce && !CanRunNonForcedGetExtent())
    5203             :     {
    5204           0 :         return OGRERR_FAILURE;
    5205             :     }
    5206             : 
    5207             :     int iCol;
    5208          21 :     if (m_bIgnoredFields)
    5209             :     {
    5210           0 :         iCol = m_anMapGeomFieldIndexToArrayIndex[iGeomField];
    5211             :     }
    5212             :     else
    5213             :     {
    5214          21 :         iCol = m_anMapGeomFieldIndexToArrowColumn[iGeomField];
    5215             :     }
    5216          21 :     if (iCol < 0)
    5217             :     {
    5218           0 :         return OGRERR_FAILURE;
    5219             :     }
    5220             : 
    5221          21 :     if (m_aeGeomEncoding[iGeomField] == OGRArrowGeomEncoding::WKB)
    5222             :     {
    5223           7 :         ResetReading();
    5224           7 :         if (m_poBatch == nullptr)
    5225             :         {
    5226           6 :             m_bEOF = !ReadNextBatch();
    5227           6 :             if (m_bEOF)
    5228           0 :                 return OGRERR_FAILURE;
    5229             :         }
    5230           7 :         *psExtent = OGREnvelope();
    5231             : 
    5232          14 :         auto array = m_poBatchColumns[iCol];
    5233           7 :         std::shared_ptr<arrow::BinaryArray> smallArray;
    5234           7 :         std::shared_ptr<arrow::LargeBinaryArray> largeArray;
    5235           7 :         if (array->type_id() == arrow::Type::BINARY)
    5236           7 :             smallArray = std::static_pointer_cast<arrow::BinaryArray>(array);
    5237             :         else
    5238             :         {
    5239           0 :             CPLAssert(array->type_id() == arrow::Type::LARGE_BINARY);
    5240             :             largeArray =
    5241           0 :                 std::static_pointer_cast<arrow::LargeBinaryArray>(array);
    5242             :         }
    5243           7 :         OGREnvelope sEnvelope;
    5244             :         while (true)
    5245             :         {
    5246          32 :             if (!array->IsNull(m_nIdxInBatch))
    5247             :             {
    5248          30 :                 if (smallArray)
    5249             :                 {
    5250          30 :                     int out_length = 0;
    5251             :                     const uint8_t *data =
    5252          30 :                         smallArray->GetValue(m_nIdxInBatch, &out_length);
    5253          30 :                     if (OGRWKBGetBoundingBox(data, out_length, sEnvelope))
    5254             :                     {
    5255          30 :                         psExtent->Merge(sEnvelope);
    5256             :                     }
    5257             :                 }
    5258             :                 else
    5259             :                 {
    5260           0 :                     int64_t out_length = 0;
    5261             :                     const uint8_t *data =
    5262           0 :                         largeArray->GetValue(m_nIdxInBatch, &out_length);
    5263           0 :                     if (out_length < INT_MAX &&
    5264           0 :                         OGRWKBGetBoundingBox(data, static_cast<int>(out_length),
    5265             :                                              sEnvelope))
    5266             :                     {
    5267           0 :                         psExtent->Merge(sEnvelope);
    5268             :                     }
    5269             :                 }
    5270             :             }
    5271             : 
    5272          32 :             m_nIdxInBatch++;
    5273          32 :             if (m_nIdxInBatch == m_poBatch->num_rows())
    5274             :             {
    5275           7 :                 m_bEOF = !ReadNextBatch();
    5276           7 :                 if (m_bEOF)
    5277             :                 {
    5278           7 :                     ResetReading();
    5279           7 :                     if (psExtent->IsInit())
    5280             :                     {
    5281           7 :                         m_oMapExtents[iGeomField] = *psExtent;
    5282           7 :                         return OGRERR_NONE;
    5283             :                     }
    5284           0 :                     return OGRERR_FAILURE;
    5285             :                 }
    5286           0 :                 array = m_poBatchColumns[iCol];
    5287           0 :                 if (array->type_id() == arrow::Type::BINARY)
    5288             :                     smallArray =
    5289           0 :                         std::static_pointer_cast<arrow::BinaryArray>(array);
    5290             :                 else
    5291             :                 {
    5292           0 :                     CPLAssert(array->type_id() == arrow::Type::LARGE_BINARY);
    5293             :                     largeArray =
    5294           0 :                         std::static_pointer_cast<arrow::LargeBinaryArray>(
    5295           0 :                             array);
    5296             :                 }
    5297             :             }
    5298          25 :         }
    5299             :     }
    5300          14 :     else if (m_aeGeomEncoding[iGeomField] ==
    5301             :              OGRArrowGeomEncoding::GEOARROW_FSL_MULTIPOLYGON)
    5302             :     {
    5303           0 :         ResetReading();
    5304           0 :         if (m_poBatch == nullptr)
    5305             :         {
    5306           0 :             m_bEOF = !ReadNextBatch();
    5307           0 :             if (m_bEOF)
    5308           0 :                 return OGRERR_FAILURE;
    5309             :         }
    5310           0 :         *psExtent = OGREnvelope();
    5311             : 
    5312             :         const auto poGeomFieldDefn =
    5313           0 :             m_poFeatureDefn->GetGeomFieldDefn(iGeomField);
    5314           0 :         const auto eGeomType = poGeomFieldDefn->GetType();
    5315           0 :         const bool bHasZ = CPL_TO_BOOL(OGR_GT_HasZ(eGeomType));
    5316           0 :         const bool bHasM = CPL_TO_BOOL(OGR_GT_HasM(eGeomType));
    5317           0 :         const int nDim = 2 + (bHasZ ? 1 : 0) + (bHasM ? 1 : 0);
    5318             : 
    5319           0 :     begin_multipolygon:
    5320           0 :         auto array = m_poBatchColumns[iCol].get();
    5321           0 :         CPLAssert(array->type_id() == arrow::Type::LIST);
    5322           0 :         auto listOfPartsArray = static_cast<const arrow::ListArray *>(array);
    5323           0 :         CPLAssert(listOfPartsArray->values()->type_id() == arrow::Type::LIST);
    5324             :         auto listOfPartsValues = std::static_pointer_cast<arrow::ListArray>(
    5325           0 :             listOfPartsArray->values());
    5326           0 :         CPLAssert(listOfPartsValues->values()->type_id() == arrow::Type::LIST);
    5327             :         auto listOfRingsValues = std::static_pointer_cast<arrow::ListArray>(
    5328           0 :             listOfPartsValues->values());
    5329           0 :         CPLAssert(listOfRingsValues->values()->type_id() ==
    5330             :                   arrow::Type::FIXED_SIZE_LIST);
    5331             :         auto listOfPointsValues =
    5332             :             std::static_pointer_cast<arrow::FixedSizeListArray>(
    5333           0 :                 listOfRingsValues->values());
    5334           0 :         CPLAssert(listOfPointsValues->values()->type_id() ==
    5335             :                   arrow::Type::DOUBLE);
    5336             :         auto pointValues = std::static_pointer_cast<arrow::DoubleArray>(
    5337           0 :             listOfPointsValues->values());
    5338             : 
    5339             :         while (true)
    5340             :         {
    5341           0 :             if (!listOfPartsArray->IsNull(m_nIdxInBatch))
    5342             :             {
    5343             :                 const auto nParts =
    5344           0 :                     listOfPartsArray->value_length(m_nIdxInBatch);
    5345             :                 const auto nPartOffset =
    5346           0 :                     listOfPartsArray->value_offset(m_nIdxInBatch);
    5347           0 :                 for (auto j = decltype(nParts){0}; j < nParts; j++)
    5348             :                 {
    5349             :                     const auto nRings =
    5350           0 :                         listOfPartsValues->value_length(nPartOffset + j);
    5351             :                     const auto nRingOffset =
    5352           0 :                         listOfPartsValues->value_offset(nPartOffset + j);
    5353           0 :                     if (nRings >= 1)
    5354             :                     {
    5355             :                         const auto nPoints =
    5356           0 :                             listOfRingsValues->value_length(nRingOffset);
    5357             :                         const auto nPointOffset =
    5358           0 :                             listOfRingsValues->value_offset(nRingOffset) * nDim;
    5359             :                         const double *padfRawValue =
    5360           0 :                             pointValues->raw_values() + nPointOffset;
    5361           0 :                         for (auto l = decltype(nPoints){0}; l < nPoints; ++l)
    5362             :                         {
    5363           0 :                             psExtent->Merge(padfRawValue[nDim * l],
    5364           0 :                                             padfRawValue[nDim * l + 1]);
    5365             :                         }
    5366             :                         // for bounding box, only the first ring matters
    5367             :                     }
    5368             :                 }
    5369             :             }
    5370             : 
    5371           0 :             m_nIdxInBatch++;
    5372           0 :             if (m_nIdxInBatch == m_poBatch->num_rows())
    5373             :             {
    5374           0 :                 m_bEOF = !ReadNextBatch();
    5375           0 :                 if (m_bEOF)
    5376             :                 {
    5377           0 :                     ResetReading();
    5378           0 :                     if (psExtent->IsInit())
    5379             :                     {
    5380           0 :                         m_oMapExtents[iGeomField] = *psExtent;
    5381           0 :                         return OGRERR_NONE;
    5382             :                     }
    5383           0 :                     return OGRERR_FAILURE;
    5384             :                 }
    5385           0 :                 goto begin_multipolygon;
    5386             :             }
    5387           0 :         }
    5388             :     }
    5389             : 
    5390          14 :     return GetExtentInternal(iGeomField, psExtent, bForce);
    5391             : }
    5392             : 
    5393             : /************************************************************************/
    5394             : /*                        FastGetExtent3D()                             */
    5395             : /************************************************************************/
    5396             : 
    5397          38 : inline bool OGRArrowLayer::FastGetExtent3D(int iGeomField,
    5398             :                                            OGREnvelope3D *psExtent) const
    5399             : {
    5400             :     const char *pszGeomFieldName =
    5401          38 :         m_poFeatureDefn->GetGeomFieldDefn(iGeomField)->GetNameRef();
    5402          38 :     const auto oIter = m_oMapGeometryColumns.find(pszGeomFieldName);
    5403          76 :     if (oIter != m_oMapGeometryColumns.end() &&
    5404          38 :         CPLTestBool(CPLGetConfigOption(
    5405          76 :             ("OGR_" + GetDriverUCName() + "_USE_BBOX").c_str(), "YES")))
    5406             :     {
    5407          36 :         const auto &oJSONDef = oIter->second;
    5408          64 :         if (GetExtentFromMetadata(oJSONDef, psExtent) == OGRERR_NONE &&
    5409          28 :             psExtent->Is3D())
    5410             :         {
    5411           2 :             return true;
    5412             :         }
    5413             :     }
    5414          36 :     return false;
    5415             : }
    5416             : 
    5417             : /************************************************************************/
    5418             : /*                           GetExtent3D()                              */
    5419             : /************************************************************************/
    5420             : 
    5421          19 : inline OGRErr OGRArrowLayer::GetExtent3D(int iGeomField,
    5422             :                                          OGREnvelope3D *psExtent, int bForce)
    5423             : {
    5424          19 :     if (iGeomField < 0 || iGeomField >= m_poFeatureDefn->GetGeomFieldCount())
    5425             :     {
    5426           0 :         if (iGeomField != 0)
    5427             :         {
    5428           0 :             CPLError(CE_Failure, CPLE_AppDefined,
    5429             :                      "Invalid geometry field index : %d", iGeomField);
    5430             :         }
    5431           0 :         return OGRERR_FAILURE;
    5432             :     }
    5433             : 
    5434          19 :     if (FastGetExtent3D(iGeomField, psExtent))
    5435             :     {
    5436           1 :         return OGRERR_NONE;
    5437             :     }
    5438             : 
    5439          18 :     return OGRLayer::GetExtent3D(iGeomField, psExtent, bForce);
    5440             : }
    5441             : 
    5442             : /************************************************************************/
    5443             : /*                  OverrideArrowSchemaRelease()                        */
    5444             : /************************************************************************/
    5445             : 
    5446             : template <class T>
    5447         609 : static void OverrideArrowRelease(OGRArrowDataset *poDS, T *obj)
    5448             : {
    5449             :     // We override the release callback, since it can use the memory pool,
    5450             :     // and we need to make sure it is still alive when the object (ArrowArray
    5451             :     // or ArrowSchema) is deleted
    5452         609 :     struct OverriddenPrivate
    5453             :     {
    5454             :         OverriddenPrivate() = default;
    5455             :         OverriddenPrivate(const OverriddenPrivate &) = delete;
    5456             :         OverriddenPrivate &operator=(const OverriddenPrivate &) = delete;
    5457             : 
    5458             :         std::shared_ptr<arrow::MemoryPool> poMemoryPool{};
    5459             :         void (*pfnPreviousRelease)(T *) = nullptr;
    5460             :         void *pPreviousPrivateData = nullptr;
    5461             : 
    5462         609 :         static void release(T *l_obj)
    5463             :         {
    5464         609 :             OverriddenPrivate *myPrivate =
    5465             :                 static_cast<OverriddenPrivate *>(l_obj->private_data);
    5466         609 :             l_obj->private_data = myPrivate->pPreviousPrivateData;
    5467         609 :             l_obj->release = myPrivate->pfnPreviousRelease;
    5468         609 :             l_obj->release(l_obj);
    5469         609 :             delete myPrivate;
    5470         609 :         }
    5471             :     };
    5472             : 
    5473         609 :     auto overriddenPrivate = new OverriddenPrivate();
    5474         609 :     overriddenPrivate->poMemoryPool = poDS->GetSharedMemoryPool();
    5475         609 :     overriddenPrivate->pPreviousPrivateData = obj->private_data;
    5476         609 :     overriddenPrivate->pfnPreviousRelease = obj->release;
    5477             : 
    5478         609 :     obj->release = OverriddenPrivate::release;
    5479         609 :     obj->private_data = overriddenPrivate;
    5480         609 : }
    5481             : 
    5482             : /************************************************************************/
    5483             : /*                   UseRecordBatchBaseImplementation()                 */
    5484             : /************************************************************************/
    5485             : 
    5486         474 : inline bool OGRArrowLayer::UseRecordBatchBaseImplementation() const
    5487             : {
    5488         474 :     if (CPLTestBool(CPLGetConfigOption("OGR_ARROW_STREAM_BASE_IMPL", "NO")))
    5489             :     {
    5490           4 :         return true;
    5491             :     }
    5492             : 
    5493         470 :     if (m_aosArrowArrayStreamOptions.FetchBool(GAS_OPT_DATETIME_AS_STRING,
    5494             :                                                false))
    5495             :     {
    5496           1 :         const int nFieldCount = m_poFeatureDefn->GetFieldCount();
    5497          14 :         for (int i = 0; i < nFieldCount; ++i)
    5498             :         {
    5499          14 :             const auto poFieldDefn = m_poFeatureDefn->GetFieldDefn(i);
    5500          28 :             if (!poFieldDefn->IsIgnored() &&
    5501          14 :                 poFieldDefn->GetType() == OFTDateTime)
    5502             :             {
    5503           1 :                 CPLDebug("ARROW",
    5504             :                          "DATETIME_AS_STRING=YES not compatible of fast "
    5505             :                          "Arrow implementation");
    5506           1 :                 return true;
    5507             :             }
    5508             :         }
    5509             :     }
    5510             : 
    5511         469 :     if (EQUAL(m_aosArrowArrayStreamOptions.FetchNameValueDef(
    5512             :                   "GEOMETRY_ENCODING", ""),
    5513             :               "WKB"))
    5514             :     {
    5515         131 :         const int nGeomFieldCount = m_poFeatureDefn->GetGeomFieldCount();
    5516         181 :         for (int i = 0; i < nGeomFieldCount; i++)
    5517             :         {
    5518         146 :             if (!m_poFeatureDefn->GetGeomFieldDefn(i)->IsIgnored() &&
    5519         259 :                 m_aeGeomEncoding[i] != OGRArrowGeomEncoding::WKB &&
    5520         113 :                 m_aeGeomEncoding[i] != OGRArrowGeomEncoding::WKT)
    5521             :             {
    5522          96 :                 CPLDebug("ARROW", "Geometry encoding not compatible of fast "
    5523             :                                   "Arrow implementation");
    5524          96 :                 return true;
    5525             :             }
    5526             :         }
    5527             :     }
    5528             : 
    5529         373 :     if (m_bIgnoredFields)
    5530             :     {
    5531             :         std::vector<int> ignoredState(m_anMapFieldIndexToArrowColumn.size(),
    5532         106 :                                       -1);
    5533        7658 :         for (size_t i = 0; i < m_anMapFieldIndexToArrowColumn.size(); i++)
    5534             :         {
    5535        7553 :             const int nArrowCol = m_anMapFieldIndexToArrowColumn[i][0];
    5536        7553 :             if (nArrowCol >= static_cast<int>(ignoredState.size()))
    5537           1 :                 ignoredState.resize(nArrowCol + 1, -1);
    5538             :             const auto bIsIgnored =
    5539        7553 :                 m_poFeatureDefn->GetFieldDefn(static_cast<int>(i))->IsIgnored();
    5540        7553 :             if (ignoredState[nArrowCol] < 0)
    5541             :             {
    5542        7095 :                 ignoredState[nArrowCol] = static_cast<int>(bIsIgnored);
    5543             :             }
    5544             :             else
    5545             :             {
    5546             :                 // struct fields will point to the same arrow column
    5547         458 :                 if (ignoredState[nArrowCol] != static_cast<int>(bIsIgnored))
    5548             :                 {
    5549           1 :                     CPLDebug("ARROW",
    5550             :                              "Inconsistent ignore state for Arrow Columns");
    5551           1 :                     return true;
    5552             :                 }
    5553             :             }
    5554             :         }
    5555             :     }
    5556             : 
    5557         372 :     if (m_poAttrQuery || m_poFilterGeom)
    5558             :     {
    5559         163 :         struct ArrowSchema *psSchema = &m_sCachedSchema;
    5560         163 :         if (psSchema->release)
    5561         104 :             psSchema->release(psSchema);
    5562         163 :         memset(psSchema, 0, sizeof(*psSchema));
    5563             : 
    5564         326 :         const bool bCanPostFilter = GetArrowSchemaInternal(psSchema) == 0 &&
    5565         163 :                                     CanPostFilterArrowArray(psSchema);
    5566         163 :         if (!bCanPostFilter)
    5567          11 :             return true;
    5568             :     }
    5569             : 
    5570         361 :     return false;
    5571             : }
    5572             : 
    5573             : /************************************************************************/
    5574             : /*                          GetArrowStream()                            */
    5575             : /************************************************************************/
    5576             : 
    5577         293 : inline bool OGRArrowLayer::GetArrowStream(struct ArrowArrayStream *out_stream,
    5578             :                                           CSLConstList papszOptions)
    5579             : {
    5580         293 :     if (!OGRLayer::GetArrowStream(out_stream, papszOptions))
    5581           0 :         return false;
    5582             : 
    5583         293 :     m_bUseRecordBatchBaseImplementation = UseRecordBatchBaseImplementation();
    5584         293 :     return true;
    5585             : }
    5586             : 
    5587             : /************************************************************************/
    5588             : /*                         GetArrowSchema()                             */
    5589             : /************************************************************************/
    5590             : 
    5591         433 : inline int OGRArrowLayer::GetArrowSchema(struct ArrowArrayStream *stream,
    5592             :                                          struct ArrowSchema *out_schema)
    5593             : {
    5594         433 :     if (m_bUseRecordBatchBaseImplementation)
    5595         209 :         return OGRLayer::GetArrowSchema(stream, out_schema);
    5596             : 
    5597         224 :     return GetArrowSchemaInternal(out_schema);
    5598             : }
    5599             : 
    5600             : /************************************************************************/
    5601             : /*                     GetArrowSchemaInternal()                         */
    5602             : /************************************************************************/
    5603             : 
    5604       10800 : static bool IsSilentlyIgnoredFormatForGetArrowSchemaArray(const char *format)
    5605             : {
    5606             :     // n: null
    5607       10800 :     return strcmp(format, "n") == 0;
    5608             : }
    5609             : 
    5610             : inline int
    5611         387 : OGRArrowLayer::GetArrowSchemaInternal(struct ArrowSchema *out_schema) const
    5612             : {
    5613         774 :     auto status = arrow::ExportSchema(*m_poSchema, out_schema);
    5614         387 :     if (!status.ok())
    5615             :     {
    5616           0 :         CPLError(CE_Failure, CPLE_AppDefined, "ExportSchema() failed with %s",
    5617           0 :                  status.message().c_str());
    5618           0 :         return EIO;
    5619             :     }
    5620             : 
    5621         387 :     CPLAssert(out_schema->n_children == m_poSchema->num_fields());
    5622             : 
    5623             :     // Remove ignored fields from the ArrowSchema.
    5624             : 
    5625             :     struct FieldDesc
    5626             :     {
    5627             :         bool bIsRegularField =
    5628             :             false;  // true = attribute field, false = geometry field
    5629             :         int nIdx = -1;
    5630             :     };
    5631             : 
    5632             :     // cppcheck-suppress unreadVariable
    5633             :     std::vector<FieldDesc> fieldDesc(
    5634         774 :         static_cast<size_t>(out_schema->n_children));
    5635       21816 :     for (size_t i = 0; i < m_anMapFieldIndexToArrowColumn.size(); i++)
    5636             :     {
    5637       21429 :         const int nArrowCol = m_anMapFieldIndexToArrowColumn[i][0];
    5638       21429 :         if (fieldDesc[nArrowCol].nIdx < 0)
    5639             :         {
    5640       20148 :             fieldDesc[nArrowCol].bIsRegularField = true;
    5641       20148 :             fieldDesc[nArrowCol].nIdx = static_cast<int>(i);
    5642             :         }
    5643             :     }
    5644         788 :     for (size_t i = 0; i < m_anMapGeomFieldIndexToArrowColumn.size(); i++)
    5645             :     {
    5646         401 :         const int nArrowCol = m_anMapGeomFieldIndexToArrowColumn[i];
    5647         401 :         CPLAssert(fieldDesc[nArrowCol].nIdx < 0);
    5648         401 :         fieldDesc[nArrowCol].bIsRegularField = false;
    5649         401 :         fieldDesc[nArrowCol].nIdx = static_cast<int>(i);
    5650             :     }
    5651             : 
    5652         387 :     int j = 0;
    5653             :     const char *pszReqGeomEncoding =
    5654         387 :         m_aosArrowArrayStreamOptions.FetchNameValueDef("GEOMETRY_ENCODING", "");
    5655             : 
    5656         387 :     const char *pszExtensionName = EXTENSION_NAME_OGC_WKB;
    5657         387 :     if (EQUAL(pszReqGeomEncoding, "WKB") || EQUAL(pszReqGeomEncoding, ""))
    5658             :     {
    5659             :         const char *const pszGeometryMetadataEncoding =
    5660         387 :             m_aosArrowArrayStreamOptions.FetchNameValue(
    5661             :                 "GEOMETRY_METADATA_ENCODING");
    5662         387 :         if (pszGeometryMetadataEncoding)
    5663             :         {
    5664           0 :             if (EQUAL(pszGeometryMetadataEncoding, "OGC"))
    5665           0 :                 pszExtensionName = EXTENSION_NAME_OGC_WKB;
    5666           0 :             else if (EQUAL(pszGeometryMetadataEncoding, "GEOARROW"))
    5667           0 :                 pszExtensionName = EXTENSION_NAME_GEOARROW_WKB;
    5668             :             else
    5669           0 :                 CPLError(CE_Warning, CPLE_NotSupported,
    5670             :                          "Unsupported GEOMETRY_METADATA_ENCODING value: %s",
    5671             :                          pszGeometryMetadataEncoding);
    5672             :         }
    5673             :     }
    5674             : 
    5675       21294 :     for (int i = 0; i < out_schema->n_children; ++i)
    5676             :     {
    5677       20907 :         if (fieldDesc[i].nIdx < 0)
    5678             :         {
    5679         358 :             if (m_iFIDArrowColumn == i)
    5680             :             {
    5681          21 :                 out_schema->children[j] = out_schema->children[i];
    5682          21 :                 ++j;
    5683             :             }
    5684         337 :             else if (cpl::contains(m_oSetBBoxArrowColumns, i))
    5685             :             {
    5686             :                 // Remove bounding box columns from exported schema
    5687          82 :                 out_schema->children[i]->release(out_schema->children[i]);
    5688          82 :                 out_schema->children[i] = nullptr;
    5689             :             }
    5690         255 :             else if (IsSilentlyIgnoredFormatForGetArrowSchemaArray(
    5691         255 :                          out_schema->children[i]->format))
    5692             :             {
    5693             :                 // Silently ignore columns with null data type...
    5694         255 :                 out_schema->children[i]->release(out_schema->children[i]);
    5695             :             }
    5696             :             else
    5697             :             {
    5698             :                 // can happen with data types we don't support
    5699           0 :                 if (m_aosArrowArrayStreamOptions.FetchBool(
    5700             :                         "SILENCE_GET_SCHEMA_ERROR", false))
    5701             :                 {
    5702           0 :                     CPLDebug(GetDriverUCName().c_str(),
    5703             :                              "GetArrowSchema() error: fieldDesc[%d].nIdx < 0 "
    5704             :                              "not expected: name=%s, format=%s",
    5705           0 :                              i, out_schema->children[i]->name,
    5706           0 :                              out_schema->children[i]->format);
    5707             :                 }
    5708             :                 else
    5709             :                 {
    5710           0 :                     CPLError(CE_Failure, CPLE_NotSupported,
    5711             :                              "GetArrowSchema() error: fieldDesc[%d].nIdx < 0 "
    5712             :                              "not expected: name=%s, format=%s",
    5713           0 :                              i, out_schema->children[i]->name,
    5714           0 :                              out_schema->children[i]->format);
    5715             :                 }
    5716           0 :                 for (; i < out_schema->n_children; ++i, ++j)
    5717           0 :                     out_schema->children[j] = out_schema->children[i];
    5718           0 :                 out_schema->n_children = j;
    5719             : 
    5720           0 :                 out_schema->release(out_schema);
    5721             : 
    5722           0 :                 return EIO;
    5723             :             }
    5724         358 :             continue;
    5725             :         }
    5726             : 
    5727             :         const auto bIsIgnored =
    5728       20549 :             fieldDesc[i].bIsRegularField
    5729       20549 :                 ? m_poFeatureDefn->GetFieldDefn(fieldDesc[i].nIdx)->IsIgnored()
    5730         401 :                 : m_poFeatureDefn->GetGeomFieldDefn(fieldDesc[i].nIdx)
    5731         401 :                       ->IsIgnored();
    5732       20549 :         if (bIsIgnored)
    5733             :         {
    5734        2911 :             out_schema->children[i]->release(out_schema->children[i]);
    5735             :         }
    5736             :         else
    5737             :         {
    5738       18021 :             if (!fieldDesc[i].bIsRegularField &&
    5739         383 :                 EQUAL(pszReqGeomEncoding, "WKB"))
    5740             :             {
    5741          64 :                 const int iGeomField = fieldDesc[i].nIdx;
    5742          64 :                 if (m_aeGeomEncoding[iGeomField] == OGRArrowGeomEncoding::WKT)
    5743             :                 {
    5744             :                     const auto poGeomFieldDefn =
    5745          17 :                         m_poFeatureDefn->GetGeomFieldDefn(iGeomField);
    5746          17 :                     CPLAssert(strcmp(out_schema->children[i]->name,
    5747             :                                      poGeomFieldDefn->GetNameRef()) == 0);
    5748          17 :                     auto poSchema = CreateSchemaForWKBGeometryColumn(
    5749             :                         poGeomFieldDefn, "z", pszExtensionName);
    5750          17 :                     out_schema->children[i]->release(out_schema->children[i]);
    5751          17 :                     *(out_schema->children[j]) = *poSchema;
    5752          17 :                     CPLFree(poSchema);
    5753             :                 }
    5754          47 :                 else if (m_aeGeomEncoding[iGeomField] !=
    5755             :                          OGRArrowGeomEncoding::WKB)
    5756             :                 {
    5757             :                     // Shouldn't happen if UseRecordBatchBaseImplementation()
    5758             :                     // is up to date
    5759           0 :                     CPLAssert(false);
    5760             :                 }
    5761             :                 else
    5762             :                 {
    5763          47 :                     out_schema->children[j] = out_schema->children[i];
    5764             :                 }
    5765             :             }
    5766             :             else
    5767             :             {
    5768       17574 :                 out_schema->children[j] = out_schema->children[i];
    5769             :             }
    5770             : 
    5771       18021 :             if (!fieldDesc[i].bIsRegularField &&
    5772         383 :                 (EQUAL(pszReqGeomEncoding, "WKB") ||
    5773         319 :                  EQUAL(pszReqGeomEncoding, "")))
    5774             :             {
    5775         383 :                 const int iGeomField = fieldDesc[i].nIdx;
    5776         383 :                 const char *pszFormat = out_schema->children[j]->format;
    5777         383 :                 if (m_aeGeomEncoding[iGeomField] == OGRArrowGeomEncoding::WKB &&
    5778         695 :                     !out_schema->children[j]->metadata &&
    5779         312 :                     (strcmp(pszFormat, "z") == 0 ||
    5780           1 :                      strcmp(pszFormat, "Z") == 0))
    5781             :                 {
    5782             :                     const auto poGeomFieldDefn =
    5783         312 :                         m_poFeatureDefn->GetGeomFieldDefn(iGeomField);
    5784             :                     // Set ARROW:extension:name = ogc:wkb
    5785         312 :                     auto poSchema = CreateSchemaForWKBGeometryColumn(
    5786             :                         poGeomFieldDefn, pszFormat, pszExtensionName);
    5787         312 :                     out_schema->children[i]->release(out_schema->children[i]);
    5788         312 :                     *(out_schema->children[j]) = *poSchema;
    5789         312 :                     CPLFree(poSchema);
    5790             :                 }
    5791             :             }
    5792             : 
    5793       17638 :             ++j;
    5794             :         }
    5795             :     }
    5796             : 
    5797         387 :     out_schema->n_children = j;
    5798             : 
    5799         387 :     OverrideArrowRelease(m_poArrowDS, out_schema);
    5800             : 
    5801         387 :     return 0;
    5802             : }
    5803             : 
    5804             : /************************************************************************/
    5805             : /*                       GetNextArrowArray()                            */
    5806             : /************************************************************************/
    5807             : 
    5808         604 : inline int OGRArrowLayer::GetNextArrowArray(struct ArrowArrayStream *stream,
    5809             :                                             struct ArrowArray *out_array)
    5810             : {
    5811         604 :     if (m_bUseRecordBatchBaseImplementation)
    5812         224 :         return OGRLayer::GetNextArrowArray(stream, out_array);
    5813             : 
    5814             :     while (true)
    5815             :     {
    5816         398 :         if (m_bEOF)
    5817             :         {
    5818          10 :             memset(out_array, 0, sizeof(*out_array));
    5819         176 :             return 0;
    5820             :         }
    5821             : 
    5822         388 :         if (m_poBatch == nullptr || m_nIdxInBatch == m_poBatch->num_rows())
    5823             :         {
    5824         363 :             if (!ReadNextBatch())
    5825             :             {
    5826         166 :                 if (m_poAttrQuery || m_poFilterGeom)
    5827             :                 {
    5828          97 :                     InvalidateCachedBatches();
    5829             :                 }
    5830         166 :                 m_bEOF = true;
    5831         166 :                 memset(out_array, 0, sizeof(*out_array));
    5832         166 :                 return 0;
    5833             :             }
    5834             :         }
    5835             : 
    5836         222 :         const bool bNeedsPostFilter =
    5837         333 :             (m_poAttrQuery && !m_bBaseArrowIgnoreAttributeFilter) ||
    5838         111 :             (m_poFilterGeom && !m_bBaseArrowIgnoreSpatialFilter);
    5839             : 
    5840             :         struct ArrowSchema schema;
    5841         222 :         memset(&schema, 0, sizeof(schema));
    5842         222 :         auto status = arrow::ExportRecordBatch(*m_poBatch, out_array, &schema);
    5843         222 :         m_nIdxInBatch = m_poBatch->num_rows();
    5844         222 :         if (!status.ok())
    5845             :         {
    5846           0 :             CPLError(CE_Failure, CPLE_AppDefined,
    5847             :                      "ExportRecordBatch() failed with %s",
    5848           0 :                      status.message().c_str());
    5849           0 :             return EIO;
    5850             :         }
    5851             : 
    5852             :         // Remove bounding box columns from exported array, or columns
    5853             :         // of unsupported data types that we voluntarily strip off.
    5854             :         const auto RemoveBBoxOrUnsupportedColumns =
    5855       32653 :             [out_array, &schema](const std::set<int> &oSetBBoxArrayIndex)
    5856             :         {
    5857         222 :             int j = 0;
    5858       10817 :             for (int i = 0; i < static_cast<int>(schema.n_children); ++i)
    5859             :             {
    5860       21140 :                 if (cpl::contains(oSetBBoxArrayIndex, i) ||
    5861       10545 :                     IsSilentlyIgnoredFormatForGetArrowSchemaArray(
    5862       10545 :                         schema.children[i]->format))
    5863             :                 {
    5864         126 :                     out_array->children[i]->release(out_array->children[i]);
    5865         126 :                     out_array->children[i] = nullptr;
    5866             : 
    5867         126 :                     schema.children[i]->release(schema.children[i]);
    5868         126 :                     schema.children[i] = nullptr;
    5869             :                 }
    5870             :                 else
    5871             :                 {
    5872       10469 :                     out_array->children[j] = out_array->children[i];
    5873       10469 :                     schema.children[j] = schema.children[i];
    5874       10469 :                     ++j;
    5875             :                 }
    5876             :             }
    5877         222 :             out_array->n_children = j;
    5878         222 :             schema.n_children = j;
    5879         222 :         };
    5880             : 
    5881         222 :         if (m_bIgnoredFields)
    5882             :         {
    5883         178 :             std::set<int> oSetBBoxArrayIndex;
    5884         126 :             for (const auto &iter : m_oMapGeomFieldIndexToGeomColBBOX)
    5885             :             {
    5886          37 :                 if (iter.second.iArrayIdx >= 0)
    5887          20 :                     oSetBBoxArrayIndex.insert(iter.second.iArrayIdx);
    5888             :             }
    5889          89 :             RemoveBBoxOrUnsupportedColumns(oSetBBoxArrayIndex);
    5890             :         }
    5891             :         else
    5892             :         {
    5893         133 :             RemoveBBoxOrUnsupportedColumns(m_oSetBBoxArrowColumns);
    5894             :         }
    5895             : 
    5896         222 :         if (EQUAL(m_aosArrowArrayStreamOptions.FetchNameValueDef(
    5897             :                       "GEOMETRY_ENCODING", ""),
    5898             :                   "WKB"))
    5899             :         {
    5900          40 :             const int nGeomFieldCount = m_poFeatureDefn->GetGeomFieldCount();
    5901          95 :             for (int i = 0; i < nGeomFieldCount; i++)
    5902             :             {
    5903             :                 const auto poGeomFieldDefn =
    5904          55 :                     m_poFeatureDefn->GetGeomFieldDefn(i);
    5905          55 :                 if (!poGeomFieldDefn->IsIgnored())
    5906             :                 {
    5907          47 :                     if (m_aeGeomEncoding[i] == OGRArrowGeomEncoding::WKT)
    5908             :                     {
    5909             :                         const int nArrayIdx =
    5910          18 :                             m_bIgnoredFields
    5911          26 :                                 ? m_anMapGeomFieldIndexToArrayIndex[i]
    5912           8 :                                 : m_anMapGeomFieldIndexToArrowColumn[i];
    5913          18 :                         auto sourceArray = out_array->children[nArrayIdx];
    5914             :                         auto targetArray =
    5915          18 :                             strcmp(schema.children[nArrayIdx]->format, "u") == 0
    5916          18 :                                 ? CreateWKBArrayFromWKTArray<uint32_t>(
    5917             :                                       sourceArray)
    5918           0 :                                 : CreateWKBArrayFromWKTArray<uint64_t>(
    5919          18 :                                       sourceArray);
    5920          18 :                         if (targetArray)
    5921             :                         {
    5922          18 :                             sourceArray->release(sourceArray);
    5923          18 :                             *(out_array->children[nArrayIdx]) = *targetArray;
    5924          18 :                             CPLFree(targetArray);
    5925             :                         }
    5926             :                         else
    5927             :                         {
    5928           0 :                             out_array->release(out_array);
    5929           0 :                             memset(out_array, 0, sizeof(*out_array));
    5930           0 :                             if (schema.release)
    5931           0 :                                 schema.release(&schema);
    5932           0 :                             return ENOMEM;
    5933             :                         }
    5934             :                     }
    5935          29 :                     else if (m_aeGeomEncoding[i] != OGRArrowGeomEncoding::WKB)
    5936             :                     {
    5937             :                         // Shouldn't happen if UseRecordBatchBaseImplementation()
    5938             :                         // is up to date
    5939           0 :                         CPLAssert(false);
    5940             :                     }
    5941             :                 }
    5942             :             }
    5943             :         }
    5944             : 
    5945         222 :         if (schema.release)
    5946         222 :             schema.release(&schema);
    5947             : 
    5948         222 :         OverrideArrowRelease(m_poArrowDS, out_array);
    5949             : 
    5950         222 :         const auto nFeatureIdxCur = m_nFeatureIdx;
    5951             :         // TODO: We likely have an issue regarding FIDs based on m_nFeatureIdx
    5952             :         // when m_iFIDArrowColumn < 0, only a subset of row groups is
    5953             :         // selected, and this batch goes across non consecutive row groups.
    5954        1119 :         for (int64_t i = 0; i < m_nIdxInBatch; ++i)
    5955         897 :             IncrFeatureIdx();
    5956             : 
    5957         222 :         if (bNeedsPostFilter)
    5958             :         {
    5959         130 :             CPLStringList aosOptions;
    5960         130 :             if (m_iFIDArrowColumn < 0)
    5961             :                 aosOptions.SetNameValue(
    5962             :                     "BASE_SEQUENTIAL_FID",
    5963             :                     CPLSPrintf(CPL_FRMT_GIB,
    5964         122 :                                static_cast<GIntBig>(nFeatureIdxCur)));
    5965             : 
    5966             :             // If there might be more than one record batch, it is more
    5967             :             // prudent to clone the array before modifying it.
    5968         130 :             if (nFeatureIdxCur > 0 || !TestCapability(OLCFastFeatureCount) ||
    5969           0 :                 out_array->length < GetFeatureCount(false))
    5970             :             {
    5971             :                 struct ArrowArray new_array;
    5972         130 :                 if (!OGRCloneArrowArray(&m_sCachedSchema, out_array,
    5973             :                                         &new_array))
    5974             :                 {
    5975           0 :                     if (out_array->release)
    5976           0 :                         out_array->release(out_array);
    5977           0 :                     memset(out_array, 0, sizeof(*out_array));
    5978           0 :                     return ENOMEM;
    5979             :                 }
    5980         130 :                 if (out_array->release)
    5981         130 :                     out_array->release(out_array);
    5982         130 :                 memcpy(out_array, &new_array, sizeof(new_array));
    5983             :             }
    5984             : 
    5985         130 :             PostFilterArrowArray(&m_sCachedSchema, out_array,
    5986         130 :                                  aosOptions.List());
    5987         130 :             if (out_array->length == 0)
    5988             :             {
    5989          18 :                 if (out_array->release)
    5990          18 :                     out_array->release(out_array);
    5991          18 :                 memset(out_array, 0, sizeof(*out_array));
    5992             :                 // If there are no records after filtering, start again
    5993             :                 // with a new batch
    5994          18 :                 continue;
    5995             :             }
    5996             :         }
    5997             : 
    5998         204 :         break;
    5999          18 :     }
    6000             : 
    6001         204 :     return 0;
    6002             : }
    6003             : 
    6004             : /************************************************************************/
    6005             : /*                    OGRArrowLayerAppendBuffer                         */
    6006             : /************************************************************************/
    6007             : 
    6008             : class OGRArrowLayerAppendBuffer : public OGRAppendBuffer
    6009             : {
    6010             :   public:
    6011          18 :     OGRArrowLayerAppendBuffer(struct ArrowArray *targetArrayIn,
    6012             :                               size_t nInitialCapacityIn)
    6013          18 :         : m_psTargetArray(targetArrayIn)
    6014             :     {
    6015          18 :         m_nCapacity = nInitialCapacityIn;
    6016          18 :         m_pRawBuffer = const_cast<void *>(m_psTargetArray->buffers[2]);
    6017          18 :     }
    6018             : 
    6019             :   protected:
    6020           4 :     bool Grow(size_t nItemSize) override
    6021             :     {
    6022           4 :         constexpr uint32_t MAX_SIZE_SINT32 =
    6023             :             static_cast<uint32_t>(std::numeric_limits<int32_t>::max());
    6024           4 :         if (nItemSize > MAX_SIZE_SINT32 - m_nSize)
    6025             :         {
    6026           0 :             CPLError(CE_Failure, CPLE_AppDefined, "Too large WKT content");
    6027           0 :             return false;
    6028             :         }
    6029           4 :         size_t nNewCapacity = m_nSize + nItemSize;
    6030           4 :         CPLAssert(m_nCapacity <= MAX_SIZE_SINT32);
    6031             :         const size_t nDoubleCapacity =
    6032           4 :             std::min<size_t>(MAX_SIZE_SINT32, 2 * m_nCapacity);
    6033           4 :         if (nNewCapacity < nDoubleCapacity)
    6034           4 :             nNewCapacity = nDoubleCapacity;
    6035           4 :         CPLAssert(nNewCapacity <= MAX_SIZE_SINT32);
    6036           4 :         void *newBuffer = VSI_MALLOC_ALIGNED_AUTO_VERBOSE(nNewCapacity);
    6037           4 :         if (newBuffer == nullptr)
    6038             :         {
    6039           0 :             return false;
    6040             :         }
    6041           4 :         m_nCapacity = nNewCapacity;
    6042           4 :         memcpy(newBuffer, m_pRawBuffer, m_nSize);
    6043           4 :         VSIFreeAligned(m_pRawBuffer);
    6044           4 :         m_pRawBuffer = newBuffer;
    6045           4 :         m_psTargetArray->buffers[2] = m_pRawBuffer;
    6046           4 :         return true;
    6047             :     }
    6048             : 
    6049             :   private:
    6050             :     struct ArrowArray *m_psTargetArray;
    6051             : 
    6052             :     OGRArrowLayerAppendBuffer(const OGRArrowLayerAppendBuffer &) = delete;
    6053             :     OGRArrowLayerAppendBuffer &
    6054             :     operator=(const OGRArrowLayerAppendBuffer &) = delete;
    6055             : };
    6056             : 
    6057             : /************************************************************************/
    6058             : /*                    CreateWKBArrayFromWKTArray()                      */
    6059             : /************************************************************************/
    6060             : 
    6061             : template <typename SourceOffset>
    6062             : inline struct ArrowArray *
    6063          18 : OGRArrowLayer::CreateWKBArrayFromWKTArray(const struct ArrowArray *sourceArray)
    6064             : {
    6065          18 :     CPLAssert(sourceArray->n_buffers == 3);
    6066          18 :     CPLAssert(sourceArray->buffers[1] != nullptr);
    6067          18 :     CPLAssert(sourceArray->buffers[2] != nullptr);
    6068             : 
    6069          18 :     const size_t nLength = static_cast<size_t>(sourceArray->length);
    6070             :     auto targetArray = static_cast<struct ArrowArray *>(
    6071          18 :         CPLCalloc(1, sizeof(struct ArrowArray)));
    6072          18 :     targetArray->release = OGRLayer::ReleaseArray;
    6073          18 :     targetArray->length = nLength;
    6074             : 
    6075          18 :     targetArray->n_buffers = 3;
    6076          18 :     targetArray->buffers =
    6077          18 :         static_cast<const void **>(CPLCalloc(3, sizeof(void *)));
    6078             : 
    6079             :     // Allocate validity map buffer if needed
    6080          18 :     const auto sourceNull =
    6081          18 :         static_cast<const uint8_t *>(sourceArray->buffers[0]);
    6082          18 :     const size_t nOffset = static_cast<size_t>(sourceArray->offset);
    6083          18 :     uint8_t *targetNull = nullptr;
    6084          18 :     if (sourceArray->null_count && sourceNull)
    6085             :     {
    6086           6 :         targetArray->buffers[0] =
    6087           3 :             VSI_MALLOC_ALIGNED_AUTO_VERBOSE((nLength + 7) / 8);
    6088           3 :         if (targetArray->buffers[0])
    6089             :         {
    6090           3 :             targetArray->null_count = sourceArray->null_count;
    6091           3 :             targetNull = static_cast<uint8_t *>(
    6092           3 :                 const_cast<void *>(targetArray->buffers[0]));
    6093           3 :             if (nOffset == 0)
    6094             :             {
    6095           3 :                 memcpy(targetNull, sourceNull, (nLength + 7) / 8);
    6096             :             }
    6097             :             else
    6098             :             {
    6099           0 :                 memset(targetNull, 0, (nLength + 7) / 8);
    6100           0 :                 for (size_t i = 0; i < nLength; ++i)
    6101             :                 {
    6102           0 :                     if ((sourceNull[(i + nOffset) / 8] >> ((i + nOffset) % 8)) &
    6103             :                         1)
    6104           0 :                         targetNull[i / 8] |= (1 << (i % 8));
    6105             :                 }
    6106             :             }
    6107             :         }
    6108             :     }
    6109             : 
    6110             :     // Allocate offset buffer
    6111          36 :     targetArray->buffers[1] =
    6112          18 :         VSI_MALLOC_ALIGNED_AUTO_VERBOSE(sizeof(uint32_t) * (1 + nLength));
    6113             : 
    6114             :     // Allocate data (WKB) buffer
    6115          18 :     constexpr size_t DEFAULT_WKB_SIZE = 100;
    6116          18 :     uint32_t nInitialCapacity = static_cast<uint32_t>(std::min<size_t>(
    6117          18 :         std::numeric_limits<int32_t>::max(), DEFAULT_WKB_SIZE * nLength));
    6118          18 :     targetArray->buffers[2] = VSI_MALLOC_ALIGNED_AUTO_VERBOSE(nInitialCapacity);
    6119             : 
    6120             :     // Check buffers have been allocated
    6121          18 :     if ((sourceArray->null_count && sourceNull && !targetNull) ||
    6122          18 :         targetArray->buffers[1] == nullptr ||
    6123          18 :         targetArray->buffers[2] == nullptr)
    6124             :     {
    6125           0 :         targetArray->release(targetArray);
    6126           0 :         return nullptr;
    6127             :     }
    6128             : 
    6129          36 :     OGRArrowLayerAppendBuffer oOGRAppendBuffer(targetArray, nInitialCapacity);
    6130          18 :     OGRWKTToWKBTranslator oTranslator(oOGRAppendBuffer);
    6131             : 
    6132          18 :     const auto sourceOffsets =
    6133          18 :         static_cast<const SourceOffset *>(sourceArray->buffers[1]) + nOffset;
    6134          18 :     auto sourceBytes =
    6135          18 :         static_cast<char *>(const_cast<void *>(sourceArray->buffers[2]));
    6136          18 :     auto targetOffsets =
    6137          18 :         static_cast<uint32_t *>(const_cast<void *>(targetArray->buffers[1]));
    6138         159 :     for (size_t i = 0; i < nLength; ++i)
    6139             :     {
    6140         141 :         targetOffsets[i] = static_cast<uint32_t>(oOGRAppendBuffer.GetSize());
    6141             : 
    6142         141 :         if (targetNull && ((targetNull[i / 8] >> (i % 8)) & 1) == 0)
    6143             :         {
    6144           3 :             continue;
    6145             :         }
    6146             : 
    6147         276 :         const size_t nWKBSize = oTranslator.TranslateWKT(
    6148         138 :             sourceBytes + sourceOffsets[i],
    6149         138 :             static_cast<size_t>(sourceOffsets[i + 1] - sourceOffsets[i]),
    6150         138 :             sourceOffsets[i + 1] < sourceOffsets[nLength]);
    6151         138 :         if (nWKBSize == static_cast<size_t>(-1))
    6152             :         {
    6153           0 :             targetArray->release(targetArray);
    6154           0 :             return nullptr;
    6155             :         }
    6156             :     }
    6157          18 :     targetOffsets[nLength] = static_cast<uint32_t>(oOGRAppendBuffer.GetSize());
    6158             : 
    6159          18 :     return targetArray;
    6160             : }
    6161             : 
    6162             : /************************************************************************/
    6163             : /*                         TestCapability()                             */
    6164             : /************************************************************************/
    6165             : 
    6166         873 : inline int OGRArrowLayer::TestCapability(const char *pszCap)
    6167             : {
    6168             : 
    6169         873 :     if (EQUAL(pszCap, OLCStringsAsUTF8))
    6170         488 :         return true;
    6171             : 
    6172         566 :     else if (EQUAL(pszCap, OLCFastGetArrowStream) &&
    6173         181 :              !UseRecordBatchBaseImplementation())
    6174             :     {
    6175         181 :         return true;
    6176             :     }
    6177             : 
    6178         204 :     if (EQUAL(pszCap, OLCFastGetExtent))
    6179             :     {
    6180          29 :         OGREnvelope sEnvelope;
    6181          44 :         for (int i = 0; i < m_poFeatureDefn->GetGeomFieldCount(); i++)
    6182             :         {
    6183          29 :             if (!FastGetExtent(i, &sEnvelope))
    6184          14 :                 return false;
    6185             :         }
    6186          15 :         return true;
    6187             :     }
    6188             : 
    6189         175 :     if (EQUAL(pszCap, OLCFastGetExtent3D))
    6190             :     {
    6191          19 :         OGREnvelope3D sEnvelope;
    6192          20 :         for (int i = 0; i < m_poFeatureDefn->GetGeomFieldCount(); i++)
    6193             :         {
    6194          19 :             if (!FastGetExtent3D(i, &sEnvelope))
    6195          18 :                 return false;
    6196             :         }
    6197           1 :         return true;
    6198             :     }
    6199             : 
    6200         156 :     return false;
    6201             : }
    6202             : 
    6203             : #endif /* OGARROWLAYER_HPP_INCLUDED */

Generated by: LCOV version 1.14