VirtualBox

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

Last change on this file since 50795 was 49082, checked in by vboxsync, 11 years ago

IPRT: darwin debugging fixes.

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