src/jdk.hotspot.agent/share/classes/sun/jvm/hotspot/runtime/linux_amd64/LinuxAMD64JavaThreadPDAccess.java
author jgeorge
Thu, 13 Dec 2018 15:11:25 +0530
changeset 53008 9a73a4e4011f
parent 47216 71c04702a3d5
permissions -rw-r--r--
8214226: Incorrect BCI and Line Number with jstack if the top frame is in the interpreter Summary: Read in the bcp from r13 for the top level interpreter frames Reviewed-by: jcbeyler, jgeorge Contributed-by: david.griffiths@gmail.com
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
     1
/*
53008
9a73a4e4011f 8214226: Incorrect BCI and Line Number with jstack if the top frame is in the interpreter
jgeorge
parents: 47216
diff changeset
     2
 * Copyright (c) 2003, 2018, Oracle and/or its affiliates. All rights reserved.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
489c9b5090e2 Initial load
duke
parents:
diff changeset
     4
 *
489c9b5090e2 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
489c9b5090e2 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
489c9b5090e2 Initial load
duke
parents:
diff changeset
     7
 * published by the Free Software Foundation.
489c9b5090e2 Initial load
duke
parents:
diff changeset
     8
 *
489c9b5090e2 Initial load
duke
parents:
diff changeset
     9
 * This code is distributed in the hope that it will be useful, but WITHOUT
489c9b5090e2 Initial load
duke
parents:
diff changeset
    10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
489c9b5090e2 Initial load
duke
parents:
diff changeset
    11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
489c9b5090e2 Initial load
duke
parents:
diff changeset
    12
 * version 2 for more details (a copy is included in the LICENSE file that
489c9b5090e2 Initial load
duke
parents:
diff changeset
    13
 * accompanied this code).
489c9b5090e2 Initial load
duke
parents:
diff changeset
    14
 *
489c9b5090e2 Initial load
duke
parents:
diff changeset
    15
 * You should have received a copy of the GNU General Public License version
489c9b5090e2 Initial load
duke
parents:
diff changeset
    16
 * 2 along with this work; if not, write to the Free Software Foundation,
489c9b5090e2 Initial load
duke
parents:
diff changeset
    17
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
489c9b5090e2 Initial load
duke
parents:
diff changeset
    18
 *
5547
f4b087cbb361 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1
diff changeset
    19
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
f4b087cbb361 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1
diff changeset
    20
 * or visit www.oracle.com if you need additional information or have any
f4b087cbb361 6941466: Oracle rebranding changes for Hotspot repositories
trims
parents: 1
diff changeset
    21
 * questions.
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
    22
 *
489c9b5090e2 Initial load
duke
parents:
diff changeset
    23
 */
489c9b5090e2 Initial load
duke
parents:
diff changeset
    24
489c9b5090e2 Initial load
duke
parents:
diff changeset
    25
package sun.jvm.hotspot.runtime.linux_amd64;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    26
489c9b5090e2 Initial load
duke
parents:
diff changeset
    27
import java.io.*;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    28
import java.util.*;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    29
import sun.jvm.hotspot.debugger.*;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    30
import sun.jvm.hotspot.debugger.amd64.*;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    31
import sun.jvm.hotspot.runtime.*;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    32
import sun.jvm.hotspot.runtime.amd64.*;
10517
f92c9ff3a15f 7051798: SA-JDI: NPE in Frame.addressOfStackSlot(Frame.java:244)
never
parents: 5547
diff changeset
    33
import sun.jvm.hotspot.runtime.x86.*;
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
    34
import sun.jvm.hotspot.types.*;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    35
import sun.jvm.hotspot.utilities.*;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    36
489c9b5090e2 Initial load
duke
parents:
diff changeset
    37
