VirtualBox

source: vbox/trunk/src/VBox/Runtime/tools/RTDbgSymCache.cpp@ 49469

Last change on this file since 49469 was 49118, checked in by vboxsync, 11 years ago

kext plugins.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 44.4 KB
Line 
1/* $Id: RTDbgSymCache.cpp 49118 2013-10-15 13:46:24Z vboxsync $ */
2/** @file
3 * IPRT - Debug Symbol Cache Utility.
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#include <iprt/zip.h>
32
33#include <iprt/buildconfig.h>
34#include <iprt/dbg.h>
35#include <iprt/file.h>
36#include <iprt/formats/mach-o.h>
37#include <iprt/getopt.h>
38#include <iprt/initterm.h>
39#include <iprt/ldr.h>
40#include <iprt/message.h>
41#include <iprt/param.h>
42#include <iprt/path.h>
43#include <iprt/stream.h>
44#include <iprt/string.h>
45#include <iprt/uuid.h>
46#include <iprt/vfs.h>
47#include <iprt/zip.h>
48
49
50/*******************************************************************************
51* Structures and Typedefs *
52*******************************************************************************/
53/**
54 * Cache file type.
55 */
56typedef enum RTDBGSYMCACHEFILETYPE
57{
58 RTDBGSYMCACHEFILETYPE_INVALID,
59 RTDBGSYMCACHEFILETYPE_DIR,
60 RTDBGSYMCACHEFILETYPE_DIR_FILTER,
61 RTDBGSYMCACHEFILETYPE_DEBUG_FILE,
62 RTDBGSYMCACHEFILETYPE_IMAGE_FILE,
63 RTDBGSYMCACHEFILETYPE_DEBUG_BUNDLE,
64 RTDBGSYMCACHEFILETYPE_IMAGE_BUNDLE,
65 RTDBGSYMCACHEFILETYPE_IGNORE
66} RTDBGSYMCACHEFILETYPE;
67
68
69/**
70 * Configuration for the 'add' command.
71 */
72typedef struct RTDBGSYMCACHEADDCFG
73{
74 bool fRecursive;
75 const char *pszFilter;
76 const char *pszCache;
77} RTDBGSYMCACHEADDCFG;
78/** Pointer to a read only 'add' config. */
79typedef RTDBGSYMCACHEADDCFG const *PCRTDBGSYMCACHEADDCFG;
80
81
82/*******************************************************************************
83* Global Variables *
84*******************************************************************************/
85/** Bundle suffixes. */
86static const char * const g_apszBundleSuffixes[] =
87{
88 ".kext",
89 ".app",
90 ".framework", /** @todo framework is different. */
91 ".component",
92 ".action",
93 ".caction",
94 ".bundle",
95 ".sourcebundle",
96 ".plugin",
97 ".ppp",
98 ".menu",
99 ".monitorpanel",
100 ".scripting",
101 ".prefPane",
102 ".qlgenerator",
103 ".brailledriver",
104 ".saver",
105 ".SpeechVoice",
106 ".SpeechRecognizer",
107 ".SpeechSynthesizer",
108 ".mdimporter",
109 ".spreporter",
110 ".xpc",
111 NULL
112};
113
114/** Debug bundle suffixes. (Same as above + .dSYM) */
115static const char * const g_apszDSymBundleSuffixes[] =
116{
117 ".kext.dSYM",
118 ".app.dSYM",
119 ".framework.dSYM",
120 ".component.dSYM",
121 ".action.dSYM",
122 ".caction.dSYM",
123 ".bundle.dSYM",
124 ".sourcebundle.dSYM",
125 ".menu.dSYM",
126 ".plugin.dSYM",
127 ".ppp.dSYM",
128 ".monitorpanel.dSYM",
129 ".scripting.dSYM",
130 ".prefPane.dSYM",
131 ".qlgenerator.dSYM",
132 ".brailledriver.dSYM",
133 ".saver.dSYM",
134 ".SpeechVoice.dSYM",
135 ".SpeechRecognizer.dSYM",
136 ".SpeechSynthesizer.dSYM",
137 ".mdimporter.dSYM",
138 ".spreporter.dSYM",
139 ".xpc.dSYM",
140 ".dSYM",
141 NULL
142};
143
144
145/*******************************************************************************
146* Internal Functions *
147*******************************************************************************/
148static int rtDbgSymCacheAddDirWorker(char *pszPath, size_t cchPath, PRTDIRENTRYEX pDirEntry, PCRTDBGSYMCACHEADDCFG pCfg);
149
150
151
152/**
153 * Display the version of the cache program.
154 *
155 * @returns exit code.
156 */
157static RTEXITCODE rtDbgSymCacheVersion(void)
158{
159 RTPrintf("%sr%d\n", RTBldCfgVersion(), RTBldCfgRevision());
160 return RTEXITCODE_SUCCESS;
161}
162
163
164/**
165 * Shows the usage of the cache program.
166 *
167 * @returns Exit code.
168 * @param pszArg0 Program name.
169 * @param pszCommand Command selector, NULL if all.
170 */
171static RTEXITCODE rtDbgSymCacheUsage(const char *pszArg0, const char *pszCommand)
172{
173 if (!pszCommand || !strcmp(pszCommand, "add"))
174 RTPrintf("Usage: %s add [-rn] <cache-root-dir> <file1> [fileN..]\n", pszArg0);
175 return RTEXITCODE_SUCCESS;
176}
177
178
179/**
180 * Creates a UUID mapping for the file.
181 *
182 * @returns IPRT status code.
183 * @param pszCacheFile The path to the file in the cache.
184 * @param pFileUuid The UUID of the file.
185 * @param pszUuidMapDir The UUID map subdirectory in the cache, if this is
186 * wanted, otherwise NULL.
187 * @param pCfg The configuration.
188 */
189static int rtDbgSymCacheAddCreateUuidMapping(const char *pszCacheFile, PRTUUID pFileUuid,
190 const char *pszUuidMapDir, PCRTDBGSYMCACHEADDCFG pCfg)
191{
192 /*
193 * Create the UUID map entry first, deep.
194 */
195 char szMapPath[RTPATH_MAX];
196 int rc = RTPathJoin(szMapPath, sizeof(szMapPath) - sizeof("/xxxx/yyyy/xxxx/yyyy/xxxx/zzzzzzzzzzzz") + 1,
197 pCfg->pszCache, pszUuidMapDir);
198 if (RT_FAILURE(rc))
199 return RTMsgErrorRc(rc, "Error constructing UUID map path (RTPathJoin): %Rrc", rc);
200
201 size_t cch = strlen(szMapPath);
202 szMapPath[cch] = '-';
203
204 rc = RTUuidToStr(pFileUuid, &szMapPath[cch + 2], sizeof(szMapPath) - cch);
205 if (RT_FAILURE(rc))
206 return RTMsgErrorRc(rc, "Error constructing UUID map path (RTUuidToStr): %Rrc", rc);
207
208 /* Uppercase the whole lot. */
209 RTStrToUpper(&szMapPath[cch + 2]);
210
211 /* Split the first dword in two. */
212 szMapPath[cch + 1] = szMapPath[cch + 2];
213 szMapPath[cch + 2] = szMapPath[cch + 3];
214 szMapPath[cch + 3] = szMapPath[cch + 4];
215 szMapPath[cch + 4] = szMapPath[cch + 5];
216 szMapPath[cch + 5] = '-';
217
218 /*
219 * Create the directories in the path.
220 */
221 char chSaved = RTPATH_SLASH;
222 for (unsigned i = 0; i < 6; i++, cch += 5)
223 {
224 Assert(szMapPath[cch] == '-');
225 szMapPath[cch] = '\0';
226 if (!RTDirExists(szMapPath))
227 {
228 rc = RTDirCreate(szMapPath, 0755, RTDIRCREATE_FLAGS_NOT_CONTENT_INDEXED_NOT_CRITICAL);
229 if (RT_FAILURE(rc))
230 return RTMsgErrorRc(rc, "RTDirCreate failed on '%s' (UUID map path): %Rrc", szMapPath, rc);
231 }
232 szMapPath[cch] = RTPATH_SLASH;
233 }
234 cch -= 5;
235
236 /*
237 * Calculate a relative path from there to the actual file.
238 */
239 char szLinkTarget[RTPATH_MAX];
240 //szMapPath[cch] = '\0';
241 rc = RTPathCalcRelative(szLinkTarget, sizeof(szLinkTarget), szMapPath, pszCacheFile);
242 //szMapPath[cch] = RTPATH_SLASH;
243 if (RT_FAILURE(rc))
244 return RTMsgErrorRc(rc, "Failed to calculate relative path from '%s' to '%s': %Rrc", szMapPath, pszCacheFile, rc);
245
246 /*
247 * If there is already a link there, check if it matches or whether
248 * perhaps it's target doesn't exist.
249 */
250 RTFSOBJINFO ObjInfo;
251 rc = RTPathQueryInfoEx(szMapPath, &ObjInfo, RTFSOBJATTRADD_NOTHING, RTPATH_F_ON_LINK);
252 if (RT_SUCCESS(rc))
253 {
254 if (RTFS_IS_SYMLINK(ObjInfo.Attr.fMode))
255 {
256 rc = RTPathQueryInfoEx(szMapPath, &ObjInfo, RTFSOBJATTRADD_NOTHING, RTPATH_F_FOLLOW_LINK);
257 if (RT_SUCCESS(rc))
258 {
259 char *pszCurTarget = NULL;
260 rc = RTSymlinkReadA(szMapPath, &pszCurTarget);
261 if (RT_FAILURE(rc))
262 return RTMsgErrorRc(rc, "UUID map: failed to read existing symlink '%s': %Rrc", szMapPath, rc);
263 if (RTPathCompare(pszCurTarget, szLinkTarget) == 0)
264 RTMsgInfo("UUID map: existing link '%s' has the same target ('%s').", szMapPath, pszCurTarget);
265 else
266 {
267 RTMsgError("UUID map: Existing mapping '%s' pointing to '%s' insted of '%s'",
268 szMapPath, pszCurTarget, szLinkTarget);
269 rc = VERR_ALREADY_EXISTS;
270 }
271 RTStrFree(pszCurTarget);
272 return rc;
273 }
274 else
275 RTMsgInfo("UUID map: replacing dangling link '%s'", szMapPath);
276 RTSymlinkDelete(szMapPath, 0 /*fFlags*/);
277 }
278 else if (RTFS_IS_FILE(ObjInfo.Attr.fMode))
279 return RTMsgErrorRc(VERR_IS_A_FILE,
280 "UUID map: found file at '%s', expect symbolic link or nothing.", szMapPath);
281 else if (RTFS_IS_DIRECTORY(ObjInfo.Attr.fMode))
282 return RTMsgErrorRc(VERR_IS_A_DIRECTORY,
283 "UUID map: found directory at '%s', expect symbolic link or nothing.", szMapPath);
284 else
285 return RTMsgErrorRc(VERR_NOT_SYMLINK,
286 "UUID map: Expected symbolic link or nothing at '%s', found: fMode=%#x",
287 szMapPath, ObjInfo.Attr.fMode);
288 }
289
290 /*
291 * Create the symbolic link.
292 */
293 rc = RTSymlinkCreate(szMapPath, szLinkTarget, RTSYMLINKTYPE_FILE, 0);
294 if (RT_FAILURE(rc))
295 return RTMsgErrorRc(rc, "Failed to create UUID map symlink '%s' to '%s': %Rrc", szMapPath, szLinkTarget, rc);
296 RTMsgInfo("UUID map: %s => %s", szMapPath, szLinkTarget);
297 return VINF_SUCCESS;
298}
299
300
301/**
302 * Adds a file to the cache.
303 *
304 * @returns IPRT status code.
305 * @param pszSrcPath Path to the source file.
306 * @param pszDstName The name of the destionation file (no path stuff).
307 * @param pszExtraSuff Optional extra suffix. Mach-O dSYM hack.
308 * @param pszDstSubDir The subdirectory to file it under. This is the
309 * stringification of a relatively unique identifier of
310 * the file in question.
311 * @param pAddToUuidMap Optional file UUID that is used to create a UUID map
312 * entry.
313 * @param pszUuidMapDir The UUID map subdirectory in the cache, if this is
314 * wanted, otherwise NULL.
315 * @param pCfg The configuration.
316 */
317static int rtDbgSymCacheAddOneFile(const char *pszSrcPath, const char *pszDstName, const char *pszExtraStuff,
318 const char *pszDstSubDir, PRTUUID pAddToUuidMap, const char *pszUuidMapDir,
319 PCRTDBGSYMCACHEADDCFG pCfg)
320{
321 /*
322 * Build and create the destination path, step by step.
323 */
324 char szDstPath[RTPATH_MAX];
325 int rc = RTPathJoin(szDstPath, sizeof(szDstPath), pCfg->pszCache, pszDstName);
326 if (RT_FAILURE(rc))
327 return RTMsgErrorRc(rc, "Error constructing cache path for '%s': %Rrc", pszSrcPath, rc);
328
329 if (!RTDirExists(szDstPath))
330 {
331 rc = RTDirCreate(szDstPath, 0755, RTDIRCREATE_FLAGS_NOT_CONTENT_INDEXED_NOT_CRITICAL);
332 if (RT_FAILURE(rc))
333 return RTMsgErrorRc(rc, "Error creating '%s': %Rrc", szDstPath, rc);
334 }
335
336 rc = RTPathAppend(szDstPath, sizeof(szDstPath), pszDstSubDir);
337 if (RT_FAILURE(rc))
338 return RTMsgErrorRc(rc, "Error constructing cache path for '%s': %Rrc", pszSrcPath, rc);
339
340 if (!RTDirExists(szDstPath))
341 {
342 rc = RTDirCreate(szDstPath, 0755, RTDIRCREATE_FLAGS_NOT_CONTENT_INDEXED_NOT_CRITICAL);
343 if (RT_FAILURE(rc))
344 return RTMsgErrorRc(rc, "Error creating '%s': %Rrc", szDstPath, rc);
345 }
346
347 rc = RTPathAppend(szDstPath, sizeof(szDstPath), pszDstName);
348 if (RT_FAILURE(rc))
349 return RTMsgErrorRc(rc, "Error constructing cache path for '%s': %Rrc", pszSrcPath, rc);
350 if (pszExtraStuff)
351 {
352 rc = RTStrCat(szDstPath, sizeof(szDstPath), pszExtraStuff);
353 if (RT_FAILURE(rc))
354 return RTMsgErrorRc(rc, "Error constructing cache path for '%s': %Rrc", pszSrcPath, rc);
355 }
356
357 /*
358 * If the file exists, we compare the two and throws an error if the doesn't match.
359 */
360 if (RTPathExists(szDstPath))
361 {
362 rc = RTFileCompare(pszSrcPath, szDstPath);
363 if (RT_SUCCESS(rc))
364 {
365 RTMsgInfo("%s is already in the cache.", pszSrcPath);
366 if (pAddToUuidMap && pszUuidMapDir)
367 return rtDbgSymCacheAddCreateUuidMapping(szDstPath, pAddToUuidMap, pszUuidMapDir, pCfg);
368 return VINF_SUCCESS;
369 }
370 if (rc == VERR_NOT_EQUAL)
371 RTMsgInfo("Cache conflict with existing entry '%s' when inserting '%s'.", szDstPath, pszSrcPath);
372 else
373 RTMsgInfo("Error comparing '%s' with '%s': %Rrc", pszSrcPath, szDstPath, rc);
374 return rc;
375 }
376
377 /*
378 * The file doesn't exist or we should overwrite it,
379 */
380 RTMsgInfo("Copying '%s' to '%s'...", pszSrcPath, szDstPath);
381 rc = RTFileCopy(pszSrcPath, szDstPath);
382 if (RT_FAILURE(rc))
383 return RTMsgErrorRc(rc, "Error copying '%s' to '%s': %Rrc", pszSrcPath, szDstPath, rc);
384 if (pAddToUuidMap && pszUuidMapDir)
385 return rtDbgSymCacheAddCreateUuidMapping(szDstPath, pAddToUuidMap, pszUuidMapDir, pCfg);
386 return VINF_SUCCESS;
387}
388
389
390/**
391 * Worker that add the image file to the right place.
392 *
393 * @returns IPRT status code.
394 * @param pszPath Path to the image file.
395 * @param pCfg Configuration data.
396 * @param hLdrMod Image handle.
397 * @param pszExtraSuff Optional extra suffix. Mach-O dSYM hack.
398 * @param pszUuidMapDir Optional UUID map cache directory if the image
399 * should be mapped by UUID.
400 * The map is a Mac OS X debug feature supported by
401 * the two native debuggers gdb and lldb. Look for
402 * descriptions of DBGFileMappedPaths in the
403 * com.apple.DebugSymbols in the user defaults.
404 */
405static int rtDbgSymCacheAddImageFileWorker(const char *pszPath, PCRTDBGSYMCACHEADDCFG pCfg, RTLDRMOD hLdrMod,
406 const char *pszExtrSuff, const char *pszUuidMapDir)
407{
408 /*
409 * Determine which subdirectory to put the files in.
410 */
411 RTUUID Uuid;
412 PRTUUID pUuid = NULL;
413 int rc;
414 char szSubDir[48];
415 RTLDRFMT enmFmt = RTLdrGetFormat(hLdrMod);
416 switch (enmFmt)
417 {
418 case RTLDRFMT_MACHO:
419 {
420 rc = RTLdrQueryProp(hLdrMod, RTLDRPROP_UUID, &Uuid, sizeof(Uuid));
421 if (RT_FAILURE(rc))
422 return RTMsgErrorRc(rc, "Error quering image UUID from image '%s': %Rrc", pszPath, rc);
423
424 rc = RTUuidToStr(&Uuid, szSubDir, sizeof(szSubDir));
425 if (RT_FAILURE(rc))
426 return RTMsgErrorRc(rc, "Error convering UUID for image '%s' to string: %Rrc", pszPath, rc);
427 pUuid = &Uuid;
428 break;
429 }
430
431 case RTLDRFMT_PE:
432 {
433 uint32_t uTimestamp;
434 rc = RTLdrQueryProp(hLdrMod, RTLDRPROP_TIMESTAMP_SECONDS, &uTimestamp, sizeof(uTimestamp));
435 if (RT_FAILURE(rc))
436 return RTMsgErrorRc(rc, "Error quering timestamp from image '%s': %Rrc", pszPath, rc);
437
438 size_t cbImage = RTLdrSize(hLdrMod);
439 if (cbImage == ~(size_t)0)
440 return RTMsgErrorRc(rc, "Error quering size of image '%s': %Rrc", pszPath, rc);
441
442 RTStrPrintf(szSubDir, sizeof(szSubDir), "%08X%x", uTimestamp, cbImage);
443 break;
444 }
445
446 case RTLDRFMT_AOUT:
447 return RTMsgErrorRc(VERR_NOT_SUPPORTED, "Caching of a.out image has not yet been implemented: %s", pszPath);
448 case RTLDRFMT_ELF:
449 return RTMsgErrorRc(VERR_NOT_SUPPORTED, "Caching of ELF image has not yet been implemented: %s", pszPath);
450 case RTLDRFMT_LX:
451 return RTMsgErrorRc(VERR_NOT_SUPPORTED, "Caching of LX image has not yet been implemented: %s", pszPath);
452 default:
453 return RTMsgErrorRc(VERR_NOT_SUPPORTED, "Unknown loader format for '%s': %d", pszPath, enmFmt);
454 }
455
456 /*
457 * Now add it.
458 */
459 return rtDbgSymCacheAddOneFile(pszPath, RTPathFilename(pszPath), pszExtrSuff,
460 szSubDir, pUuid, pszUuidMapDir, pCfg);
461}
462
463
464/**
465 * Adds what we think is an image file to the cache.
466 *
467 * @returns IPRT status code.
468 * @param pszPath Path to the image file.
469 * @param pszExtraSuff Optional extra suffix. Mach-O dSYM hack.
470 * @param pszUuidMapDir The UUID map subdirectory in the cache, if this is
471 * wanted, otherwise NULL.
472 * @param pCfg Configuration data.
473 */
474static int rtDbgSymCacheAddImageFile(const char *pszPath, const char *pszExtraSuff, const char *pszUuidMapDir,
475 PCRTDBGSYMCACHEADDCFG pCfg)
476{
477 /*
478 * Use the loader to open the alleged image file. We need to open it with
479 * arch set to amd64 and x86_32 in order to handle FAT images from the mac
480 * guys (we should actually enumerate archs, but that's currently not
481 * implemented nor necessary for our current use).
482 */
483 /* Open it as AMD64. */
484 RTLDRMOD hLdrMod64;
485 int rc = RTLdrOpen(pszPath, RTLDR_O_FOR_DEBUG, RTLDRARCH_AMD64, &hLdrMod64);
486 if (RT_FAILURE(rc))
487 {
488 if (rc != VERR_LDR_ARCH_MISMATCH)
489 {
490 if (rc != VERR_INVALID_EXE_SIGNATURE)
491 return RTMsgErrorRc(rc, "RTLdrOpen failed opening '%s' [arch=amd64]: %Rrc", pszPath, rc);
492 RTMsgInfo("Skipping '%s', no a recognizable image file...", pszPath);
493 return VINF_SUCCESS;
494 }
495 hLdrMod64 = NIL_RTLDRMOD;
496 }
497
498 /* Open it as X86. */
499 RTLDRMOD hLdrMod32;
500 rc = RTLdrOpen(pszPath, RTLDR_O_FOR_DEBUG, RTLDRARCH_X86_32, &hLdrMod32);
501 if (RT_FAILURE(rc))
502 {
503 if (rc != VERR_LDR_ARCH_MISMATCH)
504 {
505 RTLdrClose(hLdrMod32);
506 return RTMsgErrorRc(rc, "RTLdrOpen failed opening '%s' [arch=x86]: %Rrc", pszPath, rc);
507 }
508 hLdrMod32 = NIL_RTLDRMOD;
509 }
510
511 /*
512 * Add the file.
513 */
514 if (hLdrMod32 == NIL_RTLDRMOD)
515 rc = rtDbgSymCacheAddImageFileWorker(pszPath, pCfg, hLdrMod64, pszExtraSuff, pszUuidMapDir);
516 else if (hLdrMod64 == NIL_RTLDRMOD)
517 rc = rtDbgSymCacheAddImageFileWorker(pszPath, pCfg, hLdrMod32, pszExtraSuff, pszUuidMapDir);
518 else
519 {
520 /*
521 * Do we need to add it once or twice?
522 */
523 RTLDRFMT enmFmt = RTLdrGetFormat(hLdrMod32);
524 bool fSame = enmFmt == RTLdrGetFormat(hLdrMod64);
525 if (fSame && enmFmt == RTLDRFMT_MACHO)
526 {
527 RTUUID Uuid32, Uuid64;
528 int rc32 = RTLdrQueryProp(hLdrMod32, RTLDRPROP_UUID, &Uuid32, sizeof(Uuid32));
529 int rc64 = RTLdrQueryProp(hLdrMod64, RTLDRPROP_UUID, &Uuid64, sizeof(Uuid64));
530 fSame = RT_SUCCESS(rc32) == RT_SUCCESS(rc64);
531 if (fSame && RT_SUCCESS(rc32))
532 fSame = RTUuidCompare(&Uuid32, &Uuid64) == 0;
533 }
534 else if (fSame && enmFmt == RTLDRFMT_PE)
535 {
536 fSame = RTLdrSize(hLdrMod32) == RTLdrSize(hLdrMod64);
537 if (fSame)
538 {
539 uint32_t uTimestamp32, uTimestamp64;
540 int rc32 = RTLdrQueryProp(hLdrMod32, RTLDRPROP_TIMESTAMP_SECONDS, &uTimestamp32, sizeof(uTimestamp32));
541 int rc64 = RTLdrQueryProp(hLdrMod64, RTLDRPROP_TIMESTAMP_SECONDS, &uTimestamp64, sizeof(uTimestamp64));
542 fSame = RT_SUCCESS(rc32) == RT_SUCCESS(rc64);
543 if (fSame && RT_SUCCESS(rc32))
544 fSame = uTimestamp32 == uTimestamp64;
545 }
546 }
547
548 rc = rtDbgSymCacheAddImageFileWorker(pszPath, pCfg, hLdrMod64, pszExtraSuff, pszUuidMapDir);
549 if (!fSame)
550 {
551 /** @todo should symlink or hardlink this second copy. */
552 int rc2 = rtDbgSymCacheAddImageFileWorker(pszPath, pCfg, hLdrMod32, pszExtraSuff, pszUuidMapDir);
553 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
554 rc = rc2;
555 }
556 }
557
558 RTLdrClose(hLdrMod32);
559 RTLdrClose(hLdrMod64);
560 return VINF_SUCCESS;
561}
562
563
564/**
565 * Worker for rtDbgSymCacheAddDebugFile that adds a Mach-O debug file to the
566 * cache.
567 *
568 * @returns IPRT status code
569 * @param pszPath The path to the PDB file.
570 * @param pCfg The configuration.
571 * @param hFile Handle to the file.
572 */
573static int rtDbgSymCacheAddDebugMachO(const char *pszPath, PCRTDBGSYMCACHEADDCFG pCfg)
574{
575 /* This shouldn't happen, figure out what to do if it does. */
576 return RTMsgErrorRc(VERR_NOT_IMPLEMENTED,
577 "'%s' is an OS X image file, did you point me to a file inside a .dSYM or .sym file?",
578 pszPath);
579}
580
581
582/**
583 * Worker for rtDbgSymCacheAddDebugFile that adds PDBs to the cace.
584 *
585 * @returns IPRT status code
586 * @param pszPath The path to the PDB file.
587 * @param pCfg The configuration.
588 * @param hFile Handle to the file.
589 */
590static int rtDbgSymCacheAddDebugPdb(const char *pszPath, PCRTDBGSYMCACHEADDCFG pCfg, RTFILE hFile)
591{
592 return RTMsgErrorRc(VERR_NOT_IMPLEMENTED, "PDB support not implemented: '%s'");
593}
594
595
596/**
597 * Adds a debug file to the cache.
598 *
599 * @returns IPRT status code
600 * @param pszPath The path to the debug file in question.
601 * @param pCfg The configuration.
602 */
603static int rtDbgSymCacheAddDebugFile(const char *pszPath, PCRTDBGSYMCACHEADDCFG pCfg)
604{
605 /*
606 * Need to extract an identifier of sorts here in order to put them in
607 * the right place in the cache. Currently only implemnted for Mach-O
608 * files since these use executable containers.
609 *
610 * We take a look at the file header in hope to figure out what to do
611 * with the file.
612 */
613 RTFILE hFile;
614 int rc = RTFileOpen(&hFile, pszPath, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_WRITE);
615 if (RT_FAILURE(rc))
616 return RTMsgErrorRc(rc, "Error opening '%s': %Rrc", pszPath, rc);
617
618 union
619 {
620 uint64_t au64[16];
621 uint32_t au32[16];
622 uint16_t au16[32];
623 uint8_t ab[64];
624 } uBuf;
625 rc = RTFileRead(hFile, &uBuf, sizeof(uBuf), NULL);
626 if (RT_SUCCESS(rc))
627 {
628 /*
629 * Look for magics and call workers.
630 */
631 if (!memcmp(uBuf.ab, RT_STR_TUPLE("Microsoft C/C++ MSF 7.00")))
632 rc = rtDbgSymCacheAddDebugPdb(pszPath, pCfg, hFile);
633 else if ( uBuf.au32[0] == IMAGE_FAT_SIGNATURE
634 || uBuf.au32[0] == IMAGE_FAT_SIGNATURE_OE
635 || uBuf.au32[0] == IMAGE_MACHO32_SIGNATURE
636 || uBuf.au32[0] == IMAGE_MACHO64_SIGNATURE
637 || uBuf.au32[0] == IMAGE_MACHO32_SIGNATURE_OE
638 || uBuf.au32[0] == IMAGE_MACHO64_SIGNATURE_OE)
639 rc = rtDbgSymCacheAddDebugMachO(pszPath, pCfg);
640 else
641 rc = RTMsgErrorRc(VERR_INVALID_MAGIC, "Unsupported debug file '%s' magic: %#010x", pszPath, uBuf.au32[0]);
642 }
643 else
644 rc = RTMsgErrorRc(rc, "Error reading '%s': %Rrc", pszPath, rc);
645
646 /* close the file. */
647 int rc2 = RTFileClose(hFile);
648 if (RT_FAILURE(rc2))
649 {
650 RTMsgError("Error closing '%s': %Rrc", pszPath, rc2);
651 if (RT_SUCCESS(rc))
652 rc = rc2;
653 }
654 return rc;
655}
656
657
658/**
659 * Constructs the path to the file instide the bundle that we're keen on.
660 *
661 * @returns IPRT status code.
662 * @param pszPath Path to the bundle on input, on successful
663 * return it's the path to the desired file. This
664 * a RTPATH_MAX size buffer.
665 * @param cchPath The length of the path up to the bundle name.
666 * @param cchName The length of the bundle name.
667 * @param pszSubDir The bundle subdirectory the file lives in.
668 * @param papszSuffixes Pointer to an array of bundle suffixes.
669 */
670static int rtDbgSymCacheConstructBundlePath(char *pszPath, size_t cchPath, size_t cchName, const char *pszSubDir,
671 const char * const *papszSuffixes)
672{
673 /*
674 * Calc the name without the bundle extension.
675 */
676 size_t const cchOrgName = cchName;
677 const char *pszEnd = &pszPath[cchPath + cchName];
678 for (unsigned i = 0; papszSuffixes[i]; i++)
679 {
680 Assert(papszSuffixes[i][0] == '.');
681 size_t cchSuff = strlen(papszSuffixes[i]);
682 if ( cchSuff < cchName
683 && !memcmp(&pszEnd[-(ssize_t)cchSuff], papszSuffixes[i], cchSuff))
684 {
685 cchName -= cchSuff;
686 break;
687 }
688 }
689
690 /*
691 * Check the immediate directory first, in case it's layed out like
692 * IOPCIFamily.kext.
693 */
694 int rc = RTPathAppendEx(pszPath, RTPATH_MAX, &pszPath[cchPath], cchName);
695 if (RT_FAILURE(rc) || !RTFileExists(pszPath))
696 {
697 /*
698 * Not there, ok then try the given subdirectory + name.
699 */
700 pszPath[cchPath + cchOrgName] = '\0';
701 rc = RTPathAppend(pszPath, RTPATH_MAX, pszSubDir);
702 if (RT_SUCCESS(rc))
703 rc = RTPathAppendEx(pszPath, RTPATH_MAX, &pszPath[cchPath], cchName);
704 if (RT_FAILURE(rc))
705 {
706 pszPath[cchPath + cchOrgName] = '\0';
707 return RTMsgErrorRc(rc, "Error constructing image bundle path for '%s': %Rrc", pszPath, rc);
708 }
709 }
710
711 return VINF_SUCCESS;
712}
713
714
715/**
716 * Adds a image bundle of some sort.
717 *
718 * @returns IPRT status code.
719 * @param pszPath Path to the bundle. This a RTPATH_MAX size
720 * buffer that we can write to when creating the
721 * path to the file inside the bundle that we're
722 * interested in.
723 * @param cchPath The length of the path up to the bundle name.
724 * @param cchName The length of the bundle name.
725 * @param pDirEntry The directory entry buffer, for handling bundle
726 * within bundle recursion.
727 * @param pCfg The configuration.
728 */
729static int rtDbgSymCacheAddImageBundle(char *pszPath, size_t cchPath, size_t cchName,
730 PRTDIRENTRYEX pDirEntry, PCRTDBGSYMCACHEADDCFG pCfg)
731{
732 /*
733 * Assuming these are kexts or simple applications, we only add the image
734 * file itself to the cache. No Info.plist or other files.
735 */
736 /** @todo consider looking for Frameworks and handling framework bundles. */
737 int rc = rtDbgSymCacheConstructBundlePath(pszPath, cchPath, cchName, "Contents/MacOS/", g_apszBundleSuffixes);
738 if (RT_SUCCESS(rc))
739 rc = rtDbgSymCacheAddImageFile(pszPath, NULL, RTDBG_CACHE_UUID_MAP_DIR_IMAGES, pCfg);
740
741 /*
742 * Look for plugins and other sub-bundles.
743 */
744 if (pCfg->fRecursive)
745 {
746 static char const * const s_apszSubBundleDirs[] =
747 {
748 "Contents/Plugins",
749 /** @todo Frameworks ++ */
750 };
751 for (uint32_t i = 0; i < RT_ELEMENTS(s_apszSubBundleDirs); i++)
752 {
753 pszPath[cchPath + cchName] = '\0';
754 int rc2 = RTPathAppend(pszPath, RTPATH_MAX, s_apszSubBundleDirs[i]);
755 if (RT_SUCCESS(rc2))
756 rc2 = rtDbgSymCacheAddDirWorker(pszPath, strlen(pszPath), pDirEntry, pCfg);
757 else
758 {
759 pszPath[cchPath + cchName] = '\0';
760 RTMsgError("Error constructing bundle subdir path for '%s' + '%s': %Rrc", pszPath, s_apszSubBundleDirs[i], rc);
761 }
762 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
763 rc = rc2;
764 }
765 }
766
767 return rc;
768}
769
770
771/**
772 * Adds a debug bundle.
773 *
774 * @returns IPRT status code.
775 * @param pszPath Path to the bundle. This a RTPATH_MAX size
776 * buffer that we can write to when creating the
777 * path to the file inside the bundle that we're
778 * interested in.
779 * @param cchPath The length of the path up to the bundle name.
780 * @param cchName The length of the bundle name.
781 * @param pCfg The configuration.
782 */
783static int rtDbgSymCacheAddDebugBundle(char *pszPath, size_t cchPath, size_t cchName, PCRTDBGSYMCACHEADDCFG pCfg)
784{
785 /*
786 * The current policy is not to add the whole .dSYM (or .sym) bundle, but
787 * rather just the dwarf image instide it. The <UUID>.plist and Info.plist
788 * files generally doesn't contain much extra information that's really
789 * necessary, I hope. At least this is what the uuidmap example in the
790 * lldb hints at (it links to the dwarf file, not the .dSYM dir).
791 *
792 * To avoid confusion with a .dSYM bundle, as well as collision with the
793 * image file, we use .dwarf suffix for the file.
794 *
795 * For details on the uuid map see rtDbgSymCacheAddImageFile as well as
796 * http://lldb.llvm.org/symbols.html .
797 *
798 * ASSUMES bundles contains Mach-O DWARF files.
799 */
800 int rc = rtDbgSymCacheConstructBundlePath(pszPath, cchPath, cchName, "Contents/Resources/DWARF/", g_apszDSymBundleSuffixes);
801 if (RT_SUCCESS(rc))
802 rc = rtDbgSymCacheAddImageFile(pszPath, RTDBG_CACHE_DSYM_FILE_SUFFIX, RTDBG_CACHE_UUID_MAP_DIR_DSYMS, pCfg);
803 return rc;
804}
805
806
807/**
808 * Figure the type of a file/dir based on path and FS object info.
809 *
810 * @returns The type.
811 * @param pszPath The path to the file/dir.
812 * @param pObjInfo The object information, symlinks followed.
813 */
814static RTDBGSYMCACHEFILETYPE rtDbgSymCacheFigureType2(const char *pszPath, PCRTFSOBJINFO pObjInfo)
815{
816 const char *pszName = RTPathFilename(pszPath);
817 const char *pszExt = RTPathSuffix(pszName);
818 if (pszExt)
819 pszExt++;
820 else
821 pszExt = "";
822
823 if ( RTFS_IS_DIRECTORY(pObjInfo->Attr.fMode)
824 || (pObjInfo->Attr.fMode & RTFS_DOS_DIRECTORY)) /** @todo OS X samba reports reparse points in /Volumes/ that we cannot resolve. */
825 {
826 /* Skip directories shouldn't bother with. */
827 if ( !RTStrICmp(pszName, ".Trashes")
828 || !RTStrICmp(pszName, ".$RESCYCLE.BIN")
829 || !RTStrICmp(pszName, "System.kext") /* Usually only plugins here, so skip it. */
830 )
831 return RTDBGSYMCACHEFILETYPE_IGNORE;
832
833 /* Directories can also be bundles on the mac. */
834 if (!RTStrICmp(pszExt, "dSYM"))
835 return RTDBGSYMCACHEFILETYPE_DEBUG_BUNDLE;
836
837 for (unsigned i = 0; i < RT_ELEMENTS(g_apszBundleSuffixes) - 1; i++)
838 if (!RTStrICmp(pszExt, &g_apszBundleSuffixes[i][1]))
839 return RTDBGSYMCACHEFILETYPE_IMAGE_BUNDLE;
840
841 return RTDBGSYMCACHEFILETYPE_DIR;
842 }
843
844 if (!RTFS_IS_FILE(pObjInfo->Attr.fMode))
845 return RTDBGSYMCACHEFILETYPE_INVALID;
846
847 /* Select image vs debug info based on extension. */
848 if ( !RTStrICmp(pszExt, "pdb")
849 || !RTStrICmp(pszExt, "dbg")
850 || !RTStrICmp(pszExt, "sym")
851 || !RTStrICmp(pszExt, "dwo")
852 || !RTStrICmp(pszExt, "dwp")
853 || !RTStrICmp(pszExt, "debug")
854 || !RTStrICmp(pszExt, "dsym")
855 || !RTStrICmp(pszExt, "dwarf")
856 || !RTStrICmp(pszExt, "map")
857 || !RTStrICmp(pszExt, "cv"))
858 return RTDBGSYMCACHEFILETYPE_DEBUG_FILE;
859
860 /* Filter out a bunch of files which obviously shouldn't be images. */
861 if ( !RTStrICmp(pszExt, "txt")
862 || !RTStrICmp(pszExt, "html")
863 || !RTStrICmp(pszExt, "htm")
864 || !RTStrICmp(pszExt, "rtf")
865 || !RTStrICmp(pszExt, "zip")
866 || !RTStrICmp(pszExt, "doc")
867 || !RTStrICmp(pszExt, "gz")
868 || !RTStrICmp(pszExt, "bz2")
869 || !RTStrICmp(pszExt, "xz")
870 || !RTStrICmp(pszExt, "kmk")
871 || !RTStrICmp(pszExt, "c")
872 || !RTStrICmp(pszExt, "cpp")
873 || !RTStrICmp(pszExt, "h")
874 || !RTStrICmp(pszExt, "m")
875 || !RTStrICmp(pszExt, "mm")
876 || !RTStrICmp(pszExt, "asm")
877 || !RTStrICmp(pszExt, "S")
878 || !RTStrICmp(pszExt, "inc")
879 || !RTStrICmp(pszExt, "sh")
880 )
881 return RTDBGSYMCACHEFILETYPE_IGNORE;
882 if ( !RTStrICmp(pszName, "Makefile")
883 || !RTStrICmp(pszName, "GNUmakefile")
884 || !RTStrICmp(pszName, "createsymbolfiles")
885 || !RTStrICmp(pszName, "kgmacros")
886 )
887 return RTDBGSYMCACHEFILETYPE_IGNORE;
888
889 return RTDBGSYMCACHEFILETYPE_IMAGE_FILE;
890}
891
892
893/**
894 * Figure file type based on name, will stat the file/dir.
895 *
896 * @returns File type.
897 * @param pszPath The path to the file/dir to figure.
898 */
899static RTDBGSYMCACHEFILETYPE rtDbgSymCacheFigureType(const char *pszPath)
900{
901 const char *pszName = RTPathFilename(pszPath);
902
903 /* Trailing slash. */
904 if (!pszName)
905 return RTDBGSYMCACHEFILETYPE_DIR;
906
907 /* Wildcard means listing directory and filtering. */
908 if (strpbrk(pszName, "?*"))
909 return RTDBGSYMCACHEFILETYPE_DIR_FILTER;
910
911 /* Get object info, following links. */
912 RTFSOBJINFO ObjInfo;
913 int rc = RTPathQueryInfoEx(pszPath, &ObjInfo, RTFSOBJATTRADD_NOTHING, RTPATH_F_FOLLOW_LINK);
914 if (RT_FAILURE(rc))
915 return RTDBGSYMCACHEFILETYPE_INVALID;
916 return rtDbgSymCacheFigureType2(pszPath, &ObjInfo);
917}
918
919
920/**
921 * Recursive worker for rtDbgSymCacheAddDir, for minimal stack wasting.
922 *
923 * @returns IPRT status code (fully bitched).
924 * @param pszPath Pointer to a RTPATH_MAX size buffer containing
925 * the path to the current directory ending with a
926 * slash.
927 * @param cchPath The size of the current directory path.
928 * @param pDirEntry Pointer to the RTDIRENTRYEX structure to use.
929 * @param pCfg The configuration.
930 */
931static int rtDbgSymCacheAddDirWorker(char *pszPath, size_t cchPath, PRTDIRENTRYEX pDirEntry, PCRTDBGSYMCACHEADDCFG pCfg)
932{
933 /*
934 * Open the directory.
935 */
936 PRTDIR pDir;
937 int rc, rc2;
938 if (pCfg->pszFilter)
939 {
940 rc = RTStrCopy(&pszPath[cchPath], RTPATH_MAX - cchPath, pCfg->pszFilter);
941 if (RT_FAILURE(rc))
942 {
943 pszPath[cchPath] = '\0';
944 return RTMsgErrorRc(rc, "Filename too long (%Rrc): '%s" RTPATH_SLASH_STR "%s'", rc, pszPath, pCfg->pszFilter);
945 }
946 rc = RTDirOpenFiltered(&pDir, pszPath, RTDIRFILTER_WINNT, 0 /*fFlags*/);
947 }
948 else
949 rc = RTDirOpen(&pDir, pszPath);
950 if (RT_FAILURE(rc))
951 return RTMsgErrorRc(rc, "RTDirOpen%s failed on '%s': %Rrc", pCfg->pszFilter ? "Filtered" : "", pszPath, rc);
952
953 /*
954 * Enumerate the files.
955 */
956 for (;;)
957 {
958 rc2 = RTDirReadEx(pDir, pDirEntry, NULL, RTFSOBJATTRADD_NOTHING, RTPATH_F_FOLLOW_LINK);
959 if (RT_FAILURE(rc2))
960 {
961 pszPath[cchPath] = '\0';
962 if (rc2 != VERR_NO_MORE_FILES)
963 {
964 RTMsgError("RTDirReadEx failed in '%s': %Rrc\n", pszPath, rc2);
965 rc = rc2;
966 }
967 break;
968 }
969
970 /* Skip dot and dot-dot. */
971 if (RTDirEntryExIsStdDotLink(pDirEntry))
972 continue;
973
974 /* Construct a full path. */
975 rc = RTStrCopy(&pszPath[cchPath], RTPATH_MAX, pDirEntry->szName);
976 if (RT_FAILURE(rc))
977 {
978 pszPath[cchPath] = '\0';
979 RTMsgError("File name too long in '%s': '%s' (%Rrc)", pszPath, pDirEntry->szName, rc);
980 break;
981 }
982
983 switch (rtDbgSymCacheFigureType2(pszPath, &pDirEntry->Info))
984 {
985 case RTDBGSYMCACHEFILETYPE_DIR:
986 if (!pCfg->fRecursive)
987 RTMsgInfo("Skipping directory '%s'...", pszPath);
988 else
989 {
990 if (cchPath + pDirEntry->cbName + 3 <= RTPATH_MAX)
991 {
992 pszPath[cchPath + pDirEntry->cbName] = RTPATH_SLASH;
993 pszPath[cchPath + pDirEntry->cbName + 1] = '\0';
994 rc2 = rtDbgSymCacheAddDirWorker(pszPath, cchPath + pDirEntry->cbName + 1, pDirEntry, pCfg);
995 }
996 else
997 {
998 RTMsgError("File name too long in '%s': '%s' (%Rrc)", pszPath, pDirEntry->szName, rc);
999 rc2 = VERR_FILENAME_TOO_LONG;
1000 }
1001 }
1002 break;
1003
1004 case RTDBGSYMCACHEFILETYPE_DEBUG_FILE:
1005 rc2 = rtDbgSymCacheAddDebugFile(pszPath, pCfg);
1006 break;
1007
1008 case RTDBGSYMCACHEFILETYPE_IMAGE_FILE:
1009 rc2 = rtDbgSymCacheAddImageFile(pszPath, NULL /*pszExtraSuff*/, RTDBG_CACHE_UUID_MAP_DIR_IMAGES, pCfg);
1010 break;
1011
1012 case RTDBGSYMCACHEFILETYPE_DEBUG_BUNDLE:
1013 rc2 = rtDbgSymCacheAddDebugBundle(pszPath, cchPath, pDirEntry->cbName, pCfg);
1014 break;
1015
1016 case RTDBGSYMCACHEFILETYPE_IMAGE_BUNDLE:
1017 rc2 = rtDbgSymCacheAddImageBundle(pszPath, cchPath, pDirEntry->cbName, pDirEntry, pCfg);
1018 break;
1019
1020 case RTDBGSYMCACHEFILETYPE_DIR_FILTER:
1021 case RTDBGSYMCACHEFILETYPE_INVALID:
1022 rc2 = RTMsgErrorRc(VERR_INTERNAL_ERROR_2, "Invalid: '%s'", pszPath);
1023 break;
1024
1025 case RTDBGSYMCACHEFILETYPE_IGNORE:
1026 rc2 = VINF_SUCCESS;
1027 break;
1028 }
1029
1030 if (RT_FAILURE(rc2) && RT_SUCCESS(rc))
1031 rc = rc2;
1032 }
1033
1034 /*
1035 * Clean up.
1036 */
1037 rc2 = RTDirClose(pDir);
1038 if (RT_FAILURE(rc2))
1039 {
1040 RTMsgError("RTDirClose failed in '%s': %Rrc", pszPath, rc);
1041 rc = rc2;
1042 }
1043 return rc;
1044}
1045
1046
1047/**
1048 * Adds a directory.
1049 *
1050 * @returns IPRT status code (fully bitched).
1051 * @param pszPath The directory path.
1052 * @param pCfg The configuration.
1053 */
1054static int rtDbgSymCacheAddDir(const char *pszPath, PCRTDBGSYMCACHEADDCFG pCfg)
1055{
1056 /*
1057 * Set up the path buffer, stripping any filter.
1058 */
1059 char szPath[RTPATH_MAX];
1060 int rc = RTStrCopy(szPath, sizeof(szPath) - 2, pszPath);
1061 if (RT_FAILURE(rc))
1062 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Path too long: '%s'", pszPath);
1063
1064 size_t cchPath = strlen(pszPath);
1065 if (!cchPath)
1066 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Path empty: '%s'", pszPath);
1067
1068 if (pCfg->pszFilter)
1069 szPath[cchPath - strlen(pCfg->pszFilter)] = '\0';
1070 cchPath = RTPathStripTrailingSlash(szPath);
1071 if (!RTPATH_IS_SEP(pszPath[cchPath - 1]))
1072 {
1073 szPath[cchPath++] = RTPATH_SLASH;
1074 szPath[cchPath] = '\0';
1075 }
1076
1077 /*
1078 * Let the worker do the rest.
1079 */
1080 RTDIRENTRYEX DirEntry;
1081 return rtDbgSymCacheAddDirWorker(szPath, cchPath, &DirEntry, pCfg);
1082}
1083
1084
1085/**
1086 * Adds a file or directory.
1087 *
1088 * @returns Program exit code.
1089 * @param pszPath The user supplied path to the file or directory.
1090 * @param pszCache The path to the cache.
1091 * @param fRecursive Whether to process directories recursively.
1092 */
1093static RTEXITCODE rtDbgSymCacheAddFileOrDir(const char *pszPath, const char *pszCache, bool fRecursive)
1094{
1095 RTDBGSYMCACHEADDCFG Cfg;
1096 Cfg.fRecursive = fRecursive;
1097 Cfg.pszCache = pszCache;
1098 Cfg.pszFilter = NULL;
1099
1100 int rc;
1101 RTDBGSYMCACHEFILETYPE enmType = rtDbgSymCacheFigureType(pszPath);
1102 switch (enmType)
1103 {
1104 default:
1105 case RTDBGSYMCACHEFILETYPE_INVALID:
1106 return RTMsgErrorExit(RTEXITCODE_FAILURE, "Invalid: '%s'", pszPath);
1107
1108 case RTDBGSYMCACHEFILETYPE_DIR_FILTER:
1109 Cfg.pszFilter = RTPathFilename(pszPath);
1110 /* fall thru */
1111 case RTDBGSYMCACHEFILETYPE_DIR:
1112 rc = rtDbgSymCacheAddDir(pszPath, &Cfg);
1113 break;
1114
1115 case RTDBGSYMCACHEFILETYPE_DEBUG_FILE:
1116 rc = rtDbgSymCacheAddDebugFile(pszPath, &Cfg);
1117 break;
1118
1119 case RTDBGSYMCACHEFILETYPE_IMAGE_FILE:
1120 rc = rtDbgSymCacheAddImageFile(pszPath, NULL /*pszExtraSuff*/, RTDBG_CACHE_UUID_MAP_DIR_IMAGES, &Cfg);
1121 break;
1122
1123 case RTDBGSYMCACHEFILETYPE_DEBUG_BUNDLE:
1124 case RTDBGSYMCACHEFILETYPE_IMAGE_BUNDLE:
1125 {
1126 size_t cchPath = strlen(pszPath);
1127 size_t cchFilename = strlen(RTPathFilename(pszPath));
1128 char szPathBuf[RTPATH_MAX];
1129 if (cchPath < sizeof(szPathBuf))
1130 {
1131 memcpy(szPathBuf, pszPath, cchPath + 1);
1132 if (enmType == RTDBGSYMCACHEFILETYPE_DEBUG_BUNDLE)
1133 rc = rtDbgSymCacheAddDebugBundle(szPathBuf, cchPath - cchFilename, cchFilename, &Cfg);
1134 else
1135 {
1136 RTDIRENTRYEX DirEntry;
1137 rc = rtDbgSymCacheAddImageBundle(szPathBuf, cchPath - cchFilename, cchFilename, &DirEntry, &Cfg);
1138 }
1139 }
1140 else
1141 rc = RTMsgErrorRc(VERR_FILENAME_TOO_LONG, "Filename too long: '%s'", pszPath);
1142 break;
1143 }
1144
1145 case RTDBGSYMCACHEFILETYPE_IGNORE:
1146 rc = RTMsgErrorRc(VERR_INVALID_PARAMETER, "Invalid file: '%s'");
1147 break;
1148 }
1149 return RT_SUCCESS(rc) ? RTEXITCODE_SUCCESS : RTEXITCODE_FAILURE;
1150}
1151
1152
1153/**
1154 * Handles the 'add' command.
1155 *
1156 * @returns Program exit code.
1157 * @param pszArg0 The program name.
1158 * @param cArgs The number of arguments to the 'add' command.
1159 * @param papszArgs The argument vector, starting after 'add'.
1160 */
1161static RTEXITCODE rtDbgSymCacheCmdAdd(const char *pszArg0, int cArgs, char **papszArgs)
1162{
1163 /*
1164 * Parse the command line.
1165 */
1166 static RTGETOPTDEF const s_aOptions[] =
1167 {
1168 { "--recursive", 'R', RTGETOPT_REQ_NOTHING },
1169 { "--no-recursive", 'n', RTGETOPT_REQ_NOTHING },
1170 };
1171
1172 const char *pszCache = NULL;
1173 bool fRecursive = false;
1174
1175 RTGETOPTSTATE State;
1176 int rc = RTGetOptInit(&State, cArgs, papszArgs, &s_aOptions[0], RT_ELEMENTS(s_aOptions), 0, RTGETOPTINIT_FLAGS_OPTS_FIRST);
1177 if (RT_FAILURE(rc))
1178 return RTMsgErrorExit(RTEXITCODE_FAILURE, "RTGetOptInit failed: %Rrc", rc);
1179
1180 uint32_t cAdded = 0;
1181 RTGETOPTUNION ValueUnion;
1182 int chOpt;
1183 while ((chOpt = RTGetOpt(&State, &ValueUnion)) != 0)
1184 {
1185 switch (chOpt)
1186 {
1187 case 'R':
1188 fRecursive = true;
1189 break;
1190
1191 case 'n':
1192 fRecursive = false;
1193 break;
1194
1195 case VINF_GETOPT_NOT_OPTION:
1196 /* The first non-option is a cache directory. */
1197 if (!pszCache)
1198 {
1199 pszCache = ValueUnion.psz;
1200 if (!RTPathExists(pszCache))
1201 {
1202 rc = RTDirCreate(pszCache, 0755, RTDIRCREATE_FLAGS_NOT_CONTENT_INDEXED_NOT_CRITICAL);
1203 if (RT_FAILURE(rc))
1204 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Error creating cache directory '%s': %Rrc", pszCache, rc);
1205 }
1206 else if (!RTDirExists(pszCache))
1207 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "Specified cache directory is not a directory: '%s'", pszCache);
1208 }
1209 /* Subsequent non-options are files to be added to the cache. */
1210 else
1211 {
1212 RTEXITCODE rcExit = rtDbgSymCacheAddFileOrDir(ValueUnion.psz, pszCache, fRecursive);
1213 if (rcExit != RTEXITCODE_FAILURE)
1214 return rcExit;
1215 }
1216 break;
1217
1218 case 'h':
1219 return rtDbgSymCacheUsage(pszArg0, "add");
1220 case 'V':
1221 return rtDbgSymCacheVersion();
1222 default:
1223 return RTGetOptPrintError(chOpt, &ValueUnion);
1224 }
1225 }
1226
1227 if (!pszCache)
1228 return RTMsgErrorExit(RTEXITCODE_SYNTAX, "No cache directory or files to add were specified.");
1229 return RTEXITCODE_SUCCESS;
1230}
1231
1232
1233int main(int argc, char **argv)
1234{
1235 int rc = RTR3InitExe(argc, &argv, 0);
1236 if (RT_FAILURE(rc))
1237 return RTMsgInitFailure(rc);
1238
1239 /*
1240 * Switch on the command.
1241 */
1242 RTEXITCODE rcExit = RTEXITCODE_SYNTAX;
1243 if (argc < 2)
1244 rtDbgSymCacheUsage(argv[0], NULL);
1245 else if (!strcmp(argv[1], "add"))
1246 rcExit = rtDbgSymCacheCmdAdd(argv[0], argc - 2, argv + 2);
1247 else if ( !strcmp(argv[1], "-h")
1248 || !strcmp(argv[1], "-?")
1249 || !strcmp(argv[1], "--help"))
1250 rcExit = rtDbgSymCacheUsage(argv[0], NULL);
1251 else if ( !strcmp(argv[1], "-V")
1252 || !strcmp(argv[1], "--version"))
1253 rcExit = rtDbgSymCacheVersion();
1254 else
1255 RTMsgError("Unknown command: '%s'", argv[1]);
1256
1257 return rcExit;
1258}
1259
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