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

org.neo4j.hashing.WyHash Maven / Gradle / Ivy

/*
 * Copyright (c) "Neo4j"
 * Neo4j Sweden AB [https://neo4j.com]
 *
 * This file is part of Neo4j.
 *
 * Neo4j is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
package org.neo4j.hashing;

import java.lang.invoke.MethodHandles;
import java.lang.invoke.VarHandle;
import java.nio.ByteOrder;

public final class WyHash {
    private static final VarHandle LE_INTEGER = MethodHandles.byteArrayViewVarHandle(
                    int[].class, ByteOrder.LITTLE_ENDIAN)
            .withInvokeExactBehavior();
    private static final VarHandle LE_LONG = MethodHandles.byteArrayViewVarHandle(long[].class, ByteOrder.LITTLE_ENDIAN)
            .withInvokeExactBehavior();
    private static final long WYP_0 = 0xa0761d6478bd642fL;
    private static final long WYP_1 = 0xe7037ed1a0b428dbL;
    private static final long WYP_2 = 0x8ebc6af09c88c6e3L;
    private static final long WYP_3 = 0x589965cc75374cc3L;
    private static final long WYP_4 = 0x1d8e4e27c47d124fL;

    private WyHash() {}

    public static long hashLong(long input) {
        long hi = input & 0xFFFFFFFFL;
        long lo = (input >>> 32) & 0xFFFFFFFFL;
        return wymum(wymum(hi ^ WYP_0, lo ^ WYP_1), 8 ^ WYP_4);
    }

    public static long hashInt(int input) {
        long longInput = (input & 0xFFFFFFFFL);
        return wymum(wymum(longInput ^ WYP_0, longInput ^ WYP_1), 4 ^ WYP_4);
    }

    public static long hashShort(short input) {
        long hi = (input >>> 8) & 0xFFL;
        long wyr3 = hi | hi << 8 | (input & 0xFFL) << 16;
        return wymum(wymum(wyr3 ^ WYP_0, WYP_1), 2 ^ WYP_4);
    }

    public static long hashChar(final char input) {
        return hashShort((short) input);
    }

    public static long hashByte(final byte input) {
        long hi = input & 0xFFL;
        long wyr3 = hi | hi << 8 | hi << 16;
        return wymum(wymum(wyr3 ^ WYP_0, WYP_1), 1 ^ WYP_4);
    }

    public static long hash(final byte[] input, final int off, final int len) {
        return WyHash.wyHash64(input, off, len);
    }

    private static long unsignedLongMulXorFold(final long lhs, final long rhs) {
        final long upper = Math.multiplyHigh(lhs, rhs) + ((lhs >> 63) & rhs) + ((rhs >> 63) & lhs);
        final long lower = lhs * rhs;
        return lower ^ upper;
    }

    private static long wymum(final long lhs, final long rhs) {
        return unsignedLongMulXorFold(lhs, rhs);
    }

    private static long wyr3(final byte[] in, final int index, final int k) {
        return ((long) in[index] << 16) | ((long) in[index + (k >>> 1)] << 8) | ((long) in[index + k - 1]);
    }

    private static long u64Rorate32(final byte[] in, final int index) {
        return (u32(in, index) << 32) | u32(in, index + 4);
    }

    private static long u32(final byte[] input, final int offset) {
        return (int) LE_INTEGER.get(input, offset) & 0xFFFFFFFFL;
    }

    private static long i64(final byte[] input, final int offset) {
        return (long) LE_LONG.get(input, offset);
    }

    /**
     *
     * @param input the type wrapped by the Access, ex. byte[], ByteBuffer, etc.
     * @param off offset to the input
     * @param length length to read from input
     * @return hash result
     */
    static long wyHash64(final byte[] input, final int off, final int length) {
        if (length <= 0) {
            return 0;
        } else if (length < 4) {
            return wymum(wymum(wyr3(input, off, length) ^ WYP_0, WYP_1), length ^ WYP_4);
        } else if (length <= 8) {
            return wymum(wymum(u32(input, off) ^ WYP_0, u32(input, off + length - 4) ^ WYP_1), length ^ WYP_4);
        } else if (length <= 16) {
            return wymum(
                    wymum(u64Rorate32(input, off) ^ WYP_0, u64Rorate32(input, off + length - 8) ^ WYP_1),
                    length ^ WYP_4);
        } else if (length <= 24) {
            return wymum(
                    wymum(u64Rorate32(input, off) ^ WYP_0, u64Rorate32(input, off + 8) ^ WYP_1)
                            ^ wymum(u64Rorate32(input, off + length - 8) ^ WYP_2, WYP_3),
                    length ^ WYP_4);
        } else if (length <= 32) {
            return wymum(
                    wymum(u64Rorate32(input, off) ^ WYP_0, u64Rorate32(input, off + 8) ^ WYP_1)
                            ^ wymum(u64Rorate32(input, off + 16) ^ WYP_2, u64Rorate32(input, off + length - 8) ^ WYP_3),
                    length ^ WYP_4);
        }
        long seed = 0;
        long see1 = 0;
        int i = length, p = off;
        for (; i > 256; i -= 256, p += 256) {
            seed = wymum(i64(input, p) ^ seed ^ WYP_0, i64(input, p + 8) ^ seed ^ WYP_1)
                    ^ wymum(i64(input, p + 16) ^ seed ^ WYP_2, i64(input, p + 24) ^ seed ^ WYP_3);
            see1 = wymum(i64(input, p + 32) ^ see1 ^ WYP_1, i64(input, p + 40) ^ see1 ^ WYP_2)
                    ^ wymum(i64(input, p + 48) ^ see1 ^ WYP_3, i64(input, p + 56) ^ see1 ^ WYP_0);
            seed = wymum(i64(input, p + 64) ^ seed ^ WYP_0, i64(input, p + 72) ^ seed ^ WYP_1)
                    ^ wymum(i64(input, p + 80) ^ seed ^ WYP_2, i64(input, p + 88) ^ seed ^ WYP_3);
            see1 = wymum(i64(input, p + 96) ^ see1 ^ WYP_1, i64(input, p + 104) ^ see1 ^ WYP_2)
                    ^ wymum(i64(input, p + 112) ^ see1 ^ WYP_3, i64(input, p + 120) ^ see1 ^ WYP_0);
            seed = wymum(i64(input, p + 128) ^ seed ^ WYP_0, i64(input, p + 136) ^ seed ^ WYP_1)
                    ^ wymum(i64(input, p + 144) ^ seed ^ WYP_2, i64(input, p + 152) ^ seed ^ WYP_3);
            see1 = wymum(i64(input, p + 160) ^ see1 ^ WYP_1, i64(input, p + 168) ^ see1 ^ WYP_2)
                    ^ wymum(i64(input, p + 176) ^ see1 ^ WYP_3, i64(input, p + 184) ^ see1 ^ WYP_0);
            seed = wymum(i64(input, p + 192) ^ seed ^ WYP_0, i64(input, p + 200) ^ seed ^ WYP_1)
                    ^ wymum(i64(input, p + 208) ^ seed ^ WYP_2, i64(input, p + 216) ^ seed ^ WYP_3);
            see1 = wymum(i64(input, p + 224) ^ see1 ^ WYP_1, i64(input, p + 232) ^ see1 ^ WYP_2)
                    ^ wymum(i64(input, p + 240) ^ see1 ^ WYP_3, i64(input, p + 248) ^ see1 ^ WYP_0);
        }
        for (; i > 32; i -= 32, p += 32) {
            seed = wymum(i64(input, p) ^ seed ^ WYP_0, i64(input, p + 8) ^ seed ^ WYP_1);
            see1 = wymum(i64(input, p + 16) ^ see1 ^ WYP_2, i64(input, p + 24) ^ see1 ^ WYP_3);
        }
        if (i < 4) {
            seed = wymum(wyr3(input, p, i) ^ seed ^ WYP_0, seed ^ WYP_1);
        } else if (i <= 8) {
            seed = wymum(u32(input, p) ^ seed ^ WYP_0, u32(input, p + i - 4) ^ seed ^ WYP_1);
        } else if (i <= 16) {
            seed = wymum(u64Rorate32(input, p) ^ seed ^ WYP_0, u64Rorate32(input, p + i - 8) ^ seed ^ WYP_1);
        } else if (i <= 24) {
            seed = wymum(u64Rorate32(input, p) ^ seed ^ WYP_0, u64Rorate32(input, p + 8) ^ seed ^ WYP_1);
            see1 = wymum(u64Rorate32(input, p + i - 8) ^ see1 ^ WYP_2, see1 ^ WYP_3);
        } else {
            seed = wymum(u64Rorate32(input, p) ^ seed ^ WYP_0, u64Rorate32(input, p + 8) ^ seed ^ WYP_1);
            see1 = wymum(u64Rorate32(input, p + 16) ^ see1 ^ WYP_2, u64Rorate32(input, p + i - 8) ^ see1 ^ WYP_3);
        }
        return wymum(seed ^ see1, length ^ WYP_4);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy