decoder_windows.cpp 9.6 KB
Newer Older
1
/*
2
 * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved.
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 *
 */

#include "precompiled.hpp"
#include "prims/jvm.h"
27
#include "runtime/arguments.hpp"
Z
zgu 已提交
28
#include "decoder_windows.hpp"
29

Z
zgu 已提交
30 31 32 33 34
WindowsDecoder::WindowsDecoder() {
  _dbghelp_handle = NULL;
  _can_decode_in_vm = false;
  _pfnSymGetSymFromAddr64 = NULL;
  _pfnUndecorateSymbolName = NULL;
35 36 37 38 39
#ifdef AMD64
  _pfnStackWalk64 = NULL;
  _pfnSymFunctionTableAccess64 = NULL;
  _pfnSymGetModuleBase64 = NULL;
#endif
Z
zgu 已提交
40 41 42
  _decoder_status = no_error;
  initialize();
}
43

Z
zgu 已提交
44 45 46
void WindowsDecoder::initialize() {
  if (!has_error() && _dbghelp_handle == NULL) {
    HMODULE handle = ::LoadLibrary("dbghelp.dll");
47 48
    if (!handle) {
      _decoder_status = helper_not_found;
Z
zgu 已提交
49
      return;
50 51 52 53 54 55 56
    }

    _dbghelp_handle = handle;

    pfn_SymSetOptions _pfnSymSetOptions = (pfn_SymSetOptions)::GetProcAddress(handle, "SymSetOptions");
    pfn_SymInitialize _pfnSymInitialize = (pfn_SymInitialize)::GetProcAddress(handle, "SymInitialize");
    _pfnSymGetSymFromAddr64 = (pfn_SymGetSymFromAddr64)::GetProcAddress(handle, "SymGetSymFromAddr64");
57
    _pfnUndecorateSymbolName = (pfn_UndecorateSymbolName)::GetProcAddress(handle, "UnDecorateSymbolName");
58 59

    if (_pfnSymSetOptions == NULL || _pfnSymInitialize == NULL || _pfnSymGetSymFromAddr64 == NULL) {
60
      uninitialize();
61 62 63 64
      _decoder_status = helper_func_error;
      return;
    }

65 66 67 68 69 70 71 72 73 74 75 76 77
#ifdef AMD64
    _pfnStackWalk64 = (pfn_StackWalk64)::GetProcAddress(handle, "StackWalk64");
    _pfnSymFunctionTableAccess64 = (pfn_SymFunctionTableAccess64)::GetProcAddress(handle, "SymFunctionTableAccess64");
    _pfnSymGetModuleBase64 = (pfn_SymGetModuleBase64)::GetProcAddress(handle, "SymGetModuleBase64");
    if (_pfnStackWalk64 == NULL || _pfnSymFunctionTableAccess64 == NULL || _pfnSymGetModuleBase64 == NULL) {
      // We can't call StackWalk64 to walk the stack, but we are still
      // able to decode the symbols. Let's limp on.
      _pfnStackWalk64 = NULL;
      _pfnSymFunctionTableAccess64 = NULL;
      _pfnSymGetModuleBase64 = NULL;
    }
#endif

78 79 80
    HANDLE hProcess = ::GetCurrentProcess();
    _pfnSymSetOptions(SYMOPT_UNDNAME | SYMOPT_DEFERRED_LOADS | SYMOPT_EXACT_SYMBOLS);
    if (!_pfnSymInitialize(hProcess, NULL, TRUE)) {
81 82 83 84 85 86 87 88
      _pfnSymGetSymFromAddr64 = NULL;
      _pfnUndecorateSymbolName = NULL;
      ::FreeLibrary(handle);
      _dbghelp_handle = NULL;
      _decoder_status = helper_init_error;
      return;
    }

89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159
    // set pdb search paths
    pfn_SymSetSearchPath  _pfn_SymSetSearchPath =
      (pfn_SymSetSearchPath)::GetProcAddress(handle, "SymSetSearchPath");
    pfn_SymGetSearchPath  _pfn_SymGetSearchPath =
      (pfn_SymGetSearchPath)::GetProcAddress(handle, "SymGetSearchPath");
    if (_pfn_SymSetSearchPath != NULL && _pfn_SymGetSearchPath != NULL) {
      char paths[MAX_PATH];
      int  len = sizeof(paths);
      if (!_pfn_SymGetSearchPath(hProcess, paths, len)) {
        paths[0] = '\0';
      } else {
        // available spaces in path buffer
        len -= (int)strlen(paths);
      }

      char tmp_path[MAX_PATH];
      DWORD dwSize;
      HMODULE hJVM = ::GetModuleHandle("jvm.dll");
      tmp_path[0] = '\0';
      // append the path where jvm.dll is located
      if (hJVM != NULL && (dwSize = ::GetModuleFileName(hJVM, tmp_path, sizeof(tmp_path))) > 0) {
        while (dwSize > 0 && tmp_path[dwSize] != '\\') {
          dwSize --;
        }

        tmp_path[dwSize] = '\0';

        if (dwSize > 0 && len > (int)dwSize + 1) {
          strncat(paths, os::path_separator(), 1);
          strncat(paths, tmp_path, dwSize);
          len -= dwSize + 1;
        }
      }

      // append $JRE/bin. Arguments::get_java_home actually returns $JRE
      // path
      char *p = Arguments::get_java_home();
      assert(p != NULL, "empty java home");
      size_t java_home_len = strlen(p);
      if (len > (int)java_home_len + 5) {
        strncat(paths, os::path_separator(), 1);
        strncat(paths, p, java_home_len);
        strncat(paths, "\\bin", 4);
        len -= (int)(java_home_len + 5);
      }

      // append $JDK/bin path if it exists
      assert(java_home_len < MAX_PATH, "Invalid path length");
      // assume $JRE is under $JDK, construct $JDK/bin path and
      // see if it exists or not
      if (strncmp(&p[java_home_len - 3], "jre", 3) == 0) {
        strncpy(tmp_path, p, java_home_len - 3);
        tmp_path[java_home_len - 3] = '\0';
        strncat(tmp_path, "bin", 3);

        // if the directory exists
        DWORD dwAttrib = GetFileAttributes(tmp_path);
        if (dwAttrib != INVALID_FILE_ATTRIBUTES &&
            (dwAttrib & FILE_ATTRIBUTE_DIRECTORY)) {
          // tmp_path should have the same length as java_home_len, since we only
          // replaced 'jre' with 'bin'
          if (len > (int)java_home_len + 1) {
            strncat(paths, os::path_separator(), 1);
            strncat(paths, tmp_path, java_home_len);
          }
        }
      }

      _pfn_SymSetSearchPath(hProcess, paths);
    }

160 161
     // find out if jvm.dll contains private symbols, by decoding
     // current function and comparing the result
162
     address addr = (address)Decoder::demangle;
163
     char buf[MAX_PATH];
Z
zgu 已提交
164
     if (decode(addr, buf, sizeof(buf), NULL)) {
165
       _can_decode_in_vm = !strcmp(buf, "Decoder::demangle");
166 167 168 169
     }
  }
}

