summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJinyoung Park <jinyoungp@nvidia.com>2013-01-18 02:19:11 +0900
committerDan Willemsen <dwillemsen@nvidia.com>2013-09-14 12:57:17 -0700
commit2ac7f32d425a3578427dd0528aceff9d6e568a2a (patch)
tree0589bda42237f01855f6d04a89de76065915b6e2
parentc8cbab3457927b5c8c9611a0daddbea09914eaf2 (diff)
Thermal: pid_thermal_gov: Add compensation
Added compensation to get steady state transition. And added sysfs node for compensation rate. Bug 1200111 Signed-off-by: Jinyoung Park <jinyoungp@nvidia.com> Reviewed-on: http://git-master/r/193836 (cherry picked from commit 4d2d134e71efce2cd652ccb0eac4946c0e3743f2) Change-Id: I406a9fa60c705f01d5c7a2194eecdf9babe09634 Signed-off-by: Jinyoung Park <jinyoungp@nvidia.com> Reviewed-on: http://git-master/r/198875 Reviewed-by: Automatic_Commit_Validation_User Reviewed-by: Diwakar Tundlam <dtundlam@nvidia.com>
-rw-r--r--drivers/thermal/pid_thermal_gov.c69
1 files changed, 62 insertions, 7 deletions
diff --git a/drivers/thermal/pid_thermal_gov.c b/drivers/thermal/pid_thermal_gov.c
index e9d74f0316d0..ca0fcaa2bbc0 100644
--- a/drivers/thermal/pid_thermal_gov.c
+++ b/drivers/thermal/pid_thermal_gov.c
@@ -24,10 +24,11 @@
#define DRV_NAME "pid_thermal_gov"
-#define MAX_ERR_TEMP_DEFAULT 10000 /* in mC */
-#define MAX_ERR_GAIN_DEFAULT 1000
-#define GAIN_P_DEFAULT 1000
-#define GAIN_D_DEFAULT 0
+#define MAX_ERR_TEMP_DEFAULT 10000 /* in mC */
+#define MAX_ERR_GAIN_DEFAULT 1000
+#define GAIN_P_DEFAULT 1000
+#define GAIN_D_DEFAULT 0
+#define COMPENSATION_RATE_DEFAULT 20
struct pid_thermal_gov_attribute {
struct attribute attr;
@@ -46,6 +47,8 @@ struct pid_thermal_governor {
int gain_p; /* proportional gain */
int gain_i; /* integral gain */
int gain_d; /* derivative gain */
+
+ unsigned long compensation_rate;
};
#define tz_to_gov(t) \
@@ -177,11 +180,44 @@ static ssize_t gain_d_store(struct kobject *kobj, struct attribute *attr,
static struct pid_thermal_gov_attribute gain_d_attr =
__ATTR(gain_d, 0644, gain_d_show, gain_d_store);
+static ssize_t compensation_rate_show(struct kobject *kobj,
+ struct attribute *attr, char *buf)
+{
+ struct pid_thermal_governor *gov = kobj_to_gov(kobj);
+
+ if (!gov)
+ return -ENODEV;
+
+ return sprintf(buf, "%lu\n", gov->compensation_rate);
+}
+
+static ssize_t compensation_rate_store(struct kobject *kobj,
+ struct attribute *attr, const char *buf,
+ size_t count)
+{
+ struct pid_thermal_governor *gov = kobj_to_gov(kobj);
+ unsigned long val;
+
+ if (!gov)
+ return -ENODEV;
+
+ if (!sscanf(buf, "%lu\n", &val))
+ return -EINVAL;
+
+ gov->compensation_rate = val;
+ return count;
+}
+
+static struct pid_thermal_gov_attribute compensation_rate_attr =
+ __ATTR(compensation_rate, 0644,
+ compensation_rate_show, compensation_rate_store);
+
static struct attribute *pid_thermal_gov_default_attrs[] = {
&max_err_temp_attr.attr,
&max_err_gain_attr.attr,
&gain_p_attr.attr,
&gain_d_attr.attr,
+ &compensation_rate_attr.attr,
NULL,
};
@@ -243,6 +279,7 @@ static int pid_thermal_gov_start(struct thermal_zone_device *tz)
gov->max_err_gain = MAX_ERR_GAIN_DEFAULT;
gov->gain_p = GAIN_P_DEFAULT;
gov->gain_d = GAIN_D_DEFAULT;
+ gov->compensation_rate = COMPENSATION_RATE_DEFAULT;
tz->governor_data = gov;
return 0;
@@ -286,11 +323,14 @@ pid_thermal_gov_get_target(struct thermal_zone_device *tz,
int last_temperature = tz->passive ? tz->last_temperature : trip_temp;
int passive_delay = tz->passive ? tz->passive_delay : MSEC_PER_SEC;
s64 proportional, derivative, sum_err, max_err;
- unsigned long max_state;
+ unsigned long max_state, cur_state, target, compensation;
if (cdev->ops->get_max_state(cdev, &max_state) < 0)
return 0;
+ if (cdev->ops->get_cur_state(cdev, &cur_state) < 0)
+ return 0;
+
/* Calculate proportional term */
proportional = (s64)tz->temperature - (s64)trip_temp;
proportional *= gov->gain_p;
@@ -312,9 +352,24 @@ pid_thermal_gov_get_target(struct thermal_zone_device *tz,
return max_state;
sum_err = sum_err * max_state + max_err - 1;
- sum_err = div64_s64(sum_err, max_err);
+ target = (unsigned long)div64_s64(sum_err, max_err);
+
+ /* Apply compensation */
+ if (target == cur_state)
+ return target;
+
+ if (target > cur_state) {
+ compensation = DIV_ROUND_UP(gov->compensation_rate *
+ (target - cur_state), 100);
+ target = min(cur_state + compensation, max_state);
+ } else if (target < cur_state) {
+ compensation = DIV_ROUND_UP(gov->compensation_rate *
+ (cur_state - target), 100);
+ target = (cur_state > compensation) ?
+ (cur_state - compensation) : 0;
+ }
- return (unsigned long)sum_err;
+ return target;
}
static int pid_thermal_gov_throttle(struct thermal_zone_device *tz, int trip)