Coverage Report

Created: 2026-02-14 06:52

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/gdal/ogr/ogrsf_frmts/generic/ogrwarpedlayer.cpp
Line
Count
Source
1
/******************************************************************************
2
 *
3
 * Project:  OpenGIS Simple Features Reference Implementation
4
 * Purpose:  Implements OGRWarpedLayer class
5
 * Author:   Even Rouault, even dot rouault at spatialys.com
6
 *
7
 ******************************************************************************
8
 * Copyright (c) 2012-2014, Even Rouault <even dot rouault at spatialys.com>
9
 *
10
 * SPDX-License-Identifier: MIT
11
 ****************************************************************************/
12
13
#ifndef DOXYGEN_SKIP
14
15
#include <cmath>
16
17
#include "ogrwarpedlayer.h"
18
19
/************************************************************************/
20
/*                           OGRWarpedLayer()                           */
21
/************************************************************************/
22
23
OGRWarpedLayer::OGRWarpedLayer(
24
    OGRLayer *poDecoratedLayer, int iGeomField, int bTakeLayerOwnership,
25
    std::unique_ptr<OGRCoordinateTransformation> poCT,
26
    std::unique_ptr<OGRCoordinateTransformation> poReversedCT)
27
0
    : OGRLayerDecorator(poDecoratedLayer, bTakeLayerOwnership),
28
0
      m_poFeatureDefn(poDecoratedLayer->GetLayerDefn()->Clone()),
29
0
      m_iGeomField(iGeomField), m_poCT(std::move(poCT)),
30
0
      m_poReversedCT(std::move(poReversedCT)),
31
0
      m_poSRS(const_cast<OGRSpatialReference *>(m_poCT->GetTargetCS()))
32
0
{
33
0
    SetDescription(poDecoratedLayer->GetDescription());
34
35
0
    m_poFeatureDefn->Reference();
36
0
    if (m_poFeatureDefn->GetGeomFieldCount() > 0)
37
0
        m_poFeatureDefn->GetGeomFieldDefn(m_iGeomField)->SetSpatialRef(m_poSRS);
38
39
0
    if (m_poSRS != nullptr)
40
0
    {
41
0
        m_poSRS->Reference();
42
0
    }
43
0
}
Unexecuted instantiation: OGRWarpedLayer::OGRWarpedLayer(OGRLayer*, int, int, std::__1::unique_ptr<OGRCoordinateTransformation, std::__1::default_delete<OGRCoordinateTransformation> >, std::__1::unique_ptr<OGRCoordinateTransformation, std::__1::default_delete<OGRCoordinateTransformation> >)
Unexecuted instantiation: OGRWarpedLayer::OGRWarpedLayer(OGRLayer*, int, int, std::__1::unique_ptr<OGRCoordinateTransformation, std::__1::default_delete<OGRCoordinateTransformation> >, std::__1::unique_ptr<OGRCoordinateTransformation, std::__1::default_delete<OGRCoordinateTransformation> >)
44
45
/************************************************************************/
46
/*                          ~OGRWarpedLayer()                           */
47
/************************************************************************/
48
49
OGRWarpedLayer::~OGRWarpedLayer()
50
0
{
51
0
    if (m_poFeatureDefn != nullptr)
52
0
        m_poFeatureDefn->Release();
53
0
    if (m_poSRS != nullptr)
54
0
        m_poSRS->Release();
55
0
}
56
57
/************************************************************************/
58
/*                         ISetSpatialFilter()                          */
59
/************************************************************************/
60
61
OGRErr OGRWarpedLayer::ISetSpatialFilter(int iGeomField,
62
                                         const OGRGeometry *poGeom)
