1 /* 2 * Copyright (c) 2012, 2018, 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 package org.graalvm.compiler.hotspot.replacements; 26 27 import static jdk.vm.ci.code.MemoryBarriers.STORE_LOAD; 28 import static org.graalvm.compiler.hotspot.GraalHotSpotVMConfig.INJECTED_VMCONFIG; 29 import static org.graalvm.compiler.hotspot.GraalHotSpotVMConfigBase.INJECTED_METAACCESS; 30 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.cardTableShift; 31 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.dirtyCardValue; 32 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.g1CardQueueBufferOffset; 33 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.g1CardQueueIndexOffset; 34 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.g1SATBQueueBufferOffset; 35 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.g1SATBQueueIndexOffset; 36 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.g1SATBQueueMarkingOffset; 37 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.g1YoungCardValue; 38 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.registerAsWord; 39 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.verifyOop; 40 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.verifyOops; 41 import static org.graalvm.compiler.hotspot.replacements.HotSpotReplacementsUtil.wordSize; 42 import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.FREQUENT_PROBABILITY; 43 import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.LIKELY_PROBABILITY; 44 import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.NOT_FREQUENT_PROBABILITY; 45 import static org.graalvm.compiler.nodes.extended.BranchProbabilityNode.probability; 46 import static org.graalvm.compiler.replacements.SnippetTemplate.DEFAULT_REPLACER; 47 48 import org.graalvm.compiler.api.replacements.Snippet; 49 import org.graalvm.compiler.api.replacements.Snippet.ConstantParameter; 50 import org.graalvm.compiler.core.common.CompressEncoding; 51 import org.graalvm.compiler.core.common.GraalOptions; 52 import org.graalvm.compiler.core.common.spi.ForeignCallDescriptor; 53 import org.graalvm.compiler.debug.DebugHandlersFactory; 54 import org.graalvm.compiler.graph.Node.ConstantNodeParameter; 55 import org.graalvm.compiler.graph.Node.NodeIntrinsic; 56 import org.graalvm.compiler.hotspot.GraalHotSpotVMConfig; 57 import org.graalvm.compiler.hotspot.gc.g1.G1ArrayRangePostWriteBarrier; 58 import org.graalvm.compiler.hotspot.gc.g1.G1ArrayRangePreWriteBarrier; 59 import org.graalvm.compiler.hotspot.gc.g1.G1PostWriteBarrier; 60 import org.graalvm.compiler.hotspot.gc.g1.G1PreWriteBarrier; 61 import org.graalvm.compiler.hotspot.gc.g1.G1ReferentFieldReadBarrier; 62 import org.graalvm.compiler.hotspot.gc.shared.SerialArrayRangeWriteBarrier; 63 import org.graalvm.compiler.hotspot.gc.shared.SerialWriteBarrier; 64 import org.graalvm.compiler.hotspot.meta.HotSpotProviders; 65 import org.graalvm.compiler.hotspot.meta.HotSpotRegistersProvider; 66 import org.graalvm.compiler.hotspot.nodes.GraalHotSpotVMConfigNode; 67 import org.graalvm.compiler.hotspot.nodes.HotSpotCompressionNode; 68 import org.graalvm.compiler.hotspot.nodes.VMErrorNode; 69 import org.graalvm.compiler.nodes.NamedLocationIdentity; 70 import org.graalvm.compiler.nodes.NodeView; 71 import org.graalvm.compiler.nodes.PiNode; 72 import org.graalvm.compiler.nodes.SnippetAnchorNode; 73 import org.graalvm.compiler.nodes.StructuredGraph; 74 import org.graalvm.compiler.nodes.ValueNode; 75 import org.graalvm.compiler.nodes.extended.FixedValueAnchorNode; 76 import org.graalvm.compiler.nodes.extended.ForeignCallNode; 77 import org.graalvm.compiler.nodes.extended.MembarNode; 78 import org.graalvm.compiler.nodes.extended.NullCheckNode; 79 import org.graalvm.compiler.nodes.memory.HeapAccess.BarrierType; 80 import org.graalvm.compiler.nodes.memory.address.AddressNode; 81 import org.graalvm.compiler.nodes.memory.address.AddressNode.Address; 82 import org.graalvm.compiler.nodes.memory.address.OffsetAddressNode; 83 import org.graalvm.compiler.nodes.spi.LoweringTool; 84 import org.graalvm.compiler.nodes.type.NarrowOopStamp; 85 import org.graalvm.compiler.options.OptionValues; 86 import org.graalvm.compiler.replacements.Log; 87 import org.graalvm.compiler.replacements.ReplacementsUtil; 88 import org.graalvm.compiler.replacements.SnippetCounter; 89 import org.graalvm.compiler.replacements.SnippetCounter.Group; 90 import org.graalvm.compiler.replacements.SnippetTemplate.AbstractTemplates; 91 import org.graalvm.compiler.replacements.SnippetTemplate.Arguments; 92 import org.graalvm.compiler.replacements.SnippetTemplate.SnippetInfo; 93 import org.graalvm.compiler.replacements.Snippets; 94 import org.graalvm.compiler.replacements.nodes.AssertionNode; 95 import org.graalvm.compiler.replacements.nodes.DirectStoreNode; 96 import org.graalvm.compiler.word.Word; 97 import jdk.internal.vm.compiler.word.LocationIdentity; 98 import jdk.internal.vm.compiler.word.Pointer; 99 import jdk.internal.vm.compiler.word.UnsignedWord; 100 import jdk.internal.vm.compiler.word.WordFactory; 101 102 import jdk.vm.ci.code.Register; 103 import jdk.vm.ci.code.TargetDescription; 104 import jdk.vm.ci.meta.JavaKind; 105 106 public class WriteBarrierSnippets implements Snippets { 107 108 static class Counters { 109 Counters(SnippetCounter.Group.Factory factory) { 110 Group countersWriteBarriers = factory.createSnippetCounterGroup("WriteBarriers"); 111 serialWriteBarrierCounter = new SnippetCounter(countersWriteBarriers, "serialWriteBarrier", "Number of Serial Write Barriers"); 112 g1AttemptedPreWriteBarrierCounter = new SnippetCounter(countersWriteBarriers, "g1AttemptedPreWriteBarrier", "Number of attempted G1 Pre Write Barriers"); 113 g1EffectivePreWriteBarrierCounter = new SnippetCounter(countersWriteBarriers, "g1EffectivePreWriteBarrier", "Number of effective G1 Pre Write Barriers"); 114 g1ExecutedPreWriteBarrierCounter = new SnippetCounter(countersWriteBarriers, "g1ExecutedPreWriteBarrier", "Number of executed G1 Pre Write Barriers"); 115 g1AttemptedPostWriteBarrierCounter = new SnippetCounter(countersWriteBarriers, "g1AttemptedPostWriteBarrier", "Number of attempted G1 Post Write Barriers"); 116 g1EffectiveAfterXORPostWriteBarrierCounter = new SnippetCounter(countersWriteBarriers, "g1EffectiveAfterXORPostWriteBarrier", 117 "Number of effective G1 Post Write Barriers (after passing the XOR test)"); 118 g1EffectiveAfterNullPostWriteBarrierCounter = new SnippetCounter(countersWriteBarriers, "g1EffectiveAfterNullPostWriteBarrier", 119 "Number of effective G1 Post Write Barriers (after passing the NULL test)"); 120 g1ExecutedPostWriteBarrierCounter = new SnippetCounter(countersWriteBarriers, "g1ExecutedPostWriteBarrier", "Number of executed G1 Post Write Barriers"); 121 122 } 123 124 final SnippetCounter serialWriteBarrierCounter; 125 final SnippetCounter g1AttemptedPreWriteBarrierCounter; 126 final SnippetCounter g1EffectivePreWriteBarrierCounter; 127 final SnippetCounter g1ExecutedPreWriteBarrierCounter; 128 final SnippetCounter g1AttemptedPostWriteBarrierCounter; 129 final SnippetCounter g1EffectiveAfterXORPostWriteBarrierCounter; 130 final SnippetCounter g1EffectiveAfterNullPostWriteBarrierCounter; 131 final SnippetCounter g1ExecutedPostWriteBarrierCounter; 132 } 133 134 public static final LocationIdentity GC_CARD_LOCATION = NamedLocationIdentity.mutable("GC-Card"); 135 public static final LocationIdentity GC_LOG_LOCATION = NamedLocationIdentity.mutable("GC-Log"); 136 public static final LocationIdentity GC_INDEX_LOCATION = NamedLocationIdentity.mutable("GC-Index"); 137 138 private static void serialWriteBarrier(Pointer ptr, Counters counters) { 139 counters.serialWriteBarrierCounter.inc(); 140 final long startAddress = GraalHotSpotVMConfigNode.cardTableAddress(); 141 Word base = (Word) ptr.unsignedShiftRight(cardTableShift(INJECTED_VMCONFIG)); 142 if (((int) startAddress) == startAddress && GraalHotSpotVMConfigNode.isCardTableAddressConstant()) { 143 base.writeByte((int) startAddress, (byte) 0, GC_CARD_LOCATION); 144 } else { 145 base.writeByte(WordFactory.unsigned(startAddress), (byte) 0, GC_CARD_LOCATION); 146 } 147 } 148 149 @Snippet 150 public static void serialImpreciseWriteBarrier(Object object, @ConstantParameter boolean verifyBarrier, @ConstantParameter Counters counters) { 151 if (verifyBarrier) { 152 verifyNotArray(object); 153 } 154 serialWriteBarrier(Word.objectToTrackedPointer(object), counters); 155 } 156 157 @Snippet 158 public static void serialPreciseWriteBarrier(Address address, @ConstantParameter Counters counters) { 159 serialWriteBarrier(Word.fromAddress(address), counters); 160 } 161 162 @Snippet 163 public static void serialArrayRangeWriteBarrier(Address address, int length, @ConstantParameter int elementStride) { 164 if (length == 0) { 165 return; 166 } 167 int cardShift = cardTableShift(INJECTED_VMCONFIG); 168 final long cardStart = GraalHotSpotVMConfigNode.cardTableAddress(); 169 long start = getPointerToFirstArrayElement(address, length, elementStride) >>> cardShift; 170 long end = getPointerToLastArrayElement(address, length, elementStride) >>> cardShift; 171 long count = end - start + 1; 172 while (count-- > 0) { 173 DirectStoreNode.storeBoolean((start + cardStart) + count, false, JavaKind.Boolean); 174 } 175 } 176 177 @Snippet 178 public static void g1PreWriteBarrier(Address address, Object object, Object expectedObject, @ConstantParameter boolean doLoad, @ConstantParameter boolean nullCheck, 179 @ConstantParameter Register threadRegister, @ConstantParameter boolean trace, @ConstantParameter Counters counters) { 180 if (nullCheck) { 181 NullCheckNode.nullCheck(address); 182 } 183 Word thread = registerAsWord(threadRegister); 184 verifyOop(object); 185 Object fixedExpectedObject = FixedValueAnchorNode.getObject(expectedObject); 186 Word field = Word.fromAddress(address); 187 Pointer previousOop = Word.objectToTrackedPointer(fixedExpectedObject); 188 byte markingValue = thread.readByte(g1SATBQueueMarkingOffset(INJECTED_VMCONFIG)); 189 int gcCycle = 0; 190 if (trace) { 191 Pointer gcTotalCollectionsAddress = WordFactory.pointer(HotSpotReplacementsUtil.gcTotalCollectionsAddress(INJECTED_VMCONFIG)); 192 gcCycle = (int) gcTotalCollectionsAddress.readLong(0); 193 log(trace, "[%d] G1-Pre Thread %p Object %p\n", gcCycle, thread.rawValue(), Word.objectToTrackedPointer(object).rawValue()); 194 log(trace, "[%d] G1-Pre Thread %p Expected Object %p\n", gcCycle, thread.rawValue(), Word.objectToTrackedPointer(fixedExpectedObject).rawValue()); 195 log(trace, "[%d] G1-Pre Thread %p Field %p\n", gcCycle, thread.rawValue(), field.rawValue()); 196 log(trace, "[%d] G1-Pre Thread %p Marking %d\n", gcCycle, thread.rawValue(), markingValue); 197 log(trace, "[%d] G1-Pre Thread %p DoLoad %d\n", gcCycle, thread.rawValue(), doLoad ? 1L : 0L); 198 } 199 counters.g1AttemptedPreWriteBarrierCounter.inc(); 200 // If the concurrent marker is enabled, the barrier is issued. 201 if (probability(NOT_FREQUENT_PROBABILITY, markingValue != (byte) 0)) { 202 // If the previous value has to be loaded (before the write), the load is issued. 203 // The load is always issued except the cases of CAS and referent field. 204 if (probability(LIKELY_PROBABILITY, doLoad)) { 205 previousOop = Word.objectToTrackedPointer(field.readObject(0, BarrierType.NONE)); 206 if (trace) { 207 log(trace, "[%d] G1-Pre Thread %p Previous Object %p\n ", gcCycle, thread.rawValue(), previousOop.rawValue()); 208 verifyOop(previousOop.toObject()); 209 } 210 } 211 counters.g1EffectivePreWriteBarrierCounter.inc(); 212 // If the previous value is null the barrier should not be issued. 213 if (probability(FREQUENT_PROBABILITY, previousOop.notEqual(0))) { 214 counters.g1ExecutedPreWriteBarrierCounter.inc(); 215 // If the thread-local SATB buffer is full issue a native call which will 216 // initialize a new one and add the entry. 217 Word indexAddress = thread.add(g1SATBQueueIndexOffset(INJECTED_VMCONFIG)); 218 Word indexValue = indexAddress.readWord(0); 219 if (probability(FREQUENT_PROBABILITY, indexValue.notEqual(0))) { 220 Word bufferAddress = thread.readWord(g1SATBQueueBufferOffset(INJECTED_VMCONFIG)); 221 Word nextIndex = indexValue.subtract(wordSize()); 222 Word logAddress = bufferAddress.add(nextIndex); 223 // Log the object to be marked as well as update the SATB's buffer next index. 224 logAddress.writeWord(0, previousOop, GC_LOG_LOCATION); 225 indexAddress.writeWord(0, nextIndex, GC_INDEX_LOCATION); 226 } else { 227 g1PreBarrierStub(G1WBPRECALL, previousOop.toObject()); 228 } 229 } 230 } 231 } 232 233 @Snippet 234 public static void g1PostWriteBarrier(Address address, Object object, Object value, @ConstantParameter boolean usePrecise, @ConstantParameter boolean verifyBarrier, 235 @ConstantParameter Register threadRegister, @ConstantParameter boolean trace, @ConstantParameter Counters counters) { 236 Word thread = registerAsWord(threadRegister); 237 Object fixedValue = FixedValueAnchorNode.getObject(value); 238 verifyOop(object); 239 verifyOop(fixedValue); 240 validateObject(object, fixedValue); 241 Pointer oop; 242 if (usePrecise) { 243 oop = Word.fromAddress(address); 244 } else { 245 if (verifyBarrier) { 246 verifyNotArray(object); 247 } 248 oop = Word.objectToTrackedPointer(object); 249 } 250 int gcCycle = 0; 251 if (trace) { 252 Pointer gcTotalCollectionsAddress = WordFactory.pointer(HotSpotReplacementsUtil.gcTotalCollectionsAddress(INJECTED_VMCONFIG)); 253 gcCycle = (int) gcTotalCollectionsAddress.readLong(0); 254 log(trace, "[%d] G1-Post Thread: %p Object: %p\n", gcCycle, thread.rawValue(), Word.objectToTrackedPointer(object).rawValue()); 255 log(trace, "[%d] G1-Post Thread: %p Field: %p\n", gcCycle, thread.rawValue(), oop.rawValue()); 256 } 257 Pointer writtenValue = Word.objectToTrackedPointer(fixedValue); 258 // The result of the xor reveals whether the installed pointer crosses heap regions. 259 // In case it does the write barrier has to be issued. 260 final int logOfHeapRegionGrainBytes = GraalHotSpotVMConfigNode.logOfHeapRegionGrainBytes(); 261 UnsignedWord xorResult = (oop.xor(writtenValue)).unsignedShiftRight(logOfHeapRegionGrainBytes); 262 263 // Calculate the address of the card to be enqueued to the 264 // thread local card queue. 265 UnsignedWord cardBase = oop.unsignedShiftRight(cardTableShift(INJECTED_VMCONFIG)); 266 final long startAddress = GraalHotSpotVMConfigNode.cardTableAddress(); 267 int displacement = 0; 268 if (((int) startAddress) == startAddress && GraalHotSpotVMConfigNode.isCardTableAddressConstant()) { 269 displacement = (int) startAddress; 270 } else { 271 cardBase = cardBase.add(WordFactory.unsigned(startAddress)); 272 } 273 Word cardAddress = (Word) cardBase.add(displacement); 274 275 counters.g1AttemptedPostWriteBarrierCounter.inc(); 276 if (probability(FREQUENT_PROBABILITY, xorResult.notEqual(0))) { 277 counters.g1EffectiveAfterXORPostWriteBarrierCounter.inc(); 278 279 // If the written value is not null continue with the barrier addition. 280 if (probability(FREQUENT_PROBABILITY, writtenValue.notEqual(0))) { 281 byte cardByte = cardAddress.readByte(0, GC_CARD_LOCATION); 282 counters.g1EffectiveAfterNullPostWriteBarrierCounter.inc(); 283 284 // If the card is already dirty, (hence already enqueued) skip the insertion. 285 if (probability(NOT_FREQUENT_PROBABILITY, cardByte != g1YoungCardValue(INJECTED_VMCONFIG))) { 286 MembarNode.memoryBarrier(STORE_LOAD, GC_CARD_LOCATION); 287 byte cardByteReload = cardAddress.readByte(0, GC_CARD_LOCATION); 288 if (probability(NOT_FREQUENT_PROBABILITY, cardByteReload != dirtyCardValue(INJECTED_VMCONFIG))) { 289 log(trace, "[%d] G1-Post Thread: %p Card: %p \n", gcCycle, thread.rawValue(), WordFactory.unsigned((int) cardByte).rawValue()); 290 cardAddress.writeByte(0, (byte) 0, GC_CARD_LOCATION); 291 counters.g1ExecutedPostWriteBarrierCounter.inc(); 292 293 // If the thread local card queue is full, issue a native call which will 294 // initialize a new one and add the card entry. 295 Word indexAddress = thread.add(g1CardQueueIndexOffset(INJECTED_VMCONFIG)); 296 Word indexValue = thread.readWord(g1CardQueueIndexOffset(INJECTED_VMCONFIG)); 297 if (probability(FREQUENT_PROBABILITY, indexValue.notEqual(0))) { 298 Word bufferAddress = thread.readWord(g1CardQueueBufferOffset(INJECTED_VMCONFIG)); 299 Word nextIndex = indexValue.subtract(wordSize()); 300 Word logAddress = bufferAddress.add(nextIndex); 301 // Log the object to be scanned as well as update 302 // the card queue's next index. 303 logAddress.writeWord(0, cardAddress, GC_LOG_LOCATION); 304 indexAddress.writeWord(0, nextIndex, GC_INDEX_LOCATION); 305 } else { 306 g1PostBarrierStub(G1WBPOSTCALL, cardAddress); 307 } 308 } 309 } 310 } 311 } 312 } 313 314 private static void verifyNotArray(Object object) { 315 if (object != null) { 316 // Manually build the null check and cast because we're in snippet that's lowered late. 317 AssertionNode.assertion(false, !PiNode.piCastNonNull(object, SnippetAnchorNode.anchor()).getClass().isArray(), "imprecise card mark used with array"); 318 } 319 } 320 321 @Snippet 322 public static void g1ArrayRangePreWriteBarrier(Address address, int length, @ConstantParameter int elementStride, @ConstantParameter Register threadRegister) { 323 Word thread = registerAsWord(threadRegister); 324 byte markingValue = thread.readByte(g1SATBQueueMarkingOffset(INJECTED_VMCONFIG)); 325 // If the concurrent marker is not enabled or the vector length is zero, return. 326 if (markingValue == (byte) 0 || length == 0) { 327 return; 328 } 329 Word bufferAddress = thread.readWord(g1SATBQueueBufferOffset(INJECTED_VMCONFIG)); 330 Word indexAddress = thread.add(g1SATBQueueIndexOffset(INJECTED_VMCONFIG)); 331 long indexValue = indexAddress.readWord(0).rawValue(); 332 final int scale = ReplacementsUtil.arrayIndexScale(INJECTED_METAACCESS, JavaKind.Object); 333 long start = getPointerToFirstArrayElement(address, length, elementStride); 334 335 for (int i = 0; i < length; i++) { 336 Word arrElemPtr = WordFactory.pointer(start + i * scale); 337 Pointer oop = Word.objectToTrackedPointer(arrElemPtr.readObject(0, BarrierType.NONE)); 338 verifyOop(oop.toObject()); 339 if (oop.notEqual(0)) { 340 if (indexValue != 0) { 341 indexValue = indexValue - wordSize(); 342 Word logAddress = bufferAddress.add(WordFactory.unsigned(indexValue)); 343 // Log the object to be marked as well as update the SATB's buffer next index. 344 logAddress.writeWord(0, oop, GC_LOG_LOCATION); 345 indexAddress.writeWord(0, WordFactory.unsigned(indexValue), GC_INDEX_LOCATION); 346 } else { 347 g1PreBarrierStub(G1WBPRECALL, oop.toObject()); 348 } 349 } 350 } 351 } 352 353 @Snippet 354 public static void g1ArrayRangePostWriteBarrier(Address address, int length, @ConstantParameter int elementStride, @ConstantParameter Register threadRegister) { 355 if (length == 0) { 356 return; 357 } 358 Word thread = registerAsWord(threadRegister); 359 Word bufferAddress = thread.readWord(g1CardQueueBufferOffset(INJECTED_VMCONFIG)); 360 Word indexAddress = thread.add(g1CardQueueIndexOffset(INJECTED_VMCONFIG)); 361 long indexValue = thread.readWord(g1CardQueueIndexOffset(INJECTED_VMCONFIG)).rawValue(); 362 363 int cardShift = cardTableShift(INJECTED_VMCONFIG); 364 final long cardStart = GraalHotSpotVMConfigNode.cardTableAddress(); 365 long start = getPointerToFirstArrayElement(address, length, elementStride) >>> cardShift; 366 long end = getPointerToLastArrayElement(address, length, elementStride) >>> cardShift; 367 long count = end - start + 1; 368 369 while (count-- > 0) { 370 Word cardAddress = WordFactory.unsigned((start + cardStart) + count); 371 byte cardByte = cardAddress.readByte(0, GC_CARD_LOCATION); 372 // If the card is already dirty, (hence already enqueued) skip the insertion. 373 if (probability(NOT_FREQUENT_PROBABILITY, cardByte != g1YoungCardValue(INJECTED_VMCONFIG))) { 374 MembarNode.memoryBarrier(STORE_LOAD, GC_CARD_LOCATION); 375 byte cardByteReload = cardAddress.readByte(0, GC_CARD_LOCATION); 376 if (probability(NOT_FREQUENT_PROBABILITY, cardByteReload != dirtyCardValue(INJECTED_VMCONFIG))) { 377 cardAddress.writeByte(0, (byte) 0, GC_CARD_LOCATION); 378 // If the thread local card queue is full, issue a native call which will 379 // initialize a new one and add the card entry. 380 if (indexValue != 0) { 381 indexValue = indexValue - wordSize(); 382 Word logAddress = bufferAddress.add(WordFactory.unsigned(indexValue)); 383 // Log the object to be scanned as well as update 384 // the card queue's next index. 385 logAddress.writeWord(0, cardAddress, GC_LOG_LOCATION); 386 indexAddress.writeWord(0, WordFactory.unsigned(indexValue), GC_INDEX_LOCATION); 387 } else { 388 g1PostBarrierStub(G1WBPOSTCALL, cardAddress); 389 } 390 } 391 } 392 } 393 } 394 395 private static long getPointerToFirstArrayElement(Address address, int length, int elementStride) { 396 long result = Word.fromAddress(address).rawValue(); 397 if (elementStride < 0) { 398 // the address points to the place after the last array element 399 result = result + elementStride * length; 400 } 401 return result; 402 } 403 404 private static long getPointerToLastArrayElement(Address address, int length, int elementStride) { 405 long result = Word.fromAddress(address).rawValue(); 406 if (elementStride < 0) { 407 // the address points to the place after the last array element 408 result = result + elementStride; 409 } else { 410 result = result + (length - 1) * elementStride; 411 } 412 return result; 413 } 414 415 public static final ForeignCallDescriptor G1WBPRECALL = new ForeignCallDescriptor("write_barrier_pre", void.class, Object.class); 416 417 @NodeIntrinsic(ForeignCallNode.class) 418 private static native void g1PreBarrierStub(@ConstantNodeParameter ForeignCallDescriptor descriptor, Object object); 419 420 public static final ForeignCallDescriptor G1WBPOSTCALL = new ForeignCallDescriptor("write_barrier_post", void.class, Word.class); 421 422 @NodeIntrinsic(ForeignCallNode.class) 423 public static native void g1PostBarrierStub(@ConstantNodeParameter ForeignCallDescriptor descriptor, Word card); 424 425 public static class Templates extends AbstractTemplates { 426 427 private final SnippetInfo serialImpreciseWriteBarrier = snippet(WriteBarrierSnippets.class, "serialImpreciseWriteBarrier", GC_CARD_LOCATION); 428 private final SnippetInfo serialPreciseWriteBarrier = snippet(WriteBarrierSnippets.class, "serialPreciseWriteBarrier", GC_CARD_LOCATION); 429 private final SnippetInfo serialArrayRangeWriteBarrier = snippet(WriteBarrierSnippets.class, "serialArrayRangeWriteBarrier"); 430 private final SnippetInfo g1PreWriteBarrier = snippet(WriteBarrierSnippets.class, "g1PreWriteBarrier", GC_INDEX_LOCATION, GC_LOG_LOCATION); 431 private final SnippetInfo g1ReferentReadBarrier = g1PreWriteBarrier; 432 private final SnippetInfo g1PostWriteBarrier = snippet(WriteBarrierSnippets.class, "g1PostWriteBarrier", GC_CARD_LOCATION, GC_INDEX_LOCATION, GC_LOG_LOCATION); 433 private final SnippetInfo g1ArrayRangePreWriteBarrier = snippet(WriteBarrierSnippets.class, "g1ArrayRangePreWriteBarrier", GC_INDEX_LOCATION, GC_LOG_LOCATION); 434 private final SnippetInfo g1ArrayRangePostWriteBarrier = snippet(WriteBarrierSnippets.class, "g1ArrayRangePostWriteBarrier", GC_CARD_LOCATION, GC_INDEX_LOCATION, GC_LOG_LOCATION); 435 436 private final CompressEncoding oopEncoding; 437 private final Counters counters; 438 private final boolean verifyBarrier; 439 private final long gcTotalCollectionsAddress; 440 441 public Templates(OptionValues options, Iterable<DebugHandlersFactory> factories, Group.Factory factory, HotSpotProviders providers, TargetDescription target, 442 GraalHotSpotVMConfig config) { 443 super(options, factories, providers, providers.getSnippetReflection(), target); 444 this.oopEncoding = config.useCompressedOops ? config.getOopEncoding() : null; 445 this.verifyBarrier = ReplacementsUtil.REPLACEMENTS_ASSERTIONS_ENABLED || config.verifyBeforeGC || config.verifyAfterGC; 446 this.gcTotalCollectionsAddress = config.gcTotalCollectionsAddress(); 447 this.counters = new Counters(factory); 448 } 449 450 public boolean traceBarrier(StructuredGraph graph) { 451 long startCycle = GraalOptions.GCDebugStartCycle.getValue(graph.getOptions()); 452 return startCycle > 0 && ((Pointer) WordFactory.pointer(gcTotalCollectionsAddress)).readLong(0) > startCycle; 453 } 454 455 public void lower(SerialWriteBarrier writeBarrier, LoweringTool tool) { 456 Arguments args; 457 if (writeBarrier.usePrecise()) { 458 args = new Arguments(serialPreciseWriteBarrier, writeBarrier.graph().getGuardsStage(), tool.getLoweringStage()); 459 args.add("address", writeBarrier.getAddress()); 460 } else { 461 args = new Arguments(serialImpreciseWriteBarrier, writeBarrier.graph().getGuardsStage(), tool.getLoweringStage()); 462 OffsetAddressNode address = (OffsetAddressNode) writeBarrier.getAddress(); 463 args.add("object", address.getBase()); 464 args.addConst("verifyBarrier", verifyBarrier); 465 } 466 args.addConst("counters", counters); 467 template(writeBarrier, args).instantiate(providers.getMetaAccess(), writeBarrier, DEFAULT_REPLACER, args); 468 } 469 470 public void lower(SerialArrayRangeWriteBarrier arrayRangeWriteBarrier, LoweringTool tool) { 471 Arguments args = new Arguments(serialArrayRangeWriteBarrier, arrayRangeWriteBarrier.graph().getGuardsStage(), tool.getLoweringStage()); 472 args.add("address", arrayRangeWriteBarrier.getAddress()); 473 args.add("length", arrayRangeWriteBarrier.getLength()); 474 args.addConst("elementStride", arrayRangeWriteBarrier.getElementStride()); 475 template(arrayRangeWriteBarrier, args).instantiate(providers.getMetaAccess(), arrayRangeWriteBarrier, DEFAULT_REPLACER, args); 476 } 477 478 public void lower(G1PreWriteBarrier writeBarrierPre, HotSpotRegistersProvider registers, LoweringTool tool) { 479 Arguments args = new Arguments(g1PreWriteBarrier, writeBarrierPre.graph().getGuardsStage(), tool.getLoweringStage()); 480 AddressNode address = writeBarrierPre.getAddress(); 481 args.add("address", address); 482 if (address instanceof OffsetAddressNode) { 483 args.add("object", ((OffsetAddressNode) address).getBase()); 484 } else { 485 args.add("object", null); 486 } 487 488 ValueNode expected = writeBarrierPre.getExpectedObject(); 489 if (expected != null && expected.stamp(NodeView.DEFAULT) instanceof NarrowOopStamp) { 490 assert oopEncoding != null; 491 expected = HotSpotCompressionNode.uncompress(expected, oopEncoding); 492 } 493 args.add("expectedObject", expected); 494 495 args.addConst("doLoad", writeBarrierPre.doLoad()); 496 args.addConst("nullCheck", writeBarrierPre.getNullCheck()); 497 args.addConst("threadRegister", registers.getThreadRegister()); 498 args.addConst("trace", traceBarrier(writeBarrierPre.graph())); 499 args.addConst("counters", counters); 500 template(writeBarrierPre, args).instantiate(providers.getMetaAccess(), writeBarrierPre, DEFAULT_REPLACER, args); 501 } 502 503 public void lower(G1ReferentFieldReadBarrier readBarrier, HotSpotRegistersProvider registers, LoweringTool tool) { 504 Arguments args = new Arguments(g1ReferentReadBarrier, readBarrier.graph().getGuardsStage(), tool.getLoweringStage()); 505 AddressNode address = readBarrier.getAddress(); 506 args.add("address", address); 507 if (address instanceof OffsetAddressNode) { 508 args.add("object", ((OffsetAddressNode) address).getBase()); 509 } else { 510 args.add("object", null); 511 } 512 513 ValueNode expected = readBarrier.getExpectedObject(); 514 if (expected != null && expected.stamp(NodeView.DEFAULT) instanceof NarrowOopStamp) { 515 assert oopEncoding != null; 516 expected = HotSpotCompressionNode.uncompress(expected, oopEncoding); 517 } 518 519 args.add("expectedObject", expected); 520 args.addConst("doLoad", readBarrier.doLoad()); 521 args.addConst("nullCheck", false); 522 args.addConst("threadRegister", registers.getThreadRegister()); 523 args.addConst("trace", traceBarrier(readBarrier.graph())); 524 args.addConst("counters", counters); 525 template(readBarrier, args).instantiate(providers.getMetaAccess(), readBarrier, DEFAULT_REPLACER, args); 526 } 527 528 public void lower(G1PostWriteBarrier writeBarrierPost, HotSpotRegistersProvider registers, LoweringTool tool) { 529 StructuredGraph graph = writeBarrierPost.graph(); 530 if (writeBarrierPost.alwaysNull()) { 531 graph.removeFixed(writeBarrierPost); 532 return; 533 } 534 Arguments args = new Arguments(g1PostWriteBarrier, graph.getGuardsStage(), tool.getLoweringStage()); 535 AddressNode address = writeBarrierPost.getAddress(); 536 args.add("address", address); 537 if (address instanceof OffsetAddressNode) { 538 args.add("object", ((OffsetAddressNode) address).getBase()); 539 } else { 540 assert writeBarrierPost.usePrecise() : "found imprecise barrier that's not an object access " + writeBarrierPost; 541 args.add("object", null); 542 } 543 544 ValueNode value = writeBarrierPost.getValue(); 545 if (value.stamp(NodeView.DEFAULT) instanceof NarrowOopStamp) { 546 assert oopEncoding != null; 547 value = HotSpotCompressionNode.uncompress(value, oopEncoding); 548 } 549 args.add("value", value); 550 551 args.addConst("usePrecise", writeBarrierPost.usePrecise()); 552 args.addConst("verifyBarrier", verifyBarrier); 553 args.addConst("threadRegister", registers.getThreadRegister()); 554 args.addConst("trace", traceBarrier(writeBarrierPost.graph())); 555 args.addConst("counters", counters); 556 template(writeBarrierPost, args).instantiate(providers.getMetaAccess(), writeBarrierPost, DEFAULT_REPLACER, args); 557 } 558 559 public void lower(G1ArrayRangePreWriteBarrier arrayRangeWriteBarrier, HotSpotRegistersProvider registers, LoweringTool tool) { 560 Arguments args = new Arguments(g1ArrayRangePreWriteBarrier, arrayRangeWriteBarrier.graph().getGuardsStage(), tool.getLoweringStage()); 561 args.add("address", arrayRangeWriteBarrier.getAddress()); 562 args.add("length", arrayRangeWriteBarrier.getLength()); 563 args.addConst("elementStride", arrayRangeWriteBarrier.getElementStride()); 564 args.addConst("threadRegister", registers.getThreadRegister()); 565 template(arrayRangeWriteBarrier, args).instantiate(providers.getMetaAccess(), arrayRangeWriteBarrier, DEFAULT_REPLACER, args); 566 } 567 568 public void lower(G1ArrayRangePostWriteBarrier arrayRangeWriteBarrier, HotSpotRegistersProvider registers, LoweringTool tool) { 569 Arguments args = new Arguments(g1ArrayRangePostWriteBarrier, arrayRangeWriteBarrier.graph().getGuardsStage(), tool.getLoweringStage()); 570 args.add("address", arrayRangeWriteBarrier.getAddress()); 571 args.add("length", arrayRangeWriteBarrier.getLength()); 572 args.addConst("elementStride", arrayRangeWriteBarrier.getElementStride()); 573 args.addConst("threadRegister", registers.getThreadRegister()); 574 template(arrayRangeWriteBarrier, args).instantiate(providers.getMetaAccess(), arrayRangeWriteBarrier, DEFAULT_REPLACER, args); 575 } 576 } 577 578 /** 579 * Log method of debugging purposes. 580 */ 581 public static void log(boolean enabled, String format, long value) { 582 if (enabled) { 583 Log.printf(format, value); 584 } 585 } 586 587 public static void log(boolean enabled, String format, long value1, long value2) { 588 if (enabled) { 589 Log.printf(format, value1, value2); 590 } 591 } 592 593 public static void log(boolean enabled, String format, long value1, long value2, long value3) { 594 if (enabled) { 595 Log.printf(format, value1, value2, value3); 596 } 597 } 598 599 /** 600 * Validation helper method which performs sanity checks on write operations. The addresses of 601 * both the object and the value being written are checked in order to determine if they reside 602 * in a valid heap region. If an object is stale, an invalid access is performed in order to 603 * prematurely crash the VM and debug the stack trace of the faulty method. 604 */ 605 public static void validateObject(Object parent, Object child) { 606 if (verifyOops(INJECTED_VMCONFIG) && child != null) { 607 Word parentWord = Word.objectToTrackedPointer(parent); 608 Word childWord = Word.objectToTrackedPointer(child); 609 if (!validateOop(VALIDATE_OBJECT, parentWord, childWord)) { 610 log(true, "Verification ERROR, Parent: %p Child: %p\n", parentWord.rawValue(), childWord.rawValue()); 611 VMErrorNode.vmError("Verification ERROR, Parent: %p\n", parentWord.rawValue()); 612 } 613 } 614 } 615 616 public static final ForeignCallDescriptor VALIDATE_OBJECT = new ForeignCallDescriptor("validate_object", boolean.class, Word.class, Word.class); 617 618 @NodeIntrinsic(ForeignCallNode.class) 619 private static native boolean validateOop(@ConstantNodeParameter ForeignCallDescriptor descriptor, Word parent, Word object); 620 621 }