Coverage Report

Created: 2026-02-14 06:52

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/gdal/port/cpl_vsil_stdin.cpp
Line
Count
Source
1
/**********************************************************************
2
 *
3
 * Project:  CPL - Common Portability Library
4
 * Purpose:  Implement VSI large file api for stdin
5
 * Author:   Even Rouault, <even dot rouault at spatialys.com>
6
 *
7
 **********************************************************************
8
 * Copyright (c) 2010-2012, Even Rouault <even dot rouault at spatialys.com>
9
 *
10
 * SPDX-License-Identifier: MIT
11
 ****************************************************************************/
12
13
//! @cond Doxygen_Suppress
14
15
#include "cpl_port.h"
16
#include "cpl_vsi.h"
17
18
#include <cstddef>
19
#include <cstdio>
20
#include <cstdlib>
21
#include <cstring>
22
#include <fcntl.h>
23
24
#include <algorithm>
25
#include <limits>
26
27
#include "cpl_conv.h"
28
#include "cpl_error.h"
29
#include "cpl_vsi_virtual.h"
30
31
#ifdef _WIN32
32
#include <io.h>
33
#endif
34
35
static std::string gosStdinFilename{};
36
static FILE *gStdinFile = stdin;
37
static GByte *gpabyBuffer = nullptr;
38
static size_t gnBufferLimit = 0;  // maximum that can be allocated
39
static size_t gnBufferAlloc = 0;  // current allocation
40
static size_t gnBufferLen = 0;    // number of valid bytes in gpabyBuffer
41
static uint64_t gnRealPos = 0;    // current offset on stdin
42
static bool gbHasSoughtToEnd = false;
43
static bool gbHasErrored = false;
44
static uint64_t gnFileSize = 0;
45
46
/************************************************************************/
47
/*                            VSIStdinInit()                            */
48
/************************************************************************/
49
50
static void VSIStdinInit()
51
22.3k
{
52
22.3k
    if (gpabyBuffer == nullptr)
53
22.3k
    {
54
#ifdef _WIN32
55
        setmode(fileno(stdin), O_BINARY);
56
#endif
57
22.3k
        constexpr size_t MAX_INITIAL_ALLOC = 1024 * 1024;
58
22.3k
        gnBufferAlloc = std::min(gnBufferAlloc, MAX_INITIAL_ALLOC);
59
22.3k
        gpabyBuffer = static_cast<GByte *>(CPLMalloc(gnBufferAlloc));
60
22.3k
    }
61
22.3k
}
62
63
/************************************************************************/
64
/* ==================================================================== */
65
/*                       VSIStdinFilesystemHandler                     */
66
/* ==================================================================== */
67
/************************************************************************/
68
69
class VSIStdinFilesystemHandler final : public VSIFilesystemHandler
70
{
71
    CPL_DISALLOW_COPY_ASSIGN(VSIStdinFilesystemHandler)
72
73
  public:
74
    VSIStdinFilesystemHandler();
75
    ~VSIStdinFilesystemHandler() override;
76
77
    VSIVirtualHandleUniquePtr Open(const char *pszFilename,
78
                                   const char *pszAccess, bool bSetError,
79
                                   CSLConstList /* papszOptions */) override;
80
    int Stat(const char *pszFilename, VSIStatBufL *pStatBuf,
81
             int nFlags) override;
82
83
    bool SupportsSequentialWrite(const char * /* pszPath */,
84
                                 bool /* bAllowLocalTempFile */) override
85
0
    {
86
0
        return false;
87
0
    }
88
89
    bool SupportsRandomWrite(const char * /* pszPath */,
90
                             bool /* bAllowLocalTempFile */) override
91
0
    {
92
0
        return false;
93
0
    }
94
};
95
96
/************************************************************************/
97
/* ==================================================================== */
98
/*                        VSIStdinHandle                               */
99
/* ==================================================================== */
100
/************************************************************************/
101
102
class VSIStdinHandle final : public VSIVirtualHandle
103
{
104
  private:
105
    CPL_DISALLOW_COPY_ASSIGN(VSIStdinHandle)
106
107
    bool m_bEOF = false;
108
    bool m_bError = false;
109
    uint64_t m_nCurOff = 0;
110
    size_t ReadAndCache(void *pBuffer, size_t nToRead);
111
112
  public:
113
2.22k
    VSIStdinHandle() = default;
114
115
    ~VSIStdinHandle() override
116
2.22k
    {
117
2.22k
        VSIStdinHandle::Close();
118
2.22k
    }
119
120
    int Seek(vsi_l_offset nOffset, int nWhence) override;
121
    vsi_l_offset Tell() override;
122
    size_t Read(void *pBuffer, size_t nBytes) override;
123
    size_t Write(const void *pBuffer, size_t nBytes) override;
124
    void ClearErr() override;
125
    int Error() override;
126
    int Eof() override;
127
    int Close() override;
128
};
129
130
/************************************************************************/
131
/*                            ReadAndCache()                            */
132
/************************************************************************/
133
134
size_t VSIStdinHandle::ReadAndCache(void *pUserBuffer, size_t nToRead)
135
19.7k
{
136
19.7k
    CPLAssert(m_nCurOff == gnRealPos);
137
138
19.7k
    const size_t nRead = fread(pUserBuffer, 1, nToRead, gStdinFile);
139
140
19.7k
    if (gnRealPos < gnBufferLimit)
141
19.7k
    {
142
19.7k
        bool bCopyInBuffer = true;
143
19.7k
        const size_t nToCopy = static_cast<size_t>(
144
19.7k
            std::min(gnBufferLimit - gnRealPos, static_cast<uint64_t>(nRead)));
145
19.7k
        if (gnRealPos + nToCopy > gnBufferAlloc)
146
0
        {
147
0
            auto newAlloc = gnRealPos + nToCopy;
148
0
            if (newAlloc < gnBufferLimit - newAlloc / 3)
149
0
                newAlloc += newAlloc / 3;
150
0
            else
151
0
                newAlloc = gnBufferLimit;
152
0
            GByte *newBuffer = static_cast<GByte *>(VSI_REALLOC_VERBOSE(
153
0
                gpabyBuffer, static_cast<size_t>(newAlloc)));
154
0
            if (newBuffer == nullptr)
155
0
            {
156
0
                bCopyInBuffer = false;
157
0
            }
158
0
            else
159
0
            {
160
0
                gpabyBuffer = newBuffer;
161
0
                gnBufferAlloc = static_cast<size_t>(newAlloc);
162
0
            }
163
0
        }
164
19.7k
        if (bCopyInBuffer)
165
19.7k
        {
166
19.7k
            memcpy(gpabyBuffer + static_cast<size_t>(gnRealPos), pUserBuffer,
167
19.7k
                   nToCopy);
168
19.7k
            gnBufferLen += nToCopy;
169
19.7k
        }
170
19.7k
    }
171
172
19.7k
    m_nCurOff += nRead;
173
19.7k
    gnRealPos = m_nCurOff;
174
175
19.7k
    if (nRead < nToRead)
176
19.7k
    {
177
19.7k
        gbHasSoughtToEnd = feof(gStdinFile);
178
19.7k
        if (gbHasSoughtToEnd)
179
19.7k
            gnFileSize = gnRealPos;
180
19.7k
        gbHasErrored = ferror(gStdinFile);
181
19.7k
    }
182
183
19.7k
    return nRead;
184
19.7k
}
185
186
/************************************************************************/
187
/*                                Seek()                                */
188
/************************************************************************/
189
190
int VSIStdinHandle::Seek(vsi_l_offset nOffset, int nWhence)
191
192
5.86k
{
193
5.86k
    m_bEOF = false;
194
195
5.86k
    if (nWhence == SEEK_SET && nOffset == m_nCurOff)
196
3.23k
        return 0;
197
198
2.63k
    VSIStdinInit();
199
200
2.63k
    if (nWhence == SEEK_END)
201
2.63k
    {
202
2.63k
        if (nOffset != 0)
203
0
        {
204
0
            CPLError(CE_Failure, CPLE_NotSupported,
205
0
                     "Seek(xx != 0, SEEK_END) unsupported on /vsistdin");
206
0
            return -1;
207
0
        }
208
209
2.63k
        if (gbHasSoughtToEnd)
210
2.63k
        {
211
2.63k
            m_nCurOff = gnFileSize;
212
2.63k
            return 0;
213
2.63k
        }
214
215
0
        nOffset = static_cast<vsi_l_offset>(-1);
216
0
    }
217
0
    else if (nWhence == SEEK_CUR)
218
0
    {
219
0
        nOffset += m_nCurOff;
220
0
    }
221
222
0
    if (nWhence != SEEK_END && gnRealPos >= gnBufferLimit &&
223
0
        nOffset >= gnBufferLimit)
224
0
    {
225
0
        CPLError(CE_Failure, CPLE_NotSupported,
226
0
                 "Backward Seek() unsupported on /vsistdin beyond "
227
0
                 "maximum buffer limit (" CPL_FRMT_GUIB " bytes).\n"
228
0
                 "This limit can be extended by setting the "
229
0
                 "CPL_VSISTDIN_BUFFER_LIMIT "
230
0
                 "configuration option to a number of bytes, or by using the "
231
0
                 "'/vsistdin?buffer_limit=number_of_bytes' filename.\n"
232
0
                 "A limit of -1 means unlimited.",
233
0
                 static_cast<GUIntBig>(gnBufferLimit));
234
0
        return -1;
235
0
    }
236
237
0
    if (nOffset < gnBufferLen)
238
0
    {
239
0
        m_nCurOff = nOffset;
240
0
        return 0;
241
0
    }
242
243
0
    if (nOffset == m_nCurOff)
244
0
        return 0;
245
246
0
    CPLDebug("VSI", "Forward seek from " CPL_FRMT_GUIB " to " CPL_FRMT_GUIB,
247
0
             static_cast<GUIntBig>(m_nCurOff), nOffset);
248
249
0
    char abyTemp[8192] = {};
250
0
    m_nCurOff = gnRealPos;
251
0
    while (true)
252
0
    {
253
0
        const size_t nToRead = static_cast<size_t>(
254
0
            std::min(static_cast<uint64_t>(sizeof(abyTemp)),
255
0
                     static_cast<uint64_t>(nOffset - m_nCurOff)));
256
0
        const size_t nRead = ReadAndCache(abyTemp, nToRead);
257
258
0
        if (nRead < nToRead)
259
0
        {
260
0
            return nWhence == SEEK_END ? 0 : -1;
261
0
        }
262
0
        if (nToRead < sizeof(abyTemp))
263
0
            break;
264
0
    }
265
266
0
    return 0;
267
0
}
268
269
/************************************************************************/
270
/*                                Tell()                                */
271
/************************************************************************/
272
273
vsi_l_offset VSIStdinHandle::Tell()
274
2.63k
{
275
2.63k
    return m_nCurOff;
276
2.63k
}
277
278
/************************************************************************/
279
/*                                Read()                                */
280
/************************************************************************/
281
282
size_t VSIStdinHandle::Read(void *pBuffer, size_t nBytesToRead)
283
284
19.7k
{
285
19.7k
    VSIStdinInit();
286
287
19.7k
    if (nBytesToRead == 0)
288
0
        return 0;
289
290
19.7k
    if (m_nCurOff < gnRealPos && gnRealPos >= gnBufferLimit &&
291
0
        m_nCurOff + nBytesToRead > gnBufferLimit)
292
0
    {
293
0
        CPLError(CE_Failure, CPLE_NotSupported,
294
0
                 "Backward Seek() unsupported on /vsistdin beyond "
295
0
                 "maximum buffer limit (" CPL_FRMT_GUIB " bytes).\n"
296
0
                 "This limit can be extended by setting the "
297
0
                 "CPL_VSISTDIN_BUFFER_LIMIT "
298
0
                 "configuration option to a number of bytes, or by using the "
299
0
                 "'/vsistdin?buffer_limit=number_of_bytes' filename.\n"
300
0
                 "A limit of -1 means unlimited.",
301
0
                 static_cast<GUIntBig>(gnBufferLimit));
302
0
        return 0;
303
0
    }
304
305
19.7k
    if (m_nCurOff < gnBufferLen)
306
0
    {
307
0
        const size_t nAlreadyCached =
308
0
            static_cast<size_t>(gnBufferLen - m_nCurOff);
309
0
        if (nBytesToRead <= nAlreadyCached)
310
0
        {
311
0
            memcpy(pBuffer, gpabyBuffer + static_cast<size_t>(m_nCurOff),
312
0
                   nBytesToRead);
313
0
            m_nCurOff += nBytesToRead;
314
0
            return nBytesToRead;
315
0
        }
316
317
0
        memcpy(pBuffer, gpabyBuffer + static_cast<size_t>(m_nCurOff),
318
0
               nAlreadyCached);
319
0
        m_nCurOff += nAlreadyCached;
320
321
0
        const size_t nRead =
322
0
            ReadAndCache(static_cast<GByte *>(pBuffer) + nAlreadyCached,
323
0
                         nBytesToRead - nAlreadyCached);
324
0
        m_bEOF = gbHasSoughtToEnd;
325
0
        m_bError = gbHasErrored;
326
327
0
        return nRead + nAlreadyCached;
328
0
    }
329
330
19.7k
    const size_t nRead = ReadAndCache(pBuffer, nBytesToRead);
331
19.7k
    m_bEOF = gbHasSoughtToEnd;
332
19.7k
    m_bError = gbHasErrored;
333
19.7k
    return nRead;
334
19.7k
}
335
336
/************************************************************************/
337
/*                               Write()                                */
338
/************************************************************************/
339
340
size_t VSIStdinHandle::Write(const void * /* pBuffer */, size_t /* nBytes */)
341
0
{
342
0
    CPLError(CE_Failure, CPLE_NotSupported, "Write() unsupported on /vsistdin");
343
0
    return 0;
344
0
}
345
346
/************************************************************************/
347
/*                              ClearErr()                              */
348
/************************************************************************/
349
350
void VSIStdinHandle::ClearErr()
351
352
0
{
353
0
    clearerr(gStdinFile);
354
0
    m_bEOF = false;
355
0
    m_bError = false;
356
0
}
357
358
/************************************************************************/
359
/*                               Error()                                */
360
/************************************************************************/
361
362
int VSIStdinHandle::Error()
363
364
118
{
365
118
    return m_bError;
366
118
}
367
368
/************************************************************************/
369
/*                                Eof()                                 */
370
/************************************************************************/
371
372
int VSIStdinHandle::Eof()
373
374
251
{
375
251
    return m_bEOF;
376
251
}
377
378
/************************************************************************/
379
/*                               Close()                                */
380
/************************************************************************/
381
382
int VSIStdinHandle::Close()
383
384
4.45k
{
385
4.45k
    if (!gosStdinFilename.empty() &&
386
0
        CPLTestBool(CPLGetConfigOption("CPL_VSISTDIN_FILE_CLOSE", "NO")))
387
0
    {
388
0
        if (gStdinFile != stdin)
389
0
            fclose(gStdinFile);
390
0
        gStdinFile = stdin;
391
0
        gosStdinFilename.clear();
392
0
        gnRealPos = ftell(stdin);
393
0
        gnBufferLen = 0;
394
0
        gbHasSoughtToEnd = false;
395
0
        gbHasErrored = false;
396
0
        gnFileSize = 0;
397
0
    }
398
4.45k
    return 0;
399
4.45k
}
400
401
/************************************************************************/
402
/* ==================================================================== */
403
/*                       VSIStdinFilesystemHandler                     */
404
/* ==================================================================== */
405
/************************************************************************/
406
407
/************************************************************************/
408
/*                     VSIStdinFilesystemHandler()                      */
409
/************************************************************************/
410
411
VSIStdinFilesystemHandler::VSIStdinFilesystemHandler()
412
3
{
413
3
}
414
415
/************************************************************************/
416
/*                     ~VSIStdinFilesystemHandler()                     */
417
/************************************************************************/
418
419
VSIStdinFilesystemHandler::~VSIStdinFilesystemHandler()
420
0
{
421
0
    if (gStdinFile != stdin)
422
0
        fclose(gStdinFile);
423
0
    gStdinFile = stdin;
424
0
    CPLFree(gpabyBuffer);
425
0
    gpabyBuffer = nullptr;
426
0
    gnBufferLimit = 0;
427
0
    gnBufferAlloc = 0;
428
0
    gnBufferLen = 0;
429
0
    gnRealPos = 0;
430
0
    gosStdinFilename.clear();
431
0
}
432
433
/************************************************************************/
434
/*                           GetBufferLimit()                           */
435
/************************************************************************/
436
437
static size_t GetBufferLimit(const char *pszBufferLimit)
438
7.01k
{
439
7.01k
    uint64_t nVal =
440
7.01k
        static_cast<uint64_t>(std::strtoull(pszBufferLimit, nullptr, 10));
441
442
    // -1 because on 64-bit builds with size_t==uint64_t, a static analyzer
443
    // could complain that the ending nVal > MAX_BUFFER_LIMIT test is always
444
    // false
445
7.01k
    constexpr size_t MAX_BUFFER_LIMIT = std::numeric_limits<size_t>::max() - 1;
446
7.01k
    if (strstr(pszBufferLimit, "MB") != nullptr)
447
194
    {
448
194
        constexpr size_t ONE_MB = 1024 * 1024;
449
194
        if (nVal > MAX_BUFFER_LIMIT / ONE_MB)
450
88
        {
451
88
            nVal = MAX_BUFFER_LIMIT;
452
88
        }
453
106
        else
454
106
        {
455
106
            nVal *= ONE_MB;
456
106
        }
457
194
    }
458
6.81k
    else if (strstr(pszBufferLimit, "GB") != nullptr)
459
122
    {
460
122
        constexpr size_t ONE_GB = 1024 * 1024 * 1024;
461
122
        if (nVal > MAX_BUFFER_LIMIT / ONE_GB)
462
24
        {
463
24
            nVal = MAX_BUFFER_LIMIT;
464
24
        }
465
98
        else
466
98
        {
467
98
            nVal *= ONE_GB;
468
98
        }
469
122
    }
470
7.01k
    if (nVal > MAX_BUFFER_LIMIT)
471
313
    {
472
313
        nVal = MAX_BUFFER_LIMIT;
473
313
    }
474
7.01k
    return static_cast<size_t>(nVal);
475
7.01k
}
476
477
/************************************************************************/
478
/*                           ParseFilename()                            */
479
/************************************************************************/
480
481
static bool ParseFilename(const char *pszFilename)
482
6.53k
{
483
6.53k
    if (!(EQUAL(pszFilename, "/vsistdin/") ||
484
4.77k
          ((STARTS_WITH(pszFilename, "/vsistdin/?") ||
485
1.32k
            STARTS_WITH(pszFilename, "/vsistdin?")) &&
486
4.67k
           strchr(pszFilename, '.') == nullptr)))
487
196
    {
488
196
        return false;
489
196
    }
490
491
6.33k
    if (!CPLTestBool(CPLGetConfigOption("CPL_ALLOW_VSISTDIN", "YES")))
492
0
    {
493
0
        CPLError(CE_Failure, CPLE_NotSupported,
494
0
                 "/vsistdin/ disabled. Set CPL_ALLOW_VSISTDIN to YES to "
495
0
                 "enable it");
496
0
        return false;
497
0
    }
498
499
6.33k
    const char *pszBufferLimit =
500
6.33k
        CPLGetConfigOption("CPL_VSISTDIN_BUFFER_LIMIT", "1048576");
501
6.33k
    size_t nBufferLimit = GetBufferLimit(pszBufferLimit);
502
503
6.33k
    pszFilename += strlen("/vsistdin/");
504
6.33k
    if (*pszFilename == '?')
505
3.43k
        pszFilename++;
506
6.33k
    char **papszTokens = CSLTokenizeString2(pszFilename, "&", 0);
507
13.1k
    for (int i = 0; papszTokens[i] != nullptr; i++)
508
6.83k
    {
509
6.83k
        char *pszUnescaped =
510
6.83k
            CPLUnescapeString(papszTokens[i], nullptr, CPLES_URL);
511
6.83k
        CPLFree(papszTokens[i]);
512
6.83k
        papszTokens[i] = pszUnescaped;
513
6.83k
    }
514
515
13.1k
    for (int i = 0; papszTokens[i]; i++)
516
6.83k
    {
517
6.83k
        char *pszKey = nullptr;
518
6.83k
        const char *pszValue = CPLParseNameValue(papszTokens[i], &pszKey);
519
6.83k
        if (pszKey && pszValue)
520
1.63k
        {
521
1.63k
            if (EQUAL(pszKey, "buffer_limit"))
522
675
            {
523
675
                nBufferLimit = GetBufferLimit(pszValue);
524
675
            }
525
963
            else
526
963
            {
527
963
                CPLError(CE_Warning, CPLE_NotSupported,
528
963
                         "Unsupported option: %s", pszKey);
529
963
            }
530
1.63k
        }
531
6.83k
        CPLFree(pszKey);
532
6.83k
    }
533
534
6.33k
    CSLDestroy(papszTokens);
535
536
    // For testing purposes
537
6.33k
    const char *pszStdinFilename =
538
6.33k
        CPLGetConfigOption("CPL_VSISTDIN_FILE", "stdin");
539
6.33k
    if (EQUAL(pszStdinFilename, "stdin"))
540
6.33k
    {
541
6.33k
        if (!gosStdinFilename.empty())
542
0
        {
543
0
            if (gStdinFile != stdin)
544
0
                fclose(gStdinFile);
545
0
            gStdinFile = stdin;
546
0
            gosStdinFilename.clear();
547
0
            gnRealPos = ftell(stdin);
548
0
            gnBufferLen = 0;
549
0
            gbHasSoughtToEnd = false;
550
0
            gnFileSize = 0;
551
0
        }
552
6.33k
    }
553
0
    else
554
0
    {
555
0
        bool bReset = false;
556
0
        if (gosStdinFilename != pszStdinFilename)
557
0
        {
558
0
            if (gStdinFile != stdin)
559
0
                fclose(gStdinFile);
560
0
            gStdinFile = fopen(pszStdinFilename, "rb");
561
0
            if (gStdinFile == nullptr)
562
0
            {
563
0
                gStdinFile = stdin;
564
0
                return false;
565
0
            }
566
0
            gosStdinFilename = pszStdinFilename;
567
0
            bReset = true;
568
0
        }
569
0
        else
570
0
        {
571
0
            bReset = CPLTestBool(
572
0
                CPLGetConfigOption("CPL_VSISTDIN_RESET_POSITION", "NO"));
573
0
        }
574
0
        if (bReset)
575
0
        {
576
0
            gnBufferLimit = 0;
577
0
            gnBufferLen = 0;
578
0
            gnRealPos = 0;
579
0
            gbHasSoughtToEnd = false;
580
0
            gnFileSize = 0;
581
0
        }
582
0
    }
583
584
6.33k
    gnBufferLimit = std::max(gnBufferLimit, nBufferLimit);
585
586
6.33k
    return true;
587
6.33k
}
588
589
/************************************************************************/
590
/*                                Open()                                */
591
/************************************************************************/
592
593
VSIVirtualHandleUniquePtr
594
VSIStdinFilesystemHandler::Open(const char *pszFilename, const char *pszAccess,
595
                                bool /* bSetError */,
596
                                CSLConstList /* papszOptions */)