63
0
{
64
65
0
    m_iGeomFieldFilter = iGeomField;
66
0
    if (InstallFilter(poGeom))
67
0
        ResetReading();
68
69
0
    if (m_iGeomFieldFilter == m_iGeomField)
70
0
    {
71
0
        if (poGeom == nullptr || m_poReversedCT == nullptr)
72
0
        {
73
0
            return m_poDecoratedLayer->SetSpatialFilter(m_iGeomFieldFilter,
74
0
                                                        nullptr);
75
0
        }
76
0
        else
77
0
        {
78
0
            OGREnvelope sEnvelope;
79
0
            poGeom->getEnvelope(&sEnvelope);
80
0
            if (std::isinf(sEnvelope.MinX) && std::isinf(sEnvelope.MinY) &&
81
0
                std::isinf(sEnvelope.MaxX) && std::isinf(sEnvelope.MaxY))
82
0
            {
83
0
                return m_poDecoratedLayer->SetSpatialFilterRect(
84
0
                    m_iGeomFieldFilter, sEnvelope.MinX, sEnvelope.MinY,
85
0
                    sEnvelope.MaxX, sEnvelope.MaxY);
86
0
            }
87
0
            else if (ReprojectEnvelope(&sEnvelope, m_poReversedCT.get()))
88
0
            {
89
0
                return m_poDecoratedLayer->SetSpatialFilterRect(
90
0
                    m_iGeomFieldFilter, sEnvelope.MinX, sEnvelope.MinY,
91
0
                    sEnvelope.MaxX, sEnvelope.MaxY);
92
0
            }
93
0
            else
94
0
            {
95
0
                return m_poDecoratedLayer->SetSpatialFilter(m_iGeomFieldFilter,
96
0
                                                            nullptr);
97
0
            }
98
0
        }
99
0
    }
100
0
    else
101
0
    {
102
0
        return m_poDecoratedLayer->SetSpatialFilter(m_iGeomFieldFilter, poGeom);
103
0
    }
104
0
}
105
106
/************************************************************************/
107
/*                          TranslateFeature()                          */
108
/************************************************************************/
109
110
void OGRWarpedLayer::TranslateFeature(
111
    std::unique_ptr<OGRFeature> poSrcFeature,
112
    std::vector<std::unique_ptr<OGRFeature>> &apoOutFeatures)
113
0
{
114
0
    apoOutFeatures.push_back(
115
0
        SrcFeatureToWarpedFeature(std::move(poSrcFeature)));
116
0
}
117
118
/************************************************************************/
119
/*                     SrcFeatureToWarpedFeature()                      */
120
/************************************************************************/
121
122
std::unique_ptr<OGRFeature>
123
OGRWarpedLayer::SrcFeatureToWarpedFeature(std::unique_ptr<OGRFeature> poFeature)
124
0
{
125
    // This is safe to do here as they have matching attribute and geometry
126
    // fields
127
0
    OGRLayer *poThisLayer = this;
128
0
    poFeature->SetFDefnUnsafe(poThisLayer->GetLayerDefn());
129
130
0
    OGRGeometry *poGeom = poFeature->GetGeomFieldRef(m_iGeomField);
131
0
    if (poGeom)
132
0
    {
133
0
        auto poNewGeom = OGRGeometryFactory::transformWithOptions(
134
0
            poGeom, m_poCT.get(), nullptr, m_transformCacheForward);
135
0
        poFeature->SetGeomFieldDirectly(m_iGeomField, poNewGeom);
136
0
    }
137
138
0
    return poFeature;
139
0
}
140
141
/************************************************************************/
142
/*                     WarpedFeatureToSrcFeature()                      */
143
/************************************************************************/
144
145
std::unique_ptr<OGRFeature>
146
OGRWarpedLayer::WarpedFeatureToSrcFeature(std::unique_ptr<OGRFeature> poFeature)
147
0
{
148
    // This is safe to do here as they have matching attribute and geometry
149
    // fields
150
0
    poFeature->SetFDefnUnsafe(m_poDecoratedLayer->GetLayerDefn());
151
152
0
    OGRGeometry *poGeom = poFeature->GetGeomFieldRef(m_iGeomField);
153
0
    if (poGeom)
154
0
    {
155
0
        if (!m_poReversedCT)
156
0
            return nullptr;
157
0
        auto poNewGeom = OGRGeometryFactory::transformWithOptions(
158
0
            poGeom, m_poReversedCT.get(), nullptr, m_transformCacheReverse);
159
0
        if (!poNewGeom)
160
0
            return nullptr;
161
0
        poFeature->SetGeomFieldDirectly(m_iGeomField, poNewGeom);
162
0
    }
163
164
0
    return poFeature;
165
0
}
166
167
/************************************************************************/
168
/*                           GetNextFeature()                           */
169
/************************************************************************/
170
171
OGRFeature *OGRWarpedLayer::GetNextFeature()
172
0
{
173
0
    while (true)
174
0
    {
175
0
        auto poFeature =
176
0
            std::unique_ptr<OGRFeature>(m_poDecoratedLayer->GetNextFeature());
177
0
        if (!poFeature)
178
0
            return nullptr;
179
180
0
        auto poFeatureNew = SrcFeatureToWarpedFeature(std::move(poFeature));
181
0
        const OGRGeometry *poGeom = poFeatureNew->GetGeomFieldRef(m_iGeomField);
182
0
        if (m_poFilterGeom != nullptr && !FilterGeometry(poGeom))
183
0
        {
184
0
            continue;
185
0
        }
186
187
0
        return poFeatureNew.release();
188
0
    }
189
0
}
190
191
/************************************************************************/
192
/*                             GetFeature()                             */
193
/************************************************************************/
194
195
OGRFeature *OGRWarpedLayer::GetFeature(GIntBig nFID)
196
0
{
197
0
    auto poFeature =
198
0
        std::unique_ptr<OGRFeature>(m_poDecoratedLayer->GetFeature(nFID));
199
0
    if (poFeature)
200
0
    {
201
0
        poFeature = SrcFeatureToWarpedFeature(std::move(poFeature));
202
0
    }
203
0
    return poFeature.release();
204
0
}
205
206
/************************************************************************/
207
/*                            ISetFeature()                             */
208
/************************************************************************/
209
210
OGRErr OGRWarpedLayer::ISetFeature(OGRFeature *poFeature)
211
0
{
212
0
    auto poFeatureNew = WarpedFeatureToSrcFeature(
213
0
        std::unique_ptr<OGRFeature>(poFeature->Clone()));
214
0
    if (!poFeatureNew)
215
0
        return OGRERR_FAILURE;
216
217
0
    return m_poDecoratedLayer->SetFeature(poFeatureNew.get());
218
0
}
219
220
/************************************************************************/
221
/*                         ISetFeatureUniqPtr()                         */
222
/************************************************************************/
223
224
OGRErr OGRWarpedLayer::ISetFeatureUniqPtr(std::unique_ptr<OGRFeature> poFeature)
225
0
{
226
0
    auto poFeatureNew = WarpedFeatureToSrcFeature(std::move(poFeature));
227
0
    if (!poFeatureNew)
228
0
        return OGRERR_FAILURE;
229
230
0
    return m_poDecoratedLayer->SetFeature(std::move(poFeatureNew));
231
0
}
232
233
/************************************************************************/
234
/*                           ICreateFeature()                           */
235
/************************************************************************/
236
237
OGRErr OGRWarpedLayer::ICreateFeature(OGRFeature *poFeature)
238
0
{
239
0
    auto poFeatureNew = WarpedFeatureToSrcFeature(
240
0
        std::unique_ptr<OGRFeature>(poFeature->Clone()));
241
0
    if (!poFeatureNew)
242
0
        return OGRERR_FAILURE;
243
244
0
    return m_poDecoratedLayer->CreateFeature(poFeatureNew.get());
245
0
}
246
247
/************************************************************************/
248
/*                       ICreateFeatureUniqPtr()                        */
249
/************************************************************************/
250
251
OGRErr
252
OGRWarpedLayer::ICreateFeatureUniqPtr(std::unique_ptr<OGRFeature> poFeature,
253
                                      GIntBig *pnFID)
254
0
{
255
0
    auto poFeatureNew = WarpedFeatureToSrcFeature(std::move(poFeature));
256
0
    if (!poFeatureNew)
257
0
        return OGRERR_FAILURE;
258
259
0
    return m_poDecoratedLayer->CreateFeature(std::move(poFeatureNew), pnFID);
260
0
}
261
262
/************************************************************************/
263
/*                           IUpsertFeature()                           */
264
/************************************************************************/
265
266
OGRErr OGRWarpedLayer::IUpsertFeature(OGRFeature *poFeature)
267
0
{
268
0
    auto poFeatureNew = WarpedFeatureToSrcFeature(
269
0
        std::unique_ptr<OGRFeature>(poFeature->Clone()));
270
0
    if (poFeatureNew == nullptr)
271
0
        return OGRERR_FAILURE;
272
273
0
    return m_poDecoratedLayer->UpsertFeature(poFeatureNew.get());
274
0
}
275
276
/************************************************************************/
277
/*                           IUpdateFeature()                           */
278
/************************************************************************/
279
280
OGRErr OGRWarpedLayer::IUpdateFeature(OGRFeature *poFeature,
281
                                      int nUpdatedFieldsCount,
282
                                      const int *panUpdatedFieldsIdx,
283
                                      int nUpdatedGeomFieldsCount,
284
                                      const int *panUpdatedGeomFieldsIdx,
285
                                      bool bUpdateStyleString)
