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

io.timeandspace.smoothie.Utils Maven / Gradle / Ivy

There is a newer version: 2.0.2
Show newest version
/*
 * Copyright (C) The SmoothieMap Authors
 *
 * 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.timeandspace.smoothie;


import com.google.errorprone.annotations.CanIgnoreReturnValue;
import org.checkerframework.checker.nullness.qual.Nullable;
import org.jetbrains.annotations.Contract;

import java.util.ConcurrentModificationException;

/**
 * Prefixes of methods used in this class:
 *   - verifyXxx throw an AssertionError
 *   - checkXxx throw a regular unchecked exception, like NPE or ISE
 */
final class Utils {

    /**
     * = numberOfTrailingZeros({@link Byte#SIZE})
     *
     * Replacing division with shift: Java doesn't optimize division by a power-of-two constant
     * ({@link Byte#SIZE}, in this case) to a raw shift because the arithmetic in Java is
     * always signed. See
     * https://lemire.me/blog/2017/05/09/signed-integer-division-by-a-power-of-two-can-be-expensive/
     */
    @CompileTimeConstant
    public static final int BYTE_SIZE_DIVISION_SHIFT = 3;

    static {
        verifyEqual(BYTE_SIZE_DIVISION_SHIFT, Integer.numberOfTrailingZeros(Byte.SIZE));
    }

    @Contract("false -> fail")
    static void verifyThat(boolean condition) {
        if (!condition) {
            throw new AssertionError();
        }
    }

    static void verifyEqual(int actual, int expected) {
        if (actual != expected) {
            throw new AssertionError("expected: " + expected + ", actual: " + actual);
        }
    }

    static void verifyEqual(long actual, long expected) {
        if (actual != expected) {
            throw new AssertionError("expected: " + expected + ", actual: " + actual);
        }
    }

    /**
     * The difference between this method and {@link java.util.Objects#requireNonNull(Object)} is
     * that this method doesn't return the argument back, resulting in less bytecodes that
     * ultimately makes SmoothieMap friendlier for method inlining, because inlining thresholds and
     * limits are defined in terms of the numbers of bytecodes in Hotspot.
     */
    static void checkNonNull(Object obj) {
        if (obj == null) {
            throw new NullPointerException();
        }
    }

    static void checkNonNegative(long value, String meaning) {
        if (value < 0) {
            throw new IllegalArgumentException(
                    meaning + " must be non-negative, " + value + " given");
        }
    }

    @Contract(value = "null -> fail; !null -> param1", pure = true)
    static  T nonNullOrThrowCme(@Nullable T obj) {
        if (obj == null) {
            throw new ConcurrentModificationException();
        }
        return obj;
    }

    @CanIgnoreReturnValue
    @Contract(value = "null -> fail; !null -> param1", pure = true)
    static  T verifyNonNull(@Nullable T obj) {
        if (obj == null) {
            throw new AssertionError();
        }
        return obj;
    }

    static void verifyIsPowerOfTwo(int n, String meaning) {
        if (!IntMath.isPowerOfTwo(n)) {
            throw new AssertionError(meaning + ": " + n);
        }
    }

    @SuppressWarnings("unchecked")
    static  void rethrowUnchecked(final Throwable t) throws T {
        throw (T) t;
    }

    @SuppressWarnings("SuspiciousSystemArraycopy")
    static void duplicateArray(Object array, int arrayLength, int filledLowPartLength) {
        for (; filledLowPartLength < arrayLength; filledLowPartLength *= 2) {
            System.arraycopy(array, 0, array, filledLowPartLength,
                    filledLowPartLength);
        }
    }

    private Utils() {}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy