1
0
mirror of https://github.com/GOSTSec/sgminer synced 2025-01-25 14:04:25 +00:00

MMQ add api pgaset for clock

This commit is contained in:
Kano 2012-12-19 10:56:00 +11:00
parent e931b72753
commit 4ab19c7de6
2 changed files with 64 additions and 16 deletions

View File

@ -344,6 +344,9 @@ The list of requests - a (*) means it requires privileged access - and replies a
If opt=help it will return an INFO status with a If opt=help it will return an INFO status with a
help message about the options available help message about the options available
The current options are:
MMQ opt=clock val=160 to 230 (and a multiple of 2)
When you enable, disable or restart a GPU or PGA, you will also get Thread messages When you enable, disable or restart a GPU or PGA, you will also get Thread messages
in the cgminer status window in the cgminer status window
@ -400,7 +403,7 @@ Feature Changelog for external applications using the API:
API V1.23 API V1.23
Added API commands: Added API commands:
'pgaset' 'pgaset' - with: MMQ opt=clock val=160 to 230 (and a multiple of 2)
---------- ----------

View File

@ -599,7 +599,13 @@ static bool modminer_fpga_prepare(struct thr_info *thr)
* *
* N.B. clock must always be a multiple of 2 * N.B. clock must always be a multiple of 2
*/ */
static bool modminer_delta_clock(struct thr_info *thr, int delta, bool temp) static const char *CLOCKOLDWORK = "clock already changed for this work";
static const char *CLOCKTOOLOW = "clock too low";
static const char *CLOCKTOOHI = "clock too high";
static const char *CLOCKSETFAIL = "clock set command failed";
static const char *CLOCKREPLYFAIL = "clock reply failed";
static const char *modminer_delta_clock(struct thr_info *thr, int delta, bool temp, bool force)
{ {
struct cgpu_info *modminer = thr->cgpu; struct cgpu_info *modminer = thr->cgpu;
struct modminer_fpga_state *state = thr->cgpu_data; struct modminer_fpga_state *state = thr->cgpu_data;
@ -607,8 +613,8 @@ static bool modminer_delta_clock(struct thr_info *thr, int delta, bool temp)
int err, amount; int err, amount;
// Only do once if multiple shares per work or multiple reasons // Only do once if multiple shares per work or multiple reasons
if (!state->new_work) if (!state->new_work && !force)
return false; return CLOCKOLDWORK;
state->new_work = false; state->new_work = false;
@ -618,10 +624,10 @@ static bool modminer_delta_clock(struct thr_info *thr, int delta, bool temp)
// FYI clock drop has little effect on temp // FYI clock drop has little effect on temp
if (delta < 0 && modminer->clock <= MODMINER_MIN_CLOCK) if (delta < 0 && modminer->clock <= MODMINER_MIN_CLOCK)
return false; return CLOCKTOOLOW;
if (delta > 0 && modminer->clock >= MODMINER_MAX_CLOCK) if (delta > 0 && modminer->clock >= MODMINER_MAX_CLOCK)
return false; return CLOCKTOOHI;
if (delta < 0) { if (delta < 0) {
if (temp) if (temp)
@ -649,7 +655,7 @@ static bool modminer_delta_clock(struct thr_info *thr, int delta, bool temp)
applog(LOG_ERR, "%s%u: Error writing set clock speed (%d:%d)", applog(LOG_ERR, "%s%u: Error writing set clock speed (%d:%d)",
modminer->api->name, modminer->device_id, amount, err); modminer->api->name, modminer->device_id, amount, err);
return false; return CLOCKSETFAIL;
} }
if ((err = usb_read(modminer, (char *)(&buf), 1, &amount, C_REPLYSETCLOCK)) < 0 || amount != 1) { if ((err = usb_read(modminer, (char *)(&buf), 1, &amount, C_REPLYSETCLOCK)) < 0 || amount != 1) {
@ -658,7 +664,7 @@ static bool modminer_delta_clock(struct thr_info *thr, int delta, bool temp)
applog(LOG_ERR, "%s%u: Error reading set clock speed (%d:%d)", applog(LOG_ERR, "%s%u: Error reading set clock speed (%d:%d)",
modminer->api->name, modminer->device_id, amount, err); modminer->api->name, modminer->device_id, amount, err);
return false; return CLOCKREPLYFAIL;
} }
mutex_unlock(modminer->modminer_mutex); mutex_unlock(modminer->modminer_mutex);
@ -668,7 +674,7 @@ static bool modminer_delta_clock(struct thr_info *thr, int delta, bool temp)
(delta < 0) ? "down " : (delta > 0 ? "up " : ""), (delta < 0) ? "down " : (delta > 0 ? "up " : ""),
modminer->clock); modminer->clock);
return true; return NULL;
} }
static bool modminer_fpga_init(struct thr_info *thr) static bool modminer_fpga_init(struct thr_info *thr)
@ -715,7 +721,7 @@ static bool modminer_fpga_init(struct thr_info *thr)
} }
modminer->clock = MODMINER_DEF_CLOCK; modminer->clock = MODMINER_DEF_CLOCK;
modminer_delta_clock(thr, MODMINER_CLOCK_SET, false); modminer_delta_clock(thr, MODMINER_CLOCK_SET, false, false);
thr->primary_thread = true; thr->primary_thread = true;
@ -831,7 +837,7 @@ static void check_temperature(struct thr_info *thr)
modminer->api->name, modminer->device_id, modminer->api->name, modminer->device_id,
MODMINER_CUTOFF_TEMP, modminer->temp); MODMINER_CUTOFF_TEMP, modminer->temp);
modminer_delta_clock(thr, MODMINER_CLOCK_CUTOFF, true); modminer_delta_clock(thr, MODMINER_CLOCK_CUTOFF, true, false);
state->overheated = true; state->overheated = true;
dev_error(modminer, REASON_DEV_THERMAL_CUTOFF); dev_error(modminer, REASON_DEV_THERMAL_CUTOFF);
} else { } else {
@ -841,7 +847,7 @@ static void check_temperature(struct thr_info *thr)
// If it's defined to be 0 then don't call modminer_delta_clock() // If it's defined to be 0 then don't call modminer_delta_clock()
if (MODMINER_CLOCK_OVERHEAT != 0) if (MODMINER_CLOCK_OVERHEAT != 0)
modminer_delta_clock(thr, MODMINER_CLOCK_OVERHEAT, true); modminer_delta_clock(thr, MODMINER_CLOCK_OVERHEAT, true, false);
state->overheated = true; state->overheated = true;
dev_error(modminer, REASON_DEV_OVER_HEAT); dev_error(modminer, REASON_DEV_OVER_HEAT);
} }
@ -950,7 +956,7 @@ static uint64_t modminer_process_results(struct thr_info *thr)
if (modminer->clock > MODMINER_DEF_CLOCK || state->hw_errors > 1) { if (modminer->clock > MODMINER_DEF_CLOCK || state->hw_errors > 1) {
float pct = (state->hw_errors * 100.0 / (state->shares ? : 1.0)); float pct = (state->hw_errors * 100.0 / (state->shares ? : 1.0));
if (pct >= MODMINER_HW_ERROR_PERCENT) if (pct >= MODMINER_HW_ERROR_PERCENT)
modminer_delta_clock(thr, MODMINER_CLOCK_DOWN, false); modminer_delta_clock(thr, MODMINER_CLOCK_DOWN, false, false);
} }
} }
} else { } else {
@ -959,7 +965,7 @@ static uint64_t modminer_process_results(struct thr_info *thr)
// If we've reached the required good shares in a row then clock up // If we've reached the required good shares in a row then clock up
if (((state->shares - state->shares_last_hw) >= state->shares_to_good) && if (((state->shares - state->shares_last_hw) >= state->shares_to_good) &&
modminer->temp < MODMINER_TEMP_UP_LIMIT) modminer->temp < MODMINER_TEMP_UP_LIMIT)
modminer_delta_clock(thr, MODMINER_CLOCK_UP, false); modminer_delta_clock(thr, MODMINER_CLOCK_UP, false, false);
} }
} else { } else {
// on rare occasions - the MMQ can just stop returning valid nonces // on rare occasions - the MMQ can just stop returning valid nonces
@ -967,7 +973,7 @@ static uint64_t modminer_process_results(struct thr_info *thr)
gettimeofday(&now, NULL); gettimeofday(&now, NULL);
if (tdiff(&now, &state->last_nonce) >= death) { if (tdiff(&now, &state->last_nonce) >= death) {
if (state->death_stage_one) { if (state->death_stage_one) {
modminer_delta_clock(thr, MODMINER_CLOCK_DEAD, false); modminer_delta_clock(thr, MODMINER_CLOCK_DEAD, false, true);
applog(LOG_ERR, "%s%u: DEATH clock down", applog(LOG_ERR, "%s%u: DEATH clock down",
modminer->api->name, modminer->device_id); modminer->api->name, modminer->device_id);
@ -977,7 +983,7 @@ static uint64_t modminer_process_results(struct thr_info *thr)
state->death_stage_one = false; state->death_stage_one = false;
return -1; return -1;
} else { } else {
modminer_delta_clock(thr, MODMINER_CLOCK_DEAD, false); modminer_delta_clock(thr, MODMINER_CLOCK_DEAD, false, true);
applog(LOG_ERR, "%s%u: death clock down", applog(LOG_ERR, "%s%u: death clock down",
modminer->api->name, modminer->device_id); modminer->api->name, modminer->device_id);
@ -1097,11 +1103,50 @@ static void modminer_fpga_shutdown(struct thr_info *thr)
free(thr->cgpu_data); free(thr->cgpu_data);
} }
static char *modminer_set_device(struct cgpu_info *modminer, char *option, char *setting, char *replybuf)
{
const char *ret;
int val;
if (strcasecmp(option, "help") == 0) {
sprintf(replybuf, "clock: range %d-%d and a multiple of 2",
MODMINER_MIN_CLOCK, MODMINER_MAX_CLOCK);
return replybuf;
}
if (strcasecmp(option, "clock") == 0) {
if (!setting || !*setting) {
sprintf(replybuf, "missing clock setting");
return replybuf;
}
val = atoi(setting);
if (val < MODMINER_MIN_CLOCK || val > MODMINER_MAX_CLOCK || (val & 1) != 0) {
sprintf(replybuf, "invalid clock: '%s' valid range %d-%d and a multiple of 2",
setting, MODMINER_MIN_CLOCK, MODMINER_MAX_CLOCK);
return replybuf;
}
val -= (int)(modminer->clock);
ret = modminer_delta_clock(modminer->thr[0], val, false, true);
if (ret) {
sprintf(replybuf, "Set clock failed: %s", ret);
return replybuf;
} else
return NULL;
}
sprintf(replybuf, "Unknown option: %s", option);
return replybuf;
}
struct device_api modminer_api = { struct device_api modminer_api = {
.dname = "modminer", .dname = "modminer",
.name = "MMQ", .name = "MMQ",
.api_detect = modminer_detect, .api_detect = modminer_detect,
.get_statline_before = get_modminer_statline_before, .get_statline_before = get_modminer_statline_before,
.set_device = modminer_set_device,
.thread_prepare = modminer_fpga_prepare, .thread_prepare = modminer_fpga_prepare,
.thread_init = modminer_fpga_init, .thread_init = modminer_fpga_init,
.scanhash = modminer_scanhash, .scanhash = modminer_scanhash,