597
598
2.33k
{
599
2.33k
    if (!ParseFilename(pszFilename))
600
102
    {
601
102
        return nullptr;
602
102
    }
603
604
2.22k
    if (strchr(pszAccess, 'w') != nullptr || strchr(pszAccess, '+') != nullptr)
605
0
    {
606
0
        CPLError(CE_Failure, CPLE_NotSupported,
607
0
                 "Write or update mode not supported on /vsistdin");
608
0
        return nullptr;
609
0
    }
610
611
2.22k
    return VSIVirtualHandleUniquePtr(
612
2.22k
        std::make_unique<VSIStdinHandle>().release());
613
2.22k
}
614
615
/************************************************************************/
616
/*                                Stat()                                */
617
/************************************************************************/
618
619
int VSIStdinFilesystemHandler::Stat(const char *pszFilename,
620
                                    VSIStatBufL *pStatBuf, int nFlags)
621
622
4.20k
{
623
4.20k
    memset(pStatBuf, 0, sizeof(VSIStatBufL));
624
625
4.20k
    if (!ParseFilename(pszFilename))
626
94
    {
627
94
        return -1;
628
94
    }
629
630
4.11k
    if (nFlags & VSI_STAT_SIZE_FLAG)
631
1.06k
    {
632
1.06k
        if (gbHasSoughtToEnd)
633
1.06k
            pStatBuf->st_size = gnFileSize;
634
0
        else
635
0
        {
636
0
            auto handle = Open(pszFilename, "rb", false, nullptr);
637
0
            if (handle == nullptr)
638
0
                return -1;
639
0
            handle->Seek(0, SEEK_END);
640
0
            pStatBuf->st_size = handle->Tell();
641
0
        }
642
1.06k
    }
643
644
4.11k
    pStatBuf->st_mode = S_IFREG;
645
4.11k
    return 0;
646
4.11k
}
647
648
//! @endcond
649
650
/************************************************************************/
651
/*                       VSIInstallStdinHandler()                       */
652
/************************************************************************/
653
654
/*!
655
 \brief Install /vsistdin/ file system handler
656
657
 A special file handler is installed that allows reading from the standard
658
 input stream.
659
660
 The file operations available are of course limited to Read() and
661
 forward Seek() (full seek in the first MB of a file by default).
662
663
 Starting with GDAL 3.6, this limit can be configured either by setting
664
 the CPL_VSISTDIN_BUFFER_LIMIT configuration option to a number of bytes
665
 (can be -1 for unlimited), or using the "/vsistdin?buffer_limit=value"
666
 filename.
667
668
 \verbatim embed:rst
669
 See :ref:`/vsistdin/ documentation <vsistdin>`
670
 \endverbatim
671
672
 */
673
void VSIInstallStdinHandler()
674
675
3
{
676
3
    auto poHandler = std::make_shared<VSIStdinFilesystemHandler>();
677
3
    VSIFileManager::InstallHandler("/vsistdin/", poHandler);
678
3
    VSIFileManager::InstallHandler("/vsistdin?", poHandler);
679
3
}