Commit 0b627a0b authored by Ben Skeggs's avatar Ben Skeggs

drm/nouveau/pm: change volt/fan before upclock, but after downclock

Signed-off-by: default avatarBen Skeggs <bskeggs@redhat.com>
parent ff2b6c6e
...@@ -97,39 +97,62 @@ nouveau_pwmfan_set(struct drm_device *dev, int percent) ...@@ -97,39 +97,62 @@ nouveau_pwmfan_set(struct drm_device *dev, int percent)
} }
static int static int
nouveau_pm_perflvl_set(struct drm_device *dev, struct nouveau_pm_level *perflvl) nouveau_pm_perflvl_aux(struct drm_device *dev, struct nouveau_pm_level *perflvl,
struct nouveau_pm_level *a, struct nouveau_pm_level *b)
{ {
struct drm_nouveau_private *dev_priv = dev->dev_private; struct drm_nouveau_private *dev_priv = dev->dev_private;
struct nouveau_pm_engine *pm = &dev_priv->engine.pm; struct nouveau_pm_engine *pm = &dev_priv->engine.pm;
void *state;
int ret; int ret;
if (perflvl == pm->cur)
return 0;
/*XXX: not on all boards, we should control based on temperature /*XXX: not on all boards, we should control based on temperature
* on recent boards.. or maybe on some other factor we don't * on recent boards.. or maybe on some other factor we don't
* know about? * know about?
*/ */
if (perflvl->fanspeed) { if (a->fanspeed && b->fanspeed && b->fanspeed > a->fanspeed) {
ret = nouveau_pwmfan_set(dev, perflvl->fanspeed); ret = nouveau_pwmfan_set(dev, perflvl->fanspeed);
if (ret && ret != -ENODEV) if (ret && ret != -ENODEV) {
NV_ERROR(dev, "set fanspeed failed: %d\n", ret); NV_ERROR(dev, "fanspeed set failed: %d\n", ret);
return ret;
}
} }
if (pm->voltage.supported && pm->voltage_set && perflvl->volt_min) { if (pm->voltage.supported && pm->voltage_set) {
ret = pm->voltage_set(dev, perflvl->volt_min); if (a->volt_min && b->volt_min && b->volt_min > a->volt_min) {
if (ret) { ret = pm->voltage_set(dev, perflvl->volt_min);
NV_ERROR(dev, "voltage_set %d failed: %d\n", if (ret) {
perflvl->volt_min, ret); NV_ERROR(dev, "voltage set failed: %d\n", ret);
return ret;
}
} }
} }
return 0;
}
static int
nouveau_pm_perflvl_set(struct drm_device *dev, struct nouveau_pm_level *perflvl)
{
struct drm_nouveau_private *dev_priv = dev->dev_private;
struct nouveau_pm_engine *pm = &dev_priv->engine.pm;
void *state;
int ret;
if (perflvl == pm->cur)
return 0;
ret = nouveau_pm_perflvl_aux(dev, perflvl, pm->cur, perflvl);
if (ret)
return ret;
state = pm->clocks_pre(dev, perflvl); state = pm->clocks_pre(dev, perflvl);
if (IS_ERR(state)) if (IS_ERR(state))
return PTR_ERR(state); return PTR_ERR(state);
pm->clocks_set(dev, state); pm->clocks_set(dev, state);
ret = nouveau_pm_perflvl_aux(dev, perflvl, perflvl, pm->cur);
if (ret)
return ret;
pm->cur = perflvl; pm->cur = perflvl;
return 0; return 0;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment