drm/msm/gpu: Respect PM QoS constraints
Re-work the boost and idle clamping to use PM QoS requests instead, so they get aggreggated with other requests (such as cooling device). This does have the minor side-effect that devfreq sysfs min_freq/ max_freq files now reflect the boost and idle clamping, as they show (despite what they are documented to show) the aggregated min/max freq. Fixing that in devfreq does not look straightforward after considering that OPPs can be dynamically added/removed. However writes to the sysfs files still behave as expected. v2: Use 64b math to avoid potential 32b overflow Signed-off-by: Rob Clark <robdclark@chromium.org> Link: https://lore.kernel.org/r/20211120200103.1051459-3-robdclark@gmail.com Signed-off-by: Rob Clark <robdclark@chromium.org>
This commit is contained in:
parent
2a1ac5ba90
commit
7c0ffcd40b
2 changed files with 72 additions and 52 deletions
|
@ -87,6 +87,21 @@ struct msm_gpu_devfreq {
|
|||
/** devfreq: devfreq instance */
|
||||
struct devfreq *devfreq;
|
||||
|
||||
/**
|
||||
* idle_constraint:
|
||||
*
|
||||
* A PM QoS constraint to limit max freq while the GPU is idle.
|
||||
*/
|
||||
struct dev_pm_qos_request idle_freq;
|
||||
|
||||
/**
|
||||
* boost_constraint:
|
||||
*
|
||||
* A PM QoS constraint to boost min freq for a period of time
|
||||
* until the boost expires.
|
||||
*/
|
||||
struct dev_pm_qos_request boost_freq;
|
||||
|
||||
/**
|
||||
* busy_cycles:
|
||||
*
|
||||
|
@ -102,23 +117,20 @@ struct msm_gpu_devfreq {
|
|||
/** idle_time: Time of last transition to idle: */
|
||||
ktime_t idle_time;
|
||||
|
||||
/**
|
||||
* idle_freq:
|
||||
*
|
||||
* Shadow frequency used while the GPU is idle. From the PoV of
|
||||
* the devfreq governor, we are continuing to sample busyness and
|
||||
* adjust frequency while the GPU is idle, but we use this shadow
|
||||
* value as the GPU is actually clamped to minimum frequency while
|
||||
* it is inactive.
|
||||
*/
|
||||
unsigned long idle_freq;
|
||||
|
||||
/**
|
||||
* idle_work:
|
||||
*
|
||||
* Used to delay clamping to idle freq on active->idle transition.
|
||||
*/
|
||||
struct msm_hrtimer_work idle_work;
|
||||
|
||||
/**
|
||||
* boost_work:
|
||||
*
|
||||
* Used to reset the boost_constraint after the boost period has
|
||||
* elapsed
|
||||
*/
|
||||
struct msm_hrtimer_work boost_work;
|
||||
};
|
||||
|
||||
struct msm_gpu {
|
||||
|
@ -522,6 +534,7 @@ void msm_devfreq_init(struct msm_gpu *gpu);
|
|||
void msm_devfreq_cleanup(struct msm_gpu *gpu);
|
||||
void msm_devfreq_resume(struct msm_gpu *gpu);
|
||||
void msm_devfreq_suspend(struct msm_gpu *gpu);
|
||||
void msm_devfreq_boost(struct msm_gpu *gpu, unsigned factor);
|
||||
void msm_devfreq_active(struct msm_gpu *gpu);
|
||||
void msm_devfreq_idle(struct msm_gpu *gpu);
|
||||
|
||||
|
|
|
@ -9,6 +9,7 @@
|
|||
|
||||
#include <linux/devfreq.h>
|
||||
#include <linux/devfreq_cooling.h>
|
||||
#include <linux/units.h>
|
||||
|
||||
/*
|
||||
* Power Management:
|
||||
|
@ -25,17 +26,6 @@ static int msm_devfreq_target(struct device *dev, unsigned long *freq,
|
|||
* to something that actually is in the opp table:
|
||||
*/
|
||||
opp = devfreq_recommended_opp(dev, freq, flags);
|
||||
|
||||
/*
|
||||
* If the GPU is idle, devfreq is not aware, so just ignore
|
||||
* it's requests
|
||||
*/
|
||||
if (gpu->devfreq.idle_freq) {
|
||||
gpu->devfreq.idle_freq = *freq;
|
||||
dev_pm_opp_put(opp);
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (IS_ERR(opp))
|
||||
return PTR_ERR(opp);
|
||||
|
||||
|
@ -53,9 +43,6 @@ static int msm_devfreq_target(struct device *dev, unsigned long *freq,
|
|||
|
||||
static unsigned long get_freq(struct msm_gpu *gpu)
|
||||
{
|
||||
if (gpu->devfreq.idle_freq)
|
||||
return gpu->devfreq.idle_freq;
|
||||
|
||||
if (gpu->funcs->gpu_get_freq)
|
||||
return gpu->funcs->gpu_get_freq(gpu);
|
||||
|
||||
|
@ -93,6 +80,7 @@ static struct devfreq_dev_profile msm_devfreq_profile = {
|
|||
.get_cur_freq = msm_devfreq_get_cur_freq,
|
||||
};
|
||||
|
||||
static void msm_devfreq_boost_work(struct kthread_work *work);
|
||||
static void msm_devfreq_idle_work(struct kthread_work *work);
|
||||
|
||||
void msm_devfreq_init(struct msm_gpu *gpu)
|
||||
|
@ -103,6 +91,12 @@ void msm_devfreq_init(struct msm_gpu *gpu)
|
|||
if (!gpu->funcs->gpu_busy)
|
||||
return;
|
||||
|
||||
dev_pm_qos_add_request(&gpu->pdev->dev, &df->idle_freq,
|
||||
DEV_PM_QOS_MAX_FREQUENCY,
|
||||
PM_QOS_MAX_FREQUENCY_DEFAULT_VALUE);
|
||||
dev_pm_qos_add_request(&gpu->pdev->dev, &df->boost_freq,
|
||||
DEV_PM_QOS_MIN_FREQUENCY, 0);
|
||||
|
||||
msm_devfreq_profile.initial_freq = gpu->fast_rate;
|
||||
|
||||
/*
|
||||
|
@ -133,13 +127,19 @@ void msm_devfreq_init(struct msm_gpu *gpu)
|
|||
gpu->cooling = NULL;
|
||||
}
|
||||
|
||||
msm_hrtimer_work_init(&df->boost_work, gpu->worker, msm_devfreq_boost_work,
|
||||
CLOCK_MONOTONIC, HRTIMER_MODE_REL);
|
||||
msm_hrtimer_work_init(&df->idle_work, gpu->worker, msm_devfreq_idle_work,
|
||||
CLOCK_MONOTONIC, HRTIMER_MODE_REL);
|
||||
}
|
||||
|
||||
void msm_devfreq_cleanup(struct msm_gpu *gpu)
|
||||
{
|
||||
struct msm_gpu_devfreq *df = &gpu->devfreq;
|
||||
|
||||
devfreq_cooling_unregister(gpu->cooling);
|
||||
dev_pm_qos_remove_request(&df->boost_freq);
|
||||
dev_pm_qos_remove_request(&df->idle_freq);
|
||||
}
|
||||
|
||||
void msm_devfreq_resume(struct msm_gpu *gpu)
|
||||
|
@ -155,12 +155,40 @@ void msm_devfreq_suspend(struct msm_gpu *gpu)
|
|||
devfreq_suspend_device(gpu->devfreq.devfreq);
|
||||
}
|
||||
|
||||
static void msm_devfreq_boost_work(struct kthread_work *work)
|
||||
{
|
||||
struct msm_gpu_devfreq *df = container_of(work,
|
||||
struct msm_gpu_devfreq, boost_work.work);
|
||||
|
||||
dev_pm_qos_update_request(&df->boost_freq, 0);
|
||||
}
|
||||
|
||||
void msm_devfreq_boost(struct msm_gpu *gpu, unsigned factor)
|
||||
{
|
||||
struct msm_gpu_devfreq *df = &gpu->devfreq;
|
||||
uint64_t freq;
|
||||
|
||||
freq = get_freq(gpu);
|
||||
freq *= factor;
|
||||
|
||||
/*
|
||||
* A nice little trap is that PM QoS operates in terms of KHz,
|
||||
* while devfreq operates in terms of Hz:
|
||||
*/
|
||||
do_div(freq, HZ_PER_KHZ);
|
||||
|
||||
dev_pm_qos_update_request(&df->boost_freq, freq);
|
||||
|
||||
msm_hrtimer_queue_work(&df->boost_work,
|
||||
ms_to_ktime(msm_devfreq_profile.polling_ms),
|
||||
HRTIMER_MODE_REL);
|
||||
}
|
||||
|
||||
void msm_devfreq_active(struct msm_gpu *gpu)
|
||||
{
|
||||
struct msm_gpu_devfreq *df = &gpu->devfreq;
|
||||
struct devfreq_dev_status status;
|
||||
unsigned int idle_time;
|
||||
unsigned long target_freq = df->idle_freq;
|
||||
|
||||
if (!df->devfreq)
|
||||
return;
|
||||
|
@ -170,12 +198,6 @@ void msm_devfreq_active(struct msm_gpu *gpu)
|
|||
*/
|
||||
hrtimer_cancel(&df->idle_work.timer);
|
||||
|
||||
/*
|
||||
* Hold devfreq lock to synchronize with get_dev_status()/
|
||||
* target() callbacks
|
||||
*/
|
||||
mutex_lock(&df->devfreq->lock);
|
||||
|
||||
idle_time = ktime_to_ms(ktime_sub(ktime_get(), df->idle_time));
|
||||
|
||||
/*
|
||||
|
@ -184,20 +206,17 @@ void msm_devfreq_active(struct msm_gpu *gpu)
|
|||
* the governor to ramp up the freq.. so give some boost
|
||||
*/
|
||||
if (idle_time > msm_devfreq_profile.polling_ms) {
|
||||
target_freq *= 2;
|
||||
msm_devfreq_boost(gpu, 2);
|
||||
}
|
||||
|
||||
df->idle_freq = 0;
|
||||
|
||||
msm_devfreq_target(&gpu->pdev->dev, &target_freq, 0);
|
||||
dev_pm_qos_update_request(&df->idle_freq,
|
||||
PM_QOS_MAX_FREQUENCY_DEFAULT_VALUE);
|
||||
|
||||
/*
|
||||
* Reset the polling interval so we aren't inconsistent
|
||||
* about freq vs busy/total cycles
|
||||
*/
|
||||
msm_devfreq_get_dev_status(&gpu->pdev->dev, &status);
|
||||
|
||||
mutex_unlock(&df->devfreq->lock);
|
||||
}
|
||||
|
||||
|
||||
|
@ -206,23 +225,11 @@ static void msm_devfreq_idle_work(struct kthread_work *work)
|
|||
struct msm_gpu_devfreq *df = container_of(work,
|
||||
struct msm_gpu_devfreq, idle_work.work);
|
||||
struct msm_gpu *gpu = container_of(df, struct msm_gpu, devfreq);
|
||||
unsigned long idle_freq, target_freq = 0;
|
||||
|
||||
/*
|
||||
* Hold devfreq lock to synchronize with get_dev_status()/
|
||||
* target() callbacks
|
||||
*/
|
||||
mutex_lock(&df->devfreq->lock);
|
||||
|
||||
idle_freq = get_freq(gpu);
|
||||
|
||||
if (gpu->clamp_to_idle)
|
||||
msm_devfreq_target(&gpu->pdev->dev, &target_freq, 0);
|
||||
|
||||
df->idle_time = ktime_get();
|
||||
df->idle_freq = idle_freq;
|
||||
|
||||
mutex_unlock(&df->devfreq->lock);
|
||||
if (gpu->clamp_to_idle)
|
||||
dev_pm_qos_update_request(&df->idle_freq, 0);
|
||||
}
|
||||
|
||||
void msm_devfreq_idle(struct msm_gpu *gpu)
|
||||
|
|
Loading…
Add table
Reference in a new issue