VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/dbg/dbgcfg.cpp@ 66299

Last change on this file since 66299 was 62566, checked in by vboxsync, 9 years ago

IPRT: More unused parameters.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 83.9 KB
Line 
1/* $Id: dbgcfg.cpp 62566 2016-07-26 15:16:41Z vboxsync $ */
2/** @file
3 * IPRT - Debugging Configuration.
4 */
5
6/*
7 * Copyright (C) 2013-2016 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#define LOG_GROUP RTLOGGROUP_DBG
32#include <iprt/dbg.h>
33#include "internal/iprt.h"
34
35#include <iprt/alloca.h>
36#include <iprt/asm.h>
37#include <iprt/assert.h>
38#include <iprt/critsect.h>
39#include <iprt/ctype.h>
40#include <iprt/dir.h>
41#include <iprt/err.h>
42#include <iprt/env.h>
43#include <iprt/file.h>
44#ifdef IPRT_WITH_HTTP
45# include <iprt/http.h>
46#endif
47#include <iprt/list.h>
48#include <iprt/log.h>
49#include <iprt/mem.h>
50#include <iprt/path.h>
51#include <iprt/process.h>
52#include <iprt/semaphore.h>
53#include <iprt/string.h>
54#include <iprt/uuid.h>
55#include "internal/magics.h"
56
57
58/*********************************************************************************************************************************
59* Structures and Typedefs *
60*********************************************************************************************************************************/
61/**
62 * String list entry.
63 */
64typedef struct RTDBGCFGSTR
65{
66 /** List entry. */
67 RTLISTNODE ListEntry;
68 /** Domain specific flags. */
69 uint16_t fFlags;
70 /** The length of the string. */
71 uint16_t cch;
72 /** The string. */
73 char sz[1];
74} RTDBGCFGSTR;
75/** Pointer to a string list entry. */
76typedef RTDBGCFGSTR *PRTDBGCFGSTR;
77
78
79/**
80 * Configuration instance.
81 */
82typedef struct RTDBGCFGINT
83{
84 /** The magic value (RTDBGCFG_MAGIC). */
85 uint32_t u32Magic;
86 /** Reference counter. */
87 uint32_t volatile cRefs;
88 /** Flags, see RTDBGCFG_FLAGS_XXX. */
89 uint64_t fFlags;
90
91 /** List of paths to search for debug files and executable images. */
92 RTLISTANCHOR PathList;
93 /** List of debug file suffixes. */
94 RTLISTANCHOR SuffixList;
95 /** List of paths to search for source files. */
96 RTLISTANCHOR SrcPathList;
97
98#ifdef RT_OS_WINDOWS
99 /** The _NT_ALT_SYMBOL_PATH and _NT_SYMBOL_PATH combined. */
100 RTLISTANCHOR NtSymbolPathList;
101 /** The _NT_EXECUTABLE_PATH. */
102 RTLISTANCHOR NtExecutablePathList;
103 /** The _NT_SOURCE_PATH. */
104 RTLISTANCHOR NtSourcePath;
105#endif
106
107 /** Log callback function. */
108 PFNRTDBGCFGLOG pfnLogCallback;
109 /** User argument to pass to the log callback. */
110 void *pvLogUser;
111
112 /** Critical section protecting the instance data. */
113 RTCRITSECTRW CritSect;
114} *PRTDBGCFGINT;
115
116/**
117 * Mnemonics map entry for a 64-bit unsigned property value.
118 */
119typedef struct RTDBGCFGU64MNEMONIC
120{
121 /** The flags to set or clear. */
122 uint64_t fFlags;
123 /** The mnemonic. */
124 const char *pszMnemonic;
125 /** The length of the mnemonic. */
126 uint8_t cchMnemonic;
127 /** If @c true, the bits in fFlags will be set, if @c false they will be
128 * cleared. */
129 bool fSet;
130} RTDBGCFGU64MNEMONIC;
131/** Pointer to a read only mnemonic map entry for a uint64_t property. */
132typedef RTDBGCFGU64MNEMONIC const *PCRTDBGCFGU64MNEMONIC;
133
134
135/** @name Open flags.
136 * @{ */
137/** The operative system mask. The values are RT_OPSYS_XXX. */
138#define RTDBGCFG_O_OPSYS_MASK UINT32_C(0x000000ff)
139/** The files may be compressed MS styled. */
140#define RTDBGCFG_O_MAYBE_COMPRESSED_MS RT_BIT_32(26)
141/** Whether to make a recursive search. */
142#define RTDBGCFG_O_RECURSIVE RT_BIT_32(27)
143/** We're looking for a separate debug file. */
144#define RTDBGCFG_O_EXT_DEBUG_FILE RT_BIT_32(28)
145/** We're looking for an executable image. */
146#define RTDBGCFG_O_EXECUTABLE_IMAGE RT_BIT_32(29)
147/** The file search should be done in an case insensitive fashion. */
148#define RTDBGCFG_O_CASE_INSENSITIVE RT_BIT_32(30)
149/** Use Windbg style symbol servers when encountered in the path. */
150#define RTDBGCFG_O_SYMSRV RT_BIT_32(31)
151/** @} */
152
153
154/*********************************************************************************************************************************
155* Defined Constants And Macros *
156*********************************************************************************************************************************/
157/** Validates a debug module handle and returns rc if not valid. */
158#define RTDBGCFG_VALID_RETURN_RC(pThis, rc) \
159 do { \
160 AssertPtrReturn((pThis), (rc)); \
161 AssertReturn((pThis)->u32Magic == RTDBGCFG_MAGIC, (rc)); \
162 AssertReturn((pThis)->cRefs > 0, (rc)); \
163 } while (0)
164
165
166/*********************************************************************************************************************************
167* Global Variables *
168*********************************************************************************************************************************/
169/** Mnemonics map for RTDBGCFGPROP_FLAGS. */
170static const RTDBGCFGU64MNEMONIC g_aDbgCfgFlags[] =
171{
172 { RTDBGCFG_FLAGS_DEFERRED, RT_STR_TUPLE("deferred"), true },
173 { RTDBGCFG_FLAGS_DEFERRED, RT_STR_TUPLE("nodeferred"), false },
174 { RTDBGCFG_FLAGS_NO_SYM_SRV, RT_STR_TUPLE("symsrv"), false },
175 { RTDBGCFG_FLAGS_NO_SYM_SRV, RT_STR_TUPLE("nosymsrv"), true },
176 { RTDBGCFG_FLAGS_NO_SYSTEM_PATHS, RT_STR_TUPLE("syspaths"), false },
177 { RTDBGCFG_FLAGS_NO_SYSTEM_PATHS, RT_STR_TUPLE("nosyspaths"), true },
178 { RTDBGCFG_FLAGS_NO_RECURSIV_SEARCH, RT_STR_TUPLE("rec"), false },
179 { RTDBGCFG_FLAGS_NO_RECURSIV_SEARCH, RT_STR_TUPLE("norec"), true },
180 { RTDBGCFG_FLAGS_NO_RECURSIV_SRC_SEARCH, RT_STR_TUPLE("recsrc"), false },
181 { RTDBGCFG_FLAGS_NO_RECURSIV_SRC_SEARCH, RT_STR_TUPLE("norecsrc"), true },
182 { 0, NULL, 0, false }
183};
184
185
186/** Interesting bundle suffixes. */
187static const char * const g_apszBundleSuffixes[] =
188{
189 ".kext",
190 ".app",
191 ".framework",
192 ".component",
193 ".action",
194 ".caction",
195 ".bundle",
196 ".sourcebundle",
197 ".menu",
198 ".plugin",
199 ".ppp",
200 ".monitorpanel",
201 ".scripting",
202 ".prefPane",
203 ".qlgenerator",
204 ".brailledriver",
205 ".saver",
206 ".SpeechVoice",
207 ".SpeechRecognizer",
208 ".SpeechSynthesizer",
209 ".mdimporter",
210 ".spreporter",
211 ".xpc",
212 NULL
213};
214
215/** Debug bundle suffixes. (Same as above + .dSYM) */
216static const char * const g_apszDSymBundleSuffixes[] =
217{
218 ".dSYM",
219 ".kext.dSYM",
220 ".app.dSYM",
221 ".framework.dSYM",
222 ".component.dSYM",
223 ".action.dSYM",
224 ".caction.dSYM",
225 ".bundle.dSYM",
226 ".sourcebundle.dSYM",
227 ".menu.dSYM",
228 ".plugin.dSYM",
229 ".ppp.dSYM",
230 ".monitorpanel.dSYM",
231 ".scripting.dSYM",
232 ".prefPane.dSYM",
233 ".qlgenerator.dSYM",
234 ".brailledriver.dSYM",
235 ".saver.dSYM",
236 ".SpeechVoice.dSYM",
237 ".SpeechRecognizer.dSYM",
238 ".SpeechSynthesizer.dSYM",
239 ".mdimporter.dSYM",
240 ".spreporter.dSYM",
241 ".xpc.dSYM",
242 NULL
243};
244
245
246
247/**
248 * Runtime logging, level 1.
249 *
250 * @param pThis The debug config instance data.
251 * @param pszFormat The message format string.
252 * @param ... Arguments references in the format string.
253 */
254static void rtDbgCfgLog1(PRTDBGCFGINT pThis, const char *pszFormat, ...)
255{
256 if (LogIsEnabled() || (pThis && pThis->pfnLogCallback))
257 {
258 va_list va;
259 va_start(va, pszFormat);
260 char *pszMsg = RTStrAPrintf2V(pszFormat, va);
261 va_end(va);
262
263 Log(("RTDbgCfg: %s", pszMsg));
264 if (pThis && pThis->pfnLogCallback)
265 pThis->pfnLogCallback(pThis, 1, pszMsg, pThis->pvLogUser);
266 RTStrFree(pszMsg);
267 }
268}
269
270
271/**
272 * Runtime logging, level 2.
273 *
274 * @param pThis The debug config instance data.
275 * @param pszFormat The message format string.
276 * @param ... Arguments references in the format string.
277 */
278static void rtDbgCfgLog2(PRTDBGCFGINT pThis, const char *pszFormat, ...)
279{
280 if (LogIs2Enabled() || (pThis && pThis->pfnLogCallback))
281 {
282 va_list va;
283 va_start(va, pszFormat);
284 char *pszMsg = RTStrAPrintf2V(pszFormat, va);
285 va_end(va);
286
287 Log(("RTDbgCfg: %s", pszMsg));
288 if (pThis && pThis->pfnLogCallback)
289 pThis->pfnLogCallback(pThis, 2, pszMsg, pThis->pvLogUser);
290 RTStrFree(pszMsg);
291 }
292}
293
294
295/**
296 * Checks if the file system at the given path is case insensitive or not.
297 *
298 * @returns true / false
299 * @param pszPath The path to query about.
300 */
301static int rtDbgCfgIsFsCaseInsensitive(const char *pszPath)
302{
303 RTFSPROPERTIES Props;
304 int rc = RTFsQueryProperties(pszPath, &Props);
305 if (RT_FAILURE(rc))
306 return RT_OPSYS == RT_OPSYS_DARWIN
307 || RT_OPSYS == RT_OPSYS_DOS
308 || RT_OPSYS == RT_OPSYS_OS2
309 || RT_OPSYS == RT_OPSYS_NT
310 || RT_OPSYS == RT_OPSYS_WINDOWS;
311 return !Props.fCaseSensitive;
312}
313
314
315/**
316 * Worker that does case sensitive file/dir searching.
317 *
318 * @returns true / false.
319 * @param pszPath The path buffer containing an existing directory and
320 * at @a offLastComp the name we're looking for.
321 * RTPATH_MAX in size. On success, this last component
322 * will have the correct case. On failure, the last
323 * component is stripped off.
324 * @param offLastComp The offset of the last component (for chopping it
325 * off).
326 * @param enmType What kind of thing we're looking for.
327 */
328static bool rtDbgCfgIsXxxxAndFixCaseWorker(char *pszPath, size_t offLastComp, RTDIRENTRYTYPE enmType)
329{
330 /** @todo IPRT should generalize this so we can use host specific tricks to
331 * speed it up. */
332
333 char *pszName = &pszPath[offLastComp];
334
335 /* Return straight away if the name isn't case foldable. */
336 if (!RTStrIsCaseFoldable(pszName))
337 {
338 *pszName = '\0';
339 return false;
340 }
341
342 /*
343 * Try some simple case folding games.
344 */
345 RTStrToLower(pszName);
346 if (RTFileExists(pszPath))
347 return true;
348
349 RTStrToUpper(pszName);
350 if (RTFileExists(pszPath))
351 return true;
352
353 /*
354 * Open the directory and check each entry in it.
355 */
356 char chSaved = *pszName;
357 *pszName = '\0';
358
359 PRTDIR pDir;
360 int rc = RTDirOpen(&pDir, pszPath);
361 if (RT_FAILURE(rc))
362 return false;
363
364 *pszName = chSaved;
365
366 for (;;)
367 {
368 /* Read the next entry. */
369 union
370 {
371 RTDIRENTRY Entry;
372 uint8_t ab[_4K];
373 } u;
374 size_t cbBuf = sizeof(u);
375 rc = RTDirRead(pDir, &u.Entry, &cbBuf);
376 if (RT_FAILURE(rc))
377 break;
378
379 if ( !RTStrICmp(pszName, u.Entry.szName)
380 && ( u.Entry.enmType == enmType
381 || u.Entry.enmType == RTDIRENTRYTYPE_UNKNOWN
382 || u.Entry.enmType == RTDIRENTRYTYPE_SYMLINK) )
383 {
384 strcpy(pszName, u.Entry.szName);
385 if (u.Entry.enmType != enmType)
386 RTDirQueryUnknownType(pszPath, true /*fFollowSymlinks*/, &u.Entry.enmType);
387 if (u.Entry.enmType == enmType)
388 {
389 RTDirClose(pDir);
390 return true;
391 }
392 }
393 }
394
395 RTDirClose(pDir);
396 *pszName = '\0';
397
398 return false;
399}
400
401
402/**
403 * Appends @a pszSubDir to @a pszPath and check whether it exists and is a
404 * directory.
405 *
406 * If @a fCaseInsensitive is set, we will do a case insensitive search for a
407 * matching sub directory.
408 *
409 * @returns true / false
410 * @param pszPath The path buffer containing an existing
411 * directory. RTPATH_MAX in size.
412 * @param pszSubDir The sub directory to append.
413 * @param fCaseInsensitive Whether case insensitive searching is required.
414 */
415static bool rtDbgCfgIsDirAndFixCase(char *pszPath, const char *pszSubDir, bool fCaseInsensitive)
416{
417 /* Save the length of the input path so we can restore it in the case
418 insensitive branch further down. */
419 size_t const cchPath = strlen(pszPath);
420
421 /*
422 * Append the sub directory and check if we got a hit.
423 */
424 int rc = RTPathAppend(pszPath, RTPATH_MAX, pszSubDir);
425 if (RT_FAILURE(rc))
426 return false;
427
428 if (RTDirExists(pszPath))
429 return true;
430
431 /*
432 * Do case insensitive lookup if requested.
433 */
434 if (fCaseInsensitive)
435 return rtDbgCfgIsXxxxAndFixCaseWorker(pszPath, cchPath, RTDIRENTRYTYPE_DIRECTORY);
436
437 pszPath[cchPath] = '\0';
438 return false;
439}
440
441
442/**
443 * Appends @a pszSubDir1 and @a pszSuffix to @a pszPath and check whether it
444 * exists and is a directory.
445 *
446 * If @a fCaseInsensitive is set, we will do a case insensitive search for a
447 * matching sub directory.
448 *
449 * @returns true / false
450 * @param pszPath The path buffer containing an existing
451 * directory. RTPATH_MAX in size.
452 * @param pszSubDir The sub directory to append.
453 * @param pszSuffix The suffix to append.
454 * @param fCaseInsensitive Whether case insensitive searching is required.
455 */
456static bool rtDbgCfgIsDirAndFixCase2(char *pszPath, const char *pszSubDir, const char *pszSuffix, bool fCaseInsensitive)
457{
458 Assert(!strpbrk(pszSuffix, ":/\\"));
459
460 /* Save the length of the input path so we can restore it in the case
461 insensitive branch further down. */
462 size_t const cchPath = strlen(pszPath);
463
464 /*
465 * Append the subdirectory and suffix, then check if we got a hit.
466 */
467 int rc = RTPathAppend(pszPath, RTPATH_MAX, pszSubDir);
468 if (RT_SUCCESS(rc))
469 {
470 rc = RTStrCat(pszPath, RTPATH_MAX, pszSuffix);
471 if (RT_SUCCESS(rc))
472 {
473 if (RTDirExists(pszPath))
474 return true;
475
476 /*
477 * Do case insensitive lookup if requested.
478 */
479 if (fCaseInsensitive)
480 return rtDbgCfgIsXxxxAndFixCaseWorker(pszPath, cchPath, RTDIRENTRYTYPE_DIRECTORY);
481 }
482 }
483
484 pszPath[cchPath] = '\0';
485 return false;
486}
487
488
489/**
490 * Appends @a pszFilename to @a pszPath and check whether it exists and is a
491 * directory.
492 *
493 * If @a fCaseInsensitive is set, we will do a case insensitive search for a
494 * matching filename.
495 *
496 * @returns true / false
497 * @param pszPath The path buffer containing an existing
498 * directory. RTPATH_MAX in size.
499 * @param pszFilename The filename to append.
500 * @param pszSuffix Optional filename suffix to append.
501 * @param fCaseInsensitive Whether case insensitive searching is required.
502 * @param fMsCompressed Whether to look for the MS compressed file name
503 * variant.
504 * @param pfProbablyCompressed This is set to true if a MS compressed
505 * filename variant is returned. Optional.
506 */
507static bool rtDbgCfgIsFileAndFixCase(char *pszPath, const char *pszFilename, const char *pszSuffix, bool fCaseInsensitive,
508 bool fMsCompressed, bool *pfProbablyCompressed)
509{
510 /* Save the length of the input path so we can restore it in the case
511 insensitive branch further down. */
512 size_t cchPath = strlen(pszPath);
513 if (pfProbablyCompressed)
514 *pfProbablyCompressed = false;
515
516 /*
517 * Append the filename and optionally suffix, then check if we got a hit.
518 */
519 int rc = RTPathAppend(pszPath, RTPATH_MAX, pszFilename);
520 if (RT_FAILURE(rc))
521 return false;
522 if (pszSuffix)
523 {
524 Assert(!fMsCompressed);
525 rc = RTStrCat(pszPath, RTPATH_MAX, pszSuffix);
526 if (RT_FAILURE(rc))
527 return false;
528 }
529
530 if (RTFileExists(pszPath))
531 return true;
532
533 /*
534 * Do case insensitive file lookup if requested.
535 */
536 if (fCaseInsensitive)
537 {
538 if (rtDbgCfgIsXxxxAndFixCaseWorker(pszPath, cchPath, RTDIRENTRYTYPE_FILE))
539 return true;
540 }
541
542 /*
543 * Look for MS compressed file if requested.
544 */
545 if ( fMsCompressed
546 && (unsigned char)pszFilename[strlen(pszFilename) - 1] < 0x7f)
547 {
548 pszPath[cchPath] = '\0';
549 rc = RTPathAppend(pszPath, RTPATH_MAX, pszFilename);
550 AssertRCReturn(rc, false);
551 pszPath[strlen(pszPath) - 1] = '_';
552
553 if (pfProbablyCompressed)
554 *pfProbablyCompressed = true;
555
556 if ( RTFileExists(pszPath)
557 || ( fCaseInsensitive
558 && rtDbgCfgIsXxxxAndFixCaseWorker(pszPath, cchPath, RTDIRENTRYTYPE_FILE) ))
559 return true;
560
561 if (pfProbablyCompressed)
562 *pfProbablyCompressed = false;
563 }
564
565 pszPath[cchPath] = '\0';
566 return false;
567}
568
569
570static int rtDbgCfgTryOpenDir(PRTDBGCFGINT pThis, char *pszPath, PRTPATHSPLIT pSplitFn, uint32_t fFlags,
571 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
572{
573 int rcRet = VWRN_NOT_FOUND;
574 int rc2;
575
576 /* If the directory doesn't exist, just quit immediately.
577 Note! Our case insensitivity doesn't extend to the search dirs themselfs,
578 only to the bits under neath them. */
579 if (!RTDirExists(pszPath))
580 {
581 rtDbgCfgLog2(pThis, "Dir does not exist: '%s'\n", pszPath);
582 return rcRet;
583 }
584
585 /* Figure out whether we have to do a case sensitive search or not.
586 Note! As a simplification, we don't ask for case settings in each
587 directory under the user specified path, we assume the file
588 systems that mounted there have compatible settings. Faster
589 that way. */
590 bool const fCaseInsensitive = (fFlags & RTDBGCFG_O_CASE_INSENSITIVE)
591 && !rtDbgCfgIsFsCaseInsensitive(pszPath);
592
593 size_t const cchPath = strlen(pszPath);
594
595 /*
596 * Look for the file with less and less of the original path given.
597 */
598 for (unsigned i = RTPATH_PROP_HAS_ROOT_SPEC(pSplitFn->fProps); i < pSplitFn->cComps; i++)
599 {
600 pszPath[cchPath] = '\0';
601
602 rc2 = VINF_SUCCESS;
603 for (unsigned j = i; j < pSplitFn->cComps - 1U && RT_SUCCESS(rc2); j++)
604 if (!rtDbgCfgIsDirAndFixCase(pszPath, pSplitFn->apszComps[i], fCaseInsensitive))
605 rc2 = VERR_FILE_NOT_FOUND;
606
607 if (RT_SUCCESS(rc2))
608 {
609 if (rtDbgCfgIsFileAndFixCase(pszPath, pSplitFn->apszComps[pSplitFn->cComps - 1], NULL /*pszSuffix*/,
610 fCaseInsensitive, false, NULL))
611 {
612 rtDbgCfgLog1(pThis, "Trying '%s'...\n", pszPath);
613 rc2 = pfnCallback(pThis, pszPath, pvUser1, pvUser2);
614 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
615 {
616 if (rc2 == VINF_CALLBACK_RETURN)
617 rtDbgCfgLog1(pThis, "Found '%s'.\n", pszPath);
618 else
619 rtDbgCfgLog1(pThis, "Error opening '%s'.\n", pszPath);
620 return rc2;
621 }
622 rtDbgCfgLog1(pThis, "Error %Rrc opening '%s'.\n", rc2, pszPath);
623 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rcRet))
624 rcRet = rc2;
625 }
626 }
627 }
628
629 /*
630 * Do a recursive search if requested.
631 */
632 if ( (fFlags & RTDBGCFG_O_RECURSIVE)
633 && !(pThis->fFlags & RTDBGCFG_FLAGS_NO_RECURSIV_SEARCH) )
634 {
635 /** @todo Recursive searching will be done later. */
636 }
637
638 return rcRet;
639}
640
641static int rtDbgCfgUnpackMsCacheFile(PRTDBGCFGINT pThis, char *pszPath, const char *pszFilename)
642{
643 rtDbgCfgLog2(pThis, "Unpacking '%s'...\n", pszPath);
644
645 /*
646 * Duplicate the source file path, just for simplicity and restore the
647 * final character in the orignal. We cheerfully ignorining any
648 * possibility of multibyte UTF-8 sequences just like the caller did when
649 * setting it to '_'.
650 */
651 char *pszSrcArchive = RTStrDup(pszPath);
652 if (!pszSrcArchive)
653 return VERR_NO_STR_MEMORY;
654
655 pszPath[strlen(pszPath) - 1] = RT_C_TO_LOWER(pszFilename[strlen(pszFilename) - 1]);
656
657
658 /*
659 * Figuring out the argument list for the platform specific unpack util.
660 */
661#ifdef RT_OS_WINDOWS
662 RTPathChangeToDosSlashes(pszSrcArchive, false /*fForce*/);
663 RTPathChangeToDosSlashes(pszPath, false /*fForce*/);
664 const char *papszArgs[] =
665 {
666 "expand.exe",
667 pszSrcArchive,
668 pszPath,
669 NULL
670 };
671
672#else
673 char szExtractDir[RTPATH_MAX];
674 strcpy(szExtractDir, pszPath);
675 RTPathStripFilename(szExtractDir);
676
677 const char *papszArgs[] =
678 {
679 "cabextract",
680 "-L", /* Lower case extracted files. */
681 "-d", szExtractDir, /* Extraction path */
682 pszSrcArchive,
683 NULL
684 };
685#endif
686
687 /*
688 * Do the unpacking.
689 */
690 RTPROCESS hChild;
691 int rc = RTProcCreate(papszArgs[0], papszArgs, RTENV_DEFAULT,
692#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
693 RTPROC_FLAGS_NO_WINDOW | RTPROC_FLAGS_HIDDEN | RTPROC_FLAGS_SEARCH_PATH,
694#else
695 RTPROC_FLAGS_SEARCH_PATH,
696#endif
697 &hChild);
698 if (RT_SUCCESS(rc))
699 {
700 RTPROCSTATUS ProcStatus;
701 rc = RTProcWait(hChild, RTPROCWAIT_FLAGS_BLOCK, &ProcStatus);
702 if (RT_SUCCESS(rc))
703 {
704 if ( ProcStatus.enmReason == RTPROCEXITREASON_NORMAL
705 && ProcStatus.iStatus == 0)
706 {
707 if (RTPathExists(pszPath))
708 {
709 rtDbgCfgLog1(pThis, "Successfully unpacked '%s' to '%s'.\n", pszSrcArchive, pszPath);
710 rc = VINF_SUCCESS;
711 }
712 else
713 {
714 rtDbgCfgLog1(pThis, "Successfully ran unpacker on '%s', but '%s' is missing!\n", pszSrcArchive, pszPath);
715 rc = VERR_ZIP_ERROR;
716 }
717 }
718 else
719 {
720 rtDbgCfgLog2(pThis, "Unpacking '%s' failed: iStatus=%d enmReason=%d\n",
721 pszSrcArchive, ProcStatus.iStatus, ProcStatus.enmReason);
722 rc = VERR_ZIP_CORRUPTED;
723 }
724 }
725 else
726 rtDbgCfgLog1(pThis, "Error waiting for process: %Rrc\n", rc);
727
728 }
729 else
730 rtDbgCfgLog1(pThis, "Error starting unpack process '%s': %Rrc\n", papszArgs[0], rc);
731
732 return rc;
733}
734
735
736static int rtDbgCfgTryDownloadAndOpen(PRTDBGCFGINT pThis, const char *pszServer, char *pszPath,
737 const char *pszCacheSubDir, const char *pszUuidMappingSubDir,
738 PRTPATHSPLIT pSplitFn, const char *pszCacheSuffix, uint32_t fFlags,
739 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
740{
741#ifdef IPRT_WITH_HTTP
742 RT_NOREF_PV(pszUuidMappingSubDir); /** @todo do we bother trying pszUuidMappingSubDir? */
743 RT_NOREF_PV(pszCacheSuffix); /** @todo do we bother trying pszUuidMappingSubDir? */
744 RT_NOREF_PV(fFlags);
745
746 if (pThis->fFlags & RTDBGCFG_FLAGS_NO_SYM_SRV)
747 return VWRN_NOT_FOUND;
748 if (!pszCacheSubDir || !*pszCacheSubDir)
749 return VWRN_NOT_FOUND;
750
751 /*
752 * Create the path.
753 */
754 size_t cchTmp = strlen(pszPath);
755
756 int rc = RTDirCreateFullPath(pszPath, 0766);
757 if (!RTDirExists(pszPath))
758 {
759 Log(("Error creating cache dir '%s': %Rrc\n", pszPath, rc));
760 return rc;
761 }
762
763 const char *pszFilename = pSplitFn->apszComps[pSplitFn->cComps - 1];
764 rc = RTPathAppend(pszPath, RTPATH_MAX, pszFilename);
765 if (RT_FAILURE(rc))
766 return rc;
767 RTStrToLower(&pszPath[cchTmp]);
768 if (!RTDirExists(pszPath))
769 {
770 rc = RTDirCreate(pszPath, 0766, 0);
771 if (RT_FAILURE(rc))
772 {
773 Log(("RTDirCreate(%s) -> %Rrc\n", pszPath, rc));
774 }
775 }
776
777 rc = RTPathAppend(pszPath, RTPATH_MAX, pszCacheSubDir);
778 if (RT_FAILURE(rc))
779 return rc;
780 if (!RTDirExists(pszPath))
781 {
782 rc = RTDirCreate(pszPath, 0766, 0);
783 if (RT_FAILURE(rc))
784 {
785 Log(("RTDirCreate(%s) -> %Rrc\n", pszPath, rc));
786 }
787 }
788
789 /* Prepare the destination file name while we're here. */
790 cchTmp = strlen(pszPath);
791 RTStrToLower(&pszPath[cchTmp]);
792 rc = RTPathAppend(pszPath, RTPATH_MAX, pszFilename);
793 if (RT_FAILURE(rc))
794 return rc;
795
796 /*
797 * Download the file.
798 */
799 RTHTTP hHttp;
800 rc = RTHttpCreate(&hHttp);
801 if (RT_FAILURE(rc))
802 return rc;
803 RTHttpUseSystemProxySettings(hHttp);
804
805 static const char * const s_apszHeaders[] =
806 {
807 "User-Agent: Microsoft-Symbol-Server/6.6.0999.9",
808 "Pragma: no-cache",
809 };
810
811 rc = RTHttpSetHeaders(hHttp, RT_ELEMENTS(s_apszHeaders), s_apszHeaders);
812 if (RT_SUCCESS(rc))
813 {
814 char szUrl[_2K];
815 RTStrPrintf(szUrl, sizeof(szUrl), "%s/%s/%s/%s", pszServer, pszFilename, pszCacheSubDir, pszFilename);
816
817 /** @todo Use some temporary file name and rename it after the operation
818 * since not all systems support read-deny file sharing
819 * settings. */
820 rtDbgCfgLog2(pThis, "Downloading '%s' to '%s'...\n", szUrl, pszPath);
821 rc = RTHttpGetFile(hHttp, szUrl, pszPath);
822 if (RT_FAILURE(rc))
823 {
824 RTFileDelete(pszPath);
825 rtDbgCfgLog1(pThis, "%Rrc on URL '%s'\n", rc, szUrl);
826 }
827 if (rc == VERR_HTTP_NOT_FOUND)
828 {
829 /* Try the compressed version of the file. */
830 pszPath[strlen(pszPath) - 1] = '_';
831 szUrl[strlen(szUrl) - 1] = '_';
832 rtDbgCfgLog2(pThis, "Downloading '%s' to '%s'...\n", szUrl, pszPath);
833 rc = RTHttpGetFile(hHttp, szUrl, pszPath);
834 if (RT_SUCCESS(rc))
835 rc = rtDbgCfgUnpackMsCacheFile(pThis, pszPath, pszFilename);
836 else
837 {
838 rtDbgCfgLog1(pThis, "%Rrc on URL '%s'\n", rc, pszPath);
839 RTFileDelete(pszPath);
840 }
841 }
842 }
843
844 RTHttpDestroy(hHttp);
845
846 if (RT_SUCCESS(rc))
847 {
848 /*
849 * Succeeded in downloading it. Add UUID mapping?
850 */
851 if (pszUuidMappingSubDir)
852 {
853 /** @todo UUID mapping when downloading. */
854 }
855
856 /*
857 * Give the file a try.
858 */
859 Assert(RTFileExists(pszPath));
860 rtDbgCfgLog1(pThis, "Trying '%s'...\n", pszPath);
861 rc = pfnCallback(pThis, pszPath, pvUser1, pvUser2);
862 if (rc == VINF_CALLBACK_RETURN)
863 rtDbgCfgLog1(pThis, "Found '%s'.\n", pszPath);
864 else if (rc == VERR_CALLBACK_RETURN)
865 rtDbgCfgLog1(pThis, "Error opening '%s'.\n", pszPath);
866 else
867 rtDbgCfgLog1(pThis, "Error %Rrc opening '%s'.\n", rc, pszPath);
868 }
869
870 return rc;
871
872#else /* !IPRT_WITH_HTTP */
873 RT_NOREF_PV(pThis); RT_NOREF_PV(pszServer); RT_NOREF_PV(pszPath); RT_NOREF_PV(pszCacheSubDir);
874 RT_NOREF_PV(pszUuidMappingSubDir); RT_NOREF_PV(pSplitFn); RT_NOREF_PV(pszCacheSuffix); RT_NOREF_PV(fFlags);
875 RT_NOREF_PV(pfnCallback); RT_NOREF_PV(pvUser1); RT_NOREF_PV(pvUser2);
876 return VWRN_NOT_FOUND;
877#endif /* !IPRT_WITH_HTTP */
878}
879
880
881static int rtDbgCfgCopyFileToCache(PRTDBGCFGINT pThis, char const *pszSrc, const char *pchCache, size_t cchCache,
882 const char *pszCacheSubDir, const char *pszUuidMappingSubDir, PRTPATHSPLIT pSplitFn)
883{
884 RT_NOREF_PV(pThis); RT_NOREF_PV(pszSrc); RT_NOREF_PV(pchCache); RT_NOREF_PV(cchCache);
885 RT_NOREF_PV(pszUuidMappingSubDir); RT_NOREF_PV(pSplitFn);
886
887 if (!pszCacheSubDir || !*pszCacheSubDir)
888 return VINF_SUCCESS;
889
890 /** @todo copy to cache */
891 return VINF_SUCCESS;
892}
893
894
895static int rtDbgCfgTryOpenCache(PRTDBGCFGINT pThis, char *pszPath, size_t cchCachePath,
896 const char *pszCacheSubDir, const char *pszUuidMappingSubDir,
897 PCRTPATHSPLIT pSplitFn, const char *pszCacheSuffix, uint32_t fFlags,
898 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
899{
900 Assert(pszPath[cchCachePath] == '\0');
901
902 /*
903 * If the cache doesn't exist, fail right away.
904 */
905 if (!pszCacheSubDir || !*pszCacheSubDir)
906 return VWRN_NOT_FOUND;
907 if (!RTDirExists(pszPath))
908 {
909 rtDbgCfgLog2(pThis, "Cache does not exist: '%s'\n", pszPath);
910 return VWRN_NOT_FOUND;
911 }
912
913 /*
914 * If we got a UUID mapping option, try it first as we can hopefully
915 * dispense with case folding.
916 */
917 if (pszUuidMappingSubDir)
918 {
919 int rc = RTPathAppend(pszPath, RTPATH_MAX, pszUuidMappingSubDir);
920 if ( RT_SUCCESS(rc)
921 && RTFileExists(pszPath))
922 {
923 /* Try resolve the path before presenting it to the client, a
924 12 digit filename is of little worth. */
925 char szBackup[RTPATH_MAX];
926 strcpy(szBackup, pszPath);
927 rc = RTPathAbs(szBackup, pszPath, RTPATH_MAX);
928 if (RT_FAILURE(rc))
929 strcpy(pszPath, szBackup);
930
931 /* Do the callback thing. */
932 rtDbgCfgLog1(pThis, "Trying '%s'...\n", pszPath);
933 int rc2 = pfnCallback(pThis, pszPath, pvUser1, pvUser2);
934 if (rc2 == VINF_CALLBACK_RETURN)
935 rtDbgCfgLog1(pThis, "Found '%s' via uuid mapping.\n", pszPath);
936 else if (rc2 == VERR_CALLBACK_RETURN)
937 rtDbgCfgLog1(pThis, "Error opening '%s'.\n", pszPath);
938 else
939 rtDbgCfgLog1(pThis, "Error %Rrc opening '%s'.\n", rc2, pszPath);
940 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
941 return rc2;
942
943 /* Failed, restore the cache path. */
944 memcpy(pszPath, szBackup, cchCachePath);
945 }
946 pszPath[cchCachePath] = '\0';
947 }
948
949 /*
950 * Carefully construct the cache path with case insensitivity in mind.
951 */
952 bool const fCaseInsensitive = (fFlags & RTDBGCFG_O_CASE_INSENSITIVE)
953 && !rtDbgCfgIsFsCaseInsensitive(pszPath);
954 const char *pszFilename = pSplitFn->apszComps[pSplitFn->cComps - 1];
955
956 if (!rtDbgCfgIsDirAndFixCase(pszPath, pszFilename, fCaseInsensitive))
957 return VWRN_NOT_FOUND;
958
959 if (!rtDbgCfgIsDirAndFixCase(pszPath, pszCacheSubDir, fCaseInsensitive))
960 return VWRN_NOT_FOUND;
961
962 bool fProbablyCompressed = false;
963 if (!rtDbgCfgIsFileAndFixCase(pszPath, pszFilename, pszCacheSuffix, fCaseInsensitive,
964 RT_BOOL(fFlags & RTDBGCFG_O_MAYBE_COMPRESSED_MS), &fProbablyCompressed))
965 return VWRN_NOT_FOUND;
966 if (fProbablyCompressed)
967 {
968 int rc = rtDbgCfgUnpackMsCacheFile(pThis, pszPath, pszFilename);
969 if (RT_FAILURE(rc))
970 return VWRN_NOT_FOUND;
971 }
972
973 rtDbgCfgLog1(pThis, "Trying '%s'...\n", pszPath);
974 int rc2 = pfnCallback(pThis, pszPath, pvUser1, pvUser2);
975 if (rc2 == VINF_CALLBACK_RETURN)
976 rtDbgCfgLog1(pThis, "Found '%s'.\n", pszPath);
977 else if (rc2 == VERR_CALLBACK_RETURN)
978 rtDbgCfgLog1(pThis, "Error opening '%s'.\n", pszPath);
979 else
980 rtDbgCfgLog1(pThis, "Error %Rrc opening '%s'.\n", rc2, pszPath);
981 return rc2;
982}
983
984
985static int rtDbgCfgTryOpenList(PRTDBGCFGINT pThis, PRTLISTANCHOR pList, PRTPATHSPLIT pSplitFn, const char *pszCacheSubDir,
986 const char *pszUuidMappingSubDir, uint32_t fFlags, char *pszPath,
987 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
988{
989 int rcRet = VWRN_NOT_FOUND;
990 int rc2 = VINF_SUCCESS;
991
992 const char *pchCache = NULL;
993 size_t cchCache = 0;
994 int rcCache = VWRN_NOT_FOUND;
995
996 PRTDBGCFGSTR pCur;
997 RTListForEach(pList, pCur, RTDBGCFGSTR, ListEntry)
998 {
999 size_t cchDir = pCur->cch;
1000 const char *pszDir = pCur->sz;
1001 rtDbgCfgLog2(pThis, "Path list entry: '%s'\n", pszDir);
1002
1003 /* This is very simplistic, but we have a unreasonably large path
1004 buffer, so it'll work just fine and simplify things greatly below. */
1005 if (cchDir >= RTPATH_MAX - 8U)
1006 {
1007 if (RT_SUCCESS_NP(rcRet))
1008 rcRet = VERR_FILENAME_TOO_LONG;
1009 continue;
1010 }
1011
1012 /*
1013 * Process the path according to it's type.
1014 */
1015 if (!RTStrNICmp(pszDir, RT_STR_TUPLE("srv*")))
1016 {
1017 /*
1018 * Symbol server.
1019 */
1020 pszDir += sizeof("srv*") - 1;
1021 cchDir -= sizeof("srv*") - 1;
1022 bool fSearchCache = false;
1023 const char *pszServer = (const char *)memchr(pszDir, '*', cchDir);
1024 if (!pszServer)
1025 pszServer = pszDir;
1026 else if (pszServer == pszDir)
1027 continue;
1028 else
1029 {
1030 fSearchCache = true;
1031 pchCache = pszDir;
1032 cchCache = pszServer - pszDir;
1033 pszServer++;
1034 }
1035
1036 /* We don't have any default cache directory, so skip if the cache is missing. */
1037 if (cchCache == 0)
1038 continue;
1039
1040 /* Search the cache first (if we haven't already done so). */
1041 if (fSearchCache)
1042 {
1043 memcpy(pszPath, pchCache, cchCache);
1044 pszPath[cchCache] = '\0';
1045 RTPathChangeToUnixSlashes(pszPath, false);
1046
1047 rcCache = rc2 = rtDbgCfgTryOpenCache(pThis, pszPath, cchCache, pszCacheSubDir, pszUuidMappingSubDir,
1048 pSplitFn, NULL /*pszCacheSuffix*/, fFlags, pfnCallback, pvUser1, pvUser2);
1049 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
1050 return rc2;
1051 }
1052
1053 /* Try downloading the file. */
1054 if (rcCache == VWRN_NOT_FOUND)
1055 {
1056 memcpy(pszPath, pchCache, cchCache);
1057 pszPath[cchCache] = '\0';
1058 RTPathChangeToUnixSlashes(pszPath, false);
1059
1060 rc2 = rtDbgCfgTryDownloadAndOpen(pThis, pszServer, pszPath, pszCacheSubDir, pszUuidMappingSubDir,
1061 pSplitFn, NULL /*pszCacheSuffix*/, fFlags, pfnCallback, pvUser1, pvUser2);
1062 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
1063 return rc2;
1064 }
1065 }
1066 else if (!RTStrNICmp(pszDir, RT_STR_TUPLE("cache*")))
1067 {
1068 /*
1069 * Cache directory.
1070 */
1071 pszDir += sizeof("cache*") - 1;
1072 cchDir -= sizeof("cache*") - 1;
1073 if (!cchDir)
1074 continue;
1075 pchCache = pszDir;
1076 cchCache = cchDir;
1077
1078 memcpy(pszPath, pchCache, cchCache);
1079 pszPath[cchCache] = '\0';
1080 RTPathChangeToUnixSlashes(pszPath, false);
1081
1082 rcCache = rc2 = rtDbgCfgTryOpenCache(pThis, pszPath, cchCache, pszCacheSubDir, pszUuidMappingSubDir,
1083 pSplitFn, NULL /*pszCacheSuffix*/, fFlags, pfnCallback, pvUser1, pvUser2);
1084 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
1085 return rc2;
1086 }
1087 else
1088 {
1089 /*
1090 * Normal directory. Check for our own 'rec*' and 'norec*' prefix
1091 * flags governing recursive searching.
1092 */
1093 uint32_t fFlagsDir = fFlags;
1094 if (!RTStrNICmp(pszDir, RT_STR_TUPLE("rec*")))
1095 {
1096 pszDir += sizeof("rec*") - 1;
1097 cchDir -= sizeof("rec*") - 1;
1098 fFlagsDir |= RTDBGCFG_O_RECURSIVE;
1099 }
1100 else if (!RTStrNICmp(pszDir, RT_STR_TUPLE("norec*")))
1101 {
1102 pszDir += sizeof("norec*") - 1;
1103 cchDir -= sizeof("norec*") - 1;
1104 fFlagsDir &= ~RTDBGCFG_O_RECURSIVE;
1105 }
1106
1107 /* Copy the path into the buffer and do the searching. */
1108 memcpy(pszPath, pszDir, cchDir);
1109 pszPath[cchDir] = '\0';
1110 RTPathChangeToUnixSlashes(pszPath, false);
1111
1112 rc2 = rtDbgCfgTryOpenDir(pThis, pszPath, pSplitFn, fFlagsDir, pfnCallback, pvUser1, pvUser2);
1113 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
1114 {
1115 if ( rc2 == VINF_CALLBACK_RETURN
1116 && cchCache > 0)
1117 rtDbgCfgCopyFileToCache(pThis, pszPath, pchCache, cchCache,
1118 pszCacheSubDir, pszUuidMappingSubDir, pSplitFn);
1119 return rc2;
1120 }
1121 }
1122
1123 /* Propagate errors. */
1124 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rcRet))
1125 rcRet = rc2;
1126 }
1127
1128 return rcRet;
1129}
1130
1131
1132/**
1133 * Common worker routine for Image and debug info opening.
1134 *
1135 * This will not search using for suffixes.
1136 *
1137 * @returns IPRT status code.
1138 * @param hDbgCfg The debugging configuration handle.
1139 * NIL_RTDBGCFG is accepted, but the result is
1140 * that no paths will be searched beyond the
1141 * given and the current directory.
1142 * @param pszFilename The filename to search for. This may or may
1143 * not include a full or partial path.
1144 * @param pszCacheSubDir The cache subdirectory to look in.
1145 * @param pszUuidMappingSubDir UUID mapping subdirectory to check, NULL if
1146 * no mapping wanted.
1147 * @param fFlags Flags and hints.
1148 * @param pfnCallback The open callback routine.
1149 * @param pvUser1 User parameter 1.
1150 * @param pvUser2 User parameter 2.
1151 */
1152static int rtDbgCfgOpenWithSubDir(RTDBGCFG hDbgCfg, const char *pszFilename, const char *pszCacheSubDir,
1153 const char *pszUuidMappingSubDir, uint32_t fFlags,
1154 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1155{
1156 int rcRet = VINF_SUCCESS;
1157 int rc2;
1158
1159 /*
1160 * Do a little validating first.
1161 */
1162 PRTDBGCFGINT pThis = hDbgCfg;
1163 if (pThis != NIL_RTDBGCFG)
1164 RTDBGCFG_VALID_RETURN_RC(pThis, VERR_INVALID_HANDLE);
1165 else
1166 pThis = NULL;
1167 AssertPtrReturn(pszFilename, VERR_INVALID_POINTER);
1168 AssertPtrReturn(pszCacheSubDir, VERR_INVALID_POINTER);
1169 AssertPtrReturn(pfnCallback, VERR_INVALID_POINTER);
1170
1171 /*
1172 * Do some guessing as to the way we should parse the filename and whether
1173 * it's case exact or not.
1174 */
1175 bool fDosPath = strchr(pszFilename, ':') != NULL
1176 || strchr(pszFilename, '\\') != NULL
1177 || RT_OPSYS_USES_DOS_PATHS(fFlags & RTDBGCFG_O_OPSYS_MASK)
1178 || (fFlags & RTDBGCFG_O_CASE_INSENSITIVE);
1179 if (fDosPath)
1180 fFlags |= RTDBGCFG_O_CASE_INSENSITIVE;
1181
1182 rtDbgCfgLog2(pThis, "Looking for '%s' w/ cache subdir '%s' and %#x flags...\n", pszFilename, pszCacheSubDir, fFlags);
1183
1184 PRTPATHSPLIT pSplitFn;
1185 rc2 = RTPathSplitA(pszFilename, &pSplitFn, fDosPath ? RTPATH_STR_F_STYLE_DOS : RTPATH_STR_F_STYLE_UNIX);
1186 if (RT_FAILURE(rc2))
1187 return rc2;
1188 AssertReturnStmt(pSplitFn->fProps & RTPATH_PROP_FILENAME, RTPathSplitFree(pSplitFn), VERR_IS_A_DIRECTORY);
1189
1190 /*
1191 * Try the stored file name first if it has a kind of absolute path.
1192 */
1193 char szPath[RTPATH_MAX];
1194 if (RTPATH_PROP_HAS_ROOT_SPEC(pSplitFn->fProps))
1195 {
1196 rc2 = RTPathSplitReassemble(pSplitFn, RTPATH_STR_F_STYLE_HOST, szPath, sizeof(szPath));
1197 if (RT_SUCCESS(rc2) && RTFileExists(szPath))
1198 {
1199 RTPathChangeToUnixSlashes(szPath, false);
1200 rtDbgCfgLog1(pThis, "Trying '%s'...\n", szPath);
1201 rc2 = pfnCallback(pThis, szPath, pvUser1, pvUser2);
1202 if (rc2 == VINF_CALLBACK_RETURN)
1203 rtDbgCfgLog1(pThis, "Found '%s'.\n", szPath);
1204 else if (rc2 == VERR_CALLBACK_RETURN)
1205 rtDbgCfgLog1(pThis, "Error opening '%s'.\n", szPath);
1206 else
1207 rtDbgCfgLog1(pThis, "Error %Rrc opening '%s'.\n", rc2, szPath);
1208 }
1209 }
1210 if ( rc2 != VINF_CALLBACK_RETURN
1211 && rc2 != VERR_CALLBACK_RETURN)
1212 {
1213 /*
1214 * Try the current directory (will take cover relative paths
1215 * skipped above).
1216 */
1217 rc2 = RTPathGetCurrent(szPath, sizeof(szPath));
1218 if (RT_FAILURE(rc2))
1219 strcpy(szPath, ".");
1220 RTPathChangeToUnixSlashes(szPath, false);
1221
1222 rc2 = rtDbgCfgTryOpenDir(pThis, szPath, pSplitFn, fFlags, pfnCallback, pvUser1, pvUser2);
1223 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rcRet))
1224 rcRet = rc2;
1225
1226 if ( rc2 != VINF_CALLBACK_RETURN
1227 && rc2 != VERR_CALLBACK_RETURN
1228 && pThis)
1229 {
1230 rc2 = RTCritSectRwEnterShared(&pThis->CritSect);
1231 if (RT_SUCCESS(rc2))
1232 {
1233 /*
1234 * Run the applicable lists.
1235 */
1236 rc2 = rtDbgCfgTryOpenList(pThis, &pThis->PathList, pSplitFn, pszCacheSubDir,
1237 pszUuidMappingSubDir, fFlags, szPath, pfnCallback, pvUser1, pvUser2);
1238 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rcRet))
1239 rcRet = rc2;
1240
1241#ifdef RT_OS_WINDOWS
1242 if ( rc2 != VINF_CALLBACK_RETURN
1243 && rc2 != VERR_CALLBACK_RETURN
1244 && (fFlags & RTDBGCFG_O_EXECUTABLE_IMAGE)
1245 && !(pThis->fFlags & RTDBGCFG_FLAGS_NO_SYSTEM_PATHS) )
1246 {
1247 rc2 = rtDbgCfgTryOpenList(pThis, &pThis->NtExecutablePathList, pSplitFn, pszCacheSubDir,
1248 pszUuidMappingSubDir, fFlags, szPath, pfnCallback, pvUser1, pvUser2);
1249 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rcRet))
1250 rcRet = rc2;
1251 }
1252
1253 if ( rc2 != VINF_CALLBACK_RETURN
1254 && rc2 != VERR_CALLBACK_RETURN
1255 && !(pThis->fFlags & RTDBGCFG_FLAGS_NO_SYSTEM_PATHS) )
1256 {
1257 rc2 = rtDbgCfgTryOpenList(pThis, &pThis->NtSymbolPathList, pSplitFn, pszCacheSubDir,
1258 pszUuidMappingSubDir, fFlags, szPath, pfnCallback, pvUser1, pvUser2);
1259 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rcRet))
1260 rcRet = rc2;
1261 }
1262#endif
1263 RTCritSectRwLeaveShared(&pThis->CritSect);
1264 }
1265 else if (RT_SUCCESS(rcRet))
1266 rcRet = rc2;
1267 }
1268 }
1269
1270 RTPathSplitFree(pSplitFn);
1271 if ( rc2 == VINF_CALLBACK_RETURN
1272 || rc2 == VERR_CALLBACK_RETURN)
1273 rcRet = rc2;
1274 else if (RT_SUCCESS(rcRet))
1275 rcRet = VERR_NOT_FOUND;
1276 return rcRet;
1277}
1278
1279
1280RTDECL(int) RTDbgCfgOpenPeImage(RTDBGCFG hDbgCfg, const char *pszFilename, uint32_t cbImage, uint32_t uTimestamp,
1281 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1282{
1283 char szSubDir[32];
1284 RTStrPrintf(szSubDir, sizeof(szSubDir), "%08X%x", uTimestamp, cbImage);
1285 return rtDbgCfgOpenWithSubDir(hDbgCfg, pszFilename, szSubDir, NULL,
1286 RT_OPSYS_WINDOWS /* approx */ | RTDBGCFG_O_SYMSRV | RTDBGCFG_O_CASE_INSENSITIVE
1287 | RTDBGCFG_O_MAYBE_COMPRESSED_MS | RTDBGCFG_O_EXECUTABLE_IMAGE,
1288 pfnCallback, pvUser1, pvUser2);
1289}
1290
1291
1292RTDECL(int) RTDbgCfgOpenPdb70(RTDBGCFG hDbgCfg, const char *pszFilename, PCRTUUID pUuid, uint32_t uAge,
1293 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1294{
1295 char szSubDir[64];
1296 if (!pUuid)
1297 szSubDir[0] = '\0';
1298 else
1299 {
1300 /* Stringify the UUID and remove the dashes. */
1301 int rc2 = RTUuidToStr(pUuid, szSubDir, sizeof(szSubDir));
1302 AssertRCReturn(rc2, rc2);
1303
1304 char *pszSrc = szSubDir;
1305 char *pszDst = szSubDir;
1306 char ch;
1307 while ((ch = *pszSrc++))
1308 if (ch != '-')
1309 *pszDst++ = RT_C_TO_UPPER(ch);
1310
1311 RTStrPrintf(pszDst, &szSubDir[sizeof(szSubDir)] - pszDst, "%X", uAge);
1312 }
1313
1314 return rtDbgCfgOpenWithSubDir(hDbgCfg, pszFilename, szSubDir, NULL,
1315 RT_OPSYS_WINDOWS /* approx */ | RTDBGCFG_O_SYMSRV | RTDBGCFG_O_CASE_INSENSITIVE
1316 | RTDBGCFG_O_MAYBE_COMPRESSED_MS | RTDBGCFG_O_EXT_DEBUG_FILE,
1317 pfnCallback, pvUser1, pvUser2);
1318}
1319
1320
1321RTDECL(int) RTDbgCfgOpenPdb20(RTDBGCFG hDbgCfg, const char *pszFilename, uint32_t cbImage, uint32_t uTimestamp, uint32_t uAge,
1322 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1323{
1324 RT_NOREF_PV(cbImage);
1325 /** @todo test this! */
1326 char szSubDir[32];
1327 RTStrPrintf(szSubDir, sizeof(szSubDir), "%08X%x", uTimestamp, uAge);
1328 return rtDbgCfgOpenWithSubDir(hDbgCfg, pszFilename, szSubDir, NULL,
1329 RT_OPSYS_WINDOWS /* approx */ | RTDBGCFG_O_SYMSRV | RTDBGCFG_O_CASE_INSENSITIVE
1330 | RTDBGCFG_O_MAYBE_COMPRESSED_MS | RTDBGCFG_O_EXT_DEBUG_FILE,
1331 pfnCallback, pvUser1, pvUser2);
1332}
1333
1334
1335RTDECL(int) RTDbgCfgOpenDbg(RTDBGCFG hDbgCfg, const char *pszFilename, uint32_t cbImage, uint32_t uTimestamp,
1336 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1337{
1338 char szSubDir[32];
1339 RTStrPrintf(szSubDir, sizeof(szSubDir), "%08X%x", uTimestamp, cbImage);
1340 return rtDbgCfgOpenWithSubDir(hDbgCfg, pszFilename, szSubDir, NULL,
1341 RT_OPSYS_WINDOWS /* approx */ | RTDBGCFG_O_SYMSRV | RTDBGCFG_O_CASE_INSENSITIVE
1342 | RTDBGCFG_O_MAYBE_COMPRESSED_MS | RTDBGCFG_O_EXT_DEBUG_FILE,
1343 pfnCallback, pvUser1, pvUser2);
1344}
1345
1346
1347RTDECL(int) RTDbgCfgOpenDwo(RTDBGCFG hDbgCfg, const char *pszFilename, uint32_t uCrc32,
1348 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1349{
1350 char szSubDir[32];
1351 RTStrPrintf(szSubDir, sizeof(szSubDir), "%08x", uCrc32);
1352 return rtDbgCfgOpenWithSubDir(hDbgCfg, pszFilename, szSubDir, NULL,
1353 RT_OPSYS_UNKNOWN | RTDBGCFG_O_EXT_DEBUG_FILE,
1354 pfnCallback, pvUser1, pvUser2);
1355}
1356
1357
1358
1359/*
1360 *
1361 * D a r w i n . d S Y M b u n d l e s
1362 * D a r w i n . d S Y M b u n d l e s
1363 * D a r w i n . d S Y M b u n d l e s
1364 *
1365 */
1366
1367/**
1368 * Very similar to rtDbgCfgTryOpenDir.
1369 */
1370static int rtDbgCfgTryOpenDsymBundleInDir(PRTDBGCFGINT pThis, char *pszPath, PRTPATHSPLIT pSplitFn,
1371 const char * const *papszSuffixes, uint32_t fFlags,
1372 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1373{
1374 int rcRet = VWRN_NOT_FOUND;
1375 int rc2;
1376
1377 /* If the directory doesn't exist, just quit immediately.
1378 Note! Our case insensitivity doesn't extend to the search dirs themselfs,
1379 only to the bits under neath them. */
1380 if (!RTDirExists(pszPath))
1381 {
1382 rtDbgCfgLog2(pThis, "Dir does not exist: '%s'\n", pszPath);
1383 return rcRet;
1384 }
1385
1386 /* Figure out whether we have to do a case sensitive search or not.
1387 Note! As a simplification, we don't ask for case settings in each
1388 directory under the user specified path, we assume the file
1389 systems that mounted there have compatible settings. Faster
1390 that way. */
1391 bool const fCaseInsensitive = (fFlags & RTDBGCFG_O_CASE_INSENSITIVE)
1392 && !rtDbgCfgIsFsCaseInsensitive(pszPath);
1393
1394 size_t const cchPath = strlen(pszPath);
1395
1396 /*
1397 * Look for the file with less and less of the original path given.
1398 * Also try out typical bundle extension variations.
1399 */
1400 const char *pszName = pSplitFn->apszComps[pSplitFn->cComps - 1];
1401 for (unsigned i = RTPATH_PROP_HAS_ROOT_SPEC(pSplitFn->fProps); i < pSplitFn->cComps; i++)
1402 {
1403 pszPath[cchPath] = '\0';
1404
1405 rc2 = VINF_SUCCESS;
1406 for (unsigned j = i; j < pSplitFn->cComps - 1U && RT_SUCCESS(rc2); j++)
1407 if (!rtDbgCfgIsDirAndFixCase(pszPath, pSplitFn->apszComps[i], fCaseInsensitive))
1408 rc2 = VERR_FILE_NOT_FOUND;
1409 if (RT_SUCCESS(rc2))
1410 {
1411 for (uint32_t iSuffix = 0; papszSuffixes[iSuffix]; iSuffix++)
1412 {
1413 if ( !rtDbgCfgIsDirAndFixCase2(pszPath, pszName, papszSuffixes[iSuffix], fCaseInsensitive)
1414 && !rtDbgCfgIsDirAndFixCase(pszPath, "Contents", fCaseInsensitive)
1415 && !rtDbgCfgIsDirAndFixCase(pszPath, "Resources", fCaseInsensitive)
1416 && !rtDbgCfgIsDirAndFixCase(pszPath, "DWARF", fCaseInsensitive))
1417 {
1418 if (rtDbgCfgIsFileAndFixCase(pszPath, pszName, NULL /*pszSuffix*/, fCaseInsensitive, false, NULL))
1419 {
1420 rtDbgCfgLog1(pThis, "Trying '%s'...\n", pszPath);
1421 rc2 = pfnCallback(pThis, pszPath, pvUser1, pvUser2);
1422 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
1423 {
1424 if (rc2 == VINF_CALLBACK_RETURN)
1425 rtDbgCfgLog1(pThis, "Found '%s'.\n", pszPath);
1426 else
1427 rtDbgCfgLog1(pThis, "Error opening '%s'.\n", pszPath);
1428 return rc2;
1429 }
1430 rtDbgCfgLog1(pThis, "Error %Rrc opening '%s'.\n", rc2, pszPath);
1431 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rcRet))
1432 rcRet = rc2;
1433 }
1434 }
1435 }
1436 }
1437 rc2 = VERR_FILE_NOT_FOUND;
1438 }
1439
1440 /*
1441 * Do a recursive search if requested.
1442 */
1443 if ( (fFlags & RTDBGCFG_O_RECURSIVE)
1444 && !(pThis->fFlags & RTDBGCFG_FLAGS_NO_RECURSIV_SEARCH) )
1445 {
1446 /** @todo Recursive searching will be done later. */
1447 }
1448
1449 return rcRet;
1450}
1451
1452
1453/**
1454 * Very similar to rtDbgCfgTryOpenList.
1455 */
1456static int rtDbgCfgTryOpenBundleInList(PRTDBGCFGINT pThis, PRTLISTANCHOR pList, PRTPATHSPLIT pSplitFn,
1457 const char * const *papszSuffixes, const char *pszCacheSubDir,
1458 const char *pszCacheSuffix, const char *pszUuidMappingSubDir,
1459 uint32_t fFlags, char *pszPath,
1460 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1461{
1462 int rcRet = VWRN_NOT_FOUND;
1463 int rc2;
1464
1465 const char *pchCache = NULL;
1466 size_t cchCache = 0;
1467 int rcCache = VWRN_NOT_FOUND;
1468
1469 PRTDBGCFGSTR pCur;
1470 RTListForEach(pList, pCur, RTDBGCFGSTR, ListEntry)
1471 {
1472 size_t cchDir = pCur->cch;
1473 const char *pszDir = pCur->sz;
1474 rtDbgCfgLog2(pThis, "Path list entry: '%s'\n", pszDir);
1475
1476 /* This is very simplistic, but we have a unreasonably large path
1477 buffer, so it'll work just fine and simplify things greatly below. */
1478 if (cchDir >= RTPATH_MAX - 8U)
1479 {
1480 if (RT_SUCCESS_NP(rcRet))
1481 rcRet = VERR_FILENAME_TOO_LONG;
1482 continue;
1483 }
1484
1485 /*
1486 * Process the path according to it's type.
1487 */
1488 rc2 = VINF_SUCCESS;
1489 if (!RTStrNICmp(pszDir, RT_STR_TUPLE("srv*")))
1490 {
1491 /*
1492 * Symbol server.
1493 */
1494 pszDir += sizeof("srv*") - 1;
1495 cchDir -= sizeof("srv*") - 1;
1496 bool fSearchCache = false;
1497 const char *pszServer = (const char *)memchr(pszDir, '*', cchDir);
1498 if (!pszServer)
1499 pszServer = pszDir;
1500 else if (pszServer == pszDir)
1501 continue;
1502 else
1503 {
1504 fSearchCache = true;
1505 pchCache = pszDir;
1506 cchCache = pszServer - pszDir;
1507 pszServer++;
1508 }
1509
1510 /* We don't have any default cache directory, so skip if the cache is missing. */
1511 if (cchCache == 0)
1512 continue;
1513
1514 /* Search the cache first (if we haven't already done so). */
1515 if (fSearchCache)
1516 {
1517 memcpy(pszPath, pchCache, cchCache);
1518 pszPath[cchCache] = '\0';
1519 RTPathChangeToUnixSlashes(pszPath, false);
1520
1521 rcCache = rc2 = rtDbgCfgTryOpenCache(pThis, pszPath, cchCache, pszCacheSubDir, pszUuidMappingSubDir,
1522 pSplitFn, pszCacheSuffix, fFlags, pfnCallback, pvUser1, pvUser2);
1523 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
1524 return rc2;
1525 }
1526
1527 /* Try downloading the file. */
1528 if (rcCache == VWRN_NOT_FOUND)
1529 {
1530 memcpy(pszPath, pchCache, cchCache);
1531 pszPath[cchCache] = '\0';
1532 RTPathChangeToUnixSlashes(pszPath, false);
1533
1534 rc2 = rtDbgCfgTryDownloadAndOpen(pThis, pszServer, pszPath, pszCacheSubDir, pszUuidMappingSubDir,
1535 pSplitFn, pszCacheSuffix, fFlags, pfnCallback, pvUser1, pvUser2);
1536 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
1537 return rc2;
1538 }
1539 }
1540 else if (!RTStrNICmp(pszDir, RT_STR_TUPLE("cache*")))
1541 {
1542 /*
1543 * Cache directory.
1544 */
1545 pszDir += sizeof("cache*") - 1;
1546 cchDir -= sizeof("cache*") - 1;
1547 if (!cchDir)
1548 continue;
1549 pchCache = pszDir;
1550 cchCache = cchDir;
1551
1552 memcpy(pszPath, pchCache, cchCache);
1553 pszPath[cchCache] = '\0';
1554 RTPathChangeToUnixSlashes(pszPath, false);
1555
1556 rcCache = rc2 = rtDbgCfgTryOpenCache(pThis, pszPath, cchCache, pszCacheSubDir, pszUuidMappingSubDir,
1557 pSplitFn, pszCacheSuffix, fFlags, pfnCallback, pvUser1, pvUser2);
1558 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
1559 return rc2;
1560 }
1561 else
1562 {
1563 /*
1564 * Normal directory. Check for our own 'rec*' and 'norec*' prefix
1565 * flags governing recursive searching.
1566 */
1567 uint32_t fFlagsDir = fFlags;
1568 if (!RTStrNICmp(pszDir, RT_STR_TUPLE("rec*")))
1569 {
1570 pszDir += sizeof("rec*") - 1;
1571 cchDir -= sizeof("rec*") - 1;
1572 fFlagsDir |= RTDBGCFG_O_RECURSIVE;
1573 }
1574 else if (!RTStrNICmp(pszDir, RT_STR_TUPLE("norec*")))
1575 {
1576 pszDir += sizeof("norec*") - 1;
1577 cchDir -= sizeof("norec*") - 1;
1578 fFlagsDir &= ~RTDBGCFG_O_RECURSIVE;
1579 }
1580
1581 /* Copy the path into the buffer and do the searching. */
1582 memcpy(pszPath, pszDir, cchDir);
1583 pszPath[cchDir] = '\0';
1584 RTPathChangeToUnixSlashes(pszPath, false);
1585
1586 rc2 = rtDbgCfgTryOpenDsymBundleInDir(pThis, pszPath, pSplitFn, papszSuffixes, fFlagsDir,
1587 pfnCallback, pvUser1, pvUser2);
1588 if (rc2 == VINF_CALLBACK_RETURN || rc2 == VERR_CALLBACK_RETURN)
1589 {
1590 if ( rc2 == VINF_CALLBACK_RETURN
1591 && cchCache > 0)
1592 rtDbgCfgCopyFileToCache(pThis, pszPath, pchCache, cchCache,
1593 pszCacheSubDir, pszUuidMappingSubDir, pSplitFn);
1594 return rc2;
1595 }
1596 }
1597
1598 /* Propagate errors. */
1599 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rcRet))
1600 rcRet = rc2;
1601 }
1602
1603 return rcRet;
1604}
1605
1606
1607/**
1608 * Creating a UUID mapping subdirectory path for use in caches.
1609 *
1610 * @returns IPRT status code.
1611 * @param pszSubDir The output buffer.
1612 * @param cbSubDir The size of the output buffer. (Top dir length +
1613 * slash + UUID string len + extra dash.)
1614 * @param pszTopDir The top level cache directory name. No slashes
1615 * or other directory separators, please.
1616 * @param pUuid The UUID.
1617 */
1618static int rtDbgCfgConstructUuidMappingSubDir(char *pszSubDir, size_t cbSubDir, const char *pszTopDir, PCRTUUID pUuid)
1619{
1620 Assert(!strpbrk(pszTopDir, ":/\\"));
1621
1622 size_t cchTopDir = strlen(pszTopDir);
1623 if (cchTopDir + 1 + 1 + RTUUID_STR_LENGTH + 1 > cbSubDir)
1624 return VERR_BUFFER_OVERFLOW;
1625 memcpy(pszSubDir, pszTopDir, cchTopDir);
1626
1627 pszSubDir += cchTopDir;
1628 *pszSubDir++ = RTPATH_SLASH;
1629 cbSubDir -= cchTopDir + 1;
1630
1631 /* ed5a8336-35c2-4892-9122-21d5572924a3 -> ED5A/8336/35C2/4892/9122/21D5572924A3 */
1632 int rc = RTUuidToStr(pUuid, pszSubDir + 1, cbSubDir - 1); AssertRCReturn(rc, rc);
1633 RTStrToUpper(pszSubDir + 1);
1634 memmove(pszSubDir, pszSubDir + 1, 4);
1635 pszSubDir += 4;
1636 *pszSubDir = RTPATH_SLASH;
1637 pszSubDir += 5;
1638 *pszSubDir = RTPATH_SLASH;
1639 pszSubDir += 5;
1640 *pszSubDir = RTPATH_SLASH;
1641 pszSubDir += 5;
1642 *pszSubDir = RTPATH_SLASH;
1643 pszSubDir += 5;
1644 *pszSubDir = RTPATH_SLASH;
1645
1646 return VINF_SUCCESS;
1647}
1648
1649
1650static int rtDbgCfgOpenBundleFile(RTDBGCFG hDbgCfg, const char *pszImage, const char * const *papszSuffixes,
1651 const char *pszBundleSubDir, PCRTUUID pUuid, const char *pszUuidMapDirName,
1652 const char *pszCacheSuffix, bool fOpenImage,
1653 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1654{
1655 /*
1656 * Bundles are directories, means we can forget about sharing code much
1657 * with the other RTDbgCfgOpenXXX methods. Thus we're duplicating a lot of
1658 * code from rtDbgCfgOpenWithSubDir with .dSYM/.kext/.dylib/.app/.* related
1659 * adjustments, so, a bug found here or there probably means the other
1660 * version needs updating.
1661 */
1662 int rcRet = VINF_SUCCESS;
1663 int rc2;
1664
1665 /*
1666 * Do a little validating first.
1667 */
1668 PRTDBGCFGINT pThis = hDbgCfg;
1669 if (pThis != NIL_RTDBGCFG)
1670 RTDBGCFG_VALID_RETURN_RC(pThis, VERR_INVALID_HANDLE);
1671 else
1672 pThis = NULL;
1673 AssertPtrReturn(pszImage, VERR_INVALID_POINTER);
1674 AssertPtrReturn(pfnCallback, VERR_INVALID_POINTER);
1675
1676 /*
1677 * Set up rtDbgCfgOpenWithSubDir and uuid map parameters.
1678 */
1679 uint32_t fFlags = RTDBGCFG_O_EXT_DEBUG_FILE | RT_OPSYS_DARWIN;
1680 const char *pszCacheSubDir = NULL;
1681 char szCacheSubDir[RTUUID_STR_LENGTH];
1682 const char *pszUuidMappingSubDir = NULL;
1683 char szUuidMappingSubDir[RTUUID_STR_LENGTH + 16];
1684 if (pUuid)
1685 {
1686 /* Since Mac debuggers uses UUID mappings, we just the slashing default
1687 UUID string representation instead of stripping dashes like for PDB. */
1688 RTUuidToStr(pUuid, szCacheSubDir, sizeof(szCacheSubDir));
1689 pszCacheSubDir = szCacheSubDir;
1690
1691 rc2 = rtDbgCfgConstructUuidMappingSubDir(szUuidMappingSubDir, sizeof(szUuidMappingSubDir), pszUuidMapDirName, pUuid);
1692 AssertRCReturn(rc2, rc2);
1693 pszUuidMappingSubDir = szUuidMappingSubDir;
1694 }
1695
1696 /*
1697 * Do some guessing as to the way we should parse the filename and whether
1698 * it's case exact or not.
1699 */
1700 bool fDosPath = strchr(pszImage, ':') != NULL
1701 || strchr(pszImage, '\\') != NULL
1702 || RT_OPSYS_USES_DOS_PATHS(fFlags & RTDBGCFG_O_OPSYS_MASK)
1703 || (fFlags & RTDBGCFG_O_CASE_INSENSITIVE);
1704 if (fDosPath)
1705 fFlags |= RTDBGCFG_O_CASE_INSENSITIVE;
1706
1707 rtDbgCfgLog2(pThis, "Looking for '%s' with %#x flags...\n", pszImage, fFlags);
1708
1709 PRTPATHSPLIT pSplitFn;
1710 rc2 = RTPathSplitA(pszImage, &pSplitFn, fDosPath ? RTPATH_STR_F_STYLE_DOS : RTPATH_STR_F_STYLE_UNIX);
1711 if (RT_FAILURE(rc2))
1712 return rc2;
1713 AssertReturnStmt(pSplitFn->fProps & RTPATH_PROP_FILENAME, RTPathSplitFree(pSplitFn), VERR_IS_A_DIRECTORY);
1714
1715 /*
1716 * Try the image directory first.
1717 */
1718 char szPath[RTPATH_MAX];
1719 if (pSplitFn->cComps > 0)
1720 {
1721 rc2 = RTPathSplitReassemble(pSplitFn, RTPATH_STR_F_STYLE_HOST, szPath, sizeof(szPath));
1722 if (fOpenImage && RT_SUCCESS(rc2))
1723 {
1724 rc2 = RTStrCat(szPath, sizeof(szPath), papszSuffixes[0]);
1725 if (RT_SUCCESS(rc2))
1726 rc2 = RTStrCat(szPath, sizeof(szPath), pszBundleSubDir);
1727 if (RT_SUCCESS(rc2))
1728 rc2 = RTPathAppend(szPath, sizeof(szPath), pSplitFn->apszComps[pSplitFn->cComps - 1]);
1729 }
1730 if (RT_SUCCESS(rc2) && RTPathExists(szPath))
1731 {
1732 RTPathChangeToUnixSlashes(szPath, false);
1733 rtDbgCfgLog1(pThis, "Trying '%s'...\n", szPath);
1734 rc2 = pfnCallback(pThis, szPath, pvUser1, pvUser2);
1735 if (rc2 == VINF_CALLBACK_RETURN)
1736 rtDbgCfgLog1(pThis, "Found '%s'.\n", szPath);
1737 else if (rc2 == VERR_CALLBACK_RETURN)
1738 rtDbgCfgLog1(pThis, "Error opening '%s'.\n", szPath);
1739 else
1740 rtDbgCfgLog1(pThis, "Error %Rrc opening '%s'.\n", rc2, szPath);
1741 }
1742 }
1743 if ( rc2 != VINF_CALLBACK_RETURN
1744 && rc2 != VERR_CALLBACK_RETURN)
1745 {
1746 /*
1747 * Try the current directory (will take cover relative paths
1748 * skipped above).
1749 */
1750 rc2 = RTPathGetCurrent(szPath, sizeof(szPath));
1751 if (RT_FAILURE(rc2))
1752 strcpy(szPath, ".");
1753 RTPathChangeToUnixSlashes(szPath, false);
1754
1755 rc2 = rtDbgCfgTryOpenDsymBundleInDir(pThis, szPath, pSplitFn, g_apszDSymBundleSuffixes,
1756 fFlags, pfnCallback, pvUser1, pvUser2);
1757 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rcRet))
1758 rcRet = rc2;
1759
1760 if ( rc2 != VINF_CALLBACK_RETURN
1761 && rc2 != VERR_CALLBACK_RETURN
1762 && pThis)
1763 {
1764 rc2 = RTCritSectRwEnterShared(&pThis->CritSect);
1765 if (RT_SUCCESS(rc2))
1766 {
1767 /*
1768 * Run the applicable lists.
1769 */
1770 rc2 = rtDbgCfgTryOpenBundleInList(pThis, &pThis->PathList, pSplitFn, g_apszDSymBundleSuffixes,
1771 pszCacheSubDir, pszCacheSuffix,
1772 pszUuidMappingSubDir, fFlags, szPath,
1773 pfnCallback, pvUser1, pvUser2);
1774 if (RT_FAILURE(rc2) && RT_SUCCESS_NP(rcRet))
1775 rcRet = rc2;
1776
1777 RTCritSectRwLeaveShared(&pThis->CritSect);
1778 }
1779 else if (RT_SUCCESS(rcRet))
1780 rcRet = rc2;
1781 }
1782 }
1783
1784 RTPathSplitFree(pSplitFn);
1785 if ( rc2 == VINF_CALLBACK_RETURN
1786 || rc2 == VERR_CALLBACK_RETURN)
1787 rcRet = rc2;
1788 else if (RT_SUCCESS(rcRet))
1789 rcRet = VERR_NOT_FOUND;
1790 return rcRet;
1791}
1792
1793
1794RTDECL(int) RTDbgCfgOpenDsymBundle(RTDBGCFG hDbgCfg, const char *pszImage, PCRTUUID pUuid,
1795 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1796{
1797 return rtDbgCfgOpenBundleFile(hDbgCfg, pszImage, g_apszDSymBundleSuffixes,
1798 "Contents" RTPATH_SLASH_STR "Resources" RTPATH_SLASH_STR "DWARF",
1799 pUuid, RTDBG_CACHE_UUID_MAP_DIR_DSYMS, RTDBG_CACHE_DSYM_FILE_SUFFIX, false /* fOpenImage */,
1800 pfnCallback, pvUser1, pvUser2);
1801}
1802
1803
1804RTDECL(int) RTDbgCfgOpenMachOImage(RTDBGCFG hDbgCfg, const char *pszImage, PCRTUUID pUuid,
1805 PFNRTDBGCFGOPEN pfnCallback, void *pvUser1, void *pvUser2)
1806{
1807 return rtDbgCfgOpenBundleFile(hDbgCfg, pszImage, g_apszBundleSuffixes,
1808 "Contents" RTPATH_SLASH_STR "MacOS",
1809 pUuid, RTDBG_CACHE_UUID_MAP_DIR_IMAGES, NULL /*pszCacheSuffix*/, true /* fOpenImage */,
1810 pfnCallback, pvUser1, pvUser2);
1811}
1812
1813
1814
1815RTDECL(int) RTDbgCfgSetLogCallback(RTDBGCFG hDbgCfg, PFNRTDBGCFGLOG pfnCallback, void *pvUser)
1816{
1817 PRTDBGCFGINT pThis = hDbgCfg;
1818 RTDBGCFG_VALID_RETURN_RC(pThis, VERR_INVALID_HANDLE);
1819 AssertPtrNullReturn(pfnCallback, VERR_INVALID_POINTER);
1820
1821 int rc = RTCritSectRwEnterExcl(&pThis->CritSect);
1822 if (RT_SUCCESS(rc))
1823 {
1824 if ( pThis->pfnLogCallback == NULL
1825 || pfnCallback == NULL
1826 || pfnCallback == pThis->pfnLogCallback)
1827 {
1828 pThis->pfnLogCallback = NULL;
1829 pThis->pvLogUser = NULL;
1830 ASMCompilerBarrier(); /* paranoia */
1831 pThis->pvLogUser = pvUser;
1832 pThis->pfnLogCallback = pfnCallback;
1833 rc = VINF_SUCCESS;
1834 }
1835 else
1836 rc = VERR_ACCESS_DENIED;
1837 RTCritSectRwLeaveExcl(&pThis->CritSect);
1838 }
1839
1840 return rc;
1841}
1842
1843
1844/**
1845 * Frees a string list.
1846 *
1847 * @param pList The list to free.
1848 */
1849static void rtDbgCfgFreeStrList(PRTLISTANCHOR pList)
1850{
1851 PRTDBGCFGSTR pCur;
1852 PRTDBGCFGSTR pNext;
1853 RTListForEachSafe(pList, pCur, pNext, RTDBGCFGSTR, ListEntry)
1854 {
1855 RTListNodeRemove(&pCur->ListEntry);
1856 RTMemFree(pCur);
1857 }
1858}
1859
1860
1861/**
1862 * Make changes to a string list, given a semicolon separated input string.
1863 *
1864 * @returns VINF_SUCCESS, VERR_FILENAME_TOO_LONG, VERR_NO_MEMORY
1865 * @param pThis The config instance.
1866 * @param enmOp The change operation.
1867 * @param pszValue The input strings separated by semicolon.
1868 * @param fPaths Indicates that this is a path list and that we
1869 * should look for srv and cache prefixes.
1870 * @param pList The string list anchor.
1871 */
1872static int rtDbgCfgChangeStringList(PRTDBGCFGINT pThis, RTDBGCFGOP enmOp, const char *pszValue, bool fPaths,
1873 PRTLISTANCHOR pList)
1874{
1875 RT_NOREF_PV(pThis); RT_NOREF_PV(fPaths);
1876
1877 if (enmOp == RTDBGCFGOP_SET)
1878 rtDbgCfgFreeStrList(pList);
1879
1880 while (*pszValue)
1881 {
1882 /* Skip separators. */
1883 while (*pszValue == ';')
1884 pszValue++;
1885 if (!*pszValue)
1886 break;
1887
1888 /* Find the end of this path. */
1889 const char *pchPath = pszValue++;
1890 char ch;
1891 while ((ch = *pszValue) && ch != ';')
1892 pszValue++;
1893 size_t cchPath = pszValue - pchPath;
1894 if (cchPath >= UINT16_MAX)
1895 return VERR_FILENAME_TOO_LONG;
1896
1897 if (enmOp == RTDBGCFGOP_REMOVE)
1898 {
1899 /*
1900 * Remove all occurences.
1901 */
1902 PRTDBGCFGSTR pCur;
1903 PRTDBGCFGSTR pNext;
1904 RTListForEachSafe(pList, pCur, pNext, RTDBGCFGSTR, ListEntry)
1905 {
1906 if ( pCur->cch == cchPath
1907 && !memcmp(pCur->sz, pchPath, cchPath))
1908 {
1909 RTListNodeRemove(&pCur->ListEntry);
1910 RTMemFree(pCur);
1911 }
1912 }
1913 }
1914 else
1915 {
1916 /*
1917 * We're adding a new one.
1918 */
1919 PRTDBGCFGSTR pNew = (PRTDBGCFGSTR)RTMemAlloc(RT_OFFSETOF(RTDBGCFGSTR, sz[cchPath + 1]));
1920 if (!pNew)
1921 return VERR_NO_MEMORY;
1922 pNew->cch = (uint16_t)cchPath;
1923 pNew->fFlags = 0;
1924 memcpy(pNew->sz, pchPath, cchPath);
1925 pNew->sz[cchPath] = '\0';
1926
1927 if (enmOp == RTDBGCFGOP_PREPEND)
1928 RTListPrepend(pList, &pNew->ListEntry);
1929 else
1930 RTListAppend(pList, &pNew->ListEntry);
1931 }
1932 }
1933
1934 return VINF_SUCCESS;
1935}
1936
1937
1938/**
1939 * Make changes to a 64-bit value
1940 *
1941 * @returns VINF_SUCCESS, VERR_DBG_CFG_INVALID_VALUE.
1942 * @param pThis The config instance.
1943 * @param enmOp The change operation.
1944 * @param pszValue The input value.
1945 * @param paMnemonics The mnemonics map for this value.
1946 * @param puValue The value to change.
1947 */
1948static int rtDbgCfgChangeStringU64(PRTDBGCFGINT pThis, RTDBGCFGOP enmOp, const char *pszValue,
1949 PCRTDBGCFGU64MNEMONIC paMnemonics, uint64_t *puValue)
1950{
1951 RT_NOREF_PV(pThis);
1952
1953 uint64_t uNew = enmOp == RTDBGCFGOP_SET ? 0 : *puValue;
1954 char ch;
1955 while ((ch = *pszValue))
1956 {
1957 /* skip whitespace and separators */
1958 while (RT_C_IS_SPACE(ch) || RT_C_IS_CNTRL(ch) || ch == ';' || ch == ':')
1959 ch = *++pszValue;
1960 if (!ch)
1961 break;
1962
1963 if (RT_C_IS_DIGIT(ch))
1964 {
1965 uint64_t uTmp;
1966 int rc = RTStrToUInt64Ex(pszValue, (char **)&pszValue, 0, &uTmp);
1967 if (RT_FAILURE(rc) || rc == VWRN_NUMBER_TOO_BIG)
1968 return VERR_DBG_CFG_INVALID_VALUE;
1969
1970 if (enmOp != RTDBGCFGOP_REMOVE)
1971 uNew |= uTmp;
1972 else
1973 uNew &= ~uTmp;
1974 }
1975 else
1976 {
1977 /* A mnemonic, find the end of it. */
1978 const char *pszMnemonic = pszValue - 1;
1979 do
1980 ch = *++pszValue;
1981 while (ch && !RT_C_IS_SPACE(ch) && !RT_C_IS_CNTRL(ch) && ch != ';' && ch != ':');
1982 size_t cchMnemonic = pszValue - pszMnemonic;
1983
1984 /* Look it up in the map and apply it. */
1985 unsigned i = 0;
1986 while (paMnemonics[i].pszMnemonic)
1987 {
1988 if ( cchMnemonic == paMnemonics[i].cchMnemonic
1989 && !memcmp(pszMnemonic, paMnemonics[i].pszMnemonic, cchMnemonic))
1990 {
1991 if (paMnemonics[i].fSet ? enmOp != RTDBGCFGOP_REMOVE : enmOp == RTDBGCFGOP_REMOVE)
1992 uNew |= paMnemonics[i].fFlags;
1993 else
1994 uNew &= ~paMnemonics[i].fFlags;
1995 break;
1996 }
1997 i++;
1998 }
1999
2000 if (!paMnemonics[i].pszMnemonic)
2001 return VERR_DBG_CFG_INVALID_VALUE;
2002 }
2003 }
2004
2005 *puValue = uNew;
2006 return VINF_SUCCESS;
2007}
2008
2009
2010RTDECL(int) RTDbgCfgChangeString(RTDBGCFG hDbgCfg, RTDBGCFGPROP enmProp, RTDBGCFGOP enmOp, const char *pszValue)
2011{
2012 PRTDBGCFGINT pThis = hDbgCfg;
2013 RTDBGCFG_VALID_RETURN_RC(pThis, VERR_INVALID_HANDLE);
2014 AssertReturn(enmProp > RTDBGCFGPROP_INVALID && enmProp < RTDBGCFGPROP_END, VERR_INVALID_PARAMETER);
2015 AssertReturn(enmOp > RTDBGCFGOP_INVALID && enmOp < RTDBGCFGOP_END, VERR_INVALID_PARAMETER);
2016 if (!pszValue)
2017 pszValue = "";
2018 else
2019 AssertPtrReturn(pszValue, VERR_INVALID_POINTER);
2020
2021 int rc = RTCritSectRwEnterExcl(&pThis->CritSect);
2022 if (RT_SUCCESS(rc))
2023 {
2024 switch (enmProp)
2025 {
2026 case RTDBGCFGPROP_FLAGS:
2027 rc = rtDbgCfgChangeStringU64(pThis, enmOp, pszValue, g_aDbgCfgFlags, &pThis->fFlags);
2028 break;
2029 case RTDBGCFGPROP_PATH:
2030 rc = rtDbgCfgChangeStringList(pThis, enmOp, pszValue, true, &pThis->PathList);
2031 break;
2032 case RTDBGCFGPROP_SUFFIXES:
2033 rc = rtDbgCfgChangeStringList(pThis, enmOp, pszValue, false, &pThis->SuffixList);
2034 break;
2035 case RTDBGCFGPROP_SRC_PATH:
2036 rc = rtDbgCfgChangeStringList(pThis, enmOp, pszValue, true, &pThis->SrcPathList);
2037 break;
2038 default:
2039 AssertFailed();
2040 rc = VERR_INTERNAL_ERROR_3;
2041 }
2042
2043 RTCritSectRwLeaveExcl(&pThis->CritSect);
2044 }
2045
2046 return rc;
2047}
2048
2049
2050RTDECL(int) RTDbgCfgChangeUInt(RTDBGCFG hDbgCfg, RTDBGCFGPROP enmProp, RTDBGCFGOP enmOp, uint64_t uValue)
2051{
2052 PRTDBGCFGINT pThis = hDbgCfg;
2053 RTDBGCFG_VALID_RETURN_RC(pThis, VERR_INVALID_HANDLE);
2054 AssertReturn(enmProp > RTDBGCFGPROP_INVALID && enmProp < RTDBGCFGPROP_END, VERR_INVALID_PARAMETER);
2055 AssertReturn(enmOp > RTDBGCFGOP_INVALID && enmOp < RTDBGCFGOP_END, VERR_INVALID_PARAMETER);
2056
2057 int rc = RTCritSectRwEnterExcl(&pThis->CritSect);
2058 if (RT_SUCCESS(rc))
2059 {
2060 uint64_t *puValue = NULL;
2061 switch (enmProp)
2062 {
2063 case RTDBGCFGPROP_FLAGS:
2064 puValue = &pThis->fFlags;
2065 break;
2066 default:
2067 rc = VERR_DBG_CFG_NOT_UINT_PROP;
2068 }
2069 if (RT_SUCCESS(rc))
2070 {
2071 switch (enmOp)
2072 {
2073 case RTDBGCFGOP_SET:
2074 *puValue = uValue;
2075 break;
2076 case RTDBGCFGOP_APPEND:
2077 case RTDBGCFGOP_PREPEND:
2078 *puValue |= uValue;
2079 break;
2080 case RTDBGCFGOP_REMOVE:
2081 *puValue &= ~uValue;
2082 break;
2083 default:
2084 AssertFailed();
2085 rc = VERR_INTERNAL_ERROR_2;
2086 }
2087 }
2088
2089 RTCritSectRwLeaveExcl(&pThis->CritSect);
2090 }
2091
2092 return rc;
2093}
2094
2095
2096/**
2097 * Querys a string list as a single string (semicolon separators).
2098 *
2099 * @returns VINF_SUCCESS, VERR_BUFFER_OVERFLOW.
2100 * @param hDbgCfg The config instance handle.
2101 * @param pList The string list anchor.
2102 * @param pszValue The output buffer.
2103 * @param cbValue The size of the output buffer.
2104 */
2105static int rtDbgCfgQueryStringList(RTDBGCFG hDbgCfg, PRTLISTANCHOR pList,
2106 char *pszValue, size_t cbValue)
2107{
2108 RT_NOREF_PV(hDbgCfg);
2109
2110 /*
2111 * Check the length first.
2112 */
2113 size_t cbReq = 1;
2114 PRTDBGCFGSTR pCur;
2115 RTListForEach(pList, pCur, RTDBGCFGSTR, ListEntry)
2116 cbReq += pCur->cch + 1;
2117 if (cbReq > cbValue)
2118 return VERR_BUFFER_OVERFLOW;
2119
2120 /*
2121 * Construct the string list in the buffer.
2122 */
2123 char *psz = pszValue;
2124 RTListForEach(pList, pCur, RTDBGCFGSTR, ListEntry)
2125 {
2126 if (psz != pszValue)
2127 *psz++ = ';';
2128 memcpy(psz, pCur->sz, pCur->cch);
2129 psz += pCur->cch;
2130 }
2131 *psz = '\0';
2132
2133 return VINF_SUCCESS;
2134}
2135
2136
2137/**
2138 * Querys the string value of a 64-bit unsigned int.
2139 *
2140 * @returns VINF_SUCCESS, VERR_BUFFER_OVERFLOW.
2141 * @param hDbgCfg The config instance handle.
2142 * @param uValue The value to query.
2143 * @param paMnemonics The mnemonics map for this value.
2144 * @param pszValue The output buffer.
2145 * @param cbValue The size of the output buffer.
2146 */
2147static int rtDbgCfgQueryStringU64(RTDBGCFG hDbgCfg, uint64_t uValue, PCRTDBGCFGU64MNEMONIC paMnemonics,
2148 char *pszValue, size_t cbValue)
2149{
2150 RT_NOREF_PV(hDbgCfg);
2151
2152 /*
2153 * If no mnemonics, just return the hex value.
2154 */
2155 if (!paMnemonics || paMnemonics[0].pszMnemonic)
2156 {
2157 char szTmp[64];
2158 size_t cch = RTStrPrintf(szTmp, sizeof(szTmp), "%#x", uValue);
2159 if (cch + 1 > cbValue)
2160 return VERR_BUFFER_OVERFLOW;
2161 memcpy(pszValue, szTmp, cbValue);
2162 return VINF_SUCCESS;
2163 }
2164
2165 /*
2166 * Check that there is sufficient buffer space first.
2167 */
2168 size_t cbReq = 1;
2169 for (unsigned i = 0; paMnemonics[i].pszMnemonic; i++)
2170 if ( paMnemonics[i].fSet
2171 ? (paMnemonics[i].fFlags & uValue)
2172 : !(paMnemonics[i].fFlags & uValue))
2173 cbReq += (cbReq != 1) + paMnemonics[i].cchMnemonic;
2174 if (cbReq > cbValue)
2175 return VERR_BUFFER_OVERFLOW;
2176
2177 /*
2178 * Construct the string.
2179 */
2180 char *psz = pszValue;
2181 for (unsigned i = 0; paMnemonics[i].pszMnemonic; i++)
2182 if ( paMnemonics[i].fSet
2183 ? (paMnemonics[i].fFlags & uValue)
2184 : !(paMnemonics[i].fFlags & uValue))
2185 {
2186 if (psz != pszValue)
2187 *psz++ = ' ';
2188 memcpy(psz, paMnemonics[i].pszMnemonic, paMnemonics[i].cchMnemonic);
2189 psz += paMnemonics[i].cchMnemonic;
2190 }
2191 *psz = '\0';
2192 return VINF_SUCCESS;
2193}
2194
2195
2196RTDECL(int) RTDbgCfgQueryString(RTDBGCFG hDbgCfg, RTDBGCFGPROP enmProp, char *pszValue, size_t cbValue)
2197{
2198 PRTDBGCFGINT pThis = hDbgCfg;
2199 RTDBGCFG_VALID_RETURN_RC(pThis, VERR_INVALID_HANDLE);
2200 AssertReturn(enmProp > RTDBGCFGPROP_INVALID && enmProp < RTDBGCFGPROP_END, VERR_INVALID_PARAMETER);
2201 AssertPtrReturn(pszValue, VERR_INVALID_POINTER);
2202
2203 int rc = RTCritSectRwEnterShared(&pThis->CritSect);
2204 if (RT_SUCCESS(rc))
2205 {
2206 switch (enmProp)
2207 {
2208 case RTDBGCFGPROP_FLAGS:
2209 rc = rtDbgCfgQueryStringU64(pThis, pThis->fFlags, g_aDbgCfgFlags, pszValue, cbValue);
2210 break;
2211 case RTDBGCFGPROP_PATH:
2212 rc = rtDbgCfgQueryStringList(pThis, &pThis->PathList, pszValue, cbValue);
2213 break;
2214 case RTDBGCFGPROP_SUFFIXES:
2215 rc = rtDbgCfgQueryStringList(pThis, &pThis->SuffixList, pszValue, cbValue);
2216 break;
2217 case RTDBGCFGPROP_SRC_PATH:
2218 rc = rtDbgCfgQueryStringList(pThis, &pThis->SrcPathList, pszValue, cbValue);
2219 break;
2220 default:
2221 AssertFailed();
2222 rc = VERR_INTERNAL_ERROR_3;
2223 }
2224
2225 RTCritSectRwLeaveShared(&pThis->CritSect);
2226 }
2227
2228 return rc;
2229}
2230
2231
2232RTDECL(int) RTDbgCfgQueryUInt(RTDBGCFG hDbgCfg, RTDBGCFGPROP enmProp, uint64_t *puValue)
2233{
2234 PRTDBGCFGINT pThis = hDbgCfg;
2235 RTDBGCFG_VALID_RETURN_RC(pThis, VERR_INVALID_HANDLE);
2236 AssertReturn(enmProp > RTDBGCFGPROP_INVALID && enmProp < RTDBGCFGPROP_END, VERR_INVALID_PARAMETER);
2237 AssertPtrReturn(puValue, VERR_INVALID_POINTER);
2238
2239 int rc = RTCritSectRwEnterShared(&pThis->CritSect);
2240 if (RT_SUCCESS(rc))
2241 {
2242 switch (enmProp)
2243 {
2244 case RTDBGCFGPROP_FLAGS:
2245 *puValue = pThis->fFlags;
2246 break;
2247 default:
2248 rc = VERR_DBG_CFG_NOT_UINT_PROP;
2249 }
2250
2251 RTCritSectRwLeaveShared(&pThis->CritSect);
2252 }
2253
2254 return rc;
2255}
2256
2257RTDECL(uint32_t) RTDbgCfgRetain(RTDBGCFG hDbgCfg)
2258{
2259 PRTDBGCFGINT pThis = hDbgCfg;
2260 RTDBGCFG_VALID_RETURN_RC(pThis, UINT32_MAX);
2261
2262 uint32_t cRefs = ASMAtomicIncU32(&pThis->cRefs);
2263 Assert(cRefs < UINT32_MAX / 2);
2264 return cRefs;
2265}
2266
2267
2268RTDECL(uint32_t) RTDbgCfgRelease(RTDBGCFG hDbgCfg)
2269{
2270 if (hDbgCfg == NIL_RTDBGCFG)
2271 return 0;
2272
2273 PRTDBGCFGINT pThis = hDbgCfg;
2274 RTDBGCFG_VALID_RETURN_RC(pThis, UINT32_MAX);
2275
2276 uint32_t cRefs = ASMAtomicDecU32(&pThis->cRefs);
2277 if (!cRefs)
2278 {
2279 /*
2280 * Last reference - free all memory.
2281 */
2282 ASMAtomicWriteU32(&pThis->u32Magic, ~RTDBGCFG_MAGIC);
2283 rtDbgCfgFreeStrList(&pThis->PathList);
2284 rtDbgCfgFreeStrList(&pThis->SuffixList);
2285 rtDbgCfgFreeStrList(&pThis->SrcPathList);
2286#ifdef RT_OS_WINDOWS
2287 rtDbgCfgFreeStrList(&pThis->NtSymbolPathList);
2288 rtDbgCfgFreeStrList(&pThis->NtExecutablePathList);
2289 rtDbgCfgFreeStrList(&pThis->NtSourcePath);
2290#endif
2291 RTCritSectRwDelete(&pThis->CritSect);
2292 RTMemFree(pThis);
2293 }
2294 else
2295 Assert(cRefs < UINT32_MAX / 2);
2296 return cRefs;
2297}
2298
2299
2300RTDECL(int) RTDbgCfgCreate(PRTDBGCFG phDbgCfg, const char *pszEnvVarPrefix, bool fNativePaths)
2301{
2302 /*
2303 * Validate input.
2304 */
2305 AssertPtrReturn(phDbgCfg, VERR_INVALID_POINTER);
2306 if (pszEnvVarPrefix)
2307 {
2308 AssertPtrReturn(pszEnvVarPrefix, VERR_INVALID_POINTER);
2309 AssertReturn(*pszEnvVarPrefix, VERR_INVALID_PARAMETER);
2310 }
2311
2312 /*
2313 * Allocate and initialize a new instance.
2314 */
2315 PRTDBGCFGINT pThis = (PRTDBGCFGINT)RTMemAllocZ(sizeof(*pThis));
2316 if (!pThis)
2317 return VERR_NO_MEMORY;
2318
2319 pThis->u32Magic = RTDBGCFG_MAGIC;
2320 pThis->cRefs = 1;
2321 RTListInit(&pThis->PathList);
2322 RTListInit(&pThis->SuffixList);
2323 RTListInit(&pThis->SrcPathList);
2324#ifdef RT_OS_WINDOWS
2325 RTListInit(&pThis->NtSymbolPathList);
2326 RTListInit(&pThis->NtExecutablePathList);
2327 RTListInit(&pThis->NtSourcePath);
2328#endif
2329
2330 int rc = RTCritSectRwInit(&pThis->CritSect);
2331 if (RT_FAILURE(rc))
2332 {
2333 RTMemFree(pThis);
2334 return rc;
2335 }
2336
2337 /*
2338 * Read configurtion from the environment if requested to do so.
2339 */
2340 if (pszEnvVarPrefix || fNativePaths)
2341 {
2342 const size_t cbEnvVar = 256;
2343 const size_t cbEnvVal = 65536 - cbEnvVar;
2344 char *pszEnvVar = (char *)RTMemTmpAlloc(cbEnvVar + cbEnvVal);
2345 if (pszEnvVar)
2346 {
2347 char *pszEnvVal = pszEnvVar + cbEnvVar;
2348
2349 if (pszEnvVarPrefix)
2350 {
2351 static struct
2352 {
2353 RTDBGCFGPROP enmProp;
2354 const char *pszVar;
2355 } const s_aProps[] =
2356 {
2357 { RTDBGCFGPROP_FLAGS, "FLAGS" },
2358 { RTDBGCFGPROP_PATH, "PATH" },
2359 { RTDBGCFGPROP_SUFFIXES, "SUFFIXES" },
2360 { RTDBGCFGPROP_SRC_PATH, "SRC_PATH" },
2361 };
2362
2363 for (unsigned i = 0; i < RT_ELEMENTS(s_aProps); i++)
2364 {
2365 size_t cchEnvVar = RTStrPrintf(pszEnvVar, cbEnvVar, "%s_%s", pszEnvVarPrefix, s_aProps[i].pszVar);
2366 if (cchEnvVar >= cbEnvVar - 1)
2367 {
2368 rc = VERR_BUFFER_OVERFLOW;
2369 break;
2370 }
2371
2372 rc = RTEnvGetEx(RTENV_DEFAULT, pszEnvVar, pszEnvVal, cbEnvVal, NULL);
2373 if (RT_SUCCESS(rc))
2374 {
2375 rc = RTDbgCfgChangeString(pThis, s_aProps[i].enmProp, RTDBGCFGOP_SET, pszEnvVal);
2376 if (RT_FAILURE(rc))
2377 break;
2378 }
2379 else if (rc != VERR_ENV_VAR_NOT_FOUND)
2380 break;
2381 else
2382 rc = VINF_SUCCESS;
2383 }
2384 }
2385
2386 /*
2387 * Pick up system specific search paths.
2388 */
2389 if (RT_SUCCESS(rc) && fNativePaths)
2390 {
2391 struct
2392 {
2393 PRTLISTANCHOR pList;
2394 const char *pszVar;
2395 char chSep;
2396 } aNativePaths[] =
2397 {
2398#ifdef RT_OS_WINDOWS
2399 { &pThis->NtExecutablePathList, "_NT_EXECUTABLE_PATH", ';' },
2400 { &pThis->NtSymbolPathList, "_NT_ALT_SYMBOL_PATH", ';' },
2401 { &pThis->NtSymbolPathList, "_NT_SYMBOL_PATH", ';' },
2402 { &pThis->NtSourcePath, "_NT_SOURCE_PATH", ';' },
2403#endif
2404 { NULL, NULL, 0 }
2405 };
2406 for (unsigned i = 0; aNativePaths[i].pList; i++)
2407 {
2408 Assert(aNativePaths[i].chSep == ';'); /* fix when needed */
2409 rc = RTEnvGetEx(RTENV_DEFAULT, aNativePaths[i].pszVar, pszEnvVal, cbEnvVal, NULL);
2410 if (RT_SUCCESS(rc))
2411 {
2412 rc = rtDbgCfgChangeStringList(pThis, RTDBGCFGOP_APPEND, pszEnvVal, true, aNativePaths[i].pList);
2413 if (RT_FAILURE(rc))
2414 break;
2415 }
2416 else if (rc != VERR_ENV_VAR_NOT_FOUND)
2417 break;
2418 else
2419 rc = VINF_SUCCESS;
2420 }
2421 }
2422 RTMemTmpFree(pszEnvVar);
2423 }
2424 else
2425 rc = VERR_NO_TMP_MEMORY;
2426 if (RT_FAILURE(rc))
2427 {
2428 /*
2429 * Error, bail out.
2430 */
2431 RTDbgCfgRelease(pThis);
2432 return rc;
2433 }
2434 }
2435
2436 /*
2437 * Returns successfully.
2438 */
2439 *phDbgCfg = pThis;
2440
2441 return VINF_SUCCESS;
2442}
2443
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