Compute and display network diff (nbits) if max-diff

was only made on the wallets with getmininginfo

Todo: double check for all algos/pools

tested ok on x11, lyra2, skein, skein2, zr5 stratum/longpoll

Scrypt-Jane pool net diff seems incorrect, disabled
Neoscrypt could be different also (untested, disabled)...
This commit is contained in:
Tanguy Pruvot 2015-05-18 07:37:57 +02:00
parent 87a521ac14
commit a2b6488c01
5 changed files with 98 additions and 27 deletions

View File

@ -177,7 +177,7 @@ static char *getsummary(char *params)
PACKAGE_NAME, PACKAGE_VERSION, APIVERSION, PACKAGE_NAME, PACKAGE_VERSION, APIVERSION,
algo, active_gpus, (double)global_hashrate / 1000.0, algo, active_gpus, (double)global_hashrate / 1000.0,
accepted_count, rejected_count, accepted_count, rejected_count,
accps, global_diff, uptime, (uint32_t) ts); accps, net_diff > 0. ? net_diff : stratum_diff, uptime, (uint32_t) ts);
return buffer; return buffer;
} }

View File

@ -235,7 +235,8 @@ uint32_t accepted_count = 0L;
uint32_t rejected_count = 0L; uint32_t rejected_count = 0L;
static double thr_hashrates[MAX_GPUS] = { 0 }; static double thr_hashrates[MAX_GPUS] = { 0 };
uint64_t global_hashrate = 0; uint64_t global_hashrate = 0;
double global_diff = 0.0; double stratum_diff = 0.0;
double net_diff = 0;
uint64_t net_hashrate = 0; uint64_t net_hashrate = 0;
uint64_t net_blocks = 0; uint64_t net_blocks = 0;
// conditional mining // conditional mining
@ -537,6 +538,48 @@ static bool jobj_binary(const json_t *obj, const char *key,
return true; return true;
} }
/* compute nbits to get the network diff */
static void calc_network_diff(struct work *work)
{
// sample for diff 43.281 : 1c05ea29
const uint64_t diffone = 0xFFFF000000000000ull;
uint64_t *data64, d64;
uint32_t nbits = have_stratum ? swab32(work->data[18]) : work->data[18];
uint32_t shift = (swab32(nbits) & 0xff); // 0x1c = 28
uint32_t bits = (nbits & 0xffffff);
uchar rtarget[48] = { 0 };
switch (opt_algo) {
case ALGO_NEOSCRYPT:
case ALGO_SCRYPT:
case ALGO_SCRYPT_JANE: /* bad value - 1.16 should be 0.01206 */
// todo/check...
if (opt_debug)
applog(LOG_DEBUG, "diff shift: %08x -> %u %08x -> %.3f", nbits, shift, bits, net_diff);
net_diff = 0.;
return;
}
if (shift >= 3 && shift < sizeof(rtarget)-3) {
memcpy(&rtarget[shift - 3], &bits, 3); // 0029ea05 00000000
}
swab256(rtarget, rtarget);
switch (opt_algo) {
case ALGO_HEAVY:
data64 = (uint64_t *)(rtarget + 2);
break;
default:
data64 = (uint64_t *)(rtarget + 4);
}
d64 = swab64(*data64);
if (!d64)
d64 = 1;
net_diff = (double)diffone / d64; // 43.281
//applog(LOG_DEBUG, "diff shift: %08x -> %u %08x -> %.3f", nbits, shift, bits, net_diff);
}
static bool work_decode(const json_t *val, struct work *work) static bool work_decode(const json_t *val, struct work *work)
{ {
int data_size = sizeof(work->data), target_size = sizeof(work->target); int data_size = sizeof(work->data), target_size = sizeof(work->target);
@ -567,6 +610,9 @@ static bool work_decode(const json_t *val, struct work *work)
for (i = 0; i < atarget_sz; i++) for (i = 0; i < atarget_sz; i++)
work->target[i] = le32dec(work->target + i); work->target[i] = le32dec(work->target + i);
if (opt_max_diff > 0. && !allow_mininginfo)
calc_network_diff(work);
json_t *jr = json_object_get(val, "noncerange"); json_t *jr = json_object_get(val, "noncerange");
if (jr) { if (jr) {
const char * hexstr = json_string_value(jr); const char * hexstr = json_string_value(jr);
@ -586,9 +632,9 @@ static bool work_decode(const json_t *val, struct work *work)
/** /**
* Calculate the work difficulty as double * Calculate the work difficulty as double
* Not sure it works with pools * TODO: merge common code with calc_network_diff
*/ */
static void calc_diff(struct work *work, int known) static void calc_target_diff(struct work *work)
{ {
// sample for diff 32.53 : 00000007de5f0000 // sample for diff 32.53 : 00000007de5f0000
const uint64_t diffone = 0xFFFF000000000000ull; const uint64_t diffone = 0xFFFF000000000000ull;
@ -596,10 +642,19 @@ static void calc_diff(struct work *work, int known)
char rtarget[32]; char rtarget[32];
swab256(rtarget, work->target); swab256(rtarget, work->target);
data64 = (uint64_t *)(rtarget + 3); /* todo: index (3) can be tuned here */
if (opt_algo == ALGO_HEAVY) { switch (opt_algo) {
data64 = (uint64_t *)(rtarget + 2); case ALGO_NEOSCRYPT:
//case ALGO_SCRYPT:
//case ALGO_SCRYPT_JANE:
// todo/check...
work->difficulty = 0.;
return;
case ALGO_HEAVY:
data64 = (uint64_t*)(rtarget + 2);
break;
default:
data64 = (uint64_t*)(rtarget + 3); /* index (3) can be tuned here */
} }
d64 = swab64(*data64); d64 = swab64(*data64);
@ -687,7 +742,7 @@ static bool submit_upstream_work(CURL *curl, struct work *work)
applog(LOG_WARNING, "stale work detected, discarding"); applog(LOG_WARNING, "stale work detected, discarding");
return true; return true;
} }
calc_diff(work, 0); calc_target_diff(work);
if (have_stratum) { if (have_stratum) {
uint32_t sent = 0; uint32_t sent = 0;
@ -815,10 +870,10 @@ static bool gbt_work_decode(const json_t *val, struct work *work)
if (key && json_is_integer(key)) { if (key && json_is_integer(key)) {
work->height = (uint32_t) json_integer_value(key); work->height = (uint32_t) json_integer_value(key);
if (!opt_quiet && work->height > g_work.height) { if (!opt_quiet && work->height > g_work.height) {
if (!have_stratum && allow_mininginfo && global_diff > 0) { if (net_diff > 0.) {
char netinfo[64] = { 0 }; char netinfo[64] = { 0 };
char srate[32] = { 0 }; char srate[32] = { 0 };
sprintf(netinfo, "diff %.2f", global_diff); sprintf(netinfo, "diff %.2f", net_diff);
if (net_hashrate) { if (net_hashrate) {
format_hashrate((double) net_hashrate, srate); format_hashrate((double) net_hashrate, srate);
strcat(netinfo, ", net "); strcat(netinfo, ", net ");
@ -896,7 +951,7 @@ static bool get_mininginfo(CURL *curl, struct work *work)
if (res) { if (res) {
json_t *key = json_object_get(res, "difficulty"); json_t *key = json_object_get(res, "difficulty");
if (key && json_is_real(key)) { if (key && json_is_real(key)) {
global_diff = json_real_value(key); net_diff = json_real_value(key);
} }
key = json_object_get(res, "networkhashps"); key = json_object_get(res, "networkhashps");
if (key && json_is_integer(key)) { if (key && json_is_integer(key)) {
@ -1209,6 +1264,9 @@ static void stratum_gen_work(struct stratum_ctx *sctx, struct work *work)
work->data[17] = le32dec(sctx->job.ntime); work->data[17] = le32dec(sctx->job.ntime);
work->data[18] = le32dec(sctx->job.nbits); work->data[18] = le32dec(sctx->job.nbits);
if (opt_max_diff > 0.)
calc_network_diff(work);
switch (opt_algo) { switch (opt_algo) {
case ALGO_MJOLLNIR: case ALGO_MJOLLNIR:
case ALGO_HEAVY: case ALGO_HEAVY:
@ -1237,7 +1295,9 @@ static void stratum_gen_work(struct stratum_ctx *sctx, struct work *work)
pthread_mutex_unlock(&sctx->work_lock); pthread_mutex_unlock(&sctx->work_lock);
if (opt_debug) { if (opt_debug) {
char *tm = atime2str(swab32(work->data[17]) - sctx->srvtime_diff); uint32_t utm = work->data[17];
if (opt_algo != ALGO_ZR5) utm = swab32(utm);
char *tm = atime2str(utm - sctx->srvtime_diff);
char *xnonce2str = bin2hex(work->xnonce2, sctx->xnonce2_size); char *xnonce2str = bin2hex(work->xnonce2, sctx->xnonce2_size);
applog(LOG_DEBUG, "DEBUG: job_id=%s xnonce2=%s time=%s", applog(LOG_DEBUG, "DEBUG: job_id=%s xnonce2=%s time=%s",
work->job_id, xnonce2str, tm); work->job_id, xnonce2str, tm);
@ -1293,7 +1353,7 @@ static bool wanna_mine(int thr_id)
} }
#endif #endif
} }
if (opt_max_diff > 0.0 && global_diff > opt_max_diff) { if (opt_max_diff > 0.0 && net_diff > opt_max_diff) {
if (conditional_state && !opt_quiet) if (conditional_state && !opt_quiet)
applog(LOG_INFO, "network diff too high, waiting..."); applog(LOG_INFO, "network diff too high, waiting...");
state = false; state = false;
@ -1422,9 +1482,7 @@ static void *miner_thread(void *userdata)
if (!opt_benchmark && (g_work.height != work.height || memcmp(work.target, g_work.target, sizeof(work.target)))) if (!opt_benchmark && (g_work.height != work.height || memcmp(work.target, g_work.target, sizeof(work.target))))
{ {
calc_diff(&g_work, 0); calc_target_diff(&g_work);
if (!have_stratum && !allow_mininginfo)
global_diff = g_work.difficulty;
if (opt_debug) { if (opt_debug) {
uint64_t target64 = g_work.target[7] * 0x100000000ULL + g_work.target[6]; uint64_t target64 = g_work.target[7] * 0x100000000ULL + g_work.target[6];
applog(LOG_DEBUG, "job %s target change: %llx (%.1f)", g_work.job_id, target64, g_work.difficulty); applog(LOG_DEBUG, "job %s target change: %llx (%.1f)", g_work.job_id, target64, g_work.difficulty);
@ -1917,8 +1975,13 @@ start:
pthread_mutex_lock(&g_work_lock); pthread_mutex_lock(&g_work_lock);
if (work_decode(json_object_get(val, "result"), &g_work)) { if (work_decode(json_object_get(val, "result"), &g_work)) {
restart_threads(); restart_threads();
if (!opt_quiet) if (!opt_quiet) {
applog(LOG_BLUE, "%s detected new block", short_url); char netinfo[64] = { 0 };
if (net_diff > 0.) {
sprintf(netinfo, ", diff %.2f", net_diff);
}
applog(LOG_BLUE, "%s detected new block%s", short_url, netinfo);
}
g_work_time = time(NULL); g_work_time = time(NULL);
} }
pthread_mutex_unlock(&g_work_lock); pthread_mutex_unlock(&g_work_lock);
@ -2041,9 +2104,14 @@ static void *stratum_thread(void *userdata)
stratum_gen_work(&stratum, &g_work); stratum_gen_work(&stratum, &g_work);
g_work_time = time(NULL); g_work_time = time(NULL);
if (stratum.job.clean) { if (stratum.job.clean) {
if (!opt_quiet) if (!opt_quiet) {
applog(LOG_BLUE, "%s %s block %d", short_url, algo_names[opt_algo], if (net_diff > 0.)
stratum.job.height); applog(LOG_BLUE, "%s block %d, diff %.2f", algo_names[opt_algo],
stratum.job.height, net_diff);
else
applog(LOG_BLUE, "%s %s block %d", short_url, algo_names[opt_algo],
stratum.job.height);
}
restart_threads(); restart_threads();
if (check_dups) if (check_dups)
hashlog_purge_old(); hashlog_purge_old();

View File

@ -461,8 +461,9 @@ struct thr_info {
}; };
struct work_restart { struct work_restart {
volatile unsigned long restart; /* volatile to modify accross threads (vstudio thing) */
char padding[128 - sizeof(unsigned long)]; volatile uint32_t restart;
char padding[128 - sizeof(uint32_t)];
}; };
extern bool opt_benchmark; extern bool opt_benchmark;
@ -492,7 +493,9 @@ extern bool opt_trust_pool;
extern uint16_t opt_vote; extern uint16_t opt_vote;
extern uint64_t global_hashrate; extern uint64_t global_hashrate;
extern double global_diff; extern uint64_t net_hashrate;
extern double net_diff;
extern double stratum_diff;
#define MAX_GPUS 16 #define MAX_GPUS 16
extern char* device_name[MAX_GPUS]; extern char* device_name[MAX_GPUS];

View File

@ -44,7 +44,7 @@ void stats_remember_speed(int thr_id, uint32_t hashcount, double hashrate, uint8
data.hashcount = hashcount; data.hashcount = hashcount;
data.hashfound = found; data.hashfound = found;
data.hashrate = hashrate; data.hashrate = hashrate;
data.difficulty = global_diff; data.difficulty = net_diff ? net_diff : stratum_diff;
if (opt_n_threads == 1 && global_hashrate && uid > 10) { if (opt_n_threads == 1 && global_hashrate && uid > 10) {
// prevent stats on too high vardiff (erroneous rates) // prevent stats on too high vardiff (erroneous rates)
double ratio = (hashrate / (1.0 * global_hashrate)); double ratio = (hashrate / (1.0 * global_hashrate));

View File

@ -1363,8 +1363,8 @@ static bool stratum_set_difficulty(struct stratum_ctx *sctx, json_t *params)
pthread_mutex_unlock(&sctx->work_lock); pthread_mutex_unlock(&sctx->work_lock);
/* store for api stats */ /* store for api stats */
if (diff != global_diff) { if (diff != stratum_diff) {
global_diff = diff; stratum_diff = diff;
applog(LOG_WARNING, "Stratum difficulty set to %g", diff); applog(LOG_WARNING, "Stratum difficulty set to %g", diff);
g_work_time = 0; g_work_time = 0;
} }