Z
zgu 已提交
170
void WindowsDecoder::uninitialize() {
171 172
  _pfnSymGetSymFromAddr64 = NULL;
  _pfnUndecorateSymbolName = NULL;
173 174 175 176 177
#ifdef AMD64
  _pfnStackWalk64 = NULL;
  _pfnSymFunctionTableAccess64 = NULL;
  _pfnSymGetModuleBase64 = NULL;
#endif
178 179 180
  if (_dbghelp_handle != NULL) {
    ::FreeLibrary(_dbghelp_handle);
  }
Z
zgu 已提交
181
  _dbghelp_handle = NULL;
182 183
}

Z
zgu 已提交
184 185
bool WindowsDecoder::can_decode_C_frame_in_vm() const {
  return  (!has_error() && _can_decode_in_vm);
186 187 188
}


Z
zgu 已提交
189
bool WindowsDecoder::decode(address addr, char *buf, int buflen, int* offset, const char* modulepath)  {
190 191 192 193 194 195 196 197 198
  if (_pfnSymGetSymFromAddr64 != NULL) {
    PIMAGEHLP_SYMBOL64 pSymbol;
    char symbolInfo[MAX_PATH + sizeof(IMAGEHLP_SYMBOL64)];
    pSymbol = (PIMAGEHLP_SYMBOL64)symbolInfo;
    pSymbol->MaxNameLength = MAX_PATH;
    pSymbol->SizeOfStruct = sizeof(IMAGEHLP_SYMBOL64);
    DWORD64 displacement;
    if (_pfnSymGetSymFromAddr64(::GetCurrentProcess(), (DWORD64)addr, &displacement, pSymbol)) {
      if (buf != NULL) {
Z
zgu 已提交
199
        if (demangle(pSymbol->Name, buf, buflen)) {
200 201 202
          jio_snprintf(buf, buflen, "%s", pSymbol->Name);
        }
      }
Z
zgu 已提交
203 204
      if(offset != NULL) *offset = (int)displacement;
      return true;
205 206
    }
  }
Z
zgu 已提交
207 208 209
  if (buf != NULL && buflen > 0) buf[0] = '\0';
  if (offset != NULL) *offset = -1;
  return false;
210 211
}

