src/hotspot/os/windows/symbolengine.cpp
author goetz
Wed, 20 Sep 2017 15:12:11 +0200
changeset 47666 19219ec3f176
child 49039 0f81d08a578e
permissions -rw-r--r--
8185712: [windows] Improve native symbol decoder Reviewed-by: goetz, iklam, zgu
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
47666
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
     1
/*
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
     2
 * Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
     4
 *
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
     7
 * published by the Free Software Foundation.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
     8
 *
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
     9
 * This code is distributed in the hope that it will be useful, but WITHOUT
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    12
 * version 2 for more details (a copy is included in the LICENSE file that
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    13
 * accompanied this code).
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    14
 *
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    15
 * You should have received a copy of the GNU General Public License version
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    16
 * 2 along with this work; if not, write to the Free Software Foundation,
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    17
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    18
 *
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    19
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    20
 * or visit www.oracle.com if you need additional information or have any
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    21
 * questions.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    22
 *
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    23
 */
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    24
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    25
#include "precompiled.hpp"
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    26
#include "utilities/globalDefinitions.hpp"
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    27
#include "symbolengine.hpp"
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    28
#include "utilities/debug.hpp"
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    29
#include "windbghelp.hpp"
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    30
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    31
#include <windows.h>
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    32
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    33
#include <imagehlp.h>
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    34
#include <psapi.h>
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    35
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    36
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    37
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    38
// This code may be invoked normally but also as part of error reporting
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    39
// In the latter case, we may run under tight memory constraints (native oom)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    40
// or in a stack overflow situation or the C heap may be corrupted. We may
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    41
// run very early before VM initialization or very late when C exit handlers
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    42
// run. In all these cases, callstacks would still be nice, so lets be robust.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    43
//
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    44
// We need a number of buffers - for the pdb search path, module handle
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    45
// lists, for demangled symbols, etc.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    46
//
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    47
// These buffers, while typically small, may need to be large for corner
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    48
// cases (e.g. templatized C++ symbols, or many DLLs loaded). Where do we
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    49
// allocate them?
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    50
//
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    51
// We may be in error handling for a stack overflow, so lets not put them on
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    52
// the stack.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    53
//
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    54
// Dynamically allocating them may fail if we are handling a native OOM. It
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    55
// is also a bit dangerous, as the C heap may be corrupted already.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    56
//
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    57
// That leaves pre-allocating them globally, which is safe and should always
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    58
// work (if we synchronize access) but incurs an undesirable footprint for
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    59
// non-error cases.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    60
//
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    61
// We follow a two-way strategy: Allocate the buffers on the C heap in a
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    62
// reasonable large size. Failing that, fall back to static preallocated
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    63
// buffers. The size of the latter is large enough to handle common scenarios
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    64
// but small enough not to drive up the footprint too much (several kb).
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    65
//
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    66
// We keep these buffers around once allocated, for subsequent requests. This
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    67
// means that by running the initialization early at a safe time - before
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    68
// any error happens - buffers can be pre-allocated. This increases the chance
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    69
// of useful callstacks in error scenarios in exchange for a some cycles spent
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    70
// at startup. This behavior can be controlled with -XX:+InitializeDbgHelpEarly
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    71
// and is off by default.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    72
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    73
///////
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    74
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    75
// A simple buffer which attempts to allocate an optimal size but will
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    76
// fall back to a static minimally sized array on allocation error.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    77
template <class T, int MINIMAL_CAPACITY, int OPTIMAL_CAPACITY>
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    78
class SimpleBufferWithFallback {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    79
  T _fallback_buffer[MINIMAL_CAPACITY];
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    80
  T* _p;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    81
  int _capacity;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    82
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    83
  // A sentinel at the end of the buffer to catch overflows.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    84
  void imprint_sentinel() {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    85
    assert(_p && _capacity > 0, "Buffer must be allocated");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    86
    _p[_capacity - 1] = (T)'X';
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    87
    _capacity --;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    88
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    89
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    90
public:
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    91
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    92
  SimpleBufferWithFallback<T, MINIMAL_CAPACITY, OPTIMAL_CAPACITY> ()
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    93
    : _p(NULL), _capacity(0)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    94
  {}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    95
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    96
  // Note: no destructor because these buffers should, once
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    97
  // allocated, live until process end.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    98
  // ~SimpleBufferWithFallback()
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
    99
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   100
  // Note: We use raw ::malloc/::free here instead of os::malloc()/os::free
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   101
  // to prevent circularities or secondary crashes during error reporting.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   102
  virtual void initialize () {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   103
    assert(_p == NULL && _capacity == 0, "Only call once.");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   104
    const size_t bytes = OPTIMAL_CAPACITY * sizeof(T);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   105
    T* q = (T*) ::malloc(bytes);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   106
    if (q != NULL) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   107
      _p = q;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   108
      _capacity = OPTIMAL_CAPACITY;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   109
    } else {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   110
      _p = _fallback_buffer;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   111
      _capacity = (int)(sizeof(_fallback_buffer) / sizeof(T));
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   112
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   113
    _p[0] = '\0';
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   114
    imprint_sentinel();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   115
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   116
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   117
  // We need a way to reset the buffer to fallback size for one special
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   118
  // case, where two buffers need to be of identical capacity.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   119
  void reset_to_fallback_capacity() {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   120
    if (_p != _fallback_buffer) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   121
      ::free(_p);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   122
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   123
    _p = _fallback_buffer;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   124
    _capacity = (int)(sizeof(_fallback_buffer) / sizeof(T));
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   125
    _p[0] = '\0';
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   126
    imprint_sentinel();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   127
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   128
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   129
  T* ptr()                { return _p; }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   130
  const T* ptr() const    { return _p; }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   131
  int capacity() const    { return _capacity; }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   132
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   133
#ifdef ASSERT
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   134
  void check() const {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   135
    assert(_p[_capacity] == (T)'X', "sentinel lost");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   136
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   137
#else
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   138
  void check() const {}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   139
#endif
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   140
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   141
};
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   142
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   143
////
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   144
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   145
// ModuleHandleArray: a list holding module handles. Needs to be large enough
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   146
// to hold one handle per loaded DLL.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   147
// Note: a standard OpenJDK loads normally ~30 libraries, including system
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   148
// libraries, without third party libraries.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   149
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   150
typedef SimpleBufferWithFallback <HMODULE, 48, 512> ModuleHandleArrayBase;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   151
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   152
class ModuleHandleArray : public ModuleHandleArrayBase {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   153
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   154
  int _num; // Number of handles in this array (may be < capacity).
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   155
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   156
public:
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   157
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   158
  void initialize() {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   159
    ModuleHandleArrayBase::initialize();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   160
    _num = 0;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   161
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   162
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   163
  int num() const { return _num; }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   164
  void set_num(int n) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   165
    assert(n <= capacity(), "Too large");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   166
    _num = n;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   167
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   168
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   169
  // Compare with another list; returns true if all handles are equal (incl.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   170
  // sort order)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   171
  bool equals(const ModuleHandleArray& other) const {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   172
    if (_num != other._num) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   173
      return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   174
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   175
    if (::memcmp(ptr(), other.ptr(), _num * sizeof(HMODULE)) != 0) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   176
      return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   177
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   178
    return true;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   179
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   180
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   181
  // Copy content from other list.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   182
  void copy_content_from(ModuleHandleArray& other) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   183
    assert(capacity() == other.capacity(), "Different capacities.");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   184
    memcpy(ptr(), other.ptr(), other._num * sizeof(HMODULE));
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   185
    _num = other._num;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   186
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   187
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   188
};
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   189
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   190
////
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   191
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   192
// PathBuffer: a buffer to hold and work with a pdb search PATH - a concatenation
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   193
// of multiple directories separated by ';'.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   194
// A single directory name can be (NTFS) as long as 32K, but in reality is
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   195
// seldom larger than the (historical) MAX_PATH of 260.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   196
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   197
#define MINIMUM_PDB_PATH_LENGTH  MAX_PATH * 4
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   198
#define OPTIMAL_PDB_PATH_LENGTH  MAX_PATH * 64
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   199
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   200
typedef SimpleBufferWithFallback<char, MINIMUM_PDB_PATH_LENGTH, OPTIMAL_PDB_PATH_LENGTH> PathBufferBase;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   201
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   202
class PathBuffer: public PathBufferBase {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   203
public:
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   204
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   205
  // Search PDB path for a directory. Search is case insensitive. Returns
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   206
  // true if directory was found in the path, false otherwise.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   207
  bool contains_directory(const char* directory) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   208
    if (ptr() == NULL) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   209
      return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   210
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   211
    const size_t len = strlen(directory);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   212
    if (len == 0) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   213
      return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   214
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   215
    char* p = ptr();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   216
    for(;;) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   217
      char* q = strchr(p, ';');
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   218
      if (q != NULL) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   219
        if (len == (q - p)) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   220
          if (strnicmp(p, directory, len) == 0) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   221
            return true;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   222
          }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   223
        }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   224
        p = q + 1;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   225
      } else {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   226
        // tail
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   227
        return stricmp(p, directory) == 0 ? true : false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   228
      }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   229
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   230
    return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   231
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   232
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   233
  // Appends the given directory to the path. Returns false if internal
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   234
  // buffer size was not sufficient.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   235
  bool append_directory(const char* directory) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   236
    const size_t len = strlen(directory);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   237
    if (len == 0) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   238
      return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   239
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   240
    char* p = ptr();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   241
    const size_t len_now = strlen(p);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   242
    const size_t needs_capacity = len_now + 1 + len + 1; // xxx;yy\0
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   243
    if (needs_capacity > (size_t)capacity()) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   244
      return false; // OOM
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   245
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   246
    if (len_now > 0) { // Not the first path element.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   247
      p += len_now;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   248
      *p = ';';
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   249
      p ++;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   250
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   251
    strcpy(p, directory);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   252
    return true;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   253
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   254
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   255
};
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   256
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   257
// A simple buffer to hold one single file name. A file name can be (NTFS) as
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   258
// long as 32K, but in reality is seldom larger than MAX_PATH.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   259
typedef SimpleBufferWithFallback<char, MAX_PATH, 8 * K> FileNameBuffer;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   260
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   261
// A buffer to hold a C++ symbol. Usually small, but symbols may be larger for
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   262
// templates.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   263
#define MINIMUM_SYMBOL_NAME_LEN 128
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   264
#define OPTIMAL_SYMBOL_NAME_LEN 1024
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   265
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   266
typedef SimpleBufferWithFallback<uint8_t,
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   267
        sizeof(IMAGEHLP_SYMBOL64) + MINIMUM_SYMBOL_NAME_LEN,
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   268
        sizeof(IMAGEHLP_SYMBOL64) + OPTIMAL_SYMBOL_NAME_LEN> SymbolBuffer;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   269
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   270
static struct {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   271
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   272
  // Two buffers to hold lists of loaded modules. handles across invocations of
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   273
  // SymbolEngine::recalc_search_path().
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   274
  ModuleHandleArray loaded_modules;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   275
  ModuleHandleArray last_loaded_modules;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   276
  // Buffer to retrieve and assemble the pdb search path.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   277
  PathBuffer search_path;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   278
  // Buffer to retrieve directory names for loaded modules.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   279
  FileNameBuffer dir_name;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   280
  // Buffer to retrieve decoded symbol information (in SymbolEngine::decode)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   281
  SymbolBuffer decode_buffer;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   282
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   283
  void initialize() {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   284
    search_path.initialize();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   285
    dir_name.initialize();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   286
    decode_buffer.initialize();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   287
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   288
    loaded_modules.initialize();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   289
    last_loaded_modules.initialize();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   290
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   291
    // Note: both module lists must have the same capacity. If one allocation
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   292
    // did fail, let them both fall back to the fallback size.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   293
    if (loaded_modules.capacity() != last_loaded_modules.capacity()) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   294
      loaded_modules.reset_to_fallback_capacity();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   295
      last_loaded_modules.reset_to_fallback_capacity();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   296
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   297
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   298
    assert(search_path.capacity() > 0 && dir_name.capacity() > 0 &&
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   299
            decode_buffer.capacity() > 0 && loaded_modules.capacity() > 0 &&
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   300
            last_loaded_modules.capacity() > 0, "Init error.");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   301
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   302
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   303
} g_buffers;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   304
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   305
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   306
// Scan the loaded modules.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   307
//
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   308
// For each loaded module, add the directory it is located in to the pdb search
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   309
// path, but avoid duplicates. Prior search path content is preserved.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   310
//
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   311
// If p_search_path_was_updated is not NULL, points to a bool which, upon
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   312
// successful return from the function, contains true if the search path
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   313
// was updated, false if no update was needed because no new DLLs were
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   314
// loaded or unloaded.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   315
//
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   316
// Returns true for success, false for error.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   317
static bool recalc_search_path_locked(bool* p_search_path_was_updated) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   318
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   319
  if (p_search_path_was_updated) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   320
    *p_search_path_was_updated = false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   321
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   322
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   323
  HANDLE hProcess = ::GetCurrentProcess();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   324
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   325
  BOOL success = false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   326
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   327
  // 1) Retrieve current set search path.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   328
  //    (PDB search path is a global setting and someone might have modified
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   329
  //     it, so take care not to remove directories, just to add our own).
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   330
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   331
  if (!WindowsDbgHelp::symGetSearchPath(hProcess, g_buffers.search_path.ptr(),
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   332
                                       (int)g_buffers.search_path.capacity())) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   333
    return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   334
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   335
  DEBUG_ONLY(g_buffers.search_path.check();)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   336
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   337
  // 2) Retrieve list of modules handles of all currently loaded modules.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   338
  DWORD bytes_needed = 0;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   339
  const DWORD buffer_capacity_bytes = (DWORD)g_buffers.loaded_modules.capacity() * sizeof(HMODULE);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   340
  success = ::EnumProcessModules(hProcess, g_buffers.loaded_modules.ptr(),
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   341
                                 buffer_capacity_bytes, &bytes_needed);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   342
  DEBUG_ONLY(g_buffers.loaded_modules.check();)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   343
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   344
  // Note: EnumProcessModules is sloppily defined in terms of whether a
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   345
  // too-small output buffer counts as error. Will it truncate but still
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   346
  // return TRUE? Nobody knows and the manpage is not telling. So we count
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   347
  // truncation it as error, disregarding the return value.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   348
  if (!success || bytes_needed > buffer_capacity_bytes) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   349
    return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   350
  } else {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   351
    const int num_modules = bytes_needed / sizeof(HMODULE);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   352
    g_buffers.loaded_modules.set_num(num_modules);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   353
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   354
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   355
  // Compare the list of module handles with the last list. If the lists are
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   356
  // identical, no additional dlls were loaded and we can stop.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   357
  if (g_buffers.loaded_modules.equals(g_buffers.last_loaded_modules)) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   358
    return true;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   359
  } else {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   360
    // Remember the new set of module handles and continue.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   361
    g_buffers.last_loaded_modules.copy_content_from(g_buffers.loaded_modules);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   362
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   363
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   364
  // 3) For each loaded module: retrieve directory from which it was loaded.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   365
  //    Add directory to search path (but avoid duplicates).
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   366
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   367
  bool did_modify_searchpath = false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   368
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   369
  for (int i = 0; i < (int)g_buffers.loaded_modules.num(); i ++) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   370
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   371
    const HMODULE hMod = g_buffers.loaded_modules.ptr()[i];
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   372
    char* const filebuffer = g_buffers.dir_name.ptr();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   373
    const int file_buffer_capacity = g_buffers.dir_name.capacity();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   374
    const int len_returned = (int)::GetModuleFileName(hMod, filebuffer, (DWORD)file_buffer_capacity);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   375
    DEBUG_ONLY(g_buffers.dir_name.check();)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   376
    if (len_returned == 0) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   377
      // Error. This is suspicious - this may happen if a module has just been
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   378
      // unloaded concurrently after our call to EnumProcessModules and
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   379
      // GetModuleFileName, but probably just indicates a coding error.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   380
      assert(false, "GetModuleFileName failed (%u)", ::GetLastError());
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   381
    } else if (len_returned == file_buffer_capacity) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   382
      // Truncation. Just skip this module and continue with the next module.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   383
      continue;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   384
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   385
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   386
    // Cut file name part off.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   387
    char* last_slash = ::strrchr(filebuffer, '\\');
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   388
    if (last_slash == NULL) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   389
      last_slash = ::strrchr(filebuffer, '/');
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   390
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   391
    if (last_slash) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   392
      *last_slash = '\0';
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   393
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   394
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   395
    // If this is already part of the search path, ignore it, otherwise
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   396
    // append to search path.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   397
    if (!g_buffers.search_path.contains_directory(filebuffer)) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   398
      if (!g_buffers.search_path.append_directory(filebuffer)) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   399
        return false; // oom
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   400
      }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   401
      DEBUG_ONLY(g_buffers.search_path.check();)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   402
      did_modify_searchpath = true;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   403
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   404
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   405
  } // for each loaded module.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   406
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   407
  // If we did not modify the search path, nothing further needs to be done.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   408
  if (!did_modify_searchpath) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   409
    return true;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   410
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   411
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   412
  // Set the search path to its new value.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   413
  if (!WindowsDbgHelp::symSetSearchPath(hProcess, g_buffers.search_path.ptr())) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   414
    return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   415
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   416
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   417
  if (p_search_path_was_updated) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   418
    *p_search_path_was_updated = true;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   419
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   420
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   421
  return true;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   422
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   423
}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   424
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   425
static bool demangle_locked(const char* symbol, char *buf, int buflen) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   426
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   427
  return WindowsDbgHelp::unDecorateSymbolName(symbol, buf, buflen, UNDNAME_COMPLETE) > 0;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   428
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   429
}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   430
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   431
static bool decode_locked(const void* addr, char* buf, int buflen, int* offset, bool do_demangle) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   432
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   433
  assert(g_buffers.decode_buffer.capacity() >= (sizeof(IMAGEHLP_SYMBOL64) + MINIMUM_SYMBOL_NAME_LEN),
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   434
         "Decode buffer too small.");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   435
  assert(buf != NULL && buflen > 0 && offset != NULL, "invalid output buffer.");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   436
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   437
  DWORD64 displacement;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   438
  PIMAGEHLP_SYMBOL64 pSymbol = NULL;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   439
  bool success = false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   440
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   441
  pSymbol = (PIMAGEHLP_SYMBOL64) g_buffers.decode_buffer.ptr();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   442
  pSymbol->SizeOfStruct = sizeof(IMAGEHLP_SYMBOL64);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   443
  pSymbol->MaxNameLength = (DWORD)(g_buffers.decode_buffer.capacity() - sizeof(IMAGEHLP_SYMBOL64) - 1);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   444
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   445
  // It is unclear how SymGetSymFromAddr64 handles truncation. Experiments
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   446
  // show it will return TRUE but not zero terminate (which is a really bad
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   447
  // combination). Lets be super careful.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   448
  ::memset(pSymbol->Name, 0, pSymbol->MaxNameLength); // To catch truncation.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   449
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   450
  if (WindowsDbgHelp::symGetSymFromAddr64(::GetCurrentProcess(), (DWORD64)addr, &displacement, pSymbol)) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   451
    success = true;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   452
    if (pSymbol->Name[pSymbol->MaxNameLength - 1] != '\0') {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   453
      // Symbol was truncated. Do not attempt to demangle. Instead, zero terminate the
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   454
      // truncated string. We still return success - the truncated string may still
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   455
      // be usable for the caller.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   456
      pSymbol->Name[pSymbol->MaxNameLength - 1] = '\0';
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   457
      do_demangle = false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   458
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   459
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   460
    // Attempt to demangle.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   461
    if (do_demangle && demangle_locked(pSymbol->Name, buf, buflen)) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   462
      // ok.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   463
    } else {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   464
      ::strncpy(buf, pSymbol->Name, buflen - 1);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   465
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   466
    buf[buflen - 1] = '\0';
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   467
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   468
    *offset = (int)displacement;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   469
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   470
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   471
  DEBUG_ONLY(g_buffers.decode_buffer.check();)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   472
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   473
  return success;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   474
}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   475
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   476
static enum {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   477
  state_uninitialized = 0,
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   478
  state_ready = 1,
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   479
  state_error = 2
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   480
} g_state = state_uninitialized;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   481
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   482
static void initialize() {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   483
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   484
  assert(g_state == state_uninitialized, "wrong sequence");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   485
  g_state = state_error;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   486
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   487
  // 1) Initialize buffers.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   488
  g_buffers.initialize();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   489
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   490
  // 1) Call SymInitialize
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   491
  HANDLE hProcess = ::GetCurrentProcess();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   492
  WindowsDbgHelp::symSetOptions(SYMOPT_FAIL_CRITICAL_ERRORS | SYMOPT_DEFERRED_LOADS |
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   493
                        SYMOPT_EXACT_SYMBOLS | SYMOPT_LOAD_LINES);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   494
  if (!WindowsDbgHelp::symInitialize(hProcess, NULL, TRUE)) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   495
    return;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   496
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   497
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   498
  // Note: we ignore any errors from this point on. The symbol engine may be
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   499
  // usable enough.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   500
  g_state = state_ready;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   501
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   502
  (void)recalc_search_path_locked(NULL);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   503
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   504
}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   505
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   506
///////////////////// External functions //////////////////////////
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   507
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   508
// All outside facing functions are synchronized. Also, we run
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   509
// initialization on first touch.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   510
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   511
static CRITICAL_SECTION g_cs;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   512
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   513
namespace { // Do not export.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   514
  class SymbolEngineEntry {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   515
   public:
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   516
    SymbolEngineEntry() {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   517
      ::EnterCriticalSection(&g_cs);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   518
      if (g_state == state_uninitialized) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   519
        initialize();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   520
      }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   521
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   522
    ~SymbolEngineEntry() {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   523
      ::LeaveCriticalSection(&g_cs);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   524
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   525
  };
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   526
}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   527
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   528
// Called at DLL_PROCESS_ATTACH.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   529
void SymbolEngine::pre_initialize() {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   530
  ::InitializeCriticalSection(&g_cs);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   531
}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   532
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   533
bool SymbolEngine::decode(const void* addr, char* buf, int buflen, int* offset, bool do_demangle) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   534
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   535
  assert(buf != NULL && buflen > 0 && offset != NULL, "Argument error");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   536
  buf[0] = '\0';
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   537
  *offset = -1;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   538
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   539
  if (addr == NULL) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   540
    return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   541
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   542
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   543
  SymbolEngineEntry entry_guard;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   544
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   545
  // Try decoding the symbol once. If we fail, attempt to rebuild the
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   546
  // symbol search path - maybe the pc points to a dll whose pdb file is
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   547
  // outside our search path. Then do attempt the decode again.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   548
  bool success = decode_locked(addr, buf, buflen, offset, do_demangle);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   549
  if (!success) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   550
    bool did_update_search_path = false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   551
    if (recalc_search_path_locked(&did_update_search_path)) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   552
      if (did_update_search_path) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   553
        success = decode_locked(addr, buf, buflen, offset, do_demangle);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   554
      }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   555
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   556
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   557
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   558
  return success;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   559
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   560
}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   561
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   562
bool SymbolEngine::demangle(const char* symbol, char *buf, int buflen) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   563
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   564
  SymbolEngineEntry entry_guard;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   565
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   566
  return demangle_locked(symbol, buf, buflen);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   567
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   568
}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   569
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   570
bool SymbolEngine::recalc_search_path(bool* p_search_path_was_updated) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   571
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   572
  SymbolEngineEntry entry_guard;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   573
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   574
  return recalc_search_path_locked(p_search_path_was_updated);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   575
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   576
}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   577
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   578
bool SymbolEngine::get_source_info(const void* addr, char* buf, size_t buflen,
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   579
                                   int* line_no)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   580
{
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   581
  assert(buf != NULL && buflen > 0 && line_no != NULL, "Argument error");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   582
  buf[0] = '\0';
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   583
  *line_no = -1;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   584
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   585
  if (addr == NULL) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   586
    return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   587
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   588
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   589
  SymbolEngineEntry entry_guard;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   590
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   591
  IMAGEHLP_LINE64 lineinfo;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   592
  memset(&lineinfo, 0, sizeof(lineinfo));
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   593
  lineinfo.SizeOfStruct = sizeof(lineinfo);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   594
  DWORD displacement;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   595
  if (WindowsDbgHelp::symGetLineFromAddr64(::GetCurrentProcess(), (DWORD64)addr,
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   596
                                           &displacement, &lineinfo)) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   597
    if (buf != NULL && buflen > 0 && lineinfo.FileName != NULL) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   598
      // We only return the file name, not the whole path.
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   599
      char* p = lineinfo.FileName;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   600
      char* q = strrchr(lineinfo.FileName, '\\');
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   601
      if (q) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   602
        p = q + 1;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   603
      }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   604
      ::strncpy(buf, p, buflen - 1);
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   605
      buf[buflen - 1] = '\0';
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   606
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   607
    if (line_no != 0) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   608
      *line_no = lineinfo.LineNumber;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   609
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   610
    return true;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   611
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   612
  return false;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   613
}
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   614
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   615
// Print one liner describing state (if library loaded, which functions are
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   616
// missing - if any, and the dbhelp API version)
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   617
void SymbolEngine::print_state_on(outputStream* st) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   618
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   619
  SymbolEngineEntry entry_guard;
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   620
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   621
  st->print("symbol engine: ");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   622
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   623
  if (g_state == state_uninitialized) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   624
    st->print("uninitialized.");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   625
  } else if (g_state == state_error) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   626
    st->print("initialization error.");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   627
  } else {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   628
    st->print("initialized successfully");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   629
    st->print(" - sym options: 0x%X", WindowsDbgHelp::symGetOptions());
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   630
    st->print(" - pdb path: ");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   631
    if (WindowsDbgHelp::symGetSearchPath(::GetCurrentProcess(),
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   632
                                          g_buffers.search_path.ptr(),
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   633
                                          (int)g_buffers.search_path.capacity())) {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   634
      st->print_raw(g_buffers.search_path.ptr());
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   635
    } else {
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   636
      st->print_raw("(cannot be retrieved)");
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   637
    }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   638
  }
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   639
  st->cr();
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   640
19219ec3f176 8185712: [windows] Improve native symbol decoder
goetz
parents:
diff changeset
   641
}