Commit 5e00e326 authored by Karol Herbst's avatar Karol Herbst Committed by Ben Skeggs

drm/nouveau/volt: Don't require perfect fit

If we calculate the voltage in the table right, we get all kinds of values,
which never fit the hardware steps, so we use the closest higher value the
hardware can do.

v3: Simplify the implementation.
v5: Initialize best_err with volt->max_uv.
Signed-off-by: default avatarKarol Herbst <karolherbst@gmail.com>
Reviewed-by: default avatarMartin Peres <martin.peres@free.fr>
Signed-off-by: default avatarBen Skeggs <bskeggs@redhat.com>
parent 4b9ce6e7
...@@ -51,18 +51,30 @@ static int ...@@ -51,18 +51,30 @@ static int
nvkm_volt_set(struct nvkm_volt *volt, u32 uv) nvkm_volt_set(struct nvkm_volt *volt, u32 uv)
{ {
struct nvkm_subdev *subdev = &volt->subdev; struct nvkm_subdev *subdev = &volt->subdev;
int i, ret = -EINVAL; int i, ret = -EINVAL, best_err = volt->max_uv, best = -1;
if (volt->func->volt_set) if (volt->func->volt_set)
return volt->func->volt_set(volt, uv); return volt->func->volt_set(volt, uv);
for (i = 0; i < volt->vid_nr; i++) { for (i = 0; i < volt->vid_nr; i++) {
if (volt->vid[i].uv == uv) { int err = volt->vid[i].uv - uv;
ret = volt->func->vid_set(volt, volt->vid[i].vid); if (err < 0 || err > best_err)
nvkm_debug(subdev, "set %duv: %d\n", uv, ret); continue;
best_err = err;
best = i;
if (best_err == 0)
break; break;
} }
if (best == -1) {
nvkm_error(subdev, "couldn't set %iuv\n", uv);
return ret;
} }
ret = volt->func->vid_set(volt, volt->vid[best].vid);
nvkm_debug(subdev, "set req %duv to %duv: %d\n", uv,
volt->vid[best].uv, ret);
return ret; return ret;
} }
......
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