java-stack.h 4.51 KB
Newer Older
1 2
// java-stack.h - Definitions for unwinding & inspecting the call stack.

3
/* Copyright (C) 2005, 2006  Free Software Foundation
4 5 6 7 8 9 10 11 12 13

   This file is part of libgcj.

This software is copyrighted work licensed under the terms of the
Libgcj License.  Please consult the file "LIBGCJ_LICENSE" for
details.  */

#ifndef __JV_STACKTRACE_H__
#define __JV_STACKTRACE_H__

14
#include <stdlib.h>
15 16 17 18 19 20 21 22 23 24
#include <unwind.h>

#include <gcj/cni.h>
#include <gcj/javaprims.h>

#include <java-interp.h>

#include <java/lang/Class.h>
#include <java/lang/StackTraceElement.h>
#include <java/lang/Throwable.h>
25
#include <java/lang/Thread.h>
26
#include <java/util/IdentityHashMap.h>
27 28 29 30

#include <gnu/gcj/runtime/NameFinder.h>

using namespace gnu::gcj::runtime;
31
using namespace java::lang;
32

33 34 35 36 37 38 39 40 41 42 43
extern "Java"
{
  namespace gnu
  {
    namespace classpath
    {
        class VMStackWalker;
    }
  }
}

44 45 46 47 48 49 50 51 52 53 54 55 56 57 58
#ifdef INTERPRETER
struct _Jv_InterpFrameInfo
{
  _Jv_InterpMethod *meth;
  pc_t pc;
};
#endif

struct _Jv_StackFrame
{
  _Jv_FrameType type;   /* Native or interpreted.  */
  union {
#ifdef INTERPRETER
    _Jv_InterpFrameInfo interp;
#endif
59
    struct {
60 61 62 63
      jclass proxyClass;
      _Jv_Method *proxyMethod;
    };
    struct {
64 65 66
      void *ip;
      void *start_ip;
    };
67 68 69 70 71
  };
  jclass klass;
  _Jv_Method *meth;
};

72
struct _Jv_UnwindState;
73 74 75 76 77 78 79
typedef _Unwind_Reason_Code (*_Jv_TraceFn) (_Jv_UnwindState *);

struct _Jv_UnwindState
{
  jint length;                   // length of FRAMES
  jint pos;                      // current position in FRAMES
  _Jv_StackFrame *frames;        // array of stack frame data to be filled.
80
#ifdef INTERPRETER
81
  _Jv_InterpFrame *interp_frame; // current frame in the interpreter stack.
82
#endif
83 84 85 86 87 88 89 90 91
  _Jv_TraceFn trace_function;    // function to call back after each frame
  				 // is enumerated. May be NULL.
  void *trace_data;		 // additional state data for trace_function.
  
  _Jv_UnwindState (jint ln)
    {
      length = ln;
      pos = 0;
      frames = NULL;
92
#ifdef INTERPRETER
93 94 95 96 97
      Thread *thread = Thread::currentThread();
      // Check for NULL currentThread(), in case an exception is created 
      // very early during the runtime startup.
      if (thread)
	interp_frame = (_Jv_InterpFrame *) thread->interp_frame;
98 99
      else
	interp_frame = NULL;
100
#endif
101 102 103 104 105
      trace_function = NULL;
      trace_data = NULL;
    }
};

106 107 108 109 110 111
class _Jv_StackTrace
{
private:
  int length;
  _Jv_StackFrame frames[];

112
  static java::util::IdentityHashMap *ncodeMap;
113
  static void UpdateNCodeMap ();
114
  static jclass ClassForFrame (_Jv_StackFrame *frame);
115 116
  static void FillInFrameInfo (_Jv_StackFrame *frame);
  static void getLineNumberForFrame(_Jv_StackFrame *frame, NameFinder *finder, 
117 118
				    jstring *sourceFileName, jint *lineNum,
				    jstring *methodName);
119 120 121
  
  static _Unwind_Reason_Code UnwindTraceFn (struct _Unwind_Context *context, 
    void *state_ptr);
122 123 124
    
  static _Unwind_Reason_Code calling_class_trace_fn (_Jv_UnwindState *state);
  static _Unwind_Reason_Code non_system_trace_fn (_Jv_UnwindState *state);
125
  static _Unwind_Reason_Code accesscontrol_trace_fn (_Jv_UnwindState *state);
126 127
  static _Unwind_Reason_Code stackwalker_trace_fn (_Jv_UnwindState *state);
  static _Unwind_Reason_Code stackwalker_nnl_trace_fn (_Jv_UnwindState *state);
128 129 130 131 132 133

public:
  static _Jv_StackTrace *GetStackTrace (void);
  static JArray< ::java::lang::StackTraceElement *>*
    GetStackTraceElements (_Jv_StackTrace *trace, 
    java::lang::Throwable *throwable);
134 135 136
  static jclass GetCallingClass (jclass);
  static void GetCallerInfo (jclass checkClass, jclass *, _Jv_Method **);
  static ClassLoader *GetFirstNonSystemClassLoader (void);
137
  static jobjectArray GetAccessControlStack ();
138 139 140
  static JArray<jclass> *GetStackWalkerStack ();
  static jclass GetStackWalkerCallingClass ();
  static ClassLoader *GetStackWalkerFirstNonNullLoader ();
141 142

  friend jclass _Jv_GetMethodDeclaringClass (jmethodID);
143
  friend class gnu::classpath::VMStackWalker;
144 145
};

146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175
// Information about a given address.
struct _Jv_AddrInfo
{
  // File name of the defining module.
  const char *file_name;

  // Base address of the loaded module.
  void *base;

  // Name of the nearest symbol.
  const char *sym_name;

  // Address of the nearest symbol.
  void *sym_addr;

  ~_Jv_AddrInfo (void)
    {
      // On systems with a real dladdr(), the file and symbol names given by
      // _Jv_platform_dladdr() are not dynamically allocated.  On Windows,
      // they are.

#ifdef WIN32
      if (file_name)
        free ((void *)file_name);

      if (sym_name)
        free ((void *)sym_name);
#endif /* WIN32 */
    }
};
176

177
#endif /* __JV_STACKTRACE_H__ */