286
0
{
287
0
    auto poFeatureNew = WarpedFeatureToSrcFeature(
288
0
        std::unique_ptr<OGRFeature>(poFeature->Clone()));
289
0
    if (!poFeatureNew)
290
0
        return OGRERR_FAILURE;
291
292
0
    return m_poDecoratedLayer->UpdateFeature(
293
0
        poFeatureNew.get(), nUpdatedFieldsCount, panUpdatedFieldsIdx,
294
0
        nUpdatedGeomFieldsCount, panUpdatedGeomFieldsIdx, bUpdateStyleString);
295
0
}
296
297
/************************************************************************/
298
/*                            GetLayerDefn()                            */
299
/************************************************************************/
300
301
const OGRFeatureDefn *OGRWarpedLayer::GetLayerDefn() const
302
0
{
303
0
    return m_poFeatureDefn;
304
0
}
305
306
/************************************************************************/
307
/*                           GetSpatialRef()                            */
308
/************************************************************************/
309
310
const OGRSpatialReference *OGRWarpedLayer::GetSpatialRef() const
311
0
{
312
0
    if (m_iGeomField == 0)
313
0
        return m_poSRS;
314
0
    else
315
0
        return OGRLayer::GetSpatialRef();
316
0
}
317
318
/************************************************************************/
319
/*                          GetFeatureCount()                           */
320
/************************************************************************/
321
322
GIntBig OGRWarpedLayer::GetFeatureCount(int bForce)
323
0
{
324
0
    if (m_poFilterGeom == nullptr)
325
0
        return m_poDecoratedLayer->GetFeatureCount(bForce);
326
327
0
    return OGRLayer::GetFeatureCount(bForce);
328
0
}
329
330
/************************************************************************/
331
/*                             IGetExtent()                             */
332
/************************************************************************/
333
334
OGRErr OGRWarpedLayer::IGetExtent(int iGeomField, OGREnvelope *psExtent,
335
                                  bool bForce)
336
0
{
337
0
    if (iGeomField == m_iGeomField)
338
0
    {
339
0
        if (sStaticEnvelope.IsInit())
340
0
        {
341
0
            *psExtent = sStaticEnvelope;
342
0
            return OGRERR_NONE;
343
0
        }
344
345
0
        OGREnvelope sExtent;
346
0
        OGRErr eErr =
347
0
            m_poDecoratedLayer->GetExtent(m_iGeomField, &sExtent, bForce);
348
0
        if (eErr != OGRERR_NONE)
349
0
            return eErr;
350
351
0
        if (ReprojectEnvelope(&sExtent, m_poCT.get()))
352
0
        {
353
0
            *psExtent = sExtent;
354
0
            return OGRERR_NONE;
355
0
        }
356
0
        else
357
0
            return OGRERR_FAILURE;
358
0
    }
359
0
    else
360
0
        return m_poDecoratedLayer->GetExtent(iGeomField, psExtent, bForce);
361
0
}
362
363
/************************************************************************/
364
/*                   TransformAndUpdateBBAndReturnX()                   */
365
/************************************************************************/
366
367
static double TransformAndUpdateBBAndReturnX(OGRCoordinateTransformation *poCT,
368
                                             double dfX, double dfY,
369
                                             double &dfMinX, double &dfMinY,
370
                                             double &dfMaxX, double &dfMaxY)
371
0
{
372
0
    int bSuccess = FALSE;
373
0
    poCT->Transform(1, &dfX, &dfY, nullptr, nullptr, &bSuccess);
374
0
    if (bSuccess)
375
0
    {
376
0
        if (dfX < dfMinX)
377
0
            dfMinX = dfX;
378
0
        if (dfY < dfMinY)
379
0
            dfMinY = dfY;
380
0
        if (dfX > dfMaxX)
381
0
            dfMaxX = dfX;
382
0
        if (dfY > dfMaxY)
383
0
            dfMaxY = dfY;
384
0
        return dfX;
385
0
    }
386
387
0
    return 0.0;
388
0
}
389
390
/************************************************************************/
391
/*                         FindXDiscontinuity()                         */
392
/************************************************************************/
393
394
static void FindXDiscontinuity(OGRCoordinateTransformation *poCT, double dfX1,
395
                               double dfX2, double dfY, double &dfMinX,
396
                               double &dfMinY, double &dfMaxX, double &dfMaxY,
397
                               int nRecLevel = 0)
