| 1 | /* | 
|---|
| 2 | * Copyright (c) 2011, 2019, Oracle and/or its affiliates. All rights reserved. | 
|---|
| 3 | * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. | 
|---|
| 4 | * | 
|---|
| 5 | * This code is free software; you can redistribute it and/or modify it | 
|---|
| 6 | * under the terms of the GNU General Public License version 2 only, as | 
|---|
| 7 | * published by the Free Software Foundation. | 
|---|
| 8 | * | 
|---|
| 9 | * This code is distributed in the hope that it will be useful, but WITHOUT | 
|---|
| 10 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | 
|---|
| 11 | * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License | 
|---|
| 12 | * version 2 for more details (a copy is included in the LICENSE file that | 
|---|
| 13 | * accompanied this code). | 
|---|
| 14 | * | 
|---|
| 15 | * You should have received a copy of the GNU General Public License version | 
|---|
| 16 | * 2 along with this work; if not, write to the Free Software Foundation, | 
|---|
| 17 | * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. | 
|---|
| 18 | * | 
|---|
| 19 | * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA | 
|---|
| 20 | * or visit www.oracle.com if you need additional information or have any | 
|---|
| 21 | * questions. | 
|---|
| 22 | * | 
|---|
| 23 | */ | 
|---|
| 24 |  | 
|---|
| 25 | #ifndef SHARE_JFR_RECORDER_STACKTRACE_JFRSTACKTRACEREPOSITORY_HPP | 
|---|
| 26 | #define SHARE_JFR_RECORDER_STACKTRACE_JFRSTACKTRACEREPOSITORY_HPP | 
|---|
| 27 |  | 
|---|
| 28 | #include "jfr/utilities/jfrAllocation.hpp" | 
|---|
| 29 | #include "jfr/utilities/jfrTypes.hpp" | 
|---|
| 30 |  | 
|---|
| 31 | class frame; | 
|---|
| 32 | class JavaThread; | 
|---|
| 33 | class JfrCheckpointWriter; | 
|---|
| 34 | class JfrChunkWriter; | 
|---|
| 35 | class Method; | 
|---|
| 36 |  | 
|---|
| 37 | class JfrStackFrame { | 
|---|
| 38 | private: | 
|---|
| 39 | mutable const Method* _method; | 
|---|
| 40 | traceid _methodid; | 
|---|
| 41 | mutable int _line; | 
|---|
| 42 | int _bci; | 
|---|
| 43 | u1 _type; | 
|---|
| 44 |  | 
|---|
| 45 | public: | 
|---|
| 46 | enum { | 
|---|
| 47 | FRAME_INTERPRETER = 0, | 
|---|
| 48 | FRAME_JIT, | 
|---|
| 49 | FRAME_INLINE, | 
|---|
| 50 | FRAME_NATIVE, | 
|---|
| 51 | NUM_FRAME_TYPES | 
|---|
| 52 | }; | 
|---|
| 53 |  | 
|---|
| 54 | JfrStackFrame(const traceid& id, int bci, int type, const Method* method) : | 
|---|
| 55 | _method(method), _methodid(id), _line(0), _bci(bci), _type(type) {} | 
|---|
| 56 | JfrStackFrame(const traceid& id, int bci, int type, int lineno) : | 
|---|
| 57 | _method(NULL), _methodid(id), _line(lineno), _bci(bci), _type(type) {} | 
|---|
| 58 | bool equals(const JfrStackFrame& rhs) const; | 
|---|
| 59 | void write(JfrChunkWriter& cw) const; | 
|---|
| 60 | void write(JfrCheckpointWriter& cpw) const; | 
|---|
| 61 | void resolve_lineno() const; | 
|---|
| 62 | }; | 
|---|
| 63 |  | 
|---|
| 64 | class JfrStackTrace : public StackObj { | 
|---|
| 65 | friend class JfrStackTraceRepository; | 
|---|
| 66 | private: | 
|---|
| 67 | JfrStackFrame* _frames; | 
|---|
| 68 | traceid _id; | 
|---|
| 69 | u4 _nr_of_frames; | 
|---|
| 70 | unsigned int _hash; | 
|---|
| 71 | const u4 _max_frames; | 
|---|
| 72 | bool _reached_root; | 
|---|
| 73 | mutable bool _lineno; | 
|---|
| 74 |  | 
|---|
| 75 | public: | 
|---|
| 76 | JfrStackTrace(JfrStackFrame* frames, u4 max_frames) : _frames(frames), | 
|---|
| 77 | _id(0), | 
|---|
| 78 | _nr_of_frames(0), | 
|---|
| 79 | _hash(0), | 
|---|
| 80 | _max_frames(max_frames), | 
|---|
| 81 | _reached_root(false), | 
|---|
| 82 | _lineno(false) {} | 
|---|
| 83 | bool record_thread(JavaThread& thread, frame& frame); | 
|---|
| 84 | bool record_safe(JavaThread* thread, int skip, bool leakp = false); | 
|---|
| 85 | void resolve_linenos() const; | 
|---|
| 86 | void set_nr_of_frames(u4 nr_of_frames) { _nr_of_frames = nr_of_frames; } | 
|---|
| 87 | void set_hash(unsigned int hash) { _hash = hash; } | 
|---|
| 88 | unsigned int hash() const { return _hash; } | 
|---|
| 89 | void set_frame(u4 frame_pos, JfrStackFrame& frame); | 
|---|
| 90 | void set_reached_root(bool reached_root) { _reached_root = reached_root; } | 
|---|
| 91 | bool full_stacktrace() const { return _reached_root; } | 
|---|
| 92 | bool have_lineno() const { return _lineno; } | 
|---|
| 93 | }; | 
|---|
| 94 |  | 
|---|
| 95 | class JfrStackTraceRepository : public JfrCHeapObj { | 
|---|
| 96 | friend class JfrRecorder; | 
|---|
| 97 | friend class JfrRecorderService; | 
|---|
| 98 | friend class ObjectSampler; | 
|---|
| 99 | friend class WriteObjectSampleStacktrace; | 
|---|
| 100 |  | 
|---|
| 101 | class StackTrace : public JfrCHeapObj { | 
|---|
| 102 | friend class JfrStackTrace; | 
|---|
| 103 | friend class JfrStackTraceRepository; | 
|---|
| 104 | private: | 
|---|
| 105 | StackTrace* _next; | 
|---|
| 106 | JfrStackFrame* _frames; | 
|---|
| 107 | const traceid _id; | 
|---|
| 108 | u4 _nr_of_frames; | 
|---|
| 109 | unsigned int _hash; | 
|---|
| 110 | bool _reached_root; | 
|---|
| 111 | mutable bool _written; | 
|---|
| 112 |  | 
|---|
| 113 | unsigned int hash() const { return _hash; } | 
|---|
| 114 | bool should_write() const { return !_written; } | 
|---|
| 115 |  | 
|---|
| 116 | public: | 
|---|
| 117 | StackTrace(traceid id, const JfrStackTrace& trace, StackTrace* next); | 
|---|
| 118 | ~StackTrace(); | 
|---|
| 119 | traceid id() const { return _id; } | 
|---|
| 120 | StackTrace* next() const { return _next; } | 
|---|
| 121 | void write(JfrChunkWriter& cw) const; | 
|---|
| 122 | void write(JfrCheckpointWriter& cpw) const; | 
|---|
| 123 | bool equals(const JfrStackTrace& rhs) const; | 
|---|
| 124 | }; | 
|---|
| 125 |  | 
|---|
| 126 | private: | 
|---|
| 127 | static const u4 TABLE_SIZE = 2053; | 
|---|
| 128 | StackTrace* _table[TABLE_SIZE]; | 
|---|
| 129 | traceid _next_id; | 
|---|
| 130 | u4 _entries; | 
|---|
| 131 |  | 
|---|
| 132 | traceid add_trace(const JfrStackTrace& stacktrace); | 
|---|
| 133 | static traceid add(const JfrStackTrace* stacktrace, JavaThread* thread); | 
|---|
| 134 | traceid record_for(JavaThread* thread, int skip, JfrStackFrame* frames, u4 max_frames); | 
|---|
| 135 |  | 
|---|
| 136 | size_t write_impl(JfrChunkWriter& cw, bool clear); | 
|---|
| 137 | const StackTrace* resolve_entry(unsigned int hash, traceid id) const; | 
|---|
| 138 | static void write_metadata(JfrCheckpointWriter& cpw); | 
|---|
| 139 |  | 
|---|
| 140 | static bool fill_stacktrace_for(JavaThread* thread, JfrStackTrace* stacktrace, int skip); | 
|---|
| 141 |  | 
|---|
| 142 | JfrStackTraceRepository(); | 
|---|
| 143 | static JfrStackTraceRepository* create(); | 
|---|
| 144 | bool initialize(); | 
|---|
| 145 | static void destroy(); | 
|---|
| 146 |  | 
|---|
| 147 | static JfrStackTraceRepository& instance(); | 
|---|
| 148 |  | 
|---|
| 149 | public: | 
|---|
| 150 | static traceid add(const JfrStackTrace& stacktrace); | 
|---|
| 151 | static traceid record(Thread* thread, int skip = 0); | 
|---|
| 152 | traceid write(JfrCheckpointWriter& cpw, traceid id, unsigned int hash); | 
|---|
| 153 | size_t write(JfrChunkWriter& cw, bool clear); | 
|---|
| 154 | size_t clear(); | 
|---|
| 155 | }; | 
|---|
| 156 |  | 
|---|
| 157 | #endif // SHARE_JFR_RECORDER_STACKTRACE_JFRSTACKTRACEREPOSITORY_HPP | 
|---|
| 158 |  | 
|---|