1 /* 2 * Copyright (c) 2018, 2019, Red Hat, Inc. All rights reserved. 3 * 4 * This code is free software; you can redistribute it and/or modify it 5 * under the terms of the GNU General Public License version 2 only, as 6 * published by the Free Software Foundation. 7 * 8 * This code is distributed in the hope that it will be useful, but WITHOUT 9 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 10 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 11 * version 2 for more details (a copy is included in the LICENSE file that 12 * accompanied this code). 13 * 14 * You should have received a copy of the GNU General Public License version 15 * 2 along with this work; if not, write to the Free Software Foundation, 16 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 17 * 18 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 19 * or visit www.oracle.com if you need additional information or have any 20 * questions. 21 * 22 */ 23 24 #ifndef SHARE_GC_SHENANDOAH_SHENANDOAHASSERTS_HPP 25 #define SHARE_GC_SHENANDOAH_SHENANDOAHASSERTS_HPP 26 27 #include "memory/iterator.hpp" 28 #include "runtime/mutex.hpp" 29 #include "utilities/formatBuffer.hpp" 30 31 typedef FormatBuffer<8192> ShenandoahMessageBuffer; 32 33 class ShenandoahAsserts { 34 public: 35 enum SafeLevel { 36 _safe_unknown, 37 _safe_oop, 38 _safe_oop_fwd, 39 _safe_all 40 }; 41 42 static void print_obj(ShenandoahMessageBuffer &msg, oop obj); 43 44 static void print_non_obj(ShenandoahMessageBuffer &msg, void *loc); 45 46 static void print_obj_safe(ShenandoahMessageBuffer &msg, void *loc); 47 48 static void print_failure(SafeLevel level, oop obj, void *interior_loc, oop loc, 49 const char *phase, const char *label, 50 const char *file, int line); 51 52 static void print_rp_failure(const char *label, BoolObjectClosure* actual, 53 const char *file, int line); 54 55 static void assert_in_heap(void* interior_loc, oop obj, const char* file, int line); 56 static void assert_in_correct_region(void* interior_loc, oop obj, const char* file, int line); 57 58 static void assert_correct(void* interior_loc, oop obj, const char* file, int line); 59 static void assert_forwarded(void* interior_loc, oop obj, const char* file, int line); 60 static void assert_not_forwarded(void* interior_loc, oop obj, const char* file, int line); 61 static void assert_marked(void* interior_loc, oop obj, const char* file, int line); 62 static void assert_in_cset(void* interior_loc, oop obj, const char* file, int line); 63 static void assert_not_in_cset(void* interior_loc, oop obj, const char* file, int line); 64 static void assert_not_in_cset_loc(void* interior_loc, const char* file, int line); 65 66 static void assert_rp_isalive_not_installed(const char *file, int line); 67 static void assert_rp_isalive_installed(const char *file, int line); 68 69 static void assert_locked_or_shenandoah_safepoint(Mutex* lock, const char* file, int line); 70 71 #ifdef ASSERT 72 // Shenandoah uses oop's marked pattern to indicate forwarding. Unfortunately, ObjectMarker (jvmtiTagMap.cpp) 73 // also uses this pattern to indicate visited. 74 // In ObjectMarker's destructor, it invokes heap iteration to reset the marked bits to their original values. 75 // That trips over many assertions on heap iteration code paths, but there are invariants: 76 // 1. Marked pattern in collection set indicates forwarding 77 // 2. Marked pattern off collection set indicates visited by ObjectMarker (because oops seen by 78 // ObjectMarker were LRB'd) 79 // 3. No off collection set marked pattern at any shenandoah safepoint. In fact, no off collection set 80 // marked pattern at any safepoints other than VM_HeapWalkOperation safepoints. 81 // This is an important invariant, since traversal degenerated GC clears collection set before 82 // entering degenerated GC cycle. 83 // So, we will need temporary disable assertions for oops with marked pattern outside collection set. 84 // Note: We only enforce this check for debug assertion to preserve full capacities of verifier, 85 // because verifier always runs at safepoints. 86 static bool can_verify_oop(oop obj); 87 88 #define shenandoah_assert_in_heap(interior_loc, obj) \ 89 if (ShenandoahAsserts::can_verify_oop(obj)) \ 90 ShenandoahAsserts::assert_in_heap(interior_loc, obj, __FILE__, __LINE__); 91 #define shenandoah_assert_in_correct_region(interior_loc, obj) \ 92 if (ShenandoahAsserts::can_verify_oop(obj)) \ 93 ShenandoahAsserts::assert_in_correct_region(interior_loc, obj, __FILE__, __LINE__); 94 95 #define shenandoah_assert_correct_if(interior_loc, obj, condition) \ 96 if (condition && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_correct(interior_loc, obj, __FILE__, __LINE__); 97 #define shenandoah_assert_correct_except(interior_loc, obj, exception) \ 98 if (!(exception) && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_correct(interior_loc, obj, __FILE__, __LINE__); 99 #define shenandoah_assert_correct(interior_loc, obj) \ 100 if (ShenandoahAsserts::can_verify_oop(obj)) \ 101 ShenandoahAsserts::assert_correct(interior_loc, obj, __FILE__, __LINE__); 102 103 #define shenandoah_assert_forwarded_if(interior_loc, obj, condition) \ 104 if (condition && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_forwarded(interior_loc, obj, __FILE__, __LINE__); 105 #define shenandoah_assert_forwarded_except(interior_loc, obj, exception) \ 106 if (!(exception) && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_forwarded(interior_loc, obj, __FILE__, __LINE__); 107 #define shenandoah_assert_forwarded(interior_loc, obj) \ 108 if (ShenandoahAsserts::can_verify_oop(obj)) \ 109 ShenandoahAsserts::assert_forwarded(interior_loc, obj, __FILE__, __LINE__); 110 111 #define shenandoah_assert_not_forwarded_if(interior_loc, obj, condition) \ 112 if (condition && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_not_forwarded(interior_loc, obj, __FILE__, __LINE__); 113 #define shenandoah_assert_not_forwarded_except(interior_loc, obj, exception) \ 114 if (!(exception) && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_not_forwarded(interior_loc, obj, __FILE__, __LINE__); 115 #define shenandoah_assert_not_forwarded(interior_loc, obj) \ 116 if (ShenandoahAsserts::can_verify_oop(obj)) \ 117 ShenandoahAsserts::assert_not_forwarded(interior_loc, obj, __FILE__, __LINE__); 118 119 #define shenandoah_assert_marked_if(interior_loc, obj, condition) \ 120 if (condition && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_marked(interior_loc, obj, __FILE__, __LINE__); 121 #define shenandoah_assert_marked_except(interior_loc, obj, exception) \ 122 if (!(exception) && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_marked(interior_loc, obj, __FILE__, __LINE__); 123 #define shenandoah_assert_marked(interior_loc, obj) \ 124 if (ShenandoahAsserts::can_verify_oop(obj)) \ 125 ShenandoahAsserts::assert_marked(interior_loc, obj, __FILE__, __LINE__); 126 127 #define shenandoah_assert_in_cset_if(interior_loc, obj, condition) \ 128 if (condition && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_in_cset(interior_loc, obj, __FILE__, __LINE__); 129 #define shenandoah_assert_in_cset_except(interior_loc, obj, exception) \ 130 if (!(exception) && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_in_cset(interior_loc, obj, __FILE__, __LINE__); 131 #define shenandoah_assert_in_cset(interior_loc, obj) \ 132 if (ShenandoahAsserts::can_verify_oop(obj)) \ 133 ShenandoahAsserts::assert_in_cset(interior_loc, obj, __FILE__, __LINE__); 134 135 #define shenandoah_assert_not_in_cset_if(interior_loc, obj, condition) \ 136 if (condition && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_not_in_cset(interior_loc, obj, __FILE__, __LINE__); 137 #define shenandoah_assert_not_in_cset_except(interior_loc, obj, exception) \ 138 if (!(exception) && ShenandoahAsserts::can_verify_oop(obj)) ShenandoahAsserts::assert_not_in_cset(interior_loc, obj, __FILE__, __LINE__); 139 #define shenandoah_assert_not_in_cset(interior_loc, obj) \ 140 if (ShenandoahAsserts::can_verify_oop(obj)) \ 141 ShenandoahAsserts::assert_not_in_cset(interior_loc, obj, __FILE__, __LINE__); 142 143 #define shenandoah_assert_not_in_cset_loc_if(interior_loc, condition) \ 144 if (condition) ShenandoahAsserts::assert_not_in_cset_loc(interior_loc, __FILE__, __LINE__); 145 #define shenandoah_assert_not_in_cset_loc_except(interior_loc, exception) \ 146 if (!(exception)) ShenandoahAsserts::assert_not_in_cset_loc(interior_loc, __FILE__, __LINE__); 147 #define shenandoah_assert_not_in_cset_loc(interior_loc) \ 148 ShenandoahAsserts::assert_not_in_cset_loc(interior_loc, __FILE__, __LINE__); 149 150 #define shenandoah_assert_rp_isalive_installed() \ 151 ShenandoahAsserts::assert_rp_isalive_installed(__FILE__, __LINE__); 152 #define shenandoah_assert_rp_isalive_not_installed() \ 153 ShenandoahAsserts::assert_rp_isalive_not_installed(__FILE__, __LINE__); 154 155 #define shenandoah_assert_safepoint() \ 156 assert(ShenandoahSafepoint::is_at_shenandoah_safepoint(), "Should be at Shenandoah Safepoints"); 157 158 #define shenandoah_assert_locked_or_safepoint(lock) \ 159 ShenandoahAsserts::assert_locked_or_shenandoah_safepoint(lock, __FILE__, __LINE__); 160 #else 161 #define shenandoah_assert_in_heap(interior_loc, obj) 162 #define shenandoah_assert_in_correct_region(interior_loc, obj) 163 164 #define shenandoah_assert_correct_if(interior_loc, obj, condition) 165 #define shenandoah_assert_correct_except(interior_loc, obj, exception) 166 #define shenandoah_assert_correct(interior_loc, obj) 167 168 #define shenandoah_assert_forwarded_if(interior_loc, obj, condition) 169 #define shenandoah_assert_forwarded_except(interior_loc, obj, exception) 170 #define shenandoah_assert_forwarded(interior_loc, obj) 171 172 #define shenandoah_assert_not_forwarded_if(interior_loc, obj, condition) 173 #define shenandoah_assert_not_forwarded_except(interior_loc, obj, exception) 174 #define shenandoah_assert_not_forwarded(interior_loc, obj) 175 176 #define shenandoah_assert_marked_if(interior_loc, obj, condition) 177 #define shenandoah_assert_marked_except(interior_loc, obj, exception) 178 #define shenandoah_assert_marked(interior_loc, obj) 179 180 #define shenandoah_assert_in_cset_if(interior_loc, obj, condition) 181 #define shenandoah_assert_in_cset_except(interior_loc, obj, exception) 182 #define shenandoah_assert_in_cset(interior_loc, obj) 183 184 #define shenandoah_assert_not_in_cset_if(interior_loc, obj, condition) 185 #define shenandoah_assert_not_in_cset_except(interior_loc, obj, exception) 186 #define shenandoah_assert_not_in_cset(interior_loc, obj) 187 188 #define shenandoah_assert_not_in_cset_loc_if(interior_loc, condition) 189 #define shenandoah_assert_not_in_cset_loc_except(interior_loc, exception) 190 #define shenandoah_assert_not_in_cset_loc(interior_loc) 191 192 #define shenandoah_assert_rp_isalive_installed() 193 #define shenandoah_assert_rp_isalive_not_installed() 194 195 #define shenandoah_assert_safepoint() 196 #define shenandoah_assert_locked_or_safepoint(lock) 197 198 #endif 199 200 #define shenandoah_not_implemented \ 201 { fatal("Deliberately not implemented."); } 202 #define shenandoah_not_implemented_return(v) \ 203 { fatal("Deliberately not implemented."); return v; } 204 205 }; 206 207 #endif // SHARE_GC_SHENANDOAH_SHENANDOAHASSERTS_HPP