src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.hotspot/src/org/graalvm/compiler/hotspot/replacements/WriteBarrierSnippets.java
branchmetal-prototype-branch
changeset 57440 10f701a91883
parent 57431 d5ab3442e44f
parent 55519 c59f36ed7b52
child 57441 ee34e24af607
equal deleted inserted replaced
57431:d5ab3442e44f 57440:10f701a91883
     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 }