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 */
|