Z
zgu 已提交
212
bool WindowsDecoder::demangle(const char* symbol, char *buf, int buflen) {
213 214 215 216
  return _pfnUndecorateSymbolName != NULL &&
         _pfnUndecorateSymbolName(symbol, buf, buflen, UNDNAME_COMPLETE);
}

217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278
#ifdef AMD64
BOOL WindowsDbgHelp::StackWalk64(DWORD MachineType,
                                 HANDLE hProcess,
                                 HANDLE hThread,
                                 LPSTACKFRAME64 StackFrame,
                                 PVOID ContextRecord,
                                 PREAD_PROCESS_MEMORY_ROUTINE64 ReadMemoryRoutine,
                                 PFUNCTION_TABLE_ACCESS_ROUTINE64 FunctionTableAccessRoutine,
                                 PGET_MODULE_BASE_ROUTINE64 GetModuleBaseRoutine,
                                 PTRANSLATE_ADDRESS_ROUTINE64 TranslateAddress) {
  DecoderLocker locker;
  WindowsDecoder* wd = (WindowsDecoder*)locker.decoder();

  if (!wd->has_error() && wd->_pfnStackWalk64) {
    return wd->_pfnStackWalk64(MachineType,
                               hProcess,
                               hThread,
                               StackFrame,
                               ContextRecord,
                               ReadMemoryRoutine,
                               FunctionTableAccessRoutine,
                               GetModuleBaseRoutine,
                               TranslateAddress);
  } else {
    return false;
  }
}

PVOID WindowsDbgHelp::SymFunctionTableAccess64(HANDLE hProcess, DWORD64 AddrBase) {
  DecoderLocker locker;
  WindowsDecoder* wd = (WindowsDecoder*)locker.decoder();

  if (!wd->has_error() && wd->_pfnSymFunctionTableAccess64) {
    return wd->_pfnSymFunctionTableAccess64(hProcess, AddrBase);
  } else {
    return NULL;
  }
}

pfn_SymFunctionTableAccess64 WindowsDbgHelp::pfnSymFunctionTableAccess64() {
  DecoderLocker locker;
  WindowsDecoder* wd = (WindowsDecoder*)locker.decoder();

  if (!wd->has_error()) {
    return wd->_pfnSymFunctionTableAccess64;
  } else {
    return NULL;
  }
}

pfn_SymGetModuleBase64 WindowsDbgHelp::pfnSymGetModuleBase64() {
  DecoderLocker locker;
  WindowsDecoder* wd = (WindowsDecoder*)locker.decoder();

  if (!wd->has_error()) {
    return wd->_pfnSymGetModuleBase64;
  } else {
    return NULL;
  }
}

#endif // AMD64