diff --git a/api.c b/api.c
index ff26e54f..ac5b9a6b 100644
--- a/api.c
+++ b/api.c
@@ -1174,10 +1174,12 @@ static void pgaenable(__maybe_unused SOCKETTYPE c, char *param, bool isjson)
return;
}
+#if 0 /* A DISABLED device wont change status FIXME: should disabling make it WELL? */
if (cgpu->status != LIFE_WELL) {
strcpy(io_buffer, message(MSG_PGAUNW, id, NULL, isjson));
return;
}
+#endif
for (i = 0; i < mining_threads; i++) {
pga = thr_info[i].cgpu->device_id;
diff --git a/cgminer.c b/cgminer.c
index 9b77d693..8b7de73f 100644
--- a/cgminer.c
+++ b/cgminer.c
@@ -2165,7 +2165,10 @@ static bool stale_work(struct work *work, bool share)
return false;
if (share) {
- if (work->rolltime)
+ /* Technically the rolltime should be correct but some pools
+ * advertise a broken expire= that is lower than a meaningful
+ * scantime */
+ if (work->rolltime > opt_scantime)
work_expiry = work->rolltime;
else
work_expiry = opt_expiry;
@@ -3641,9 +3644,11 @@ static inline bool should_roll(struct work *work)
return false;
}
+/* Limit rolls to 7000 to not beyond 2 hours in the future where bitcoind will
+ * reject blocks as invalid. */
static inline bool can_roll(struct work *work)
{
- return (work->pool && !stale_work(work, false) && work->rolltime && !work->clone);
+ return (work->pool && work->rolltime && !work->clone && work->rolls < 7000);
}
static void roll_work(struct work *work)
diff --git a/miner.php b/miner.php
index 7c4d8891..1c64e5a8 100644
--- a/miner.php
+++ b/miner.php
@@ -4,6 +4,7 @@ session_start();
global $miner, $port, $readonly, $notify, $rigs, $socktimeoutsec;
global $checklastshare, $hidefields;
global $ignorerefresh, $changerefresh, $autorefresh;
+global $allowcustompages, $customsummarypages;
#
# Don't touch these 2 - see $rigs below
$miner = null;
@@ -59,6 +60,32 @@ $ignorerefresh = false;
$changerefresh = true;
$autorefresh = 0;
#
+# Should we allow custom pages?
+# (or just completely ignore then and don't display the buttons)
+$allowcustompages = true;
+#
+# OK this is a bit more complex item: Custom Summary Pages
+# A custom summary page in an array of 'section' => array('FieldA','FieldB'...)
+# This makes up what is displayed with each 'section' separately as a table
+# - empty tables are not shown
+# - empty columns (an unknown field) are not shown
+# - and missing field data shows as blank
+# There is a second array, listing fields to be totaled for each section
+# see the example below (if there is no matching data, no total will show)
+$mobilepage = array(
+ 'SUMMARY' => array('Elapsed', 'MHS av', 'Found Blocks', 'Accepted', 'Rejected', 'Utility'),
+ 'GPU' => array('GPU', 'Status', 'MHS av', 'Accepted', 'Rejected', 'Utility'),
+ 'PGA' => array('ID', 'Name', 'Status', 'MHS av', 'Accepted', 'Rejected', 'Utility'),
+ 'POOL' => array('POOL', 'Status', 'Accepted', 'Rejected', 'Last Share Time'));
+$mobilesum = array(
+ 'SUMMARY' => array('MHS av' => 1, 'Found Blocks' => 1, 'Accepted' => 1, 'Rejected' => 1, 'Utility' => 1),
+ 'GPU' => array('MHS av' => 1, 'Accepted' => 1, 'Rejected' => 1, 'Utility' => 1),
+ 'PGA' => array('MHS av' => 1, 'Accepted' => 1, 'Rejected' => 1, 'Utility' => 1),
+ 'POOL' => array('Accepted' => 1, 'Rejected' => 1));
+#
+# customsummarypages is an array of these Custom Summary Pages
+$customsummarypages = array('Mobile' => array($mobilepage, $mobilesum));
+#
$here = $_SERVER['PHP_SELF'];
#
global $tablebegin, $tableend, $warnfont, $warnoff, $dfmt;
@@ -90,26 +117,30 @@ $showndate = false;
global $rigerror;
$rigerror = array();
#
-function htmlhead($checkapi, $rig)
+function htmlhead($checkapi, $rig, $pg = null)
{
global $miner_font_family, $miner_font_size;
global $error, $readonly, $here;
global $ignorerefresh, $autorefresh;
- $paramrig = '';
+ $extraparams = '';
if ($rig != null && $rig != '')
- $paramrig = "&rig=$rig";
+ $extraparams = "&rig=$rig";
+ else
+ if ($pg != null && $pg != '')
+ $extraparams = "&pg=$pg";
if ($ignorerefresh == true || $autorefresh == 0)
$refreshmeta = '';
else
{
- $url = "$here?ref=$autorefresh$paramrig";
+ $url = "$here?ref=$autorefresh$extraparams";
$refreshmeta = "\n";
}
if ($readonly === false && $checkapi === true)
{
+ $error = null;
$access = api('privileged');
if ($error != null
|| !isset($access['STATUS']['STATUS'])
@@ -134,7 +165,7 @@ td.lst { color:blue; $miner_font background:#ffffdd }
function pr(a,m){if(m!=null){if(!confirm(m+'?'))return}window.location='$here?ref=$autorefresh'+a}\n";
if ($ignorerefresh == false)
- echo "function prr(a){if(a){v=document.getElementById('refval').value}else{v=0}window.location='$here?ref='+v+'$paramrig'}\n";
+ echo "function prr(a){if(a){v=document.getElementById('refval').value}else{v=0}window.location='$here?ref='+v+'$extraparams'}\n";
if ($readonly === false && $checkapi === true)
{
@@ -150,17 +181,20 @@ function prs2(a,n,r){var v=document.getElementById('gi'+n).value;var c=a.substr(
'switchpool',
'Enable' => 'enablepool',
'Disable' => 'disablepool' );
#
-function showhead($cmd, $item, $values)
+function showhead($cmd, $values, $justnames = false)
{
global $poolcmd, $readonly;
@@ -532,7 +569,7 @@ function showhead($cmd, $item, $values)
echo "
$name | ";
}
- if ($cmd == 'pools' && $readonly === false)
+ if ($justnames === false && $cmd == 'pools' && $readonly === false)
foreach ($poolcmd as $name => $pcmd)
echo "$name | ";
@@ -588,7 +625,7 @@ function details($cmd, $list, $rig)
if ($sectionname != $section)
{
echo $tableend.$tablebegin;
- showhead($cmd, $item, $values);
+ showhead($cmd, $values);
$section = $sectionname;
}
@@ -796,7 +833,7 @@ function doforeach($cmd, $des, $sum, $head, $datetime)
echo $tableend.$tablebegin;
$dthead = array('' => 1, 'STATUS' => 1, 'Description' => 1, 'When' => 1, 'API' => 1, 'CGMiner' => 1);
- showhead('', null, $dthead);
+ showhead('', $dthead);
foreach ($anss as $rig => $ans)
{
@@ -865,7 +902,7 @@ function doforeach($cmd, $des, $sum, $head, $datetime)
if ($sum != null)
$anss['total']['total'] = $total;
- showhead('', null, $header);
+ showhead('', $header);
$section = '';
@@ -924,26 +961,38 @@ function refreshbuttons()
if ($ignorerefresh == false && $changerefresh == true)
{
echo ' ';
- echo "";
+ echo "";
echo "";
echo "";
}
}
#
-function doOne($rig, $preprocess)
+function pagetop($rig, $pg)
{
- global $error, $readonly, $notify, $rigs;
+ global $readonly, $rigs;
+ global $allowcustompages, $customsummarypages;
- htmlhead(true, $rig);
+ if ($rig === null)
+ {
+ if ($pg === null)
+ $refresh = '';
+ else
+ $refresh = "&pg=$pg";
+ }
+ else
+ $refresh = "&rig=$rig";
- $error = null;
-
- echo "";
- echo " | ";
+ echo ' | ";
+}
+#
+function doOne($rig, $preprocess)
+{
+ global $haderror, $readonly, $notify, $rigs;
+
+ htmlhead(true, $rig);
+
+ pagetop($rig, null);
if ($preprocess != null)
process(array($preprocess => $preprocess), $rig);
@@ -968,16 +1026,226 @@ function doOne($rig, $preprocess)
process($cmds, $rig);
- if ($error == null && $readonly === false)
+ if ($haderror == false && $readonly === false)
processgpus($rig);
}
#
+global $sectionmap;
+# map sections to their api command
+$sectionmap = array(
+ 'SUMMARY' => 'summary',
+ 'POOL' => 'pools',
+ 'GPU' => 'devs',
+ 'PGA' => 'devs',
+ 'NOTIFY' => 'notify',
+ 'CONFIG' => 'config');
+#
+function customset($showfields, $sum, $section, $num, $result, $total)
+{
+ foreach ($result as $sec => $row)
+ {
+ $secname = preg_replace('/\d/', '', $sec);
+
+ if ($sec != 'total')
+ {
+ if ($secname != $section)
+ continue;
+ }
+
+ echo '';
+
+ $when = 0;
+ if (isset($result['STATUS']['When']))
+ $when = $result['STATUS']['When'];
+
+ if ($sec === 'total')
+ $class = ' class=tot';
+ else
+ $class = '';
+
+ echo "$num | ";
+
+ foreach ($showfields as $name => $one)
+ {
+ if (isset($row[$name]))
+ {
+ $value = $row[$name];
+
+ if (isset($sum[$secname][$name]))
+ {
+ if (isset($total[$name]))
+ $total[$name] += $value;
+ else
+ $total[$name] = $value;
+ }
+ }
+ else
+ {
+ if ($sec == 'total' && isset($total[$name]))
+ $value = $total[$name];
+ else
+ $value = null;
+ }
+
+ list($showvalue, $class) = fmt($section, $name, $value, $when, $row);
+
+ if ($sec === 'total' and $class == '')
+ $class = ' class=tot';
+
+
+ echo "$showvalue | ";
+ }
+
+ echo ' ';
+ }
+ return $total;
+}
+#
+function processcustompage($pagename, $sections, $sum)
+{
+ global $sectionmap;
+ global $miner, $port;
+ global $rigs, $error;
+ global $warnfont, $warnoff;
+ global $tablebegin, $tableend, $dfmt;
+ global $readonly, $showndate;
+
+ $cmds = array();
+ $errors = array();
+ foreach ($sections as $section => $fields)
+ {
+ if (isset($sectionmap[$section]))
+ {
+ $cmd = $sectionmap[$section];
+ if (!isset($cmds[$cmd]))
+ $cmds[$cmd] = 1;
+ }
+ else
+ $errors[] = "Error: unknown section '$section' in custom summary page '$pagename'";
+ }
+
+ $results = array();
+ foreach ($rigs as $num => $rig)
+ {
+ $parts = explode(':', $rig, 2);
+ if (count($parts) == 2)
+ {
+ $miner = $parts[0];
+ $port = $parts[1];
+
+ foreach ($cmds as $cmd => $one)
+ {
+ $process = api($cmd);
+
+ if ($error != null)
+ {
+ $errors[] = "Error getting $cmd for $rig $warnfont$error$warnoff";
+ break;
+ }
+ else
+ $results[$cmd][$num] = $process;
+ }
+ }
+ }
+
+ if (count($results) > 0)
+ {
+ $first = true;
+ foreach ($sections as $section => $fields)
+ {
+ if (isset($results[$sectionmap[$section]]))
+ {
+ $rigresults = $results[$sectionmap[$section]];
+ $showfields = array();
+ foreach ($fields as $field)
+ foreach ($rigresults as $result)
+ foreach ($result as $sec => $row)
+ {
+ $secname = preg_replace('/\d/', '', $sec);
+ if ($secname == $section && isset($row[$field]))
+ $showfields[$field] = 1;
+ }
+
+ if (count($showfields) > 0)
+ {
+ if ($first === false)
+ echo ' | ';
+
+ echo $tablebegin;
+
+ showhead('', array('Rig'=>1)+$showfields, true);
+
+ $total = array();
+ $add = array('total' => array());
+
+ foreach ($rigresults as $num => $result)
+ {
+ $rg = "";
+ $total = customset($showfields, $sum, $section, $rg, $result, $total);
+ }
+
+ if (count($total) > 0)
+ customset($showfields, $sum, $section, 'Σ', $add, $total);
+
+ $first = false;
+
+ echo $tableend;
+ }
+ }
+ }
+ }
+
+ if (count($errors) > 0)
+ {
+ if (count($results) > 0)
+ echo ' | ';
+
+ foreach ($errors as $err)
+ echo "$err | ";
+ }
+}
+#
+function showcustompage($pagename)
+{
+ global $customsummarypages;
+
+ htmlhead(false, null, $pagename);
+
+ pagetop(null, $pagename);
+
+ if (!isset($customsummarypages[$pagename]))
+ {
+ echo "Unknown custom summary page '$pagename' | ";
+ return;
+ }
+
+ if (count($customsummarypages[$pagename]) != 2)
+ {
+ echo "Invalid custom summary page '$pagename' (".count($customsummarypages[$pagename]).") | ";
+ return;
+ }
+
+ $page = $customsummarypages[$pagename][0];
+ $sum = $customsummarypages[$pagename][1];
+ if ($sum === null)
+ $sum = array();
+
+ if (count($page) <= 1)
+ {
+ echo "Invalid custom summary page '$pagename' no content | ";
+ return;
+ }
+
+ processcustompage($pagename, $page, $sum);
+}
+#
function display()
{
global $tablebegin, $tableend;
global $miner, $port;
- global $error, $readonly, $notify, $rigs;
+ global $readonly, $notify, $rigs;
global $ignorerefresh, $autorefresh;
+ global $allowcustompages;
if ($ignorerefresh == false)
{
@@ -1021,6 +1289,16 @@ function display()
return;
}
+ if ($allowcustompages === true)
+ {
+ $pg = trim(getparam('pg', true));
+ if ($pg != null && $pg != '')
+ {
+ showcustompage($pg);
+ return;
+ }
+ }
+
if (count($rigs) == 1)
{
$parts = explode(':', $rigs[0], 2);
@@ -1055,11 +1333,7 @@ function display()
htmlhead(false, null);
- echo " | ";
+ pagetop(null, null);
if ($preprocess != null)
process(array($preprocess => $preprocess), $rig);
|