You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

76 lines
1.6 KiB

#include "ethash_cuda_miner_kernel_globals.h"
#include "ethash_cuda_miner_kernel.h"
#include "keccak_u64.cuh"
#include "fnv.cuh"
#define copy(dst, src, count) for (int i = 0; i != count; ++i) { (dst)[i] = (src)[i]; }
typedef union {
uint4 uint4s[4];
uint64_t ulongs[8];
uint32_t uints[16];
} compute_hash_share;
__device__ uint64_t compute_hash(
uint64_t nonce
)
{
// sha3_512(header .. nonce)
uint64_t state[25];
state[4] = nonce;
keccak_f1600_init(state);
// Threads work together in this phase in groups of 8.
const int thread_id = threadIdx.x & (THREADS_PER_HASH - 1);
const int hash_id = threadIdx.x >> 3;
extern __shared__ compute_hash_share share[];
for (int i = 0; i < THREADS_PER_HASH; i++)
{
// share init with other threads
if (i == thread_id)
copy(share[hash_id].ulongs, state, 8);
__syncthreads();
uint4 mix = share[hash_id].uint4s[thread_id & 3];
__syncthreads();
uint32_t *share0 = share[hash_id].uints;
// share init0
if (thread_id == 0)
*share0 = mix.x;
__syncthreads();
uint32_t init0 = *share0;
for (uint32_t a = 0; a < ACCESSES; a += 4)
{
bool update_share = thread_id == ((a >> 2) & (THREADS_PER_HASH - 1));
for (uint32_t i = 0; i != 4; ++i)
{
if (update_share)
{
*share0 = fnv(init0 ^ (a + i), ((uint32_t *)&mix)[i]) % d_dag_size;
}
__syncthreads();
mix = fnv4(mix, d_dag[*share0].uint4s[thread_id]);
}
}
share[hash_id].uints[thread_id] = fnv_reduce(mix);
__syncthreads();
if (i == thread_id)
copy(state + 8, share[hash_id].ulongs, 4);
__syncthreads();
}
// keccak_256(keccak_512(header..nonce) .. mix);
return keccak_f1600_final(state);
}