Browse Source

Implement auto fanspeed adjustment to maintain a target temperature and fanspeed below 85%, with an overheat check that will speed the fan up to 100%.

nfactor-troky
Con Kolivas 13 years ago
parent
commit
495adcbf5f
  1. 100
      adl.c
  2. 2
      adl.h
  3. 2
      main.c
  4. 7
      miner.h

100
adl.c

@ -199,19 +199,36 @@ void init_adl(int nDevs)
/* We're only interested in the top performance level */ /* We're only interested in the top performance level */
lpOdPerformanceLevels = alloca(sizeof(ADLODPerformanceLevels) + (lev * sizeof(ADLODPerformanceLevel))); lpOdPerformanceLevels = alloca(sizeof(ADLODPerformanceLevels) + (lev * sizeof(ADLODPerformanceLevel)));
lpOdPerformanceLevels->iSize = sizeof(ADLODPerformanceLevels) + sizeof(ADLODPerformanceLevel) * lev; lpOdPerformanceLevels->iSize = sizeof(ADLODPerformanceLevels) + sizeof(ADLODPerformanceLevel) * lev;
if (ADL_Overdrive5_ODPerformanceLevels_Get(iAdapterIndex, 0, lpOdPerformanceLevels) != ADL_OK) {
/* Get default performance levels first */
if (ADL_Overdrive5_ODPerformanceLevels_Get(iAdapterIndex, 1, lpOdPerformanceLevels) != ADL_OK) {
applog(LOG_INFO, "Failed to ADL_Overdrive5_ODPerformanceLevels_Get"); applog(LOG_INFO, "Failed to ADL_Overdrive5_ODPerformanceLevels_Get");
continue; continue;
} }
/* Set the limits we'd use based on default gpu speeds */
ga->maxspeed = ga->minspeed = lpOdPerformanceLevels->aLevels[lev].iEngineClock;
/* Now get the current performance levels for any existing overclock */
ADL_Overdrive5_ODPerformanceLevels_Get(iAdapterIndex, 0, lpOdPerformanceLevels);
ga->iEngineClock = lpOdPerformanceLevels->aLevels[lev].iEngineClock; ga->iEngineClock = lpOdPerformanceLevels->aLevels[lev].iEngineClock;
ga->iMemoryClock = lpOdPerformanceLevels->aLevels[lev].iMemoryClock; ga->iMemoryClock = lpOdPerformanceLevels->aLevels[lev].iMemoryClock;
ga->iVddc = lpOdPerformanceLevels->aLevels[lev].iVddc; ga->iVddc = lpOdPerformanceLevels->aLevels[lev].iVddc;
/* Now use any parameters already set outside default as limits */
if (ga->iEngineClock < ga->minspeed)
ga->minspeed = ga->iEngineClock;
if (ga->iEngineClock > ga->maxspeed)
ga->maxspeed = ga->iEngineClock;
if (ADL_Overdrive5_FanSpeedInfo_Get(iAdapterIndex, 0, &ga->lpFanSpeedInfo) != ADL_OK) { if (ADL_Overdrive5_FanSpeedInfo_Get(iAdapterIndex, 0, &ga->lpFanSpeedInfo) != ADL_OK) {
applog(LOG_INFO, "Failed to ADL_Overdrive5_FanSpeedInfo_Get"); applog(LOG_INFO, "Failed to ADL_Overdrive5_FanSpeedInfo_Get");
continue; continue;
} }
/* Set some default temperatures for autotune when enabled */
ga->targettemp = 75;
ga->overtemp = 85;
gpus[devices - 1].has_adl = true; gpus[devices - 1].has_adl = true;
} }
@ -293,7 +310,7 @@ int gpu_fanspeed(int gpu)
return 0; return 0;
ga = &gpus[gpu].adl; ga = &gpus[gpu].adl;
if (!(ga->lpFanSpeedInfo.iFlags & (ADL_DL_FANCTRL_SUPPORTS_RPM_READ | ADL_DL_FANCTRL_SUPPORTS_PERCENT_READ ))) if (!(ga->lpFanSpeedInfo.iFlags & ADL_DL_FANCTRL_SUPPORTS_RPM_READ))
return 0; return 0;
ga->lpFanSpeedValue.iSpeedType = ADL_DL_FANCTRL_SPEED_TYPE_RPM; ga->lpFanSpeedValue.iSpeedType = ADL_DL_FANCTRL_SPEED_TYPE_RPM;
if (ADL_Overdrive5_FanSpeed_Get(ga->iAdapterIndex, 0, &ga->lpFanSpeedValue) != ADL_OK) if (ADL_Overdrive5_FanSpeed_Get(ga->iAdapterIndex, 0, &ga->lpFanSpeedValue) != ADL_OK)
@ -301,6 +318,22 @@ int gpu_fanspeed(int gpu)
return ga->lpFanSpeedValue.iFanSpeed; return ga->lpFanSpeedValue.iFanSpeed;
} }
static int gpu_fanpercent(int gpu)
{
struct gpu_adl *ga;
if (!gpus[gpu].has_adl || !adl_active)
return -1;
ga = &gpus[gpu].adl;
if (!(ga->lpFanSpeedInfo.iFlags & ADL_DL_FANCTRL_SUPPORTS_PERCENT_READ ))
return -1;
ga->lpFanSpeedValue.iSpeedType = ADL_DL_FANCTRL_SPEED_TYPE_PERCENT;
if (ADL_Overdrive5_FanSpeed_Get(ga->iAdapterIndex, 0, &ga->lpFanSpeedValue) != ADL_OK)
return -1;
return ga->lpFanSpeedValue.iFanSpeed;
}
static void get_enginerange(int gpu, int *imin, int *imax) static void get_enginerange(int gpu, int *imin, int *imax)
{ {
struct gpu_adl *ga; struct gpu_adl *ga;
@ -518,17 +551,74 @@ static int set_fanspeed(int gpu, int iFanSpeed)
return 0; return 0;
} }
void gpu_autotune(int gpu)
{
struct gpu_adl *ga;
int temp, fanpercent;
if (!gpus[gpu].has_adl || !adl_active)
return;
temp = gpu_temp(gpu);
if (!temp)
return;
fanpercent = gpu_fanpercent(gpu);
if (fanpercent < 0)
return;
ga = &gpus[gpu].adl;
if (temp > ga->overtemp) {
applog(LOG_WARNING, "Overhead detected, increasing fan to 100%");
set_fanspeed(gpu, 100);
return;
}
if (temp > ga->targettemp && fanpercent < 85) {
if (opt_debug)
applog(LOG_DEBUG, "Temperature over target, increasing fanspeed");
set_fanspeed(gpu, fanpercent + 5);
return;
}
if (fanpercent && temp < ga->targettemp - 5) {
if (opt_debug)
applog(LOG_DEBUG, "Temperature 5 degrees below target, decreasing fanspeed");
set_fanspeed(gpu, fanpercent - 1);
}
}
void change_autosettings(int gpu)
{
struct gpu_adl *ga = &gpus[gpu].adl;
char input;
wlogprint("Fan autotune is %s\n", ga->autofan ? "enabled" : "disabled");
wlogprint("Target temperature: %d\n", ga->targettemp);
wlogprint("Overheat temperature: %d\n", ga->overtemp);
wlogprint("Change [F]an [T]arget [O]verheat\n");
wlogprint("Or press any other key to continue\n");
input = getch();
if (!strncasecmp(&input, "f", 1)) {
ga->autofan ^= true;
wlogprint("Fan autotune is now %s\n", ga->autofan ? "enabled" : "disabled");
if (!ga->autofan) {
wlogprint("Setting fan to 85 %% as safety precaution\n");
set_fanspeed(gpu, 85);
}
}
}
void change_gpusettings(int gpu) void change_gpusettings(int gpu)
{ {
int val, imin = 0, imax = 0; int val, imin = 0, imax = 0;
float fval, fmin = 0, fmax = 0; float fval, fmin = 0, fmax = 0;
char input; char input;
wlogprint("Change [E]ngine [F]an [M]emory [V]oltage\n"); wlogprint("Change [A]utomatic [E]ngine [F]an [M]emory [V]oltage\n");
wlogprint("Or press any other key to continue\n"); wlogprint("Or press any other key to continue\n");
input = getch(); input = getch();
if (!strncasecmp(&input, "e", 1)) { if (!strncasecmp(&input, "a", 1)) {
change_autosettings(gpu);
} else if (!strncasecmp(&input, "e", 1)) {
get_enginerange(gpu, &imin, &imax); get_enginerange(gpu, &imin, &imax);
wlogprint("Enter GPU engine clock speed (%d - %d Mhz)", imin, imax); wlogprint("Enter GPU engine clock speed (%d - %d Mhz)", imin, imax);
val = curses_int(""); val = curses_int("");
@ -542,7 +632,7 @@ void change_gpusettings(int gpu)
wlogprint("Successfully modified engine clock speed\n"); wlogprint("Successfully modified engine clock speed\n");
else else
wlogprint("Failed to modify engine clock speed\n"); wlogprint("Failed to modify engine clock speed\n");
} else if (!strncasecmp(&input, "f", 1)) { } else if (!strncasecmp(&input, "f", 1)) {
get_fanrange(gpu, &imin, &imax); get_fanrange(gpu, &imin, &imax);
wlogprint("Enter fan percentage (%d - %d %)", imin, imax); wlogprint("Enter fan percentage (%d - %d %)", imin, imax);
val = curses_int(""); val = curses_int("");

2
adl.h

@ -10,10 +10,12 @@ float gpu_vddc(int gpu);
int gpu_activity(int gpu); int gpu_activity(int gpu);
int gpu_fanspeed(int gpu); int gpu_fanspeed(int gpu);
void change_gpusettings(int gpu); void change_gpusettings(int gpu);
void gpu_autotune(int gpu);
void clear_adl(void); void clear_adl(void);
#else /* HAVE_ADL */ #else /* HAVE_ADL */
void init_adl(int nDevs) {} void init_adl(int nDevs) {}
void change_gpusettings(int gpu) { } void change_gpusettings(int gpu) { }
void gpu_autotune(int gpu) { }
void clear_adl(void) {} void clear_adl(void) {}
#endif #endif
#endif #endif

2
main.c

@ -4337,6 +4337,8 @@ static void *watchdog_thread(void *userdata)
break; break;
thr = &thr_info[i]; thr = &thr_info[i];
gpu = thr->cgpu->cpu_gpu; gpu = thr->cgpu->cpu_gpu;
if (adl_active)
gpu_autotune(gpu);
/* Thread is waiting on getwork or disabled */ /* Thread is waiting on getwork or disabled */
if (thr->getwork || !gpu_devices[gpu]) if (thr->getwork || !gpu_devices[gpu])
continue; continue;

7
miner.h

@ -168,6 +168,13 @@ struct gpu_adl {
int iEngineClock; int iEngineClock;
int iMemoryClock; int iMemoryClock;
int iVddc; int iVddc;
bool autofan;
bool autoengine;
int targettemp;
int overtemp;
int minspeed;
int maxspeed;
}; };
#endif #endif

Loading…
Cancel
Save