2012-03-11 16:58:50 +00:00
/**
* ztex . c - cgminer worker for Ztex 1.15 x fpga board
*
* Copyright ( c ) 2012 nelisky . btc @ gmail . com
*
* This work is based upon the Java SDK provided by ztex which is
* Copyright ( C ) 2009 - 2011 ZTEX GmbH .
* http : //www.ztex.de
*
* This work is based upon the icarus . c worker which is
* Copyright 2012 Luke Dashjr
* Copyright 2012 Xiangfu < xiangfu @ openmobilefree . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the GNU
* General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , see http : //www.gnu.org/licenses/.
* */
2012-03-11 00:49:25 +00:00
# include <unistd.h>
2012-03-12 00:55:38 +00:00
# include <sha2.h>
2012-03-11 00:49:25 +00:00
# include "miner.h"
# include "libztex.h"
# define GOLDEN_BACKLOG 5
2012-03-18 22:37:37 +00:00
struct device_api ztex_api , ztex_hotplug_api ;
2012-03-11 00:49:25 +00:00
2012-03-16 15:15:46 +00:00
// Forward declarations
static void ztex_disable ( struct thr_info * thr ) ;
2012-03-18 22:37:37 +00:00
static bool ztex_prepare ( struct thr_info * thr ) ;
2012-03-16 15:15:46 +00:00
2012-03-11 00:49:25 +00:00
static void ztex_detect ( )
{
2012-04-17 08:51:53 +00:00
int cnt ;
int i ;
struct libztex_dev_list * * ztex_devices ;
cnt = libztex_scanDevices ( & ztex_devices ) ;
applog ( LOG_WARNING , " Found %d ztex board(s) " , cnt ) ;
for ( i = 0 ; i < cnt ; i + + ) {
if ( total_devices = = MAX_DEVICES )
break ;
struct cgpu_info * ztex ;
ztex = calloc ( 1 , sizeof ( struct cgpu_info ) ) ;
ztex - > api = & ztex_api ;
ztex - > device_id = total_devices ;
ztex - > device = ztex_devices [ i ] - > dev ;
ztex - > threads = 1 ;
devices [ total_devices + + ] = ztex ;
applog ( LOG_WARNING , " %s: Found Ztex, mark as %d " , ztex - > device - > repr , ztex - > device_id ) ;
}
2012-04-18 08:54:51 +00:00
if ( cnt > 0 )
2012-04-17 08:51:53 +00:00
libztex_freeDevList ( ztex_devices ) ;
2012-03-11 00:49:25 +00:00
}
2012-03-12 00:55:38 +00:00
static bool ztex_updateFreq ( struct libztex_device * ztex )
2012-03-11 00:49:25 +00:00
{
2012-04-17 08:51:53 +00:00
int i , maxM , bestM ;
double bestR , r ;
2012-04-18 08:54:51 +00:00
for ( i = 0 ; i < ztex - > freqMaxM ; i + + )
2012-04-17 08:51:53 +00:00
if ( ztex - > maxErrorRate [ i + 1 ] * i < ztex - > maxErrorRate [ i ] * ( i + 20 ) )
ztex - > maxErrorRate [ i + 1 ] = ztex - > maxErrorRate [ i ] * ( 1.0 + 20.0 / i ) ;
maxM = 0 ;
while ( maxM < ztex - > freqMDefault & & ztex - > maxErrorRate [ maxM + 1 ] < LIBZTEX_MAXMAXERRORRATE )
maxM + + ;
while ( maxM < ztex - > freqMaxM & & ztex - > errorWeight [ maxM ] > 150 & & ztex - > maxErrorRate [ maxM + 1 ] < LIBZTEX_MAXMAXERRORRATE )
maxM + + ;
bestM = 0 ;
bestR = 0 ;
for ( i = 0 ; i < = maxM ; i + + ) {
r = ( i + 1 + ( i = = ztex - > freqM ? LIBZTEX_ERRORHYSTERESIS : 0 ) ) * ( 1 - ztex - > maxErrorRate [ i ] ) ;
if ( r > bestR ) {
bestM = i ;
bestR = r ;
}
}
2012-04-18 08:54:51 +00:00
if ( bestM ! = ztex - > freqM )
2012-04-17 08:51:53 +00:00
libztex_setFreq ( ztex , bestM ) ;
maxM = ztex - > freqMDefault ;
while ( maxM < ztex - > freqMaxM & & 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 ;
2012-03-12 00:55:38 +00:00
}
static bool ztex_checkNonce ( struct libztex_device * ztex ,
struct work * work ,
struct libztex_hash_data * hdata )
{
2012-04-17 08:51:53 +00:00
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 ;
2012-03-31 12:13:53 +00:00
# if defined(__BIGENDIAN__) || defined(MIPSEB)
2012-04-17 08:51:53 +00:00
hdata - > nonce = swab32 ( hdata - > nonce ) ;
hdata - > hash7 = swab32 ( hdata - > hash7 ) ;
2012-03-31 12:13:53 +00:00
# endif
2012-04-17 08:51:53 +00:00
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 ;
2012-03-12 00:55:38 +00:00
2012-04-17 08:51:53 +00:00
for ( i = 0 ; i < 80 / 4 ; i + + )
swap32 [ i ] = swab32 ( data32 [ i ] ) ;
sha2 ( swap , 80 , hash1 , false ) ;
sha2 ( hash1 , 32 , hash2 , false ) ;
2012-03-31 12:13:53 +00:00
# if defined(__BIGENDIAN__) || defined(MIPSEB)
2012-04-17 08:51:53 +00:00
if ( hash2_32 [ 7 ] ! = ( ( hdata - > hash7 + 0x5be0cd19 ) & 0xFFFFFFFF ) ) {
2012-03-31 12:13:53 +00:00
# else
2012-04-17 08:51:53 +00:00
if ( swab32 ( hash2_32 [ 7 ] ) ! = ( ( hdata - > hash7 + 0x5be0cd19 ) & 0xFFFFFFFF ) ) {
2012-03-31 12:13:53 +00:00
# endif
2012-04-17 08:51:53 +00:00
ztex - > errorCount [ ztex - > freqM ] + = 1.0 / ztex - > numNonces ;
applog ( LOG_DEBUG , " %s: checkNonce failed for %0.8X " , ztex - > repr , hdata - > nonce ) ;
return false ;
}
return true ;
2012-03-11 00:49:25 +00:00
}
static uint64_t ztex_scanhash ( struct thr_info * thr , struct work * work ,
2012-04-17 08:51:53 +00:00
__maybe_unused uint64_t max_nonce )
2012-03-11 00:49:25 +00:00
{
2012-04-17 08:51:53 +00:00
struct libztex_device * ztex ;
unsigned char sendbuf [ 44 ] ;
int i , j ;
uint32_t backlog [ GOLDEN_BACKLOG ] ;
int backlog_p = 0 ;
uint32_t lastnonce [ GOLDEN_BACKLOG ] , nonce , noncecnt = 0 ;
bool overflow , found , rv ;
struct libztex_hash_data hdata [ GOLDEN_BACKLOG ] ;
ztex = thr - > cgpu - > device ;
memcpy ( sendbuf , work - > data + 64 , 12 ) ;
memcpy ( sendbuf + 12 , work - > midstate , 32 ) ;
memset ( backlog , 0 , sizeof ( backlog ) ) ;
i = libztex_sendHashData ( ztex , sendbuf ) ;
if ( i < 0 ) {
// Something wrong happened in send
2012-04-18 08:54:51 +00:00
applog ( LOG_ERR , " %s: Failed to send hash data with err %d, retrying " , ztex - > repr , i ) ;
2012-04-17 08:51:53 +00:00
usleep ( 500000 ) ;
i = libztex_sendHashData ( ztex , sendbuf ) ;
if ( i < 0 ) {
// And there's nothing we can do about it
ztex_disable ( thr ) ;
2012-04-18 08:54:51 +00:00
applog ( LOG_ERR , " %s: Failed to send hash data with err %d, giving up " , ztex - > repr , i ) ;
2012-04-17 08:51:53 +00:00
return 0 ;
}
}
applog ( LOG_DEBUG , " sent hashdata " ) ;
2012-04-18 08:54:51 +00:00
for ( i = 0 ; i < ztex - > numNonces ; i + + )
2012-04-17 08:51:53 +00:00
lastnonce [ i ] = 0 ;
2012-04-18 08:54:51 +00:00
2012-04-17 08:51:53 +00:00
overflow = false ;
while ( ! ( overflow | | work_restart [ thr - > id ] . restart ) ) {
usleep ( 250000 ) ;
if ( work_restart [ thr - > id ] . restart ) {
2012-04-18 08:54:51 +00:00
applog ( LOG_DEBUG , " %s: New work detected " , ztex - > repr ) ;
2012-04-17 08:51:53 +00:00
break ;
}
i = libztex_readHashData ( ztex , & hdata [ 0 ] ) ;
if ( i < 0 ) {
// Something wrong happened in read
2012-04-18 08:54:51 +00:00
applog ( LOG_ERR , " %s: Failed to read hash data with err %d, retrying " , ztex - > repr , i ) ;
2012-04-17 08:51:53 +00:00
usleep ( 500000 ) ;
i = libztex_readHashData ( ztex , & hdata [ 0 ] ) ;
if ( i < 0 ) {
// And there's nothing we can do about it
ztex_disable ( thr ) ;
2012-04-18 08:54:51 +00:00
applog ( LOG_ERR , " %s: Failed to read hash data with err %d, giving up " , ztex - > repr , i ) ;
2012-04-17 08:51:53 +00:00
return 0 ;
}
}
if ( work_restart [ thr - > id ] . restart ) {
2012-04-18 08:54:51 +00:00
applog ( LOG_DEBUG , " %s: New work detected " , ztex - > repr ) ;
2012-04-17 08:51:53 +00:00
break ;
}
ztex - > errorCount [ ztex - > freqM ] * = 0.995 ;
ztex - > errorWeight [ ztex - > freqM ] = ztex - > errorWeight [ ztex - > freqM ] * 0.995 + 1.0 ;
2012-03-12 00:55:38 +00:00
2012-04-17 08:51:53 +00:00
for ( i = 0 ; i < ztex - > numNonces ; i + + ) {
2012-04-18 08:59:48 +00:00
nonce = hdata [ i ] . nonce ;
# if defined(__BIGENDIAN__) || defined(MIPSEB)
nonce = swab32 ( nonce ) ;
# endif
2012-04-17 08:51:53 +00:00
if ( nonce > noncecnt )
noncecnt = nonce ;
2012-04-18 08:59:48 +00:00
if ( ( ( nonce & 0x7fffffff ) > > 4 ) < ( ( lastnonce [ i ] & 0x7fffffff ) > > 4 ) ) {
applog ( LOG_DEBUG , " %s: overflow nonce=%0.8x lastnonce=%0.8x " , ztex - > repr , nonce , lastnonce [ i ] ) ;
2012-04-17 08:51:53 +00:00
overflow = true ;
} else {
lastnonce [ i ] = nonce ;
}
2012-04-18 08:59:48 +00:00
# if !(defined(__BIGENDIAN__) || defined(MIPSEB))
nonce = swab32 ( nonce ) ;
# endif
2012-04-17 08:51:53 +00:00
if ( ! ztex_checkNonce ( ztex , work , & hdata [ i ] ) ) {
thr - > cgpu - > hw_errors + + ;
continue ;
}
nonce = hdata [ i ] . goldenNonce ;
if ( nonce > 0 ) {
found = false ;
for ( j = 0 ; j < GOLDEN_BACKLOG ; j + + ) {
if ( backlog [ j ] = = nonce ) {
found = true ;
break ;
}
}
if ( ! found ) {
2012-04-18 08:54:51 +00:00
applog ( LOG_DEBUG , " %s: Share found " , ztex - > repr ) ;
2012-04-17 08:51:53 +00:00
backlog [ backlog_p + + ] = nonce ;
2012-04-18 08:54:51 +00:00
if ( backlog_p > = GOLDEN_BACKLOG )
2012-04-17 08:51:53 +00:00
backlog_p = 0 ;
2012-03-31 12:13:53 +00:00
# if defined(__BIGENDIAN__) || defined(MIPSEB)
2012-04-17 08:51:53 +00:00
nonce = swab32 ( nonce ) ;
2012-03-11 00:49:25 +00:00
# endif
2012-04-18 08:54:51 +00:00
work - > blk . nonce = 0xffffffff ;
rv = submit_nonce ( thr , work , nonce ) ;
applog ( LOG_DEBUG , " %s: submitted %0.8x %d " , ztex - > repr , nonce , rv ) ;
2012-04-17 08:51:53 +00:00
}
}
2012-03-12 00:55:38 +00:00
2012-04-17 08:51:53 +00:00
}
2012-03-11 00:49:25 +00:00
2012-04-17 08:51:53 +00:00
}
2012-03-11 00:49:25 +00:00
2012-04-17 08:51:53 +00:00
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 ) ;
2012-04-18 08:54:51 +00:00
if ( ztex - > errorRate [ ztex - > freqM ] > ztex - > maxErrorRate [ ztex - > freqM ] )
2012-04-17 08:51:53 +00:00
ztex - > maxErrorRate [ ztex - > freqM ] = ztex - > errorRate [ ztex - > freqM ] ;
2012-03-12 16:20:03 +00:00
2012-04-18 08:54:51 +00:00
if ( ! ztex_updateFreq ( ztex ) )
2012-04-17 08:51:53 +00:00
// Something really serious happened, so mark this thread as dead!
return 0 ;
2012-04-18 08:54:51 +00:00
applog ( LOG_DEBUG , " %s: exit %1.8X " , ztex - > repr , noncecnt ) ;
2012-03-12 16:20:03 +00:00
2012-04-17 08:51:53 +00:00
work - > blk . nonce = 0xffffffff ;
2012-03-12 16:20:03 +00:00
2012-04-17 08:51:53 +00:00
return noncecnt > 0 ? noncecnt : 1 ;
2012-03-11 00:49:25 +00:00
}
2012-03-19 11:19:16 +00:00
static void ztex_statline_before ( char * buf , struct cgpu_info * cgpu )
{
2012-04-17 08:51:53 +00:00
if ( cgpu - > deven = = DEV_ENABLED ) {
tailsprintf ( buf , " %s | " , cgpu - > device - > snString ) ;
tailsprintf ( buf , " %0.2fMhz | " , cgpu - > device - > freqM1 * ( cgpu - > device - > freqM + 1 ) ) ;
}
2012-03-19 11:19:16 +00:00
}
2012-03-12 00:55:38 +00:00
static bool ztex_prepare ( struct thr_info * thr )
{
2012-04-17 08:51:53 +00:00
struct timeval now ;
struct cgpu_info * ztex = thr - > cgpu ;
2012-03-12 00:55:38 +00:00
2012-04-17 08:51:53 +00:00
gettimeofday ( & now , NULL ) ;
get_datestamp ( ztex - > init , & now ) ;
2012-03-12 00:55:38 +00:00
2012-04-18 08:54:51 +00:00
if ( libztex_configureFpga ( ztex - > device ) ! = 0 )
2012-04-17 08:51:53 +00:00
return false ;
2012-04-18 08:54:51 +00:00
2012-04-17 08:51:53 +00:00
ztex - > device - > freqM = - 1 ;
ztex_updateFreq ( ztex - > device ) ;
2012-03-12 16:20:03 +00:00
2012-04-18 08:54:51 +00:00
applog ( LOG_DEBUG , " %s: prepare " , ztex - > device - > repr ) ;
2012-04-17 08:51:53 +00:00
return true ;
2012-03-12 00:55:38 +00:00
}
2012-03-11 00:49:25 +00:00
static void ztex_shutdown ( struct thr_info * thr )
{
2012-04-17 08:51:53 +00:00
if ( thr - > cgpu - > device ! = NULL ) {
2012-04-18 08:54:51 +00:00
applog ( LOG_DEBUG , " %s: shutdown " , thr - > cgpu - > device - > repr ) ;
2012-04-17 08:51:53 +00:00
libztex_destroy_device ( thr - > cgpu - > device ) ;
thr - > cgpu - > device = NULL ;
}
2012-03-16 15:15:46 +00:00
}
static void ztex_disable ( struct thr_info * thr )
{
2012-04-17 08:51:53 +00:00
applog ( LOG_ERR , " %s: Disabling! " , thr - > cgpu - > device - > repr ) ;
devices [ thr - > cgpu - > device_id ] - > deven = DEV_DISABLED ;
ztex_shutdown ( thr ) ;
2012-03-11 00:49:25 +00:00
}
struct device_api ztex_api = {
. name = " ZTX " ,
. api_detect = ztex_detect ,
2012-03-19 11:19:16 +00:00
. get_statline_before = ztex_statline_before ,
2012-03-11 00:49:25 +00:00
. thread_prepare = ztex_prepare ,
. scanhash = ztex_scanhash ,
. thread_shutdown = ztex_shutdown ,
} ;
2012-03-11 16:58:50 +00:00