All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.airlift.slice.Slices Maven / Gradle / Ivy

The newest version!
/*
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package io.airlift.slice;

import java.nio.ByteBuffer;
import java.nio.charset.Charset;
import java.util.Arrays;
import java.util.Random;
import java.util.concurrent.ThreadLocalRandom;

import static java.lang.String.format;
import static java.nio.charset.StandardCharsets.UTF_8;
import static java.util.Objects.requireNonNull;

public final class Slices
{
    /**
     * A slice with size {@code 0}.
     */
    public static final Slice EMPTY_SLICE = Slice.EMPTY_SLICE;

    // see java.util.ArrayList for an explanation
    static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;

    static final int SLICE_ALLOC_THRESHOLD = 524_288; // 2^19
    static final double SLICE_ALLOW_SKEW = 1.25; // must be > 1!

    private Slices() {}

    public static Slice ensureSize(Slice existingSlice, int minWritableBytes)
    {
        if (minWritableBytes < 0) {
            throw new IllegalArgumentException("minWritableBytes is negative");
        }

        if (existingSlice == null) {
            return allocate(minWritableBytes);
        }

        if (minWritableBytes <= existingSlice.length()) {
            return existingSlice;
        }

        int newCapacity;
        if (existingSlice.length() == 0) {
            newCapacity = 1;
        }
        else {
            newCapacity = existingSlice.length();
        }
        while (newCapacity < minWritableBytes) {
            if (newCapacity < SLICE_ALLOC_THRESHOLD) {
                newCapacity <<= 1;
            }
            else {
                newCapacity *= SLICE_ALLOW_SKEW; // double to int cast is saturating
                if (newCapacity > MAX_ARRAY_SIZE && minWritableBytes <= MAX_ARRAY_SIZE) {
                    newCapacity = MAX_ARRAY_SIZE;
                }
            }
        }

        byte[] bytes = existingSlice.byteArray();
        int offset = existingSlice.byteArrayOffset();
        // copy of range can be more efficient because it avoids zeroing memory
        byte[] copy = Arrays.copyOfRange(bytes, offset, offset + newCapacity);
        // but if the slice is a view, we need to zero the end of the array
        Arrays.fill(copy, existingSlice.length(), bytes.length - offset, (byte) 0);
        return new Slice(copy);
    }

    public static Slice allocate(int capacity)
    {
        if (capacity == 0) {
            return EMPTY_SLICE;
        }
        if (capacity > MAX_ARRAY_SIZE) {
            throw new SliceTooLargeException(format("Cannot allocate slice larger than %s bytes", MAX_ARRAY_SIZE));
        }
        return new Slice(new byte[capacity]);
    }

    /**
     * Allocates a new slice containing random bytes from ThreadLocalRandom.
     */
    public static Slice random(int capacity)
    {
        // Note: this only exists because many static checkers do not allow passing ThreadLocalRandom to methods
        Slice slice = allocate(capacity);
        ThreadLocalRandom.current().nextBytes(slice.byteArray());
        return slice;
    }

    /**
     * Allocates a new slice containing random bytes.
     */
    public static Slice random(int capacity, Random random)
    {
        Slice slice = allocate(capacity);
        random.nextBytes(slice.byteArray());
        return slice;
    }

    public static Slice wrappedHeapBuffer(ByteBuffer buffer)
    {
        if (!buffer.hasArray()) {
            throw new IllegalArgumentException("cannot wrap " + buffer.getClass().getName());
        }

        int length = buffer.remaining();
        if (length == 0) {
            return EMPTY_SLICE;
        }
        return new Slice(buffer.array(), buffer.arrayOffset() + buffer.position(), length);
    }

    /**
     * Creates a slice over the specified array.
     */
    public static Slice wrappedBuffer(byte... array)
    {
        if (array.length == 0) {
            return EMPTY_SLICE;
        }
        return new Slice(array);
    }

    /**
     * Creates a slice over the specified array range.
     *
     * @param offset the array position at which the slice begins
     * @param length the number of array positions to include in the slice
     */
    public static Slice wrappedBuffer(byte[] array, int offset, int length)
    {
        if (length == 0) {
            return EMPTY_SLICE;
        }
        return new Slice(array, offset, length);
    }

    public static Slice copiedBuffer(String string, Charset charset)
    {
        requireNonNull(string, "string is null");
        requireNonNull(charset, "charset is null");

        return wrappedBuffer(string.getBytes(charset));
    }

    public static Slice utf8Slice(String string)
    {
        return copiedBuffer(string, UTF_8);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy