Line data Source code
1 : /**********************************************************************
2 : *
3 : * Project: CPL - Common Portability Library
4 : * Purpose: Implement VSI large file api for encrypted files.
5 : * Author: Even Rouault, even.rouault at spatialys.com
6 : *
7 : ******************************************************************************
8 : * Copyright (c) 2015, Even Rouault <even.rouault at spatialys.com>
9 : *
10 : * SPDX-License-Identifier: MIT
11 : ****************************************************************************/
12 :
13 : #include "cpl_port.h"
14 : #include "cpl_vsi_virtual.h"
15 :
16 : #include <cstddef>
17 : #include <algorithm>
18 :
19 : #include "cpl_error.h"
20 : #include "cpl_vsi.h"
21 :
22 : CPL_C_START
23 : void CPL_DLL VSIInstallCryptFileHandler();
24 : void CPL_DLL VSISetCryptKey(const GByte *pabyKey, int nKeySize);
25 : CPL_C_END
26 :
27 : constexpr char VSICRYPT_PREFIX[] = "/vsicrypt/";
28 :
29 : #if defined(HAVE_CRYPTOPP) || defined(DOXYGEN_SKIP)
30 :
31 : //! @cond Doxygen_Suppress
32 :
33 : /* Increase Major in case of backward incompatible changes */
34 : constexpr int VSICRYPT_CURRENT_MAJOR = 1;
35 : constexpr int VSICRYPT_CURRENT_MINOR = 0;
36 : constexpr char VSICRYPT_SIGNATURE[] = "VSICRYPT"; // Must be 8 chars.
37 :
38 : constexpr char VSICRYPT_PREFIX_WITHOUT_SLASH[] = "/vsicrypt";
39 :
40 : constexpr unsigned int VSICRYPT_READ = 0x1;
41 : constexpr unsigned int VSICRYPT_WRITE = 0x2;
42 :
43 : /* Begin of crypto++ headers */
44 : #ifdef _MSC_VER
45 : #pragma warning(push)
46 : #pragma warning(disable : 4189)
47 : #pragma warning(disable : 4512)
48 : #pragma warning(disable : 4244)
49 : #pragma warning(disable : 4505)
50 : #endif
51 :
52 : #ifdef __GNUC__
53 : #pragma GCC diagnostic push
54 : #pragma GCC diagnostic ignored "-Weffc++"
55 : #pragma GCC diagnostic ignored "-Wnon-virtual-dtor"
56 : #pragma GCC diagnostic ignored "-Wold-style-cast"
57 : #endif
58 :
59 : #ifdef USE_ONLY_CRYPTODLL_ALG
60 : #include "cryptopp/dll.h"
61 : #else
62 : #include "cryptopp/aes.h"
63 : #include "cryptopp/blowfish.h"
64 : #include "cryptopp/camellia.h"
65 : #include "cryptopp/cast.h"
66 : #include "cryptopp/des.h"
67 : #include "cryptopp/mars.h"
68 : #include "cryptopp/idea.h"
69 : #include "cryptopp/rc5.h"
70 : #include "cryptopp/rc6.h"
71 : #include "cryptopp/serpent.h"
72 : #include "cryptopp/shacal2.h"
73 : #include "cryptopp/skipjack.h"
74 : #include "cryptopp/tea.h"
75 : #include "cryptopp/twofish.h"
76 : #endif
77 :
78 : #include "cryptopp/filters.h"
79 : #include "cryptopp/modes.h"
80 : #include "cryptopp/osrng.h"
81 :
82 : #ifdef __GNUC__
83 : #pragma GCC diagnostic pop
84 : #endif
85 :
86 : #ifdef _MSC_VER
87 : #pragma warning(pop)
88 : #endif
89 :
90 : // Fix compatibility with Crypto++
91 : #if CRYPTOPP_VERSION >= 600
92 : typedef CryptoPP::byte cryptopp_byte;
93 : #else
94 : typedef byte cryptopp_byte;
95 : #endif
96 :
97 : /* End of crypto++ headers */
98 :
99 : // I don't really understand why this is necessary, especially
100 : // when cryptopp.dll and GDAL have been compiled with the same
101 : // VC version and /MD. But otherwise you'll get crashes
102 : // Borrowed from dlltest.cpp of crypto++
103 : #if defined(_WIN32) && defined(USE_ONLY_CRYPTODLL_ALG)
104 :
105 : static CryptoPP::PNew s_pNew = nullptr;
106 : static CryptoPP::PDelete s_pDelete = nullptr;
107 :
108 : extern "C" __declspec(dllexport) void __cdecl SetNewAndDeleteFromCryptoPP(
109 : CryptoPP::PNew pNew, CryptoPP::PDelete pDelete,
110 : CryptoPP::PSetNewHandler pSetNewHandler)
111 : {
112 : s_pNew = pNew;
113 : s_pDelete = pDelete;
114 : }
115 :
116 : void *__cdecl operator new(size_t size)
117 : {
118 : return s_pNew(size);
119 : }
120 :
121 : void __cdecl operator delete(void *p)
122 : {
123 : s_pDelete(p);
124 : }
125 :
126 : #endif // defined(_WIN32) && defined(USE_ONLY_CRYPTODLL_ALG)
127 :
128 : static GByte *pabyGlobalKey = nullptr;
129 : static int nGlobalKeySize = 0;
130 :
131 : typedef enum
132 : {
133 : ALG_AES,
134 : ALG_Blowfish,
135 : ALG_Camellia,
136 : // ALG_CAST128, (obsolete)
137 : ALG_CAST256,
138 : // ALG_DES, (obsolete)
139 : ALG_DES_EDE2,
140 : ALG_DES_EDE3,
141 : // ALG_DES_XEX3, (obsolete)
142 : // ALG_Gost, (obsolete)
143 : ALG_MARS,
144 : ALG_IDEA,
145 : // ALG_RC2, (obsolete)
146 : ALG_RC5,
147 : ALG_RC6,
148 : // ALG_SAFER_K, (obsolete)
149 : // ALG_SAFER_SK, (obsolete)
150 : ALG_Serpent,
151 : ALG_SHACAL2,
152 : // ALG_SHARK, (obsolete)
153 : ALG_SKIPJACK,
154 : ALG_Twofish,
155 : // ALG_ThreeWay, (obsolete)
156 : ALG_XTEA,
157 : ALG_MAX = ALG_XTEA
158 : } VSICryptAlg;
159 :
160 : typedef enum
161 : {
162 : MODE_CBC,
163 : MODE_CFB,
164 : MODE_OFB,
165 : MODE_CTR,
166 : MODE_CBC_CTS,
167 : MODE_MAX = MODE_CBC_CTS
168 : } VSICryptMode;
169 :
170 : //! @endcond
171 :
172 : /************************************************************************/
173 : /* VSISetCryptKey() */
174 : /************************************************************************/
175 :
176 : /** Installs the encryption/decryption key.
177 : *
178 : * By passing a NULL key, the previously installed key will be cleared.
179 : * Note, however, that it is not guaranteed that there won't be trace of it
180 : * in other places in memory or in on-disk temporary file.
181 : *
182 : * @param pabyKey key. Might be NULL to clear previously set key.
183 : * @param nKeySize length of the key in bytes. Might be 0 to clear
184 : * previously set key.
185 : *
186 : * @see VSIInstallCryptFileHandler() for documentation on /vsicrypt/
187 : */
188 3 : void VSISetCryptKey(const GByte *pabyKey, int nKeySize)
189 : {
190 3 : CPLAssert((pabyKey != nullptr && nKeySize != 0) ||
191 : (pabyKey == nullptr && nKeySize == 0));
192 3 : if (pabyGlobalKey)
193 : {
194 : // Make some effort to clear the memory, although it could have leaked
195 : // elsewhere...
196 2 : memset(pabyGlobalKey, 0, nGlobalKeySize);
197 2 : CPLFree(pabyGlobalKey);
198 2 : pabyGlobalKey = nullptr;
199 2 : nGlobalKeySize = 0;
200 : }
201 3 : if (pabyKey)
202 : {
203 2 : pabyGlobalKey = static_cast<GByte *>(CPLMalloc(nKeySize));
204 2 : memcpy(pabyGlobalKey, pabyKey, nKeySize);
205 2 : nGlobalKeySize = nKeySize;
206 : }
207 3 : }
208 :
209 : //! @cond Doxygen_Suppress
210 :
211 : /************************************************************************/
212 : /* GetAlg() */
213 : /************************************************************************/
214 :
215 : #undef CASE_ALG
216 : #define CASE_ALG(alg) \
217 : if (EQUAL(pszName, #alg)) \
218 : return ALG_##alg;
219 :
220 1044 : static VSICryptAlg GetAlg(const char *pszName)
221 : {
222 1044 : CASE_ALG(AES)
223 16 : CASE_ALG(Blowfish)
224 14 : CASE_ALG(Camellia)
225 : // CASE_ALG(CAST128) (obsolete)
226 13 : CASE_ALG(CAST256)
227 : // CASE_ALG(DES) (obsolete)
228 12 : CASE_ALG(DES_EDE2)
229 11 : CASE_ALG(DES_EDE3)
230 : // CASE_ALG(DES_XEX3) (obsolete)
231 : // CASE_ALG(Ghost) (obsolete)
232 10 : CASE_ALG(MARS)
233 9 : CASE_ALG(IDEA)
234 : // CASE_ALG(RC2) (obsolete)
235 8 : CASE_ALG(RC5)
236 7 : CASE_ALG(RC6)
237 : // CASE_ALG(SAFER_K) (obsolete)
238 : // CASE_ALG(SAFER_SK) (obsolete)
239 6 : CASE_ALG(Serpent)
240 5 : CASE_ALG(SHACAL2)
241 : // CASE_ALG(SHARK) (obsolete)
242 4 : CASE_ALG(SKIPJACK)
243 : // CASE_ALG(ThreeWay) (obsolete)
244 3 : CASE_ALG(Twofish)
245 2 : CASE_ALG(XTEA)
246 :
247 1 : CPLError(CE_Warning, CPLE_NotSupported,
248 : "Unsupported cipher algorithm: %s. Using AES instead", pszName);
249 1 : return ALG_AES;
250 : }
251 :
252 : /************************************************************************/
253 : /* GetEncBlockCipher() */
254 : /************************************************************************/
255 :
256 : #undef CASE_ALG
257 : #define CASE_ALG(alg) \
258 : case ALG_##alg: \
259 : return new CryptoPP::alg::Encryption();
260 :
261 2226 : static CryptoPP::BlockCipher *GetEncBlockCipher(VSICryptAlg eAlg)
262 : {
263 2226 : switch (eAlg)
264 : {
265 2182 : CASE_ALG(AES)
266 : #ifndef USE_ONLY_CRYPTODLL_ALG
267 5 : CASE_ALG(Blowfish)
268 3 : CASE_ALG(Camellia)
269 : // CASE_ALG(CAST128) (obsolete)
270 3 : CASE_ALG(CAST256)
271 : #endif
272 : // CASE_ALG(DES) (obsolete)
273 3 : CASE_ALG(DES_EDE2)
274 3 : CASE_ALG(DES_EDE3)
275 : // CASE_ALG(DES_XEX3) (obsolete)
276 : #ifndef USE_ONLY_CRYPTODLL_ALG
277 : // CASE_ALG(Gost) (obsolete)
278 3 : CASE_ALG(MARS)
279 3 : CASE_ALG(IDEA)
280 : // CASE_ALG(RC2) (obsolete)
281 3 : CASE_ALG(RC5)
282 3 : CASE_ALG(RC6)
283 : // CASE_ALG(SAFER_K) (obsolete)
284 : // CASE_ALG(SAFER_SK) (obsolete)
285 3 : CASE_ALG(Serpent)
286 3 : CASE_ALG(SHACAL2)
287 : // CASE_ALG(SHARK) (obsolete)
288 : #endif
289 3 : CASE_ALG(SKIPJACK)
290 : #ifndef USE_ONLY_CRYPTODLL_ALG
291 : // CASE_ALG(ThreeWay) (obsolete)
292 3 : CASE_ALG(Twofish)
293 3 : CASE_ALG(XTEA)
294 : #endif
295 0 : default:
296 0 : return nullptr;
297 : }
298 : }
299 :
300 : /************************************************************************/
301 : /* GetDecBlockCipher() */
302 : /************************************************************************/
303 :
304 : #undef CASE_ALG
305 : #define CASE_ALG(alg) \
306 : case ALG_##alg: \
307 : return new CryptoPP::alg::Decryption();
308 :
309 1173 : static CryptoPP::BlockCipher *GetDecBlockCipher(VSICryptAlg eAlg)
310 : {
311 1173 : switch (eAlg)
312 : {
313 1144 : CASE_ALG(AES)
314 : #ifndef USE_ONLY_CRYPTODLL_ALG
315 3 : CASE_ALG(Blowfish)
316 2 : CASE_ALG(Camellia)
317 : // CASE_ALG(CAST128) (obsolete)
318 2 : CASE_ALG(CAST256)
319 : #endif
320 : // CASE_ALG(DES) (obsolete)
321 2 : CASE_ALG(DES_EDE2)
322 2 : CASE_ALG(DES_EDE3)
323 : // CASE_ALG(DES_XEX3) (obsolete)
324 : #ifndef USE_ONLY_CRYPTODLL_ALG
325 : // CASE_ALG(Gost) (obsolete)
326 2 : CASE_ALG(MARS)
327 2 : CASE_ALG(IDEA)
328 : // CASE_ALG(RC2) (obsolete)
329 2 : CASE_ALG(RC5)
330 2 : CASE_ALG(RC6)
331 : // CASE_ALG(SAFER_K) (obsolete)
332 : // CASE_ALG(SAFER_SK) (obsolete)
333 2 : CASE_ALG(Serpent)
334 2 : CASE_ALG(SHACAL2)
335 : // CASE_ALG(SHARK) (obsolete)
336 : #endif
337 2 : CASE_ALG(SKIPJACK)
338 : #ifndef USE_ONLY_CRYPTODLL_ALG
339 : // CASE_ALG(ThreeWay) (obsolete)
340 2 : CASE_ALG(Twofish)
341 2 : CASE_ALG(XTEA)
342 : #endif
343 0 : default:
344 0 : return nullptr;
345 : }
346 : }
347 :
348 : /************************************************************************/
349 : /* GetMode() */
350 : /************************************************************************/
351 :
352 1044 : static VSICryptMode GetMode(const char *pszName)
353 : {
354 1044 : if (EQUAL(pszName, "CBC"))
355 1038 : return MODE_CBC;
356 6 : if (EQUAL(pszName, "CFB"))
357 1 : return MODE_CFB;
358 5 : if (EQUAL(pszName, "OFB"))
359 1 : return MODE_OFB;
360 4 : if (EQUAL(pszName, "CTR"))
361 1 : return MODE_CTR;
362 3 : if (EQUAL(pszName, "CBC_CTS"))
363 2 : return MODE_CBC_CTS;
364 :
365 1 : CPLError(CE_Warning, CPLE_NotSupported,
366 : "Unsupported cipher block mode: %s. Using CBC instead", pszName);
367 1 : return MODE_CBC;
368 : }
369 :
370 : /************************************************************************/
371 : /* VSICryptFileHeader */
372 : /************************************************************************/
373 :
374 : class VSICryptFileHeader
375 : {
376 : CPL_DISALLOW_COPY_ASSIGN(VSICryptFileHeader)
377 :
378 : std::string CryptKeyCheck(CryptoPP::BlockCipher *poEncCipher);
379 :
380 : public:
381 1274 : VSICryptFileHeader() = default;
382 :
383 : int ReadFromFile(VSIVirtualHandle *fp, const CPLString &osKey);
384 : int WriteToFile(VSIVirtualHandle *fp, CryptoPP::BlockCipher *poEncCipher);
385 :
386 : GUInt16 nHeaderSize = 0;
387 : GByte nMajorVersion = 0;
388 : GByte nMinorVersion = 0;
389 : GUInt16 nSectorSize = 512;
390 : VSICryptAlg eAlg = ALG_AES;
391 : VSICryptMode eMode = MODE_CBC;
392 : CPLString osIV{};
393 : bool bAddKeyCheck = false;
394 : GUIntBig nPayloadFileSize = 0;
395 : CPLString osFreeText{};
396 : CPLString osExtraContent{};
397 : };
398 :
399 : /************************************************************************/
400 : /* VSICryptReadError() */
401 : /************************************************************************/
402 :
403 51 : static bool VSICryptReadError()
404 : {
405 51 : CPLError(CE_Failure, CPLE_FileIO, "Cannot read header");
406 51 : return false;
407 : }
408 :
409 : /************************************************************************/
410 : /* VSICryptGenerateSectorIV() */
411 : /************************************************************************/
412 :
413 : // TODO(rouault): This function really needs a comment saying what it does.
414 145344 : static std::string VSICryptGenerateSectorIV(const std::string &osIV,
415 : vsi_l_offset nOffset)
416 : {
417 145344 : std::string osSectorIV(osIV);
418 145344 : const size_t nLength = std::min(sizeof(vsi_l_offset), osSectorIV.size());
419 1308100 : for (size_t i = 0; i < nLength; i++)
420 : {
421 : // TODO(rouault): Explain what this block is trying to do?
422 1162750 : osSectorIV[i] = static_cast<char>((osSectorIV[i] ^ nOffset) & 0xff);
423 1162750 : nOffset >>= 8;
424 : }
425 145344 : return osSectorIV;
426 : }
427 :
428 : /************************************************************************/
429 : /* CryptKeyCheck() */
430 : /************************************************************************/
431 :
432 : std::string
433 14 : VSICryptFileHeader::CryptKeyCheck(CryptoPP::BlockCipher *poEncCipher)
434 : {
435 28 : std::string osKeyCheckRes;
436 :
437 14 : CPLAssert(osIV.size() == poEncCipher->BlockSize());
438 : // Generate a unique IV with a sector offset of 0xFFFFFFFFFFFFFFFF.
439 : std::string osCheckIV(
440 28 : VSICryptGenerateSectorIV(osIV, ~(static_cast<vsi_l_offset>(0))));
441 : CryptoPP::StreamTransformation *poMode;
442 : try
443 : {
444 28 : poMode = new CryptoPP::CBC_Mode_ExternalCipher::Encryption(
445 : *poEncCipher,
446 14 : reinterpret_cast<const cryptopp_byte *>(osCheckIV.c_str()));
447 : }
448 0 : catch (const std::exception &e)
449 : {
450 0 : CPLError(CE_Failure, CPLE_AppDefined, "CryptoPP exception: %s",
451 0 : e.what());
452 0 : return std::string();
453 : }
454 14 : CryptoPP::StringSink *poSink = new CryptoPP::StringSink(osKeyCheckRes);
455 : CryptoPP::StreamTransformationFilter *poEnc =
456 : new CryptoPP::StreamTransformationFilter(
457 14 : *poMode, poSink, CryptoPP::StreamTransformationFilter::NO_PADDING);
458 : // Not sure if it is add extra security, but pick up something that is
459 : // unlikely to be a plain text (random number).
460 28 : poEnc->Put(
461 : reinterpret_cast<const cryptopp_byte *>(
462 : "\xDB\x31\xB9\x1B\xD3\x1C\xFA\x3E\x84\x06\xC1\x42\xC3\xEC\xCD\x9A"
463 : "\x02\x36\x22\x15\x58\x88\x74\x65\x00\x2F\x98\xBC\x69\x22\xE1\x63"),
464 14 : std::min(32U, poEncCipher->BlockSize()));
465 14 : poEnc->MessageEnd();
466 14 : delete poEnc;
467 14 : delete poMode;
468 :
469 14 : return osKeyCheckRes;
470 : }
471 :
472 : /************************************************************************/
473 : /* ReadFromFile() */
474 : /************************************************************************/
475 :
476 235 : int VSICryptFileHeader::ReadFromFile(VSIVirtualHandle *fp,
477 : const CPLString &osKey)
478 : {
479 235 : GByte abySignature[8] = {};
480 235 : fp->Seek(0, SEEK_SET);
481 235 : CPL_STATIC_ASSERT(sizeof(VSICRYPT_SIGNATURE) == 8 + 1);
482 461 : if (fp->Read(abySignature, 8, 1) == 0 ||
483 226 : memcmp(abySignature, VSICRYPT_SIGNATURE, 8) != 0)
484 : {
485 33 : CPLError(CE_Failure, CPLE_AppDefined, "Invalid signature");
486 33 : return FALSE;
487 : }
488 :
489 202 : if (fp->Read(&nHeaderSize, 2, 1) == 0)
490 2 : return VSICryptReadError();
491 200 : nHeaderSize = CPL_LSBWORD16(nHeaderSize);
492 200 : if (nHeaderSize < 8 + 2 + 1 + 1)
493 : {
494 1 : CPLError(CE_Failure, CPLE_AppDefined, "Invalid header size : %d",
495 1 : nHeaderSize);
496 1 : return FALSE;
497 : }
498 :
499 199 : if (fp->Read(&nMajorVersion, 1, 1) == 0)
500 1 : return VSICryptReadError();
501 198 : if (fp->Read(&nMinorVersion, 1, 1) == 0)
502 1 : return VSICryptReadError();
503 :
504 197 : if (nMajorVersion != VSICRYPT_CURRENT_MAJOR)
505 : {
506 3 : CPLError(CE_Failure, CPLE_AppDefined, "Unhandled major version : %d",
507 3 : nMajorVersion);
508 3 : return FALSE;
509 : }
510 194 : if (nMinorVersion != VSICRYPT_CURRENT_MINOR)
511 : {
512 2 : CPLDebug("VSICRYPT", "Minor version in file is %d", nMinorVersion);
513 : }
514 :
515 194 : if (fp->Read(&nSectorSize, 2, 1) == 0)
516 2 : return VSICryptReadError();
517 192 : nSectorSize = CPL_LSBWORD16(nSectorSize);
518 :
519 : GByte nAlg, nMode;
520 192 : if (fp->Read(&nAlg, 1, 1) == 0 || fp->Read(&nMode, 1, 1) == 0)
521 2 : return VSICryptReadError();
522 190 : if (nAlg > ALG_MAX)
523 : {
524 2 : CPLError(CE_Failure, CPLE_NotSupported,
525 : "Unsupported cipher algorithm %d", nAlg);
526 2 : return FALSE;
527 : }
528 188 : if (nMode > MODE_MAX)
529 : {
530 2 : CPLError(CE_Failure, CPLE_NotSupported,
531 : "Unsupported cipher block mode %d", nMode);
532 2 : return FALSE;
533 : }
534 186 : eAlg = static_cast<VSICryptAlg>(nAlg);
535 186 : eMode = static_cast<VSICryptMode>(nMode);
536 :
537 : GByte nIVSize;
538 186 : if (fp->Read(&nIVSize, 1, 1) == 0)
539 1 : return VSICryptReadError();
540 :
541 185 : osIV.resize(nIVSize);
542 185 : if (fp->Read(osIV.data(), 1, nIVSize) != nIVSize)
543 18 : return VSICryptReadError();
544 :
545 : GUInt16 nFreeTextSize;
546 167 : if (fp->Read(&nFreeTextSize, 2, 1) == 0)
547 2 : return VSICryptReadError();
548 :
549 165 : osFreeText.resize(nFreeTextSize);
550 165 : if (fp->Read(osFreeText.data(), 1, nFreeTextSize) != nFreeTextSize)
551 5 : return VSICryptReadError();
552 :
553 : GByte nKeyCheckSize;
554 160 : if (fp->Read(&nKeyCheckSize, 1, 1) == 0)
555 1 : return VSICryptReadError();
556 159 : bAddKeyCheck = nKeyCheckSize != 0;
557 159 : if (nKeyCheckSize)
558 : {
559 11 : CPLString osKeyCheck;
560 11 : osKeyCheck.resize(nKeyCheckSize);
561 11 : if (fp->Read(osKeyCheck.data(), 1, nKeyCheckSize) != nKeyCheckSize)
562 2 : return VSICryptReadError();
563 :
564 9 : if (osKey.empty() && pabyGlobalKey == nullptr)
565 : {
566 1 : CPLError(CE_Failure, CPLE_AppDefined,
567 : "Encryption key not defined as key/key_b64 parameter, "
568 : "VSICRYPT_KEY/VSICRYPT_KEY_B64 configuration option or "
569 : "VSISetCryptKey() API");
570 1 : return FALSE;
571 : }
572 :
573 8 : CryptoPP::BlockCipher *poEncCipher = GetEncBlockCipher(eAlg);
574 8 : if (poEncCipher == nullptr)
575 0 : return FALSE;
576 :
577 8 : if (osIV.size() != poEncCipher->BlockSize())
578 : {
579 1 : CPLError(CE_Failure, CPLE_AppDefined,
580 : "Inconsistent initial vector");
581 1 : delete poEncCipher;
582 1 : return FALSE;
583 : }
584 :
585 7 : int nMaxKeySize = static_cast<int>(poEncCipher->MaxKeyLength());
586 :
587 : try
588 : {
589 7 : if (!osKey.empty())
590 : {
591 : const int nKeySize =
592 6 : std::min(nMaxKeySize, static_cast<int>(osKey.size()));
593 6 : poEncCipher->SetKey(
594 6 : reinterpret_cast<const cryptopp_byte *>(osKey.c_str()),
595 6 : nKeySize);
596 : }
597 1 : else if (pabyGlobalKey)
598 : {
599 1 : const int nKeySize = std::min(nMaxKeySize, nGlobalKeySize);
600 1 : poEncCipher->SetKey(pabyGlobalKey, nKeySize);
601 : }
602 : }
603 1 : catch (const std::exception &e)
604 : {
605 1 : CPLError(CE_Failure, CPLE_AppDefined, "CryptoPP exception: %s",
606 1 : e.what());
607 1 : delete poEncCipher;
608 1 : return FALSE;
609 : }
610 :
611 6 : std::string osKeyCheckRes = CryptKeyCheck(poEncCipher);
612 :
613 6 : delete poEncCipher;
614 :
615 12 : if (osKeyCheck.size() != osKeyCheckRes.size() ||
616 6 : memcmp(osKeyCheck.c_str(), osKeyCheckRes.c_str(),
617 : osKeyCheck.size()) != 0)
618 : {
619 2 : CPLError(CE_Failure, CPLE_AppDefined, "Bad key");
620 2 : return FALSE;
621 : }
622 : }
623 :
624 152 : if (fp->Read(&nPayloadFileSize, 8, 1) == 0)
625 8 : return VSICryptReadError();
626 144 : CPL_LSBPTR64(&nPayloadFileSize);
627 : #ifdef VERBOSE_VSICRYPT
628 : CPLDebug("VSICRYPT", "nPayloadFileSize read = " CPL_FRMT_GUIB,
629 : nPayloadFileSize);
630 : #endif
631 :
632 144 : GUInt16 nExtraContentSize = 0;
633 144 : if (fp->Read(&nExtraContentSize, 2, 1) == 0)
634 2 : return VSICryptReadError();
635 142 : nExtraContentSize = CPL_LSBWORD16(nExtraContentSize);
636 :
637 142 : osExtraContent.resize(nExtraContentSize);
638 142 : if (fp->Read(osExtraContent.data(), 1, nExtraContentSize) !=
639 142 : nExtraContentSize)
640 4 : return VSICryptReadError();
641 :
642 138 : return TRUE;
643 : }
644 :
645 : /************************************************************************/
646 : /* WriteToFile() */
647 : /************************************************************************/
648 :
649 2075 : int VSICryptFileHeader::WriteToFile(VSIVirtualHandle *fp,
650 : CryptoPP::BlockCipher *poEncCipher)
651 : {
652 2075 : fp->Seek(0, SEEK_SET);
653 :
654 2075 : bool bRet = fp->Write(VSICRYPT_SIGNATURE, 8, 1) == 1;
655 :
656 2075 : std::string osKeyCheckRes;
657 2075 : if (bAddKeyCheck)
658 : {
659 8 : osKeyCheckRes = CryptKeyCheck(poEncCipher);
660 : }
661 :
662 : GUInt16 nHeaderSizeNew =
663 : static_cast<GUInt16>(8 + /* signature */
664 : 2 + /* header size */
665 : 1 + /* major version */
666 : 1 + /* minor version */
667 : 2 + /* sector size */
668 : 1 + /* alg */
669 : 1 + /* mode */
670 2075 : 1 + osIV.size() + /* IV */
671 2075 : 2 + osFreeText.size() + /* free text */
672 2075 : 1 + osKeyCheckRes.size() + /* key check */
673 : 8 + /* payload size */
674 2075 : 2 + osExtraContent.size()); /* extra content */
675 2075 : if (nHeaderSize != 0)
676 1038 : CPLAssert(nHeaderSizeNew == nHeaderSize);
677 : else
678 1037 : nHeaderSize = nHeaderSizeNew;
679 :
680 2075 : GUInt16 nHeaderSizeToWrite = CPL_LSBWORD16(nHeaderSizeNew);
681 2075 : bRet &= (fp->Write(&nHeaderSizeToWrite, 2, 1) == 1);
682 :
683 2075 : GByte nMajorVersionToWrite = VSICRYPT_CURRENT_MAJOR;
684 2075 : bRet &= (fp->Write(&nMajorVersionToWrite, 1, 1) == 1);
685 :
686 2075 : GByte nMinorVersionToWrite = VSICRYPT_CURRENT_MINOR;
687 2075 : bRet &= (fp->Write(&nMinorVersionToWrite, 1, 1) == 1);
688 :
689 2075 : GUInt16 nSectorSizeToWrite = CPL_LSBWORD16(nSectorSize);
690 2075 : bRet &= (fp->Write(&nSectorSizeToWrite, 2, 1) == 1);
691 :
692 2075 : GByte nAlg = static_cast<GByte>(eAlg);
693 2075 : bRet &= (fp->Write(&nAlg, 1, 1) == 1);
694 :
695 2075 : GByte nMode = static_cast<GByte>(eMode);
696 2075 : bRet &= (fp->Write(&nMode, 1, 1) == 1);
697 :
698 2075 : GByte nIVSizeToWrite = static_cast<GByte>(osIV.size());
699 2075 : CPLAssert(nIVSizeToWrite == osIV.size());
700 2075 : bRet &= (fp->Write(&nIVSizeToWrite, 1, 1) == 1);
701 2075 : bRet &= (fp->Write(osIV.c_str(), 1, osIV.size()) == osIV.size());
702 :
703 : GUInt16 nFreeTextSizeToWrite =
704 2075 : CPL_LSBWORD16(static_cast<GUInt16>(osFreeText.size()));
705 2075 : bRet &= (fp->Write(&nFreeTextSizeToWrite, 2, 1) == 1);
706 2075 : bRet &= (fp->Write(osFreeText.c_str(), 1, osFreeText.size()) ==
707 2075 : osFreeText.size());
708 :
709 2075 : GByte nSize = static_cast<GByte>(osKeyCheckRes.size());
710 2075 : bRet &= (fp->Write(&nSize, 1, 1) == 1);
711 2075 : bRet &= (fp->Write(osKeyCheckRes.c_str(), 1, osKeyCheckRes.size()) ==
712 2075 : osKeyCheckRes.size());
713 :
714 2075 : GUIntBig nPayloadFileSizeToWrite = nPayloadFileSize;
715 2075 : CPL_LSBPTR64(&nPayloadFileSizeToWrite);
716 2075 : bRet &= (fp->Write(&nPayloadFileSizeToWrite, 8, 1) == 1);
717 :
718 : GUInt16 nExtraContentSizeToWrite =
719 2075 : CPL_LSBWORD16(static_cast<GUInt16>(osExtraContent.size()));
720 2075 : bRet &= (fp->Write(&nExtraContentSizeToWrite, 2, 1) == 1);
721 2075 : bRet &= (fp->Write(osExtraContent.c_str(), 1, osExtraContent.size()) ==
722 2075 : osExtraContent.size());
723 :
724 2075 : CPLAssert(fp->Tell() == nHeaderSize);
725 :
726 4150 : return bRet;
727 : }
728 :
729 : /************************************************************************/
730 : /* VSICryptFileHandle */
731 : /************************************************************************/
732 :
733 : class VSICryptFileHandle final : public VSIVirtualHandle
734 : {
735 : CPL_DISALLOW_COPY_ASSIGN(VSICryptFileHandle)
736 :
737 : private:
738 : CPLString osBaseFilename{};
739 : int nPerms = 0;
740 : VSIVirtualHandle *poBaseHandle = nullptr;
741 : VSICryptFileHeader *poHeader = nullptr;
742 : bool bUpdateHeader = false;
743 : vsi_l_offset nCurPos = 0;
744 : bool bEOF = false;
745 : bool bError = false;
746 :
747 : CryptoPP::BlockCipher *poEncCipher = nullptr;
748 : CryptoPP::BlockCipher *poDecCipher = nullptr;
749 : int nBlockSize = 0;
750 :
751 : vsi_l_offset nWBOffset = 0;
752 : GByte *pabyWB = nullptr;
753 : int nWBSize = 0;
754 : bool bWBDirty = false;
755 :
756 : bool bLastSectorWasModified = false;
757 :
758 : void EncryptBlock(GByte *pabyData, vsi_l_offset nOffset);
759 : bool DecryptBlock(GByte *pabyData, vsi_l_offset nOffset);
760 : bool FlushDirty();
761 :
762 : public:
763 : VSICryptFileHandle(const CPLString &osBaseFilename,
764 : VSIVirtualHandle *poBaseHandle,
765 : VSICryptFileHeader *poHeader, int nPerms);
766 : ~VSICryptFileHandle() override;
767 :
768 : int Init(const CPLString &osKey, bool bWriteHeader = false);
769 :
770 : int Seek(vsi_l_offset nOffset, int nWhence) override;
771 : vsi_l_offset Tell() override;
772 : size_t Read(void *pBuffer, size_t nSize, size_t nMemb) override;
773 : size_t Write(const void *pBuffer, size_t nSize, size_t nMemb) override;
774 : int Eof() override;
775 : int Error() override;
776 : void ClearErr() override;
777 : int Flush() override;
778 : int Close() override;
779 : int Truncate(vsi_l_offset nNewSize) override;
780 : };
781 :
782 : /************************************************************************/
783 : /* VSICryptFileHandle() */
784 : /************************************************************************/
785 :
786 1174 : VSICryptFileHandle::VSICryptFileHandle(const CPLString &osBaseFilenameIn,
787 : VSIVirtualHandle *poBaseHandleIn,
788 : VSICryptFileHeader *poHeaderIn,
789 1174 : int nPermsIn)
790 : : osBaseFilename(osBaseFilenameIn), nPerms(nPermsIn),
791 1174 : poBaseHandle(poBaseHandleIn), poHeader(poHeaderIn)
792 : {
793 1174 : }
794 :
795 : /************************************************************************/
796 : /* ~VSICryptFileHandle() */
797 : /************************************************************************/
798 :
799 2348 : VSICryptFileHandle::~VSICryptFileHandle()
800 : {
801 1174 : Close();
802 1174 : delete poHeader;
803 1174 : delete poEncCipher;
804 1174 : delete poDecCipher;
805 1174 : CPLFree(pabyWB);
806 2348 : }
807 :
808 : /************************************************************************/
809 : /* Init() */
810 : /************************************************************************/
811 :
812 1174 : int VSICryptFileHandle::Init(const CPLString &osKey, bool bWriteHeader)
813 : {
814 1174 : poEncCipher = GetEncBlockCipher(poHeader->eAlg);
815 1174 : if (poEncCipher == nullptr)
816 : {
817 0 : CPLError(CE_Failure, CPLE_AppDefined,
818 : "Cipher algorithm not supported in this build: %d",
819 0 : static_cast<int>(poHeader->eAlg));
820 0 : return FALSE;
821 : }
822 :
823 1174 : if (poHeader->osIV.size() != poEncCipher->BlockSize())
824 : {
825 1 : CPLError(CE_Failure, CPLE_AppDefined, "Inconsistent initial vector");
826 1 : return FALSE;
827 : }
828 :
829 1173 : poDecCipher = GetDecBlockCipher(poHeader->eAlg);
830 1173 : nBlockSize = poEncCipher->BlockSize();
831 1173 : int nMaxKeySize = static_cast<int>(poEncCipher->MaxKeyLength());
832 :
833 : try
834 : {
835 1173 : if (!osKey.empty())
836 : {
837 : const int nKeySize =
838 1168 : std::min(nMaxKeySize, static_cast<int>(osKey.size()));
839 1168 : poEncCipher->SetKey(
840 1168 : reinterpret_cast<const cryptopp_byte *>(osKey.c_str()),
841 1168 : nKeySize);
842 1167 : poDecCipher->SetKey(
843 1167 : reinterpret_cast<const cryptopp_byte *>(osKey.c_str()),
844 1167 : nKeySize);
845 : }
846 5 : else if (pabyGlobalKey)
847 : {
848 5 : const int nKeySize = std::min(nMaxKeySize, nGlobalKeySize);
849 5 : poEncCipher->SetKey(pabyGlobalKey, nKeySize);
850 4 : poDecCipher->SetKey(pabyGlobalKey, nKeySize);
851 : }
852 : else
853 0 : return FALSE;
854 : }
855 2 : catch (const std::exception &e)
856 : {
857 2 : CPLError(CE_Failure, CPLE_AppDefined, "CryptoPP exception: %s",
858 2 : e.what());
859 2 : return FALSE;
860 : }
861 :
862 1171 : pabyWB = static_cast<GByte *>(CPLCalloc(1, poHeader->nSectorSize));
863 :
864 1171 : if ((poHeader->nSectorSize % nBlockSize) != 0)
865 : {
866 3 : CPLError(CE_Failure, CPLE_AppDefined,
867 : "Sector size (%d) is not a multiple of block size (%d)",
868 3 : poHeader->nSectorSize, nBlockSize);
869 3 : return FALSE;
870 : }
871 1168 : if (poHeader->eMode == MODE_CBC_CTS &&
872 3 : poHeader->nSectorSize < 2 * nBlockSize)
873 : {
874 1 : CPLError(CE_Failure, CPLE_AppDefined,
875 : "Sector size (%d) should be at least twice larger than "
876 : "the block size (%d) in CBC_CTS.",
877 1 : poHeader->nSectorSize, nBlockSize);
878 1 : return FALSE;
879 : }
880 :
881 1167 : if (bWriteHeader && !poHeader->WriteToFile(poBaseHandle, poEncCipher))
882 : {
883 0 : return FALSE;
884 : }
885 :
886 1167 : return TRUE;
887 : }
888 :
889 : /************************************************************************/
890 : /* EncryptBlock() */
891 : /************************************************************************/
892 :
893 70111 : void VSICryptFileHandle::EncryptBlock(GByte *pabyData, vsi_l_offset nOffset)
894 : {
895 70111 : std::string osRes;
896 70111 : std::string osIV(VSICryptGenerateSectorIV(poHeader->osIV, nOffset));
897 70111 : CPLAssert(static_cast<int>(osIV.size()) == nBlockSize);
898 :
899 : CryptoPP::StreamTransformation *poMode;
900 : try
901 : {
902 70111 : if (poHeader->eMode == MODE_CBC)
903 70107 : poMode = new CryptoPP::CBC_Mode_ExternalCipher::Encryption(
904 70107 : *poEncCipher,
905 70107 : reinterpret_cast<const cryptopp_byte *>(osIV.c_str()));
906 4 : else if (poHeader->eMode == MODE_CFB)
907 1 : poMode = new CryptoPP::CFB_Mode_ExternalCipher::Encryption(
908 1 : *poEncCipher,
909 1 : reinterpret_cast<const cryptopp_byte *>(osIV.c_str()));
910 3 : else if (poHeader->eMode == MODE_OFB)
911 1 : poMode = new CryptoPP::OFB_Mode_ExternalCipher::Encryption(
912 1 : *poEncCipher,
913 1 : reinterpret_cast<const cryptopp_byte *>(osIV.c_str()));
914 2 : else if (poHeader->eMode == MODE_CTR)
915 1 : poMode = new CryptoPP::CTR_Mode_ExternalCipher::Encryption(
916 1 : *poEncCipher,
917 1 : reinterpret_cast<const cryptopp_byte *>(osIV.c_str()));
918 : else
919 1 : poMode = new CryptoPP::CBC_CTS_Mode_ExternalCipher::Encryption(
920 1 : *poEncCipher,
921 1 : reinterpret_cast<const cryptopp_byte *>(osIV.c_str()));
922 : }
923 0 : catch (const std::exception &e)
924 : {
925 0 : CPLError(CE_Failure, CPLE_AppDefined, "cryptopp exception: %s",
926 0 : e.what());
927 0 : return;
928 : }
929 70111 : CryptoPP::StringSink *poSink = new CryptoPP::StringSink(osRes);
930 : CryptoPP::StreamTransformationFilter *poEnc =
931 : new CryptoPP::StreamTransformationFilter(
932 70111 : *poMode, poSink, CryptoPP::StreamTransformationFilter::NO_PADDING);
933 70111 : poEnc->Put(pabyData, poHeader->nSectorSize);
934 70111 : poEnc->MessageEnd();
935 70111 : delete poEnc;
936 :
937 70111 : delete poMode;
938 :
939 70111 : CPLAssert(static_cast<int>(osRes.length()) == poHeader->nSectorSize);
940 70111 : memcpy(pabyData, osRes.c_str(), osRes.length());
941 : }
942 :
943 : /************************************************************************/
944 : /* DecryptBlock() */
945 : /************************************************************************/
946 :
947 75219 : bool VSICryptFileHandle::DecryptBlock(GByte *pabyData, vsi_l_offset nOffset)
948 : {
949 150438 : std::string osRes;
950 150438 : std::string osIV(VSICryptGenerateSectorIV(poHeader->osIV, nOffset));
951 75219 : CPLAssert(static_cast<int>(osIV.size()) == nBlockSize);
952 75219 : CryptoPP::StringSink *poSink = new CryptoPP::StringSink(osRes);
953 75219 : CryptoPP::StreamTransformation *poMode = nullptr;
954 75219 : CryptoPP::StreamTransformationFilter *poDec = nullptr;
955 :
956 : try
957 : {
958 : // Yes, some modes need the encryption cipher.
959 75219 : if (poHeader->eMode == MODE_CBC)
960 75215 : poMode = new CryptoPP::CBC_Mode_ExternalCipher::Decryption(
961 75215 : *poDecCipher,
962 75215 : reinterpret_cast<const cryptopp_byte *>(osIV.c_str()));
963 4 : else if (poHeader->eMode == MODE_CFB)
964 1 : poMode = new CryptoPP::CFB_Mode_ExternalCipher::Decryption(
965 1 : *poEncCipher,
966 1 : reinterpret_cast<const cryptopp_byte *>(osIV.c_str()));
967 3 : else if (poHeader->eMode == MODE_OFB)
968 1 : poMode = new CryptoPP::OFB_Mode_ExternalCipher::Decryption(
969 1 : *poEncCipher,
970 1 : reinterpret_cast<const cryptopp_byte *>(osIV.c_str()));
971 2 : else if (poHeader->eMode == MODE_CTR)
972 1 : poMode = new CryptoPP::CTR_Mode_ExternalCipher::Decryption(
973 1 : *poEncCipher,
974 1 : reinterpret_cast<const cryptopp_byte *>(osIV.c_str()));
975 : else
976 1 : poMode = new CryptoPP::CBC_CTS_Mode_ExternalCipher::Decryption(
977 1 : *poDecCipher,
978 1 : reinterpret_cast<const cryptopp_byte *>(osIV.c_str()));
979 75219 : poDec = new CryptoPP::StreamTransformationFilter(
980 75219 : *poMode, poSink, CryptoPP::StreamTransformationFilter::NO_PADDING);
981 75219 : poDec->Put(reinterpret_cast<const cryptopp_byte *>(pabyData),
982 75219 : poHeader->nSectorSize);
983 75219 : poDec->MessageEnd();
984 75219 : delete poDec;
985 75219 : delete poMode;
986 : }
987 0 : catch (const std::exception &e)
988 : {
989 0 : delete poDec;
990 0 : delete poMode;
991 :
992 0 : CPLError(CE_Failure, CPLE_AppDefined, "CryptoPP exception: %s",
993 0 : e.what());
994 0 : return false;
995 : }
996 :
997 75219 : CPLAssert(static_cast<int>(osRes.length()) == poHeader->nSectorSize);
998 75219 : memcpy(pabyData, osRes.c_str(), osRes.length());
999 :
1000 75219 : return true;
1001 : }
1002 :
1003 : /************************************************************************/
1004 : /* FlushDirty() */
1005 : /************************************************************************/
1006 :
1007 55328 : bool VSICryptFileHandle::FlushDirty()
1008 : {
1009 55328 : if (!bWBDirty)
1010 11097 : return true;
1011 44231 : bWBDirty = false;
1012 :
1013 44231 : EncryptBlock(pabyWB, nWBOffset);
1014 44231 : poBaseHandle->Seek(poHeader->nHeaderSize + nWBOffset, SEEK_SET);
1015 :
1016 44231 : nWBOffset = 0;
1017 44231 : nWBSize = 0;
1018 :
1019 44231 : if (poBaseHandle->Write(pabyWB, poHeader->nSectorSize, 1) != 1)
1020 0 : return false;
1021 :
1022 44231 : return true;
1023 : }
1024 :
1025 : /************************************************************************/
1026 : /* Seek() */
1027 : /************************************************************************/
1028 :
1029 30973 : int VSICryptFileHandle::Seek(vsi_l_offset nOffset, int nWhence)
1030 : {
1031 : #ifdef VERBOSE_VSICRYPT
1032 : CPLDebug("VSICRYPT", "Seek(nOffset=" CPL_FRMT_GUIB ", nWhence=%d)", nOffset,
1033 : nWhence);
1034 : #endif
1035 :
1036 30973 : bEOF = false;
1037 :
1038 30973 : if (nWhence == SEEK_SET)
1039 30971 : nCurPos = nOffset;
1040 2 : else if (nWhence == SEEK_CUR)
1041 0 : nCurPos += nOffset;
1042 : else
1043 2 : nCurPos = poHeader->nPayloadFileSize;
1044 30973 : return 0;
1045 : }
1046 :
1047 : /************************************************************************/
1048 : /* Tell() */
1049 : /************************************************************************/
1050 :
1051 3 : vsi_l_offset VSICryptFileHandle::Tell()
1052 : {
1053 : #ifdef VERBOSE_VSICRYPT
1054 : CPLDebug("VSICRYPT", "Tell()=" CPL_FRMT_GUIB, nCurPos);
1055 : #endif
1056 3 : return nCurPos;
1057 : }
1058 :
1059 : /************************************************************************/
1060 : /* Read() */
1061 : /************************************************************************/
1062 :
1063 11005 : size_t VSICryptFileHandle::Read(void *pBuffer, size_t nSize, size_t nMemb)
1064 : {
1065 11005 : size_t nToRead = nSize * nMemb;
1066 11005 : GByte *pabyBuffer = static_cast<GByte *>(pBuffer);
1067 :
1068 : #ifdef VERBOSE_VSICRYPT
1069 : CPLDebug("VSICRYPT", "Read(nCurPos=" CPL_FRMT_GUIB ", nToRead=%d)", nCurPos,
1070 : static_cast<int>(nToRead));
1071 : #endif
1072 :
1073 11005 : if ((nPerms & VSICRYPT_READ) == 0)
1074 : {
1075 1 : bError = true;
1076 1 : return 0;
1077 : }
1078 :
1079 11004 : if (nCurPos >= poHeader->nPayloadFileSize)
1080 : {
1081 3882 : bEOF = true;
1082 3882 : return 0;
1083 : }
1084 :
1085 7122 : if (!FlushDirty())
1086 0 : return 0;
1087 :
1088 52089 : while (nToRead > 0)
1089 : {
1090 52089 : if (nCurPos >= nWBOffset && nCurPos < nWBOffset + nWBSize)
1091 : {
1092 : // TODO(schwehr): Can nToCopy be a size_t to simplify casting?
1093 : int nToCopy =
1094 89938 : std::min(static_cast<int>(nToRead),
1095 44969 : static_cast<int>(nWBSize - (nCurPos - nWBOffset)));
1096 44969 : if (nCurPos + nToCopy > poHeader->nPayloadFileSize)
1097 : {
1098 1234 : bEOF = true;
1099 1234 : nToCopy =
1100 1234 : static_cast<int>(poHeader->nPayloadFileSize - nCurPos);
1101 : }
1102 44969 : memcpy(pabyBuffer, pabyWB + nCurPos - nWBOffset, nToCopy);
1103 44969 : pabyBuffer += nToCopy;
1104 44969 : nToRead -= nToCopy;
1105 44969 : nCurPos += nToCopy;
1106 44969 : if (bEOF || nToRead == 0)
1107 : break;
1108 37888 : CPLAssert((nCurPos % poHeader->nSectorSize) == 0);
1109 : }
1110 :
1111 45008 : vsi_l_offset nSectorOffset =
1112 45008 : (nCurPos / poHeader->nSectorSize) * poHeader->nSectorSize;
1113 45008 : poBaseHandle->Seek(poHeader->nHeaderSize + nSectorOffset, SEEK_SET);
1114 45008 : if (poBaseHandle->Read(pabyWB, poHeader->nSectorSize, 1) != 1)
1115 : {
1116 41 : bEOF = poBaseHandle->Eof();
1117 41 : bError = poBaseHandle->Error();
1118 41 : break;
1119 : }
1120 44967 : if (!DecryptBlock(pabyWB, nSectorOffset))
1121 : {
1122 0 : bError = true;
1123 0 : break;
1124 : }
1125 44967 : if ((nPerms & VSICRYPT_WRITE) &&
1126 44933 : nSectorOffset + poHeader->nSectorSize > poHeader->nPayloadFileSize)
1127 : {
1128 : // If the last sector was padded with random values, decrypt it to 0
1129 : // in case of update scenarios.
1130 1347 : CPLAssert(nSectorOffset < poHeader->nPayloadFileSize);
1131 1347 : memset(pabyWB + poHeader->nPayloadFileSize - nSectorOffset, 0,
1132 1347 : nSectorOffset + poHeader->nSectorSize -
1133 1347 : poHeader->nPayloadFileSize);
1134 : }
1135 44967 : nWBOffset = nSectorOffset;
1136 44967 : nWBSize = poHeader->nSectorSize;
1137 : }
1138 :
1139 7122 : int nRet = static_cast<int>((nSize * nMemb - nToRead) / nSize);
1140 : #ifdef VERBOSE_VSICRYPT
1141 : CPLDebug("VSICRYPT", "Read ret = %d (nMemb = %d)", nRet,
1142 : static_cast<int>(nMemb));
1143 : #endif
1144 7122 : return nRet;
1145 : }
1146 :
1147 : /************************************************************************/
1148 : /* Write() */
1149 : /************************************************************************/
1150 :
1151 20040 : size_t VSICryptFileHandle::Write(const void *pBuffer, size_t nSize,
1152 : size_t nMemb)
1153 : {
1154 20040 : size_t nToWrite = nSize * nMemb;
1155 20040 : const GByte *pabyBuffer = static_cast<const GByte *>(pBuffer);
1156 :
1157 : #ifdef VERBOSE_VSICRYPT
1158 : CPLDebug("VSICRYPT",
1159 : "Write(nCurPos=" CPL_FRMT_GUIB ", nToWrite=%d,"
1160 : "nPayloadFileSize=" CPL_FRMT_GUIB
1161 : ",bWBDirty=%d,nWBOffset=" CPL_FRMT_GUIB ",nWBSize=%d)",
1162 : nCurPos, static_cast<int>(nToWrite), poHeader->nPayloadFileSize,
1163 : static_cast<int>(bWBDirty), nWBOffset, nWBSize);
1164 : #endif
1165 :
1166 20040 : if ((nPerms & VSICRYPT_WRITE) == 0)
1167 1 : return 0;
1168 :
1169 20039 : if (nCurPos >= (poHeader->nPayloadFileSize / poHeader->nSectorSize) *
1170 20039 : poHeader->nSectorSize)
1171 : {
1172 3266 : bLastSectorWasModified = true;
1173 : }
1174 :
1175 : // If seeking past end of file, we need to explicitly encrypt the
1176 : // padding zeroes.
1177 20039 : if (nCurPos > poHeader->nPayloadFileSize && nCurPos > nWBOffset + nWBSize)
1178 : {
1179 2962 : if (!FlushDirty())
1180 0 : return 0;
1181 2962 : vsi_l_offset nOffset =
1182 2962 : (poHeader->nPayloadFileSize + poHeader->nSectorSize - 1) /
1183 2962 : poHeader->nSectorSize * poHeader->nSectorSize;
1184 2962 : const vsi_l_offset nEndOffset =
1185 2962 : nCurPos / poHeader->nSectorSize * poHeader->nSectorSize;
1186 27578 : for (; nOffset < nEndOffset; nOffset += poHeader->nSectorSize)
1187 : {
1188 24616 : memset(pabyWB, 0, poHeader->nSectorSize);
1189 24616 : EncryptBlock(pabyWB, nOffset);
1190 24616 : poBaseHandle->Seek(poHeader->nHeaderSize + nOffset, SEEK_SET);
1191 24616 : if (poBaseHandle->Write(pabyWB, poHeader->nSectorSize, 1) != 1)
1192 0 : return 0;
1193 24616 : poHeader->nPayloadFileSize = nOffset + poHeader->nSectorSize;
1194 24616 : bUpdateHeader = true;
1195 : }
1196 : }
1197 :
1198 79522 : while (nToWrite > 0)
1199 : {
1200 79133 : if (nCurPos >= nWBOffset && nCurPos < nWBOffset + nWBSize)
1201 : {
1202 35066 : bWBDirty = true;
1203 : const int nToCopy =
1204 70132 : std::min(static_cast<int>(nToWrite),
1205 35066 : static_cast<int>(nWBSize - (nCurPos - nWBOffset)));
1206 35066 : memcpy(pabyWB + nCurPos - nWBOffset, pabyBuffer, nToCopy);
1207 35066 : pabyBuffer += nToCopy;
1208 35066 : nToWrite -= nToCopy;
1209 35066 : nCurPos += nToCopy;
1210 35066 : if (nCurPos > poHeader->nPayloadFileSize)
1211 : {
1212 6102 : bUpdateHeader = true;
1213 6102 : poHeader->nPayloadFileSize = nCurPos;
1214 : }
1215 35066 : if (nToWrite == 0)
1216 19650 : break;
1217 15416 : CPLAssert((nCurPos % poHeader->nSectorSize) == 0);
1218 : }
1219 44067 : else if ((nCurPos % poHeader->nSectorSize) == 0 &&
1220 25241 : nToWrite >= static_cast<size_t>(poHeader->nSectorSize))
1221 : {
1222 9558 : if (!FlushDirty())
1223 0 : break;
1224 :
1225 9558 : bWBDirty = true;
1226 9558 : nWBOffset = nCurPos;
1227 9558 : nWBSize = poHeader->nSectorSize;
1228 9558 : memcpy(pabyWB, pabyBuffer, poHeader->nSectorSize);
1229 9558 : pabyBuffer += poHeader->nSectorSize;
1230 9558 : nToWrite -= poHeader->nSectorSize;
1231 9558 : nCurPos += poHeader->nSectorSize;
1232 9558 : if (nCurPos > poHeader->nPayloadFileSize)
1233 : {
1234 1837 : bUpdateHeader = true;
1235 1837 : poHeader->nPayloadFileSize = nCurPos;
1236 : }
1237 : }
1238 : else
1239 : {
1240 34509 : if (!FlushDirty())
1241 0 : break;
1242 :
1243 34509 : const vsi_l_offset nSectorOffset =
1244 34509 : (nCurPos / poHeader->nSectorSize) * poHeader->nSectorSize;
1245 34509 : const vsi_l_offset nLastSectorOffset =
1246 34509 : (poHeader->nPayloadFileSize / poHeader->nSectorSize) *
1247 34509 : poHeader->nSectorSize;
1248 34509 : if (nSectorOffset > nLastSectorOffset &&
1249 286 : (poHeader->nPayloadFileSize % poHeader->nSectorSize) != 0)
1250 : {
1251 858 : if (poBaseHandle->Seek(
1252 286 : poHeader->nHeaderSize + nLastSectorOffset, 0) == 0 &&
1253 572 : poBaseHandle->Read(pabyWB, poHeader->nSectorSize, 1) == 1 &&
1254 286 : DecryptBlock(pabyWB, nLastSectorOffset))
1255 : {
1256 : #ifdef VERBOSE_VSICRYPT
1257 : CPLDebug("VSICRYPT", "Filling %d trailing bytes with 0",
1258 : static_cast<int>(poHeader->nSectorSize -
1259 : (poHeader->nPayloadFileSize -
1260 : nLastSectorOffset)));
1261 : #endif
1262 : // Fill with 0.
1263 286 : memset(
1264 286 : pabyWB + poHeader->nPayloadFileSize - nLastSectorOffset,
1265 : 0,
1266 : static_cast<int>(
1267 286 : poHeader->nSectorSize -
1268 286 : (poHeader->nPayloadFileSize - nLastSectorOffset)));
1269 :
1270 572 : if (poBaseHandle->Seek(
1271 286 : poHeader->nHeaderSize + nLastSectorOffset, 0) == 0)
1272 : {
1273 286 : EncryptBlock(pabyWB, nLastSectorOffset);
1274 286 : poBaseHandle->Write(pabyWB, poHeader->nSectorSize, 1);
1275 : }
1276 : }
1277 : }
1278 34509 : poBaseHandle->Seek(poHeader->nHeaderSize + nSectorOffset, SEEK_SET);
1279 63497 : if (poBaseHandle->Read(pabyWB, poHeader->nSectorSize, 1) == 0 ||
1280 28988 : !DecryptBlock(pabyWB, nSectorOffset))
1281 : {
1282 5521 : memset(pabyWB, 0, poHeader->nSectorSize);
1283 : }
1284 28988 : else if (nSectorOffset + poHeader->nSectorSize >
1285 28988 : poHeader->nPayloadFileSize)
1286 : {
1287 : // If the last sector was padded with random values,
1288 : // decrypt it to 0 in case of update scenarios.
1289 735 : CPLAssert(nSectorOffset < poHeader->nPayloadFileSize);
1290 735 : memset(pabyWB + poHeader->nPayloadFileSize - nSectorOffset, 0,
1291 735 : nSectorOffset + poHeader->nSectorSize -
1292 735 : poHeader->nPayloadFileSize);
1293 : }
1294 34509 : nWBOffset = nSectorOffset;
1295 34509 : nWBSize = poHeader->nSectorSize;
1296 : }
1297 : }
1298 :
1299 20039 : int nRet = static_cast<int>((nSize * nMemb - nToWrite) / nSize);
1300 : #ifdef VERBOSE_VSICRYPT
1301 : CPLDebug("VSICRYPT", "Write ret = %d (nMemb = %d)", nRet,
1302 : static_cast<int>(nMemb));
1303 : #endif
1304 20039 : return nRet;
1305 : }
1306 :
1307 : /************************************************************************/
1308 : /* Truncate() */
1309 : /************************************************************************/
1310 :
1311 : // Returns 0 on success. Returns -1 on error.
1312 3 : int VSICryptFileHandle::Truncate(vsi_l_offset nNewSize)
1313 : {
1314 : #ifdef VERBOSE_VSICRYPT
1315 : CPLDebug("VSICRYPT", "Truncate(" CPL_FRMT_GUIB ")", nNewSize);
1316 : #endif
1317 3 : if ((nPerms & VSICRYPT_WRITE) == 0)
1318 1 : return -1;
1319 :
1320 2 : if (!FlushDirty())
1321 0 : return -1;
1322 4 : if (poBaseHandle->Truncate(
1323 2 : poHeader->nHeaderSize +
1324 2 : ((nNewSize + poHeader->nSectorSize - 1) / poHeader->nSectorSize) *
1325 2 : poHeader->nSectorSize) != 0)
1326 0 : return -1;
1327 2 : bUpdateHeader = true;
1328 2 : poHeader->nPayloadFileSize = nNewSize;
1329 2 : return 0;
1330 : }
1331 :
1332 : /************************************************************************/
1333 : /* Eof() */
1334 : /************************************************************************/
1335 :
1336 4 : int VSICryptFileHandle::Eof()
1337 : {
1338 : #ifdef VERBOSE_VSICRYPT
1339 : CPLDebug("VSICRYPT", "Eof() = %d", static_cast<int>(bEOF));
1340 : #endif
1341 4 : return bEOF;
1342 : }
1343 :
1344 : /************************************************************************/
1345 : /* Error() */
1346 : /************************************************************************/
1347 :
1348 4 : int VSICryptFileHandle::Error()
1349 : {
1350 : #ifdef VERBOSE_VSICRYPT
1351 : CPLDebug("VSICRYPT", "Error() = %d", static_cast<int>(bError));
1352 : #endif
1353 4 : return bError;
1354 : }
1355 :
1356 : /************************************************************************/
1357 : /* ClearErr() */
1358 : /************************************************************************/
1359 :
1360 1 : void VSICryptFileHandle::ClearErr()
1361 : {
1362 : #ifdef VERBOSE_VSICRYPT
1363 : CPLDebug("VSICRYPT", "ClearErr()");
1364 : #endif
1365 1 : bEOF = false;
1366 1 : bError = false;
1367 1 : poBaseHandle->ClearErr();
1368 1 : }
1369 :
1370 : /************************************************************************/
1371 : /* Flush() */
1372 : /************************************************************************/
1373 :
1374 1175 : int VSICryptFileHandle::Flush()
1375 : {
1376 : #ifdef VERBOSE_VSICRYPT
1377 : CPLDebug("VSICRYPT", "Flush()");
1378 : #endif
1379 1175 : if (!FlushDirty())
1380 : {
1381 0 : return -1;
1382 : }
1383 1175 : if ((nPerms & VSICRYPT_WRITE))
1384 : {
1385 1045 : if (bLastSectorWasModified &&
1386 1038 : (poHeader->nPayloadFileSize % poHeader->nSectorSize) != 0)
1387 : {
1388 1006 : const vsi_l_offset nLastSectorOffset =
1389 1006 : (poHeader->nPayloadFileSize / poHeader->nSectorSize) *
1390 1006 : poHeader->nSectorSize;
1391 3018 : if (poBaseHandle->Seek(poHeader->nHeaderSize + nLastSectorOffset,
1392 1006 : 0) == 0 &&
1393 1984 : poBaseHandle->Read(pabyWB, poHeader->nSectorSize, 1) == 1 &&
1394 978 : DecryptBlock(pabyWB, nLastSectorOffset))
1395 : {
1396 : // Fill with random
1397 : #ifdef VERBOSE_VSICRYPT
1398 : CPLDebug("VSICRYPT", "Filling %d trailing bytes with random",
1399 : static_cast<int>(
1400 : poHeader->nSectorSize -
1401 : (poHeader->nPayloadFileSize - nLastSectorOffset)));
1402 : #endif
1403 978 : CryptoPP::OS_GenerateRandomBlock(
1404 : false, // Do not need cryptographic randomness.
1405 : reinterpret_cast<cryptopp_byte *>(
1406 978 : pabyWB + poHeader->nPayloadFileSize -
1407 : nLastSectorOffset),
1408 : static_cast<int>(
1409 978 : poHeader->nSectorSize -
1410 978 : (poHeader->nPayloadFileSize - nLastSectorOffset)));
1411 :
1412 1956 : if (poBaseHandle->Seek(
1413 978 : poHeader->nHeaderSize + nLastSectorOffset, 0) == 0)
1414 : {
1415 978 : EncryptBlock(pabyWB, nLastSectorOffset);
1416 978 : poBaseHandle->Write(pabyWB, poHeader->nSectorSize, 1);
1417 : }
1418 : }
1419 : }
1420 1045 : bLastSectorWasModified = false;
1421 1045 : if (poBaseHandle->Flush() != 0)
1422 0 : return -1;
1423 : }
1424 1175 : if (bUpdateHeader)
1425 : {
1426 : #ifdef VERBOSE_VSICRYPT
1427 : CPLDebug("VSICRYPT", "nPayloadFileSize = " CPL_FRMT_GUIB,
1428 : poHeader->nPayloadFileSize);
1429 : #endif
1430 1038 : if (!poHeader->WriteToFile(poBaseHandle, poEncCipher))
1431 0 : return -1;
1432 : }
1433 :
1434 1175 : return 0;
1435 : }
1436 :
1437 : /************************************************************************/
1438 : /* Close() */
1439 : /************************************************************************/
1440 :
1441 2341 : int VSICryptFileHandle::Close()
1442 : {
1443 2341 : int nRet = 0;
1444 2341 : if (poBaseHandle != nullptr && poHeader != nullptr)
1445 : {
1446 1174 : if (Flush() != 0)
1447 0 : return -1;
1448 1174 : nRet = poBaseHandle->Close();
1449 1174 : delete poBaseHandle;
1450 1174 : poBaseHandle = nullptr;
1451 : }
1452 : #ifdef VERBOSE_VSICRYPT
1453 : CPLDebug("VSICRYPT", "Close(%s)", osBaseFilename.c_str());
1454 : #endif
1455 2341 : return nRet;
1456 : }
1457 :
1458 : /************************************************************************/
1459 : /* VSICryptFilesystemHandler */
1460 : /************************************************************************/
1461 :
1462 : class VSICryptFilesystemHandler final : public VSIFilesystemHandler
1463 : {
1464 : public:
1465 : VSICryptFilesystemHandler();
1466 : ~VSICryptFilesystemHandler() override;
1467 :
1468 : VSIVirtualHandle *Open(const char *pszFilename, const char *pszAccess,
1469 : bool bSetError,
1470 : CSLConstList /* papszOptions */) override;
1471 : int Stat(const char *pszFilename, VSIStatBufL *pStatBuf,
1472 : int nFlags) override;
1473 : int Unlink(const char *pszFilename) override;
1474 : int Rename(const char *oldpath, const char *newpath) override;
1475 : char **ReadDirEx(const char *pszDirname, int nMaxFiles) override;
1476 : };
1477 :
1478 : /************************************************************************/
1479 : /* VSICryptFilesystemHandler() */
1480 : /************************************************************************/
1481 :
1482 1392 : VSICryptFilesystemHandler::VSICryptFilesystemHandler()
1483 : {
1484 1392 : }
1485 :
1486 : /************************************************************************/
1487 : /* ~VSICryptFilesystemHandler() */
1488 : /************************************************************************/
1489 :
1490 1882 : VSICryptFilesystemHandler::~VSICryptFilesystemHandler()
1491 : {
1492 1882 : }
1493 :
1494 : /************************************************************************/
1495 : /* GetFilename() */
1496 : /************************************************************************/
1497 :
1498 2296 : static CPLString GetFilename(const char *pszFilename)
1499 : {
1500 2296 : if (strcmp(pszFilename, VSICRYPT_PREFIX_WITHOUT_SLASH) == 0)
1501 0 : pszFilename = VSICRYPT_PREFIX;
1502 :
1503 2296 : CPLAssert(strncmp(pszFilename, VSICRYPT_PREFIX, strlen(VSICRYPT_PREFIX)) ==
1504 : 0);
1505 2296 : pszFilename += strlen(VSICRYPT_PREFIX);
1506 2296 : const char *pszFileArg = strstr(pszFilename, "file=");
1507 2296 : if (pszFileArg == nullptr)
1508 13 : return pszFilename;
1509 4566 : CPLString osRet(pszFileArg + strlen("file="));
1510 2283 : return osRet;
1511 : }
1512 :
1513 : /************************************************************************/
1514 : /* GetArgument() */
1515 : /************************************************************************/
1516 :
1517 7561 : static CPLString GetArgument(const char *pszFilename, const char *pszParamName,
1518 : const char *pszDefault = "")
1519 : {
1520 15122 : CPLString osParamName(pszParamName);
1521 7561 : osParamName += "=";
1522 :
1523 7561 : const char *pszNeedle = strstr(pszFilename, osParamName);
1524 7561 : if (pszNeedle == nullptr)
1525 5254 : return pszDefault;
1526 :
1527 4614 : CPLString osRet(pszNeedle + osParamName.size());
1528 2307 : size_t nCommaPos = osRet.find(",");
1529 2307 : if (nCommaPos != std::string::npos)
1530 2307 : osRet.resize(nCommaPos);
1531 2307 : return osRet;
1532 : }
1533 :
1534 : /************************************************************************/
1535 : /* GetKey() */
1536 : /************************************************************************/
1537 :
1538 1285 : static CPLString GetKey(const char *pszFilename)
1539 : {
1540 1285 : CPLString osKey = GetArgument(pszFilename, "key");
1541 : // TODO(schwehr): Make 10U and 1024U into symbolic constants.
1542 1285 : if (osKey.empty())
1543 : {
1544 11 : const char *pszKey = CPLGetConfigOption("VSICRYPT_KEY", "");
1545 : // Do some form of validation to please Coverity
1546 11 : CPLAssert(strlen(pszKey) < 10U * 1024U);
1547 : // coverity [tainted_data_transitive]
1548 11 : osKey = pszKey;
1549 : }
1550 1285 : if (osKey.empty() || EQUAL(osKey, "GENERATE_IT"))
1551 : {
1552 12 : CPLString osKeyB64(GetArgument(pszFilename, "key_b64"));
1553 12 : if (osKeyB64.empty())
1554 : {
1555 11 : const char *pszKey = CPLGetConfigOption("VSICRYPT_KEY_B64", "");
1556 : // Do some form of validation to please Coverity
1557 11 : CPLAssert(strlen(pszKey) < 10U * 1024U);
1558 : // coverity [tainted_data_transitive]
1559 11 : osKeyB64 = pszKey;
1560 : }
1561 12 : if (!osKeyB64.empty())
1562 : {
1563 2 : GByte *key = reinterpret_cast<GByte *>(CPLStrdup(osKeyB64));
1564 2 : int nLength = CPLBase64DecodeInPlace(key);
1565 2 : osKey.assign(reinterpret_cast<const char *>(key), nLength);
1566 2 : memset(key, 0, osKeyB64.size());
1567 2 : CPLFree(key);
1568 : }
1569 : // coverity[tainted_data]
1570 12 : memset(osKeyB64.data(), 0, osKeyB64.size());
1571 : }
1572 1285 : return osKey;
1573 : }
1574 :
1575 : /************************************************************************/
1576 : /* Open() */
1577 : /************************************************************************/
1578 :
1579 : VSIVirtualHandle *
1580 1281 : VSICryptFilesystemHandler::Open(const char *pszFilename, const char *pszAccess,
1581 : bool /* bSetError */,
1582 : CSLConstList /* papszOptions */)
1583 : {
1584 : #ifdef VERBOSE_VSICRYPT
1585 : CPLDebug("VSICRYPT", "Open(%s, %s)", pszFilename, pszAccess);
1586 : #endif
1587 2562 : CPLString osFilename(GetFilename(pszFilename));
1588 :
1589 2562 : CPLString osKey(GetKey(pszFilename));
1590 1281 : if (osKey.empty() && pabyGlobalKey == nullptr)
1591 : {
1592 2 : CPLError(CE_Failure, CPLE_AppDefined,
1593 : "Encryption key not defined as key/key_b64 parameter, "
1594 : "VSICRYPT_KEY/VSICRYPT_KEY_B64 configuration option or "
1595 : "VSISetCryptKey() API");
1596 2 : return nullptr;
1597 : }
1598 :
1599 2558 : CPLString osAccess(pszAccess);
1600 1279 : if (strchr(pszAccess, 'b') == nullptr)
1601 15 : osAccess += "b";
1602 1279 : if (strchr(pszAccess, 'r'))
1603 : {
1604 228 : VSIVirtualHandle *fpBase = reinterpret_cast<VSIVirtualHandle *>(
1605 : VSIFOpenL(osFilename, osAccess));
1606 228 : if (fpBase == nullptr)
1607 1 : return nullptr;
1608 227 : VSICryptFileHeader *poHeader = new VSICryptFileHeader();
1609 227 : if (!poHeader->ReadFromFile(fpBase, osKey))
1610 : {
1611 94 : memset(osKey.data(), 0, osKey.size());
1612 94 : fpBase->Close();
1613 94 : delete fpBase;
1614 94 : delete poHeader;
1615 94 : return nullptr;
1616 : }
1617 :
1618 : VSICryptFileHandle *poHandle = new VSICryptFileHandle(
1619 : osFilename, fpBase, poHeader,
1620 133 : strchr(pszAccess, '+') ? VSICRYPT_READ | VSICRYPT_WRITE
1621 133 : : VSICRYPT_READ);
1622 133 : if (!poHandle->Init(osKey, false))
1623 : {
1624 4 : memset(osKey.data(), 0, osKey.size());
1625 4 : delete poHandle;
1626 4 : poHandle = nullptr;
1627 : }
1628 133 : memset(osKey.data(), 0, osKey.size());
1629 133 : return poHandle;
1630 : }
1631 1051 : else if (strchr(pszAccess, 'w'))
1632 : {
1633 : CPLString osAlg(GetArgument(pszFilename, "alg",
1634 2088 : CPLGetConfigOption("VSICRYPT_ALG", "AES")));
1635 1044 : VSICryptAlg eAlg = GetAlg(osAlg);
1636 :
1637 1044 : VSICryptMode eMode = GetMode(GetArgument(
1638 : pszFilename, "mode", CPLGetConfigOption("VSICRYPT_MODE", "CBC")));
1639 :
1640 : CPLString osFreeText =
1641 : GetArgument(pszFilename, "freetext",
1642 2088 : CPLGetConfigOption("VSICRYPT_FREETEXT", ""));
1643 :
1644 : CPLString osIV = GetArgument(pszFilename, "iv",
1645 2088 : CPLGetConfigOption("VSICRYPT_IV", ""));
1646 :
1647 1044 : int nSectorSize = atoi(
1648 2088 : GetArgument(pszFilename, "sector_size",
1649 : CPLGetConfigOption("VSICRYPT_SECTOR_SIZE", "512")));
1650 1044 : if (nSectorSize <= 0 || nSectorSize >= 65535)
1651 : {
1652 0 : CPLError(CE_Warning, CPLE_NotSupported,
1653 : "Invalid value for sector_size. Defaulting to 512.");
1654 0 : nSectorSize = 512;
1655 : }
1656 :
1657 1044 : const bool bAddKeyCheck = CPLTestBool(
1658 2088 : GetArgument(pszFilename, "add_key_check",
1659 : CPLGetConfigOption("VSICRYPT_ADD_KEY_CHECK", "NO")));
1660 :
1661 : /* Generate random initial vector */
1662 1044 : CryptoPP::BlockCipher *poBlock = GetEncBlockCipher(eAlg);
1663 1044 : if (poBlock == nullptr)
1664 : {
1665 0 : CPLError(CE_Failure, CPLE_AppDefined,
1666 : "Cipher algorithm not supported in this build: %s",
1667 : osAlg.c_str());
1668 0 : memset(osKey.data(), 0, osKey.size());
1669 0 : return nullptr;
1670 : }
1671 1044 : int nMinKeySize = static_cast<int>(poBlock->MinKeyLength());
1672 1044 : int nMaxKeySize = static_cast<int>(poBlock->MaxKeyLength());
1673 1044 : int nBlockSize = static_cast<int>(poBlock->BlockSize());
1674 1044 : delete poBlock;
1675 :
1676 1044 : if (!osIV.empty())
1677 : {
1678 1 : if (static_cast<int>(osIV.size()) != nBlockSize)
1679 : {
1680 1 : CPLError(CE_Failure, CPLE_AppDefined,
1681 : "IV should be %d byte large", nBlockSize);
1682 1 : memset(osKey.data(), 0, osKey.size());
1683 1 : return nullptr;
1684 : }
1685 : }
1686 : else
1687 : {
1688 1043 : osIV.resize(nBlockSize);
1689 2086 : CryptoPP::OS_GenerateRandomBlock(
1690 : false, // Do not need cryptographic randomness.
1691 1043 : reinterpret_cast<cryptopp_byte *>(osIV.data()), osIV.size());
1692 : }
1693 :
1694 1043 : if (EQUAL(osKey, "GENERATE_IT"))
1695 : {
1696 1 : osKey.resize(nMaxKeySize);
1697 1 : CPLDebug("VSICRYPT",
1698 : "Generating key. This might take some time...");
1699 1 : CryptoPP::OS_GenerateRandomBlock(
1700 : // Need cryptographic randomness.
1701 : // Config option for speeding tests.
1702 1 : CPLTestBool(
1703 : CPLGetConfigOption("VSICRYPT_CRYPTO_RANDOM", "TRUE")),
1704 1 : reinterpret_cast<cryptopp_byte *>(osKey.data()), osKey.size());
1705 :
1706 : char *pszB64 =
1707 1 : CPLBase64Encode(static_cast<int>(osKey.size()),
1708 1 : reinterpret_cast<const GByte *>(osKey.c_str()));
1709 1 : if (CPLTestBool(CPLGetConfigOption("VSICRYPT_DISPLAY_GENERATED_KEY",
1710 : "TRUE")))
1711 : {
1712 1 : CPLError(CE_Failure, CPLE_AppDefined,
1713 : "BASE64 key '%s' has been generated, and installed in "
1714 : "the VSICRYPT_KEY_B64 configuration option.",
1715 : pszB64);
1716 : }
1717 1 : CPLSetConfigOption("VSICRYPT_KEY_B64", pszB64);
1718 1 : CPLFree(pszB64);
1719 : }
1720 :
1721 : const int nKeyLength =
1722 1043 : !osKey.empty() ? static_cast<int>(osKey.size()) : nGlobalKeySize;
1723 1043 : if (nKeyLength < nMinKeySize)
1724 : {
1725 2 : CPLError(CE_Failure, CPLE_AppDefined,
1726 : "Key is too short: %d bytes. Should be at least %d bytes",
1727 : nKeyLength, nMinKeySize);
1728 2 : memset(osKey.data(), 0, osKey.size());
1729 2 : return nullptr;
1730 : }
1731 :
1732 1041 : VSIVirtualHandle *fpBase = reinterpret_cast<VSIVirtualHandle *>(
1733 : VSIFOpenL(osFilename, osAccess.c_str()));
1734 1041 : if (fpBase == nullptr)
1735 : {
1736 2 : memset(osKey.data(), 0, osKey.size());
1737 2 : return nullptr;
1738 : }
1739 :
1740 1039 : VSICryptFileHeader *poHeader = new VSICryptFileHeader();
1741 1039 : poHeader->osIV = osIV;
1742 1039 : CPL_IGNORE_RET_VAL(osIV);
1743 1039 : poHeader->eAlg = eAlg;
1744 1039 : poHeader->eMode = eMode;
1745 1039 : poHeader->nSectorSize = static_cast<GUInt16>(nSectorSize);
1746 1039 : poHeader->osFreeText = std::move(osFreeText);
1747 1039 : poHeader->bAddKeyCheck = bAddKeyCheck;
1748 :
1749 : VSICryptFileHandle *poHandle = new VSICryptFileHandle(
1750 : osFilename, fpBase, poHeader,
1751 1039 : strchr(pszAccess, '+') ? VSICRYPT_READ | VSICRYPT_WRITE
1752 1039 : : VSICRYPT_WRITE);
1753 1039 : if (!poHandle->Init(osKey, true))
1754 : {
1755 2 : memset(osKey.data(), 0, osKey.size());
1756 2 : delete poHandle;
1757 2 : poHandle = nullptr;
1758 : }
1759 1039 : memset(osKey.data(), 0, osKey.size());
1760 1039 : return poHandle;
1761 : }
1762 7 : else if (strchr(pszAccess, 'a'))
1763 : {
1764 : VSIVirtualHandle *fpBase =
1765 6 : reinterpret_cast<VSIVirtualHandle *>(VSIFOpenL(osFilename, "rb+"));
1766 6 : if (fpBase == nullptr)
1767 : {
1768 2 : memset(osKey.data(), 0, osKey.size());
1769 2 : return VSIFilesystemHandler::Open(pszFilename, "wb+");
1770 : }
1771 4 : VSICryptFileHeader *poHeader = new VSICryptFileHeader();
1772 4 : if (!poHeader->ReadFromFile(fpBase, osKey))
1773 : {
1774 2 : memset(osKey.data(), 0, osKey.size());
1775 2 : fpBase->Close();
1776 2 : delete fpBase;
1777 2 : delete poHeader;
1778 2 : return nullptr;
1779 : }
1780 :
1781 : VSICryptFileHandle *poHandle = new VSICryptFileHandle(
1782 2 : osFilename, fpBase, poHeader, VSICRYPT_READ | VSICRYPT_WRITE);
1783 2 : if (!poHandle->Init(osKey))
1784 : {
1785 1 : delete poHandle;
1786 1 : poHandle = nullptr;
1787 : }
1788 2 : memset(osKey.data(), 0, osKey.size());
1789 2 : if (poHandle != nullptr)
1790 1 : poHandle->Seek(0, SEEK_END);
1791 2 : return poHandle;
1792 : }
1793 :
1794 1 : return nullptr;
1795 : }
1796 :
1797 : /************************************************************************/
1798 : /* Stat() */
1799 : /************************************************************************/
1800 :
1801 5 : int VSICryptFilesystemHandler::Stat(const char *pszFilename,
1802 : VSIStatBufL *pStatBuf, int nFlags)
1803 : {
1804 : #ifdef VERBOSE_VSICRYPT
1805 : CPLDebug("VSICRYPT", "Stat(%s)", pszFilename);
1806 : #endif
1807 10 : CPLString osFilename(GetFilename(pszFilename));
1808 5 : if (VSIStatExL(osFilename, pStatBuf, nFlags) != 0)
1809 1 : return -1;
1810 : VSIVirtualHandle *fp =
1811 4 : reinterpret_cast<VSIVirtualHandle *>(VSIFOpenL(osFilename, "rb"));
1812 4 : if (fp == nullptr)
1813 0 : return -1;
1814 4 : VSICryptFileHeader *poHeader = new VSICryptFileHeader();
1815 8 : CPLString osKey(GetKey(pszFilename));
1816 4 : if (!poHeader->ReadFromFile(fp, osKey))
1817 : {
1818 1 : memset(osKey.data(), 0, osKey.size());
1819 1 : fp->Close();
1820 1 : delete fp;
1821 1 : delete poHeader;
1822 1 : return -1;
1823 : }
1824 3 : memset(osKey.data(), 0, osKey.size());
1825 3 : fp->Close();
1826 3 : delete fp;
1827 3 : if (poHeader)
1828 : {
1829 3 : pStatBuf->st_size = poHeader->nPayloadFileSize;
1830 3 : delete poHeader;
1831 3 : return 0;
1832 : }
1833 : else
1834 0 : return -1;
1835 : }
1836 :
1837 : /************************************************************************/
1838 : /* Unlink() */
1839 : /************************************************************************/
1840 :
1841 1006 : int VSICryptFilesystemHandler::Unlink(const char *pszFilename)
1842 : {
1843 1006 : return VSIUnlink(GetFilename(pszFilename));
1844 : }
1845 :
1846 : /************************************************************************/
1847 : /* Rename() */
1848 : /************************************************************************/
1849 :
1850 2 : int VSICryptFilesystemHandler::Rename(const char *oldpath, const char *newpath)
1851 : {
1852 2 : CPLString osNewPath;
1853 2 : if (strncmp(newpath, VSICRYPT_PREFIX, strlen(VSICRYPT_PREFIX)) == 0)
1854 1 : osNewPath = GetFilename(newpath);
1855 : else
1856 1 : osNewPath = newpath;
1857 :
1858 4 : return VSIRename(GetFilename(oldpath), osNewPath);
1859 : }
1860 :
1861 : /************************************************************************/
1862 : /* ReadDirEx() */
1863 : /************************************************************************/
1864 :
1865 1 : char **VSICryptFilesystemHandler::ReadDirEx(const char *pszDirname,
1866 : int nMaxFiles)
1867 : {
1868 : #ifdef VERBOSE_VSICRYPT
1869 : CPLDebug("VSICRYPT", "ReadDir(%s)", pszDirname);
1870 : #endif
1871 1 : return VSIReadDirEx(GetFilename(pszDirname), nMaxFiles);
1872 : }
1873 :
1874 : #ifdef VSICRYPT_DRIVER
1875 :
1876 : #include "gdal_priv.h"
1877 :
1878 : /**
1879 : * \brief Evaluate if this is a crypt file.
1880 : *
1881 : * The function signature must match GDALDataset::Identify.
1882 : *
1883 : * @param poOpenInfo The header bytes used for file identification.
1884 : *
1885 : * @return 1 if this is a crypt file or 0 otherwise.
1886 : */
1887 :
1888 : static int VSICryptIdentify(GDALOpenInfo *poOpenInfo)
1889 : {
1890 : return poOpenInfo->nHeaderBytes > 8 &&
1891 : memcmp(poOpenInfo->pabyHeader, VSICRYPT_SIGNATURE, 8) == 0;
1892 : }
1893 :
1894 : static GDALDataset *VSICryptOpen(GDALOpenInfo *poOpenInfo)
1895 : {
1896 : if (!VSICryptIdentify(poOpenInfo))
1897 : return nullptr;
1898 : return GDALOpen(
1899 : (CPLString(VSICRYPT_PREFIX) + poOpenInfo->pszFilename).c_str(),
1900 : poOpenInfo->eAccess);
1901 : }
1902 :
1903 : #endif
1904 :
1905 : //! @endcond
1906 :
1907 : /************************************************************************/
1908 : /* VSIInstallCryptFileHandler() */
1909 : /************************************************************************/
1910 :
1911 : /**
1912 : * \brief Install /vsicrypt/ encrypted file system handler
1913 : * (requires <a href="http://www.cryptopp.com/">libcrypto++</a>)
1914 : *
1915 : * A special file handler is installed that allows reading/creating/update
1916 : * encrypted files on the fly, with random access capabilities.
1917 : *
1918 : * The cryptographic algorithms used are
1919 : * <a href="https://en.wikipedia.org/wiki/Block_cipher">block ciphers</a>,
1920 : * with symmetric key.
1921 : *
1922 : * In their simplest form, recognized filenames are of the form
1923 : * /vsicrypt//absolute_path/to/file, /vsicrypt/c:/absolute_path/to/file or
1924 : * /vsicrypt/relative/path/to/file.
1925 : *
1926 : * Options can also be used with the following format :
1927 : * /vsicrypt/option1=val1,option2=val2,...,file=/path/to/file
1928 : *
1929 : * They can also be passed as configuration option/environment variable, because
1930 : * in some use cases, the syntax with option in the filename might not properly
1931 : * work with some drivers.
1932 : *
1933 : * In all modes, the encryption key must be provided. There are several ways
1934 : * of doing so :
1935 : * <ul>
1936 : * <li>By adding a key= parameter to the filename, like
1937 : * /vsicrypt/key=my_secret_key,file=/path/to/file. Note that this restricts
1938 : * the key to be in text format, whereas at its full power, it can be binary
1939 : * content.</li>
1940 : * <li>By adding a key_b64= parameter to the filename, to specify a binary key
1941 : * expressed in Base64 encoding, like
1942 : * /vsicrypt/key_b64=th1sl00kslikebase64=,file=/path/to/file.</li>
1943 : * <li>By setting the VSICRYPT_KEY configuration option. The key should be in
1944 : * text format.</li>
1945 : * <li>By setting the VSICRYPT_KEY_B64 configuration option. The key should be
1946 : * encoded in Base64.</li>
1947 : * <li>By using the VSISetCryptKey() C function.</li>
1948 : * </ul>
1949 : *
1950 : * When creating a file, if key=GENERATE_IT or VSICRYPT_KEY=GENERATE_IT is
1951 : * passed, the encryption key will be generated from the pseudo-random number
1952 : * generator of the operating system. The key will be displayed on the standard
1953 : * error stream in a Base64 form (unless the VSICRYPT_DISPLAY_GENERATED_KEY
1954 : * configuration option is set to OFF), and the VSICRYPT_KEY_B64 configuration
1955 : * option will also be set with the Base64 form of the key (so that
1956 : * CPLGetConfigOption("VSICRYPT_KEY_B64", NULL) can be used to get it back).
1957 : *
1958 : * The available options are :
1959 : * <ul>
1960 :
1961 : * <li>alg=AES/Blowfish/Camellia/CAST256/DES_EDE2/DES_EDE3/MARS/IDEA/RC5/RC6/Serpent/SHACAL2/SKIPJACK/Twofish/XTEA:
1962 : * to specify the <a href="https://en.wikipedia.org/wiki/Block_cipher">block
1963 : * cipher</a> algorithm. The default is AES. Only used on
1964 : * creation. Ignored otherwise. Note: depending on how GDAL is build, if
1965 : * linked against the DLL version of libcrypto++, only a subset of those
1966 : * algorithms will be available, namely AES, DES_EDE2, DES_EDE3 and
1967 : * SKIPJACK. Also available as VSICRYPT_ALG configuration option.</li>
1968 : * <li>mode=CBC/CFB/OFB/CTR/CBC_CTS: to specify the
1969 : * <a href="https://en.wikipedia.org/wiki/Block_cipher_mode_of_operation">
1970 : * block cipher mode of operation</a>.
1971 : * The default is CBC.
1972 : * Only used on creation. Ignored otherwise.
1973 : * Also available as VSICRYPT_MODE configuration option.</li>
1974 : * <li>key=text_key: see above.</li>
1975 : * <li>key_b64=base64_encoded_key: see above.</li>
1976 : * <li>freetext=some_text: to specify a text content that will be written
1977 : * *unencrypted* in the file header, for informational purposes. Default to
1978 : * empty. Only used on creation. Ignored otherwise.
1979 : * Also available as VSICRYPT_FREETEXT configuration option.</li>
1980 : * <li>sector_size=int_value: to specify the size of the "sector", which is the
1981 : * unit chunk of information that is encrypted/decrypted. Default to 512
1982 : * bytes. The valid values depend on the algorithm and block cipher mode of
1983 : * operation. Only used on creation. Ignored otherwise. Also available as
1984 : * VSICRYPT_SECTOR_SIZE configuration option.</li>
1985 : * <li>iv=initial_vector_as_text: to specify the Initial Vector. This is an
1986 : * advanced option that should generally *NOT* be used. It is only useful to
1987 : * get completely deterministic output given the plaintext, key and other
1988 : * parameters, which in general *NOT* what you want to do. By default, a
1989 : * random initial vector of the appropriate size will be generated for each
1990 : * new file created. Only used on creation. Ignored otherwise. Also
1991 : * available as VSICRYPT_IV configuration option.</li>
1992 :
1993 : * <li>add_key_check=YES/NO: whether a special value should be encrypted in the
1994 : * header, so as to be quickly able to determine if the decryption key is
1995 : * correct. Defaults to NO. Only used on creation. Ignored otherwise.
1996 : * Also available as VSICRYPT_ADD_KEY_CHECK configuration option.</li>
1997 : * <li>file=filename. To specify the filename. This must be the last option put
1998 : * in the option list (so as to make it possible to use filenames with comma
1999 : * in them. )
2000 : * </ul>
2001 : *
2002 : * This special file handler can be combined with other virtual filesystems
2003 : * handlers, such as /vsizip. For example,
2004 : * /vsicrypt//vsicurl/path/to/remote/encrypted/file.tif
2005 : *
2006 : * Implementation details:
2007 : *
2008 : * The structure of encrypted files is the following: a header, immediately
2009 : * followed by the encrypted payload (by sectors, i.e. chunks of sector_size
2010 : * bytes).
2011 : *
2012 : * The header structure is the following :
2013 : * <ol>
2014 : * <li>8 bytes. Signature. Fixed value: VSICRYPT.</li>
2015 : * <li>UINT16_LE. Header size (including previous signature bytes).</li>
2016 : * <li>UINT8. Format major version. Current value: 1.</li>
2017 : * <li>UINT8. Format minor version. Current value: 0.</li>
2018 : * <li>UINT16. Sector size.</li>
2019 : * <li>UINT8. Cipher algorithm. Valid values are: 0 = AES (Rijndael), 1 =
2020 : * Blowfish, 2 = Camellia, 3 = CAST256, 4 = DES_EDE2, 5 = DES_EDE3, 6 =
2021 : * MARS, 7 = IDEA, 8 = RC5, 9 = RC6, 10 = Serpent, 11 = SHACAL2, 12 =
2022 : * SKIPJACK, 13 = Twofish, 14 = XTEA.</li>
2023 : * <li>UINT8. Block cipher mode of operation. Valid values are: 0 = CBC, 1 =
2024 : * CFB, 2 = OFB, 3 = CTR, 4 = CBC_CTS.</li>
2025 : * <li>UINT8. Size in bytes of the Initial Vector.</li>
2026 : * <li>N bytes with the content of the Initial Vector, where N is the value of
2027 : * the previous field.</li>
2028 : * <li>UINT16_LE. Size in bytes of the free text.</li>
2029 : * <li>N bytes with the content of the free text, where N is the value of the
2030 : * previous field.</li>
2031 : * <li>UINT8. Size in bytes of encrypted content (key check), or 0 if key check
2032 : * is absent.</li>
2033 : * <li>N bytes with encrypted content (key check), where N is the value of the
2034 : * previous field.</li>
2035 : * <li>UINT64_LE. Size of the unencrypted file, in bytes.</li>
2036 : * <li>UINT16_LE. Size in bytes of extra content (of unspecified semantics). For
2037 : * v1.0, fixed value of 0</li>
2038 : * <li>N bytes with extra content (of unspecified semantics), where N is the
2039 : * value of the previous field.</li>
2040 : * </ol>
2041 : *
2042 : * This design does not provide any means of authentication or integrity check.
2043 : *
2044 : * Each sector is encrypted/decrypted independently of other sectors. For that,
2045 : * the Initial Vector contained in the header is XOR'ed with the file offset
2046 : * (relative to plain text file) of the start of the sector being processed, as
2047 : * a 8-byte integer. More precisely, the first byte of the main IV is XOR'ed
2048 : * with the 8 least-significant bits of the sector offset, the second byte of
2049 : * the main IV is XOR'ed with the following 8 bits of the sector offset,
2050 : * etc... until the 8th byte.
2051 : *
2052 : * This design could potentially be prone to chosen-plaintext attack, for
2053 : * example if the attacker managed to get (part of) an existing encrypted file
2054 : * to be encrypted from plaintext he might have selected.
2055 : *
2056 : * Note: if "hostile" code can explore process content, or attach to it with a
2057 : * debugger, it might be relatively easy to retrieve the encryption key. A GDAL
2058 : * plugin could for example get the content of configuration options, or list
2059 : * opened datasets and see the key/key_b64 values, so disabling plugin loading
2060 : * might be a first step, as well as linking statically GDAL to application
2061 : * code. If plugin loading is enabled or GDAL dynamically linked, using
2062 : * VSISetCryptKey() to set the key might make it a bit more complicated to spy
2063 : * the key. But, as said initially, this is in no way a perfect protection.
2064 : *
2065 : * @since GDAL 2.1.0
2066 : */
2067 1392 : void VSIInstallCryptFileHandler(void)
2068 :
2069 : {
2070 1392 : VSIFileManager::InstallHandler(VSICRYPT_PREFIX,
2071 1392 : new VSICryptFilesystemHandler);
2072 :
2073 : #ifdef VSICRYPT_DRIVER
2074 : if (GDALGetDriverByName("VSICRYPT") != nullptr)
2075 : return;
2076 :
2077 : GDALDriver *poDriver = new GDALDriver();
2078 :
2079 : poDriver->SetDescription("VSICRYPT");
2080 : #ifdef GDAL_DCAP_RASTER
2081 : poDriver->SetMetadataItem(GDAL_DCAP_RASTER, "YES");
2082 : poDriver->SetMetadataItem(GDAL_DCAP_VECTOR, "YES");
2083 : #endif
2084 : poDriver->SetMetadataItem(
2085 : GDAL_DMD_LONGNAME, CPLSPrintf("Wrapper for %s files", VSICRYPT_PREFIX));
2086 :
2087 : poDriver->pfnOpen = VSICryptOpen;
2088 : poDriver->pfnIdentify = VSICryptIdentify;
2089 :
2090 : GetGDALDriverManager()->RegisterDriver(poDriver);
2091 : #endif
2092 1392 : }
2093 :
2094 : #else /* HAVE_CRYPTOPP */
2095 :
2096 : class VSIDummyCryptFilesystemHandler : public VSIFilesystemHandler
2097 : {
2098 : public:
2099 : VSIDummyCryptFilesystemHandler()
2100 : {
2101 : }
2102 :
2103 : VSIVirtualHandle *Open(const char * /* pszFilename */,
2104 : const char * /* pszAccess */, bool /* bSetError */,
2105 : CSLConstList /* papszOptions */) override
2106 : {
2107 : CPLError(CE_Failure, CPLE_NotSupported,
2108 : "%s support not available in this build", VSICRYPT_PREFIX);
2109 : return nullptr;
2110 : }
2111 :
2112 : int Stat(const char * /* pszFilename */, VSIStatBufL * /*pStatBuf */,
2113 : int /* nFlags */) override
2114 : {
2115 : CPLError(CE_Failure, CPLE_NotSupported,
2116 : "%s support not available in this build", VSICRYPT_PREFIX);
2117 : return -1;
2118 : }
2119 : };
2120 :
2121 : void VSIInstallCryptFileHandler(void)
2122 : {
2123 : VSIFileManager::InstallHandler(VSICRYPT_PREFIX,
2124 : new VSIDummyCryptFilesystemHandler);
2125 : }
2126 :
2127 : void VSISetCryptKey(const GByte * /* pabyKey */, int /* nKeySize */)
2128 : {
2129 : // Not supported.
2130 : }
2131 :
2132 : #endif // HAVE_CRYPTOPP
2133 :
2134 : // Below is only useful if using as a plugin over GDAL >= 2.0.
2135 : #ifdef VSICRYPT_AUTOLOAD
2136 :
2137 : CPL_C_START
2138 : void CPL_DLL GDALRegisterMe();
2139 : CPL_C_END
2140 :
2141 : void GDALRegisterMe()
2142 : {
2143 : VSIFilesystemHandler *poExistingHandler =
2144 : VSIFileManager::GetHandler(VSICRYPT_PREFIX);
2145 : if (poExistingHandler == VSIFileManager::GetHandler("."))
2146 : {
2147 : // In the case where VSICRYPT_PREFIX is just handled by the regular
2148 : // handler, install the vsicrypt handler (shouldn't happen)
2149 : VSIInstallCryptFileHandler();
2150 : }
2151 : else
2152 : {
2153 : // If there's already an installed handler, then check if it is a
2154 : // dummy one (should normally be the case) or a real one
2155 : CPLErrorReset();
2156 : CPLPushErrorHandler(CPLQuietErrorHandler);
2157 : VSIStatBufL sStat;
2158 : CPL_IGNORE_RET_VAL(VSIStatL(
2159 : (CPLString(VSICRYPT_PREFIX) + "i_do_not_exist").c_str(), &sStat));
2160 : CPLPopErrorHandler();
2161 : if (strstr(CPLGetLastErrorMsg(), "support not available in this build"))
2162 : {
2163 : // Dummy handler. Register the new one, and delete the old one
2164 : VSIInstallCryptFileHandler();
2165 : delete poExistingHandler;
2166 : }
2167 : else
2168 : {
2169 : CPLDebug("VSICRYPT", "GDAL has already a working %s implementation",
2170 : VSICRYPT_PREFIX);
2171 : }
2172 : CPLErrorReset();
2173 : }
2174 : }
2175 :
2176 : #endif /* VSICRYPT_AUTOLOAD */
|