diff --git a/libztex.c b/libztex.c index 57200c87..89f5f0e7 100644 --- a/libztex.c +++ b/libztex.c @@ -156,7 +156,7 @@ static int libztex_configureFpgaLS (struct libztex_device *ztex, const char* fir for (tries=10; tries>0; tries--) { //* Reset fpga - cnt = libusb_control_transfer(ztex->hndl, 0x40, 0x31, 0, 0, NULL, 0, 1000); + cnt = libztex_resetFpga(ztex); if (unlikely(cnt < 0)) { applog(LOG_ERR, "%s: Failed reset fpga with err %d", ztex->repr, cnt); continue; @@ -185,7 +185,7 @@ static int libztex_configureFpgaLS (struct libztex_device *ztex, const char* fir return 3; } } - sleep(0.2); + usleep(200000); applog(LOG_ERR, "%s: FPGA configuration done", ztex->repr); return 0; } @@ -193,9 +193,9 @@ static int libztex_configureFpgaLS (struct libztex_device *ztex, const char* fir int libztex_configureFpga (struct libztex_device *ztex) { int rv; rv = libztex_configureFpgaLS(ztex, "bitstreams/ztex_ufm1_15d3.bit", true, 2); - if (rv == 0) { - libztex_setFreq(ztex, ztex->freqMDefault); - } + //if (rv == 0) { + // libztex_setFreq(ztex, ztex->freqMDefault); + //} return rv; } @@ -216,6 +216,10 @@ int libztex_setFreq (struct libztex_device *ztex, uint16_t freq) { return 0; } +int libztex_resetFpga (struct libztex_device *ztex) { + return libusb_control_transfer(ztex->hndl, 0x40, 0x31, 0, 0, NULL, 0, 1000); +} + int libztex_prepare_device (struct libusb_device *dev, struct libztex_device** ztex) { struct libztex_device *newdev; int cnt, err; @@ -306,6 +310,13 @@ int libztex_prepare_device (struct libusb_device *dev, struct libztex_device** z newdev->freqM = (buf[6] & 255); newdev->freqMDefault = newdev->freqM; + for (cnt=0; cnt<255; cnt++) { + newdev->errorCount[cnt] = 0; + newdev->errorWeight[cnt] = 0; + newdev->errorRate[cnt] = 0; + newdev->maxErrorRate[cnt] = 0; + } + newdev->usbbus = libusb_get_bus_number(dev); newdev->usbaddress = libusb_get_device_address(dev); sprintf(newdev->repr, "ZTEX %.3d:%.3d-%s", newdev->usbbus, newdev->usbaddress, newdev->snString); diff --git a/libztex.h b/libztex.h index bce2498c..4b9c5deb 100644 --- a/libztex.h +++ b/libztex.h @@ -30,6 +30,10 @@ #define LIBZTEX_IDVENDOR 0x221A #define LIBZTEX_IDPRODUCT 0x0100 +#define LIBZTEX_MAXMAXERRORRATE 0.05 +#define LIBZTEX_ERRORHYSTERESIS 0.1 +#define LIBZTEX_OVERHEATTHRESHOLD 0.5 + struct libztex_fpgastate { bool fpgaConfigured; unsigned char fpgaChecksum; @@ -58,6 +62,11 @@ struct libztex_device { uint8_t freqMaxM; uint8_t freqMDefault; + double errorCount[256]; + double errorWeight[256]; + double errorRate[256]; + double maxErrorRate[256]; + char repr[64]; }; @@ -79,5 +88,6 @@ extern int libztex_configureFpga (struct libztex_device *dev); extern int libztex_setFreq (struct libztex_device *ztex, uint16_t freq); extern int libztex_sendHashData (struct libztex_device *ztex, unsigned char *sendbuf); extern int libztex_readHashData (struct libztex_device *ztex, struct libztex_hash_data nonces[]); +extern int libztex_resetFpga (struct libztex_device *ztex); #endif /* __LIBZTEX_H__ */ diff --git a/todo_ztex.txt b/todo_ztex.txt index 411744c1..79fc208e 100644 --- a/todo_ztex.txt +++ b/todo_ztex.txt @@ -1,5 +1,5 @@ +- verify setting cgpu.status=DEAD does in fact stop the thread - make it compile on osx / windows - read actual bitstream -- nonce errors from fpga -- clock scaling - HS fpga config +- allow configuring LIBZTEX_OVERHEATTHRESHOLD diff --git a/ztex.c b/ztex.c index e478ab04..452d3c26 100644 --- a/ztex.c +++ b/ztex.c @@ -24,7 +24,7 @@ * along with this program; if not, see http://www.gnu.org/licenses/. **/ #include - +#include #include "miner.h" #include "libztex.h" @@ -61,15 +61,78 @@ static void ztex_detect() } -static bool ztex_prepare(struct thr_info *thr) +static bool ztex_updateFreq (struct libztex_device* ztex) { - struct timeval now; - struct cgpu_info *ztex = thr->cgpu; + int i, maxM, bestM; + double bestR, r; - gettimeofday(&now, NULL); - get_datestamp(ztex->init, &now); + for (i=0; ifreqMaxM; i++) { + if (ztex->maxErrorRate[i+1]*i < ztex->maxErrorRate[i]*(i+20)) + ztex->maxErrorRate[i+1] = ztex->maxErrorRate[i]*(1.0+20.0/i); + } - if (libztex_configureFpga(ztex->device) != 0) { + maxM = 0; + while (maxMfreqMDefault && ztex->maxErrorRate[maxM+1]freqMaxM, ztex->errorWeight[maxM], ztex->maxErrorRate[maxM+1], LIBZTEX_MAXMAXERRORRATE); + while (maxMfreqMaxM && ztex->errorWeight[maxM]>150 && ztex->maxErrorRate[maxM+1]freqM ? LIBZTEX_ERRORHYSTERESIS : 0))*(1-ztex->maxErrorRate[i]); + if (r > bestR) { + bestM = i; + bestR = r; + } + } + + //applog(LOG_WARNING, "maxM:%d bestM:%d bestR:%f freqM:%d", maxM, bestM, bestR, ztex->freqM); + + + if (bestM != ztex->freqM) { + libztex_setFreq(ztex, bestM); + } + + maxM = ztex->freqMDefault; + while (maxMfreqMaxM && ztex->errorWeight[maxM+1] > 100) + maxM++; + if ((bestM < (1.0-LIBZTEX_OVERHEATTHRESHOLD) * maxM) && bestM < maxM - 1) { + libztex_resetFpga(ztex); + applog(LOG_ERR, "%s: frequency drop of %.1f%% detect. This may be caused by overheating. FPGA is shut down to prevent damage.", ztex->repr, (1.0-1.0*bestM/maxM)*100); + return false; + } + return true; +} + + +static bool ztex_checkNonce (struct libztex_device *ztex, + struct work *work, + struct libztex_hash_data *hdata) +{ + uint32_t *data32 = (uint32_t *)(work->data); + unsigned char swap[128]; + uint32_t *swap32 = (uint32_t *)swap; + unsigned char hash1[32]; + unsigned char hash2[32]; + uint32_t *hash2_32 = (uint32_t *)hash2; + int i; + + work->data[64 + 12 + 0] = (hdata->nonce >> 0) & 0xff; + work->data[64 + 12 + 1] = (hdata->nonce >> 8) & 0xff; + work->data[64 + 12 + 2] = (hdata->nonce >> 16) & 0xff; + work->data[64 + 12 + 3] = (hdata->nonce >> 24) & 0xff; + + for (i = 0; i < 80 / 4; i++) + swap32[i] = swab32(data32[i]); + + sha2(swap, 80, hash1, false); + sha2(hash1, 32, hash2, false); + + if (swab32(hash2_32[7]) != ((hdata->hash7 + 0x5be0cd19) & 0xFFFFFFFF)) { + ztex->errorCount[ztex->freqM] += 1.0/ztex->numNonces; + applog(LOG_ERR, "%s: checkNonce failed for %0.8X", ztex->repr, hdata->nonce); return false; } return true; @@ -100,16 +163,24 @@ static uint64_t ztex_scanhash(struct thr_info *thr, struct work *work, } overflow = false; while (!(overflow || work_restart[thr->id].restart)) { + usleep(250000); libztex_readHashData(ztex, &hdata[0]); + ztex->errorCount[ztex->freqM] *= 0.995; + ztex->errorWeight[ztex->freqM] = ztex->errorWeight[ztex->freqM] * 0.995 + 1.0; + for (i=0; inumNonces; i++) { nonce = hdata[i].nonce; if (nonce > noncecnt) noncecnt = nonce; - if ((nonce >> 4) < (lastnonce[i] >> 4)) + if ((nonce >> 4) < (lastnonce[i] >> 4)) { overflow = true; - else + } else { lastnonce[i] = nonce; + } + if (!ztex_checkNonce(ztex, work, &hdata[i])) { + continue; + } nonce = hdata[i].goldenNonce; if (nonce > 0) { found = false; @@ -133,14 +204,40 @@ static uint64_t ztex_scanhash(struct thr_info *thr, struct work *work, applog(LOG_DEBUG, "submitted %0.8X %d", nonce, rv); } } + } } + ztex->errorRate[ztex->freqM] = ztex->errorCount[ztex->freqM] / ztex->errorWeight[ztex->freqM] * (ztex->errorWeight[ztex->freqM]<100 ? ztex->errorWeight[ztex->freqM]*0.01 : 1.0); + if (ztex->errorRate[ztex->freqM] > ztex->maxErrorRate[ztex->freqM]) { + ztex->maxErrorRate[ztex->freqM] = ztex->errorRate[ztex->freqM]; + } + if (!ztex_updateFreq(ztex)) { + // Something really serious happened, so mark this thread as dead! + thr->cgpu->status = LIFE_DEAD; + } + applog(LOG_WARNING, "freqM:%d errorRate:%0.3f errorCount:%d", ztex->freqM, ztex->errorRate[ztex->freqM], ztex->errorCount[ztex->freqM]); applog(LOG_DEBUG, "exit %0.8X", noncecnt); return noncecnt; } +static bool ztex_prepare(struct thr_info *thr) +{ + struct timeval now; + struct cgpu_info *ztex = thr->cgpu; + + gettimeofday(&now, NULL); + get_datestamp(ztex->init, &now); + + if (libztex_configureFpga(ztex->device) != 0) { + return false; + } + ztex->device->freqM = -1; + ztex_updateFreq(ztex->device); + return true; +} + static void ztex_shutdown(struct thr_info *thr) { if (thr->cgpu) {