2014-06-14 01:43:28 +02:00
|
|
|
// Auf QuarkCoin spezialisierte Version von Groestl inkl. Bitslice
|
2014-05-01 08:36:56 +02:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <memory.h>
|
2015-06-08 16:54:55 +02:00
|
|
|
#include <sys/types.h> // off_t
|
2014-05-01 08:36:56 +02:00
|
|
|
|
2014-08-18 03:45:48 +02:00
|
|
|
#include "cuda_helper.h"
|
|
|
|
|
2016-07-25 22:30:07 +02:00
|
|
|
#ifdef __INTELLISENSE__
|
|
|
|
#define __CUDA_ARCH__ 500
|
|
|
|
#endif
|
|
|
|
|
2014-08-19 15:34:15 +02:00
|
|
|
#define TPB 256
|
2016-07-25 22:30:07 +02:00
|
|
|
#define THF 4U
|
2014-08-19 15:34:15 +02:00
|
|
|
|
2015-01-18 23:00:03 +01:00
|
|
|
#if __CUDA_ARCH__ >= 300
|
2015-04-06 23:39:15 +02:00
|
|
|
#include "quark/groestl_functions_quad.h"
|
|
|
|
#include "quark/groestl_transf_quad.h"
|
2015-01-18 23:00:03 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "quark/cuda_quark_groestl512_sm20.cu"
|
2014-05-01 08:36:56 +02:00
|
|
|
|
2014-08-19 15:34:15 +02:00
|
|
|
__global__ __launch_bounds__(TPB, THF)
|
2016-07-25 22:30:07 +02:00
|
|
|
void quark_groestl512_gpu_hash_64_quad(const uint32_t threads, const uint32_t startNounce, uint32_t * g_hash, uint32_t * __restrict g_nonceVector)
|
2014-06-14 01:43:28 +02:00
|
|
|
{
|
2015-01-18 23:00:03 +01:00
|
|
|
#if __CUDA_ARCH__ >= 300
|
2014-06-14 01:43:28 +02:00
|
|
|
// durch 4 dividieren, weil jeweils 4 Threads zusammen ein Hash berechnen
|
2016-07-25 22:30:07 +02:00
|
|
|
const uint32_t thread = (blockDim.x * blockIdx.x + threadIdx.x) >> 2;
|
2014-06-14 01:43:28 +02:00
|
|
|
if (thread < threads)
|
2014-05-01 08:36:56 +02:00
|
|
|
{
|
2014-06-14 01:43:28 +02:00
|
|
|
// GROESTL
|
|
|
|
uint32_t message[8];
|
|
|
|
uint32_t state[8];
|
2014-05-01 08:36:56 +02:00
|
|
|
|
2014-08-22 19:42:57 +02:00
|
|
|
uint32_t nounce = g_nonceVector ? g_nonceVector[thread] : (startNounce + thread);
|
2015-05-30 14:57:52 +02:00
|
|
|
off_t hashPosition = nounce - startNounce;
|
|
|
|
uint32_t *pHash = &g_hash[hashPosition << 4];
|
2014-08-22 19:42:57 +02:00
|
|
|
|
2016-07-25 22:30:07 +02:00
|
|
|
const uint32_t thr = threadIdx.x & 0x3; // % THF
|
2014-05-01 08:36:56 +02:00
|
|
|
|
2014-08-22 19:42:57 +02:00
|
|
|
#pragma unroll
|
2015-05-30 14:57:52 +02:00
|
|
|
for(int k=0;k<4;k++) message[k] = pHash[thr + (k * THF)];
|
2014-08-22 19:42:57 +02:00
|
|
|
|
|
|
|
#pragma unroll
|
2014-06-14 01:43:28 +02:00
|
|
|
for(int k=4;k<8;k++) message[k] = 0;
|
2014-05-01 08:36:56 +02:00
|
|
|
|
2015-05-30 14:57:52 +02:00
|
|
|
if (thr == 0) message[4] = 0x80U;
|
|
|
|
if (thr == 3) message[7] = 0x01000000U;
|
2014-06-14 01:43:28 +02:00
|
|
|
|
|
|
|
uint32_t msgBitsliced[8];
|
|
|
|
to_bitslice_quad(message, msgBitsliced);
|
|
|
|
|
|
|
|
groestl512_progressMessage_quad(state, msgBitsliced);
|
|
|
|
|
|
|
|
// Nur der erste von jeweils 4 Threads bekommt das Ergebns-Hash
|
2016-07-25 22:30:07 +02:00
|
|
|
uint32_t __align__(16) hash[16];
|
2014-06-14 01:43:28 +02:00
|
|
|
from_bitslice_quad(state, hash);
|
|
|
|
|
2015-03-27 11:21:13 +01:00
|
|
|
// uint4 = 4x4 uint32_t = 16 bytes
|
|
|
|
if (thr == 0) {
|
|
|
|
uint4 *phash = (uint4*) hash;
|
2015-05-30 14:57:52 +02:00
|
|
|
uint4 *outpt = (uint4*) pHash;
|
2015-03-27 11:21:13 +01:00
|
|
|
outpt[0] = phash[0];
|
|
|
|
outpt[1] = phash[1];
|
|
|
|
outpt[2] = phash[2];
|
|
|
|
outpt[3] = phash[3];
|
|
|
|
}
|
2014-05-01 08:36:56 +02:00
|
|
|
}
|
2015-01-18 23:00:03 +01:00
|
|
|
#endif
|
2014-05-01 08:36:56 +02:00
|
|
|
}
|
|
|
|
|
2015-03-28 10:09:55 +01:00
|
|
|
__host__
|
|
|
|
void quark_groestl512_cpu_init(int thr_id, uint32_t threads)
|
2014-05-01 08:36:56 +02:00
|
|
|
{
|
2015-03-28 10:09:55 +01:00
|
|
|
int dev_id = device_map[thr_id];
|
|
|
|
cuda_get_arch(thr_id);
|
|
|
|
if (device_sm[dev_id] < 300 || cuda_arch[dev_id] < 300)
|
2015-01-18 23:00:03 +01:00
|
|
|
quark_groestl512_sm20_init(thr_id, threads);
|
2014-05-01 08:36:56 +02:00
|
|
|
}
|
|
|
|
|
2015-10-08 21:31:16 +02:00
|
|
|
__host__
|
|
|
|
void quark_groestl512_cpu_free(int thr_id)
|
|
|
|
{
|
|
|
|
int dev_id = device_map[thr_id];
|
|
|
|
if (device_sm[dev_id] < 300 || cuda_arch[dev_id] < 300)
|
|
|
|
quark_groestl512_sm20_free(thr_id);
|
|
|
|
}
|
|
|
|
|
2015-03-28 10:09:55 +01:00
|
|
|
__host__
|
|
|
|
void quark_groestl512_cpu_hash_64(int thr_id, uint32_t threads, uint32_t startNounce, uint32_t *d_nonceVector, uint32_t *d_hash, int order)
|
2014-05-01 08:36:56 +02:00
|
|
|
{
|
2016-07-25 22:30:07 +02:00
|
|
|
uint32_t threadsperblock = TPB;
|
2014-06-14 01:43:28 +02:00
|
|
|
|
|
|
|
// Compute 3.0 benutzt die registeroptimierte Quad Variante mit Warp Shuffle
|
|
|
|
// mit den Quad Funktionen brauchen wir jetzt 4 threads pro Hash, daher Faktor 4 bei der Blockzahl
|
2016-07-25 22:30:07 +02:00
|
|
|
const uint32_t factor = THF;
|
2014-05-01 08:36:56 +02:00
|
|
|
|
|
|
|
// berechne wie viele Thread Blocks wir brauchen
|
2014-06-14 01:43:28 +02:00
|
|
|
dim3 grid(factor*((threads + threadsperblock-1)/threadsperblock));
|
2014-05-01 08:36:56 +02:00
|
|
|
dim3 block(threadsperblock);
|
|
|
|
|
2015-03-28 10:09:55 +01:00
|
|
|
int dev_id = device_map[thr_id];
|
2014-05-01 08:36:56 +02:00
|
|
|
|
2015-03-28 10:09:55 +01:00
|
|
|
if (device_sm[dev_id] >= 300 && cuda_arch[dev_id] >= 300)
|
|
|
|
quark_groestl512_gpu_hash_64_quad<<<grid, block>>>(threads, startNounce, d_hash, d_nonceVector);
|
2015-01-18 23:00:03 +01:00
|
|
|
else
|
|
|
|
quark_groestl512_sm20_hash_64(thr_id, threads, startNounce, d_nonceVector, d_hash, order);
|
2014-05-01 08:36:56 +02:00
|
|
|
|
|
|
|
// Strategisches Sleep Kommando zur Senkung der CPU Last
|
2016-07-25 22:30:07 +02:00
|
|
|
// MyStreamSynchronize(NULL, order, thr_id);
|
2014-05-01 08:36:56 +02:00
|
|
|
}
|