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

org.ethereum.mine.EthashAlgo Maven / Gradle / Ivy

Go to download

Java implementation of the Ethereum protocol adapted to use for Hedera Smart Contract Service

The newest version!
/*
 * Copyright (c) [2016] [  ]
 * This file is part of the ethereumJ library.
 *
 * The ethereumJ library is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * The ethereumJ library 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 Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with the ethereumJ library. If not, see .
 */
package org.ethereum.mine;

import org.apache.commons.lang3.tuple.Pair;
import org.ethereum.crypto.HashUtil;
import org.spongycastle.util.Arrays;

import java.math.BigInteger;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.util.Random;

import static java.lang.System.arraycopy;
import static java.math.BigInteger.valueOf;
import static org.ethereum.crypto.HashUtil.sha3;
import static org.ethereum.util.ByteUtil.*;
import static org.spongycastle.util.Arrays.reverse;

/**
 * The Ethash algorithm described in https://github.com/ethereum/wiki/wiki/Ethash
 *
 * Created by Anton Nashatyrev on 27.11.2015.
 */
public class EthashAlgo {
    EthashParams params;

    public EthashAlgo() {
        this(new EthashParams());
    }

    public EthashAlgo(EthashParams params) {
        this.params = params;
    }

    public EthashParams getParams() {
        return params;
    }

    // Little-Endian !
    static int getWord(byte[] arr, int wordOff) {
        return ByteBuffer.wrap(arr, wordOff * 4, 4).order(ByteOrder.LITTLE_ENDIAN).getInt();
    }

    static void setWord(byte[] arr, int wordOff, long val) {
        ByteBuffer bb = ByteBuffer.allocate(4).order(ByteOrder.LITTLE_ENDIAN).putInt((int) val);
        bb.rewind();
        bb.get(arr, wordOff * 4, 4);
    }

    public static int remainderUnsigned(int dividend, int divisor) {
        if (divisor >= 0) {
            if (dividend >= 0) {
                return dividend % divisor;
            }
            // The implementation is a Java port of algorithm described in the book
            // "Hacker's Delight" (section "Unsigned short division from signed division").
            int q = ((dividend >>> 1) / divisor) << 1;
            dividend -= q * divisor;
            if (dividend < 0 || dividend >= divisor) {
                dividend -= divisor;
            }
            return dividend;
        }
        return dividend >= 0 || dividend < divisor ? dividend : dividend - divisor;
    }


    private byte[][] makeCacheBytes(long cacheSize, byte[] seed) {
        int n = (int) (cacheSize / params.getHASH_BYTES());
        byte[][] o = new byte[n][];
        o[0] = HashUtil.sha512(seed);
        for (int i = 1; i < n; i++) {
            o[i] = HashUtil.sha512(o[i - 1]);
        }

        for (int cacheRound = 0; cacheRound < params.getCACHE_ROUNDS(); cacheRound++) {
            for (int i = 0; i < n; i++) {
                int v = remainderUnsigned(getWord(o[i], 0), n);
                o[i] = HashUtil.sha512(xor(o[(i - 1 + n) % n], o[v]));
            }
        }
        return o;
    }

    public int[] makeCache(long cacheSize, byte[] seed) {
        byte[][] bytes = makeCacheBytes(cacheSize, seed);
        int[] ret = new int[bytes.length * bytes[0].length / 4];
        int[] ints = new int[bytes[0].length / 4];
        for (int i = 0; i < bytes.length; i++) {
            bytesToInts(bytes[i], ints, false);
            arraycopy(ints, 0, ret, i * ints.length, ints.length);
        }
        return ret;
    }

    private static final int FNV_PRIME = 0x01000193;
    private static int fnv(int v1, int v2) {
        return (v1 * FNV_PRIME) ^ v2;
    }

    int[] sha512(int[] arr, boolean bigEndian) {
        byte[] bytesTmp = new byte[arr.length << 2];
        intsToBytes(arr, bytesTmp, bigEndian);
        bytesTmp = HashUtil.sha512(bytesTmp);
        bytesToInts(bytesTmp, arr, bigEndian);
        return arr;
    }

    public final int[] calcDatasetItem(final int[] cache, final int i) {
        final int r = params.getHASH_BYTES() / params.getWORD_BYTES();
        final int n = cache.length / r;
        int[] mix = Arrays.copyOfRange(cache, i % n * r, (i % n + 1) * r);

        mix[0] = i ^ mix[0];
        mix = sha512(mix, false);
        final int dsParents = (int) params.getDATASET_PARENTS();
        final int mixLen = mix.length;
        for (int j = 0; j < dsParents; j++) {
            int cacheIdx = fnv(i ^ j, mix[j % r]);
            cacheIdx = remainderUnsigned(cacheIdx, n);
            int off = cacheIdx * r;
            for (int k = 0; k < mixLen; k++) {
                mix[k] = fnv(mix[k], cache[off + k]);
            }
        }
        return sha512(mix, false);
    }

