Commit 22aa1a20 authored by AngeloGioacchino Del Regno's avatar AngeloGioacchino Del Regno Committed by Steven Price

drm/panfrost: Really power off GPU cores in panfrost_gpu_power_off()

The layout of the registers {TILER,SHADER,L2}_PWROFF_LO, used to request
powering off cores, is the same as the {TILER,SHADER,L2}_PWRON_LO ones:
this means that in order to request poweroff of cores, we are supposed
to write a bitmask of cores that should be powered off!
This means that the panfrost_gpu_power_off() function has always been
doing nothing.

Fix powering off the GPU by writing a bitmask of the cores to poweroff
to the relevant PWROFF_LO registers and then check that the transition
(from ON to OFF) has finished by polling the relevant PWRTRANS_LO
registers.

While at it, in order to avoid code duplication, move the core mask
logic from panfrost_gpu_power_on() to a new panfrost_get_core_mask()
function, used in both poweron and poweroff.

Fixes: f3ba9122 ("drm/panfrost: Add initial panfrost driver")
Signed-off-by: default avatarAngeloGioacchino Del Regno <angelogioacchino.delregno@collabora.com>
Reviewed-by: default avatarSteven Price <steven.price@arm.com>
Signed-off-by: default avatarSteven Price <steven.price@arm.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20231102141507.73481-1-angelogioacchino.delregno@collabora.com
parent a78422e9
...@@ -362,15 +362,13 @@ unsigned long long panfrost_cycle_counter_read(struct panfrost_device *pfdev) ...@@ -362,15 +362,13 @@ unsigned long long panfrost_cycle_counter_read(struct panfrost_device *pfdev)
return ((u64)hi << 32) | lo; return ((u64)hi << 32) | lo;
} }
void panfrost_gpu_power_on(struct panfrost_device *pfdev) static u64 panfrost_get_core_mask(struct panfrost_device *pfdev)
{ {
int ret; u64 core_mask;
u32 val;
u64 core_mask = U64_MAX;
panfrost_gpu_init_quirks(pfdev); if (pfdev->features.l2_present == 1)
return U64_MAX;
if (pfdev->features.l2_present != 1) {
/* /*
* Only support one core group now. * Only support one core group now.
* ~(l2_present - 1) unsets all bits in l2_present except * ~(l2_present - 1) unsets all bits in l2_present except
...@@ -383,7 +381,19 @@ void panfrost_gpu_power_on(struct panfrost_device *pfdev) ...@@ -383,7 +381,19 @@ void panfrost_gpu_power_on(struct panfrost_device *pfdev)
dev_info_once(pfdev->dev, "using only 1st core group (%lu cores from %lu)\n", dev_info_once(pfdev->dev, "using only 1st core group (%lu cores from %lu)\n",
hweight64(core_mask), hweight64(core_mask),
hweight64(pfdev->features.shader_present)); hweight64(pfdev->features.shader_present));
}
return core_mask;
}
void panfrost_gpu_power_on(struct panfrost_device *pfdev)
{
int ret;
u32 val;
u64 core_mask;
panfrost_gpu_init_quirks(pfdev);
core_mask = panfrost_get_core_mask(pfdev);
gpu_write(pfdev, L2_PWRON_LO, pfdev->features.l2_present & core_mask); gpu_write(pfdev, L2_PWRON_LO, pfdev->features.l2_present & core_mask);
ret = readl_relaxed_poll_timeout(pfdev->iomem + L2_READY_LO, ret = readl_relaxed_poll_timeout(pfdev->iomem + L2_READY_LO,
val, val == (pfdev->features.l2_present & core_mask), val, val == (pfdev->features.l2_present & core_mask),
...@@ -408,9 +418,27 @@ void panfrost_gpu_power_on(struct panfrost_device *pfdev) ...@@ -408,9 +418,27 @@ void panfrost_gpu_power_on(struct panfrost_device *pfdev)
void panfrost_gpu_power_off(struct panfrost_device *pfdev) void panfrost_gpu_power_off(struct panfrost_device *pfdev)
{ {
gpu_write(pfdev, TILER_PWROFF_LO, 0); u64 core_mask = panfrost_get_core_mask(pfdev);
gpu_write(pfdev, SHADER_PWROFF_LO, 0); int ret;
gpu_write(pfdev, L2_PWROFF_LO, 0); u32 val;
gpu_write(pfdev, SHADER_PWROFF_LO, pfdev->features.shader_present & core_mask);
ret = readl_relaxed_poll_timeout(pfdev->iomem + SHADER_PWRTRANS_LO,
val, !val, 1, 1000);
if (ret)
dev_err(pfdev->dev, "shader power transition timeout");
gpu_write(pfdev, TILER_PWROFF_LO, pfdev->features.tiler_present);
ret = readl_relaxed_poll_timeout(pfdev->iomem + TILER_PWRTRANS_LO,
val, !val, 1, 1000);
if (ret)
dev_err(pfdev->dev, "tiler power transition timeout");
gpu_write(pfdev, L2_PWROFF_LO, pfdev->features.l2_present & core_mask);
ret = readl_poll_timeout(pfdev->iomem + L2_PWRTRANS_LO,
val, !val, 0, 1000);
if (ret)
dev_err(pfdev->dev, "l2 power transition timeout");
} }
int panfrost_gpu_init(struct panfrost_device *pfdev) int panfrost_gpu_init(struct panfrost_device *pfdev)
......
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