@@ -251,6 +251,7 @@ static void
nouveau_perf_voltage(struct drm_device *dev, struct nouveau_pm_level *perflvl)
{
struct nouveau_drm *drm = nouveau_drm(dev);
+ struct nouveau_pm_voltage *volt = &nouveau_pm(dev)->voltage;
struct bit_entry P;
u8 *vmap;
int id;
@@ -287,6 +288,20 @@ nouveau_perf_voltage(struct drm_device *dev, struct nouveau_pm_level *perflvl)
perflvl->volt_min = ROM32(vmap[0]);
perflvl->volt_max = ROM32(vmap[4]);
}
+
+ /* Adjust for buggy bioses */
+ if (!volt->step_uv || !volt->step_ofs)
+ return;
+
+ if (perflvl->volt_min % volt->step_uv != volt->step_ofs) {
+ perflvl->volt_min -= perflvl->volt_min % volt->step_uv;
+ perflvl->volt_min += volt->step_ofs;
+ }
+
+ if (perflvl->volt_max % volt->step_uv != volt->step_ofs) {
+ perflvl->volt_max -= perflvl->volt_max % volt->step_uv;
+ perflvl->volt_max += volt->step_ofs;
+ }
}
void
@@ -40,6 +40,7 @@ struct nouveau_pm_voltage {
struct nouveau_pm_voltage_level *level;
int nr_level;
+ u16 step_uv, step_ofs;
};
/* Exclusive upper limits */
@@ -236,6 +236,9 @@ nouveau_volt_init(struct drm_device *dev)
voltage->level[vid].vid = vid;
volt_uv += step_uv;
}
+ voltage->step_uv = step_uv >= 0 ? step_uv : -step_uv;
+ if (step_uv)
+ voltage->step_ofs = volt_uv % voltage->step_uv;
}
voltage->supported = true;
Required to find all performance levels on my nvc0. Signed-off-by: Maarten Lankhorst <maarten.lankhorst@canonical.com> ---