public class LinuxAMD64JavaThreadPDAccess implements JavaThreadPDAccess {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    38
  private static AddressField  lastJavaFPField;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    39
  private static AddressField  osThreadField;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    40
489c9b5090e2 Initial load
duke
parents:
diff changeset
    41
  // Field from OSThread
489c9b5090e2 Initial load
duke
parents:
diff changeset
    42
  private static CIntegerField osThreadThreadIDField;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    43
489c9b5090e2 Initial load
duke
parents:
diff changeset
    44
  // This is currently unneeded but is being kept in case we change
489c9b5090e2 Initial load
duke
parents:
diff changeset
    45
  // the currentFrameGuess algorithm
489c9b5090e2 Initial load
duke
parents:
diff changeset
    46
  private static final long GUESS_SCAN_RANGE = 128 * 1024;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    47
489c9b5090e2 Initial load
duke
parents:
diff changeset
    48
  static {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    49
    VM.registerVMInitializedObserver(new Observer() {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    50
        public void update(Observable o, Object data) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    51
          initialize(VM.getVM().getTypeDataBase());
489c9b5090e2 Initial load
duke
parents:
diff changeset
    52
        }
489c9b5090e2 Initial load
duke
parents:
diff changeset
    53
      });
489c9b5090e2 Initial load
duke
parents:
diff changeset
    54
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
    55
489c9b5090e2 Initial load
duke
parents:
diff changeset
    56
  private static synchronized void initialize(TypeDataBase db) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    57
    Type type = db.lookupType("JavaThread");
489c9b5090e2 Initial load
duke
parents:
diff changeset
    58
    osThreadField           = type.getAddressField("_osthread");
489c9b5090e2 Initial load
duke
parents:
diff changeset
    59
489c9b5090e2 Initial load
duke
parents:
diff changeset
    60
    Type anchorType = db.lookupType("JavaFrameAnchor");
489c9b5090e2 Initial load
duke
parents:
diff changeset
    61
    lastJavaFPField         = anchorType.getAddressField("_last_Java_fp");
489c9b5090e2 Initial load
duke
parents:
diff changeset
    62
489c9b5090e2 Initial load
duke
parents:
diff changeset
    63
    Type osThreadType = db.lookupType("OSThread");
489c9b5090e2 Initial load
duke
parents:
diff changeset
    64
    osThreadThreadIDField   = osThreadType.getCIntegerField("_thread_id");
489c9b5090e2 Initial load
duke
parents:
diff changeset
    65
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
    66
489c9b5090e2 Initial load
duke
parents:
diff changeset
    67
  public    Address getLastJavaFP(Address addr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    68
    return lastJavaFPField.getValue(addr.addOffsetTo(sun.jvm.hotspot.runtime.JavaThread.getAnchorField().getOffset()));
489c9b5090e2 Initial load
duke
parents:
diff changeset
    69
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
    70
489c9b5090e2 Initial load
duke
parents:
diff changeset
    71
  public    Address getLastJavaPC(Address addr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    72
    return null;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    73
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
    74
489c9b5090e2 Initial load
duke
parents:
diff changeset
    75
  public    Address getBaseOfStackPointer(Address addr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    76
    return null;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    77
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
    78
489c9b5090e2 Initial load
duke
parents:
diff changeset
    79
  public    Frame getLastFramePD(JavaThread thread, Address addr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    80
    Address fp = thread.getLastJavaFP();
489c9b5090e2 Initial load
duke
parents:
diff changeset
    81
    if (fp == null) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    82
      return null; // no information
489c9b5090e2 Initial load
duke
parents:
diff changeset
    83
    }
10517
f92c9ff3a15f 7051798: SA-JDI: NPE in Frame.addressOfStackSlot(Frame.java:244)
never
parents: 5547
diff changeset
    84
    return new X86Frame(thread.getLastJavaSP(), fp);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
    85
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
    86
489c9b5090e2 Initial load
duke
parents:
diff changeset
    87
  public    RegisterMap newRegisterMap(JavaThread thread, boolean updateMap) {
10517
f92c9ff3a15f 7051798: SA-JDI: NPE in Frame.addressOfStackSlot(Frame.java:244)
never
parents: 5547
diff changeset
    88
    return new X86RegisterMap(thread, updateMap);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
    89
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
    90
489c9b5090e2 Initial load
duke
parents:
diff changeset
    91
  public    Frame getCurrentFrameGuess(JavaThread thread, Address addr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    92
    ThreadProxy t = getThreadProxy(addr);
489c9b5090e2 Initial load
duke
parents:
diff changeset
    93
    AMD64ThreadContext context = (AMD64ThreadContext) t.getContext();
489c9b5090e2 Initial load
duke
parents:
diff changeset
    94
    AMD64CurrentFrameGuess guesser = new AMD64CurrentFrameGuess(context, thread);
489c9b5090e2 Initial load
duke
parents:
diff changeset
    95
    if (!guesser.run(GUESS_SCAN_RANGE)) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
    96
      return null;
489c9b5090e2 Initial load
duke
parents:
diff changeset
    97
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
    98
    if (guesser.getPC() == null) {
10517
f92c9ff3a15f 7051798: SA-JDI: NPE in Frame.addressOfStackSlot(Frame.java:244)
never
parents: 5547
diff changeset
    99
      return new X86Frame(guesser.getSP(), guesser.getFP());
53008
9a73a4e4011f 8214226: Incorrect BCI and Line Number with jstack if the top frame is in the interpreter
jgeorge
parents: 47216
diff changeset
   100
    } else if (VM.getVM().getInterpreter().contains(guesser.getPC())) {
9a73a4e4011f 8214226: Incorrect BCI and Line Number with jstack if the top frame is in the interpreter
jgeorge
parents: 47216
diff changeset
   101
      // pass the value of R13 which contains the bcp for the top level frame
9a73a4e4011f 8214226: Incorrect BCI and Line Number with jstack if the top frame is in the interpreter
jgeorge
parents: 47216
diff changeset
   102
      Address bcp = context.getRegisterAsAddress(AMD64ThreadContext.R13);
9a73a4e4011f 8214226: Incorrect BCI and Line Number with jstack if the top frame is in the interpreter
jgeorge
parents: 47216
diff changeset
   103
      return new X86Frame(guesser.getSP(), guesser.getFP(), guesser.getPC(), null, bcp);
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   104
    } else {
10517
f92c9ff3a15f 7051798: SA-JDI: NPE in Frame.addressOfStackSlot(Frame.java:244)
never
parents: 5547
diff changeset
   105
      return new X86Frame(guesser.getSP(), guesser.getFP(), guesser.getPC());
1
489c9b5090e2 Initial load
duke
parents:
diff changeset
   106
    }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   107
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   108
489c9b5090e2 Initial load
duke
parents:
diff changeset
   109
  public    void printThreadIDOn(Address addr, PrintStream tty) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   110
    tty.print(getThreadProxy(addr));
489c9b5090e2 Initial load
duke
parents:
diff changeset
   111
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   112
489c9b5090e2 Initial load
duke
parents:
diff changeset
   113
  public    void printInfoOn(Address threadAddr, PrintStream tty) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   114
    tty.print("Thread id: ");
489c9b5090e2 Initial load
duke
parents:
diff changeset
   115
    printThreadIDOn(threadAddr, tty);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   116
//    tty.println("\nPostJavaState: " + getPostJavaState(threadAddr));
489c9b5090e2 Initial load
duke
parents:
diff changeset
   117
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   118
489c9b5090e2 Initial load
duke
parents:
diff changeset
   119
  public    Address getLastSP(Address addr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   120
    ThreadProxy t = getThreadProxy(addr);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   121
    AMD64ThreadContext context = (AMD64ThreadContext) t.getContext();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   122
    return context.getRegisterAsAddress(AMD64ThreadContext.RSP);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   123
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   124
489c9b5090e2 Initial load
duke
parents:
diff changeset
   125
  public    ThreadProxy getThreadProxy(Address addr) {
489c9b5090e2 Initial load
duke
parents:
diff changeset
   126
    // Addr is the address of the JavaThread.
489c9b5090e2 Initial load
duke
parents:
diff changeset
   127
    // Fetch the OSThread (for now and for simplicity, not making a
489c9b5090e2 Initial load
duke
parents:
diff changeset
   128
    // separate "OSThread" class in this package)
489c9b5090e2 Initial load
duke
parents:
diff changeset
   129
    Address osThreadAddr = osThreadField.getValue(addr);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   130
    // Get the address of the _thread_id from the OSThread
489c9b5090e2 Initial load
duke
parents:
diff changeset
   131
    Address threadIdAddr = osThreadAddr.addOffsetTo(osThreadThreadIDField.getOffset());
489c9b5090e2 Initial load
duke
parents:
diff changeset
   132
489c9b5090e2 Initial load
duke
parents:
diff changeset
   133
    JVMDebugger debugger = VM.getVM().getDebugger();
489c9b5090e2 Initial load
duke
parents:
diff changeset
   134
    return debugger.getThreadForIdentifierAddress(threadIdAddr);
489c9b5090e2 Initial load
duke
parents:
diff changeset
   135
  }
489c9b5090e2 Initial load
duke
parents:
diff changeset
   136
}