VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/misc/inifile.cpp@ 96809

Last change on this file since 96809 was 96407, checked in by vboxsync, 2 years ago

scm copyright and license note update

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 27.2 KB
Line 
1/* $Id: inifile.cpp 96407 2022-08-22 17:43:14Z vboxsync $ */
2/** @file
3 * IPRT - INI-file parser.
4 */
5
6/*
7 * Copyright (C) 2017-2022 Oracle and/or its affiliates.
8 *
9 * This file is part of VirtualBox base platform packages, as
10 * available from https://www.virtualbox.org.
11 *
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation, in version 3 of the
15 * License.
16 *
17 * This program is distributed in the hope that it will be useful, but
18 * WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 * General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with this program; if not, see <https://www.gnu.org/licenses>.
24 *
25 * The contents of this file may alternatively be used under the terms
26 * of the Common Development and Distribution License Version 1.0
27 * (CDDL), a copy of it is provided in the "COPYING.CDDL" file included
28 * in the VirtualBox distribution, in which case the provisions of the
29 * CDDL are applicable instead of those of the GPL.
30 *
31 * You may elect to license modified versions of this file under the
32 * terms and conditions of either the GPL or the CDDL or both.
33 *
34 * SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
35 */
36
37
38/*********************************************************************************************************************************
39* Header Files *
40*********************************************************************************************************************************/
41#include <iprt/inifile.h>
42#include "internal/iprt.h"
43
44#include <iprt/asm.h>
45#include <iprt/assert.h>
46#include <iprt/ctype.h>
47#include <iprt/err.h>
48#include <iprt/latin1.h>
49#include <iprt/mem.h>
50#include <iprt/string.h>
51#include <iprt/utf16.h>
52#include <iprt/vfs.h>
53
54#include "internal/magics.h"
55
56
57/*********************************************************************************************************************************
58* Defined Constants And Macros *
59*********************************************************************************************************************************/
60/** @def RTINIFILE_MAX_SIZE
61 * The maximum INI-file size we accept loading. */
62#if ARCH_BITS > 32
63# define RTINIFILE_MAX_SIZE (_64M - 2U)
64#elif ARCH_BITS > 16
65# define RTINIFILE_MAX_SIZE (_16M - 2U)
66#else
67# define RTINIFILE_MAX_SIZE (_64K - 2U)
68#endif
69
70/** @def RTINIFILE_MAX_SECTIONS
71 * The maximum number of sections we accept in an INI-file. */
72#if ARCH_BITS > 32
73# define RTINIFILE_MAX_SECTIONS (_1M)
74#elif ARCH_BITS > 16
75# define RTINIFILE_MAX_SECTIONS (_256K)
76#else
77# define RTINIFILE_MAX_SECTIONS (_1K)
78#endif
79
80
81/*********************************************************************************************************************************
82* Structures and Typedefs *
83*********************************************************************************************************************************/
84/**
85 * File encoding types.
86 */
87typedef enum RTINIFILEENCODING
88{
89 /** The customary invalid zero value. */
90 RTINIFILEENCODING_INVALID = 0,
91 /** We treat this as latin-1. */
92 RTINIFILEENCODING_ANSI,
93 /** UTF-8. */
94 RTINIFILEENCODING_UTF8,
95 /** Little endian UTF-16. */
96 RTINIFILEENCODING_UTF16LE,
97 /** Big endian UTF-16. */
98 RTINIFILEENCODING_UTF16BE,
99 /** End of valid encoding types. */
100 RTINIFILEENCODING_END
101} RTINIFILEENCODING;
102
103
104/**
105 * Preparsed section info.
106 */
107typedef struct RTINIFILESECTION
108{
109 /** The section name offset (byte). */
110 uint32_t offName;
111 /** The section length in bytes starting with the name. */
112 uint32_t cchSection;
113 /** The UTF-8 length of the section name. */
114 uint32_t cchName;
115 /** Offset into the section where to start looking for values. */
116 uint32_t cchSkipToValues : 24;
117 /** @todo use 4 bits for flags and stuff. like escaped name. */
118} RTINIFILESECTION;
119/** Pointer to preparsed section info. */
120typedef RTINIFILESECTION *PRTINIFILESECTION;
121
122
123/**
124 * INI-file instance data.
125 */
126typedef struct RTINIFILEINT
127{
128 /** Magic value (RTINIFILEINT_MAGIC). */
129 uint32_t u32Magic;
130 /** Reference counter. */
131 uint32_t volatile cRefs;
132 /** The file we're working on. */
133 RTVFSFILE hVfsFile;
134 /** Flags, RTINIFILE_F_XXX. */
135 uint32_t fFlags;
136
137 /** The original file encoding. */
138 RTINIFILEENCODING enmEncoding;
139 /** Pointer to the file content (converted to UTF-8). */
140 char *pszFile;
141 /** The file size. */
142 uint32_t cbFile;
143 /** Number of sections. */
144 uint32_t cSections;
145 /** Sections in the loaded file. */
146 PRTINIFILESECTION paSections;
147
148} RTINIFILEINT;
149/** Pointer to an INI-file instance. */
150typedef RTINIFILEINT *PRTINIFILEINT;
151
152
153static int rtIniFileLoad(PRTINIFILEINT pThis)
154{
155 /*
156 * Load the entire file into memory, ensuring two terminating zeros.
157 */
158 uint64_t cbFile;
159 int rc = RTVfsFileQuerySize(pThis->hVfsFile, &cbFile);
160 AssertRCReturn(rc, rc);
161
162 if (cbFile > RTINIFILE_MAX_SIZE)
163 return VERR_TOO_MUCH_DATA;
164 if (cbFile == 0)
165 return VINF_SUCCESS; /* Nothing to do. */
166
167 pThis->cbFile = (uint32_t)cbFile;
168 pThis->pszFile = (char *)RTMemAllocZ(pThis->cbFile + 2);
169 if (!pThis->pszFile)
170 return VERR_NO_MEMORY;
171
172 rc = RTVfsFileReadAt(pThis->hVfsFile, 0, pThis->pszFile, pThis->cbFile, NULL);
173 AssertRCReturn(rc, rc);
174
175 /*
176 * Detect encoding and convert to BOM prefixed UTF-8.
177 */
178 if ( (uint8_t)pThis->pszFile[0] == UINT8_C(0xef)
179 && (uint8_t)pThis->pszFile[1] == UINT8_C(0xbb)
180 && (uint8_t)pThis->pszFile[2] == UINT8_C(0xbf))
181 {
182 pThis->enmEncoding = RTINIFILEENCODING_UTF8;
183 rc = RTStrValidateEncoding(&pThis->pszFile[3]);
184 if (RT_FAILURE(rc))
185 return rc;
186 }
187 else
188 {
189 size_t cchUtf8;
190 if ( (uint8_t)pThis->pszFile[0] == UINT8_C(0xfe)
191 && (uint8_t)pThis->pszFile[1] == UINT8_C(0xff))
192 {
193 pThis->enmEncoding = RTINIFILEENCODING_UTF16BE;
194 rc = RTUtf16BigCalcUtf8LenEx((PCRTUTF16)&pThis->pszFile[2], RTSTR_MAX, &cchUtf8);
195 }
196 else if ( (uint8_t)pThis->pszFile[0] == UINT8_C(0xff)
197 && (uint8_t)pThis->pszFile[1] == UINT8_C(0xfe))
198 {
199 pThis->enmEncoding = RTINIFILEENCODING_UTF16LE;
200 rc = RTUtf16LittleCalcUtf8LenEx((PCRTUTF16)&pThis->pszFile[2], RTSTR_MAX, &cchUtf8);
201 }
202 else
203 {
204 pThis->enmEncoding = RTINIFILEENCODING_ANSI;
205 rc = RTLatin1CalcUtf8LenEx(pThis->pszFile, RTSTR_MAX, &cchUtf8);
206 }
207 if (RT_FAILURE(rc))
208 return rc;
209
210 char *pszUtf8Bom = (char *)RTMemAllocZ(3 + cchUtf8 + 1);
211 if (!pszUtf8Bom)
212 return VERR_NO_MEMORY;
213 pszUtf8Bom[0] = '\xEF';
214 pszUtf8Bom[1] = '\xBB';
215 pszUtf8Bom[2] = '\xBF';
216
217 char *pszUtf8 = pszUtf8Bom + 3;
218 if (pThis->enmEncoding == RTINIFILEENCODING_UTF16BE)
219 rc = RTUtf16BigToUtf8Ex((PCRTUTF16)&pThis->pszFile[2], RTSTR_MAX, &pszUtf8, cchUtf8 + 1, NULL);
220 else if (pThis->enmEncoding == RTINIFILEENCODING_UTF16LE)
221 rc = RTUtf16LittleToUtf8Ex((PCRTUTF16)&pThis->pszFile[2], RTSTR_MAX, &pszUtf8, cchUtf8 + 1, NULL);
222 else
223 rc = RTLatin1ToUtf8Ex(pThis->pszFile, RTSTR_MAX, &pszUtf8, cchUtf8 + 1, NULL);
224 AssertRCReturnStmt(rc, RTMemFree(pszUtf8Bom), rc);
225
226 RTMemFree(pThis->pszFile);
227 pThis->pszFile = pszUtf8Bom;
228 pThis->cbFile = 3 + (uint32_t)cchUtf8;
229 }
230
231 /*
232 * Do a rough section count.
233 * Section zero is for unsectioned values at the start of the file.
234 */
235 uint32_t cSections = 1;
236 const char *psz = pThis->pszFile + 3;
237 char ch;
238 while ((ch = *psz) != '\0')
239 {
240 while (RT_C_IS_SPACE(ch))
241 ch = *++psz;
242 if (ch == '[')
243 cSections++;
244
245 /* next line. */
246 psz = strchr(psz, '\n');
247 if (psz)
248 psz++;
249 else
250 break;
251 }
252 if (cSections > RTINIFILE_MAX_SECTIONS)
253 return VERR_TOO_MUCH_DATA;
254
255 /*
256 * Allocation section array and do the preparsing.
257 */
258 pThis->paSections = (PRTINIFILESECTION)RTMemAllocZ(sizeof(pThis->paSections[0]) * cSections);
259 if (!pThis->paSections)
260 return VERR_NO_MEMORY;
261
262 uint32_t iSection = 0;
263 pThis->paSections[0].offName = 3;
264 pThis->paSections[0].cchName = 0;
265 pThis->paSections[0].cchSkipToValues = 0;
266 psz = pThis->pszFile + 3;
267 while ((ch = *psz) != '\0')
268 {
269 const char *const pszLine = psz;
270
271 while (RT_C_IS_SPACE(ch))
272 ch = *++psz;
273 if (ch == '[')
274 {
275 /* Complete previous section. */
276 pThis->paSections[iSection].cchSection = (uint32_t)(pszLine - &pThis->pszFile[pThis->paSections[iSection].offName]);
277
278 /* New section. */
279 iSection++;
280 AssertReturn(iSection < cSections, VERR_INTERNAL_ERROR_3);
281 const char * const pszName = ++psz;
282 pThis->paSections[iSection].offName = (uint32_t)(psz - pThis->pszFile);
283
284 /* Figure the name length. We're very very relaxed about terminating bracket. */
285 while ((ch = *psz) != '\0' && ch != ']' && ch != '\r' && ch != '\n')
286 psz++;
287 pThis->paSections[iSection].cchName = (uint32_t)(psz - pszName);
288
289 /* Set skip count to the start of the next line. */
290 while (ch != '\0' && ch != '\n')
291 ch = *++psz;
292 pThis->paSections[iSection].cchSkipToValues = (uint32_t)(psz - pszName + 1);
293
294 if (ch == '\n')
295 psz++;
296 else
297 break;
298 }
299 else
300 {
301 psz = strchr(psz, '\n');
302 if (psz)
303 psz++;
304 else
305 break;
306 }
307 }
308
309 /* Complete the final section. */
310 pThis->paSections[iSection].cchSection = pThis->cbFile - pThis->paSections[iSection].offName;
311 pThis->cSections = iSection + 1;
312
313 return VINF_SUCCESS;
314}
315
316
317/**
318 * Creates a INI-file instance from a VFS file handle.
319 *
320 * @returns IPRT status code
321 * @param phIniFile Where to return the INI-file handle.
322 * @param hVfsFile The VFS file handle (not consumed, additional
323 * reference is retained).
324 * @param fFlags Flags, RTINIFILE_F_XXX.
325 */
326RTDECL(int) RTIniFileCreateFromVfsFile(PRTINIFILE phIniFile, RTVFSFILE hVfsFile, uint32_t fFlags)
327{
328 /*
329 * Validate input, retaining a reference to the file.
330 */
331 AssertPtrReturn(phIniFile, VERR_INVALID_POINTER);
332 AssertReturn(!(fFlags & ~RTINIFILE_F_VALID_MASK), VERR_INVALID_FLAGS);
333
334 uint32_t cRefs = RTVfsFileRetain(hVfsFile);
335 AssertReturn(cRefs != UINT32_MAX, VERR_INVALID_HANDLE);
336
337 /*
338 * Create an instance.
339 */
340 PRTINIFILEINT pThis = (PRTINIFILEINT)RTMemAllocZ(sizeof(*pThis));
341 if (pThis)
342 {
343 pThis->u32Magic = RTINIFILE_MAGIC;
344 pThis->cRefs = 1;
345 pThis->hVfsFile = hVfsFile;
346 pThis->fFlags = fFlags;
347
348 int rc = rtIniFileLoad(pThis);
349 if (RT_SUCCESS(rc))
350 {
351
352 *phIniFile = pThis;
353 return VINF_SUCCESS;
354 }
355 RTIniFileRelease(pThis);
356 return rc;
357 }
358 RTVfsFileRelease(hVfsFile);
359 return VERR_NO_MEMORY;
360}
361
362
363/**
364 * Retains a reference to an INI-file instance.
365 *
366 * @returns New reference count, UINT32_MAX on failure.
367 * @param hIniFile The INI-file handle.
368 */
369RTDECL(uint32_t) RTIniFileRetain(RTINIFILE hIniFile)
370{
371 PRTINIFILEINT pThis = hIniFile;
372 AssertPtrReturn(pThis, UINT32_MAX);
373 AssertReturn(pThis->u32Magic == RTINIFILE_MAGIC, UINT32_MAX);
374
375 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
376 Assert(cRefs > 1);
377 Assert(cRefs < _64K);
378 return cRefs;
379}
380
381
382/**
383 * Releases a reference to an INI-file instance, destroying it if the count
384 * reaches zero.
385 *
386 * @returns New reference count, UINT32_MAX on failure.
387 * @param hIniFile The INI-file handle. NIL is ignored.
388 */
389RTDECL(uint32_t) RTIniFileRelease(RTINIFILE hIniFile)
390{
391 if (hIniFile == NIL_RTINIFILE)
392 return 0;
393 PRTINIFILEINT pThis = hIniFile;
394 AssertPtrReturn(pThis, UINT32_MAX);
395 AssertReturn(pThis->u32Magic == RTINIFILE_MAGIC, UINT32_MAX);
396
397 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
398 Assert(cRefs < _64K);
399 if (cRefs == 0)
400 {
401 AssertReturn(ASMAtomicCmpXchgU32(&pThis->u32Magic, RTINIFILE_MAGIC_DEAD, RTINIFILE_MAGIC), UINT32_MAX);
402 RTMemFree(pThis->paSections);
403 pThis->paSections = NULL;
404 RTMemFree(pThis->pszFile);
405 pThis->pszFile = NULL;
406 RTVfsFileRelease(pThis->hVfsFile);
407 pThis->hVfsFile = NIL_RTVFSFILE;
408 RTMemFree(pThis);
409 }
410 return cRefs;
411}
412
413
414/**
415 * Worker for RTIniFileQueryValue.
416 */
417static int rtIniFileQueryValueInSection(PRTINIFILEINT pThis, PRTINIFILESECTION pSection, const char *pszKey, size_t cchKey,
418 char *pszValue, size_t cbValue, size_t *pcbActual)
419{
420 /*
421 * Scan the section, looking for the matching key.
422 */
423 Assert(pSection->cchSkipToValues <= pSection->cchSection);
424 const char * const pszEnd = &pThis->pszFile[pSection->offName + pSection->cchSection];
425 const char * pszNext = pszEnd;
426 for (const char *psz = &pThis->pszFile[pSection->offName + pSection->cchSkipToValues];
427 (uintptr_t)psz < (uintptr_t)pszEnd;
428 psz = pszNext)
429 {
430 /* Find start of next line so we can use 'continue' to skip a line. */
431 pszNext = strchr(psz, '\n');
432 if (pszNext)
433 pszNext++;
434 else
435 pszNext = pszEnd;
436
437 /* Skip leading spaces. */
438 char ch;
439 while ((ch = *psz) != '\0' && RT_C_IS_SPACE(ch))
440 psz++;
441 if ( ch != ';' /* comment line */
442 && ch != '\n' /* empty line */
443 && ch != '\r' /* empty line */
444 && (uintptr_t)psz < (uintptr_t)pszEnd)
445 {
446 /* Find end of key name, if any. */
447 const char *pszCurKey = psz;
448 size_t cchCurKey;
449 const char *pszEqual;
450 if (ch != '=')
451 {
452 /** @todo deal with escaped equal signs? */
453 pszEqual = strchr(psz, '=');
454 if (pszEqual)
455 {
456 if ((uintptr_t)pszEqual < (uintptr_t)pszNext)
457 cchCurKey = pszEqual - pszCurKey;
458 else
459 continue;
460 }
461 else
462 break;
463
464 /* Strip trailing spaces from the current key name. */
465 while (cchCurKey > 0 && RT_C_IS_SPACE(pszCurKey[cchCurKey - 1]))
466 cchCurKey--;
467 }
468 else
469 {
470 cchCurKey = 0;
471 pszEqual = psz;
472 }
473
474 /* Match the keys. */
475 /** @todo escape sequences? */
476 if ( cchCurKey == cchKey
477 && RTStrNICmp(pszCurKey, pszKey, cchKey) == 0)
478 {
479 /*
480 * Copy out the return value, without quotes.
481 */
482
483 /* Skip leading blanks. */
484 psz = pszEqual + 1;
485 while ((ch = *psz) && RT_C_IS_SPACE(ch) && ch != '\n')
486 psz++;
487
488 /* Strip trailing spaces. */
489 size_t cchCurValue = pszNext - psz;
490 while (cchCurValue > 1 && RT_C_IS_SPACE(psz[cchCurValue - 1]))
491 cchCurValue--;
492
493 /* Strip quotes. */
494 if ( cchCurValue > 2
495 && ( (ch = *psz) == '"'
496 || ch == '\'' )
497 && psz[cchCurValue - 1] == ch)
498 {
499 cchCurValue -= 2;
500 psz++;
501 }
502
503 /* Do the copying. */
504 if (cchCurValue < cbValue)
505 {
506 memcpy(pszValue, psz, cchCurValue);
507 pszValue[cchCurValue] = '\0';
508 if (pcbActual)
509 *pcbActual = cchCurValue;
510 return VINF_SUCCESS;
511 }
512
513 if (cbValue > 0)
514 {
515 memcpy(pszValue, psz, cbValue - 1);
516 pszValue[cbValue - 1] = '\0';
517 }
518 if (pcbActual)
519 *pcbActual = cchCurValue + 1;
520 return VERR_BUFFER_OVERFLOW;
521 }
522 }
523 }
524 return VERR_NOT_FOUND;
525}
526
527
528/**
529 * Queries a value in a section.
530 *
531 * The first matching value is returned. The matching is by default case
532 * insensitive.
533 *
534 * @returns IPRT status code.
535 * @retval VERR_NOT_FOUND if section or key not found.
536 *
537 * @param hIniFile The INI-file handle.
538 * @param pszSection The section name. Pass NULL to refer to the
539 * unsectioned key space at the top of the file.
540 * @param pszKey The key name.
541 * @param pszValue Where to return the value.
542 * @param cbValue Size of the buffer @a pszValue points to.
543 * @param pcbActual Where to return the actual value size excluding
544 * terminator on success. On VERR_BUFFER_OVERFLOW this
545 * will be set to the buffer size needed to hold the
546 * value, terminator included. Optional.
547 */
548RTDECL(int) RTIniFileQueryValue(RTINIFILE hIniFile, const char *pszSection, const char *pszKey,
549 char *pszValue, size_t cbValue, size_t *pcbActual)
550{
551 /*
552 * Validate input.
553 */
554 PRTINIFILEINT pThis = hIniFile;
555 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
556 AssertReturn(pThis->u32Magic == RTINIFILE_MAGIC, VERR_INVALID_HANDLE);
557 AssertPtrNullReturn(pszSection, VERR_INVALID_POINTER);
558 AssertPtrReturn(pszKey, VERR_INVALID_POINTER);
559 size_t const cchKey = strlen(pszKey);
560 if (cbValue)
561 AssertPtrReturn(pszValue, VERR_INVALID_POINTER);
562 AssertPtrNullReturn(pcbActual, VERR_INVALID_POINTER);
563
564 /*
565 * Search relevant sections.
566 */
567 int rc;
568 if (pszSection == NULL)
569 rc = rtIniFileQueryValueInSection(pThis, &pThis->paSections[0], pszKey, cchKey, pszValue, cbValue, pcbActual);
570 else
571 {
572 rc = VERR_NOT_FOUND;
573 uint32_t const cchSection = (uint32_t)strlen(pszSection);
574 for (uint32_t iSection = 1; iSection < pThis->cSections; iSection++)
575 if ( pThis->paSections[iSection].cchName == cchSection
576 && RTStrNICmp(&pThis->pszFile[pThis->paSections[iSection].offName], pszSection, cchSection) == 0)
577 {
578 rc = rtIniFileQueryValueInSection(pThis, &pThis->paSections[iSection], pszKey, cchKey,
579 pszValue, cbValue, pcbActual);
580 if (rc != VERR_NOT_FOUND)
581 break;
582 }
583 }
584 return rc;
585}
586
587
588/**
589 * Worker for RTIniFileQueryPair.
590 *
591 * This can also be used to count the number of pairs in a section.
592 */
593static int rtIniFileQueryPairInSection(PRTINIFILEINT pThis, PRTINIFILESECTION pSection, uint32_t *pidxPair,
594 char *pszKey, size_t cbKey, size_t *pcbKeyActual,
595 char *pszValue, size_t cbValue, size_t *pcbValueActual)
596{
597 uint32_t idxPair = *pidxPair;
598
599 /*
600 * Scan the section, looking for the matching key.
601 */
602 Assert(pSection->cchSkipToValues <= pSection->cchSection);
603 const char * const pszEnd = &pThis->pszFile[pSection->offName + pSection->cchSection];
604 const char * pszNext = pszEnd;
605 for (const char *psz = &pThis->pszFile[pSection->offName + pSection->cchSkipToValues];
606 (uintptr_t)psz < (uintptr_t)pszEnd;
607 psz = pszNext)
608 {
609 /* Find start of next line so we can use 'continue' to skip a line. */
610 pszNext = strchr(psz, '\n');
611 if (pszNext)
612 pszNext++;
613 else
614 pszNext = pszEnd;
615
616 /* Skip leading spaces. */
617 char ch;
618 while ((ch = *psz) != '\0' && RT_C_IS_SPACE(ch))
619 psz++;
620 if ( ch != ';' /* comment line */
621 && ch != '\n' /* empty line */
622 && ch != '\r' /* empty line */
623 && (uintptr_t)psz < (uintptr_t)pszEnd)
624 {
625 /* Find end of key name, if any. */
626 const char *pszCurKey = psz;
627 size_t cchCurKey;
628 const char *pszEqual;
629 if (ch != '=')
630 {
631 /** @todo deal with escaped equal signs? */
632 pszEqual = strchr(psz, '=');
633 if (pszEqual)
634 {
635 if ((uintptr_t)pszEqual < (uintptr_t)pszNext)
636 cchCurKey = pszEqual - pszCurKey;
637 else
638 continue;
639 }
640 else
641 break;
642 }
643 else
644 {
645 cchCurKey = 0;
646 pszEqual = psz;
647 }
648
649 /* Is this the pair we're looking for? */
650 if (idxPair > 0)
651 idxPair--;
652 else
653 {
654 /*
655 * Yes it's the stuff we're looking for.
656 * Prepare the the return stuff.
657 */
658
659 /* Strip trailing spaces from the key name. */
660 while (cchCurKey > 0 && RT_C_IS_SPACE(pszCurKey[cchCurKey - 1]))
661 cchCurKey--;
662
663 /* Skip leading blanks from the value. */
664 psz = pszEqual + 1;
665 while ((ch = *psz) && RT_C_IS_SPACE(ch) && ch != '\n')
666 psz++;
667
668 /* Strip trailing spaces from the value. */
669 size_t cchCurValue = pszNext - psz;
670 while (cchCurValue > 1 && RT_C_IS_SPACE(psz[cchCurValue - 1]))
671 cchCurValue--;
672
673 /* Strip value quotes. */
674 if ( cchCurValue > 2
675 && ( (ch = *psz) == '"'
676 || ch == '\'' )
677 && psz[cchCurValue - 1] == ch)
678 {
679 cchCurValue -= 2;
680 psz++;
681 }
682
683 /*
684 * Copy the stuff out.
685 */
686 if ( cchCurValue < cbValue
687 && cchCurKey < cbKey)
688 {
689 memcpy(pszKey, pszCurKey, cchCurKey);
690 pszKey[cchCurKey] = '\0';
691 if (pcbKeyActual)
692 *pcbKeyActual = cchCurKey;
693
694 memcpy(pszValue, psz, cchCurValue);
695 pszValue[cchCurValue] = '\0';
696 if (pcbValueActual)
697 *pcbValueActual = cchCurValue;
698
699 *pidxPair = 0;
700 return VINF_SUCCESS;
701 }
702
703 /* Buffer overflow. Copy out what we can. */
704 if (cbKey > 0)
705 {
706 if (cchCurKey < cbKey)
707 cbKey = cchCurKey + 1;
708 memcpy(pszKey, pszCurKey, cbKey - 1);
709 pszKey[cbKey - 1] = '\0';
710 }
711 if (pcbKeyActual)
712 *pcbKeyActual = cchCurKey + 1;
713
714 if (cbValue > 0)
715 {
716 if (cchCurValue < cbValue)
717 cbValue = cchCurValue + 1;
718 memcpy(pszValue, psz, cbValue - 1);
719 pszValue[cbValue - 1] = '\0';
720 }
721 if (pcbValueActual)
722 *pcbValueActual = cchCurValue + 1;
723
724 *pidxPair = 0;
725 return VERR_BUFFER_OVERFLOW;
726 }
727 }
728 }
729 *pidxPair = idxPair;
730 return VERR_NOT_FOUND;
731}
732
733
734/**
735 * Queries a key-value pair in a section by ordinal.
736 *
737 * @returns IPRT status code.
738 * @retval VERR_NOT_FOUND if the section wasn't found or if it contains no pair
739 * with the given ordinal value.
740 *
741 * @param hIniFile The INI-file handle.
742 * @param pszSection The section name. Pass NULL to refer to the
743 * unsectioned key space at the top of the file.
744 * @param idxPair The pair to fetch (counting from 0).
745 *
746 * @param pszKey Where to return the key name.
747 * @param cbKey Size of the buffer @a pszKey points to.
748 * @param pcbKeyActual Where to return the actual key size excluding
749 * terminator on success. On VERR_BUFFER_OVERFLOW this
750 * will be set to the buffer size needed to hold the
751 * value, terminator included. Optional.
752 *
753 * @param pszValue Where to return the value.
754 * @param cbValue Size of the buffer @a pszValue points to.
755 * @param pcbValueActual Where to return the actual value size excluding
756 * terminator on success. On VERR_BUFFER_OVERFLOW this
757 * will be set to the buffer size needed to hold the
758 * value, terminator included. Optional.
759 */
760RTDECL(int) RTIniFileQueryPair(RTINIFILE hIniFile, const char *pszSection, uint32_t idxPair,
761 char *pszKey, size_t cbKey, size_t *pcbKeyActual,
762 char *pszValue, size_t cbValue, size_t *pcbValueActual)
763{
764 /*
765 * Validate input.
766 */
767 PRTINIFILEINT pThis = hIniFile;
768 AssertPtrReturn(pThis, VERR_INVALID_HANDLE);
769 AssertReturn(pThis->u32Magic == RTINIFILE_MAGIC, VERR_INVALID_HANDLE);
770 AssertPtrNullReturn(pszSection, VERR_INVALID_POINTER);
771 if (cbKey)
772 AssertPtrReturn(pszKey, VERR_INVALID_POINTER);
773 AssertPtrNullReturn(pcbKeyActual, VERR_INVALID_POINTER);
774 if (cbValue)
775 AssertPtrReturn(pszValue, VERR_INVALID_POINTER);
776 AssertPtrNullReturn(pcbValueActual, VERR_INVALID_POINTER);
777
778 /*
779 * Search relevant sections.
780 */
781 int rc;
782 if (pszSection == NULL)
783 rc = rtIniFileQueryPairInSection(pThis, &pThis->paSections[0], &idxPair,
784 pszKey, cbKey, pcbKeyActual, pszValue, cbValue, pcbValueActual);
785 else
786 {
787 rc = VERR_NOT_FOUND;
788 uint32_t const cchSection = (uint32_t)strlen(pszSection);
789 for (uint32_t iSection = 1; iSection < pThis->cSections; iSection++)
790 if ( pThis->paSections[iSection].cchName == cchSection
791 && RTStrNICmp(&pThis->pszFile[pThis->paSections[iSection].offName], pszSection, cchSection) == 0)
792 {
793 rc = rtIniFileQueryPairInSection(pThis, &pThis->paSections[iSection], &idxPair,
794 pszKey, cbKey, pcbKeyActual, pszValue, cbValue, pcbValueActual);
795 if (rc != VERR_NOT_FOUND)
796 break;
797 }
798 }
799 return rc;
800}
801
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette