1 /*
   2  * Copyright (c) 2015, 2015, 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.core.common.util;
  26 
  27 import static org.graalvm.compiler.core.common.util.TypeConversion.asS1;
  28 import static org.graalvm.compiler.core.common.util.TypeConversion.asS2;
  29 import static org.graalvm.compiler.core.common.util.TypeConversion.asS4;
  30 import static org.graalvm.compiler.core.common.util.TypeConversion.asU1;
  31 import static org.graalvm.compiler.core.common.util.TypeConversion.asU2;
  32 import static org.graalvm.compiler.core.common.util.TypeConversion.asU4;
  33 import sun.misc.Unsafe;
  34 
  35 /**
  36  * Provides low-level sequential write access to a byte[] array for signed and unsigned values of
  37  * size 1, 2, 4, and 8 bytes. To avoid copying an array when the buffer size is no longer
  38  * sufficient, the buffer is split into chunks of a fixed size.
  39  *
  40  * The flag {@code supportsUnalignedMemoryAccess} must be set according to the capabilities of the
  41  * hardware architecture: the value {@code true} allows more efficient memory access on
  42  * architectures that support unaligned memory accesses; the value {@code false} is the safe
  43  * fallback that works on every hardware.
  44  */
  45 public abstract class UnsafeArrayTypeWriter implements TypeWriter {
  46 
  47     private static final int MIN_CHUNK_LENGTH = 200;
  48     private static final int MAX_CHUNK_LENGTH = 16000;
  49 
  50     static class Chunk {
  51         protected final byte[] data;
  52         protected int size;
  53         protected Chunk next;
  54 
  55         protected Chunk(int arrayLength) {
  56             data = new byte[arrayLength];
  57         }
  58     }
  59 
  60     protected final Chunk firstChunk;
  61     protected Chunk writeChunk;
  62     protected int totalSize;
  63 
  64     public static UnsafeArrayTypeWriter create(boolean supportsUnalignedMemoryAccess) {
  65         if (supportsUnalignedMemoryAccess) {
  66             return new UnalignedUnsafeArrayTypeWriter();
  67         } else {
  68             return new AlignedUnsafeArrayTypeWriter();
  69         }
  70     }
  71 
  72     protected UnsafeArrayTypeWriter() {
  73         firstChunk = new Chunk(MIN_CHUNK_LENGTH);
  74         writeChunk = firstChunk;
  75     }
  76 
  77     @Override
  78     public final long getBytesWritten() {
  79         return totalSize;
  80     }
  81 
  82     /**
  83      * Copies the buffer into the provided byte[] array of length {@link #getBytesWritten()}.
  84      */
  85     public final byte[] toArray(byte[] result) {
  86         assert result.length == totalSize;
  87         int resultIdx = 0;
  88         for (Chunk cur = firstChunk; cur != null; cur = cur.next) {
  89             System.arraycopy(cur.data, 0, result, resultIdx, cur.size);
  90             resultIdx += cur.size;
  91         }
  92         assert resultIdx == totalSize;
  93         return result;
  94     }
  95 
  96     @Override
  97     public final void putS1(long value) {
  98         long offset = writeOffset(Byte.BYTES);
  99         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset, asS1(value));
 100     }
 101 
 102     @Override
 103     public final void putU1(long value) {
 104         long offset = writeOffset(Byte.BYTES);
 105         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset, asU1(value));
 106     }
 107 
 108     @Override
 109     public final void putU2(long value) {
 110         putS2(asU2(value));
 111     }
 112 
 113     @Override
 114     public final void putU4(long value) {
 115         putS4(asU4(value));
 116     }
 117 
 118     protected long writeOffset(int writeBytes) {
 119         if (writeChunk.size + writeBytes >= writeChunk.data.length) {
 120             Chunk newChunk = new Chunk(Math.min(writeChunk.data.length * 2, MAX_CHUNK_LENGTH));
 121             writeChunk.next = newChunk;
 122             writeChunk = newChunk;
 123         }
 124 
 125         assert Unsafe.ARRAY_BYTE_INDEX_SCALE == 1;
 126         long result = writeChunk.size + Unsafe.ARRAY_BYTE_BASE_OFFSET;
 127 
 128         totalSize += writeBytes;
 129         writeChunk.size += writeBytes;
 130         assert writeChunk.size <= writeChunk.data.length;
 131 
 132         return result;
 133     }
 134 }
 135 
 136 final class UnalignedUnsafeArrayTypeWriter extends UnsafeArrayTypeWriter {
 137     @Override
 138     public void putS2(long value) {
 139         long offset = writeOffset(Short.BYTES);
 140         UnsafeAccess.UNSAFE.putShort(writeChunk.data, offset, asS2(value));
 141     }
 142 
 143     @Override
 144     public void putS4(long value) {
 145         long offset = writeOffset(Integer.BYTES);
 146         UnsafeAccess.UNSAFE.putInt(writeChunk.data, offset, asS4(value));
 147     }
 148 
 149     @Override
 150     public void putS8(long value) {
 151         long offset = writeOffset(Long.BYTES);
 152         UnsafeAccess.UNSAFE.putLong(writeChunk.data, offset, value);
 153     }
 154 }
 155 
 156 final class AlignedUnsafeArrayTypeWriter extends UnsafeArrayTypeWriter {
 157     @Override
 158     public void putS2(long value) {
 159         long offset = writeOffset(Short.BYTES);
 160         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 0, (byte) (value >> 0));
 161         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 1, (byte) (value >> 8));
 162     }
 163 
 164     @Override
 165     public void putS4(long value) {
 166         long offset = writeOffset(Integer.BYTES);
 167         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 0, (byte) (value >> 0));
 168         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 1, (byte) (value >> 8));
 169         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 2, (byte) (value >> 16));
 170         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 3, (byte) (value >> 24));
 171     }
 172 
 173     @Override
 174     public void putS8(long value) {
 175         long offset = writeOffset(Long.BYTES);
 176         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 0, (byte) (value >> 0));
 177         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 1, (byte) (value >> 8));
 178         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 2, (byte) (value >> 16));
 179         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 3, (byte) (value >> 24));
 180         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 4, (byte) (value >> 32));
 181         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 5, (byte) (value >> 40));
 182         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 6, (byte) (value >> 48));
 183         UnsafeAccess.UNSAFE.putByte(writeChunk.data, offset + 7, (byte) (value >> 56));
 184     }
 185 }