1 /* 2 * Copyright (c) 2013, 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_VM_OOPS_METHODCOUNTERS_HPP 26 #define SHARE_VM_OOPS_METHODCOUNTERS_HPP 27 28 #include "oops/metadata.hpp" 29 #include "interpreter/invocationCounter.hpp" 30 31 class MethodCounters: public MetaspaceObj { 32 friend class VMStructs; 33 private: 34 int _interpreter_invocation_count; // Count of times invoked (reused as prev_event_count in tiered) 35 u2 _interpreter_throwout_count; // Count of times method was exited via exception while interpreting 36 u2 _number_of_breakpoints; // fullspeed debugging support 37 InvocationCounter _invocation_counter; // Incremented before each activation of the method - used to trigger frequency-based optimizations 38 InvocationCounter _backedge_counter; // Incremented before each backedge taken - used to trigger frequencey-based optimizations 39 // NMethod age is a counter for warm methods detection in the code cache sweeper. 40 // The counter is reset by the sweeper and is decremented by some of the compiled 41 // code. The counter values are interpreted as follows: 42 // 1. (HotMethodDetection..INT_MAX] - initial value, no counters inserted 43 // 2. (1..HotMethodDetectionLimit) - the method is warm, the counter is used 44 // to figure out which methods can be flushed. 45 // 3. (INT_MIN..0] - method is hot and will deopt and get 46 // recompiled without the counters 47 int _nmethod_age; 48 49 #ifdef TIERED 50 float _rate; // Events (invocation and backedge counter increments) per millisecond 51 jlong _prev_time; // Previous time the rate was acquired 52 u1 _highest_comp_level; // Highest compile level this method has ever seen. 53 u1 _highest_osr_comp_level; // Same for OSR level 54 #endif 55 56 MethodCounters() : _interpreter_invocation_count(0), 57 _interpreter_throwout_count(0), 58 _number_of_breakpoints(0), 59 _nmethod_age(INT_MAX) 60 #ifdef TIERED 61 , _rate(0), 62 _prev_time(0), 63 _highest_comp_level(0), 64 _highest_osr_comp_level(0) 65 #endif 66 { 67 invocation_counter()->init(); 68 backedge_counter()->init(); 69 70 if (StressCodeAging) { 71 set_nmethod_age(HotMethodDetectionLimit); 72 } 73 } 74 75 public: 76 static MethodCounters* allocate(ClassLoaderData* loader_data, TRAPS); 77 78 void deallocate_contents(ClassLoaderData* loader_data) {} 79 DEBUG_ONLY(bool on_stack() { return false; }) // for template 80 81 static int size() { return sizeof(MethodCounters) / wordSize; } 82 83 bool is_klass() const { return false; } 84 85 void clear_counters(); 86 87 int interpreter_invocation_count() { 88 return _interpreter_invocation_count; 89 } 90 void set_interpreter_invocation_count(int count) { 91 _interpreter_invocation_count = count; 92 } 93 int increment_interpreter_invocation_count() { 94 return ++_interpreter_invocation_count; 95 } 96 97 void interpreter_throwout_increment() { 98 if (_interpreter_throwout_count < 65534) { 99 _interpreter_throwout_count++; 100 } 101 } 102 int interpreter_throwout_count() const { 103 return _interpreter_throwout_count; 104 } 105 void set_interpreter_throwout_count(int count) { 106 _interpreter_throwout_count = count; 107 } 108 109 u2 number_of_breakpoints() const { return _number_of_breakpoints; } 110 void incr_number_of_breakpoints() { ++_number_of_breakpoints; } 111 void decr_number_of_breakpoints() { --_number_of_breakpoints; } 112 void clear_number_of_breakpoints() { _number_of_breakpoints = 0; } 113 114 #ifdef TIERED 115 jlong prev_time() const { return _prev_time; } 116 void set_prev_time(jlong time) { _prev_time = time; } 117 float rate() const { return _rate; } 118 void set_rate(float rate) { _rate = rate; } 119 #endif 120 121 int highest_comp_level() const; 122 void set_highest_comp_level(int level); 123 int highest_osr_comp_level() const; 124 void set_highest_osr_comp_level(int level); 125 126 // invocation counter 127 InvocationCounter* invocation_counter() { return &_invocation_counter; } 128 InvocationCounter* backedge_counter() { return &_backedge_counter; } 129 130 int nmethod_age() { 131 return _nmethod_age; 132 } 133 void set_nmethod_age(int age) { 134 _nmethod_age = age; 135 } 136 void reset_nmethod_age() { 137 set_nmethod_age(HotMethodDetectionLimit); 138 } 139 140 static bool is_nmethod_hot(int age) { return age <= 0; } 141 static bool is_nmethod_warm(int age) { return age < HotMethodDetectionLimit; } 142 static bool is_nmethod_age_unset(int age) { return age > HotMethodDetectionLimit; } 143 144 static ByteSize nmethod_age_offset() { 145 return byte_offset_of(MethodCounters, _nmethod_age); 146 } 147 148 static ByteSize interpreter_invocation_counter_offset() { 149 return byte_offset_of(MethodCounters, _interpreter_invocation_count); 150 } 151 152 static ByteSize invocation_counter_offset() { 153 return byte_offset_of(MethodCounters, _invocation_counter); 154 } 155 156 static ByteSize backedge_counter_offset() { 157 return byte_offset_of(MethodCounters, _backedge_counter); 158 } 159 160 static int interpreter_invocation_counter_offset_in_bytes() { 161 return offset_of(MethodCounters, _interpreter_invocation_count); 162 } 163 164 }; 165 #endif //SHARE_VM_OOPS_METHODCOUNTERS_HPP