2014-04-27 01:26:08 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <memory.h>
|
|
|
|
|
2014-08-18 03:45:48 +02:00
|
|
|
#include "cuda_helper.h"
|
|
|
|
|
2014-08-08 08:06:06 +02:00
|
|
|
// Hash Target gegen das wir testen sollen
|
2014-04-27 01:26:08 +02:00
|
|
|
__constant__ uint32_t pTarget[8];
|
|
|
|
|
2014-08-31 08:57:48 +02:00
|
|
|
static uint32_t *d_resNounce[8];
|
|
|
|
static uint32_t *h_resNounce[8];
|
2014-04-27 01:26:08 +02:00
|
|
|
|
2014-09-09 23:04:32 +02:00
|
|
|
__global__
|
|
|
|
void cuda_check_gpu_hash_64(int threads, uint32_t startNounce, uint32_t *g_nonceVector, uint32_t *g_hash, uint32_t *resNounce)
|
2014-04-27 01:26:08 +02:00
|
|
|
{
|
|
|
|
int thread = (blockDim.x * blockIdx.x + threadIdx.x);
|
|
|
|
if (thread < threads)
|
|
|
|
{
|
2014-09-09 23:04:32 +02:00
|
|
|
// bestimme den aktuellen Zähler
|
2014-04-27 01:26:08 +02:00
|
|
|
uint32_t nounce = (g_nonceVector != NULL) ? g_nonceVector[thread] : (startNounce + thread);
|
|
|
|
|
2014-11-09 10:55:35 +01:00
|
|
|
uint32_t hashPosition = (nounce - startNounce) << 4;
|
|
|
|
uint32_t *inpHash = &g_hash[hashPosition];
|
2014-04-27 01:26:08 +02:00
|
|
|
uint32_t hash[8];
|
2014-09-09 23:04:32 +02:00
|
|
|
|
|
|
|
#pragma unroll 8
|
2014-04-27 01:26:08 +02:00
|
|
|
for (int i=0; i < 8; i++)
|
|
|
|
hash[i] = inpHash[i];
|
|
|
|
|
2014-09-09 23:04:32 +02:00
|
|
|
for (int i = 7; i >= 0; i--) {
|
2014-04-27 01:26:08 +02:00
|
|
|
if (hash[i] > pTarget[i]) {
|
2014-09-09 23:04:32 +02:00
|
|
|
return;
|
|
|
|
}
|
2014-11-09 10:55:35 +01:00
|
|
|
if (hash[i] <= pTarget[i]) {
|
2014-09-09 23:04:32 +02:00
|
|
|
break;
|
|
|
|
}
|
2014-04-27 01:26:08 +02:00
|
|
|
}
|
2014-11-09 10:55:35 +01:00
|
|
|
if (resNounce[0] > nounce)
|
2014-09-09 23:04:32 +02:00
|
|
|
resNounce[0] = nounce;
|
2014-04-27 01:26:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Setup-Funktionen
|
2014-09-09 23:04:32 +02:00
|
|
|
__host__
|
|
|
|
void cuda_check_cpu_init(int thr_id, int threads)
|
2014-04-27 01:26:08 +02:00
|
|
|
{
|
2014-11-20 17:34:37 +01:00
|
|
|
CUDA_CALL_OR_RET(cudaMallocHost(&h_resNounce[thr_id], 1*sizeof(uint32_t)));
|
|
|
|
CUDA_CALL_OR_RET(cudaMalloc(&d_resNounce[thr_id], 1*sizeof(uint32_t)));
|
2014-04-27 01:26:08 +02:00
|
|
|
}
|
|
|
|
|
2014-11-20 17:34:37 +01:00
|
|
|
// Target Difficulty
|
2014-09-09 23:04:32 +02:00
|
|
|
__host__
|
|
|
|
void cuda_check_cpu_setTarget(const void *ptarget)
|
2014-04-27 01:26:08 +02:00
|
|
|
{
|
2014-11-09 10:55:35 +01:00
|
|
|
CUDA_SAFE_CALL(cudaMemcpyToSymbol(pTarget, ptarget, 8*sizeof(uint32_t), 0, cudaMemcpyHostToDevice));
|
2014-04-27 01:26:08 +02:00
|
|
|
}
|
|
|
|
|
2014-09-09 23:04:32 +02:00
|
|
|
__host__
|
|
|
|
uint32_t cuda_check_cpu_hash_64(int thr_id, int threads, uint32_t startNounce, uint32_t *d_nonceVector, uint32_t *d_inputHash, int order)
|
2014-04-27 01:26:08 +02:00
|
|
|
{
|
|
|
|
uint32_t result = 0xffffffff;
|
|
|
|
cudaMemset(d_resNounce[thr_id], 0xff, sizeof(uint32_t));
|
|
|
|
|
|
|
|
const int threadsperblock = 256;
|
|
|
|
|
|
|
|
dim3 grid((threads + threadsperblock-1)/threadsperblock);
|
|
|
|
dim3 block(threadsperblock);
|
|
|
|
|
2014-11-09 10:55:35 +01:00
|
|
|
cuda_check_gpu_hash_64 <<<grid, block>>> (threads, startNounce, d_nonceVector, d_inputHash, d_resNounce[thr_id]);
|
2014-04-27 01:26:08 +02:00
|
|
|
|
|
|
|
// Strategisches Sleep Kommando zur Senkung der CPU Last
|
|
|
|
MyStreamSynchronize(NULL, order, thr_id);
|
|
|
|
|
|
|
|
// Ergebnis zum Host kopieren (in page locked memory, damits schneller geht)
|
|
|
|
cudaMemcpy(h_resNounce[thr_id], d_resNounce[thr_id], sizeof(uint32_t), cudaMemcpyDeviceToHost);
|
|
|
|
|
|
|
|
// cudaMemcpy() ist asynchron!
|
|
|
|
cudaThreadSynchronize();
|
|
|
|
result = *h_resNounce[thr_id];
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2014-11-09 10:55:35 +01:00
|
|
|
|
|
|
|
__global__
|
|
|
|
void cuda_check_gpu_hash_fast(int threads, uint32_t startNounce, uint32_t *hashEnd, uint32_t *resNounce)
|
|
|
|
{
|
|
|
|
int thread = (blockDim.x * blockIdx.x + threadIdx.x);
|
|
|
|
if (thread < threads)
|
|
|
|
{
|
|
|
|
/* only test the last 2 dwords, ok for most algos */
|
|
|
|
int hashPos = thread << 4;
|
|
|
|
uint32_t *inpHash = &hashEnd[hashPos];
|
|
|
|
|
|
|
|
if (inpHash[7] <= pTarget[7] && inpHash[6] <= pTarget[6]) {
|
|
|
|
uint32_t nounce = (startNounce + thread);
|
|
|
|
if (resNounce[0] > nounce)
|
|
|
|
resNounce[0] = nounce;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
__host__
|
|
|
|
uint32_t cuda_check_hash_fast(int thr_id, int threads, uint32_t startNounce, uint32_t *d_inputHash, int order)
|
|
|
|
{
|
|
|
|
uint32_t result = 0xffffffff;
|
|
|
|
cudaMemset(d_resNounce[thr_id], 0xff, sizeof(uint32_t));
|
|
|
|
|
|
|
|
const int threadsperblock = 256;
|
|
|
|
|
|
|
|
dim3 grid((threads + threadsperblock - 1) / threadsperblock);
|
|
|
|
dim3 block(threadsperblock);
|
|
|
|
|
|
|
|
cuda_check_gpu_hash_fast <<<grid, block>>> (threads, startNounce, d_inputHash, d_resNounce[thr_id]);
|
|
|
|
|
|
|
|
// MyStreamSynchronize(NULL, order, thr_id);
|
|
|
|
cudaThreadSynchronize();
|
|
|
|
|
|
|
|
cudaMemcpy(h_resNounce[thr_id], d_resNounce[thr_id], sizeof(uint32_t), cudaMemcpyDeviceToHost);
|
|
|
|
|
|
|
|
// cudaMemcpy() was asynchron ?
|
|
|
|
// cudaThreadSynchronize();
|
|
|
|
result = *h_resNounce[thr_id];
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|