VirtualBox

source: vbox/trunk/src/VBox/Runtime/common/dbg/dbgmodcontainer.cpp@ 25524

Last change on this file since 25524 was 22111, checked in by vboxsync, 15 years ago

RTDbgAs,RTDbgMod: Bugfixes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 25.0 KB
Line 
1/* $Id: dbgmodcontainer.cpp 22111 2009-08-09 20:12:09Z vboxsync $ */
2/** @file
3 * IPRT - Debug Info Container.
4 */
5
6/*
7 * Copyright (C) 2009 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
27 * Clara, CA 95054 USA or visit http://www.sun.com if you need
28 * additional information or have any questions.
29 */
30
31
32/*******************************************************************************
33* Header Files *
34*******************************************************************************/
35#include <iprt/dbg.h>
36#include "internal/iprt.h"
37
38#include <iprt/avl.h>
39#include <iprt/err.h>
40#include <iprt/mem.h>
41#include <iprt/string.h>
42#include <iprt/strcache.h>
43#include "internal/dbgmod.h"
44
45
46/*******************************************************************************
47* Structures and Typedefs *
48*******************************************************************************/
49/**
50 * Symbol entry.
51 */
52typedef struct RTDBGMODCTNSYMBOL
53{
54 /** The address core. */
55 AVLRUINTPTRNODECORE AddrCore;
56 /** The name space core. */
57 RTSTRSPACECORE NameCore;
58 /** The ordinal number core. */
59 AVLU32NODECORE OrdinalCore;
60 /** The segment index. */
61 RTDBGSEGIDX iSeg;
62 /** The symbol flags. */
63 uint32_t fFlags;
64 /** The symbol size.
65 * This may be zero while the range in AddrCore indicates 0. */
66 RTUINTPTR cb;
67} RTDBGMODCTNSYMBOL;
68/** Pointer to a symbol entry in the debug info container. */
69typedef RTDBGMODCTNSYMBOL *PRTDBGMODCTNSYMBOL;
70/** Pointer to a const symbol entry in the debug info container. */
71typedef RTDBGMODCTNSYMBOL const *PCRTDBGMODCTNSYMBOL;
72
73/**
74 * Line number entry.
75 */
76typedef struct RTDBGMODCTNLINE
77{
78 /** The address core.
79 * The Key is the address of the line number. */
80 AVLUINTPTRNODECORE AddrCore;
81 /** The ordinal number core. */
82 AVLU32NODECORE OrdinalCore;
83 /** Pointer to the file name (in string cache). */
84 const char *pszFile;
85 /** The line number. */
86 uint32_t uLineNo;
87 /** The segment index. */
88 RTDBGSEGIDX iSeg;
89} RTDBGMODCTNLINE;
90/** Pointer to a line number entry. */
91typedef RTDBGMODCTNLINE *PRTDBGMODCTNLINE;
92/** Pointer to const a line number entry. */
93typedef RTDBGMODCTNLINE const *PCRTDBGMODCTNLINE;
94
95/**
96 * Segment entry.
97 */
98typedef struct RTDBGMODCTNSEGMENT
99{
100 /** The symbol address space tree. */
101 AVLRUINTPTRTREE SymAddrTree;
102 /** The line number address space tree. */
103 AVLUINTPTRTREE LineAddrTree;
104 /** The segment offset. */
105 RTUINTPTR off;
106 /** The segment size. */
107 RTUINTPTR cb;
108 /** The segment flags. */
109 uint32_t fFlags;
110 /** The segment name. */
111 const char *pszName;
112} RTDBGMODCTNSEGMENT;
113/** Pointer to a segment entry in the debug info container. */
114typedef RTDBGMODCTNSEGMENT *PRTDBGMODCTNSEGMENT;
115/** Pointer to a const segment entry in the debug info container. */
116typedef RTDBGMODCTNSEGMENT const *PCRTDBGMODCTNSEGMENT;
117
118/**
119 * Instance data.
120 */
121typedef struct RTDBGMODCTN
122{
123 /** The name space. */
124 RTSTRSPACE Names;
125 /** Tree containing any absolute addresses. */
126 AVLRUINTPTRTREE AbsAddrTree;
127 /** Tree organizing the symbols by ordinal number. */
128 AVLU32TREE SymbolOrdinalTree;
129 /** Tree organizing the line numbers by ordinal number. */
130 AVLU32TREE LineOrdinalTree;
131 /** Segment table. */
132 PRTDBGMODCTNSEGMENT paSegs;
133 /** The number of segments in the segment table. */
134 RTDBGSEGIDX cSegs;
135 /** The image size. 0 means unlimited. */
136 RTUINTPTR cb;
137 /** The next symbol ordinal. */
138 uint32_t iNextSymbolOrdinal;
139 /** The next line number ordinal. */
140 uint32_t iNextLineOrdinal;
141} RTDBGMODCTN;
142/** Pointer to instance data for the debug info container. */
143typedef RTDBGMODCTN *PRTDBGMODCTN;
144
145
146/**
147 * Fills in a RTDBGSYMBOL structure.
148 *
149 * @returns VINF_SUCCESS.
150 * @param pMySym Our internal symbol representation.
151 * @param pExtSym The external symbol representation.
152 */
153DECLINLINE(int) rtDbgModContainerReturnSymbol(PCRTDBGMODCTNSYMBOL pMySym, PRTDBGSYMBOL pExtSym)
154{
155 pExtSym->Value = pMySym->AddrCore.Key;
156 pExtSym->offSeg = pMySym->AddrCore.Key;
157 pExtSym->iSeg = pMySym->iSeg;
158 pExtSym->fFlags = pMySym->fFlags;
159 pExtSym->cb = pMySym->cb;
160 pExtSym->iOrdinal = pMySym->OrdinalCore.Key;
161 Assert(pMySym->NameCore.cchString < sizeof(pExtSym->szName));
162 memcpy(pExtSym->szName, pMySym->NameCore.pszString, pMySym->NameCore.cchString + 1);
163 return VINF_SUCCESS;
164}
165
166
167
168/** @copydoc RTDBGMODVTDBG::pfnLineByAddr */
169static DECLCALLBACK(int) rtDbgModContainer_LineByAddr(PRTDBGMODINT pMod, RTDBGSEGIDX iSeg, RTUINTPTR off,
170 PRTINTPTR poffDisp, PRTDBGLINE pLineInfo)
171{
172 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
173
174 /*
175 * Validate the input address.
176 */
177 AssertMsgReturn(iSeg < pThis->cSegs,
178 ("iSeg=%#x cSegs=%#x\n", pThis->cSegs),
179 VERR_DBG_INVALID_SEGMENT_INDEX);
180 AssertMsgReturn(off < pThis->paSegs[iSeg].cb,
181 ("off=%RTptr cbSeg=%RTptr\n", off, pThis->paSegs[iSeg].cb),
182 VERR_DBG_INVALID_SEGMENT_OFFSET);
183
184 /*
185 * Lookup the nearest line number with an address less or equal to the specified address.
186 */
187 PAVLUINTPTRNODECORE pAvlCore = RTAvlUIntPtrGetBestFit(&pThis->paSegs[iSeg].LineAddrTree, off, false /*fAbove*/);
188 if (!pAvlCore)
189 return pThis->iNextLineOrdinal
190 ? VERR_DBG_LINE_NOT_FOUND
191 : VERR_DBG_NO_LINE_NUMBERS;
192 PCRTDBGMODCTNLINE pMyLine = RT_FROM_MEMBER(pAvlCore, RTDBGMODCTNLINE const, AddrCore);
193 pLineInfo->Address = pMyLine->AddrCore.Key;
194 pLineInfo->offSeg = pMyLine->AddrCore.Key;
195 pLineInfo->iSeg = iSeg;
196 pLineInfo->uLineNo = pMyLine->uLineNo;
197 pLineInfo->iOrdinal = pMyLine->OrdinalCore.Key;
198 strcpy(pLineInfo->szFilename, pMyLine->pszFile);
199 if (poffDisp)
200 *poffDisp = off - pMyLine->AddrCore.Key;
201 return VINF_SUCCESS;
202}
203
204
205/** @copydoc RTDBGMODVTDBG::pfnLineByOrdinal */
206static DECLCALLBACK(int) rtDbgModContainer_LineByOrdinal(PRTDBGMODINT pMod, uint32_t iOrdinal, PRTDBGLINE pLineInfo)
207{
208 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
209
210 /*
211 * Look it up.
212 */
213 if (iOrdinal >= pThis->iNextLineOrdinal)
214 return pThis->iNextLineOrdinal
215 ? VERR_DBG_LINE_NOT_FOUND
216 : VERR_DBG_NO_LINE_NUMBERS;
217 PAVLU32NODECORE pAvlCore = RTAvlU32Get(&pThis->LineOrdinalTree, iOrdinal);
218 AssertReturn(pAvlCore, VERR_DBG_LINE_NOT_FOUND);
219 PCRTDBGMODCTNLINE pMyLine = RT_FROM_MEMBER(pAvlCore, RTDBGMODCTNLINE const, OrdinalCore);
220 pLineInfo->Address = pMyLine->AddrCore.Key;
221 pLineInfo->offSeg = pMyLine->AddrCore.Key;
222 pLineInfo->iSeg = pMyLine->iSeg;
223 pLineInfo->uLineNo = pMyLine->uLineNo;
224 pLineInfo->iOrdinal = pMyLine->OrdinalCore.Key;
225 strcpy(pLineInfo->szFilename, pMyLine->pszFile);
226 return VINF_SUCCESS;
227}
228
229
230/** @copydoc RTDBGMODVTDBG::pfnLineCount */
231static DECLCALLBACK(uint32_t) rtDbgModContainer_LineCount(PRTDBGMODINT pMod)
232{
233 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
234
235 /* Note! The ordinal numbers are 0-based. */
236 return pThis->iNextLineOrdinal;
237}
238
239
240/** @copydoc RTDBGMODVTDBG::pfnLineAdd */
241static DECLCALLBACK(int) rtDbgModContainer_LineAdd(PRTDBGMODINT pMod, const char *pszFile, size_t cchFile, uint32_t uLineNo,
242 uint32_t iSeg, RTUINTPTR off, uint32_t *piOrdinal)
243{
244 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
245
246 /*
247 * Validate the input address.
248 */
249 AssertMsgReturn(iSeg < pThis->cSegs, ("iSeg=%#x cSegs=%#x\n", pThis->cSegs),
250 VERR_DBG_INVALID_SEGMENT_INDEX);
251 AssertMsgReturn(off < pThis->paSegs[iSeg].cb, ("off=%RTptr cbSeg=%RTptr\n", off, pThis->paSegs[iSeg].cb),
252 VERR_DBG_INVALID_SEGMENT_OFFSET);
253
254 /*
255 * Create a new entry.
256 */
257 PRTDBGMODCTNLINE pLine = (PRTDBGMODCTNLINE)RTMemAllocZ(sizeof(*pLine));
258 if (!pLine)
259 return VERR_NO_MEMORY;
260 pLine->AddrCore.Key = off;
261 pLine->OrdinalCore.Key = pThis->iNextLineOrdinal;
262 pLine->uLineNo = uLineNo;
263 pLine->iSeg = iSeg;
264 pLine->pszFile = RTStrCacheEnterN(g_hDbgModStrCache, pszFile, cchFile);
265 int rc;
266 if (pLine->pszFile)
267 {
268 if (RTAvlUIntPtrInsert(&pThis->paSegs[iSeg].LineAddrTree, &pLine->AddrCore))
269 {
270 if (RTAvlU32Insert(&pThis->LineOrdinalTree, &pLine->OrdinalCore))
271 {
272 if (piOrdinal)
273 *piOrdinal = pThis->iNextLineOrdinal;
274 pThis->iNextLineOrdinal++;
275 return VINF_SUCCESS;
276 }
277
278 rc = VERR_INTERNAL_ERROR_5;
279 RTAvlUIntPtrRemove(&pThis->paSegs[iSeg].LineAddrTree, pLine->AddrCore.Key);
280 }
281
282 /* bail out */
283 rc = VERR_DBG_ADDRESS_CONFLICT;
284 RTStrCacheRelease(g_hDbgModStrCache, pLine->pszFile);
285 }
286 else
287 rc = VERR_NO_MEMORY;
288 RTMemFree(pLine);
289 return rc;
290}
291
292
293/** @copydoc RTDBGMODVTDBG::pfnSymbolByAddr */
294static DECLCALLBACK(int) rtDbgModContainer_SymbolByAddr(PRTDBGMODINT pMod, RTDBGSEGIDX iSeg, RTUINTPTR off,
295 PRTINTPTR poffDisp, PRTDBGSYMBOL pSymInfo)
296{
297 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
298
299 /*
300 * Validate the input address.
301 */
302 AssertMsgReturn( iSeg == RTDBGSEGIDX_ABS
303 || iSeg < pThis->cSegs,
304 ("iSeg=%#x cSegs=%#x\n", pThis->cSegs),
305 VERR_DBG_INVALID_SEGMENT_INDEX);
306 AssertMsgReturn( iSeg >= RTDBGSEGIDX_SPECIAL_FIRST
307 || off <= pThis->paSegs[iSeg].cb,
308 ("off=%RTptr cbSeg=%RTptr\n", off, pThis->paSegs[iSeg].cb),
309 VERR_DBG_INVALID_SEGMENT_OFFSET);
310
311 /*
312 * Lookup the nearest symbol with an address less or equal to the specified address.
313 */
314 PAVLRUINTPTRNODECORE pAvlCore = RTAvlrUIntPtrGetBestFit( iSeg == RTDBGSEGIDX_ABS
315 ? &pThis->AbsAddrTree
316 : &pThis->paSegs[iSeg].SymAddrTree,
317 off,
318 false /*fAbove*/);
319 if (!pAvlCore)
320 return VERR_SYMBOL_NOT_FOUND;
321 PCRTDBGMODCTNSYMBOL pMySym = RT_FROM_MEMBER(pAvlCore, RTDBGMODCTNSYMBOL const, AddrCore);
322 if (poffDisp)
323 *poffDisp = off - pMySym->AddrCore.Key;
324 return rtDbgModContainerReturnSymbol(pMySym, pSymInfo);
325}
326
327
328/** @copydoc RTDBGMODVTDBG::pfnSymbolByName */
329static DECLCALLBACK(int) rtDbgModContainer_SymbolByName(PRTDBGMODINT pMod, const char *pszSymbol, size_t cchSymbol, PRTDBGSYMBOL pSymInfo)
330{
331 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
332 NOREF(cchSymbol);
333
334 /*
335 * Look it up in the name space.
336 */
337 PRTSTRSPACECORE pStrCore = RTStrSpaceGet(&pThis->Names, pszSymbol);
338 if (!pStrCore)
339 return VERR_SYMBOL_NOT_FOUND;
340 PCRTDBGMODCTNSYMBOL pMySym = RT_FROM_MEMBER(pStrCore, RTDBGMODCTNSYMBOL const, NameCore);
341 return rtDbgModContainerReturnSymbol(pMySym, pSymInfo);
342}
343
344
345/** @copydoc RTDBGMODVTDBG::pfnSymbolByOrdinal */
346static DECLCALLBACK(int) rtDbgModContainer_SymbolByOrdinal(PRTDBGMODINT pMod, uint32_t iOrdinal, PRTDBGSYMBOL pSymInfo)
347{
348 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
349
350 /*
351 * Look it up in the ordinal tree.
352 */
353 if (iOrdinal >= pThis->iNextSymbolOrdinal)
354 return pThis->iNextSymbolOrdinal
355 ? VERR_DBG_NO_SYMBOLS
356 : VERR_SYMBOL_NOT_FOUND;
357 PAVLU32NODECORE pAvlCore = RTAvlU32Get(&pThis->SymbolOrdinalTree, iOrdinal);
358 AssertReturn(pAvlCore, VERR_SYMBOL_NOT_FOUND);
359 PCRTDBGMODCTNSYMBOL pMySym = RT_FROM_MEMBER(pAvlCore, RTDBGMODCTNSYMBOL const, OrdinalCore);
360 return rtDbgModContainerReturnSymbol(pMySym, pSymInfo);
361}
362
363
364/** @copydoc RTDBGMODVTDBG::pfnSymbolCount */
365static DECLCALLBACK(uint32_t) rtDbgModContainer_SymbolCount(PRTDBGMODINT pMod)
366{
367 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
368
369 /* Note! The ordinal numbers are 0-based. */
370 return pThis->iNextSymbolOrdinal;
371}
372
373
374/** @copydoc RTDBGMODVTDBG::pfnSymbolAdd */
375static DECLCALLBACK(int) rtDbgModContainer_SymbolAdd(PRTDBGMODINT pMod, const char *pszSymbol, size_t cchSymbol,
376 RTDBGSEGIDX iSeg, RTUINTPTR off, RTUINTPTR cb, uint32_t fFlags,
377 uint32_t *piOrdinal)
378{
379 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
380
381 /*
382 * Address validation. The other arguments have already been validated.
383 */
384 AssertMsgReturn( iSeg == RTDBGSEGIDX_ABS
385 || iSeg < pThis->cSegs,
386 ("iSeg=%#x cSegs=%#x\n", pThis->cSegs),
387 VERR_DBG_INVALID_SEGMENT_INDEX);
388 AssertMsgReturn( iSeg >= RTDBGSEGIDX_SPECIAL_FIRST
389 || off + cb <= pThis->paSegs[iSeg].cb,
390 ("off=%RTptr cb=%RTptr cbSeg=%RTptr\n", off, cb, pThis->paSegs[iSeg].cb),
391 VERR_DBG_INVALID_SEGMENT_OFFSET);
392
393 /*
394 * Create a new entry.
395 */
396 PRTDBGMODCTNSYMBOL pSymbol = (PRTDBGMODCTNSYMBOL)RTMemAllocZ(sizeof(*pSymbol));
397 if (!pSymbol)
398 return VERR_NO_MEMORY;
399
400 pSymbol->AddrCore.Key = off;
401 pSymbol->AddrCore.KeyLast = off + (cb ? cb - 1 : 0);
402 pSymbol->OrdinalCore.Key = pThis->iNextSymbolOrdinal;
403 pSymbol->iSeg = iSeg;
404 pSymbol->cb = cb;
405 pSymbol->fFlags = fFlags;
406 pSymbol->NameCore.pszString = RTStrCacheEnter(g_hDbgModStrCache, pszSymbol);
407 int rc;
408 if (pSymbol->NameCore.pszString)
409 {
410 if (RTStrSpaceInsert(&pThis->Names, &pSymbol->NameCore))
411 {
412 PAVLRUINTPTRTREE pAddrTree = iSeg == RTDBGSEGIDX_ABS
413 ? &pThis->AbsAddrTree
414 : &pThis->paSegs[iSeg].SymAddrTree;
415 if (RTAvlrUIntPtrInsert(pAddrTree, &pSymbol->AddrCore))
416 {
417 if (RTAvlU32Insert(&pThis->SymbolOrdinalTree, &pSymbol->OrdinalCore))
418 {
419 if (piOrdinal)
420 *piOrdinal = pThis->iNextSymbolOrdinal;
421 pThis->iNextSymbolOrdinal++;
422 return VINF_SUCCESS;
423 }
424
425 /* bail out */
426 rc = VERR_INTERNAL_ERROR_5;
427 RTAvlrUIntPtrRemove(pAddrTree, pSymbol->AddrCore.Key);
428 }
429 else
430 rc = VERR_DBG_ADDRESS_CONFLICT;
431 RTStrSpaceRemove(&pThis->Names, pSymbol->NameCore.pszString);
432 }
433 else
434 rc = VERR_DBG_DUPLICATE_SYMBOL;
435 RTStrCacheRelease(g_hDbgModStrCache, pSymbol->NameCore.pszString);
436 }
437 else
438 rc = VERR_NO_MEMORY;
439 RTMemFree(pSymbol);
440 return rc;
441}
442
443
444/** @copydoc RTDBGMODVTDBG::pfnSegmentByIndex */
445static DECLCALLBACK(int) rtDbgModContainer_SegmentByIndex(PRTDBGMODINT pMod, RTDBGSEGIDX iSeg, PRTDBGSEGMENT pSegInfo)
446{
447 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
448 if (iSeg >= pThis->cSegs)
449 return VERR_DBG_INVALID_SEGMENT_INDEX;
450 pSegInfo->Address = RTUINTPTR_MAX;
451 pSegInfo->uRva = pThis->paSegs[iSeg].off;
452 pSegInfo->cb = pThis->paSegs[iSeg].cb;
453 pSegInfo->fFlags = pThis->paSegs[iSeg].fFlags;
454 pSegInfo->iSeg = iSeg;
455 strcpy(pSegInfo->szName, pThis->paSegs[iSeg].pszName);
456 return VINF_SUCCESS;
457}
458
459
460/** @copydoc RTDBGMODVTDBG::pfnSegmentCount */
461static DECLCALLBACK(RTDBGSEGIDX) rtDbgModContainer_SegmentCount(PRTDBGMODINT pMod)
462{
463 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
464 return pThis->cSegs;
465}
466
467
468/** @copydoc RTDBGMODVTDBG::pfnSegmentAdd */
469static DECLCALLBACK(int) rtDbgModContainer_SegmentAdd(PRTDBGMODINT pMod, RTUINTPTR uRva, RTUINTPTR cb, const char *pszName, size_t cchName,
470 uint32_t fFlags, PRTDBGSEGIDX piSeg)
471{
472 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
473
474 /*
475 * Input validation (the bits the caller cannot do).
476 */
477 /* Overlapping segments are not yet supported. Will use flags to deal with it if it becomes necessary. */
478 RTUINTPTR uRvaLast = uRva + RT_MAX(cb, 1) - 1;
479 RTUINTPTR uRvaLastMax = uRvaLast;
480 RTDBGSEGIDX iSeg = pThis->cSegs;
481 while (iSeg-- > 0)
482 {
483 RTUINTPTR uCurRva = pThis->paSegs[iSeg].off;
484 RTUINTPTR uCurRvaLast = uCurRva + RT_MAX(pThis->paSegs[iSeg].cb, 1) - 1;
485 if ( uRva <= uCurRvaLast
486 && uRvaLast >= uCurRva)
487 AssertMsgFailedReturn(("uRva=%RTptr uRvaLast=%RTptr (cb=%RTptr) \"%s\";\n"
488 "uRva=%RTptr uRvaLast=%RTptr (cb=%RTptr) \"%s\" iSeg=%#x\n",
489 uRva, uRvaLast, cb, pszName,
490 uCurRva, uCurRvaLast, pThis->paSegs[iSeg].cb, pThis->paSegs[iSeg].pszName, iSeg),
491 VERR_DBG_SEGMENT_INDEX_CONFLICT);
492 if (uRvaLastMax < uCurRvaLast)
493 uRvaLastMax = uCurRvaLast;
494 }
495 /* Strict ordered segment addition at the moment. */
496 iSeg = pThis->cSegs;
497 AssertMsgReturn(!piSeg || *piSeg == NIL_RTDBGSEGIDX || *piSeg == iSeg,
498 ("iSeg=%#x *piSeg=%#x\n", iSeg, *piSeg),
499 VERR_DBG_INVALID_SEGMENT_INDEX);
500
501 /*
502 * Add an entry to the segment table, extending it if necessary.
503 */
504 if (!(iSeg % 8))
505 {
506 void *pvSegs = RTMemRealloc(pThis->paSegs, sizeof(RTDBGMODCTNSEGMENT) * (iSeg + 8));
507 if (!pvSegs)
508 return VERR_NO_MEMORY;
509 pThis->paSegs = (PRTDBGMODCTNSEGMENT)pvSegs;
510 }
511
512 pThis->paSegs[iSeg].SymAddrTree = NULL;
513 pThis->paSegs[iSeg].LineAddrTree = NULL;
514 pThis->paSegs[iSeg].off = uRva;
515 pThis->paSegs[iSeg].cb = cb;
516 pThis->paSegs[iSeg].fFlags = fFlags;
517 pThis->paSegs[iSeg].pszName = RTStrCacheEnterN(g_hDbgModStrCache, pszName, cchName);
518 if (pThis->paSegs[iSeg].pszName)
519 {
520 if (piSeg)
521 *piSeg = iSeg;
522 pThis->cSegs++;
523 pThis->cb = uRvaLastMax + 1;
524 if (!pThis->cb)
525 pThis->cb = RTUINTPTR_MAX;
526 return VINF_SUCCESS;
527 }
528 return VERR_NO_MEMORY;
529}
530
531
532/** @copydoc RTDBGMODVTDBG::pfnRvaToSegOff */
533static DECLCALLBACK(RTUINTPTR) rtDbgModContainer_ImageSize(PRTDBGMODINT pMod)
534{
535 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
536 return pThis->cb;
537}
538
539
540/** @copydoc RTDBGMODVTDBG::pfnRvaToSegOff */
541static DECLCALLBACK(RTDBGSEGIDX) rtDbgModContainer_RvaToSegOff(PRTDBGMODINT pMod, RTUINTPTR uRva, PRTUINTPTR poffSeg)
542{
543 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
544 PCRTDBGMODCTNSEGMENT paSeg = pThis->paSegs;
545 uint32_t const cSegs = pThis->cSegs;
546 if (cSegs <= 7)
547 {
548 /*
549 * Linear search.
550 */
551 for (uint32_t iSeg = 0; iSeg < cSegs; iSeg++)
552 {
553 RTUINTPTR offSeg = uRva - paSeg[iSeg].off;
554 if (offSeg < paSeg[iSeg].cb)
555 {
556 if (poffSeg)
557 *poffSeg = offSeg;
558 return iSeg;
559 }
560 }
561 }
562 else
563 {
564 /*
565 * Binary search.
566 */
567 uint32_t iFirst = 0;
568 uint32_t iLast = cSegs - 1;
569 for (;;)
570 {
571 uint32_t iSeg = iFirst + (iFirst - iLast) / 2;
572 RTUINTPTR offSeg = uRva - paSeg[iSeg].off;
573 if (offSeg < paSeg[iSeg].cb)
574 {
575 if (poffSeg)
576 *poffSeg = offSeg;
577 return iSeg;
578 }
579
580 /* advance */
581 if (uRva < paSeg[iSeg].off)
582 {
583 /* between iFirst and iSeg. */
584 if (iSeg == iFirst)
585 break;
586 iLast = iSeg - 1;
587 }
588 else
589 {
590 /* between iSeg and iLast. */
591 if (iSeg == iLast)
592 break;
593 iFirst = iSeg + 1;
594 }
595 }
596 }
597
598 /* Invalid. */
599 return NIL_RTDBGSEGIDX;
600}
601
602
603/** Destroy a symbol node. */
604static DECLCALLBACK(int) rtDbgModContainer_DestroyTreeNode(PAVLRUINTPTRNODECORE pNode, void *pvUser)
605{
606 PRTDBGMODCTNSYMBOL pSym = RT_FROM_MEMBER(pNode, RTDBGMODCTNSYMBOL, AddrCore);
607 RTStrCacheRelease(g_hDbgModStrCache, pSym->NameCore.pszString);
608 pSym->NameCore.pszString = NULL;
609 RTMemFree(pSym);
610 return 0;
611}
612
613
614/** @copydoc RTDBGMODVTDBG::pfnClose */
615static DECLCALLBACK(int) rtDbgModContainer_Close(PRTDBGMODINT pMod)
616{
617 PRTDBGMODCTN pThis = (PRTDBGMODCTN)pMod->pvDbgPriv;
618
619 /*
620 * Destroy the symbols and instance data.
621 */
622 for (uint32_t iSeg = 0; iSeg < pThis->cSegs; iSeg++)
623 {
624 RTAvlrUIntPtrDestroy(&pThis->paSegs[iSeg].SymAddrTree, rtDbgModContainer_DestroyTreeNode, NULL);
625 RTStrCacheRelease(g_hDbgModStrCache, pThis->paSegs[iSeg].pszName);
626 pThis->paSegs[iSeg].pszName = NULL;
627 }
628
629 RTAvlrUIntPtrDestroy(&pThis->AbsAddrTree, rtDbgModContainer_DestroyTreeNode, NULL);
630 pThis->Names = NULL;
631
632 RTMemFree(pThis->paSegs);
633 pThis->paSegs = NULL;
634
635 RTMemFree(pThis);
636
637 return VINF_SUCCESS;
638}
639
640
641/** @copydoc RTDBGMODVTDBG::pfnTryOpen */
642static DECLCALLBACK(int) rtDbgModContainer_TryOpen(PRTDBGMODINT pMod)
643{
644 return VERR_INTERNAL_ERROR_5;
645}
646
647
648
649/** Virtual function table for the debug info container. */
650static RTDBGMODVTDBG const g_rtDbgModVtDbgContainer =
651{
652 /*.u32Magic = */ RTDBGMODVTDBG_MAGIC,
653 /*.fSupports = */ 0, /* (Don't call my TryOpen, please.) */
654 /*.pszName = */ "container",
655 /*.pfnTryOpen = */ rtDbgModContainer_TryOpen,
656 /*.pfnClose = */ rtDbgModContainer_Close,
657
658 /*.pfnRvaToSegOff = */ rtDbgModContainer_RvaToSegOff,
659 /*.pfnImageSize = */ rtDbgModContainer_ImageSize,
660
661 /*.pfnSegmentAdd = */ rtDbgModContainer_SegmentAdd,
662 /*.pfnSegmentCount = */ rtDbgModContainer_SegmentCount,
663 /*.pfnSegmentByIndex = */ rtDbgModContainer_SegmentByIndex,
664
665 /*.pfnSymbolAdd = */ rtDbgModContainer_SymbolAdd,
666 /*.pfnSymbolCount = */ rtDbgModContainer_SymbolCount,
667 /*.pfnSymbolByOrdinal = */ rtDbgModContainer_SymbolByOrdinal,
668 /*.pfnSymbolByName = */ rtDbgModContainer_SymbolByName,
669 /*.pfnSymbolByAddr = */ rtDbgModContainer_SymbolByAddr,
670
671 /*.pfnLineAdd = */ rtDbgModContainer_LineAdd,
672 /*.pfnLineCount = */ rtDbgModContainer_LineCount,
673 /*.pfnLineByOrdinal = */ rtDbgModContainer_LineByOrdinal,
674 /*.pfnLineByAddr = */ rtDbgModContainer_LineByAddr,
675
676 /*.u32EndMagic = */ RTDBGMODVTDBG_MAGIC
677};
678
679
680
681/**
682 * Creates a generic debug info container and associates it with the module.
683 *
684 * @returns IPRT status code.
685 * @param pMod The module instance.
686 * @param cbSeg The size of the initial segment. 0 if segments are to be
687 * created manually later on.
688 */
689int rtDbgModContainerCreate(PRTDBGMODINT pMod, RTUINTPTR cbSeg)
690{
691 PRTDBGMODCTN pThis = (PRTDBGMODCTN)RTMemAlloc(sizeof(*pThis));
692 if (!pThis)
693 return VERR_NO_MEMORY;
694
695 pThis->Names = NULL;
696 pThis->AbsAddrTree = NULL;
697 pThis->SymbolOrdinalTree = NULL;
698 pThis->LineOrdinalTree = NULL;
699 pThis->paSegs = NULL;
700 pThis->cSegs = 0;
701 pThis->cb = 0;
702 pThis->iNextSymbolOrdinal = 0;
703 pThis->iNextLineOrdinal = 0;
704
705 pMod->pDbgVt = &g_rtDbgModVtDbgContainer;
706 pMod->pvDbgPriv = pThis;
707
708 /*
709 * Add the initial segment.
710 */
711 if (cbSeg)
712 {
713 int rc = rtDbgModContainer_SegmentAdd(pMod, 0, cbSeg, "default", sizeof("default") - 1, 0, NULL);
714 if (RT_FAILURE(rc))
715 {
716 RTMemFree(pThis);
717 pMod->pDbgVt = NULL;
718 pMod->pvDbgPriv = NULL;
719 return rc;
720 }
721 }
722
723 return VINF_SUCCESS;
724}
725
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