/* * This file is part of the Monero P2Pool * Copyright (c) 2021-2025 SChernykh * * This program 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, version 3. * * 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 . */ #pragma once namespace p2pool { enum KeccakParams { HASH_DATA_AREA = 136, ROUNDS = 24, }; extern const uint64_t keccakf_rndc[24]; // keccak hash of a single 0x00 byte constexpr hash keccak_0x00{ 0xbc, 0x36, 0x78, 0x9e, 0x7a, 0x1e, 0x28, 0x14, 0x36, 0x46, 0x42, 0x29, 0x82, 0x8f, 0x81, 0x7d, 0x66, 0x12, 0xf7, 0xb4, 0x77, 0xd6, 0x65, 0x91, 0xff, 0x96, 0xa9, 0xe0, 0x64, 0xbc, 0xc9, 0x8a }; typedef void (*keccakf_func)(std::array&); extern keccakf_func keccakf; void keccakf_plain(std::array& st); void keccakf_bmi(std::array& st); void keccak_step(const uint8_t* &in, int &inlen, std::array& st); void keccak_finish(const uint8_t* in, int inlen, std::array& st); template FORCEINLINE void keccak(const uint8_t* in, int inlen, uint8_t (&md)[N]) { static_assert((N == 32) || (N == 200), "invalid size"); std::array st = {}; keccak_step(in, inlen, st); keccak_finish(in, inlen, st); memcpy(md, st.data(), N); } template FORCEINLINE void keccak_custom(T&& in, int inlen, uint8_t* md, int mdlen) { std::array st = {}; const int rsiz = sizeof(st) == mdlen ? KeccakParams::HASH_DATA_AREA : 200 - 2 * mdlen; const int rsizw = rsiz / 8; int offset = 0; for (; inlen >= rsiz; inlen -= rsiz, offset += rsiz) { for (int i = 0; i < rsizw; ++i) { uint64_t k = 0; for (int j = 0; j < 8; ++j) { k |= static_cast(in(offset + i * 8 + j)) << (j * 8); } st[i] ^= k; } keccakf(st); } // last block and padding alignas(8) uint8_t temp[144]; for (int i = 0; i < inlen; ++i) { temp[i] = in(offset + i); } temp[inlen++] = 1; memset(temp + inlen, 0, rsiz - inlen); temp[rsiz - 1] |= 0x80; for (int i = 0; i < rsizw; i++) { st[i] ^= reinterpret_cast(temp)[i]; } keccakf(st); memcpy(md, st.data(), mdlen); } } // namespace p2pool