398
0
{
399
0
    double dfXMid = (dfX1 + dfX2) / 2;
400
401
0
    double dfWrkX1 = TransformAndUpdateBBAndReturnX(poCT, dfX1, dfY, dfMinX,
402
0
                                                    dfMinY, dfMaxX, dfMaxY);
403
0
    double dfWrkXMid = TransformAndUpdateBBAndReturnX(poCT, dfXMid, dfY, dfMinX,
404
0
                                                      dfMinY, dfMaxX, dfMaxY);
405
0
    double dfWrkX2 = TransformAndUpdateBBAndReturnX(poCT, dfX2, dfY, dfMinX,
406
0
                                                    dfMinY, dfMaxX, dfMaxY);
407
408
0
    double dfDX1 = dfWrkXMid - dfWrkX1;
409
0
    double dfDX2 = dfWrkX2 - dfWrkXMid;
410
411
0
    if (dfDX1 * dfDX2 < 0 && nRecLevel < 30)
412
0
    {
413
0
        FindXDiscontinuity(poCT, dfX1, dfXMid, dfY, dfMinX, dfMinY, dfMaxX,
414
0
                           dfMaxY, nRecLevel + 1);
415
0
        FindXDiscontinuity(poCT, dfXMid, dfX2, dfY, dfMinX, dfMinY, dfMaxX,
416
0
                           dfMaxY, nRecLevel + 1);
417
0
    }
418
0
}
419
420
/************************************************************************/
421
/*                         ReprojectEnvelope()                          */
422
/************************************************************************/
423
424
int OGRWarpedLayer::ReprojectEnvelope(OGREnvelope *psEnvelope,
425
                                      OGRCoordinateTransformation *poCT)
426
0
{
427
0
    const int NSTEP = 20;
428
0
    double dfXStep = (psEnvelope->MaxX - psEnvelope->MinX) / NSTEP;
429
0
    double dfYStep = (psEnvelope->MaxY - psEnvelope->MinY) / NSTEP;
430
431
0
    double *padfX = static_cast<double *>(
432
0
        VSI_MALLOC_VERBOSE((NSTEP + 1) * (NSTEP + 1) * sizeof(double)));
433
0
    double *padfY = static_cast<double *>(
434
0
        VSI_MALLOC_VERBOSE((NSTEP + 1) * (NSTEP + 1) * sizeof(double)));
435
0
    int *pabSuccess = static_cast<int *>(
436
0
        VSI_MALLOC_VERBOSE((NSTEP + 1) * (NSTEP + 1) * sizeof(int)));
437
0
    if (padfX == nullptr || padfY == nullptr || pabSuccess == nullptr)
438
0
    {
439
0
        VSIFree(padfX);
440
0
        VSIFree(padfY);
441
0
        VSIFree(pabSuccess);
442
0
        return FALSE;
443
0
    }
444
445
0
    for (int j = 0; j <= NSTEP; j++)
446
0
    {
447
0
        for (int i = 0; i <= NSTEP; i++)
448
0
        {
449
0
            padfX[j * (NSTEP + 1) + i] = psEnvelope->MinX + i * dfXStep;
450
0
            padfY[j * (NSTEP + 1) + i] = psEnvelope->MinY + j * dfYStep;
451
0
        }
452
0
    }
453
454
0
    int bRet = FALSE;
455
456
0
    if (poCT->Transform((NSTEP + 1) * (NSTEP + 1), padfX, padfY, nullptr,
457
0
                        nullptr, pabSuccess))
458
0
    {
459
0
        double dfMinX = 0.0;
460
0
        double dfMinY = 0.0;
461
0
        double dfMaxX = 0.0;
462
0
        double dfMaxY = 0.0;
463
0
        int bSet = FALSE;
464
0
        for (int j = 0; j <= NSTEP; j++)
465
0
        {
466
0
            double dfXOld = 0.0;
467
0
            double dfDXOld = 0.0;
468
0
            int iOld = -1;
469
0
            int iOldOld = -1;
470
0
            for (int i = 0; i <= NSTEP; i++)
471
0
            {
472
0
                if (pabSuccess[j * (NSTEP + 1) + i])
473
0
                {
474
0
                    double dfX = padfX[j * (NSTEP + 1) + i];
475
0
                    double dfY = padfY[j * (NSTEP + 1) + i];
476
477
0
                    if (!bSet)
478
0
                    {
479
0
                        dfMinX = dfX;
480
0
                        dfMaxX = dfX;
481
0
                        dfMinY = dfY;
482
0
                        dfMaxY = dfY;
483
0
                        bSet = TRUE;
484
0
                    }
485
0
                    else
486
0
                    {
487
0
                        if (dfX < dfMinX)
488
0
                            dfMinX = dfX;
489
0
                        if (dfY < dfMinY)
490
0
                            dfMinY = dfY;
491
0
                        if (dfX > dfMaxX)
492
0
                            dfMaxX = dfX;
493
0
                        if (dfY > dfMaxY)
494
0
                            dfMaxY = dfY;
495
0
                    }
496
497
0
                    if (iOld >= 0)
498
0
                    {
499
0
                        double dfDXNew = dfX - dfXOld;
500
0
                        if (iOldOld >= 0 && dfDXNew * dfDXOld < 0)
501
0
                        {
502
0
                            FindXDiscontinuity(
503
0
                                poCT, psEnvelope->MinX + iOldOld * dfXStep,
504
0
                                psEnvelope->MinX + i * dfXStep,
505
0
                                psEnvelope->MinY + j * dfYStep, dfMinX, dfMinY,
506
0
                                dfMaxX, dfMaxY);
507
0
                        }
508
0
                        dfDXOld = dfDXNew;
509
0
                    }
510
511
0
                    dfXOld = dfX;
512
0
                    iOldOld = iOld;
513
0
                    iOld = i;
514
0
                }
515
0
            }
516
0
        }
517
0
        if (bSet)
518
0
        {
519
0
            psEnvelope->MinX = dfMinX;
520
0
            psEnvelope->MinY = dfMinY;
521
0
            psEnvelope->MaxX = dfMaxX;
522
0
            psEnvelope->MaxY = dfMaxY;
523
0
            bRet = TRUE;
524
0
        }
525
0
    }
526
527
0
    VSIFree(padfX);
528
0
    VSIFree(padfY);
529
0
    VSIFree(pabSuccess);
530
531
0
    return bRet;
532
0
}
533
534
/************************************************************************/
535
/*                           TestCapability()                           */
536
/************************************************************************/
537
538
int OGRWarpedLayer::TestCapability(const char *pszCapability) const
539
0
{
540
0
    if (EQUAL(pszCapability, OLCFastGetExtent) && sStaticEnvelope.IsInit())
541
0
        return TRUE;
542
543
0
    int bVal = m_poDecoratedLayer->TestCapability(pszCapability);
544
545
0
    if (EQUAL(pszCapability, OLCFastGetArrowStream))
546
0
        return false;
547
548
0
    if (EQUAL(pszCapability, OLCFastSpatialFilter) ||
549
0
        EQUAL(pszCapability, OLCRandomWrite) ||
550
0
        EQUAL(pszCapability, OLCSequentialWrite))
551
0
    {
552
0
        if (bVal)
553
0
            bVal = m_poReversedCT != nullptr;
554
0
    }
555
0
    else if (EQUAL(pszCapability, OLCFastFeatureCount))
556
0
    {
557
0
        if (bVal)
558
0
            bVal = m_poFilterGeom == nullptr;
559
0
    }
560
561
0
    return bVal;
562
0
}
563
564
/************************************************************************/
565
/*                             SetExtent()                              */
566
/************************************************************************/
567
568
void OGRWarpedLayer::SetExtent(double dfXMin, double dfYMin, double dfXMax,
569
                               double dfYMax)
570
0
{
571
0
    sStaticEnvelope.MinX = dfXMin;
572
0
    sStaticEnvelope.MinY = dfYMin;
573
0
    sStaticEnvelope.MaxX = dfXMax;
574
0
    sStaticEnvelope.MaxY = dfYMax;
575
0
}
576
577
/************************************************************************/
578
/*                           GetArrowStream()                           */
579
/************************************************************************/
580
581
bool OGRWarpedLayer::GetArrowStream(struct ArrowArrayStream *out_stream,
582
                                    CSLConstList papszOptions)
583
0
{
584
0
    return OGRLayer::GetArrowStream(out_stream, papszOptions);
585
0
}
586
587
#endif /* #ifndef DOXYGEN_SKIP */