    public int[] calcDataset(long fullSize, int[] cache) {
        int hashesCount = (int) (fullSize / params.getHASH_BYTES());
        int[] ret = new int[hashesCount * (params.getHASH_BYTES() / 4)];
        for (int i = 0; i < hashesCount; i++) {
            int[] item = calcDatasetItem(cache, i);
            arraycopy(item, 0, ret, i * (params.getHASH_BYTES() / 4), item.length);
        }
        return ret;
    }

    public Pair hashimoto(byte[] blockHeaderTruncHash, byte[] nonce, long fullSize,
                                          int[] cacheOrDataset, boolean full) {
        if (nonce.length != 8) throw new RuntimeException("nonce.length != 8");

        int hashWords = params.getHASH_BYTES() / 4;
        int w = params.getMIX_BYTES() / params.getWORD_BYTES();
        int mixhashes = params.getMIX_BYTES() / params.getHASH_BYTES();
        int[] s = bytesToInts(HashUtil.sha512(merge(blockHeaderTruncHash, reverse(nonce))), false);
        int[] mix = new int[params.getMIX_BYTES() / 4];
        for (int i = 0; i < mixhashes; i++) {
            arraycopy(s, 0, mix, i * s.length, s.length);
        }

        int numFullPages = (int) (fullSize / params.getMIX_BYTES());
        for (int i = 0; i < params.getACCESSES(); i++) {
            int p = remainderUnsigned(fnv(i ^ s[0], mix[i % w]), numFullPages);
            int[] newData = new int[mix.length];
            int off = p * mixhashes;
            for (int j = 0; j < mixhashes; j++) {
                int itemIdx = off + j;
                if (!full) {
                    int[] lookup1 = calcDatasetItem(cacheOrDataset, itemIdx);
                    arraycopy(lookup1, 0, newData, j * lookup1.length, lookup1.length);
                } else {
                    arraycopy(cacheOrDataset, itemIdx * hashWords, newData, j * hashWords, hashWords);
                }
            }
            for (int i1 = 0; i1 < mix.length; i1++) {
                mix[i1] = fnv(mix[i1], newData[i1]);
            }
        }

        int[] cmix = new int[mix.length / 4];
        for (int i = 0; i < mix.length; i += 4 /* ? */) {
            int fnv1 = fnv(mix[i], mix[i + 1]);
            int fnv2 = fnv(fnv1, mix[i + 2]);
            int fnv3 = fnv(fnv2, mix[i + 3]);
            cmix[i >> 2] = fnv3;
        }

        return Pair.of(intsToBytes(cmix, false), sha3(merge(intsToBytes(s, false), intsToBytes(cmix, false))));
    }

    public Pair hashimotoLight(long fullSize, final int[] cache, byte[] blockHeaderTruncHash,
                                               byte[]  nonce) {
        return hashimoto(blockHeaderTruncHash, nonce, fullSize, cache, false);
    }

    public Pair hashimotoFull(long fullSize, final int[] dataset, byte[] blockHeaderTruncHash,
                                              byte[]  nonce) {
        return hashimoto(blockHeaderTruncHash, nonce, fullSize, dataset, true);
    }

    public long mine(long fullSize, int[] dataset, byte[] blockHeaderTruncHash, long difficulty) {
        return mine(fullSize, dataset, blockHeaderTruncHash, difficulty, new Random().nextLong());
    }

    public long mine(long fullSize, int[] dataset, byte[] blockHeaderTruncHash, long difficulty, long startNonce) {
        long nonce = startNonce;
        BigInteger target = valueOf(2).pow(256).divide(valueOf(difficulty));
        while (!Thread.currentThread().isInterrupted()) {
            nonce++;
            Pair pair = hashimotoFull(fullSize, dataset, blockHeaderTruncHash, longToBytes(nonce));
            BigInteger h = new BigInteger(1, pair.getRight() /* ?? */);
            if (h.compareTo(target) < 0) break;
        }
        return nonce;
    }

    /**
     * This the slower miner version which uses only cache thus taking much less memory than
     * regular {@link #mine} method
     */
    public long mineLight(long fullSize, final int[] cache, byte[] blockHeaderTruncHash, long difficulty) {
        return mineLight(fullSize, cache, blockHeaderTruncHash, difficulty, new Random().nextLong());
    }

    public long mineLight(long fullSize, final int[] cache, byte[] blockHeaderTruncHash, long difficulty, long startNonce) {
        long nonce = startNonce;
        BigInteger target = valueOf(2).pow(256).divide(valueOf(difficulty));
        while(!Thread.currentThread().isInterrupted()) {
            nonce++;
            Pair pair = hashimotoLight(fullSize, cache, blockHeaderTruncHash, longToBytes(nonce));
            BigInteger h = new BigInteger(1, pair.getRight() /* ?? */);
            if (h.compareTo(target) < 0) break;
        }
        return nonce;
    }

    public byte[] getSeedHash(long blockNumber) {
        byte[] ret = new byte[32];
        for (int i = 0; i < blockNumber / params.getEPOCH_LENGTH(); i++) {
            ret = sha3(ret);
        }
        return ret;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy